From dde0252d27b22a950a29a5f13253e2c24b153989 Mon Sep 17 00:00:00 2001 From: zhangliang <1466013297@qq.com> Date: Fri, 18 Oct 2024 16:21:57 +0800 Subject: [PATCH] =?UTF-8?q?=E7=AB=99=E7=82=B9=E9=85=8D=E7=BD=AE=E5=8A=9F?= =?UTF-8?q?=E8=83=BD=E5=AE=8C=E6=88=90=EF=BC=8C=E5=85=B3=E8=81=94=E9=85=8D?= =?UTF-8?q?=E7=BD=AE=E5=8A=9F=E8=83=BD=E5=AE=8C=E6=88=90?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- package-lock.json | 36 +- package.json | 4 +- public/images/access/media-plugin.png | Bin 5211 -> 0 bytes public/images/access/onvif.png | Bin 3915 -> 0 bytes public/images/calendar/hover.png | Bin 1392 -> 0 bytes public/images/media/doc5.png | Bin 53025 -> 0 bytes public/images/media/doc6.png | Bin 58930 -> 0 bytes public/images/media/doc7.png | Bin 71451 -> 0 bytes public/static/common/css/common.css | 977 - public/static/common/css/real.css | 0 .../img/LiveData/adaptive-noise_off.png | Bin 4121 -> 0 bytes .../common/img/LiveData/adaptive-noise_on.png | Bin 1501 -> 0 bytes public/static/common/img/LiveData/big.png | Bin 4325 -> 0 bytes .../static/common/img/LiveData/big_ready.png | Bin 4098 -> 0 bytes public/static/common/img/LiveData/event.png | Bin 8681 -> 0 bytes .../common/img/LiveData/event_ready.png | Bin 7054 -> 0 bytes public/static/common/img/LiveData/hide.png | Bin 1984 -> 0 bytes public/static/common/img/LiveData/history.png | Bin 6161 -> 0 bytes .../common/img/LiveData/history_ready.png | Bin 5971 -> 0 bytes public/static/common/img/LiveData/noise.png | Bin 4020 -> 0 bytes .../common/img/LiveData/noise_ready.png | Bin 1401 -> 0 bytes public/static/common/img/LiveData/pause.png | Bin 4220 -> 0 bytes public/static/common/img/LiveData/play.png | Bin 2138 -> 0 bytes public/static/common/img/LiveData/prpd.png | Bin 6780 -> 0 bytes .../common/img/LiveData/prpd_cluster.png | Bin 3764 -> 0 bytes public/static/common/img/LiveData/prps.png | Bin 8038 -> 0 bytes .../common/img/LiveData/prps_cluster.png | Bin 7467 -> 0 bytes public/static/common/img/LiveData/record.png | Bin 3914 -> 0 bytes .../common/img/LiveData/record_ready.png | Bin 1294 -> 0 bytes .../static/common/img/LiveData/recording.png | Bin 4270 -> 0 bytes public/static/common/img/LiveData/refresh.png | Bin 4273 -> 0 bytes public/static/common/img/LiveData/save.png | Bin 4153 -> 0 bytes .../static/common/img/LiveData/save_ready.png | Bin 1533 -> 0 bytes public/static/common/img/LiveData/show.png | Bin 1999 -> 0 bytes public/static/common/img/LiveData/small.png | Bin 4274 -> 0 bytes .../common/img/LiveData/small_ready.png | Bin 4064 -> 0 bytes public/static/common/img/LiveData/start.png | Bin 1476 -> 0 bytes public/static/common/img/LiveData/stop.png | Bin 2071 -> 0 bytes public/static/common/img/LiveData/total.png | Bin 5157 -> 0 bytes .../common/img/LiveData/total_ready.png | Bin 5008 -> 0 bytes public/static/common/img/download.png | Bin 3619 -> 0 bytes .../static/common/img/sprite-skin-nice-1.png | Bin 4433 -> 0 bytes public/static/common/img/sprite-skin-nice.png | Bin 4263 -> 0 bytes public/static/common/js/axisInfo/constant.js | 603 - public/static/common/js/controller.js | 27 - public/static/common/js/draw.js | 280 - public/static/common/js/drawHistory.js | 280 - public/static/common/js/echarts/echarts.js | 95739 ---------------- public/static/common/js/echartsOptions.js | 1050 - public/static/common/js/history.js | 211 - public/static/common/js/index.js | 1464 - public/static/common/js/jquery-ui.min.js | 13 - public/static/common/js/jquery.min.js | 2 - public/static/common/js/simpleAlert.js | 72 - public/static/common/js/three/Lut.js | 200 - .../static/common/js/three/OrbitControls.js | 1065 - public/static/common/js/three/stats.module.js | 167 - public/static/common/js/three/three.min.js | 6 - public/static/history.html | 91 - public/static/index.html | 137 - public/static/threePublic.zip | Bin 0 -> 1021983 bytes .../common/js/axisInfo/constant.js | 10 +- public/static/threePublic/common/js/index.js | 34 +- public/static/threePublic/index.html | 6 +- src/api/location/associateConfig.ts | 19 + src/api/monitor.ts | 29 + src/api/system/calendar.ts | 17 - src/components/AMapComponent/AMap.vue | 92 - .../AMapComponent/DistrictSearch.vue | 116 - src/components/AMapComponent/GeoJson.vue | 91 - src/components/AMapComponent/index.ts | 7 - src/components/AMapComponent/useMap.ts | 4 - src/components/AlarmLeveIcon/index.vue | 27 - src/components/CardSelect/CardSelect.vue | 282 - src/components/CardSelect/RadioButton.vue | 72 - src/components/CardSelect/index.ts | 11 - src/components/CheckButton/CheckButton.vue | 163 - src/components/CheckButton/index.ts | 3 - src/components/ConfirmModal/index.vue | 76 - src/components/Metadata/index.ts | 0 src/components/Modal/DragModal.vue | 326 - src/components/Modal/index.ts | 1 - src/components/Player/utils.ts | 13 - src/hook/useAlarmLevel.ts | 30 - src/i18n/index.ts | 17 - src/i18n/locale/en.json | 1773 - src/i18n/locale/zh.json | 1773 - src/i18n/temp/home_2024-7-4-926838.json | 8 - src/i18n/temp/home_2024-7-4-926841.json | 36 - src/i18n/temp/home_2024-7-4-926843.json | 10 - src/i18n/temp/home_2024-7-4-926845.json | 8 - src/i18n/temp/home_2024-7-4-926846.json | 8 - src/i18n/temp/home_2024-7-4-926849.json | 14 - src/i18n/temp/home_2024-7-4-926853.json | 12 - src/i18n/temp/home_2024-7-4-926855.json | 38 - src/i18n/temp/home_2024-7-4-926856.json | 24 - src/i18n/temp/home_2024-7-4-926861.json | 8 - src/i18n/temp/home_2024-7-4-926866.json | 14 - src/i18n/temp/home_2024-7-4-926873.json | 8 - src/i18n/temp/home_2024-7-4-926911.json | 68 - src/i18n/temp/user_2024-7-4-265143.json | 38 - src/style/global.less | 2 - src/style/modal.less | 9 - src/style/scrollbar.less | 28 - src/views/Northbound/AliCloud/Tree/index.vue | 134 - src/views/Northbound/DuerOS/Tree/index.vue | 138 - .../AlarmRecord/Alarm/components/AlarmLog.vue | 83 - .../Detail/AlarmRecord/Alarm/index.vue | 444 - .../Detail/AlarmRecord/Invalid/index.vue | 128 - .../Instance/Detail/AlarmRecord/index.vue | 33 - .../device/Instance/Detail/Firmware/index.vue | 193 - .../Metadata/Base/DetailModal/utils.ts | 25 - .../Base/components/Import/Import.vue | 230 - .../Metadata/Base/components/Import/index.ts | 1 - .../Metadata/Base/components/Import/util.ts | 110 - .../Base/components/Properties/hooks.ts | 106 - .../components/VirtualRule/DetailModal.vue | 52 - .../Metadata/Base/hooks/useGroup.ts | 36 - .../History/Detail/EventStatistics.vue | 2 +- .../AnalyzeDiagnosis/History/Detail/Prpd.vue | 6 +- .../History/Detail/PrpdTotal.vue | 2 +- .../Detail/Snipaste_2024-10-08_10-10-25.png | Bin 0 -> 42615 bytes .../AnalyzeDiagnosis/History/Detail/index.vue | 136 +- .../History/components/Network.vue | 196 +- .../History/components/TopChart.vue | 56 +- .../gis/AnalyzeDiagnosis/History/index.vue | 7 +- .../Graph/components/LeftMonitor/index.vue | 66 + src/views/gis/RealTime/Graph/index.vue | 152 +- src/views/gis/Site/RoleLeft/AddGroup.vue | 205 + src/views/gis/Site/RoleLeft/BindDevice.vue | 253 + src/views/gis/Site/RoleLeft/index.vue | 222 +- src/views/gis/Site/index.vue | 90 +- src/views/gis/Site/type.d.ts | 42 +- .../gis/components/LeftMonitor/index.vue | 66 + .../Outline/components/FifthKind.vue | 100 - .../Outline/components/FirstKind.vue | 197 - .../Outline/components/FourthKind.vue | 19 - .../Outline/components/SecondKind.vue | 143 - .../Outline/components/SixthKind.vue | 67 - .../Outline/components/ThirdKind.vue | 84 - src/views/link/AccessConfig/Outline/index.vue | 85 - .../components/EditUserDialog.vue | 326 + .../AssociationConfig/components/Progress.vue | 76 + .../location/AssociationConfig/index.vue | 321 +- .../location/Site/Detail/Permiss/index.vue | 153 + .../location/Site/Detail/User/index.d.ts | 23 + src/views/location/Site/Detail/User/index.vue | 234 + .../Site/Detail/components/AddUserDialog.vue | 123 + .../Site/Detail/components/PermissTree.vue | 548 + .../location/Site/Detail/components/util.ts | 41 + src/views/location/Site/Detail/index.vue | 26 + src/views/location/Site/RoleLeft/AddGroup.vue | 205 + .../location/Site/RoleLeft/BindDevice.vue | 163 + src/views/location/Site/RoleLeft/index.vue | 332 + .../Site/RoleRight/components/AddDialog.vue | 10 + src/views/location/Site/RoleRight/index.vue | 119 + src/views/location/Site/data.ts | 27 + src/views/location/Site/index.vue | 59 + src/views/location/Site/type.d.ts | 63 + src/views/media/Device/Channel/data.ts | 10 - src/views/media/Device/Summary/index.vue | 148 - .../HandTrigger/components/Actions.vue | 188 - .../HandTrigger/components/BranchesTabs.vue | 72 - .../HandTrigger/components/CardBox.vue | 566 - .../Configuration/Save/Scene/Save/Actions.vue | 395 - .../Save/Scene/Save/BranchesTabs.vue | 168 - .../Configuration/Save/Scene/Save/CardBox.vue | 599 - .../Save/Scene/Save/Terms/ActionsFilter.vue | 71 - .../Save/Scene/Save/Terms/Terms.vue | 87 - .../Save/Scene/Save/Terms/WhenItem.vue | 123 - .../Configuration/Save/Scene/Save/tags.vue | 70 - .../Configuration/Save/Scene/SceneDrawer.vue | 131 - .../TabComponent/components/DetailDrawer.vue | 148 - .../Alarm/Log/TabComponent/components/Log.vue | 164 - .../Log/TabComponent/components/LogDetail.vue | 78 - .../Log/TabComponent/components/Record.vue | 134 - .../Alarm/Log/components/Duration.vue | 43 - .../Scene/Save/action/Device/device/util.ts | 32 - .../Save/action/ListItem/CheckFilterItem.vue | 102 - .../Save/action/TriggerAlarm/AlarmModal.vue | 233 - .../Scene/Save/components/Description.vue | 84 - .../components/Terms/BranchesNameEdit.vue | 50 - .../Save/components/Terms/TermsTabPane.vue | 53 - .../Scene/Save/components/Timer/Calendar.vue | 263 - .../system/Calendar/CalendarRight/index.vue | 142 - .../system/Calendar/FullCalendar/index.vue | 761 - .../Calendar/Tags/components/editTag.vue | 127 - src/views/system/Calendar/Tags/index.vue | 200 - src/views/system/Calendar/index.vue | 24 - src/views/system/Region/Save/GeoJsonModal.vue | 46 - src/views/system/Region/util.ts | 31 - src/views/system/User/index.vue | 122 +- src/views/user/Login/utils.ts | 9 - vite.config.ts | 2 +- yarn.lock | 10 +- 195 files changed, 4136 insertions(+), 115857 deletions(-) delete mode 100644 public/images/access/media-plugin.png delete mode 100644 public/images/access/onvif.png delete mode 100644 public/images/calendar/hover.png delete mode 100644 public/images/media/doc5.png delete mode 100644 public/images/media/doc6.png delete mode 100644 public/images/media/doc7.png delete mode 100644 public/static/common/css/common.css delete mode 100644 public/static/common/css/real.css delete mode 100644 public/static/common/img/LiveData/adaptive-noise_off.png delete mode 100644 public/static/common/img/LiveData/adaptive-noise_on.png delete mode 100644 public/static/common/img/LiveData/big.png delete mode 100644 public/static/common/img/LiveData/big_ready.png delete mode 100644 public/static/common/img/LiveData/event.png delete mode 100644 public/static/common/img/LiveData/event_ready.png delete mode 100644 public/static/common/img/LiveData/hide.png delete mode 100644 public/static/common/img/LiveData/history.png delete mode 100644 public/static/common/img/LiveData/history_ready.png delete mode 100644 public/static/common/img/LiveData/noise.png delete mode 100644 public/static/common/img/LiveData/noise_ready.png delete mode 100644 public/static/common/img/LiveData/pause.png delete mode 100644 public/static/common/img/LiveData/play.png delete mode 100644 public/static/common/img/LiveData/prpd.png delete mode 100644 public/static/common/img/LiveData/prpd_cluster.png delete mode 100644 public/static/common/img/LiveData/prps.png delete mode 100644 public/static/common/img/LiveData/prps_cluster.png delete mode 100644 public/static/common/img/LiveData/record.png delete mode 100644 public/static/common/img/LiveData/record_ready.png delete mode 100644 public/static/common/img/LiveData/recording.png delete mode 100644 public/static/common/img/LiveData/refresh.png delete mode 100644 public/static/common/img/LiveData/save.png delete mode 100644 public/static/common/img/LiveData/save_ready.png delete mode 100644 public/static/common/img/LiveData/show.png delete mode 100644 public/static/common/img/LiveData/small.png delete mode 100644 public/static/common/img/LiveData/small_ready.png delete mode 100644 public/static/common/img/LiveData/start.png delete mode 100644 public/static/common/img/LiveData/stop.png delete mode 100644 public/static/common/img/LiveData/total.png delete mode 100644 public/static/common/img/LiveData/total_ready.png delete mode 100644 public/static/common/img/download.png delete mode 100644 public/static/common/img/sprite-skin-nice-1.png delete mode 100644 public/static/common/img/sprite-skin-nice.png delete mode 100644 public/static/common/js/axisInfo/constant.js delete mode 100644 public/static/common/js/controller.js delete mode 100644 public/static/common/js/draw.js delete mode 100644 public/static/common/js/drawHistory.js delete mode 100644 public/static/common/js/echarts/echarts.js delete mode 100644 public/static/common/js/echartsOptions.js delete mode 100644 public/static/common/js/history.js delete mode 100644 public/static/common/js/index.js delete mode 100644 public/static/common/js/jquery-ui.min.js delete mode 100644 public/static/common/js/jquery.min.js delete mode 100644 public/static/common/js/simpleAlert.js delete mode 100644 public/static/common/js/three/Lut.js delete mode 100644 public/static/common/js/three/OrbitControls.js delete mode 100644 public/static/common/js/three/stats.module.js delete mode 100644 public/static/common/js/three/three.min.js delete mode 100644 public/static/history.html delete mode 100644 public/static/index.html create mode 100644 public/static/threePublic.zip create mode 100644 src/api/location/associateConfig.ts create mode 100644 src/api/monitor.ts delete mode 100644 src/api/system/calendar.ts delete mode 100644 src/components/AMapComponent/AMap.vue delete mode 100644 src/components/AMapComponent/DistrictSearch.vue delete mode 100644 src/components/AMapComponent/GeoJson.vue delete mode 100644 src/components/AMapComponent/index.ts delete mode 100644 src/components/AMapComponent/useMap.ts delete mode 100644 src/components/AlarmLeveIcon/index.vue delete mode 100644 src/components/CardSelect/CardSelect.vue delete mode 100644 src/components/CardSelect/RadioButton.vue delete mode 100644 src/components/CardSelect/index.ts delete mode 100644 src/components/CheckButton/CheckButton.vue delete mode 100644 src/components/CheckButton/index.ts delete mode 100644 src/components/ConfirmModal/index.vue delete mode 100644 src/components/Metadata/index.ts delete mode 100644 src/components/Modal/DragModal.vue delete mode 100644 src/components/Modal/index.ts delete mode 100644 src/components/Player/utils.ts delete mode 100644 src/hook/useAlarmLevel.ts delete mode 100644 src/i18n/index.ts delete mode 100644 src/i18n/locale/en.json delete mode 100644 src/i18n/locale/zh.json delete mode 100644 src/i18n/temp/home_2024-7-4-926838.json delete mode 100644 src/i18n/temp/home_2024-7-4-926841.json delete mode 100644 src/i18n/temp/home_2024-7-4-926843.json delete mode 100644 src/i18n/temp/home_2024-7-4-926845.json delete mode 100644 src/i18n/temp/home_2024-7-4-926846.json delete mode 100644 src/i18n/temp/home_2024-7-4-926849.json delete mode 100644 src/i18n/temp/home_2024-7-4-926853.json delete mode 100644 src/i18n/temp/home_2024-7-4-926855.json delete mode 100644 src/i18n/temp/home_2024-7-4-926856.json delete mode 100644 src/i18n/temp/home_2024-7-4-926861.json delete mode 100644 src/i18n/temp/home_2024-7-4-926866.json delete mode 100644 src/i18n/temp/home_2024-7-4-926873.json delete mode 100644 src/i18n/temp/home_2024-7-4-926911.json delete mode 100644 src/i18n/temp/user_2024-7-4-265143.json delete mode 100644 src/style/global.less delete mode 100644 src/style/modal.less delete mode 100644 src/style/scrollbar.less delete mode 100644 src/views/Northbound/AliCloud/Tree/index.vue delete mode 100644 src/views/Northbound/DuerOS/Tree/index.vue delete mode 100644 src/views/device/Instance/Detail/AlarmRecord/Alarm/components/AlarmLog.vue delete mode 100644 src/views/device/Instance/Detail/AlarmRecord/Alarm/index.vue delete mode 100644 src/views/device/Instance/Detail/AlarmRecord/Invalid/index.vue delete mode 100644 src/views/device/Instance/Detail/AlarmRecord/index.vue delete mode 100644 src/views/device/Instance/Detail/Firmware/index.vue delete mode 100644 src/views/device/components/Metadata/Base/DetailModal/utils.ts delete mode 100644 src/views/device/components/Metadata/Base/components/Import/Import.vue delete mode 100644 src/views/device/components/Metadata/Base/components/Import/index.ts delete mode 100644 src/views/device/components/Metadata/Base/components/Import/util.ts delete mode 100644 src/views/device/components/Metadata/Base/components/Properties/hooks.ts delete mode 100644 src/views/device/components/Metadata/Base/components/VirtualRule/DetailModal.vue delete mode 100644 src/views/device/components/Metadata/Base/hooks/useGroup.ts create mode 100644 src/views/gis/AnalyzeDiagnosis/History/Detail/Snipaste_2024-10-08_10-10-25.png create mode 100644 src/views/gis/RealTime/Graph/components/LeftMonitor/index.vue create mode 100644 src/views/gis/Site/RoleLeft/AddGroup.vue create mode 100644 src/views/gis/Site/RoleLeft/BindDevice.vue create mode 100644 src/views/gis/components/LeftMonitor/index.vue delete mode 100644 src/views/link/AccessConfig/Outline/components/FifthKind.vue delete mode 100644 src/views/link/AccessConfig/Outline/components/FirstKind.vue delete mode 100644 src/views/link/AccessConfig/Outline/components/FourthKind.vue delete mode 100644 src/views/link/AccessConfig/Outline/components/SecondKind.vue delete mode 100644 src/views/link/AccessConfig/Outline/components/SixthKind.vue delete mode 100644 src/views/link/AccessConfig/Outline/components/ThirdKind.vue delete mode 100644 src/views/link/AccessConfig/Outline/index.vue create mode 100644 src/views/location/AssociationConfig/components/EditUserDialog.vue create mode 100644 src/views/location/AssociationConfig/components/Progress.vue create mode 100644 src/views/location/Site/Detail/Permiss/index.vue create mode 100644 src/views/location/Site/Detail/User/index.d.ts create mode 100644 src/views/location/Site/Detail/User/index.vue create mode 100644 src/views/location/Site/Detail/components/AddUserDialog.vue create mode 100644 src/views/location/Site/Detail/components/PermissTree.vue create mode 100644 src/views/location/Site/Detail/components/util.ts create mode 100644 src/views/location/Site/Detail/index.vue create mode 100644 src/views/location/Site/RoleLeft/AddGroup.vue create mode 100644 src/views/location/Site/RoleLeft/BindDevice.vue create mode 100644 src/views/location/Site/RoleLeft/index.vue create mode 100644 src/views/location/Site/RoleRight/components/AddDialog.vue create mode 100644 src/views/location/Site/RoleRight/index.vue create mode 100644 src/views/location/Site/data.ts create mode 100644 src/views/location/Site/index.vue create mode 100644 src/views/location/Site/type.d.ts delete mode 100644 src/views/media/Device/Channel/data.ts delete mode 100644 src/views/media/Device/Summary/index.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/HandTrigger/components/Actions.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/HandTrigger/components/BranchesTabs.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/HandTrigger/components/CardBox.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Actions.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/BranchesTabs.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/CardBox.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/ActionsFilter.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/Terms.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/WhenItem.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/tags.vue delete mode 100644 src/views/rule-engine/Alarm/Configuration/Save/Scene/SceneDrawer.vue delete mode 100644 src/views/rule-engine/Alarm/Log/TabComponent/components/DetailDrawer.vue delete mode 100644 src/views/rule-engine/Alarm/Log/TabComponent/components/Log.vue delete mode 100644 src/views/rule-engine/Alarm/Log/TabComponent/components/LogDetail.vue delete mode 100644 src/views/rule-engine/Alarm/Log/TabComponent/components/Record.vue delete mode 100644 src/views/rule-engine/Alarm/Log/components/Duration.vue delete mode 100644 src/views/rule-engine/Scene/Save/action/Device/device/util.ts delete mode 100644 src/views/rule-engine/Scene/Save/action/ListItem/CheckFilterItem.vue delete mode 100644 src/views/rule-engine/Scene/Save/action/TriggerAlarm/AlarmModal.vue delete mode 100644 src/views/rule-engine/Scene/Save/components/Description.vue delete mode 100644 src/views/rule-engine/Scene/Save/components/Terms/BranchesNameEdit.vue delete mode 100644 src/views/rule-engine/Scene/Save/components/Terms/TermsTabPane.vue delete mode 100644 src/views/rule-engine/Scene/Save/components/Timer/Calendar.vue delete mode 100644 src/views/system/Calendar/CalendarRight/index.vue delete mode 100644 src/views/system/Calendar/FullCalendar/index.vue delete mode 100644 src/views/system/Calendar/Tags/components/editTag.vue delete mode 100644 src/views/system/Calendar/Tags/index.vue delete mode 100644 src/views/system/Calendar/index.vue delete mode 100644 src/views/system/Region/Save/GeoJsonModal.vue delete mode 100644 src/views/system/Region/util.ts delete mode 100644 src/views/user/Login/utils.ts diff --git a/package-lock.json b/package-lock.json index 5e09885..a50d16d 100644 --- a/package-lock.json +++ b/package-lock.json @@ -47,6 +47,7 @@ "pinia": "^2.0.28", "rollup-plugin-copy": "^3.4.0", "rxjs": "^7.8.1", + "splitpanes": "^3.1.5", "three": "^0.146.0", "unplugin-auto-import": "^0.12.1", "unplugin-vue-components": "^0.22.12", @@ -58,8 +59,7 @@ "vue-json-viewer": "^3.0.4", "vue-router": "^4.1.6", "vue3-json-viewer": "^2.2.2", - "vue3-ts-jsoneditor": "^2.7.1", - "yarn": "^1.22.22" + "vue3-ts-jsoneditor": "^2.7.1" }, "devDependencies": { "@commitlint/cli": "^17.4.1", @@ -5055,6 +5055,7 @@ "version": "4.4.0", "resolved": "https://registry.npmjs.org/global/-/global-4.4.0.tgz", "integrity": "sha512-wv/LAoHdRE3BeTGz53FAamhGlPLhlssK45usmGFThIi4XqnBmjKQ16u+RNbP7WvigRZDxUsM0J3gcQ5yicaL0w==", + "license": "MIT", "dependencies": { "min-document": "^2.19.0", "process": "^0.11.10" @@ -9818,6 +9819,14 @@ "readable-stream": "^3.0.0" } }, + "node_modules/splitpanes": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/splitpanes/-/splitpanes-3.1.5.tgz", + "integrity": "sha512-r3Mq2ITFQ5a2VXLOy4/Sb2Ptp7OfEO8YIbhVJqJXoFc9hc5nTXXkCvtVDjIGbvC0vdE7tse+xTM9BMjsszP6bw==", + "funding": { + "url": "https://github.com/sponsors/antoniandre" + } + }, "node_modules/string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -11582,19 +11591,6 @@ "node": ">=10" } }, - "node_modules/yarn": { - "version": "1.22.22", - "resolved": "https://registry.npmjs.org/yarn/-/yarn-1.22.22.tgz", - "integrity": "sha512-prL3kGtyG7o9Z9Sv8IPfBNrWTDmXB4Qbes8A9rEzt6wkJV8mUvoirjU0Mp3GGAU06Y0XQyA3/2/RQFVuK7MTfg==", - "hasInstallScript": true, - "bin": { - "yarn": "bin/yarn.js", - "yarnpkg": "bin/yarn.js" - }, - "engines": { - "node": ">=4.0.0" - } - }, "node_modules/yn": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", @@ -18524,6 +18520,11 @@ "readable-stream": "^3.0.0" } }, + "splitpanes": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/splitpanes/-/splitpanes-3.1.5.tgz", + "integrity": "sha512-r3Mq2ITFQ5a2VXLOy4/Sb2Ptp7OfEO8YIbhVJqJXoFc9hc5nTXXkCvtVDjIGbvC0vdE7tse+xTM9BMjsszP6bw==" + }, "string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -19745,11 +19746,6 @@ "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", "dev": true }, - "yarn": { - "version": "1.22.22", - "resolved": "https://registry.npmjs.org/yarn/-/yarn-1.22.22.tgz", - "integrity": "sha512-prL3kGtyG7o9Z9Sv8IPfBNrWTDmXB4Qbes8A9rEzt6wkJV8mUvoirjU0Mp3GGAU06Y0XQyA3/2/RQFVuK7MTfg==" - }, "yn": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", diff --git a/package.json b/package.json index 6d6937b..1b04a25 100644 --- a/package.json +++ b/package.json @@ -52,6 +52,7 @@ "pinia": "^2.0.28", "rollup-plugin-copy": "^3.4.0", "rxjs": "^7.8.1", + "splitpanes": "^3.1.5", "three": "^0.146.0", "unplugin-auto-import": "^0.12.1", "unplugin-vue-components": "^0.22.12", @@ -63,8 +64,7 @@ "vue-json-viewer": "^3.0.4", "vue-router": "^4.1.6", "vue3-json-viewer": "^2.2.2", - "vue3-ts-jsoneditor": "^2.7.1", - "yarn": "^1.22.22" + "vue3-ts-jsoneditor": "^2.7.1" }, "devDependencies": { "@commitlint/cli": "^17.4.1", diff --git a/public/images/access/media-plugin.png b/public/images/access/media-plugin.png deleted file mode 100644 index 6eab68fb60625e0f4f79b6e188087927f2980aea..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5211 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 Z{vVU*H^5OLJoEqn002ovPDHLkV1g=#YOw$S diff --git a/public/images/calendar/hover.png b/public/images/calendar/hover.png deleted file mode 100644 index 10756cad510e1d55b8466a95ac3f9352982b5592..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1392 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 diff --git a/public/images/media/doc6.png b/public/images/media/doc6.png deleted file mode 100644 index 95b828414dc666e21226551c41b9a06ba3de2600..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 58930 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%0AM+Zl>56tR)X18n| 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 F{{y~yRZIW? diff --git a/public/images/media/doc7.png b/public/images/media/doc7.png deleted file mode 100644 index 97903fa5c55574a35254cfb0c36cd0385d0bb0ae..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 71451 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 diff --git a/public/static/common/css/common.css b/public/static/common/css/common.css deleted file mode 100644 index b4c9879..0000000 --- a/public/static/common/css/common.css +++ /dev/null @@ -1,977 +0,0 @@ -*{ - margin:0; - padding:0; -} -ul{ - list-style: none; -} -input { - /* border: 0; */ - outline: none; - /* background-color: rgba(0, 0, 0, 0); */ -} -a{ - text-decoration: none; - /* color: #333333; */ - color: #fff; - outline: medium none; - overflow: hidden; -} -/* 隐藏 input type="number" 的箭头按钮 */ -input[type="number"]::-webkit-outer-spin-button, -input[type="number"]::-webkit-inner-spin-button { - -webkit-appearance: none; /* WebKit 浏览器 */ - margin: 0; /* Firefox 不显示箭头 */ -} - -input[type="number"] { - -moz-appearance: textfield; /* Firefox */ -} - - -@media screen and (min-width: 1280px) { - body{ - overflow-x:hidden; - } -} -header .nav-list{ - color:#fff; -} - -.hide{ - display:none; -} - -.col{ - box-sizing: border-box; - padding: 0; - outline: 0; - border: 0; - vertical-align: baseline; - font-weight: inherit; - font-style: inherit; - font-size: 100%; - font-family: inherit; - overflow:hidden; -} -/* select option 样式 */ -select{ - box-sizing: border-box; - font-size:12px; -} - - -/* 实时数据表 */ - -.col.echarts-chart:after{ - display:block; - content:''; - clear:both; -} -.echarts-btn{ - position: relative; - height:20px; -} -.echarts-btn>span{ - /* float:right; */ - display: inline-block; -} - -.echarts-text span{ - font-size:14px; - color:#92A3A8; /*#333*/ - margin-right:8px; -} - -.realdata{ - padding-left: 10px; - padding-right: 10px; - position: absolute; - top: 10px; - z-index: 10; - width: 95vw; -} - -.btn{ - width: 22px; - height: 22px; - cursor: pointer; - border-radius: 5px; - box-sizing: border-box; -} -.record-btn{ - display: inline-block; - width:100%; - height:100%; - background: url(../img/LiveData/record.png) 0 0 no-repeat transparent; - background-size: cover; - /* border-radius: 50%; */ - transition: all .5s; -} -.record-btn.ready{ - background: url(../img/LiveData/record_ready.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} -.record-btn.live{ - background: url(../img/LiveData/recording.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} -.record-btn.live{ - transform: scale(0.9); - animation: geetest_wait 2s linear infinite both; -} -@keyframes geetest_wait{ - 0%{transform: scale(0.9);} - 50%{transform: scale(0.5);} - 100%{transform: scale(0.9);} -} - - -.refresh-btn{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.8); - background: url(../img/LiveData/total.png) 0 0 no-repeat transparent; - background-size: cover; -} - -.refresh-btn.ready{ - background: url(../img/LiveData/total_ready.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} - - -.btn:active{ - background:#eee; - border-color:1px solid #333; -} -.associated-noise{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/noise.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -.prps{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/prps_cluster.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; - pointer-events: auto; - cursor: pointer; -} - -.prps.ready{ - background: url(../img/LiveData/prps.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; - pointer-events: none; -} - -.prpd{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/prpd_cluster.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -.prpd.ready{ - background: url(../img/LiveData/prpd.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; - pointer-events: none; -} - -.start{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/play.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -.start.ready{ - background: url(../img/LiveData/stop.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} - -.show{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/show.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -#menu-right{ - margin-left: auto; -} - -.show.ready{ - background: url(../img/LiveData/hide.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} - -.big{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/big.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -.big.ready{ - background: url(../img/LiveData/big_ready.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; - cursor: not-allowed; - pointer-events: none; -} - -.small{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/small.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -.small.ready{ - background: url(../img/LiveData/small_ready.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; - cursor: not-allowed; - pointer-events: none; -} - -.zoom{ - position: absolute; - z-index: 101; - bottom: 50px; - left: 10px; - display: flex; - flex-direction: column; -} -#total-time{ - width:70px -} - -.adaptive-noise{ - display: inline-block; - width:100%; - height:100%; - transform: scale(0.9); - box-sizing: border-box; - background: url(../img/LiveData/adaptive-noise_off.png) 0 0 no-repeat transparent; - background-size: cover; - cursor: pointer; - transition: all .5s; -} - -.associated-noise.ready{ - background: url(../img/LiveData/noise_ready.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} - -.adaptive-noise.ready{ - background: url(../img/LiveData/adaptive-noise_on.png) 0 0 no-repeat transparent; - background-size: cover; - transition: all .5s; -} - -.slide{ - text-align: left; - border-radius: 5px; - cursor: pointer; -} -.slide-name{ - position: absolute; - display: block; - cursor: default; - white-space: nowrap; - content: attr(data-content); - color: #fff; - font-size: 10px; - line-height: 1.333; - text-shadow: none; - padding: 1px 5px; - background: #1890ff; - border-radius: 5px; -} -.pm-slide-box .slide-name{ - left:-65px; - top:6px; -} -.noisy-slide-box .slide-name{ - left:3px; - top:0px; -} -.cdf-slide-box .slide-name{ - left:-70px; - top:10px; -} -.pm-slide-box{ - position:absolute; - width:600px; - height:30px; - left:40%; - margin-left:-300px; -} -.pm-slide{ - position:absolute; - width:100%; - top:12px; - margin:0 auto; - font-size:12px; - z-index:100; - height:8px; - background: url(../img/sprite-skin-nice.png) repeat-x; - background-position: 0 0; -} -#addPointNumDiv{ - position: absolute; - bottom: 25px; - left: 20%; -} -.noisy-slide-box{ - width:26px; -} -.noisy-slide{ - position: absolute; - margin:auto; - left:-6px; - height:350px; - margin:25px 0 0 20px; - width:8px!important; - background: url(../img/sprite-skin-nice-1.png) repeat-y!important; - background-position: 0 0; -} -.cdf-slide-box{ - width:40vw; - height:20px; -} -.cdf-slide{ - position:absolute; - width:100%; - top:15px; - margin:0 auto; - font-size:12px; - z-index:100; - height:8px; - background: url(../img/sprite-skin-nice.png) repeat-x; - background-position: 0 0; -} - -.menu-top{ - display: flex; - gap: 8px; -} -.menu-right{ - display: flex; - flex-direction: column; - gap: 8px; -} -.text-bottom{ - height: 50px; - position: absolute; - z-index: 10; - bottom: 0; - right: 10px; - display: flex; - flex-direction: column; - justify-content: space-between; -} -.content{ - position: relative; - width:100%; - margin-top:80px; - user-select: none; -} -.left-box{ - position: fixed; - width:180px; - height:600px; - background-color:#fbfbfb; - border:1px solid #999; - border-top-color:transparent; - -webkit-box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.05); - -moz-box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.05); - box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.05); - z-index: 101; -} -.right-box{ - position: relative; - width:100%; - min-width:1000px; - top:0; - left:180px; -} -.crumbs{ - padding-top:40px; - padding-bottom:30px; -} -.selectStyle{ - border-radius: 2px; - height: 22px; - width: 80px; - background-color: #fff; - border: 1px solid #d9d9d9; - text-align:center; - text-align-last:center -} -.selectStyle:hover{ - border-color: #1890ff -} -.selectStyle option{ - text-align:center; - text-align-last:center; -} - - - -.site-box{ - width:98%; - min-width:1045px; - /* max-width:1600px; - margin:auto; */ - /* height:500px; */ - margin-top:30px; - margin-left:1%; - overflow: hidden; -} -.echarts-box{ - /* max-width:1600px; */ - height:auto; - padding:5px; - border: 1px solid #ddd; - background-color: #fff; - border-radius: 15px; - box-shadow: 0 1px 2px rgba(0,0,0,0.2); - min-width:1040px; - margin:auto; - margin-top:10px; - /* margin-bottom:20px; */ -} -.mainer{ - display:flex; - flex-flow:row wrap; -} - -/* @media screen and (min-width: 1920px) { */ - /* body { - background-color:#b9e8f898; - } */ - .auto-box{ - /* width:788px; */ - width:50%; - /* max-width:788px; */ - border:1px solid #d9edf7; - /* padding:10px; */ - box-sizing: border-box; - border-radius:10px; - /* margin-right:1%; - margin-bottom:10px; */ - } - - .loading-box{ - /* width:360px; */ - width:90%; - /* width:45%; */ - /* height:300px; */ - min-height:180px; - float:left; - /*background: @body-background;*/ - } - .loading-box-mini{ - min-height:106.5px; - display: inline-block; - width: 100%; - } - .auto-box canvas{ - width:100%; - height:100%; - } - .pm-slide-box{ - left:auto; - position:relative; - } - .pm-slide-box .slide-name { - left: auto; - right: 104%; - } - - .cdf-slide{ - position:absolute; - width:100%; - top:15px; - margin:0 auto; - font-size:12px; - z-index:100; - height:8px; - background: url(../img/sprite-skin-nice.png) repeat-x; - background-position: 0 0; - } -@media screen and (min-width: 1800px) { - .noisy-slide-box{ - margin-right:20px!important; - } -} -@media screen and (min-width: 1600px) { - .pm-slide-box{ - width:40%; - right:-60%; - margin-left:-32%; - } - .echarts-text span { - font-size: 0.8rem; - } -} -@media screen and (max-width: 1599px){ - .pm-slide-box{ - width:40%; - right:-35%; - margin-left:-32%; - } - .pm-slide-box .slide-name { - opacity:0; - left: -20px; - top: 21px; - } - .echarts-text span { - font-size: 0.7rem; - } - -} -@media screen and (max-width: 1439px) { - .site-box { - margin-left: 15px; - } - .pm-slide-box{ - /* width:30%; */ - width:38%; - /*right:-60%;*/ - margin-left:-33%; - } - .sel-opt.sm{ - width:80px; - } - .pm-slide-box .slide-name { - opacity:0; - left: -20px; - top: 21px; - } - .echarts-text span { - font-size: 0.6rem; - } -} -@media screen and (max-width: 750px) { - -} - -@media screen and (max-width: 590px) { - -} - - .pm-slide{ - position:absolute; - width:100%; - top:12px; - margin:0 auto; - font-size:12px; - z-index:100; - height:8px; - background: url(../img/sprite-skin-nice.png) repeat-x; - background-position: 0 0; - } - .noisy-slide-box{ - position: relative; - } - .noisy-slide{ - position: absolute; - margin:auto; - left:-6px; - height:80%; - /* height:250px; */ - margin:25px 0 0 20px; - width:8px!important; - background: url(../img/sprite-skin-nice-1.png) repeat-y!important; - background-position: 0 0; - } - .noisy-slide .ui-slider-handle:after { - top: -18px; - left: -28px; - } - .ui-slider-handle{ - width:20px; - height:20px; - outline:none; - } - .pm-slide .ui-slider-handle { - top: -8px; - } - .cdf-slide .ui-slider-handle { - top: -8px; - } - .noisy-slide-box{ - float:right!important; - margin-right:10px; - margin-top: 45px; - } - .btn.mr50{ - margin-right:1%!important; - } -/* } */ -/* @media screen and (max-width:1919px){ - body { - background-color:#b9e8f898; - } - .auto-box{ - width:1022px; - margin:0 auto; - border:1px solid #d9edf7; - padding:10px; - box-sizing: border-box; - border-radius:10px; - margin-bottom:20px; - } - - #box3D1,#box2D1,#box3D2,#box2D2,#box3D3,#box2D3,#box3D4,#box2D4{ - width:480px; - height:380px; - float:left; - } - .noisy-slide .ui-slider-handle:after{ - top: -18px; - left: -20px; - opacity: 0; - } - .noisy-slide-box{ - float:right!important; - } -} */ - -/* .col{ - min-width:1000px; -} */ -/************* 数据图表 ************/ - - -#box3D canvas,#box2D canvas{ - width:100%; - height:100%; -} -.col{ - overflow: visible; -} - - -/*量程*/ -.radiogroup{ - margin-top:5px; - user-select: none; -} - -.el-radio { - color: #fff; - font-weight: 300; - line-height: 1; - cursor: pointer; - white-space: nowrap; - outline: 0; -} - -.el-radio, .el-radio--medium.is-bordered .el-radio__label { - font-size: 14px; -} -.el-radio, .el-radio__inner, .el-radio__input { - position: relative; - display: inline-block; -} - -.el-radio__input { - white-space: nowrap; - cursor: pointer; - outline: 0; - line-height: 1; - vertical-align: middle; -} - - -.el-radio__input.is-checked .el-radio__inner { - border-color: #2DAFB8; - background: #2DAFB8; -} -.el-radio__inner { - border: 1px solid #99a4ac; - border-radius: 100%; - width: 14px; - height: 14px; - background-color: #fff; - cursor: pointer; - -webkit-box-sizing: border-box; - box-sizing: border-box; -} - -.el-radio__input.is-checked .el-radio__inner::after { - -webkit-transform: translate(-50%,-50%) scale(1); - transform: translate(-50%,-50%) scale(1); -} -.el-radio__inner::after { - width: 4px; - height: 4px; - border-radius: 100%; - background-color: #fff; - content: ""; - position: absolute; - left: 50%; - top: 50%; - -webkit-transform: translate(-50%,-50%) scale(0); - transform: translate(-50%,-50%) scale(0); - -webkit-transition: -webkit-transform .15s cubic-bezier(.71,-.46,.88,.6); - transition: -webkit-transform .15s cubic-bezier(.71,-.46,.88,.6); - transition: transform .15s cubic-bezier(.71,-.46,.88,.6); - transition: transform .15s cubic-bezier(.71,-.46,.88,.6),-webkit-transform .15s cubic-bezier(.71,-.46,.88,.6); -} - -.el-radio__original { - opacity: 0; - outline: 0; - position: absolute; - z-index: -1; - top: 0; - left: 0; - right: 0; - bottom: 0; - margin: 0; -} - -.el-radio__input.is-checked+.el-radio__label { - color: #2DAFB8; -} -.el-radio__label { - font-size: 12px; - font-weight: bold; - /*padding-left: 5px;*/ -} -.el-radio+.el-radio { - /*margin-left: 30px;*/ -} - -/*量程end*/ - -/* 弹出修改框 */ -.alert { - z-index: 2; - border: 1px solid rgba(0,0,0,.2); - width: 598px; - height: auto; - border-radius: 6px; - box-shadow: 0 5px 15px rgba(0,0,0,.5); - background: #fff; - z-index: 1000; - position: fixed; - left: 50%; - top: 20%; - margin-left: -299px; - display: none; - line-height: 1.428571429; -} -.model-head { - padding: 15px; - color: #73879C; - border-bottom: 1px solid #e5e5e5; -} -.close { - padding: 0; - cursor: pointer; - background: 0 0; - border: 0; - float: right; - font-size: 14px !important; - font-weight: 700; - text-shadow: 0 1px 0 #fff; - opacity: 0.4; - margin-top: 5px; - line-height: 1; - color: #000000; -} -#close:hover { - cursor: pointer; - color: #000; -} -#mask { - position: fixed; - top: 0; - left: 0; - height: 100%; - width: 100%; - background: #000; - opacity: 0.3; - display: none; - z-index: 99; -} -.model-content { - position: relative; - padding: 15px; - padding-bottom: 0; -} -.model-foot { - padding: 15px; - text-align: right; -} -.modal-title { - margin: 0; - line-height: 1.428571429; - font-size: 18px; - font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; - font-weight: 500; -} -.foot-btn { - /* width: 100%; */ - /* height: 58px; */ - border-top: 1px solid #ddd; - padding: 10px 10px; - margin-top: 10px; - text-align: center; -} -.main.page{ - margin-top:0; - border:0; - min-width:800px; -} -.alert .block{ - margin-top:0; - margin-bottom:0; -} -.text-right input.lg { - width: 300px; -} -#type-imgf{ - display:none; -} -#select-type-img{ - margin-top:4px; - margin-left:10px; -} -#type-describe{ - margin-top:5px; - padding:10px; - color:#666; - font-size:14px; -} -#type-describe.err{ - border:1px solid #f00; -} -#save-btn{ - margin-left:-10px; -} -#cancel-btn{ - margin-left:100px; -} -button:hover { - color: #fff; -} - -/* 左侧专家库树 */ -.libraryList li span{ - /* max-width:120px; */ - /* white-space:normal; */ - /* word-break:break-all; */ - /* overflow:hidden; */ - white-space: nowrap; - /* text-overflow: ellipsis; */ -} -.libraryList li span:hover{ - background:#ddd; -} -.libraryList li span.select{ - background:#468847; -} -.tree{ - padding-right:0; -} -.tree ul{ - margin:0; -} - -/* 按钮组样式 */ -.btn-group button { - line-height: 20px; - border-radius: 2px; - background-color: #FFF; /* Green background */ - border: 1px solid #d9d9d9; /* Green border */ - box-sizing: border-box; - color: #000000d9; /* White text */ - padding: 0 7px; /* Some padding */ - cursor: pointer; /* Pointer/hand icon */ - float: left; /* Float the buttons side by side */ -} - -/* Clear floats (clearfix hack) */ -.btn-group:after { - content: ""; - clear: both; - display: table; -} - -.btn-group button:not(:last-child) { - border-right: none; /* Prevent double borders */ -} - -/* Add a background color on hover */ -.btn-group button:hover { - background-color: #1890ff; - color: #fff; -} -.btn-group .active { - background-color: #1890ff; - color: #fff; -} - -.channelNum{ - display:inline-block; - width:100%; -} -.monitorType{ - position: absolute; - right: 20px; - font-size: 20px!important; - color:#A1BEC6!important; -} \ No newline at end of file diff --git a/public/static/common/css/real.css b/public/static/common/css/real.css deleted file mode 100644 index e69de29..0000000 diff --git a/public/static/common/img/LiveData/adaptive-noise_off.png b/public/static/common/img/LiveData/adaptive-noise_off.png deleted file mode 100644 index 15afce28ab73449db1b7d8d7d711891c1636d608..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4121 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 RwgSos(Q7M-&()2A{{UQ*0j&T4 diff --git a/public/static/common/img/LiveData/big.png b/public/static/common/img/LiveData/big.png deleted file mode 100644 index 8c77b7320f76a2fa57c04da68e9b9c8fb010045c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4325 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 C8pf^w diff --git a/public/static/common/img/LiveData/big_ready.png b/public/static/common/img/LiveData/big_ready.png deleted file mode 100644 index a9c4d828ffcf89c039e892e58e9e1616e8373eea..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4098 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 diff --git a/public/static/common/img/LiveData/event.png b/public/static/common/img/LiveData/event.png deleted file mode 100644 index 49a8ea9f7bcd4074004b6a411f91e9c2cbbd011b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8681 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 diff --git a/public/static/common/img/LiveData/event_ready.png b/public/static/common/img/LiveData/event_ready.png deleted file mode 100644 index 1da520233b4b348e05c8b9134be192ec0b11ad1b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7054 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 P$pAMdVM{~w-ok$Xi<(3M diff --git a/public/static/common/img/LiveData/history.png b/public/static/common/img/LiveData/history.png deleted file mode 100644 index b9524def5664f6c5cf783b4eca63e36f3cdf20ca..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6161 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 diff --git a/public/static/common/img/LiveData/noise.png b/public/static/common/img/LiveData/noise.png deleted file mode 100644 index dd1cb25af8f96bdc24f83a8dd910327ec08c0f11..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4020 zcmZ`*Wl+@r_Wsf#B^}bCNT;w!gDglc-6fsUONWF~OAAYbQVRm2gp}mcf^;{MOC$Z~ zcmH!|?tO8enRA{qbI#}U;?$h?n(9h~__X){01zsJ6(A3c{6}yx9&}ce8RCKP+`)#P z0Pu+HA3+1MvZ(<8U(G>YUQ^TF&CAWx-p!p!SzeyW-NVh+!Py1?{AP2seGO7BD5aJz zY>4&brVnc}oWlSXRs=4R#ITSg8L)guLMqc88H2A!PVqP!tlJenfe+4!SX|vD~5Qjv*1Whw0YJj`?T0+q& zz{{9SB%1>rc39m2aEJl*3-+p(zCruq2A~7(6TEDsy({+-O`06~08fy24p@{Vndht z0a)mULAcS|JAizI3=1!2dlg!!;denL^w1V8z$d`~Ml53igl5Wd4+2bL=zu%}*$@Vk zDn>s4Sle=$xqD;-7!7D2NE^y{JtjonkoJq#_xNymm3g(pxkk>Cij%|^Z=EGO1I|hY z-oj8FCIA51%h8{AV$!g!&GnT{+bw6*X?NkB)1e!S6!X#2U2_~dCP1P-f|@no*qFp{ z??tQjZgVO#24?jD`=v`$p*g&KBm7_fm$6W>8ySY&0!`#g`X~609__PR6}0fL#Y)`^ zt+*}62A@B8@89=?(zJvfGN~2x^gE1dK zCisLWj`mR!Kw>8>2)zJ+l(x%687?$nD5H1{0Gj_$o0O(fsrKLkfI@!Q^J-Zff}iAw zcB1T`^v&&5=$66Bd+dbL zMrrF9BtyR!mW;ms8%nl^Yxa($!#NxVQ!-FBkS}H5W33}?!qvx%cqh>DlqIE9qmJF5^s>v( z3RD9#c!l6NU}j*>WjA|DL}s1>W+UoLURTFwW`_wXKT>uA&w+W0wTq?3NF1_NId)+# zeT(b3wV?>6jHG%PF-#-*y03G+o*dL2v#N?$$Ob7+B~=lzk>0@klqV>cnP09oRF*rA zTOI?-Gg5ZR_tE6jjxHk;_ReA-f>X*56!R2K{K|5$bGWeQ+xBY0TZmv#%=nct=JW?= zduu!5)R%xTiZG%VZ1s%Bi6(%igu0%FFH`k1*6)l=qz?a-AYG>2N8*oex(3hBp9fMT z^^>N7()7}T)3(ZWbUk&YC!pnzbSiX-E6ntQKEKlas<~E%H%?hGQAV%(qfE4F7~%^_ z)Ev<=FKRWv4ZpG`tc{9%l^+Ez(y^}Zp^0av>=Cp%GLW0}(8Ji}cKWDhWn6o?r|N@s zP3OZGpsFqJS#FE4O|s3|d^?0j)Kjy3C_a;|GR!3g>gk;G|mEI)`x>I>pxB(9U4Xt->sBB9

as5yRW5qkz;O_5 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 diff --git a/public/static/common/img/LiveData/noise_ready.png b/public/static/common/img/LiveData/noise_ready.png deleted file mode 100644 index ced4026d59bea9bc1b8324517f9351cfd85aa72a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1401 zcmeAS@N?(olHy`uVBq!ia0vp^CqS5k8A#4*i(3PvBuiW)N`mv#O3D+9QW+dm@{>{( zJaZG%Q-e|yQz{EjrrIztFso&TM3hAM`dB6B=jtVb)aX^@765fKFxc2v6eK2Rr<N!r zrKV@**($x?y<bToGsRXZ+|<{=x4=0yBh#a*C^fjsFC@7tJJ~Wt$=*)EroswnKyGSD zqC!P(PF}H9g{>0UTq__OB&@Hb09I0xZL0)vRD^GUf^&XRs)DJWnQpS7iK&9QrJkXg zv5BRnj)IYap@qJIg}$M&uA!NgiGh`okpdJb0c|TvNwW%aaf8|g<k>1^l#~=$>Fbx5 zm+O@q>*W`v>l<2HTIw4Z=^Gj80#)c1SLT%@R_NvxE5l51Ni9w;$}A|!%+FH*nV6WA zUs__T1av9H3%LbwWAlok!2}F2{ffi_eM3D1ke6TzeSPsO&CP|YE-nd5MYtEM!Nnn! z1*!T$sm1xFMajU3OH&3}Rbb^@l$uzQUlfv`p92fUfQ<Z-{NjxK0tIJhP^eidXoP3x zrR0|vYl8Lo`dWGB6_+IDC8v72*eU@{(aX$Cv2rnYGPiWKa5FJ=axpY?b+WWDG;%XF zHgs|^bulnCF@foI$xklLP0cHT=}m#_wKTw~7nB@w3xGDeq!wkCrKY$Q<>xAZy=|3= z+bu3Q&4cPq!RZzYSDbqFfsWA!MJ`fg!-Rk-2*iXZT_6XZ_*3(MDZU7pv=c9??FVKH zwg8_HSEHMqK<1Cc-yeV!Q%R6t@PCG<+YP)J7#L4_x;TbZ+<J4zv+uBi0K<W2|L4nP z@-4QsNHlCarnN4R<ERkl3$;_;UJa;VkBrd%rxhN5rfyag+^uSOccZ7!XW93SM^;W? zV8jHHnI+aKQ+3&__ls`InUPc?wAJyN?+%GL9^DBiI5V@}{r3<1$KJr>prolH*VMD} zvm+b3fow{NR$4QU{;HG-6`CpzX3k73AfO`oWYMgh6YhPU@!Bj@;eLG;*OKox%R<~V zZ~50QbE-O@?3KIks=`N}CFS4a?-#fpf0ib<WcRXPZQ535=F*CWj6$4>j7&JeCf-dC zjJABaHsx8>uK)A?+NC%iaCq6O!uv3z+jXUdohy56UFs{L+*hksv<jQHTH4C4zj`4w zFn2PCl>i7#*fZh!vk1`%cPD)0`Ef>b-O~(@-1~(BDQi5=e1GQ5bmHvg6_Z3RRd-D0 zEK)btx<1*Od1qF(MDqEdtCxQ5S<i~;tfKfJhcf3=b?hhHy)@IE^%lpUZ^_Yv6)8~K e@^n7~GsBaal^#AP%hQ2T%;4$j=d#Wzp$P!dk<3E? diff --git a/public/static/common/img/LiveData/pause.png b/public/static/common/img/LiveData/pause.png deleted file mode 100644 index 59e0a9915dd92864d069d6c0ca913f70e673938f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4220 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 SXR!bP0000<MNUMnLSTYn@bH=d diff --git a/public/static/common/img/LiveData/play.png b/public/static/common/img/LiveData/play.png deleted file mode 100644 index 36e192f7f7f0115ea2a8a05945b1a299ce515682..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2138 zcmaJ@do)z-8lRaqwo%$N?$Qi$t%PW#h8fJDnrK4QlqNdvjLRIC`Z7B-BQn0P<T6nt z*Kaf`w<*b*522LEtx_XSIfEp{7dpFj{yppb@$UDxp6B`fo_DR^`#x*)z38+JSY0d% zh1%fZ?s^#6yM8TD5&8akR8)*?=(xi)3QF`${}T!Y;yhf*M-orWT(`b_g$7aok+*J1 z8^5KC%P^*tv=uof$Z_uavFeyk5?bp{kX3)cna86A%aWFANp&mD^PdUdYmy72{X&Q( zOY6|!apRbb4FEo-00r;d3^h?uWbKV09<)WF_0S}8#OvX$NXS+9scj?xNjtKch5+Px z>&4!eRMh`W=3&1~M4w-YPfncwv|8jh=eqQzR|e#%#FE~uCX=p{gtN1=<E_6fi-}I6 zabcS#xj27qq37iIxt|MjPQss2&kFvfvGtC>fEIS1v{yLCXS$hIF5kBWzwq*h2ha9r zlB}}vQc>xqft68Z1p>#er1@}Pj|C{M8PHyQ7!gM1G0!R0b@ya~6nV9c*PB^?4~XU0 z_{s8faV!i;um^m~EHyRLh6h7bUy{mO5UG1*gA=Un0D-fp-H}xnVgiaSCj>sN3>qSJ z)zoI>%@~#<_p}dnt24qQZ9YcaZfeLxc(@7QIrYPc=JbY-vwD}1qS1YcKF|CRai3t% z-ESEuAl6XFC(D>!@YylrR_D^T=b{IO@Lrc9E5T6*xVeuas9AlVB8kn3UYUVD*q>$% zMN%XE@Li8zsY_Jhu!i;I3_VY|F4Z@29Xwn}lh?B?8&$|A0aSTms$Y;L+_WomcoMcg zr3TE0W5qo=!G)fn>@}8k&Gz}zNyxYsEfwv&r=nuyzAb{~w{86MG^l<#|EGQkT6V_> zGG0Fum(FMIXFKX=;8h`|{g6bFJ9GRpzUbq^nCRd;r5rZ_IdH9q&s;mOYGKIR(o^6m z73m~%Q`C&1*^p4ykePUC5mfIRj=_j~P}_r4;IzO5b5D8fW1=5Go?1$uGvTBgwI-ur z`J_|K9&tD-Z|pF@c4@u^iNlqrDqNr->zJGg9@7AqGw3WhE&N?%$Z^}PNG?_G95Ur7 z*iZ$|O_uvNNmwC9?(2uLefZ4c&uY>?&J+hocT}9f^EP#51@o8-OtK&;L%{-))E=Zj zmd(jOc+8Aq0uLQ9xm_v}&l0+{>!>w+CasW<r<pA>UF0n<jsc$>S1z8U%3HFqWYwJ9 zb_#3grz8uHjdS`&Wexc~fPx|tgM2qh)*|iTuB-=(H%G-P<l4Bh`-ki~1m3F~haUP` za|m4nU6oO<_)M^L&&0x99*+syIGOcpv~=zOat)&UI=^QzL79WmK$w;UO1sH1eQu7B ze<XT+<~y$rr{jC$c5(X=-O|nrfB#%9y1lVNBKgr+ck;K|G=OZ8Q1cfv{I4BAsCE3L zfwu{V;NCnk9rqslXoeynE%+{NO**Xw;|yxf_#M)bK(|J&_?)LVAgY6=)ibMcQiSLj zxZR*(A40t3FulsVj&w@y^SwLtXATqWk4OWG@9sj5)8=ZIVhHaFH|t{?IZI<*JZ_SJ zCBnl)R!Mv-ZaI6pxn{wZ7wu9tMGZVBq(F^R-e#F?&xq-GZ)Ukji95r_lJ}0fXQMC< zG8_!t%ln;!89~r+$`krYjQIo(Ua~I;Gr=SwC||eRWFtlc$tK1Z>XrfBQgc8ja{Iqc zl?O5zpv_2B!xkeiJ366DyjI^$zY&kLOPLr+`onn@X}7zm7xu<nN61;6lEoqL-X#Xu z-sY(GQKtfuhVDJm6Q9FCd>HxV10v>R25ZD4#o*<}#V47_A!}cjBIeP5nWuN6W-~(C zX@fn+V_cK;4y~DK$1G<{tMpcBkUyEnpatY0SH&*CMW`EY0%y%vKPV&ye>8x#Zw!u6 zg!SLnlWRX<<wo?UyOqgPI%jH~g|X9En8@vAWDqAuRDk(g&0<Zqg(3--#xVV_5;ffQ z$e8yX*G%Ycwp2k#J9?4NKzmtTaTO%g?xP#hlf3jRx%hQ=zU|mbpLv&qcdXnf^bqQ% zYTUt*F*o*HHKsdoh2|IVwqM)7saLc=SJk8<PcyfN&&Z9Bysswhc$%Wc?k=ii>;u@C zP%9T<{oPCBeEfV?$v!UezVgaDF=zw!o^oJ!qh;0VV14G-2?uFadDCW97f6ZM=m}Ls zsdfMO4yupo$&UPYPGsa0H8MD-P66zUgTh$FmS%l;$u5;R*=cidE5Hs_V0SB=ZZ`yN zUf^T{zyrf$YSLF`vMbrgnvpl~+D108v%Xyq`VhFlV<?gBj%DB*^*n_wPb+7zK=WvT z?8w*?r=^`hk|K^h{q8Hc%|#GhbU>)C{dcaZVV72bv?}+px4)$o$o32#h@3m3nMMpQ z(y3yW!cfJn+h=8BWaw6yn7GB)AY<741-j&1Ln>ZcHA3)yJMIg~u6#NA<Ke?HB=u-n zP&aS@iLAIft_vZ6?CvBBv+wE735dz-<IK42hz`Z{f-DE<R;CD9v3}~6#vnUvGAPEk z{17C=Ewo&8o<_hx@y&MwrWk((0^PdY95lwI5wEgrnp{OCh|qx`Dn^Hd03Fi{j=@U+ zQgg9cIbs<BlLpwR#`(sh6&JNi-ablNylmop1H&yZ!E2<((7>|?QRI9j@HinApQQvq zX*#K?*{Bk{eUw}s2^nTj71(1y)<nk+H4JFjA5t7@l=}aYpLKSZDdWDWnYSW83n&ja Kx~qs12>%BJ|Ffw8 diff --git a/public/static/common/img/LiveData/prpd.png b/public/static/common/img/LiveData/prpd.png deleted file mode 100644 index ca86cb114d97ef5980c11ee1a847b45e9b9b30a9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6780 zcmaKRXE<D4*!5%#MvKww!w@aedksTI3q!Pw-l7vFdS`T^chOsvAjE?pdMC;t(K2d? z8l!}$?>yJ_eZSu8dVidAe(ZJjeXslMv-jF-$Gy-}B_n1e1^@tL>S}OZ+;il=3rv8! z*BRCq;vNq?bybxB^^?rI005(eI$TlT&-}1}B%gY)V7Ru1^>igS_~U`VM;&(58~F{# zQhL@=&IfPe6?-^SMA?9-LN07uESQ~wLJa;&6UJU8tU>_<2^E?jrBzi4J6Th+LQ!qF zw+{q5PWN_q>)UI?&A;2)`}np51_WHK7A3RF_vzo>-~a9o(KGn*^u7-)00n?VD0_Mc zz<}@x5DY3L69%INLLu}lFaQYn3IN_x0obLiv?=mtLjd7z95WC*R~+oKN5o43HUQ8= zy_9WY1sn!@vq?~AsssRSq`{x%e5L?|Zw-+de~tuz2RP%Rg*+btfTM{_&2NYvKw>p0 z$*i_j0H8D`Nr#`5xLDa7jC~V{05FH7>+~2Wu6F@-vB+Ge2M{&amEuedupv9dW4%5> z1Us9RxU^2$0<oij2$?}mnT^lZ5^@(}d-oC)Ih2Lom{Ou_ffSt)DbQ>$>$!O_I`a`2 zmr$w!hm|oGL|Tm5Bb7dz))`i9_y}CCMv!FEhhxM!3$fS2#gw*+P;vc_kwzukWd47Q z`UnzN{$rG-`7B@hKSq&cjGv=$jQDe-r99bjjEI=T-jLuJnTU{iY{PMks3aV={;OWA z4dcc{9F7sU%k&68u6Nf_v4LDF93!oT;#3n{?<;AJv7+2SXiEM^#Ye6GmG=KgWcx@Z zHD)FwL#rPBn28;-66@Vu<{}VWOkW)aix;k?9h~c;P5jPaPLr6v{sRHJ^|CMuLk!H$ zn9%07R+vCy`4<Z8n>Tg524{S%crWCri${1Lae(>$QR55OQGE;5fr-)2HIROVE$7o& zO|eA4;yKf4hdtbB;;J4qGn=ZUs>*Y{t<SD}H0K3}H4T)`f1lMRE;TyHp8pX6`@m%i zx^0z%v&0W#`Q$w#Be>)H!NQf6b4yw;s*CEtUn@!Cp<W>1<7hr(=4a!ASOsgMH!ojj z7S5*$CxC*^>f)Uj7il{(vyT*^{=Ek-kG|Hb>nRP+e<6xg9nCh1{8)wyCt|M}JP`;{ z6SkSzpwC05Uu7i4%#)7EN4QJwkrRz@f5OZBK4YBlV@%@$n$l_M(~`O^emO-HWt!(0 zfe9wpr7SiXw`+`>Ok%W4u^zE7x0bpRy2|!5y|i%UTjqPymRMXKuk$)Fgt7iv$H_+i zpo%3yBfWd0*J=x`Tl~NXhYqZ&NpojPPDZdx+U-z^E9r#tkNYvB%qPPQf^cdLu2YEW zDdSC$5?eO?#<3cyBXvbouV7Jj*9++wvphoNi5(D>COJ1~#dF1zN2{G8%b>))FQb*r znyl#?VACQeh4<8mSto=NA6&j5LX6FM4ggs{(faja0XI<M`O6^5)wm&4xe+-TkNZou z^cZXn%}p&-&$k@lI!XfnoEQd$6gMuUJ;a_B;F_g)9oO$vb!U?8yp&_`$tlaHWv$&W zK|du{ySiRQY185cELbUaBCsyk39D!dDQyK(d<F@dDGx0k<&3m%H57gjU@E-pg_s_( ztp$X~lKi<*Oppx9EZUu%^7$Pyqfhv}<xmO!M1AHqFv#JR*>AUl&-ORe!R7U|u{#2` zHYd}y9kJ9(g-tpP>2<uSDNBvyM0Yi0S0}TTw6|4Z09m3oV`v}hBD8Fn9I>;p#y;xv znBR>}bHJzlq*Iq(0T<ONJoeQ0PFc@*^5SKF)aBiy<-u$tO{AC4D*epUemr*XR+ua1 zZ&Q{F!z^7m^E<^YYO}vnBBV?g5chv!<t7&&GzS^C&u!x}rEVD6-#FskVi)^YMCG8m zz9d^MzLPW7sohws9{W_y|2G*>M)V)eWEF_4?pe*=K=0DEusd_q0yISE*q&0LmD|fn zTZ@(@iL-l0U_C8Zm5qO;gNvrFJIf9Ib9$8fK$+A7T!>b=E&LQ?EFEV2y*NR?z(t$n z8EoWth1z$I9XEkC!_KC=pC{4|F_|H-)gp-d7nl^W$K7PMSjA7JE*)-NB1$wlak`qG zF*=k|7=lH<mEjC<R{xY4@gDntElP-T{24c`y>l{4gSlFY#FmO#6Po-N6X;DJ{V|{G z)rEei6$4_hZ%Dyf@mnlpq2&IOZ#-Kf2ANJyO_MzGlD?N`V*7Mjq5lo9f=Dz&InxfB z^MhE;kvx+4`_c1>3)T^BM#$A=NH~5cX=aS$%k)kmRJ~P>@<|3gM6D2JFWP^>qND7W znh=q|v24rtul#B7kFymTKs?H`*K1^>&y3D-CPMJ!BJc7VF?2p!P`sD$uWgxS`l2mQ za~+#PEYD>ZW={*tH6+6OYhEz=Yx{WsFx+se4M{RYiZ=0Zz)%Q2{Hm<$A__K%;E*G% z2#CtuArFI{kjkZhj;K{*s%R54v63j0YDU@x5?H8e^FzLS`jl-_h_O{{R4@+OpQ06S zzO`^PJ!7G`l@Uw06WU2axPXDe6x*<N5xZ1Prt|v2#pj&|l}*xT@9`S#eV0dMPCm8p zbfsSRM)f+oFleaV+4rcf3JTiVugTc#Pm#c&KFJvZfBsUP!PA1`$a7w#u%|N14vtFo z?k@CL?LVon_(VK%7}9}mOyZMe9Be(2>QFdcw*aYh%I6+7bS4$9xZ!b00K#*?vv2D! zHA}1dip1jO&%X(U#vi3Gk{8H(x~mW(sgw;GxtBeNhs=4IEyw#l{bCng*hUkAIp~~d zTi4mwx#+n>!B#YC!&7t-LVlWu`y+?<p|t_(ZmrFUw^<yG-}3eyqXtJreaL~037r~; zEp>y-?7|}9t8af913I{wcVkLrDxbMdcVvuB3Q9rLLi{^0OR3_Q6>Ula2jh+gLAjZl z;V@GLhZPX?M99CGALaII{MpP|^=EcxCb8_S<w39=q+_IG?r28b!6k-WZa6t@n|;&^ zyTcw7M?;Lhe^M}dH!olc{SUO@dM$Hf9hhO7vHSHwm!g}iY|CL4!w2$y@2cMa4><}i zDobTa?MzYIcp{$;3OYz9jEdXr4y1sDdYt|_<_m}^sKS*V?b~>GWl>BaPt=giJ@Wa7 z@Q6r)zQJ@{+r!%#S2!-3=-Qj>o7`lclyUC?6`{y<TTU;s!gzBYYCb50L`2x;G`fMv zknwr0xfZJA{QT9hQj{Nw)J~ZOBsMN1M*hnop%{fpzAlxmN^1^SEi1SqI6U4la43}1 zOc7sT2e%UEWYn^~lW@_?`~?yKCv%5=bUzSyoAta@UnB{wgay%xYvwggaHLNoSa8#` zW|*AI>2_eozzl1A8X>+?Cij_IbyDeJ%xtIy<p+k(zI_YinnKP_bO%HkcYnDHlag|l z8vAAO!gEr3_TIB-JLK(=A#DWON9@xKRYfUnrU+x|ufC~KO+0KPX(1wWmPSdQvhr0w z$x7~Ksu-udH5r6T5NWIU^BN1US@P}S#2f5me^KC3+DK~jPH?YCSLE0d6J2b*bCv&+ zujUSBRj4$%xK?(fuR)A#oh4x|E;j1s$``e&+<oM>I(X(kOZBs*luZ2Hw+6&xY;G`x zng}L5ZIqrkl3+f_YtfeQkWuY`$jeyo-LZf*_(H!tse5~7NfFW)JD;&zhV?q@GV5A; zO$QR`Athbzbc{-KD4M!p2_jEBj8`?8W%_emTw=hQ#N_LuXWj38sn~>6-&kKXrx(LS zodhg&^?L5wzjprw)kMJva-ywFTD5db`QLSRK~63|vG&c*e2WAr>XUy9=Iux$_@&pf zeZWBNw8Yzdx*B#-tyYkvgQ#X1x{sFO^QyJ%Aiz~LuZFBUNvGf{V!byur7(2)l?$3v z`uKZgZDpYwYB9gva(@Q@JSGwuaIMVxdxb|A$vDRyWmn2C&EVzt2og&~#M}~mP4-;% z6z|4JRhwOSfbBWU52~7~9||Es=|py(d+aAXHO|g-g?kCGx9r}wR>4lAq|I#H5f9x! zK;g&X;!H(1zccbK8R>{t6OQk5>K_$24Miw%X<+=x^eQp-X`}h=qZMjqhnB%oh0*e3 zqs9G9;V?~2%fl9aQGL#UqQd9iEN7$mkJP&)E7Fe3Z#RbyMQ2PW&Hgb}UV1!OGS)QF zO(Gf3bt!kC_R}<AV|FUPPPUc}Jb(&H^xMuIYh&#0*(5p;yl*@CBDtN9e8DOHD!Fvu zcqz)@<}Z%Wu5q~l^T&Hh2HNtfDKix8>TOM%`~ptn#0`wVH`z)qE@#T)<6o%0Esx+Q znqJNPYK^2tZrvsjfY1u&%ICY6uN*Kq38}2DyoWMx5fOVQz*7_E2_^ug;Wb-3_x*OM zek@|n{!2|gMczzJ!LaF_IC$WQH0&0|1pq$&br$C7Lj?$5D0uMEVj3rC@C#VnCtY!B zxkD82ha?XGE-#+Nn_NWT#E&mQPlzH8OYIfVBi%tnoe3K-@d?cw>3V`l%$i|)|1`QQ z#_aDS)(%1g;YC5>@R5Xf+s@dbK{1+hrqNy$A<>&({K9fKOk=<zUW<8w%5WznUlLuc z)O|s>!&vI+XDHkae<Son1Nle|nRI;rKCRoXCSXUs`(F}~1#f`|sw8mZHaG7lS6I_t zS@3AdI=4-DrOb<_*GOXyJ})N*W8DWD0y{|tCUU;@up?XfptX4(rC2}S$uV&96uZ-~ zJO_bOi?eTHXCLjD<y<3(nHPfnM(+CEI#J_tqSrpfg|0{$QtHuuvT~65gZZ6``^pnw z5>4#KwcEsin`vtKK>F(>&6GA6J%ZimH-6=?vw?N-Y9G|{Dr^*KtL(Urv8XIP|Mb+h z43gUPy=+5+uWL@}!on$_Gb&Z(xbF$Dk~}AfXyDUU&x*}*Ozky0q~a$(d;s<?WGeTs zlombm=|%qibeo58Z5qMtc5>J&e3YxdKj<aVs_B_yx&WxAb<a03&w*xXiQnW%Z+=0% zy)xR5lPZL$v>BOkbMJ4t0Y}oBeTw&%zxJ+&L6p#?UlcTy_uRHL_@uo~hyp1#B?MES z>odx6zs<57bZP514_O?}dwBDx8m=wwWbMK9c5k=e6CKJuG9@vOxNcu;IoxXqSZ-+` z$@?P?pHGLRUT74qy<#<!_=D)UX++nT;rI2%O8#Cc;WX(Qk=%nnQAC-dath&j=(Ia3 zDx=}TwEf+9(GW-ckv%f)pbr%pONCCSK2N2Fc?q+>YZnI9(#UVsPU#f}`@q@1-31!N z6}WfB9grRT!SBlsmI}Ek4Emc+E}KLLj@dF=xxCfybbO3Z)*y8lwh;I4-@Kjp*~qD= zk;EwUq(g&6;e;>{gYlw>!^j=VTFa+7MI}z%-I3d?$S~edRtX#n;whd$*<)k(7}0v_ z<s34XF4YNL$F>M6Zh8ACa-TCJYM^jW90}&0oekq8EM?;+_xMX<mgU#yRR&Cj4%2#L zDK6X^9@kDH(Udb19M{j?-15bCAMyq_)%>n`&*HXoU8^}PcPxlfGohk<+O-{_S4q|~ zmK=0pq7NLE>>T3`RaPF25dF#$tuVBfYO{KZKq9VnyKb4QzLc0q@FRwn>CSxy@D!l2 zQyeUzmy^-F+^Z5VI-;goKq6s$?%`lmY@kfrr@+=je!`RkZ1<eh&v$fWJ>~p}m>Zw| z)p?sUPflBLhGgHhBnAsm6ZfUvW=6ZGi^n&cGHA*JKwZEdmZu^Bs{b<mU%FRuoc_RM zNkneIIY4s>-5jt<g~+SXzi7$Wn#4Xs#ln(UYC76aKlqvCON{b>Z2-_1SIniR2*uw3 zx8VkCzwBRri(a~$u@ZIfQ06dolf0L1L8a#3b^8OZ+1Dnqxw;u&2p^h@fB8mhkB}e+ zG61G)5c`5X?RUfqpCx8@b%#DjHc1|<Y;INWkoDxhMmS5<hq6@+2_79L6nn<~B$59i zRqcv;{qUUwJ#HDz`FFW>65-YSVVfNEv=Fv)sLfwdPCmIV8pAX8ns9Yl6wGg=MCdS? zwJolF?`m4}<=lwnRg2L;&f^OAzn7Kc37ank!2^+O$PVvQ$?D2`scLw%_wUIgQ6dCb z@W)^Zf~lqYxff7DdYq2$e$K97iaE}~`D(mpUoh~dCle{-T9g?_1s?c75eeDg#_Z*t zkiE?zq}s)!<S{Z+@Np@N<8nE48mWA5{xWlcUOpX+$_Ysv@~eoJ<qE`W5Ys9QD5=xR zb72Nqi@d1q*f&hzVuyEC3F{-k0fKxp*VtAa42GGd<@K^AA`z3sQ!9=Ur{btK&xlZv zaUZ1F|B$4xdK&>dLCC2F&Zth$>qOls8F=jMi*@qqz%?m%NisbRF|4p8%tytL)_Se4 z!W^n%c;Qk|#6RSzwyK*_x23OmU=#`h2nhZ2N};`2%7H0r(#ze+_{9TF!pKb?Iz=dc zp+E~u0mcjNm)()O+w;{B_<o9pk7)YP_7gpIrI7G@D1^$&cxHaxz$QDb?-60P&Q_{X zGS^!@NAG{i^1{rcyAzU(*AtkoLE$!|?Y}4e>XsUJC^REW^z~{WTy;4kg1>z4XT|?A z@+B(Ji78qF>C{?ytF66oyZettw>)z11V*z%2VDt4X>9%IzAxnErRS3L_Y<KOeW784 zf|^?y)XZ0}y?5=unMlJH7`Gabk%m3vFZ`uCV3&Moob#cm5Lgo=aa)-g^10d#qDH=$ zMSB=1MDy40Azc(TJ=(}Ofg}>Csy@g_!NtOb@3i-U&v7gZ+US^)vcYtMu}jH8={U=- z1N}RLB>3|Py?qiWKxlrvaGJfv#!4h+tJ`yjMlYbf?FBkhx7;$aUQRObYsmw<%e5zz zCDeq@1iMCoA|PFsZ|@21i@6CMzMX7upv+pKH3(Zxh1{78I6U+zZN=*QBW7$-7Srz- zoYy32xN%d^;PiZ)E`XPN&ImoH;8)UebJlUy+iAi2+C)PFZhiSABv6uhqEmfmoY%-2 zySMMyi%h0=^S@v<E|oSD%$h>y_$AT&nN8mn{AZM@UIQb&u^1BG^s675;%XW7Zq;=r zz3yjMZ_!lzN{%wz?eD|GBT>N?9)(?3@_H_cn$6<Xz*4ohvW7wFEPp-?t}l93XLgY` zyJ52~Ftd~~+WU^t->k`Yt(jtl&5pCpOZjDQe}*qHIRvVLBk9UE!$rhDQ&V7{sZ49N z2c2mXG6rcMVi64vhR4LYSxB>Gj)&btwK7MQyi>dW{B~&({QJ}NO4uy5eEm3}5=)%7 zJkM6&S}5IFW8gE<UAc5kV{~G(`|sOohue$GE-g~h!x;jcQrnEfjQKWqs1{kpJ-YqX zyoS#{5$ao4FE(|+LPcwXIL-_;GLdjnRoFN@FSEg{vcIkFb)nguwSL+~PH`(f<uI;l z{>^!qE2#Ld;*EmL;Hx=mEmrCb+62v20YR(pYmUFXsWc6j@N+LzA{+G^ToQE~OS|{j zU;lL!G}_Y_ETC(KjdXrruA=1?KjOl+BX9OP(>X>MQ~SEnzgeD+^TpymV(5HE*3?A{ zp>Wr3=lYa+?TXHIiqW5IG<hH?ehQtx`#(;H<2xi9qP)iaRlA2}xsz{p_sSZ(Pb0NI z%li+1ocFo1_y`oXOVWh>m26wWQdsK?dvjs=u8!Y`V>%1CIcNFM{&X6oD!bZ~w{znW zK3tztUZ2L)$8CR?pKR$AxsiNgP24pWb52vHZTNIcbP`1azPdZ*Tdw<7KhDe^^=CSB zC*O%yYSRH<-V>WvX;wSTBud!I8KYHLBk<HQ!pJ9~z33Go%i!N9#qVc(+J`Wo$+1Ym zxvb=-gNIfYestxq5tMHzZ1d~I`MEo@e8R`TVYOhQwuYiq+C!6yL2|Qf0gHf}mmK$Y zn{=rtq8SChmH^d5x2q-ApoeTCzt79wI$CM(R(-H<KRz{;+Mia>*7v$0Z1H9=N&yOI zA%mrv-=8x5Kz&jei0Tlu%J?xL_EQe7#|b}UwWR-2(jXN~pwzY#Ls=a879^Lj@1|cy zrmz`PcT(Z0O~@MjhOTy_Vd;l`zMJGEiaJ&};;cny`*_Wrxg+Mq{Xv4{?;rz>mY{G! zlL|}36<Z0mkZD!#Bo~kB@o3*G)XVIo!-O6D$B<1(Vb=<Fts%&AL)e=fP3L}`yFwI8 z;^Q`-UWcR?4D~sDe0$T6^tLL=S2IlksyCeOe8CLmyj%)SUzxQ_y5x0h+UvDC>IWFx zJ%VEX^j74=&*m1kh3SgJ;rFueW4V_Ol23_WLWlU!w~5Ah?Ts%vt=D*NNBxy`wkbIb z{6M%iW*WM<LWf(bS%VoVXw0rtKtW!iEsQxzvNV383`HBuMtS#;wrp=F>8U}s_}dDe zv|9y~twCzy5Rml!8oqAWL=Pt&&IoVgS%f&<6NSO9rbvz3VsVbR@@VvnYi$4s9cnk- zr;O9~0?#EPmqKy&_MK=1tpLs|_t0r#o4dnF`m-X2W_JplX`YZqR`RC^hQaS!?c{{> zToN<isuJ-Za9R39F|lZ4@U11`A1yjT1D5B5b_pDAz;(B^6G@MCAkdC#dS0!aI-?BV zT7vo)IhY<Kv(F*xPY~_}d{B$ql7C<)xuwN_ph_G`X9}_5kujux^sy(xeT${MCl?i{ ztkxs^?u^}roJ~3`Y_Zjk-~ZtTAZd*q@=Nr=Y*ED7qDT+~EER?$^*|j+cZfT<XiR=; z=mWrIjF$!ySW?ApPgny#9sqC~7eWpq)8=FxRQ`|U;sYG*hia5<4y-t6@Ih}A9xfN= z6b|<pc^ve5A;OjrXD&!V)z?acanL`FgyRpPxWd5FdX%qm(1TaxS^wY;T{glv!V~`B k%Dgbl!YBK`6A8b^GjYlpQReOY@0<ysuB-*GSF%L?4?MDwJOBUy diff --git a/public/static/common/img/LiveData/prpd_cluster.png b/public/static/common/img/LiveData/prpd_cluster.png deleted file mode 100644 index 72a7612d98e2c9a7b5bc9c9c33f261f93b17e53d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3764 zcmb_f`#%%@_n%psTjrLru#IBUq(~~7H8J-l#2aBsZz6J8ZqqWimfK83Qkr`%tK8pG z%%$8~7uWD+#3=WBynWUW-#_5{`MiI4y<U%VUXOEL=lOV?=XsrzU}I$>E}|d;006}C zrZ@us?D)OFLj1k#O8FE1ArMF~!2-&A4)Fkh1Acg%(d9d?KXWXLa!lmgbM@OXrKgS{ z2j(@7%rwXBdxwP$VMgqHF(1)v>SZu}^yESwJ5eD)ltw$b0g_TjU2Ld)3Nd#%nDt5X zWzp*^My)p*)H98zgoK?u<{x<ZjY(Tu^|W_A4jm(Xf3!C;x)nz5T{(Ypr6&WvLjiy_ zRpD=iKo~b|0WgRo05k_<;9v>JT?|AJ4!H<0Foo((YES_nLi}_0>rer(>I2wqSu6mE zVsnn4d;kDRwTah>AOIAZi5r>n7Xd(FZme?*-&`a>r%Ox)K%oX)D97@B#++8T9xHhl z(_D*iK!D&W*hAtcTljkSZz-h^AjA}r6#K}XuRreBVF#vM1ifUjmKKrZ=bRY=;->o% zK_F+JM=NJ;&Gjn#qE93M=?>A4$^4g}6#z3KCM{jZ`QiNQsAY3~jzb?yV<R5MyMA(N zdbd`f0A_C5YzV^hzajGs0&lx;oRr6p$bi{ehW9O9#z^ZLl98Um)NH4@=*Ku9MFzG^ zj@9DnUl*+}Vw{<A2&*qfqy$B#S#&ZP^!V6v@e^(5n(FGd7(UyK&qi=mxLRs89`*C% z?4{)Il{{lOM4}=i>kILYu2GOBswzWJ3rvlcgH;=uZ8;!^rZ0(yr5IQGH#C?)uhCz- zuAH9bID<jJC2&Sp+8Odh&XrJGRMpYIv|GB$7c|jNZVntsP6*3q5XAKgt0oXBeKylq zFCCNm6*lm#6>F7CQX9R8Sxz3dKUNg?M`8R74YwMAthd4~1b(~j@eT_wi<Q&pOay@F z+3g=k?S(P$vWbh=>iEfVdj&c)tRVpQ6sln?laXOqpUA5IITw4g#^DkZ8%F{^KZX>| zLteP5=BYr;2m!-RL^uR_d*DI)Jlw`d@?ofPt0>Ma^xc8tLpvSU<4U5-ObOEJPVlt| zu44a<ziSCUyu*um_YZcs9p5a5Oi)W#VjvE2E#r~pyi{})&{&jB6dX+|jDaYN#kDl7 zkj`rQAGWE-I8Q4>;(c)1<_`6kztAdB>+fR(GlRF9LT)`$z<g0QT4adJ?3@qFYU}^t zxc}^`H1^z$2X&?8J42hFKn+U7x%TF3G&<S|{_quJc^VfqK-rSb|3Z!ndWS!C>eQRB zb89{5+B$!)1Fst!cV(^QUhQk~MiG~Q#l|mZowDuwFigf?@JzR#6o<U3`coYQEY|P+ zwN~9U(3UBePB6&|iDN8xZ$s$jb~n++{@T@@I*^V-+v{JPhg+bi{3S5RUQXrCz(^PY z%nY6++1JjNF!x#Rh{ZZ~pOpaq`DE#XLd3$Gs(s#N60b)(%XFyjK(MMn;r6`nD@%Jb z;|&*}I^72w)KMBMx;d4tENEx9>*YN{w0r98U^enf<I?P2MHCbbMi}1mtx+~^yE<AS zPkdrI<#!CMnyOyXl(+jKU*Bm?&Ez?UhW^vml6(Ax#l%wu5Zwk;ck{_)<<uy*{pm`X z0mKvwWLXwRPsiws#)eQ9A6<@BnsW<0{Vue0vpn1!?5Sw2$LLS$uIiHzK`6blgA!fe zN1EyyO>x41c=EQz4uC${q&y$k0-Z}bZ4LUwlRpForGC>8$O!GTh(0r-3!uDKP%6@_ zA@)GP9@Tb00a{*@otdI{RjO^e5!@wJtJXC+)U>drG$$Fv+NGwi4&b)Z9X61+`WI-L zCo9vXA9pGMD2ucvP0bQx59`35@Td_bZ8AJ&<O9U?;5fS`HLbHbVMdbJbv_1i9&ZGd zo`9GNvh*BhUKs~E{_O*>J!p)DOOALtB>dX_=OZ+5!Gd*5lJkr~mRHH@u2%=E(!=*h zztcysIfq*ZVG~V>N{6KuAKs^6kayPyu2Nxm<f7jzFjy7n+uLH5b8Gy`sk+f^_uQb- z3po3?>kD+ktQ$yYP)>mIf`=VTAHR%#Ti###BhroLm(G4}bVc3SRuMPZP*YUST7lu6 zwTf-cemdn6-W-0q-|;i)fQiXM(X%;GIVNM>a3V6PK#x3CLOgp0FGP8f5c#z)YjDCA z>Y=}`lvY4`qwH-dSC*~_bQB$ndt{)-%ee}h4W$t|B!45gBxquY#{3vwTOF)m{@y|N zx5k6!e;Sp+&KE@hd0{;=?~RRGF>s@S?_MR*3sd?F;X#Hkviw-RWk?(qAI65V`uom( z`(5fUUq_7uH`bH&TX!8wBXqb8J-wdn2i9XRKSpm#5$U7u3z?06Gh5x=-G!N%hdhNq zQq)9~g2Saed{+kHsThU%cTCc-TMUwVT9dk|5|<4lPEau(pKy*p_&?7NXOU<Rsk{GA z+3j#jNr}?Qwum9<D+iz5&d-I`dq<CGGVGKD4g%%Hgw&{z^Qw8tJ++GxxYacoJ+X&! z-l0y98~Gg0NSNs~U-Ut47Z7IkLhHe0^K?w#ZyT@n<a0`afTEh3nhlSRDU1l-PbH-c zQ&Lht3=PbQ4@)wPbVRM`w!bzN0-goEl<nA@peQGbpXQCXM;*;K{GO_WH-h_6cCzB% zjB{J)z4pKSu|s>(9o74bOJ?<nU8Kmz!j#RA*r6A3S70z$dsySBql2CZqB^!^T+QXP ziXzza7$4QTJn^3iZG03SwT83xt8qvXL0DNR^|DBA{EHV=<S-ot7w-~R#;_*kX55v7 zG{$Wl&bk7tO08v8RjnngO!r1rYWQ~S<ueu_oL$l7p_P%Qx#hA2NcO`imfjGIm?Vra zG{IF!rbc$|j0~(XZCv~KHO?a{N7)G&_=nGKBj?7dv-I11#;qhl_L~aUs<qFqYFvN* z!-4SB_#}U6vvnZrlB3<%yYW`?#Q8@FpT@>&1Gq~ky?UHoWZxq8HvSqknCs7<CE5Ha zTcnLjv%rImsAeoY?QcW!&gO+(^4=)9X>Vhsfz%YaJo)}T4UbrAI+5(Npp8<Wc4c!< zwJ}{kI`P=D!dHYnU7&Ba3~mM2v<|bj5o2NXRtrl)Tr1l|?Os1WKXTJtNPCeo&@no1 z`Yf!vfH^-CtUEI9F+Z61@^zq+1emRB_MN!`4e{LVbh7(;{`+6MZ&JS95#)Ux_rEiy za3k%#2<64N2x@8{a@p|@-i~iGB+wxD<}16(M{yW9Leh4cbv1+vi8|5tZOX=?x+UqZ z@;%LICq4mjtXop;g`jd&{`&UA9J4SJSYdYdHEUstj!Zx>ps~_TH>b6t*hK5LD=L_Y z_);cEL)hac*N*X$r8$KUv@86t=;yX^7j=&gyHDmFzl%YNFcb4;M`z93GPQ!1oGC8S zyp3NMeswt;Y`mng7QD-%-u~SGZd)4or%H*+7uMwN1=!OJ24TO-<!Iz}K8xh3RPGb4 zkd|y2XTjfmllAp?VYO%9Tgnlpu)P$jaJ|O$*=EHqmtZz`+#wyaR~H=SRM)%Of3;=y z4@GQ!t=h;u@qETBn>Z&Abi@R;);3iIN^s{AA0&U)?pl6Ub><l8L|iJDSu2~mxen($ zc)EcXpmfQ_&sBtQ|1gapiTRXCO#0(-yMx<bz64D5#Z{oO6wGyX{TBy&Z~F#yorj9` z2qTi}PX|UM8*}dk-)#Tnc@%o}vs09=Z0ZF#Nre5RVRTv&oLF)G&MGqT{@3XD&CH}~ z)fHW|C}m)NKx$zvY~3TaNw==Gnbc*Cb6^mD5kb9qCkkR!lF_C_htIi}%!-c;v-}2= z0vqtUiGjf^wNQ$tyob&Se8^5rLu++6KEObSm7BfsMZCj1J3(z&H7u36ZvwDdp#5D= zyZ>;+G`!B@z6<$Sww3h9vm`@L3wyXy+UFtzyuo7@ExL0)DvM1xm^M)$I99cDs6U7< z8?V+?#I<t>$lLHwZ;Rtagr~@C9zj8R-#@;l9Ql`1&A69n7@cWoydCo=saP?#t40?c zGp#|FiJ#Nj)agFr)~5!N5Jf1l3PC@871xhCb?IA^#)rt=hF-(NJDaQrI66>x#;8b` z@>&_Y;}f(4rgwcq<0+dvpS~bugAX*Hw`i!UR6;~}y<4nWig8e$OOr7*cs881J=*l4 zpy&Mla&P!QQD-TS#^=J<o7U>qnMyD-KBh*~6C>KVx5=27Kvt1rpDO$SS^E*g7%%E+ zrkdu<38BS|xCTT<sB|XVj)gNAg`7+{@V%&NhlVGRV#;s+OyQTrtbq`)byIW6e=nDb zQsXts)?nuXnoH2aIst>uazGv(F?-$OX~M}L5HR;Hs#^!l)Jbk})dE7|<<P2VezQu* zYIm=Y69BVC<)kwBgAS^jTiH*;oB}Sr6iG0GOHnlYjF5x(Juq-#AgAAmUMZ3Y_!ooz fUqSz${XhG?StJ|%_>FD;7z)4}Tj9#FZukBJ`sTCC diff --git a/public/static/common/img/LiveData/prps.png b/public/static/common/img/LiveData/prps.png deleted file mode 100644 index 3a35453f282a9c351fe47fde4c6c8e29e56c6251..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8038 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 diff --git a/public/static/common/img/LiveData/prps_cluster.png b/public/static/common/img/LiveData/prps_cluster.png deleted file mode 100644 index 1f6e57da17508112a0fced3fd800a053dcfb878b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7467 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 WqU`M1K@H*b6VTDnSFcpF4*fsO;maHV diff --git a/public/static/common/img/LiveData/record.png b/public/static/common/img/LiveData/record.png deleted file mode 100644 index d915a1858881abe792e3d8b4c901ce6fb4298293..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3914 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 PYX{U+G?l9rEhGL9YxYru diff --git a/public/static/common/img/LiveData/record_ready.png b/public/static/common/img/LiveData/record_ready.png deleted file mode 100644 index f40b2ef59803422f8d22302c0c5834c79571ea7f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1294 zcma)5TWB0r7#_3fnqERJZG^2ahpE&myE~V?G;@+oyStN^mEC38U1>x~oy?qUhGgbU zXHK${AevYa^hMF9SVgIQRQpnFpv4kI*g{FcD1{~^6jP;Cj2fYRQ0kd&Hua$!{9)#t z|L}d^e>wkor;vZXeaHSC48yc%`sE^BYu?x9qyLZI9{hqX0n$4_N_d!56&o>04G$rZ zF_a1_B1M}#aT#?pjPIac9v}m`EL5=(Ry<7DH7v?znC>I4rKqEbfFV@T%>=u4;SvjU zEx{g(<@lVHMk9Lvl#NPL`La4Ss!AGrqz81nkO~-xD8MzwOb5CNwyq25-aF=4P=}Dw z1p6SVfm{Kkv5i1191p2NGzJb!VIdxgN>UJrywJ(<otzK}32_+Zp(uh4V`(&78-_)> zZzC4<66^>e7UZ~UwHmHQ!q~2Gf+R^CFLI(7qKJ?)X%fW^na+WRf{Yy1)-9rA6L^Zs z5FRH9mS*~(1jE{tHJyz%(S~uZVsS#4_eyF2bGfadhOvouND)2E_mRR*dD22$5jl9= zR_VnJAMiq1Fl{4+V7rX*Sfh)D5lpZ%f-R6vdtH;jp<}wK;i~g&os-MKjOh@?R8dAw zuv8_i>l&2PeIg%^MWeBlEC{`+WM}$tDw^U&Ns7zyq}<@jSRFTzNg7=3Ave~P>y3k9 z(a18g^$Dc)+1LQ}vZ3BQ7qO|{23KpIOQb26qsegI<^Jcg8%K14yzS=bQfG7ckx3`s zrlSpyiUNHW{zA6g`%~Wwbo=C&-<9anLW=p)edgAAo*q6nn30oZ_xi8PN%PyIPx$6% z@4o@L{$SV4Y2%0G?Q^a7zFr+$^|x+itH*(}^3ScaJ$uAGwJ2C=eeYzpRcx>Ap8p)| zoyq$CWZhlb{`arVN9wK}y7yX528+|L?CyE0&RE;_>CV|U3b}jo>RWYcW&W4s!IoM7 z>N@v(unzwFx2xM){<qnWKfOLJ{dMNzdhJr_LR*Ide>{_aDOg^)v+udCj%$V6b0y!} zzV&oz`zL?AyYO*s!aoQ<>^c=lUS9~lSUEHnSu~aci%VB7-kk71)3x`rQvq~iA!}CV t+(_yBrNG^9;Q7~oo|)dtmaXG0yO_$(mpib2?gQF)Cexdjze>IO=D%VspFjWr diff --git a/public/static/common/img/LiveData/recording.png b/public/static/common/img/LiveData/recording.png deleted file mode 100644 index 09902bff298b1281daf97d3af1e8d14784a432eb..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4270 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 Q<NyEw07*qoM6N<$g3r(x_5c6? diff --git a/public/static/common/img/LiveData/refresh.png b/public/static/common/img/LiveData/refresh.png deleted file mode 100644 index 51ca60443201848d9cda6615e7ffb8fc32f65aa7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4273 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 TWSJ>|00000NkvXXu0mjfv)KKz diff --git a/public/static/common/img/LiveData/save.png b/public/static/common/img/LiveData/save.png deleted file mode 100644 index c6d8b35d0f4a0a7f3faf09b9356f040ead205ac8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4153 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 DW@pC8 diff --git a/public/static/common/img/LiveData/save_ready.png b/public/static/common/img/LiveData/save_ready.png deleted file mode 100644 index 58fd65a1a9b225e8e8752c77567a041e99a25427..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1533 zcmaJ=eM}Q)7(YJ54k?gIG{ZQrM-g4tUO!KIqq84ZY#GH;YCxU1Ywv&#?ylaQwwRDP zhmL01a&D+`iO5)Ef*)IkIwZ`P5uHGsi_W>R#JDMbEKHql0nM_pw@A@H#xA*gp7+i3 z`#oR3i+L+n$usjaAqbMYU4#$V!Ne;|1K*E!)*b=odC^%TR&jM=gyd<+MsWceb~9u> z?W0MmdD9SG0zqj@L;f1E##4?5IYvV!Vl+`E4A2l%vN9SbgN?KZ2k80`YeDY!pGM#i zWkIS<9<3+rpc_K2tvp?|wZb3V+88ub$jVpXk|+)c7+NIZDAU9Wc+`SS@#4TwT%!m) z6(TlTkS9gec)YNK<7wEW!PG&W(FCtBYjl{wXg0qD>$SRKR9lSd3~C*Q8@0Gz4^KY` z$i`E3xQ{5E&IL{uq(KzJIEqFh5lzIP;rMz~XEvKrtsd3u)gVGGG_xWZRkK1tl7XOw zARh{gA&!L;jAVe@ELspy>5~?i@C+?0Opgf+42_auRHxA<T1p0bJpUicFf-AD=%b(3 z`=7*uzd1~!K3d>5^Fgq<x`ITiFz(=KQsj6)$2BFV=xyLcPH5o5u)~p<ni*bJ9bzdi zA}pW6cs#h96-1H^(r&_n02WOsMB#+lX~K+lo6%$^bULTq?r_>|2AcshV_L%slOc&E zxZq}nX2m3ydWyBricQFY34_c8&4;$oR4LCf@KoD)X!cr)XXTs5QnS}$o)wFNV$j5L zpILVD3MeSy&PW#=X2egkK=C|~_LA}|FN5DA!&~lm-Y6{t?t-IdtAI%reHB%c(9K>g zxJ>o937bFq({EST^J&T~qVn@ePkmbVaOYg({=@fmrrYOs<d2-ucSEV^vRv4@4N94E zj!DJUZ%f4+9MVZ|nsl=2uGH46`uNshnf=19_STMqqFt@TLe+JP+l4C>Yg|8CyRu`^ zN+j#KMZh1(P<$huFYvr~Taec59<&}(EOu2Q>D}&jxG<I%M=oy9Qw5|i3Hb+m25mhG zi_3?|hTM6|<uR42;9{#v)g`sWUZmC|^G0^xv15e;k|n06UPa_<F-aaPrbeuL<fDg) zB|S~>*fPqvWp1+e$KyJ<JKmDrvn`&zDMLBd8C-B_mHRk(N)qNgX|E*b*Vk?=7`<Z; zO4t89oEiVZ-Tv0)iGGFDarCbiSM_{;(WM_MuI>=AoePy$E%(%}H+!?=>HB0})>_xq z`G?f)o##^M_Zm8OlrKG)FMXTia*QA7lRqARKPUd+K)vm(^4c}DurD(d&2tsSO?UB( zSiY)Nc`8Mf3lGJzVCz4NTRAmsEtyz{^=};edidb?E%O5(c0C$hhq*slJ>DoB`8n;E zJD%)-aA-`pz#0cgga263+Tq;2%|~w<Y_|>HJ&O5if6qEY-<v?o&b@IC#*RHW{n_Oc z7xph1IIiBb=#qD8iK9$gZAwaNJf){rgql-(UJ%=APwbPC=apmTcdpO*s!v>ffA|(v zwmcHsf-!6Quaqp#Z`@$qC_6ndDXp%2kn*q`z@O2C6!=@Fg|0meI8)cX+XQ3=xt$fn Jr}o#k{|mcb73u&0 diff --git a/public/static/common/img/LiveData/show.png b/public/static/common/img/LiveData/show.png deleted file mode 100644 index dd491087840b6ed6f1a0af5fb0db5310e48464cd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1999 zcmeH|?Nbw16vhL@$ctGL0#cfVEK7`{h!9JNQ-uT<5~hIGYSkj3CTgvKZhWIfz^G$3 z0%0SL!lXjSwxXt{1{ep5N|l!+idc2T5Jk)eK>}1LH43z`+rOgy;D>vk^PKa$Gxt7U zu6lKbgzo#HFNH#(r%I7Eq?-R--ZXN*yunaIipSnHk|c_u8yu!kW~);X@!EpyvA*>D zz#V}p_WC53{o=u!{ByX+9PCsUSC0k4^R$fH(dOvo+rk%UqPA`+dy-YuQvB@FOpM{@ zqSn3BdonkVcRUaF^ai<mlyp*f5aSg<t$`rMZhtD3SaDcXewe<-F}RlH>A4fO#qFX} z-@E@diIzD7GyR5~8#5Emdpsb*Fx93T8R8@Y9JhPexw2ivq8nR8+1Hwj6X&!|>bAU{ z>KoDL@Q8tJHPATmW+d^r&i-QYWJw}WG8TC<YnCF|boLW%SO4oQ@J;or$di54KVkrF z6vq@6(VGmZ2!`d5cKK7#zx6?OLVjKXE<-W-v{ch!tV^#S7Ik)ppB#om*rsf_Hy5fS z-DJeFIL!*A-Xgk$Vn2kX#%V4~>YGK~<tw>$5=B0ksgZxCObb#;J0<*eU}m{|sWL58 zCDlmy`@zgIIZN3^&HxF2J(yX_O~9KS49jC}SE%VUtOn$fpTbk1&tQ)GI5X6Iur)iU zcyesKxG5yMqGW5$pt`&NdDA|ytz+CyY%h52di^Nzp;hNzch0N%x<PvWjN7{CgahrK zP-1eos2#1E;CV91*I|yGS9LxJ_n`#VIMU*SFsa5bbmo|_Q)}A<!6EZq7wg(%YwSEP zrlsJfe{G;?=cs^<Ni$eQGmXNG9ao1%2tT_?*xB&QyMaLxLMI4e_sic{E#3%M)lVTV zn9xE+yb>E2i9aIjE|nc5%Sm<;IAk*35h27X{rObKL;a{TqNgY(tJTjsSjDdho&5l< z#AG(%f$QSmSMTrT5>*MBxypT3vY4+CC`XqCpk2Wdy~y=}S~yt#eHK)B0s$vY_xbn= z1lDXLzGRzEM=(tav=NY(YB5=_&^EjNM`^q95a>Bo`}K4V?}Y5EkgBLC{EAC#SOO@Q zZ?2uI7WR~{get~YsS$9*6pniVJsJcILA?ij{C6cQ(AZ5teA}IlXE6IMlUiyl81Nf& zanjN~_Lob~R~7!b3v(=3s0780Po(YWM*zo|68^=I>TMZ#*;OS>+1nsaIcK}eW16In z+Ax;cr%Aw33p1`P#w)k<g;{gC#Db}-VqdEuSS|EDorm)q)AIt*Z(sD%mU-p;MIH+| zkcT(elXnN8X~bh%HAVK6Y*IhZ!!1w>Nv1g;(W)6kP8|ZWY>i@`s@X}-qZ0D)kSozQ zQrlyXCbQ*&3#IvFa?bm-YUZm^5?yL*5N}s~JVK(dI~Si7vx-EGhvV4tSaT(bMmg`& zHZtDylBl1pUOZP-)I*{RZ{*@ZS8A`3=#hDBdECqS<OP0mT4)=Y@9w9=nxHW&q;D%x zRALdgD(C^<+4W?TvPd8;JGMk%8GF~UZ|SvfW~QCB7d&);>?l>;+Q+?DLq9{9wJ;6_ p^L)4vE{aUK9S*&B|NA7yImW;1Tm|cI{Yw7Ol+@%5#E`W4$Un}X2-E-o diff --git a/public/static/common/img/LiveData/small.png b/public/static/common/img/LiveData/small.png deleted file mode 100644 index 856580ebe4111c1b457272d3289dedac00939874..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4274 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 UP~+}!RsaA107*qoM6N<$f`n(qWB>pF diff --git a/public/static/common/img/LiveData/small_ready.png b/public/static/common/img/LiveData/small_ready.png deleted file mode 100644 index a4f824e0e6b055a87220598f21808638e061d69f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4064 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 MY<L-4Y2Xs^KTz>#fB*mh diff --git a/public/static/common/img/LiveData/start.png b/public/static/common/img/LiveData/start.png deleted file mode 100644 index f47aa50441513c80042ba1d3052146cd7760fb19..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1476 zcmeAS@N?(olHy`uVBq!ia0vp^CqS5k4M?tyST_$yF%}28J29*~C-ahlfwj)l#WAEJ z?(Nl$zRV5+tbupfMIOxRh!x$Ky@)}hWy+BSdl`CfxSxvrGxuZ1>B;ge0u>$)ns}Kw zY}6D4?(?zKDJTfU^Ruu#vTisa$@t&#+>W1DgqWBjm2S;UXJIKic6RwR0f7@j7th%! zD@-uXv)kIz;4rPD?3pBE<DA0JD^*#JHSXEBw_Ydl@tJy$2X(vm`JA)jeNy?!eUGbx z(}Q3ILE$tNP7aloM^|t$Hir1zn5d#KVTsscEf$s_!$kK<4Gxo{dR7StcpPn!^<-l5 z%rac%<e(rbTsEp{G$2M(#Awzaw~RP2pLNondm1ct$_4vl<LjzF|NTDwWOBaX4L1jc z+daF>T^tmYW90IIg<9^hyVgny69o18w+jlKFnjz?7D#bSKYp8oBV~5s_dX!ia{6&D w3(KanJF4ATSd0!m{b<I-bTjSFUS+0umEC#YbkaZV1QyE-p00i_>zopr0Gi?PQ2+n{ diff --git a/public/static/common/img/LiveData/stop.png b/public/static/common/img/LiveData/stop.png deleted file mode 100644 index a92e86cd71d2c153820b54bf12d8bb554a586dbc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2071 zcmeAS@N?(olHy`uVBq!ia0vp^CqS5k4M?tyST_$yF%}28J29*~C-ahlf&HSVi(^Pd z+}oRmc`>X44G+_w3v?(i@e<tOUX>C2#A8YC6eIS|DSU-2T`~tZrL;VmWcpA2aMz34 zEAtG0zqhI1ed_E(dsdMh5f9v?fmF-^=0{c{6^k0!k5p<rU@T<jI>RN9a_#IDPp%3F zzfW$}GdDBIO^7rPR@Z#MXn9ilZR$CoMA}`Sxl0<@pQz+-nQp25fN^H2ap@$c|G(Lv zT)+SC$H#}?o_*eWzrX!)#pjQ2PwQX*xA*$L+WmX}|9$@W+0o_M|L@)Yz5BZSy}iGx z-rkR{zyJMH!=K3b8{7Z=Ib8UsWD)z5SM?2f@88#bXsCHTRqpsljSlvdg-`9*R)`#7 zE?TJXsMn%+qW+7E-8$71LGevGt$Q}KDfoX^<#7D>Z_+#EY5fsz^*8oMtPTI0DZepm zBe(r~<!eD6@*CY0Yma;qJng<lT~3z2gZ&8eBgU7ze(_zJJ(csmqt{yXx@+YVmQL8$ z+~f4C<0;4(yQHu7azG+2^^p6TAfU#jQMJAGp5OIf{>|!ptZ+i(_U3kuPj5o}|1Z#g z>8)7nDc|}<_3reN_)E2(@9S-9j=Va1s--_-=`^71)?ZR!;qb@azg9#_DtWAQod7hB z*OBAW`Zk5dX{j9wOT`?gWqmbeQS{g^)zmX<s$$cWD3-`oSFa1SDE!UhFv<#R;|R(U zOj#Kg?<~;tWwqjoQG-UqXfz#-W|7figwzu9+J5d&m6>-pU$>DtKD$3ZzOM4yf%0F? z{Y^EyzWx9A_V3--{O{vx_kMf7cK`cjd_O8#&xIUdE?QQ8JNqoFje_OOMq|+r4Rc=J z{T5Zs$nW8o_P_@yoL;qk?=@hBI%%?`MKB}(iJA9u*PDueXn2yshFg2%euk^f3j@S- SrUn9QdInEdKbLh*2~7Z9^Mn}y diff --git a/public/static/common/img/LiveData/total.png b/public/static/common/img/LiveData/total.png deleted file mode 100644 index e208da2d8d55ac1bb9a282aa0069baf9d351a081..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5157 zcmaJ_c{tQx)c?+mWiXh=HYf&>ESYSTEW=nvS%#2A&Di&SUq={9mhAg36jCWmWGrPT z3fb34$i6SVe((S9`#hg>KhM4Q+;jf8&pr3^oTqpl4Mut{dH?_zHF2u?=iK_AL21rM zrI6xR=S1g<d*A^83@rZ{2*}7n0sy_X9Ttnn+q!tVc-XqQqBOBsl&iaowVk6C0Qin& z-}5m}+-Fyu-nU{lQu(>@Hq|i{;1?)|cEBGLz!LysI2?ib5)nmj#L9Ld3uoB&WQZP@ z9W|FHTx+BJEXpL>(DdeC-ieYALH_G&YsVwU4NF-|Cw+eX%%lqlp%;q<_P{Ou2zj}m zym7j?w$4Tpm<=~Wg@P}3x7bjB0e}sNpKq{dmC_05jSBz<T!*e*MzqiV3#-Qq-3L%F zfEj+o5(6k+ALzJd(7_6HAc3lT?=9(pBmi*q3sm6;UNHizJDQ5rz|XS82@c?A!NoEt z5Ca0zg*7gNF`oc}VW_?sn9>5|m1Fp?QBf*Eqz4}*HNd1sYQQ_zIG!0p07&U7Vm$!r zCK$l-u=GMuT97>SVAD+5W7mjU90;IvC-<h_9h73NP5#E^v$ZiZ$2ZsF_*R9;aTRV& zx4@s3nkm47TY_ly!2p2kaG?3@rc(UU;==5r^^)WI&X<BShYc5gMZV4HvxetjDgdt2 zzdrW(*RNrSYdfgQtI474AuwhH*iIjsNsZIxnb3dpKaBRfd4l1|$;WrdU%E)oz_2E0 zp5KU`k5>FAHS01H9lUqGdp7RB0k~yg(km#1VW)0HXYWe=*rjeNmFv#?O9E%@5h16s zR|n>#-<pcCJ=-*XT8rn`ZuhIxf1%3PnLLMB4w2mCO*y4Y^kD?b$+{GskF+1U9Vx+o zLEva6<3Vo#6TB_{oQvTF0Jo8r^xFr3#HPbIj3@}`O)Z)SfQH{l)6ygkt*=l3P|XV! zsZyqeHM5pen6jELHBdOfL>g5L>@r0ju0pdwx=5doOi|`Epve8C`cMVo%Da#;3$Jdz zev@%&24Nby;>tloK<2a{8_0B2${kP^BA;B;1@qXS{C!Z4^jIo>B2GGiRD@cNV%6jS z(&1hNE>p38VTKdf5_f046=RAyiXZPlNj^Jy)}V1cDfFEl<OkRbQ&-B%_3Ue&$AG6a zvl%I+z+r%^nn!{3v7lt&?r$$oAm~}F1!>?qllAjaqUrNUHNsP4E}?dIi(kevMsYG0 zmyB<CvvE^$i0jbH;Aj%dD)dn}yv6ZJe7uISmNB+ea8;lhQ4hUO7Z!fK<qCgdscyBP zKjN^>*Ie#xys>(@&|f|tz8t|vSD08HC*m$MbtWw6(DMn#OKLJ`I^f1};zjq06bIpU zSz1CX@lKsn3($9@a#ZSznt0}T-Gt-L)`c2Yxi3+3T66`M^^1}acVw)Teo+<YN~)yi zz0&I~%lQF)6(yHzqUn_9jhDC=S;ivml_A)h$&T$V5-%M3mSJaOw{I)4>{(AY8OEcQ z`Ym<P;WzE_(sJ024~&t_7}kz67v^-}4B#w5)^JLsYrUlYo|@iafF6<LPPa*8PIECd z7TFUCWP8ztNR~@BN)Aq5dSzhfVW>3Z_lm*bwITECM@B&})eYa{=ga7Ru)iKEyJYyO zOt!L5-$y?V->>($@blx-Cr1{H@5mAAd1PFnfkn+%&KN%SuaZ`q#wx?^MvxUThcs>T zhwlzowY;g1xxFO<wC-U&URjr0zp&0;bn68%eSJH#aceZ|P9I+%V_$$cL|jSy?XZ1? zbwzN+jWPeR&tv9m?rZF8D`V?kuGyGuvFu$rC%Jtux_a4qz50myaj*VAir+0({Fdo< zyLOqDCkHfkopuU$IsU8<uMhCvKubeq<2fC#-?Z51iMROekY-mqun|66dZF94{zaQ@ z;ecl0m0F>{kv8)(z8s3y<t`V^%cZBRCLH96=5~_fMQuf27Cv9?O>)Svt{<rW?%C7R zAvRI|v%GUP_SR6ALk{utXv6qPOhmC-)|WfcveE4+V<~~%)G692`{ijyibj4$yGA4B zn!~0mVk=I=_Z{)v_cLLca+%#9ODi%f+PpKJrdj7$<E{=~-M!~Lc763^m3}p2m1gbM z9&t}_&uOp!QqM)irQzG2w*%chy?VSIT^m-4W~E!p+R9p&@7r+O<a=aq%W%BB8{dsc z^-kru=T!2n#5vc$JNXya-}=E1-**c$KV?o-j5__8>{0j<^!ix)S|)d=S*yaj^3KM- z@tj-Fe0E7@Ny~QplEfqLM-LtyH}dGd&l1WRMmu$kw>_L`><%EvDJobU^({jbzbb-r z{c|Hfzx>QUt?*eOcsMvcnD!*+`1VovY1~26tlt*<rVcd$)C~FsY7FHHeGhSf@KU8y z71PR4$54Nwk$@?(slezMAMm;-x==W_xDPF)CTzqq2H3iw*=#1<N9Z0QEuP^EvE+W} zG(6bl<)PMc&qrv~jD}-WEw3GCI>%uU>s1lSSJ(dNRp0Y;^V}C3E?;<XY||LK8CUka z?)JF-M|W?Va(hR8M~l_WT2k$++ECxf1GUXmgeS{oy0xbZZ9m%-p3ZxKmmD*svKBH7 zQamjxe@ztBaV#$Pb?5lVKahD4A9yn&4Uzgz=~oy^1uucr{Z-<ueTii&JTpR9?Q=m) zf!^Di+B@<Ii`hoWdp9R<2VswF$-3%ZU!~}D^+o+&a3<6VY-<~-4o{W$kGhSD-%83# z$|}@nv`eWL9YNO!*XUY{iY78H_$@khaVJK0?Jo!%dTfugIYwv=6jtk3;+fdBBuYhn zA0i+6*5Vp*CO@R6t<U}(3NSoj4{VID!q-;4|5@w%U~`7QN$nH6K=eHhpKI66^y@8s z?KTF3x^27PrJUmDjP(t-?ENOTV)erYi;J<MPwq=6mnC&}J6G<Q)LF0*lPwL`?Nf&y z8ij~(BtI^9EibJ+s8X+8oV%~{cqnk!&uMII>P}79+6r!^XvMjF|3TB6JB{|rL*K|7 zj1}CQF~7JUJKdS`%=fq7jqR=O?dpZd5M|aL_IxZK-^!OQ-1E;4G3z!ut$1DYc~od@ zxM|WSV4ErI65H*m^`zB?52C*J9U9ewcAW?RKADW;GUu*uS`417)$q~%k`otD96-CL zw!UKT>Ebx?rm5*2os+(73=dAu^;agzWh`?o!{>vmi3n4R@~`GA(=P)9=JCJt`&uVj zd#BSwYIdP3I6u)rxhAi2@j|COh5W|#-_HxPFFm@X@ww(-@cQ+63$J#WxcfPI*}@8& zhYL><=SwV0a<X-l6n>;n`0Z<*q&t}DnKjyA`*V~yJpXFdJl^lzo?41C+h}mz;omnO z7p%Ekp-`!y5$t+8i+l6XWyNa0d+ok_rRR?J%I$&m?;%I_lp&9igOO`yxqgl7Ust(n zoP{PY)(Y0|9(eDqarn^x)tWH#kMn(V@_Fp^5A6~2pOEUaw^kIVnF9ia(BJxid7X(a zGUqJm-}lJbL5&`bApv!Pn7C|SUT#LN#x2Xf#MzQ}B`p12!AAf5Ry<xWO`pyk-|aU( z`MJSx7~S7g=~uSxzW%!Gja<7xJ8wgupTg$LlZw9ap!M4I`swyY-gMy;%o%=r;K-uA z<eTok;!@z*iNt>R_Oak#uGG2OL|NeUwE@6i5CBMF0C4#4oR<N>3k?8&2mm0L1^^r` z36H<1p9@ocO;u$B-?7!~aku%Ka5c_nDsD)TQQp2bVriE_vWgz^z*0DaIhre$*CK!O zPrI_=ztvbD1r*cqlsZdV<;aREyv@rT0R<!fPne@%&rjYey_`Ht^Oo@NH<{nadBnWF zyo)ZW37OmLuL<!!^|RjE2f?1PpVU)wb(c@%OS-qSDh=n8QXXdAw@nKoAU_{4d-yxN zn<<X(Sw>t!5Ta@ic>DR6a0CEbxG7muk~vYA5f@XvwW>tQ*QTkhkG*HY<l(<fLxcVS zp27M&#zsh8rG^Pgb|ldLP7){a!C^NfssNG_9W{(Owo1Pg_WZq|FbM?hV5N{Qw%h|+ z-wg4w9Ybo&W~cq*6x$nJmK_}0)u<<)<pbpY1lu3v;jF1)>dE4Z6ZOo0LYrznCWxU= zdTH+V-zfZ8Ge5>T{>e%8yL@<L{#{wLvGXc@`&DtSMIkZRvBj87P)DB$d&UiJkqrbN zv38B;MnK-wZH3G|LiQjhQmmVwwgM`s-r)ZcX<6Ay`KHQF9g)|4!jv`+M;oN7VZO0E zlR^tqgP=4Rw>u2zq@yrqg@+8`40la!s~uKnJ%yX<Bl1n|MzmxPRt=SW#ngu98~Ljv zWa4#-9~AJx{}M{U;8BcEO6g9ZBK8+O`3@(@<=g8jy#zt}(Aw_bF}HjmSVnlf+E)=B zaOckCKaYqmYM9C3H4Bns7cf^Mc|5DoSWn8Y4IC>Y_u3w)1k7-rtv9WM$@JXz2M{O4 z0!err?)o+wc;djTSL`)EAY;x&E}-L{5bCl0#QT<A+1)bL4oNQxC(T@pn#h%6z~&Et z!-W$%Z^I%KL2&?FkszGKim5X7t{4wG5$tiQe;NcnhOT0LpD4>;(7_z+S@-j{Z1>7l zkkAn9Q~9lL<}Vt7q%7qh-;_*ST2QCm0n0K8fkg`+`bDH+)L%Peh&1Fqp)o30fi{N? z`Ia24S68tb2>T)C7a*!m&L8cJOd-T)SjX`$sfOq8v$eb;!e6STaytwVekU_(QBc{i zH*85b^z(ZBBXL+k`AjWHf*w}!@LEqZs+T@-9t2Y+##=^wn1>@BTk*TXcUI6)QQY58 zdWD?7t$h@YpM+)`Em|2oHROd^QWKDu^9`CQ5L$A1P*hjyutOM;PZVesrYokQ(bVh> zpr%6}aXHJNEtOUYxo%)A4^nVjhKJ@I7!8x15Fy0lF85C#fU&1;zPfLkdh_X#@4OyS zuoO~$<LslyRfR+hKd9BVHRz2NQ<yUU3w}j;!tWd8W>hJEb|?th&{DNQI%le2zCRE! z%@W%3ID(?l2?_H+y7RiWi6GEzp+sUKBa#Ga0$>!^FA@L*?MS5jM^IwaDhH`q3C0i% zLX!AK<UM$~(EwzT6PeYD5u`(sl%d$2I|%emGT(ml7XCbo@M|yxjW?nu3!^Q00Lnaw zh5*nIpr}N4*$L8ljB}iy$xHlr?t4B_;kNzzpTqwGCVd_o{rt%+y5xqnPLozWfDJ$n z@A>$DI`@?Ih_W0MO|<>7Lf5J;`$F))rJ-9SJQenDV`I_5EA+-L3jLGk!?m-orlMi9 zRyw*#&~}^1AyY^j2D#T>ZNXuZ#-~{6YG#iOt)rnyv<T7q#Ev0U(o)dcECGxk0#&2M zE=-uNkV9D{gD;*JEc*mOaOTR{z7$@s#T3%IyUrc+EP-*9wa4^zB#1@9Se9CHWRloX zDNHPA?|z9|^#4H}$q|Ufj<)55UqZTTOMm7InULI|_Ncf>Q<MPlSi{RQhRzBlA5?O9 zFv1YiBm;f>N?p6n1VM6hyqUS0NLh+@hE&^mwTwFnZY98_*Z7j{Ggi7Vw!IVo8a#FF z4yV^|!DyZu%(Y;P-)XuHpLxe*4{inCS?ZS<$wxz-|EY>xl^!vX+Qa^_NE9cxoG18T zdo5={&v`K*{rh)3QdR%yBSGy&lil|nlQ%e-!t&hWR`Zl)ucnsQnK89mB^Zbhi`wtP z)+8=>`g1Xbg`3A}ukLuC=FjRL)0FOfc4I@wUrNec2?@oou~22!*{~H4@eB0UDNS;u zN2H{!%I0c|KIKAQR?k?a9PqD5j(po&8ls?jnS5!<2KFJ&Xanjn;oDW~lr+_2!^iAJ zBFLhUt2!x6*W#!mr9x_~e>9_Zn0CbY;+~B}%OQ(KK~gdDB8CVNV!SnNk2ET&6q=0I zN~{{;sbjT<MvMP*63}A^!!huu0{H}|(g<RPWv1{&9ykTfprn3OVWBJon~T%gU_PK~ zX3&>`MYz-b1E3Ealoep#4P561&zhN`=NBgb3ZX^87qDzH?`d@IHr?u<rh=)t)1e&0 zP+|qRqv0M3UhX<@fhm0T-kDZ2zUWdJIs&Y`Ro>f-_mt+xgagX2z&*|QKfJ-WT@r~S zVyVH?<_Pj!4K3!kSZ>Ocv_Wl+B|DkR<pRl-N+sfbmR2hO-_xl0v}EtAyp3Lsv!jK} zM%Erz4=e{S5=-~IXD<plsMj6suU{jS(osEVA&c3&!Na+ce_70C{e?*1#G_5ln?3a8 z3=AsM(>g3g15y{3pHikSySzoC(AVKkhc!fzRMcdDyIl!?h~#QazlJM`mB{Z?u=-0Z zWoi?@3#idT<!;<3lZY8zxA{T~q^KZF$xIOJ(4Fr$I2<*?3a(ROyT<bgw%L76pS-5b zynfH;)Od(O)q_0{Ok^+lz%LG=;agz!6|p=kmn&j0?om(YQ<8X!)eXQj1Dj_#q9hG7 zoA&3vTD<9=?8Q^xd;TSzZJ=SHa}r3O+<|-gYjJipQ&XecYi}B9VMu?2EGkktrzN%3 z`Fqru%3H~O$k9MnbFtb+Af+vzO1iQSHI?Ll*CaK|v8to0@Tve!US0wX3<IK~0IcGC z=)fyC&`Bzr1Ay<-9BTO;YY>F6(aeJOjMq!)$_d=0g-uGTuke(Tqo?0712i+jE7|l% z0|+#X#v;N3(otO3FjgJpYcE4fM&(T$M>_}I^>I~fUz~oU^nUN==Hy3yB(%i>O-BWS g{eRGtIfDToWjy!pefKRmS7LysnvQCv3gOBB0I#@GN&o-= diff --git a/public/static/common/img/LiveData/total_ready.png b/public/static/common/img/LiveData/total_ready.png deleted file mode 100644 index 31cbed0d069fd5f772cfbc8fdb41d1189d76f81c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5008 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 diff --git a/public/static/common/img/download.png b/public/static/common/img/download.png deleted file mode 100644 index 554605b4026582527db92cc81c65d691b31e14cd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3619 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 diff --git a/public/static/common/img/sprite-skin-nice-1.png b/public/static/common/img/sprite-skin-nice-1.png deleted file mode 100644 index 7184ae871cdb7246a260855bcd01b3561a1c5a69..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4433 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 diff --git a/public/static/common/img/sprite-skin-nice.png b/public/static/common/img/sprite-skin-nice.png deleted file mode 100644 index 114eb97b5b8d20d7c63be08a38f41e0b94431add..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4263 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 diff --git a/public/static/common/js/axisInfo/constant.js b/public/static/common/js/axisInfo/constant.js deleted file mode 100644 index 26d81c2..0000000 --- a/public/static/common/js/axisInfo/constant.js +++ /dev/null @@ -1,603 +0,0 @@ -const createAxisInfo = (size, minSize = 0) => { - const textY = minSize ? (360 / (size + (-minSize))) * minSize - 30 : -30; - let data = {}; - let prps = [ - { - type: 'plan', - size: 50, - color: 0x1890ff, - }, - { - type: 'curveLine', - line: [ - [0, minSize ? 0 : size / 2, 0], - [0, minSize ? minSize : 0, 90], - [0, size, 270], - [0, minSize ? 0 : size / 2, 360], - ], - color: 0x1890ff, - }, - { - type: 'line', - line: [ - 0, minSize, 0, - 0, size, 0, - 0, size, 360, - 0, minSize, 360, - 0, minSize, 0, - - 0, 0, 0, - 50, 0, 0, - 50, 0, 360, - 0, 0, 360, - 0, 0, 0, - - // 0, minSize, 0, - // 0, size, 0, - // 50, size, 0, - // 50, minSize, 0, - // 0, minSize, 0, - ], - color: 0x1890ff, - }, - { - type: 'line', - line: [ - 0, 0, 90, - 50, 0, 90, - ], - color: 0xcccccc, - position: 'Z', - num: 3, - margin: 90, - }, - { - type: 'line', - line: [ - 10, 0, 0, - 10, 0, 360, - ], - color: 0xcccccc, - position: 'X', - num: 4, - margin: 10, - }, - { - type: 'line', - line: [ - 0, (minSize ? minSize + (size / 2) : size / 4), 0, - 0, (minSize ? minSize + (size / 2) : size / 4), 360, - ], - color: 0xcccccc, - position: 'Y', - num: 3, - margin: minSize ? size / 2 : size / 4, - }, - { - type: 'line', - line: [ - 0, minSize, 90, - 0, size, 90, - ], - color: 0xcccccc, - position: 'Z', - num: 3, - margin: 90, - }, - // { - // type: 'line', - // line: [ - // 10, minSize, 0, - // 10, size, 0, - // ], - // color: 0xcccccc, - // position: 'X', - // num: 4, - // margin: 10, - // }, - // { - // type: 'line', - // line: [ - // 0, (minSize ? minSize + (size / 4) : size / 4), 0, - // 50, (minSize ? minSize + (size / 4) : size / 4), 0, - // ], - // color: 0xcccccc, - // position: 'Y', - // num: 3 + (minSize ? -minSize / (size / 4) : 0), - // margin: size / 4, - // }, - { - type: 'text', - content: ['0'], - w: 55, - h: 25, - color: '#FF0000', - font: 'normal 500px Arial,sans-serif', - xyz: [360, -40, 360], - }, - { - type: 'text', - content: ['10'], - w: 55, - h: 25, - color: '#FF0000', - font: 'normal 500px Arial,sans-serif', - xyz: [288, -40, 360], - }, - { - type: 'text', - content: ['20'], - w: 55, - h: 25, - color: '#FF0000', - font: 'normal 500px Arial,sans-serif', - xyz: [216, -40, 360], - }, - { - type: 'text', - content: ['30'], - w: 55, - h: 25, - color: '#FF0000', - font: 'normal 500px Arial,sans-serif', - xyz: [144, -40, 360], - }, - { - type: 'text', - content: ['40'], - w: 55, - h: 25, - color: '#FF0000', - font: 'normal 500px Arial,sans-serif', - xyz: [72, -40, 360], - }, - { - type: 'text', - content: ['50'], - w: 55, - h: 25, - color: '#FF0000', - font: 'normal 500px Arial,sans-serif', - xyz: [0, -40, 360], - }, - { - type: 'text', - content: ['0'], - w: 55, - h: 25, - color: '#0000cc', - font: 'normal 500px Arial,sans-serif', - xyz: [390, -40, 330], - textAlign: 'right', - }, - { - type: 'text', - content: ['90'], - w: 55, - h: 25, - color: '#0000cc', - font: 'normal 500px Arial,sans-serif', - xyz: [380, -40, 250], - }, - { - type: 'text', - content: ['180'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: '#0000cc', - size: 20.3, - font: 'normal 500px Arial,sans-serif', - xyz: [380, -40, 160], - }, - { - type: 'text', - content: ['270'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: '#0000cc', - size: 20.3, - font: 'normal 500px Arial,sans-serif', - xyz: [380, -40, 70], - }, - { - type: 'text', - content: ['360'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: '#0000cc', - size: 20.3, - font: 'normal 500px Arial,sans-serif', - xyz: [380, -40, -20], - }, - ] - let prpd = [ - { - type: 'curveLine', - line: [ - [0, minSize ? 0 : size / 2, 0], - [0, minSize ? minSize : 0, 90], - [0, size, 270], - [0, minSize ? 0 : size / 2, 360], - ], - color: 0x1890ff, - }, - { - type: 'line', - line: [ - 0, minSize, 0, - 0, size, 0, - 0, size, 360, - 0, minSize, 360, - 0, minSize, 0, - - ], - color: 0x1890ff, - }, - { - type: 'line', - line: [ - 0, 0, 90, - 50, 0, 90, - ], - color: 0xcccccc, - position: 'Z', - num: 3, - margin: 90, - }, - { - type: 'line', - line: [ - 0, (minSize ? minSize + (size / 2) : size / 4), 0, - 0, (minSize ? minSize + (size / 2) : size / 4), 360, - ], - color: 0xcccccc, - position: 'Y', - num: 3, - margin: minSize ? size / 2 : size / 4, - }, - { - type: 'line', - line: [ - 0, minSize, 90, - 0, size, 90, - ], - color: 0xcccccc, - position: 'Z', - num: 3, - margin: 90, - }, - { - type: 'text', - content: ['0'], - w: 40, - h: 25, - color: '#0000cc', - font: 'normal 500px Arial,sans-serif', - xyz: [360, textY, 340], - }, - { - type: 'text', - content: ['90'], - w: 47, - h: 25, - color: '#0000cc', - font: 'normal 500px Arial,sans-serif', - xyz: [360, textY, 270], - }, - { - type: 'text', - content: ['180'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: '#0000cc', - size: 22.3, - font: 'normal 500px Arial,sans-serif', - xyz: [360, textY - 2, 180], - }, - { - type: 'text', - content: ['270'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: '#0000cc', - size: 22.3, - font: 'normal 500px Arial,sans-serif', - xyz: [360, textY - 2, 90], - }, - { - type: 'text', - content: ['360'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: '#0000cc', - size: 22.3, - font: 'normal 500px Arial,sans-serif', - xyz: [360, textY - 2, 0], - }, - ]; - let prpd3d = [ - { - type: 'plan', - line: [minSize ? size * 2 : size, 360, minSize], - color: 0x32494B, - }, - { - type: 'curveLine', - line: [ - [minSize ? 0 : size / 2, 0, 0], - [size, 0, 90], - [minSize, 0, 270], - [minSize ? 0 : size / 2, 0, 360], - ], - color: 0xcccccc, - }, - { - type: 'line', - line: [ - minSize, 0, 0, - size, 0, 0, - size, 0, 360, - minSize, 0, 360, - minSize, 0, 0, - - 0, 0, 0, - 0, 50, 0, - 0, 50, 360, - 0, 0, 360, - 0, 0, 0, - ], - color: 0xcccccc, - }, - { - type: 'line', - line: [ - minSize, 0, 90, - size, 0, 90, - ], - color: 0xcccccc, - position: 'Z', - num: 3, - margin: 90, - }, - { - type: 'line', - line: [ - (minSize ? minSize + (size / 2) : size / 4), 0, 0, - (minSize ? minSize + (size / 2) : size / 4), 0, 360, - ], - color: 0xcccccc, - position: 'X', - num: 3, - margin: (minSize ? size / 2 : size / 4), - }, - { - type: 'line', - line: [ - 0, 0, 90, - 0, 50, 90, - ], - color: 0xcccccc, - position: 'Z', - num: 3, - margin: 90, - }, - { - type: 'line', - line: [ - 0, 10, 0, - 0, 10, 360, - ], - color: 0xcccccc, - position: 'Y', - num: 4, - margin: 10, - }, - { - type: 'text', - content: ['0'], - w: 40, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [0, 20, 380], - }, - { - type: 'text', - content: ['10'], - w: 47, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [0, 72, 380], - }, - { - type: 'text', - content: ['20'], - w: 47, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [0, 144, 380], - }, - { - type: 'text', - content: ['30'], - w: 47, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [0, 216, 380], - }, - { - type: 'text', - content: ['40'], - w: 47, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [0, 288, 380], - }, - { - type: 'text', - content: ['50'], - w: 47, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [0, 360, 380], - }, - { - type: 'text', - content: ['0'], - w: 40, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [minSize ? 200 : 400, -40, 330], - }, - { - type: 'text', - content: ['90'], - w: 47, - h: 25, - color: 0xcccccc, - font: 'normal 500px Arial,sans-serif', - xyz: [minSize ? 200 : 400, -40, 230], - }, - { - type: 'text', - content: ['180'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: 0xcccccc, - size: 22.3, - font: 'normal 500px Arial,sans-serif', - xyz: [minSize ? 200 : 400, -40, 140], - }, - { - type: 'text', - content: ['270'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: 0xcccccc, - size: 22.3, - font: 'normal 500px Arial,sans-serif', - xyz: [minSize ? 200 : 400, -40, 50], - }, - { - type: 'text', - content: ['360'], - w: 75, - h: 25, - ch: 9, - cw: 5.7, - color: 0xcccccc, - size: 22.3, - font: 'normal 500px Arial,sans-serif', - xyz: [minSize ? 200 : 400, -40, -40], - }, - ] - let set = [7.2, 360 / (size + (-minSize)), 1]; - let isGIS = size <= 80; - let YAxisText = minSize ? [ - [size, 180], - [size / 2, 90], - [0, 0], - [minSize / 2, -90], - [minSize, -180], - ] : [ - [isGIS ? size == 80 ? 0 : `-${80 - size}` : size], - [isGIS ? `-${80 - (size * 0.75)}` : size * 0.75,], - [isGIS ? `-${80 - (size * 0.5)}` : size * 0.5,], - [isGIS ? `-${80 - (0.25 * size)}` : size * 0.25,], - [isGIS ? `-80` : 0,], - ]; - let infoNum = [ - { - w: 40, - h: 20, - size: 16, - }, { - w: 55, - h: 25, - size: 22.3, - }, { - w: 70, - h: 25, - size: 22.3, - }, { - w: 100, - h: 30, - size: 26.3, - }, { - w: 115, - h: 32, - size: 28.3, - } - ] - YAxisText.forEach((item, index) => { - item[0] = item[0].toString(); - let length = item[0].length; - let info = { - type: 'text', - content: item, - w: infoNum[length - 1].w, - h: infoNum[length - 1].h, - cw: length >= 3 ? 5.7 : '', - ch: length >= 3 ? 9 : '', - size: infoNum[length - 1].size, - color: '#00ff00', - font: 'normal 500px Arial,sans-serif', - xyz: [index, minSize ? item[1] : 360 - (index * 90), length === 1 ? 410 : 430], - textAlign: 'right', - }; - let infoPrpd = { - type: 'text', - content: item, - w: infoNum[length - 1].w, - h: infoNum[length - 1].h, - cw: length >= 3 ? 5.7 : '', - ch: length >= 3 ? 9 : '', - size: infoNum[length - 1].size, - color: '#00ff00', - font: 'normal 500px Arial,sans-serif', - xyz: [0, minSize ? item[1] : 360 - (index * 90), length === 1 ? 385 : 395], - textAlign: 'right', - }; - let info3d = { - type: 'text', - content: item, - w: infoNum[length - 1].w, - h: infoNum[length - 1].h, - cw: length >= 3 ? 5.7 : '', - ch: length >= 3 ? 9 : '', - size: infoNum[length - 1].size, - color: '#00ff00', - font: 'normal 500px Arial,sans-serif', - xyz: [minSize ? item[1] - 20 : 360 - (index * 90), -40, 380], - textAlign: 'right', - }; - prps.push(info); - prpd.push(infoPrpd); - prpd3d.push(info3d); - }) - data[size] = { set, prps, prpd, prpd3d } - return data; -} -export { createAxisInfo } \ No newline at end of file diff --git a/public/static/common/js/controller.js b/public/static/common/js/controller.js deleted file mode 100644 index 7612a21..0000000 --- a/public/static/common/js/controller.js +++ /dev/null @@ -1,27 +0,0 @@ -//全屏 -const fullScreen = () => { - var element = document.documentElement; - if (element.requestFullscreen) { - element.requestFullscreen(); - } else if (element.msRequestFullscreen) { - element.msRequestFullscreen(); - } else if (element.mozRequestFullScreen) { - element.mozRequestFullScreen(); - } else if (element.webkitRequestFullscreen) { - element.webkitRequestFullscreen(); - } -} - -//退出全屏 -const exitFullscreen = () => { - if (document.exitFullscreen) { - document.exitFullscreen(); - } else if (document.msExitFullscreen) { - document.msExitFullscreen(); - } else if (document.mozCancelFullScreen) { - document.mozCancelFullScreen(); - } else if (document.webkitExitFullscreen) { - document.webkitExitFullscreen(); - } -} -export { fullScreen, exitFullscreen } \ No newline at end of file diff --git a/public/static/common/js/draw.js b/public/static/common/js/draw.js deleted file mode 100644 index 13f7ee6..0000000 --- a/public/static/common/js/draw.js +++ /dev/null @@ -1,280 +0,0 @@ -import { scene, axis, meshList, pointsList, axis2D, scene2D } from './index.js'; -import { createAxisInfo } from './axisInfo/constant.js'; -const lut = new THREE.Lut(); -lut.addColorMap('axis', [[0.0, 0x00ee00], [0.25, 0xeeee00], [0.75, 0xee0000], [1.0, 0x4e0211]],); -lut.setColorMap('axis',1024) -let pointColors = [ - { - material: new THREE.PointsMaterial({ color: 0xD94E5D, size: 2, sizeAttenuation: false }), - colors: 0xD94E5D, - num: 15, - }, - { - material: new THREE.PointsMaterial({ color: 0xE28B4A, size: 2, sizeAttenuation: false }), - colors: 0xE28B4A, - num: 14, - }, - { - material: new THREE.PointsMaterial({ color: 0xEAC736, size: 2, sizeAttenuation: false }), - colors: 0xEAC736, - num: 10, - }, - { - material: new THREE.PointsMaterial({ color: 0x9DB578, size: 2, sizeAttenuation: false }), - colors: 0x9DB578, - num: 7, - }, - { - material: new THREE.PointsMaterial({ color: 0x50A3BA, size: 2, sizeAttenuation: false }), - colors: 0x50A3BA, - num: 5, - }, -]; -const initPoints = new THREE.Points( - new THREE.BufferGeometry().setAttribute('position', new THREE.Float32BufferAttribute([0.5, 0, 0], 3)), - new THREE.PointsMaterial({ color: 0x0082df, size: 2, sizeAttenuation: false }) -); -//柱子材质数据 -let meterials = [ - { - material: new THREE.MeshBasicMaterial({ - color: '#4e0211', - }), - max: 0.75 - }, - { - material: new THREE.MeshBasicMaterial({ - color: '#ee0000', - }), - max: 0.5 - }, - { - material: new THREE.MeshBasicMaterial({ - color: '#eeee00', - }), - max: 0.25 - }, - { - material: new THREE.MeshBasicMaterial({ - color: '#00ee00', - }), - max: 0, - }, -]; -const initGeometry = new THREE.BoxGeometry(0.6, 1, 2.5); -//绘制图表柱子 -const drawChartsContent = (px, pz, py = 50, max, pxNum) => { - let y = py < 0 ? -py : py; - const material = new THREE.MeshBasicMaterial({color:lut.getColor(Number(y)/max)});// - //const material = meterials.find(item => (y / max) >= item.max).material; - //柱子宽高长 - const geometry = initGeometry.clone(); - geometry.scale(1, y, 1); - geometry.translate(49.4 - pxNum, Math.fround(py / 2), 360 - pz); - return { geometry, material } -} -//正弦曲线绘制 -const drawSinLine = (obj) => { - let { line, color } = obj, list = []; - line.forEach(item => { - list.push(new THREE.Vector3(...item)); - }) - let geometry = new THREE.BufferGeometry(); //声明一个几何体对象Geometry - // 三维样条曲线 Catmull-Rom算法 - let curve = new THREE.CatmullRomCurve3(list); - //getPoints是基类Curve的方法,返回一个vector3对象作为元素组成的数组 - let points = curve.getPoints(200); //分段数100,返回101个顶点 - // setFromPoints方法从points中提取数据改变几何体的顶点属性vertices - geometry.setFromPoints(points); - //材质对象 - let material = new THREE.LineBasicMaterial({ - color, - linewidth: 1, - }); - //线条模型对象 - let modelLine = new THREE.Line(geometry, material); - return modelLine - //scene.add(line); //线条对象添加到场景中 -} -//绘制网格线条 -const drawLine = (obj) => { - let { line, color, position, margin, num } = obj - let geometry = new THREE.BufferGeometry(); //创建一个Buffer类型几何体对象 - //3个为一组,表示一个顶点的xyz坐标 - geometry.setAttribute('position', new THREE.Float32BufferAttribute(line, 3)) - // 线条渲染模式 - let material = new THREE.LineBasicMaterial({ - color, //线条颜色 - });//材质对象 - let LineList = []; - let modelLine = new THREE.Line(geometry, material);//线条模型对象 - LineList.push(modelLine) - for (let i = 1; i < num; i++) { - let lineChildren = modelLine.clone()['translate' + position](i * margin) - LineList.push(lineChildren); - } - return LineList -} -// 添加文字geometry -const add3DText = (text, callback) => { - let { content, url, size, set, rotation, xyz } = text; - let loader = new THREE.FontLoader(); - //加载相应的字体,下面字体名和样式组成gentilis_bold.typeface.json 完整文件名 - let fontJsonUrl = url ? url : 'common/fonts/Century Gothic_Regular.json'; //你的字体json路径,文件名规格随意 - loader.load(fontJsonUrl, function (font) { - const color = 0xffffff; - const matLite = new THREE.MeshBasicMaterial({ - color: color, // 字体大小 - }); - const geometry = new THREE.TextGeometry(content, { - font: font, - size: size ? size : 5, - height: 1, - }); - geometry.translate(...xyz); - if (rotation) { - for (let i of rotation) { - geometry['rotate' + i[0]](Math.PI / i[1]) - } - - } - let mesh = new THREE.Mesh(geometry, matLite); - // mesh.scale.set(2, 2, 2) - if (set) { - mesh.scale.set(...set); - } - mesh.castShadow = true; - mesh.receiveShadow = true; - mesh.name = 'text'; - callback(mesh) - }); -} -// 添加精灵图文字 -const draw2DText = (obj) => { - let scale = 1;//window.devicePixelRatio; - let { content, w, h, color, font, xyz, size, cw, ch, rotate } = obj; - cw = cw ? cw : 6; - ch = ch ? ch : 8; - let canvas = document.createElement('canvas') - canvas.style.width = w * scale; - canvas.style.height = h * scale; - if (rotate) { - canvas.width = w * cw * scale * 3; - canvas.height = h * ch * scale * 2; - } else { - canvas.width = w * cw * scale * 2; - canvas.height = h * ch * scale * 3; - } - - let ctx = canvas.getContext('2d'); - ctx.fillStyle = color; - ctx.strokeStyle = color; - ctx.font = font; - ctx.direction = "ltr"; // 文本方向从左向右 - ctx.textAlign = "left"; // 左对齐 - ctx.textBaseline = 'middle'//基线对齐选项,决定文字垂直方向的对齐方式 - if (rotate) { - ctx.translate(canvas.width / 2, canvas.height); - ctx.rotate(-Math.PI / 2); - ctx.fillText(content, h, 0); - //ctx.strokeText(content, h, 0); - } else { - for (let i = 0; i < 1; i++) { - ctx.fillText(content[i], 0, canvas.height / 2); - ctx.strokeText(content[i], 0, canvas.height / 2); - } - } - let texture = new THREE.CanvasTexture(canvas) - // texture.needsUpdate = true//如果编码类型在纹理已被一个材质使用之后发生了改变, 你需要来设置Material.needsUpdate为true来使得材质重新编译 - let geometry = new THREE.BufferGeometry(); //创建一个Buffer类型几何体对象 - //3个为一组,表示一个顶点的xyz坐标 - geometry.setAttribute('position', new THREE.Float32BufferAttribute(xyz, 3)); - const material = new THREE.PointsMaterial({ - map: texture, - transparent: true,//材质透明 - size: size ? size : 16.0, //点对象像素尺寸 - }); - - const pointsText = new THREE.Points(geometry, material); - pointsText.name = 'pointsText'; - return pointsText; -} - -const drawAxis = (size, minNum, is2DInit) => { - scene.remove(...scene.children.filter(item => item.name === 'axisNumberText')) - if (axis.children.length) { - axis.remove(...axis.children) - } - let axisText = new THREE.Group(); - axisText.name = 'axisNumberText'; - scene.add(axisText) - const data = createAxisInfo(size, minNum); - axis.scale.set(...data[size]['set']) - meshList.scale.set(...data[size]['set']) - pointsList.scale.set(...data[size]['set']) - data[size]['prps'].forEach(async item => { - switch (item.type) { - case 'plan': - axis.add(drawPlan(item)); - break; - case 'line': - axis.add(...drawLine(item)); - break; - case 'curveLine': - axis.add(drawSinLine(item)); - break; - case 'text': - axisText.add(draw2DText(item)); - } - }); - if (is2DInit) { - return; - } - drawAxis2D(size, minNum); -} -const drawAxis2D = (size, minNum) => { - if (axis2D.children.length) { - axis2D.remove(...axis2D.children) - } - let axisList = scene2D?.children.filter(item => item.name === 'axisNumberText'); - axisList && scene2D.remove(...axisList) - let axisText = new THREE.Group(); - const data = createAxisInfo(size, minNum); - axisText.name = 'axisNumberText'; - scene2D.add(axisText) - axis2D.scale.set(...data[size]['set']) - let list = scene2D.getObjectByName('chartsPoint') - if (list) { - list.scale.set(...data[size]['set']) - } - data[size]['prpd'].forEach(async item => { - switch (item.type) { - case 'line': - axis2D.add(...drawLine(item)); - break; - case 'curveLine': - axis2D.add(drawSinLine(item)); - break; - case 'text': - axisText.add(draw2DText(item)); - } - }); -} -//绘制平面 -const drawPlan = (obj) => { - const { size, color } = obj; - let geometry = new THREE.PlaneGeometry(size, 360); //创建一个Buffer类型几何体对象 - geometry.rotateX(Math.PI / 2).translate(size/2, 0, 180); - // 三角面(网格)渲染模式 - let material = new THREE.MeshBasicMaterial({ - color, //三角面颜色 - transparent: true,//透明 - opacity: 0.2, - side: THREE.DoubleSide, //两面可见 - }); //材质对象 - let mesh = new THREE.Mesh(geometry, material); //网格模型对象Mesh - mesh.name = 'plan'; - //scene.add(mesh); - return mesh -} -export { drawChartsContent, drawSinLine, drawLine, add3DText, draw2DText, drawAxis, drawAxis2D, initPoints, pointColors, drawPlan } \ No newline at end of file diff --git a/public/static/common/js/drawHistory.js b/public/static/common/js/drawHistory.js deleted file mode 100644 index f5a7b85..0000000 --- a/public/static/common/js/drawHistory.js +++ /dev/null @@ -1,280 +0,0 @@ -import { scene, axis, meshList, pointsList, axis2D, scene2D } from './history.js'; -import { createAxisInfo } from './axisInfo/constant.js'; -const lut = new THREE.Lut(); -lut.addColorMap('axis', [[0.0, 0x00ee00], [0.25, 0xeeee00], [0.75, 0xee0000], [1.0, 0x4e0211]],); -lut.setColorMap('axis',1024) -let pointColors = [ - { - material: new THREE.PointsMaterial({ color: 0xD94E5D, size: 2, sizeAttenuation: false }), - colors: 0xD94E5D, - num: 15, - }, - { - material: new THREE.PointsMaterial({ color: 0xE28B4A, size: 2, sizeAttenuation: false }), - colors: 0xE28B4A, - num: 14, - }, - { - material: new THREE.PointsMaterial({ color: 0xEAC736, size: 2, sizeAttenuation: false }), - colors: 0xEAC736, - num: 10, - }, - { - material: new THREE.PointsMaterial({ color: 0x9DB578, size: 2, sizeAttenuation: false }), - colors: 0x9DB578, - num: 7, - }, - { - material: new THREE.PointsMaterial({ color: 0x50A3BA, size: 2, sizeAttenuation: false }), - colors: 0x50A3BA, - num: 5, - }, -]; -const initPoints = new THREE.Points( - new THREE.BufferGeometry().setAttribute('position', new THREE.Float32BufferAttribute([0.5, 0, 0], 3)), - new THREE.PointsMaterial({ color: 0x0082df, size: 2, sizeAttenuation: false }) -); -//柱子材质数据 -let meterials = [ - { - material: new THREE.MeshBasicMaterial({ - color: '#4e0211', - }), - max: 0.75 - }, - { - material: new THREE.MeshBasicMaterial({ - color: '#ee0000', - }), - max: 0.5 - }, - { - material: new THREE.MeshBasicMaterial({ - color: '#eeee00', - }), - max: 0.25 - }, - { - material: new THREE.MeshBasicMaterial({ - color: '#00ee00', - }), - max: 0, - }, -]; -const initGeometry = new THREE.BoxGeometry(0.6, 1, 2.5); -//绘制图表柱子 -const drawChartsContent = (px, pz, py = 50, max, pxNum) => { - let y = py < 0 ? -py : py; - const material = new THREE.MeshBasicMaterial({color:lut.getColor(Number(y)/max)});// - //const material = meterials.find(item => (y / max) >= item.max).material; - //柱子宽高长 - const geometry = initGeometry.clone(); - geometry.scale(1, y, 1); - geometry.translate(49.4 - pxNum, Math.fround(py / 2), 360 - pz); - return { geometry, material } -} -//正弦曲线绘制 -const drawSinLine = (obj) => { - let { line, color } = obj, list = []; - line.forEach(item => { - list.push(new THREE.Vector3(...item)); - }) - let geometry = new THREE.BufferGeometry(); //声明一个几何体对象Geometry - // 三维样条曲线 Catmull-Rom算法 - let curve = new THREE.CatmullRomCurve3(list); - //getPoints是基类Curve的方法,返回一个vector3对象作为元素组成的数组 - let points = curve.getPoints(200); //分段数100,返回101个顶点 - // setFromPoints方法从points中提取数据改变几何体的顶点属性vertices - geometry.setFromPoints(points); - //材质对象 - let material = new THREE.LineBasicMaterial({ - color, - linewidth: 1, - }); - //线条模型对象 - let modelLine = new THREE.Line(geometry, material); - return modelLine - //scene.add(line); //线条对象添加到场景中 -} -//绘制网格线条 -const drawLine = (obj) => { - let { line, color, position, margin, num } = obj - let geometry = new THREE.BufferGeometry(); //创建一个Buffer类型几何体对象 - //3个为一组,表示一个顶点的xyz坐标 - geometry.setAttribute('position', new THREE.Float32BufferAttribute(line, 3)) - // 线条渲染模式 - let material = new THREE.LineBasicMaterial({ - color, //线条颜色 - });//材质对象 - let LineList = []; - let modelLine = new THREE.Line(geometry, material);//线条模型对象 - LineList.push(modelLine) - for (let i = 1; i < num; i++) { - let lineChildren = modelLine.clone()['translate' + position](i * margin) - LineList.push(lineChildren); - } - return LineList -} -// 添加文字geometry -const add3DText = (text, callback) => { - let { content, url, size, set, rotation, xyz } = text; - let loader = new THREE.FontLoader(); - //加载相应的字体,下面字体名和样式组成gentilis_bold.typeface.json 完整文件名 - let fontJsonUrl = url ? url : 'common/fonts/Century Gothic_Regular.json'; //你的字体json路径,文件名规格随意 - loader.load(fontJsonUrl, function (font) { - const color = 0xffffff; - const matLite = new THREE.MeshBasicMaterial({ - color: color, // 字体大小 - }); - const geometry = new THREE.TextGeometry(content, { - font: font, - size: size ? size : 5, - height: 1, - }); - geometry.translate(...xyz); - if (rotation) { - for (let i of rotation) { - geometry['rotate' + i[0]](Math.PI / i[1]) - } - - } - let mesh = new THREE.Mesh(geometry, matLite); - // mesh.scale.set(2, 2, 2) - if (set) { - mesh.scale.set(...set); - } - mesh.castShadow = true; - mesh.receiveShadow = true; - mesh.name = 'text'; - callback(mesh) - }); -} -// 添加精灵图文字 -const draw2DText = (obj) => { - let scale = 1;//window.devicePixelRatio; - let { content, w, h, color, font, xyz, size, cw, ch, rotate } = obj; - cw = cw ? cw : 6; - ch = ch ? ch : 8; - let canvas = document.createElement('canvas') - canvas.style.width = w * scale; - canvas.style.height = h * scale; - if (rotate) { - canvas.width = w * cw * scale * 3; - canvas.height = h * ch * scale * 2; - } else { - canvas.width = w * cw * scale * 2; - canvas.height = h * ch * scale * 3; - } - - let ctx = canvas.getContext('2d'); - ctx.fillStyle = color; - ctx.strokeStyle = color; - ctx.font = font; - ctx.direction = "ltr"; // 文本方向从左向右 - ctx.textAlign = "left"; // 左对齐 - ctx.textBaseline = 'middle'//基线对齐选项,决定文字垂直方向的对齐方式 - if (rotate) { - ctx.translate(canvas.width / 2, canvas.height); - ctx.rotate(-Math.PI / 2); - ctx.fillText(content, h, 0); - //ctx.strokeText(content, h, 0); - } else { - for (let i = 0; i < 1; i++) { - ctx.fillText(content[i], 0, canvas.height / 2); - ctx.strokeText(content[i], 0, canvas.height / 2); - } - } - let texture = new THREE.CanvasTexture(canvas) - // texture.needsUpdate = true//如果编码类型在纹理已被一个材质使用之后发生了改变, 你需要来设置Material.needsUpdate为true来使得材质重新编译 - let geometry = new THREE.BufferGeometry(); //创建一个Buffer类型几何体对象 - //3个为一组,表示一个顶点的xyz坐标 - geometry.setAttribute('position', new THREE.Float32BufferAttribute(xyz, 3)); - const material = new THREE.PointsMaterial({ - map: texture, - transparent: true,//材质透明 - size: size ? size : 16.0, //点对象像素尺寸 - }); - - const pointsText = new THREE.Points(geometry, material); - pointsText.name = 'pointsText'; - return pointsText; -} - -const drawAxis = (size, minNum, is2DInit) => { - scene.remove(...scene.children.filter(item => item.name === 'axisNumberText')) - if (axis.children.length) { - axis.remove(...axis.children) - } - let axisText = new THREE.Group(); - axisText.name = 'axisNumberText'; - scene.add(axisText) - const data = createAxisInfo(size, minNum); - axis.scale.set(...data[size]['set']) - meshList.scale.set(...data[size]['set']) - pointsList.scale.set(...data[size]['set']) - data[size]['prps'].forEach(async item => { - switch (item.type) { - case 'plan': - axis.add(drawPlan(item)); - break; - case 'line': - axis.add(...drawLine(item)); - break; - case 'curveLine': - axis.add(drawSinLine(item)); - break; - case 'text': - axisText.add(draw2DText(item)); - } - }); - if (is2DInit) { - return; - } - drawAxis2D(size, minNum); -} -const drawAxis2D = (size, minNum) => { - if (axis2D.children.length) { - axis2D.remove(...axis2D.children) - } - let axisList = scene2D?.children.filter(item => item.name === 'axisNumberText'); - axisList && scene2D.remove(...axisList) - let axisText = new THREE.Group(); - const data = createAxisInfo(size, minNum); - axisText.name = 'axisNumberText'; - scene2D.add(axisText) - axis2D.scale.set(...data[size]['set']) - let list = scene2D.getObjectByName('chartsPoint') - if (list) { - list.scale.set(...data[size]['set']) - } - data[size]['prpd'].forEach(async item => { - switch (item.type) { - case 'line': - axis2D.add(...drawLine(item)); - break; - case 'curveLine': - axis2D.add(drawSinLine(item)); - break; - case 'text': - axisText.add(draw2DText(item)); - } - }); -} -//绘制平面 -const drawPlan = (obj) => { - const { size, color } = obj; - let geometry = new THREE.PlaneGeometry(size, 360); //创建一个Buffer类型几何体对象 - geometry.rotateX(Math.PI / 2).translate(size/2, 0, 180); - // 三角面(网格)渲染模式 - let material = new THREE.MeshBasicMaterial({ - color, //三角面颜色 - transparent: true,//透明 - opacity: 0.2, - side: THREE.DoubleSide, //两面可见 - }); //材质对象 - let mesh = new THREE.Mesh(geometry, material); //网格模型对象Mesh - mesh.name = 'plan'; - //scene.add(mesh); - return mesh -} -export { drawChartsContent, drawSinLine, drawLine, add3DText, draw2DText, drawAxis, drawAxis2D, initPoints, pointColors, drawPlan } \ No newline at end of file diff --git a/public/static/common/js/echarts/echarts.js b/public/static/common/js/echarts/echarts.js deleted file mode 100644 index 297dd59..0000000 --- a/public/static/common/js/echarts/echarts.js +++ /dev/null @@ -1,95739 +0,0 @@ - -/* -* Licensed to the Apache Software Foundation (ASF) under one -* or more contributor license agreements. See the NOTICE file -* distributed with this work for additional information -* regarding copyright ownership. The ASF licenses this file -* to you under the Apache License, Version 2.0 (the -* "License"); you may not use this file except in compliance -* with the License. You may obtain a copy of the License at -* -* http://www.apache.org/licenses/LICENSE-2.0 -* -* Unless required by applicable law or agreed to in writing, -* software distributed under the License is distributed on an -* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY -* KIND, either express or implied. See the License for the -* specific language governing permissions and limitations -* under the License. -*/ - -(function (global, factory) { - typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) : - typeof define === 'function' && define.amd ? define(['exports'], factory) : - (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.echarts = {})); -}(this, (function (exports) { 'use strict'; - - /*! ***************************************************************************** - Copyright (c) Microsoft Corporation. - - Permission to use, copy, modify, and/or distribute this software for any - purpose with or without fee is hereby granted. - - THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH - REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY - AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, - INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM - LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR - OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR - PERFORMANCE OF THIS SOFTWARE. - ***************************************************************************** */ - /* global Reflect, Promise */ - - var extendStatics = function(d, b) { - extendStatics = Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || - function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; }; - return extendStatics(d, b); - }; - - function __extends(d, b) { - if (typeof b !== "function" && b !== null) - throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); - extendStatics(d, b); - function __() { this.constructor = d; } - d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); - } - - var Browser = (function () { - function Browser() { - this.firefox = false; - this.ie = false; - this.edge = false; - this.newEdge = false; - this.weChat = false; - } - return Browser; - }()); - var Env = (function () { - function Env() { - this.browser = new Browser(); - this.node = false; - this.wxa = false; - this.worker = false; - this.svgSupported = false; - this.touchEventsSupported = false; - this.pointerEventsSupported = false; - this.domSupported = false; - this.transformSupported = false; - this.transform3dSupported = false; - this.hasGlobalWindow = typeof window !== 'undefined'; - } - return Env; - }()); - var env = new Env(); - if (typeof wx === 'object' && typeof wx.getSystemInfoSync === 'function') { - env.wxa = true; - env.touchEventsSupported = true; - } - else if (typeof document === 'undefined' && typeof self !== 'undefined') { - env.worker = true; - } - else if (typeof navigator === 'undefined') { - env.node = true; - env.svgSupported = true; - } - else { - detect(navigator.userAgent, env); - } - function detect(ua, env) { - var browser = env.browser; - var firefox = ua.match(/Firefox\/([\d.]+)/); - var ie = ua.match(/MSIE\s([\d.]+)/) - || ua.match(/Trident\/.+?rv:(([\d.]+))/); - var edge = ua.match(/Edge?\/([\d.]+)/); - var weChat = (/micromessenger/i).test(ua); - if (firefox) { - browser.firefox = true; - browser.version = firefox[1]; - } - if (ie) { - browser.ie = true; - browser.version = ie[1]; - } - if (edge) { - browser.edge = true; - browser.version = edge[1]; - browser.newEdge = +edge[1].split('.')[0] > 18; - } - if (weChat) { - browser.weChat = true; - } - env.svgSupported = typeof SVGRect !== 'undefined'; - env.touchEventsSupported = 'ontouchstart' in window && !browser.ie && !browser.edge; - env.pointerEventsSupported = 'onpointerdown' in window - && (browser.edge || (browser.ie && +browser.version >= 11)); - env.domSupported = typeof document !== 'undefined'; - var style = document.documentElement.style; - env.transform3dSupported = ((browser.ie && 'transition' in style) - || browser.edge - || (('WebKitCSSMatrix' in window) && ('m11' in new WebKitCSSMatrix())) - || 'MozPerspective' in style) - && !('OTransition' in style); - env.transformSupported = env.transform3dSupported - || (browser.ie && +browser.version >= 9); - } - - var DEFAULT_FONT_SIZE = 12; - var DEFAULT_FONT_FAMILY = 'sans-serif'; - var DEFAULT_FONT = DEFAULT_FONT_SIZE + "px " + DEFAULT_FONT_FAMILY; - var OFFSET = 20; - var SCALE = 100; - var defaultWidthMapStr = "007LLmW'55;N0500LLLLLLLLLL00NNNLzWW\\\\WQb\\0FWLg\\bWb\\WQ\\WrWWQ000CL5LLFLL0LL**F*gLLLL5F0LF\\FFF5.5N"; - function getTextWidthMap(mapStr) { - var map = {}; - if (typeof JSON === 'undefined') { - return map; - } - for (var i = 0; i < mapStr.length; i++) { - var char = String.fromCharCode(i + 32); - var size = (mapStr.charCodeAt(i) - OFFSET) / SCALE; - map[char] = size; - } - return map; - } - var DEFAULT_TEXT_WIDTH_MAP = getTextWidthMap(defaultWidthMapStr); - var platformApi = { - createCanvas: function () { - return typeof document !== 'undefined' - && document.createElement('canvas'); - }, - measureText: (function () { - var _ctx; - var _cachedFont; - return function (text, font) { - if (!_ctx) { - var canvas = platformApi.createCanvas(); - _ctx = canvas && canvas.getContext('2d'); - } - if (_ctx) { - if (_cachedFont !== font) { - _cachedFont = _ctx.font = font || DEFAULT_FONT; - } - return _ctx.measureText(text); - } - else { - text = text || ''; - font = font || DEFAULT_FONT; - var res = /(\d+)px/.exec(font); - var fontSize = res && +res[1] || DEFAULT_FONT_SIZE; - var width = 0; - if (font.indexOf('mono') >= 0) { - width = fontSize * text.length; - } - else { - for (var i = 0; i < text.length; i++) { - var preCalcWidth = DEFAULT_TEXT_WIDTH_MAP[text[i]]; - width += preCalcWidth == null ? fontSize : (preCalcWidth * fontSize); - } - } - return { width: width }; - } - }; - })(), - loadImage: function (src, onload, onerror) { - var image = new Image(); - image.onload = onload; - image.onerror = onerror; - image.src = src; - return image; - } - }; - function setPlatformAPI(newPlatformApis) { - for (var key in platformApi) { - if (newPlatformApis[key]) { - platformApi[key] = newPlatformApis[key]; - } - } - } - - var BUILTIN_OBJECT = reduce([ - 'Function', - 'RegExp', - 'Date', - 'Error', - 'CanvasGradient', - 'CanvasPattern', - 'Image', - 'Canvas' - ], function (obj, val) { - obj['[object ' + val + ']'] = true; - return obj; - }, {}); - var TYPED_ARRAY = reduce([ - 'Int8', - 'Uint8', - 'Uint8Clamped', - 'Int16', - 'Uint16', - 'Int32', - 'Uint32', - 'Float32', - 'Float64' - ], function (obj, val) { - obj['[object ' + val + 'Array]'] = true; - return obj; - }, {}); - var objToString = Object.prototype.toString; - var arrayProto = Array.prototype; - var nativeForEach = arrayProto.forEach; - var nativeFilter = arrayProto.filter; - var nativeSlice = arrayProto.slice; - var nativeMap = arrayProto.map; - var ctorFunction = function () { }.constructor; - var protoFunction = ctorFunction ? ctorFunction.prototype : null; - var protoKey = '__proto__'; - var idStart = 0x0907; - function guid() { - return idStart++; - } - function logError() { - var args = []; - for (var _i = 0; _i < arguments.length; _i++) { - args[_i] = arguments[_i]; - } - if (typeof console !== 'undefined') { - console.error.apply(console, args); - } - } - function clone(source) { - if (source == null || typeof source !== 'object') { - return source; - } - var result = source; - var typeStr = objToString.call(source); - if (typeStr === '[object Array]') { - if (!isPrimitive(source)) { - result = []; - for (var i = 0, len = source.length; i < len; i++) { - result[i] = clone(source[i]); - } - } - } - else if (TYPED_ARRAY[typeStr]) { - if (!isPrimitive(source)) { - var Ctor = source.constructor; - if (Ctor.from) { - result = Ctor.from(source); - } - else { - result = new Ctor(source.length); - for (var i = 0, len = source.length; i < len; i++) { - result[i] = source[i]; - } - } - } - } - else if (!BUILTIN_OBJECT[typeStr] && !isPrimitive(source) && !isDom(source)) { - result = {}; - for (var key in source) { - if (source.hasOwnProperty(key) && key !== protoKey) { - result[key] = clone(source[key]); - } - } - } - return result; - } - function merge(target, source, overwrite) { - if (!isObject(source) || !isObject(target)) { - return overwrite ? clone(source) : target; - } - for (var key in source) { - if (source.hasOwnProperty(key) && key !== protoKey) { - var targetProp = target[key]; - var sourceProp = source[key]; - if (isObject(sourceProp) - && isObject(targetProp) - && !isArray(sourceProp) - && !isArray(targetProp) - && !isDom(sourceProp) - && !isDom(targetProp) - && !isBuiltInObject(sourceProp) - && !isBuiltInObject(targetProp) - && !isPrimitive(sourceProp) - && !isPrimitive(targetProp)) { - merge(targetProp, sourceProp, overwrite); - } - else if (overwrite || !(key in target)) { - target[key] = clone(source[key]); - } - } - } - return target; - } - function mergeAll(targetAndSources, overwrite) { - var result = targetAndSources[0]; - for (var i = 1, len = targetAndSources.length; i < len; i++) { - result = merge(result, targetAndSources[i], overwrite); - } - return result; - } - function extend(target, source) { - if (Object.assign) { - Object.assign(target, source); - } - else { - for (var key in source) { - if (source.hasOwnProperty(key) && key !== protoKey) { - target[key] = source[key]; - } - } - } - return target; - } - function defaults(target, source, overlay) { - var keysArr = keys(source); - for (var i = 0; i < keysArr.length; i++) { - var key = keysArr[i]; - if ((overlay ? source[key] != null : target[key] == null)) { - target[key] = source[key]; - } - } - return target; - } - var createCanvas = platformApi.createCanvas; - function indexOf(array, value) { - if (array) { - if (array.indexOf) { - return array.indexOf(value); - } - for (var i = 0, len = array.length; i < len; i++) { - if (array[i] === value) { - return i; - } - } - } - return -1; - } - function inherits(clazz, baseClazz) { - var clazzPrototype = clazz.prototype; - function F() { } - F.prototype = baseClazz.prototype; - clazz.prototype = new F(); - for (var prop in clazzPrototype) { - if (clazzPrototype.hasOwnProperty(prop)) { - clazz.prototype[prop] = clazzPrototype[prop]; - } - } - clazz.prototype.constructor = clazz; - clazz.superClass = baseClazz; - } - function mixin(target, source, override) { - target = 'prototype' in target ? target.prototype : target; - source = 'prototype' in source ? source.prototype : source; - if (Object.getOwnPropertyNames) { - var keyList = Object.getOwnPropertyNames(source); - for (var i = 0; i < keyList.length; i++) { - var key = keyList[i]; - if (key !== 'constructor') { - if ((override ? source[key] != null : target[key] == null)) { - target[key] = source[key]; - } - } - } - } - else { - defaults(target, source, override); - } - } - function isArrayLike(data) { - if (!data) { - return false; - } - if (typeof data === 'string') { - return false; - } - return typeof data.length === 'number'; - } - function each(arr, cb, context) { - if (!(arr && cb)) { - return; - } - if (arr.forEach && arr.forEach === nativeForEach) { - arr.forEach(cb, context); - } - else if (arr.length === +arr.length) { - for (var i = 0, len = arr.length; i < len; i++) { - cb.call(context, arr[i], i, arr); - } - } - else { - for (var key in arr) { - if (arr.hasOwnProperty(key)) { - cb.call(context, arr[key], key, arr); - } - } - } - } - function map(arr, cb, context) { - if (!arr) { - return []; - } - if (!cb) { - return slice(arr); - } - if (arr.map && arr.map === nativeMap) { - return arr.map(cb, context); - } - else { - var result = []; - for (var i = 0, len = arr.length; i < len; i++) { - result.push(cb.call(context, arr[i], i, arr)); - } - return result; - } - } - function reduce(arr, cb, memo, context) { - if (!(arr && cb)) { - return; - } - for (var i = 0, len = arr.length; i < len; i++) { - memo = cb.call(context, memo, arr[i], i, arr); - } - return memo; - } - function filter(arr, cb, context) { - if (!arr) { - return []; - } - if (!cb) { - return slice(arr); - } - if (arr.filter && arr.filter === nativeFilter) { - return arr.filter(cb, context); - } - else { - var result = []; - for (var i = 0, len = arr.length; i < len; i++) { - if (cb.call(context, arr[i], i, arr)) { - result.push(arr[i]); - } - } - return result; - } - } - function find(arr, cb, context) { - if (!(arr && cb)) { - return; - } - for (var i = 0, len = arr.length; i < len; i++) { - if (cb.call(context, arr[i], i, arr)) { - return arr[i]; - } - } - } - function keys(obj) { - if (!obj) { - return []; - } - if (Object.keys) { - return Object.keys(obj); - } - var keyList = []; - for (var key in obj) { - if (obj.hasOwnProperty(key)) { - keyList.push(key); - } - } - return keyList; - } - function bindPolyfill(func, context) { - var args = []; - for (var _i = 2; _i < arguments.length; _i++) { - args[_i - 2] = arguments[_i]; - } - return function () { - return func.apply(context, args.concat(nativeSlice.call(arguments))); - }; - } - var bind = (protoFunction && isFunction(protoFunction.bind)) - ? protoFunction.call.bind(protoFunction.bind) - : bindPolyfill; - function curry(func) { - var args = []; - for (var _i = 1; _i < arguments.length; _i++) { - args[_i - 1] = arguments[_i]; - } - return function () { - return func.apply(this, args.concat(nativeSlice.call(arguments))); - }; - } - function isArray(value) { - if (Array.isArray) { - return Array.isArray(value); - } - return objToString.call(value) === '[object Array]'; - } - function isFunction(value) { - return typeof value === 'function'; - } - function isString(value) { - return typeof value === 'string'; - } - function isStringSafe(value) { - return objToString.call(value) === '[object String]'; - } - function isNumber(value) { - return typeof value === 'number'; - } - function isObject(value) { - var type = typeof value; - return type === 'function' || (!!value && type === 'object'); - } - function isBuiltInObject(value) { - return !!BUILTIN_OBJECT[objToString.call(value)]; - } - function isTypedArray(value) { - return !!TYPED_ARRAY[objToString.call(value)]; - } - function isDom(value) { - return typeof value === 'object' - && typeof value.nodeType === 'number' - && typeof value.ownerDocument === 'object'; - } - function isGradientObject(value) { - return value.colorStops != null; - } - function isImagePatternObject(value) { - return value.image != null; - } - function isRegExp(value) { - return objToString.call(value) === '[object RegExp]'; - } - function eqNaN(value) { - return value !== value; - } - function retrieve() { - var args = []; - for (var _i = 0; _i < arguments.length; _i++) { - args[_i] = arguments[_i]; - } - for (var i = 0, len = args.length; i < len; i++) { - if (args[i] != null) { - return args[i]; - } - } - } - function retrieve2(value0, value1) { - return value0 != null - ? value0 - : value1; - } - function retrieve3(value0, value1, value2) { - return value0 != null - ? value0 - : value1 != null - ? value1 - : value2; - } - function slice(arr) { - var args = []; - for (var _i = 1; _i < arguments.length; _i++) { - args[_i - 1] = arguments[_i]; - } - return nativeSlice.apply(arr, args); - } - function normalizeCssArray(val) { - if (typeof (val) === 'number') { - return [val, val, val, val]; - } - var len = val.length; - if (len === 2) { - return [val[0], val[1], val[0], val[1]]; - } - else if (len === 3) { - return [val[0], val[1], val[2], val[1]]; - } - return val; - } - function assert(condition, message) { - if (!condition) { - throw new Error(message); - } - } - function trim(str) { - if (str == null) { - return null; - } - else if (typeof str.trim === 'function') { - return str.trim(); - } - else { - return str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); - } - } - var primitiveKey = '__ec_primitive__'; - function setAsPrimitive(obj) { - obj[primitiveKey] = true; - } - function isPrimitive(obj) { - return obj[primitiveKey]; - } - var MapPolyfill = (function () { - function MapPolyfill() { - this.data = {}; - } - MapPolyfill.prototype["delete"] = function (key) { - var existed = this.has(key); - if (existed) { - delete this.data[key]; - } - return existed; - }; - MapPolyfill.prototype.has = function (key) { - return this.data.hasOwnProperty(key); - }; - MapPolyfill.prototype.get = function (key) { - return this.data[key]; - }; - MapPolyfill.prototype.set = function (key, value) { - this.data[key] = value; - return this; - }; - MapPolyfill.prototype.keys = function () { - return keys(this.data); - }; - MapPolyfill.prototype.forEach = function (callback) { - var data = this.data; - for (var key in data) { - if (data.hasOwnProperty(key)) { - callback(data[key], key); - } - } - }; - return MapPolyfill; - }()); - var isNativeMapSupported = typeof Map === 'function'; - function maybeNativeMap() { - return (isNativeMapSupported ? new Map() : new MapPolyfill()); - } - var HashMap = (function () { - function HashMap(obj) { - var isArr = isArray(obj); - this.data = maybeNativeMap(); - var thisMap = this; - (obj instanceof HashMap) - ? obj.each(visit) - : (obj && each(obj, visit)); - function visit(value, key) { - isArr ? thisMap.set(value, key) : thisMap.set(key, value); - } - } - HashMap.prototype.hasKey = function (key) { - return this.data.has(key); - }; - HashMap.prototype.get = function (key) { - return this.data.get(key); - }; - HashMap.prototype.set = function (key, value) { - this.data.set(key, value); - return value; - }; - HashMap.prototype.each = function (cb, context) { - this.data.forEach(function (value, key) { - cb.call(context, value, key); - }); - }; - HashMap.prototype.keys = function () { - var keys = this.data.keys(); - return isNativeMapSupported - ? Array.from(keys) - : keys; - }; - HashMap.prototype.removeKey = function (key) { - this.data["delete"](key); - }; - return HashMap; - }()); - function createHashMap(obj) { - return new HashMap(obj); - } - function concatArray(a, b) { - var newArray = new a.constructor(a.length + b.length); - for (var i = 0; i < a.length; i++) { - newArray[i] = a[i]; - } - var offset = a.length; - for (var i = 0; i < b.length; i++) { - newArray[i + offset] = b[i]; - } - return newArray; - } - function createObject(proto, properties) { - var obj; - if (Object.create) { - obj = Object.create(proto); - } - else { - var StyleCtor = function () { }; - StyleCtor.prototype = proto; - obj = new StyleCtor(); - } - if (properties) { - extend(obj, properties); - } - return obj; - } - function disableUserSelect(dom) { - var domStyle = dom.style; - domStyle.webkitUserSelect = 'none'; - domStyle.userSelect = 'none'; - domStyle.webkitTapHighlightColor = 'rgba(0,0,0,0)'; - domStyle['-webkit-touch-callout'] = 'none'; - } - function hasOwn(own, prop) { - return own.hasOwnProperty(prop); - } - function noop() { } - var RADIAN_TO_DEGREE = 180 / Math.PI; - - var util = /*#__PURE__*/Object.freeze({ - __proto__: null, - guid: guid, - logError: logError, - clone: clone, - merge: merge, - mergeAll: mergeAll, - extend: extend, - defaults: defaults, - createCanvas: createCanvas, - indexOf: indexOf, - inherits: inherits, - mixin: mixin, - isArrayLike: isArrayLike, - each: each, - map: map, - reduce: reduce, - filter: filter, - find: find, - keys: keys, - bind: bind, - curry: curry, - isArray: isArray, - isFunction: isFunction, - isString: isString, - isStringSafe: isStringSafe, - isNumber: isNumber, - isObject: isObject, - isBuiltInObject: isBuiltInObject, - isTypedArray: isTypedArray, - isDom: isDom, - isGradientObject: isGradientObject, - isImagePatternObject: isImagePatternObject, - isRegExp: isRegExp, - eqNaN: eqNaN, - retrieve: retrieve, - retrieve2: retrieve2, - retrieve3: retrieve3, - slice: slice, - normalizeCssArray: normalizeCssArray, - assert: assert, - trim: trim, - setAsPrimitive: setAsPrimitive, - isPrimitive: isPrimitive, - HashMap: HashMap, - createHashMap: createHashMap, - concatArray: concatArray, - createObject: createObject, - disableUserSelect: disableUserSelect, - hasOwn: hasOwn, - noop: noop, - RADIAN_TO_DEGREE: RADIAN_TO_DEGREE - }); - - function create(x, y) { - if (x == null) { - x = 0; - } - if (y == null) { - y = 0; - } - return [x, y]; - } - function copy(out, v) { - out[0] = v[0]; - out[1] = v[1]; - return out; - } - function clone$1(v) { - return [v[0], v[1]]; - } - function set(out, a, b) { - out[0] = a; - out[1] = b; - return out; - } - function add(out, v1, v2) { - out[0] = v1[0] + v2[0]; - out[1] = v1[1] + v2[1]; - return out; - } - function scaleAndAdd(out, v1, v2, a) { - out[0] = v1[0] + v2[0] * a; - out[1] = v1[1] + v2[1] * a; - return out; - } - function sub(out, v1, v2) { - out[0] = v1[0] - v2[0]; - out[1] = v1[1] - v2[1]; - return out; - } - function len(v) { - return Math.sqrt(lenSquare(v)); - } - var length = len; - function lenSquare(v) { - return v[0] * v[0] + v[1] * v[1]; - } - var lengthSquare = lenSquare; - function mul(out, v1, v2) { - out[0] = v1[0] * v2[0]; - out[1] = v1[1] * v2[1]; - return out; - } - function div(out, v1, v2) { - out[0] = v1[0] / v2[0]; - out[1] = v1[1] / v2[1]; - return out; - } - function dot(v1, v2) { - return v1[0] * v2[0] + v1[1] * v2[1]; - } - function scale(out, v, s) { - out[0] = v[0] * s; - out[1] = v[1] * s; - return out; - } - function normalize(out, v) { - var d = len(v); - if (d === 0) { - out[0] = 0; - out[1] = 0; - } - else { - out[0] = v[0] / d; - out[1] = v[1] / d; - } - return out; - } - function distance(v1, v2) { - return Math.sqrt((v1[0] - v2[0]) * (v1[0] - v2[0]) - + (v1[1] - v2[1]) * (v1[1] - v2[1])); - } - var dist = distance; - function distanceSquare(v1, v2) { - return (v1[0] - v2[0]) * (v1[0] - v2[0]) - + (v1[1] - v2[1]) * (v1[1] - v2[1]); - } - var distSquare = distanceSquare; - function negate(out, v) { - out[0] = -v[0]; - out[1] = -v[1]; - return out; - } - function lerp(out, v1, v2, t) { - out[0] = v1[0] + t * (v2[0] - v1[0]); - out[1] = v1[1] + t * (v2[1] - v1[1]); - return out; - } - function applyTransform(out, v, m) { - var x = v[0]; - var y = v[1]; - out[0] = m[0] * x + m[2] * y + m[4]; - out[1] = m[1] * x + m[3] * y + m[5]; - return out; - } - function min(out, v1, v2) { - out[0] = Math.min(v1[0], v2[0]); - out[1] = Math.min(v1[1], v2[1]); - return out; - } - function max(out, v1, v2) { - out[0] = Math.max(v1[0], v2[0]); - out[1] = Math.max(v1[1], v2[1]); - return out; - } - - var vector = /*#__PURE__*/Object.freeze({ - __proto__: null, - create: create, - copy: copy, - clone: clone$1, - set: set, - add: add, - scaleAndAdd: scaleAndAdd, - sub: sub, - len: len, - length: length, - lenSquare: lenSquare, - lengthSquare: lengthSquare, - mul: mul, - div: div, - dot: dot, - scale: scale, - normalize: normalize, - distance: distance, - dist: dist, - distanceSquare: distanceSquare, - distSquare: distSquare, - negate: negate, - lerp: lerp, - applyTransform: applyTransform, - min: min, - max: max - }); - - var Param = (function () { - function Param(target, e) { - this.target = target; - this.topTarget = e && e.topTarget; - } - return Param; - }()); - var Draggable = (function () { - function Draggable(handler) { - this.handler = handler; - handler.on('mousedown', this._dragStart, this); - handler.on('mousemove', this._drag, this); - handler.on('mouseup', this._dragEnd, this); - } - Draggable.prototype._dragStart = function (e) { - var draggingTarget = e.target; - while (draggingTarget && !draggingTarget.draggable) { - draggingTarget = draggingTarget.parent || draggingTarget.__hostTarget; - } - if (draggingTarget) { - this._draggingTarget = draggingTarget; - draggingTarget.dragging = true; - this._x = e.offsetX; - this._y = e.offsetY; - this.handler.dispatchToElement(new Param(draggingTarget, e), 'dragstart', e.event); - } - }; - Draggable.prototype._drag = function (e) { - var draggingTarget = this._draggingTarget; - if (draggingTarget) { - var x = e.offsetX; - var y = e.offsetY; - var dx = x - this._x; - var dy = y - this._y; - this._x = x; - this._y = y; - draggingTarget.drift(dx, dy, e); - this.handler.dispatchToElement(new Param(draggingTarget, e), 'drag', e.event); - var dropTarget = this.handler.findHover(x, y, draggingTarget).target; - var lastDropTarget = this._dropTarget; - this._dropTarget = dropTarget; - if (draggingTarget !== dropTarget) { - if (lastDropTarget && dropTarget !== lastDropTarget) { - this.handler.dispatchToElement(new Param(lastDropTarget, e), 'dragleave', e.event); - } - if (dropTarget && dropTarget !== lastDropTarget) { - this.handler.dispatchToElement(new Param(dropTarget, e), 'dragenter', e.event); - } - } - } - }; - Draggable.prototype._dragEnd = function (e) { - var draggingTarget = this._draggingTarget; - if (draggingTarget) { - draggingTarget.dragging = false; - } - this.handler.dispatchToElement(new Param(draggingTarget, e), 'dragend', e.event); - if (this._dropTarget) { - this.handler.dispatchToElement(new Param(this._dropTarget, e), 'drop', e.event); - } - this._draggingTarget = null; - this._dropTarget = null; - }; - return Draggable; - }()); - - var Eventful = (function () { - function Eventful(eventProcessors) { - if (eventProcessors) { - this._$eventProcessor = eventProcessors; - } - } - Eventful.prototype.on = function (event, query, handler, context) { - if (!this._$handlers) { - this._$handlers = {}; - } - var _h = this._$handlers; - if (typeof query === 'function') { - context = handler; - handler = query; - query = null; - } - if (!handler || !event) { - return this; - } - var eventProcessor = this._$eventProcessor; - if (query != null && eventProcessor && eventProcessor.normalizeQuery) { - query = eventProcessor.normalizeQuery(query); - } - if (!_h[event]) { - _h[event] = []; - } - for (var i = 0; i < _h[event].length; i++) { - if (_h[event][i].h === handler) { - return this; - } - } - var wrap = { - h: handler, - query: query, - ctx: (context || this), - callAtLast: handler.zrEventfulCallAtLast - }; - var lastIndex = _h[event].length - 1; - var lastWrap = _h[event][lastIndex]; - (lastWrap && lastWrap.callAtLast) - ? _h[event].splice(lastIndex, 0, wrap) - : _h[event].push(wrap); - return this; - }; - Eventful.prototype.isSilent = function (eventName) { - var _h = this._$handlers; - return !_h || !_h[eventName] || !_h[eventName].length; - }; - Eventful.prototype.off = function (eventType, handler) { - var _h = this._$handlers; - if (!_h) { - return this; - } - if (!eventType) { - this._$handlers = {}; - return this; - } - if (handler) { - if (_h[eventType]) { - var newList = []; - for (var i = 0, l = _h[eventType].length; i < l; i++) { - if (_h[eventType][i].h !== handler) { - newList.push(_h[eventType][i]); - } - } - _h[eventType] = newList; - } - if (_h[eventType] && _h[eventType].length === 0) { - delete _h[eventType]; - } - } - else { - delete _h[eventType]; - } - return this; - }; - Eventful.prototype.trigger = function (eventType) { - var args = []; - for (var _i = 1; _i < arguments.length; _i++) { - args[_i - 1] = arguments[_i]; - } - if (!this._$handlers) { - return this; - } - var _h = this._$handlers[eventType]; - var eventProcessor = this._$eventProcessor; - if (_h) { - var argLen = args.length; - var len = _h.length; - for (var i = 0; i < len; i++) { - var hItem = _h[i]; - if (eventProcessor - && eventProcessor.filter - && hItem.query != null - && !eventProcessor.filter(eventType, hItem.query)) { - continue; - } - switch (argLen) { - case 0: - hItem.h.call(hItem.ctx); - break; - case 1: - hItem.h.call(hItem.ctx, args[0]); - break; - case 2: - hItem.h.call(hItem.ctx, args[0], args[1]); - break; - default: - hItem.h.apply(hItem.ctx, args); - break; - } - } - } - eventProcessor && eventProcessor.afterTrigger - && eventProcessor.afterTrigger(eventType); - return this; - }; - Eventful.prototype.triggerWithContext = function (type) { - var args = []; - for (var _i = 1; _i < arguments.length; _i++) { - args[_i - 1] = arguments[_i]; - } - if (!this._$handlers) { - return this; - } - var _h = this._$handlers[type]; - var eventProcessor = this._$eventProcessor; - if (_h) { - var argLen = args.length; - var ctx = args[argLen - 1]; - var len = _h.length; - for (var i = 0; i < len; i++) { - var hItem = _h[i]; - if (eventProcessor - && eventProcessor.filter - && hItem.query != null - && !eventProcessor.filter(type, hItem.query)) { - continue; - } - switch (argLen) { - case 0: - hItem.h.call(ctx); - break; - case 1: - hItem.h.call(ctx, args[0]); - break; - case 2: - hItem.h.call(ctx, args[0], args[1]); - break; - default: - hItem.h.apply(ctx, args.slice(1, argLen - 1)); - break; - } - } - } - eventProcessor && eventProcessor.afterTrigger - && eventProcessor.afterTrigger(type); - return this; - }; - return Eventful; - }()); - - var LN2 = Math.log(2); - function determinant(rows, rank, rowStart, rowMask, colMask, detCache) { - var cacheKey = rowMask + '-' + colMask; - var fullRank = rows.length; - if (detCache.hasOwnProperty(cacheKey)) { - return detCache[cacheKey]; - } - if (rank === 1) { - var colStart = Math.round(Math.log(((1 << fullRank) - 1) & ~colMask) / LN2); - return rows[rowStart][colStart]; - } - var subRowMask = rowMask | (1 << rowStart); - var subRowStart = rowStart + 1; - while (rowMask & (1 << subRowStart)) { - subRowStart++; - } - var sum = 0; - for (var j = 0, colLocalIdx = 0; j < fullRank; j++) { - var colTag = 1 << j; - if (!(colTag & colMask)) { - sum += (colLocalIdx % 2 ? -1 : 1) * rows[rowStart][j] - * determinant(rows, rank - 1, subRowStart, subRowMask, colMask | colTag, detCache); - colLocalIdx++; - } - } - detCache[cacheKey] = sum; - return sum; - } - function buildTransformer(src, dest) { - var mA = [ - [src[0], src[1], 1, 0, 0, 0, -dest[0] * src[0], -dest[0] * src[1]], - [0, 0, 0, src[0], src[1], 1, -dest[1] * src[0], -dest[1] * src[1]], - [src[2], src[3], 1, 0, 0, 0, -dest[2] * src[2], -dest[2] * src[3]], - [0, 0, 0, src[2], src[3], 1, -dest[3] * src[2], -dest[3] * src[3]], - [src[4], src[5], 1, 0, 0, 0, -dest[4] * src[4], -dest[4] * src[5]], - [0, 0, 0, src[4], src[5], 1, -dest[5] * src[4], -dest[5] * src[5]], - [src[6], src[7], 1, 0, 0, 0, -dest[6] * src[6], -dest[6] * src[7]], - [0, 0, 0, src[6], src[7], 1, -dest[7] * src[6], -dest[7] * src[7]] - ]; - var detCache = {}; - var det = determinant(mA, 8, 0, 0, 0, detCache); - if (det === 0) { - return; - } - var vh = []; - for (var i = 0; i < 8; i++) { - for (var j = 0; j < 8; j++) { - vh[j] == null && (vh[j] = 0); - vh[j] += ((i + j) % 2 ? -1 : 1) - * determinant(mA, 7, i === 0 ? 1 : 0, 1 << i, 1 << j, detCache) - / det * dest[i]; - } - } - return function (out, srcPointX, srcPointY) { - var pk = srcPointX * vh[6] + srcPointY * vh[7] + 1; - out[0] = (srcPointX * vh[0] + srcPointY * vh[1] + vh[2]) / pk; - out[1] = (srcPointX * vh[3] + srcPointY * vh[4] + vh[5]) / pk; - }; - } - - var EVENT_SAVED_PROP = '___zrEVENTSAVED'; - var _calcOut = []; - function transformLocalCoord(out, elFrom, elTarget, inX, inY) { - return transformCoordWithViewport(_calcOut, elFrom, inX, inY, true) - && transformCoordWithViewport(out, elTarget, _calcOut[0], _calcOut[1]); - } - function transformCoordWithViewport(out, el, inX, inY, inverse) { - if (el.getBoundingClientRect && env.domSupported && !isCanvasEl(el)) { - var saved = el[EVENT_SAVED_PROP] || (el[EVENT_SAVED_PROP] = {}); - var markers = prepareCoordMarkers(el, saved); - var transformer = preparePointerTransformer(markers, saved, inverse); - if (transformer) { - transformer(out, inX, inY); - return true; - } - } - return false; - } - function prepareCoordMarkers(el, saved) { - var markers = saved.markers; - if (markers) { - return markers; - } - markers = saved.markers = []; - var propLR = ['left', 'right']; - var propTB = ['top', 'bottom']; - for (var i = 0; i < 4; i++) { - var marker = document.createElement('div'); - var stl = marker.style; - var idxLR = i % 2; - var idxTB = (i >> 1) % 2; - stl.cssText = [ - 'position: absolute', - 'visibility: hidden', - 'padding: 0', - 'margin: 0', - 'border-width: 0', - 'user-select: none', - 'width:0', - 'height:0', - propLR[idxLR] + ':0', - propTB[idxTB] + ':0', - propLR[1 - idxLR] + ':auto', - propTB[1 - idxTB] + ':auto', - '' - ].join('!important;'); - el.appendChild(marker); - markers.push(marker); - } - return markers; - } - function preparePointerTransformer(markers, saved, inverse) { - var transformerName = inverse ? 'invTrans' : 'trans'; - var transformer = saved[transformerName]; - var oldSrcCoords = saved.srcCoords; - var srcCoords = []; - var destCoords = []; - var oldCoordTheSame = true; - for (var i = 0; i < 4; i++) { - var rect = markers[i].getBoundingClientRect(); - var ii = 2 * i; - var x = rect.left; - var y = rect.top; - srcCoords.push(x, y); - oldCoordTheSame = oldCoordTheSame && oldSrcCoords && x === oldSrcCoords[ii] && y === oldSrcCoords[ii + 1]; - destCoords.push(markers[i].offsetLeft, markers[i].offsetTop); - } - return (oldCoordTheSame && transformer) - ? transformer - : (saved.srcCoords = srcCoords, - saved[transformerName] = inverse - ? buildTransformer(destCoords, srcCoords) - : buildTransformer(srcCoords, destCoords)); - } - function isCanvasEl(el) { - return el.nodeName.toUpperCase() === 'CANVAS'; - } - var replaceReg = /([&<>"'])/g; - var replaceMap = { - '&': '&amp;', - '<': '&lt;', - '>': '&gt;', - '"': '&quot;', - '\'': '&#39;' - }; - function encodeHTML(source) { - return source == null - ? '' - : (source + '').replace(replaceReg, function (str, c) { - return replaceMap[c]; - }); - } - - var MOUSE_EVENT_REG = /^(?:mouse|pointer|contextmenu|drag|drop)|click/; - var _calcOut$1 = []; - var firefoxNotSupportOffsetXY = env.browser.firefox - && +env.browser.version.split('.')[0] < 39; - function clientToLocal(el, e, out, calculate) { - out = out || {}; - if (calculate) { - calculateZrXY(el, e, out); - } - else if (firefoxNotSupportOffsetXY - && e.layerX != null - && e.layerX !== e.offsetX) { - out.zrX = e.layerX; - out.zrY = e.layerY; - } - else if (e.offsetX != null) { - out.zrX = e.offsetX; - out.zrY = e.offsetY; - } - else { - calculateZrXY(el, e, out); - } - return out; - } - function calculateZrXY(el, e, out) { - if (env.domSupported && el.getBoundingClientRect) { - var ex = e.clientX; - var ey = e.clientY; - if (isCanvasEl(el)) { - var box = el.getBoundingClientRect(); - out.zrX = ex - box.left; - out.zrY = ey - box.top; - return; - } - else { - if (transformCoordWithViewport(_calcOut$1, el, ex, ey)) { - out.zrX = _calcOut$1[0]; - out.zrY = _calcOut$1[1]; - return; - } - } - } - out.zrX = out.zrY = 0; - } - function getNativeEvent(e) { - return e - || window.event; - } - function normalizeEvent(el, e, calculate) { - e = getNativeEvent(e); - if (e.zrX != null) { - return e; - } - var eventType = e.type; - var isTouch = eventType && eventType.indexOf('touch') >= 0; - if (!isTouch) { - clientToLocal(el, e, e, calculate); - var wheelDelta = getWheelDeltaMayPolyfill(e); - e.zrDelta = wheelDelta ? wheelDelta / 120 : -(e.detail || 0) / 3; - } - else { - var touch = eventType !== 'touchend' - ? e.targetTouches[0] - : e.changedTouches[0]; - touch && clientToLocal(el, touch, e, calculate); - } - var button = e.button; - if (e.which == null && button !== undefined && MOUSE_EVENT_REG.test(e.type)) { - e.which = (button & 1 ? 1 : (button & 2 ? 3 : (button & 4 ? 2 : 0))); - } - return e; - } - function getWheelDeltaMayPolyfill(e) { - var rawWheelDelta = e.wheelDelta; - if (rawWheelDelta) { - return rawWheelDelta; - } - var deltaX = e.deltaX; - var deltaY = e.deltaY; - if (deltaX == null || deltaY == null) { - return rawWheelDelta; - } - var delta = deltaY !== 0 ? Math.abs(deltaY) : Math.abs(deltaX); - var sign = deltaY > 0 ? -1 - : deltaY < 0 ? 1 - : deltaX > 0 ? -1 - : 1; - return 3 * delta * sign; - } - function addEventListener(el, name, handler, opt) { - el.addEventListener(name, handler, opt); - } - function removeEventListener(el, name, handler, opt) { - el.removeEventListener(name, handler, opt); - } - var stop = function (e) { - e.preventDefault(); - e.stopPropagation(); - e.cancelBubble = true; - }; - function isMiddleOrRightButtonOnMouseUpDown(e) { - return e.which === 2 || e.which === 3; - } - - var GestureMgr = (function () { - function GestureMgr() { - this._track = []; - } - GestureMgr.prototype.recognize = function (event, target, root) { - this._doTrack(event, target, root); - return this._recognize(event); - }; - GestureMgr.prototype.clear = function () { - this._track.length = 0; - return this; - }; - GestureMgr.prototype._doTrack = function (event, target, root) { - var touches = event.touches; - if (!touches) { - return; - } - var trackItem = { - points: [], - touches: [], - target: target, - event: event - }; - for (var i = 0, len = touches.length; i < len; i++) { - var touch = touches[i]; - var pos = clientToLocal(root, touch, {}); - trackItem.points.push([pos.zrX, pos.zrY]); - trackItem.touches.push(touch); - } - this._track.push(trackItem); - }; - GestureMgr.prototype._recognize = function (event) { - for (var eventName in recognizers) { - if (recognizers.hasOwnProperty(eventName)) { - var gestureInfo = recognizers[eventName](this._track, event); - if (gestureInfo) { - return gestureInfo; - } - } - } - }; - return GestureMgr; - }()); - function dist$1(pointPair) { - var dx = pointPair[1][0] - pointPair[0][0]; - var dy = pointPair[1][1] - pointPair[0][1]; - return Math.sqrt(dx * dx + dy * dy); - } - function center(pointPair) { - return [ - (pointPair[0][0] + pointPair[1][0]) / 2, - (pointPair[0][1] + pointPair[1][1]) / 2 - ]; - } - var recognizers = { - pinch: function (tracks, event) { - var trackLen = tracks.length; - if (!trackLen) { - return; - } - var pinchEnd = (tracks[trackLen - 1] || {}).points; - var pinchPre = (tracks[trackLen - 2] || {}).points || pinchEnd; - if (pinchPre - && pinchPre.length > 1 - && pinchEnd - && pinchEnd.length > 1) { - var pinchScale = dist$1(pinchEnd) / dist$1(pinchPre); - !isFinite(pinchScale) && (pinchScale = 1); - event.pinchScale = pinchScale; - var pinchCenter = center(pinchEnd); - event.pinchX = pinchCenter[0]; - event.pinchY = pinchCenter[1]; - return { - type: 'pinch', - target: tracks[0].target, - event: event - }; - } - } - }; - - function create$1() { - return [1, 0, 0, 1, 0, 0]; - } - function identity(out) { - out[0] = 1; - out[1] = 0; - out[2] = 0; - out[3] = 1; - out[4] = 0; - out[5] = 0; - return out; - } - function copy$1(out, m) { - out[0] = m[0]; - out[1] = m[1]; - out[2] = m[2]; - out[3] = m[3]; - out[4] = m[4]; - out[5] = m[5]; - return out; - } - function mul$1(out, m1, m2) { - var out0 = m1[0] * m2[0] + m1[2] * m2[1]; - var out1 = m1[1] * m2[0] + m1[3] * m2[1]; - var out2 = m1[0] * m2[2] + m1[2] * m2[3]; - var out3 = m1[1] * m2[2] + m1[3] * m2[3]; - var out4 = m1[0] * m2[4] + m1[2] * m2[5] + m1[4]; - var out5 = m1[1] * m2[4] + m1[3] * m2[5] + m1[5]; - out[0] = out0; - out[1] = out1; - out[2] = out2; - out[3] = out3; - out[4] = out4; - out[5] = out5; - return out; - } - function translate(out, a, v) { - out[0] = a[0]; - out[1] = a[1]; - out[2] = a[2]; - out[3] = a[3]; - out[4] = a[4] + v[0]; - out[5] = a[5] + v[1]; - return out; - } - function rotate(out, a, rad) { - var aa = a[0]; - var ac = a[2]; - var atx = a[4]; - var ab = a[1]; - var ad = a[3]; - var aty = a[5]; - var st = Math.sin(rad); - var ct = Math.cos(rad); - out[0] = aa * ct + ab * st; - out[1] = -aa * st + ab * ct; - out[2] = ac * ct + ad * st; - out[3] = -ac * st + ct * ad; - out[4] = ct * atx + st * aty; - out[5] = ct * aty - st * atx; - return out; - } - function scale$1(out, a, v) { - var vx = v[0]; - var vy = v[1]; - out[0] = a[0] * vx; - out[1] = a[1] * vy; - out[2] = a[2] * vx; - out[3] = a[3] * vy; - out[4] = a[4] * vx; - out[5] = a[5] * vy; - return out; - } - function invert(out, a) { - var aa = a[0]; - var ac = a[2]; - var atx = a[4]; - var ab = a[1]; - var ad = a[3]; - var aty = a[5]; - var det = aa * ad - ab * ac; - if (!det) { - return null; - } - det = 1.0 / det; - out[0] = ad * det; - out[1] = -ab * det; - out[2] = -ac * det; - out[3] = aa * det; - out[4] = (ac * aty - ad * atx) * det; - out[5] = (ab * atx - aa * aty) * det; - return out; - } - function clone$2(a) { - var b = create$1(); - copy$1(b, a); - return b; - } - - var matrix = /*#__PURE__*/Object.freeze({ - __proto__: null, - create: create$1, - identity: identity, - copy: copy$1, - mul: mul$1, - translate: translate, - rotate: rotate, - scale: scale$1, - invert: invert, - clone: clone$2 - }); - - var Point = (function () { - function Point(x, y) { - this.x = x || 0; - this.y = y || 0; - } - Point.prototype.copy = function (other) { - this.x = other.x; - this.y = other.y; - return this; - }; - Point.prototype.clone = function () { - return new Point(this.x, this.y); - }; - Point.prototype.set = function (x, y) { - this.x = x; - this.y = y; - return this; - }; - Point.prototype.equal = function (other) { - return other.x === this.x && other.y === this.y; - }; - Point.prototype.add = function (other) { - this.x += other.x; - this.y += other.y; - return this; - }; - Point.prototype.scale = function (scalar) { - this.x *= scalar; - this.y *= scalar; - }; - Point.prototype.scaleAndAdd = function (other, scalar) { - this.x += other.x * scalar; - this.y += other.y * scalar; - }; - Point.prototype.sub = function (other) { - this.x -= other.x; - this.y -= other.y; - return this; - }; - Point.prototype.dot = function (other) { - return this.x * other.x + this.y * other.y; - }; - Point.prototype.len = function () { - return Math.sqrt(this.x * this.x + this.y * this.y); - }; - Point.prototype.lenSquare = function () { - return this.x * this.x + this.y * this.y; - }; - Point.prototype.normalize = function () { - var len = this.len(); - this.x /= len; - this.y /= len; - return this; - }; - Point.prototype.distance = function (other) { - var dx = this.x - other.x; - var dy = this.y - other.y; - return Math.sqrt(dx * dx + dy * dy); - }; - Point.prototype.distanceSquare = function (other) { - var dx = this.x - other.x; - var dy = this.y - other.y; - return dx * dx + dy * dy; - }; - Point.prototype.negate = function () { - this.x = -this.x; - this.y = -this.y; - return this; - }; - Point.prototype.transform = function (m) { - if (!m) { - return; - } - var x = this.x; - var y = this.y; - this.x = m[0] * x + m[2] * y + m[4]; - this.y = m[1] * x + m[3] * y + m[5]; - return this; - }; - Point.prototype.toArray = function (out) { - out[0] = this.x; - out[1] = this.y; - return out; - }; - Point.prototype.fromArray = function (input) { - this.x = input[0]; - this.y = input[1]; - }; - Point.set = function (p, x, y) { - p.x = x; - p.y = y; - }; - Point.copy = function (p, p2) { - p.x = p2.x; - p.y = p2.y; - }; - Point.len = function (p) { - return Math.sqrt(p.x * p.x + p.y * p.y); - }; - Point.lenSquare = function (p) { - return p.x * p.x + p.y * p.y; - }; - Point.dot = function (p0, p1) { - return p0.x * p1.x + p0.y * p1.y; - }; - Point.add = function (out, p0, p1) { - out.x = p0.x + p1.x; - out.y = p0.y + p1.y; - }; - Point.sub = function (out, p0, p1) { - out.x = p0.x - p1.x; - out.y = p0.y - p1.y; - }; - Point.scale = function (out, p0, scalar) { - out.x = p0.x * scalar; - out.y = p0.y * scalar; - }; - Point.scaleAndAdd = function (out, p0, p1, scalar) { - out.x = p0.x + p1.x * scalar; - out.y = p0.y + p1.y * scalar; - }; - Point.lerp = function (out, p0, p1, t) { - var onet = 1 - t; - out.x = onet * p0.x + t * p1.x; - out.y = onet * p0.y + t * p1.y; - }; - return Point; - }()); - - var mathMin = Math.min; - var mathMax = Math.max; - var lt = new Point(); - var rb = new Point(); - var lb = new Point(); - var rt = new Point(); - var minTv = new Point(); - var maxTv = new Point(); - var BoundingRect = (function () { - function BoundingRect(x, y, width, height) { - if (width < 0) { - x = x + width; - width = -width; - } - if (height < 0) { - y = y + height; - height = -height; - } - this.x = x; - this.y = y; - this.width = width; - this.height = height; - } - BoundingRect.prototype.union = function (other) { - var x = mathMin(other.x, this.x); - var y = mathMin(other.y, this.y); - if (isFinite(this.x) && isFinite(this.width)) { - this.width = mathMax(other.x + other.width, this.x + this.width) - x; - } - else { - this.width = other.width; - } - if (isFinite(this.y) && isFinite(this.height)) { - this.height = mathMax(other.y + other.height, this.y + this.height) - y; - } - else { - this.height = other.height; - } - this.x = x; - this.y = y; - }; - BoundingRect.prototype.applyTransform = function (m) { - BoundingRect.applyTransform(this, this, m); - }; - BoundingRect.prototype.calculateTransform = function (b) { - var a = this; - var sx = b.width / a.width; - var sy = b.height / a.height; - var m = create$1(); - translate(m, m, [-a.x, -a.y]); - scale$1(m, m, [sx, sy]); - translate(m, m, [b.x, b.y]); - return m; - }; - BoundingRect.prototype.intersect = function (b, mtv) { - if (!b) { - return false; - } - if (!(b instanceof BoundingRect)) { - b = BoundingRect.create(b); - } - var a = this; - var ax0 = a.x; - var ax1 = a.x + a.width; - var ay0 = a.y; - var ay1 = a.y + a.height; - var bx0 = b.x; - var bx1 = b.x + b.width; - var by0 = b.y; - var by1 = b.y + b.height; - var overlap = !(ax1 < bx0 || bx1 < ax0 || ay1 < by0 || by1 < ay0); - if (mtv) { - var dMin = Infinity; - var dMax = 0; - var d0 = Math.abs(ax1 - bx0); - var d1 = Math.abs(bx1 - ax0); - var d2 = Math.abs(ay1 - by0); - var d3 = Math.abs(by1 - ay0); - var dx = Math.min(d0, d1); - var dy = Math.min(d2, d3); - if (ax1 < bx0 || bx1 < ax0) { - if (dx > dMax) { - dMax = dx; - if (d0 < d1) { - Point.set(maxTv, -d0, 0); - } - else { - Point.set(maxTv, d1, 0); - } - } - } - else { - if (dx < dMin) { - dMin = dx; - if (d0 < d1) { - Point.set(minTv, d0, 0); - } - else { - Point.set(minTv, -d1, 0); - } - } - } - if (ay1 < by0 || by1 < ay0) { - if (dy > dMax) { - dMax = dy; - if (d2 < d3) { - Point.set(maxTv, 0, -d2); - } - else { - Point.set(maxTv, 0, d3); - } - } - } - else { - if (dx < dMin) { - dMin = dx; - if (d2 < d3) { - Point.set(minTv, 0, d2); - } - else { - Point.set(minTv, 0, -d3); - } - } - } - } - if (mtv) { - Point.copy(mtv, overlap ? minTv : maxTv); - } - return overlap; - }; - BoundingRect.prototype.contain = function (x, y) { - var rect = this; - return x >= rect.x - && x <= (rect.x + rect.width) - && y >= rect.y - && y <= (rect.y + rect.height); - }; - BoundingRect.prototype.clone = function () { - return new BoundingRect(this.x, this.y, this.width, this.height); - }; - BoundingRect.prototype.copy = function (other) { - BoundingRect.copy(this, other); - }; - BoundingRect.prototype.plain = function () { - return { - x: this.x, - y: this.y, - width: this.width, - height: this.height - }; - }; - BoundingRect.prototype.isFinite = function () { - return isFinite(this.x) - && isFinite(this.y) - && isFinite(this.width) - && isFinite(this.height); - }; - BoundingRect.prototype.isZero = function () { - return this.width === 0 || this.height === 0; - }; - BoundingRect.create = function (rect) { - return new BoundingRect(rect.x, rect.y, rect.width, rect.height); - }; - BoundingRect.copy = function (target, source) { - target.x = source.x; - target.y = source.y; - target.width = source.width; - target.height = source.height; - }; - BoundingRect.applyTransform = function (target, source, m) { - if (!m) { - if (target !== source) { - BoundingRect.copy(target, source); - } - return; - } - if (m[1] < 1e-5 && m[1] > -1e-5 && m[2] < 1e-5 && m[2] > -1e-5) { - var sx = m[0]; - var sy = m[3]; - var tx = m[4]; - var ty = m[5]; - target.x = source.x * sx + tx; - target.y = source.y * sy + ty; - target.width = source.width * sx; - target.height = source.height * sy; - if (target.width < 0) { - target.x += target.width; - target.width = -target.width; - } - if (target.height < 0) { - target.y += target.height; - target.height = -target.height; - } - return; - } - lt.x = lb.x = source.x; - lt.y = rt.y = source.y; - rb.x = rt.x = source.x + source.width; - rb.y = lb.y = source.y + source.height; - lt.transform(m); - rt.transform(m); - rb.transform(m); - lb.transform(m); - target.x = mathMin(lt.x, rb.x, lb.x, rt.x); - target.y = mathMin(lt.y, rb.y, lb.y, rt.y); - var maxX = mathMax(lt.x, rb.x, lb.x, rt.x); - var maxY = mathMax(lt.y, rb.y, lb.y, rt.y); - target.width = maxX - target.x; - target.height = maxY - target.y; - }; - return BoundingRect; - }()); - - var SILENT = 'silent'; - function makeEventPacket(eveType, targetInfo, event) { - return { - type: eveType, - event: event, - target: targetInfo.target, - topTarget: targetInfo.topTarget, - cancelBubble: false, - offsetX: event.zrX, - offsetY: event.zrY, - gestureEvent: event.gestureEvent, - pinchX: event.pinchX, - pinchY: event.pinchY, - pinchScale: event.pinchScale, - wheelDelta: event.zrDelta, - zrByTouch: event.zrByTouch, - which: event.which, - stop: stopEvent - }; - } - function stopEvent() { - stop(this.event); - } - var EmptyProxy = (function (_super) { - __extends(EmptyProxy, _super); - function EmptyProxy() { - var _this = _super !== null && _super.apply(this, arguments) || this; - _this.handler = null; - return _this; - } - EmptyProxy.prototype.dispose = function () { }; - EmptyProxy.prototype.setCursor = function () { }; - return EmptyProxy; - }(Eventful)); - var HoveredResult = (function () { - function HoveredResult(x, y) { - this.x = x; - this.y = y; - } - return HoveredResult; - }()); - var handlerNames = [ - 'click', 'dblclick', 'mousewheel', 'mouseout', - 'mouseup', 'mousedown', 'mousemove', 'contextmenu' - ]; - var tmpRect = new BoundingRect(0, 0, 0, 0); - var Handler = (function (_super) { - __extends(Handler, _super); - function Handler(storage, painter, proxy, painterRoot, pointerSize) { - var _this = _super.call(this) || this; - _this._hovered = new HoveredResult(0, 0); - _this.storage = storage; - _this.painter = painter; - _this.painterRoot = painterRoot; - _this._pointerSize = pointerSize; - proxy = proxy || new EmptyProxy(); - _this.proxy = null; - _this.setHandlerProxy(proxy); - _this._draggingMgr = new Draggable(_this); - return _this; - } - Handler.prototype.setHandlerProxy = function (proxy) { - if (this.proxy) { - this.proxy.dispose(); - } - if (proxy) { - each(handlerNames, function (name) { - proxy.on && proxy.on(name, this[name], this); - }, this); - proxy.handler = this; - } - this.proxy = proxy; - }; - Handler.prototype.mousemove = function (event) { - var x = event.zrX; - var y = event.zrY; - var isOutside = isOutsideBoundary(this, x, y); - var lastHovered = this._hovered; - var lastHoveredTarget = lastHovered.target; - if (lastHoveredTarget && !lastHoveredTarget.__zr) { - lastHovered = this.findHover(lastHovered.x, lastHovered.y); - lastHoveredTarget = lastHovered.target; - } - var hovered = this._hovered = isOutside ? new HoveredResult(x, y) : this.findHover(x, y); - var hoveredTarget = hovered.target; - var proxy = this.proxy; - proxy.setCursor && proxy.setCursor(hoveredTarget ? hoveredTarget.cursor : 'default'); - if (lastHoveredTarget && hoveredTarget !== lastHoveredTarget) { - this.dispatchToElement(lastHovered, 'mouseout', event); - } - this.dispatchToElement(hovered, 'mousemove', event); - if (hoveredTarget && hoveredTarget !== lastHoveredTarget) { - this.dispatchToElement(hovered, 'mouseover', event); - } - }; - Handler.prototype.mouseout = function (event) { - var eventControl = event.zrEventControl; - if (eventControl !== 'only_globalout') { - this.dispatchToElement(this._hovered, 'mouseout', event); - } - if (eventControl !== 'no_globalout') { - this.trigger('globalout', { type: 'globalout', event: event }); - } - }; - Handler.prototype.resize = function () { - this._hovered = new HoveredResult(0, 0); - }; - Handler.prototype.dispatch = function (eventName, eventArgs) { - var handler = this[eventName]; - handler && handler.call(this, eventArgs); - }; - Handler.prototype.dispose = function () { - this.proxy.dispose(); - this.storage = null; - this.proxy = null; - this.painter = null; - }; - Handler.prototype.setCursorStyle = function (cursorStyle) { - var proxy = this.proxy; - proxy.setCursor && proxy.setCursor(cursorStyle); - }; - Handler.prototype.dispatchToElement = function (targetInfo, eventName, event) { - targetInfo = targetInfo || {}; - var el = targetInfo.target; - if (el && el.silent) { - return; - } - var eventKey = ('on' + eventName); - var eventPacket = makeEventPacket(eventName, targetInfo, event); - while (el) { - el[eventKey] - && (eventPacket.cancelBubble = !!el[eventKey].call(el, eventPacket)); - el.trigger(eventName, eventPacket); - el = el.__hostTarget ? el.__hostTarget : el.parent; - if (eventPacket.cancelBubble) { - break; - } - } - if (!eventPacket.cancelBubble) { - this.trigger(eventName, eventPacket); - if (this.painter && this.painter.eachOtherLayer) { - this.painter.eachOtherLayer(function (layer) { - if (typeof (layer[eventKey]) === 'function') { - layer[eventKey].call(layer, eventPacket); - } - if (layer.trigger) { - layer.trigger(eventName, eventPacket); - } - }); - } - } - }; - Handler.prototype.findHover = function (x, y, exclude) { - var list = this.storage.getDisplayList(); - var out = new HoveredResult(x, y); - setHoverTarget(list, out, x, y, exclude); - if (this._pointerSize && !out.target) { - var candidates = []; - var pointerSize = this._pointerSize; - var targetSizeHalf = pointerSize / 2; - var pointerRect = new BoundingRect(x - targetSizeHalf, y - targetSizeHalf, pointerSize, pointerSize); - for (var i = list.length - 1; i >= 0; i--) { - var el = list[i]; - if (el !== exclude - && !el.ignore - && !el.ignoreCoarsePointer - && (!el.parent || !el.parent.ignoreCoarsePointer)) { - tmpRect.copy(el.getBoundingRect()); - if (el.transform) { - tmpRect.applyTransform(el.transform); - } - if (tmpRect.intersect(pointerRect)) { - candidates.push(el); - } - } - } - if (candidates.length) { - var rStep = 4; - var thetaStep = Math.PI / 12; - var PI2 = Math.PI * 2; - for (var r = 0; r < targetSizeHalf; r += rStep) { - for (var theta = 0; theta < PI2; theta += thetaStep) { - var x1 = x + r * Math.cos(theta); - var y1 = y + r * Math.sin(theta); - setHoverTarget(candidates, out, x1, y1, exclude); - if (out.target) { - return out; - } - } - } - } - } - return out; - }; - Handler.prototype.processGesture = function (event, stage) { - if (!this._gestureMgr) { - this._gestureMgr = new GestureMgr(); - } - var gestureMgr = this._gestureMgr; - stage === 'start' && gestureMgr.clear(); - var gestureInfo = gestureMgr.recognize(event, this.findHover(event.zrX, event.zrY, null).target, this.proxy.dom); - stage === 'end' && gestureMgr.clear(); - if (gestureInfo) { - var type = gestureInfo.type; - event.gestureEvent = type; - var res = new HoveredResult(); - res.target = gestureInfo.target; - this.dispatchToElement(res, type, gestureInfo.event); - } - }; - return Handler; - }(Eventful)); - each(['click', 'mousedown', 'mouseup', 'mousewheel', 'dblclick', 'contextmenu'], function (name) { - Handler.prototype[name] = function (event) { - var x = event.zrX; - var y = event.zrY; - var isOutside = isOutsideBoundary(this, x, y); - var hovered; - var hoveredTarget; - if (name !== 'mouseup' || !isOutside) { - hovered = this.findHover(x, y); - hoveredTarget = hovered.target; - } - if (name === 'mousedown') { - this._downEl = hoveredTarget; - this._downPoint = [event.zrX, event.zrY]; - this._upEl = hoveredTarget; - } - else if (name === 'mouseup') { - this._upEl = hoveredTarget; - } - else if (name === 'click') { - if (this._downEl !== this._upEl - || !this._downPoint - || dist(this._downPoint, [event.zrX, event.zrY]) > 4) { - return; - } - this._downPoint = null; - } - this.dispatchToElement(hovered, name, event); - }; - }); - function isHover(displayable, x, y) { - if (displayable[displayable.rectHover ? 'rectContain' : 'contain'](x, y)) { - var el = displayable; - var isSilent = void 0; - var ignoreClip = false; - while (el) { - if (el.ignoreClip) { - ignoreClip = true; - } - if (!ignoreClip) { - var clipPath = el.getClipPath(); - if (clipPath && !clipPath.contain(x, y)) { - return false; - } - if (el.silent) { - isSilent = true; - } - } - var hostEl = el.__hostTarget; - el = hostEl ? hostEl : el.parent; - } - return isSilent ? SILENT : true; - } - return false; - } - function setHoverTarget(list, out, x, y, exclude) { - for (var i = list.length - 1; i >= 0; i--) { - var el = list[i]; - var hoverCheckResult = void 0; - if (el !== exclude - && !el.ignore - && (hoverCheckResult = isHover(el, x, y))) { - !out.topTarget && (out.topTarget = el); - if (hoverCheckResult !== SILENT) { - out.target = el; - break; - } - } - } - } - function isOutsideBoundary(handlerInstance, x, y) { - var painter = handlerInstance.painter; - return x < 0 || x > painter.getWidth() || y < 0 || y > painter.getHeight(); - } - - var DEFAULT_MIN_MERGE = 32; - var DEFAULT_MIN_GALLOPING = 7; - function minRunLength(n) { - var r = 0; - while (n >= DEFAULT_MIN_MERGE) { - r |= n & 1; - n >>= 1; - } - return n + r; - } - function makeAscendingRun(array, lo, hi, compare) { - var runHi = lo + 1; - if (runHi === hi) { - return 1; - } - if (compare(array[runHi++], array[lo]) < 0) { - while (runHi < hi && compare(array[runHi], array[runHi - 1]) < 0) { - runHi++; - } - reverseRun(array, lo, runHi); - } - else { - while (runHi < hi && compare(array[runHi], array[runHi - 1]) >= 0) { - runHi++; - } - } - return runHi - lo; - } - function reverseRun(array, lo, hi) { - hi--; - while (lo < hi) { - var t = array[lo]; - array[lo++] = array[hi]; - array[hi--] = t; - } - } - function binaryInsertionSort(array, lo, hi, start, compare) { - if (start === lo) { - start++; - } - for (; start < hi; start++) { - var pivot = array[start]; - var left = lo; - var right = start; - var mid; - while (left < right) { - mid = left + right >>> 1; - if (compare(pivot, array[mid]) < 0) { - right = mid; - } - else { - left = mid + 1; - } - } - var n = start - left; - switch (n) { - case 3: - array[left + 3] = array[left + 2]; - case 2: - array[left + 2] = array[left + 1]; - case 1: - array[left + 1] = array[left]; - break; - default: - while (n > 0) { - array[left + n] = array[left + n - 1]; - n--; - } - } - array[left] = pivot; - } - } - function gallopLeft(value, array, start, length, hint, compare) { - var lastOffset = 0; - var maxOffset = 0; - var offset = 1; - if (compare(value, array[start + hint]) > 0) { - maxOffset = length - hint; - while (offset < maxOffset && compare(value, array[start + hint + offset]) > 0) { - lastOffset = offset; - offset = (offset << 1) + 1; - if (offset <= 0) { - offset = maxOffset; - } - } - if (offset > maxOffset) { - offset = maxOffset; - } - lastOffset += hint; - offset += hint; - } - else { - maxOffset = hint + 1; - while (offset < maxOffset && compare(value, array[start + hint - offset]) <= 0) { - lastOffset = offset; - offset = (offset << 1) + 1; - if (offset <= 0) { - offset = maxOffset; - } - } - if (offset > maxOffset) { - offset = maxOffset; - } - var tmp = lastOffset; - lastOffset = hint - offset; - offset = hint - tmp; - } - lastOffset++; - while (lastOffset < offset) { - var m = lastOffset + (offset - lastOffset >>> 1); - if (compare(value, array[start + m]) > 0) { - lastOffset = m + 1; - } - else { - offset = m; - } - } - return offset; - } - function gallopRight(value, array, start, length, hint, compare) { - var lastOffset = 0; - var maxOffset = 0; - var offset = 1; - if (compare(value, array[start + hint]) < 0) { - maxOffset = hint + 1; - while (offset < maxOffset && compare(value, array[start + hint - offset]) < 0) { - lastOffset = offset; - offset = (offset << 1) + 1; - if (offset <= 0) { - offset = maxOffset; - } - } - if (offset > maxOffset) { - offset = maxOffset; - } - var tmp = lastOffset; - lastOffset = hint - offset; - offset = hint - tmp; - } - else { - maxOffset = length - hint; - while (offset < maxOffset && compare(value, array[start + hint + offset]) >= 0) { - lastOffset = offset; - offset = (offset << 1) + 1; - if (offset <= 0) { - offset = maxOffset; - } - } - if (offset > maxOffset) { - offset = maxOffset; - } - lastOffset += hint; - offset += hint; - } - lastOffset++; - while (lastOffset < offset) { - var m = lastOffset + (offset - lastOffset >>> 1); - if (compare(value, array[start + m]) < 0) { - offset = m; - } - else { - lastOffset = m + 1; - } - } - return offset; - } - function TimSort(array, compare) { - var minGallop = DEFAULT_MIN_GALLOPING; - var length = 0; - var runStart; - var runLength; - var stackSize = 0; - length = array.length; - var tmp = []; - runStart = []; - runLength = []; - function pushRun(_runStart, _runLength) { - runStart[stackSize] = _runStart; - runLength[stackSize] = _runLength; - stackSize += 1; - } - function mergeRuns() { - while (stackSize > 1) { - var n = stackSize - 2; - if ((n >= 1 && runLength[n - 1] <= runLength[n] + runLength[n + 1]) - || (n >= 2 && runLength[n - 2] <= runLength[n] + runLength[n - 1])) { - if (runLength[n - 1] < runLength[n + 1]) { - n--; - } - } - else if (runLength[n] > runLength[n + 1]) { - break; - } - mergeAt(n); - } - } - function forceMergeRuns() { - while (stackSize > 1) { - var n = stackSize - 2; - if (n > 0 && runLength[n - 1] < runLength[n + 1]) { - n--; - } - mergeAt(n); - } - } - function mergeAt(i) { - var start1 = runStart[i]; - var length1 = runLength[i]; - var start2 = runStart[i + 1]; - var length2 = runLength[i + 1]; - runLength[i] = length1 + length2; - if (i === stackSize - 3) { - runStart[i + 1] = runStart[i + 2]; - runLength[i + 1] = runLength[i + 2]; - } - stackSize--; - var k = gallopRight(array[start2], array, start1, length1, 0, compare); - start1 += k; - length1 -= k; - if (length1 === 0) { - return; - } - length2 = gallopLeft(array[start1 + length1 - 1], array, start2, length2, length2 - 1, compare); - if (length2 === 0) { - return; - } - if (length1 <= length2) { - mergeLow(start1, length1, start2, length2); - } - else { - mergeHigh(start1, length1, start2, length2); - } - } - function mergeLow(start1, length1, start2, length2) { - var i = 0; - for (i = 0; i < length1; i++) { - tmp[i] = array[start1 + i]; - } - var cursor1 = 0; - var cursor2 = start2; - var dest = start1; - array[dest++] = array[cursor2++]; - if (--length2 === 0) { - for (i = 0; i < length1; i++) { - array[dest + i] = tmp[cursor1 + i]; - } - return; - } - if (length1 === 1) { - for (i = 0; i < length2; i++) { - array[dest + i] = array[cursor2 + i]; - } - array[dest + length2] = tmp[cursor1]; - return; - } - var _minGallop = minGallop; - var count1; - var count2; - var exit; - while (1) { - count1 = 0; - count2 = 0; - exit = false; - do { - if (compare(array[cursor2], tmp[cursor1]) < 0) { - array[dest++] = array[cursor2++]; - count2++; - count1 = 0; - if (--length2 === 0) { - exit = true; - break; - } - } - else { - array[dest++] = tmp[cursor1++]; - count1++; - count2 = 0; - if (--length1 === 1) { - exit = true; - break; - } - } - } while ((count1 | count2) < _minGallop); - if (exit) { - break; - } - do { - count1 = gallopRight(array[cursor2], tmp, cursor1, length1, 0, compare); - if (count1 !== 0) { - for (i = 0; i < count1; i++) { - array[dest + i] = tmp[cursor1 + i]; - } - dest += count1; - cursor1 += count1; - length1 -= count1; - if (length1 <= 1) { - exit = true; - break; - } - } - array[dest++] = array[cursor2++]; - if (--length2 === 0) { - exit = true; - break; - } - count2 = gallopLeft(tmp[cursor1], array, cursor2, length2, 0, compare); - if (count2 !== 0) { - for (i = 0; i < count2; i++) { - array[dest + i] = array[cursor2 + i]; - } - dest += count2; - cursor2 += count2; - length2 -= count2; - if (length2 === 0) { - exit = true; - break; - } - } - array[dest++] = tmp[cursor1++]; - if (--length1 === 1) { - exit = true; - break; - } - _minGallop--; - } while (count1 >= DEFAULT_MIN_GALLOPING || count2 >= DEFAULT_MIN_GALLOPING); - if (exit) { - break; - } - if (_minGallop < 0) { - _minGallop = 0; - } - _minGallop += 2; - } - minGallop = _minGallop; - minGallop < 1 && (minGallop = 1); - if (length1 === 1) { - for (i = 0; i < length2; i++) { - array[dest + i] = array[cursor2 + i]; - } - array[dest + length2] = tmp[cursor1]; - } - else if (length1 === 0) { - throw new Error(); - } - else { - for (i = 0; i < length1; i++) { - array[dest + i] = tmp[cursor1 + i]; - } - } - } - function mergeHigh(start1, length1, start2, length2) { - var i = 0; - for (i = 0; i < length2; i++) { - tmp[i] = array[start2 + i]; - } - var cursor1 = start1 + length1 - 1; - var cursor2 = length2 - 1; - var dest = start2 + length2 - 1; - var customCursor = 0; - var customDest = 0; - array[dest--] = array[cursor1--]; - if (--length1 === 0) { - customCursor = dest - (length2 - 1); - for (i = 0; i < length2; i++) { - array[customCursor + i] = tmp[i]; - } - return; - } - if (length2 === 1) { - dest -= length1; - cursor1 -= length1; - customDest = dest + 1; - customCursor = cursor1 + 1; - for (i = length1 - 1; i >= 0; i--) { - array[customDest + i] = array[customCursor + i]; - } - array[dest] = tmp[cursor2]; - return; - } - var _minGallop = minGallop; - while (true) { - var count1 = 0; - var count2 = 0; - var exit = false; - do { - if (compare(tmp[cursor2], array[cursor1]) < 0) { - array[dest--] = array[cursor1--]; - count1++; - count2 = 0; - if (--length1 === 0) { - exit = true; - break; - } - } - else { - array[dest--] = tmp[cursor2--]; - count2++; - count1 = 0; - if (--length2 === 1) { - exit = true; - break; - } - } - } while ((count1 | count2) < _minGallop); - if (exit) { - break; - } - do { - count1 = length1 - gallopRight(tmp[cursor2], array, start1, length1, length1 - 1, compare); - if (count1 !== 0) { - dest -= count1; - cursor1 -= count1; - length1 -= count1; - customDest = dest + 1; - customCursor = cursor1 + 1; - for (i = count1 - 1; i >= 0; i--) { - array[customDest + i] = array[customCursor + i]; - } - if (length1 === 0) { - exit = true; - break; - } - } - array[dest--] = tmp[cursor2--]; - if (--length2 === 1) { - exit = true; - break; - } - count2 = length2 - gallopLeft(array[cursor1], tmp, 0, length2, length2 - 1, compare); - if (count2 !== 0) { - dest -= count2; - cursor2 -= count2; - length2 -= count2; - customDest = dest + 1; - customCursor = cursor2 + 1; - for (i = 0; i < count2; i++) { - array[customDest + i] = tmp[customCursor + i]; - } - if (length2 <= 1) { - exit = true; - break; - } - } - array[dest--] = array[cursor1--]; - if (--length1 === 0) { - exit = true; - break; - } - _minGallop--; - } while (count1 >= DEFAULT_MIN_GALLOPING || count2 >= DEFAULT_MIN_GALLOPING); - if (exit) { - break; - } - if (_minGallop < 0) { - _minGallop = 0; - } - _minGallop += 2; - } - minGallop = _minGallop; - if (minGallop < 1) { - minGallop = 1; - } - if (length2 === 1) { - dest -= length1; - cursor1 -= length1; - customDest = dest + 1; - customCursor = cursor1 + 1; - for (i = length1 - 1; i >= 0; i--) { - array[customDest + i] = array[customCursor + i]; - } - array[dest] = tmp[cursor2]; - } - else if (length2 === 0) { - throw new Error(); - } - else { - customCursor = dest - (length2 - 1); - for (i = 0; i < length2; i++) { - array[customCursor + i] = tmp[i]; - } - } - } - return { - mergeRuns: mergeRuns, - forceMergeRuns: forceMergeRuns, - pushRun: pushRun - }; - } - function sort(array, compare, lo, hi) { - if (!lo) { - lo = 0; - } - if (!hi) { - hi = array.length; - } - var remaining = hi - lo; - if (remaining < 2) { - return; - } - var runLength = 0; - if (remaining < DEFAULT_MIN_MERGE) { - runLength = makeAscendingRun(array, lo, hi, compare); - binaryInsertionSort(array, lo, hi, lo + runLength, compare); - return; - } - var ts = TimSort(array, compare); - var minRun = minRunLength(remaining); - do { - runLength = makeAscendingRun(array, lo, hi, compare); - if (runLength < minRun) { - var force = remaining; - if (force > minRun) { - force = minRun; - } - binaryInsertionSort(array, lo, lo + force, lo + runLength, compare); - runLength = force; - } - ts.pushRun(lo, runLength); - ts.mergeRuns(); - remaining -= runLength; - lo += runLength; - } while (remaining !== 0); - ts.forceMergeRuns(); - } - - var REDRAW_BIT = 1; - var STYLE_CHANGED_BIT = 2; - var SHAPE_CHANGED_BIT = 4; - - var invalidZErrorLogged = false; - function logInvalidZError() { - if (invalidZErrorLogged) { - return; - } - invalidZErrorLogged = true; - console.warn('z / z2 / zlevel of displayable is invalid, which may cause unexpected errors'); - } - function shapeCompareFunc(a, b) { - if (a.zlevel === b.zlevel) { - if (a.z === b.z) { - return a.z2 - b.z2; - } - return a.z - b.z; - } - return a.zlevel - b.zlevel; - } - var Storage = (function () { - function Storage() { - this._roots = []; - this._displayList = []; - this._displayListLen = 0; - this.displayableSortFunc = shapeCompareFunc; - } - Storage.prototype.traverse = function (cb, context) { - for (var i = 0; i < this._roots.length; i++) { - this._roots[i].traverse(cb, context); - } - }; - Storage.prototype.getDisplayList = function (update, includeIgnore) { - includeIgnore = includeIgnore || false; - var displayList = this._displayList; - if (update || !displayList.length) { - this.updateDisplayList(includeIgnore); - } - return displayList; - }; - Storage.prototype.updateDisplayList = function (includeIgnore) { - this._displayListLen = 0; - var roots = this._roots; - var displayList = this._displayList; - for (var i = 0, len = roots.length; i < len; i++) { - this._updateAndAddDisplayable(roots[i], null, includeIgnore); - } - displayList.length = this._displayListLen; - sort(displayList, shapeCompareFunc); - }; - Storage.prototype._updateAndAddDisplayable = function (el, clipPaths, includeIgnore) { - if (el.ignore && !includeIgnore) { - return; - } - el.beforeUpdate(); - el.update(); - el.afterUpdate(); - var userSetClipPath = el.getClipPath(); - if (el.ignoreClip) { - clipPaths = null; - } - else if (userSetClipPath) { - if (clipPaths) { - clipPaths = clipPaths.slice(); - } - else { - clipPaths = []; - } - var currentClipPath = userSetClipPath; - var parentClipPath = el; - while (currentClipPath) { - currentClipPath.parent = parentClipPath; - currentClipPath.updateTransform(); - clipPaths.push(currentClipPath); - parentClipPath = currentClipPath; - currentClipPath = currentClipPath.getClipPath(); - } - } - if (el.childrenRef) { - var children = el.childrenRef(); - for (var i = 0; i < children.length; i++) { - var child = children[i]; - if (el.__dirty) { - child.__dirty |= REDRAW_BIT; - } - this._updateAndAddDisplayable(child, clipPaths, includeIgnore); - } - el.__dirty = 0; - } - else { - var disp = el; - if (clipPaths && clipPaths.length) { - disp.__clipPaths = clipPaths; - } - else if (disp.__clipPaths && disp.__clipPaths.length > 0) { - disp.__clipPaths = []; - } - if (isNaN(disp.z)) { - logInvalidZError(); - disp.z = 0; - } - if (isNaN(disp.z2)) { - logInvalidZError(); - disp.z2 = 0; - } - if (isNaN(disp.zlevel)) { - logInvalidZError(); - disp.zlevel = 0; - } - this._displayList[this._displayListLen++] = disp; - } - var decalEl = el.getDecalElement && el.getDecalElement(); - if (decalEl) { - this._updateAndAddDisplayable(decalEl, clipPaths, includeIgnore); - } - var textGuide = el.getTextGuideLine(); - if (textGuide) { - this._updateAndAddDisplayable(textGuide, clipPaths, includeIgnore); - } - var textEl = el.getTextContent(); - if (textEl) { - this._updateAndAddDisplayable(textEl, clipPaths, includeIgnore); - } - }; - Storage.prototype.addRoot = function (el) { - if (el.__zr && el.__zr.storage === this) { - return; - } - this._roots.push(el); - }; - Storage.prototype.delRoot = function (el) { - if (el instanceof Array) { - for (var i = 0, l = el.length; i < l; i++) { - this.delRoot(el[i]); - } - return; - } - var idx = indexOf(this._roots, el); - if (idx >= 0) { - this._roots.splice(idx, 1); - } - }; - Storage.prototype.delAllRoots = function () { - this._roots = []; - this._displayList = []; - this._displayListLen = 0; - return; - }; - Storage.prototype.getRoots = function () { - return this._roots; - }; - Storage.prototype.dispose = function () { - this._displayList = null; - this._roots = null; - }; - return Storage; - }()); - - var requestAnimationFrame; - requestAnimationFrame = (env.hasGlobalWindow - && ((window.requestAnimationFrame && window.requestAnimationFrame.bind(window)) - || (window.msRequestAnimationFrame && window.msRequestAnimationFrame.bind(window)) - || window.mozRequestAnimationFrame - || window.webkitRequestAnimationFrame)) || function (func) { - return setTimeout(func, 16); - }; - var requestAnimationFrame$1 = requestAnimationFrame; - - var easingFuncs = { - linear: function (k) { - return k; - }, - quadraticIn: function (k) { - return k * k; - }, - quadraticOut: function (k) { - return k * (2 - k); - }, - quadraticInOut: function (k) { - if ((k *= 2) < 1) { - return 0.5 * k * k; - } - return -0.5 * (--k * (k - 2) - 1); - }, - cubicIn: function (k) { - return k * k * k; - }, - cubicOut: function (k) { - return --k * k * k + 1; - }, - cubicInOut: function (k) { - if ((k *= 2) < 1) { - return 0.5 * k * k * k; - } - return 0.5 * ((k -= 2) * k * k + 2); - }, - quarticIn: function (k) { - return k * k * k * k; - }, - quarticOut: function (k) { - return 1 - (--k * k * k * k); - }, - quarticInOut: function (k) { - if ((k *= 2) < 1) { - return 0.5 * k * k * k * k; - } - return -0.5 * ((k -= 2) * k * k * k - 2); - }, - quinticIn: function (k) { - return k * k * k * k * k; - }, - quinticOut: function (k) { - return --k * k * k * k * k + 1; - }, - quinticInOut: function (k) { - if ((k *= 2) < 1) { - return 0.5 * k * k * k * k * k; - } - return 0.5 * ((k -= 2) * k * k * k * k + 2); - }, - sinusoidalIn: function (k) { - return 1 - Math.cos(k * Math.PI / 2); - }, - sinusoidalOut: function (k) { - return Math.sin(k * Math.PI / 2); - }, - sinusoidalInOut: function (k) { - return 0.5 * (1 - Math.cos(Math.PI * k)); - }, - exponentialIn: function (k) { - return k === 0 ? 0 : Math.pow(1024, k - 1); - }, - exponentialOut: function (k) { - return k === 1 ? 1 : 1 - Math.pow(2, -10 * k); - }, - exponentialInOut: function (k) { - if (k === 0) { - return 0; - } - if (k === 1) { - return 1; - } - if ((k *= 2) < 1) { - return 0.5 * Math.pow(1024, k - 1); - } - return 0.5 * (-Math.pow(2, -10 * (k - 1)) + 2); - }, - circularIn: function (k) { - return 1 - Math.sqrt(1 - k * k); - }, - circularOut: function (k) { - return Math.sqrt(1 - (--k * k)); - }, - circularInOut: function (k) { - if ((k *= 2) < 1) { - return -0.5 * (Math.sqrt(1 - k * k) - 1); - } - return 0.5 * (Math.sqrt(1 - (k -= 2) * k) + 1); - }, - elasticIn: function (k) { - var s; - var a = 0.1; - var p = 0.4; - if (k === 0) { - return 0; - } - if (k === 1) { - return 1; - } - if (!a || a < 1) { - a = 1; - s = p / 4; - } - else { - s = p * Math.asin(1 / a) / (2 * Math.PI); - } - return -(a * Math.pow(2, 10 * (k -= 1)) - * Math.sin((k - s) * (2 * Math.PI) / p)); - }, - elasticOut: function (k) { - var s; - var a = 0.1; - var p = 0.4; - if (k === 0) { - return 0; - } - if (k === 1) { - return 1; - } - if (!a || a < 1) { - a = 1; - s = p / 4; - } - else { - s = p * Math.asin(1 / a) / (2 * Math.PI); - } - return (a * Math.pow(2, -10 * k) - * Math.sin((k - s) * (2 * Math.PI) / p) + 1); - }, - elasticInOut: function (k) { - var s; - var a = 0.1; - var p = 0.4; - if (k === 0) { - return 0; - } - if (k === 1) { - return 1; - } - if (!a || a < 1) { - a = 1; - s = p / 4; - } - else { - s = p * Math.asin(1 / a) / (2 * Math.PI); - } - if ((k *= 2) < 1) { - return -0.5 * (a * Math.pow(2, 10 * (k -= 1)) - * Math.sin((k - s) * (2 * Math.PI) / p)); - } - return a * Math.pow(2, -10 * (k -= 1)) - * Math.sin((k - s) * (2 * Math.PI) / p) * 0.5 + 1; - }, - backIn: function (k) { - var s = 1.70158; - return k * k * ((s + 1) * k - s); - }, - backOut: function (k) { - var s = 1.70158; - return --k * k * ((s + 1) * k + s) + 1; - }, - backInOut: function (k) { - var s = 1.70158 * 1.525; - if ((k *= 2) < 1) { - return 0.5 * (k * k * ((s + 1) * k - s)); - } - return 0.5 * ((k -= 2) * k * ((s + 1) * k + s) + 2); - }, - bounceIn: function (k) { - return 1 - easingFuncs.bounceOut(1 - k); - }, - bounceOut: function (k) { - if (k < (1 / 2.75)) { - return 7.5625 * k * k; - } - else if (k < (2 / 2.75)) { - return 7.5625 * (k -= (1.5 / 2.75)) * k + 0.75; - } - else if (k < (2.5 / 2.75)) { - return 7.5625 * (k -= (2.25 / 2.75)) * k + 0.9375; - } - else { - return 7.5625 * (k -= (2.625 / 2.75)) * k + 0.984375; - } - }, - bounceInOut: function (k) { - if (k < 0.5) { - return easingFuncs.bounceIn(k * 2) * 0.5; - } - return easingFuncs.bounceOut(k * 2 - 1) * 0.5 + 0.5; - } - }; - - var mathPow = Math.pow; - var mathSqrt = Math.sqrt; - var EPSILON = 1e-8; - var EPSILON_NUMERIC = 1e-4; - var THREE_SQRT = mathSqrt(3); - var ONE_THIRD = 1 / 3; - var _v0 = create(); - var _v1 = create(); - var _v2 = create(); - function isAroundZero(val) { - return val > -EPSILON && val < EPSILON; - } - function isNotAroundZero(val) { - return val > EPSILON || val < -EPSILON; - } - function cubicAt(p0, p1, p2, p3, t) { - var onet = 1 - t; - return onet * onet * (onet * p0 + 3 * t * p1) - + t * t * (t * p3 + 3 * onet * p2); - } - function cubicDerivativeAt(p0, p1, p2, p3, t) { - var onet = 1 - t; - return 3 * (((p1 - p0) * onet + 2 * (p2 - p1) * t) * onet - + (p3 - p2) * t * t); - } - function cubicRootAt(p0, p1, p2, p3, val, roots) { - var a = p3 + 3 * (p1 - p2) - p0; - var b = 3 * (p2 - p1 * 2 + p0); - var c = 3 * (p1 - p0); - var d = p0 - val; - var A = b * b - 3 * a * c; - var B = b * c - 9 * a * d; - var C = c * c - 3 * b * d; - var n = 0; - if (isAroundZero(A) && isAroundZero(B)) { - if (isAroundZero(b)) { - roots[0] = 0; - } - else { - var t1 = -c / b; - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - } - } - else { - var disc = B * B - 4 * A * C; - if (isAroundZero(disc)) { - var K = B / A; - var t1 = -b / a + K; - var t2 = -K / 2; - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - if (t2 >= 0 && t2 <= 1) { - roots[n++] = t2; - } - } - else if (disc > 0) { - var discSqrt = mathSqrt(disc); - var Y1 = A * b + 1.5 * a * (-B + discSqrt); - var Y2 = A * b + 1.5 * a * (-B - discSqrt); - if (Y1 < 0) { - Y1 = -mathPow(-Y1, ONE_THIRD); - } - else { - Y1 = mathPow(Y1, ONE_THIRD); - } - if (Y2 < 0) { - Y2 = -mathPow(-Y2, ONE_THIRD); - } - else { - Y2 = mathPow(Y2, ONE_THIRD); - } - var t1 = (-b - (Y1 + Y2)) / (3 * a); - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - } - else { - var T = (2 * A * b - 3 * a * B) / (2 * mathSqrt(A * A * A)); - var theta = Math.acos(T) / 3; - var ASqrt = mathSqrt(A); - var tmp = Math.cos(theta); - var t1 = (-b - 2 * ASqrt * tmp) / (3 * a); - var t2 = (-b + ASqrt * (tmp + THREE_SQRT * Math.sin(theta))) / (3 * a); - var t3 = (-b + ASqrt * (tmp - THREE_SQRT * Math.sin(theta))) / (3 * a); - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - if (t2 >= 0 && t2 <= 1) { - roots[n++] = t2; - } - if (t3 >= 0 && t3 <= 1) { - roots[n++] = t3; - } - } - } - return n; - } - function cubicExtrema(p0, p1, p2, p3, extrema) { - var b = 6 * p2 - 12 * p1 + 6 * p0; - var a = 9 * p1 + 3 * p3 - 3 * p0 - 9 * p2; - var c = 3 * p1 - 3 * p0; - var n = 0; - if (isAroundZero(a)) { - if (isNotAroundZero(b)) { - var t1 = -c / b; - if (t1 >= 0 && t1 <= 1) { - extrema[n++] = t1; - } - } - } - else { - var disc = b * b - 4 * a * c; - if (isAroundZero(disc)) { - extrema[0] = -b / (2 * a); - } - else if (disc > 0) { - var discSqrt = mathSqrt(disc); - var t1 = (-b + discSqrt) / (2 * a); - var t2 = (-b - discSqrt) / (2 * a); - if (t1 >= 0 && t1 <= 1) { - extrema[n++] = t1; - } - if (t2 >= 0 && t2 <= 1) { - extrema[n++] = t2; - } - } - } - return n; - } - function cubicSubdivide(p0, p1, p2, p3, t, out) { - var p01 = (p1 - p0) * t + p0; - var p12 = (p2 - p1) * t + p1; - var p23 = (p3 - p2) * t + p2; - var p012 = (p12 - p01) * t + p01; - var p123 = (p23 - p12) * t + p12; - var p0123 = (p123 - p012) * t + p012; - out[0] = p0; - out[1] = p01; - out[2] = p012; - out[3] = p0123; - out[4] = p0123; - out[5] = p123; - out[6] = p23; - out[7] = p3; - } - function cubicProjectPoint(x0, y0, x1, y1, x2, y2, x3, y3, x, y, out) { - var t; - var interval = 0.005; - var d = Infinity; - var prev; - var next; - var d1; - var d2; - _v0[0] = x; - _v0[1] = y; - for (var _t = 0; _t < 1; _t += 0.05) { - _v1[0] = cubicAt(x0, x1, x2, x3, _t); - _v1[1] = cubicAt(y0, y1, y2, y3, _t); - d1 = distSquare(_v0, _v1); - if (d1 < d) { - t = _t; - d = d1; - } - } - d = Infinity; - for (var i = 0; i < 32; i++) { - if (interval < EPSILON_NUMERIC) { - break; - } - prev = t - interval; - next = t + interval; - _v1[0] = cubicAt(x0, x1, x2, x3, prev); - _v1[1] = cubicAt(y0, y1, y2, y3, prev); - d1 = distSquare(_v1, _v0); - if (prev >= 0 && d1 < d) { - t = prev; - d = d1; - } - else { - _v2[0] = cubicAt(x0, x1, x2, x3, next); - _v2[1] = cubicAt(y0, y1, y2, y3, next); - d2 = distSquare(_v2, _v0); - if (next <= 1 && d2 < d) { - t = next; - d = d2; - } - else { - interval *= 0.5; - } - } - } - if (out) { - out[0] = cubicAt(x0, x1, x2, x3, t); - out[1] = cubicAt(y0, y1, y2, y3, t); - } - return mathSqrt(d); - } - function cubicLength(x0, y0, x1, y1, x2, y2, x3, y3, iteration) { - var px = x0; - var py = y0; - var d = 0; - var step = 1 / iteration; - for (var i = 1; i <= iteration; i++) { - var t = i * step; - var x = cubicAt(x0, x1, x2, x3, t); - var y = cubicAt(y0, y1, y2, y3, t); - var dx = x - px; - var dy = y - py; - d += Math.sqrt(dx * dx + dy * dy); - px = x; - py = y; - } - return d; - } - function quadraticAt(p0, p1, p2, t) { - var onet = 1 - t; - return onet * (onet * p0 + 2 * t * p1) + t * t * p2; - } - function quadraticDerivativeAt(p0, p1, p2, t) { - return 2 * ((1 - t) * (p1 - p0) + t * (p2 - p1)); - } - function quadraticRootAt(p0, p1, p2, val, roots) { - var a = p0 - 2 * p1 + p2; - var b = 2 * (p1 - p0); - var c = p0 - val; - var n = 0; - if (isAroundZero(a)) { - if (isNotAroundZero(b)) { - var t1 = -c / b; - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - } - } - else { - var disc = b * b - 4 * a * c; - if (isAroundZero(disc)) { - var t1 = -b / (2 * a); - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - } - else if (disc > 0) { - var discSqrt = mathSqrt(disc); - var t1 = (-b + discSqrt) / (2 * a); - var t2 = (-b - discSqrt) / (2 * a); - if (t1 >= 0 && t1 <= 1) { - roots[n++] = t1; - } - if (t2 >= 0 && t2 <= 1) { - roots[n++] = t2; - } - } - } - return n; - } - function quadraticExtremum(p0, p1, p2) { - var divider = p0 + p2 - 2 * p1; - if (divider === 0) { - return 0.5; - } - else { - return (p0 - p1) / divider; - } - } - function quadraticSubdivide(p0, p1, p2, t, out) { - var p01 = (p1 - p0) * t + p0; - var p12 = (p2 - p1) * t + p1; - var p012 = (p12 - p01) * t + p01; - out[0] = p0; - out[1] = p01; - out[2] = p012; - out[3] = p012; - out[4] = p12; - out[5] = p2; - } - function quadraticProjectPoint(x0, y0, x1, y1, x2, y2, x, y, out) { - var t; - var interval = 0.005; - var d = Infinity; - _v0[0] = x; - _v0[1] = y; - for (var _t = 0; _t < 1; _t += 0.05) { - _v1[0] = quadraticAt(x0, x1, x2, _t); - _v1[1] = quadraticAt(y0, y1, y2, _t); - var d1 = distSquare(_v0, _v1); - if (d1 < d) { - t = _t; - d = d1; - } - } - d = Infinity; - for (var i = 0; i < 32; i++) { - if (interval < EPSILON_NUMERIC) { - break; - } - var prev = t - interval; - var next = t + interval; - _v1[0] = quadraticAt(x0, x1, x2, prev); - _v1[1] = quadraticAt(y0, y1, y2, prev); - var d1 = distSquare(_v1, _v0); - if (prev >= 0 && d1 < d) { - t = prev; - d = d1; - } - else { - _v2[0] = quadraticAt(x0, x1, x2, next); - _v2[1] = quadraticAt(y0, y1, y2, next); - var d2 = distSquare(_v2, _v0); - if (next <= 1 && d2 < d) { - t = next; - d = d2; - } - else { - interval *= 0.5; - } - } - } - if (out) { - out[0] = quadraticAt(x0, x1, x2, t); - out[1] = quadraticAt(y0, y1, y2, t); - } - return mathSqrt(d); - } - function quadraticLength(x0, y0, x1, y1, x2, y2, iteration) { - var px = x0; - var py = y0; - var d = 0; - var step = 1 / iteration; - for (var i = 1; i <= iteration; i++) { - var t = i * step; - var x = quadraticAt(x0, x1, x2, t); - var y = quadraticAt(y0, y1, y2, t); - var dx = x - px; - var dy = y - py; - d += Math.sqrt(dx * dx + dy * dy); - px = x; - py = y; - } - return d; - } - - var regexp = /cubic-bezier\(([0-9,\.e ]+)\)/; - function createCubicEasingFunc(cubicEasingStr) { - var cubic = cubicEasingStr && regexp.exec(cubicEasingStr); - if (cubic) { - var points = cubic[1].split(','); - var a_1 = +trim(points[0]); - var b_1 = +trim(points[1]); - var c_1 = +trim(points[2]); - var d_1 = +trim(points[3]); - if (isNaN(a_1 + b_1 + c_1 + d_1)) { - return; - } - var roots_1 = []; - return function (p) { - return p <= 0 - ? 0 : p >= 1 - ? 1 - : cubicRootAt(0, a_1, c_1, 1, p, roots_1) && cubicAt(0, b_1, d_1, 1, roots_1[0]); - }; - } - } - - var Clip = (function () { - function Clip(opts) { - this._inited = false; - this._startTime = 0; - this._pausedTime = 0; - this._paused = false; - this._life = opts.life || 1000; - this._delay = opts.delay || 0; - this.loop = opts.loop || false; - this.onframe = opts.onframe || noop; - this.ondestroy = opts.ondestroy || noop; - this.onrestart = opts.onrestart || noop; - opts.easing && this.setEasing(opts.easing); - } - Clip.prototype.step = function (globalTime, deltaTime) { - if (!this._inited) { - this._startTime = globalTime + this._delay; - this._inited = true; - } - if (this._paused) { - this._pausedTime += deltaTime; - return; - } - var life = this._life; - var elapsedTime = globalTime - this._startTime - this._pausedTime; - var percent = elapsedTime / life; - if (percent < 0) { - percent = 0; - } - percent = Math.min(percent, 1); - var easingFunc = this.easingFunc; - var schedule = easingFunc ? easingFunc(percent) : percent; - this.onframe(schedule); - if (percent === 1) { - if (this.loop) { - var remainder = elapsedTime % life; - this._startTime = globalTime - remainder; - this._pausedTime = 0; - this.onrestart(); - } - else { - return true; - } - } - return false; - }; - Clip.prototype.pause = function () { - this._paused = true; - }; - Clip.prototype.resume = function () { - this._paused = false; - }; - Clip.prototype.setEasing = function (easing) { - this.easing = easing; - this.easingFunc = isFunction(easing) - ? easing - : easingFuncs[easing] || createCubicEasingFunc(easing); - }; - return Clip; - }()); - - var Entry = (function () { - function Entry(val) { - this.value = val; - } - return Entry; - }()); - var LinkedList = (function () { - function LinkedList() { - this._len = 0; - } - LinkedList.prototype.insert = function (val) { - var entry = new Entry(val); - this.insertEntry(entry); - return entry; - }; - LinkedList.prototype.insertEntry = function (entry) { - if (!this.head) { - this.head = this.tail = entry; - } - else { - this.tail.next = entry; - entry.prev = this.tail; - entry.next = null; - this.tail = entry; - } - this._len++; - }; - LinkedList.prototype.remove = function (entry) { - var prev = entry.prev; - var next = entry.next; - if (prev) { - prev.next = next; - } - else { - this.head = next; - } - if (next) { - next.prev = prev; - } - else { - this.tail = prev; - } - entry.next = entry.prev = null; - this._len--; - }; - LinkedList.prototype.len = function () { - return this._len; - }; - LinkedList.prototype.clear = function () { - this.head = this.tail = null; - this._len = 0; - }; - return LinkedList; - }()); - var LRU = (function () { - function LRU(maxSize) { - this._list = new LinkedList(); - this._maxSize = 10; - this._map = {}; - this._maxSize = maxSize; - } - LRU.prototype.put = function (key, value) { - var list = this._list; - var map = this._map; - var removed = null; - if (map[key] == null) { - var len = list.len(); - var entry = this._lastRemovedEntry; - if (len >= this._maxSize && len > 0) { - var leastUsedEntry = list.head; - list.remove(leastUsedEntry); - delete map[leastUsedEntry.key]; - removed = leastUsedEntry.value; - this._lastRemovedEntry = leastUsedEntry; - } - if (entry) { - entry.value = value; - } - else { - entry = new Entry(value); - } - entry.key = key; - list.insertEntry(entry); - map[key] = entry; - } - return removed; - }; - LRU.prototype.get = function (key) { - var entry = this._map[key]; - var list = this._list; - if (entry != null) { - if (entry !== list.tail) { - list.remove(entry); - list.insertEntry(entry); - } - return entry.value; - } - }; - LRU.prototype.clear = function () { - this._list.clear(); - this._map = {}; - }; - LRU.prototype.len = function () { - return this._list.len(); - }; - return LRU; - }()); - - var kCSSColorTable = { - 'transparent': [0, 0, 0, 0], 'aliceblue': [240, 248, 255, 1], - 'antiquewhite': [250, 235, 215, 1], 'aqua': [0, 255, 255, 1], - 'aquamarine': [127, 255, 212, 1], 'azure': [240, 255, 255, 1], - 'beige': [245, 245, 220, 1], 'bisque': [255, 228, 196, 1], - 'black': [0, 0, 0, 1], 'blanchedalmond': [255, 235, 205, 1], - 'blue': [0, 0, 255, 1], 'blueviolet': [138, 43, 226, 1], - 'brown': [165, 42, 42, 1], 'burlywood': [222, 184, 135, 1], - 'cadetblue': [95, 158, 160, 1], 'chartreuse': [127, 255, 0, 1], - 'chocolate': [210, 105, 30, 1], 'coral': [255, 127, 80, 1], - 'cornflowerblue': [100, 149, 237, 1], 'cornsilk': [255, 248, 220, 1], - 'crimson': [220, 20, 60, 1], 'cyan': [0, 255, 255, 1], - 'darkblue': [0, 0, 139, 1], 'darkcyan': [0, 139, 139, 1], - 'darkgoldenrod': [184, 134, 11, 1], 'darkgray': [169, 169, 169, 1], - 'darkgreen': [0, 100, 0, 1], 'darkgrey': [169, 169, 169, 1], - 'darkkhaki': [189, 183, 107, 1], 'darkmagenta': [139, 0, 139, 1], - 'darkolivegreen': [85, 107, 47, 1], 'darkorange': [255, 140, 0, 1], - 'darkorchid': [153, 50, 204, 1], 'darkred': [139, 0, 0, 1], - 'darksalmon': [233, 150, 122, 1], 'darkseagreen': [143, 188, 143, 1], - 'darkslateblue': [72, 61, 139, 1], 'darkslategray': [47, 79, 79, 1], - 'darkslategrey': [47, 79, 79, 1], 'darkturquoise': [0, 206, 209, 1], - 'darkviolet': [148, 0, 211, 1], 'deeppink': [255, 20, 147, 1], - 'deepskyblue': [0, 191, 255, 1], 'dimgray': [105, 105, 105, 1], - 'dimgrey': [105, 105, 105, 1], 'dodgerblue': [30, 144, 255, 1], - 'firebrick': [178, 34, 34, 1], 'floralwhite': [255, 250, 240, 1], - 'forestgreen': [34, 139, 34, 1], 'fuchsia': [255, 0, 255, 1], - 'gainsboro': [220, 220, 220, 1], 'ghostwhite': [248, 248, 255, 1], - 'gold': [255, 215, 0, 1], 'goldenrod': [218, 165, 32, 1], - 'gray': [128, 128, 128, 1], 'green': [0, 128, 0, 1], - 'greenyellow': [173, 255, 47, 1], 'grey': [128, 128, 128, 1], - 'honeydew': [240, 255, 240, 1], 'hotpink': [255, 105, 180, 1], - 'indianred': [205, 92, 92, 1], 'indigo': [75, 0, 130, 1], - 'ivory': [255, 255, 240, 1], 'khaki': [240, 230, 140, 1], - 'lavender': [230, 230, 250, 1], 'lavenderblush': [255, 240, 245, 1], - 'lawngreen': [124, 252, 0, 1], 'lemonchiffon': [255, 250, 205, 1], - 'lightblue': [173, 216, 230, 1], 'lightcoral': [240, 128, 128, 1], - 'lightcyan': [224, 255, 255, 1], 'lightgoldenrodyellow': [250, 250, 210, 1], - 'lightgray': [211, 211, 211, 1], 'lightgreen': [144, 238, 144, 1], - 'lightgrey': [211, 211, 211, 1], 'lightpink': [255, 182, 193, 1], - 'lightsalmon': [255, 160, 122, 1], 'lightseagreen': [32, 178, 170, 1], - 'lightskyblue': [135, 206, 250, 1], 'lightslategray': [119, 136, 153, 1], - 'lightslategrey': [119, 136, 153, 1], 'lightsteelblue': [176, 196, 222, 1], - 'lightyellow': [255, 255, 224, 1], 'lime': [0, 255, 0, 1], - 'limegreen': [50, 205, 50, 1], 'linen': [250, 240, 230, 1], - 'magenta': [255, 0, 255, 1], 'maroon': [128, 0, 0, 1], - 'mediumaquamarine': [102, 205, 170, 1], 'mediumblue': [0, 0, 205, 1], - 'mediumorchid': [186, 85, 211, 1], 'mediumpurple': [147, 112, 219, 1], - 'mediumseagreen': [60, 179, 113, 1], 'mediumslateblue': [123, 104, 238, 1], - 'mediumspringgreen': [0, 250, 154, 1], 'mediumturquoise': [72, 209, 204, 1], - 'mediumvioletred': [199, 21, 133, 1], 'midnightblue': [25, 25, 112, 1], - 'mintcream': [245, 255, 250, 1], 'mistyrose': [255, 228, 225, 1], - 'moccasin': [255, 228, 181, 1], 'navajowhite': [255, 222, 173, 1], - 'navy': [0, 0, 128, 1], 'oldlace': [253, 245, 230, 1], - 'olive': [128, 128, 0, 1], 'olivedrab': [107, 142, 35, 1], - 'orange': [255, 165, 0, 1], 'orangered': [255, 69, 0, 1], - 'orchid': [218, 112, 214, 1], 'palegoldenrod': [238, 232, 170, 1], - 'palegreen': [152, 251, 152, 1], 'paleturquoise': [175, 238, 238, 1], - 'palevioletred': [219, 112, 147, 1], 'papayawhip': [255, 239, 213, 1], - 'peachpuff': [255, 218, 185, 1], 'peru': [205, 133, 63, 1], - 'pink': [255, 192, 203, 1], 'plum': [221, 160, 221, 1], - 'powderblue': [176, 224, 230, 1], 'purple': [128, 0, 128, 1], - 'red': [255, 0, 0, 1], 'rosybrown': [188, 143, 143, 1], - 'royalblue': [65, 105, 225, 1], 'saddlebrown': [139, 69, 19, 1], - 'salmon': [250, 128, 114, 1], 'sandybrown': [244, 164, 96, 1], - 'seagreen': [46, 139, 87, 1], 'seashell': [255, 245, 238, 1], - 'sienna': [160, 82, 45, 1], 'silver': [192, 192, 192, 1], - 'skyblue': [135, 206, 235, 1], 'slateblue': [106, 90, 205, 1], - 'slategray': [112, 128, 144, 1], 'slategrey': [112, 128, 144, 1], - 'snow': [255, 250, 250, 1], 'springgreen': [0, 255, 127, 1], - 'steelblue': [70, 130, 180, 1], 'tan': [210, 180, 140, 1], - 'teal': [0, 128, 128, 1], 'thistle': [216, 191, 216, 1], - 'tomato': [255, 99, 71, 1], 'turquoise': [64, 224, 208, 1], - 'violet': [238, 130, 238, 1], 'wheat': [245, 222, 179, 1], - 'white': [255, 255, 255, 1], 'whitesmoke': [245, 245, 245, 1], - 'yellow': [255, 255, 0, 1], 'yellowgreen': [154, 205, 50, 1] - }; - function clampCssByte(i) { - i = Math.round(i); - return i < 0 ? 0 : i > 255 ? 255 : i; - } - function clampCssAngle(i) { - i = Math.round(i); - return i < 0 ? 0 : i > 360 ? 360 : i; - } - function clampCssFloat(f) { - return f < 0 ? 0 : f > 1 ? 1 : f; - } - function parseCssInt(val) { - var str = val; - if (str.length && str.charAt(str.length - 1) === '%') { - return clampCssByte(parseFloat(str) / 100 * 255); - } - return clampCssByte(parseInt(str, 10)); - } - function parseCssFloat(val) { - var str = val; - if (str.length && str.charAt(str.length - 1) === '%') { - return clampCssFloat(parseFloat(str) / 100); - } - return clampCssFloat(parseFloat(str)); - } - function cssHueToRgb(m1, m2, h) { - if (h < 0) { - h += 1; - } - else if (h > 1) { - h -= 1; - } - if (h * 6 < 1) { - return m1 + (m2 - m1) * h * 6; - } - if (h * 2 < 1) { - return m2; - } - if (h * 3 < 2) { - return m1 + (m2 - m1) * (2 / 3 - h) * 6; - } - return m1; - } - function lerpNumber(a, b, p) { - return a + (b - a) * p; - } - function setRgba(out, r, g, b, a) { - out[0] = r; - out[1] = g; - out[2] = b; - out[3] = a; - return out; - } - function copyRgba(out, a) { - out[0] = a[0]; - out[1] = a[1]; - out[2] = a[2]; - out[3] = a[3]; - return out; - } - var colorCache = new LRU(20); - var lastRemovedArr = null; - function putToCache(colorStr, rgbaArr) { - if (lastRemovedArr) { - copyRgba(lastRemovedArr, rgbaArr); - } - lastRemovedArr = colorCache.put(colorStr, lastRemovedArr || (rgbaArr.slice())); - } - function parse(colorStr, rgbaArr) { - if (!colorStr) { - return; - } - rgbaArr = rgbaArr || []; - var cached = colorCache.get(colorStr); - if (cached) { - return copyRgba(rgbaArr, cached); - } - colorStr = colorStr + ''; - var str = colorStr.replace(/ /g, '').toLowerCase(); - if (str in kCSSColorTable) { - copyRgba(rgbaArr, kCSSColorTable[str]); - putToCache(colorStr, rgbaArr); - return rgbaArr; - } - var strLen = str.length; - if (str.charAt(0) === '#') { - if (strLen === 4 || strLen === 5) { - var iv = parseInt(str.slice(1, 4), 16); - if (!(iv >= 0 && iv <= 0xfff)) { - setRgba(rgbaArr, 0, 0, 0, 1); - return; - } - setRgba(rgbaArr, ((iv & 0xf00) >> 4) | ((iv & 0xf00) >> 8), (iv & 0xf0) | ((iv & 0xf0) >> 4), (iv & 0xf) | ((iv & 0xf) << 4), strLen === 5 ? parseInt(str.slice(4), 16) / 0xf : 1); - putToCache(colorStr, rgbaArr); - return rgbaArr; - } - else if (strLen === 7 || strLen === 9) { - var iv = parseInt(str.slice(1, 7), 16); - if (!(iv >= 0 && iv <= 0xffffff)) { - setRgba(rgbaArr, 0, 0, 0, 1); - return; - } - setRgba(rgbaArr, (iv & 0xff0000) >> 16, (iv & 0xff00) >> 8, iv & 0xff, strLen === 9 ? parseInt(str.slice(7), 16) / 0xff : 1); - putToCache(colorStr, rgbaArr); - return rgbaArr; - } - return; - } - var op = str.indexOf('('); - var ep = str.indexOf(')'); - if (op !== -1 && ep + 1 === strLen) { - var fname = str.substr(0, op); - var params = str.substr(op + 1, ep - (op + 1)).split(','); - var alpha = 1; - switch (fname) { - case 'rgba': - if (params.length !== 4) { - return params.length === 3 - ? setRgba(rgbaArr, +params[0], +params[1], +params[2], 1) - : setRgba(rgbaArr, 0, 0, 0, 1); - } - alpha = parseCssFloat(params.pop()); - case 'rgb': - if (params.length >= 3) { - setRgba(rgbaArr, parseCssInt(params[0]), parseCssInt(params[1]), parseCssInt(params[2]), params.length === 3 ? alpha : parseCssFloat(params[3])); - putToCache(colorStr, rgbaArr); - return rgbaArr; - } - else { - setRgba(rgbaArr, 0, 0, 0, 1); - return; - } - case 'hsla': - if (params.length !== 4) { - setRgba(rgbaArr, 0, 0, 0, 1); - return; - } - params[3] = parseCssFloat(params[3]); - hsla2rgba(params, rgbaArr); - putToCache(colorStr, rgbaArr); - return rgbaArr; - case 'hsl': - if (params.length !== 3) { - setRgba(rgbaArr, 0, 0, 0, 1); - return; - } - hsla2rgba(params, rgbaArr); - putToCache(colorStr, rgbaArr); - return rgbaArr; - default: - return; - } - } - setRgba(rgbaArr, 0, 0, 0, 1); - return; - } - function hsla2rgba(hsla, rgba) { - var h = (((parseFloat(hsla[0]) % 360) + 360) % 360) / 360; - var s = parseCssFloat(hsla[1]); - var l = parseCssFloat(hsla[2]); - var m2 = l <= 0.5 ? l * (s + 1) : l + s - l * s; - var m1 = l * 2 - m2; - rgba = rgba || []; - setRgba(rgba, clampCssByte(cssHueToRgb(m1, m2, h + 1 / 3) * 255), clampCssByte(cssHueToRgb(m1, m2, h) * 255), clampCssByte(cssHueToRgb(m1, m2, h - 1 / 3) * 255), 1); - if (hsla.length === 4) { - rgba[3] = hsla[3]; - } - return rgba; - } - function rgba2hsla(rgba) { - if (!rgba) { - return; - } - var R = rgba[0] / 255; - var G = rgba[1] / 255; - var B = rgba[2] / 255; - var vMin = Math.min(R, G, B); - var vMax = Math.max(R, G, B); - var delta = vMax - vMin; - var L = (vMax + vMin) / 2; - var H; - var S; - if (delta === 0) { - H = 0; - S = 0; - } - else { - if (L < 0.5) { - S = delta / (vMax + vMin); - } - else { - S = delta / (2 - vMax - vMin); - } - var deltaR = (((vMax - R) / 6) + (delta / 2)) / delta; - var deltaG = (((vMax - G) / 6) + (delta / 2)) / delta; - var deltaB = (((vMax - B) / 6) + (delta / 2)) / delta; - if (R === vMax) { - H = deltaB - deltaG; - } - else if (G === vMax) { - H = (1 / 3) + deltaR - deltaB; - } - else if (B === vMax) { - H = (2 / 3) + deltaG - deltaR; - } - if (H < 0) { - H += 1; - } - if (H > 1) { - H -= 1; - } - } - var hsla = [H * 360, S, L]; - if (rgba[3] != null) { - hsla.push(rgba[3]); - } - return hsla; - } - function lift(color, level) { - var colorArr = parse(color); - if (colorArr) { - for (var i = 0; i < 3; i++) { - if (level < 0) { - colorArr[i] = colorArr[i] * (1 - level) | 0; - } - else { - colorArr[i] = ((255 - colorArr[i]) * level + colorArr[i]) | 0; - } - if (colorArr[i] > 255) { - colorArr[i] = 255; - } - else if (colorArr[i] < 0) { - colorArr[i] = 0; - } - } - return stringify(colorArr, colorArr.length === 4 ? 'rgba' : 'rgb'); - } - } - function toHex(color) { - var colorArr = parse(color); - if (colorArr) { - return ((1 << 24) + (colorArr[0] << 16) + (colorArr[1] << 8) + (+colorArr[2])).toString(16).slice(1); - } - } - function fastLerp(normalizedValue, colors, out) { - if (!(colors && colors.length) - || !(normalizedValue >= 0 && normalizedValue <= 1)) { - return; - } - out = out || []; - var value = normalizedValue * (colors.length - 1); - var leftIndex = Math.floor(value); - var rightIndex = Math.ceil(value); - var leftColor = colors[leftIndex]; - var rightColor = colors[rightIndex]; - var dv = value - leftIndex; - out[0] = clampCssByte(lerpNumber(leftColor[0], rightColor[0], dv)); - out[1] = clampCssByte(lerpNumber(leftColor[1], rightColor[1], dv)); - out[2] = clampCssByte(lerpNumber(leftColor[2], rightColor[2], dv)); - out[3] = clampCssFloat(lerpNumber(leftColor[3], rightColor[3], dv)); - return out; - } - var fastMapToColor = fastLerp; - function lerp$1(normalizedValue, colors, fullOutput) { - if (!(colors && colors.length) - || !(normalizedValue >= 0 && normalizedValue <= 1)) { - return; - } - var value = normalizedValue * (colors.length - 1); - var leftIndex = Math.floor(value); - var rightIndex = Math.ceil(value); - var leftColor = parse(colors[leftIndex]); - var rightColor = parse(colors[rightIndex]); - var dv = value - leftIndex; - var color = stringify([ - clampCssByte(lerpNumber(leftColor[0], rightColor[0], dv)), - clampCssByte(lerpNumber(leftColor[1], rightColor[1], dv)), - clampCssByte(lerpNumber(leftColor[2], rightColor[2], dv)), - clampCssFloat(lerpNumber(leftColor[3], rightColor[3], dv)) - ], 'rgba'); - return fullOutput - ? { - color: color, - leftIndex: leftIndex, - rightIndex: rightIndex, - value: value - } - : color; - } - var mapToColor = lerp$1; - function modifyHSL(color, h, s, l) { - var colorArr = parse(color); - if (color) { - colorArr = rgba2hsla(colorArr); - h != null && (colorArr[0] = clampCssAngle(h)); - s != null && (colorArr[1] = parseCssFloat(s)); - l != null && (colorArr[2] = parseCssFloat(l)); - return stringify(hsla2rgba(colorArr), 'rgba'); - } - } - function modifyAlpha(color, alpha) { - var colorArr = parse(color); - if (colorArr && alpha != null) { - colorArr[3] = clampCssFloat(alpha); - return stringify(colorArr, 'rgba'); - } - } - function stringify(arrColor, type) { - if (!arrColor || !arrColor.length) { - return; - } - var colorStr = arrColor[0] + ',' + arrColor[1] + ',' + arrColor[2]; - if (type === 'rgba' || type === 'hsva' || type === 'hsla') { - colorStr += ',' + arrColor[3]; - } - return type + '(' + colorStr + ')'; - } - function lum(color, backgroundLum) { - var arr = parse(color); - return arr - ? (0.299 * arr[0] + 0.587 * arr[1] + 0.114 * arr[2]) * arr[3] / 255 - + (1 - arr[3]) * backgroundLum - : 0; - } - function random() { - return stringify([ - Math.round(Math.random() * 255), - Math.round(Math.random() * 255), - Math.round(Math.random() * 255) - ], 'rgb'); - } - - var color = /*#__PURE__*/Object.freeze({ - __proto__: null, - parse: parse, - lift: lift, - toHex: toHex, - fastLerp: fastLerp, - fastMapToColor: fastMapToColor, - lerp: lerp$1, - mapToColor: mapToColor, - modifyHSL: modifyHSL, - modifyAlpha: modifyAlpha, - stringify: stringify, - lum: lum, - random: random - }); - - var mathRound = Math.round; - function normalizeColor(color) { - var opacity; - if (!color || color === 'transparent') { - color = 'none'; - } - else if (typeof color === 'string' && color.indexOf('rgba') > -1) { - var arr = parse(color); - if (arr) { - color = 'rgb(' + arr[0] + ',' + arr[1] + ',' + arr[2] + ')'; - opacity = arr[3]; - } - } - return { - color: color, - opacity: opacity == null ? 1 : opacity - }; - } - var EPSILON$1 = 1e-4; - function isAroundZero$1(transform) { - return transform < EPSILON$1 && transform > -EPSILON$1; - } - function round3(transform) { - return mathRound(transform * 1e3) / 1e3; - } - function round4(transform) { - return mathRound(transform * 1e4) / 1e4; - } - function getMatrixStr(m) { - return 'matrix(' - + round3(m[0]) + ',' - + round3(m[1]) + ',' - + round3(m[2]) + ',' - + round3(m[3]) + ',' - + round4(m[4]) + ',' - + round4(m[5]) - + ')'; - } - var TEXT_ALIGN_TO_ANCHOR = { - left: 'start', - right: 'end', - center: 'middle', - middle: 'middle' - }; - function adjustTextY(y, lineHeight, textBaseline) { - if (textBaseline === 'top') { - y += lineHeight / 2; - } - else if (textBaseline === 'bottom') { - y -= lineHeight / 2; - } - return y; - } - function hasShadow(style) { - return style - && (style.shadowBlur || style.shadowOffsetX || style.shadowOffsetY); - } - function getShadowKey(displayable) { - var style = displayable.style; - var globalScale = displayable.getGlobalScale(); - return [ - style.shadowColor, - (style.shadowBlur || 0).toFixed(2), - (style.shadowOffsetX || 0).toFixed(2), - (style.shadowOffsetY || 0).toFixed(2), - globalScale[0], - globalScale[1] - ].join(','); - } - function isImagePattern(val) { - return val && (!!val.image); - } - function isSVGPattern(val) { - return val && (!!val.svgElement); - } - function isPattern(val) { - return isImagePattern(val) || isSVGPattern(val); - } - function isLinearGradient(val) { - return val.type === 'linear'; - } - function isRadialGradient(val) { - return val.type === 'radial'; - } - function isGradient(val) { - return val && (val.type === 'linear' - || val.type === 'radial'); - } - function getIdURL(id) { - return "url(#" + id + ")"; - } - function getPathPrecision(el) { - var scale = el.getGlobalScale(); - var size = Math.max(scale[0], scale[1]); - return Math.max(Math.ceil(Math.log(size) / Math.log(10)), 1); - } - function getSRTTransformString(transform) { - var x = transform.x || 0; - var y = transform.y || 0; - var rotation = (transform.rotation || 0) * RADIAN_TO_DEGREE; - var scaleX = retrieve2(transform.scaleX, 1); - var scaleY = retrieve2(transform.scaleY, 1); - var skewX = transform.skewX || 0; - var skewY = transform.skewY || 0; - var res = []; - if (x || y) { - res.push("translate(" + x + "px," + y + "px)"); - } - if (rotation) { - res.push("rotate(" + rotation + ")"); - } - if (scaleX !== 1 || scaleY !== 1) { - res.push("scale(" + scaleX + "," + scaleY + ")"); - } - if (skewX || skewY) { - res.push("skew(" + mathRound(skewX * RADIAN_TO_DEGREE) + "deg, " + mathRound(skewY * RADIAN_TO_DEGREE) + "deg)"); - } - return res.join(' '); - } - var encodeBase64 = (function () { - if (env.hasGlobalWindow && isFunction(window.btoa)) { - return function (str) { - return window.btoa(unescape(encodeURIComponent(str))); - }; - } - if (typeof Buffer !== 'undefined') { - return function (str) { - return Buffer.from(str).toString('base64'); - }; - } - return function (str) { - if ("development" !== 'production') { - logError('Base64 isn\'t natively supported in the current environment.'); - } - return null; - }; - })(); - - var arraySlice = Array.prototype.slice; - function interpolateNumber(p0, p1, percent) { - return (p1 - p0) * percent + p0; - } - function interpolate1DArray(out, p0, p1, percent) { - var len = p0.length; - for (var i = 0; i < len; i++) { - out[i] = interpolateNumber(p0[i], p1[i], percent); - } - return out; - } - function interpolate2DArray(out, p0, p1, percent) { - var len = p0.length; - var len2 = len && p0[0].length; - for (var i = 0; i < len; i++) { - if (!out[i]) { - out[i] = []; - } - for (var j = 0; j < len2; j++) { - out[i][j] = interpolateNumber(p0[i][j], p1[i][j], percent); - } - } - return out; - } - function add1DArray(out, p0, p1, sign) { - var len = p0.length; - for (var i = 0; i < len; i++) { - out[i] = p0[i] + p1[i] * sign; - } - return out; - } - function add2DArray(out, p0, p1, sign) { - var len = p0.length; - var len2 = len && p0[0].length; - for (var i = 0; i < len; i++) { - if (!out[i]) { - out[i] = []; - } - for (var j = 0; j < len2; j++) { - out[i][j] = p0[i][j] + p1[i][j] * sign; - } - } - return out; - } - function fillColorStops(val0, val1) { - var len0 = val0.length; - var len1 = val1.length; - var shorterArr = len0 > len1 ? val1 : val0; - var shorterLen = Math.min(len0, len1); - var last = shorterArr[shorterLen - 1] || { color: [0, 0, 0, 0], offset: 0 }; - for (var i = shorterLen; i < Math.max(len0, len1); i++) { - shorterArr.push({ - offset: last.offset, - color: last.color.slice() - }); - } - } - function fillArray(val0, val1, arrDim) { - var arr0 = val0; - var arr1 = val1; - if (!arr0.push || !arr1.push) { - return; - } - var arr0Len = arr0.length; - var arr1Len = arr1.length; - if (arr0Len !== arr1Len) { - var isPreviousLarger = arr0Len > arr1Len; - if (isPreviousLarger) { - arr0.length = arr1Len; - } - else { - for (var i = arr0Len; i < arr1Len; i++) { - arr0.push(arrDim === 1 ? arr1[i] : arraySlice.call(arr1[i])); - } - } - } - var len2 = arr0[0] && arr0[0].length; - for (var i = 0; i < arr0.length; i++) { - if (arrDim === 1) { - if (isNaN(arr0[i])) { - arr0[i] = arr1[i]; - } - } - else { - for (var j = 0; j < len2; j++) { - if (isNaN(arr0[i][j])) { - arr0[i][j] = arr1[i][j]; - } - } - } - } - } - function cloneValue(value) { - if (isArrayLike(value)) { - var len = value.length; - if (isArrayLike(value[0])) { - var ret = []; - for (var i = 0; i < len; i++) { - ret.push(arraySlice.call(value[i])); - } - return ret; - } - return arraySlice.call(value); - } - return value; - } - function rgba2String(rgba) { - rgba[0] = Math.floor(rgba[0]) || 0; - rgba[1] = Math.floor(rgba[1]) || 0; - rgba[2] = Math.floor(rgba[2]) || 0; - rgba[3] = rgba[3] == null ? 1 : rgba[3]; - return 'rgba(' + rgba.join(',') + ')'; - } - function guessArrayDim(value) { - return isArrayLike(value && value[0]) ? 2 : 1; - } - var VALUE_TYPE_NUMBER = 0; - var VALUE_TYPE_1D_ARRAY = 1; - var VALUE_TYPE_2D_ARRAY = 2; - var VALUE_TYPE_COLOR = 3; - var VALUE_TYPE_LINEAR_GRADIENT = 4; - var VALUE_TYPE_RADIAL_GRADIENT = 5; - var VALUE_TYPE_UNKOWN = 6; - function isGradientValueType(valType) { - return valType === VALUE_TYPE_LINEAR_GRADIENT || valType === VALUE_TYPE_RADIAL_GRADIENT; - } - function isArrayValueType(valType) { - return valType === VALUE_TYPE_1D_ARRAY || valType === VALUE_TYPE_2D_ARRAY; - } - var tmpRgba = [0, 0, 0, 0]; - var Track = (function () { - function Track(propName) { - this.keyframes = []; - this.discrete = false; - this._invalid = false; - this._needsSort = false; - this._lastFr = 0; - this._lastFrP = 0; - this.propName = propName; - } - Track.prototype.isFinished = function () { - return this._finished; - }; - Track.prototype.setFinished = function () { - this._finished = true; - if (this._additiveTrack) { - this._additiveTrack.setFinished(); - } - }; - Track.prototype.needsAnimate = function () { - return this.keyframes.length >= 1; - }; - Track.prototype.getAdditiveTrack = function () { - return this._additiveTrack; - }; - Track.prototype.addKeyframe = function (time, rawValue, easing) { - this._needsSort = true; - var keyframes = this.keyframes; - var len = keyframes.length; - var discrete = false; - var valType = VALUE_TYPE_UNKOWN; - var value = rawValue; - if (isArrayLike(rawValue)) { - var arrayDim = guessArrayDim(rawValue); - valType = arrayDim; - if (arrayDim === 1 && !isNumber(rawValue[0]) - || arrayDim === 2 && !isNumber(rawValue[0][0])) { - discrete = true; - } - } - else { - if (isNumber(rawValue) && !eqNaN(rawValue)) { - valType = VALUE_TYPE_NUMBER; - } - else if (isString(rawValue)) { - if (!isNaN(+rawValue)) { - valType = VALUE_TYPE_NUMBER; - } - else { - var colorArray = parse(rawValue); - if (colorArray) { - value = colorArray; - valType = VALUE_TYPE_COLOR; - } - } - } - else if (isGradientObject(rawValue)) { - var parsedGradient = extend({}, value); - parsedGradient.colorStops = map(rawValue.colorStops, function (colorStop) { return ({ - offset: colorStop.offset, - color: parse(colorStop.color) - }); }); - if (isLinearGradient(rawValue)) { - valType = VALUE_TYPE_LINEAR_GRADIENT; - } - else if (isRadialGradient(rawValue)) { - valType = VALUE_TYPE_RADIAL_GRADIENT; - } - value = parsedGradient; - } - } - if (len === 0) { - this.valType = valType; - } - else if (valType !== this.valType || valType === VALUE_TYPE_UNKOWN) { - discrete = true; - } - this.discrete = this.discrete || discrete; - var kf = { - time: time, - value: value, - rawValue: rawValue, - percent: 0 - }; - if (easing) { - kf.easing = easing; - kf.easingFunc = isFunction(easing) - ? easing - : easingFuncs[easing] || createCubicEasingFunc(easing); - } - keyframes.push(kf); - return kf; - }; - Track.prototype.prepare = function (maxTime, additiveTrack) { - var kfs = this.keyframes; - if (this._needsSort) { - kfs.sort(function (a, b) { - return a.time - b.time; - }); - } - var valType = this.valType; - var kfsLen = kfs.length; - var lastKf = kfs[kfsLen - 1]; - var isDiscrete = this.discrete; - var isArr = isArrayValueType(valType); - var isGradient = isGradientValueType(valType); - for (var i = 0; i < kfsLen; i++) { - var kf = kfs[i]; - var value = kf.value; - var lastValue = lastKf.value; - kf.percent = kf.time / maxTime; - if (!isDiscrete) { - if (isArr && i !== kfsLen - 1) { - fillArray(value, lastValue, valType); - } - else if (isGradient) { - fillColorStops(value.colorStops, lastValue.colorStops); - } - } - } - if (!isDiscrete - && valType !== VALUE_TYPE_RADIAL_GRADIENT - && additiveTrack - && this.needsAnimate() - && additiveTrack.needsAnimate() - && valType === additiveTrack.valType - && !additiveTrack._finished) { - this._additiveTrack = additiveTrack; - var startValue = kfs[0].value; - for (var i = 0; i < kfsLen; i++) { - if (valType === VALUE_TYPE_NUMBER) { - kfs[i].additiveValue = kfs[i].value - startValue; - } - else if (valType === VALUE_TYPE_COLOR) { - kfs[i].additiveValue = - add1DArray([], kfs[i].value, startValue, -1); - } - else if (isArrayValueType(valType)) { - kfs[i].additiveValue = valType === VALUE_TYPE_1D_ARRAY - ? add1DArray([], kfs[i].value, startValue, -1) - : add2DArray([], kfs[i].value, startValue, -1); - } - } - } - }; - Track.prototype.step = function (target, percent) { - if (this._finished) { - return; - } - if (this._additiveTrack && this._additiveTrack._finished) { - this._additiveTrack = null; - } - var isAdditive = this._additiveTrack != null; - var valueKey = isAdditive ? 'additiveValue' : 'value'; - var valType = this.valType; - var keyframes = this.keyframes; - var kfsNum = keyframes.length; - var propName = this.propName; - var isValueColor = valType === VALUE_TYPE_COLOR; - var frameIdx; - var lastFrame = this._lastFr; - var mathMin = Math.min; - var frame; - var nextFrame; - if (kfsNum === 1) { - frame = nextFrame = keyframes[0]; - } - else { - if (percent < 0) { - frameIdx = 0; - } - else if (percent < this._lastFrP) { - var start = mathMin(lastFrame + 1, kfsNum - 1); - for (frameIdx = start; frameIdx >= 0; frameIdx--) { - if (keyframes[frameIdx].percent <= percent) { - break; - } - } - frameIdx = mathMin(frameIdx, kfsNum - 2); - } - else { - for (frameIdx = lastFrame; frameIdx < kfsNum; frameIdx++) { - if (keyframes[frameIdx].percent > percent) { - break; - } - } - frameIdx = mathMin(frameIdx - 1, kfsNum - 2); - } - nextFrame = keyframes[frameIdx + 1]; - frame = keyframes[frameIdx]; - } - if (!(frame && nextFrame)) { - return; - } - this._lastFr = frameIdx; - this._lastFrP = percent; - var interval = (nextFrame.percent - frame.percent); - var w = interval === 0 ? 1 : mathMin((percent - frame.percent) / interval, 1); - if (nextFrame.easingFunc) { - w = nextFrame.easingFunc(w); - } - var targetArr = isAdditive ? this._additiveValue - : (isValueColor ? tmpRgba : target[propName]); - if ((isArrayValueType(valType) || isValueColor) && !targetArr) { - targetArr = this._additiveValue = []; - } - if (this.discrete) { - target[propName] = w < 1 ? frame.rawValue : nextFrame.rawValue; - } - else if (isArrayValueType(valType)) { - valType === VALUE_TYPE_1D_ARRAY - ? interpolate1DArray(targetArr, frame[valueKey], nextFrame[valueKey], w) - : interpolate2DArray(targetArr, frame[valueKey], nextFrame[valueKey], w); - } - else if (isGradientValueType(valType)) { - var val = frame[valueKey]; - var nextVal_1 = nextFrame[valueKey]; - var isLinearGradient_1 = valType === VALUE_TYPE_LINEAR_GRADIENT; - target[propName] = { - type: isLinearGradient_1 ? 'linear' : 'radial', - x: interpolateNumber(val.x, nextVal_1.x, w), - y: interpolateNumber(val.y, nextVal_1.y, w), - colorStops: map(val.colorStops, function (colorStop, idx) { - var nextColorStop = nextVal_1.colorStops[idx]; - return { - offset: interpolateNumber(colorStop.offset, nextColorStop.offset, w), - color: rgba2String(interpolate1DArray([], colorStop.color, nextColorStop.color, w)) - }; - }), - global: nextVal_1.global - }; - if (isLinearGradient_1) { - target[propName].x2 = interpolateNumber(val.x2, nextVal_1.x2, w); - target[propName].y2 = interpolateNumber(val.y2, nextVal_1.y2, w); - } - else { - target[propName].r = interpolateNumber(val.r, nextVal_1.r, w); - } - } - else if (isValueColor) { - interpolate1DArray(targetArr, frame[valueKey], nextFrame[valueKey], w); - if (!isAdditive) { - target[propName] = rgba2String(targetArr); - } - } - else { - var value = interpolateNumber(frame[valueKey], nextFrame[valueKey], w); - if (isAdditive) { - this._additiveValue = value; - } - else { - target[propName] = value; - } - } - if (isAdditive) { - this._addToTarget(target); - } - }; - Track.prototype._addToTarget = function (target) { - var valType = this.valType; - var propName = this.propName; - var additiveValue = this._additiveValue; - if (valType === VALUE_TYPE_NUMBER) { - target[propName] = target[propName] + additiveValue; - } - else if (valType === VALUE_TYPE_COLOR) { - parse(target[propName], tmpRgba); - add1DArray(tmpRgba, tmpRgba, additiveValue, 1); - target[propName] = rgba2String(tmpRgba); - } - else if (valType === VALUE_TYPE_1D_ARRAY) { - add1DArray(target[propName], target[propName], additiveValue, 1); - } - else if (valType === VALUE_TYPE_2D_ARRAY) { - add2DArray(target[propName], target[propName], additiveValue, 1); - } - }; - return Track; - }()); - var Animator = (function () { - function Animator(target, loop, allowDiscreteAnimation, additiveTo) { - this._tracks = {}; - this._trackKeys = []; - this._maxTime = 0; - this._started = 0; - this._clip = null; - this._target = target; - this._loop = loop; - if (loop && additiveTo) { - logError('Can\' use additive animation on looped animation.'); - return; - } - this._additiveAnimators = additiveTo; - this._allowDiscrete = allowDiscreteAnimation; - } - Animator.prototype.getMaxTime = function () { - return this._maxTime; - }; - Animator.prototype.getDelay = function () { - return this._delay; - }; - Animator.prototype.getLoop = function () { - return this._loop; - }; - Animator.prototype.getTarget = function () { - return this._target; - }; - Animator.prototype.changeTarget = function (target) { - this._target = target; - }; - Animator.prototype.when = function (time, props, easing) { - return this.whenWithKeys(time, props, keys(props), easing); - }; - Animator.prototype.whenWithKeys = function (time, props, propNames, easing) { - var tracks = this._tracks; - for (var i = 0; i < propNames.length; i++) { - var propName = propNames[i]; - var track = tracks[propName]; - if (!track) { - track = tracks[propName] = new Track(propName); - var initialValue = void 0; - var additiveTrack = this._getAdditiveTrack(propName); - if (additiveTrack) { - var addtiveTrackKfs = additiveTrack.keyframes; - var lastFinalKf = addtiveTrackKfs[addtiveTrackKfs.length - 1]; - initialValue = lastFinalKf && lastFinalKf.value; - if (additiveTrack.valType === VALUE_TYPE_COLOR && initialValue) { - initialValue = rgba2String(initialValue); - } - } - else { - initialValue = this._target[propName]; - } - if (initialValue == null) { - continue; - } - if (time > 0) { - track.addKeyframe(0, cloneValue(initialValue), easing); - } - this._trackKeys.push(propName); - } - track.addKeyframe(time, cloneValue(props[propName]), easing); - } - this._maxTime = Math.max(this._maxTime, time); - return this; - }; - Animator.prototype.pause = function () { - this._clip.pause(); - this._paused = true; - }; - Animator.prototype.resume = function () { - this._clip.resume(); - this._paused = false; - }; - Animator.prototype.isPaused = function () { - return !!this._paused; - }; - Animator.prototype.duration = function (duration) { - this._maxTime = duration; - this._force = true; - return this; - }; - Animator.prototype._doneCallback = function () { - this._setTracksFinished(); - this._clip = null; - var doneList = this._doneCbs; - if (doneList) { - var len = doneList.length; - for (var i = 0; i < len; i++) { - doneList[i].call(this); - } - } - }; - Animator.prototype._abortedCallback = function () { - this._setTracksFinished(); - var animation = this.animation; - var abortedList = this._abortedCbs; - if (animation) { - animation.removeClip(this._clip); - } - this._clip = null; - if (abortedList) { - for (var i = 0; i < abortedList.length; i++) { - abortedList[i].call(this); - } - } - }; - Animator.prototype._setTracksFinished = function () { - var tracks = this._tracks; - var tracksKeys = this._trackKeys; - for (var i = 0; i < tracksKeys.length; i++) { - tracks[tracksKeys[i]].setFinished(); - } - }; - Animator.prototype._getAdditiveTrack = function (trackName) { - var additiveTrack; - var additiveAnimators = this._additiveAnimators; - if (additiveAnimators) { - for (var i = 0; i < additiveAnimators.length; i++) { - var track = additiveAnimators[i].getTrack(trackName); - if (track) { - additiveTrack = track; - } - } - } - return additiveTrack; - }; - Animator.prototype.start = function (easing) { - if (this._started > 0) { - return; - } - this._started = 1; - var self = this; - var tracks = []; - var maxTime = this._maxTime || 0; - for (var i = 0; i < this._trackKeys.length; i++) { - var propName = this._trackKeys[i]; - var track = this._tracks[propName]; - var additiveTrack = this._getAdditiveTrack(propName); - var kfs = track.keyframes; - var kfsNum = kfs.length; - track.prepare(maxTime, additiveTrack); - if (track.needsAnimate()) { - if (!this._allowDiscrete && track.discrete) { - var lastKf = kfs[kfsNum - 1]; - if (lastKf) { - self._target[track.propName] = lastKf.rawValue; - } - track.setFinished(); - } - else { - tracks.push(track); - } - } - } - if (tracks.length || this._force) { - var clip = new Clip({ - life: maxTime, - loop: this._loop, - delay: this._delay || 0, - onframe: function (percent) { - self._started = 2; - var additiveAnimators = self._additiveAnimators; - if (additiveAnimators) { - var stillHasAdditiveAnimator = false; - for (var i = 0; i < additiveAnimators.length; i++) { - if (additiveAnimators[i]._clip) { - stillHasAdditiveAnimator = true; - break; - } - } - if (!stillHasAdditiveAnimator) { - self._additiveAnimators = null; - } - } - for (var i = 0; i < tracks.length; i++) { - tracks[i].step(self._target, percent); - } - var onframeList = self._onframeCbs; - if (onframeList) { - for (var i = 0; i < onframeList.length; i++) { - onframeList[i](self._target, percent); - } - } - }, - ondestroy: function () { - self._doneCallback(); - } - }); - this._clip = clip; - if (this.animation) { - this.animation.addClip(clip); - } - if (easing) { - clip.setEasing(easing); - } - } - else { - this._doneCallback(); - } - return this; - }; - Animator.prototype.stop = function (forwardToLast) { - if (!this._clip) { - return; - } - var clip = this._clip; - if (forwardToLast) { - clip.onframe(1); - } - this._abortedCallback(); - }; - Animator.prototype.delay = function (time) { - this._delay = time; - return this; - }; - Animator.prototype.during = function (cb) { - if (cb) { - if (!this._onframeCbs) { - this._onframeCbs = []; - } - this._onframeCbs.push(cb); - } - return this; - }; - Animator.prototype.done = function (cb) { - if (cb) { - if (!this._doneCbs) { - this._doneCbs = []; - } - this._doneCbs.push(cb); - } - return this; - }; - Animator.prototype.aborted = function (cb) { - if (cb) { - if (!this._abortedCbs) { - this._abortedCbs = []; - } - this._abortedCbs.push(cb); - } - return this; - }; - Animator.prototype.getClip = function () { - return this._clip; - }; - Animator.prototype.getTrack = function (propName) { - return this._tracks[propName]; - }; - Animator.prototype.getTracks = function () { - var _this = this; - return map(this._trackKeys, function (key) { return _this._tracks[key]; }); - }; - Animator.prototype.stopTracks = function (propNames, forwardToLast) { - if (!propNames.length || !this._clip) { - return true; - } - var tracks = this._tracks; - var tracksKeys = this._trackKeys; - for (var i = 0; i < propNames.length; i++) { - var track = tracks[propNames[i]]; - if (track && !track.isFinished()) { - if (forwardToLast) { - track.step(this._target, 1); - } - else if (this._started === 1) { - track.step(this._target, 0); - } - track.setFinished(); - } - } - var allAborted = true; - for (var i = 0; i < tracksKeys.length; i++) { - if (!tracks[tracksKeys[i]].isFinished()) { - allAborted = false; - break; - } - } - if (allAborted) { - this._abortedCallback(); - } - return allAborted; - }; - Animator.prototype.saveTo = function (target, trackKeys, firstOrLast) { - if (!target) { - return; - } - trackKeys = trackKeys || this._trackKeys; - for (var i = 0; i < trackKeys.length; i++) { - var propName = trackKeys[i]; - var track = this._tracks[propName]; - if (!track || track.isFinished()) { - continue; - } - var kfs = track.keyframes; - var kf = kfs[firstOrLast ? 0 : kfs.length - 1]; - if (kf) { - target[propName] = cloneValue(kf.rawValue); - } - } - }; - Animator.prototype.__changeFinalValue = function (finalProps, trackKeys) { - trackKeys = trackKeys || keys(finalProps); - for (var i = 0; i < trackKeys.length; i++) { - var propName = trackKeys[i]; - var track = this._tracks[propName]; - if (!track) { - continue; - } - var kfs = track.keyframes; - if (kfs.length > 1) { - var lastKf = kfs.pop(); - track.addKeyframe(lastKf.time, finalProps[propName]); - track.prepare(this._maxTime, track.getAdditiveTrack()); - } - } - }; - return Animator; - }()); - - function getTime() { - return new Date().getTime(); - } - var Animation = (function (_super) { - __extends(Animation, _super); - function Animation(opts) { - var _this = _super.call(this) || this; - _this._running = false; - _this._time = 0; - _this._pausedTime = 0; - _this._pauseStart = 0; - _this._paused = false; - opts = opts || {}; - _this.stage = opts.stage || {}; - return _this; - } - Animation.prototype.addClip = function (clip) { - if (clip.animation) { - this.removeClip(clip); - } - if (!this._head) { - this._head = this._tail = clip; - } - else { - this._tail.next = clip; - clip.prev = this._tail; - clip.next = null; - this._tail = clip; - } - clip.animation = this; - }; - Animation.prototype.addAnimator = function (animator) { - animator.animation = this; - var clip = animator.getClip(); - if (clip) { - this.addClip(clip); - } - }; - Animation.prototype.removeClip = function (clip) { - if (!clip.animation) { - return; - } - var prev = clip.prev; - var next = clip.next; - if (prev) { - prev.next = next; - } - else { - this._head = next; - } - if (next) { - next.prev = prev; - } - else { - this._tail = prev; - } - clip.next = clip.prev = clip.animation = null; - }; - Animation.prototype.removeAnimator = function (animator) { - var clip = animator.getClip(); - if (clip) { - this.removeClip(clip); - } - animator.animation = null; - }; - Animation.prototype.update = function (notTriggerFrameAndStageUpdate) { - var time = getTime() - this._pausedTime; - var delta = time - this._time; - var clip = this._head; - while (clip) { - var nextClip = clip.next; - var finished = clip.step(time, delta); - if (finished) { - clip.ondestroy(); - this.removeClip(clip); - clip = nextClip; - } - else { - clip = nextClip; - } - } - this._time = time; - if (!notTriggerFrameAndStageUpdate) { - this.trigger('frame', delta); - this.stage.update && this.stage.update(); - } - }; - Animation.prototype._startLoop = function () { - var self = this; - this._running = true; - function step() { - if (self._running) { - requestAnimationFrame$1(step); - !self._paused && self.update(); - } - } - requestAnimationFrame$1(step); - }; - Animation.prototype.start = function () { - if (this._running) { - return; - } - this._time = getTime(); - this._pausedTime = 0; - this._startLoop(); - }; - Animation.prototype.stop = function () { - this._running = false; - }; - Animation.prototype.pause = function () { - if (!this._paused) { - this._pauseStart = getTime(); - this._paused = true; - } - }; - Animation.prototype.resume = function () { - if (this._paused) { - this._pausedTime += getTime() - this._pauseStart; - this._paused = false; - } - }; - Animation.prototype.clear = function () { - var clip = this._head; - while (clip) { - var nextClip = clip.next; - clip.prev = clip.next = clip.animation = null; - clip = nextClip; - } - this._head = this._tail = null; - }; - Animation.prototype.isFinished = function () { - return this._head == null; - }; - Animation.prototype.animate = function (target, options) { - options = options || {}; - this.start(); - var animator = new Animator(target, options.loop); - this.addAnimator(animator); - return animator; - }; - return Animation; - }(Eventful)); - - var TOUCH_CLICK_DELAY = 300; - var globalEventSupported = env.domSupported; - var localNativeListenerNames = (function () { - var mouseHandlerNames = [ - 'click', 'dblclick', 'mousewheel', 'wheel', 'mouseout', - 'mouseup', 'mousedown', 'mousemove', 'contextmenu' - ]; - var touchHandlerNames = [ - 'touchstart', 'touchend', 'touchmove' - ]; - var pointerEventNameMap = { - pointerdown: 1, pointerup: 1, pointermove: 1, pointerout: 1 - }; - var pointerHandlerNames = map(mouseHandlerNames, function (name) { - var nm = name.replace('mouse', 'pointer'); - return pointerEventNameMap.hasOwnProperty(nm) ? nm : name; - }); - return { - mouse: mouseHandlerNames, - touch: touchHandlerNames, - pointer: pointerHandlerNames - }; - })(); - var globalNativeListenerNames = { - mouse: ['mousemove', 'mouseup'], - pointer: ['pointermove', 'pointerup'] - }; - var wheelEventSupported = false; - function isPointerFromTouch(event) { - var pointerType = event.pointerType; - return pointerType === 'pen' || pointerType === 'touch'; - } - function setTouchTimer(scope) { - scope.touching = true; - if (scope.touchTimer != null) { - clearTimeout(scope.touchTimer); - scope.touchTimer = null; - } - scope.touchTimer = setTimeout(function () { - scope.touching = false; - scope.touchTimer = null; - }, 700); - } - function markTouch(event) { - event && (event.zrByTouch = true); - } - function normalizeGlobalEvent(instance, event) { - return normalizeEvent(instance.dom, new FakeGlobalEvent(instance, event), true); - } - function isLocalEl(instance, el) { - var elTmp = el; - var isLocal = false; - while (elTmp && elTmp.nodeType !== 9 - && !(isLocal = elTmp.domBelongToZr - || (elTmp !== el && elTmp === instance.painterRoot))) { - elTmp = elTmp.parentNode; - } - return isLocal; - } - var FakeGlobalEvent = (function () { - function FakeGlobalEvent(instance, event) { - this.stopPropagation = noop; - this.stopImmediatePropagation = noop; - this.preventDefault = noop; - this.type = event.type; - this.target = this.currentTarget = instance.dom; - this.pointerType = event.pointerType; - this.clientX = event.clientX; - this.clientY = event.clientY; - } - return FakeGlobalEvent; - }()); - var localDOMHandlers = { - mousedown: function (event) { - event = normalizeEvent(this.dom, event); - this.__mayPointerCapture = [event.zrX, event.zrY]; - this.trigger('mousedown', event); - }, - mousemove: function (event) { - event = normalizeEvent(this.dom, event); - var downPoint = this.__mayPointerCapture; - if (downPoint && (event.zrX !== downPoint[0] || event.zrY !== downPoint[1])) { - this.__togglePointerCapture(true); - } - this.trigger('mousemove', event); - }, - mouseup: function (event) { - event = normalizeEvent(this.dom, event); - this.__togglePointerCapture(false); - this.trigger('mouseup', event); - }, - mouseout: function (event) { - event = normalizeEvent(this.dom, event); - var element = event.toElement || event.relatedTarget; - if (!isLocalEl(this, element)) { - if (this.__pointerCapturing) { - event.zrEventControl = 'no_globalout'; - } - this.trigger('mouseout', event); - } - }, - wheel: function (event) { - wheelEventSupported = true; - event = normalizeEvent(this.dom, event); - this.trigger('mousewheel', event); - }, - mousewheel: function (event) { - if (wheelEventSupported) { - return; - } - event = normalizeEvent(this.dom, event); - this.trigger('mousewheel', event); - }, - touchstart: function (event) { - event = normalizeEvent(this.dom, event); - markTouch(event); - this.__lastTouchMoment = new Date(); - this.handler.processGesture(event, 'start'); - localDOMHandlers.mousemove.call(this, event); - localDOMHandlers.mousedown.call(this, event); - }, - touchmove: function (event) { - event = normalizeEvent(this.dom, event); - markTouch(event); - this.handler.processGesture(event, 'change'); - localDOMHandlers.mousemove.call(this, event); - }, - touchend: function (event) { - event = normalizeEvent(this.dom, event); - markTouch(event); - this.handler.processGesture(event, 'end'); - localDOMHandlers.mouseup.call(this, event); - if (+new Date() - (+this.__lastTouchMoment) < TOUCH_CLICK_DELAY) { - localDOMHandlers.click.call(this, event); - } - }, - pointerdown: function (event) { - localDOMHandlers.mousedown.call(this, event); - }, - pointermove: function (event) { - if (!isPointerFromTouch(event)) { - localDOMHandlers.mousemove.call(this, event); - } - }, - pointerup: function (event) { - localDOMHandlers.mouseup.call(this, event); - }, - pointerout: function (event) { - if (!isPointerFromTouch(event)) { - localDOMHandlers.mouseout.call(this, event); - } - } - }; - each(['click', 'dblclick', 'contextmenu'], function (name) { - localDOMHandlers[name] = function (event) { - event = normalizeEvent(this.dom, event); - this.trigger(name, event); - }; - }); - var globalDOMHandlers = { - pointermove: function (event) { - if (!isPointerFromTouch(event)) { - globalDOMHandlers.mousemove.call(this, event); - } - }, - pointerup: function (event) { - globalDOMHandlers.mouseup.call(this, event); - }, - mousemove: function (event) { - this.trigger('mousemove', event); - }, - mouseup: function (event) { - var pointerCaptureReleasing = this.__pointerCapturing; - this.__togglePointerCapture(false); - this.trigger('mouseup', event); - if (pointerCaptureReleasing) { - event.zrEventControl = 'only_globalout'; - this.trigger('mouseout', event); - } - } - }; - function mountLocalDOMEventListeners(instance, scope) { - var domHandlers = scope.domHandlers; - if (env.pointerEventsSupported) { - each(localNativeListenerNames.pointer, function (nativeEventName) { - mountSingleDOMEventListener(scope, nativeEventName, function (event) { - domHandlers[nativeEventName].call(instance, event); - }); - }); - } - else { - if (env.touchEventsSupported) { - each(localNativeListenerNames.touch, function (nativeEventName) { - mountSingleDOMEventListener(scope, nativeEventName, function (event) { - domHandlers[nativeEventName].call(instance, event); - setTouchTimer(scope); - }); - }); - } - each(localNativeListenerNames.mouse, function (nativeEventName) { - mountSingleDOMEventListener(scope, nativeEventName, function (event) { - event = getNativeEvent(event); - if (!scope.touching) { - domHandlers[nativeEventName].call(instance, event); - } - }); - }); - } - } - function mountGlobalDOMEventListeners(instance, scope) { - if (env.pointerEventsSupported) { - each(globalNativeListenerNames.pointer, mount); - } - else if (!env.touchEventsSupported) { - each(globalNativeListenerNames.mouse, mount); - } - function mount(nativeEventName) { - function nativeEventListener(event) { - event = getNativeEvent(event); - if (!isLocalEl(instance, event.target)) { - event = normalizeGlobalEvent(instance, event); - scope.domHandlers[nativeEventName].call(instance, event); - } - } - mountSingleDOMEventListener(scope, nativeEventName, nativeEventListener, { capture: true }); - } - } - function mountSingleDOMEventListener(scope, nativeEventName, listener, opt) { - scope.mounted[nativeEventName] = listener; - scope.listenerOpts[nativeEventName] = opt; - addEventListener(scope.domTarget, nativeEventName, listener, opt); - } - function unmountDOMEventListeners(scope) { - var mounted = scope.mounted; - for (var nativeEventName in mounted) { - if (mounted.hasOwnProperty(nativeEventName)) { - removeEventListener(scope.domTarget, nativeEventName, mounted[nativeEventName], scope.listenerOpts[nativeEventName]); - } - } - scope.mounted = {}; - } - var DOMHandlerScope = (function () { - function DOMHandlerScope(domTarget, domHandlers) { - this.mounted = {}; - this.listenerOpts = {}; - this.touching = false; - this.domTarget = domTarget; - this.domHandlers = domHandlers; - } - return DOMHandlerScope; - }()); - var HandlerDomProxy = (function (_super) { - __extends(HandlerDomProxy, _super); - function HandlerDomProxy(dom, painterRoot) { - var _this = _super.call(this) || this; - _this.__pointerCapturing = false; - _this.dom = dom; - _this.painterRoot = painterRoot; - _this._localHandlerScope = new DOMHandlerScope(dom, localDOMHandlers); - if (globalEventSupported) { - _this._globalHandlerScope = new DOMHandlerScope(document, globalDOMHandlers); - } - mountLocalDOMEventListeners(_this, _this._localHandlerScope); - return _this; - } - HandlerDomProxy.prototype.dispose = function () { - unmountDOMEventListeners(this._localHandlerScope); - if (globalEventSupported) { - unmountDOMEventListeners(this._globalHandlerScope); - } - }; - HandlerDomProxy.prototype.setCursor = function (cursorStyle) { - this.dom.style && (this.dom.style.cursor = cursorStyle || 'default'); - }; - HandlerDomProxy.prototype.__togglePointerCapture = function (isPointerCapturing) { - this.__mayPointerCapture = null; - if (globalEventSupported - && ((+this.__pointerCapturing) ^ (+isPointerCapturing))) { - this.__pointerCapturing = isPointerCapturing; - var globalHandlerScope = this._globalHandlerScope; - isPointerCapturing - ? mountGlobalDOMEventListeners(this, globalHandlerScope) - : unmountDOMEventListeners(globalHandlerScope); - } - }; - return HandlerDomProxy; - }(Eventful)); - - var dpr = 1; - if (env.hasGlobalWindow) { - dpr = Math.max(window.devicePixelRatio - || (window.screen && window.screen.deviceXDPI / window.screen.logicalXDPI) - || 1, 1); - } - var devicePixelRatio = dpr; - var DARK_MODE_THRESHOLD = 0.4; - var DARK_LABEL_COLOR = '#333'; - var LIGHT_LABEL_COLOR = '#ccc'; - var LIGHTER_LABEL_COLOR = '#eee'; - - var mIdentity = identity; - var EPSILON$2 = 5e-5; - function isNotAroundZero$1(val) { - return val > EPSILON$2 || val < -EPSILON$2; - } - var scaleTmp = []; - var tmpTransform = []; - var originTransform = create$1(); - var abs = Math.abs; - var Transformable = (function () { - function Transformable() { - } - Transformable.prototype.getLocalTransform = function (m) { - return Transformable.getLocalTransform(this, m); - }; - Transformable.prototype.setPosition = function (arr) { - this.x = arr[0]; - this.y = arr[1]; - }; - Transformable.prototype.setScale = function (arr) { - this.scaleX = arr[0]; - this.scaleY = arr[1]; - }; - Transformable.prototype.setSkew = function (arr) { - this.skewX = arr[0]; - this.skewY = arr[1]; - }; - Transformable.prototype.setOrigin = function (arr) { - this.originX = arr[0]; - this.originY = arr[1]; - }; - Transformable.prototype.needLocalTransform = function () { - return isNotAroundZero$1(this.rotation) - || isNotAroundZero$1(this.x) - || isNotAroundZero$1(this.y) - || isNotAroundZero$1(this.scaleX - 1) - || isNotAroundZero$1(this.scaleY - 1) - || isNotAroundZero$1(this.skewX) - || isNotAroundZero$1(this.skewY); - }; - Transformable.prototype.updateTransform = function () { - var parentTransform = this.parent && this.parent.transform; - var needLocalTransform = this.needLocalTransform(); - var m = this.transform; - if (!(needLocalTransform || parentTransform)) { - m && mIdentity(m); - return; - } - m = m || create$1(); - if (needLocalTransform) { - this.getLocalTransform(m); - } - else { - mIdentity(m); - } - if (parentTransform) { - if (needLocalTransform) { - mul$1(m, parentTransform, m); - } - else { - copy$1(m, parentTransform); - } - } - this.transform = m; - this._resolveGlobalScaleRatio(m); - }; - Transformable.prototype._resolveGlobalScaleRatio = function (m) { - var globalScaleRatio = this.globalScaleRatio; - if (globalScaleRatio != null && globalScaleRatio !== 1) { - this.getGlobalScale(scaleTmp); - var relX = scaleTmp[0] < 0 ? -1 : 1; - var relY = scaleTmp[1] < 0 ? -1 : 1; - var sx = ((scaleTmp[0] - relX) * globalScaleRatio + relX) / scaleTmp[0] || 0; - var sy = ((scaleTmp[1] - relY) * globalScaleRatio + relY) / scaleTmp[1] || 0; - m[0] *= sx; - m[1] *= sx; - m[2] *= sy; - m[3] *= sy; - } - this.invTransform = this.invTransform || create$1(); - invert(this.invTransform, m); - }; - Transformable.prototype.getComputedTransform = function () { - var transformNode = this; - var ancestors = []; - while (transformNode) { - ancestors.push(transformNode); - transformNode = transformNode.parent; - } - while (transformNode = ancestors.pop()) { - transformNode.updateTransform(); - } - return this.transform; - }; - Transformable.prototype.setLocalTransform = function (m) { - if (!m) { - return; - } - var sx = m[0] * m[0] + m[1] * m[1]; - var sy = m[2] * m[2] + m[3] * m[3]; - var rotation = Math.atan2(m[1], m[0]); - var shearX = Math.PI / 2 + rotation - Math.atan2(m[3], m[2]); - sy = Math.sqrt(sy) * Math.cos(shearX); - sx = Math.sqrt(sx); - this.skewX = shearX; - this.skewY = 0; - this.rotation = -rotation; - this.x = +m[4]; - this.y = +m[5]; - this.scaleX = sx; - this.scaleY = sy; - this.originX = 0; - this.originY = 0; - }; - Transformable.prototype.decomposeTransform = function () { - if (!this.transform) { - return; - } - var parent = this.parent; - var m = this.transform; - if (parent && parent.transform) { - mul$1(tmpTransform, parent.invTransform, m); - m = tmpTransform; - } - var ox = this.originX; - var oy = this.originY; - if (ox || oy) { - originTransform[4] = ox; - originTransform[5] = oy; - mul$1(tmpTransform, m, originTransform); - tmpTransform[4] -= ox; - tmpTransform[5] -= oy; - m = tmpTransform; - } - this.setLocalTransform(m); - }; - Transformable.prototype.getGlobalScale = function (out) { - var m = this.transform; - out = out || []; - if (!m) { - out[0] = 1; - out[1] = 1; - return out; - } - out[0] = Math.sqrt(m[0] * m[0] + m[1] * m[1]); - out[1] = Math.sqrt(m[2] * m[2] + m[3] * m[3]); - if (m[0] < 0) { - out[0] = -out[0]; - } - if (m[3] < 0) { - out[1] = -out[1]; - } - return out; - }; - Transformable.prototype.transformCoordToLocal = function (x, y) { - var v2 = [x, y]; - var invTransform = this.invTransform; - if (invTransform) { - applyTransform(v2, v2, invTransform); - } - return v2; - }; - Transformable.prototype.transformCoordToGlobal = function (x, y) { - var v2 = [x, y]; - var transform = this.transform; - if (transform) { - applyTransform(v2, v2, transform); - } - return v2; - }; - Transformable.prototype.getLineScale = function () { - var m = this.transform; - return m && abs(m[0] - 1) > 1e-10 && abs(m[3] - 1) > 1e-10 - ? Math.sqrt(abs(m[0] * m[3] - m[2] * m[1])) - : 1; - }; - Transformable.prototype.copyTransform = function (source) { - copyTransform(this, source); - }; - Transformable.getLocalTransform = function (target, m) { - m = m || []; - var ox = target.originX || 0; - var oy = target.originY || 0; - var sx = target.scaleX; - var sy = target.scaleY; - var ax = target.anchorX; - var ay = target.anchorY; - var rotation = target.rotation || 0; - var x = target.x; - var y = target.y; - var skewX = target.skewX ? Math.tan(target.skewX) : 0; - var skewY = target.skewY ? Math.tan(-target.skewY) : 0; - if (ox || oy || ax || ay) { - var dx = ox + ax; - var dy = oy + ay; - m[4] = -dx * sx - skewX * dy * sy; - m[5] = -dy * sy - skewY * dx * sx; - } - else { - m[4] = m[5] = 0; - } - m[0] = sx; - m[3] = sy; - m[1] = skewY * sx; - m[2] = skewX * sy; - rotation && rotate(m, m, rotation); - m[4] += ox + x; - m[5] += oy + y; - return m; - }; - Transformable.initDefaultProps = (function () { - var proto = Transformable.prototype; - proto.scaleX = - proto.scaleY = - proto.globalScaleRatio = 1; - proto.x = - proto.y = - proto.originX = - proto.originY = - proto.skewX = - proto.skewY = - proto.rotation = - proto.anchorX = - proto.anchorY = 0; - })(); - return Transformable; - }()); - var TRANSFORMABLE_PROPS = [ - 'x', 'y', 'originX', 'originY', 'anchorX', 'anchorY', 'rotation', 'scaleX', 'scaleY', 'skewX', 'skewY' - ]; - function copyTransform(target, source) { - for (var i = 0; i < TRANSFORMABLE_PROPS.length; i++) { - var propName = TRANSFORMABLE_PROPS[i]; - target[propName] = source[propName]; - } - } - - var textWidthCache = {}; - function getWidth(text, font) { - font = font || DEFAULT_FONT; - var cacheOfFont = textWidthCache[font]; - if (!cacheOfFont) { - cacheOfFont = textWidthCache[font] = new LRU(500); - } - var width = cacheOfFont.get(text); - if (width == null) { - width = platformApi.measureText(text, font).width; - cacheOfFont.put(text, width); - } - return width; - } - function innerGetBoundingRect(text, font, textAlign, textBaseline) { - var width = getWidth(text, font); - var height = getLineHeight(font); - var x = adjustTextX(0, width, textAlign); - var y = adjustTextY$1(0, height, textBaseline); - var rect = new BoundingRect(x, y, width, height); - return rect; - } - function getBoundingRect(text, font, textAlign, textBaseline) { - var textLines = ((text || '') + '').split('\n'); - var len = textLines.length; - if (len === 1) { - return innerGetBoundingRect(textLines[0], font, textAlign, textBaseline); - } - else { - var uniondRect = new BoundingRect(0, 0, 0, 0); - for (var i = 0; i < textLines.length; i++) { - var rect = innerGetBoundingRect(textLines[i], font, textAlign, textBaseline); - i === 0 ? uniondRect.copy(rect) : uniondRect.union(rect); - } - return uniondRect; - } - } - function adjustTextX(x, width, textAlign) { - if (textAlign === 'right') { - x -= width; - } - else if (textAlign === 'center') { - x -= width / 2; - } - return x; - } - function adjustTextY$1(y, height, verticalAlign) { - if (verticalAlign === 'middle') { - y -= height / 2; - } - else if (verticalAlign === 'bottom') { - y -= height; - } - return y; - } - function getLineHeight(font) { - return getWidth('国', font); - } - function parsePercent(value, maxValue) { - if (typeof value === 'string') { - if (value.lastIndexOf('%') >= 0) { - return parseFloat(value) / 100 * maxValue; - } - return parseFloat(value); - } - return value; - } - function calculateTextPosition(out, opts, rect) { - var textPosition = opts.position || 'inside'; - var distance = opts.distance != null ? opts.distance : 5; - var height = rect.height; - var width = rect.width; - var halfHeight = height / 2; - var x = rect.x; - var y = rect.y; - var textAlign = 'left'; - var textVerticalAlign = 'top'; - if (textPosition instanceof Array) { - x += parsePercent(textPosition[0], rect.width); - y += parsePercent(textPosition[1], rect.height); - textAlign = null; - textVerticalAlign = null; - } - else { - switch (textPosition) { - case 'left': - x -= distance; - y += halfHeight; - textAlign = 'right'; - textVerticalAlign = 'middle'; - break; - case 'right': - x += distance + width; - y += halfHeight; - textVerticalAlign = 'middle'; - break; - case 'top': - x += width / 2; - y -= distance; - textAlign = 'center'; - textVerticalAlign = 'bottom'; - break; - case 'bottom': - x += width / 2; - y += height + distance; - textAlign = 'center'; - break; - case 'inside': - x += width / 2; - y += halfHeight; - textAlign = 'center'; - textVerticalAlign = 'middle'; - break; - case 'insideLeft': - x += distance; - y += halfHeight; - textVerticalAlign = 'middle'; - break; - case 'insideRight': - x += width - distance; - y += halfHeight; - textAlign = 'right'; - textVerticalAlign = 'middle'; - break; - case 'insideTop': - x += width / 2; - y += distance; - textAlign = 'center'; - break; - case 'insideBottom': - x += width / 2; - y += height - distance; - textAlign = 'center'; - textVerticalAlign = 'bottom'; - break; - case 'insideTopLeft': - x += distance; - y += distance; - break; - case 'insideTopRight': - x += width - distance; - y += distance; - textAlign = 'right'; - break; - case 'insideBottomLeft': - x += distance; - y += height - distance; - textVerticalAlign = 'bottom'; - break; - case 'insideBottomRight': - x += width - distance; - y += height - distance; - textAlign = 'right'; - textVerticalAlign = 'bottom'; - break; - } - } - out = out || {}; - out.x = x; - out.y = y; - out.align = textAlign; - out.verticalAlign = textVerticalAlign; - return out; - } - - var PRESERVED_NORMAL_STATE = '__zr_normal__'; - var PRIMARY_STATES_KEYS = TRANSFORMABLE_PROPS.concat(['ignore']); - var DEFAULT_ANIMATABLE_MAP = reduce(TRANSFORMABLE_PROPS, function (obj, key) { - obj[key] = true; - return obj; - }, { ignore: false }); - var tmpTextPosCalcRes = {}; - var tmpBoundingRect = new BoundingRect(0, 0, 0, 0); - var Element = (function () { - function Element(props) { - this.id = guid(); - this.animators = []; - this.currentStates = []; - this.states = {}; - this._init(props); - } - Element.prototype._init = function (props) { - this.attr(props); - }; - Element.prototype.drift = function (dx, dy, e) { - switch (this.draggable) { - case 'horizontal': - dy = 0; - break; - case 'vertical': - dx = 0; - break; - } - var m = this.transform; - if (!m) { - m = this.transform = [1, 0, 0, 1, 0, 0]; - } - m[4] += dx; - m[5] += dy; - this.decomposeTransform(); - this.markRedraw(); - }; - Element.prototype.beforeUpdate = function () { }; - Element.prototype.afterUpdate = function () { }; - Element.prototype.update = function () { - this.updateTransform(); - if (this.__dirty) { - this.updateInnerText(); - } - }; - Element.prototype.updateInnerText = function (forceUpdate) { - var textEl = this._textContent; - if (textEl && (!textEl.ignore || forceUpdate)) { - if (!this.textConfig) { - this.textConfig = {}; - } - var textConfig = this.textConfig; - var isLocal = textConfig.local; - var innerTransformable = textEl.innerTransformable; - var textAlign = void 0; - var textVerticalAlign = void 0; - var textStyleChanged = false; - innerTransformable.parent = isLocal ? this : null; - var innerOrigin = false; - innerTransformable.copyTransform(textEl); - if (textConfig.position != null) { - var layoutRect = tmpBoundingRect; - if (textConfig.layoutRect) { - layoutRect.copy(textConfig.layoutRect); - } - else { - layoutRect.copy(this.getBoundingRect()); - } - if (!isLocal) { - layoutRect.applyTransform(this.transform); - } - if (this.calculateTextPosition) { - this.calculateTextPosition(tmpTextPosCalcRes, textConfig, layoutRect); - } - else { - calculateTextPosition(tmpTextPosCalcRes, textConfig, layoutRect); - } - innerTransformable.x = tmpTextPosCalcRes.x; - innerTransformable.y = tmpTextPosCalcRes.y; - textAlign = tmpTextPosCalcRes.align; - textVerticalAlign = tmpTextPosCalcRes.verticalAlign; - var textOrigin = textConfig.origin; - if (textOrigin && textConfig.rotation != null) { - var relOriginX = void 0; - var relOriginY = void 0; - if (textOrigin === 'center') { - relOriginX = layoutRect.width * 0.5; - relOriginY = layoutRect.height * 0.5; - } - else { - relOriginX = parsePercent(textOrigin[0], layoutRect.width); - relOriginY = parsePercent(textOrigin[1], layoutRect.height); - } - innerOrigin = true; - innerTransformable.originX = -innerTransformable.x + relOriginX + (isLocal ? 0 : layoutRect.x); - innerTransformable.originY = -innerTransformable.y + relOriginY + (isLocal ? 0 : layoutRect.y); - } - } - if (textConfig.rotation != null) { - innerTransformable.rotation = textConfig.rotation; - } - var textOffset = textConfig.offset; - if (textOffset) { - innerTransformable.x += textOffset[0]; - innerTransformable.y += textOffset[1]; - if (!innerOrigin) { - innerTransformable.originX = -textOffset[0]; - innerTransformable.originY = -textOffset[1]; - } - } - var isInside = textConfig.inside == null - ? (typeof textConfig.position === 'string' && textConfig.position.indexOf('inside') >= 0) - : textConfig.inside; - var innerTextDefaultStyle = this._innerTextDefaultStyle || (this._innerTextDefaultStyle = {}); - var textFill = void 0; - var textStroke = void 0; - var autoStroke = void 0; - if (isInside && this.canBeInsideText()) { - textFill = textConfig.insideFill; - textStroke = textConfig.insideStroke; - if (textFill == null || textFill === 'auto') { - textFill = this.getInsideTextFill(); - } - if (textStroke == null || textStroke === 'auto') { - textStroke = this.getInsideTextStroke(textFill); - autoStroke = true; - } - } - else { - textFill = textConfig.outsideFill; - textStroke = textConfig.outsideStroke; - if (textFill == null || textFill === 'auto') { - textFill = this.getOutsideFill(); - } - if (textStroke == null || textStroke === 'auto') { - textStroke = this.getOutsideStroke(textFill); - autoStroke = true; - } - } - textFill = textFill || '#000'; - if (textFill !== innerTextDefaultStyle.fill - || textStroke !== innerTextDefaultStyle.stroke - || autoStroke !== innerTextDefaultStyle.autoStroke - || textAlign !== innerTextDefaultStyle.align - || textVerticalAlign !== innerTextDefaultStyle.verticalAlign) { - textStyleChanged = true; - innerTextDefaultStyle.fill = textFill; - innerTextDefaultStyle.stroke = textStroke; - innerTextDefaultStyle.autoStroke = autoStroke; - innerTextDefaultStyle.align = textAlign; - innerTextDefaultStyle.verticalAlign = textVerticalAlign; - textEl.setDefaultTextStyle(innerTextDefaultStyle); - } - textEl.__dirty |= REDRAW_BIT; - if (textStyleChanged) { - textEl.dirtyStyle(true); - } - } - }; - Element.prototype.canBeInsideText = function () { - return true; - }; - Element.prototype.getInsideTextFill = function () { - return '#fff'; - }; - Element.prototype.getInsideTextStroke = function (textFill) { - return '#000'; - }; - Element.prototype.getOutsideFill = function () { - return this.__zr && this.__zr.isDarkMode() ? LIGHT_LABEL_COLOR : DARK_LABEL_COLOR; - }; - Element.prototype.getOutsideStroke = function (textFill) { - var backgroundColor = this.__zr && this.__zr.getBackgroundColor(); - var colorArr = typeof backgroundColor === 'string' && parse(backgroundColor); - if (!colorArr) { - colorArr = [255, 255, 255, 1]; - } - var alpha = colorArr[3]; - var isDark = this.__zr.isDarkMode(); - for (var i = 0; i < 3; i++) { - colorArr[i] = colorArr[i] * alpha + (isDark ? 0 : 255) * (1 - alpha); - } - colorArr[3] = 1; - return stringify(colorArr, 'rgba'); - }; - Element.prototype.traverse = function (cb, context) { }; - Element.prototype.attrKV = function (key, value) { - if (key === 'textConfig') { - this.setTextConfig(value); - } - else if (key === 'textContent') { - this.setTextContent(value); - } - else if (key === 'clipPath') { - this.setClipPath(value); - } - else if (key === 'extra') { - this.extra = this.extra || {}; - extend(this.extra, value); - } - else { - this[key] = value; - } - }; - Element.prototype.hide = function () { - this.ignore = true; - this.markRedraw(); - }; - Element.prototype.show = function () { - this.ignore = false; - this.markRedraw(); - }; - Element.prototype.attr = function (keyOrObj, value) { - if (typeof keyOrObj === 'string') { - this.attrKV(keyOrObj, value); - } - else if (isObject(keyOrObj)) { - var obj = keyOrObj; - var keysArr = keys(obj); - for (var i = 0; i < keysArr.length; i++) { - var key = keysArr[i]; - this.attrKV(key, keyOrObj[key]); - } - } - this.markRedraw(); - return this; - }; - Element.prototype.saveCurrentToNormalState = function (toState) { - this._innerSaveToNormal(toState); - var normalState = this._normalState; - for (var i = 0; i < this.animators.length; i++) { - var animator = this.animators[i]; - var fromStateTransition = animator.__fromStateTransition; - if (animator.getLoop() || fromStateTransition && fromStateTransition !== PRESERVED_NORMAL_STATE) { - continue; - } - var targetName = animator.targetName; - var target = targetName - ? normalState[targetName] : normalState; - animator.saveTo(target); - } - }; - Element.prototype._innerSaveToNormal = function (toState) { - var normalState = this._normalState; - if (!normalState) { - normalState = this._normalState = {}; - } - if (toState.textConfig && !normalState.textConfig) { - normalState.textConfig = this.textConfig; - } - this._savePrimaryToNormal(toState, normalState, PRIMARY_STATES_KEYS); - }; - Element.prototype._savePrimaryToNormal = function (toState, normalState, primaryKeys) { - for (var i = 0; i < primaryKeys.length; i++) { - var key = primaryKeys[i]; - if (toState[key] != null && !(key in normalState)) { - normalState[key] = this[key]; - } - } - }; - Element.prototype.hasState = function () { - return this.currentStates.length > 0; - }; - Element.prototype.getState = function (name) { - return this.states[name]; - }; - Element.prototype.ensureState = function (name) { - var states = this.states; - if (!states[name]) { - states[name] = {}; - } - return states[name]; - }; - Element.prototype.clearStates = function (noAnimation) { - this.useState(PRESERVED_NORMAL_STATE, false, noAnimation); - }; - Element.prototype.useState = function (stateName, keepCurrentStates, noAnimation, forceUseHoverLayer) { - var toNormalState = stateName === PRESERVED_NORMAL_STATE; - var hasStates = this.hasState(); - if (!hasStates && toNormalState) { - return; - } - var currentStates = this.currentStates; - var animationCfg = this.stateTransition; - if (indexOf(currentStates, stateName) >= 0 && (keepCurrentStates || currentStates.length === 1)) { - return; - } - var state; - if (this.stateProxy && !toNormalState) { - state = this.stateProxy(stateName); - } - if (!state) { - state = (this.states && this.states[stateName]); - } - if (!state && !toNormalState) { - logError("State " + stateName + " not exists."); - return; - } - if (!toNormalState) { - this.saveCurrentToNormalState(state); - } - var useHoverLayer = !!((state && state.hoverLayer) || forceUseHoverLayer); - if (useHoverLayer) { - this._toggleHoverLayerFlag(true); - } - this._applyStateObj(stateName, state, this._normalState, keepCurrentStates, !noAnimation && !this.__inHover && animationCfg && animationCfg.duration > 0, animationCfg); - var textContent = this._textContent; - var textGuide = this._textGuide; - if (textContent) { - textContent.useState(stateName, keepCurrentStates, noAnimation, useHoverLayer); - } - if (textGuide) { - textGuide.useState(stateName, keepCurrentStates, noAnimation, useHoverLayer); - } - if (toNormalState) { - this.currentStates = []; - this._normalState = {}; - } - else { - if (!keepCurrentStates) { - this.currentStates = [stateName]; - } - else { - this.currentStates.push(stateName); - } - } - this._updateAnimationTargets(); - this.markRedraw(); - if (!useHoverLayer && this.__inHover) { - this._toggleHoverLayerFlag(false); - this.__dirty &= ~REDRAW_BIT; - } - return state; - }; - Element.prototype.useStates = function (states, noAnimation, forceUseHoverLayer) { - if (!states.length) { - this.clearStates(); - } - else { - var stateObjects = []; - var currentStates = this.currentStates; - var len = states.length; - var notChange = len === currentStates.length; - if (notChange) { - for (var i = 0; i < len; i++) { - if (states[i] !== currentStates[i]) { - notChange = false; - break; - } - } - } - if (notChange) { - return; - } - for (var i = 0; i < len; i++) { - var stateName = states[i]; - var stateObj = void 0; - if (this.stateProxy) { - stateObj = this.stateProxy(stateName, states); - } - if (!stateObj) { - stateObj = this.states[stateName]; - } - if (stateObj) { - stateObjects.push(stateObj); - } - } - var lastStateObj = stateObjects[len - 1]; - var useHoverLayer = !!((lastStateObj && lastStateObj.hoverLayer) || forceUseHoverLayer); - if (useHoverLayer) { - this._toggleHoverLayerFlag(true); - } - var mergedState = this._mergeStates(stateObjects); - var animationCfg = this.stateTransition; - this.saveCurrentToNormalState(mergedState); - this._applyStateObj(states.join(','), mergedState, this._normalState, false, !noAnimation && !this.__inHover && animationCfg && animationCfg.duration > 0, animationCfg); - var textContent = this._textContent; - var textGuide = this._textGuide; - if (textContent) { - textContent.useStates(states, noAnimation, useHoverLayer); - } - if (textGuide) { - textGuide.useStates(states, noAnimation, useHoverLayer); - } - this._updateAnimationTargets(); - this.currentStates = states.slice(); - this.markRedraw(); - if (!useHoverLayer && this.__inHover) { - this._toggleHoverLayerFlag(false); - this.__dirty &= ~REDRAW_BIT; - } - } - }; - Element.prototype._updateAnimationTargets = function () { - for (var i = 0; i < this.animators.length; i++) { - var animator = this.animators[i]; - if (animator.targetName) { - animator.changeTarget(this[animator.targetName]); - } - } - }; - Element.prototype.removeState = function (state) { - var idx = indexOf(this.currentStates, state); - if (idx >= 0) { - var currentStates = this.currentStates.slice(); - currentStates.splice(idx, 1); - this.useStates(currentStates); - } - }; - Element.prototype.replaceState = function (oldState, newState, forceAdd) { - var currentStates = this.currentStates.slice(); - var idx = indexOf(currentStates, oldState); - var newStateExists = indexOf(currentStates, newState) >= 0; - if (idx >= 0) { - if (!newStateExists) { - currentStates[idx] = newState; - } - else { - currentStates.splice(idx, 1); - } - } - else if (forceAdd && !newStateExists) { - currentStates.push(newState); - } - this.useStates(currentStates); - }; - Element.prototype.toggleState = function (state, enable) { - if (enable) { - this.useState(state, true); - } - else { - this.removeState(state); - } - }; - Element.prototype._mergeStates = function (states) { - var mergedState = {}; - var mergedTextConfig; - for (var i = 0; i < states.length; i++) { - var state = states[i]; - extend(mergedState, state); - if (state.textConfig) { - mergedTextConfig = mergedTextConfig || {}; - extend(mergedTextConfig, state.textConfig); - } - } - if (mergedTextConfig) { - mergedState.textConfig = mergedTextConfig; - } - return mergedState; - }; - Element.prototype._applyStateObj = function (stateName, state, normalState, keepCurrentStates, transition, animationCfg) { - var needsRestoreToNormal = !(state && keepCurrentStates); - if (state && state.textConfig) { - this.textConfig = extend({}, keepCurrentStates ? this.textConfig : normalState.textConfig); - extend(this.textConfig, state.textConfig); - } - else if (needsRestoreToNormal) { - if (normalState.textConfig) { - this.textConfig = normalState.textConfig; - } - } - var transitionTarget = {}; - var hasTransition = false; - for (var i = 0; i < PRIMARY_STATES_KEYS.length; i++) { - var key = PRIMARY_STATES_KEYS[i]; - var propNeedsTransition = transition && DEFAULT_ANIMATABLE_MAP[key]; - if (state && state[key] != null) { - if (propNeedsTransition) { - hasTransition = true; - transitionTarget[key] = state[key]; - } - else { - this[key] = state[key]; - } - } - else if (needsRestoreToNormal) { - if (normalState[key] != null) { - if (propNeedsTransition) { - hasTransition = true; - transitionTarget[key] = normalState[key]; - } - else { - this[key] = normalState[key]; - } - } - } - } - if (!transition) { - for (var i = 0; i < this.animators.length; i++) { - var animator = this.animators[i]; - var targetName = animator.targetName; - if (!animator.getLoop()) { - animator.__changeFinalValue(targetName - ? (state || normalState)[targetName] - : (state || normalState)); - } - } - } - if (hasTransition) { - this._transitionState(stateName, transitionTarget, animationCfg); - } - }; - Element.prototype._attachComponent = function (componentEl) { - if (componentEl.__zr && !componentEl.__hostTarget) { - if ("development" !== 'production') { - throw new Error('Text element has been added to zrender.'); - } - return; - } - if (componentEl === this) { - if ("development" !== 'production') { - throw new Error('Recursive component attachment.'); - } - return; - } - var zr = this.__zr; - if (zr) { - componentEl.addSelfToZr(zr); - } - componentEl.__zr = zr; - componentEl.__hostTarget = this; - }; - Element.prototype._detachComponent = function (componentEl) { - if (componentEl.__zr) { - componentEl.removeSelfFromZr(componentEl.__zr); - } - componentEl.__zr = null; - componentEl.__hostTarget = null; - }; - Element.prototype.getClipPath = function () { - return this._clipPath; - }; - Element.prototype.setClipPath = function (clipPath) { - if (this._clipPath && this._clipPath !== clipPath) { - this.removeClipPath(); - } - this._attachComponent(clipPath); - this._clipPath = clipPath; - this.markRedraw(); - }; - Element.prototype.removeClipPath = function () { - var clipPath = this._clipPath; - if (clipPath) { - this._detachComponent(clipPath); - this._clipPath = null; - this.markRedraw(); - } - }; - Element.prototype.getTextContent = function () { - return this._textContent; - }; - Element.prototype.setTextContent = function (textEl) { - var previousTextContent = this._textContent; - if (previousTextContent === textEl) { - return; - } - if (previousTextContent && previousTextContent !== textEl) { - this.removeTextContent(); - } - if ("development" !== 'production') { - if (textEl.__zr && !textEl.__hostTarget) { - throw new Error('Text element has been added to zrender.'); - } - } - textEl.innerTransformable = new Transformable(); - this._attachComponent(textEl); - this._textContent = textEl; - this.markRedraw(); - }; - Element.prototype.setTextConfig = function (cfg) { - if (!this.textConfig) { - this.textConfig = {}; - } - extend(this.textConfig, cfg); - this.markRedraw(); - }; - Element.prototype.removeTextConfig = function () { - this.textConfig = null; - this.markRedraw(); - }; - Element.prototype.removeTextContent = function () { - var textEl = this._textContent; - if (textEl) { - textEl.innerTransformable = null; - this._detachComponent(textEl); - this._textContent = null; - this._innerTextDefaultStyle = null; - this.markRedraw(); - } - }; - Element.prototype.getTextGuideLine = function () { - return this._textGuide; - }; - Element.prototype.setTextGuideLine = function (guideLine) { - if (this._textGuide && this._textGuide !== guideLine) { - this.removeTextGuideLine(); - } - this._attachComponent(guideLine); - this._textGuide = guideLine; - this.markRedraw(); - }; - Element.prototype.removeTextGuideLine = function () { - var textGuide = this._textGuide; - if (textGuide) { - this._detachComponent(textGuide); - this._textGuide = null; - this.markRedraw(); - } - }; - Element.prototype.markRedraw = function () { - this.__dirty |= REDRAW_BIT; - var zr = this.__zr; - if (zr) { - if (this.__inHover) { - zr.refreshHover(); - } - else { - zr.refresh(); - } - } - if (this.__hostTarget) { - this.__hostTarget.markRedraw(); - } - }; - Element.prototype.dirty = function () { - this.markRedraw(); - }; - Element.prototype._toggleHoverLayerFlag = function (inHover) { - this.__inHover = inHover; - var textContent = this._textContent; - var textGuide = this._textGuide; - if (textContent) { - textContent.__inHover = inHover; - } - if (textGuide) { - textGuide.__inHover = inHover; - } - }; - Element.prototype.addSelfToZr = function (zr) { - if (this.__zr === zr) { - return; - } - this.__zr = zr; - var animators = this.animators; - if (animators) { - for (var i = 0; i < animators.length; i++) { - zr.animation.addAnimator(animators[i]); - } - } - if (this._clipPath) { - this._clipPath.addSelfToZr(zr); - } - if (this._textContent) { - this._textContent.addSelfToZr(zr); - } - if (this._textGuide) { - this._textGuide.addSelfToZr(zr); - } - }; - Element.prototype.removeSelfFromZr = function (zr) { - if (!this.__zr) { - return; - } - this.__zr = null; - var animators = this.animators; - if (animators) { - for (var i = 0; i < animators.length; i++) { - zr.animation.removeAnimator(animators[i]); - } - } - if (this._clipPath) { - this._clipPath.removeSelfFromZr(zr); - } - if (this._textContent) { - this._textContent.removeSelfFromZr(zr); - } - if (this._textGuide) { - this._textGuide.removeSelfFromZr(zr); - } - }; - Element.prototype.animate = function (key, loop, allowDiscreteAnimation) { - var target = key ? this[key] : this; - if ("development" !== 'production') { - if (!target) { - logError('Property "' - + key - + '" is not existed in element ' - + this.id); - return; - } - } - var animator = new Animator(target, loop, allowDiscreteAnimation); - key && (animator.targetName = key); - this.addAnimator(animator, key); - return animator; - }; - Element.prototype.addAnimator = function (animator, key) { - var zr = this.__zr; - var el = this; - animator.during(function () { - el.updateDuringAnimation(key); - }).done(function () { - var animators = el.animators; - var idx = indexOf(animators, animator); - if (idx >= 0) { - animators.splice(idx, 1); - } - }); - this.animators.push(animator); - if (zr) { - zr.animation.addAnimator(animator); - } - zr && zr.wakeUp(); - }; - Element.prototype.updateDuringAnimation = function (key) { - this.markRedraw(); - }; - Element.prototype.stopAnimation = function (scope, forwardToLast) { - var animators = this.animators; - var len = animators.length; - var leftAnimators = []; - for (var i = 0; i < len; i++) { - var animator = animators[i]; - if (!scope || scope === animator.scope) { - animator.stop(forwardToLast); - } - else { - leftAnimators.push(animator); - } - } - this.animators = leftAnimators; - return this; - }; - Element.prototype.animateTo = function (target, cfg, animationProps) { - animateTo(this, target, cfg, animationProps); - }; - Element.prototype.animateFrom = function (target, cfg, animationProps) { - animateTo(this, target, cfg, animationProps, true); - }; - Element.prototype._transitionState = function (stateName, target, cfg, animationProps) { - var animators = animateTo(this, target, cfg, animationProps); - for (var i = 0; i < animators.length; i++) { - animators[i].__fromStateTransition = stateName; - } - }; - Element.prototype.getBoundingRect = function () { - return null; - }; - Element.prototype.getPaintRect = function () { - return null; - }; - Element.initDefaultProps = (function () { - var elProto = Element.prototype; - elProto.type = 'element'; - elProto.name = ''; - elProto.ignore = - elProto.silent = - elProto.isGroup = - elProto.draggable = - elProto.dragging = - elProto.ignoreClip = - elProto.__inHover = false; - elProto.__dirty = REDRAW_BIT; - var logs = {}; - function logDeprecatedError(key, xKey, yKey) { - if (!logs[key + xKey + yKey]) { - console.warn("DEPRECATED: '" + key + "' has been deprecated. use '" + xKey + "', '" + yKey + "' instead"); - logs[key + xKey + yKey] = true; - } - } - function createLegacyProperty(key, privateKey, xKey, yKey) { - Object.defineProperty(elProto, key, { - get: function () { - if ("development" !== 'production') { - logDeprecatedError(key, xKey, yKey); - } - if (!this[privateKey]) { - var pos = this[privateKey] = []; - enhanceArray(this, pos); - } - return this[privateKey]; - }, - set: function (pos) { - if ("development" !== 'production') { - logDeprecatedError(key, xKey, yKey); - } - this[xKey] = pos[0]; - this[yKey] = pos[1]; - this[privateKey] = pos; - enhanceArray(this, pos); - } - }); - function enhanceArray(self, pos) { - Object.defineProperty(pos, 0, { - get: function () { - return self[xKey]; - }, - set: function (val) { - self[xKey] = val; - } - }); - Object.defineProperty(pos, 1, { - get: function () { - return self[yKey]; - }, - set: function (val) { - self[yKey] = val; - } - }); - } - } - if (Object.defineProperty) { - createLegacyProperty('position', '_legacyPos', 'x', 'y'); - createLegacyProperty('scale', '_legacyScale', 'scaleX', 'scaleY'); - createLegacyProperty('origin', '_legacyOrigin', 'originX', 'originY'); - } - })(); - return Element; - }()); - mixin(Element, Eventful); - mixin(Element, Transformable); - function animateTo(animatable, target, cfg, animationProps, reverse) { - cfg = cfg || {}; - var animators = []; - animateToShallow(animatable, '', animatable, target, cfg, animationProps, animators, reverse); - var finishCount = animators.length; - var doneHappened = false; - var cfgDone = cfg.done; - var cfgAborted = cfg.aborted; - var doneCb = function () { - doneHappened = true; - finishCount--; - if (finishCount <= 0) { - doneHappened - ? (cfgDone && cfgDone()) - : (cfgAborted && cfgAborted()); - } - }; - var abortedCb = function () { - finishCount--; - if (finishCount <= 0) { - doneHappened - ? (cfgDone && cfgDone()) - : (cfgAborted && cfgAborted()); - } - }; - if (!finishCount) { - cfgDone && cfgDone(); - } - if (animators.length > 0 && cfg.during) { - animators[0].during(function (target, percent) { - cfg.during(percent); - }); - } - for (var i = 0; i < animators.length; i++) { - var animator = animators[i]; - if (doneCb) { - animator.done(doneCb); - } - if (abortedCb) { - animator.aborted(abortedCb); - } - if (cfg.force) { - animator.duration(cfg.duration); - } - animator.start(cfg.easing); - } - return animators; - } - function copyArrShallow(source, target, len) { - for (var i = 0; i < len; i++) { - source[i] = target[i]; - } - } - function is2DArray(value) { - return isArrayLike(value[0]); - } - function copyValue(target, source, key) { - if (isArrayLike(source[key])) { - if (!isArrayLike(target[key])) { - target[key] = []; - } - if (isTypedArray(source[key])) { - var len = source[key].length; - if (target[key].length !== len) { - target[key] = new (source[key].constructor)(len); - copyArrShallow(target[key], source[key], len); - } - } - else { - var sourceArr = source[key]; - var targetArr = target[key]; - var len0 = sourceArr.length; - if (is2DArray(sourceArr)) { - var len1 = sourceArr[0].length; - for (var i = 0; i < len0; i++) { - if (!targetArr[i]) { - targetArr[i] = Array.prototype.slice.call(sourceArr[i]); - } - else { - copyArrShallow(targetArr[i], sourceArr[i], len1); - } - } - } - else { - copyArrShallow(targetArr, sourceArr, len0); - } - targetArr.length = sourceArr.length; - } - } - else { - target[key] = source[key]; - } - } - function isValueSame(val1, val2) { - return val1 === val2 - || isArrayLike(val1) && isArrayLike(val2) && is1DArraySame(val1, val2); - } - function is1DArraySame(arr0, arr1) { - var len = arr0.length; - if (len !== arr1.length) { - return false; - } - for (var i = 0; i < len; i++) { - if (arr0[i] !== arr1[i]) { - return false; - } - } - return true; - } - function animateToShallow(animatable, topKey, animateObj, target, cfg, animationProps, animators, reverse) { - var targetKeys = keys(target); - var duration = cfg.duration; - var delay = cfg.delay; - var additive = cfg.additive; - var setToFinal = cfg.setToFinal; - var animateAll = !isObject(animationProps); - var existsAnimators = animatable.animators; - var animationKeys = []; - for (var k = 0; k < targetKeys.length; k++) { - var innerKey = targetKeys[k]; - var targetVal = target[innerKey]; - if (targetVal != null && animateObj[innerKey] != null - && (animateAll || animationProps[innerKey])) { - if (isObject(targetVal) - && !isArrayLike(targetVal) - && !isGradientObject(targetVal)) { - if (topKey) { - if (!reverse) { - animateObj[innerKey] = targetVal; - animatable.updateDuringAnimation(topKey); - } - continue; - } - animateToShallow(animatable, innerKey, animateObj[innerKey], targetVal, cfg, animationProps && animationProps[innerKey], animators, reverse); - } - else { - animationKeys.push(innerKey); - } - } - else if (!reverse) { - animateObj[innerKey] = targetVal; - animatable.updateDuringAnimation(topKey); - animationKeys.push(innerKey); - } - } - var keyLen = animationKeys.length; - if (!additive && keyLen) { - for (var i = 0; i < existsAnimators.length; i++) { - var animator = existsAnimators[i]; - if (animator.targetName === topKey) { - var allAborted = animator.stopTracks(animationKeys); - if (allAborted) { - var idx = indexOf(existsAnimators, animator); - existsAnimators.splice(idx, 1); - } - } - } - } - if (!cfg.force) { - animationKeys = filter(animationKeys, function (key) { return !isValueSame(target[key], animateObj[key]); }); - keyLen = animationKeys.length; - } - if (keyLen > 0 - || (cfg.force && !animators.length)) { - var revertedSource = void 0; - var reversedTarget = void 0; - var sourceClone = void 0; - if (reverse) { - reversedTarget = {}; - if (setToFinal) { - revertedSource = {}; - } - for (var i = 0; i < keyLen; i++) { - var innerKey = animationKeys[i]; - reversedTarget[innerKey] = animateObj[innerKey]; - if (setToFinal) { - revertedSource[innerKey] = target[innerKey]; - } - else { - animateObj[innerKey] = target[innerKey]; - } - } - } - else if (setToFinal) { - sourceClone = {}; - for (var i = 0; i < keyLen; i++) { - var innerKey = animationKeys[i]; - sourceClone[innerKey] = cloneValue(animateObj[innerKey]); - copyValue(animateObj, target, innerKey); - } - } - var animator = new Animator(animateObj, false, false, additive ? filter(existsAnimators, function (animator) { return animator.targetName === topKey; }) : null); - animator.targetName = topKey; - if (cfg.scope) { - animator.scope = cfg.scope; - } - if (setToFinal && revertedSource) { - animator.whenWithKeys(0, revertedSource, animationKeys); - } - if (sourceClone) { - animator.whenWithKeys(0, sourceClone, animationKeys); - } - animator.whenWithKeys(duration == null ? 500 : duration, reverse ? reversedTarget : target, animationKeys).delay(delay || 0); - animatable.addAnimator(animator, topKey); - animators.push(animator); - } - } - - var Group = (function (_super) { - __extends(Group, _super); - function Group(opts) { - var _this = _super.call(this) || this; - _this.isGroup = true; - _this._children = []; - _this.attr(opts); - return _this; - } - Group.prototype.childrenRef = function () { - return this._children; - }; - Group.prototype.children = function () { - return this._children.slice(); - }; - Group.prototype.childAt = function (idx) { - return this._children[idx]; - }; - Group.prototype.childOfName = function (name) { - var children = this._children; - for (var i = 0; i < children.length; i++) { - if (children[i].name === name) { - return children[i]; - } - } - }; - Group.prototype.childCount = function () { - return this._children.length; - }; - Group.prototype.add = function (child) { - if (child) { - if (child !== this && child.parent !== this) { - this._children.push(child); - this._doAdd(child); - } - if ("development" !== 'production') { - if (child.__hostTarget) { - throw 'This elemenet has been used as an attachment'; - } - } - } - return this; - }; - Group.prototype.addBefore = function (child, nextSibling) { - if (child && child !== this && child.parent !== this - && nextSibling && nextSibling.parent === this) { - var children = this._children; - var idx = children.indexOf(nextSibling); - if (idx >= 0) { - children.splice(idx, 0, child); - this._doAdd(child); - } - } - return this; - }; - Group.prototype.replace = function (oldChild, newChild) { - var idx = indexOf(this._children, oldChild); - if (idx >= 0) { - this.replaceAt(newChild, idx); - } - return this; - }; - Group.prototype.replaceAt = function (child, index) { - var children = this._children; - var old = children[index]; - if (child && child !== this && child.parent !== this && child !== old) { - children[index] = child; - old.parent = null; - var zr = this.__zr; - if (zr) { - old.removeSelfFromZr(zr); - } - this._doAdd(child); - } - return this; - }; - Group.prototype._doAdd = function (child) { - if (child.parent) { - child.parent.remove(child); - } - child.parent = this; - var zr = this.__zr; - if (zr && zr !== child.__zr) { - child.addSelfToZr(zr); - } - zr && zr.refresh(); - }; - Group.prototype.remove = function (child) { - var zr = this.__zr; - var children = this._children; - var idx = indexOf(children, child); - if (idx < 0) { - return this; - } - children.splice(idx, 1); - child.parent = null; - if (zr) { - child.removeSelfFromZr(zr); - } - zr && zr.refresh(); - return this; - }; - Group.prototype.removeAll = function () { - var children = this._children; - var zr = this.__zr; - for (var i = 0; i < children.length; i++) { - var child = children[i]; - if (zr) { - child.removeSelfFromZr(zr); - } - child.parent = null; - } - children.length = 0; - return this; - }; - Group.prototype.eachChild = function (cb, context) { - var children = this._children; - for (var i = 0; i < children.length; i++) { - var child = children[i]; - cb.call(context, child, i); - } - return this; - }; - Group.prototype.traverse = function (cb, context) { - for (var i = 0; i < this._children.length; i++) { - var child = this._children[i]; - var stopped = cb.call(context, child); - if (child.isGroup && !stopped) { - child.traverse(cb, context); - } - } - return this; - }; - Group.prototype.addSelfToZr = function (zr) { - _super.prototype.addSelfToZr.call(this, zr); - for (var i = 0; i < this._children.length; i++) { - var child = this._children[i]; - child.addSelfToZr(zr); - } - }; - Group.prototype.removeSelfFromZr = function (zr) { - _super.prototype.removeSelfFromZr.call(this, zr); - for (var i = 0; i < this._children.length; i++) { - var child = this._children[i]; - child.removeSelfFromZr(zr); - } - }; - Group.prototype.getBoundingRect = function (includeChildren) { - var tmpRect = new BoundingRect(0, 0, 0, 0); - var children = includeChildren || this._children; - var tmpMat = []; - var rect = null; - for (var i = 0; i < children.length; i++) { - var child = children[i]; - if (child.ignore || child.invisible) { - continue; - } - var childRect = child.getBoundingRect(); - var transform = child.getLocalTransform(tmpMat); - if (transform) { - BoundingRect.applyTransform(tmpRect, childRect, transform); - rect = rect || tmpRect.clone(); - rect.union(tmpRect); - } - else { - rect = rect || childRect.clone(); - rect.union(childRect); - } - } - return rect || tmpRect; - }; - return Group; - }(Element)); - Group.prototype.type = 'group'; - - /*! - * ZRender, a high performance 2d drawing library. - * - * Copyright (c) 2013, Baidu Inc. - * All rights reserved. - * - * LICENSE - * https://github.com/ecomfe/zrender/blob/master/LICENSE.txt - */ - var painterCtors = {}; - var instances = {}; - function delInstance(id) { - delete instances[id]; - } - function isDarkMode(backgroundColor) { - if (!backgroundColor) { - return false; - } - if (typeof backgroundColor === 'string') { - return lum(backgroundColor, 1) < DARK_MODE_THRESHOLD; - } - else if (backgroundColor.colorStops) { - var colorStops = backgroundColor.colorStops; - var totalLum = 0; - var len = colorStops.length; - for (var i = 0; i < len; i++) { - totalLum += lum(colorStops[i].color, 1); - } - totalLum /= len; - return totalLum < DARK_MODE_THRESHOLD; - } - return false; - } - var ZRender = (function () { - function ZRender(id, dom, opts) { - var _this = this; - this._sleepAfterStill = 10; - this._stillFrameAccum = 0; - this._needsRefresh = true; - this._needsRefreshHover = true; - this._darkMode = false; - opts = opts || {}; - this.dom = dom; - this.id = id; - var storage = new Storage(); - var rendererType = opts.renderer || 'canvas'; - if (!painterCtors[rendererType]) { - rendererType = keys(painterCtors)[0]; - } - if ("development" !== 'production') { - if (!painterCtors[rendererType]) { - throw new Error("Renderer '" + rendererType + "' is not imported. Please import it first."); - } - } - opts.useDirtyRect = opts.useDirtyRect == null - ? false - : opts.useDirtyRect; - var painter = new painterCtors[rendererType](dom, storage, opts, id); - var ssrMode = opts.ssr || painter.ssrOnly; - this.storage = storage; - this.painter = painter; - var handerProxy = (!env.node && !env.worker && !ssrMode) - ? new HandlerDomProxy(painter.getViewportRoot(), painter.root) - : null; - var useCoarsePointer = opts.useCoarsePointer; - var usePointerSize = (useCoarsePointer == null || useCoarsePointer === 'auto') - ? env.touchEventsSupported - : !!useCoarsePointer; - var defaultPointerSize = 44; - var pointerSize; - if (usePointerSize) { - pointerSize = retrieve2(opts.pointerSize, defaultPointerSize); - } - this.handler = new Handler(storage, painter, handerProxy, painter.root, pointerSize); - this.animation = new Animation({ - stage: { - update: ssrMode ? null : function () { return _this._flush(true); } - } - }); - if (!ssrMode) { - this.animation.start(); - } - } - ZRender.prototype.add = function (el) { - if (!el) { - return; - } - this.storage.addRoot(el); - el.addSelfToZr(this); - this.refresh(); - }; - ZRender.prototype.remove = function (el) { - if (!el) { - return; - } - this.storage.delRoot(el); - el.removeSelfFromZr(this); - this.refresh(); - }; - ZRender.prototype.configLayer = function (zLevel, config) { - if (this.painter.configLayer) { - this.painter.configLayer(zLevel, config); - } - this.refresh(); - }; - ZRender.prototype.setBackgroundColor = function (backgroundColor) { - if (this.painter.setBackgroundColor) { - this.painter.setBackgroundColor(backgroundColor); - } - this.refresh(); - this._backgroundColor = backgroundColor; - this._darkMode = isDarkMode(backgroundColor); - }; - ZRender.prototype.getBackgroundColor = function () { - return this._backgroundColor; - }; - ZRender.prototype.setDarkMode = function (darkMode) { - this._darkMode = darkMode; - }; - ZRender.prototype.isDarkMode = function () { - return this._darkMode; - }; - ZRender.prototype.refreshImmediately = function (fromInside) { - if (!fromInside) { - this.animation.update(true); - } - this._needsRefresh = false; - this.painter.refresh(); - this._needsRefresh = false; - }; - ZRender.prototype.refresh = function () { - this._needsRefresh = true; - this.animation.start(); - }; - ZRender.prototype.flush = function () { - this._flush(false); - }; - ZRender.prototype._flush = function (fromInside) { - var triggerRendered; - var start = getTime(); - if (this._needsRefresh) { - triggerRendered = true; - this.refreshImmediately(fromInside); - } - if (this._needsRefreshHover) { - triggerRendered = true; - this.refreshHoverImmediately(); - } - var end = getTime(); - if (triggerRendered) { - this._stillFrameAccum = 0; - this.trigger('rendered', { - elapsedTime: end - start - }); - } - else if (this._sleepAfterStill > 0) { - this._stillFrameAccum++; - if (this._stillFrameAccum > this._sleepAfterStill) { - this.animation.stop(); - } - } - }; - ZRender.prototype.setSleepAfterStill = function (stillFramesCount) { - this._sleepAfterStill = stillFramesCount; - }; - ZRender.prototype.wakeUp = function () { - this.animation.start(); - this._stillFrameAccum = 0; - }; - ZRender.prototype.refreshHover = function () { - this._needsRefreshHover = true; - }; - ZRender.prototype.refreshHoverImmediately = function () { - this._needsRefreshHover = false; - if (this.painter.refreshHover && this.painter.getType() === 'canvas') { - this.painter.refreshHover(); - } - }; - ZRender.prototype.resize = function (opts) { - opts = opts || {}; - this.painter.resize(opts.width, opts.height); - this.handler.resize(); - }; - ZRender.prototype.clearAnimation = function () { - this.animation.clear(); - }; - ZRender.prototype.getWidth = function () { - return this.painter.getWidth(); - }; - ZRender.prototype.getHeight = function () { - return this.painter.getHeight(); - }; - ZRender.prototype.setCursorStyle = function (cursorStyle) { - this.handler.setCursorStyle(cursorStyle); - }; - ZRender.prototype.findHover = function (x, y) { - return this.handler.findHover(x, y); - }; - ZRender.prototype.on = function (eventName, eventHandler, context) { - this.handler.on(eventName, eventHandler, context); - return this; - }; - ZRender.prototype.off = function (eventName, eventHandler) { - this.handler.off(eventName, eventHandler); - }; - ZRender.prototype.trigger = function (eventName, event) { - this.handler.trigger(eventName, event); - }; - ZRender.prototype.clear = function () { - var roots = this.storage.getRoots(); - for (var i = 0; i < roots.length; i++) { - if (roots[i] instanceof Group) { - roots[i].removeSelfFromZr(this); - } - } - this.storage.delAllRoots(); - this.painter.clear(); - }; - ZRender.prototype.dispose = function () { - this.animation.stop(); - this.clear(); - this.storage.dispose(); - this.painter.dispose(); - this.handler.dispose(); - this.animation = - this.storage = - this.painter = - this.handler = null; - delInstance(this.id); - }; - return ZRender; - }()); - function init(dom, opts) { - var zr = new ZRender(guid(), dom, opts); - instances[zr.id] = zr; - return zr; - } - function dispose(zr) { - zr.dispose(); - } - function disposeAll() { - for (var key in instances) { - if (instances.hasOwnProperty(key)) { - instances[key].dispose(); - } - } - instances = {}; - } - function getInstance(id) { - return instances[id]; - } - function registerPainter(name, Ctor) { - painterCtors[name] = Ctor; - } - var version = '5.4.3'; - - var zrender = /*#__PURE__*/Object.freeze({ - __proto__: null, - init: init, - dispose: dispose, - disposeAll: disposeAll, - getInstance: getInstance, - registerPainter: registerPainter, - version: version - }); - - var RADIAN_EPSILON = 1e-4; // Although chrome already enlarge this number to 100 for `toFixed`, but - // we sill follow the spec for compatibility. - - var ROUND_SUPPORTED_PRECISION_MAX = 20; - - function _trim(str) { - return str.replace(/^\s+|\s+$/g, ''); - } - /** - * Linear mapping a value from domain to range - * @param val - * @param domain Domain extent domain[0] can be bigger than domain[1] - * @param range Range extent range[0] can be bigger than range[1] - * @param clamp Default to be false - */ - - - function linearMap(val, domain, range, clamp) { - var d0 = domain[0]; - var d1 = domain[1]; - var r0 = range[0]; - var r1 = range[1]; - var subDomain = d1 - d0; - var subRange = r1 - r0; - - if (subDomain === 0) { - return subRange === 0 ? r0 : (r0 + r1) / 2; - } // Avoid accuracy problem in edge, such as - // 146.39 - 62.83 === 83.55999999999999. - // See echarts/test/ut/spec/util/number.js#linearMap#accuracyError - // It is a little verbose for efficiency considering this method - // is a hotspot. - - - if (clamp) { - if (subDomain > 0) { - if (val <= d0) { - return r0; - } else if (val >= d1) { - return r1; - } - } else { - if (val >= d0) { - return r0; - } else if (val <= d1) { - return r1; - } - } - } else { - if (val === d0) { - return r0; - } - - if (val === d1) { - return r1; - } - } - - return (val - d0) / subDomain * subRange + r0; - } - /** - * Convert a percent string to absolute number. - * Returns NaN if percent is not a valid string or number - */ - - function parsePercent$1(percent, all) { - switch (percent) { - case 'center': - case 'middle': - percent = '50%'; - break; - - case 'left': - case 'top': - percent = '0%'; - break; - - case 'right': - case 'bottom': - percent = '100%'; - break; - } - - if (isString(percent)) { - if (_trim(percent).match(/%$/)) { - return parseFloat(percent) / 100 * all; - } - - return parseFloat(percent); - } - - return percent == null ? NaN : +percent; - } - function round(x, precision, returnStr) { - if (precision == null) { - precision = 10; - } // Avoid range error - - - precision = Math.min(Math.max(0, precision), ROUND_SUPPORTED_PRECISION_MAX); // PENDING: 1.005.toFixed(2) is '1.00' rather than '1.01' - - x = (+x).toFixed(precision); - return returnStr ? x : +x; - } - /** - * Inplacd asc sort arr. - * The input arr will be modified. - */ - - function asc(arr) { - arr.sort(function (a, b) { - return a - b; - }); - return arr; - } - /** - * Get precision. - */ - - function getPrecision(val) { - val = +val; - - if (isNaN(val)) { - return 0; - } // It is much faster than methods converting number to string as follows - // let tmp = val.toString(); - // return tmp.length - 1 - tmp.indexOf('.'); - // especially when precision is low - // Notice: - // (1) If the loop count is over about 20, it is slower than `getPrecisionSafe`. - // (see https://jsbench.me/2vkpcekkvw/1) - // (2) If the val is less than for example 1e-15, the result may be incorrect. - // (see test/ut/spec/util/number.test.ts `getPrecision_equal_random`) - - - if (val > 1e-14) { - var e = 1; - - for (var i = 0; i < 15; i++, e *= 10) { - if (Math.round(val * e) / e === val) { - return i; - } - } - } - - return getPrecisionSafe(val); - } - /** - * Get precision with slow but safe method - */ - - function getPrecisionSafe(val) { - // toLowerCase for: '3.4E-12' - var str = val.toString().toLowerCase(); // Consider scientific notation: '3.4e-12' '3.4e+12' - - var eIndex = str.indexOf('e'); - var exp = eIndex > 0 ? +str.slice(eIndex + 1) : 0; - var significandPartLen = eIndex > 0 ? eIndex : str.length; - var dotIndex = str.indexOf('.'); - var decimalPartLen = dotIndex < 0 ? 0 : significandPartLen - 1 - dotIndex; - return Math.max(0, decimalPartLen - exp); - } - /** - * Minimal dicernible data precisioin according to a single pixel. - */ - - function getPixelPrecision(dataExtent, pixelExtent) { - var log = Math.log; - var LN10 = Math.LN10; - var dataQuantity = Math.floor(log(dataExtent[1] - dataExtent[0]) / LN10); - var sizeQuantity = Math.round(log(Math.abs(pixelExtent[1] - pixelExtent[0])) / LN10); // toFixed() digits argument must be between 0 and 20. - - var precision = Math.min(Math.max(-dataQuantity + sizeQuantity, 0), 20); - return !isFinite(precision) ? 20 : precision; - } - /** - * Get a data of given precision, assuring the sum of percentages - * in valueList is 1. - * The largest remainder method is used. - * https://en.wikipedia.org/wiki/Largest_remainder_method - * - * @param valueList a list of all data - * @param idx index of the data to be processed in valueList - * @param precision integer number showing digits of precision - * @return percent ranging from 0 to 100 - */ - - function getPercentWithPrecision(valueList, idx, precision) { - if (!valueList[idx]) { - return 0; - } - - var seats = getPercentSeats(valueList, precision); - return seats[idx] || 0; - } - /** - * Get a data of given precision, assuring the sum of percentages - * in valueList is 1. - * The largest remainder method is used. - * https://en.wikipedia.org/wiki/Largest_remainder_method - * - * @param valueList a list of all data - * @param precision integer number showing digits of precision - * @return {Array<number>} - */ - - function getPercentSeats(valueList, precision) { - var sum = reduce(valueList, function (acc, val) { - return acc + (isNaN(val) ? 0 : val); - }, 0); - - if (sum === 0) { - return []; - } - - var digits = Math.pow(10, precision); - var votesPerQuota = map(valueList, function (val) { - return (isNaN(val) ? 0 : val) / sum * digits * 100; - }); - var targetSeats = digits * 100; - var seats = map(votesPerQuota, function (votes) { - // Assign automatic seats. - return Math.floor(votes); - }); - var currentSum = reduce(seats, function (acc, val) { - return acc + val; - }, 0); - var remainder = map(votesPerQuota, function (votes, idx) { - return votes - seats[idx]; - }); // Has remainding votes. - - while (currentSum < targetSeats) { - // Find next largest remainder. - var max = Number.NEGATIVE_INFINITY; - var maxId = null; - - for (var i = 0, len = remainder.length; i < len; ++i) { - if (remainder[i] > max) { - max = remainder[i]; - maxId = i; - } - } // Add a vote to max remainder. - - - ++seats[maxId]; - remainder[maxId] = 0; - ++currentSum; - } - - return map(seats, function (seat) { - return seat / digits; - }); - } - /** - * Solve the floating point adding problem like 0.1 + 0.2 === 0.30000000000000004 - * See <http://0.30000000000000004.com/> - */ - - function addSafe(val0, val1) { - var maxPrecision = Math.max(getPrecision(val0), getPrecision(val1)); // const multiplier = Math.pow(10, maxPrecision); - // return (Math.round(val0 * multiplier) + Math.round(val1 * multiplier)) / multiplier; - - var sum = val0 + val1; // // PENDING: support more? - - return maxPrecision > ROUND_SUPPORTED_PRECISION_MAX ? sum : round(sum, maxPrecision); - } // Number.MAX_SAFE_INTEGER, ie do not support. - - var MAX_SAFE_INTEGER = 9007199254740991; - /** - * To 0 - 2 * PI, considering negative radian. - */ - - function remRadian(radian) { - var pi2 = Math.PI * 2; - return (radian % pi2 + pi2) % pi2; - } - /** - * @param {type} radian - * @return {boolean} - */ - - function isRadianAroundZero(val) { - return val > -RADIAN_EPSILON && val < RADIAN_EPSILON; - } // eslint-disable-next-line - - var TIME_REG = /^(?:(\d{4})(?:[-\/](\d{1,2})(?:[-\/](\d{1,2})(?:[T ](\d{1,2})(?::(\d{1,2})(?::(\d{1,2})(?:[.,](\d+))?)?)?(Z|[\+\-]\d\d:?\d\d)?)?)?)?)?$/; // jshint ignore:line - - /** - * @param value valid type: number | string | Date, otherwise return `new Date(NaN)` - * These values can be accepted: - * + An instance of Date, represent a time in its own time zone. - * + Or string in a subset of ISO 8601, only including: - * + only year, month, date: '2012-03', '2012-03-01', '2012-03-01 05', '2012-03-01 05:06', - * + separated with T or space: '2012-03-01T12:22:33.123', '2012-03-01 12:22:33.123', - * + time zone: '2012-03-01T12:22:33Z', '2012-03-01T12:22:33+8000', '2012-03-01T12:22:33-05:00', - * all of which will be treated as local time if time zone is not specified - * (see <https://momentjs.com/>). - * + Or other string format, including (all of which will be treated as local time): - * '2012', '2012-3-1', '2012/3/1', '2012/03/01', - * '2009/6/12 2:00', '2009/6/12 2:05:08', '2009/6/12 2:05:08.123' - * + a timestamp, which represent a time in UTC. - * @return date Never be null/undefined. If invalid, return `new Date(NaN)`. - */ - - function parseDate(value) { - if (value instanceof Date) { - return value; - } else if (isString(value)) { - // Different browsers parse date in different way, so we parse it manually. - // Some other issues: - // new Date('1970-01-01') is UTC, - // new Date('1970/01/01') and new Date('1970-1-01') is local. - // See issue #3623 - var match = TIME_REG.exec(value); - - if (!match) { - // return Invalid Date. - return new Date(NaN); - } // Use local time when no timezone offset is specified. - - - if (!match[8]) { - // match[n] can only be string or undefined. - // But take care of '12' + 1 => '121'. - return new Date(+match[1], +(match[2] || 1) - 1, +match[3] || 1, +match[4] || 0, +(match[5] || 0), +match[6] || 0, match[7] ? +match[7].substring(0, 3) : 0); - } // Timezoneoffset of Javascript Date has considered DST (Daylight Saving Time, - // https://tc39.github.io/ecma262/#sec-daylight-saving-time-adjustment). - // For example, system timezone is set as "Time Zone: America/Toronto", - // then these code will get different result: - // `new Date(1478411999999).getTimezoneOffset(); // get 240` - // `new Date(1478412000000).getTimezoneOffset(); // get 300` - // So we should not use `new Date`, but use `Date.UTC`. - else { - var hour = +match[4] || 0; - - if (match[8].toUpperCase() !== 'Z') { - hour -= +match[8].slice(0, 3); - } - - return new Date(Date.UTC(+match[1], +(match[2] || 1) - 1, +match[3] || 1, hour, +(match[5] || 0), +match[6] || 0, match[7] ? +match[7].substring(0, 3) : 0)); - } - } else if (value == null) { - return new Date(NaN); - } - - return new Date(Math.round(value)); - } - /** - * Quantity of a number. e.g. 0.1, 1, 10, 100 - * - * @param val - * @return - */ - - function quantity(val) { - return Math.pow(10, quantityExponent(val)); - } - /** - * Exponent of the quantity of a number - * e.g., 1234 equals to 1.234*10^3, so quantityExponent(1234) is 3 - * - * @param val non-negative value - * @return - */ - - function quantityExponent(val) { - if (val === 0) { - return 0; - } - - var exp = Math.floor(Math.log(val) / Math.LN10); - /** - * exp is expected to be the rounded-down result of the base-10 log of val. - * But due to the precision loss with Math.log(val), we need to restore it - * using 10^exp to make sure we can get val back from exp. #11249 - */ - - if (val / Math.pow(10, exp) >= 10) { - exp++; - } - - return exp; - } - /** - * find a “nice” number approximately equal to x. Round the number if round = true, - * take ceiling if round = false. The primary observation is that the “nicest” - * numbers in decimal are 1, 2, and 5, and all power-of-ten multiples of these numbers. - * - * See "Nice Numbers for Graph Labels" of Graphic Gems. - * - * @param val Non-negative value. - * @param round - * @return Niced number - */ - - function nice(val, round) { - var exponent = quantityExponent(val); - var exp10 = Math.pow(10, exponent); - var f = val / exp10; // 1 <= f < 10 - - var nf; - - if (round) { - if (f < 1.5) { - nf = 1; - } else if (f < 2.5) { - nf = 2; - } else if (f < 4) { - nf = 3; - } else if (f < 7) { - nf = 5; - } else { - nf = 10; - } - } else { - if (f < 1) { - nf = 1; - } else if (f < 2) { - nf = 2; - } else if (f < 3) { - nf = 3; - } else if (f < 5) { - nf = 5; - } else { - nf = 10; - } - } - - val = nf * exp10; // Fix 3 * 0.1 === 0.30000000000000004 issue (see IEEE 754). - // 20 is the uppper bound of toFixed. - - return exponent >= -20 ? +val.toFixed(exponent < 0 ? -exponent : 0) : val; - } - /** - * This code was copied from "d3.js" - * <https://github.com/d3/d3/blob/9cc9a875e636a1dcf36cc1e07bdf77e1ad6e2c74/src/arrays/quantile.js>. - * See the license statement at the head of this file. - * @param ascArr - */ - - function quantile(ascArr, p) { - var H = (ascArr.length - 1) * p + 1; - var h = Math.floor(H); - var v = +ascArr[h - 1]; - var e = H - h; - return e ? v + e * (ascArr[h] - v) : v; - } - /** - * Order intervals asc, and split them when overlap. - * expect(numberUtil.reformIntervals([ - * {interval: [18, 62], close: [1, 1]}, - * {interval: [-Infinity, -70], close: [0, 0]}, - * {interval: [-70, -26], close: [1, 1]}, - * {interval: [-26, 18], close: [1, 1]}, - * {interval: [62, 150], close: [1, 1]}, - * {interval: [106, 150], close: [1, 1]}, - * {interval: [150, Infinity], close: [0, 0]} - * ])).toEqual([ - * {interval: [-Infinity, -70], close: [0, 0]}, - * {interval: [-70, -26], close: [1, 1]}, - * {interval: [-26, 18], close: [0, 1]}, - * {interval: [18, 62], close: [0, 1]}, - * {interval: [62, 150], close: [0, 1]}, - * {interval: [150, Infinity], close: [0, 0]} - * ]); - * @param list, where `close` mean open or close - * of the interval, and Infinity can be used. - * @return The origin list, which has been reformed. - */ - - function reformIntervals(list) { - list.sort(function (a, b) { - return littleThan(a, b, 0) ? -1 : 1; - }); - var curr = -Infinity; - var currClose = 1; - - for (var i = 0; i < list.length;) { - var interval = list[i].interval; - var close_1 = list[i].close; - - for (var lg = 0; lg < 2; lg++) { - if (interval[lg] <= curr) { - interval[lg] = curr; - close_1[lg] = !lg ? 1 - currClose : 1; - } - - curr = interval[lg]; - currClose = close_1[lg]; - } - - if (interval[0] === interval[1] && close_1[0] * close_1[1] !== 1) { - list.splice(i, 1); - } else { - i++; - } - } - - return list; - - function littleThan(a, b, lg) { - return a.interval[lg] < b.interval[lg] || a.interval[lg] === b.interval[lg] && (a.close[lg] - b.close[lg] === (!lg ? 1 : -1) || !lg && littleThan(a, b, 1)); - } - } - /** - * [Numeric is defined as]: - * `parseFloat(val) == val` - * For example: - * numeric: - * typeof number except NaN, '-123', '123', '2e3', '-2e3', '011', 'Infinity', Infinity, - * and they rounded by white-spaces or line-terminal like ' -123 \n ' (see es spec) - * not-numeric: - * null, undefined, [], {}, true, false, 'NaN', NaN, '123ab', - * empty string, string with only white-spaces or line-terminal (see es spec), - * 0x12, '0x12', '-0x12', 012, '012', '-012', - * non-string, ... - * - * @test See full test cases in `test/ut/spec/util/number.js`. - * @return Must be a typeof number. If not numeric, return NaN. - */ - - function numericToNumber(val) { - var valFloat = parseFloat(val); - return valFloat == val // eslint-disable-line eqeqeq - && (valFloat !== 0 || !isString(val) || val.indexOf('x') <= 0) // For case ' 0x0 '. - ? valFloat : NaN; - } - /** - * Definition of "numeric": see `numericToNumber`. - */ - - function isNumeric(val) { - return !isNaN(numericToNumber(val)); - } - /** - * Use random base to prevent users hard code depending on - * this auto generated marker id. - * @return An positive integer. - */ - - function getRandomIdBase() { - return Math.round(Math.random() * 9); - } - /** - * Get the greatest common divisor. - * - * @param {number} a one number - * @param {number} b the other number - */ - - function getGreatestCommonDividor(a, b) { - if (b === 0) { - return a; - } - - return getGreatestCommonDividor(b, a % b); - } - /** - * Get the least common multiple. - * - * @param {number} a one number - * @param {number} b the other number - */ - - function getLeastCommonMultiple(a, b) { - if (a == null) { - return b; - } - - if (b == null) { - return a; - } - - return a * b / getGreatestCommonDividor(a, b); - } - - var ECHARTS_PREFIX = '[ECharts] '; - var storedLogs = {}; - var hasConsole = typeof console !== 'undefined' // eslint-disable-next-line - && console.warn && console.log; - - function outputLog(type, str, onlyOnce) { - if (hasConsole) { - if (onlyOnce) { - if (storedLogs[str]) { - return; - } - - storedLogs[str] = true; - } // eslint-disable-next-line - - - console[type](ECHARTS_PREFIX + str); - } - } - - function log(str, onlyOnce) { - outputLog('log', str, onlyOnce); - } - function warn(str, onlyOnce) { - outputLog('warn', str, onlyOnce); - } - function error(str, onlyOnce) { - outputLog('error', str, onlyOnce); - } - function deprecateLog(str) { - if ("development" !== 'production') { - // Not display duplicate message. - outputLog('warn', 'DEPRECATED: ' + str, true); - } - } - function deprecateReplaceLog(oldOpt, newOpt, scope) { - if ("development" !== 'production') { - deprecateLog((scope ? "[" + scope + "]" : '') + (oldOpt + " is deprecated, use " + newOpt + " instead.")); - } - } - /** - * If in __DEV__ environment, get console printable message for users hint. - * Parameters are separated by ' '. - * @usage - * makePrintable('This is an error on', someVar, someObj); - * - * @param hintInfo anything about the current execution context to hint users. - * @throws Error - */ - - function makePrintable() { - var hintInfo = []; - - for (var _i = 0; _i < arguments.length; _i++) { - hintInfo[_i] = arguments[_i]; - } - - var msg = ''; - - if ("development" !== 'production') { - // Fuzzy stringify for print. - // This code only exist in dev environment. - var makePrintableStringIfPossible_1 = function (val) { - return val === void 0 ? 'undefined' : val === Infinity ? 'Infinity' : val === -Infinity ? '-Infinity' : eqNaN(val) ? 'NaN' : val instanceof Date ? 'Date(' + val.toISOString() + ')' : isFunction(val) ? 'function () { ... }' : isRegExp(val) ? val + '' : null; - }; - - msg = map(hintInfo, function (arg) { - if (isString(arg)) { - // Print without quotation mark for some statement. - return arg; - } else { - var printableStr = makePrintableStringIfPossible_1(arg); - - if (printableStr != null) { - return printableStr; - } else if (typeof JSON !== 'undefined' && JSON.stringify) { - try { - return JSON.stringify(arg, function (n, val) { - var printableStr = makePrintableStringIfPossible_1(val); - return printableStr == null ? val : printableStr; - }); // In most cases the info object is small, so do not line break. - } catch (err) { - return '?'; - } - } else { - return '?'; - } - } - }).join(' '); - } - - return msg; - } - /** - * @throws Error - */ - - function throwError(msg) { - throw new Error(msg); - } - - function interpolateNumber$1(p0, p1, percent) { - return (p1 - p0) * percent + p0; - } - /** - * Make the name displayable. But we should - * make sure it is not duplicated with user - * specified name, so use '\0'; - */ - - - var DUMMY_COMPONENT_NAME_PREFIX = 'series\0'; - var INTERNAL_COMPONENT_ID_PREFIX = '\0_ec_\0'; - /** - * If value is not array, then translate it to array. - * @param {*} value - * @return {Array} [value] or value - */ - - function normalizeToArray(value) { - return value instanceof Array ? value : value == null ? [] : [value]; - } - /** - * Sync default option between normal and emphasis like `position` and `show` - * In case some one will write code like - * label: { - * show: false, - * position: 'outside', - * fontSize: 18 - * }, - * emphasis: { - * label: { show: true } - * } - */ - - function defaultEmphasis(opt, key, subOpts) { - // Caution: performance sensitive. - if (opt) { - opt[key] = opt[key] || {}; - opt.emphasis = opt.emphasis || {}; - opt.emphasis[key] = opt.emphasis[key] || {}; // Default emphasis option from normal - - for (var i = 0, len = subOpts.length; i < len; i++) { - var subOptName = subOpts[i]; - - if (!opt.emphasis[key].hasOwnProperty(subOptName) && opt[key].hasOwnProperty(subOptName)) { - opt.emphasis[key][subOptName] = opt[key][subOptName]; - } - } - } - } - var TEXT_STYLE_OPTIONS = ['fontStyle', 'fontWeight', 'fontSize', 'fontFamily', 'rich', 'tag', 'color', 'textBorderColor', 'textBorderWidth', 'width', 'height', 'lineHeight', 'align', 'verticalAlign', 'baseline', 'shadowColor', 'shadowBlur', 'shadowOffsetX', 'shadowOffsetY', 'textShadowColor', 'textShadowBlur', 'textShadowOffsetX', 'textShadowOffsetY', 'backgroundColor', 'borderColor', 'borderWidth', 'borderRadius', 'padding']; // modelUtil.LABEL_OPTIONS = modelUtil.TEXT_STYLE_OPTIONS.concat([ - // 'position', 'offset', 'rotate', 'origin', 'show', 'distance', 'formatter', - // 'fontStyle', 'fontWeight', 'fontSize', 'fontFamily', - // // FIXME: deprecated, check and remove it. - // 'textStyle' - // ]); - - /** - * The method does not ensure performance. - * data could be [12, 2323, {value: 223}, [1221, 23], {value: [2, 23]}] - * This helper method retrieves value from data. - */ - - function getDataItemValue(dataItem) { - return isObject(dataItem) && !isArray(dataItem) && !(dataItem instanceof Date) ? dataItem.value : dataItem; - } - /** - * data could be [12, 2323, {value: 223}, [1221, 23], {value: [2, 23]}] - * This helper method determine if dataItem has extra option besides value - */ - - function isDataItemOption(dataItem) { - return isObject(dataItem) && !(dataItem instanceof Array); // // markLine data can be array - // && !(dataItem[0] && isObject(dataItem[0]) && !(dataItem[0] instanceof Array)); - } - /** - * Mapping to existings for merge. - * - * Mode "normalMege": - * The mapping result (merge result) will keep the order of the existing - * component, rather than the order of new option. Because we should ensure - * some specified index reference (like xAxisIndex) keep work. - * And in most cases, "merge option" is used to update partial option but not - * be expected to change the order. - * - * Mode "replaceMege": - * (1) Only the id mapped components will be merged. - * (2) Other existing components (except internal components) will be removed. - * (3) Other new options will be used to create new component. - * (4) The index of the existing components will not be modified. - * That means their might be "hole" after the removal. - * The new components are created first at those available index. - * - * Mode "replaceAll": - * This mode try to support that reproduce an echarts instance from another - * echarts instance (via `getOption`) in some simple cases. - * In this scenario, the `result` index are exactly the consistent with the `newCmptOptions`, - * which ensures the component index referring (like `xAxisIndex: ?`) corrent. That is, - * the "hole" in `newCmptOptions` will also be kept. - * On the contrary, other modes try best to eliminate holes. - * PENDING: This is an experimental mode yet. - * - * @return See the comment of <MappingResult>. - */ - - function mappingToExists(existings, newCmptOptions, mode) { - var isNormalMergeMode = mode === 'normalMerge'; - var isReplaceMergeMode = mode === 'replaceMerge'; - var isReplaceAllMode = mode === 'replaceAll'; - existings = existings || []; - newCmptOptions = (newCmptOptions || []).slice(); - var existingIdIdxMap = createHashMap(); // Validate id and name on user input option. - - each(newCmptOptions, function (cmptOption, index) { - if (!isObject(cmptOption)) { - newCmptOptions[index] = null; - return; - } - - if ("development" !== 'production') { - // There is some legacy case that name is set as `false`. - // But should work normally rather than throw error. - if (cmptOption.id != null && !isValidIdOrName(cmptOption.id)) { - warnInvalidateIdOrName(cmptOption.id); - } - - if (cmptOption.name != null && !isValidIdOrName(cmptOption.name)) { - warnInvalidateIdOrName(cmptOption.name); - } - } - }); - var result = prepareResult(existings, existingIdIdxMap, mode); - - if (isNormalMergeMode || isReplaceMergeMode) { - mappingById(result, existings, existingIdIdxMap, newCmptOptions); - } - - if (isNormalMergeMode) { - mappingByName(result, newCmptOptions); - } - - if (isNormalMergeMode || isReplaceMergeMode) { - mappingByIndex(result, newCmptOptions, isReplaceMergeMode); - } else if (isReplaceAllMode) { - mappingInReplaceAllMode(result, newCmptOptions); - } - - makeIdAndName(result); // The array `result` MUST NOT contain elided items, otherwise the - // forEach will omit those items and result in incorrect result. - - return result; - } - - function prepareResult(existings, existingIdIdxMap, mode) { - var result = []; - - if (mode === 'replaceAll') { - return result; - } // Do not use native `map` to in case that the array `existings` - // contains elided items, which will be omitted. - - - for (var index = 0; index < existings.length; index++) { - var existing = existings[index]; // Because of replaceMerge, `existing` may be null/undefined. - - if (existing && existing.id != null) { - existingIdIdxMap.set(existing.id, index); - } // For non-internal-componnets: - // Mode "normalMerge": all existings kept. - // Mode "replaceMerge": all existing removed unless mapped by id. - // For internal-components: - // go with "replaceMerge" approach in both mode. - - - result.push({ - existing: mode === 'replaceMerge' || isComponentIdInternal(existing) ? null : existing, - newOption: null, - keyInfo: null, - brandNew: null - }); - } - - return result; - } - - function mappingById(result, existings, existingIdIdxMap, newCmptOptions) { - // Mapping by id if specified. - each(newCmptOptions, function (cmptOption, index) { - if (!cmptOption || cmptOption.id == null) { - return; - } - - var optionId = makeComparableKey(cmptOption.id); - var existingIdx = existingIdIdxMap.get(optionId); - - if (existingIdx != null) { - var resultItem = result[existingIdx]; - assert(!resultItem.newOption, 'Duplicated option on id "' + optionId + '".'); - resultItem.newOption = cmptOption; // In both mode, if id matched, new option will be merged to - // the existings rather than creating new component model. - - resultItem.existing = existings[existingIdx]; - newCmptOptions[index] = null; - } - }); - } - - function mappingByName(result, newCmptOptions) { - // Mapping by name if specified. - each(newCmptOptions, function (cmptOption, index) { - if (!cmptOption || cmptOption.name == null) { - return; - } - - for (var i = 0; i < result.length; i++) { - var existing = result[i].existing; - - if (!result[i].newOption // Consider name: two map to one. - // Can not match when both ids existing but different. - && existing && (existing.id == null || cmptOption.id == null) && !isComponentIdInternal(cmptOption) && !isComponentIdInternal(existing) && keyExistAndEqual('name', existing, cmptOption)) { - result[i].newOption = cmptOption; - newCmptOptions[index] = null; - return; - } - } - }); - } - - function mappingByIndex(result, newCmptOptions, brandNew) { - each(newCmptOptions, function (cmptOption) { - if (!cmptOption) { - return; - } // Find the first place that not mapped by id and not internal component (consider the "hole"). - - - var resultItem; - var nextIdx = 0; - - while ( // Be `!resultItem` only when `nextIdx >= result.length`. - (resultItem = result[nextIdx]) && ( // (1) Existing models that already have id should be able to mapped to. Because - // after mapping performed, model will always be assigned with an id if user not given. - // After that all models have id. - // (2) If new option has id, it can only set to a hole or append to the last. It should - // not be merged to the existings with different id. Because id should not be overwritten. - // (3) Name can be overwritten, because axis use name as 'show label text'. - resultItem.newOption || isComponentIdInternal(resultItem.existing) || // In mode "replaceMerge", here no not-mapped-non-internal-existing. - resultItem.existing && cmptOption.id != null && !keyExistAndEqual('id', cmptOption, resultItem.existing))) { - nextIdx++; - } - - if (resultItem) { - resultItem.newOption = cmptOption; - resultItem.brandNew = brandNew; - } else { - result.push({ - newOption: cmptOption, - brandNew: brandNew, - existing: null, - keyInfo: null - }); - } - - nextIdx++; - }); - } - - function mappingInReplaceAllMode(result, newCmptOptions) { - each(newCmptOptions, function (cmptOption) { - // The feature "reproduce" requires "hole" will also reproduced - // in case that component index referring are broken. - result.push({ - newOption: cmptOption, - brandNew: true, - existing: null, - keyInfo: null - }); - }); - } - /** - * Make id and name for mapping result (result of mappingToExists) - * into `keyInfo` field. - */ - - - function makeIdAndName(mapResult) { - // We use this id to hash component models and view instances - // in echarts. id can be specified by user, or auto generated. - // The id generation rule ensures new view instance are able - // to mapped to old instance when setOption are called in - // no-merge mode. So we generate model id by name and plus - // type in view id. - // name can be duplicated among components, which is convenient - // to specify multi components (like series) by one name. - // Ensure that each id is distinct. - var idMap = createHashMap(); - each(mapResult, function (item) { - var existing = item.existing; - existing && idMap.set(existing.id, item); - }); - each(mapResult, function (item) { - var opt = item.newOption; // Force ensure id not duplicated. - - assert(!opt || opt.id == null || !idMap.get(opt.id) || idMap.get(opt.id) === item, 'id duplicates: ' + (opt && opt.id)); - opt && opt.id != null && idMap.set(opt.id, item); - !item.keyInfo && (item.keyInfo = {}); - }); // Make name and id. - - each(mapResult, function (item, index) { - var existing = item.existing; - var opt = item.newOption; - var keyInfo = item.keyInfo; - - if (!isObject(opt)) { - return; - } // Name can be overwritten. Consider case: axis.name = '20km'. - // But id generated by name will not be changed, which affect - // only in that case: setOption with 'not merge mode' and view - // instance will be recreated, which can be accepted. - - - keyInfo.name = opt.name != null ? makeComparableKey(opt.name) : existing ? existing.name // Avoid that different series has the same name, - // because name may be used like in color pallet. - : DUMMY_COMPONENT_NAME_PREFIX + index; - - if (existing) { - keyInfo.id = makeComparableKey(existing.id); - } else if (opt.id != null) { - keyInfo.id = makeComparableKey(opt.id); - } else { - // Consider this situatoin: - // optionA: [{name: 'a'}, {name: 'a'}, {..}] - // optionB [{..}, {name: 'a'}, {name: 'a'}] - // Series with the same name between optionA and optionB - // should be mapped. - var idNum = 0; - - do { - keyInfo.id = '\0' + keyInfo.name + '\0' + idNum++; - } while (idMap.get(keyInfo.id)); - } - - idMap.set(keyInfo.id, item); - }); - } - - function keyExistAndEqual(attr, obj1, obj2) { - var key1 = convertOptionIdName(obj1[attr], null); - var key2 = convertOptionIdName(obj2[attr], null); // See `MappingExistingItem`. `id` and `name` trade string equals to number. - - return key1 != null && key2 != null && key1 === key2; - } - /** - * @return return null if not exist. - */ - - - function makeComparableKey(val) { - if ("development" !== 'production') { - if (val == null) { - throw new Error(); - } - } - - return convertOptionIdName(val, ''); - } - - function convertOptionIdName(idOrName, defaultValue) { - if (idOrName == null) { - return defaultValue; - } - - return isString(idOrName) ? idOrName : isNumber(idOrName) || isStringSafe(idOrName) ? idOrName + '' : defaultValue; - } - - function warnInvalidateIdOrName(idOrName) { - if ("development" !== 'production') { - warn('`' + idOrName + '` is invalid id or name. Must be a string or number.'); - } - } - - function isValidIdOrName(idOrName) { - return isStringSafe(idOrName) || isNumeric(idOrName); - } - - function isNameSpecified(componentModel) { - var name = componentModel.name; // Is specified when `indexOf` get -1 or > 0. - - return !!(name && name.indexOf(DUMMY_COMPONENT_NAME_PREFIX)); - } - /** - * @public - * @param {Object} cmptOption - * @return {boolean} - */ - - function isComponentIdInternal(cmptOption) { - return cmptOption && cmptOption.id != null && makeComparableKey(cmptOption.id).indexOf(INTERNAL_COMPONENT_ID_PREFIX) === 0; - } - function makeInternalComponentId(idSuffix) { - return INTERNAL_COMPONENT_ID_PREFIX + idSuffix; - } - function setComponentTypeToKeyInfo(mappingResult, mainType, componentModelCtor) { - // Set mainType and complete subType. - each(mappingResult, function (item) { - var newOption = item.newOption; - - if (isObject(newOption)) { - item.keyInfo.mainType = mainType; - item.keyInfo.subType = determineSubType(mainType, newOption, item.existing, componentModelCtor); - } - }); - } - - function determineSubType(mainType, newCmptOption, existComponent, componentModelCtor) { - var subType = newCmptOption.type ? newCmptOption.type : existComponent ? existComponent.subType // Use determineSubType only when there is no existComponent. - : componentModelCtor.determineSubType(mainType, newCmptOption); // tooltip, markline, markpoint may always has no subType - - return subType; - } - /** - * A helper for removing duplicate items between batchA and batchB, - * and in themselves, and categorize by series. - * - * @param batchA Like: [{seriesId: 2, dataIndex: [32, 4, 5]}, ...] - * @param batchB Like: [{seriesId: 2, dataIndex: [32, 4, 5]}, ...] - * @return result: [resultBatchA, resultBatchB] - */ - - - function compressBatches(batchA, batchB) { - var mapA = {}; - var mapB = {}; - makeMap(batchA || [], mapA); - makeMap(batchB || [], mapB, mapA); - return [mapToArray(mapA), mapToArray(mapB)]; - - function makeMap(sourceBatch, map, otherMap) { - for (var i = 0, len = sourceBatch.length; i < len; i++) { - var seriesId = convertOptionIdName(sourceBatch[i].seriesId, null); - - if (seriesId == null) { - return; - } - - var dataIndices = normalizeToArray(sourceBatch[i].dataIndex); - var otherDataIndices = otherMap && otherMap[seriesId]; - - for (var j = 0, lenj = dataIndices.length; j < lenj; j++) { - var dataIndex = dataIndices[j]; - - if (otherDataIndices && otherDataIndices[dataIndex]) { - otherDataIndices[dataIndex] = null; - } else { - (map[seriesId] || (map[seriesId] = {}))[dataIndex] = 1; - } - } - } - } - - function mapToArray(map, isData) { - var result = []; - - for (var i in map) { - if (map.hasOwnProperty(i) && map[i] != null) { - if (isData) { - result.push(+i); - } else { - var dataIndices = mapToArray(map[i], true); - dataIndices.length && result.push({ - seriesId: i, - dataIndex: dataIndices - }); - } - } - } - - return result; - } - } - /** - * @param payload Contains dataIndex (means rawIndex) / dataIndexInside / name - * each of which can be Array or primary type. - * @return dataIndex If not found, return undefined/null. - */ - - function queryDataIndex(data, payload) { - if (payload.dataIndexInside != null) { - return payload.dataIndexInside; - } else if (payload.dataIndex != null) { - return isArray(payload.dataIndex) ? map(payload.dataIndex, function (value) { - return data.indexOfRawIndex(value); - }) : data.indexOfRawIndex(payload.dataIndex); - } else if (payload.name != null) { - return isArray(payload.name) ? map(payload.name, function (value) { - return data.indexOfName(value); - }) : data.indexOfName(payload.name); - } - } - /** - * Enable property storage to any host object. - * Notice: Serialization is not supported. - * - * For example: - * let inner = zrUitl.makeInner(); - * - * function some1(hostObj) { - * inner(hostObj).someProperty = 1212; - * ... - * } - * function some2() { - * let fields = inner(this); - * fields.someProperty1 = 1212; - * fields.someProperty2 = 'xx'; - * ... - * } - * - * @return {Function} - */ - - function makeInner() { - var key = '__ec_inner_' + innerUniqueIndex++; - return function (hostObj) { - return hostObj[key] || (hostObj[key] = {}); - }; - } - var innerUniqueIndex = getRandomIdBase(); - /** - * The same behavior as `component.getReferringComponents`. - */ - - function parseFinder(ecModel, finderInput, opt) { - var _a = preParseFinder(finderInput, opt), - mainTypeSpecified = _a.mainTypeSpecified, - queryOptionMap = _a.queryOptionMap, - others = _a.others; - - var result = others; - var defaultMainType = opt ? opt.defaultMainType : null; - - if (!mainTypeSpecified && defaultMainType) { - queryOptionMap.set(defaultMainType, {}); - } - - queryOptionMap.each(function (queryOption, mainType) { - var queryResult = queryReferringComponents(ecModel, mainType, queryOption, { - useDefault: defaultMainType === mainType, - enableAll: opt && opt.enableAll != null ? opt.enableAll : true, - enableNone: opt && opt.enableNone != null ? opt.enableNone : true - }); - result[mainType + 'Models'] = queryResult.models; - result[mainType + 'Model'] = queryResult.models[0]; - }); - return result; - } - function preParseFinder(finderInput, opt) { - var finder; - - if (isString(finderInput)) { - var obj = {}; - obj[finderInput + 'Index'] = 0; - finder = obj; - } else { - finder = finderInput; - } - - var queryOptionMap = createHashMap(); - var others = {}; - var mainTypeSpecified = false; - each(finder, function (value, key) { - // Exclude 'dataIndex' and other illgal keys. - if (key === 'dataIndex' || key === 'dataIndexInside') { - others[key] = value; - return; - } - - var parsedKey = key.match(/^(\w+)(Index|Id|Name)$/) || []; - var mainType = parsedKey[1]; - var queryType = (parsedKey[2] || '').toLowerCase(); - - if (!mainType || !queryType || opt && opt.includeMainTypes && indexOf(opt.includeMainTypes, mainType) < 0) { - return; - } - - mainTypeSpecified = mainTypeSpecified || !!mainType; - var queryOption = queryOptionMap.get(mainType) || queryOptionMap.set(mainType, {}); - queryOption[queryType] = value; - }); - return { - mainTypeSpecified: mainTypeSpecified, - queryOptionMap: queryOptionMap, - others: others - }; - } - var SINGLE_REFERRING = { - useDefault: true, - enableAll: false, - enableNone: false - }; - var MULTIPLE_REFERRING = { - useDefault: false, - enableAll: true, - enableNone: true - }; - function queryReferringComponents(ecModel, mainType, userOption, opt) { - opt = opt || SINGLE_REFERRING; - var indexOption = userOption.index; - var idOption = userOption.id; - var nameOption = userOption.name; - var result = { - models: null, - specified: indexOption != null || idOption != null || nameOption != null - }; - - if (!result.specified) { - // Use the first as default if `useDefault`. - var firstCmpt = void 0; - result.models = opt.useDefault && (firstCmpt = ecModel.getComponent(mainType)) ? [firstCmpt] : []; - return result; - } - - if (indexOption === 'none' || indexOption === false) { - assert(opt.enableNone, '`"none"` or `false` is not a valid value on index option.'); - result.models = []; - return result; - } // `queryComponents` will return all components if - // both all of index/id/name are null/undefined. - - - if (indexOption === 'all') { - assert(opt.enableAll, '`"all"` is not a valid value on index option.'); - indexOption = idOption = nameOption = null; - } - - result.models = ecModel.queryComponents({ - mainType: mainType, - index: indexOption, - id: idOption, - name: nameOption - }); - return result; - } - function setAttribute(dom, key, value) { - dom.setAttribute ? dom.setAttribute(key, value) : dom[key] = value; - } - function getAttribute(dom, key) { - return dom.getAttribute ? dom.getAttribute(key) : dom[key]; - } - function getTooltipRenderMode(renderModeOption) { - if (renderModeOption === 'auto') { - // Using html when `document` exists, use richText otherwise - return env.domSupported ? 'html' : 'richText'; - } else { - return renderModeOption || 'html'; - } - } - /** - * Group a list by key. - */ - - function groupData(array, getKey // return key - ) { - var buckets = createHashMap(); - var keys = []; - each(array, function (item) { - var key = getKey(item); - (buckets.get(key) || (keys.push(key), buckets.set(key, []))).push(item); - }); - return { - keys: keys, - buckets: buckets - }; - } - /** - * Interpolate raw values of a series with percent - * - * @param data data - * @param labelModel label model of the text element - * @param sourceValue start value. May be null/undefined when init. - * @param targetValue end value - * @param percent 0~1 percentage; 0 uses start value while 1 uses end value - * @return interpolated values - * If `sourceValue` and `targetValue` are `number`, return `number`. - * If `sourceValue` and `targetValue` are `string`, return `string`. - * If `sourceValue` and `targetValue` are `(string | number)[]`, return `(string | number)[]`. - * Other cases do not supported. - */ - - function interpolateRawValues(data, precision, sourceValue, targetValue, percent) { - var isAutoPrecision = precision == null || precision === 'auto'; - - if (targetValue == null) { - return targetValue; - } - - if (isNumber(targetValue)) { - var value = interpolateNumber$1(sourceValue || 0, targetValue, percent); - return round(value, isAutoPrecision ? Math.max(getPrecision(sourceValue || 0), getPrecision(targetValue)) : precision); - } else if (isString(targetValue)) { - return percent < 1 ? sourceValue : targetValue; - } else { - var interpolated = []; - var leftArr = sourceValue; - var rightArr = targetValue; - var length_1 = Math.max(leftArr ? leftArr.length : 0, rightArr.length); - - for (var i = 0; i < length_1; ++i) { - var info = data.getDimensionInfo(i); // Don't interpolate ordinal dims - - if (info && info.type === 'ordinal') { - // In init, there is no `sourceValue`, but should better not to get undefined result. - interpolated[i] = (percent < 1 && leftArr ? leftArr : rightArr)[i]; - } else { - var leftVal = leftArr && leftArr[i] ? leftArr[i] : 0; - var rightVal = rightArr[i]; - var value = interpolateNumber$1(leftVal, rightVal, percent); - interpolated[i] = round(value, isAutoPrecision ? Math.max(getPrecision(leftVal), getPrecision(rightVal)) : precision); - } - } - - return interpolated; - } - } - - var TYPE_DELIMITER = '.'; - var IS_CONTAINER = '___EC__COMPONENT__CONTAINER___'; - var IS_EXTENDED_CLASS = '___EC__EXTENDED_CLASS___'; - /** - * Notice, parseClassType('') should returns {main: '', sub: ''} - * @public - */ - - function parseClassType(componentType) { - var ret = { - main: '', - sub: '' - }; - - if (componentType) { - var typeArr = componentType.split(TYPE_DELIMITER); - ret.main = typeArr[0] || ''; - ret.sub = typeArr[1] || ''; - } - - return ret; - } - /** - * @public - */ - - function checkClassType(componentType) { - assert(/^[a-zA-Z0-9_]+([.][a-zA-Z0-9_]+)?$/.test(componentType), 'componentType "' + componentType + '" illegal'); - } - - function isExtendedClass(clz) { - return !!(clz && clz[IS_EXTENDED_CLASS]); - } - /** - * Implements `ExtendableConstructor` for `rootClz`. - * - * @usage - * ```ts - * class Xxx {} - * type XxxConstructor = typeof Xxx & ExtendableConstructor - * enableClassExtend(Xxx as XxxConstructor); - * ``` - */ - - function enableClassExtend(rootClz, mandatoryMethods) { - rootClz.$constructor = rootClz; // FIXME: not necessary? - - rootClz.extend = function (proto) { - if ("development" !== 'production') { - each(mandatoryMethods, function (method) { - if (!proto[method]) { - console.warn('Method `' + method + '` should be implemented' + (proto.type ? ' in ' + proto.type : '') + '.'); - } - }); - } - - var superClass = this; - var ExtendedClass; - - if (isESClass(superClass)) { - ExtendedClass = - /** @class */ - function (_super) { - __extends(class_1, _super); - - function class_1() { - return _super.apply(this, arguments) || this; - } - - return class_1; - }(superClass); - } else { - // For backward compat, we both support ts class inheritance and this - // "extend" approach. - // The constructor should keep the same behavior as ts class inheritance: - // If this constructor/$constructor is not declared, auto invoke the super - // constructor. - // If this constructor/$constructor is declared, it is responsible for - // calling the super constructor. - ExtendedClass = function () { - (proto.$constructor || superClass).apply(this, arguments); - }; - - inherits(ExtendedClass, this); - } - - extend(ExtendedClass.prototype, proto); - ExtendedClass[IS_EXTENDED_CLASS] = true; - ExtendedClass.extend = this.extend; - ExtendedClass.superCall = superCall; - ExtendedClass.superApply = superApply; - ExtendedClass.superClass = superClass; - return ExtendedClass; - }; - } - - function isESClass(fn) { - return isFunction(fn) && /^class\s/.test(Function.prototype.toString.call(fn)); - } - /** - * A work around to both support ts extend and this extend mechanism. - * on sub-class. - * @usage - * ```ts - * class Component { ... } - * classUtil.enableClassExtend(Component); - * classUtil.enableClassManagement(Component, {registerWhenExtend: true}); - * - * class Series extends Component { ... } - * // Without calling `markExtend`, `registerWhenExtend` will not work. - * Component.markExtend(Series); - * ``` - */ - - - function mountExtend(SubClz, SupperClz) { - SubClz.extend = SupperClz.extend; - } // A random offset. - - var classBase = Math.round(Math.random() * 10); - /** - * Implements `CheckableConstructor` for `target`. - * Can not use instanceof, consider different scope by - * cross domain or es module import in ec extensions. - * Mount a method "isInstance()" to Clz. - * - * @usage - * ```ts - * class Xxx {} - * type XxxConstructor = typeof Xxx & CheckableConstructor; - * enableClassCheck(Xxx as XxxConstructor) - * ``` - */ - - function enableClassCheck(target) { - var classAttr = ['__\0is_clz', classBase++].join('_'); - target.prototype[classAttr] = true; - - if ("development" !== 'production') { - assert(!target.isInstance, 'The method "is" can not be defined.'); - } - - target.isInstance = function (obj) { - return !!(obj && obj[classAttr]); - }; - } // superCall should have class info, which can not be fetched from 'this'. - // Consider this case: - // class A has method f, - // class B inherits class A, overrides method f, f call superApply('f'), - // class C inherits class B, does not override method f, - // then when method of class C is called, dead loop occurred. - - function superCall(context, methodName) { - var args = []; - - for (var _i = 2; _i < arguments.length; _i++) { - args[_i - 2] = arguments[_i]; - } - - return this.superClass.prototype[methodName].apply(context, args); - } - - function superApply(context, methodName, args) { - return this.superClass.prototype[methodName].apply(context, args); - } - /** - * Implements `ClassManager` for `target` - * - * @usage - * ```ts - * class Xxx {} - * type XxxConstructor = typeof Xxx & ClassManager - * enableClassManagement(Xxx as XxxConstructor); - * ``` - */ - - - function enableClassManagement(target) { - /** - * Component model classes - * key: componentType, - * value: - * componentClass, when componentType is 'a' - * or Object.<subKey, componentClass>, when componentType is 'a.b' - */ - var storage = {}; - - target.registerClass = function (clz) { - // `type` should not be a "instance member". - // If using TS class, should better declared as `static type = 'series.pie'`. - // otherwise users have to mount `type` on prototype manually. - // For backward compat and enable instance visit type via `this.type`, - // we still support fetch `type` from prototype. - var componentFullType = clz.type || clz.prototype.type; - - if (componentFullType) { - checkClassType(componentFullType); // If only static type declared, we assign it to prototype mandatorily. - - clz.prototype.type = componentFullType; - var componentTypeInfo = parseClassType(componentFullType); - - if (!componentTypeInfo.sub) { - if ("development" !== 'production') { - if (storage[componentTypeInfo.main]) { - console.warn(componentTypeInfo.main + ' exists.'); - } - } - - storage[componentTypeInfo.main] = clz; - } else if (componentTypeInfo.sub !== IS_CONTAINER) { - var container = makeContainer(componentTypeInfo); - container[componentTypeInfo.sub] = clz; - } - } - - return clz; - }; - - target.getClass = function (mainType, subType, throwWhenNotFound) { - var clz = storage[mainType]; - - if (clz && clz[IS_CONTAINER]) { - clz = subType ? clz[subType] : null; - } - - if (throwWhenNotFound && !clz) { - throw new Error(!subType ? mainType + '.' + 'type should be specified.' : 'Component ' + mainType + '.' + (subType || '') + ' is used but not imported.'); - } - - return clz; - }; - - target.getClassesByMainType = function (componentType) { - var componentTypeInfo = parseClassType(componentType); - var result = []; - var obj = storage[componentTypeInfo.main]; - - if (obj && obj[IS_CONTAINER]) { - each(obj, function (o, type) { - type !== IS_CONTAINER && result.push(o); - }); - } else { - result.push(obj); - } - - return result; - }; - - target.hasClass = function (componentType) { - // Just consider componentType.main. - var componentTypeInfo = parseClassType(componentType); - return !!storage[componentTypeInfo.main]; - }; - /** - * @return Like ['aa', 'bb'], but can not be ['aa.xx'] - */ - - - target.getAllClassMainTypes = function () { - var types = []; - each(storage, function (obj, type) { - types.push(type); - }); - return types; - }; - /** - * If a main type is container and has sub types - */ - - - target.hasSubTypes = function (componentType) { - var componentTypeInfo = parseClassType(componentType); - var obj = storage[componentTypeInfo.main]; - return obj && obj[IS_CONTAINER]; - }; - - function makeContainer(componentTypeInfo) { - var container = storage[componentTypeInfo.main]; - - if (!container || !container[IS_CONTAINER]) { - container = storage[componentTypeInfo.main] = {}; - container[IS_CONTAINER] = true; - } - - return container; - } - } // /** - // * @param {string|Array.<string>} properties - // */ - // export function setReadOnly(obj, properties) { - // FIXME It seems broken in IE8 simulation of IE11 - // if (!zrUtil.isArray(properties)) { - // properties = properties != null ? [properties] : []; - // } - // zrUtil.each(properties, function (prop) { - // let value = obj[prop]; - // Object.defineProperty - // && Object.defineProperty(obj, prop, { - // value: value, writable: false - // }); - // zrUtil.isArray(obj[prop]) - // && Object.freeze - // && Object.freeze(obj[prop]); - // }); - // } - - function makeStyleMapper(properties, ignoreParent) { - // Normalize - for (var i = 0; i < properties.length; i++) { - if (!properties[i][1]) { - properties[i][1] = properties[i][0]; - } - } - - ignoreParent = ignoreParent || false; - return function (model, excludes, includes) { - var style = {}; - - for (var i = 0; i < properties.length; i++) { - var propName = properties[i][1]; - - if (excludes && indexOf(excludes, propName) >= 0 || includes && indexOf(includes, propName) < 0) { - continue; - } - - var val = model.getShallow(propName, ignoreParent); - - if (val != null) { - style[properties[i][0]] = val; - } - } // TODO Text or image? - - - return style; - }; - } - - var AREA_STYLE_KEY_MAP = [['fill', 'color'], ['shadowBlur'], ['shadowOffsetX'], ['shadowOffsetY'], ['opacity'], ['shadowColor'] // Option decal is in `DecalObject` but style.decal is in `PatternObject`. - // So do not transfer decal directly. - ]; - var getAreaStyle = makeStyleMapper(AREA_STYLE_KEY_MAP); - - var AreaStyleMixin = - /** @class */ - function () { - function AreaStyleMixin() {} - - AreaStyleMixin.prototype.getAreaStyle = function (excludes, includes) { - return getAreaStyle(this, excludes, includes); - }; - - return AreaStyleMixin; - }(); - - var globalImageCache = new LRU(50); - function findExistImage(newImageOrSrc) { - if (typeof newImageOrSrc === 'string') { - var cachedImgObj = globalImageCache.get(newImageOrSrc); - return cachedImgObj && cachedImgObj.image; - } - else { - return newImageOrSrc; - } - } - function createOrUpdateImage(newImageOrSrc, image, hostEl, onload, cbPayload) { - if (!newImageOrSrc) { - return image; - } - else if (typeof newImageOrSrc === 'string') { - if ((image && image.__zrImageSrc === newImageOrSrc) || !hostEl) { - return image; - } - var cachedImgObj = globalImageCache.get(newImageOrSrc); - var pendingWrap = { hostEl: hostEl, cb: onload, cbPayload: cbPayload }; - if (cachedImgObj) { - image = cachedImgObj.image; - !isImageReady(image) && cachedImgObj.pending.push(pendingWrap); - } - else { - image = platformApi.loadImage(newImageOrSrc, imageOnLoad, imageOnLoad); - image.__zrImageSrc = newImageOrSrc; - globalImageCache.put(newImageOrSrc, image.__cachedImgObj = { - image: image, - pending: [pendingWrap] - }); - } - return image; - } - else { - return newImageOrSrc; - } - } - function imageOnLoad() { - var cachedImgObj = this.__cachedImgObj; - this.onload = this.onerror = this.__cachedImgObj = null; - for (var i = 0; i < cachedImgObj.pending.length; i++) { - var pendingWrap = cachedImgObj.pending[i]; - var cb = pendingWrap.cb; - cb && cb(this, pendingWrap.cbPayload); - pendingWrap.hostEl.dirty(); - } - cachedImgObj.pending.length = 0; - } - function isImageReady(image) { - return image && image.width && image.height; - } - - var STYLE_REG = /\{([a-zA-Z0-9_]+)\|([^}]*)\}/g; - function truncateText(text, containerWidth, font, ellipsis, options) { - if (!containerWidth) { - return ''; - } - var textLines = (text + '').split('\n'); - options = prepareTruncateOptions(containerWidth, font, ellipsis, options); - for (var i = 0, len = textLines.length; i < len; i++) { - textLines[i] = truncateSingleLine(textLines[i], options); - } - return textLines.join('\n'); - } - function prepareTruncateOptions(containerWidth, font, ellipsis, options) { - options = options || {}; - var preparedOpts = extend({}, options); - preparedOpts.font = font; - ellipsis = retrieve2(ellipsis, '...'); - preparedOpts.maxIterations = retrieve2(options.maxIterations, 2); - var minChar = preparedOpts.minChar = retrieve2(options.minChar, 0); - preparedOpts.cnCharWidth = getWidth('国', font); - var ascCharWidth = preparedOpts.ascCharWidth = getWidth('a', font); - preparedOpts.placeholder = retrieve2(options.placeholder, ''); - var contentWidth = containerWidth = Math.max(0, containerWidth - 1); - for (var i = 0; i < minChar && contentWidth >= ascCharWidth; i++) { - contentWidth -= ascCharWidth; - } - var ellipsisWidth = getWidth(ellipsis, font); - if (ellipsisWidth > contentWidth) { - ellipsis = ''; - ellipsisWidth = 0; - } - contentWidth = containerWidth - ellipsisWidth; - preparedOpts.ellipsis = ellipsis; - preparedOpts.ellipsisWidth = ellipsisWidth; - preparedOpts.contentWidth = contentWidth; - preparedOpts.containerWidth = containerWidth; - return preparedOpts; - } - function truncateSingleLine(textLine, options) { - var containerWidth = options.containerWidth; - var font = options.font; - var contentWidth = options.contentWidth; - if (!containerWidth) { - return ''; - } - var lineWidth = getWidth(textLine, font); - if (lineWidth <= containerWidth) { - return textLine; - } - for (var j = 0;; j++) { - if (lineWidth <= contentWidth || j >= options.maxIterations) { - textLine += options.ellipsis; - break; - } - var subLength = j === 0 - ? estimateLength(textLine, contentWidth, options.ascCharWidth, options.cnCharWidth) - : lineWidth > 0 - ? Math.floor(textLine.length * contentWidth / lineWidth) - : 0; - textLine = textLine.substr(0, subLength); - lineWidth = getWidth(textLine, font); - } - if (textLine === '') { - textLine = options.placeholder; - } - return textLine; - } - function estimateLength(text, contentWidth, ascCharWidth, cnCharWidth) { - var width = 0; - var i = 0; - for (var len = text.length; i < len && width < contentWidth; i++) { - var charCode = text.charCodeAt(i); - width += (0 <= charCode && charCode <= 127) ? ascCharWidth : cnCharWidth; - } - return i; - } - function parsePlainText(text, style) { - text != null && (text += ''); - var overflow = style.overflow; - var padding = style.padding; - var font = style.font; - var truncate = overflow === 'truncate'; - var calculatedLineHeight = getLineHeight(font); - var lineHeight = retrieve2(style.lineHeight, calculatedLineHeight); - var bgColorDrawn = !!(style.backgroundColor); - var truncateLineOverflow = style.lineOverflow === 'truncate'; - var width = style.width; - var lines; - if (width != null && (overflow === 'break' || overflow === 'breakAll')) { - lines = text ? wrapText(text, style.font, width, overflow === 'breakAll', 0).lines : []; - } - else { - lines = text ? text.split('\n') : []; - } - var contentHeight = lines.length * lineHeight; - var height = retrieve2(style.height, contentHeight); - if (contentHeight > height && truncateLineOverflow) { - var lineCount = Math.floor(height / lineHeight); - lines = lines.slice(0, lineCount); - } - if (text && truncate && width != null) { - var options = prepareTruncateOptions(width, font, style.ellipsis, { - minChar: style.truncateMinChar, - placeholder: style.placeholder - }); - for (var i = 0; i < lines.length; i++) { - lines[i] = truncateSingleLine(lines[i], options); - } - } - var outerHeight = height; - var contentWidth = 0; - for (var i = 0; i < lines.length; i++) { - contentWidth = Math.max(getWidth(lines[i], font), contentWidth); - } - if (width == null) { - width = contentWidth; - } - var outerWidth = contentWidth; - if (padding) { - outerHeight += padding[0] + padding[2]; - outerWidth += padding[1] + padding[3]; - width += padding[1] + padding[3]; - } - if (bgColorDrawn) { - outerWidth = width; - } - return { - lines: lines, - height: height, - outerWidth: outerWidth, - outerHeight: outerHeight, - lineHeight: lineHeight, - calculatedLineHeight: calculatedLineHeight, - contentWidth: contentWidth, - contentHeight: contentHeight, - width: width - }; - } - var RichTextToken = (function () { - function RichTextToken() { - } - return RichTextToken; - }()); - var RichTextLine = (function () { - function RichTextLine(tokens) { - this.tokens = []; - if (tokens) { - this.tokens = tokens; - } - } - return RichTextLine; - }()); - var RichTextContentBlock = (function () { - function RichTextContentBlock() { - this.width = 0; - this.height = 0; - this.contentWidth = 0; - this.contentHeight = 0; - this.outerWidth = 0; - this.outerHeight = 0; - this.lines = []; - } - return RichTextContentBlock; - }()); - function parseRichText(text, style) { - var contentBlock = new RichTextContentBlock(); - text != null && (text += ''); - if (!text) { - return contentBlock; - } - var topWidth = style.width; - var topHeight = style.height; - var overflow = style.overflow; - var wrapInfo = (overflow === 'break' || overflow === 'breakAll') && topWidth != null - ? { width: topWidth, accumWidth: 0, breakAll: overflow === 'breakAll' } - : null; - var lastIndex = STYLE_REG.lastIndex = 0; - var result; - while ((result = STYLE_REG.exec(text)) != null) { - var matchedIndex = result.index; - if (matchedIndex > lastIndex) { - pushTokens(contentBlock, text.substring(lastIndex, matchedIndex), style, wrapInfo); - } - pushTokens(contentBlock, result[2], style, wrapInfo, result[1]); - lastIndex = STYLE_REG.lastIndex; - } - if (lastIndex < text.length) { - pushTokens(contentBlock, text.substring(lastIndex, text.length), style, wrapInfo); - } - var pendingList = []; - var calculatedHeight = 0; - var calculatedWidth = 0; - var stlPadding = style.padding; - var truncate = overflow === 'truncate'; - var truncateLine = style.lineOverflow === 'truncate'; - function finishLine(line, lineWidth, lineHeight) { - line.width = lineWidth; - line.lineHeight = lineHeight; - calculatedHeight += lineHeight; - calculatedWidth = Math.max(calculatedWidth, lineWidth); - } - outer: for (var i = 0; i < contentBlock.lines.length; i++) { - var line = contentBlock.lines[i]; - var lineHeight = 0; - var lineWidth = 0; - for (var j = 0; j < line.tokens.length; j++) { - var token = line.tokens[j]; - var tokenStyle = token.styleName && style.rich[token.styleName] || {}; - var textPadding = token.textPadding = tokenStyle.padding; - var paddingH = textPadding ? textPadding[1] + textPadding[3] : 0; - var font = token.font = tokenStyle.font || style.font; - token.contentHeight = getLineHeight(font); - var tokenHeight = retrieve2(tokenStyle.height, token.contentHeight); - token.innerHeight = tokenHeight; - textPadding && (tokenHeight += textPadding[0] + textPadding[2]); - token.height = tokenHeight; - token.lineHeight = retrieve3(tokenStyle.lineHeight, style.lineHeight, tokenHeight); - token.align = tokenStyle && tokenStyle.align || style.align; - token.verticalAlign = tokenStyle && tokenStyle.verticalAlign || 'middle'; - if (truncateLine && topHeight != null && calculatedHeight + token.lineHeight > topHeight) { - if (j > 0) { - line.tokens = line.tokens.slice(0, j); - finishLine(line, lineWidth, lineHeight); - contentBlock.lines = contentBlock.lines.slice(0, i + 1); - } - else { - contentBlock.lines = contentBlock.lines.slice(0, i); - } - break outer; - } - var styleTokenWidth = tokenStyle.width; - var tokenWidthNotSpecified = styleTokenWidth == null || styleTokenWidth === 'auto'; - if (typeof styleTokenWidth === 'string' && styleTokenWidth.charAt(styleTokenWidth.length - 1) === '%') { - token.percentWidth = styleTokenWidth; - pendingList.push(token); - token.contentWidth = getWidth(token.text, font); - } - else { - if (tokenWidthNotSpecified) { - var textBackgroundColor = tokenStyle.backgroundColor; - var bgImg = textBackgroundColor && textBackgroundColor.image; - if (bgImg) { - bgImg = findExistImage(bgImg); - if (isImageReady(bgImg)) { - token.width = Math.max(token.width, bgImg.width * tokenHeight / bgImg.height); - } - } - } - var remainTruncWidth = truncate && topWidth != null - ? topWidth - lineWidth : null; - if (remainTruncWidth != null && remainTruncWidth < token.width) { - if (!tokenWidthNotSpecified || remainTruncWidth < paddingH) { - token.text = ''; - token.width = token.contentWidth = 0; - } - else { - token.text = truncateText(token.text, remainTruncWidth - paddingH, font, style.ellipsis, { minChar: style.truncateMinChar }); - token.width = token.contentWidth = getWidth(token.text, font); - } - } - else { - token.contentWidth = getWidth(token.text, font); - } - } - token.width += paddingH; - lineWidth += token.width; - tokenStyle && (lineHeight = Math.max(lineHeight, token.lineHeight)); - } - finishLine(line, lineWidth, lineHeight); - } - contentBlock.outerWidth = contentBlock.width = retrieve2(topWidth, calculatedWidth); - contentBlock.outerHeight = contentBlock.height = retrieve2(topHeight, calculatedHeight); - contentBlock.contentHeight = calculatedHeight; - contentBlock.contentWidth = calculatedWidth; - if (stlPadding) { - contentBlock.outerWidth += stlPadding[1] + stlPadding[3]; - contentBlock.outerHeight += stlPadding[0] + stlPadding[2]; - } - for (var i = 0; i < pendingList.length; i++) { - var token = pendingList[i]; - var percentWidth = token.percentWidth; - token.width = parseInt(percentWidth, 10) / 100 * contentBlock.width; - } - return contentBlock; - } - function pushTokens(block, str, style, wrapInfo, styleName) { - var isEmptyStr = str === ''; - var tokenStyle = styleName && style.rich[styleName] || {}; - var lines = block.lines; - var font = tokenStyle.font || style.font; - var newLine = false; - var strLines; - var linesWidths; - if (wrapInfo) { - var tokenPadding = tokenStyle.padding; - var tokenPaddingH = tokenPadding ? tokenPadding[1] + tokenPadding[3] : 0; - if (tokenStyle.width != null && tokenStyle.width !== 'auto') { - var outerWidth_1 = parsePercent(tokenStyle.width, wrapInfo.width) + tokenPaddingH; - if (lines.length > 0) { - if (outerWidth_1 + wrapInfo.accumWidth > wrapInfo.width) { - strLines = str.split('\n'); - newLine = true; - } - } - wrapInfo.accumWidth = outerWidth_1; - } - else { - var res = wrapText(str, font, wrapInfo.width, wrapInfo.breakAll, wrapInfo.accumWidth); - wrapInfo.accumWidth = res.accumWidth + tokenPaddingH; - linesWidths = res.linesWidths; - strLines = res.lines; - } - } - else { - strLines = str.split('\n'); - } - for (var i = 0; i < strLines.length; i++) { - var text = strLines[i]; - var token = new RichTextToken(); - token.styleName = styleName; - token.text = text; - token.isLineHolder = !text && !isEmptyStr; - if (typeof tokenStyle.width === 'number') { - token.width = tokenStyle.width; - } - else { - token.width = linesWidths - ? linesWidths[i] - : getWidth(text, font); - } - if (!i && !newLine) { - var tokens = (lines[lines.length - 1] || (lines[0] = new RichTextLine())).tokens; - var tokensLen = tokens.length; - (tokensLen === 1 && tokens[0].isLineHolder) - ? (tokens[0] = token) - : ((text || !tokensLen || isEmptyStr) && tokens.push(token)); - } - else { - lines.push(new RichTextLine([token])); - } - } - } - function isAlphabeticLetter(ch) { - var code = ch.charCodeAt(0); - return code >= 0x20 && code <= 0x24F - || code >= 0x370 && code <= 0x10FF - || code >= 0x1200 && code <= 0x13FF - || code >= 0x1E00 && code <= 0x206F; - } - var breakCharMap = reduce(',&?/;] '.split(''), function (obj, ch) { - obj[ch] = true; - return obj; - }, {}); - function isWordBreakChar(ch) { - if (isAlphabeticLetter(ch)) { - if (breakCharMap[ch]) { - return true; - } - return false; - } - return true; - } - function wrapText(text, font, lineWidth, isBreakAll, lastAccumWidth) { - var lines = []; - var linesWidths = []; - var line = ''; - var currentWord = ''; - var currentWordWidth = 0; - var accumWidth = 0; - for (var i = 0; i < text.length; i++) { - var ch = text.charAt(i); - if (ch === '\n') { - if (currentWord) { - line += currentWord; - accumWidth += currentWordWidth; - } - lines.push(line); - linesWidths.push(accumWidth); - line = ''; - currentWord = ''; - currentWordWidth = 0; - accumWidth = 0; - continue; - } - var chWidth = getWidth(ch, font); - var inWord = isBreakAll ? false : !isWordBreakChar(ch); - if (!lines.length - ? lastAccumWidth + accumWidth + chWidth > lineWidth - : accumWidth + chWidth > lineWidth) { - if (!accumWidth) { - if (inWord) { - lines.push(currentWord); - linesWidths.push(currentWordWidth); - currentWord = ch; - currentWordWidth = chWidth; - } - else { - lines.push(ch); - linesWidths.push(chWidth); - } - } - else if (line || currentWord) { - if (inWord) { - if (!line) { - line = currentWord; - currentWord = ''; - currentWordWidth = 0; - accumWidth = currentWordWidth; - } - lines.push(line); - linesWidths.push(accumWidth - currentWordWidth); - currentWord += ch; - currentWordWidth += chWidth; - line = ''; - accumWidth = currentWordWidth; - } - else { - if (currentWord) { - line += currentWord; - currentWord = ''; - currentWordWidth = 0; - } - lines.push(line); - linesWidths.push(accumWidth); - line = ch; - accumWidth = chWidth; - } - } - continue; - } - accumWidth += chWidth; - if (inWord) { - currentWord += ch; - currentWordWidth += chWidth; - } - else { - if (currentWord) { - line += currentWord; - currentWord = ''; - currentWordWidth = 0; - } - line += ch; - } - } - if (!lines.length && !line) { - line = text; - currentWord = ''; - currentWordWidth = 0; - } - if (currentWord) { - line += currentWord; - } - if (line) { - lines.push(line); - linesWidths.push(accumWidth); - } - if (lines.length === 1) { - accumWidth += lastAccumWidth; - } - return { - accumWidth: accumWidth, - lines: lines, - linesWidths: linesWidths - }; - } - - var STYLE_MAGIC_KEY = '__zr_style_' + Math.round((Math.random() * 10)); - var DEFAULT_COMMON_STYLE = { - shadowBlur: 0, - shadowOffsetX: 0, - shadowOffsetY: 0, - shadowColor: '#000', - opacity: 1, - blend: 'source-over' - }; - var DEFAULT_COMMON_ANIMATION_PROPS = { - style: { - shadowBlur: true, - shadowOffsetX: true, - shadowOffsetY: true, - shadowColor: true, - opacity: true - } - }; - DEFAULT_COMMON_STYLE[STYLE_MAGIC_KEY] = true; - var PRIMARY_STATES_KEYS$1 = ['z', 'z2', 'invisible']; - var PRIMARY_STATES_KEYS_IN_HOVER_LAYER = ['invisible']; - var Displayable = (function (_super) { - __extends(Displayable, _super); - function Displayable(props) { - return _super.call(this, props) || this; - } - Displayable.prototype._init = function (props) { - var keysArr = keys(props); - for (var i = 0; i < keysArr.length; i++) { - var key = keysArr[i]; - if (key === 'style') { - this.useStyle(props[key]); - } - else { - _super.prototype.attrKV.call(this, key, props[key]); - } - } - if (!this.style) { - this.useStyle({}); - } - }; - Displayable.prototype.beforeBrush = function () { }; - Displayable.prototype.afterBrush = function () { }; - Displayable.prototype.innerBeforeBrush = function () { }; - Displayable.prototype.innerAfterBrush = function () { }; - Displayable.prototype.shouldBePainted = function (viewWidth, viewHeight, considerClipPath, considerAncestors) { - var m = this.transform; - if (this.ignore - || this.invisible - || this.style.opacity === 0 - || (this.culling - && isDisplayableCulled(this, viewWidth, viewHeight)) - || (m && !m[0] && !m[3])) { - return false; - } - if (considerClipPath && this.__clipPaths) { - for (var i = 0; i < this.__clipPaths.length; ++i) { - if (this.__clipPaths[i].isZeroArea()) { - return false; - } - } - } - if (considerAncestors && this.parent) { - var parent_1 = this.parent; - while (parent_1) { - if (parent_1.ignore) { - return false; - } - parent_1 = parent_1.parent; - } - } - return true; - }; - Displayable.prototype.contain = function (x, y) { - return this.rectContain(x, y); - }; - Displayable.prototype.traverse = function (cb, context) { - cb.call(context, this); - }; - Displayable.prototype.rectContain = function (x, y) { - var coord = this.transformCoordToLocal(x, y); - var rect = this.getBoundingRect(); - return rect.contain(coord[0], coord[1]); - }; - Displayable.prototype.getPaintRect = function () { - var rect = this._paintRect; - if (!this._paintRect || this.__dirty) { - var transform = this.transform; - var elRect = this.getBoundingRect(); - var style = this.style; - var shadowSize = style.shadowBlur || 0; - var shadowOffsetX = style.shadowOffsetX || 0; - var shadowOffsetY = style.shadowOffsetY || 0; - rect = this._paintRect || (this._paintRect = new BoundingRect(0, 0, 0, 0)); - if (transform) { - BoundingRect.applyTransform(rect, elRect, transform); - } - else { - rect.copy(elRect); - } - if (shadowSize || shadowOffsetX || shadowOffsetY) { - rect.width += shadowSize * 2 + Math.abs(shadowOffsetX); - rect.height += shadowSize * 2 + Math.abs(shadowOffsetY); - rect.x = Math.min(rect.x, rect.x + shadowOffsetX - shadowSize); - rect.y = Math.min(rect.y, rect.y + shadowOffsetY - shadowSize); - } - var tolerance = this.dirtyRectTolerance; - if (!rect.isZero()) { - rect.x = Math.floor(rect.x - tolerance); - rect.y = Math.floor(rect.y - tolerance); - rect.width = Math.ceil(rect.width + 1 + tolerance * 2); - rect.height = Math.ceil(rect.height + 1 + tolerance * 2); - } - } - return rect; - }; - Displayable.prototype.setPrevPaintRect = function (paintRect) { - if (paintRect) { - this._prevPaintRect = this._prevPaintRect || new BoundingRect(0, 0, 0, 0); - this._prevPaintRect.copy(paintRect); - } - else { - this._prevPaintRect = null; - } - }; - Displayable.prototype.getPrevPaintRect = function () { - return this._prevPaintRect; - }; - Displayable.prototype.animateStyle = function (loop) { - return this.animate('style', loop); - }; - Displayable.prototype.updateDuringAnimation = function (targetKey) { - if (targetKey === 'style') { - this.dirtyStyle(); - } - else { - this.markRedraw(); - } - }; - Displayable.prototype.attrKV = function (key, value) { - if (key !== 'style') { - _super.prototype.attrKV.call(this, key, value); - } - else { - if (!this.style) { - this.useStyle(value); - } - else { - this.setStyle(value); - } - } - }; - Displayable.prototype.setStyle = function (keyOrObj, value) { - if (typeof keyOrObj === 'string') { - this.style[keyOrObj] = value; - } - else { - extend(this.style, keyOrObj); - } - this.dirtyStyle(); - return this; - }; - Displayable.prototype.dirtyStyle = function (notRedraw) { - if (!notRedraw) { - this.markRedraw(); - } - this.__dirty |= STYLE_CHANGED_BIT; - if (this._rect) { - this._rect = null; - } - }; - Displayable.prototype.dirty = function () { - this.dirtyStyle(); - }; - Displayable.prototype.styleChanged = function () { - return !!(this.__dirty & STYLE_CHANGED_BIT); - }; - Displayable.prototype.styleUpdated = function () { - this.__dirty &= ~STYLE_CHANGED_BIT; - }; - Displayable.prototype.createStyle = function (obj) { - return createObject(DEFAULT_COMMON_STYLE, obj); - }; - Displayable.prototype.useStyle = function (obj) { - if (!obj[STYLE_MAGIC_KEY]) { - obj = this.createStyle(obj); - } - if (this.__inHover) { - this.__hoverStyle = obj; - } - else { - this.style = obj; - } - this.dirtyStyle(); - }; - Displayable.prototype.isStyleObject = function (obj) { - return obj[STYLE_MAGIC_KEY]; - }; - Displayable.prototype._innerSaveToNormal = function (toState) { - _super.prototype._innerSaveToNormal.call(this, toState); - var normalState = this._normalState; - if (toState.style && !normalState.style) { - normalState.style = this._mergeStyle(this.createStyle(), this.style); - } - this._savePrimaryToNormal(toState, normalState, PRIMARY_STATES_KEYS$1); - }; - Displayable.prototype._applyStateObj = function (stateName, state, normalState, keepCurrentStates, transition, animationCfg) { - _super.prototype._applyStateObj.call(this, stateName, state, normalState, keepCurrentStates, transition, animationCfg); - var needsRestoreToNormal = !(state && keepCurrentStates); - var targetStyle; - if (state && state.style) { - if (transition) { - if (keepCurrentStates) { - targetStyle = state.style; - } - else { - targetStyle = this._mergeStyle(this.createStyle(), normalState.style); - this._mergeStyle(targetStyle, state.style); - } - } - else { - targetStyle = this._mergeStyle(this.createStyle(), keepCurrentStates ? this.style : normalState.style); - this._mergeStyle(targetStyle, state.style); - } - } - else if (needsRestoreToNormal) { - targetStyle = normalState.style; - } - if (targetStyle) { - if (transition) { - var sourceStyle = this.style; - this.style = this.createStyle(needsRestoreToNormal ? {} : sourceStyle); - if (needsRestoreToNormal) { - var changedKeys = keys(sourceStyle); - for (var i = 0; i < changedKeys.length; i++) { - var key = changedKeys[i]; - if (key in targetStyle) { - targetStyle[key] = targetStyle[key]; - this.style[key] = sourceStyle[key]; - } - } - } - var targetKeys = keys(targetStyle); - for (var i = 0; i < targetKeys.length; i++) { - var key = targetKeys[i]; - this.style[key] = this.style[key]; - } - this._transitionState(stateName, { - style: targetStyle - }, animationCfg, this.getAnimationStyleProps()); - } - else { - this.useStyle(targetStyle); - } - } - var statesKeys = this.__inHover ? PRIMARY_STATES_KEYS_IN_HOVER_LAYER : PRIMARY_STATES_KEYS$1; - for (var i = 0; i < statesKeys.length; i++) { - var key = statesKeys[i]; - if (state && state[key] != null) { - this[key] = state[key]; - } - else if (needsRestoreToNormal) { - if (normalState[key] != null) { - this[key] = normalState[key]; - } - } - } - }; - Displayable.prototype._mergeStates = function (states) { - var mergedState = _super.prototype._mergeStates.call(this, states); - var mergedStyle; - for (var i = 0; i < states.length; i++) { - var state = states[i]; - if (state.style) { - mergedStyle = mergedStyle || {}; - this._mergeStyle(mergedStyle, state.style); - } - } - if (mergedStyle) { - mergedState.style = mergedStyle; - } - return mergedState; - }; - Displayable.prototype._mergeStyle = function (targetStyle, sourceStyle) { - extend(targetStyle, sourceStyle); - return targetStyle; - }; - Displayable.prototype.getAnimationStyleProps = function () { - return DEFAULT_COMMON_ANIMATION_PROPS; - }; - Displayable.initDefaultProps = (function () { - var dispProto = Displayable.prototype; - dispProto.type = 'displayable'; - dispProto.invisible = false; - dispProto.z = 0; - dispProto.z2 = 0; - dispProto.zlevel = 0; - dispProto.culling = false; - dispProto.cursor = 'pointer'; - dispProto.rectHover = false; - dispProto.incremental = false; - dispProto._rect = null; - dispProto.dirtyRectTolerance = 0; - dispProto.__dirty = REDRAW_BIT | STYLE_CHANGED_BIT; - })(); - return Displayable; - }(Element)); - var tmpRect$1 = new BoundingRect(0, 0, 0, 0); - var viewRect = new BoundingRect(0, 0, 0, 0); - function isDisplayableCulled(el, width, height) { - tmpRect$1.copy(el.getBoundingRect()); - if (el.transform) { - tmpRect$1.applyTransform(el.transform); - } - viewRect.width = width; - viewRect.height = height; - return !tmpRect$1.intersect(viewRect); - } - - var mathMin$1 = Math.min; - var mathMax$1 = Math.max; - var mathSin = Math.sin; - var mathCos = Math.cos; - var PI2 = Math.PI * 2; - var start = create(); - var end = create(); - var extremity = create(); - function fromPoints(points, min, max) { - if (points.length === 0) { - return; - } - var p = points[0]; - var left = p[0]; - var right = p[0]; - var top = p[1]; - var bottom = p[1]; - for (var i = 1; i < points.length; i++) { - p = points[i]; - left = mathMin$1(left, p[0]); - right = mathMax$1(right, p[0]); - top = mathMin$1(top, p[1]); - bottom = mathMax$1(bottom, p[1]); - } - min[0] = left; - min[1] = top; - max[0] = right; - max[1] = bottom; - } - function fromLine(x0, y0, x1, y1, min, max) { - min[0] = mathMin$1(x0, x1); - min[1] = mathMin$1(y0, y1); - max[0] = mathMax$1(x0, x1); - max[1] = mathMax$1(y0, y1); - } - var xDim = []; - var yDim = []; - function fromCubic(x0, y0, x1, y1, x2, y2, x3, y3, min, max) { - var cubicExtrema$1 = cubicExtrema; - var cubicAt$1 = cubicAt; - var n = cubicExtrema$1(x0, x1, x2, x3, xDim); - min[0] = Infinity; - min[1] = Infinity; - max[0] = -Infinity; - max[1] = -Infinity; - for (var i = 0; i < n; i++) { - var x = cubicAt$1(x0, x1, x2, x3, xDim[i]); - min[0] = mathMin$1(x, min[0]); - max[0] = mathMax$1(x, max[0]); - } - n = cubicExtrema$1(y0, y1, y2, y3, yDim); - for (var i = 0; i < n; i++) { - var y = cubicAt$1(y0, y1, y2, y3, yDim[i]); - min[1] = mathMin$1(y, min[1]); - max[1] = mathMax$1(y, max[1]); - } - min[0] = mathMin$1(x0, min[0]); - max[0] = mathMax$1(x0, max[0]); - min[0] = mathMin$1(x3, min[0]); - max[0] = mathMax$1(x3, max[0]); - min[1] = mathMin$1(y0, min[1]); - max[1] = mathMax$1(y0, max[1]); - min[1] = mathMin$1(y3, min[1]); - max[1] = mathMax$1(y3, max[1]); - } - function fromQuadratic(x0, y0, x1, y1, x2, y2, min, max) { - var quadraticExtremum$1 = quadraticExtremum; - var quadraticAt$1 = quadraticAt; - var tx = mathMax$1(mathMin$1(quadraticExtremum$1(x0, x1, x2), 1), 0); - var ty = mathMax$1(mathMin$1(quadraticExtremum$1(y0, y1, y2), 1), 0); - var x = quadraticAt$1(x0, x1, x2, tx); - var y = quadraticAt$1(y0, y1, y2, ty); - min[0] = mathMin$1(x0, x2, x); - min[1] = mathMin$1(y0, y2, y); - max[0] = mathMax$1(x0, x2, x); - max[1] = mathMax$1(y0, y2, y); - } - function fromArc(x, y, rx, ry, startAngle, endAngle, anticlockwise, min$1, max$1) { - var vec2Min = min; - var vec2Max = max; - var diff = Math.abs(startAngle - endAngle); - if (diff % PI2 < 1e-4 && diff > 1e-4) { - min$1[0] = x - rx; - min$1[1] = y - ry; - max$1[0] = x + rx; - max$1[1] = y + ry; - return; - } - start[0] = mathCos(startAngle) * rx + x; - start[1] = mathSin(startAngle) * ry + y; - end[0] = mathCos(endAngle) * rx + x; - end[1] = mathSin(endAngle) * ry + y; - vec2Min(min$1, start, end); - vec2Max(max$1, start, end); - startAngle = startAngle % (PI2); - if (startAngle < 0) { - startAngle = startAngle + PI2; - } - endAngle = endAngle % (PI2); - if (endAngle < 0) { - endAngle = endAngle + PI2; - } - if (startAngle > endAngle && !anticlockwise) { - endAngle += PI2; - } - else if (startAngle < endAngle && anticlockwise) { - startAngle += PI2; - } - if (anticlockwise) { - var tmp = endAngle; - endAngle = startAngle; - startAngle = tmp; - } - for (var angle = 0; angle < endAngle; angle += Math.PI / 2) { - if (angle > startAngle) { - extremity[0] = mathCos(angle) * rx + x; - extremity[1] = mathSin(angle) * ry + y; - vec2Min(min$1, extremity, min$1); - vec2Max(max$1, extremity, max$1); - } - } - } - - var CMD = { - M: 1, - L: 2, - C: 3, - Q: 4, - A: 5, - Z: 6, - R: 7 - }; - var tmpOutX = []; - var tmpOutY = []; - var min$1 = []; - var max$1 = []; - var min2 = []; - var max2 = []; - var mathMin$2 = Math.min; - var mathMax$2 = Math.max; - var mathCos$1 = Math.cos; - var mathSin$1 = Math.sin; - var mathAbs = Math.abs; - var PI = Math.PI; - var PI2$1 = PI * 2; - var hasTypedArray = typeof Float32Array !== 'undefined'; - var tmpAngles = []; - function modPI2(radian) { - var n = Math.round(radian / PI * 1e8) / 1e8; - return (n % 2) * PI; - } - function normalizeArcAngles(angles, anticlockwise) { - var newStartAngle = modPI2(angles[0]); - if (newStartAngle < 0) { - newStartAngle += PI2$1; - } - var delta = newStartAngle - angles[0]; - var newEndAngle = angles[1]; - newEndAngle += delta; - if (!anticlockwise && newEndAngle - newStartAngle >= PI2$1) { - newEndAngle = newStartAngle + PI2$1; - } - else if (anticlockwise && newStartAngle - newEndAngle >= PI2$1) { - newEndAngle = newStartAngle - PI2$1; - } - else if (!anticlockwise && newStartAngle > newEndAngle) { - newEndAngle = newStartAngle + (PI2$1 - modPI2(newStartAngle - newEndAngle)); - } - else if (anticlockwise && newStartAngle < newEndAngle) { - newEndAngle = newStartAngle - (PI2$1 - modPI2(newEndAngle - newStartAngle)); - } - angles[0] = newStartAngle; - angles[1] = newEndAngle; - } - var PathProxy = (function () { - function PathProxy(notSaveData) { - this.dpr = 1; - this._xi = 0; - this._yi = 0; - this._x0 = 0; - this._y0 = 0; - this._len = 0; - if (notSaveData) { - this._saveData = false; - } - if (this._saveData) { - this.data = []; - } - } - PathProxy.prototype.increaseVersion = function () { - this._version++; - }; - PathProxy.prototype.getVersion = function () { - return this._version; - }; - PathProxy.prototype.setScale = function (sx, sy, segmentIgnoreThreshold) { - segmentIgnoreThreshold = segmentIgnoreThreshold || 0; - if (segmentIgnoreThreshold > 0) { - this._ux = mathAbs(segmentIgnoreThreshold / devicePixelRatio / sx) || 0; - this._uy = mathAbs(segmentIgnoreThreshold / devicePixelRatio / sy) || 0; - } - }; - PathProxy.prototype.setDPR = function (dpr) { - this.dpr = dpr; - }; - PathProxy.prototype.setContext = function (ctx) { - this._ctx = ctx; - }; - PathProxy.prototype.getContext = function () { - return this._ctx; - }; - PathProxy.prototype.beginPath = function () { - this._ctx && this._ctx.beginPath(); - this.reset(); - return this; - }; - PathProxy.prototype.reset = function () { - if (this._saveData) { - this._len = 0; - } - if (this._pathSegLen) { - this._pathSegLen = null; - this._pathLen = 0; - } - this._version++; - }; - PathProxy.prototype.moveTo = function (x, y) { - this._drawPendingPt(); - this.addData(CMD.M, x, y); - this._ctx && this._ctx.moveTo(x, y); - this._x0 = x; - this._y0 = y; - this._xi = x; - this._yi = y; - return this; - }; - PathProxy.prototype.lineTo = function (x, y) { - var dx = mathAbs(x - this._xi); - var dy = mathAbs(y - this._yi); - var exceedUnit = dx > this._ux || dy > this._uy; - this.addData(CMD.L, x, y); - if (this._ctx && exceedUnit) { - this._ctx.lineTo(x, y); - } - if (exceedUnit) { - this._xi = x; - this._yi = y; - this._pendingPtDist = 0; - } - else { - var d2 = dx * dx + dy * dy; - if (d2 > this._pendingPtDist) { - this._pendingPtX = x; - this._pendingPtY = y; - this._pendingPtDist = d2; - } - } - return this; - }; - PathProxy.prototype.bezierCurveTo = function (x1, y1, x2, y2, x3, y3) { - this._drawPendingPt(); - this.addData(CMD.C, x1, y1, x2, y2, x3, y3); - if (this._ctx) { - this._ctx.bezierCurveTo(x1, y1, x2, y2, x3, y3); - } - this._xi = x3; - this._yi = y3; - return this; - }; - PathProxy.prototype.quadraticCurveTo = function (x1, y1, x2, y2) { - this._drawPendingPt(); - this.addData(CMD.Q, x1, y1, x2, y2); - if (this._ctx) { - this._ctx.quadraticCurveTo(x1, y1, x2, y2); - } - this._xi = x2; - this._yi = y2; - return this; - }; - PathProxy.prototype.arc = function (cx, cy, r, startAngle, endAngle, anticlockwise) { - this._drawPendingPt(); - tmpAngles[0] = startAngle; - tmpAngles[1] = endAngle; - normalizeArcAngles(tmpAngles, anticlockwise); - startAngle = tmpAngles[0]; - endAngle = tmpAngles[1]; - var delta = endAngle - startAngle; - this.addData(CMD.A, cx, cy, r, r, startAngle, delta, 0, anticlockwise ? 0 : 1); - this._ctx && this._ctx.arc(cx, cy, r, startAngle, endAngle, anticlockwise); - this._xi = mathCos$1(endAngle) * r + cx; - this._yi = mathSin$1(endAngle) * r + cy; - return this; - }; - PathProxy.prototype.arcTo = function (x1, y1, x2, y2, radius) { - this._drawPendingPt(); - if (this._ctx) { - this._ctx.arcTo(x1, y1, x2, y2, radius); - } - return this; - }; - PathProxy.prototype.rect = function (x, y, w, h) { - this._drawPendingPt(); - this._ctx && this._ctx.rect(x, y, w, h); - this.addData(CMD.R, x, y, w, h); - return this; - }; - PathProxy.prototype.closePath = function () { - this._drawPendingPt(); - this.addData(CMD.Z); - var ctx = this._ctx; - var x0 = this._x0; - var y0 = this._y0; - if (ctx) { - ctx.closePath(); - } - this._xi = x0; - this._yi = y0; - return this; - }; - PathProxy.prototype.fill = function (ctx) { - ctx && ctx.fill(); - this.toStatic(); - }; - PathProxy.prototype.stroke = function (ctx) { - ctx && ctx.stroke(); - this.toStatic(); - }; - PathProxy.prototype.len = function () { - return this._len; - }; - PathProxy.prototype.setData = function (data) { - var len = data.length; - if (!(this.data && this.data.length === len) && hasTypedArray) { - this.data = new Float32Array(len); - } - for (var i = 0; i < len; i++) { - this.data[i] = data[i]; - } - this._len = len; - }; - PathProxy.prototype.appendPath = function (path) { - if (!(path instanceof Array)) { - path = [path]; - } - var len = path.length; - var appendSize = 0; - var offset = this._len; - for (var i = 0; i < len; i++) { - appendSize += path[i].len(); - } - if (hasTypedArray && (this.data instanceof Float32Array)) { - this.data = new Float32Array(offset + appendSize); - } - for (var i = 0; i < len; i++) { - var appendPathData = path[i].data; - for (var k = 0; k < appendPathData.length; k++) { - this.data[offset++] = appendPathData[k]; - } - } - this._len = offset; - }; - PathProxy.prototype.addData = function (cmd, a, b, c, d, e, f, g, h) { - if (!this._saveData) { - return; - } - var data = this.data; - if (this._len + arguments.length > data.length) { - this._expandData(); - data = this.data; - } - for (var i = 0; i < arguments.length; i++) { - data[this._len++] = arguments[i]; - } - }; - PathProxy.prototype._drawPendingPt = function () { - if (this._pendingPtDist > 0) { - this._ctx && this._ctx.lineTo(this._pendingPtX, this._pendingPtY); - this._pendingPtDist = 0; - } - }; - PathProxy.prototype._expandData = function () { - if (!(this.data instanceof Array)) { - var newData = []; - for (var i = 0; i < this._len; i++) { - newData[i] = this.data[i]; - } - this.data = newData; - } - }; - PathProxy.prototype.toStatic = function () { - if (!this._saveData) { - return; - } - this._drawPendingPt(); - var data = this.data; - if (data instanceof Array) { - data.length = this._len; - if (hasTypedArray && this._len > 11) { - this.data = new Float32Array(data); - } - } - }; - PathProxy.prototype.getBoundingRect = function () { - min$1[0] = min$1[1] = min2[0] = min2[1] = Number.MAX_VALUE; - max$1[0] = max$1[1] = max2[0] = max2[1] = -Number.MAX_VALUE; - var data = this.data; - var xi = 0; - var yi = 0; - var x0 = 0; - var y0 = 0; - var i; - for (i = 0; i < this._len;) { - var cmd = data[i++]; - var isFirst = i === 1; - if (isFirst) { - xi = data[i]; - yi = data[i + 1]; - x0 = xi; - y0 = yi; - } - switch (cmd) { - case CMD.M: - xi = x0 = data[i++]; - yi = y0 = data[i++]; - min2[0] = x0; - min2[1] = y0; - max2[0] = x0; - max2[1] = y0; - break; - case CMD.L: - fromLine(xi, yi, data[i], data[i + 1], min2, max2); - xi = data[i++]; - yi = data[i++]; - break; - case CMD.C: - fromCubic(xi, yi, data[i++], data[i++], data[i++], data[i++], data[i], data[i + 1], min2, max2); - xi = data[i++]; - yi = data[i++]; - break; - case CMD.Q: - fromQuadratic(xi, yi, data[i++], data[i++], data[i], data[i + 1], min2, max2); - xi = data[i++]; - yi = data[i++]; - break; - case CMD.A: - var cx = data[i++]; - var cy = data[i++]; - var rx = data[i++]; - var ry = data[i++]; - var startAngle = data[i++]; - var endAngle = data[i++] + startAngle; - i += 1; - var anticlockwise = !data[i++]; - if (isFirst) { - x0 = mathCos$1(startAngle) * rx + cx; - y0 = mathSin$1(startAngle) * ry + cy; - } - fromArc(cx, cy, rx, ry, startAngle, endAngle, anticlockwise, min2, max2); - xi = mathCos$1(endAngle) * rx + cx; - yi = mathSin$1(endAngle) * ry + cy; - break; - case CMD.R: - x0 = xi = data[i++]; - y0 = yi = data[i++]; - var width = data[i++]; - var height = data[i++]; - fromLine(x0, y0, x0 + width, y0 + height, min2, max2); - break; - case CMD.Z: - xi = x0; - yi = y0; - break; - } - min(min$1, min$1, min2); - max(max$1, max$1, max2); - } - if (i === 0) { - min$1[0] = min$1[1] = max$1[0] = max$1[1] = 0; - } - return new BoundingRect(min$1[0], min$1[1], max$1[0] - min$1[0], max$1[1] - min$1[1]); - }; - PathProxy.prototype._calculateLength = function () { - var data = this.data; - var len = this._len; - var ux = this._ux; - var uy = this._uy; - var xi = 0; - var yi = 0; - var x0 = 0; - var y0 = 0; - if (!this._pathSegLen) { - this._pathSegLen = []; - } - var pathSegLen = this._pathSegLen; - var pathTotalLen = 0; - var segCount = 0; - for (var i = 0; i < len;) { - var cmd = data[i++]; - var isFirst = i === 1; - if (isFirst) { - xi = data[i]; - yi = data[i + 1]; - x0 = xi; - y0 = yi; - } - var l = -1; - switch (cmd) { - case CMD.M: - xi = x0 = data[i++]; - yi = y0 = data[i++]; - break; - case CMD.L: { - var x2 = data[i++]; - var y2 = data[i++]; - var dx = x2 - xi; - var dy = y2 - yi; - if (mathAbs(dx) > ux || mathAbs(dy) > uy || i === len - 1) { - l = Math.sqrt(dx * dx + dy * dy); - xi = x2; - yi = y2; - } - break; - } - case CMD.C: { - var x1 = data[i++]; - var y1 = data[i++]; - var x2 = data[i++]; - var y2 = data[i++]; - var x3 = data[i++]; - var y3 = data[i++]; - l = cubicLength(xi, yi, x1, y1, x2, y2, x3, y3, 10); - xi = x3; - yi = y3; - break; - } - case CMD.Q: { - var x1 = data[i++]; - var y1 = data[i++]; - var x2 = data[i++]; - var y2 = data[i++]; - l = quadraticLength(xi, yi, x1, y1, x2, y2, 10); - xi = x2; - yi = y2; - break; - } - case CMD.A: - var cx = data[i++]; - var cy = data[i++]; - var rx = data[i++]; - var ry = data[i++]; - var startAngle = data[i++]; - var delta = data[i++]; - var endAngle = delta + startAngle; - i += 1; - var anticlockwise = !data[i++]; - if (isFirst) { - x0 = mathCos$1(startAngle) * rx + cx; - y0 = mathSin$1(startAngle) * ry + cy; - } - l = mathMax$2(rx, ry) * mathMin$2(PI2$1, Math.abs(delta)); - xi = mathCos$1(endAngle) * rx + cx; - yi = mathSin$1(endAngle) * ry + cy; - break; - case CMD.R: { - x0 = xi = data[i++]; - y0 = yi = data[i++]; - var width = data[i++]; - var height = data[i++]; - l = width * 2 + height * 2; - break; - } - case CMD.Z: { - var dx = x0 - xi; - var dy = y0 - yi; - l = Math.sqrt(dx * dx + dy * dy); - xi = x0; - yi = y0; - break; - } - } - if (l >= 0) { - pathSegLen[segCount++] = l; - pathTotalLen += l; - } - } - this._pathLen = pathTotalLen; - return pathTotalLen; - }; - PathProxy.prototype.rebuildPath = function (ctx, percent) { - var d = this.data; - var ux = this._ux; - var uy = this._uy; - var len = this._len; - var x0; - var y0; - var xi; - var yi; - var x; - var y; - var drawPart = percent < 1; - var pathSegLen; - var pathTotalLen; - var accumLength = 0; - var segCount = 0; - var displayedLength; - var pendingPtDist = 0; - var pendingPtX; - var pendingPtY; - if (drawPart) { - if (!this._pathSegLen) { - this._calculateLength(); - } - pathSegLen = this._pathSegLen; - pathTotalLen = this._pathLen; - displayedLength = percent * pathTotalLen; - if (!displayedLength) { - return; - } - } - lo: for (var i = 0; i < len;) { - var cmd = d[i++]; - var isFirst = i === 1; - if (isFirst) { - xi = d[i]; - yi = d[i + 1]; - x0 = xi; - y0 = yi; - } - if (cmd !== CMD.L && pendingPtDist > 0) { - ctx.lineTo(pendingPtX, pendingPtY); - pendingPtDist = 0; - } - switch (cmd) { - case CMD.M: - x0 = xi = d[i++]; - y0 = yi = d[i++]; - ctx.moveTo(xi, yi); - break; - case CMD.L: { - x = d[i++]; - y = d[i++]; - var dx = mathAbs(x - xi); - var dy = mathAbs(y - yi); - if (dx > ux || dy > uy) { - if (drawPart) { - var l = pathSegLen[segCount++]; - if (accumLength + l > displayedLength) { - var t = (displayedLength - accumLength) / l; - ctx.lineTo(xi * (1 - t) + x * t, yi * (1 - t) + y * t); - break lo; - } - accumLength += l; - } - ctx.lineTo(x, y); - xi = x; - yi = y; - pendingPtDist = 0; - } - else { - var d2 = dx * dx + dy * dy; - if (d2 > pendingPtDist) { - pendingPtX = x; - pendingPtY = y; - pendingPtDist = d2; - } - } - break; - } - case CMD.C: { - var x1 = d[i++]; - var y1 = d[i++]; - var x2 = d[i++]; - var y2 = d[i++]; - var x3 = d[i++]; - var y3 = d[i++]; - if (drawPart) { - var l = pathSegLen[segCount++]; - if (accumLength + l > displayedLength) { - var t = (displayedLength - accumLength) / l; - cubicSubdivide(xi, x1, x2, x3, t, tmpOutX); - cubicSubdivide(yi, y1, y2, y3, t, tmpOutY); - ctx.bezierCurveTo(tmpOutX[1], tmpOutY[1], tmpOutX[2], tmpOutY[2], tmpOutX[3], tmpOutY[3]); - break lo; - } - accumLength += l; - } - ctx.bezierCurveTo(x1, y1, x2, y2, x3, y3); - xi = x3; - yi = y3; - break; - } - case CMD.Q: { - var x1 = d[i++]; - var y1 = d[i++]; - var x2 = d[i++]; - var y2 = d[i++]; - if (drawPart) { - var l = pathSegLen[segCount++]; - if (accumLength + l > displayedLength) { - var t = (displayedLength - accumLength) / l; - quadraticSubdivide(xi, x1, x2, t, tmpOutX); - quadraticSubdivide(yi, y1, y2, t, tmpOutY); - ctx.quadraticCurveTo(tmpOutX[1], tmpOutY[1], tmpOutX[2], tmpOutY[2]); - break lo; - } - accumLength += l; - } - ctx.quadraticCurveTo(x1, y1, x2, y2); - xi = x2; - yi = y2; - break; - } - case CMD.A: - var cx = d[i++]; - var cy = d[i++]; - var rx = d[i++]; - var ry = d[i++]; - var startAngle = d[i++]; - var delta = d[i++]; - var psi = d[i++]; - var anticlockwise = !d[i++]; - var r = (rx > ry) ? rx : ry; - var isEllipse = mathAbs(rx - ry) > 1e-3; - var endAngle = startAngle + delta; - var breakBuild = false; - if (drawPart) { - var l = pathSegLen[segCount++]; - if (accumLength + l > displayedLength) { - endAngle = startAngle + delta * (displayedLength - accumLength) / l; - breakBuild = true; - } - accumLength += l; - } - if (isEllipse && ctx.ellipse) { - ctx.ellipse(cx, cy, rx, ry, psi, startAngle, endAngle, anticlockwise); - } - else { - ctx.arc(cx, cy, r, startAngle, endAngle, anticlockwise); - } - if (breakBuild) { - break lo; - } - if (isFirst) { - x0 = mathCos$1(startAngle) * rx + cx; - y0 = mathSin$1(startAngle) * ry + cy; - } - xi = mathCos$1(endAngle) * rx + cx; - yi = mathSin$1(endAngle) * ry + cy; - break; - case CMD.R: - x0 = xi = d[i]; - y0 = yi = d[i + 1]; - x = d[i++]; - y = d[i++]; - var width = d[i++]; - var height = d[i++]; - if (drawPart) { - var l = pathSegLen[segCount++]; - if (accumLength + l > displayedLength) { - var d_1 = displayedLength - accumLength; - ctx.moveTo(x, y); - ctx.lineTo(x + mathMin$2(d_1, width), y); - d_1 -= width; - if (d_1 > 0) { - ctx.lineTo(x + width, y + mathMin$2(d_1, height)); - } - d_1 -= height; - if (d_1 > 0) { - ctx.lineTo(x + mathMax$2(width - d_1, 0), y + height); - } - d_1 -= width; - if (d_1 > 0) { - ctx.lineTo(x, y + mathMax$2(height - d_1, 0)); - } - break lo; - } - accumLength += l; - } - ctx.rect(x, y, width, height); - break; - case CMD.Z: - if (drawPart) { - var l = pathSegLen[segCount++]; - if (accumLength + l > displayedLength) { - var t = (displayedLength - accumLength) / l; - ctx.lineTo(xi * (1 - t) + x0 * t, yi * (1 - t) + y0 * t); - break lo; - } - accumLength += l; - } - ctx.closePath(); - xi = x0; - yi = y0; - } - } - }; - PathProxy.prototype.clone = function () { - var newProxy = new PathProxy(); - var data = this.data; - newProxy.data = data.slice ? data.slice() - : Array.prototype.slice.call(data); - newProxy._len = this._len; - return newProxy; - }; - PathProxy.CMD = CMD; - PathProxy.initDefaultProps = (function () { - var proto = PathProxy.prototype; - proto._saveData = true; - proto._ux = 0; - proto._uy = 0; - proto._pendingPtDist = 0; - proto._version = 0; - })(); - return PathProxy; - }()); - - function containStroke(x0, y0, x1, y1, lineWidth, x, y) { - if (lineWidth === 0) { - return false; - } - var _l = lineWidth; - var _a = 0; - var _b = x0; - if ((y > y0 + _l && y > y1 + _l) - || (y < y0 - _l && y < y1 - _l) - || (x > x0 + _l && x > x1 + _l) - || (x < x0 - _l && x < x1 - _l)) { - return false; - } - if (x0 !== x1) { - _a = (y0 - y1) / (x0 - x1); - _b = (x0 * y1 - x1 * y0) / (x0 - x1); - } - else { - return Math.abs(x - x0) <= _l / 2; - } - var tmp = _a * x - y + _b; - var _s = tmp * tmp / (_a * _a + 1); - return _s <= _l / 2 * _l / 2; - } - - function containStroke$1(x0, y0, x1, y1, x2, y2, x3, y3, lineWidth, x, y) { - if (lineWidth === 0) { - return false; - } - var _l = lineWidth; - if ((y > y0 + _l && y > y1 + _l && y > y2 + _l && y > y3 + _l) - || (y < y0 - _l && y < y1 - _l && y < y2 - _l && y < y3 - _l) - || (x > x0 + _l && x > x1 + _l && x > x2 + _l && x > x3 + _l) - || (x < x0 - _l && x < x1 - _l && x < x2 - _l && x < x3 - _l)) { - return false; - } - var d = cubicProjectPoint(x0, y0, x1, y1, x2, y2, x3, y3, x, y, null); - return d <= _l / 2; - } - - function containStroke$2(x0, y0, x1, y1, x2, y2, lineWidth, x, y) { - if (lineWidth === 0) { - return false; - } - var _l = lineWidth; - if ((y > y0 + _l && y > y1 + _l && y > y2 + _l) - || (y < y0 - _l && y < y1 - _l && y < y2 - _l) - || (x > x0 + _l && x > x1 + _l && x > x2 + _l) - || (x < x0 - _l && x < x1 - _l && x < x2 - _l)) { - return false; - } - var d = quadraticProjectPoint(x0, y0, x1, y1, x2, y2, x, y, null); - return d <= _l / 2; - } - - var PI2$2 = Math.PI * 2; - function normalizeRadian(angle) { - angle %= PI2$2; - if (angle < 0) { - angle += PI2$2; - } - return angle; - } - - var PI2$3 = Math.PI * 2; - function containStroke$3(cx, cy, r, startAngle, endAngle, anticlockwise, lineWidth, x, y) { - if (lineWidth === 0) { - return false; - } - var _l = lineWidth; - x -= cx; - y -= cy; - var d = Math.sqrt(x * x + y * y); - if ((d - _l > r) || (d + _l < r)) { - return false; - } - if (Math.abs(startAngle - endAngle) % PI2$3 < 1e-4) { - return true; - } - if (anticlockwise) { - var tmp = startAngle; - startAngle = normalizeRadian(endAngle); - endAngle = normalizeRadian(tmp); - } - else { - startAngle = normalizeRadian(startAngle); - endAngle = normalizeRadian(endAngle); - } - if (startAngle > endAngle) { - endAngle += PI2$3; - } - var angle = Math.atan2(y, x); - if (angle < 0) { - angle += PI2$3; - } - return (angle >= startAngle && angle <= endAngle) - || (angle + PI2$3 >= startAngle && angle + PI2$3 <= endAngle); - } - - function windingLine(x0, y0, x1, y1, x, y) { - if ((y > y0 && y > y1) || (y < y0 && y < y1)) { - return 0; - } - if (y1 === y0) { - return 0; - } - var t = (y - y0) / (y1 - y0); - var dir = y1 < y0 ? 1 : -1; - if (t === 1 || t === 0) { - dir = y1 < y0 ? 0.5 : -0.5; - } - var x_ = t * (x1 - x0) + x0; - return x_ === x ? Infinity : x_ > x ? dir : 0; - } - - var CMD$1 = PathProxy.CMD; - var PI2$4 = Math.PI * 2; - var EPSILON$3 = 1e-4; - function isAroundEqual(a, b) { - return Math.abs(a - b) < EPSILON$3; - } - var roots = [-1, -1, -1]; - var extrema = [-1, -1]; - function swapExtrema() { - var tmp = extrema[0]; - extrema[0] = extrema[1]; - extrema[1] = tmp; - } - function windingCubic(x0, y0, x1, y1, x2, y2, x3, y3, x, y) { - if ((y > y0 && y > y1 && y > y2 && y > y3) - || (y < y0 && y < y1 && y < y2 && y < y3)) { - return 0; - } - var nRoots = cubicRootAt(y0, y1, y2, y3, y, roots); - if (nRoots === 0) { - return 0; - } - else { - var w = 0; - var nExtrema = -1; - var y0_ = void 0; - var y1_ = void 0; - for (var i = 0; i < nRoots; i++) { - var t = roots[i]; - var unit = (t === 0 || t === 1) ? 0.5 : 1; - var x_ = cubicAt(x0, x1, x2, x3, t); - if (x_ < x) { - continue; - } - if (nExtrema < 0) { - nExtrema = cubicExtrema(y0, y1, y2, y3, extrema); - if (extrema[1] < extrema[0] && nExtrema > 1) { - swapExtrema(); - } - y0_ = cubicAt(y0, y1, y2, y3, extrema[0]); - if (nExtrema > 1) { - y1_ = cubicAt(y0, y1, y2, y3, extrema[1]); - } - } - if (nExtrema === 2) { - if (t < extrema[0]) { - w += y0_ < y0 ? unit : -unit; - } - else if (t < extrema[1]) { - w += y1_ < y0_ ? unit : -unit; - } - else { - w += y3 < y1_ ? unit : -unit; - } - } - else { - if (t < extrema[0]) { - w += y0_ < y0 ? unit : -unit; - } - else { - w += y3 < y0_ ? unit : -unit; - } - } - } - return w; - } - } - function windingQuadratic(x0, y0, x1, y1, x2, y2, x, y) { - if ((y > y0 && y > y1 && y > y2) - || (y < y0 && y < y1 && y < y2)) { - return 0; - } - var nRoots = quadraticRootAt(y0, y1, y2, y, roots); - if (nRoots === 0) { - return 0; - } - else { - var t = quadraticExtremum(y0, y1, y2); - if (t >= 0 && t <= 1) { - var w = 0; - var y_ = quadraticAt(y0, y1, y2, t); - for (var i = 0; i < nRoots; i++) { - var unit = (roots[i] === 0 || roots[i] === 1) ? 0.5 : 1; - var x_ = quadraticAt(x0, x1, x2, roots[i]); - if (x_ < x) { - continue; - } - if (roots[i] < t) { - w += y_ < y0 ? unit : -unit; - } - else { - w += y2 < y_ ? unit : -unit; - } - } - return w; - } - else { - var unit = (roots[0] === 0 || roots[0] === 1) ? 0.5 : 1; - var x_ = quadraticAt(x0, x1, x2, roots[0]); - if (x_ < x) { - return 0; - } - return y2 < y0 ? unit : -unit; - } - } - } - function windingArc(cx, cy, r, startAngle, endAngle, anticlockwise, x, y) { - y -= cy; - if (y > r || y < -r) { - return 0; - } - var tmp = Math.sqrt(r * r - y * y); - roots[0] = -tmp; - roots[1] = tmp; - var dTheta = Math.abs(startAngle - endAngle); - if (dTheta < 1e-4) { - return 0; - } - if (dTheta >= PI2$4 - 1e-4) { - startAngle = 0; - endAngle = PI2$4; - var dir = anticlockwise ? 1 : -1; - if (x >= roots[0] + cx && x <= roots[1] + cx) { - return dir; - } - else { - return 0; - } - } - if (startAngle > endAngle) { - var tmp_1 = startAngle; - startAngle = endAngle; - endAngle = tmp_1; - } - if (startAngle < 0) { - startAngle += PI2$4; - endAngle += PI2$4; - } - var w = 0; - for (var i = 0; i < 2; i++) { - var x_ = roots[i]; - if (x_ + cx > x) { - var angle = Math.atan2(y, x_); - var dir = anticlockwise ? 1 : -1; - if (angle < 0) { - angle = PI2$4 + angle; - } - if ((angle >= startAngle && angle <= endAngle) - || (angle + PI2$4 >= startAngle && angle + PI2$4 <= endAngle)) { - if (angle > Math.PI / 2 && angle < Math.PI * 1.5) { - dir = -dir; - } - w += dir; - } - } - } - return w; - } - function containPath(path, lineWidth, isStroke, x, y) { - var data = path.data; - var len = path.len(); - var w = 0; - var xi = 0; - var yi = 0; - var x0 = 0; - var y0 = 0; - var x1; - var y1; - for (var i = 0; i < len;) { - var cmd = data[i++]; - var isFirst = i === 1; - if (cmd === CMD$1.M && i > 1) { - if (!isStroke) { - w += windingLine(xi, yi, x0, y0, x, y); - } - } - if (isFirst) { - xi = data[i]; - yi = data[i + 1]; - x0 = xi; - y0 = yi; - } - switch (cmd) { - case CMD$1.M: - x0 = data[i++]; - y0 = data[i++]; - xi = x0; - yi = y0; - break; - case CMD$1.L: - if (isStroke) { - if (containStroke(xi, yi, data[i], data[i + 1], lineWidth, x, y)) { - return true; - } - } - else { - w += windingLine(xi, yi, data[i], data[i + 1], x, y) || 0; - } - xi = data[i++]; - yi = data[i++]; - break; - case CMD$1.C: - if (isStroke) { - if (containStroke$1(xi, yi, data[i++], data[i++], data[i++], data[i++], data[i], data[i + 1], lineWidth, x, y)) { - return true; - } - } - else { - w += windingCubic(xi, yi, data[i++], data[i++], data[i++], data[i++], data[i], data[i + 1], x, y) || 0; - } - xi = data[i++]; - yi = data[i++]; - break; - case CMD$1.Q: - if (isStroke) { - if (containStroke$2(xi, yi, data[i++], data[i++], data[i], data[i + 1], lineWidth, x, y)) { - return true; - } - } - else { - w += windingQuadratic(xi, yi, data[i++], data[i++], data[i], data[i + 1], x, y) || 0; - } - xi = data[i++]; - yi = data[i++]; - break; - case CMD$1.A: - var cx = data[i++]; - var cy = data[i++]; - var rx = data[i++]; - var ry = data[i++]; - var theta = data[i++]; - var dTheta = data[i++]; - i += 1; - var anticlockwise = !!(1 - data[i++]); - x1 = Math.cos(theta) * rx + cx; - y1 = Math.sin(theta) * ry + cy; - if (!isFirst) { - w += windingLine(xi, yi, x1, y1, x, y); - } - else { - x0 = x1; - y0 = y1; - } - var _x = (x - cx) * ry / rx + cx; - if (isStroke) { - if (containStroke$3(cx, cy, ry, theta, theta + dTheta, anticlockwise, lineWidth, _x, y)) { - return true; - } - } - else { - w += windingArc(cx, cy, ry, theta, theta + dTheta, anticlockwise, _x, y); - } - xi = Math.cos(theta + dTheta) * rx + cx; - yi = Math.sin(theta + dTheta) * ry + cy; - break; - case CMD$1.R: - x0 = xi = data[i++]; - y0 = yi = data[i++]; - var width = data[i++]; - var height = data[i++]; - x1 = x0 + width; - y1 = y0 + height; - if (isStroke) { - if (containStroke(x0, y0, x1, y0, lineWidth, x, y) - || containStroke(x1, y0, x1, y1, lineWidth, x, y) - || containStroke(x1, y1, x0, y1, lineWidth, x, y) - || containStroke(x0, y1, x0, y0, lineWidth, x, y)) { - return true; - } - } - else { - w += windingLine(x1, y0, x1, y1, x, y); - w += windingLine(x0, y1, x0, y0, x, y); - } - break; - case CMD$1.Z: - if (isStroke) { - if (containStroke(xi, yi, x0, y0, lineWidth, x, y)) { - return true; - } - } - else { - w += windingLine(xi, yi, x0, y0, x, y); - } - xi = x0; - yi = y0; - break; - } - } - if (!isStroke && !isAroundEqual(yi, y0)) { - w += windingLine(xi, yi, x0, y0, x, y) || 0; - } - return w !== 0; - } - function contain(pathProxy, x, y) { - return containPath(pathProxy, 0, false, x, y); - } - function containStroke$4(pathProxy, lineWidth, x, y) { - return containPath(pathProxy, lineWidth, true, x, y); - } - - var DEFAULT_PATH_STYLE = defaults({ - fill: '#000', - stroke: null, - strokePercent: 1, - fillOpacity: 1, - strokeOpacity: 1, - lineDashOffset: 0, - lineWidth: 1, - lineCap: 'butt', - miterLimit: 10, - strokeNoScale: false, - strokeFirst: false - }, DEFAULT_COMMON_STYLE); - var DEFAULT_PATH_ANIMATION_PROPS = { - style: defaults({ - fill: true, - stroke: true, - strokePercent: true, - fillOpacity: true, - strokeOpacity: true, - lineDashOffset: true, - lineWidth: true, - miterLimit: true - }, DEFAULT_COMMON_ANIMATION_PROPS.style) - }; - var pathCopyParams = TRANSFORMABLE_PROPS.concat(['invisible', - 'culling', 'z', 'z2', 'zlevel', 'parent' - ]); - var Path = (function (_super) { - __extends(Path, _super); - function Path(opts) { - return _super.call(this, opts) || this; - } - Path.prototype.update = function () { - var _this = this; - _super.prototype.update.call(this); - var style = this.style; - if (style.decal) { - var decalEl = this._decalEl = this._decalEl || new Path(); - if (decalEl.buildPath === Path.prototype.buildPath) { - decalEl.buildPath = function (ctx) { - _this.buildPath(ctx, _this.shape); - }; - } - decalEl.silent = true; - var decalElStyle = decalEl.style; - for (var key in style) { - if (decalElStyle[key] !== style[key]) { - decalElStyle[key] = style[key]; - } - } - decalElStyle.fill = style.fill ? style.decal : null; - decalElStyle.decal = null; - decalElStyle.shadowColor = null; - style.strokeFirst && (decalElStyle.stroke = null); - for (var i = 0; i < pathCopyParams.length; ++i) { - decalEl[pathCopyParams[i]] = this[pathCopyParams[i]]; - } - decalEl.__dirty |= REDRAW_BIT; - } - else if (this._decalEl) { - this._decalEl = null; - } - }; - Path.prototype.getDecalElement = function () { - return this._decalEl; - }; - Path.prototype._init = function (props) { - var keysArr = keys(props); - this.shape = this.getDefaultShape(); - var defaultStyle = this.getDefaultStyle(); - if (defaultStyle) { - this.useStyle(defaultStyle); - } - for (var i = 0; i < keysArr.length; i++) { - var key = keysArr[i]; - var value = props[key]; - if (key === 'style') { - if (!this.style) { - this.useStyle(value); - } - else { - extend(this.style, value); - } - } - else if (key === 'shape') { - extend(this.shape, value); - } - else { - _super.prototype.attrKV.call(this, key, value); - } - } - if (!this.style) { - this.useStyle({}); - } - }; - Path.prototype.getDefaultStyle = function () { - return null; - }; - Path.prototype.getDefaultShape = function () { - return {}; - }; - Path.prototype.canBeInsideText = function () { - return this.hasFill(); - }; - Path.prototype.getInsideTextFill = function () { - var pathFill = this.style.fill; - if (pathFill !== 'none') { - if (isString(pathFill)) { - var fillLum = lum(pathFill, 0); - if (fillLum > 0.5) { - return DARK_LABEL_COLOR; - } - else if (fillLum > 0.2) { - return LIGHTER_LABEL_COLOR; - } - return LIGHT_LABEL_COLOR; - } - else if (pathFill) { - return LIGHT_LABEL_COLOR; - } - } - return DARK_LABEL_COLOR; - }; - Path.prototype.getInsideTextStroke = function (textFill) { - var pathFill = this.style.fill; - if (isString(pathFill)) { - var zr = this.__zr; - var isDarkMode = !!(zr && zr.isDarkMode()); - var isDarkLabel = lum(textFill, 0) < DARK_MODE_THRESHOLD; - if (isDarkMode === isDarkLabel) { - return pathFill; - } - } - }; - Path.prototype.buildPath = function (ctx, shapeCfg, inBatch) { }; - Path.prototype.pathUpdated = function () { - this.__dirty &= ~SHAPE_CHANGED_BIT; - }; - Path.prototype.getUpdatedPathProxy = function (inBatch) { - !this.path && this.createPathProxy(); - this.path.beginPath(); - this.buildPath(this.path, this.shape, inBatch); - return this.path; - }; - Path.prototype.createPathProxy = function () { - this.path = new PathProxy(false); - }; - Path.prototype.hasStroke = function () { - var style = this.style; - var stroke = style.stroke; - return !(stroke == null || stroke === 'none' || !(style.lineWidth > 0)); - }; - Path.prototype.hasFill = function () { - var style = this.style; - var fill = style.fill; - return fill != null && fill !== 'none'; - }; - Path.prototype.getBoundingRect = function () { - var rect = this._rect; - var style = this.style; - var needsUpdateRect = !rect; - if (needsUpdateRect) { - var firstInvoke = false; - if (!this.path) { - firstInvoke = true; - this.createPathProxy(); - } - var path = this.path; - if (firstInvoke || (this.__dirty & SHAPE_CHANGED_BIT)) { - path.beginPath(); - this.buildPath(path, this.shape, false); - this.pathUpdated(); - } - rect = path.getBoundingRect(); - } - this._rect = rect; - if (this.hasStroke() && this.path && this.path.len() > 0) { - var rectStroke = this._rectStroke || (this._rectStroke = rect.clone()); - if (this.__dirty || needsUpdateRect) { - rectStroke.copy(rect); - var lineScale = style.strokeNoScale ? this.getLineScale() : 1; - var w = style.lineWidth; - if (!this.hasFill()) { - var strokeContainThreshold = this.strokeContainThreshold; - w = Math.max(w, strokeContainThreshold == null ? 4 : strokeContainThreshold); - } - if (lineScale > 1e-10) { - rectStroke.width += w / lineScale; - rectStroke.height += w / lineScale; - rectStroke.x -= w / lineScale / 2; - rectStroke.y -= w / lineScale / 2; - } - } - return rectStroke; - } - return rect; - }; - Path.prototype.contain = function (x, y) { - var localPos = this.transformCoordToLocal(x, y); - var rect = this.getBoundingRect(); - var style = this.style; - x = localPos[0]; - y = localPos[1]; - if (rect.contain(x, y)) { - var pathProxy = this.path; - if (this.hasStroke()) { - var lineWidth = style.lineWidth; - var lineScale = style.strokeNoScale ? this.getLineScale() : 1; - if (lineScale > 1e-10) { - if (!this.hasFill()) { - lineWidth = Math.max(lineWidth, this.strokeContainThreshold); - } - if (containStroke$4(pathProxy, lineWidth / lineScale, x, y)) { - return true; - } - } - } - if (this.hasFill()) { - return contain(pathProxy, x, y); - } - } - return false; - }; - Path.prototype.dirtyShape = function () { - this.__dirty |= SHAPE_CHANGED_BIT; - if (this._rect) { - this._rect = null; - } - if (this._decalEl) { - this._decalEl.dirtyShape(); - } - this.markRedraw(); - }; - Path.prototype.dirty = function () { - this.dirtyStyle(); - this.dirtyShape(); - }; - Path.prototype.animateShape = function (loop) { - return this.animate('shape', loop); - }; - Path.prototype.updateDuringAnimation = function (targetKey) { - if (targetKey === 'style') { - this.dirtyStyle(); - } - else if (targetKey === 'shape') { - this.dirtyShape(); - } - else { - this.markRedraw(); - } - }; - Path.prototype.attrKV = function (key, value) { - if (key === 'shape') { - this.setShape(value); - } - else { - _super.prototype.attrKV.call(this, key, value); - } - }; - Path.prototype.setShape = function (keyOrObj, value) { - var shape = this.shape; - if (!shape) { - shape = this.shape = {}; - } - if (typeof keyOrObj === 'string') { - shape[keyOrObj] = value; - } - else { - extend(shape, keyOrObj); - } - this.dirtyShape(); - return this; - }; - Path.prototype.shapeChanged = function () { - return !!(this.__dirty & SHAPE_CHANGED_BIT); - }; - Path.prototype.createStyle = function (obj) { - return createObject(DEFAULT_PATH_STYLE, obj); - }; - Path.prototype._innerSaveToNormal = function (toState) { - _super.prototype._innerSaveToNormal.call(this, toState); - var normalState = this._normalState; - if (toState.shape && !normalState.shape) { - normalState.shape = extend({}, this.shape); - } - }; - Path.prototype._applyStateObj = function (stateName, state, normalState, keepCurrentStates, transition, animationCfg) { - _super.prototype._applyStateObj.call(this, stateName, state, normalState, keepCurrentStates, transition, animationCfg); - var needsRestoreToNormal = !(state && keepCurrentStates); - var targetShape; - if (state && state.shape) { - if (transition) { - if (keepCurrentStates) { - targetShape = state.shape; - } - else { - targetShape = extend({}, normalState.shape); - extend(targetShape, state.shape); - } - } - else { - targetShape = extend({}, keepCurrentStates ? this.shape : normalState.shape); - extend(targetShape, state.shape); - } - } - else if (needsRestoreToNormal) { - targetShape = normalState.shape; - } - if (targetShape) { - if (transition) { - this.shape = extend({}, this.shape); - var targetShapePrimaryProps = {}; - var shapeKeys = keys(targetShape); - for (var i = 0; i < shapeKeys.length; i++) { - var key = shapeKeys[i]; - if (typeof targetShape[key] === 'object') { - this.shape[key] = targetShape[key]; - } - else { - targetShapePrimaryProps[key] = targetShape[key]; - } - } - this._transitionState(stateName, { - shape: targetShapePrimaryProps - }, animationCfg); - } - else { - this.shape = targetShape; - this.dirtyShape(); - } - } - }; - Path.prototype._mergeStates = function (states) { - var mergedState = _super.prototype._mergeStates.call(this, states); - var mergedShape; - for (var i = 0; i < states.length; i++) { - var state = states[i]; - if (state.shape) { - mergedShape = mergedShape || {}; - this._mergeStyle(mergedShape, state.shape); - } - } - if (mergedShape) { - mergedState.shape = mergedShape; - } - return mergedState; - }; - Path.prototype.getAnimationStyleProps = function () { - return DEFAULT_PATH_ANIMATION_PROPS; - }; - Path.prototype.isZeroArea = function () { - return false; - }; - Path.extend = function (defaultProps) { - var Sub = (function (_super) { - __extends(Sub, _super); - function Sub(opts) { - var _this = _super.call(this, opts) || this; - defaultProps.init && defaultProps.init.call(_this, opts); - return _this; - } - Sub.prototype.getDefaultStyle = function () { - return clone(defaultProps.style); - }; - Sub.prototype.getDefaultShape = function () { - return clone(defaultProps.shape); - }; - return Sub; - }(Path)); - for (var key in defaultProps) { - if (typeof defaultProps[key] === 'function') { - Sub.prototype[key] = defaultProps[key]; - } - } - return Sub; - }; - Path.initDefaultProps = (function () { - var pathProto = Path.prototype; - pathProto.type = 'path'; - pathProto.strokeContainThreshold = 5; - pathProto.segmentIgnoreThreshold = 0; - pathProto.subPixelOptimize = false; - pathProto.autoBatch = false; - pathProto.__dirty = REDRAW_BIT | STYLE_CHANGED_BIT | SHAPE_CHANGED_BIT; - })(); - return Path; - }(Displayable)); - - var DEFAULT_TSPAN_STYLE = defaults({ - strokeFirst: true, - font: DEFAULT_FONT, - x: 0, - y: 0, - textAlign: 'left', - textBaseline: 'top', - miterLimit: 2 - }, DEFAULT_PATH_STYLE); - var TSpan = (function (_super) { - __extends(TSpan, _super); - function TSpan() { - return _super !== null && _super.apply(this, arguments) || this; - } - TSpan.prototype.hasStroke = function () { - var style = this.style; - var stroke = style.stroke; - return stroke != null && stroke !== 'none' && style.lineWidth > 0; - }; - TSpan.prototype.hasFill = function () { - var style = this.style; - var fill = style.fill; - return fill != null && fill !== 'none'; - }; - TSpan.prototype.createStyle = function (obj) { - return createObject(DEFAULT_TSPAN_STYLE, obj); - }; - TSpan.prototype.setBoundingRect = function (rect) { - this._rect = rect; - }; - TSpan.prototype.getBoundingRect = function () { - var style = this.style; - if (!this._rect) { - var text = style.text; - text != null ? (text += '') : (text = ''); - var rect = getBoundingRect(text, style.font, style.textAlign, style.textBaseline); - rect.x += style.x || 0; - rect.y += style.y || 0; - if (this.hasStroke()) { - var w = style.lineWidth; - rect.x -= w / 2; - rect.y -= w / 2; - rect.width += w; - rect.height += w; - } - this._rect = rect; - } - return this._rect; - }; - TSpan.initDefaultProps = (function () { - var tspanProto = TSpan.prototype; - tspanProto.dirtyRectTolerance = 10; - })(); - return TSpan; - }(Displayable)); - TSpan.prototype.type = 'tspan'; - - var DEFAULT_IMAGE_STYLE = defaults({ - x: 0, - y: 0 - }, DEFAULT_COMMON_STYLE); - var DEFAULT_IMAGE_ANIMATION_PROPS = { - style: defaults({ - x: true, - y: true, - width: true, - height: true, - sx: true, - sy: true, - sWidth: true, - sHeight: true - }, DEFAULT_COMMON_ANIMATION_PROPS.style) - }; - function isImageLike(source) { - return !!(source - && typeof source !== 'string' - && source.width && source.height); - } - var ZRImage = (function (_super) { - __extends(ZRImage, _super); - function ZRImage() { - return _super !== null && _super.apply(this, arguments) || this; - } - ZRImage.prototype.createStyle = function (obj) { - return createObject(DEFAULT_IMAGE_STYLE, obj); - }; - ZRImage.prototype._getSize = function (dim) { - var style = this.style; - var size = style[dim]; - if (size != null) { - return size; - } - var imageSource = isImageLike(style.image) - ? style.image : this.__image; - if (!imageSource) { - return 0; - } - var otherDim = dim === 'width' ? 'height' : 'width'; - var otherDimSize = style[otherDim]; - if (otherDimSize == null) { - return imageSource[dim]; - } - else { - return imageSource[dim] / imageSource[otherDim] * otherDimSize; - } - }; - ZRImage.prototype.getWidth = function () { - return this._getSize('width'); - }; - ZRImage.prototype.getHeight = function () { - return this._getSize('height'); - }; - ZRImage.prototype.getAnimationStyleProps = function () { - return DEFAULT_IMAGE_ANIMATION_PROPS; - }; - ZRImage.prototype.getBoundingRect = function () { - var style = this.style; - if (!this._rect) { - this._rect = new BoundingRect(style.x || 0, style.y || 0, this.getWidth(), this.getHeight()); - } - return this._rect; - }; - return ZRImage; - }(Displayable)); - ZRImage.prototype.type = 'image'; - - function buildPath(ctx, shape) { - var x = shape.x; - var y = shape.y; - var width = shape.width; - var height = shape.height; - var r = shape.r; - var r1; - var r2; - var r3; - var r4; - if (width < 0) { - x = x + width; - width = -width; - } - if (height < 0) { - y = y + height; - height = -height; - } - if (typeof r === 'number') { - r1 = r2 = r3 = r4 = r; - } - else if (r instanceof Array) { - if (r.length === 1) { - r1 = r2 = r3 = r4 = r[0]; - } - else if (r.length === 2) { - r1 = r3 = r[0]; - r2 = r4 = r[1]; - } - else if (r.length === 3) { - r1 = r[0]; - r2 = r4 = r[1]; - r3 = r[2]; - } - else { - r1 = r[0]; - r2 = r[1]; - r3 = r[2]; - r4 = r[3]; - } - } - else { - r1 = r2 = r3 = r4 = 0; - } - var total; - if (r1 + r2 > width) { - total = r1 + r2; - r1 *= width / total; - r2 *= width / total; - } - if (r3 + r4 > width) { - total = r3 + r4; - r3 *= width / total; - r4 *= width / total; - } - if (r2 + r3 > height) { - total = r2 + r3; - r2 *= height / total; - r3 *= height / total; - } - if (r1 + r4 > height) { - total = r1 + r4; - r1 *= height / total; - r4 *= height / total; - } - ctx.moveTo(x + r1, y); - ctx.lineTo(x + width - r2, y); - r2 !== 0 && ctx.arc(x + width - r2, y + r2, r2, -Math.PI / 2, 0); - ctx.lineTo(x + width, y + height - r3); - r3 !== 0 && ctx.arc(x + width - r3, y + height - r3, r3, 0, Math.PI / 2); - ctx.lineTo(x + r4, y + height); - r4 !== 0 && ctx.arc(x + r4, y + height - r4, r4, Math.PI / 2, Math.PI); - ctx.lineTo(x, y + r1); - r1 !== 0 && ctx.arc(x + r1, y + r1, r1, Math.PI, Math.PI * 1.5); - } - - var round$1 = Math.round; - function subPixelOptimizeLine(outputShape, inputShape, style) { - if (!inputShape) { - return; - } - var x1 = inputShape.x1; - var x2 = inputShape.x2; - var y1 = inputShape.y1; - var y2 = inputShape.y2; - outputShape.x1 = x1; - outputShape.x2 = x2; - outputShape.y1 = y1; - outputShape.y2 = y2; - var lineWidth = style && style.lineWidth; - if (!lineWidth) { - return outputShape; - } - if (round$1(x1 * 2) === round$1(x2 * 2)) { - outputShape.x1 = outputShape.x2 = subPixelOptimize(x1, lineWidth, true); - } - if (round$1(y1 * 2) === round$1(y2 * 2)) { - outputShape.y1 = outputShape.y2 = subPixelOptimize(y1, lineWidth, true); - } - return outputShape; - } - function subPixelOptimizeRect(outputShape, inputShape, style) { - if (!inputShape) { - return; - } - var originX = inputShape.x; - var originY = inputShape.y; - var originWidth = inputShape.width; - var originHeight = inputShape.height; - outputShape.x = originX; - outputShape.y = originY; - outputShape.width = originWidth; - outputShape.height = originHeight; - var lineWidth = style && style.lineWidth; - if (!lineWidth) { - return outputShape; - } - outputShape.x = subPixelOptimize(originX, lineWidth, true); - outputShape.y = subPixelOptimize(originY, lineWidth, true); - outputShape.width = Math.max(subPixelOptimize(originX + originWidth, lineWidth, false) - outputShape.x, originWidth === 0 ? 0 : 1); - outputShape.height = Math.max(subPixelOptimize(originY + originHeight, lineWidth, false) - outputShape.y, originHeight === 0 ? 0 : 1); - return outputShape; - } - function subPixelOptimize(position, lineWidth, positiveOrNegative) { - if (!lineWidth) { - return position; - } - var doubledPosition = round$1(position * 2); - return (doubledPosition + round$1(lineWidth)) % 2 === 0 - ? doubledPosition / 2 - : (doubledPosition + (positiveOrNegative ? 1 : -1)) / 2; - } - - var RectShape = (function () { - function RectShape() { - this.x = 0; - this.y = 0; - this.width = 0; - this.height = 0; - } - return RectShape; - }()); - var subPixelOptimizeOutputShape = {}; - var Rect = (function (_super) { - __extends(Rect, _super); - function Rect(opts) { - return _super.call(this, opts) || this; - } - Rect.prototype.getDefaultShape = function () { - return new RectShape(); - }; - Rect.prototype.buildPath = function (ctx, shape) { - var x; - var y; - var width; - var height; - if (this.subPixelOptimize) { - var optimizedShape = subPixelOptimizeRect(subPixelOptimizeOutputShape, shape, this.style); - x = optimizedShape.x; - y = optimizedShape.y; - width = optimizedShape.width; - height = optimizedShape.height; - optimizedShape.r = shape.r; - shape = optimizedShape; - } - else { - x = shape.x; - y = shape.y; - width = shape.width; - height = shape.height; - } - if (!shape.r) { - ctx.rect(x, y, width, height); - } - else { - buildPath(ctx, shape); - } - }; - Rect.prototype.isZeroArea = function () { - return !this.shape.width || !this.shape.height; - }; - return Rect; - }(Path)); - Rect.prototype.type = 'rect'; - - var DEFAULT_RICH_TEXT_COLOR = { - fill: '#000' - }; - var DEFAULT_STROKE_LINE_WIDTH = 2; - var DEFAULT_TEXT_ANIMATION_PROPS = { - style: defaults({ - fill: true, - stroke: true, - fillOpacity: true, - strokeOpacity: true, - lineWidth: true, - fontSize: true, - lineHeight: true, - width: true, - height: true, - textShadowColor: true, - textShadowBlur: true, - textShadowOffsetX: true, - textShadowOffsetY: true, - backgroundColor: true, - padding: true, - borderColor: true, - borderWidth: true, - borderRadius: true - }, DEFAULT_COMMON_ANIMATION_PROPS.style) - }; - var ZRText = (function (_super) { - __extends(ZRText, _super); - function ZRText(opts) { - var _this = _super.call(this) || this; - _this.type = 'text'; - _this._children = []; - _this._defaultStyle = DEFAULT_RICH_TEXT_COLOR; - _this.attr(opts); - return _this; - } - ZRText.prototype.childrenRef = function () { - return this._children; - }; - ZRText.prototype.update = function () { - _super.prototype.update.call(this); - if (this.styleChanged()) { - this._updateSubTexts(); - } - for (var i = 0; i < this._children.length; i++) { - var child = this._children[i]; - child.zlevel = this.zlevel; - child.z = this.z; - child.z2 = this.z2; - child.culling = this.culling; - child.cursor = this.cursor; - child.invisible = this.invisible; - } - }; - ZRText.prototype.updateTransform = function () { - var innerTransformable = this.innerTransformable; - if (innerTransformable) { - innerTransformable.updateTransform(); - if (innerTransformable.transform) { - this.transform = innerTransformable.transform; - } - } - else { - _super.prototype.updateTransform.call(this); - } - }; - ZRText.prototype.getLocalTransform = function (m) { - var innerTransformable = this.innerTransformable; - return innerTransformable - ? innerTransformable.getLocalTransform(m) - : _super.prototype.getLocalTransform.call(this, m); - }; - ZRText.prototype.getComputedTransform = function () { - if (this.__hostTarget) { - this.__hostTarget.getComputedTransform(); - this.__hostTarget.updateInnerText(true); - } - return _super.prototype.getComputedTransform.call(this); - }; - ZRText.prototype._updateSubTexts = function () { - this._childCursor = 0; - normalizeTextStyle(this.style); - this.style.rich - ? this._updateRichTexts() - : this._updatePlainTexts(); - this._children.length = this._childCursor; - this.styleUpdated(); - }; - ZRText.prototype.addSelfToZr = function (zr) { - _super.prototype.addSelfToZr.call(this, zr); - for (var i = 0; i < this._children.length; i++) { - this._children[i].__zr = zr; - } - }; - ZRText.prototype.removeSelfFromZr = function (zr) { - _super.prototype.removeSelfFromZr.call(this, zr); - for (var i = 0; i < this._children.length; i++) { - this._children[i].__zr = null; - } - }; - ZRText.prototype.getBoundingRect = function () { - if (this.styleChanged()) { - this._updateSubTexts(); - } - if (!this._rect) { - var tmpRect = new BoundingRect(0, 0, 0, 0); - var children = this._children; - var tmpMat = []; - var rect = null; - for (var i = 0; i < children.length; i++) { - var child = children[i]; - var childRect = child.getBoundingRect(); - var transform = child.getLocalTransform(tmpMat); - if (transform) { - tmpRect.copy(childRect); - tmpRect.applyTransform(transform); - rect = rect || tmpRect.clone(); - rect.union(tmpRect); - } - else { - rect = rect || childRect.clone(); - rect.union(childRect); - } - } - this._rect = rect || tmpRect; - } - return this._rect; - }; - ZRText.prototype.setDefaultTextStyle = function (defaultTextStyle) { - this._defaultStyle = defaultTextStyle || DEFAULT_RICH_TEXT_COLOR; - }; - ZRText.prototype.setTextContent = function (textContent) { - if ("development" !== 'production') { - throw new Error('Can\'t attach text on another text'); - } - }; - ZRText.prototype._mergeStyle = function (targetStyle, sourceStyle) { - if (!sourceStyle) { - return targetStyle; - } - var sourceRich = sourceStyle.rich; - var targetRich = targetStyle.rich || (sourceRich && {}); - extend(targetStyle, sourceStyle); - if (sourceRich && targetRich) { - this._mergeRich(targetRich, sourceRich); - targetStyle.rich = targetRich; - } - else if (targetRich) { - targetStyle.rich = targetRich; - } - return targetStyle; - }; - ZRText.prototype._mergeRich = function (targetRich, sourceRich) { - var richNames = keys(sourceRich); - for (var i = 0; i < richNames.length; i++) { - var richName = richNames[i]; - targetRich[richName] = targetRich[richName] || {}; - extend(targetRich[richName], sourceRich[richName]); - } - }; - ZRText.prototype.getAnimationStyleProps = function () { - return DEFAULT_TEXT_ANIMATION_PROPS; - }; - ZRText.prototype._getOrCreateChild = function (Ctor) { - var child = this._children[this._childCursor]; - if (!child || !(child instanceof Ctor)) { - child = new Ctor(); - } - this._children[this._childCursor++] = child; - child.__zr = this.__zr; - child.parent = this; - return child; - }; - ZRText.prototype._updatePlainTexts = function () { - var style = this.style; - var textFont = style.font || DEFAULT_FONT; - var textPadding = style.padding; - var text = getStyleText(style); - var contentBlock = parsePlainText(text, style); - var needDrawBg = needDrawBackground(style); - var bgColorDrawn = !!(style.backgroundColor); - var outerHeight = contentBlock.outerHeight; - var outerWidth = contentBlock.outerWidth; - var contentWidth = contentBlock.contentWidth; - var textLines = contentBlock.lines; - var lineHeight = contentBlock.lineHeight; - var defaultStyle = this._defaultStyle; - var baseX = style.x || 0; - var baseY = style.y || 0; - var textAlign = style.align || defaultStyle.align || 'left'; - var verticalAlign = style.verticalAlign || defaultStyle.verticalAlign || 'top'; - var textX = baseX; - var textY = adjustTextY$1(baseY, contentBlock.contentHeight, verticalAlign); - if (needDrawBg || textPadding) { - var boxX = adjustTextX(baseX, outerWidth, textAlign); - var boxY = adjustTextY$1(baseY, outerHeight, verticalAlign); - needDrawBg && this._renderBackground(style, style, boxX, boxY, outerWidth, outerHeight); - } - textY += lineHeight / 2; - if (textPadding) { - textX = getTextXForPadding(baseX, textAlign, textPadding); - if (verticalAlign === 'top') { - textY += textPadding[0]; - } - else if (verticalAlign === 'bottom') { - textY -= textPadding[2]; - } - } - var defaultLineWidth = 0; - var useDefaultFill = false; - var textFill = getFill('fill' in style - ? style.fill - : (useDefaultFill = true, defaultStyle.fill)); - var textStroke = getStroke('stroke' in style - ? style.stroke - : (!bgColorDrawn - && (!defaultStyle.autoStroke || useDefaultFill)) - ? (defaultLineWidth = DEFAULT_STROKE_LINE_WIDTH, defaultStyle.stroke) - : null); - var hasShadow = style.textShadowBlur > 0; - var fixedBoundingRect = style.width != null - && (style.overflow === 'truncate' || style.overflow === 'break' || style.overflow === 'breakAll'); - var calculatedLineHeight = contentBlock.calculatedLineHeight; - for (var i = 0; i < textLines.length; i++) { - var el = this._getOrCreateChild(TSpan); - var subElStyle = el.createStyle(); - el.useStyle(subElStyle); - subElStyle.text = textLines[i]; - subElStyle.x = textX; - subElStyle.y = textY; - if (textAlign) { - subElStyle.textAlign = textAlign; - } - subElStyle.textBaseline = 'middle'; - subElStyle.opacity = style.opacity; - subElStyle.strokeFirst = true; - if (hasShadow) { - subElStyle.shadowBlur = style.textShadowBlur || 0; - subElStyle.shadowColor = style.textShadowColor || 'transparent'; - subElStyle.shadowOffsetX = style.textShadowOffsetX || 0; - subElStyle.shadowOffsetY = style.textShadowOffsetY || 0; - } - subElStyle.stroke = textStroke; - subElStyle.fill = textFill; - if (textStroke) { - subElStyle.lineWidth = style.lineWidth || defaultLineWidth; - subElStyle.lineDash = style.lineDash; - subElStyle.lineDashOffset = style.lineDashOffset || 0; - } - subElStyle.font = textFont; - setSeparateFont(subElStyle, style); - textY += lineHeight; - if (fixedBoundingRect) { - el.setBoundingRect(new BoundingRect(adjustTextX(subElStyle.x, style.width, subElStyle.textAlign), adjustTextY$1(subElStyle.y, calculatedLineHeight, subElStyle.textBaseline), contentWidth, calculatedLineHeight)); - } - } - }; - ZRText.prototype._updateRichTexts = function () { - var style = this.style; - var text = getStyleText(style); - var contentBlock = parseRichText(text, style); - var contentWidth = contentBlock.width; - var outerWidth = contentBlock.outerWidth; - var outerHeight = contentBlock.outerHeight; - var textPadding = style.padding; - var baseX = style.x || 0; - var baseY = style.y || 0; - var defaultStyle = this._defaultStyle; - var textAlign = style.align || defaultStyle.align; - var verticalAlign = style.verticalAlign || defaultStyle.verticalAlign; - var boxX = adjustTextX(baseX, outerWidth, textAlign); - var boxY = adjustTextY$1(baseY, outerHeight, verticalAlign); - var xLeft = boxX; - var lineTop = boxY; - if (textPadding) { - xLeft += textPadding[3]; - lineTop += textPadding[0]; - } - var xRight = xLeft + contentWidth; - if (needDrawBackground(style)) { - this._renderBackground(style, style, boxX, boxY, outerWidth, outerHeight); - } - var bgColorDrawn = !!(style.backgroundColor); - for (var i = 0; i < contentBlock.lines.length; i++) { - var line = contentBlock.lines[i]; - var tokens = line.tokens; - var tokenCount = tokens.length; - var lineHeight = line.lineHeight; - var remainedWidth = line.width; - var leftIndex = 0; - var lineXLeft = xLeft; - var lineXRight = xRight; - var rightIndex = tokenCount - 1; - var token = void 0; - while (leftIndex < tokenCount - && (token = tokens[leftIndex], !token.align || token.align === 'left')) { - this._placeToken(token, style, lineHeight, lineTop, lineXLeft, 'left', bgColorDrawn); - remainedWidth -= token.width; - lineXLeft += token.width; - leftIndex++; - } - while (rightIndex >= 0 - && (token = tokens[rightIndex], token.align === 'right')) { - this._placeToken(token, style, lineHeight, lineTop, lineXRight, 'right', bgColorDrawn); - remainedWidth -= token.width; - lineXRight -= token.width; - rightIndex--; - } - lineXLeft += (contentWidth - (lineXLeft - xLeft) - (xRight - lineXRight) - remainedWidth) / 2; - while (leftIndex <= rightIndex) { - token = tokens[leftIndex]; - this._placeToken(token, style, lineHeight, lineTop, lineXLeft + token.width / 2, 'center', bgColorDrawn); - lineXLeft += token.width; - leftIndex++; - } - lineTop += lineHeight; - } - }; - ZRText.prototype._placeToken = function (token, style, lineHeight, lineTop, x, textAlign, parentBgColorDrawn) { - var tokenStyle = style.rich[token.styleName] || {}; - tokenStyle.text = token.text; - var verticalAlign = token.verticalAlign; - var y = lineTop + lineHeight / 2; - if (verticalAlign === 'top') { - y = lineTop + token.height / 2; - } - else if (verticalAlign === 'bottom') { - y = lineTop + lineHeight - token.height / 2; - } - var needDrawBg = !token.isLineHolder && needDrawBackground(tokenStyle); - needDrawBg && this._renderBackground(tokenStyle, style, textAlign === 'right' - ? x - token.width - : textAlign === 'center' - ? x - token.width / 2 - : x, y - token.height / 2, token.width, token.height); - var bgColorDrawn = !!tokenStyle.backgroundColor; - var textPadding = token.textPadding; - if (textPadding) { - x = getTextXForPadding(x, textAlign, textPadding); - y -= token.height / 2 - textPadding[0] - token.innerHeight / 2; - } - var el = this._getOrCreateChild(TSpan); - var subElStyle = el.createStyle(); - el.useStyle(subElStyle); - var defaultStyle = this._defaultStyle; - var useDefaultFill = false; - var defaultLineWidth = 0; - var textFill = getFill('fill' in tokenStyle ? tokenStyle.fill - : 'fill' in style ? style.fill - : (useDefaultFill = true, defaultStyle.fill)); - var textStroke = getStroke('stroke' in tokenStyle ? tokenStyle.stroke - : 'stroke' in style ? style.stroke - : (!bgColorDrawn - && !parentBgColorDrawn - && (!defaultStyle.autoStroke || useDefaultFill)) ? (defaultLineWidth = DEFAULT_STROKE_LINE_WIDTH, defaultStyle.stroke) - : null); - var hasShadow = tokenStyle.textShadowBlur > 0 - || style.textShadowBlur > 0; - subElStyle.text = token.text; - subElStyle.x = x; - subElStyle.y = y; - if (hasShadow) { - subElStyle.shadowBlur = tokenStyle.textShadowBlur || style.textShadowBlur || 0; - subElStyle.shadowColor = tokenStyle.textShadowColor || style.textShadowColor || 'transparent'; - subElStyle.shadowOffsetX = tokenStyle.textShadowOffsetX || style.textShadowOffsetX || 0; - subElStyle.shadowOffsetY = tokenStyle.textShadowOffsetY || style.textShadowOffsetY || 0; - } - subElStyle.textAlign = textAlign; - subElStyle.textBaseline = 'middle'; - subElStyle.font = token.font || DEFAULT_FONT; - subElStyle.opacity = retrieve3(tokenStyle.opacity, style.opacity, 1); - setSeparateFont(subElStyle, tokenStyle); - if (textStroke) { - subElStyle.lineWidth = retrieve3(tokenStyle.lineWidth, style.lineWidth, defaultLineWidth); - subElStyle.lineDash = retrieve2(tokenStyle.lineDash, style.lineDash); - subElStyle.lineDashOffset = style.lineDashOffset || 0; - subElStyle.stroke = textStroke; - } - if (textFill) { - subElStyle.fill = textFill; - } - var textWidth = token.contentWidth; - var textHeight = token.contentHeight; - el.setBoundingRect(new BoundingRect(adjustTextX(subElStyle.x, textWidth, subElStyle.textAlign), adjustTextY$1(subElStyle.y, textHeight, subElStyle.textBaseline), textWidth, textHeight)); - }; - ZRText.prototype._renderBackground = function (style, topStyle, x, y, width, height) { - var textBackgroundColor = style.backgroundColor; - var textBorderWidth = style.borderWidth; - var textBorderColor = style.borderColor; - var isImageBg = textBackgroundColor && textBackgroundColor.image; - var isPlainOrGradientBg = textBackgroundColor && !isImageBg; - var textBorderRadius = style.borderRadius; - var self = this; - var rectEl; - var imgEl; - if (isPlainOrGradientBg || style.lineHeight || (textBorderWidth && textBorderColor)) { - rectEl = this._getOrCreateChild(Rect); - rectEl.useStyle(rectEl.createStyle()); - rectEl.style.fill = null; - var rectShape = rectEl.shape; - rectShape.x = x; - rectShape.y = y; - rectShape.width = width; - rectShape.height = height; - rectShape.r = textBorderRadius; - rectEl.dirtyShape(); - } - if (isPlainOrGradientBg) { - var rectStyle = rectEl.style; - rectStyle.fill = textBackgroundColor || null; - rectStyle.fillOpacity = retrieve2(style.fillOpacity, 1); - } - else if (isImageBg) { - imgEl = this._getOrCreateChild(ZRImage); - imgEl.onload = function () { - self.dirtyStyle(); - }; - var imgStyle = imgEl.style; - imgStyle.image = textBackgroundColor.image; - imgStyle.x = x; - imgStyle.y = y; - imgStyle.width = width; - imgStyle.height = height; - } - if (textBorderWidth && textBorderColor) { - var rectStyle = rectEl.style; - rectStyle.lineWidth = textBorderWidth; - rectStyle.stroke = textBorderColor; - rectStyle.strokeOpacity = retrieve2(style.strokeOpacity, 1); - rectStyle.lineDash = style.borderDash; - rectStyle.lineDashOffset = style.borderDashOffset || 0; - rectEl.strokeContainThreshold = 0; - if (rectEl.hasFill() && rectEl.hasStroke()) { - rectStyle.strokeFirst = true; - rectStyle.lineWidth *= 2; - } - } - var commonStyle = (rectEl || imgEl).style; - commonStyle.shadowBlur = style.shadowBlur || 0; - commonStyle.shadowColor = style.shadowColor || 'transparent'; - commonStyle.shadowOffsetX = style.shadowOffsetX || 0; - commonStyle.shadowOffsetY = style.shadowOffsetY || 0; - commonStyle.opacity = retrieve3(style.opacity, topStyle.opacity, 1); - }; - ZRText.makeFont = function (style) { - var font = ''; - if (hasSeparateFont(style)) { - font = [ - style.fontStyle, - style.fontWeight, - parseFontSize(style.fontSize), - style.fontFamily || 'sans-serif' - ].join(' '); - } - return font && trim(font) || style.textFont || style.font; - }; - return ZRText; - }(Displayable)); - var VALID_TEXT_ALIGN = { left: true, right: 1, center: 1 }; - var VALID_TEXT_VERTICAL_ALIGN = { top: 1, bottom: 1, middle: 1 }; - var FONT_PARTS = ['fontStyle', 'fontWeight', 'fontSize', 'fontFamily']; - function parseFontSize(fontSize) { - if (typeof fontSize === 'string' - && (fontSize.indexOf('px') !== -1 - || fontSize.indexOf('rem') !== -1 - || fontSize.indexOf('em') !== -1)) { - return fontSize; - } - else if (!isNaN(+fontSize)) { - return fontSize + 'px'; - } - else { - return DEFAULT_FONT_SIZE + 'px'; - } - } - function setSeparateFont(targetStyle, sourceStyle) { - for (var i = 0; i < FONT_PARTS.length; i++) { - var fontProp = FONT_PARTS[i]; - var val = sourceStyle[fontProp]; - if (val != null) { - targetStyle[fontProp] = val; - } - } - } - function hasSeparateFont(style) { - return style.fontSize != null || style.fontFamily || style.fontWeight; - } - function normalizeTextStyle(style) { - normalizeStyle(style); - each(style.rich, normalizeStyle); - return style; - } - function normalizeStyle(style) { - if (style) { - style.font = ZRText.makeFont(style); - var textAlign = style.align; - textAlign === 'middle' && (textAlign = 'center'); - style.align = (textAlign == null || VALID_TEXT_ALIGN[textAlign]) ? textAlign : 'left'; - var verticalAlign = style.verticalAlign; - verticalAlign === 'center' && (verticalAlign = 'middle'); - style.verticalAlign = (verticalAlign == null || VALID_TEXT_VERTICAL_ALIGN[verticalAlign]) ? verticalAlign : 'top'; - var textPadding = style.padding; - if (textPadding) { - style.padding = normalizeCssArray(style.padding); - } - } - } - function getStroke(stroke, lineWidth) { - return (stroke == null || lineWidth <= 0 || stroke === 'transparent' || stroke === 'none') - ? null - : (stroke.image || stroke.colorStops) - ? '#000' - : stroke; - } - function getFill(fill) { - return (fill == null || fill === 'none') - ? null - : (fill.image || fill.colorStops) - ? '#000' - : fill; - } - function getTextXForPadding(x, textAlign, textPadding) { - return textAlign === 'right' - ? (x - textPadding[1]) - : textAlign === 'center' - ? (x + textPadding[3] / 2 - textPadding[1] / 2) - : (x + textPadding[3]); - } - function getStyleText(style) { - var text = style.text; - text != null && (text += ''); - return text; - } - function needDrawBackground(style) { - return !!(style.backgroundColor - || style.lineHeight - || (style.borderWidth && style.borderColor)); - } - - var getECData = makeInner(); - var setCommonECData = function (seriesIndex, dataType, dataIdx, el) { - if (el) { - var ecData = getECData(el); // Add data index and series index for indexing the data by element - // Useful in tooltip - - ecData.dataIndex = dataIdx; - ecData.dataType = dataType; - ecData.seriesIndex = seriesIndex; // TODO: not store dataIndex on children. - - if (el.type === 'group') { - el.traverse(function (child) { - var childECData = getECData(child); - childECData.seriesIndex = seriesIndex; - childECData.dataIndex = dataIdx; - childECData.dataType = dataType; - }); - } - } - }; - - var _highlightNextDigit = 1; - var _highlightKeyMap = {}; - var getSavedStates = makeInner(); - var getComponentStates = makeInner(); - var HOVER_STATE_NORMAL = 0; - var HOVER_STATE_BLUR = 1; - var HOVER_STATE_EMPHASIS = 2; - var SPECIAL_STATES = ['emphasis', 'blur', 'select']; - var DISPLAY_STATES = ['normal', 'emphasis', 'blur', 'select']; - var Z2_EMPHASIS_LIFT = 10; - var Z2_SELECT_LIFT = 9; - var HIGHLIGHT_ACTION_TYPE = 'highlight'; - var DOWNPLAY_ACTION_TYPE = 'downplay'; - var SELECT_ACTION_TYPE = 'select'; - var UNSELECT_ACTION_TYPE = 'unselect'; - var TOGGLE_SELECT_ACTION_TYPE = 'toggleSelect'; - - function hasFillOrStroke(fillOrStroke) { - return fillOrStroke != null && fillOrStroke !== 'none'; - } // Most lifted color are duplicated. - - - var liftedColorCache = new LRU(100); - - function liftColor(color$1) { - if (isString(color$1)) { - var liftedColor = liftedColorCache.get(color$1); - - if (!liftedColor) { - liftedColor = lift(color$1, -0.1); - liftedColorCache.put(color$1, liftedColor); - } - - return liftedColor; - } else if (isGradientObject(color$1)) { - var ret = extend({}, color$1); - ret.colorStops = map(color$1.colorStops, function (stop) { - return { - offset: stop.offset, - color: lift(stop.color, -0.1) - }; - }); - return ret; - } // Change nothing. - - - return color$1; - } - - function doChangeHoverState(el, stateName, hoverStateEnum) { - if (el.onHoverStateChange && (el.hoverState || 0) !== hoverStateEnum) { - el.onHoverStateChange(stateName); - } - - el.hoverState = hoverStateEnum; - } - - function singleEnterEmphasis(el) { - // Only mark the flag. - // States will be applied in the echarts.ts in next frame. - doChangeHoverState(el, 'emphasis', HOVER_STATE_EMPHASIS); - } - - function singleLeaveEmphasis(el) { - // Only mark the flag. - // States will be applied in the echarts.ts in next frame. - if (el.hoverState === HOVER_STATE_EMPHASIS) { - doChangeHoverState(el, 'normal', HOVER_STATE_NORMAL); - } - } - - function singleEnterBlur(el) { - doChangeHoverState(el, 'blur', HOVER_STATE_BLUR); - } - - function singleLeaveBlur(el) { - if (el.hoverState === HOVER_STATE_BLUR) { - doChangeHoverState(el, 'normal', HOVER_STATE_NORMAL); - } - } - - function singleEnterSelect(el) { - el.selected = true; - } - - function singleLeaveSelect(el) { - el.selected = false; - } - - function updateElementState(el, updater, commonParam) { - updater(el, commonParam); - } - - function traverseUpdateState(el, updater, commonParam) { - updateElementState(el, updater, commonParam); - el.isGroup && el.traverse(function (child) { - updateElementState(child, updater, commonParam); - }); - } - - function setStatesFlag(el, stateName) { - switch (stateName) { - case 'emphasis': - el.hoverState = HOVER_STATE_EMPHASIS; - break; - - case 'normal': - el.hoverState = HOVER_STATE_NORMAL; - break; - - case 'blur': - el.hoverState = HOVER_STATE_BLUR; - break; - - case 'select': - el.selected = true; - } - } - - function getFromStateStyle(el, props, toStateName, defaultValue) { - var style = el.style; - var fromState = {}; - - for (var i = 0; i < props.length; i++) { - var propName = props[i]; - var val = style[propName]; - fromState[propName] = val == null ? defaultValue && defaultValue[propName] : val; - } - - for (var i = 0; i < el.animators.length; i++) { - var animator = el.animators[i]; - - if (animator.__fromStateTransition // Don't consider the animation to emphasis state. - && animator.__fromStateTransition.indexOf(toStateName) < 0 && animator.targetName === 'style') { - animator.saveTo(fromState, props); - } - } - - return fromState; - } - - function createEmphasisDefaultState(el, stateName, targetStates, state) { - var hasSelect = targetStates && indexOf(targetStates, 'select') >= 0; - var cloned = false; - - if (el instanceof Path) { - var store = getSavedStates(el); - var fromFill = hasSelect ? store.selectFill || store.normalFill : store.normalFill; - var fromStroke = hasSelect ? store.selectStroke || store.normalStroke : store.normalStroke; - - if (hasFillOrStroke(fromFill) || hasFillOrStroke(fromStroke)) { - state = state || {}; - var emphasisStyle = state.style || {}; // inherit case - - if (emphasisStyle.fill === 'inherit') { - cloned = true; - state = extend({}, state); - emphasisStyle = extend({}, emphasisStyle); - emphasisStyle.fill = fromFill; - } // Apply default color lift - else if (!hasFillOrStroke(emphasisStyle.fill) && hasFillOrStroke(fromFill)) { - cloned = true; // Not modify the original value. - - state = extend({}, state); - emphasisStyle = extend({}, emphasisStyle); // Already being applied 'emphasis'. DON'T lift color multiple times. - - emphasisStyle.fill = liftColor(fromFill); - } // Not highlight stroke if fill has been highlighted. - else if (!hasFillOrStroke(emphasisStyle.stroke) && hasFillOrStroke(fromStroke)) { - if (!cloned) { - state = extend({}, state); - emphasisStyle = extend({}, emphasisStyle); - } - - emphasisStyle.stroke = liftColor(fromStroke); - } - - state.style = emphasisStyle; - } - } - - if (state) { - // TODO Share with textContent? - if (state.z2 == null) { - if (!cloned) { - state = extend({}, state); - } - - var z2EmphasisLift = el.z2EmphasisLift; - state.z2 = el.z2 + (z2EmphasisLift != null ? z2EmphasisLift : Z2_EMPHASIS_LIFT); - } - } - - return state; - } - - function createSelectDefaultState(el, stateName, state) { - // const hasSelect = indexOf(el.currentStates, stateName) >= 0; - if (state) { - // TODO Share with textContent? - if (state.z2 == null) { - state = extend({}, state); - var z2SelectLift = el.z2SelectLift; - state.z2 = el.z2 + (z2SelectLift != null ? z2SelectLift : Z2_SELECT_LIFT); - } - } - - return state; - } - - function createBlurDefaultState(el, stateName, state) { - var hasBlur = indexOf(el.currentStates, stateName) >= 0; - var currentOpacity = el.style.opacity; - var fromState = !hasBlur ? getFromStateStyle(el, ['opacity'], stateName, { - opacity: 1 - }) : null; - state = state || {}; - var blurStyle = state.style || {}; - - if (blurStyle.opacity == null) { - // clone state - state = extend({}, state); - blurStyle = extend({ - // Already being applied 'emphasis'. DON'T mul opacity multiple times. - opacity: hasBlur ? currentOpacity : fromState.opacity * 0.1 - }, blurStyle); - state.style = blurStyle; - } - - return state; - } - - function elementStateProxy(stateName, targetStates) { - var state = this.states[stateName]; - - if (this.style) { - if (stateName === 'emphasis') { - return createEmphasisDefaultState(this, stateName, targetStates, state); - } else if (stateName === 'blur') { - return createBlurDefaultState(this, stateName, state); - } else if (stateName === 'select') { - return createSelectDefaultState(this, stateName, state); - } - } - - return state; - } - /** - * Set hover style (namely "emphasis style") of element. - * @param el Should not be `zrender/graphic/Group`. - * @param focus 'self' | 'selfInSeries' | 'series' - */ - - - function setDefaultStateProxy(el) { - el.stateProxy = elementStateProxy; - var textContent = el.getTextContent(); - var textGuide = el.getTextGuideLine(); - - if (textContent) { - textContent.stateProxy = elementStateProxy; - } - - if (textGuide) { - textGuide.stateProxy = elementStateProxy; - } - } - function enterEmphasisWhenMouseOver(el, e) { - !shouldSilent(el, e) // "emphasis" event highlight has higher priority than mouse highlight. - && !el.__highByOuter && traverseUpdateState(el, singleEnterEmphasis); - } - function leaveEmphasisWhenMouseOut(el, e) { - !shouldSilent(el, e) // "emphasis" event highlight has higher priority than mouse highlight. - && !el.__highByOuter && traverseUpdateState(el, singleLeaveEmphasis); - } - function enterEmphasis(el, highlightDigit) { - el.__highByOuter |= 1 << (highlightDigit || 0); - traverseUpdateState(el, singleEnterEmphasis); - } - function leaveEmphasis(el, highlightDigit) { - !(el.__highByOuter &= ~(1 << (highlightDigit || 0))) && traverseUpdateState(el, singleLeaveEmphasis); - } - function enterBlur(el) { - traverseUpdateState(el, singleEnterBlur); - } - function leaveBlur(el) { - traverseUpdateState(el, singleLeaveBlur); - } - function enterSelect(el) { - traverseUpdateState(el, singleEnterSelect); - } - function leaveSelect(el) { - traverseUpdateState(el, singleLeaveSelect); - } - - function shouldSilent(el, e) { - return el.__highDownSilentOnTouch && e.zrByTouch; - } - - function allLeaveBlur(api) { - var model = api.getModel(); - var leaveBlurredSeries = []; - var allComponentViews = []; - model.eachComponent(function (componentType, componentModel) { - var componentStates = getComponentStates(componentModel); - var isSeries = componentType === 'series'; - var view = isSeries ? api.getViewOfSeriesModel(componentModel) : api.getViewOfComponentModel(componentModel); - !isSeries && allComponentViews.push(view); - - if (componentStates.isBlured) { - // Leave blur anyway - view.group.traverse(function (child) { - singleLeaveBlur(child); - }); - isSeries && leaveBlurredSeries.push(componentModel); - } - - componentStates.isBlured = false; - }); - each(allComponentViews, function (view) { - if (view && view.toggleBlurSeries) { - view.toggleBlurSeries(leaveBlurredSeries, false, model); - } - }); - } - function blurSeries(targetSeriesIndex, focus, blurScope, api) { - var ecModel = api.getModel(); - blurScope = blurScope || 'coordinateSystem'; - - function leaveBlurOfIndices(data, dataIndices) { - for (var i = 0; i < dataIndices.length; i++) { - var itemEl = data.getItemGraphicEl(dataIndices[i]); - itemEl && leaveBlur(itemEl); - } - } - - if (targetSeriesIndex == null) { - return; - } - - if (!focus || focus === 'none') { - return; - } - - var targetSeriesModel = ecModel.getSeriesByIndex(targetSeriesIndex); - var targetCoordSys = targetSeriesModel.coordinateSystem; - - if (targetCoordSys && targetCoordSys.master) { - targetCoordSys = targetCoordSys.master; - } - - var blurredSeries = []; - ecModel.eachSeries(function (seriesModel) { - var sameSeries = targetSeriesModel === seriesModel; - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && coordSys.master) { - coordSys = coordSys.master; - } - - var sameCoordSys = coordSys && targetCoordSys ? coordSys === targetCoordSys : sameSeries; // If there is no coordinate system. use sameSeries instead. - - if (!( // Not blur other series if blurScope series - blurScope === 'series' && !sameSeries // Not blur other coordinate system if blurScope is coordinateSystem - || blurScope === 'coordinateSystem' && !sameCoordSys // Not blur self series if focus is series. - || focus === 'series' && sameSeries // TODO blurScope: coordinate system - )) { - var view = api.getViewOfSeriesModel(seriesModel); - view.group.traverse(function (child) { - singleEnterBlur(child); - }); - - if (isArrayLike(focus)) { - leaveBlurOfIndices(seriesModel.getData(), focus); - } else if (isObject(focus)) { - var dataTypes = keys(focus); - - for (var d = 0; d < dataTypes.length; d++) { - leaveBlurOfIndices(seriesModel.getData(dataTypes[d]), focus[dataTypes[d]]); - } - } - - blurredSeries.push(seriesModel); - getComponentStates(seriesModel).isBlured = true; - } - }); - ecModel.eachComponent(function (componentType, componentModel) { - if (componentType === 'series') { - return; - } - - var view = api.getViewOfComponentModel(componentModel); - - if (view && view.toggleBlurSeries) { - view.toggleBlurSeries(blurredSeries, true, ecModel); - } - }); - } - function blurComponent(componentMainType, componentIndex, api) { - if (componentMainType == null || componentIndex == null) { - return; - } - - var componentModel = api.getModel().getComponent(componentMainType, componentIndex); - - if (!componentModel) { - return; - } - - getComponentStates(componentModel).isBlured = true; - var view = api.getViewOfComponentModel(componentModel); - - if (!view || !view.focusBlurEnabled) { - return; - } - - view.group.traverse(function (child) { - singleEnterBlur(child); - }); - } - function blurSeriesFromHighlightPayload(seriesModel, payload, api) { - var seriesIndex = seriesModel.seriesIndex; - var data = seriesModel.getData(payload.dataType); - - if (!data) { - if ("development" !== 'production') { - error("Unknown dataType " + payload.dataType); - } - - return; - } - - var dataIndex = queryDataIndex(data, payload); // Pick the first one if there is multiple/none exists. - - dataIndex = (isArray(dataIndex) ? dataIndex[0] : dataIndex) || 0; - var el = data.getItemGraphicEl(dataIndex); - - if (!el) { - var count = data.count(); - var current = 0; // If data on dataIndex is NaN. - - while (!el && current < count) { - el = data.getItemGraphicEl(current++); - } - } - - if (el) { - var ecData = getECData(el); - blurSeries(seriesIndex, ecData.focus, ecData.blurScope, api); - } else { - // If there is no element put on the data. Try getting it from raw option - // TODO Should put it on seriesModel? - var focus_1 = seriesModel.get(['emphasis', 'focus']); - var blurScope = seriesModel.get(['emphasis', 'blurScope']); - - if (focus_1 != null) { - blurSeries(seriesIndex, focus_1, blurScope, api); - } - } - } - function findComponentHighDownDispatchers(componentMainType, componentIndex, name, api) { - var ret = { - focusSelf: false, - dispatchers: null - }; - - if (componentMainType == null || componentMainType === 'series' || componentIndex == null || name == null) { - return ret; - } - - var componentModel = api.getModel().getComponent(componentMainType, componentIndex); - - if (!componentModel) { - return ret; - } - - var view = api.getViewOfComponentModel(componentModel); - - if (!view || !view.findHighDownDispatchers) { - return ret; - } - - var dispatchers = view.findHighDownDispatchers(name); // At presnet, the component (like Geo) only blur inside itself. - // So we do not use `blurScope` in component. - - var focusSelf; - - for (var i = 0; i < dispatchers.length; i++) { - if ("development" !== 'production' && !isHighDownDispatcher(dispatchers[i])) { - error('param should be highDownDispatcher'); - } - - if (getECData(dispatchers[i]).focus === 'self') { - focusSelf = true; - break; - } - } - - return { - focusSelf: focusSelf, - dispatchers: dispatchers - }; - } - function handleGlobalMouseOverForHighDown(dispatcher, e, api) { - if ("development" !== 'production' && !isHighDownDispatcher(dispatcher)) { - error('param should be highDownDispatcher'); - } - - var ecData = getECData(dispatcher); - - var _a = findComponentHighDownDispatchers(ecData.componentMainType, ecData.componentIndex, ecData.componentHighDownName, api), - dispatchers = _a.dispatchers, - focusSelf = _a.focusSelf; // If `findHighDownDispatchers` is supported on the component, - // highlight/downplay elements with the same name. - - - if (dispatchers) { - if (focusSelf) { - blurComponent(ecData.componentMainType, ecData.componentIndex, api); - } - - each(dispatchers, function (dispatcher) { - return enterEmphasisWhenMouseOver(dispatcher, e); - }); - } else { - // Try blur all in the related series. Then emphasis the hoverred. - // TODO. progressive mode. - blurSeries(ecData.seriesIndex, ecData.focus, ecData.blurScope, api); - - if (ecData.focus === 'self') { - blurComponent(ecData.componentMainType, ecData.componentIndex, api); - } // Other than series, component that not support `findHighDownDispatcher` will - // also use it. But in this case, highlight/downplay are only supported in - // mouse hover but not in dispatchAction. - - - enterEmphasisWhenMouseOver(dispatcher, e); - } - } - function handleGlobalMouseOutForHighDown(dispatcher, e, api) { - if ("development" !== 'production' && !isHighDownDispatcher(dispatcher)) { - error('param should be highDownDispatcher'); - } - - allLeaveBlur(api); - var ecData = getECData(dispatcher); - var dispatchers = findComponentHighDownDispatchers(ecData.componentMainType, ecData.componentIndex, ecData.componentHighDownName, api).dispatchers; - - if (dispatchers) { - each(dispatchers, function (dispatcher) { - return leaveEmphasisWhenMouseOut(dispatcher, e); - }); - } else { - leaveEmphasisWhenMouseOut(dispatcher, e); - } - } - function toggleSelectionFromPayload(seriesModel, payload, api) { - if (!isSelectChangePayload(payload)) { - return; - } - - var dataType = payload.dataType; - var data = seriesModel.getData(dataType); - var dataIndex = queryDataIndex(data, payload); - - if (!isArray(dataIndex)) { - dataIndex = [dataIndex]; - } - - seriesModel[payload.type === TOGGLE_SELECT_ACTION_TYPE ? 'toggleSelect' : payload.type === SELECT_ACTION_TYPE ? 'select' : 'unselect'](dataIndex, dataType); - } - function updateSeriesElementSelection(seriesModel) { - var allData = seriesModel.getAllData(); - each(allData, function (_a) { - var data = _a.data, - type = _a.type; - data.eachItemGraphicEl(function (el, idx) { - seriesModel.isSelected(idx, type) ? enterSelect(el) : leaveSelect(el); - }); - }); - } - function getAllSelectedIndices(ecModel) { - var ret = []; - ecModel.eachSeries(function (seriesModel) { - var allData = seriesModel.getAllData(); - each(allData, function (_a) { - var data = _a.data, - type = _a.type; - var dataIndices = seriesModel.getSelectedDataIndices(); - - if (dataIndices.length > 0) { - var item = { - dataIndex: dataIndices, - seriesIndex: seriesModel.seriesIndex - }; - - if (type != null) { - item.dataType = type; - } - - ret.push(item); - } - }); - }); - return ret; - } - /** - * Enable the function that mouseover will trigger the emphasis state. - * - * NOTE: - * This function should be used on the element with dataIndex, seriesIndex. - * - */ - - function enableHoverEmphasis(el, focus, blurScope) { - setAsHighDownDispatcher(el, true); - traverseUpdateState(el, setDefaultStateProxy); - enableHoverFocus(el, focus, blurScope); - } - function disableHoverEmphasis(el) { - setAsHighDownDispatcher(el, false); - } - function toggleHoverEmphasis(el, focus, blurScope, isDisabled) { - isDisabled ? disableHoverEmphasis(el) : enableHoverEmphasis(el, focus, blurScope); - } - function enableHoverFocus(el, focus, blurScope) { - var ecData = getECData(el); - - if (focus != null) { - // TODO dataIndex may be set after this function. This check is not useful. - // if (ecData.dataIndex == null) { - // if (__DEV__) { - // console.warn('focus can only been set on element with dataIndex'); - // } - // } - // else { - ecData.focus = focus; - ecData.blurScope = blurScope; // } - } else if (ecData.focus) { - ecData.focus = null; - } - } - var OTHER_STATES = ['emphasis', 'blur', 'select']; - var defaultStyleGetterMap = { - itemStyle: 'getItemStyle', - lineStyle: 'getLineStyle', - areaStyle: 'getAreaStyle' - }; - /** - * Set emphasis/blur/selected states of element. - */ - - function setStatesStylesFromModel(el, itemModel, styleType, // default itemStyle - getter) { - styleType = styleType || 'itemStyle'; - - for (var i = 0; i < OTHER_STATES.length; i++) { - var stateName = OTHER_STATES[i]; - var model = itemModel.getModel([stateName, styleType]); - var state = el.ensureState(stateName); // Let it throw error if getterType is not found. - - state.style = getter ? getter(model) : model[defaultStyleGetterMap[styleType]](); - } - } - /** - * - * Set element as highlight / downplay dispatcher. - * It will be checked when element received mouseover event or from highlight action. - * It's in change of all highlight/downplay behavior of it's children. - * - * @param el - * @param el.highDownSilentOnTouch - * In touch device, mouseover event will be trigger on touchstart event - * (see module:zrender/dom/HandlerProxy). By this mechanism, we can - * conveniently use hoverStyle when tap on touch screen without additional - * code for compatibility. - * But if the chart/component has select feature, which usually also use - * hoverStyle, there might be conflict between 'select-highlight' and - * 'hover-highlight' especially when roam is enabled (see geo for example). - * In this case, `highDownSilentOnTouch` should be used to disable - * hover-highlight on touch device. - * @param asDispatcher If `false`, do not set as "highDownDispatcher". - */ - - function setAsHighDownDispatcher(el, asDispatcher) { - var disable = asDispatcher === false; - var extendedEl = el; // Make `highDownSilentOnTouch` and `onStateChange` only work after - // `setAsHighDownDispatcher` called. Avoid it is modified by user unexpectedly. - - if (el.highDownSilentOnTouch) { - extendedEl.__highDownSilentOnTouch = el.highDownSilentOnTouch; - } // Simple optimize, since this method might be - // called for each elements of a group in some cases. - - - if (!disable || extendedEl.__highDownDispatcher) { - // Emphasis, normal can be triggered manually by API or other components like hover link. - // el[method]('emphasis', onElementEmphasisEvent)[method]('normal', onElementNormalEvent); - // Also keep previous record. - extendedEl.__highByOuter = extendedEl.__highByOuter || 0; - extendedEl.__highDownDispatcher = !disable; - } - } - function isHighDownDispatcher(el) { - return !!(el && el.__highDownDispatcher); - } - /** - * Enable component highlight/downplay features: - * + hover link (within the same name) - * + focus blur in component - */ - - function enableComponentHighDownFeatures(el, componentModel, componentHighDownName) { - var ecData = getECData(el); - ecData.componentMainType = componentModel.mainType; - ecData.componentIndex = componentModel.componentIndex; - ecData.componentHighDownName = componentHighDownName; - } - /** - * Support highlight/downplay record on each elements. - * For the case: hover highlight/downplay (legend, visualMap, ...) and - * user triggered highlight/downplay should not conflict. - * Only all of the highlightDigit cleared, return to normal. - * @param {string} highlightKey - * @return {number} highlightDigit - */ - - function getHighlightDigit(highlightKey) { - var highlightDigit = _highlightKeyMap[highlightKey]; - - if (highlightDigit == null && _highlightNextDigit <= 32) { - highlightDigit = _highlightKeyMap[highlightKey] = _highlightNextDigit++; - } - - return highlightDigit; - } - function isSelectChangePayload(payload) { - var payloadType = payload.type; - return payloadType === SELECT_ACTION_TYPE || payloadType === UNSELECT_ACTION_TYPE || payloadType === TOGGLE_SELECT_ACTION_TYPE; - } - function isHighDownPayload(payload) { - var payloadType = payload.type; - return payloadType === HIGHLIGHT_ACTION_TYPE || payloadType === DOWNPLAY_ACTION_TYPE; - } - function savePathStates(el) { - var store = getSavedStates(el); - store.normalFill = el.style.fill; - store.normalStroke = el.style.stroke; - var selectState = el.states.select || {}; - store.selectFill = selectState.style && selectState.style.fill || null; - store.selectStroke = selectState.style && selectState.style.stroke || null; - } - - var CMD$2 = PathProxy.CMD; - var points = [[], [], []]; - var mathSqrt$1 = Math.sqrt; - var mathAtan2 = Math.atan2; - function transformPath(path, m) { - if (!m) { - return; - } - var data = path.data; - var len = path.len(); - var cmd; - var nPoint; - var i; - var j; - var k; - var p; - var M = CMD$2.M; - var C = CMD$2.C; - var L = CMD$2.L; - var R = CMD$2.R; - var A = CMD$2.A; - var Q = CMD$2.Q; - for (i = 0, j = 0; i < len;) { - cmd = data[i++]; - j = i; - nPoint = 0; - switch (cmd) { - case M: - nPoint = 1; - break; - case L: - nPoint = 1; - break; - case C: - nPoint = 3; - break; - case Q: - nPoint = 2; - break; - case A: - var x = m[4]; - var y = m[5]; - var sx = mathSqrt$1(m[0] * m[0] + m[1] * m[1]); - var sy = mathSqrt$1(m[2] * m[2] + m[3] * m[3]); - var angle = mathAtan2(-m[1] / sy, m[0] / sx); - data[i] *= sx; - data[i++] += x; - data[i] *= sy; - data[i++] += y; - data[i++] *= sx; - data[i++] *= sy; - data[i++] += angle; - data[i++] += angle; - i += 2; - j = i; - break; - case R: - p[0] = data[i++]; - p[1] = data[i++]; - applyTransform(p, p, m); - data[j++] = p[0]; - data[j++] = p[1]; - p[0] += data[i++]; - p[1] += data[i++]; - applyTransform(p, p, m); - data[j++] = p[0]; - data[j++] = p[1]; - } - for (k = 0; k < nPoint; k++) { - var p_1 = points[k]; - p_1[0] = data[i++]; - p_1[1] = data[i++]; - applyTransform(p_1, p_1, m); - data[j++] = p_1[0]; - data[j++] = p_1[1]; - } - } - path.increaseVersion(); - } - - var mathSqrt$2 = Math.sqrt; - var mathSin$2 = Math.sin; - var mathCos$2 = Math.cos; - var PI$1 = Math.PI; - function vMag(v) { - return Math.sqrt(v[0] * v[0] + v[1] * v[1]); - } - function vRatio(u, v) { - return (u[0] * v[0] + u[1] * v[1]) / (vMag(u) * vMag(v)); - } - function vAngle(u, v) { - return (u[0] * v[1] < u[1] * v[0] ? -1 : 1) - * Math.acos(vRatio(u, v)); - } - function processArc(x1, y1, x2, y2, fa, fs, rx, ry, psiDeg, cmd, path) { - var psi = psiDeg * (PI$1 / 180.0); - var xp = mathCos$2(psi) * (x1 - x2) / 2.0 - + mathSin$2(psi) * (y1 - y2) / 2.0; - var yp = -1 * mathSin$2(psi) * (x1 - x2) / 2.0 - + mathCos$2(psi) * (y1 - y2) / 2.0; - var lambda = (xp * xp) / (rx * rx) + (yp * yp) / (ry * ry); - if (lambda > 1) { - rx *= mathSqrt$2(lambda); - ry *= mathSqrt$2(lambda); - } - var f = (fa === fs ? -1 : 1) - * mathSqrt$2((((rx * rx) * (ry * ry)) - - ((rx * rx) * (yp * yp)) - - ((ry * ry) * (xp * xp))) / ((rx * rx) * (yp * yp) - + (ry * ry) * (xp * xp))) || 0; - var cxp = f * rx * yp / ry; - var cyp = f * -ry * xp / rx; - var cx = (x1 + x2) / 2.0 - + mathCos$2(psi) * cxp - - mathSin$2(psi) * cyp; - var cy = (y1 + y2) / 2.0 - + mathSin$2(psi) * cxp - + mathCos$2(psi) * cyp; - var theta = vAngle([1, 0], [(xp - cxp) / rx, (yp - cyp) / ry]); - var u = [(xp - cxp) / rx, (yp - cyp) / ry]; - var v = [(-1 * xp - cxp) / rx, (-1 * yp - cyp) / ry]; - var dTheta = vAngle(u, v); - if (vRatio(u, v) <= -1) { - dTheta = PI$1; - } - if (vRatio(u, v) >= 1) { - dTheta = 0; - } - if (dTheta < 0) { - var n = Math.round(dTheta / PI$1 * 1e6) / 1e6; - dTheta = PI$1 * 2 + (n % 2) * PI$1; - } - path.addData(cmd, cx, cy, rx, ry, theta, dTheta, psi, fs); - } - var commandReg = /([mlvhzcqtsa])([^mlvhzcqtsa]*)/ig; - var numberReg = /-?([0-9]*\.)?[0-9]+([eE]-?[0-9]+)?/g; - function createPathProxyFromString(data) { - var path = new PathProxy(); - if (!data) { - return path; - } - var cpx = 0; - var cpy = 0; - var subpathX = cpx; - var subpathY = cpy; - var prevCmd; - var CMD = PathProxy.CMD; - var cmdList = data.match(commandReg); - if (!cmdList) { - return path; - } - for (var l = 0; l < cmdList.length; l++) { - var cmdText = cmdList[l]; - var cmdStr = cmdText.charAt(0); - var cmd = void 0; - var p = cmdText.match(numberReg) || []; - var pLen = p.length; - for (var i = 0; i < pLen; i++) { - p[i] = parseFloat(p[i]); - } - var off = 0; - while (off < pLen) { - var ctlPtx = void 0; - var ctlPty = void 0; - var rx = void 0; - var ry = void 0; - var psi = void 0; - var fa = void 0; - var fs = void 0; - var x1 = cpx; - var y1 = cpy; - var len = void 0; - var pathData = void 0; - switch (cmdStr) { - case 'l': - cpx += p[off++]; - cpy += p[off++]; - cmd = CMD.L; - path.addData(cmd, cpx, cpy); - break; - case 'L': - cpx = p[off++]; - cpy = p[off++]; - cmd = CMD.L; - path.addData(cmd, cpx, cpy); - break; - case 'm': - cpx += p[off++]; - cpy += p[off++]; - cmd = CMD.M; - path.addData(cmd, cpx, cpy); - subpathX = cpx; - subpathY = cpy; - cmdStr = 'l'; - break; - case 'M': - cpx = p[off++]; - cpy = p[off++]; - cmd = CMD.M; - path.addData(cmd, cpx, cpy); - subpathX = cpx; - subpathY = cpy; - cmdStr = 'L'; - break; - case 'h': - cpx += p[off++]; - cmd = CMD.L; - path.addData(cmd, cpx, cpy); - break; - case 'H': - cpx = p[off++]; - cmd = CMD.L; - path.addData(cmd, cpx, cpy); - break; - case 'v': - cpy += p[off++]; - cmd = CMD.L; - path.addData(cmd, cpx, cpy); - break; - case 'V': - cpy = p[off++]; - cmd = CMD.L; - path.addData(cmd, cpx, cpy); - break; - case 'C': - cmd = CMD.C; - path.addData(cmd, p[off++], p[off++], p[off++], p[off++], p[off++], p[off++]); - cpx = p[off - 2]; - cpy = p[off - 1]; - break; - case 'c': - cmd = CMD.C; - path.addData(cmd, p[off++] + cpx, p[off++] + cpy, p[off++] + cpx, p[off++] + cpy, p[off++] + cpx, p[off++] + cpy); - cpx += p[off - 2]; - cpy += p[off - 1]; - break; - case 'S': - ctlPtx = cpx; - ctlPty = cpy; - len = path.len(); - pathData = path.data; - if (prevCmd === CMD.C) { - ctlPtx += cpx - pathData[len - 4]; - ctlPty += cpy - pathData[len - 3]; - } - cmd = CMD.C; - x1 = p[off++]; - y1 = p[off++]; - cpx = p[off++]; - cpy = p[off++]; - path.addData(cmd, ctlPtx, ctlPty, x1, y1, cpx, cpy); - break; - case 's': - ctlPtx = cpx; - ctlPty = cpy; - len = path.len(); - pathData = path.data; - if (prevCmd === CMD.C) { - ctlPtx += cpx - pathData[len - 4]; - ctlPty += cpy - pathData[len - 3]; - } - cmd = CMD.C; - x1 = cpx + p[off++]; - y1 = cpy + p[off++]; - cpx += p[off++]; - cpy += p[off++]; - path.addData(cmd, ctlPtx, ctlPty, x1, y1, cpx, cpy); - break; - case 'Q': - x1 = p[off++]; - y1 = p[off++]; - cpx = p[off++]; - cpy = p[off++]; - cmd = CMD.Q; - path.addData(cmd, x1, y1, cpx, cpy); - break; - case 'q': - x1 = p[off++] + cpx; - y1 = p[off++] + cpy; - cpx += p[off++]; - cpy += p[off++]; - cmd = CMD.Q; - path.addData(cmd, x1, y1, cpx, cpy); - break; - case 'T': - ctlPtx = cpx; - ctlPty = cpy; - len = path.len(); - pathData = path.data; - if (prevCmd === CMD.Q) { - ctlPtx += cpx - pathData[len - 4]; - ctlPty += cpy - pathData[len - 3]; - } - cpx = p[off++]; - cpy = p[off++]; - cmd = CMD.Q; - path.addData(cmd, ctlPtx, ctlPty, cpx, cpy); - break; - case 't': - ctlPtx = cpx; - ctlPty = cpy; - len = path.len(); - pathData = path.data; - if (prevCmd === CMD.Q) { - ctlPtx += cpx - pathData[len - 4]; - ctlPty += cpy - pathData[len - 3]; - } - cpx += p[off++]; - cpy += p[off++]; - cmd = CMD.Q; - path.addData(cmd, ctlPtx, ctlPty, cpx, cpy); - break; - case 'A': - rx = p[off++]; - ry = p[off++]; - psi = p[off++]; - fa = p[off++]; - fs = p[off++]; - x1 = cpx, y1 = cpy; - cpx = p[off++]; - cpy = p[off++]; - cmd = CMD.A; - processArc(x1, y1, cpx, cpy, fa, fs, rx, ry, psi, cmd, path); - break; - case 'a': - rx = p[off++]; - ry = p[off++]; - psi = p[off++]; - fa = p[off++]; - fs = p[off++]; - x1 = cpx, y1 = cpy; - cpx += p[off++]; - cpy += p[off++]; - cmd = CMD.A; - processArc(x1, y1, cpx, cpy, fa, fs, rx, ry, psi, cmd, path); - break; - } - } - if (cmdStr === 'z' || cmdStr === 'Z') { - cmd = CMD.Z; - path.addData(cmd); - cpx = subpathX; - cpy = subpathY; - } - prevCmd = cmd; - } - path.toStatic(); - return path; - } - var SVGPath = (function (_super) { - __extends(SVGPath, _super); - function SVGPath() { - return _super !== null && _super.apply(this, arguments) || this; - } - SVGPath.prototype.applyTransform = function (m) { }; - return SVGPath; - }(Path)); - function isPathProxy(path) { - return path.setData != null; - } - function createPathOptions(str, opts) { - var pathProxy = createPathProxyFromString(str); - var innerOpts = extend({}, opts); - innerOpts.buildPath = function (path) { - if (isPathProxy(path)) { - path.setData(pathProxy.data); - var ctx = path.getContext(); - if (ctx) { - path.rebuildPath(ctx, 1); - } - } - else { - var ctx = path; - pathProxy.rebuildPath(ctx, 1); - } - }; - innerOpts.applyTransform = function (m) { - transformPath(pathProxy, m); - this.dirtyShape(); - }; - return innerOpts; - } - function createFromString(str, opts) { - return new SVGPath(createPathOptions(str, opts)); - } - function extendFromString(str, defaultOpts) { - var innerOpts = createPathOptions(str, defaultOpts); - var Sub = (function (_super) { - __extends(Sub, _super); - function Sub(opts) { - var _this = _super.call(this, opts) || this; - _this.applyTransform = innerOpts.applyTransform; - _this.buildPath = innerOpts.buildPath; - return _this; - } - return Sub; - }(SVGPath)); - return Sub; - } - function mergePath(pathEls, opts) { - var pathList = []; - var len = pathEls.length; - for (var i = 0; i < len; i++) { - var pathEl = pathEls[i]; - pathList.push(pathEl.getUpdatedPathProxy(true)); - } - var pathBundle = new Path(opts); - pathBundle.createPathProxy(); - pathBundle.buildPath = function (path) { - if (isPathProxy(path)) { - path.appendPath(pathList); - var ctx = path.getContext(); - if (ctx) { - path.rebuildPath(ctx, 1); - } - } - }; - return pathBundle; - } - function clonePath(sourcePath, opts) { - opts = opts || {}; - var path = new Path(); - if (sourcePath.shape) { - path.setShape(sourcePath.shape); - } - path.setStyle(sourcePath.style); - if (opts.bakeTransform) { - transformPath(path.path, sourcePath.getComputedTransform()); - } - else { - if (opts.toLocal) { - path.setLocalTransform(sourcePath.getComputedTransform()); - } - else { - path.copyTransform(sourcePath); - } - } - path.buildPath = sourcePath.buildPath; - path.applyTransform = path.applyTransform; - path.z = sourcePath.z; - path.z2 = sourcePath.z2; - path.zlevel = sourcePath.zlevel; - return path; - } - - var CircleShape = (function () { - function CircleShape() { - this.cx = 0; - this.cy = 0; - this.r = 0; - } - return CircleShape; - }()); - var Circle = (function (_super) { - __extends(Circle, _super); - function Circle(opts) { - return _super.call(this, opts) || this; - } - Circle.prototype.getDefaultShape = function () { - return new CircleShape(); - }; - Circle.prototype.buildPath = function (ctx, shape) { - ctx.moveTo(shape.cx + shape.r, shape.cy); - ctx.arc(shape.cx, shape.cy, shape.r, 0, Math.PI * 2); - }; - return Circle; - }(Path)); - Circle.prototype.type = 'circle'; - - var EllipseShape = (function () { - function EllipseShape() { - this.cx = 0; - this.cy = 0; - this.rx = 0; - this.ry = 0; - } - return EllipseShape; - }()); - var Ellipse = (function (_super) { - __extends(Ellipse, _super); - function Ellipse(opts) { - return _super.call(this, opts) || this; - } - Ellipse.prototype.getDefaultShape = function () { - return new EllipseShape(); - }; - Ellipse.prototype.buildPath = function (ctx, shape) { - var k = 0.5522848; - var x = shape.cx; - var y = shape.cy; - var a = shape.rx; - var b = shape.ry; - var ox = a * k; - var oy = b * k; - ctx.moveTo(x - a, y); - ctx.bezierCurveTo(x - a, y - oy, x - ox, y - b, x, y - b); - ctx.bezierCurveTo(x + ox, y - b, x + a, y - oy, x + a, y); - ctx.bezierCurveTo(x + a, y + oy, x + ox, y + b, x, y + b); - ctx.bezierCurveTo(x - ox, y + b, x - a, y + oy, x - a, y); - ctx.closePath(); - }; - return Ellipse; - }(Path)); - Ellipse.prototype.type = 'ellipse'; - - var PI$2 = Math.PI; - var PI2$5 = PI$2 * 2; - var mathSin$3 = Math.sin; - var mathCos$3 = Math.cos; - var mathACos = Math.acos; - var mathATan2 = Math.atan2; - var mathAbs$1 = Math.abs; - var mathSqrt$3 = Math.sqrt; - var mathMax$3 = Math.max; - var mathMin$3 = Math.min; - var e = 1e-4; - function intersect(x0, y0, x1, y1, x2, y2, x3, y3) { - var dx10 = x1 - x0; - var dy10 = y1 - y0; - var dx32 = x3 - x2; - var dy32 = y3 - y2; - var t = dy32 * dx10 - dx32 * dy10; - if (t * t < e) { - return; - } - t = (dx32 * (y0 - y2) - dy32 * (x0 - x2)) / t; - return [x0 + t * dx10, y0 + t * dy10]; - } - function computeCornerTangents(x0, y0, x1, y1, radius, cr, clockwise) { - var x01 = x0 - x1; - var y01 = y0 - y1; - var lo = (clockwise ? cr : -cr) / mathSqrt$3(x01 * x01 + y01 * y01); - var ox = lo * y01; - var oy = -lo * x01; - var x11 = x0 + ox; - var y11 = y0 + oy; - var x10 = x1 + ox; - var y10 = y1 + oy; - var x00 = (x11 + x10) / 2; - var y00 = (y11 + y10) / 2; - var dx = x10 - x11; - var dy = y10 - y11; - var d2 = dx * dx + dy * dy; - var r = radius - cr; - var s = x11 * y10 - x10 * y11; - var d = (dy < 0 ? -1 : 1) * mathSqrt$3(mathMax$3(0, r * r * d2 - s * s)); - var cx0 = (s * dy - dx * d) / d2; - var cy0 = (-s * dx - dy * d) / d2; - var cx1 = (s * dy + dx * d) / d2; - var cy1 = (-s * dx + dy * d) / d2; - var dx0 = cx0 - x00; - var dy0 = cy0 - y00; - var dx1 = cx1 - x00; - var dy1 = cy1 - y00; - if (dx0 * dx0 + dy0 * dy0 > dx1 * dx1 + dy1 * dy1) { - cx0 = cx1; - cy0 = cy1; - } - return { - cx: cx0, - cy: cy0, - x0: -ox, - y0: -oy, - x1: cx0 * (radius / r - 1), - y1: cy0 * (radius / r - 1) - }; - } - function normalizeCornerRadius(cr) { - var arr; - if (isArray(cr)) { - var len = cr.length; - if (!len) { - return cr; - } - if (len === 1) { - arr = [cr[0], cr[0], 0, 0]; - } - else if (len === 2) { - arr = [cr[0], cr[0], cr[1], cr[1]]; - } - else if (len === 3) { - arr = cr.concat(cr[2]); - } - else { - arr = cr; - } - } - else { - arr = [cr, cr, cr, cr]; - } - return arr; - } - function buildPath$1(ctx, shape) { - var _a; - var radius = mathMax$3(shape.r, 0); - var innerRadius = mathMax$3(shape.r0 || 0, 0); - var hasRadius = radius > 0; - var hasInnerRadius = innerRadius > 0; - if (!hasRadius && !hasInnerRadius) { - return; - } - if (!hasRadius) { - radius = innerRadius; - innerRadius = 0; - } - if (innerRadius > radius) { - var tmp = radius; - radius = innerRadius; - innerRadius = tmp; - } - var startAngle = shape.startAngle, endAngle = shape.endAngle; - if (isNaN(startAngle) || isNaN(endAngle)) { - return; - } - var cx = shape.cx, cy = shape.cy; - var clockwise = !!shape.clockwise; - var arc = mathAbs$1(endAngle - startAngle); - var mod = arc > PI2$5 && arc % PI2$5; - mod > e && (arc = mod); - if (!(radius > e)) { - ctx.moveTo(cx, cy); - } - else if (arc > PI2$5 - e) { - ctx.moveTo(cx + radius * mathCos$3(startAngle), cy + radius * mathSin$3(startAngle)); - ctx.arc(cx, cy, radius, startAngle, endAngle, !clockwise); - if (innerRadius > e) { - ctx.moveTo(cx + innerRadius * mathCos$3(endAngle), cy + innerRadius * mathSin$3(endAngle)); - ctx.arc(cx, cy, innerRadius, endAngle, startAngle, clockwise); - } - } - else { - var icrStart = void 0; - var icrEnd = void 0; - var ocrStart = void 0; - var ocrEnd = void 0; - var ocrs = void 0; - var ocre = void 0; - var icrs = void 0; - var icre = void 0; - var ocrMax = void 0; - var icrMax = void 0; - var limitedOcrMax = void 0; - var limitedIcrMax = void 0; - var xre = void 0; - var yre = void 0; - var xirs = void 0; - var yirs = void 0; - var xrs = radius * mathCos$3(startAngle); - var yrs = radius * mathSin$3(startAngle); - var xire = innerRadius * mathCos$3(endAngle); - var yire = innerRadius * mathSin$3(endAngle); - var hasArc = arc > e; - if (hasArc) { - var cornerRadius = shape.cornerRadius; - if (cornerRadius) { - _a = normalizeCornerRadius(cornerRadius), icrStart = _a[0], icrEnd = _a[1], ocrStart = _a[2], ocrEnd = _a[3]; - } - var halfRd = mathAbs$1(radius - innerRadius) / 2; - ocrs = mathMin$3(halfRd, ocrStart); - ocre = mathMin$3(halfRd, ocrEnd); - icrs = mathMin$3(halfRd, icrStart); - icre = mathMin$3(halfRd, icrEnd); - limitedOcrMax = ocrMax = mathMax$3(ocrs, ocre); - limitedIcrMax = icrMax = mathMax$3(icrs, icre); - if (ocrMax > e || icrMax > e) { - xre = radius * mathCos$3(endAngle); - yre = radius * mathSin$3(endAngle); - xirs = innerRadius * mathCos$3(startAngle); - yirs = innerRadius * mathSin$3(startAngle); - if (arc < PI$2) { - var it_1 = intersect(xrs, yrs, xirs, yirs, xre, yre, xire, yire); - if (it_1) { - var x0 = xrs - it_1[0]; - var y0 = yrs - it_1[1]; - var x1 = xre - it_1[0]; - var y1 = yre - it_1[1]; - var a = 1 / mathSin$3(mathACos((x0 * x1 + y0 * y1) / (mathSqrt$3(x0 * x0 + y0 * y0) * mathSqrt$3(x1 * x1 + y1 * y1))) / 2); - var b = mathSqrt$3(it_1[0] * it_1[0] + it_1[1] * it_1[1]); - limitedOcrMax = mathMin$3(ocrMax, (radius - b) / (a + 1)); - limitedIcrMax = mathMin$3(icrMax, (innerRadius - b) / (a - 1)); - } - } - } - } - if (!hasArc) { - ctx.moveTo(cx + xrs, cy + yrs); - } - else if (limitedOcrMax > e) { - var crStart = mathMin$3(ocrStart, limitedOcrMax); - var crEnd = mathMin$3(ocrEnd, limitedOcrMax); - var ct0 = computeCornerTangents(xirs, yirs, xrs, yrs, radius, crStart, clockwise); - var ct1 = computeCornerTangents(xre, yre, xire, yire, radius, crEnd, clockwise); - ctx.moveTo(cx + ct0.cx + ct0.x0, cy + ct0.cy + ct0.y0); - if (limitedOcrMax < ocrMax && crStart === crEnd) { - ctx.arc(cx + ct0.cx, cy + ct0.cy, limitedOcrMax, mathATan2(ct0.y0, ct0.x0), mathATan2(ct1.y0, ct1.x0), !clockwise); - } - else { - crStart > 0 && ctx.arc(cx + ct0.cx, cy + ct0.cy, crStart, mathATan2(ct0.y0, ct0.x0), mathATan2(ct0.y1, ct0.x1), !clockwise); - ctx.arc(cx, cy, radius, mathATan2(ct0.cy + ct0.y1, ct0.cx + ct0.x1), mathATan2(ct1.cy + ct1.y1, ct1.cx + ct1.x1), !clockwise); - crEnd > 0 && ctx.arc(cx + ct1.cx, cy + ct1.cy, crEnd, mathATan2(ct1.y1, ct1.x1), mathATan2(ct1.y0, ct1.x0), !clockwise); - } - } - else { - ctx.moveTo(cx + xrs, cy + yrs); - ctx.arc(cx, cy, radius, startAngle, endAngle, !clockwise); - } - if (!(innerRadius > e) || !hasArc) { - ctx.lineTo(cx + xire, cy + yire); - } - else if (limitedIcrMax > e) { - var crStart = mathMin$3(icrStart, limitedIcrMax); - var crEnd = mathMin$3(icrEnd, limitedIcrMax); - var ct0 = computeCornerTangents(xire, yire, xre, yre, innerRadius, -crEnd, clockwise); - var ct1 = computeCornerTangents(xrs, yrs, xirs, yirs, innerRadius, -crStart, clockwise); - ctx.lineTo(cx + ct0.cx + ct0.x0, cy + ct0.cy + ct0.y0); - if (limitedIcrMax < icrMax && crStart === crEnd) { - ctx.arc(cx + ct0.cx, cy + ct0.cy, limitedIcrMax, mathATan2(ct0.y0, ct0.x0), mathATan2(ct1.y0, ct1.x0), !clockwise); - } - else { - crEnd > 0 && ctx.arc(cx + ct0.cx, cy + ct0.cy, crEnd, mathATan2(ct0.y0, ct0.x0), mathATan2(ct0.y1, ct0.x1), !clockwise); - ctx.arc(cx, cy, innerRadius, mathATan2(ct0.cy + ct0.y1, ct0.cx + ct0.x1), mathATan2(ct1.cy + ct1.y1, ct1.cx + ct1.x1), clockwise); - crStart > 0 && ctx.arc(cx + ct1.cx, cy + ct1.cy, crStart, mathATan2(ct1.y1, ct1.x1), mathATan2(ct1.y0, ct1.x0), !clockwise); - } - } - else { - ctx.lineTo(cx + xire, cy + yire); - ctx.arc(cx, cy, innerRadius, endAngle, startAngle, clockwise); - } - } - ctx.closePath(); - } - - var SectorShape = (function () { - function SectorShape() { - this.cx = 0; - this.cy = 0; - this.r0 = 0; - this.r = 0; - this.startAngle = 0; - this.endAngle = Math.PI * 2; - this.clockwise = true; - this.cornerRadius = 0; - } - return SectorShape; - }()); - var Sector = (function (_super) { - __extends(Sector, _super); - function Sector(opts) { - return _super.call(this, opts) || this; - } - Sector.prototype.getDefaultShape = function () { - return new SectorShape(); - }; - Sector.prototype.buildPath = function (ctx, shape) { - buildPath$1(ctx, shape); - }; - Sector.prototype.isZeroArea = function () { - return this.shape.startAngle === this.shape.endAngle - || this.shape.r === this.shape.r0; - }; - return Sector; - }(Path)); - Sector.prototype.type = 'sector'; - - var RingShape = (function () { - function RingShape() { - this.cx = 0; - this.cy = 0; - this.r = 0; - this.r0 = 0; - } - return RingShape; - }()); - var Ring = (function (_super) { - __extends(Ring, _super); - function Ring(opts) { - return _super.call(this, opts) || this; - } - Ring.prototype.getDefaultShape = function () { - return new RingShape(); - }; - Ring.prototype.buildPath = function (ctx, shape) { - var x = shape.cx; - var y = shape.cy; - var PI2 = Math.PI * 2; - ctx.moveTo(x + shape.r, y); - ctx.arc(x, y, shape.r, 0, PI2, false); - ctx.moveTo(x + shape.r0, y); - ctx.arc(x, y, shape.r0, 0, PI2, true); - }; - return Ring; - }(Path)); - Ring.prototype.type = 'ring'; - - function smoothBezier(points, smooth, isLoop, constraint) { - var cps = []; - var v = []; - var v1 = []; - var v2 = []; - var prevPoint; - var nextPoint; - var min$1; - var max$1; - if (constraint) { - min$1 = [Infinity, Infinity]; - max$1 = [-Infinity, -Infinity]; - for (var i = 0, len = points.length; i < len; i++) { - min(min$1, min$1, points[i]); - max(max$1, max$1, points[i]); - } - min(min$1, min$1, constraint[0]); - max(max$1, max$1, constraint[1]); - } - for (var i = 0, len = points.length; i < len; i++) { - var point = points[i]; - if (isLoop) { - prevPoint = points[i ? i - 1 : len - 1]; - nextPoint = points[(i + 1) % len]; - } - else { - if (i === 0 || i === len - 1) { - cps.push(clone$1(points[i])); - continue; - } - else { - prevPoint = points[i - 1]; - nextPoint = points[i + 1]; - } - } - sub(v, nextPoint, prevPoint); - scale(v, v, smooth); - var d0 = distance(point, prevPoint); - var d1 = distance(point, nextPoint); - var sum = d0 + d1; - if (sum !== 0) { - d0 /= sum; - d1 /= sum; - } - scale(v1, v, -d0); - scale(v2, v, d1); - var cp0 = add([], point, v1); - var cp1 = add([], point, v2); - if (constraint) { - max(cp0, cp0, min$1); - min(cp0, cp0, max$1); - max(cp1, cp1, min$1); - min(cp1, cp1, max$1); - } - cps.push(cp0); - cps.push(cp1); - } - if (isLoop) { - cps.push(cps.shift()); - } - return cps; - } - - function buildPath$2(ctx, shape, closePath) { - var smooth = shape.smooth; - var points = shape.points; - if (points && points.length >= 2) { - if (smooth) { - var controlPoints = smoothBezier(points, smooth, closePath, shape.smoothConstraint); - ctx.moveTo(points[0][0], points[0][1]); - var len = points.length; - for (var i = 0; i < (closePath ? len : len - 1); i++) { - var cp1 = controlPoints[i * 2]; - var cp2 = controlPoints[i * 2 + 1]; - var p = points[(i + 1) % len]; - ctx.bezierCurveTo(cp1[0], cp1[1], cp2[0], cp2[1], p[0], p[1]); - } - } - else { - ctx.moveTo(points[0][0], points[0][1]); - for (var i = 1, l = points.length; i < l; i++) { - ctx.lineTo(points[i][0], points[i][1]); - } - } - closePath && ctx.closePath(); - } - } - - var PolygonShape = (function () { - function PolygonShape() { - this.points = null; - this.smooth = 0; - this.smoothConstraint = null; - } - return PolygonShape; - }()); - var Polygon = (function (_super) { - __extends(Polygon, _super); - function Polygon(opts) { - return _super.call(this, opts) || this; - } - Polygon.prototype.getDefaultShape = function () { - return new PolygonShape(); - }; - Polygon.prototype.buildPath = function (ctx, shape) { - buildPath$2(ctx, shape, true); - }; - return Polygon; - }(Path)); - Polygon.prototype.type = 'polygon'; - - var PolylineShape = (function () { - function PolylineShape() { - this.points = null; - this.percent = 1; - this.smooth = 0; - this.smoothConstraint = null; - } - return PolylineShape; - }()); - var Polyline = (function (_super) { - __extends(Polyline, _super); - function Polyline(opts) { - return _super.call(this, opts) || this; - } - Polyline.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - Polyline.prototype.getDefaultShape = function () { - return new PolylineShape(); - }; - Polyline.prototype.buildPath = function (ctx, shape) { - buildPath$2(ctx, shape, false); - }; - return Polyline; - }(Path)); - Polyline.prototype.type = 'polyline'; - - var subPixelOptimizeOutputShape$1 = {}; - var LineShape = (function () { - function LineShape() { - this.x1 = 0; - this.y1 = 0; - this.x2 = 0; - this.y2 = 0; - this.percent = 1; - } - return LineShape; - }()); - var Line = (function (_super) { - __extends(Line, _super); - function Line(opts) { - return _super.call(this, opts) || this; - } - Line.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - Line.prototype.getDefaultShape = function () { - return new LineShape(); - }; - Line.prototype.buildPath = function (ctx, shape) { - var x1; - var y1; - var x2; - var y2; - if (this.subPixelOptimize) { - var optimizedShape = subPixelOptimizeLine(subPixelOptimizeOutputShape$1, shape, this.style); - x1 = optimizedShape.x1; - y1 = optimizedShape.y1; - x2 = optimizedShape.x2; - y2 = optimizedShape.y2; - } - else { - x1 = shape.x1; - y1 = shape.y1; - x2 = shape.x2; - y2 = shape.y2; - } - var percent = shape.percent; - if (percent === 0) { - return; - } - ctx.moveTo(x1, y1); - if (percent < 1) { - x2 = x1 * (1 - percent) + x2 * percent; - y2 = y1 * (1 - percent) + y2 * percent; - } - ctx.lineTo(x2, y2); - }; - Line.prototype.pointAt = function (p) { - var shape = this.shape; - return [ - shape.x1 * (1 - p) + shape.x2 * p, - shape.y1 * (1 - p) + shape.y2 * p - ]; - }; - return Line; - }(Path)); - Line.prototype.type = 'line'; - - var out = []; - var BezierCurveShape = (function () { - function BezierCurveShape() { - this.x1 = 0; - this.y1 = 0; - this.x2 = 0; - this.y2 = 0; - this.cpx1 = 0; - this.cpy1 = 0; - this.percent = 1; - } - return BezierCurveShape; - }()); - function someVectorAt(shape, t, isTangent) { - var cpx2 = shape.cpx2; - var cpy2 = shape.cpy2; - if (cpx2 != null || cpy2 != null) { - return [ - (isTangent ? cubicDerivativeAt : cubicAt)(shape.x1, shape.cpx1, shape.cpx2, shape.x2, t), - (isTangent ? cubicDerivativeAt : cubicAt)(shape.y1, shape.cpy1, shape.cpy2, shape.y2, t) - ]; - } - else { - return [ - (isTangent ? quadraticDerivativeAt : quadraticAt)(shape.x1, shape.cpx1, shape.x2, t), - (isTangent ? quadraticDerivativeAt : quadraticAt)(shape.y1, shape.cpy1, shape.y2, t) - ]; - } - } - var BezierCurve = (function (_super) { - __extends(BezierCurve, _super); - function BezierCurve(opts) { - return _super.call(this, opts) || this; - } - BezierCurve.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - BezierCurve.prototype.getDefaultShape = function () { - return new BezierCurveShape(); - }; - BezierCurve.prototype.buildPath = function (ctx, shape) { - var x1 = shape.x1; - var y1 = shape.y1; - var x2 = shape.x2; - var y2 = shape.y2; - var cpx1 = shape.cpx1; - var cpy1 = shape.cpy1; - var cpx2 = shape.cpx2; - var cpy2 = shape.cpy2; - var percent = shape.percent; - if (percent === 0) { - return; - } - ctx.moveTo(x1, y1); - if (cpx2 == null || cpy2 == null) { - if (percent < 1) { - quadraticSubdivide(x1, cpx1, x2, percent, out); - cpx1 = out[1]; - x2 = out[2]; - quadraticSubdivide(y1, cpy1, y2, percent, out); - cpy1 = out[1]; - y2 = out[2]; - } - ctx.quadraticCurveTo(cpx1, cpy1, x2, y2); - } - else { - if (percent < 1) { - cubicSubdivide(x1, cpx1, cpx2, x2, percent, out); - cpx1 = out[1]; - cpx2 = out[2]; - x2 = out[3]; - cubicSubdivide(y1, cpy1, cpy2, y2, percent, out); - cpy1 = out[1]; - cpy2 = out[2]; - y2 = out[3]; - } - ctx.bezierCurveTo(cpx1, cpy1, cpx2, cpy2, x2, y2); - } - }; - BezierCurve.prototype.pointAt = function (t) { - return someVectorAt(this.shape, t, false); - }; - BezierCurve.prototype.tangentAt = function (t) { - var p = someVectorAt(this.shape, t, true); - return normalize(p, p); - }; - return BezierCurve; - }(Path)); - BezierCurve.prototype.type = 'bezier-curve'; - - var ArcShape = (function () { - function ArcShape() { - this.cx = 0; - this.cy = 0; - this.r = 0; - this.startAngle = 0; - this.endAngle = Math.PI * 2; - this.clockwise = true; - } - return ArcShape; - }()); - var Arc = (function (_super) { - __extends(Arc, _super); - function Arc(opts) { - return _super.call(this, opts) || this; - } - Arc.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - Arc.prototype.getDefaultShape = function () { - return new ArcShape(); - }; - Arc.prototype.buildPath = function (ctx, shape) { - var x = shape.cx; - var y = shape.cy; - var r = Math.max(shape.r, 0); - var startAngle = shape.startAngle; - var endAngle = shape.endAngle; - var clockwise = shape.clockwise; - var unitX = Math.cos(startAngle); - var unitY = Math.sin(startAngle); - ctx.moveTo(unitX * r + x, unitY * r + y); - ctx.arc(x, y, r, startAngle, endAngle, !clockwise); - }; - return Arc; - }(Path)); - Arc.prototype.type = 'arc'; - - var CompoundPath = (function (_super) { - __extends(CompoundPath, _super); - function CompoundPath() { - var _this = _super !== null && _super.apply(this, arguments) || this; - _this.type = 'compound'; - return _this; - } - CompoundPath.prototype._updatePathDirty = function () { - var paths = this.shape.paths; - var dirtyPath = this.shapeChanged(); - for (var i = 0; i < paths.length; i++) { - dirtyPath = dirtyPath || paths[i].shapeChanged(); - } - if (dirtyPath) { - this.dirtyShape(); - } - }; - CompoundPath.prototype.beforeBrush = function () { - this._updatePathDirty(); - var paths = this.shape.paths || []; - var scale = this.getGlobalScale(); - for (var i = 0; i < paths.length; i++) { - if (!paths[i].path) { - paths[i].createPathProxy(); - } - paths[i].path.setScale(scale[0], scale[1], paths[i].segmentIgnoreThreshold); - } - }; - CompoundPath.prototype.buildPath = function (ctx, shape) { - var paths = shape.paths || []; - for (var i = 0; i < paths.length; i++) { - paths[i].buildPath(ctx, paths[i].shape, true); - } - }; - CompoundPath.prototype.afterBrush = function () { - var paths = this.shape.paths || []; - for (var i = 0; i < paths.length; i++) { - paths[i].pathUpdated(); - } - }; - CompoundPath.prototype.getBoundingRect = function () { - this._updatePathDirty.call(this); - return Path.prototype.getBoundingRect.call(this); - }; - return CompoundPath; - }(Path)); - - var Gradient = (function () { - function Gradient(colorStops) { - this.colorStops = colorStops || []; - } - Gradient.prototype.addColorStop = function (offset, color) { - this.colorStops.push({ - offset: offset, - color: color - }); - }; - return Gradient; - }()); - - var LinearGradient = (function (_super) { - __extends(LinearGradient, _super); - function LinearGradient(x, y, x2, y2, colorStops, globalCoord) { - var _this = _super.call(this, colorStops) || this; - _this.x = x == null ? 0 : x; - _this.y = y == null ? 0 : y; - _this.x2 = x2 == null ? 1 : x2; - _this.y2 = y2 == null ? 0 : y2; - _this.type = 'linear'; - _this.global = globalCoord || false; - return _this; - } - return LinearGradient; - }(Gradient)); - - var RadialGradient = (function (_super) { - __extends(RadialGradient, _super); - function RadialGradient(x, y, r, colorStops, globalCoord) { - var _this = _super.call(this, colorStops) || this; - _this.x = x == null ? 0.5 : x; - _this.y = y == null ? 0.5 : y; - _this.r = r == null ? 0.5 : r; - _this.type = 'radial'; - _this.global = globalCoord || false; - return _this; - } - return RadialGradient; - }(Gradient)); - - var extent = [0, 0]; - var extent2 = [0, 0]; - var minTv$1 = new Point(); - var maxTv$1 = new Point(); - var OrientedBoundingRect = (function () { - function OrientedBoundingRect(rect, transform) { - this._corners = []; - this._axes = []; - this._origin = [0, 0]; - for (var i = 0; i < 4; i++) { - this._corners[i] = new Point(); - } - for (var i = 0; i < 2; i++) { - this._axes[i] = new Point(); - } - if (rect) { - this.fromBoundingRect(rect, transform); - } - } - OrientedBoundingRect.prototype.fromBoundingRect = function (rect, transform) { - var corners = this._corners; - var axes = this._axes; - var x = rect.x; - var y = rect.y; - var x2 = x + rect.width; - var y2 = y + rect.height; - corners[0].set(x, y); - corners[1].set(x2, y); - corners[2].set(x2, y2); - corners[3].set(x, y2); - if (transform) { - for (var i = 0; i < 4; i++) { - corners[i].transform(transform); - } - } - Point.sub(axes[0], corners[1], corners[0]); - Point.sub(axes[1], corners[3], corners[0]); - axes[0].normalize(); - axes[1].normalize(); - for (var i = 0; i < 2; i++) { - this._origin[i] = axes[i].dot(corners[0]); - } - }; - OrientedBoundingRect.prototype.intersect = function (other, mtv) { - var overlapped = true; - var noMtv = !mtv; - minTv$1.set(Infinity, Infinity); - maxTv$1.set(0, 0); - if (!this._intersectCheckOneSide(this, other, minTv$1, maxTv$1, noMtv, 1)) { - overlapped = false; - if (noMtv) { - return overlapped; - } - } - if (!this._intersectCheckOneSide(other, this, minTv$1, maxTv$1, noMtv, -1)) { - overlapped = false; - if (noMtv) { - return overlapped; - } - } - if (!noMtv) { - Point.copy(mtv, overlapped ? minTv$1 : maxTv$1); - } - return overlapped; - }; - OrientedBoundingRect.prototype._intersectCheckOneSide = function (self, other, minTv, maxTv, noMtv, inverse) { - var overlapped = true; - for (var i = 0; i < 2; i++) { - var axis = this._axes[i]; - this._getProjMinMaxOnAxis(i, self._corners, extent); - this._getProjMinMaxOnAxis(i, other._corners, extent2); - if (extent[1] < extent2[0] || extent[0] > extent2[1]) { - overlapped = false; - if (noMtv) { - return overlapped; - } - var dist0 = Math.abs(extent2[0] - extent[1]); - var dist1 = Math.abs(extent[0] - extent2[1]); - if (Math.min(dist0, dist1) > maxTv.len()) { - if (dist0 < dist1) { - Point.scale(maxTv, axis, -dist0 * inverse); - } - else { - Point.scale(maxTv, axis, dist1 * inverse); - } - } - } - else if (minTv) { - var dist0 = Math.abs(extent2[0] - extent[1]); - var dist1 = Math.abs(extent[0] - extent2[1]); - if (Math.min(dist0, dist1) < minTv.len()) { - if (dist0 < dist1) { - Point.scale(minTv, axis, dist0 * inverse); - } - else { - Point.scale(minTv, axis, -dist1 * inverse); - } - } - } - } - return overlapped; - }; - OrientedBoundingRect.prototype._getProjMinMaxOnAxis = function (dim, corners, out) { - var axis = this._axes[dim]; - var origin = this._origin; - var proj = corners[0].dot(axis) + origin[dim]; - var min = proj; - var max = proj; - for (var i = 1; i < corners.length; i++) { - var proj_1 = corners[i].dot(axis) + origin[dim]; - min = Math.min(proj_1, min); - max = Math.max(proj_1, max); - } - out[0] = min; - out[1] = max; - }; - return OrientedBoundingRect; - }()); - - var m = []; - var IncrementalDisplayable = (function (_super) { - __extends(IncrementalDisplayable, _super); - function IncrementalDisplayable() { - var _this = _super !== null && _super.apply(this, arguments) || this; - _this.notClear = true; - _this.incremental = true; - _this._displayables = []; - _this._temporaryDisplayables = []; - _this._cursor = 0; - return _this; - } - IncrementalDisplayable.prototype.traverse = function (cb, context) { - cb.call(context, this); - }; - IncrementalDisplayable.prototype.useStyle = function () { - this.style = {}; - }; - IncrementalDisplayable.prototype.getCursor = function () { - return this._cursor; - }; - IncrementalDisplayable.prototype.innerAfterBrush = function () { - this._cursor = this._displayables.length; - }; - IncrementalDisplayable.prototype.clearDisplaybles = function () { - this._displayables = []; - this._temporaryDisplayables = []; - this._cursor = 0; - this.markRedraw(); - this.notClear = false; - }; - IncrementalDisplayable.prototype.clearTemporalDisplayables = function () { - this._temporaryDisplayables = []; - }; - IncrementalDisplayable.prototype.addDisplayable = function (displayable, notPersistent) { - if (notPersistent) { - this._temporaryDisplayables.push(displayable); - } - else { - this._displayables.push(displayable); - } - this.markRedraw(); - }; - IncrementalDisplayable.prototype.addDisplayables = function (displayables, notPersistent) { - notPersistent = notPersistent || false; - for (var i = 0; i < displayables.length; i++) { - this.addDisplayable(displayables[i], notPersistent); - } - }; - IncrementalDisplayable.prototype.getDisplayables = function () { - return this._displayables; - }; - IncrementalDisplayable.prototype.getTemporalDisplayables = function () { - return this._temporaryDisplayables; - }; - IncrementalDisplayable.prototype.eachPendingDisplayable = function (cb) { - for (var i = this._cursor; i < this._displayables.length; i++) { - cb && cb(this._displayables[i]); - } - for (var i = 0; i < this._temporaryDisplayables.length; i++) { - cb && cb(this._temporaryDisplayables[i]); - } - }; - IncrementalDisplayable.prototype.update = function () { - this.updateTransform(); - for (var i = this._cursor; i < this._displayables.length; i++) { - var displayable = this._displayables[i]; - displayable.parent = this; - displayable.update(); - displayable.parent = null; - } - for (var i = 0; i < this._temporaryDisplayables.length; i++) { - var displayable = this._temporaryDisplayables[i]; - displayable.parent = this; - displayable.update(); - displayable.parent = null; - } - }; - IncrementalDisplayable.prototype.getBoundingRect = function () { - if (!this._rect) { - var rect = new BoundingRect(Infinity, Infinity, -Infinity, -Infinity); - for (var i = 0; i < this._displayables.length; i++) { - var displayable = this._displayables[i]; - var childRect = displayable.getBoundingRect().clone(); - if (displayable.needLocalTransform()) { - childRect.applyTransform(displayable.getLocalTransform(m)); - } - rect.union(childRect); - } - this._rect = rect; - } - return this._rect; - }; - IncrementalDisplayable.prototype.contain = function (x, y) { - var localPos = this.transformCoordToLocal(x, y); - var rect = this.getBoundingRect(); - if (rect.contain(localPos[0], localPos[1])) { - for (var i = 0; i < this._displayables.length; i++) { - var displayable = this._displayables[i]; - if (displayable.contain(x, y)) { - return true; - } - } - } - return false; - }; - return IncrementalDisplayable; - }(Displayable)); - - var transitionStore = makeInner(); - /** - * Return null if animation is disabled. - */ - - function getAnimationConfig(animationType, animatableModel, dataIndex, // Extra opts can override the option in animatable model. - extraOpts, // TODO It's only for pictorial bar now. - extraDelayParams) { - var animationPayload; // Check if there is global animation configuration from dataZoom/resize can override the config in option. - // If animation is enabled. Will use this animation config in payload. - // If animation is disabled. Just ignore it. - - if (animatableModel && animatableModel.ecModel) { - var updatePayload = animatableModel.ecModel.getUpdatePayload(); - animationPayload = updatePayload && updatePayload.animation; - } - - var animationEnabled = animatableModel && animatableModel.isAnimationEnabled(); - var isUpdate = animationType === 'update'; - - if (animationEnabled) { - var duration = void 0; - var easing = void 0; - var delay = void 0; - - if (extraOpts) { - duration = retrieve2(extraOpts.duration, 200); - easing = retrieve2(extraOpts.easing, 'cubicOut'); - delay = 0; - } else { - duration = animatableModel.getShallow(isUpdate ? 'animationDurationUpdate' : 'animationDuration'); - easing = animatableModel.getShallow(isUpdate ? 'animationEasingUpdate' : 'animationEasing'); - delay = animatableModel.getShallow(isUpdate ? 'animationDelayUpdate' : 'animationDelay'); - } // animation from payload has highest priority. - - - if (animationPayload) { - animationPayload.duration != null && (duration = animationPayload.duration); - animationPayload.easing != null && (easing = animationPayload.easing); - animationPayload.delay != null && (delay = animationPayload.delay); - } - - if (isFunction(delay)) { - delay = delay(dataIndex, extraDelayParams); - } - - if (isFunction(duration)) { - duration = duration(dataIndex); - } - - var config = { - duration: duration || 0, - delay: delay, - easing: easing - }; - return config; - } else { - return null; - } - } - - function animateOrSetProps(animationType, el, props, animatableModel, dataIndex, cb, during) { - var isFrom = false; - var removeOpt; - - if (isFunction(dataIndex)) { - during = cb; - cb = dataIndex; - dataIndex = null; - } else if (isObject(dataIndex)) { - cb = dataIndex.cb; - during = dataIndex.during; - isFrom = dataIndex.isFrom; - removeOpt = dataIndex.removeOpt; - dataIndex = dataIndex.dataIndex; - } - - var isRemove = animationType === 'leave'; - - if (!isRemove) { - // Must stop the remove animation. - el.stopAnimation('leave'); - } - - var animationConfig = getAnimationConfig(animationType, animatableModel, dataIndex, isRemove ? removeOpt || {} : null, animatableModel && animatableModel.getAnimationDelayParams ? animatableModel.getAnimationDelayParams(el, dataIndex) : null); - - if (animationConfig && animationConfig.duration > 0) { - var duration = animationConfig.duration; - var animationDelay = animationConfig.delay; - var animationEasing = animationConfig.easing; - var animateConfig = { - duration: duration, - delay: animationDelay || 0, - easing: animationEasing, - done: cb, - force: !!cb || !!during, - // Set to final state in update/init animation. - // So the post processing based on the path shape can be done correctly. - setToFinal: !isRemove, - scope: animationType, - during: during - }; - isFrom ? el.animateFrom(props, animateConfig) : el.animateTo(props, animateConfig); - } else { - el.stopAnimation(); // If `isFrom`, the props is the "from" props. - - !isFrom && el.attr(props); // Call during at least once. - - during && during(1); - cb && cb(); - } - } - /** - * Update graphic element properties with or without animation according to the - * configuration in series. - * - * Caution: this method will stop previous animation. - * So do not use this method to one element twice before - * animation starts, unless you know what you are doing. - * @example - * graphic.updateProps(el, { - * position: [100, 100] - * }, seriesModel, dataIndex, function () { console.log('Animation done!'); }); - * // Or - * graphic.updateProps(el, { - * position: [100, 100] - * }, seriesModel, function () { console.log('Animation done!'); }); - */ - - - function updateProps(el, props, // TODO: TYPE AnimatableModel - animatableModel, dataIndex, cb, during) { - animateOrSetProps('update', el, props, animatableModel, dataIndex, cb, during); - } - /** - * Init graphic element properties with or without animation according to the - * configuration in series. - * - * Caution: this method will stop previous animation. - * So do not use this method to one element twice before - * animation starts, unless you know what you are doing. - */ - - function initProps(el, props, animatableModel, dataIndex, cb, during) { - animateOrSetProps('enter', el, props, animatableModel, dataIndex, cb, during); - } - /** - * If element is removed. - * It can determine if element is having remove animation. - */ - - function isElementRemoved(el) { - if (!el.__zr) { - return true; - } - - for (var i = 0; i < el.animators.length; i++) { - var animator = el.animators[i]; - - if (animator.scope === 'leave') { - return true; - } - } - - return false; - } - /** - * Remove graphic element - */ - - function removeElement(el, props, animatableModel, dataIndex, cb, during) { - // Don't do remove animation twice. - if (isElementRemoved(el)) { - return; - } - - animateOrSetProps('leave', el, props, animatableModel, dataIndex, cb, during); - } - - function fadeOutDisplayable(el, animatableModel, dataIndex, done) { - el.removeTextContent(); - el.removeTextGuideLine(); - removeElement(el, { - style: { - opacity: 0 - } - }, animatableModel, dataIndex, done); - } - - function removeElementWithFadeOut(el, animatableModel, dataIndex) { - function doRemove() { - el.parent && el.parent.remove(el); - } // Hide label and labelLine first - // TODO Also use fade out animation? - - - if (!el.isGroup) { - fadeOutDisplayable(el, animatableModel, dataIndex, doRemove); - } else { - el.traverse(function (disp) { - if (!disp.isGroup) { - // Can invoke doRemove multiple times. - fadeOutDisplayable(disp, animatableModel, dataIndex, doRemove); - } - }); - } - } - /** - * Save old style for style transition in universalTransition module. - * It's used when element will be reused in each render. - * For chart like map, heatmap, which will always create new element. - * We don't need to save this because universalTransition can get old style from the old element - */ - - function saveOldStyle(el) { - transitionStore(el).oldStyle = el.style; - } - function getOldStyle(el) { - return transitionStore(el).oldStyle; - } - - var mathMax$4 = Math.max; - var mathMin$4 = Math.min; - var _customShapeMap = {}; - /** - * Extend shape with parameters - */ - - function extendShape(opts) { - return Path.extend(opts); - } - var extendPathFromString = extendFromString; - /** - * Extend path - */ - - function extendPath(pathData, opts) { - return extendPathFromString(pathData, opts); - } - /** - * Register a user defined shape. - * The shape class can be fetched by `getShapeClass` - * This method will overwrite the registered shapes, including - * the registered built-in shapes, if using the same `name`. - * The shape can be used in `custom series` and - * `graphic component` by declaring `{type: name}`. - * - * @param name - * @param ShapeClass Can be generated by `extendShape`. - */ - - function registerShape(name, ShapeClass) { - _customShapeMap[name] = ShapeClass; - } - /** - * Find shape class registered by `registerShape`. Usually used in - * fetching user defined shape. - * - * [Caution]: - * (1) This method **MUST NOT be used inside echarts !!!**, unless it is prepared - * to use user registered shapes. - * Because the built-in shape (see `getBuiltInShape`) will be registered by - * `registerShape` by default. That enables users to get both built-in - * shapes as well as the shapes belonging to themsleves. But users can overwrite - * the built-in shapes by using names like 'circle', 'rect' via calling - * `registerShape`. So the echarts inner featrues should not fetch shapes from here - * in case that it is overwritten by users, except that some features, like - * `custom series`, `graphic component`, do it deliberately. - * - * (2) In the features like `custom series`, `graphic component`, the user input - * `{tpye: 'xxx'}` does not only specify shapes but also specify other graphic - * elements like `'group'`, `'text'`, `'image'` or event `'path'`. Those names - * are reserved names, that is, if some user registers a shape named `'image'`, - * the shape will not be used. If we intending to add some more reserved names - * in feature, that might bring break changes (disable some existing user shape - * names). But that case probably rarely happens. So we don't make more mechanism - * to resolve this issue here. - * - * @param name - * @return The shape class. If not found, return nothing. - */ - - function getShapeClass(name) { - if (_customShapeMap.hasOwnProperty(name)) { - return _customShapeMap[name]; - } - } - /** - * Create a path element from path data string - * @param pathData - * @param opts - * @param rect - * @param layout 'center' or 'cover' default to be cover - */ - - function makePath(pathData, opts, rect, layout) { - var path = createFromString(pathData, opts); - - if (rect) { - if (layout === 'center') { - rect = centerGraphic(rect, path.getBoundingRect()); - } - - resizePath(path, rect); - } - - return path; - } - /** - * Create a image element from image url - * @param imageUrl image url - * @param opts options - * @param rect constrain rect - * @param layout 'center' or 'cover'. Default to be 'cover' - */ - - function makeImage(imageUrl, rect, layout) { - var zrImg = new ZRImage({ - style: { - image: imageUrl, - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height - }, - onload: function (img) { - if (layout === 'center') { - var boundingRect = { - width: img.width, - height: img.height - }; - zrImg.setStyle(centerGraphic(rect, boundingRect)); - } - } - }); - return zrImg; - } - /** - * Get position of centered element in bounding box. - * - * @param rect element local bounding box - * @param boundingRect constraint bounding box - * @return element position containing x, y, width, and height - */ - - function centerGraphic(rect, boundingRect) { - // Set rect to center, keep width / height ratio. - var aspect = boundingRect.width / boundingRect.height; - var width = rect.height * aspect; - var height; - - if (width <= rect.width) { - height = rect.height; - } else { - width = rect.width; - height = width / aspect; - } - - var cx = rect.x + rect.width / 2; - var cy = rect.y + rect.height / 2; - return { - x: cx - width / 2, - y: cy - height / 2, - width: width, - height: height - }; - } - - var mergePath$1 = mergePath; - /** - * Resize a path to fit the rect - * @param path - * @param rect - */ - - function resizePath(path, rect) { - if (!path.applyTransform) { - return; - } - - var pathRect = path.getBoundingRect(); - var m = pathRect.calculateTransform(rect); - path.applyTransform(m); - } - /** - * Sub pixel optimize line for canvas - */ - - function subPixelOptimizeLine$1(shape, lineWidth) { - subPixelOptimizeLine(shape, shape, { - lineWidth: lineWidth - }); - return shape; - } - /** - * Sub pixel optimize rect for canvas - */ - - function subPixelOptimizeRect$1(param) { - subPixelOptimizeRect(param.shape, param.shape, param.style); - return param; - } - /** - * Sub pixel optimize for canvas - * - * @param position Coordinate, such as x, y - * @param lineWidth Should be nonnegative integer. - * @param positiveOrNegative Default false (negative). - * @return Optimized position. - */ - - var subPixelOptimize$1 = subPixelOptimize; - /** - * Get transform matrix of target (param target), - * in coordinate of its ancestor (param ancestor) - * - * @param target - * @param [ancestor] - */ - - function getTransform(target, ancestor) { - var mat = identity([]); - - while (target && target !== ancestor) { - mul$1(mat, target.getLocalTransform(), mat); - target = target.parent; - } - - return mat; - } - /** - * Apply transform to an vertex. - * @param target [x, y] - * @param transform Can be: - * + Transform matrix: like [1, 0, 0, 1, 0, 0] - * + {position, rotation, scale}, the same as `zrender/Transformable`. - * @param invert Whether use invert matrix. - * @return [x, y] - */ - - function applyTransform$1(target, transform, invert$1) { - if (transform && !isArrayLike(transform)) { - transform = Transformable.getLocalTransform(transform); - } - - if (invert$1) { - transform = invert([], transform); - } - - return applyTransform([], target, transform); - } - /** - * @param direction 'left' 'right' 'top' 'bottom' - * @param transform Transform matrix: like [1, 0, 0, 1, 0, 0] - * @param invert Whether use invert matrix. - * @return Transformed direction. 'left' 'right' 'top' 'bottom' - */ - - function transformDirection(direction, transform, invert) { - // Pick a base, ensure that transform result will not be (0, 0). - var hBase = transform[4] === 0 || transform[5] === 0 || transform[0] === 0 ? 1 : Math.abs(2 * transform[4] / transform[0]); - var vBase = transform[4] === 0 || transform[5] === 0 || transform[2] === 0 ? 1 : Math.abs(2 * transform[4] / transform[2]); - var vertex = [direction === 'left' ? -hBase : direction === 'right' ? hBase : 0, direction === 'top' ? -vBase : direction === 'bottom' ? vBase : 0]; - vertex = applyTransform$1(vertex, transform, invert); - return Math.abs(vertex[0]) > Math.abs(vertex[1]) ? vertex[0] > 0 ? 'right' : 'left' : vertex[1] > 0 ? 'bottom' : 'top'; - } - - function isNotGroup(el) { - return !el.isGroup; - } - - function isPath(el) { - return el.shape != null; - } - /** - * Apply group transition animation from g1 to g2. - * If no animatableModel, no animation. - */ - - - function groupTransition(g1, g2, animatableModel) { - if (!g1 || !g2) { - return; - } - - function getElMap(g) { - var elMap = {}; - g.traverse(function (el) { - if (isNotGroup(el) && el.anid) { - elMap[el.anid] = el; - } - }); - return elMap; - } - - function getAnimatableProps(el) { - var obj = { - x: el.x, - y: el.y, - rotation: el.rotation - }; - - if (isPath(el)) { - obj.shape = extend({}, el.shape); - } - - return obj; - } - - var elMap1 = getElMap(g1); - g2.traverse(function (el) { - if (isNotGroup(el) && el.anid) { - var oldEl = elMap1[el.anid]; - - if (oldEl) { - var newProp = getAnimatableProps(el); - el.attr(getAnimatableProps(oldEl)); - updateProps(el, newProp, animatableModel, getECData(el).dataIndex); - } - } - }); - } - function clipPointsByRect(points, rect) { - // FIXME: This way might be incorrect when graphic clipped by a corner - // and when element has a border. - return map(points, function (point) { - var x = point[0]; - x = mathMax$4(x, rect.x); - x = mathMin$4(x, rect.x + rect.width); - var y = point[1]; - y = mathMax$4(y, rect.y); - y = mathMin$4(y, rect.y + rect.height); - return [x, y]; - }); - } - /** - * Return a new clipped rect. If rect size are negative, return undefined. - */ - - function clipRectByRect(targetRect, rect) { - var x = mathMax$4(targetRect.x, rect.x); - var x2 = mathMin$4(targetRect.x + targetRect.width, rect.x + rect.width); - var y = mathMax$4(targetRect.y, rect.y); - var y2 = mathMin$4(targetRect.y + targetRect.height, rect.y + rect.height); // If the total rect is cliped, nothing, including the border, - // should be painted. So return undefined. - - if (x2 >= x && y2 >= y) { - return { - x: x, - y: y, - width: x2 - x, - height: y2 - y - }; - } - } - function createIcon(iconStr, // Support 'image://' or 'path://' or direct svg path. - opt, rect) { - var innerOpts = extend({ - rectHover: true - }, opt); - var style = innerOpts.style = { - strokeNoScale: true - }; - rect = rect || { - x: -1, - y: -1, - width: 2, - height: 2 - }; - - if (iconStr) { - return iconStr.indexOf('image://') === 0 ? (style.image = iconStr.slice(8), defaults(style, rect), new ZRImage(innerOpts)) : makePath(iconStr.replace('path://', ''), innerOpts, rect, 'center'); - } - } - /** - * Return `true` if the given line (line `a`) and the given polygon - * are intersect. - * Note that we do not count colinear as intersect here because no - * requirement for that. We could do that if required in future. - */ - - function linePolygonIntersect(a1x, a1y, a2x, a2y, points) { - for (var i = 0, p2 = points[points.length - 1]; i < points.length; i++) { - var p = points[i]; - - if (lineLineIntersect(a1x, a1y, a2x, a2y, p[0], p[1], p2[0], p2[1])) { - return true; - } - - p2 = p; - } - } - /** - * Return `true` if the given two lines (line `a` and line `b`) - * are intersect. - * Note that we do not count colinear as intersect here because no - * requirement for that. We could do that if required in future. - */ - - function lineLineIntersect(a1x, a1y, a2x, a2y, b1x, b1y, b2x, b2y) { - // let `vec_m` to be `vec_a2 - vec_a1` and `vec_n` to be `vec_b2 - vec_b1`. - var mx = a2x - a1x; - var my = a2y - a1y; - var nx = b2x - b1x; - var ny = b2y - b1y; // `vec_m` and `vec_n` are parallel iff - // existing `k` such that `vec_m = k · vec_n`, equivalent to `vec_m X vec_n = 0`. - - var nmCrossProduct = crossProduct2d(nx, ny, mx, my); - - if (nearZero(nmCrossProduct)) { - return false; - } // `vec_m` and `vec_n` are intersect iff - // existing `p` and `q` in [0, 1] such that `vec_a1 + p * vec_m = vec_b1 + q * vec_n`, - // such that `q = ((vec_a1 - vec_b1) X vec_m) / (vec_n X vec_m)` - // and `p = ((vec_a1 - vec_b1) X vec_n) / (vec_n X vec_m)`. - - - var b1a1x = a1x - b1x; - var b1a1y = a1y - b1y; - var q = crossProduct2d(b1a1x, b1a1y, mx, my) / nmCrossProduct; - - if (q < 0 || q > 1) { - return false; - } - - var p = crossProduct2d(b1a1x, b1a1y, nx, ny) / nmCrossProduct; - - if (p < 0 || p > 1) { - return false; - } - - return true; - } - /** - * Cross product of 2-dimension vector. - */ - - function crossProduct2d(x1, y1, x2, y2) { - return x1 * y2 - x2 * y1; - } - - function nearZero(val) { - return val <= 1e-6 && val >= -1e-6; - } - - function setTooltipConfig(opt) { - var itemTooltipOption = opt.itemTooltipOption; - var componentModel = opt.componentModel; - var itemName = opt.itemName; - var itemTooltipOptionObj = isString(itemTooltipOption) ? { - formatter: itemTooltipOption - } : itemTooltipOption; - var mainType = componentModel.mainType; - var componentIndex = componentModel.componentIndex; - var formatterParams = { - componentType: mainType, - name: itemName, - $vars: ['name'] - }; - formatterParams[mainType + 'Index'] = componentIndex; - var formatterParamsExtra = opt.formatterParamsExtra; - - if (formatterParamsExtra) { - each(keys(formatterParamsExtra), function (key) { - if (!hasOwn(formatterParams, key)) { - formatterParams[key] = formatterParamsExtra[key]; - formatterParams.$vars.push(key); - } - }); - } - - var ecData = getECData(opt.el); - ecData.componentMainType = mainType; - ecData.componentIndex = componentIndex; - ecData.tooltipConfig = { - name: itemName, - option: defaults({ - content: itemName, - formatterParams: formatterParams - }, itemTooltipOptionObj) - }; - } - - function traverseElement(el, cb) { - var stopped; // TODO - // Polyfill for fixing zrender group traverse don't visit it's root issue. - - if (el.isGroup) { - stopped = cb(el); - } - - if (!stopped) { - el.traverse(cb); - } - } - - function traverseElements(els, cb) { - if (els) { - if (isArray(els)) { - for (var i = 0; i < els.length; i++) { - traverseElement(els[i], cb); - } - } else { - traverseElement(els, cb); - } - } - } // Register built-in shapes. These shapes might be overwritten - // by users, although we do not recommend that. - - registerShape('circle', Circle); - registerShape('ellipse', Ellipse); - registerShape('sector', Sector); - registerShape('ring', Ring); - registerShape('polygon', Polygon); - registerShape('polyline', Polyline); - registerShape('rect', Rect); - registerShape('line', Line); - registerShape('bezierCurve', BezierCurve); - registerShape('arc', Arc); - - var graphic = /*#__PURE__*/Object.freeze({ - __proto__: null, - updateProps: updateProps, - initProps: initProps, - removeElement: removeElement, - removeElementWithFadeOut: removeElementWithFadeOut, - isElementRemoved: isElementRemoved, - extendShape: extendShape, - extendPath: extendPath, - registerShape: registerShape, - getShapeClass: getShapeClass, - makePath: makePath, - makeImage: makeImage, - mergePath: mergePath$1, - resizePath: resizePath, - subPixelOptimizeLine: subPixelOptimizeLine$1, - subPixelOptimizeRect: subPixelOptimizeRect$1, - subPixelOptimize: subPixelOptimize$1, - getTransform: getTransform, - applyTransform: applyTransform$1, - transformDirection: transformDirection, - groupTransition: groupTransition, - clipPointsByRect: clipPointsByRect, - clipRectByRect: clipRectByRect, - createIcon: createIcon, - linePolygonIntersect: linePolygonIntersect, - lineLineIntersect: lineLineIntersect, - setTooltipConfig: setTooltipConfig, - traverseElements: traverseElements, - Group: Group, - Image: ZRImage, - Text: ZRText, - Circle: Circle, - Ellipse: Ellipse, - Sector: Sector, - Ring: Ring, - Polygon: Polygon, - Polyline: Polyline, - Rect: Rect, - Line: Line, - BezierCurve: BezierCurve, - Arc: Arc, - IncrementalDisplayable: IncrementalDisplayable, - CompoundPath: CompoundPath, - LinearGradient: LinearGradient, - RadialGradient: RadialGradient, - BoundingRect: BoundingRect, - OrientedBoundingRect: OrientedBoundingRect, - Point: Point, - Path: Path - }); - - var EMPTY_OBJ = {}; - function setLabelText(label, labelTexts) { - for (var i = 0; i < SPECIAL_STATES.length; i++) { - var stateName = SPECIAL_STATES[i]; - var text = labelTexts[stateName]; - var state = label.ensureState(stateName); - state.style = state.style || {}; - state.style.text = text; - } - - var oldStates = label.currentStates.slice(); - label.clearStates(true); - label.setStyle({ - text: labelTexts.normal - }); - label.useStates(oldStates, true); - } - - function getLabelText(opt, stateModels, interpolatedValue) { - var labelFetcher = opt.labelFetcher; - var labelDataIndex = opt.labelDataIndex; - var labelDimIndex = opt.labelDimIndex; - var normalModel = stateModels.normal; - var baseText; - - if (labelFetcher) { - baseText = labelFetcher.getFormattedLabel(labelDataIndex, 'normal', null, labelDimIndex, normalModel && normalModel.get('formatter'), interpolatedValue != null ? { - interpolatedValue: interpolatedValue - } : null); - } - - if (baseText == null) { - baseText = isFunction(opt.defaultText) ? opt.defaultText(labelDataIndex, opt, interpolatedValue) : opt.defaultText; - } - - var statesText = { - normal: baseText - }; - - for (var i = 0; i < SPECIAL_STATES.length; i++) { - var stateName = SPECIAL_STATES[i]; - var stateModel = stateModels[stateName]; - statesText[stateName] = retrieve2(labelFetcher ? labelFetcher.getFormattedLabel(labelDataIndex, stateName, null, labelDimIndex, stateModel && stateModel.get('formatter')) : null, baseText); - } - - return statesText; - } - - function setLabelStyle(targetEl, labelStatesModels, opt, stateSpecified // TODO specified position? - ) { - opt = opt || EMPTY_OBJ; - var isSetOnText = targetEl instanceof ZRText; - var needsCreateText = false; - - for (var i = 0; i < DISPLAY_STATES.length; i++) { - var stateModel = labelStatesModels[DISPLAY_STATES[i]]; - - if (stateModel && stateModel.getShallow('show')) { - needsCreateText = true; - break; - } - } - - var textContent = isSetOnText ? targetEl : targetEl.getTextContent(); - - if (needsCreateText) { - if (!isSetOnText) { - // Reuse the previous - if (!textContent) { - textContent = new ZRText(); - targetEl.setTextContent(textContent); - } // Use same state proxy - - - if (targetEl.stateProxy) { - textContent.stateProxy = targetEl.stateProxy; - } - } - - var labelStatesTexts = getLabelText(opt, labelStatesModels); - var normalModel = labelStatesModels.normal; - var showNormal = !!normalModel.getShallow('show'); - var normalStyle = createTextStyle(normalModel, stateSpecified && stateSpecified.normal, opt, false, !isSetOnText); - normalStyle.text = labelStatesTexts.normal; - - if (!isSetOnText) { - // Always create new - targetEl.setTextConfig(createTextConfig(normalModel, opt, false)); - } - - for (var i = 0; i < SPECIAL_STATES.length; i++) { - var stateName = SPECIAL_STATES[i]; - var stateModel = labelStatesModels[stateName]; - - if (stateModel) { - var stateObj = textContent.ensureState(stateName); - var stateShow = !!retrieve2(stateModel.getShallow('show'), showNormal); - - if (stateShow !== showNormal) { - stateObj.ignore = !stateShow; - } - - stateObj.style = createTextStyle(stateModel, stateSpecified && stateSpecified[stateName], opt, true, !isSetOnText); - stateObj.style.text = labelStatesTexts[stateName]; - - if (!isSetOnText) { - var targetElEmphasisState = targetEl.ensureState(stateName); - targetElEmphasisState.textConfig = createTextConfig(stateModel, opt, true); - } - } - } // PENDING: if there is many requirements that emphasis position - // need to be different from normal position, we might consider - // auto silent is those cases. - - - textContent.silent = !!normalModel.getShallow('silent'); // Keep x and y - - if (textContent.style.x != null) { - normalStyle.x = textContent.style.x; - } - - if (textContent.style.y != null) { - normalStyle.y = textContent.style.y; - } - - textContent.ignore = !showNormal; // Always create new style. - - textContent.useStyle(normalStyle); - textContent.dirty(); - - if (opt.enableTextSetter) { - labelInner(textContent).setLabelText = function (interpolatedValue) { - var labelStatesTexts = getLabelText(opt, labelStatesModels, interpolatedValue); - setLabelText(textContent, labelStatesTexts); - }; - } - } else if (textContent) { - // Not display rich text. - textContent.ignore = true; - } - - targetEl.dirty(); - } - function getLabelStatesModels(itemModel, labelName) { - labelName = labelName || 'label'; - var statesModels = { - normal: itemModel.getModel(labelName) - }; - - for (var i = 0; i < SPECIAL_STATES.length; i++) { - var stateName = SPECIAL_STATES[i]; - statesModels[stateName] = itemModel.getModel([stateName, labelName]); - } - - return statesModels; - } - /** - * Set basic textStyle properties. - */ - - function createTextStyle(textStyleModel, specifiedTextStyle, // Fixed style in the code. Can't be set by model. - opt, isNotNormal, isAttached // If text is attached on an element. If so, auto color will handling in zrender. - ) { - var textStyle = {}; - setTextStyleCommon(textStyle, textStyleModel, opt, isNotNormal, isAttached); - specifiedTextStyle && extend(textStyle, specifiedTextStyle); // textStyle.host && textStyle.host.dirty && textStyle.host.dirty(false); - - return textStyle; - } - function createTextConfig(textStyleModel, opt, isNotNormal) { - opt = opt || {}; - var textConfig = {}; - var labelPosition; - var labelRotate = textStyleModel.getShallow('rotate'); - var labelDistance = retrieve2(textStyleModel.getShallow('distance'), isNotNormal ? null : 5); - var labelOffset = textStyleModel.getShallow('offset'); - labelPosition = textStyleModel.getShallow('position') || (isNotNormal ? null : 'inside'); // 'outside' is not a valid zr textPostion value, but used - // in bar series, and magric type should be considered. - - labelPosition === 'outside' && (labelPosition = opt.defaultOutsidePosition || 'top'); - - if (labelPosition != null) { - textConfig.position = labelPosition; - } - - if (labelOffset != null) { - textConfig.offset = labelOffset; - } - - if (labelRotate != null) { - labelRotate *= Math.PI / 180; - textConfig.rotation = labelRotate; - } - - if (labelDistance != null) { - textConfig.distance = labelDistance; - } // fill and auto is determined by the color of path fill if it's not specified by developers. - - - textConfig.outsideFill = textStyleModel.get('color') === 'inherit' ? opt.inheritColor || null : 'auto'; - return textConfig; - } - /** - * The uniform entry of set text style, that is, retrieve style definitions - * from `model` and set to `textStyle` object. - * - * Never in merge mode, but in overwrite mode, that is, all of the text style - * properties will be set. (Consider the states of normal and emphasis and - * default value can be adopted, merge would make the logic too complicated - * to manage.) - */ - - function setTextStyleCommon(textStyle, textStyleModel, opt, isNotNormal, isAttached) { - // Consider there will be abnormal when merge hover style to normal style if given default value. - opt = opt || EMPTY_OBJ; - var ecModel = textStyleModel.ecModel; - var globalTextStyle = ecModel && ecModel.option.textStyle; // Consider case: - // { - // data: [{ - // value: 12, - // label: { - // rich: { - // // no 'a' here but using parent 'a'. - // } - // } - // }], - // rich: { - // a: { ... } - // } - // } - - var richItemNames = getRichItemNames(textStyleModel); - var richResult; - - if (richItemNames) { - richResult = {}; - - for (var name_1 in richItemNames) { - if (richItemNames.hasOwnProperty(name_1)) { - // Cascade is supported in rich. - var richTextStyle = textStyleModel.getModel(['rich', name_1]); // In rich, never `disableBox`. - // FIXME: consider `label: {formatter: '{a|xx}', color: 'blue', rich: {a: {}}}`, - // the default color `'blue'` will not be adopted if no color declared in `rich`. - // That might confuses users. So probably we should put `textStyleModel` as the - // root ancestor of the `richTextStyle`. But that would be a break change. - - setTokenTextStyle(richResult[name_1] = {}, richTextStyle, globalTextStyle, opt, isNotNormal, isAttached, false, true); - } - } - } - - if (richResult) { - textStyle.rich = richResult; - } - - var overflow = textStyleModel.get('overflow'); - - if (overflow) { - textStyle.overflow = overflow; - } - - var margin = textStyleModel.get('minMargin'); - - if (margin != null) { - textStyle.margin = margin; - } - - setTokenTextStyle(textStyle, textStyleModel, globalTextStyle, opt, isNotNormal, isAttached, true, false); - } // Consider case: - // { - // data: [{ - // value: 12, - // label: { - // rich: { - // // no 'a' here but using parent 'a'. - // } - // } - // }], - // rich: { - // a: { ... } - // } - // } - // TODO TextStyleModel - - - function getRichItemNames(textStyleModel) { - // Use object to remove duplicated names. - var richItemNameMap; - - while (textStyleModel && textStyleModel !== textStyleModel.ecModel) { - var rich = (textStyleModel.option || EMPTY_OBJ).rich; - - if (rich) { - richItemNameMap = richItemNameMap || {}; - var richKeys = keys(rich); - - for (var i = 0; i < richKeys.length; i++) { - var richKey = richKeys[i]; - richItemNameMap[richKey] = 1; - } - } - - textStyleModel = textStyleModel.parentModel; - } - - return richItemNameMap; - } - - var TEXT_PROPS_WITH_GLOBAL = ['fontStyle', 'fontWeight', 'fontSize', 'fontFamily', 'textShadowColor', 'textShadowBlur', 'textShadowOffsetX', 'textShadowOffsetY']; - var TEXT_PROPS_SELF = ['align', 'lineHeight', 'width', 'height', 'tag', 'verticalAlign']; - var TEXT_PROPS_BOX = ['padding', 'borderWidth', 'borderRadius', 'borderDashOffset', 'backgroundColor', 'borderColor', 'shadowColor', 'shadowBlur', 'shadowOffsetX', 'shadowOffsetY']; - - function setTokenTextStyle(textStyle, textStyleModel, globalTextStyle, opt, isNotNormal, isAttached, isBlock, inRich) { - // In merge mode, default value should not be given. - globalTextStyle = !isNotNormal && globalTextStyle || EMPTY_OBJ; - var inheritColor = opt && opt.inheritColor; - var fillColor = textStyleModel.getShallow('color'); - var strokeColor = textStyleModel.getShallow('textBorderColor'); - var opacity = retrieve2(textStyleModel.getShallow('opacity'), globalTextStyle.opacity); - - if (fillColor === 'inherit' || fillColor === 'auto') { - if ("development" !== 'production') { - if (fillColor === 'auto') { - deprecateReplaceLog('color: \'auto\'', 'color: \'inherit\''); - } - } - - if (inheritColor) { - fillColor = inheritColor; - } else { - fillColor = null; - } - } - - if (strokeColor === 'inherit' || strokeColor === 'auto') { - if ("development" !== 'production') { - if (strokeColor === 'auto') { - deprecateReplaceLog('color: \'auto\'', 'color: \'inherit\''); - } - } - - if (inheritColor) { - strokeColor = inheritColor; - } else { - strokeColor = null; - } - } - - if (!isAttached) { - // Only use default global textStyle.color if text is individual. - // Otherwise it will use the strategy of attached text color because text may be on a path. - fillColor = fillColor || globalTextStyle.color; - strokeColor = strokeColor || globalTextStyle.textBorderColor; - } - - if (fillColor != null) { - textStyle.fill = fillColor; - } - - if (strokeColor != null) { - textStyle.stroke = strokeColor; - } - - var textBorderWidth = retrieve2(textStyleModel.getShallow('textBorderWidth'), globalTextStyle.textBorderWidth); - - if (textBorderWidth != null) { - textStyle.lineWidth = textBorderWidth; - } - - var textBorderType = retrieve2(textStyleModel.getShallow('textBorderType'), globalTextStyle.textBorderType); - - if (textBorderType != null) { - textStyle.lineDash = textBorderType; - } - - var textBorderDashOffset = retrieve2(textStyleModel.getShallow('textBorderDashOffset'), globalTextStyle.textBorderDashOffset); - - if (textBorderDashOffset != null) { - textStyle.lineDashOffset = textBorderDashOffset; - } - - if (!isNotNormal && opacity == null && !inRich) { - opacity = opt && opt.defaultOpacity; - } - - if (opacity != null) { - textStyle.opacity = opacity; - } // TODO - - - if (!isNotNormal && !isAttached) { - // Set default finally. - if (textStyle.fill == null && opt.inheritColor) { - textStyle.fill = opt.inheritColor; - } - } // Do not use `getFont` here, because merge should be supported, where - // part of these properties may be changed in emphasis style, and the - // others should remain their original value got from normal style. - - - for (var i = 0; i < TEXT_PROPS_WITH_GLOBAL.length; i++) { - var key = TEXT_PROPS_WITH_GLOBAL[i]; - var val = retrieve2(textStyleModel.getShallow(key), globalTextStyle[key]); - - if (val != null) { - textStyle[key] = val; - } - } - - for (var i = 0; i < TEXT_PROPS_SELF.length; i++) { - var key = TEXT_PROPS_SELF[i]; - var val = textStyleModel.getShallow(key); - - if (val != null) { - textStyle[key] = val; - } - } - - if (textStyle.verticalAlign == null) { - var baseline = textStyleModel.getShallow('baseline'); - - if (baseline != null) { - textStyle.verticalAlign = baseline; - } - } - - if (!isBlock || !opt.disableBox) { - for (var i = 0; i < TEXT_PROPS_BOX.length; i++) { - var key = TEXT_PROPS_BOX[i]; - var val = textStyleModel.getShallow(key); - - if (val != null) { - textStyle[key] = val; - } - } - - var borderType = textStyleModel.getShallow('borderType'); - - if (borderType != null) { - textStyle.borderDash = borderType; - } - - if ((textStyle.backgroundColor === 'auto' || textStyle.backgroundColor === 'inherit') && inheritColor) { - if ("development" !== 'production') { - if (textStyle.backgroundColor === 'auto') { - deprecateReplaceLog('backgroundColor: \'auto\'', 'backgroundColor: \'inherit\''); - } - } - - textStyle.backgroundColor = inheritColor; - } - - if ((textStyle.borderColor === 'auto' || textStyle.borderColor === 'inherit') && inheritColor) { - if ("development" !== 'production') { - if (textStyle.borderColor === 'auto') { - deprecateReplaceLog('borderColor: \'auto\'', 'borderColor: \'inherit\''); - } - } - - textStyle.borderColor = inheritColor; - } - } - } - - function getFont(opt, ecModel) { - var gTextStyleModel = ecModel && ecModel.getModel('textStyle'); - return trim([// FIXME in node-canvas fontWeight is before fontStyle - opt.fontStyle || gTextStyleModel && gTextStyleModel.getShallow('fontStyle') || '', opt.fontWeight || gTextStyleModel && gTextStyleModel.getShallow('fontWeight') || '', (opt.fontSize || gTextStyleModel && gTextStyleModel.getShallow('fontSize') || 12) + 'px', opt.fontFamily || gTextStyleModel && gTextStyleModel.getShallow('fontFamily') || 'sans-serif'].join(' ')); - } - var labelInner = makeInner(); - function setLabelValueAnimation(label, labelStatesModels, value, getDefaultText) { - if (!label) { - return; - } - - var obj = labelInner(label); - obj.prevValue = obj.value; - obj.value = value; - var normalLabelModel = labelStatesModels.normal; - obj.valueAnimation = normalLabelModel.get('valueAnimation'); - - if (obj.valueAnimation) { - obj.precision = normalLabelModel.get('precision'); - obj.defaultInterpolatedText = getDefaultText; - obj.statesModels = labelStatesModels; - } - } - function animateLabelValue(textEl, dataIndex, data, animatableModel, labelFetcher) { - var labelInnerStore = labelInner(textEl); - - if (!labelInnerStore.valueAnimation || labelInnerStore.prevValue === labelInnerStore.value) { - // Value not changed, no new label animation - return; - } - - var defaultInterpolatedText = labelInnerStore.defaultInterpolatedText; // Consider the case that being animating, do not use the `obj.value`, - // Otherwise it will jump to the `obj.value` when this new animation started. - - var currValue = retrieve2(labelInnerStore.interpolatedValue, labelInnerStore.prevValue); - var targetValue = labelInnerStore.value; - - function during(percent) { - var interpolated = interpolateRawValues(data, labelInnerStore.precision, currValue, targetValue, percent); - labelInnerStore.interpolatedValue = percent === 1 ? null : interpolated; - var labelText = getLabelText({ - labelDataIndex: dataIndex, - labelFetcher: labelFetcher, - defaultText: defaultInterpolatedText ? defaultInterpolatedText(interpolated) : interpolated + '' - }, labelInnerStore.statesModels, interpolated); - setLabelText(textEl, labelText); - } - - textEl.percent = 0; - (labelInnerStore.prevValue == null ? initProps : updateProps)(textEl, { - // percent is used to prevent animation from being aborted #15916 - percent: 1 - }, animatableModel, dataIndex, null, during); - } - - var PATH_COLOR = ['textStyle', 'color']; - var textStyleParams = ['fontStyle', 'fontWeight', 'fontSize', 'fontFamily', 'padding', 'lineHeight', 'rich', 'width', 'height', 'overflow']; // TODO Performance improvement? - - var tmpText = new ZRText(); - - var TextStyleMixin = - /** @class */ - function () { - function TextStyleMixin() {} - /** - * Get color property or get color from option.textStyle.color - */ - // TODO Callback - - - TextStyleMixin.prototype.getTextColor = function (isEmphasis) { - var ecModel = this.ecModel; - return this.getShallow('color') || (!isEmphasis && ecModel ? ecModel.get(PATH_COLOR) : null); - }; - /** - * Create font string from fontStyle, fontWeight, fontSize, fontFamily - * @return {string} - */ - - - TextStyleMixin.prototype.getFont = function () { - return getFont({ - fontStyle: this.getShallow('fontStyle'), - fontWeight: this.getShallow('fontWeight'), - fontSize: this.getShallow('fontSize'), - fontFamily: this.getShallow('fontFamily') - }, this.ecModel); - }; - - TextStyleMixin.prototype.getTextRect = function (text) { - var style = { - text: text, - verticalAlign: this.getShallow('verticalAlign') || this.getShallow('baseline') - }; - - for (var i = 0; i < textStyleParams.length; i++) { - style[textStyleParams[i]] = this.getShallow(textStyleParams[i]); - } - - tmpText.useStyle(style); - tmpText.update(); - return tmpText.getBoundingRect(); - }; - - return TextStyleMixin; - }(); - - var LINE_STYLE_KEY_MAP = [['lineWidth', 'width'], ['stroke', 'color'], ['opacity'], ['shadowBlur'], ['shadowOffsetX'], ['shadowOffsetY'], ['shadowColor'], ['lineDash', 'type'], ['lineDashOffset', 'dashOffset'], ['lineCap', 'cap'], ['lineJoin', 'join'], ['miterLimit'] // Option decal is in `DecalObject` but style.decal is in `PatternObject`. - // So do not transfer decal directly. - ]; - var getLineStyle = makeStyleMapper(LINE_STYLE_KEY_MAP); - - var LineStyleMixin = - /** @class */ - function () { - function LineStyleMixin() {} - - LineStyleMixin.prototype.getLineStyle = function (excludes) { - return getLineStyle(this, excludes); - }; - - return LineStyleMixin; - }(); - - var ITEM_STYLE_KEY_MAP = [['fill', 'color'], ['stroke', 'borderColor'], ['lineWidth', 'borderWidth'], ['opacity'], ['shadowBlur'], ['shadowOffsetX'], ['shadowOffsetY'], ['shadowColor'], ['lineDash', 'borderType'], ['lineDashOffset', 'borderDashOffset'], ['lineCap', 'borderCap'], ['lineJoin', 'borderJoin'], ['miterLimit', 'borderMiterLimit'] // Option decal is in `DecalObject` but style.decal is in `PatternObject`. - // So do not transfer decal directly. - ]; - var getItemStyle = makeStyleMapper(ITEM_STYLE_KEY_MAP); - - var ItemStyleMixin = - /** @class */ - function () { - function ItemStyleMixin() {} - - ItemStyleMixin.prototype.getItemStyle = function (excludes, includes) { - return getItemStyle(this, excludes, includes); - }; - - return ItemStyleMixin; - }(); - - var Model = - /** @class */ - function () { - function Model(option, parentModel, ecModel) { - this.parentModel = parentModel; - this.ecModel = ecModel; - this.option = option; // Simple optimization - // if (this.init) { - // if (arguments.length <= 4) { - // this.init(option, parentModel, ecModel, extraOpt); - // } - // else { - // this.init.apply(this, arguments); - // } - // } - } - - Model.prototype.init = function (option, parentModel, ecModel) { - var rest = []; - - for (var _i = 3; _i < arguments.length; _i++) { - rest[_i - 3] = arguments[_i]; - } - }; - /** - * Merge the input option to me. - */ - - - Model.prototype.mergeOption = function (option, ecModel) { - merge(this.option, option, true); - }; // `path` can be 'a.b.c', so the return value type have to be `ModelOption` - // TODO: TYPE strict key check? - // get(path: string | string[], ignoreParent?: boolean): ModelOption; - - - Model.prototype.get = function (path, ignoreParent) { - if (path == null) { - return this.option; - } - - return this._doGet(this.parsePath(path), !ignoreParent && this.parentModel); - }; - - Model.prototype.getShallow = function (key, ignoreParent) { - var option = this.option; - var val = option == null ? option : option[key]; - - if (val == null && !ignoreParent) { - var parentModel = this.parentModel; - - if (parentModel) { - // FIXME:TS do not know how to make it works - val = parentModel.getShallow(key); - } - } - - return val; - }; // `path` can be 'a.b.c', so the return value type have to be `Model<ModelOption>` - // getModel(path: string | string[], parentModel?: Model): Model; - // TODO 'a.b.c' is deprecated - - - Model.prototype.getModel = function (path, parentModel) { - var hasPath = path != null; - var pathFinal = hasPath ? this.parsePath(path) : null; - var obj = hasPath ? this._doGet(pathFinal) : this.option; - parentModel = parentModel || this.parentModel && this.parentModel.getModel(this.resolveParentPath(pathFinal)); - return new Model(obj, parentModel, this.ecModel); - }; - /** - * If model has option - */ - - - Model.prototype.isEmpty = function () { - return this.option == null; - }; - - Model.prototype.restoreData = function () {}; // Pending - - - Model.prototype.clone = function () { - var Ctor = this.constructor; - return new Ctor(clone(this.option)); - }; // setReadOnly(properties): void { - // clazzUtil.setReadOnly(this, properties); - // } - // If path is null/undefined, return null/undefined. - - - Model.prototype.parsePath = function (path) { - if (typeof path === 'string') { - return path.split('.'); - } - - return path; - }; // Resolve path for parent. Perhaps useful when parent use a different property. - // Default to be a identity resolver. - // Can be modified to a different resolver. - - - Model.prototype.resolveParentPath = function (path) { - return path; - }; // FIXME:TS check whether put this method here - - - Model.prototype.isAnimationEnabled = function () { - if (!env.node && this.option) { - if (this.option.animation != null) { - return !!this.option.animation; - } else if (this.parentModel) { - return this.parentModel.isAnimationEnabled(); - } - } - }; - - Model.prototype._doGet = function (pathArr, parentModel) { - var obj = this.option; - - if (!pathArr) { - return obj; - } - - for (var i = 0; i < pathArr.length; i++) { - // Ignore empty - if (!pathArr[i]) { - continue; - } // obj could be number/string/... (like 0) - - - obj = obj && typeof obj === 'object' ? obj[pathArr[i]] : null; - - if (obj == null) { - break; - } - } - - if (obj == null && parentModel) { - obj = parentModel._doGet(this.resolveParentPath(pathArr), parentModel.parentModel); - } - - return obj; - }; - - return Model; - }(); - - enableClassExtend(Model); - enableClassCheck(Model); - mixin(Model, LineStyleMixin); - mixin(Model, ItemStyleMixin); - mixin(Model, AreaStyleMixin); - mixin(Model, TextStyleMixin); - - var base = Math.round(Math.random() * 10); - /** - * @public - * @param {string} type - * @return {string} - */ - - function getUID(type) { - // Considering the case of crossing js context, - // use Math.random to make id as unique as possible. - return [type || '', base++].join('_'); - } - /** - * Implements `SubTypeDefaulterManager` for `target`. - */ - - function enableSubTypeDefaulter(target) { - var subTypeDefaulters = {}; - - target.registerSubTypeDefaulter = function (componentType, defaulter) { - var componentTypeInfo = parseClassType(componentType); - subTypeDefaulters[componentTypeInfo.main] = defaulter; - }; - - target.determineSubType = function (componentType, option) { - var type = option.type; - - if (!type) { - var componentTypeMain = parseClassType(componentType).main; - - if (target.hasSubTypes(componentType) && subTypeDefaulters[componentTypeMain]) { - type = subTypeDefaulters[componentTypeMain](option); - } - } - - return type; - }; - } - /** - * Implements `TopologicalTravelable<any>` for `entity`. - * - * Topological travel on Activity Network (Activity On Vertices). - * Dependencies is defined in Model.prototype.dependencies, like ['xAxis', 'yAxis']. - * If 'xAxis' or 'yAxis' is absent in componentTypeList, just ignore it in topology. - * If there is circular dependencey, Error will be thrown. - */ - - function enableTopologicalTravel(entity, dependencyGetter) { - /** - * @param targetNameList Target Component type list. - * Can be ['aa', 'bb', 'aa.xx'] - * @param fullNameList By which we can build dependency graph. - * @param callback Params: componentType, dependencies. - * @param context Scope of callback. - */ - entity.topologicalTravel = function (targetNameList, fullNameList, callback, context) { - if (!targetNameList.length) { - return; - } - - var result = makeDepndencyGraph(fullNameList); - var graph = result.graph; - var noEntryList = result.noEntryList; - var targetNameSet = {}; - each(targetNameList, function (name) { - targetNameSet[name] = true; - }); - - while (noEntryList.length) { - var currComponentType = noEntryList.pop(); - var currVertex = graph[currComponentType]; - var isInTargetNameSet = !!targetNameSet[currComponentType]; - - if (isInTargetNameSet) { - callback.call(context, currComponentType, currVertex.originalDeps.slice()); - delete targetNameSet[currComponentType]; - } - - each(currVertex.successor, isInTargetNameSet ? removeEdgeAndAdd : removeEdge); - } - - each(targetNameSet, function () { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = makePrintable('Circular dependency may exists: ', targetNameSet, targetNameList, fullNameList); - } - - throw new Error(errMsg); - }); - - function removeEdge(succComponentType) { - graph[succComponentType].entryCount--; - - if (graph[succComponentType].entryCount === 0) { - noEntryList.push(succComponentType); - } - } // Consider this case: legend depends on series, and we call - // chart.setOption({series: [...]}), where only series is in option. - // If we do not have 'removeEdgeAndAdd', legendModel.mergeOption will - // not be called, but only sereis.mergeOption is called. Thus legend - // have no chance to update its local record about series (like which - // name of series is available in legend). - - - function removeEdgeAndAdd(succComponentType) { - targetNameSet[succComponentType] = true; - removeEdge(succComponentType); - } - }; - - function makeDepndencyGraph(fullNameList) { - var graph = {}; - var noEntryList = []; - each(fullNameList, function (name) { - var thisItem = createDependencyGraphItem(graph, name); - var originalDeps = thisItem.originalDeps = dependencyGetter(name); - var availableDeps = getAvailableDependencies(originalDeps, fullNameList); - thisItem.entryCount = availableDeps.length; - - if (thisItem.entryCount === 0) { - noEntryList.push(name); - } - - each(availableDeps, function (dependentName) { - if (indexOf(thisItem.predecessor, dependentName) < 0) { - thisItem.predecessor.push(dependentName); - } - - var thatItem = createDependencyGraphItem(graph, dependentName); - - if (indexOf(thatItem.successor, dependentName) < 0) { - thatItem.successor.push(name); - } - }); - }); - return { - graph: graph, - noEntryList: noEntryList - }; - } - - function createDependencyGraphItem(graph, name) { - if (!graph[name]) { - graph[name] = { - predecessor: [], - successor: [] - }; - } - - return graph[name]; - } - - function getAvailableDependencies(originalDeps, fullNameList) { - var availableDeps = []; - each(originalDeps, function (dep) { - indexOf(fullNameList, dep) >= 0 && availableDeps.push(dep); - }); - return availableDeps; - } - } - function inheritDefaultOption(superOption, subOption) { - // See also `model/Component.ts#getDefaultOption` - return merge(merge({}, superOption, true), subOption, true); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - /** - * Language: English. - */ - var langEN = { - time: { - month: ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December'], - monthAbbr: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'], - dayOfWeek: ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'], - dayOfWeekAbbr: ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'] - }, - legend: { - selector: { - all: 'All', - inverse: 'Inv' - } - }, - toolbox: { - brush: { - title: { - rect: 'Box Select', - polygon: 'Lasso Select', - lineX: 'Horizontally Select', - lineY: 'Vertically Select', - keep: 'Keep Selections', - clear: 'Clear Selections' - } - }, - dataView: { - title: 'Data View', - lang: ['Data View', 'Close', 'Refresh'] - }, - dataZoom: { - title: { - zoom: 'Zoom', - back: 'Zoom Reset' - } - }, - magicType: { - title: { - line: 'Switch to Line Chart', - bar: 'Switch to Bar Chart', - stack: 'Stack', - tiled: 'Tile' - } - }, - restore: { - title: 'Restore' - }, - saveAsImage: { - title: 'Save as Image', - lang: ['Right Click to Save Image'] - } - }, - series: { - typeNames: { - pie: 'Pie chart', - bar: 'Bar chart', - line: 'Line chart', - scatter: 'Scatter plot', - effectScatter: 'Ripple scatter plot', - radar: 'Radar chart', - tree: 'Tree', - treemap: 'Treemap', - boxplot: 'Boxplot', - candlestick: 'Candlestick', - k: 'K line chart', - heatmap: 'Heat map', - map: 'Map', - parallel: 'Parallel coordinate map', - lines: 'Line graph', - graph: 'Relationship graph', - sankey: 'Sankey diagram', - funnel: 'Funnel chart', - gauge: 'Gauge', - pictorialBar: 'Pictorial bar', - themeRiver: 'Theme River Map', - sunburst: 'Sunburst' - } - }, - aria: { - general: { - withTitle: 'This is a chart about "{title}"', - withoutTitle: 'This is a chart' - }, - series: { - single: { - prefix: '', - withName: ' with type {seriesType} named {seriesName}.', - withoutName: ' with type {seriesType}.' - }, - multiple: { - prefix: '. It consists of {seriesCount} series count.', - withName: ' The {seriesId} series is a {seriesType} representing {seriesName}.', - withoutName: ' The {seriesId} series is a {seriesType}.', - separator: { - middle: '', - end: '' - } - } - }, - data: { - allData: 'The data is as follows: ', - partialData: 'The first {displayCnt} items are: ', - withName: 'the data for {name} is {value}', - withoutName: '{value}', - separator: { - middle: ', ', - end: '. ' - } - } - } - }; - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var langZH = { - time: { - month: ['一月', '二月', '三月', '四月', '五月', '六月', '七月', '八月', '九月', '十月', '十一月', '十二月'], - monthAbbr: ['1月', '2月', '3月', '4月', '5月', '6月', '7月', '8月', '9月', '10月', '11月', '12月'], - dayOfWeek: ['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六'], - dayOfWeekAbbr: ['日', '一', '二', '三', '四', '五', '六'] - }, - legend: { - selector: { - all: '全选', - inverse: '反选' - } - }, - toolbox: { - brush: { - title: { - rect: '矩形选择', - polygon: '圈选', - lineX: '横向选择', - lineY: '纵向选择', - keep: '保持选择', - clear: '清除选择' - } - }, - dataView: { - title: '数据视图', - lang: ['数据视图', '关闭', '刷新'] - }, - dataZoom: { - title: { - zoom: '区域缩放', - back: '区域缩放还原' - } - }, - magicType: { - title: { - line: '切换为折线图', - bar: '切换为柱状图', - stack: '切换为堆叠', - tiled: '切换为平铺' - } - }, - restore: { - title: '还原' - }, - saveAsImage: { - title: '保存为图片', - lang: ['右键另存为图片'] - } - }, - series: { - typeNames: { - pie: '饼图', - bar: '柱状图', - line: '折线图', - scatter: '散点图', - effectScatter: '涟漪散点图', - radar: '雷达图', - tree: '树图', - treemap: '矩形树图', - boxplot: '箱型图', - candlestick: 'K线图', - k: 'K线图', - heatmap: '热力图', - map: '地图', - parallel: '平行坐标图', - lines: '线图', - graph: '关系图', - sankey: '桑基图', - funnel: '漏斗图', - gauge: '仪表盘图', - pictorialBar: '象形柱图', - themeRiver: '主题河流图', - sunburst: '旭日图' - } - }, - aria: { - general: { - withTitle: '这是一个关于“{title}”的图表。', - withoutTitle: '这是一个图表,' - }, - series: { - single: { - prefix: '', - withName: '图表类型是{seriesType},表示{seriesName}。', - withoutName: '图表类型是{seriesType}。' - }, - multiple: { - prefix: '它由{seriesCount}个图表系列组成。', - withName: '第{seriesId}个系列是一个表示{seriesName}的{seriesType},', - withoutName: '第{seriesId}个系列是一个{seriesType},', - separator: { - middle: ';', - end: '。' - } - } - }, - data: { - allData: '其数据是——', - partialData: '其中,前{displayCnt}项是——', - withName: '{name}的数据是{value}', - withoutName: '{value}', - separator: { - middle: ',', - end: '' - } - } - } - }; - - var LOCALE_ZH = 'ZH'; - var LOCALE_EN = 'EN'; - var DEFAULT_LOCALE = LOCALE_EN; - var localeStorage = {}; - var localeModels = {}; - var SYSTEM_LANG = !env.domSupported ? DEFAULT_LOCALE : function () { - var langStr = ( - /* eslint-disable-next-line */ - document.documentElement.lang || navigator.language || navigator.browserLanguage).toUpperCase(); - return langStr.indexOf(LOCALE_ZH) > -1 ? LOCALE_ZH : DEFAULT_LOCALE; - }(); - function registerLocale(locale, localeObj) { - locale = locale.toUpperCase(); - localeModels[locale] = new Model(localeObj); - localeStorage[locale] = localeObj; - } // export function getLocale(locale: string) { - // return localeStorage[locale]; - // } - - function createLocaleObject(locale) { - if (isString(locale)) { - var localeObj = localeStorage[locale.toUpperCase()] || {}; - - if (locale === LOCALE_ZH || locale === LOCALE_EN) { - return clone(localeObj); - } else { - return merge(clone(localeObj), clone(localeStorage[DEFAULT_LOCALE]), false); - } - } else { - return merge(clone(locale), clone(localeStorage[DEFAULT_LOCALE]), false); - } - } - function getLocaleModel(lang) { - return localeModels[lang]; - } - function getDefaultLocaleModel() { - return localeModels[DEFAULT_LOCALE]; - } // Default locale - - registerLocale(LOCALE_EN, langEN); - registerLocale(LOCALE_ZH, langZH); - - var ONE_SECOND = 1000; - var ONE_MINUTE = ONE_SECOND * 60; - var ONE_HOUR = ONE_MINUTE * 60; - var ONE_DAY = ONE_HOUR * 24; - var ONE_YEAR = ONE_DAY * 365; - var defaultLeveledFormatter = { - year: '{yyyy}', - month: '{MMM}', - day: '{d}', - hour: '{HH}:{mm}', - minute: '{HH}:{mm}', - second: '{HH}:{mm}:{ss}', - millisecond: '{HH}:{mm}:{ss} {SSS}', - none: '{yyyy}-{MM}-{dd} {HH}:{mm}:{ss} {SSS}' - }; - var fullDayFormatter = '{yyyy}-{MM}-{dd}'; - var fullLeveledFormatter = { - year: '{yyyy}', - month: '{yyyy}-{MM}', - day: fullDayFormatter, - hour: fullDayFormatter + ' ' + defaultLeveledFormatter.hour, - minute: fullDayFormatter + ' ' + defaultLeveledFormatter.minute, - second: fullDayFormatter + ' ' + defaultLeveledFormatter.second, - millisecond: defaultLeveledFormatter.none - }; - var primaryTimeUnits = ['year', 'month', 'day', 'hour', 'minute', 'second', 'millisecond']; - var timeUnits = ['year', 'half-year', 'quarter', 'month', 'week', 'half-week', 'day', 'half-day', 'quarter-day', 'hour', 'minute', 'second', 'millisecond']; - function pad(str, len) { - str += ''; - return '0000'.substr(0, len - str.length) + str; - } - function getPrimaryTimeUnit(timeUnit) { - switch (timeUnit) { - case 'half-year': - case 'quarter': - return 'month'; - - case 'week': - case 'half-week': - return 'day'; - - case 'half-day': - case 'quarter-day': - return 'hour'; - - default: - // year, minutes, second, milliseconds - return timeUnit; - } - } - function isPrimaryTimeUnit(timeUnit) { - return timeUnit === getPrimaryTimeUnit(timeUnit); - } - function getDefaultFormatPrecisionOfInterval(timeUnit) { - switch (timeUnit) { - case 'year': - case 'month': - return 'day'; - - case 'millisecond': - return 'millisecond'; - - default: - // Also for day, hour, minute, second - return 'second'; - } - } - function format( // Note: The result based on `isUTC` are totally different, which can not be just simply - // substituted by the result without `isUTC`. So we make the param `isUTC` mandatory. - time, template, isUTC, lang) { - var date = parseDate(time); - var y = date[fullYearGetterName(isUTC)](); - var M = date[monthGetterName(isUTC)]() + 1; - var q = Math.floor((M - 1) / 3) + 1; - var d = date[dateGetterName(isUTC)](); - var e = date['get' + (isUTC ? 'UTC' : '') + 'Day'](); - var H = date[hoursGetterName(isUTC)](); - var h = (H - 1) % 12 + 1; - var m = date[minutesGetterName(isUTC)](); - var s = date[secondsGetterName(isUTC)](); - var S = date[millisecondsGetterName(isUTC)](); - var localeModel = lang instanceof Model ? lang : getLocaleModel(lang || SYSTEM_LANG) || getDefaultLocaleModel(); - var timeModel = localeModel.getModel('time'); - var month = timeModel.get('month'); - var monthAbbr = timeModel.get('monthAbbr'); - var dayOfWeek = timeModel.get('dayOfWeek'); - var dayOfWeekAbbr = timeModel.get('dayOfWeekAbbr'); - return (template || '').replace(/{yyyy}/g, y + '').replace(/{yy}/g, y % 100 + '').replace(/{Q}/g, q + '').replace(/{MMMM}/g, month[M - 1]).replace(/{MMM}/g, monthAbbr[M - 1]).replace(/{MM}/g, pad(M, 2)).replace(/{M}/g, M + '').replace(/{dd}/g, pad(d, 2)).replace(/{d}/g, d + '').replace(/{eeee}/g, dayOfWeek[e]).replace(/{ee}/g, dayOfWeekAbbr[e]).replace(/{e}/g, e + '').replace(/{HH}/g, pad(H, 2)).replace(/{H}/g, H + '').replace(/{hh}/g, pad(h + '', 2)).replace(/{h}/g, h + '').replace(/{mm}/g, pad(m, 2)).replace(/{m}/g, m + '').replace(/{ss}/g, pad(s, 2)).replace(/{s}/g, s + '').replace(/{SSS}/g, pad(S, 3)).replace(/{S}/g, S + ''); - } - function leveledFormat(tick, idx, formatter, lang, isUTC) { - var template = null; - - if (isString(formatter)) { - // Single formatter for all units at all levels - template = formatter; - } else if (isFunction(formatter)) { - // Callback formatter - template = formatter(tick.value, idx, { - level: tick.level - }); - } else { - var defaults$1 = extend({}, defaultLeveledFormatter); - - if (tick.level > 0) { - for (var i = 0; i < primaryTimeUnits.length; ++i) { - defaults$1[primaryTimeUnits[i]] = "{primary|" + defaults$1[primaryTimeUnits[i]] + "}"; - } - } - - var mergedFormatter = formatter ? formatter.inherit === false ? formatter // Use formatter with bigger units - : defaults(formatter, defaults$1) : defaults$1; - var unit = getUnitFromValue(tick.value, isUTC); - - if (mergedFormatter[unit]) { - template = mergedFormatter[unit]; - } else if (mergedFormatter.inherit) { - // Unit formatter is not defined and should inherit from bigger units - var targetId = timeUnits.indexOf(unit); - - for (var i = targetId - 1; i >= 0; --i) { - if (mergedFormatter[unit]) { - template = mergedFormatter[unit]; - break; - } - } - - template = template || defaults$1.none; - } - - if (isArray(template)) { - var levelId = tick.level == null ? 0 : tick.level >= 0 ? tick.level : template.length + tick.level; - levelId = Math.min(levelId, template.length - 1); - template = template[levelId]; - } - } - - return format(new Date(tick.value), template, isUTC, lang); - } - function getUnitFromValue(value, isUTC) { - var date = parseDate(value); - var M = date[monthGetterName(isUTC)]() + 1; - var d = date[dateGetterName(isUTC)](); - var h = date[hoursGetterName(isUTC)](); - var m = date[minutesGetterName(isUTC)](); - var s = date[secondsGetterName(isUTC)](); - var S = date[millisecondsGetterName(isUTC)](); - var isSecond = S === 0; - var isMinute = isSecond && s === 0; - var isHour = isMinute && m === 0; - var isDay = isHour && h === 0; - var isMonth = isDay && d === 1; - var isYear = isMonth && M === 1; - - if (isYear) { - return 'year'; - } else if (isMonth) { - return 'month'; - } else if (isDay) { - return 'day'; - } else if (isHour) { - return 'hour'; - } else if (isMinute) { - return 'minute'; - } else if (isSecond) { - return 'second'; - } else { - return 'millisecond'; - } - } - function getUnitValue(value, unit, isUTC) { - var date = isNumber(value) ? parseDate(value) : value; - unit = unit || getUnitFromValue(value, isUTC); - - switch (unit) { - case 'year': - return date[fullYearGetterName(isUTC)](); - - case 'half-year': - return date[monthGetterName(isUTC)]() >= 6 ? 1 : 0; - - case 'quarter': - return Math.floor((date[monthGetterName(isUTC)]() + 1) / 4); - - case 'month': - return date[monthGetterName(isUTC)](); - - case 'day': - return date[dateGetterName(isUTC)](); - - case 'half-day': - return date[hoursGetterName(isUTC)]() / 24; - - case 'hour': - return date[hoursGetterName(isUTC)](); - - case 'minute': - return date[minutesGetterName(isUTC)](); - - case 'second': - return date[secondsGetterName(isUTC)](); - - case 'millisecond': - return date[millisecondsGetterName(isUTC)](); - } - } - function fullYearGetterName(isUTC) { - return isUTC ? 'getUTCFullYear' : 'getFullYear'; - } - function monthGetterName(isUTC) { - return isUTC ? 'getUTCMonth' : 'getMonth'; - } - function dateGetterName(isUTC) { - return isUTC ? 'getUTCDate' : 'getDate'; - } - function hoursGetterName(isUTC) { - return isUTC ? 'getUTCHours' : 'getHours'; - } - function minutesGetterName(isUTC) { - return isUTC ? 'getUTCMinutes' : 'getMinutes'; - } - function secondsGetterName(isUTC) { - return isUTC ? 'getUTCSeconds' : 'getSeconds'; - } - function millisecondsGetterName(isUTC) { - return isUTC ? 'getUTCMilliseconds' : 'getMilliseconds'; - } - function fullYearSetterName(isUTC) { - return isUTC ? 'setUTCFullYear' : 'setFullYear'; - } - function monthSetterName(isUTC) { - return isUTC ? 'setUTCMonth' : 'setMonth'; - } - function dateSetterName(isUTC) { - return isUTC ? 'setUTCDate' : 'setDate'; - } - function hoursSetterName(isUTC) { - return isUTC ? 'setUTCHours' : 'setHours'; - } - function minutesSetterName(isUTC) { - return isUTC ? 'setUTCMinutes' : 'setMinutes'; - } - function secondsSetterName(isUTC) { - return isUTC ? 'setUTCSeconds' : 'setSeconds'; - } - function millisecondsSetterName(isUTC) { - return isUTC ? 'setUTCMilliseconds' : 'setMilliseconds'; - } - - function getTextRect(text, font, align, verticalAlign, padding, rich, truncate, lineHeight) { - var textEl = new ZRText({ - style: { - text: text, - font: font, - align: align, - verticalAlign: verticalAlign, - padding: padding, - rich: rich, - overflow: truncate ? 'truncate' : null, - lineHeight: lineHeight - } - }); - return textEl.getBoundingRect(); - } - - /** - * Add a comma each three digit. - */ - - function addCommas(x) { - if (!isNumeric(x)) { - return isString(x) ? x : '-'; - } - - var parts = (x + '').split('.'); - return parts[0].replace(/(\d{1,3})(?=(?:\d{3})+(?!\d))/g, '$1,') + (parts.length > 1 ? '.' + parts[1] : ''); - } - function toCamelCase(str, upperCaseFirst) { - str = (str || '').toLowerCase().replace(/-(.)/g, function (match, group1) { - return group1.toUpperCase(); - }); - - if (upperCaseFirst && str) { - str = str.charAt(0).toUpperCase() + str.slice(1); - } - - return str; - } - var normalizeCssArray$1 = normalizeCssArray; - /** - * Make value user readable for tooltip and label. - * "User readable": - * Try to not print programmer-specific text like NaN, Infinity, null, undefined. - * Avoid to display an empty string, which users can not recognize there is - * a value and it might look like a bug. - */ - - function makeValueReadable(value, valueType, useUTC) { - var USER_READABLE_DEFUALT_TIME_PATTERN = '{yyyy}-{MM}-{dd} {HH}:{mm}:{ss}'; - - function stringToUserReadable(str) { - return str && trim(str) ? str : '-'; - } - - function isNumberUserReadable(num) { - return !!(num != null && !isNaN(num) && isFinite(num)); - } - - var isTypeTime = valueType === 'time'; - var isValueDate = value instanceof Date; - - if (isTypeTime || isValueDate) { - var date = isTypeTime ? parseDate(value) : value; - - if (!isNaN(+date)) { - return format(date, USER_READABLE_DEFUALT_TIME_PATTERN, useUTC); - } else if (isValueDate) { - return '-'; - } // In other cases, continue to try to display the value in the following code. - - } - - if (valueType === 'ordinal') { - return isStringSafe(value) ? stringToUserReadable(value) : isNumber(value) ? isNumberUserReadable(value) ? value + '' : '-' : '-'; - } // By default. - - - var numericResult = numericToNumber(value); - return isNumberUserReadable(numericResult) ? addCommas(numericResult) : isStringSafe(value) ? stringToUserReadable(value) : typeof value === 'boolean' ? value + '' : '-'; - } - var TPL_VAR_ALIAS = ['a', 'b', 'c', 'd', 'e', 'f', 'g']; - - var wrapVar = function (varName, seriesIdx) { - return '{' + varName + (seriesIdx == null ? '' : seriesIdx) + '}'; - }; - /** - * Template formatter - * @param {Array.<Object>|Object} paramsList - */ - - - function formatTpl(tpl, paramsList, encode) { - if (!isArray(paramsList)) { - paramsList = [paramsList]; - } - - var seriesLen = paramsList.length; - - if (!seriesLen) { - return ''; - } - - var $vars = paramsList[0].$vars || []; - - for (var i = 0; i < $vars.length; i++) { - var alias = TPL_VAR_ALIAS[i]; - tpl = tpl.replace(wrapVar(alias), wrapVar(alias, 0)); - } - - for (var seriesIdx = 0; seriesIdx < seriesLen; seriesIdx++) { - for (var k = 0; k < $vars.length; k++) { - var val = paramsList[seriesIdx][$vars[k]]; - tpl = tpl.replace(wrapVar(TPL_VAR_ALIAS[k], seriesIdx), encode ? encodeHTML(val) : val); - } - } - - return tpl; - } - /** - * simple Template formatter - */ - - function formatTplSimple(tpl, param, encode) { - each(param, function (value, key) { - tpl = tpl.replace('{' + key + '}', encode ? encodeHTML(value) : value); - }); - return tpl; - } - function getTooltipMarker(inOpt, extraCssText) { - var opt = isString(inOpt) ? { - color: inOpt, - extraCssText: extraCssText - } : inOpt || {}; - var color = opt.color; - var type = opt.type; - extraCssText = opt.extraCssText; - var renderMode = opt.renderMode || 'html'; - - if (!color) { - return ''; - } - - if (renderMode === 'html') { - return type === 'subItem' ? '<span style="display:inline-block;vertical-align:middle;margin-right:8px;margin-left:3px;' + 'border-radius:4px;width:4px;height:4px;background-color:' // Only support string - + encodeHTML(color) + ';' + (extraCssText || '') + '"></span>' : '<span style="display:inline-block;margin-right:4px;' + 'border-radius:10px;width:10px;height:10px;background-color:' + encodeHTML(color) + ';' + (extraCssText || '') + '"></span>'; - } else { - // Should better not to auto generate style name by auto-increment number here. - // Because this util is usually called in tooltip formatter, which is probably - // called repeatedly when mouse move and the auto-increment number increases fast. - // Users can make their own style name by theirselves, make it unique and readable. - var markerId = opt.markerId || 'markerX'; - return { - renderMode: renderMode, - content: '{' + markerId + '|} ', - style: type === 'subItem' ? { - width: 4, - height: 4, - borderRadius: 2, - backgroundColor: color - } : { - width: 10, - height: 10, - borderRadius: 5, - backgroundColor: color - } - }; - } - } - /** - * @deprecated Use `time/format` instead. - * ISO Date format - * @param {string} tpl - * @param {number} value - * @param {boolean} [isUTC=false] Default in local time. - * see `module:echarts/scale/Time` - * and `module:echarts/util/number#parseDate`. - * @inner - */ - - function formatTime(tpl, value, isUTC) { - if ("development" !== 'production') { - deprecateReplaceLog('echarts.format.formatTime', 'echarts.time.format'); - } - - if (tpl === 'week' || tpl === 'month' || tpl === 'quarter' || tpl === 'half-year' || tpl === 'year') { - tpl = 'MM-dd\nyyyy'; - } - - var date = parseDate(value); - var getUTC = isUTC ? 'getUTC' : 'get'; - var y = date[getUTC + 'FullYear'](); - var M = date[getUTC + 'Month']() + 1; - var d = date[getUTC + 'Date'](); - var h = date[getUTC + 'Hours'](); - var m = date[getUTC + 'Minutes'](); - var s = date[getUTC + 'Seconds'](); - var S = date[getUTC + 'Milliseconds'](); - tpl = tpl.replace('MM', pad(M, 2)).replace('M', M).replace('yyyy', y).replace('yy', pad(y % 100 + '', 2)).replace('dd', pad(d, 2)).replace('d', d).replace('hh', pad(h, 2)).replace('h', h).replace('mm', pad(m, 2)).replace('m', m).replace('ss', pad(s, 2)).replace('s', s).replace('SSS', pad(S, 3)); - return tpl; - } - /** - * Capital first - * @param {string} str - * @return {string} - */ - - function capitalFirst(str) { - return str ? str.charAt(0).toUpperCase() + str.substr(1) : str; - } - /** - * @return Never be null/undefined. - */ - - function convertToColorString(color, defaultColor) { - defaultColor = defaultColor || 'transparent'; - return isString(color) ? color : isObject(color) ? color.colorStops && (color.colorStops[0] || {}).color || defaultColor : defaultColor; - } - /** - * open new tab - * @param link url - * @param target blank or self - */ - - function windowOpen(link, target) { - /* global window */ - if (target === '_blank' || target === 'blank') { - var blank = window.open(); - blank.opener = null; - blank.location.href = link; - } else { - window.open(link, target); - } - } - - var each$1 = each; - /** - * @public - */ - - var LOCATION_PARAMS = ['left', 'right', 'top', 'bottom', 'width', 'height']; - /** - * @public - */ - - var HV_NAMES = [['width', 'left', 'right'], ['height', 'top', 'bottom']]; - - function boxLayout(orient, group, gap, maxWidth, maxHeight) { - var x = 0; - var y = 0; - - if (maxWidth == null) { - maxWidth = Infinity; - } - - if (maxHeight == null) { - maxHeight = Infinity; - } - - var currentLineMaxSize = 0; - group.eachChild(function (child, idx) { - var rect = child.getBoundingRect(); - var nextChild = group.childAt(idx + 1); - var nextChildRect = nextChild && nextChild.getBoundingRect(); - var nextX; - var nextY; - - if (orient === 'horizontal') { - var moveX = rect.width + (nextChildRect ? -nextChildRect.x + rect.x : 0); - nextX = x + moveX; // Wrap when width exceeds maxWidth or meet a `newline` group - // FIXME compare before adding gap? - - if (nextX > maxWidth || child.newline) { - x = 0; - nextX = moveX; - y += currentLineMaxSize + gap; - currentLineMaxSize = rect.height; - } else { - // FIXME: consider rect.y is not `0`? - currentLineMaxSize = Math.max(currentLineMaxSize, rect.height); - } - } else { - var moveY = rect.height + (nextChildRect ? -nextChildRect.y + rect.y : 0); - nextY = y + moveY; // Wrap when width exceeds maxHeight or meet a `newline` group - - if (nextY > maxHeight || child.newline) { - x += currentLineMaxSize + gap; - y = 0; - nextY = moveY; - currentLineMaxSize = rect.width; - } else { - currentLineMaxSize = Math.max(currentLineMaxSize, rect.width); - } - } - - if (child.newline) { - return; - } - - child.x = x; - child.y = y; - child.markRedraw(); - orient === 'horizontal' ? x = nextX + gap : y = nextY + gap; - }); - } - /** - * VBox or HBox layouting - * @param {string} orient - * @param {module:zrender/graphic/Group} group - * @param {number} gap - * @param {number} [width=Infinity] - * @param {number} [height=Infinity] - */ - - - var box = boxLayout; - /** - * VBox layouting - * @param {module:zrender/graphic/Group} group - * @param {number} gap - * @param {number} [width=Infinity] - * @param {number} [height=Infinity] - */ - - var vbox = curry(boxLayout, 'vertical'); - /** - * HBox layouting - * @param {module:zrender/graphic/Group} group - * @param {number} gap - * @param {number} [width=Infinity] - * @param {number} [height=Infinity] - */ - - var hbox = curry(boxLayout, 'horizontal'); - /** - * If x or x2 is not specified or 'center' 'left' 'right', - * the width would be as long as possible. - * If y or y2 is not specified or 'middle' 'top' 'bottom', - * the height would be as long as possible. - */ - - function getAvailableSize(positionInfo, containerRect, margin) { - var containerWidth = containerRect.width; - var containerHeight = containerRect.height; - var x = parsePercent$1(positionInfo.left, containerWidth); - var y = parsePercent$1(positionInfo.top, containerHeight); - var x2 = parsePercent$1(positionInfo.right, containerWidth); - var y2 = parsePercent$1(positionInfo.bottom, containerHeight); - (isNaN(x) || isNaN(parseFloat(positionInfo.left))) && (x = 0); - (isNaN(x2) || isNaN(parseFloat(positionInfo.right))) && (x2 = containerWidth); - (isNaN(y) || isNaN(parseFloat(positionInfo.top))) && (y = 0); - (isNaN(y2) || isNaN(parseFloat(positionInfo.bottom))) && (y2 = containerHeight); - margin = normalizeCssArray$1(margin || 0); - return { - width: Math.max(x2 - x - margin[1] - margin[3], 0), - height: Math.max(y2 - y - margin[0] - margin[2], 0) - }; - } - /** - * Parse position info. - */ - - function getLayoutRect(positionInfo, containerRect, margin) { - margin = normalizeCssArray$1(margin || 0); - var containerWidth = containerRect.width; - var containerHeight = containerRect.height; - var left = parsePercent$1(positionInfo.left, containerWidth); - var top = parsePercent$1(positionInfo.top, containerHeight); - var right = parsePercent$1(positionInfo.right, containerWidth); - var bottom = parsePercent$1(positionInfo.bottom, containerHeight); - var width = parsePercent$1(positionInfo.width, containerWidth); - var height = parsePercent$1(positionInfo.height, containerHeight); - var verticalMargin = margin[2] + margin[0]; - var horizontalMargin = margin[1] + margin[3]; - var aspect = positionInfo.aspect; // If width is not specified, calculate width from left and right - - if (isNaN(width)) { - width = containerWidth - right - horizontalMargin - left; - } - - if (isNaN(height)) { - height = containerHeight - bottom - verticalMargin - top; - } - - if (aspect != null) { - // If width and height are not given - // 1. Graph should not exceeds the container - // 2. Aspect must be keeped - // 3. Graph should take the space as more as possible - // FIXME - // Margin is not considered, because there is no case that both - // using margin and aspect so far. - if (isNaN(width) && isNaN(height)) { - if (aspect > containerWidth / containerHeight) { - width = containerWidth * 0.8; - } else { - height = containerHeight * 0.8; - } - } // Calculate width or height with given aspect - - - if (isNaN(width)) { - width = aspect * height; - } - - if (isNaN(height)) { - height = width / aspect; - } - } // If left is not specified, calculate left from right and width - - - if (isNaN(left)) { - left = containerWidth - right - width - horizontalMargin; - } - - if (isNaN(top)) { - top = containerHeight - bottom - height - verticalMargin; - } // Align left and top - - - switch (positionInfo.left || positionInfo.right) { - case 'center': - left = containerWidth / 2 - width / 2 - margin[3]; - break; - - case 'right': - left = containerWidth - width - horizontalMargin; - break; - } - - switch (positionInfo.top || positionInfo.bottom) { - case 'middle': - case 'center': - top = containerHeight / 2 - height / 2 - margin[0]; - break; - - case 'bottom': - top = containerHeight - height - verticalMargin; - break; - } // If something is wrong and left, top, width, height are calculated as NaN - - - left = left || 0; - top = top || 0; - - if (isNaN(width)) { - // Width may be NaN if only one value is given except width - width = containerWidth - horizontalMargin - left - (right || 0); - } - - if (isNaN(height)) { - // Height may be NaN if only one value is given except height - height = containerHeight - verticalMargin - top - (bottom || 0); - } - - var rect = new BoundingRect(left + margin[3], top + margin[0], width, height); - rect.margin = margin; - return rect; - } - /** - * Position a zr element in viewport - * Group position is specified by either - * {left, top}, {right, bottom} - * If all properties exists, right and bottom will be igonred. - * - * Logic: - * 1. Scale (against origin point in parent coord) - * 2. Rotate (against origin point in parent coord) - * 3. Translate (with el.position by this method) - * So this method only fixes the last step 'Translate', which does not affect - * scaling and rotating. - * - * If be called repeatedly with the same input el, the same result will be gotten. - * - * Return true if the layout happened. - * - * @param el Should have `getBoundingRect` method. - * @param positionInfo - * @param positionInfo.left - * @param positionInfo.top - * @param positionInfo.right - * @param positionInfo.bottom - * @param positionInfo.width Only for opt.boundingModel: 'raw' - * @param positionInfo.height Only for opt.boundingModel: 'raw' - * @param containerRect - * @param margin - * @param opt - * @param opt.hv Only horizontal or only vertical. Default to be [1, 1] - * @param opt.boundingMode - * Specify how to calculate boundingRect when locating. - * 'all': Position the boundingRect that is transformed and uioned - * both itself and its descendants. - * This mode simplies confine the elements in the bounding - * of their container (e.g., using 'right: 0'). - * 'raw': Position the boundingRect that is not transformed and only itself. - * This mode is useful when you want a element can overflow its - * container. (Consider a rotated circle needs to be located in a corner.) - * In this mode positionInfo.width/height can only be number. - */ - - function positionElement(el, positionInfo, containerRect, margin, opt, out) { - var h = !opt || !opt.hv || opt.hv[0]; - var v = !opt || !opt.hv || opt.hv[1]; - var boundingMode = opt && opt.boundingMode || 'all'; - out = out || el; - out.x = el.x; - out.y = el.y; - - if (!h && !v) { - return false; - } - - var rect; - - if (boundingMode === 'raw') { - rect = el.type === 'group' ? new BoundingRect(0, 0, +positionInfo.width || 0, +positionInfo.height || 0) : el.getBoundingRect(); - } else { - rect = el.getBoundingRect(); - - if (el.needLocalTransform()) { - var transform = el.getLocalTransform(); // Notice: raw rect may be inner object of el, - // which should not be modified. - - rect = rect.clone(); - rect.applyTransform(transform); - } - } // The real width and height can not be specified but calculated by the given el. - - - var layoutRect = getLayoutRect(defaults({ - width: rect.width, - height: rect.height - }, positionInfo), containerRect, margin); // Because 'tranlate' is the last step in transform - // (see zrender/core/Transformable#getLocalTransform), - // we can just only modify el.position to get final result. - - var dx = h ? layoutRect.x - rect.x : 0; - var dy = v ? layoutRect.y - rect.y : 0; - - if (boundingMode === 'raw') { - out.x = dx; - out.y = dy; - } else { - out.x += dx; - out.y += dy; - } - - if (out === el) { - el.markRedraw(); - } - - return true; - } - /** - * @param option Contains some of the properties in HV_NAMES. - * @param hvIdx 0: horizontal; 1: vertical. - */ - - function sizeCalculable(option, hvIdx) { - return option[HV_NAMES[hvIdx][0]] != null || option[HV_NAMES[hvIdx][1]] != null && option[HV_NAMES[hvIdx][2]] != null; - } - function fetchLayoutMode(ins) { - var layoutMode = ins.layoutMode || ins.constructor.layoutMode; - return isObject(layoutMode) ? layoutMode : layoutMode ? { - type: layoutMode - } : null; - } - /** - * Consider Case: - * When default option has {left: 0, width: 100}, and we set {right: 0} - * through setOption or media query, using normal zrUtil.merge will cause - * {right: 0} does not take effect. - * - * @example - * ComponentModel.extend({ - * init: function () { - * ... - * let inputPositionParams = layout.getLayoutParams(option); - * this.mergeOption(inputPositionParams); - * }, - * mergeOption: function (newOption) { - * newOption && zrUtil.merge(thisOption, newOption, true); - * layout.mergeLayoutParam(thisOption, newOption); - * } - * }); - * - * @param targetOption - * @param newOption - * @param opt - */ - - function mergeLayoutParam(targetOption, newOption, opt) { - var ignoreSize = opt && opt.ignoreSize; - !isArray(ignoreSize) && (ignoreSize = [ignoreSize, ignoreSize]); - var hResult = merge(HV_NAMES[0], 0); - var vResult = merge(HV_NAMES[1], 1); - copy(HV_NAMES[0], targetOption, hResult); - copy(HV_NAMES[1], targetOption, vResult); - - function merge(names, hvIdx) { - var newParams = {}; - var newValueCount = 0; - var merged = {}; - var mergedValueCount = 0; - var enoughParamNumber = 2; - each$1(names, function (name) { - merged[name] = targetOption[name]; - }); - each$1(names, function (name) { - // Consider case: newOption.width is null, which is - // set by user for removing width setting. - hasProp(newOption, name) && (newParams[name] = merged[name] = newOption[name]); - hasValue(newParams, name) && newValueCount++; - hasValue(merged, name) && mergedValueCount++; - }); - - if (ignoreSize[hvIdx]) { - // Only one of left/right is premitted to exist. - if (hasValue(newOption, names[1])) { - merged[names[2]] = null; - } else if (hasValue(newOption, names[2])) { - merged[names[1]] = null; - } - - return merged; - } // Case: newOption: {width: ..., right: ...}, - // or targetOption: {right: ...} and newOption: {width: ...}, - // There is no conflict when merged only has params count - // little than enoughParamNumber. - - - if (mergedValueCount === enoughParamNumber || !newValueCount) { - return merged; - } // Case: newOption: {width: ..., right: ...}, - // Than we can make sure user only want those two, and ignore - // all origin params in targetOption. - else if (newValueCount >= enoughParamNumber) { - return newParams; - } else { - // Chose another param from targetOption by priority. - for (var i = 0; i < names.length; i++) { - var name_1 = names[i]; - - if (!hasProp(newParams, name_1) && hasProp(targetOption, name_1)) { - newParams[name_1] = targetOption[name_1]; - break; - } - } - - return newParams; - } - } - - function hasProp(obj, name) { - return obj.hasOwnProperty(name); - } - - function hasValue(obj, name) { - return obj[name] != null && obj[name] !== 'auto'; - } - - function copy(names, target, source) { - each$1(names, function (name) { - target[name] = source[name]; - }); - } - } - /** - * Retrieve 'left', 'right', 'top', 'bottom', 'width', 'height' from object. - */ - - function getLayoutParams(source) { - return copyLayoutParams({}, source); - } - /** - * Retrieve 'left', 'right', 'top', 'bottom', 'width', 'height' from object. - * @param {Object} source - * @return {Object} Result contains those props. - */ - - function copyLayoutParams(target, source) { - source && target && each$1(LOCATION_PARAMS, function (name) { - source.hasOwnProperty(name) && (target[name] = source[name]); - }); - return target; - } - - var inner = makeInner(); - - var ComponentModel = - /** @class */ - function (_super) { - __extends(ComponentModel, _super); - - function ComponentModel(option, parentModel, ecModel) { - var _this = _super.call(this, option, parentModel, ecModel) || this; - - _this.uid = getUID('ec_cpt_model'); - return _this; - } - - ComponentModel.prototype.init = function (option, parentModel, ecModel) { - this.mergeDefaultAndTheme(option, ecModel); - }; - - ComponentModel.prototype.mergeDefaultAndTheme = function (option, ecModel) { - var layoutMode = fetchLayoutMode(this); - var inputPositionParams = layoutMode ? getLayoutParams(option) : {}; - var themeModel = ecModel.getTheme(); - merge(option, themeModel.get(this.mainType)); - merge(option, this.getDefaultOption()); - - if (layoutMode) { - mergeLayoutParam(option, inputPositionParams, layoutMode); - } - }; - - ComponentModel.prototype.mergeOption = function (option, ecModel) { - merge(this.option, option, true); - var layoutMode = fetchLayoutMode(this); - - if (layoutMode) { - mergeLayoutParam(this.option, option, layoutMode); - } - }; - /** - * Called immediately after `init` or `mergeOption` of this instance called. - */ - - - ComponentModel.prototype.optionUpdated = function (newCptOption, isInit) {}; - /** - * [How to declare defaultOption]: - * - * (A) If using class declaration in typescript (since echarts 5): - * ```ts - * import {ComponentOption} from '../model/option.js'; - * export interface XxxOption extends ComponentOption { - * aaa: number - * } - * export class XxxModel extends Component { - * static type = 'xxx'; - * static defaultOption: XxxOption = { - * aaa: 123 - * } - * } - * Component.registerClass(XxxModel); - * ``` - * ```ts - * import {inheritDefaultOption} from '../util/component.js'; - * import {XxxModel, XxxOption} from './XxxModel.js'; - * export interface XxxSubOption extends XxxOption { - * bbb: number - * } - * class XxxSubModel extends XxxModel { - * static defaultOption: XxxSubOption = inheritDefaultOption(XxxModel.defaultOption, { - * bbb: 456 - * }) - * fn() { - * let opt = this.getDefaultOption(); - * // opt is {aaa: 123, bbb: 456} - * } - * } - * ``` - * - * (B) If using class extend (previous approach in echarts 3 & 4): - * ```js - * let XxxComponent = Component.extend({ - * defaultOption: { - * xx: 123 - * } - * }) - * ``` - * ```js - * let XxxSubComponent = XxxComponent.extend({ - * defaultOption: { - * yy: 456 - * }, - * fn: function () { - * let opt = this.getDefaultOption(); - * // opt is {xx: 123, yy: 456} - * } - * }) - * ``` - */ - - - ComponentModel.prototype.getDefaultOption = function () { - var ctor = this.constructor; // If using class declaration, it is different to travel super class - // in legacy env and auto merge defaultOption. So if using class - // declaration, defaultOption should be merged manually. - - if (!isExtendedClass(ctor)) { - // When using ts class, defaultOption must be declared as static. - return ctor.defaultOption; - } // FIXME: remove this approach? - - - var fields = inner(this); - - if (!fields.defaultOption) { - var optList = []; - var clz = ctor; - - while (clz) { - var opt = clz.prototype.defaultOption; - opt && optList.push(opt); - clz = clz.superClass; - } - - var defaultOption = {}; - - for (var i = optList.length - 1; i >= 0; i--) { - defaultOption = merge(defaultOption, optList[i], true); - } - - fields.defaultOption = defaultOption; - } - - return fields.defaultOption; - }; - /** - * Notice: always force to input param `useDefault` in case that forget to consider it. - * The same behavior as `modelUtil.parseFinder`. - * - * @param useDefault In many cases like series refer axis and axis refer grid, - * If axis index / axis id not specified, use the first target as default. - * In other cases like dataZoom refer axis, if not specified, measn no refer. - */ - - - ComponentModel.prototype.getReferringComponents = function (mainType, opt) { - var indexKey = mainType + 'Index'; - var idKey = mainType + 'Id'; - return queryReferringComponents(this.ecModel, mainType, { - index: this.get(indexKey, true), - id: this.get(idKey, true) - }, opt); - }; - - ComponentModel.prototype.getBoxLayoutParams = function () { - // Consider itself having box layout configs. - var boxLayoutModel = this; - return { - left: boxLayoutModel.get('left'), - top: boxLayoutModel.get('top'), - right: boxLayoutModel.get('right'), - bottom: boxLayoutModel.get('bottom'), - width: boxLayoutModel.get('width'), - height: boxLayoutModel.get('height') - }; - }; - /** - * Get key for zlevel. - * If developers don't configure zlevel. We will assign zlevel to series based on the key. - * For example, lines with trail effect and progressive series will in an individual zlevel. - */ - - - ComponentModel.prototype.getZLevelKey = function () { - return ''; - }; - - ComponentModel.prototype.setZLevel = function (zlevel) { - this.option.zlevel = zlevel; - }; - - ComponentModel.protoInitialize = function () { - var proto = ComponentModel.prototype; - proto.type = 'component'; - proto.id = ''; - proto.name = ''; - proto.mainType = ''; - proto.subType = ''; - proto.componentIndex = 0; - }(); - - return ComponentModel; - }(Model); - - mountExtend(ComponentModel, Model); - enableClassManagement(ComponentModel); - enableSubTypeDefaulter(ComponentModel); - enableTopologicalTravel(ComponentModel, getDependencies); - - function getDependencies(componentType) { - var deps = []; - each(ComponentModel.getClassesByMainType(componentType), function (clz) { - deps = deps.concat(clz.dependencies || clz.prototype.dependencies || []); - }); // Ensure main type. - - deps = map(deps, function (type) { - return parseClassType(type).main; - }); // Hack dataset for convenience. - - if (componentType !== 'dataset' && indexOf(deps, 'dataset') <= 0) { - deps.unshift('dataset'); - } - - return deps; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var platform = ''; // Navigator not exists in node - - if (typeof navigator !== 'undefined') { - /* global navigator */ - platform = navigator.platform || ''; - } - - var decalColor = 'rgba(0, 0, 0, 0.2)'; - var globalDefault = { - darkMode: 'auto', - // backgroundColor: 'rgba(0,0,0,0)', - colorBy: 'series', - color: ['#5470c6', '#91cc75', '#fac858', '#ee6666', '#73c0de', '#3ba272', '#fc8452', '#9a60b4', '#ea7ccc'], - gradientColor: ['#f6efa6', '#d88273', '#bf444c'], - aria: { - decal: { - decals: [{ - color: decalColor, - dashArrayX: [1, 0], - dashArrayY: [2, 5], - symbolSize: 1, - rotation: Math.PI / 6 - }, { - color: decalColor, - symbol: 'circle', - dashArrayX: [[8, 8], [0, 8, 8, 0]], - dashArrayY: [6, 0], - symbolSize: 0.8 - }, { - color: decalColor, - dashArrayX: [1, 0], - dashArrayY: [4, 3], - rotation: -Math.PI / 4 - }, { - color: decalColor, - dashArrayX: [[6, 6], [0, 6, 6, 0]], - dashArrayY: [6, 0] - }, { - color: decalColor, - dashArrayX: [[1, 0], [1, 6]], - dashArrayY: [1, 0, 6, 0], - rotation: Math.PI / 4 - }, { - color: decalColor, - symbol: 'triangle', - dashArrayX: [[9, 9], [0, 9, 9, 0]], - dashArrayY: [7, 2], - symbolSize: 0.75 - }] - } - }, - // If xAxis and yAxis declared, grid is created by default. - // grid: {}, - textStyle: { - // color: '#000', - // decoration: 'none', - // PENDING - fontFamily: platform.match(/^Win/) ? 'Microsoft YaHei' : 'sans-serif', - // fontFamily: 'Arial, Verdana, sans-serif', - fontSize: 12, - fontStyle: 'normal', - fontWeight: 'normal' - }, - // http://blogs.adobe.com/webplatform/2014/02/24/using-blend-modes-in-html-canvas/ - // https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/globalCompositeOperation - // Default is source-over - blendMode: null, - stateAnimation: { - duration: 300, - easing: 'cubicOut' - }, - animation: 'auto', - animationDuration: 1000, - animationDurationUpdate: 500, - animationEasing: 'cubicInOut', - animationEasingUpdate: 'cubicInOut', - animationThreshold: 2000, - // Configuration for progressive/incremental rendering - progressiveThreshold: 3000, - progressive: 400, - // Threshold of if use single hover layer to optimize. - // It is recommended that `hoverLayerThreshold` is equivalent to or less than - // `progressiveThreshold`, otherwise hover will cause restart of progressive, - // which is unexpected. - // see example <echarts/test/heatmap-large.html>. - hoverLayerThreshold: 3000, - // See: module:echarts/scale/Time - useUTC: false - }; - - var VISUAL_DIMENSIONS = createHashMap(['tooltip', 'label', 'itemName', 'itemId', 'itemGroupId', 'seriesName']); - var SOURCE_FORMAT_ORIGINAL = 'original'; - var SOURCE_FORMAT_ARRAY_ROWS = 'arrayRows'; - var SOURCE_FORMAT_OBJECT_ROWS = 'objectRows'; - var SOURCE_FORMAT_KEYED_COLUMNS = 'keyedColumns'; - var SOURCE_FORMAT_TYPED_ARRAY = 'typedArray'; - var SOURCE_FORMAT_UNKNOWN = 'unknown'; - var SERIES_LAYOUT_BY_COLUMN = 'column'; - var SERIES_LAYOUT_BY_ROW = 'row'; - - var BE_ORDINAL = { - Must: 1, - Might: 2, - Not: 3 // Other cases - - }; - var innerGlobalModel = makeInner(); - /** - * MUST be called before mergeOption of all series. - */ - - function resetSourceDefaulter(ecModel) { - // `datasetMap` is used to make default encode. - innerGlobalModel(ecModel).datasetMap = createHashMap(); - } - /** - * [The strategy of the arrengment of data dimensions for dataset]: - * "value way": all axes are non-category axes. So series one by one take - * several (the number is coordSysDims.length) dimensions from dataset. - * The result of data arrengment of data dimensions like: - * | ser0_x | ser0_y | ser1_x | ser1_y | ser2_x | ser2_y | - * "category way": at least one axis is category axis. So the the first data - * dimension is always mapped to the first category axis and shared by - * all of the series. The other data dimensions are taken by series like - * "value way" does. - * The result of data arrengment of data dimensions like: - * | ser_shared_x | ser0_y | ser1_y | ser2_y | - * - * @return encode Never be `null/undefined`. - */ - - function makeSeriesEncodeForAxisCoordSys(coordDimensions, seriesModel, source) { - var encode = {}; - var datasetModel = querySeriesUpstreamDatasetModel(seriesModel); // Currently only make default when using dataset, util more reqirements occur. - - if (!datasetModel || !coordDimensions) { - return encode; - } - - var encodeItemName = []; - var encodeSeriesName = []; - var ecModel = seriesModel.ecModel; - var datasetMap = innerGlobalModel(ecModel).datasetMap; - var key = datasetModel.uid + '_' + source.seriesLayoutBy; - var baseCategoryDimIndex; - var categoryWayValueDimStart; - coordDimensions = coordDimensions.slice(); - each(coordDimensions, function (coordDimInfoLoose, coordDimIdx) { - var coordDimInfo = isObject(coordDimInfoLoose) ? coordDimInfoLoose : coordDimensions[coordDimIdx] = { - name: coordDimInfoLoose - }; - - if (coordDimInfo.type === 'ordinal' && baseCategoryDimIndex == null) { - baseCategoryDimIndex = coordDimIdx; - categoryWayValueDimStart = getDataDimCountOnCoordDim(coordDimInfo); - } - - encode[coordDimInfo.name] = []; - }); - var datasetRecord = datasetMap.get(key) || datasetMap.set(key, { - categoryWayDim: categoryWayValueDimStart, - valueWayDim: 0 - }); // TODO - // Auto detect first time axis and do arrangement. - - each(coordDimensions, function (coordDimInfo, coordDimIdx) { - var coordDimName = coordDimInfo.name; - var count = getDataDimCountOnCoordDim(coordDimInfo); // In value way. - - if (baseCategoryDimIndex == null) { - var start = datasetRecord.valueWayDim; - pushDim(encode[coordDimName], start, count); - pushDim(encodeSeriesName, start, count); - datasetRecord.valueWayDim += count; // ??? TODO give a better default series name rule? - // especially when encode x y specified. - // consider: when multiple series share one dimension - // category axis, series name should better use - // the other dimension name. On the other hand, use - // both dimensions name. - } // In category way, the first category axis. - else if (baseCategoryDimIndex === coordDimIdx) { - pushDim(encode[coordDimName], 0, count); - pushDim(encodeItemName, 0, count); - } // In category way, the other axis. - else { - var start = datasetRecord.categoryWayDim; - pushDim(encode[coordDimName], start, count); - pushDim(encodeSeriesName, start, count); - datasetRecord.categoryWayDim += count; - } - }); - - function pushDim(dimIdxArr, idxFrom, idxCount) { - for (var i = 0; i < idxCount; i++) { - dimIdxArr.push(idxFrom + i); - } - } - - function getDataDimCountOnCoordDim(coordDimInfo) { - var dimsDef = coordDimInfo.dimsDef; - return dimsDef ? dimsDef.length : 1; - } - - encodeItemName.length && (encode.itemName = encodeItemName); - encodeSeriesName.length && (encode.seriesName = encodeSeriesName); - return encode; - } - /** - * Work for data like [{name: ..., value: ...}, ...]. - * - * @return encode Never be `null/undefined`. - */ - - function makeSeriesEncodeForNameBased(seriesModel, source, dimCount) { - var encode = {}; - var datasetModel = querySeriesUpstreamDatasetModel(seriesModel); // Currently only make default when using dataset, util more reqirements occur. - - if (!datasetModel) { - return encode; - } - - var sourceFormat = source.sourceFormat; - var dimensionsDefine = source.dimensionsDefine; - var potentialNameDimIndex; - - if (sourceFormat === SOURCE_FORMAT_OBJECT_ROWS || sourceFormat === SOURCE_FORMAT_KEYED_COLUMNS) { - each(dimensionsDefine, function (dim, idx) { - if ((isObject(dim) ? dim.name : dim) === 'name') { - potentialNameDimIndex = idx; - } - }); - } - - var idxResult = function () { - var idxRes0 = {}; - var idxRes1 = {}; - var guessRecords = []; // 5 is an experience value. - - for (var i = 0, len = Math.min(5, dimCount); i < len; i++) { - var guessResult = doGuessOrdinal(source.data, sourceFormat, source.seriesLayoutBy, dimensionsDefine, source.startIndex, i); - guessRecords.push(guessResult); - var isPureNumber = guessResult === BE_ORDINAL.Not; // [Strategy of idxRes0]: find the first BE_ORDINAL.Not as the value dim, - // and then find a name dim with the priority: - // "BE_ORDINAL.Might|BE_ORDINAL.Must" > "other dim" > "the value dim itself". - - if (isPureNumber && idxRes0.v == null && i !== potentialNameDimIndex) { - idxRes0.v = i; - } - - if (idxRes0.n == null || idxRes0.n === idxRes0.v || !isPureNumber && guessRecords[idxRes0.n] === BE_ORDINAL.Not) { - idxRes0.n = i; - } - - if (fulfilled(idxRes0) && guessRecords[idxRes0.n] !== BE_ORDINAL.Not) { - return idxRes0; - } // [Strategy of idxRes1]: if idxRes0 not satisfied (that is, no BE_ORDINAL.Not), - // find the first BE_ORDINAL.Might as the value dim, - // and then find a name dim with the priority: - // "other dim" > "the value dim itself". - // That is for backward compat: number-like (e.g., `'3'`, `'55'`) can be - // treated as number. - - - if (!isPureNumber) { - if (guessResult === BE_ORDINAL.Might && idxRes1.v == null && i !== potentialNameDimIndex) { - idxRes1.v = i; - } - - if (idxRes1.n == null || idxRes1.n === idxRes1.v) { - idxRes1.n = i; - } - } - } - - function fulfilled(idxResult) { - return idxResult.v != null && idxResult.n != null; - } - - return fulfilled(idxRes0) ? idxRes0 : fulfilled(idxRes1) ? idxRes1 : null; - }(); - - if (idxResult) { - encode.value = [idxResult.v]; // `potentialNameDimIndex` has highest priority. - - var nameDimIndex = potentialNameDimIndex != null ? potentialNameDimIndex : idxResult.n; // By default, label uses itemName in charts. - // So we don't set encodeLabel here. - - encode.itemName = [nameDimIndex]; - encode.seriesName = [nameDimIndex]; - } - - return encode; - } - /** - * @return If return null/undefined, indicate that should not use datasetModel. - */ - - function querySeriesUpstreamDatasetModel(seriesModel) { - // Caution: consider the scenario: - // A dataset is declared and a series is not expected to use the dataset, - // and at the beginning `setOption({series: { noData })` (just prepare other - // option but no data), then `setOption({series: {data: [...]}); In this case, - // the user should set an empty array to avoid that dataset is used by default. - var thisData = seriesModel.get('data', true); - - if (!thisData) { - return queryReferringComponents(seriesModel.ecModel, 'dataset', { - index: seriesModel.get('datasetIndex', true), - id: seriesModel.get('datasetId', true) - }, SINGLE_REFERRING).models[0]; - } - } - /** - * @return Always return an array event empty. - */ - - function queryDatasetUpstreamDatasetModels(datasetModel) { - // Only these attributes declared, we by defualt reference to `datasetIndex: 0`. - // Otherwise, no reference. - if (!datasetModel.get('transform', true) && !datasetModel.get('fromTransformResult', true)) { - return []; - } - - return queryReferringComponents(datasetModel.ecModel, 'dataset', { - index: datasetModel.get('fromDatasetIndex', true), - id: datasetModel.get('fromDatasetId', true) - }, SINGLE_REFERRING).models; - } - /** - * The rule should not be complex, otherwise user might not - * be able to known where the data is wrong. - * The code is ugly, but how to make it neat? - */ - - function guessOrdinal(source, dimIndex) { - return doGuessOrdinal(source.data, source.sourceFormat, source.seriesLayoutBy, source.dimensionsDefine, source.startIndex, dimIndex); - } // dimIndex may be overflow source data. - // return {BE_ORDINAL} - - function doGuessOrdinal(data, sourceFormat, seriesLayoutBy, dimensionsDefine, startIndex, dimIndex) { - var result; // Experience value. - - var maxLoop = 5; - - if (isTypedArray(data)) { - return BE_ORDINAL.Not; - } // When sourceType is 'objectRows' or 'keyedColumns', dimensionsDefine - // always exists in source. - - - var dimName; - var dimType; - - if (dimensionsDefine) { - var dimDefItem = dimensionsDefine[dimIndex]; - - if (isObject(dimDefItem)) { - dimName = dimDefItem.name; - dimType = dimDefItem.type; - } else if (isString(dimDefItem)) { - dimName = dimDefItem; - } - } - - if (dimType != null) { - return dimType === 'ordinal' ? BE_ORDINAL.Must : BE_ORDINAL.Not; - } - - if (sourceFormat === SOURCE_FORMAT_ARRAY_ROWS) { - var dataArrayRows = data; - - if (seriesLayoutBy === SERIES_LAYOUT_BY_ROW) { - var sample = dataArrayRows[dimIndex]; - - for (var i = 0; i < (sample || []).length && i < maxLoop; i++) { - if ((result = detectValue(sample[startIndex + i])) != null) { - return result; - } - } - } else { - for (var i = 0; i < dataArrayRows.length && i < maxLoop; i++) { - var row = dataArrayRows[startIndex + i]; - - if (row && (result = detectValue(row[dimIndex])) != null) { - return result; - } - } - } - } else if (sourceFormat === SOURCE_FORMAT_OBJECT_ROWS) { - var dataObjectRows = data; - - if (!dimName) { - return BE_ORDINAL.Not; - } - - for (var i = 0; i < dataObjectRows.length && i < maxLoop; i++) { - var item = dataObjectRows[i]; - - if (item && (result = detectValue(item[dimName])) != null) { - return result; - } - } - } else if (sourceFormat === SOURCE_FORMAT_KEYED_COLUMNS) { - var dataKeyedColumns = data; - - if (!dimName) { - return BE_ORDINAL.Not; - } - - var sample = dataKeyedColumns[dimName]; - - if (!sample || isTypedArray(sample)) { - return BE_ORDINAL.Not; - } - - for (var i = 0; i < sample.length && i < maxLoop; i++) { - if ((result = detectValue(sample[i])) != null) { - return result; - } - } - } else if (sourceFormat === SOURCE_FORMAT_ORIGINAL) { - var dataOriginal = data; - - for (var i = 0; i < dataOriginal.length && i < maxLoop; i++) { - var item = dataOriginal[i]; - var val = getDataItemValue(item); - - if (!isArray(val)) { - return BE_ORDINAL.Not; - } - - if ((result = detectValue(val[dimIndex])) != null) { - return result; - } - } - } - - function detectValue(val) { - var beStr = isString(val); // Consider usage convenience, '1', '2' will be treated as "number". - // `isFinit('')` get `true`. - - if (val != null && isFinite(val) && val !== '') { - return beStr ? BE_ORDINAL.Might : BE_ORDINAL.Not; - } else if (beStr && val !== '-') { - return BE_ORDINAL.Must; - } - } - - return BE_ORDINAL.Not; - } - - var internalOptionCreatorMap = createHashMap(); - function registerInternalOptionCreator(mainType, creator) { - assert(internalOptionCreatorMap.get(mainType) == null && creator); - internalOptionCreatorMap.set(mainType, creator); - } - function concatInternalOptions(ecModel, mainType, newCmptOptionList) { - var internalOptionCreator = internalOptionCreatorMap.get(mainType); - - if (!internalOptionCreator) { - return newCmptOptionList; - } - - var internalOptions = internalOptionCreator(ecModel); - - if (!internalOptions) { - return newCmptOptionList; - } - - if ("development" !== 'production') { - for (var i = 0; i < internalOptions.length; i++) { - assert(isComponentIdInternal(internalOptions[i])); - } - } - - return newCmptOptionList.concat(internalOptions); - } - - var innerColor = makeInner(); - var innerDecal = makeInner(); - - var PaletteMixin = - /** @class */ - function () { - function PaletteMixin() {} - - PaletteMixin.prototype.getColorFromPalette = function (name, scope, requestNum) { - var defaultPalette = normalizeToArray(this.get('color', true)); - var layeredPalette = this.get('colorLayer', true); - return getFromPalette(this, innerColor, defaultPalette, layeredPalette, name, scope, requestNum); - }; - - PaletteMixin.prototype.clearColorPalette = function () { - clearPalette(this, innerColor); - }; - - return PaletteMixin; - }(); - - function getDecalFromPalette(ecModel, name, scope, requestNum) { - var defaultDecals = normalizeToArray(ecModel.get(['aria', 'decal', 'decals'])); - return getFromPalette(ecModel, innerDecal, defaultDecals, null, name, scope, requestNum); - } - - function getNearestPalette(palettes, requestColorNum) { - var paletteNum = palettes.length; // TODO palettes must be in order - - for (var i = 0; i < paletteNum; i++) { - if (palettes[i].length > requestColorNum) { - return palettes[i]; - } - } - - return palettes[paletteNum - 1]; - } - /** - * @param name MUST NOT be null/undefined. Otherwise call this function - * twise with the same parameters will get different result. - * @param scope default this. - * @return Can be null/undefined - */ - - - function getFromPalette(that, inner, defaultPalette, layeredPalette, name, scope, requestNum) { - scope = scope || that; - var scopeFields = inner(scope); - var paletteIdx = scopeFields.paletteIdx || 0; - var paletteNameMap = scopeFields.paletteNameMap = scopeFields.paletteNameMap || {}; // Use `hasOwnProperty` to avoid conflict with Object.prototype. - - if (paletteNameMap.hasOwnProperty(name)) { - return paletteNameMap[name]; - } - - var palette = requestNum == null || !layeredPalette ? defaultPalette : getNearestPalette(layeredPalette, requestNum); // In case can't find in layered color palette. - - palette = palette || defaultPalette; - - if (!palette || !palette.length) { - return; - } - - var pickedPaletteItem = palette[paletteIdx]; - - if (name) { - paletteNameMap[name] = pickedPaletteItem; - } - - scopeFields.paletteIdx = (paletteIdx + 1) % palette.length; - return pickedPaletteItem; - } - - function clearPalette(that, inner) { - inner(that).paletteIdx = 0; - inner(that).paletteNameMap = {}; - } - - // Internal method names: - // ----------------------- - - var reCreateSeriesIndices; - var assertSeriesInitialized; - var initBase; - var OPTION_INNER_KEY = '\0_ec_inner'; - var OPTION_INNER_VALUE = 1; - var BUITIN_COMPONENTS_MAP = { - grid: 'GridComponent', - polar: 'PolarComponent', - geo: 'GeoComponent', - singleAxis: 'SingleAxisComponent', - parallel: 'ParallelComponent', - calendar: 'CalendarComponent', - graphic: 'GraphicComponent', - toolbox: 'ToolboxComponent', - tooltip: 'TooltipComponent', - axisPointer: 'AxisPointerComponent', - brush: 'BrushComponent', - title: 'TitleComponent', - timeline: 'TimelineComponent', - markPoint: 'MarkPointComponent', - markLine: 'MarkLineComponent', - markArea: 'MarkAreaComponent', - legend: 'LegendComponent', - dataZoom: 'DataZoomComponent', - visualMap: 'VisualMapComponent', - // aria: 'AriaComponent', - // dataset: 'DatasetComponent', - // Dependencies - xAxis: 'GridComponent', - yAxis: 'GridComponent', - angleAxis: 'PolarComponent', - radiusAxis: 'PolarComponent' - }; - var BUILTIN_CHARTS_MAP = { - line: 'LineChart', - bar: 'BarChart', - pie: 'PieChart', - scatter: 'ScatterChart', - radar: 'RadarChart', - map: 'MapChart', - tree: 'TreeChart', - treemap: 'TreemapChart', - graph: 'GraphChart', - gauge: 'GaugeChart', - funnel: 'FunnelChart', - parallel: 'ParallelChart', - sankey: 'SankeyChart', - boxplot: 'BoxplotChart', - candlestick: 'CandlestickChart', - effectScatter: 'EffectScatterChart', - lines: 'LinesChart', - heatmap: 'HeatmapChart', - pictorialBar: 'PictorialBarChart', - themeRiver: 'ThemeRiverChart', - sunburst: 'SunburstChart', - custom: 'CustomChart' - }; - var componetsMissingLogPrinted = {}; - - function checkMissingComponents(option) { - each(option, function (componentOption, mainType) { - if (!ComponentModel.hasClass(mainType)) { - var componentImportName = BUITIN_COMPONENTS_MAP[mainType]; - - if (componentImportName && !componetsMissingLogPrinted[componentImportName]) { - error("Component " + mainType + " is used but not imported.\nimport { " + componentImportName + " } from 'echarts/components';\necharts.use([" + componentImportName + "]);"); - componetsMissingLogPrinted[componentImportName] = true; - } - } - }); - } - - var GlobalModel = - /** @class */ - function (_super) { - __extends(GlobalModel, _super); - - function GlobalModel() { - return _super !== null && _super.apply(this, arguments) || this; - } - - GlobalModel.prototype.init = function (option, parentModel, ecModel, theme, locale, optionManager) { - theme = theme || {}; - this.option = null; // Mark as not initialized. - - this._theme = new Model(theme); - this._locale = new Model(locale); - this._optionManager = optionManager; - }; - - GlobalModel.prototype.setOption = function (option, opts, optionPreprocessorFuncs) { - if ("development" !== 'production') { - assert(option != null, 'option is null/undefined'); - assert(option[OPTION_INNER_KEY] !== OPTION_INNER_VALUE, 'please use chart.getOption()'); - } - - var innerOpt = normalizeSetOptionInput(opts); - - this._optionManager.setOption(option, optionPreprocessorFuncs, innerOpt); - - this._resetOption(null, innerOpt); - }; - /** - * @param type null/undefined: reset all. - * 'recreate': force recreate all. - * 'timeline': only reset timeline option - * 'media': only reset media query option - * @return Whether option changed. - */ - - - GlobalModel.prototype.resetOption = function (type, opt) { - return this._resetOption(type, normalizeSetOptionInput(opt)); - }; - - GlobalModel.prototype._resetOption = function (type, opt) { - var optionChanged = false; - var optionManager = this._optionManager; - - if (!type || type === 'recreate') { - var baseOption = optionManager.mountOption(type === 'recreate'); - - if ("development" !== 'production') { - checkMissingComponents(baseOption); - } - - if (!this.option || type === 'recreate') { - initBase(this, baseOption); - } else { - this.restoreData(); - - this._mergeOption(baseOption, opt); - } - - optionChanged = true; - } - - if (type === 'timeline' || type === 'media') { - this.restoreData(); - } // By design, if `setOption(option2)` at the second time, and `option2` is a `ECUnitOption`, - // it should better not have the same props with `MediaUnit['option']`. - // Because either `option2` or `MediaUnit['option']` will be always merged to "current option" - // rather than original "baseOption". If they both override a prop, the result might be - // unexpected when media state changed after `setOption` called. - // If we really need to modify a props in each `MediaUnit['option']`, use the full version - // (`{baseOption, media}`) in `setOption`. - // For `timeline`, the case is the same. - - - if (!type || type === 'recreate' || type === 'timeline') { - var timelineOption = optionManager.getTimelineOption(this); - - if (timelineOption) { - optionChanged = true; - - this._mergeOption(timelineOption, opt); - } - } - - if (!type || type === 'recreate' || type === 'media') { - var mediaOptions = optionManager.getMediaOption(this); - - if (mediaOptions.length) { - each(mediaOptions, function (mediaOption) { - optionChanged = true; - - this._mergeOption(mediaOption, opt); - }, this); - } - } - - return optionChanged; - }; - - GlobalModel.prototype.mergeOption = function (option) { - this._mergeOption(option, null); - }; - - GlobalModel.prototype._mergeOption = function (newOption, opt) { - var option = this.option; - var componentsMap = this._componentsMap; - var componentsCount = this._componentsCount; - var newCmptTypes = []; - var newCmptTypeMap = createHashMap(); - var replaceMergeMainTypeMap = opt && opt.replaceMergeMainTypeMap; - resetSourceDefaulter(this); // If no component class, merge directly. - // For example: color, animaiton options, etc. - - each(newOption, function (componentOption, mainType) { - if (componentOption == null) { - return; - } - - if (!ComponentModel.hasClass(mainType)) { - // globalSettingTask.dirty(); - option[mainType] = option[mainType] == null ? clone(componentOption) : merge(option[mainType], componentOption, true); - } else if (mainType) { - newCmptTypes.push(mainType); - newCmptTypeMap.set(mainType, true); - } - }); - - if (replaceMergeMainTypeMap) { - // If there is a mainType `xxx` in `replaceMerge` but not declared in option, - // we trade it as it is declared in option as `{xxx: []}`. Because: - // (1) for normal merge, `{xxx: null/undefined}` are the same meaning as `{xxx: []}`. - // (2) some preprocessor may convert some of `{xxx: null/undefined}` to `{xxx: []}`. - replaceMergeMainTypeMap.each(function (val, mainTypeInReplaceMerge) { - if (ComponentModel.hasClass(mainTypeInReplaceMerge) && !newCmptTypeMap.get(mainTypeInReplaceMerge)) { - newCmptTypes.push(mainTypeInReplaceMerge); - newCmptTypeMap.set(mainTypeInReplaceMerge, true); - } - }); - } - - ComponentModel.topologicalTravel(newCmptTypes, ComponentModel.getAllClassMainTypes(), visitComponent, this); - - function visitComponent(mainType) { - var newCmptOptionList = concatInternalOptions(this, mainType, normalizeToArray(newOption[mainType])); - var oldCmptList = componentsMap.get(mainType); - var mergeMode = // `!oldCmptList` means init. See the comment in `mappingToExists` - !oldCmptList ? 'replaceAll' : replaceMergeMainTypeMap && replaceMergeMainTypeMap.get(mainType) ? 'replaceMerge' : 'normalMerge'; - var mappingResult = mappingToExists(oldCmptList, newCmptOptionList, mergeMode); // Set mainType and complete subType. - - setComponentTypeToKeyInfo(mappingResult, mainType, ComponentModel); // Empty it before the travel, in order to prevent `this._componentsMap` - // from being used in the `init`/`mergeOption`/`optionUpdated` of some - // components, which is probably incorrect logic. - - option[mainType] = null; - componentsMap.set(mainType, null); - componentsCount.set(mainType, 0); - var optionsByMainType = []; - var cmptsByMainType = []; - var cmptsCountByMainType = 0; - var tooltipExists; - var tooltipWarningLogged; - each(mappingResult, function (resultItem, index) { - var componentModel = resultItem.existing; - var newCmptOption = resultItem.newOption; - - if (!newCmptOption) { - if (componentModel) { - // Consider where is no new option and should be merged using {}, - // see removeEdgeAndAdd in topologicalTravel and - // ComponentModel.getAllClassMainTypes. - componentModel.mergeOption({}, this); - componentModel.optionUpdated({}, false); - } // If no both `resultItem.exist` and `resultItem.option`, - // either it is in `replaceMerge` and not matched by any id, - // or it has been removed in previous `replaceMerge` and left a "hole" in this component index. - - } else { - var isSeriesType = mainType === 'series'; - var ComponentModelClass = ComponentModel.getClass(mainType, resultItem.keyInfo.subType, !isSeriesType // Give a more detailed warn later if series don't exists - ); - - if (!ComponentModelClass) { - if ("development" !== 'production') { - var subType = resultItem.keyInfo.subType; - var seriesImportName = BUILTIN_CHARTS_MAP[subType]; - - if (!componetsMissingLogPrinted[subType]) { - componetsMissingLogPrinted[subType] = true; - - if (seriesImportName) { - error("Series " + subType + " is used but not imported.\nimport { " + seriesImportName + " } from 'echarts/charts';\necharts.use([" + seriesImportName + "]);"); - } else { - error("Unknown series " + subType); - } - } - } - - return; - } // TODO Before multiple tooltips get supported, we do this check to avoid unexpected exception. - - - if (mainType === 'tooltip') { - if (tooltipExists) { - if ("development" !== 'production') { - if (!tooltipWarningLogged) { - warn('Currently only one tooltip component is allowed.'); - tooltipWarningLogged = true; - } - } - - return; - } - - tooltipExists = true; - } - - if (componentModel && componentModel.constructor === ComponentModelClass) { - componentModel.name = resultItem.keyInfo.name; // componentModel.settingTask && componentModel.settingTask.dirty(); - - componentModel.mergeOption(newCmptOption, this); - componentModel.optionUpdated(newCmptOption, false); - } else { - // PENDING Global as parent ? - var extraOpt = extend({ - componentIndex: index - }, resultItem.keyInfo); - componentModel = new ComponentModelClass(newCmptOption, this, this, extraOpt); // Assign `keyInfo` - - extend(componentModel, extraOpt); - - if (resultItem.brandNew) { - componentModel.__requireNewView = true; - } - - componentModel.init(newCmptOption, this, this); // Call optionUpdated after init. - // newCmptOption has been used as componentModel.option - // and may be merged with theme and default, so pass null - // to avoid confusion. - - componentModel.optionUpdated(null, true); - } - } - - if (componentModel) { - optionsByMainType.push(componentModel.option); - cmptsByMainType.push(componentModel); - cmptsCountByMainType++; - } else { - // Always do assign to avoid elided item in array. - optionsByMainType.push(void 0); - cmptsByMainType.push(void 0); - } - }, this); - option[mainType] = optionsByMainType; - componentsMap.set(mainType, cmptsByMainType); - componentsCount.set(mainType, cmptsCountByMainType); // Backup series for filtering. - - if (mainType === 'series') { - reCreateSeriesIndices(this); - } - } // If no series declared, ensure `_seriesIndices` initialized. - - - if (!this._seriesIndices) { - reCreateSeriesIndices(this); - } - }; - /** - * Get option for output (cloned option and inner info removed) - */ - - - GlobalModel.prototype.getOption = function () { - var option = clone(this.option); - each(option, function (optInMainType, mainType) { - if (ComponentModel.hasClass(mainType)) { - var opts = normalizeToArray(optInMainType); // Inner cmpts need to be removed. - // Inner cmpts might not be at last since ec5.0, but still - // compatible for users: if inner cmpt at last, splice the returned array. - - var realLen = opts.length; - var metNonInner = false; - - for (var i = realLen - 1; i >= 0; i--) { - // Remove options with inner id. - if (opts[i] && !isComponentIdInternal(opts[i])) { - metNonInner = true; - } else { - opts[i] = null; - !metNonInner && realLen--; - } - } - - opts.length = realLen; - option[mainType] = opts; - } - }); - delete option[OPTION_INNER_KEY]; - return option; - }; - - GlobalModel.prototype.getTheme = function () { - return this._theme; - }; - - GlobalModel.prototype.getLocaleModel = function () { - return this._locale; - }; - - GlobalModel.prototype.setUpdatePayload = function (payload) { - this._payload = payload; - }; - - GlobalModel.prototype.getUpdatePayload = function () { - return this._payload; - }; - /** - * @param idx If not specified, return the first one. - */ - - - GlobalModel.prototype.getComponent = function (mainType, idx) { - var list = this._componentsMap.get(mainType); - - if (list) { - var cmpt = list[idx || 0]; - - if (cmpt) { - return cmpt; - } else if (idx == null) { - for (var i = 0; i < list.length; i++) { - if (list[i]) { - return list[i]; - } - } - } - } - }; - /** - * @return Never be null/undefined. - */ - - - GlobalModel.prototype.queryComponents = function (condition) { - var mainType = condition.mainType; - - if (!mainType) { - return []; - } - - var index = condition.index; - var id = condition.id; - var name = condition.name; - - var cmpts = this._componentsMap.get(mainType); - - if (!cmpts || !cmpts.length) { - return []; - } - - var result; - - if (index != null) { - result = []; - each(normalizeToArray(index), function (idx) { - cmpts[idx] && result.push(cmpts[idx]); - }); - } else if (id != null) { - result = queryByIdOrName('id', id, cmpts); - } else if (name != null) { - result = queryByIdOrName('name', name, cmpts); - } else { - // Return all non-empty components in that mainType - result = filter(cmpts, function (cmpt) { - return !!cmpt; - }); - } - - return filterBySubType(result, condition); - }; - /** - * The interface is different from queryComponents, - * which is convenient for inner usage. - * - * @usage - * let result = findComponents( - * {mainType: 'dataZoom', query: {dataZoomId: 'abc'}} - * ); - * let result = findComponents( - * {mainType: 'series', subType: 'pie', query: {seriesName: 'uio'}} - * ); - * let result = findComponents( - * {mainType: 'series', - * filter: function (model, index) {...}} - * ); - * // result like [component0, componnet1, ...] - */ - - - GlobalModel.prototype.findComponents = function (condition) { - var query = condition.query; - var mainType = condition.mainType; - var queryCond = getQueryCond(query); - var result = queryCond ? this.queryComponents(queryCond) // Retrieve all non-empty components. - : filter(this._componentsMap.get(mainType), function (cmpt) { - return !!cmpt; - }); - return doFilter(filterBySubType(result, condition)); - - function getQueryCond(q) { - var indexAttr = mainType + 'Index'; - var idAttr = mainType + 'Id'; - var nameAttr = mainType + 'Name'; - return q && (q[indexAttr] != null || q[idAttr] != null || q[nameAttr] != null) ? { - mainType: mainType, - // subType will be filtered finally. - index: q[indexAttr], - id: q[idAttr], - name: q[nameAttr] - } : null; - } - - function doFilter(res) { - return condition.filter ? filter(res, condition.filter) : res; - } - }; - - GlobalModel.prototype.eachComponent = function (mainType, cb, context) { - var componentsMap = this._componentsMap; - - if (isFunction(mainType)) { - var ctxForAll_1 = cb; - var cbForAll_1 = mainType; - componentsMap.each(function (cmpts, componentType) { - for (var i = 0; cmpts && i < cmpts.length; i++) { - var cmpt = cmpts[i]; - cmpt && cbForAll_1.call(ctxForAll_1, componentType, cmpt, cmpt.componentIndex); - } - }); - } else { - var cmpts = isString(mainType) ? componentsMap.get(mainType) : isObject(mainType) ? this.findComponents(mainType) : null; - - for (var i = 0; cmpts && i < cmpts.length; i++) { - var cmpt = cmpts[i]; - cmpt && cb.call(context, cmpt, cmpt.componentIndex); - } - } - }; - /** - * Get series list before filtered by name. - */ - - - GlobalModel.prototype.getSeriesByName = function (name) { - var nameStr = convertOptionIdName(name, null); - return filter(this._componentsMap.get('series'), function (oneSeries) { - return !!oneSeries && nameStr != null && oneSeries.name === nameStr; - }); - }; - /** - * Get series list before filtered by index. - */ - - - GlobalModel.prototype.getSeriesByIndex = function (seriesIndex) { - return this._componentsMap.get('series')[seriesIndex]; - }; - /** - * Get series list before filtered by type. - * FIXME: rename to getRawSeriesByType? - */ - - - GlobalModel.prototype.getSeriesByType = function (subType) { - return filter(this._componentsMap.get('series'), function (oneSeries) { - return !!oneSeries && oneSeries.subType === subType; - }); - }; - /** - * Get all series before filtered. - */ - - - GlobalModel.prototype.getSeries = function () { - return filter(this._componentsMap.get('series'), function (oneSeries) { - return !!oneSeries; - }); - }; - /** - * Count series before filtered. - */ - - - GlobalModel.prototype.getSeriesCount = function () { - return this._componentsCount.get('series'); - }; - /** - * After filtering, series may be different - * from raw series. - */ - - - GlobalModel.prototype.eachSeries = function (cb, context) { - assertSeriesInitialized(this); - each(this._seriesIndices, function (rawSeriesIndex) { - var series = this._componentsMap.get('series')[rawSeriesIndex]; - - cb.call(context, series, rawSeriesIndex); - }, this); - }; - /** - * Iterate raw series before filtered. - * - * @param {Function} cb - * @param {*} context - */ - - - GlobalModel.prototype.eachRawSeries = function (cb, context) { - each(this._componentsMap.get('series'), function (series) { - series && cb.call(context, series, series.componentIndex); - }); - }; - /** - * After filtering, series may be different. - * from raw series. - */ - - - GlobalModel.prototype.eachSeriesByType = function (subType, cb, context) { - assertSeriesInitialized(this); - each(this._seriesIndices, function (rawSeriesIndex) { - var series = this._componentsMap.get('series')[rawSeriesIndex]; - - if (series.subType === subType) { - cb.call(context, series, rawSeriesIndex); - } - }, this); - }; - /** - * Iterate raw series before filtered of given type. - */ - - - GlobalModel.prototype.eachRawSeriesByType = function (subType, cb, context) { - return each(this.getSeriesByType(subType), cb, context); - }; - - GlobalModel.prototype.isSeriesFiltered = function (seriesModel) { - assertSeriesInitialized(this); - return this._seriesIndicesMap.get(seriesModel.componentIndex) == null; - }; - - GlobalModel.prototype.getCurrentSeriesIndices = function () { - return (this._seriesIndices || []).slice(); - }; - - GlobalModel.prototype.filterSeries = function (cb, context) { - assertSeriesInitialized(this); - var newSeriesIndices = []; - each(this._seriesIndices, function (seriesRawIdx) { - var series = this._componentsMap.get('series')[seriesRawIdx]; - - cb.call(context, series, seriesRawIdx) && newSeriesIndices.push(seriesRawIdx); - }, this); - this._seriesIndices = newSeriesIndices; - this._seriesIndicesMap = createHashMap(newSeriesIndices); - }; - - GlobalModel.prototype.restoreData = function (payload) { - reCreateSeriesIndices(this); - var componentsMap = this._componentsMap; - var componentTypes = []; - componentsMap.each(function (components, componentType) { - if (ComponentModel.hasClass(componentType)) { - componentTypes.push(componentType); - } - }); - ComponentModel.topologicalTravel(componentTypes, ComponentModel.getAllClassMainTypes(), function (componentType) { - each(componentsMap.get(componentType), function (component) { - if (component && (componentType !== 'series' || !isNotTargetSeries(component, payload))) { - component.restoreData(); - } - }); - }); - }; - - GlobalModel.internalField = function () { - reCreateSeriesIndices = function (ecModel) { - var seriesIndices = ecModel._seriesIndices = []; - each(ecModel._componentsMap.get('series'), function (series) { - // series may have been removed by `replaceMerge`. - series && seriesIndices.push(series.componentIndex); - }); - ecModel._seriesIndicesMap = createHashMap(seriesIndices); - }; - - assertSeriesInitialized = function (ecModel) { - // Components that use _seriesIndices should depends on series component, - // which make sure that their initialization is after series. - if ("development" !== 'production') { - if (!ecModel._seriesIndices) { - throw new Error('Option should contains series.'); - } - } - }; - - initBase = function (ecModel, baseOption) { - // Using OPTION_INNER_KEY to mark that this option cannot be used outside, - // i.e. `chart.setOption(chart.getModel().option);` is forbidden. - ecModel.option = {}; - ecModel.option[OPTION_INNER_KEY] = OPTION_INNER_VALUE; // Init with series: [], in case of calling findSeries method - // before series initialized. - - ecModel._componentsMap = createHashMap({ - series: [] - }); - ecModel._componentsCount = createHashMap(); // If user spefied `option.aria`, aria will be enable. This detection should be - // performed before theme and globalDefault merge. - - var airaOption = baseOption.aria; - - if (isObject(airaOption) && airaOption.enabled == null) { - airaOption.enabled = true; - } - - mergeTheme(baseOption, ecModel._theme.option); // TODO Needs clone when merging to the unexisted property - - merge(baseOption, globalDefault, false); - - ecModel._mergeOption(baseOption, null); - }; - }(); - - return GlobalModel; - }(Model); - - function isNotTargetSeries(seriesModel, payload) { - if (payload) { - var index = payload.seriesIndex; - var id = payload.seriesId; - var name_1 = payload.seriesName; - return index != null && seriesModel.componentIndex !== index || id != null && seriesModel.id !== id || name_1 != null && seriesModel.name !== name_1; - } - } - - function mergeTheme(option, theme) { - // PENDING - // NOT use `colorLayer` in theme if option has `color` - var notMergeColorLayer = option.color && !option.colorLayer; - each(theme, function (themeItem, name) { - if (name === 'colorLayer' && notMergeColorLayer) { - return; - } // If it is component model mainType, the model handles that merge later. - // otherwise, merge them here. - - - if (!ComponentModel.hasClass(name)) { - if (typeof themeItem === 'object') { - option[name] = !option[name] ? clone(themeItem) : merge(option[name], themeItem, false); - } else { - if (option[name] == null) { - option[name] = themeItem; - } - } - } - }); - } - - function queryByIdOrName(attr, idOrName, cmpts) { - // Here is a break from echarts4: string and number are - // treated as equal. - if (isArray(idOrName)) { - var keyMap_1 = createHashMap(); - each(idOrName, function (idOrNameItem) { - if (idOrNameItem != null) { - var idName = convertOptionIdName(idOrNameItem, null); - idName != null && keyMap_1.set(idOrNameItem, true); - } - }); - return filter(cmpts, function (cmpt) { - return cmpt && keyMap_1.get(cmpt[attr]); - }); - } else { - var idName_1 = convertOptionIdName(idOrName, null); - return filter(cmpts, function (cmpt) { - return cmpt && idName_1 != null && cmpt[attr] === idName_1; - }); - } - } - - function filterBySubType(components, condition) { - // Using hasOwnProperty for restrict. Consider - // subType is undefined in user payload. - return condition.hasOwnProperty('subType') ? filter(components, function (cmpt) { - return cmpt && cmpt.subType === condition.subType; - }) : components; - } - - function normalizeSetOptionInput(opts) { - var replaceMergeMainTypeMap = createHashMap(); - opts && each(normalizeToArray(opts.replaceMerge), function (mainType) { - if ("development" !== 'production') { - assert(ComponentModel.hasClass(mainType), '"' + mainType + '" is not valid component main type in "replaceMerge"'); - } - - replaceMergeMainTypeMap.set(mainType, true); - }); - return { - replaceMergeMainTypeMap: replaceMergeMainTypeMap - }; - } - - mixin(GlobalModel, PaletteMixin); - - var availableMethods = ['getDom', 'getZr', 'getWidth', 'getHeight', 'getDevicePixelRatio', 'dispatchAction', 'isSSR', 'isDisposed', 'on', 'off', 'getDataURL', 'getConnectedDataURL', // 'getModel', - 'getOption', // 'getViewOfComponentModel', - // 'getViewOfSeriesModel', - 'getId', 'updateLabelLayout']; - - var ExtensionAPI = - /** @class */ - function () { - function ExtensionAPI(ecInstance) { - each(availableMethods, function (methodName) { - this[methodName] = bind(ecInstance[methodName], ecInstance); - }, this); - } - - return ExtensionAPI; - }(); - - var coordinateSystemCreators = {}; - - var CoordinateSystemManager = - /** @class */ - function () { - function CoordinateSystemManager() { - this._coordinateSystems = []; - } - - CoordinateSystemManager.prototype.create = function (ecModel, api) { - var coordinateSystems = []; - each(coordinateSystemCreators, function (creator, type) { - var list = creator.create(ecModel, api); - coordinateSystems = coordinateSystems.concat(list || []); - }); - this._coordinateSystems = coordinateSystems; - }; - - CoordinateSystemManager.prototype.update = function (ecModel, api) { - each(this._coordinateSystems, function (coordSys) { - coordSys.update && coordSys.update(ecModel, api); - }); - }; - - CoordinateSystemManager.prototype.getCoordinateSystems = function () { - return this._coordinateSystems.slice(); - }; - - CoordinateSystemManager.register = function (type, creator) { - coordinateSystemCreators[type] = creator; - }; - - CoordinateSystemManager.get = function (type) { - return coordinateSystemCreators[type]; - }; - - return CoordinateSystemManager; - }(); - - var QUERY_REG = /^(min|max)?(.+)$/; // Key: mainType - // type FakeComponentsMap = HashMap<(MappingExistingItem & { subType: string })[]>; - - /** - * TERM EXPLANATIONS: - * See `ECOption` and `ECUnitOption` in `src/util/types.ts`. - */ - - var OptionManager = - /** @class */ - function () { - // timeline.notMerge is not supported in ec3. Firstly there is rearly - // case that notMerge is needed. Secondly supporting 'notMerge' requires - // rawOption cloned and backuped when timeline changed, which does no - // good to performance. What's more, that both timeline and setOption - // method supply 'notMerge' brings complex and some problems. - // Consider this case: - // (step1) chart.setOption({timeline: {notMerge: false}, ...}, false); - // (step2) chart.setOption({timeline: {notMerge: true}, ...}, false); - function OptionManager(api) { - this._timelineOptions = []; - this._mediaList = []; - /** - * -1, means default. - * empty means no media. - */ - - this._currentMediaIndices = []; - this._api = api; - } - - OptionManager.prototype.setOption = function (rawOption, optionPreprocessorFuncs, opt) { - if (rawOption) { - // That set dat primitive is dangerous if user reuse the data when setOption again. - each(normalizeToArray(rawOption.series), function (series) { - series && series.data && isTypedArray(series.data) && setAsPrimitive(series.data); - }); - each(normalizeToArray(rawOption.dataset), function (dataset) { - dataset && dataset.source && isTypedArray(dataset.source) && setAsPrimitive(dataset.source); - }); - } // Caution: some series modify option data, if do not clone, - // it should ensure that the repeat modify correctly - // (create a new object when modify itself). - - - rawOption = clone(rawOption); // FIXME - // If some property is set in timeline options or media option but - // not set in baseOption, a warning should be given. - - var optionBackup = this._optionBackup; - var newParsedOption = parseRawOption(rawOption, optionPreprocessorFuncs, !optionBackup); - this._newBaseOption = newParsedOption.baseOption; // For setOption at second time (using merge mode); - - if (optionBackup) { - // FIXME - // the restore merge solution is essentially incorrect. - // the mapping can not be 100% consistent with ecModel, which probably brings - // potential bug! - // The first merge is delayed, because in most cases, users do not call `setOption` twice. - // let fakeCmptsMap = this._fakeCmptsMap; - // if (!fakeCmptsMap) { - // fakeCmptsMap = this._fakeCmptsMap = createHashMap(); - // mergeToBackupOption(fakeCmptsMap, null, optionBackup.baseOption, null); - // } - // mergeToBackupOption( - // fakeCmptsMap, optionBackup.baseOption, newParsedOption.baseOption, opt - // ); - // For simplicity, timeline options and media options do not support merge, - // that is, if you `setOption` twice and both has timeline options, the latter - // timeline options will not be merged to the former, but just substitute them. - if (newParsedOption.timelineOptions.length) { - optionBackup.timelineOptions = newParsedOption.timelineOptions; - } - - if (newParsedOption.mediaList.length) { - optionBackup.mediaList = newParsedOption.mediaList; - } - - if (newParsedOption.mediaDefault) { - optionBackup.mediaDefault = newParsedOption.mediaDefault; - } - } else { - this._optionBackup = newParsedOption; - } - }; - - OptionManager.prototype.mountOption = function (isRecreate) { - var optionBackup = this._optionBackup; - this._timelineOptions = optionBackup.timelineOptions; - this._mediaList = optionBackup.mediaList; - this._mediaDefault = optionBackup.mediaDefault; - this._currentMediaIndices = []; - return clone(isRecreate // this._optionBackup.baseOption, which is created at the first `setOption` - // called, and is merged into every new option by inner method `mergeToBackupOption` - // each time `setOption` called, can be only used in `isRecreate`, because - // its reliability is under suspicion. In other cases option merge is - // performed by `model.mergeOption`. - ? optionBackup.baseOption : this._newBaseOption); - }; - - OptionManager.prototype.getTimelineOption = function (ecModel) { - var option; - var timelineOptions = this._timelineOptions; - - if (timelineOptions.length) { - // getTimelineOption can only be called after ecModel inited, - // so we can get currentIndex from timelineModel. - var timelineModel = ecModel.getComponent('timeline'); - - if (timelineModel) { - option = clone( // FIXME:TS as TimelineModel or quivlant interface - timelineOptions[timelineModel.getCurrentIndex()]); - } - } - - return option; - }; - - OptionManager.prototype.getMediaOption = function (ecModel) { - var ecWidth = this._api.getWidth(); - - var ecHeight = this._api.getHeight(); - - var mediaList = this._mediaList; - var mediaDefault = this._mediaDefault; - var indices = []; - var result = []; // No media defined. - - if (!mediaList.length && !mediaDefault) { - return result; - } // Multi media may be applied, the latter defined media has higher priority. - - - for (var i = 0, len = mediaList.length; i < len; i++) { - if (applyMediaQuery(mediaList[i].query, ecWidth, ecHeight)) { - indices.push(i); - } - } // FIXME - // Whether mediaDefault should force users to provide? Otherwise - // the change by media query can not be recorvered. - - - if (!indices.length && mediaDefault) { - indices = [-1]; - } - - if (indices.length && !indicesEquals(indices, this._currentMediaIndices)) { - result = map(indices, function (index) { - return clone(index === -1 ? mediaDefault.option : mediaList[index].option); - }); - } // Otherwise return nothing. - - - this._currentMediaIndices = indices; - return result; - }; - - return OptionManager; - }(); - /** - * [RAW_OPTION_PATTERNS] - * (Note: "series: []" represents all other props in `ECUnitOption`) - * - * (1) No prop "baseOption" declared: - * Root option is used as "baseOption" (except prop "options" and "media"). - * ```js - * option = { - * series: [], - * timeline: {}, - * options: [], - * }; - * option = { - * series: [], - * media: {}, - * }; - * option = { - * series: [], - * timeline: {}, - * options: [], - * media: {}, - * } - * ``` - * - * (2) Prop "baseOption" declared: - * If "baseOption" declared, `ECUnitOption` props can only be declared - * inside "baseOption" except prop "timeline" (compat ec2). - * ```js - * option = { - * baseOption: { - * timeline: {}, - * series: [], - * }, - * options: [] - * }; - * option = { - * baseOption: { - * series: [], - * }, - * media: [] - * }; - * option = { - * baseOption: { - * timeline: {}, - * series: [], - * }, - * options: [] - * media: [] - * }; - * option = { - * // ec3 compat ec2: allow (only) `timeline` declared - * // outside baseOption. Keep this setting for compat. - * timeline: {}, - * baseOption: { - * series: [], - * }, - * options: [], - * media: [] - * }; - * ``` - */ - - - function parseRawOption( // `rawOption` May be modified - rawOption, optionPreprocessorFuncs, isNew) { - var mediaList = []; - var mediaDefault; - var baseOption; - var declaredBaseOption = rawOption.baseOption; // Compatible with ec2, [RAW_OPTION_PATTERNS] above. - - var timelineOnRoot = rawOption.timeline; - var timelineOptionsOnRoot = rawOption.options; - var mediaOnRoot = rawOption.media; - var hasMedia = !!rawOption.media; - var hasTimeline = !!(timelineOptionsOnRoot || timelineOnRoot || declaredBaseOption && declaredBaseOption.timeline); - - if (declaredBaseOption) { - baseOption = declaredBaseOption; // For merge option. - - if (!baseOption.timeline) { - baseOption.timeline = timelineOnRoot; - } - } // For convenience, enable to use the root option as the `baseOption`: - // `{ ...normalOptionProps, media: [{ ... }, { ... }] }` - else { - if (hasTimeline || hasMedia) { - rawOption.options = rawOption.media = null; - } - - baseOption = rawOption; - } - - if (hasMedia) { - if (isArray(mediaOnRoot)) { - each(mediaOnRoot, function (singleMedia) { - if ("development" !== 'production') { - // Real case of wrong config. - if (singleMedia && !singleMedia.option && isObject(singleMedia.query) && isObject(singleMedia.query.option)) { - error('Illegal media option. Must be like { media: [ { query: {}, option: {} } ] }'); - } - } - - if (singleMedia && singleMedia.option) { - if (singleMedia.query) { - mediaList.push(singleMedia); - } else if (!mediaDefault) { - // Use the first media default. - mediaDefault = singleMedia; - } - } - }); - } else { - if ("development" !== 'production') { - // Real case of wrong config. - error('Illegal media option. Must be an array. Like { media: [ {...}, {...} ] }'); - } - } - } - - doPreprocess(baseOption); - each(timelineOptionsOnRoot, function (option) { - return doPreprocess(option); - }); - each(mediaList, function (media) { - return doPreprocess(media.option); - }); - - function doPreprocess(option) { - each(optionPreprocessorFuncs, function (preProcess) { - preProcess(option, isNew); - }); - } - - return { - baseOption: baseOption, - timelineOptions: timelineOptionsOnRoot || [], - mediaDefault: mediaDefault, - mediaList: mediaList - }; - } - /** - * @see <http://www.w3.org/TR/css3-mediaqueries/#media1> - * Support: width, height, aspectRatio - * Can use max or min as prefix. - */ - - - function applyMediaQuery(query, ecWidth, ecHeight) { - var realMap = { - width: ecWidth, - height: ecHeight, - aspectratio: ecWidth / ecHeight // lower case for convenience. - - }; - var applicable = true; - each(query, function (value, attr) { - var matched = attr.match(QUERY_REG); - - if (!matched || !matched[1] || !matched[2]) { - return; - } - - var operator = matched[1]; - var realAttr = matched[2].toLowerCase(); - - if (!compare(realMap[realAttr], value, operator)) { - applicable = false; - } - }); - return applicable; - } - - function compare(real, expect, operator) { - if (operator === 'min') { - return real >= expect; - } else if (operator === 'max') { - return real <= expect; - } else { - // Equals - return real === expect; - } - } - - function indicesEquals(indices1, indices2) { - // indices is always order by asc and has only finite number. - return indices1.join(',') === indices2.join(','); - } - - var each$2 = each; - var isObject$1 = isObject; - var POSSIBLE_STYLES = ['areaStyle', 'lineStyle', 'nodeStyle', 'linkStyle', 'chordStyle', 'label', 'labelLine']; - - function compatEC2ItemStyle(opt) { - var itemStyleOpt = opt && opt.itemStyle; - - if (!itemStyleOpt) { - return; - } - - for (var i = 0, len = POSSIBLE_STYLES.length; i < len; i++) { - var styleName = POSSIBLE_STYLES[i]; - var normalItemStyleOpt = itemStyleOpt.normal; - var emphasisItemStyleOpt = itemStyleOpt.emphasis; - - if (normalItemStyleOpt && normalItemStyleOpt[styleName]) { - if ("development" !== 'production') { - deprecateReplaceLog("itemStyle.normal." + styleName, styleName); - } - - opt[styleName] = opt[styleName] || {}; - - if (!opt[styleName].normal) { - opt[styleName].normal = normalItemStyleOpt[styleName]; - } else { - merge(opt[styleName].normal, normalItemStyleOpt[styleName]); - } - - normalItemStyleOpt[styleName] = null; - } - - if (emphasisItemStyleOpt && emphasisItemStyleOpt[styleName]) { - if ("development" !== 'production') { - deprecateReplaceLog("itemStyle.emphasis." + styleName, "emphasis." + styleName); - } - - opt[styleName] = opt[styleName] || {}; - - if (!opt[styleName].emphasis) { - opt[styleName].emphasis = emphasisItemStyleOpt[styleName]; - } else { - merge(opt[styleName].emphasis, emphasisItemStyleOpt[styleName]); - } - - emphasisItemStyleOpt[styleName] = null; - } - } - } - - function convertNormalEmphasis(opt, optType, useExtend) { - if (opt && opt[optType] && (opt[optType].normal || opt[optType].emphasis)) { - var normalOpt = opt[optType].normal; - var emphasisOpt = opt[optType].emphasis; - - if (normalOpt) { - if ("development" !== 'production') { - // eslint-disable-next-line max-len - deprecateLog("'normal' hierarchy in " + optType + " has been removed since 4.0. All style properties are configured in " + optType + " directly now."); - } // Timeline controlStyle has other properties besides normal and emphasis - - - if (useExtend) { - opt[optType].normal = opt[optType].emphasis = null; - defaults(opt[optType], normalOpt); - } else { - opt[optType] = normalOpt; - } - } - - if (emphasisOpt) { - if ("development" !== 'production') { - deprecateLog(optType + ".emphasis has been changed to emphasis." + optType + " since 4.0"); - } - - opt.emphasis = opt.emphasis || {}; - opt.emphasis[optType] = emphasisOpt; // Also compat the case user mix the style and focus together in ec3 style - // for example: { itemStyle: { normal: {}, emphasis: {focus, shadowBlur} } } - - if (emphasisOpt.focus) { - opt.emphasis.focus = emphasisOpt.focus; - } - - if (emphasisOpt.blurScope) { - opt.emphasis.blurScope = emphasisOpt.blurScope; - } - } - } - } - - function removeEC3NormalStatus(opt) { - convertNormalEmphasis(opt, 'itemStyle'); - convertNormalEmphasis(opt, 'lineStyle'); - convertNormalEmphasis(opt, 'areaStyle'); - convertNormalEmphasis(opt, 'label'); - convertNormalEmphasis(opt, 'labelLine'); // treemap - - convertNormalEmphasis(opt, 'upperLabel'); // graph - - convertNormalEmphasis(opt, 'edgeLabel'); - } - - function compatTextStyle(opt, propName) { - // Check whether is not object (string\null\undefined ...) - var labelOptSingle = isObject$1(opt) && opt[propName]; - var textStyle = isObject$1(labelOptSingle) && labelOptSingle.textStyle; - - if (textStyle) { - if ("development" !== 'production') { - // eslint-disable-next-line max-len - deprecateLog("textStyle hierarchy in " + propName + " has been removed since 4.0. All textStyle properties are configured in " + propName + " directly now."); - } - - for (var i = 0, len = TEXT_STYLE_OPTIONS.length; i < len; i++) { - var textPropName = TEXT_STYLE_OPTIONS[i]; - - if (textStyle.hasOwnProperty(textPropName)) { - labelOptSingle[textPropName] = textStyle[textPropName]; - } - } - } - } - - function compatEC3CommonStyles(opt) { - if (opt) { - removeEC3NormalStatus(opt); - compatTextStyle(opt, 'label'); - opt.emphasis && compatTextStyle(opt.emphasis, 'label'); - } - } - - function processSeries(seriesOpt) { - if (!isObject$1(seriesOpt)) { - return; - } - - compatEC2ItemStyle(seriesOpt); - removeEC3NormalStatus(seriesOpt); - compatTextStyle(seriesOpt, 'label'); // treemap - - compatTextStyle(seriesOpt, 'upperLabel'); // graph - - compatTextStyle(seriesOpt, 'edgeLabel'); - - if (seriesOpt.emphasis) { - compatTextStyle(seriesOpt.emphasis, 'label'); // treemap - - compatTextStyle(seriesOpt.emphasis, 'upperLabel'); // graph - - compatTextStyle(seriesOpt.emphasis, 'edgeLabel'); - } - - var markPoint = seriesOpt.markPoint; - - if (markPoint) { - compatEC2ItemStyle(markPoint); - compatEC3CommonStyles(markPoint); - } - - var markLine = seriesOpt.markLine; - - if (markLine) { - compatEC2ItemStyle(markLine); - compatEC3CommonStyles(markLine); - } - - var markArea = seriesOpt.markArea; - - if (markArea) { - compatEC3CommonStyles(markArea); - } - - var data = seriesOpt.data; // Break with ec3: if `setOption` again, there may be no `type` in option, - // then the backward compat based on option type will not be performed. - - if (seriesOpt.type === 'graph') { - data = data || seriesOpt.nodes; - var edgeData = seriesOpt.links || seriesOpt.edges; - - if (edgeData && !isTypedArray(edgeData)) { - for (var i = 0; i < edgeData.length; i++) { - compatEC3CommonStyles(edgeData[i]); - } - } - - each(seriesOpt.categories, function (opt) { - removeEC3NormalStatus(opt); - }); - } - - if (data && !isTypedArray(data)) { - for (var i = 0; i < data.length; i++) { - compatEC3CommonStyles(data[i]); - } - } // mark point data - - - markPoint = seriesOpt.markPoint; - - if (markPoint && markPoint.data) { - var mpData = markPoint.data; - - for (var i = 0; i < mpData.length; i++) { - compatEC3CommonStyles(mpData[i]); - } - } // mark line data - - - markLine = seriesOpt.markLine; - - if (markLine && markLine.data) { - var mlData = markLine.data; - - for (var i = 0; i < mlData.length; i++) { - if (isArray(mlData[i])) { - compatEC3CommonStyles(mlData[i][0]); - compatEC3CommonStyles(mlData[i][1]); - } else { - compatEC3CommonStyles(mlData[i]); - } - } - } // Series - - - if (seriesOpt.type === 'gauge') { - compatTextStyle(seriesOpt, 'axisLabel'); - compatTextStyle(seriesOpt, 'title'); - compatTextStyle(seriesOpt, 'detail'); - } else if (seriesOpt.type === 'treemap') { - convertNormalEmphasis(seriesOpt.breadcrumb, 'itemStyle'); - each(seriesOpt.levels, function (opt) { - removeEC3NormalStatus(opt); - }); - } else if (seriesOpt.type === 'tree') { - removeEC3NormalStatus(seriesOpt.leaves); - } // sunburst starts from ec4, so it does not need to compat levels. - - } - - function toArr(o) { - return isArray(o) ? o : o ? [o] : []; - } - - function toObj(o) { - return (isArray(o) ? o[0] : o) || {}; - } - - function globalCompatStyle(option, isTheme) { - each$2(toArr(option.series), function (seriesOpt) { - isObject$1(seriesOpt) && processSeries(seriesOpt); - }); - var axes = ['xAxis', 'yAxis', 'radiusAxis', 'angleAxis', 'singleAxis', 'parallelAxis', 'radar']; - isTheme && axes.push('valueAxis', 'categoryAxis', 'logAxis', 'timeAxis'); - each$2(axes, function (axisName) { - each$2(toArr(option[axisName]), function (axisOpt) { - if (axisOpt) { - compatTextStyle(axisOpt, 'axisLabel'); - compatTextStyle(axisOpt.axisPointer, 'label'); - } - }); - }); - each$2(toArr(option.parallel), function (parallelOpt) { - var parallelAxisDefault = parallelOpt && parallelOpt.parallelAxisDefault; - compatTextStyle(parallelAxisDefault, 'axisLabel'); - compatTextStyle(parallelAxisDefault && parallelAxisDefault.axisPointer, 'label'); - }); - each$2(toArr(option.calendar), function (calendarOpt) { - convertNormalEmphasis(calendarOpt, 'itemStyle'); - compatTextStyle(calendarOpt, 'dayLabel'); - compatTextStyle(calendarOpt, 'monthLabel'); - compatTextStyle(calendarOpt, 'yearLabel'); - }); // radar.name.textStyle - - each$2(toArr(option.radar), function (radarOpt) { - compatTextStyle(radarOpt, 'name'); // Use axisName instead of name because component has name property - - if (radarOpt.name && radarOpt.axisName == null) { - radarOpt.axisName = radarOpt.name; - delete radarOpt.name; - - if ("development" !== 'production') { - deprecateLog('name property in radar component has been changed to axisName'); - } - } - - if (radarOpt.nameGap != null && radarOpt.axisNameGap == null) { - radarOpt.axisNameGap = radarOpt.nameGap; - delete radarOpt.nameGap; - - if ("development" !== 'production') { - deprecateLog('nameGap property in radar component has been changed to axisNameGap'); - } - } - - if ("development" !== 'production') { - each$2(radarOpt.indicator, function (indicatorOpt) { - if (indicatorOpt.text) { - deprecateReplaceLog('text', 'name', 'radar.indicator'); - } - }); - } - }); - each$2(toArr(option.geo), function (geoOpt) { - if (isObject$1(geoOpt)) { - compatEC3CommonStyles(geoOpt); - each$2(toArr(geoOpt.regions), function (regionObj) { - compatEC3CommonStyles(regionObj); - }); - } - }); - each$2(toArr(option.timeline), function (timelineOpt) { - compatEC3CommonStyles(timelineOpt); - convertNormalEmphasis(timelineOpt, 'label'); - convertNormalEmphasis(timelineOpt, 'itemStyle'); - convertNormalEmphasis(timelineOpt, 'controlStyle', true); - var data = timelineOpt.data; - isArray(data) && each(data, function (item) { - if (isObject(item)) { - convertNormalEmphasis(item, 'label'); - convertNormalEmphasis(item, 'itemStyle'); - } - }); - }); - each$2(toArr(option.toolbox), function (toolboxOpt) { - convertNormalEmphasis(toolboxOpt, 'iconStyle'); - each$2(toolboxOpt.feature, function (featureOpt) { - convertNormalEmphasis(featureOpt, 'iconStyle'); - }); - }); - compatTextStyle(toObj(option.axisPointer), 'label'); - compatTextStyle(toObj(option.tooltip).axisPointer, 'label'); // Clean logs - // storedLogs = {}; - } - - function get(opt, path) { - var pathArr = path.split(','); - var obj = opt; - - for (var i = 0; i < pathArr.length; i++) { - obj = obj && obj[pathArr[i]]; - - if (obj == null) { - break; - } - } - - return obj; - } - - function set$1(opt, path, val, overwrite) { - var pathArr = path.split(','); - var obj = opt; - var key; - var i = 0; - - for (; i < pathArr.length - 1; i++) { - key = pathArr[i]; - - if (obj[key] == null) { - obj[key] = {}; - } - - obj = obj[key]; - } - - if (overwrite || obj[pathArr[i]] == null) { - obj[pathArr[i]] = val; - } - } - - function compatLayoutProperties(option) { - option && each(LAYOUT_PROPERTIES, function (prop) { - if (prop[0] in option && !(prop[1] in option)) { - option[prop[1]] = option[prop[0]]; - } - }); - } - - var LAYOUT_PROPERTIES = [['x', 'left'], ['y', 'top'], ['x2', 'right'], ['y2', 'bottom']]; - var COMPATITABLE_COMPONENTS = ['grid', 'geo', 'parallel', 'legend', 'toolbox', 'title', 'visualMap', 'dataZoom', 'timeline']; - var BAR_ITEM_STYLE_MAP = [['borderRadius', 'barBorderRadius'], ['borderColor', 'barBorderColor'], ['borderWidth', 'barBorderWidth']]; - - function compatBarItemStyle(option) { - var itemStyle = option && option.itemStyle; - - if (itemStyle) { - for (var i = 0; i < BAR_ITEM_STYLE_MAP.length; i++) { - var oldName = BAR_ITEM_STYLE_MAP[i][1]; - var newName = BAR_ITEM_STYLE_MAP[i][0]; - - if (itemStyle[oldName] != null) { - itemStyle[newName] = itemStyle[oldName]; - - if ("development" !== 'production') { - deprecateReplaceLog(oldName, newName); - } - } - } - } - } - - function compatPieLabel(option) { - if (!option) { - return; - } - - if (option.alignTo === 'edge' && option.margin != null && option.edgeDistance == null) { - if ("development" !== 'production') { - deprecateReplaceLog('label.margin', 'label.edgeDistance', 'pie'); - } - - option.edgeDistance = option.margin; - } - } - - function compatSunburstState(option) { - if (!option) { - return; - } - - if (option.downplay && !option.blur) { - option.blur = option.downplay; - - if ("development" !== 'production') { - deprecateReplaceLog('downplay', 'blur', 'sunburst'); - } - } - } - - function compatGraphFocus(option) { - if (!option) { - return; - } - - if (option.focusNodeAdjacency != null) { - option.emphasis = option.emphasis || {}; - - if (option.emphasis.focus == null) { - if ("development" !== 'production') { - deprecateReplaceLog('focusNodeAdjacency', 'emphasis: { focus: \'adjacency\'}', 'graph/sankey'); - } - - option.emphasis.focus = 'adjacency'; - } - } - } - - function traverseTree(data, cb) { - if (data) { - for (var i = 0; i < data.length; i++) { - cb(data[i]); - data[i] && traverseTree(data[i].children, cb); - } - } - } - - function globalBackwardCompat(option, isTheme) { - globalCompatStyle(option, isTheme); // Make sure series array for model initialization. - - option.series = normalizeToArray(option.series); - each(option.series, function (seriesOpt) { - if (!isObject(seriesOpt)) { - return; - } - - var seriesType = seriesOpt.type; - - if (seriesType === 'line') { - if (seriesOpt.clipOverflow != null) { - seriesOpt.clip = seriesOpt.clipOverflow; - - if ("development" !== 'production') { - deprecateReplaceLog('clipOverflow', 'clip', 'line'); - } - } - } else if (seriesType === 'pie' || seriesType === 'gauge') { - if (seriesOpt.clockWise != null) { - seriesOpt.clockwise = seriesOpt.clockWise; - - if ("development" !== 'production') { - deprecateReplaceLog('clockWise', 'clockwise'); - } - } - - compatPieLabel(seriesOpt.label); - var data = seriesOpt.data; - - if (data && !isTypedArray(data)) { - for (var i = 0; i < data.length; i++) { - compatPieLabel(data[i]); - } - } - - if (seriesOpt.hoverOffset != null) { - seriesOpt.emphasis = seriesOpt.emphasis || {}; - - if (seriesOpt.emphasis.scaleSize = null) { - if ("development" !== 'production') { - deprecateReplaceLog('hoverOffset', 'emphasis.scaleSize'); - } - - seriesOpt.emphasis.scaleSize = seriesOpt.hoverOffset; - } - } - } else if (seriesType === 'gauge') { - var pointerColor = get(seriesOpt, 'pointer.color'); - pointerColor != null && set$1(seriesOpt, 'itemStyle.color', pointerColor); - } else if (seriesType === 'bar') { - compatBarItemStyle(seriesOpt); - compatBarItemStyle(seriesOpt.backgroundStyle); - compatBarItemStyle(seriesOpt.emphasis); - var data = seriesOpt.data; - - if (data && !isTypedArray(data)) { - for (var i = 0; i < data.length; i++) { - if (typeof data[i] === 'object') { - compatBarItemStyle(data[i]); - compatBarItemStyle(data[i] && data[i].emphasis); - } - } - } - } else if (seriesType === 'sunburst') { - var highlightPolicy = seriesOpt.highlightPolicy; - - if (highlightPolicy) { - seriesOpt.emphasis = seriesOpt.emphasis || {}; - - if (!seriesOpt.emphasis.focus) { - seriesOpt.emphasis.focus = highlightPolicy; - - if ("development" !== 'production') { - deprecateReplaceLog('highlightPolicy', 'emphasis.focus', 'sunburst'); - } - } - } - - compatSunburstState(seriesOpt); - traverseTree(seriesOpt.data, compatSunburstState); - } else if (seriesType === 'graph' || seriesType === 'sankey') { - compatGraphFocus(seriesOpt); // TODO nodes, edges? - } else if (seriesType === 'map') { - if (seriesOpt.mapType && !seriesOpt.map) { - if ("development" !== 'production') { - deprecateReplaceLog('mapType', 'map', 'map'); - } - - seriesOpt.map = seriesOpt.mapType; - } - - if (seriesOpt.mapLocation) { - if ("development" !== 'production') { - deprecateLog('`mapLocation` is not used anymore.'); - } - - defaults(seriesOpt, seriesOpt.mapLocation); - } - } - - if (seriesOpt.hoverAnimation != null) { - seriesOpt.emphasis = seriesOpt.emphasis || {}; - - if (seriesOpt.emphasis && seriesOpt.emphasis.scale == null) { - if ("development" !== 'production') { - deprecateReplaceLog('hoverAnimation', 'emphasis.scale'); - } - - seriesOpt.emphasis.scale = seriesOpt.hoverAnimation; - } - } - - compatLayoutProperties(seriesOpt); - }); // dataRange has changed to visualMap - - if (option.dataRange) { - option.visualMap = option.dataRange; - } - - each(COMPATITABLE_COMPONENTS, function (componentName) { - var options = option[componentName]; - - if (options) { - if (!isArray(options)) { - options = [options]; - } - - each(options, function (option) { - compatLayoutProperties(option); - }); - } - }); - } - - // data processing stage is blocked in stream. - // See <module:echarts/stream/Scheduler#performDataProcessorTasks> - // (2) Only register once when import repeatedly. - // Should be executed after series is filtered and before stack calculation. - - function dataStack(ecModel) { - var stackInfoMap = createHashMap(); - ecModel.eachSeries(function (seriesModel) { - var stack = seriesModel.get('stack'); // Compatible: when `stack` is set as '', do not stack. - - if (stack) { - var stackInfoList = stackInfoMap.get(stack) || stackInfoMap.set(stack, []); - var data = seriesModel.getData(); - var stackInfo = { - // Used for calculate axis extent automatically. - // TODO: Type getCalculationInfo return more specific type? - stackResultDimension: data.getCalculationInfo('stackResultDimension'), - stackedOverDimension: data.getCalculationInfo('stackedOverDimension'), - stackedDimension: data.getCalculationInfo('stackedDimension'), - stackedByDimension: data.getCalculationInfo('stackedByDimension'), - isStackedByIndex: data.getCalculationInfo('isStackedByIndex'), - data: data, - seriesModel: seriesModel - }; // If stacked on axis that do not support data stack. - - if (!stackInfo.stackedDimension || !(stackInfo.isStackedByIndex || stackInfo.stackedByDimension)) { - return; - } - - stackInfoList.length && data.setCalculationInfo('stackedOnSeries', stackInfoList[stackInfoList.length - 1].seriesModel); - stackInfoList.push(stackInfo); - } - }); - stackInfoMap.each(calculateStack); - } - - function calculateStack(stackInfoList) { - each(stackInfoList, function (targetStackInfo, idxInStack) { - var resultVal = []; - var resultNaN = [NaN, NaN]; - var dims = [targetStackInfo.stackResultDimension, targetStackInfo.stackedOverDimension]; - var targetData = targetStackInfo.data; - var isStackedByIndex = targetStackInfo.isStackedByIndex; - var stackStrategy = targetStackInfo.seriesModel.get('stackStrategy') || 'samesign'; // Should not write on raw data, because stack series model list changes - // depending on legend selection. - - targetData.modify(dims, function (v0, v1, dataIndex) { - var sum = targetData.get(targetStackInfo.stackedDimension, dataIndex); // Consider `connectNulls` of line area, if value is NaN, stackedOver - // should also be NaN, to draw a appropriate belt area. - - if (isNaN(sum)) { - return resultNaN; - } - - var byValue; - var stackedDataRawIndex; - - if (isStackedByIndex) { - stackedDataRawIndex = targetData.getRawIndex(dataIndex); - } else { - byValue = targetData.get(targetStackInfo.stackedByDimension, dataIndex); - } // If stackOver is NaN, chart view will render point on value start. - - - var stackedOver = NaN; - - for (var j = idxInStack - 1; j >= 0; j--) { - var stackInfo = stackInfoList[j]; // Has been optimized by inverted indices on `stackedByDimension`. - - if (!isStackedByIndex) { - stackedDataRawIndex = stackInfo.data.rawIndexOf(stackInfo.stackedByDimension, byValue); - } - - if (stackedDataRawIndex >= 0) { - var val = stackInfo.data.getByRawIndex(stackInfo.stackResultDimension, stackedDataRawIndex); // Considering positive stack, negative stack and empty data - - if (stackStrategy === 'all' // single stack group - || stackStrategy === 'positive' && val > 0 || stackStrategy === 'negative' && val < 0 || stackStrategy === 'samesign' && sum >= 0 && val > 0 // All positive stack - || stackStrategy === 'samesign' && sum <= 0 && val < 0 // All negative stack - ) { - // The sum has to be very small to be affected by the - // floating arithmetic problem. An incorrect result will probably - // cause axis min/max to be filtered incorrectly. - sum = addSafe(sum, val); - stackedOver = val; - break; - } - } - } - - resultVal[0] = sum; - resultVal[1] = stackedOver; - return resultVal; - }); - }); - } - - var SourceImpl = - /** @class */ - function () { - function SourceImpl(fields) { - this.data = fields.data || (fields.sourceFormat === SOURCE_FORMAT_KEYED_COLUMNS ? {} : []); - this.sourceFormat = fields.sourceFormat || SOURCE_FORMAT_UNKNOWN; // Visit config - - this.seriesLayoutBy = fields.seriesLayoutBy || SERIES_LAYOUT_BY_COLUMN; - this.startIndex = fields.startIndex || 0; - this.dimensionsDetectedCount = fields.dimensionsDetectedCount; - this.metaRawOption = fields.metaRawOption; - var dimensionsDefine = this.dimensionsDefine = fields.dimensionsDefine; - - if (dimensionsDefine) { - for (var i = 0; i < dimensionsDefine.length; i++) { - var dim = dimensionsDefine[i]; - - if (dim.type == null) { - if (guessOrdinal(this, i) === BE_ORDINAL.Must) { - dim.type = 'ordinal'; - } - } - } - } - } - - return SourceImpl; - }(); - - function isSourceInstance(val) { - return val instanceof SourceImpl; - } - /** - * Create a source from option. - * NOTE: Created source is immutable. Don't change any properties in it. - */ - - function createSource(sourceData, thisMetaRawOption, // can be null. If not provided, auto detect it from `sourceData`. - sourceFormat) { - sourceFormat = sourceFormat || detectSourceFormat(sourceData); - var seriesLayoutBy = thisMetaRawOption.seriesLayoutBy; - var determined = determineSourceDimensions(sourceData, sourceFormat, seriesLayoutBy, thisMetaRawOption.sourceHeader, thisMetaRawOption.dimensions); - var source = new SourceImpl({ - data: sourceData, - sourceFormat: sourceFormat, - seriesLayoutBy: seriesLayoutBy, - dimensionsDefine: determined.dimensionsDefine, - startIndex: determined.startIndex, - dimensionsDetectedCount: determined.dimensionsDetectedCount, - metaRawOption: clone(thisMetaRawOption) - }); - return source; - } - /** - * Wrap original series data for some compatibility cases. - */ - - function createSourceFromSeriesDataOption(data) { - return new SourceImpl({ - data: data, - sourceFormat: isTypedArray(data) ? SOURCE_FORMAT_TYPED_ARRAY : SOURCE_FORMAT_ORIGINAL - }); - } - /** - * Clone source but excludes source data. - */ - - function cloneSourceShallow(source) { - return new SourceImpl({ - data: source.data, - sourceFormat: source.sourceFormat, - seriesLayoutBy: source.seriesLayoutBy, - dimensionsDefine: clone(source.dimensionsDefine), - startIndex: source.startIndex, - dimensionsDetectedCount: source.dimensionsDetectedCount - }); - } - /** - * Note: An empty array will be detected as `SOURCE_FORMAT_ARRAY_ROWS`. - */ - - function detectSourceFormat(data) { - var sourceFormat = SOURCE_FORMAT_UNKNOWN; - - if (isTypedArray(data)) { - sourceFormat = SOURCE_FORMAT_TYPED_ARRAY; - } else if (isArray(data)) { - // FIXME Whether tolerate null in top level array? - if (data.length === 0) { - sourceFormat = SOURCE_FORMAT_ARRAY_ROWS; - } - - for (var i = 0, len = data.length; i < len; i++) { - var item = data[i]; - - if (item == null) { - continue; - } else if (isArray(item)) { - sourceFormat = SOURCE_FORMAT_ARRAY_ROWS; - break; - } else if (isObject(item)) { - sourceFormat = SOURCE_FORMAT_OBJECT_ROWS; - break; - } - } - } else if (isObject(data)) { - for (var key in data) { - if (hasOwn(data, key) && isArrayLike(data[key])) { - sourceFormat = SOURCE_FORMAT_KEYED_COLUMNS; - break; - } - } - } - - return sourceFormat; - } - /** - * Determine the source definitions from data standalone dimensions definitions - * are not specified. - */ - - function determineSourceDimensions(data, sourceFormat, seriesLayoutBy, sourceHeader, // standalone raw dimensions definition, like: - // { - // dimensions: ['aa', 'bb', { name: 'cc', type: 'time' }] - // } - // in `dataset` or `series` - dimensionsDefine) { - var dimensionsDetectedCount; - var startIndex; // PENDING: Could data be null/undefined here? - // currently, if `dataset.source` not specified, error thrown. - // if `series.data` not specified, nothing rendered without error thrown. - // Should test these cases. - - if (!data) { - return { - dimensionsDefine: normalizeDimensionsOption(dimensionsDefine), - startIndex: startIndex, - dimensionsDetectedCount: dimensionsDetectedCount - }; - } - - if (sourceFormat === SOURCE_FORMAT_ARRAY_ROWS) { - var dataArrayRows = data; // Rule: Most of the first line are string: it is header. - // Caution: consider a line with 5 string and 1 number, - // it still can not be sure it is a head, because the - // 5 string may be 5 values of category columns. - - if (sourceHeader === 'auto' || sourceHeader == null) { - arrayRowsTravelFirst(function (val) { - // '-' is regarded as null/undefined. - if (val != null && val !== '-') { - if (isString(val)) { - startIndex == null && (startIndex = 1); - } else { - startIndex = 0; - } - } // 10 is an experience number, avoid long loop. - - }, seriesLayoutBy, dataArrayRows, 10); - } else { - startIndex = isNumber(sourceHeader) ? sourceHeader : sourceHeader ? 1 : 0; - } - - if (!dimensionsDefine && startIndex === 1) { - dimensionsDefine = []; - arrayRowsTravelFirst(function (val, index) { - dimensionsDefine[index] = val != null ? val + '' : ''; - }, seriesLayoutBy, dataArrayRows, Infinity); - } - - dimensionsDetectedCount = dimensionsDefine ? dimensionsDefine.length : seriesLayoutBy === SERIES_LAYOUT_BY_ROW ? dataArrayRows.length : dataArrayRows[0] ? dataArrayRows[0].length : null; - } else if (sourceFormat === SOURCE_FORMAT_OBJECT_ROWS) { - if (!dimensionsDefine) { - dimensionsDefine = objectRowsCollectDimensions(data); - } - } else if (sourceFormat === SOURCE_FORMAT_KEYED_COLUMNS) { - if (!dimensionsDefine) { - dimensionsDefine = []; - each(data, function (colArr, key) { - dimensionsDefine.push(key); - }); - } - } else if (sourceFormat === SOURCE_FORMAT_ORIGINAL) { - var value0 = getDataItemValue(data[0]); - dimensionsDetectedCount = isArray(value0) && value0.length || 1; - } else if (sourceFormat === SOURCE_FORMAT_TYPED_ARRAY) { - if ("development" !== 'production') { - assert(!!dimensionsDefine, 'dimensions must be given if data is TypedArray.'); - } - } - - return { - startIndex: startIndex, - dimensionsDefine: normalizeDimensionsOption(dimensionsDefine), - dimensionsDetectedCount: dimensionsDetectedCount - }; - } - - function objectRowsCollectDimensions(data) { - var firstIndex = 0; - var obj; - - while (firstIndex < data.length && !(obj = data[firstIndex++])) {} // jshint ignore: line - - - if (obj) { - return keys(obj); - } - } // Consider dimensions defined like ['A', 'price', 'B', 'price', 'C', 'price'], - // which is reasonable. But dimension name is duplicated. - // Returns undefined or an array contains only object without null/undefined or string. - - - function normalizeDimensionsOption(dimensionsDefine) { - if (!dimensionsDefine) { - // The meaning of null/undefined is different from empty array. - return; - } - - var nameMap = createHashMap(); - return map(dimensionsDefine, function (rawItem, index) { - rawItem = isObject(rawItem) ? rawItem : { - name: rawItem - }; // Other fields will be discarded. - - var item = { - name: rawItem.name, - displayName: rawItem.displayName, - type: rawItem.type - }; // User can set null in dimensions. - // We don't auto specify name, otherwise a given name may - // cause it to be referred unexpectedly. - - if (item.name == null) { - return item; - } // Also consider number form like 2012. - - - item.name += ''; // User may also specify displayName. - // displayName will always exists except user not - // specified or dim name is not specified or detected. - // (A auto generated dim name will not be used as - // displayName). - - if (item.displayName == null) { - item.displayName = item.name; - } - - var exist = nameMap.get(item.name); - - if (!exist) { - nameMap.set(item.name, { - count: 1 - }); - } else { - item.name += '-' + exist.count++; - } - - return item; - }); - } - - function arrayRowsTravelFirst(cb, seriesLayoutBy, data, maxLoop) { - if (seriesLayoutBy === SERIES_LAYOUT_BY_ROW) { - for (var i = 0; i < data.length && i < maxLoop; i++) { - cb(data[i] ? data[i][0] : null, i); - } - } else { - var value0 = data[0] || []; - - for (var i = 0; i < value0.length && i < maxLoop; i++) { - cb(value0[i], i); - } - } - } - - function shouldRetrieveDataByName(source) { - var sourceFormat = source.sourceFormat; - return sourceFormat === SOURCE_FORMAT_OBJECT_ROWS || sourceFormat === SOURCE_FORMAT_KEYED_COLUMNS; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var _a, _b, _c; // TODO - var providerMethods; - var mountMethods; - /** - * If normal array used, mutable chunk size is supported. - * If typed array used, chunk size must be fixed. - */ - - var DefaultDataProvider = - /** @class */ - function () { - function DefaultDataProvider(sourceParam, dimSize) { - // let source: Source; - var source = !isSourceInstance(sourceParam) ? createSourceFromSeriesDataOption(sourceParam) : sourceParam; // declare source is Source; - - this._source = source; - var data = this._data = source.data; // Typed array. TODO IE10+? - - if (source.sourceFormat === SOURCE_FORMAT_TYPED_ARRAY) { - if ("development" !== 'production') { - if (dimSize == null) { - throw new Error('Typed array data must specify dimension size'); - } - } - - this._offset = 0; - this._dimSize = dimSize; - this._data = data; - } - - mountMethods(this, data, source); - } - - DefaultDataProvider.prototype.getSource = function () { - return this._source; - }; - - DefaultDataProvider.prototype.count = function () { - return 0; - }; - - DefaultDataProvider.prototype.getItem = function (idx, out) { - return; - }; - - DefaultDataProvider.prototype.appendData = function (newData) {}; - - DefaultDataProvider.prototype.clean = function () {}; - - DefaultDataProvider.protoInitialize = function () { - // PENDING: To avoid potential incompat (e.g., prototype - // is visited somewhere), still init them on prototype. - var proto = DefaultDataProvider.prototype; - proto.pure = false; - proto.persistent = true; - }(); - - DefaultDataProvider.internalField = function () { - var _a; - - mountMethods = function (provider, data, source) { - var sourceFormat = source.sourceFormat; - var seriesLayoutBy = source.seriesLayoutBy; - var startIndex = source.startIndex; - var dimsDef = source.dimensionsDefine; - var methods = providerMethods[getMethodMapKey(sourceFormat, seriesLayoutBy)]; - - if ("development" !== 'production') { - assert(methods, 'Invalide sourceFormat: ' + sourceFormat); - } - - extend(provider, methods); - - if (sourceFormat === SOURCE_FORMAT_TYPED_ARRAY) { - provider.getItem = getItemForTypedArray; - provider.count = countForTypedArray; - provider.fillStorage = fillStorageForTypedArray; - } else { - var rawItemGetter = getRawSourceItemGetter(sourceFormat, seriesLayoutBy); - provider.getItem = bind(rawItemGetter, null, data, startIndex, dimsDef); - var rawCounter = getRawSourceDataCounter(sourceFormat, seriesLayoutBy); - provider.count = bind(rawCounter, null, data, startIndex, dimsDef); - } - }; - - var getItemForTypedArray = function (idx, out) { - idx = idx - this._offset; - out = out || []; - var data = this._data; - var dimSize = this._dimSize; - var offset = dimSize * idx; - - for (var i = 0; i < dimSize; i++) { - out[i] = data[offset + i]; - } - - return out; - }; - - var fillStorageForTypedArray = function (start, end, storage, extent) { - var data = this._data; - var dimSize = this._dimSize; - - for (var dim = 0; dim < dimSize; dim++) { - var dimExtent = extent[dim]; - var min = dimExtent[0] == null ? Infinity : dimExtent[0]; - var max = dimExtent[1] == null ? -Infinity : dimExtent[1]; - var count = end - start; - var arr = storage[dim]; - - for (var i = 0; i < count; i++) { - // appendData with TypedArray will always do replace in provider. - var val = data[i * dimSize + dim]; - arr[start + i] = val; - val < min && (min = val); - val > max && (max = val); - } - - dimExtent[0] = min; - dimExtent[1] = max; - } - }; - - var countForTypedArray = function () { - return this._data ? this._data.length / this._dimSize : 0; - }; - - providerMethods = (_a = {}, _a[SOURCE_FORMAT_ARRAY_ROWS + '_' + SERIES_LAYOUT_BY_COLUMN] = { - pure: true, - appendData: appendDataSimply - }, _a[SOURCE_FORMAT_ARRAY_ROWS + '_' + SERIES_LAYOUT_BY_ROW] = { - pure: true, - appendData: function () { - throw new Error('Do not support appendData when set seriesLayoutBy: "row".'); - } - }, _a[SOURCE_FORMAT_OBJECT_ROWS] = { - pure: true, - appendData: appendDataSimply - }, _a[SOURCE_FORMAT_KEYED_COLUMNS] = { - pure: true, - appendData: function (newData) { - var data = this._data; - each(newData, function (newCol, key) { - var oldCol = data[key] || (data[key] = []); - - for (var i = 0; i < (newCol || []).length; i++) { - oldCol.push(newCol[i]); - } - }); - } - }, _a[SOURCE_FORMAT_ORIGINAL] = { - appendData: appendDataSimply - }, _a[SOURCE_FORMAT_TYPED_ARRAY] = { - persistent: false, - pure: true, - appendData: function (newData) { - if ("development" !== 'production') { - assert(isTypedArray(newData), 'Added data must be TypedArray if data in initialization is TypedArray'); - } - - this._data = newData; - }, - // Clean self if data is already used. - clean: function () { - // PENDING - this._offset += this.count(); - this._data = null; - } - }, _a); - - function appendDataSimply(newData) { - for (var i = 0; i < newData.length; i++) { - this._data.push(newData[i]); - } - } - }(); - - return DefaultDataProvider; - }(); - - var getItemSimply = function (rawData, startIndex, dimsDef, idx) { - return rawData[idx]; - }; - - var rawSourceItemGetterMap = (_a = {}, _a[SOURCE_FORMAT_ARRAY_ROWS + '_' + SERIES_LAYOUT_BY_COLUMN] = function (rawData, startIndex, dimsDef, idx) { - return rawData[idx + startIndex]; - }, _a[SOURCE_FORMAT_ARRAY_ROWS + '_' + SERIES_LAYOUT_BY_ROW] = function (rawData, startIndex, dimsDef, idx, out) { - idx += startIndex; - var item = out || []; - var data = rawData; - - for (var i = 0; i < data.length; i++) { - var row = data[i]; - item[i] = row ? row[idx] : null; - } - - return item; - }, _a[SOURCE_FORMAT_OBJECT_ROWS] = getItemSimply, _a[SOURCE_FORMAT_KEYED_COLUMNS] = function (rawData, startIndex, dimsDef, idx, out) { - var item = out || []; - - for (var i = 0; i < dimsDef.length; i++) { - var dimName = dimsDef[i].name; - - if ("development" !== 'production') { - if (dimName == null) { - throw new Error(); - } - } - - var col = rawData[dimName]; - item[i] = col ? col[idx] : null; - } - - return item; - }, _a[SOURCE_FORMAT_ORIGINAL] = getItemSimply, _a); - function getRawSourceItemGetter(sourceFormat, seriesLayoutBy) { - var method = rawSourceItemGetterMap[getMethodMapKey(sourceFormat, seriesLayoutBy)]; - - if ("development" !== 'production') { - assert(method, 'Do not support get item on "' + sourceFormat + '", "' + seriesLayoutBy + '".'); - } - - return method; - } - - var countSimply = function (rawData, startIndex, dimsDef) { - return rawData.length; - }; - - var rawSourceDataCounterMap = (_b = {}, _b[SOURCE_FORMAT_ARRAY_ROWS + '_' + SERIES_LAYOUT_BY_COLUMN] = function (rawData, startIndex, dimsDef) { - return Math.max(0, rawData.length - startIndex); - }, _b[SOURCE_FORMAT_ARRAY_ROWS + '_' + SERIES_LAYOUT_BY_ROW] = function (rawData, startIndex, dimsDef) { - var row = rawData[0]; - return row ? Math.max(0, row.length - startIndex) : 0; - }, _b[SOURCE_FORMAT_OBJECT_ROWS] = countSimply, _b[SOURCE_FORMAT_KEYED_COLUMNS] = function (rawData, startIndex, dimsDef) { - var dimName = dimsDef[0].name; - - if ("development" !== 'production') { - if (dimName == null) { - throw new Error(); - } - } - - var col = rawData[dimName]; - return col ? col.length : 0; - }, _b[SOURCE_FORMAT_ORIGINAL] = countSimply, _b); - function getRawSourceDataCounter(sourceFormat, seriesLayoutBy) { - var method = rawSourceDataCounterMap[getMethodMapKey(sourceFormat, seriesLayoutBy)]; - - if ("development" !== 'production') { - assert(method, 'Do not support count on "' + sourceFormat + '", "' + seriesLayoutBy + '".'); - } - - return method; - } - - var getRawValueSimply = function (dataItem, dimIndex, property) { - return dataItem[dimIndex]; - }; - - var rawSourceValueGetterMap = (_c = {}, _c[SOURCE_FORMAT_ARRAY_ROWS] = getRawValueSimply, _c[SOURCE_FORMAT_OBJECT_ROWS] = function (dataItem, dimIndex, property) { - return dataItem[property]; - }, _c[SOURCE_FORMAT_KEYED_COLUMNS] = getRawValueSimply, _c[SOURCE_FORMAT_ORIGINAL] = function (dataItem, dimIndex, property) { - // FIXME: In some case (markpoint in geo (geo-map.html)), - // dataItem is {coord: [...]} - var value = getDataItemValue(dataItem); - return !(value instanceof Array) ? value : value[dimIndex]; - }, _c[SOURCE_FORMAT_TYPED_ARRAY] = getRawValueSimply, _c); - function getRawSourceValueGetter(sourceFormat) { - var method = rawSourceValueGetterMap[sourceFormat]; - - if ("development" !== 'production') { - assert(method, 'Do not support get value on "' + sourceFormat + '".'); - } - - return method; - } - - function getMethodMapKey(sourceFormat, seriesLayoutBy) { - return sourceFormat === SOURCE_FORMAT_ARRAY_ROWS ? sourceFormat + '_' + seriesLayoutBy : sourceFormat; - } // ??? FIXME can these logic be more neat: getRawValue, getRawDataItem, - // Consider persistent. - // Caution: why use raw value to display on label or tooltip? - // A reason is to avoid format. For example time value we do not know - // how to format is expected. More over, if stack is used, calculated - // value may be 0.91000000001, which have brings trouble to display. - // TODO: consider how to treat null/undefined/NaN when display? - - - function retrieveRawValue(data, dataIndex, // If dimIndex is null/undefined, return OptionDataItem. - // Otherwise, return OptionDataValue. - dim) { - if (!data) { - return; - } // Consider data may be not persistent. - - - var dataItem = data.getRawDataItem(dataIndex); - - if (dataItem == null) { - return; - } - - var store = data.getStore(); - var sourceFormat = store.getSource().sourceFormat; - - if (dim != null) { - var dimIndex = data.getDimensionIndex(dim); - var property = store.getDimensionProperty(dimIndex); - return getRawSourceValueGetter(sourceFormat)(dataItem, dimIndex, property); - } else { - var result = dataItem; - - if (sourceFormat === SOURCE_FORMAT_ORIGINAL) { - result = getDataItemValue(dataItem); - } - - return result; - } - } - - var DIMENSION_LABEL_REG = /\{@(.+?)\}/g; - - var DataFormatMixin = - /** @class */ - function () { - function DataFormatMixin() {} - /** - * Get params for formatter - */ - - - DataFormatMixin.prototype.getDataParams = function (dataIndex, dataType) { - var data = this.getData(dataType); - var rawValue = this.getRawValue(dataIndex, dataType); - var rawDataIndex = data.getRawIndex(dataIndex); - var name = data.getName(dataIndex); - var itemOpt = data.getRawDataItem(dataIndex); - var style = data.getItemVisual(dataIndex, 'style'); - var color = style && style[data.getItemVisual(dataIndex, 'drawType') || 'fill']; - var borderColor = style && style.stroke; - var mainType = this.mainType; - var isSeries = mainType === 'series'; - var userOutput = data.userOutput && data.userOutput.get(); - return { - componentType: mainType, - componentSubType: this.subType, - componentIndex: this.componentIndex, - seriesType: isSeries ? this.subType : null, - seriesIndex: this.seriesIndex, - seriesId: isSeries ? this.id : null, - seriesName: isSeries ? this.name : null, - name: name, - dataIndex: rawDataIndex, - data: itemOpt, - dataType: dataType, - value: rawValue, - color: color, - borderColor: borderColor, - dimensionNames: userOutput ? userOutput.fullDimensions : null, - encode: userOutput ? userOutput.encode : null, - // Param name list for mapping `a`, `b`, `c`, `d`, `e` - $vars: ['seriesName', 'name', 'value'] - }; - }; - /** - * Format label - * @param dataIndex - * @param status 'normal' by default - * @param dataType - * @param labelDimIndex Only used in some chart that - * use formatter in different dimensions, like radar. - * @param formatter Formatter given outside. - * @return return null/undefined if no formatter - */ - - - DataFormatMixin.prototype.getFormattedLabel = function (dataIndex, status, dataType, labelDimIndex, formatter, extendParams) { - status = status || 'normal'; - var data = this.getData(dataType); - var params = this.getDataParams(dataIndex, dataType); - - if (extendParams) { - params.value = extendParams.interpolatedValue; - } - - if (labelDimIndex != null && isArray(params.value)) { - params.value = params.value[labelDimIndex]; - } - - if (!formatter) { - var itemModel = data.getItemModel(dataIndex); // @ts-ignore - - formatter = itemModel.get(status === 'normal' ? ['label', 'formatter'] : [status, 'label', 'formatter']); - } - - if (isFunction(formatter)) { - params.status = status; - params.dimensionIndex = labelDimIndex; - return formatter(params); - } else if (isString(formatter)) { - var str = formatTpl(formatter, params); // Support 'aaa{@[3]}bbb{@product}ccc'. - // Do not support '}' in dim name util have to. - - return str.replace(DIMENSION_LABEL_REG, function (origin, dimStr) { - var len = dimStr.length; - var dimLoose = dimStr; - - if (dimLoose.charAt(0) === '[' && dimLoose.charAt(len - 1) === ']') { - dimLoose = +dimLoose.slice(1, len - 1); // Also support: '[]' => 0 - - if ("development" !== 'production') { - if (isNaN(dimLoose)) { - error("Invalide label formatter: @" + dimStr + ", only support @[0], @[1], @[2], ..."); - } - } - } - - var val = retrieveRawValue(data, dataIndex, dimLoose); - - if (extendParams && isArray(extendParams.interpolatedValue)) { - var dimIndex = data.getDimensionIndex(dimLoose); - - if (dimIndex >= 0) { - val = extendParams.interpolatedValue[dimIndex]; - } - } - - return val != null ? val + '' : ''; - }); - } - }; - /** - * Get raw value in option - */ - - - DataFormatMixin.prototype.getRawValue = function (idx, dataType) { - return retrieveRawValue(this.getData(dataType), idx); - }; - /** - * Should be implemented. - * @param {number} dataIndex - * @param {boolean} [multipleSeries=false] - * @param {string} [dataType] - */ - - - DataFormatMixin.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - // Empty function - return; - }; - - return DataFormatMixin; - }(); - // but guess little chance has been used outside. Do we need to backward - // compat it? - // type TooltipFormatResultLegacyObject = { - // // `html` means the markup language text, either in 'html' or 'richText'. - // // The name `html` is not appropriate because in 'richText' it is not a HTML - // // string. But still support it for backward compatibility. - // html: string; - // markers: Dictionary<ColorString>; - // }; - - /** - * For backward compat, normalize the return from `formatTooltip`. - */ - - function normalizeTooltipFormatResult(result) { - var markupText; // let markers: Dictionary<ColorString>; - - var markupFragment; - - if (isObject(result)) { - if (result.type) { - markupFragment = result; - } else { - if ("development" !== 'production') { - console.warn('The return type of `formatTooltip` is not supported: ' + makePrintable(result)); - } - } // else { - // markupText = (result as TooltipFormatResultLegacyObject).html; - // markers = (result as TooltipFormatResultLegacyObject).markers; - // if (markersExisting) { - // markers = zrUtil.merge(markersExisting, markers); - // } - // } - - } else { - markupText = result; - } - - return { - text: markupText, - // markers: markers || markersExisting, - frag: markupFragment - }; - } - - /** - * @param {Object} define - * @return See the return of `createTask`. - */ - - function createTask(define) { - return new Task(define); - } - - var Task = - /** @class */ - function () { - function Task(define) { - define = define || {}; - this._reset = define.reset; - this._plan = define.plan; - this._count = define.count; - this._onDirty = define.onDirty; - this._dirty = true; - } - /** - * @param step Specified step. - * @param skip Skip customer perform call. - * @param modBy Sampling window size. - * @param modDataCount Sampling count. - * @return whether unfinished. - */ - - - Task.prototype.perform = function (performArgs) { - var upTask = this._upstream; - var skip = performArgs && performArgs.skip; // TODO some refactor. - // Pull data. Must pull data each time, because context.data - // may be updated by Series.setData. - - if (this._dirty && upTask) { - var context = this.context; - context.data = context.outputData = upTask.context.outputData; - } - - if (this.__pipeline) { - this.__pipeline.currentTask = this; - } - - var planResult; - - if (this._plan && !skip) { - planResult = this._plan(this.context); - } // Support sharding by mod, which changes the render sequence and makes the rendered graphic - // elements uniformed distributed when progress, especially when moving or zooming. - - - var lastModBy = normalizeModBy(this._modBy); - var lastModDataCount = this._modDataCount || 0; - var modBy = normalizeModBy(performArgs && performArgs.modBy); - var modDataCount = performArgs && performArgs.modDataCount || 0; - - if (lastModBy !== modBy || lastModDataCount !== modDataCount) { - planResult = 'reset'; - } - - function normalizeModBy(val) { - !(val >= 1) && (val = 1); // jshint ignore:line - - return val; - } - - var forceFirstProgress; - - if (this._dirty || planResult === 'reset') { - this._dirty = false; - forceFirstProgress = this._doReset(skip); - } - - this._modBy = modBy; - this._modDataCount = modDataCount; - var step = performArgs && performArgs.step; - - if (upTask) { - if ("development" !== 'production') { - assert(upTask._outputDueEnd != null); - } - - this._dueEnd = upTask._outputDueEnd; - } // DataTask or overallTask - else { - if ("development" !== 'production') { - assert(!this._progress || this._count); - } - - this._dueEnd = this._count ? this._count(this.context) : Infinity; - } // Note: Stubs, that its host overall task let it has progress, has progress. - // If no progress, pass index from upstream to downstream each time plan called. - - - if (this._progress) { - var start = this._dueIndex; - var end = Math.min(step != null ? this._dueIndex + step : Infinity, this._dueEnd); - - if (!skip && (forceFirstProgress || start < end)) { - var progress = this._progress; - - if (isArray(progress)) { - for (var i = 0; i < progress.length; i++) { - this._doProgress(progress[i], start, end, modBy, modDataCount); - } - } else { - this._doProgress(progress, start, end, modBy, modDataCount); - } - } - - this._dueIndex = end; // If no `outputDueEnd`, assume that output data and - // input data is the same, so use `dueIndex` as `outputDueEnd`. - - var outputDueEnd = this._settedOutputEnd != null ? this._settedOutputEnd : end; - - if ("development" !== 'production') { - // ??? Can not rollback. - assert(outputDueEnd >= this._outputDueEnd); - } - - this._outputDueEnd = outputDueEnd; - } else { - // (1) Some overall task has no progress. - // (2) Stubs, that its host overall task do not let it has progress, has no progress. - // This should always be performed so it can be passed to downstream. - this._dueIndex = this._outputDueEnd = this._settedOutputEnd != null ? this._settedOutputEnd : this._dueEnd; - } - - return this.unfinished(); - }; - - Task.prototype.dirty = function () { - this._dirty = true; - this._onDirty && this._onDirty(this.context); - }; - - Task.prototype._doProgress = function (progress, start, end, modBy, modDataCount) { - iterator.reset(start, end, modBy, modDataCount); - this._callingProgress = progress; - - this._callingProgress({ - start: start, - end: end, - count: end - start, - next: iterator.next - }, this.context); - }; - - Task.prototype._doReset = function (skip) { - this._dueIndex = this._outputDueEnd = this._dueEnd = 0; - this._settedOutputEnd = null; - var progress; - var forceFirstProgress; - - if (!skip && this._reset) { - progress = this._reset(this.context); - - if (progress && progress.progress) { - forceFirstProgress = progress.forceFirstProgress; - progress = progress.progress; - } // To simplify no progress checking, array must has item. - - - if (isArray(progress) && !progress.length) { - progress = null; - } - } - - this._progress = progress; - this._modBy = this._modDataCount = null; - var downstream = this._downstream; - downstream && downstream.dirty(); - return forceFirstProgress; - }; - - Task.prototype.unfinished = function () { - return this._progress && this._dueIndex < this._dueEnd; - }; - /** - * @param downTask The downstream task. - * @return The downstream task. - */ - - - Task.prototype.pipe = function (downTask) { - if ("development" !== 'production') { - assert(downTask && !downTask._disposed && downTask !== this); - } // If already downstream, do not dirty downTask. - - - if (this._downstream !== downTask || this._dirty) { - this._downstream = downTask; - downTask._upstream = this; - downTask.dirty(); - } - }; - - Task.prototype.dispose = function () { - if (this._disposed) { - return; - } - - this._upstream && (this._upstream._downstream = null); - this._downstream && (this._downstream._upstream = null); - this._dirty = false; - this._disposed = true; - }; - - Task.prototype.getUpstream = function () { - return this._upstream; - }; - - Task.prototype.getDownstream = function () { - return this._downstream; - }; - - Task.prototype.setOutputEnd = function (end) { - // This only happens in dataTask, dataZoom, map, currently. - // where dataZoom do not set end each time, but only set - // when reset. So we should record the set end, in case - // that the stub of dataZoom perform again and earse the - // set end by upstream. - this._outputDueEnd = this._settedOutputEnd = end; - }; - - return Task; - }(); - - var iterator = function () { - var end; - var current; - var modBy; - var modDataCount; - var winCount; - var it = { - reset: function (s, e, sStep, sCount) { - current = s; - end = e; - modBy = sStep; - modDataCount = sCount; - winCount = Math.ceil(modDataCount / modBy); - it.next = modBy > 1 && modDataCount > 0 ? modNext : sequentialNext; - } - }; - return it; - - function sequentialNext() { - return current < end ? current++ : null; - } - - function modNext() { - var dataIndex = current % winCount * modBy + Math.ceil(current / winCount); - var result = current >= end ? null : dataIndex < modDataCount ? dataIndex // If modDataCount is smaller than data.count() (consider `appendData` case), - // Use normal linear rendering mode. - : current; - current++; - return result; - } - }(); // ----------------------------------------------------------------------------- - // For stream debug (Should be commented out after used!) - // @usage: printTask(this, 'begin'); - // @usage: printTask(this, null, {someExtraProp}); - // @usage: Use `__idxInPipeline` as conditional breakpiont. - // - // window.printTask = function (task: any, prefix: string, extra: { [key: string]: unknown }): void { - // window.ecTaskUID == null && (window.ecTaskUID = 0); - // task.uidDebug == null && (task.uidDebug = `task_${window.ecTaskUID++}`); - // task.agent && task.agent.uidDebug == null && (task.agent.uidDebug = `task_${window.ecTaskUID++}`); - // let props = []; - // if (task.__pipeline) { - // let val = `${task.__idxInPipeline}/${task.__pipeline.tail.__idxInPipeline} ${task.agent ? '(stub)' : ''}`; - // props.push({text: '__idxInPipeline/total', value: val}); - // } else { - // let stubCount = 0; - // task.agentStubMap.each(() => stubCount++); - // props.push({text: 'idx', value: `overall (stubs: ${stubCount})`}); - // } - // props.push({text: 'uid', value: task.uidDebug}); - // if (task.__pipeline) { - // props.push({text: 'pipelineId', value: task.__pipeline.id}); - // task.agent && props.push( - // {text: 'stubFor', value: task.agent.uidDebug} - // ); - // } - // props.push( - // {text: 'dirty', value: task._dirty}, - // {text: 'dueIndex', value: task._dueIndex}, - // {text: 'dueEnd', value: task._dueEnd}, - // {text: 'outputDueEnd', value: task._outputDueEnd} - // ); - // if (extra) { - // Object.keys(extra).forEach(key => { - // props.push({text: key, value: extra[key]}); - // }); - // } - // let args = ['color: blue']; - // let msg = `%c[${prefix || 'T'}] %c` + props.map(item => ( - // args.push('color: green', 'color: red'), - // `${item.text}: %c${item.value}` - // )).join('%c, '); - // console.log.apply(console, [msg].concat(args)); - // // console.log(this); - // }; - // window.printPipeline = function (task: any, prefix: string) { - // const pipeline = task.__pipeline; - // let currTask = pipeline.head; - // while (currTask) { - // window.printTask(currTask, prefix); - // currTask = currTask._downstream; - // } - // }; - // window.showChain = function (chainHeadTask) { - // var chain = []; - // var task = chainHeadTask; - // while (task) { - // chain.push({ - // task: task, - // up: task._upstream, - // down: task._downstream, - // idxInPipeline: task.__idxInPipeline - // }); - // task = task._downstream; - // } - // return chain; - // }; - // window.findTaskInChain = function (task, chainHeadTask) { - // let chain = window.showChain(chainHeadTask); - // let result = []; - // for (let i = 0; i < chain.length; i++) { - // let chainItem = chain[i]; - // if (chainItem.task === task) { - // result.push(i); - // } - // } - // return result; - // }; - // window.printChainAEachInChainB = function (chainHeadTaskA, chainHeadTaskB) { - // let chainA = window.showChain(chainHeadTaskA); - // for (let i = 0; i < chainA.length; i++) { - // console.log('chainAIdx:', i, 'inChainB:', window.findTaskInChain(chainA[i].task, chainHeadTaskB)); - // } - // }; - - /** - * Convert raw the value in to inner value in List. - * - * [Performance sensitive] - * - * [Caution]: this is the key logic of user value parser. - * For backward compatibility, do not modify it until you have to! - */ - - function parseDataValue(value, // For high performance, do not omit the second param. - opt) { - // Performance sensitive. - var dimType = opt && opt.type; - - if (dimType === 'ordinal') { - // If given value is a category string - return value; - } - - if (dimType === 'time' // spead up when using timestamp - && !isNumber(value) && value != null && value !== '-') { - value = +parseDate(value); - } // dimType defaults 'number'. - // If dimType is not ordinal and value is null or undefined or NaN or '-', - // parse to NaN. - // number-like string (like ' 123 ') can be converted to a number. - // where null/undefined or other string will be converted to NaN. - - - return value == null || value === '' ? NaN // If string (like '-'), using '+' parse to NaN - // If object, also parse to NaN - : +value; - } - var valueParserMap = createHashMap({ - 'number': function (val) { - // Do not use `numericToNumber` here. We have `numericToNumber` by default. - // Here the number parser can have loose rule: - // enable to cut suffix: "120px" => 120, "14%" => 14. - return parseFloat(val); - }, - 'time': function (val) { - // return timestamp. - return +parseDate(val); - }, - 'trim': function (val) { - return isString(val) ? trim(val) : val; - } - }); - function getRawValueParser(type) { - return valueParserMap.get(type); - } - var ORDER_COMPARISON_OP_MAP = { - lt: function (lval, rval) { - return lval < rval; - }, - lte: function (lval, rval) { - return lval <= rval; - }, - gt: function (lval, rval) { - return lval > rval; - }, - gte: function (lval, rval) { - return lval >= rval; - } - }; - - var FilterOrderComparator = - /** @class */ - function () { - function FilterOrderComparator(op, rval) { - if (!isNumber(rval)) { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = 'rvalue of "<", ">", "<=", ">=" can only be number in filter.'; - } - - throwError(errMsg); - } - - this._opFn = ORDER_COMPARISON_OP_MAP[op]; - this._rvalFloat = numericToNumber(rval); - } // Performance sensitive. - - - FilterOrderComparator.prototype.evaluate = function (lval) { - // Most cases is 'number', and typeof maybe 10 times faseter than parseFloat. - return isNumber(lval) ? this._opFn(lval, this._rvalFloat) : this._opFn(numericToNumber(lval), this._rvalFloat); - }; - - return FilterOrderComparator; - }(); - - var SortOrderComparator = - /** @class */ - function () { - /** - * @param order by default: 'asc' - * @param incomparable by default: Always on the tail. - * That is, if 'asc' => 'max', if 'desc' => 'min' - * See the definition of "incomparable" in [SORT_COMPARISON_RULE]. - */ - function SortOrderComparator(order, incomparable) { - var isDesc = order === 'desc'; - this._resultLT = isDesc ? 1 : -1; - - if (incomparable == null) { - incomparable = isDesc ? 'min' : 'max'; - } - - this._incomparable = incomparable === 'min' ? -Infinity : Infinity; - } // See [SORT_COMPARISON_RULE]. - // Performance sensitive. - - - SortOrderComparator.prototype.evaluate = function (lval, rval) { - // Most cases is 'number', and typeof maybe 10 times faseter than parseFloat. - var lvalFloat = isNumber(lval) ? lval : numericToNumber(lval); - var rvalFloat = isNumber(rval) ? rval : numericToNumber(rval); - var lvalNotNumeric = isNaN(lvalFloat); - var rvalNotNumeric = isNaN(rvalFloat); - - if (lvalNotNumeric) { - lvalFloat = this._incomparable; - } - - if (rvalNotNumeric) { - rvalFloat = this._incomparable; - } - - if (lvalNotNumeric && rvalNotNumeric) { - var lvalIsStr = isString(lval); - var rvalIsStr = isString(rval); - - if (lvalIsStr) { - lvalFloat = rvalIsStr ? lval : 0; - } - - if (rvalIsStr) { - rvalFloat = lvalIsStr ? rval : 0; - } - } - - return lvalFloat < rvalFloat ? this._resultLT : lvalFloat > rvalFloat ? -this._resultLT : 0; - }; - - return SortOrderComparator; - }(); - - var FilterEqualityComparator = - /** @class */ - function () { - function FilterEqualityComparator(isEq, rval) { - this._rval = rval; - this._isEQ = isEq; - this._rvalTypeof = typeof rval; - this._rvalFloat = numericToNumber(rval); - } // Performance sensitive. - - - FilterEqualityComparator.prototype.evaluate = function (lval) { - var eqResult = lval === this._rval; - - if (!eqResult) { - var lvalTypeof = typeof lval; - - if (lvalTypeof !== this._rvalTypeof && (lvalTypeof === 'number' || this._rvalTypeof === 'number')) { - eqResult = numericToNumber(lval) === this._rvalFloat; - } - } - - return this._isEQ ? eqResult : !eqResult; - }; - - return FilterEqualityComparator; - }(); - /** - * [FILTER_COMPARISON_RULE] - * `lt`|`lte`|`gt`|`gte`: - * + rval must be a number. And lval will be converted to number (`numericToNumber`) to compare. - * `eq`: - * + If same type, compare with `===`. - * + If there is one number, convert to number (`numericToNumber`) to compare. - * + Else return `false`. - * `ne`: - * + Not `eq`. - * - * - * [SORT_COMPARISON_RULE] - * All the values are grouped into three categories: - * + "numeric" (number and numeric string) - * + "non-numeric-string" (string that excluding numeric string) - * + "others" - * "numeric" vs "numeric": values are ordered by number order. - * "non-numeric-string" vs "non-numeric-string": values are ordered by ES spec (#sec-abstract-relational-comparison). - * "others" vs "others": do not change order (always return 0). - * "numeric" vs "non-numeric-string": "non-numeric-string" is treated as "incomparable". - * "number" vs "others": "others" is treated as "incomparable". - * "non-numeric-string" vs "others": "others" is treated as "incomparable". - * "incomparable" will be seen as -Infinity or Infinity (depends on the settings). - * MEMO: - * Non-numeric string sort makes sense when we need to put the items with the same tag together. - * But if we support string sort, we still need to avoid the misleading like `'2' > '12'`, - * So we treat "numeric-string" sorted by number order rather than string comparison. - * - * - * [CHECK_LIST_OF_THE_RULE_DESIGN] - * + Do not support string comparison until required. And also need to - * avoid the misleading of "2" > "12". - * + Should avoid the misleading case: - * `" 22 " gte "22"` is `true` but `" 22 " eq "22"` is `false`. - * + JS bad case should be avoided: null <= 0, [] <= 0, ' ' <= 0, ... - * + Only "numeric" can be converted to comparable number, otherwise converted to NaN. - * See `util/number.ts#numericToNumber`. - * - * @return If `op` is not `RelationalOperator`, return null; - */ - - - function createFilterComparator(op, rval) { - return op === 'eq' || op === 'ne' ? new FilterEqualityComparator(op === 'eq', rval) : hasOwn(ORDER_COMPARISON_OP_MAP, op) ? new FilterOrderComparator(op, rval) : null; - } - - /** - * TODO: disable writable. - * This structure will be exposed to users. - */ - - var ExternalSource = - /** @class */ - function () { - function ExternalSource() {} - - ExternalSource.prototype.getRawData = function () { - // Only built-in transform available. - throw new Error('not supported'); - }; - - ExternalSource.prototype.getRawDataItem = function (dataIndex) { - // Only built-in transform available. - throw new Error('not supported'); - }; - - ExternalSource.prototype.cloneRawData = function () { - return; - }; - /** - * @return If dimension not found, return null/undefined. - */ - - - ExternalSource.prototype.getDimensionInfo = function (dim) { - return; - }; - /** - * dimensions defined if and only if either: - * (a) dataset.dimensions are declared. - * (b) dataset data include dimensions definitions in data (detected or via specified `sourceHeader`). - * If dimensions are defined, `dimensionInfoAll` is corresponding to - * the defined dimensions. - * Otherwise, `dimensionInfoAll` is determined by data columns. - * @return Always return an array (even empty array). - */ - - - ExternalSource.prototype.cloneAllDimensionInfo = function () { - return; - }; - - ExternalSource.prototype.count = function () { - return; - }; - /** - * Only support by dimension index. - * No need to support by dimension name in transform function, - * because transform function is not case-specific, no need to use name literally. - */ - - - ExternalSource.prototype.retrieveValue = function (dataIndex, dimIndex) { - return; - }; - - ExternalSource.prototype.retrieveValueFromItem = function (dataItem, dimIndex) { - return; - }; - - ExternalSource.prototype.convertValue = function (rawVal, dimInfo) { - return parseDataValue(rawVal, dimInfo); - }; - - return ExternalSource; - }(); - - function createExternalSource(internalSource, externalTransform) { - var extSource = new ExternalSource(); - var data = internalSource.data; - var sourceFormat = extSource.sourceFormat = internalSource.sourceFormat; - var sourceHeaderCount = internalSource.startIndex; - var errMsg = ''; - - if (internalSource.seriesLayoutBy !== SERIES_LAYOUT_BY_COLUMN) { - // For the logic simplicity in transformer, only 'culumn' is - // supported in data transform. Otherwise, the `dimensionsDefine` - // might be detected by 'row', which probably confuses users. - if ("development" !== 'production') { - errMsg = '`seriesLayoutBy` of upstream dataset can only be "column" in data transform.'; - } - - throwError(errMsg); - } // [MEMO] - // Create a new dimensions structure for exposing. - // Do not expose all dimension info to users directly. - // Because the dimension is probably auto detected from data and not might reliable. - // Should not lead the transformers to think that is reliable and return it. - // See [DIMENSION_INHERIT_RULE] in `sourceManager.ts`. - - - var dimensions = []; - var dimsByName = {}; - var dimsDef = internalSource.dimensionsDefine; - - if (dimsDef) { - each(dimsDef, function (dimDef, idx) { - var name = dimDef.name; - var dimDefExt = { - index: idx, - name: name, - displayName: dimDef.displayName - }; - dimensions.push(dimDefExt); // Users probably do not specify dimension name. For simplicity, data transform - // does not generate dimension name. - - if (name != null) { - // Dimension name should not be duplicated. - // For simplicity, data transform forbids name duplication, do not generate - // new name like module `completeDimensions.ts` did, but just tell users. - var errMsg_1 = ''; - - if (hasOwn(dimsByName, name)) { - if ("development" !== 'production') { - errMsg_1 = 'dimension name "' + name + '" duplicated.'; - } - - throwError(errMsg_1); - } - - dimsByName[name] = dimDefExt; - } - }); - } // If dimension definitions are not defined and can not be detected. - // e.g., pure data `[[11, 22], ...]`. - else { - for (var i = 0; i < internalSource.dimensionsDetectedCount || 0; i++) { - // Do not generete name or anything others. The consequence process in - // `transform` or `series` probably have there own name generation strategry. - dimensions.push({ - index: i - }); - } - } // Implement public methods: - - - var rawItemGetter = getRawSourceItemGetter(sourceFormat, SERIES_LAYOUT_BY_COLUMN); - - if (externalTransform.__isBuiltIn) { - extSource.getRawDataItem = function (dataIndex) { - return rawItemGetter(data, sourceHeaderCount, dimensions, dataIndex); - }; - - extSource.getRawData = bind(getRawData, null, internalSource); - } - - extSource.cloneRawData = bind(cloneRawData, null, internalSource); - var rawCounter = getRawSourceDataCounter(sourceFormat, SERIES_LAYOUT_BY_COLUMN); - extSource.count = bind(rawCounter, null, data, sourceHeaderCount, dimensions); - var rawValueGetter = getRawSourceValueGetter(sourceFormat); - - extSource.retrieveValue = function (dataIndex, dimIndex) { - var rawItem = rawItemGetter(data, sourceHeaderCount, dimensions, dataIndex); - return retrieveValueFromItem(rawItem, dimIndex); - }; - - var retrieveValueFromItem = extSource.retrieveValueFromItem = function (dataItem, dimIndex) { - if (dataItem == null) { - return; - } - - var dimDef = dimensions[dimIndex]; // When `dimIndex` is `null`, `rawValueGetter` return the whole item. - - if (dimDef) { - return rawValueGetter(dataItem, dimIndex, dimDef.name); - } - }; - - extSource.getDimensionInfo = bind(getDimensionInfo, null, dimensions, dimsByName); - extSource.cloneAllDimensionInfo = bind(cloneAllDimensionInfo, null, dimensions); - return extSource; - } - - function getRawData(upstream) { - var sourceFormat = upstream.sourceFormat; - - if (!isSupportedSourceFormat(sourceFormat)) { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = '`getRawData` is not supported in source format ' + sourceFormat; - } - - throwError(errMsg); - } - - return upstream.data; - } - - function cloneRawData(upstream) { - var sourceFormat = upstream.sourceFormat; - var data = upstream.data; - - if (!isSupportedSourceFormat(sourceFormat)) { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = '`cloneRawData` is not supported in source format ' + sourceFormat; - } - - throwError(errMsg); - } - - if (sourceFormat === SOURCE_FORMAT_ARRAY_ROWS) { - var result = []; - - for (var i = 0, len = data.length; i < len; i++) { - // Not strictly clone for performance - result.push(data[i].slice()); - } - - return result; - } else if (sourceFormat === SOURCE_FORMAT_OBJECT_ROWS) { - var result = []; - - for (var i = 0, len = data.length; i < len; i++) { - // Not strictly clone for performance - result.push(extend({}, data[i])); - } - - return result; - } - } - - function getDimensionInfo(dimensions, dimsByName, dim) { - if (dim == null) { - return; - } // Keep the same logic as `List::getDimension` did. - - - if (isNumber(dim) // If being a number-like string but not being defined a dimension name. - || !isNaN(dim) && !hasOwn(dimsByName, dim)) { - return dimensions[dim]; - } else if (hasOwn(dimsByName, dim)) { - return dimsByName[dim]; - } - } - - function cloneAllDimensionInfo(dimensions) { - return clone(dimensions); - } - - var externalTransformMap = createHashMap(); - function registerExternalTransform(externalTransform) { - externalTransform = clone(externalTransform); - var type = externalTransform.type; - var errMsg = ''; - - if (!type) { - if ("development" !== 'production') { - errMsg = 'Must have a `type` when `registerTransform`.'; - } - - throwError(errMsg); - } - - var typeParsed = type.split(':'); - - if (typeParsed.length !== 2) { - if ("development" !== 'production') { - errMsg = 'Name must include namespace like "ns:regression".'; - } - - throwError(errMsg); - } // Namespace 'echarts:xxx' is official namespace, where the transforms should - // be called directly via 'xxx' rather than 'echarts:xxx'. - - - var isBuiltIn = false; - - if (typeParsed[0] === 'echarts') { - type = typeParsed[1]; - isBuiltIn = true; - } - - externalTransform.__isBuiltIn = isBuiltIn; - externalTransformMap.set(type, externalTransform); - } - function applyDataTransform(rawTransOption, sourceList, infoForPrint) { - var pipedTransOption = normalizeToArray(rawTransOption); - var pipeLen = pipedTransOption.length; - var errMsg = ''; - - if (!pipeLen) { - if ("development" !== 'production') { - errMsg = 'If `transform` declared, it should at least contain one transform.'; - } - - throwError(errMsg); - } - - for (var i = 0, len = pipeLen; i < len; i++) { - var transOption = pipedTransOption[i]; - sourceList = applySingleDataTransform(transOption, sourceList, infoForPrint, pipeLen === 1 ? null : i); // piped transform only support single input, except the fist one. - // piped transform only support single output, except the last one. - - if (i !== len - 1) { - sourceList.length = Math.max(sourceList.length, 1); - } - } - - return sourceList; - } - - function applySingleDataTransform(transOption, upSourceList, infoForPrint, // If `pipeIndex` is null/undefined, no piped transform. - pipeIndex) { - var errMsg = ''; - - if (!upSourceList.length) { - if ("development" !== 'production') { - errMsg = 'Must have at least one upstream dataset.'; - } - - throwError(errMsg); - } - - if (!isObject(transOption)) { - if ("development" !== 'production') { - errMsg = 'transform declaration must be an object rather than ' + typeof transOption + '.'; - } - - throwError(errMsg); - } - - var transType = transOption.type; - var externalTransform = externalTransformMap.get(transType); - - if (!externalTransform) { - if ("development" !== 'production') { - errMsg = 'Can not find transform on type "' + transType + '".'; - } - - throwError(errMsg); - } // Prepare source - - - var extUpSourceList = map(upSourceList, function (upSource) { - return createExternalSource(upSource, externalTransform); - }); - var resultList = normalizeToArray(externalTransform.transform({ - upstream: extUpSourceList[0], - upstreamList: extUpSourceList, - config: clone(transOption.config) - })); - - if ("development" !== 'production') { - if (transOption.print) { - var printStrArr = map(resultList, function (extSource) { - var pipeIndexStr = pipeIndex != null ? ' === pipe index: ' + pipeIndex : ''; - return ['=== dataset index: ' + infoForPrint.datasetIndex + pipeIndexStr + ' ===', '- transform result data:', makePrintable(extSource.data), '- transform result dimensions:', makePrintable(extSource.dimensions)].join('\n'); - }).join('\n'); - log(printStrArr); - } - } - - return map(resultList, function (result, resultIndex) { - var errMsg = ''; - - if (!isObject(result)) { - if ("development" !== 'production') { - errMsg = 'A transform should not return some empty results.'; - } - - throwError(errMsg); - } - - if (!result.data) { - if ("development" !== 'production') { - errMsg = 'Transform result data should be not be null or undefined'; - } - - throwError(errMsg); - } - - var sourceFormat = detectSourceFormat(result.data); - - if (!isSupportedSourceFormat(sourceFormat)) { - if ("development" !== 'production') { - errMsg = 'Transform result data should be array rows or object rows.'; - } - - throwError(errMsg); - } - - var resultMetaRawOption; - var firstUpSource = upSourceList[0]; - /** - * Intuitively, the end users known the content of the original `dataset.source`, - * calucating the transform result in mind. - * Suppose the original `dataset.source` is: - * ```js - * [ - * ['product', '2012', '2013', '2014', '2015'], - * ['AAA', 41.1, 30.4, 65.1, 53.3], - * ['BBB', 86.5, 92.1, 85.7, 83.1], - * ['CCC', 24.1, 67.2, 79.5, 86.4] - * ] - * ``` - * The dimension info have to be detected from the source data. - * Some of the transformers (like filter, sort) will follow the dimension info - * of upstream, while others use new dimensions (like aggregate). - * Transformer can output a field `dimensions` to define the its own output dimensions. - * We also allow transformers to ignore the output `dimensions` field, and - * inherit the upstream dimensions definition. It can reduce the burden of handling - * dimensions in transformers. - * - * See also [DIMENSION_INHERIT_RULE] in `sourceManager.ts`. - */ - - if (firstUpSource && resultIndex === 0 // If transformer returns `dimensions`, it means that the transformer has different - // dimensions definitions. We do not inherit anything from upstream. - && !result.dimensions) { - var startIndex = firstUpSource.startIndex; // We copy the header of upstream to the result, because: - // (1) The returned data always does not contain header line and can not be used - // as dimension-detection. In this case we can not use "detected dimensions" of - // upstream directly, because it might be detected based on different `seriesLayoutBy`. - // (2) We should support that the series read the upstream source in `seriesLayoutBy: 'row'`. - // So the original detected header should be add to the result, otherwise they can not be read. - - if (startIndex) { - result.data = firstUpSource.data.slice(0, startIndex).concat(result.data); - } - - resultMetaRawOption = { - seriesLayoutBy: SERIES_LAYOUT_BY_COLUMN, - sourceHeader: startIndex, - dimensions: firstUpSource.metaRawOption.dimensions - }; - } else { - resultMetaRawOption = { - seriesLayoutBy: SERIES_LAYOUT_BY_COLUMN, - sourceHeader: 0, - dimensions: result.dimensions - }; - } - - return createSource(result.data, resultMetaRawOption, null); - }); - } - - function isSupportedSourceFormat(sourceFormat) { - return sourceFormat === SOURCE_FORMAT_ARRAY_ROWS || sourceFormat === SOURCE_FORMAT_OBJECT_ROWS; - } - - var UNDEFINED = 'undefined'; - /* global Float64Array, Int32Array, Uint32Array, Uint16Array */ - // Caution: MUST not use `new CtorUint32Array(arr, 0, len)`, because the Ctor of array is - // different from the Ctor of typed array. - - var CtorUint32Array = typeof Uint32Array === UNDEFINED ? Array : Uint32Array; - var CtorUint16Array = typeof Uint16Array === UNDEFINED ? Array : Uint16Array; - var CtorInt32Array = typeof Int32Array === UNDEFINED ? Array : Int32Array; - var CtorFloat64Array = typeof Float64Array === UNDEFINED ? Array : Float64Array; - /** - * Multi dimensional data store - */ - - var dataCtors = { - 'float': CtorFloat64Array, - 'int': CtorInt32Array, - // Ordinal data type can be string or int - 'ordinal': Array, - 'number': Array, - 'time': CtorFloat64Array - }; - var defaultDimValueGetters; - - function getIndicesCtor(rawCount) { - // The possible max value in this._indicies is always this._rawCount despite of filtering. - return rawCount > 65535 ? CtorUint32Array : CtorUint16Array; - } - - function getInitialExtent() { - return [Infinity, -Infinity]; - } - - function cloneChunk(originalChunk) { - var Ctor = originalChunk.constructor; // Only shallow clone is enough when Array. - - return Ctor === Array ? originalChunk.slice() : new Ctor(originalChunk); - } - - function prepareStore(store, dimIdx, dimType, end, append) { - var DataCtor = dataCtors[dimType || 'float']; - - if (append) { - var oldStore = store[dimIdx]; - var oldLen = oldStore && oldStore.length; - - if (!(oldLen === end)) { - var newStore = new DataCtor(end); // The cost of the copy is probably inconsiderable - // within the initial chunkSize. - - for (var j = 0; j < oldLen; j++) { - newStore[j] = oldStore[j]; - } - - store[dimIdx] = newStore; - } - } else { - store[dimIdx] = new DataCtor(end); - } - } - /** - * Basically, DataStore API keep immutable. - */ - - var DataStore = - /** @class */ - function () { - function DataStore() { - this._chunks = []; // It will not be calculated until needed. - - this._rawExtent = []; - this._extent = []; - this._count = 0; - this._rawCount = 0; - this._calcDimNameToIdx = createHashMap(); - } - /** - * Initialize from data - */ - - - DataStore.prototype.initData = function (provider, inputDimensions, dimValueGetter) { - if ("development" !== 'production') { - assert(isFunction(provider.getItem) && isFunction(provider.count), 'Invalid data provider.'); - } - - this._provider = provider; // Clear - - this._chunks = []; - this._indices = null; - this.getRawIndex = this._getRawIdxIdentity; - var source = provider.getSource(); - var defaultGetter = this.defaultDimValueGetter = defaultDimValueGetters[source.sourceFormat]; // Default dim value getter - - this._dimValueGetter = dimValueGetter || defaultGetter; // Reset raw extent. - - this._rawExtent = []; - var willRetrieveDataByName = shouldRetrieveDataByName(source); - this._dimensions = map(inputDimensions, function (dim) { - if ("development" !== 'production') { - if (willRetrieveDataByName) { - assert(dim.property != null); - } - } - - return { - // Only pick these two props. Not leak other properties like orderMeta. - type: dim.type, - property: dim.property - }; - }); - - this._initDataFromProvider(0, provider.count()); - }; - - DataStore.prototype.getProvider = function () { - return this._provider; - }; - /** - * Caution: even when a `source` instance owned by a series, the created data store - * may still be shared by different sereis (the source hash does not use all `source` - * props, see `sourceManager`). In this case, the `source` props that are not used in - * hash (like `source.dimensionDefine`) probably only belongs to a certain series and - * thus should not be fetch here. - */ - - - DataStore.prototype.getSource = function () { - return this._provider.getSource(); - }; - /** - * @caution Only used in dataStack. - */ - - - DataStore.prototype.ensureCalculationDimension = function (dimName, type) { - var calcDimNameToIdx = this._calcDimNameToIdx; - var dimensions = this._dimensions; - var calcDimIdx = calcDimNameToIdx.get(dimName); - - if (calcDimIdx != null) { - if (dimensions[calcDimIdx].type === type) { - return calcDimIdx; - } - } else { - calcDimIdx = dimensions.length; - } - - dimensions[calcDimIdx] = { - type: type - }; - calcDimNameToIdx.set(dimName, calcDimIdx); - this._chunks[calcDimIdx] = new dataCtors[type || 'float'](this._rawCount); - this._rawExtent[calcDimIdx] = getInitialExtent(); - return calcDimIdx; - }; - - DataStore.prototype.collectOrdinalMeta = function (dimIdx, ordinalMeta) { - var chunk = this._chunks[dimIdx]; - var dim = this._dimensions[dimIdx]; - var rawExtents = this._rawExtent; - var offset = dim.ordinalOffset || 0; - var len = chunk.length; - - if (offset === 0) { - // We need to reset the rawExtent if collect is from start. - // Because this dimension may be guessed as number and calcuating a wrong extent. - rawExtents[dimIdx] = getInitialExtent(); - } - - var dimRawExtent = rawExtents[dimIdx]; // Parse from previous data offset. len may be changed after appendData - - for (var i = offset; i < len; i++) { - var val = chunk[i] = ordinalMeta.parseAndCollect(chunk[i]); - - if (!isNaN(val)) { - dimRawExtent[0] = Math.min(val, dimRawExtent[0]); - dimRawExtent[1] = Math.max(val, dimRawExtent[1]); - } - } - - dim.ordinalMeta = ordinalMeta; - dim.ordinalOffset = len; - dim.type = 'ordinal'; // Force to be ordinal - }; - - DataStore.prototype.getOrdinalMeta = function (dimIdx) { - var dimInfo = this._dimensions[dimIdx]; - var ordinalMeta = dimInfo.ordinalMeta; - return ordinalMeta; - }; - - DataStore.prototype.getDimensionProperty = function (dimIndex) { - var item = this._dimensions[dimIndex]; - return item && item.property; - }; - /** - * Caution: Can be only called on raw data (before `this._indices` created). - */ - - - DataStore.prototype.appendData = function (data) { - if ("development" !== 'production') { - assert(!this._indices, 'appendData can only be called on raw data.'); - } - - var provider = this._provider; - var start = this.count(); - provider.appendData(data); - var end = provider.count(); - - if (!provider.persistent) { - end += start; - } - - if (start < end) { - this._initDataFromProvider(start, end, true); - } - - return [start, end]; - }; - - DataStore.prototype.appendValues = function (values, minFillLen) { - var chunks = this._chunks; - var dimensions = this._dimensions; - var dimLen = dimensions.length; - var rawExtent = this._rawExtent; - var start = this.count(); - var end = start + Math.max(values.length, minFillLen || 0); - - for (var i = 0; i < dimLen; i++) { - var dim = dimensions[i]; - prepareStore(chunks, i, dim.type, end, true); - } - - var emptyDataItem = []; - - for (var idx = start; idx < end; idx++) { - var sourceIdx = idx - start; // Store the data by dimensions - - for (var dimIdx = 0; dimIdx < dimLen; dimIdx++) { - var dim = dimensions[dimIdx]; - var val = defaultDimValueGetters.arrayRows.call(this, values[sourceIdx] || emptyDataItem, dim.property, sourceIdx, dimIdx); - chunks[dimIdx][idx] = val; - var dimRawExtent = rawExtent[dimIdx]; - val < dimRawExtent[0] && (dimRawExtent[0] = val); - val > dimRawExtent[1] && (dimRawExtent[1] = val); - } - } - - this._rawCount = this._count = end; - return { - start: start, - end: end - }; - }; - - DataStore.prototype._initDataFromProvider = function (start, end, append) { - var provider = this._provider; - var chunks = this._chunks; - var dimensions = this._dimensions; - var dimLen = dimensions.length; - var rawExtent = this._rawExtent; - var dimNames = map(dimensions, function (dim) { - return dim.property; - }); - - for (var i = 0; i < dimLen; i++) { - var dim = dimensions[i]; - - if (!rawExtent[i]) { - rawExtent[i] = getInitialExtent(); - } - - prepareStore(chunks, i, dim.type, end, append); - } - - if (provider.fillStorage) { - provider.fillStorage(start, end, chunks, rawExtent); - } else { - var dataItem = []; - - for (var idx = start; idx < end; idx++) { - // NOTICE: Try not to write things into dataItem - dataItem = provider.getItem(idx, dataItem); // Each data item is value - // [1, 2] - // 2 - // Bar chart, line chart which uses category axis - // only gives the 'y' value. 'x' value is the indices of category - // Use a tempValue to normalize the value to be a (x, y) value - // Store the data by dimensions - - for (var dimIdx = 0; dimIdx < dimLen; dimIdx++) { - var dimStorage = chunks[dimIdx]; // PENDING NULL is empty or zero - - var val = this._dimValueGetter(dataItem, dimNames[dimIdx], idx, dimIdx); - - dimStorage[idx] = val; - var dimRawExtent = rawExtent[dimIdx]; - val < dimRawExtent[0] && (dimRawExtent[0] = val); - val > dimRawExtent[1] && (dimRawExtent[1] = val); - } - } - } - - if (!provider.persistent && provider.clean) { - // Clean unused data if data source is typed array. - provider.clean(); - } - - this._rawCount = this._count = end; // Reset data extent - - this._extent = []; - }; - - DataStore.prototype.count = function () { - return this._count; - }; - /** - * Get value. Return NaN if idx is out of range. - */ - - - DataStore.prototype.get = function (dim, idx) { - if (!(idx >= 0 && idx < this._count)) { - return NaN; - } - - var dimStore = this._chunks[dim]; - return dimStore ? dimStore[this.getRawIndex(idx)] : NaN; - }; - - DataStore.prototype.getValues = function (dimensions, idx) { - var values = []; - var dimArr = []; - - if (idx == null) { - idx = dimensions; // TODO get all from store? - - dimensions = []; // All dimensions - - for (var i = 0; i < this._dimensions.length; i++) { - dimArr.push(i); - } - } else { - dimArr = dimensions; - } - - for (var i = 0, len = dimArr.length; i < len; i++) { - values.push(this.get(dimArr[i], idx)); - } - - return values; - }; - /** - * @param dim concrete dim - */ - - - DataStore.prototype.getByRawIndex = function (dim, rawIdx) { - if (!(rawIdx >= 0 && rawIdx < this._rawCount)) { - return NaN; - } - - var dimStore = this._chunks[dim]; - return dimStore ? dimStore[rawIdx] : NaN; - }; - /** - * Get sum of data in one dimension - */ - - - DataStore.prototype.getSum = function (dim) { - var dimData = this._chunks[dim]; - var sum = 0; - - if (dimData) { - for (var i = 0, len = this.count(); i < len; i++) { - var value = this.get(dim, i); - - if (!isNaN(value)) { - sum += value; - } - } - } - - return sum; - }; - /** - * Get median of data in one dimension - */ - - - DataStore.prototype.getMedian = function (dim) { - var dimDataArray = []; // map all data of one dimension - - this.each([dim], function (val) { - if (!isNaN(val)) { - dimDataArray.push(val); - } - }); // TODO - // Use quick select? - - var sortedDimDataArray = dimDataArray.sort(function (a, b) { - return a - b; - }); - var len = this.count(); // calculate median - - return len === 0 ? 0 : len % 2 === 1 ? sortedDimDataArray[(len - 1) / 2] : (sortedDimDataArray[len / 2] + sortedDimDataArray[len / 2 - 1]) / 2; - }; - /** - * Retrieve the index with given raw data index. - */ - - - DataStore.prototype.indexOfRawIndex = function (rawIndex) { - if (rawIndex >= this._rawCount || rawIndex < 0) { - return -1; - } - - if (!this._indices) { - return rawIndex; - } // Indices are ascending - - - var indices = this._indices; // If rawIndex === dataIndex - - var rawDataIndex = indices[rawIndex]; - - if (rawDataIndex != null && rawDataIndex < this._count && rawDataIndex === rawIndex) { - return rawIndex; - } - - var left = 0; - var right = this._count - 1; - - while (left <= right) { - var mid = (left + right) / 2 | 0; - - if (indices[mid] < rawIndex) { - left = mid + 1; - } else if (indices[mid] > rawIndex) { - right = mid - 1; - } else { - return mid; - } - } - - return -1; - }; - /** - * Retrieve the index of nearest value. - * @param dim - * @param value - * @param [maxDistance=Infinity] - * @return If and only if multiple indices have - * the same value, they are put to the result. - */ - - - DataStore.prototype.indicesOfNearest = function (dim, value, maxDistance) { - var chunks = this._chunks; - var dimData = chunks[dim]; - var nearestIndices = []; - - if (!dimData) { - return nearestIndices; - } - - if (maxDistance == null) { - maxDistance = Infinity; - } - - var minDist = Infinity; - var minDiff = -1; - var nearestIndicesLen = 0; // Check the test case of `test/ut/spec/data/SeriesData.js`. - - for (var i = 0, len = this.count(); i < len; i++) { - var dataIndex = this.getRawIndex(i); - var diff = value - dimData[dataIndex]; - var dist = Math.abs(diff); - - if (dist <= maxDistance) { - // When the `value` is at the middle of `this.get(dim, i)` and `this.get(dim, i+1)`, - // we'd better not push both of them to `nearestIndices`, otherwise it is easy to - // get more than one item in `nearestIndices` (more specifically, in `tooltip`). - // So we choose the one that `diff >= 0` in this case. - // But if `this.get(dim, i)` and `this.get(dim, j)` get the same value, both of them - // should be push to `nearestIndices`. - if (dist < minDist || dist === minDist && diff >= 0 && minDiff < 0) { - minDist = dist; - minDiff = diff; - nearestIndicesLen = 0; - } - - if (diff === minDiff) { - nearestIndices[nearestIndicesLen++] = i; - } - } - } - - nearestIndices.length = nearestIndicesLen; - return nearestIndices; - }; - - DataStore.prototype.getIndices = function () { - var newIndices; - var indices = this._indices; - - if (indices) { - var Ctor = indices.constructor; - var thisCount = this._count; // `new Array(a, b, c)` is different from `new Uint32Array(a, b, c)`. - - if (Ctor === Array) { - newIndices = new Ctor(thisCount); - - for (var i = 0; i < thisCount; i++) { - newIndices[i] = indices[i]; - } - } else { - newIndices = new Ctor(indices.buffer, 0, thisCount); - } - } else { - var Ctor = getIndicesCtor(this._rawCount); - newIndices = new Ctor(this.count()); - - for (var i = 0; i < newIndices.length; i++) { - newIndices[i] = i; - } - } - - return newIndices; - }; - /** - * Data filter. - */ - - - DataStore.prototype.filter = function (dims, cb) { - if (!this._count) { - return this; - } - - var newStore = this.clone(); - var count = newStore.count(); - var Ctor = getIndicesCtor(newStore._rawCount); - var newIndices = new Ctor(count); - var value = []; - var dimSize = dims.length; - var offset = 0; - var dim0 = dims[0]; - var chunks = newStore._chunks; - - for (var i = 0; i < count; i++) { - var keep = void 0; - var rawIdx = newStore.getRawIndex(i); // Simple optimization - - if (dimSize === 0) { - keep = cb(i); - } else if (dimSize === 1) { - var val = chunks[dim0][rawIdx]; - keep = cb(val, i); - } else { - var k = 0; - - for (; k < dimSize; k++) { - value[k] = chunks[dims[k]][rawIdx]; - } - - value[k] = i; - keep = cb.apply(null, value); - } - - if (keep) { - newIndices[offset++] = rawIdx; - } - } // Set indices after filtered. - - - if (offset < count) { - newStore._indices = newIndices; - } - - newStore._count = offset; // Reset data extent - - newStore._extent = []; - - newStore._updateGetRawIdx(); - - return newStore; - }; - /** - * Select data in range. (For optimization of filter) - * (Manually inline code, support 5 million data filtering in data zoom.) - */ - - - DataStore.prototype.selectRange = function (range) { - var newStore = this.clone(); - var len = newStore._count; - - if (!len) { - return this; - } - - var dims = keys(range); - var dimSize = dims.length; - - if (!dimSize) { - return this; - } - - var originalCount = newStore.count(); - var Ctor = getIndicesCtor(newStore._rawCount); - var newIndices = new Ctor(originalCount); - var offset = 0; - var dim0 = dims[0]; - var min = range[dim0][0]; - var max = range[dim0][1]; - var storeArr = newStore._chunks; - var quickFinished = false; - - if (!newStore._indices) { - // Extreme optimization for common case. About 2x faster in chrome. - var idx = 0; - - if (dimSize === 1) { - var dimStorage = storeArr[dims[0]]; - - for (var i = 0; i < len; i++) { - var val = dimStorage[i]; // NaN will not be filtered. Consider the case, in line chart, empty - // value indicates the line should be broken. But for the case like - // scatter plot, a data item with empty value will not be rendered, - // but the axis extent may be effected if some other dim of the data - // item has value. Fortunately it is not a significant negative effect. - - if (val >= min && val <= max || isNaN(val)) { - newIndices[offset++] = idx; - } - - idx++; - } - - quickFinished = true; - } else if (dimSize === 2) { - var dimStorage = storeArr[dims[0]]; - var dimStorage2 = storeArr[dims[1]]; - var min2 = range[dims[1]][0]; - var max2 = range[dims[1]][1]; - - for (var i = 0; i < len; i++) { - var val = dimStorage[i]; - var val2 = dimStorage2[i]; // Do not filter NaN, see comment above. - - if ((val >= min && val <= max || isNaN(val)) && (val2 >= min2 && val2 <= max2 || isNaN(val2))) { - newIndices[offset++] = idx; - } - - idx++; - } - - quickFinished = true; - } - } - - if (!quickFinished) { - if (dimSize === 1) { - for (var i = 0; i < originalCount; i++) { - var rawIndex = newStore.getRawIndex(i); - var val = storeArr[dims[0]][rawIndex]; // Do not filter NaN, see comment above. - - if (val >= min && val <= max || isNaN(val)) { - newIndices[offset++] = rawIndex; - } - } - } else { - for (var i = 0; i < originalCount; i++) { - var keep = true; - var rawIndex = newStore.getRawIndex(i); - - for (var k = 0; k < dimSize; k++) { - var dimk = dims[k]; - var val = storeArr[dimk][rawIndex]; // Do not filter NaN, see comment above. - - if (val < range[dimk][0] || val > range[dimk][1]) { - keep = false; - } - } - - if (keep) { - newIndices[offset++] = newStore.getRawIndex(i); - } - } - } - } // Set indices after filtered. - - - if (offset < originalCount) { - newStore._indices = newIndices; - } - - newStore._count = offset; // Reset data extent - - newStore._extent = []; - - newStore._updateGetRawIdx(); - - return newStore; - }; // /** - // * Data mapping to a plain array - // */ - // mapArray(dims: DimensionIndex[], cb: MapArrayCb): any[] { - // const result: any[] = []; - // this.each(dims, function () { - // result.push(cb && (cb as MapArrayCb).apply(null, arguments)); - // }); - // return result; - // } - - /** - * Data mapping to a new List with given dimensions - */ - - - DataStore.prototype.map = function (dims, cb) { - // TODO only clone picked chunks. - var target = this.clone(dims); - - this._updateDims(target, dims, cb); - - return target; - }; - /** - * @caution Danger!! Only used in dataStack. - */ - - - DataStore.prototype.modify = function (dims, cb) { - this._updateDims(this, dims, cb); - }; - - DataStore.prototype._updateDims = function (target, dims, cb) { - var targetChunks = target._chunks; - var tmpRetValue = []; - var dimSize = dims.length; - var dataCount = target.count(); - var values = []; - var rawExtent = target._rawExtent; - - for (var i = 0; i < dims.length; i++) { - rawExtent[dims[i]] = getInitialExtent(); - } - - for (var dataIndex = 0; dataIndex < dataCount; dataIndex++) { - var rawIndex = target.getRawIndex(dataIndex); - - for (var k = 0; k < dimSize; k++) { - values[k] = targetChunks[dims[k]][rawIndex]; - } - - values[dimSize] = dataIndex; - var retValue = cb && cb.apply(null, values); - - if (retValue != null) { - // a number or string (in oridinal dimension)? - if (typeof retValue !== 'object') { - tmpRetValue[0] = retValue; - retValue = tmpRetValue; - } - - for (var i = 0; i < retValue.length; i++) { - var dim = dims[i]; - var val = retValue[i]; - var rawExtentOnDim = rawExtent[dim]; - var dimStore = targetChunks[dim]; - - if (dimStore) { - dimStore[rawIndex] = val; - } - - if (val < rawExtentOnDim[0]) { - rawExtentOnDim[0] = val; - } - - if (val > rawExtentOnDim[1]) { - rawExtentOnDim[1] = val; - } - } - } - } - }; - /** - * Large data down sampling using largest-triangle-three-buckets - * @param {string} valueDimension - * @param {number} targetCount - */ - - - DataStore.prototype.lttbDownSample = function (valueDimension, rate) { - var target = this.clone([valueDimension], true); - var targetStorage = target._chunks; - var dimStore = targetStorage[valueDimension]; - var len = this.count(); - var sampledIndex = 0; - var frameSize = Math.floor(1 / rate); - var currentRawIndex = this.getRawIndex(0); - var maxArea; - var area; - var nextRawIndex; - var newIndices = new (getIndicesCtor(this._rawCount))(Math.min((Math.ceil(len / frameSize) + 2) * 2, len)); // First frame use the first data. - - newIndices[sampledIndex++] = currentRawIndex; - - for (var i = 1; i < len - 1; i += frameSize) { - var nextFrameStart = Math.min(i + frameSize, len - 1); - var nextFrameEnd = Math.min(i + frameSize * 2, len); - var avgX = (nextFrameEnd + nextFrameStart) / 2; - var avgY = 0; - - for (var idx = nextFrameStart; idx < nextFrameEnd; idx++) { - var rawIndex = this.getRawIndex(idx); - var y = dimStore[rawIndex]; - - if (isNaN(y)) { - continue; - } - - avgY += y; - } - - avgY /= nextFrameEnd - nextFrameStart; - var frameStart = i; - var frameEnd = Math.min(i + frameSize, len); - var pointAX = i - 1; - var pointAY = dimStore[currentRawIndex]; - maxArea = -1; - nextRawIndex = frameStart; - var firstNaNIndex = -1; - var countNaN = 0; // Find a point from current frame that construct a triangle with largest area with previous selected point - // And the average of next frame. - - for (var idx = frameStart; idx < frameEnd; idx++) { - var rawIndex = this.getRawIndex(idx); - var y = dimStore[rawIndex]; - - if (isNaN(y)) { - countNaN++; - - if (firstNaNIndex < 0) { - firstNaNIndex = rawIndex; - } - - continue; - } // Calculate triangle area over three buckets - - - area = Math.abs((pointAX - avgX) * (y - pointAY) - (pointAX - idx) * (avgY - pointAY)); - - if (area > maxArea) { - maxArea = area; - nextRawIndex = rawIndex; // Next a is this b - } - } - - if (countNaN > 0 && countNaN < frameEnd - frameStart) { - // Append first NaN point in every bucket. - // It is necessary to ensure the correct order of indices. - newIndices[sampledIndex++] = Math.min(firstNaNIndex, nextRawIndex); - nextRawIndex = Math.max(firstNaNIndex, nextRawIndex); - } - - newIndices[sampledIndex++] = nextRawIndex; - currentRawIndex = nextRawIndex; // This a is the next a (chosen b) - } // First frame use the last data. - - - newIndices[sampledIndex++] = this.getRawIndex(len - 1); - target._count = sampledIndex; - target._indices = newIndices; - target.getRawIndex = this._getRawIdx; - return target; - }; - /** - * Large data down sampling on given dimension - * @param sampleIndex Sample index for name and id - */ - - - DataStore.prototype.downSample = function (dimension, rate, sampleValue, sampleIndex) { - var target = this.clone([dimension], true); - var targetStorage = target._chunks; - var frameValues = []; - var frameSize = Math.floor(1 / rate); - var dimStore = targetStorage[dimension]; - var len = this.count(); - var rawExtentOnDim = target._rawExtent[dimension] = getInitialExtent(); - var newIndices = new (getIndicesCtor(this._rawCount))(Math.ceil(len / frameSize)); - var offset = 0; - - for (var i = 0; i < len; i += frameSize) { - // Last frame - if (frameSize > len - i) { - frameSize = len - i; - frameValues.length = frameSize; - } - - for (var k = 0; k < frameSize; k++) { - var dataIdx = this.getRawIndex(i + k); - frameValues[k] = dimStore[dataIdx]; - } - - var value = sampleValue(frameValues); - var sampleFrameIdx = this.getRawIndex(Math.min(i + sampleIndex(frameValues, value) || 0, len - 1)); // Only write value on the filtered data - - dimStore[sampleFrameIdx] = value; - - if (value < rawExtentOnDim[0]) { - rawExtentOnDim[0] = value; - } - - if (value > rawExtentOnDim[1]) { - rawExtentOnDim[1] = value; - } - - newIndices[offset++] = sampleFrameIdx; - } - - target._count = offset; - target._indices = newIndices; - - target._updateGetRawIdx(); - - return target; - }; - /** - * Data iteration - * @param ctx default this - * @example - * list.each('x', function (x, idx) {}); - * list.each(['x', 'y'], function (x, y, idx) {}); - * list.each(function (idx) {}) - */ - - - DataStore.prototype.each = function (dims, cb) { - if (!this._count) { - return; - } - - var dimSize = dims.length; - var chunks = this._chunks; - - for (var i = 0, len = this.count(); i < len; i++) { - var rawIdx = this.getRawIndex(i); // Simple optimization - - switch (dimSize) { - case 0: - cb(i); - break; - - case 1: - cb(chunks[dims[0]][rawIdx], i); - break; - - case 2: - cb(chunks[dims[0]][rawIdx], chunks[dims[1]][rawIdx], i); - break; - - default: - var k = 0; - var value = []; - - for (; k < dimSize; k++) { - value[k] = chunks[dims[k]][rawIdx]; - } // Index - - - value[k] = i; - cb.apply(null, value); - } - } - }; - /** - * Get extent of data in one dimension - */ - - - DataStore.prototype.getDataExtent = function (dim) { - // Make sure use concrete dim as cache name. - var dimData = this._chunks[dim]; - var initialExtent = getInitialExtent(); - - if (!dimData) { - return initialExtent; - } // Make more strict checkings to ensure hitting cache. - - - var currEnd = this.count(); // Consider the most cases when using data zoom, `getDataExtent` - // happened before filtering. We cache raw extent, which is not - // necessary to be cleared and recalculated when restore data. - - var useRaw = !this._indices; - var dimExtent; - - if (useRaw) { - return this._rawExtent[dim].slice(); - } - - dimExtent = this._extent[dim]; - - if (dimExtent) { - return dimExtent.slice(); - } - - dimExtent = initialExtent; - var min = dimExtent[0]; - var max = dimExtent[1]; - - for (var i = 0; i < currEnd; i++) { - var rawIdx = this.getRawIndex(i); - var value = dimData[rawIdx]; - value < min && (min = value); - value > max && (max = value); - } - - dimExtent = [min, max]; - this._extent[dim] = dimExtent; - return dimExtent; - }; - /** - * Get raw data item - */ - - - DataStore.prototype.getRawDataItem = function (idx) { - var rawIdx = this.getRawIndex(idx); - - if (!this._provider.persistent) { - var val = []; - var chunks = this._chunks; - - for (var i = 0; i < chunks.length; i++) { - val.push(chunks[i][rawIdx]); - } - - return val; - } else { - return this._provider.getItem(rawIdx); - } - }; - /** - * Clone shallow. - * - * @param clonedDims Determine which dims to clone. Will share the data if not specified. - */ - - - DataStore.prototype.clone = function (clonedDims, ignoreIndices) { - var target = new DataStore(); - var chunks = this._chunks; - var clonedDimsMap = clonedDims && reduce(clonedDims, function (obj, dimIdx) { - obj[dimIdx] = true; - return obj; - }, {}); - - if (clonedDimsMap) { - for (var i = 0; i < chunks.length; i++) { - // Not clone if dim is not picked. - target._chunks[i] = !clonedDimsMap[i] ? chunks[i] : cloneChunk(chunks[i]); - } - } else { - target._chunks = chunks; - } - - this._copyCommonProps(target); - - if (!ignoreIndices) { - target._indices = this._cloneIndices(); - } - - target._updateGetRawIdx(); - - return target; - }; - - DataStore.prototype._copyCommonProps = function (target) { - target._count = this._count; - target._rawCount = this._rawCount; - target._provider = this._provider; - target._dimensions = this._dimensions; - target._extent = clone(this._extent); - target._rawExtent = clone(this._rawExtent); - }; - - DataStore.prototype._cloneIndices = function () { - if (this._indices) { - var Ctor = this._indices.constructor; - var indices = void 0; - - if (Ctor === Array) { - var thisCount = this._indices.length; - indices = new Ctor(thisCount); - - for (var i = 0; i < thisCount; i++) { - indices[i] = this._indices[i]; - } - } else { - indices = new Ctor(this._indices); - } - - return indices; - } - - return null; - }; - - DataStore.prototype._getRawIdxIdentity = function (idx) { - return idx; - }; - - DataStore.prototype._getRawIdx = function (idx) { - if (idx < this._count && idx >= 0) { - return this._indices[idx]; - } - - return -1; - }; - - DataStore.prototype._updateGetRawIdx = function () { - this.getRawIndex = this._indices ? this._getRawIdx : this._getRawIdxIdentity; - }; - - DataStore.internalField = function () { - function getDimValueSimply(dataItem, property, dataIndex, dimIndex) { - return parseDataValue(dataItem[dimIndex], this._dimensions[dimIndex]); - } - - defaultDimValueGetters = { - arrayRows: getDimValueSimply, - objectRows: function (dataItem, property, dataIndex, dimIndex) { - return parseDataValue(dataItem[property], this._dimensions[dimIndex]); - }, - keyedColumns: getDimValueSimply, - original: function (dataItem, property, dataIndex, dimIndex) { - // Performance sensitive, do not use modelUtil.getDataItemValue. - // If dataItem is an plain object with no value field, the let `value` - // will be assigned with the object, but it will be tread correctly - // in the `convertValue`. - var value = dataItem && (dataItem.value == null ? dataItem : dataItem.value); - return parseDataValue(value instanceof Array ? value[dimIndex] // If value is a single number or something else not array. - : value, this._dimensions[dimIndex]); - }, - typedArray: function (dataItem, property, dataIndex, dimIndex) { - return dataItem[dimIndex]; - } - }; - }(); - - return DataStore; - }(); - - /** - * [REQUIREMENT_MEMO]: - * (0) `metaRawOption` means `dimensions`/`sourceHeader`/`seriesLayoutBy` in raw option. - * (1) Keep support the feature: `metaRawOption` can be specified both on `series` and - * `root-dataset`. Them on `series` has higher priority. - * (2) Do not support to set `metaRawOption` on a `non-root-dataset`, because it might - * confuse users: whether those props indicate how to visit the upstream source or visit - * the transform result source, and some transforms has nothing to do with these props, - * and some transforms might have multiple upstream. - * (3) Transforms should specify `metaRawOption` in each output, just like they can be - * declared in `root-dataset`. - * (4) At present only support visit source in `SERIES_LAYOUT_BY_COLUMN` in transforms. - * That is for reducing complexity in transforms. - * PENDING: Whether to provide transposition transform? - * - * [IMPLEMENTAION_MEMO]: - * "sourceVisitConfig" are calculated from `metaRawOption` and `data`. - * They will not be calculated until `source` is about to be visited (to prevent from - * duplicate calcuation). `source` is visited only in series and input to transforms. - * - * [DIMENSION_INHERIT_RULE]: - * By default the dimensions are inherited from ancestors, unless a transform return - * a new dimensions definition. - * Consider the case: - * ```js - * dataset: [{ - * source: [ ['Product', 'Sales', 'Prise'], ['Cookies', 321, 44.21], ...] - * }, { - * transform: { type: 'filter', ... } - * }] - * dataset: [{ - * dimension: ['Product', 'Sales', 'Prise'], - * source: [ ['Cookies', 321, 44.21], ...] - * }, { - * transform: { type: 'filter', ... } - * }] - * ``` - * The two types of option should have the same behavior after transform. - * - * - * [SCENARIO]: - * (1) Provide source data directly: - * ```js - * series: { - * encode: {...}, - * dimensions: [...] - * seriesLayoutBy: 'row', - * data: [[...]] - * } - * ``` - * (2) Series refer to dataset. - * ```js - * series: [{ - * encode: {...} - * // Ignore datasetIndex means `datasetIndex: 0` - * // and the dimensions defination in dataset is used - * }, { - * encode: {...}, - * seriesLayoutBy: 'column', - * datasetIndex: 1 - * }] - * ``` - * (3) dataset transform - * ```js - * dataset: [{ - * source: [...] - * }, { - * source: [...] - * }, { - * // By default from 0. - * transform: { type: 'filter', config: {...} } - * }, { - * // Piped. - * transform: [ - * { type: 'filter', config: {...} }, - * { type: 'sort', config: {...} } - * ] - * }, { - * id: 'regressionData', - * fromDatasetIndex: 1, - * // Third-party transform - * transform: { type: 'ecStat:regression', config: {...} } - * }, { - * // retrieve the extra result. - * id: 'regressionFormula', - * fromDatasetId: 'regressionData', - * fromTransformResult: 1 - * }] - * ``` - */ - - var SourceManager = - /** @class */ - function () { - function SourceManager(sourceHost) { - // Cached source. Do not repeat calculating if not dirty. - this._sourceList = []; - this._storeList = []; // version sign of each upstream source manager. - - this._upstreamSignList = []; - this._versionSignBase = 0; - this._dirty = true; - this._sourceHost = sourceHost; - } - /** - * Mark dirty. - */ - - - SourceManager.prototype.dirty = function () { - this._setLocalSource([], []); - - this._storeList = []; - this._dirty = true; - }; - - SourceManager.prototype._setLocalSource = function (sourceList, upstreamSignList) { - this._sourceList = sourceList; - this._upstreamSignList = upstreamSignList; - this._versionSignBase++; - - if (this._versionSignBase > 9e10) { - this._versionSignBase = 0; - } - }; - /** - * For detecting whether the upstream source is dirty, so that - * the local cached source (in `_sourceList`) should be discarded. - */ - - - SourceManager.prototype._getVersionSign = function () { - return this._sourceHost.uid + '_' + this._versionSignBase; - }; - /** - * Always return a source instance. Otherwise throw error. - */ - - - SourceManager.prototype.prepareSource = function () { - // For the case that call `setOption` multiple time but no data changed, - // cache the result source to prevent from repeating transform. - if (this._isDirty()) { - this._createSource(); - - this._dirty = false; - } - }; - - SourceManager.prototype._createSource = function () { - this._setLocalSource([], []); - - var sourceHost = this._sourceHost; - - var upSourceMgrList = this._getUpstreamSourceManagers(); - - var hasUpstream = !!upSourceMgrList.length; - var resultSourceList; - var upstreamSignList; - - if (isSeries(sourceHost)) { - var seriesModel = sourceHost; - var data = void 0; - var sourceFormat = void 0; - var upSource = void 0; // Has upstream dataset - - if (hasUpstream) { - var upSourceMgr = upSourceMgrList[0]; - upSourceMgr.prepareSource(); - upSource = upSourceMgr.getSource(); - data = upSource.data; - sourceFormat = upSource.sourceFormat; - upstreamSignList = [upSourceMgr._getVersionSign()]; - } // Series data is from own. - else { - data = seriesModel.get('data', true); - sourceFormat = isTypedArray(data) ? SOURCE_FORMAT_TYPED_ARRAY : SOURCE_FORMAT_ORIGINAL; - upstreamSignList = []; - } // See [REQUIREMENT_MEMO], merge settings on series and parent dataset if it is root. - - - var newMetaRawOption = this._getSourceMetaRawOption() || {}; - var upMetaRawOption = upSource && upSource.metaRawOption || {}; - var seriesLayoutBy = retrieve2(newMetaRawOption.seriesLayoutBy, upMetaRawOption.seriesLayoutBy) || null; - var sourceHeader = retrieve2(newMetaRawOption.sourceHeader, upMetaRawOption.sourceHeader); // Note here we should not use `upSource.dimensionsDefine`. Consider the case: - // `upSource.dimensionsDefine` is detected by `seriesLayoutBy: 'column'`, - // but series need `seriesLayoutBy: 'row'`. - - var dimensions = retrieve2(newMetaRawOption.dimensions, upMetaRawOption.dimensions); // We share source with dataset as much as possible - // to avoid extra memory cost of high dimensional data. - - var needsCreateSource = seriesLayoutBy !== upMetaRawOption.seriesLayoutBy || !!sourceHeader !== !!upMetaRawOption.sourceHeader || dimensions; - resultSourceList = needsCreateSource ? [createSource(data, { - seriesLayoutBy: seriesLayoutBy, - sourceHeader: sourceHeader, - dimensions: dimensions - }, sourceFormat)] : []; - } else { - var datasetModel = sourceHost; // Has upstream dataset. - - if (hasUpstream) { - var result = this._applyTransform(upSourceMgrList); - - resultSourceList = result.sourceList; - upstreamSignList = result.upstreamSignList; - } // Is root dataset. - else { - var sourceData = datasetModel.get('source', true); - resultSourceList = [createSource(sourceData, this._getSourceMetaRawOption(), null)]; - upstreamSignList = []; - } - } - - if ("development" !== 'production') { - assert(resultSourceList && upstreamSignList); - } - - this._setLocalSource(resultSourceList, upstreamSignList); - }; - - SourceManager.prototype._applyTransform = function (upMgrList) { - var datasetModel = this._sourceHost; - var transformOption = datasetModel.get('transform', true); - var fromTransformResult = datasetModel.get('fromTransformResult', true); - - if ("development" !== 'production') { - assert(fromTransformResult != null || transformOption != null); - } - - if (fromTransformResult != null) { - var errMsg = ''; - - if (upMgrList.length !== 1) { - if ("development" !== 'production') { - errMsg = 'When using `fromTransformResult`, there should be only one upstream dataset'; - } - - doThrow(errMsg); - } - } - - var sourceList; - var upSourceList = []; - var upstreamSignList = []; - each(upMgrList, function (upMgr) { - upMgr.prepareSource(); - var upSource = upMgr.getSource(fromTransformResult || 0); - var errMsg = ''; - - if (fromTransformResult != null && !upSource) { - if ("development" !== 'production') { - errMsg = 'Can not retrieve result by `fromTransformResult`: ' + fromTransformResult; - } - - doThrow(errMsg); - } - - upSourceList.push(upSource); - upstreamSignList.push(upMgr._getVersionSign()); - }); - - if (transformOption) { - sourceList = applyDataTransform(transformOption, upSourceList, { - datasetIndex: datasetModel.componentIndex - }); - } else if (fromTransformResult != null) { - sourceList = [cloneSourceShallow(upSourceList[0])]; - } - - return { - sourceList: sourceList, - upstreamSignList: upstreamSignList - }; - }; - - SourceManager.prototype._isDirty = function () { - if (this._dirty) { - return true; - } // All sourceList is from the some upstream. - - - var upSourceMgrList = this._getUpstreamSourceManagers(); - - for (var i = 0; i < upSourceMgrList.length; i++) { - var upSrcMgr = upSourceMgrList[i]; - - if ( // Consider the case that there is ancestor diry, call it recursively. - // The performance is probably not an issue because usually the chain is not long. - upSrcMgr._isDirty() || this._upstreamSignList[i] !== upSrcMgr._getVersionSign()) { - return true; - } - } - }; - /** - * @param sourceIndex By default 0, means "main source". - * In most cases there is only one source. - */ - - - SourceManager.prototype.getSource = function (sourceIndex) { - sourceIndex = sourceIndex || 0; - var source = this._sourceList[sourceIndex]; - - if (!source) { - // Series may share source instance with dataset. - var upSourceMgrList = this._getUpstreamSourceManagers(); - - return upSourceMgrList[0] && upSourceMgrList[0].getSource(sourceIndex); - } - - return source; - }; - /** - * - * Get a data store which can be shared across series. - * Only available for series. - * - * @param seriesDimRequest Dimensions that are generated in series. - * Should have been sorted by `storeDimIndex` asc. - */ - - - SourceManager.prototype.getSharedDataStore = function (seriesDimRequest) { - if ("development" !== 'production') { - assert(isSeries(this._sourceHost), 'Can only call getDataStore on series source manager.'); - } - - var schema = seriesDimRequest.makeStoreSchema(); - return this._innerGetDataStore(schema.dimensions, seriesDimRequest.source, schema.hash); - }; - - SourceManager.prototype._innerGetDataStore = function (storeDims, seriesSource, sourceReadKey) { - // TODO Can use other sourceIndex? - var sourceIndex = 0; - var storeList = this._storeList; - var cachedStoreMap = storeList[sourceIndex]; - - if (!cachedStoreMap) { - cachedStoreMap = storeList[sourceIndex] = {}; - } - - var cachedStore = cachedStoreMap[sourceReadKey]; - - if (!cachedStore) { - var upSourceMgr = this._getUpstreamSourceManagers()[0]; - - if (isSeries(this._sourceHost) && upSourceMgr) { - cachedStore = upSourceMgr._innerGetDataStore(storeDims, seriesSource, sourceReadKey); - } else { - cachedStore = new DataStore(); // Always create store from source of series. - - cachedStore.initData(new DefaultDataProvider(seriesSource, storeDims.length), storeDims); - } - - cachedStoreMap[sourceReadKey] = cachedStore; - } - - return cachedStore; - }; - /** - * PENDING: Is it fast enough? - * If no upstream, return empty array. - */ - - - SourceManager.prototype._getUpstreamSourceManagers = function () { - // Always get the relationship from the raw option. - // Do not cache the link of the dependency graph, so that - // there is no need to update them when change happens. - var sourceHost = this._sourceHost; - - if (isSeries(sourceHost)) { - var datasetModel = querySeriesUpstreamDatasetModel(sourceHost); - return !datasetModel ? [] : [datasetModel.getSourceManager()]; - } else { - return map(queryDatasetUpstreamDatasetModels(sourceHost), function (datasetModel) { - return datasetModel.getSourceManager(); - }); - } - }; - - SourceManager.prototype._getSourceMetaRawOption = function () { - var sourceHost = this._sourceHost; - var seriesLayoutBy; - var sourceHeader; - var dimensions; - - if (isSeries(sourceHost)) { - seriesLayoutBy = sourceHost.get('seriesLayoutBy', true); - sourceHeader = sourceHost.get('sourceHeader', true); - dimensions = sourceHost.get('dimensions', true); - } // See [REQUIREMENT_MEMO], `non-root-dataset` do not support them. - else if (!this._getUpstreamSourceManagers().length) { - var model = sourceHost; - seriesLayoutBy = model.get('seriesLayoutBy', true); - sourceHeader = model.get('sourceHeader', true); - dimensions = model.get('dimensions', true); - } - - return { - seriesLayoutBy: seriesLayoutBy, - sourceHeader: sourceHeader, - dimensions: dimensions - }; - }; - - return SourceManager; - }(); - // disable the transform merge, but do not disable transform clone from rawOption. - - function disableTransformOptionMerge(datasetModel) { - var transformOption = datasetModel.option.transform; - transformOption && setAsPrimitive(datasetModel.option.transform); - } - - function isSeries(sourceHost) { - // Avoid circular dependency with Series.ts - return sourceHost.mainType === 'series'; - } - - function doThrow(errMsg) { - throw new Error(errMsg); - } - - var TOOLTIP_LINE_HEIGHT_CSS = 'line-height:1'; // TODO: more textStyle option - - function getTooltipTextStyle(textStyle, renderMode) { - var nameFontColor = textStyle.color || '#6e7079'; - var nameFontSize = textStyle.fontSize || 12; - var nameFontWeight = textStyle.fontWeight || '400'; - var valueFontColor = textStyle.color || '#464646'; - var valueFontSize = textStyle.fontSize || 14; - var valueFontWeight = textStyle.fontWeight || '900'; - - if (renderMode === 'html') { - // `textStyle` is probably from user input, should be encoded to reduce security risk. - return { - // eslint-disable-next-line max-len - nameStyle: "font-size:" + encodeHTML(nameFontSize + '') + "px;color:" + encodeHTML(nameFontColor) + ";font-weight:" + encodeHTML(nameFontWeight + ''), - // eslint-disable-next-line max-len - valueStyle: "font-size:" + encodeHTML(valueFontSize + '') + "px;color:" + encodeHTML(valueFontColor) + ";font-weight:" + encodeHTML(valueFontWeight + '') - }; - } else { - return { - nameStyle: { - fontSize: nameFontSize, - fill: nameFontColor, - fontWeight: nameFontWeight - }, - valueStyle: { - fontSize: valueFontSize, - fill: valueFontColor, - fontWeight: valueFontWeight - } - }; - } - } // See `TooltipMarkupLayoutIntent['innerGapLevel']`. - // (value from UI design) - - - var HTML_GAPS = [0, 10, 20, 30]; - var RICH_TEXT_GAPS = ['', '\n', '\n\n', '\n\n\n']; // eslint-disable-next-line max-len - - function createTooltipMarkup(type, option) { - option.type = type; - return option; - } - - function isSectionFragment(frag) { - return frag.type === 'section'; - } - - function getBuilder(frag) { - return isSectionFragment(frag) ? buildSection : buildNameValue; - } - - function getBlockGapLevel(frag) { - if (isSectionFragment(frag)) { - var gapLevel_1 = 0; - var subBlockLen = frag.blocks.length; - var hasInnerGap_1 = subBlockLen > 1 || subBlockLen > 0 && !frag.noHeader; - each(frag.blocks, function (subBlock) { - var subGapLevel = getBlockGapLevel(subBlock); // If the some of the sub-blocks have some gaps (like 10px) inside, this block - // should use a larger gap (like 20px) to distinguish those sub-blocks. - - if (subGapLevel >= gapLevel_1) { - gapLevel_1 = subGapLevel + +(hasInnerGap_1 && ( // 0 always can not be readable gap level. - !subGapLevel // If no header, always keep the sub gap level. Otherwise - // look weird in case `multipleSeries`. - || isSectionFragment(subBlock) && !subBlock.noHeader)); - } - }); - return gapLevel_1; - } - - return 0; - } - - function buildSection(ctx, fragment, topMarginForOuterGap, toolTipTextStyle) { - var noHeader = fragment.noHeader; - var gaps = getGap(getBlockGapLevel(fragment)); - var subMarkupTextList = []; - var subBlocks = fragment.blocks || []; - assert(!subBlocks || isArray(subBlocks)); - subBlocks = subBlocks || []; - var orderMode = ctx.orderMode; - - if (fragment.sortBlocks && orderMode) { - subBlocks = subBlocks.slice(); - var orderMap = { - valueAsc: 'asc', - valueDesc: 'desc' - }; - - if (hasOwn(orderMap, orderMode)) { - var comparator_1 = new SortOrderComparator(orderMap[orderMode], null); - subBlocks.sort(function (a, b) { - return comparator_1.evaluate(a.sortParam, b.sortParam); - }); - } // FIXME 'seriesDesc' necessary? - else if (orderMode === 'seriesDesc') { - subBlocks.reverse(); - } - } - - each(subBlocks, function (subBlock, idx) { - var valueFormatter = fragment.valueFormatter; - var subMarkupText = getBuilder(subBlock)( // Inherit valueFormatter - valueFormatter ? extend(extend({}, ctx), { - valueFormatter: valueFormatter - }) : ctx, subBlock, idx > 0 ? gaps.html : 0, toolTipTextStyle); - subMarkupText != null && subMarkupTextList.push(subMarkupText); - }); - var subMarkupText = ctx.renderMode === 'richText' ? subMarkupTextList.join(gaps.richText) : wrapBlockHTML(subMarkupTextList.join(''), noHeader ? topMarginForOuterGap : gaps.html); - - if (noHeader) { - return subMarkupText; - } - - var displayableHeader = makeValueReadable(fragment.header, 'ordinal', ctx.useUTC); - var nameStyle = getTooltipTextStyle(toolTipTextStyle, ctx.renderMode).nameStyle; - - if (ctx.renderMode === 'richText') { - return wrapInlineNameRichText(ctx, displayableHeader, nameStyle) + gaps.richText + subMarkupText; - } else { - return wrapBlockHTML("<div style=\"" + nameStyle + ";" + TOOLTIP_LINE_HEIGHT_CSS + ";\">" + encodeHTML(displayableHeader) + '</div>' + subMarkupText, topMarginForOuterGap); - } - } - - function buildNameValue(ctx, fragment, topMarginForOuterGap, toolTipTextStyle) { - var renderMode = ctx.renderMode; - var noName = fragment.noName; - var noValue = fragment.noValue; - var noMarker = !fragment.markerType; - var name = fragment.name; - var useUTC = ctx.useUTC; - - var valueFormatter = fragment.valueFormatter || ctx.valueFormatter || function (value) { - value = isArray(value) ? value : [value]; - return map(value, function (val, idx) { - return makeValueReadable(val, isArray(valueTypeOption) ? valueTypeOption[idx] : valueTypeOption, useUTC); - }); - }; - - if (noName && noValue) { - return; - } - - var markerStr = noMarker ? '' : ctx.markupStyleCreator.makeTooltipMarker(fragment.markerType, fragment.markerColor || '#333', renderMode); - var readableName = noName ? '' : makeValueReadable(name, 'ordinal', useUTC); - var valueTypeOption = fragment.valueType; - var readableValueList = noValue ? [] : valueFormatter(fragment.value); - var valueAlignRight = !noMarker || !noName; // It little weird if only value next to marker but far from marker. - - var valueCloseToMarker = !noMarker && noName; - - var _a = getTooltipTextStyle(toolTipTextStyle, renderMode), - nameStyle = _a.nameStyle, - valueStyle = _a.valueStyle; - - return renderMode === 'richText' ? (noMarker ? '' : markerStr) + (noName ? '' : wrapInlineNameRichText(ctx, readableName, nameStyle)) // Value has commas inside, so use ' ' as delimiter for multiple values. - + (noValue ? '' : wrapInlineValueRichText(ctx, readableValueList, valueAlignRight, valueCloseToMarker, valueStyle)) : wrapBlockHTML((noMarker ? '' : markerStr) + (noName ? '' : wrapInlineNameHTML(readableName, !noMarker, nameStyle)) + (noValue ? '' : wrapInlineValueHTML(readableValueList, valueAlignRight, valueCloseToMarker, valueStyle)), topMarginForOuterGap); - } - /** - * @return markupText. null/undefined means no content. - */ - - - function buildTooltipMarkup(fragment, markupStyleCreator, renderMode, orderMode, useUTC, toolTipTextStyle) { - if (!fragment) { - return; - } - - var builder = getBuilder(fragment); - var ctx = { - useUTC: useUTC, - renderMode: renderMode, - orderMode: orderMode, - markupStyleCreator: markupStyleCreator, - valueFormatter: fragment.valueFormatter - }; - return builder(ctx, fragment, 0, toolTipTextStyle); - } - - function getGap(gapLevel) { - return { - html: HTML_GAPS[gapLevel], - richText: RICH_TEXT_GAPS[gapLevel] - }; - } - - function wrapBlockHTML(encodedContent, topGap) { - var clearfix = '<div style="clear:both"></div>'; - var marginCSS = "margin: " + topGap + "px 0 0"; - return "<div style=\"" + marginCSS + ";" + TOOLTIP_LINE_HEIGHT_CSS + ";\">" + encodedContent + clearfix + '</div>'; - } - - function wrapInlineNameHTML(name, leftHasMarker, style) { - var marginCss = leftHasMarker ? 'margin-left:2px' : ''; - return "<span style=\"" + style + ";" + marginCss + "\">" + encodeHTML(name) + '</span>'; - } - - function wrapInlineValueHTML(valueList, alignRight, valueCloseToMarker, style) { - // Do not too close to marker, considering there are multiple values separated by spaces. - var paddingStr = valueCloseToMarker ? '10px' : '20px'; - var alignCSS = alignRight ? "float:right;margin-left:" + paddingStr : ''; - valueList = isArray(valueList) ? valueList : [valueList]; - return "<span style=\"" + alignCSS + ";" + style + "\">" // Value has commas inside, so use ' ' as delimiter for multiple values. - + map(valueList, function (value) { - return encodeHTML(value); - }).join('&nbsp;&nbsp;') + '</span>'; - } - - function wrapInlineNameRichText(ctx, name, style) { - return ctx.markupStyleCreator.wrapRichTextStyle(name, style); - } - - function wrapInlineValueRichText(ctx, values, alignRight, valueCloseToMarker, style) { - var styles = [style]; - var paddingLeft = valueCloseToMarker ? 10 : 20; - alignRight && styles.push({ - padding: [0, 0, 0, paddingLeft], - align: 'right' - }); // Value has commas inside, so use ' ' as delimiter for multiple values. - - return ctx.markupStyleCreator.wrapRichTextStyle(isArray(values) ? values.join(' ') : values, styles); - } - - function retrieveVisualColorForTooltipMarker(series, dataIndex) { - var style = series.getData().getItemVisual(dataIndex, 'style'); - var color = style[series.visualDrawType]; - return convertToColorString(color); - } - function getPaddingFromTooltipModel(model, renderMode) { - var padding = model.get('padding'); - return padding != null ? padding // We give slightly different to look pretty. - : renderMode === 'richText' ? [8, 10] : 10; - } - /** - * The major feature is generate styles for `renderMode: 'richText'`. - * But it also serves `renderMode: 'html'` to provide - * "renderMode-independent" API. - */ - - var TooltipMarkupStyleCreator = - /** @class */ - function () { - function TooltipMarkupStyleCreator() { - this.richTextStyles = {}; // Notice that "generate a style name" usually happens repeatedly when mouse is moving and - // a tooltip is displayed. So we put the `_nextStyleNameId` as a member of each creator - // rather than static shared by all creators (which will cause it increase to fast). - - this._nextStyleNameId = getRandomIdBase(); - } - - TooltipMarkupStyleCreator.prototype._generateStyleName = function () { - return '__EC_aUTo_' + this._nextStyleNameId++; - }; - - TooltipMarkupStyleCreator.prototype.makeTooltipMarker = function (markerType, colorStr, renderMode) { - var markerId = renderMode === 'richText' ? this._generateStyleName() : null; - var marker = getTooltipMarker({ - color: colorStr, - type: markerType, - renderMode: renderMode, - markerId: markerId - }); - - if (isString(marker)) { - return marker; - } else { - if ("development" !== 'production') { - assert(markerId); - } - - this.richTextStyles[markerId] = marker.style; - return marker.content; - } - }; - /** - * @usage - * ```ts - * const styledText = markupStyleCreator.wrapRichTextStyle([ - * // The styles will be auto merged. - * { - * fontSize: 12, - * color: 'blue' - * }, - * { - * padding: 20 - * } - * ]); - * ``` - */ - - - TooltipMarkupStyleCreator.prototype.wrapRichTextStyle = function (text, styles) { - var finalStl = {}; - - if (isArray(styles)) { - each(styles, function (stl) { - return extend(finalStl, stl); - }); - } else { - extend(finalStl, styles); - } - - var styleName = this._generateStyleName(); - - this.richTextStyles[styleName] = finalStl; - return "{" + styleName + "|" + text + "}"; - }; - - return TooltipMarkupStyleCreator; - }(); - - function defaultSeriesFormatTooltip(opt) { - var series = opt.series; - var dataIndex = opt.dataIndex; - var multipleSeries = opt.multipleSeries; - var data = series.getData(); - var tooltipDims = data.mapDimensionsAll('defaultedTooltip'); - var tooltipDimLen = tooltipDims.length; - var value = series.getRawValue(dataIndex); - var isValueArr = isArray(value); - var markerColor = retrieveVisualColorForTooltipMarker(series, dataIndex); // Complicated rule for pretty tooltip. - - var inlineValue; - var inlineValueType; - var subBlocks; - var sortParam; - - if (tooltipDimLen > 1 || isValueArr && !tooltipDimLen) { - var formatArrResult = formatTooltipArrayValue(value, series, dataIndex, tooltipDims, markerColor); - inlineValue = formatArrResult.inlineValues; - inlineValueType = formatArrResult.inlineValueTypes; - subBlocks = formatArrResult.blocks; // Only support tooltip sort by the first inline value. It's enough in most cases. - - sortParam = formatArrResult.inlineValues[0]; - } else if (tooltipDimLen) { - var dimInfo = data.getDimensionInfo(tooltipDims[0]); - sortParam = inlineValue = retrieveRawValue(data, dataIndex, tooltipDims[0]); - inlineValueType = dimInfo.type; - } else { - sortParam = inlineValue = isValueArr ? value[0] : value; - } // Do not show generated series name. It might not be readable. - - - var seriesNameSpecified = isNameSpecified(series); - var seriesName = seriesNameSpecified && series.name || ''; - var itemName = data.getName(dataIndex); - var inlineName = multipleSeries ? seriesName : itemName; - return createTooltipMarkup('section', { - header: seriesName, - // When series name is not specified, do not show a header line with only '-'. - // This case always happens in tooltip.trigger: 'item'. - noHeader: multipleSeries || !seriesNameSpecified, - sortParam: sortParam, - blocks: [createTooltipMarkup('nameValue', { - markerType: 'item', - markerColor: markerColor, - // Do not mix display seriesName and itemName in one tooltip, - // which might confuses users. - name: inlineName, - // name dimension might be auto assigned, where the name might - // be not readable. So we check trim here. - noName: !trim(inlineName), - value: inlineValue, - valueType: inlineValueType - })].concat(subBlocks || []) - }); - } - - function formatTooltipArrayValue(value, series, dataIndex, tooltipDims, colorStr) { - // check: category-no-encode-has-axis-data in dataset.html - var data = series.getData(); - var isValueMultipleLine = reduce(value, function (isValueMultipleLine, val, idx) { - var dimItem = data.getDimensionInfo(idx); - return isValueMultipleLine = isValueMultipleLine || dimItem && dimItem.tooltip !== false && dimItem.displayName != null; - }, false); - var inlineValues = []; - var inlineValueTypes = []; - var blocks = []; - tooltipDims.length ? each(tooltipDims, function (dim) { - setEachItem(retrieveRawValue(data, dataIndex, dim), dim); - }) // By default, all dims is used on tooltip. - : each(value, setEachItem); - - function setEachItem(val, dim) { - var dimInfo = data.getDimensionInfo(dim); // If `dimInfo.tooltip` is not set, show tooltip. - - if (!dimInfo || dimInfo.otherDims.tooltip === false) { - return; - } - - if (isValueMultipleLine) { - blocks.push(createTooltipMarkup('nameValue', { - markerType: 'subItem', - markerColor: colorStr, - name: dimInfo.displayName, - value: val, - valueType: dimInfo.type - })); - } else { - inlineValues.push(val); - inlineValueTypes.push(dimInfo.type); - } - } - - return { - inlineValues: inlineValues, - inlineValueTypes: inlineValueTypes, - blocks: blocks - }; - } - - var inner$1 = makeInner(); - - function getSelectionKey(data, dataIndex) { - return data.getName(dataIndex) || data.getId(dataIndex); - } - - var SERIES_UNIVERSAL_TRANSITION_PROP = '__universalTransitionEnabled'; - - var SeriesModel = - /** @class */ - function (_super) { - __extends(SeriesModel, _super); - - function SeriesModel() { - // [Caution]: Because this class or desecendants can be used as `XXX.extend(subProto)`, - // the class members must not be initialized in constructor or declaration place. - // Otherwise there is bad case: - // class A {xxx = 1;} - // enableClassExtend(A); - // class B extends A {} - // var C = B.extend({xxx: 5}); - // var c = new C(); - // console.log(c.xxx); // expect 5 but always 1. - var _this = _super !== null && _super.apply(this, arguments) || this; // --------------------------------------- - // Props about data selection - // --------------------------------------- - - - _this._selectedDataIndicesMap = {}; - return _this; - } - - SeriesModel.prototype.init = function (option, parentModel, ecModel) { - this.seriesIndex = this.componentIndex; - this.dataTask = createTask({ - count: dataTaskCount, - reset: dataTaskReset - }); - this.dataTask.context = { - model: this - }; - this.mergeDefaultAndTheme(option, ecModel); - var sourceManager = inner$1(this).sourceManager = new SourceManager(this); - sourceManager.prepareSource(); - var data = this.getInitialData(option, ecModel); - wrapData(data, this); - this.dataTask.context.data = data; - - if ("development" !== 'production') { - assert(data, 'getInitialData returned invalid data.'); - } - - inner$1(this).dataBeforeProcessed = data; // If we reverse the order (make data firstly, and then make - // dataBeforeProcessed by cloneShallow), cloneShallow will - // cause data.graph.data !== data when using - // module:echarts/data/Graph or module:echarts/data/Tree. - // See module:echarts/data/helper/linkSeriesData - // Theoretically, it is unreasonable to call `seriesModel.getData()` in the model - // init or merge stage, because the data can be restored. So we do not `restoreData` - // and `setData` here, which forbids calling `seriesModel.getData()` in this stage. - // Call `seriesModel.getRawData()` instead. - // this.restoreData(); - - autoSeriesName(this); - - this._initSelectedMapFromData(data); - }; - /** - * Util for merge default and theme to option - */ - - - SeriesModel.prototype.mergeDefaultAndTheme = function (option, ecModel) { - var layoutMode = fetchLayoutMode(this); - var inputPositionParams = layoutMode ? getLayoutParams(option) : {}; // Backward compat: using subType on theme. - // But if name duplicate between series subType - // (for example: parallel) add component mainType, - // add suffix 'Series'. - - var themeSubType = this.subType; - - if (ComponentModel.hasClass(themeSubType)) { - themeSubType += 'Series'; - } - - merge(option, ecModel.getTheme().get(this.subType)); - merge(option, this.getDefaultOption()); // Default label emphasis `show` - - defaultEmphasis(option, 'label', ['show']); - this.fillDataTextStyle(option.data); - - if (layoutMode) { - mergeLayoutParam(option, inputPositionParams, layoutMode); - } - }; - - SeriesModel.prototype.mergeOption = function (newSeriesOption, ecModel) { - // this.settingTask.dirty(); - newSeriesOption = merge(this.option, newSeriesOption, true); - this.fillDataTextStyle(newSeriesOption.data); - var layoutMode = fetchLayoutMode(this); - - if (layoutMode) { - mergeLayoutParam(this.option, newSeriesOption, layoutMode); - } - - var sourceManager = inner$1(this).sourceManager; - sourceManager.dirty(); - sourceManager.prepareSource(); - var data = this.getInitialData(newSeriesOption, ecModel); - wrapData(data, this); - this.dataTask.dirty(); - this.dataTask.context.data = data; - inner$1(this).dataBeforeProcessed = data; - autoSeriesName(this); - - this._initSelectedMapFromData(data); - }; - - SeriesModel.prototype.fillDataTextStyle = function (data) { - // Default data label emphasis `show` - // FIXME Tree structure data ? - // FIXME Performance ? - if (data && !isTypedArray(data)) { - var props = ['show']; - - for (var i = 0; i < data.length; i++) { - if (data[i] && data[i].label) { - defaultEmphasis(data[i], 'label', props); - } - } - } - }; - /** - * Init a data structure from data related option in series - * Must be overridden. - */ - - - SeriesModel.prototype.getInitialData = function (option, ecModel) { - return; - }; - /** - * Append data to list - */ - - - SeriesModel.prototype.appendData = function (params) { - // FIXME ??? - // (1) If data from dataset, forbidden append. - // (2) support append data of dataset. - var data = this.getRawData(); - data.appendData(params.data); - }; - /** - * Consider some method like `filter`, `map` need make new data, - * We should make sure that `seriesModel.getData()` get correct - * data in the stream procedure. So we fetch data from upstream - * each time `task.perform` called. - */ - - - SeriesModel.prototype.getData = function (dataType) { - var task = getCurrentTask(this); - - if (task) { - var data = task.context.data; - return dataType == null ? data : data.getLinkedData(dataType); - } else { - // When series is not alive (that may happen when click toolbox - // restore or setOption with not merge mode), series data may - // be still need to judge animation or something when graphic - // elements want to know whether fade out. - return inner$1(this).data; - } - }; - - SeriesModel.prototype.getAllData = function () { - var mainData = this.getData(); - return mainData && mainData.getLinkedDataAll ? mainData.getLinkedDataAll() : [{ - data: mainData - }]; - }; - - SeriesModel.prototype.setData = function (data) { - var task = getCurrentTask(this); - - if (task) { - var context = task.context; // Consider case: filter, data sample. - // FIXME:TS never used, so comment it - // if (context.data !== data && task.modifyOutputEnd) { - // task.setOutputEnd(data.count()); - // } - - context.outputData = data; // Caution: setData should update context.data, - // Because getData may be called multiply in a - // single stage and expect to get the data just - // set. (For example, AxisProxy, x y both call - // getData and setDate sequentially). - // So the context.data should be fetched from - // upstream each time when a stage starts to be - // performed. - - if (task !== this.dataTask) { - context.data = data; - } - } - - inner$1(this).data = data; - }; - - SeriesModel.prototype.getEncode = function () { - var encode = this.get('encode', true); - - if (encode) { - return createHashMap(encode); - } - }; - - SeriesModel.prototype.getSourceManager = function () { - return inner$1(this).sourceManager; - }; - - SeriesModel.prototype.getSource = function () { - return this.getSourceManager().getSource(); - }; - /** - * Get data before processed - */ - - - SeriesModel.prototype.getRawData = function () { - return inner$1(this).dataBeforeProcessed; - }; - - SeriesModel.prototype.getColorBy = function () { - var colorBy = this.get('colorBy'); - return colorBy || 'series'; - }; - - SeriesModel.prototype.isColorBySeries = function () { - return this.getColorBy() === 'series'; - }; - /** - * Get base axis if has coordinate system and has axis. - * By default use coordSys.getBaseAxis(); - * Can be overridden for some chart. - * @return {type} description - */ - - - SeriesModel.prototype.getBaseAxis = function () { - var coordSys = this.coordinateSystem; // @ts-ignore - - return coordSys && coordSys.getBaseAxis && coordSys.getBaseAxis(); - }; - /** - * Default tooltip formatter - * - * @param dataIndex - * @param multipleSeries - * @param dataType - * @param renderMode valid values: 'html'(by default) and 'richText'. - * 'html' is used for rendering tooltip in extra DOM form, and the result - * string is used as DOM HTML content. - * 'richText' is used for rendering tooltip in rich text form, for those where - * DOM operation is not supported. - * @return formatted tooltip with `html` and `markers` - * Notice: The override method can also return string - */ - - - SeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - return defaultSeriesFormatTooltip({ - series: this, - dataIndex: dataIndex, - multipleSeries: multipleSeries - }); - }; - - SeriesModel.prototype.isAnimationEnabled = function () { - var ecModel = this.ecModel; // Disable animation if using echarts in node but not give ssr flag. - // In ssr mode, renderToString will generate svg with css animation. - - if (env.node && !(ecModel && ecModel.ssr)) { - return false; - } - - var animationEnabled = this.getShallow('animation'); - - if (animationEnabled) { - if (this.getData().count() > this.getShallow('animationThreshold')) { - animationEnabled = false; - } - } - - return !!animationEnabled; - }; - - SeriesModel.prototype.restoreData = function () { - this.dataTask.dirty(); - }; - - SeriesModel.prototype.getColorFromPalette = function (name, scope, requestColorNum) { - var ecModel = this.ecModel; // PENDING - - var color = PaletteMixin.prototype.getColorFromPalette.call(this, name, scope, requestColorNum); - - if (!color) { - color = ecModel.getColorFromPalette(name, scope, requestColorNum); - } - - return color; - }; - /** - * Use `data.mapDimensionsAll(coordDim)` instead. - * @deprecated - */ - - - SeriesModel.prototype.coordDimToDataDim = function (coordDim) { - return this.getRawData().mapDimensionsAll(coordDim); - }; - /** - * Get progressive rendering count each step - */ - - - SeriesModel.prototype.getProgressive = function () { - return this.get('progressive'); - }; - /** - * Get progressive rendering count each step - */ - - - SeriesModel.prototype.getProgressiveThreshold = function () { - return this.get('progressiveThreshold'); - }; // PENGING If selectedMode is null ? - - - SeriesModel.prototype.select = function (innerDataIndices, dataType) { - this._innerSelect(this.getData(dataType), innerDataIndices); - }; - - SeriesModel.prototype.unselect = function (innerDataIndices, dataType) { - var selectedMap = this.option.selectedMap; - - if (!selectedMap) { - return; - } - - var selectedMode = this.option.selectedMode; - var data = this.getData(dataType); - - if (selectedMode === 'series' || selectedMap === 'all') { - this.option.selectedMap = {}; - this._selectedDataIndicesMap = {}; - return; - } - - for (var i = 0; i < innerDataIndices.length; i++) { - var dataIndex = innerDataIndices[i]; - var nameOrId = getSelectionKey(data, dataIndex); - selectedMap[nameOrId] = false; - this._selectedDataIndicesMap[nameOrId] = -1; - } - }; - - SeriesModel.prototype.toggleSelect = function (innerDataIndices, dataType) { - var tmpArr = []; - - for (var i = 0; i < innerDataIndices.length; i++) { - tmpArr[0] = innerDataIndices[i]; - this.isSelected(innerDataIndices[i], dataType) ? this.unselect(tmpArr, dataType) : this.select(tmpArr, dataType); - } - }; - - SeriesModel.prototype.getSelectedDataIndices = function () { - if (this.option.selectedMap === 'all') { - return [].slice.call(this.getData().getIndices()); - } - - var selectedDataIndicesMap = this._selectedDataIndicesMap; - var nameOrIds = keys(selectedDataIndicesMap); - var dataIndices = []; - - for (var i = 0; i < nameOrIds.length; i++) { - var dataIndex = selectedDataIndicesMap[nameOrIds[i]]; - - if (dataIndex >= 0) { - dataIndices.push(dataIndex); - } - } - - return dataIndices; - }; - - SeriesModel.prototype.isSelected = function (dataIndex, dataType) { - var selectedMap = this.option.selectedMap; - - if (!selectedMap) { - return false; - } - - var data = this.getData(dataType); - return (selectedMap === 'all' || selectedMap[getSelectionKey(data, dataIndex)]) && !data.getItemModel(dataIndex).get(['select', 'disabled']); - }; - - SeriesModel.prototype.isUniversalTransitionEnabled = function () { - if (this[SERIES_UNIVERSAL_TRANSITION_PROP]) { - return true; - } - - var universalTransitionOpt = this.option.universalTransition; // Quick reject - - if (!universalTransitionOpt) { - return false; - } - - if (universalTransitionOpt === true) { - return true; - } // Can be simply 'universalTransition: true' - - - return universalTransitionOpt && universalTransitionOpt.enabled; - }; - - SeriesModel.prototype._innerSelect = function (data, innerDataIndices) { - var _a, _b; - - var option = this.option; - var selectedMode = option.selectedMode; - var len = innerDataIndices.length; - - if (!selectedMode || !len) { - return; - } - - if (selectedMode === 'series') { - option.selectedMap = 'all'; - } else if (selectedMode === 'multiple') { - if (!isObject(option.selectedMap)) { - option.selectedMap = {}; - } - - var selectedMap = option.selectedMap; - - for (var i = 0; i < len; i++) { - var dataIndex = innerDataIndices[i]; // TODO different types of data share same object. - - var nameOrId = getSelectionKey(data, dataIndex); - selectedMap[nameOrId] = true; - this._selectedDataIndicesMap[nameOrId] = data.getRawIndex(dataIndex); - } - } else if (selectedMode === 'single' || selectedMode === true) { - var lastDataIndex = innerDataIndices[len - 1]; - var nameOrId = getSelectionKey(data, lastDataIndex); - option.selectedMap = (_a = {}, _a[nameOrId] = true, _a); - this._selectedDataIndicesMap = (_b = {}, _b[nameOrId] = data.getRawIndex(lastDataIndex), _b); - } - }; - - SeriesModel.prototype._initSelectedMapFromData = function (data) { - // Ignore select info in data if selectedMap exists. - // NOTE It's only for legacy usage. edge data is not supported. - if (this.option.selectedMap) { - return; - } - - var dataIndices = []; - - if (data.hasItemOption) { - data.each(function (idx) { - var rawItem = data.getRawDataItem(idx); - - if (rawItem && rawItem.selected) { - dataIndices.push(idx); - } - }); - } - - if (dataIndices.length > 0) { - this._innerSelect(data, dataIndices); - } - }; // /** - // * @see {module:echarts/stream/Scheduler} - // */ - // abstract pipeTask: null - - - SeriesModel.registerClass = function (clz) { - return ComponentModel.registerClass(clz); - }; - - SeriesModel.protoInitialize = function () { - var proto = SeriesModel.prototype; - proto.type = 'series.__base__'; - proto.seriesIndex = 0; - proto.ignoreStyleOnData = false; - proto.hasSymbolVisual = false; - proto.defaultSymbol = 'circle'; // Make sure the values can be accessed! - - proto.visualStyleAccessPath = 'itemStyle'; - proto.visualDrawType = 'fill'; - }(); - - return SeriesModel; - }(ComponentModel); - - mixin(SeriesModel, DataFormatMixin); - mixin(SeriesModel, PaletteMixin); - mountExtend(SeriesModel, ComponentModel); - /** - * MUST be called after `prepareSource` called - * Here we need to make auto series, especially for auto legend. But we - * do not modify series.name in option to avoid side effects. - */ - - function autoSeriesName(seriesModel) { - // User specified name has higher priority, otherwise it may cause - // series can not be queried unexpectedly. - var name = seriesModel.name; - - if (!isNameSpecified(seriesModel)) { - seriesModel.name = getSeriesAutoName(seriesModel) || name; - } - } - - function getSeriesAutoName(seriesModel) { - var data = seriesModel.getRawData(); - var dataDims = data.mapDimensionsAll('seriesName'); - var nameArr = []; - each(dataDims, function (dataDim) { - var dimInfo = data.getDimensionInfo(dataDim); - dimInfo.displayName && nameArr.push(dimInfo.displayName); - }); - return nameArr.join(' '); - } - - function dataTaskCount(context) { - return context.model.getRawData().count(); - } - - function dataTaskReset(context) { - var seriesModel = context.model; - seriesModel.setData(seriesModel.getRawData().cloneShallow()); - return dataTaskProgress; - } - - function dataTaskProgress(param, context) { - // Avoid repeat cloneShallow when data just created in reset. - if (context.outputData && param.end > context.outputData.count()) { - context.model.getRawData().cloneShallow(context.outputData); - } - } // TODO refactor - - - function wrapData(data, seriesModel) { - each(concatArray(data.CHANGABLE_METHODS, data.DOWNSAMPLE_METHODS), function (methodName) { - data.wrapMethod(methodName, curry(onDataChange, seriesModel)); - }); - } - - function onDataChange(seriesModel, newList) { - var task = getCurrentTask(seriesModel); - - if (task) { - // Consider case: filter, selectRange - task.setOutputEnd((newList || this).count()); - } - - return newList; - } - - function getCurrentTask(seriesModel) { - var scheduler = (seriesModel.ecModel || {}).scheduler; - var pipeline = scheduler && scheduler.getPipeline(seriesModel.uid); - - if (pipeline) { - // When pipline finished, the currrentTask keep the last - // task (renderTask). - var task = pipeline.currentTask; - - if (task) { - var agentStubMap = task.agentStubMap; - - if (agentStubMap) { - task = agentStubMap.get(seriesModel.uid); - } - } - - return task; - } - } - - var ComponentView = - /** @class */ - function () { - function ComponentView() { - this.group = new Group(); - this.uid = getUID('viewComponent'); - } - - ComponentView.prototype.init = function (ecModel, api) {}; - - ComponentView.prototype.render = function (model, ecModel, api, payload) {}; - - ComponentView.prototype.dispose = function (ecModel, api) {}; - - ComponentView.prototype.updateView = function (model, ecModel, api, payload) {// Do nothing; - }; - - ComponentView.prototype.updateLayout = function (model, ecModel, api, payload) {// Do nothing; - }; - - ComponentView.prototype.updateVisual = function (model, ecModel, api, payload) {// Do nothing; - }; - /** - * Hook for toggle blur target series. - * Can be used in marker for blur or leave blur the markers - */ - - - ComponentView.prototype.toggleBlurSeries = function (seriesModels, isBlur, ecModel) {// Do nothing; - }; - /** - * Traverse the new rendered elements. - * - * It will traverse the new added element in progressive rendering. - * And traverse all in normal rendering. - */ - - - ComponentView.prototype.eachRendered = function (cb) { - var group = this.group; - - if (group) { - group.traverse(cb); - } - }; - - return ComponentView; - }(); - enableClassExtend(ComponentView); - enableClassManagement(ComponentView); - - /** - * @return {string} If large mode changed, return string 'reset'; - */ - - function createRenderPlanner() { - var inner = makeInner(); - return function (seriesModel) { - var fields = inner(seriesModel); - var pipelineContext = seriesModel.pipelineContext; - var originalLarge = !!fields.large; - var originalProgressive = !!fields.progressiveRender; // FIXME: if the planner works on a filtered series, `pipelineContext` does not - // exists. See #11611 . Probably we need to modify this structure, see the comment - // on `performRawSeries` in `Schedular.js`. - - var large = fields.large = !!(pipelineContext && pipelineContext.large); - var progressive = fields.progressiveRender = !!(pipelineContext && pipelineContext.progressiveRender); - return !!(originalLarge !== large || originalProgressive !== progressive) && 'reset'; - }; - } - - var inner$2 = makeInner(); - var renderPlanner = createRenderPlanner(); - - var ChartView = - /** @class */ - function () { - function ChartView() { - this.group = new Group(); - this.uid = getUID('viewChart'); - this.renderTask = createTask({ - plan: renderTaskPlan, - reset: renderTaskReset - }); - this.renderTask.context = { - view: this - }; - } - - ChartView.prototype.init = function (ecModel, api) {}; - - ChartView.prototype.render = function (seriesModel, ecModel, api, payload) { - if ("development" !== 'production') { - throw new Error('render method must been implemented'); - } - }; - /** - * Highlight series or specified data item. - */ - - - ChartView.prototype.highlight = function (seriesModel, ecModel, api, payload) { - var data = seriesModel.getData(payload && payload.dataType); - - if (!data) { - if ("development" !== 'production') { - error("Unknown dataType " + payload.dataType); - } - - return; - } - - toggleHighlight(data, payload, 'emphasis'); - }; - /** - * Downplay series or specified data item. - */ - - - ChartView.prototype.downplay = function (seriesModel, ecModel, api, payload) { - var data = seriesModel.getData(payload && payload.dataType); - - if (!data) { - if ("development" !== 'production') { - error("Unknown dataType " + payload.dataType); - } - - return; - } - - toggleHighlight(data, payload, 'normal'); - }; - /** - * Remove self. - */ - - - ChartView.prototype.remove = function (ecModel, api) { - this.group.removeAll(); - }; - /** - * Dispose self. - */ - - - ChartView.prototype.dispose = function (ecModel, api) {}; - - ChartView.prototype.updateView = function (seriesModel, ecModel, api, payload) { - this.render(seriesModel, ecModel, api, payload); - }; // FIXME never used? - - - ChartView.prototype.updateLayout = function (seriesModel, ecModel, api, payload) { - this.render(seriesModel, ecModel, api, payload); - }; // FIXME never used? - - - ChartView.prototype.updateVisual = function (seriesModel, ecModel, api, payload) { - this.render(seriesModel, ecModel, api, payload); - }; - /** - * Traverse the new rendered elements. - * - * It will traverse the new added element in progressive rendering. - * And traverse all in normal rendering. - */ - - - ChartView.prototype.eachRendered = function (cb) { - traverseElements(this.group, cb); - }; - - ChartView.markUpdateMethod = function (payload, methodName) { - inner$2(payload).updateMethod = methodName; - }; - - ChartView.protoInitialize = function () { - var proto = ChartView.prototype; - proto.type = 'chart'; - }(); - - return ChartView; - }(); - /** - * Set state of single element - */ - - function elSetState(el, state, highlightDigit) { - if (el && isHighDownDispatcher(el)) { - (state === 'emphasis' ? enterEmphasis : leaveEmphasis)(el, highlightDigit); - } - } - - function toggleHighlight(data, payload, state) { - var dataIndex = queryDataIndex(data, payload); - var highlightDigit = payload && payload.highlightKey != null ? getHighlightDigit(payload.highlightKey) : null; - - if (dataIndex != null) { - each(normalizeToArray(dataIndex), function (dataIdx) { - elSetState(data.getItemGraphicEl(dataIdx), state, highlightDigit); - }); - } else { - data.eachItemGraphicEl(function (el) { - elSetState(el, state, highlightDigit); - }); - } - } - - enableClassExtend(ChartView, ['dispose']); - enableClassManagement(ChartView); - - function renderTaskPlan(context) { - return renderPlanner(context.model); - } - - function renderTaskReset(context) { - var seriesModel = context.model; - var ecModel = context.ecModel; - var api = context.api; - var payload = context.payload; // FIXME: remove updateView updateVisual - - var progressiveRender = seriesModel.pipelineContext.progressiveRender; - var view = context.view; - var updateMethod = payload && inner$2(payload).updateMethod; - var methodName = progressiveRender ? 'incrementalPrepareRender' : updateMethod && view[updateMethod] ? updateMethod // `appendData` is also supported when data amount - // is less than progressive threshold. - : 'render'; - - if (methodName !== 'render') { - view[methodName](seriesModel, ecModel, api, payload); - } - - return progressMethodMap[methodName]; - } - - var progressMethodMap = { - incrementalPrepareRender: { - progress: function (params, context) { - context.view.incrementalRender(params, context.model, context.ecModel, context.api, context.payload); - } - }, - render: { - // Put view.render in `progress` to support appendData. But in this case - // view.render should not be called in reset, otherwise it will be called - // twise. Use `forceFirstProgress` to make sure that view.render is called - // in any cases. - forceFirstProgress: true, - progress: function (params, context) { - context.view.render(context.model, context.ecModel, context.api, context.payload); - } - } - }; - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var ORIGIN_METHOD = '\0__throttleOriginMethod'; - var RATE = '\0__throttleRate'; - var THROTTLE_TYPE = '\0__throttleType'; - /** - * @public - * @param {(Function)} fn - * @param {number} [delay=0] Unit: ms. - * @param {boolean} [debounce=false] - * true: If call interval less than `delay`, only the last call works. - * false: If call interval less than `delay, call works on fixed rate. - * @return {(Function)} throttled fn. - */ - - function throttle(fn, delay, debounce) { - var currCall; - var lastCall = 0; - var lastExec = 0; - var timer = null; - var diff; - var scope; - var args; - var debounceNextCall; - delay = delay || 0; - - function exec() { - lastExec = new Date().getTime(); - timer = null; - fn.apply(scope, args || []); - } - - var cb = function () { - var cbArgs = []; - - for (var _i = 0; _i < arguments.length; _i++) { - cbArgs[_i] = arguments[_i]; - } - - currCall = new Date().getTime(); - scope = this; - args = cbArgs; - var thisDelay = debounceNextCall || delay; - var thisDebounce = debounceNextCall || debounce; - debounceNextCall = null; - diff = currCall - (thisDebounce ? lastCall : lastExec) - thisDelay; - clearTimeout(timer); // Here we should make sure that: the `exec` SHOULD NOT be called later - // than a new call of `cb`, that is, preserving the command order. Consider - // calculating "scale rate" when roaming as an example. When a call of `cb` - // happens, either the `exec` is called dierectly, or the call is delayed. - // But the delayed call should never be later than next call of `cb`. Under - // this assurance, we can simply update view state each time `dispatchAction` - // triggered by user roaming, but not need to add extra code to avoid the - // state being "rolled-back". - - if (thisDebounce) { - timer = setTimeout(exec, thisDelay); - } else { - if (diff >= 0) { - exec(); - } else { - timer = setTimeout(exec, -diff); - } - } - - lastCall = currCall; - }; - /** - * Clear throttle. - * @public - */ - - - cb.clear = function () { - if (timer) { - clearTimeout(timer); - timer = null; - } - }; - /** - * Enable debounce once. - */ - - - cb.debounceNextCall = function (debounceDelay) { - debounceNextCall = debounceDelay; - }; - - return cb; - } - /** - * Create throttle method or update throttle rate. - * - * @example - * ComponentView.prototype.render = function () { - * ... - * throttle.createOrUpdate( - * this, - * '_dispatchAction', - * this.model.get('throttle'), - * 'fixRate' - * ); - * }; - * ComponentView.prototype.remove = function () { - * throttle.clear(this, '_dispatchAction'); - * }; - * ComponentView.prototype.dispose = function () { - * throttle.clear(this, '_dispatchAction'); - * }; - * - */ - - function createOrUpdate(obj, fnAttr, rate, throttleType) { - var fn = obj[fnAttr]; - - if (!fn) { - return; - } - - var originFn = fn[ORIGIN_METHOD] || fn; - var lastThrottleType = fn[THROTTLE_TYPE]; - var lastRate = fn[RATE]; - - if (lastRate !== rate || lastThrottleType !== throttleType) { - if (rate == null || !throttleType) { - return obj[fnAttr] = originFn; - } - - fn = obj[fnAttr] = throttle(originFn, rate, throttleType === 'debounce'); - fn[ORIGIN_METHOD] = originFn; - fn[THROTTLE_TYPE] = throttleType; - fn[RATE] = rate; - } - - return fn; - } - /** - * Clear throttle. Example see throttle.createOrUpdate. - */ - - function clear(obj, fnAttr) { - var fn = obj[fnAttr]; - - if (fn && fn[ORIGIN_METHOD]) { - // Clear throttle - fn.clear && fn.clear(); - obj[fnAttr] = fn[ORIGIN_METHOD]; - } - } - - var inner$3 = makeInner(); - var defaultStyleMappers = { - itemStyle: makeStyleMapper(ITEM_STYLE_KEY_MAP, true), - lineStyle: makeStyleMapper(LINE_STYLE_KEY_MAP, true) - }; - var defaultColorKey = { - lineStyle: 'stroke', - itemStyle: 'fill' - }; - - function getStyleMapper(seriesModel, stylePath) { - var styleMapper = seriesModel.visualStyleMapper || defaultStyleMappers[stylePath]; - - if (!styleMapper) { - console.warn("Unknown style type '" + stylePath + "'."); - return defaultStyleMappers.itemStyle; - } - - return styleMapper; - } - - function getDefaultColorKey(seriesModel, stylePath) { - // return defaultColorKey[stylePath] || - var colorKey = seriesModel.visualDrawType || defaultColorKey[stylePath]; - - if (!colorKey) { - console.warn("Unknown style type '" + stylePath + "'."); - return 'fill'; - } - - return colorKey; - } - - var seriesStyleTask = { - createOnAllSeries: true, - performRawSeries: true, - reset: function (seriesModel, ecModel) { - var data = seriesModel.getData(); - var stylePath = seriesModel.visualStyleAccessPath || 'itemStyle'; // Set in itemStyle - - var styleModel = seriesModel.getModel(stylePath); - var getStyle = getStyleMapper(seriesModel, stylePath); - var globalStyle = getStyle(styleModel); - var decalOption = styleModel.getShallow('decal'); - - if (decalOption) { - data.setVisual('decal', decalOption); - decalOption.dirty = true; - } // TODO - - - var colorKey = getDefaultColorKey(seriesModel, stylePath); - var color = globalStyle[colorKey]; // TODO style callback - - var colorCallback = isFunction(color) ? color : null; - var hasAutoColor = globalStyle.fill === 'auto' || globalStyle.stroke === 'auto'; // Get from color palette by default. - - if (!globalStyle[colorKey] || colorCallback || hasAutoColor) { - // Note: If some series has color specified (e.g., by itemStyle.color), we DO NOT - // make it effect palette. Because some scenarios users need to make some series - // transparent or as background, which should better not effect the palette. - var colorPalette = seriesModel.getColorFromPalette( // TODO series count changed. - seriesModel.name, null, ecModel.getSeriesCount()); - - if (!globalStyle[colorKey]) { - globalStyle[colorKey] = colorPalette; - data.setVisual('colorFromPalette', true); - } - - globalStyle.fill = globalStyle.fill === 'auto' || isFunction(globalStyle.fill) ? colorPalette : globalStyle.fill; - globalStyle.stroke = globalStyle.stroke === 'auto' || isFunction(globalStyle.stroke) ? colorPalette : globalStyle.stroke; - } - - data.setVisual('style', globalStyle); - data.setVisual('drawType', colorKey); // Only visible series has each data be visual encoded - - if (!ecModel.isSeriesFiltered(seriesModel) && colorCallback) { - data.setVisual('colorFromPalette', false); - return { - dataEach: function (data, idx) { - var dataParams = seriesModel.getDataParams(idx); - var itemStyle = extend({}, globalStyle); - itemStyle[colorKey] = colorCallback(dataParams); - data.setItemVisual(idx, 'style', itemStyle); - } - }; - } - } - }; - var sharedModel = new Model(); - var dataStyleTask = { - createOnAllSeries: true, - performRawSeries: true, - reset: function (seriesModel, ecModel) { - if (seriesModel.ignoreStyleOnData || ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - var data = seriesModel.getData(); - var stylePath = seriesModel.visualStyleAccessPath || 'itemStyle'; // Set in itemStyle - - var getStyle = getStyleMapper(seriesModel, stylePath); - var colorKey = data.getVisual('drawType'); - return { - dataEach: data.hasItemOption ? function (data, idx) { - // Not use getItemModel for performance considuration - var rawItem = data.getRawDataItem(idx); - - if (rawItem && rawItem[stylePath]) { - sharedModel.option = rawItem[stylePath]; - var style = getStyle(sharedModel); - var existsStyle = data.ensureUniqueItemVisual(idx, 'style'); - extend(existsStyle, style); - - if (sharedModel.option.decal) { - data.setItemVisual(idx, 'decal', sharedModel.option.decal); - sharedModel.option.decal.dirty = true; - } - - if (colorKey in style) { - data.setItemVisual(idx, 'colorFromPalette', false); - } - } - } : null - }; - } - }; // Pick color from palette for the data which has not been set with color yet. - // Note: do not support stream rendering. No such cases yet. - - var dataColorPaletteTask = { - performRawSeries: true, - overallReset: function (ecModel) { - // Each type of series uses one scope. - // Pie and funnel are using different scopes. - var paletteScopeGroupByType = createHashMap(); - ecModel.eachSeries(function (seriesModel) { - var colorBy = seriesModel.getColorBy(); - - if (seriesModel.isColorBySeries()) { - return; - } - - var key = seriesModel.type + '-' + colorBy; - var colorScope = paletteScopeGroupByType.get(key); - - if (!colorScope) { - colorScope = {}; - paletteScopeGroupByType.set(key, colorScope); - } - - inner$3(seriesModel).scope = colorScope; - }); - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.isColorBySeries() || ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - var dataAll = seriesModel.getRawData(); - var idxMap = {}; - var data = seriesModel.getData(); - var colorScope = inner$3(seriesModel).scope; - var stylePath = seriesModel.visualStyleAccessPath || 'itemStyle'; - var colorKey = getDefaultColorKey(seriesModel, stylePath); - data.each(function (idx) { - var rawIdx = data.getRawIndex(idx); - idxMap[rawIdx] = idx; - }); // Iterate on data before filtered. To make sure color from palette can be - // Consistent when toggling legend. - - dataAll.each(function (rawIdx) { - var idx = idxMap[rawIdx]; - var fromPalette = data.getItemVisual(idx, 'colorFromPalette'); // Get color from palette for each data only when the color is inherited from series color, which is - // also picked from color palette. So following situation is not in the case: - // 1. series.itemStyle.color is set - // 2. color is encoded by visualMap - - if (fromPalette) { - var itemStyle = data.ensureUniqueItemVisual(idx, 'style'); - var name_1 = dataAll.getName(rawIdx) || rawIdx + ''; - var dataCount = dataAll.count(); - itemStyle[colorKey] = seriesModel.getColorFromPalette(name_1, colorScope, dataCount); - } - }); - }); - } - }; - - var PI$3 = Math.PI; - /** - * @param {module:echarts/ExtensionAPI} api - * @param {Object} [opts] - * @param {string} [opts.text] - * @param {string} [opts.color] - * @param {string} [opts.textColor] - * @return {module:zrender/Element} - */ - - function defaultLoading(api, opts) { - opts = opts || {}; - defaults(opts, { - text: 'loading', - textColor: '#000', - fontSize: 12, - fontWeight: 'normal', - fontStyle: 'normal', - fontFamily: 'sans-serif', - maskColor: 'rgba(255, 255, 255, 0.8)', - showSpinner: true, - color: '#5470c6', - spinnerRadius: 10, - lineWidth: 5, - zlevel: 0 - }); - var group = new Group(); - var mask = new Rect({ - style: { - fill: opts.maskColor - }, - zlevel: opts.zlevel, - z: 10000 - }); - group.add(mask); - var textContent = new ZRText({ - style: { - text: opts.text, - fill: opts.textColor, - fontSize: opts.fontSize, - fontWeight: opts.fontWeight, - fontStyle: opts.fontStyle, - fontFamily: opts.fontFamily - }, - zlevel: opts.zlevel, - z: 10001 - }); - var labelRect = new Rect({ - style: { - fill: 'none' - }, - textContent: textContent, - textConfig: { - position: 'right', - distance: 10 - }, - zlevel: opts.zlevel, - z: 10001 - }); - group.add(labelRect); - var arc; - - if (opts.showSpinner) { - arc = new Arc({ - shape: { - startAngle: -PI$3 / 2, - endAngle: -PI$3 / 2 + 0.1, - r: opts.spinnerRadius - }, - style: { - stroke: opts.color, - lineCap: 'round', - lineWidth: opts.lineWidth - }, - zlevel: opts.zlevel, - z: 10001 - }); - arc.animateShape(true).when(1000, { - endAngle: PI$3 * 3 / 2 - }).start('circularInOut'); - arc.animateShape(true).when(1000, { - startAngle: PI$3 * 3 / 2 - }).delay(300).start('circularInOut'); - group.add(arc); - } // Inject resize - - - group.resize = function () { - var textWidth = textContent.getBoundingRect().width; - var r = opts.showSpinner ? opts.spinnerRadius : 0; // cx = (containerWidth - arcDiameter - textDistance - textWidth) / 2 - // textDistance needs to be calculated when both animation and text exist - - var cx = (api.getWidth() - r * 2 - (opts.showSpinner && textWidth ? 10 : 0) - textWidth) / 2 - (opts.showSpinner && textWidth ? 0 : 5 + textWidth / 2) // only show the text - + (opts.showSpinner ? 0 : textWidth / 2) // only show the spinner - + (textWidth ? 0 : r); - var cy = api.getHeight() / 2; - opts.showSpinner && arc.setShape({ - cx: cx, - cy: cy - }); - labelRect.setShape({ - x: cx - r, - y: cy - r, - width: r * 2, - height: r * 2 - }); - mask.setShape({ - x: 0, - y: 0, - width: api.getWidth(), - height: api.getHeight() - }); - }; - - group.resize(); - return group; - } - - var Scheduler = - /** @class */ - function () { - function Scheduler(ecInstance, api, dataProcessorHandlers, visualHandlers) { - // key: handlerUID - this._stageTaskMap = createHashMap(); - this.ecInstance = ecInstance; - this.api = api; // Fix current processors in case that in some rear cases that - // processors might be registered after echarts instance created. - // Register processors incrementally for a echarts instance is - // not supported by this stream architecture. - - dataProcessorHandlers = this._dataProcessorHandlers = dataProcessorHandlers.slice(); - visualHandlers = this._visualHandlers = visualHandlers.slice(); - this._allHandlers = dataProcessorHandlers.concat(visualHandlers); - } - - Scheduler.prototype.restoreData = function (ecModel, payload) { - // TODO: Only restore needed series and components, but not all components. - // Currently `restoreData` of all of the series and component will be called. - // But some independent components like `title`, `legend`, `graphic`, `toolbox`, - // `tooltip`, `axisPointer`, etc, do not need series refresh when `setOption`, - // and some components like coordinate system, axes, dataZoom, visualMap only - // need their target series refresh. - // (1) If we are implementing this feature some day, we should consider these cases: - // if a data processor depends on a component (e.g., dataZoomProcessor depends - // on the settings of `dataZoom`), it should be re-performed if the component - // is modified by `setOption`. - // (2) If a processor depends on sevral series, speicified by its `getTargetSeries`, - // it should be re-performed when the result array of `getTargetSeries` changed. - // We use `dependencies` to cover these issues. - // (3) How to update target series when coordinate system related components modified. - // TODO: simply the dirty mechanism? Check whether only the case here can set tasks dirty, - // and this case all of the tasks will be set as dirty. - ecModel.restoreData(payload); // Theoretically an overall task not only depends on each of its target series, but also - // depends on all of the series. - // The overall task is not in pipeline, and `ecModel.restoreData` only set pipeline tasks - // dirty. If `getTargetSeries` of an overall task returns nothing, we should also ensure - // that the overall task is set as dirty and to be performed, otherwise it probably cause - // state chaos. So we have to set dirty of all of the overall tasks manually, otherwise it - // probably cause state chaos (consider `dataZoomProcessor`). - - this._stageTaskMap.each(function (taskRecord) { - var overallTask = taskRecord.overallTask; - overallTask && overallTask.dirty(); - }); - }; // If seriesModel provided, incremental threshold is check by series data. - - - Scheduler.prototype.getPerformArgs = function (task, isBlock) { - // For overall task - if (!task.__pipeline) { - return; - } - - var pipeline = this._pipelineMap.get(task.__pipeline.id); - - var pCtx = pipeline.context; - var incremental = !isBlock && pipeline.progressiveEnabled && (!pCtx || pCtx.progressiveRender) && task.__idxInPipeline > pipeline.blockIndex; - var step = incremental ? pipeline.step : null; - var modDataCount = pCtx && pCtx.modDataCount; - var modBy = modDataCount != null ? Math.ceil(modDataCount / step) : null; - return { - step: step, - modBy: modBy, - modDataCount: modDataCount - }; - }; - - Scheduler.prototype.getPipeline = function (pipelineId) { - return this._pipelineMap.get(pipelineId); - }; - /** - * Current, progressive rendering starts from visual and layout. - * Always detect render mode in the same stage, avoiding that incorrect - * detection caused by data filtering. - * Caution: - * `updateStreamModes` use `seriesModel.getData()`. - */ - - - Scheduler.prototype.updateStreamModes = function (seriesModel, view) { - var pipeline = this._pipelineMap.get(seriesModel.uid); - - var data = seriesModel.getData(); - var dataLen = data.count(); // `progressiveRender` means that can render progressively in each - // animation frame. Note that some types of series do not provide - // `view.incrementalPrepareRender` but support `chart.appendData`. We - // use the term `incremental` but not `progressive` to describe the - // case that `chart.appendData`. - - var progressiveRender = pipeline.progressiveEnabled && view.incrementalPrepareRender && dataLen >= pipeline.threshold; - var large = seriesModel.get('large') && dataLen >= seriesModel.get('largeThreshold'); // TODO: modDataCount should not updated if `appendData`, otherwise cause whole repaint. - // see `test/candlestick-large3.html` - - var modDataCount = seriesModel.get('progressiveChunkMode') === 'mod' ? dataLen : null; - seriesModel.pipelineContext = pipeline.context = { - progressiveRender: progressiveRender, - modDataCount: modDataCount, - large: large - }; - }; - - Scheduler.prototype.restorePipelines = function (ecModel) { - var scheduler = this; - var pipelineMap = scheduler._pipelineMap = createHashMap(); - ecModel.eachSeries(function (seriesModel) { - var progressive = seriesModel.getProgressive(); - var pipelineId = seriesModel.uid; - pipelineMap.set(pipelineId, { - id: pipelineId, - head: null, - tail: null, - threshold: seriesModel.getProgressiveThreshold(), - progressiveEnabled: progressive && !(seriesModel.preventIncremental && seriesModel.preventIncremental()), - blockIndex: -1, - step: Math.round(progressive || 700), - count: 0 - }); - - scheduler._pipe(seriesModel, seriesModel.dataTask); - }); - }; - - Scheduler.prototype.prepareStageTasks = function () { - var stageTaskMap = this._stageTaskMap; - var ecModel = this.api.getModel(); - var api = this.api; - each(this._allHandlers, function (handler) { - var record = stageTaskMap.get(handler.uid) || stageTaskMap.set(handler.uid, {}); - var errMsg = ''; - - if ("development" !== 'production') { - // Currently do not need to support to sepecify them both. - errMsg = '"reset" and "overallReset" must not be both specified.'; - } - - assert(!(handler.reset && handler.overallReset), errMsg); - handler.reset && this._createSeriesStageTask(handler, record, ecModel, api); - handler.overallReset && this._createOverallStageTask(handler, record, ecModel, api); - }, this); - }; - - Scheduler.prototype.prepareView = function (view, model, ecModel, api) { - var renderTask = view.renderTask; - var context = renderTask.context; - context.model = model; - context.ecModel = ecModel; - context.api = api; - renderTask.__block = !view.incrementalPrepareRender; - - this._pipe(model, renderTask); - }; - - Scheduler.prototype.performDataProcessorTasks = function (ecModel, payload) { - // If we do not use `block` here, it should be considered when to update modes. - this._performStageTasks(this._dataProcessorHandlers, ecModel, payload, { - block: true - }); - }; - - Scheduler.prototype.performVisualTasks = function (ecModel, payload, opt) { - this._performStageTasks(this._visualHandlers, ecModel, payload, opt); - }; - - Scheduler.prototype._performStageTasks = function (stageHandlers, ecModel, payload, opt) { - opt = opt || {}; - var unfinished = false; - var scheduler = this; - each(stageHandlers, function (stageHandler, idx) { - if (opt.visualType && opt.visualType !== stageHandler.visualType) { - return; - } - - var stageHandlerRecord = scheduler._stageTaskMap.get(stageHandler.uid); - - var seriesTaskMap = stageHandlerRecord.seriesTaskMap; - var overallTask = stageHandlerRecord.overallTask; - - if (overallTask) { - var overallNeedDirty_1; - var agentStubMap = overallTask.agentStubMap; - agentStubMap.each(function (stub) { - if (needSetDirty(opt, stub)) { - stub.dirty(); - overallNeedDirty_1 = true; - } - }); - overallNeedDirty_1 && overallTask.dirty(); - scheduler.updatePayload(overallTask, payload); - var performArgs_1 = scheduler.getPerformArgs(overallTask, opt.block); // Execute stubs firstly, which may set the overall task dirty, - // then execute the overall task. And stub will call seriesModel.setData, - // which ensures that in the overallTask seriesModel.getData() will not - // return incorrect data. - - agentStubMap.each(function (stub) { - stub.perform(performArgs_1); - }); - - if (overallTask.perform(performArgs_1)) { - unfinished = true; - } - } else if (seriesTaskMap) { - seriesTaskMap.each(function (task, pipelineId) { - if (needSetDirty(opt, task)) { - task.dirty(); - } - - var performArgs = scheduler.getPerformArgs(task, opt.block); // FIXME - // if intending to declare `performRawSeries` in handlers, only - // stream-independent (specifically, data item independent) operations can be - // performed. Because if a series is filtered, most of the tasks will not - // be performed. A stream-dependent operation probably cause wrong biz logic. - // Perhaps we should not provide a separate callback for this case instead - // of providing the config `performRawSeries`. The stream-dependent operations - // and stream-independent operations should better not be mixed. - - performArgs.skip = !stageHandler.performRawSeries && ecModel.isSeriesFiltered(task.context.model); - scheduler.updatePayload(task, payload); - - if (task.perform(performArgs)) { - unfinished = true; - } - }); - } - }); - - function needSetDirty(opt, task) { - return opt.setDirty && (!opt.dirtyMap || opt.dirtyMap.get(task.__pipeline.id)); - } - - this.unfinished = unfinished || this.unfinished; - }; - - Scheduler.prototype.performSeriesTasks = function (ecModel) { - var unfinished; - ecModel.eachSeries(function (seriesModel) { - // Progress to the end for dataInit and dataRestore. - unfinished = seriesModel.dataTask.perform() || unfinished; - }); - this.unfinished = unfinished || this.unfinished; - }; - - Scheduler.prototype.plan = function () { - // Travel pipelines, check block. - this._pipelineMap.each(function (pipeline) { - var task = pipeline.tail; - - do { - if (task.__block) { - pipeline.blockIndex = task.__idxInPipeline; - break; - } - - task = task.getUpstream(); - } while (task); - }); - }; - - Scheduler.prototype.updatePayload = function (task, payload) { - payload !== 'remain' && (task.context.payload = payload); - }; - - Scheduler.prototype._createSeriesStageTask = function (stageHandler, stageHandlerRecord, ecModel, api) { - var scheduler = this; - var oldSeriesTaskMap = stageHandlerRecord.seriesTaskMap; // The count of stages are totally about only several dozen, so - // do not need to reuse the map. - - var newSeriesTaskMap = stageHandlerRecord.seriesTaskMap = createHashMap(); - var seriesType = stageHandler.seriesType; - var getTargetSeries = stageHandler.getTargetSeries; // If a stageHandler should cover all series, `createOnAllSeries` should be declared mandatorily, - // to avoid some typo or abuse. Otherwise if an extension do not specify a `seriesType`, - // it works but it may cause other irrelevant charts blocked. - - if (stageHandler.createOnAllSeries) { - ecModel.eachRawSeries(create); - } else if (seriesType) { - ecModel.eachRawSeriesByType(seriesType, create); - } else if (getTargetSeries) { - getTargetSeries(ecModel, api).each(create); - } - - function create(seriesModel) { - var pipelineId = seriesModel.uid; // Init tasks for each seriesModel only once. - // Reuse original task instance. - - var task = newSeriesTaskMap.set(pipelineId, oldSeriesTaskMap && oldSeriesTaskMap.get(pipelineId) || createTask({ - plan: seriesTaskPlan, - reset: seriesTaskReset, - count: seriesTaskCount - })); - task.context = { - model: seriesModel, - ecModel: ecModel, - api: api, - // PENDING: `useClearVisual` not used? - useClearVisual: stageHandler.isVisual && !stageHandler.isLayout, - plan: stageHandler.plan, - reset: stageHandler.reset, - scheduler: scheduler - }; - - scheduler._pipe(seriesModel, task); - } - }; - - Scheduler.prototype._createOverallStageTask = function (stageHandler, stageHandlerRecord, ecModel, api) { - var scheduler = this; - var overallTask = stageHandlerRecord.overallTask = stageHandlerRecord.overallTask // For overall task, the function only be called on reset stage. - || createTask({ - reset: overallTaskReset - }); - overallTask.context = { - ecModel: ecModel, - api: api, - overallReset: stageHandler.overallReset, - scheduler: scheduler - }; - var oldAgentStubMap = overallTask.agentStubMap; // The count of stages are totally about only several dozen, so - // do not need to reuse the map. - - var newAgentStubMap = overallTask.agentStubMap = createHashMap(); - var seriesType = stageHandler.seriesType; - var getTargetSeries = stageHandler.getTargetSeries; - var overallProgress = true; - var shouldOverallTaskDirty = false; // FIXME:TS never used, so comment it - // let modifyOutputEnd = stageHandler.modifyOutputEnd; - // An overall task with seriesType detected or has `getTargetSeries`, we add - // stub in each pipelines, it will set the overall task dirty when the pipeline - // progress. Moreover, to avoid call the overall task each frame (too frequent), - // we set the pipeline block. - - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = '"createOnAllSeries" is not supported for "overallReset", ' + 'because it will block all streams.'; - } - - assert(!stageHandler.createOnAllSeries, errMsg); - - if (seriesType) { - ecModel.eachRawSeriesByType(seriesType, createStub); - } else if (getTargetSeries) { - getTargetSeries(ecModel, api).each(createStub); - } // Otherwise, (usually it is legacy case), the overall task will only be - // executed when upstream is dirty. Otherwise the progressive rendering of all - // pipelines will be disabled unexpectedly. But it still needs stubs to receive - // dirty info from upstream. - else { - overallProgress = false; - each(ecModel.getSeries(), createStub); - } - - function createStub(seriesModel) { - var pipelineId = seriesModel.uid; - var stub = newAgentStubMap.set(pipelineId, oldAgentStubMap && oldAgentStubMap.get(pipelineId) || ( // When the result of `getTargetSeries` changed, the overallTask - // should be set as dirty and re-performed. - shouldOverallTaskDirty = true, createTask({ - reset: stubReset, - onDirty: stubOnDirty - }))); - stub.context = { - model: seriesModel, - overallProgress: overallProgress // FIXME:TS never used, so comment it - // modifyOutputEnd: modifyOutputEnd - - }; - stub.agent = overallTask; - stub.__block = overallProgress; - - scheduler._pipe(seriesModel, stub); - } - - if (shouldOverallTaskDirty) { - overallTask.dirty(); - } - }; - - Scheduler.prototype._pipe = function (seriesModel, task) { - var pipelineId = seriesModel.uid; - - var pipeline = this._pipelineMap.get(pipelineId); - - !pipeline.head && (pipeline.head = task); - pipeline.tail && pipeline.tail.pipe(task); - pipeline.tail = task; - task.__idxInPipeline = pipeline.count++; - task.__pipeline = pipeline; - }; - - Scheduler.wrapStageHandler = function (stageHandler, visualType) { - if (isFunction(stageHandler)) { - stageHandler = { - overallReset: stageHandler, - seriesType: detectSeriseType(stageHandler) - }; - } - - stageHandler.uid = getUID('stageHandler'); - visualType && (stageHandler.visualType = visualType); - return stageHandler; - }; - return Scheduler; - }(); - - function overallTaskReset(context) { - context.overallReset(context.ecModel, context.api, context.payload); - } - - function stubReset(context) { - return context.overallProgress && stubProgress; - } - - function stubProgress() { - this.agent.dirty(); - this.getDownstream().dirty(); - } - - function stubOnDirty() { - this.agent && this.agent.dirty(); - } - - function seriesTaskPlan(context) { - return context.plan ? context.plan(context.model, context.ecModel, context.api, context.payload) : null; - } - - function seriesTaskReset(context) { - if (context.useClearVisual) { - context.data.clearAllVisual(); - } - - var resetDefines = context.resetDefines = normalizeToArray(context.reset(context.model, context.ecModel, context.api, context.payload)); - return resetDefines.length > 1 ? map(resetDefines, function (v, idx) { - return makeSeriesTaskProgress(idx); - }) : singleSeriesTaskProgress; - } - - var singleSeriesTaskProgress = makeSeriesTaskProgress(0); - - function makeSeriesTaskProgress(resetDefineIdx) { - return function (params, context) { - var data = context.data; - var resetDefine = context.resetDefines[resetDefineIdx]; - - if (resetDefine && resetDefine.dataEach) { - for (var i = params.start; i < params.end; i++) { - resetDefine.dataEach(data, i); - } - } else if (resetDefine && resetDefine.progress) { - resetDefine.progress(params, data); - } - }; - } - - function seriesTaskCount(context) { - return context.data.count(); - } - /** - * Only some legacy stage handlers (usually in echarts extensions) are pure function. - * To ensure that they can work normally, they should work in block mode, that is, - * they should not be started util the previous tasks finished. So they cause the - * progressive rendering disabled. We try to detect the series type, to narrow down - * the block range to only the series type they concern, but not all series. - */ - - - function detectSeriseType(legacyFunc) { - seriesType = null; - - try { - // Assume there is no async when calling `eachSeriesByType`. - legacyFunc(ecModelMock, apiMock); - } catch (e) {} - - return seriesType; - } - - var ecModelMock = {}; - var apiMock = {}; - var seriesType; - mockMethods(ecModelMock, GlobalModel); - mockMethods(apiMock, ExtensionAPI); - - ecModelMock.eachSeriesByType = ecModelMock.eachRawSeriesByType = function (type) { - seriesType = type; - }; - - ecModelMock.eachComponent = function (cond) { - if (cond.mainType === 'series' && cond.subType) { - seriesType = cond.subType; - } - }; - - function mockMethods(target, Clz) { - /* eslint-disable */ - for (var name_1 in Clz.prototype) { - // Do not use hasOwnProperty - target[name_1] = noop; - } - /* eslint-enable */ - - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var colorAll = ['#37A2DA', '#32C5E9', '#67E0E3', '#9FE6B8', '#FFDB5C', '#ff9f7f', '#fb7293', '#E062AE', '#E690D1', '#e7bcf3', '#9d96f5', '#8378EA', '#96BFFF']; - var lightTheme = { - color: colorAll, - colorLayer: [['#37A2DA', '#ffd85c', '#fd7b5f'], ['#37A2DA', '#67E0E3', '#FFDB5C', '#ff9f7f', '#E062AE', '#9d96f5'], ['#37A2DA', '#32C5E9', '#9FE6B8', '#FFDB5C', '#ff9f7f', '#fb7293', '#e7bcf3', '#8378EA', '#96BFFF'], colorAll] - }; - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var contrastColor = '#B9B8CE'; - var backgroundColor = '#100C2A'; - - var axisCommon = function () { - return { - axisLine: { - lineStyle: { - color: contrastColor - } - }, - splitLine: { - lineStyle: { - color: '#484753' - } - }, - splitArea: { - areaStyle: { - color: ['rgba(255,255,255,0.02)', 'rgba(255,255,255,0.05)'] - } - }, - minorSplitLine: { - lineStyle: { - color: '#20203B' - } - } - }; - }; - - var colorPalette = ['#4992ff', '#7cffb2', '#fddd60', '#ff6e76', '#58d9f9', '#05c091', '#ff8a45', '#8d48e3', '#dd79ff']; - var theme = { - darkMode: true, - color: colorPalette, - backgroundColor: backgroundColor, - axisPointer: { - lineStyle: { - color: '#817f91' - }, - crossStyle: { - color: '#817f91' - }, - label: { - // TODO Contrast of label backgorundColor - color: '#fff' - } - }, - legend: { - textStyle: { - color: contrastColor - } - }, - textStyle: { - color: contrastColor - }, - title: { - textStyle: { - color: '#EEF1FA' - }, - subtextStyle: { - color: '#B9B8CE' - } - }, - toolbox: { - iconStyle: { - borderColor: contrastColor - } - }, - dataZoom: { - borderColor: '#71708A', - textStyle: { - color: contrastColor - }, - brushStyle: { - color: 'rgba(135,163,206,0.3)' - }, - handleStyle: { - color: '#353450', - borderColor: '#C5CBE3' - }, - moveHandleStyle: { - color: '#B0B6C3', - opacity: 0.3 - }, - fillerColor: 'rgba(135,163,206,0.2)', - emphasis: { - handleStyle: { - borderColor: '#91B7F2', - color: '#4D587D' - }, - moveHandleStyle: { - color: '#636D9A', - opacity: 0.7 - } - }, - dataBackground: { - lineStyle: { - color: '#71708A', - width: 1 - }, - areaStyle: { - color: '#71708A' - } - }, - selectedDataBackground: { - lineStyle: { - color: '#87A3CE' - }, - areaStyle: { - color: '#87A3CE' - } - } - }, - visualMap: { - textStyle: { - color: contrastColor - } - }, - timeline: { - lineStyle: { - color: contrastColor - }, - label: { - color: contrastColor - }, - controlStyle: { - color: contrastColor, - borderColor: contrastColor - } - }, - calendar: { - itemStyle: { - color: backgroundColor - }, - dayLabel: { - color: contrastColor - }, - monthLabel: { - color: contrastColor - }, - yearLabel: { - color: contrastColor - } - }, - timeAxis: axisCommon(), - logAxis: axisCommon(), - valueAxis: axisCommon(), - categoryAxis: axisCommon(), - line: { - symbol: 'circle' - }, - graph: { - color: colorPalette - }, - gauge: { - title: { - color: contrastColor - }, - axisLine: { - lineStyle: { - color: [[1, 'rgba(207,212,219,0.2)']] - } - }, - axisLabel: { - color: contrastColor - }, - detail: { - color: '#EEF1FA' - } - }, - candlestick: { - itemStyle: { - color: '#f64e56', - color0: '#54ea92', - borderColor: '#f64e56', - borderColor0: '#54ea92' // borderColor: '#ca2824', - // borderColor0: '#09a443' - - } - } - }; - theme.categoryAxis.splitLine.show = false; - - /** - * Usage of query: - * `chart.on('click', query, handler);` - * The `query` can be: - * + The component type query string, only `mainType` or `mainType.subType`, - * like: 'xAxis', 'series', 'xAxis.category' or 'series.line'. - * + The component query object, like: - * `{seriesIndex: 2}`, `{seriesName: 'xx'}`, `{seriesId: 'some'}`, - * `{xAxisIndex: 2}`, `{xAxisName: 'xx'}`, `{xAxisId: 'some'}`. - * + The data query object, like: - * `{dataIndex: 123}`, `{dataType: 'link'}`, `{name: 'some'}`. - * + The other query object (cmponent customized query), like: - * `{element: 'some'}` (only available in custom series). - * - * Caveat: If a prop in the `query` object is `null/undefined`, it is the - * same as there is no such prop in the `query` object. - */ - - var ECEventProcessor = - /** @class */ - function () { - function ECEventProcessor() {} - - ECEventProcessor.prototype.normalizeQuery = function (query) { - var cptQuery = {}; - var dataQuery = {}; - var otherQuery = {}; // `query` is `mainType` or `mainType.subType` of component. - - if (isString(query)) { - var condCptType = parseClassType(query); // `.main` and `.sub` may be ''. - - cptQuery.mainType = condCptType.main || null; - cptQuery.subType = condCptType.sub || null; - } // `query` is an object, convert to {mainType, index, name, id}. - else { - // `xxxIndex`, `xxxName`, `xxxId`, `name`, `dataIndex`, `dataType` is reserved, - // can not be used in `compomentModel.filterForExposedEvent`. - var suffixes_1 = ['Index', 'Name', 'Id']; - var dataKeys_1 = { - name: 1, - dataIndex: 1, - dataType: 1 - }; - each(query, function (val, key) { - var reserved = false; - - for (var i = 0; i < suffixes_1.length; i++) { - var propSuffix = suffixes_1[i]; - var suffixPos = key.lastIndexOf(propSuffix); - - if (suffixPos > 0 && suffixPos === key.length - propSuffix.length) { - var mainType = key.slice(0, suffixPos); // Consider `dataIndex`. - - if (mainType !== 'data') { - cptQuery.mainType = mainType; - cptQuery[propSuffix.toLowerCase()] = val; - reserved = true; - } - } - } - - if (dataKeys_1.hasOwnProperty(key)) { - dataQuery[key] = val; - reserved = true; - } - - if (!reserved) { - otherQuery[key] = val; - } - }); - } - - return { - cptQuery: cptQuery, - dataQuery: dataQuery, - otherQuery: otherQuery - }; - }; - - ECEventProcessor.prototype.filter = function (eventType, query) { - // They should be assigned before each trigger call. - var eventInfo = this.eventInfo; - - if (!eventInfo) { - return true; - } - - var targetEl = eventInfo.targetEl; - var packedEvent = eventInfo.packedEvent; - var model = eventInfo.model; - var view = eventInfo.view; // For event like 'globalout'. - - if (!model || !view) { - return true; - } - - var cptQuery = query.cptQuery; - var dataQuery = query.dataQuery; - return check(cptQuery, model, 'mainType') && check(cptQuery, model, 'subType') && check(cptQuery, model, 'index', 'componentIndex') && check(cptQuery, model, 'name') && check(cptQuery, model, 'id') && check(dataQuery, packedEvent, 'name') && check(dataQuery, packedEvent, 'dataIndex') && check(dataQuery, packedEvent, 'dataType') && (!view.filterForExposedEvent || view.filterForExposedEvent(eventType, query.otherQuery, targetEl, packedEvent)); - - function check(query, host, prop, propOnHost) { - return query[prop] == null || host[propOnHost || prop] === query[prop]; - } - }; - - ECEventProcessor.prototype.afterTrigger = function () { - // Make sure the eventInfo won't be used in next trigger. - this.eventInfo = null; - }; - - return ECEventProcessor; - }(); - - var SYMBOL_PROPS_WITH_CB = ['symbol', 'symbolSize', 'symbolRotate', 'symbolOffset']; - var SYMBOL_PROPS = SYMBOL_PROPS_WITH_CB.concat(['symbolKeepAspect']); // Encoding visual for all series include which is filtered for legend drawing - - var seriesSymbolTask = { - createOnAllSeries: true, - // For legend. - performRawSeries: true, - reset: function (seriesModel, ecModel) { - var data = seriesModel.getData(); - - if (seriesModel.legendIcon) { - data.setVisual('legendIcon', seriesModel.legendIcon); - } - - if (!seriesModel.hasSymbolVisual) { - return; - } - - var symbolOptions = {}; - var symbolOptionsCb = {}; - var hasCallback = false; - - for (var i = 0; i < SYMBOL_PROPS_WITH_CB.length; i++) { - var symbolPropName = SYMBOL_PROPS_WITH_CB[i]; - var val = seriesModel.get(symbolPropName); - - if (isFunction(val)) { - hasCallback = true; - symbolOptionsCb[symbolPropName] = val; - } else { - symbolOptions[symbolPropName] = val; - } - } - - symbolOptions.symbol = symbolOptions.symbol || seriesModel.defaultSymbol; - data.setVisual(extend({ - legendIcon: seriesModel.legendIcon || symbolOptions.symbol, - symbolKeepAspect: seriesModel.get('symbolKeepAspect') - }, symbolOptions)); // Only visible series has each data be visual encoded - - if (ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - var symbolPropsCb = keys(symbolOptionsCb); - - function dataEach(data, idx) { - var rawValue = seriesModel.getRawValue(idx); - var params = seriesModel.getDataParams(idx); - - for (var i = 0; i < symbolPropsCb.length; i++) { - var symbolPropName = symbolPropsCb[i]; - data.setItemVisual(idx, symbolPropName, symbolOptionsCb[symbolPropName](rawValue, params)); - } - } - - return { - dataEach: hasCallback ? dataEach : null - }; - } - }; - var dataSymbolTask = { - createOnAllSeries: true, - // For legend. - performRawSeries: true, - reset: function (seriesModel, ecModel) { - if (!seriesModel.hasSymbolVisual) { - return; - } // Only visible series has each data be visual encoded - - - if (ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - var data = seriesModel.getData(); - - function dataEach(data, idx) { - var itemModel = data.getItemModel(idx); - - for (var i = 0; i < SYMBOL_PROPS.length; i++) { - var symbolPropName = SYMBOL_PROPS[i]; - var val = itemModel.getShallow(symbolPropName, true); - - if (val != null) { - data.setItemVisual(idx, symbolPropName, val); - } - } - } - - return { - dataEach: data.hasItemOption ? dataEach : null - }; - } - }; - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function getItemVisualFromData(data, dataIndex, key) { - switch (key) { - case 'color': - var style = data.getItemVisual(dataIndex, 'style'); - return style[data.getVisual('drawType')]; - - case 'opacity': - return data.getItemVisual(dataIndex, 'style').opacity; - - case 'symbol': - case 'symbolSize': - case 'liftZ': - return data.getItemVisual(dataIndex, key); - - default: - if ("development" !== 'production') { - console.warn("Unknown visual type " + key); - } - - } - } - function getVisualFromData(data, key) { - switch (key) { - case 'color': - var style = data.getVisual('style'); - return style[data.getVisual('drawType')]; - - case 'opacity': - return data.getVisual('style').opacity; - - case 'symbol': - case 'symbolSize': - case 'liftZ': - return data.getVisual(key); - - default: - if ("development" !== 'production') { - console.warn("Unknown visual type " + key); - } - - } - } - function setItemVisualFromData(data, dataIndex, key, value) { - switch (key) { - case 'color': - // Make sure not sharing style object. - var style = data.ensureUniqueItemVisual(dataIndex, 'style'); - style[data.getVisual('drawType')] = value; // Mark the color has been changed, not from palette anymore - - data.setItemVisual(dataIndex, 'colorFromPalette', false); - break; - - case 'opacity': - data.ensureUniqueItemVisual(dataIndex, 'style').opacity = value; - break; - - case 'symbol': - case 'symbolSize': - case 'liftZ': - data.setItemVisual(dataIndex, key, value); - break; - - default: - if ("development" !== 'production') { - console.warn("Unknown visual type " + key); - } - - } - } - - // Includes: pieSelect, pieUnSelect, pieToggleSelect, mapSelect, mapUnSelect, mapToggleSelect - - function createLegacyDataSelectAction(seriesType, ecRegisterAction) { - function getSeriesIndices(ecModel, payload) { - var seriesIndices = []; - ecModel.eachComponent({ - mainType: 'series', - subType: seriesType, - query: payload - }, function (seriesModel) { - seriesIndices.push(seriesModel.seriesIndex); - }); - return seriesIndices; - } - - each([[seriesType + 'ToggleSelect', 'toggleSelect'], [seriesType + 'Select', 'select'], [seriesType + 'UnSelect', 'unselect']], function (eventsMap) { - ecRegisterAction(eventsMap[0], function (payload, ecModel, api) { - payload = extend({}, payload); - - if ("development" !== 'production') { - deprecateReplaceLog(payload.type, eventsMap[1]); - } - - api.dispatchAction(extend(payload, { - type: eventsMap[1], - seriesIndex: getSeriesIndices(ecModel, payload) - })); - }); - }); - } - - function handleSeriesLegacySelectEvents(type, eventPostfix, ecIns, ecModel, payload) { - var legacyEventName = type + eventPostfix; - - if (!ecIns.isSilent(legacyEventName)) { - if ("development" !== 'production') { - deprecateLog("event " + legacyEventName + " is deprecated."); - } - - ecModel.eachComponent({ - mainType: 'series', - subType: 'pie' - }, function (seriesModel) { - var seriesIndex = seriesModel.seriesIndex; - var selectedMap = seriesModel.option.selectedMap; - var selected = payload.selected; - - for (var i = 0; i < selected.length; i++) { - if (selected[i].seriesIndex === seriesIndex) { - var data = seriesModel.getData(); - var dataIndex = queryDataIndex(data, payload.fromActionPayload); - ecIns.trigger(legacyEventName, { - type: legacyEventName, - seriesId: seriesModel.id, - name: isArray(dataIndex) ? data.getName(dataIndex[0]) : data.getName(dataIndex), - selected: isString(selectedMap) ? selectedMap : extend({}, selectedMap) - }); - } - } - }); - } - } - - function handleLegacySelectEvents(messageCenter, ecIns, api) { - messageCenter.on('selectchanged', function (params) { - var ecModel = api.getModel(); - - if (params.isFromClick) { - handleSeriesLegacySelectEvents('map', 'selectchanged', ecIns, ecModel, params); - handleSeriesLegacySelectEvents('pie', 'selectchanged', ecIns, ecModel, params); - } else if (params.fromAction === 'select') { - handleSeriesLegacySelectEvents('map', 'selected', ecIns, ecModel, params); - handleSeriesLegacySelectEvents('pie', 'selected', ecIns, ecModel, params); - } else if (params.fromAction === 'unselect') { - handleSeriesLegacySelectEvents('map', 'unselected', ecIns, ecModel, params); - handleSeriesLegacySelectEvents('pie', 'unselected', ecIns, ecModel, params); - } - }); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function findEventDispatcher(target, det, returnFirstMatch) { - var found; - - while (target) { - if (det(target)) { - found = target; - - if (returnFirstMatch) { - break; - } - } - - target = target.__hostTarget || target.parent; - } - - return found; - } - - var wmUniqueIndex = Math.round(Math.random() * 9); - var supportDefineProperty = typeof Object.defineProperty === 'function'; - var WeakMap = (function () { - function WeakMap() { - this._id = '__ec_inner_' + wmUniqueIndex++; - } - WeakMap.prototype.get = function (key) { - return this._guard(key)[this._id]; - }; - WeakMap.prototype.set = function (key, value) { - var target = this._guard(key); - if (supportDefineProperty) { - Object.defineProperty(target, this._id, { - value: value, - enumerable: false, - configurable: true - }); - } - else { - target[this._id] = value; - } - return this; - }; - WeakMap.prototype["delete"] = function (key) { - if (this.has(key)) { - delete this._guard(key)[this._id]; - return true; - } - return false; - }; - WeakMap.prototype.has = function (key) { - return !!this._guard(key)[this._id]; - }; - WeakMap.prototype._guard = function (key) { - if (key !== Object(key)) { - throw TypeError('Value of WeakMap is not a non-null object.'); - } - return key; - }; - return WeakMap; - }()); - - /** - * Triangle shape - * @inner - */ - - var Triangle = Path.extend({ - type: 'triangle', - shape: { - cx: 0, - cy: 0, - width: 0, - height: 0 - }, - buildPath: function (path, shape) { - var cx = shape.cx; - var cy = shape.cy; - var width = shape.width / 2; - var height = shape.height / 2; - path.moveTo(cx, cy - height); - path.lineTo(cx + width, cy + height); - path.lineTo(cx - width, cy + height); - path.closePath(); - } - }); - /** - * Diamond shape - * @inner - */ - - var Diamond = Path.extend({ - type: 'diamond', - shape: { - cx: 0, - cy: 0, - width: 0, - height: 0 - }, - buildPath: function (path, shape) { - var cx = shape.cx; - var cy = shape.cy; - var width = shape.width / 2; - var height = shape.height / 2; - path.moveTo(cx, cy - height); - path.lineTo(cx + width, cy); - path.lineTo(cx, cy + height); - path.lineTo(cx - width, cy); - path.closePath(); - } - }); - /** - * Pin shape - * @inner - */ - - var Pin = Path.extend({ - type: 'pin', - shape: { - // x, y on the cusp - x: 0, - y: 0, - width: 0, - height: 0 - }, - buildPath: function (path, shape) { - var x = shape.x; - var y = shape.y; - var w = shape.width / 5 * 3; // Height must be larger than width - - var h = Math.max(w, shape.height); - var r = w / 2; // Dist on y with tangent point and circle center - - var dy = r * r / (h - r); - var cy = y - h + r + dy; - var angle = Math.asin(dy / r); // Dist on x with tangent point and circle center - - var dx = Math.cos(angle) * r; - var tanX = Math.sin(angle); - var tanY = Math.cos(angle); - var cpLen = r * 0.6; - var cpLen2 = r * 0.7; - path.moveTo(x - dx, cy + dy); - path.arc(x, cy, r, Math.PI - angle, Math.PI * 2 + angle); - path.bezierCurveTo(x + dx - tanX * cpLen, cy + dy + tanY * cpLen, x, y - cpLen2, x, y); - path.bezierCurveTo(x, y - cpLen2, x - dx + tanX * cpLen, cy + dy + tanY * cpLen, x - dx, cy + dy); - path.closePath(); - } - }); - /** - * Arrow shape - * @inner - */ - - var Arrow = Path.extend({ - type: 'arrow', - shape: { - x: 0, - y: 0, - width: 0, - height: 0 - }, - buildPath: function (ctx, shape) { - var height = shape.height; - var width = shape.width; - var x = shape.x; - var y = shape.y; - var dx = width / 3 * 2; - ctx.moveTo(x, y); - ctx.lineTo(x + dx, y + height); - ctx.lineTo(x, y + height / 4 * 3); - ctx.lineTo(x - dx, y + height); - ctx.lineTo(x, y); - ctx.closePath(); - } - }); - /** - * Map of path constructors - */ - // TODO Use function to build symbol path. - - var symbolCtors = { - line: Line, - rect: Rect, - roundRect: Rect, - square: Rect, - circle: Circle, - diamond: Diamond, - pin: Pin, - arrow: Arrow, - triangle: Triangle - }; - var symbolShapeMakers = { - line: function (x, y, w, h, shape) { - shape.x1 = x; - shape.y1 = y + h / 2; - shape.x2 = x + w; - shape.y2 = y + h / 2; - }, - rect: function (x, y, w, h, shape) { - shape.x = x; - shape.y = y; - shape.width = w; - shape.height = h; - }, - roundRect: function (x, y, w, h, shape) { - shape.x = x; - shape.y = y; - shape.width = w; - shape.height = h; - shape.r = Math.min(w, h) / 4; - }, - square: function (x, y, w, h, shape) { - var size = Math.min(w, h); - shape.x = x; - shape.y = y; - shape.width = size; - shape.height = size; - }, - circle: function (x, y, w, h, shape) { - // Put circle in the center of square - shape.cx = x + w / 2; - shape.cy = y + h / 2; - shape.r = Math.min(w, h) / 2; - }, - diamond: function (x, y, w, h, shape) { - shape.cx = x + w / 2; - shape.cy = y + h / 2; - shape.width = w; - shape.height = h; - }, - pin: function (x, y, w, h, shape) { - shape.x = x + w / 2; - shape.y = y + h / 2; - shape.width = w; - shape.height = h; - }, - arrow: function (x, y, w, h, shape) { - shape.x = x + w / 2; - shape.y = y + h / 2; - shape.width = w; - shape.height = h; - }, - triangle: function (x, y, w, h, shape) { - shape.cx = x + w / 2; - shape.cy = y + h / 2; - shape.width = w; - shape.height = h; - } - }; - var symbolBuildProxies = {}; - each(symbolCtors, function (Ctor, name) { - symbolBuildProxies[name] = new Ctor(); - }); - var SymbolClz = Path.extend({ - type: 'symbol', - shape: { - symbolType: '', - x: 0, - y: 0, - width: 0, - height: 0 - }, - calculateTextPosition: function (out, config, rect) { - var res = calculateTextPosition(out, config, rect); - var shape = this.shape; - - if (shape && shape.symbolType === 'pin' && config.position === 'inside') { - res.y = rect.y + rect.height * 0.4; - } - - return res; - }, - buildPath: function (ctx, shape, inBundle) { - var symbolType = shape.symbolType; - - if (symbolType !== 'none') { - var proxySymbol = symbolBuildProxies[symbolType]; - - if (!proxySymbol) { - // Default rect - symbolType = 'rect'; - proxySymbol = symbolBuildProxies[symbolType]; - } - - symbolShapeMakers[symbolType](shape.x, shape.y, shape.width, shape.height, proxySymbol.shape); - proxySymbol.buildPath(ctx, proxySymbol.shape, inBundle); - } - } - }); // Provide setColor helper method to avoid determine if set the fill or stroke outside - - function symbolPathSetColor(color, innerColor) { - if (this.type !== 'image') { - var symbolStyle = this.style; - - if (this.__isEmptyBrush) { - symbolStyle.stroke = color; - symbolStyle.fill = innerColor || '#fff'; // TODO Same width with lineStyle in LineView - - symbolStyle.lineWidth = 2; - } else if (this.shape.symbolType === 'line') { - symbolStyle.stroke = color; - } else { - symbolStyle.fill = color; - } - - this.markRedraw(); - } - } - /** - * Create a symbol element with given symbol configuration: shape, x, y, width, height, color - */ - - - function createSymbol(symbolType, x, y, w, h, color, // whether to keep the ratio of w/h, - keepAspect) { - // TODO Support image object, DynamicImage. - var isEmpty = symbolType.indexOf('empty') === 0; - - if (isEmpty) { - symbolType = symbolType.substr(5, 1).toLowerCase() + symbolType.substr(6); - } - - var symbolPath; - - if (symbolType.indexOf('image://') === 0) { - symbolPath = makeImage(symbolType.slice(8), new BoundingRect(x, y, w, h), keepAspect ? 'center' : 'cover'); - } else if (symbolType.indexOf('path://') === 0) { - symbolPath = makePath(symbolType.slice(7), {}, new BoundingRect(x, y, w, h), keepAspect ? 'center' : 'cover'); - } else { - symbolPath = new SymbolClz({ - shape: { - symbolType: symbolType, - x: x, - y: y, - width: w, - height: h - } - }); - } - - symbolPath.__isEmptyBrush = isEmpty; // TODO Should deprecate setColor - - symbolPath.setColor = symbolPathSetColor; - - if (color) { - symbolPath.setColor(color); - } - - return symbolPath; - } - function normalizeSymbolSize(symbolSize) { - if (!isArray(symbolSize)) { - symbolSize = [+symbolSize, +symbolSize]; - } - - return [symbolSize[0] || 0, symbolSize[1] || 0]; - } - function normalizeSymbolOffset(symbolOffset, symbolSize) { - if (symbolOffset == null) { - return; - } - - if (!isArray(symbolOffset)) { - symbolOffset = [symbolOffset, symbolOffset]; - } - - return [parsePercent$1(symbolOffset[0], symbolSize[0]) || 0, parsePercent$1(retrieve2(symbolOffset[1], symbolOffset[0]), symbolSize[1]) || 0]; - } - - function isSafeNum(num) { - return isFinite(num); - } - function createLinearGradient(ctx, obj, rect) { - var x = obj.x == null ? 0 : obj.x; - var x2 = obj.x2 == null ? 1 : obj.x2; - var y = obj.y == null ? 0 : obj.y; - var y2 = obj.y2 == null ? 0 : obj.y2; - if (!obj.global) { - x = x * rect.width + rect.x; - x2 = x2 * rect.width + rect.x; - y = y * rect.height + rect.y; - y2 = y2 * rect.height + rect.y; - } - x = isSafeNum(x) ? x : 0; - x2 = isSafeNum(x2) ? x2 : 1; - y = isSafeNum(y) ? y : 0; - y2 = isSafeNum(y2) ? y2 : 0; - var canvasGradient = ctx.createLinearGradient(x, y, x2, y2); - return canvasGradient; - } - function createRadialGradient(ctx, obj, rect) { - var width = rect.width; - var height = rect.height; - var min = Math.min(width, height); - var x = obj.x == null ? 0.5 : obj.x; - var y = obj.y == null ? 0.5 : obj.y; - var r = obj.r == null ? 0.5 : obj.r; - if (!obj.global) { - x = x * width + rect.x; - y = y * height + rect.y; - r = r * min; - } - x = isSafeNum(x) ? x : 0.5; - y = isSafeNum(y) ? y : 0.5; - r = r >= 0 && isSafeNum(r) ? r : 0.5; - var canvasGradient = ctx.createRadialGradient(x, y, 0, x, y, r); - return canvasGradient; - } - function getCanvasGradient(ctx, obj, rect) { - var canvasGradient = obj.type === 'radial' - ? createRadialGradient(ctx, obj, rect) - : createLinearGradient(ctx, obj, rect); - var colorStops = obj.colorStops; - for (var i = 0; i < colorStops.length; i++) { - canvasGradient.addColorStop(colorStops[i].offset, colorStops[i].color); - } - return canvasGradient; - } - function isClipPathChanged(clipPaths, prevClipPaths) { - if (clipPaths === prevClipPaths || (!clipPaths && !prevClipPaths)) { - return false; - } - if (!clipPaths || !prevClipPaths || (clipPaths.length !== prevClipPaths.length)) { - return true; - } - for (var i = 0; i < clipPaths.length; i++) { - if (clipPaths[i] !== prevClipPaths[i]) { - return true; - } - } - return false; - } - function parseInt10(val) { - return parseInt(val, 10); - } - function getSize(root, whIdx, opts) { - var wh = ['width', 'height'][whIdx]; - var cwh = ['clientWidth', 'clientHeight'][whIdx]; - var plt = ['paddingLeft', 'paddingTop'][whIdx]; - var prb = ['paddingRight', 'paddingBottom'][whIdx]; - if (opts[wh] != null && opts[wh] !== 'auto') { - return parseFloat(opts[wh]); - } - var stl = document.defaultView.getComputedStyle(root); - return ((root[cwh] || parseInt10(stl[wh]) || parseInt10(root.style[wh])) - - (parseInt10(stl[plt]) || 0) - - (parseInt10(stl[prb]) || 0)) | 0; - } - - function normalizeLineDash(lineType, lineWidth) { - if (!lineType || lineType === 'solid' || !(lineWidth > 0)) { - return null; - } - return lineType === 'dashed' - ? [4 * lineWidth, 2 * lineWidth] - : lineType === 'dotted' - ? [lineWidth] - : isNumber(lineType) - ? [lineType] : isArray(lineType) ? lineType : null; - } - function getLineDash(el) { - var style = el.style; - var lineDash = style.lineDash && style.lineWidth > 0 && normalizeLineDash(style.lineDash, style.lineWidth); - var lineDashOffset = style.lineDashOffset; - if (lineDash) { - var lineScale_1 = (style.strokeNoScale && el.getLineScale) ? el.getLineScale() : 1; - if (lineScale_1 && lineScale_1 !== 1) { - lineDash = map(lineDash, function (rawVal) { - return rawVal / lineScale_1; - }); - lineDashOffset /= lineScale_1; - } - } - return [lineDash, lineDashOffset]; - } - - var pathProxyForDraw = new PathProxy(true); - function styleHasStroke(style) { - var stroke = style.stroke; - return !(stroke == null || stroke === 'none' || !(style.lineWidth > 0)); - } - function isValidStrokeFillStyle(strokeOrFill) { - return typeof strokeOrFill === 'string' && strokeOrFill !== 'none'; - } - function styleHasFill(style) { - var fill = style.fill; - return fill != null && fill !== 'none'; - } - function doFillPath(ctx, style) { - if (style.fillOpacity != null && style.fillOpacity !== 1) { - var originalGlobalAlpha = ctx.globalAlpha; - ctx.globalAlpha = style.fillOpacity * style.opacity; - ctx.fill(); - ctx.globalAlpha = originalGlobalAlpha; - } - else { - ctx.fill(); - } - } - function doStrokePath(ctx, style) { - if (style.strokeOpacity != null && style.strokeOpacity !== 1) { - var originalGlobalAlpha = ctx.globalAlpha; - ctx.globalAlpha = style.strokeOpacity * style.opacity; - ctx.stroke(); - ctx.globalAlpha = originalGlobalAlpha; - } - else { - ctx.stroke(); - } - } - function createCanvasPattern(ctx, pattern, el) { - var image = createOrUpdateImage(pattern.image, pattern.__image, el); - if (isImageReady(image)) { - var canvasPattern = ctx.createPattern(image, pattern.repeat || 'repeat'); - if (typeof DOMMatrix === 'function' - && canvasPattern - && canvasPattern.setTransform) { - var matrix = new DOMMatrix(); - matrix.translateSelf((pattern.x || 0), (pattern.y || 0)); - matrix.rotateSelf(0, 0, (pattern.rotation || 0) * RADIAN_TO_DEGREE); - matrix.scaleSelf((pattern.scaleX || 1), (pattern.scaleY || 1)); - canvasPattern.setTransform(matrix); - } - return canvasPattern; - } - } - function brushPath(ctx, el, style, inBatch) { - var _a; - var hasStroke = styleHasStroke(style); - var hasFill = styleHasFill(style); - var strokePercent = style.strokePercent; - var strokePart = strokePercent < 1; - var firstDraw = !el.path; - if ((!el.silent || strokePart) && firstDraw) { - el.createPathProxy(); - } - var path = el.path || pathProxyForDraw; - var dirtyFlag = el.__dirty; - if (!inBatch) { - var fill = style.fill; - var stroke = style.stroke; - var hasFillGradient = hasFill && !!fill.colorStops; - var hasStrokeGradient = hasStroke && !!stroke.colorStops; - var hasFillPattern = hasFill && !!fill.image; - var hasStrokePattern = hasStroke && !!stroke.image; - var fillGradient = void 0; - var strokeGradient = void 0; - var fillPattern = void 0; - var strokePattern = void 0; - var rect = void 0; - if (hasFillGradient || hasStrokeGradient) { - rect = el.getBoundingRect(); - } - if (hasFillGradient) { - fillGradient = dirtyFlag - ? getCanvasGradient(ctx, fill, rect) - : el.__canvasFillGradient; - el.__canvasFillGradient = fillGradient; - } - if (hasStrokeGradient) { - strokeGradient = dirtyFlag - ? getCanvasGradient(ctx, stroke, rect) - : el.__canvasStrokeGradient; - el.__canvasStrokeGradient = strokeGradient; - } - if (hasFillPattern) { - fillPattern = (dirtyFlag || !el.__canvasFillPattern) - ? createCanvasPattern(ctx, fill, el) - : el.__canvasFillPattern; - el.__canvasFillPattern = fillPattern; - } - if (hasStrokePattern) { - strokePattern = (dirtyFlag || !el.__canvasStrokePattern) - ? createCanvasPattern(ctx, stroke, el) - : el.__canvasStrokePattern; - el.__canvasStrokePattern = fillPattern; - } - if (hasFillGradient) { - ctx.fillStyle = fillGradient; - } - else if (hasFillPattern) { - if (fillPattern) { - ctx.fillStyle = fillPattern; - } - else { - hasFill = false; - } - } - if (hasStrokeGradient) { - ctx.strokeStyle = strokeGradient; - } - else if (hasStrokePattern) { - if (strokePattern) { - ctx.strokeStyle = strokePattern; - } - else { - hasStroke = false; - } - } - } - var scale = el.getGlobalScale(); - path.setScale(scale[0], scale[1], el.segmentIgnoreThreshold); - var lineDash; - var lineDashOffset; - if (ctx.setLineDash && style.lineDash) { - _a = getLineDash(el), lineDash = _a[0], lineDashOffset = _a[1]; - } - var needsRebuild = true; - if (firstDraw || (dirtyFlag & SHAPE_CHANGED_BIT)) { - path.setDPR(ctx.dpr); - if (strokePart) { - path.setContext(null); - } - else { - path.setContext(ctx); - needsRebuild = false; - } - path.reset(); - el.buildPath(path, el.shape, inBatch); - path.toStatic(); - el.pathUpdated(); - } - if (needsRebuild) { - path.rebuildPath(ctx, strokePart ? strokePercent : 1); - } - if (lineDash) { - ctx.setLineDash(lineDash); - ctx.lineDashOffset = lineDashOffset; - } - if (!inBatch) { - if (style.strokeFirst) { - if (hasStroke) { - doStrokePath(ctx, style); - } - if (hasFill) { - doFillPath(ctx, style); - } - } - else { - if (hasFill) { - doFillPath(ctx, style); - } - if (hasStroke) { - doStrokePath(ctx, style); - } - } - } - if (lineDash) { - ctx.setLineDash([]); - } - } - function brushImage(ctx, el, style) { - var image = el.__image = createOrUpdateImage(style.image, el.__image, el, el.onload); - if (!image || !isImageReady(image)) { - return; - } - var x = style.x || 0; - var y = style.y || 0; - var width = el.getWidth(); - var height = el.getHeight(); - var aspect = image.width / image.height; - if (width == null && height != null) { - width = height * aspect; - } - else if (height == null && width != null) { - height = width / aspect; - } - else if (width == null && height == null) { - width = image.width; - height = image.height; - } - if (style.sWidth && style.sHeight) { - var sx = style.sx || 0; - var sy = style.sy || 0; - ctx.drawImage(image, sx, sy, style.sWidth, style.sHeight, x, y, width, height); - } - else if (style.sx && style.sy) { - var sx = style.sx; - var sy = style.sy; - var sWidth = width - sx; - var sHeight = height - sy; - ctx.drawImage(image, sx, sy, sWidth, sHeight, x, y, width, height); - } - else { - ctx.drawImage(image, x, y, width, height); - } - } - function brushText(ctx, el, style) { - var _a; - var text = style.text; - text != null && (text += ''); - if (text) { - ctx.font = style.font || DEFAULT_FONT; - ctx.textAlign = style.textAlign; - ctx.textBaseline = style.textBaseline; - var lineDash = void 0; - var lineDashOffset = void 0; - if (ctx.setLineDash && style.lineDash) { - _a = getLineDash(el), lineDash = _a[0], lineDashOffset = _a[1]; - } - if (lineDash) { - ctx.setLineDash(lineDash); - ctx.lineDashOffset = lineDashOffset; - } - if (style.strokeFirst) { - if (styleHasStroke(style)) { - ctx.strokeText(text, style.x, style.y); - } - if (styleHasFill(style)) { - ctx.fillText(text, style.x, style.y); - } - } - else { - if (styleHasFill(style)) { - ctx.fillText(text, style.x, style.y); - } - if (styleHasStroke(style)) { - ctx.strokeText(text, style.x, style.y); - } - } - if (lineDash) { - ctx.setLineDash([]); - } - } - } - var SHADOW_NUMBER_PROPS = ['shadowBlur', 'shadowOffsetX', 'shadowOffsetY']; - var STROKE_PROPS = [ - ['lineCap', 'butt'], ['lineJoin', 'miter'], ['miterLimit', 10] - ]; - function bindCommonProps(ctx, style, prevStyle, forceSetAll, scope) { - var styleChanged = false; - if (!forceSetAll) { - prevStyle = prevStyle || {}; - if (style === prevStyle) { - return false; - } - } - if (forceSetAll || style.opacity !== prevStyle.opacity) { - flushPathDrawn(ctx, scope); - styleChanged = true; - var opacity = Math.max(Math.min(style.opacity, 1), 0); - ctx.globalAlpha = isNaN(opacity) ? DEFAULT_COMMON_STYLE.opacity : opacity; - } - if (forceSetAll || style.blend !== prevStyle.blend) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - ctx.globalCompositeOperation = style.blend || DEFAULT_COMMON_STYLE.blend; - } - for (var i = 0; i < SHADOW_NUMBER_PROPS.length; i++) { - var propName = SHADOW_NUMBER_PROPS[i]; - if (forceSetAll || style[propName] !== prevStyle[propName]) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - ctx[propName] = ctx.dpr * (style[propName] || 0); - } - } - if (forceSetAll || style.shadowColor !== prevStyle.shadowColor) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - ctx.shadowColor = style.shadowColor || DEFAULT_COMMON_STYLE.shadowColor; - } - return styleChanged; - } - function bindPathAndTextCommonStyle(ctx, el, prevEl, forceSetAll, scope) { - var style = getStyle(el, scope.inHover); - var prevStyle = forceSetAll - ? null - : (prevEl && getStyle(prevEl, scope.inHover) || {}); - if (style === prevStyle) { - return false; - } - var styleChanged = bindCommonProps(ctx, style, prevStyle, forceSetAll, scope); - if (forceSetAll || style.fill !== prevStyle.fill) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - isValidStrokeFillStyle(style.fill) && (ctx.fillStyle = style.fill); - } - if (forceSetAll || style.stroke !== prevStyle.stroke) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - isValidStrokeFillStyle(style.stroke) && (ctx.strokeStyle = style.stroke); - } - if (forceSetAll || style.opacity !== prevStyle.opacity) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - ctx.globalAlpha = style.opacity == null ? 1 : style.opacity; - } - if (el.hasStroke()) { - var lineWidth = style.lineWidth; - var newLineWidth = lineWidth / ((style.strokeNoScale && el.getLineScale) ? el.getLineScale() : 1); - if (ctx.lineWidth !== newLineWidth) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - ctx.lineWidth = newLineWidth; - } - } - for (var i = 0; i < STROKE_PROPS.length; i++) { - var prop = STROKE_PROPS[i]; - var propName = prop[0]; - if (forceSetAll || style[propName] !== prevStyle[propName]) { - if (!styleChanged) { - flushPathDrawn(ctx, scope); - styleChanged = true; - } - ctx[propName] = style[propName] || prop[1]; - } - } - return styleChanged; - } - function bindImageStyle(ctx, el, prevEl, forceSetAll, scope) { - return bindCommonProps(ctx, getStyle(el, scope.inHover), prevEl && getStyle(prevEl, scope.inHover), forceSetAll, scope); - } - function setContextTransform(ctx, el) { - var m = el.transform; - var dpr = ctx.dpr || 1; - if (m) { - ctx.setTransform(dpr * m[0], dpr * m[1], dpr * m[2], dpr * m[3], dpr * m[4], dpr * m[5]); - } - else { - ctx.setTransform(dpr, 0, 0, dpr, 0, 0); - } - } - function updateClipStatus(clipPaths, ctx, scope) { - var allClipped = false; - for (var i = 0; i < clipPaths.length; i++) { - var clipPath = clipPaths[i]; - allClipped = allClipped || clipPath.isZeroArea(); - setContextTransform(ctx, clipPath); - ctx.beginPath(); - clipPath.buildPath(ctx, clipPath.shape); - ctx.clip(); - } - scope.allClipped = allClipped; - } - function isTransformChanged(m0, m1) { - if (m0 && m1) { - return m0[0] !== m1[0] - || m0[1] !== m1[1] - || m0[2] !== m1[2] - || m0[3] !== m1[3] - || m0[4] !== m1[4] - || m0[5] !== m1[5]; - } - else if (!m0 && !m1) { - return false; - } - return true; - } - var DRAW_TYPE_PATH = 1; - var DRAW_TYPE_IMAGE = 2; - var DRAW_TYPE_TEXT = 3; - var DRAW_TYPE_INCREMENTAL = 4; - function canPathBatch(style) { - var hasFill = styleHasFill(style); - var hasStroke = styleHasStroke(style); - return !(style.lineDash - || !(+hasFill ^ +hasStroke) - || (hasFill && typeof style.fill !== 'string') - || (hasStroke && typeof style.stroke !== 'string') - || style.strokePercent < 1 - || style.strokeOpacity < 1 - || style.fillOpacity < 1); - } - function flushPathDrawn(ctx, scope) { - scope.batchFill && ctx.fill(); - scope.batchStroke && ctx.stroke(); - scope.batchFill = ''; - scope.batchStroke = ''; - } - function getStyle(el, inHover) { - return inHover ? (el.__hoverStyle || el.style) : el.style; - } - function brushSingle(ctx, el) { - brush(ctx, el, { inHover: false, viewWidth: 0, viewHeight: 0 }, true); - } - function brush(ctx, el, scope, isLast) { - var m = el.transform; - if (!el.shouldBePainted(scope.viewWidth, scope.viewHeight, false, false)) { - el.__dirty &= ~REDRAW_BIT; - el.__isRendered = false; - return; - } - var clipPaths = el.__clipPaths; - var prevElClipPaths = scope.prevElClipPaths; - var forceSetTransform = false; - var forceSetStyle = false; - if (!prevElClipPaths || isClipPathChanged(clipPaths, prevElClipPaths)) { - if (prevElClipPaths && prevElClipPaths.length) { - flushPathDrawn(ctx, scope); - ctx.restore(); - forceSetStyle = forceSetTransform = true; - scope.prevElClipPaths = null; - scope.allClipped = false; - scope.prevEl = null; - } - if (clipPaths && clipPaths.length) { - flushPathDrawn(ctx, scope); - ctx.save(); - updateClipStatus(clipPaths, ctx, scope); - forceSetTransform = true; - } - scope.prevElClipPaths = clipPaths; - } - if (scope.allClipped) { - el.__isRendered = false; - return; - } - el.beforeBrush && el.beforeBrush(); - el.innerBeforeBrush(); - var prevEl = scope.prevEl; - if (!prevEl) { - forceSetStyle = forceSetTransform = true; - } - var canBatchPath = el instanceof Path - && el.autoBatch - && canPathBatch(el.style); - if (forceSetTransform || isTransformChanged(m, prevEl.transform)) { - flushPathDrawn(ctx, scope); - setContextTransform(ctx, el); - } - else if (!canBatchPath) { - flushPathDrawn(ctx, scope); - } - var style = getStyle(el, scope.inHover); - if (el instanceof Path) { - if (scope.lastDrawType !== DRAW_TYPE_PATH) { - forceSetStyle = true; - scope.lastDrawType = DRAW_TYPE_PATH; - } - bindPathAndTextCommonStyle(ctx, el, prevEl, forceSetStyle, scope); - if (!canBatchPath || (!scope.batchFill && !scope.batchStroke)) { - ctx.beginPath(); - } - brushPath(ctx, el, style, canBatchPath); - if (canBatchPath) { - scope.batchFill = style.fill || ''; - scope.batchStroke = style.stroke || ''; - } - } - else { - if (el instanceof TSpan) { - if (scope.lastDrawType !== DRAW_TYPE_TEXT) { - forceSetStyle = true; - scope.lastDrawType = DRAW_TYPE_TEXT; - } - bindPathAndTextCommonStyle(ctx, el, prevEl, forceSetStyle, scope); - brushText(ctx, el, style); - } - else if (el instanceof ZRImage) { - if (scope.lastDrawType !== DRAW_TYPE_IMAGE) { - forceSetStyle = true; - scope.lastDrawType = DRAW_TYPE_IMAGE; - } - bindImageStyle(ctx, el, prevEl, forceSetStyle, scope); - brushImage(ctx, el, style); - } - else if (el.getTemporalDisplayables) { - if (scope.lastDrawType !== DRAW_TYPE_INCREMENTAL) { - forceSetStyle = true; - scope.lastDrawType = DRAW_TYPE_INCREMENTAL; - } - brushIncremental(ctx, el, scope); - } - } - if (canBatchPath && isLast) { - flushPathDrawn(ctx, scope); - } - el.innerAfterBrush(); - el.afterBrush && el.afterBrush(); - scope.prevEl = el; - el.__dirty = 0; - el.__isRendered = true; - } - function brushIncremental(ctx, el, scope) { - var displayables = el.getDisplayables(); - var temporalDisplayables = el.getTemporalDisplayables(); - ctx.save(); - var innerScope = { - prevElClipPaths: null, - prevEl: null, - allClipped: false, - viewWidth: scope.viewWidth, - viewHeight: scope.viewHeight, - inHover: scope.inHover - }; - var i; - var len; - for (i = el.getCursor(), len = displayables.length; i < len; i++) { - var displayable = displayables[i]; - displayable.beforeBrush && displayable.beforeBrush(); - displayable.innerBeforeBrush(); - brush(ctx, displayable, innerScope, i === len - 1); - displayable.innerAfterBrush(); - displayable.afterBrush && displayable.afterBrush(); - innerScope.prevEl = displayable; - } - for (var i_1 = 0, len_1 = temporalDisplayables.length; i_1 < len_1; i_1++) { - var displayable = temporalDisplayables[i_1]; - displayable.beforeBrush && displayable.beforeBrush(); - displayable.innerBeforeBrush(); - brush(ctx, displayable, innerScope, i_1 === len_1 - 1); - displayable.innerAfterBrush(); - displayable.afterBrush && displayable.afterBrush(); - innerScope.prevEl = displayable; - } - el.clearTemporalDisplayables(); - el.notClear = true; - ctx.restore(); - } - - var decalMap = new WeakMap(); - var decalCache = new LRU(100); - var decalKeys = ['symbol', 'symbolSize', 'symbolKeepAspect', 'color', 'backgroundColor', 'dashArrayX', 'dashArrayY', 'maxTileWidth', 'maxTileHeight']; - /** - * Create or update pattern image from decal options - * - * @param {InnerDecalObject | 'none'} decalObject decal options, 'none' if no decal - * @return {Pattern} pattern with generated image, null if no decal - */ - - function createOrUpdatePatternFromDecal(decalObject, api) { - if (decalObject === 'none') { - return null; - } - - var dpr = api.getDevicePixelRatio(); - var zr = api.getZr(); - var isSVG = zr.painter.type === 'svg'; - - if (decalObject.dirty) { - decalMap["delete"](decalObject); - } - - var oldPattern = decalMap.get(decalObject); - - if (oldPattern) { - return oldPattern; - } - - var decalOpt = defaults(decalObject, { - symbol: 'rect', - symbolSize: 1, - symbolKeepAspect: true, - color: 'rgba(0, 0, 0, 0.2)', - backgroundColor: null, - dashArrayX: 5, - dashArrayY: 5, - rotation: 0, - maxTileWidth: 512, - maxTileHeight: 512 - }); - - if (decalOpt.backgroundColor === 'none') { - decalOpt.backgroundColor = null; - } - - var pattern = { - repeat: 'repeat' - }; - setPatternnSource(pattern); - pattern.rotation = decalOpt.rotation; - pattern.scaleX = pattern.scaleY = isSVG ? 1 : 1 / dpr; - decalMap.set(decalObject, pattern); - decalObject.dirty = false; - return pattern; - - function setPatternnSource(pattern) { - var keys = [dpr]; - var isValidKey = true; - - for (var i = 0; i < decalKeys.length; ++i) { - var value = decalOpt[decalKeys[i]]; - - if (value != null && !isArray(value) && !isString(value) && !isNumber(value) && typeof value !== 'boolean') { - isValidKey = false; - break; - } - - keys.push(value); - } - - var cacheKey; - - if (isValidKey) { - cacheKey = keys.join(',') + (isSVG ? '-svg' : ''); - var cache = decalCache.get(cacheKey); - - if (cache) { - isSVG ? pattern.svgElement = cache : pattern.image = cache; - } - } - - var dashArrayX = normalizeDashArrayX(decalOpt.dashArrayX); - var dashArrayY = normalizeDashArrayY(decalOpt.dashArrayY); - var symbolArray = normalizeSymbolArray(decalOpt.symbol); - var lineBlockLengthsX = getLineBlockLengthX(dashArrayX); - var lineBlockLengthY = getLineBlockLengthY(dashArrayY); - var canvas = !isSVG && platformApi.createCanvas(); - var svgRoot = isSVG && { - tag: 'g', - attrs: {}, - key: 'dcl', - children: [] - }; - var pSize = getPatternSize(); - var ctx; - - if (canvas) { - canvas.width = pSize.width * dpr; - canvas.height = pSize.height * dpr; - ctx = canvas.getContext('2d'); - } - - brushDecal(); - - if (isValidKey) { - decalCache.put(cacheKey, canvas || svgRoot); - } - - pattern.image = canvas; - pattern.svgElement = svgRoot; - pattern.svgWidth = pSize.width; - pattern.svgHeight = pSize.height; - /** - * Get minimum length that can make a repeatable pattern. - * - * @return {Object} pattern width and height - */ - - function getPatternSize() { - /** - * For example, if dash is [[3, 2], [2, 1]] for X, it looks like - * |--- --- --- --- --- ... - * |-- -- -- -- -- -- -- -- ... - * |--- --- --- --- --- ... - * |-- -- -- -- -- -- -- -- ... - * So the minimum length of X is 15, - * which is the least common multiple of `3 + 2` and `2 + 1` - * |--- --- --- |--- --- ... - * |-- -- -- -- -- |-- -- -- ... - */ - var width = 1; - - for (var i = 0, xlen = lineBlockLengthsX.length; i < xlen; ++i) { - width = getLeastCommonMultiple(width, lineBlockLengthsX[i]); - } - - var symbolRepeats = 1; - - for (var i = 0, xlen = symbolArray.length; i < xlen; ++i) { - symbolRepeats = getLeastCommonMultiple(symbolRepeats, symbolArray[i].length); - } - - width *= symbolRepeats; - var height = lineBlockLengthY * lineBlockLengthsX.length * symbolArray.length; - - if ("development" !== 'production') { - var warn = function (attrName) { - /* eslint-disable-next-line */ - console.warn("Calculated decal size is greater than " + attrName + " due to decal option settings so " + attrName + " is used for the decal size. Please consider changing the decal option to make a smaller decal or set " + attrName + " to be larger to avoid incontinuity."); - }; - - if (width > decalOpt.maxTileWidth) { - warn('maxTileWidth'); - } - - if (height > decalOpt.maxTileHeight) { - warn('maxTileHeight'); - } - } - - return { - width: Math.max(1, Math.min(width, decalOpt.maxTileWidth)), - height: Math.max(1, Math.min(height, decalOpt.maxTileHeight)) - }; - } - - function brushDecal() { - if (ctx) { - ctx.clearRect(0, 0, canvas.width, canvas.height); - - if (decalOpt.backgroundColor) { - ctx.fillStyle = decalOpt.backgroundColor; - ctx.fillRect(0, 0, canvas.width, canvas.height); - } - } - - var ySum = 0; - - for (var i = 0; i < dashArrayY.length; ++i) { - ySum += dashArrayY[i]; - } - - if (ySum <= 0) { - // dashArrayY is 0, draw nothing - return; - } - - var y = -lineBlockLengthY; - var yId = 0; - var yIdTotal = 0; - var xId0 = 0; - - while (y < pSize.height) { - if (yId % 2 === 0) { - var symbolYId = yIdTotal / 2 % symbolArray.length; - var x = 0; - var xId1 = 0; - var xId1Total = 0; - - while (x < pSize.width * 2) { - var xSum = 0; - - for (var i = 0; i < dashArrayX[xId0].length; ++i) { - xSum += dashArrayX[xId0][i]; - } - - if (xSum <= 0) { - // Skip empty line - break; - } // E.g., [15, 5, 20, 5] draws only for 15 and 20 - - - if (xId1 % 2 === 0) { - var size = (1 - decalOpt.symbolSize) * 0.5; - var left = x + dashArrayX[xId0][xId1] * size; - var top_1 = y + dashArrayY[yId] * size; - var width = dashArrayX[xId0][xId1] * decalOpt.symbolSize; - var height = dashArrayY[yId] * decalOpt.symbolSize; - var symbolXId = xId1Total / 2 % symbolArray[symbolYId].length; - brushSymbol(left, top_1, width, height, symbolArray[symbolYId][symbolXId]); - } - - x += dashArrayX[xId0][xId1]; - ++xId1Total; - ++xId1; - - if (xId1 === dashArrayX[xId0].length) { - xId1 = 0; - } - } - - ++xId0; - - if (xId0 === dashArrayX.length) { - xId0 = 0; - } - } - - y += dashArrayY[yId]; - ++yIdTotal; - ++yId; - - if (yId === dashArrayY.length) { - yId = 0; - } - } - - function brushSymbol(x, y, width, height, symbolType) { - var scale = isSVG ? 1 : dpr; - var symbol = createSymbol(symbolType, x * scale, y * scale, width * scale, height * scale, decalOpt.color, decalOpt.symbolKeepAspect); - - if (isSVG) { - var symbolVNode = zr.painter.renderOneToVNode(symbol); - - if (symbolVNode) { - svgRoot.children.push(symbolVNode); - } - } else { - // Paint to canvas for all other renderers. - brushSingle(ctx, symbol); - } - } - } - } - } - /** - * Convert symbol array into normalized array - * - * @param {string | (string | string[])[]} symbol symbol input - * @return {string[][]} normolized symbol array - */ - - function normalizeSymbolArray(symbol) { - if (!symbol || symbol.length === 0) { - return [['rect']]; - } - - if (isString(symbol)) { - return [[symbol]]; - } - - var isAllString = true; - - for (var i = 0; i < symbol.length; ++i) { - if (!isString(symbol[i])) { - isAllString = false; - break; - } - } - - if (isAllString) { - return normalizeSymbolArray([symbol]); - } - - var result = []; - - for (var i = 0; i < symbol.length; ++i) { - if (isString(symbol[i])) { - result.push([symbol[i]]); - } else { - result.push(symbol[i]); - } - } - - return result; - } - /** - * Convert dash input into dashArray - * - * @param {DecalDashArrayX} dash dash input - * @return {number[][]} normolized dash array - */ - - - function normalizeDashArrayX(dash) { - if (!dash || dash.length === 0) { - return [[0, 0]]; - } - - if (isNumber(dash)) { - var dashValue = Math.ceil(dash); - return [[dashValue, dashValue]]; - } - /** - * [20, 5] should be normalized into [[20, 5]], - * while [20, [5, 10]] should be normalized into [[20, 20], [5, 10]] - */ - - - var isAllNumber = true; - - for (var i = 0; i < dash.length; ++i) { - if (!isNumber(dash[i])) { - isAllNumber = false; - break; - } - } - - if (isAllNumber) { - return normalizeDashArrayX([dash]); - } - - var result = []; - - for (var i = 0; i < dash.length; ++i) { - if (isNumber(dash[i])) { - var dashValue = Math.ceil(dash[i]); - result.push([dashValue, dashValue]); - } else { - var dashValue = map(dash[i], function (n) { - return Math.ceil(n); - }); - - if (dashValue.length % 2 === 1) { - // [4, 2, 1] means |---- - -- |---- - -- | - // so normalize it to be [4, 2, 1, 4, 2, 1] - result.push(dashValue.concat(dashValue)); - } else { - result.push(dashValue); - } - } - } - - return result; - } - /** - * Convert dash input into dashArray - * - * @param {DecalDashArrayY} dash dash input - * @return {number[]} normolized dash array - */ - - - function normalizeDashArrayY(dash) { - if (!dash || typeof dash === 'object' && dash.length === 0) { - return [0, 0]; - } - - if (isNumber(dash)) { - var dashValue_1 = Math.ceil(dash); - return [dashValue_1, dashValue_1]; - } - - var dashValue = map(dash, function (n) { - return Math.ceil(n); - }); - return dash.length % 2 ? dashValue.concat(dashValue) : dashValue; - } - /** - * Get block length of each line. A block is the length of dash line and space. - * For example, a line with [4, 1] has a dash line of 4 and a space of 1 after - * that, so the block length of this line is 5. - * - * @param {number[][]} dash dash array of X or Y - * @return {number[]} block length of each line - */ - - - function getLineBlockLengthX(dash) { - return map(dash, function (line) { - return getLineBlockLengthY(line); - }); - } - - function getLineBlockLengthY(dash) { - var blockLength = 0; - - for (var i = 0; i < dash.length; ++i) { - blockLength += dash[i]; - } - - if (dash.length % 2 === 1) { - // [4, 2, 1] means |---- - -- |---- - -- | - // So total length is (4 + 2 + 1) * 2 - return blockLength * 2; - } - - return blockLength; - } - - function decalVisual(ecModel, api) { - ecModel.eachRawSeries(function (seriesModel) { - if (ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - var data = seriesModel.getData(); - - if (data.hasItemVisual()) { - data.each(function (idx) { - var decal = data.getItemVisual(idx, 'decal'); - - if (decal) { - var itemStyle = data.ensureUniqueItemVisual(idx, 'style'); - itemStyle.decal = createOrUpdatePatternFromDecal(decal, api); - } - }); - } - - var decal = data.getVisual('decal'); - - if (decal) { - var style = data.getVisual('style'); - style.decal = createOrUpdatePatternFromDecal(decal, api); - } - }); - } - - var lifecycle = new Eventful(); - - // The implementations will be registered when installing the component. - // Avoid these code being bundled to the core module. - - var implsStore = {}; // TODO Type - - function registerImpl(name, impl) { - if ("development" !== 'production') { - if (implsStore[name]) { - error("Already has an implementation of " + name + "."); - } - } - - implsStore[name] = impl; - } - function getImpl(name) { - if ("development" !== 'production') { - if (!implsStore[name]) { - error("Implementation of " + name + " doesn't exists."); - } - } - - return implsStore[name]; - } - - var version$1 = '5.4.2'; - var dependencies = { - zrender: '5.4.3' - }; - var TEST_FRAME_REMAIN_TIME = 1; - var PRIORITY_PROCESSOR_SERIES_FILTER = 800; // Some data processors depends on the stack result dimension (to calculate data extent). - // So data stack stage should be in front of data processing stage. - - var PRIORITY_PROCESSOR_DATASTACK = 900; // "Data filter" will block the stream, so it should be - // put at the beginning of data processing. - - var PRIORITY_PROCESSOR_FILTER = 1000; - var PRIORITY_PROCESSOR_DEFAULT = 2000; - var PRIORITY_PROCESSOR_STATISTIC = 5000; - var PRIORITY_VISUAL_LAYOUT = 1000; - var PRIORITY_VISUAL_PROGRESSIVE_LAYOUT = 1100; - var PRIORITY_VISUAL_GLOBAL = 2000; - var PRIORITY_VISUAL_CHART = 3000; - var PRIORITY_VISUAL_COMPONENT = 4000; // Visual property in data. Greater than `PRIORITY_VISUAL_COMPONENT` to enable to - // overwrite the viusal result of component (like `visualMap`) - // using data item specific setting (like itemStyle.xxx on data item) - - var PRIORITY_VISUAL_CHART_DATA_CUSTOM = 4500; // Greater than `PRIORITY_VISUAL_CHART_DATA_CUSTOM` to enable to layout based on - // visual result like `symbolSize`. - - var PRIORITY_VISUAL_POST_CHART_LAYOUT = 4600; - var PRIORITY_VISUAL_BRUSH = 5000; - var PRIORITY_VISUAL_ARIA = 6000; - var PRIORITY_VISUAL_DECAL = 7000; - var PRIORITY = { - PROCESSOR: { - FILTER: PRIORITY_PROCESSOR_FILTER, - SERIES_FILTER: PRIORITY_PROCESSOR_SERIES_FILTER, - STATISTIC: PRIORITY_PROCESSOR_STATISTIC - }, - VISUAL: { - LAYOUT: PRIORITY_VISUAL_LAYOUT, - PROGRESSIVE_LAYOUT: PRIORITY_VISUAL_PROGRESSIVE_LAYOUT, - GLOBAL: PRIORITY_VISUAL_GLOBAL, - CHART: PRIORITY_VISUAL_CHART, - POST_CHART_LAYOUT: PRIORITY_VISUAL_POST_CHART_LAYOUT, - COMPONENT: PRIORITY_VISUAL_COMPONENT, - BRUSH: PRIORITY_VISUAL_BRUSH, - CHART_ITEM: PRIORITY_VISUAL_CHART_DATA_CUSTOM, - ARIA: PRIORITY_VISUAL_ARIA, - DECAL: PRIORITY_VISUAL_DECAL - } - }; // Main process have three entries: `setOption`, `dispatchAction` and `resize`, - // where they must not be invoked nestedly, except the only case: invoke - // dispatchAction with updateMethod "none" in main process. - // This flag is used to carry out this rule. - // All events will be triggered out side main process (i.e. when !this[IN_MAIN_PROCESS]). - - var IN_MAIN_PROCESS_KEY = '__flagInMainProcess'; - var PENDING_UPDATE = '__pendingUpdate'; - var STATUS_NEEDS_UPDATE_KEY = '__needsUpdateStatus'; - var ACTION_REG = /^[a-zA-Z0-9_]+$/; - var CONNECT_STATUS_KEY = '__connectUpdateStatus'; - var CONNECT_STATUS_PENDING = 0; - var CONNECT_STATUS_UPDATING = 1; - var CONNECT_STATUS_UPDATED = 2; - - function createRegisterEventWithLowercaseECharts(method) { - return function () { - var args = []; - - for (var _i = 0; _i < arguments.length; _i++) { - args[_i] = arguments[_i]; - } - - if (this.isDisposed()) { - disposedWarning(this.id); - return; - } - - return toLowercaseNameAndCallEventful(this, method, args); - }; - } - - function createRegisterEventWithLowercaseMessageCenter(method) { - return function () { - var args = []; - - for (var _i = 0; _i < arguments.length; _i++) { - args[_i] = arguments[_i]; - } - - return toLowercaseNameAndCallEventful(this, method, args); - }; - } - - function toLowercaseNameAndCallEventful(host, method, args) { - // `args[0]` is event name. Event name is all lowercase. - args[0] = args[0] && args[0].toLowerCase(); - return Eventful.prototype[method].apply(host, args); - } - - var MessageCenter = - /** @class */ - function (_super) { - __extends(MessageCenter, _super); - - function MessageCenter() { - return _super !== null && _super.apply(this, arguments) || this; - } - - return MessageCenter; - }(Eventful); - - var messageCenterProto = MessageCenter.prototype; - messageCenterProto.on = createRegisterEventWithLowercaseMessageCenter('on'); - messageCenterProto.off = createRegisterEventWithLowercaseMessageCenter('off'); // --------------------------------------- - // Internal method names for class ECharts - // --------------------------------------- - - var prepare; - var prepareView; - var updateDirectly; - var updateMethods; - var doConvertPixel; - var updateStreamModes; - var doDispatchAction; - var flushPendingActions; - var triggerUpdatedEvent; - var bindRenderedEvent; - var bindMouseEvent; - var render; - var renderComponents; - var renderSeries; - var createExtensionAPI; - var enableConnect; - var markStatusToUpdate; - var applyChangedStates; - - var ECharts = - /** @class */ - function (_super) { - __extends(ECharts, _super); - - function ECharts(dom, // Theme name or themeOption. - theme, opts) { - var _this = _super.call(this, new ECEventProcessor()) || this; - - _this._chartsViews = []; - _this._chartsMap = {}; - _this._componentsViews = []; - _this._componentsMap = {}; // Can't dispatch action during rendering procedure - - _this._pendingActions = []; - opts = opts || {}; // Get theme by name - - if (isString(theme)) { - theme = themeStorage[theme]; - } - - _this._dom = dom; - var defaultRenderer = 'canvas'; - var defaultCoarsePointer = 'auto'; - var defaultUseDirtyRect = false; - - if ("development" !== 'production') { - var root = - /* eslint-disable-next-line */ - env.hasGlobalWindow ? window : global; - defaultRenderer = root.__ECHARTS__DEFAULT__RENDERER__ || defaultRenderer; - defaultCoarsePointer = retrieve2(root.__ECHARTS__DEFAULT__COARSE_POINTER, defaultCoarsePointer); - var devUseDirtyRect = root.__ECHARTS__DEFAULT__USE_DIRTY_RECT__; - defaultUseDirtyRect = devUseDirtyRect == null ? defaultUseDirtyRect : devUseDirtyRect; - } - - var zr = _this._zr = init(dom, { - renderer: opts.renderer || defaultRenderer, - devicePixelRatio: opts.devicePixelRatio, - width: opts.width, - height: opts.height, - ssr: opts.ssr, - useDirtyRect: retrieve2(opts.useDirtyRect, defaultUseDirtyRect), - useCoarsePointer: retrieve2(opts.useCoarsePointer, defaultCoarsePointer), - pointerSize: opts.pointerSize - }); - _this._ssr = opts.ssr; // Expect 60 fps. - - _this._throttledZrFlush = throttle(bind(zr.flush, zr), 17); - theme = clone(theme); - theme && globalBackwardCompat(theme, true); - _this._theme = theme; - _this._locale = createLocaleObject(opts.locale || SYSTEM_LANG); - _this._coordSysMgr = new CoordinateSystemManager(); - var api = _this._api = createExtensionAPI(_this); // Sort on demand - - function prioritySortFunc(a, b) { - return a.__prio - b.__prio; - } - - sort(visualFuncs, prioritySortFunc); - sort(dataProcessorFuncs, prioritySortFunc); - _this._scheduler = new Scheduler(_this, api, dataProcessorFuncs, visualFuncs); - _this._messageCenter = new MessageCenter(); // Init mouse events - - _this._initEvents(); // In case some people write `window.onresize = chart.resize` - - - _this.resize = bind(_this.resize, _this); - zr.animation.on('frame', _this._onframe, _this); - bindRenderedEvent(zr, _this); - bindMouseEvent(zr, _this); // ECharts instance can be used as value. - - setAsPrimitive(_this); - return _this; - } - - ECharts.prototype._onframe = function () { - if (this._disposed) { - return; - } - - applyChangedStates(this); - var scheduler = this._scheduler; // Lazy update - - if (this[PENDING_UPDATE]) { - var silent = this[PENDING_UPDATE].silent; - this[IN_MAIN_PROCESS_KEY] = true; - - try { - prepare(this); - updateMethods.update.call(this, null, this[PENDING_UPDATE].updateParams); - } catch (e) { - this[IN_MAIN_PROCESS_KEY] = false; - this[PENDING_UPDATE] = null; - throw e; - } // At present, in each frame, zrender performs: - // (1) animation step forward. - // (2) trigger('frame') (where this `_onframe` is called) - // (3) zrender flush (render). - // If we do nothing here, since we use `setToFinal: true`, the step (3) above - // will render the final state of the elements before the real animation started. - - - this._zr.flush(); - - this[IN_MAIN_PROCESS_KEY] = false; - this[PENDING_UPDATE] = null; - flushPendingActions.call(this, silent); - triggerUpdatedEvent.call(this, silent); - } // Avoid do both lazy update and progress in one frame. - else if (scheduler.unfinished) { - // Stream progress. - var remainTime = TEST_FRAME_REMAIN_TIME; - var ecModel = this._model; - var api = this._api; - scheduler.unfinished = false; - - do { - var startTime = +new Date(); - scheduler.performSeriesTasks(ecModel); // Currently dataProcessorFuncs do not check threshold. - - scheduler.performDataProcessorTasks(ecModel); - updateStreamModes(this, ecModel); // Do not update coordinate system here. Because that coord system update in - // each frame is not a good user experience. So we follow the rule that - // the extent of the coordinate system is determined in the first frame (the - // frame is executed immediately after task reset. - // this._coordSysMgr.update(ecModel, api); - // console.log('--- ec frame visual ---', remainTime); - - scheduler.performVisualTasks(ecModel); - renderSeries(this, this._model, api, 'remain', {}); - remainTime -= +new Date() - startTime; - } while (remainTime > 0 && scheduler.unfinished); // Call flush explicitly for trigger finished event. - - - if (!scheduler.unfinished) { - this._zr.flush(); - } // Else, zr flushing be ensue within the same frame, - // because zr flushing is after onframe event. - - } - }; - - ECharts.prototype.getDom = function () { - return this._dom; - }; - - ECharts.prototype.getId = function () { - return this.id; - }; - - ECharts.prototype.getZr = function () { - return this._zr; - }; - - ECharts.prototype.isSSR = function () { - return this._ssr; - }; - /* eslint-disable-next-line */ - - - ECharts.prototype.setOption = function (option, notMerge, lazyUpdate) { - if (this[IN_MAIN_PROCESS_KEY]) { - if ("development" !== 'production') { - error('`setOption` should not be called during main process.'); - } - - return; - } - - if (this._disposed) { - disposedWarning(this.id); - return; - } - - var silent; - var replaceMerge; - var transitionOpt; - - if (isObject(notMerge)) { - lazyUpdate = notMerge.lazyUpdate; - silent = notMerge.silent; - replaceMerge = notMerge.replaceMerge; - transitionOpt = notMerge.transition; - notMerge = notMerge.notMerge; - } - - this[IN_MAIN_PROCESS_KEY] = true; - - if (!this._model || notMerge) { - var optionManager = new OptionManager(this._api); - var theme = this._theme; - var ecModel = this._model = new GlobalModel(); - ecModel.scheduler = this._scheduler; - ecModel.ssr = this._ssr; - ecModel.init(null, null, null, theme, this._locale, optionManager); - } - - this._model.setOption(option, { - replaceMerge: replaceMerge - }, optionPreprocessorFuncs); - - var updateParams = { - seriesTransition: transitionOpt, - optionChanged: true - }; - - if (lazyUpdate) { - this[PENDING_UPDATE] = { - silent: silent, - updateParams: updateParams - }; - this[IN_MAIN_PROCESS_KEY] = false; // `setOption(option, {lazyMode: true})` may be called when zrender has been slept. - // It should wake it up to make sure zrender start to render at the next frame. - - this.getZr().wakeUp(); - } else { - try { - prepare(this); - updateMethods.update.call(this, null, updateParams); - } catch (e) { - this[PENDING_UPDATE] = null; - this[IN_MAIN_PROCESS_KEY] = false; - throw e; - } // Ensure zr refresh sychronously, and then pixel in canvas can be - // fetched after `setOption`. - - - if (!this._ssr) { - // not use flush when using ssr mode. - this._zr.flush(); - } - - this[PENDING_UPDATE] = null; - this[IN_MAIN_PROCESS_KEY] = false; - flushPendingActions.call(this, silent); - triggerUpdatedEvent.call(this, silent); - } - }; - /** - * @deprecated - */ - - - ECharts.prototype.setTheme = function () { - deprecateLog('ECharts#setTheme() is DEPRECATED in ECharts 3.0'); - }; // We don't want developers to use getModel directly. - - - ECharts.prototype.getModel = function () { - return this._model; - }; - - ECharts.prototype.getOption = function () { - return this._model && this._model.getOption(); - }; - - ECharts.prototype.getWidth = function () { - return this._zr.getWidth(); - }; - - ECharts.prototype.getHeight = function () { - return this._zr.getHeight(); - }; - - ECharts.prototype.getDevicePixelRatio = function () { - return this._zr.painter.dpr - /* eslint-disable-next-line */ - || env.hasGlobalWindow && window.devicePixelRatio || 1; - }; - /** - * Get canvas which has all thing rendered - * @deprecated Use renderToCanvas instead. - */ - - - ECharts.prototype.getRenderedCanvas = function (opts) { - if ("development" !== 'production') { - deprecateReplaceLog('getRenderedCanvas', 'renderToCanvas'); - } - - return this.renderToCanvas(opts); - }; - - ECharts.prototype.renderToCanvas = function (opts) { - opts = opts || {}; - var painter = this._zr.painter; - - if ("development" !== 'production') { - if (painter.type !== 'canvas') { - throw new Error('renderToCanvas can only be used in the canvas renderer.'); - } - } - - return painter.getRenderedCanvas({ - backgroundColor: opts.backgroundColor || this._model.get('backgroundColor'), - pixelRatio: opts.pixelRatio || this.getDevicePixelRatio() - }); - }; - - ECharts.prototype.renderToSVGString = function (opts) { - opts = opts || {}; - var painter = this._zr.painter; - - if ("development" !== 'production') { - if (painter.type !== 'svg') { - throw new Error('renderToSVGString can only be used in the svg renderer.'); - } - } - - return painter.renderToString({ - useViewBox: opts.useViewBox - }); - }; - /** - * Get svg data url - */ - - - ECharts.prototype.getSvgDataURL = function () { - if (!env.svgSupported) { - return; - } - - var zr = this._zr; - var list = zr.storage.getDisplayList(); // Stop animations - - each(list, function (el) { - el.stopAnimation(null, true); - }); - return zr.painter.toDataURL(); - }; - - ECharts.prototype.getDataURL = function (opts) { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - opts = opts || {}; - var excludeComponents = opts.excludeComponents; - var ecModel = this._model; - var excludesComponentViews = []; - var self = this; - each(excludeComponents, function (componentType) { - ecModel.eachComponent({ - mainType: componentType - }, function (component) { - var view = self._componentsMap[component.__viewId]; - - if (!view.group.ignore) { - excludesComponentViews.push(view); - view.group.ignore = true; - } - }); - }); - var url = this._zr.painter.getType() === 'svg' ? this.getSvgDataURL() : this.renderToCanvas(opts).toDataURL('image/' + (opts && opts.type || 'png')); - each(excludesComponentViews, function (view) { - view.group.ignore = false; - }); - return url; - }; - - ECharts.prototype.getConnectedDataURL = function (opts) { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - var isSvg = opts.type === 'svg'; - var groupId = this.group; - var mathMin = Math.min; - var mathMax = Math.max; - var MAX_NUMBER = Infinity; - - if (connectedGroups[groupId]) { - var left_1 = MAX_NUMBER; - var top_1 = MAX_NUMBER; - var right_1 = -MAX_NUMBER; - var bottom_1 = -MAX_NUMBER; - var canvasList_1 = []; - var dpr_1 = opts && opts.pixelRatio || this.getDevicePixelRatio(); - each(instances$1, function (chart, id) { - if (chart.group === groupId) { - var canvas = isSvg ? chart.getZr().painter.getSvgDom().innerHTML : chart.renderToCanvas(clone(opts)); - var boundingRect = chart.getDom().getBoundingClientRect(); - left_1 = mathMin(boundingRect.left, left_1); - top_1 = mathMin(boundingRect.top, top_1); - right_1 = mathMax(boundingRect.right, right_1); - bottom_1 = mathMax(boundingRect.bottom, bottom_1); - canvasList_1.push({ - dom: canvas, - left: boundingRect.left, - top: boundingRect.top - }); - } - }); - left_1 *= dpr_1; - top_1 *= dpr_1; - right_1 *= dpr_1; - bottom_1 *= dpr_1; - var width = right_1 - left_1; - var height = bottom_1 - top_1; - var targetCanvas = platformApi.createCanvas(); - var zr_1 = init(targetCanvas, { - renderer: isSvg ? 'svg' : 'canvas' - }); - zr_1.resize({ - width: width, - height: height - }); - - if (isSvg) { - var content_1 = ''; - each(canvasList_1, function (item) { - var x = item.left - left_1; - var y = item.top - top_1; - content_1 += '<g transform="translate(' + x + ',' + y + ')">' + item.dom + '</g>'; - }); - zr_1.painter.getSvgRoot().innerHTML = content_1; - - if (opts.connectedBackgroundColor) { - zr_1.painter.setBackgroundColor(opts.connectedBackgroundColor); - } - - zr_1.refreshImmediately(); - return zr_1.painter.toDataURL(); - } else { - // Background between the charts - if (opts.connectedBackgroundColor) { - zr_1.add(new Rect({ - shape: { - x: 0, - y: 0, - width: width, - height: height - }, - style: { - fill: opts.connectedBackgroundColor - } - })); - } - - each(canvasList_1, function (item) { - var img = new ZRImage({ - style: { - x: item.left * dpr_1 - left_1, - y: item.top * dpr_1 - top_1, - image: item.dom - } - }); - zr_1.add(img); - }); - zr_1.refreshImmediately(); - return targetCanvas.toDataURL('image/' + (opts && opts.type || 'png')); - } - } else { - return this.getDataURL(opts); - } - }; - - ECharts.prototype.convertToPixel = function (finder, value) { - return doConvertPixel(this, 'convertToPixel', finder, value); - }; - - ECharts.prototype.convertFromPixel = function (finder, value) { - return doConvertPixel(this, 'convertFromPixel', finder, value); - }; - /** - * Is the specified coordinate systems or components contain the given pixel point. - * @param {Array|number} value - * @return {boolean} result - */ - - - ECharts.prototype.containPixel = function (finder, value) { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - var ecModel = this._model; - var result; - var findResult = parseFinder(ecModel, finder); - each(findResult, function (models, key) { - key.indexOf('Models') >= 0 && each(models, function (model) { - var coordSys = model.coordinateSystem; - - if (coordSys && coordSys.containPoint) { - result = result || !!coordSys.containPoint(value); - } else if (key === 'seriesModels') { - var view = this._chartsMap[model.__viewId]; - - if (view && view.containPoint) { - result = result || view.containPoint(value, model); - } else { - if ("development" !== 'production') { - warn(key + ': ' + (view ? 'The found component do not support containPoint.' : 'No view mapping to the found component.')); - } - } - } else { - if ("development" !== 'production') { - warn(key + ': containPoint is not supported'); - } - } - }, this); - }, this); - return !!result; - }; - /** - * Get visual from series or data. - * @param finder - * If string, e.g., 'series', means {seriesIndex: 0}. - * If Object, could contain some of these properties below: - * { - * seriesIndex / seriesId / seriesName, - * dataIndex / dataIndexInside - * } - * If dataIndex is not specified, series visual will be fetched, - * but not data item visual. - * If all of seriesIndex, seriesId, seriesName are not specified, - * visual will be fetched from first series. - * @param visualType 'color', 'symbol', 'symbolSize' - */ - - - ECharts.prototype.getVisual = function (finder, visualType) { - var ecModel = this._model; - var parsedFinder = parseFinder(ecModel, finder, { - defaultMainType: 'series' - }); - var seriesModel = parsedFinder.seriesModel; - - if ("development" !== 'production') { - if (!seriesModel) { - warn('There is no specified series model'); - } - } - - var data = seriesModel.getData(); - var dataIndexInside = parsedFinder.hasOwnProperty('dataIndexInside') ? parsedFinder.dataIndexInside : parsedFinder.hasOwnProperty('dataIndex') ? data.indexOfRawIndex(parsedFinder.dataIndex) : null; - return dataIndexInside != null ? getItemVisualFromData(data, dataIndexInside, visualType) : getVisualFromData(data, visualType); - }; - /** - * Get view of corresponding component model - */ - - - ECharts.prototype.getViewOfComponentModel = function (componentModel) { - return this._componentsMap[componentModel.__viewId]; - }; - /** - * Get view of corresponding series model - */ - - - ECharts.prototype.getViewOfSeriesModel = function (seriesModel) { - return this._chartsMap[seriesModel.__viewId]; - }; - - ECharts.prototype._initEvents = function () { - var _this = this; - - each(MOUSE_EVENT_NAMES, function (eveName) { - var handler = function (e) { - var ecModel = _this.getModel(); - - var el = e.target; - var params; - var isGlobalOut = eveName === 'globalout'; // no e.target when 'globalout'. - - if (isGlobalOut) { - params = {}; - } else { - el && findEventDispatcher(el, function (parent) { - var ecData = getECData(parent); - - if (ecData && ecData.dataIndex != null) { - var dataModel = ecData.dataModel || ecModel.getSeriesByIndex(ecData.seriesIndex); - params = dataModel && dataModel.getDataParams(ecData.dataIndex, ecData.dataType) || {}; - return true; - } // If element has custom eventData of components - else if (ecData.eventData) { - params = extend({}, ecData.eventData); - return true; - } - }, true); - } // Contract: if params prepared in mouse event, - // these properties must be specified: - // { - // componentType: string (component main type) - // componentIndex: number - // } - // Otherwise event query can not work. - - - if (params) { - var componentType = params.componentType; - var componentIndex = params.componentIndex; // Special handling for historic reason: when trigger by - // markLine/markPoint/markArea, the componentType is - // 'markLine'/'markPoint'/'markArea', but we should better - // enable them to be queried by seriesIndex, since their - // option is set in each series. - - if (componentType === 'markLine' || componentType === 'markPoint' || componentType === 'markArea') { - componentType = 'series'; - componentIndex = params.seriesIndex; - } - - var model = componentType && componentIndex != null && ecModel.getComponent(componentType, componentIndex); - var view = model && _this[model.mainType === 'series' ? '_chartsMap' : '_componentsMap'][model.__viewId]; - - if ("development" !== 'production') { - // `event.componentType` and `event[componentTpype + 'Index']` must not - // be missed, otherwise there is no way to distinguish source component. - // See `dataFormat.getDataParams`. - if (!isGlobalOut && !(model && view)) { - warn('model or view can not be found by params'); - } - } - - params.event = e; - params.type = eveName; - _this._$eventProcessor.eventInfo = { - targetEl: el, - packedEvent: params, - model: model, - view: view - }; - - _this.trigger(eveName, params); - } - }; // Consider that some component (like tooltip, brush, ...) - // register zr event handler, but user event handler might - // do anything, such as call `setOption` or `dispatchAction`, - // which probably update any of the content and probably - // cause problem if it is called previous other inner handlers. - - - handler.zrEventfulCallAtLast = true; - - _this._zr.on(eveName, handler, _this); - }); - each(eventActionMap, function (actionType, eventType) { - _this._messageCenter.on(eventType, function (event) { - this.trigger(eventType, event); - }, _this); - }); // Extra events - // TODO register? - - each(['selectchanged'], function (eventType) { - _this._messageCenter.on(eventType, function (event) { - this.trigger(eventType, event); - }, _this); - }); - handleLegacySelectEvents(this._messageCenter, this, this._api); - }; - - ECharts.prototype.isDisposed = function () { - return this._disposed; - }; - - ECharts.prototype.clear = function () { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - this.setOption({ - series: [] - }, true); - }; - - ECharts.prototype.dispose = function () { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - this._disposed = true; - var dom = this.getDom(); - - if (dom) { - setAttribute(this.getDom(), DOM_ATTRIBUTE_KEY, ''); - } - - var chart = this; - var api = chart._api; - var ecModel = chart._model; - each(chart._componentsViews, function (component) { - component.dispose(ecModel, api); - }); - each(chart._chartsViews, function (chart) { - chart.dispose(ecModel, api); - }); // Dispose after all views disposed - - chart._zr.dispose(); // Set properties to null. - // To reduce the memory cost in case the top code still holds this instance unexpectedly. - - - chart._dom = chart._model = chart._chartsMap = chart._componentsMap = chart._chartsViews = chart._componentsViews = chart._scheduler = chart._api = chart._zr = chart._throttledZrFlush = chart._theme = chart._coordSysMgr = chart._messageCenter = null; - delete instances$1[chart.id]; - }; - /** - * Resize the chart - */ - - - ECharts.prototype.resize = function (opts) { - if (this[IN_MAIN_PROCESS_KEY]) { - if ("development" !== 'production') { - error('`resize` should not be called during main process.'); - } - - return; - } - - if (this._disposed) { - disposedWarning(this.id); - return; - } - - this._zr.resize(opts); - - var ecModel = this._model; // Resize loading effect - - this._loadingFX && this._loadingFX.resize(); - - if (!ecModel) { - return; - } - - var needPrepare = ecModel.resetOption('media'); - var silent = opts && opts.silent; // There is some real cases that: - // chart.setOption(option, { lazyUpdate: true }); - // chart.resize(); - - if (this[PENDING_UPDATE]) { - if (silent == null) { - silent = this[PENDING_UPDATE].silent; - } - - needPrepare = true; - this[PENDING_UPDATE] = null; - } - - this[IN_MAIN_PROCESS_KEY] = true; - - try { - needPrepare && prepare(this); - updateMethods.update.call(this, { - type: 'resize', - animation: extend({ - // Disable animation - duration: 0 - }, opts && opts.animation) - }); - } catch (e) { - this[IN_MAIN_PROCESS_KEY] = false; - throw e; - } - - this[IN_MAIN_PROCESS_KEY] = false; - flushPendingActions.call(this, silent); - triggerUpdatedEvent.call(this, silent); - }; - - ECharts.prototype.showLoading = function (name, cfg) { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - if (isObject(name)) { - cfg = name; - name = ''; - } - - name = name || 'default'; - this.hideLoading(); - - if (!loadingEffects[name]) { - if ("development" !== 'production') { - warn('Loading effects ' + name + ' not exists.'); - } - - return; - } - - var el = loadingEffects[name](this._api, cfg); - var zr = this._zr; - this._loadingFX = el; - zr.add(el); - }; - /** - * Hide loading effect - */ - - - ECharts.prototype.hideLoading = function () { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - this._loadingFX && this._zr.remove(this._loadingFX); - this._loadingFX = null; - }; - - ECharts.prototype.makeActionFromEvent = function (eventObj) { - var payload = extend({}, eventObj); - payload.type = eventActionMap[eventObj.type]; - return payload; - }; - /** - * @param opt If pass boolean, means opt.silent - * @param opt.silent Default `false`. Whether trigger events. - * @param opt.flush Default `undefined`. - * true: Flush immediately, and then pixel in canvas can be fetched - * immediately. Caution: it might affect performance. - * false: Not flush. - * undefined: Auto decide whether perform flush. - */ - - - ECharts.prototype.dispatchAction = function (payload, opt) { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - if (!isObject(opt)) { - opt = { - silent: !!opt - }; - } - - if (!actions[payload.type]) { - return; - } // Avoid dispatch action before setOption. Especially in `connect`. - - - if (!this._model) { - return; - } // May dispatchAction in rendering procedure - - - if (this[IN_MAIN_PROCESS_KEY]) { - this._pendingActions.push(payload); - - return; - } - - var silent = opt.silent; - doDispatchAction.call(this, payload, silent); - var flush = opt.flush; - - if (flush) { - this._zr.flush(); - } else if (flush !== false && env.browser.weChat) { - // In WeChat embedded browser, `requestAnimationFrame` and `setInterval` - // hang when sliding page (on touch event), which cause that zr does not - // refresh until user interaction finished, which is not expected. - // But `dispatchAction` may be called too frequently when pan on touch - // screen, which impacts performance if do not throttle them. - this._throttledZrFlush(); - } - - flushPendingActions.call(this, silent); - triggerUpdatedEvent.call(this, silent); - }; - - ECharts.prototype.updateLabelLayout = function () { - lifecycle.trigger('series:layoutlabels', this._model, this._api, { - // Not adding series labels. - // TODO - updatedSeries: [] - }); - }; - - ECharts.prototype.appendData = function (params) { - if (this._disposed) { - disposedWarning(this.id); - return; - } - - var seriesIndex = params.seriesIndex; - var ecModel = this.getModel(); - var seriesModel = ecModel.getSeriesByIndex(seriesIndex); - - if ("development" !== 'production') { - assert(params.data && seriesModel); - } - - seriesModel.appendData(params); // Note: `appendData` does not support that update extent of coordinate - // system, util some scenario require that. In the expected usage of - // `appendData`, the initial extent of coordinate system should better - // be fixed by axis `min`/`max` setting or initial data, otherwise if - // the extent changed while `appendData`, the location of the painted - // graphic elements have to be changed, which make the usage of - // `appendData` meaningless. - - this._scheduler.unfinished = true; - this.getZr().wakeUp(); - }; // A work around for no `internal` modifier in ts yet but - // need to strictly hide private methods to JS users. - - - ECharts.internalField = function () { - prepare = function (ecIns) { - var scheduler = ecIns._scheduler; - scheduler.restorePipelines(ecIns._model); - scheduler.prepareStageTasks(); - prepareView(ecIns, true); - prepareView(ecIns, false); - scheduler.plan(); - }; - /** - * Prepare view instances of charts and components - */ - - - prepareView = function (ecIns, isComponent) { - var ecModel = ecIns._model; - var scheduler = ecIns._scheduler; - var viewList = isComponent ? ecIns._componentsViews : ecIns._chartsViews; - var viewMap = isComponent ? ecIns._componentsMap : ecIns._chartsMap; - var zr = ecIns._zr; - var api = ecIns._api; - - for (var i = 0; i < viewList.length; i++) { - viewList[i].__alive = false; - } - - isComponent ? ecModel.eachComponent(function (componentType, model) { - componentType !== 'series' && doPrepare(model); - }) : ecModel.eachSeries(doPrepare); - - function doPrepare(model) { - // By default view will be reused if possible for the case that `setOption` with "notMerge" - // mode and need to enable transition animation. (Usually, when they have the same id, or - // especially no id but have the same type & name & index. See the `model.id` generation - // rule in `makeIdAndName` and `viewId` generation rule here). - // But in `replaceMerge` mode, this feature should be able to disabled when it is clear that - // the new model has nothing to do with the old model. - var requireNewView = model.__requireNewView; // This command should not work twice. - - model.__requireNewView = false; // Consider: id same and type changed. - - var viewId = '_ec_' + model.id + '_' + model.type; - var view = !requireNewView && viewMap[viewId]; - - if (!view) { - var classType = parseClassType(model.type); - var Clazz = isComponent ? ComponentView.getClass(classType.main, classType.sub) : // FIXME:TS - // (ChartView as ChartViewConstructor).getClass('series', classType.sub) - // For backward compat, still support a chart type declared as only subType - // like "liquidfill", but recommend "series.liquidfill" - // But need a base class to make a type series. - ChartView.getClass(classType.sub); - - if ("development" !== 'production') { - assert(Clazz, classType.sub + ' does not exist.'); - } - - view = new Clazz(); - view.init(ecModel, api); - viewMap[viewId] = view; - viewList.push(view); - zr.add(view.group); - } - - model.__viewId = view.__id = viewId; - view.__alive = true; - view.__model = model; - view.group.__ecComponentInfo = { - mainType: model.mainType, - index: model.componentIndex - }; - !isComponent && scheduler.prepareView(view, model, ecModel, api); - } - - for (var i = 0; i < viewList.length;) { - var view = viewList[i]; - - if (!view.__alive) { - !isComponent && view.renderTask.dispose(); - zr.remove(view.group); - view.dispose(ecModel, api); - viewList.splice(i, 1); - - if (viewMap[view.__id] === view) { - delete viewMap[view.__id]; - } - - view.__id = view.group.__ecComponentInfo = null; - } else { - i++; - } - } - }; - - updateDirectly = function (ecIns, method, payload, mainType, subType) { - var ecModel = ecIns._model; - ecModel.setUpdatePayload(payload); // broadcast - - if (!mainType) { - // FIXME - // Chart will not be update directly here, except set dirty. - // But there is no such scenario now. - each([].concat(ecIns._componentsViews).concat(ecIns._chartsViews), callView); - return; - } - - var query = {}; - query[mainType + 'Id'] = payload[mainType + 'Id']; - query[mainType + 'Index'] = payload[mainType + 'Index']; - query[mainType + 'Name'] = payload[mainType + 'Name']; - var condition = { - mainType: mainType, - query: query - }; - subType && (condition.subType = subType); // subType may be '' by parseClassType; - - var excludeSeriesId = payload.excludeSeriesId; - var excludeSeriesIdMap; - - if (excludeSeriesId != null) { - excludeSeriesIdMap = createHashMap(); - each(normalizeToArray(excludeSeriesId), function (id) { - var modelId = convertOptionIdName(id, null); - - if (modelId != null) { - excludeSeriesIdMap.set(modelId, true); - } - }); - } // If dispatchAction before setOption, do nothing. - - - ecModel && ecModel.eachComponent(condition, function (model) { - var isExcluded = excludeSeriesIdMap && excludeSeriesIdMap.get(model.id) != null; - - if (isExcluded) { - return; - } - - if (isHighDownPayload(payload)) { - if (model instanceof SeriesModel) { - if (payload.type === HIGHLIGHT_ACTION_TYPE && !payload.notBlur && !model.get(['emphasis', 'disabled'])) { - blurSeriesFromHighlightPayload(model, payload, ecIns._api); - } - } else { - var _a = findComponentHighDownDispatchers(model.mainType, model.componentIndex, payload.name, ecIns._api), - focusSelf = _a.focusSelf, - dispatchers = _a.dispatchers; - - if (payload.type === HIGHLIGHT_ACTION_TYPE && focusSelf && !payload.notBlur) { - blurComponent(model.mainType, model.componentIndex, ecIns._api); - } // PENDING: - // Whether to put this "enter emphasis" code in `ComponentView`, - // which will be the same as `ChartView` but might be not necessary - // and will be far from this logic. - - - if (dispatchers) { - each(dispatchers, function (dispatcher) { - payload.type === HIGHLIGHT_ACTION_TYPE ? enterEmphasis(dispatcher) : leaveEmphasis(dispatcher); - }); - } - } - } else if (isSelectChangePayload(payload)) { - // TODO geo - if (model instanceof SeriesModel) { - toggleSelectionFromPayload(model, payload, ecIns._api); - updateSeriesElementSelection(model); - markStatusToUpdate(ecIns); - } - } - }, ecIns); - ecModel && ecModel.eachComponent(condition, function (model) { - var isExcluded = excludeSeriesIdMap && excludeSeriesIdMap.get(model.id) != null; - - if (isExcluded) { - return; - } - callView(ecIns[mainType === 'series' ? '_chartsMap' : '_componentsMap'][model.__viewId]); - }, ecIns); - - function callView(view) { - view && view.__alive && view[method] && view[method](view.__model, ecModel, ecIns._api, payload); - } - }; - - updateMethods = { - prepareAndUpdate: function (payload) { - prepare(this); - updateMethods.update.call(this, payload, { - // Needs to mark option changed if newOption is given. - // It's from MagicType. - // TODO If use a separate flag optionChanged in payload? - optionChanged: payload.newOption != null - }); - }, - update: function (payload, updateParams) { - var ecModel = this._model; - var api = this._api; - var zr = this._zr; - var coordSysMgr = this._coordSysMgr; - var scheduler = this._scheduler; // update before setOption - - if (!ecModel) { - return; - } - - ecModel.setUpdatePayload(payload); - scheduler.restoreData(ecModel, payload); - scheduler.performSeriesTasks(ecModel); // TODO - // Save total ecModel here for undo/redo (after restoring data and before processing data). - // Undo (restoration of total ecModel) can be carried out in 'action' or outside API call. - // Create new coordinate system each update - // In LineView may save the old coordinate system and use it to get the original point. - - coordSysMgr.create(ecModel, api); - scheduler.performDataProcessorTasks(ecModel, payload); // Current stream render is not supported in data process. So we can update - // stream modes after data processing, where the filtered data is used to - // determine whether to use progressive rendering. - - updateStreamModes(this, ecModel); // We update stream modes before coordinate system updated, then the modes info - // can be fetched when coord sys updating (consider the barGrid extent fix). But - // the drawback is the full coord info can not be fetched. Fortunately this full - // coord is not required in stream mode updater currently. - - coordSysMgr.update(ecModel, api); - clearColorPalette(ecModel); - scheduler.performVisualTasks(ecModel, payload); - render(this, ecModel, api, payload, updateParams); // Set background - - var backgroundColor = ecModel.get('backgroundColor') || 'transparent'; - var darkMode = ecModel.get('darkMode'); - zr.setBackgroundColor(backgroundColor); // Force set dark mode. - - if (darkMode != null && darkMode !== 'auto') { - zr.setDarkMode(darkMode); - } - - lifecycle.trigger('afterupdate', ecModel, api); - }, - updateTransform: function (payload) { - var _this = this; - - var ecModel = this._model; - var api = this._api; // update before setOption - - if (!ecModel) { - return; - } - - ecModel.setUpdatePayload(payload); // ChartView.markUpdateMethod(payload, 'updateTransform'); - - var componentDirtyList = []; - ecModel.eachComponent(function (componentType, componentModel) { - if (componentType === 'series') { - return; - } - - var componentView = _this.getViewOfComponentModel(componentModel); - - if (componentView && componentView.__alive) { - if (componentView.updateTransform) { - var result = componentView.updateTransform(componentModel, ecModel, api, payload); - result && result.update && componentDirtyList.push(componentView); - } else { - componentDirtyList.push(componentView); - } - } - }); - var seriesDirtyMap = createHashMap(); - ecModel.eachSeries(function (seriesModel) { - var chartView = _this._chartsMap[seriesModel.__viewId]; - - if (chartView.updateTransform) { - var result = chartView.updateTransform(seriesModel, ecModel, api, payload); - result && result.update && seriesDirtyMap.set(seriesModel.uid, 1); - } else { - seriesDirtyMap.set(seriesModel.uid, 1); - } - }); - clearColorPalette(ecModel); // Keep pipe to the exist pipeline because it depends on the render task of the full pipeline. - // this._scheduler.performVisualTasks(ecModel, payload, 'layout', true); - - this._scheduler.performVisualTasks(ecModel, payload, { - setDirty: true, - dirtyMap: seriesDirtyMap - }); // Currently, not call render of components. Geo render cost a lot. - // renderComponents(ecIns, ecModel, api, payload, componentDirtyList); - - - renderSeries(this, ecModel, api, payload, {}, seriesDirtyMap); - lifecycle.trigger('afterupdate', ecModel, api); - }, - updateView: function (payload) { - var ecModel = this._model; // update before setOption - - if (!ecModel) { - return; - } - - ecModel.setUpdatePayload(payload); - ChartView.markUpdateMethod(payload, 'updateView'); - clearColorPalette(ecModel); // Keep pipe to the exist pipeline because it depends on the render task of the full pipeline. - - this._scheduler.performVisualTasks(ecModel, payload, { - setDirty: true - }); - - render(this, ecModel, this._api, payload, {}); - lifecycle.trigger('afterupdate', ecModel, this._api); - }, - updateVisual: function (payload) { - // updateMethods.update.call(this, payload); - var _this = this; - - var ecModel = this._model; // update before setOption - - if (!ecModel) { - return; - } - - ecModel.setUpdatePayload(payload); // clear all visual - - ecModel.eachSeries(function (seriesModel) { - seriesModel.getData().clearAllVisual(); - }); // Perform visual - - ChartView.markUpdateMethod(payload, 'updateVisual'); - clearColorPalette(ecModel); // Keep pipe to the exist pipeline because it depends on the render task of the full pipeline. - - this._scheduler.performVisualTasks(ecModel, payload, { - visualType: 'visual', - setDirty: true - }); - - ecModel.eachComponent(function (componentType, componentModel) { - if (componentType !== 'series') { - var componentView = _this.getViewOfComponentModel(componentModel); - - componentView && componentView.__alive && componentView.updateVisual(componentModel, ecModel, _this._api, payload); - } - }); - ecModel.eachSeries(function (seriesModel) { - var chartView = _this._chartsMap[seriesModel.__viewId]; - chartView.updateVisual(seriesModel, ecModel, _this._api, payload); - }); - lifecycle.trigger('afterupdate', ecModel, this._api); - }, - updateLayout: function (payload) { - updateMethods.update.call(this, payload); - } - }; - - doConvertPixel = function (ecIns, methodName, finder, value) { - if (ecIns._disposed) { - disposedWarning(ecIns.id); - return; - } - - var ecModel = ecIns._model; - - var coordSysList = ecIns._coordSysMgr.getCoordinateSystems(); - - var result; - var parsedFinder = parseFinder(ecModel, finder); - - for (var i = 0; i < coordSysList.length; i++) { - var coordSys = coordSysList[i]; - - if (coordSys[methodName] && (result = coordSys[methodName](ecModel, parsedFinder, value)) != null) { - return result; - } - } - - if ("development" !== 'production') { - warn('No coordinate system that supports ' + methodName + ' found by the given finder.'); - } - }; - - updateStreamModes = function (ecIns, ecModel) { - var chartsMap = ecIns._chartsMap; - var scheduler = ecIns._scheduler; - ecModel.eachSeries(function (seriesModel) { - scheduler.updateStreamModes(seriesModel, chartsMap[seriesModel.__viewId]); - }); - }; - - doDispatchAction = function (payload, silent) { - var _this = this; - - var ecModel = this.getModel(); - var payloadType = payload.type; - var escapeConnect = payload.escapeConnect; - var actionWrap = actions[payloadType]; - var actionInfo = actionWrap.actionInfo; - var cptTypeTmp = (actionInfo.update || 'update').split(':'); - var updateMethod = cptTypeTmp.pop(); - var cptType = cptTypeTmp[0] != null && parseClassType(cptTypeTmp[0]); - this[IN_MAIN_PROCESS_KEY] = true; - var payloads = [payload]; - var batched = false; // Batch action - - if (payload.batch) { - batched = true; - payloads = map(payload.batch, function (item) { - item = defaults(extend({}, item), payload); - item.batch = null; - return item; - }); - } - - var eventObjBatch = []; - var eventObj; - var isSelectChange = isSelectChangePayload(payload); - var isHighDown = isHighDownPayload(payload); // Only leave blur once if there are multiple batches. - - if (isHighDown) { - allLeaveBlur(this._api); - } - - each(payloads, function (batchItem) { - // Action can specify the event by return it. - eventObj = actionWrap.action(batchItem, _this._model, _this._api); // Emit event outside - - eventObj = eventObj || extend({}, batchItem); // Convert type to eventType - - eventObj.type = actionInfo.event || eventObj.type; - eventObjBatch.push(eventObj); // light update does not perform data process, layout and visual. - - if (isHighDown) { - var _a = preParseFinder(payload), - queryOptionMap = _a.queryOptionMap, - mainTypeSpecified = _a.mainTypeSpecified; - - var componentMainType = mainTypeSpecified ? queryOptionMap.keys()[0] : 'series'; - updateDirectly(_this, updateMethod, batchItem, componentMainType); - markStatusToUpdate(_this); - } else if (isSelectChange) { - // At present `dispatchAction({ type: 'select', ... })` is not supported on components. - // geo still use 'geoselect'. - updateDirectly(_this, updateMethod, batchItem, 'series'); - markStatusToUpdate(_this); - } else if (cptType) { - updateDirectly(_this, updateMethod, batchItem, cptType.main, cptType.sub); - } - }); - - if (updateMethod !== 'none' && !isHighDown && !isSelectChange && !cptType) { - try { - // Still dirty - if (this[PENDING_UPDATE]) { - prepare(this); - updateMethods.update.call(this, payload); - this[PENDING_UPDATE] = null; - } else { - updateMethods[updateMethod].call(this, payload); - } - } catch (e) { - this[IN_MAIN_PROCESS_KEY] = false; - throw e; - } - } // Follow the rule of action batch - - - if (batched) { - eventObj = { - type: actionInfo.event || payloadType, - escapeConnect: escapeConnect, - batch: eventObjBatch - }; - } else { - eventObj = eventObjBatch[0]; - } - - this[IN_MAIN_PROCESS_KEY] = false; - - if (!silent) { - var messageCenter = this._messageCenter; - messageCenter.trigger(eventObj.type, eventObj); // Extra triggered 'selectchanged' event - - if (isSelectChange) { - var newObj = { - type: 'selectchanged', - escapeConnect: escapeConnect, - selected: getAllSelectedIndices(ecModel), - isFromClick: payload.isFromClick || false, - fromAction: payload.type, - fromActionPayload: payload - }; - messageCenter.trigger(newObj.type, newObj); - } - } - }; - - flushPendingActions = function (silent) { - var pendingActions = this._pendingActions; - - while (pendingActions.length) { - var payload = pendingActions.shift(); - doDispatchAction.call(this, payload, silent); - } - }; - - triggerUpdatedEvent = function (silent) { - !silent && this.trigger('updated'); - }; - /** - * Event `rendered` is triggered when zr - * rendered. It is useful for realtime - * snapshot (reflect animation). - * - * Event `finished` is triggered when: - * (1) zrender rendering finished. - * (2) initial animation finished. - * (3) progressive rendering finished. - * (4) no pending action. - * (5) no delayed setOption needs to be processed. - */ - - - bindRenderedEvent = function (zr, ecIns) { - zr.on('rendered', function (params) { - ecIns.trigger('rendered', params); // The `finished` event should not be triggered repeatedly, - // so it should only be triggered when rendering indeed happens - // in zrender. (Consider the case that dipatchAction is keep - // triggering when mouse move). - - if ( // Although zr is dirty if initial animation is not finished - // and this checking is called on frame, we also check - // animation finished for robustness. - zr.animation.isFinished() && !ecIns[PENDING_UPDATE] && !ecIns._scheduler.unfinished && !ecIns._pendingActions.length) { - ecIns.trigger('finished'); - } - }); - }; - - bindMouseEvent = function (zr, ecIns) { - zr.on('mouseover', function (e) { - var el = e.target; - var dispatcher = findEventDispatcher(el, isHighDownDispatcher); - - if (dispatcher) { - handleGlobalMouseOverForHighDown(dispatcher, e, ecIns._api); - markStatusToUpdate(ecIns); - } - }).on('mouseout', function (e) { - var el = e.target; - var dispatcher = findEventDispatcher(el, isHighDownDispatcher); - - if (dispatcher) { - handleGlobalMouseOutForHighDown(dispatcher, e, ecIns._api); - markStatusToUpdate(ecIns); - } - }).on('click', function (e) { - var el = e.target; - var dispatcher = findEventDispatcher(el, function (target) { - return getECData(target).dataIndex != null; - }, true); - - if (dispatcher) { - var actionType = dispatcher.selected ? 'unselect' : 'select'; - var ecData = getECData(dispatcher); - - ecIns._api.dispatchAction({ - type: actionType, - dataType: ecData.dataType, - dataIndexInside: ecData.dataIndex, - seriesIndex: ecData.seriesIndex, - isFromClick: true - }); - } - }); - }; - - function clearColorPalette(ecModel) { - ecModel.clearColorPalette(); - ecModel.eachSeries(function (seriesModel) { - seriesModel.clearColorPalette(); - }); - } - - function allocateZlevels(ecModel) { - var componentZLevels = []; - var seriesZLevels = []; - var hasSeperateZLevel = false; - ecModel.eachComponent(function (componentType, componentModel) { - var zlevel = componentModel.get('zlevel') || 0; - var z = componentModel.get('z') || 0; - var zlevelKey = componentModel.getZLevelKey(); - hasSeperateZLevel = hasSeperateZLevel || !!zlevelKey; - (componentType === 'series' ? seriesZLevels : componentZLevels).push({ - zlevel: zlevel, - z: z, - idx: componentModel.componentIndex, - type: componentType, - key: zlevelKey - }); - }); - - if (hasSeperateZLevel) { - // Series after component - var zLevels = componentZLevels.concat(seriesZLevels); - var lastSeriesZLevel_1; - var lastSeriesKey_1; - sort(zLevels, function (a, b) { - if (a.zlevel === b.zlevel) { - return a.z - b.z; - } - - return a.zlevel - b.zlevel; - }); - each(zLevels, function (item) { - var componentModel = ecModel.getComponent(item.type, item.idx); - var zlevel = item.zlevel; - var key = item.key; - - if (lastSeriesZLevel_1 != null) { - zlevel = Math.max(lastSeriesZLevel_1, zlevel); - } - - if (key) { - if (zlevel === lastSeriesZLevel_1 && key !== lastSeriesKey_1) { - zlevel++; - } - - lastSeriesKey_1 = key; - } else if (lastSeriesKey_1) { - if (zlevel === lastSeriesZLevel_1) { - zlevel++; - } - - lastSeriesKey_1 = ''; - } - - lastSeriesZLevel_1 = zlevel; - componentModel.setZLevel(zlevel); - }); - } - } - - render = function (ecIns, ecModel, api, payload, updateParams) { - allocateZlevels(ecModel); - renderComponents(ecIns, ecModel, api, payload, updateParams); - each(ecIns._chartsViews, function (chart) { - chart.__alive = false; - }); - renderSeries(ecIns, ecModel, api, payload, updateParams); // Remove groups of unrendered charts - - each(ecIns._chartsViews, function (chart) { - if (!chart.__alive) { - chart.remove(ecModel, api); - } - }); - }; - - renderComponents = function (ecIns, ecModel, api, payload, updateParams, dirtyList) { - each(dirtyList || ecIns._componentsViews, function (componentView) { - var componentModel = componentView.__model; - clearStates(componentModel, componentView); - componentView.render(componentModel, ecModel, api, payload); - updateZ(componentModel, componentView); - updateStates(componentModel, componentView); - }); - }; - /** - * Render each chart and component - */ - - - renderSeries = function (ecIns, ecModel, api, payload, updateParams, dirtyMap) { - // Render all charts - var scheduler = ecIns._scheduler; - updateParams = extend(updateParams || {}, { - updatedSeries: ecModel.getSeries() - }); // TODO progressive? - - lifecycle.trigger('series:beforeupdate', ecModel, api, updateParams); - var unfinished = false; - ecModel.eachSeries(function (seriesModel) { - var chartView = ecIns._chartsMap[seriesModel.__viewId]; - chartView.__alive = true; - var renderTask = chartView.renderTask; - scheduler.updatePayload(renderTask, payload); // TODO states on marker. - - clearStates(seriesModel, chartView); - - if (dirtyMap && dirtyMap.get(seriesModel.uid)) { - renderTask.dirty(); - } - - if (renderTask.perform(scheduler.getPerformArgs(renderTask))) { - unfinished = true; - } - - chartView.group.silent = !!seriesModel.get('silent'); // Should not call markRedraw on group, because it will disable zrender - // incremental render (always render from the __startIndex each frame) - // chartView.group.markRedraw(); - - updateBlend(seriesModel, chartView); - updateSeriesElementSelection(seriesModel); - }); - scheduler.unfinished = unfinished || scheduler.unfinished; - lifecycle.trigger('series:layoutlabels', ecModel, api, updateParams); // transition after label is layouted. - - lifecycle.trigger('series:transition', ecModel, api, updateParams); - ecModel.eachSeries(function (seriesModel) { - var chartView = ecIns._chartsMap[seriesModel.__viewId]; // Update Z after labels updated. Before applying states. - - updateZ(seriesModel, chartView); // NOTE: Update states after label is updated. - // label should be in normal status when layouting. - - updateStates(seriesModel, chartView); - }); // If use hover layer - - updateHoverLayerStatus(ecIns, ecModel); - lifecycle.trigger('series:afterupdate', ecModel, api, updateParams); - }; - - markStatusToUpdate = function (ecIns) { - ecIns[STATUS_NEEDS_UPDATE_KEY] = true; // Wake up zrender if it's sleep. Let it update states in the next frame. - - ecIns.getZr().wakeUp(); - }; - - applyChangedStates = function (ecIns) { - if (!ecIns[STATUS_NEEDS_UPDATE_KEY]) { - return; - } - - ecIns.getZr().storage.traverse(function (el) { - // Not applied on removed elements, it may still in fading. - if (isElementRemoved(el)) { - return; - } - - applyElementStates(el); - }); - ecIns[STATUS_NEEDS_UPDATE_KEY] = false; - }; - - function applyElementStates(el) { - var newStates = []; - var oldStates = el.currentStates; // Keep other states. - - for (var i = 0; i < oldStates.length; i++) { - var stateName = oldStates[i]; - - if (!(stateName === 'emphasis' || stateName === 'blur' || stateName === 'select')) { - newStates.push(stateName); - } - } // Only use states when it's exists. - - - if (el.selected && el.states.select) { - newStates.push('select'); - } - - if (el.hoverState === HOVER_STATE_EMPHASIS && el.states.emphasis) { - newStates.push('emphasis'); - } else if (el.hoverState === HOVER_STATE_BLUR && el.states.blur) { - newStates.push('blur'); - } - - el.useStates(newStates); - } - - function updateHoverLayerStatus(ecIns, ecModel) { - var zr = ecIns._zr; - var storage = zr.storage; - var elCount = 0; - storage.traverse(function (el) { - if (!el.isGroup) { - elCount++; - } - }); - - if (elCount > ecModel.get('hoverLayerThreshold') && !env.node && !env.worker) { - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.preventUsingHoverLayer) { - return; - } - - var chartView = ecIns._chartsMap[seriesModel.__viewId]; - - if (chartView.__alive) { - chartView.eachRendered(function (el) { - if (el.states.emphasis) { - el.states.emphasis.hoverLayer = true; - } - }); - } - }); - } - } - /** - * Update chart and blend. - */ - - function updateBlend(seriesModel, chartView) { - var blendMode = seriesModel.get('blendMode') || null; - chartView.eachRendered(function (el) { - // FIXME marker and other components - if (!el.isGroup) { - // DON'T mark the element dirty. In case element is incremental and don't want to rerender. - el.style.blend = blendMode; - } - }); - } - - function updateZ(model, view) { - if (model.preventAutoZ) { - return; - } - - var z = model.get('z') || 0; - var zlevel = model.get('zlevel') || 0; // Set z and zlevel - - view.eachRendered(function (el) { - doUpdateZ(el, z, zlevel, -Infinity); // Don't traverse the children because it has been traversed in _updateZ. - - return true; - }); - } - - function doUpdateZ(el, z, zlevel, maxZ2) { - // Group may also have textContent - var label = el.getTextContent(); - var labelLine = el.getTextGuideLine(); - var isGroup = el.isGroup; - - if (isGroup) { - // set z & zlevel of children elements of Group - var children = el.childrenRef(); - - for (var i = 0; i < children.length; i++) { - maxZ2 = Math.max(doUpdateZ(children[i], z, zlevel, maxZ2), maxZ2); - } - } else { - // not Group - el.z = z; - el.zlevel = zlevel; - maxZ2 = Math.max(el.z2, maxZ2); - } // always set z and zlevel if label/labelLine exists - - - if (label) { - label.z = z; - label.zlevel = zlevel; // lift z2 of text content - // TODO if el.emphasis.z2 is spcefied, what about textContent. - - isFinite(maxZ2) && (label.z2 = maxZ2 + 2); - } - - if (labelLine) { - var textGuideLineConfig = el.textGuideLineConfig; - labelLine.z = z; - labelLine.zlevel = zlevel; - isFinite(maxZ2) && (labelLine.z2 = maxZ2 + (textGuideLineConfig && textGuideLineConfig.showAbove ? 1 : -1)); - } - - return maxZ2; - } // Clear states without animation. - // TODO States on component. - - - function clearStates(model, view) { - view.eachRendered(function (el) { - // Not applied on removed elements, it may still in fading. - if (isElementRemoved(el)) { - return; - } - - var textContent = el.getTextContent(); - var textGuide = el.getTextGuideLine(); - - if (el.stateTransition) { - el.stateTransition = null; - } - - if (textContent && textContent.stateTransition) { - textContent.stateTransition = null; - } - - if (textGuide && textGuide.stateTransition) { - textGuide.stateTransition = null; - } // TODO If el is incremental. - - - if (el.hasState()) { - el.prevStates = el.currentStates; - el.clearStates(); - } else if (el.prevStates) { - el.prevStates = null; - } - }); - } - - function updateStates(model, view) { - var stateAnimationModel = model.getModel('stateAnimation'); - var enableAnimation = model.isAnimationEnabled(); - var duration = stateAnimationModel.get('duration'); - var stateTransition = duration > 0 ? { - duration: duration, - delay: stateAnimationModel.get('delay'), - easing: stateAnimationModel.get('easing') // additive: stateAnimationModel.get('additive') - - } : null; - view.eachRendered(function (el) { - if (el.states && el.states.emphasis) { - // Not applied on removed elements, it may still in fading. - if (isElementRemoved(el)) { - return; - } - - if (el instanceof Path) { - savePathStates(el); - } // Only updated on changed element. In case element is incremental and don't want to rerender. - // TODO, a more proper way? - - - if (el.__dirty) { - var prevStates = el.prevStates; // Restore states without animation - - if (prevStates) { - el.useStates(prevStates); - } - } // Update state transition and enable animation again. - - - if (enableAnimation) { - el.stateTransition = stateTransition; - var textContent = el.getTextContent(); - var textGuide = el.getTextGuideLine(); // TODO Is it necessary to animate label? - - if (textContent) { - textContent.stateTransition = stateTransition; - } - - if (textGuide) { - textGuide.stateTransition = stateTransition; - } - } // Use highlighted and selected flag to toggle states. - - - if (el.__dirty) { - applyElementStates(el); - } - } - }); - } - - createExtensionAPI = function (ecIns) { - return new ( - /** @class */ - function (_super) { - __extends(class_1, _super); - - function class_1() { - return _super !== null && _super.apply(this, arguments) || this; - } - - class_1.prototype.getCoordinateSystems = function () { - return ecIns._coordSysMgr.getCoordinateSystems(); - }; - - class_1.prototype.getComponentByElement = function (el) { - while (el) { - var modelInfo = el.__ecComponentInfo; - - if (modelInfo != null) { - return ecIns._model.getComponent(modelInfo.mainType, modelInfo.index); - } - - el = el.parent; - } - }; - - class_1.prototype.enterEmphasis = function (el, highlightDigit) { - enterEmphasis(el, highlightDigit); - markStatusToUpdate(ecIns); - }; - - class_1.prototype.leaveEmphasis = function (el, highlightDigit) { - leaveEmphasis(el, highlightDigit); - markStatusToUpdate(ecIns); - }; - - class_1.prototype.enterBlur = function (el) { - enterBlur(el); - markStatusToUpdate(ecIns); - }; - - class_1.prototype.leaveBlur = function (el) { - leaveBlur(el); - markStatusToUpdate(ecIns); - }; - - class_1.prototype.enterSelect = function (el) { - enterSelect(el); - markStatusToUpdate(ecIns); - }; - - class_1.prototype.leaveSelect = function (el) { - leaveSelect(el); - markStatusToUpdate(ecIns); - }; - - class_1.prototype.getModel = function () { - return ecIns.getModel(); - }; - - class_1.prototype.getViewOfComponentModel = function (componentModel) { - return ecIns.getViewOfComponentModel(componentModel); - }; - - class_1.prototype.getViewOfSeriesModel = function (seriesModel) { - return ecIns.getViewOfSeriesModel(seriesModel); - }; - - return class_1; - }(ExtensionAPI))(ecIns); - }; - - enableConnect = function (chart) { - function updateConnectedChartsStatus(charts, status) { - for (var i = 0; i < charts.length; i++) { - var otherChart = charts[i]; - otherChart[CONNECT_STATUS_KEY] = status; - } - } - - each(eventActionMap, function (actionType, eventType) { - chart._messageCenter.on(eventType, function (event) { - if (connectedGroups[chart.group] && chart[CONNECT_STATUS_KEY] !== CONNECT_STATUS_PENDING) { - if (event && event.escapeConnect) { - return; - } - - var action_1 = chart.makeActionFromEvent(event); - var otherCharts_1 = []; - each(instances$1, function (otherChart) { - if (otherChart !== chart && otherChart.group === chart.group) { - otherCharts_1.push(otherChart); - } - }); - updateConnectedChartsStatus(otherCharts_1, CONNECT_STATUS_PENDING); - each(otherCharts_1, function (otherChart) { - if (otherChart[CONNECT_STATUS_KEY] !== CONNECT_STATUS_UPDATING) { - otherChart.dispatchAction(action_1); - } - }); - updateConnectedChartsStatus(otherCharts_1, CONNECT_STATUS_UPDATED); - } - }); - }); - }; - }(); - - return ECharts; - }(Eventful); - - var echartsProto = ECharts.prototype; - echartsProto.on = createRegisterEventWithLowercaseECharts('on'); - echartsProto.off = createRegisterEventWithLowercaseECharts('off'); - /** - * @deprecated - */ - // @ts-ignore - - echartsProto.one = function (eventName, cb, ctx) { - var self = this; - deprecateLog('ECharts#one is deprecated.'); - - function wrapped() { - var args2 = []; - - for (var _i = 0; _i < arguments.length; _i++) { - args2[_i] = arguments[_i]; - } - - cb && cb.apply && cb.apply(this, args2); // @ts-ignore - - self.off(eventName, wrapped); - } - - this.on.call(this, eventName, wrapped, ctx); - }; - - var MOUSE_EVENT_NAMES = ['click', 'dblclick', 'mouseover', 'mouseout', 'mousemove', 'mousedown', 'mouseup', 'globalout', 'contextmenu']; - - function disposedWarning(id) { - if ("development" !== 'production') { - warn('Instance ' + id + ' has been disposed'); - } - } - - var actions = {}; - /** - * Map eventType to actionType - */ - - var eventActionMap = {}; - var dataProcessorFuncs = []; - var optionPreprocessorFuncs = []; - var visualFuncs = []; - var themeStorage = {}; - var loadingEffects = {}; - var instances$1 = {}; - var connectedGroups = {}; - var idBase = +new Date() - 0; - var groupIdBase = +new Date() - 0; - var DOM_ATTRIBUTE_KEY = '_echarts_instance_'; - /** - * @param opts.devicePixelRatio Use window.devicePixelRatio by default - * @param opts.renderer Can choose 'canvas' or 'svg' to render the chart. - * @param opts.width Use clientWidth of the input `dom` by default. - * Can be 'auto' (the same as null/undefined) - * @param opts.height Use clientHeight of the input `dom` by default. - * Can be 'auto' (the same as null/undefined) - * @param opts.locale Specify the locale. - * @param opts.useDirtyRect Enable dirty rectangle rendering or not. - */ - - function init$1(dom, theme, opts) { - var isClient = !(opts && opts.ssr); - - if (isClient) { - if ("development" !== 'production') { - if (!dom) { - throw new Error('Initialize failed: invalid dom.'); - } - } - - var existInstance = getInstanceByDom(dom); - - if (existInstance) { - if ("development" !== 'production') { - warn('There is a chart instance already initialized on the dom.'); - } - - return existInstance; - } - - if ("development" !== 'production') { - if (isDom(dom) && dom.nodeName.toUpperCase() !== 'CANVAS' && (!dom.clientWidth && (!opts || opts.width == null) || !dom.clientHeight && (!opts || opts.height == null))) { - warn('Can\'t get DOM width or height. Please check ' + 'dom.clientWidth and dom.clientHeight. They should not be 0.' + 'For example, you may need to call this in the callback ' + 'of window.onload.'); - } - } - } - - var chart = new ECharts(dom, theme, opts); - chart.id = 'ec_' + idBase++; - instances$1[chart.id] = chart; - isClient && setAttribute(dom, DOM_ATTRIBUTE_KEY, chart.id); - enableConnect(chart); - lifecycle.trigger('afterinit', chart); - return chart; - } - /** - * @usage - * (A) - * ```js - * let chart1 = echarts.init(dom1); - * let chart2 = echarts.init(dom2); - * chart1.group = 'xxx'; - * chart2.group = 'xxx'; - * echarts.connect('xxx'); - * ``` - * (B) - * ```js - * let chart1 = echarts.init(dom1); - * let chart2 = echarts.init(dom2); - * echarts.connect('xxx', [chart1, chart2]); - * ``` - */ - - function connect(groupId) { - // Is array of charts - if (isArray(groupId)) { - var charts = groupId; - groupId = null; // If any chart has group - - each(charts, function (chart) { - if (chart.group != null) { - groupId = chart.group; - } - }); - groupId = groupId || 'g_' + groupIdBase++; - each(charts, function (chart) { - chart.group = groupId; - }); - } - - connectedGroups[groupId] = true; - return groupId; - } - /** - * @deprecated - */ - - function disConnect(groupId) { - connectedGroups[groupId] = false; - } - /** - * Alias and backward compatibility - */ - - var disconnect = disConnect; - /** - * Dispose a chart instance - */ - - function dispose$1(chart) { - if (isString(chart)) { - chart = instances$1[chart]; - } else if (!(chart instanceof ECharts)) { - // Try to treat as dom - chart = getInstanceByDom(chart); - } - - if (chart instanceof ECharts && !chart.isDisposed()) { - chart.dispose(); - } - } - function getInstanceByDom(dom) { - return instances$1[getAttribute(dom, DOM_ATTRIBUTE_KEY)]; - } - function getInstanceById(key) { - return instances$1[key]; - } - /** - * Register theme - */ - - function registerTheme(name, theme) { - themeStorage[name] = theme; - } - /** - * Register option preprocessor - */ - - function registerPreprocessor(preprocessorFunc) { - if (indexOf(optionPreprocessorFuncs, preprocessorFunc) < 0) { - optionPreprocessorFuncs.push(preprocessorFunc); - } - } - function registerProcessor(priority, processor) { - normalizeRegister(dataProcessorFuncs, priority, processor, PRIORITY_PROCESSOR_DEFAULT); - } - /** - * Register postIniter - * @param {Function} postInitFunc - */ - - function registerPostInit(postInitFunc) { - registerUpdateLifecycle('afterinit', postInitFunc); - } - /** - * Register postUpdater - * @param {Function} postUpdateFunc - */ - - function registerPostUpdate(postUpdateFunc) { - registerUpdateLifecycle('afterupdate', postUpdateFunc); - } - function registerUpdateLifecycle(name, cb) { - lifecycle.on(name, cb); - } - function registerAction(actionInfo, eventName, action) { - if (isFunction(eventName)) { - action = eventName; - eventName = ''; - } - - var actionType = isObject(actionInfo) ? actionInfo.type : [actionInfo, actionInfo = { - event: eventName - }][0]; // Event name is all lowercase - - actionInfo.event = (actionInfo.event || actionType).toLowerCase(); - eventName = actionInfo.event; - - if (eventActionMap[eventName]) { - // Already registered. - return; - } // Validate action type and event name. - - - assert(ACTION_REG.test(actionType) && ACTION_REG.test(eventName)); - - if (!actions[actionType]) { - actions[actionType] = { - action: action, - actionInfo: actionInfo - }; - } - - eventActionMap[eventName] = actionType; - } - function registerCoordinateSystem(type, coordSysCreator) { - CoordinateSystemManager.register(type, coordSysCreator); - } - /** - * Get dimensions of specified coordinate system. - * @param {string} type - * @return {Array.<string|Object>} - */ - - function getCoordinateSystemDimensions(type) { - var coordSysCreator = CoordinateSystemManager.get(type); - - if (coordSysCreator) { - return coordSysCreator.getDimensionsInfo ? coordSysCreator.getDimensionsInfo() : coordSysCreator.dimensions.slice(); - } - } - - function registerLayout(priority, layoutTask) { - normalizeRegister(visualFuncs, priority, layoutTask, PRIORITY_VISUAL_LAYOUT, 'layout'); - } - - function registerVisual(priority, visualTask) { - normalizeRegister(visualFuncs, priority, visualTask, PRIORITY_VISUAL_CHART, 'visual'); - } - var registeredTasks = []; - - function normalizeRegister(targetList, priority, fn, defaultPriority, visualType) { - if (isFunction(priority) || isObject(priority)) { - fn = priority; - priority = defaultPriority; - } - - if ("development" !== 'production') { - if (isNaN(priority) || priority == null) { - throw new Error('Illegal priority'); - } // Check duplicate - - - each(targetList, function (wrap) { - assert(wrap.__raw !== fn); - }); - } // Already registered - - - if (indexOf(registeredTasks, fn) >= 0) { - return; - } - - registeredTasks.push(fn); - var stageHandler = Scheduler.wrapStageHandler(fn, visualType); - stageHandler.__prio = priority; - stageHandler.__raw = fn; - targetList.push(stageHandler); - } - - function registerLoading(name, loadingFx) { - loadingEffects[name] = loadingFx; - } - /** - * ZRender need a canvas context to do measureText. - * But in node environment canvas may be created by node-canvas. - * So we need to specify how to create a canvas instead of using document.createElement('canvas') - * - * - * @deprecated use setPlatformAPI({ createCanvas }) instead. - * - * @example - * let Canvas = require('canvas'); - * let echarts = require('echarts'); - * echarts.setCanvasCreator(function () { - * // Small size is enough. - * return new Canvas(32, 32); - * }); - */ - - function setCanvasCreator(creator) { - if ("development" !== 'production') { - deprecateLog('setCanvasCreator is deprecated. Use setPlatformAPI({ createCanvas }) instead.'); - } - - setPlatformAPI({ - createCanvas: creator - }); - } - /** - * The parameters and usage: see `geoSourceManager.registerMap`. - * Compatible with previous `echarts.registerMap`. - */ - - function registerMap(mapName, geoJson, specialAreas) { - var registerMap = getImpl('registerMap'); - registerMap && registerMap(mapName, geoJson, specialAreas); - } - function getMap(mapName) { - var getMap = getImpl('getMap'); - return getMap && getMap(mapName); - } - var registerTransform = registerExternalTransform; - /** - * Globa dispatchAction to a specified chart instance. - */ - // export function dispatchAction(payload: { chartId: string } & Payload, opt?: Parameters<ECharts['dispatchAction']>[1]) { - // if (!payload || !payload.chartId) { - // // Must have chartId to find chart - // return; - // } - // const chart = instances[payload.chartId]; - // if (chart) { - // chart.dispatchAction(payload, opt); - // } - // } - // Builtin global visual - - registerVisual(PRIORITY_VISUAL_GLOBAL, seriesStyleTask); - registerVisual(PRIORITY_VISUAL_CHART_DATA_CUSTOM, dataStyleTask); - registerVisual(PRIORITY_VISUAL_CHART_DATA_CUSTOM, dataColorPaletteTask); - registerVisual(PRIORITY_VISUAL_GLOBAL, seriesSymbolTask); - registerVisual(PRIORITY_VISUAL_CHART_DATA_CUSTOM, dataSymbolTask); - registerVisual(PRIORITY_VISUAL_DECAL, decalVisual); - registerPreprocessor(globalBackwardCompat); - registerProcessor(PRIORITY_PROCESSOR_DATASTACK, dataStack); - registerLoading('default', defaultLoading); // Default actions - - registerAction({ - type: HIGHLIGHT_ACTION_TYPE, - event: HIGHLIGHT_ACTION_TYPE, - update: HIGHLIGHT_ACTION_TYPE - }, noop); - registerAction({ - type: DOWNPLAY_ACTION_TYPE, - event: DOWNPLAY_ACTION_TYPE, - update: DOWNPLAY_ACTION_TYPE - }, noop); - registerAction({ - type: SELECT_ACTION_TYPE, - event: SELECT_ACTION_TYPE, - update: SELECT_ACTION_TYPE - }, noop); - registerAction({ - type: UNSELECT_ACTION_TYPE, - event: UNSELECT_ACTION_TYPE, - update: UNSELECT_ACTION_TYPE - }, noop); - registerAction({ - type: TOGGLE_SELECT_ACTION_TYPE, - event: TOGGLE_SELECT_ACTION_TYPE, - update: TOGGLE_SELECT_ACTION_TYPE - }, noop); // Default theme - - registerTheme('light', lightTheme); - registerTheme('dark', theme); // For backward compatibility, where the namespace `dataTool` will - // be mounted on `echarts` is the extension `dataTool` is imported. - - var dataTool = {}; - - var extensions = []; - var extensionRegisters = { - registerPreprocessor: registerPreprocessor, - registerProcessor: registerProcessor, - registerPostInit: registerPostInit, - registerPostUpdate: registerPostUpdate, - registerUpdateLifecycle: registerUpdateLifecycle, - registerAction: registerAction, - registerCoordinateSystem: registerCoordinateSystem, - registerLayout: registerLayout, - registerVisual: registerVisual, - registerTransform: registerTransform, - registerLoading: registerLoading, - registerMap: registerMap, - registerImpl: registerImpl, - PRIORITY: PRIORITY, - ComponentModel: ComponentModel, - ComponentView: ComponentView, - SeriesModel: SeriesModel, - ChartView: ChartView, - // TODO Use ComponentModel and SeriesModel instead of Constructor - registerComponentModel: function (ComponentModelClass) { - ComponentModel.registerClass(ComponentModelClass); - }, - registerComponentView: function (ComponentViewClass) { - ComponentView.registerClass(ComponentViewClass); - }, - registerSeriesModel: function (SeriesModelClass) { - SeriesModel.registerClass(SeriesModelClass); - }, - registerChartView: function (ChartViewClass) { - ChartView.registerClass(ChartViewClass); - }, - registerSubTypeDefaulter: function (componentType, defaulter) { - ComponentModel.registerSubTypeDefaulter(componentType, defaulter); - }, - registerPainter: function (painterType, PainterCtor) { - registerPainter(painterType, PainterCtor); - } - }; - function use(ext) { - if (isArray(ext)) { - // use([ChartLine, ChartBar]); - each(ext, function (singleExt) { - use(singleExt); - }); - return; - } - - if (indexOf(extensions, ext) >= 0) { - return; - } - - extensions.push(ext); - - if (isFunction(ext)) { - ext = { - install: ext - }; - } - - ext.install(extensionRegisters); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function dataIndexMapValueLength(valNumOrArrLengthMoreThan2) { - return valNumOrArrLengthMoreThan2 == null ? 0 : valNumOrArrLengthMoreThan2.length || 1; - } - - function defaultKeyGetter(item) { - return item; - } - - var DataDiffer = - /** @class */ - function () { - /** - * @param context Can be visited by this.context in callback. - */ - function DataDiffer(oldArr, newArr, oldKeyGetter, newKeyGetter, context, // By default: 'oneToOne'. - diffMode) { - this._old = oldArr; - this._new = newArr; - this._oldKeyGetter = oldKeyGetter || defaultKeyGetter; - this._newKeyGetter = newKeyGetter || defaultKeyGetter; // Visible in callback via `this.context`; - - this.context = context; - this._diffModeMultiple = diffMode === 'multiple'; - } - /** - * Callback function when add a data - */ - - - DataDiffer.prototype.add = function (func) { - this._add = func; - return this; - }; - /** - * Callback function when update a data - */ - - - DataDiffer.prototype.update = function (func) { - this._update = func; - return this; - }; - /** - * Callback function when update a data and only work in `cbMode: 'byKey'`. - */ - - - DataDiffer.prototype.updateManyToOne = function (func) { - this._updateManyToOne = func; - return this; - }; - /** - * Callback function when update a data and only work in `cbMode: 'byKey'`. - */ - - - DataDiffer.prototype.updateOneToMany = function (func) { - this._updateOneToMany = func; - return this; - }; - /** - * Callback function when update a data and only work in `cbMode: 'byKey'`. - */ - - - DataDiffer.prototype.updateManyToMany = function (func) { - this._updateManyToMany = func; - return this; - }; - /** - * Callback function when remove a data - */ - - - DataDiffer.prototype.remove = function (func) { - this._remove = func; - return this; - }; - - DataDiffer.prototype.execute = function () { - this[this._diffModeMultiple ? '_executeMultiple' : '_executeOneToOne'](); - }; - - DataDiffer.prototype._executeOneToOne = function () { - var oldArr = this._old; - var newArr = this._new; - var newDataIndexMap = {}; - var oldDataKeyArr = new Array(oldArr.length); - var newDataKeyArr = new Array(newArr.length); - - this._initIndexMap(oldArr, null, oldDataKeyArr, '_oldKeyGetter'); - - this._initIndexMap(newArr, newDataIndexMap, newDataKeyArr, '_newKeyGetter'); - - for (var i = 0; i < oldArr.length; i++) { - var oldKey = oldDataKeyArr[i]; - var newIdxMapVal = newDataIndexMap[oldKey]; - var newIdxMapValLen = dataIndexMapValueLength(newIdxMapVal); // idx can never be empty array here. see 'set null' logic below. - - if (newIdxMapValLen > 1) { - // Consider there is duplicate key (for example, use dataItem.name as key). - // We should make sure every item in newArr and oldArr can be visited. - var newIdx = newIdxMapVal.shift(); - - if (newIdxMapVal.length === 1) { - newDataIndexMap[oldKey] = newIdxMapVal[0]; - } - - this._update && this._update(newIdx, i); - } else if (newIdxMapValLen === 1) { - newDataIndexMap[oldKey] = null; - this._update && this._update(newIdxMapVal, i); - } else { - this._remove && this._remove(i); - } - } - - this._performRestAdd(newDataKeyArr, newDataIndexMap); - }; - /** - * For example, consider the case: - * oldData: [o0, o1, o2, o3, o4, o5, o6, o7], - * newData: [n0, n1, n2, n3, n4, n5, n6, n7, n8], - * Where: - * o0, o1, n0 has key 'a' (many to one) - * o5, n4, n5, n6 has key 'b' (one to many) - * o2, n1 has key 'c' (one to one) - * n2, n3 has key 'd' (add) - * o3, o4 has key 'e' (remove) - * o6, o7, n7, n8 has key 'f' (many to many, treated as add and remove) - * Then: - * (The order of the following directives are not ensured.) - * this._updateManyToOne(n0, [o0, o1]); - * this._updateOneToMany([n4, n5, n6], o5); - * this._update(n1, o2); - * this._remove(o3); - * this._remove(o4); - * this._remove(o6); - * this._remove(o7); - * this._add(n2); - * this._add(n3); - * this._add(n7); - * this._add(n8); - */ - - - DataDiffer.prototype._executeMultiple = function () { - var oldArr = this._old; - var newArr = this._new; - var oldDataIndexMap = {}; - var newDataIndexMap = {}; - var oldDataKeyArr = []; - var newDataKeyArr = []; - - this._initIndexMap(oldArr, oldDataIndexMap, oldDataKeyArr, '_oldKeyGetter'); - - this._initIndexMap(newArr, newDataIndexMap, newDataKeyArr, '_newKeyGetter'); - - for (var i = 0; i < oldDataKeyArr.length; i++) { - var oldKey = oldDataKeyArr[i]; - var oldIdxMapVal = oldDataIndexMap[oldKey]; - var newIdxMapVal = newDataIndexMap[oldKey]; - var oldIdxMapValLen = dataIndexMapValueLength(oldIdxMapVal); - var newIdxMapValLen = dataIndexMapValueLength(newIdxMapVal); - - if (oldIdxMapValLen > 1 && newIdxMapValLen === 1) { - this._updateManyToOne && this._updateManyToOne(newIdxMapVal, oldIdxMapVal); - newDataIndexMap[oldKey] = null; - } else if (oldIdxMapValLen === 1 && newIdxMapValLen > 1) { - this._updateOneToMany && this._updateOneToMany(newIdxMapVal, oldIdxMapVal); - newDataIndexMap[oldKey] = null; - } else if (oldIdxMapValLen === 1 && newIdxMapValLen === 1) { - this._update && this._update(newIdxMapVal, oldIdxMapVal); - newDataIndexMap[oldKey] = null; - } else if (oldIdxMapValLen > 1 && newIdxMapValLen > 1) { - this._updateManyToMany && this._updateManyToMany(newIdxMapVal, oldIdxMapVal); - newDataIndexMap[oldKey] = null; - } else if (oldIdxMapValLen > 1) { - for (var i_1 = 0; i_1 < oldIdxMapValLen; i_1++) { - this._remove && this._remove(oldIdxMapVal[i_1]); - } - } else { - this._remove && this._remove(oldIdxMapVal); - } - } - - this._performRestAdd(newDataKeyArr, newDataIndexMap); - }; - - DataDiffer.prototype._performRestAdd = function (newDataKeyArr, newDataIndexMap) { - for (var i = 0; i < newDataKeyArr.length; i++) { - var newKey = newDataKeyArr[i]; - var newIdxMapVal = newDataIndexMap[newKey]; - var idxMapValLen = dataIndexMapValueLength(newIdxMapVal); - - if (idxMapValLen > 1) { - for (var j = 0; j < idxMapValLen; j++) { - this._add && this._add(newIdxMapVal[j]); - } - } else if (idxMapValLen === 1) { - this._add && this._add(newIdxMapVal); - } // Support both `newDataKeyArr` are duplication removed or not removed. - - - newDataIndexMap[newKey] = null; - } - }; - - DataDiffer.prototype._initIndexMap = function (arr, // Can be null. - map, // In 'byKey', the output `keyArr` is duplication removed. - // In 'byIndex', the output `keyArr` is not duplication removed and - // its indices are accurately corresponding to `arr`. - keyArr, keyGetterName) { - var cbModeMultiple = this._diffModeMultiple; - - for (var i = 0; i < arr.length; i++) { - // Add prefix to avoid conflict with Object.prototype. - var key = '_ec_' + this[keyGetterName](arr[i], i); - - if (!cbModeMultiple) { - keyArr[i] = key; - } - - if (!map) { - continue; - } - - var idxMapVal = map[key]; - var idxMapValLen = dataIndexMapValueLength(idxMapVal); - - if (idxMapValLen === 0) { - // Simple optimize: in most cases, one index has one key, - // do not need array. - map[key] = i; - - if (cbModeMultiple) { - keyArr.push(key); - } - } else if (idxMapValLen === 1) { - map[key] = [idxMapVal, i]; - } else { - idxMapVal.push(i); - } - } - }; - - return DataDiffer; - }(); - - var DimensionUserOuput = - /** @class */ - function () { - function DimensionUserOuput(encode, dimRequest) { - this._encode = encode; - this._schema = dimRequest; - } - - DimensionUserOuput.prototype.get = function () { - return { - // Do not generate full dimension name until fist used. - fullDimensions: this._getFullDimensionNames(), - encode: this._encode - }; - }; - /** - * Get all data store dimension names. - * Theoretically a series data store is defined both by series and used dataset (if any). - * If some dimensions are omitted for performance reason in `this.dimensions`, - * the dimension name may not be auto-generated if user does not specify a dimension name. - * In this case, the dimension name is `null`/`undefined`. - */ - - - DimensionUserOuput.prototype._getFullDimensionNames = function () { - if (!this._cachedDimNames) { - this._cachedDimNames = this._schema ? this._schema.makeOutputDimensionNames() : []; - } - - return this._cachedDimNames; - }; - - return DimensionUserOuput; - }(); - function summarizeDimensions(data, schema) { - var summary = {}; - var encode = summary.encode = {}; - var notExtraCoordDimMap = createHashMap(); - var defaultedLabel = []; - var defaultedTooltip = []; - var userOutputEncode = {}; - each(data.dimensions, function (dimName) { - var dimItem = data.getDimensionInfo(dimName); - var coordDim = dimItem.coordDim; - - if (coordDim) { - if ("development" !== 'production') { - assert(VISUAL_DIMENSIONS.get(coordDim) == null); - } - - var coordDimIndex = dimItem.coordDimIndex; - getOrCreateEncodeArr(encode, coordDim)[coordDimIndex] = dimName; - - if (!dimItem.isExtraCoord) { - notExtraCoordDimMap.set(coordDim, 1); // Use the last coord dim (and label friendly) as default label, - // because when dataset is used, it is hard to guess which dimension - // can be value dimension. If both show x, y on label is not look good, - // and conventionally y axis is focused more. - - if (mayLabelDimType(dimItem.type)) { - defaultedLabel[0] = dimName; - } // User output encode do not contain generated coords. - // And it only has index. User can use index to retrieve value from the raw item array. - - - getOrCreateEncodeArr(userOutputEncode, coordDim)[coordDimIndex] = data.getDimensionIndex(dimItem.name); - } - - if (dimItem.defaultTooltip) { - defaultedTooltip.push(dimName); - } - } - - VISUAL_DIMENSIONS.each(function (v, otherDim) { - var encodeArr = getOrCreateEncodeArr(encode, otherDim); - var dimIndex = dimItem.otherDims[otherDim]; - - if (dimIndex != null && dimIndex !== false) { - encodeArr[dimIndex] = dimItem.name; - } - }); - }); - var dataDimsOnCoord = []; - var encodeFirstDimNotExtra = {}; - notExtraCoordDimMap.each(function (v, coordDim) { - var dimArr = encode[coordDim]; - encodeFirstDimNotExtra[coordDim] = dimArr[0]; // Not necessary to remove duplicate, because a data - // dim canot on more than one coordDim. - - dataDimsOnCoord = dataDimsOnCoord.concat(dimArr); - }); - summary.dataDimsOnCoord = dataDimsOnCoord; - summary.dataDimIndicesOnCoord = map(dataDimsOnCoord, function (dimName) { - return data.getDimensionInfo(dimName).storeDimIndex; - }); - summary.encodeFirstDimNotExtra = encodeFirstDimNotExtra; - var encodeLabel = encode.label; // FIXME `encode.label` is not recommended, because formatter cannot be set - // in this way. Use label.formatter instead. Maybe remove this approach someday. - - if (encodeLabel && encodeLabel.length) { - defaultedLabel = encodeLabel.slice(); - } - - var encodeTooltip = encode.tooltip; - - if (encodeTooltip && encodeTooltip.length) { - defaultedTooltip = encodeTooltip.slice(); - } else if (!defaultedTooltip.length) { - defaultedTooltip = defaultedLabel.slice(); - } - - encode.defaultedLabel = defaultedLabel; - encode.defaultedTooltip = defaultedTooltip; - summary.userOutput = new DimensionUserOuput(userOutputEncode, schema); - return summary; - } - - function getOrCreateEncodeArr(encode, dim) { - if (!encode.hasOwnProperty(dim)) { - encode[dim] = []; - } - - return encode[dim]; - } // FIXME:TS should be type `AxisType` - - - function getDimensionTypeByAxis(axisType) { - return axisType === 'category' ? 'ordinal' : axisType === 'time' ? 'time' : 'float'; - } - - function mayLabelDimType(dimType) { - // In most cases, ordinal and time do not suitable for label. - // Ordinal info can be displayed on axis. Time is too long. - return !(dimType === 'ordinal' || dimType === 'time'); - } // function findTheLastDimMayLabel(data) { - // // Get last value dim - // let dimensions = data.dimensions.slice(); - // let valueType; - // let valueDim; - // while (dimensions.length && ( - // valueDim = dimensions.pop(), - // valueType = data.getDimensionInfo(valueDim).type, - // valueType === 'ordinal' || valueType === 'time' - // )) {} // jshint ignore:line - // return valueDim; - // } - - var SeriesDimensionDefine = - /** @class */ - function () { - /** - * @param opt All of the fields will be shallow copied. - */ - function SeriesDimensionDefine(opt) { - /** - * The format of `otherDims` is: - * ```js - * { - * tooltip?: number - * label?: number - * itemName?: number - * seriesName?: number - * } - * ``` - * - * A `series.encode` can specified these fields: - * ```js - * encode: { - * // "3, 1, 5" is the index of data dimension. - * tooltip: [3, 1, 5], - * label: [0, 3], - * ... - * } - * ``` - * `otherDims` is the parse result of the `series.encode` above, like: - * ```js - * // Suppose the index of this data dimension is `3`. - * this.otherDims = { - * // `3` is at the index `0` of the `encode.tooltip` - * tooltip: 0, - * // `3` is at the index `1` of the `encode.label` - * label: 1 - * }; - * ``` - * - * This prop should never be `null`/`undefined` after initialized. - */ - this.otherDims = {}; - - if (opt != null) { - extend(this, opt); - } - } - - return SeriesDimensionDefine; - }(); - - var inner$4 = makeInner(); - var dimTypeShort = { - float: 'f', - int: 'i', - ordinal: 'o', - number: 'n', - time: 't' - }; - /** - * Represents the dimension requirement of a series. - * - * NOTICE: - * When there are too many dimensions in dataset and many series, only the used dimensions - * (i.e., used by coord sys and declared in `series.encode`) are add to `dimensionDefineList`. - * But users may query data by other unused dimension names. - * In this case, users can only query data if and only if they have defined dimension names - * via ec option, so we provide `getDimensionIndexFromSource`, which only query them from - * `source` dimensions. - */ - - var SeriesDataSchema = - /** @class */ - function () { - function SeriesDataSchema(opt) { - this.dimensions = opt.dimensions; - this._dimOmitted = opt.dimensionOmitted; - this.source = opt.source; - this._fullDimCount = opt.fullDimensionCount; - - this._updateDimOmitted(opt.dimensionOmitted); - } - - SeriesDataSchema.prototype.isDimensionOmitted = function () { - return this._dimOmitted; - }; - - SeriesDataSchema.prototype._updateDimOmitted = function (dimensionOmitted) { - this._dimOmitted = dimensionOmitted; - - if (!dimensionOmitted) { - return; - } - - if (!this._dimNameMap) { - this._dimNameMap = ensureSourceDimNameMap(this.source); - } - }; - /** - * @caution Can only be used when `dimensionOmitted: true`. - * - * Get index by user defined dimension name (i.e., not internal generate name). - * That is, get index from `dimensionsDefine`. - * If no `dimensionsDefine`, or no name get, return -1. - */ - - - SeriesDataSchema.prototype.getSourceDimensionIndex = function (dimName) { - return retrieve2(this._dimNameMap.get(dimName), -1); - }; - /** - * @caution Can only be used when `dimensionOmitted: true`. - * - * Notice: may return `null`/`undefined` if user not specify dimension names. - */ - - - SeriesDataSchema.prototype.getSourceDimension = function (dimIndex) { - var dimensionsDefine = this.source.dimensionsDefine; - - if (dimensionsDefine) { - return dimensionsDefine[dimIndex]; - } - }; - - SeriesDataSchema.prototype.makeStoreSchema = function () { - var dimCount = this._fullDimCount; - var willRetrieveDataByName = shouldRetrieveDataByName(this.source); - var makeHashStrict = !shouldOmitUnusedDimensions(dimCount); // If source don't have dimensions or series don't omit unsed dimensions. - // Generate from seriesDimList directly - - var dimHash = ''; - var dims = []; - - for (var fullDimIdx = 0, seriesDimIdx = 0; fullDimIdx < dimCount; fullDimIdx++) { - var property = void 0; - var type = void 0; - var ordinalMeta = void 0; - var seriesDimDef = this.dimensions[seriesDimIdx]; // The list has been sorted by `storeDimIndex` asc. - - if (seriesDimDef && seriesDimDef.storeDimIndex === fullDimIdx) { - property = willRetrieveDataByName ? seriesDimDef.name : null; - type = seriesDimDef.type; - ordinalMeta = seriesDimDef.ordinalMeta; - seriesDimIdx++; - } else { - var sourceDimDef = this.getSourceDimension(fullDimIdx); - - if (sourceDimDef) { - property = willRetrieveDataByName ? sourceDimDef.name : null; - type = sourceDimDef.type; - } - } - - dims.push({ - property: property, - type: type, - ordinalMeta: ordinalMeta - }); // If retrieving data by index, - // use <index, type, ordinalMeta> to determine whether data can be shared. - // (Because in this case there might be no dimension name defined in dataset, but indices always exists). - // (Indices are always 0, 1, 2, ..., so we can ignore them to shorten the hash). - // Otherwise if retrieving data by property name (like `data: [{aa: 123, bb: 765}, ...]`), - // use <property, type, ordinalMeta> in hash. - - if (willRetrieveDataByName && property != null // For data stack, we have make sure each series has its own dim on this store. - // So we do not add property to hash to make sure they can share this store. - && (!seriesDimDef || !seriesDimDef.isCalculationCoord)) { - dimHash += makeHashStrict // Use escape character '`' in case that property name contains '$'. - ? property.replace(/\`/g, '`1').replace(/\$/g, '`2') // For better performance, when there are large dimensions, tolerant this defects that hardly meet. - : property; - } - - dimHash += '$'; - dimHash += dimTypeShort[type] || 'f'; - - if (ordinalMeta) { - dimHash += ordinalMeta.uid; - } - - dimHash += '$'; - } // Source from endpoint(usually series) will be read differently - // when seriesLayoutBy or startIndex(which is affected by sourceHeader) are different. - // So we use this three props as key. - - - var source = this.source; - var hash = [source.seriesLayoutBy, source.startIndex, dimHash].join('$$'); - return { - dimensions: dims, - hash: hash - }; - }; - - SeriesDataSchema.prototype.makeOutputDimensionNames = function () { - var result = []; - - for (var fullDimIdx = 0, seriesDimIdx = 0; fullDimIdx < this._fullDimCount; fullDimIdx++) { - var name_1 = void 0; - var seriesDimDef = this.dimensions[seriesDimIdx]; // The list has been sorted by `storeDimIndex` asc. - - if (seriesDimDef && seriesDimDef.storeDimIndex === fullDimIdx) { - if (!seriesDimDef.isCalculationCoord) { - name_1 = seriesDimDef.name; - } - - seriesDimIdx++; - } else { - var sourceDimDef = this.getSourceDimension(fullDimIdx); - - if (sourceDimDef) { - name_1 = sourceDimDef.name; - } - } - - result.push(name_1); - } - - return result; - }; - - SeriesDataSchema.prototype.appendCalculationDimension = function (dimDef) { - this.dimensions.push(dimDef); - dimDef.isCalculationCoord = true; - this._fullDimCount++; // If append dimension on a data store, consider the store - // might be shared by different series, series dimensions not - // really map to store dimensions. - - this._updateDimOmitted(true); - }; - - return SeriesDataSchema; - }(); - function isSeriesDataSchema(schema) { - return schema instanceof SeriesDataSchema; - } - function createDimNameMap(dimsDef) { - var dataDimNameMap = createHashMap(); - - for (var i = 0; i < (dimsDef || []).length; i++) { - var dimDefItemRaw = dimsDef[i]; - var userDimName = isObject(dimDefItemRaw) ? dimDefItemRaw.name : dimDefItemRaw; - - if (userDimName != null && dataDimNameMap.get(userDimName) == null) { - dataDimNameMap.set(userDimName, i); - } - } - - return dataDimNameMap; - } - function ensureSourceDimNameMap(source) { - var innerSource = inner$4(source); - return innerSource.dimNameMap || (innerSource.dimNameMap = createDimNameMap(source.dimensionsDefine)); - } - function shouldOmitUnusedDimensions(dimCount) { - return dimCount > 30; - } - - var isObject$2 = isObject; - var map$1 = map; - var CtorInt32Array$1 = typeof Int32Array === 'undefined' ? Array : Int32Array; // Use prefix to avoid index to be the same as otherIdList[idx], - // which will cause weird update animation. - - var ID_PREFIX = 'e\0\0'; - var INDEX_NOT_FOUND = -1; // type SeriesDimensionIndex = DimensionIndex; - - var TRANSFERABLE_PROPERTIES = ['hasItemOption', '_nameList', '_idList', '_invertedIndicesMap', '_dimSummary', 'userOutput', '_rawData', '_dimValueGetter', '_nameDimIdx', '_idDimIdx', '_nameRepeatCount']; - var CLONE_PROPERTIES = ['_approximateExtent']; // ----------------------------- - // Internal method declarations: - // ----------------------------- - - var prepareInvertedIndex; - var getId; - var getIdNameFromStore; - var normalizeDimensions; - var transferProperties; - var cloneListForMapAndSample; - var makeIdFromName; - - var SeriesData = - /** @class */ - function () { - /** - * @param dimensionsInput.dimensions - * For example, ['someDimName', {name: 'someDimName', type: 'someDimType'}, ...]. - * Dimensions should be concrete names like x, y, z, lng, lat, angle, radius - */ - function SeriesData(dimensionsInput, hostModel) { - this.type = 'list'; - this._dimOmitted = false; - this._nameList = []; - this._idList = []; // Models of data option is stored sparse for optimizing memory cost - // Never used yet (not used yet). - // private _optionModels: Model[] = []; - // Global visual properties after visual coding - - this._visual = {}; // Global layout properties. - - this._layout = {}; // Item visual properties after visual coding - - this._itemVisuals = []; // Item layout properties after layout - - this._itemLayouts = []; // Graphic elements - - this._graphicEls = []; // key: dim, value: extent - - this._approximateExtent = {}; - this._calculationInfo = {}; // Having detected that there is data item is non primitive type - // (in type `OptionDataItemObject`). - // Like `data: [ { value: xx, itemStyle: {...} }, ...]` - // At present it only happen in `SOURCE_FORMAT_ORIGINAL`. - - this.hasItemOption = false; // Methods that create a new list based on this list should be listed here. - // Notice that those method should `RETURN` the new list. - - this.TRANSFERABLE_METHODS = ['cloneShallow', 'downSample', 'lttbDownSample', 'map']; // Methods that change indices of this list should be listed here. - - this.CHANGABLE_METHODS = ['filterSelf', 'selectRange']; - this.DOWNSAMPLE_METHODS = ['downSample', 'lttbDownSample']; - var dimensions; - var assignStoreDimIdx = false; - - if (isSeriesDataSchema(dimensionsInput)) { - dimensions = dimensionsInput.dimensions; - this._dimOmitted = dimensionsInput.isDimensionOmitted(); - this._schema = dimensionsInput; - } else { - assignStoreDimIdx = true; - dimensions = dimensionsInput; - } - - dimensions = dimensions || ['x', 'y']; - var dimensionInfos = {}; - var dimensionNames = []; - var invertedIndicesMap = {}; - var needsHasOwn = false; - var emptyObj = {}; - - for (var i = 0; i < dimensions.length; i++) { - // Use the original dimensions[i], where other flag props may exists. - var dimInfoInput = dimensions[i]; - var dimensionInfo = isString(dimInfoInput) ? new SeriesDimensionDefine({ - name: dimInfoInput - }) : !(dimInfoInput instanceof SeriesDimensionDefine) ? new SeriesDimensionDefine(dimInfoInput) : dimInfoInput; - var dimensionName = dimensionInfo.name; - dimensionInfo.type = dimensionInfo.type || 'float'; - - if (!dimensionInfo.coordDim) { - dimensionInfo.coordDim = dimensionName; - dimensionInfo.coordDimIndex = 0; - } - - var otherDims = dimensionInfo.otherDims = dimensionInfo.otherDims || {}; - dimensionNames.push(dimensionName); - dimensionInfos[dimensionName] = dimensionInfo; - - if (emptyObj[dimensionName] != null) { - needsHasOwn = true; - } - - if (dimensionInfo.createInvertedIndices) { - invertedIndicesMap[dimensionName] = []; - } - - if (otherDims.itemName === 0) { - this._nameDimIdx = i; - } - - if (otherDims.itemId === 0) { - this._idDimIdx = i; - } - - if ("development" !== 'production') { - assert(assignStoreDimIdx || dimensionInfo.storeDimIndex >= 0); - } - - if (assignStoreDimIdx) { - dimensionInfo.storeDimIndex = i; - } - } - - this.dimensions = dimensionNames; - this._dimInfos = dimensionInfos; - - this._initGetDimensionInfo(needsHasOwn); - - this.hostModel = hostModel; - this._invertedIndicesMap = invertedIndicesMap; - - if (this._dimOmitted) { - var dimIdxToName_1 = this._dimIdxToName = createHashMap(); - each(dimensionNames, function (dimName) { - dimIdxToName_1.set(dimensionInfos[dimName].storeDimIndex, dimName); - }); - } - } - /** - * - * Get concrete dimension name by dimension name or dimension index. - * If input a dimension name, do not validate whether the dimension name exits. - * - * @caution - * @param dim Must make sure the dimension is `SeriesDimensionLoose`. - * Because only those dimensions will have auto-generated dimension names if not - * have a user-specified name, and other dimensions will get a return of null/undefined. - * - * @notice Because of this reason, should better use `getDimensionIndex` instead, for examples: - * ```js - * const val = data.getStore().get(data.getDimensionIndex(dim), dataIdx); - * ``` - * - * @return Concrete dim name. - */ - - - SeriesData.prototype.getDimension = function (dim) { - var dimIdx = this._recognizeDimIndex(dim); - - if (dimIdx == null) { - return dim; - } - - dimIdx = dim; - - if (!this._dimOmitted) { - return this.dimensions[dimIdx]; - } // Retrieve from series dimension definition because it probably contains - // generated dimension name (like 'x', 'y'). - - - var dimName = this._dimIdxToName.get(dimIdx); - - if (dimName != null) { - return dimName; - } - - var sourceDimDef = this._schema.getSourceDimension(dimIdx); - - if (sourceDimDef) { - return sourceDimDef.name; - } - }; - /** - * Get dimension index in data store. Return -1 if not found. - * Can be used to index value from getRawValue. - */ - - - SeriesData.prototype.getDimensionIndex = function (dim) { - var dimIdx = this._recognizeDimIndex(dim); - - if (dimIdx != null) { - return dimIdx; - } - - if (dim == null) { - return -1; - } - - var dimInfo = this._getDimInfo(dim); - - return dimInfo ? dimInfo.storeDimIndex : this._dimOmitted ? this._schema.getSourceDimensionIndex(dim) : -1; - }; - /** - * The meanings of the input parameter `dim`: - * - * + If dim is a number (e.g., `1`), it means the index of the dimension. - * For example, `getDimension(0)` will return 'x' or 'lng' or 'radius'. - * + If dim is a number-like string (e.g., `"1"`): - * + If there is the same concrete dim name defined in `series.dimensions` or `dataset.dimensions`, - * it means that concrete name. - * + If not, it will be converted to a number, which means the index of the dimension. - * (why? because of the backward compatibility. We have been tolerating number-like string in - * dimension setting, although now it seems that it is not a good idea.) - * For example, `visualMap[i].dimension: "1"` is the same meaning as `visualMap[i].dimension: 1`, - * if no dimension name is defined as `"1"`. - * + If dim is a not-number-like string, it means the concrete dim name. - * For example, it can be be default name `"x"`, `"y"`, `"z"`, `"lng"`, `"lat"`, `"angle"`, `"radius"`, - * or customized in `dimensions` property of option like `"age"`. - * - * @return recognized `DimensionIndex`. Otherwise return null/undefined (means that dim is `DimensionName`). - */ - - - SeriesData.prototype._recognizeDimIndex = function (dim) { - if (isNumber(dim) // If being a number-like string but not being defined as a dimension name. - || dim != null && !isNaN(dim) && !this._getDimInfo(dim) && (!this._dimOmitted || this._schema.getSourceDimensionIndex(dim) < 0)) { - return +dim; - } - }; - - SeriesData.prototype._getStoreDimIndex = function (dim) { - var dimIdx = this.getDimensionIndex(dim); - - if ("development" !== 'production') { - if (dimIdx == null) { - throw new Error('Unknown dimension ' + dim); - } - } - - return dimIdx; - }; - /** - * Get type and calculation info of particular dimension - * @param dim - * Dimension can be concrete names like x, y, z, lng, lat, angle, radius - * Or a ordinal number. For example getDimensionInfo(0) will return 'x' or 'lng' or 'radius' - */ - - - SeriesData.prototype.getDimensionInfo = function (dim) { - // Do not clone, because there may be categories in dimInfo. - return this._getDimInfo(this.getDimension(dim)); - }; - - SeriesData.prototype._initGetDimensionInfo = function (needsHasOwn) { - var dimensionInfos = this._dimInfos; - this._getDimInfo = needsHasOwn ? function (dimName) { - return dimensionInfos.hasOwnProperty(dimName) ? dimensionInfos[dimName] : undefined; - } : function (dimName) { - return dimensionInfos[dimName]; - }; - }; - /** - * concrete dimension name list on coord. - */ - - - SeriesData.prototype.getDimensionsOnCoord = function () { - return this._dimSummary.dataDimsOnCoord.slice(); - }; - - SeriesData.prototype.mapDimension = function (coordDim, idx) { - var dimensionsSummary = this._dimSummary; - - if (idx == null) { - return dimensionsSummary.encodeFirstDimNotExtra[coordDim]; - } - - var dims = dimensionsSummary.encode[coordDim]; - return dims ? dims[idx] : null; - }; - - SeriesData.prototype.mapDimensionsAll = function (coordDim) { - var dimensionsSummary = this._dimSummary; - var dims = dimensionsSummary.encode[coordDim]; - return (dims || []).slice(); - }; - - SeriesData.prototype.getStore = function () { - return this._store; - }; - /** - * Initialize from data - * @param data source or data or data store. - * @param nameList The name of a datum is used on data diff and - * default label/tooltip. - * A name can be specified in encode.itemName, - * or dataItem.name (only for series option data), - * or provided in nameList from outside. - */ - - - SeriesData.prototype.initData = function (data, nameList, dimValueGetter) { - var _this = this; - - var store; - - if (data instanceof DataStore) { - store = data; - } - - if (!store) { - var dimensions = this.dimensions; - var provider = isSourceInstance(data) || isArrayLike(data) ? new DefaultDataProvider(data, dimensions.length) : data; - store = new DataStore(); - var dimensionInfos = map$1(dimensions, function (dimName) { - return { - type: _this._dimInfos[dimName].type, - property: dimName - }; - }); - store.initData(provider, dimensionInfos, dimValueGetter); - } - - this._store = store; // Reset - - this._nameList = (nameList || []).slice(); - this._idList = []; - this._nameRepeatCount = {}; - - this._doInit(0, store.count()); // Cache summary info for fast visit. See "dimensionHelper". - // Needs to be initialized after store is prepared. - - - this._dimSummary = summarizeDimensions(this, this._schema); - this.userOutput = this._dimSummary.userOutput; - }; - /** - * Caution: Can be only called on raw data (before `this._indices` created). - */ - - - SeriesData.prototype.appendData = function (data) { - var range = this._store.appendData(data); - - this._doInit(range[0], range[1]); - }; - /** - * Caution: Can be only called on raw data (before `this._indices` created). - * This method does not modify `rawData` (`dataProvider`), but only - * add values to store. - * - * The final count will be increased by `Math.max(values.length, names.length)`. - * - * @param values That is the SourceType: 'arrayRows', like - * [ - * [12, 33, 44], - * [NaN, 43, 1], - * ['-', 'asdf', 0] - * ] - * Each item is exactly corresponding to a dimension. - */ - - - SeriesData.prototype.appendValues = function (values, names) { - var _a = this._store.appendValues(values, names.length), - start = _a.start, - end = _a.end; - - var shouldMakeIdFromName = this._shouldMakeIdFromName(); - - this._updateOrdinalMeta(); - - if (names) { - for (var idx = start; idx < end; idx++) { - var sourceIdx = idx - start; - this._nameList[idx] = names[sourceIdx]; - - if (shouldMakeIdFromName) { - makeIdFromName(this, idx); - } - } - } - }; - - SeriesData.prototype._updateOrdinalMeta = function () { - var store = this._store; - var dimensions = this.dimensions; - - for (var i = 0; i < dimensions.length; i++) { - var dimInfo = this._dimInfos[dimensions[i]]; - - if (dimInfo.ordinalMeta) { - store.collectOrdinalMeta(dimInfo.storeDimIndex, dimInfo.ordinalMeta); - } - } - }; - - SeriesData.prototype._shouldMakeIdFromName = function () { - var provider = this._store.getProvider(); - - return this._idDimIdx == null && provider.getSource().sourceFormat !== SOURCE_FORMAT_TYPED_ARRAY && !provider.fillStorage; - }; - - SeriesData.prototype._doInit = function (start, end) { - if (start >= end) { - return; - } - - var store = this._store; - var provider = store.getProvider(); - - this._updateOrdinalMeta(); - - var nameList = this._nameList; - var idList = this._idList; - var sourceFormat = provider.getSource().sourceFormat; - var isFormatOriginal = sourceFormat === SOURCE_FORMAT_ORIGINAL; // Each data item is value - // [1, 2] - // 2 - // Bar chart, line chart which uses category axis - // only gives the 'y' value. 'x' value is the indices of category - // Use a tempValue to normalize the value to be a (x, y) value - // If dataItem is {name: ...} or {id: ...}, it has highest priority. - // This kind of ids and names are always stored `_nameList` and `_idList`. - - if (isFormatOriginal && !provider.pure) { - var sharedDataItem = []; - - for (var idx = start; idx < end; idx++) { - // NOTICE: Try not to write things into dataItem - var dataItem = provider.getItem(idx, sharedDataItem); - - if (!this.hasItemOption && isDataItemOption(dataItem)) { - this.hasItemOption = true; - } - - if (dataItem) { - var itemName = dataItem.name; - - if (nameList[idx] == null && itemName != null) { - nameList[idx] = convertOptionIdName(itemName, null); - } - - var itemId = dataItem.id; - - if (idList[idx] == null && itemId != null) { - idList[idx] = convertOptionIdName(itemId, null); - } - } - } - } - - if (this._shouldMakeIdFromName()) { - for (var idx = start; idx < end; idx++) { - makeIdFromName(this, idx); - } - } - - prepareInvertedIndex(this); - }; - /** - * PENDING: In fact currently this function is only used to short-circuit - * the calling of `scale.unionExtentFromData` when data have been filtered by modules - * like "dataZoom". `scale.unionExtentFromData` is used to calculate data extent for series on - * an axis, but if a "axis related data filter module" is used, the extent of the axis have - * been fixed and no need to calling `scale.unionExtentFromData` actually. - * But if we add "custom data filter" in future, which is not "axis related", this method may - * be still needed. - * - * Optimize for the scenario that data is filtered by a given extent. - * Consider that if data amount is more than hundreds of thousand, - * extent calculation will cost more than 10ms and the cache will - * be erased because of the filtering. - */ - - - SeriesData.prototype.getApproximateExtent = function (dim) { - return this._approximateExtent[dim] || this._store.getDataExtent(this._getStoreDimIndex(dim)); - }; - /** - * Calculate extent on a filtered data might be time consuming. - * Approximate extent is only used for: calculate extent of filtered data outside. - */ - - - SeriesData.prototype.setApproximateExtent = function (extent, dim) { - dim = this.getDimension(dim); - this._approximateExtent[dim] = extent.slice(); - }; - - SeriesData.prototype.getCalculationInfo = function (key) { - return this._calculationInfo[key]; - }; - - SeriesData.prototype.setCalculationInfo = function (key, value) { - isObject$2(key) ? extend(this._calculationInfo, key) : this._calculationInfo[key] = value; - }; - /** - * @return Never be null/undefined. `number` will be converted to string. Because: - * In most cases, name is used in display, where returning a string is more convenient. - * In other cases, name is used in query (see `indexOfName`), where we can keep the - * rule that name `2` equals to name `'2'`. - */ - - - SeriesData.prototype.getName = function (idx) { - var rawIndex = this.getRawIndex(idx); - var name = this._nameList[rawIndex]; - - if (name == null && this._nameDimIdx != null) { - name = getIdNameFromStore(this, this._nameDimIdx, rawIndex); - } - - if (name == null) { - name = ''; - } - - return name; - }; - - SeriesData.prototype._getCategory = function (dimIdx, idx) { - var ordinal = this._store.get(dimIdx, idx); - - var ordinalMeta = this._store.getOrdinalMeta(dimIdx); - - if (ordinalMeta) { - return ordinalMeta.categories[ordinal]; - } - - return ordinal; - }; - /** - * @return Never null/undefined. `number` will be converted to string. Because: - * In all cases having encountered at present, id is used in making diff comparison, which - * are usually based on hash map. We can keep the rule that the internal id are always string - * (treat `2` is the same as `'2'`) to make the related logic simple. - */ - - - SeriesData.prototype.getId = function (idx) { - return getId(this, this.getRawIndex(idx)); - }; - - SeriesData.prototype.count = function () { - return this._store.count(); - }; - /** - * Get value. Return NaN if idx is out of range. - * - * @notice Should better to use `data.getStore().get(dimIndex, dataIdx)` instead. - */ - - - SeriesData.prototype.get = function (dim, idx) { - var store = this._store; - var dimInfo = this._dimInfos[dim]; - - if (dimInfo) { - return store.get(dimInfo.storeDimIndex, idx); - } - }; - /** - * @notice Should better to use `data.getStore().getByRawIndex(dimIndex, dataIdx)` instead. - */ - - - SeriesData.prototype.getByRawIndex = function (dim, rawIdx) { - var store = this._store; - var dimInfo = this._dimInfos[dim]; - - if (dimInfo) { - return store.getByRawIndex(dimInfo.storeDimIndex, rawIdx); - } - }; - - SeriesData.prototype.getIndices = function () { - return this._store.getIndices(); - }; - - SeriesData.prototype.getDataExtent = function (dim) { - return this._store.getDataExtent(this._getStoreDimIndex(dim)); - }; - - SeriesData.prototype.getSum = function (dim) { - return this._store.getSum(this._getStoreDimIndex(dim)); - }; - - SeriesData.prototype.getMedian = function (dim) { - return this._store.getMedian(this._getStoreDimIndex(dim)); - }; - - SeriesData.prototype.getValues = function (dimensions, idx) { - var _this = this; - - var store = this._store; - return isArray(dimensions) ? store.getValues(map$1(dimensions, function (dim) { - return _this._getStoreDimIndex(dim); - }), idx) : store.getValues(dimensions); - }; - /** - * If value is NaN. Including '-' - * Only check the coord dimensions. - */ - - - SeriesData.prototype.hasValue = function (idx) { - var dataDimIndicesOnCoord = this._dimSummary.dataDimIndicesOnCoord; - - for (var i = 0, len = dataDimIndicesOnCoord.length; i < len; i++) { - // Ordinal type originally can be string or number. - // But when an ordinal type is used on coord, it can - // not be string but only number. So we can also use isNaN. - if (isNaN(this._store.get(dataDimIndicesOnCoord[i], idx))) { - return false; - } - } - - return true; - }; - /** - * Retrieve the index with given name - */ - - - SeriesData.prototype.indexOfName = function (name) { - for (var i = 0, len = this._store.count(); i < len; i++) { - if (this.getName(i) === name) { - return i; - } - } - - return -1; - }; - - SeriesData.prototype.getRawIndex = function (idx) { - return this._store.getRawIndex(idx); - }; - - SeriesData.prototype.indexOfRawIndex = function (rawIndex) { - return this._store.indexOfRawIndex(rawIndex); - }; - /** - * Only support the dimension which inverted index created. - * Do not support other cases until required. - * @param dim concrete dim - * @param value ordinal index - * @return rawIndex - */ - - - SeriesData.prototype.rawIndexOf = function (dim, value) { - var invertedIndices = dim && this._invertedIndicesMap[dim]; - - if ("development" !== 'production') { - if (!invertedIndices) { - throw new Error('Do not supported yet'); - } - } - - var rawIndex = invertedIndices[value]; - - if (rawIndex == null || isNaN(rawIndex)) { - return INDEX_NOT_FOUND; - } - - return rawIndex; - }; - /** - * Retrieve the index of nearest value - * @param dim - * @param value - * @param [maxDistance=Infinity] - * @return If and only if multiple indices has - * the same value, they are put to the result. - */ - - - SeriesData.prototype.indicesOfNearest = function (dim, value, maxDistance) { - return this._store.indicesOfNearest(this._getStoreDimIndex(dim), value, maxDistance); - }; - - SeriesData.prototype.each = function (dims, cb, ctx) { - - if (isFunction(dims)) { - ctx = cb; - cb = dims; - dims = []; - } // ctxCompat just for compat echarts3 - - - var fCtx = ctx || this; - var dimIndices = map$1(normalizeDimensions(dims), this._getStoreDimIndex, this); - - this._store.each(dimIndices, fCtx ? bind(cb, fCtx) : cb); - }; - - SeriesData.prototype.filterSelf = function (dims, cb, ctx) { - - if (isFunction(dims)) { - ctx = cb; - cb = dims; - dims = []; - } // ctxCompat just for compat echarts3 - - - var fCtx = ctx || this; - var dimIndices = map$1(normalizeDimensions(dims), this._getStoreDimIndex, this); - this._store = this._store.filter(dimIndices, fCtx ? bind(cb, fCtx) : cb); - return this; - }; - /** - * Select data in range. (For optimization of filter) - * (Manually inline code, support 5 million data filtering in data zoom.) - */ - - - SeriesData.prototype.selectRange = function (range) { - - var _this = this; - - var innerRange = {}; - var dims = keys(range); - each(dims, function (dim) { - var dimIdx = _this._getStoreDimIndex(dim); - - innerRange[dimIdx] = range[dim]; - }); - this._store = this._store.selectRange(innerRange); - return this; - }; - /* eslint-enable max-len */ - - - SeriesData.prototype.mapArray = function (dims, cb, ctx) { - - if (isFunction(dims)) { - ctx = cb; - cb = dims; - dims = []; - } // ctxCompat just for compat echarts3 - - - ctx = ctx || this; - var result = []; - this.each(dims, function () { - result.push(cb && cb.apply(this, arguments)); - }, ctx); - return result; - }; - - SeriesData.prototype.map = function (dims, cb, ctx, ctxCompat) { - - var fCtx = ctx || ctxCompat || this; - var dimIndices = map$1(normalizeDimensions(dims), this._getStoreDimIndex, this); - var list = cloneListForMapAndSample(this); - list._store = this._store.map(dimIndices, fCtx ? bind(cb, fCtx) : cb); - return list; - }; - - SeriesData.prototype.modify = function (dims, cb, ctx, ctxCompat) { - var _this = this; // ctxCompat just for compat echarts3 - - - var fCtx = ctx || ctxCompat || this; - - if ("development" !== 'production') { - each(normalizeDimensions(dims), function (dim) { - var dimInfo = _this.getDimensionInfo(dim); - - if (!dimInfo.isCalculationCoord) { - console.error('Danger: only stack dimension can be modified'); - } - }); - } - - var dimIndices = map$1(normalizeDimensions(dims), this._getStoreDimIndex, this); // If do shallow clone here, if there are too many stacked series, - // it still cost lots of memory, because `_store.dimensions` are not shared. - // We should consider there probably be shallow clone happen in each series - // in consequent filter/map. - - this._store.modify(dimIndices, fCtx ? bind(cb, fCtx) : cb); - }; - /** - * Large data down sampling on given dimension - * @param sampleIndex Sample index for name and id - */ - - - SeriesData.prototype.downSample = function (dimension, rate, sampleValue, sampleIndex) { - var list = cloneListForMapAndSample(this); - list._store = this._store.downSample(this._getStoreDimIndex(dimension), rate, sampleValue, sampleIndex); - return list; - }; - /** - * Large data down sampling using largest-triangle-three-buckets - * @param {string} valueDimension - * @param {number} targetCount - */ - - - SeriesData.prototype.lttbDownSample = function (valueDimension, rate) { - var list = cloneListForMapAndSample(this); - list._store = this._store.lttbDownSample(this._getStoreDimIndex(valueDimension), rate); - return list; - }; - - SeriesData.prototype.getRawDataItem = function (idx) { - return this._store.getRawDataItem(idx); - }; - /** - * Get model of one data item. - */ - // TODO: Type of data item - - - SeriesData.prototype.getItemModel = function (idx) { - var hostModel = this.hostModel; - var dataItem = this.getRawDataItem(idx); - return new Model(dataItem, hostModel, hostModel && hostModel.ecModel); - }; - /** - * Create a data differ - */ - - - SeriesData.prototype.diff = function (otherList) { - var thisList = this; - return new DataDiffer(otherList ? otherList.getStore().getIndices() : [], this.getStore().getIndices(), function (idx) { - return getId(otherList, idx); - }, function (idx) { - return getId(thisList, idx); - }); - }; - /** - * Get visual property. - */ - - - SeriesData.prototype.getVisual = function (key) { - var visual = this._visual; - return visual && visual[key]; - }; - - SeriesData.prototype.setVisual = function (kvObj, val) { - this._visual = this._visual || {}; - - if (isObject$2(kvObj)) { - extend(this._visual, kvObj); - } else { - this._visual[kvObj] = val; - } - }; - /** - * Get visual property of single data item - */ - // eslint-disable-next-line - - - SeriesData.prototype.getItemVisual = function (idx, key) { - var itemVisual = this._itemVisuals[idx]; - var val = itemVisual && itemVisual[key]; - - if (val == null) { - // Use global visual property - return this.getVisual(key); - } - - return val; - }; - /** - * If exists visual property of single data item - */ - - - SeriesData.prototype.hasItemVisual = function () { - return this._itemVisuals.length > 0; - }; - /** - * Make sure itemVisual property is unique - */ - // TODO: use key to save visual to reduce memory. - - - SeriesData.prototype.ensureUniqueItemVisual = function (idx, key) { - var itemVisuals = this._itemVisuals; - var itemVisual = itemVisuals[idx]; - - if (!itemVisual) { - itemVisual = itemVisuals[idx] = {}; - } - - var val = itemVisual[key]; - - if (val == null) { - val = this.getVisual(key); // TODO Performance? - - if (isArray(val)) { - val = val.slice(); - } else if (isObject$2(val)) { - val = extend({}, val); - } - - itemVisual[key] = val; - } - - return val; - }; // eslint-disable-next-line - - - SeriesData.prototype.setItemVisual = function (idx, key, value) { - var itemVisual = this._itemVisuals[idx] || {}; - this._itemVisuals[idx] = itemVisual; - - if (isObject$2(key)) { - extend(itemVisual, key); - } else { - itemVisual[key] = value; - } - }; - /** - * Clear itemVisuals and list visual. - */ - - - SeriesData.prototype.clearAllVisual = function () { - this._visual = {}; - this._itemVisuals = []; - }; - - SeriesData.prototype.setLayout = function (key, val) { - isObject$2(key) ? extend(this._layout, key) : this._layout[key] = val; - }; - /** - * Get layout property. - */ - - - SeriesData.prototype.getLayout = function (key) { - return this._layout[key]; - }; - /** - * Get layout of single data item - */ - - - SeriesData.prototype.getItemLayout = function (idx) { - return this._itemLayouts[idx]; - }; - /** - * Set layout of single data item - */ - - - SeriesData.prototype.setItemLayout = function (idx, layout, merge) { - this._itemLayouts[idx] = merge ? extend(this._itemLayouts[idx] || {}, layout) : layout; - }; - /** - * Clear all layout of single data item - */ - - - SeriesData.prototype.clearItemLayouts = function () { - this._itemLayouts.length = 0; - }; - /** - * Set graphic element relative to data. It can be set as null - */ - - - SeriesData.prototype.setItemGraphicEl = function (idx, el) { - var seriesIndex = this.hostModel && this.hostModel.seriesIndex; - setCommonECData(seriesIndex, this.dataType, idx, el); - this._graphicEls[idx] = el; - }; - - SeriesData.prototype.getItemGraphicEl = function (idx) { - return this._graphicEls[idx]; - }; - - SeriesData.prototype.eachItemGraphicEl = function (cb, context) { - each(this._graphicEls, function (el, idx) { - if (el) { - cb && cb.call(context, el, idx); - } - }); - }; - /** - * Shallow clone a new list except visual and layout properties, and graph elements. - * New list only change the indices. - */ - - - SeriesData.prototype.cloneShallow = function (list) { - if (!list) { - list = new SeriesData(this._schema ? this._schema : map$1(this.dimensions, this._getDimInfo, this), this.hostModel); - } - - transferProperties(list, this); - list._store = this._store; - return list; - }; - /** - * Wrap some method to add more feature - */ - - - SeriesData.prototype.wrapMethod = function (methodName, injectFunction) { - var originalMethod = this[methodName]; - - if (!isFunction(originalMethod)) { - return; - } - - this.__wrappedMethods = this.__wrappedMethods || []; - - this.__wrappedMethods.push(methodName); - - this[methodName] = function () { - var res = originalMethod.apply(this, arguments); - return injectFunction.apply(this, [res].concat(slice(arguments))); - }; - }; // ---------------------------------------------------------- - // A work around for internal method visiting private member. - // ---------------------------------------------------------- - - - SeriesData.internalField = function () { - prepareInvertedIndex = function (data) { - var invertedIndicesMap = data._invertedIndicesMap; - each(invertedIndicesMap, function (invertedIndices, dim) { - var dimInfo = data._dimInfos[dim]; // Currently, only dimensions that has ordinalMeta can create inverted indices. - - var ordinalMeta = dimInfo.ordinalMeta; - var store = data._store; - - if (ordinalMeta) { - invertedIndices = invertedIndicesMap[dim] = new CtorInt32Array$1(ordinalMeta.categories.length); // The default value of TypedArray is 0. To avoid miss - // mapping to 0, we should set it as INDEX_NOT_FOUND. - - for (var i = 0; i < invertedIndices.length; i++) { - invertedIndices[i] = INDEX_NOT_FOUND; - } - - for (var i = 0; i < store.count(); i++) { - // Only support the case that all values are distinct. - invertedIndices[store.get(dimInfo.storeDimIndex, i)] = i; - } - } - }); - }; - - getIdNameFromStore = function (data, dimIdx, idx) { - return convertOptionIdName(data._getCategory(dimIdx, idx), null); - }; - /** - * @see the comment of `List['getId']`. - */ - - - getId = function (data, rawIndex) { - var id = data._idList[rawIndex]; - - if (id == null && data._idDimIdx != null) { - id = getIdNameFromStore(data, data._idDimIdx, rawIndex); - } - - if (id == null) { - id = ID_PREFIX + rawIndex; - } - - return id; - }; - - normalizeDimensions = function (dimensions) { - if (!isArray(dimensions)) { - dimensions = dimensions != null ? [dimensions] : []; - } - - return dimensions; - }; - /** - * Data in excludeDimensions is copied, otherwise transferred. - */ - - - cloneListForMapAndSample = function (original) { - var list = new SeriesData(original._schema ? original._schema : map$1(original.dimensions, original._getDimInfo, original), original.hostModel); // FIXME If needs stackedOn, value may already been stacked - - transferProperties(list, original); - return list; - }; - - transferProperties = function (target, source) { - each(TRANSFERABLE_PROPERTIES.concat(source.__wrappedMethods || []), function (propName) { - if (source.hasOwnProperty(propName)) { - target[propName] = source[propName]; - } - }); - target.__wrappedMethods = source.__wrappedMethods; - each(CLONE_PROPERTIES, function (propName) { - target[propName] = clone(source[propName]); - }); - target._calculationInfo = extend({}, source._calculationInfo); - }; - - makeIdFromName = function (data, idx) { - var nameList = data._nameList; - var idList = data._idList; - var nameDimIdx = data._nameDimIdx; - var idDimIdx = data._idDimIdx; - var name = nameList[idx]; - var id = idList[idx]; - - if (name == null && nameDimIdx != null) { - nameList[idx] = name = getIdNameFromStore(data, nameDimIdx, idx); - } - - if (id == null && idDimIdx != null) { - idList[idx] = id = getIdNameFromStore(data, idDimIdx, idx); - } - - if (id == null && name != null) { - var nameRepeatCount = data._nameRepeatCount; - var nmCnt = nameRepeatCount[name] = (nameRepeatCount[name] || 0) + 1; - id = name; - - if (nmCnt > 1) { - id += '__ec__' + nmCnt; - } - - idList[idx] = id; - } - }; - }(); - - return SeriesData; - }(); - - /** - * For outside usage compat (like echarts-gl are using it). - */ - - function createDimensions(source, opt) { - return prepareSeriesDataSchema(source, opt).dimensions; - } - /** - * This method builds the relationship between: - * + "what the coord sys or series requires (see `coordDimensions`)", - * + "what the user defines (in `encode` and `dimensions`, see `opt.dimensionsDefine` and `opt.encodeDefine`)" - * + "what the data source provids (see `source`)". - * - * Some guess strategy will be adapted if user does not define something. - * If no 'value' dimension specified, the first no-named dimension will be - * named as 'value'. - * - * @return The results are always sorted by `storeDimIndex` asc. - */ - - function prepareSeriesDataSchema( // TODO: TYPE completeDimensions type - source, opt) { - if (!isSourceInstance(source)) { - source = createSourceFromSeriesDataOption(source); - } - - opt = opt || {}; - var sysDims = opt.coordDimensions || []; - var dimsDef = opt.dimensionsDefine || source.dimensionsDefine || []; - var coordDimNameMap = createHashMap(); - var resultList = []; - var dimCount = getDimCount(source, sysDims, dimsDef, opt.dimensionsCount); // Try to ignore unused dimensions if sharing a high dimension datastore - // 30 is an experience value. - - var omitUnusedDimensions = opt.canOmitUnusedDimensions && shouldOmitUnusedDimensions(dimCount); - var isUsingSourceDimensionsDef = dimsDef === source.dimensionsDefine; - var dataDimNameMap = isUsingSourceDimensionsDef ? ensureSourceDimNameMap(source) : createDimNameMap(dimsDef); - var encodeDef = opt.encodeDefine; - - if (!encodeDef && opt.encodeDefaulter) { - encodeDef = opt.encodeDefaulter(source, dimCount); - } - - var encodeDefMap = createHashMap(encodeDef); - var indicesMap = new CtorInt32Array(dimCount); - - for (var i = 0; i < indicesMap.length; i++) { - indicesMap[i] = -1; - } - - function getResultItem(dimIdx) { - var idx = indicesMap[dimIdx]; - - if (idx < 0) { - var dimDefItemRaw = dimsDef[dimIdx]; - var dimDefItem = isObject(dimDefItemRaw) ? dimDefItemRaw : { - name: dimDefItemRaw - }; - var resultItem = new SeriesDimensionDefine(); - var userDimName = dimDefItem.name; - - if (userDimName != null && dataDimNameMap.get(userDimName) != null) { - // Only if `series.dimensions` is defined in option - // displayName, will be set, and dimension will be displayed vertically in - // tooltip by default. - resultItem.name = resultItem.displayName = userDimName; - } - - dimDefItem.type != null && (resultItem.type = dimDefItem.type); - dimDefItem.displayName != null && (resultItem.displayName = dimDefItem.displayName); - var newIdx = resultList.length; - indicesMap[dimIdx] = newIdx; - resultItem.storeDimIndex = dimIdx; - resultList.push(resultItem); - return resultItem; - } - - return resultList[idx]; - } - - if (!omitUnusedDimensions) { - for (var i = 0; i < dimCount; i++) { - getResultItem(i); - } - } // Set `coordDim` and `coordDimIndex` by `encodeDefMap` and normalize `encodeDefMap`. - - - encodeDefMap.each(function (dataDimsRaw, coordDim) { - var dataDims = normalizeToArray(dataDimsRaw).slice(); // Note: It is allowed that `dataDims.length` is `0`, e.g., options is - // `{encode: {x: -1, y: 1}}`. Should not filter anything in - // this case. - - if (dataDims.length === 1 && !isString(dataDims[0]) && dataDims[0] < 0) { - encodeDefMap.set(coordDim, false); - return; - } - - var validDataDims = encodeDefMap.set(coordDim, []); - each(dataDims, function (resultDimIdxOrName, idx) { - // The input resultDimIdx can be dim name or index. - var resultDimIdx = isString(resultDimIdxOrName) ? dataDimNameMap.get(resultDimIdxOrName) : resultDimIdxOrName; - - if (resultDimIdx != null && resultDimIdx < dimCount) { - validDataDims[idx] = resultDimIdx; - applyDim(getResultItem(resultDimIdx), coordDim, idx); - } - }); - }); // Apply templates and default order from `sysDims`. - - var availDimIdx = 0; - each(sysDims, function (sysDimItemRaw) { - var coordDim; - var sysDimItemDimsDef; - var sysDimItemOtherDims; - var sysDimItem; - - if (isString(sysDimItemRaw)) { - coordDim = sysDimItemRaw; - sysDimItem = {}; - } else { - sysDimItem = sysDimItemRaw; - coordDim = sysDimItem.name; - var ordinalMeta = sysDimItem.ordinalMeta; - sysDimItem.ordinalMeta = null; - sysDimItem = extend({}, sysDimItem); - sysDimItem.ordinalMeta = ordinalMeta; // `coordDimIndex` should not be set directly. - - sysDimItemDimsDef = sysDimItem.dimsDef; - sysDimItemOtherDims = sysDimItem.otherDims; - sysDimItem.name = sysDimItem.coordDim = sysDimItem.coordDimIndex = sysDimItem.dimsDef = sysDimItem.otherDims = null; - } - - var dataDims = encodeDefMap.get(coordDim); // negative resultDimIdx means no need to mapping. - - if (dataDims === false) { - return; - } - - dataDims = normalizeToArray(dataDims); // dimensions provides default dim sequences. - - if (!dataDims.length) { - for (var i = 0; i < (sysDimItemDimsDef && sysDimItemDimsDef.length || 1); i++) { - while (availDimIdx < dimCount && getResultItem(availDimIdx).coordDim != null) { - availDimIdx++; - } - - availDimIdx < dimCount && dataDims.push(availDimIdx++); - } - } // Apply templates. - - - each(dataDims, function (resultDimIdx, coordDimIndex) { - var resultItem = getResultItem(resultDimIdx); // Coordinate system has a higher priority on dim type than source. - - if (isUsingSourceDimensionsDef && sysDimItem.type != null) { - resultItem.type = sysDimItem.type; - } - - applyDim(defaults(resultItem, sysDimItem), coordDim, coordDimIndex); - - if (resultItem.name == null && sysDimItemDimsDef) { - var sysDimItemDimsDefItem = sysDimItemDimsDef[coordDimIndex]; - !isObject(sysDimItemDimsDefItem) && (sysDimItemDimsDefItem = { - name: sysDimItemDimsDefItem - }); - resultItem.name = resultItem.displayName = sysDimItemDimsDefItem.name; - resultItem.defaultTooltip = sysDimItemDimsDefItem.defaultTooltip; - } // FIXME refactor, currently only used in case: {otherDims: {tooltip: false}} - - - sysDimItemOtherDims && defaults(resultItem.otherDims, sysDimItemOtherDims); - }); - }); - - function applyDim(resultItem, coordDim, coordDimIndex) { - if (VISUAL_DIMENSIONS.get(coordDim) != null) { - resultItem.otherDims[coordDim] = coordDimIndex; - } else { - resultItem.coordDim = coordDim; - resultItem.coordDimIndex = coordDimIndex; - coordDimNameMap.set(coordDim, true); - } - } // Make sure the first extra dim is 'value'. - - - var generateCoord = opt.generateCoord; - var generateCoordCount = opt.generateCoordCount; - var fromZero = generateCoordCount != null; - generateCoordCount = generateCoord ? generateCoordCount || 1 : 0; - var extra = generateCoord || 'value'; - - function ifNoNameFillWithCoordName(resultItem) { - if (resultItem.name == null) { - // Duplication will be removed in the next step. - resultItem.name = resultItem.coordDim; - } - } // Set dim `name` and other `coordDim` and other props. - - - if (!omitUnusedDimensions) { - for (var resultDimIdx = 0; resultDimIdx < dimCount; resultDimIdx++) { - var resultItem = getResultItem(resultDimIdx); - var coordDim = resultItem.coordDim; - - if (coordDim == null) { - // TODO no need to generate coordDim for isExtraCoord? - resultItem.coordDim = genCoordDimName(extra, coordDimNameMap, fromZero); - resultItem.coordDimIndex = 0; // Series specified generateCoord is using out. - - if (!generateCoord || generateCoordCount <= 0) { - resultItem.isExtraCoord = true; - } - - generateCoordCount--; - } - - ifNoNameFillWithCoordName(resultItem); - - if (resultItem.type == null && (guessOrdinal(source, resultDimIdx) === BE_ORDINAL.Must // Consider the case: - // { - // dataset: {source: [ - // ['2001', 123], - // ['2002', 456], - // ... - // ['The others', 987], - // ]}, - // series: {type: 'pie'} - // } - // The first column should better be treated as a "ordinal" although it - // might not be detected as an "ordinal" by `guessOrdinal`. - || resultItem.isExtraCoord && (resultItem.otherDims.itemName != null || resultItem.otherDims.seriesName != null))) { - resultItem.type = 'ordinal'; - } - } - } else { - each(resultList, function (resultItem) { - // PENDING: guessOrdinal or let user specify type: 'ordinal' manually? - ifNoNameFillWithCoordName(resultItem); - }); // Sort dimensions: there are some rule that use the last dim as label, - // and for some latter travel process easier. - - resultList.sort(function (item0, item1) { - return item0.storeDimIndex - item1.storeDimIndex; - }); - } - - removeDuplication(resultList); - return new SeriesDataSchema({ - source: source, - dimensions: resultList, - fullDimensionCount: dimCount, - dimensionOmitted: omitUnusedDimensions - }); - } - - function removeDuplication(result) { - var duplicationMap = createHashMap(); - - for (var i = 0; i < result.length; i++) { - var dim = result[i]; - var dimOriginalName = dim.name; - var count = duplicationMap.get(dimOriginalName) || 0; - - if (count > 0) { - // Starts from 0. - dim.name = dimOriginalName + (count - 1); - } - - count++; - duplicationMap.set(dimOriginalName, count); - } - } // ??? TODO - // Originally detect dimCount by data[0]. Should we - // optimize it to only by sysDims and dimensions and encode. - // So only necessary dims will be initialized. - // But - // (1) custom series should be considered. where other dims - // may be visited. - // (2) sometimes user need to calculate bubble size or use visualMap - // on other dimensions besides coordSys needed. - // So, dims that is not used by system, should be shared in data store? - - - function getDimCount(source, sysDims, dimsDef, optDimCount) { - // Note that the result dimCount should not small than columns count - // of data, otherwise `dataDimNameMap` checking will be incorrect. - var dimCount = Math.max(source.dimensionsDetectedCount || 1, sysDims.length, dimsDef.length, optDimCount || 0); - each(sysDims, function (sysDimItem) { - var sysDimItemDimsDef; - - if (isObject(sysDimItem) && (sysDimItemDimsDef = sysDimItem.dimsDef)) { - dimCount = Math.max(dimCount, sysDimItemDimsDef.length); - } - }); - return dimCount; - } - - function genCoordDimName(name, map, fromZero) { - if (fromZero || map.hasKey(name)) { - var i = 0; - - while (map.hasKey(name + i)) { - i++; - } - - name += i; - } - - map.set(name, true); - return name; - } - - /** - * @class - * For example: - * { - * coordSysName: 'cartesian2d', - * coordSysDims: ['x', 'y', ...], - * axisMap: HashMap({ - * x: xAxisModel, - * y: yAxisModel - * }), - * categoryAxisMap: HashMap({ - * x: xAxisModel, - * y: undefined - * }), - * // The index of the first category axis in `coordSysDims`. - * // `null/undefined` means no category axis exists. - * firstCategoryDimIndex: 1, - * // To replace user specified encode. - * } - */ - - var CoordSysInfo = - /** @class */ - function () { - function CoordSysInfo(coordSysName) { - this.coordSysDims = []; - this.axisMap = createHashMap(); - this.categoryAxisMap = createHashMap(); - this.coordSysName = coordSysName; - } - - return CoordSysInfo; - }(); - - function getCoordSysInfoBySeries(seriesModel) { - var coordSysName = seriesModel.get('coordinateSystem'); - var result = new CoordSysInfo(coordSysName); - var fetch = fetchers[coordSysName]; - - if (fetch) { - fetch(seriesModel, result, result.axisMap, result.categoryAxisMap); - return result; - } - } - var fetchers = { - cartesian2d: function (seriesModel, result, axisMap, categoryAxisMap) { - var xAxisModel = seriesModel.getReferringComponents('xAxis', SINGLE_REFERRING).models[0]; - var yAxisModel = seriesModel.getReferringComponents('yAxis', SINGLE_REFERRING).models[0]; - - if ("development" !== 'production') { - if (!xAxisModel) { - throw new Error('xAxis "' + retrieve(seriesModel.get('xAxisIndex'), seriesModel.get('xAxisId'), 0) + '" not found'); - } - - if (!yAxisModel) { - throw new Error('yAxis "' + retrieve(seriesModel.get('xAxisIndex'), seriesModel.get('yAxisId'), 0) + '" not found'); - } - } - - result.coordSysDims = ['x', 'y']; - axisMap.set('x', xAxisModel); - axisMap.set('y', yAxisModel); - - if (isCategory(xAxisModel)) { - categoryAxisMap.set('x', xAxisModel); - result.firstCategoryDimIndex = 0; - } - - if (isCategory(yAxisModel)) { - categoryAxisMap.set('y', yAxisModel); - result.firstCategoryDimIndex == null && (result.firstCategoryDimIndex = 1); - } - }, - singleAxis: function (seriesModel, result, axisMap, categoryAxisMap) { - var singleAxisModel = seriesModel.getReferringComponents('singleAxis', SINGLE_REFERRING).models[0]; - - if ("development" !== 'production') { - if (!singleAxisModel) { - throw new Error('singleAxis should be specified.'); - } - } - - result.coordSysDims = ['single']; - axisMap.set('single', singleAxisModel); - - if (isCategory(singleAxisModel)) { - categoryAxisMap.set('single', singleAxisModel); - result.firstCategoryDimIndex = 0; - } - }, - polar: function (seriesModel, result, axisMap, categoryAxisMap) { - var polarModel = seriesModel.getReferringComponents('polar', SINGLE_REFERRING).models[0]; - var radiusAxisModel = polarModel.findAxisModel('radiusAxis'); - var angleAxisModel = polarModel.findAxisModel('angleAxis'); - - if ("development" !== 'production') { - if (!angleAxisModel) { - throw new Error('angleAxis option not found'); - } - - if (!radiusAxisModel) { - throw new Error('radiusAxis option not found'); - } - } - - result.coordSysDims = ['radius', 'angle']; - axisMap.set('radius', radiusAxisModel); - axisMap.set('angle', angleAxisModel); - - if (isCategory(radiusAxisModel)) { - categoryAxisMap.set('radius', radiusAxisModel); - result.firstCategoryDimIndex = 0; - } - - if (isCategory(angleAxisModel)) { - categoryAxisMap.set('angle', angleAxisModel); - result.firstCategoryDimIndex == null && (result.firstCategoryDimIndex = 1); - } - }, - geo: function (seriesModel, result, axisMap, categoryAxisMap) { - result.coordSysDims = ['lng', 'lat']; - }, - parallel: function (seriesModel, result, axisMap, categoryAxisMap) { - var ecModel = seriesModel.ecModel; - var parallelModel = ecModel.getComponent('parallel', seriesModel.get('parallelIndex')); - var coordSysDims = result.coordSysDims = parallelModel.dimensions.slice(); - each(parallelModel.parallelAxisIndex, function (axisIndex, index) { - var axisModel = ecModel.getComponent('parallelAxis', axisIndex); - var axisDim = coordSysDims[index]; - axisMap.set(axisDim, axisModel); - - if (isCategory(axisModel)) { - categoryAxisMap.set(axisDim, axisModel); - - if (result.firstCategoryDimIndex == null) { - result.firstCategoryDimIndex = index; - } - } - }); - } - }; - - function isCategory(axisModel) { - return axisModel.get('type') === 'category'; - } - - /** - * Note that it is too complicated to support 3d stack by value - * (have to create two-dimension inverted index), so in 3d case - * we just support that stacked by index. - * - * @param seriesModel - * @param dimensionsInput The same as the input of <module:echarts/data/SeriesData>. - * The input will be modified. - * @param opt - * @param opt.stackedCoordDimension Specify a coord dimension if needed. - * @param opt.byIndex=false - * @return calculationInfo - * { - * stackedDimension: string - * stackedByDimension: string - * isStackedByIndex: boolean - * stackedOverDimension: string - * stackResultDimension: string - * } - */ - - function enableDataStack(seriesModel, dimensionsInput, opt) { - opt = opt || {}; - var byIndex = opt.byIndex; - var stackedCoordDimension = opt.stackedCoordDimension; - var dimensionDefineList; - var schema; - var store; - - if (isLegacyDimensionsInput(dimensionsInput)) { - dimensionDefineList = dimensionsInput; - } else { - schema = dimensionsInput.schema; - dimensionDefineList = schema.dimensions; - store = dimensionsInput.store; - } // Compatibal: when `stack` is set as '', do not stack. - - - var mayStack = !!(seriesModel && seriesModel.get('stack')); - var stackedByDimInfo; - var stackedDimInfo; - var stackResultDimension; - var stackedOverDimension; - each(dimensionDefineList, function (dimensionInfo, index) { - if (isString(dimensionInfo)) { - dimensionDefineList[index] = dimensionInfo = { - name: dimensionInfo - }; - } - - if (mayStack && !dimensionInfo.isExtraCoord) { - // Find the first ordinal dimension as the stackedByDimInfo. - if (!byIndex && !stackedByDimInfo && dimensionInfo.ordinalMeta) { - stackedByDimInfo = dimensionInfo; - } // Find the first stackable dimension as the stackedDimInfo. - - - if (!stackedDimInfo && dimensionInfo.type !== 'ordinal' && dimensionInfo.type !== 'time' && (!stackedCoordDimension || stackedCoordDimension === dimensionInfo.coordDim)) { - stackedDimInfo = dimensionInfo; - } - } - }); - - if (stackedDimInfo && !byIndex && !stackedByDimInfo) { - // Compatible with previous design, value axis (time axis) only stack by index. - // It may make sense if the user provides elaborately constructed data. - byIndex = true; - } // Add stack dimension, they can be both calculated by coordinate system in `unionExtent`. - // That put stack logic in List is for using conveniently in echarts extensions, but it - // might not be a good way. - - - if (stackedDimInfo) { - // Use a weird name that not duplicated with other names. - // Also need to use seriesModel.id as postfix because different - // series may share same data store. The stack dimension needs to be distinguished. - stackResultDimension = '__\0ecstackresult_' + seriesModel.id; - stackedOverDimension = '__\0ecstackedover_' + seriesModel.id; // Create inverted index to fast query index by value. - - if (stackedByDimInfo) { - stackedByDimInfo.createInvertedIndices = true; - } - - var stackedDimCoordDim_1 = stackedDimInfo.coordDim; - var stackedDimType = stackedDimInfo.type; - var stackedDimCoordIndex_1 = 0; - each(dimensionDefineList, function (dimensionInfo) { - if (dimensionInfo.coordDim === stackedDimCoordDim_1) { - stackedDimCoordIndex_1++; - } - }); - var stackedOverDimensionDefine = { - name: stackResultDimension, - coordDim: stackedDimCoordDim_1, - coordDimIndex: stackedDimCoordIndex_1, - type: stackedDimType, - isExtraCoord: true, - isCalculationCoord: true, - storeDimIndex: dimensionDefineList.length - }; - var stackResultDimensionDefine = { - name: stackedOverDimension, - // This dimension contains stack base (generally, 0), so do not set it as - // `stackedDimCoordDim` to avoid extent calculation, consider log scale. - coordDim: stackedOverDimension, - coordDimIndex: stackedDimCoordIndex_1 + 1, - type: stackedDimType, - isExtraCoord: true, - isCalculationCoord: true, - storeDimIndex: dimensionDefineList.length + 1 - }; - - if (schema) { - if (store) { - stackedOverDimensionDefine.storeDimIndex = store.ensureCalculationDimension(stackedOverDimension, stackedDimType); - stackResultDimensionDefine.storeDimIndex = store.ensureCalculationDimension(stackResultDimension, stackedDimType); - } - - schema.appendCalculationDimension(stackedOverDimensionDefine); - schema.appendCalculationDimension(stackResultDimensionDefine); - } else { - dimensionDefineList.push(stackedOverDimensionDefine); - dimensionDefineList.push(stackResultDimensionDefine); - } - } - - return { - stackedDimension: stackedDimInfo && stackedDimInfo.name, - stackedByDimension: stackedByDimInfo && stackedByDimInfo.name, - isStackedByIndex: byIndex, - stackedOverDimension: stackedOverDimension, - stackResultDimension: stackResultDimension - }; - } - - function isLegacyDimensionsInput(dimensionsInput) { - return !isSeriesDataSchema(dimensionsInput.schema); - } - - function isDimensionStacked(data, stackedDim) { - // Each single series only maps to one pair of axis. So we do not need to - // check stackByDim, whatever stacked by a dimension or stacked by index. - return !!stackedDim && stackedDim === data.getCalculationInfo('stackedDimension'); - } - function getStackedDimension(data, targetDim) { - return isDimensionStacked(data, targetDim) ? data.getCalculationInfo('stackResultDimension') : targetDim; - } - - function getCoordSysDimDefs(seriesModel, coordSysInfo) { - var coordSysName = seriesModel.get('coordinateSystem'); - var registeredCoordSys = CoordinateSystemManager.get(coordSysName); - var coordSysDimDefs; - - if (coordSysInfo && coordSysInfo.coordSysDims) { - coordSysDimDefs = map(coordSysInfo.coordSysDims, function (dim) { - var dimInfo = { - name: dim - }; - var axisModel = coordSysInfo.axisMap.get(dim); - - if (axisModel) { - var axisType = axisModel.get('type'); - dimInfo.type = getDimensionTypeByAxis(axisType); - } - - return dimInfo; - }); - } - - if (!coordSysDimDefs) { - // Get dimensions from registered coordinate system - coordSysDimDefs = registeredCoordSys && (registeredCoordSys.getDimensionsInfo ? registeredCoordSys.getDimensionsInfo() : registeredCoordSys.dimensions.slice()) || ['x', 'y']; - } - - return coordSysDimDefs; - } - - function injectOrdinalMeta(dimInfoList, createInvertedIndices, coordSysInfo) { - var firstCategoryDimIndex; - var hasNameEncode; - coordSysInfo && each(dimInfoList, function (dimInfo, dimIndex) { - var coordDim = dimInfo.coordDim; - var categoryAxisModel = coordSysInfo.categoryAxisMap.get(coordDim); - - if (categoryAxisModel) { - if (firstCategoryDimIndex == null) { - firstCategoryDimIndex = dimIndex; - } - - dimInfo.ordinalMeta = categoryAxisModel.getOrdinalMeta(); - - if (createInvertedIndices) { - dimInfo.createInvertedIndices = true; - } - } - - if (dimInfo.otherDims.itemName != null) { - hasNameEncode = true; - } - }); - - if (!hasNameEncode && firstCategoryDimIndex != null) { - dimInfoList[firstCategoryDimIndex].otherDims.itemName = 0; - } - - return firstCategoryDimIndex; - } - /** - * Caution: there are side effects to `sourceManager` in this method. - * Should better only be called in `Series['getInitialData']`. - */ - - - function createSeriesData(sourceRaw, seriesModel, opt) { - opt = opt || {}; - var sourceManager = seriesModel.getSourceManager(); - var source; - var isOriginalSource = false; - - if (sourceRaw) { - isOriginalSource = true; - source = createSourceFromSeriesDataOption(sourceRaw); - } else { - source = sourceManager.getSource(); // Is series.data. not dataset. - - isOriginalSource = source.sourceFormat === SOURCE_FORMAT_ORIGINAL; - } - - var coordSysInfo = getCoordSysInfoBySeries(seriesModel); - var coordSysDimDefs = getCoordSysDimDefs(seriesModel, coordSysInfo); - var useEncodeDefaulter = opt.useEncodeDefaulter; - var encodeDefaulter = isFunction(useEncodeDefaulter) ? useEncodeDefaulter : useEncodeDefaulter ? curry(makeSeriesEncodeForAxisCoordSys, coordSysDimDefs, seriesModel) : null; - var createDimensionOptions = { - coordDimensions: coordSysDimDefs, - generateCoord: opt.generateCoord, - encodeDefine: seriesModel.getEncode(), - encodeDefaulter: encodeDefaulter, - canOmitUnusedDimensions: !isOriginalSource - }; - var schema = prepareSeriesDataSchema(source, createDimensionOptions); - var firstCategoryDimIndex = injectOrdinalMeta(schema.dimensions, opt.createInvertedIndices, coordSysInfo); - var store = !isOriginalSource ? sourceManager.getSharedDataStore(schema) : null; - var stackCalculationInfo = enableDataStack(seriesModel, { - schema: schema, - store: store - }); - var data = new SeriesData(schema, seriesModel); - data.setCalculationInfo(stackCalculationInfo); - var dimValueGetter = firstCategoryDimIndex != null && isNeedCompleteOrdinalData(source) ? function (itemOpt, dimName, dataIndex, dimIndex) { - // Use dataIndex as ordinal value in categoryAxis - return dimIndex === firstCategoryDimIndex ? dataIndex : this.defaultDimValueGetter(itemOpt, dimName, dataIndex, dimIndex); - } : null; - data.hasItemOption = false; - data.initData( // Try to reuse the data store in sourceManager if using dataset. - isOriginalSource ? source : store, null, dimValueGetter); - return data; - } - - function isNeedCompleteOrdinalData(source) { - if (source.sourceFormat === SOURCE_FORMAT_ORIGINAL) { - var sampleItem = firstDataNotNull(source.data || []); - return !isArray(getDataItemValue(sampleItem)); - } - } - - function firstDataNotNull(arr) { - var i = 0; - - while (i < arr.length && arr[i] == null) { - i++; - } - - return arr[i]; - } - - var Scale = - /** @class */ - function () { - function Scale(setting) { - this._setting = setting || {}; - this._extent = [Infinity, -Infinity]; - } - - Scale.prototype.getSetting = function (name) { - return this._setting[name]; - }; - /** - * Set extent from data - */ - - - Scale.prototype.unionExtent = function (other) { - var extent = this._extent; - other[0] < extent[0] && (extent[0] = other[0]); - other[1] > extent[1] && (extent[1] = other[1]); // not setExtent because in log axis it may transformed to power - // this.setExtent(extent[0], extent[1]); - }; - /** - * Set extent from data - */ - - - Scale.prototype.unionExtentFromData = function (data, dim) { - this.unionExtent(data.getApproximateExtent(dim)); - }; - /** - * Get extent - * - * Extent is always in increase order. - */ - - - Scale.prototype.getExtent = function () { - return this._extent.slice(); - }; - /** - * Set extent - */ - - - Scale.prototype.setExtent = function (start, end) { - var thisExtent = this._extent; - - if (!isNaN(start)) { - thisExtent[0] = start; - } - - if (!isNaN(end)) { - thisExtent[1] = end; - } - }; - /** - * If value is in extent range - */ - - - Scale.prototype.isInExtentRange = function (value) { - return this._extent[0] <= value && this._extent[1] >= value; - }; - /** - * When axis extent depends on data and no data exists, - * axis ticks should not be drawn, which is named 'blank'. - */ - - - Scale.prototype.isBlank = function () { - return this._isBlank; - }; - /** - * When axis extent depends on data and no data exists, - * axis ticks should not be drawn, which is named 'blank'. - */ - - - Scale.prototype.setBlank = function (isBlank) { - this._isBlank = isBlank; - }; - - return Scale; - }(); - - enableClassManagement(Scale); - - var uidBase = 0; - - var OrdinalMeta = - /** @class */ - function () { - function OrdinalMeta(opt) { - this.categories = opt.categories || []; - this._needCollect = opt.needCollect; - this._deduplication = opt.deduplication; - this.uid = ++uidBase; - } - - OrdinalMeta.createByAxisModel = function (axisModel) { - var option = axisModel.option; - var data = option.data; - var categories = data && map(data, getName); - return new OrdinalMeta({ - categories: categories, - needCollect: !categories, - // deduplication is default in axis. - deduplication: option.dedplication !== false - }); - }; - - OrdinalMeta.prototype.getOrdinal = function (category) { - // @ts-ignore - return this._getOrCreateMap().get(category); - }; - /** - * @return The ordinal. If not found, return NaN. - */ - - - OrdinalMeta.prototype.parseAndCollect = function (category) { - var index; - var needCollect = this._needCollect; // The value of category dim can be the index of the given category set. - // This feature is only supported when !needCollect, because we should - // consider a common case: a value is 2017, which is a number but is - // expected to be tread as a category. This case usually happen in dataset, - // where it happent to be no need of the index feature. - - if (!isString(category) && !needCollect) { - return category; - } // Optimize for the scenario: - // category is ['2012-01-01', '2012-01-02', ...], where the input - // data has been ensured not duplicate and is large data. - // Notice, if a dataset dimension provide categroies, usually echarts - // should remove duplication except user tell echarts dont do that - // (set axis.deduplication = false), because echarts do not know whether - // the values in the category dimension has duplication (consider the - // parallel-aqi example) - - - if (needCollect && !this._deduplication) { - index = this.categories.length; - this.categories[index] = category; - return index; - } - - var map = this._getOrCreateMap(); // @ts-ignore - - - index = map.get(category); - - if (index == null) { - if (needCollect) { - index = this.categories.length; - this.categories[index] = category; // @ts-ignore - - map.set(category, index); - } else { - index = NaN; - } - } - - return index; - }; // Consider big data, do not create map until needed. - - - OrdinalMeta.prototype._getOrCreateMap = function () { - return this._map || (this._map = createHashMap(this.categories)); - }; - - return OrdinalMeta; - }(); - - function getName(obj) { - if (isObject(obj) && obj.value != null) { - return obj.value; - } else { - return obj + ''; - } - } - - function isValueNice(val) { - var exp10 = Math.pow(10, quantityExponent(Math.abs(val))); - var f = Math.abs(val / exp10); - return f === 0 || f === 1 || f === 2 || f === 3 || f === 5; - } - function isIntervalOrLogScale(scale) { - return scale.type === 'interval' || scale.type === 'log'; - } - /** - * @param extent Both extent[0] and extent[1] should be valid number. - * Should be extent[0] < extent[1]. - * @param splitNumber splitNumber should be >= 1. - */ - - function intervalScaleNiceTicks(extent, splitNumber, minInterval, maxInterval) { - var result = {}; - var span = extent[1] - extent[0]; - var interval = result.interval = nice(span / splitNumber, true); - - if (minInterval != null && interval < minInterval) { - interval = result.interval = minInterval; - } - - if (maxInterval != null && interval > maxInterval) { - interval = result.interval = maxInterval; - } // Tow more digital for tick. - - - var precision = result.intervalPrecision = getIntervalPrecision(interval); // Niced extent inside original extent - - var niceTickExtent = result.niceTickExtent = [round(Math.ceil(extent[0] / interval) * interval, precision), round(Math.floor(extent[1] / interval) * interval, precision)]; - fixExtent(niceTickExtent, extent); - return result; - } - function increaseInterval(interval) { - var exp10 = Math.pow(10, quantityExponent(interval)); // Increase interval - - var f = interval / exp10; - - if (!f) { - f = 1; - } else if (f === 2) { - f = 3; - } else if (f === 3) { - f = 5; - } else { - // f is 1 or 5 - f *= 2; - } - - return round(f * exp10); - } - /** - * @return interval precision - */ - - function getIntervalPrecision(interval) { - // Tow more digital for tick. - return getPrecision(interval) + 2; - } - - function clamp(niceTickExtent, idx, extent) { - niceTickExtent[idx] = Math.max(Math.min(niceTickExtent[idx], extent[1]), extent[0]); - } // In some cases (e.g., splitNumber is 1), niceTickExtent may be out of extent. - - - function fixExtent(niceTickExtent, extent) { - !isFinite(niceTickExtent[0]) && (niceTickExtent[0] = extent[0]); - !isFinite(niceTickExtent[1]) && (niceTickExtent[1] = extent[1]); - clamp(niceTickExtent, 0, extent); - clamp(niceTickExtent, 1, extent); - - if (niceTickExtent[0] > niceTickExtent[1]) { - niceTickExtent[0] = niceTickExtent[1]; - } - } - function contain$1(val, extent) { - return val >= extent[0] && val <= extent[1]; - } - function normalize$1(val, extent) { - if (extent[1] === extent[0]) { - return 0.5; - } - - return (val - extent[0]) / (extent[1] - extent[0]); - } - function scale$2(val, extent) { - return val * (extent[1] - extent[0]) + extent[0]; - } - - var OrdinalScale = - /** @class */ - function (_super) { - __extends(OrdinalScale, _super); - - function OrdinalScale(setting) { - var _this = _super.call(this, setting) || this; - - _this.type = 'ordinal'; - - var ordinalMeta = _this.getSetting('ordinalMeta'); // Caution: Should not use instanceof, consider ec-extensions using - // import approach to get OrdinalMeta class. - - - if (!ordinalMeta) { - ordinalMeta = new OrdinalMeta({}); - } - - if (isArray(ordinalMeta)) { - ordinalMeta = new OrdinalMeta({ - categories: map(ordinalMeta, function (item) { - return isObject(item) ? item.value : item; - }) - }); - } - - _this._ordinalMeta = ordinalMeta; - _this._extent = _this.getSetting('extent') || [0, ordinalMeta.categories.length - 1]; - return _this; - } - - OrdinalScale.prototype.parse = function (val) { - // Caution: Math.round(null) will return `0` rather than `NaN` - if (val == null) { - return NaN; - } - - return isString(val) ? this._ordinalMeta.getOrdinal(val) // val might be float. - : Math.round(val); - }; - - OrdinalScale.prototype.contain = function (rank) { - rank = this.parse(rank); - return contain$1(rank, this._extent) && this._ordinalMeta.categories[rank] != null; - }; - /** - * Normalize given rank or name to linear [0, 1] - * @param val raw ordinal number. - * @return normalized value in [0, 1]. - */ - - - OrdinalScale.prototype.normalize = function (val) { - val = this._getTickNumber(this.parse(val)); - return normalize$1(val, this._extent); - }; - /** - * @param val normalized value in [0, 1]. - * @return raw ordinal number. - */ - - - OrdinalScale.prototype.scale = function (val) { - val = Math.round(scale$2(val, this._extent)); - return this.getRawOrdinalNumber(val); - }; - - OrdinalScale.prototype.getTicks = function () { - var ticks = []; - var extent = this._extent; - var rank = extent[0]; - - while (rank <= extent[1]) { - ticks.push({ - value: rank - }); - rank++; - } - - return ticks; - }; - - OrdinalScale.prototype.getMinorTicks = function (splitNumber) { - // Not support. - return; - }; - /** - * @see `Ordinal['_ordinalNumbersByTick']` - */ - - - OrdinalScale.prototype.setSortInfo = function (info) { - if (info == null) { - this._ordinalNumbersByTick = this._ticksByOrdinalNumber = null; - return; - } - - var infoOrdinalNumbers = info.ordinalNumbers; - var ordinalsByTick = this._ordinalNumbersByTick = []; - var ticksByOrdinal = this._ticksByOrdinalNumber = []; // Unnecessary support negative tick in `realtimeSort`. - - var tickNum = 0; - var allCategoryLen = this._ordinalMeta.categories.length; - - for (var len = Math.min(allCategoryLen, infoOrdinalNumbers.length); tickNum < len; ++tickNum) { - var ordinalNumber = infoOrdinalNumbers[tickNum]; - ordinalsByTick[tickNum] = ordinalNumber; - ticksByOrdinal[ordinalNumber] = tickNum; - } // Handle that `series.data` only covers part of the `axis.category.data`. - - - var unusedOrdinal = 0; - - for (; tickNum < allCategoryLen; ++tickNum) { - while (ticksByOrdinal[unusedOrdinal] != null) { - unusedOrdinal++; - } - ordinalsByTick.push(unusedOrdinal); - ticksByOrdinal[unusedOrdinal] = tickNum; - } - }; - - OrdinalScale.prototype._getTickNumber = function (ordinal) { - var ticksByOrdinalNumber = this._ticksByOrdinalNumber; // also support ordinal out of range of `ordinalMeta.categories.length`, - // where ordinal numbers are used as tick value directly. - - return ticksByOrdinalNumber && ordinal >= 0 && ordinal < ticksByOrdinalNumber.length ? ticksByOrdinalNumber[ordinal] : ordinal; - }; - /** - * @usage - * ```js - * const ordinalNumber = ordinalScale.getRawOrdinalNumber(tickVal); - * - * // case0 - * const rawOrdinalValue = axisModel.getCategories()[ordinalNumber]; - * // case1 - * const rawOrdinalValue = this._ordinalMeta.categories[ordinalNumber]; - * // case2 - * const coord = axis.dataToCoord(ordinalNumber); - * ``` - * - * @param {OrdinalNumber} tickNumber index of display - */ - - - OrdinalScale.prototype.getRawOrdinalNumber = function (tickNumber) { - var ordinalNumbersByTick = this._ordinalNumbersByTick; // tickNumber may be out of range, e.g., when axis max is larger than `ordinalMeta.categories.length`., - // where ordinal numbers are used as tick value directly. - - return ordinalNumbersByTick && tickNumber >= 0 && tickNumber < ordinalNumbersByTick.length ? ordinalNumbersByTick[tickNumber] : tickNumber; - }; - /** - * Get item on tick - */ - - - OrdinalScale.prototype.getLabel = function (tick) { - if (!this.isBlank()) { - var ordinalNumber = this.getRawOrdinalNumber(tick.value); - var cateogry = this._ordinalMeta.categories[ordinalNumber]; // Note that if no data, ordinalMeta.categories is an empty array. - // Return empty if it's not exist. - - return cateogry == null ? '' : cateogry + ''; - } - }; - - OrdinalScale.prototype.count = function () { - return this._extent[1] - this._extent[0] + 1; - }; - - OrdinalScale.prototype.unionExtentFromData = function (data, dim) { - this.unionExtent(data.getApproximateExtent(dim)); - }; - /** - * @override - * If value is in extent range - */ - - - OrdinalScale.prototype.isInExtentRange = function (value) { - value = this._getTickNumber(value); - return this._extent[0] <= value && this._extent[1] >= value; - }; - - OrdinalScale.prototype.getOrdinalMeta = function () { - return this._ordinalMeta; - }; - - OrdinalScale.prototype.calcNiceTicks = function () {}; - - OrdinalScale.prototype.calcNiceExtent = function () {}; - - OrdinalScale.type = 'ordinal'; - return OrdinalScale; - }(Scale); - - Scale.registerClass(OrdinalScale); - - var roundNumber = round; - - var IntervalScale = - /** @class */ - function (_super) { - __extends(IntervalScale, _super); - - function IntervalScale() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'interval'; // Step is calculated in adjustExtent. - - _this._interval = 0; - _this._intervalPrecision = 2; - return _this; - } - - IntervalScale.prototype.parse = function (val) { - return val; - }; - - IntervalScale.prototype.contain = function (val) { - return contain$1(val, this._extent); - }; - - IntervalScale.prototype.normalize = function (val) { - return normalize$1(val, this._extent); - }; - - IntervalScale.prototype.scale = function (val) { - return scale$2(val, this._extent); - }; - - IntervalScale.prototype.setExtent = function (start, end) { - var thisExtent = this._extent; // start,end may be a Number like '25',so... - - if (!isNaN(start)) { - thisExtent[0] = parseFloat(start); - } - - if (!isNaN(end)) { - thisExtent[1] = parseFloat(end); - } - }; - - IntervalScale.prototype.unionExtent = function (other) { - var extent = this._extent; - other[0] < extent[0] && (extent[0] = other[0]); - other[1] > extent[1] && (extent[1] = other[1]); // unionExtent may called by it's sub classes - - this.setExtent(extent[0], extent[1]); - }; - - IntervalScale.prototype.getInterval = function () { - return this._interval; - }; - - IntervalScale.prototype.setInterval = function (interval) { - this._interval = interval; // Dropped auto calculated niceExtent and use user-set extent. - // We assume user wants to set both interval, min, max to get a better result. - - this._niceExtent = this._extent.slice(); - this._intervalPrecision = getIntervalPrecision(interval); - }; - /** - * @param expandToNicedExtent Whether expand the ticks to niced extent. - */ - - - IntervalScale.prototype.getTicks = function (expandToNicedExtent) { - var interval = this._interval; - var extent = this._extent; - var niceTickExtent = this._niceExtent; - var intervalPrecision = this._intervalPrecision; - var ticks = []; // If interval is 0, return []; - - if (!interval) { - return ticks; - } // Consider this case: using dataZoom toolbox, zoom and zoom. - - - var safeLimit = 10000; - - if (extent[0] < niceTickExtent[0]) { - if (expandToNicedExtent) { - ticks.push({ - value: roundNumber(niceTickExtent[0] - interval, intervalPrecision) - }); - } else { - ticks.push({ - value: extent[0] - }); - } - } - - var tick = niceTickExtent[0]; - - while (tick <= niceTickExtent[1]) { - ticks.push({ - value: tick - }); // Avoid rounding error - - tick = roundNumber(tick + interval, intervalPrecision); - - if (tick === ticks[ticks.length - 1].value) { - // Consider out of safe float point, e.g., - // -3711126.9907707 + 2e-10 === -3711126.9907707 - break; - } - - if (ticks.length > safeLimit) { - return []; - } - } // Consider this case: the last item of ticks is smaller - // than niceTickExtent[1] and niceTickExtent[1] === extent[1]. - - - var lastNiceTick = ticks.length ? ticks[ticks.length - 1].value : niceTickExtent[1]; - - if (extent[1] > lastNiceTick) { - if (expandToNicedExtent) { - ticks.push({ - value: roundNumber(lastNiceTick + interval, intervalPrecision) - }); - } else { - ticks.push({ - value: extent[1] - }); - } - } - - return ticks; - }; - - IntervalScale.prototype.getMinorTicks = function (splitNumber) { - var ticks = this.getTicks(true); - var minorTicks = []; - var extent = this.getExtent(); - - for (var i = 1; i < ticks.length; i++) { - var nextTick = ticks[i]; - var prevTick = ticks[i - 1]; - var count = 0; - var minorTicksGroup = []; - var interval = nextTick.value - prevTick.value; - var minorInterval = interval / splitNumber; - - while (count < splitNumber - 1) { - var minorTick = roundNumber(prevTick.value + (count + 1) * minorInterval); // For the first and last interval. The count may be less than splitNumber. - - if (minorTick > extent[0] && minorTick < extent[1]) { - minorTicksGroup.push(minorTick); - } - - count++; - } - - minorTicks.push(minorTicksGroup); - } - - return minorTicks; - }; - /** - * @param opt.precision If 'auto', use nice presision. - * @param opt.pad returns 1.50 but not 1.5 if precision is 2. - */ - - - IntervalScale.prototype.getLabel = function (data, opt) { - if (data == null) { - return ''; - } - - var precision = opt && opt.precision; - - if (precision == null) { - precision = getPrecision(data.value) || 0; - } else if (precision === 'auto') { - // Should be more precise then tick. - precision = this._intervalPrecision; - } // (1) If `precision` is set, 12.005 should be display as '12.00500'. - // (2) Use roundNumber (toFixed) to avoid scientific notation like '3.5e-7'. - - - var dataNum = roundNumber(data.value, precision, true); - return addCommas(dataNum); - }; - /** - * @param splitNumber By default `5`. - */ - - - IntervalScale.prototype.calcNiceTicks = function (splitNumber, minInterval, maxInterval) { - splitNumber = splitNumber || 5; - var extent = this._extent; - var span = extent[1] - extent[0]; - - if (!isFinite(span)) { - return; - } // User may set axis min 0 and data are all negative - // FIXME If it needs to reverse ? - - - if (span < 0) { - span = -span; - extent.reverse(); - } - - var result = intervalScaleNiceTicks(extent, splitNumber, minInterval, maxInterval); - this._intervalPrecision = result.intervalPrecision; - this._interval = result.interval; - this._niceExtent = result.niceTickExtent; - }; - - IntervalScale.prototype.calcNiceExtent = function (opt) { - var extent = this._extent; // If extent start and end are same, expand them - - if (extent[0] === extent[1]) { - if (extent[0] !== 0) { - // Expand extent - // Note that extents can be both negative. See #13154 - var expandSize = Math.abs(extent[0]); // In the fowllowing case - // Axis has been fixed max 100 - // Plus data are all 100 and axis extent are [100, 100]. - // Extend to the both side will cause expanded max is larger than fixed max. - // So only expand to the smaller side. - - if (!opt.fixMax) { - extent[1] += expandSize / 2; - extent[0] -= expandSize / 2; - } else { - extent[0] -= expandSize / 2; - } - } else { - extent[1] = 1; - } - } - - var span = extent[1] - extent[0]; // If there are no data and extent are [Infinity, -Infinity] - - if (!isFinite(span)) { - extent[0] = 0; - extent[1] = 1; - } - - this.calcNiceTicks(opt.splitNumber, opt.minInterval, opt.maxInterval); // let extent = this._extent; - - var interval = this._interval; - - if (!opt.fixMin) { - extent[0] = roundNumber(Math.floor(extent[0] / interval) * interval); - } - - if (!opt.fixMax) { - extent[1] = roundNumber(Math.ceil(extent[1] / interval) * interval); - } - }; - - IntervalScale.prototype.setNiceExtent = function (min, max) { - this._niceExtent = [min, max]; - }; - - IntervalScale.type = 'interval'; - return IntervalScale; - }(Scale); - - Scale.registerClass(IntervalScale); - - /* global Float32Array */ - - var supportFloat32Array = typeof Float32Array !== 'undefined'; - var Float32ArrayCtor = !supportFloat32Array ? Array : Float32Array; - function createFloat32Array(arg) { - if (isArray(arg)) { - // Return self directly if don't support TypedArray. - return supportFloat32Array ? new Float32Array(arg) : arg; - } // Else is number - - - return new Float32ArrayCtor(arg); - } - - var STACK_PREFIX = '__ec_stack_'; - - function getSeriesStackId(seriesModel) { - return seriesModel.get('stack') || STACK_PREFIX + seriesModel.seriesIndex; - } - - function getAxisKey(axis) { - return axis.dim + axis.index; - } - /** - * @return {Object} {width, offset, offsetCenter} If axis.type is not 'category', return undefined. - */ - - - function getLayoutOnAxis(opt) { - var params = []; - var baseAxis = opt.axis; - var axisKey = 'axis0'; - - if (baseAxis.type !== 'category') { - return; - } - - var bandWidth = baseAxis.getBandWidth(); - - for (var i = 0; i < opt.count || 0; i++) { - params.push(defaults({ - bandWidth: bandWidth, - axisKey: axisKey, - stackId: STACK_PREFIX + i - }, opt)); - } - - var widthAndOffsets = doCalBarWidthAndOffset(params); - var result = []; - - for (var i = 0; i < opt.count; i++) { - var item = widthAndOffsets[axisKey][STACK_PREFIX + i]; - item.offsetCenter = item.offset + item.width / 2; - result.push(item); - } - - return result; - } - function prepareLayoutBarSeries(seriesType, ecModel) { - var seriesModels = []; - ecModel.eachSeriesByType(seriesType, function (seriesModel) { - // Check series coordinate, do layout for cartesian2d only - if (isOnCartesian(seriesModel)) { - seriesModels.push(seriesModel); - } - }); - return seriesModels; - } - /** - * Map from (baseAxis.dim + '_' + baseAxis.index) to min gap of two adjacent - * values. - * This works for time axes, value axes, and log axes. - * For a single time axis, return value is in the form like - * {'x_0': [1000000]}. - * The value of 1000000 is in milliseconds. - */ - - function getValueAxesMinGaps(barSeries) { - /** - * Map from axis.index to values. - * For a single time axis, axisValues is in the form like - * {'x_0': [1495555200000, 1495641600000, 1495728000000]}. - * Items in axisValues[x], e.g. 1495555200000, are time values of all - * series. - */ - var axisValues = {}; - each(barSeries, function (seriesModel) { - var cartesian = seriesModel.coordinateSystem; - var baseAxis = cartesian.getBaseAxis(); - - if (baseAxis.type !== 'time' && baseAxis.type !== 'value') { - return; - } - - var data = seriesModel.getData(); - var key = baseAxis.dim + '_' + baseAxis.index; - var dimIdx = data.getDimensionIndex(data.mapDimension(baseAxis.dim)); - var store = data.getStore(); - - for (var i = 0, cnt = store.count(); i < cnt; ++i) { - var value = store.get(dimIdx, i); - - if (!axisValues[key]) { - // No previous data for the axis - axisValues[key] = [value]; - } else { - // No value in previous series - axisValues[key].push(value); - } // Ignore duplicated time values in the same axis - - } - }); - var axisMinGaps = {}; - - for (var key in axisValues) { - if (axisValues.hasOwnProperty(key)) { - var valuesInAxis = axisValues[key]; - - if (valuesInAxis) { - // Sort axis values into ascending order to calculate gaps - valuesInAxis.sort(function (a, b) { - return a - b; - }); - var min = null; - - for (var j = 1; j < valuesInAxis.length; ++j) { - var delta = valuesInAxis[j] - valuesInAxis[j - 1]; - - if (delta > 0) { - // Ignore 0 delta because they are of the same axis value - min = min === null ? delta : Math.min(min, delta); - } - } // Set to null if only have one data - - - axisMinGaps[key] = min; - } - } - } - - return axisMinGaps; - } - - function makeColumnLayout(barSeries) { - var axisMinGaps = getValueAxesMinGaps(barSeries); - var seriesInfoList = []; - each(barSeries, function (seriesModel) { - var cartesian = seriesModel.coordinateSystem; - var baseAxis = cartesian.getBaseAxis(); - var axisExtent = baseAxis.getExtent(); - var bandWidth; - - if (baseAxis.type === 'category') { - bandWidth = baseAxis.getBandWidth(); - } else if (baseAxis.type === 'value' || baseAxis.type === 'time') { - var key = baseAxis.dim + '_' + baseAxis.index; - var minGap = axisMinGaps[key]; - var extentSpan = Math.abs(axisExtent[1] - axisExtent[0]); - var scale = baseAxis.scale.getExtent(); - var scaleSpan = Math.abs(scale[1] - scale[0]); - bandWidth = minGap ? extentSpan / scaleSpan * minGap : extentSpan; // When there is only one data value - } else { - var data = seriesModel.getData(); - bandWidth = Math.abs(axisExtent[1] - axisExtent[0]) / data.count(); - } - - var barWidth = parsePercent$1(seriesModel.get('barWidth'), bandWidth); - var barMaxWidth = parsePercent$1(seriesModel.get('barMaxWidth'), bandWidth); - var barMinWidth = parsePercent$1( // barMinWidth by default is 0.5 / 1 in cartesian. Because in value axis, - // the auto-calculated bar width might be less than 0.5 / 1. - seriesModel.get('barMinWidth') || (isInLargeMode(seriesModel) ? 0.5 : 1), bandWidth); - var barGap = seriesModel.get('barGap'); - var barCategoryGap = seriesModel.get('barCategoryGap'); - seriesInfoList.push({ - bandWidth: bandWidth, - barWidth: barWidth, - barMaxWidth: barMaxWidth, - barMinWidth: barMinWidth, - barGap: barGap, - barCategoryGap: barCategoryGap, - axisKey: getAxisKey(baseAxis), - stackId: getSeriesStackId(seriesModel) - }); - }); - return doCalBarWidthAndOffset(seriesInfoList); - } - - function doCalBarWidthAndOffset(seriesInfoList) { - // Columns info on each category axis. Key is cartesian name - var columnsMap = {}; - each(seriesInfoList, function (seriesInfo, idx) { - var axisKey = seriesInfo.axisKey; - var bandWidth = seriesInfo.bandWidth; - var columnsOnAxis = columnsMap[axisKey] || { - bandWidth: bandWidth, - remainedWidth: bandWidth, - autoWidthCount: 0, - categoryGap: null, - gap: '20%', - stacks: {} - }; - var stacks = columnsOnAxis.stacks; - columnsMap[axisKey] = columnsOnAxis; - var stackId = seriesInfo.stackId; - - if (!stacks[stackId]) { - columnsOnAxis.autoWidthCount++; - } - - stacks[stackId] = stacks[stackId] || { - width: 0, - maxWidth: 0 - }; // Caution: In a single coordinate system, these barGrid attributes - // will be shared by series. Consider that they have default values, - // only the attributes set on the last series will work. - // Do not change this fact unless there will be a break change. - - var barWidth = seriesInfo.barWidth; - - if (barWidth && !stacks[stackId].width) { - // See #6312, do not restrict width. - stacks[stackId].width = barWidth; - barWidth = Math.min(columnsOnAxis.remainedWidth, barWidth); - columnsOnAxis.remainedWidth -= barWidth; - } - - var barMaxWidth = seriesInfo.barMaxWidth; - barMaxWidth && (stacks[stackId].maxWidth = barMaxWidth); - var barMinWidth = seriesInfo.barMinWidth; - barMinWidth && (stacks[stackId].minWidth = barMinWidth); - var barGap = seriesInfo.barGap; - barGap != null && (columnsOnAxis.gap = barGap); - var barCategoryGap = seriesInfo.barCategoryGap; - barCategoryGap != null && (columnsOnAxis.categoryGap = barCategoryGap); - }); - var result = {}; - each(columnsMap, function (columnsOnAxis, coordSysName) { - result[coordSysName] = {}; - var stacks = columnsOnAxis.stacks; - var bandWidth = columnsOnAxis.bandWidth; - var categoryGapPercent = columnsOnAxis.categoryGap; - - if (categoryGapPercent == null) { - var columnCount = keys(stacks).length; // More columns in one group - // the spaces between group is smaller. Or the column will be too thin. - - categoryGapPercent = Math.max(35 - columnCount * 4, 15) + '%'; - } - - var categoryGap = parsePercent$1(categoryGapPercent, bandWidth); - var barGapPercent = parsePercent$1(columnsOnAxis.gap, 1); - var remainedWidth = columnsOnAxis.remainedWidth; - var autoWidthCount = columnsOnAxis.autoWidthCount; - var autoWidth = (remainedWidth - categoryGap) / (autoWidthCount + (autoWidthCount - 1) * barGapPercent); - autoWidth = Math.max(autoWidth, 0); // Find if any auto calculated bar exceeded maxBarWidth - - each(stacks, function (column) { - var maxWidth = column.maxWidth; - var minWidth = column.minWidth; - - if (!column.width) { - var finalWidth = autoWidth; - - if (maxWidth && maxWidth < finalWidth) { - finalWidth = Math.min(maxWidth, remainedWidth); - } // `minWidth` has higher priority. `minWidth` decide that whether the - // bar is able to be visible. So `minWidth` should not be restricted - // by `maxWidth` or `remainedWidth` (which is from `bandWidth`). In - // the extreme cases for `value` axis, bars are allowed to overlap - // with each other if `minWidth` specified. - - - if (minWidth && minWidth > finalWidth) { - finalWidth = minWidth; - } - - if (finalWidth !== autoWidth) { - column.width = finalWidth; - remainedWidth -= finalWidth + barGapPercent * finalWidth; - autoWidthCount--; - } - } else { - // `barMinWidth/barMaxWidth` has higher priority than `barWidth`, as - // CSS does. Because barWidth can be a percent value, where - // `barMaxWidth` can be used to restrict the final width. - var finalWidth = column.width; - - if (maxWidth) { - finalWidth = Math.min(finalWidth, maxWidth); - } // `minWidth` has higher priority, as described above - - - if (minWidth) { - finalWidth = Math.max(finalWidth, minWidth); - } - - column.width = finalWidth; - remainedWidth -= finalWidth + barGapPercent * finalWidth; - autoWidthCount--; - } - }); // Recalculate width again - - autoWidth = (remainedWidth - categoryGap) / (autoWidthCount + (autoWidthCount - 1) * barGapPercent); - autoWidth = Math.max(autoWidth, 0); - var widthSum = 0; - var lastColumn; - each(stacks, function (column, idx) { - if (!column.width) { - column.width = autoWidth; - } - - lastColumn = column; - widthSum += column.width * (1 + barGapPercent); - }); - - if (lastColumn) { - widthSum -= lastColumn.width * barGapPercent; - } - - var offset = -widthSum / 2; - each(stacks, function (column, stackId) { - result[coordSysName][stackId] = result[coordSysName][stackId] || { - bandWidth: bandWidth, - offset: offset, - width: column.width - }; - offset += column.width * (1 + barGapPercent); - }); - }); - return result; - } - - function retrieveColumnLayout(barWidthAndOffset, axis, seriesModel) { - if (barWidthAndOffset && axis) { - var result = barWidthAndOffset[getAxisKey(axis)]; - - if (result != null && seriesModel != null) { - return result[getSeriesStackId(seriesModel)]; - } - - return result; - } - } - function layout(seriesType, ecModel) { - var seriesModels = prepareLayoutBarSeries(seriesType, ecModel); - var barWidthAndOffset = makeColumnLayout(seriesModels); - each(seriesModels, function (seriesModel) { - var data = seriesModel.getData(); - var cartesian = seriesModel.coordinateSystem; - var baseAxis = cartesian.getBaseAxis(); - var stackId = getSeriesStackId(seriesModel); - var columnLayoutInfo = barWidthAndOffset[getAxisKey(baseAxis)][stackId]; - var columnOffset = columnLayoutInfo.offset; - var columnWidth = columnLayoutInfo.width; - data.setLayout({ - bandWidth: columnLayoutInfo.bandWidth, - offset: columnOffset, - size: columnWidth - }); - }); - } // TODO: Do not support stack in large mode yet. - - function createProgressiveLayout(seriesType) { - return { - seriesType: seriesType, - plan: createRenderPlanner(), - reset: function (seriesModel) { - if (!isOnCartesian(seriesModel)) { - return; - } - - var data = seriesModel.getData(); - var cartesian = seriesModel.coordinateSystem; - var baseAxis = cartesian.getBaseAxis(); - var valueAxis = cartesian.getOtherAxis(baseAxis); - var valueDimIdx = data.getDimensionIndex(data.mapDimension(valueAxis.dim)); - var baseDimIdx = data.getDimensionIndex(data.mapDimension(baseAxis.dim)); - var drawBackground = seriesModel.get('showBackground', true); - var valueDim = data.mapDimension(valueAxis.dim); - var stackResultDim = data.getCalculationInfo('stackResultDimension'); - var stacked = isDimensionStacked(data, valueDim) && !!data.getCalculationInfo('stackedOnSeries'); - var isValueAxisH = valueAxis.isHorizontal(); - var valueAxisStart = getValueAxisStart(baseAxis, valueAxis); - var isLarge = isInLargeMode(seriesModel); - var barMinHeight = seriesModel.get('barMinHeight') || 0; - var stackedDimIdx = stackResultDim && data.getDimensionIndex(stackResultDim); // Layout info. - - var columnWidth = data.getLayout('size'); - var columnOffset = data.getLayout('offset'); - return { - progress: function (params, data) { - var count = params.count; - var largePoints = isLarge && createFloat32Array(count * 3); - var largeBackgroundPoints = isLarge && drawBackground && createFloat32Array(count * 3); - var largeDataIndices = isLarge && createFloat32Array(count); - var coordLayout = cartesian.master.getRect(); - var bgSize = isValueAxisH ? coordLayout.width : coordLayout.height; - var dataIndex; - var store = data.getStore(); - var idxOffset = 0; - - while ((dataIndex = params.next()) != null) { - var value = store.get(stacked ? stackedDimIdx : valueDimIdx, dataIndex); - var baseValue = store.get(baseDimIdx, dataIndex); - var baseCoord = valueAxisStart; - var startValue = void 0; // Because of the barMinHeight, we can not use the value in - // stackResultDimension directly. - - if (stacked) { - startValue = +value - store.get(valueDimIdx, dataIndex); - } - - var x = void 0; - var y = void 0; - var width = void 0; - var height = void 0; - - if (isValueAxisH) { - var coord = cartesian.dataToPoint([value, baseValue]); - - if (stacked) { - var startCoord = cartesian.dataToPoint([startValue, baseValue]); - baseCoord = startCoord[0]; - } - - x = baseCoord; - y = coord[1] + columnOffset; - width = coord[0] - baseCoord; - height = columnWidth; - - if (Math.abs(width) < barMinHeight) { - width = (width < 0 ? -1 : 1) * barMinHeight; - } - } else { - var coord = cartesian.dataToPoint([baseValue, value]); - - if (stacked) { - var startCoord = cartesian.dataToPoint([baseValue, startValue]); - baseCoord = startCoord[1]; - } - - x = coord[0] + columnOffset; - y = baseCoord; - width = columnWidth; - height = coord[1] - baseCoord; - - if (Math.abs(height) < barMinHeight) { - // Include zero to has a positive bar - height = (height <= 0 ? -1 : 1) * barMinHeight; - } - } - - if (!isLarge) { - data.setItemLayout(dataIndex, { - x: x, - y: y, - width: width, - height: height - }); - } else { - largePoints[idxOffset] = x; - largePoints[idxOffset + 1] = y; - largePoints[idxOffset + 2] = isValueAxisH ? width : height; - - if (largeBackgroundPoints) { - largeBackgroundPoints[idxOffset] = isValueAxisH ? coordLayout.x : x; - largeBackgroundPoints[idxOffset + 1] = isValueAxisH ? y : coordLayout.y; - largeBackgroundPoints[idxOffset + 2] = bgSize; - } - - largeDataIndices[dataIndex] = dataIndex; - } - - idxOffset += 3; - } - - if (isLarge) { - data.setLayout({ - largePoints: largePoints, - largeDataIndices: largeDataIndices, - largeBackgroundPoints: largeBackgroundPoints, - valueAxisHorizontal: isValueAxisH - }); - } - } - }; - } - }; - } - - function isOnCartesian(seriesModel) { - return seriesModel.coordinateSystem && seriesModel.coordinateSystem.type === 'cartesian2d'; - } - - function isInLargeMode(seriesModel) { - return seriesModel.pipelineContext && seriesModel.pipelineContext.large; - } // See cases in `test/bar-start.html` and `#7412`, `#8747`. - - - function getValueAxisStart(baseAxis, valueAxis) { - return valueAxis.toGlobalCoord(valueAxis.dataToCoord(valueAxis.type === 'log' ? 1 : 0)); - } - - var bisect = function (a, x, lo, hi) { - while (lo < hi) { - var mid = lo + hi >>> 1; - - if (a[mid][1] < x) { - lo = mid + 1; - } else { - hi = mid; - } - } - - return lo; - }; - - var TimeScale = - /** @class */ - function (_super) { - __extends(TimeScale, _super); - - function TimeScale(settings) { - var _this = _super.call(this, settings) || this; - - _this.type = 'time'; - return _this; - } - /** - * Get label is mainly for other components like dataZoom, tooltip. - */ - - - TimeScale.prototype.getLabel = function (tick) { - var useUTC = this.getSetting('useUTC'); - return format(tick.value, fullLeveledFormatter[getDefaultFormatPrecisionOfInterval(getPrimaryTimeUnit(this._minLevelUnit))] || fullLeveledFormatter.second, useUTC, this.getSetting('locale')); - }; - - TimeScale.prototype.getFormattedLabel = function (tick, idx, labelFormatter) { - var isUTC = this.getSetting('useUTC'); - var lang = this.getSetting('locale'); - return leveledFormat(tick, idx, labelFormatter, lang, isUTC); - }; - /** - * @override - */ - - - TimeScale.prototype.getTicks = function () { - var interval = this._interval; - var extent = this._extent; - var ticks = []; // If interval is 0, return []; - - if (!interval) { - return ticks; - } - - ticks.push({ - value: extent[0], - level: 0 - }); - var useUTC = this.getSetting('useUTC'); - var innerTicks = getIntervalTicks(this._minLevelUnit, this._approxInterval, useUTC, extent); - ticks = ticks.concat(innerTicks); - ticks.push({ - value: extent[1], - level: 0 - }); - return ticks; - }; - - TimeScale.prototype.calcNiceExtent = function (opt) { - var extent = this._extent; // If extent start and end are same, expand them - - if (extent[0] === extent[1]) { - // Expand extent - extent[0] -= ONE_DAY; - extent[1] += ONE_DAY; - } // If there are no data and extent are [Infinity, -Infinity] - - - if (extent[1] === -Infinity && extent[0] === Infinity) { - var d = new Date(); - extent[1] = +new Date(d.getFullYear(), d.getMonth(), d.getDate()); - extent[0] = extent[1] - ONE_DAY; - } - - this.calcNiceTicks(opt.splitNumber, opt.minInterval, opt.maxInterval); - }; - - TimeScale.prototype.calcNiceTicks = function (approxTickNum, minInterval, maxInterval) { - approxTickNum = approxTickNum || 10; - var extent = this._extent; - var span = extent[1] - extent[0]; - this._approxInterval = span / approxTickNum; - - if (minInterval != null && this._approxInterval < minInterval) { - this._approxInterval = minInterval; - } - - if (maxInterval != null && this._approxInterval > maxInterval) { - this._approxInterval = maxInterval; - } - - var scaleIntervalsLen = scaleIntervals.length; - var idx = Math.min(bisect(scaleIntervals, this._approxInterval, 0, scaleIntervalsLen), scaleIntervalsLen - 1); // Interval that can be used to calculate ticks - - this._interval = scaleIntervals[idx][1]; // Min level used when picking ticks from top down. - // We check one more level to avoid the ticks are to sparse in some case. - - this._minLevelUnit = scaleIntervals[Math.max(idx - 1, 0)][0]; - }; - - TimeScale.prototype.parse = function (val) { - // val might be float. - return isNumber(val) ? val : +parseDate(val); - }; - - TimeScale.prototype.contain = function (val) { - return contain$1(this.parse(val), this._extent); - }; - - TimeScale.prototype.normalize = function (val) { - return normalize$1(this.parse(val), this._extent); - }; - - TimeScale.prototype.scale = function (val) { - return scale$2(val, this._extent); - }; - - TimeScale.type = 'time'; - return TimeScale; - }(IntervalScale); - /** - * This implementation was originally copied from "d3.js" - * <https://github.com/d3/d3/blob/b516d77fb8566b576088e73410437494717ada26/src/time/scale.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - */ - - - var scaleIntervals = [// Format interval - ['second', ONE_SECOND], ['minute', ONE_MINUTE], ['hour', ONE_HOUR], ['quarter-day', ONE_HOUR * 6], ['half-day', ONE_HOUR * 12], ['day', ONE_DAY * 1.2], ['half-week', ONE_DAY * 3.5], ['week', ONE_DAY * 7], ['month', ONE_DAY * 31], ['quarter', ONE_DAY * 95], ['half-year', ONE_YEAR / 2], ['year', ONE_YEAR] // 1Y - ]; - - function isUnitValueSame(unit, valueA, valueB, isUTC) { - var dateA = parseDate(valueA); - var dateB = parseDate(valueB); - - var isSame = function (unit) { - return getUnitValue(dateA, unit, isUTC) === getUnitValue(dateB, unit, isUTC); - }; - - var isSameYear = function () { - return isSame('year'); - }; // const isSameHalfYear = () => isSameYear() && isSame('half-year'); - // const isSameQuater = () => isSameYear() && isSame('quarter'); - - - var isSameMonth = function () { - return isSameYear() && isSame('month'); - }; - - var isSameDay = function () { - return isSameMonth() && isSame('day'); - }; // const isSameHalfDay = () => isSameDay() && isSame('half-day'); - - - var isSameHour = function () { - return isSameDay() && isSame('hour'); - }; - - var isSameMinute = function () { - return isSameHour() && isSame('minute'); - }; - - var isSameSecond = function () { - return isSameMinute() && isSame('second'); - }; - - var isSameMilliSecond = function () { - return isSameSecond() && isSame('millisecond'); - }; - - switch (unit) { - case 'year': - return isSameYear(); - - case 'month': - return isSameMonth(); - - case 'day': - return isSameDay(); - - case 'hour': - return isSameHour(); - - case 'minute': - return isSameMinute(); - - case 'second': - return isSameSecond(); - - case 'millisecond': - return isSameMilliSecond(); - } - } // const primaryUnitGetters = { - // year: fullYearGetterName(), - // month: monthGetterName(), - // day: dateGetterName(), - // hour: hoursGetterName(), - // minute: minutesGetterName(), - // second: secondsGetterName(), - // millisecond: millisecondsGetterName() - // }; - // const primaryUnitUTCGetters = { - // year: fullYearGetterName(true), - // month: monthGetterName(true), - // day: dateGetterName(true), - // hour: hoursGetterName(true), - // minute: minutesGetterName(true), - // second: secondsGetterName(true), - // millisecond: millisecondsGetterName(true) - // }; - // function moveTick(date: Date, unitName: TimeUnit, step: number, isUTC: boolean) { - // step = step || 1; - // switch (getPrimaryTimeUnit(unitName)) { - // case 'year': - // date[fullYearSetterName(isUTC)](date[fullYearGetterName(isUTC)]() + step); - // break; - // case 'month': - // date[monthSetterName(isUTC)](date[monthGetterName(isUTC)]() + step); - // break; - // case 'day': - // date[dateSetterName(isUTC)](date[dateGetterName(isUTC)]() + step); - // break; - // case 'hour': - // date[hoursSetterName(isUTC)](date[hoursGetterName(isUTC)]() + step); - // break; - // case 'minute': - // date[minutesSetterName(isUTC)](date[minutesGetterName(isUTC)]() + step); - // break; - // case 'second': - // date[secondsSetterName(isUTC)](date[secondsGetterName(isUTC)]() + step); - // break; - // case 'millisecond': - // date[millisecondsSetterName(isUTC)](date[millisecondsGetterName(isUTC)]() + step); - // break; - // } - // return date.getTime(); - // } - // const DATE_INTERVALS = [[8, 7.5], [4, 3.5], [2, 1.5]]; - // const MONTH_INTERVALS = [[6, 5.5], [3, 2.5], [2, 1.5]]; - // const MINUTES_SECONDS_INTERVALS = [[30, 30], [20, 20], [15, 15], [10, 10], [5, 5], [2, 2]]; - - - function getDateInterval(approxInterval, daysInMonth) { - approxInterval /= ONE_DAY; - return approxInterval > 16 ? 16 // Math.floor(daysInMonth / 2) + 1 // In this case we only want one tick between two months. - : approxInterval > 7.5 ? 7 // TODO week 7 or day 8? - : approxInterval > 3.5 ? 4 : approxInterval > 1.5 ? 2 : 1; - } - - function getMonthInterval(approxInterval) { - var APPROX_ONE_MONTH = 30 * ONE_DAY; - approxInterval /= APPROX_ONE_MONTH; - return approxInterval > 6 ? 6 : approxInterval > 3 ? 3 : approxInterval > 2 ? 2 : 1; - } - - function getHourInterval(approxInterval) { - approxInterval /= ONE_HOUR; - return approxInterval > 12 ? 12 : approxInterval > 6 ? 6 : approxInterval > 3.5 ? 4 : approxInterval > 2 ? 2 : 1; - } - - function getMinutesAndSecondsInterval(approxInterval, isMinutes) { - approxInterval /= isMinutes ? ONE_MINUTE : ONE_SECOND; - return approxInterval > 30 ? 30 : approxInterval > 20 ? 20 : approxInterval > 15 ? 15 : approxInterval > 10 ? 10 : approxInterval > 5 ? 5 : approxInterval > 2 ? 2 : 1; - } - - function getMillisecondsInterval(approxInterval) { - return nice(approxInterval, true); - } - - function getFirstTimestampOfUnit(date, unitName, isUTC) { - var outDate = new Date(date); - - switch (getPrimaryTimeUnit(unitName)) { - case 'year': - case 'month': - outDate[monthSetterName(isUTC)](0); - - case 'day': - outDate[dateSetterName(isUTC)](1); - - case 'hour': - outDate[hoursSetterName(isUTC)](0); - - case 'minute': - outDate[minutesSetterName(isUTC)](0); - - case 'second': - outDate[secondsSetterName(isUTC)](0); - outDate[millisecondsSetterName(isUTC)](0); - } - - return outDate.getTime(); - } - - function getIntervalTicks(bottomUnitName, approxInterval, isUTC, extent) { - var safeLimit = 10000; - var unitNames = timeUnits; - var iter = 0; - - function addTicksInSpan(interval, minTimestamp, maxTimestamp, getMethodName, setMethodName, isDate, out) { - var date = new Date(minTimestamp); - var dateTime = minTimestamp; - var d = date[getMethodName](); // if (isDate) { - // d -= 1; // Starts with 0; PENDING - // } - - while (dateTime < maxTimestamp && dateTime <= extent[1]) { - out.push({ - value: dateTime - }); - d += interval; - date[setMethodName](d); - dateTime = date.getTime(); - } // This extra tick is for calcuating ticks of next level. Will not been added to the final result - - - out.push({ - value: dateTime, - notAdd: true - }); - } - - function addLevelTicks(unitName, lastLevelTicks, levelTicks) { - var newAddedTicks = []; - var isFirstLevel = !lastLevelTicks.length; - - if (isUnitValueSame(getPrimaryTimeUnit(unitName), extent[0], extent[1], isUTC)) { - return; - } - - if (isFirstLevel) { - lastLevelTicks = [{ - // TODO Optimize. Not include so may ticks. - value: getFirstTimestampOfUnit(new Date(extent[0]), unitName, isUTC) - }, { - value: extent[1] - }]; - } - - for (var i = 0; i < lastLevelTicks.length - 1; i++) { - var startTick = lastLevelTicks[i].value; - var endTick = lastLevelTicks[i + 1].value; - - if (startTick === endTick) { - continue; - } - - var interval = void 0; - var getterName = void 0; - var setterName = void 0; - var isDate = false; - - switch (unitName) { - case 'year': - interval = Math.max(1, Math.round(approxInterval / ONE_DAY / 365)); - getterName = fullYearGetterName(isUTC); - setterName = fullYearSetterName(isUTC); - break; - - case 'half-year': - case 'quarter': - case 'month': - interval = getMonthInterval(approxInterval); - getterName = monthGetterName(isUTC); - setterName = monthSetterName(isUTC); - break; - - case 'week': // PENDING If week is added. Ignore day. - - case 'half-week': - case 'day': - interval = getDateInterval(approxInterval); // Use 32 days and let interval been 16 - - getterName = dateGetterName(isUTC); - setterName = dateSetterName(isUTC); - isDate = true; - break; - - case 'half-day': - case 'quarter-day': - case 'hour': - interval = getHourInterval(approxInterval); - getterName = hoursGetterName(isUTC); - setterName = hoursSetterName(isUTC); - break; - - case 'minute': - interval = getMinutesAndSecondsInterval(approxInterval, true); - getterName = minutesGetterName(isUTC); - setterName = minutesSetterName(isUTC); - break; - - case 'second': - interval = getMinutesAndSecondsInterval(approxInterval, false); - getterName = secondsGetterName(isUTC); - setterName = secondsSetterName(isUTC); - break; - - case 'millisecond': - interval = getMillisecondsInterval(approxInterval); - getterName = millisecondsGetterName(isUTC); - setterName = millisecondsSetterName(isUTC); - break; - } - - addTicksInSpan(interval, startTick, endTick, getterName, setterName, isDate, newAddedTicks); - - if (unitName === 'year' && levelTicks.length > 1 && i === 0) { - // Add nearest years to the left extent. - levelTicks.unshift({ - value: levelTicks[0].value - interval - }); - } - } - - for (var i = 0; i < newAddedTicks.length; i++) { - levelTicks.push(newAddedTicks[i]); - } // newAddedTicks.length && console.log(unitName, newAddedTicks); - - - return newAddedTicks; - } - - var levelsTicks = []; - var currentLevelTicks = []; - var tickCount = 0; - var lastLevelTickCount = 0; - - for (var i = 0; i < unitNames.length && iter++ < safeLimit; ++i) { - var primaryTimeUnit = getPrimaryTimeUnit(unitNames[i]); - - if (!isPrimaryTimeUnit(unitNames[i])) { - // TODO - continue; - } - - addLevelTicks(unitNames[i], levelsTicks[levelsTicks.length - 1] || [], currentLevelTicks); - var nextPrimaryTimeUnit = unitNames[i + 1] ? getPrimaryTimeUnit(unitNames[i + 1]) : null; - - if (primaryTimeUnit !== nextPrimaryTimeUnit) { - if (currentLevelTicks.length) { - lastLevelTickCount = tickCount; // Remove the duplicate so the tick count can be precisely. - - currentLevelTicks.sort(function (a, b) { - return a.value - b.value; - }); - var levelTicksRemoveDuplicated = []; - - for (var i_1 = 0; i_1 < currentLevelTicks.length; ++i_1) { - var tickValue = currentLevelTicks[i_1].value; - - if (i_1 === 0 || currentLevelTicks[i_1 - 1].value !== tickValue) { - levelTicksRemoveDuplicated.push(currentLevelTicks[i_1]); - - if (tickValue >= extent[0] && tickValue <= extent[1]) { - tickCount++; - } - } - } - - var targetTickNum = (extent[1] - extent[0]) / approxInterval; // Added too much in this level and not too less in last level - - if (tickCount > targetTickNum * 1.5 && lastLevelTickCount > targetTickNum / 1.5) { - break; - } // Only treat primary time unit as one level. - - - levelsTicks.push(levelTicksRemoveDuplicated); - - if (tickCount > targetTickNum || bottomUnitName === unitNames[i]) { - break; - } - } // Reset if next unitName is primary - - - currentLevelTicks = []; - } - } - - if ("development" !== 'production') { - if (iter >= safeLimit) { - warn('Exceed safe limit.'); - } - } - - var levelsTicksInExtent = filter(map(levelsTicks, function (levelTicks) { - return filter(levelTicks, function (tick) { - return tick.value >= extent[0] && tick.value <= extent[1] && !tick.notAdd; - }); - }), function (levelTicks) { - return levelTicks.length > 0; - }); - var ticks = []; - var maxLevel = levelsTicksInExtent.length - 1; - - for (var i = 0; i < levelsTicksInExtent.length; ++i) { - var levelTicks = levelsTicksInExtent[i]; - - for (var k = 0; k < levelTicks.length; ++k) { - ticks.push({ - value: levelTicks[k].value, - level: maxLevel - i - }); - } - } - - ticks.sort(function (a, b) { - return a.value - b.value; - }); // Remove duplicates - - var result = []; - - for (var i = 0; i < ticks.length; ++i) { - if (i === 0 || ticks[i].value !== ticks[i - 1].value) { - result.push(ticks[i]); - } - } - - return result; - } - - Scale.registerClass(TimeScale); - - var scaleProto = Scale.prototype; // FIXME:TS refactor: not good to call it directly with `this`? - - var intervalScaleProto = IntervalScale.prototype; - var roundingErrorFix = round; - var mathFloor = Math.floor; - var mathCeil = Math.ceil; - var mathPow$1 = Math.pow; - var mathLog = Math.log; - - var LogScale = - /** @class */ - function (_super) { - __extends(LogScale, _super); - - function LogScale() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'log'; - _this.base = 10; - _this._originalScale = new IntervalScale(); // FIXME:TS actually used by `IntervalScale` - - _this._interval = 0; - return _this; - } - /** - * @param Whether expand the ticks to niced extent. - */ - - - LogScale.prototype.getTicks = function (expandToNicedExtent) { - var originalScale = this._originalScale; - var extent = this._extent; - var originalExtent = originalScale.getExtent(); - var ticks = intervalScaleProto.getTicks.call(this, expandToNicedExtent); - return map(ticks, function (tick) { - var val = tick.value; - var powVal = round(mathPow$1(this.base, val)); // Fix #4158 - - powVal = val === extent[0] && this._fixMin ? fixRoundingError(powVal, originalExtent[0]) : powVal; - powVal = val === extent[1] && this._fixMax ? fixRoundingError(powVal, originalExtent[1]) : powVal; - return { - value: powVal - }; - }, this); - }; - - LogScale.prototype.setExtent = function (start, end) { - var base = mathLog(this.base); // log(-Infinity) is NaN, so safe guard here - - start = mathLog(Math.max(0, start)) / base; - end = mathLog(Math.max(0, end)) / base; - intervalScaleProto.setExtent.call(this, start, end); - }; - /** - * @return {number} end - */ - - - LogScale.prototype.getExtent = function () { - var base = this.base; - var extent = scaleProto.getExtent.call(this); - extent[0] = mathPow$1(base, extent[0]); - extent[1] = mathPow$1(base, extent[1]); // Fix #4158 - - var originalScale = this._originalScale; - var originalExtent = originalScale.getExtent(); - this._fixMin && (extent[0] = fixRoundingError(extent[0], originalExtent[0])); - this._fixMax && (extent[1] = fixRoundingError(extent[1], originalExtent[1])); - return extent; - }; - - LogScale.prototype.unionExtent = function (extent) { - this._originalScale.unionExtent(extent); - - var base = this.base; - extent[0] = mathLog(extent[0]) / mathLog(base); - extent[1] = mathLog(extent[1]) / mathLog(base); - scaleProto.unionExtent.call(this, extent); - }; - - LogScale.prototype.unionExtentFromData = function (data, dim) { - // TODO - // filter value that <= 0 - this.unionExtent(data.getApproximateExtent(dim)); - }; - /** - * Update interval and extent of intervals for nice ticks - * @param approxTickNum default 10 Given approx tick number - */ - - - LogScale.prototype.calcNiceTicks = function (approxTickNum) { - approxTickNum = approxTickNum || 10; - var extent = this._extent; - var span = extent[1] - extent[0]; - - if (span === Infinity || span <= 0) { - return; - } - - var interval = quantity(span); - var err = approxTickNum / span * interval; // Filter ticks to get closer to the desired count. - - if (err <= 0.5) { - interval *= 10; - } // Interval should be integer - - - while (!isNaN(interval) && Math.abs(interval) < 1 && Math.abs(interval) > 0) { - interval *= 10; - } - - var niceExtent = [round(mathCeil(extent[0] / interval) * interval), round(mathFloor(extent[1] / interval) * interval)]; - this._interval = interval; - this._niceExtent = niceExtent; - }; - - LogScale.prototype.calcNiceExtent = function (opt) { - intervalScaleProto.calcNiceExtent.call(this, opt); - this._fixMin = opt.fixMin; - this._fixMax = opt.fixMax; - }; - - LogScale.prototype.parse = function (val) { - return val; - }; - - LogScale.prototype.contain = function (val) { - val = mathLog(val) / mathLog(this.base); - return contain$1(val, this._extent); - }; - - LogScale.prototype.normalize = function (val) { - val = mathLog(val) / mathLog(this.base); - return normalize$1(val, this._extent); - }; - - LogScale.prototype.scale = function (val) { - val = scale$2(val, this._extent); - return mathPow$1(this.base, val); - }; - - LogScale.type = 'log'; - return LogScale; - }(Scale); - - var proto = LogScale.prototype; - proto.getMinorTicks = intervalScaleProto.getMinorTicks; - proto.getLabel = intervalScaleProto.getLabel; - - function fixRoundingError(val, originalVal) { - return roundingErrorFix(val, getPrecision(originalVal)); - } - - Scale.registerClass(LogScale); - - var ScaleRawExtentInfo = - /** @class */ - function () { - function ScaleRawExtentInfo(scale, model, // Usually: data extent from all series on this axis. - originalExtent) { - this._prepareParams(scale, model, originalExtent); - } - /** - * Parameters depending on outside (like model, user callback) - * are prepared and fixed here. - */ - - - ScaleRawExtentInfo.prototype._prepareParams = function (scale, model, // Usually: data extent from all series on this axis. - dataExtent) { - if (dataExtent[1] < dataExtent[0]) { - dataExtent = [NaN, NaN]; - } - - this._dataMin = dataExtent[0]; - this._dataMax = dataExtent[1]; - var isOrdinal = this._isOrdinal = scale.type === 'ordinal'; - this._needCrossZero = scale.type === 'interval' && model.getNeedCrossZero && model.getNeedCrossZero(); - var modelMinRaw = this._modelMinRaw = model.get('min', true); - - if (isFunction(modelMinRaw)) { - // This callback always provides users the full data extent (before data is filtered). - this._modelMinNum = parseAxisModelMinMax(scale, modelMinRaw({ - min: dataExtent[0], - max: dataExtent[1] - })); - } else if (modelMinRaw !== 'dataMin') { - this._modelMinNum = parseAxisModelMinMax(scale, modelMinRaw); - } - - var modelMaxRaw = this._modelMaxRaw = model.get('max', true); - - if (isFunction(modelMaxRaw)) { - // This callback always provides users the full data extent (before data is filtered). - this._modelMaxNum = parseAxisModelMinMax(scale, modelMaxRaw({ - min: dataExtent[0], - max: dataExtent[1] - })); - } else if (modelMaxRaw !== 'dataMax') { - this._modelMaxNum = parseAxisModelMinMax(scale, modelMaxRaw); - } - - if (isOrdinal) { - // FIXME: there is a flaw here: if there is no "block" data processor like `dataZoom`, - // and progressive rendering is using, here the category result might just only contain - // the processed chunk rather than the entire result. - this._axisDataLen = model.getCategories().length; - } else { - var boundaryGap = model.get('boundaryGap'); - var boundaryGapArr = isArray(boundaryGap) ? boundaryGap : [boundaryGap || 0, boundaryGap || 0]; - - if (typeof boundaryGapArr[0] === 'boolean' || typeof boundaryGapArr[1] === 'boolean') { - if ("development" !== 'production') { - console.warn('Boolean type for boundaryGap is only ' + 'allowed for ordinal axis. Please use string in ' + 'percentage instead, e.g., "20%". Currently, ' + 'boundaryGap is set to be 0.'); - } - - this._boundaryGapInner = [0, 0]; - } else { - this._boundaryGapInner = [parsePercent(boundaryGapArr[0], 1), parsePercent(boundaryGapArr[1], 1)]; - } - } - }; - /** - * Calculate extent by prepared parameters. - * This method has no external dependency and can be called duplicatedly, - * getting the same result. - * If parameters changed, should call this method to recalcuate. - */ - - - ScaleRawExtentInfo.prototype.calculate = function () { - // Notice: When min/max is not set (that is, when there are null/undefined, - // which is the most common case), these cases should be ensured: - // (1) For 'ordinal', show all axis.data. - // (2) For others: - // + `boundaryGap` is applied (if min/max set, boundaryGap is - // disabled). - // + If `needCrossZero`, min/max should be zero, otherwise, min/max should - // be the result that originalExtent enlarged by boundaryGap. - // (3) If no data, it should be ensured that `scale.setBlank` is set. - var isOrdinal = this._isOrdinal; - var dataMin = this._dataMin; - var dataMax = this._dataMax; - var axisDataLen = this._axisDataLen; - var boundaryGapInner = this._boundaryGapInner; - var span = !isOrdinal ? dataMax - dataMin || Math.abs(dataMin) : null; // Currently if a `'value'` axis model min is specified as 'dataMin'/'dataMax', - // `boundaryGap` will not be used. It's the different from specifying as `null`/`undefined`. - - var min = this._modelMinRaw === 'dataMin' ? dataMin : this._modelMinNum; - var max = this._modelMaxRaw === 'dataMax' ? dataMax : this._modelMaxNum; // If `_modelMinNum`/`_modelMaxNum` is `null`/`undefined`, should not be fixed. - - var minFixed = min != null; - var maxFixed = max != null; - - if (min == null) { - min = isOrdinal ? axisDataLen ? 0 : NaN : dataMin - boundaryGapInner[0] * span; - } - - if (max == null) { - max = isOrdinal ? axisDataLen ? axisDataLen - 1 : NaN : dataMax + boundaryGapInner[1] * span; - } - - (min == null || !isFinite(min)) && (min = NaN); - (max == null || !isFinite(max)) && (max = NaN); - var isBlank = eqNaN(min) || eqNaN(max) || isOrdinal && !axisDataLen; // If data extent modified, need to recalculated to ensure cross zero. - - if (this._needCrossZero) { - // Axis is over zero and min is not set - if (min > 0 && max > 0 && !minFixed) { - min = 0; // minFixed = true; - } // Axis is under zero and max is not set - - - if (min < 0 && max < 0 && !maxFixed) { - max = 0; // maxFixed = true; - } // PENDING: - // When `needCrossZero` and all data is positive/negative, should it be ensured - // that the results processed by boundaryGap are positive/negative? - // If so, here `minFixed`/`maxFixed` need to be set. - - } - - var determinedMin = this._determinedMin; - var determinedMax = this._determinedMax; - - if (determinedMin != null) { - min = determinedMin; - minFixed = true; - } - - if (determinedMax != null) { - max = determinedMax; - maxFixed = true; - } // Ensure min/max be finite number or NaN here. (not to be null/undefined) - // `NaN` means min/max axis is blank. - - - return { - min: min, - max: max, - minFixed: minFixed, - maxFixed: maxFixed, - isBlank: isBlank - }; - }; - - ScaleRawExtentInfo.prototype.modifyDataMinMax = function (minMaxName, val) { - if ("development" !== 'production') { - assert(!this.frozen); - } - - this[DATA_MIN_MAX_ATTR[minMaxName]] = val; - }; - - ScaleRawExtentInfo.prototype.setDeterminedMinMax = function (minMaxName, val) { - var attr = DETERMINED_MIN_MAX_ATTR[minMaxName]; - - if ("development" !== 'production') { - assert(!this.frozen // Earse them usually means logic flaw. - && this[attr] == null); - } - - this[attr] = val; - }; - - ScaleRawExtentInfo.prototype.freeze = function () { - // @ts-ignore - this.frozen = true; - }; - - return ScaleRawExtentInfo; - }(); - var DETERMINED_MIN_MAX_ATTR = { - min: '_determinedMin', - max: '_determinedMax' - }; - var DATA_MIN_MAX_ATTR = { - min: '_dataMin', - max: '_dataMax' - }; - /** - * Get scale min max and related info only depends on model settings. - * This method can be called after coordinate system created. - * For example, in data processing stage. - * - * Scale extent info probably be required multiple times during a workflow. - * For example: - * (1) `dataZoom` depends it to get the axis extent in "100%" state. - * (2) `processor/extentCalculator` depends it to make sure whether axis extent is specified. - * (3) `coordSys.update` use it to finally decide the scale extent. - * But the callback of `min`/`max` should not be called multiple times. - * The code below should not be implemented repeatedly either. - * So we cache the result in the scale instance, which will be recreated at the beginning - * of the workflow (because `scale` instance will be recreated each round of the workflow). - */ - - function ensureScaleRawExtentInfo(scale, model, // Usually: data extent from all series on this axis. - originalExtent) { - // Do not permit to recreate. - var rawExtentInfo = scale.rawExtentInfo; - - if (rawExtentInfo) { - return rawExtentInfo; - } - - rawExtentInfo = new ScaleRawExtentInfo(scale, model, originalExtent); // @ts-ignore - - scale.rawExtentInfo = rawExtentInfo; - return rawExtentInfo; - } - function parseAxisModelMinMax(scale, minMax) { - return minMax == null ? null : eqNaN(minMax) ? NaN : scale.parse(minMax); - } - - /** - * Get axis scale extent before niced. - * Item of returned array can only be number (including Infinity and NaN). - * - * Caution: - * Precondition of calling this method: - * The scale extent has been initialized using series data extent via - * `scale.setExtent` or `scale.unionExtentFromData`; - */ - - function getScaleExtent(scale, model) { - var scaleType = scale.type; - var rawExtentResult = ensureScaleRawExtentInfo(scale, model, scale.getExtent()).calculate(); - scale.setBlank(rawExtentResult.isBlank); - var min = rawExtentResult.min; - var max = rawExtentResult.max; // If bars are placed on a base axis of type time or interval account for axis boundary overflow and current axis - // is base axis - // FIXME - // (1) Consider support value axis, where below zero and axis `onZero` should be handled properly. - // (2) Refactor the logic with `barGrid`. Is it not need to `makeBarWidthAndOffsetInfo` twice with different extent? - // Should not depend on series type `bar`? - // (3) Fix that might overlap when using dataZoom. - // (4) Consider other chart types using `barGrid`? - // See #6728, #4862, `test/bar-overflow-time-plot.html` - - var ecModel = model.ecModel; - - if (ecModel && scaleType === 'time' - /* || scaleType === 'interval' */ - ) { - var barSeriesModels = prepareLayoutBarSeries('bar', ecModel); - var isBaseAxisAndHasBarSeries_1 = false; - each(barSeriesModels, function (seriesModel) { - isBaseAxisAndHasBarSeries_1 = isBaseAxisAndHasBarSeries_1 || seriesModel.getBaseAxis() === model.axis; - }); - - if (isBaseAxisAndHasBarSeries_1) { - // Calculate placement of bars on axis. TODO should be decoupled - // with barLayout - var barWidthAndOffset = makeColumnLayout(barSeriesModels); // Adjust axis min and max to account for overflow - - var adjustedScale = adjustScaleForOverflow(min, max, model, barWidthAndOffset); - min = adjustedScale.min; - max = adjustedScale.max; - } - } - - return { - extent: [min, max], - // "fix" means "fixed", the value should not be - // changed in the subsequent steps. - fixMin: rawExtentResult.minFixed, - fixMax: rawExtentResult.maxFixed - }; - } - - function adjustScaleForOverflow(min, max, model, // Only support cartesian coord yet. - barWidthAndOffset) { - // Get Axis Length - var axisExtent = model.axis.getExtent(); - var axisLength = axisExtent[1] - axisExtent[0]; // Get bars on current base axis and calculate min and max overflow - - var barsOnCurrentAxis = retrieveColumnLayout(barWidthAndOffset, model.axis); - - if (barsOnCurrentAxis === undefined) { - return { - min: min, - max: max - }; - } - - var minOverflow = Infinity; - each(barsOnCurrentAxis, function (item) { - minOverflow = Math.min(item.offset, minOverflow); - }); - var maxOverflow = -Infinity; - each(barsOnCurrentAxis, function (item) { - maxOverflow = Math.max(item.offset + item.width, maxOverflow); - }); - minOverflow = Math.abs(minOverflow); - maxOverflow = Math.abs(maxOverflow); - var totalOverFlow = minOverflow + maxOverflow; // Calculate required buffer based on old range and overflow - - var oldRange = max - min; - var oldRangePercentOfNew = 1 - (minOverflow + maxOverflow) / axisLength; - var overflowBuffer = oldRange / oldRangePercentOfNew - oldRange; - max += overflowBuffer * (maxOverflow / totalOverFlow); - min -= overflowBuffer * (minOverflow / totalOverFlow); - return { - min: min, - max: max - }; - } // Precondition of calling this method: - // The scale extent has been initialized using series data extent via - // `scale.setExtent` or `scale.unionExtentFromData`; - - - function niceScaleExtent(scale, inModel) { - var model = inModel; - var extentInfo = getScaleExtent(scale, model); - var extent = extentInfo.extent; - var splitNumber = model.get('splitNumber'); - - if (scale instanceof LogScale) { - scale.base = model.get('logBase'); - } - - var scaleType = scale.type; - var interval = model.get('interval'); - var isIntervalOrTime = scaleType === 'interval' || scaleType === 'time'; - scale.setExtent(extent[0], extent[1]); - scale.calcNiceExtent({ - splitNumber: splitNumber, - fixMin: extentInfo.fixMin, - fixMax: extentInfo.fixMax, - minInterval: isIntervalOrTime ? model.get('minInterval') : null, - maxInterval: isIntervalOrTime ? model.get('maxInterval') : null - }); // If some one specified the min, max. And the default calculated interval - // is not good enough. He can specify the interval. It is often appeared - // in angle axis with angle 0 - 360. Interval calculated in interval scale is hard - // to be 60. - // FIXME - - if (interval != null) { - scale.setInterval && scale.setInterval(interval); - } - } - /** - * @param axisType Default retrieve from model.type - */ - - function createScaleByModel(model, axisType) { - axisType = axisType || model.get('type'); - - if (axisType) { - switch (axisType) { - // Buildin scale - case 'category': - return new OrdinalScale({ - ordinalMeta: model.getOrdinalMeta ? model.getOrdinalMeta() : model.getCategories(), - extent: [Infinity, -Infinity] - }); - - case 'time': - return new TimeScale({ - locale: model.ecModel.getLocaleModel(), - useUTC: model.ecModel.get('useUTC') - }); - - default: - // case 'value'/'interval', 'log', or others. - return new (Scale.getClass(axisType) || IntervalScale)(); - } - } - } - /** - * Check if the axis cross 0 - */ - - function ifAxisCrossZero(axis) { - var dataExtent = axis.scale.getExtent(); - var min = dataExtent[0]; - var max = dataExtent[1]; - return !(min > 0 && max > 0 || min < 0 && max < 0); - } - /** - * @param axis - * @return Label formatter function. - * param: {number} tickValue, - * param: {number} idx, the index in all ticks. - * If category axis, this param is not required. - * return: {string} label string. - */ - - function makeLabelFormatter(axis) { - var labelFormatter = axis.getLabelModel().get('formatter'); - var categoryTickStart = axis.type === 'category' ? axis.scale.getExtent()[0] : null; - - if (axis.scale.type === 'time') { - return function (tpl) { - return function (tick, idx) { - return axis.scale.getFormattedLabel(tick, idx, tpl); - }; - }(labelFormatter); - } else if (isString(labelFormatter)) { - return function (tpl) { - return function (tick) { - // For category axis, get raw value; for numeric axis, - // get formatted label like '1,333,444'. - var label = axis.scale.getLabel(tick); - var text = tpl.replace('{value}', label != null ? label : ''); - return text; - }; - }(labelFormatter); - } else if (isFunction(labelFormatter)) { - return function (cb) { - return function (tick, idx) { - // The original intention of `idx` is "the index of the tick in all ticks". - // But the previous implementation of category axis do not consider the - // `axisLabel.interval`, which cause that, for example, the `interval` is - // `1`, then the ticks "name5", "name7", "name9" are displayed, where the - // corresponding `idx` are `0`, `2`, `4`, but not `0`, `1`, `2`. So we keep - // the definition here for back compatibility. - if (categoryTickStart != null) { - idx = tick.value - categoryTickStart; - } - - return cb(getAxisRawValue(axis, tick), idx, tick.level != null ? { - level: tick.level - } : null); - }; - }(labelFormatter); - } else { - return function (tick) { - return axis.scale.getLabel(tick); - }; - } - } - function getAxisRawValue(axis, tick) { - // In category axis with data zoom, tick is not the original - // index of axis.data. So tick should not be exposed to user - // in category axis. - return axis.type === 'category' ? axis.scale.getLabel(tick) : tick.value; - } - /** - * @param axis - * @return Be null/undefined if no labels. - */ - - function estimateLabelUnionRect(axis) { - var axisModel = axis.model; - var scale = axis.scale; - - if (!axisModel.get(['axisLabel', 'show']) || scale.isBlank()) { - return; - } - - var realNumberScaleTicks; - var tickCount; - var categoryScaleExtent = scale.getExtent(); // Optimize for large category data, avoid call `getTicks()`. - - if (scale instanceof OrdinalScale) { - tickCount = scale.count(); - } else { - realNumberScaleTicks = scale.getTicks(); - tickCount = realNumberScaleTicks.length; - } - - var axisLabelModel = axis.getLabelModel(); - var labelFormatter = makeLabelFormatter(axis); - var rect; - var step = 1; // Simple optimization for large amount of labels - - if (tickCount > 40) { - step = Math.ceil(tickCount / 40); - } - - for (var i = 0; i < tickCount; i += step) { - var tick = realNumberScaleTicks ? realNumberScaleTicks[i] : { - value: categoryScaleExtent[0] + i - }; - var label = labelFormatter(tick, i); - var unrotatedSingleRect = axisLabelModel.getTextRect(label); - var singleRect = rotateTextRect(unrotatedSingleRect, axisLabelModel.get('rotate') || 0); - rect ? rect.union(singleRect) : rect = singleRect; - } - - return rect; - } - - function rotateTextRect(textRect, rotate) { - var rotateRadians = rotate * Math.PI / 180; - var beforeWidth = textRect.width; - var beforeHeight = textRect.height; - var afterWidth = beforeWidth * Math.abs(Math.cos(rotateRadians)) + Math.abs(beforeHeight * Math.sin(rotateRadians)); - var afterHeight = beforeWidth * Math.abs(Math.sin(rotateRadians)) + Math.abs(beforeHeight * Math.cos(rotateRadians)); - var rotatedRect = new BoundingRect(textRect.x, textRect.y, afterWidth, afterHeight); - return rotatedRect; - } - /** - * @param model axisLabelModel or axisTickModel - * @return {number|String} Can be null|'auto'|number|function - */ - - - function getOptionCategoryInterval(model) { - var interval = model.get('interval'); - return interval == null ? 'auto' : interval; - } - /** - * Set `categoryInterval` as 0 implicitly indicates that - * show all labels regardless of overlap. - * @param {Object} axis axisModel.axis - */ - - function shouldShowAllLabels(axis) { - return axis.type === 'category' && getOptionCategoryInterval(axis.getLabelModel()) === 0; - } - function getDataDimensionsOnAxis(data, axisDim) { - // Remove duplicated dat dimensions caused by `getStackedDimension`. - var dataDimMap = {}; // Currently `mapDimensionsAll` will contain stack result dimension ('__\0ecstackresult'). - // PENDING: is it reasonable? Do we need to remove the original dim from "coord dim" since - // there has been stacked result dim? - - each(data.mapDimensionsAll(axisDim), function (dataDim) { - // For example, the extent of the original dimension - // is [0.1, 0.5], the extent of the `stackResultDimension` - // is [7, 9], the final extent should NOT include [0.1, 0.5], - // because there is no graphic corresponding to [0.1, 0.5]. - // See the case in `test/area-stack.html` `main1`, where area line - // stack needs `yAxis` not start from 0. - dataDimMap[getStackedDimension(data, dataDim)] = true; - }); - return keys(dataDimMap); - } - function unionAxisExtentFromData(dataExtent, data, axisDim) { - if (data) { - each(getDataDimensionsOnAxis(data, axisDim), function (dim) { - var seriesExtent = data.getApproximateExtent(dim); - seriesExtent[0] < dataExtent[0] && (dataExtent[0] = seriesExtent[0]); - seriesExtent[1] > dataExtent[1] && (dataExtent[1] = seriesExtent[1]); - }); - } - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - // eslint-disable-next-line @typescript-eslint/no-unused-vars - var AxisModelCommonMixin = - /** @class */ - function () { - function AxisModelCommonMixin() {} - - AxisModelCommonMixin.prototype.getNeedCrossZero = function () { - var option = this.option; - return !option.scale; - }; - /** - * Should be implemented by each axis model if necessary. - * @return coordinate system model - */ - - - AxisModelCommonMixin.prototype.getCoordSysModel = function () { - return; - }; - - return AxisModelCommonMixin; - }(); - - /** - * Create a multi dimension List structure from seriesModel. - */ - - function createList(seriesModel) { - return createSeriesData(null, seriesModel); - } // export function createGraph(seriesModel) { - var dataStack$1 = { - isDimensionStacked: isDimensionStacked, - enableDataStack: enableDataStack, - getStackedDimension: getStackedDimension - }; - /** - * Create scale - * @param {Array.<number>} dataExtent - * @param {Object|module:echarts/Model} option If `optoin.type` - * is secified, it can only be `'value'` currently. - */ - - function createScale(dataExtent, option) { - var axisModel = option; - - if (!(option instanceof Model)) { - axisModel = new Model(option); // FIXME - // Currently AxisModelCommonMixin has nothing to do with the - // the requirements of `axisHelper.createScaleByModel`. For - // example the methods `getCategories` and `getOrdinalMeta` - // are required for `'category'` axis, and ecModel is required - // for `'time'` axis. But occasionally echarts-gl happened - // to only use `'value'` axis. - // zrUtil.mixin(axisModel, AxisModelCommonMixin); - } - - var scale = createScaleByModel(axisModel); - scale.setExtent(dataExtent[0], dataExtent[1]); - niceScaleExtent(scale, axisModel); - return scale; - } - /** - * Mixin common methods to axis model, - * - * Include methods - * `getFormattedLabels() => Array.<string>` - * `getCategories() => Array.<string>` - * `getMin(origin: boolean) => number` - * `getMax(origin: boolean) => number` - * `getNeedCrossZero() => boolean` - */ - - function mixinAxisModelCommonMethods(Model) { - mixin(Model, AxisModelCommonMixin); - } - function createTextStyle$1(textStyleModel, opts) { - opts = opts || {}; - return createTextStyle(textStyleModel, null, null, opts.state !== 'normal'); - } - - var helper = /*#__PURE__*/Object.freeze({ - __proto__: null, - createList: createList, - getLayoutRect: getLayoutRect, - dataStack: dataStack$1, - createScale: createScale, - mixinAxisModelCommonMethods: mixinAxisModelCommonMethods, - getECData: getECData, - createTextStyle: createTextStyle$1, - createDimensions: createDimensions, - createSymbol: createSymbol, - enableHoverEmphasis: enableHoverEmphasis - }); - - var EPSILON$4 = 1e-8; - function isAroundEqual$1(a, b) { - return Math.abs(a - b) < EPSILON$4; - } - function contain$2(points, x, y) { - var w = 0; - var p = points[0]; - if (!p) { - return false; - } - for (var i = 1; i < points.length; i++) { - var p2 = points[i]; - w += windingLine(p[0], p[1], p2[0], p2[1], x, y); - p = p2; - } - var p0 = points[0]; - if (!isAroundEqual$1(p[0], p0[0]) || !isAroundEqual$1(p[1], p0[1])) { - w += windingLine(p[0], p[1], p0[0], p0[1], x, y); - } - return w !== 0; - } - - var TMP_TRANSFORM = []; - - function transformPoints(points, transform) { - for (var p = 0; p < points.length; p++) { - applyTransform(points[p], points[p], transform); - } - } - - function updateBBoxFromPoints(points, min$1, max$1, projection) { - for (var i = 0; i < points.length; i++) { - var p = points[i]; - - if (projection) { - // projection may return null point. - p = projection.project(p); - } - - if (p && isFinite(p[0]) && isFinite(p[1])) { - min(min$1, min$1, p); - max(max$1, max$1, p); - } - } - } - - function centroid(points) { - var signedArea = 0; - var cx = 0; - var cy = 0; - var len = points.length; - var x0 = points[len - 1][0]; - var y0 = points[len - 1][1]; // Polygon should been closed. - - for (var i = 0; i < len; i++) { - var x1 = points[i][0]; - var y1 = points[i][1]; - var a = x0 * y1 - x1 * y0; - signedArea += a; - cx += (x0 + x1) * a; - cy += (y0 + y1) * a; - x0 = x1; - y0 = y1; - } - - return signedArea ? [cx / signedArea / 3, cy / signedArea / 3, signedArea] : [points[0][0] || 0, points[0][1] || 0]; - } - - var Region = - /** @class */ - function () { - function Region(name) { - this.name = name; - } - - Region.prototype.setCenter = function (center) { - this._center = center; - }; - /** - * Get center point in data unit. That is, - * for GeoJSONRegion, the unit is lat/lng, - * for GeoSVGRegion, the unit is SVG local coord. - */ - - - Region.prototype.getCenter = function () { - var center = this._center; - - if (!center) { - // In most cases there are no need to calculate this center. - // So calculate only when called. - center = this._center = this.calcCenter(); - } - - return center; - }; - - return Region; - }(); - - var GeoJSONPolygonGeometry = - /** @class */ - function () { - function GeoJSONPolygonGeometry(exterior, interiors) { - this.type = 'polygon'; - this.exterior = exterior; - this.interiors = interiors; - } - - return GeoJSONPolygonGeometry; - }(); - - var GeoJSONLineStringGeometry = - /** @class */ - function () { - function GeoJSONLineStringGeometry(points) { - this.type = 'linestring'; - this.points = points; - } - - return GeoJSONLineStringGeometry; - }(); - - var GeoJSONRegion = - /** @class */ - function (_super) { - __extends(GeoJSONRegion, _super); - - function GeoJSONRegion(name, geometries, cp) { - var _this = _super.call(this, name) || this; - - _this.type = 'geoJSON'; - _this.geometries = geometries; - _this._center = cp && [cp[0], cp[1]]; - return _this; - } - - GeoJSONRegion.prototype.calcCenter = function () { - var geometries = this.geometries; - var largestGeo; - var largestGeoSize = 0; - - for (var i = 0; i < geometries.length; i++) { - var geo = geometries[i]; - var exterior = geo.exterior; // Simple trick to use points count instead of polygon area as region size. - // Ignore linestring - - var size = exterior && exterior.length; - - if (size > largestGeoSize) { - largestGeo = geo; - largestGeoSize = size; - } - } - - if (largestGeo) { - return centroid(largestGeo.exterior); - } // from bounding rect by default. - - - var rect = this.getBoundingRect(); - return [rect.x + rect.width / 2, rect.y + rect.height / 2]; - }; - - GeoJSONRegion.prototype.getBoundingRect = function (projection) { - var rect = this._rect; // Always recalculate if using projection. - - if (rect && !projection) { - return rect; - } - - var min = [Infinity, Infinity]; - var max = [-Infinity, -Infinity]; - var geometries = this.geometries; - each(geometries, function (geo) { - if (geo.type === 'polygon') { - // Doesn't consider hole - updateBBoxFromPoints(geo.exterior, min, max, projection); - } else { - each(geo.points, function (points) { - updateBBoxFromPoints(points, min, max, projection); - }); - } - }); // Normalie invalid bounding. - - if (!(isFinite(min[0]) && isFinite(min[1]) && isFinite(max[0]) && isFinite(max[1]))) { - min[0] = min[1] = max[0] = max[1] = 0; - } - - rect = new BoundingRect(min[0], min[1], max[0] - min[0], max[1] - min[1]); - - if (!projection) { - this._rect = rect; - } - - return rect; - }; - - GeoJSONRegion.prototype.contain = function (coord) { - var rect = this.getBoundingRect(); - var geometries = this.geometries; - - if (!rect.contain(coord[0], coord[1])) { - return false; - } - - loopGeo: for (var i = 0, len = geometries.length; i < len; i++) { - var geo = geometries[i]; // Only support polygon. - - if (geo.type !== 'polygon') { - continue; - } - - var exterior = geo.exterior; - var interiors = geo.interiors; - - if (contain$2(exterior, coord[0], coord[1])) { - // Not in the region if point is in the hole. - for (var k = 0; k < (interiors ? interiors.length : 0); k++) { - if (contain$2(interiors[k], coord[0], coord[1])) { - continue loopGeo; - } - } - - return true; - } - } - - return false; - }; - /** - * Transform the raw coords to target bounding. - * @param x - * @param y - * @param width - * @param height - */ - - - GeoJSONRegion.prototype.transformTo = function (x, y, width, height) { - var rect = this.getBoundingRect(); - var aspect = rect.width / rect.height; - - if (!width) { - width = aspect * height; - } else if (!height) { - height = width / aspect; - } - - var target = new BoundingRect(x, y, width, height); - var transform = rect.calculateTransform(target); - var geometries = this.geometries; - - for (var i = 0; i < geometries.length; i++) { - var geo = geometries[i]; - - if (geo.type === 'polygon') { - transformPoints(geo.exterior, transform); - each(geo.interiors, function (interior) { - transformPoints(interior, transform); - }); - } else { - each(geo.points, function (points) { - transformPoints(points, transform); - }); - } - } - - rect = this._rect; - rect.copy(target); // Update center - - this._center = [rect.x + rect.width / 2, rect.y + rect.height / 2]; - }; - - GeoJSONRegion.prototype.cloneShallow = function (name) { - name == null && (name = this.name); - var newRegion = new GeoJSONRegion(name, this.geometries, this._center); - newRegion._rect = this._rect; - newRegion.transformTo = null; // Simply avoid to be called. - - return newRegion; - }; - - return GeoJSONRegion; - }(Region); - - var GeoSVGRegion = - /** @class */ - function (_super) { - __extends(GeoSVGRegion, _super); - - function GeoSVGRegion(name, elOnlyForCalculate) { - var _this = _super.call(this, name) || this; - - _this.type = 'geoSVG'; - _this._elOnlyForCalculate = elOnlyForCalculate; - return _this; - } - - GeoSVGRegion.prototype.calcCenter = function () { - var el = this._elOnlyForCalculate; - var rect = el.getBoundingRect(); - var center = [rect.x + rect.width / 2, rect.y + rect.height / 2]; - var mat = identity(TMP_TRANSFORM); - var target = el; - - while (target && !target.isGeoSVGGraphicRoot) { - mul$1(mat, target.getLocalTransform(), mat); - target = target.parent; - } - - invert(mat, mat); - applyTransform(center, center, mat); - return center; - }; - - return GeoSVGRegion; - }(Region); - - function decode(json) { - if (!json.UTF8Encoding) { - return json; - } - - var jsonCompressed = json; - var encodeScale = jsonCompressed.UTF8Scale; - - if (encodeScale == null) { - encodeScale = 1024; - } - - var features = jsonCompressed.features; - each(features, function (feature) { - var geometry = feature.geometry; - var encodeOffsets = geometry.encodeOffsets; - var coordinates = geometry.coordinates; // Geometry may be appeded manually in the script after json loaded. - // In this case this geometry is usually not encoded. - - if (!encodeOffsets) { - return; - } - - switch (geometry.type) { - case 'LineString': - geometry.coordinates = decodeRing(coordinates, encodeOffsets, encodeScale); - break; - - case 'Polygon': - decodeRings(coordinates, encodeOffsets, encodeScale); - break; - - case 'MultiLineString': - decodeRings(coordinates, encodeOffsets, encodeScale); - break; - - case 'MultiPolygon': - each(coordinates, function (rings, idx) { - return decodeRings(rings, encodeOffsets[idx], encodeScale); - }); - } - }); // Has been decoded - - jsonCompressed.UTF8Encoding = false; - return jsonCompressed; - } - - function decodeRings(rings, encodeOffsets, encodeScale) { - for (var c = 0; c < rings.length; c++) { - rings[c] = decodeRing(rings[c], encodeOffsets[c], encodeScale); - } - } - - function decodeRing(coordinate, encodeOffsets, encodeScale) { - var result = []; - var prevX = encodeOffsets[0]; - var prevY = encodeOffsets[1]; - - for (var i = 0; i < coordinate.length; i += 2) { - var x = coordinate.charCodeAt(i) - 64; - var y = coordinate.charCodeAt(i + 1) - 64; // ZigZag decoding - - x = x >> 1 ^ -(x & 1); - y = y >> 1 ^ -(y & 1); // Delta deocding - - x += prevX; - y += prevY; - prevX = x; - prevY = y; // Dequantize - - result.push([x / encodeScale, y / encodeScale]); - } - - return result; - } - - function parseGeoJSON(geoJson, nameProperty) { - geoJson = decode(geoJson); - return map(filter(geoJson.features, function (featureObj) { - // Output of mapshaper may have geometry null - return featureObj.geometry && featureObj.properties && featureObj.geometry.coordinates.length > 0; - }), function (featureObj) { - var properties = featureObj.properties; - var geo = featureObj.geometry; - var geometries = []; - - switch (geo.type) { - case 'Polygon': - var coordinates = geo.coordinates; // According to the GeoJSON specification. - // First must be exterior, and the rest are all interior(holes). - - geometries.push(new GeoJSONPolygonGeometry(coordinates[0], coordinates.slice(1))); - break; - - case 'MultiPolygon': - each(geo.coordinates, function (item) { - if (item[0]) { - geometries.push(new GeoJSONPolygonGeometry(item[0], item.slice(1))); - } - }); - break; - - case 'LineString': - geometries.push(new GeoJSONLineStringGeometry([geo.coordinates])); - break; - - case 'MultiLineString': - geometries.push(new GeoJSONLineStringGeometry(geo.coordinates)); - } - - var region = new GeoJSONRegion(properties[nameProperty || 'name'], geometries, properties.cp); - region.properties = properties; - return region; - }); - } - - var number = /*#__PURE__*/Object.freeze({ - __proto__: null, - linearMap: linearMap, - round: round, - asc: asc, - getPrecision: getPrecision, - getPrecisionSafe: getPrecisionSafe, - getPixelPrecision: getPixelPrecision, - getPercentWithPrecision: getPercentWithPrecision, - MAX_SAFE_INTEGER: MAX_SAFE_INTEGER, - remRadian: remRadian, - isRadianAroundZero: isRadianAroundZero, - parseDate: parseDate, - quantity: quantity, - quantityExponent: quantityExponent, - nice: nice, - quantile: quantile, - reformIntervals: reformIntervals, - isNumeric: isNumeric, - numericToNumber: numericToNumber - }); - - var time = /*#__PURE__*/Object.freeze({ - __proto__: null, - parse: parseDate, - format: format - }); - - var graphic$1 = /*#__PURE__*/Object.freeze({ - __proto__: null, - extendShape: extendShape, - extendPath: extendPath, - makePath: makePath, - makeImage: makeImage, - mergePath: mergePath$1, - resizePath: resizePath, - createIcon: createIcon, - updateProps: updateProps, - initProps: initProps, - getTransform: getTransform, - clipPointsByRect: clipPointsByRect, - clipRectByRect: clipRectByRect, - registerShape: registerShape, - getShapeClass: getShapeClass, - Group: Group, - Image: ZRImage, - Text: ZRText, - Circle: Circle, - Ellipse: Ellipse, - Sector: Sector, - Ring: Ring, - Polygon: Polygon, - Polyline: Polyline, - Rect: Rect, - Line: Line, - BezierCurve: BezierCurve, - Arc: Arc, - IncrementalDisplayable: IncrementalDisplayable, - CompoundPath: CompoundPath, - LinearGradient: LinearGradient, - RadialGradient: RadialGradient, - BoundingRect: BoundingRect - }); - - var format$1 = /*#__PURE__*/Object.freeze({ - __proto__: null, - addCommas: addCommas, - toCamelCase: toCamelCase, - normalizeCssArray: normalizeCssArray$1, - encodeHTML: encodeHTML, - formatTpl: formatTpl, - getTooltipMarker: getTooltipMarker, - formatTime: formatTime, - capitalFirst: capitalFirst, - truncateText: truncateText, - getTextRect: getTextRect - }); - - var util$1 = /*#__PURE__*/Object.freeze({ - __proto__: null, - map: map, - each: each, - indexOf: indexOf, - inherits: inherits, - reduce: reduce, - filter: filter, - bind: bind, - curry: curry, - isArray: isArray, - isString: isString, - isObject: isObject, - isFunction: isFunction, - extend: extend, - defaults: defaults, - clone: clone, - merge: merge - }); - - var inner$5 = makeInner(); - function createAxisLabels(axis) { - // Only ordinal scale support tick interval - return axis.type === 'category' ? makeCategoryLabels(axis) : makeRealNumberLabels(axis); - } - /** - * @param {module:echats/coord/Axis} axis - * @param {module:echarts/model/Model} tickModel For example, can be axisTick, splitLine, splitArea. - * @return {Object} { - * ticks: Array.<number> - * tickCategoryInterval: number - * } - */ - - function createAxisTicks(axis, tickModel) { - // Only ordinal scale support tick interval - return axis.type === 'category' ? makeCategoryTicks(axis, tickModel) : { - ticks: map(axis.scale.getTicks(), function (tick) { - return tick.value; - }) - }; - } - - function makeCategoryLabels(axis) { - var labelModel = axis.getLabelModel(); - var result = makeCategoryLabelsActually(axis, labelModel); - return !labelModel.get('show') || axis.scale.isBlank() ? { - labels: [], - labelCategoryInterval: result.labelCategoryInterval - } : result; - } - - function makeCategoryLabelsActually(axis, labelModel) { - var labelsCache = getListCache(axis, 'labels'); - var optionLabelInterval = getOptionCategoryInterval(labelModel); - var result = listCacheGet(labelsCache, optionLabelInterval); - - if (result) { - return result; - } - - var labels; - var numericLabelInterval; - - if (isFunction(optionLabelInterval)) { - labels = makeLabelsByCustomizedCategoryInterval(axis, optionLabelInterval); - } else { - numericLabelInterval = optionLabelInterval === 'auto' ? makeAutoCategoryInterval(axis) : optionLabelInterval; - labels = makeLabelsByNumericCategoryInterval(axis, numericLabelInterval); - } // Cache to avoid calling interval function repeatedly. - - - return listCacheSet(labelsCache, optionLabelInterval, { - labels: labels, - labelCategoryInterval: numericLabelInterval - }); - } - - function makeCategoryTicks(axis, tickModel) { - var ticksCache = getListCache(axis, 'ticks'); - var optionTickInterval = getOptionCategoryInterval(tickModel); - var result = listCacheGet(ticksCache, optionTickInterval); - - if (result) { - return result; - } - - var ticks; - var tickCategoryInterval; // Optimize for the case that large category data and no label displayed, - // we should not return all ticks. - - if (!tickModel.get('show') || axis.scale.isBlank()) { - ticks = []; - } - - if (isFunction(optionTickInterval)) { - ticks = makeLabelsByCustomizedCategoryInterval(axis, optionTickInterval, true); - } // Always use label interval by default despite label show. Consider this - // scenario, Use multiple grid with the xAxis sync, and only one xAxis shows - // labels. `splitLine` and `axisTick` should be consistent in this case. - else if (optionTickInterval === 'auto') { - var labelsResult = makeCategoryLabelsActually(axis, axis.getLabelModel()); - tickCategoryInterval = labelsResult.labelCategoryInterval; - ticks = map(labelsResult.labels, function (labelItem) { - return labelItem.tickValue; - }); - } else { - tickCategoryInterval = optionTickInterval; - ticks = makeLabelsByNumericCategoryInterval(axis, tickCategoryInterval, true); - } // Cache to avoid calling interval function repeatedly. - - - return listCacheSet(ticksCache, optionTickInterval, { - ticks: ticks, - tickCategoryInterval: tickCategoryInterval - }); - } - - function makeRealNumberLabels(axis) { - var ticks = axis.scale.getTicks(); - var labelFormatter = makeLabelFormatter(axis); - return { - labels: map(ticks, function (tick, idx) { - return { - level: tick.level, - formattedLabel: labelFormatter(tick, idx), - rawLabel: axis.scale.getLabel(tick), - tickValue: tick.value - }; - }) - }; - } - - function getListCache(axis, prop) { - // Because key can be a function, and cache size always is small, we use array cache. - return inner$5(axis)[prop] || (inner$5(axis)[prop] = []); - } - - function listCacheGet(cache, key) { - for (var i = 0; i < cache.length; i++) { - if (cache[i].key === key) { - return cache[i].value; - } - } - } - - function listCacheSet(cache, key, value) { - cache.push({ - key: key, - value: value - }); - return value; - } - - function makeAutoCategoryInterval(axis) { - var result = inner$5(axis).autoInterval; - return result != null ? result : inner$5(axis).autoInterval = axis.calculateCategoryInterval(); - } - /** - * Calculate interval for category axis ticks and labels. - * To get precise result, at least one of `getRotate` and `isHorizontal` - * should be implemented in axis. - */ - - - function calculateCategoryInterval(axis) { - var params = fetchAutoCategoryIntervalCalculationParams(axis); - var labelFormatter = makeLabelFormatter(axis); - var rotation = (params.axisRotate - params.labelRotate) / 180 * Math.PI; - var ordinalScale = axis.scale; - var ordinalExtent = ordinalScale.getExtent(); // Providing this method is for optimization: - // avoid generating a long array by `getTicks` - // in large category data case. - - var tickCount = ordinalScale.count(); - - if (ordinalExtent[1] - ordinalExtent[0] < 1) { - return 0; - } - - var step = 1; // Simple optimization. Empirical value: tick count should less than 40. - - if (tickCount > 40) { - step = Math.max(1, Math.floor(tickCount / 40)); - } - - var tickValue = ordinalExtent[0]; - var unitSpan = axis.dataToCoord(tickValue + 1) - axis.dataToCoord(tickValue); - var unitW = Math.abs(unitSpan * Math.cos(rotation)); - var unitH = Math.abs(unitSpan * Math.sin(rotation)); - var maxW = 0; - var maxH = 0; // Caution: Performance sensitive for large category data. - // Consider dataZoom, we should make appropriate step to avoid O(n) loop. - - for (; tickValue <= ordinalExtent[1]; tickValue += step) { - var width = 0; - var height = 0; // Not precise, do not consider align and vertical align - // and each distance from axis line yet. - - var rect = getBoundingRect(labelFormatter({ - value: tickValue - }), params.font, 'center', 'top'); // Magic number - - width = rect.width * 1.3; - height = rect.height * 1.3; // Min size, void long loop. - - maxW = Math.max(maxW, width, 7); - maxH = Math.max(maxH, height, 7); - } - - var dw = maxW / unitW; - var dh = maxH / unitH; // 0/0 is NaN, 1/0 is Infinity. - - isNaN(dw) && (dw = Infinity); - isNaN(dh) && (dh = Infinity); - var interval = Math.max(0, Math.floor(Math.min(dw, dh))); - var cache = inner$5(axis.model); - var axisExtent = axis.getExtent(); - var lastAutoInterval = cache.lastAutoInterval; - var lastTickCount = cache.lastTickCount; // Use cache to keep interval stable while moving zoom window, - // otherwise the calculated interval might jitter when the zoom - // window size is close to the interval-changing size. - // For example, if all of the axis labels are `a, b, c, d, e, f, g`. - // The jitter will cause that sometimes the displayed labels are - // `a, d, g` (interval: 2) sometimes `a, c, e`(interval: 1). - - if (lastAutoInterval != null && lastTickCount != null && Math.abs(lastAutoInterval - interval) <= 1 && Math.abs(lastTickCount - tickCount) <= 1 // Always choose the bigger one, otherwise the critical - // point is not the same when zooming in or zooming out. - && lastAutoInterval > interval // If the axis change is caused by chart resize, the cache should not - // be used. Otherwise some hidden labels might not be shown again. - && cache.axisExtent0 === axisExtent[0] && cache.axisExtent1 === axisExtent[1]) { - interval = lastAutoInterval; - } // Only update cache if cache not used, otherwise the - // changing of interval is too insensitive. - else { - cache.lastTickCount = tickCount; - cache.lastAutoInterval = interval; - cache.axisExtent0 = axisExtent[0]; - cache.axisExtent1 = axisExtent[1]; - } - - return interval; - } - - function fetchAutoCategoryIntervalCalculationParams(axis) { - var labelModel = axis.getLabelModel(); - return { - axisRotate: axis.getRotate ? axis.getRotate() : axis.isHorizontal && !axis.isHorizontal() ? 90 : 0, - labelRotate: labelModel.get('rotate') || 0, - font: labelModel.getFont() - }; - } - - function makeLabelsByNumericCategoryInterval(axis, categoryInterval, onlyTick) { - var labelFormatter = makeLabelFormatter(axis); - var ordinalScale = axis.scale; - var ordinalExtent = ordinalScale.getExtent(); - var labelModel = axis.getLabelModel(); - var result = []; // TODO: axisType: ordinalTime, pick the tick from each month/day/year/... - - var step = Math.max((categoryInterval || 0) + 1, 1); - var startTick = ordinalExtent[0]; - var tickCount = ordinalScale.count(); // Calculate start tick based on zero if possible to keep label consistent - // while zooming and moving while interval > 0. Otherwise the selection - // of displayable ticks and symbols probably keep changing. - // 3 is empirical value. - - if (startTick !== 0 && step > 1 && tickCount / step > 2) { - startTick = Math.round(Math.ceil(startTick / step) * step); - } // (1) Only add min max label here but leave overlap checking - // to render stage, which also ensure the returned list - // suitable for splitLine and splitArea rendering. - // (2) Scales except category always contain min max label so - // do not need to perform this process. - - - var showAllLabel = shouldShowAllLabels(axis); - var includeMinLabel = labelModel.get('showMinLabel') || showAllLabel; - var includeMaxLabel = labelModel.get('showMaxLabel') || showAllLabel; - - if (includeMinLabel && startTick !== ordinalExtent[0]) { - addItem(ordinalExtent[0]); - } // Optimize: avoid generating large array by `ordinalScale.getTicks()`. - - - var tickValue = startTick; - - for (; tickValue <= ordinalExtent[1]; tickValue += step) { - addItem(tickValue); - } - - if (includeMaxLabel && tickValue - step !== ordinalExtent[1]) { - addItem(ordinalExtent[1]); - } - - function addItem(tickValue) { - var tickObj = { - value: tickValue - }; - result.push(onlyTick ? tickValue : { - formattedLabel: labelFormatter(tickObj), - rawLabel: ordinalScale.getLabel(tickObj), - tickValue: tickValue - }); - } - - return result; - } - - function makeLabelsByCustomizedCategoryInterval(axis, categoryInterval, onlyTick) { - var ordinalScale = axis.scale; - var labelFormatter = makeLabelFormatter(axis); - var result = []; - each(ordinalScale.getTicks(), function (tick) { - var rawLabel = ordinalScale.getLabel(tick); - var tickValue = tick.value; - - if (categoryInterval(tick.value, rawLabel)) { - result.push(onlyTick ? tickValue : { - formattedLabel: labelFormatter(tick), - rawLabel: rawLabel, - tickValue: tickValue - }); - } - }); - return result; - } - - var NORMALIZED_EXTENT = [0, 1]; - /** - * Base class of Axis. - */ - - var Axis = - /** @class */ - function () { - function Axis(dim, scale, extent) { - this.onBand = false; - this.inverse = false; - this.dim = dim; - this.scale = scale; - this._extent = extent || [0, 0]; - } - /** - * If axis extent contain given coord - */ - - - Axis.prototype.contain = function (coord) { - var extent = this._extent; - var min = Math.min(extent[0], extent[1]); - var max = Math.max(extent[0], extent[1]); - return coord >= min && coord <= max; - }; - /** - * If axis extent contain given data - */ - - - Axis.prototype.containData = function (data) { - return this.scale.contain(data); - }; - /** - * Get coord extent. - */ - - - Axis.prototype.getExtent = function () { - return this._extent.slice(); - }; - /** - * Get precision used for formatting - */ - - - Axis.prototype.getPixelPrecision = function (dataExtent) { - return getPixelPrecision(dataExtent || this.scale.getExtent(), this._extent); - }; - /** - * Set coord extent - */ - - - Axis.prototype.setExtent = function (start, end) { - var extent = this._extent; - extent[0] = start; - extent[1] = end; - }; - /** - * Convert data to coord. Data is the rank if it has an ordinal scale - */ - - - Axis.prototype.dataToCoord = function (data, clamp) { - var extent = this._extent; - var scale = this.scale; - data = scale.normalize(data); - - if (this.onBand && scale.type === 'ordinal') { - extent = extent.slice(); - fixExtentWithBands(extent, scale.count()); - } - - return linearMap(data, NORMALIZED_EXTENT, extent, clamp); - }; - /** - * Convert coord to data. Data is the rank if it has an ordinal scale - */ - - - Axis.prototype.coordToData = function (coord, clamp) { - var extent = this._extent; - var scale = this.scale; - - if (this.onBand && scale.type === 'ordinal') { - extent = extent.slice(); - fixExtentWithBands(extent, scale.count()); - } - - var t = linearMap(coord, extent, NORMALIZED_EXTENT, clamp); - return this.scale.scale(t); - }; - /** - * Convert pixel point to data in axis - */ - - - Axis.prototype.pointToData = function (point, clamp) { - // Should be implemented in derived class if necessary. - return; - }; - /** - * Different from `zrUtil.map(axis.getTicks(), axis.dataToCoord, axis)`, - * `axis.getTicksCoords` considers `onBand`, which is used by - * `boundaryGap:true` of category axis and splitLine and splitArea. - * @param opt.tickModel default: axis.model.getModel('axisTick') - * @param opt.clamp If `true`, the first and the last - * tick must be at the axis end points. Otherwise, clip ticks - * that outside the axis extent. - */ - - - Axis.prototype.getTicksCoords = function (opt) { - opt = opt || {}; - var tickModel = opt.tickModel || this.getTickModel(); - var result = createAxisTicks(this, tickModel); - var ticks = result.ticks; - var ticksCoords = map(ticks, function (tickVal) { - return { - coord: this.dataToCoord(this.scale.type === 'ordinal' ? this.scale.getRawOrdinalNumber(tickVal) : tickVal), - tickValue: tickVal - }; - }, this); - var alignWithLabel = tickModel.get('alignWithLabel'); - fixOnBandTicksCoords(this, ticksCoords, alignWithLabel, opt.clamp); - return ticksCoords; - }; - - Axis.prototype.getMinorTicksCoords = function () { - if (this.scale.type === 'ordinal') { - // Category axis doesn't support minor ticks - return []; - } - - var minorTickModel = this.model.getModel('minorTick'); - var splitNumber = minorTickModel.get('splitNumber'); // Protection. - - if (!(splitNumber > 0 && splitNumber < 100)) { - splitNumber = 5; - } - - var minorTicks = this.scale.getMinorTicks(splitNumber); - var minorTicksCoords = map(minorTicks, function (minorTicksGroup) { - return map(minorTicksGroup, function (minorTick) { - return { - coord: this.dataToCoord(minorTick), - tickValue: minorTick - }; - }, this); - }, this); - return minorTicksCoords; - }; - - Axis.prototype.getViewLabels = function () { - return createAxisLabels(this).labels; - }; - - Axis.prototype.getLabelModel = function () { - return this.model.getModel('axisLabel'); - }; - /** - * Notice here we only get the default tick model. For splitLine - * or splitArea, we should pass the splitLineModel or splitAreaModel - * manually when calling `getTicksCoords`. - * In GL, this method may be overridden to: - * `axisModel.getModel('axisTick', grid3DModel.getModel('axisTick'));` - */ - - - Axis.prototype.getTickModel = function () { - return this.model.getModel('axisTick'); - }; - /** - * Get width of band - */ - - - Axis.prototype.getBandWidth = function () { - var axisExtent = this._extent; - var dataExtent = this.scale.getExtent(); - var len = dataExtent[1] - dataExtent[0] + (this.onBand ? 1 : 0); // Fix #2728, avoid NaN when only one data. - - len === 0 && (len = 1); - var size = Math.abs(axisExtent[1] - axisExtent[0]); - return Math.abs(size) / len; - }; - /** - * Only be called in category axis. - * Can be overridden, consider other axes like in 3D. - * @return Auto interval for cateogry axis tick and label - */ - - - Axis.prototype.calculateCategoryInterval = function () { - return calculateCategoryInterval(this); - }; - - return Axis; - }(); - - function fixExtentWithBands(extent, nTick) { - var size = extent[1] - extent[0]; - var len = nTick; - var margin = size / len / 2; - extent[0] += margin; - extent[1] -= margin; - } // If axis has labels [1, 2, 3, 4]. Bands on the axis are - // |---1---|---2---|---3---|---4---|. - // So the displayed ticks and splitLine/splitArea should between - // each data item, otherwise cause misleading (e.g., split tow bars - // of a single data item when there are two bar series). - // Also consider if tickCategoryInterval > 0 and onBand, ticks and - // splitLine/spliteArea should layout appropriately corresponding - // to displayed labels. (So we should not use `getBandWidth` in this - // case). - - - function fixOnBandTicksCoords(axis, ticksCoords, alignWithLabel, clamp) { - var ticksLen = ticksCoords.length; - - if (!axis.onBand || alignWithLabel || !ticksLen) { - return; - } - - var axisExtent = axis.getExtent(); - var last; - var diffSize; - - if (ticksLen === 1) { - ticksCoords[0].coord = axisExtent[0]; - last = ticksCoords[1] = { - coord: axisExtent[0] - }; - } else { - var crossLen = ticksCoords[ticksLen - 1].tickValue - ticksCoords[0].tickValue; - var shift_1 = (ticksCoords[ticksLen - 1].coord - ticksCoords[0].coord) / crossLen; - each(ticksCoords, function (ticksItem) { - ticksItem.coord -= shift_1 / 2; - }); - var dataExtent = axis.scale.getExtent(); - diffSize = 1 + dataExtent[1] - ticksCoords[ticksLen - 1].tickValue; - last = { - coord: ticksCoords[ticksLen - 1].coord + shift_1 * diffSize - }; - ticksCoords.push(last); - } - - var inverse = axisExtent[0] > axisExtent[1]; // Handling clamp. - - if (littleThan(ticksCoords[0].coord, axisExtent[0])) { - clamp ? ticksCoords[0].coord = axisExtent[0] : ticksCoords.shift(); - } - - if (clamp && littleThan(axisExtent[0], ticksCoords[0].coord)) { - ticksCoords.unshift({ - coord: axisExtent[0] - }); - } - - if (littleThan(axisExtent[1], last.coord)) { - clamp ? last.coord = axisExtent[1] : ticksCoords.pop(); - } - - if (clamp && littleThan(last.coord, axisExtent[1])) { - ticksCoords.push({ - coord: axisExtent[1] - }); - } - - function littleThan(a, b) { - // Avoid rounding error cause calculated tick coord different with extent. - // It may cause an extra unnecessary tick added. - a = round(a); - b = round(b); - return inverse ? a > b : a < b; - } - } - - // Should use `ComponentModel.extend` or `class XXXX extend ComponentModel` to create class. - // Then use `registerComponentModel` in `install` parameter when `use` this extension. For example: - // class Bar3DModel extends ComponentModel {} - // export function install(registers) { registers.registerComponentModel(Bar3DModel); } - // echarts.use(install); - - function extendComponentModel(proto) { - var Model = ComponentModel.extend(proto); - ComponentModel.registerClass(Model); - return Model; - } - function extendComponentView(proto) { - var View = ComponentView.extend(proto); - ComponentView.registerClass(View); - return View; - } - function extendSeriesModel(proto) { - var Model = SeriesModel.extend(proto); - SeriesModel.registerClass(Model); - return Model; - } - function extendChartView(proto) { - var View = ChartView.extend(proto); - ChartView.registerClass(View); - return View; - } - - var PI2$6 = Math.PI * 2; - var CMD$3 = PathProxy.CMD; - var DEFAULT_SEARCH_SPACE = ['top', 'right', 'bottom', 'left']; - - function getCandidateAnchor(pos, distance, rect, outPt, outDir) { - var width = rect.width; - var height = rect.height; - - switch (pos) { - case 'top': - outPt.set(rect.x + width / 2, rect.y - distance); - outDir.set(0, -1); - break; - - case 'bottom': - outPt.set(rect.x + width / 2, rect.y + height + distance); - outDir.set(0, 1); - break; - - case 'left': - outPt.set(rect.x - distance, rect.y + height / 2); - outDir.set(-1, 0); - break; - - case 'right': - outPt.set(rect.x + width + distance, rect.y + height / 2); - outDir.set(1, 0); - break; - } - } - - function projectPointToArc(cx, cy, r, startAngle, endAngle, anticlockwise, x, y, out) { - x -= cx; - y -= cy; - var d = Math.sqrt(x * x + y * y); - x /= d; - y /= d; // Intersect point. - - var ox = x * r + cx; - var oy = y * r + cy; - - if (Math.abs(startAngle - endAngle) % PI2$6 < 1e-4) { - // Is a circle - out[0] = ox; - out[1] = oy; - return d - r; - } - - if (anticlockwise) { - var tmp = startAngle; - startAngle = normalizeRadian(endAngle); - endAngle = normalizeRadian(tmp); - } else { - startAngle = normalizeRadian(startAngle); - endAngle = normalizeRadian(endAngle); - } - - if (startAngle > endAngle) { - endAngle += PI2$6; - } - - var angle = Math.atan2(y, x); - - if (angle < 0) { - angle += PI2$6; - } - - if (angle >= startAngle && angle <= endAngle || angle + PI2$6 >= startAngle && angle + PI2$6 <= endAngle) { - // Project point is on the arc. - out[0] = ox; - out[1] = oy; - return d - r; - } - - var x1 = r * Math.cos(startAngle) + cx; - var y1 = r * Math.sin(startAngle) + cy; - var x2 = r * Math.cos(endAngle) + cx; - var y2 = r * Math.sin(endAngle) + cy; - var d1 = (x1 - x) * (x1 - x) + (y1 - y) * (y1 - y); - var d2 = (x2 - x) * (x2 - x) + (y2 - y) * (y2 - y); - - if (d1 < d2) { - out[0] = x1; - out[1] = y1; - return Math.sqrt(d1); - } else { - out[0] = x2; - out[1] = y2; - return Math.sqrt(d2); - } - } - - function projectPointToLine(x1, y1, x2, y2, x, y, out, limitToEnds) { - var dx = x - x1; - var dy = y - y1; - var dx1 = x2 - x1; - var dy1 = y2 - y1; - var lineLen = Math.sqrt(dx1 * dx1 + dy1 * dy1); - dx1 /= lineLen; - dy1 /= lineLen; // dot product - - var projectedLen = dx * dx1 + dy * dy1; - var t = projectedLen / lineLen; - - if (limitToEnds) { - t = Math.min(Math.max(t, 0), 1); - } - - t *= lineLen; - var ox = out[0] = x1 + t * dx1; - var oy = out[1] = y1 + t * dy1; - return Math.sqrt((ox - x) * (ox - x) + (oy - y) * (oy - y)); - } - - function projectPointToRect(x1, y1, width, height, x, y, out) { - if (width < 0) { - x1 = x1 + width; - width = -width; - } - - if (height < 0) { - y1 = y1 + height; - height = -height; - } - - var x2 = x1 + width; - var y2 = y1 + height; - var ox = out[0] = Math.min(Math.max(x, x1), x2); - var oy = out[1] = Math.min(Math.max(y, y1), y2); - return Math.sqrt((ox - x) * (ox - x) + (oy - y) * (oy - y)); - } - - var tmpPt = []; - - function nearestPointOnRect(pt, rect, out) { - var dist = projectPointToRect(rect.x, rect.y, rect.width, rect.height, pt.x, pt.y, tmpPt); - out.set(tmpPt[0], tmpPt[1]); - return dist; - } - /** - * Calculate min distance corresponding point. - * This method won't evaluate if point is in the path. - */ - - - function nearestPointOnPath(pt, path, out) { - var xi = 0; - var yi = 0; - var x0 = 0; - var y0 = 0; - var x1; - var y1; - var minDist = Infinity; - var data = path.data; - var x = pt.x; - var y = pt.y; - - for (var i = 0; i < data.length;) { - var cmd = data[i++]; - - if (i === 1) { - xi = data[i]; - yi = data[i + 1]; - x0 = xi; - y0 = yi; - } - - var d = minDist; - - switch (cmd) { - case CMD$3.M: - // moveTo 命令重新创建一个新的 subpath, 并且更新新的起点 - // 在 closePath 的时候使用 - x0 = data[i++]; - y0 = data[i++]; - xi = x0; - yi = y0; - break; - - case CMD$3.L: - d = projectPointToLine(xi, yi, data[i], data[i + 1], x, y, tmpPt, true); - xi = data[i++]; - yi = data[i++]; - break; - - case CMD$3.C: - d = cubicProjectPoint(xi, yi, data[i++], data[i++], data[i++], data[i++], data[i], data[i + 1], x, y, tmpPt); - xi = data[i++]; - yi = data[i++]; - break; - - case CMD$3.Q: - d = quadraticProjectPoint(xi, yi, data[i++], data[i++], data[i], data[i + 1], x, y, tmpPt); - xi = data[i++]; - yi = data[i++]; - break; - - case CMD$3.A: - // TODO Arc 判断的开销比较大 - var cx = data[i++]; - var cy = data[i++]; - var rx = data[i++]; - var ry = data[i++]; - var theta = data[i++]; - var dTheta = data[i++]; // TODO Arc 旋转 - - i += 1; - var anticlockwise = !!(1 - data[i++]); - x1 = Math.cos(theta) * rx + cx; - y1 = Math.sin(theta) * ry + cy; // 不是直接使用 arc 命令 - - if (i <= 1) { - // 第一个命令起点还未定义 - x0 = x1; - y0 = y1; - } // zr 使用scale来模拟椭圆, 这里也对x做一定的缩放 - - - var _x = (x - cx) * ry / rx + cx; - - d = projectPointToArc(cx, cy, ry, theta, theta + dTheta, anticlockwise, _x, y, tmpPt); - xi = Math.cos(theta + dTheta) * rx + cx; - yi = Math.sin(theta + dTheta) * ry + cy; - break; - - case CMD$3.R: - x0 = xi = data[i++]; - y0 = yi = data[i++]; - var width = data[i++]; - var height = data[i++]; - d = projectPointToRect(x0, y0, width, height, x, y, tmpPt); - break; - - case CMD$3.Z: - d = projectPointToLine(xi, yi, x0, y0, x, y, tmpPt, true); - xi = x0; - yi = y0; - break; - } - - if (d < minDist) { - minDist = d; - out.set(tmpPt[0], tmpPt[1]); - } - } - - return minDist; - } // Temporal variable for intermediate usage. - - - var pt0 = new Point(); - var pt1 = new Point(); - var pt2 = new Point(); - var dir = new Point(); - var dir2 = new Point(); - /** - * Calculate a proper guide line based on the label position and graphic element definition - * @param label - * @param labelRect - * @param target - * @param targetRect - */ - - function updateLabelLinePoints(target, labelLineModel) { - if (!target) { - return; - } - - var labelLine = target.getTextGuideLine(); - var label = target.getTextContent(); // Needs to create text guide in each charts. - - if (!(label && labelLine)) { - return; - } - - var labelGuideConfig = target.textGuideLineConfig || {}; - var points = [[0, 0], [0, 0], [0, 0]]; - var searchSpace = labelGuideConfig.candidates || DEFAULT_SEARCH_SPACE; - var labelRect = label.getBoundingRect().clone(); - labelRect.applyTransform(label.getComputedTransform()); - var minDist = Infinity; - var anchorPoint = labelGuideConfig.anchor; - var targetTransform = target.getComputedTransform(); - var targetInversedTransform = targetTransform && invert([], targetTransform); - var len = labelLineModel.get('length2') || 0; - - if (anchorPoint) { - pt2.copy(anchorPoint); - } - - for (var i = 0; i < searchSpace.length; i++) { - var candidate = searchSpace[i]; - getCandidateAnchor(candidate, 0, labelRect, pt0, dir); - Point.scaleAndAdd(pt1, pt0, dir, len); // Transform to target coord space. - - pt1.transform(targetInversedTransform); // Note: getBoundingRect will ensure the `path` being created. - - var boundingRect = target.getBoundingRect(); - var dist = anchorPoint ? anchorPoint.distance(pt1) : target instanceof Path ? nearestPointOnPath(pt1, target.path, pt2) : nearestPointOnRect(pt1, boundingRect, pt2); // TODO pt2 is in the path - - if (dist < minDist) { - minDist = dist; // Transform back to global space. - - pt1.transform(targetTransform); - pt2.transform(targetTransform); - pt2.toArray(points[0]); - pt1.toArray(points[1]); - pt0.toArray(points[2]); - } - } - - limitTurnAngle(points, labelLineModel.get('minTurnAngle')); - labelLine.setShape({ - points: points - }); - } // Temporal variable for the limitTurnAngle function - - var tmpArr = []; - var tmpProjPoint = new Point(); - /** - * Reduce the line segment attached to the label to limit the turn angle between two segments. - * @param linePoints - * @param minTurnAngle Radian of minimum turn angle. 0 - 180 - */ - - function limitTurnAngle(linePoints, minTurnAngle) { - if (!(minTurnAngle <= 180 && minTurnAngle > 0)) { - return; - } - - minTurnAngle = minTurnAngle / 180 * Math.PI; // The line points can be - // /pt1----pt2 (label) - // / - // pt0/ - - pt0.fromArray(linePoints[0]); - pt1.fromArray(linePoints[1]); - pt2.fromArray(linePoints[2]); - Point.sub(dir, pt0, pt1); - Point.sub(dir2, pt2, pt1); - var len1 = dir.len(); - var len2 = dir2.len(); - - if (len1 < 1e-3 || len2 < 1e-3) { - return; - } - - dir.scale(1 / len1); - dir2.scale(1 / len2); - var angleCos = dir.dot(dir2); - var minTurnAngleCos = Math.cos(minTurnAngle); - - if (minTurnAngleCos < angleCos) { - // Smaller than minTurnAngle - // Calculate project point of pt0 on pt1-pt2 - var d = projectPointToLine(pt1.x, pt1.y, pt2.x, pt2.y, pt0.x, pt0.y, tmpArr, false); - tmpProjPoint.fromArray(tmpArr); // Calculate new projected length with limited minTurnAngle and get the new connect point - - tmpProjPoint.scaleAndAdd(dir2, d / Math.tan(Math.PI - minTurnAngle)); // Limit the new calculated connect point between pt1 and pt2. - - var t = pt2.x !== pt1.x ? (tmpProjPoint.x - pt1.x) / (pt2.x - pt1.x) : (tmpProjPoint.y - pt1.y) / (pt2.y - pt1.y); - - if (isNaN(t)) { - return; - } - - if (t < 0) { - Point.copy(tmpProjPoint, pt1); - } else if (t > 1) { - Point.copy(tmpProjPoint, pt2); - } - - tmpProjPoint.toArray(linePoints[1]); - } - } - /** - * Limit the angle of line and the surface - * @param maxSurfaceAngle Radian of minimum turn angle. 0 - 180. 0 is same direction to normal. 180 is opposite - */ - - function limitSurfaceAngle(linePoints, surfaceNormal, maxSurfaceAngle) { - if (!(maxSurfaceAngle <= 180 && maxSurfaceAngle > 0)) { - return; - } - - maxSurfaceAngle = maxSurfaceAngle / 180 * Math.PI; - pt0.fromArray(linePoints[0]); - pt1.fromArray(linePoints[1]); - pt2.fromArray(linePoints[2]); - Point.sub(dir, pt1, pt0); - Point.sub(dir2, pt2, pt1); - var len1 = dir.len(); - var len2 = dir2.len(); - - if (len1 < 1e-3 || len2 < 1e-3) { - return; - } - - dir.scale(1 / len1); - dir2.scale(1 / len2); - var angleCos = dir.dot(surfaceNormal); - var maxSurfaceAngleCos = Math.cos(maxSurfaceAngle); - - if (angleCos < maxSurfaceAngleCos) { - // Calculate project point of pt0 on pt1-pt2 - var d = projectPointToLine(pt1.x, pt1.y, pt2.x, pt2.y, pt0.x, pt0.y, tmpArr, false); - tmpProjPoint.fromArray(tmpArr); - var HALF_PI = Math.PI / 2; - var angle2 = Math.acos(dir2.dot(surfaceNormal)); - var newAngle = HALF_PI + angle2 - maxSurfaceAngle; - - if (newAngle >= HALF_PI) { - // parallel - Point.copy(tmpProjPoint, pt2); - } else { - // Calculate new projected length with limited minTurnAngle and get the new connect point - tmpProjPoint.scaleAndAdd(dir2, d / Math.tan(Math.PI / 2 - newAngle)); // Limit the new calculated connect point between pt1 and pt2. - - var t = pt2.x !== pt1.x ? (tmpProjPoint.x - pt1.x) / (pt2.x - pt1.x) : (tmpProjPoint.y - pt1.y) / (pt2.y - pt1.y); - - if (isNaN(t)) { - return; - } - - if (t < 0) { - Point.copy(tmpProjPoint, pt1); - } else if (t > 1) { - Point.copy(tmpProjPoint, pt2); - } - } - - tmpProjPoint.toArray(linePoints[1]); - } - } - - function setLabelLineState(labelLine, ignore, stateName, stateModel) { - var isNormal = stateName === 'normal'; - var stateObj = isNormal ? labelLine : labelLine.ensureState(stateName); // Make sure display. - - stateObj.ignore = ignore; // Set smooth - - var smooth = stateModel.get('smooth'); - - if (smooth && smooth === true) { - smooth = 0.3; - } - - stateObj.shape = stateObj.shape || {}; - - if (smooth > 0) { - stateObj.shape.smooth = smooth; - } - - var styleObj = stateModel.getModel('lineStyle').getLineStyle(); - isNormal ? labelLine.useStyle(styleObj) : stateObj.style = styleObj; - } - - function buildLabelLinePath(path, shape) { - var smooth = shape.smooth; - var points = shape.points; - - if (!points) { - return; - } - - path.moveTo(points[0][0], points[0][1]); - - if (smooth > 0 && points.length >= 3) { - var len1 = dist(points[0], points[1]); - var len2 = dist(points[1], points[2]); - - if (!len1 || !len2) { - path.lineTo(points[1][0], points[1][1]); - path.lineTo(points[2][0], points[2][1]); - return; - } - - var moveLen = Math.min(len1, len2) * smooth; - var midPoint0 = lerp([], points[1], points[0], moveLen / len1); - var midPoint2 = lerp([], points[1], points[2], moveLen / len2); - var midPoint1 = lerp([], midPoint0, midPoint2, 0.5); - path.bezierCurveTo(midPoint0[0], midPoint0[1], midPoint0[0], midPoint0[1], midPoint1[0], midPoint1[1]); - path.bezierCurveTo(midPoint2[0], midPoint2[1], midPoint2[0], midPoint2[1], points[2][0], points[2][1]); - } else { - for (var i = 1; i < points.length; i++) { - path.lineTo(points[i][0], points[i][1]); - } - } - } - /** - * Create a label line if necessary and set it's style. - */ - - - function setLabelLineStyle(targetEl, statesModels, defaultStyle) { - var labelLine = targetEl.getTextGuideLine(); - var label = targetEl.getTextContent(); - - if (!label) { - // Not show label line if there is no label. - if (labelLine) { - targetEl.removeTextGuideLine(); - } - - return; - } - - var normalModel = statesModels.normal; - var showNormal = normalModel.get('show'); - var labelIgnoreNormal = label.ignore; - - for (var i = 0; i < DISPLAY_STATES.length; i++) { - var stateName = DISPLAY_STATES[i]; - var stateModel = statesModels[stateName]; - var isNormal = stateName === 'normal'; - - if (stateModel) { - var stateShow = stateModel.get('show'); - var isLabelIgnored = isNormal ? labelIgnoreNormal : retrieve2(label.states[stateName] && label.states[stateName].ignore, labelIgnoreNormal); - - if (isLabelIgnored // Not show when label is not shown in this state. - || !retrieve2(stateShow, showNormal) // Use normal state by default if not set. - ) { - var stateObj = isNormal ? labelLine : labelLine && labelLine.states[stateName]; - - if (stateObj) { - stateObj.ignore = true; - } - - continue; - } // Create labelLine if not exists - - - if (!labelLine) { - labelLine = new Polyline(); - targetEl.setTextGuideLine(labelLine); // Reset state of normal because it's new created. - // NOTE: NORMAL should always been the first! - - if (!isNormal && (labelIgnoreNormal || !showNormal)) { - setLabelLineState(labelLine, true, 'normal', statesModels.normal); - } // Use same state proxy. - - - if (targetEl.stateProxy) { - labelLine.stateProxy = targetEl.stateProxy; - } - } - - setLabelLineState(labelLine, false, stateName, stateModel); - } - } - - if (labelLine) { - defaults(labelLine.style, defaultStyle); // Not fill. - - labelLine.style.fill = null; - var showAbove = normalModel.get('showAbove'); - var labelLineConfig = targetEl.textGuideLineConfig = targetEl.textGuideLineConfig || {}; - labelLineConfig.showAbove = showAbove || false; // Custom the buildPath. - - labelLine.buildPath = buildLabelLinePath; - } - } - function getLabelLineStatesModels(itemModel, labelLineName) { - labelLineName = labelLineName || 'labelLine'; - var statesModels = { - normal: itemModel.getModel(labelLineName) - }; - - for (var i = 0; i < SPECIAL_STATES.length; i++) { - var stateName = SPECIAL_STATES[i]; - statesModels[stateName] = itemModel.getModel([stateName, labelLineName]); - } - - return statesModels; - } - - function prepareLayoutList(input) { - var list = []; - - for (var i = 0; i < input.length; i++) { - var rawItem = input[i]; - - if (rawItem.defaultAttr.ignore) { - continue; - } - - var label = rawItem.label; - var transform = label.getComputedTransform(); // NOTE: Get bounding rect after getComputedTransform, or label may not been updated by the host el. - - var localRect = label.getBoundingRect(); - var isAxisAligned = !transform || transform[1] < 1e-5 && transform[2] < 1e-5; - var minMargin = label.style.margin || 0; - var globalRect = localRect.clone(); - globalRect.applyTransform(transform); - globalRect.x -= minMargin / 2; - globalRect.y -= minMargin / 2; - globalRect.width += minMargin; - globalRect.height += minMargin; - var obb = isAxisAligned ? new OrientedBoundingRect(localRect, transform) : null; - list.push({ - label: label, - labelLine: rawItem.labelLine, - rect: globalRect, - localRect: localRect, - obb: obb, - priority: rawItem.priority, - defaultAttr: rawItem.defaultAttr, - layoutOption: rawItem.computedLayoutOption, - axisAligned: isAxisAligned, - transform: transform - }); - } - - return list; - } - - function shiftLayout(list, xyDim, sizeDim, minBound, maxBound, balanceShift) { - var len = list.length; - - if (len < 2) { - return; - } - - list.sort(function (a, b) { - return a.rect[xyDim] - b.rect[xyDim]; - }); - var lastPos = 0; - var delta; - var adjusted = false; - var totalShifts = 0; - - for (var i = 0; i < len; i++) { - var item = list[i]; - var rect = item.rect; - delta = rect[xyDim] - lastPos; - - if (delta < 0) { - // shiftForward(i, len, -delta); - rect[xyDim] -= delta; - item.label[xyDim] -= delta; - adjusted = true; - } - - var shift = Math.max(-delta, 0); - totalShifts += shift; - lastPos = rect[xyDim] + rect[sizeDim]; - } - - if (totalShifts > 0 && balanceShift) { - // Shift back to make the distribution more equally. - shiftList(-totalShifts / len, 0, len); - } // TODO bleedMargin? - - - var first = list[0]; - var last = list[len - 1]; - var minGap; - var maxGap; - updateMinMaxGap(); // If ends exceed two bounds, squeeze at most 80%, then take the gap of two bounds. - - minGap < 0 && squeezeGaps(-minGap, 0.8); - maxGap < 0 && squeezeGaps(maxGap, 0.8); - updateMinMaxGap(); - takeBoundsGap(minGap, maxGap, 1); - takeBoundsGap(maxGap, minGap, -1); // Handle bailout when there is not enough space. - - updateMinMaxGap(); - - if (minGap < 0) { - squeezeWhenBailout(-minGap); - } - - if (maxGap < 0) { - squeezeWhenBailout(maxGap); - } - - function updateMinMaxGap() { - minGap = first.rect[xyDim] - minBound; - maxGap = maxBound - last.rect[xyDim] - last.rect[sizeDim]; - } - - function takeBoundsGap(gapThisBound, gapOtherBound, moveDir) { - if (gapThisBound < 0) { - // Move from other gap if can. - var moveFromMaxGap = Math.min(gapOtherBound, -gapThisBound); - - if (moveFromMaxGap > 0) { - shiftList(moveFromMaxGap * moveDir, 0, len); - var remained = moveFromMaxGap + gapThisBound; - - if (remained < 0) { - squeezeGaps(-remained * moveDir, 1); - } - } else { - squeezeGaps(-gapThisBound * moveDir, 1); - } - } - } - - function shiftList(delta, start, end) { - if (delta !== 0) { - adjusted = true; - } - - for (var i = start; i < end; i++) { - var item = list[i]; - var rect = item.rect; - rect[xyDim] += delta; - item.label[xyDim] += delta; - } - } // Squeeze gaps if the labels exceed margin. - - - function squeezeGaps(delta, maxSqeezePercent) { - var gaps = []; - var totalGaps = 0; - - for (var i = 1; i < len; i++) { - var prevItemRect = list[i - 1].rect; - var gap = Math.max(list[i].rect[xyDim] - prevItemRect[xyDim] - prevItemRect[sizeDim], 0); - gaps.push(gap); - totalGaps += gap; - } - - if (!totalGaps) { - return; - } - - var squeezePercent = Math.min(Math.abs(delta) / totalGaps, maxSqeezePercent); - - if (delta > 0) { - for (var i = 0; i < len - 1; i++) { - // Distribute the shift delta to all gaps. - var movement = gaps[i] * squeezePercent; // Forward - - shiftList(movement, 0, i + 1); - } - } else { - // Backward - for (var i = len - 1; i > 0; i--) { - // Distribute the shift delta to all gaps. - var movement = gaps[i - 1] * squeezePercent; - shiftList(-movement, i, len); - } - } - } - /** - * Squeeze to allow overlap if there is no more space available. - * Let other overlapping strategy like hideOverlap do the job instead of keep exceeding the bounds. - */ - - - function squeezeWhenBailout(delta) { - var dir = delta < 0 ? -1 : 1; - delta = Math.abs(delta); - var moveForEachLabel = Math.ceil(delta / (len - 1)); - - for (var i = 0; i < len - 1; i++) { - if (dir > 0) { - // Forward - shiftList(moveForEachLabel, 0, i + 1); - } else { - // Backward - shiftList(-moveForEachLabel, len - i - 1, len); - } - - delta -= moveForEachLabel; - - if (delta <= 0) { - return; - } - } - } - - return adjusted; - } - /** - * Adjust labels on x direction to avoid overlap. - */ - - - function shiftLayoutOnX(list, leftBound, rightBound, // If average the shifts on all labels and add them to 0 - // TODO: Not sure if should enable it. - // Pros: The angle of lines will distribute more equally - // Cons: In some layout. It may not what user wanted. like in pie. the label of last sector is usually changed unexpectedly. - balanceShift) { - return shiftLayout(list, 'x', 'width', leftBound, rightBound, balanceShift); - } - /** - * Adjust labels on y direction to avoid overlap. - */ - - function shiftLayoutOnY(list, topBound, bottomBound, // If average the shifts on all labels and add them to 0 - balanceShift) { - return shiftLayout(list, 'y', 'height', topBound, bottomBound, balanceShift); - } - function hideOverlap(labelList) { - var displayedLabels = []; // TODO, render overflow visible first, put in the displayedLabels. - - labelList.sort(function (a, b) { - return b.priority - a.priority; - }); - var globalRect = new BoundingRect(0, 0, 0, 0); - - function hideEl(el) { - if (!el.ignore) { - // Show on emphasis. - var emphasisState = el.ensureState('emphasis'); - - if (emphasisState.ignore == null) { - emphasisState.ignore = false; - } - } - - el.ignore = true; - } - - for (var i = 0; i < labelList.length; i++) { - var labelItem = labelList[i]; - var isAxisAligned = labelItem.axisAligned; - var localRect = labelItem.localRect; - var transform = labelItem.transform; - var label = labelItem.label; - var labelLine = labelItem.labelLine; - globalRect.copy(labelItem.rect); // Add a threshold because layout may be aligned precisely. - - globalRect.width -= 0.1; - globalRect.height -= 0.1; - globalRect.x += 0.05; - globalRect.y += 0.05; - var obb = labelItem.obb; - var overlapped = false; - - for (var j = 0; j < displayedLabels.length; j++) { - var existsTextCfg = displayedLabels[j]; // Fast rejection. - - if (!globalRect.intersect(existsTextCfg.rect)) { - continue; - } - - if (isAxisAligned && existsTextCfg.axisAligned) { - // Is overlapped - overlapped = true; - break; - } - - if (!existsTextCfg.obb) { - // If self is not axis aligned. But other is. - existsTextCfg.obb = new OrientedBoundingRect(existsTextCfg.localRect, existsTextCfg.transform); - } - - if (!obb) { - // If self is axis aligned. But other is not. - obb = new OrientedBoundingRect(localRect, transform); - } - - if (obb.intersect(existsTextCfg.obb)) { - overlapped = true; - break; - } - } // TODO Callback to determine if this overlap should be handled? - - - if (overlapped) { - hideEl(label); - labelLine && hideEl(labelLine); - } else { - label.attr('ignore', labelItem.defaultAttr.ignore); - labelLine && labelLine.attr('ignore', labelItem.defaultAttr.labelGuideIgnore); - displayedLabels.push(labelItem); - } - } - } - - function cloneArr(points) { - if (points) { - var newPoints = []; - - for (var i = 0; i < points.length; i++) { - newPoints.push(points[i].slice()); - } - - return newPoints; - } - } - - function prepareLayoutCallbackParams(labelItem, hostEl) { - var label = labelItem.label; - var labelLine = hostEl && hostEl.getTextGuideLine(); - return { - dataIndex: labelItem.dataIndex, - dataType: labelItem.dataType, - seriesIndex: labelItem.seriesModel.seriesIndex, - text: labelItem.label.style.text, - rect: labelItem.hostRect, - labelRect: labelItem.rect, - // x: labelAttr.x, - // y: labelAttr.y, - align: label.style.align, - verticalAlign: label.style.verticalAlign, - labelLinePoints: cloneArr(labelLine && labelLine.shape.points) - }; - } - - var LABEL_OPTION_TO_STYLE_KEYS = ['align', 'verticalAlign', 'width', 'height', 'fontSize']; - var dummyTransformable = new Transformable(); - var labelLayoutInnerStore = makeInner(); - var labelLineAnimationStore = makeInner(); - - function extendWithKeys(target, source, keys) { - for (var i = 0; i < keys.length; i++) { - var key = keys[i]; - - if (source[key] != null) { - target[key] = source[key]; - } - } - } - - var LABEL_LAYOUT_PROPS = ['x', 'y', 'rotation']; - - var LabelManager = - /** @class */ - function () { - function LabelManager() { - this._labelList = []; - this._chartViewList = []; - } - - LabelManager.prototype.clearLabels = function () { - this._labelList = []; - this._chartViewList = []; - }; - /** - * Add label to manager - */ - - - LabelManager.prototype._addLabel = function (dataIndex, dataType, seriesModel, label, layoutOption) { - var labelStyle = label.style; - var hostEl = label.__hostTarget; - var textConfig = hostEl.textConfig || {}; // TODO: If label is in other state. - - var labelTransform = label.getComputedTransform(); - var labelRect = label.getBoundingRect().plain(); - BoundingRect.applyTransform(labelRect, labelRect, labelTransform); - - if (labelTransform) { - dummyTransformable.setLocalTransform(labelTransform); - } else { - // Identity transform. - dummyTransformable.x = dummyTransformable.y = dummyTransformable.rotation = dummyTransformable.originX = dummyTransformable.originY = 0; - dummyTransformable.scaleX = dummyTransformable.scaleY = 1; - } - - var host = label.__hostTarget; - var hostRect; - - if (host) { - hostRect = host.getBoundingRect().plain(); - var transform = host.getComputedTransform(); - BoundingRect.applyTransform(hostRect, hostRect, transform); - } - - var labelGuide = hostRect && host.getTextGuideLine(); - - this._labelList.push({ - label: label, - labelLine: labelGuide, - seriesModel: seriesModel, - dataIndex: dataIndex, - dataType: dataType, - layoutOption: layoutOption, - computedLayoutOption: null, - rect: labelRect, - hostRect: hostRect, - // Label with lower priority will be hidden when overlapped - // Use rect size as default priority - priority: hostRect ? hostRect.width * hostRect.height : 0, - // Save default label attributes. - // For restore if developers want get back to default value in callback. - defaultAttr: { - ignore: label.ignore, - labelGuideIgnore: labelGuide && labelGuide.ignore, - x: dummyTransformable.x, - y: dummyTransformable.y, - scaleX: dummyTransformable.scaleX, - scaleY: dummyTransformable.scaleY, - rotation: dummyTransformable.rotation, - style: { - x: labelStyle.x, - y: labelStyle.y, - align: labelStyle.align, - verticalAlign: labelStyle.verticalAlign, - width: labelStyle.width, - height: labelStyle.height, - fontSize: labelStyle.fontSize - }, - cursor: label.cursor, - attachedPos: textConfig.position, - attachedRot: textConfig.rotation - } - }); - }; - - LabelManager.prototype.addLabelsOfSeries = function (chartView) { - var _this = this; - - this._chartViewList.push(chartView); - - var seriesModel = chartView.__model; - var layoutOption = seriesModel.get('labelLayout'); - /** - * Ignore layouting if it's not specified anything. - */ - - if (!(isFunction(layoutOption) || keys(layoutOption).length)) { - return; - } - - chartView.group.traverse(function (child) { - if (child.ignore) { - return true; // Stop traverse descendants. - } // Only support label being hosted on graphic elements. - - - var textEl = child.getTextContent(); - var ecData = getECData(child); // Can only attach the text on the element with dataIndex - - if (textEl && !textEl.disableLabelLayout) { - _this._addLabel(ecData.dataIndex, ecData.dataType, seriesModel, textEl, layoutOption); - } - }); - }; - - LabelManager.prototype.updateLayoutConfig = function (api) { - var width = api.getWidth(); - var height = api.getHeight(); - - function createDragHandler(el, labelLineModel) { - return function () { - updateLabelLinePoints(el, labelLineModel); - }; - } - - for (var i = 0; i < this._labelList.length; i++) { - var labelItem = this._labelList[i]; - var label = labelItem.label; - var hostEl = label.__hostTarget; - var defaultLabelAttr = labelItem.defaultAttr; - var layoutOption = void 0; // TODO A global layout option? - - if (isFunction(labelItem.layoutOption)) { - layoutOption = labelItem.layoutOption(prepareLayoutCallbackParams(labelItem, hostEl)); - } else { - layoutOption = labelItem.layoutOption; - } - - layoutOption = layoutOption || {}; - labelItem.computedLayoutOption = layoutOption; - var degreeToRadian = Math.PI / 180; // TODO hostEl should always exists. - // Or label should not have parent because the x, y is all in global space. - - if (hostEl) { - hostEl.setTextConfig({ - // Force to set local false. - local: false, - // Ignore position and rotation config on the host el if x or y is changed. - position: layoutOption.x != null || layoutOption.y != null ? null : defaultLabelAttr.attachedPos, - // Ignore rotation config on the host el if rotation is changed. - rotation: layoutOption.rotate != null ? layoutOption.rotate * degreeToRadian : defaultLabelAttr.attachedRot, - offset: [layoutOption.dx || 0, layoutOption.dy || 0] - }); - } - - var needsUpdateLabelLine = false; - - if (layoutOption.x != null) { - // TODO width of chart view. - label.x = parsePercent$1(layoutOption.x, width); - label.setStyle('x', 0); // Ignore movement in style. TODO: origin. - - needsUpdateLabelLine = true; - } else { - label.x = defaultLabelAttr.x; - label.setStyle('x', defaultLabelAttr.style.x); - } - - if (layoutOption.y != null) { - // TODO height of chart view. - label.y = parsePercent$1(layoutOption.y, height); - label.setStyle('y', 0); // Ignore movement in style. - - needsUpdateLabelLine = true; - } else { - label.y = defaultLabelAttr.y; - label.setStyle('y', defaultLabelAttr.style.y); - } - - if (layoutOption.labelLinePoints) { - var guideLine = hostEl.getTextGuideLine(); - - if (guideLine) { - guideLine.setShape({ - points: layoutOption.labelLinePoints - }); // Not update - - needsUpdateLabelLine = false; - } - } - - var labelLayoutStore = labelLayoutInnerStore(label); - labelLayoutStore.needsUpdateLabelLine = needsUpdateLabelLine; - label.rotation = layoutOption.rotate != null ? layoutOption.rotate * degreeToRadian : defaultLabelAttr.rotation; - label.scaleX = defaultLabelAttr.scaleX; - label.scaleY = defaultLabelAttr.scaleY; - - for (var k = 0; k < LABEL_OPTION_TO_STYLE_KEYS.length; k++) { - var key = LABEL_OPTION_TO_STYLE_KEYS[k]; - label.setStyle(key, layoutOption[key] != null ? layoutOption[key] : defaultLabelAttr.style[key]); - } - - if (layoutOption.draggable) { - label.draggable = true; - label.cursor = 'move'; - - if (hostEl) { - var hostModel = labelItem.seriesModel; - - if (labelItem.dataIndex != null) { - var data = labelItem.seriesModel.getData(labelItem.dataType); - hostModel = data.getItemModel(labelItem.dataIndex); - } - - label.on('drag', createDragHandler(hostEl, hostModel.getModel('labelLine'))); - } - } else { - // TODO Other drag functions? - label.off('drag'); - label.cursor = defaultLabelAttr.cursor; - } - } - }; - - LabelManager.prototype.layout = function (api) { - var width = api.getWidth(); - var height = api.getHeight(); - var labelList = prepareLayoutList(this._labelList); - var labelsNeedsAdjustOnX = filter(labelList, function (item) { - return item.layoutOption.moveOverlap === 'shiftX'; - }); - var labelsNeedsAdjustOnY = filter(labelList, function (item) { - return item.layoutOption.moveOverlap === 'shiftY'; - }); - shiftLayoutOnX(labelsNeedsAdjustOnX, 0, width); - shiftLayoutOnY(labelsNeedsAdjustOnY, 0, height); - var labelsNeedsHideOverlap = filter(labelList, function (item) { - return item.layoutOption.hideOverlap; - }); - hideOverlap(labelsNeedsHideOverlap); - }; - /** - * Process all labels. Not only labels with layoutOption. - */ - - - LabelManager.prototype.processLabelsOverall = function () { - var _this = this; - - each(this._chartViewList, function (chartView) { - var seriesModel = chartView.__model; - var ignoreLabelLineUpdate = chartView.ignoreLabelLineUpdate; - var animationEnabled = seriesModel.isAnimationEnabled(); - chartView.group.traverse(function (child) { - if (child.ignore && !child.forceLabelAnimation) { - return true; // Stop traverse descendants. - } - - var needsUpdateLabelLine = !ignoreLabelLineUpdate; - var label = child.getTextContent(); - - if (!needsUpdateLabelLine && label) { - needsUpdateLabelLine = labelLayoutInnerStore(label).needsUpdateLabelLine; - } - - if (needsUpdateLabelLine) { - _this._updateLabelLine(child, seriesModel); - } - - if (animationEnabled) { - _this._animateLabels(child, seriesModel); - } - }); - }); - }; - - LabelManager.prototype._updateLabelLine = function (el, seriesModel) { - // Only support label being hosted on graphic elements. - var textEl = el.getTextContent(); // Update label line style. - - var ecData = getECData(el); - var dataIndex = ecData.dataIndex; // Only support labelLine on the labels represent data. - - if (textEl && dataIndex != null) { - var data = seriesModel.getData(ecData.dataType); - var itemModel = data.getItemModel(dataIndex); - var defaultStyle = {}; - var visualStyle = data.getItemVisual(dataIndex, 'style'); - var visualType = data.getVisual('drawType'); // Default to be same with main color - - defaultStyle.stroke = visualStyle[visualType]; - var labelLineModel = itemModel.getModel('labelLine'); - setLabelLineStyle(el, getLabelLineStatesModels(itemModel), defaultStyle); - updateLabelLinePoints(el, labelLineModel); - } - }; - - LabelManager.prototype._animateLabels = function (el, seriesModel) { - var textEl = el.getTextContent(); - var guideLine = el.getTextGuideLine(); // Animate - - if (textEl // `forceLabelAnimation` has the highest priority - && (el.forceLabelAnimation || !textEl.ignore && !textEl.invisible && !el.disableLabelAnimation && !isElementRemoved(el))) { - var layoutStore = labelLayoutInnerStore(textEl); - var oldLayout = layoutStore.oldLayout; - var ecData = getECData(el); - var dataIndex = ecData.dataIndex; - var newProps = { - x: textEl.x, - y: textEl.y, - rotation: textEl.rotation - }; - var data = seriesModel.getData(ecData.dataType); - - if (!oldLayout) { - textEl.attr(newProps); // Disable fade in animation if value animation is enabled. - - if (!labelInner(textEl).valueAnimation) { - var oldOpacity = retrieve2(textEl.style.opacity, 1); // Fade in animation - - textEl.style.opacity = 0; - initProps(textEl, { - style: { - opacity: oldOpacity - } - }, seriesModel, dataIndex); - } - } else { - textEl.attr(oldLayout); // Make sure the animation from is in the right status. - - var prevStates = el.prevStates; - - if (prevStates) { - if (indexOf(prevStates, 'select') >= 0) { - textEl.attr(layoutStore.oldLayoutSelect); - } - - if (indexOf(prevStates, 'emphasis') >= 0) { - textEl.attr(layoutStore.oldLayoutEmphasis); - } - } - - updateProps(textEl, newProps, seriesModel, dataIndex); - } - - layoutStore.oldLayout = newProps; - - if (textEl.states.select) { - var layoutSelect = layoutStore.oldLayoutSelect = {}; - extendWithKeys(layoutSelect, newProps, LABEL_LAYOUT_PROPS); - extendWithKeys(layoutSelect, textEl.states.select, LABEL_LAYOUT_PROPS); - } - - if (textEl.states.emphasis) { - var layoutEmphasis = layoutStore.oldLayoutEmphasis = {}; - extendWithKeys(layoutEmphasis, newProps, LABEL_LAYOUT_PROPS); - extendWithKeys(layoutEmphasis, textEl.states.emphasis, LABEL_LAYOUT_PROPS); - } - - animateLabelValue(textEl, dataIndex, data, seriesModel, seriesModel); - } - - if (guideLine && !guideLine.ignore && !guideLine.invisible) { - var layoutStore = labelLineAnimationStore(guideLine); - var oldLayout = layoutStore.oldLayout; - var newLayout = { - points: guideLine.shape.points - }; - - if (!oldLayout) { - guideLine.setShape(newLayout); - guideLine.style.strokePercent = 0; - initProps(guideLine, { - style: { - strokePercent: 1 - } - }, seriesModel); - } else { - guideLine.attr({ - shape: oldLayout - }); - updateProps(guideLine, { - shape: newLayout - }, seriesModel); - } - - layoutStore.oldLayout = newLayout; - } - }; - - return LabelManager; - }(); - - var getLabelManager = makeInner(); - function installLabelLayout(registers) { - registers.registerUpdateLifecycle('series:beforeupdate', function (ecModel, api, params) { - // TODO api provide an namespace that can save stuff per instance - var labelManager = getLabelManager(api).labelManager; - - if (!labelManager) { - labelManager = getLabelManager(api).labelManager = new LabelManager(); - } - - labelManager.clearLabels(); - }); - registers.registerUpdateLifecycle('series:layoutlabels', function (ecModel, api, params) { - var labelManager = getLabelManager(api).labelManager; - params.updatedSeries.forEach(function (series) { - labelManager.addLabelsOfSeries(api.getViewOfSeriesModel(series)); - }); - labelManager.updateLayoutConfig(api); - labelManager.layout(api); - labelManager.processLabelsOverall(); - }); - } - - var mathSin$4 = Math.sin; - var mathCos$4 = Math.cos; - var PI$4 = Math.PI; - var PI2$7 = Math.PI * 2; - var degree = 180 / PI$4; - var SVGPathRebuilder = (function () { - function SVGPathRebuilder() { - } - SVGPathRebuilder.prototype.reset = function (precision) { - this._start = true; - this._d = []; - this._str = ''; - this._p = Math.pow(10, precision || 4); - }; - SVGPathRebuilder.prototype.moveTo = function (x, y) { - this._add('M', x, y); - }; - SVGPathRebuilder.prototype.lineTo = function (x, y) { - this._add('L', x, y); - }; - SVGPathRebuilder.prototype.bezierCurveTo = function (x, y, x2, y2, x3, y3) { - this._add('C', x, y, x2, y2, x3, y3); - }; - SVGPathRebuilder.prototype.quadraticCurveTo = function (x, y, x2, y2) { - this._add('Q', x, y, x2, y2); - }; - SVGPathRebuilder.prototype.arc = function (cx, cy, r, startAngle, endAngle, anticlockwise) { - this.ellipse(cx, cy, r, r, 0, startAngle, endAngle, anticlockwise); - }; - SVGPathRebuilder.prototype.ellipse = function (cx, cy, rx, ry, psi, startAngle, endAngle, anticlockwise) { - var dTheta = endAngle - startAngle; - var clockwise = !anticlockwise; - var dThetaPositive = Math.abs(dTheta); - var isCircle = isAroundZero$1(dThetaPositive - PI2$7) - || (clockwise ? dTheta >= PI2$7 : -dTheta >= PI2$7); - var unifiedTheta = dTheta > 0 ? dTheta % PI2$7 : (dTheta % PI2$7 + PI2$7); - var large = false; - if (isCircle) { - large = true; - } - else if (isAroundZero$1(dThetaPositive)) { - large = false; - } - else { - large = (unifiedTheta >= PI$4) === !!clockwise; - } - var x0 = cx + rx * mathCos$4(startAngle); - var y0 = cy + ry * mathSin$4(startAngle); - if (this._start) { - this._add('M', x0, y0); - } - var xRot = Math.round(psi * degree); - if (isCircle) { - var p = 1 / this._p; - var dTheta_1 = (clockwise ? 1 : -1) * (PI2$7 - p); - this._add('A', rx, ry, xRot, 1, +clockwise, cx + rx * mathCos$4(startAngle + dTheta_1), cy + ry * mathSin$4(startAngle + dTheta_1)); - if (p > 1e-2) { - this._add('A', rx, ry, xRot, 0, +clockwise, x0, y0); - } - } - else { - var x = cx + rx * mathCos$4(endAngle); - var y = cy + ry * mathSin$4(endAngle); - this._add('A', rx, ry, xRot, +large, +clockwise, x, y); - } - }; - SVGPathRebuilder.prototype.rect = function (x, y, w, h) { - this._add('M', x, y); - this._add('l', w, 0); - this._add('l', 0, h); - this._add('l', -w, 0); - this._add('Z'); - }; - SVGPathRebuilder.prototype.closePath = function () { - if (this._d.length > 0) { - this._add('Z'); - } - }; - SVGPathRebuilder.prototype._add = function (cmd, a, b, c, d, e, f, g, h) { - var vals = []; - var p = this._p; - for (var i = 1; i < arguments.length; i++) { - var val = arguments[i]; - if (isNaN(val)) { - this._invalid = true; - return; - } - vals.push(Math.round(val * p) / p); - } - this._d.push(cmd + vals.join(' ')); - this._start = cmd === 'Z'; - }; - SVGPathRebuilder.prototype.generateStr = function () { - this._str = this._invalid ? '' : this._d.join(''); - this._d = []; - }; - SVGPathRebuilder.prototype.getStr = function () { - return this._str; - }; - return SVGPathRebuilder; - }()); - - var NONE = 'none'; - var mathRound$1 = Math.round; - function pathHasFill(style) { - var fill = style.fill; - return fill != null && fill !== NONE; - } - function pathHasStroke(style) { - var stroke = style.stroke; - return stroke != null && stroke !== NONE; - } - var strokeProps = ['lineCap', 'miterLimit', 'lineJoin']; - var svgStrokeProps = map(strokeProps, function (prop) { return "stroke-" + prop.toLowerCase(); }); - function mapStyleToAttrs(updateAttr, style, el, forceUpdate) { - var opacity = style.opacity == null ? 1 : style.opacity; - if (el instanceof ZRImage) { - updateAttr('opacity', opacity); - return; - } - if (pathHasFill(style)) { - var fill = normalizeColor(style.fill); - updateAttr('fill', fill.color); - var fillOpacity = style.fillOpacity != null - ? style.fillOpacity * fill.opacity * opacity - : fill.opacity * opacity; - if (forceUpdate || fillOpacity < 1) { - updateAttr('fill-opacity', fillOpacity); - } - } - else { - updateAttr('fill', NONE); - } - if (pathHasStroke(style)) { - var stroke = normalizeColor(style.stroke); - updateAttr('stroke', stroke.color); - var strokeScale = style.strokeNoScale - ? el.getLineScale() - : 1; - var strokeWidth = (strokeScale ? (style.lineWidth || 0) / strokeScale : 0); - var strokeOpacity = style.strokeOpacity != null - ? style.strokeOpacity * stroke.opacity * opacity - : stroke.opacity * opacity; - var strokeFirst = style.strokeFirst; - if (forceUpdate || strokeWidth !== 1) { - updateAttr('stroke-width', strokeWidth); - } - if (forceUpdate || strokeFirst) { - updateAttr('paint-order', strokeFirst ? 'stroke' : 'fill'); - } - if (forceUpdate || strokeOpacity < 1) { - updateAttr('stroke-opacity', strokeOpacity); - } - if (style.lineDash) { - var _a = getLineDash(el), lineDash = _a[0], lineDashOffset = _a[1]; - if (lineDash) { - lineDashOffset = mathRound$1(lineDashOffset || 0); - updateAttr('stroke-dasharray', lineDash.join(',')); - if (lineDashOffset || forceUpdate) { - updateAttr('stroke-dashoffset', lineDashOffset); - } - } - } - else if (forceUpdate) { - updateAttr('stroke-dasharray', NONE); - } - for (var i = 0; i < strokeProps.length; i++) { - var propName = strokeProps[i]; - if (forceUpdate || style[propName] !== DEFAULT_PATH_STYLE[propName]) { - var val = style[propName] || DEFAULT_PATH_STYLE[propName]; - val && updateAttr(svgStrokeProps[i], val); - } - } - } - else if (forceUpdate) { - updateAttr('stroke', NONE); - } - } - - var SVGNS = 'http://www.w3.org/2000/svg'; - var XLINKNS = 'http://www.w3.org/1999/xlink'; - var XMLNS = 'http://www.w3.org/2000/xmlns/'; - var XML_NAMESPACE = 'http://www.w3.org/XML/1998/namespace'; - function createElement(name) { - return document.createElementNS(SVGNS, name); - } - function createVNode(tag, key, attrs, children, text) { - return { - tag: tag, - attrs: attrs || {}, - children: children, - text: text, - key: key - }; - } - function createElementOpen(name, attrs) { - var attrsStr = []; - if (attrs) { - for (var key in attrs) { - var val = attrs[key]; - var part = key; - if (val === false) { - continue; - } - else if (val !== true && val != null) { - part += "=\"" + val + "\""; - } - attrsStr.push(part); - } - } - return "<" + name + " " + attrsStr.join(' ') + ">"; - } - function createElementClose(name) { - return "</" + name + ">"; - } - function vNodeToString(el, opts) { - opts = opts || {}; - var S = opts.newline ? '\n' : ''; - function convertElToString(el) { - var children = el.children, tag = el.tag, attrs = el.attrs, text = el.text; - return createElementOpen(tag, attrs) - + (tag !== 'style' ? encodeHTML(text) : text || '') - + (children ? "" + S + map(children, function (child) { return convertElToString(child); }).join(S) + S : '') - + createElementClose(tag); - } - return convertElToString(el); - } - function getCssString(selectorNodes, animationNodes, opts) { - opts = opts || {}; - var S = opts.newline ? '\n' : ''; - var bracketBegin = " {" + S; - var bracketEnd = S + "}"; - var selectors = map(keys(selectorNodes), function (className) { - return className + bracketBegin + map(keys(selectorNodes[className]), function (attrName) { - return attrName + ":" + selectorNodes[className][attrName] + ";"; - }).join(S) + bracketEnd; - }).join(S); - var animations = map(keys(animationNodes), function (animationName) { - return "@keyframes " + animationName + bracketBegin + map(keys(animationNodes[animationName]), function (percent) { - return percent + bracketBegin + map(keys(animationNodes[animationName][percent]), function (attrName) { - var val = animationNodes[animationName][percent][attrName]; - if (attrName === 'd') { - val = "path(\"" + val + "\")"; - } - return attrName + ":" + val + ";"; - }).join(S) + bracketEnd; - }).join(S) + bracketEnd; - }).join(S); - if (!selectors && !animations) { - return ''; - } - return ['<![CDATA[', selectors, animations, ']]>'].join(S); - } - function createBrushScope(zrId) { - return { - zrId: zrId, - shadowCache: {}, - patternCache: {}, - gradientCache: {}, - clipPathCache: {}, - defs: {}, - cssNodes: {}, - cssAnims: {}, - cssClassIdx: 0, - cssAnimIdx: 0, - shadowIdx: 0, - gradientIdx: 0, - patternIdx: 0, - clipPathIdx: 0 - }; - } - function createSVGVNode(width, height, children, useViewBox) { - return createVNode('svg', 'root', { - 'width': width, - 'height': height, - 'xmlns': SVGNS, - 'xmlns:xlink': XLINKNS, - 'version': '1.1', - 'baseProfile': 'full', - 'viewBox': useViewBox ? "0 0 " + width + " " + height : false - }, children); - } - - var EASING_MAP = { - cubicIn: '0.32,0,0.67,0', - cubicOut: '0.33,1,0.68,1', - cubicInOut: '0.65,0,0.35,1', - quadraticIn: '0.11,0,0.5,0', - quadraticOut: '0.5,1,0.89,1', - quadraticInOut: '0.45,0,0.55,1', - quarticIn: '0.5,0,0.75,0', - quarticOut: '0.25,1,0.5,1', - quarticInOut: '0.76,0,0.24,1', - quinticIn: '0.64,0,0.78,0', - quinticOut: '0.22,1,0.36,1', - quinticInOut: '0.83,0,0.17,1', - sinusoidalIn: '0.12,0,0.39,0', - sinusoidalOut: '0.61,1,0.88,1', - sinusoidalInOut: '0.37,0,0.63,1', - exponentialIn: '0.7,0,0.84,0', - exponentialOut: '0.16,1,0.3,1', - exponentialInOut: '0.87,0,0.13,1', - circularIn: '0.55,0,1,0.45', - circularOut: '0,0.55,0.45,1', - circularInOut: '0.85,0,0.15,1' - }; - var transformOriginKey = 'transform-origin'; - function buildPathString(el, kfShape, path) { - var shape = extend({}, el.shape); - extend(shape, kfShape); - el.buildPath(path, shape); - var svgPathBuilder = new SVGPathRebuilder(); - svgPathBuilder.reset(getPathPrecision(el)); - path.rebuildPath(svgPathBuilder, 1); - svgPathBuilder.generateStr(); - return svgPathBuilder.getStr(); - } - function setTransformOrigin(target, transform) { - var originX = transform.originX, originY = transform.originY; - if (originX || originY) { - target[transformOriginKey] = originX + "px " + originY + "px"; - } - } - var ANIMATE_STYLE_MAP = { - fill: 'fill', - opacity: 'opacity', - lineWidth: 'stroke-width', - lineDashOffset: 'stroke-dashoffset' - }; - function addAnimation(cssAnim, scope) { - var animationName = scope.zrId + '-ani-' + scope.cssAnimIdx++; - scope.cssAnims[animationName] = cssAnim; - return animationName; - } - function createCompoundPathCSSAnimation(el, attrs, scope) { - var paths = el.shape.paths; - var composedAnim = {}; - var cssAnimationCfg; - var cssAnimationName; - each(paths, function (path) { - var subScope = createBrushScope(scope.zrId); - subScope.animation = true; - createCSSAnimation(path, {}, subScope, true); - var cssAnims = subScope.cssAnims; - var cssNodes = subScope.cssNodes; - var animNames = keys(cssAnims); - var len = animNames.length; - if (!len) { - return; - } - cssAnimationName = animNames[len - 1]; - var lastAnim = cssAnims[cssAnimationName]; - for (var percent in lastAnim) { - var kf = lastAnim[percent]; - composedAnim[percent] = composedAnim[percent] || { d: '' }; - composedAnim[percent].d += kf.d || ''; - } - for (var className in cssNodes) { - var val = cssNodes[className].animation; - if (val.indexOf(cssAnimationName) >= 0) { - cssAnimationCfg = val; - } - } - }); - if (!cssAnimationCfg) { - return; - } - attrs.d = false; - var animationName = addAnimation(composedAnim, scope); - return cssAnimationCfg.replace(cssAnimationName, animationName); - } - function getEasingFunc(easing) { - return isString(easing) - ? EASING_MAP[easing] - ? "cubic-bezier(" + EASING_MAP[easing] + ")" - : createCubicEasingFunc(easing) ? easing : '' - : ''; - } - function createCSSAnimation(el, attrs, scope, onlyShape) { - var animators = el.animators; - var len = animators.length; - var cssAnimations = []; - if (el instanceof CompoundPath) { - var animationCfg = createCompoundPathCSSAnimation(el, attrs, scope); - if (animationCfg) { - cssAnimations.push(animationCfg); - } - else if (!len) { - return; - } - } - else if (!len) { - return; - } - var groupAnimators = {}; - for (var i = 0; i < len; i++) { - var animator = animators[i]; - var cfgArr = [animator.getMaxTime() / 1000 + 's']; - var easing = getEasingFunc(animator.getClip().easing); - var delay = animator.getDelay(); - if (easing) { - cfgArr.push(easing); - } - else { - cfgArr.push('linear'); - } - if (delay) { - cfgArr.push(delay / 1000 + 's'); - } - if (animator.getLoop()) { - cfgArr.push('infinite'); - } - var cfg = cfgArr.join(' '); - groupAnimators[cfg] = groupAnimators[cfg] || [cfg, []]; - groupAnimators[cfg][1].push(animator); - } - function createSingleCSSAnimation(groupAnimator) { - var animators = groupAnimator[1]; - var len = animators.length; - var transformKfs = {}; - var shapeKfs = {}; - var finalKfs = {}; - var animationTimingFunctionAttrName = 'animation-timing-function'; - function saveAnimatorTrackToCssKfs(animator, cssKfs, toCssAttrName) { - var tracks = animator.getTracks(); - var maxTime = animator.getMaxTime(); - for (var k = 0; k < tracks.length; k++) { - var track = tracks[k]; - if (track.needsAnimate()) { - var kfs = track.keyframes; - var attrName = track.propName; - toCssAttrName && (attrName = toCssAttrName(attrName)); - if (attrName) { - for (var i = 0; i < kfs.length; i++) { - var kf = kfs[i]; - var percent = Math.round(kf.time / maxTime * 100) + '%'; - var kfEasing = getEasingFunc(kf.easing); - var rawValue = kf.rawValue; - if (isString(rawValue) || isNumber(rawValue)) { - cssKfs[percent] = cssKfs[percent] || {}; - cssKfs[percent][attrName] = kf.rawValue; - if (kfEasing) { - cssKfs[percent][animationTimingFunctionAttrName] = kfEasing; - } - } - } - } - } - } - } - for (var i = 0; i < len; i++) { - var animator = animators[i]; - var targetProp = animator.targetName; - if (!targetProp) { - !onlyShape && saveAnimatorTrackToCssKfs(animator, transformKfs); - } - else if (targetProp === 'shape') { - saveAnimatorTrackToCssKfs(animator, shapeKfs); - } - } - for (var percent in transformKfs) { - var transform = {}; - copyTransform(transform, el); - extend(transform, transformKfs[percent]); - var str = getSRTTransformString(transform); - var timingFunction = transformKfs[percent][animationTimingFunctionAttrName]; - finalKfs[percent] = str ? { - transform: str - } : {}; - setTransformOrigin(finalKfs[percent], transform); - if (timingFunction) { - finalKfs[percent][animationTimingFunctionAttrName] = timingFunction; - } - } - var path; - var canAnimateShape = true; - for (var percent in shapeKfs) { - finalKfs[percent] = finalKfs[percent] || {}; - var isFirst = !path; - var timingFunction = shapeKfs[percent][animationTimingFunctionAttrName]; - if (isFirst) { - path = new PathProxy(); - } - var len_1 = path.len(); - path.reset(); - finalKfs[percent].d = buildPathString(el, shapeKfs[percent], path); - var newLen = path.len(); - if (!isFirst && len_1 !== newLen) { - canAnimateShape = false; - break; - } - if (timingFunction) { - finalKfs[percent][animationTimingFunctionAttrName] = timingFunction; - } - } - if (!canAnimateShape) { - for (var percent in finalKfs) { - delete finalKfs[percent].d; - } - } - if (!onlyShape) { - for (var i = 0; i < len; i++) { - var animator = animators[i]; - var targetProp = animator.targetName; - if (targetProp === 'style') { - saveAnimatorTrackToCssKfs(animator, finalKfs, function (propName) { return ANIMATE_STYLE_MAP[propName]; }); - } - } - } - var percents = keys(finalKfs); - var allTransformOriginSame = true; - var transformOrigin; - for (var i = 1; i < percents.length; i++) { - var p0 = percents[i - 1]; - var p1 = percents[i]; - if (finalKfs[p0][transformOriginKey] !== finalKfs[p1][transformOriginKey]) { - allTransformOriginSame = false; - break; - } - transformOrigin = finalKfs[p0][transformOriginKey]; - } - if (allTransformOriginSame && transformOrigin) { - for (var percent in finalKfs) { - if (finalKfs[percent][transformOriginKey]) { - delete finalKfs[percent][transformOriginKey]; - } - } - attrs[transformOriginKey] = transformOrigin; - } - if (filter(percents, function (percent) { return keys(finalKfs[percent]).length > 0; }).length) { - var animationName = addAnimation(finalKfs, scope); - return animationName + " " + groupAnimator[0] + " both"; - } - } - for (var key in groupAnimators) { - var animationCfg = createSingleCSSAnimation(groupAnimators[key]); - if (animationCfg) { - cssAnimations.push(animationCfg); - } - } - if (cssAnimations.length) { - var className = scope.zrId + '-cls-' + scope.cssClassIdx++; - scope.cssNodes['.' + className] = { - animation: cssAnimations.join(',') - }; - attrs["class"] = className; - } - } - - var round$2 = Math.round; - function isImageLike$1(val) { - return val && isString(val.src); - } - function isCanvasLike(val) { - return val && isFunction(val.toDataURL); - } - function setStyleAttrs(attrs, style, el, scope) { - mapStyleToAttrs(function (key, val) { - var isFillStroke = key === 'fill' || key === 'stroke'; - if (isFillStroke && isGradient(val)) { - setGradient(style, attrs, key, scope); - } - else if (isFillStroke && isPattern(val)) { - setPattern(el, attrs, key, scope); - } - else { - attrs[key] = val; - } - }, style, el, false); - setShadow(el, attrs, scope); - } - function noRotateScale(m) { - return isAroundZero$1(m[0] - 1) - && isAroundZero$1(m[1]) - && isAroundZero$1(m[2]) - && isAroundZero$1(m[3] - 1); - } - function noTranslate(m) { - return isAroundZero$1(m[4]) && isAroundZero$1(m[5]); - } - function setTransform(attrs, m, compress) { - if (m && !(noTranslate(m) && noRotateScale(m))) { - var mul = compress ? 10 : 1e4; - attrs.transform = noRotateScale(m) - ? "translate(" + round$2(m[4] * mul) / mul + " " + round$2(m[5] * mul) / mul + ")" : getMatrixStr(m); - } - } - function convertPolyShape(shape, attrs, mul) { - var points = shape.points; - var strArr = []; - for (var i = 0; i < points.length; i++) { - strArr.push(round$2(points[i][0] * mul) / mul); - strArr.push(round$2(points[i][1] * mul) / mul); - } - attrs.points = strArr.join(' '); - } - function validatePolyShape(shape) { - return !shape.smooth; - } - function createAttrsConvert(desc) { - var normalizedDesc = map(desc, function (item) { - return (typeof item === 'string' ? [item, item] : item); - }); - return function (shape, attrs, mul) { - for (var i = 0; i < normalizedDesc.length; i++) { - var item = normalizedDesc[i]; - var val = shape[item[0]]; - if (val != null) { - attrs[item[1]] = round$2(val * mul) / mul; - } - } - }; - } - var builtinShapesDef = { - circle: [createAttrsConvert(['cx', 'cy', 'r'])], - polyline: [convertPolyShape, validatePolyShape], - polygon: [convertPolyShape, validatePolyShape] - }; - function hasShapeAnimation(el) { - var animators = el.animators; - for (var i = 0; i < animators.length; i++) { - if (animators[i].targetName === 'shape') { - return true; - } - } - return false; - } - function brushSVGPath(el, scope) { - var style = el.style; - var shape = el.shape; - var builtinShpDef = builtinShapesDef[el.type]; - var attrs = {}; - var needsAnimate = scope.animation; - var svgElType = 'path'; - var strokePercent = el.style.strokePercent; - var precision = (scope.compress && getPathPrecision(el)) || 4; - if (builtinShpDef - && !scope.willUpdate - && !(builtinShpDef[1] && !builtinShpDef[1](shape)) - && !(needsAnimate && hasShapeAnimation(el)) - && !(strokePercent < 1)) { - svgElType = el.type; - var mul = Math.pow(10, precision); - builtinShpDef[0](shape, attrs, mul); - } - else { - var needBuildPath = !el.path || el.shapeChanged(); - if (!el.path) { - el.createPathProxy(); - } - var path = el.path; - if (needBuildPath) { - path.beginPath(); - el.buildPath(path, el.shape); - el.pathUpdated(); - } - var pathVersion = path.getVersion(); - var elExt = el; - var svgPathBuilder = elExt.__svgPathBuilder; - if (elExt.__svgPathVersion !== pathVersion - || !svgPathBuilder - || strokePercent !== elExt.__svgPathStrokePercent) { - if (!svgPathBuilder) { - svgPathBuilder = elExt.__svgPathBuilder = new SVGPathRebuilder(); - } - svgPathBuilder.reset(precision); - path.rebuildPath(svgPathBuilder, strokePercent); - svgPathBuilder.generateStr(); - elExt.__svgPathVersion = pathVersion; - elExt.__svgPathStrokePercent = strokePercent; - } - attrs.d = svgPathBuilder.getStr(); - } - setTransform(attrs, el.transform); - setStyleAttrs(attrs, style, el, scope); - scope.animation && createCSSAnimation(el, attrs, scope); - return createVNode(svgElType, el.id + '', attrs); - } - function brushSVGImage(el, scope) { - var style = el.style; - var image = style.image; - if (image && !isString(image)) { - if (isImageLike$1(image)) { - image = image.src; - } - else if (isCanvasLike(image)) { - image = image.toDataURL(); - } - } - if (!image) { - return; - } - var x = style.x || 0; - var y = style.y || 0; - var dw = style.width; - var dh = style.height; - var attrs = { - href: image, - width: dw, - height: dh - }; - if (x) { - attrs.x = x; - } - if (y) { - attrs.y = y; - } - setTransform(attrs, el.transform); - setStyleAttrs(attrs, style, el, scope); - scope.animation && createCSSAnimation(el, attrs, scope); - return createVNode('image', el.id + '', attrs); - } - function brushSVGTSpan(el, scope) { - var style = el.style; - var text = style.text; - text != null && (text += ''); - if (!text || isNaN(style.x) || isNaN(style.y)) { - return; - } - var font = style.font || DEFAULT_FONT; - var x = style.x || 0; - var y = adjustTextY(style.y || 0, getLineHeight(font), style.textBaseline); - var textAlign = TEXT_ALIGN_TO_ANCHOR[style.textAlign] - || style.textAlign; - var attrs = { - 'dominant-baseline': 'central', - 'text-anchor': textAlign - }; - if (hasSeparateFont(style)) { - var separatedFontStr = ''; - var fontStyle = style.fontStyle; - var fontSize = parseFontSize(style.fontSize); - if (!parseFloat(fontSize)) { - return; - } - var fontFamily = style.fontFamily || DEFAULT_FONT_FAMILY; - var fontWeight = style.fontWeight; - separatedFontStr += "font-size:" + fontSize + ";font-family:" + fontFamily + ";"; - if (fontStyle && fontStyle !== 'normal') { - separatedFontStr += "font-style:" + fontStyle + ";"; - } - if (fontWeight && fontWeight !== 'normal') { - separatedFontStr += "font-weight:" + fontWeight + ";"; - } - attrs.style = separatedFontStr; - } - else { - attrs.style = "font: " + font; - } - if (text.match(/\s/)) { - attrs['xml:space'] = 'preserve'; - } - if (x) { - attrs.x = x; - } - if (y) { - attrs.y = y; - } - setTransform(attrs, el.transform); - setStyleAttrs(attrs, style, el, scope); - scope.animation && createCSSAnimation(el, attrs, scope); - return createVNode('text', el.id + '', attrs, undefined, text); - } - function brush$1(el, scope) { - if (el instanceof Path) { - return brushSVGPath(el, scope); - } - else if (el instanceof ZRImage) { - return brushSVGImage(el, scope); - } - else if (el instanceof TSpan) { - return brushSVGTSpan(el, scope); - } - } - function setShadow(el, attrs, scope) { - var style = el.style; - if (hasShadow(style)) { - var shadowKey = getShadowKey(el); - var shadowCache = scope.shadowCache; - var shadowId = shadowCache[shadowKey]; - if (!shadowId) { - var globalScale = el.getGlobalScale(); - var scaleX = globalScale[0]; - var scaleY = globalScale[1]; - if (!scaleX || !scaleY) { - return; - } - var offsetX = style.shadowOffsetX || 0; - var offsetY = style.shadowOffsetY || 0; - var blur_1 = style.shadowBlur; - var _a = normalizeColor(style.shadowColor), opacity = _a.opacity, color = _a.color; - var stdDx = blur_1 / 2 / scaleX; - var stdDy = blur_1 / 2 / scaleY; - var stdDeviation = stdDx + ' ' + stdDy; - shadowId = scope.zrId + '-s' + scope.shadowIdx++; - scope.defs[shadowId] = createVNode('filter', shadowId, { - 'id': shadowId, - 'x': '-100%', - 'y': '-100%', - 'width': '300%', - 'height': '300%' - }, [ - createVNode('feDropShadow', '', { - 'dx': offsetX / scaleX, - 'dy': offsetY / scaleY, - 'stdDeviation': stdDeviation, - 'flood-color': color, - 'flood-opacity': opacity - }) - ]); - shadowCache[shadowKey] = shadowId; - } - attrs.filter = getIdURL(shadowId); - } - } - function setGradient(style, attrs, target, scope) { - var val = style[target]; - var gradientTag; - var gradientAttrs = { - 'gradientUnits': val.global - ? 'userSpaceOnUse' - : 'objectBoundingBox' - }; - if (isLinearGradient(val)) { - gradientTag = 'linearGradient'; - gradientAttrs.x1 = val.x; - gradientAttrs.y1 = val.y; - gradientAttrs.x2 = val.x2; - gradientAttrs.y2 = val.y2; - } - else if (isRadialGradient(val)) { - gradientTag = 'radialGradient'; - gradientAttrs.cx = retrieve2(val.x, 0.5); - gradientAttrs.cy = retrieve2(val.y, 0.5); - gradientAttrs.r = retrieve2(val.r, 0.5); - } - else { - if ("development" !== 'production') { - logError('Illegal gradient type.'); - } - return; - } - var colors = val.colorStops; - var colorStops = []; - for (var i = 0, len = colors.length; i < len; ++i) { - var offset = round4(colors[i].offset) * 100 + '%'; - var stopColor = colors[i].color; - var _a = normalizeColor(stopColor), color = _a.color, opacity = _a.opacity; - var stopsAttrs = { - 'offset': offset - }; - stopsAttrs['stop-color'] = color; - if (opacity < 1) { - stopsAttrs['stop-opacity'] = opacity; - } - colorStops.push(createVNode('stop', i + '', stopsAttrs)); - } - var gradientVNode = createVNode(gradientTag, '', gradientAttrs, colorStops); - var gradientKey = vNodeToString(gradientVNode); - var gradientCache = scope.gradientCache; - var gradientId = gradientCache[gradientKey]; - if (!gradientId) { - gradientId = scope.zrId + '-g' + scope.gradientIdx++; - gradientCache[gradientKey] = gradientId; - gradientAttrs.id = gradientId; - scope.defs[gradientId] = createVNode(gradientTag, gradientId, gradientAttrs, colorStops); - } - attrs[target] = getIdURL(gradientId); - } - function setPattern(el, attrs, target, scope) { - var val = el.style[target]; - var boundingRect = el.getBoundingRect(); - var patternAttrs = {}; - var repeat = val.repeat; - var noRepeat = repeat === 'no-repeat'; - var repeatX = repeat === 'repeat-x'; - var repeatY = repeat === 'repeat-y'; - var child; - if (isImagePattern(val)) { - var imageWidth_1 = val.imageWidth; - var imageHeight_1 = val.imageHeight; - var imageSrc = void 0; - var patternImage = val.image; - if (isString(patternImage)) { - imageSrc = patternImage; - } - else if (isImageLike$1(patternImage)) { - imageSrc = patternImage.src; - } - else if (isCanvasLike(patternImage)) { - imageSrc = patternImage.toDataURL(); - } - if (typeof Image === 'undefined') { - var errMsg = 'Image width/height must been given explictly in svg-ssr renderer.'; - assert(imageWidth_1, errMsg); - assert(imageHeight_1, errMsg); - } - else if (imageWidth_1 == null || imageHeight_1 == null) { - var setSizeToVNode_1 = function (vNode, img) { - if (vNode) { - var svgEl = vNode.elm; - var width = imageWidth_1 || img.width; - var height = imageHeight_1 || img.height; - if (vNode.tag === 'pattern') { - if (repeatX) { - height = 1; - width /= boundingRect.width; - } - else if (repeatY) { - width = 1; - height /= boundingRect.height; - } - } - vNode.attrs.width = width; - vNode.attrs.height = height; - if (svgEl) { - svgEl.setAttribute('width', width); - svgEl.setAttribute('height', height); - } - } - }; - var createdImage = createOrUpdateImage(imageSrc, null, el, function (img) { - noRepeat || setSizeToVNode_1(patternVNode, img); - setSizeToVNode_1(child, img); - }); - if (createdImage && createdImage.width && createdImage.height) { - imageWidth_1 = imageWidth_1 || createdImage.width; - imageHeight_1 = imageHeight_1 || createdImage.height; - } - } - child = createVNode('image', 'img', { - href: imageSrc, - width: imageWidth_1, - height: imageHeight_1 - }); - patternAttrs.width = imageWidth_1; - patternAttrs.height = imageHeight_1; - } - else if (val.svgElement) { - child = clone(val.svgElement); - patternAttrs.width = val.svgWidth; - patternAttrs.height = val.svgHeight; - } - if (!child) { - return; - } - var patternWidth; - var patternHeight; - if (noRepeat) { - patternWidth = patternHeight = 1; - } - else if (repeatX) { - patternHeight = 1; - patternWidth = patternAttrs.width / boundingRect.width; - } - else if (repeatY) { - patternWidth = 1; - patternHeight = patternAttrs.height / boundingRect.height; - } - else { - patternAttrs.patternUnits = 'userSpaceOnUse'; - } - if (patternWidth != null && !isNaN(patternWidth)) { - patternAttrs.width = patternWidth; - } - if (patternHeight != null && !isNaN(patternHeight)) { - patternAttrs.height = patternHeight; - } - var patternTransform = getSRTTransformString(val); - patternTransform && (patternAttrs.patternTransform = patternTransform); - var patternVNode = createVNode('pattern', '', patternAttrs, [child]); - var patternKey = vNodeToString(patternVNode); - var patternCache = scope.patternCache; - var patternId = patternCache[patternKey]; - if (!patternId) { - patternId = scope.zrId + '-p' + scope.patternIdx++; - patternCache[patternKey] = patternId; - patternAttrs.id = patternId; - patternVNode = scope.defs[patternId] = createVNode('pattern', patternId, patternAttrs, [child]); - } - attrs[target] = getIdURL(patternId); - } - function setClipPath(clipPath, attrs, scope) { - var clipPathCache = scope.clipPathCache, defs = scope.defs; - var clipPathId = clipPathCache[clipPath.id]; - if (!clipPathId) { - clipPathId = scope.zrId + '-c' + scope.clipPathIdx++; - var clipPathAttrs = { - id: clipPathId - }; - clipPathCache[clipPath.id] = clipPathId; - defs[clipPathId] = createVNode('clipPath', clipPathId, clipPathAttrs, [brushSVGPath(clipPath, scope)]); - } - attrs['clip-path'] = getIdURL(clipPathId); - } - - function createTextNode(text) { - return document.createTextNode(text); - } - function insertBefore(parentNode, newNode, referenceNode) { - parentNode.insertBefore(newNode, referenceNode); - } - function removeChild(node, child) { - node.removeChild(child); - } - function appendChild(node, child) { - node.appendChild(child); - } - function parentNode(node) { - return node.parentNode; - } - function nextSibling(node) { - return node.nextSibling; - } - function setTextContent(node, text) { - node.textContent = text; - } - - var colonChar = 58; - var xChar = 120; - var emptyNode = createVNode('', ''); - function isUndef(s) { - return s === undefined; - } - function isDef(s) { - return s !== undefined; - } - function createKeyToOldIdx(children, beginIdx, endIdx) { - var map = {}; - for (var i = beginIdx; i <= endIdx; ++i) { - var key = children[i].key; - if (key !== undefined) { - if ("development" !== 'production') { - if (map[key] != null) { - console.error("Duplicate key " + key); - } - } - map[key] = i; - } - } - return map; - } - function sameVnode(vnode1, vnode2) { - var isSameKey = vnode1.key === vnode2.key; - var isSameTag = vnode1.tag === vnode2.tag; - return isSameTag && isSameKey; - } - function createElm(vnode) { - var i; - var children = vnode.children; - var tag = vnode.tag; - if (isDef(tag)) { - var elm = (vnode.elm = createElement(tag)); - updateAttrs(emptyNode, vnode); - if (isArray(children)) { - for (i = 0; i < children.length; ++i) { - var ch = children[i]; - if (ch != null) { - appendChild(elm, createElm(ch)); - } - } - } - else if (isDef(vnode.text) && !isObject(vnode.text)) { - appendChild(elm, createTextNode(vnode.text)); - } - } - else { - vnode.elm = createTextNode(vnode.text); - } - return vnode.elm; - } - function addVnodes(parentElm, before, vnodes, startIdx, endIdx) { - for (; startIdx <= endIdx; ++startIdx) { - var ch = vnodes[startIdx]; - if (ch != null) { - insertBefore(parentElm, createElm(ch), before); - } - } - } - function removeVnodes(parentElm, vnodes, startIdx, endIdx) { - for (; startIdx <= endIdx; ++startIdx) { - var ch = vnodes[startIdx]; - if (ch != null) { - if (isDef(ch.tag)) { - var parent_1 = parentNode(ch.elm); - removeChild(parent_1, ch.elm); - } - else { - removeChild(parentElm, ch.elm); - } - } - } - } - function updateAttrs(oldVnode, vnode) { - var key; - var elm = vnode.elm; - var oldAttrs = oldVnode && oldVnode.attrs || {}; - var attrs = vnode.attrs || {}; - if (oldAttrs === attrs) { - return; - } - for (key in attrs) { - var cur = attrs[key]; - var old = oldAttrs[key]; - if (old !== cur) { - if (cur === true) { - elm.setAttribute(key, ''); - } - else if (cur === false) { - elm.removeAttribute(key); - } - else { - if (key.charCodeAt(0) !== xChar) { - elm.setAttribute(key, cur); - } - else if (key === 'xmlns:xlink' || key === 'xmlns') { - elm.setAttributeNS(XMLNS, key, cur); - } - else if (key.charCodeAt(3) === colonChar) { - elm.setAttributeNS(XML_NAMESPACE, key, cur); - } - else if (key.charCodeAt(5) === colonChar) { - elm.setAttributeNS(XLINKNS, key, cur); - } - else { - elm.setAttribute(key, cur); - } - } - } - } - for (key in oldAttrs) { - if (!(key in attrs)) { - elm.removeAttribute(key); - } - } - } - function updateChildren(parentElm, oldCh, newCh) { - var oldStartIdx = 0; - var newStartIdx = 0; - var oldEndIdx = oldCh.length - 1; - var oldStartVnode = oldCh[0]; - var oldEndVnode = oldCh[oldEndIdx]; - var newEndIdx = newCh.length - 1; - var newStartVnode = newCh[0]; - var newEndVnode = newCh[newEndIdx]; - var oldKeyToIdx; - var idxInOld; - var elmToMove; - var before; - while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) { - if (oldStartVnode == null) { - oldStartVnode = oldCh[++oldStartIdx]; - } - else if (oldEndVnode == null) { - oldEndVnode = oldCh[--oldEndIdx]; - } - else if (newStartVnode == null) { - newStartVnode = newCh[++newStartIdx]; - } - else if (newEndVnode == null) { - newEndVnode = newCh[--newEndIdx]; - } - else if (sameVnode(oldStartVnode, newStartVnode)) { - patchVnode(oldStartVnode, newStartVnode); - oldStartVnode = oldCh[++oldStartIdx]; - newStartVnode = newCh[++newStartIdx]; - } - else if (sameVnode(oldEndVnode, newEndVnode)) { - patchVnode(oldEndVnode, newEndVnode); - oldEndVnode = oldCh[--oldEndIdx]; - newEndVnode = newCh[--newEndIdx]; - } - else if (sameVnode(oldStartVnode, newEndVnode)) { - patchVnode(oldStartVnode, newEndVnode); - insertBefore(parentElm, oldStartVnode.elm, nextSibling(oldEndVnode.elm)); - oldStartVnode = oldCh[++oldStartIdx]; - newEndVnode = newCh[--newEndIdx]; - } - else if (sameVnode(oldEndVnode, newStartVnode)) { - patchVnode(oldEndVnode, newStartVnode); - insertBefore(parentElm, oldEndVnode.elm, oldStartVnode.elm); - oldEndVnode = oldCh[--oldEndIdx]; - newStartVnode = newCh[++newStartIdx]; - } - else { - if (isUndef(oldKeyToIdx)) { - oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx); - } - idxInOld = oldKeyToIdx[newStartVnode.key]; - if (isUndef(idxInOld)) { - insertBefore(parentElm, createElm(newStartVnode), oldStartVnode.elm); - } - else { - elmToMove = oldCh[idxInOld]; - if (elmToMove.tag !== newStartVnode.tag) { - insertBefore(parentElm, createElm(newStartVnode), oldStartVnode.elm); - } - else { - patchVnode(elmToMove, newStartVnode); - oldCh[idxInOld] = undefined; - insertBefore(parentElm, elmToMove.elm, oldStartVnode.elm); - } - } - newStartVnode = newCh[++newStartIdx]; - } - } - if (oldStartIdx <= oldEndIdx || newStartIdx <= newEndIdx) { - if (oldStartIdx > oldEndIdx) { - before = newCh[newEndIdx + 1] == null ? null : newCh[newEndIdx + 1].elm; - addVnodes(parentElm, before, newCh, newStartIdx, newEndIdx); - } - else { - removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx); - } - } - } - function patchVnode(oldVnode, vnode) { - var elm = (vnode.elm = oldVnode.elm); - var oldCh = oldVnode.children; - var ch = vnode.children; - if (oldVnode === vnode) { - return; - } - updateAttrs(oldVnode, vnode); - if (isUndef(vnode.text)) { - if (isDef(oldCh) && isDef(ch)) { - if (oldCh !== ch) { - updateChildren(elm, oldCh, ch); - } - } - else if (isDef(ch)) { - if (isDef(oldVnode.text)) { - setTextContent(elm, ''); - } - addVnodes(elm, null, ch, 0, ch.length - 1); - } - else if (isDef(oldCh)) { - removeVnodes(elm, oldCh, 0, oldCh.length - 1); - } - else if (isDef(oldVnode.text)) { - setTextContent(elm, ''); - } - } - else if (oldVnode.text !== vnode.text) { - if (isDef(oldCh)) { - removeVnodes(elm, oldCh, 0, oldCh.length - 1); - } - setTextContent(elm, vnode.text); - } - } - function patch(oldVnode, vnode) { - if (sameVnode(oldVnode, vnode)) { - patchVnode(oldVnode, vnode); - } - else { - var elm = oldVnode.elm; - var parent_2 = parentNode(elm); - createElm(vnode); - if (parent_2 !== null) { - insertBefore(parent_2, vnode.elm, nextSibling(elm)); - removeVnodes(parent_2, [oldVnode], 0, 0); - } - } - return vnode; - } - - var svgId = 0; - var SVGPainter = (function () { - function SVGPainter(root, storage, opts) { - this.type = 'svg'; - this.refreshHover = createMethodNotSupport('refreshHover'); - this.configLayer = createMethodNotSupport('configLayer'); - this.storage = storage; - this._opts = opts = extend({}, opts); - this.root = root; - this._id = 'zr' + svgId++; - this._oldVNode = createSVGVNode(opts.width, opts.height); - if (root && !opts.ssr) { - var viewport = this._viewport = document.createElement('div'); - viewport.style.cssText = 'position:relative;overflow:hidden'; - var svgDom = this._svgDom = this._oldVNode.elm = createElement('svg'); - updateAttrs(null, this._oldVNode); - viewport.appendChild(svgDom); - root.appendChild(viewport); - } - this.resize(opts.width, opts.height); - } - SVGPainter.prototype.getType = function () { - return this.type; - }; - SVGPainter.prototype.getViewportRoot = function () { - return this._viewport; - }; - SVGPainter.prototype.getViewportRootOffset = function () { - var viewportRoot = this.getViewportRoot(); - if (viewportRoot) { - return { - offsetLeft: viewportRoot.offsetLeft || 0, - offsetTop: viewportRoot.offsetTop || 0 - }; - } - }; - SVGPainter.prototype.getSvgDom = function () { - return this._svgDom; - }; - SVGPainter.prototype.refresh = function () { - if (this.root) { - var vnode = this.renderToVNode({ - willUpdate: true - }); - vnode.attrs.style = 'position:absolute;left:0;top:0;user-select:none'; - patch(this._oldVNode, vnode); - this._oldVNode = vnode; - } - }; - SVGPainter.prototype.renderOneToVNode = function (el) { - return brush$1(el, createBrushScope(this._id)); - }; - SVGPainter.prototype.renderToVNode = function (opts) { - opts = opts || {}; - var list = this.storage.getDisplayList(true); - var width = this._width; - var height = this._height; - var scope = createBrushScope(this._id); - scope.animation = opts.animation; - scope.willUpdate = opts.willUpdate; - scope.compress = opts.compress; - var children = []; - var bgVNode = this._bgVNode = createBackgroundVNode(width, height, this._backgroundColor, scope); - bgVNode && children.push(bgVNode); - var mainVNode = !opts.compress - ? (this._mainVNode = createVNode('g', 'main', {}, [])) : null; - this._paintList(list, scope, mainVNode ? mainVNode.children : children); - mainVNode && children.push(mainVNode); - var defs = map(keys(scope.defs), function (id) { return scope.defs[id]; }); - if (defs.length) { - children.push(createVNode('defs', 'defs', {}, defs)); - } - if (opts.animation) { - var animationCssStr = getCssString(scope.cssNodes, scope.cssAnims, { newline: true }); - if (animationCssStr) { - var styleNode = createVNode('style', 'stl', {}, [], animationCssStr); - children.push(styleNode); - } - } - return createSVGVNode(width, height, children, opts.useViewBox); - }; - SVGPainter.prototype.renderToString = function (opts) { - opts = opts || {}; - return vNodeToString(this.renderToVNode({ - animation: retrieve2(opts.cssAnimation, true), - willUpdate: false, - compress: true, - useViewBox: retrieve2(opts.useViewBox, true) - }), { newline: true }); - }; - SVGPainter.prototype.setBackgroundColor = function (backgroundColor) { - this._backgroundColor = backgroundColor; - }; - SVGPainter.prototype.getSvgRoot = function () { - return this._mainVNode && this._mainVNode.elm; - }; - SVGPainter.prototype._paintList = function (list, scope, out) { - var listLen = list.length; - var clipPathsGroupsStack = []; - var clipPathsGroupsStackDepth = 0; - var currentClipPathGroup; - var prevClipPaths; - var clipGroupNodeIdx = 0; - for (var i = 0; i < listLen; i++) { - var displayable = list[i]; - if (!displayable.invisible) { - var clipPaths = displayable.__clipPaths; - var len = clipPaths && clipPaths.length || 0; - var prevLen = prevClipPaths && prevClipPaths.length || 0; - var lca = void 0; - for (lca = Math.max(len - 1, prevLen - 1); lca >= 0; lca--) { - if (clipPaths && prevClipPaths - && clipPaths[lca] === prevClipPaths[lca]) { - break; - } - } - for (var i_1 = prevLen - 1; i_1 > lca; i_1--) { - clipPathsGroupsStackDepth--; - currentClipPathGroup = clipPathsGroupsStack[clipPathsGroupsStackDepth - 1]; - } - for (var i_2 = lca + 1; i_2 < len; i_2++) { - var groupAttrs = {}; - setClipPath(clipPaths[i_2], groupAttrs, scope); - var g = createVNode('g', 'clip-g-' + clipGroupNodeIdx++, groupAttrs, []); - (currentClipPathGroup ? currentClipPathGroup.children : out).push(g); - clipPathsGroupsStack[clipPathsGroupsStackDepth++] = g; - currentClipPathGroup = g; - } - prevClipPaths = clipPaths; - var ret = brush$1(displayable, scope); - if (ret) { - (currentClipPathGroup ? currentClipPathGroup.children : out).push(ret); - } - } - } - }; - SVGPainter.prototype.resize = function (width, height) { - var opts = this._opts; - var root = this.root; - var viewport = this._viewport; - width != null && (opts.width = width); - height != null && (opts.height = height); - if (root && viewport) { - viewport.style.display = 'none'; - width = getSize(root, 0, opts); - height = getSize(root, 1, opts); - viewport.style.display = ''; - } - if (this._width !== width || this._height !== height) { - this._width = width; - this._height = height; - if (viewport) { - var viewportStyle = viewport.style; - viewportStyle.width = width + 'px'; - viewportStyle.height = height + 'px'; - } - if (!isPattern(this._backgroundColor)) { - var svgDom = this._svgDom; - if (svgDom) { - svgDom.setAttribute('width', width); - svgDom.setAttribute('height', height); - } - var bgEl = this._bgVNode && this._bgVNode.elm; - if (bgEl) { - bgEl.setAttribute('width', width); - bgEl.setAttribute('height', height); - } - } - else { - this.refresh(); - } - } - }; - SVGPainter.prototype.getWidth = function () { - return this._width; - }; - SVGPainter.prototype.getHeight = function () { - return this._height; - }; - SVGPainter.prototype.dispose = function () { - if (this.root) { - this.root.innerHTML = ''; - } - this._svgDom = - this._viewport = - this.storage = - this._oldVNode = - this._bgVNode = - this._mainVNode = null; - }; - SVGPainter.prototype.clear = function () { - if (this._svgDom) { - this._svgDom.innerHTML = null; - } - this._oldVNode = null; - }; - SVGPainter.prototype.toDataURL = function (base64) { - var str = this.renderToString(); - var prefix = 'data:image/svg+xml;'; - if (base64) { - str = encodeBase64(str); - return str && prefix + 'base64,' + str; - } - return prefix + 'charset=UTF-8,' + encodeURIComponent(str); - }; - return SVGPainter; - }()); - function createMethodNotSupport(method) { - return function () { - if ("development" !== 'production') { - logError('In SVG mode painter not support method "' + method + '"'); - } - }; - } - function createBackgroundVNode(width, height, backgroundColor, scope) { - var bgVNode; - if (backgroundColor && backgroundColor !== 'none') { - bgVNode = createVNode('rect', 'bg', { - width: width, - height: height, - x: '0', - y: '0', - id: '0' - }); - if (isGradient(backgroundColor)) { - setGradient({ fill: backgroundColor }, bgVNode.attrs, 'fill', scope); - } - else if (isPattern(backgroundColor)) { - setPattern({ - style: { - fill: backgroundColor - }, - dirty: noop, - getBoundingRect: function () { return ({ width: width, height: height }); } - }, bgVNode.attrs, 'fill', scope); - } - else { - var _a = normalizeColor(backgroundColor), color = _a.color, opacity = _a.opacity; - bgVNode.attrs.fill = color; - opacity < 1 && (bgVNode.attrs['fill-opacity'] = opacity); - } - } - return bgVNode; - } - - function install(registers) { - registers.registerPainter('svg', SVGPainter); - } - - function createDom(id, painter, dpr) { - var newDom = platformApi.createCanvas(); - var width = painter.getWidth(); - var height = painter.getHeight(); - var newDomStyle = newDom.style; - if (newDomStyle) { - newDomStyle.position = 'absolute'; - newDomStyle.left = '0'; - newDomStyle.top = '0'; - newDomStyle.width = width + 'px'; - newDomStyle.height = height + 'px'; - newDom.setAttribute('data-zr-dom-id', id); - } - newDom.width = width * dpr; - newDom.height = height * dpr; - return newDom; - } - var Layer = (function (_super) { - __extends(Layer, _super); - function Layer(id, painter, dpr) { - var _this = _super.call(this) || this; - _this.motionBlur = false; - _this.lastFrameAlpha = 0.7; - _this.dpr = 1; - _this.virtual = false; - _this.config = {}; - _this.incremental = false; - _this.zlevel = 0; - _this.maxRepaintRectCount = 5; - _this.__dirty = true; - _this.__firstTimePaint = true; - _this.__used = false; - _this.__drawIndex = 0; - _this.__startIndex = 0; - _this.__endIndex = 0; - _this.__prevStartIndex = null; - _this.__prevEndIndex = null; - var dom; - dpr = dpr || devicePixelRatio; - if (typeof id === 'string') { - dom = createDom(id, painter, dpr); - } - else if (isObject(id)) { - dom = id; - id = dom.id; - } - _this.id = id; - _this.dom = dom; - var domStyle = dom.style; - if (domStyle) { - disableUserSelect(dom); - dom.onselectstart = function () { return false; }; - domStyle.padding = '0'; - domStyle.margin = '0'; - domStyle.borderWidth = '0'; - } - _this.painter = painter; - _this.dpr = dpr; - return _this; - } - Layer.prototype.getElementCount = function () { - return this.__endIndex - this.__startIndex; - }; - Layer.prototype.afterBrush = function () { - this.__prevStartIndex = this.__startIndex; - this.__prevEndIndex = this.__endIndex; - }; - Layer.prototype.initContext = function () { - this.ctx = this.dom.getContext('2d'); - this.ctx.dpr = this.dpr; - }; - Layer.prototype.setUnpainted = function () { - this.__firstTimePaint = true; - }; - Layer.prototype.createBackBuffer = function () { - var dpr = this.dpr; - this.domBack = createDom('back-' + this.id, this.painter, dpr); - this.ctxBack = this.domBack.getContext('2d'); - if (dpr !== 1) { - this.ctxBack.scale(dpr, dpr); - } - }; - Layer.prototype.createRepaintRects = function (displayList, prevList, viewWidth, viewHeight) { - if (this.__firstTimePaint) { - this.__firstTimePaint = false; - return null; - } - var mergedRepaintRects = []; - var maxRepaintRectCount = this.maxRepaintRectCount; - var full = false; - var pendingRect = new BoundingRect(0, 0, 0, 0); - function addRectToMergePool(rect) { - if (!rect.isFinite() || rect.isZero()) { - return; - } - if (mergedRepaintRects.length === 0) { - var boundingRect = new BoundingRect(0, 0, 0, 0); - boundingRect.copy(rect); - mergedRepaintRects.push(boundingRect); - } - else { - var isMerged = false; - var minDeltaArea = Infinity; - var bestRectToMergeIdx = 0; - for (var i = 0; i < mergedRepaintRects.length; ++i) { - var mergedRect = mergedRepaintRects[i]; - if (mergedRect.intersect(rect)) { - var pendingRect_1 = new BoundingRect(0, 0, 0, 0); - pendingRect_1.copy(mergedRect); - pendingRect_1.union(rect); - mergedRepaintRects[i] = pendingRect_1; - isMerged = true; - break; - } - else if (full) { - pendingRect.copy(rect); - pendingRect.union(mergedRect); - var aArea = rect.width * rect.height; - var bArea = mergedRect.width * mergedRect.height; - var pendingArea = pendingRect.width * pendingRect.height; - var deltaArea = pendingArea - aArea - bArea; - if (deltaArea < minDeltaArea) { - minDeltaArea = deltaArea; - bestRectToMergeIdx = i; - } - } - } - if (full) { - mergedRepaintRects[bestRectToMergeIdx].union(rect); - isMerged = true; - } - if (!isMerged) { - var boundingRect = new BoundingRect(0, 0, 0, 0); - boundingRect.copy(rect); - mergedRepaintRects.push(boundingRect); - } - if (!full) { - full = mergedRepaintRects.length >= maxRepaintRectCount; - } - } - } - for (var i = this.__startIndex; i < this.__endIndex; ++i) { - var el = displayList[i]; - if (el) { - var shouldPaint = el.shouldBePainted(viewWidth, viewHeight, true, true); - var prevRect = el.__isRendered && ((el.__dirty & REDRAW_BIT) || !shouldPaint) - ? el.getPrevPaintRect() - : null; - if (prevRect) { - addRectToMergePool(prevRect); - } - var curRect = shouldPaint && ((el.__dirty & REDRAW_BIT) || !el.__isRendered) - ? el.getPaintRect() - : null; - if (curRect) { - addRectToMergePool(curRect); - } - } - } - for (var i = this.__prevStartIndex; i < this.__prevEndIndex; ++i) { - var el = prevList[i]; - var shouldPaint = el.shouldBePainted(viewWidth, viewHeight, true, true); - if (el && (!shouldPaint || !el.__zr) && el.__isRendered) { - var prevRect = el.getPrevPaintRect(); - if (prevRect) { - addRectToMergePool(prevRect); - } - } - } - var hasIntersections; - do { - hasIntersections = false; - for (var i = 0; i < mergedRepaintRects.length;) { - if (mergedRepaintRects[i].isZero()) { - mergedRepaintRects.splice(i, 1); - continue; - } - for (var j = i + 1; j < mergedRepaintRects.length;) { - if (mergedRepaintRects[i].intersect(mergedRepaintRects[j])) { - hasIntersections = true; - mergedRepaintRects[i].union(mergedRepaintRects[j]); - mergedRepaintRects.splice(j, 1); - } - else { - j++; - } - } - i++; - } - } while (hasIntersections); - this._paintRects = mergedRepaintRects; - return mergedRepaintRects; - }; - Layer.prototype.debugGetPaintRects = function () { - return (this._paintRects || []).slice(); - }; - Layer.prototype.resize = function (width, height) { - var dpr = this.dpr; - var dom = this.dom; - var domStyle = dom.style; - var domBack = this.domBack; - if (domStyle) { - domStyle.width = width + 'px'; - domStyle.height = height + 'px'; - } - dom.width = width * dpr; - dom.height = height * dpr; - if (domBack) { - domBack.width = width * dpr; - domBack.height = height * dpr; - if (dpr !== 1) { - this.ctxBack.scale(dpr, dpr); - } - } - }; - Layer.prototype.clear = function (clearAll, clearColor, repaintRects) { - var dom = this.dom; - var ctx = this.ctx; - var width = dom.width; - var height = dom.height; - clearColor = clearColor || this.clearColor; - var haveMotionBLur = this.motionBlur && !clearAll; - var lastFrameAlpha = this.lastFrameAlpha; - var dpr = this.dpr; - var self = this; - if (haveMotionBLur) { - if (!this.domBack) { - this.createBackBuffer(); - } - this.ctxBack.globalCompositeOperation = 'copy'; - this.ctxBack.drawImage(dom, 0, 0, width / dpr, height / dpr); - } - var domBack = this.domBack; - function doClear(x, y, width, height) { - ctx.clearRect(x, y, width, height); - if (clearColor && clearColor !== 'transparent') { - var clearColorGradientOrPattern = void 0; - if (isGradientObject(clearColor)) { - var shouldCache = clearColor.global || (clearColor.__width === width - && clearColor.__height === height); - clearColorGradientOrPattern = shouldCache - && clearColor.__canvasGradient - || getCanvasGradient(ctx, clearColor, { - x: 0, - y: 0, - width: width, - height: height - }); - clearColor.__canvasGradient = clearColorGradientOrPattern; - clearColor.__width = width; - clearColor.__height = height; - } - else if (isImagePatternObject(clearColor)) { - clearColor.scaleX = clearColor.scaleX || dpr; - clearColor.scaleY = clearColor.scaleY || dpr; - clearColorGradientOrPattern = createCanvasPattern(ctx, clearColor, { - dirty: function () { - self.setUnpainted(); - self.__painter.refresh(); - } - }); - } - ctx.save(); - ctx.fillStyle = clearColorGradientOrPattern || clearColor; - ctx.fillRect(x, y, width, height); - ctx.restore(); - } - if (haveMotionBLur) { - ctx.save(); - ctx.globalAlpha = lastFrameAlpha; - ctx.drawImage(domBack, x, y, width, height); - ctx.restore(); - } - } - if (!repaintRects || haveMotionBLur) { - doClear(0, 0, width, height); - } - else if (repaintRects.length) { - each(repaintRects, function (rect) { - doClear(rect.x * dpr, rect.y * dpr, rect.width * dpr, rect.height * dpr); - }); - } - }; - return Layer; - }(Eventful)); - - var HOVER_LAYER_ZLEVEL = 1e5; - var CANVAS_ZLEVEL = 314159; - var EL_AFTER_INCREMENTAL_INC = 0.01; - var INCREMENTAL_INC = 0.001; - function isLayerValid(layer) { - if (!layer) { - return false; - } - if (layer.__builtin__) { - return true; - } - if (typeof (layer.resize) !== 'function' - || typeof (layer.refresh) !== 'function') { - return false; - } - return true; - } - function createRoot(width, height) { - var domRoot = document.createElement('div'); - domRoot.style.cssText = [ - 'position:relative', - 'width:' + width + 'px', - 'height:' + height + 'px', - 'padding:0', - 'margin:0', - 'border-width:0' - ].join(';') + ';'; - return domRoot; - } - var CanvasPainter = (function () { - function CanvasPainter(root, storage, opts, id) { - this.type = 'canvas'; - this._zlevelList = []; - this._prevDisplayList = []; - this._layers = {}; - this._layerConfig = {}; - this._needsManuallyCompositing = false; - this.type = 'canvas'; - var singleCanvas = !root.nodeName - || root.nodeName.toUpperCase() === 'CANVAS'; - this._opts = opts = extend({}, opts || {}); - this.dpr = opts.devicePixelRatio || devicePixelRatio; - this._singleCanvas = singleCanvas; - this.root = root; - var rootStyle = root.style; - if (rootStyle) { - disableUserSelect(root); - root.innerHTML = ''; - } - this.storage = storage; - var zlevelList = this._zlevelList; - this._prevDisplayList = []; - var layers = this._layers; - if (!singleCanvas) { - this._width = getSize(root, 0, opts); - this._height = getSize(root, 1, opts); - var domRoot = this._domRoot = createRoot(this._width, this._height); - root.appendChild(domRoot); - } - else { - var rootCanvas = root; - var width = rootCanvas.width; - var height = rootCanvas.height; - if (opts.width != null) { - width = opts.width; - } - if (opts.height != null) { - height = opts.height; - } - this.dpr = opts.devicePixelRatio || 1; - rootCanvas.width = width * this.dpr; - rootCanvas.height = height * this.dpr; - this._width = width; - this._height = height; - var mainLayer = new Layer(rootCanvas, this, this.dpr); - mainLayer.__builtin__ = true; - mainLayer.initContext(); - layers[CANVAS_ZLEVEL] = mainLayer; - mainLayer.zlevel = CANVAS_ZLEVEL; - zlevelList.push(CANVAS_ZLEVEL); - this._domRoot = root; - } - } - CanvasPainter.prototype.getType = function () { - return 'canvas'; - }; - CanvasPainter.prototype.isSingleCanvas = function () { - return this._singleCanvas; - }; - CanvasPainter.prototype.getViewportRoot = function () { - return this._domRoot; - }; - CanvasPainter.prototype.getViewportRootOffset = function () { - var viewportRoot = this.getViewportRoot(); - if (viewportRoot) { - return { - offsetLeft: viewportRoot.offsetLeft || 0, - offsetTop: viewportRoot.offsetTop || 0 - }; - } - }; - CanvasPainter.prototype.refresh = function (paintAll) { - var list = this.storage.getDisplayList(true); - var prevList = this._prevDisplayList; - var zlevelList = this._zlevelList; - this._redrawId = Math.random(); - this._paintList(list, prevList, paintAll, this._redrawId); - for (var i = 0; i < zlevelList.length; i++) { - var z = zlevelList[i]; - var layer = this._layers[z]; - if (!layer.__builtin__ && layer.refresh) { - var clearColor = i === 0 ? this._backgroundColor : null; - layer.refresh(clearColor); - } - } - if (this._opts.useDirtyRect) { - this._prevDisplayList = list.slice(); - } - return this; - }; - CanvasPainter.prototype.refreshHover = function () { - this._paintHoverList(this.storage.getDisplayList(false)); - }; - CanvasPainter.prototype._paintHoverList = function (list) { - var len = list.length; - var hoverLayer = this._hoverlayer; - hoverLayer && hoverLayer.clear(); - if (!len) { - return; - } - var scope = { - inHover: true, - viewWidth: this._width, - viewHeight: this._height - }; - var ctx; - for (var i = 0; i < len; i++) { - var el = list[i]; - if (el.__inHover) { - if (!hoverLayer) { - hoverLayer = this._hoverlayer = this.getLayer(HOVER_LAYER_ZLEVEL); - } - if (!ctx) { - ctx = hoverLayer.ctx; - ctx.save(); - } - brush(ctx, el, scope, i === len - 1); - } - } - if (ctx) { - ctx.restore(); - } - }; - CanvasPainter.prototype.getHoverLayer = function () { - return this.getLayer(HOVER_LAYER_ZLEVEL); - }; - CanvasPainter.prototype.paintOne = function (ctx, el) { - brushSingle(ctx, el); - }; - CanvasPainter.prototype._paintList = function (list, prevList, paintAll, redrawId) { - if (this._redrawId !== redrawId) { - return; - } - paintAll = paintAll || false; - this._updateLayerStatus(list); - var _a = this._doPaintList(list, prevList, paintAll), finished = _a.finished, needsRefreshHover = _a.needsRefreshHover; - if (this._needsManuallyCompositing) { - this._compositeManually(); - } - if (needsRefreshHover) { - this._paintHoverList(list); - } - if (!finished) { - var self_1 = this; - requestAnimationFrame$1(function () { - self_1._paintList(list, prevList, paintAll, redrawId); - }); - } - else { - this.eachLayer(function (layer) { - layer.afterBrush && layer.afterBrush(); - }); - } - }; - CanvasPainter.prototype._compositeManually = function () { - var ctx = this.getLayer(CANVAS_ZLEVEL).ctx; - var width = this._domRoot.width; - var height = this._domRoot.height; - ctx.clearRect(0, 0, width, height); - this.eachBuiltinLayer(function (layer) { - if (layer.virtual) { - ctx.drawImage(layer.dom, 0, 0, width, height); - } - }); - }; - CanvasPainter.prototype._doPaintList = function (list, prevList, paintAll) { - var _this = this; - var layerList = []; - var useDirtyRect = this._opts.useDirtyRect; - for (var zi = 0; zi < this._zlevelList.length; zi++) { - var zlevel = this._zlevelList[zi]; - var layer = this._layers[zlevel]; - if (layer.__builtin__ - && layer !== this._hoverlayer - && (layer.__dirty || paintAll)) { - layerList.push(layer); - } - } - var finished = true; - var needsRefreshHover = false; - var _loop_1 = function (k) { - var layer = layerList[k]; - var ctx = layer.ctx; - var repaintRects = useDirtyRect - && layer.createRepaintRects(list, prevList, this_1._width, this_1._height); - var start = paintAll ? layer.__startIndex : layer.__drawIndex; - var useTimer = !paintAll && layer.incremental && Date.now; - var startTime = useTimer && Date.now(); - var clearColor = layer.zlevel === this_1._zlevelList[0] - ? this_1._backgroundColor : null; - if (layer.__startIndex === layer.__endIndex) { - layer.clear(false, clearColor, repaintRects); - } - else if (start === layer.__startIndex) { - var firstEl = list[start]; - if (!firstEl.incremental || !firstEl.notClear || paintAll) { - layer.clear(false, clearColor, repaintRects); - } - } - if (start === -1) { - console.error('For some unknown reason. drawIndex is -1'); - start = layer.__startIndex; - } - var i; - var repaint = function (repaintRect) { - var scope = { - inHover: false, - allClipped: false, - prevEl: null, - viewWidth: _this._width, - viewHeight: _this._height - }; - for (i = start; i < layer.__endIndex; i++) { - var el = list[i]; - if (el.__inHover) { - needsRefreshHover = true; - } - _this._doPaintEl(el, layer, useDirtyRect, repaintRect, scope, i === layer.__endIndex - 1); - if (useTimer) { - var dTime = Date.now() - startTime; - if (dTime > 15) { - break; - } - } - } - if (scope.prevElClipPaths) { - ctx.restore(); - } - }; - if (repaintRects) { - if (repaintRects.length === 0) { - i = layer.__endIndex; - } - else { - var dpr = this_1.dpr; - for (var r = 0; r < repaintRects.length; ++r) { - var rect = repaintRects[r]; - ctx.save(); - ctx.beginPath(); - ctx.rect(rect.x * dpr, rect.y * dpr, rect.width * dpr, rect.height * dpr); - ctx.clip(); - repaint(rect); - ctx.restore(); - } - } - } - else { - ctx.save(); - repaint(); - ctx.restore(); - } - layer.__drawIndex = i; - if (layer.__drawIndex < layer.__endIndex) { - finished = false; - } - }; - var this_1 = this; - for (var k = 0; k < layerList.length; k++) { - _loop_1(k); - } - if (env.wxa) { - each(this._layers, function (layer) { - if (layer && layer.ctx && layer.ctx.draw) { - layer.ctx.draw(); - } - }); - } - return { - finished: finished, - needsRefreshHover: needsRefreshHover - }; - }; - CanvasPainter.prototype._doPaintEl = function (el, currentLayer, useDirtyRect, repaintRect, scope, isLast) { - var ctx = currentLayer.ctx; - if (useDirtyRect) { - var paintRect = el.getPaintRect(); - if (!repaintRect || paintRect && paintRect.intersect(repaintRect)) { - brush(ctx, el, scope, isLast); - el.setPrevPaintRect(paintRect); - } - } - else { - brush(ctx, el, scope, isLast); - } - }; - CanvasPainter.prototype.getLayer = function (zlevel, virtual) { - if (this._singleCanvas && !this._needsManuallyCompositing) { - zlevel = CANVAS_ZLEVEL; - } - var layer = this._layers[zlevel]; - if (!layer) { - layer = new Layer('zr_' + zlevel, this, this.dpr); - layer.zlevel = zlevel; - layer.__builtin__ = true; - if (this._layerConfig[zlevel]) { - merge(layer, this._layerConfig[zlevel], true); - } - else if (this._layerConfig[zlevel - EL_AFTER_INCREMENTAL_INC]) { - merge(layer, this._layerConfig[zlevel - EL_AFTER_INCREMENTAL_INC], true); - } - if (virtual) { - layer.virtual = virtual; - } - this.insertLayer(zlevel, layer); - layer.initContext(); - } - return layer; - }; - CanvasPainter.prototype.insertLayer = function (zlevel, layer) { - var layersMap = this._layers; - var zlevelList = this._zlevelList; - var len = zlevelList.length; - var domRoot = this._domRoot; - var prevLayer = null; - var i = -1; - if (layersMap[zlevel]) { - if ("development" !== 'production') { - logError('ZLevel ' + zlevel + ' has been used already'); - } - return; - } - if (!isLayerValid(layer)) { - if ("development" !== 'production') { - logError('Layer of zlevel ' + zlevel + ' is not valid'); - } - return; - } - if (len > 0 && zlevel > zlevelList[0]) { - for (i = 0; i < len - 1; i++) { - if (zlevelList[i] < zlevel - && zlevelList[i + 1] > zlevel) { - break; - } - } - prevLayer = layersMap[zlevelList[i]]; - } - zlevelList.splice(i + 1, 0, zlevel); - layersMap[zlevel] = layer; - if (!layer.virtual) { - if (prevLayer) { - var prevDom = prevLayer.dom; - if (prevDom.nextSibling) { - domRoot.insertBefore(layer.dom, prevDom.nextSibling); - } - else { - domRoot.appendChild(layer.dom); - } - } - else { - if (domRoot.firstChild) { - domRoot.insertBefore(layer.dom, domRoot.firstChild); - } - else { - domRoot.appendChild(layer.dom); - } - } - } - layer.__painter = this; - }; - CanvasPainter.prototype.eachLayer = function (cb, context) { - var zlevelList = this._zlevelList; - for (var i = 0; i < zlevelList.length; i++) { - var z = zlevelList[i]; - cb.call(context, this._layers[z], z); - } - }; - CanvasPainter.prototype.eachBuiltinLayer = function (cb, context) { - var zlevelList = this._zlevelList; - for (var i = 0; i < zlevelList.length; i++) { - var z = zlevelList[i]; - var layer = this._layers[z]; - if (layer.__builtin__) { - cb.call(context, layer, z); - } - } - }; - CanvasPainter.prototype.eachOtherLayer = function (cb, context) { - var zlevelList = this._zlevelList; - for (var i = 0; i < zlevelList.length; i++) { - var z = zlevelList[i]; - var layer = this._layers[z]; - if (!layer.__builtin__) { - cb.call(context, layer, z); - } - } - }; - CanvasPainter.prototype.getLayers = function () { - return this._layers; - }; - CanvasPainter.prototype._updateLayerStatus = function (list) { - this.eachBuiltinLayer(function (layer, z) { - layer.__dirty = layer.__used = false; - }); - function updatePrevLayer(idx) { - if (prevLayer) { - if (prevLayer.__endIndex !== idx) { - prevLayer.__dirty = true; - } - prevLayer.__endIndex = idx; - } - } - if (this._singleCanvas) { - for (var i_1 = 1; i_1 < list.length; i_1++) { - var el = list[i_1]; - if (el.zlevel !== list[i_1 - 1].zlevel || el.incremental) { - this._needsManuallyCompositing = true; - break; - } - } - } - var prevLayer = null; - var incrementalLayerCount = 0; - var prevZlevel; - var i; - for (i = 0; i < list.length; i++) { - var el = list[i]; - var zlevel = el.zlevel; - var layer = void 0; - if (prevZlevel !== zlevel) { - prevZlevel = zlevel; - incrementalLayerCount = 0; - } - if (el.incremental) { - layer = this.getLayer(zlevel + INCREMENTAL_INC, this._needsManuallyCompositing); - layer.incremental = true; - incrementalLayerCount = 1; - } - else { - layer = this.getLayer(zlevel + (incrementalLayerCount > 0 ? EL_AFTER_INCREMENTAL_INC : 0), this._needsManuallyCompositing); - } - if (!layer.__builtin__) { - logError('ZLevel ' + zlevel + ' has been used by unkown layer ' + layer.id); - } - if (layer !== prevLayer) { - layer.__used = true; - if (layer.__startIndex !== i) { - layer.__dirty = true; - } - layer.__startIndex = i; - if (!layer.incremental) { - layer.__drawIndex = i; - } - else { - layer.__drawIndex = -1; - } - updatePrevLayer(i); - prevLayer = layer; - } - if ((el.__dirty & REDRAW_BIT) && !el.__inHover) { - layer.__dirty = true; - if (layer.incremental && layer.__drawIndex < 0) { - layer.__drawIndex = i; - } - } - } - updatePrevLayer(i); - this.eachBuiltinLayer(function (layer, z) { - if (!layer.__used && layer.getElementCount() > 0) { - layer.__dirty = true; - layer.__startIndex = layer.__endIndex = layer.__drawIndex = 0; - } - if (layer.__dirty && layer.__drawIndex < 0) { - layer.__drawIndex = layer.__startIndex; - } - }); - }; - CanvasPainter.prototype.clear = function () { - this.eachBuiltinLayer(this._clearLayer); - return this; - }; - CanvasPainter.prototype._clearLayer = function (layer) { - layer.clear(); - }; - CanvasPainter.prototype.setBackgroundColor = function (backgroundColor) { - this._backgroundColor = backgroundColor; - each(this._layers, function (layer) { - layer.setUnpainted(); - }); - }; - CanvasPainter.prototype.configLayer = function (zlevel, config) { - if (config) { - var layerConfig = this._layerConfig; - if (!layerConfig[zlevel]) { - layerConfig[zlevel] = config; - } - else { - merge(layerConfig[zlevel], config, true); - } - for (var i = 0; i < this._zlevelList.length; i++) { - var _zlevel = this._zlevelList[i]; - if (_zlevel === zlevel || _zlevel === zlevel + EL_AFTER_INCREMENTAL_INC) { - var layer = this._layers[_zlevel]; - merge(layer, layerConfig[zlevel], true); - } - } - } - }; - CanvasPainter.prototype.delLayer = function (zlevel) { - var layers = this._layers; - var zlevelList = this._zlevelList; - var layer = layers[zlevel]; - if (!layer) { - return; - } - layer.dom.parentNode.removeChild(layer.dom); - delete layers[zlevel]; - zlevelList.splice(indexOf(zlevelList, zlevel), 1); - }; - CanvasPainter.prototype.resize = function (width, height) { - if (!this._domRoot.style) { - if (width == null || height == null) { - return; - } - this._width = width; - this._height = height; - this.getLayer(CANVAS_ZLEVEL).resize(width, height); - } - else { - var domRoot = this._domRoot; - domRoot.style.display = 'none'; - var opts = this._opts; - var root = this.root; - width != null && (opts.width = width); - height != null && (opts.height = height); - width = getSize(root, 0, opts); - height = getSize(root, 1, opts); - domRoot.style.display = ''; - if (this._width !== width || height !== this._height) { - domRoot.style.width = width + 'px'; - domRoot.style.height = height + 'px'; - for (var id in this._layers) { - if (this._layers.hasOwnProperty(id)) { - this._layers[id].resize(width, height); - } - } - this.refresh(true); - } - this._width = width; - this._height = height; - } - return this; - }; - CanvasPainter.prototype.clearLayer = function (zlevel) { - var layer = this._layers[zlevel]; - if (layer) { - layer.clear(); - } - }; - CanvasPainter.prototype.dispose = function () { - this.root.innerHTML = ''; - this.root = - this.storage = - this._domRoot = - this._layers = null; - }; - CanvasPainter.prototype.getRenderedCanvas = function (opts) { - opts = opts || {}; - if (this._singleCanvas && !this._compositeManually) { - return this._layers[CANVAS_ZLEVEL].dom; - } - var imageLayer = new Layer('image', this, opts.pixelRatio || this.dpr); - imageLayer.initContext(); - imageLayer.clear(false, opts.backgroundColor || this._backgroundColor); - var ctx = imageLayer.ctx; - if (opts.pixelRatio <= this.dpr) { - this.refresh(); - var width_1 = imageLayer.dom.width; - var height_1 = imageLayer.dom.height; - this.eachLayer(function (layer) { - if (layer.__builtin__) { - ctx.drawImage(layer.dom, 0, 0, width_1, height_1); - } - else if (layer.renderToCanvas) { - ctx.save(); - layer.renderToCanvas(ctx); - ctx.restore(); - } - }); - } - else { - var scope = { - inHover: false, - viewWidth: this._width, - viewHeight: this._height - }; - var displayList = this.storage.getDisplayList(true); - for (var i = 0, len = displayList.length; i < len; i++) { - var el = displayList[i]; - brush(ctx, el, scope, i === len - 1); - } - } - return imageLayer.dom; - }; - CanvasPainter.prototype.getWidth = function () { - return this._width; - }; - CanvasPainter.prototype.getHeight = function () { - return this._height; - }; - return CanvasPainter; - }()); - - function install$1(registers) { - registers.registerPainter('canvas', CanvasPainter); - } - - var LineSeriesModel = - /** @class */ - function (_super) { - __extends(LineSeriesModel, _super); - - function LineSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = LineSeriesModel.type; - _this.hasSymbolVisual = true; - return _this; - } - - LineSeriesModel.prototype.getInitialData = function (option) { - if ("development" !== 'production') { - var coordSys = option.coordinateSystem; - - if (coordSys !== 'polar' && coordSys !== 'cartesian2d') { - throw new Error('Line not support coordinateSystem besides cartesian and polar'); - } - } - - return createSeriesData(null, this, { - useEncodeDefaulter: true - }); - }; - - LineSeriesModel.prototype.getLegendIcon = function (opt) { - var group = new Group(); - var line = createSymbol('line', 0, opt.itemHeight / 2, opt.itemWidth, 0, opt.lineStyle.stroke, false); - group.add(line); - line.setStyle(opt.lineStyle); - var visualType = this.getData().getVisual('symbol'); - var visualRotate = this.getData().getVisual('symbolRotate'); - var symbolType = visualType === 'none' ? 'circle' : visualType; // Symbol size is 80% when there is a line - - var size = opt.itemHeight * 0.8; - var symbol = createSymbol(symbolType, (opt.itemWidth - size) / 2, (opt.itemHeight - size) / 2, size, size, opt.itemStyle.fill); - group.add(symbol); - symbol.setStyle(opt.itemStyle); - var symbolRotate = opt.iconRotate === 'inherit' ? visualRotate : opt.iconRotate || 0; - symbol.rotation = symbolRotate * Math.PI / 180; - symbol.setOrigin([opt.itemWidth / 2, opt.itemHeight / 2]); - - if (symbolType.indexOf('empty') > -1) { - symbol.style.stroke = symbol.style.fill; - symbol.style.fill = '#fff'; - symbol.style.lineWidth = 2; - } - - return group; - }; - - LineSeriesModel.type = 'series.line'; - LineSeriesModel.dependencies = ['grid', 'polar']; - LineSeriesModel.defaultOption = { - // zlevel: 0, - z: 3, - coordinateSystem: 'cartesian2d', - legendHoverLink: true, - clip: true, - label: { - position: 'top' - }, - // itemStyle: { - // }, - endLabel: { - show: false, - valueAnimation: true, - distance: 8 - }, - lineStyle: { - width: 2, - type: 'solid' - }, - emphasis: { - scale: true - }, - // areaStyle: { - // origin of areaStyle. Valid values: - // `'auto'/null/undefined`: from axisLine to data - // `'start'`: from min to data - // `'end'`: from data to max - // origin: 'auto' - // }, - // false, 'start', 'end', 'middle' - step: false, - // Disabled if step is true - smooth: false, - smoothMonotone: null, - symbol: 'emptyCircle', - symbolSize: 4, - symbolRotate: null, - showSymbol: true, - // `false`: follow the label interval strategy. - // `true`: show all symbols. - // `'auto'`: If possible, show all symbols, otherwise - // follow the label interval strategy. - showAllSymbol: 'auto', - // Whether to connect break point. - connectNulls: false, - // Sampling for large data. Can be: 'average', 'max', 'min', 'sum', 'lttb'. - sampling: 'none', - animationEasing: 'linear', - // Disable progressive - progressive: 0, - hoverLayerThreshold: Infinity, - universalTransition: { - divideShape: 'clone' - }, - triggerLineEvent: false - }; - return LineSeriesModel; - }(SeriesModel); - - /** - * @return label string. Not null/undefined - */ - - function getDefaultLabel(data, dataIndex) { - var labelDims = data.mapDimensionsAll('defaultedLabel'); - var len = labelDims.length; // Simple optimization (in lots of cases, label dims length is 1) - - if (len === 1) { - var rawVal = retrieveRawValue(data, dataIndex, labelDims[0]); - return rawVal != null ? rawVal + '' : null; - } else if (len) { - var vals = []; - - for (var i = 0; i < labelDims.length; i++) { - vals.push(retrieveRawValue(data, dataIndex, labelDims[i])); - } - - return vals.join(' '); - } - } - function getDefaultInterpolatedLabel(data, interpolatedValue) { - var labelDims = data.mapDimensionsAll('defaultedLabel'); - - if (!isArray(interpolatedValue)) { - return interpolatedValue + ''; - } - - var vals = []; - - for (var i = 0; i < labelDims.length; i++) { - var dimIndex = data.getDimensionIndex(labelDims[i]); - - if (dimIndex >= 0) { - vals.push(interpolatedValue[dimIndex]); - } - } - - return vals.join(' '); - } - - var Symbol = - /** @class */ - function (_super) { - __extends(Symbol, _super); - - function Symbol(data, idx, seriesScope, opts) { - var _this = _super.call(this) || this; - - _this.updateData(data, idx, seriesScope, opts); - - return _this; - } - - Symbol.prototype._createSymbol = function (symbolType, data, idx, symbolSize, keepAspect) { - // Remove paths created before - this.removeAll(); // let symbolPath = createSymbol( - // symbolType, -0.5, -0.5, 1, 1, color - // ); - // If width/height are set too small (e.g., set to 1) on ios10 - // and macOS Sierra, a circle stroke become a rect, no matter what - // the scale is set. So we set width/height as 2. See #4150. - - var symbolPath = createSymbol(symbolType, -1, -1, 2, 2, null, keepAspect); - symbolPath.attr({ - z2: 100, - culling: true, - scaleX: symbolSize[0] / 2, - scaleY: symbolSize[1] / 2 - }); // Rewrite drift method - - symbolPath.drift = driftSymbol; - this._symbolType = symbolType; - this.add(symbolPath); - }; - /** - * Stop animation - * @param {boolean} toLastFrame - */ - - - Symbol.prototype.stopSymbolAnimation = function (toLastFrame) { - this.childAt(0).stopAnimation(null, toLastFrame); - }; - - Symbol.prototype.getSymbolType = function () { - return this._symbolType; - }; - /** - * FIXME: - * Caution: This method breaks the encapsulation of this module, - * but it indeed brings convenience. So do not use the method - * unless you detailedly know all the implements of `Symbol`, - * especially animation. - * - * Get symbol path element. - */ - - - Symbol.prototype.getSymbolPath = function () { - return this.childAt(0); - }; - /** - * Highlight symbol - */ - - - Symbol.prototype.highlight = function () { - enterEmphasis(this.childAt(0)); - }; - /** - * Downplay symbol - */ - - - Symbol.prototype.downplay = function () { - leaveEmphasis(this.childAt(0)); - }; - /** - * @param {number} zlevel - * @param {number} z - */ - - - Symbol.prototype.setZ = function (zlevel, z) { - var symbolPath = this.childAt(0); - symbolPath.zlevel = zlevel; - symbolPath.z = z; - }; - - Symbol.prototype.setDraggable = function (draggable, hasCursorOption) { - var symbolPath = this.childAt(0); - symbolPath.draggable = draggable; - symbolPath.cursor = !hasCursorOption && draggable ? 'move' : symbolPath.cursor; - }; - /** - * Update symbol properties - */ - - - Symbol.prototype.updateData = function (data, idx, seriesScope, opts) { - this.silent = false; - var symbolType = data.getItemVisual(idx, 'symbol') || 'circle'; - var seriesModel = data.hostModel; - var symbolSize = Symbol.getSymbolSize(data, idx); - var isInit = symbolType !== this._symbolType; - var disableAnimation = opts && opts.disableAnimation; - - if (isInit) { - var keepAspect = data.getItemVisual(idx, 'symbolKeepAspect'); - - this._createSymbol(symbolType, data, idx, symbolSize, keepAspect); - } else { - var symbolPath = this.childAt(0); - symbolPath.silent = false; - var target = { - scaleX: symbolSize[0] / 2, - scaleY: symbolSize[1] / 2 - }; - disableAnimation ? symbolPath.attr(target) : updateProps(symbolPath, target, seriesModel, idx); - saveOldStyle(symbolPath); - } - - this._updateCommon(data, idx, symbolSize, seriesScope, opts); - - if (isInit) { - var symbolPath = this.childAt(0); - - if (!disableAnimation) { - var target = { - scaleX: this._sizeX, - scaleY: this._sizeY, - style: { - // Always fadeIn. Because it has fadeOut animation when symbol is removed.. - opacity: symbolPath.style.opacity - } - }; - symbolPath.scaleX = symbolPath.scaleY = 0; - symbolPath.style.opacity = 0; - initProps(symbolPath, target, seriesModel, idx); - } - } - - if (disableAnimation) { - // Must stop leave transition manually if don't call initProps or updateProps. - this.childAt(0).stopAnimation('leave'); - } - }; - - Symbol.prototype._updateCommon = function (data, idx, symbolSize, seriesScope, opts) { - var symbolPath = this.childAt(0); - var seriesModel = data.hostModel; - var emphasisItemStyle; - var blurItemStyle; - var selectItemStyle; - var focus; - var blurScope; - var emphasisDisabled; - var labelStatesModels; - var hoverScale; - var cursorStyle; - - if (seriesScope) { - emphasisItemStyle = seriesScope.emphasisItemStyle; - blurItemStyle = seriesScope.blurItemStyle; - selectItemStyle = seriesScope.selectItemStyle; - focus = seriesScope.focus; - blurScope = seriesScope.blurScope; - labelStatesModels = seriesScope.labelStatesModels; - hoverScale = seriesScope.hoverScale; - cursorStyle = seriesScope.cursorStyle; - emphasisDisabled = seriesScope.emphasisDisabled; - } - - if (!seriesScope || data.hasItemOption) { - var itemModel = seriesScope && seriesScope.itemModel ? seriesScope.itemModel : data.getItemModel(idx); - var emphasisModel = itemModel.getModel('emphasis'); - emphasisItemStyle = emphasisModel.getModel('itemStyle').getItemStyle(); - selectItemStyle = itemModel.getModel(['select', 'itemStyle']).getItemStyle(); - blurItemStyle = itemModel.getModel(['blur', 'itemStyle']).getItemStyle(); - focus = emphasisModel.get('focus'); - blurScope = emphasisModel.get('blurScope'); - emphasisDisabled = emphasisModel.get('disabled'); - labelStatesModels = getLabelStatesModels(itemModel); - hoverScale = emphasisModel.getShallow('scale'); - cursorStyle = itemModel.getShallow('cursor'); - } - - var symbolRotate = data.getItemVisual(idx, 'symbolRotate'); - symbolPath.attr('rotation', (symbolRotate || 0) * Math.PI / 180 || 0); - var symbolOffset = normalizeSymbolOffset(data.getItemVisual(idx, 'symbolOffset'), symbolSize); - - if (symbolOffset) { - symbolPath.x = symbolOffset[0]; - symbolPath.y = symbolOffset[1]; - } - - cursorStyle && symbolPath.attr('cursor', cursorStyle); - var symbolStyle = data.getItemVisual(idx, 'style'); - var visualColor = symbolStyle.fill; - - if (symbolPath instanceof ZRImage) { - var pathStyle = symbolPath.style; - symbolPath.useStyle(extend({ - // TODO other properties like x, y ? - image: pathStyle.image, - x: pathStyle.x, - y: pathStyle.y, - width: pathStyle.width, - height: pathStyle.height - }, symbolStyle)); - } else { - if (symbolPath.__isEmptyBrush) { - // fill and stroke will be swapped if it's empty. - // So we cloned a new style to avoid it affecting the original style in visual storage. - // TODO Better implementation. No empty logic! - symbolPath.useStyle(extend({}, symbolStyle)); - } else { - symbolPath.useStyle(symbolStyle); - } // Disable decal because symbol scale will been applied on the decal. - - - symbolPath.style.decal = null; - symbolPath.setColor(visualColor, opts && opts.symbolInnerColor); - symbolPath.style.strokeNoScale = true; - } - - var liftZ = data.getItemVisual(idx, 'liftZ'); - var z2Origin = this._z2; - - if (liftZ != null) { - if (z2Origin == null) { - this._z2 = symbolPath.z2; - symbolPath.z2 += liftZ; - } - } else if (z2Origin != null) { - symbolPath.z2 = z2Origin; - this._z2 = null; - } - - var useNameLabel = opts && opts.useNameLabel; - setLabelStyle(symbolPath, labelStatesModels, { - labelFetcher: seriesModel, - labelDataIndex: idx, - defaultText: getLabelDefaultText, - inheritColor: visualColor, - defaultOpacity: symbolStyle.opacity - }); // Do not execute util needed. - - function getLabelDefaultText(idx) { - return useNameLabel ? data.getName(idx) : getDefaultLabel(data, idx); - } - - this._sizeX = symbolSize[0] / 2; - this._sizeY = symbolSize[1] / 2; - var emphasisState = symbolPath.ensureState('emphasis'); - emphasisState.style = emphasisItemStyle; - symbolPath.ensureState('select').style = selectItemStyle; - symbolPath.ensureState('blur').style = blurItemStyle; // null / undefined / true means to use default strategy. - // 0 / false / negative number / NaN / Infinity means no scale. - - var scaleRatio = hoverScale == null || hoverScale === true ? Math.max(1.1, 3 / this._sizeY) // PENDING: restrict hoverScale > 1? It seems unreasonable to scale down - : isFinite(hoverScale) && hoverScale > 0 ? +hoverScale : 1; // always set scale to allow resetting - - emphasisState.scaleX = this._sizeX * scaleRatio; - emphasisState.scaleY = this._sizeY * scaleRatio; - this.setSymbolScale(1); - toggleHoverEmphasis(this, focus, blurScope, emphasisDisabled); - }; - - Symbol.prototype.setSymbolScale = function (scale) { - this.scaleX = this.scaleY = scale; - }; - - Symbol.prototype.fadeOut = function (cb, seriesModel, opt) { - var symbolPath = this.childAt(0); - var dataIndex = getECData(this).dataIndex; - var animationOpt = opt && opt.animation; // Avoid mistaken hover when fading out - - this.silent = symbolPath.silent = true; // Not show text when animating - - if (opt && opt.fadeLabel) { - var textContent = symbolPath.getTextContent(); - - if (textContent) { - removeElement(textContent, { - style: { - opacity: 0 - } - }, seriesModel, { - dataIndex: dataIndex, - removeOpt: animationOpt, - cb: function () { - symbolPath.removeTextContent(); - } - }); - } - } else { - symbolPath.removeTextContent(); - } - - removeElement(symbolPath, { - style: { - opacity: 0 - }, - scaleX: 0, - scaleY: 0 - }, seriesModel, { - dataIndex: dataIndex, - cb: cb, - removeOpt: animationOpt - }); - }; - - Symbol.getSymbolSize = function (data, idx) { - return normalizeSymbolSize(data.getItemVisual(idx, 'symbolSize')); - }; - - return Symbol; - }(Group); - - function driftSymbol(dx, dy) { - this.parent.drift(dx, dy); - } - - function symbolNeedsDraw(data, point, idx, opt) { - return point && !isNaN(point[0]) && !isNaN(point[1]) && !(opt.isIgnore && opt.isIgnore(idx)) // We do not set clipShape on group, because it will cut part of - // the symbol element shape. We use the same clip shape here as - // the line clip. - && !(opt.clipShape && !opt.clipShape.contain(point[0], point[1])) && data.getItemVisual(idx, 'symbol') !== 'none'; - } - - function normalizeUpdateOpt(opt) { - if (opt != null && !isObject(opt)) { - opt = { - isIgnore: opt - }; - } - - return opt || {}; - } - - function makeSeriesScope(data) { - var seriesModel = data.hostModel; - var emphasisModel = seriesModel.getModel('emphasis'); - return { - emphasisItemStyle: emphasisModel.getModel('itemStyle').getItemStyle(), - blurItemStyle: seriesModel.getModel(['blur', 'itemStyle']).getItemStyle(), - selectItemStyle: seriesModel.getModel(['select', 'itemStyle']).getItemStyle(), - focus: emphasisModel.get('focus'), - blurScope: emphasisModel.get('blurScope'), - emphasisDisabled: emphasisModel.get('disabled'), - hoverScale: emphasisModel.get('scale'), - labelStatesModels: getLabelStatesModels(seriesModel), - cursorStyle: seriesModel.get('cursor') - }; - } - - var SymbolDraw = - /** @class */ - function () { - function SymbolDraw(SymbolCtor) { - this.group = new Group(); - this._SymbolCtor = SymbolCtor || Symbol; - } - /** - * Update symbols draw by new data - */ - - - SymbolDraw.prototype.updateData = function (data, opt) { - // Remove progressive els. - this._progressiveEls = null; - opt = normalizeUpdateOpt(opt); - var group = this.group; - var seriesModel = data.hostModel; - var oldData = this._data; - var SymbolCtor = this._SymbolCtor; - var disableAnimation = opt.disableAnimation; - var seriesScope = makeSeriesScope(data); - var symbolUpdateOpt = { - disableAnimation: disableAnimation - }; - - var getSymbolPoint = opt.getSymbolPoint || function (idx) { - return data.getItemLayout(idx); - }; // There is no oldLineData only when first rendering or switching from - // stream mode to normal mode, where previous elements should be removed. - - - if (!oldData) { - group.removeAll(); - } - - data.diff(oldData).add(function (newIdx) { - var point = getSymbolPoint(newIdx); - - if (symbolNeedsDraw(data, point, newIdx, opt)) { - var symbolEl = new SymbolCtor(data, newIdx, seriesScope, symbolUpdateOpt); - symbolEl.setPosition(point); - data.setItemGraphicEl(newIdx, symbolEl); - group.add(symbolEl); - } - }).update(function (newIdx, oldIdx) { - var symbolEl = oldData.getItemGraphicEl(oldIdx); - var point = getSymbolPoint(newIdx); - - if (!symbolNeedsDraw(data, point, newIdx, opt)) { - group.remove(symbolEl); - return; - } - - var newSymbolType = data.getItemVisual(newIdx, 'symbol') || 'circle'; - var oldSymbolType = symbolEl && symbolEl.getSymbolType && symbolEl.getSymbolType(); - - if (!symbolEl // Create a new if symbol type changed. - || oldSymbolType && oldSymbolType !== newSymbolType) { - group.remove(symbolEl); - symbolEl = new SymbolCtor(data, newIdx, seriesScope, symbolUpdateOpt); - symbolEl.setPosition(point); - } else { - symbolEl.updateData(data, newIdx, seriesScope, symbolUpdateOpt); - var target = { - x: point[0], - y: point[1] - }; - disableAnimation ? symbolEl.attr(target) : updateProps(symbolEl, target, seriesModel); - } // Add back - - - group.add(symbolEl); - data.setItemGraphicEl(newIdx, symbolEl); - }).remove(function (oldIdx) { - var el = oldData.getItemGraphicEl(oldIdx); - el && el.fadeOut(function () { - group.remove(el); - }, seriesModel); - }).execute(); - this._getSymbolPoint = getSymbolPoint; - this._data = data; - }; - - SymbolDraw.prototype.updateLayout = function () { - var _this = this; - - var data = this._data; - - if (data) { - // Not use animation - data.eachItemGraphicEl(function (el, idx) { - var point = _this._getSymbolPoint(idx); - - el.setPosition(point); - el.markRedraw(); - }); - } - }; - - SymbolDraw.prototype.incrementalPrepareUpdate = function (data) { - this._seriesScope = makeSeriesScope(data); - this._data = null; - this.group.removeAll(); - }; - /** - * Update symbols draw by new data - */ - - SymbolDraw.prototype.incrementalUpdate = function (taskParams, data, opt) { - // Clear - this._progressiveEls = []; - opt = normalizeUpdateOpt(opt); - - function updateIncrementalAndHover(el) { - if (!el.isGroup) { - el.incremental = true; - el.ensureState('emphasis').hoverLayer = true; - } - } - - for (var idx = taskParams.start; idx < taskParams.end; idx++) { - var point = data.getItemLayout(idx); - - if (symbolNeedsDraw(data, point, idx, opt)) { - var el = new this._SymbolCtor(data, idx, this._seriesScope); - el.traverse(updateIncrementalAndHover); - el.setPosition(point); - this.group.add(el); - data.setItemGraphicEl(idx, el); - - this._progressiveEls.push(el); - } - } - }; - - SymbolDraw.prototype.eachRendered = function (cb) { - traverseElements(this._progressiveEls || this.group, cb); - }; - - SymbolDraw.prototype.remove = function (enableAnimation) { - var group = this.group; - var data = this._data; // Incremental model do not have this._data. - - if (data && enableAnimation) { - data.eachItemGraphicEl(function (el) { - el.fadeOut(function () { - group.remove(el); - }, data.hostModel); - }); - } else { - group.removeAll(); - } - }; - return SymbolDraw; - }(); - - function prepareDataCoordInfo(coordSys, data, valueOrigin) { - var baseAxis = coordSys.getBaseAxis(); - var valueAxis = coordSys.getOtherAxis(baseAxis); - var valueStart = getValueStart(valueAxis, valueOrigin); - var baseAxisDim = baseAxis.dim; - var valueAxisDim = valueAxis.dim; - var valueDim = data.mapDimension(valueAxisDim); - var baseDim = data.mapDimension(baseAxisDim); - var baseDataOffset = valueAxisDim === 'x' || valueAxisDim === 'radius' ? 1 : 0; - var dims = map(coordSys.dimensions, function (coordDim) { - return data.mapDimension(coordDim); - }); - var stacked = false; - var stackResultDim = data.getCalculationInfo('stackResultDimension'); - - if (isDimensionStacked(data, dims[0] - /* , dims[1] */ - )) { - // jshint ignore:line - stacked = true; - dims[0] = stackResultDim; - } - - if (isDimensionStacked(data, dims[1] - /* , dims[0] */ - )) { - // jshint ignore:line - stacked = true; - dims[1] = stackResultDim; - } - - return { - dataDimsForPoint: dims, - valueStart: valueStart, - valueAxisDim: valueAxisDim, - baseAxisDim: baseAxisDim, - stacked: !!stacked, - valueDim: valueDim, - baseDim: baseDim, - baseDataOffset: baseDataOffset, - stackedOverDimension: data.getCalculationInfo('stackedOverDimension') - }; - } - - function getValueStart(valueAxis, valueOrigin) { - var valueStart = 0; - var extent = valueAxis.scale.getExtent(); - - if (valueOrigin === 'start') { - valueStart = extent[0]; - } else if (valueOrigin === 'end') { - valueStart = extent[1]; - } // If origin is specified as a number, use it as - // valueStart directly - else if (isNumber(valueOrigin) && !isNaN(valueOrigin)) { - valueStart = valueOrigin; - } // auto - else { - // Both positive - if (extent[0] > 0) { - valueStart = extent[0]; - } // Both negative - else if (extent[1] < 0) { - valueStart = extent[1]; - } // If is one positive, and one negative, onZero shall be true - - } - - return valueStart; - } - - function getStackedOnPoint(dataCoordInfo, coordSys, data, idx) { - var value = NaN; - - if (dataCoordInfo.stacked) { - value = data.get(data.getCalculationInfo('stackedOverDimension'), idx); - } - - if (isNaN(value)) { - value = dataCoordInfo.valueStart; - } - - var baseDataOffset = dataCoordInfo.baseDataOffset; - var stackedData = []; - stackedData[baseDataOffset] = data.get(dataCoordInfo.baseDim, idx); - stackedData[1 - baseDataOffset] = value; - return coordSys.dataToPoint(stackedData); - } - - function diffData(oldData, newData) { - var diffResult = []; - newData.diff(oldData).add(function (idx) { - diffResult.push({ - cmd: '+', - idx: idx - }); - }).update(function (newIdx, oldIdx) { - diffResult.push({ - cmd: '=', - idx: oldIdx, - idx1: newIdx - }); - }).remove(function (idx) { - diffResult.push({ - cmd: '-', - idx: idx - }); - }).execute(); - return diffResult; - } - - function lineAnimationDiff(oldData, newData, oldStackedOnPoints, newStackedOnPoints, oldCoordSys, newCoordSys, oldValueOrigin, newValueOrigin) { - var diff = diffData(oldData, newData); // let newIdList = newData.mapArray(newData.getId); - // let oldIdList = oldData.mapArray(oldData.getId); - // convertToIntId(newIdList, oldIdList); - // // FIXME One data ? - // diff = arrayDiff(oldIdList, newIdList); - - var currPoints = []; - var nextPoints = []; // Points for stacking base line - - var currStackedPoints = []; - var nextStackedPoints = []; - var status = []; - var sortedIndices = []; - var rawIndices = []; - var newDataOldCoordInfo = prepareDataCoordInfo(oldCoordSys, newData, oldValueOrigin); // const oldDataNewCoordInfo = prepareDataCoordInfo(newCoordSys, oldData, newValueOrigin); - - var oldPoints = oldData.getLayout('points') || []; - var newPoints = newData.getLayout('points') || []; - - for (var i = 0; i < diff.length; i++) { - var diffItem = diff[i]; - var pointAdded = true; - var oldIdx2 = void 0; - var newIdx2 = void 0; // FIXME, animation is not so perfect when dataZoom window moves fast - // Which is in case remvoing or add more than one data in the tail or head - - switch (diffItem.cmd) { - case '=': - oldIdx2 = diffItem.idx * 2; - newIdx2 = diffItem.idx1 * 2; - var currentX = oldPoints[oldIdx2]; - var currentY = oldPoints[oldIdx2 + 1]; - var nextX = newPoints[newIdx2]; - var nextY = newPoints[newIdx2 + 1]; // If previous data is NaN, use next point directly - - if (isNaN(currentX) || isNaN(currentY)) { - currentX = nextX; - currentY = nextY; - } - - currPoints.push(currentX, currentY); - nextPoints.push(nextX, nextY); - currStackedPoints.push(oldStackedOnPoints[oldIdx2], oldStackedOnPoints[oldIdx2 + 1]); - nextStackedPoints.push(newStackedOnPoints[newIdx2], newStackedOnPoints[newIdx2 + 1]); - rawIndices.push(newData.getRawIndex(diffItem.idx1)); - break; - - case '+': - var newIdx = diffItem.idx; - var newDataDimsForPoint = newDataOldCoordInfo.dataDimsForPoint; - var oldPt = oldCoordSys.dataToPoint([newData.get(newDataDimsForPoint[0], newIdx), newData.get(newDataDimsForPoint[1], newIdx)]); - newIdx2 = newIdx * 2; - currPoints.push(oldPt[0], oldPt[1]); - nextPoints.push(newPoints[newIdx2], newPoints[newIdx2 + 1]); - var stackedOnPoint = getStackedOnPoint(newDataOldCoordInfo, oldCoordSys, newData, newIdx); - currStackedPoints.push(stackedOnPoint[0], stackedOnPoint[1]); - nextStackedPoints.push(newStackedOnPoints[newIdx2], newStackedOnPoints[newIdx2 + 1]); - rawIndices.push(newData.getRawIndex(newIdx)); - break; - - case '-': - pointAdded = false; - } // Original indices - - - if (pointAdded) { - status.push(diffItem); - sortedIndices.push(sortedIndices.length); - } - } // Diff result may be crossed if all items are changed - // Sort by data index - - - sortedIndices.sort(function (a, b) { - return rawIndices[a] - rawIndices[b]; - }); - var len = currPoints.length; - var sortedCurrPoints = createFloat32Array(len); - var sortedNextPoints = createFloat32Array(len); - var sortedCurrStackedPoints = createFloat32Array(len); - var sortedNextStackedPoints = createFloat32Array(len); - var sortedStatus = []; - - for (var i = 0; i < sortedIndices.length; i++) { - var idx = sortedIndices[i]; - var i2 = i * 2; - var idx2 = idx * 2; - sortedCurrPoints[i2] = currPoints[idx2]; - sortedCurrPoints[i2 + 1] = currPoints[idx2 + 1]; - sortedNextPoints[i2] = nextPoints[idx2]; - sortedNextPoints[i2 + 1] = nextPoints[idx2 + 1]; - sortedCurrStackedPoints[i2] = currStackedPoints[idx2]; - sortedCurrStackedPoints[i2 + 1] = currStackedPoints[idx2 + 1]; - sortedNextStackedPoints[i2] = nextStackedPoints[idx2]; - sortedNextStackedPoints[i2 + 1] = nextStackedPoints[idx2 + 1]; - sortedStatus[i] = status[idx]; - } - - return { - current: sortedCurrPoints, - next: sortedNextPoints, - stackedOnCurrent: sortedCurrStackedPoints, - stackedOnNext: sortedNextStackedPoints, - status: sortedStatus - }; - } - - var mathMin$5 = Math.min; - var mathMax$5 = Math.max; - - function isPointNull(x, y) { - return isNaN(x) || isNaN(y); - } - /** - * Draw smoothed line in non-monotone, in may cause undesired curve in extreme - * situations. This should be used when points are non-monotone neither in x or - * y dimension. - */ - - - function drawSegment(ctx, points, start, segLen, allLen, dir, smooth, smoothMonotone, connectNulls) { - var prevX; - var prevY; - var cpx0; - var cpy0; - var cpx1; - var cpy1; - var idx = start; - var k = 0; - - for (; k < segLen; k++) { - var x = points[idx * 2]; - var y = points[idx * 2 + 1]; - - if (idx >= allLen || idx < 0) { - break; - } - - if (isPointNull(x, y)) { - if (connectNulls) { - idx += dir; - continue; - } - - break; - } - - if (idx === start) { - ctx[dir > 0 ? 'moveTo' : 'lineTo'](x, y); - cpx0 = x; - cpy0 = y; - } else { - var dx = x - prevX; - var dy = y - prevY; // Ignore tiny segment. - - if (dx * dx + dy * dy < 0.5) { - idx += dir; - continue; - } - - if (smooth > 0) { - var nextIdx = idx + dir; - var nextX = points[nextIdx * 2]; - var nextY = points[nextIdx * 2 + 1]; // Ignore duplicate point - - while (nextX === x && nextY === y && k < segLen) { - k++; - nextIdx += dir; - idx += dir; - nextX = points[nextIdx * 2]; - nextY = points[nextIdx * 2 + 1]; - x = points[idx * 2]; - y = points[idx * 2 + 1]; - dx = x - prevX; - dy = y - prevY; - } - - var tmpK = k + 1; - - if (connectNulls) { - // Find next point not null - while (isPointNull(nextX, nextY) && tmpK < segLen) { - tmpK++; - nextIdx += dir; - nextX = points[nextIdx * 2]; - nextY = points[nextIdx * 2 + 1]; - } - } - - var ratioNextSeg = 0.5; - var vx = 0; - var vy = 0; - var nextCpx0 = void 0; - var nextCpy0 = void 0; // Is last point - - if (tmpK >= segLen || isPointNull(nextX, nextY)) { - cpx1 = x; - cpy1 = y; - } else { - vx = nextX - prevX; - vy = nextY - prevY; - var dx0 = x - prevX; - var dx1 = nextX - x; - var dy0 = y - prevY; - var dy1 = nextY - y; - var lenPrevSeg = void 0; - var lenNextSeg = void 0; - - if (smoothMonotone === 'x') { - lenPrevSeg = Math.abs(dx0); - lenNextSeg = Math.abs(dx1); - var dir_1 = vx > 0 ? 1 : -1; - cpx1 = x - dir_1 * lenPrevSeg * smooth; - cpy1 = y; - nextCpx0 = x + dir_1 * lenNextSeg * smooth; - nextCpy0 = y; - } else if (smoothMonotone === 'y') { - lenPrevSeg = Math.abs(dy0); - lenNextSeg = Math.abs(dy1); - var dir_2 = vy > 0 ? 1 : -1; - cpx1 = x; - cpy1 = y - dir_2 * lenPrevSeg * smooth; - nextCpx0 = x; - nextCpy0 = y + dir_2 * lenNextSeg * smooth; - } else { - lenPrevSeg = Math.sqrt(dx0 * dx0 + dy0 * dy0); - lenNextSeg = Math.sqrt(dx1 * dx1 + dy1 * dy1); // Use ratio of seg length - - ratioNextSeg = lenNextSeg / (lenNextSeg + lenPrevSeg); - cpx1 = x - vx * smooth * (1 - ratioNextSeg); - cpy1 = y - vy * smooth * (1 - ratioNextSeg); // cp0 of next segment - - nextCpx0 = x + vx * smooth * ratioNextSeg; - nextCpy0 = y + vy * smooth * ratioNextSeg; // Smooth constraint between point and next point. - // Avoid exceeding extreme after smoothing. - - nextCpx0 = mathMin$5(nextCpx0, mathMax$5(nextX, x)); - nextCpy0 = mathMin$5(nextCpy0, mathMax$5(nextY, y)); - nextCpx0 = mathMax$5(nextCpx0, mathMin$5(nextX, x)); - nextCpy0 = mathMax$5(nextCpy0, mathMin$5(nextY, y)); // Reclaculate cp1 based on the adjusted cp0 of next seg. - - vx = nextCpx0 - x; - vy = nextCpy0 - y; - cpx1 = x - vx * lenPrevSeg / lenNextSeg; - cpy1 = y - vy * lenPrevSeg / lenNextSeg; // Smooth constraint between point and prev point. - // Avoid exceeding extreme after smoothing. - - cpx1 = mathMin$5(cpx1, mathMax$5(prevX, x)); - cpy1 = mathMin$5(cpy1, mathMax$5(prevY, y)); - cpx1 = mathMax$5(cpx1, mathMin$5(prevX, x)); - cpy1 = mathMax$5(cpy1, mathMin$5(prevY, y)); // Adjust next cp0 again. - - vx = x - cpx1; - vy = y - cpy1; - nextCpx0 = x + vx * lenNextSeg / lenPrevSeg; - nextCpy0 = y + vy * lenNextSeg / lenPrevSeg; - } - } - - ctx.bezierCurveTo(cpx0, cpy0, cpx1, cpy1, x, y); - cpx0 = nextCpx0; - cpy0 = nextCpy0; - } else { - ctx.lineTo(x, y); - } - } - - prevX = x; - prevY = y; - idx += dir; - } - - return k; - } - - var ECPolylineShape = - /** @class */ - function () { - function ECPolylineShape() { - this.smooth = 0; - this.smoothConstraint = true; - } - - return ECPolylineShape; - }(); - - var ECPolyline = - /** @class */ - function (_super) { - __extends(ECPolyline, _super); - - function ECPolyline(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'ec-polyline'; - return _this; - } - - ECPolyline.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - - ECPolyline.prototype.getDefaultShape = function () { - return new ECPolylineShape(); - }; - - ECPolyline.prototype.buildPath = function (ctx, shape) { - var points = shape.points; - var i = 0; - var len = points.length / 2; // const result = getBoundingBox(points, shape.smoothConstraint); - - if (shape.connectNulls) { - // Must remove first and last null values avoid draw error in polygon - for (; len > 0; len--) { - if (!isPointNull(points[len * 2 - 2], points[len * 2 - 1])) { - break; - } - } - - for (; i < len; i++) { - if (!isPointNull(points[i * 2], points[i * 2 + 1])) { - break; - } - } - } - - while (i < len) { - i += drawSegment(ctx, points, i, len, len, 1, shape.smooth, shape.smoothMonotone, shape.connectNulls) + 1; - } - }; - - ECPolyline.prototype.getPointOn = function (xOrY, dim) { - if (!this.path) { - this.createPathProxy(); - this.buildPath(this.path, this.shape); - } - - var path = this.path; - var data = path.data; - var CMD = PathProxy.CMD; - var x0; - var y0; - var isDimX = dim === 'x'; - var roots = []; - - for (var i = 0; i < data.length;) { - var cmd = data[i++]; - var x = void 0; - var y = void 0; - var x2 = void 0; - var y2 = void 0; - var x3 = void 0; - var y3 = void 0; - var t = void 0; - - switch (cmd) { - case CMD.M: - x0 = data[i++]; - y0 = data[i++]; - break; - - case CMD.L: - x = data[i++]; - y = data[i++]; - t = isDimX ? (xOrY - x0) / (x - x0) : (xOrY - y0) / (y - y0); - - if (t <= 1 && t >= 0) { - var val = isDimX ? (y - y0) * t + y0 : (x - x0) * t + x0; - return isDimX ? [xOrY, val] : [val, xOrY]; - } - - x0 = x; - y0 = y; - break; - - case CMD.C: - x = data[i++]; - y = data[i++]; - x2 = data[i++]; - y2 = data[i++]; - x3 = data[i++]; - y3 = data[i++]; - var nRoot = isDimX ? cubicRootAt(x0, x, x2, x3, xOrY, roots) : cubicRootAt(y0, y, y2, y3, xOrY, roots); - - if (nRoot > 0) { - for (var i_1 = 0; i_1 < nRoot; i_1++) { - var t_1 = roots[i_1]; - - if (t_1 <= 1 && t_1 >= 0) { - var val = isDimX ? cubicAt(y0, y, y2, y3, t_1) : cubicAt(x0, x, x2, x3, t_1); - return isDimX ? [xOrY, val] : [val, xOrY]; - } - } - } - - x0 = x3; - y0 = y3; - break; - } - } - }; - - return ECPolyline; - }(Path); - - var ECPolygonShape = - /** @class */ - function (_super) { - __extends(ECPolygonShape, _super); - - function ECPolygonShape() { - return _super !== null && _super.apply(this, arguments) || this; - } - - return ECPolygonShape; - }(ECPolylineShape); - - var ECPolygon = - /** @class */ - function (_super) { - __extends(ECPolygon, _super); - - function ECPolygon(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'ec-polygon'; - return _this; - } - - ECPolygon.prototype.getDefaultShape = function () { - return new ECPolygonShape(); - }; - - ECPolygon.prototype.buildPath = function (ctx, shape) { - var points = shape.points; - var stackedOnPoints = shape.stackedOnPoints; - var i = 0; - var len = points.length / 2; - var smoothMonotone = shape.smoothMonotone; - - if (shape.connectNulls) { - // Must remove first and last null values avoid draw error in polygon - for (; len > 0; len--) { - if (!isPointNull(points[len * 2 - 2], points[len * 2 - 1])) { - break; - } - } - - for (; i < len; i++) { - if (!isPointNull(points[i * 2], points[i * 2 + 1])) { - break; - } - } - } - - while (i < len) { - var k = drawSegment(ctx, points, i, len, len, 1, shape.smooth, smoothMonotone, shape.connectNulls); - drawSegment(ctx, stackedOnPoints, i + k - 1, k, len, -1, shape.stackedOnSmooth, smoothMonotone, shape.connectNulls); - i += k + 1; - ctx.closePath(); - } - }; - - return ECPolygon; - }(Path); - - function createGridClipPath(cartesian, hasAnimation, seriesModel, done, during) { - var rect = cartesian.getArea(); - var x = rect.x; - var y = rect.y; - var width = rect.width; - var height = rect.height; - var lineWidth = seriesModel.get(['lineStyle', 'width']) || 2; // Expand the clip path a bit to avoid the border is clipped and looks thinner - - x -= lineWidth / 2; - y -= lineWidth / 2; - width += lineWidth; - height += lineWidth; // fix: https://github.com/apache/incubator-echarts/issues/11369 - - x = Math.floor(x); - width = Math.round(width); - var clipPath = new Rect({ - shape: { - x: x, - y: y, - width: width, - height: height - } - }); - - if (hasAnimation) { - var baseAxis = cartesian.getBaseAxis(); - var isHorizontal = baseAxis.isHorizontal(); - var isAxisInversed = baseAxis.inverse; - - if (isHorizontal) { - if (isAxisInversed) { - clipPath.shape.x += width; - } - - clipPath.shape.width = 0; - } else { - if (!isAxisInversed) { - clipPath.shape.y += height; - } - - clipPath.shape.height = 0; - } - - var duringCb = isFunction(during) ? function (percent) { - during(percent, clipPath); - } : null; - initProps(clipPath, { - shape: { - width: width, - height: height, - x: x, - y: y - } - }, seriesModel, null, done, duringCb); - } - - return clipPath; - } - - function createPolarClipPath(polar, hasAnimation, seriesModel) { - var sectorArea = polar.getArea(); // Avoid float number rounding error for symbol on the edge of axis extent. - - var r0 = round(sectorArea.r0, 1); - var r = round(sectorArea.r, 1); - var clipPath = new Sector({ - shape: { - cx: round(polar.cx, 1), - cy: round(polar.cy, 1), - r0: r0, - r: r, - startAngle: sectorArea.startAngle, - endAngle: sectorArea.endAngle, - clockwise: sectorArea.clockwise - } - }); - - if (hasAnimation) { - var isRadial = polar.getBaseAxis().dim === 'angle'; - - if (isRadial) { - clipPath.shape.endAngle = sectorArea.startAngle; - } else { - clipPath.shape.r = r0; - } - - initProps(clipPath, { - shape: { - endAngle: sectorArea.endAngle, - r: r - } - }, seriesModel); - } - - return clipPath; - } - - function createClipPath(coordSys, hasAnimation, seriesModel, done, during) { - if (!coordSys) { - return null; - } else if (coordSys.type === 'polar') { - return createPolarClipPath(coordSys, hasAnimation, seriesModel); - } else if (coordSys.type === 'cartesian2d') { - return createGridClipPath(coordSys, hasAnimation, seriesModel, done, during); - } - - return null; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function isCoordinateSystemType(coordSys, type) { - return coordSys.type === type; - } - - function isPointsSame(points1, points2) { - if (points1.length !== points2.length) { - return; - } - - for (var i = 0; i < points1.length; i++) { - if (points1[i] !== points2[i]) { - return; - } - } - - return true; - } - - function bboxFromPoints(points) { - var minX = Infinity; - var minY = Infinity; - var maxX = -Infinity; - var maxY = -Infinity; - - for (var i = 0; i < points.length;) { - var x = points[i++]; - var y = points[i++]; - - if (!isNaN(x)) { - minX = Math.min(x, minX); - maxX = Math.max(x, maxX); - } - - if (!isNaN(y)) { - minY = Math.min(y, minY); - maxY = Math.max(y, maxY); - } - } - - return [[minX, minY], [maxX, maxY]]; - } - - function getBoundingDiff(points1, points2) { - var _a = bboxFromPoints(points1), - min1 = _a[0], - max1 = _a[1]; - - var _b = bboxFromPoints(points2), - min2 = _b[0], - max2 = _b[1]; // Get a max value from each corner of two boundings. - - - return Math.max(Math.abs(min1[0] - min2[0]), Math.abs(min1[1] - min2[1]), Math.abs(max1[0] - max2[0]), Math.abs(max1[1] - max2[1])); - } - - function getSmooth(smooth) { - return isNumber(smooth) ? smooth : smooth ? 0.5 : 0; - } - - function getStackedOnPoints(coordSys, data, dataCoordInfo) { - if (!dataCoordInfo.valueDim) { - return []; - } - - var len = data.count(); - var points = createFloat32Array(len * 2); - - for (var idx = 0; idx < len; idx++) { - var pt = getStackedOnPoint(dataCoordInfo, coordSys, data, idx); - points[idx * 2] = pt[0]; - points[idx * 2 + 1] = pt[1]; - } - - return points; - } - - function turnPointsIntoStep(points, coordSys, stepTurnAt, connectNulls) { - var baseAxis = coordSys.getBaseAxis(); - var baseIndex = baseAxis.dim === 'x' || baseAxis.dim === 'radius' ? 0 : 1; - var stepPoints = []; - var i = 0; - var stepPt = []; - var pt = []; - var nextPt = []; - var filteredPoints = []; - - if (connectNulls) { - for (i = 0; i < points.length; i += 2) { - if (!isNaN(points[i]) && !isNaN(points[i + 1])) { - filteredPoints.push(points[i], points[i + 1]); - } - } - - points = filteredPoints; - } - - for (i = 0; i < points.length - 2; i += 2) { - nextPt[0] = points[i + 2]; - nextPt[1] = points[i + 3]; - pt[0] = points[i]; - pt[1] = points[i + 1]; - stepPoints.push(pt[0], pt[1]); - - switch (stepTurnAt) { - case 'end': - stepPt[baseIndex] = nextPt[baseIndex]; - stepPt[1 - baseIndex] = pt[1 - baseIndex]; - stepPoints.push(stepPt[0], stepPt[1]); - break; - - case 'middle': - var middle = (pt[baseIndex] + nextPt[baseIndex]) / 2; - var stepPt2 = []; - stepPt[baseIndex] = stepPt2[baseIndex] = middle; - stepPt[1 - baseIndex] = pt[1 - baseIndex]; - stepPt2[1 - baseIndex] = nextPt[1 - baseIndex]; - stepPoints.push(stepPt[0], stepPt[1]); - stepPoints.push(stepPt2[0], stepPt2[1]); - break; - - default: - // default is start - stepPt[baseIndex] = pt[baseIndex]; - stepPt[1 - baseIndex] = nextPt[1 - baseIndex]; - stepPoints.push(stepPt[0], stepPt[1]); - } - } // Last points - - - stepPoints.push(points[i++], points[i++]); - return stepPoints; - } - /** - * Clip color stops to edge. Avoid creating too large gradients. - * Which may lead to blurry when GPU acceleration is enabled. See #15680 - * - * The stops has been sorted from small to large. - */ - - - function clipColorStops(colorStops, maxSize) { - var newColorStops = []; - var len = colorStops.length; // coord will always < 0 in prevOutOfRangeColorStop. - - var prevOutOfRangeColorStop; - var prevInRangeColorStop; - - function lerpStop(stop0, stop1, clippedCoord) { - var coord0 = stop0.coord; - var p = (clippedCoord - coord0) / (stop1.coord - coord0); - var color = lerp$1(p, [stop0.color, stop1.color]); - return { - coord: clippedCoord, - color: color - }; - } - - for (var i = 0; i < len; i++) { - var stop_1 = colorStops[i]; - var coord = stop_1.coord; - - if (coord < 0) { - prevOutOfRangeColorStop = stop_1; - } else if (coord > maxSize) { - if (prevInRangeColorStop) { - newColorStops.push(lerpStop(prevInRangeColorStop, stop_1, maxSize)); - } else if (prevOutOfRangeColorStop) { - // If there are two stops and coord range is between these two stops - newColorStops.push(lerpStop(prevOutOfRangeColorStop, stop_1, 0), lerpStop(prevOutOfRangeColorStop, stop_1, maxSize)); - } // All following stop will be out of range. So just ignore them. - - - break; - } else { - if (prevOutOfRangeColorStop) { - newColorStops.push(lerpStop(prevOutOfRangeColorStop, stop_1, 0)); // Reset - - prevOutOfRangeColorStop = null; - } - - newColorStops.push(stop_1); - prevInRangeColorStop = stop_1; - } - } - - return newColorStops; - } - - function getVisualGradient(data, coordSys, api) { - var visualMetaList = data.getVisual('visualMeta'); - - if (!visualMetaList || !visualMetaList.length || !data.count()) { - // When data.count() is 0, gradient range can not be calculated. - return; - } - - if (coordSys.type !== 'cartesian2d') { - if ("development" !== 'production') { - console.warn('Visual map on line style is only supported on cartesian2d.'); - } - - return; - } - - var coordDim; - var visualMeta; - - for (var i = visualMetaList.length - 1; i >= 0; i--) { - var dimInfo = data.getDimensionInfo(visualMetaList[i].dimension); - coordDim = dimInfo && dimInfo.coordDim; // Can only be x or y - - if (coordDim === 'x' || coordDim === 'y') { - visualMeta = visualMetaList[i]; - break; - } - } - - if (!visualMeta) { - if ("development" !== 'production') { - console.warn('Visual map on line style only support x or y dimension.'); - } - - return; - } // If the area to be rendered is bigger than area defined by LinearGradient, - // the canvas spec prescribes that the color of the first stop and the last - // stop should be used. But if two stops are added at offset 0, in effect - // browsers use the color of the second stop to render area outside - // LinearGradient. So we can only infinitesimally extend area defined in - // LinearGradient to render `outerColors`. - - - var axis = coordSys.getAxis(coordDim); // dataToCoord mapping may not be linear, but must be monotonic. - - var colorStops = map(visualMeta.stops, function (stop) { - // offset will be calculated later. - return { - coord: axis.toGlobalCoord(axis.dataToCoord(stop.value)), - color: stop.color - }; - }); - var stopLen = colorStops.length; - var outerColors = visualMeta.outerColors.slice(); - - if (stopLen && colorStops[0].coord > colorStops[stopLen - 1].coord) { - colorStops.reverse(); - outerColors.reverse(); - } - - var colorStopsInRange = clipColorStops(colorStops, coordDim === 'x' ? api.getWidth() : api.getHeight()); - var inRangeStopLen = colorStopsInRange.length; - - if (!inRangeStopLen && stopLen) { - // All stops are out of range. All will be the same color. - return colorStops[0].coord < 0 ? outerColors[1] ? outerColors[1] : colorStops[stopLen - 1].color : outerColors[0] ? outerColors[0] : colorStops[0].color; - } - - var tinyExtent = 10; // Arbitrary value: 10px - - var minCoord = colorStopsInRange[0].coord - tinyExtent; - var maxCoord = colorStopsInRange[inRangeStopLen - 1].coord + tinyExtent; - var coordSpan = maxCoord - minCoord; - - if (coordSpan < 1e-3) { - return 'transparent'; - } - - each(colorStopsInRange, function (stop) { - stop.offset = (stop.coord - minCoord) / coordSpan; - }); - colorStopsInRange.push({ - // NOTE: inRangeStopLen may still be 0 if stoplen is zero. - offset: inRangeStopLen ? colorStopsInRange[inRangeStopLen - 1].offset : 0.5, - color: outerColors[1] || 'transparent' - }); - colorStopsInRange.unshift({ - offset: inRangeStopLen ? colorStopsInRange[0].offset : 0.5, - color: outerColors[0] || 'transparent' - }); - var gradient = new LinearGradient(0, 0, 0, 0, colorStopsInRange, true); - gradient[coordDim] = minCoord; - gradient[coordDim + '2'] = maxCoord; - return gradient; - } - - function getIsIgnoreFunc(seriesModel, data, coordSys) { - var showAllSymbol = seriesModel.get('showAllSymbol'); - var isAuto = showAllSymbol === 'auto'; - - if (showAllSymbol && !isAuto) { - return; - } - - var categoryAxis = coordSys.getAxesByScale('ordinal')[0]; - - if (!categoryAxis) { - return; - } // Note that category label interval strategy might bring some weird effect - // in some scenario: users may wonder why some of the symbols are not - // displayed. So we show all symbols as possible as we can. - - - if (isAuto // Simplify the logic, do not determine label overlap here. - && canShowAllSymbolForCategory(categoryAxis, data)) { - return; - } // Otherwise follow the label interval strategy on category axis. - - - var categoryDataDim = data.mapDimension(categoryAxis.dim); - var labelMap = {}; - each(categoryAxis.getViewLabels(), function (labelItem) { - var ordinalNumber = categoryAxis.scale.getRawOrdinalNumber(labelItem.tickValue); - labelMap[ordinalNumber] = 1; - }); - return function (dataIndex) { - return !labelMap.hasOwnProperty(data.get(categoryDataDim, dataIndex)); - }; - } - - function canShowAllSymbolForCategory(categoryAxis, data) { - // In most cases, line is monotonous on category axis, and the label size - // is close with each other. So we check the symbol size and some of the - // label size alone with the category axis to estimate whether all symbol - // can be shown without overlap. - var axisExtent = categoryAxis.getExtent(); - var availSize = Math.abs(axisExtent[1] - axisExtent[0]) / categoryAxis.scale.count(); - isNaN(availSize) && (availSize = 0); // 0/0 is NaN. - // Sampling some points, max 5. - - var dataLen = data.count(); - var step = Math.max(1, Math.round(dataLen / 5)); - - for (var dataIndex = 0; dataIndex < dataLen; dataIndex += step) { - if (Symbol.getSymbolSize(data, dataIndex // Only for cartesian, where `isHorizontal` exists. - )[categoryAxis.isHorizontal() ? 1 : 0] // Empirical number - * 1.5 > availSize) { - return false; - } - } - - return true; - } - - function isPointNull$1(x, y) { - return isNaN(x) || isNaN(y); - } - - function getLastIndexNotNull(points) { - var len = points.length / 2; - - for (; len > 0; len--) { - if (!isPointNull$1(points[len * 2 - 2], points[len * 2 - 1])) { - break; - } - } - - return len - 1; - } - - function getPointAtIndex(points, idx) { - return [points[idx * 2], points[idx * 2 + 1]]; - } - - function getIndexRange(points, xOrY, dim) { - var len = points.length / 2; - var dimIdx = dim === 'x' ? 0 : 1; - var a; - var b; - var prevIndex = 0; - var nextIndex = -1; - - for (var i = 0; i < len; i++) { - b = points[i * 2 + dimIdx]; - - if (isNaN(b) || isNaN(points[i * 2 + 1 - dimIdx])) { - continue; - } - - if (i === 0) { - a = b; - continue; - } - - if (a <= xOrY && b >= xOrY || a >= xOrY && b <= xOrY) { - nextIndex = i; - break; - } - - prevIndex = i; - a = b; - } - - return { - range: [prevIndex, nextIndex], - t: (xOrY - a) / (b - a) - }; - } - - function anyStateShowEndLabel(seriesModel) { - if (seriesModel.get(['endLabel', 'show'])) { - return true; - } - - for (var i = 0; i < SPECIAL_STATES.length; i++) { - if (seriesModel.get([SPECIAL_STATES[i], 'endLabel', 'show'])) { - return true; - } - } - - return false; - } - - function createLineClipPath(lineView, coordSys, hasAnimation, seriesModel) { - if (isCoordinateSystemType(coordSys, 'cartesian2d')) { - var endLabelModel_1 = seriesModel.getModel('endLabel'); - var valueAnimation_1 = endLabelModel_1.get('valueAnimation'); - var data_1 = seriesModel.getData(); - var labelAnimationRecord_1 = { - lastFrameIndex: 0 - }; - var during = anyStateShowEndLabel(seriesModel) ? function (percent, clipRect) { - lineView._endLabelOnDuring(percent, clipRect, data_1, labelAnimationRecord_1, valueAnimation_1, endLabelModel_1, coordSys); - } : null; - var isHorizontal = coordSys.getBaseAxis().isHorizontal(); - var clipPath = createGridClipPath(coordSys, hasAnimation, seriesModel, function () { - var endLabel = lineView._endLabel; - - if (endLabel && hasAnimation) { - if (labelAnimationRecord_1.originalX != null) { - endLabel.attr({ - x: labelAnimationRecord_1.originalX, - y: labelAnimationRecord_1.originalY - }); - } - } - }, during); // Expand clip shape to avoid clipping when line value exceeds axis - - if (!seriesModel.get('clip', true)) { - var rectShape = clipPath.shape; - var expandSize = Math.max(rectShape.width, rectShape.height); - - if (isHorizontal) { - rectShape.y -= expandSize; - rectShape.height += expandSize * 2; - } else { - rectShape.x -= expandSize; - rectShape.width += expandSize * 2; - } - } // Set to the final frame. To make sure label layout is right. - - - if (during) { - during(1, clipPath); - } - - return clipPath; - } else { - if ("development" !== 'production') { - if (seriesModel.get(['endLabel', 'show'])) { - console.warn('endLabel is not supported for lines in polar systems.'); - } - } - - return createPolarClipPath(coordSys, hasAnimation, seriesModel); - } - } - - function getEndLabelStateSpecified(endLabelModel, coordSys) { - var baseAxis = coordSys.getBaseAxis(); - var isHorizontal = baseAxis.isHorizontal(); - var isBaseInversed = baseAxis.inverse; - var align = isHorizontal ? isBaseInversed ? 'right' : 'left' : 'center'; - var verticalAlign = isHorizontal ? 'middle' : isBaseInversed ? 'top' : 'bottom'; - return { - normal: { - align: endLabelModel.get('align') || align, - verticalAlign: endLabelModel.get('verticalAlign') || verticalAlign - } - }; - } - - var LineView = - /** @class */ - function (_super) { - __extends(LineView, _super); - - function LineView() { - return _super !== null && _super.apply(this, arguments) || this; - } - - LineView.prototype.init = function () { - var lineGroup = new Group(); - var symbolDraw = new SymbolDraw(); - this.group.add(symbolDraw.group); - this._symbolDraw = symbolDraw; - this._lineGroup = lineGroup; - }; - - LineView.prototype.render = function (seriesModel, ecModel, api) { - var _this = this; - - var coordSys = seriesModel.coordinateSystem; - var group = this.group; - var data = seriesModel.getData(); - var lineStyleModel = seriesModel.getModel('lineStyle'); - var areaStyleModel = seriesModel.getModel('areaStyle'); - var points = data.getLayout('points') || []; - var isCoordSysPolar = coordSys.type === 'polar'; - var prevCoordSys = this._coordSys; - var symbolDraw = this._symbolDraw; - var polyline = this._polyline; - var polygon = this._polygon; - var lineGroup = this._lineGroup; - var hasAnimation = !ecModel.ssr && seriesModel.isAnimationEnabled(); - var isAreaChart = !areaStyleModel.isEmpty(); - var valueOrigin = areaStyleModel.get('origin'); - var dataCoordInfo = prepareDataCoordInfo(coordSys, data, valueOrigin); - var stackedOnPoints = isAreaChart && getStackedOnPoints(coordSys, data, dataCoordInfo); - var showSymbol = seriesModel.get('showSymbol'); - var connectNulls = seriesModel.get('connectNulls'); - var isIgnoreFunc = showSymbol && !isCoordSysPolar && getIsIgnoreFunc(seriesModel, data, coordSys); // Remove temporary symbols - - var oldData = this._data; - oldData && oldData.eachItemGraphicEl(function (el, idx) { - if (el.__temp) { - group.remove(el); - oldData.setItemGraphicEl(idx, null); - } - }); // Remove previous created symbols if showSymbol changed to false - - if (!showSymbol) { - symbolDraw.remove(); - } - - group.add(lineGroup); // FIXME step not support polar - - var step = !isCoordSysPolar ? seriesModel.get('step') : false; - var clipShapeForSymbol; - - if (coordSys && coordSys.getArea && seriesModel.get('clip', true)) { - clipShapeForSymbol = coordSys.getArea(); // Avoid float number rounding error for symbol on the edge of axis extent. - // See #7913 and `test/dataZoom-clip.html`. - - if (clipShapeForSymbol.width != null) { - clipShapeForSymbol.x -= 0.1; - clipShapeForSymbol.y -= 0.1; - clipShapeForSymbol.width += 0.2; - clipShapeForSymbol.height += 0.2; - } else if (clipShapeForSymbol.r0) { - clipShapeForSymbol.r0 -= 0.5; - clipShapeForSymbol.r += 0.5; - } - } - - this._clipShapeForSymbol = clipShapeForSymbol; - var visualColor = getVisualGradient(data, coordSys, api) || data.getVisual('style')[data.getVisual('drawType')]; // Initialization animation or coordinate system changed - - if (!(polyline && prevCoordSys.type === coordSys.type && step === this._step)) { - showSymbol && symbolDraw.updateData(data, { - isIgnore: isIgnoreFunc, - clipShape: clipShapeForSymbol, - disableAnimation: true, - getSymbolPoint: function (idx) { - return [points[idx * 2], points[idx * 2 + 1]]; - } - }); - hasAnimation && this._initSymbolLabelAnimation(data, coordSys, clipShapeForSymbol); - - if (step) { - // TODO If stacked series is not step - points = turnPointsIntoStep(points, coordSys, step, connectNulls); - - if (stackedOnPoints) { - stackedOnPoints = turnPointsIntoStep(stackedOnPoints, coordSys, step, connectNulls); - } - } - - polyline = this._newPolyline(points); - - if (isAreaChart) { - polygon = this._newPolygon(points, stackedOnPoints); - } // If areaStyle is removed - else if (polygon) { - lineGroup.remove(polygon); - polygon = this._polygon = null; - } // NOTE: Must update _endLabel before setClipPath. - - - if (!isCoordSysPolar) { - this._initOrUpdateEndLabel(seriesModel, coordSys, convertToColorString(visualColor)); - } - - lineGroup.setClipPath(createLineClipPath(this, coordSys, true, seriesModel)); - } else { - if (isAreaChart && !polygon) { - // If areaStyle is added - polygon = this._newPolygon(points, stackedOnPoints); - } else if (polygon && !isAreaChart) { - // If areaStyle is removed - lineGroup.remove(polygon); - polygon = this._polygon = null; - } // NOTE: Must update _endLabel before setClipPath. - - - if (!isCoordSysPolar) { - this._initOrUpdateEndLabel(seriesModel, coordSys, convertToColorString(visualColor)); - } // Update clipPath - - - var oldClipPath = lineGroup.getClipPath(); - - if (oldClipPath) { - var newClipPath = createLineClipPath(this, coordSys, false, seriesModel); - initProps(oldClipPath, { - shape: newClipPath.shape - }, seriesModel); - } else { - lineGroup.setClipPath(createLineClipPath(this, coordSys, true, seriesModel)); - } // Always update, or it is wrong in the case turning on legend - // because points are not changed. - - - showSymbol && symbolDraw.updateData(data, { - isIgnore: isIgnoreFunc, - clipShape: clipShapeForSymbol, - disableAnimation: true, - getSymbolPoint: function (idx) { - return [points[idx * 2], points[idx * 2 + 1]]; - } - }); // In the case data zoom triggered refreshing frequently - // Data may not change if line has a category axis. So it should animate nothing. - - if (!isPointsSame(this._stackedOnPoints, stackedOnPoints) || !isPointsSame(this._points, points)) { - if (hasAnimation) { - this._doUpdateAnimation(data, stackedOnPoints, coordSys, api, step, valueOrigin, connectNulls); - } else { - // Not do it in update with animation - if (step) { - // TODO If stacked series is not step - points = turnPointsIntoStep(points, coordSys, step, connectNulls); - - if (stackedOnPoints) { - stackedOnPoints = turnPointsIntoStep(stackedOnPoints, coordSys, step, connectNulls); - } - } - - polyline.setShape({ - points: points - }); - polygon && polygon.setShape({ - points: points, - stackedOnPoints: stackedOnPoints - }); - } - } - } - - var emphasisModel = seriesModel.getModel('emphasis'); - var focus = emphasisModel.get('focus'); - var blurScope = emphasisModel.get('blurScope'); - var emphasisDisabled = emphasisModel.get('disabled'); - polyline.useStyle(defaults( // Use color in lineStyle first - lineStyleModel.getLineStyle(), { - fill: 'none', - stroke: visualColor, - lineJoin: 'bevel' - })); - setStatesStylesFromModel(polyline, seriesModel, 'lineStyle'); - - if (polyline.style.lineWidth > 0 && seriesModel.get(['emphasis', 'lineStyle', 'width']) === 'bolder') { - var emphasisLineStyle = polyline.getState('emphasis').style; - emphasisLineStyle.lineWidth = +polyline.style.lineWidth + 1; - } // Needs seriesIndex for focus - - - getECData(polyline).seriesIndex = seriesModel.seriesIndex; - toggleHoverEmphasis(polyline, focus, blurScope, emphasisDisabled); - var smooth = getSmooth(seriesModel.get('smooth')); - var smoothMonotone = seriesModel.get('smoothMonotone'); - polyline.setShape({ - smooth: smooth, - smoothMonotone: smoothMonotone, - connectNulls: connectNulls - }); - - if (polygon) { - var stackedOnSeries = data.getCalculationInfo('stackedOnSeries'); - var stackedOnSmooth = 0; - polygon.useStyle(defaults(areaStyleModel.getAreaStyle(), { - fill: visualColor, - opacity: 0.7, - lineJoin: 'bevel', - decal: data.getVisual('style').decal - })); - - if (stackedOnSeries) { - stackedOnSmooth = getSmooth(stackedOnSeries.get('smooth')); - } - - polygon.setShape({ - smooth: smooth, - stackedOnSmooth: stackedOnSmooth, - smoothMonotone: smoothMonotone, - connectNulls: connectNulls - }); - setStatesStylesFromModel(polygon, seriesModel, 'areaStyle'); // Needs seriesIndex for focus - - getECData(polygon).seriesIndex = seriesModel.seriesIndex; - toggleHoverEmphasis(polygon, focus, blurScope, emphasisDisabled); - } - - var changePolyState = function (toState) { - _this._changePolyState(toState); - }; - - data.eachItemGraphicEl(function (el) { - // Switch polyline / polygon state if element changed its state. - el && (el.onHoverStateChange = changePolyState); - }); - this._polyline.onHoverStateChange = changePolyState; - this._data = data; // Save the coordinate system for transition animation when data changed - - this._coordSys = coordSys; - this._stackedOnPoints = stackedOnPoints; - this._points = points; - this._step = step; - this._valueOrigin = valueOrigin; - - if (seriesModel.get('triggerLineEvent')) { - this.packEventData(seriesModel, polyline); - polygon && this.packEventData(seriesModel, polygon); - } - }; - - LineView.prototype.packEventData = function (seriesModel, el) { - getECData(el).eventData = { - componentType: 'series', - componentSubType: 'line', - componentIndex: seriesModel.componentIndex, - seriesIndex: seriesModel.seriesIndex, - seriesName: seriesModel.name, - seriesType: 'line' - }; - }; - - LineView.prototype.highlight = function (seriesModel, ecModel, api, payload) { - var data = seriesModel.getData(); - var dataIndex = queryDataIndex(data, payload); - - this._changePolyState('emphasis'); - - if (!(dataIndex instanceof Array) && dataIndex != null && dataIndex >= 0) { - var points = data.getLayout('points'); - var symbol = data.getItemGraphicEl(dataIndex); - - if (!symbol) { - // Create a temporary symbol if it is not exists - var x = points[dataIndex * 2]; - var y = points[dataIndex * 2 + 1]; - - if (isNaN(x) || isNaN(y)) { - // Null data - return; - } // fix #11360: shouldn't draw symbol outside clipShapeForSymbol - - - if (this._clipShapeForSymbol && !this._clipShapeForSymbol.contain(x, y)) { - return; - } - - var zlevel = seriesModel.get('zlevel') || 0; - var z = seriesModel.get('z') || 0; - symbol = new Symbol(data, dataIndex); - symbol.x = x; - symbol.y = y; - symbol.setZ(zlevel, z); // ensure label text of the temporary symbol is in front of line and area polygon - - var symbolLabel = symbol.getSymbolPath().getTextContent(); - - if (symbolLabel) { - symbolLabel.zlevel = zlevel; - symbolLabel.z = z; - symbolLabel.z2 = this._polyline.z2 + 1; - } - - symbol.__temp = true; - data.setItemGraphicEl(dataIndex, symbol); // Stop scale animation - - symbol.stopSymbolAnimation(true); - this.group.add(symbol); - } - - symbol.highlight(); - } else { - // Highlight whole series - ChartView.prototype.highlight.call(this, seriesModel, ecModel, api, payload); - } - }; - - LineView.prototype.downplay = function (seriesModel, ecModel, api, payload) { - var data = seriesModel.getData(); - var dataIndex = queryDataIndex(data, payload); - - this._changePolyState('normal'); - - if (dataIndex != null && dataIndex >= 0) { - var symbol = data.getItemGraphicEl(dataIndex); - - if (symbol) { - if (symbol.__temp) { - data.setItemGraphicEl(dataIndex, null); - this.group.remove(symbol); - } else { - symbol.downplay(); - } - } - } else { - // FIXME - // can not downplay completely. - // Downplay whole series - ChartView.prototype.downplay.call(this, seriesModel, ecModel, api, payload); - } - }; - - LineView.prototype._changePolyState = function (toState) { - var polygon = this._polygon; - setStatesFlag(this._polyline, toState); - polygon && setStatesFlag(polygon, toState); - }; - - LineView.prototype._newPolyline = function (points) { - var polyline = this._polyline; // Remove previous created polyline - - if (polyline) { - this._lineGroup.remove(polyline); - } - - polyline = new ECPolyline({ - shape: { - points: points - }, - segmentIgnoreThreshold: 2, - z2: 10 - }); - - this._lineGroup.add(polyline); - - this._polyline = polyline; - return polyline; - }; - - LineView.prototype._newPolygon = function (points, stackedOnPoints) { - var polygon = this._polygon; // Remove previous created polygon - - if (polygon) { - this._lineGroup.remove(polygon); - } - - polygon = new ECPolygon({ - shape: { - points: points, - stackedOnPoints: stackedOnPoints - }, - segmentIgnoreThreshold: 2 - }); - - this._lineGroup.add(polygon); - - this._polygon = polygon; - return polygon; - }; - - LineView.prototype._initSymbolLabelAnimation = function (data, coordSys, clipShape) { - var isHorizontalOrRadial; - var isCoordSysPolar; - var baseAxis = coordSys.getBaseAxis(); - var isAxisInverse = baseAxis.inverse; - - if (coordSys.type === 'cartesian2d') { - isHorizontalOrRadial = baseAxis.isHorizontal(); - isCoordSysPolar = false; - } else if (coordSys.type === 'polar') { - isHorizontalOrRadial = baseAxis.dim === 'angle'; - isCoordSysPolar = true; - } - - var seriesModel = data.hostModel; - var seriesDuration = seriesModel.get('animationDuration'); - - if (isFunction(seriesDuration)) { - seriesDuration = seriesDuration(null); - } - - var seriesDelay = seriesModel.get('animationDelay') || 0; - var seriesDelayValue = isFunction(seriesDelay) ? seriesDelay(null) : seriesDelay; - data.eachItemGraphicEl(function (symbol, idx) { - var el = symbol; - - if (el) { - var point = [symbol.x, symbol.y]; - var start = void 0; - var end = void 0; - var current = void 0; - - if (clipShape) { - if (isCoordSysPolar) { - var polarClip = clipShape; - var coord = coordSys.pointToCoord(point); - - if (isHorizontalOrRadial) { - start = polarClip.startAngle; - end = polarClip.endAngle; - current = -coord[1] / 180 * Math.PI; - } else { - start = polarClip.r0; - end = polarClip.r; - current = coord[0]; - } - } else { - var gridClip = clipShape; - - if (isHorizontalOrRadial) { - start = gridClip.x; - end = gridClip.x + gridClip.width; - current = symbol.x; - } else { - start = gridClip.y + gridClip.height; - end = gridClip.y; - current = symbol.y; - } - } - } - - var ratio = end === start ? 0 : (current - start) / (end - start); - - if (isAxisInverse) { - ratio = 1 - ratio; - } - - var delay = isFunction(seriesDelay) ? seriesDelay(idx) : seriesDuration * ratio + seriesDelayValue; - var symbolPath = el.getSymbolPath(); - var text = symbolPath.getTextContent(); - el.attr({ - scaleX: 0, - scaleY: 0 - }); - el.animateTo({ - scaleX: 1, - scaleY: 1 - }, { - duration: 200, - setToFinal: true, - delay: delay - }); - - if (text) { - text.animateFrom({ - style: { - opacity: 0 - } - }, { - duration: 300, - delay: delay - }); - } - - symbolPath.disableLabelAnimation = true; - } - }); - }; - - LineView.prototype._initOrUpdateEndLabel = function (seriesModel, coordSys, inheritColor) { - var endLabelModel = seriesModel.getModel('endLabel'); - - if (anyStateShowEndLabel(seriesModel)) { - var data_2 = seriesModel.getData(); - var polyline = this._polyline; // series may be filtered. - - var points = data_2.getLayout('points'); - - if (!points) { - polyline.removeTextContent(); - this._endLabel = null; - return; - } - - var endLabel = this._endLabel; - - if (!endLabel) { - endLabel = this._endLabel = new ZRText({ - z2: 200 // should be higher than item symbol - - }); - endLabel.ignoreClip = true; - polyline.setTextContent(this._endLabel); - polyline.disableLabelAnimation = true; - } // Find last non-NaN data to display data - - - var dataIndex = getLastIndexNotNull(points); - - if (dataIndex >= 0) { - setLabelStyle(polyline, getLabelStatesModels(seriesModel, 'endLabel'), { - inheritColor: inheritColor, - labelFetcher: seriesModel, - labelDataIndex: dataIndex, - defaultText: function (dataIndex, opt, interpolatedValue) { - return interpolatedValue != null ? getDefaultInterpolatedLabel(data_2, interpolatedValue) : getDefaultLabel(data_2, dataIndex); - }, - enableTextSetter: true - }, getEndLabelStateSpecified(endLabelModel, coordSys)); - polyline.textConfig.position = null; - } - } else if (this._endLabel) { - this._polyline.removeTextContent(); - - this._endLabel = null; - } - }; - - LineView.prototype._endLabelOnDuring = function (percent, clipRect, data, animationRecord, valueAnimation, endLabelModel, coordSys) { - var endLabel = this._endLabel; - var polyline = this._polyline; - - if (endLabel) { - // NOTE: Don't remove percent < 1. percent === 1 means the first frame during render. - // The label is not prepared at this time. - if (percent < 1 && animationRecord.originalX == null) { - animationRecord.originalX = endLabel.x; - animationRecord.originalY = endLabel.y; - } - - var points = data.getLayout('points'); - var seriesModel = data.hostModel; - var connectNulls = seriesModel.get('connectNulls'); - var precision = endLabelModel.get('precision'); - var distance = endLabelModel.get('distance') || 0; - var baseAxis = coordSys.getBaseAxis(); - var isHorizontal = baseAxis.isHorizontal(); - var isBaseInversed = baseAxis.inverse; - var clipShape = clipRect.shape; - var xOrY = isBaseInversed ? isHorizontal ? clipShape.x : clipShape.y + clipShape.height : isHorizontal ? clipShape.x + clipShape.width : clipShape.y; - var distanceX = (isHorizontal ? distance : 0) * (isBaseInversed ? -1 : 1); - var distanceY = (isHorizontal ? 0 : -distance) * (isBaseInversed ? -1 : 1); - var dim = isHorizontal ? 'x' : 'y'; - var dataIndexRange = getIndexRange(points, xOrY, dim); - var indices = dataIndexRange.range; - var diff = indices[1] - indices[0]; - var value = void 0; - - if (diff >= 1) { - // diff > 1 && connectNulls, which is on the null data. - if (diff > 1 && !connectNulls) { - var pt = getPointAtIndex(points, indices[0]); - endLabel.attr({ - x: pt[0] + distanceX, - y: pt[1] + distanceY - }); - valueAnimation && (value = seriesModel.getRawValue(indices[0])); - } else { - var pt = polyline.getPointOn(xOrY, dim); - pt && endLabel.attr({ - x: pt[0] + distanceX, - y: pt[1] + distanceY - }); - var startValue = seriesModel.getRawValue(indices[0]); - var endValue = seriesModel.getRawValue(indices[1]); - valueAnimation && (value = interpolateRawValues(data, precision, startValue, endValue, dataIndexRange.t)); - } - - animationRecord.lastFrameIndex = indices[0]; - } else { - // If diff <= 0, which is the range is not found(Include NaN) - // Choose the first point or last point. - var idx = percent === 1 || animationRecord.lastFrameIndex > 0 ? indices[0] : 0; - var pt = getPointAtIndex(points, idx); - valueAnimation && (value = seriesModel.getRawValue(idx)); - endLabel.attr({ - x: pt[0] + distanceX, - y: pt[1] + distanceY - }); - } - - if (valueAnimation) { - labelInner(endLabel).setLabelText(value); - } - } - }; - /** - * @private - */ - // FIXME Two value axis - - - LineView.prototype._doUpdateAnimation = function (data, stackedOnPoints, coordSys, api, step, valueOrigin, connectNulls) { - var polyline = this._polyline; - var polygon = this._polygon; - var seriesModel = data.hostModel; - var diff = lineAnimationDiff(this._data, data, this._stackedOnPoints, stackedOnPoints, this._coordSys, coordSys, this._valueOrigin); - var current = diff.current; - var stackedOnCurrent = diff.stackedOnCurrent; - var next = diff.next; - var stackedOnNext = diff.stackedOnNext; - - if (step) { - // TODO If stacked series is not step - current = turnPointsIntoStep(diff.current, coordSys, step, connectNulls); - stackedOnCurrent = turnPointsIntoStep(diff.stackedOnCurrent, coordSys, step, connectNulls); - next = turnPointsIntoStep(diff.next, coordSys, step, connectNulls); - stackedOnNext = turnPointsIntoStep(diff.stackedOnNext, coordSys, step, connectNulls); - } // Don't apply animation if diff is large. - // For better result and avoid memory explosion problems like - // https://github.com/apache/incubator-echarts/issues/12229 - - - if (getBoundingDiff(current, next) > 3000 || polygon && getBoundingDiff(stackedOnCurrent, stackedOnNext) > 3000) { - polyline.stopAnimation(); - polyline.setShape({ - points: next - }); - - if (polygon) { - polygon.stopAnimation(); - polygon.setShape({ - points: next, - stackedOnPoints: stackedOnNext - }); - } - - return; - } - - polyline.shape.__points = diff.current; - polyline.shape.points = current; - var target = { - shape: { - points: next - } - }; // Also animate the original points. - // If points reference is changed when turning into step line. - - if (diff.current !== current) { - target.shape.__points = diff.next; - } // Stop previous animation. - - - polyline.stopAnimation(); - updateProps(polyline, target, seriesModel); - - if (polygon) { - polygon.setShape({ - // Reuse the points with polyline. - points: current, - stackedOnPoints: stackedOnCurrent - }); - polygon.stopAnimation(); - updateProps(polygon, { - shape: { - stackedOnPoints: stackedOnNext - } - }, seriesModel); // If use attr directly in updateProps. - - if (polyline.shape.points !== polygon.shape.points) { - polygon.shape.points = polyline.shape.points; - } - } - - var updatedDataInfo = []; - var diffStatus = diff.status; - - for (var i = 0; i < diffStatus.length; i++) { - var cmd = diffStatus[i].cmd; - - if (cmd === '=') { - var el = data.getItemGraphicEl(diffStatus[i].idx1); - - if (el) { - updatedDataInfo.push({ - el: el, - ptIdx: i // Index of points - - }); - } - } - } - - if (polyline.animators && polyline.animators.length) { - polyline.animators[0].during(function () { - polygon && polygon.dirtyShape(); - var points = polyline.shape.__points; - - for (var i = 0; i < updatedDataInfo.length; i++) { - var el = updatedDataInfo[i].el; - var offset = updatedDataInfo[i].ptIdx * 2; - el.x = points[offset]; - el.y = points[offset + 1]; - el.markRedraw(); - } - }); - } - }; - - LineView.prototype.remove = function (ecModel) { - var group = this.group; - var oldData = this._data; - - this._lineGroup.removeAll(); - - this._symbolDraw.remove(true); // Remove temporary created elements when highlighting - - - oldData && oldData.eachItemGraphicEl(function (el, idx) { - if (el.__temp) { - group.remove(el); - oldData.setItemGraphicEl(idx, null); - } - }); - this._polyline = this._polygon = this._coordSys = this._points = this._stackedOnPoints = this._endLabel = this._data = null; - }; - - LineView.type = 'line'; - return LineView; - }(ChartView); - - function pointsLayout(seriesType, forceStoreInTypedArray) { - return { - seriesType: seriesType, - plan: createRenderPlanner(), - reset: function (seriesModel) { - var data = seriesModel.getData(); - var coordSys = seriesModel.coordinateSystem; - var pipelineContext = seriesModel.pipelineContext; - var useTypedArray = forceStoreInTypedArray || pipelineContext.large; - - if (!coordSys) { - return; - } - - var dims = map(coordSys.dimensions, function (dim) { - return data.mapDimension(dim); - }).slice(0, 2); - var dimLen = dims.length; - var stackResultDim = data.getCalculationInfo('stackResultDimension'); - - if (isDimensionStacked(data, dims[0])) { - dims[0] = stackResultDim; - } - - if (isDimensionStacked(data, dims[1])) { - dims[1] = stackResultDim; - } - - var store = data.getStore(); - var dimIdx0 = data.getDimensionIndex(dims[0]); - var dimIdx1 = data.getDimensionIndex(dims[1]); - return dimLen && { - progress: function (params, data) { - var segCount = params.end - params.start; - var points = useTypedArray && createFloat32Array(segCount * dimLen); - var tmpIn = []; - var tmpOut = []; - - for (var i = params.start, offset = 0; i < params.end; i++) { - var point = void 0; - - if (dimLen === 1) { - var x = store.get(dimIdx0, i); // NOTE: Make sure the second parameter is null to use default strategy. - - point = coordSys.dataToPoint(x, null, tmpOut); - } else { - tmpIn[0] = store.get(dimIdx0, i); - tmpIn[1] = store.get(dimIdx1, i); // Let coordinate system to handle the NaN data. - - point = coordSys.dataToPoint(tmpIn, null, tmpOut); - } - - if (useTypedArray) { - points[offset++] = point[0]; - points[offset++] = point[1]; - } else { - data.setItemLayout(i, point.slice()); - } - } - - useTypedArray && data.setLayout('points', points); - } - }; - } - }; - } - - var samplers = { - average: function (frame) { - var sum = 0; - var count = 0; - - for (var i = 0; i < frame.length; i++) { - if (!isNaN(frame[i])) { - sum += frame[i]; - count++; - } - } // Return NaN if count is 0 - - - return count === 0 ? NaN : sum / count; - }, - sum: function (frame) { - var sum = 0; - - for (var i = 0; i < frame.length; i++) { - // Ignore NaN - sum += frame[i] || 0; - } - - return sum; - }, - max: function (frame) { - var max = -Infinity; - - for (var i = 0; i < frame.length; i++) { - frame[i] > max && (max = frame[i]); - } // NaN will cause illegal axis extent. - - - return isFinite(max) ? max : NaN; - }, - min: function (frame) { - var min = Infinity; - - for (var i = 0; i < frame.length; i++) { - frame[i] < min && (min = frame[i]); - } // NaN will cause illegal axis extent. - - - return isFinite(min) ? min : NaN; - }, - // TODO - // Median - nearest: function (frame) { - return frame[0]; - } - }; - - var indexSampler = function (frame) { - return Math.round(frame.length / 2); - }; - - function dataSample(seriesType) { - return { - seriesType: seriesType, - // FIXME:TS never used, so comment it - // modifyOutputEnd: true, - reset: function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var sampling = seriesModel.get('sampling'); - var coordSys = seriesModel.coordinateSystem; - var count = data.count(); // Only cartesian2d support down sampling. Disable it when there is few data. - - if (count > 10 && coordSys.type === 'cartesian2d' && sampling) { - var baseAxis = coordSys.getBaseAxis(); - var valueAxis = coordSys.getOtherAxis(baseAxis); - var extent = baseAxis.getExtent(); - var dpr = api.getDevicePixelRatio(); // Coordinste system has been resized - - var size = Math.abs(extent[1] - extent[0]) * (dpr || 1); - var rate = Math.round(count / size); - - if (isFinite(rate) && rate > 1) { - if (sampling === 'lttb') { - seriesModel.setData(data.lttbDownSample(data.mapDimension(valueAxis.dim), 1 / rate)); - } - - var sampler = void 0; - - if (isString(sampling)) { - sampler = samplers[sampling]; - } else if (isFunction(sampling)) { - sampler = sampling; - } - - if (sampler) { - // Only support sample the first dim mapped from value axis. - seriesModel.setData(data.downSample(data.mapDimension(valueAxis.dim), 1 / rate, sampler, indexSampler)); - } - } - } - } - }; - } - - function install$2(registers) { - registers.registerChartView(LineView); - registers.registerSeriesModel(LineSeriesModel); - registers.registerLayout(pointsLayout('line', true)); - registers.registerVisual({ - seriesType: 'line', - reset: function (seriesModel) { - var data = seriesModel.getData(); // Visual coding for legend - - var lineStyle = seriesModel.getModel('lineStyle').getLineStyle(); - - if (lineStyle && !lineStyle.stroke) { - // Fill in visual should be palette color if - // has color callback - lineStyle.stroke = data.getVisual('style').fill; - } - - data.setVisual('legendLineStyle', lineStyle); - } - }); // Down sample after filter - - registers.registerProcessor(registers.PRIORITY.PROCESSOR.STATISTIC, dataSample('line')); - } - - var BaseBarSeriesModel = - /** @class */ - function (_super) { - __extends(BaseBarSeriesModel, _super); - - function BaseBarSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = BaseBarSeriesModel.type; - return _this; - } - - BaseBarSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesData(null, this, { - useEncodeDefaulter: true - }); - }; - - BaseBarSeriesModel.prototype.getMarkerPosition = function (value, dims, startingAtTick) { - var coordSys = this.coordinateSystem; - - if (coordSys && coordSys.clampData) { - // PENDING if clamp ? - var clampData_1 = coordSys.clampData(value); - var pt_1 = coordSys.dataToPoint(clampData_1); - - if (startingAtTick) { - each(coordSys.getAxes(), function (axis, idx) { - // If axis type is category, use tick coords instead - if (axis.type === 'category' && dims != null) { - var tickCoords = axis.getTicksCoords(); - var targetTickId = clampData_1[idx]; // The index of rightmost tick of markArea is 1 larger than x1/y1 index - - var isEnd = dims[idx] === 'x1' || dims[idx] === 'y1'; - - if (isEnd) { - targetTickId += 1; - } // The only contains one tick, tickCoords is - // like [{coord: 0, tickValue: 0}, {coord: 0}] - // to the length should always be larger than 1 - - - if (tickCoords.length < 2) { - return; - } else if (tickCoords.length === 2) { - // The left value and right value of the axis are - // the same. coord is 0 in both items. Use the max - // value of the axis as the coord - pt_1[idx] = axis.toGlobalCoord(axis.getExtent()[isEnd ? 1 : 0]); - return; - } - - var leftCoord = void 0; - var coord = void 0; - var stepTickValue = 1; - - for (var i = 0; i < tickCoords.length; i++) { - var tickCoord = tickCoords[i].coord; // The last item of tickCoords doesn't contain - // tickValue - - var tickValue = i === tickCoords.length - 1 ? tickCoords[i - 1].tickValue + stepTickValue : tickCoords[i].tickValue; - - if (tickValue === targetTickId) { - coord = tickCoord; - break; - } else if (tickValue < targetTickId) { - leftCoord = tickCoord; - } else if (leftCoord != null && tickValue > targetTickId) { - coord = (tickCoord + leftCoord) / 2; - break; - } - - if (i === 1) { - // Here we assume the step of category axes is - // the same - stepTickValue = tickValue - tickCoords[0].tickValue; - } - } - - if (coord == null) { - if (!leftCoord) { - // targetTickId is smaller than all tick ids in the - // visible area, use the leftmost tick coord - coord = tickCoords[0].coord; - } else if (leftCoord) { - // targetTickId is larger than all tick ids in the - // visible area, use the rightmost tick coord - coord = tickCoords[tickCoords.length - 1].coord; - } - } - - pt_1[idx] = axis.toGlobalCoord(coord); - } - }); - } else { - var data = this.getData(); - var offset = data.getLayout('offset'); - var size = data.getLayout('size'); - var offsetIndex = coordSys.getBaseAxis().isHorizontal() ? 0 : 1; - pt_1[offsetIndex] += offset + size / 2; - } - - return pt_1; - } - - return [NaN, NaN]; - }; - - BaseBarSeriesModel.type = 'series.__base_bar__'; - BaseBarSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'cartesian2d', - legendHoverLink: true, - // stack: null - // Cartesian coordinate system - // xAxisIndex: 0, - // yAxisIndex: 0, - barMinHeight: 0, - barMinAngle: 0, - // cursor: null, - large: false, - largeThreshold: 400, - progressive: 3e3, - progressiveChunkMode: 'mod' - }; - return BaseBarSeriesModel; - }(SeriesModel); - - SeriesModel.registerClass(BaseBarSeriesModel); - - var BarSeriesModel = - /** @class */ - function (_super) { - __extends(BarSeriesModel, _super); - - function BarSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = BarSeriesModel.type; - return _this; - } - - BarSeriesModel.prototype.getInitialData = function () { - return createSeriesData(null, this, { - useEncodeDefaulter: true, - createInvertedIndices: !!this.get('realtimeSort', true) || null - }); - }; - /** - * @override - */ - - - BarSeriesModel.prototype.getProgressive = function () { - // Do not support progressive in normal mode. - return this.get('large') ? this.get('progressive') : false; - }; - /** - * @override - */ - - - BarSeriesModel.prototype.getProgressiveThreshold = function () { - // Do not support progressive in normal mode. - var progressiveThreshold = this.get('progressiveThreshold'); - var largeThreshold = this.get('largeThreshold'); - - if (largeThreshold > progressiveThreshold) { - progressiveThreshold = largeThreshold; - } - - return progressiveThreshold; - }; - - BarSeriesModel.prototype.brushSelector = function (dataIndex, data, selectors) { - return selectors.rect(data.getItemLayout(dataIndex)); - }; - - BarSeriesModel.type = 'series.bar'; - BarSeriesModel.dependencies = ['grid', 'polar']; - BarSeriesModel.defaultOption = inheritDefaultOption(BaseBarSeriesModel.defaultOption, { - // If clipped - // Only available on cartesian2d - clip: true, - roundCap: false, - showBackground: false, - backgroundStyle: { - color: 'rgba(180, 180, 180, 0.2)', - borderColor: null, - borderWidth: 0, - borderType: 'solid', - borderRadius: 0, - shadowBlur: 0, - shadowColor: null, - shadowOffsetX: 0, - shadowOffsetY: 0, - opacity: 1 - }, - select: { - itemStyle: { - borderColor: '#212121' - } - }, - realtimeSort: false - }); - return BarSeriesModel; - }(BaseBarSeriesModel); - - /** - * Sausage: similar to sector, but have half circle on both sides - */ - - var SausageShape = - /** @class */ - function () { - function SausageShape() { - this.cx = 0; - this.cy = 0; - this.r0 = 0; - this.r = 0; - this.startAngle = 0; - this.endAngle = Math.PI * 2; - this.clockwise = true; - } - - return SausageShape; - }(); - - var SausagePath = - /** @class */ - function (_super) { - __extends(SausagePath, _super); - - function SausagePath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'sausage'; - return _this; - } - - SausagePath.prototype.getDefaultShape = function () { - return new SausageShape(); - }; - - SausagePath.prototype.buildPath = function (ctx, shape) { - var cx = shape.cx; - var cy = shape.cy; - var r0 = Math.max(shape.r0 || 0, 0); - var r = Math.max(shape.r, 0); - var dr = (r - r0) * 0.5; - var rCenter = r0 + dr; - var startAngle = shape.startAngle; - var endAngle = shape.endAngle; - var clockwise = shape.clockwise; - var PI2 = Math.PI * 2; - var lessThanCircle = clockwise ? endAngle - startAngle < PI2 : startAngle - endAngle < PI2; - - if (!lessThanCircle) { - // Normalize angles - startAngle = endAngle - (clockwise ? PI2 : -PI2); - } - - var unitStartX = Math.cos(startAngle); - var unitStartY = Math.sin(startAngle); - var unitEndX = Math.cos(endAngle); - var unitEndY = Math.sin(endAngle); - - if (lessThanCircle) { - ctx.moveTo(unitStartX * r0 + cx, unitStartY * r0 + cy); - ctx.arc(unitStartX * rCenter + cx, unitStartY * rCenter + cy, dr, -Math.PI + startAngle, startAngle, !clockwise); - } else { - ctx.moveTo(unitStartX * r + cx, unitStartY * r + cy); - } - - ctx.arc(cx, cy, r, startAngle, endAngle, !clockwise); - ctx.arc(unitEndX * rCenter + cx, unitEndY * rCenter + cy, dr, endAngle - Math.PI * 2, endAngle - Math.PI, !clockwise); - - if (r0 !== 0) { - ctx.arc(cx, cy, r0, endAngle, startAngle, clockwise); - } // ctx.closePath(); - - }; - - return SausagePath; - }(Path); - - function createSectorCalculateTextPosition(positionMapping, opts) { - opts = opts || {}; - var isRoundCap = opts.isRoundCap; - return function (out, opts, boundingRect) { - var textPosition = opts.position; - - if (!textPosition || textPosition instanceof Array) { - return calculateTextPosition(out, opts, boundingRect); - } - - var mappedSectorPosition = positionMapping(textPosition); - var distance = opts.distance != null ? opts.distance : 5; - var sector = this.shape; - var cx = sector.cx; - var cy = sector.cy; - var r = sector.r; - var r0 = sector.r0; - var middleR = (r + r0) / 2; - var startAngle = sector.startAngle; - var endAngle = sector.endAngle; - var middleAngle = (startAngle + endAngle) / 2; - var extraDist = isRoundCap ? Math.abs(r - r0) / 2 : 0; - var mathCos = Math.cos; - var mathSin = Math.sin; // base position: top-left - - var x = cx + r * mathCos(startAngle); - var y = cy + r * mathSin(startAngle); - var textAlign = 'left'; - var textVerticalAlign = 'top'; - - switch (mappedSectorPosition) { - case 'startArc': - x = cx + (r0 - distance) * mathCos(middleAngle); - y = cy + (r0 - distance) * mathSin(middleAngle); - textAlign = 'center'; - textVerticalAlign = 'top'; - break; - - case 'insideStartArc': - x = cx + (r0 + distance) * mathCos(middleAngle); - y = cy + (r0 + distance) * mathSin(middleAngle); - textAlign = 'center'; - textVerticalAlign = 'bottom'; - break; - - case 'startAngle': - x = cx + middleR * mathCos(startAngle) + adjustAngleDistanceX(startAngle, distance + extraDist, false); - y = cy + middleR * mathSin(startAngle) + adjustAngleDistanceY(startAngle, distance + extraDist, false); - textAlign = 'right'; - textVerticalAlign = 'middle'; - break; - - case 'insideStartAngle': - x = cx + middleR * mathCos(startAngle) + adjustAngleDistanceX(startAngle, -distance + extraDist, false); - y = cy + middleR * mathSin(startAngle) + adjustAngleDistanceY(startAngle, -distance + extraDist, false); - textAlign = 'left'; - textVerticalAlign = 'middle'; - break; - - case 'middle': - x = cx + middleR * mathCos(middleAngle); - y = cy + middleR * mathSin(middleAngle); - textAlign = 'center'; - textVerticalAlign = 'middle'; - break; - - case 'endArc': - x = cx + (r + distance) * mathCos(middleAngle); - y = cy + (r + distance) * mathSin(middleAngle); - textAlign = 'center'; - textVerticalAlign = 'bottom'; - break; - - case 'insideEndArc': - x = cx + (r - distance) * mathCos(middleAngle); - y = cy + (r - distance) * mathSin(middleAngle); - textAlign = 'center'; - textVerticalAlign = 'top'; - break; - - case 'endAngle': - x = cx + middleR * mathCos(endAngle) + adjustAngleDistanceX(endAngle, distance + extraDist, true); - y = cy + middleR * mathSin(endAngle) + adjustAngleDistanceY(endAngle, distance + extraDist, true); - textAlign = 'left'; - textVerticalAlign = 'middle'; - break; - - case 'insideEndAngle': - x = cx + middleR * mathCos(endAngle) + adjustAngleDistanceX(endAngle, -distance + extraDist, true); - y = cy + middleR * mathSin(endAngle) + adjustAngleDistanceY(endAngle, -distance + extraDist, true); - textAlign = 'right'; - textVerticalAlign = 'middle'; - break; - - default: - return calculateTextPosition(out, opts, boundingRect); - } - - out = out || {}; - out.x = x; - out.y = y; - out.align = textAlign; - out.verticalAlign = textVerticalAlign; - return out; - }; - } - function setSectorTextRotation(sector, textPosition, positionMapping, rotateType) { - if (isNumber(rotateType)) { - // user-set rotation - sector.setTextConfig({ - rotation: rotateType - }); - return; - } else if (isArray(textPosition)) { - // user-set position, use 0 as auto rotation - sector.setTextConfig({ - rotation: 0 - }); - return; - } - - var shape = sector.shape; - var startAngle = shape.clockwise ? shape.startAngle : shape.endAngle; - var endAngle = shape.clockwise ? shape.endAngle : shape.startAngle; - var middleAngle = (startAngle + endAngle) / 2; - var anchorAngle; - var mappedSectorPosition = positionMapping(textPosition); - - switch (mappedSectorPosition) { - case 'startArc': - case 'insideStartArc': - case 'middle': - case 'insideEndArc': - case 'endArc': - anchorAngle = middleAngle; - break; - - case 'startAngle': - case 'insideStartAngle': - anchorAngle = startAngle; - break; - - case 'endAngle': - case 'insideEndAngle': - anchorAngle = endAngle; - break; - - default: - sector.setTextConfig({ - rotation: 0 - }); - return; - } - - var rotate = Math.PI * 1.5 - anchorAngle; - /** - * TODO: labels with rotate > Math.PI / 2 should be rotate another - * half round flipped to increase readability. However, only middle - * position supports this for now, because in other positions, the - * anchor point is not at the center of the text, so the positions - * after rotating is not as expected. - */ - - if (mappedSectorPosition === 'middle' && rotate > Math.PI / 2 && rotate < Math.PI * 1.5) { - rotate -= Math.PI; - } - - sector.setTextConfig({ - rotation: rotate - }); - } - - function adjustAngleDistanceX(angle, distance, isEnd) { - return distance * Math.sin(angle) * (isEnd ? -1 : 1); - } - - function adjustAngleDistanceY(angle, distance, isEnd) { - return distance * Math.cos(angle) * (isEnd ? 1 : -1); - } - - function getSectorCornerRadius(model, shape, zeroIfNull) { - var cornerRadius = model.get('borderRadius'); - - if (cornerRadius == null) { - return zeroIfNull ? { - cornerRadius: 0 - } : null; - } - - if (!isArray(cornerRadius)) { - cornerRadius = [cornerRadius, cornerRadius, cornerRadius, cornerRadius]; - } - - var dr = Math.abs(shape.r || 0 - shape.r0 || 0); - return { - cornerRadius: map(cornerRadius, function (cr) { - return parsePercent(cr, dr); - }) - }; - } - - var mathMax$6 = Math.max; - var mathMin$6 = Math.min; - - function getClipArea(coord, data) { - var coordSysClipArea = coord.getArea && coord.getArea(); - - if (isCoordinateSystemType(coord, 'cartesian2d')) { - var baseAxis = coord.getBaseAxis(); // When boundaryGap is false or using time axis. bar may exceed the grid. - // We should not clip this part. - // See test/bar2.html - - if (baseAxis.type !== 'category' || !baseAxis.onBand) { - var expandWidth = data.getLayout('bandWidth'); - - if (baseAxis.isHorizontal()) { - coordSysClipArea.x -= expandWidth; - coordSysClipArea.width += expandWidth * 2; - } else { - coordSysClipArea.y -= expandWidth; - coordSysClipArea.height += expandWidth * 2; - } - } - } - - return coordSysClipArea; - } - - var BarView = - /** @class */ - function (_super) { - __extends(BarView, _super); - - function BarView() { - var _this = _super.call(this) || this; - - _this.type = BarView.type; - _this._isFirstFrame = true; - return _this; - } - - BarView.prototype.render = function (seriesModel, ecModel, api, payload) { - this._model = seriesModel; - - this._removeOnRenderedListener(api); - - this._updateDrawMode(seriesModel); - - var coordinateSystemType = seriesModel.get('coordinateSystem'); - - if (coordinateSystemType === 'cartesian2d' || coordinateSystemType === 'polar') { - // Clear previously rendered progressive elements. - this._progressiveEls = null; - this._isLargeDraw ? this._renderLarge(seriesModel, ecModel, api) : this._renderNormal(seriesModel, ecModel, api, payload); - } else if ("development" !== 'production') { - warn('Only cartesian2d and polar supported for bar.'); - } - }; - - BarView.prototype.incrementalPrepareRender = function (seriesModel) { - this._clear(); - - this._updateDrawMode(seriesModel); // incremental also need to clip, otherwise might be overlow. - // But must not set clip in each frame, otherwise all of the children will be marked redraw. - - - this._updateLargeClip(seriesModel); - }; - - BarView.prototype.incrementalRender = function (params, seriesModel) { - // Reset - this._progressiveEls = []; // Do not support progressive in normal mode. - - this._incrementalRenderLarge(params, seriesModel); - }; - - BarView.prototype.eachRendered = function (cb) { - traverseElements(this._progressiveEls || this.group, cb); - }; - - BarView.prototype._updateDrawMode = function (seriesModel) { - var isLargeDraw = seriesModel.pipelineContext.large; - - if (this._isLargeDraw == null || isLargeDraw !== this._isLargeDraw) { - this._isLargeDraw = isLargeDraw; - - this._clear(); - } - }; - - BarView.prototype._renderNormal = function (seriesModel, ecModel, api, payload) { - var group = this.group; - var data = seriesModel.getData(); - var oldData = this._data; - var coord = seriesModel.coordinateSystem; - var baseAxis = coord.getBaseAxis(); - var isHorizontalOrRadial; - - if (coord.type === 'cartesian2d') { - isHorizontalOrRadial = baseAxis.isHorizontal(); - } else if (coord.type === 'polar') { - isHorizontalOrRadial = baseAxis.dim === 'angle'; - } - - var animationModel = seriesModel.isAnimationEnabled() ? seriesModel : null; - var realtimeSortCfg = shouldRealtimeSort(seriesModel, coord); - - if (realtimeSortCfg) { - this._enableRealtimeSort(realtimeSortCfg, data, api); - } - - var needsClip = seriesModel.get('clip', true) || realtimeSortCfg; - var coordSysClipArea = getClipArea(coord, data); // If there is clipPath created in large mode. Remove it. - - group.removeClipPath(); // We don't use clipPath in normal mode because we needs a perfect animation - // And don't want the label are clipped. - - var roundCap = seriesModel.get('roundCap', true); - var drawBackground = seriesModel.get('showBackground', true); - var backgroundModel = seriesModel.getModel('backgroundStyle'); - var barBorderRadius = backgroundModel.get('borderRadius') || 0; - var bgEls = []; - var oldBgEls = this._backgroundEls; - var isInitSort = payload && payload.isInitSort; - var isChangeOrder = payload && payload.type === 'changeAxisOrder'; - - function createBackground(dataIndex) { - var bgLayout = getLayout[coord.type](data, dataIndex); - var bgEl = createBackgroundEl(coord, isHorizontalOrRadial, bgLayout); - bgEl.useStyle(backgroundModel.getItemStyle()); // Only cartesian2d support borderRadius. - - if (coord.type === 'cartesian2d') { - bgEl.setShape('r', barBorderRadius); - } else { - bgEl.setShape('cornerRadius', barBorderRadius); - } - - bgEls[dataIndex] = bgEl; - return bgEl; - } - data.diff(oldData).add(function (dataIndex) { - var itemModel = data.getItemModel(dataIndex); - var layout = getLayout[coord.type](data, dataIndex, itemModel); - - if (drawBackground) { - createBackground(dataIndex); - } // If dataZoom in filteMode: 'empty', the baseValue can be set as NaN in "axisProxy". - - - if (!data.hasValue(dataIndex) || !isValidLayout[coord.type](layout)) { - return; - } - - var isClipped = false; - - if (needsClip) { - // Clip will modify the layout params. - // And return a boolean to determine if the shape are fully clipped. - isClipped = clip[coord.type](coordSysClipArea, layout); - } - - var el = elementCreator[coord.type](seriesModel, data, dataIndex, layout, isHorizontalOrRadial, animationModel, baseAxis.model, false, roundCap); - - if (realtimeSortCfg) { - /** - * Force label animation because even if the element is - * ignored because it's clipped, it may not be clipped after - * changing order. Then, if not using forceLabelAnimation, - * the label animation was never started, in which case, - * the label will be the final value and doesn't have label - * animation. - */ - el.forceLabelAnimation = true; - } - - updateStyle(el, data, dataIndex, itemModel, layout, seriesModel, isHorizontalOrRadial, coord.type === 'polar'); - - if (isInitSort) { - el.attr({ - shape: layout - }); - } else if (realtimeSortCfg) { - updateRealtimeAnimation(realtimeSortCfg, animationModel, el, layout, dataIndex, isHorizontalOrRadial, false, false); - } else { - initProps(el, { - shape: layout - }, seriesModel, dataIndex); - } - - data.setItemGraphicEl(dataIndex, el); - group.add(el); - el.ignore = isClipped; - }).update(function (newIndex, oldIndex) { - var itemModel = data.getItemModel(newIndex); - var layout = getLayout[coord.type](data, newIndex, itemModel); - - if (drawBackground) { - var bgEl = void 0; - - if (oldBgEls.length === 0) { - bgEl = createBackground(oldIndex); - } else { - bgEl = oldBgEls[oldIndex]; - bgEl.useStyle(backgroundModel.getItemStyle()); // Only cartesian2d support borderRadius. - - if (coord.type === 'cartesian2d') { - bgEl.setShape('r', barBorderRadius); - } else { - bgEl.setShape('cornerRadius', barBorderRadius); - } - - bgEls[newIndex] = bgEl; - } - - var bgLayout = getLayout[coord.type](data, newIndex); - var shape = createBackgroundShape(isHorizontalOrRadial, bgLayout, coord); - updateProps(bgEl, { - shape: shape - }, animationModel, newIndex); - } - - var el = oldData.getItemGraphicEl(oldIndex); - - if (!data.hasValue(newIndex) || !isValidLayout[coord.type](layout)) { - group.remove(el); - return; - } - - var isClipped = false; - - if (needsClip) { - isClipped = clip[coord.type](coordSysClipArea, layout); - - if (isClipped) { - group.remove(el); - } - } - - if (!el) { - el = elementCreator[coord.type](seriesModel, data, newIndex, layout, isHorizontalOrRadial, animationModel, baseAxis.model, !!el, roundCap); - } else { - saveOldStyle(el); - } - - if (realtimeSortCfg) { - el.forceLabelAnimation = true; - } - - if (isChangeOrder) { - var textEl = el.getTextContent(); - - if (textEl) { - var labelInnerStore = labelInner(textEl); - - if (labelInnerStore.prevValue != null) { - /** - * Set preValue to be value so that no new label - * should be started, otherwise, it will take a full - * `animationDurationUpdate` time to finish the - * animation, which is not expected. - */ - labelInnerStore.prevValue = labelInnerStore.value; - } - } - } // Not change anything if only order changed. - // Especially not change label. - else { - updateStyle(el, data, newIndex, itemModel, layout, seriesModel, isHorizontalOrRadial, coord.type === 'polar'); - } - - if (isInitSort) { - el.attr({ - shape: layout - }); - } else if (realtimeSortCfg) { - updateRealtimeAnimation(realtimeSortCfg, animationModel, el, layout, newIndex, isHorizontalOrRadial, true, isChangeOrder); - } else { - updateProps(el, { - shape: layout - }, seriesModel, newIndex, null); - } - - data.setItemGraphicEl(newIndex, el); - el.ignore = isClipped; - group.add(el); - }).remove(function (dataIndex) { - var el = oldData.getItemGraphicEl(dataIndex); - el && removeElementWithFadeOut(el, seriesModel, dataIndex); - }).execute(); - var bgGroup = this._backgroundGroup || (this._backgroundGroup = new Group()); - bgGroup.removeAll(); - - for (var i = 0; i < bgEls.length; ++i) { - bgGroup.add(bgEls[i]); - } - - group.add(bgGroup); - this._backgroundEls = bgEls; - this._data = data; - }; - - BarView.prototype._renderLarge = function (seriesModel, ecModel, api) { - this._clear(); - - createLarge(seriesModel, this.group); - - this._updateLargeClip(seriesModel); - }; - - BarView.prototype._incrementalRenderLarge = function (params, seriesModel) { - this._removeBackground(); - - createLarge(seriesModel, this.group, this._progressiveEls, true); - }; - - BarView.prototype._updateLargeClip = function (seriesModel) { - // Use clipPath in large mode. - var clipPath = seriesModel.get('clip', true) && createClipPath(seriesModel.coordinateSystem, false, seriesModel); - var group = this.group; - - if (clipPath) { - group.setClipPath(clipPath); - } else { - group.removeClipPath(); - } - }; - - BarView.prototype._enableRealtimeSort = function (realtimeSortCfg, data, api) { - var _this = this; // If no data in the first frame, wait for data to initSort - - - if (!data.count()) { - return; - } - - var baseAxis = realtimeSortCfg.baseAxis; - - if (this._isFirstFrame) { - this._dispatchInitSort(data, realtimeSortCfg, api); - - this._isFirstFrame = false; - } else { - var orderMapping_1 = function (idx) { - var el = data.getItemGraphicEl(idx); - var shape = el && el.shape; - return shape && // The result should be consistent with the initial sort by data value. - // Do not support the case that both positive and negative exist. - Math.abs(baseAxis.isHorizontal() ? shape.height : shape.width) // If data is NaN, shape.xxx may be NaN, so use || 0 here in case - || 0; - }; - - this._onRendered = function () { - _this._updateSortWithinSameData(data, orderMapping_1, baseAxis, api); - }; - - api.getZr().on('rendered', this._onRendered); - } - }; - - BarView.prototype._dataSort = function (data, baseAxis, orderMapping) { - var info = []; - data.each(data.mapDimension(baseAxis.dim), function (ordinalNumber, dataIdx) { - var mappedValue = orderMapping(dataIdx); - mappedValue = mappedValue == null ? NaN : mappedValue; - info.push({ - dataIndex: dataIdx, - mappedValue: mappedValue, - ordinalNumber: ordinalNumber - }); - }); - info.sort(function (a, b) { - // If NaN, it will be treated as min val. - return b.mappedValue - a.mappedValue; - }); - return { - ordinalNumbers: map(info, function (item) { - return item.ordinalNumber; - }) - }; - }; - - BarView.prototype._isOrderChangedWithinSameData = function (data, orderMapping, baseAxis) { - var scale = baseAxis.scale; - var ordinalDataDim = data.mapDimension(baseAxis.dim); - var lastValue = Number.MAX_VALUE; - - for (var tickNum = 0, len = scale.getOrdinalMeta().categories.length; tickNum < len; ++tickNum) { - var rawIdx = data.rawIndexOf(ordinalDataDim, scale.getRawOrdinalNumber(tickNum)); - var value = rawIdx < 0 // If some tick have no bar, the tick will be treated as min. - ? Number.MIN_VALUE // PENDING: if dataZoom on baseAxis exits, is it a performance issue? - : orderMapping(data.indexOfRawIndex(rawIdx)); - - if (value > lastValue) { - return true; - } - - lastValue = value; - } - - return false; - }; - /* - * Consider the case when A and B changed order, whose representing - * bars are both out of sight, we don't wish to trigger reorder action - * as long as the order in the view doesn't change. - */ - - - BarView.prototype._isOrderDifferentInView = function (orderInfo, baseAxis) { - var scale = baseAxis.scale; - var extent = scale.getExtent(); - var tickNum = Math.max(0, extent[0]); - var tickMax = Math.min(extent[1], scale.getOrdinalMeta().categories.length - 1); - - for (; tickNum <= tickMax; ++tickNum) { - if (orderInfo.ordinalNumbers[tickNum] !== scale.getRawOrdinalNumber(tickNum)) { - return true; - } - } - }; - - BarView.prototype._updateSortWithinSameData = function (data, orderMapping, baseAxis, api) { - if (!this._isOrderChangedWithinSameData(data, orderMapping, baseAxis)) { - return; - } - - var sortInfo = this._dataSort(data, baseAxis, orderMapping); - - if (this._isOrderDifferentInView(sortInfo, baseAxis)) { - this._removeOnRenderedListener(api); - - api.dispatchAction({ - type: 'changeAxisOrder', - componentType: baseAxis.dim + 'Axis', - axisId: baseAxis.index, - sortInfo: sortInfo - }); - } - }; - - BarView.prototype._dispatchInitSort = function (data, realtimeSortCfg, api) { - var baseAxis = realtimeSortCfg.baseAxis; - - var sortResult = this._dataSort(data, baseAxis, function (dataIdx) { - return data.get(data.mapDimension(realtimeSortCfg.otherAxis.dim), dataIdx); - }); - - api.dispatchAction({ - type: 'changeAxisOrder', - componentType: baseAxis.dim + 'Axis', - isInitSort: true, - axisId: baseAxis.index, - sortInfo: sortResult - }); - }; - - BarView.prototype.remove = function (ecModel, api) { - this._clear(this._model); - - this._removeOnRenderedListener(api); - }; - - BarView.prototype.dispose = function (ecModel, api) { - this._removeOnRenderedListener(api); - }; - - BarView.prototype._removeOnRenderedListener = function (api) { - if (this._onRendered) { - api.getZr().off('rendered', this._onRendered); - this._onRendered = null; - } - }; - - BarView.prototype._clear = function (model) { - var group = this.group; - var data = this._data; - - if (model && model.isAnimationEnabled() && data && !this._isLargeDraw) { - this._removeBackground(); - - this._backgroundEls = []; - data.eachItemGraphicEl(function (el) { - removeElementWithFadeOut(el, model, getECData(el).dataIndex); - }); - } else { - group.removeAll(); - } - - this._data = null; - this._isFirstFrame = true; - }; - - BarView.prototype._removeBackground = function () { - this.group.remove(this._backgroundGroup); - this._backgroundGroup = null; - }; - - BarView.type = 'bar'; - return BarView; - }(ChartView); - - var clip = { - cartesian2d: function (coordSysBoundingRect, layout) { - var signWidth = layout.width < 0 ? -1 : 1; - var signHeight = layout.height < 0 ? -1 : 1; // Needs positive width and height - - if (signWidth < 0) { - layout.x += layout.width; - layout.width = -layout.width; - } - - if (signHeight < 0) { - layout.y += layout.height; - layout.height = -layout.height; - } - - var coordSysX2 = coordSysBoundingRect.x + coordSysBoundingRect.width; - var coordSysY2 = coordSysBoundingRect.y + coordSysBoundingRect.height; - var x = mathMax$6(layout.x, coordSysBoundingRect.x); - var x2 = mathMin$6(layout.x + layout.width, coordSysX2); - var y = mathMax$6(layout.y, coordSysBoundingRect.y); - var y2 = mathMin$6(layout.y + layout.height, coordSysY2); - var xClipped = x2 < x; - var yClipped = y2 < y; // When xClipped or yClipped, the element will be marked as `ignore`. - // But we should also place the element at the edge of the coord sys bounding rect. - // Because if data changed and the bar shows again, its transition animation - // will begin at this place. - - layout.x = xClipped && x > coordSysX2 ? x2 : x; - layout.y = yClipped && y > coordSysY2 ? y2 : y; - layout.width = xClipped ? 0 : x2 - x; - layout.height = yClipped ? 0 : y2 - y; // Reverse back - - if (signWidth < 0) { - layout.x += layout.width; - layout.width = -layout.width; - } - - if (signHeight < 0) { - layout.y += layout.height; - layout.height = -layout.height; - } - - return xClipped || yClipped; - }, - polar: function (coordSysClipArea, layout) { - var signR = layout.r0 <= layout.r ? 1 : -1; // Make sure r is larger than r0 - - if (signR < 0) { - var tmp = layout.r; - layout.r = layout.r0; - layout.r0 = tmp; - } - - var r = mathMin$6(layout.r, coordSysClipArea.r); - var r0 = mathMax$6(layout.r0, coordSysClipArea.r0); - layout.r = r; - layout.r0 = r0; - var clipped = r - r0 < 0; // Reverse back - - if (signR < 0) { - var tmp = layout.r; - layout.r = layout.r0; - layout.r0 = tmp; - } - - return clipped; - } - }; - var elementCreator = { - cartesian2d: function (seriesModel, data, newIndex, layout, isHorizontal, animationModel, axisModel, isUpdate, roundCap) { - var rect = new Rect({ - shape: extend({}, layout), - z2: 1 - }); - rect.__dataIndex = newIndex; - rect.name = 'item'; - - if (animationModel) { - var rectShape = rect.shape; - var animateProperty = isHorizontal ? 'height' : 'width'; - rectShape[animateProperty] = 0; - } - - return rect; - }, - polar: function (seriesModel, data, newIndex, layout, isRadial, animationModel, axisModel, isUpdate, roundCap) { - var ShapeClass = !isRadial && roundCap ? SausagePath : Sector; - var sector = new ShapeClass({ - shape: layout, - z2: 1 - }); - sector.name = 'item'; - var positionMap = createPolarPositionMapping(isRadial); - sector.calculateTextPosition = createSectorCalculateTextPosition(positionMap, { - isRoundCap: ShapeClass === SausagePath - }); // Animation - - if (animationModel) { - var sectorShape = sector.shape; - var animateProperty = isRadial ? 'r' : 'endAngle'; - var animateTarget = {}; - sectorShape[animateProperty] = isRadial ? layout.r0 : layout.startAngle; - animateTarget[animateProperty] = layout[animateProperty]; - (isUpdate ? updateProps : initProps)(sector, { - shape: animateTarget // __value: typeof dataValue === 'string' ? parseInt(dataValue, 10) : dataValue - - }, animationModel); - } - - return sector; - } - }; - - function shouldRealtimeSort(seriesModel, coordSys) { - var realtimeSortOption = seriesModel.get('realtimeSort', true); - var baseAxis = coordSys.getBaseAxis(); - - if ("development" !== 'production') { - if (realtimeSortOption) { - if (baseAxis.type !== 'category') { - warn('`realtimeSort` will not work because this bar series is not based on a category axis.'); - } - - if (coordSys.type !== 'cartesian2d') { - warn('`realtimeSort` will not work because this bar series is not on cartesian2d.'); - } - } - } - - if (realtimeSortOption && baseAxis.type === 'category' && coordSys.type === 'cartesian2d') { - return { - baseAxis: baseAxis, - otherAxis: coordSys.getOtherAxis(baseAxis) - }; - } - } - - function updateRealtimeAnimation(realtimeSortCfg, seriesAnimationModel, el, layout, newIndex, isHorizontal, isUpdate, isChangeOrder) { - var seriesTarget; - var axisTarget; - - if (isHorizontal) { - axisTarget = { - x: layout.x, - width: layout.width - }; - seriesTarget = { - y: layout.y, - height: layout.height - }; - } else { - axisTarget = { - y: layout.y, - height: layout.height - }; - seriesTarget = { - x: layout.x, - width: layout.width - }; - } - - if (!isChangeOrder) { - // Keep the original growth animation if only axis order changed. - // Not start a new animation. - (isUpdate ? updateProps : initProps)(el, { - shape: seriesTarget - }, seriesAnimationModel, newIndex, null); - } - - var axisAnimationModel = seriesAnimationModel ? realtimeSortCfg.baseAxis.model : null; - (isUpdate ? updateProps : initProps)(el, { - shape: axisTarget - }, axisAnimationModel, newIndex); - } - - function checkPropertiesNotValid(obj, props) { - for (var i = 0; i < props.length; i++) { - if (!isFinite(obj[props[i]])) { - return true; - } - } - - return false; - } - - var rectPropties = ['x', 'y', 'width', 'height']; - var polarPropties = ['cx', 'cy', 'r', 'startAngle', 'endAngle']; - var isValidLayout = { - cartesian2d: function (layout) { - return !checkPropertiesNotValid(layout, rectPropties); - }, - polar: function (layout) { - return !checkPropertiesNotValid(layout, polarPropties); - } - }; - var getLayout = { - // itemModel is only used to get borderWidth, which is not needed - // when calculating bar background layout. - cartesian2d: function (data, dataIndex, itemModel) { - var layout = data.getItemLayout(dataIndex); - var fixedLineWidth = itemModel ? getLineWidth(itemModel, layout) : 0; // fix layout with lineWidth - - var signX = layout.width > 0 ? 1 : -1; - var signY = layout.height > 0 ? 1 : -1; - return { - x: layout.x + signX * fixedLineWidth / 2, - y: layout.y + signY * fixedLineWidth / 2, - width: layout.width - signX * fixedLineWidth, - height: layout.height - signY * fixedLineWidth - }; - }, - polar: function (data, dataIndex, itemModel) { - var layout = data.getItemLayout(dataIndex); - return { - cx: layout.cx, - cy: layout.cy, - r0: layout.r0, - r: layout.r, - startAngle: layout.startAngle, - endAngle: layout.endAngle, - clockwise: layout.clockwise - }; - } - }; - - function isZeroOnPolar(layout) { - return layout.startAngle != null && layout.endAngle != null && layout.startAngle === layout.endAngle; - } - - function createPolarPositionMapping(isRadial) { - return function (isRadial) { - var arcOrAngle = isRadial ? 'Arc' : 'Angle'; - return function (position) { - switch (position) { - case 'start': - case 'insideStart': - case 'end': - case 'insideEnd': - return position + arcOrAngle; - - default: - return position; - } - }; - }(isRadial); - } - - function updateStyle(el, data, dataIndex, itemModel, layout, seriesModel, isHorizontalOrRadial, isPolar) { - var style = data.getItemVisual(dataIndex, 'style'); - - if (!isPolar) { - var borderRadius = itemModel.get(['itemStyle', 'borderRadius']) || 0; - el.setShape('r', borderRadius); - } else if (!seriesModel.get('roundCap')) { - var sectorShape = el.shape; - var cornerRadius = getSectorCornerRadius(itemModel.getModel('itemStyle'), sectorShape, true); - extend(sectorShape, cornerRadius); - el.setShape(sectorShape); - } - - el.useStyle(style); - var cursorStyle = itemModel.getShallow('cursor'); - cursorStyle && el.attr('cursor', cursorStyle); - var labelPositionOutside = isPolar ? isHorizontalOrRadial ? layout.r >= layout.r0 ? 'endArc' : 'startArc' : layout.endAngle >= layout.startAngle ? 'endAngle' : 'startAngle' : isHorizontalOrRadial ? layout.height >= 0 ? 'bottom' : 'top' : layout.width >= 0 ? 'right' : 'left'; - var labelStatesModels = getLabelStatesModels(itemModel); - setLabelStyle(el, labelStatesModels, { - labelFetcher: seriesModel, - labelDataIndex: dataIndex, - defaultText: getDefaultLabel(seriesModel.getData(), dataIndex), - inheritColor: style.fill, - defaultOpacity: style.opacity, - defaultOutsidePosition: labelPositionOutside - }); - var label = el.getTextContent(); - - if (isPolar && label) { - var position = itemModel.get(['label', 'position']); - el.textConfig.inside = position === 'middle' ? true : null; - setSectorTextRotation(el, position === 'outside' ? labelPositionOutside : position, createPolarPositionMapping(isHorizontalOrRadial), itemModel.get(['label', 'rotate'])); - } - - setLabelValueAnimation(label, labelStatesModels, seriesModel.getRawValue(dataIndex), function (value) { - return getDefaultInterpolatedLabel(data, value); - }); - var emphasisModel = itemModel.getModel(['emphasis']); - toggleHoverEmphasis(el, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - setStatesStylesFromModel(el, itemModel); - - if (isZeroOnPolar(layout)) { - el.style.fill = 'none'; - el.style.stroke = 'none'; - each(el.states, function (state) { - if (state.style) { - state.style.fill = state.style.stroke = 'none'; - } - }); - } - } // In case width or height are too small. - - - function getLineWidth(itemModel, rawLayout) { - // Has no border. - var borderColor = itemModel.get(['itemStyle', 'borderColor']); - - if (!borderColor || borderColor === 'none') { - return 0; - } - - var lineWidth = itemModel.get(['itemStyle', 'borderWidth']) || 0; // width or height may be NaN for empty data - - var width = isNaN(rawLayout.width) ? Number.MAX_VALUE : Math.abs(rawLayout.width); - var height = isNaN(rawLayout.height) ? Number.MAX_VALUE : Math.abs(rawLayout.height); - return Math.min(lineWidth, width, height); - } - - var LagePathShape = - /** @class */ - function () { - function LagePathShape() {} - - return LagePathShape; - }(); - - var LargePath = - /** @class */ - function (_super) { - __extends(LargePath, _super); - - function LargePath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'largeBar'; - return _this; - } - - LargePath.prototype.getDefaultShape = function () { - return new LagePathShape(); - }; - - LargePath.prototype.buildPath = function (ctx, shape) { - // Drawing lines is more efficient than drawing - // a whole line or drawing rects. - var points = shape.points; - var baseDimIdx = this.baseDimIdx; - var valueDimIdx = 1 - this.baseDimIdx; - var startPoint = []; - var size = []; - var barWidth = this.barWidth; - - for (var i = 0; i < points.length; i += 3) { - size[baseDimIdx] = barWidth; - size[valueDimIdx] = points[i + 2]; - startPoint[baseDimIdx] = points[i + baseDimIdx]; - startPoint[valueDimIdx] = points[i + valueDimIdx]; - ctx.rect(startPoint[0], startPoint[1], size[0], size[1]); - } - }; - - return LargePath; - }(Path); - - function createLarge(seriesModel, group, progressiveEls, incremental) { - // TODO support polar - var data = seriesModel.getData(); - var baseDimIdx = data.getLayout('valueAxisHorizontal') ? 1 : 0; - var largeDataIndices = data.getLayout('largeDataIndices'); - var barWidth = data.getLayout('size'); - var backgroundModel = seriesModel.getModel('backgroundStyle'); - var bgPoints = data.getLayout('largeBackgroundPoints'); - - if (bgPoints) { - var bgEl = new LargePath({ - shape: { - points: bgPoints - }, - incremental: !!incremental, - silent: true, - z2: 0 - }); - bgEl.baseDimIdx = baseDimIdx; - bgEl.largeDataIndices = largeDataIndices; - bgEl.barWidth = barWidth; - bgEl.useStyle(backgroundModel.getItemStyle()); - group.add(bgEl); - progressiveEls && progressiveEls.push(bgEl); - } - - var el = new LargePath({ - shape: { - points: data.getLayout('largePoints') - }, - incremental: !!incremental, - ignoreCoarsePointer: true, - z2: 1 - }); - el.baseDimIdx = baseDimIdx; - el.largeDataIndices = largeDataIndices; - el.barWidth = barWidth; - group.add(el); - el.useStyle(data.getVisual('style')); // Enable tooltip and user mouse/touch event handlers. - - getECData(el).seriesIndex = seriesModel.seriesIndex; - - if (!seriesModel.get('silent')) { - el.on('mousedown', largePathUpdateDataIndex); - el.on('mousemove', largePathUpdateDataIndex); - } - - progressiveEls && progressiveEls.push(el); - } // Use throttle to avoid frequently traverse to find dataIndex. - - - var largePathUpdateDataIndex = throttle(function (event) { - var largePath = this; - var dataIndex = largePathFindDataIndex(largePath, event.offsetX, event.offsetY); - getECData(largePath).dataIndex = dataIndex >= 0 ? dataIndex : null; - }, 30, false); - - function largePathFindDataIndex(largePath, x, y) { - var baseDimIdx = largePath.baseDimIdx; - var valueDimIdx = 1 - baseDimIdx; - var points = largePath.shape.points; - var largeDataIndices = largePath.largeDataIndices; - var startPoint = []; - var size = []; - var barWidth = largePath.barWidth; - - for (var i = 0, len = points.length / 3; i < len; i++) { - var ii = i * 3; - size[baseDimIdx] = barWidth; - size[valueDimIdx] = points[ii + 2]; - startPoint[baseDimIdx] = points[ii + baseDimIdx]; - startPoint[valueDimIdx] = points[ii + valueDimIdx]; - - if (size[valueDimIdx] < 0) { - startPoint[valueDimIdx] += size[valueDimIdx]; - size[valueDimIdx] = -size[valueDimIdx]; - } - - if (x >= startPoint[0] && x <= startPoint[0] + size[0] && y >= startPoint[1] && y <= startPoint[1] + size[1]) { - return largeDataIndices[i]; - } - } - - return -1; - } - - function createBackgroundShape(isHorizontalOrRadial, layout, coord) { - if (isCoordinateSystemType(coord, 'cartesian2d')) { - var rectShape = layout; - var coordLayout = coord.getArea(); - return { - x: isHorizontalOrRadial ? rectShape.x : coordLayout.x, - y: isHorizontalOrRadial ? coordLayout.y : rectShape.y, - width: isHorizontalOrRadial ? rectShape.width : coordLayout.width, - height: isHorizontalOrRadial ? coordLayout.height : rectShape.height - }; - } else { - var coordLayout = coord.getArea(); - var sectorShape = layout; - return { - cx: coordLayout.cx, - cy: coordLayout.cy, - r0: isHorizontalOrRadial ? coordLayout.r0 : sectorShape.r0, - r: isHorizontalOrRadial ? coordLayout.r : sectorShape.r, - startAngle: isHorizontalOrRadial ? sectorShape.startAngle : 0, - endAngle: isHorizontalOrRadial ? sectorShape.endAngle : Math.PI * 2 - }; - } - } - - function createBackgroundEl(coord, isHorizontalOrRadial, layout) { - var ElementClz = coord.type === 'polar' ? Sector : Rect; - return new ElementClz({ - shape: createBackgroundShape(isHorizontalOrRadial, layout, coord), - silent: true, - z2: 0 - }); - } - - function install$3(registers) { - registers.registerChartView(BarView); - registers.registerSeriesModel(BarSeriesModel); - registers.registerLayout(registers.PRIORITY.VISUAL.LAYOUT, curry(layout, 'bar')); // Do layout after other overall layout, which can prepare some information. - - registers.registerLayout(registers.PRIORITY.VISUAL.PROGRESSIVE_LAYOUT, createProgressiveLayout('bar')); // Down sample after filter - - registers.registerProcessor(registers.PRIORITY.PROCESSOR.STATISTIC, dataSample('bar')); - /** - * @payload - * @property {string} [componentType=series] - * @property {number} [dx] - * @property {number} [dy] - * @property {number} [zoom] - * @property {number} [originX] - * @property {number} [originY] - */ - - registers.registerAction({ - type: 'changeAxisOrder', - event: 'changeAxisOrder', - update: 'update' - }, function (payload, ecModel) { - var componentType = payload.componentType || 'series'; - ecModel.eachComponent({ - mainType: componentType, - query: payload - }, function (componentModel) { - if (payload.sortInfo) { - componentModel.axis.setCategorySortInfo(payload.sortInfo); - } - }); - }); - } - - var PI2$8 = Math.PI * 2; - var RADIAN = Math.PI / 180; - - function getViewRect(seriesModel, api) { - return getLayoutRect(seriesModel.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - } - - function getBasicPieLayout(seriesModel, api) { - var viewRect = getViewRect(seriesModel, api); // center can be string or number when coordinateSystem is specified - - var center = seriesModel.get('center'); - var radius = seriesModel.get('radius'); - - if (!isArray(radius)) { - radius = [0, radius]; - } - - var width = parsePercent$1(viewRect.width, api.getWidth()); - var height = parsePercent$1(viewRect.height, api.getHeight()); - var size = Math.min(width, height); - var r0 = parsePercent$1(radius[0], size / 2); - var r = parsePercent$1(radius[1], size / 2); - var cx; - var cy; - var coordSys = seriesModel.coordinateSystem; - - if (coordSys) { - // percentage is not allowed when coordinate system is specified - var point = coordSys.dataToPoint(center); - cx = point[0] || 0; - cy = point[1] || 0; - } else { - if (!isArray(center)) { - center = [center, center]; - } - - cx = parsePercent$1(center[0], width) + viewRect.x; - cy = parsePercent$1(center[1], height) + viewRect.y; - } - - return { - cx: cx, - cy: cy, - r0: r0, - r: r - }; - } - function pieLayout(seriesType, ecModel, api) { - ecModel.eachSeriesByType(seriesType, function (seriesModel) { - var data = seriesModel.getData(); - var valueDim = data.mapDimension('value'); - var viewRect = getViewRect(seriesModel, api); - - var _a = getBasicPieLayout(seriesModel, api), - cx = _a.cx, - cy = _a.cy, - r = _a.r, - r0 = _a.r0; - - var startAngle = -seriesModel.get('startAngle') * RADIAN; - var minAngle = seriesModel.get('minAngle') * RADIAN; - var validDataCount = 0; - data.each(valueDim, function (value) { - !isNaN(value) && validDataCount++; - }); - var sum = data.getSum(valueDim); // Sum may be 0 - - var unitRadian = Math.PI / (sum || validDataCount) * 2; - var clockwise = seriesModel.get('clockwise'); - var roseType = seriesModel.get('roseType'); - var stillShowZeroSum = seriesModel.get('stillShowZeroSum'); // [0...max] - - var extent = data.getDataExtent(valueDim); - extent[0] = 0; // In the case some sector angle is smaller than minAngle - - var restAngle = PI2$8; - var valueSumLargerThanMinAngle = 0; - var currentAngle = startAngle; - var dir = clockwise ? 1 : -1; - data.setLayout({ - viewRect: viewRect, - r: r - }); - data.each(valueDim, function (value, idx) { - var angle; - - if (isNaN(value)) { - data.setItemLayout(idx, { - angle: NaN, - startAngle: NaN, - endAngle: NaN, - clockwise: clockwise, - cx: cx, - cy: cy, - r0: r0, - r: roseType ? NaN : r - }); - return; - } // FIXME 兼容 2.0 但是 roseType 是 area 的时候才是这样? - - - if (roseType !== 'area') { - angle = sum === 0 && stillShowZeroSum ? unitRadian : value * unitRadian; - } else { - angle = PI2$8 / validDataCount; - } - - if (angle < minAngle) { - angle = minAngle; - restAngle -= minAngle; - } else { - valueSumLargerThanMinAngle += value; - } - - var endAngle = currentAngle + dir * angle; - data.setItemLayout(idx, { - angle: angle, - startAngle: currentAngle, - endAngle: endAngle, - clockwise: clockwise, - cx: cx, - cy: cy, - r0: r0, - r: roseType ? linearMap(value, extent, [r0, r]) : r - }); - currentAngle = endAngle; - }); // Some sector is constrained by minAngle - // Rest sectors needs recalculate angle - - if (restAngle < PI2$8 && validDataCount) { - // Average the angle if rest angle is not enough after all angles is - // Constrained by minAngle - if (restAngle <= 1e-3) { - var angle_1 = PI2$8 / validDataCount; - data.each(valueDim, function (value, idx) { - if (!isNaN(value)) { - var layout_1 = data.getItemLayout(idx); - layout_1.angle = angle_1; - layout_1.startAngle = startAngle + dir * idx * angle_1; - layout_1.endAngle = startAngle + dir * (idx + 1) * angle_1; - } - }); - } else { - unitRadian = restAngle / valueSumLargerThanMinAngle; - currentAngle = startAngle; - data.each(valueDim, function (value, idx) { - if (!isNaN(value)) { - var layout_2 = data.getItemLayout(idx); - var angle = layout_2.angle === minAngle ? minAngle : value * unitRadian; - layout_2.startAngle = currentAngle; - layout_2.endAngle = currentAngle + dir * angle; - currentAngle += dir * angle; - } - }); - } - } - }); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function dataFilter(seriesType) { - return { - seriesType: seriesType, - reset: function (seriesModel, ecModel) { - var legendModels = ecModel.findComponents({ - mainType: 'legend' - }); - - if (!legendModels || !legendModels.length) { - return; - } - - var data = seriesModel.getData(); - data.filterSelf(function (idx) { - var name = data.getName(idx); // If in any legend component the status is not selected. - - for (var i = 0; i < legendModels.length; i++) { - // @ts-ignore FIXME: LegendModel - if (!legendModels[i].isSelected(name)) { - return false; - } - } - - return true; - }); - } - }; - } - - var RADIAN$1 = Math.PI / 180; - - function adjustSingleSide(list, cx, cy, r, dir, viewWidth, viewHeight, viewLeft, viewTop, farthestX) { - if (list.length < 2) { - return; - } - - function recalculateXOnSemiToAlignOnEllipseCurve(semi) { - var rB = semi.rB; - var rB2 = rB * rB; - - for (var i = 0; i < semi.list.length; i++) { - var item = semi.list[i]; - var dy = Math.abs(item.label.y - cy); // horizontal r is always same with original r because x is not changed. - - var rA = r + item.len; - var rA2 = rA * rA; // Use ellipse implicit function to calculate x - - var dx = Math.sqrt((1 - Math.abs(dy * dy / rB2)) * rA2); - var newX = cx + (dx + item.len2) * dir; - var deltaX = newX - item.label.x; - var newTargetWidth = item.targetTextWidth - deltaX * dir; // text x is changed, so need to recalculate width. - - constrainTextWidth(item, newTargetWidth, true); - item.label.x = newX; - } - } // Adjust X based on the shifted y. Make tight labels aligned on an ellipse curve. - - - function recalculateX(items) { - // Extremes of - var topSemi = { - list: [], - maxY: 0 - }; - var bottomSemi = { - list: [], - maxY: 0 - }; - - for (var i = 0; i < items.length; i++) { - if (items[i].labelAlignTo !== 'none') { - continue; - } - - var item = items[i]; - var semi = item.label.y > cy ? bottomSemi : topSemi; - var dy = Math.abs(item.label.y - cy); - - if (dy >= semi.maxY) { - var dx = item.label.x - cx - item.len2 * dir; // horizontal r is always same with original r because x is not changed. - - var rA = r + item.len; // Canculate rB based on the topest / bottemest label. - - var rB = Math.abs(dx) < rA ? Math.sqrt(dy * dy / (1 - dx * dx / rA / rA)) : rA; - semi.rB = rB; - semi.maxY = dy; - } - - semi.list.push(item); - } - - recalculateXOnSemiToAlignOnEllipseCurve(topSemi); - recalculateXOnSemiToAlignOnEllipseCurve(bottomSemi); - } - - var len = list.length; - - for (var i = 0; i < len; i++) { - if (list[i].position === 'outer' && list[i].labelAlignTo === 'labelLine') { - var dx = list[i].label.x - farthestX; - list[i].linePoints[1][0] += dx; - list[i].label.x = farthestX; - } - } - - if (shiftLayoutOnY(list, viewTop, viewTop + viewHeight)) { - recalculateX(list); - } - } - - function avoidOverlap(labelLayoutList, cx, cy, r, viewWidth, viewHeight, viewLeft, viewTop) { - var leftList = []; - var rightList = []; - var leftmostX = Number.MAX_VALUE; - var rightmostX = -Number.MAX_VALUE; - - for (var i = 0; i < labelLayoutList.length; i++) { - var label = labelLayoutList[i].label; - - if (isPositionCenter(labelLayoutList[i])) { - continue; - } - - if (label.x < cx) { - leftmostX = Math.min(leftmostX, label.x); - leftList.push(labelLayoutList[i]); - } else { - rightmostX = Math.max(rightmostX, label.x); - rightList.push(labelLayoutList[i]); - } - } - - for (var i = 0; i < labelLayoutList.length; i++) { - var layout = labelLayoutList[i]; - - if (!isPositionCenter(layout) && layout.linePoints) { - if (layout.labelStyleWidth != null) { - continue; - } - - var label = layout.label; - var linePoints = layout.linePoints; - var targetTextWidth = void 0; - - if (layout.labelAlignTo === 'edge') { - if (label.x < cx) { - targetTextWidth = linePoints[2][0] - layout.labelDistance - viewLeft - layout.edgeDistance; - } else { - targetTextWidth = viewLeft + viewWidth - layout.edgeDistance - linePoints[2][0] - layout.labelDistance; - } - } else if (layout.labelAlignTo === 'labelLine') { - if (label.x < cx) { - targetTextWidth = leftmostX - viewLeft - layout.bleedMargin; - } else { - targetTextWidth = viewLeft + viewWidth - rightmostX - layout.bleedMargin; - } - } else { - if (label.x < cx) { - targetTextWidth = label.x - viewLeft - layout.bleedMargin; - } else { - targetTextWidth = viewLeft + viewWidth - label.x - layout.bleedMargin; - } - } - - layout.targetTextWidth = targetTextWidth; - constrainTextWidth(layout, targetTextWidth); - } - } - - adjustSingleSide(rightList, cx, cy, r, 1, viewWidth, viewHeight, viewLeft, viewTop, rightmostX); - adjustSingleSide(leftList, cx, cy, r, -1, viewWidth, viewHeight, viewLeft, viewTop, leftmostX); - - for (var i = 0; i < labelLayoutList.length; i++) { - var layout = labelLayoutList[i]; - - if (!isPositionCenter(layout) && layout.linePoints) { - var label = layout.label; - var linePoints = layout.linePoints; - var isAlignToEdge = layout.labelAlignTo === 'edge'; - var padding = label.style.padding; - var paddingH = padding ? padding[1] + padding[3] : 0; // textRect.width already contains paddingH if bgColor is set - - var extraPaddingH = label.style.backgroundColor ? 0 : paddingH; - var realTextWidth = layout.rect.width + extraPaddingH; - var dist = linePoints[1][0] - linePoints[2][0]; - - if (isAlignToEdge) { - if (label.x < cx) { - linePoints[2][0] = viewLeft + layout.edgeDistance + realTextWidth + layout.labelDistance; - } else { - linePoints[2][0] = viewLeft + viewWidth - layout.edgeDistance - realTextWidth - layout.labelDistance; - } - } else { - if (label.x < cx) { - linePoints[2][0] = label.x + layout.labelDistance; - } else { - linePoints[2][0] = label.x - layout.labelDistance; - } - - linePoints[1][0] = linePoints[2][0] + dist; - } - - linePoints[1][1] = linePoints[2][1] = label.y; - } - } - } - /** - * Set max width of each label, and then wrap each label to the max width. - * - * @param layout label layout - * @param availableWidth max width for the label to display - * @param forceRecalculate recaculate the text layout even if the current width - * is smaller than `availableWidth`. This is useful when the text was previously - * wrapped by calling `constrainTextWidth` but now `availableWidth` changed, in - * which case, previous wrapping should be redo. - */ - - - function constrainTextWidth(layout, availableWidth, forceRecalculate) { - if (forceRecalculate === void 0) { - forceRecalculate = false; - } - - if (layout.labelStyleWidth != null) { - // User-defined style.width has the highest priority. - return; - } - - var label = layout.label; - var style = label.style; - var textRect = layout.rect; - var bgColor = style.backgroundColor; - var padding = style.padding; - var paddingH = padding ? padding[1] + padding[3] : 0; - var overflow = style.overflow; // textRect.width already contains paddingH if bgColor is set - - var oldOuterWidth = textRect.width + (bgColor ? 0 : paddingH); - - if (availableWidth < oldOuterWidth || forceRecalculate) { - var oldHeight = textRect.height; - - if (overflow && overflow.match('break')) { - // Temporarily set background to be null to calculate - // the bounding box without background. - label.setStyle('backgroundColor', null); // Set constraining width - - label.setStyle('width', availableWidth - paddingH); // This is the real bounding box of the text without padding. - - var innerRect = label.getBoundingRect(); - label.setStyle('width', Math.ceil(innerRect.width)); - label.setStyle('backgroundColor', bgColor); - } else { - var availableInnerWidth = availableWidth - paddingH; - var newWidth = availableWidth < oldOuterWidth // Current text is too wide, use `availableWidth` as max width. - ? availableInnerWidth : // Current available width is enough, but the text may have - // already been wrapped with a smaller available width. - forceRecalculate ? availableInnerWidth > layout.unconstrainedWidth // Current available is larger than text width, - // so don't constrain width (otherwise it may have - // empty space in the background). - ? null // Current available is smaller than text width, so - // use the current available width as constraining - // width. - : availableInnerWidth : // Current available width is enough, so no need to - // constrain. - null; - label.setStyle('width', newWidth); - } - - var newRect = label.getBoundingRect(); - textRect.width = newRect.width; - var margin = (label.style.margin || 0) + 2.1; - textRect.height = newRect.height + margin; - textRect.y -= (textRect.height - oldHeight) / 2; - } - } - - function isPositionCenter(sectorShape) { - // Not change x for center label - return sectorShape.position === 'center'; - } - - function pieLabelLayout(seriesModel) { - var data = seriesModel.getData(); - var labelLayoutList = []; - var cx; - var cy; - var hasLabelRotate = false; - var minShowLabelRadian = (seriesModel.get('minShowLabelAngle') || 0) * RADIAN$1; - var viewRect = data.getLayout('viewRect'); - var r = data.getLayout('r'); - var viewWidth = viewRect.width; - var viewLeft = viewRect.x; - var viewTop = viewRect.y; - var viewHeight = viewRect.height; - - function setNotShow(el) { - el.ignore = true; - } - - function isLabelShown(label) { - if (!label.ignore) { - return true; - } - - for (var key in label.states) { - if (label.states[key].ignore === false) { - return true; - } - } - - return false; - } - - data.each(function (idx) { - var sector = data.getItemGraphicEl(idx); - var sectorShape = sector.shape; - var label = sector.getTextContent(); - var labelLine = sector.getTextGuideLine(); - var itemModel = data.getItemModel(idx); - var labelModel = itemModel.getModel('label'); // Use position in normal or emphasis - - var labelPosition = labelModel.get('position') || itemModel.get(['emphasis', 'label', 'position']); - var labelDistance = labelModel.get('distanceToLabelLine'); - var labelAlignTo = labelModel.get('alignTo'); - var edgeDistance = parsePercent$1(labelModel.get('edgeDistance'), viewWidth); - var bleedMargin = labelModel.get('bleedMargin'); - var labelLineModel = itemModel.getModel('labelLine'); - var labelLineLen = labelLineModel.get('length'); - labelLineLen = parsePercent$1(labelLineLen, viewWidth); - var labelLineLen2 = labelLineModel.get('length2'); - labelLineLen2 = parsePercent$1(labelLineLen2, viewWidth); - - if (Math.abs(sectorShape.endAngle - sectorShape.startAngle) < minShowLabelRadian) { - each(label.states, setNotShow); - label.ignore = true; - - if (labelLine) { - each(labelLine.states, setNotShow); - labelLine.ignore = true; - } - - return; - } - - if (!isLabelShown(label)) { - return; - } - - var midAngle = (sectorShape.startAngle + sectorShape.endAngle) / 2; - var nx = Math.cos(midAngle); - var ny = Math.sin(midAngle); - var textX; - var textY; - var linePoints; - var textAlign; - cx = sectorShape.cx; - cy = sectorShape.cy; - var isLabelInside = labelPosition === 'inside' || labelPosition === 'inner'; - - if (labelPosition === 'center') { - textX = sectorShape.cx; - textY = sectorShape.cy; - textAlign = 'center'; - } else { - var x1 = (isLabelInside ? (sectorShape.r + sectorShape.r0) / 2 * nx : sectorShape.r * nx) + cx; - var y1 = (isLabelInside ? (sectorShape.r + sectorShape.r0) / 2 * ny : sectorShape.r * ny) + cy; - textX = x1 + nx * 3; - textY = y1 + ny * 3; - - if (!isLabelInside) { - // For roseType - var x2 = x1 + nx * (labelLineLen + r - sectorShape.r); - var y2 = y1 + ny * (labelLineLen + r - sectorShape.r); - var x3 = x2 + (nx < 0 ? -1 : 1) * labelLineLen2; - var y3 = y2; - - if (labelAlignTo === 'edge') { - // Adjust textX because text align of edge is opposite - textX = nx < 0 ? viewLeft + edgeDistance : viewLeft + viewWidth - edgeDistance; - } else { - textX = x3 + (nx < 0 ? -labelDistance : labelDistance); - } - - textY = y3; - linePoints = [[x1, y1], [x2, y2], [x3, y3]]; - } - - textAlign = isLabelInside ? 'center' : labelAlignTo === 'edge' ? nx > 0 ? 'right' : 'left' : nx > 0 ? 'left' : 'right'; - } - - var PI = Math.PI; - var labelRotate = 0; - var rotate = labelModel.get('rotate'); - - if (isNumber(rotate)) { - labelRotate = rotate * (PI / 180); - } else if (labelPosition === 'center') { - labelRotate = 0; - } else if (rotate === 'radial' || rotate === true) { - var radialAngle = nx < 0 ? -midAngle + PI : -midAngle; - labelRotate = radialAngle; - } else if (rotate === 'tangential' && labelPosition !== 'outside' && labelPosition !== 'outer') { - var rad = Math.atan2(nx, ny); - - if (rad < 0) { - rad = PI * 2 + rad; - } - - var isDown = ny > 0; - - if (isDown) { - rad = PI + rad; - } - - labelRotate = rad - PI; - } - - hasLabelRotate = !!labelRotate; - label.x = textX; - label.y = textY; - label.rotation = labelRotate; - label.setStyle({ - verticalAlign: 'middle' - }); // Not sectorShape the inside label - - if (!isLabelInside) { - var textRect = label.getBoundingRect().clone(); - textRect.applyTransform(label.getComputedTransform()); // Text has a default 1px stroke. Exclude this. - - var margin = (label.style.margin || 0) + 2.1; - textRect.y -= margin / 2; - textRect.height += margin; - labelLayoutList.push({ - label: label, - labelLine: labelLine, - position: labelPosition, - len: labelLineLen, - len2: labelLineLen2, - minTurnAngle: labelLineModel.get('minTurnAngle'), - maxSurfaceAngle: labelLineModel.get('maxSurfaceAngle'), - surfaceNormal: new Point(nx, ny), - linePoints: linePoints, - textAlign: textAlign, - labelDistance: labelDistance, - labelAlignTo: labelAlignTo, - edgeDistance: edgeDistance, - bleedMargin: bleedMargin, - rect: textRect, - unconstrainedWidth: textRect.width, - labelStyleWidth: label.style.width - }); - } else { - label.setStyle({ - align: textAlign - }); - var selectState = label.states.select; - - if (selectState) { - selectState.x += label.x; - selectState.y += label.y; - } - } - - sector.setTextConfig({ - inside: isLabelInside - }); - }); - - if (!hasLabelRotate && seriesModel.get('avoidLabelOverlap')) { - avoidOverlap(labelLayoutList, cx, cy, r, viewWidth, viewHeight, viewLeft, viewTop); - } - - for (var i = 0; i < labelLayoutList.length; i++) { - var layout = labelLayoutList[i]; - var label = layout.label; - var labelLine = layout.labelLine; - var notShowLabel = isNaN(label.x) || isNaN(label.y); - - if (label) { - label.setStyle({ - align: layout.textAlign - }); - - if (notShowLabel) { - each(label.states, setNotShow); - label.ignore = true; - } - - var selectState = label.states.select; - - if (selectState) { - selectState.x += label.x; - selectState.y += label.y; - } - } - - if (labelLine) { - var linePoints = layout.linePoints; - - if (notShowLabel || !linePoints) { - each(labelLine.states, setNotShow); - labelLine.ignore = true; - } else { - limitTurnAngle(linePoints, layout.minTurnAngle); - limitSurfaceAngle(linePoints, layout.surfaceNormal, layout.maxSurfaceAngle); - labelLine.setShape({ - points: linePoints - }); // Set the anchor to the midpoint of sector - - label.__hostTarget.textGuideLineConfig = { - anchor: new Point(linePoints[0][0], linePoints[0][1]) - }; - } - } - } - } - - /** - * Piece of pie including Sector, Label, LabelLine - */ - - var PiePiece = - /** @class */ - function (_super) { - __extends(PiePiece, _super); - - function PiePiece(data, idx, startAngle) { - var _this = _super.call(this) || this; - - _this.z2 = 2; - var text = new ZRText(); - - _this.setTextContent(text); - - _this.updateData(data, idx, startAngle, true); - - return _this; - } - - PiePiece.prototype.updateData = function (data, idx, startAngle, firstCreate) { - var sector = this; - var seriesModel = data.hostModel; - var itemModel = data.getItemModel(idx); - var emphasisModel = itemModel.getModel('emphasis'); - var layout = data.getItemLayout(idx); // cornerRadius & innerCornerRadius doesn't exist in the item layout. Use `0` if null value is specified. - // see `setItemLayout` in `pieLayout.ts`. - - var sectorShape = extend(getSectorCornerRadius(itemModel.getModel('itemStyle'), layout, true), layout); // Ignore NaN data. - - if (isNaN(sectorShape.startAngle)) { - // Use NaN shape to avoid drawing shape. - sector.setShape(sectorShape); - return; - } - - if (firstCreate) { - sector.setShape(sectorShape); - var animationType = seriesModel.getShallow('animationType'); - - if (seriesModel.ecModel.ssr) { - // Use scale animation in SSR mode(opacity?) - // Because CSS SVG animation doesn't support very customized shape animation. - initProps(sector, { - scaleX: 0, - scaleY: 0 - }, seriesModel, { - dataIndex: idx, - isFrom: true - }); - sector.originX = sectorShape.cx; - sector.originY = sectorShape.cy; - } else if (animationType === 'scale') { - sector.shape.r = layout.r0; - initProps(sector, { - shape: { - r: layout.r - } - }, seriesModel, idx); - } // Expansion - else { - if (startAngle != null) { - sector.setShape({ - startAngle: startAngle, - endAngle: startAngle - }); - initProps(sector, { - shape: { - startAngle: layout.startAngle, - endAngle: layout.endAngle - } - }, seriesModel, idx); - } else { - sector.shape.endAngle = layout.startAngle; - updateProps(sector, { - shape: { - endAngle: layout.endAngle - } - }, seriesModel, idx); - } - } - } else { - saveOldStyle(sector); // Transition animation from the old shape - - updateProps(sector, { - shape: sectorShape - }, seriesModel, idx); - } - - sector.useStyle(data.getItemVisual(idx, 'style')); - setStatesStylesFromModel(sector, itemModel); - var midAngle = (layout.startAngle + layout.endAngle) / 2; - var offset = seriesModel.get('selectedOffset'); - var dx = Math.cos(midAngle) * offset; - var dy = Math.sin(midAngle) * offset; - var cursorStyle = itemModel.getShallow('cursor'); - cursorStyle && sector.attr('cursor', cursorStyle); - - this._updateLabel(seriesModel, data, idx); - - sector.ensureState('emphasis').shape = extend({ - r: layout.r + (emphasisModel.get('scale') ? emphasisModel.get('scaleSize') || 0 : 0) - }, getSectorCornerRadius(emphasisModel.getModel('itemStyle'), layout)); - extend(sector.ensureState('select'), { - x: dx, - y: dy, - shape: getSectorCornerRadius(itemModel.getModel(['select', 'itemStyle']), layout) - }); - extend(sector.ensureState('blur'), { - shape: getSectorCornerRadius(itemModel.getModel(['blur', 'itemStyle']), layout) - }); - var labelLine = sector.getTextGuideLine(); - var labelText = sector.getTextContent(); - labelLine && extend(labelLine.ensureState('select'), { - x: dx, - y: dy - }); // TODO: needs dx, dy in zrender? - - extend(labelText.ensureState('select'), { - x: dx, - y: dy - }); - toggleHoverEmphasis(this, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - }; - - PiePiece.prototype._updateLabel = function (seriesModel, data, idx) { - var sector = this; - var itemModel = data.getItemModel(idx); - var labelLineModel = itemModel.getModel('labelLine'); - var style = data.getItemVisual(idx, 'style'); - var visualColor = style && style.fill; - var visualOpacity = style && style.opacity; - setLabelStyle(sector, getLabelStatesModels(itemModel), { - labelFetcher: data.hostModel, - labelDataIndex: idx, - inheritColor: visualColor, - defaultOpacity: visualOpacity, - defaultText: seriesModel.getFormattedLabel(idx, 'normal') || data.getName(idx) - }); - var labelText = sector.getTextContent(); // Set textConfig on sector. - - sector.setTextConfig({ - // reset position, rotation - position: null, - rotation: null - }); // Make sure update style on labelText after setLabelStyle. - // Because setLabelStyle will replace a new style on it. - - labelText.attr({ - z2: 10 - }); - var labelPosition = seriesModel.get(['label', 'position']); - - if (labelPosition !== 'outside' && labelPosition !== 'outer') { - sector.removeTextGuideLine(); - } else { - var polyline = this.getTextGuideLine(); - - if (!polyline) { - polyline = new Polyline(); - this.setTextGuideLine(polyline); - } // Default use item visual color - - - setLabelLineStyle(this, getLabelLineStatesModels(itemModel), { - stroke: visualColor, - opacity: retrieve3(labelLineModel.get(['lineStyle', 'opacity']), visualOpacity, 1) - }); - } - }; - - return PiePiece; - }(Sector); // Pie view - - - var PieView = - /** @class */ - function (_super) { - __extends(PieView, _super); - - function PieView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.ignoreLabelLineUpdate = true; - return _this; - } - - PieView.prototype.render = function (seriesModel, ecModel, api, payload) { - var data = seriesModel.getData(); - var oldData = this._data; - var group = this.group; - var startAngle; // First render - - if (!oldData && data.count() > 0) { - var shape = data.getItemLayout(0); - - for (var s = 1; isNaN(shape && shape.startAngle) && s < data.count(); ++s) { - shape = data.getItemLayout(s); - } - - if (shape) { - startAngle = shape.startAngle; - } - } // remove empty-circle if it exists - - - if (this._emptyCircleSector) { - group.remove(this._emptyCircleSector); - } // when all data are filtered, show lightgray empty circle - - - if (data.count() === 0 && seriesModel.get('showEmptyCircle')) { - var sector = new Sector({ - shape: getBasicPieLayout(seriesModel, api) - }); - sector.useStyle(seriesModel.getModel('emptyCircleStyle').getItemStyle()); - this._emptyCircleSector = sector; - group.add(sector); - } - - data.diff(oldData).add(function (idx) { - var piePiece = new PiePiece(data, idx, startAngle); - data.setItemGraphicEl(idx, piePiece); - group.add(piePiece); - }).update(function (newIdx, oldIdx) { - var piePiece = oldData.getItemGraphicEl(oldIdx); - piePiece.updateData(data, newIdx, startAngle); - piePiece.off('click'); - group.add(piePiece); - data.setItemGraphicEl(newIdx, piePiece); - }).remove(function (idx) { - var piePiece = oldData.getItemGraphicEl(idx); - removeElementWithFadeOut(piePiece, seriesModel, idx); - }).execute(); - pieLabelLayout(seriesModel); // Always use initial animation. - - if (seriesModel.get('animationTypeUpdate') !== 'expansion') { - this._data = data; - } - }; - - PieView.prototype.dispose = function () {}; - - PieView.prototype.containPoint = function (point, seriesModel) { - var data = seriesModel.getData(); - var itemLayout = data.getItemLayout(0); - - if (itemLayout) { - var dx = point[0] - itemLayout.cx; - var dy = point[1] - itemLayout.cy; - var radius = Math.sqrt(dx * dx + dy * dy); - return radius <= itemLayout.r && radius >= itemLayout.r0; - } - }; - - PieView.type = 'pie'; - return PieView; - }(ChartView); - - /** - * [Usage]: - * (1) - * createListSimply(seriesModel, ['value']); - * (2) - * createListSimply(seriesModel, { - * coordDimensions: ['value'], - * dimensionsCount: 5 - * }); - */ - - function createSeriesDataSimply(seriesModel, opt, nameList) { - opt = isArray(opt) && { - coordDimensions: opt - } || extend({ - encodeDefine: seriesModel.getEncode() - }, opt); - var source = seriesModel.getSource(); - var dimensions = prepareSeriesDataSchema(source, opt).dimensions; - var list = new SeriesData(dimensions, seriesModel); - list.initData(source, nameList); - return list; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - /** - * LegendVisualProvider is an bridge that pick encoded color from data and - * provide to the legend component. - */ - var LegendVisualProvider = - /** @class */ - function () { - function LegendVisualProvider( // Function to get data after filtered. It stores all the encoding info - getDataWithEncodedVisual, // Function to get raw data before filtered. - getRawData) { - this._getDataWithEncodedVisual = getDataWithEncodedVisual; - this._getRawData = getRawData; - } - - LegendVisualProvider.prototype.getAllNames = function () { - var rawData = this._getRawData(); // We find the name from the raw data. In case it's filtered by the legend component. - // Normally, the name can be found in rawData, but can't be found in filtered data will display as gray. - - - return rawData.mapArray(rawData.getName); - }; - - LegendVisualProvider.prototype.containName = function (name) { - var rawData = this._getRawData(); - - return rawData.indexOfName(name) >= 0; - }; - - LegendVisualProvider.prototype.indexOfName = function (name) { - // Only get data when necessary. - // Because LegendVisualProvider constructor may be new in the stage that data is not prepared yet. - // Invoking Series#getData immediately will throw an error. - var dataWithEncodedVisual = this._getDataWithEncodedVisual(); - - return dataWithEncodedVisual.indexOfName(name); - }; - - LegendVisualProvider.prototype.getItemVisual = function (dataIndex, key) { - // Get encoded visual properties from final filtered data. - var dataWithEncodedVisual = this._getDataWithEncodedVisual(); - - return dataWithEncodedVisual.getItemVisual(dataIndex, key); - }; - - return LegendVisualProvider; - }(); - - var innerData = makeInner(); - - var PieSeriesModel = - /** @class */ - function (_super) { - __extends(PieSeriesModel, _super); - - function PieSeriesModel() { - return _super !== null && _super.apply(this, arguments) || this; - } - /** - * @overwrite - */ - - - PieSeriesModel.prototype.init = function (option) { - _super.prototype.init.apply(this, arguments); // Enable legend selection for each data item - // Use a function instead of direct access because data reference may changed - - - this.legendVisualProvider = new LegendVisualProvider(bind(this.getData, this), bind(this.getRawData, this)); - - this._defaultLabelLine(option); - }; - /** - * @overwrite - */ - - - PieSeriesModel.prototype.mergeOption = function () { - _super.prototype.mergeOption.apply(this, arguments); - }; - /** - * @overwrite - */ - - - PieSeriesModel.prototype.getInitialData = function () { - return createSeriesDataSimply(this, { - coordDimensions: ['value'], - encodeDefaulter: curry(makeSeriesEncodeForNameBased, this) - }); - }; - /** - * @overwrite - */ - - - PieSeriesModel.prototype.getDataParams = function (dataIndex) { - var data = this.getData(); // update seats when data is changed - - var dataInner = innerData(data); - var seats = dataInner.seats; - - if (!seats) { - var valueList_1 = []; - data.each(data.mapDimension('value'), function (value) { - valueList_1.push(value); - }); - seats = dataInner.seats = getPercentSeats(valueList_1, data.hostModel.get('percentPrecision')); - } - - var params = _super.prototype.getDataParams.call(this, dataIndex); // seats may be empty when sum is 0 - - - params.percent = seats[dataIndex] || 0; - params.$vars.push('percent'); - return params; - }; - - PieSeriesModel.prototype._defaultLabelLine = function (option) { - // Extend labelLine emphasis - defaultEmphasis(option, 'labelLine', ['show']); - var labelLineNormalOpt = option.labelLine; - var labelLineEmphasisOpt = option.emphasis.labelLine; // Not show label line if `label.normal.show = false` - - labelLineNormalOpt.show = labelLineNormalOpt.show && option.label.show; - labelLineEmphasisOpt.show = labelLineEmphasisOpt.show && option.emphasis.label.show; - }; - - PieSeriesModel.type = 'series.pie'; - PieSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - legendHoverLink: true, - colorBy: 'data', - // 默认全局居中 - center: ['50%', '50%'], - radius: [0, '75%'], - // 默认顺时针 - clockwise: true, - startAngle: 90, - // 最小角度改为0 - minAngle: 0, - // If the angle of a sector less than `minShowLabelAngle`, - // the label will not be displayed. - minShowLabelAngle: 0, - // 选中时扇区偏移量 - selectedOffset: 10, - // 选择模式,默认关闭,可选single,multiple - // selectedMode: false, - // 南丁格尔玫瑰图模式,'radius'(半径) | 'area'(面积) - // roseType: null, - percentPrecision: 2, - // If still show when all data zero. - stillShowZeroSum: true, - // cursor: null, - left: 0, - top: 0, - right: 0, - bottom: 0, - width: null, - height: null, - label: { - // color: 'inherit', - // If rotate around circle - rotate: 0, - show: true, - overflow: 'truncate', - // 'outer', 'inside', 'center' - position: 'outer', - // 'none', 'labelLine', 'edge'. Works only when position is 'outer' - alignTo: 'none', - // Closest distance between label and chart edge. - // Works only position is 'outer' and alignTo is 'edge'. - edgeDistance: '25%', - // Works only position is 'outer' and alignTo is not 'edge'. - bleedMargin: 10, - // Distance between text and label line. - distanceToLabelLine: 5 // formatter: 标签文本格式器,同 tooltip.formatter,不支持异步回调 - // 默认使用全局文本样式,详见 textStyle - // distance: 当position为inner时有效,为label位置到圆心的距离与圆半径(环状图为内外半径和)的比例系数 - - }, - // Enabled when label.normal.position is 'outer' - labelLine: { - show: true, - // 引导线两段中的第一段长度 - length: 15, - // 引导线两段中的第二段长度 - length2: 15, - smooth: false, - minTurnAngle: 90, - maxSurfaceAngle: 90, - lineStyle: { - // color: 各异, - width: 1, - type: 'solid' - } - }, - itemStyle: { - borderWidth: 1, - borderJoin: 'round' - }, - showEmptyCircle: true, - emptyCircleStyle: { - color: 'lightgray', - opacity: 1 - }, - labelLayout: { - // Hide the overlapped label. - hideOverlap: true - }, - emphasis: { - scale: true, - scaleSize: 5 - }, - // If use strategy to avoid label overlapping - avoidLabelOverlap: true, - // Animation type. Valid values: expansion, scale - animationType: 'expansion', - animationDuration: 1000, - // Animation type when update. Valid values: transition, expansion - animationTypeUpdate: 'transition', - animationEasingUpdate: 'cubicInOut', - animationDurationUpdate: 500, - animationEasing: 'cubicInOut' - }; - return PieSeriesModel; - }(SeriesModel); - - function negativeDataFilter(seriesType) { - return { - seriesType: seriesType, - reset: function (seriesModel, ecModel) { - var data = seriesModel.getData(); - data.filterSelf(function (idx) { - // handle negative value condition - var valueDim = data.mapDimension('value'); - var curValue = data.get(valueDim, idx); - - if (isNumber(curValue) && !isNaN(curValue) && curValue < 0) { - return false; - } - - return true; - }); - } - }; - } - - function install$4(registers) { - registers.registerChartView(PieView); - registers.registerSeriesModel(PieSeriesModel); - createLegacyDataSelectAction('pie', registers.registerAction); - registers.registerLayout(curry(pieLayout, 'pie')); - registers.registerProcessor(dataFilter('pie')); - registers.registerProcessor(negativeDataFilter('pie')); - } - - var ScatterSeriesModel = - /** @class */ - function (_super) { - __extends(ScatterSeriesModel, _super); - - function ScatterSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ScatterSeriesModel.type; - _this.hasSymbolVisual = true; - return _this; - } - - ScatterSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesData(null, this, { - useEncodeDefaulter: true - }); - }; - - ScatterSeriesModel.prototype.getProgressive = function () { - var progressive = this.option.progressive; - - if (progressive == null) { - // PENDING - return this.option.large ? 5e3 : this.get('progressive'); - } - - return progressive; - }; - - ScatterSeriesModel.prototype.getProgressiveThreshold = function () { - var progressiveThreshold = this.option.progressiveThreshold; - - if (progressiveThreshold == null) { - // PENDING - return this.option.large ? 1e4 : this.get('progressiveThreshold'); - } - - return progressiveThreshold; - }; - - ScatterSeriesModel.prototype.brushSelector = function (dataIndex, data, selectors) { - return selectors.point(data.getItemLayout(dataIndex)); - }; - - ScatterSeriesModel.prototype.getZLevelKey = function () { - // Each progressive series has individual key. - return this.getData().count() > this.getProgressiveThreshold() ? this.id : ''; - }; - - ScatterSeriesModel.type = 'series.scatter'; - ScatterSeriesModel.dependencies = ['grid', 'polar', 'geo', 'singleAxis', 'calendar']; - ScatterSeriesModel.defaultOption = { - coordinateSystem: 'cartesian2d', - // zlevel: 0, - z: 2, - legendHoverLink: true, - symbolSize: 10, - // symbolRotate: null, // 图形旋转控制 - large: false, - // Available when large is true - largeThreshold: 2000, - // cursor: null, - itemStyle: { - opacity: 0.8 // color: 各异 - - }, - emphasis: { - scale: true - }, - // If clip the overflow graphics - // Works on cartesian / polar series - clip: true, - select: { - itemStyle: { - borderColor: '#212121' - } - }, - universalTransition: { - divideShape: 'clone' - } // progressive: null - - }; - return ScatterSeriesModel; - }(SeriesModel); - - var BOOST_SIZE_THRESHOLD = 4; - - var LargeSymbolPathShape = - /** @class */ - function () { - function LargeSymbolPathShape() {} - - return LargeSymbolPathShape; - }(); - - var LargeSymbolPath = - /** @class */ - function (_super) { - __extends(LargeSymbolPath, _super); - - function LargeSymbolPath(opts) { - var _this = _super.call(this, opts) || this; - - _this._off = 0; - _this.hoverDataIdx = -1; - return _this; - } - - LargeSymbolPath.prototype.getDefaultShape = function () { - return new LargeSymbolPathShape(); - }; - - LargeSymbolPath.prototype.reset = function () { - this.notClear = false; - this._off = 0; - }; - - LargeSymbolPath.prototype.buildPath = function (path, shape) { - var points = shape.points; - var size = shape.size; - var symbolProxy = this.symbolProxy; - var symbolProxyShape = symbolProxy.shape; - var ctx = path.getContext ? path.getContext() : path; - var canBoost = ctx && size[0] < BOOST_SIZE_THRESHOLD; - var softClipShape = this.softClipShape; - var i; // Do draw in afterBrush. - - if (canBoost) { - this._ctx = ctx; - return; - } - - this._ctx = null; - - for (i = this._off; i < points.length;) { - var x = points[i++]; - var y = points[i++]; - - if (isNaN(x) || isNaN(y)) { - continue; - } - - if (softClipShape && !softClipShape.contain(x, y)) { - continue; - } - - symbolProxyShape.x = x - size[0] / 2; - symbolProxyShape.y = y - size[1] / 2; - symbolProxyShape.width = size[0]; - symbolProxyShape.height = size[1]; - symbolProxy.buildPath(path, symbolProxyShape, true); - } - - if (this.incremental) { - this._off = i; - this.notClear = true; - } - }; - - LargeSymbolPath.prototype.afterBrush = function () { - var shape = this.shape; - var points = shape.points; - var size = shape.size; - var ctx = this._ctx; - var softClipShape = this.softClipShape; - var i; - - if (!ctx) { - return; - } // PENDING If style or other canvas status changed? - - - for (i = this._off; i < points.length;) { - var x = points[i++]; - var y = points[i++]; - - if (isNaN(x) || isNaN(y)) { - continue; - } - - if (softClipShape && !softClipShape.contain(x, y)) { - continue; - } // fillRect is faster than building a rect path and draw. - // And it support light globalCompositeOperation. - - - ctx.fillRect(x - size[0] / 2, y - size[1] / 2, size[0], size[1]); - } - - if (this.incremental) { - this._off = i; - this.notClear = true; - } - }; - - LargeSymbolPath.prototype.findDataIndex = function (x, y) { - // TODO ??? - // Consider transform - var shape = this.shape; - var points = shape.points; - var size = shape.size; - var w = Math.max(size[0], 4); - var h = Math.max(size[1], 4); // Not consider transform - // Treat each element as a rect - // top down traverse - - for (var idx = points.length / 2 - 1; idx >= 0; idx--) { - var i = idx * 2; - var x0 = points[i] - w / 2; - var y0 = points[i + 1] - h / 2; - - if (x >= x0 && y >= y0 && x <= x0 + w && y <= y0 + h) { - return idx; - } - } - - return -1; - }; - - LargeSymbolPath.prototype.contain = function (x, y) { - var localPos = this.transformCoordToLocal(x, y); - var rect = this.getBoundingRect(); - x = localPos[0]; - y = localPos[1]; - - if (rect.contain(x, y)) { - // Cache found data index. - var dataIdx = this.hoverDataIdx = this.findDataIndex(x, y); - return dataIdx >= 0; - } - - this.hoverDataIdx = -1; - return false; - }; - - LargeSymbolPath.prototype.getBoundingRect = function () { - // Ignore stroke for large symbol draw. - var rect = this._rect; - - if (!rect) { - var shape = this.shape; - var points = shape.points; - var size = shape.size; - var w = size[0]; - var h = size[1]; - var minX = Infinity; - var minY = Infinity; - var maxX = -Infinity; - var maxY = -Infinity; - - for (var i = 0; i < points.length;) { - var x = points[i++]; - var y = points[i++]; - minX = Math.min(x, minX); - maxX = Math.max(x, maxX); - minY = Math.min(y, minY); - maxY = Math.max(y, maxY); - } - - rect = this._rect = new BoundingRect(minX - w / 2, minY - h / 2, maxX - minX + w, maxY - minY + h); - } - - return rect; - }; - - return LargeSymbolPath; - }(Path); - - var LargeSymbolDraw = - /** @class */ - function () { - function LargeSymbolDraw() { - this.group = new Group(); - } - /** - * Update symbols draw by new data - */ - - - LargeSymbolDraw.prototype.updateData = function (data, opt) { - this._clear(); - - var symbolEl = this._create(); - - symbolEl.setShape({ - points: data.getLayout('points') - }); - - this._setCommon(symbolEl, data, opt); - }; - - LargeSymbolDraw.prototype.updateLayout = function (data) { - var points = data.getLayout('points'); - this.group.eachChild(function (child) { - if (child.startIndex != null) { - var len = (child.endIndex - child.startIndex) * 2; - var byteOffset = child.startIndex * 4 * 2; - points = new Float32Array(points.buffer, byteOffset, len); - } - - child.setShape('points', points); // Reset draw cursor. - - child.reset(); - }); - }; - - LargeSymbolDraw.prototype.incrementalPrepareUpdate = function (data) { - this._clear(); - }; - - LargeSymbolDraw.prototype.incrementalUpdate = function (taskParams, data, opt) { - var lastAdded = this._newAdded[0]; - var points = data.getLayout('points'); - var oldPoints = lastAdded && lastAdded.shape.points; // Merging the exists. Each element has 1e4 points. - // Consider the performance balance between too much elements and too much points in one shape(may affect hover optimization) - - if (oldPoints && oldPoints.length < 2e4) { - var oldLen = oldPoints.length; - var newPoints = new Float32Array(oldLen + points.length); // Concat two array - - newPoints.set(oldPoints); - newPoints.set(points, oldLen); // Update endIndex - - lastAdded.endIndex = taskParams.end; - lastAdded.setShape({ - points: newPoints - }); - } else { - // Clear - this._newAdded = []; - - var symbolEl = this._create(); - - symbolEl.startIndex = taskParams.start; - symbolEl.endIndex = taskParams.end; - symbolEl.incremental = true; - symbolEl.setShape({ - points: points - }); - - this._setCommon(symbolEl, data, opt); - } - }; - - LargeSymbolDraw.prototype.eachRendered = function (cb) { - this._newAdded[0] && cb(this._newAdded[0]); - }; - - LargeSymbolDraw.prototype._create = function () { - var symbolEl = new LargeSymbolPath({ - cursor: 'default' - }); - symbolEl.ignoreCoarsePointer = true; - this.group.add(symbolEl); - - this._newAdded.push(symbolEl); - - return symbolEl; - }; - - LargeSymbolDraw.prototype._setCommon = function (symbolEl, data, opt) { - var hostModel = data.hostModel; - opt = opt || {}; - var size = data.getVisual('symbolSize'); - symbolEl.setShape('size', size instanceof Array ? size : [size, size]); - symbolEl.softClipShape = opt.clipShape || null; // Create symbolProxy to build path for each data - - symbolEl.symbolProxy = createSymbol(data.getVisual('symbol'), 0, 0, 0, 0); // Use symbolProxy setColor method - - symbolEl.setColor = symbolEl.symbolProxy.setColor; - var extrudeShadow = symbolEl.shape.size[0] < BOOST_SIZE_THRESHOLD; - symbolEl.useStyle( // Draw shadow when doing fillRect is extremely slow. - hostModel.getModel('itemStyle').getItemStyle(extrudeShadow ? ['color', 'shadowBlur', 'shadowColor'] : ['color'])); - var globalStyle = data.getVisual('style'); - var visualColor = globalStyle && globalStyle.fill; - - if (visualColor) { - symbolEl.setColor(visualColor); - } - - var ecData = getECData(symbolEl); // Enable tooltip - // PENDING May have performance issue when path is extremely large - - ecData.seriesIndex = hostModel.seriesIndex; - symbolEl.on('mousemove', function (e) { - ecData.dataIndex = null; - var dataIndex = symbolEl.hoverDataIdx; - - if (dataIndex >= 0) { - // Provide dataIndex for tooltip - ecData.dataIndex = dataIndex + (symbolEl.startIndex || 0); - } - }); - }; - - LargeSymbolDraw.prototype.remove = function () { - this._clear(); - }; - - LargeSymbolDraw.prototype._clear = function () { - this._newAdded = []; - this.group.removeAll(); - }; - - return LargeSymbolDraw; - }(); - - var ScatterView = - /** @class */ - function (_super) { - __extends(ScatterView, _super); - - function ScatterView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ScatterView.type; - return _this; - } - - ScatterView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - - var symbolDraw = this._updateSymbolDraw(data, seriesModel); - - symbolDraw.updateData(data, { - // TODO - // If this parameter should be a shape or a bounding volume - // shape will be more general. - // But bounding volume like bounding rect will be much faster in the contain calculation - clipShape: this._getClipShape(seriesModel) - }); - this._finished = true; - }; - - ScatterView.prototype.incrementalPrepareRender = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - - var symbolDraw = this._updateSymbolDraw(data, seriesModel); - - symbolDraw.incrementalPrepareUpdate(data); - this._finished = false; - }; - - ScatterView.prototype.incrementalRender = function (taskParams, seriesModel, ecModel) { - this._symbolDraw.incrementalUpdate(taskParams, seriesModel.getData(), { - clipShape: this._getClipShape(seriesModel) - }); - - this._finished = taskParams.end === seriesModel.getData().count(); - }; - - ScatterView.prototype.updateTransform = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); // Must mark group dirty and make sure the incremental layer will be cleared - // PENDING - - this.group.dirty(); - - if (!this._finished || data.count() > 1e4) { - return { - update: true - }; - } else { - var res = pointsLayout('').reset(seriesModel, ecModel, api); - - if (res.progress) { - res.progress({ - start: 0, - end: data.count(), - count: data.count() - }, data); - } - - this._symbolDraw.updateLayout(data); - } - }; - - ScatterView.prototype.eachRendered = function (cb) { - this._symbolDraw && this._symbolDraw.eachRendered(cb); - }; - - ScatterView.prototype._getClipShape = function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - var clipArea = coordSys && coordSys.getArea && coordSys.getArea(); - return seriesModel.get('clip', true) ? clipArea : null; - }; - - ScatterView.prototype._updateSymbolDraw = function (data, seriesModel) { - var symbolDraw = this._symbolDraw; - var pipelineContext = seriesModel.pipelineContext; - var isLargeDraw = pipelineContext.large; - - if (!symbolDraw || isLargeDraw !== this._isLargeDraw) { - symbolDraw && symbolDraw.remove(); - symbolDraw = this._symbolDraw = isLargeDraw ? new LargeSymbolDraw() : new SymbolDraw(); - this._isLargeDraw = isLargeDraw; - this.group.removeAll(); - } - - this.group.add(symbolDraw.group); - return symbolDraw; - }; - - ScatterView.prototype.remove = function (ecModel, api) { - this._symbolDraw && this._symbolDraw.remove(true); - this._symbolDraw = null; - }; - - ScatterView.prototype.dispose = function () {}; - - ScatterView.type = 'scatter'; - return ScatterView; - }(ChartView); - - var GridModel = - /** @class */ - function (_super) { - __extends(GridModel, _super); - - function GridModel() { - return _super !== null && _super.apply(this, arguments) || this; - } - - GridModel.type = 'grid'; - GridModel.dependencies = ['xAxis', 'yAxis']; - GridModel.layoutMode = 'box'; - GridModel.defaultOption = { - show: false, - // zlevel: 0, - z: 0, - left: '10%', - top: 60, - right: '10%', - bottom: 70, - // If grid size contain label - containLabel: false, - // width: {totalWidth} - left - right, - // height: {totalHeight} - top - bottom, - backgroundColor: 'rgba(0,0,0,0)', - borderWidth: 1, - borderColor: '#ccc' - }; - return GridModel; - }(ComponentModel); - - var CartesianAxisModel = - /** @class */ - function (_super) { - __extends(CartesianAxisModel, _super); - - function CartesianAxisModel() { - return _super !== null && _super.apply(this, arguments) || this; - } - - CartesianAxisModel.prototype.getCoordSysModel = function () { - return this.getReferringComponents('grid', SINGLE_REFERRING).models[0]; - }; - - CartesianAxisModel.type = 'cartesian2dAxis'; - return CartesianAxisModel; - }(ComponentModel); - mixin(CartesianAxisModel, AxisModelCommonMixin); - - var defaultOption = { - show: true, - // zlevel: 0, - z: 0, - // Inverse the axis. - inverse: false, - // Axis name displayed. - name: '', - // 'start' | 'middle' | 'end' - nameLocation: 'end', - // By degree. By default auto rotate by nameLocation. - nameRotate: null, - nameTruncate: { - maxWidth: null, - ellipsis: '...', - placeholder: '.' - }, - // Use global text style by default. - nameTextStyle: {}, - // The gap between axisName and axisLine. - nameGap: 15, - // Default `false` to support tooltip. - silent: false, - // Default `false` to avoid legacy user event listener fail. - triggerEvent: false, - tooltip: { - show: false - }, - axisPointer: {}, - axisLine: { - show: true, - onZero: true, - onZeroAxisIndex: null, - lineStyle: { - color: '#6E7079', - width: 1, - type: 'solid' - }, - // The arrow at both ends the the axis. - symbol: ['none', 'none'], - symbolSize: [10, 15] - }, - axisTick: { - show: true, - // Whether axisTick is inside the grid or outside the grid. - inside: false, - // The length of axisTick. - length: 5, - lineStyle: { - width: 1 - } - }, - axisLabel: { - show: true, - // Whether axisLabel is inside the grid or outside the grid. - inside: false, - rotate: 0, - // true | false | null/undefined (auto) - showMinLabel: null, - // true | false | null/undefined (auto) - showMaxLabel: null, - margin: 8, - // formatter: null, - fontSize: 12 - }, - splitLine: { - show: true, - lineStyle: { - color: ['#E0E6F1'], - width: 1, - type: 'solid' - } - }, - splitArea: { - show: false, - areaStyle: { - color: ['rgba(250,250,250,0.2)', 'rgba(210,219,238,0.2)'] - } - } - }; - var categoryAxis = merge({ - // The gap at both ends of the axis. For categoryAxis, boolean. - boundaryGap: true, - // Set false to faster category collection. - deduplication: null, - // splitArea: { - // show: false - // }, - splitLine: { - show: false - }, - axisTick: { - // If tick is align with label when boundaryGap is true - alignWithLabel: false, - interval: 'auto' - }, - axisLabel: { - interval: 'auto' - } - }, defaultOption); - var valueAxis = merge({ - boundaryGap: [0, 0], - axisLine: { - // Not shown when other axis is categoryAxis in cartesian - show: 'auto' - }, - axisTick: { - // Not shown when other axis is categoryAxis in cartesian - show: 'auto' - }, - // TODO - // min/max: [30, datamin, 60] or [20, datamin] or [datamin, 60] - splitNumber: 5, - minorTick: { - // Minor tick, not available for cateogry axis. - show: false, - // Split number of minor ticks. The value should be in range of (0, 100) - splitNumber: 5, - // Length of minor tick - length: 3, - // Line style - lineStyle: {// Default to be same with axisTick - } - }, - minorSplitLine: { - show: false, - lineStyle: { - color: '#F4F7FD', - width: 1 - } - } - }, defaultOption); - var timeAxis = merge({ - splitNumber: 6, - axisLabel: { - // To eliminate labels that are not nice - showMinLabel: false, - showMaxLabel: false, - rich: { - primary: { - fontWeight: 'bold' - } - } - }, - splitLine: { - show: false - } - }, valueAxis); - var logAxis = defaults({ - logBase: 10 - }, valueAxis); - var axisDefault = { - category: categoryAxis, - value: valueAxis, - time: timeAxis, - log: logAxis - }; - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var AXIS_TYPES = { - value: 1, - category: 1, - time: 1, - log: 1 - }; - - /** - * Generate sub axis model class - * @param axisName 'x' 'y' 'radius' 'angle' 'parallel' ... - */ - - function axisModelCreator(registers, axisName, BaseAxisModelClass, extraDefaultOption) { - each(AXIS_TYPES, function (v, axisType) { - var defaultOption = merge(merge({}, axisDefault[axisType], true), extraDefaultOption, true); - - var AxisModel = - /** @class */ - function (_super) { - __extends(AxisModel, _super); - - function AxisModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = axisName + 'Axis.' + axisType; - return _this; - } - - AxisModel.prototype.mergeDefaultAndTheme = function (option, ecModel) { - var layoutMode = fetchLayoutMode(this); - var inputPositionParams = layoutMode ? getLayoutParams(option) : {}; - var themeModel = ecModel.getTheme(); - merge(option, themeModel.get(axisType + 'Axis')); - merge(option, this.getDefaultOption()); - option.type = getAxisType(option); - - if (layoutMode) { - mergeLayoutParam(option, inputPositionParams, layoutMode); - } - }; - - AxisModel.prototype.optionUpdated = function () { - var thisOption = this.option; - - if (thisOption.type === 'category') { - this.__ordinalMeta = OrdinalMeta.createByAxisModel(this); - } - }; - /** - * Should not be called before all of 'getInitailData' finished. - * Because categories are collected during initializing data. - */ - - - AxisModel.prototype.getCategories = function (rawData) { - var option = this.option; // FIXME - // warning if called before all of 'getInitailData' finished. - - if (option.type === 'category') { - if (rawData) { - return option.data; - } - - return this.__ordinalMeta.categories; - } - }; - - AxisModel.prototype.getOrdinalMeta = function () { - return this.__ordinalMeta; - }; - - AxisModel.type = axisName + 'Axis.' + axisType; - AxisModel.defaultOption = defaultOption; - return AxisModel; - }(BaseAxisModelClass); - - registers.registerComponentModel(AxisModel); - }); - registers.registerSubTypeDefaulter(axisName + 'Axis', getAxisType); - } - - function getAxisType(option) { - // Default axis with data is category axis - return option.type || (option.data ? 'category' : 'value'); - } - - var Cartesian = - /** @class */ - function () { - function Cartesian(name) { - this.type = 'cartesian'; - this._dimList = []; - this._axes = {}; - this.name = name || ''; - } - - Cartesian.prototype.getAxis = function (dim) { - return this._axes[dim]; - }; - - Cartesian.prototype.getAxes = function () { - return map(this._dimList, function (dim) { - return this._axes[dim]; - }, this); - }; - - Cartesian.prototype.getAxesByScale = function (scaleType) { - scaleType = scaleType.toLowerCase(); - return filter(this.getAxes(), function (axis) { - return axis.scale.type === scaleType; - }); - }; - - Cartesian.prototype.addAxis = function (axis) { - var dim = axis.dim; - this._axes[dim] = axis; - - this._dimList.push(dim); - }; - - return Cartesian; - }(); - - var cartesian2DDimensions = ['x', 'y']; - - function canCalculateAffineTransform(scale) { - return scale.type === 'interval' || scale.type === 'time'; - } - - var Cartesian2D = - /** @class */ - function (_super) { - __extends(Cartesian2D, _super); - - function Cartesian2D() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'cartesian2d'; - _this.dimensions = cartesian2DDimensions; - return _this; - } - /** - * Calculate an affine transform matrix if two axes are time or value. - * It's mainly for accelartion on the large time series data. - */ - - - Cartesian2D.prototype.calcAffineTransform = function () { - this._transform = this._invTransform = null; - var xAxisScale = this.getAxis('x').scale; - var yAxisScale = this.getAxis('y').scale; - - if (!canCalculateAffineTransform(xAxisScale) || !canCalculateAffineTransform(yAxisScale)) { - return; - } - - var xScaleExtent = xAxisScale.getExtent(); - var yScaleExtent = yAxisScale.getExtent(); - var start = this.dataToPoint([xScaleExtent[0], yScaleExtent[0]]); - var end = this.dataToPoint([xScaleExtent[1], yScaleExtent[1]]); - var xScaleSpan = xScaleExtent[1] - xScaleExtent[0]; - var yScaleSpan = yScaleExtent[1] - yScaleExtent[0]; - - if (!xScaleSpan || !yScaleSpan) { - return; - } // Accelerate data to point calculation on the special large time series data. - - - var scaleX = (end[0] - start[0]) / xScaleSpan; - var scaleY = (end[1] - start[1]) / yScaleSpan; - var translateX = start[0] - xScaleExtent[0] * scaleX; - var translateY = start[1] - yScaleExtent[0] * scaleY; - var m = this._transform = [scaleX, 0, 0, scaleY, translateX, translateY]; - this._invTransform = invert([], m); - }; - /** - * Base axis will be used on stacking. - */ - - - Cartesian2D.prototype.getBaseAxis = function () { - return this.getAxesByScale('ordinal')[0] || this.getAxesByScale('time')[0] || this.getAxis('x'); - }; - - Cartesian2D.prototype.containPoint = function (point) { - var axisX = this.getAxis('x'); - var axisY = this.getAxis('y'); - return axisX.contain(axisX.toLocalCoord(point[0])) && axisY.contain(axisY.toLocalCoord(point[1])); - }; - - Cartesian2D.prototype.containData = function (data) { - return this.getAxis('x').containData(data[0]) && this.getAxis('y').containData(data[1]); - }; - - Cartesian2D.prototype.containZone = function (data1, data2) { - var zoneDiag1 = this.dataToPoint(data1); - var zoneDiag2 = this.dataToPoint(data2); - var area = this.getArea(); - var zone = new BoundingRect(zoneDiag1[0], zoneDiag1[1], zoneDiag2[0] - zoneDiag1[0], zoneDiag2[1] - zoneDiag1[1]); - return area.intersect(zone); - }; - - Cartesian2D.prototype.dataToPoint = function (data, clamp, out) { - out = out || []; - var xVal = data[0]; - var yVal = data[1]; // Fast path - - if (this._transform // It's supported that if data is like `[Inifity, 123]`, where only Y pixel calculated. - && xVal != null && isFinite(xVal) && yVal != null && isFinite(yVal)) { - return applyTransform(out, data, this._transform); - } - - var xAxis = this.getAxis('x'); - var yAxis = this.getAxis('y'); - out[0] = xAxis.toGlobalCoord(xAxis.dataToCoord(xVal, clamp)); - out[1] = yAxis.toGlobalCoord(yAxis.dataToCoord(yVal, clamp)); - return out; - }; - - Cartesian2D.prototype.clampData = function (data, out) { - var xScale = this.getAxis('x').scale; - var yScale = this.getAxis('y').scale; - var xAxisExtent = xScale.getExtent(); - var yAxisExtent = yScale.getExtent(); - var x = xScale.parse(data[0]); - var y = yScale.parse(data[1]); - out = out || []; - out[0] = Math.min(Math.max(Math.min(xAxisExtent[0], xAxisExtent[1]), x), Math.max(xAxisExtent[0], xAxisExtent[1])); - out[1] = Math.min(Math.max(Math.min(yAxisExtent[0], yAxisExtent[1]), y), Math.max(yAxisExtent[0], yAxisExtent[1])); - return out; - }; - - Cartesian2D.prototype.pointToData = function (point, clamp) { - var out = []; - - if (this._invTransform) { - return applyTransform(out, point, this._invTransform); - } - - var xAxis = this.getAxis('x'); - var yAxis = this.getAxis('y'); - out[0] = xAxis.coordToData(xAxis.toLocalCoord(point[0]), clamp); - out[1] = yAxis.coordToData(yAxis.toLocalCoord(point[1]), clamp); - return out; - }; - - Cartesian2D.prototype.getOtherAxis = function (axis) { - return this.getAxis(axis.dim === 'x' ? 'y' : 'x'); - }; - /** - * Get rect area of cartesian. - * Area will have a contain function to determine if a point is in the coordinate system. - */ - - - Cartesian2D.prototype.getArea = function () { - var xExtent = this.getAxis('x').getGlobalExtent(); - var yExtent = this.getAxis('y').getGlobalExtent(); - var x = Math.min(xExtent[0], xExtent[1]); - var y = Math.min(yExtent[0], yExtent[1]); - var width = Math.max(xExtent[0], xExtent[1]) - x; - var height = Math.max(yExtent[0], yExtent[1]) - y; - return new BoundingRect(x, y, width, height); - }; - - return Cartesian2D; - }(Cartesian); - - var Axis2D = - /** @class */ - function (_super) { - __extends(Axis2D, _super); - - function Axis2D(dim, scale, coordExtent, axisType, position) { - var _this = _super.call(this, dim, scale, coordExtent) || this; - /** - * Index of axis, can be used as key - * Injected outside. - */ - - - _this.index = 0; - _this.type = axisType || 'value'; - _this.position = position || 'bottom'; - return _this; - } - - Axis2D.prototype.isHorizontal = function () { - var position = this.position; - return position === 'top' || position === 'bottom'; - }; - /** - * Each item cooresponds to this.getExtent(), which - * means globalExtent[0] may greater than globalExtent[1], - * unless `asc` is input. - * - * @param {boolean} [asc] - * @return {Array.<number>} - */ - - - Axis2D.prototype.getGlobalExtent = function (asc) { - var ret = this.getExtent(); - ret[0] = this.toGlobalCoord(ret[0]); - ret[1] = this.toGlobalCoord(ret[1]); - asc && ret[0] > ret[1] && ret.reverse(); - return ret; - }; - - Axis2D.prototype.pointToData = function (point, clamp) { - return this.coordToData(this.toLocalCoord(point[this.dim === 'x' ? 0 : 1]), clamp); - }; - /** - * Set ordinalSortInfo - * @param info new OrdinalSortInfo - */ - - - Axis2D.prototype.setCategorySortInfo = function (info) { - if (this.type !== 'category') { - return false; - } - - this.model.option.categorySortInfo = info; - this.scale.setSortInfo(info); - }; - - return Axis2D; - }(Axis); - - /** - * Can only be called after coordinate system creation stage. - * (Can be called before coordinate system update stage). - */ - - function layout$1(gridModel, axisModel, opt) { - opt = opt || {}; - var grid = gridModel.coordinateSystem; - var axis = axisModel.axis; - var layout = {}; - var otherAxisOnZeroOf = axis.getAxesOnZeroOf()[0]; - var rawAxisPosition = axis.position; - var axisPosition = otherAxisOnZeroOf ? 'onZero' : rawAxisPosition; - var axisDim = axis.dim; - var rect = grid.getRect(); - var rectBound = [rect.x, rect.x + rect.width, rect.y, rect.y + rect.height]; - var idx = { - left: 0, - right: 1, - top: 0, - bottom: 1, - onZero: 2 - }; - var axisOffset = axisModel.get('offset') || 0; - var posBound = axisDim === 'x' ? [rectBound[2] - axisOffset, rectBound[3] + axisOffset] : [rectBound[0] - axisOffset, rectBound[1] + axisOffset]; - - if (otherAxisOnZeroOf) { - var onZeroCoord = otherAxisOnZeroOf.toGlobalCoord(otherAxisOnZeroOf.dataToCoord(0)); - posBound[idx.onZero] = Math.max(Math.min(onZeroCoord, posBound[1]), posBound[0]); - } // Axis position - - - layout.position = [axisDim === 'y' ? posBound[idx[axisPosition]] : rectBound[0], axisDim === 'x' ? posBound[idx[axisPosition]] : rectBound[3]]; // Axis rotation - - layout.rotation = Math.PI / 2 * (axisDim === 'x' ? 0 : 1); // Tick and label direction, x y is axisDim - - var dirMap = { - top: -1, - bottom: 1, - left: -1, - right: 1 - }; - layout.labelDirection = layout.tickDirection = layout.nameDirection = dirMap[rawAxisPosition]; - layout.labelOffset = otherAxisOnZeroOf ? posBound[idx[rawAxisPosition]] - posBound[idx.onZero] : 0; - - if (axisModel.get(['axisTick', 'inside'])) { - layout.tickDirection = -layout.tickDirection; - } - - if (retrieve(opt.labelInside, axisModel.get(['axisLabel', 'inside']))) { - layout.labelDirection = -layout.labelDirection; - } // Special label rotation - - - var labelRotate = axisModel.get(['axisLabel', 'rotate']); - layout.labelRotate = axisPosition === 'top' ? -labelRotate : labelRotate; // Over splitLine and splitArea - - layout.z2 = 1; - return layout; - } - function isCartesian2DSeries(seriesModel) { - return seriesModel.get('coordinateSystem') === 'cartesian2d'; - } - function findAxisModels(seriesModel) { - var axisModelMap = { - xAxisModel: null, - yAxisModel: null - }; - each(axisModelMap, function (v, key) { - var axisType = key.replace(/Model$/, ''); - var axisModel = seriesModel.getReferringComponents(axisType, SINGLE_REFERRING).models[0]; - - if ("development" !== 'production') { - if (!axisModel) { - throw new Error(axisType + ' "' + retrieve3(seriesModel.get(axisType + 'Index'), seriesModel.get(axisType + 'Id'), 0) + '" not found'); - } - } - - axisModelMap[key] = axisModel; - }); - return axisModelMap; - } - - var mathLog$1 = Math.log; - function alignScaleTicks(scale, axisModel, alignToScale) { - var intervalScaleProto = IntervalScale.prototype; // NOTE: There is a precondition for log scale here: - // In log scale we store _interval and _extent of exponent value. - // So if we use the method of InternalScale to set/get these data. - // It process the exponent value, which is linear and what we want here. - - var alignToTicks = intervalScaleProto.getTicks.call(alignToScale); - var alignToNicedTicks = intervalScaleProto.getTicks.call(alignToScale, true); - var alignToSplitNumber = alignToTicks.length - 1; - var alignToInterval = intervalScaleProto.getInterval.call(alignToScale); - var scaleExtent = getScaleExtent(scale, axisModel); - var rawExtent = scaleExtent.extent; - var isMinFixed = scaleExtent.fixMin; - var isMaxFixed = scaleExtent.fixMax; - - if (scale.type === 'log') { - var logBase = mathLog$1(scale.base); - rawExtent = [mathLog$1(rawExtent[0]) / logBase, mathLog$1(rawExtent[1]) / logBase]; - } - - scale.setExtent(rawExtent[0], rawExtent[1]); - scale.calcNiceExtent({ - splitNumber: alignToSplitNumber, - fixMin: isMinFixed, - fixMax: isMaxFixed - }); - var extent = intervalScaleProto.getExtent.call(scale); // Need to update the rawExtent. - // Because value in rawExtent may be not parsed. e.g. 'dataMin', 'dataMax' - - if (isMinFixed) { - rawExtent[0] = extent[0]; - } - - if (isMaxFixed) { - rawExtent[1] = extent[1]; - } - - var interval = intervalScaleProto.getInterval.call(scale); - var min = rawExtent[0]; - var max = rawExtent[1]; - - if (isMinFixed && isMaxFixed) { - // User set min, max, divide to get new interval - interval = (max - min) / alignToSplitNumber; - } else if (isMinFixed) { - max = rawExtent[0] + interval * alignToSplitNumber; // User set min, expand extent on the other side - - while (max < rawExtent[1] && isFinite(max) && isFinite(rawExtent[1])) { - interval = increaseInterval(interval); - max = rawExtent[0] + interval * alignToSplitNumber; - } - } else if (isMaxFixed) { - // User set max, expand extent on the other side - min = rawExtent[1] - interval * alignToSplitNumber; - - while (min > rawExtent[0] && isFinite(min) && isFinite(rawExtent[0])) { - interval = increaseInterval(interval); - min = rawExtent[1] - interval * alignToSplitNumber; - } - } else { - var nicedSplitNumber = scale.getTicks().length - 1; - - if (nicedSplitNumber > alignToSplitNumber) { - interval = increaseInterval(interval); - } - - var range = interval * alignToSplitNumber; - max = Math.ceil(rawExtent[1] / interval) * interval; - min = round(max - range); // Not change the result that crossing zero. - - if (min < 0 && rawExtent[0] >= 0) { - min = 0; - max = round(range); - } else if (max > 0 && rawExtent[1] <= 0) { - max = 0; - min = -round(range); - } - } // Adjust min, max based on the extent of alignTo. When min or max is set in alignTo scale - - - var t0 = (alignToTicks[0].value - alignToNicedTicks[0].value) / alignToInterval; - var t1 = (alignToTicks[alignToSplitNumber].value - alignToNicedTicks[alignToSplitNumber].value) / alignToInterval; // NOTE: Must in setExtent -> setInterval -> setNiceExtent order. - - intervalScaleProto.setExtent.call(scale, min + interval * t0, max + interval * t1); - intervalScaleProto.setInterval.call(scale, interval); - - if (t0 || t1) { - intervalScaleProto.setNiceExtent.call(scale, min + interval, max - interval); - } - - if ("development" !== 'production') { - var ticks = intervalScaleProto.getTicks.call(scale); - - if (ticks[1] && (!isValueNice(interval) || getPrecisionSafe(ticks[1].value) > getPrecisionSafe(interval))) { - warn( // eslint-disable-next-line - "The ticks may be not readable when set min: " + axisModel.get('min') + ", max: " + axisModel.get('max') + " and alignTicks: true"); - } - } - } - - var Grid = - /** @class */ - function () { - function Grid(gridModel, ecModel, api) { - // FIXME:TS where used (different from registered type 'cartesian2d')? - this.type = 'grid'; - this._coordsMap = {}; - this._coordsList = []; - this._axesMap = {}; - this._axesList = []; - this.axisPointerEnabled = true; - this.dimensions = cartesian2DDimensions; - - this._initCartesian(gridModel, ecModel, api); - - this.model = gridModel; - } - - Grid.prototype.getRect = function () { - return this._rect; - }; - - Grid.prototype.update = function (ecModel, api) { - var axesMap = this._axesMap; - - this._updateScale(ecModel, this.model); - - function updateAxisTicks(axes) { - var alignTo; // Axis is added in order of axisIndex. - - var axesIndices = keys(axes); - var len = axesIndices.length; - - if (!len) { - return; - } - - var axisNeedsAlign = []; // Process once and calculate the ticks for those don't use alignTicks. - - for (var i = len - 1; i >= 0; i--) { - var idx = +axesIndices[i]; // Convert to number. - - var axis = axes[idx]; - var model = axis.model; - var scale = axis.scale; - - if ( // Only value and log axis without interval support alignTicks. - isIntervalOrLogScale(scale) && model.get('alignTicks') && model.get('interval') == null) { - axisNeedsAlign.push(axis); - } else { - niceScaleExtent(scale, model); - - if (isIntervalOrLogScale(scale)) { - // Can only align to interval or log axis. - alignTo = axis; - } - } - } - // PENDING. Should we find the axis that both set interval, min, max and align to this one? - - if (axisNeedsAlign.length) { - if (!alignTo) { - alignTo = axisNeedsAlign.pop(); - niceScaleExtent(alignTo.scale, alignTo.model); - } - - each(axisNeedsAlign, function (axis) { - alignScaleTicks(axis.scale, axis.model, alignTo.scale); - }); - } - } - - updateAxisTicks(axesMap.x); - updateAxisTicks(axesMap.y); // Key: axisDim_axisIndex, value: boolean, whether onZero target. - - var onZeroRecords = {}; - each(axesMap.x, function (xAxis) { - fixAxisOnZero(axesMap, 'y', xAxis, onZeroRecords); - }); - each(axesMap.y, function (yAxis) { - fixAxisOnZero(axesMap, 'x', yAxis, onZeroRecords); - }); // Resize again if containLabel is enabled - // FIXME It may cause getting wrong grid size in data processing stage - - this.resize(this.model, api); - }; - /** - * Resize the grid - */ - - - Grid.prototype.resize = function (gridModel, api, ignoreContainLabel) { - var boxLayoutParams = gridModel.getBoxLayoutParams(); - var isContainLabel = !ignoreContainLabel && gridModel.get('containLabel'); - var gridRect = getLayoutRect(boxLayoutParams, { - width: api.getWidth(), - height: api.getHeight() - }); - this._rect = gridRect; - var axesList = this._axesList; - adjustAxes(); // Minus label size - - if (isContainLabel) { - each(axesList, function (axis) { - if (!axis.model.get(['axisLabel', 'inside'])) { - var labelUnionRect = estimateLabelUnionRect(axis); - - if (labelUnionRect) { - var dim = axis.isHorizontal() ? 'height' : 'width'; - var margin = axis.model.get(['axisLabel', 'margin']); - gridRect[dim] -= labelUnionRect[dim] + margin; - - if (axis.position === 'top') { - gridRect.y += labelUnionRect.height + margin; - } else if (axis.position === 'left') { - gridRect.x += labelUnionRect.width + margin; - } - } - } - }); - adjustAxes(); - } - - each(this._coordsList, function (coord) { - // Calculate affine matrix to accelerate the data to point transform. - // If all the axes scales are time or value. - coord.calcAffineTransform(); - }); - - function adjustAxes() { - each(axesList, function (axis) { - var isHorizontal = axis.isHorizontal(); - var extent = isHorizontal ? [0, gridRect.width] : [0, gridRect.height]; - var idx = axis.inverse ? 1 : 0; - axis.setExtent(extent[idx], extent[1 - idx]); - updateAxisTransform(axis, isHorizontal ? gridRect.x : gridRect.y); - }); - } - }; - - Grid.prototype.getAxis = function (dim, axisIndex) { - var axesMapOnDim = this._axesMap[dim]; - - if (axesMapOnDim != null) { - return axesMapOnDim[axisIndex || 0]; - } - }; - - Grid.prototype.getAxes = function () { - return this._axesList.slice(); - }; - - Grid.prototype.getCartesian = function (xAxisIndex, yAxisIndex) { - if (xAxisIndex != null && yAxisIndex != null) { - var key = 'x' + xAxisIndex + 'y' + yAxisIndex; - return this._coordsMap[key]; - } - - if (isObject(xAxisIndex)) { - yAxisIndex = xAxisIndex.yAxisIndex; - xAxisIndex = xAxisIndex.xAxisIndex; - } - - for (var i = 0, coordList = this._coordsList; i < coordList.length; i++) { - if (coordList[i].getAxis('x').index === xAxisIndex || coordList[i].getAxis('y').index === yAxisIndex) { - return coordList[i]; - } - } - }; - - Grid.prototype.getCartesians = function () { - return this._coordsList.slice(); - }; - /** - * @implements - */ - - - Grid.prototype.convertToPixel = function (ecModel, finder, value) { - var target = this._findConvertTarget(finder); - - return target.cartesian ? target.cartesian.dataToPoint(value) : target.axis ? target.axis.toGlobalCoord(target.axis.dataToCoord(value)) : null; - }; - /** - * @implements - */ - - - Grid.prototype.convertFromPixel = function (ecModel, finder, value) { - var target = this._findConvertTarget(finder); - - return target.cartesian ? target.cartesian.pointToData(value) : target.axis ? target.axis.coordToData(target.axis.toLocalCoord(value)) : null; - }; - - Grid.prototype._findConvertTarget = function (finder) { - var seriesModel = finder.seriesModel; - var xAxisModel = finder.xAxisModel || seriesModel && seriesModel.getReferringComponents('xAxis', SINGLE_REFERRING).models[0]; - var yAxisModel = finder.yAxisModel || seriesModel && seriesModel.getReferringComponents('yAxis', SINGLE_REFERRING).models[0]; - var gridModel = finder.gridModel; - var coordsList = this._coordsList; - var cartesian; - var axis; - - if (seriesModel) { - cartesian = seriesModel.coordinateSystem; - indexOf(coordsList, cartesian) < 0 && (cartesian = null); - } else if (xAxisModel && yAxisModel) { - cartesian = this.getCartesian(xAxisModel.componentIndex, yAxisModel.componentIndex); - } else if (xAxisModel) { - axis = this.getAxis('x', xAxisModel.componentIndex); - } else if (yAxisModel) { - axis = this.getAxis('y', yAxisModel.componentIndex); - } // Lowest priority. - else if (gridModel) { - var grid = gridModel.coordinateSystem; - - if (grid === this) { - cartesian = this._coordsList[0]; - } - } - - return { - cartesian: cartesian, - axis: axis - }; - }; - /** - * @implements - */ - - - Grid.prototype.containPoint = function (point) { - var coord = this._coordsList[0]; - - if (coord) { - return coord.containPoint(point); - } - }; - /** - * Initialize cartesian coordinate systems - */ - - - Grid.prototype._initCartesian = function (gridModel, ecModel, api) { - var _this = this; - - var grid = this; - var axisPositionUsed = { - left: false, - right: false, - top: false, - bottom: false - }; - var axesMap = { - x: {}, - y: {} - }; - var axesCount = { - x: 0, - y: 0 - }; // Create axis - - ecModel.eachComponent('xAxis', createAxisCreator('x'), this); - ecModel.eachComponent('yAxis', createAxisCreator('y'), this); - - if (!axesCount.x || !axesCount.y) { - // Roll back when there no either x or y axis - this._axesMap = {}; - this._axesList = []; - return; - } - - this._axesMap = axesMap; // Create cartesian2d - - each(axesMap.x, function (xAxis, xAxisIndex) { - each(axesMap.y, function (yAxis, yAxisIndex) { - var key = 'x' + xAxisIndex + 'y' + yAxisIndex; - var cartesian = new Cartesian2D(key); - cartesian.master = _this; - cartesian.model = gridModel; - _this._coordsMap[key] = cartesian; - - _this._coordsList.push(cartesian); - - cartesian.addAxis(xAxis); - cartesian.addAxis(yAxis); - }); - }); - - function createAxisCreator(dimName) { - return function (axisModel, idx) { - if (!isAxisUsedInTheGrid(axisModel, gridModel)) { - return; - } - - var axisPosition = axisModel.get('position'); - - if (dimName === 'x') { - // Fix position - if (axisPosition !== 'top' && axisPosition !== 'bottom') { - // Default bottom of X - axisPosition = axisPositionUsed.bottom ? 'top' : 'bottom'; - } - } else { - // Fix position - if (axisPosition !== 'left' && axisPosition !== 'right') { - // Default left of Y - axisPosition = axisPositionUsed.left ? 'right' : 'left'; - } - } - - axisPositionUsed[axisPosition] = true; - var axis = new Axis2D(dimName, createScaleByModel(axisModel), [0, 0], axisModel.get('type'), axisPosition); - var isCategory = axis.type === 'category'; - axis.onBand = isCategory && axisModel.get('boundaryGap'); - axis.inverse = axisModel.get('inverse'); // Inject axis into axisModel - - axisModel.axis = axis; // Inject axisModel into axis - - axis.model = axisModel; // Inject grid info axis - - axis.grid = grid; // Index of axis, can be used as key - - axis.index = idx; - - grid._axesList.push(axis); - - axesMap[dimName][idx] = axis; - axesCount[dimName]++; - }; - } - }; - /** - * Update cartesian properties from series. - */ - - - Grid.prototype._updateScale = function (ecModel, gridModel) { - // Reset scale - each(this._axesList, function (axis) { - axis.scale.setExtent(Infinity, -Infinity); - - if (axis.type === 'category') { - var categorySortInfo = axis.model.get('categorySortInfo'); - axis.scale.setSortInfo(categorySortInfo); - } - }); - ecModel.eachSeries(function (seriesModel) { - if (isCartesian2DSeries(seriesModel)) { - var axesModelMap = findAxisModels(seriesModel); - var xAxisModel = axesModelMap.xAxisModel; - var yAxisModel = axesModelMap.yAxisModel; - - if (!isAxisUsedInTheGrid(xAxisModel, gridModel) || !isAxisUsedInTheGrid(yAxisModel, gridModel)) { - return; - } - - var cartesian = this.getCartesian(xAxisModel.componentIndex, yAxisModel.componentIndex); - var data = seriesModel.getData(); - var xAxis = cartesian.getAxis('x'); - var yAxis = cartesian.getAxis('y'); - unionExtent(data, xAxis); - unionExtent(data, yAxis); - } - }, this); - - function unionExtent(data, axis) { - each(getDataDimensionsOnAxis(data, axis.dim), function (dim) { - axis.scale.unionExtentFromData(data, dim); - }); - } - }; - /** - * @param dim 'x' or 'y' or 'auto' or null/undefined - */ - - - Grid.prototype.getTooltipAxes = function (dim) { - var baseAxes = []; - var otherAxes = []; - each(this.getCartesians(), function (cartesian) { - var baseAxis = dim != null && dim !== 'auto' ? cartesian.getAxis(dim) : cartesian.getBaseAxis(); - var otherAxis = cartesian.getOtherAxis(baseAxis); - indexOf(baseAxes, baseAxis) < 0 && baseAxes.push(baseAxis); - indexOf(otherAxes, otherAxis) < 0 && otherAxes.push(otherAxis); - }); - return { - baseAxes: baseAxes, - otherAxes: otherAxes - }; - }; - - Grid.create = function (ecModel, api) { - var grids = []; - ecModel.eachComponent('grid', function (gridModel, idx) { - var grid = new Grid(gridModel, ecModel, api); - grid.name = 'grid_' + idx; // dataSampling requires axis extent, so resize - // should be performed in create stage. - - grid.resize(gridModel, api, true); - gridModel.coordinateSystem = grid; - grids.push(grid); - }); // Inject the coordinateSystems into seriesModel - - ecModel.eachSeries(function (seriesModel) { - if (!isCartesian2DSeries(seriesModel)) { - return; - } - - var axesModelMap = findAxisModels(seriesModel); - var xAxisModel = axesModelMap.xAxisModel; - var yAxisModel = axesModelMap.yAxisModel; - var gridModel = xAxisModel.getCoordSysModel(); - - if ("development" !== 'production') { - if (!gridModel) { - throw new Error('Grid "' + retrieve3(xAxisModel.get('gridIndex'), xAxisModel.get('gridId'), 0) + '" not found'); - } - - if (xAxisModel.getCoordSysModel() !== yAxisModel.getCoordSysModel()) { - throw new Error('xAxis and yAxis must use the same grid'); - } - } - - var grid = gridModel.coordinateSystem; - seriesModel.coordinateSystem = grid.getCartesian(xAxisModel.componentIndex, yAxisModel.componentIndex); - }); - return grids; - }; // For deciding which dimensions to use when creating list data - - - Grid.dimensions = cartesian2DDimensions; - return Grid; - }(); - /** - * Check if the axis is used in the specified grid. - */ - - - function isAxisUsedInTheGrid(axisModel, gridModel) { - return axisModel.getCoordSysModel() === gridModel; - } - - function fixAxisOnZero(axesMap, otherAxisDim, axis, // Key: see `getOnZeroRecordKey` - onZeroRecords) { - axis.getAxesOnZeroOf = function () { - // TODO: onZero of multiple axes. - return otherAxisOnZeroOf ? [otherAxisOnZeroOf] : []; - }; // onZero can not be enabled in these two situations: - // 1. When any other axis is a category axis. - // 2. When no axis is cross 0 point. - - - var otherAxes = axesMap[otherAxisDim]; - var otherAxisOnZeroOf; - var axisModel = axis.model; - var onZero = axisModel.get(['axisLine', 'onZero']); - var onZeroAxisIndex = axisModel.get(['axisLine', 'onZeroAxisIndex']); - - if (!onZero) { - return; - } // If target axis is specified. - - - if (onZeroAxisIndex != null) { - if (canOnZeroToAxis(otherAxes[onZeroAxisIndex])) { - otherAxisOnZeroOf = otherAxes[onZeroAxisIndex]; - } - } else { - // Find the first available other axis. - for (var idx in otherAxes) { - if (otherAxes.hasOwnProperty(idx) && canOnZeroToAxis(otherAxes[idx]) // Consider that two Y axes on one value axis, - // if both onZero, the two Y axes overlap. - && !onZeroRecords[getOnZeroRecordKey(otherAxes[idx])]) { - otherAxisOnZeroOf = otherAxes[idx]; - break; - } - } - } - - if (otherAxisOnZeroOf) { - onZeroRecords[getOnZeroRecordKey(otherAxisOnZeroOf)] = true; - } - - function getOnZeroRecordKey(axis) { - return axis.dim + '_' + axis.index; - } - } - - function canOnZeroToAxis(axis) { - return axis && axis.type !== 'category' && axis.type !== 'time' && ifAxisCrossZero(axis); - } - - function updateAxisTransform(axis, coordBase) { - var axisExtent = axis.getExtent(); - var axisExtentSum = axisExtent[0] + axisExtent[1]; // Fast transform - - axis.toGlobalCoord = axis.dim === 'x' ? function (coord) { - return coord + coordBase; - } : function (coord) { - return axisExtentSum - coord + coordBase; - }; - axis.toLocalCoord = axis.dim === 'x' ? function (coord) { - return coord - coordBase; - } : function (coord) { - return axisExtentSum - coord + coordBase; - }; - } - - var PI$5 = Math.PI; - /** - * A final axis is translated and rotated from a "standard axis". - * So opt.position and opt.rotation is required. - * - * A standard axis is and axis from [0, 0] to [0, axisExtent[1]], - * for example: (0, 0) ------------> (0, 50) - * - * nameDirection or tickDirection or labelDirection is 1 means tick - * or label is below the standard axis, whereas is -1 means above - * the standard axis. labelOffset means offset between label and axis, - * which is useful when 'onZero', where axisLabel is in the grid and - * label in outside grid. - * - * Tips: like always, - * positive rotation represents anticlockwise, and negative rotation - * represents clockwise. - * The direction of position coordinate is the same as the direction - * of screen coordinate. - * - * Do not need to consider axis 'inverse', which is auto processed by - * axis extent. - */ - - var AxisBuilder = - /** @class */ - function () { - function AxisBuilder(axisModel, opt) { - this.group = new Group(); - this.opt = opt; - this.axisModel = axisModel; // Default value - - defaults(opt, { - labelOffset: 0, - nameDirection: 1, - tickDirection: 1, - labelDirection: 1, - silent: true, - handleAutoShown: function () { - return true; - } - }); // FIXME Not use a separate text group? - - var transformGroup = new Group({ - x: opt.position[0], - y: opt.position[1], - rotation: opt.rotation - }); // this.group.add(transformGroup); - // this._transformGroup = transformGroup; - - transformGroup.updateTransform(); - this._transformGroup = transformGroup; - } - - AxisBuilder.prototype.hasBuilder = function (name) { - return !!builders[name]; - }; - - AxisBuilder.prototype.add = function (name) { - builders[name](this.opt, this.axisModel, this.group, this._transformGroup); - }; - - AxisBuilder.prototype.getGroup = function () { - return this.group; - }; - - AxisBuilder.innerTextLayout = function (axisRotation, textRotation, direction) { - var rotationDiff = remRadian(textRotation - axisRotation); - var textAlign; - var textVerticalAlign; - - if (isRadianAroundZero(rotationDiff)) { - // Label is parallel with axis line. - textVerticalAlign = direction > 0 ? 'top' : 'bottom'; - textAlign = 'center'; - } else if (isRadianAroundZero(rotationDiff - PI$5)) { - // Label is inverse parallel with axis line. - textVerticalAlign = direction > 0 ? 'bottom' : 'top'; - textAlign = 'center'; - } else { - textVerticalAlign = 'middle'; - - if (rotationDiff > 0 && rotationDiff < PI$5) { - textAlign = direction > 0 ? 'right' : 'left'; - } else { - textAlign = direction > 0 ? 'left' : 'right'; - } - } - - return { - rotation: rotationDiff, - textAlign: textAlign, - textVerticalAlign: textVerticalAlign - }; - }; - - AxisBuilder.makeAxisEventDataBase = function (axisModel) { - var eventData = { - componentType: axisModel.mainType, - componentIndex: axisModel.componentIndex - }; - eventData[axisModel.mainType + 'Index'] = axisModel.componentIndex; - return eventData; - }; - - AxisBuilder.isLabelSilent = function (axisModel) { - var tooltipOpt = axisModel.get('tooltip'); - return axisModel.get('silent') // Consider mouse cursor, add these restrictions. - || !(axisModel.get('triggerEvent') || tooltipOpt && tooltipOpt.show); - }; - - return AxisBuilder; - }(); - var builders = { - axisLine: function (opt, axisModel, group, transformGroup) { - var shown = axisModel.get(['axisLine', 'show']); - - if (shown === 'auto' && opt.handleAutoShown) { - shown = opt.handleAutoShown('axisLine'); - } - - if (!shown) { - return; - } - - var extent = axisModel.axis.getExtent(); - var matrix = transformGroup.transform; - var pt1 = [extent[0], 0]; - var pt2 = [extent[1], 0]; - var inverse = pt1[0] > pt2[0]; - - if (matrix) { - applyTransform(pt1, pt1, matrix); - applyTransform(pt2, pt2, matrix); - } - - var lineStyle = extend({ - lineCap: 'round' - }, axisModel.getModel(['axisLine', 'lineStyle']).getLineStyle()); - var line = new Line({ - shape: { - x1: pt1[0], - y1: pt1[1], - x2: pt2[0], - y2: pt2[1] - }, - style: lineStyle, - strokeContainThreshold: opt.strokeContainThreshold || 5, - silent: true, - z2: 1 - }); - subPixelOptimizeLine$1(line.shape, line.style.lineWidth); - line.anid = 'line'; - group.add(line); - var arrows = axisModel.get(['axisLine', 'symbol']); - - if (arrows != null) { - var arrowSize = axisModel.get(['axisLine', 'symbolSize']); - - if (isString(arrows)) { - // Use the same arrow for start and end point - arrows = [arrows, arrows]; - } - - if (isString(arrowSize) || isNumber(arrowSize)) { - // Use the same size for width and height - arrowSize = [arrowSize, arrowSize]; - } - - var arrowOffset = normalizeSymbolOffset(axisModel.get(['axisLine', 'symbolOffset']) || 0, arrowSize); - var symbolWidth_1 = arrowSize[0]; - var symbolHeight_1 = arrowSize[1]; - each([{ - rotate: opt.rotation + Math.PI / 2, - offset: arrowOffset[0], - r: 0 - }, { - rotate: opt.rotation - Math.PI / 2, - offset: arrowOffset[1], - r: Math.sqrt((pt1[0] - pt2[0]) * (pt1[0] - pt2[0]) + (pt1[1] - pt2[1]) * (pt1[1] - pt2[1])) - }], function (point, index) { - if (arrows[index] !== 'none' && arrows[index] != null) { - var symbol = createSymbol(arrows[index], -symbolWidth_1 / 2, -symbolHeight_1 / 2, symbolWidth_1, symbolHeight_1, lineStyle.stroke, true); // Calculate arrow position with offset - - var r = point.r + point.offset; - var pt = inverse ? pt2 : pt1; - symbol.attr({ - rotation: point.rotate, - x: pt[0] + r * Math.cos(opt.rotation), - y: pt[1] - r * Math.sin(opt.rotation), - silent: true, - z2: 11 - }); - group.add(symbol); - } - }); - } - }, - axisTickLabel: function (opt, axisModel, group, transformGroup) { - var ticksEls = buildAxisMajorTicks(group, transformGroup, axisModel, opt); - var labelEls = buildAxisLabel(group, transformGroup, axisModel, opt); - fixMinMaxLabelShow(axisModel, labelEls, ticksEls); - buildAxisMinorTicks(group, transformGroup, axisModel, opt.tickDirection); // This bit fixes the label overlap issue for the time chart. - // See https://github.com/apache/echarts/issues/14266 for more. - - if (axisModel.get(['axisLabel', 'hideOverlap'])) { - var labelList = prepareLayoutList(map(labelEls, function (label) { - return { - label: label, - priority: label.z2, - defaultAttr: { - ignore: label.ignore - } - }; - })); - hideOverlap(labelList); - } - }, - axisName: function (opt, axisModel, group, transformGroup) { - var name = retrieve(opt.axisName, axisModel.get('name')); - - if (!name) { - return; - } - - var nameLocation = axisModel.get('nameLocation'); - var nameDirection = opt.nameDirection; - var textStyleModel = axisModel.getModel('nameTextStyle'); - var gap = axisModel.get('nameGap') || 0; - var extent = axisModel.axis.getExtent(); - var gapSignal = extent[0] > extent[1] ? -1 : 1; - var pos = [nameLocation === 'start' ? extent[0] - gapSignal * gap : nameLocation === 'end' ? extent[1] + gapSignal * gap : (extent[0] + extent[1]) / 2, // Reuse labelOffset. - isNameLocationCenter(nameLocation) ? opt.labelOffset + nameDirection * gap : 0]; - var labelLayout; - var nameRotation = axisModel.get('nameRotate'); - - if (nameRotation != null) { - nameRotation = nameRotation * PI$5 / 180; // To radian. - } - - var axisNameAvailableWidth; - - if (isNameLocationCenter(nameLocation)) { - labelLayout = AxisBuilder.innerTextLayout(opt.rotation, nameRotation != null ? nameRotation : opt.rotation, // Adapt to axis. - nameDirection); - } else { - labelLayout = endTextLayout(opt.rotation, nameLocation, nameRotation || 0, extent); - axisNameAvailableWidth = opt.axisNameAvailableWidth; - - if (axisNameAvailableWidth != null) { - axisNameAvailableWidth = Math.abs(axisNameAvailableWidth / Math.sin(labelLayout.rotation)); - !isFinite(axisNameAvailableWidth) && (axisNameAvailableWidth = null); - } - } - - var textFont = textStyleModel.getFont(); - var truncateOpt = axisModel.get('nameTruncate', true) || {}; - var ellipsis = truncateOpt.ellipsis; - var maxWidth = retrieve(opt.nameTruncateMaxWidth, truncateOpt.maxWidth, axisNameAvailableWidth); - var textEl = new ZRText({ - x: pos[0], - y: pos[1], - rotation: labelLayout.rotation, - silent: AxisBuilder.isLabelSilent(axisModel), - style: createTextStyle(textStyleModel, { - text: name, - font: textFont, - overflow: 'truncate', - width: maxWidth, - ellipsis: ellipsis, - fill: textStyleModel.getTextColor() || axisModel.get(['axisLine', 'lineStyle', 'color']), - align: textStyleModel.get('align') || labelLayout.textAlign, - verticalAlign: textStyleModel.get('verticalAlign') || labelLayout.textVerticalAlign - }), - z2: 1 - }); - setTooltipConfig({ - el: textEl, - componentModel: axisModel, - itemName: name - }); - textEl.__fullText = name; // Id for animation - - textEl.anid = 'name'; - - if (axisModel.get('triggerEvent')) { - var eventData = AxisBuilder.makeAxisEventDataBase(axisModel); - eventData.targetType = 'axisName'; - eventData.name = name; - getECData(textEl).eventData = eventData; - } // FIXME - - - transformGroup.add(textEl); - textEl.updateTransform(); - group.add(textEl); - textEl.decomposeTransform(); - } - }; - - function endTextLayout(rotation, textPosition, textRotate, extent) { - var rotationDiff = remRadian(textRotate - rotation); - var textAlign; - var textVerticalAlign; - var inverse = extent[0] > extent[1]; - var onLeft = textPosition === 'start' && !inverse || textPosition !== 'start' && inverse; - - if (isRadianAroundZero(rotationDiff - PI$5 / 2)) { - textVerticalAlign = onLeft ? 'bottom' : 'top'; - textAlign = 'center'; - } else if (isRadianAroundZero(rotationDiff - PI$5 * 1.5)) { - textVerticalAlign = onLeft ? 'top' : 'bottom'; - textAlign = 'center'; - } else { - textVerticalAlign = 'middle'; - - if (rotationDiff < PI$5 * 1.5 && rotationDiff > PI$5 / 2) { - textAlign = onLeft ? 'left' : 'right'; - } else { - textAlign = onLeft ? 'right' : 'left'; - } - } - - return { - rotation: rotationDiff, - textAlign: textAlign, - textVerticalAlign: textVerticalAlign - }; - } - - function fixMinMaxLabelShow(axisModel, labelEls, tickEls) { - if (shouldShowAllLabels(axisModel.axis)) { - return; - } // If min or max are user set, we need to check - // If the tick on min(max) are overlap on their neighbour tick - // If they are overlapped, we need to hide the min(max) tick label - - - var showMinLabel = axisModel.get(['axisLabel', 'showMinLabel']); - var showMaxLabel = axisModel.get(['axisLabel', 'showMaxLabel']); // FIXME - // Have not consider onBand yet, where tick els is more than label els. - - labelEls = labelEls || []; - tickEls = tickEls || []; - var firstLabel = labelEls[0]; - var nextLabel = labelEls[1]; - var lastLabel = labelEls[labelEls.length - 1]; - var prevLabel = labelEls[labelEls.length - 2]; - var firstTick = tickEls[0]; - var nextTick = tickEls[1]; - var lastTick = tickEls[tickEls.length - 1]; - var prevTick = tickEls[tickEls.length - 2]; - - if (showMinLabel === false) { - ignoreEl(firstLabel); - ignoreEl(firstTick); - } else if (isTwoLabelOverlapped(firstLabel, nextLabel)) { - if (showMinLabel) { - ignoreEl(nextLabel); - ignoreEl(nextTick); - } else { - ignoreEl(firstLabel); - ignoreEl(firstTick); - } - } - - if (showMaxLabel === false) { - ignoreEl(lastLabel); - ignoreEl(lastTick); - } else if (isTwoLabelOverlapped(prevLabel, lastLabel)) { - if (showMaxLabel) { - ignoreEl(prevLabel); - ignoreEl(prevTick); - } else { - ignoreEl(lastLabel); - ignoreEl(lastTick); - } - } - } - - function ignoreEl(el) { - el && (el.ignore = true); - } - - function isTwoLabelOverlapped(current, next) { - // current and next has the same rotation. - var firstRect = current && current.getBoundingRect().clone(); - var nextRect = next && next.getBoundingRect().clone(); - - if (!firstRect || !nextRect) { - return; - } // When checking intersect of two rotated labels, we use mRotationBack - // to avoid that boundingRect is enlarge when using `boundingRect.applyTransform`. - - - var mRotationBack = identity([]); - rotate(mRotationBack, mRotationBack, -current.rotation); - firstRect.applyTransform(mul$1([], mRotationBack, current.getLocalTransform())); - nextRect.applyTransform(mul$1([], mRotationBack, next.getLocalTransform())); - return firstRect.intersect(nextRect); - } - - function isNameLocationCenter(nameLocation) { - return nameLocation === 'middle' || nameLocation === 'center'; - } - - function createTicks(ticksCoords, tickTransform, tickEndCoord, tickLineStyle, anidPrefix) { - var tickEls = []; - var pt1 = []; - var pt2 = []; - - for (var i = 0; i < ticksCoords.length; i++) { - var tickCoord = ticksCoords[i].coord; - pt1[0] = tickCoord; - pt1[1] = 0; - pt2[0] = tickCoord; - pt2[1] = tickEndCoord; - - if (tickTransform) { - applyTransform(pt1, pt1, tickTransform); - applyTransform(pt2, pt2, tickTransform); - } // Tick line, Not use group transform to have better line draw - - - var tickEl = new Line({ - shape: { - x1: pt1[0], - y1: pt1[1], - x2: pt2[0], - y2: pt2[1] - }, - style: tickLineStyle, - z2: 2, - autoBatch: true, - silent: true - }); - subPixelOptimizeLine$1(tickEl.shape, tickEl.style.lineWidth); - tickEl.anid = anidPrefix + '_' + ticksCoords[i].tickValue; - tickEls.push(tickEl); - } - - return tickEls; - } - - function buildAxisMajorTicks(group, transformGroup, axisModel, opt) { - var axis = axisModel.axis; - var tickModel = axisModel.getModel('axisTick'); - var shown = tickModel.get('show'); - - if (shown === 'auto' && opt.handleAutoShown) { - shown = opt.handleAutoShown('axisTick'); - } - - if (!shown || axis.scale.isBlank()) { - return; - } - - var lineStyleModel = tickModel.getModel('lineStyle'); - var tickEndCoord = opt.tickDirection * tickModel.get('length'); - var ticksCoords = axis.getTicksCoords(); - var ticksEls = createTicks(ticksCoords, transformGroup.transform, tickEndCoord, defaults(lineStyleModel.getLineStyle(), { - stroke: axisModel.get(['axisLine', 'lineStyle', 'color']) - }), 'ticks'); - - for (var i = 0; i < ticksEls.length; i++) { - group.add(ticksEls[i]); - } - - return ticksEls; - } - - function buildAxisMinorTicks(group, transformGroup, axisModel, tickDirection) { - var axis = axisModel.axis; - var minorTickModel = axisModel.getModel('minorTick'); - - if (!minorTickModel.get('show') || axis.scale.isBlank()) { - return; - } - - var minorTicksCoords = axis.getMinorTicksCoords(); - - if (!minorTicksCoords.length) { - return; - } - - var lineStyleModel = minorTickModel.getModel('lineStyle'); - var tickEndCoord = tickDirection * minorTickModel.get('length'); - var minorTickLineStyle = defaults(lineStyleModel.getLineStyle(), defaults(axisModel.getModel('axisTick').getLineStyle(), { - stroke: axisModel.get(['axisLine', 'lineStyle', 'color']) - })); - - for (var i = 0; i < minorTicksCoords.length; i++) { - var minorTicksEls = createTicks(minorTicksCoords[i], transformGroup.transform, tickEndCoord, minorTickLineStyle, 'minorticks_' + i); - - for (var k = 0; k < minorTicksEls.length; k++) { - group.add(minorTicksEls[k]); - } - } - } - - function buildAxisLabel(group, transformGroup, axisModel, opt) { - var axis = axisModel.axis; - var show = retrieve(opt.axisLabelShow, axisModel.get(['axisLabel', 'show'])); - - if (!show || axis.scale.isBlank()) { - return; - } - - var labelModel = axisModel.getModel('axisLabel'); - var labelMargin = labelModel.get('margin'); - var labels = axis.getViewLabels(); // Special label rotate. - - var labelRotation = (retrieve(opt.labelRotate, labelModel.get('rotate')) || 0) * PI$5 / 180; - var labelLayout = AxisBuilder.innerTextLayout(opt.rotation, labelRotation, opt.labelDirection); - var rawCategoryData = axisModel.getCategories && axisModel.getCategories(true); - var labelEls = []; - var silent = AxisBuilder.isLabelSilent(axisModel); - var triggerEvent = axisModel.get('triggerEvent'); - each(labels, function (labelItem, index) { - var tickValue = axis.scale.type === 'ordinal' ? axis.scale.getRawOrdinalNumber(labelItem.tickValue) : labelItem.tickValue; - var formattedLabel = labelItem.formattedLabel; - var rawLabel = labelItem.rawLabel; - var itemLabelModel = labelModel; - - if (rawCategoryData && rawCategoryData[tickValue]) { - var rawCategoryItem = rawCategoryData[tickValue]; - - if (isObject(rawCategoryItem) && rawCategoryItem.textStyle) { - itemLabelModel = new Model(rawCategoryItem.textStyle, labelModel, axisModel.ecModel); - } - } - - var textColor = itemLabelModel.getTextColor() || axisModel.get(['axisLine', 'lineStyle', 'color']); - var tickCoord = axis.dataToCoord(tickValue); - var textEl = new ZRText({ - x: tickCoord, - y: opt.labelOffset + opt.labelDirection * labelMargin, - rotation: labelLayout.rotation, - silent: silent, - z2: 10 + (labelItem.level || 0), - style: createTextStyle(itemLabelModel, { - text: formattedLabel, - align: itemLabelModel.getShallow('align', true) || labelLayout.textAlign, - verticalAlign: itemLabelModel.getShallow('verticalAlign', true) || itemLabelModel.getShallow('baseline', true) || labelLayout.textVerticalAlign, - fill: isFunction(textColor) ? textColor( // (1) In category axis with data zoom, tick is not the original - // index of axis.data. So tick should not be exposed to user - // in category axis. - // (2) Compatible with previous version, which always use formatted label as - // input. But in interval scale the formatted label is like '223,445', which - // maked user replace ','. So we modify it to return original val but remain - // it as 'string' to avoid error in replacing. - axis.type === 'category' ? rawLabel : axis.type === 'value' ? tickValue + '' : tickValue, index) : textColor - }) - }); - textEl.anid = 'label_' + tickValue; // Pack data for mouse event - - if (triggerEvent) { - var eventData = AxisBuilder.makeAxisEventDataBase(axisModel); - eventData.targetType = 'axisLabel'; - eventData.value = rawLabel; - eventData.tickIndex = index; - - if (axis.type === 'category') { - eventData.dataIndex = tickValue; - } - - getECData(textEl).eventData = eventData; - } // FIXME - - - transformGroup.add(textEl); - textEl.updateTransform(); - labelEls.push(textEl); - group.add(textEl); - textEl.decomposeTransform(); - }); - return labelEls; - } - - // allAxesInfo should be updated when setOption performed. - - function collect(ecModel, api) { - var result = { - /** - * key: makeKey(axis.model) - * value: { - * axis, - * coordSys, - * axisPointerModel, - * triggerTooltip, - * involveSeries, - * snap, - * seriesModels, - * seriesDataCount - * } - */ - axesInfo: {}, - seriesInvolved: false, - - /** - * key: makeKey(coordSys.model) - * value: Object: key makeKey(axis.model), value: axisInfo - */ - coordSysAxesInfo: {}, - coordSysMap: {} - }; - collectAxesInfo(result, ecModel, api); // Check seriesInvolved for performance, in case too many series in some chart. - - result.seriesInvolved && collectSeriesInfo(result, ecModel); - return result; - } - - function collectAxesInfo(result, ecModel, api) { - var globalTooltipModel = ecModel.getComponent('tooltip'); - var globalAxisPointerModel = ecModel.getComponent('axisPointer'); // links can only be set on global. - - var linksOption = globalAxisPointerModel.get('link', true) || []; - var linkGroups = []; // Collect axes info. - - each(api.getCoordinateSystems(), function (coordSys) { - // Some coordinate system do not support axes, like geo. - if (!coordSys.axisPointerEnabled) { - return; - } - - var coordSysKey = makeKey(coordSys.model); - var axesInfoInCoordSys = result.coordSysAxesInfo[coordSysKey] = {}; - result.coordSysMap[coordSysKey] = coordSys; // Set tooltip (like 'cross') is a convenient way to show axisPointer - // for user. So we enable setting tooltip on coordSys model. - - var coordSysModel = coordSys.model; - var baseTooltipModel = coordSysModel.getModel('tooltip', globalTooltipModel); - each(coordSys.getAxes(), curry(saveTooltipAxisInfo, false, null)); // If axis tooltip used, choose tooltip axis for each coordSys. - // Notice this case: coordSys is `grid` but not `cartesian2D` here. - - if (coordSys.getTooltipAxes && globalTooltipModel // If tooltip.showContent is set as false, tooltip will not - // show but axisPointer will show as normal. - && baseTooltipModel.get('show')) { - // Compatible with previous logic. But series.tooltip.trigger: 'axis' - // or series.data[n].tooltip.trigger: 'axis' are not support any more. - var triggerAxis = baseTooltipModel.get('trigger') === 'axis'; - var cross = baseTooltipModel.get(['axisPointer', 'type']) === 'cross'; - var tooltipAxes = coordSys.getTooltipAxes(baseTooltipModel.get(['axisPointer', 'axis'])); - - if (triggerAxis || cross) { - each(tooltipAxes.baseAxes, curry(saveTooltipAxisInfo, cross ? 'cross' : true, triggerAxis)); - } - - if (cross) { - each(tooltipAxes.otherAxes, curry(saveTooltipAxisInfo, 'cross', false)); - } - } // fromTooltip: true | false | 'cross' - // triggerTooltip: true | false | null - - - function saveTooltipAxisInfo(fromTooltip, triggerTooltip, axis) { - var axisPointerModel = axis.model.getModel('axisPointer', globalAxisPointerModel); - var axisPointerShow = axisPointerModel.get('show'); - - if (!axisPointerShow || axisPointerShow === 'auto' && !fromTooltip && !isHandleTrigger(axisPointerModel)) { - return; - } - - if (triggerTooltip == null) { - triggerTooltip = axisPointerModel.get('triggerTooltip'); - } - - axisPointerModel = fromTooltip ? makeAxisPointerModel(axis, baseTooltipModel, globalAxisPointerModel, ecModel, fromTooltip, triggerTooltip) : axisPointerModel; - var snap = axisPointerModel.get('snap'); - var axisKey = makeKey(axis.model); - var involveSeries = triggerTooltip || snap || axis.type === 'category'; // If result.axesInfo[key] exist, override it (tooltip has higher priority). - - var axisInfo = result.axesInfo[axisKey] = { - key: axisKey, - axis: axis, - coordSys: coordSys, - axisPointerModel: axisPointerModel, - triggerTooltip: triggerTooltip, - involveSeries: involveSeries, - snap: snap, - useHandle: isHandleTrigger(axisPointerModel), - seriesModels: [], - linkGroup: null - }; - axesInfoInCoordSys[axisKey] = axisInfo; - result.seriesInvolved = result.seriesInvolved || involveSeries; - var groupIndex = getLinkGroupIndex(linksOption, axis); - - if (groupIndex != null) { - var linkGroup = linkGroups[groupIndex] || (linkGroups[groupIndex] = { - axesInfo: {} - }); - linkGroup.axesInfo[axisKey] = axisInfo; - linkGroup.mapper = linksOption[groupIndex].mapper; - axisInfo.linkGroup = linkGroup; - } - } - }); - } - - function makeAxisPointerModel(axis, baseTooltipModel, globalAxisPointerModel, ecModel, fromTooltip, triggerTooltip) { - var tooltipAxisPointerModel = baseTooltipModel.getModel('axisPointer'); - var fields = ['type', 'snap', 'lineStyle', 'shadowStyle', 'label', 'animation', 'animationDurationUpdate', 'animationEasingUpdate', 'z']; - var volatileOption = {}; - each(fields, function (field) { - volatileOption[field] = clone(tooltipAxisPointerModel.get(field)); - }); // category axis do not auto snap, otherwise some tick that do not - // has value can not be hovered. value/time/log axis default snap if - // triggered from tooltip and trigger tooltip. - - volatileOption.snap = axis.type !== 'category' && !!triggerTooltip; // Compatible with previous behavior, tooltip axis does not show label by default. - // Only these properties can be overridden from tooltip to axisPointer. - - if (tooltipAxisPointerModel.get('type') === 'cross') { - volatileOption.type = 'line'; - } - - var labelOption = volatileOption.label || (volatileOption.label = {}); // Follow the convention, do not show label when triggered by tooltip by default. - - labelOption.show == null && (labelOption.show = false); - - if (fromTooltip === 'cross') { - // When 'cross', both axes show labels. - var tooltipAxisPointerLabelShow = tooltipAxisPointerModel.get(['label', 'show']); - labelOption.show = tooltipAxisPointerLabelShow != null ? tooltipAxisPointerLabelShow : true; // If triggerTooltip, this is a base axis, which should better not use cross style - // (cross style is dashed by default) - - if (!triggerTooltip) { - var crossStyle = volatileOption.lineStyle = tooltipAxisPointerModel.get('crossStyle'); - crossStyle && defaults(labelOption, crossStyle.textStyle); - } - } - - return axis.model.getModel('axisPointer', new Model(volatileOption, globalAxisPointerModel, ecModel)); - } - - function collectSeriesInfo(result, ecModel) { - // Prepare data for axis trigger - ecModel.eachSeries(function (seriesModel) { - // Notice this case: this coordSys is `cartesian2D` but not `grid`. - var coordSys = seriesModel.coordinateSystem; - var seriesTooltipTrigger = seriesModel.get(['tooltip', 'trigger'], true); - var seriesTooltipShow = seriesModel.get(['tooltip', 'show'], true); - - if (!coordSys || seriesTooltipTrigger === 'none' || seriesTooltipTrigger === false || seriesTooltipTrigger === 'item' || seriesTooltipShow === false || seriesModel.get(['axisPointer', 'show'], true) === false) { - return; - } - - each(result.coordSysAxesInfo[makeKey(coordSys.model)], function (axisInfo) { - var axis = axisInfo.axis; - - if (coordSys.getAxis(axis.dim) === axis) { - axisInfo.seriesModels.push(seriesModel); - axisInfo.seriesDataCount == null && (axisInfo.seriesDataCount = 0); - axisInfo.seriesDataCount += seriesModel.getData().count(); - } - }); - }); - } - /** - * For example: - * { - * axisPointer: { - * links: [{ - * xAxisIndex: [2, 4], - * yAxisIndex: 'all' - * }, { - * xAxisId: ['a5', 'a7'], - * xAxisName: 'xxx' - * }] - * } - * } - */ - - - function getLinkGroupIndex(linksOption, axis) { - var axisModel = axis.model; - var dim = axis.dim; - - for (var i = 0; i < linksOption.length; i++) { - var linkOption = linksOption[i] || {}; - - if (checkPropInLink(linkOption[dim + 'AxisId'], axisModel.id) || checkPropInLink(linkOption[dim + 'AxisIndex'], axisModel.componentIndex) || checkPropInLink(linkOption[dim + 'AxisName'], axisModel.name)) { - return i; - } - } - } - - function checkPropInLink(linkPropValue, axisPropValue) { - return linkPropValue === 'all' || isArray(linkPropValue) && indexOf(linkPropValue, axisPropValue) >= 0 || linkPropValue === axisPropValue; - } - - function fixValue(axisModel) { - var axisInfo = getAxisInfo(axisModel); - - if (!axisInfo) { - return; - } - - var axisPointerModel = axisInfo.axisPointerModel; - var scale = axisInfo.axis.scale; - var option = axisPointerModel.option; - var status = axisPointerModel.get('status'); - var value = axisPointerModel.get('value'); // Parse init value for category and time axis. - - if (value != null) { - value = scale.parse(value); - } - - var useHandle = isHandleTrigger(axisPointerModel); // If `handle` used, `axisPointer` will always be displayed, so value - // and status should be initialized. - - if (status == null) { - option.status = useHandle ? 'show' : 'hide'; - } - - var extent = scale.getExtent().slice(); - extent[0] > extent[1] && extent.reverse(); - - if ( // Pick a value on axis when initializing. - value == null // If both `handle` and `dataZoom` are used, value may be out of axis extent, - // where we should re-pick a value to keep `handle` displaying normally. - || value > extent[1]) { - // Make handle displayed on the end of the axis when init, which looks better. - value = extent[1]; - } - - if (value < extent[0]) { - value = extent[0]; - } - - option.value = value; - - if (useHandle) { - option.status = axisInfo.axis.scale.isBlank() ? 'hide' : 'show'; - } - } - function getAxisInfo(axisModel) { - var coordSysAxesInfo = (axisModel.ecModel.getComponent('axisPointer') || {}).coordSysAxesInfo; - return coordSysAxesInfo && coordSysAxesInfo.axesInfo[makeKey(axisModel)]; - } - function getAxisPointerModel(axisModel) { - var axisInfo = getAxisInfo(axisModel); - return axisInfo && axisInfo.axisPointerModel; - } - - function isHandleTrigger(axisPointerModel) { - return !!axisPointerModel.get(['handle', 'show']); - } - /** - * @param {module:echarts/model/Model} model - * @return {string} unique key - */ - - - function makeKey(model) { - return model.type + '||' + model.id; - } - - var axisPointerClazz = {}; - /** - * Base class of AxisView. - */ - - var AxisView = - /** @class */ - function (_super) { - __extends(AxisView, _super); - - function AxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = AxisView.type; - return _this; - } - /** - * @override - */ - - - AxisView.prototype.render = function (axisModel, ecModel, api, payload) { - // FIXME - // This process should proformed after coordinate systems updated - // (axis scale updated), and should be performed each time update. - // So put it here temporarily, although it is not appropriate to - // put a model-writing procedure in `view`. - this.axisPointerClass && fixValue(axisModel); - - _super.prototype.render.apply(this, arguments); - - this._doUpdateAxisPointerClass(axisModel, api, true); - }; - /** - * Action handler. - */ - - - AxisView.prototype.updateAxisPointer = function (axisModel, ecModel, api, payload) { - this._doUpdateAxisPointerClass(axisModel, api, false); - }; - /** - * @override - */ - - - AxisView.prototype.remove = function (ecModel, api) { - var axisPointer = this._axisPointer; - axisPointer && axisPointer.remove(api); - }; - /** - * @override - */ - - - AxisView.prototype.dispose = function (ecModel, api) { - this._disposeAxisPointer(api); - - _super.prototype.dispose.apply(this, arguments); - }; - - AxisView.prototype._doUpdateAxisPointerClass = function (axisModel, api, forceRender) { - var Clazz = AxisView.getAxisPointerClass(this.axisPointerClass); - - if (!Clazz) { - return; - } - - var axisPointerModel = getAxisPointerModel(axisModel); - axisPointerModel ? (this._axisPointer || (this._axisPointer = new Clazz())).render(axisModel, axisPointerModel, api, forceRender) : this._disposeAxisPointer(api); - }; - - AxisView.prototype._disposeAxisPointer = function (api) { - this._axisPointer && this._axisPointer.dispose(api); - this._axisPointer = null; - }; - - AxisView.registerAxisPointerClass = function (type, clazz) { - if ("development" !== 'production') { - if (axisPointerClazz[type]) { - throw new Error('axisPointer ' + type + ' exists'); - } - } - - axisPointerClazz[type] = clazz; - }; - - AxisView.getAxisPointerClass = function (type) { - return type && axisPointerClazz[type]; - }; - AxisView.type = 'axis'; - return AxisView; - }(ComponentView); - - var inner$6 = makeInner(); - function rectCoordAxisBuildSplitArea(axisView, axisGroup, axisModel, gridModel) { - var axis = axisModel.axis; - - if (axis.scale.isBlank()) { - return; - } // TODO: TYPE - - - var splitAreaModel = axisModel.getModel('splitArea'); - var areaStyleModel = splitAreaModel.getModel('areaStyle'); - var areaColors = areaStyleModel.get('color'); - var gridRect = gridModel.coordinateSystem.getRect(); - var ticksCoords = axis.getTicksCoords({ - tickModel: splitAreaModel, - clamp: true - }); - - if (!ticksCoords.length) { - return; - } // For Making appropriate splitArea animation, the color and anid - // should be corresponding to previous one if possible. - - - var areaColorsLen = areaColors.length; - var lastSplitAreaColors = inner$6(axisView).splitAreaColors; - var newSplitAreaColors = createHashMap(); - var colorIndex = 0; - - if (lastSplitAreaColors) { - for (var i = 0; i < ticksCoords.length; i++) { - var cIndex = lastSplitAreaColors.get(ticksCoords[i].tickValue); - - if (cIndex != null) { - colorIndex = (cIndex + (areaColorsLen - 1) * i) % areaColorsLen; - break; - } - } - } - - var prev = axis.toGlobalCoord(ticksCoords[0].coord); - var areaStyle = areaStyleModel.getAreaStyle(); - areaColors = isArray(areaColors) ? areaColors : [areaColors]; - - for (var i = 1; i < ticksCoords.length; i++) { - var tickCoord = axis.toGlobalCoord(ticksCoords[i].coord); - var x = void 0; - var y = void 0; - var width = void 0; - var height = void 0; - - if (axis.isHorizontal()) { - x = prev; - y = gridRect.y; - width = tickCoord - x; - height = gridRect.height; - prev = x + width; - } else { - x = gridRect.x; - y = prev; - width = gridRect.width; - height = tickCoord - y; - prev = y + height; - } - - var tickValue = ticksCoords[i - 1].tickValue; - tickValue != null && newSplitAreaColors.set(tickValue, colorIndex); - axisGroup.add(new Rect({ - anid: tickValue != null ? 'area_' + tickValue : null, - shape: { - x: x, - y: y, - width: width, - height: height - }, - style: defaults({ - fill: areaColors[colorIndex] - }, areaStyle), - autoBatch: true, - silent: true - })); - colorIndex = (colorIndex + 1) % areaColorsLen; - } - - inner$6(axisView).splitAreaColors = newSplitAreaColors; - } - function rectCoordAxisHandleRemove(axisView) { - inner$6(axisView).splitAreaColors = null; - } - - var axisBuilderAttrs = ['axisLine', 'axisTickLabel', 'axisName']; - var selfBuilderAttrs = ['splitArea', 'splitLine', 'minorSplitLine']; - - var CartesianAxisView = - /** @class */ - function (_super) { - __extends(CartesianAxisView, _super); - - function CartesianAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CartesianAxisView.type; - _this.axisPointerClass = 'CartesianAxisPointer'; - return _this; - } - /** - * @override - */ - - - CartesianAxisView.prototype.render = function (axisModel, ecModel, api, payload) { - this.group.removeAll(); - var oldAxisGroup = this._axisGroup; - this._axisGroup = new Group(); - this.group.add(this._axisGroup); - - if (!axisModel.get('show')) { - return; - } - - var gridModel = axisModel.getCoordSysModel(); - var layout = layout$1(gridModel, axisModel); - var axisBuilder = new AxisBuilder(axisModel, extend({ - handleAutoShown: function (elementType) { - var cartesians = gridModel.coordinateSystem.getCartesians(); - - for (var i = 0; i < cartesians.length; i++) { - if (isIntervalOrLogScale(cartesians[i].getOtherAxis(axisModel.axis).scale)) { - // Still show axis tick or axisLine if other axis is value / log - return true; - } - } // Not show axisTick or axisLine if other axis is category / time - - - return false; - } - }, layout)); - each(axisBuilderAttrs, axisBuilder.add, axisBuilder); - - this._axisGroup.add(axisBuilder.getGroup()); - - each(selfBuilderAttrs, function (name) { - if (axisModel.get([name, 'show'])) { - axisElementBuilders[name](this, this._axisGroup, axisModel, gridModel); - } - }, this); // THIS is a special case for bar racing chart. - // Update the axis label from the natural initial layout to - // sorted layout should has no animation. - - var isInitialSortFromBarRacing = payload && payload.type === 'changeAxisOrder' && payload.isInitSort; - - if (!isInitialSortFromBarRacing) { - groupTransition(oldAxisGroup, this._axisGroup, axisModel); - } - - _super.prototype.render.call(this, axisModel, ecModel, api, payload); - }; - - CartesianAxisView.prototype.remove = function () { - rectCoordAxisHandleRemove(this); - }; - - CartesianAxisView.type = 'cartesianAxis'; - return CartesianAxisView; - }(AxisView); - - var axisElementBuilders = { - splitLine: function (axisView, axisGroup, axisModel, gridModel) { - var axis = axisModel.axis; - - if (axis.scale.isBlank()) { - return; - } - - var splitLineModel = axisModel.getModel('splitLine'); - var lineStyleModel = splitLineModel.getModel('lineStyle'); - var lineColors = lineStyleModel.get('color'); - lineColors = isArray(lineColors) ? lineColors : [lineColors]; - var gridRect = gridModel.coordinateSystem.getRect(); - var isHorizontal = axis.isHorizontal(); - var lineCount = 0; - var ticksCoords = axis.getTicksCoords({ - tickModel: splitLineModel - }); - var p1 = []; - var p2 = []; - var lineStyle = lineStyleModel.getLineStyle(); - - for (var i = 0; i < ticksCoords.length; i++) { - var tickCoord = axis.toGlobalCoord(ticksCoords[i].coord); - - if (isHorizontal) { - p1[0] = tickCoord; - p1[1] = gridRect.y; - p2[0] = tickCoord; - p2[1] = gridRect.y + gridRect.height; - } else { - p1[0] = gridRect.x; - p1[1] = tickCoord; - p2[0] = gridRect.x + gridRect.width; - p2[1] = tickCoord; - } - - var colorIndex = lineCount++ % lineColors.length; - var tickValue = ticksCoords[i].tickValue; - var line = new Line({ - anid: tickValue != null ? 'line_' + ticksCoords[i].tickValue : null, - autoBatch: true, - shape: { - x1: p1[0], - y1: p1[1], - x2: p2[0], - y2: p2[1] - }, - style: defaults({ - stroke: lineColors[colorIndex] - }, lineStyle), - silent: true - }); - subPixelOptimizeLine$1(line.shape, lineStyle.lineWidth); - axisGroup.add(line); - } - }, - minorSplitLine: function (axisView, axisGroup, axisModel, gridModel) { - var axis = axisModel.axis; - var minorSplitLineModel = axisModel.getModel('minorSplitLine'); - var lineStyleModel = minorSplitLineModel.getModel('lineStyle'); - var gridRect = gridModel.coordinateSystem.getRect(); - var isHorizontal = axis.isHorizontal(); - var minorTicksCoords = axis.getMinorTicksCoords(); - - if (!minorTicksCoords.length) { - return; - } - - var p1 = []; - var p2 = []; - var lineStyle = lineStyleModel.getLineStyle(); - - for (var i = 0; i < minorTicksCoords.length; i++) { - for (var k = 0; k < minorTicksCoords[i].length; k++) { - var tickCoord = axis.toGlobalCoord(minorTicksCoords[i][k].coord); - - if (isHorizontal) { - p1[0] = tickCoord; - p1[1] = gridRect.y; - p2[0] = tickCoord; - p2[1] = gridRect.y + gridRect.height; - } else { - p1[0] = gridRect.x; - p1[1] = tickCoord; - p2[0] = gridRect.x + gridRect.width; - p2[1] = tickCoord; - } - - var line = new Line({ - anid: 'minor_line_' + minorTicksCoords[i][k].tickValue, - autoBatch: true, - shape: { - x1: p1[0], - y1: p1[1], - x2: p2[0], - y2: p2[1] - }, - style: lineStyle, - silent: true - }); - subPixelOptimizeLine$1(line.shape, lineStyle.lineWidth); - axisGroup.add(line); - } - } - }, - splitArea: function (axisView, axisGroup, axisModel, gridModel) { - rectCoordAxisBuildSplitArea(axisView, axisGroup, axisModel, gridModel); - } - }; - - var CartesianXAxisView = - /** @class */ - function (_super) { - __extends(CartesianXAxisView, _super); - - function CartesianXAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CartesianXAxisView.type; - return _this; - } - - CartesianXAxisView.type = 'xAxis'; - return CartesianXAxisView; - }(CartesianAxisView); - - var CartesianYAxisView = - /** @class */ - function (_super) { - __extends(CartesianYAxisView, _super); - - function CartesianYAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CartesianXAxisView.type; - return _this; - } - - CartesianYAxisView.type = 'yAxis'; - return CartesianYAxisView; - }(CartesianAxisView); - - var GridView = - /** @class */ - function (_super) { - __extends(GridView, _super); - - function GridView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'grid'; - return _this; - } - - GridView.prototype.render = function (gridModel, ecModel) { - this.group.removeAll(); - - if (gridModel.get('show')) { - this.group.add(new Rect({ - shape: gridModel.coordinateSystem.getRect(), - style: defaults({ - fill: gridModel.get('backgroundColor') - }, gridModel.getItemStyle()), - silent: true, - z2: -1 - })); - } - }; - - GridView.type = 'grid'; - return GridView; - }(ComponentView); - - var extraOption = { - // gridIndex: 0, - // gridId: '', - offset: 0 - }; - function install$5(registers) { - registers.registerComponentView(GridView); - registers.registerComponentModel(GridModel); - registers.registerCoordinateSystem('cartesian2d', Grid); - axisModelCreator(registers, 'x', CartesianAxisModel, extraOption); - axisModelCreator(registers, 'y', CartesianAxisModel, extraOption); - registers.registerComponentView(CartesianXAxisView); - registers.registerComponentView(CartesianYAxisView); - registers.registerPreprocessor(function (option) { - // Only create grid when need - if (option.xAxis && option.yAxis && !option.grid) { - option.grid = {}; - } - }); - } - - function install$6(registers) { - // In case developer forget to include grid component - use(install$5); - registers.registerSeriesModel(ScatterSeriesModel); - registers.registerChartView(ScatterView); - registers.registerLayout(pointsLayout('scatter')); - } - - function radarLayout(ecModel) { - ecModel.eachSeriesByType('radar', function (seriesModel) { - var data = seriesModel.getData(); - var points = []; - var coordSys = seriesModel.coordinateSystem; - - if (!coordSys) { - return; - } - - var axes = coordSys.getIndicatorAxes(); - each(axes, function (axis, axisIndex) { - data.each(data.mapDimension(axes[axisIndex].dim), function (val, dataIndex) { - points[dataIndex] = points[dataIndex] || []; - var point = coordSys.dataToPoint(val, axisIndex); - points[dataIndex][axisIndex] = isValidPoint(point) ? point : getValueMissingPoint(coordSys); - }); - }); // Close polygon - - data.each(function (idx) { - // TODO - // Is it appropriate to connect to the next data when some data is missing? - // Or, should trade it like `connectNull` in line chart? - var firstPoint = find(points[idx], function (point) { - return isValidPoint(point); - }) || getValueMissingPoint(coordSys); // Copy the first actual point to the end of the array - - points[idx].push(firstPoint.slice()); - data.setItemLayout(idx, points[idx]); - }); - }); - } - - function isValidPoint(point) { - return !isNaN(point[0]) && !isNaN(point[1]); - } - - function getValueMissingPoint(coordSys) { - // It is error-prone to input [NaN, NaN] into polygon, polygon. - // (probably cause problem when refreshing or animating) - return [coordSys.cx, coordSys.cy]; - } - - function radarBackwardCompat(option) { - var polarOptArr = option.polar; - - if (polarOptArr) { - if (!isArray(polarOptArr)) { - polarOptArr = [polarOptArr]; - } - - var polarNotRadar_1 = []; - each(polarOptArr, function (polarOpt, idx) { - if (polarOpt.indicator) { - if (polarOpt.type && !polarOpt.shape) { - polarOpt.shape = polarOpt.type; - } - - option.radar = option.radar || []; - - if (!isArray(option.radar)) { - option.radar = [option.radar]; - } - - option.radar.push(polarOpt); - } else { - polarNotRadar_1.push(polarOpt); - } - }); - option.polar = polarNotRadar_1; - } - - each(option.series, function (seriesOpt) { - if (seriesOpt && seriesOpt.type === 'radar' && seriesOpt.polarIndex) { - seriesOpt.radarIndex = seriesOpt.polarIndex; - } - }); - } - - var RadarView = - /** @class */ - function (_super) { - __extends(RadarView, _super); - - function RadarView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = RadarView.type; - return _this; - } - - RadarView.prototype.render = function (seriesModel, ecModel, api) { - var polar = seriesModel.coordinateSystem; - var group = this.group; - var data = seriesModel.getData(); - var oldData = this._data; - - function createSymbol$1(data, idx) { - var symbolType = data.getItemVisual(idx, 'symbol') || 'circle'; - - if (symbolType === 'none') { - return; - } - - var symbolSize = normalizeSymbolSize(data.getItemVisual(idx, 'symbolSize')); - var symbolPath = createSymbol(symbolType, -1, -1, 2, 2); - var symbolRotate = data.getItemVisual(idx, 'symbolRotate') || 0; - symbolPath.attr({ - style: { - strokeNoScale: true - }, - z2: 100, - scaleX: symbolSize[0] / 2, - scaleY: symbolSize[1] / 2, - rotation: symbolRotate * Math.PI / 180 || 0 - }); - return symbolPath; - } - - function updateSymbols(oldPoints, newPoints, symbolGroup, data, idx, isInit) { - // Simply rerender all - symbolGroup.removeAll(); - - for (var i = 0; i < newPoints.length - 1; i++) { - var symbolPath = createSymbol$1(data, idx); - - if (symbolPath) { - symbolPath.__dimIdx = i; - - if (oldPoints[i]) { - symbolPath.setPosition(oldPoints[i]); - graphic[isInit ? 'initProps' : 'updateProps'](symbolPath, { - x: newPoints[i][0], - y: newPoints[i][1] - }, seriesModel, idx); - } else { - symbolPath.setPosition(newPoints[i]); - } - - symbolGroup.add(symbolPath); - } - } - } - - function getInitialPoints(points) { - return map(points, function (pt) { - return [polar.cx, polar.cy]; - }); - } - - data.diff(oldData).add(function (idx) { - var points = data.getItemLayout(idx); - - if (!points) { - return; - } - - var polygon = new Polygon(); - var polyline = new Polyline(); - var target = { - shape: { - points: points - } - }; - polygon.shape.points = getInitialPoints(points); - polyline.shape.points = getInitialPoints(points); - initProps(polygon, target, seriesModel, idx); - initProps(polyline, target, seriesModel, idx); - var itemGroup = new Group(); - var symbolGroup = new Group(); - itemGroup.add(polyline); - itemGroup.add(polygon); - itemGroup.add(symbolGroup); - updateSymbols(polyline.shape.points, points, symbolGroup, data, idx, true); - data.setItemGraphicEl(idx, itemGroup); - }).update(function (newIdx, oldIdx) { - var itemGroup = oldData.getItemGraphicEl(oldIdx); - var polyline = itemGroup.childAt(0); - var polygon = itemGroup.childAt(1); - var symbolGroup = itemGroup.childAt(2); - var target = { - shape: { - points: data.getItemLayout(newIdx) - } - }; - - if (!target.shape.points) { - return; - } - - updateSymbols(polyline.shape.points, target.shape.points, symbolGroup, data, newIdx, false); - saveOldStyle(polygon); - saveOldStyle(polyline); - updateProps(polyline, target, seriesModel); - updateProps(polygon, target, seriesModel); - data.setItemGraphicEl(newIdx, itemGroup); - }).remove(function (idx) { - group.remove(oldData.getItemGraphicEl(idx)); - }).execute(); - data.eachItemGraphicEl(function (itemGroup, idx) { - var itemModel = data.getItemModel(idx); - var polyline = itemGroup.childAt(0); - var polygon = itemGroup.childAt(1); - var symbolGroup = itemGroup.childAt(2); // Radar uses the visual encoded from itemStyle. - - var itemStyle = data.getItemVisual(idx, 'style'); - var color = itemStyle.fill; - group.add(itemGroup); - polyline.useStyle(defaults(itemModel.getModel('lineStyle').getLineStyle(), { - fill: 'none', - stroke: color - })); - setStatesStylesFromModel(polyline, itemModel, 'lineStyle'); - setStatesStylesFromModel(polygon, itemModel, 'areaStyle'); - var areaStyleModel = itemModel.getModel('areaStyle'); - var polygonIgnore = areaStyleModel.isEmpty() && areaStyleModel.parentModel.isEmpty(); - polygon.ignore = polygonIgnore; - each(['emphasis', 'select', 'blur'], function (stateName) { - var stateModel = itemModel.getModel([stateName, 'areaStyle']); - var stateIgnore = stateModel.isEmpty() && stateModel.parentModel.isEmpty(); // Won't be ignore if normal state is not ignore. - - polygon.ensureState(stateName).ignore = stateIgnore && polygonIgnore; - }); - polygon.useStyle(defaults(areaStyleModel.getAreaStyle(), { - fill: color, - opacity: 0.7, - decal: itemStyle.decal - })); - var emphasisModel = itemModel.getModel('emphasis'); - var itemHoverStyle = emphasisModel.getModel('itemStyle').getItemStyle(); - symbolGroup.eachChild(function (symbolPath) { - if (symbolPath instanceof ZRImage) { - var pathStyle = symbolPath.style; - symbolPath.useStyle(extend({ - // TODO other properties like x, y ? - image: pathStyle.image, - x: pathStyle.x, - y: pathStyle.y, - width: pathStyle.width, - height: pathStyle.height - }, itemStyle)); - } else { - symbolPath.useStyle(itemStyle); - symbolPath.setColor(color); - symbolPath.style.strokeNoScale = true; - } - - var pathEmphasisState = symbolPath.ensureState('emphasis'); - pathEmphasisState.style = clone(itemHoverStyle); - var defaultText = data.getStore().get(data.getDimensionIndex(symbolPath.__dimIdx), idx); - (defaultText == null || isNaN(defaultText)) && (defaultText = ''); - setLabelStyle(symbolPath, getLabelStatesModels(itemModel), { - labelFetcher: data.hostModel, - labelDataIndex: idx, - labelDimIndex: symbolPath.__dimIdx, - defaultText: defaultText, - inheritColor: color, - defaultOpacity: itemStyle.opacity - }); - }); - toggleHoverEmphasis(itemGroup, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - }); - this._data = data; - }; - - RadarView.prototype.remove = function () { - this.group.removeAll(); - this._data = null; - }; - - RadarView.type = 'radar'; - return RadarView; - }(ChartView); - - var RadarSeriesModel = - /** @class */ - function (_super) { - __extends(RadarSeriesModel, _super); - - function RadarSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = RadarSeriesModel.type; - _this.hasSymbolVisual = true; - return _this; - } // Overwrite - - - RadarSeriesModel.prototype.init = function (option) { - _super.prototype.init.apply(this, arguments); // Enable legend selection for each data item - // Use a function instead of direct access because data reference may changed - - - this.legendVisualProvider = new LegendVisualProvider(bind(this.getData, this), bind(this.getRawData, this)); - }; - - RadarSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesDataSimply(this, { - generateCoord: 'indicator_', - generateCoordCount: Infinity - }); - }; - - RadarSeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - var data = this.getData(); - var coordSys = this.coordinateSystem; - var indicatorAxes = coordSys.getIndicatorAxes(); - var name = this.getData().getName(dataIndex); - var nameToDisplay = name === '' ? this.name : name; - var markerColor = retrieveVisualColorForTooltipMarker(this, dataIndex); - return createTooltipMarkup('section', { - header: nameToDisplay, - sortBlocks: true, - blocks: map(indicatorAxes, function (axis) { - var val = data.get(data.mapDimension(axis.dim), dataIndex); - return createTooltipMarkup('nameValue', { - markerType: 'subItem', - markerColor: markerColor, - name: axis.name, - value: val, - sortParam: val - }); - }) - }); - }; - - RadarSeriesModel.prototype.getTooltipPosition = function (dataIndex) { - if (dataIndex != null) { - var data_1 = this.getData(); - var coordSys = this.coordinateSystem; - var values = data_1.getValues(map(coordSys.dimensions, function (dim) { - return data_1.mapDimension(dim); - }), dataIndex); - - for (var i = 0, len = values.length; i < len; i++) { - if (!isNaN(values[i])) { - var indicatorAxes = coordSys.getIndicatorAxes(); - return coordSys.coordToPoint(indicatorAxes[i].dataToCoord(values[i]), i); - } - } - } - }; - - RadarSeriesModel.type = 'series.radar'; - RadarSeriesModel.dependencies = ['radar']; - RadarSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - colorBy: 'data', - coordinateSystem: 'radar', - legendHoverLink: true, - radarIndex: 0, - lineStyle: { - width: 2, - type: 'solid', - join: 'round' - }, - label: { - position: 'top' - }, - // areaStyle: { - // }, - // itemStyle: {} - symbolSize: 8 // symbolRotate: null - - }; - return RadarSeriesModel; - }(SeriesModel); - - var valueAxisDefault = axisDefault.value; - - function defaultsShow(opt, show) { - return defaults({ - show: show - }, opt); - } - - var RadarModel = - /** @class */ - function (_super) { - __extends(RadarModel, _super); - - function RadarModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = RadarModel.type; - return _this; - } - - RadarModel.prototype.optionUpdated = function () { - var boundaryGap = this.get('boundaryGap'); - var splitNumber = this.get('splitNumber'); - var scale = this.get('scale'); - var axisLine = this.get('axisLine'); - var axisTick = this.get('axisTick'); // let axisType = this.get('axisType'); - - var axisLabel = this.get('axisLabel'); - var nameTextStyle = this.get('axisName'); - var showName = this.get(['axisName', 'show']); - var nameFormatter = this.get(['axisName', 'formatter']); - var nameGap = this.get('axisNameGap'); - var triggerEvent = this.get('triggerEvent'); - var indicatorModels = map(this.get('indicator') || [], function (indicatorOpt) { - // PENDING - if (indicatorOpt.max != null && indicatorOpt.max > 0 && !indicatorOpt.min) { - indicatorOpt.min = 0; - } else if (indicatorOpt.min != null && indicatorOpt.min < 0 && !indicatorOpt.max) { - indicatorOpt.max = 0; - } - - var iNameTextStyle = nameTextStyle; - - if (indicatorOpt.color != null) { - iNameTextStyle = defaults({ - color: indicatorOpt.color - }, nameTextStyle); - } // Use same configuration - - - var innerIndicatorOpt = merge(clone(indicatorOpt), { - boundaryGap: boundaryGap, - splitNumber: splitNumber, - scale: scale, - axisLine: axisLine, - axisTick: axisTick, - // axisType: axisType, - axisLabel: axisLabel, - // Compatible with 2 and use text - name: indicatorOpt.text, - showName: showName, - nameLocation: 'end', - nameGap: nameGap, - // min: 0, - nameTextStyle: iNameTextStyle, - triggerEvent: triggerEvent - }, false); - - if (isString(nameFormatter)) { - var indName = innerIndicatorOpt.name; - innerIndicatorOpt.name = nameFormatter.replace('{value}', indName != null ? indName : ''); - } else if (isFunction(nameFormatter)) { - innerIndicatorOpt.name = nameFormatter(innerIndicatorOpt.name, innerIndicatorOpt); - } - - var model = new Model(innerIndicatorOpt, null, this.ecModel); - mixin(model, AxisModelCommonMixin.prototype); // For triggerEvent. - - model.mainType = 'radar'; - model.componentIndex = this.componentIndex; - return model; - }, this); - this._indicatorModels = indicatorModels; - }; - - RadarModel.prototype.getIndicatorModels = function () { - return this._indicatorModels; - }; - - RadarModel.type = 'radar'; - RadarModel.defaultOption = { - // zlevel: 0, - z: 0, - center: ['50%', '50%'], - radius: '75%', - startAngle: 90, - axisName: { - show: true // formatter: null - // textStyle: {} - - }, - boundaryGap: [0, 0], - splitNumber: 5, - axisNameGap: 15, - scale: false, - // Polygon or circle - shape: 'polygon', - axisLine: merge({ - lineStyle: { - color: '#bbb' - } - }, valueAxisDefault.axisLine), - axisLabel: defaultsShow(valueAxisDefault.axisLabel, false), - axisTick: defaultsShow(valueAxisDefault.axisTick, false), - // axisType: 'value', - splitLine: defaultsShow(valueAxisDefault.splitLine, true), - splitArea: defaultsShow(valueAxisDefault.splitArea, true), - // {text, min, max} - indicator: [] - }; - return RadarModel; - }(ComponentModel); - - var axisBuilderAttrs$1 = ['axisLine', 'axisTickLabel', 'axisName']; - - var RadarView$1 = - /** @class */ - function (_super) { - __extends(RadarView, _super); - - function RadarView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = RadarView.type; - return _this; - } - - RadarView.prototype.render = function (radarModel, ecModel, api) { - var group = this.group; - group.removeAll(); - - this._buildAxes(radarModel); - - this._buildSplitLineAndArea(radarModel); - }; - - RadarView.prototype._buildAxes = function (radarModel) { - var radar = radarModel.coordinateSystem; - var indicatorAxes = radar.getIndicatorAxes(); - var axisBuilders = map(indicatorAxes, function (indicatorAxis) { - var axisName = indicatorAxis.model.get('showName') ? indicatorAxis.name : ''; // hide name - - var axisBuilder = new AxisBuilder(indicatorAxis.model, { - axisName: axisName, - position: [radar.cx, radar.cy], - rotation: indicatorAxis.angle, - labelDirection: -1, - tickDirection: -1, - nameDirection: 1 - }); - return axisBuilder; - }); - each(axisBuilders, function (axisBuilder) { - each(axisBuilderAttrs$1, axisBuilder.add, axisBuilder); - this.group.add(axisBuilder.getGroup()); - }, this); - }; - - RadarView.prototype._buildSplitLineAndArea = function (radarModel) { - var radar = radarModel.coordinateSystem; - var indicatorAxes = radar.getIndicatorAxes(); - - if (!indicatorAxes.length) { - return; - } - - var shape = radarModel.get('shape'); - var splitLineModel = radarModel.getModel('splitLine'); - var splitAreaModel = radarModel.getModel('splitArea'); - var lineStyleModel = splitLineModel.getModel('lineStyle'); - var areaStyleModel = splitAreaModel.getModel('areaStyle'); - var showSplitLine = splitLineModel.get('show'); - var showSplitArea = splitAreaModel.get('show'); - var splitLineColors = lineStyleModel.get('color'); - var splitAreaColors = areaStyleModel.get('color'); - var splitLineColorsArr = isArray(splitLineColors) ? splitLineColors : [splitLineColors]; - var splitAreaColorsArr = isArray(splitAreaColors) ? splitAreaColors : [splitAreaColors]; - var splitLines = []; - var splitAreas = []; - - function getColorIndex(areaOrLine, areaOrLineColorList, idx) { - var colorIndex = idx % areaOrLineColorList.length; - areaOrLine[colorIndex] = areaOrLine[colorIndex] || []; - return colorIndex; - } - - if (shape === 'circle') { - var ticksRadius = indicatorAxes[0].getTicksCoords(); - var cx = radar.cx; - var cy = radar.cy; - - for (var i = 0; i < ticksRadius.length; i++) { - if (showSplitLine) { - var colorIndex = getColorIndex(splitLines, splitLineColorsArr, i); - splitLines[colorIndex].push(new Circle({ - shape: { - cx: cx, - cy: cy, - r: ticksRadius[i].coord - } - })); - } - - if (showSplitArea && i < ticksRadius.length - 1) { - var colorIndex = getColorIndex(splitAreas, splitAreaColorsArr, i); - splitAreas[colorIndex].push(new Ring({ - shape: { - cx: cx, - cy: cy, - r0: ticksRadius[i].coord, - r: ticksRadius[i + 1].coord - } - })); - } - } - } // Polyyon - else { - var realSplitNumber_1; - var axesTicksPoints = map(indicatorAxes, function (indicatorAxis, idx) { - var ticksCoords = indicatorAxis.getTicksCoords(); - realSplitNumber_1 = realSplitNumber_1 == null ? ticksCoords.length - 1 : Math.min(ticksCoords.length - 1, realSplitNumber_1); - return map(ticksCoords, function (tickCoord) { - return radar.coordToPoint(tickCoord.coord, idx); - }); - }); - var prevPoints = []; - - for (var i = 0; i <= realSplitNumber_1; i++) { - var points = []; - - for (var j = 0; j < indicatorAxes.length; j++) { - points.push(axesTicksPoints[j][i]); - } // Close - - - if (points[0]) { - points.push(points[0].slice()); - } else { - if ("development" !== 'production') { - console.error('Can\'t draw value axis ' + i); - } - } - - if (showSplitLine) { - var colorIndex = getColorIndex(splitLines, splitLineColorsArr, i); - splitLines[colorIndex].push(new Polyline({ - shape: { - points: points - } - })); - } - - if (showSplitArea && prevPoints) { - var colorIndex = getColorIndex(splitAreas, splitAreaColorsArr, i - 1); - splitAreas[colorIndex].push(new Polygon({ - shape: { - points: points.concat(prevPoints) - } - })); - } - - prevPoints = points.slice().reverse(); - } - } - - var lineStyle = lineStyleModel.getLineStyle(); - var areaStyle = areaStyleModel.getAreaStyle(); // Add splitArea before splitLine - - each(splitAreas, function (splitAreas, idx) { - this.group.add(mergePath$1(splitAreas, { - style: defaults({ - stroke: 'none', - fill: splitAreaColorsArr[idx % splitAreaColorsArr.length] - }, areaStyle), - silent: true - })); - }, this); - each(splitLines, function (splitLines, idx) { - this.group.add(mergePath$1(splitLines, { - style: defaults({ - fill: 'none', - stroke: splitLineColorsArr[idx % splitLineColorsArr.length] - }, lineStyle), - silent: true - })); - }, this); - }; - - RadarView.type = 'radar'; - return RadarView; - }(ComponentView); - - var IndicatorAxis = - /** @class */ - function (_super) { - __extends(IndicatorAxis, _super); - - function IndicatorAxis(dim, scale, radiusExtent) { - var _this = _super.call(this, dim, scale, radiusExtent) || this; - - _this.type = 'value'; - _this.angle = 0; - _this.name = ''; - return _this; - } - - return IndicatorAxis; - }(Axis); - - var Radar = - /** @class */ - function () { - function Radar(radarModel, ecModel, api) { - /** - * - * Radar dimensions - */ - this.dimensions = []; - this._model = radarModel; - this._indicatorAxes = map(radarModel.getIndicatorModels(), function (indicatorModel, idx) { - var dim = 'indicator_' + idx; - var indicatorAxis = new IndicatorAxis(dim, new IntervalScale() // (indicatorModel.get('axisType') === 'log') ? new LogScale() : new IntervalScale() - ); - indicatorAxis.name = indicatorModel.get('name'); // Inject model and axis - - indicatorAxis.model = indicatorModel; - indicatorModel.axis = indicatorAxis; - this.dimensions.push(dim); - return indicatorAxis; - }, this); - this.resize(radarModel, api); - } - - Radar.prototype.getIndicatorAxes = function () { - return this._indicatorAxes; - }; - - Radar.prototype.dataToPoint = function (value, indicatorIndex) { - var indicatorAxis = this._indicatorAxes[indicatorIndex]; - return this.coordToPoint(indicatorAxis.dataToCoord(value), indicatorIndex); - }; // TODO: API should be coordToPoint([coord, indicatorIndex]) - - - Radar.prototype.coordToPoint = function (coord, indicatorIndex) { - var indicatorAxis = this._indicatorAxes[indicatorIndex]; - var angle = indicatorAxis.angle; - var x = this.cx + coord * Math.cos(angle); - var y = this.cy - coord * Math.sin(angle); - return [x, y]; - }; - - Radar.prototype.pointToData = function (pt) { - var dx = pt[0] - this.cx; - var dy = pt[1] - this.cy; - var radius = Math.sqrt(dx * dx + dy * dy); - dx /= radius; - dy /= radius; - var radian = Math.atan2(-dy, dx); // Find the closest angle - // FIXME index can calculated directly - - var minRadianDiff = Infinity; - var closestAxis; - var closestAxisIdx = -1; - - for (var i = 0; i < this._indicatorAxes.length; i++) { - var indicatorAxis = this._indicatorAxes[i]; - var diff = Math.abs(radian - indicatorAxis.angle); - - if (diff < minRadianDiff) { - closestAxis = indicatorAxis; - closestAxisIdx = i; - minRadianDiff = diff; - } - } - - return [closestAxisIdx, +(closestAxis && closestAxis.coordToData(radius))]; - }; - - Radar.prototype.resize = function (radarModel, api) { - var center = radarModel.get('center'); - var viewWidth = api.getWidth(); - var viewHeight = api.getHeight(); - var viewSize = Math.min(viewWidth, viewHeight) / 2; - this.cx = parsePercent$1(center[0], viewWidth); - this.cy = parsePercent$1(center[1], viewHeight); - this.startAngle = radarModel.get('startAngle') * Math.PI / 180; // radius may be single value like `20`, `'80%'`, or array like `[10, '80%']` - - var radius = radarModel.get('radius'); - - if (isString(radius) || isNumber(radius)) { - radius = [0, radius]; - } - - this.r0 = parsePercent$1(radius[0], viewSize); - this.r = parsePercent$1(radius[1], viewSize); - each(this._indicatorAxes, function (indicatorAxis, idx) { - indicatorAxis.setExtent(this.r0, this.r); - var angle = this.startAngle + idx * Math.PI * 2 / this._indicatorAxes.length; // Normalize to [-PI, PI] - - angle = Math.atan2(Math.sin(angle), Math.cos(angle)); - indicatorAxis.angle = angle; - }, this); - }; - - Radar.prototype.update = function (ecModel, api) { - var indicatorAxes = this._indicatorAxes; - var radarModel = this._model; - each(indicatorAxes, function (indicatorAxis) { - indicatorAxis.scale.setExtent(Infinity, -Infinity); - }); - ecModel.eachSeriesByType('radar', function (radarSeries, idx) { - if (radarSeries.get('coordinateSystem') !== 'radar' // @ts-ignore - || ecModel.getComponent('radar', radarSeries.get('radarIndex')) !== radarModel) { - return; - } - - var data = radarSeries.getData(); - each(indicatorAxes, function (indicatorAxis) { - indicatorAxis.scale.unionExtentFromData(data, data.mapDimension(indicatorAxis.dim)); - }); - }, this); - var splitNumber = radarModel.get('splitNumber'); - var dummyScale = new IntervalScale(); - dummyScale.setExtent(0, splitNumber); - dummyScale.setInterval(1); // Force all the axis fixing the maxSplitNumber. - - each(indicatorAxes, function (indicatorAxis, idx) { - alignScaleTicks(indicatorAxis.scale, indicatorAxis.model, dummyScale); - }); - }; - - Radar.prototype.convertToPixel = function (ecModel, finder, value) { - console.warn('Not implemented.'); - return null; - }; - - Radar.prototype.convertFromPixel = function (ecModel, finder, pixel) { - console.warn('Not implemented.'); - return null; - }; - - Radar.prototype.containPoint = function (point) { - console.warn('Not implemented.'); - return false; - }; - - Radar.create = function (ecModel, api) { - var radarList = []; - ecModel.eachComponent('radar', function (radarModel) { - var radar = new Radar(radarModel, ecModel, api); - radarList.push(radar); - radarModel.coordinateSystem = radar; - }); - ecModel.eachSeriesByType('radar', function (radarSeries) { - if (radarSeries.get('coordinateSystem') === 'radar') { - // Inject coordinate system - // @ts-ignore - radarSeries.coordinateSystem = radarList[radarSeries.get('radarIndex') || 0]; - } - }); - return radarList; - }; - /** - * Radar dimensions is based on the data - */ - - - Radar.dimensions = []; - return Radar; - }(); - - function install$7(registers) { - registers.registerCoordinateSystem('radar', Radar); - registers.registerComponentModel(RadarModel); - registers.registerComponentView(RadarView$1); - registers.registerVisual({ - seriesType: 'radar', - reset: function (seriesModel) { - var data = seriesModel.getData(); // itemVisual symbol is for selected data - - data.each(function (idx) { - data.setItemVisual(idx, 'legendIcon', 'roundRect'); - }); // visual is for unselected data - - data.setVisual('legendIcon', 'roundRect'); - } - }); - } - - function install$8(registers) { - use(install$7); - registers.registerChartView(RadarView); - registers.registerSeriesModel(RadarSeriesModel); - registers.registerLayout(radarLayout); - registers.registerProcessor(dataFilter('radar')); - registers.registerPreprocessor(radarBackwardCompat); - } - - var ATTR = '\0_ec_interaction_mutex'; - function take(zr, resourceKey, userKey) { - var store = getStore(zr); - store[resourceKey] = userKey; - } - function release(zr, resourceKey, userKey) { - var store = getStore(zr); - var uKey = store[resourceKey]; - - if (uKey === userKey) { - store[resourceKey] = null; - } - } - function isTaken(zr, resourceKey) { - return !!getStore(zr)[resourceKey]; - } - - function getStore(zr) { - return zr[ATTR] || (zr[ATTR] = {}); - } - /** - * payload: { - * type: 'takeGlobalCursor', - * key: 'dataZoomSelect', or 'brush', or ..., - * If no userKey, release global cursor. - * } - */ - // TODO: SELF REGISTERED. - - - registerAction({ - type: 'takeGlobalCursor', - event: 'globalCursorTaken', - update: 'update' - }, noop); - - var RoamController = - /** @class */ - function (_super) { - __extends(RoamController, _super); - - function RoamController(zr) { - var _this = _super.call(this) || this; - - _this._zr = zr; // Avoid two roamController bind the same handler - - var mousedownHandler = bind(_this._mousedownHandler, _this); - var mousemoveHandler = bind(_this._mousemoveHandler, _this); - var mouseupHandler = bind(_this._mouseupHandler, _this); - var mousewheelHandler = bind(_this._mousewheelHandler, _this); - var pinchHandler = bind(_this._pinchHandler, _this); - /** - * Notice: only enable needed types. For example, if 'zoom' - * is not needed, 'zoom' should not be enabled, otherwise - * default mousewheel behaviour (scroll page) will be disabled. - */ - - _this.enable = function (controlType, opt) { - // Disable previous first - this.disable(); - this._opt = defaults(clone(opt) || {}, { - zoomOnMouseWheel: true, - moveOnMouseMove: true, - // By default, wheel do not trigger move. - moveOnMouseWheel: false, - preventDefaultMouseMove: true - }); - - if (controlType == null) { - controlType = true; - } - - if (controlType === true || controlType === 'move' || controlType === 'pan') { - zr.on('mousedown', mousedownHandler); - zr.on('mousemove', mousemoveHandler); - zr.on('mouseup', mouseupHandler); - } - - if (controlType === true || controlType === 'scale' || controlType === 'zoom') { - zr.on('mousewheel', mousewheelHandler); - zr.on('pinch', pinchHandler); - } - }; - - _this.disable = function () { - zr.off('mousedown', mousedownHandler); - zr.off('mousemove', mousemoveHandler); - zr.off('mouseup', mouseupHandler); - zr.off('mousewheel', mousewheelHandler); - zr.off('pinch', pinchHandler); - }; - - return _this; - } - - RoamController.prototype.isDragging = function () { - return this._dragging; - }; - - RoamController.prototype.isPinching = function () { - return this._pinching; - }; - - RoamController.prototype.setPointerChecker = function (pointerChecker) { - this.pointerChecker = pointerChecker; - }; - - RoamController.prototype.dispose = function () { - this.disable(); - }; - - RoamController.prototype._mousedownHandler = function (e) { - if (isMiddleOrRightButtonOnMouseUpDown(e)) { - return; - } - - var el = e.target; - - while (el) { - if (el.draggable) { - return; - } // check if host is draggable - - - el = el.__hostTarget || el.parent; - } - - var x = e.offsetX; - var y = e.offsetY; // Only check on mosedown, but not mousemove. - // Mouse can be out of target when mouse moving. - - if (this.pointerChecker && this.pointerChecker(e, x, y)) { - this._x = x; - this._y = y; - this._dragging = true; - } - }; - - RoamController.prototype._mousemoveHandler = function (e) { - if (!this._dragging || !isAvailableBehavior('moveOnMouseMove', e, this._opt) || e.gestureEvent === 'pinch' || isTaken(this._zr, 'globalPan')) { - return; - } - - var x = e.offsetX; - var y = e.offsetY; - var oldX = this._x; - var oldY = this._y; - var dx = x - oldX; - var dy = y - oldY; - this._x = x; - this._y = y; - this._opt.preventDefaultMouseMove && stop(e.event); - trigger(this, 'pan', 'moveOnMouseMove', e, { - dx: dx, - dy: dy, - oldX: oldX, - oldY: oldY, - newX: x, - newY: y, - isAvailableBehavior: null - }); - }; - - RoamController.prototype._mouseupHandler = function (e) { - if (!isMiddleOrRightButtonOnMouseUpDown(e)) { - this._dragging = false; - } - }; - - RoamController.prototype._mousewheelHandler = function (e) { - var shouldZoom = isAvailableBehavior('zoomOnMouseWheel', e, this._opt); - var shouldMove = isAvailableBehavior('moveOnMouseWheel', e, this._opt); - var wheelDelta = e.wheelDelta; - var absWheelDeltaDelta = Math.abs(wheelDelta); - var originX = e.offsetX; - var originY = e.offsetY; // wheelDelta maybe -0 in chrome mac. - - if (wheelDelta === 0 || !shouldZoom && !shouldMove) { - return; - } // If both `shouldZoom` and `shouldMove` is true, trigger - // their event both, and the final behavior is determined - // by event listener themselves. - - - if (shouldZoom) { - // Convenience: - // Mac and VM Windows on Mac: scroll up: zoom out. - // Windows: scroll up: zoom in. - // FIXME: Should do more test in different environment. - // wheelDelta is too complicated in difference nvironment - // (https://developer.mozilla.org/en-US/docs/Web/Events/mousewheel), - // although it has been normallized by zrender. - // wheelDelta of mouse wheel is bigger than touch pad. - var factor = absWheelDeltaDelta > 3 ? 1.4 : absWheelDeltaDelta > 1 ? 1.2 : 1.1; - var scale = wheelDelta > 0 ? factor : 1 / factor; - checkPointerAndTrigger(this, 'zoom', 'zoomOnMouseWheel', e, { - scale: scale, - originX: originX, - originY: originY, - isAvailableBehavior: null - }); - } - - if (shouldMove) { - // FIXME: Should do more test in different environment. - var absDelta = Math.abs(wheelDelta); // wheelDelta of mouse wheel is bigger than touch pad. - - var scrollDelta = (wheelDelta > 0 ? 1 : -1) * (absDelta > 3 ? 0.4 : absDelta > 1 ? 0.15 : 0.05); - checkPointerAndTrigger(this, 'scrollMove', 'moveOnMouseWheel', e, { - scrollDelta: scrollDelta, - originX: originX, - originY: originY, - isAvailableBehavior: null - }); - } - }; - - RoamController.prototype._pinchHandler = function (e) { - if (isTaken(this._zr, 'globalPan')) { - return; - } - - var scale = e.pinchScale > 1 ? 1.1 : 1 / 1.1; - checkPointerAndTrigger(this, 'zoom', null, e, { - scale: scale, - originX: e.pinchX, - originY: e.pinchY, - isAvailableBehavior: null - }); - }; - - return RoamController; - }(Eventful); - - function checkPointerAndTrigger(controller, eventName, behaviorToCheck, e, contollerEvent) { - if (controller.pointerChecker && controller.pointerChecker(e, contollerEvent.originX, contollerEvent.originY)) { - // When mouse is out of roamController rect, - // default befavoius should not be be disabled, otherwise - // page sliding is disabled, contrary to expectation. - stop(e.event); - trigger(controller, eventName, behaviorToCheck, e, contollerEvent); - } - } - - function trigger(controller, eventName, behaviorToCheck, e, contollerEvent) { - // Also provide behavior checker for event listener, for some case that - // multiple components share one listener. - contollerEvent.isAvailableBehavior = bind(isAvailableBehavior, null, behaviorToCheck, e); // TODO should not have type issue. - - controller.trigger(eventName, contollerEvent); - } // settings: { - // zoomOnMouseWheel - // moveOnMouseMove - // moveOnMouseWheel - // } - // The value can be: true / false / 'shift' / 'ctrl' / 'alt'. - - - function isAvailableBehavior(behaviorToCheck, e, settings) { - var setting = settings[behaviorToCheck]; - return !behaviorToCheck || setting && (!isString(setting) || e.event[setting + 'Key']); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - /** - * For geo and graph. - */ - function updateViewOnPan(controllerHost, dx, dy) { - var target = controllerHost.target; - target.x += dx; - target.y += dy; - target.dirty(); - } - /** - * For geo and graph. - */ - - function updateViewOnZoom(controllerHost, zoomDelta, zoomX, zoomY) { - var target = controllerHost.target; - var zoomLimit = controllerHost.zoomLimit; - var newZoom = controllerHost.zoom = controllerHost.zoom || 1; - newZoom *= zoomDelta; - - if (zoomLimit) { - var zoomMin = zoomLimit.min || 0; - var zoomMax = zoomLimit.max || Infinity; - newZoom = Math.max(Math.min(zoomMax, newZoom), zoomMin); - } - - var zoomScale = newZoom / controllerHost.zoom; - controllerHost.zoom = newZoom; // Keep the mouse center when scaling - - target.x -= (zoomX - target.x) * (zoomScale - 1); - target.y -= (zoomY - target.y) * (zoomScale - 1); - target.scaleX *= zoomScale; - target.scaleY *= zoomScale; - target.dirty(); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var IRRELEVANT_EXCLUDES = { - 'axisPointer': 1, - 'tooltip': 1, - 'brush': 1 - }; - /** - * Avoid that: mouse click on a elements that is over geo or graph, - * but roam is triggered. - */ - - function onIrrelevantElement(e, api, targetCoordSysModel) { - var model = api.getComponentByElement(e.topTarget); // If model is axisModel, it works only if it is injected with coordinateSystem. - - var coordSys = model && model.coordinateSystem; - return model && model !== targetCoordSysModel && !IRRELEVANT_EXCLUDES.hasOwnProperty(model.mainType) && coordSys && coordSys.model !== targetCoordSysModel; - } - - function parseXML(svg) { - if (isString(svg)) { - var parser = new DOMParser(); - svg = parser.parseFromString(svg, 'text/xml'); - } - var svgNode = svg; - if (svgNode.nodeType === 9) { - svgNode = svgNode.firstChild; - } - while (svgNode.nodeName.toLowerCase() !== 'svg' || svgNode.nodeType !== 1) { - svgNode = svgNode.nextSibling; - } - return svgNode; - } - - var nodeParsers; - var INHERITABLE_STYLE_ATTRIBUTES_MAP = { - 'fill': 'fill', - 'stroke': 'stroke', - 'stroke-width': 'lineWidth', - 'opacity': 'opacity', - 'fill-opacity': 'fillOpacity', - 'stroke-opacity': 'strokeOpacity', - 'stroke-dasharray': 'lineDash', - 'stroke-dashoffset': 'lineDashOffset', - 'stroke-linecap': 'lineCap', - 'stroke-linejoin': 'lineJoin', - 'stroke-miterlimit': 'miterLimit', - 'font-family': 'fontFamily', - 'font-size': 'fontSize', - 'font-style': 'fontStyle', - 'font-weight': 'fontWeight', - 'text-anchor': 'textAlign', - 'visibility': 'visibility', - 'display': 'display' - }; - var INHERITABLE_STYLE_ATTRIBUTES_MAP_KEYS = keys(INHERITABLE_STYLE_ATTRIBUTES_MAP); - var SELF_STYLE_ATTRIBUTES_MAP = { - 'alignment-baseline': 'textBaseline', - 'stop-color': 'stopColor' - }; - var SELF_STYLE_ATTRIBUTES_MAP_KEYS = keys(SELF_STYLE_ATTRIBUTES_MAP); - var SVGParser = (function () { - function SVGParser() { - this._defs = {}; - this._root = null; - } - SVGParser.prototype.parse = function (xml, opt) { - opt = opt || {}; - var svg = parseXML(xml); - if ("development" !== 'production') { - if (!svg) { - throw new Error('Illegal svg'); - } - } - this._defsUsePending = []; - var root = new Group(); - this._root = root; - var named = []; - var viewBox = svg.getAttribute('viewBox') || ''; - var width = parseFloat((svg.getAttribute('width') || opt.width)); - var height = parseFloat((svg.getAttribute('height') || opt.height)); - isNaN(width) && (width = null); - isNaN(height) && (height = null); - parseAttributes(svg, root, null, true, false); - var child = svg.firstChild; - while (child) { - this._parseNode(child, root, named, null, false, false); - child = child.nextSibling; - } - applyDefs(this._defs, this._defsUsePending); - this._defsUsePending = []; - var viewBoxRect; - var viewBoxTransform; - if (viewBox) { - var viewBoxArr = splitNumberSequence(viewBox); - if (viewBoxArr.length >= 4) { - viewBoxRect = { - x: parseFloat((viewBoxArr[0] || 0)), - y: parseFloat((viewBoxArr[1] || 0)), - width: parseFloat(viewBoxArr[2]), - height: parseFloat(viewBoxArr[3]) - }; - } - } - if (viewBoxRect && width != null && height != null) { - viewBoxTransform = makeViewBoxTransform(viewBoxRect, { x: 0, y: 0, width: width, height: height }); - if (!opt.ignoreViewBox) { - var elRoot = root; - root = new Group(); - root.add(elRoot); - elRoot.scaleX = elRoot.scaleY = viewBoxTransform.scale; - elRoot.x = viewBoxTransform.x; - elRoot.y = viewBoxTransform.y; - } - } - if (!opt.ignoreRootClip && width != null && height != null) { - root.setClipPath(new Rect({ - shape: { x: 0, y: 0, width: width, height: height } - })); - } - return { - root: root, - width: width, - height: height, - viewBoxRect: viewBoxRect, - viewBoxTransform: viewBoxTransform, - named: named - }; - }; - SVGParser.prototype._parseNode = function (xmlNode, parentGroup, named, namedFrom, isInDefs, isInText) { - var nodeName = xmlNode.nodeName.toLowerCase(); - var el; - var namedFromForSub = namedFrom; - if (nodeName === 'defs') { - isInDefs = true; - } - if (nodeName === 'text') { - isInText = true; - } - if (nodeName === 'defs' || nodeName === 'switch') { - el = parentGroup; - } - else { - if (!isInDefs) { - var parser_1 = nodeParsers[nodeName]; - if (parser_1 && hasOwn(nodeParsers, nodeName)) { - el = parser_1.call(this, xmlNode, parentGroup); - var nameAttr = xmlNode.getAttribute('name'); - if (nameAttr) { - var newNamed = { - name: nameAttr, - namedFrom: null, - svgNodeTagLower: nodeName, - el: el - }; - named.push(newNamed); - if (nodeName === 'g') { - namedFromForSub = newNamed; - } - } - else if (namedFrom) { - named.push({ - name: namedFrom.name, - namedFrom: namedFrom, - svgNodeTagLower: nodeName, - el: el - }); - } - parentGroup.add(el); - } - } - var parser = paintServerParsers[nodeName]; - if (parser && hasOwn(paintServerParsers, nodeName)) { - var def = parser.call(this, xmlNode); - var id = xmlNode.getAttribute('id'); - if (id) { - this._defs[id] = def; - } - } - } - if (el && el.isGroup) { - var child = xmlNode.firstChild; - while (child) { - if (child.nodeType === 1) { - this._parseNode(child, el, named, namedFromForSub, isInDefs, isInText); - } - else if (child.nodeType === 3 && isInText) { - this._parseText(child, el); - } - child = child.nextSibling; - } - } - }; - SVGParser.prototype._parseText = function (xmlNode, parentGroup) { - var text = new TSpan({ - style: { - text: xmlNode.textContent - }, - silent: true, - x: this._textX || 0, - y: this._textY || 0 - }); - inheritStyle(parentGroup, text); - parseAttributes(xmlNode, text, this._defsUsePending, false, false); - applyTextAlignment(text, parentGroup); - var textStyle = text.style; - var fontSize = textStyle.fontSize; - if (fontSize && fontSize < 9) { - textStyle.fontSize = 9; - text.scaleX *= fontSize / 9; - text.scaleY *= fontSize / 9; - } - var font = (textStyle.fontSize || textStyle.fontFamily) && [ - textStyle.fontStyle, - textStyle.fontWeight, - (textStyle.fontSize || 12) + 'px', - textStyle.fontFamily || 'sans-serif' - ].join(' '); - textStyle.font = font; - var rect = text.getBoundingRect(); - this._textX += rect.width; - parentGroup.add(text); - return text; - }; - SVGParser.internalField = (function () { - nodeParsers = { - 'g': function (xmlNode, parentGroup) { - var g = new Group(); - inheritStyle(parentGroup, g); - parseAttributes(xmlNode, g, this._defsUsePending, false, false); - return g; - }, - 'rect': function (xmlNode, parentGroup) { - var rect = new Rect(); - inheritStyle(parentGroup, rect); - parseAttributes(xmlNode, rect, this._defsUsePending, false, false); - rect.setShape({ - x: parseFloat(xmlNode.getAttribute('x') || '0'), - y: parseFloat(xmlNode.getAttribute('y') || '0'), - width: parseFloat(xmlNode.getAttribute('width') || '0'), - height: parseFloat(xmlNode.getAttribute('height') || '0') - }); - rect.silent = true; - return rect; - }, - 'circle': function (xmlNode, parentGroup) { - var circle = new Circle(); - inheritStyle(parentGroup, circle); - parseAttributes(xmlNode, circle, this._defsUsePending, false, false); - circle.setShape({ - cx: parseFloat(xmlNode.getAttribute('cx') || '0'), - cy: parseFloat(xmlNode.getAttribute('cy') || '0'), - r: parseFloat(xmlNode.getAttribute('r') || '0') - }); - circle.silent = true; - return circle; - }, - 'line': function (xmlNode, parentGroup) { - var line = new Line(); - inheritStyle(parentGroup, line); - parseAttributes(xmlNode, line, this._defsUsePending, false, false); - line.setShape({ - x1: parseFloat(xmlNode.getAttribute('x1') || '0'), - y1: parseFloat(xmlNode.getAttribute('y1') || '0'), - x2: parseFloat(xmlNode.getAttribute('x2') || '0'), - y2: parseFloat(xmlNode.getAttribute('y2') || '0') - }); - line.silent = true; - return line; - }, - 'ellipse': function (xmlNode, parentGroup) { - var ellipse = new Ellipse(); - inheritStyle(parentGroup, ellipse); - parseAttributes(xmlNode, ellipse, this._defsUsePending, false, false); - ellipse.setShape({ - cx: parseFloat(xmlNode.getAttribute('cx') || '0'), - cy: parseFloat(xmlNode.getAttribute('cy') || '0'), - rx: parseFloat(xmlNode.getAttribute('rx') || '0'), - ry: parseFloat(xmlNode.getAttribute('ry') || '0') - }); - ellipse.silent = true; - return ellipse; - }, - 'polygon': function (xmlNode, parentGroup) { - var pointsStr = xmlNode.getAttribute('points'); - var pointsArr; - if (pointsStr) { - pointsArr = parsePoints(pointsStr); - } - var polygon = new Polygon({ - shape: { - points: pointsArr || [] - }, - silent: true - }); - inheritStyle(parentGroup, polygon); - parseAttributes(xmlNode, polygon, this._defsUsePending, false, false); - return polygon; - }, - 'polyline': function (xmlNode, parentGroup) { - var pointsStr = xmlNode.getAttribute('points'); - var pointsArr; - if (pointsStr) { - pointsArr = parsePoints(pointsStr); - } - var polyline = new Polyline({ - shape: { - points: pointsArr || [] - }, - silent: true - }); - inheritStyle(parentGroup, polyline); - parseAttributes(xmlNode, polyline, this._defsUsePending, false, false); - return polyline; - }, - 'image': function (xmlNode, parentGroup) { - var img = new ZRImage(); - inheritStyle(parentGroup, img); - parseAttributes(xmlNode, img, this._defsUsePending, false, false); - img.setStyle({ - image: xmlNode.getAttribute('xlink:href') || xmlNode.getAttribute('href'), - x: +xmlNode.getAttribute('x'), - y: +xmlNode.getAttribute('y'), - width: +xmlNode.getAttribute('width'), - height: +xmlNode.getAttribute('height') - }); - img.silent = true; - return img; - }, - 'text': function (xmlNode, parentGroup) { - var x = xmlNode.getAttribute('x') || '0'; - var y = xmlNode.getAttribute('y') || '0'; - var dx = xmlNode.getAttribute('dx') || '0'; - var dy = xmlNode.getAttribute('dy') || '0'; - this._textX = parseFloat(x) + parseFloat(dx); - this._textY = parseFloat(y) + parseFloat(dy); - var g = new Group(); - inheritStyle(parentGroup, g); - parseAttributes(xmlNode, g, this._defsUsePending, false, true); - return g; - }, - 'tspan': function (xmlNode, parentGroup) { - var x = xmlNode.getAttribute('x'); - var y = xmlNode.getAttribute('y'); - if (x != null) { - this._textX = parseFloat(x); - } - if (y != null) { - this._textY = parseFloat(y); - } - var dx = xmlNode.getAttribute('dx') || '0'; - var dy = xmlNode.getAttribute('dy') || '0'; - var g = new Group(); - inheritStyle(parentGroup, g); - parseAttributes(xmlNode, g, this._defsUsePending, false, true); - this._textX += parseFloat(dx); - this._textY += parseFloat(dy); - return g; - }, - 'path': function (xmlNode, parentGroup) { - var d = xmlNode.getAttribute('d') || ''; - var path = createFromString(d); - inheritStyle(parentGroup, path); - parseAttributes(xmlNode, path, this._defsUsePending, false, false); - path.silent = true; - return path; - } - }; - })(); - return SVGParser; - }()); - var paintServerParsers = { - 'lineargradient': function (xmlNode) { - var x1 = parseInt(xmlNode.getAttribute('x1') || '0', 10); - var y1 = parseInt(xmlNode.getAttribute('y1') || '0', 10); - var x2 = parseInt(xmlNode.getAttribute('x2') || '10', 10); - var y2 = parseInt(xmlNode.getAttribute('y2') || '0', 10); - var gradient = new LinearGradient(x1, y1, x2, y2); - parsePaintServerUnit(xmlNode, gradient); - parseGradientColorStops(xmlNode, gradient); - return gradient; - }, - 'radialgradient': function (xmlNode) { - var cx = parseInt(xmlNode.getAttribute('cx') || '0', 10); - var cy = parseInt(xmlNode.getAttribute('cy') || '0', 10); - var r = parseInt(xmlNode.getAttribute('r') || '0', 10); - var gradient = new RadialGradient(cx, cy, r); - parsePaintServerUnit(xmlNode, gradient); - parseGradientColorStops(xmlNode, gradient); - return gradient; - } - }; - function parsePaintServerUnit(xmlNode, gradient) { - var gradientUnits = xmlNode.getAttribute('gradientUnits'); - if (gradientUnits === 'userSpaceOnUse') { - gradient.global = true; - } - } - function parseGradientColorStops(xmlNode, gradient) { - var stop = xmlNode.firstChild; - while (stop) { - if (stop.nodeType === 1 - && stop.nodeName.toLocaleLowerCase() === 'stop') { - var offsetStr = stop.getAttribute('offset'); - var offset = void 0; - if (offsetStr && offsetStr.indexOf('%') > 0) { - offset = parseInt(offsetStr, 10) / 100; - } - else if (offsetStr) { - offset = parseFloat(offsetStr); - } - else { - offset = 0; - } - var styleVals = {}; - parseInlineStyle(stop, styleVals, styleVals); - var stopColor = styleVals.stopColor - || stop.getAttribute('stop-color') - || '#000000'; - gradient.colorStops.push({ - offset: offset, - color: stopColor - }); - } - stop = stop.nextSibling; - } - } - function inheritStyle(parent, child) { - if (parent && parent.__inheritedStyle) { - if (!child.__inheritedStyle) { - child.__inheritedStyle = {}; - } - defaults(child.__inheritedStyle, parent.__inheritedStyle); - } - } - function parsePoints(pointsString) { - var list = splitNumberSequence(pointsString); - var points = []; - for (var i = 0; i < list.length; i += 2) { - var x = parseFloat(list[i]); - var y = parseFloat(list[i + 1]); - points.push([x, y]); - } - return points; - } - function parseAttributes(xmlNode, el, defsUsePending, onlyInlineStyle, isTextGroup) { - var disp = el; - var inheritedStyle = disp.__inheritedStyle = disp.__inheritedStyle || {}; - var selfStyle = {}; - if (xmlNode.nodeType === 1) { - parseTransformAttribute(xmlNode, el); - parseInlineStyle(xmlNode, inheritedStyle, selfStyle); - if (!onlyInlineStyle) { - parseAttributeStyle(xmlNode, inheritedStyle, selfStyle); - } - } - disp.style = disp.style || {}; - if (inheritedStyle.fill != null) { - disp.style.fill = getFillStrokeStyle(disp, 'fill', inheritedStyle.fill, defsUsePending); - } - if (inheritedStyle.stroke != null) { - disp.style.stroke = getFillStrokeStyle(disp, 'stroke', inheritedStyle.stroke, defsUsePending); - } - each([ - 'lineWidth', 'opacity', 'fillOpacity', 'strokeOpacity', 'miterLimit', 'fontSize' - ], function (propName) { - if (inheritedStyle[propName] != null) { - disp.style[propName] = parseFloat(inheritedStyle[propName]); - } - }); - each([ - 'lineDashOffset', 'lineCap', 'lineJoin', 'fontWeight', 'fontFamily', 'fontStyle', 'textAlign' - ], function (propName) { - if (inheritedStyle[propName] != null) { - disp.style[propName] = inheritedStyle[propName]; - } - }); - if (isTextGroup) { - disp.__selfStyle = selfStyle; - } - if (inheritedStyle.lineDash) { - disp.style.lineDash = map(splitNumberSequence(inheritedStyle.lineDash), function (str) { - return parseFloat(str); - }); - } - if (inheritedStyle.visibility === 'hidden' || inheritedStyle.visibility === 'collapse') { - disp.invisible = true; - } - if (inheritedStyle.display === 'none') { - disp.ignore = true; - } - } - function applyTextAlignment(text, parentGroup) { - var parentSelfStyle = parentGroup.__selfStyle; - if (parentSelfStyle) { - var textBaseline = parentSelfStyle.textBaseline; - var zrTextBaseline = textBaseline; - if (!textBaseline || textBaseline === 'auto') { - zrTextBaseline = 'alphabetic'; - } - else if (textBaseline === 'baseline') { - zrTextBaseline = 'alphabetic'; - } - else if (textBaseline === 'before-edge' || textBaseline === 'text-before-edge') { - zrTextBaseline = 'top'; - } - else if (textBaseline === 'after-edge' || textBaseline === 'text-after-edge') { - zrTextBaseline = 'bottom'; - } - else if (textBaseline === 'central' || textBaseline === 'mathematical') { - zrTextBaseline = 'middle'; - } - text.style.textBaseline = zrTextBaseline; - } - var parentInheritedStyle = parentGroup.__inheritedStyle; - if (parentInheritedStyle) { - var textAlign = parentInheritedStyle.textAlign; - var zrTextAlign = textAlign; - if (textAlign) { - if (textAlign === 'middle') { - zrTextAlign = 'center'; - } - text.style.textAlign = zrTextAlign; - } - } - } - var urlRegex = /^url\(\s*#(.*?)\)/; - function getFillStrokeStyle(el, method, str, defsUsePending) { - var urlMatch = str && str.match(urlRegex); - if (urlMatch) { - var url = trim(urlMatch[1]); - defsUsePending.push([el, method, url]); - return; - } - if (str === 'none') { - str = null; - } - return str; - } - function applyDefs(defs, defsUsePending) { - for (var i = 0; i < defsUsePending.length; i++) { - var item = defsUsePending[i]; - item[0].style[item[1]] = defs[item[2]]; - } - } - var numberReg$1 = /-?([0-9]*\.)?[0-9]+([eE]-?[0-9]+)?/g; - function splitNumberSequence(rawStr) { - return rawStr.match(numberReg$1) || []; - } - var transformRegex = /(translate|scale|rotate|skewX|skewY|matrix)\(([\-\s0-9\.eE,]*)\)/g; - var DEGREE_TO_ANGLE = Math.PI / 180; - function parseTransformAttribute(xmlNode, node) { - var transform = xmlNode.getAttribute('transform'); - if (transform) { - transform = transform.replace(/,/g, ' '); - var transformOps_1 = []; - var mt = null; - transform.replace(transformRegex, function (str, type, value) { - transformOps_1.push(type, value); - return ''; - }); - for (var i = transformOps_1.length - 1; i > 0; i -= 2) { - var value = transformOps_1[i]; - var type = transformOps_1[i - 1]; - var valueArr = splitNumberSequence(value); - mt = mt || create$1(); - switch (type) { - case 'translate': - translate(mt, mt, [parseFloat(valueArr[0]), parseFloat(valueArr[1] || '0')]); - break; - case 'scale': - scale$1(mt, mt, [parseFloat(valueArr[0]), parseFloat(valueArr[1] || valueArr[0])]); - break; - case 'rotate': - rotate(mt, mt, -parseFloat(valueArr[0]) * DEGREE_TO_ANGLE); - break; - case 'skewX': - var sx = Math.tan(parseFloat(valueArr[0]) * DEGREE_TO_ANGLE); - mul$1(mt, [1, 0, sx, 1, 0, 0], mt); - break; - case 'skewY': - var sy = Math.tan(parseFloat(valueArr[0]) * DEGREE_TO_ANGLE); - mul$1(mt, [1, sy, 0, 1, 0, 0], mt); - break; - case 'matrix': - mt[0] = parseFloat(valueArr[0]); - mt[1] = parseFloat(valueArr[1]); - mt[2] = parseFloat(valueArr[2]); - mt[3] = parseFloat(valueArr[3]); - mt[4] = parseFloat(valueArr[4]); - mt[5] = parseFloat(valueArr[5]); - break; - } - } - node.setLocalTransform(mt); - } - } - var styleRegex = /([^\s:;]+)\s*:\s*([^:;]+)/g; - function parseInlineStyle(xmlNode, inheritableStyleResult, selfStyleResult) { - var style = xmlNode.getAttribute('style'); - if (!style) { - return; - } - styleRegex.lastIndex = 0; - var styleRegResult; - while ((styleRegResult = styleRegex.exec(style)) != null) { - var svgStlAttr = styleRegResult[1]; - var zrInheritableStlAttr = hasOwn(INHERITABLE_STYLE_ATTRIBUTES_MAP, svgStlAttr) - ? INHERITABLE_STYLE_ATTRIBUTES_MAP[svgStlAttr] - : null; - if (zrInheritableStlAttr) { - inheritableStyleResult[zrInheritableStlAttr] = styleRegResult[2]; - } - var zrSelfStlAttr = hasOwn(SELF_STYLE_ATTRIBUTES_MAP, svgStlAttr) - ? SELF_STYLE_ATTRIBUTES_MAP[svgStlAttr] - : null; - if (zrSelfStlAttr) { - selfStyleResult[zrSelfStlAttr] = styleRegResult[2]; - } - } - } - function parseAttributeStyle(xmlNode, inheritableStyleResult, selfStyleResult) { - for (var i = 0; i < INHERITABLE_STYLE_ATTRIBUTES_MAP_KEYS.length; i++) { - var svgAttrName = INHERITABLE_STYLE_ATTRIBUTES_MAP_KEYS[i]; - var attrValue = xmlNode.getAttribute(svgAttrName); - if (attrValue != null) { - inheritableStyleResult[INHERITABLE_STYLE_ATTRIBUTES_MAP[svgAttrName]] = attrValue; - } - } - for (var i = 0; i < SELF_STYLE_ATTRIBUTES_MAP_KEYS.length; i++) { - var svgAttrName = SELF_STYLE_ATTRIBUTES_MAP_KEYS[i]; - var attrValue = xmlNode.getAttribute(svgAttrName); - if (attrValue != null) { - selfStyleResult[SELF_STYLE_ATTRIBUTES_MAP[svgAttrName]] = attrValue; - } - } - } - function makeViewBoxTransform(viewBoxRect, boundingRect) { - var scaleX = boundingRect.width / viewBoxRect.width; - var scaleY = boundingRect.height / viewBoxRect.height; - var scale = Math.min(scaleX, scaleY); - return { - scale: scale, - x: -(viewBoxRect.x + viewBoxRect.width / 2) * scale + (boundingRect.x + boundingRect.width / 2), - y: -(viewBoxRect.y + viewBoxRect.height / 2) * scale + (boundingRect.y + boundingRect.height / 2) - }; - } - function parseSVG(xml, opt) { - var parser = new SVGParser(); - return parser.parse(xml, opt); - } - - /** - * "region available" means that: enable users to set attribute `name="xxx"` on those tags - * to make it be a region. - * 1. region styles and its label styles can be defined in echarts opton: - * ```js - * geo: { - * regions: [{ - * name: 'xxx', - * itemStyle: { ... }, - * label: { ... } - * }, { - * ... - * }, - * ...] - * }; - * ``` - * 2. name can be duplicated in different SVG tag. All of the tags with the same name share - * a region option. For exampel if there are two <path> representing two lung lobes. They have - * no common parents but both of them need to display label "lung" inside. - */ - - var REGION_AVAILABLE_SVG_TAG_MAP = createHashMap(['rect', 'circle', 'line', 'ellipse', 'polygon', 'polyline', 'path', // <text> <tspan> are also enabled because some SVG might paint text itself, - // but still need to trigger events or tooltip. - 'text', 'tspan', // <g> is also enabled because this case: if multiple tags share one name - // and need label displayed, every tags will display the name, which is not - // expected. So we can put them into a <g name="xxx">. Thereby only one label - // displayed and located based on the bounding rect of the <g>. - 'g']); - - var GeoSVGResource = - /** @class */ - function () { - function GeoSVGResource(mapName, svg) { - this.type = 'geoSVG'; // All used graphics. key: hostKey, value: root - - this._usedGraphicMap = createHashMap(); // All unused graphics. - - this._freedGraphics = []; - this._mapName = mapName; // Only perform parse to XML object here, which might be time - // consiming for large SVG. - // Although convert XML to zrender element is also time consiming, - // if we do it here, the clone of zrender elements has to be - // required. So we do it once for each geo instance, util real - // performance issues call for optimizing it. - - this._parsedXML = parseXML(svg); - } - - GeoSVGResource.prototype.load = function () - /* nameMap: NameMap */ - { - // In the "load" stage, graphic need to be built to - // get boundingRect for geo coordinate system. - var firstGraphic = this._firstGraphic; // Create the return data structure only when first graphic created. - // Because they will be used in geo coordinate system update stage, - // and `regions` will be mounted at `geo` coordinate system, - // in which there is no "view" info, so that it should better not to - // make references to graphic elements. - - if (!firstGraphic) { - firstGraphic = this._firstGraphic = this._buildGraphic(this._parsedXML); - - this._freedGraphics.push(firstGraphic); - - this._boundingRect = this._firstGraphic.boundingRect.clone(); // PENDING: `nameMap` will not be supported until some real requirement come. - // if (nameMap) { - // named = applyNameMap(named, nameMap); - // } - - var _a = createRegions(firstGraphic.named), - regions = _a.regions, - regionsMap = _a.regionsMap; - - this._regions = regions; - this._regionsMap = regionsMap; - } - - return { - boundingRect: this._boundingRect, - regions: this._regions, - regionsMap: this._regionsMap - }; - }; - - GeoSVGResource.prototype._buildGraphic = function (svgXML) { - var result; - var rootFromParse; - - try { - result = svgXML && parseSVG(svgXML, { - ignoreViewBox: true, - ignoreRootClip: true - }) || {}; - rootFromParse = result.root; - assert(rootFromParse != null); - } catch (e) { - throw new Error('Invalid svg format\n' + e.message); - } // Note: we keep the covenant that the root has no transform. So always add an extra root. - - - var root = new Group(); - root.add(rootFromParse); - root.isGeoSVGGraphicRoot = true; // [THE_RULE_OF_VIEWPORT_AND_VIEWBOX] - // - // Consider: `<svg width="..." height="..." viewBox="...">` - // - the `width/height` we call it `svgWidth/svgHeight` for short. - // - `(0, 0, svgWidth, svgHeight)` defines the viewport of the SVG, or say, - // "viewport boundingRect", or `boundingRect` for short. - // - `viewBox` defines the transform from the real content ot the viewport. - // `viewBox` has the same unit as the content of SVG. - // If `viewBox` exists, a transform is defined, so the unit of `svgWidth/svgHeight` become - // different from the content of SVG. Otherwise, they are the same. - // - // If both `svgWidth/svgHeight/viewBox` are specified in a SVG file, the transform rule will be: - // 0. `boundingRect` is `(0, 0, svgWidth, svgHeight)`. Set it to Geo['_rect'] (View['_rect']). - // 1. Make a transform from `viewBox` to `boundingRect`. - // Note: only support `preserveAspectRatio 'xMidYMid'` here. That is, this transform will preserve - // the aspect ratio. - // 2. Make a transform from boundingRect to Geo['_viewRect'] (View['_viewRect']) - // (`Geo`/`View` will do this job). - // Note: this transform might not preserve aspect radio, which depending on how users specify - // viewRect in echarts option (e.g., `geo.left/top/width/height` will not preserve aspect ratio, - // but `geo.layoutCenter/layoutSize` will preserve aspect ratio). - // - // If `svgWidth/svgHeight` not specified, we use `viewBox` as the `boundingRect` to make the SVG - // layout look good. - // - // If neither `svgWidth/svgHeight` nor `viewBox` are not specified, we calculate the boundingRect - // of the SVG content and use them to make SVG layout look good. - - var svgWidth = result.width; - var svgHeight = result.height; - var viewBoxRect = result.viewBoxRect; - var boundingRect = this._boundingRect; - - if (!boundingRect) { - var bRectX = void 0; - var bRectY = void 0; - var bRectWidth = void 0; - var bRectHeight = void 0; - - if (svgWidth != null) { - bRectX = 0; - bRectWidth = svgWidth; - } else if (viewBoxRect) { - bRectX = viewBoxRect.x; - bRectWidth = viewBoxRect.width; - } - - if (svgHeight != null) { - bRectY = 0; - bRectHeight = svgHeight; - } else if (viewBoxRect) { - bRectY = viewBoxRect.y; - bRectHeight = viewBoxRect.height; - } // If both viewBox and svgWidth/svgHeight not specified, - // we have to determine how to layout those element to make them look good. - - - if (bRectX == null || bRectY == null) { - var calculatedBoundingRect = rootFromParse.getBoundingRect(); - - if (bRectX == null) { - bRectX = calculatedBoundingRect.x; - bRectWidth = calculatedBoundingRect.width; - } - - if (bRectY == null) { - bRectY = calculatedBoundingRect.y; - bRectHeight = calculatedBoundingRect.height; - } - } - - boundingRect = this._boundingRect = new BoundingRect(bRectX, bRectY, bRectWidth, bRectHeight); - } - - if (viewBoxRect) { - var viewBoxTransform = makeViewBoxTransform(viewBoxRect, boundingRect); // Only support `preserveAspectRatio 'xMidYMid'` - - rootFromParse.scaleX = rootFromParse.scaleY = viewBoxTransform.scale; - rootFromParse.x = viewBoxTransform.x; - rootFromParse.y = viewBoxTransform.y; - } // SVG needs to clip based on `viewBox`. And some SVG files really rely on this feature. - // They do not strictly confine all of the content inside a display rect, but deliberately - // use a `viewBox` to define a displayable rect. - // PENDING: - // The drawback of the `setClipPath` here is: the region label (genereted by echarts) near the - // edge might also be clipped, because region labels are put as `textContent` of the SVG path. - - - root.setClipPath(new Rect({ - shape: boundingRect.plain() - })); - var named = []; - each(result.named, function (namedItem) { - if (REGION_AVAILABLE_SVG_TAG_MAP.get(namedItem.svgNodeTagLower) != null) { - named.push(namedItem); - setSilent(namedItem.el); - } - }); - return { - root: root, - boundingRect: boundingRect, - named: named - }; - }; - /** - * Consider: - * (1) One graphic element can not be shared by different `geoView` running simultaneously. - * Notice, also need to consider multiple echarts instances share a `mapRecord`. - * (2) Converting SVG to graphic elements is time consuming. - * (3) In the current architecture, `load` should be called frequently to get boundingRect, - * and it is called without view info. - * So we maintain graphic elements in this module, and enables `view` to use/return these - * graphics from/to the pool with it's uid. - */ - - - GeoSVGResource.prototype.useGraphic = function (hostKey - /* , nameMap: NameMap */ - ) { - var usedRootMap = this._usedGraphicMap; - var svgGraphic = usedRootMap.get(hostKey); - - if (svgGraphic) { - return svgGraphic; - } - - svgGraphic = this._freedGraphics.pop() // use the first boundingRect to avoid duplicated boundingRect calculation. - || this._buildGraphic(this._parsedXML); - usedRootMap.set(hostKey, svgGraphic); // PENDING: `nameMap` will not be supported until some real requirement come. - // `nameMap` can only be obtained from echarts option. - // The original `named` must not be modified. - // if (nameMap) { - // svgGraphic = extend({}, svgGraphic); - // svgGraphic.named = applyNameMap(svgGraphic.named, nameMap); - // } - - return svgGraphic; - }; - - GeoSVGResource.prototype.freeGraphic = function (hostKey) { - var usedRootMap = this._usedGraphicMap; - var svgGraphic = usedRootMap.get(hostKey); - - if (svgGraphic) { - usedRootMap.removeKey(hostKey); - - this._freedGraphics.push(svgGraphic); - } - }; - - return GeoSVGResource; - }(); - - function setSilent(el) { - // Only named element has silent: false, other elements should - // act as background and has no user interaction. - el.silent = false; // text|tspan will be converted to group. - - if (el.isGroup) { - el.traverse(function (child) { - child.silent = false; - }); - } - } - - function createRegions(named) { - var regions = []; - var regionsMap = createHashMap(); // Create resions only for the first graphic. - - each(named, function (namedItem) { - // Region has feature to calculate center for tooltip or other features. - // If there is a <g name="xxx">, the center should be the center of the - // bounding rect of the g. - if (namedItem.namedFrom != null) { - return; - } - - var region = new GeoSVGRegion(namedItem.name, namedItem.el); // PENDING: if `nameMap` supported, this region can not be mounted on - // `this`, but can only be created each time `load()` called. - - regions.push(region); // PENDING: if multiple tag named with the same name, only one will be - // found by `_regionsMap`. `_regionsMap` is used to find a coordinate - // by name. We use `region.getCenter()` as the coordinate. - - regionsMap.set(namedItem.name, region); - }); - return { - regions: regions, - regionsMap: regionsMap - }; - } // PENDING: `nameMap` will not be supported until some real requirement come. - // /** - // * Use the alias in geoNameMap. - // * The input `named` must not be modified. - // */ - // function applyNameMap( - // named: GeoSVGGraphicRecord['named'], - // nameMap: NameMap - // ): GeoSVGGraphicRecord['named'] { - // const result = [] as GeoSVGGraphicRecord['named']; - // for (let i = 0; i < named.length; i++) { - // let regionGraphic = named[i]; - // const name = regionGraphic.name; - // if (nameMap && nameMap.hasOwnProperty(name)) { - // regionGraphic = extend({}, regionGraphic); - // regionGraphic.name = name; - // } - // result.push(regionGraphic); - // } - // return result; - // } - - var geoCoord = [126, 25]; - var nanhaiName = '南海诸岛'; - var points$1 = [[[0, 3.5], [7, 11.2], [15, 11.9], [30, 7], [42, 0.7], [52, 0.7], [56, 7.7], [59, 0.7], [64, 0.7], [64, 0], [5, 0], [0, 3.5]], [[13, 16.1], [19, 14.7], [16, 21.7], [11, 23.1], [13, 16.1]], [[12, 32.2], [14, 38.5], [15, 38.5], [13, 32.2], [12, 32.2]], [[16, 47.6], [12, 53.2], [13, 53.2], [18, 47.6], [16, 47.6]], [[6, 64.4], [8, 70], [9, 70], [8, 64.4], [6, 64.4]], [[23, 82.6], [29, 79.8], [30, 79.8], [25, 82.6], [23, 82.6]], [[37, 70.7], [43, 62.3], [44, 62.3], [39, 70.7], [37, 70.7]], [[48, 51.1], [51, 45.5], [53, 45.5], [50, 51.1], [48, 51.1]], [[51, 35], [51, 28.7], [53, 28.7], [53, 35], [51, 35]], [[52, 22.4], [55, 17.5], [56, 17.5], [53, 22.4], [52, 22.4]], [[58, 12.6], [62, 7], [63, 7], [60, 12.6], [58, 12.6]], [[0, 3.5], [0, 93.1], [64, 93.1], [64, 0], [63, 0], [63, 92.4], [1, 92.4], [1, 3.5], [0, 3.5]]]; - - for (var i = 0; i < points$1.length; i++) { - for (var k = 0; k < points$1[i].length; k++) { - points$1[i][k][0] /= 10.5; - points$1[i][k][1] /= -10.5 / 0.75; - points$1[i][k][0] += geoCoord[0]; - points$1[i][k][1] += geoCoord[1]; - } - } - - function fixNanhai(mapType, regions) { - if (mapType === 'china') { - for (var i = 0; i < regions.length; i++) { - // Already exists. - if (regions[i].name === nanhaiName) { - return; - } - } - - regions.push(new GeoJSONRegion(nanhaiName, map(points$1, function (exterior) { - return { - type: 'polygon', - exterior: exterior - }; - }), geoCoord)); - } - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var coordsOffsetMap = { - '南海诸岛': [32, 80], - // 全国 - '广东': [0, -10], - '香港': [10, 5], - '澳门': [-10, 10], - // '北京': [-10, 0], - '天津': [5, 5] - }; - function fixTextCoords(mapType, region) { - if (mapType === 'china') { - var coordFix = coordsOffsetMap[region.name]; - - if (coordFix) { - var cp = region.getCenter(); - cp[0] += coordFix[0] / 10.5; - cp[1] += -coordFix[1] / (10.5 / 0.75); - region.setCenter(cp); - } - } - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - // Fix for 钓鱼岛 - // let Region = require('../Region'); - // let zrUtil = require('zrender/lib/core/util'); - // let geoCoord = [126, 25]; - var points$2 = [[[123.45165252685547, 25.73527164402261], [123.49731445312499, 25.73527164402261], [123.49731445312499, 25.750734064600884], [123.45165252685547, 25.750734064600884], [123.45165252685547, 25.73527164402261]]]; - function fixDiaoyuIsland(mapType, region) { - if (mapType === 'china' && region.name === '台湾') { - region.geometries.push({ - type: 'polygon', - exterior: points$2[0] - }); - } - } - - var DEFAULT_NAME_PROPERTY = 'name'; - - var GeoJSONResource = - /** @class */ - function () { - function GeoJSONResource(mapName, geoJSON, specialAreas) { - this.type = 'geoJSON'; - this._parsedMap = createHashMap(); - this._mapName = mapName; - this._specialAreas = specialAreas; // PENDING: delay the parse to the first usage to rapid up the FMP? - - this._geoJSON = parseInput(geoJSON); - } - /** - * @param nameMap can be null/undefined - * @param nameProperty can be null/undefined - */ - - - GeoJSONResource.prototype.load = function (nameMap, nameProperty) { - nameProperty = nameProperty || DEFAULT_NAME_PROPERTY; - - var parsed = this._parsedMap.get(nameProperty); - - if (!parsed) { - var rawRegions = this._parseToRegions(nameProperty); - - parsed = this._parsedMap.set(nameProperty, { - regions: rawRegions, - boundingRect: calculateBoundingRect(rawRegions) - }); - } - - var regionsMap = createHashMap(); - var finalRegions = []; - each(parsed.regions, function (region) { - var regionName = region.name; // Try use the alias in geoNameMap - - if (nameMap && hasOwn(nameMap, regionName)) { - region = region.cloneShallow(regionName = nameMap[regionName]); - } - - finalRegions.push(region); - regionsMap.set(regionName, region); - }); - return { - regions: finalRegions, - boundingRect: parsed.boundingRect || new BoundingRect(0, 0, 0, 0), - regionsMap: regionsMap - }; - }; - - GeoJSONResource.prototype._parseToRegions = function (nameProperty) { - var mapName = this._mapName; - var geoJSON = this._geoJSON; - var rawRegions; // https://jsperf.com/try-catch-performance-overhead - - try { - rawRegions = geoJSON ? parseGeoJSON(geoJSON, nameProperty) : []; - } catch (e) { - throw new Error('Invalid geoJson format\n' + e.message); - } - - fixNanhai(mapName, rawRegions); - each(rawRegions, function (region) { - var regionName = region.name; - fixTextCoords(mapName, region); - fixDiaoyuIsland(mapName, region); // Some area like Alaska in USA map needs to be tansformed - // to look better - - var specialArea = this._specialAreas && this._specialAreas[regionName]; - - if (specialArea) { - region.transformTo(specialArea.left, specialArea.top, specialArea.width, specialArea.height); - } - }, this); - return rawRegions; - }; - /** - * Only for exporting to users. - * **MUST NOT** used internally. - */ - - - GeoJSONResource.prototype.getMapForUser = function () { - return { - // For backward compatibility, use geoJson - // PENDING: it has been returning them without clone. - // do we need to avoid outsite modification? - geoJson: this._geoJSON, - geoJSON: this._geoJSON, - specialAreas: this._specialAreas - }; - }; - - return GeoJSONResource; - }(); - - function calculateBoundingRect(regions) { - var rect; - - for (var i = 0; i < regions.length; i++) { - var regionRect = regions[i].getBoundingRect(); - rect = rect || regionRect.clone(); - rect.union(regionRect); - } - - return rect; - } - - function parseInput(source) { - return !isString(source) ? source : typeof JSON !== 'undefined' && JSON.parse ? JSON.parse(source) : new Function('return (' + source + ');')(); - } - - var storage = createHashMap(); - var geoSourceManager = { - /** - * Compatible with previous `echarts.registerMap`. - * - * @usage - * ```js - * - * echarts.registerMap('USA', geoJson, specialAreas); - * - * echarts.registerMap('USA', { - * geoJson: geoJson, - * specialAreas: {...} - * }); - * echarts.registerMap('USA', { - * geoJSON: geoJson, - * specialAreas: {...} - * }); - * - * echarts.registerMap('airport', { - * svg: svg - * } - * ``` - * - * Note: - * Do not support that register multiple geoJSON or SVG - * one map name. Because different geoJSON and SVG have - * different unit. It's not easy to make sure how those - * units are mapping/normalize. - * If intending to use multiple geoJSON or SVG, we can - * use multiple geo coordinate system. - */ - registerMap: function (mapName, rawDef, rawSpecialAreas) { - if (rawDef.svg) { - var resource = new GeoSVGResource(mapName, rawDef.svg); - storage.set(mapName, resource); - } else { - // Recommend: - // echarts.registerMap('eu', { geoJSON: xxx, specialAreas: xxx }); - // Backward compatibility: - // echarts.registerMap('eu', geoJSON, specialAreas); - // echarts.registerMap('eu', { geoJson: xxx, specialAreas: xxx }); - var geoJSON = rawDef.geoJson || rawDef.geoJSON; - - if (geoJSON && !rawDef.features) { - rawSpecialAreas = rawDef.specialAreas; - } else { - geoJSON = rawDef; - } - - var resource = new GeoJSONResource(mapName, geoJSON, rawSpecialAreas); - storage.set(mapName, resource); - } - }, - getGeoResource: function (mapName) { - return storage.get(mapName); - }, - - /** - * Only for exporting to users. - * **MUST NOT** used internally. - */ - getMapForUser: function (mapName) { - var resource = storage.get(mapName); // Do not support return SVG until some real requirement come. - - return resource && resource.type === 'geoJSON' && resource.getMapForUser(); - }, - load: function (mapName, nameMap, nameProperty) { - var resource = storage.get(mapName); - - if (!resource) { - if ("development" !== 'production') { - console.error('Map ' + mapName + ' not exists. The GeoJSON of the map must be provided.'); - } - - return; - } - - return resource.load(nameMap, nameProperty); - } - }; - - /** - * Only these tags enable use `itemStyle` if they are named in SVG. - * Other tags like <text> <tspan> <image> might not suitable for `itemStyle`. - * They will not be considered to be styled until some requirements come. - */ - - var OPTION_STYLE_ENABLED_TAGS = ['rect', 'circle', 'line', 'ellipse', 'polygon', 'polyline', 'path']; - var OPTION_STYLE_ENABLED_TAG_MAP = createHashMap(OPTION_STYLE_ENABLED_TAGS); - var STATE_TRIGGER_TAG_MAP = createHashMap(OPTION_STYLE_ENABLED_TAGS.concat(['g'])); - var LABEL_HOST_MAP = createHashMap(OPTION_STYLE_ENABLED_TAGS.concat(['g'])); - var mapLabelRaw = makeInner(); - - function getFixedItemStyle(model) { - var itemStyle = model.getItemStyle(); - var areaColor = model.get('areaColor'); // If user want the color not to be changed when hover, - // they should both set areaColor and color to be null. - - if (areaColor != null) { - itemStyle.fill = areaColor; - } - - return itemStyle; - } // Only stroke can be used for line. - // Using fill in style if stroke not exits. - // TODO Not sure yet. Perhaps a separate `lineStyle`? - - - function fixLineStyle(styleHost) { - var style = styleHost.style; - - if (style) { - style.stroke = style.stroke || style.fill; - style.fill = null; - } - } - - var MapDraw = - /** @class */ - function () { - function MapDraw(api) { - var group = new Group(); - this.uid = getUID('ec_map_draw'); - this._controller = new RoamController(api.getZr()); - this._controllerHost = { - target: group - }; - this.group = group; - group.add(this._regionsGroup = new Group()); - group.add(this._svgGroup = new Group()); - } - - MapDraw.prototype.draw = function (mapOrGeoModel, ecModel, api, fromView, payload) { - var isGeo = mapOrGeoModel.mainType === 'geo'; // Map series has data. GEO model that controlled by map series - // will be assigned with map data. Other GEO model has no data. - - var data = mapOrGeoModel.getData && mapOrGeoModel.getData(); - isGeo && ecModel.eachComponent({ - mainType: 'series', - subType: 'map' - }, function (mapSeries) { - if (!data && mapSeries.getHostGeoModel() === mapOrGeoModel) { - data = mapSeries.getData(); - } - }); - var geo = mapOrGeoModel.coordinateSystem; - var regionsGroup = this._regionsGroup; - var group = this.group; - var transformInfo = geo.getTransformInfo(); - var transformInfoRaw = transformInfo.raw; - var transformInfoRoam = transformInfo.roam; // No animation when first draw or in action - - var isFirstDraw = !regionsGroup.childAt(0) || payload; - - if (isFirstDraw) { - group.x = transformInfoRoam.x; - group.y = transformInfoRoam.y; - group.scaleX = transformInfoRoam.scaleX; - group.scaleY = transformInfoRoam.scaleY; - group.dirty(); - } else { - updateProps(group, transformInfoRoam, mapOrGeoModel); - } - - var isVisualEncodedByVisualMap = data && data.getVisual('visualMeta') && data.getVisual('visualMeta').length > 0; - var viewBuildCtx = { - api: api, - geo: geo, - mapOrGeoModel: mapOrGeoModel, - data: data, - isVisualEncodedByVisualMap: isVisualEncodedByVisualMap, - isGeo: isGeo, - transformInfoRaw: transformInfoRaw - }; - - if (geo.resourceType === 'geoJSON') { - this._buildGeoJSON(viewBuildCtx); - } else if (geo.resourceType === 'geoSVG') { - this._buildSVG(viewBuildCtx); - } - - this._updateController(mapOrGeoModel, ecModel, api); - - this._updateMapSelectHandler(mapOrGeoModel, regionsGroup, api, fromView); - }; - - MapDraw.prototype._buildGeoJSON = function (viewBuildCtx) { - var regionsGroupByName = this._regionsGroupByName = createHashMap(); - var regionsInfoByName = createHashMap(); - var regionsGroup = this._regionsGroup; - var transformInfoRaw = viewBuildCtx.transformInfoRaw; - var mapOrGeoModel = viewBuildCtx.mapOrGeoModel; - var data = viewBuildCtx.data; - var projection = viewBuildCtx.geo.projection; - var projectionStream = projection && projection.stream; - - function transformPoint(point, project) { - if (project) { - // projection may return null point. - point = project(point); - } - - return point && [point[0] * transformInfoRaw.scaleX + transformInfoRaw.x, point[1] * transformInfoRaw.scaleY + transformInfoRaw.y]; - } - - function transformPolygonPoints(inPoints) { - var outPoints = []; // If projectionStream is provided. Use it instead of single point project. - - var project = !projectionStream && projection && projection.project; - - for (var i = 0; i < inPoints.length; ++i) { - var newPt = transformPoint(inPoints[i], project); - newPt && outPoints.push(newPt); - } - - return outPoints; - } - - function getPolyShape(points) { - return { - shape: { - points: transformPolygonPoints(points) - } - }; - } - - regionsGroup.removeAll(); // Only when the resource is GeoJSON, there is `geo.regions`. - - each(viewBuildCtx.geo.regions, function (region) { - var regionName = region.name; // Consider in GeoJson properties.name may be duplicated, for example, - // there is multiple region named "United Kindom" or "France" (so many - // colonies). And it is not appropriate to merge them in geo, which - // will make them share the same label and bring trouble in label - // location calculation. - - var regionGroup = regionsGroupByName.get(regionName); - - var _a = regionsInfoByName.get(regionName) || {}, - dataIdx = _a.dataIdx, - regionModel = _a.regionModel; - - if (!regionGroup) { - regionGroup = regionsGroupByName.set(regionName, new Group()); - regionsGroup.add(regionGroup); - dataIdx = data ? data.indexOfName(regionName) : null; - regionModel = viewBuildCtx.isGeo ? mapOrGeoModel.getRegionModel(regionName) : data ? data.getItemModel(dataIdx) : null; - regionsInfoByName.set(regionName, { - dataIdx: dataIdx, - regionModel: regionModel - }); - } - - var polygonSubpaths = []; - var polylineSubpaths = []; - each(region.geometries, function (geometry) { - // Polygon and MultiPolygon - if (geometry.type === 'polygon') { - var polys = [geometry.exterior].concat(geometry.interiors || []); - - if (projectionStream) { - polys = projectPolys(polys, projectionStream); - } - - each(polys, function (poly) { - polygonSubpaths.push(new Polygon(getPolyShape(poly))); - }); - } // LineString and MultiLineString - else { - var points = geometry.points; - - if (projectionStream) { - points = projectPolys(points, projectionStream, true); - } - - each(points, function (points) { - polylineSubpaths.push(new Polyline(getPolyShape(points))); - }); - } - }); - var centerPt = transformPoint(region.getCenter(), projection && projection.project); - - function createCompoundPath(subpaths, isLine) { - if (!subpaths.length) { - return; - } - - var compoundPath = new CompoundPath({ - culling: true, - segmentIgnoreThreshold: 1, - shape: { - paths: subpaths - } - }); - regionGroup.add(compoundPath); - applyOptionStyleForRegion(viewBuildCtx, compoundPath, dataIdx, regionModel); - resetLabelForRegion(viewBuildCtx, compoundPath, regionName, regionModel, mapOrGeoModel, dataIdx, centerPt); - - if (isLine) { - fixLineStyle(compoundPath); - each(compoundPath.states, fixLineStyle); - } - } - - createCompoundPath(polygonSubpaths); - createCompoundPath(polylineSubpaths, true); - }); // Ensure children have been added to `regionGroup` before calling them. - - regionsGroupByName.each(function (regionGroup, regionName) { - var _a = regionsInfoByName.get(regionName), - dataIdx = _a.dataIdx, - regionModel = _a.regionModel; - - resetEventTriggerForRegion(viewBuildCtx, regionGroup, regionName, regionModel, mapOrGeoModel, dataIdx); - resetTooltipForRegion(viewBuildCtx, regionGroup, regionName, regionModel, mapOrGeoModel); - resetStateTriggerForRegion(viewBuildCtx, regionGroup, regionName, regionModel, mapOrGeoModel); - }, this); - }; - - MapDraw.prototype._buildSVG = function (viewBuildCtx) { - var mapName = viewBuildCtx.geo.map; - var transformInfoRaw = viewBuildCtx.transformInfoRaw; - this._svgGroup.x = transformInfoRaw.x; - this._svgGroup.y = transformInfoRaw.y; - this._svgGroup.scaleX = transformInfoRaw.scaleX; - this._svgGroup.scaleY = transformInfoRaw.scaleY; - - if (this._svgResourceChanged(mapName)) { - this._freeSVG(); - - this._useSVG(mapName); - } - - var svgDispatcherMap = this._svgDispatcherMap = createHashMap(); - var focusSelf = false; - each(this._svgGraphicRecord.named, function (namedItem) { - // Note that we also allow different elements have the same name. - // For example, a glyph of a city and the label of the city have - // the same name and their tooltip info can be defined in a single - // region option. - var regionName = namedItem.name; - var mapOrGeoModel = viewBuildCtx.mapOrGeoModel; - var data = viewBuildCtx.data; - var svgNodeTagLower = namedItem.svgNodeTagLower; - var el = namedItem.el; - var dataIdx = data ? data.indexOfName(regionName) : null; - var regionModel = mapOrGeoModel.getRegionModel(regionName); - - if (OPTION_STYLE_ENABLED_TAG_MAP.get(svgNodeTagLower) != null && el instanceof Displayable) { - applyOptionStyleForRegion(viewBuildCtx, el, dataIdx, regionModel); - } - - if (el instanceof Displayable) { - el.culling = true; - } // We do not know how the SVG like so we'd better not to change z2. - // Otherwise it might bring some unexpected result. For example, - // an area hovered that make some inner city can not be clicked. - - - el.z2EmphasisLift = 0; // If self named: - - if (!namedItem.namedFrom) { - // label should batter to be displayed based on the center of <g> - // if it is named rather than displayed on each child. - if (LABEL_HOST_MAP.get(svgNodeTagLower) != null) { - resetLabelForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel, dataIdx, null); - } - - resetEventTriggerForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel, dataIdx); - resetTooltipForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel); - - if (STATE_TRIGGER_TAG_MAP.get(svgNodeTagLower) != null) { - var focus_1 = resetStateTriggerForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel); - - if (focus_1 === 'self') { - focusSelf = true; - } - - var els = svgDispatcherMap.get(regionName) || svgDispatcherMap.set(regionName, []); - els.push(el); - } - } - }, this); - - this._enableBlurEntireSVG(focusSelf, viewBuildCtx); - }; - - MapDraw.prototype._enableBlurEntireSVG = function (focusSelf, viewBuildCtx) { - // It's a little complicated to support blurring the entire geoSVG in series-map. - // So do not support it until some requirements come. - // At present, in series-map, only regions can be blurred. - if (focusSelf && viewBuildCtx.isGeo) { - var blurStyle = viewBuildCtx.mapOrGeoModel.getModel(['blur', 'itemStyle']).getItemStyle(); // Only support `opacity` here. Because not sure that other props are suitable for - // all of the elements generated by SVG (especially for Text/TSpan/Image/... ). - - var opacity_1 = blurStyle.opacity; - - this._svgGraphicRecord.root.traverse(function (el) { - if (!el.isGroup) { - // PENDING: clear those settings to SVG elements when `_freeSVG`. - // (Currently it happen not to be needed.) - setDefaultStateProxy(el); - var style = el.ensureState('blur').style || {}; // Do not overwrite the region style that already set from region option. - - if (style.opacity == null && opacity_1 != null) { - style.opacity = opacity_1; - } // If `ensureState('blur').style = {}`, there will be default opacity. - // Enable `stateTransition` (animation). - - - el.ensureState('emphasis'); - } - }); - } - }; - - MapDraw.prototype.remove = function () { - this._regionsGroup.removeAll(); - - this._regionsGroupByName = null; - - this._svgGroup.removeAll(); - - this._freeSVG(); - - this._controller.dispose(); - - this._controllerHost = null; - }; - - MapDraw.prototype.findHighDownDispatchers = function (name, geoModel) { - if (name == null) { - return []; - } - - var geo = geoModel.coordinateSystem; - - if (geo.resourceType === 'geoJSON') { - var regionsGroupByName = this._regionsGroupByName; - - if (regionsGroupByName) { - var regionGroup = regionsGroupByName.get(name); - return regionGroup ? [regionGroup] : []; - } - } else if (geo.resourceType === 'geoSVG') { - return this._svgDispatcherMap && this._svgDispatcherMap.get(name) || []; - } - }; - - MapDraw.prototype._svgResourceChanged = function (mapName) { - return this._svgMapName !== mapName; - }; - - MapDraw.prototype._useSVG = function (mapName) { - var resource = geoSourceManager.getGeoResource(mapName); - - if (resource && resource.type === 'geoSVG') { - var svgGraphic = resource.useGraphic(this.uid); - - this._svgGroup.add(svgGraphic.root); - - this._svgGraphicRecord = svgGraphic; - this._svgMapName = mapName; - } - }; - - MapDraw.prototype._freeSVG = function () { - var mapName = this._svgMapName; - - if (mapName == null) { - return; - } - - var resource = geoSourceManager.getGeoResource(mapName); - - if (resource && resource.type === 'geoSVG') { - resource.freeGraphic(this.uid); - } - - this._svgGraphicRecord = null; - this._svgDispatcherMap = null; - - this._svgGroup.removeAll(); - - this._svgMapName = null; - }; - - MapDraw.prototype._updateController = function (mapOrGeoModel, ecModel, api) { - var geo = mapOrGeoModel.coordinateSystem; - var controller = this._controller; - var controllerHost = this._controllerHost; // @ts-ignore FIXME:TS - - controllerHost.zoomLimit = mapOrGeoModel.get('scaleLimit'); - controllerHost.zoom = geo.getZoom(); // roamType is will be set default true if it is null - // @ts-ignore FIXME:TS - - controller.enable(mapOrGeoModel.get('roam') || false); - var mainType = mapOrGeoModel.mainType; - - function makeActionBase() { - var action = { - type: 'geoRoam', - componentType: mainType - }; - action[mainType + 'Id'] = mapOrGeoModel.id; - return action; - } - - controller.off('pan').on('pan', function (e) { - this._mouseDownFlag = false; - updateViewOnPan(controllerHost, e.dx, e.dy); - api.dispatchAction(extend(makeActionBase(), { - dx: e.dx, - dy: e.dy, - animation: { - duration: 0 - } - })); - }, this); - controller.off('zoom').on('zoom', function (e) { - this._mouseDownFlag = false; - updateViewOnZoom(controllerHost, e.scale, e.originX, e.originY); - api.dispatchAction(extend(makeActionBase(), { - zoom: e.scale, - originX: e.originX, - originY: e.originY, - animation: { - duration: 0 - } - })); - }, this); - controller.setPointerChecker(function (e, x, y) { - return geo.containPoint([x, y]) && !onIrrelevantElement(e, api, mapOrGeoModel); - }); - }; - /** - * FIXME: this is a temporarily workaround. - * When `geoRoam` the elements need to be reset in `MapView['render']`, because the props like - * `ignore` might have been modified by `LabelManager`, and `LabelManager#addLabelsOfSeries` - * will subsequently cache `defaultAttr` like `ignore`. If do not do this reset, the modified - * props will have no chance to be restored. - * Note: This reset should be after `clearStates` in `renderSeries` because `useStates` in - * `renderSeries` will cache the modified `ignore` to `el._normalState`. - * TODO: - * Use clone/immutable in `LabelManager`? - */ - - - MapDraw.prototype.resetForLabelLayout = function () { - this.group.traverse(function (el) { - var label = el.getTextContent(); - - if (label) { - label.ignore = mapLabelRaw(label).ignore; - } - }); - }; - - MapDraw.prototype._updateMapSelectHandler = function (mapOrGeoModel, regionsGroup, api, fromView) { - var mapDraw = this; - regionsGroup.off('mousedown'); - regionsGroup.off('click'); // @ts-ignore FIXME:TS resolve type conflict - - if (mapOrGeoModel.get('selectedMode')) { - regionsGroup.on('mousedown', function () { - mapDraw._mouseDownFlag = true; - }); - regionsGroup.on('click', function (e) { - if (!mapDraw._mouseDownFlag) { - return; - } - - mapDraw._mouseDownFlag = false; - }); - } - }; - - return MapDraw; - }(); - - function applyOptionStyleForRegion(viewBuildCtx, el, dataIndex, regionModel) { - // All of the path are using `itemStyle`, because - // (1) Some SVG also use fill on polyline (The different between - // polyline and polygon is "open" or "close" but not fill or not). - // (2) For the common props like opacity, if some use itemStyle - // and some use `lineStyle`, it might confuse users. - // (3) Most SVG use <path>, where can not detect whether to draw a "line" - // or a filled shape, so use `itemStyle` for <path>. - var normalStyleModel = regionModel.getModel('itemStyle'); - var emphasisStyleModel = regionModel.getModel(['emphasis', 'itemStyle']); - var blurStyleModel = regionModel.getModel(['blur', 'itemStyle']); - var selectStyleModel = regionModel.getModel(['select', 'itemStyle']); // NOTE: DON'T use 'style' in visual when drawing map. - // This component is used for drawing underlying map for both geo component and map series. - - var normalStyle = getFixedItemStyle(normalStyleModel); - var emphasisStyle = getFixedItemStyle(emphasisStyleModel); - var selectStyle = getFixedItemStyle(selectStyleModel); - var blurStyle = getFixedItemStyle(blurStyleModel); // Update the itemStyle if has data visual - - var data = viewBuildCtx.data; - - if (data) { - // Only visual color of each item will be used. It can be encoded by visualMap - // But visual color of series is used in symbol drawing - // Visual color for each series is for the symbol draw - var style = data.getItemVisual(dataIndex, 'style'); - var decal = data.getItemVisual(dataIndex, 'decal'); - - if (viewBuildCtx.isVisualEncodedByVisualMap && style.fill) { - normalStyle.fill = style.fill; - } - - if (decal) { - normalStyle.decal = createOrUpdatePatternFromDecal(decal, viewBuildCtx.api); - } - } // SVG text, tspan and image can be named but not supporeted - // to be styled by region option yet. - - - el.setStyle(normalStyle); - el.style.strokeNoScale = true; - el.ensureState('emphasis').style = emphasisStyle; - el.ensureState('select').style = selectStyle; - el.ensureState('blur').style = blurStyle; // Enable blur - - setDefaultStateProxy(el); - } - - function resetLabelForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel, // Exist only if `viewBuildCtx.data` exists. - dataIdx, // If labelXY not provided, use `textConfig.position: 'inside'` - labelXY) { - var data = viewBuildCtx.data; - var isGeo = viewBuildCtx.isGeo; - var isDataNaN = data && isNaN(data.get(data.mapDimension('value'), dataIdx)); - var itemLayout = data && data.getItemLayout(dataIdx); // In the following cases label will be drawn - // 1. In map series and data value is NaN - // 2. In geo component - // 3. Region has no series legendIcon, which will be add a showLabel flag in mapSymbolLayout - - if (isGeo || isDataNaN || itemLayout && itemLayout.showLabel) { - var query = !isGeo ? dataIdx : regionName; - var labelFetcher = void 0; // Consider dataIdx not found. - - if (!data || dataIdx >= 0) { - labelFetcher = mapOrGeoModel; - } - - var specifiedTextOpt = labelXY ? { - normal: { - align: 'center', - verticalAlign: 'middle' - } - } : null; // Caveat: must be called after `setDefaultStateProxy(el);` called. - // because textContent will be assign with `el.stateProxy` inside. - - setLabelStyle(el, getLabelStatesModels(regionModel), { - labelFetcher: labelFetcher, - labelDataIndex: query, - defaultText: regionName - }, specifiedTextOpt); - var textEl = el.getTextContent(); - - if (textEl) { - mapLabelRaw(textEl).ignore = textEl.ignore; - - if (el.textConfig && labelXY) { - // Compute a relative offset based on the el bounding rect. - var rect = el.getBoundingRect().clone(); // Need to make sure the percent position base on the same rect in normal and - // emphasis state. Otherwise if using boundingRect of el, but the emphasis state - // has borderWidth (even 0.5px), the text position will be changed obviously - // if the position is very big like ['1234%', '1345%']. - - el.textConfig.layoutRect = rect; - el.textConfig.position = [(labelXY[0] - rect.x) / rect.width * 100 + '%', (labelXY[1] - rect.y) / rect.height * 100 + '%']; - } - } // PENDING: - // If labelLayout is enabled (test/label-layout.html), el.dataIndex should be specified. - // But el.dataIndex is also used to determine whether user event should be triggered, - // where el.seriesIndex or el.dataModel must be specified. At present for a single el - // there is not case that "only label layout enabled but user event disabled", so here - // we depends `resetEventTriggerForRegion` to do the job of setting `el.dataIndex`. - - - el.disableLabelAnimation = true; - } else { - el.removeTextContent(); - el.removeTextConfig(); - el.disableLabelAnimation = null; - } - } - - function resetEventTriggerForRegion(viewBuildCtx, eventTrigger, regionName, regionModel, mapOrGeoModel, // Exist only if `viewBuildCtx.data` exists. - dataIdx) { - // setItemGraphicEl, setHoverStyle after all polygons and labels - // are added to the regionGroup - if (viewBuildCtx.data) { - // FIXME: when series-map use a SVG map, and there are duplicated name specified - // on different SVG elements, after `data.setItemGraphicEl(...)`: - // (1) all of them will be mounted with `dataIndex`, `seriesIndex`, so that tooltip - // can be triggered only mouse hover. That's correct. - // (2) only the last element will be kept in `data`, so that if trigger tooltip - // by `dispatchAction`, only the last one can be found and triggered. That might be - // not correct. We will fix it in future if anyone demanding that. - viewBuildCtx.data.setItemGraphicEl(dataIdx, eventTrigger); - } // series-map will not trigger "geoselectchange" no matter it is - // based on a declared geo component. Because series-map will - // trigger "selectchange". If it trigger both the two events, - // If users call `chart.dispatchAction({type: 'toggleSelect'})`, - // it not easy to also fire event "geoselectchanged". - else { - // Package custom mouse event for geo component - getECData(eventTrigger).eventData = { - componentType: 'geo', - componentIndex: mapOrGeoModel.componentIndex, - geoIndex: mapOrGeoModel.componentIndex, - name: regionName, - region: regionModel && regionModel.option || {} - }; - } - } - - function resetTooltipForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel) { - if (!viewBuildCtx.data) { - setTooltipConfig({ - el: el, - componentModel: mapOrGeoModel, - itemName: regionName, - // @ts-ignore FIXME:TS fix the "compatible with each other"? - itemTooltipOption: regionModel.get('tooltip') - }); - } - } - - function resetStateTriggerForRegion(viewBuildCtx, el, regionName, regionModel, mapOrGeoModel) { - // @ts-ignore FIXME:TS fix the "compatible with each other"? - el.highDownSilentOnTouch = !!mapOrGeoModel.get('selectedMode'); // @ts-ignore FIXME:TS fix the "compatible with each other"? - - var emphasisModel = regionModel.getModel('emphasis'); - var focus = emphasisModel.get('focus'); - toggleHoverEmphasis(el, focus, emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - - if (viewBuildCtx.isGeo) { - enableComponentHighDownFeatures(el, mapOrGeoModel, regionName); - } - - return focus; - } - - function projectPolys(rings, // Polygons include exterior and interiors. Or polylines. - createStream, isLine) { - var polygons = []; - var curPoly; - - function startPolygon() { - curPoly = []; - } - - function endPolygon() { - if (curPoly.length) { - polygons.push(curPoly); - curPoly = []; - } - } - - var stream = createStream({ - polygonStart: startPolygon, - polygonEnd: endPolygon, - lineStart: startPolygon, - lineEnd: endPolygon, - point: function (x, y) { - // May have NaN values from stream. - if (isFinite(x) && isFinite(y)) { - curPoly.push([x, y]); - } - }, - sphere: function () {} - }); - !isLine && stream.polygonStart(); - each(rings, function (ring) { - stream.lineStart(); - - for (var i = 0; i < ring.length; i++) { - stream.point(ring[i][0], ring[i][1]); - } - - stream.lineEnd(); - }); - !isLine && stream.polygonEnd(); - return polygons; - } - // @ts-ignore FIXME:TS fix the "compatible with each other"? - - var MapView = - /** @class */ - function (_super) { - __extends(MapView, _super); - - function MapView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MapView.type; - return _this; - } - - MapView.prototype.render = function (mapModel, ecModel, api, payload) { - // Not render if it is an toggleSelect action from self - if (payload && payload.type === 'mapToggleSelect' && payload.from === this.uid) { - return; - } - - var group = this.group; - group.removeAll(); - - if (mapModel.getHostGeoModel()) { - return; - } - - if (this._mapDraw && payload && payload.type === 'geoRoam') { - this._mapDraw.resetForLabelLayout(); - } // Not update map if it is an roam action from self - - - if (!(payload && payload.type === 'geoRoam' && payload.componentType === 'series' && payload.seriesId === mapModel.id)) { - if (mapModel.needsDrawMap) { - var mapDraw = this._mapDraw || new MapDraw(api); - group.add(mapDraw.group); - mapDraw.draw(mapModel, ecModel, api, this, payload); - this._mapDraw = mapDraw; - } else { - // Remove drawn map - this._mapDraw && this._mapDraw.remove(); - this._mapDraw = null; - } - } else { - var mapDraw = this._mapDraw; - mapDraw && group.add(mapDraw.group); - } - - mapModel.get('showLegendSymbol') && ecModel.getComponent('legend') && this._renderSymbols(mapModel, ecModel, api); - }; - - MapView.prototype.remove = function () { - this._mapDraw && this._mapDraw.remove(); - this._mapDraw = null; - this.group.removeAll(); - }; - - MapView.prototype.dispose = function () { - this._mapDraw && this._mapDraw.remove(); - this._mapDraw = null; - }; - - MapView.prototype._renderSymbols = function (mapModel, ecModel, api) { - var originalData = mapModel.originalData; - var group = this.group; - originalData.each(originalData.mapDimension('value'), function (value, originalDataIndex) { - if (isNaN(value)) { - return; - } - - var layout = originalData.getItemLayout(originalDataIndex); - - if (!layout || !layout.point) { - // Not exists in map - return; - } - - var point = layout.point; - var offset = layout.offset; - var circle = new Circle({ - style: { - // Because the special of map draw. - // Which needs statistic of multiple series and draw on one map. - // And each series also need a symbol with legend color - // - // Layout and visual are put one the different data - // TODO - fill: mapModel.getData().getVisual('style').fill - }, - shape: { - cx: point[0] + offset * 9, - cy: point[1], - r: 3 - }, - silent: true, - // Do not overlap the first series, on which labels are displayed. - z2: 8 + (!offset ? Z2_EMPHASIS_LIFT + 1 : 0) - }); // Only the series that has the first value on the same region is in charge of rendering the label. - // But consider the case: - // series: [ - // {id: 'X', type: 'map', map: 'm', {data: [{name: 'A', value: 11}, {name: 'B', {value: 22}]}, - // {id: 'Y', type: 'map', map: 'm', {data: [{name: 'A', value: 21}, {name: 'C', {value: 33}]} - // ] - // The offset `0` of item `A` is at series `X`, but of item `C` is at series `Y`. - // For backward compatibility, we follow the rule that render label `A` by the - // settings on series `X` but render label `C` by the settings on series `Y`. - - if (!offset) { - var fullData = mapModel.mainSeries.getData(); - var name_1 = originalData.getName(originalDataIndex); - var fullIndex_1 = fullData.indexOfName(name_1); - var itemModel = originalData.getItemModel(originalDataIndex); - var labelModel = itemModel.getModel('label'); - var regionGroup = fullData.getItemGraphicEl(fullIndex_1); // `getFormattedLabel` needs to use `getData` inside. Here - // `mapModel.getData()` is shallow cloned from `mainSeries.getData()`. - // FIXME - // If this is not the `mainSeries`, the item model (like label formatter) - // set on original data item will never get. But it has been working - // like that from the beginning, and this scenario is rarely encountered. - // So it won't be fixed until we have to. - - setLabelStyle(circle, getLabelStatesModels(itemModel), { - labelFetcher: { - getFormattedLabel: function (idx, state) { - return mapModel.getFormattedLabel(fullIndex_1, state); - } - }, - defaultText: name_1 - }); - circle.disableLabelAnimation = true; - - if (!labelModel.get('position')) { - circle.setTextConfig({ - position: 'bottom' - }); - } - - regionGroup.onHoverStateChange = function (toState) { - setStatesFlag(circle, toState); - }; - } - - group.add(circle); - }); - }; - - MapView.type = 'map'; - return MapView; - }(ChartView); - - var MapSeries = - /** @class */ - function (_super) { - __extends(MapSeries, _super); - - function MapSeries() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MapSeries.type; // Only first map series of same mapType will drawMap. - - _this.needsDrawMap = false; // Group of all map series with same mapType - - _this.seriesGroup = []; - - _this.getTooltipPosition = function (dataIndex) { - if (dataIndex != null) { - var name_1 = this.getData().getName(dataIndex); - var geo = this.coordinateSystem; - var region = geo.getRegion(name_1); - return region && geo.dataToPoint(region.getCenter()); - } - }; - - return _this; - } - - MapSeries.prototype.getInitialData = function (option) { - var data = createSeriesDataSimply(this, { - coordDimensions: ['value'], - encodeDefaulter: curry(makeSeriesEncodeForNameBased, this) - }); - var dataNameMap = createHashMap(); - var toAppendNames = []; - - for (var i = 0, len = data.count(); i < len; i++) { - var name_2 = data.getName(i); - dataNameMap.set(name_2, true); - } - - var geoSource = geoSourceManager.load(this.getMapType(), this.option.nameMap, this.option.nameProperty); - each(geoSource.regions, function (region) { - var name = region.name; - - if (!dataNameMap.get(name)) { - toAppendNames.push(name); - } - }); // Complete data with missing regions. The consequent processes (like visual - // map and render) can not be performed without a "full data". For example, - // find `dataIndex` by name. - - data.appendValues([], toAppendNames); - return data; - }; - /** - * If no host geo model, return null, which means using a - * inner exclusive geo model. - */ - - - MapSeries.prototype.getHostGeoModel = function () { - var geoIndex = this.option.geoIndex; - return geoIndex != null ? this.ecModel.getComponent('geo', geoIndex) : null; - }; - - MapSeries.prototype.getMapType = function () { - return (this.getHostGeoModel() || this).option.map; - }; // _fillOption(option, mapName) { - // Shallow clone - // option = zrUtil.extend({}, option); - // option.data = geoCreator.getFilledRegions(option.data, mapName, option.nameMap); - // return option; - // } - - - MapSeries.prototype.getRawValue = function (dataIndex) { - // Use value stored in data instead because it is calculated from multiple series - // FIXME Provide all value of multiple series ? - var data = this.getData(); - return data.get(data.mapDimension('value'), dataIndex); - }; - /** - * Get model of region - */ - - - MapSeries.prototype.getRegionModel = function (regionName) { - var data = this.getData(); - return data.getItemModel(data.indexOfName(regionName)); - }; - /** - * Map tooltip formatter - */ - - - MapSeries.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - // FIXME orignalData and data is a bit confusing - var data = this.getData(); - var value = this.getRawValue(dataIndex); - var name = data.getName(dataIndex); - var seriesGroup = this.seriesGroup; - var seriesNames = []; - - for (var i = 0; i < seriesGroup.length; i++) { - var otherIndex = seriesGroup[i].originalData.indexOfName(name); - var valueDim = data.mapDimension('value'); - - if (!isNaN(seriesGroup[i].originalData.get(valueDim, otherIndex))) { - seriesNames.push(seriesGroup[i].name); - } - } - - return createTooltipMarkup('section', { - header: seriesNames.join(', '), - noHeader: !seriesNames.length, - blocks: [createTooltipMarkup('nameValue', { - name: name, - value: value - })] - }); - }; - - MapSeries.prototype.setZoom = function (zoom) { - this.option.zoom = zoom; - }; - - MapSeries.prototype.setCenter = function (center) { - this.option.center = center; - }; - - MapSeries.prototype.getLegendIcon = function (opt) { - var iconType = opt.icon || 'roundRect'; - var icon = createSymbol(iconType, 0, 0, opt.itemWidth, opt.itemHeight, opt.itemStyle.fill); - icon.setStyle(opt.itemStyle); // Map do not use itemStyle.borderWidth as border width - - icon.style.stroke = 'none'; // No rotation because no series visual symbol for map - - if (iconType.indexOf('empty') > -1) { - icon.style.stroke = icon.style.fill; - icon.style.fill = '#fff'; - icon.style.lineWidth = 2; - } - - return icon; - }; - - MapSeries.type = 'series.map'; - MapSeries.dependencies = ['geo']; - MapSeries.layoutMode = 'box'; - MapSeries.defaultOption = { - // 一级层叠 - // zlevel: 0, - // 二级层叠 - z: 2, - coordinateSystem: 'geo', - // map should be explicitly specified since ec3. - map: '', - // If `geoIndex` is not specified, a exclusive geo will be - // created. Otherwise use the specified geo component, and - // `map` and `mapType` are ignored. - // geoIndex: 0, - // 'center' | 'left' | 'right' | 'x%' | {number} - left: 'center', - // 'center' | 'top' | 'bottom' | 'x%' | {number} - top: 'center', - // right - // bottom - // width: - // height - // Aspect is width / height. Inited to be geoJson bbox aspect - // This parameter is used for scale this aspect - // Default value: - // for geoSVG source: 1, - // for geoJSON source: 0.75. - aspectScale: null, - // Layout with center and size - // If you want to put map in a fixed size box with right aspect ratio - // This two properties may be more convenient. - // layoutCenter: [50%, 50%] - // layoutSize: 100 - showLegendSymbol: true, - // Define left-top, right-bottom coords to control view - // For example, [ [180, 90], [-180, -90] ], - // higher priority than center and zoom - boundingCoords: null, - // Default on center of map - center: null, - zoom: 1, - scaleLimit: null, - selectedMode: true, - label: { - show: false, - color: '#000' - }, - // scaleLimit: null, - itemStyle: { - borderWidth: 0.5, - borderColor: '#444', - areaColor: '#eee' - }, - emphasis: { - label: { - show: true, - color: 'rgb(100,0,0)' - }, - itemStyle: { - areaColor: 'rgba(255,215,0,0.8)' - } - }, - select: { - label: { - show: true, - color: 'rgb(100,0,0)' - }, - itemStyle: { - color: 'rgba(255,215,0,0.8)' - } - }, - nameProperty: 'name' - }; - return MapSeries; - }(SeriesModel); - - function dataStatistics(datas, statisticType) { - var dataNameMap = {}; - each(datas, function (data) { - data.each(data.mapDimension('value'), function (value, idx) { - // Add prefix to avoid conflict with Object.prototype. - var mapKey = 'ec-' + data.getName(idx); - dataNameMap[mapKey] = dataNameMap[mapKey] || []; - - if (!isNaN(value)) { - dataNameMap[mapKey].push(value); - } - }); - }); - return datas[0].map(datas[0].mapDimension('value'), function (value, idx) { - var mapKey = 'ec-' + datas[0].getName(idx); - var sum = 0; - var min = Infinity; - var max = -Infinity; - var len = dataNameMap[mapKey].length; - - for (var i = 0; i < len; i++) { - min = Math.min(min, dataNameMap[mapKey][i]); - max = Math.max(max, dataNameMap[mapKey][i]); - sum += dataNameMap[mapKey][i]; - } - - var result; - - if (statisticType === 'min') { - result = min; - } else if (statisticType === 'max') { - result = max; - } else if (statisticType === 'average') { - result = sum / len; - } else { - result = sum; - } - - return len === 0 ? NaN : result; - }); - } - - function mapDataStatistic(ecModel) { - var seriesGroups = {}; - ecModel.eachSeriesByType('map', function (seriesModel) { - var hostGeoModel = seriesModel.getHostGeoModel(); - var key = hostGeoModel ? 'o' + hostGeoModel.id : 'i' + seriesModel.getMapType(); - (seriesGroups[key] = seriesGroups[key] || []).push(seriesModel); - }); - each(seriesGroups, function (seriesList, key) { - var data = dataStatistics(map(seriesList, function (seriesModel) { - return seriesModel.getData(); - }), seriesList[0].get('mapValueCalculation')); - - for (var i = 0; i < seriesList.length; i++) { - seriesList[i].originalData = seriesList[i].getData(); - } // FIXME Put where? - - - for (var i = 0; i < seriesList.length; i++) { - seriesList[i].seriesGroup = seriesList; - seriesList[i].needsDrawMap = i === 0 && !seriesList[i].getHostGeoModel(); - seriesList[i].setData(data.cloneShallow()); - seriesList[i].mainSeries = seriesList[0]; - } - }); - } - - function mapSymbolLayout(ecModel) { - var processedMapType = {}; - ecModel.eachSeriesByType('map', function (mapSeries) { - var mapType = mapSeries.getMapType(); - - if (mapSeries.getHostGeoModel() || processedMapType[mapType]) { - return; - } - - var mapSymbolOffsets = {}; - each(mapSeries.seriesGroup, function (subMapSeries) { - var geo = subMapSeries.coordinateSystem; - var data = subMapSeries.originalData; - - if (subMapSeries.get('showLegendSymbol') && ecModel.getComponent('legend')) { - data.each(data.mapDimension('value'), function (value, idx) { - var name = data.getName(idx); - var region = geo.getRegion(name); // If input series.data is [11, 22, '-'/null/undefined, 44], - // it will be filled with NaN: [11, 22, NaN, 44] and NaN will - // not be drawn. So here must validate if value is NaN. - - if (!region || isNaN(value)) { - return; - } - - var offset = mapSymbolOffsets[name] || 0; - var point = geo.dataToPoint(region.getCenter()); - mapSymbolOffsets[name] = offset + 1; - data.setItemLayout(idx, { - point: point, - offset: offset - }); - }); - } - }); // Show label of those region not has legendIcon (which is offset 0) - - var data = mapSeries.getData(); - data.each(function (idx) { - var name = data.getName(idx); - var layout = data.getItemLayout(idx) || {}; - layout.showLabel = !mapSymbolOffsets[name]; - data.setItemLayout(idx, layout); - }); - processedMapType[mapType] = true; - }); - } - - var v2ApplyTransform = applyTransform; - - var View = - /** @class */ - function (_super) { - __extends(View, _super); - - function View(name) { - var _this = _super.call(this) || this; - - _this.type = 'view'; - _this.dimensions = ['x', 'y']; - /** - * Represents the transform brought by roam/zoom. - * If `View['_viewRect']` applies roam transform, - * we can get the final displayed rect. - */ - - _this._roamTransformable = new Transformable(); - /** - * Represents the transform from `View['_rect']` to `View['_viewRect']`. - */ - - _this._rawTransformable = new Transformable(); - _this.name = name; - return _this; - } - - View.prototype.setBoundingRect = function (x, y, width, height) { - this._rect = new BoundingRect(x, y, width, height); - return this._rect; - }; - /** - * @return {module:zrender/core/BoundingRect} - */ - - - View.prototype.getBoundingRect = function () { - return this._rect; - }; - - View.prototype.setViewRect = function (x, y, width, height) { - this._transformTo(x, y, width, height); - - this._viewRect = new BoundingRect(x, y, width, height); - }; - /** - * Transformed to particular position and size - */ - - - View.prototype._transformTo = function (x, y, width, height) { - var rect = this.getBoundingRect(); - var rawTransform = this._rawTransformable; - rawTransform.transform = rect.calculateTransform(new BoundingRect(x, y, width, height)); - var rawParent = rawTransform.parent; - rawTransform.parent = null; - rawTransform.decomposeTransform(); - rawTransform.parent = rawParent; - - this._updateTransform(); - }; - /** - * Set center of view - */ - - - View.prototype.setCenter = function (centerCoord, api) { - if (!centerCoord) { - return; - } - - this._center = [parsePercent$1(centerCoord[0], api.getWidth()), parsePercent$1(centerCoord[1], api.getHeight())]; - - this._updateCenterAndZoom(); - }; - - View.prototype.setZoom = function (zoom) { - zoom = zoom || 1; - var zoomLimit = this.zoomLimit; - - if (zoomLimit) { - if (zoomLimit.max != null) { - zoom = Math.min(zoomLimit.max, zoom); - } - - if (zoomLimit.min != null) { - zoom = Math.max(zoomLimit.min, zoom); - } - } - - this._zoom = zoom; - - this._updateCenterAndZoom(); - }; - /** - * Get default center without roam - */ - - - View.prototype.getDefaultCenter = function () { - // Rect before any transform - var rawRect = this.getBoundingRect(); - var cx = rawRect.x + rawRect.width / 2; - var cy = rawRect.y + rawRect.height / 2; - return [cx, cy]; - }; - - View.prototype.getCenter = function () { - return this._center || this.getDefaultCenter(); - }; - - View.prototype.getZoom = function () { - return this._zoom || 1; - }; - - View.prototype.getRoamTransform = function () { - return this._roamTransformable.getLocalTransform(); - }; - /** - * Remove roam - */ - - - View.prototype._updateCenterAndZoom = function () { - // Must update after view transform updated - var rawTransformMatrix = this._rawTransformable.getLocalTransform(); - - var roamTransform = this._roamTransformable; - var defaultCenter = this.getDefaultCenter(); - var center = this.getCenter(); - var zoom = this.getZoom(); - center = applyTransform([], center, rawTransformMatrix); - defaultCenter = applyTransform([], defaultCenter, rawTransformMatrix); - roamTransform.originX = center[0]; - roamTransform.originY = center[1]; - roamTransform.x = defaultCenter[0] - center[0]; - roamTransform.y = defaultCenter[1] - center[1]; - roamTransform.scaleX = roamTransform.scaleY = zoom; - - this._updateTransform(); - }; - /** - * Update transform props on `this` based on the current - * `this._roamTransformable` and `this._rawTransformable`. - */ - - - View.prototype._updateTransform = function () { - var roamTransformable = this._roamTransformable; - var rawTransformable = this._rawTransformable; - rawTransformable.parent = roamTransformable; - roamTransformable.updateTransform(); - rawTransformable.updateTransform(); - copy$1(this.transform || (this.transform = []), rawTransformable.transform || create$1()); - this._rawTransform = rawTransformable.getLocalTransform(); - this.invTransform = this.invTransform || []; - invert(this.invTransform, this.transform); - this.decomposeTransform(); - }; - - View.prototype.getTransformInfo = function () { - var rawTransformable = this._rawTransformable; - var roamTransformable = this._roamTransformable; // Because roamTransformabel has `originX/originY` modified, - // but the caller of `getTransformInfo` can not handle `originX/originY`, - // so need to recalculate them. - - var dummyTransformable = new Transformable(); - dummyTransformable.transform = roamTransformable.transform; - dummyTransformable.decomposeTransform(); - return { - roam: { - x: dummyTransformable.x, - y: dummyTransformable.y, - scaleX: dummyTransformable.scaleX, - scaleY: dummyTransformable.scaleY - }, - raw: { - x: rawTransformable.x, - y: rawTransformable.y, - scaleX: rawTransformable.scaleX, - scaleY: rawTransformable.scaleY - } - }; - }; - - View.prototype.getViewRect = function () { - return this._viewRect; - }; - /** - * Get view rect after roam transform - */ - - - View.prototype.getViewRectAfterRoam = function () { - var rect = this.getBoundingRect().clone(); - rect.applyTransform(this.transform); - return rect; - }; - /** - * Convert a single (lon, lat) data item to (x, y) point. - */ - - - View.prototype.dataToPoint = function (data, noRoam, out) { - var transform = noRoam ? this._rawTransform : this.transform; - out = out || []; - return transform ? v2ApplyTransform(out, data, transform) : copy(out, data); - }; - /** - * Convert a (x, y) point to (lon, lat) data - */ - - - View.prototype.pointToData = function (point) { - var invTransform = this.invTransform; - return invTransform ? v2ApplyTransform([], point, invTransform) : [point[0], point[1]]; - }; - - View.prototype.convertToPixel = function (ecModel, finder, value) { - var coordSys = getCoordSys(finder); - return coordSys === this ? coordSys.dataToPoint(value) : null; - }; - - View.prototype.convertFromPixel = function (ecModel, finder, pixel) { - var coordSys = getCoordSys(finder); - return coordSys === this ? coordSys.pointToData(pixel) : null; - }; - /** - * @implements - */ - - - View.prototype.containPoint = function (point) { - return this.getViewRectAfterRoam().contain(point[0], point[1]); - }; - - View.dimensions = ['x', 'y']; - return View; - }(Transformable); - - function getCoordSys(finder) { - var seriesModel = finder.seriesModel; - return seriesModel ? seriesModel.coordinateSystem : null; // e.g., graph. - } - - var GEO_DEFAULT_PARAMS = { - 'geoJSON': { - aspectScale: 0.75, - invertLongitute: true - }, - 'geoSVG': { - aspectScale: 1, - invertLongitute: false - } - }; - var geo2DDimensions = ['lng', 'lat']; - - var Geo = - /** @class */ - function (_super) { - __extends(Geo, _super); - - function Geo(name, map, opt) { - var _this = _super.call(this, name) || this; - - _this.dimensions = geo2DDimensions; - _this.type = 'geo'; // Only store specified name coord via `addGeoCoord`. - - _this._nameCoordMap = createHashMap(); - _this.map = map; - var projection = opt.projection; - var source = geoSourceManager.load(map, opt.nameMap, opt.nameProperty); - var resource = geoSourceManager.getGeoResource(map); - var resourceType = _this.resourceType = resource ? resource.type : null; - var regions = _this.regions = source.regions; - var defaultParams = GEO_DEFAULT_PARAMS[resource.type]; - _this._regionsMap = source.regionsMap; - _this.regions = source.regions; - - if ("development" !== 'production' && projection) { - // Do some check - if (resourceType === 'geoSVG') { - if ("development" !== 'production') { - warn("Map " + map + " with SVG source can't use projection. Only GeoJSON source supports projection."); - } - - projection = null; - } - - if (!(projection.project && projection.unproject)) { - if ("development" !== 'production') { - warn('project and unproject must be both provided in the projeciton.'); - } - - projection = null; - } - } - - _this.projection = projection; - var boundingRect; - - if (projection) { - // Can't reuse the raw bounding rect - for (var i = 0; i < regions.length; i++) { - var regionRect = regions[i].getBoundingRect(projection); - boundingRect = boundingRect || regionRect.clone(); - boundingRect.union(regionRect); - } - } else { - boundingRect = source.boundingRect; - } - - _this.setBoundingRect(boundingRect.x, boundingRect.y, boundingRect.width, boundingRect.height); // aspectScale and invertLongitute actually is the parameters default raw projection. - // So we ignore them if projection is given. - // Ignore default aspect scale if projection exits. - - - _this.aspectScale = projection ? 1 : retrieve2(opt.aspectScale, defaultParams.aspectScale); // Not invert longitude if projection exits. - - _this._invertLongitute = projection ? false : defaultParams.invertLongitute; - return _this; - } - - Geo.prototype._transformTo = function (x, y, width, height) { - var rect = this.getBoundingRect(); - var invertLongitute = this._invertLongitute; - rect = rect.clone(); - - if (invertLongitute) { - // Longitude is inverted. - rect.y = -rect.y - rect.height; - } - - var rawTransformable = this._rawTransformable; - rawTransformable.transform = rect.calculateTransform(new BoundingRect(x, y, width, height)); - var rawParent = rawTransformable.parent; - rawTransformable.parent = null; - rawTransformable.decomposeTransform(); - rawTransformable.parent = rawParent; - - if (invertLongitute) { - rawTransformable.scaleY = -rawTransformable.scaleY; - } - - this._updateTransform(); - }; - - Geo.prototype.getRegion = function (name) { - return this._regionsMap.get(name); - }; - - Geo.prototype.getRegionByCoord = function (coord) { - var regions = this.regions; - - for (var i = 0; i < regions.length; i++) { - var region = regions[i]; - - if (region.type === 'geoJSON' && region.contain(coord)) { - return regions[i]; - } - } - }; - /** - * Add geoCoord for indexing by name - */ - - - Geo.prototype.addGeoCoord = function (name, geoCoord) { - this._nameCoordMap.set(name, geoCoord); - }; - /** - * Get geoCoord by name - */ - - - Geo.prototype.getGeoCoord = function (name) { - var region = this._regionsMap.get(name); // Calculate center only on demand. - - - return this._nameCoordMap.get(name) || region && region.getCenter(); - }; - - Geo.prototype.dataToPoint = function (data, noRoam, out) { - if (isString(data)) { - // Map area name to geoCoord - data = this.getGeoCoord(data); - } - - if (data) { - var projection = this.projection; - - if (projection) { - // projection may return null point. - data = projection.project(data); - } - - return data && this.projectedToPoint(data, noRoam, out); - } - }; - - Geo.prototype.pointToData = function (point) { - var projection = this.projection; - - if (projection) { - // projection may return null point. - point = projection.unproject(point); - } - - return point && this.pointToProjected(point); - }; - /** - * Point to projected data. Same with pointToData when projection is used. - */ - - - Geo.prototype.pointToProjected = function (point) { - return _super.prototype.pointToData.call(this, point); - }; - - Geo.prototype.projectedToPoint = function (projected, noRoam, out) { - return _super.prototype.dataToPoint.call(this, projected, noRoam, out); - }; - - Geo.prototype.convertToPixel = function (ecModel, finder, value) { - var coordSys = getCoordSys$1(finder); - return coordSys === this ? coordSys.dataToPoint(value) : null; - }; - - Geo.prototype.convertFromPixel = function (ecModel, finder, pixel) { - var coordSys = getCoordSys$1(finder); - return coordSys === this ? coordSys.pointToData(pixel) : null; - }; - - return Geo; - }(View); - mixin(Geo, View); - - function getCoordSys$1(finder) { - var geoModel = finder.geoModel; - var seriesModel = finder.seriesModel; - return geoModel ? geoModel.coordinateSystem : seriesModel ? seriesModel.coordinateSystem // For map series. - || (seriesModel.getReferringComponents('geo', SINGLE_REFERRING).models[0] || {}).coordinateSystem : null; - } - - /** - * Resize method bound to the geo - */ - - function resizeGeo(geoModel, api) { - var boundingCoords = geoModel.get('boundingCoords'); - - if (boundingCoords != null) { - var leftTop_1 = boundingCoords[0]; - var rightBottom_1 = boundingCoords[1]; - - if (!(isFinite(leftTop_1[0]) && isFinite(leftTop_1[1]) && isFinite(rightBottom_1[0]) && isFinite(rightBottom_1[1]))) { - if ("development" !== 'production') { - console.error('Invalid boundingCoords'); - } - } else { - // Sample around the lng/lat rect and use projection to calculate actual bounding rect. - var projection_1 = this.projection; - - if (projection_1) { - var xMin = leftTop_1[0]; - var yMin = leftTop_1[1]; - var xMax = rightBottom_1[0]; - var yMax = rightBottom_1[1]; - leftTop_1 = [Infinity, Infinity]; - rightBottom_1 = [-Infinity, -Infinity]; // TODO better way? - - var sampleLine = function (x0, y0, x1, y1) { - var dx = x1 - x0; - var dy = y1 - y0; - - for (var i = 0; i <= 100; i++) { - var p = i / 100; - var pt = projection_1.project([x0 + dx * p, y0 + dy * p]); - min(leftTop_1, leftTop_1, pt); - max(rightBottom_1, rightBottom_1, pt); - } - }; // Top - - - sampleLine(xMin, yMin, xMax, yMin); // Right - - sampleLine(xMax, yMin, xMax, yMax); // Bottom - - sampleLine(xMax, yMax, xMin, yMax); // Left - - sampleLine(xMin, yMax, xMax, yMin); - } - - this.setBoundingRect(leftTop_1[0], leftTop_1[1], rightBottom_1[0] - leftTop_1[0], rightBottom_1[1] - leftTop_1[1]); - } - } - - var rect = this.getBoundingRect(); - var centerOption = geoModel.get('layoutCenter'); - var sizeOption = geoModel.get('layoutSize'); - var viewWidth = api.getWidth(); - var viewHeight = api.getHeight(); - var aspect = rect.width / rect.height * this.aspectScale; - var useCenterAndSize = false; - var center; - var size; - - if (centerOption && sizeOption) { - center = [parsePercent$1(centerOption[0], viewWidth), parsePercent$1(centerOption[1], viewHeight)]; - size = parsePercent$1(sizeOption, Math.min(viewWidth, viewHeight)); - - if (!isNaN(center[0]) && !isNaN(center[1]) && !isNaN(size)) { - useCenterAndSize = true; - } else { - if ("development" !== 'production') { - console.warn('Given layoutCenter or layoutSize data are invalid. Use left/top/width/height instead.'); - } - } - } - - var viewRect; - - if (useCenterAndSize) { - viewRect = {}; - - if (aspect > 1) { - // Width is same with size - viewRect.width = size; - viewRect.height = size / aspect; - } else { - viewRect.height = size; - viewRect.width = size * aspect; - } - - viewRect.y = center[1] - viewRect.height / 2; - viewRect.x = center[0] - viewRect.width / 2; - } else { - // Use left/top/width/height - var boxLayoutOption = geoModel.getBoxLayoutParams(); - boxLayoutOption.aspect = aspect; - viewRect = getLayoutRect(boxLayoutOption, { - width: viewWidth, - height: viewHeight - }); - } - - this.setViewRect(viewRect.x, viewRect.y, viewRect.width, viewRect.height); - this.setCenter(geoModel.get('center'), api); - this.setZoom(geoModel.get('zoom')); - } // Back compat for ECharts2, where the coord map is set on map series: - // {type: 'map', geoCoord: {'cityA': [116.46,39.92], 'cityA': [119.12,24.61]}}, - - - function setGeoCoords(geo, model) { - each(model.get('geoCoord'), function (geoCoord, name) { - geo.addGeoCoord(name, geoCoord); - }); - } - - var GeoCreator = - /** @class */ - function () { - function GeoCreator() { - // For deciding which dimensions to use when creating list data - this.dimensions = geo2DDimensions; - } - - GeoCreator.prototype.create = function (ecModel, api) { - var geoList = []; - - function getCommonGeoProperties(model) { - return { - nameProperty: model.get('nameProperty'), - aspectScale: model.get('aspectScale'), - projection: model.get('projection') - }; - } // FIXME Create each time may be slow - - - ecModel.eachComponent('geo', function (geoModel, idx) { - var mapName = geoModel.get('map'); - var geo = new Geo(mapName + idx, mapName, extend({ - nameMap: geoModel.get('nameMap') - }, getCommonGeoProperties(geoModel))); - geo.zoomLimit = geoModel.get('scaleLimit'); - geoList.push(geo); // setGeoCoords(geo, geoModel); - - geoModel.coordinateSystem = geo; - geo.model = geoModel; // Inject resize method - - geo.resize = resizeGeo; - geo.resize(geoModel, api); - }); - ecModel.eachSeries(function (seriesModel) { - var coordSys = seriesModel.get('coordinateSystem'); - - if (coordSys === 'geo') { - var geoIndex = seriesModel.get('geoIndex') || 0; - seriesModel.coordinateSystem = geoList[geoIndex]; - } - }); // If has map series - - var mapModelGroupBySeries = {}; - ecModel.eachSeriesByType('map', function (seriesModel) { - if (!seriesModel.getHostGeoModel()) { - var mapType = seriesModel.getMapType(); - mapModelGroupBySeries[mapType] = mapModelGroupBySeries[mapType] || []; - mapModelGroupBySeries[mapType].push(seriesModel); - } - }); - each(mapModelGroupBySeries, function (mapSeries, mapType) { - var nameMapList = map(mapSeries, function (singleMapSeries) { - return singleMapSeries.get('nameMap'); - }); - var geo = new Geo(mapType, mapType, extend({ - nameMap: mergeAll(nameMapList) - }, getCommonGeoProperties(mapSeries[0]))); - geo.zoomLimit = retrieve.apply(null, map(mapSeries, function (singleMapSeries) { - return singleMapSeries.get('scaleLimit'); - })); - geoList.push(geo); // Inject resize method - - geo.resize = resizeGeo; - geo.resize(mapSeries[0], api); - each(mapSeries, function (singleMapSeries) { - singleMapSeries.coordinateSystem = geo; - setGeoCoords(geo, singleMapSeries); - }); - }); - return geoList; - }; - /** - * Fill given regions array - */ - - - GeoCreator.prototype.getFilledRegions = function (originRegionArr, mapName, nameMap, nameProperty) { - // Not use the original - var regionsArr = (originRegionArr || []).slice(); - var dataNameMap = createHashMap(); - - for (var i = 0; i < regionsArr.length; i++) { - dataNameMap.set(regionsArr[i].name, regionsArr[i]); - } - - var source = geoSourceManager.load(mapName, nameMap, nameProperty); - each(source.regions, function (region) { - var name = region.name; - !dataNameMap.get(name) && regionsArr.push({ - name: name - }); - }); - return regionsArr; - }; - - return GeoCreator; - }(); - - var geoCreator = new GeoCreator(); - - var GeoModel = - /** @class */ - function (_super) { - __extends(GeoModel, _super); - - function GeoModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GeoModel.type; - return _this; - } - - GeoModel.prototype.init = function (option, parentModel, ecModel) { - var source = geoSourceManager.getGeoResource(option.map); - - if (source && source.type === 'geoJSON') { - var itemStyle = option.itemStyle = option.itemStyle || {}; - - if (!('color' in itemStyle)) { - itemStyle.color = '#eee'; - } - } - - this.mergeDefaultAndTheme(option, ecModel); // Default label emphasis `show` - - defaultEmphasis(option, 'label', ['show']); - }; - - GeoModel.prototype.optionUpdated = function () { - var _this = this; - - var option = this.option; - option.regions = geoCreator.getFilledRegions(option.regions, option.map, option.nameMap, option.nameProperty); - var selectedMap = {}; - this._optionModelMap = reduce(option.regions || [], function (optionModelMap, regionOpt) { - var regionName = regionOpt.name; - - if (regionName) { - optionModelMap.set(regionName, new Model(regionOpt, _this, _this.ecModel)); - - if (regionOpt.selected) { - selectedMap[regionName] = true; - } - } - - return optionModelMap; - }, createHashMap()); - - if (!option.selectedMap) { - option.selectedMap = selectedMap; - } - }; - /** - * Get model of region. - */ - - - GeoModel.prototype.getRegionModel = function (name) { - return this._optionModelMap.get(name) || new Model(null, this, this.ecModel); - }; - /** - * Format label - * @param name Region name - */ - - - GeoModel.prototype.getFormattedLabel = function (name, status) { - var regionModel = this.getRegionModel(name); - var formatter = status === 'normal' ? regionModel.get(['label', 'formatter']) : regionModel.get(['emphasis', 'label', 'formatter']); - var params = { - name: name - }; - - if (isFunction(formatter)) { - params.status = status; - return formatter(params); - } else if (isString(formatter)) { - return formatter.replace('{a}', name != null ? name : ''); - } - }; - - GeoModel.prototype.setZoom = function (zoom) { - this.option.zoom = zoom; - }; - - GeoModel.prototype.setCenter = function (center) { - this.option.center = center; - }; // PENGING If selectedMode is null ? - - - GeoModel.prototype.select = function (name) { - var option = this.option; - var selectedMode = option.selectedMode; - - if (!selectedMode) { - return; - } - - if (selectedMode !== 'multiple') { - option.selectedMap = null; - } - - var selectedMap = option.selectedMap || (option.selectedMap = {}); - selectedMap[name] = true; - }; - - GeoModel.prototype.unSelect = function (name) { - var selectedMap = this.option.selectedMap; - - if (selectedMap) { - selectedMap[name] = false; - } - }; - - GeoModel.prototype.toggleSelected = function (name) { - this[this.isSelected(name) ? 'unSelect' : 'select'](name); - }; - - GeoModel.prototype.isSelected = function (name) { - var selectedMap = this.option.selectedMap; - return !!(selectedMap && selectedMap[name]); - }; - - GeoModel.type = 'geo'; - GeoModel.layoutMode = 'box'; - GeoModel.defaultOption = { - // zlevel: 0, - z: 0, - show: true, - left: 'center', - top: 'center', - // Default value: - // for geoSVG source: 1, - // for geoJSON source: 0.75. - aspectScale: null, - // /// Layout with center and size - // If you want to put map in a fixed size box with right aspect ratio - // This two properties may be more convenient - // layoutCenter: [50%, 50%] - // layoutSize: 100 - silent: false, - // Map type - map: '', - // Define left-top, right-bottom coords to control view - // For example, [ [180, 90], [-180, -90] ] - boundingCoords: null, - // Default on center of map - center: null, - zoom: 1, - scaleLimit: null, - // selectedMode: false - label: { - show: false, - color: '#000' - }, - itemStyle: { - borderWidth: 0.5, - borderColor: '#444' // Default color: - // + geoJSON: #eee - // + geoSVG: null (use SVG original `fill`) - // color: '#eee' - - }, - emphasis: { - label: { - show: true, - color: 'rgb(100,0,0)' - }, - itemStyle: { - color: 'rgba(255,215,0,0.8)' - } - }, - select: { - label: { - show: true, - color: 'rgb(100,0,0)' - }, - itemStyle: { - color: 'rgba(255,215,0,0.8)' - } - }, - regions: [] // tooltip: { - // show: false - // } - - }; - return GeoModel; - }(ComponentModel); - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function getCenterCoord(view, point) { - // Use projected coord as center because it's linear. - return view.pointToProjected ? view.pointToProjected(point) : view.pointToData(point); - } - - function updateCenterAndZoom(view, payload, zoomLimit, api) { - var previousZoom = view.getZoom(); - var center = view.getCenter(); - var zoom = payload.zoom; - var point = view.projectedToPoint ? view.projectedToPoint(center) : view.dataToPoint(center); - - if (payload.dx != null && payload.dy != null) { - point[0] -= payload.dx; - point[1] -= payload.dy; - view.setCenter(getCenterCoord(view, point), api); - } - - if (zoom != null) { - if (zoomLimit) { - var zoomMin = zoomLimit.min || 0; - var zoomMax = zoomLimit.max || Infinity; - zoom = Math.max(Math.min(previousZoom * zoom, zoomMax), zoomMin) / previousZoom; - } // Zoom on given point(originX, originY) - - - view.scaleX *= zoom; - view.scaleY *= zoom; - var fixX = (payload.originX - view.x) * (zoom - 1); - var fixY = (payload.originY - view.y) * (zoom - 1); - view.x -= fixX; - view.y -= fixY; - view.updateTransform(); // Get the new center - - view.setCenter(getCenterCoord(view, point), api); - view.setZoom(zoom * previousZoom); - } - - return { - center: view.getCenter(), - zoom: view.getZoom() - }; - } - - var GeoView = - /** @class */ - function (_super) { - __extends(GeoView, _super); - - function GeoView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GeoView.type; - _this.focusBlurEnabled = true; - return _this; - } - - GeoView.prototype.init = function (ecModel, api) { - this._api = api; - }; - - GeoView.prototype.render = function (geoModel, ecModel, api, payload) { - this._model = geoModel; - - if (!geoModel.get('show')) { - this._mapDraw && this._mapDraw.remove(); - this._mapDraw = null; - return; - } - - if (!this._mapDraw) { - this._mapDraw = new MapDraw(api); - } - - var mapDraw = this._mapDraw; - mapDraw.draw(geoModel, ecModel, api, this, payload); - mapDraw.group.on('click', this._handleRegionClick, this); - mapDraw.group.silent = geoModel.get('silent'); - this.group.add(mapDraw.group); - this.updateSelectStatus(geoModel, ecModel, api); - }; - - GeoView.prototype._handleRegionClick = function (e) { - var eventData; - findEventDispatcher(e.target, function (current) { - return (eventData = getECData(current).eventData) != null; - }, true); - - if (eventData) { - this._api.dispatchAction({ - type: 'geoToggleSelect', - geoId: this._model.id, - name: eventData.name - }); - } - }; - - GeoView.prototype.updateSelectStatus = function (model, ecModel, api) { - var _this = this; - - this._mapDraw.group.traverse(function (node) { - var eventData = getECData(node).eventData; - - if (eventData) { - _this._model.isSelected(eventData.name) ? api.enterSelect(node) : api.leaveSelect(node); // No need to traverse children. - - return true; - } - }); - }; - - GeoView.prototype.findHighDownDispatchers = function (name) { - return this._mapDraw && this._mapDraw.findHighDownDispatchers(name, this._model); - }; - - GeoView.prototype.dispose = function () { - this._mapDraw && this._mapDraw.remove(); - }; - - GeoView.type = 'geo'; - return GeoView; - }(ComponentView); - - function registerMap$1(mapName, geoJson, specialAreas) { - geoSourceManager.registerMap(mapName, geoJson, specialAreas); - } - - function install$9(registers) { - registers.registerCoordinateSystem('geo', geoCreator); - registers.registerComponentModel(GeoModel); - registers.registerComponentView(GeoView); - registers.registerImpl('registerMap', registerMap$1); - registers.registerImpl('getMap', function (mapName) { - return geoSourceManager.getMapForUser(mapName); - }); - - function makeAction(method, actionInfo) { - actionInfo.update = 'geo:updateSelectStatus'; - registers.registerAction(actionInfo, function (payload, ecModel) { - var selected = {}; - var allSelected = []; - ecModel.eachComponent({ - mainType: 'geo', - query: payload - }, function (geoModel) { - geoModel[method](payload.name); - var geo = geoModel.coordinateSystem; - each(geo.regions, function (region) { - selected[region.name] = geoModel.isSelected(region.name) || false; - }); // Notice: there might be duplicated name in different regions. - - var names = []; - each(selected, function (v, name) { - selected[name] && names.push(name); - }); - allSelected.push({ - geoIndex: geoModel.componentIndex, - // Use singular, the same naming convention as the event `selectchanged`. - name: names - }); - }); - return { - selected: selected, - allSelected: allSelected, - name: payload.name - }; - }); - } - - makeAction('toggleSelected', { - type: 'geoToggleSelect', - event: 'geoselectchanged' - }); - makeAction('select', { - type: 'geoSelect', - event: 'geoselected' - }); - makeAction('unSelect', { - type: 'geoUnSelect', - event: 'geounselected' - }); - /** - * @payload - * @property {string} [componentType=series] - * @property {number} [dx] - * @property {number} [dy] - * @property {number} [zoom] - * @property {number} [originX] - * @property {number} [originY] - */ - - registers.registerAction({ - type: 'geoRoam', - event: 'geoRoam', - update: 'updateTransform' - }, function (payload, ecModel, api) { - var componentType = payload.componentType || 'series'; - ecModel.eachComponent({ - mainType: componentType, - query: payload - }, function (componentModel) { - var geo = componentModel.coordinateSystem; - - if (geo.type !== 'geo') { - return; - } - - var res = updateCenterAndZoom(geo, payload, componentModel.get('scaleLimit'), api); - componentModel.setCenter && componentModel.setCenter(res.center); - componentModel.setZoom && componentModel.setZoom(res.zoom); // All map series with same `map` use the same geo coordinate system - // So the center and zoom must be in sync. Include the series not selected by legend - - if (componentType === 'series') { - each(componentModel.seriesGroup, function (seriesModel) { - seriesModel.setCenter(res.center); - seriesModel.setZoom(res.zoom); - }); - } - }); - }); - } - - function install$a(registers) { - use(install$9); - registers.registerChartView(MapView); - registers.registerSeriesModel(MapSeries); - registers.registerLayout(mapSymbolLayout); - registers.registerProcessor(registers.PRIORITY.PROCESSOR.STATISTIC, mapDataStatistic); - createLegacyDataSelectAction('map', registers.registerAction); - } - - /** - * Initialize all computational message for following algorithm. - */ - - function init$2(inRoot) { - var root = inRoot; - root.hierNode = { - defaultAncestor: null, - ancestor: root, - prelim: 0, - modifier: 0, - change: 0, - shift: 0, - i: 0, - thread: null - }; - var nodes = [root]; - var node; - var children; - - while (node = nodes.pop()) { - // jshint ignore:line - children = node.children; - - if (node.isExpand && children.length) { - var n = children.length; - - for (var i = n - 1; i >= 0; i--) { - var child = children[i]; - child.hierNode = { - defaultAncestor: null, - ancestor: child, - prelim: 0, - modifier: 0, - change: 0, - shift: 0, - i: i, - thread: null - }; - nodes.push(child); - } - } - } - } - /** - * The implementation of this function was originally copied from "d3.js" - * <https://github.com/d3/d3-hierarchy/blob/4c1f038f2725d6eae2e49b61d01456400694bac4/src/tree.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - * - * Computes a preliminary x coordinate for node. Before that, this function is - * applied recursively to the children of node, as well as the function - * apportion(). After spacing out the children by calling executeShifts(), the - * node is placed to the midpoint of its outermost children. - */ - - function firstWalk(node, separation) { - var children = node.isExpand ? node.children : []; - var siblings = node.parentNode.children; - var subtreeW = node.hierNode.i ? siblings[node.hierNode.i - 1] : null; - - if (children.length) { - executeShifts(node); - var midPoint = (children[0].hierNode.prelim + children[children.length - 1].hierNode.prelim) / 2; - - if (subtreeW) { - node.hierNode.prelim = subtreeW.hierNode.prelim + separation(node, subtreeW); - node.hierNode.modifier = node.hierNode.prelim - midPoint; - } else { - node.hierNode.prelim = midPoint; - } - } else if (subtreeW) { - node.hierNode.prelim = subtreeW.hierNode.prelim + separation(node, subtreeW); - } - - node.parentNode.hierNode.defaultAncestor = apportion(node, subtreeW, node.parentNode.hierNode.defaultAncestor || siblings[0], separation); - } - /** - * The implementation of this function was originally copied from "d3.js" - * <https://github.com/d3/d3-hierarchy/blob/4c1f038f2725d6eae2e49b61d01456400694bac4/src/tree.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - * - * Computes all real x-coordinates by summing up the modifiers recursively. - */ - - function secondWalk(node) { - var nodeX = node.hierNode.prelim + node.parentNode.hierNode.modifier; - node.setLayout({ - x: nodeX - }, true); - node.hierNode.modifier += node.parentNode.hierNode.modifier; - } - function separation(cb) { - return arguments.length ? cb : defaultSeparation; - } - /** - * Transform the common coordinate to radial coordinate. - */ - - function radialCoordinate(rad, r) { - rad -= Math.PI / 2; - return { - x: r * Math.cos(rad), - y: r * Math.sin(rad) - }; - } - /** - * Get the layout position of the whole view. - */ - - function getViewRect$1(seriesModel, api) { - return getLayoutRect(seriesModel.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - } - /** - * All other shifts, applied to the smaller subtrees between w- and w+, are - * performed by this function. - * - * The implementation of this function was originally copied from "d3.js" - * <https://github.com/d3/d3-hierarchy/blob/4c1f038f2725d6eae2e49b61d01456400694bac4/src/tree.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - */ - - function executeShifts(node) { - var children = node.children; - var n = children.length; - var shift = 0; - var change = 0; - - while (--n >= 0) { - var child = children[n]; - child.hierNode.prelim += shift; - child.hierNode.modifier += shift; - change += child.hierNode.change; - shift += child.hierNode.shift + change; - } - } - /** - * The implementation of this function was originally copied from "d3.js" - * <https://github.com/d3/d3-hierarchy/blob/4c1f038f2725d6eae2e49b61d01456400694bac4/src/tree.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - * - * The core of the algorithm. Here, a new subtree is combined with the - * previous subtrees. Threads are used to traverse the inside and outside - * contours of the left and right subtree up to the highest common level. - * Whenever two nodes of the inside contours conflict, we compute the left - * one of the greatest uncommon ancestors using the function nextAncestor() - * and call moveSubtree() to shift the subtree and prepare the shifts of - * smaller subtrees. Finally, we add a new thread (if necessary). - */ - - - function apportion(subtreeV, subtreeW, ancestor, separation) { - if (subtreeW) { - var nodeOutRight = subtreeV; - var nodeInRight = subtreeV; - var nodeOutLeft = nodeInRight.parentNode.children[0]; - var nodeInLeft = subtreeW; - var sumOutRight = nodeOutRight.hierNode.modifier; - var sumInRight = nodeInRight.hierNode.modifier; - var sumOutLeft = nodeOutLeft.hierNode.modifier; - var sumInLeft = nodeInLeft.hierNode.modifier; - - while (nodeInLeft = nextRight(nodeInLeft), nodeInRight = nextLeft(nodeInRight), nodeInLeft && nodeInRight) { - nodeOutRight = nextRight(nodeOutRight); - nodeOutLeft = nextLeft(nodeOutLeft); - nodeOutRight.hierNode.ancestor = subtreeV; - var shift = nodeInLeft.hierNode.prelim + sumInLeft - nodeInRight.hierNode.prelim - sumInRight + separation(nodeInLeft, nodeInRight); - - if (shift > 0) { - moveSubtree(nextAncestor(nodeInLeft, subtreeV, ancestor), subtreeV, shift); - sumInRight += shift; - sumOutRight += shift; - } - - sumInLeft += nodeInLeft.hierNode.modifier; - sumInRight += nodeInRight.hierNode.modifier; - sumOutRight += nodeOutRight.hierNode.modifier; - sumOutLeft += nodeOutLeft.hierNode.modifier; - } - - if (nodeInLeft && !nextRight(nodeOutRight)) { - nodeOutRight.hierNode.thread = nodeInLeft; - nodeOutRight.hierNode.modifier += sumInLeft - sumOutRight; - } - - if (nodeInRight && !nextLeft(nodeOutLeft)) { - nodeOutLeft.hierNode.thread = nodeInRight; - nodeOutLeft.hierNode.modifier += sumInRight - sumOutLeft; - ancestor = subtreeV; - } - } - - return ancestor; - } - /** - * This function is used to traverse the right contour of a subtree. - * It returns the rightmost child of node or the thread of node. The function - * returns null if and only if node is on the highest depth of its subtree. - */ - - - function nextRight(node) { - var children = node.children; - return children.length && node.isExpand ? children[children.length - 1] : node.hierNode.thread; - } - /** - * This function is used to traverse the left contour of a subtree (or a subforest). - * It returns the leftmost child of node or the thread of node. The function - * returns null if and only if node is on the highest depth of its subtree. - */ - - - function nextLeft(node) { - var children = node.children; - return children.length && node.isExpand ? children[0] : node.hierNode.thread; - } - /** - * If nodeInLeft’s ancestor is a sibling of node, returns nodeInLeft’s ancestor. - * Otherwise, returns the specified ancestor. - */ - - - function nextAncestor(nodeInLeft, node, ancestor) { - return nodeInLeft.hierNode.ancestor.parentNode === node.parentNode ? nodeInLeft.hierNode.ancestor : ancestor; - } - /** - * The implementation of this function was originally copied from "d3.js" - * <https://github.com/d3/d3-hierarchy/blob/4c1f038f2725d6eae2e49b61d01456400694bac4/src/tree.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - * - * Shifts the current subtree rooted at wr. - * This is done by increasing prelim(w+) and modifier(w+) by shift. - */ - - - function moveSubtree(wl, wr, shift) { - var change = shift / (wr.hierNode.i - wl.hierNode.i); - wr.hierNode.change -= change; - wr.hierNode.shift += shift; - wr.hierNode.modifier += shift; - wr.hierNode.prelim += shift; - wl.hierNode.change += change; - } - /** - * The implementation of this function was originally copied from "d3.js" - * <https://github.com/d3/d3-hierarchy/blob/4c1f038f2725d6eae2e49b61d01456400694bac4/src/tree.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - */ - - - function defaultSeparation(node1, node2) { - return node1.parentNode === node2.parentNode ? 1 : 2; - } - - var TreeEdgeShape = - /** @class */ - function () { - function TreeEdgeShape() { - this.parentPoint = []; - this.childPoints = []; - } - - return TreeEdgeShape; - }(); - - var TreePath = - /** @class */ - function (_super) { - __extends(TreePath, _super); - - function TreePath(opts) { - return _super.call(this, opts) || this; - } - - TreePath.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - - TreePath.prototype.getDefaultShape = function () { - return new TreeEdgeShape(); - }; - - TreePath.prototype.buildPath = function (ctx, shape) { - var childPoints = shape.childPoints; - var childLen = childPoints.length; - var parentPoint = shape.parentPoint; - var firstChildPos = childPoints[0]; - var lastChildPos = childPoints[childLen - 1]; - - if (childLen === 1) { - ctx.moveTo(parentPoint[0], parentPoint[1]); - ctx.lineTo(firstChildPos[0], firstChildPos[1]); - return; - } - - var orient = shape.orient; - var forkDim = orient === 'TB' || orient === 'BT' ? 0 : 1; - var otherDim = 1 - forkDim; - var forkPosition = parsePercent$1(shape.forkPosition, 1); - var tmpPoint = []; - tmpPoint[forkDim] = parentPoint[forkDim]; - tmpPoint[otherDim] = parentPoint[otherDim] + (lastChildPos[otherDim] - parentPoint[otherDim]) * forkPosition; - ctx.moveTo(parentPoint[0], parentPoint[1]); - ctx.lineTo(tmpPoint[0], tmpPoint[1]); - ctx.moveTo(firstChildPos[0], firstChildPos[1]); - tmpPoint[forkDim] = firstChildPos[forkDim]; - ctx.lineTo(tmpPoint[0], tmpPoint[1]); - tmpPoint[forkDim] = lastChildPos[forkDim]; - ctx.lineTo(tmpPoint[0], tmpPoint[1]); - ctx.lineTo(lastChildPos[0], lastChildPos[1]); - - for (var i = 1; i < childLen - 1; i++) { - var point = childPoints[i]; - ctx.moveTo(point[0], point[1]); - tmpPoint[forkDim] = point[forkDim]; - ctx.lineTo(tmpPoint[0], tmpPoint[1]); - } - }; - - return TreePath; - }(Path); - - var TreeView = - /** @class */ - function (_super) { - __extends(TreeView, _super); - - function TreeView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TreeView.type; - _this._mainGroup = new Group(); - return _this; - } - - TreeView.prototype.init = function (ecModel, api) { - this._controller = new RoamController(api.getZr()); - this._controllerHost = { - target: this.group - }; - this.group.add(this._mainGroup); - }; - - TreeView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var layoutInfo = seriesModel.layoutInfo; - var group = this._mainGroup; - var layout = seriesModel.get('layout'); - - if (layout === 'radial') { - group.x = layoutInfo.x + layoutInfo.width / 2; - group.y = layoutInfo.y + layoutInfo.height / 2; - } else { - group.x = layoutInfo.x; - group.y = layoutInfo.y; - } - - this._updateViewCoordSys(seriesModel, api); - - this._updateController(seriesModel, ecModel, api); - - var oldData = this._data; - data.diff(oldData).add(function (newIdx) { - if (symbolNeedsDraw$1(data, newIdx)) { - // Create node and edge - updateNode(data, newIdx, null, group, seriesModel); - } - }).update(function (newIdx, oldIdx) { - var symbolEl = oldData.getItemGraphicEl(oldIdx); - - if (!symbolNeedsDraw$1(data, newIdx)) { - symbolEl && removeNode(oldData, oldIdx, symbolEl, group, seriesModel); - return; - } // Update node and edge - - - updateNode(data, newIdx, symbolEl, group, seriesModel); - }).remove(function (oldIdx) { - var symbolEl = oldData.getItemGraphicEl(oldIdx); // When remove a collapsed node of subtree, since the collapsed - // node haven't been initialized with a symbol element, - // you can't found it's symbol element through index. - // so if we want to remove the symbol element we should insure - // that the symbol element is not null. - - if (symbolEl) { - removeNode(oldData, oldIdx, symbolEl, group, seriesModel); - } - }).execute(); - this._nodeScaleRatio = seriesModel.get('nodeScaleRatio'); - - this._updateNodeAndLinkScale(seriesModel); - - if (seriesModel.get('expandAndCollapse') === true) { - data.eachItemGraphicEl(function (el, dataIndex) { - el.off('click').on('click', function () { - api.dispatchAction({ - type: 'treeExpandAndCollapse', - seriesId: seriesModel.id, - dataIndex: dataIndex - }); - }); - }); - } - - this._data = data; - }; - - TreeView.prototype._updateViewCoordSys = function (seriesModel, api) { - var data = seriesModel.getData(); - var points = []; - data.each(function (idx) { - var layout = data.getItemLayout(idx); - - if (layout && !isNaN(layout.x) && !isNaN(layout.y)) { - points.push([+layout.x, +layout.y]); - } - }); - var min = []; - var max = []; - fromPoints(points, min, max); // If don't Store min max when collapse the root node after roam, - // the root node will disappear. - - var oldMin = this._min; - var oldMax = this._max; // If width or height is 0 - - if (max[0] - min[0] === 0) { - min[0] = oldMin ? oldMin[0] : min[0] - 1; - max[0] = oldMax ? oldMax[0] : max[0] + 1; - } - - if (max[1] - min[1] === 0) { - min[1] = oldMin ? oldMin[1] : min[1] - 1; - max[1] = oldMax ? oldMax[1] : max[1] + 1; - } - - var viewCoordSys = seriesModel.coordinateSystem = new View(); - viewCoordSys.zoomLimit = seriesModel.get('scaleLimit'); - viewCoordSys.setBoundingRect(min[0], min[1], max[0] - min[0], max[1] - min[1]); - viewCoordSys.setCenter(seriesModel.get('center'), api); - viewCoordSys.setZoom(seriesModel.get('zoom')); // Here we use viewCoordSys just for computing the 'position' and 'scale' of the group - - this.group.attr({ - x: viewCoordSys.x, - y: viewCoordSys.y, - scaleX: viewCoordSys.scaleX, - scaleY: viewCoordSys.scaleY - }); - this._min = min; - this._max = max; - }; - - TreeView.prototype._updateController = function (seriesModel, ecModel, api) { - var _this = this; - - var controller = this._controller; - var controllerHost = this._controllerHost; - var group = this.group; - controller.setPointerChecker(function (e, x, y) { - var rect = group.getBoundingRect(); - rect.applyTransform(group.transform); - return rect.contain(x, y) && !onIrrelevantElement(e, api, seriesModel); - }); - controller.enable(seriesModel.get('roam')); - controllerHost.zoomLimit = seriesModel.get('scaleLimit'); - controllerHost.zoom = seriesModel.coordinateSystem.getZoom(); - controller.off('pan').off('zoom').on('pan', function (e) { - updateViewOnPan(controllerHost, e.dx, e.dy); - api.dispatchAction({ - seriesId: seriesModel.id, - type: 'treeRoam', - dx: e.dx, - dy: e.dy - }); - }).on('zoom', function (e) { - updateViewOnZoom(controllerHost, e.scale, e.originX, e.originY); - api.dispatchAction({ - seriesId: seriesModel.id, - type: 'treeRoam', - zoom: e.scale, - originX: e.originX, - originY: e.originY - }); - - _this._updateNodeAndLinkScale(seriesModel); // Only update label layout on zoom - - - api.updateLabelLayout(); - }); - }; - - TreeView.prototype._updateNodeAndLinkScale = function (seriesModel) { - var data = seriesModel.getData(); - - var nodeScale = this._getNodeGlobalScale(seriesModel); - - data.eachItemGraphicEl(function (el, idx) { - el.setSymbolScale(nodeScale); - }); - }; - - TreeView.prototype._getNodeGlobalScale = function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys.type !== 'view') { - return 1; - } - - var nodeScaleRatio = this._nodeScaleRatio; - var groupZoom = coordSys.scaleX || 1; // Scale node when zoom changes - - var roamZoom = coordSys.getZoom(); - var nodeScale = (roamZoom - 1) * nodeScaleRatio + 1; - return nodeScale / groupZoom; - }; - - TreeView.prototype.dispose = function () { - this._controller && this._controller.dispose(); - this._controllerHost = null; - }; - - TreeView.prototype.remove = function () { - this._mainGroup.removeAll(); - - this._data = null; - }; - - TreeView.type = 'tree'; - return TreeView; - }(ChartView); - - function symbolNeedsDraw$1(data, dataIndex) { - var layout = data.getItemLayout(dataIndex); - return layout && !isNaN(layout.x) && !isNaN(layout.y); - } - - function updateNode(data, dataIndex, symbolEl, group, seriesModel) { - var isInit = !symbolEl; - var node = data.tree.getNodeByDataIndex(dataIndex); - var itemModel = node.getModel(); - var visualColor = node.getVisual('style').fill; - var symbolInnerColor = node.isExpand === false && node.children.length !== 0 ? visualColor : '#fff'; - var virtualRoot = data.tree.root; - var source = node.parentNode === virtualRoot ? node : node.parentNode || node; - var sourceSymbolEl = data.getItemGraphicEl(source.dataIndex); - var sourceLayout = source.getLayout(); - var sourceOldLayout = sourceSymbolEl ? { - x: sourceSymbolEl.__oldX, - y: sourceSymbolEl.__oldY, - rawX: sourceSymbolEl.__radialOldRawX, - rawY: sourceSymbolEl.__radialOldRawY - } : sourceLayout; - var targetLayout = node.getLayout(); - - if (isInit) { - symbolEl = new Symbol(data, dataIndex, null, { - symbolInnerColor: symbolInnerColor, - useNameLabel: true - }); - symbolEl.x = sourceOldLayout.x; - symbolEl.y = sourceOldLayout.y; - } else { - symbolEl.updateData(data, dataIndex, null, { - symbolInnerColor: symbolInnerColor, - useNameLabel: true - }); - } - - symbolEl.__radialOldRawX = symbolEl.__radialRawX; - symbolEl.__radialOldRawY = symbolEl.__radialRawY; - symbolEl.__radialRawX = targetLayout.rawX; - symbolEl.__radialRawY = targetLayout.rawY; - group.add(symbolEl); - data.setItemGraphicEl(dataIndex, symbolEl); - symbolEl.__oldX = symbolEl.x; - symbolEl.__oldY = symbolEl.y; - updateProps(symbolEl, { - x: targetLayout.x, - y: targetLayout.y - }, seriesModel); - var symbolPath = symbolEl.getSymbolPath(); - - if (seriesModel.get('layout') === 'radial') { - var realRoot = virtualRoot.children[0]; - var rootLayout = realRoot.getLayout(); - var length_1 = realRoot.children.length; - var rad = void 0; - var isLeft = void 0; - - if (targetLayout.x === rootLayout.x && node.isExpand === true && realRoot.children.length) { - var center = { - x: (realRoot.children[0].getLayout().x + realRoot.children[length_1 - 1].getLayout().x) / 2, - y: (realRoot.children[0].getLayout().y + realRoot.children[length_1 - 1].getLayout().y) / 2 - }; - rad = Math.atan2(center.y - rootLayout.y, center.x - rootLayout.x); - - if (rad < 0) { - rad = Math.PI * 2 + rad; - } - - isLeft = center.x < rootLayout.x; - - if (isLeft) { - rad = rad - Math.PI; - } - } else { - rad = Math.atan2(targetLayout.y - rootLayout.y, targetLayout.x - rootLayout.x); - - if (rad < 0) { - rad = Math.PI * 2 + rad; - } - - if (node.children.length === 0 || node.children.length !== 0 && node.isExpand === false) { - isLeft = targetLayout.x < rootLayout.x; - - if (isLeft) { - rad = rad - Math.PI; - } - } else { - isLeft = targetLayout.x > rootLayout.x; - - if (!isLeft) { - rad = rad - Math.PI; - } - } - } - - var textPosition = isLeft ? 'left' : 'right'; - var normalLabelModel = itemModel.getModel('label'); - var rotate = normalLabelModel.get('rotate'); - var labelRotateRadian = rotate * (Math.PI / 180); - var textContent = symbolPath.getTextContent(); - - if (textContent) { - symbolPath.setTextConfig({ - position: normalLabelModel.get('position') || textPosition, - rotation: rotate == null ? -rad : labelRotateRadian, - origin: 'center' - }); - textContent.setStyle('verticalAlign', 'middle'); - } - } // Handle status - - - var focus = itemModel.get(['emphasis', 'focus']); - var focusDataIndices = focus === 'relative' ? concatArray(node.getAncestorsIndices(), node.getDescendantIndices()) : focus === 'ancestor' ? node.getAncestorsIndices() : focus === 'descendant' ? node.getDescendantIndices() : null; - - if (focusDataIndices) { - // Modify the focus to data indices. - getECData(symbolEl).focus = focusDataIndices; - } - - drawEdge(seriesModel, node, virtualRoot, symbolEl, sourceOldLayout, sourceLayout, targetLayout, group); - - if (symbolEl.__edge) { - symbolEl.onHoverStateChange = function (toState) { - if (toState !== 'blur') { - // NOTE: Ensure the parent elements will been blurred firstly. - // According to the return of getAncestorsIndices and getDescendantIndices - // TODO: A bit tricky. - var parentEl = node.parentNode && data.getItemGraphicEl(node.parentNode.dataIndex); - - if (!(parentEl && parentEl.hoverState === HOVER_STATE_BLUR)) { - setStatesFlag(symbolEl.__edge, toState); - } - } - }; - } - } - - function drawEdge(seriesModel, node, virtualRoot, symbolEl, sourceOldLayout, sourceLayout, targetLayout, group) { - var itemModel = node.getModel(); - var edgeShape = seriesModel.get('edgeShape'); - var layout = seriesModel.get('layout'); - var orient = seriesModel.getOrient(); - var curvature = seriesModel.get(['lineStyle', 'curveness']); - var edgeForkPosition = seriesModel.get('edgeForkPosition'); - var lineStyle = itemModel.getModel('lineStyle').getLineStyle(); - var edge = symbolEl.__edge; // curve edge from node -> parent - // polyline edge from node -> children - - if (edgeShape === 'curve') { - if (node.parentNode && node.parentNode !== virtualRoot) { - if (!edge) { - edge = symbolEl.__edge = new BezierCurve({ - shape: getEdgeShape(layout, orient, curvature, sourceOldLayout, sourceOldLayout) - }); - } - - updateProps(edge, { - shape: getEdgeShape(layout, orient, curvature, sourceLayout, targetLayout) - }, seriesModel); - } - } else if (edgeShape === 'polyline') { - if (layout === 'orthogonal') { - if (node !== virtualRoot && node.children && node.children.length !== 0 && node.isExpand === true) { - var children = node.children; - var childPoints = []; - - for (var i = 0; i < children.length; i++) { - var childLayout = children[i].getLayout(); - childPoints.push([childLayout.x, childLayout.y]); - } - - if (!edge) { - edge = symbolEl.__edge = new TreePath({ - shape: { - parentPoint: [targetLayout.x, targetLayout.y], - childPoints: [[targetLayout.x, targetLayout.y]], - orient: orient, - forkPosition: edgeForkPosition - } - }); - } - - updateProps(edge, { - shape: { - parentPoint: [targetLayout.x, targetLayout.y], - childPoints: childPoints - } - }, seriesModel); - } - } else { - if ("development" !== 'production') { - throw new Error('The polyline edgeShape can only be used in orthogonal layout'); - } - } - } // show all edge when edgeShape is 'curve', filter node `isExpand` is false when edgeShape is 'polyline' - - - if (edge && !(edgeShape === 'polyline' && !node.isExpand)) { - edge.useStyle(defaults({ - strokeNoScale: true, - fill: null - }, lineStyle)); - setStatesStylesFromModel(edge, itemModel, 'lineStyle'); - setDefaultStateProxy(edge); - group.add(edge); - } - } - - function removeNodeEdge(node, data, group, seriesModel, removeAnimationOpt) { - var virtualRoot = data.tree.root; - - var _a = getSourceNode(virtualRoot, node), - source = _a.source, - sourceLayout = _a.sourceLayout; - - var symbolEl = data.getItemGraphicEl(node.dataIndex); - - if (!symbolEl) { - return; - } - - var sourceSymbolEl = data.getItemGraphicEl(source.dataIndex); - var sourceEdge = sourceSymbolEl.__edge; // 1. when expand the sub tree, delete the children node should delete the edge of - // the source at the same time. because the polyline edge shape is only owned by the source. - // 2.when the node is the only children of the source, delete the node should delete the edge of - // the source at the same time. the same reason as above. - - var edge = symbolEl.__edge || (source.isExpand === false || source.children.length === 1 ? sourceEdge : undefined); - var edgeShape = seriesModel.get('edgeShape'); - var layoutOpt = seriesModel.get('layout'); - var orient = seriesModel.get('orient'); - var curvature = seriesModel.get(['lineStyle', 'curveness']); - - if (edge) { - if (edgeShape === 'curve') { - removeElement(edge, { - shape: getEdgeShape(layoutOpt, orient, curvature, sourceLayout, sourceLayout), - style: { - opacity: 0 - } - }, seriesModel, { - cb: function () { - group.remove(edge); - }, - removeOpt: removeAnimationOpt - }); - } else if (edgeShape === 'polyline' && seriesModel.get('layout') === 'orthogonal') { - removeElement(edge, { - shape: { - parentPoint: [sourceLayout.x, sourceLayout.y], - childPoints: [[sourceLayout.x, sourceLayout.y]] - }, - style: { - opacity: 0 - } - }, seriesModel, { - cb: function () { - group.remove(edge); - }, - removeOpt: removeAnimationOpt - }); - } - } - } - - function getSourceNode(virtualRoot, node) { - var source = node.parentNode === virtualRoot ? node : node.parentNode || node; - var sourceLayout; - - while (sourceLayout = source.getLayout(), sourceLayout == null) { - source = source.parentNode === virtualRoot ? source : source.parentNode || source; - } - - return { - source: source, - sourceLayout: sourceLayout - }; - } - - function removeNode(data, dataIndex, symbolEl, group, seriesModel) { - var node = data.tree.getNodeByDataIndex(dataIndex); - var virtualRoot = data.tree.root; - var sourceLayout = getSourceNode(virtualRoot, node).sourceLayout; // Use same duration and easing with update to have more consistent animation. - - var removeAnimationOpt = { - duration: seriesModel.get('animationDurationUpdate'), - easing: seriesModel.get('animationEasingUpdate') - }; - removeElement(symbolEl, { - x: sourceLayout.x + 1, - y: sourceLayout.y + 1 - }, seriesModel, { - cb: function () { - group.remove(symbolEl); - data.setItemGraphicEl(dataIndex, null); - }, - removeOpt: removeAnimationOpt - }); - symbolEl.fadeOut(null, data.hostModel, { - fadeLabel: true, - animation: removeAnimationOpt - }); // remove edge as parent node - - node.children.forEach(function (childNode) { - removeNodeEdge(childNode, data, group, seriesModel, removeAnimationOpt); - }); // remove edge as child node - - removeNodeEdge(node, data, group, seriesModel, removeAnimationOpt); - } - - function getEdgeShape(layoutOpt, orient, curvature, sourceLayout, targetLayout) { - var cpx1; - var cpy1; - var cpx2; - var cpy2; - var x1; - var x2; - var y1; - var y2; - - if (layoutOpt === 'radial') { - x1 = sourceLayout.rawX; - y1 = sourceLayout.rawY; - x2 = targetLayout.rawX; - y2 = targetLayout.rawY; - var radialCoor1 = radialCoordinate(x1, y1); - var radialCoor2 = radialCoordinate(x1, y1 + (y2 - y1) * curvature); - var radialCoor3 = radialCoordinate(x2, y2 + (y1 - y2) * curvature); - var radialCoor4 = radialCoordinate(x2, y2); - return { - x1: radialCoor1.x || 0, - y1: radialCoor1.y || 0, - x2: radialCoor4.x || 0, - y2: radialCoor4.y || 0, - cpx1: radialCoor2.x || 0, - cpy1: radialCoor2.y || 0, - cpx2: radialCoor3.x || 0, - cpy2: radialCoor3.y || 0 - }; - } else { - x1 = sourceLayout.x; - y1 = sourceLayout.y; - x2 = targetLayout.x; - y2 = targetLayout.y; - - if (orient === 'LR' || orient === 'RL') { - cpx1 = x1 + (x2 - x1) * curvature; - cpy1 = y1; - cpx2 = x2 + (x1 - x2) * curvature; - cpy2 = y2; - } - - if (orient === 'TB' || orient === 'BT') { - cpx1 = x1; - cpy1 = y1 + (y2 - y1) * curvature; - cpx2 = x2; - cpy2 = y2 + (y1 - y2) * curvature; - } - } - - return { - x1: x1, - y1: y1, - x2: x2, - y2: y2, - cpx1: cpx1, - cpy1: cpy1, - cpx2: cpx2, - cpy2: cpy2 - }; - } - - var inner$7 = makeInner(); - - function linkSeriesData(opt) { - var mainData = opt.mainData; - var datas = opt.datas; - - if (!datas) { - datas = { - main: mainData - }; - opt.datasAttr = { - main: 'data' - }; - } - - opt.datas = opt.mainData = null; - linkAll(mainData, datas, opt); // Porxy data original methods. - - each(datas, function (data) { - each(mainData.TRANSFERABLE_METHODS, function (methodName) { - data.wrapMethod(methodName, curry(transferInjection, opt)); - }); - }); // Beyond transfer, additional features should be added to `cloneShallow`. - - mainData.wrapMethod('cloneShallow', curry(cloneShallowInjection, opt)); // Only mainData trigger change, because struct.update may trigger - // another changable methods, which may bring about dead lock. - - each(mainData.CHANGABLE_METHODS, function (methodName) { - mainData.wrapMethod(methodName, curry(changeInjection, opt)); - }); // Make sure datas contains mainData. - - assert(datas[mainData.dataType] === mainData); - } - - function transferInjection(opt, res) { - if (isMainData(this)) { - // Transfer datas to new main data. - var datas = extend({}, inner$7(this).datas); - datas[this.dataType] = res; - linkAll(res, datas, opt); - } else { - // Modify the reference in main data to point newData. - linkSingle(res, this.dataType, inner$7(this).mainData, opt); - } - - return res; - } - - function changeInjection(opt, res) { - opt.struct && opt.struct.update(); - return res; - } - - function cloneShallowInjection(opt, res) { - // cloneShallow, which brings about some fragilities, may be inappropriate - // to be exposed as an API. So for implementation simplicity we can make - // the restriction that cloneShallow of not-mainData should not be invoked - // outside, but only be invoked here. - each(inner$7(res).datas, function (data, dataType) { - data !== res && linkSingle(data.cloneShallow(), dataType, res, opt); - }); - return res; - } - /** - * Supplement method to List. - * - * @public - * @param [dataType] If not specified, return mainData. - */ - - - function getLinkedData(dataType) { - var mainData = inner$7(this).mainData; - return dataType == null || mainData == null ? mainData : inner$7(mainData).datas[dataType]; - } - /** - * Get list of all linked data - */ - - - function getLinkedDataAll() { - var mainData = inner$7(this).mainData; - return mainData == null ? [{ - data: mainData - }] : map(keys(inner$7(mainData).datas), function (type) { - return { - type: type, - data: inner$7(mainData).datas[type] - }; - }); - } - - function isMainData(data) { - return inner$7(data).mainData === data; - } - - function linkAll(mainData, datas, opt) { - inner$7(mainData).datas = {}; - each(datas, function (data, dataType) { - linkSingle(data, dataType, mainData, opt); - }); - } - - function linkSingle(data, dataType, mainData, opt) { - inner$7(mainData).datas[dataType] = data; - inner$7(data).mainData = mainData; - data.dataType = dataType; - - if (opt.struct) { - data[opt.structAttr] = opt.struct; - opt.struct[opt.datasAttr[dataType]] = data; - } // Supplement method. - - - data.getLinkedData = getLinkedData; - data.getLinkedDataAll = getLinkedDataAll; - } - - var TreeNode = - /** @class */ - function () { - function TreeNode(name, hostTree) { - this.depth = 0; - this.height = 0; - /** - * Reference to list item. - * Do not persistent dataIndex outside, - * besause it may be changed by list. - * If dataIndex -1, - * this node is logical deleted (filtered) in list. - */ - - this.dataIndex = -1; - this.children = []; - this.viewChildren = []; - this.isExpand = false; - this.name = name || ''; - this.hostTree = hostTree; - } - /** - * The node is removed. - */ - - - TreeNode.prototype.isRemoved = function () { - return this.dataIndex < 0; - }; - - TreeNode.prototype.eachNode = function (options, cb, context) { - if (isFunction(options)) { - context = cb; - cb = options; - options = null; - } - - options = options || {}; - - if (isString(options)) { - options = { - order: options - }; - } - - var order = options.order || 'preorder'; - var children = this[options.attr || 'children']; - var suppressVisitSub; - order === 'preorder' && (suppressVisitSub = cb.call(context, this)); - - for (var i = 0; !suppressVisitSub && i < children.length; i++) { - children[i].eachNode(options, cb, context); - } - - order === 'postorder' && cb.call(context, this); - }; - /** - * Update depth and height of this subtree. - */ - - - TreeNode.prototype.updateDepthAndHeight = function (depth) { - var height = 0; - this.depth = depth; - - for (var i = 0; i < this.children.length; i++) { - var child = this.children[i]; - child.updateDepthAndHeight(depth + 1); - - if (child.height > height) { - height = child.height; - } - } - - this.height = height + 1; - }; - - TreeNode.prototype.getNodeById = function (id) { - if (this.getId() === id) { - return this; - } - - for (var i = 0, children = this.children, len = children.length; i < len; i++) { - var res = children[i].getNodeById(id); - - if (res) { - return res; - } - } - }; - - TreeNode.prototype.contains = function (node) { - if (node === this) { - return true; - } - - for (var i = 0, children = this.children, len = children.length; i < len; i++) { - var res = children[i].contains(node); - - if (res) { - return res; - } - } - }; - /** - * @param includeSelf Default false. - * @return order: [root, child, grandchild, ...] - */ - - - TreeNode.prototype.getAncestors = function (includeSelf) { - var ancestors = []; - var node = includeSelf ? this : this.parentNode; - - while (node) { - ancestors.push(node); - node = node.parentNode; - } - - ancestors.reverse(); - return ancestors; - }; - - TreeNode.prototype.getAncestorsIndices = function () { - var indices = []; - var currNode = this; - - while (currNode) { - indices.push(currNode.dataIndex); - currNode = currNode.parentNode; - } - - indices.reverse(); - return indices; - }; - - TreeNode.prototype.getDescendantIndices = function () { - var indices = []; - this.eachNode(function (childNode) { - indices.push(childNode.dataIndex); - }); - return indices; - }; - - TreeNode.prototype.getValue = function (dimension) { - var data = this.hostTree.data; - return data.getStore().get(data.getDimensionIndex(dimension || 'value'), this.dataIndex); - }; - - TreeNode.prototype.setLayout = function (layout, merge) { - this.dataIndex >= 0 && this.hostTree.data.setItemLayout(this.dataIndex, layout, merge); - }; - /** - * @return {Object} layout - */ - - - TreeNode.prototype.getLayout = function () { - return this.hostTree.data.getItemLayout(this.dataIndex); - }; // @depcrecated - // getModel<T = unknown, S extends keyof T = keyof T>(path: S): Model<T[S]> - // eslint-disable-next-line @typescript-eslint/no-unused-vars - - - TreeNode.prototype.getModel = function (path) { - if (this.dataIndex < 0) { - return; - } - - var hostTree = this.hostTree; - var itemModel = hostTree.data.getItemModel(this.dataIndex); - return itemModel.getModel(path); - }; // TODO: TYPE More specific model - - - TreeNode.prototype.getLevelModel = function () { - return (this.hostTree.levelModels || [])[this.depth]; - }; - - TreeNode.prototype.setVisual = function (key, value) { - this.dataIndex >= 0 && this.hostTree.data.setItemVisual(this.dataIndex, key, value); - }; - /** - * Get item visual - * FIXME: make return type better - */ - - - TreeNode.prototype.getVisual = function (key) { - return this.hostTree.data.getItemVisual(this.dataIndex, key); - }; - - TreeNode.prototype.getRawIndex = function () { - return this.hostTree.data.getRawIndex(this.dataIndex); - }; - - TreeNode.prototype.getId = function () { - return this.hostTree.data.getId(this.dataIndex); - }; - /** - * index in parent's children - */ - - - TreeNode.prototype.getChildIndex = function () { - if (this.parentNode) { - var children = this.parentNode.children; - - for (var i = 0; i < children.length; ++i) { - if (children[i] === this) { - return i; - } - } - - return -1; - } - - return -1; - }; - /** - * if this is an ancestor of another node - * - * @param node another node - * @return if is ancestor - */ - - - TreeNode.prototype.isAncestorOf = function (node) { - var parent = node.parentNode; - - while (parent) { - if (parent === this) { - return true; - } - - parent = parent.parentNode; - } - - return false; - }; - /** - * if this is an descendant of another node - * - * @param node another node - * @return if is descendant - */ - - - TreeNode.prototype.isDescendantOf = function (node) { - return node !== this && node.isAncestorOf(this); - }; - - return TreeNode; - }(); - - var Tree = - /** @class */ - function () { - function Tree(hostModel) { - this.type = 'tree'; - this._nodes = []; - this.hostModel = hostModel; - } - - Tree.prototype.eachNode = function (options, cb, context) { - this.root.eachNode(options, cb, context); - }; - - Tree.prototype.getNodeByDataIndex = function (dataIndex) { - var rawIndex = this.data.getRawIndex(dataIndex); - return this._nodes[rawIndex]; - }; - - Tree.prototype.getNodeById = function (name) { - return this.root.getNodeById(name); - }; - /** - * Update item available by list, - * when list has been performed options like 'filterSelf' or 'map'. - */ - - - Tree.prototype.update = function () { - var data = this.data; - var nodes = this._nodes; - - for (var i = 0, len = nodes.length; i < len; i++) { - nodes[i].dataIndex = -1; - } - - for (var i = 0, len = data.count(); i < len; i++) { - nodes[data.getRawIndex(i)].dataIndex = i; - } - }; - /** - * Clear all layouts - */ - - - Tree.prototype.clearLayouts = function () { - this.data.clearItemLayouts(); - }; - /** - * data node format: - * { - * name: ... - * value: ... - * children: [ - * { - * name: ... - * value: ... - * children: ... - * }, - * ... - * ] - * } - */ - - - Tree.createTree = function (dataRoot, hostModel, beforeLink) { - var tree = new Tree(hostModel); - var listData = []; - var dimMax = 1; - buildHierarchy(dataRoot); - - function buildHierarchy(dataNode, parentNode) { - var value = dataNode.value; - dimMax = Math.max(dimMax, isArray(value) ? value.length : 1); - listData.push(dataNode); - var node = new TreeNode(convertOptionIdName(dataNode.name, ''), tree); - parentNode ? addChild(node, parentNode) : tree.root = node; - - tree._nodes.push(node); - - var children = dataNode.children; - - if (children) { - for (var i = 0; i < children.length; i++) { - buildHierarchy(children[i], node); - } - } - } - - tree.root.updateDepthAndHeight(0); - var dimensions = prepareSeriesDataSchema(listData, { - coordDimensions: ['value'], - dimensionsCount: dimMax - }).dimensions; - var list = new SeriesData(dimensions, hostModel); - list.initData(listData); - beforeLink && beforeLink(list); - linkSeriesData({ - mainData: list, - struct: tree, - structAttr: 'tree' - }); - tree.update(); - return tree; - }; - - return Tree; - }(); - /** - * It is needed to consider the mess of 'list', 'hostModel' when creating a TreeNote, - * so this function is not ready and not necessary to be public. - */ - - - function addChild(child, node) { - var children = node.children; - - if (child.parentNode === node) { - return; - } - - children.push(child); - child.parentNode = node; - } - - function retrieveTargetInfo(payload, validPayloadTypes, seriesModel) { - if (payload && indexOf(validPayloadTypes, payload.type) >= 0) { - var root = seriesModel.getData().tree.root; - var targetNode = payload.targetNode; - - if (isString(targetNode)) { - targetNode = root.getNodeById(targetNode); - } - - if (targetNode && root.contains(targetNode)) { - return { - node: targetNode - }; - } - - var targetNodeId = payload.targetNodeId; - - if (targetNodeId != null && (targetNode = root.getNodeById(targetNodeId))) { - return { - node: targetNode - }; - } - } - } // Not includes the given node at the last item. - - function getPathToRoot(node) { - var path = []; - - while (node) { - node = node.parentNode; - node && path.push(node); - } - - return path.reverse(); - } - function aboveViewRoot(viewRoot, node) { - var viewPath = getPathToRoot(viewRoot); - return indexOf(viewPath, node) >= 0; - } // From root to the input node (the input node will be included). - - function wrapTreePathInfo(node, seriesModel) { - var treePathInfo = []; - - while (node) { - var nodeDataIndex = node.dataIndex; - treePathInfo.push({ - name: node.name, - dataIndex: nodeDataIndex, - value: seriesModel.getRawValue(nodeDataIndex) - }); - node = node.parentNode; - } - - treePathInfo.reverse(); - return treePathInfo; - } - - var TreeSeriesModel = - /** @class */ - function (_super) { - __extends(TreeSeriesModel, _super); - - function TreeSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.hasSymbolVisual = true; // Do it self. - - _this.ignoreStyleOnData = true; - return _this; - } - /** - * Init a tree data structure from data in option series - */ - - - TreeSeriesModel.prototype.getInitialData = function (option) { - // create a virtual root - var root = { - name: option.name, - children: option.data - }; - var leaves = option.leaves || {}; - var leavesModel = new Model(leaves, this, this.ecModel); - var tree = Tree.createTree(root, this, beforeLink); - - function beforeLink(nodeData) { - nodeData.wrapMethod('getItemModel', function (model, idx) { - var node = tree.getNodeByDataIndex(idx); - - if (!(node && node.children.length && node.isExpand)) { - model.parentModel = leavesModel; - } - - return model; - }); - } - - var treeDepth = 0; - tree.eachNode('preorder', function (node) { - if (node.depth > treeDepth) { - treeDepth = node.depth; - } - }); - var expandAndCollapse = option.expandAndCollapse; - var expandTreeDepth = expandAndCollapse && option.initialTreeDepth >= 0 ? option.initialTreeDepth : treeDepth; - tree.root.eachNode('preorder', function (node) { - var item = node.hostTree.data.getRawDataItem(node.dataIndex); // Add item.collapsed != null, because users can collapse node original in the series.data. - - node.isExpand = item && item.collapsed != null ? !item.collapsed : node.depth <= expandTreeDepth; - }); - return tree.data; - }; - /** - * Make the configuration 'orient' backward compatibly, with 'horizontal = LR', 'vertical = TB'. - * @returns {string} orient - */ - - - TreeSeriesModel.prototype.getOrient = function () { - var orient = this.get('orient'); - - if (orient === 'horizontal') { - orient = 'LR'; - } else if (orient === 'vertical') { - orient = 'TB'; - } - - return orient; - }; - - TreeSeriesModel.prototype.setZoom = function (zoom) { - this.option.zoom = zoom; - }; - - TreeSeriesModel.prototype.setCenter = function (center) { - this.option.center = center; - }; - - TreeSeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - var tree = this.getData().tree; - var realRoot = tree.root.children[0]; - var node = tree.getNodeByDataIndex(dataIndex); - var value = node.getValue(); - var name = node.name; - - while (node && node !== realRoot) { - name = node.parentNode.name + '.' + name; - node = node.parentNode; - } - - return createTooltipMarkup('nameValue', { - name: name, - value: value, - noValue: isNaN(value) || value == null - }); - }; // Add tree path to tooltip param - - - TreeSeriesModel.prototype.getDataParams = function (dataIndex) { - var params = _super.prototype.getDataParams.apply(this, arguments); - - var node = this.getData().tree.getNodeByDataIndex(dataIndex); - params.treeAncestors = wrapTreePathInfo(node, this); - params.collapsed = !node.isExpand; - return params; - }; - - TreeSeriesModel.type = 'series.tree'; // can support the position parameters 'left', 'top','right','bottom', 'width', - // 'height' in the setOption() with 'merge' mode normal. - - TreeSeriesModel.layoutMode = 'box'; - TreeSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'view', - // the position of the whole view - left: '12%', - top: '12%', - right: '12%', - bottom: '12%', - // the layout of the tree, two value can be selected, 'orthogonal' or 'radial' - layout: 'orthogonal', - // value can be 'polyline' - edgeShape: 'curve', - edgeForkPosition: '50%', - // true | false | 'move' | 'scale', see module:component/helper/RoamController. - roam: false, - // Symbol size scale ratio in roam - nodeScaleRatio: 0.4, - // Default on center of graph - center: null, - zoom: 1, - orient: 'LR', - symbol: 'emptyCircle', - symbolSize: 7, - expandAndCollapse: true, - initialTreeDepth: 2, - lineStyle: { - color: '#ccc', - width: 1.5, - curveness: 0.5 - }, - itemStyle: { - color: 'lightsteelblue', - // borderColor: '#c23531', - borderWidth: 1.5 - }, - label: { - show: true - }, - animationEasing: 'linear', - animationDuration: 700, - animationDurationUpdate: 500 - }; - return TreeSeriesModel; - }(SeriesModel); - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - /** - * Traverse the tree from bottom to top and do something - */ - function eachAfter(root, callback, separation) { - var nodes = [root]; - var next = []; - var node; - - while (node = nodes.pop()) { - // jshint ignore:line - next.push(node); - - if (node.isExpand) { - var children = node.children; - - if (children.length) { - for (var i = 0; i < children.length; i++) { - nodes.push(children[i]); - } - } - } - } - - while (node = next.pop()) { - // jshint ignore:line - callback(node, separation); - } - } - /** - * Traverse the tree from top to bottom and do something - */ - - - function eachBefore(root, callback) { - var nodes = [root]; - var node; - - while (node = nodes.pop()) { - // jshint ignore:line - callback(node); - - if (node.isExpand) { - var children = node.children; - - if (children.length) { - for (var i = children.length - 1; i >= 0; i--) { - nodes.push(children[i]); - } - } - } - } - } - - function treeLayout(ecModel, api) { - ecModel.eachSeriesByType('tree', function (seriesModel) { - commonLayout(seriesModel, api); - }); - } - - function commonLayout(seriesModel, api) { - var layoutInfo = getViewRect$1(seriesModel, api); - seriesModel.layoutInfo = layoutInfo; - var layout = seriesModel.get('layout'); - var width = 0; - var height = 0; - var separation$1 = null; - - if (layout === 'radial') { - width = 2 * Math.PI; - height = Math.min(layoutInfo.height, layoutInfo.width) / 2; - separation$1 = separation(function (node1, node2) { - return (node1.parentNode === node2.parentNode ? 1 : 2) / node1.depth; - }); - } else { - width = layoutInfo.width; - height = layoutInfo.height; - separation$1 = separation(); - } - - var virtualRoot = seriesModel.getData().tree.root; - var realRoot = virtualRoot.children[0]; - - if (realRoot) { - init$2(virtualRoot); - eachAfter(realRoot, firstWalk, separation$1); - virtualRoot.hierNode.modifier = -realRoot.hierNode.prelim; - eachBefore(realRoot, secondWalk); - var left_1 = realRoot; - var right_1 = realRoot; - var bottom_1 = realRoot; - eachBefore(realRoot, function (node) { - var x = node.getLayout().x; - - if (x < left_1.getLayout().x) { - left_1 = node; - } - - if (x > right_1.getLayout().x) { - right_1 = node; - } - - if (node.depth > bottom_1.depth) { - bottom_1 = node; - } - }); - var delta = left_1 === right_1 ? 1 : separation$1(left_1, right_1) / 2; - var tx_1 = delta - left_1.getLayout().x; - var kx_1 = 0; - var ky_1 = 0; - var coorX_1 = 0; - var coorY_1 = 0; - - if (layout === 'radial') { - kx_1 = width / (right_1.getLayout().x + delta + tx_1); // here we use (node.depth - 1), bucause the real root's depth is 1 - - ky_1 = height / (bottom_1.depth - 1 || 1); - eachBefore(realRoot, function (node) { - coorX_1 = (node.getLayout().x + tx_1) * kx_1; - coorY_1 = (node.depth - 1) * ky_1; - var finalCoor = radialCoordinate(coorX_1, coorY_1); - node.setLayout({ - x: finalCoor.x, - y: finalCoor.y, - rawX: coorX_1, - rawY: coorY_1 - }, true); - }); - } else { - var orient_1 = seriesModel.getOrient(); - - if (orient_1 === 'RL' || orient_1 === 'LR') { - ky_1 = height / (right_1.getLayout().x + delta + tx_1); - kx_1 = width / (bottom_1.depth - 1 || 1); - eachBefore(realRoot, function (node) { - coorY_1 = (node.getLayout().x + tx_1) * ky_1; - coorX_1 = orient_1 === 'LR' ? (node.depth - 1) * kx_1 : width - (node.depth - 1) * kx_1; - node.setLayout({ - x: coorX_1, - y: coorY_1 - }, true); - }); - } else if (orient_1 === 'TB' || orient_1 === 'BT') { - kx_1 = width / (right_1.getLayout().x + delta + tx_1); - ky_1 = height / (bottom_1.depth - 1 || 1); - eachBefore(realRoot, function (node) { - coorX_1 = (node.getLayout().x + tx_1) * kx_1; - coorY_1 = orient_1 === 'TB' ? (node.depth - 1) * ky_1 : height - (node.depth - 1) * ky_1; - node.setLayout({ - x: coorX_1, - y: coorY_1 - }, true); - }); - } - } - } - } - - function treeVisual(ecModel) { - ecModel.eachSeriesByType('tree', function (seriesModel) { - var data = seriesModel.getData(); - var tree = data.tree; - tree.eachNode(function (node) { - var model = node.getModel(); // TODO Optimize - - var style = model.getModel('itemStyle').getItemStyle(); - var existsStyle = data.ensureUniqueItemVisual(node.dataIndex, 'style'); - extend(existsStyle, style); - }); - }); - } - - function installTreeAction(registers) { - registers.registerAction({ - type: 'treeExpandAndCollapse', - event: 'treeExpandAndCollapse', - update: 'update' - }, function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'series', - subType: 'tree', - query: payload - }, function (seriesModel) { - var dataIndex = payload.dataIndex; - var tree = seriesModel.getData().tree; - var node = tree.getNodeByDataIndex(dataIndex); - node.isExpand = !node.isExpand; - }); - }); - registers.registerAction({ - type: 'treeRoam', - event: 'treeRoam', - // Here we set 'none' instead of 'update', because roam action - // just need to update the transform matrix without having to recalculate - // the layout. So don't need to go through the whole update process, such - // as 'dataPrcocess', 'coordSystemUpdate', 'layout' and so on. - update: 'none' - }, function (payload, ecModel, api) { - ecModel.eachComponent({ - mainType: 'series', - subType: 'tree', - query: payload - }, function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - var res = updateCenterAndZoom(coordSys, payload, undefined, api); - seriesModel.setCenter && seriesModel.setCenter(res.center); - seriesModel.setZoom && seriesModel.setZoom(res.zoom); - }); - }); - } - - function install$b(registers) { - registers.registerChartView(TreeView); - registers.registerSeriesModel(TreeSeriesModel); - registers.registerLayout(treeLayout); - registers.registerVisual(treeVisual); - installTreeAction(registers); - } - - var actionTypes = ['treemapZoomToNode', 'treemapRender', 'treemapMove']; - function installTreemapAction(registers) { - for (var i = 0; i < actionTypes.length; i++) { - registers.registerAction({ - type: actionTypes[i], - update: 'updateView' - }, noop); - } - - registers.registerAction({ - type: 'treemapRootToNode', - update: 'updateView' - }, function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'series', - subType: 'treemap', - query: payload - }, handleRootToNode); - - function handleRootToNode(model, index) { - var types = ['treemapZoomToNode', 'treemapRootToNode']; - var targetInfo = retrieveTargetInfo(payload, types, model); - - if (targetInfo) { - var originViewRoot = model.getViewRoot(); - - if (originViewRoot) { - payload.direction = aboveViewRoot(originViewRoot, targetInfo.node) ? 'rollUp' : 'drillDown'; - } - - model.resetViewRoot(targetInfo.node); - } - } - }); - } - - function enableAriaDecalForTree(seriesModel) { - var data = seriesModel.getData(); - var tree = data.tree; - var decalPaletteScope = {}; - tree.eachNode(function (node) { - // Use decal of level 1 node - var current = node; - - while (current && current.depth > 1) { - current = current.parentNode; - } - - var decal = getDecalFromPalette(seriesModel.ecModel, current.name || current.dataIndex + '', decalPaletteScope); - node.setVisual('decal', decal); - }); - } - - var TreemapSeriesModel = - /** @class */ - function (_super) { - __extends(TreemapSeriesModel, _super); - - function TreemapSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TreemapSeriesModel.type; - _this.preventUsingHoverLayer = true; - return _this; - } - /** - * @override - */ - - - TreemapSeriesModel.prototype.getInitialData = function (option, ecModel) { - // Create a virtual root. - var root = { - name: option.name, - children: option.data - }; - completeTreeValue(root); - var levels = option.levels || []; // Used in "visual priority" in `treemapVisual.js`. - // This way is a little tricky, must satisfy the precondition: - // 1. There is no `treeNode.getModel('itemStyle.xxx')` used. - // 2. The `Model.prototype.getModel()` will not use any clone-like way. - - var designatedVisualItemStyle = this.designatedVisualItemStyle = {}; - var designatedVisualModel = new Model({ - itemStyle: designatedVisualItemStyle - }, this, ecModel); - levels = option.levels = setDefault(levels, ecModel); - var levelModels = map(levels || [], function (levelDefine) { - return new Model(levelDefine, designatedVisualModel, ecModel); - }, this); // Make sure always a new tree is created when setOption, - // in TreemapView, we check whether oldTree === newTree - // to choose mappings approach among old shapes and new shapes. - - var tree = Tree.createTree(root, this, beforeLink); - - function beforeLink(nodeData) { - nodeData.wrapMethod('getItemModel', function (model, idx) { - var node = tree.getNodeByDataIndex(idx); - var levelModel = node ? levelModels[node.depth] : null; // If no levelModel, we also need `designatedVisualModel`. - - model.parentModel = levelModel || designatedVisualModel; - return model; - }); - } - - return tree.data; - }; - - TreemapSeriesModel.prototype.optionUpdated = function () { - this.resetViewRoot(); - }; - /** - * @override - * @param {number} dataIndex - * @param {boolean} [mutipleSeries=false] - */ - - - TreemapSeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - var data = this.getData(); - var value = this.getRawValue(dataIndex); - var name = data.getName(dataIndex); - return createTooltipMarkup('nameValue', { - name: name, - value: value - }); - }; - /** - * Add tree path to tooltip param - * - * @override - * @param {number} dataIndex - * @return {Object} - */ - - - TreemapSeriesModel.prototype.getDataParams = function (dataIndex) { - var params = _super.prototype.getDataParams.apply(this, arguments); - - var node = this.getData().tree.getNodeByDataIndex(dataIndex); - params.treeAncestors = wrapTreePathInfo(node, this); // compatitable the previous code. - - params.treePathInfo = params.treeAncestors; - return params; - }; - /** - * @public - * @param {Object} layoutInfo { - * x: containerGroup x - * y: containerGroup y - * width: containerGroup width - * height: containerGroup height - * } - */ - - - TreemapSeriesModel.prototype.setLayoutInfo = function (layoutInfo) { - /** - * @readOnly - * @type {Object} - */ - this.layoutInfo = this.layoutInfo || {}; - extend(this.layoutInfo, layoutInfo); - }; - /** - * @param {string} id - * @return {number} index - */ - - - TreemapSeriesModel.prototype.mapIdToIndex = function (id) { - // A feature is implemented: - // index is monotone increasing with the sequence of - // input id at the first time. - // This feature can make sure that each data item and its - // mapped color have the same index between data list and - // color list at the beginning, which is useful for user - // to adjust data-color mapping. - - /** - * @private - * @type {Object} - */ - var idIndexMap = this._idIndexMap; - - if (!idIndexMap) { - idIndexMap = this._idIndexMap = createHashMap(); - /** - * @private - * @type {number} - */ - - this._idIndexMapCount = 0; - } - - var index = idIndexMap.get(id); - - if (index == null) { - idIndexMap.set(id, index = this._idIndexMapCount++); - } - - return index; - }; - - TreemapSeriesModel.prototype.getViewRoot = function () { - return this._viewRoot; - }; - - TreemapSeriesModel.prototype.resetViewRoot = function (viewRoot) { - viewRoot ? this._viewRoot = viewRoot : viewRoot = this._viewRoot; - var root = this.getRawData().tree.root; - - if (!viewRoot || viewRoot !== root && !root.contains(viewRoot)) { - this._viewRoot = root; - } - }; - - TreemapSeriesModel.prototype.enableAriaDecal = function () { - enableAriaDecalForTree(this); - }; - - TreemapSeriesModel.type = 'series.treemap'; - TreemapSeriesModel.layoutMode = 'box'; - TreemapSeriesModel.defaultOption = { - // Disable progressive rendering - progressive: 0, - // size: ['80%', '80%'], // deprecated, compatible with ec2. - left: 'center', - top: 'middle', - width: '80%', - height: '80%', - sort: true, - clipWindow: 'origin', - squareRatio: 0.5 * (1 + Math.sqrt(5)), - leafDepth: null, - drillDownIcon: '▶', - // to align specialized icon. ▷▶❒❐▼✚ - zoomToNodeRatio: 0.32 * 0.32, - roam: true, - nodeClick: 'zoomToNode', - animation: true, - animationDurationUpdate: 900, - animationEasing: 'quinticInOut', - breadcrumb: { - show: true, - height: 22, - left: 'center', - top: 'bottom', - // right - // bottom - emptyItemWidth: 25, - itemStyle: { - color: 'rgba(0,0,0,0.7)', - textStyle: { - color: '#fff' - } - }, - emphasis: { - itemStyle: { - color: 'rgba(0,0,0,0.9)' // '#5793f3', - - } - } - }, - label: { - show: true, - // Do not use textDistance, for ellipsis rect just the same as treemap node rect. - distance: 0, - padding: 5, - position: 'inside', - // formatter: null, - color: '#fff', - overflow: 'truncate' // align - // verticalAlign - - }, - upperLabel: { - show: false, - position: [0, '50%'], - height: 20, - // formatter: null, - // color: '#fff', - overflow: 'truncate', - // align: null, - verticalAlign: 'middle' - }, - itemStyle: { - color: null, - colorAlpha: null, - colorSaturation: null, - borderWidth: 0, - gapWidth: 0, - borderColor: '#fff', - borderColorSaturation: null // If specified, borderColor will be ineffective, and the - // border color is evaluated by color of current node and - // borderColorSaturation. - - }, - emphasis: { - upperLabel: { - show: true, - position: [0, '50%'], - overflow: 'truncate', - verticalAlign: 'middle' - } - }, - visualDimension: 0, - visualMin: null, - visualMax: null, - color: [], - // level[n].color (if necessary). - // + Specify color list of each level. level[0].color would be global - // color list if not specified. (see method `setDefault`). - // + But set as a empty array to forbid fetch color from global palette - // when using nodeModel.get('color'), otherwise nodes on deep level - // will always has color palette set and are not able to inherit color - // from parent node. - // + TreemapSeries.color can not be set as 'none', otherwise effect - // legend color fetching (see seriesColor.js). - colorAlpha: null, - colorSaturation: null, - colorMappingBy: 'index', - visibleMin: 10, - // be rendered. Only works when sort is 'asc' or 'desc'. - childrenVisibleMin: null, - // grandchildren will not show. - // Why grandchildren? If not grandchildren but children, - // some siblings show children and some not, - // the appearance may be mess and not consistent, - levels: [] // Each item: { - // visibleMin, itemStyle, visualDimension, label - // } - - }; - return TreemapSeriesModel; - }(SeriesModel); - /** - * @param {Object} dataNode - */ - - - function completeTreeValue(dataNode) { - // Postorder travel tree. - // If value of none-leaf node is not set, - // calculate it by suming up the value of all children. - var sum = 0; - each(dataNode.children, function (child) { - completeTreeValue(child); - var childValue = child.value; - isArray(childValue) && (childValue = childValue[0]); - sum += childValue; - }); - var thisValue = dataNode.value; - - if (isArray(thisValue)) { - thisValue = thisValue[0]; - } - - if (thisValue == null || isNaN(thisValue)) { - thisValue = sum; - } // Value should not less than 0. - - - if (thisValue < 0) { - thisValue = 0; - } - - isArray(dataNode.value) ? dataNode.value[0] = thisValue : dataNode.value = thisValue; - } - /** - * set default to level configuration - */ - - - function setDefault(levels, ecModel) { - var globalColorList = normalizeToArray(ecModel.get('color')); - var globalDecalList = normalizeToArray(ecModel.get(['aria', 'decal', 'decals'])); - - if (!globalColorList) { - return; - } - - levels = levels || []; - var hasColorDefine; - var hasDecalDefine; - each(levels, function (levelDefine) { - var model = new Model(levelDefine); - var modelColor = model.get('color'); - var modelDecal = model.get('decal'); - - if (model.get(['itemStyle', 'color']) || modelColor && modelColor !== 'none') { - hasColorDefine = true; - } - - if (model.get(['itemStyle', 'decal']) || modelDecal && modelDecal !== 'none') { - hasDecalDefine = true; - } - }); - var level0 = levels[0] || (levels[0] = {}); - - if (!hasColorDefine) { - level0.color = globalColorList.slice(); - } - - if (!hasDecalDefine && globalDecalList) { - level0.decal = globalDecalList.slice(); - } - - return levels; - } - - var TEXT_PADDING = 8; - var ITEM_GAP = 8; - var ARRAY_LENGTH = 5; - - var Breadcrumb = - /** @class */ - function () { - function Breadcrumb(containerGroup) { - this.group = new Group(); - containerGroup.add(this.group); - } - - Breadcrumb.prototype.render = function (seriesModel, api, targetNode, onSelect) { - var model = seriesModel.getModel('breadcrumb'); - var thisGroup = this.group; - thisGroup.removeAll(); - - if (!model.get('show') || !targetNode) { - return; - } - - var normalStyleModel = model.getModel('itemStyle'); - var emphasisModel = model.getModel('emphasis'); - var textStyleModel = normalStyleModel.getModel('textStyle'); - var emphasisTextStyleModel = emphasisModel.getModel(['itemStyle', 'textStyle']); - var layoutParam = { - pos: { - left: model.get('left'), - right: model.get('right'), - top: model.get('top'), - bottom: model.get('bottom') - }, - box: { - width: api.getWidth(), - height: api.getHeight() - }, - emptyItemWidth: model.get('emptyItemWidth'), - totalWidth: 0, - renderList: [] - }; - - this._prepare(targetNode, layoutParam, textStyleModel); - - this._renderContent(seriesModel, layoutParam, normalStyleModel, emphasisModel, textStyleModel, emphasisTextStyleModel, onSelect); - - positionElement(thisGroup, layoutParam.pos, layoutParam.box); - }; - /** - * Prepare render list and total width - * @private - */ - - - Breadcrumb.prototype._prepare = function (targetNode, layoutParam, textStyleModel) { - for (var node = targetNode; node; node = node.parentNode) { - var text = convertOptionIdName(node.getModel().get('name'), ''); - var textRect = textStyleModel.getTextRect(text); - var itemWidth = Math.max(textRect.width + TEXT_PADDING * 2, layoutParam.emptyItemWidth); - layoutParam.totalWidth += itemWidth + ITEM_GAP; - layoutParam.renderList.push({ - node: node, - text: text, - width: itemWidth - }); - } - }; - /** - * @private - */ - - - Breadcrumb.prototype._renderContent = function (seriesModel, layoutParam, normalStyleModel, emphasisModel, textStyleModel, emphasisTextStyleModel, onSelect) { - // Start rendering. - var lastX = 0; - var emptyItemWidth = layoutParam.emptyItemWidth; - var height = seriesModel.get(['breadcrumb', 'height']); - var availableSize = getAvailableSize(layoutParam.pos, layoutParam.box); - var totalWidth = layoutParam.totalWidth; - var renderList = layoutParam.renderList; - var emphasisItemStyle = emphasisModel.getModel('itemStyle').getItemStyle(); - - for (var i = renderList.length - 1; i >= 0; i--) { - var item = renderList[i]; - var itemNode = item.node; - var itemWidth = item.width; - var text = item.text; // Hdie text and shorten width if necessary. - - if (totalWidth > availableSize.width) { - totalWidth -= itemWidth - emptyItemWidth; - itemWidth = emptyItemWidth; - text = null; - } - - var el = new Polygon({ - shape: { - points: makeItemPoints(lastX, 0, itemWidth, height, i === renderList.length - 1, i === 0) - }, - style: defaults(normalStyleModel.getItemStyle(), { - lineJoin: 'bevel' - }), - textContent: new ZRText({ - style: createTextStyle(textStyleModel, { - text: text - }) - }), - textConfig: { - position: 'inside' - }, - z2: Z2_EMPHASIS_LIFT * 1e4, - onclick: curry(onSelect, itemNode) - }); - el.disableLabelAnimation = true; - el.getTextContent().ensureState('emphasis').style = createTextStyle(emphasisTextStyleModel, { - text: text - }); - el.ensureState('emphasis').style = emphasisItemStyle; - toggleHoverEmphasis(el, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - this.group.add(el); - packEventData(el, seriesModel, itemNode); - lastX += itemWidth + ITEM_GAP; - } - }; - - Breadcrumb.prototype.remove = function () { - this.group.removeAll(); - }; - - return Breadcrumb; - }(); - - function makeItemPoints(x, y, itemWidth, itemHeight, head, tail) { - var points = [[head ? x : x - ARRAY_LENGTH, y], [x + itemWidth, y], [x + itemWidth, y + itemHeight], [head ? x : x - ARRAY_LENGTH, y + itemHeight]]; - !tail && points.splice(2, 0, [x + itemWidth + ARRAY_LENGTH, y + itemHeight / 2]); - !head && points.push([x, y + itemHeight / 2]); - return points; - } // Package custom mouse event. - - - function packEventData(el, seriesModel, itemNode) { - getECData(el).eventData = { - componentType: 'series', - componentSubType: 'treemap', - componentIndex: seriesModel.componentIndex, - seriesIndex: seriesModel.seriesIndex, - seriesName: seriesModel.name, - seriesType: 'treemap', - selfType: 'breadcrumb', - nodeData: { - dataIndex: itemNode && itemNode.dataIndex, - name: itemNode && itemNode.name - }, - treePathInfo: itemNode && wrapTreePathInfo(itemNode, seriesModel) - }; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - /** - * Animate multiple elements with a single done-callback. - * - * @example - * animation - * .createWrap() - * .add(el1, {x: 10, y: 10}) - * .add(el2, {shape: {width: 500}, style: {fill: 'red'}}, 400) - * .done(function () { // done }) - * .start('cubicOut'); - */ - var AnimationWrap = - /** @class */ - function () { - function AnimationWrap() { - this._storage = []; - this._elExistsMap = {}; - } - /** - * Caution: a el can only be added once, otherwise 'done' - * might not be called. This method checks this (by el.id), - * suppresses adding and returns false when existing el found. - * - * @return Whether adding succeeded. - */ - - - AnimationWrap.prototype.add = function (el, target, duration, delay, easing) { - if (this._elExistsMap[el.id]) { - return false; - } - - this._elExistsMap[el.id] = true; - - this._storage.push({ - el: el, - target: target, - duration: duration, - delay: delay, - easing: easing - }); - - return true; - }; - /** - * Only execute when animation done/aborted. - */ - - - AnimationWrap.prototype.finished = function (callback) { - this._finishedCallback = callback; - return this; - }; - /** - * Will stop exist animation firstly. - */ - - - AnimationWrap.prototype.start = function () { - var _this = this; - - var count = this._storage.length; - - var checkTerminate = function () { - count--; - - if (count <= 0) { - // Guard. - _this._storage.length = 0; - _this._elExistsMap = {}; - _this._finishedCallback && _this._finishedCallback(); - } - }; - - for (var i = 0, len = this._storage.length; i < len; i++) { - var item = this._storage[i]; - item.el.animateTo(item.target, { - duration: item.duration, - delay: item.delay, - easing: item.easing, - setToFinal: true, - done: checkTerminate, - aborted: checkTerminate - }); - } - - return this; - }; - - return AnimationWrap; - }(); - - function createWrap() { - return new AnimationWrap(); - } - - var Group$1 = Group; - var Rect$1 = Rect; - var DRAG_THRESHOLD = 3; - var PATH_LABEL_NOAMAL = 'label'; - var PATH_UPPERLABEL_NORMAL = 'upperLabel'; // Should larger than emphasis states lift z - - var Z2_BASE = Z2_EMPHASIS_LIFT * 10; // Should bigger than every z2. - - var Z2_BG = Z2_EMPHASIS_LIFT * 2; - var Z2_CONTENT = Z2_EMPHASIS_LIFT * 3; - var getStateItemStyle = makeStyleMapper([['fill', 'color'], // `borderColor` and `borderWidth` has been occupied, - // so use `stroke` to indicate the stroke of the rect. - ['stroke', 'strokeColor'], ['lineWidth', 'strokeWidth'], ['shadowBlur'], ['shadowOffsetX'], ['shadowOffsetY'], ['shadowColor'] // Option decal is in `DecalObject` but style.decal is in `PatternObject`. - // So do not transfer decal directly. - ]); - - var getItemStyleNormal = function (model) { - // Normal style props should include emphasis style props. - var itemStyle = getStateItemStyle(model); // Clear styles set by emphasis. - - itemStyle.stroke = itemStyle.fill = itemStyle.lineWidth = null; - return itemStyle; - }; - - var inner$8 = makeInner(); - - var TreemapView = - /** @class */ - function (_super) { - __extends(TreemapView, _super); - - function TreemapView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TreemapView.type; - _this._state = 'ready'; - _this._storage = createStorage(); - return _this; - } - /** - * @override - */ - - - TreemapView.prototype.render = function (seriesModel, ecModel, api, payload) { - var models = ecModel.findComponents({ - mainType: 'series', - subType: 'treemap', - query: payload - }); - - if (indexOf(models, seriesModel) < 0) { - return; - } - - this.seriesModel = seriesModel; - this.api = api; - this.ecModel = ecModel; - var types = ['treemapZoomToNode', 'treemapRootToNode']; - var targetInfo = retrieveTargetInfo(payload, types, seriesModel); - var payloadType = payload && payload.type; - var layoutInfo = seriesModel.layoutInfo; - var isInit = !this._oldTree; - var thisStorage = this._storage; // Mark new root when action is treemapRootToNode. - - var reRoot = payloadType === 'treemapRootToNode' && targetInfo && thisStorage ? { - rootNodeGroup: thisStorage.nodeGroup[targetInfo.node.getRawIndex()], - direction: payload.direction - } : null; - - var containerGroup = this._giveContainerGroup(layoutInfo); - - var hasAnimation = seriesModel.get('animation'); - - var renderResult = this._doRender(containerGroup, seriesModel, reRoot); - - hasAnimation && !isInit && (!payloadType || payloadType === 'treemapZoomToNode' || payloadType === 'treemapRootToNode') ? this._doAnimation(containerGroup, renderResult, seriesModel, reRoot) : renderResult.renderFinally(); - - this._resetController(api); - - this._renderBreadcrumb(seriesModel, api, targetInfo); - }; - - TreemapView.prototype._giveContainerGroup = function (layoutInfo) { - var containerGroup = this._containerGroup; - - if (!containerGroup) { - // FIXME - // 加一层containerGroup是为了clip,但是现在clip功能并没有实现。 - containerGroup = this._containerGroup = new Group$1(); - - this._initEvents(containerGroup); - - this.group.add(containerGroup); - } - - containerGroup.x = layoutInfo.x; - containerGroup.y = layoutInfo.y; - return containerGroup; - }; - - TreemapView.prototype._doRender = function (containerGroup, seriesModel, reRoot) { - var thisTree = seriesModel.getData().tree; - var oldTree = this._oldTree; // Clear last shape records. - - var lastsForAnimation = createStorage(); - var thisStorage = createStorage(); - var oldStorage = this._storage; - var willInvisibleEls = []; - - function doRenderNode(thisNode, oldNode, parentGroup, depth) { - return renderNode(seriesModel, thisStorage, oldStorage, reRoot, lastsForAnimation, willInvisibleEls, thisNode, oldNode, parentGroup, depth); - } // Notice: When thisTree and oldTree are the same tree (see list.cloneShallow), - // the oldTree is actually losted, so we cannot find all of the old graphic - // elements from tree. So we use this strategy: make element storage, move - // from old storage to new storage, clear old storage. - - - dualTravel(thisTree.root ? [thisTree.root] : [], oldTree && oldTree.root ? [oldTree.root] : [], containerGroup, thisTree === oldTree || !oldTree, 0); // Process all removing. - - var willDeleteEls = clearStorage(oldStorage); - this._oldTree = thisTree; - this._storage = thisStorage; - return { - lastsForAnimation: lastsForAnimation, - willDeleteEls: willDeleteEls, - renderFinally: renderFinally - }; - - function dualTravel(thisViewChildren, oldViewChildren, parentGroup, sameTree, depth) { - // When 'render' is triggered by action, - // 'this' and 'old' may be the same tree, - // we use rawIndex in that case. - if (sameTree) { - oldViewChildren = thisViewChildren; - each(thisViewChildren, function (child, index) { - !child.isRemoved() && processNode(index, index); - }); - } // Diff hierarchically (diff only in each subtree, but not whole). - // because, consistency of view is important. - else { - new DataDiffer(oldViewChildren, thisViewChildren, getKey, getKey).add(processNode).update(processNode).remove(curry(processNode, null)).execute(); - } - - function getKey(node) { - // Identify by name or raw index. - return node.getId(); - } - - function processNode(newIndex, oldIndex) { - var thisNode = newIndex != null ? thisViewChildren[newIndex] : null; - var oldNode = oldIndex != null ? oldViewChildren[oldIndex] : null; - var group = doRenderNode(thisNode, oldNode, parentGroup, depth); - group && dualTravel(thisNode && thisNode.viewChildren || [], oldNode && oldNode.viewChildren || [], group, sameTree, depth + 1); - } - } - - function clearStorage(storage) { - var willDeleteEls = createStorage(); - storage && each(storage, function (store, storageName) { - var delEls = willDeleteEls[storageName]; - each(store, function (el) { - el && (delEls.push(el), inner$8(el).willDelete = true); - }); - }); - return willDeleteEls; - } - - function renderFinally() { - each(willDeleteEls, function (els) { - each(els, function (el) { - el.parent && el.parent.remove(el); - }); - }); - each(willInvisibleEls, function (el) { - el.invisible = true; // Setting invisible is for optimizing, so no need to set dirty, - // just mark as invisible. - - el.dirty(); - }); - } - }; - - TreemapView.prototype._doAnimation = function (containerGroup, renderResult, seriesModel, reRoot) { - var durationOption = seriesModel.get('animationDurationUpdate'); - var easingOption = seriesModel.get('animationEasing'); // TODO: do not support function until necessary. - - var duration = (isFunction(durationOption) ? 0 : durationOption) || 0; - var easing = (isFunction(easingOption) ? null : easingOption) || 'cubicOut'; - var animationWrap = createWrap(); // Make delete animations. - - each(renderResult.willDeleteEls, function (store, storageName) { - each(store, function (el, rawIndex) { - if (el.invisible) { - return; - } - - var parent = el.parent; // Always has parent, and parent is nodeGroup. - - var target; - var innerStore = inner$8(parent); - - if (reRoot && reRoot.direction === 'drillDown') { - target = parent === reRoot.rootNodeGroup // This is the content element of view root. - // Only `content` will enter this branch, because - // `background` and `nodeGroup` will not be deleted. - ? { - shape: { - x: 0, - y: 0, - width: innerStore.nodeWidth, - height: innerStore.nodeHeight - }, - style: { - opacity: 0 - } - } // Others. - : { - style: { - opacity: 0 - } - }; - } else { - var targetX = 0; - var targetY = 0; - - if (!innerStore.willDelete) { - // Let node animate to right-bottom corner, cooperating with fadeout, - // which is appropriate for user understanding. - // Divided by 2 for reRoot rolling up effect. - targetX = innerStore.nodeWidth / 2; - targetY = innerStore.nodeHeight / 2; - } - - target = storageName === 'nodeGroup' ? { - x: targetX, - y: targetY, - style: { - opacity: 0 - } - } : { - shape: { - x: targetX, - y: targetY, - width: 0, - height: 0 - }, - style: { - opacity: 0 - } - }; - } // TODO: do not support delay until necessary. - - - target && animationWrap.add(el, target, duration, 0, easing); - }); - }); // Make other animations - - each(this._storage, function (store, storageName) { - each(store, function (el, rawIndex) { - var last = renderResult.lastsForAnimation[storageName][rawIndex]; - var target = {}; - - if (!last) { - return; - } - - if (el instanceof Group) { - if (last.oldX != null) { - target.x = el.x; - target.y = el.y; - el.x = last.oldX; - el.y = last.oldY; - } - } else { - if (last.oldShape) { - target.shape = extend({}, el.shape); - el.setShape(last.oldShape); - } - - if (last.fadein) { - el.setStyle('opacity', 0); - target.style = { - opacity: 1 - }; - } // When animation is stopped for succedent animation starting, - // el.style.opacity might not be 1 - else if (el.style.opacity !== 1) { - target.style = { - opacity: 1 - }; - } - } - - animationWrap.add(el, target, duration, 0, easing); - }); - }, this); - this._state = 'animating'; - animationWrap.finished(bind(function () { - this._state = 'ready'; - renderResult.renderFinally(); - }, this)).start(); - }; - - TreemapView.prototype._resetController = function (api) { - var controller = this._controller; // Init controller. - - if (!controller) { - controller = this._controller = new RoamController(api.getZr()); - controller.enable(this.seriesModel.get('roam')); - controller.on('pan', bind(this._onPan, this)); - controller.on('zoom', bind(this._onZoom, this)); - } - - var rect = new BoundingRect(0, 0, api.getWidth(), api.getHeight()); - controller.setPointerChecker(function (e, x, y) { - return rect.contain(x, y); - }); - }; - - TreemapView.prototype._clearController = function () { - var controller = this._controller; - - if (controller) { - controller.dispose(); - controller = null; - } - }; - - TreemapView.prototype._onPan = function (e) { - if (this._state !== 'animating' && (Math.abs(e.dx) > DRAG_THRESHOLD || Math.abs(e.dy) > DRAG_THRESHOLD)) { - // These param must not be cached. - var root = this.seriesModel.getData().tree.root; - - if (!root) { - return; - } - - var rootLayout = root.getLayout(); - - if (!rootLayout) { - return; - } - - this.api.dispatchAction({ - type: 'treemapMove', - from: this.uid, - seriesId: this.seriesModel.id, - rootRect: { - x: rootLayout.x + e.dx, - y: rootLayout.y + e.dy, - width: rootLayout.width, - height: rootLayout.height - } - }); - } - }; - - TreemapView.prototype._onZoom = function (e) { - var mouseX = e.originX; - var mouseY = e.originY; - - if (this._state !== 'animating') { - // These param must not be cached. - var root = this.seriesModel.getData().tree.root; - - if (!root) { - return; - } - - var rootLayout = root.getLayout(); - - if (!rootLayout) { - return; - } - - var rect = new BoundingRect(rootLayout.x, rootLayout.y, rootLayout.width, rootLayout.height); - var layoutInfo = this.seriesModel.layoutInfo; // Transform mouse coord from global to containerGroup. - - mouseX -= layoutInfo.x; - mouseY -= layoutInfo.y; // Scale root bounding rect. - - var m = create$1(); - translate(m, m, [-mouseX, -mouseY]); - scale$1(m, m, [e.scale, e.scale]); - translate(m, m, [mouseX, mouseY]); - rect.applyTransform(m); - this.api.dispatchAction({ - type: 'treemapRender', - from: this.uid, - seriesId: this.seriesModel.id, - rootRect: { - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height - } - }); - } - }; - - TreemapView.prototype._initEvents = function (containerGroup) { - var _this = this; - - containerGroup.on('click', function (e) { - if (_this._state !== 'ready') { - return; - } - - var nodeClick = _this.seriesModel.get('nodeClick', true); - - if (!nodeClick) { - return; - } - - var targetInfo = _this.findTarget(e.offsetX, e.offsetY); - - if (!targetInfo) { - return; - } - - var node = targetInfo.node; - - if (node.getLayout().isLeafRoot) { - _this._rootToNode(targetInfo); - } else { - if (nodeClick === 'zoomToNode') { - _this._zoomToNode(targetInfo); - } else if (nodeClick === 'link') { - var itemModel = node.hostTree.data.getItemModel(node.dataIndex); - var link = itemModel.get('link', true); - var linkTarget = itemModel.get('target', true) || 'blank'; - link && windowOpen(link, linkTarget); - } - } - }, this); - }; - - TreemapView.prototype._renderBreadcrumb = function (seriesModel, api, targetInfo) { - var _this = this; - - if (!targetInfo) { - targetInfo = seriesModel.get('leafDepth', true) != null ? { - node: seriesModel.getViewRoot() - } // FIXME - // better way? - // Find breadcrumb tail on center of containerGroup. - : this.findTarget(api.getWidth() / 2, api.getHeight() / 2); - - if (!targetInfo) { - targetInfo = { - node: seriesModel.getData().tree.root - }; - } - } - - (this._breadcrumb || (this._breadcrumb = new Breadcrumb(this.group))).render(seriesModel, api, targetInfo.node, function (node) { - if (_this._state !== 'animating') { - aboveViewRoot(seriesModel.getViewRoot(), node) ? _this._rootToNode({ - node: node - }) : _this._zoomToNode({ - node: node - }); - } - }); - }; - /** - * @override - */ - - - TreemapView.prototype.remove = function () { - this._clearController(); - - this._containerGroup && this._containerGroup.removeAll(); - this._storage = createStorage(); - this._state = 'ready'; - this._breadcrumb && this._breadcrumb.remove(); - }; - - TreemapView.prototype.dispose = function () { - this._clearController(); - }; - - TreemapView.prototype._zoomToNode = function (targetInfo) { - this.api.dispatchAction({ - type: 'treemapZoomToNode', - from: this.uid, - seriesId: this.seriesModel.id, - targetNode: targetInfo.node - }); - }; - - TreemapView.prototype._rootToNode = function (targetInfo) { - this.api.dispatchAction({ - type: 'treemapRootToNode', - from: this.uid, - seriesId: this.seriesModel.id, - targetNode: targetInfo.node - }); - }; - /** - * @public - * @param {number} x Global coord x. - * @param {number} y Global coord y. - * @return {Object} info If not found, return undefined; - * @return {number} info.node Target node. - * @return {number} info.offsetX x refer to target node. - * @return {number} info.offsetY y refer to target node. - */ - - - TreemapView.prototype.findTarget = function (x, y) { - var targetInfo; - var viewRoot = this.seriesModel.getViewRoot(); - viewRoot.eachNode({ - attr: 'viewChildren', - order: 'preorder' - }, function (node) { - var bgEl = this._storage.background[node.getRawIndex()]; // If invisible, there might be no element. - - - if (bgEl) { - var point = bgEl.transformCoordToLocal(x, y); - var shape = bgEl.shape; // For performance consideration, don't use 'getBoundingRect'. - - if (shape.x <= point[0] && point[0] <= shape.x + shape.width && shape.y <= point[1] && point[1] <= shape.y + shape.height) { - targetInfo = { - node: node, - offsetX: point[0], - offsetY: point[1] - }; - } else { - return false; // Suppress visit subtree. - } - } - }, this); - return targetInfo; - }; - - TreemapView.type = 'treemap'; - return TreemapView; - }(ChartView); - /** - * @inner - */ - - - function createStorage() { - return { - nodeGroup: [], - background: [], - content: [] - }; - } - /** - * @inner - * @return Return undefined means do not travel further. - */ - - - function renderNode(seriesModel, thisStorage, oldStorage, reRoot, lastsForAnimation, willInvisibleEls, thisNode, oldNode, parentGroup, depth) { - // Whether under viewRoot. - if (!thisNode) { - // Deleting nodes will be performed finally. This method just find - // element from old storage, or create new element, set them to new - // storage, and set styles. - return; - } // ------------------------------------------------------------------- - // Start of closure variables available in "Procedures in renderNode". - - - var thisLayout = thisNode.getLayout(); - var data = seriesModel.getData(); - var nodeModel = thisNode.getModel(); // Only for enabling highlight/downplay. Clear firstly. - // Because some node will not be rendered. - - data.setItemGraphicEl(thisNode.dataIndex, null); - - if (!thisLayout || !thisLayout.isInView) { - return; - } - - var thisWidth = thisLayout.width; - var thisHeight = thisLayout.height; - var borderWidth = thisLayout.borderWidth; - var thisInvisible = thisLayout.invisible; - var thisRawIndex = thisNode.getRawIndex(); - var oldRawIndex = oldNode && oldNode.getRawIndex(); - var thisViewChildren = thisNode.viewChildren; - var upperHeight = thisLayout.upperHeight; - var isParent = thisViewChildren && thisViewChildren.length; - var itemStyleNormalModel = nodeModel.getModel('itemStyle'); - var itemStyleEmphasisModel = nodeModel.getModel(['emphasis', 'itemStyle']); - var itemStyleBlurModel = nodeModel.getModel(['blur', 'itemStyle']); - var itemStyleSelectModel = nodeModel.getModel(['select', 'itemStyle']); - var borderRadius = itemStyleNormalModel.get('borderRadius') || 0; // End of closure ariables available in "Procedures in renderNode". - // ----------------------------------------------------------------- - // Node group - - var group = giveGraphic('nodeGroup', Group$1); - - if (!group) { - return; - } - - parentGroup.add(group); // x,y are not set when el is above view root. - - group.x = thisLayout.x || 0; - group.y = thisLayout.y || 0; - group.markRedraw(); - inner$8(group).nodeWidth = thisWidth; - inner$8(group).nodeHeight = thisHeight; - - if (thisLayout.isAboveViewRoot) { - return group; - } // Background - - - var bg = giveGraphic('background', Rect$1, depth, Z2_BG); - bg && renderBackground(group, bg, isParent && thisLayout.upperLabelHeight); - var emphasisModel = nodeModel.getModel('emphasis'); - var focus = emphasisModel.get('focus'); - var blurScope = emphasisModel.get('blurScope'); - var isDisabled = emphasisModel.get('disabled'); - var focusOrIndices = focus === 'ancestor' ? thisNode.getAncestorsIndices() : focus === 'descendant' ? thisNode.getDescendantIndices() : focus; // No children, render content. - - if (isParent) { - // Because of the implementation about "traverse" in graphic hover style, we - // can not set hover listener on the "group" of non-leaf node. Otherwise the - // hover event from the descendents will be listenered. - if (isHighDownDispatcher(group)) { - setAsHighDownDispatcher(group, false); - } - - if (bg) { - setAsHighDownDispatcher(bg, !isDisabled); // Only for enabling highlight/downplay. - - data.setItemGraphicEl(thisNode.dataIndex, bg); - enableHoverFocus(bg, focusOrIndices, blurScope); - } - } else { - var content = giveGraphic('content', Rect$1, depth, Z2_CONTENT); - content && renderContent(group, content); - bg.disableMorphing = true; - - if (bg && isHighDownDispatcher(bg)) { - setAsHighDownDispatcher(bg, false); - } - - setAsHighDownDispatcher(group, !isDisabled); // Only for enabling highlight/downplay. - - data.setItemGraphicEl(thisNode.dataIndex, group); - enableHoverFocus(group, focusOrIndices, blurScope); - } - - return group; // ---------------------------- - // | Procedures in renderNode | - // ---------------------------- - - function renderBackground(group, bg, useUpperLabel) { - var ecData = getECData(bg); // For tooltip. - - ecData.dataIndex = thisNode.dataIndex; - ecData.seriesIndex = seriesModel.seriesIndex; - bg.setShape({ - x: 0, - y: 0, - width: thisWidth, - height: thisHeight, - r: borderRadius - }); - - if (thisInvisible) { - // If invisible, do not set visual, otherwise the element will - // change immediately before animation. We think it is OK to - // remain its origin color when moving out of the view window. - processInvisible(bg); - } else { - bg.invisible = false; - var style = thisNode.getVisual('style'); - var visualBorderColor = style.stroke; - var normalStyle = getItemStyleNormal(itemStyleNormalModel); - normalStyle.fill = visualBorderColor; - var emphasisStyle = getStateItemStyle(itemStyleEmphasisModel); - emphasisStyle.fill = itemStyleEmphasisModel.get('borderColor'); - var blurStyle = getStateItemStyle(itemStyleBlurModel); - blurStyle.fill = itemStyleBlurModel.get('borderColor'); - var selectStyle = getStateItemStyle(itemStyleSelectModel); - selectStyle.fill = itemStyleSelectModel.get('borderColor'); - - if (useUpperLabel) { - var upperLabelWidth = thisWidth - 2 * borderWidth; - prepareText( // PENDING: convert ZRColor to ColorString for text. - bg, visualBorderColor, style.opacity, { - x: borderWidth, - y: 0, - width: upperLabelWidth, - height: upperHeight - }); - } // For old bg. - else { - bg.removeTextContent(); - } - - bg.setStyle(normalStyle); - bg.ensureState('emphasis').style = emphasisStyle; - bg.ensureState('blur').style = blurStyle; - bg.ensureState('select').style = selectStyle; - setDefaultStateProxy(bg); - } - - group.add(bg); - } - - function renderContent(group, content) { - var ecData = getECData(content); // For tooltip. - - ecData.dataIndex = thisNode.dataIndex; - ecData.seriesIndex = seriesModel.seriesIndex; - var contentWidth = Math.max(thisWidth - 2 * borderWidth, 0); - var contentHeight = Math.max(thisHeight - 2 * borderWidth, 0); - content.culling = true; - content.setShape({ - x: borderWidth, - y: borderWidth, - width: contentWidth, - height: contentHeight, - r: borderRadius - }); - - if (thisInvisible) { - // If invisible, do not set visual, otherwise the element will - // change immediately before animation. We think it is OK to - // remain its origin color when moving out of the view window. - processInvisible(content); - } else { - content.invisible = false; - var nodeStyle = thisNode.getVisual('style'); - var visualColor = nodeStyle.fill; - var normalStyle = getItemStyleNormal(itemStyleNormalModel); - normalStyle.fill = visualColor; - normalStyle.decal = nodeStyle.decal; - var emphasisStyle = getStateItemStyle(itemStyleEmphasisModel); - var blurStyle = getStateItemStyle(itemStyleBlurModel); - var selectStyle = getStateItemStyle(itemStyleSelectModel); // PENDING: convert ZRColor to ColorString for text. - - prepareText(content, visualColor, nodeStyle.opacity, null); - content.setStyle(normalStyle); - content.ensureState('emphasis').style = emphasisStyle; - content.ensureState('blur').style = blurStyle; - content.ensureState('select').style = selectStyle; - setDefaultStateProxy(content); - } - - group.add(content); - } - - function processInvisible(element) { - // Delay invisible setting utill animation finished, - // avoid element vanish suddenly before animation. - !element.invisible && willInvisibleEls.push(element); - } - - function prepareText(rectEl, visualColor, visualOpacity, // Can be null/undefined - upperLabelRect) { - var normalLabelModel = nodeModel.getModel(upperLabelRect ? PATH_UPPERLABEL_NORMAL : PATH_LABEL_NOAMAL); - var defaultText = convertOptionIdName(nodeModel.get('name'), null); - var isShow = normalLabelModel.getShallow('show'); - setLabelStyle(rectEl, getLabelStatesModels(nodeModel, upperLabelRect ? PATH_UPPERLABEL_NORMAL : PATH_LABEL_NOAMAL), { - defaultText: isShow ? defaultText : null, - inheritColor: visualColor, - defaultOpacity: visualOpacity, - labelFetcher: seriesModel, - labelDataIndex: thisNode.dataIndex - }); - var textEl = rectEl.getTextContent(); - - if (!textEl) { - return; - } - - var textStyle = textEl.style; - var textPadding = normalizeCssArray(textStyle.padding || 0); - - if (upperLabelRect) { - rectEl.setTextConfig({ - layoutRect: upperLabelRect - }); - textEl.disableLabelLayout = true; - } - - textEl.beforeUpdate = function () { - var width = Math.max((upperLabelRect ? upperLabelRect.width : rectEl.shape.width) - textPadding[1] - textPadding[3], 0); - var height = Math.max((upperLabelRect ? upperLabelRect.height : rectEl.shape.height) - textPadding[0] - textPadding[2], 0); - - if (textStyle.width !== width || textStyle.height !== height) { - textEl.setStyle({ - width: width, - height: height - }); - } - }; - - textStyle.truncateMinChar = 2; - textStyle.lineOverflow = 'truncate'; - addDrillDownIcon(textStyle, upperLabelRect, thisLayout); - var textEmphasisState = textEl.getState('emphasis'); - addDrillDownIcon(textEmphasisState ? textEmphasisState.style : null, upperLabelRect, thisLayout); - } - - function addDrillDownIcon(style, upperLabelRect, thisLayout) { - var text = style ? style.text : null; - - if (!upperLabelRect && thisLayout.isLeafRoot && text != null) { - var iconChar = seriesModel.get('drillDownIcon', true); - style.text = iconChar ? iconChar + ' ' + text : text; - } - } - - function giveGraphic(storageName, Ctor, depth, z) { - var element = oldRawIndex != null && oldStorage[storageName][oldRawIndex]; - var lasts = lastsForAnimation[storageName]; - - if (element) { - // Remove from oldStorage - oldStorage[storageName][oldRawIndex] = null; - prepareAnimationWhenHasOld(lasts, element); - } // If invisible and no old element, do not create new element (for optimizing). - else if (!thisInvisible) { - element = new Ctor(); - - if (element instanceof Displayable) { - element.z2 = calculateZ2(depth, z); - } - - prepareAnimationWhenNoOld(lasts, element); - } // Set to thisStorage - - - return thisStorage[storageName][thisRawIndex] = element; - } - - function prepareAnimationWhenHasOld(lasts, element) { - var lastCfg = lasts[thisRawIndex] = {}; - - if (element instanceof Group$1) { - lastCfg.oldX = element.x; - lastCfg.oldY = element.y; - } else { - lastCfg.oldShape = extend({}, element.shape); - } - } // If a element is new, we need to find the animation start point carefully, - // otherwise it will looks strange when 'zoomToNode'. - - - function prepareAnimationWhenNoOld(lasts, element) { - var lastCfg = lasts[thisRawIndex] = {}; - var parentNode = thisNode.parentNode; - var isGroup = element instanceof Group; - - if (parentNode && (!reRoot || reRoot.direction === 'drillDown')) { - var parentOldX = 0; - var parentOldY = 0; // New nodes appear from right-bottom corner in 'zoomToNode' animation. - // For convenience, get old bounding rect from background. - - var parentOldBg = lastsForAnimation.background[parentNode.getRawIndex()]; - - if (!reRoot && parentOldBg && parentOldBg.oldShape) { - parentOldX = parentOldBg.oldShape.width; - parentOldY = parentOldBg.oldShape.height; - } // When no parent old shape found, its parent is new too, - // so we can just use {x:0, y:0}. - - - if (isGroup) { - lastCfg.oldX = 0; - lastCfg.oldY = parentOldY; - } else { - lastCfg.oldShape = { - x: parentOldX, - y: parentOldY, - width: 0, - height: 0 - }; - } - } // Fade in, user can be aware that these nodes are new. - - - lastCfg.fadein = !isGroup; - } - } // We cannot set all background with the same z, because the behaviour of - // drill down and roll up differ background creation sequence from tree - // hierarchy sequence, which cause lower background elements to overlap - // upper ones. So we calculate z based on depth. - // Moreover, we try to shrink down z interval to [0, 1] to avoid that - // treemap with large z overlaps other components. - - - function calculateZ2(depth, z2InLevel) { - return depth * Z2_BASE + z2InLevel; - } - - var each$3 = each; - var isObject$3 = isObject; - var CATEGORY_DEFAULT_VISUAL_INDEX = -1; - - var VisualMapping = - /** @class */ - function () { - function VisualMapping(option) { - var mappingMethod = option.mappingMethod; - var visualType = option.type; - var thisOption = this.option = clone(option); - this.type = visualType; - this.mappingMethod = mappingMethod; - this._normalizeData = normalizers[mappingMethod]; - var visualHandler = VisualMapping.visualHandlers[visualType]; - this.applyVisual = visualHandler.applyVisual; - this.getColorMapper = visualHandler.getColorMapper; - this._normalizedToVisual = visualHandler._normalizedToVisual[mappingMethod]; - - if (mappingMethod === 'piecewise') { - normalizeVisualRange(thisOption); - preprocessForPiecewise(thisOption); - } else if (mappingMethod === 'category') { - thisOption.categories ? preprocessForSpecifiedCategory(thisOption) // categories is ordinal when thisOption.categories not specified, - // which need no more preprocess except normalize visual. - : normalizeVisualRange(thisOption, true); - } else { - // mappingMethod === 'linear' or 'fixed' - assert(mappingMethod !== 'linear' || thisOption.dataExtent); - normalizeVisualRange(thisOption); - } - } - - VisualMapping.prototype.mapValueToVisual = function (value) { - var normalized = this._normalizeData(value); - - return this._normalizedToVisual(normalized, value); - }; - - VisualMapping.prototype.getNormalizer = function () { - return bind(this._normalizeData, this); - }; - /** - * List available visual types. - * - * @public - * @return {Array.<string>} - */ - - - VisualMapping.listVisualTypes = function () { - return keys(VisualMapping.visualHandlers); - }; // /** - // * @public - // */ - // static addVisualHandler(name, handler) { - // visualHandlers[name] = handler; - // } - - /** - * @public - */ - - - VisualMapping.isValidType = function (visualType) { - return VisualMapping.visualHandlers.hasOwnProperty(visualType); - }; - /** - * Convenient method. - * Visual can be Object or Array or primary type. - */ - - - VisualMapping.eachVisual = function (visual, callback, context) { - if (isObject(visual)) { - each(visual, callback, context); - } else { - callback.call(context, visual); - } - }; - - VisualMapping.mapVisual = function (visual, callback, context) { - var isPrimary; - var newVisual = isArray(visual) ? [] : isObject(visual) ? {} : (isPrimary = true, null); - VisualMapping.eachVisual(visual, function (v, key) { - var newVal = callback.call(context, v, key); - isPrimary ? newVisual = newVal : newVisual[key] = newVal; - }); - return newVisual; - }; - /** - * Retrieve visual properties from given object. - */ - - - VisualMapping.retrieveVisuals = function (obj) { - var ret = {}; - var hasVisual; - obj && each$3(VisualMapping.visualHandlers, function (h, visualType) { - if (obj.hasOwnProperty(visualType)) { - ret[visualType] = obj[visualType]; - hasVisual = true; - } - }); - return hasVisual ? ret : null; - }; - /** - * Give order to visual types, considering colorSaturation, colorAlpha depends on color. - * - * @public - * @param {(Object|Array)} visualTypes If Object, like: {color: ..., colorSaturation: ...} - * IF Array, like: ['color', 'symbol', 'colorSaturation'] - * @return {Array.<string>} Sorted visual types. - */ - - - VisualMapping.prepareVisualTypes = function (visualTypes) { - if (isArray(visualTypes)) { - visualTypes = visualTypes.slice(); - } else if (isObject$3(visualTypes)) { - var types_1 = []; - each$3(visualTypes, function (item, type) { - types_1.push(type); - }); - visualTypes = types_1; - } else { - return []; - } - - visualTypes.sort(function (type1, type2) { - // color should be front of colorSaturation, colorAlpha, ... - // symbol and symbolSize do not matter. - return type2 === 'color' && type1 !== 'color' && type1.indexOf('color') === 0 ? 1 : -1; - }); - return visualTypes; - }; - /** - * 'color', 'colorSaturation', 'colorAlpha', ... are depends on 'color'. - * Other visuals are only depends on themself. - */ - - - VisualMapping.dependsOn = function (visualType1, visualType2) { - return visualType2 === 'color' ? !!(visualType1 && visualType1.indexOf(visualType2) === 0) : visualType1 === visualType2; - }; - /** - * @param value - * @param pieceList [{value: ..., interval: [min, max]}, ...] - * Always from small to big. - * @param findClosestWhenOutside Default to be false - * @return index - */ - - - VisualMapping.findPieceIndex = function (value, pieceList, findClosestWhenOutside) { - var possibleI; - var abs = Infinity; // value has the higher priority. - - for (var i = 0, len = pieceList.length; i < len; i++) { - var pieceValue = pieceList[i].value; - - if (pieceValue != null) { - if (pieceValue === value // FIXME - // It is supposed to compare value according to value type of dimension, - // but currently value type can exactly be string or number. - // Compromise for numeric-like string (like '12'), especially - // in the case that visualMap.categories is ['22', '33']. - || isString(pieceValue) && pieceValue === value + '') { - return i; - } - - findClosestWhenOutside && updatePossible(pieceValue, i); - } - } - - for (var i = 0, len = pieceList.length; i < len; i++) { - var piece = pieceList[i]; - var interval = piece.interval; - var close_1 = piece.close; - - if (interval) { - if (interval[0] === -Infinity) { - if (littleThan(close_1[1], value, interval[1])) { - return i; - } - } else if (interval[1] === Infinity) { - if (littleThan(close_1[0], interval[0], value)) { - return i; - } - } else if (littleThan(close_1[0], interval[0], value) && littleThan(close_1[1], value, interval[1])) { - return i; - } - - findClosestWhenOutside && updatePossible(interval[0], i); - findClosestWhenOutside && updatePossible(interval[1], i); - } - } - - if (findClosestWhenOutside) { - return value === Infinity ? pieceList.length - 1 : value === -Infinity ? 0 : possibleI; - } - - function updatePossible(val, index) { - var newAbs = Math.abs(val - value); - - if (newAbs < abs) { - abs = newAbs; - possibleI = index; - } - } - }; - - VisualMapping.visualHandlers = { - color: { - applyVisual: makeApplyVisual('color'), - getColorMapper: function () { - var thisOption = this.option; - return bind(thisOption.mappingMethod === 'category' ? function (value, isNormalized) { - !isNormalized && (value = this._normalizeData(value)); - return doMapCategory.call(this, value); - } : function (value, isNormalized, out) { - // If output rgb array - // which will be much faster and useful in pixel manipulation - var returnRGBArray = !!out; - !isNormalized && (value = this._normalizeData(value)); - out = fastLerp(value, thisOption.parsedVisual, out); - return returnRGBArray ? out : stringify(out, 'rgba'); - }, this); - }, - _normalizedToVisual: { - linear: function (normalized) { - return stringify(fastLerp(normalized, this.option.parsedVisual), 'rgba'); - }, - category: doMapCategory, - piecewise: function (normalized, value) { - var result = getSpecifiedVisual.call(this, value); - - if (result == null) { - result = stringify(fastLerp(normalized, this.option.parsedVisual), 'rgba'); - } - - return result; - }, - fixed: doMapFixed - } - }, - colorHue: makePartialColorVisualHandler(function (color$1, value) { - return modifyHSL(color$1, value); - }), - colorSaturation: makePartialColorVisualHandler(function (color$1, value) { - return modifyHSL(color$1, null, value); - }), - colorLightness: makePartialColorVisualHandler(function (color$1, value) { - return modifyHSL(color$1, null, null, value); - }), - colorAlpha: makePartialColorVisualHandler(function (color$1, value) { - return modifyAlpha(color$1, value); - }), - decal: { - applyVisual: makeApplyVisual('decal'), - _normalizedToVisual: { - linear: null, - category: doMapCategory, - piecewise: null, - fixed: null - } - }, - opacity: { - applyVisual: makeApplyVisual('opacity'), - _normalizedToVisual: createNormalizedToNumericVisual([0, 1]) - }, - liftZ: { - applyVisual: makeApplyVisual('liftZ'), - _normalizedToVisual: { - linear: doMapFixed, - category: doMapFixed, - piecewise: doMapFixed, - fixed: doMapFixed - } - }, - symbol: { - applyVisual: function (value, getter, setter) { - var symbolCfg = this.mapValueToVisual(value); - setter('symbol', symbolCfg); - }, - _normalizedToVisual: { - linear: doMapToArray, - category: doMapCategory, - piecewise: function (normalized, value) { - var result = getSpecifiedVisual.call(this, value); - - if (result == null) { - result = doMapToArray.call(this, normalized); - } - - return result; - }, - fixed: doMapFixed - } - }, - symbolSize: { - applyVisual: makeApplyVisual('symbolSize'), - _normalizedToVisual: createNormalizedToNumericVisual([0, 1]) - } - }; - return VisualMapping; - }(); - - function preprocessForPiecewise(thisOption) { - var pieceList = thisOption.pieceList; - thisOption.hasSpecialVisual = false; - each(pieceList, function (piece, index) { - piece.originIndex = index; // piece.visual is "result visual value" but not - // a visual range, so it does not need to be normalized. - - if (piece.visual != null) { - thisOption.hasSpecialVisual = true; - } - }); - } - - function preprocessForSpecifiedCategory(thisOption) { - // Hash categories. - var categories = thisOption.categories; - var categoryMap = thisOption.categoryMap = {}; - var visual = thisOption.visual; - each$3(categories, function (cate, index) { - categoryMap[cate] = index; - }); // Process visual map input. - - if (!isArray(visual)) { - var visualArr_1 = []; - - if (isObject(visual)) { - each$3(visual, function (v, cate) { - var index = categoryMap[cate]; - visualArr_1[index != null ? index : CATEGORY_DEFAULT_VISUAL_INDEX] = v; - }); - } else { - // Is primary type, represents default visual. - visualArr_1[CATEGORY_DEFAULT_VISUAL_INDEX] = visual; - } - - visual = setVisualToOption(thisOption, visualArr_1); - } // Remove categories that has no visual, - // then we can mapping them to CATEGORY_DEFAULT_VISUAL_INDEX. - - - for (var i = categories.length - 1; i >= 0; i--) { - if (visual[i] == null) { - delete categoryMap[categories[i]]; - categories.pop(); - } - } - } - - function normalizeVisualRange(thisOption, isCategory) { - var visual = thisOption.visual; - var visualArr = []; - - if (isObject(visual)) { - each$3(visual, function (v) { - visualArr.push(v); - }); - } else if (visual != null) { - visualArr.push(visual); - } - - var doNotNeedPair = { - color: 1, - symbol: 1 - }; - - if (!isCategory && visualArr.length === 1 && !doNotNeedPair.hasOwnProperty(thisOption.type)) { - // Do not care visualArr.length === 0, which is illegal. - visualArr[1] = visualArr[0]; - } - - setVisualToOption(thisOption, visualArr); - } - - function makePartialColorVisualHandler(applyValue) { - return { - applyVisual: function (value, getter, setter) { - // Only used in HSL - var colorChannel = this.mapValueToVisual(value); // Must not be array value - - setter('color', applyValue(getter('color'), colorChannel)); - }, - _normalizedToVisual: createNormalizedToNumericVisual([0, 1]) - }; - } - - function doMapToArray(normalized) { - var visual = this.option.visual; - return visual[Math.round(linearMap(normalized, [0, 1], [0, visual.length - 1], true))] || {}; // TODO {}? - } - - function makeApplyVisual(visualType) { - return function (value, getter, setter) { - setter(visualType, this.mapValueToVisual(value)); - }; - } - - function doMapCategory(normalized) { - var visual = this.option.visual; - return visual[this.option.loop && normalized !== CATEGORY_DEFAULT_VISUAL_INDEX ? normalized % visual.length : normalized]; - } - - function doMapFixed() { - // visual will be convert to array. - return this.option.visual[0]; - } - /** - * Create mapped to numeric visual - */ - - - function createNormalizedToNumericVisual(sourceExtent) { - return { - linear: function (normalized) { - return linearMap(normalized, sourceExtent, this.option.visual, true); - }, - category: doMapCategory, - piecewise: function (normalized, value) { - var result = getSpecifiedVisual.call(this, value); - - if (result == null) { - result = linearMap(normalized, sourceExtent, this.option.visual, true); - } - - return result; - }, - fixed: doMapFixed - }; - } - - function getSpecifiedVisual(value) { - var thisOption = this.option; - var pieceList = thisOption.pieceList; - - if (thisOption.hasSpecialVisual) { - var pieceIndex = VisualMapping.findPieceIndex(value, pieceList); - var piece = pieceList[pieceIndex]; - - if (piece && piece.visual) { - return piece.visual[this.type]; - } - } - } - - function setVisualToOption(thisOption, visualArr) { - thisOption.visual = visualArr; - - if (thisOption.type === 'color') { - thisOption.parsedVisual = map(visualArr, function (item) { - var color$1 = parse(item); - - if (!color$1 && "development" !== 'production') { - warn("'" + item + "' is an illegal color, fallback to '#000000'", true); - } - - return color$1 || [0, 0, 0, 1]; - }); - } - - return visualArr; - } - /** - * Normalizers by mapping methods. - */ - - - var normalizers = { - linear: function (value) { - return linearMap(value, this.option.dataExtent, [0, 1], true); - }, - piecewise: function (value) { - var pieceList = this.option.pieceList; - var pieceIndex = VisualMapping.findPieceIndex(value, pieceList, true); - - if (pieceIndex != null) { - return linearMap(pieceIndex, [0, pieceList.length - 1], [0, 1], true); - } - }, - category: function (value) { - var index = this.option.categories ? this.option.categoryMap[value] : value; // ordinal value - - return index == null ? CATEGORY_DEFAULT_VISUAL_INDEX : index; - }, - fixed: noop - }; - - function littleThan(close, a, b) { - return close ? a <= b : a < b; - } - - var ITEM_STYLE_NORMAL = 'itemStyle'; - var inner$9 = makeInner(); - var treemapVisual = { - seriesType: 'treemap', - reset: function (seriesModel) { - var tree = seriesModel.getData().tree; - var root = tree.root; - - if (root.isRemoved()) { - return; - } - - travelTree(root, // Visual should calculate from tree root but not view root. - {}, seriesModel.getViewRoot().getAncestors(), seriesModel); - } - }; - - function travelTree(node, designatedVisual, viewRootAncestors, seriesModel) { - var nodeModel = node.getModel(); - var nodeLayout = node.getLayout(); - var data = node.hostTree.data; // Optimize - - if (!nodeLayout || nodeLayout.invisible || !nodeLayout.isInView) { - return; - } - - var nodeItemStyleModel = nodeModel.getModel(ITEM_STYLE_NORMAL); - var visuals = buildVisuals(nodeItemStyleModel, designatedVisual, seriesModel); - var existsStyle = data.ensureUniqueItemVisual(node.dataIndex, 'style'); // calculate border color - - var borderColor = nodeItemStyleModel.get('borderColor'); - var borderColorSaturation = nodeItemStyleModel.get('borderColorSaturation'); - var thisNodeColor; - - if (borderColorSaturation != null) { - // For performance, do not always execute 'calculateColor'. - thisNodeColor = calculateColor(visuals); - borderColor = calculateBorderColor(borderColorSaturation, thisNodeColor); - } - - existsStyle.stroke = borderColor; - var viewChildren = node.viewChildren; - - if (!viewChildren || !viewChildren.length) { - thisNodeColor = calculateColor(visuals); // Apply visual to this node. - - existsStyle.fill = thisNodeColor; - } else { - var mapping_1 = buildVisualMapping(node, nodeModel, nodeLayout, nodeItemStyleModel, visuals, viewChildren); // Designate visual to children. - - each(viewChildren, function (child, index) { - // If higher than viewRoot, only ancestors of viewRoot is needed to visit. - if (child.depth >= viewRootAncestors.length || child === viewRootAncestors[child.depth]) { - var childVisual = mapVisual(nodeModel, visuals, child, index, mapping_1, seriesModel); - travelTree(child, childVisual, viewRootAncestors, seriesModel); - } - }); - } - } - - function buildVisuals(nodeItemStyleModel, designatedVisual, seriesModel) { - var visuals = extend({}, designatedVisual); - var designatedVisualItemStyle = seriesModel.designatedVisualItemStyle; - each(['color', 'colorAlpha', 'colorSaturation'], function (visualName) { - // Priority: thisNode > thisLevel > parentNodeDesignated > seriesModel - designatedVisualItemStyle[visualName] = designatedVisual[visualName]; - var val = nodeItemStyleModel.get(visualName); - designatedVisualItemStyle[visualName] = null; - val != null && (visuals[visualName] = val); - }); - return visuals; - } - - function calculateColor(visuals) { - var color = getValueVisualDefine(visuals, 'color'); - - if (color) { - var colorAlpha = getValueVisualDefine(visuals, 'colorAlpha'); - var colorSaturation = getValueVisualDefine(visuals, 'colorSaturation'); - - if (colorSaturation) { - color = modifyHSL(color, null, null, colorSaturation); - } - - if (colorAlpha) { - color = modifyAlpha(color, colorAlpha); - } - - return color; - } - } - - function calculateBorderColor(borderColorSaturation, thisNodeColor) { - return thisNodeColor != null // Can only be string - ? modifyHSL(thisNodeColor, null, null, borderColorSaturation) : null; - } - - function getValueVisualDefine(visuals, name) { - var value = visuals[name]; - - if (value != null && value !== 'none') { - return value; - } - } - - function buildVisualMapping(node, nodeModel, nodeLayout, nodeItemStyleModel, visuals, viewChildren) { - if (!viewChildren || !viewChildren.length) { - return; - } - - var rangeVisual = getRangeVisual(nodeModel, 'color') || visuals.color != null && visuals.color !== 'none' && (getRangeVisual(nodeModel, 'colorAlpha') || getRangeVisual(nodeModel, 'colorSaturation')); - - if (!rangeVisual) { - return; - } - - var visualMin = nodeModel.get('visualMin'); - var visualMax = nodeModel.get('visualMax'); - var dataExtent = nodeLayout.dataExtent.slice(); - visualMin != null && visualMin < dataExtent[0] && (dataExtent[0] = visualMin); - visualMax != null && visualMax > dataExtent[1] && (dataExtent[1] = visualMax); - var colorMappingBy = nodeModel.get('colorMappingBy'); - var opt = { - type: rangeVisual.name, - dataExtent: dataExtent, - visual: rangeVisual.range - }; - - if (opt.type === 'color' && (colorMappingBy === 'index' || colorMappingBy === 'id')) { - opt.mappingMethod = 'category'; - opt.loop = true; // categories is ordinal, so do not set opt.categories. - } else { - opt.mappingMethod = 'linear'; - } - - var mapping = new VisualMapping(opt); - inner$9(mapping).drColorMappingBy = colorMappingBy; - return mapping; - } // Notice: If we don't have the attribute 'colorRange', but only use - // attribute 'color' to represent both concepts of 'colorRange' and 'color', - // (It means 'colorRange' when 'color' is Array, means 'color' when not array), - // this problem will be encountered: - // If a level-1 node doesn't have children, and its siblings have children, - // and colorRange is set on level-1, then the node cannot be colored. - // So we separate 'colorRange' and 'color' to different attributes. - - - function getRangeVisual(nodeModel, name) { - // 'colorRange', 'colorARange', 'colorSRange'. - // If not exists on this node, fetch from levels and series. - var range = nodeModel.get(name); - return isArray(range) && range.length ? { - name: name, - range: range - } : null; - } - - function mapVisual(nodeModel, visuals, child, index, mapping, seriesModel) { - var childVisuals = extend({}, visuals); - - if (mapping) { - // Only support color, colorAlpha, colorSaturation. - var mappingType = mapping.type; - var colorMappingBy = mappingType === 'color' && inner$9(mapping).drColorMappingBy; - var value = colorMappingBy === 'index' ? index : colorMappingBy === 'id' ? seriesModel.mapIdToIndex(child.getId()) : child.getValue(nodeModel.get('visualDimension')); - childVisuals[mappingType] = mapping.mapValueToVisual(value); - } - - return childVisuals; - } - - var mathMax$7 = Math.max; - var mathMin$7 = Math.min; - var retrieveValue = retrieve; - var each$4 = each; - var PATH_BORDER_WIDTH = ['itemStyle', 'borderWidth']; - var PATH_GAP_WIDTH = ['itemStyle', 'gapWidth']; - var PATH_UPPER_LABEL_SHOW = ['upperLabel', 'show']; - var PATH_UPPER_LABEL_HEIGHT = ['upperLabel', 'height']; - /** - * @public - */ - - var treemapLayout = { - seriesType: 'treemap', - reset: function (seriesModel, ecModel, api, payload) { - // Layout result in each node: - // {x, y, width, height, area, borderWidth} - var ecWidth = api.getWidth(); - var ecHeight = api.getHeight(); - var seriesOption = seriesModel.option; - var layoutInfo = getLayoutRect(seriesModel.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - var size = seriesOption.size || []; // Compatible with ec2. - - var containerWidth = parsePercent$1(retrieveValue(layoutInfo.width, size[0]), ecWidth); - var containerHeight = parsePercent$1(retrieveValue(layoutInfo.height, size[1]), ecHeight); // Fetch payload info. - - var payloadType = payload && payload.type; - var types = ['treemapZoomToNode', 'treemapRootToNode']; - var targetInfo = retrieveTargetInfo(payload, types, seriesModel); - var rootRect = payloadType === 'treemapRender' || payloadType === 'treemapMove' ? payload.rootRect : null; - var viewRoot = seriesModel.getViewRoot(); - var viewAbovePath = getPathToRoot(viewRoot); - - if (payloadType !== 'treemapMove') { - var rootSize = payloadType === 'treemapZoomToNode' ? estimateRootSize(seriesModel, targetInfo, viewRoot, containerWidth, containerHeight) : rootRect ? [rootRect.width, rootRect.height] : [containerWidth, containerHeight]; - var sort_1 = seriesOption.sort; - - if (sort_1 && sort_1 !== 'asc' && sort_1 !== 'desc') { - // Default to be desc order. - sort_1 = 'desc'; - } - - var options = { - squareRatio: seriesOption.squareRatio, - sort: sort_1, - leafDepth: seriesOption.leafDepth - }; // layout should be cleared because using updateView but not update. - - viewRoot.hostTree.clearLayouts(); // TODO - // optimize: if out of view clip, do not layout. - // But take care that if do not render node out of view clip, - // how to calculate start po - - var viewRootLayout_1 = { - x: 0, - y: 0, - width: rootSize[0], - height: rootSize[1], - area: rootSize[0] * rootSize[1] - }; - viewRoot.setLayout(viewRootLayout_1); - squarify(viewRoot, options, false, 0); // Supplement layout. - - viewRootLayout_1 = viewRoot.getLayout(); - each$4(viewAbovePath, function (node, index) { - var childValue = (viewAbovePath[index + 1] || viewRoot).getValue(); - node.setLayout(extend({ - dataExtent: [childValue, childValue], - borderWidth: 0, - upperHeight: 0 - }, viewRootLayout_1)); - }); - } - - var treeRoot = seriesModel.getData().tree.root; - treeRoot.setLayout(calculateRootPosition(layoutInfo, rootRect, targetInfo), true); - seriesModel.setLayoutInfo(layoutInfo); // FIXME - // 现在没有clip功能,暂时取ec高宽。 - - prunning(treeRoot, // Transform to base element coordinate system. - new BoundingRect(-layoutInfo.x, -layoutInfo.y, ecWidth, ecHeight), viewAbovePath, viewRoot, 0); - } - }; - /** - * Layout treemap with squarify algorithm. - * The original presentation of this algorithm - * was made by Mark Bruls, Kees Huizing, and Jarke J. van Wijk - * <https://graphics.ethz.ch/teaching/scivis_common/Literature/squarifiedTreeMaps.pdf>. - * The implementation of this algorithm was originally copied from "d3.js" - * <https://github.com/d3/d3/blob/9cc9a875e636a1dcf36cc1e07bdf77e1ad6e2c74/src/layout/treemap.js> - * with some modifications made for this program. - * See the license statement at the head of this file. - * - * @protected - * @param {module:echarts/data/Tree~TreeNode} node - * @param {Object} options - * @param {string} options.sort 'asc' or 'desc' - * @param {number} options.squareRatio - * @param {boolean} hideChildren - * @param {number} depth - */ - - function squarify(node, options, hideChildren, depth) { - var width; - var height; - - if (node.isRemoved()) { - return; - } - - var thisLayout = node.getLayout(); - width = thisLayout.width; - height = thisLayout.height; // Considering border and gap - - var nodeModel = node.getModel(); - var borderWidth = nodeModel.get(PATH_BORDER_WIDTH); - var halfGapWidth = nodeModel.get(PATH_GAP_WIDTH) / 2; - var upperLabelHeight = getUpperLabelHeight(nodeModel); - var upperHeight = Math.max(borderWidth, upperLabelHeight); - var layoutOffset = borderWidth - halfGapWidth; - var layoutOffsetUpper = upperHeight - halfGapWidth; - node.setLayout({ - borderWidth: borderWidth, - upperHeight: upperHeight, - upperLabelHeight: upperLabelHeight - }, true); - width = mathMax$7(width - 2 * layoutOffset, 0); - height = mathMax$7(height - layoutOffset - layoutOffsetUpper, 0); - var totalArea = width * height; - var viewChildren = initChildren(node, nodeModel, totalArea, options, hideChildren, depth); - - if (!viewChildren.length) { - return; - } - - var rect = { - x: layoutOffset, - y: layoutOffsetUpper, - width: width, - height: height - }; - var rowFixedLength = mathMin$7(width, height); - var best = Infinity; // the best row score so far - - var row = []; - row.area = 0; - - for (var i = 0, len = viewChildren.length; i < len;) { - var child = viewChildren[i]; - row.push(child); - row.area += child.getLayout().area; - var score = worst(row, rowFixedLength, options.squareRatio); // continue with this orientation - - if (score <= best) { - i++; - best = score; - } // abort, and try a different orientation - else { - row.area -= row.pop().getLayout().area; - position(row, rowFixedLength, rect, halfGapWidth, false); - rowFixedLength = mathMin$7(rect.width, rect.height); - row.length = row.area = 0; - best = Infinity; - } - } - - if (row.length) { - position(row, rowFixedLength, rect, halfGapWidth, true); - } - - if (!hideChildren) { - var childrenVisibleMin = nodeModel.get('childrenVisibleMin'); - - if (childrenVisibleMin != null && totalArea < childrenVisibleMin) { - hideChildren = true; - } - } - - for (var i = 0, len = viewChildren.length; i < len; i++) { - squarify(viewChildren[i], options, hideChildren, depth + 1); - } - } - /** - * Set area to each child, and calculate data extent for visual coding. - */ - - - function initChildren(node, nodeModel, totalArea, options, hideChildren, depth) { - var viewChildren = node.children || []; - var orderBy = options.sort; - orderBy !== 'asc' && orderBy !== 'desc' && (orderBy = null); - var overLeafDepth = options.leafDepth != null && options.leafDepth <= depth; // leafDepth has higher priority. - - if (hideChildren && !overLeafDepth) { - return node.viewChildren = []; - } // Sort children, order by desc. - - - viewChildren = filter(viewChildren, function (child) { - return !child.isRemoved(); - }); - sort$1(viewChildren, orderBy); - var info = statistic(nodeModel, viewChildren, orderBy); - - if (info.sum === 0) { - return node.viewChildren = []; - } - - info.sum = filterByThreshold(nodeModel, totalArea, info.sum, orderBy, viewChildren); - - if (info.sum === 0) { - return node.viewChildren = []; - } // Set area to each child. - - - for (var i = 0, len = viewChildren.length; i < len; i++) { - var area = viewChildren[i].getValue() / info.sum * totalArea; // Do not use setLayout({...}, true), because it is needed to clear last layout. - - viewChildren[i].setLayout({ - area: area - }); - } - - if (overLeafDepth) { - viewChildren.length && node.setLayout({ - isLeafRoot: true - }, true); - viewChildren.length = 0; - } - - node.viewChildren = viewChildren; - node.setLayout({ - dataExtent: info.dataExtent - }, true); - return viewChildren; - } - /** - * Consider 'visibleMin'. Modify viewChildren and get new sum. - */ - - - function filterByThreshold(nodeModel, totalArea, sum, orderBy, orderedChildren) { - // visibleMin is not supported yet when no option.sort. - if (!orderBy) { - return sum; - } - - var visibleMin = nodeModel.get('visibleMin'); - var len = orderedChildren.length; - var deletePoint = len; // Always travel from little value to big value. - - for (var i = len - 1; i >= 0; i--) { - var value = orderedChildren[orderBy === 'asc' ? len - i - 1 : i].getValue(); - - if (value / sum * totalArea < visibleMin) { - deletePoint = i; - sum -= value; - } - } - - orderBy === 'asc' ? orderedChildren.splice(0, len - deletePoint) : orderedChildren.splice(deletePoint, len - deletePoint); - return sum; - } - /** - * Sort - */ - - - function sort$1(viewChildren, orderBy) { - if (orderBy) { - viewChildren.sort(function (a, b) { - var diff = orderBy === 'asc' ? a.getValue() - b.getValue() : b.getValue() - a.getValue(); - return diff === 0 ? orderBy === 'asc' ? a.dataIndex - b.dataIndex : b.dataIndex - a.dataIndex : diff; - }); - } - - return viewChildren; - } - /** - * Statistic - */ - - - function statistic(nodeModel, children, orderBy) { - // Calculate sum. - var sum = 0; - - for (var i = 0, len = children.length; i < len; i++) { - sum += children[i].getValue(); - } // Statistic data extent for latter visual coding. - // Notice: data extent should be calculate based on raw children - // but not filtered view children, otherwise visual mapping will not - // be stable when zoom (where children is filtered by visibleMin). - - - var dimension = nodeModel.get('visualDimension'); - var dataExtent; // The same as area dimension. - - if (!children || !children.length) { - dataExtent = [NaN, NaN]; - } else if (dimension === 'value' && orderBy) { - dataExtent = [children[children.length - 1].getValue(), children[0].getValue()]; - orderBy === 'asc' && dataExtent.reverse(); - } // Other dimension. - else { - dataExtent = [Infinity, -Infinity]; - each$4(children, function (child) { - var value = child.getValue(dimension); - value < dataExtent[0] && (dataExtent[0] = value); - value > dataExtent[1] && (dataExtent[1] = value); - }); - } - - return { - sum: sum, - dataExtent: dataExtent - }; - } - /** - * Computes the score for the specified row, - * as the worst aspect ratio. - */ - - - function worst(row, rowFixedLength, ratio) { - var areaMax = 0; - var areaMin = Infinity; - - for (var i = 0, area = void 0, len = row.length; i < len; i++) { - area = row[i].getLayout().area; - - if (area) { - area < areaMin && (areaMin = area); - area > areaMax && (areaMax = area); - } - } - - var squareArea = row.area * row.area; - var f = rowFixedLength * rowFixedLength * ratio; - return squareArea ? mathMax$7(f * areaMax / squareArea, squareArea / (f * areaMin)) : Infinity; - } - /** - * Positions the specified row of nodes. Modifies `rect`. - */ - - - function position(row, rowFixedLength, rect, halfGapWidth, flush) { - // When rowFixedLength === rect.width, - // it is horizontal subdivision, - // rowFixedLength is the width of the subdivision, - // rowOtherLength is the height of the subdivision, - // and nodes will be positioned from left to right. - // wh[idx0WhenH] means: when horizontal, - // wh[idx0WhenH] => wh[0] => 'width'. - // xy[idx1WhenH] => xy[1] => 'y'. - var idx0WhenH = rowFixedLength === rect.width ? 0 : 1; - var idx1WhenH = 1 - idx0WhenH; - var xy = ['x', 'y']; - var wh = ['width', 'height']; - var last = rect[xy[idx0WhenH]]; - var rowOtherLength = rowFixedLength ? row.area / rowFixedLength : 0; - - if (flush || rowOtherLength > rect[wh[idx1WhenH]]) { - rowOtherLength = rect[wh[idx1WhenH]]; // over+underflow - } - - for (var i = 0, rowLen = row.length; i < rowLen; i++) { - var node = row[i]; - var nodeLayout = {}; - var step = rowOtherLength ? node.getLayout().area / rowOtherLength : 0; - var wh1 = nodeLayout[wh[idx1WhenH]] = mathMax$7(rowOtherLength - 2 * halfGapWidth, 0); // We use Math.max/min to avoid negative width/height when considering gap width. - - var remain = rect[xy[idx0WhenH]] + rect[wh[idx0WhenH]] - last; - var modWH = i === rowLen - 1 || remain < step ? remain : step; - var wh0 = nodeLayout[wh[idx0WhenH]] = mathMax$7(modWH - 2 * halfGapWidth, 0); - nodeLayout[xy[idx1WhenH]] = rect[xy[idx1WhenH]] + mathMin$7(halfGapWidth, wh1 / 2); - nodeLayout[xy[idx0WhenH]] = last + mathMin$7(halfGapWidth, wh0 / 2); - last += modWH; - node.setLayout(nodeLayout, true); - } - - rect[xy[idx1WhenH]] += rowOtherLength; - rect[wh[idx1WhenH]] -= rowOtherLength; - } // Return [containerWidth, containerHeight] as default. - - - function estimateRootSize(seriesModel, targetInfo, viewRoot, containerWidth, containerHeight) { - // If targetInfo.node exists, we zoom to the node, - // so estimate whole width and height by target node. - var currNode = (targetInfo || {}).node; - var defaultSize = [containerWidth, containerHeight]; - - if (!currNode || currNode === viewRoot) { - return defaultSize; - } - - var parent; - var viewArea = containerWidth * containerHeight; - var area = viewArea * seriesModel.option.zoomToNodeRatio; - - while (parent = currNode.parentNode) { - // jshint ignore:line - var sum = 0; - var siblings = parent.children; - - for (var i = 0, len = siblings.length; i < len; i++) { - sum += siblings[i].getValue(); - } - - var currNodeValue = currNode.getValue(); - - if (currNodeValue === 0) { - return defaultSize; - } - - area *= sum / currNodeValue; // Considering border, suppose aspect ratio is 1. - - var parentModel = parent.getModel(); - var borderWidth = parentModel.get(PATH_BORDER_WIDTH); - var upperHeight = Math.max(borderWidth, getUpperLabelHeight(parentModel)); - area += 4 * borderWidth * borderWidth + (3 * borderWidth + upperHeight) * Math.pow(area, 0.5); - area > MAX_SAFE_INTEGER && (area = MAX_SAFE_INTEGER); - currNode = parent; - } - - area < viewArea && (area = viewArea); - var scale = Math.pow(area / viewArea, 0.5); - return [containerWidth * scale, containerHeight * scale]; - } // Root position based on coord of containerGroup - - - function calculateRootPosition(layoutInfo, rootRect, targetInfo) { - if (rootRect) { - return { - x: rootRect.x, - y: rootRect.y - }; - } - - var defaultPosition = { - x: 0, - y: 0 - }; - - if (!targetInfo) { - return defaultPosition; - } // If targetInfo is fetched by 'retrieveTargetInfo', - // old tree and new tree are the same tree, - // so the node still exists and we can visit it. - - - var targetNode = targetInfo.node; - var layout = targetNode.getLayout(); - - if (!layout) { - return defaultPosition; - } // Transform coord from local to container. - - - var targetCenter = [layout.width / 2, layout.height / 2]; - var node = targetNode; - - while (node) { - var nodeLayout = node.getLayout(); - targetCenter[0] += nodeLayout.x; - targetCenter[1] += nodeLayout.y; - node = node.parentNode; - } - - return { - x: layoutInfo.width / 2 - targetCenter[0], - y: layoutInfo.height / 2 - targetCenter[1] - }; - } // Mark nodes visible for prunning when visual coding and rendering. - // Prunning depends on layout and root position, so we have to do it after layout. - - - function prunning(node, clipRect, viewAbovePath, viewRoot, depth) { - var nodeLayout = node.getLayout(); - var nodeInViewAbovePath = viewAbovePath[depth]; - var isAboveViewRoot = nodeInViewAbovePath && nodeInViewAbovePath === node; - - if (nodeInViewAbovePath && !isAboveViewRoot || depth === viewAbovePath.length && node !== viewRoot) { - return; - } - - node.setLayout({ - // isInView means: viewRoot sub tree + viewAbovePath - isInView: true, - // invisible only means: outside view clip so that the node can not - // see but still layout for animation preparation but not render. - invisible: !isAboveViewRoot && !clipRect.intersect(nodeLayout), - isAboveViewRoot: isAboveViewRoot - }, true); // Transform to child coordinate. - - var childClipRect = new BoundingRect(clipRect.x - nodeLayout.x, clipRect.y - nodeLayout.y, clipRect.width, clipRect.height); - each$4(node.viewChildren || [], function (child) { - prunning(child, childClipRect, viewAbovePath, viewRoot, depth + 1); - }); - } - - function getUpperLabelHeight(model) { - return model.get(PATH_UPPER_LABEL_SHOW) ? model.get(PATH_UPPER_LABEL_HEIGHT) : 0; - } - - function install$c(registers) { - registers.registerSeriesModel(TreemapSeriesModel); - registers.registerChartView(TreemapView); - registers.registerVisual(treemapVisual); - registers.registerLayout(treemapLayout); - installTreemapAction(registers); - } - - function categoryFilter(ecModel) { - var legendModels = ecModel.findComponents({ - mainType: 'legend' - }); - - if (!legendModels || !legendModels.length) { - return; - } - - ecModel.eachSeriesByType('graph', function (graphSeries) { - var categoriesData = graphSeries.getCategoriesData(); - var graph = graphSeries.getGraph(); - var data = graph.data; - var categoryNames = categoriesData.mapArray(categoriesData.getName); - data.filterSelf(function (idx) { - var model = data.getItemModel(idx); - var category = model.getShallow('category'); - - if (category != null) { - if (isNumber(category)) { - category = categoryNames[category]; - } // If in any legend component the status is not selected. - - - for (var i = 0; i < legendModels.length; i++) { - if (!legendModels[i].isSelected(category)) { - return false; - } - } - } - - return true; - }); - }); - } - - function categoryVisual(ecModel) { - var paletteScope = {}; - ecModel.eachSeriesByType('graph', function (seriesModel) { - var categoriesData = seriesModel.getCategoriesData(); - var data = seriesModel.getData(); - var categoryNameIdxMap = {}; - categoriesData.each(function (idx) { - var name = categoriesData.getName(idx); // Add prefix to avoid conflict with Object.prototype. - - categoryNameIdxMap['ec-' + name] = idx; - var itemModel = categoriesData.getItemModel(idx); - var style = itemModel.getModel('itemStyle').getItemStyle(); - - if (!style.fill) { - // Get color from palette. - style.fill = seriesModel.getColorFromPalette(name, paletteScope); - } - - categoriesData.setItemVisual(idx, 'style', style); - var symbolVisualList = ['symbol', 'symbolSize', 'symbolKeepAspect']; - - for (var i = 0; i < symbolVisualList.length; i++) { - var symbolVisual = itemModel.getShallow(symbolVisualList[i], true); - - if (symbolVisual != null) { - categoriesData.setItemVisual(idx, symbolVisualList[i], symbolVisual); - } - } - }); // Assign category color to visual - - if (categoriesData.count()) { - data.each(function (idx) { - var model = data.getItemModel(idx); - var categoryIdx = model.getShallow('category'); - - if (categoryIdx != null) { - if (isString(categoryIdx)) { - categoryIdx = categoryNameIdxMap['ec-' + categoryIdx]; - } - - var categoryStyle = categoriesData.getItemVisual(categoryIdx, 'style'); - var style = data.ensureUniqueItemVisual(idx, 'style'); - extend(style, categoryStyle); - var visualList = ['symbol', 'symbolSize', 'symbolKeepAspect']; - - for (var i = 0; i < visualList.length; i++) { - data.setItemVisual(idx, visualList[i], categoriesData.getItemVisual(categoryIdx, visualList[i])); - } - } - }); - } - }); - } - - function normalize$2(a) { - if (!(a instanceof Array)) { - a = [a, a]; - } - - return a; - } - - function graphEdgeVisual(ecModel) { - ecModel.eachSeriesByType('graph', function (seriesModel) { - var graph = seriesModel.getGraph(); - var edgeData = seriesModel.getEdgeData(); - var symbolType = normalize$2(seriesModel.get('edgeSymbol')); - var symbolSize = normalize$2(seriesModel.get('edgeSymbolSize')); // const colorQuery = ['lineStyle', 'color'] as const; - // const opacityQuery = ['lineStyle', 'opacity'] as const; - - edgeData.setVisual('fromSymbol', symbolType && symbolType[0]); - edgeData.setVisual('toSymbol', symbolType && symbolType[1]); - edgeData.setVisual('fromSymbolSize', symbolSize && symbolSize[0]); - edgeData.setVisual('toSymbolSize', symbolSize && symbolSize[1]); - edgeData.setVisual('style', seriesModel.getModel('lineStyle').getLineStyle()); - edgeData.each(function (idx) { - var itemModel = edgeData.getItemModel(idx); - var edge = graph.getEdgeByIndex(idx); - var symbolType = normalize$2(itemModel.getShallow('symbol', true)); - var symbolSize = normalize$2(itemModel.getShallow('symbolSize', true)); // Edge visual must after node visual - - var style = itemModel.getModel('lineStyle').getLineStyle(); - var existsStyle = edgeData.ensureUniqueItemVisual(idx, 'style'); - extend(existsStyle, style); - - switch (existsStyle.stroke) { - case 'source': - { - var nodeStyle = edge.node1.getVisual('style'); - existsStyle.stroke = nodeStyle && nodeStyle.fill; - break; - } - - case 'target': - { - var nodeStyle = edge.node2.getVisual('style'); - existsStyle.stroke = nodeStyle && nodeStyle.fill; - break; - } - } - - symbolType[0] && edge.setVisual('fromSymbol', symbolType[0]); - symbolType[1] && edge.setVisual('toSymbol', symbolType[1]); - symbolSize[0] && edge.setVisual('fromSymbolSize', symbolSize[0]); - symbolSize[1] && edge.setVisual('toSymbolSize', symbolSize[1]); - }); - }); - } - - var KEY_DELIMITER = '-->'; - /** - * params handler - * @param {module:echarts/model/SeriesModel} seriesModel - * @returns {*} - */ - - var getAutoCurvenessParams = function (seriesModel) { - return seriesModel.get('autoCurveness') || null; - }; - /** - * Generate a list of edge curvatures, 20 is the default - * @param {module:echarts/model/SeriesModel} seriesModel - * @param {number} appendLength - * @return 20 => [0, -0.2, 0.2, -0.4, 0.4, -0.6, 0.6, -0.8, 0.8, -1, 1, -1.2, 1.2, -1.4, 1.4, -1.6, 1.6, -1.8, 1.8, -2] - */ - - - var createCurveness = function (seriesModel, appendLength) { - var autoCurvenessParmas = getAutoCurvenessParams(seriesModel); - var length = 20; - var curvenessList = []; // handler the function set - - if (isNumber(autoCurvenessParmas)) { - length = autoCurvenessParmas; - } else if (isArray(autoCurvenessParmas)) { - seriesModel.__curvenessList = autoCurvenessParmas; - return; - } // append length - - - if (appendLength > length) { - length = appendLength; - } // make sure the length is even - - - var len = length % 2 ? length + 2 : length + 3; - curvenessList = []; - - for (var i = 0; i < len; i++) { - curvenessList.push((i % 2 ? i + 1 : i) / 10 * (i % 2 ? -1 : 1)); - } - - seriesModel.__curvenessList = curvenessList; - }; - /** - * Create different cache key data in the positive and negative directions, in order to set the curvature later - * @param {number|string|module:echarts/data/Graph.Node} n1 - * @param {number|string|module:echarts/data/Graph.Node} n2 - * @param {module:echarts/model/SeriesModel} seriesModel - * @returns {string} key - */ - - - var getKeyOfEdges = function (n1, n2, seriesModel) { - var source = [n1.id, n1.dataIndex].join('.'); - var target = [n2.id, n2.dataIndex].join('.'); - return [seriesModel.uid, source, target].join(KEY_DELIMITER); - }; - /** - * get opposite key - * @param {string} key - * @returns {string} - */ - - - var getOppositeKey = function (key) { - var keys = key.split(KEY_DELIMITER); - return [keys[0], keys[2], keys[1]].join(KEY_DELIMITER); - }; - /** - * get edgeMap with key - * @param edge - * @param {module:echarts/model/SeriesModel} seriesModel - */ - - - var getEdgeFromMap = function (edge, seriesModel) { - var key = getKeyOfEdges(edge.node1, edge.node2, seriesModel); - return seriesModel.__edgeMap[key]; - }; - /** - * calculate all cases total length - * @param edge - * @param seriesModel - * @returns {number} - */ - - - var getTotalLengthBetweenNodes = function (edge, seriesModel) { - var len = getEdgeMapLengthWithKey(getKeyOfEdges(edge.node1, edge.node2, seriesModel), seriesModel); - var lenV = getEdgeMapLengthWithKey(getKeyOfEdges(edge.node2, edge.node1, seriesModel), seriesModel); - return len + lenV; - }; - /** - * - * @param key - */ - - - var getEdgeMapLengthWithKey = function (key, seriesModel) { - var edgeMap = seriesModel.__edgeMap; - return edgeMap[key] ? edgeMap[key].length : 0; - }; - /** - * Count the number of edges between the same two points, used to obtain the curvature table and the parity of the edge - * @see /graph/GraphSeries.js@getInitialData - * @param {module:echarts/model/SeriesModel} seriesModel - */ - - - function initCurvenessList(seriesModel) { - if (!getAutoCurvenessParams(seriesModel)) { - return; - } - - seriesModel.__curvenessList = []; - seriesModel.__edgeMap = {}; // calc the array of curveness List - - createCurveness(seriesModel); - } - /** - * set edgeMap with key - * @param {number|string|module:echarts/data/Graph.Node} n1 - * @param {number|string|module:echarts/data/Graph.Node} n2 - * @param {module:echarts/model/SeriesModel} seriesModel - * @param {number} index - */ - - function createEdgeMapForCurveness(n1, n2, seriesModel, index) { - if (!getAutoCurvenessParams(seriesModel)) { - return; - } - - var key = getKeyOfEdges(n1, n2, seriesModel); - var edgeMap = seriesModel.__edgeMap; - var oppositeEdges = edgeMap[getOppositeKey(key)]; // set direction - - if (edgeMap[key] && !oppositeEdges) { - edgeMap[key].isForward = true; - } else if (oppositeEdges && edgeMap[key]) { - oppositeEdges.isForward = true; - edgeMap[key].isForward = false; - } - - edgeMap[key] = edgeMap[key] || []; - edgeMap[key].push(index); - } - /** - * get curvature for edge - * @param edge - * @param {module:echarts/model/SeriesModel} seriesModel - * @param index - */ - - function getCurvenessForEdge(edge, seriesModel, index, needReverse) { - var autoCurvenessParams = getAutoCurvenessParams(seriesModel); - var isArrayParam = isArray(autoCurvenessParams); - - if (!autoCurvenessParams) { - return null; - } - - var edgeArray = getEdgeFromMap(edge, seriesModel); - - if (!edgeArray) { - return null; - } - - var edgeIndex = -1; - - for (var i = 0; i < edgeArray.length; i++) { - if (edgeArray[i] === index) { - edgeIndex = i; - break; - } - } // if totalLen is Longer createCurveness - - - var totalLen = getTotalLengthBetweenNodes(edge, seriesModel); - createCurveness(seriesModel, totalLen); - edge.lineStyle = edge.lineStyle || {}; // if is opposite edge, must set curvenss to opposite number - - var curKey = getKeyOfEdges(edge.node1, edge.node2, seriesModel); - var curvenessList = seriesModel.__curvenessList; // if pass array no need parity - - var parityCorrection = isArrayParam ? 0 : totalLen % 2 ? 0 : 1; - - if (!edgeArray.isForward) { - // the opposite edge show outside - var oppositeKey = getOppositeKey(curKey); - var len = getEdgeMapLengthWithKey(oppositeKey, seriesModel); - var resValue = curvenessList[edgeIndex + len + parityCorrection]; // isNeedReverse, simple, force type need reverse the curveness in the junction of the forword and the opposite - - if (needReverse) { - // set as array may make the parity handle with the len of opposite - if (isArrayParam) { - if (autoCurvenessParams && autoCurvenessParams[0] === 0) { - return (len + parityCorrection) % 2 ? resValue : -resValue; - } else { - return ((len % 2 ? 0 : 1) + parityCorrection) % 2 ? resValue : -resValue; - } - } else { - return (len + parityCorrection) % 2 ? resValue : -resValue; - } - } else { - return curvenessList[edgeIndex + len + parityCorrection]; - } - } else { - return curvenessList[parityCorrection + edgeIndex]; - } - } - - function simpleLayout(seriesModel) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && coordSys.type !== 'view') { - return; - } - - var graph = seriesModel.getGraph(); - graph.eachNode(function (node) { - var model = node.getModel(); - node.setLayout([+model.get('x'), +model.get('y')]); - }); - simpleLayoutEdge(graph, seriesModel); - } - function simpleLayoutEdge(graph, seriesModel) { - graph.eachEdge(function (edge, index) { - var curveness = retrieve3(edge.getModel().get(['lineStyle', 'curveness']), -getCurvenessForEdge(edge, seriesModel, index, true), 0); - var p1 = clone$1(edge.node1.getLayout()); - var p2 = clone$1(edge.node2.getLayout()); - var points = [p1, p2]; - - if (+curveness) { - points.push([(p1[0] + p2[0]) / 2 - (p1[1] - p2[1]) * curveness, (p1[1] + p2[1]) / 2 - (p2[0] - p1[0]) * curveness]); - } - - edge.setLayout(points); - }); - } - - function graphSimpleLayout(ecModel, api) { - ecModel.eachSeriesByType('graph', function (seriesModel) { - var layout = seriesModel.get('layout'); - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && coordSys.type !== 'view') { - var data_1 = seriesModel.getData(); - var dimensions_1 = []; - each(coordSys.dimensions, function (coordDim) { - dimensions_1 = dimensions_1.concat(data_1.mapDimensionsAll(coordDim)); - }); - - for (var dataIndex = 0; dataIndex < data_1.count(); dataIndex++) { - var value = []; - var hasValue = false; - - for (var i = 0; i < dimensions_1.length; i++) { - var val = data_1.get(dimensions_1[i], dataIndex); - - if (!isNaN(val)) { - hasValue = true; - } - - value.push(val); - } - - if (hasValue) { - data_1.setItemLayout(dataIndex, coordSys.dataToPoint(value)); - } else { - // Also {Array.<number>}, not undefined to avoid if...else... statement - data_1.setItemLayout(dataIndex, [NaN, NaN]); - } - } - - simpleLayoutEdge(data_1.graph, seriesModel); - } else if (!layout || layout === 'none') { - simpleLayout(seriesModel); - } - }); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function getNodeGlobalScale(seriesModel) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys.type !== 'view') { - return 1; - } - - var nodeScaleRatio = seriesModel.option.nodeScaleRatio; - var groupZoom = coordSys.scaleX; // Scale node when zoom changes - - var roamZoom = coordSys.getZoom(); - var nodeScale = (roamZoom - 1) * nodeScaleRatio + 1; - return nodeScale / groupZoom; - } - function getSymbolSize(node) { - var symbolSize = node.getVisual('symbolSize'); - - if (symbolSize instanceof Array) { - symbolSize = (symbolSize[0] + symbolSize[1]) / 2; - } - - return +symbolSize; - } - - var PI$6 = Math.PI; - var _symbolRadiansHalf = []; - /** - * `basedOn` can be: - * 'value': - * This layout is not accurate and have same bad case. For example, - * if the min value is very smaller than the max value, the nodes - * with the min value probably overlap even though there is enough - * space to layout them. So we only use this approach in the as the - * init layout of the force layout. - * FIXME - * Probably we do not need this method any more but use - * `basedOn: 'symbolSize'` in force layout if - * delay its init operations to GraphView. - * 'symbolSize': - * This approach work only if all of the symbol size calculated. - * That is, the progressive rendering is not applied to graph. - * FIXME - * If progressive rendering is applied to graph some day, - * probably we have to use `basedOn: 'value'`. - */ - - function circularLayout(seriesModel, basedOn, draggingNode, pointer) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && coordSys.type !== 'view') { - return; - } - - var rect = coordSys.getBoundingRect(); - var nodeData = seriesModel.getData(); - var graph = nodeData.graph; - var cx = rect.width / 2 + rect.x; - var cy = rect.height / 2 + rect.y; - var r = Math.min(rect.width, rect.height) / 2; - var count = nodeData.count(); - nodeData.setLayout({ - cx: cx, - cy: cy - }); - - if (!count) { - return; - } - - if (draggingNode) { - var _a = coordSys.pointToData(pointer), - tempX = _a[0], - tempY = _a[1]; - - var v = [tempX - cx, tempY - cy]; - normalize(v, v); - scale(v, v, r); - draggingNode.setLayout([cx + v[0], cy + v[1]], true); - var circularRotateLabel = seriesModel.get(['circular', 'rotateLabel']); - rotateNodeLabel(draggingNode, circularRotateLabel, cx, cy); - } - - _layoutNodesBasedOn[basedOn](seriesModel, graph, nodeData, r, cx, cy, count); - - graph.eachEdge(function (edge, index) { - var curveness = retrieve3(edge.getModel().get(['lineStyle', 'curveness']), getCurvenessForEdge(edge, seriesModel, index), 0); - var p1 = clone$1(edge.node1.getLayout()); - var p2 = clone$1(edge.node2.getLayout()); - var cp1; - var x12 = (p1[0] + p2[0]) / 2; - var y12 = (p1[1] + p2[1]) / 2; - - if (+curveness) { - curveness *= 3; - cp1 = [cx * curveness + x12 * (1 - curveness), cy * curveness + y12 * (1 - curveness)]; - } - - edge.setLayout([p1, p2, cp1]); - }); - } - var _layoutNodesBasedOn = { - value: function (seriesModel, graph, nodeData, r, cx, cy, count) { - var angle = 0; - var sum = nodeData.getSum('value'); - var unitAngle = Math.PI * 2 / (sum || count); - graph.eachNode(function (node) { - var value = node.getValue('value'); - var radianHalf = unitAngle * (sum ? value : 1) / 2; - angle += radianHalf; - node.setLayout([r * Math.cos(angle) + cx, r * Math.sin(angle) + cy]); - angle += radianHalf; - }); - }, - symbolSize: function (seriesModel, graph, nodeData, r, cx, cy, count) { - var sumRadian = 0; - _symbolRadiansHalf.length = count; - var nodeScale = getNodeGlobalScale(seriesModel); - graph.eachNode(function (node) { - var symbolSize = getSymbolSize(node); // Normally this case will not happen, but we still add - // some the defensive code (2px is an arbitrary value). - - isNaN(symbolSize) && (symbolSize = 2); - symbolSize < 0 && (symbolSize = 0); - symbolSize *= nodeScale; - var symbolRadianHalf = Math.asin(symbolSize / 2 / r); // when `symbolSize / 2` is bigger than `r`. - - isNaN(symbolRadianHalf) && (symbolRadianHalf = PI$6 / 2); - _symbolRadiansHalf[node.dataIndex] = symbolRadianHalf; - sumRadian += symbolRadianHalf * 2; - }); - var halfRemainRadian = (2 * PI$6 - sumRadian) / count / 2; - var angle = 0; - graph.eachNode(function (node) { - var radianHalf = halfRemainRadian + _symbolRadiansHalf[node.dataIndex]; - angle += radianHalf; // init circular layout for - // 1. layout undefined node - // 2. not fixed node - - (!node.getLayout() || !node.getLayout().fixed) && node.setLayout([r * Math.cos(angle) + cx, r * Math.sin(angle) + cy]); - angle += radianHalf; - }); - } - }; - function rotateNodeLabel(node, circularRotateLabel, cx, cy) { - var el = node.getGraphicEl(); // need to check if el exists. '-' value may not create node element. - - if (!el) { - return; - } - - var nodeModel = node.getModel(); - var labelRotate = nodeModel.get(['label', 'rotate']) || 0; - var symbolPath = el.getSymbolPath(); - - if (circularRotateLabel) { - var pos = node.getLayout(); - var rad = Math.atan2(pos[1] - cy, pos[0] - cx); - - if (rad < 0) { - rad = Math.PI * 2 + rad; - } - - var isLeft = pos[0] < cx; - - if (isLeft) { - rad = rad - Math.PI; - } - - var textPosition = isLeft ? 'left' : 'right'; - symbolPath.setTextConfig({ - rotation: -rad, - position: textPosition, - origin: 'center' - }); - var emphasisState = symbolPath.ensureState('emphasis'); - extend(emphasisState.textConfig || (emphasisState.textConfig = {}), { - position: textPosition - }); - } else { - symbolPath.setTextConfig({ - rotation: labelRotate *= Math.PI / 180 - }); - } - } - - function graphCircularLayout(ecModel) { - ecModel.eachSeriesByType('graph', function (seriesModel) { - if (seriesModel.get('layout') === 'circular') { - circularLayout(seriesModel, 'symbolSize'); - } - }); - } - - var scaleAndAdd$1 = scaleAndAdd; // function adjacentNode(n, e) { - // return e.n1 === n ? e.n2 : e.n1; - // } - - function forceLayout(inNodes, inEdges, opts) { - var nodes = inNodes; - var edges = inEdges; - var rect = opts.rect; - var width = rect.width; - var height = rect.height; - var center = [rect.x + width / 2, rect.y + height / 2]; // let scale = opts.scale || 1; - - var gravity = opts.gravity == null ? 0.1 : opts.gravity; // for (let i = 0; i < edges.length; i++) { - // let e = edges[i]; - // let n1 = e.n1; - // let n2 = e.n2; - // n1.edges = n1.edges || []; - // n2.edges = n2.edges || []; - // n1.edges.push(e); - // n2.edges.push(e); - // } - // Init position - - for (var i = 0; i < nodes.length; i++) { - var n = nodes[i]; - - if (!n.p) { - n.p = create(width * (Math.random() - 0.5) + center[0], height * (Math.random() - 0.5) + center[1]); - } - - n.pp = clone$1(n.p); - n.edges = null; - } // Formula in 'Graph Drawing by Force-directed Placement' - // let k = scale * Math.sqrt(width * height / nodes.length); - // let k2 = k * k; - - - var initialFriction = opts.friction == null ? 0.6 : opts.friction; - var friction = initialFriction; - var beforeStepCallback; - var afterStepCallback; - return { - warmUp: function () { - friction = initialFriction * 0.8; - }, - setFixed: function (idx) { - nodes[idx].fixed = true; - }, - setUnfixed: function (idx) { - nodes[idx].fixed = false; - }, - - /** - * Before step hook - */ - beforeStep: function (cb) { - beforeStepCallback = cb; - }, - - /** - * After step hook - */ - afterStep: function (cb) { - afterStepCallback = cb; - }, - - /** - * Some formulas were originally copied from "d3.js" - * https://github.com/d3/d3/blob/b516d77fb8566b576088e73410437494717ada26/src/layout/force.js - * with some modifications made for this project. - * See the license statement at the head of this file. - */ - step: function (cb) { - beforeStepCallback && beforeStepCallback(nodes, edges); - var v12 = []; - var nLen = nodes.length; - - for (var i = 0; i < edges.length; i++) { - var e = edges[i]; - - if (e.ignoreForceLayout) { - continue; - } - - var n1 = e.n1; - var n2 = e.n2; - sub(v12, n2.p, n1.p); - var d = len(v12) - e.d; - var w = n2.w / (n1.w + n2.w); - - if (isNaN(w)) { - w = 0; - } - - normalize(v12, v12); - !n1.fixed && scaleAndAdd$1(n1.p, n1.p, v12, w * d * friction); - !n2.fixed && scaleAndAdd$1(n2.p, n2.p, v12, -(1 - w) * d * friction); - } // Gravity - - - for (var i = 0; i < nLen; i++) { - var n = nodes[i]; - - if (!n.fixed) { - sub(v12, center, n.p); // let d = vec2.len(v12); - // vec2.scale(v12, v12, 1 / d); - // let gravityFactor = gravity; - - scaleAndAdd$1(n.p, n.p, v12, gravity * friction); - } - } // Repulsive - // PENDING - - - for (var i = 0; i < nLen; i++) { - var n1 = nodes[i]; - - for (var j = i + 1; j < nLen; j++) { - var n2 = nodes[j]; - sub(v12, n2.p, n1.p); - var d = len(v12); - - if (d === 0) { - // Random repulse - set(v12, Math.random() - 0.5, Math.random() - 0.5); - d = 1; - } - - var repFact = (n1.rep + n2.rep) / d / d; - !n1.fixed && scaleAndAdd$1(n1.pp, n1.pp, v12, repFact); - !n2.fixed && scaleAndAdd$1(n2.pp, n2.pp, v12, -repFact); - } - } - - var v = []; - - for (var i = 0; i < nLen; i++) { - var n = nodes[i]; - - if (!n.fixed) { - sub(v, n.p, n.pp); - scaleAndAdd$1(n.p, n.p, v, friction); - copy(n.pp, n.p); - } - } - - friction = friction * 0.992; - var finished = friction < 0.01; - afterStepCallback && afterStepCallback(nodes, edges, finished); - cb && cb(finished); - } - }; - } - - function graphForceLayout(ecModel) { - ecModel.eachSeriesByType('graph', function (graphSeries) { - var coordSys = graphSeries.coordinateSystem; - - if (coordSys && coordSys.type !== 'view') { - return; - } - - if (graphSeries.get('layout') === 'force') { - var preservedPoints_1 = graphSeries.preservedPoints || {}; - var graph_1 = graphSeries.getGraph(); - var nodeData_1 = graph_1.data; - var edgeData = graph_1.edgeData; - var forceModel = graphSeries.getModel('force'); - var initLayout = forceModel.get('initLayout'); - - if (graphSeries.preservedPoints) { - nodeData_1.each(function (idx) { - var id = nodeData_1.getId(idx); - nodeData_1.setItemLayout(idx, preservedPoints_1[id] || [NaN, NaN]); - }); - } else if (!initLayout || initLayout === 'none') { - simpleLayout(graphSeries); - } else if (initLayout === 'circular') { - circularLayout(graphSeries, 'value'); - } - - var nodeDataExtent_1 = nodeData_1.getDataExtent('value'); - var edgeDataExtent_1 = edgeData.getDataExtent('value'); // let edgeDataExtent = edgeData.getDataExtent('value'); - - var repulsion = forceModel.get('repulsion'); - var edgeLength = forceModel.get('edgeLength'); - var repulsionArr_1 = isArray(repulsion) ? repulsion : [repulsion, repulsion]; - var edgeLengthArr_1 = isArray(edgeLength) ? edgeLength : [edgeLength, edgeLength]; // Larger value has smaller length - - edgeLengthArr_1 = [edgeLengthArr_1[1], edgeLengthArr_1[0]]; - var nodes_1 = nodeData_1.mapArray('value', function (value, idx) { - var point = nodeData_1.getItemLayout(idx); - var rep = linearMap(value, nodeDataExtent_1, repulsionArr_1); - - if (isNaN(rep)) { - rep = (repulsionArr_1[0] + repulsionArr_1[1]) / 2; - } - - return { - w: rep, - rep: rep, - fixed: nodeData_1.getItemModel(idx).get('fixed'), - p: !point || isNaN(point[0]) || isNaN(point[1]) ? null : point - }; - }); - var edges = edgeData.mapArray('value', function (value, idx) { - var edge = graph_1.getEdgeByIndex(idx); - var d = linearMap(value, edgeDataExtent_1, edgeLengthArr_1); - - if (isNaN(d)) { - d = (edgeLengthArr_1[0] + edgeLengthArr_1[1]) / 2; - } - - var edgeModel = edge.getModel(); - var curveness = retrieve3(edge.getModel().get(['lineStyle', 'curveness']), -getCurvenessForEdge(edge, graphSeries, idx, true), 0); - return { - n1: nodes_1[edge.node1.dataIndex], - n2: nodes_1[edge.node2.dataIndex], - d: d, - curveness: curveness, - ignoreForceLayout: edgeModel.get('ignoreForceLayout') - }; - }); // let coordSys = graphSeries.coordinateSystem; - - var rect = coordSys.getBoundingRect(); - var forceInstance = forceLayout(nodes_1, edges, { - rect: rect, - gravity: forceModel.get('gravity'), - friction: forceModel.get('friction') - }); - forceInstance.beforeStep(function (nodes, edges) { - for (var i = 0, l = nodes.length; i < l; i++) { - if (nodes[i].fixed) { - // Write back to layout instance - copy(nodes[i].p, graph_1.getNodeByIndex(i).getLayout()); - } - } - }); - forceInstance.afterStep(function (nodes, edges, stopped) { - for (var i = 0, l = nodes.length; i < l; i++) { - if (!nodes[i].fixed) { - graph_1.getNodeByIndex(i).setLayout(nodes[i].p); - } - - preservedPoints_1[nodeData_1.getId(i)] = nodes[i].p; - } - - for (var i = 0, l = edges.length; i < l; i++) { - var e = edges[i]; - var edge = graph_1.getEdgeByIndex(i); - var p1 = e.n1.p; - var p2 = e.n2.p; - var points = edge.getLayout(); - points = points ? points.slice() : []; - points[0] = points[0] || []; - points[1] = points[1] || []; - copy(points[0], p1); - copy(points[1], p2); - - if (+e.curveness) { - points[2] = [(p1[0] + p2[0]) / 2 - (p1[1] - p2[1]) * e.curveness, (p1[1] + p2[1]) / 2 - (p2[0] - p1[0]) * e.curveness]; - } - - edge.setLayout(points); - } - }); - graphSeries.forceLayout = forceInstance; - graphSeries.preservedPoints = preservedPoints_1; // Step to get the layout - - forceInstance.step(); - } else { - // Remove prev injected forceLayout instance - graphSeries.forceLayout = null; - } - }); - } - - function getViewRect$2(seriesModel, api, aspect) { - var option = extend(seriesModel.getBoxLayoutParams(), { - aspect: aspect - }); - return getLayoutRect(option, { - width: api.getWidth(), - height: api.getHeight() - }); - } - - function createViewCoordSys(ecModel, api) { - var viewList = []; - ecModel.eachSeriesByType('graph', function (seriesModel) { - var coordSysType = seriesModel.get('coordinateSystem'); - - if (!coordSysType || coordSysType === 'view') { - var data_1 = seriesModel.getData(); - var positions = data_1.mapArray(function (idx) { - var itemModel = data_1.getItemModel(idx); - return [+itemModel.get('x'), +itemModel.get('y')]; - }); - var min = []; - var max = []; - fromPoints(positions, min, max); // If width or height is 0 - - if (max[0] - min[0] === 0) { - max[0] += 1; - min[0] -= 1; - } - - if (max[1] - min[1] === 0) { - max[1] += 1; - min[1] -= 1; - } - - var aspect = (max[0] - min[0]) / (max[1] - min[1]); // FIXME If get view rect after data processed? - - var viewRect = getViewRect$2(seriesModel, api, aspect); // Position may be NaN, use view rect instead - - if (isNaN(aspect)) { - min = [viewRect.x, viewRect.y]; - max = [viewRect.x + viewRect.width, viewRect.y + viewRect.height]; - } - - var bbWidth = max[0] - min[0]; - var bbHeight = max[1] - min[1]; - var viewWidth = viewRect.width; - var viewHeight = viewRect.height; - var viewCoordSys = seriesModel.coordinateSystem = new View(); - viewCoordSys.zoomLimit = seriesModel.get('scaleLimit'); - viewCoordSys.setBoundingRect(min[0], min[1], bbWidth, bbHeight); - viewCoordSys.setViewRect(viewRect.x, viewRect.y, viewWidth, viewHeight); // Update roam info - - viewCoordSys.setCenter(seriesModel.get('center'), api); - viewCoordSys.setZoom(seriesModel.get('zoom')); - viewList.push(viewCoordSys); - } - }); - return viewList; - } - - var straightLineProto = Line.prototype; - var bezierCurveProto = BezierCurve.prototype; - - var StraightLineShape = - /** @class */ - function () { - function StraightLineShape() { - // Start point - this.x1 = 0; - this.y1 = 0; // End point - - this.x2 = 0; - this.y2 = 0; - this.percent = 1; - } - - return StraightLineShape; - }(); - - var CurveShape = - /** @class */ - function (_super) { - __extends(CurveShape, _super); - - function CurveShape() { - return _super !== null && _super.apply(this, arguments) || this; - } - - return CurveShape; - }(StraightLineShape); - - function isStraightLine(shape) { - return isNaN(+shape.cpx1) || isNaN(+shape.cpy1); - } - - var ECLinePath = - /** @class */ - function (_super) { - __extends(ECLinePath, _super); - - function ECLinePath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'ec-line'; - return _this; - } - - ECLinePath.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - - ECLinePath.prototype.getDefaultShape = function () { - return new StraightLineShape(); - }; - - ECLinePath.prototype.buildPath = function (ctx, shape) { - if (isStraightLine(shape)) { - straightLineProto.buildPath.call(this, ctx, shape); - } else { - bezierCurveProto.buildPath.call(this, ctx, shape); - } - }; - - ECLinePath.prototype.pointAt = function (t) { - if (isStraightLine(this.shape)) { - return straightLineProto.pointAt.call(this, t); - } else { - return bezierCurveProto.pointAt.call(this, t); - } - }; - - ECLinePath.prototype.tangentAt = function (t) { - var shape = this.shape; - var p = isStraightLine(shape) ? [shape.x2 - shape.x1, shape.y2 - shape.y1] : bezierCurveProto.tangentAt.call(this, t); - return normalize(p, p); - }; - - return ECLinePath; - }(Path); - - var SYMBOL_CATEGORIES = ['fromSymbol', 'toSymbol']; - - function makeSymbolTypeKey(symbolCategory) { - return '_' + symbolCategory + 'Type'; - } - /** - * @inner - */ - - - function createSymbol$1(name, lineData, idx) { - var symbolType = lineData.getItemVisual(idx, name); - - if (!symbolType || symbolType === 'none') { - return; - } - - var symbolSize = lineData.getItemVisual(idx, name + 'Size'); - var symbolRotate = lineData.getItemVisual(idx, name + 'Rotate'); - var symbolOffset = lineData.getItemVisual(idx, name + 'Offset'); - var symbolKeepAspect = lineData.getItemVisual(idx, name + 'KeepAspect'); - var symbolSizeArr = normalizeSymbolSize(symbolSize); - var symbolOffsetArr = normalizeSymbolOffset(symbolOffset || 0, symbolSizeArr); - var symbolPath = createSymbol(symbolType, -symbolSizeArr[0] / 2 + symbolOffsetArr[0], -symbolSizeArr[1] / 2 + symbolOffsetArr[1], symbolSizeArr[0], symbolSizeArr[1], null, symbolKeepAspect); - symbolPath.__specifiedRotation = symbolRotate == null || isNaN(symbolRotate) ? void 0 : +symbolRotate * Math.PI / 180 || 0; - symbolPath.name = name; - return symbolPath; - } - - function createLine(points) { - var line = new ECLinePath({ - name: 'line', - subPixelOptimize: true - }); - setLinePoints(line.shape, points); - return line; - } - - function setLinePoints(targetShape, points) { - targetShape.x1 = points[0][0]; - targetShape.y1 = points[0][1]; - targetShape.x2 = points[1][0]; - targetShape.y2 = points[1][1]; - targetShape.percent = 1; - var cp1 = points[2]; - - if (cp1) { - targetShape.cpx1 = cp1[0]; - targetShape.cpy1 = cp1[1]; - } else { - targetShape.cpx1 = NaN; - targetShape.cpy1 = NaN; - } - } - - var Line$1 = - /** @class */ - function (_super) { - __extends(Line, _super); - - function Line(lineData, idx, seriesScope) { - var _this = _super.call(this) || this; - - _this._createLine(lineData, idx, seriesScope); - - return _this; - } - - Line.prototype._createLine = function (lineData, idx, seriesScope) { - var seriesModel = lineData.hostModel; - var linePoints = lineData.getItemLayout(idx); - var line = createLine(linePoints); - line.shape.percent = 0; - initProps(line, { - shape: { - percent: 1 - } - }, seriesModel, idx); - this.add(line); - each(SYMBOL_CATEGORIES, function (symbolCategory) { - var symbol = createSymbol$1(symbolCategory, lineData, idx); // symbols must added after line to make sure - // it will be updated after line#update. - // Or symbol position and rotation update in line#beforeUpdate will be one frame slow - - this.add(symbol); - this[makeSymbolTypeKey(symbolCategory)] = lineData.getItemVisual(idx, symbolCategory); - }, this); - - this._updateCommonStl(lineData, idx, seriesScope); - }; // TODO More strict on the List type in parameters? - - - Line.prototype.updateData = function (lineData, idx, seriesScope) { - var seriesModel = lineData.hostModel; - var line = this.childOfName('line'); - var linePoints = lineData.getItemLayout(idx); - var target = { - shape: {} - }; - setLinePoints(target.shape, linePoints); - updateProps(line, target, seriesModel, idx); - each(SYMBOL_CATEGORIES, function (symbolCategory) { - var symbolType = lineData.getItemVisual(idx, symbolCategory); - var key = makeSymbolTypeKey(symbolCategory); // Symbol changed - - if (this[key] !== symbolType) { - this.remove(this.childOfName(symbolCategory)); - var symbol = createSymbol$1(symbolCategory, lineData, idx); - this.add(symbol); - } - - this[key] = symbolType; - }, this); - - this._updateCommonStl(lineData, idx, seriesScope); - }; - - Line.prototype.getLinePath = function () { - return this.childAt(0); - }; - - Line.prototype._updateCommonStl = function (lineData, idx, seriesScope) { - var seriesModel = lineData.hostModel; - var line = this.childOfName('line'); - var emphasisLineStyle = seriesScope && seriesScope.emphasisLineStyle; - var blurLineStyle = seriesScope && seriesScope.blurLineStyle; - var selectLineStyle = seriesScope && seriesScope.selectLineStyle; - var labelStatesModels = seriesScope && seriesScope.labelStatesModels; - var emphasisDisabled = seriesScope && seriesScope.emphasisDisabled; - var focus = seriesScope && seriesScope.focus; - var blurScope = seriesScope && seriesScope.blurScope; // Optimization for large dataset - - if (!seriesScope || lineData.hasItemOption) { - var itemModel = lineData.getItemModel(idx); - var emphasisModel = itemModel.getModel('emphasis'); - emphasisLineStyle = emphasisModel.getModel('lineStyle').getLineStyle(); - blurLineStyle = itemModel.getModel(['blur', 'lineStyle']).getLineStyle(); - selectLineStyle = itemModel.getModel(['select', 'lineStyle']).getLineStyle(); - emphasisDisabled = emphasisModel.get('disabled'); - focus = emphasisModel.get('focus'); - blurScope = emphasisModel.get('blurScope'); - labelStatesModels = getLabelStatesModels(itemModel); - } - - var lineStyle = lineData.getItemVisual(idx, 'style'); - var visualColor = lineStyle.stroke; - line.useStyle(lineStyle); - line.style.fill = null; - line.style.strokeNoScale = true; - line.ensureState('emphasis').style = emphasisLineStyle; - line.ensureState('blur').style = blurLineStyle; - line.ensureState('select').style = selectLineStyle; // Update symbol - - each(SYMBOL_CATEGORIES, function (symbolCategory) { - var symbol = this.childOfName(symbolCategory); - - if (symbol) { - // Share opacity and color with line. - symbol.setColor(visualColor); - symbol.style.opacity = lineStyle.opacity; - - for (var i = 0; i < SPECIAL_STATES.length; i++) { - var stateName = SPECIAL_STATES[i]; - var lineState = line.getState(stateName); - - if (lineState) { - var lineStateStyle = lineState.style || {}; - var state = symbol.ensureState(stateName); - var stateStyle = state.style || (state.style = {}); - - if (lineStateStyle.stroke != null) { - stateStyle[symbol.__isEmptyBrush ? 'stroke' : 'fill'] = lineStateStyle.stroke; - } - - if (lineStateStyle.opacity != null) { - stateStyle.opacity = lineStateStyle.opacity; - } - } - } - - symbol.markRedraw(); - } - }, this); - var rawVal = seriesModel.getRawValue(idx); - setLabelStyle(this, labelStatesModels, { - labelDataIndex: idx, - labelFetcher: { - getFormattedLabel: function (dataIndex, stateName) { - return seriesModel.getFormattedLabel(dataIndex, stateName, lineData.dataType); - } - }, - inheritColor: visualColor || '#000', - defaultOpacity: lineStyle.opacity, - defaultText: (rawVal == null ? lineData.getName(idx) : isFinite(rawVal) ? round(rawVal) : rawVal) + '' - }); - var label = this.getTextContent(); // Always set `textStyle` even if `normalStyle.text` is null, because default - // values have to be set on `normalStyle`. - - if (label) { - var labelNormalModel = labelStatesModels.normal; - label.__align = label.style.align; - label.__verticalAlign = label.style.verticalAlign; // 'start', 'middle', 'end' - - label.__position = labelNormalModel.get('position') || 'middle'; - var distance = labelNormalModel.get('distance'); - - if (!isArray(distance)) { - distance = [distance, distance]; - } - - label.__labelDistance = distance; - } - - this.setTextConfig({ - position: null, - local: true, - inside: false // Can't be inside for stroke element. - - }); - toggleHoverEmphasis(this, focus, blurScope, emphasisDisabled); - }; - - Line.prototype.highlight = function () { - enterEmphasis(this); - }; - - Line.prototype.downplay = function () { - leaveEmphasis(this); - }; - - Line.prototype.updateLayout = function (lineData, idx) { - this.setLinePoints(lineData.getItemLayout(idx)); - }; - - Line.prototype.setLinePoints = function (points) { - var linePath = this.childOfName('line'); - setLinePoints(linePath.shape, points); - linePath.dirty(); - }; - - Line.prototype.beforeUpdate = function () { - var lineGroup = this; - var symbolFrom = lineGroup.childOfName('fromSymbol'); - var symbolTo = lineGroup.childOfName('toSymbol'); - var label = lineGroup.getTextContent(); // Quick reject - - if (!symbolFrom && !symbolTo && (!label || label.ignore)) { - return; - } - - var invScale = 1; - var parentNode = this.parent; - - while (parentNode) { - if (parentNode.scaleX) { - invScale /= parentNode.scaleX; - } - - parentNode = parentNode.parent; - } - - var line = lineGroup.childOfName('line'); // If line not changed - // FIXME Parent scale changed - - if (!this.__dirty && !line.__dirty) { - return; - } - - var percent = line.shape.percent; - var fromPos = line.pointAt(0); - var toPos = line.pointAt(percent); - var d = sub([], toPos, fromPos); - normalize(d, d); - - function setSymbolRotation(symbol, percent) { - // Fix #12388 - // when symbol is set to be 'arrow' in markLine, - // symbolRotate value will be ignored, and compulsively use tangent angle. - // rotate by default if symbol rotation is not specified - var specifiedRotation = symbol.__specifiedRotation; - - if (specifiedRotation == null) { - var tangent = line.tangentAt(percent); - symbol.attr('rotation', (percent === 1 ? -1 : 1) * Math.PI / 2 - Math.atan2(tangent[1], tangent[0])); - } else { - symbol.attr('rotation', specifiedRotation); - } - } - - if (symbolFrom) { - symbolFrom.setPosition(fromPos); - setSymbolRotation(symbolFrom, 0); - symbolFrom.scaleX = symbolFrom.scaleY = invScale * percent; - symbolFrom.markRedraw(); - } - - if (symbolTo) { - symbolTo.setPosition(toPos); - setSymbolRotation(symbolTo, 1); - symbolTo.scaleX = symbolTo.scaleY = invScale * percent; - symbolTo.markRedraw(); - } - - if (label && !label.ignore) { - label.x = label.y = 0; - label.originX = label.originY = 0; - var textAlign = void 0; - var textVerticalAlign = void 0; - var distance = label.__labelDistance; - var distanceX = distance[0] * invScale; - var distanceY = distance[1] * invScale; - var halfPercent = percent / 2; - var tangent = line.tangentAt(halfPercent); - var n = [tangent[1], -tangent[0]]; - var cp = line.pointAt(halfPercent); - - if (n[1] > 0) { - n[0] = -n[0]; - n[1] = -n[1]; - } - - var dir = tangent[0] < 0 ? -1 : 1; - - if (label.__position !== 'start' && label.__position !== 'end') { - var rotation = -Math.atan2(tangent[1], tangent[0]); - - if (toPos[0] < fromPos[0]) { - rotation = Math.PI + rotation; - } - - label.rotation = rotation; - } - - var dy = void 0; - - switch (label.__position) { - case 'insideStartTop': - case 'insideMiddleTop': - case 'insideEndTop': - case 'middle': - dy = -distanceY; - textVerticalAlign = 'bottom'; - break; - - case 'insideStartBottom': - case 'insideMiddleBottom': - case 'insideEndBottom': - dy = distanceY; - textVerticalAlign = 'top'; - break; - - default: - dy = 0; - textVerticalAlign = 'middle'; - } - - switch (label.__position) { - case 'end': - label.x = d[0] * distanceX + toPos[0]; - label.y = d[1] * distanceY + toPos[1]; - textAlign = d[0] > 0.8 ? 'left' : d[0] < -0.8 ? 'right' : 'center'; - textVerticalAlign = d[1] > 0.8 ? 'top' : d[1] < -0.8 ? 'bottom' : 'middle'; - break; - - case 'start': - label.x = -d[0] * distanceX + fromPos[0]; - label.y = -d[1] * distanceY + fromPos[1]; - textAlign = d[0] > 0.8 ? 'right' : d[0] < -0.8 ? 'left' : 'center'; - textVerticalAlign = d[1] > 0.8 ? 'bottom' : d[1] < -0.8 ? 'top' : 'middle'; - break; - - case 'insideStartTop': - case 'insideStart': - case 'insideStartBottom': - label.x = distanceX * dir + fromPos[0]; - label.y = fromPos[1] + dy; - textAlign = tangent[0] < 0 ? 'right' : 'left'; - label.originX = -distanceX * dir; - label.originY = -dy; - break; - - case 'insideMiddleTop': - case 'insideMiddle': - case 'insideMiddleBottom': - case 'middle': - label.x = cp[0]; - label.y = cp[1] + dy; - textAlign = 'center'; - label.originY = -dy; - break; - - case 'insideEndTop': - case 'insideEnd': - case 'insideEndBottom': - label.x = -distanceX * dir + toPos[0]; - label.y = toPos[1] + dy; - textAlign = tangent[0] >= 0 ? 'right' : 'left'; - label.originX = distanceX * dir; - label.originY = -dy; - break; - } - - label.scaleX = label.scaleY = invScale; - label.setStyle({ - // Use the user specified text align and baseline first - verticalAlign: label.__verticalAlign || textVerticalAlign, - align: label.__align || textAlign - }); - } - }; - - return Line; - }(Group); - - var LineDraw = - /** @class */ - function () { - function LineDraw(LineCtor) { - this.group = new Group(); - this._LineCtor = LineCtor || Line$1; - } - - LineDraw.prototype.updateData = function (lineData) { - var _this = this; // Remove progressive els. - - - this._progressiveEls = null; - var lineDraw = this; - var group = lineDraw.group; - var oldLineData = lineDraw._lineData; - lineDraw._lineData = lineData; // There is no oldLineData only when first rendering or switching from - // stream mode to normal mode, where previous elements should be removed. - - if (!oldLineData) { - group.removeAll(); - } - - var seriesScope = makeSeriesScope$1(lineData); - lineData.diff(oldLineData).add(function (idx) { - _this._doAdd(lineData, idx, seriesScope); - }).update(function (newIdx, oldIdx) { - _this._doUpdate(oldLineData, lineData, oldIdx, newIdx, seriesScope); - }).remove(function (idx) { - group.remove(oldLineData.getItemGraphicEl(idx)); - }).execute(); - }; - - LineDraw.prototype.updateLayout = function () { - var lineData = this._lineData; // Do not support update layout in incremental mode. - - if (!lineData) { - return; - } - - lineData.eachItemGraphicEl(function (el, idx) { - el.updateLayout(lineData, idx); - }, this); - }; - - LineDraw.prototype.incrementalPrepareUpdate = function (lineData) { - this._seriesScope = makeSeriesScope$1(lineData); - this._lineData = null; - this.group.removeAll(); - }; - - LineDraw.prototype.incrementalUpdate = function (taskParams, lineData) { - this._progressiveEls = []; - - function updateIncrementalAndHover(el) { - if (!el.isGroup && !isEffectObject(el)) { - el.incremental = true; - el.ensureState('emphasis').hoverLayer = true; - } - } - - for (var idx = taskParams.start; idx < taskParams.end; idx++) { - var itemLayout = lineData.getItemLayout(idx); - - if (lineNeedsDraw(itemLayout)) { - var el = new this._LineCtor(lineData, idx, this._seriesScope); - el.traverse(updateIncrementalAndHover); - this.group.add(el); - lineData.setItemGraphicEl(idx, el); - - this._progressiveEls.push(el); - } - } - }; - - LineDraw.prototype.remove = function () { - this.group.removeAll(); - }; - - LineDraw.prototype.eachRendered = function (cb) { - traverseElements(this._progressiveEls || this.group, cb); - }; - - LineDraw.prototype._doAdd = function (lineData, idx, seriesScope) { - var itemLayout = lineData.getItemLayout(idx); - - if (!lineNeedsDraw(itemLayout)) { - return; - } - - var el = new this._LineCtor(lineData, idx, seriesScope); - lineData.setItemGraphicEl(idx, el); - this.group.add(el); - }; - - LineDraw.prototype._doUpdate = function (oldLineData, newLineData, oldIdx, newIdx, seriesScope) { - var itemEl = oldLineData.getItemGraphicEl(oldIdx); - - if (!lineNeedsDraw(newLineData.getItemLayout(newIdx))) { - this.group.remove(itemEl); - return; - } - - if (!itemEl) { - itemEl = new this._LineCtor(newLineData, newIdx, seriesScope); - } else { - itemEl.updateData(newLineData, newIdx, seriesScope); - } - - newLineData.setItemGraphicEl(newIdx, itemEl); - this.group.add(itemEl); - }; - - return LineDraw; - }(); - - function isEffectObject(el) { - return el.animators && el.animators.length > 0; - } - - function makeSeriesScope$1(lineData) { - var hostModel = lineData.hostModel; - var emphasisModel = hostModel.getModel('emphasis'); - return { - lineStyle: hostModel.getModel('lineStyle').getLineStyle(), - emphasisLineStyle: emphasisModel.getModel(['lineStyle']).getLineStyle(), - blurLineStyle: hostModel.getModel(['blur', 'lineStyle']).getLineStyle(), - selectLineStyle: hostModel.getModel(['select', 'lineStyle']).getLineStyle(), - emphasisDisabled: emphasisModel.get('disabled'), - blurScope: emphasisModel.get('blurScope'), - focus: emphasisModel.get('focus'), - labelStatesModels: getLabelStatesModels(hostModel) - }; - } - - function isPointNaN(pt) { - return isNaN(pt[0]) || isNaN(pt[1]); - } - - function lineNeedsDraw(pts) { - return pts && !isPointNaN(pts[0]) && !isPointNaN(pts[1]); - } - - var v1 = []; - var v2 = []; - var v3 = []; - var quadraticAt$1 = quadraticAt; - var v2DistSquare = distSquare; - var mathAbs$2 = Math.abs; - - function intersectCurveCircle(curvePoints, center, radius) { - var p0 = curvePoints[0]; - var p1 = curvePoints[1]; - var p2 = curvePoints[2]; - var d = Infinity; - var t; - var radiusSquare = radius * radius; - var interval = 0.1; - - for (var _t = 0.1; _t <= 0.9; _t += 0.1) { - v1[0] = quadraticAt$1(p0[0], p1[0], p2[0], _t); - v1[1] = quadraticAt$1(p0[1], p1[1], p2[1], _t); - var diff = mathAbs$2(v2DistSquare(v1, center) - radiusSquare); - - if (diff < d) { - d = diff; - t = _t; - } - } // Assume the segment is monotone,Find root through Bisection method - // At most 32 iteration - - - for (var i = 0; i < 32; i++) { - // let prev = t - interval; - var next = t + interval; // v1[0] = quadraticAt(p0[0], p1[0], p2[0], prev); - // v1[1] = quadraticAt(p0[1], p1[1], p2[1], prev); - - v2[0] = quadraticAt$1(p0[0], p1[0], p2[0], t); - v2[1] = quadraticAt$1(p0[1], p1[1], p2[1], t); - v3[0] = quadraticAt$1(p0[0], p1[0], p2[0], next); - v3[1] = quadraticAt$1(p0[1], p1[1], p2[1], next); - var diff = v2DistSquare(v2, center) - radiusSquare; - - if (mathAbs$2(diff) < 1e-2) { - break; - } // let prevDiff = v2DistSquare(v1, center) - radiusSquare; - - - var nextDiff = v2DistSquare(v3, center) - radiusSquare; - interval /= 2; - - if (diff < 0) { - if (nextDiff >= 0) { - t = t + interval; - } else { - t = t - interval; - } - } else { - if (nextDiff >= 0) { - t = t - interval; - } else { - t = t + interval; - } - } - } - - return t; - } // Adjust edge to avoid - - - function adjustEdge(graph, scale) { - var tmp0 = []; - var quadraticSubdivide$1 = quadraticSubdivide; - var pts = [[], [], []]; - var pts2 = [[], []]; - var v = []; - scale /= 2; - graph.eachEdge(function (edge, idx) { - var linePoints = edge.getLayout(); - var fromSymbol = edge.getVisual('fromSymbol'); - var toSymbol = edge.getVisual('toSymbol'); - - if (!linePoints.__original) { - linePoints.__original = [clone$1(linePoints[0]), clone$1(linePoints[1])]; - - if (linePoints[2]) { - linePoints.__original.push(clone$1(linePoints[2])); - } - } - - var originalPoints = linePoints.__original; // Quadratic curve - - if (linePoints[2] != null) { - copy(pts[0], originalPoints[0]); - copy(pts[1], originalPoints[2]); - copy(pts[2], originalPoints[1]); - - if (fromSymbol && fromSymbol !== 'none') { - var symbolSize = getSymbolSize(edge.node1); - var t = intersectCurveCircle(pts, originalPoints[0], symbolSize * scale); // Subdivide and get the second - - quadraticSubdivide$1(pts[0][0], pts[1][0], pts[2][0], t, tmp0); - pts[0][0] = tmp0[3]; - pts[1][0] = tmp0[4]; - quadraticSubdivide$1(pts[0][1], pts[1][1], pts[2][1], t, tmp0); - pts[0][1] = tmp0[3]; - pts[1][1] = tmp0[4]; - } - - if (toSymbol && toSymbol !== 'none') { - var symbolSize = getSymbolSize(edge.node2); - var t = intersectCurveCircle(pts, originalPoints[1], symbolSize * scale); // Subdivide and get the first - - quadraticSubdivide$1(pts[0][0], pts[1][0], pts[2][0], t, tmp0); - pts[1][0] = tmp0[1]; - pts[2][0] = tmp0[2]; - quadraticSubdivide$1(pts[0][1], pts[1][1], pts[2][1], t, tmp0); - pts[1][1] = tmp0[1]; - pts[2][1] = tmp0[2]; - } // Copy back to layout - - - copy(linePoints[0], pts[0]); - copy(linePoints[1], pts[2]); - copy(linePoints[2], pts[1]); - } // Line - else { - copy(pts2[0], originalPoints[0]); - copy(pts2[1], originalPoints[1]); - sub(v, pts2[1], pts2[0]); - normalize(v, v); - - if (fromSymbol && fromSymbol !== 'none') { - var symbolSize = getSymbolSize(edge.node1); - scaleAndAdd(pts2[0], pts2[0], v, symbolSize * scale); - } - - if (toSymbol && toSymbol !== 'none') { - var symbolSize = getSymbolSize(edge.node2); - scaleAndAdd(pts2[1], pts2[1], v, -symbolSize * scale); - } - - copy(linePoints[0], pts2[0]); - copy(linePoints[1], pts2[1]); - } - }); - } - - function isViewCoordSys(coordSys) { - return coordSys.type === 'view'; - } - - var GraphView = - /** @class */ - function (_super) { - __extends(GraphView, _super); - - function GraphView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GraphView.type; - return _this; - } - - GraphView.prototype.init = function (ecModel, api) { - var symbolDraw = new SymbolDraw(); - var lineDraw = new LineDraw(); - var group = this.group; - this._controller = new RoamController(api.getZr()); - this._controllerHost = { - target: group - }; - group.add(symbolDraw.group); - group.add(lineDraw.group); - this._symbolDraw = symbolDraw; - this._lineDraw = lineDraw; - this._firstRender = true; - }; - - GraphView.prototype.render = function (seriesModel, ecModel, api) { - var _this = this; - - var coordSys = seriesModel.coordinateSystem; - this._model = seriesModel; - var symbolDraw = this._symbolDraw; - var lineDraw = this._lineDraw; - var group = this.group; - - if (isViewCoordSys(coordSys)) { - var groupNewProp = { - x: coordSys.x, - y: coordSys.y, - scaleX: coordSys.scaleX, - scaleY: coordSys.scaleY - }; - - if (this._firstRender) { - group.attr(groupNewProp); - } else { - updateProps(group, groupNewProp, seriesModel); - } - } // Fix edge contact point with node - - - adjustEdge(seriesModel.getGraph(), getNodeGlobalScale(seriesModel)); - var data = seriesModel.getData(); - symbolDraw.updateData(data); - var edgeData = seriesModel.getEdgeData(); // TODO: TYPE - - lineDraw.updateData(edgeData); - - this._updateNodeAndLinkScale(); - - this._updateController(seriesModel, ecModel, api); - - clearTimeout(this._layoutTimeout); - var forceLayout = seriesModel.forceLayout; - var layoutAnimation = seriesModel.get(['force', 'layoutAnimation']); - - if (forceLayout) { - this._startForceLayoutIteration(forceLayout, layoutAnimation); - } - - var layout = seriesModel.get('layout'); - data.graph.eachNode(function (node) { - var idx = node.dataIndex; - var el = node.getGraphicEl(); - var itemModel = node.getModel(); - - if (!el) { - return; - } // Update draggable - - - el.off('drag').off('dragend'); - var draggable = itemModel.get('draggable'); - - if (draggable) { - el.on('drag', function (e) { - switch (layout) { - case 'force': - forceLayout.warmUp(); - !_this._layouting && _this._startForceLayoutIteration(forceLayout, layoutAnimation); - forceLayout.setFixed(idx); // Write position back to layout - - data.setItemLayout(idx, [el.x, el.y]); - break; - - case 'circular': - data.setItemLayout(idx, [el.x, el.y]); // mark node fixed - - node.setLayout({ - fixed: true - }, true); // recalculate circular layout - - circularLayout(seriesModel, 'symbolSize', node, [e.offsetX, e.offsetY]); - - _this.updateLayout(seriesModel); - - break; - - case 'none': - default: - data.setItemLayout(idx, [el.x, el.y]); // update edge - - simpleLayoutEdge(seriesModel.getGraph(), seriesModel); - - _this.updateLayout(seriesModel); - - break; - } - }).on('dragend', function () { - if (forceLayout) { - forceLayout.setUnfixed(idx); - } - }); - } - - el.setDraggable(draggable, !!itemModel.get('cursor')); - var focus = itemModel.get(['emphasis', 'focus']); - - if (focus === 'adjacency') { - getECData(el).focus = node.getAdjacentDataIndices(); - } - }); - data.graph.eachEdge(function (edge) { - var el = edge.getGraphicEl(); - var focus = edge.getModel().get(['emphasis', 'focus']); - - if (!el) { - return; - } - - if (focus === 'adjacency') { - getECData(el).focus = { - edge: [edge.dataIndex], - node: [edge.node1.dataIndex, edge.node2.dataIndex] - }; - } - }); - var circularRotateLabel = seriesModel.get('layout') === 'circular' && seriesModel.get(['circular', 'rotateLabel']); - var cx = data.getLayout('cx'); - var cy = data.getLayout('cy'); - data.graph.eachNode(function (node) { - rotateNodeLabel(node, circularRotateLabel, cx, cy); - }); - this._firstRender = false; - }; - - GraphView.prototype.dispose = function () { - this._controller && this._controller.dispose(); - this._controllerHost = null; - }; - - GraphView.prototype._startForceLayoutIteration = function (forceLayout, layoutAnimation) { - var self = this; - - (function step() { - forceLayout.step(function (stopped) { - self.updateLayout(self._model); - (self._layouting = !stopped) && (layoutAnimation ? self._layoutTimeout = setTimeout(step, 16) : step()); - }); - })(); - }; - - GraphView.prototype._updateController = function (seriesModel, ecModel, api) { - var _this = this; - - var controller = this._controller; - var controllerHost = this._controllerHost; - var group = this.group; - controller.setPointerChecker(function (e, x, y) { - var rect = group.getBoundingRect(); - rect.applyTransform(group.transform); - return rect.contain(x, y) && !onIrrelevantElement(e, api, seriesModel); - }); - - if (!isViewCoordSys(seriesModel.coordinateSystem)) { - controller.disable(); - return; - } - - controller.enable(seriesModel.get('roam')); - controllerHost.zoomLimit = seriesModel.get('scaleLimit'); - controllerHost.zoom = seriesModel.coordinateSystem.getZoom(); - controller.off('pan').off('zoom').on('pan', function (e) { - updateViewOnPan(controllerHost, e.dx, e.dy); - api.dispatchAction({ - seriesId: seriesModel.id, - type: 'graphRoam', - dx: e.dx, - dy: e.dy - }); - }).on('zoom', function (e) { - updateViewOnZoom(controllerHost, e.scale, e.originX, e.originY); - api.dispatchAction({ - seriesId: seriesModel.id, - type: 'graphRoam', - zoom: e.scale, - originX: e.originX, - originY: e.originY - }); - - _this._updateNodeAndLinkScale(); - - adjustEdge(seriesModel.getGraph(), getNodeGlobalScale(seriesModel)); - - _this._lineDraw.updateLayout(); // Only update label layout on zoom - - - api.updateLabelLayout(); - }); - }; - - GraphView.prototype._updateNodeAndLinkScale = function () { - var seriesModel = this._model; - var data = seriesModel.getData(); - var nodeScale = getNodeGlobalScale(seriesModel); - data.eachItemGraphicEl(function (el, idx) { - el && el.setSymbolScale(nodeScale); - }); - }; - - GraphView.prototype.updateLayout = function (seriesModel) { - adjustEdge(seriesModel.getGraph(), getNodeGlobalScale(seriesModel)); - - this._symbolDraw.updateLayout(); - - this._lineDraw.updateLayout(); - }; - - GraphView.prototype.remove = function (ecModel, api) { - this._symbolDraw && this._symbolDraw.remove(); - this._lineDraw && this._lineDraw.remove(); - }; - - GraphView.type = 'graph'; - return GraphView; - }(ChartView); - - function generateNodeKey(id) { - return '_EC_' + id; - } - - var Graph = - /** @class */ - function () { - function Graph(directed) { - this.type = 'graph'; - this.nodes = []; - this.edges = []; - this._nodesMap = {}; - /** - * @type {Object.<string, module:echarts/data/Graph.Edge>} - * @private - */ - - this._edgesMap = {}; - this._directed = directed || false; - } - /** - * If is directed graph - */ - - - Graph.prototype.isDirected = function () { - return this._directed; - }; - /** - * Add a new node - */ - - Graph.prototype.addNode = function (id, dataIndex) { - id = id == null ? '' + dataIndex : '' + id; - var nodesMap = this._nodesMap; - - if (nodesMap[generateNodeKey(id)]) { - if ("development" !== 'production') { - console.error('Graph nodes have duplicate name or id'); - } - - return; - } - - var node = new GraphNode(id, dataIndex); - node.hostGraph = this; - this.nodes.push(node); - nodesMap[generateNodeKey(id)] = node; - return node; - }; - /** - * Get node by data index - */ - - Graph.prototype.getNodeByIndex = function (dataIndex) { - var rawIdx = this.data.getRawIndex(dataIndex); - return this.nodes[rawIdx]; - }; - /** - * Get node by id - */ - - Graph.prototype.getNodeById = function (id) { - return this._nodesMap[generateNodeKey(id)]; - }; - /** - * Add a new edge - */ - - Graph.prototype.addEdge = function (n1, n2, dataIndex) { - var nodesMap = this._nodesMap; - var edgesMap = this._edgesMap; // PENDING - - if (isNumber(n1)) { - n1 = this.nodes[n1]; - } - - if (isNumber(n2)) { - n2 = this.nodes[n2]; - } - - if (!(n1 instanceof GraphNode)) { - n1 = nodesMap[generateNodeKey(n1)]; - } - - if (!(n2 instanceof GraphNode)) { - n2 = nodesMap[generateNodeKey(n2)]; - } - - if (!n1 || !n2) { - return; - } - - var key = n1.id + '-' + n2.id; - var edge = new GraphEdge(n1, n2, dataIndex); - edge.hostGraph = this; - - if (this._directed) { - n1.outEdges.push(edge); - n2.inEdges.push(edge); - } - - n1.edges.push(edge); - - if (n1 !== n2) { - n2.edges.push(edge); - } - - this.edges.push(edge); - edgesMap[key] = edge; - return edge; - }; - /** - * Get edge by data index - */ - - Graph.prototype.getEdgeByIndex = function (dataIndex) { - var rawIdx = this.edgeData.getRawIndex(dataIndex); - return this.edges[rawIdx]; - }; - /** - * Get edge by two linked nodes - */ - - Graph.prototype.getEdge = function (n1, n2) { - if (n1 instanceof GraphNode) { - n1 = n1.id; - } - - if (n2 instanceof GraphNode) { - n2 = n2.id; - } - - var edgesMap = this._edgesMap; - - if (this._directed) { - return edgesMap[n1 + '-' + n2]; - } else { - return edgesMap[n1 + '-' + n2] || edgesMap[n2 + '-' + n1]; - } - }; - /** - * Iterate all nodes - */ - - Graph.prototype.eachNode = function (cb, context) { - var nodes = this.nodes; - var len = nodes.length; - - for (var i = 0; i < len; i++) { - if (nodes[i].dataIndex >= 0) { - cb.call(context, nodes[i], i); - } - } - }; - /** - * Iterate all edges - */ - - Graph.prototype.eachEdge = function (cb, context) { - var edges = this.edges; - var len = edges.length; - - for (var i = 0; i < len; i++) { - if (edges[i].dataIndex >= 0 && edges[i].node1.dataIndex >= 0 && edges[i].node2.dataIndex >= 0) { - cb.call(context, edges[i], i); - } - } - }; - /** - * Breadth first traverse - * Return true to stop traversing - */ - - Graph.prototype.breadthFirstTraverse = function (cb, startNode, direction, context) { - if (!(startNode instanceof GraphNode)) { - startNode = this._nodesMap[generateNodeKey(startNode)]; - } - - if (!startNode) { - return; - } - - var edgeType = direction === 'out' ? 'outEdges' : direction === 'in' ? 'inEdges' : 'edges'; - - for (var i = 0; i < this.nodes.length; i++) { - this.nodes[i].__visited = false; - } - - if (cb.call(context, startNode, null)) { - return; - } - - var queue = [startNode]; - - while (queue.length) { - var currentNode = queue.shift(); - var edges = currentNode[edgeType]; - - for (var i = 0; i < edges.length; i++) { - var e = edges[i]; - var otherNode = e.node1 === currentNode ? e.node2 : e.node1; - - if (!otherNode.__visited) { - if (cb.call(context, otherNode, currentNode)) { - // Stop traversing - return; - } - - queue.push(otherNode); - otherNode.__visited = true; - } - } - } - }; - // depthFirstTraverse( - // cb, startNode, direction, context - // ) { - // }; - // Filter update - - Graph.prototype.update = function () { - var data = this.data; - var edgeData = this.edgeData; - var nodes = this.nodes; - var edges = this.edges; - - for (var i = 0, len = nodes.length; i < len; i++) { - nodes[i].dataIndex = -1; - } - - for (var i = 0, len = data.count(); i < len; i++) { - nodes[data.getRawIndex(i)].dataIndex = i; - } - - edgeData.filterSelf(function (idx) { - var edge = edges[edgeData.getRawIndex(idx)]; - return edge.node1.dataIndex >= 0 && edge.node2.dataIndex >= 0; - }); // Update edge - - for (var i = 0, len = edges.length; i < len; i++) { - edges[i].dataIndex = -1; - } - - for (var i = 0, len = edgeData.count(); i < len; i++) { - edges[edgeData.getRawIndex(i)].dataIndex = i; - } - }; - /** - * @return {module:echarts/data/Graph} - */ - - Graph.prototype.clone = function () { - var graph = new Graph(this._directed); - var nodes = this.nodes; - var edges = this.edges; - - for (var i = 0; i < nodes.length; i++) { - graph.addNode(nodes[i].id, nodes[i].dataIndex); - } - - for (var i = 0; i < edges.length; i++) { - var e = edges[i]; - graph.addEdge(e.node1.id, e.node2.id, e.dataIndex); - } - - return graph; - }; - return Graph; - }(); - - var GraphNode = - /** @class */ - function () { - function GraphNode(id, dataIndex) { - this.inEdges = []; - this.outEdges = []; - this.edges = []; - this.dataIndex = -1; - this.id = id == null ? '' : id; - this.dataIndex = dataIndex == null ? -1 : dataIndex; - } - /** - * @return {number} - */ - - - GraphNode.prototype.degree = function () { - return this.edges.length; - }; - /** - * @return {number} - */ - - - GraphNode.prototype.inDegree = function () { - return this.inEdges.length; - }; - /** - * @return {number} - */ - - - GraphNode.prototype.outDegree = function () { - return this.outEdges.length; - }; - - GraphNode.prototype.getModel = function (path) { - if (this.dataIndex < 0) { - return; - } - - var graph = this.hostGraph; - var itemModel = graph.data.getItemModel(this.dataIndex); - return itemModel.getModel(path); - }; - - GraphNode.prototype.getAdjacentDataIndices = function () { - var dataIndices = { - edge: [], - node: [] - }; - - for (var i = 0; i < this.edges.length; i++) { - var adjacentEdge = this.edges[i]; - - if (adjacentEdge.dataIndex < 0) { - continue; - } - - dataIndices.edge.push(adjacentEdge.dataIndex); - dataIndices.node.push(adjacentEdge.node1.dataIndex, adjacentEdge.node2.dataIndex); - } - - return dataIndices; - }; - - return GraphNode; - }(); - - var GraphEdge = - /** @class */ - function () { - function GraphEdge(n1, n2, dataIndex) { - this.dataIndex = -1; - this.node1 = n1; - this.node2 = n2; - this.dataIndex = dataIndex == null ? -1 : dataIndex; - } // eslint-disable-next-line @typescript-eslint/no-unused-vars - - - GraphEdge.prototype.getModel = function (path) { - if (this.dataIndex < 0) { - return; - } - - var graph = this.hostGraph; - var itemModel = graph.edgeData.getItemModel(this.dataIndex); - return itemModel.getModel(path); - }; - - GraphEdge.prototype.getAdjacentDataIndices = function () { - return { - edge: [this.dataIndex], - node: [this.node1.dataIndex, this.node2.dataIndex] - }; - }; - - return GraphEdge; - }(); - - function createGraphDataProxyMixin(hostName, dataName) { - return { - /** - * @param Default 'value'. can be 'a', 'b', 'c', 'd', 'e'. - */ - getValue: function (dimension) { - var data = this[hostName][dataName]; - return data.getStore().get(data.getDimensionIndex(dimension || 'value'), this.dataIndex); - }, - // TODO: TYPE stricter type. - setVisual: function (key, value) { - this.dataIndex >= 0 && this[hostName][dataName].setItemVisual(this.dataIndex, key, value); - }, - getVisual: function (key) { - return this[hostName][dataName].getItemVisual(this.dataIndex, key); - }, - setLayout: function (layout, merge) { - this.dataIndex >= 0 && this[hostName][dataName].setItemLayout(this.dataIndex, layout, merge); - }, - getLayout: function () { - return this[hostName][dataName].getItemLayout(this.dataIndex); - }, - getGraphicEl: function () { - return this[hostName][dataName].getItemGraphicEl(this.dataIndex); - }, - getRawIndex: function () { - return this[hostName][dataName].getRawIndex(this.dataIndex); - } - }; - } - mixin(GraphNode, createGraphDataProxyMixin('hostGraph', 'data')); - mixin(GraphEdge, createGraphDataProxyMixin('hostGraph', 'edgeData')); - - function createGraphFromNodeEdge(nodes, edges, seriesModel, directed, beforeLink) { - // ??? TODO - // support dataset? - var graph = new Graph(directed); - - for (var i = 0; i < nodes.length; i++) { - graph.addNode(retrieve( // Id, name, dataIndex - nodes[i].id, nodes[i].name, i), i); - } - - var linkNameList = []; - var validEdges = []; - var linkCount = 0; - - for (var i = 0; i < edges.length; i++) { - var link = edges[i]; - var source = link.source; - var target = link.target; // addEdge may fail when source or target not exists - - if (graph.addEdge(source, target, linkCount)) { - validEdges.push(link); - linkNameList.push(retrieve(convertOptionIdName(link.id, null), source + ' > ' + target)); - linkCount++; - } - } - - var coordSys = seriesModel.get('coordinateSystem'); - var nodeData; - - if (coordSys === 'cartesian2d' || coordSys === 'polar') { - nodeData = createSeriesData(nodes, seriesModel); - } else { - var coordSysCtor = CoordinateSystemManager.get(coordSys); - var coordDimensions = coordSysCtor ? coordSysCtor.dimensions || [] : []; // FIXME: Some geo do not need `value` dimenson, whereas `calendar` needs - // `value` dimension, but graph need `value` dimension. It's better to - // uniform this behavior. - - if (indexOf(coordDimensions, 'value') < 0) { - coordDimensions.concat(['value']); - } - - var dimensions = prepareSeriesDataSchema(nodes, { - coordDimensions: coordDimensions, - encodeDefine: seriesModel.getEncode() - }).dimensions; - nodeData = new SeriesData(dimensions, seriesModel); - nodeData.initData(nodes); - } - - var edgeData = new SeriesData(['value'], seriesModel); - edgeData.initData(validEdges, linkNameList); - beforeLink && beforeLink(nodeData, edgeData); - linkSeriesData({ - mainData: nodeData, - struct: graph, - structAttr: 'graph', - datas: { - node: nodeData, - edge: edgeData - }, - datasAttr: { - node: 'data', - edge: 'edgeData' - } - }); // Update dataIndex of nodes and edges because invalid edge may be removed - - graph.update(); - return graph; - } - - var GraphSeriesModel = - /** @class */ - function (_super) { - __extends(GraphSeriesModel, _super); - - function GraphSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GraphSeriesModel.type; - _this.hasSymbolVisual = true; - return _this; - } - - GraphSeriesModel.prototype.init = function (option) { - _super.prototype.init.apply(this, arguments); - - var self = this; - - function getCategoriesData() { - return self._categoriesData; - } // Provide data for legend select - - - this.legendVisualProvider = new LegendVisualProvider(getCategoriesData, getCategoriesData); - this.fillDataTextStyle(option.edges || option.links); - - this._updateCategoriesData(); - }; - - GraphSeriesModel.prototype.mergeOption = function (option) { - _super.prototype.mergeOption.apply(this, arguments); - - this.fillDataTextStyle(option.edges || option.links); - - this._updateCategoriesData(); - }; - - GraphSeriesModel.prototype.mergeDefaultAndTheme = function (option) { - _super.prototype.mergeDefaultAndTheme.apply(this, arguments); - - defaultEmphasis(option, 'edgeLabel', ['show']); - }; - - GraphSeriesModel.prototype.getInitialData = function (option, ecModel) { - var edges = option.edges || option.links || []; - var nodes = option.data || option.nodes || []; - var self = this; - - if (nodes && edges) { - // auto curveness - initCurvenessList(this); - var graph = createGraphFromNodeEdge(nodes, edges, this, true, beforeLink); - each(graph.edges, function (edge) { - createEdgeMapForCurveness(edge.node1, edge.node2, this, edge.dataIndex); - }, this); - return graph.data; - } - - function beforeLink(nodeData, edgeData) { - // Overwrite nodeData.getItemModel to - nodeData.wrapMethod('getItemModel', function (model) { - var categoriesModels = self._categoriesModels; - var categoryIdx = model.getShallow('category'); - var categoryModel = categoriesModels[categoryIdx]; - - if (categoryModel) { - categoryModel.parentModel = model.parentModel; - model.parentModel = categoryModel; - } - - return model; - }); // TODO Inherit resolveParentPath by default in Model#getModel? - - var oldGetModel = Model.prototype.getModel; - - function newGetModel(path, parentModel) { - var model = oldGetModel.call(this, path, parentModel); - model.resolveParentPath = resolveParentPath; - return model; - } - - edgeData.wrapMethod('getItemModel', function (model) { - model.resolveParentPath = resolveParentPath; - model.getModel = newGetModel; - return model; - }); - - function resolveParentPath(pathArr) { - if (pathArr && (pathArr[0] === 'label' || pathArr[1] === 'label')) { - var newPathArr = pathArr.slice(); - - if (pathArr[0] === 'label') { - newPathArr[0] = 'edgeLabel'; - } else if (pathArr[1] === 'label') { - newPathArr[1] = 'edgeLabel'; - } - - return newPathArr; - } - - return pathArr; - } - } - }; - - GraphSeriesModel.prototype.getGraph = function () { - return this.getData().graph; - }; - - GraphSeriesModel.prototype.getEdgeData = function () { - return this.getGraph().edgeData; - }; - - GraphSeriesModel.prototype.getCategoriesData = function () { - return this._categoriesData; - }; - - GraphSeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - if (dataType === 'edge') { - var nodeData = this.getData(); - var params = this.getDataParams(dataIndex, dataType); - var edge = nodeData.graph.getEdgeByIndex(dataIndex); - var sourceName = nodeData.getName(edge.node1.dataIndex); - var targetName = nodeData.getName(edge.node2.dataIndex); - var nameArr = []; - sourceName != null && nameArr.push(sourceName); - targetName != null && nameArr.push(targetName); - return createTooltipMarkup('nameValue', { - name: nameArr.join(' > '), - value: params.value, - noValue: params.value == null - }); - } // dataType === 'node' or empty - - - var nodeMarkup = defaultSeriesFormatTooltip({ - series: this, - dataIndex: dataIndex, - multipleSeries: multipleSeries - }); - return nodeMarkup; - }; - - GraphSeriesModel.prototype._updateCategoriesData = function () { - var categories = map(this.option.categories || [], function (category) { - // Data must has value - return category.value != null ? category : extend({ - value: 0 - }, category); - }); - var categoriesData = new SeriesData(['value'], this); - categoriesData.initData(categories); - this._categoriesData = categoriesData; - this._categoriesModels = categoriesData.mapArray(function (idx) { - return categoriesData.getItemModel(idx); - }); - }; - - GraphSeriesModel.prototype.setZoom = function (zoom) { - this.option.zoom = zoom; - }; - - GraphSeriesModel.prototype.setCenter = function (center) { - this.option.center = center; - }; - - GraphSeriesModel.prototype.isAnimationEnabled = function () { - return _super.prototype.isAnimationEnabled.call(this) // Not enable animation when do force layout - && !(this.get('layout') === 'force' && this.get(['force', 'layoutAnimation'])); - }; - - GraphSeriesModel.type = 'series.graph'; - GraphSeriesModel.dependencies = ['grid', 'polar', 'geo', 'singleAxis', 'calendar']; - GraphSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'view', - // Default option for all coordinate systems - // xAxisIndex: 0, - // yAxisIndex: 0, - // polarIndex: 0, - // geoIndex: 0, - legendHoverLink: true, - layout: null, - // Configuration of circular layout - circular: { - rotateLabel: false - }, - // Configuration of force directed layout - force: { - initLayout: null, - // Node repulsion. Can be an array to represent range. - repulsion: [0, 50], - gravity: 0.1, - // Initial friction - friction: 0.6, - // Edge length. Can be an array to represent range. - edgeLength: 30, - layoutAnimation: true - }, - left: 'center', - top: 'center', - // right: null, - // bottom: null, - // width: '80%', - // height: '80%', - symbol: 'circle', - symbolSize: 10, - edgeSymbol: ['none', 'none'], - edgeSymbolSize: 10, - edgeLabel: { - position: 'middle', - distance: 5 - }, - draggable: false, - roam: false, - // Default on center of graph - center: null, - zoom: 1, - // Symbol size scale ratio in roam - nodeScaleRatio: 0.6, - // cursor: null, - // categories: [], - // data: [] - // Or - // nodes: [] - // - // links: [] - // Or - // edges: [] - label: { - show: false, - formatter: '{b}' - }, - itemStyle: {}, - lineStyle: { - color: '#aaa', - width: 1, - opacity: 0.5 - }, - emphasis: { - scale: true, - label: { - show: true - } - }, - select: { - itemStyle: { - borderColor: '#212121' - } - } - }; - return GraphSeriesModel; - }(SeriesModel); - - var actionInfo = { - type: 'graphRoam', - event: 'graphRoam', - update: 'none' - }; - function install$d(registers) { - registers.registerChartView(GraphView); - registers.registerSeriesModel(GraphSeriesModel); - registers.registerProcessor(categoryFilter); - registers.registerVisual(categoryVisual); - registers.registerVisual(graphEdgeVisual); - registers.registerLayout(graphSimpleLayout); - registers.registerLayout(registers.PRIORITY.VISUAL.POST_CHART_LAYOUT, graphCircularLayout); - registers.registerLayout(graphForceLayout); - registers.registerCoordinateSystem('graphView', { - dimensions: View.dimensions, - create: createViewCoordSys - }); // Register legacy focus actions - - registers.registerAction({ - type: 'focusNodeAdjacency', - event: 'focusNodeAdjacency', - update: 'series:focusNodeAdjacency' - }, noop); - registers.registerAction({ - type: 'unfocusNodeAdjacency', - event: 'unfocusNodeAdjacency', - update: 'series:unfocusNodeAdjacency' - }, noop); // Register roam action. - - registers.registerAction(actionInfo, function (payload, ecModel, api) { - ecModel.eachComponent({ - mainType: 'series', - query: payload - }, function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - var res = updateCenterAndZoom(coordSys, payload, undefined, api); - seriesModel.setCenter && seriesModel.setCenter(res.center); - seriesModel.setZoom && seriesModel.setZoom(res.zoom); - }); - }); - } - - var PointerShape = - /** @class */ - function () { - function PointerShape() { - this.angle = 0; - this.width = 10; - this.r = 10; - this.x = 0; - this.y = 0; - } - - return PointerShape; - }(); - - var PointerPath = - /** @class */ - function (_super) { - __extends(PointerPath, _super); - - function PointerPath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'pointer'; - return _this; - } - - PointerPath.prototype.getDefaultShape = function () { - return new PointerShape(); - }; - - PointerPath.prototype.buildPath = function (ctx, shape) { - var mathCos = Math.cos; - var mathSin = Math.sin; - var r = shape.r; - var width = shape.width; - var angle = shape.angle; - var x = shape.x - mathCos(angle) * width * (width >= r / 3 ? 1 : 2); - var y = shape.y - mathSin(angle) * width * (width >= r / 3 ? 1 : 2); - angle = shape.angle - Math.PI / 2; - ctx.moveTo(x, y); - ctx.lineTo(shape.x + mathCos(angle) * width, shape.y + mathSin(angle) * width); - ctx.lineTo(shape.x + mathCos(shape.angle) * r, shape.y + mathSin(shape.angle) * r); - ctx.lineTo(shape.x - mathCos(angle) * width, shape.y - mathSin(angle) * width); - ctx.lineTo(x, y); - }; - - return PointerPath; - }(Path); - - function parsePosition(seriesModel, api) { - var center = seriesModel.get('center'); - var width = api.getWidth(); - var height = api.getHeight(); - var size = Math.min(width, height); - var cx = parsePercent$1(center[0], api.getWidth()); - var cy = parsePercent$1(center[1], api.getHeight()); - var r = parsePercent$1(seriesModel.get('radius'), size / 2); - return { - cx: cx, - cy: cy, - r: r - }; - } - - function formatLabel(value, labelFormatter) { - var label = value == null ? '' : value + ''; - - if (labelFormatter) { - if (isString(labelFormatter)) { - label = labelFormatter.replace('{value}', label); - } else if (isFunction(labelFormatter)) { - label = labelFormatter(value); - } - } - - return label; - } - - var GaugeView = - /** @class */ - function (_super) { - __extends(GaugeView, _super); - - function GaugeView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GaugeView.type; - return _this; - } - - GaugeView.prototype.render = function (seriesModel, ecModel, api) { - this.group.removeAll(); - var colorList = seriesModel.get(['axisLine', 'lineStyle', 'color']); - var posInfo = parsePosition(seriesModel, api); - - this._renderMain(seriesModel, ecModel, api, colorList, posInfo); - - this._data = seriesModel.getData(); - }; - - GaugeView.prototype.dispose = function () {}; - - GaugeView.prototype._renderMain = function (seriesModel, ecModel, api, colorList, posInfo) { - var group = this.group; - var clockwise = seriesModel.get('clockwise'); - var startAngle = -seriesModel.get('startAngle') / 180 * Math.PI; - var endAngle = -seriesModel.get('endAngle') / 180 * Math.PI; - var axisLineModel = seriesModel.getModel('axisLine'); - var roundCap = axisLineModel.get('roundCap'); - var MainPath = roundCap ? SausagePath : Sector; - var showAxis = axisLineModel.get('show'); - var lineStyleModel = axisLineModel.getModel('lineStyle'); - var axisLineWidth = lineStyleModel.get('width'); - var angles = [startAngle, endAngle]; - normalizeArcAngles(angles, !clockwise); - startAngle = angles[0]; - endAngle = angles[1]; - var angleRangeSpan = endAngle - startAngle; - var prevEndAngle = startAngle; - var sectors = []; - - for (var i = 0; showAxis && i < colorList.length; i++) { - // Clamp - var percent = Math.min(Math.max(colorList[i][0], 0), 1); - endAngle = startAngle + angleRangeSpan * percent; - var sector = new MainPath({ - shape: { - startAngle: prevEndAngle, - endAngle: endAngle, - cx: posInfo.cx, - cy: posInfo.cy, - clockwise: clockwise, - r0: posInfo.r - axisLineWidth, - r: posInfo.r - }, - silent: true - }); - sector.setStyle({ - fill: colorList[i][1] - }); - sector.setStyle(lineStyleModel.getLineStyle( // Because we use sector to simulate arc - // so the properties for stroking are useless - ['color', 'width'])); - sectors.push(sector); - prevEndAngle = endAngle; - } - - sectors.reverse(); - each(sectors, function (sector) { - return group.add(sector); - }); - - var getColor = function (percent) { - // Less than 0 - if (percent <= 0) { - return colorList[0][1]; - } - - var i; - - for (i = 0; i < colorList.length; i++) { - if (colorList[i][0] >= percent && (i === 0 ? 0 : colorList[i - 1][0]) < percent) { - return colorList[i][1]; - } - } // More than 1 - - - return colorList[i - 1][1]; - }; - - this._renderTicks(seriesModel, ecModel, api, getColor, posInfo, startAngle, endAngle, clockwise, axisLineWidth); - - this._renderTitleAndDetail(seriesModel, ecModel, api, getColor, posInfo); - - this._renderAnchor(seriesModel, posInfo); - - this._renderPointer(seriesModel, ecModel, api, getColor, posInfo, startAngle, endAngle, clockwise, axisLineWidth); - }; - - GaugeView.prototype._renderTicks = function (seriesModel, ecModel, api, getColor, posInfo, startAngle, endAngle, clockwise, axisLineWidth) { - var group = this.group; - var cx = posInfo.cx; - var cy = posInfo.cy; - var r = posInfo.r; - var minVal = +seriesModel.get('min'); - var maxVal = +seriesModel.get('max'); - var splitLineModel = seriesModel.getModel('splitLine'); - var tickModel = seriesModel.getModel('axisTick'); - var labelModel = seriesModel.getModel('axisLabel'); - var splitNumber = seriesModel.get('splitNumber'); - var subSplitNumber = tickModel.get('splitNumber'); - var splitLineLen = parsePercent$1(splitLineModel.get('length'), r); - var tickLen = parsePercent$1(tickModel.get('length'), r); - var angle = startAngle; - var step = (endAngle - startAngle) / splitNumber; - var subStep = step / subSplitNumber; - var splitLineStyle = splitLineModel.getModel('lineStyle').getLineStyle(); - var tickLineStyle = tickModel.getModel('lineStyle').getLineStyle(); - var splitLineDistance = splitLineModel.get('distance'); - var unitX; - var unitY; - - for (var i = 0; i <= splitNumber; i++) { - unitX = Math.cos(angle); - unitY = Math.sin(angle); // Split line - - if (splitLineModel.get('show')) { - var distance = splitLineDistance ? splitLineDistance + axisLineWidth : axisLineWidth; - var splitLine = new Line({ - shape: { - x1: unitX * (r - distance) + cx, - y1: unitY * (r - distance) + cy, - x2: unitX * (r - splitLineLen - distance) + cx, - y2: unitY * (r - splitLineLen - distance) + cy - }, - style: splitLineStyle, - silent: true - }); - - if (splitLineStyle.stroke === 'auto') { - splitLine.setStyle({ - stroke: getColor(i / splitNumber) - }); - } - - group.add(splitLine); - } // Label - - - if (labelModel.get('show')) { - var distance = labelModel.get('distance') + splitLineDistance; - var label = formatLabel(round(i / splitNumber * (maxVal - minVal) + minVal), labelModel.get('formatter')); - var autoColor = getColor(i / splitNumber); - var textStyleX = unitX * (r - splitLineLen - distance) + cx; - var textStyleY = unitY * (r - splitLineLen - distance) + cy; - var rotateType = labelModel.get('rotate'); - var rotate = 0; - - if (rotateType === 'radial') { - rotate = -angle + 2 * Math.PI; - - if (rotate > Math.PI / 2) { - rotate += Math.PI; - } - } else if (rotateType === 'tangential') { - rotate = -angle - Math.PI / 2; - } else if (isNumber(rotateType)) { - rotate = rotateType * Math.PI / 180; - } - - if (rotate === 0) { - group.add(new ZRText({ - style: createTextStyle(labelModel, { - text: label, - x: textStyleX, - y: textStyleY, - verticalAlign: unitY < -0.8 ? 'top' : unitY > 0.8 ? 'bottom' : 'middle', - align: unitX < -0.4 ? 'left' : unitX > 0.4 ? 'right' : 'center' - }, { - inheritColor: autoColor - }), - silent: true - })); - } else { - group.add(new ZRText({ - style: createTextStyle(labelModel, { - text: label, - x: textStyleX, - y: textStyleY, - verticalAlign: 'middle', - align: 'center' - }, { - inheritColor: autoColor - }), - silent: true, - originX: textStyleX, - originY: textStyleY, - rotation: rotate - })); - } - } // Axis tick - - - if (tickModel.get('show') && i !== splitNumber) { - var distance = tickModel.get('distance'); - distance = distance ? distance + axisLineWidth : axisLineWidth; - - for (var j = 0; j <= subSplitNumber; j++) { - unitX = Math.cos(angle); - unitY = Math.sin(angle); - var tickLine = new Line({ - shape: { - x1: unitX * (r - distance) + cx, - y1: unitY * (r - distance) + cy, - x2: unitX * (r - tickLen - distance) + cx, - y2: unitY * (r - tickLen - distance) + cy - }, - silent: true, - style: tickLineStyle - }); - - if (tickLineStyle.stroke === 'auto') { - tickLine.setStyle({ - stroke: getColor((i + j / subSplitNumber) / splitNumber) - }); - } - - group.add(tickLine); - angle += subStep; - } - - angle -= subStep; - } else { - angle += step; - } - } - }; - - GaugeView.prototype._renderPointer = function (seriesModel, ecModel, api, getColor, posInfo, startAngle, endAngle, clockwise, axisLineWidth) { - var group = this.group; - var oldData = this._data; - var oldProgressData = this._progressEls; - var progressList = []; - var showPointer = seriesModel.get(['pointer', 'show']); - var progressModel = seriesModel.getModel('progress'); - var showProgress = progressModel.get('show'); - var data = seriesModel.getData(); - var valueDim = data.mapDimension('value'); - var minVal = +seriesModel.get('min'); - var maxVal = +seriesModel.get('max'); - var valueExtent = [minVal, maxVal]; - var angleExtent = [startAngle, endAngle]; - - function createPointer(idx, angle) { - var itemModel = data.getItemModel(idx); - var pointerModel = itemModel.getModel('pointer'); - var pointerWidth = parsePercent$1(pointerModel.get('width'), posInfo.r); - var pointerLength = parsePercent$1(pointerModel.get('length'), posInfo.r); - var pointerStr = seriesModel.get(['pointer', 'icon']); - var pointerOffset = pointerModel.get('offsetCenter'); - var pointerOffsetX = parsePercent$1(pointerOffset[0], posInfo.r); - var pointerOffsetY = parsePercent$1(pointerOffset[1], posInfo.r); - var pointerKeepAspect = pointerModel.get('keepAspect'); - var pointer; // not exist icon type will be set 'rect' - - if (pointerStr) { - pointer = createSymbol(pointerStr, pointerOffsetX - pointerWidth / 2, pointerOffsetY - pointerLength, pointerWidth, pointerLength, null, pointerKeepAspect); - } else { - pointer = new PointerPath({ - shape: { - angle: -Math.PI / 2, - width: pointerWidth, - r: pointerLength, - x: pointerOffsetX, - y: pointerOffsetY - } - }); - } - - pointer.rotation = -(angle + Math.PI / 2); - pointer.x = posInfo.cx; - pointer.y = posInfo.cy; - return pointer; - } - - function createProgress(idx, endAngle) { - var roundCap = progressModel.get('roundCap'); - var ProgressPath = roundCap ? SausagePath : Sector; - var isOverlap = progressModel.get('overlap'); - var progressWidth = isOverlap ? progressModel.get('width') : axisLineWidth / data.count(); - var r0 = isOverlap ? posInfo.r - progressWidth : posInfo.r - (idx + 1) * progressWidth; - var r = isOverlap ? posInfo.r : posInfo.r - idx * progressWidth; - var progress = new ProgressPath({ - shape: { - startAngle: startAngle, - endAngle: endAngle, - cx: posInfo.cx, - cy: posInfo.cy, - clockwise: clockwise, - r0: r0, - r: r - } - }); - isOverlap && (progress.z2 = maxVal - data.get(valueDim, idx) % maxVal); - return progress; - } - - if (showProgress || showPointer) { - data.diff(oldData).add(function (idx) { - var val = data.get(valueDim, idx); - - if (showPointer) { - var pointer = createPointer(idx, startAngle); // TODO hide pointer on NaN value? - - initProps(pointer, { - rotation: -((isNaN(+val) ? angleExtent[0] : linearMap(val, valueExtent, angleExtent, true)) + Math.PI / 2) - }, seriesModel); - group.add(pointer); - data.setItemGraphicEl(idx, pointer); - } - - if (showProgress) { - var progress = createProgress(idx, startAngle); - var isClip = progressModel.get('clip'); - initProps(progress, { - shape: { - endAngle: linearMap(val, valueExtent, angleExtent, isClip) - } - }, seriesModel); - group.add(progress); // Add data index and series index for indexing the data by element - // Useful in tooltip - - setCommonECData(seriesModel.seriesIndex, data.dataType, idx, progress); - progressList[idx] = progress; - } - }).update(function (newIdx, oldIdx) { - var val = data.get(valueDim, newIdx); - - if (showPointer) { - var previousPointer = oldData.getItemGraphicEl(oldIdx); - var previousRotate = previousPointer ? previousPointer.rotation : startAngle; - var pointer = createPointer(newIdx, previousRotate); - pointer.rotation = previousRotate; - updateProps(pointer, { - rotation: -((isNaN(+val) ? angleExtent[0] : linearMap(val, valueExtent, angleExtent, true)) + Math.PI / 2) - }, seriesModel); - group.add(pointer); - data.setItemGraphicEl(newIdx, pointer); - } - - if (showProgress) { - var previousProgress = oldProgressData[oldIdx]; - var previousEndAngle = previousProgress ? previousProgress.shape.endAngle : startAngle; - var progress = createProgress(newIdx, previousEndAngle); - var isClip = progressModel.get('clip'); - updateProps(progress, { - shape: { - endAngle: linearMap(val, valueExtent, angleExtent, isClip) - } - }, seriesModel); - group.add(progress); // Add data index and series index for indexing the data by element - // Useful in tooltip - - setCommonECData(seriesModel.seriesIndex, data.dataType, newIdx, progress); - progressList[newIdx] = progress; - } - }).execute(); - data.each(function (idx) { - var itemModel = data.getItemModel(idx); - var emphasisModel = itemModel.getModel('emphasis'); - var focus = emphasisModel.get('focus'); - var blurScope = emphasisModel.get('blurScope'); - var emphasisDisabled = emphasisModel.get('disabled'); - - if (showPointer) { - var pointer = data.getItemGraphicEl(idx); - var symbolStyle = data.getItemVisual(idx, 'style'); - var visualColor = symbolStyle.fill; - - if (pointer instanceof ZRImage) { - var pathStyle = pointer.style; - pointer.useStyle(extend({ - image: pathStyle.image, - x: pathStyle.x, - y: pathStyle.y, - width: pathStyle.width, - height: pathStyle.height - }, symbolStyle)); - } else { - pointer.useStyle(symbolStyle); - pointer.type !== 'pointer' && pointer.setColor(visualColor); - } - - pointer.setStyle(itemModel.getModel(['pointer', 'itemStyle']).getItemStyle()); - - if (pointer.style.fill === 'auto') { - pointer.setStyle('fill', getColor(linearMap(data.get(valueDim, idx), valueExtent, [0, 1], true))); - } - - pointer.z2EmphasisLift = 0; - setStatesStylesFromModel(pointer, itemModel); - toggleHoverEmphasis(pointer, focus, blurScope, emphasisDisabled); - } - - if (showProgress) { - var progress = progressList[idx]; - progress.useStyle(data.getItemVisual(idx, 'style')); - progress.setStyle(itemModel.getModel(['progress', 'itemStyle']).getItemStyle()); - progress.z2EmphasisLift = 0; - setStatesStylesFromModel(progress, itemModel); - toggleHoverEmphasis(progress, focus, blurScope, emphasisDisabled); - } - }); - this._progressEls = progressList; - } - }; - - GaugeView.prototype._renderAnchor = function (seriesModel, posInfo) { - var anchorModel = seriesModel.getModel('anchor'); - var showAnchor = anchorModel.get('show'); - - if (showAnchor) { - var anchorSize = anchorModel.get('size'); - var anchorType = anchorModel.get('icon'); - var offsetCenter = anchorModel.get('offsetCenter'); - var anchorKeepAspect = anchorModel.get('keepAspect'); - var anchor = createSymbol(anchorType, posInfo.cx - anchorSize / 2 + parsePercent$1(offsetCenter[0], posInfo.r), posInfo.cy - anchorSize / 2 + parsePercent$1(offsetCenter[1], posInfo.r), anchorSize, anchorSize, null, anchorKeepAspect); - anchor.z2 = anchorModel.get('showAbove') ? 1 : 0; - anchor.setStyle(anchorModel.getModel('itemStyle').getItemStyle()); - this.group.add(anchor); - } - }; - - GaugeView.prototype._renderTitleAndDetail = function (seriesModel, ecModel, api, getColor, posInfo) { - var _this = this; - - var data = seriesModel.getData(); - var valueDim = data.mapDimension('value'); - var minVal = +seriesModel.get('min'); - var maxVal = +seriesModel.get('max'); - var contentGroup = new Group(); - var newTitleEls = []; - var newDetailEls = []; - var hasAnimation = seriesModel.isAnimationEnabled(); - var showPointerAbove = seriesModel.get(['pointer', 'showAbove']); - data.diff(this._data).add(function (idx) { - newTitleEls[idx] = new ZRText({ - silent: true - }); - newDetailEls[idx] = new ZRText({ - silent: true - }); - }).update(function (idx, oldIdx) { - newTitleEls[idx] = _this._titleEls[oldIdx]; - newDetailEls[idx] = _this._detailEls[oldIdx]; - }).execute(); - data.each(function (idx) { - var itemModel = data.getItemModel(idx); - var value = data.get(valueDim, idx); - var itemGroup = new Group(); - var autoColor = getColor(linearMap(value, [minVal, maxVal], [0, 1], true)); - var itemTitleModel = itemModel.getModel('title'); - - if (itemTitleModel.get('show')) { - var titleOffsetCenter = itemTitleModel.get('offsetCenter'); - var titleX = posInfo.cx + parsePercent$1(titleOffsetCenter[0], posInfo.r); - var titleY = posInfo.cy + parsePercent$1(titleOffsetCenter[1], posInfo.r); - var labelEl = newTitleEls[idx]; - labelEl.attr({ - z2: showPointerAbove ? 0 : 2, - style: createTextStyle(itemTitleModel, { - x: titleX, - y: titleY, - text: data.getName(idx), - align: 'center', - verticalAlign: 'middle' - }, { - inheritColor: autoColor - }) - }); - itemGroup.add(labelEl); - } - - var itemDetailModel = itemModel.getModel('detail'); - - if (itemDetailModel.get('show')) { - var detailOffsetCenter = itemDetailModel.get('offsetCenter'); - var detailX = posInfo.cx + parsePercent$1(detailOffsetCenter[0], posInfo.r); - var detailY = posInfo.cy + parsePercent$1(detailOffsetCenter[1], posInfo.r); - var width = parsePercent$1(itemDetailModel.get('width'), posInfo.r); - var height = parsePercent$1(itemDetailModel.get('height'), posInfo.r); - var detailColor = seriesModel.get(['progress', 'show']) ? data.getItemVisual(idx, 'style').fill : autoColor; - var labelEl = newDetailEls[idx]; - var formatter_1 = itemDetailModel.get('formatter'); - labelEl.attr({ - z2: showPointerAbove ? 0 : 2, - style: createTextStyle(itemDetailModel, { - x: detailX, - y: detailY, - text: formatLabel(value, formatter_1), - width: isNaN(width) ? null : width, - height: isNaN(height) ? null : height, - align: 'center', - verticalAlign: 'middle' - }, { - inheritColor: detailColor - }) - }); - setLabelValueAnimation(labelEl, { - normal: itemDetailModel - }, value, function (value) { - return formatLabel(value, formatter_1); - }); - hasAnimation && animateLabelValue(labelEl, idx, data, seriesModel, { - getFormattedLabel: function (labelDataIndex, status, dataType, labelDimIndex, fmt, extendParams) { - return formatLabel(extendParams ? extendParams.interpolatedValue : value, formatter_1); - } - }); - itemGroup.add(labelEl); - } - - contentGroup.add(itemGroup); - }); - this.group.add(contentGroup); - this._titleEls = newTitleEls; - this._detailEls = newDetailEls; - }; - - GaugeView.type = 'gauge'; - return GaugeView; - }(ChartView); - - var GaugeSeriesModel = - /** @class */ - function (_super) { - __extends(GaugeSeriesModel, _super); - - function GaugeSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GaugeSeriesModel.type; - _this.visualStyleAccessPath = 'itemStyle'; - return _this; - } - - GaugeSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesDataSimply(this, ['value']); - }; - - GaugeSeriesModel.type = 'series.gauge'; - GaugeSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - colorBy: 'data', - // 默认全局居中 - center: ['50%', '50%'], - legendHoverLink: true, - radius: '75%', - startAngle: 225, - endAngle: -45, - clockwise: true, - // 最小值 - min: 0, - // 最大值 - max: 100, - // 分割段数,默认为10 - splitNumber: 10, - // 坐标轴线 - axisLine: { - // 默认显示,属性show控制显示与否 - show: true, - roundCap: false, - lineStyle: { - color: [[1, '#E6EBF8']], - width: 10 - } - }, - // 坐标轴线 - progress: { - // 默认显示,属性show控制显示与否 - show: false, - overlap: true, - width: 10, - roundCap: false, - clip: true - }, - // 分隔线 - splitLine: { - // 默认显示,属性show控制显示与否 - show: true, - // 属性length控制线长 - length: 10, - distance: 10, - // 属性lineStyle(详见lineStyle)控制线条样式 - lineStyle: { - color: '#63677A', - width: 3, - type: 'solid' - } - }, - // 坐标轴小标记 - axisTick: { - // 属性show控制显示与否,默认不显示 - show: true, - // 每份split细分多少段 - splitNumber: 5, - // 属性length控制线长 - length: 6, - distance: 10, - // 属性lineStyle控制线条样式 - lineStyle: { - color: '#63677A', - width: 1, - type: 'solid' - } - }, - axisLabel: { - show: true, - distance: 15, - // formatter: null, - color: '#464646', - fontSize: 12, - rotate: 0 - }, - pointer: { - icon: null, - offsetCenter: [0, 0], - show: true, - showAbove: true, - length: '60%', - width: 6, - keepAspect: false - }, - anchor: { - show: false, - showAbove: false, - size: 6, - icon: 'circle', - offsetCenter: [0, 0], - keepAspect: false, - itemStyle: { - color: '#fff', - borderWidth: 0, - borderColor: '#5470c6' - } - }, - title: { - show: true, - // x, y,单位px - offsetCenter: [0, '20%'], - // 其余属性默认使用全局文本样式,详见TEXTSTYLE - color: '#464646', - fontSize: 16, - valueAnimation: false - }, - detail: { - show: true, - backgroundColor: 'rgba(0,0,0,0)', - borderWidth: 0, - borderColor: '#ccc', - width: 100, - height: null, - padding: [5, 10], - // x, y,单位px - offsetCenter: [0, '40%'], - // formatter: null, - // 其余属性默认使用全局文本样式,详见TEXTSTYLE - color: '#464646', - fontSize: 30, - fontWeight: 'bold', - lineHeight: 30, - valueAnimation: false - } - }; - return GaugeSeriesModel; - }(SeriesModel); - - function install$e(registers) { - registers.registerChartView(GaugeView); - registers.registerSeriesModel(GaugeSeriesModel); - } - - var opacityAccessPath = ['itemStyle', 'opacity']; - /** - * Piece of pie including Sector, Label, LabelLine - */ - - var FunnelPiece = - /** @class */ - function (_super) { - __extends(FunnelPiece, _super); - - function FunnelPiece(data, idx) { - var _this = _super.call(this) || this; - - var polygon = _this; - var labelLine = new Polyline(); - var text = new ZRText(); - polygon.setTextContent(text); - - _this.setTextGuideLine(labelLine); - - _this.updateData(data, idx, true); - - return _this; - } - - FunnelPiece.prototype.updateData = function (data, idx, firstCreate) { - var polygon = this; - var seriesModel = data.hostModel; - var itemModel = data.getItemModel(idx); - var layout = data.getItemLayout(idx); - var emphasisModel = itemModel.getModel('emphasis'); - var opacity = itemModel.get(opacityAccessPath); - opacity = opacity == null ? 1 : opacity; - - if (!firstCreate) { - saveOldStyle(polygon); - } // Update common style - - - polygon.useStyle(data.getItemVisual(idx, 'style')); - polygon.style.lineJoin = 'round'; - - if (firstCreate) { - polygon.setShape({ - points: layout.points - }); - polygon.style.opacity = 0; - initProps(polygon, { - style: { - opacity: opacity - } - }, seriesModel, idx); - } else { - updateProps(polygon, { - style: { - opacity: opacity - }, - shape: { - points: layout.points - } - }, seriesModel, idx); - } - - setStatesStylesFromModel(polygon, itemModel); - - this._updateLabel(data, idx); - - toggleHoverEmphasis(this, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - }; - - FunnelPiece.prototype._updateLabel = function (data, idx) { - var polygon = this; - var labelLine = this.getTextGuideLine(); - var labelText = polygon.getTextContent(); - var seriesModel = data.hostModel; - var itemModel = data.getItemModel(idx); - var layout = data.getItemLayout(idx); - var labelLayout = layout.label; - var style = data.getItemVisual(idx, 'style'); - var visualColor = style.fill; - setLabelStyle( // position will not be used in setLabelStyle - labelText, getLabelStatesModels(itemModel), { - labelFetcher: data.hostModel, - labelDataIndex: idx, - defaultOpacity: style.opacity, - defaultText: data.getName(idx) - }, { - normal: { - align: labelLayout.textAlign, - verticalAlign: labelLayout.verticalAlign - } - }); - polygon.setTextConfig({ - local: true, - inside: !!labelLayout.inside, - insideStroke: visualColor, - // insideFill: 'auto', - outsideFill: visualColor - }); - var linePoints = labelLayout.linePoints; - labelLine.setShape({ - points: linePoints - }); - polygon.textGuideLineConfig = { - anchor: linePoints ? new Point(linePoints[0][0], linePoints[0][1]) : null - }; // Make sure update style on labelText after setLabelStyle. - // Because setLabelStyle will replace a new style on it. - - updateProps(labelText, { - style: { - x: labelLayout.x, - y: labelLayout.y - } - }, seriesModel, idx); - labelText.attr({ - rotation: labelLayout.rotation, - originX: labelLayout.x, - originY: labelLayout.y, - z2: 10 - }); - setLabelLineStyle(polygon, getLabelLineStatesModels(itemModel), { - // Default use item visual color - stroke: visualColor - }); - }; - - return FunnelPiece; - }(Polygon); - - var FunnelView = - /** @class */ - function (_super) { - __extends(FunnelView, _super); - - function FunnelView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = FunnelView.type; - _this.ignoreLabelLineUpdate = true; - return _this; - } - - FunnelView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var oldData = this._data; - var group = this.group; - data.diff(oldData).add(function (idx) { - var funnelPiece = new FunnelPiece(data, idx); - data.setItemGraphicEl(idx, funnelPiece); - group.add(funnelPiece); - }).update(function (newIdx, oldIdx) { - var piece = oldData.getItemGraphicEl(oldIdx); - piece.updateData(data, newIdx); - group.add(piece); - data.setItemGraphicEl(newIdx, piece); - }).remove(function (idx) { - var piece = oldData.getItemGraphicEl(idx); - removeElementWithFadeOut(piece, seriesModel, idx); - }).execute(); - this._data = data; - }; - - FunnelView.prototype.remove = function () { - this.group.removeAll(); - this._data = null; - }; - - FunnelView.prototype.dispose = function () {}; - - FunnelView.type = 'funnel'; - return FunnelView; - }(ChartView); - - var FunnelSeriesModel = - /** @class */ - function (_super) { - __extends(FunnelSeriesModel, _super); - - function FunnelSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = FunnelSeriesModel.type; - return _this; - } - - FunnelSeriesModel.prototype.init = function (option) { - _super.prototype.init.apply(this, arguments); // Enable legend selection for each data item - // Use a function instead of direct access because data reference may changed - - - this.legendVisualProvider = new LegendVisualProvider(bind(this.getData, this), bind(this.getRawData, this)); // Extend labelLine emphasis - - this._defaultLabelLine(option); - }; - - FunnelSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesDataSimply(this, { - coordDimensions: ['value'], - encodeDefaulter: curry(makeSeriesEncodeForNameBased, this) - }); - }; - - FunnelSeriesModel.prototype._defaultLabelLine = function (option) { - // Extend labelLine emphasis - defaultEmphasis(option, 'labelLine', ['show']); - var labelLineNormalOpt = option.labelLine; - var labelLineEmphasisOpt = option.emphasis.labelLine; // Not show label line if `label.normal.show = false` - - labelLineNormalOpt.show = labelLineNormalOpt.show && option.label.show; - labelLineEmphasisOpt.show = labelLineEmphasisOpt.show && option.emphasis.label.show; - }; // Overwrite - - - FunnelSeriesModel.prototype.getDataParams = function (dataIndex) { - var data = this.getData(); - - var params = _super.prototype.getDataParams.call(this, dataIndex); - - var valueDim = data.mapDimension('value'); - var sum = data.getSum(valueDim); // Percent is 0 if sum is 0 - - params.percent = !sum ? 0 : +(data.get(valueDim, dataIndex) / sum * 100).toFixed(2); - params.$vars.push('percent'); - return params; - }; - - FunnelSeriesModel.type = 'series.funnel'; - FunnelSeriesModel.defaultOption = { - // zlevel: 0, // 一级层叠 - z: 2, - legendHoverLink: true, - colorBy: 'data', - left: 80, - top: 60, - right: 80, - bottom: 60, - // width: {totalWidth} - left - right, - // height: {totalHeight} - top - bottom, - // 默认取数据最小最大值 - // min: 0, - // max: 100, - minSize: '0%', - maxSize: '100%', - sort: 'descending', - orient: 'vertical', - gap: 0, - funnelAlign: 'center', - label: { - show: true, - position: 'outer' // formatter: 标签文本格式器,同Tooltip.formatter,不支持异步回调 - - }, - labelLine: { - show: true, - length: 20, - lineStyle: { - // color: 各异, - width: 1 - } - }, - itemStyle: { - // color: 各异, - borderColor: '#fff', - borderWidth: 1 - }, - emphasis: { - label: { - show: true - } - }, - select: { - itemStyle: { - borderColor: '#212121' - } - } - }; - return FunnelSeriesModel; - }(SeriesModel); - - function getViewRect$3(seriesModel, api) { - return getLayoutRect(seriesModel.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - } - - function getSortedIndices(data, sort) { - var valueDim = data.mapDimension('value'); - var valueArr = data.mapArray(valueDim, function (val) { - return val; - }); - var indices = []; - var isAscending = sort === 'ascending'; - - for (var i = 0, len = data.count(); i < len; i++) { - indices[i] = i; - } // Add custom sortable function & none sortable opetion by "options.sort" - - - if (isFunction(sort)) { - indices.sort(sort); - } else if (sort !== 'none') { - indices.sort(function (a, b) { - return isAscending ? valueArr[a] - valueArr[b] : valueArr[b] - valueArr[a]; - }); - } - - return indices; - } - - function labelLayout(data) { - var seriesModel = data.hostModel; - var orient = seriesModel.get('orient'); - data.each(function (idx) { - var itemModel = data.getItemModel(idx); - var labelModel = itemModel.getModel('label'); - var labelPosition = labelModel.get('position'); - var labelLineModel = itemModel.getModel('labelLine'); - var layout = data.getItemLayout(idx); - var points = layout.points; - var isLabelInside = labelPosition === 'inner' || labelPosition === 'inside' || labelPosition === 'center' || labelPosition === 'insideLeft' || labelPosition === 'insideRight'; - var textAlign; - var textX; - var textY; - var linePoints; - - if (isLabelInside) { - if (labelPosition === 'insideLeft') { - textX = (points[0][0] + points[3][0]) / 2 + 5; - textY = (points[0][1] + points[3][1]) / 2; - textAlign = 'left'; - } else if (labelPosition === 'insideRight') { - textX = (points[1][0] + points[2][0]) / 2 - 5; - textY = (points[1][1] + points[2][1]) / 2; - textAlign = 'right'; - } else { - textX = (points[0][0] + points[1][0] + points[2][0] + points[3][0]) / 4; - textY = (points[0][1] + points[1][1] + points[2][1] + points[3][1]) / 4; - textAlign = 'center'; - } - - linePoints = [[textX, textY], [textX, textY]]; - } else { - var x1 = void 0; - var y1 = void 0; - var x2 = void 0; - var y2 = void 0; - var labelLineLen = labelLineModel.get('length'); - - if ("development" !== 'production') { - if (orient === 'vertical' && ['top', 'bottom'].indexOf(labelPosition) > -1) { - labelPosition = 'left'; - console.warn('Position error: Funnel chart on vertical orient dose not support top and bottom.'); - } - - if (orient === 'horizontal' && ['left', 'right'].indexOf(labelPosition) > -1) { - labelPosition = 'bottom'; - console.warn('Position error: Funnel chart on horizontal orient dose not support left and right.'); - } - } - - if (labelPosition === 'left') { - // Left side - x1 = (points[3][0] + points[0][0]) / 2; - y1 = (points[3][1] + points[0][1]) / 2; - x2 = x1 - labelLineLen; - textX = x2 - 5; - textAlign = 'right'; - } else if (labelPosition === 'right') { - // Right side - x1 = (points[1][0] + points[2][0]) / 2; - y1 = (points[1][1] + points[2][1]) / 2; - x2 = x1 + labelLineLen; - textX = x2 + 5; - textAlign = 'left'; - } else if (labelPosition === 'top') { - // Top side - x1 = (points[3][0] + points[0][0]) / 2; - y1 = (points[3][1] + points[0][1]) / 2; - y2 = y1 - labelLineLen; - textY = y2 - 5; - textAlign = 'center'; - } else if (labelPosition === 'bottom') { - // Bottom side - x1 = (points[1][0] + points[2][0]) / 2; - y1 = (points[1][1] + points[2][1]) / 2; - y2 = y1 + labelLineLen; - textY = y2 + 5; - textAlign = 'center'; - } else if (labelPosition === 'rightTop') { - // RightTop side - x1 = orient === 'horizontal' ? points[3][0] : points[1][0]; - y1 = orient === 'horizontal' ? points[3][1] : points[1][1]; - - if (orient === 'horizontal') { - y2 = y1 - labelLineLen; - textY = y2 - 5; - textAlign = 'center'; - } else { - x2 = x1 + labelLineLen; - textX = x2 + 5; - textAlign = 'top'; - } - } else if (labelPosition === 'rightBottom') { - // RightBottom side - x1 = points[2][0]; - y1 = points[2][1]; - - if (orient === 'horizontal') { - y2 = y1 + labelLineLen; - textY = y2 + 5; - textAlign = 'center'; - } else { - x2 = x1 + labelLineLen; - textX = x2 + 5; - textAlign = 'bottom'; - } - } else if (labelPosition === 'leftTop') { - // LeftTop side - x1 = points[0][0]; - y1 = orient === 'horizontal' ? points[0][1] : points[1][1]; - - if (orient === 'horizontal') { - y2 = y1 - labelLineLen; - textY = y2 - 5; - textAlign = 'center'; - } else { - x2 = x1 - labelLineLen; - textX = x2 - 5; - textAlign = 'right'; - } - } else if (labelPosition === 'leftBottom') { - // LeftBottom side - x1 = orient === 'horizontal' ? points[1][0] : points[3][0]; - y1 = orient === 'horizontal' ? points[1][1] : points[2][1]; - - if (orient === 'horizontal') { - y2 = y1 + labelLineLen; - textY = y2 + 5; - textAlign = 'center'; - } else { - x2 = x1 - labelLineLen; - textX = x2 - 5; - textAlign = 'right'; - } - } else { - // Right side or Bottom side - x1 = (points[1][0] + points[2][0]) / 2; - y1 = (points[1][1] + points[2][1]) / 2; - - if (orient === 'horizontal') { - y2 = y1 + labelLineLen; - textY = y2 + 5; - textAlign = 'center'; - } else { - x2 = x1 + labelLineLen; - textX = x2 + 5; - textAlign = 'left'; - } - } - - if (orient === 'horizontal') { - x2 = x1; - textX = x2; - } else { - y2 = y1; - textY = y2; - } - - linePoints = [[x1, y1], [x2, y2]]; - } - - layout.label = { - linePoints: linePoints, - x: textX, - y: textY, - verticalAlign: 'middle', - textAlign: textAlign, - inside: isLabelInside - }; - }); - } - - function funnelLayout(ecModel, api) { - ecModel.eachSeriesByType('funnel', function (seriesModel) { - var data = seriesModel.getData(); - var valueDim = data.mapDimension('value'); - var sort = seriesModel.get('sort'); - var viewRect = getViewRect$3(seriesModel, api); - var orient = seriesModel.get('orient'); - var viewWidth = viewRect.width; - var viewHeight = viewRect.height; - var indices = getSortedIndices(data, sort); - var x = viewRect.x; - var y = viewRect.y; - var sizeExtent = orient === 'horizontal' ? [parsePercent$1(seriesModel.get('minSize'), viewHeight), parsePercent$1(seriesModel.get('maxSize'), viewHeight)] : [parsePercent$1(seriesModel.get('minSize'), viewWidth), parsePercent$1(seriesModel.get('maxSize'), viewWidth)]; - var dataExtent = data.getDataExtent(valueDim); - var min = seriesModel.get('min'); - var max = seriesModel.get('max'); - - if (min == null) { - min = Math.min(dataExtent[0], 0); - } - - if (max == null) { - max = dataExtent[1]; - } - - var funnelAlign = seriesModel.get('funnelAlign'); - var gap = seriesModel.get('gap'); - var viewSize = orient === 'horizontal' ? viewWidth : viewHeight; - var itemSize = (viewSize - gap * (data.count() - 1)) / data.count(); - - var getLinePoints = function (idx, offset) { - // End point index is data.count() and we assign it 0 - if (orient === 'horizontal') { - var val_1 = data.get(valueDim, idx) || 0; - var itemHeight = linearMap(val_1, [min, max], sizeExtent, true); - var y0 = void 0; - - switch (funnelAlign) { - case 'top': - y0 = y; - break; - - case 'center': - y0 = y + (viewHeight - itemHeight) / 2; - break; - - case 'bottom': - y0 = y + (viewHeight - itemHeight); - break; - } - - return [[offset, y0], [offset, y0 + itemHeight]]; - } - - var val = data.get(valueDim, idx) || 0; - var itemWidth = linearMap(val, [min, max], sizeExtent, true); - var x0; - - switch (funnelAlign) { - case 'left': - x0 = x; - break; - - case 'center': - x0 = x + (viewWidth - itemWidth) / 2; - break; - - case 'right': - x0 = x + viewWidth - itemWidth; - break; - } - - return [[x0, offset], [x0 + itemWidth, offset]]; - }; - - if (sort === 'ascending') { - // From bottom to top - itemSize = -itemSize; - gap = -gap; - - if (orient === 'horizontal') { - x += viewWidth; - } else { - y += viewHeight; - } - - indices = indices.reverse(); - } - - for (var i = 0; i < indices.length; i++) { - var idx = indices[i]; - var nextIdx = indices[i + 1]; - var itemModel = data.getItemModel(idx); - - if (orient === 'horizontal') { - var width = itemModel.get(['itemStyle', 'width']); - - if (width == null) { - width = itemSize; - } else { - width = parsePercent$1(width, viewWidth); - - if (sort === 'ascending') { - width = -width; - } - } - - var start = getLinePoints(idx, x); - var end = getLinePoints(nextIdx, x + width); - x += width + gap; - data.setItemLayout(idx, { - points: start.concat(end.slice().reverse()) - }); - } else { - var height = itemModel.get(['itemStyle', 'height']); - - if (height == null) { - height = itemSize; - } else { - height = parsePercent$1(height, viewHeight); - - if (sort === 'ascending') { - height = -height; - } - } - - var start = getLinePoints(idx, y); - var end = getLinePoints(nextIdx, y + height); - y += height + gap; - data.setItemLayout(idx, { - points: start.concat(end.slice().reverse()) - }); - } - } - - labelLayout(data); - }); - } - - function install$f(registers) { - registers.registerChartView(FunnelView); - registers.registerSeriesModel(FunnelSeriesModel); - registers.registerLayout(funnelLayout); - registers.registerProcessor(dataFilter('funnel')); - } - - var DEFAULT_SMOOTH = 0.3; - - var ParallelView = - /** @class */ - function (_super) { - __extends(ParallelView, _super); - - function ParallelView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ParallelView.type; - _this._dataGroup = new Group(); - _this._initialized = false; - return _this; - } - - ParallelView.prototype.init = function () { - this.group.add(this._dataGroup); - }; - /** - * @override - */ - - - ParallelView.prototype.render = function (seriesModel, ecModel, api, payload) { - // Clear previously rendered progressive elements. - this._progressiveEls = null; - var dataGroup = this._dataGroup; - var data = seriesModel.getData(); - var oldData = this._data; - var coordSys = seriesModel.coordinateSystem; - var dimensions = coordSys.dimensions; - var seriesScope = makeSeriesScope$2(seriesModel); - data.diff(oldData).add(add).update(update).remove(remove).execute(); - - function add(newDataIndex) { - var line = addEl(data, dataGroup, newDataIndex, dimensions, coordSys); - updateElCommon(line, data, newDataIndex, seriesScope); - } - - function update(newDataIndex, oldDataIndex) { - var line = oldData.getItemGraphicEl(oldDataIndex); - var points = createLinePoints(data, newDataIndex, dimensions, coordSys); - data.setItemGraphicEl(newDataIndex, line); - updateProps(line, { - shape: { - points: points - } - }, seriesModel, newDataIndex); - saveOldStyle(line); - updateElCommon(line, data, newDataIndex, seriesScope); - } - - function remove(oldDataIndex) { - var line = oldData.getItemGraphicEl(oldDataIndex); - dataGroup.remove(line); - } // First create - - - if (!this._initialized) { - this._initialized = true; - var clipPath = createGridClipShape(coordSys, seriesModel, function () { - // Callback will be invoked immediately if there is no animation - setTimeout(function () { - dataGroup.removeClipPath(); - }); - }); - dataGroup.setClipPath(clipPath); - } - - this._data = data; - }; - - ParallelView.prototype.incrementalPrepareRender = function (seriesModel, ecModel, api) { - this._initialized = true; - this._data = null; - - this._dataGroup.removeAll(); - }; - - ParallelView.prototype.incrementalRender = function (taskParams, seriesModel, ecModel) { - var data = seriesModel.getData(); - var coordSys = seriesModel.coordinateSystem; - var dimensions = coordSys.dimensions; - var seriesScope = makeSeriesScope$2(seriesModel); - var progressiveEls = this._progressiveEls = []; - - for (var dataIndex = taskParams.start; dataIndex < taskParams.end; dataIndex++) { - var line = addEl(data, this._dataGroup, dataIndex, dimensions, coordSys); - line.incremental = true; - updateElCommon(line, data, dataIndex, seriesScope); - progressiveEls.push(line); - } - }; - - ParallelView.prototype.remove = function () { - this._dataGroup && this._dataGroup.removeAll(); - this._data = null; - }; - - ParallelView.type = 'parallel'; - return ParallelView; - }(ChartView); - - function createGridClipShape(coordSys, seriesModel, cb) { - var parallelModel = coordSys.model; - var rect = coordSys.getRect(); - var rectEl = new Rect({ - shape: { - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height - } - }); - var dim = parallelModel.get('layout') === 'horizontal' ? 'width' : 'height'; - rectEl.setShape(dim, 0); - initProps(rectEl, { - shape: { - width: rect.width, - height: rect.height - } - }, seriesModel, cb); - return rectEl; - } - - function createLinePoints(data, dataIndex, dimensions, coordSys) { - var points = []; - - for (var i = 0; i < dimensions.length; i++) { - var dimName = dimensions[i]; - var value = data.get(data.mapDimension(dimName), dataIndex); - - if (!isEmptyValue(value, coordSys.getAxis(dimName).type)) { - points.push(coordSys.dataToPoint(value, dimName)); - } - } - - return points; - } - - function addEl(data, dataGroup, dataIndex, dimensions, coordSys) { - var points = createLinePoints(data, dataIndex, dimensions, coordSys); - var line = new Polyline({ - shape: { - points: points - }, - // silent: true, - z2: 10 - }); - dataGroup.add(line); - data.setItemGraphicEl(dataIndex, line); - return line; - } - - function makeSeriesScope$2(seriesModel) { - var smooth = seriesModel.get('smooth', true); - smooth === true && (smooth = DEFAULT_SMOOTH); - smooth = numericToNumber(smooth); - eqNaN(smooth) && (smooth = 0); - return { - smooth: smooth - }; - } - - function updateElCommon(el, data, dataIndex, seriesScope) { - el.useStyle(data.getItemVisual(dataIndex, 'style')); - el.style.fill = null; - el.setShape('smooth', seriesScope.smooth); - var itemModel = data.getItemModel(dataIndex); - var emphasisModel = itemModel.getModel('emphasis'); - setStatesStylesFromModel(el, itemModel, 'lineStyle'); - toggleHoverEmphasis(el, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - } // function simpleDiff(oldData, newData, dimensions) { - // let oldLen; - // if (!oldData - // || !oldData.__plProgressive - // || (oldLen = oldData.count()) !== newData.count() - // ) { - // return true; - // } - // let dimLen = dimensions.length; - // for (let i = 0; i < oldLen; i++) { - // for (let j = 0; j < dimLen; j++) { - // if (oldData.get(dimensions[j], i) !== newData.get(dimensions[j], i)) { - // return true; - // } - // } - // } - // return false; - // } - // FIXME put in common util? - - - function isEmptyValue(val, axisType) { - return axisType === 'category' ? val == null : val == null || isNaN(val); // axisType === 'value' - } - - var ParallelSeriesModel = - /** @class */ - function (_super) { - __extends(ParallelSeriesModel, _super); - - function ParallelSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ParallelSeriesModel.type; - _this.visualStyleAccessPath = 'lineStyle'; - _this.visualDrawType = 'stroke'; - return _this; - } - - ParallelSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesData(null, this, { - useEncodeDefaulter: bind(makeDefaultEncode, null, this) - }); - }; - /** - * User can get data raw indices on 'axisAreaSelected' event received. - * - * @return Raw indices - */ - - - ParallelSeriesModel.prototype.getRawIndicesByActiveState = function (activeState) { - var coordSys = this.coordinateSystem; - var data = this.getData(); - var indices = []; - coordSys.eachActiveState(data, function (theActiveState, dataIndex) { - if (activeState === theActiveState) { - indices.push(data.getRawIndex(dataIndex)); - } - }); - return indices; - }; - - ParallelSeriesModel.type = 'series.parallel'; - ParallelSeriesModel.dependencies = ['parallel']; - ParallelSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'parallel', - parallelIndex: 0, - label: { - show: false - }, - inactiveOpacity: 0.05, - activeOpacity: 1, - lineStyle: { - width: 1, - opacity: 0.45, - type: 'solid' - }, - emphasis: { - label: { - show: false - } - }, - progressive: 500, - smooth: false, - animationEasing: 'linear' - }; - return ParallelSeriesModel; - }(SeriesModel); - - function makeDefaultEncode(seriesModel) { - // The mapping of parallelAxis dimension to data dimension can - // be specified in parallelAxis.option.dim. For example, if - // parallelAxis.option.dim is 'dim3', it mapping to the third - // dimension of data. But `data.encode` has higher priority. - // Moreover, parallelModel.dimension should not be regarded as data - // dimensions. Consider dimensions = ['dim4', 'dim2', 'dim6']; - var parallelModel = seriesModel.ecModel.getComponent('parallel', seriesModel.get('parallelIndex')); - - if (!parallelModel) { - return; - } - - var encodeDefine = {}; - each(parallelModel.dimensions, function (axisDim) { - var dataDimIndex = convertDimNameToNumber(axisDim); - encodeDefine[axisDim] = dataDimIndex; - }); - return encodeDefine; - } - - function convertDimNameToNumber(dimName) { - return +dimName.replace('dim', ''); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var opacityAccessPath$1 = ['lineStyle', 'opacity']; - var parallelVisual = { - seriesType: 'parallel', - reset: function (seriesModel, ecModel) { - var coordSys = seriesModel.coordinateSystem; - var opacityMap = { - normal: seriesModel.get(['lineStyle', 'opacity']), - active: seriesModel.get('activeOpacity'), - inactive: seriesModel.get('inactiveOpacity') - }; - return { - progress: function (params, data) { - coordSys.eachActiveState(data, function (activeState, dataIndex) { - var opacity = opacityMap[activeState]; - - if (activeState === 'normal' && data.hasItemOption) { - var itemOpacity = data.getItemModel(dataIndex).get(opacityAccessPath$1, true); - itemOpacity != null && (opacity = itemOpacity); - } - - var existsStyle = data.ensureUniqueItemVisual(dataIndex, 'style'); - existsStyle.opacity = opacity; - }, params.start, params.end); - } - }; - } - }; - - function parallelPreprocessor(option) { - createParallelIfNeeded(option); - mergeAxisOptionFromParallel(option); - } - /** - * Create a parallel coordinate if not exists. - * @inner - */ - - function createParallelIfNeeded(option) { - if (option.parallel) { - return; - } - - var hasParallelSeries = false; - each(option.series, function (seriesOpt) { - if (seriesOpt && seriesOpt.type === 'parallel') { - hasParallelSeries = true; - } - }); - - if (hasParallelSeries) { - option.parallel = [{}]; - } - } - /** - * Merge aixs definition from parallel option (if exists) to axis option. - * @inner - */ - - - function mergeAxisOptionFromParallel(option) { - var axes = normalizeToArray(option.parallelAxis); - each(axes, function (axisOption) { - if (!isObject(axisOption)) { - return; - } - - var parallelIndex = axisOption.parallelIndex || 0; - var parallelOption = normalizeToArray(option.parallel)[parallelIndex]; - - if (parallelOption && parallelOption.parallelAxisDefault) { - merge(axisOption, parallelOption.parallelAxisDefault, false); - } - }); - } - - var CLICK_THRESHOLD = 5; // > 4 - - var ParallelView$1 = - /** @class */ - function (_super) { - __extends(ParallelView, _super); - - function ParallelView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ParallelView.type; - return _this; - } - - ParallelView.prototype.render = function (parallelModel, ecModel, api) { - this._model = parallelModel; - this._api = api; - - if (!this._handlers) { - this._handlers = {}; - each(handlers, function (handler, eventName) { - api.getZr().on(eventName, this._handlers[eventName] = bind(handler, this)); - }, this); - } - - createOrUpdate(this, '_throttledDispatchExpand', parallelModel.get('axisExpandRate'), 'fixRate'); - }; - - ParallelView.prototype.dispose = function (ecModel, api) { - clear(this, '_throttledDispatchExpand'); - each(this._handlers, function (handler, eventName) { - api.getZr().off(eventName, handler); - }); - this._handlers = null; - }; - /** - * @internal - * @param {Object} [opt] If null, cancel the last action triggering for debounce. - */ - - - ParallelView.prototype._throttledDispatchExpand = function (opt) { - this._dispatchExpand(opt); - }; - /** - * @internal - */ - - - ParallelView.prototype._dispatchExpand = function (opt) { - opt && this._api.dispatchAction(extend({ - type: 'parallelAxisExpand' - }, opt)); - }; - - ParallelView.type = 'parallel'; - return ParallelView; - }(ComponentView); - - var handlers = { - mousedown: function (e) { - if (checkTrigger(this, 'click')) { - this._mouseDownPoint = [e.offsetX, e.offsetY]; - } - }, - mouseup: function (e) { - var mouseDownPoint = this._mouseDownPoint; - - if (checkTrigger(this, 'click') && mouseDownPoint) { - var point = [e.offsetX, e.offsetY]; - var dist = Math.pow(mouseDownPoint[0] - point[0], 2) + Math.pow(mouseDownPoint[1] - point[1], 2); - - if (dist > CLICK_THRESHOLD) { - return; - } - - var result = this._model.coordinateSystem.getSlidedAxisExpandWindow([e.offsetX, e.offsetY]); - - result.behavior !== 'none' && this._dispatchExpand({ - axisExpandWindow: result.axisExpandWindow - }); - } - - this._mouseDownPoint = null; - }, - mousemove: function (e) { - // Should do nothing when brushing. - if (this._mouseDownPoint || !checkTrigger(this, 'mousemove')) { - return; - } - - var model = this._model; - var result = model.coordinateSystem.getSlidedAxisExpandWindow([e.offsetX, e.offsetY]); - var behavior = result.behavior; - behavior === 'jump' && this._throttledDispatchExpand.debounceNextCall(model.get('axisExpandDebounce')); - - this._throttledDispatchExpand(behavior === 'none' ? null // Cancel the last trigger, in case that mouse slide out of the area quickly. - : { - axisExpandWindow: result.axisExpandWindow, - // Jumping uses animation, and sliding suppresses animation. - animation: behavior === 'jump' ? null : { - duration: 0 // Disable animation. - - } - }); - } - }; - - function checkTrigger(view, triggerOn) { - var model = view._model; - return model.get('axisExpandable') && model.get('axisExpandTriggerOn') === triggerOn; - } - - var ParallelModel = - /** @class */ - function (_super) { - __extends(ParallelModel, _super); - - function ParallelModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ParallelModel.type; - return _this; - } - - ParallelModel.prototype.init = function () { - _super.prototype.init.apply(this, arguments); - - this.mergeOption({}); - }; - - ParallelModel.prototype.mergeOption = function (newOption) { - var thisOption = this.option; - newOption && merge(thisOption, newOption, true); - - this._initDimensions(); - }; - /** - * Whether series or axis is in this coordinate system. - */ - - - ParallelModel.prototype.contains = function (model, ecModel) { - var parallelIndex = model.get('parallelIndex'); - return parallelIndex != null && ecModel.getComponent('parallel', parallelIndex) === this; - }; - - ParallelModel.prototype.setAxisExpand = function (opt) { - each(['axisExpandable', 'axisExpandCenter', 'axisExpandCount', 'axisExpandWidth', 'axisExpandWindow'], function (name) { - if (opt.hasOwnProperty(name)) { - // @ts-ignore FIXME: why "never" inferred in this.option[name]? - this.option[name] = opt[name]; - } - }, this); - }; - - ParallelModel.prototype._initDimensions = function () { - var dimensions = this.dimensions = []; - var parallelAxisIndex = this.parallelAxisIndex = []; - var axisModels = filter(this.ecModel.queryComponents({ - mainType: 'parallelAxis' - }), function (axisModel) { - // Can not use this.contains here, because - // initialization has not been completed yet. - return (axisModel.get('parallelIndex') || 0) === this.componentIndex; - }, this); - each(axisModels, function (axisModel) { - dimensions.push('dim' + axisModel.get('dim')); - parallelAxisIndex.push(axisModel.componentIndex); - }); - }; - - ParallelModel.type = 'parallel'; - ParallelModel.dependencies = ['parallelAxis']; - ParallelModel.layoutMode = 'box'; - ParallelModel.defaultOption = { - // zlevel: 0, - z: 0, - left: 80, - top: 60, - right: 80, - bottom: 60, - // width: {totalWidth} - left - right, - // height: {totalHeight} - top - bottom, - layout: 'horizontal', - // FIXME - // naming? - axisExpandable: false, - axisExpandCenter: null, - axisExpandCount: 0, - axisExpandWidth: 50, - axisExpandRate: 17, - axisExpandDebounce: 50, - // [out, in, jumpTarget]. In percentage. If use [null, 0.05], null means full. - // Do not doc to user until necessary. - axisExpandSlideTriggerArea: [-0.15, 0.05, 0.4], - axisExpandTriggerOn: 'click', - parallelAxisDefault: null - }; - return ParallelModel; - }(ComponentModel); - - var ParallelAxis = - /** @class */ - function (_super) { - __extends(ParallelAxis, _super); - - function ParallelAxis(dim, scale, coordExtent, axisType, axisIndex) { - var _this = _super.call(this, dim, scale, coordExtent) || this; - - _this.type = axisType || 'value'; - _this.axisIndex = axisIndex; - return _this; - } - - ParallelAxis.prototype.isHorizontal = function () { - return this.coordinateSystem.getModel().get('layout') !== 'horizontal'; - }; - - return ParallelAxis; - }(Axis); - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - /** - * Calculate slider move result. - * Usage: - * (1) If both handle0 and handle1 are needed to be moved, set minSpan the same as - * maxSpan and the same as `Math.abs(handleEnd[1] - handleEnds[0])`. - * (2) If handle0 is forbidden to cross handle1, set minSpan as `0`. - * - * @param delta Move length. - * @param handleEnds handleEnds[0] can be bigger then handleEnds[1]. - * handleEnds will be modified in this method. - * @param extent handleEnds is restricted by extent. - * extent[0] should less or equals than extent[1]. - * @param handleIndex Can be 'all', means that both move the two handleEnds. - * @param minSpan The range of dataZoom can not be smaller than that. - * If not set, handle0 and cross handle1. If set as a non-negative - * number (including `0`), handles will push each other when reaching - * the minSpan. - * @param maxSpan The range of dataZoom can not be larger than that. - * @return The input handleEnds. - */ - function sliderMove(delta, handleEnds, extent, handleIndex, minSpan, maxSpan) { - delta = delta || 0; - var extentSpan = extent[1] - extent[0]; // Notice maxSpan and minSpan can be null/undefined. - - if (minSpan != null) { - minSpan = restrict(minSpan, [0, extentSpan]); - } - - if (maxSpan != null) { - maxSpan = Math.max(maxSpan, minSpan != null ? minSpan : 0); - } - - if (handleIndex === 'all') { - var handleSpan = Math.abs(handleEnds[1] - handleEnds[0]); - handleSpan = restrict(handleSpan, [0, extentSpan]); - minSpan = maxSpan = restrict(handleSpan, [minSpan, maxSpan]); - handleIndex = 0; - } - - handleEnds[0] = restrict(handleEnds[0], extent); - handleEnds[1] = restrict(handleEnds[1], extent); - var originalDistSign = getSpanSign(handleEnds, handleIndex); - handleEnds[handleIndex] += delta; // Restrict in extent. - - var extentMinSpan = minSpan || 0; - var realExtent = extent.slice(); - originalDistSign.sign < 0 ? realExtent[0] += extentMinSpan : realExtent[1] -= extentMinSpan; - handleEnds[handleIndex] = restrict(handleEnds[handleIndex], realExtent); // Expand span. - - var currDistSign; - currDistSign = getSpanSign(handleEnds, handleIndex); - - if (minSpan != null && (currDistSign.sign !== originalDistSign.sign || currDistSign.span < minSpan)) { - // If minSpan exists, 'cross' is forbidden. - handleEnds[1 - handleIndex] = handleEnds[handleIndex] + originalDistSign.sign * minSpan; - } // Shrink span. - - - currDistSign = getSpanSign(handleEnds, handleIndex); - - if (maxSpan != null && currDistSign.span > maxSpan) { - handleEnds[1 - handleIndex] = handleEnds[handleIndex] + currDistSign.sign * maxSpan; - } - - return handleEnds; - } - - function getSpanSign(handleEnds, handleIndex) { - var dist = handleEnds[handleIndex] - handleEnds[1 - handleIndex]; // If `handleEnds[0] === handleEnds[1]`, always believe that handleEnd[0] - // is at left of handleEnds[1] for non-cross case. - - return { - span: Math.abs(dist), - sign: dist > 0 ? -1 : dist < 0 ? 1 : handleIndex ? -1 : 1 - }; - } - - function restrict(value, extend) { - return Math.min(extend[1] != null ? extend[1] : Infinity, Math.max(extend[0] != null ? extend[0] : -Infinity, value)); - } - - var each$5 = each; - var mathMin$8 = Math.min; - var mathMax$8 = Math.max; - var mathFloor$1 = Math.floor; - var mathCeil$1 = Math.ceil; - var round$3 = round; - var PI$7 = Math.PI; - - var Parallel = - /** @class */ - function () { - function Parallel(parallelModel, ecModel, api) { - this.type = 'parallel'; - /** - * key: dimension - */ - - this._axesMap = createHashMap(); - /** - * key: dimension - * value: {position: [], rotation, } - */ - - this._axesLayout = {}; - this.dimensions = parallelModel.dimensions; - this._model = parallelModel; - - this._init(parallelModel, ecModel, api); - } - - Parallel.prototype._init = function (parallelModel, ecModel, api) { - var dimensions = parallelModel.dimensions; - var parallelAxisIndex = parallelModel.parallelAxisIndex; - each$5(dimensions, function (dim, idx) { - var axisIndex = parallelAxisIndex[idx]; - var axisModel = ecModel.getComponent('parallelAxis', axisIndex); - - var axis = this._axesMap.set(dim, new ParallelAxis(dim, createScaleByModel(axisModel), [0, 0], axisModel.get('type'), axisIndex)); - - var isCategory = axis.type === 'category'; - axis.onBand = isCategory && axisModel.get('boundaryGap'); - axis.inverse = axisModel.get('inverse'); // Injection - - axisModel.axis = axis; - axis.model = axisModel; - axis.coordinateSystem = axisModel.coordinateSystem = this; - }, this); - }; - /** - * Update axis scale after data processed - */ - - - Parallel.prototype.update = function (ecModel, api) { - this._updateAxesFromSeries(this._model, ecModel); - }; - - Parallel.prototype.containPoint = function (point) { - var layoutInfo = this._makeLayoutInfo(); - - var axisBase = layoutInfo.axisBase; - var layoutBase = layoutInfo.layoutBase; - var pixelDimIndex = layoutInfo.pixelDimIndex; - var pAxis = point[1 - pixelDimIndex]; - var pLayout = point[pixelDimIndex]; - return pAxis >= axisBase && pAxis <= axisBase + layoutInfo.axisLength && pLayout >= layoutBase && pLayout <= layoutBase + layoutInfo.layoutLength; - }; - - Parallel.prototype.getModel = function () { - return this._model; - }; - /** - * Update properties from series - */ - - - Parallel.prototype._updateAxesFromSeries = function (parallelModel, ecModel) { - ecModel.eachSeries(function (seriesModel) { - if (!parallelModel.contains(seriesModel, ecModel)) { - return; - } - - var data = seriesModel.getData(); - each$5(this.dimensions, function (dim) { - var axis = this._axesMap.get(dim); - - axis.scale.unionExtentFromData(data, data.mapDimension(dim)); - niceScaleExtent(axis.scale, axis.model); - }, this); - }, this); - }; - /** - * Resize the parallel coordinate system. - */ - - - Parallel.prototype.resize = function (parallelModel, api) { - this._rect = getLayoutRect(parallelModel.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - - this._layoutAxes(); - }; - - Parallel.prototype.getRect = function () { - return this._rect; - }; - - Parallel.prototype._makeLayoutInfo = function () { - var parallelModel = this._model; - var rect = this._rect; - var xy = ['x', 'y']; - var wh = ['width', 'height']; - var layout = parallelModel.get('layout'); - var pixelDimIndex = layout === 'horizontal' ? 0 : 1; - var layoutLength = rect[wh[pixelDimIndex]]; - var layoutExtent = [0, layoutLength]; - var axisCount = this.dimensions.length; - var axisExpandWidth = restrict$1(parallelModel.get('axisExpandWidth'), layoutExtent); - var axisExpandCount = restrict$1(parallelModel.get('axisExpandCount') || 0, [0, axisCount]); - var axisExpandable = parallelModel.get('axisExpandable') && axisCount > 3 && axisCount > axisExpandCount && axisExpandCount > 1 && axisExpandWidth > 0 && layoutLength > 0; // `axisExpandWindow` is According to the coordinates of [0, axisExpandLength], - // for sake of consider the case that axisCollapseWidth is 0 (when screen is narrow), - // where collapsed axes should be overlapped. - - var axisExpandWindow = parallelModel.get('axisExpandWindow'); - var winSize; - - if (!axisExpandWindow) { - winSize = restrict$1(axisExpandWidth * (axisExpandCount - 1), layoutExtent); - var axisExpandCenter = parallelModel.get('axisExpandCenter') || mathFloor$1(axisCount / 2); - axisExpandWindow = [axisExpandWidth * axisExpandCenter - winSize / 2]; - axisExpandWindow[1] = axisExpandWindow[0] + winSize; - } else { - winSize = restrict$1(axisExpandWindow[1] - axisExpandWindow[0], layoutExtent); - axisExpandWindow[1] = axisExpandWindow[0] + winSize; - } - - var axisCollapseWidth = (layoutLength - winSize) / (axisCount - axisExpandCount); // Avoid axisCollapseWidth is too small. - - axisCollapseWidth < 3 && (axisCollapseWidth = 0); // Find the first and last indices > ewin[0] and < ewin[1]. - - var winInnerIndices = [mathFloor$1(round$3(axisExpandWindow[0] / axisExpandWidth, 1)) + 1, mathCeil$1(round$3(axisExpandWindow[1] / axisExpandWidth, 1)) - 1]; // Pos in ec coordinates. - - var axisExpandWindow0Pos = axisCollapseWidth / axisExpandWidth * axisExpandWindow[0]; - return { - layout: layout, - pixelDimIndex: pixelDimIndex, - layoutBase: rect[xy[pixelDimIndex]], - layoutLength: layoutLength, - axisBase: rect[xy[1 - pixelDimIndex]], - axisLength: rect[wh[1 - pixelDimIndex]], - axisExpandable: axisExpandable, - axisExpandWidth: axisExpandWidth, - axisCollapseWidth: axisCollapseWidth, - axisExpandWindow: axisExpandWindow, - axisCount: axisCount, - winInnerIndices: winInnerIndices, - axisExpandWindow0Pos: axisExpandWindow0Pos - }; - }; - - Parallel.prototype._layoutAxes = function () { - var rect = this._rect; - var axes = this._axesMap; - var dimensions = this.dimensions; - - var layoutInfo = this._makeLayoutInfo(); - - var layout = layoutInfo.layout; - axes.each(function (axis) { - var axisExtent = [0, layoutInfo.axisLength]; - var idx = axis.inverse ? 1 : 0; - axis.setExtent(axisExtent[idx], axisExtent[1 - idx]); - }); - each$5(dimensions, function (dim, idx) { - var posInfo = (layoutInfo.axisExpandable ? layoutAxisWithExpand : layoutAxisWithoutExpand)(idx, layoutInfo); - var positionTable = { - horizontal: { - x: posInfo.position, - y: layoutInfo.axisLength - }, - vertical: { - x: 0, - y: posInfo.position - } - }; - var rotationTable = { - horizontal: PI$7 / 2, - vertical: 0 - }; - var position = [positionTable[layout].x + rect.x, positionTable[layout].y + rect.y]; - var rotation = rotationTable[layout]; - var transform = create$1(); - rotate(transform, transform, rotation); - translate(transform, transform, position); // TODO - // tick layout info - // TODO - // update dimensions info based on axis order. - - this._axesLayout[dim] = { - position: position, - rotation: rotation, - transform: transform, - axisNameAvailableWidth: posInfo.axisNameAvailableWidth, - axisLabelShow: posInfo.axisLabelShow, - nameTruncateMaxWidth: posInfo.nameTruncateMaxWidth, - tickDirection: 1, - labelDirection: 1 - }; - }, this); - }; - /** - * Get axis by dim. - */ - - - Parallel.prototype.getAxis = function (dim) { - return this._axesMap.get(dim); - }; - /** - * Convert a dim value of a single item of series data to Point. - */ - - - Parallel.prototype.dataToPoint = function (value, dim) { - return this.axisCoordToPoint(this._axesMap.get(dim).dataToCoord(value), dim); - }; - /** - * Travel data for one time, get activeState of each data item. - * @param start the start dataIndex that travel from. - * @param end the next dataIndex of the last dataIndex will be travel. - */ - - - Parallel.prototype.eachActiveState = function (data, callback, start, end) { - start == null && (start = 0); - end == null && (end = data.count()); - var axesMap = this._axesMap; - var dimensions = this.dimensions; - var dataDimensions = []; - var axisModels = []; - each(dimensions, function (axisDim) { - dataDimensions.push(data.mapDimension(axisDim)); - axisModels.push(axesMap.get(axisDim).model); - }); - var hasActiveSet = this.hasAxisBrushed(); - - for (var dataIndex = start; dataIndex < end; dataIndex++) { - var activeState = void 0; - - if (!hasActiveSet) { - activeState = 'normal'; - } else { - activeState = 'active'; - var values = data.getValues(dataDimensions, dataIndex); - - for (var j = 0, lenj = dimensions.length; j < lenj; j++) { - var state = axisModels[j].getActiveState(values[j]); - - if (state === 'inactive') { - activeState = 'inactive'; - break; - } - } - } - - callback(activeState, dataIndex); - } - }; - /** - * Whether has any activeSet. - */ - - - Parallel.prototype.hasAxisBrushed = function () { - var dimensions = this.dimensions; - var axesMap = this._axesMap; - var hasActiveSet = false; - - for (var j = 0, lenj = dimensions.length; j < lenj; j++) { - if (axesMap.get(dimensions[j]).model.getActiveState() !== 'normal') { - hasActiveSet = true; - } - } - - return hasActiveSet; - }; - /** - * Convert coords of each axis to Point. - * Return point. For example: [10, 20] - */ - - - Parallel.prototype.axisCoordToPoint = function (coord, dim) { - var axisLayout = this._axesLayout[dim]; - return applyTransform$1([coord, 0], axisLayout.transform); - }; - /** - * Get axis layout. - */ - - - Parallel.prototype.getAxisLayout = function (dim) { - return clone(this._axesLayout[dim]); - }; - /** - * @return {Object} {axisExpandWindow, delta, behavior: 'jump' | 'slide' | 'none'}. - */ - - - Parallel.prototype.getSlidedAxisExpandWindow = function (point) { - var layoutInfo = this._makeLayoutInfo(); - - var pixelDimIndex = layoutInfo.pixelDimIndex; - var axisExpandWindow = layoutInfo.axisExpandWindow.slice(); - var winSize = axisExpandWindow[1] - axisExpandWindow[0]; - var extent = [0, layoutInfo.axisExpandWidth * (layoutInfo.axisCount - 1)]; // Out of the area of coordinate system. - - if (!this.containPoint(point)) { - return { - behavior: 'none', - axisExpandWindow: axisExpandWindow - }; - } // Convert the point from global to expand coordinates. - - - var pointCoord = point[pixelDimIndex] - layoutInfo.layoutBase - layoutInfo.axisExpandWindow0Pos; // For dragging operation convenience, the window should not be - // slided when mouse is the center area of the window. - - var delta; - var behavior = 'slide'; - var axisCollapseWidth = layoutInfo.axisCollapseWidth; - - var triggerArea = this._model.get('axisExpandSlideTriggerArea'); // But consider touch device, jump is necessary. - - - var useJump = triggerArea[0] != null; - - if (axisCollapseWidth) { - if (useJump && axisCollapseWidth && pointCoord < winSize * triggerArea[0]) { - behavior = 'jump'; - delta = pointCoord - winSize * triggerArea[2]; - } else if (useJump && axisCollapseWidth && pointCoord > winSize * (1 - triggerArea[0])) { - behavior = 'jump'; - delta = pointCoord - winSize * (1 - triggerArea[2]); - } else { - (delta = pointCoord - winSize * triggerArea[1]) >= 0 && (delta = pointCoord - winSize * (1 - triggerArea[1])) <= 0 && (delta = 0); - } - - delta *= layoutInfo.axisExpandWidth / axisCollapseWidth; - delta ? sliderMove(delta, axisExpandWindow, extent, 'all') // Avoid nonsense triger on mousemove. - : behavior = 'none'; - } // When screen is too narrow, make it visible and slidable, although it is hard to interact. - else { - var winSize2 = axisExpandWindow[1] - axisExpandWindow[0]; - var pos = extent[1] * pointCoord / winSize2; - axisExpandWindow = [mathMax$8(0, pos - winSize2 / 2)]; - axisExpandWindow[1] = mathMin$8(extent[1], axisExpandWindow[0] + winSize2); - axisExpandWindow[0] = axisExpandWindow[1] - winSize2; - } - - return { - axisExpandWindow: axisExpandWindow, - behavior: behavior - }; - }; - - return Parallel; - }(); - - function restrict$1(len, extent) { - return mathMin$8(mathMax$8(len, extent[0]), extent[1]); - } - - function layoutAxisWithoutExpand(axisIndex, layoutInfo) { - var step = layoutInfo.layoutLength / (layoutInfo.axisCount - 1); - return { - position: step * axisIndex, - axisNameAvailableWidth: step, - axisLabelShow: true - }; - } - - function layoutAxisWithExpand(axisIndex, layoutInfo) { - var layoutLength = layoutInfo.layoutLength; - var axisExpandWidth = layoutInfo.axisExpandWidth; - var axisCount = layoutInfo.axisCount; - var axisCollapseWidth = layoutInfo.axisCollapseWidth; - var winInnerIndices = layoutInfo.winInnerIndices; - var position; - var axisNameAvailableWidth = axisCollapseWidth; - var axisLabelShow = false; - var nameTruncateMaxWidth; - - if (axisIndex < winInnerIndices[0]) { - position = axisIndex * axisCollapseWidth; - nameTruncateMaxWidth = axisCollapseWidth; - } else if (axisIndex <= winInnerIndices[1]) { - position = layoutInfo.axisExpandWindow0Pos + axisIndex * axisExpandWidth - layoutInfo.axisExpandWindow[0]; - axisNameAvailableWidth = axisExpandWidth; - axisLabelShow = true; - } else { - position = layoutLength - (axisCount - 1 - axisIndex) * axisCollapseWidth; - nameTruncateMaxWidth = axisCollapseWidth; - } - - return { - position: position, - axisNameAvailableWidth: axisNameAvailableWidth, - axisLabelShow: axisLabelShow, - nameTruncateMaxWidth: nameTruncateMaxWidth - }; - } - - function createParallelCoordSys(ecModel, api) { - var coordSysList = []; - ecModel.eachComponent('parallel', function (parallelModel, idx) { - var coordSys = new Parallel(parallelModel, ecModel, api); - coordSys.name = 'parallel_' + idx; - coordSys.resize(parallelModel, api); - parallelModel.coordinateSystem = coordSys; - coordSys.model = parallelModel; - coordSysList.push(coordSys); - }); // Inject the coordinateSystems into seriesModel - - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.get('coordinateSystem') === 'parallel') { - var parallelModel = seriesModel.getReferringComponents('parallel', SINGLE_REFERRING).models[0]; - seriesModel.coordinateSystem = parallelModel.coordinateSystem; - } - }); - return coordSysList; - } - - var parallelCoordSysCreator = { - create: createParallelCoordSys - }; - - var ParallelAxisModel = - /** @class */ - function (_super) { - __extends(ParallelAxisModel, _super); - - function ParallelAxisModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ParallelAxisModel.type; - /** - * @readOnly - */ - - _this.activeIntervals = []; - return _this; - } - - ParallelAxisModel.prototype.getAreaSelectStyle = function () { - return makeStyleMapper([['fill', 'color'], ['lineWidth', 'borderWidth'], ['stroke', 'borderColor'], ['width', 'width'], ['opacity', 'opacity'] // Option decal is in `DecalObject` but style.decal is in `PatternObject`. - // So do not transfer decal directly. - ])(this.getModel('areaSelectStyle')); - }; - /** - * The code of this feature is put on AxisModel but not ParallelAxis, - * because axisModel can be alive after echarts updating but instance of - * ParallelAxis having been disposed. this._activeInterval should be kept - * when action dispatched (i.e. legend click). - * - * @param intervals `interval.length === 0` means set all active. - */ - - - ParallelAxisModel.prototype.setActiveIntervals = function (intervals) { - var activeIntervals = this.activeIntervals = clone(intervals); // Normalize - - if (activeIntervals) { - for (var i = activeIntervals.length - 1; i >= 0; i--) { - asc(activeIntervals[i]); - } - } - }; - /** - * @param value When only attempting detect whether 'no activeIntervals set', - * `value` is not needed to be input. - */ - - - ParallelAxisModel.prototype.getActiveState = function (value) { - var activeIntervals = this.activeIntervals; - - if (!activeIntervals.length) { - return 'normal'; - } - - if (value == null || isNaN(+value)) { - return 'inactive'; - } // Simple optimization - - - if (activeIntervals.length === 1) { - var interval = activeIntervals[0]; - - if (interval[0] <= value && value <= interval[1]) { - return 'active'; - } - } else { - for (var i = 0, len = activeIntervals.length; i < len; i++) { - if (activeIntervals[i][0] <= value && value <= activeIntervals[i][1]) { - return 'active'; - } - } - } - - return 'inactive'; - }; - - return ParallelAxisModel; - }(ComponentModel); - - mixin(ParallelAxisModel, AxisModelCommonMixin); - - var BRUSH_PANEL_GLOBAL = true; - var mathMin$9 = Math.min; - var mathMax$9 = Math.max; - var mathPow$2 = Math.pow; - var COVER_Z = 10000; - var UNSELECT_THRESHOLD = 6; - var MIN_RESIZE_LINE_WIDTH = 6; - var MUTEX_RESOURCE_KEY = 'globalPan'; - var DIRECTION_MAP = { - w: [0, 0], - e: [0, 1], - n: [1, 0], - s: [1, 1] - }; - var CURSOR_MAP = { - w: 'ew', - e: 'ew', - n: 'ns', - s: 'ns', - ne: 'nesw', - sw: 'nesw', - nw: 'nwse', - se: 'nwse' - }; - var DEFAULT_BRUSH_OPT = { - brushStyle: { - lineWidth: 2, - stroke: 'rgba(210,219,238,0.3)', - fill: '#D2DBEE' - }, - transformable: true, - brushMode: 'single', - removeOnClick: false - }; - var baseUID = 0; - /** - * params: - * areas: Array.<Array>, coord relates to container group, - * If no container specified, to global. - * opt { - * isEnd: boolean, - * removeOnClick: boolean - * } - */ - - var BrushController = - /** @class */ - function (_super) { - __extends(BrushController, _super); - - function BrushController(zr) { - var _this = _super.call(this) || this; - /** - * @internal - */ - - - _this._track = []; - /** - * @internal - */ - - _this._covers = []; - _this._handlers = {}; - - if ("development" !== 'production') { - assert(zr); - } - - _this._zr = zr; - _this.group = new Group(); - _this._uid = 'brushController_' + baseUID++; - each(pointerHandlers, function (handler, eventName) { - this._handlers[eventName] = bind(handler, this); - }, _this); - return _this; - } - /** - * If set to `false`, select disabled. - */ - - - BrushController.prototype.enableBrush = function (brushOption) { - if ("development" !== 'production') { - assert(this._mounted); - } - - this._brushType && this._doDisableBrush(); - brushOption.brushType && this._doEnableBrush(brushOption); - return this; - }; - - BrushController.prototype._doEnableBrush = function (brushOption) { - var zr = this._zr; // Consider roam, which takes globalPan too. - - if (!this._enableGlobalPan) { - take(zr, MUTEX_RESOURCE_KEY, this._uid); - } - - each(this._handlers, function (handler, eventName) { - zr.on(eventName, handler); - }); - this._brushType = brushOption.brushType; - this._brushOption = merge(clone(DEFAULT_BRUSH_OPT), brushOption, true); - }; - - BrushController.prototype._doDisableBrush = function () { - var zr = this._zr; - release(zr, MUTEX_RESOURCE_KEY, this._uid); - each(this._handlers, function (handler, eventName) { - zr.off(eventName, handler); - }); - this._brushType = this._brushOption = null; - }; - /** - * @param panelOpts If not pass, it is global brush. - */ - - - BrushController.prototype.setPanels = function (panelOpts) { - if (panelOpts && panelOpts.length) { - var panels_1 = this._panels = {}; - each(panelOpts, function (panelOpts) { - panels_1[panelOpts.panelId] = clone(panelOpts); - }); - } else { - this._panels = null; - } - - return this; - }; - - BrushController.prototype.mount = function (opt) { - opt = opt || {}; - - if ("development" !== 'production') { - this._mounted = true; // should be at first. - } - - this._enableGlobalPan = opt.enableGlobalPan; - var thisGroup = this.group; - - this._zr.add(thisGroup); - - thisGroup.attr({ - x: opt.x || 0, - y: opt.y || 0, - rotation: opt.rotation || 0, - scaleX: opt.scaleX || 1, - scaleY: opt.scaleY || 1 - }); - this._transform = thisGroup.getLocalTransform(); - return this; - }; // eachCover(cb, context): void { - // each(this._covers, cb, context); - // } - - /** - * Update covers. - * @param coverConfigList - * If coverConfigList is null/undefined, all covers removed. - */ - - - BrushController.prototype.updateCovers = function (coverConfigList) { - if ("development" !== 'production') { - assert(this._mounted); - } - - coverConfigList = map(coverConfigList, function (coverConfig) { - return merge(clone(DEFAULT_BRUSH_OPT), coverConfig, true); - }); - var tmpIdPrefix = '\0-brush-index-'; - var oldCovers = this._covers; - var newCovers = this._covers = []; - var controller = this; - var creatingCover = this._creatingCover; - new DataDiffer(oldCovers, coverConfigList, oldGetKey, getKey).add(addOrUpdate).update(addOrUpdate).remove(remove).execute(); - return this; - - function getKey(brushOption, index) { - return (brushOption.id != null ? brushOption.id : tmpIdPrefix + index) + '-' + brushOption.brushType; - } - - function oldGetKey(cover, index) { - return getKey(cover.__brushOption, index); - } - - function addOrUpdate(newIndex, oldIndex) { - var newBrushInternal = coverConfigList[newIndex]; // Consider setOption in event listener of brushSelect, - // where updating cover when creating should be forbidden. - - if (oldIndex != null && oldCovers[oldIndex] === creatingCover) { - newCovers[newIndex] = oldCovers[oldIndex]; - } else { - var cover = newCovers[newIndex] = oldIndex != null ? (oldCovers[oldIndex].__brushOption = newBrushInternal, oldCovers[oldIndex]) : endCreating(controller, createCover(controller, newBrushInternal)); - updateCoverAfterCreation(controller, cover); - } - } - - function remove(oldIndex) { - if (oldCovers[oldIndex] !== creatingCover) { - controller.group.remove(oldCovers[oldIndex]); - } - } - }; - - BrushController.prototype.unmount = function () { - if ("development" !== 'production') { - if (!this._mounted) { - return; - } - } - - this.enableBrush(false); // container may 'removeAll' outside. - - clearCovers(this); - - this._zr.remove(this.group); - - if ("development" !== 'production') { - this._mounted = false; // should be at last. - } - - return this; - }; - - BrushController.prototype.dispose = function () { - this.unmount(); - this.off(); - }; - - return BrushController; - }(Eventful); - - function createCover(controller, brushOption) { - var cover = coverRenderers[brushOption.brushType].createCover(controller, brushOption); - cover.__brushOption = brushOption; - updateZ(cover, brushOption); - controller.group.add(cover); - return cover; - } - - function endCreating(controller, creatingCover) { - var coverRenderer = getCoverRenderer(creatingCover); - - if (coverRenderer.endCreating) { - coverRenderer.endCreating(controller, creatingCover); - updateZ(creatingCover, creatingCover.__brushOption); - } - - return creatingCover; - } - - function updateCoverShape(controller, cover) { - var brushOption = cover.__brushOption; - getCoverRenderer(cover).updateCoverShape(controller, cover, brushOption.range, brushOption); - } - - function updateZ(cover, brushOption) { - var z = brushOption.z; - z == null && (z = COVER_Z); - cover.traverse(function (el) { - el.z = z; - el.z2 = z; // Consider in given container. - }); - } - - function updateCoverAfterCreation(controller, cover) { - getCoverRenderer(cover).updateCommon(controller, cover); - updateCoverShape(controller, cover); - } - - function getCoverRenderer(cover) { - return coverRenderers[cover.__brushOption.brushType]; - } // return target panel or `true` (means global panel) - - - function getPanelByPoint(controller, e, localCursorPoint) { - var panels = controller._panels; - - if (!panels) { - return BRUSH_PANEL_GLOBAL; // Global panel - } - - var panel; - var transform = controller._transform; - each(panels, function (pn) { - pn.isTargetByCursor(e, localCursorPoint, transform) && (panel = pn); - }); - return panel; - } // Return a panel or true - - - function getPanelByCover(controller, cover) { - var panels = controller._panels; - - if (!panels) { - return BRUSH_PANEL_GLOBAL; // Global panel - } - - var panelId = cover.__brushOption.panelId; // User may give cover without coord sys info, - // which is then treated as global panel. - - return panelId != null ? panels[panelId] : BRUSH_PANEL_GLOBAL; - } - - function clearCovers(controller) { - var covers = controller._covers; - var originalLength = covers.length; - each(covers, function (cover) { - controller.group.remove(cover); - }, controller); - covers.length = 0; - return !!originalLength; - } - - function trigger$1(controller, opt) { - var areas = map(controller._covers, function (cover) { - var brushOption = cover.__brushOption; - var range = clone(brushOption.range); - return { - brushType: brushOption.brushType, - panelId: brushOption.panelId, - range: range - }; - }); - controller.trigger('brush', { - areas: areas, - isEnd: !!opt.isEnd, - removeOnClick: !!opt.removeOnClick - }); - } - - function shouldShowCover(controller) { - var track = controller._track; - - if (!track.length) { - return false; - } - - var p2 = track[track.length - 1]; - var p1 = track[0]; - var dx = p2[0] - p1[0]; - var dy = p2[1] - p1[1]; - var dist = mathPow$2(dx * dx + dy * dy, 0.5); - return dist > UNSELECT_THRESHOLD; - } - - function getTrackEnds(track) { - var tail = track.length - 1; - tail < 0 && (tail = 0); - return [track[0], track[tail]]; - } - - function createBaseRectCover(rectRangeConverter, controller, brushOption, edgeNameSequences) { - var cover = new Group(); - cover.add(new Rect({ - name: 'main', - style: makeStyle(brushOption), - silent: true, - draggable: true, - cursor: 'move', - drift: curry(driftRect, rectRangeConverter, controller, cover, ['n', 's', 'w', 'e']), - ondragend: curry(trigger$1, controller, { - isEnd: true - }) - })); - each(edgeNameSequences, function (nameSequence) { - cover.add(new Rect({ - name: nameSequence.join(''), - style: { - opacity: 0 - }, - draggable: true, - silent: true, - invisible: true, - drift: curry(driftRect, rectRangeConverter, controller, cover, nameSequence), - ondragend: curry(trigger$1, controller, { - isEnd: true - }) - })); - }); - return cover; - } - - function updateBaseRect(controller, cover, localRange, brushOption) { - var lineWidth = brushOption.brushStyle.lineWidth || 0; - var handleSize = mathMax$9(lineWidth, MIN_RESIZE_LINE_WIDTH); - var x = localRange[0][0]; - var y = localRange[1][0]; - var xa = x - lineWidth / 2; - var ya = y - lineWidth / 2; - var x2 = localRange[0][1]; - var y2 = localRange[1][1]; - var x2a = x2 - handleSize + lineWidth / 2; - var y2a = y2 - handleSize + lineWidth / 2; - var width = x2 - x; - var height = y2 - y; - var widtha = width + lineWidth; - var heighta = height + lineWidth; - updateRectShape(controller, cover, 'main', x, y, width, height); - - if (brushOption.transformable) { - updateRectShape(controller, cover, 'w', xa, ya, handleSize, heighta); - updateRectShape(controller, cover, 'e', x2a, ya, handleSize, heighta); - updateRectShape(controller, cover, 'n', xa, ya, widtha, handleSize); - updateRectShape(controller, cover, 's', xa, y2a, widtha, handleSize); - updateRectShape(controller, cover, 'nw', xa, ya, handleSize, handleSize); - updateRectShape(controller, cover, 'ne', x2a, ya, handleSize, handleSize); - updateRectShape(controller, cover, 'sw', xa, y2a, handleSize, handleSize); - updateRectShape(controller, cover, 'se', x2a, y2a, handleSize, handleSize); - } - } - - function updateCommon(controller, cover) { - var brushOption = cover.__brushOption; - var transformable = brushOption.transformable; - var mainEl = cover.childAt(0); - mainEl.useStyle(makeStyle(brushOption)); - mainEl.attr({ - silent: !transformable, - cursor: transformable ? 'move' : 'default' - }); - each([['w'], ['e'], ['n'], ['s'], ['s', 'e'], ['s', 'w'], ['n', 'e'], ['n', 'w']], function (nameSequence) { - var el = cover.childOfName(nameSequence.join('')); - var globalDir = nameSequence.length === 1 ? getGlobalDirection1(controller, nameSequence[0]) : getGlobalDirection2(controller, nameSequence); - el && el.attr({ - silent: !transformable, - invisible: !transformable, - cursor: transformable ? CURSOR_MAP[globalDir] + '-resize' : null - }); - }); - } - - function updateRectShape(controller, cover, name, x, y, w, h) { - var el = cover.childOfName(name); - el && el.setShape(pointsToRect(clipByPanel(controller, cover, [[x, y], [x + w, y + h]]))); - } - - function makeStyle(brushOption) { - return defaults({ - strokeNoScale: true - }, brushOption.brushStyle); - } - - function formatRectRange(x, y, x2, y2) { - var min = [mathMin$9(x, x2), mathMin$9(y, y2)]; - var max = [mathMax$9(x, x2), mathMax$9(y, y2)]; - return [[min[0], max[0]], [min[1], max[1]] // y range - ]; - } - - function getTransform$1(controller) { - return getTransform(controller.group); - } - - function getGlobalDirection1(controller, localDirName) { - var map = { - w: 'left', - e: 'right', - n: 'top', - s: 'bottom' - }; - var inverseMap = { - left: 'w', - right: 'e', - top: 'n', - bottom: 's' - }; - var dir = transformDirection(map[localDirName], getTransform$1(controller)); - return inverseMap[dir]; - } - - function getGlobalDirection2(controller, localDirNameSeq) { - var globalDir = [getGlobalDirection1(controller, localDirNameSeq[0]), getGlobalDirection1(controller, localDirNameSeq[1])]; - (globalDir[0] === 'e' || globalDir[0] === 'w') && globalDir.reverse(); - return globalDir.join(''); - } - - function driftRect(rectRangeConverter, controller, cover, dirNameSequence, dx, dy) { - var brushOption = cover.__brushOption; - var rectRange = rectRangeConverter.toRectRange(brushOption.range); - var localDelta = toLocalDelta(controller, dx, dy); - each(dirNameSequence, function (dirName) { - var ind = DIRECTION_MAP[dirName]; - rectRange[ind[0]][ind[1]] += localDelta[ind[0]]; - }); - brushOption.range = rectRangeConverter.fromRectRange(formatRectRange(rectRange[0][0], rectRange[1][0], rectRange[0][1], rectRange[1][1])); - updateCoverAfterCreation(controller, cover); - trigger$1(controller, { - isEnd: false - }); - } - - function driftPolygon(controller, cover, dx, dy) { - var range = cover.__brushOption.range; - var localDelta = toLocalDelta(controller, dx, dy); - each(range, function (point) { - point[0] += localDelta[0]; - point[1] += localDelta[1]; - }); - updateCoverAfterCreation(controller, cover); - trigger$1(controller, { - isEnd: false - }); - } - - function toLocalDelta(controller, dx, dy) { - var thisGroup = controller.group; - var localD = thisGroup.transformCoordToLocal(dx, dy); - var localZero = thisGroup.transformCoordToLocal(0, 0); - return [localD[0] - localZero[0], localD[1] - localZero[1]]; - } - - function clipByPanel(controller, cover, data) { - var panel = getPanelByCover(controller, cover); - return panel && panel !== BRUSH_PANEL_GLOBAL ? panel.clipPath(data, controller._transform) : clone(data); - } - - function pointsToRect(points) { - var xmin = mathMin$9(points[0][0], points[1][0]); - var ymin = mathMin$9(points[0][1], points[1][1]); - var xmax = mathMax$9(points[0][0], points[1][0]); - var ymax = mathMax$9(points[0][1], points[1][1]); - return { - x: xmin, - y: ymin, - width: xmax - xmin, - height: ymax - ymin - }; - } - - function resetCursor(controller, e, localCursorPoint) { - if ( // Check active - !controller._brushType // resetCursor should be always called when mouse is in zr area, - // but not called when mouse is out of zr area to avoid bad influence - // if `mousemove`, `mouseup` are triggered from `document` event. - || isOutsideZrArea(controller, e.offsetX, e.offsetY)) { - return; - } - - var zr = controller._zr; - var covers = controller._covers; - var currPanel = getPanelByPoint(controller, e, localCursorPoint); // Check whether in covers. - - if (!controller._dragging) { - for (var i = 0; i < covers.length; i++) { - var brushOption = covers[i].__brushOption; - - if (currPanel && (currPanel === BRUSH_PANEL_GLOBAL || brushOption.panelId === currPanel.panelId) && coverRenderers[brushOption.brushType].contain(covers[i], localCursorPoint[0], localCursorPoint[1])) { - // Use cursor style set on cover. - return; - } - } - } - - currPanel && zr.setCursorStyle('crosshair'); - } - - function preventDefault(e) { - var rawE = e.event; - rawE.preventDefault && rawE.preventDefault(); - } - - function mainShapeContain(cover, x, y) { - return cover.childOfName('main').contain(x, y); - } - - function updateCoverByMouse(controller, e, localCursorPoint, isEnd) { - var creatingCover = controller._creatingCover; - var panel = controller._creatingPanel; - var thisBrushOption = controller._brushOption; - var eventParams; - - controller._track.push(localCursorPoint.slice()); - - if (shouldShowCover(controller) || creatingCover) { - if (panel && !creatingCover) { - thisBrushOption.brushMode === 'single' && clearCovers(controller); - var brushOption = clone(thisBrushOption); - brushOption.brushType = determineBrushType(brushOption.brushType, panel); - brushOption.panelId = panel === BRUSH_PANEL_GLOBAL ? null : panel.panelId; - creatingCover = controller._creatingCover = createCover(controller, brushOption); - - controller._covers.push(creatingCover); - } - - if (creatingCover) { - var coverRenderer = coverRenderers[determineBrushType(controller._brushType, panel)]; - var coverBrushOption = creatingCover.__brushOption; - coverBrushOption.range = coverRenderer.getCreatingRange(clipByPanel(controller, creatingCover, controller._track)); - - if (isEnd) { - endCreating(controller, creatingCover); - coverRenderer.updateCommon(controller, creatingCover); - } - - updateCoverShape(controller, creatingCover); - eventParams = { - isEnd: isEnd - }; - } - } else if (isEnd && thisBrushOption.brushMode === 'single' && thisBrushOption.removeOnClick) { - // Help user to remove covers easily, only by a tiny drag, in 'single' mode. - // But a single click do not clear covers, because user may have casual - // clicks (for example, click on other component and do not expect covers - // disappear). - // Only some cover removed, trigger action, but not every click trigger action. - if (getPanelByPoint(controller, e, localCursorPoint) && clearCovers(controller)) { - eventParams = { - isEnd: isEnd, - removeOnClick: true - }; - } - } - - return eventParams; - } - - function determineBrushType(brushType, panel) { - if (brushType === 'auto') { - if ("development" !== 'production') { - assert(panel && panel.defaultBrushType, 'MUST have defaultBrushType when brushType is "atuo"'); - } - - return panel.defaultBrushType; - } - - return brushType; - } - - var pointerHandlers = { - mousedown: function (e) { - if (this._dragging) { - // In case some browser do not support globalOut, - // and release mouse out side the browser. - handleDragEnd(this, e); - } else if (!e.target || !e.target.draggable) { - preventDefault(e); - var localCursorPoint = this.group.transformCoordToLocal(e.offsetX, e.offsetY); - this._creatingCover = null; - var panel = this._creatingPanel = getPanelByPoint(this, e, localCursorPoint); - - if (panel) { - this._dragging = true; - this._track = [localCursorPoint.slice()]; - } - } - }, - mousemove: function (e) { - var x = e.offsetX; - var y = e.offsetY; - var localCursorPoint = this.group.transformCoordToLocal(x, y); - resetCursor(this, e, localCursorPoint); - - if (this._dragging) { - preventDefault(e); - var eventParams = updateCoverByMouse(this, e, localCursorPoint, false); - eventParams && trigger$1(this, eventParams); - } - }, - mouseup: function (e) { - handleDragEnd(this, e); - } - }; - - function handleDragEnd(controller, e) { - if (controller._dragging) { - preventDefault(e); - var x = e.offsetX; - var y = e.offsetY; - var localCursorPoint = controller.group.transformCoordToLocal(x, y); - var eventParams = updateCoverByMouse(controller, e, localCursorPoint, true); - controller._dragging = false; - controller._track = []; - controller._creatingCover = null; // trigger event should be at final, after procedure will be nested. - - eventParams && trigger$1(controller, eventParams); - } - } - - function isOutsideZrArea(controller, x, y) { - var zr = controller._zr; - return x < 0 || x > zr.getWidth() || y < 0 || y > zr.getHeight(); - } - /** - * key: brushType - */ - - - var coverRenderers = { - lineX: getLineRenderer(0), - lineY: getLineRenderer(1), - rect: { - createCover: function (controller, brushOption) { - function returnInput(range) { - return range; - } - - return createBaseRectCover({ - toRectRange: returnInput, - fromRectRange: returnInput - }, controller, brushOption, [['w'], ['e'], ['n'], ['s'], ['s', 'e'], ['s', 'w'], ['n', 'e'], ['n', 'w']]); - }, - getCreatingRange: function (localTrack) { - var ends = getTrackEnds(localTrack); - return formatRectRange(ends[1][0], ends[1][1], ends[0][0], ends[0][1]); - }, - updateCoverShape: function (controller, cover, localRange, brushOption) { - updateBaseRect(controller, cover, localRange, brushOption); - }, - updateCommon: updateCommon, - contain: mainShapeContain - }, - polygon: { - createCover: function (controller, brushOption) { - var cover = new Group(); // Do not use graphic.Polygon because graphic.Polyline do not close the - // border of the shape when drawing, which is a better experience for user. - - cover.add(new Polyline({ - name: 'main', - style: makeStyle(brushOption), - silent: true - })); - return cover; - }, - getCreatingRange: function (localTrack) { - return localTrack; - }, - endCreating: function (controller, cover) { - cover.remove(cover.childAt(0)); // Use graphic.Polygon close the shape. - - cover.add(new Polygon({ - name: 'main', - draggable: true, - drift: curry(driftPolygon, controller, cover), - ondragend: curry(trigger$1, controller, { - isEnd: true - }) - })); - }, - updateCoverShape: function (controller, cover, localRange, brushOption) { - cover.childAt(0).setShape({ - points: clipByPanel(controller, cover, localRange) - }); - }, - updateCommon: updateCommon, - contain: mainShapeContain - } - }; - - function getLineRenderer(xyIndex) { - return { - createCover: function (controller, brushOption) { - return createBaseRectCover({ - toRectRange: function (range) { - var rectRange = [range, [0, 100]]; - xyIndex && rectRange.reverse(); - return rectRange; - }, - fromRectRange: function (rectRange) { - return rectRange[xyIndex]; - } - }, controller, brushOption, [[['w'], ['e']], [['n'], ['s']]][xyIndex]); - }, - getCreatingRange: function (localTrack) { - var ends = getTrackEnds(localTrack); - var min = mathMin$9(ends[0][xyIndex], ends[1][xyIndex]); - var max = mathMax$9(ends[0][xyIndex], ends[1][xyIndex]); - return [min, max]; - }, - updateCoverShape: function (controller, cover, localRange, brushOption) { - var otherExtent; // If brushWidth not specified, fit the panel. - - var panel = getPanelByCover(controller, cover); - - if (panel !== BRUSH_PANEL_GLOBAL && panel.getLinearBrushOtherExtent) { - otherExtent = panel.getLinearBrushOtherExtent(xyIndex); - } else { - var zr = controller._zr; - otherExtent = [0, [zr.getWidth(), zr.getHeight()][1 - xyIndex]]; - } - - var rectRange = [localRange, otherExtent]; - xyIndex && rectRange.reverse(); - updateBaseRect(controller, cover, rectRange, brushOption); - }, - updateCommon: updateCommon, - contain: mainShapeContain - }; - } - - function makeRectPanelClipPath(rect) { - rect = normalizeRect(rect); - return function (localPoints) { - return clipPointsByRect(localPoints, rect); - }; - } - function makeLinearBrushOtherExtent(rect, specifiedXYIndex) { - rect = normalizeRect(rect); - return function (xyIndex) { - var idx = specifiedXYIndex != null ? specifiedXYIndex : xyIndex; - var brushWidth = idx ? rect.width : rect.height; - var base = idx ? rect.x : rect.y; - return [base, base + (brushWidth || 0)]; - }; - } - function makeRectIsTargetByCursor(rect, api, targetModel) { - var boundingRect = normalizeRect(rect); - return function (e, localCursorPoint) { - return boundingRect.contain(localCursorPoint[0], localCursorPoint[1]) && !onIrrelevantElement(e, api, targetModel); - }; - } // Consider width/height is negative. - - function normalizeRect(rect) { - return BoundingRect.create(rect); - } - - var elementList = ['axisLine', 'axisTickLabel', 'axisName']; - - var ParallelAxisView = - /** @class */ - function (_super) { - __extends(ParallelAxisView, _super); - - function ParallelAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ParallelAxisView.type; - return _this; - } - - ParallelAxisView.prototype.init = function (ecModel, api) { - _super.prototype.init.apply(this, arguments); - - (this._brushController = new BrushController(api.getZr())).on('brush', bind(this._onBrush, this)); - }; - - ParallelAxisView.prototype.render = function (axisModel, ecModel, api, payload) { - if (fromAxisAreaSelect(axisModel, ecModel, payload)) { - return; - } - - this.axisModel = axisModel; - this.api = api; - this.group.removeAll(); - var oldAxisGroup = this._axisGroup; - this._axisGroup = new Group(); - this.group.add(this._axisGroup); - - if (!axisModel.get('show')) { - return; - } - - var coordSysModel = getCoordSysModel(axisModel, ecModel); - var coordSys = coordSysModel.coordinateSystem; - var areaSelectStyle = axisModel.getAreaSelectStyle(); - var areaWidth = areaSelectStyle.width; - var dim = axisModel.axis.dim; - var axisLayout = coordSys.getAxisLayout(dim); - var builderOpt = extend({ - strokeContainThreshold: areaWidth - }, axisLayout); - var axisBuilder = new AxisBuilder(axisModel, builderOpt); - each(elementList, axisBuilder.add, axisBuilder); - - this._axisGroup.add(axisBuilder.getGroup()); - - this._refreshBrushController(builderOpt, areaSelectStyle, axisModel, coordSysModel, areaWidth, api); - - groupTransition(oldAxisGroup, this._axisGroup, axisModel); - }; // /** - // * @override - // */ - // updateVisual(axisModel, ecModel, api, payload) { - // this._brushController && this._brushController - // .updateCovers(getCoverInfoList(axisModel)); - // } - - - ParallelAxisView.prototype._refreshBrushController = function (builderOpt, areaSelectStyle, axisModel, coordSysModel, areaWidth, api) { - // After filtering, axis may change, select area needs to be update. - var extent = axisModel.axis.getExtent(); - var extentLen = extent[1] - extent[0]; - var extra = Math.min(30, Math.abs(extentLen) * 0.1); // Arbitrary value. - // width/height might be negative, which will be - // normalized in BoundingRect. - - var rect = BoundingRect.create({ - x: extent[0], - y: -areaWidth / 2, - width: extentLen, - height: areaWidth - }); - rect.x -= extra; - rect.width += 2 * extra; - - this._brushController.mount({ - enableGlobalPan: true, - rotation: builderOpt.rotation, - x: builderOpt.position[0], - y: builderOpt.position[1] - }).setPanels([{ - panelId: 'pl', - clipPath: makeRectPanelClipPath(rect), - isTargetByCursor: makeRectIsTargetByCursor(rect, api, coordSysModel), - getLinearBrushOtherExtent: makeLinearBrushOtherExtent(rect, 0) - }]).enableBrush({ - brushType: 'lineX', - brushStyle: areaSelectStyle, - removeOnClick: true - }).updateCovers(getCoverInfoList(axisModel)); - }; - - ParallelAxisView.prototype._onBrush = function (eventParam) { - var coverInfoList = eventParam.areas; // Do not cache these object, because the mey be changed. - - var axisModel = this.axisModel; - var axis = axisModel.axis; - var intervals = map(coverInfoList, function (coverInfo) { - return [axis.coordToData(coverInfo.range[0], true), axis.coordToData(coverInfo.range[1], true)]; - }); // If realtime is true, action is not dispatched on drag end, because - // the drag end emits the same params with the last drag move event, - // and may have some delay when using touch pad. - - if (!axisModel.option.realtime === eventParam.isEnd || eventParam.removeOnClick) { - // jshint ignore:line - this.api.dispatchAction({ - type: 'axisAreaSelect', - parallelAxisId: axisModel.id, - intervals: intervals - }); - } - }; - - ParallelAxisView.prototype.dispose = function () { - this._brushController.dispose(); - }; - - ParallelAxisView.type = 'parallelAxis'; - return ParallelAxisView; - }(ComponentView); - - function fromAxisAreaSelect(axisModel, ecModel, payload) { - return payload && payload.type === 'axisAreaSelect' && ecModel.findComponents({ - mainType: 'parallelAxis', - query: payload - })[0] === axisModel; - } - - function getCoverInfoList(axisModel) { - var axis = axisModel.axis; - return map(axisModel.activeIntervals, function (interval) { - return { - brushType: 'lineX', - panelId: 'pl', - range: [axis.dataToCoord(interval[0], true), axis.dataToCoord(interval[1], true)] - }; - }); - } - - function getCoordSysModel(axisModel, ecModel) { - return ecModel.getComponent('parallel', axisModel.get('parallelIndex')); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var actionInfo$1 = { - type: 'axisAreaSelect', - event: 'axisAreaSelected' // update: 'updateVisual' - - }; - function installParallelActions(registers) { - registers.registerAction(actionInfo$1, function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'parallelAxis', - query: payload - }, function (parallelAxisModel) { - parallelAxisModel.axis.model.setActiveIntervals(payload.intervals); - }); - }); - /** - * @payload - */ - - registers.registerAction('parallelAxisExpand', function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'parallel', - query: payload - }, function (parallelModel) { - parallelModel.setAxisExpand(payload); - }); - }); - } - - var defaultAxisOption = { - type: 'value', - areaSelectStyle: { - width: 20, - borderWidth: 1, - borderColor: 'rgba(160,197,232)', - color: 'rgba(160,197,232)', - opacity: 0.3 - }, - realtime: true, - z: 10 - }; - function install$g(registers) { - registers.registerComponentView(ParallelView$1); - registers.registerComponentModel(ParallelModel); - registers.registerCoordinateSystem('parallel', parallelCoordSysCreator); - registers.registerPreprocessor(parallelPreprocessor); - registers.registerComponentModel(ParallelAxisModel); - registers.registerComponentView(ParallelAxisView); - axisModelCreator(registers, 'parallel', ParallelAxisModel, defaultAxisOption); - installParallelActions(registers); - } - - function install$h(registers) { - use(install$g); - registers.registerChartView(ParallelView); - registers.registerSeriesModel(ParallelSeriesModel); - registers.registerVisual(registers.PRIORITY.VISUAL.BRUSH, parallelVisual); - } - - var SankeyPathShape = - /** @class */ - function () { - function SankeyPathShape() { - this.x1 = 0; - this.y1 = 0; - this.x2 = 0; - this.y2 = 0; - this.cpx1 = 0; - this.cpy1 = 0; - this.cpx2 = 0; - this.cpy2 = 0; - this.extent = 0; - } - - return SankeyPathShape; - }(); - - var SankeyPath = - /** @class */ - function (_super) { - __extends(SankeyPath, _super); - - function SankeyPath(opts) { - return _super.call(this, opts) || this; - } - - SankeyPath.prototype.getDefaultShape = function () { - return new SankeyPathShape(); - }; - - SankeyPath.prototype.buildPath = function (ctx, shape) { - var extent = shape.extent; - ctx.moveTo(shape.x1, shape.y1); - ctx.bezierCurveTo(shape.cpx1, shape.cpy1, shape.cpx2, shape.cpy2, shape.x2, shape.y2); - - if (shape.orient === 'vertical') { - ctx.lineTo(shape.x2 + extent, shape.y2); - ctx.bezierCurveTo(shape.cpx2 + extent, shape.cpy2, shape.cpx1 + extent, shape.cpy1, shape.x1 + extent, shape.y1); - } else { - ctx.lineTo(shape.x2, shape.y2 + extent); - ctx.bezierCurveTo(shape.cpx2, shape.cpy2 + extent, shape.cpx1, shape.cpy1 + extent, shape.x1, shape.y1 + extent); - } - - ctx.closePath(); - }; - - SankeyPath.prototype.highlight = function () { - enterEmphasis(this); - }; - - SankeyPath.prototype.downplay = function () { - leaveEmphasis(this); - }; - - return SankeyPath; - }(Path); - - var SankeyView = - /** @class */ - function (_super) { - __extends(SankeyView, _super); - - function SankeyView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SankeyView.type; - _this._focusAdjacencyDisabled = false; - return _this; - } - - SankeyView.prototype.render = function (seriesModel, ecModel, api) { - var sankeyView = this; - var graph = seriesModel.getGraph(); - var group = this.group; - var layoutInfo = seriesModel.layoutInfo; // view width - - var width = layoutInfo.width; // view height - - var height = layoutInfo.height; - var nodeData = seriesModel.getData(); - var edgeData = seriesModel.getData('edge'); - var orient = seriesModel.get('orient'); - this._model = seriesModel; - group.removeAll(); - group.x = layoutInfo.x; - group.y = layoutInfo.y; // generate a bezire Curve for each edge - - graph.eachEdge(function (edge) { - var curve = new SankeyPath(); - var ecData = getECData(curve); - ecData.dataIndex = edge.dataIndex; - ecData.seriesIndex = seriesModel.seriesIndex; - ecData.dataType = 'edge'; - var edgeModel = edge.getModel(); - var lineStyleModel = edgeModel.getModel('lineStyle'); - var curvature = lineStyleModel.get('curveness'); - var n1Layout = edge.node1.getLayout(); - var node1Model = edge.node1.getModel(); - var dragX1 = node1Model.get('localX'); - var dragY1 = node1Model.get('localY'); - var n2Layout = edge.node2.getLayout(); - var node2Model = edge.node2.getModel(); - var dragX2 = node2Model.get('localX'); - var dragY2 = node2Model.get('localY'); - var edgeLayout = edge.getLayout(); - var x1; - var y1; - var x2; - var y2; - var cpx1; - var cpy1; - var cpx2; - var cpy2; - curve.shape.extent = Math.max(1, edgeLayout.dy); - curve.shape.orient = orient; - - if (orient === 'vertical') { - x1 = (dragX1 != null ? dragX1 * width : n1Layout.x) + edgeLayout.sy; - y1 = (dragY1 != null ? dragY1 * height : n1Layout.y) + n1Layout.dy; - x2 = (dragX2 != null ? dragX2 * width : n2Layout.x) + edgeLayout.ty; - y2 = dragY2 != null ? dragY2 * height : n2Layout.y; - cpx1 = x1; - cpy1 = y1 * (1 - curvature) + y2 * curvature; - cpx2 = x2; - cpy2 = y1 * curvature + y2 * (1 - curvature); - } else { - x1 = (dragX1 != null ? dragX1 * width : n1Layout.x) + n1Layout.dx; - y1 = (dragY1 != null ? dragY1 * height : n1Layout.y) + edgeLayout.sy; - x2 = dragX2 != null ? dragX2 * width : n2Layout.x; - y2 = (dragY2 != null ? dragY2 * height : n2Layout.y) + edgeLayout.ty; - cpx1 = x1 * (1 - curvature) + x2 * curvature; - cpy1 = y1; - cpx2 = x1 * curvature + x2 * (1 - curvature); - cpy2 = y2; - } - - curve.setShape({ - x1: x1, - y1: y1, - x2: x2, - y2: y2, - cpx1: cpx1, - cpy1: cpy1, - cpx2: cpx2, - cpy2: cpy2 - }); - curve.useStyle(lineStyleModel.getItemStyle()); // Special color, use source node color or target node color - - switch (curve.style.fill) { - case 'source': - curve.style.fill = edge.node1.getVisual('color'); - curve.style.decal = edge.node1.getVisual('style').decal; - break; - - case 'target': - curve.style.fill = edge.node2.getVisual('color'); - curve.style.decal = edge.node2.getVisual('style').decal; - break; - - case 'gradient': - var sourceColor = edge.node1.getVisual('color'); - var targetColor = edge.node2.getVisual('color'); - - if (isString(sourceColor) && isString(targetColor)) { - curve.style.fill = new LinearGradient(0, 0, +(orient === 'horizontal'), +(orient === 'vertical'), [{ - color: sourceColor, - offset: 0 - }, { - color: targetColor, - offset: 1 - }]); - } - - } - - setLabelStyle(curve, getLabelStatesModels(edgeModel, 'edgeLabel'), { - labelFetcher: seriesModel, - labelDataIndex: edge.dataIndex, - defaultText: "" + edgeModel.get('value') - }); - curve.setTextConfig({ - position: 'inside' - }); - var emphasisModel = edgeModel.getModel('emphasis'); - setStatesStylesFromModel(curve, edgeModel, 'lineStyle', function (model) { - return model.getItemStyle(); - }); - group.add(curve); - edgeData.setItemGraphicEl(edge.dataIndex, curve); - var focus = emphasisModel.get('focus'); - toggleHoverEmphasis(curve, focus === 'adjacency' ? edge.getAdjacentDataIndices() : focus, emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - getECData(curve).dataType = 'edge'; - }); // Generate a rect for each node - - graph.eachNode(function (node) { - var layout = node.getLayout(); - var itemModel = node.getModel(); - var dragX = itemModel.get('localX'); - var dragY = itemModel.get('localY'); - var emphasisModel = itemModel.getModel('emphasis'); - var rect = new Rect({ - shape: { - x: dragX != null ? dragX * width : layout.x, - y: dragY != null ? dragY * height : layout.y, - width: layout.dx, - height: layout.dy - }, - style: itemModel.getModel('itemStyle').getItemStyle(), - z2: 10 - }); - setLabelStyle(rect, getLabelStatesModels(itemModel), { - labelFetcher: seriesModel, - labelDataIndex: node.dataIndex, - defaultText: node.id - }); - rect.disableLabelAnimation = true; - rect.setStyle('fill', node.getVisual('color')); - rect.setStyle('decal', node.getVisual('style').decal); - setStatesStylesFromModel(rect, itemModel); - group.add(rect); - nodeData.setItemGraphicEl(node.dataIndex, rect); - getECData(rect).dataType = 'node'; - var focus = emphasisModel.get('focus'); - toggleHoverEmphasis(rect, focus === 'adjacency' ? node.getAdjacentDataIndices() : focus, emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - }); - nodeData.eachItemGraphicEl(function (el, dataIndex) { - var itemModel = nodeData.getItemModel(dataIndex); - - if (itemModel.get('draggable')) { - el.drift = function (dx, dy) { - sankeyView._focusAdjacencyDisabled = true; - this.shape.x += dx; - this.shape.y += dy; - this.dirty(); - api.dispatchAction({ - type: 'dragNode', - seriesId: seriesModel.id, - dataIndex: nodeData.getRawIndex(dataIndex), - localX: this.shape.x / width, - localY: this.shape.y / height - }); - }; - - el.ondragend = function () { - sankeyView._focusAdjacencyDisabled = false; - }; - - el.draggable = true; - el.cursor = 'move'; - } - }); - - if (!this._data && seriesModel.isAnimationEnabled()) { - group.setClipPath(createGridClipShape$1(group.getBoundingRect(), seriesModel, function () { - group.removeClipPath(); - })); - } - - this._data = seriesModel.getData(); - }; - - SankeyView.prototype.dispose = function () {}; - - SankeyView.type = 'sankey'; - return SankeyView; - }(ChartView); // Add animation to the view - - - function createGridClipShape$1(rect, seriesModel, cb) { - var rectEl = new Rect({ - shape: { - x: rect.x - 10, - y: rect.y - 10, - width: 0, - height: rect.height + 20 - } - }); - initProps(rectEl, { - shape: { - width: rect.width + 20 - } - }, seriesModel, cb); - return rectEl; - } - - var SankeySeriesModel = - /** @class */ - function (_super) { - __extends(SankeySeriesModel, _super); - - function SankeySeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SankeySeriesModel.type; - return _this; - } - /** - * Init a graph data structure from data in option series - */ - - - SankeySeriesModel.prototype.getInitialData = function (option, ecModel) { - var links = option.edges || option.links; - var nodes = option.data || option.nodes; - var levels = option.levels; - this.levelModels = []; - var levelModels = this.levelModels; - - for (var i = 0; i < levels.length; i++) { - if (levels[i].depth != null && levels[i].depth >= 0) { - levelModels[levels[i].depth] = new Model(levels[i], this, ecModel); - } else { - if ("development" !== 'production') { - throw new Error('levels[i].depth is mandatory and should be natural number'); - } - } - } - - if (nodes && links) { - var graph = createGraphFromNodeEdge(nodes, links, this, true, beforeLink); - return graph.data; - } - - function beforeLink(nodeData, edgeData) { - nodeData.wrapMethod('getItemModel', function (model, idx) { - var seriesModel = model.parentModel; - var layout = seriesModel.getData().getItemLayout(idx); - - if (layout) { - var nodeDepth = layout.depth; - var levelModel = seriesModel.levelModels[nodeDepth]; - - if (levelModel) { - model.parentModel = levelModel; - } - } - - return model; - }); - edgeData.wrapMethod('getItemModel', function (model, idx) { - var seriesModel = model.parentModel; - var edge = seriesModel.getGraph().getEdgeByIndex(idx); - var layout = edge.node1.getLayout(); - - if (layout) { - var depth = layout.depth; - var levelModel = seriesModel.levelModels[depth]; - - if (levelModel) { - model.parentModel = levelModel; - } - } - - return model; - }); - } - }; - - SankeySeriesModel.prototype.setNodePosition = function (dataIndex, localPosition) { - var nodes = this.option.data || this.option.nodes; - var dataItem = nodes[dataIndex]; - dataItem.localX = localPosition[0]; - dataItem.localY = localPosition[1]; - }; - /** - * Return the graphic data structure - * - * @return graphic data structure - */ - - - SankeySeriesModel.prototype.getGraph = function () { - return this.getData().graph; - }; - /** - * Get edge data of graphic data structure - * - * @return data structure of list - */ - - - SankeySeriesModel.prototype.getEdgeData = function () { - return this.getGraph().edgeData; - }; - - SankeySeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - function noValue(val) { - return isNaN(val) || val == null; - } // dataType === 'node' or empty do not show tooltip by default - - - if (dataType === 'edge') { - var params = this.getDataParams(dataIndex, dataType); - var rawDataOpt = params.data; - var edgeValue = params.value; - var edgeName = rawDataOpt.source + ' -- ' + rawDataOpt.target; - return createTooltipMarkup('nameValue', { - name: edgeName, - value: edgeValue, - noValue: noValue(edgeValue) - }); - } // dataType === 'node' - else { - var node = this.getGraph().getNodeByIndex(dataIndex); - var value = node.getLayout().value; - var name_1 = this.getDataParams(dataIndex, dataType).data.name; - return createTooltipMarkup('nameValue', { - name: name_1 != null ? name_1 + '' : null, - value: value, - noValue: noValue(value) - }); - } - }; - - SankeySeriesModel.prototype.optionUpdated = function () {}; // Override Series.getDataParams() - - - SankeySeriesModel.prototype.getDataParams = function (dataIndex, dataType) { - var params = _super.prototype.getDataParams.call(this, dataIndex, dataType); - - if (params.value == null && dataType === 'node') { - var node = this.getGraph().getNodeByIndex(dataIndex); - var nodeValue = node.getLayout().value; - params.value = nodeValue; - } - - return params; - }; - - SankeySeriesModel.type = 'series.sankey'; - SankeySeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'view', - left: '5%', - top: '5%', - right: '20%', - bottom: '5%', - orient: 'horizontal', - nodeWidth: 20, - nodeGap: 8, - draggable: true, - layoutIterations: 32, - label: { - show: true, - position: 'right', - fontSize: 12 - }, - edgeLabel: { - show: false, - fontSize: 12 - }, - levels: [], - nodeAlign: 'justify', - lineStyle: { - color: '#314656', - opacity: 0.2, - curveness: 0.5 - }, - emphasis: { - label: { - show: true - }, - lineStyle: { - opacity: 0.5 - } - }, - select: { - itemStyle: { - borderColor: '#212121' - } - }, - animationEasing: 'linear', - animationDuration: 1000 - }; - return SankeySeriesModel; - }(SeriesModel); - - function sankeyLayout(ecModel, api) { - ecModel.eachSeriesByType('sankey', function (seriesModel) { - var nodeWidth = seriesModel.get('nodeWidth'); - var nodeGap = seriesModel.get('nodeGap'); - var layoutInfo = getViewRect$4(seriesModel, api); - seriesModel.layoutInfo = layoutInfo; - var width = layoutInfo.width; - var height = layoutInfo.height; - var graph = seriesModel.getGraph(); - var nodes = graph.nodes; - var edges = graph.edges; - computeNodeValues(nodes); - var filteredNodes = filter(nodes, function (node) { - return node.getLayout().value === 0; - }); - var iterations = filteredNodes.length !== 0 ? 0 : seriesModel.get('layoutIterations'); - var orient = seriesModel.get('orient'); - var nodeAlign = seriesModel.get('nodeAlign'); - layoutSankey(nodes, edges, nodeWidth, nodeGap, width, height, iterations, orient, nodeAlign); - }); - } - /** - * Get the layout position of the whole view - */ - - function getViewRect$4(seriesModel, api) { - return getLayoutRect(seriesModel.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - } - - function layoutSankey(nodes, edges, nodeWidth, nodeGap, width, height, iterations, orient, nodeAlign) { - computeNodeBreadths(nodes, edges, nodeWidth, width, height, orient, nodeAlign); - computeNodeDepths(nodes, edges, height, width, nodeGap, iterations, orient); - computeEdgeDepths(nodes, orient); - } - /** - * Compute the value of each node by summing the associated edge's value - */ - - - function computeNodeValues(nodes) { - each(nodes, function (node) { - var value1 = sum(node.outEdges, getEdgeValue); - var value2 = sum(node.inEdges, getEdgeValue); - var nodeRawValue = node.getValue() || 0; - var value = Math.max(value1, value2, nodeRawValue); - node.setLayout({ - value: value - }, true); - }); - } - /** - * Compute the x-position for each node. - * - * Here we use Kahn algorithm to detect cycle when we traverse - * the node to computer the initial x position. - */ - - - function computeNodeBreadths(nodes, edges, nodeWidth, width, height, orient, nodeAlign) { - // Used to mark whether the edge is deleted. if it is deleted, - // the value is 0, otherwise it is 1. - var remainEdges = []; // Storage each node's indegree. - - var indegreeArr = []; // Used to storage the node with indegree is equal to 0. - - var zeroIndegrees = []; - var nextTargetNode = []; - var x = 0; // let kx = 0; - - for (var i = 0; i < edges.length; i++) { - remainEdges[i] = 1; - } - - for (var i = 0; i < nodes.length; i++) { - indegreeArr[i] = nodes[i].inEdges.length; - - if (indegreeArr[i] === 0) { - zeroIndegrees.push(nodes[i]); - } - } - - var maxNodeDepth = -1; // Traversing nodes using topological sorting to calculate the - // horizontal(if orient === 'horizontal') or vertical(if orient === 'vertical') - // position of the nodes. - - while (zeroIndegrees.length) { - for (var idx = 0; idx < zeroIndegrees.length; idx++) { - var node = zeroIndegrees[idx]; - var item = node.hostGraph.data.getRawDataItem(node.dataIndex); - var isItemDepth = item.depth != null && item.depth >= 0; - - if (isItemDepth && item.depth > maxNodeDepth) { - maxNodeDepth = item.depth; - } - - node.setLayout({ - depth: isItemDepth ? item.depth : x - }, true); - orient === 'vertical' ? node.setLayout({ - dy: nodeWidth - }, true) : node.setLayout({ - dx: nodeWidth - }, true); - - for (var edgeIdx = 0; edgeIdx < node.outEdges.length; edgeIdx++) { - var edge = node.outEdges[edgeIdx]; - var indexEdge = edges.indexOf(edge); - remainEdges[indexEdge] = 0; - var targetNode = edge.node2; - var nodeIndex = nodes.indexOf(targetNode); - - if (--indegreeArr[nodeIndex] === 0 && nextTargetNode.indexOf(targetNode) < 0) { - nextTargetNode.push(targetNode); - } - } - } - - ++x; - zeroIndegrees = nextTargetNode; - nextTargetNode = []; - } - - for (var i = 0; i < remainEdges.length; i++) { - if (remainEdges[i] === 1) { - throw new Error('Sankey is a DAG, the original data has cycle!'); - } - } - - var maxDepth = maxNodeDepth > x - 1 ? maxNodeDepth : x - 1; - - if (nodeAlign && nodeAlign !== 'left') { - adjustNodeWithNodeAlign(nodes, nodeAlign, orient, maxDepth); - } - - var kx = orient === 'vertical' ? (height - nodeWidth) / maxDepth : (width - nodeWidth) / maxDepth; - scaleNodeBreadths(nodes, kx, orient); - } - - function isNodeDepth(node) { - var item = node.hostGraph.data.getRawDataItem(node.dataIndex); - return item.depth != null && item.depth >= 0; - } - - function adjustNodeWithNodeAlign(nodes, nodeAlign, orient, maxDepth) { - if (nodeAlign === 'right') { - var nextSourceNode = []; - var remainNodes = nodes; - var nodeHeight = 0; - - while (remainNodes.length) { - for (var i = 0; i < remainNodes.length; i++) { - var node = remainNodes[i]; - node.setLayout({ - skNodeHeight: nodeHeight - }, true); - - for (var j = 0; j < node.inEdges.length; j++) { - var edge = node.inEdges[j]; - - if (nextSourceNode.indexOf(edge.node1) < 0) { - nextSourceNode.push(edge.node1); - } - } - } - - remainNodes = nextSourceNode; - nextSourceNode = []; - ++nodeHeight; - } - - each(nodes, function (node) { - if (!isNodeDepth(node)) { - node.setLayout({ - depth: Math.max(0, maxDepth - node.getLayout().skNodeHeight) - }, true); - } - }); - } else if (nodeAlign === 'justify') { - moveSinksRight(nodes, maxDepth); - } - } - /** - * All the node without outEgdes are assigned maximum x-position and - * be aligned in the last column. - * - * @param nodes. node of sankey view. - * @param maxDepth. use to assign to node without outEdges as x-position. - */ - - - function moveSinksRight(nodes, maxDepth) { - each(nodes, function (node) { - if (!isNodeDepth(node) && !node.outEdges.length) { - node.setLayout({ - depth: maxDepth - }, true); - } - }); - } - /** - * Scale node x-position to the width - * - * @param nodes node of sankey view - * @param kx multiple used to scale nodes - */ - - - function scaleNodeBreadths(nodes, kx, orient) { - each(nodes, function (node) { - var nodeDepth = node.getLayout().depth * kx; - orient === 'vertical' ? node.setLayout({ - y: nodeDepth - }, true) : node.setLayout({ - x: nodeDepth - }, true); - }); - } - /** - * Using Gauss-Seidel iterations method to compute the node depth(y-position) - * - * @param nodes node of sankey view - * @param edges edge of sankey view - * @param height the whole height of the area to draw the view - * @param nodeGap the vertical distance between two nodes - * in the same column. - * @param iterations the number of iterations for the algorithm - */ - - - function computeNodeDepths(nodes, edges, height, width, nodeGap, iterations, orient) { - var nodesByBreadth = prepareNodesByBreadth(nodes, orient); - initializeNodeDepth(nodesByBreadth, edges, height, width, nodeGap, orient); - resolveCollisions(nodesByBreadth, nodeGap, height, width, orient); - - for (var alpha = 1; iterations > 0; iterations--) { - // 0.99 is a experience parameter, ensure that each iterations of - // changes as small as possible. - alpha *= 0.99; - relaxRightToLeft(nodesByBreadth, alpha, orient); - resolveCollisions(nodesByBreadth, nodeGap, height, width, orient); - relaxLeftToRight(nodesByBreadth, alpha, orient); - resolveCollisions(nodesByBreadth, nodeGap, height, width, orient); - } - } - - function prepareNodesByBreadth(nodes, orient) { - var nodesByBreadth = []; - var keyAttr = orient === 'vertical' ? 'y' : 'x'; - var groupResult = groupData(nodes, function (node) { - return node.getLayout()[keyAttr]; - }); - groupResult.keys.sort(function (a, b) { - return a - b; - }); - each(groupResult.keys, function (key) { - nodesByBreadth.push(groupResult.buckets.get(key)); - }); - return nodesByBreadth; - } - /** - * Compute the original y-position for each node - */ - - - function initializeNodeDepth(nodesByBreadth, edges, height, width, nodeGap, orient) { - var minKy = Infinity; - each(nodesByBreadth, function (nodes) { - var n = nodes.length; - var sum = 0; - each(nodes, function (node) { - sum += node.getLayout().value; - }); - var ky = orient === 'vertical' ? (width - (n - 1) * nodeGap) / sum : (height - (n - 1) * nodeGap) / sum; - - if (ky < minKy) { - minKy = ky; - } - }); - each(nodesByBreadth, function (nodes) { - each(nodes, function (node, i) { - var nodeDy = node.getLayout().value * minKy; - - if (orient === 'vertical') { - node.setLayout({ - x: i - }, true); - node.setLayout({ - dx: nodeDy - }, true); - } else { - node.setLayout({ - y: i - }, true); - node.setLayout({ - dy: nodeDy - }, true); - } - }); - }); - each(edges, function (edge) { - var edgeDy = +edge.getValue() * minKy; - edge.setLayout({ - dy: edgeDy - }, true); - }); - } - /** - * Resolve the collision of initialized depth (y-position) - */ - - - function resolveCollisions(nodesByBreadth, nodeGap, height, width, orient) { - var keyAttr = orient === 'vertical' ? 'x' : 'y'; - each(nodesByBreadth, function (nodes) { - nodes.sort(function (a, b) { - return a.getLayout()[keyAttr] - b.getLayout()[keyAttr]; - }); - var nodeX; - var node; - var dy; - var y0 = 0; - var n = nodes.length; - var nodeDyAttr = orient === 'vertical' ? 'dx' : 'dy'; - - for (var i = 0; i < n; i++) { - node = nodes[i]; - dy = y0 - node.getLayout()[keyAttr]; - - if (dy > 0) { - nodeX = node.getLayout()[keyAttr] + dy; - orient === 'vertical' ? node.setLayout({ - x: nodeX - }, true) : node.setLayout({ - y: nodeX - }, true); - } - - y0 = node.getLayout()[keyAttr] + node.getLayout()[nodeDyAttr] + nodeGap; - } - - var viewWidth = orient === 'vertical' ? width : height; // If the bottommost node goes outside the bounds, push it back up - - dy = y0 - nodeGap - viewWidth; - - if (dy > 0) { - nodeX = node.getLayout()[keyAttr] - dy; - orient === 'vertical' ? node.setLayout({ - x: nodeX - }, true) : node.setLayout({ - y: nodeX - }, true); - y0 = nodeX; - - for (var i = n - 2; i >= 0; --i) { - node = nodes[i]; - dy = node.getLayout()[keyAttr] + node.getLayout()[nodeDyAttr] + nodeGap - y0; - - if (dy > 0) { - nodeX = node.getLayout()[keyAttr] - dy; - orient === 'vertical' ? node.setLayout({ - x: nodeX - }, true) : node.setLayout({ - y: nodeX - }, true); - } - - y0 = node.getLayout()[keyAttr]; - } - } - }); - } - /** - * Change the y-position of the nodes, except most the right side nodes - * @param nodesByBreadth - * @param alpha parameter used to adjust the nodes y-position - */ - - - function relaxRightToLeft(nodesByBreadth, alpha, orient) { - each(nodesByBreadth.slice().reverse(), function (nodes) { - each(nodes, function (node) { - if (node.outEdges.length) { - var y = sum(node.outEdges, weightedTarget, orient) / sum(node.outEdges, getEdgeValue); - - if (isNaN(y)) { - var len = node.outEdges.length; - y = len ? sum(node.outEdges, centerTarget, orient) / len : 0; - } - - if (orient === 'vertical') { - var nodeX = node.getLayout().x + (y - center$1(node, orient)) * alpha; - node.setLayout({ - x: nodeX - }, true); - } else { - var nodeY = node.getLayout().y + (y - center$1(node, orient)) * alpha; - node.setLayout({ - y: nodeY - }, true); - } - } - }); - }); - } - - function weightedTarget(edge, orient) { - return center$1(edge.node2, orient) * edge.getValue(); - } - - function centerTarget(edge, orient) { - return center$1(edge.node2, orient); - } - - function weightedSource(edge, orient) { - return center$1(edge.node1, orient) * edge.getValue(); - } - - function centerSource(edge, orient) { - return center$1(edge.node1, orient); - } - - function center$1(node, orient) { - return orient === 'vertical' ? node.getLayout().x + node.getLayout().dx / 2 : node.getLayout().y + node.getLayout().dy / 2; - } - - function getEdgeValue(edge) { - return edge.getValue(); - } - - function sum(array, cb, orient) { - var sum = 0; - var len = array.length; - var i = -1; - - while (++i < len) { - var value = +cb(array[i], orient); - - if (!isNaN(value)) { - sum += value; - } - } - - return sum; - } - /** - * Change the y-position of the nodes, except most the left side nodes - */ - - - function relaxLeftToRight(nodesByBreadth, alpha, orient) { - each(nodesByBreadth, function (nodes) { - each(nodes, function (node) { - if (node.inEdges.length) { - var y = sum(node.inEdges, weightedSource, orient) / sum(node.inEdges, getEdgeValue); - - if (isNaN(y)) { - var len = node.inEdges.length; - y = len ? sum(node.inEdges, centerSource, orient) / len : 0; - } - - if (orient === 'vertical') { - var nodeX = node.getLayout().x + (y - center$1(node, orient)) * alpha; - node.setLayout({ - x: nodeX - }, true); - } else { - var nodeY = node.getLayout().y + (y - center$1(node, orient)) * alpha; - node.setLayout({ - y: nodeY - }, true); - } - } - }); - }); - } - /** - * Compute the depth(y-position) of each edge - */ - - - function computeEdgeDepths(nodes, orient) { - var keyAttr = orient === 'vertical' ? 'x' : 'y'; - each(nodes, function (node) { - node.outEdges.sort(function (a, b) { - return a.node2.getLayout()[keyAttr] - b.node2.getLayout()[keyAttr]; - }); - node.inEdges.sort(function (a, b) { - return a.node1.getLayout()[keyAttr] - b.node1.getLayout()[keyAttr]; - }); - }); - each(nodes, function (node) { - var sy = 0; - var ty = 0; - each(node.outEdges, function (edge) { - edge.setLayout({ - sy: sy - }, true); - sy += edge.getLayout().dy; - }); - each(node.inEdges, function (edge) { - edge.setLayout({ - ty: ty - }, true); - ty += edge.getLayout().dy; - }); - }); - } - - function sankeyVisual(ecModel) { - ecModel.eachSeriesByType('sankey', function (seriesModel) { - var graph = seriesModel.getGraph(); - var nodes = graph.nodes; - var edges = graph.edges; - - if (nodes.length) { - var minValue_1 = Infinity; - var maxValue_1 = -Infinity; - each(nodes, function (node) { - var nodeValue = node.getLayout().value; - - if (nodeValue < minValue_1) { - minValue_1 = nodeValue; - } - - if (nodeValue > maxValue_1) { - maxValue_1 = nodeValue; - } - }); - each(nodes, function (node) { - var mapping = new VisualMapping({ - type: 'color', - mappingMethod: 'linear', - dataExtent: [minValue_1, maxValue_1], - visual: seriesModel.get('color') - }); - var mapValueToColor = mapping.mapValueToVisual(node.getLayout().value); - var customColor = node.getModel().get(['itemStyle', 'color']); - - if (customColor != null) { - node.setVisual('color', customColor); - node.setVisual('style', { - fill: customColor - }); - } else { - node.setVisual('color', mapValueToColor); - node.setVisual('style', { - fill: mapValueToColor - }); - } - }); - } - - if (edges.length) { - each(edges, function (edge) { - var edgeStyle = edge.getModel().get('lineStyle'); - edge.setVisual('style', edgeStyle); - }); - } - }); - } - - function install$i(registers) { - registers.registerChartView(SankeyView); - registers.registerSeriesModel(SankeySeriesModel); - registers.registerLayout(sankeyLayout); - registers.registerVisual(sankeyVisual); - registers.registerAction({ - type: 'dragNode', - event: 'dragnode', - // here can only use 'update' now, other value is not support in echarts. - update: 'update' - }, function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'series', - subType: 'sankey', - query: payload - }, function (seriesModel) { - seriesModel.setNodePosition(payload.dataIndex, [payload.localX, payload.localY]); - }); - }); - } - - var WhiskerBoxCommonMixin = - /** @class */ - function () { - function WhiskerBoxCommonMixin() {} - /** - * @override - */ - - - WhiskerBoxCommonMixin.prototype.getInitialData = function (option, ecModel) { - // When both types of xAxis and yAxis are 'value', layout is - // needed to be specified by user. Otherwise, layout can be - // judged by which axis is category. - var ordinalMeta; - var xAxisModel = ecModel.getComponent('xAxis', this.get('xAxisIndex')); - var yAxisModel = ecModel.getComponent('yAxis', this.get('yAxisIndex')); - var xAxisType = xAxisModel.get('type'); - var yAxisType = yAxisModel.get('type'); - var addOrdinal; // FIXME - // Consider time axis. - - if (xAxisType === 'category') { - option.layout = 'horizontal'; - ordinalMeta = xAxisModel.getOrdinalMeta(); - addOrdinal = true; - } else if (yAxisType === 'category') { - option.layout = 'vertical'; - ordinalMeta = yAxisModel.getOrdinalMeta(); - addOrdinal = true; - } else { - option.layout = option.layout || 'horizontal'; - } - - var coordDims = ['x', 'y']; - var baseAxisDimIndex = option.layout === 'horizontal' ? 0 : 1; - var baseAxisDim = this._baseAxisDim = coordDims[baseAxisDimIndex]; - var otherAxisDim = coordDims[1 - baseAxisDimIndex]; - var axisModels = [xAxisModel, yAxisModel]; - var baseAxisType = axisModels[baseAxisDimIndex].get('type'); - var otherAxisType = axisModels[1 - baseAxisDimIndex].get('type'); - var data = option.data; // Clone a new data for next setOption({}) usage. - // Avoid modifying current data will affect further update. - - if (data && addOrdinal) { - var newOptionData_1 = []; - each(data, function (item, index) { - var newItem; - - if (isArray(item)) { - newItem = item.slice(); // Modify current using data. - - item.unshift(index); - } else if (isArray(item.value)) { - newItem = extend({}, item); - newItem.value = newItem.value.slice(); // Modify current using data. - - item.value.unshift(index); - } else { - newItem = item; - } - - newOptionData_1.push(newItem); - }); - option.data = newOptionData_1; - } - - var defaultValueDimensions = this.defaultValueDimensions; - var coordDimensions = [{ - name: baseAxisDim, - type: getDimensionTypeByAxis(baseAxisType), - ordinalMeta: ordinalMeta, - otherDims: { - tooltip: false, - itemName: 0 - }, - dimsDef: ['base'] - }, { - name: otherAxisDim, - type: getDimensionTypeByAxis(otherAxisType), - dimsDef: defaultValueDimensions.slice() - }]; - return createSeriesDataSimply(this, { - coordDimensions: coordDimensions, - dimensionsCount: defaultValueDimensions.length + 1, - encodeDefaulter: curry(makeSeriesEncodeForAxisCoordSys, coordDimensions, this) - }); - }; - /** - * If horizontal, base axis is x, otherwise y. - * @override - */ - - - WhiskerBoxCommonMixin.prototype.getBaseAxis = function () { - var dim = this._baseAxisDim; - return this.ecModel.getComponent(dim + 'Axis', this.get(dim + 'AxisIndex')).axis; - }; - - return WhiskerBoxCommonMixin; - }(); - - var BoxplotSeriesModel = - /** @class */ - function (_super) { - __extends(BoxplotSeriesModel, _super); - - function BoxplotSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = BoxplotSeriesModel.type; // TODO - // box width represents group size, so dimension should have 'size'. - - /** - * @see <https://en.wikipedia.org/wiki/Box_plot> - * The meanings of 'min' and 'max' depend on user, - * and echarts do not need to know it. - * @readOnly - */ - - _this.defaultValueDimensions = [{ - name: 'min', - defaultTooltip: true - }, { - name: 'Q1', - defaultTooltip: true - }, { - name: 'median', - defaultTooltip: true - }, { - name: 'Q3', - defaultTooltip: true - }, { - name: 'max', - defaultTooltip: true - }]; - _this.visualDrawType = 'stroke'; - return _this; - } - - BoxplotSeriesModel.type = 'series.boxplot'; - BoxplotSeriesModel.dependencies = ['xAxis', 'yAxis', 'grid']; - BoxplotSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'cartesian2d', - legendHoverLink: true, - layout: null, - boxWidth: [7, 50], - itemStyle: { - color: '#fff', - borderWidth: 1 - }, - emphasis: { - scale: true, - itemStyle: { - borderWidth: 2, - shadowBlur: 5, - shadowOffsetX: 1, - shadowOffsetY: 1, - shadowColor: 'rgba(0,0,0,0.2)' - } - }, - animationDuration: 800 - }; - return BoxplotSeriesModel; - }(SeriesModel); - - mixin(BoxplotSeriesModel, WhiskerBoxCommonMixin, true); - - var BoxplotView = - /** @class */ - function (_super) { - __extends(BoxplotView, _super); - - function BoxplotView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = BoxplotView.type; - return _this; - } - - BoxplotView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var group = this.group; - var oldData = this._data; // There is no old data only when first rendering or switching from - // stream mode to normal mode, where previous elements should be removed. - - if (!this._data) { - group.removeAll(); - } - - var constDim = seriesModel.get('layout') === 'horizontal' ? 1 : 0; - data.diff(oldData).add(function (newIdx) { - if (data.hasValue(newIdx)) { - var itemLayout = data.getItemLayout(newIdx); - var symbolEl = createNormalBox(itemLayout, data, newIdx, constDim, true); - data.setItemGraphicEl(newIdx, symbolEl); - group.add(symbolEl); - } - }).update(function (newIdx, oldIdx) { - var symbolEl = oldData.getItemGraphicEl(oldIdx); // Empty data - - if (!data.hasValue(newIdx)) { - group.remove(symbolEl); - return; - } - - var itemLayout = data.getItemLayout(newIdx); - - if (!symbolEl) { - symbolEl = createNormalBox(itemLayout, data, newIdx, constDim); - } else { - saveOldStyle(symbolEl); - updateNormalBoxData(itemLayout, symbolEl, data, newIdx); - } - - group.add(symbolEl); - data.setItemGraphicEl(newIdx, symbolEl); - }).remove(function (oldIdx) { - var el = oldData.getItemGraphicEl(oldIdx); - el && group.remove(el); - }).execute(); - this._data = data; - }; - - BoxplotView.prototype.remove = function (ecModel) { - var group = this.group; - var data = this._data; - this._data = null; - data && data.eachItemGraphicEl(function (el) { - el && group.remove(el); - }); - }; - - BoxplotView.type = 'boxplot'; - return BoxplotView; - }(ChartView); - - var BoxPathShape = - /** @class */ - function () { - function BoxPathShape() {} - - return BoxPathShape; - }(); - - var BoxPath = - /** @class */ - function (_super) { - __extends(BoxPath, _super); - - function BoxPath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'boxplotBoxPath'; - return _this; - } - - BoxPath.prototype.getDefaultShape = function () { - return new BoxPathShape(); - }; - - BoxPath.prototype.buildPath = function (ctx, shape) { - var ends = shape.points; - var i = 0; - ctx.moveTo(ends[i][0], ends[i][1]); - i++; - - for (; i < 4; i++) { - ctx.lineTo(ends[i][0], ends[i][1]); - } - - ctx.closePath(); - - for (; i < ends.length; i++) { - ctx.moveTo(ends[i][0], ends[i][1]); - i++; - ctx.lineTo(ends[i][0], ends[i][1]); - } - }; - - return BoxPath; - }(Path); - - function createNormalBox(itemLayout, data, dataIndex, constDim, isInit) { - var ends = itemLayout.ends; - var el = new BoxPath({ - shape: { - points: isInit ? transInit(ends, constDim, itemLayout) : ends - } - }); - updateNormalBoxData(itemLayout, el, data, dataIndex, isInit); - return el; - } - - function updateNormalBoxData(itemLayout, el, data, dataIndex, isInit) { - var seriesModel = data.hostModel; - var updateMethod = graphic[isInit ? 'initProps' : 'updateProps']; - updateMethod(el, { - shape: { - points: itemLayout.ends - } - }, seriesModel, dataIndex); - el.useStyle(data.getItemVisual(dataIndex, 'style')); - el.style.strokeNoScale = true; - el.z2 = 100; - var itemModel = data.getItemModel(dataIndex); - var emphasisModel = itemModel.getModel('emphasis'); - setStatesStylesFromModel(el, itemModel); - toggleHoverEmphasis(el, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - } - - function transInit(points, dim, itemLayout) { - return map(points, function (point) { - point = point.slice(); - point[dim] = itemLayout.initBaseline; - return point; - }); - } - - var each$6 = each; - function boxplotLayout(ecModel) { - var groupResult = groupSeriesByAxis(ecModel); - each$6(groupResult, function (groupItem) { - var seriesModels = groupItem.seriesModels; - - if (!seriesModels.length) { - return; - } - - calculateBase(groupItem); - each$6(seriesModels, function (seriesModel, idx) { - layoutSingleSeries(seriesModel, groupItem.boxOffsetList[idx], groupItem.boxWidthList[idx]); - }); - }); - } - /** - * Group series by axis. - */ - - function groupSeriesByAxis(ecModel) { - var result = []; - var axisList = []; - ecModel.eachSeriesByType('boxplot', function (seriesModel) { - var baseAxis = seriesModel.getBaseAxis(); - var idx = indexOf(axisList, baseAxis); - - if (idx < 0) { - idx = axisList.length; - axisList[idx] = baseAxis; - result[idx] = { - axis: baseAxis, - seriesModels: [] - }; - } - - result[idx].seriesModels.push(seriesModel); - }); - return result; - } - /** - * Calculate offset and box width for each series. - */ - - - function calculateBase(groupItem) { - var baseAxis = groupItem.axis; - var seriesModels = groupItem.seriesModels; - var seriesCount = seriesModels.length; - var boxWidthList = groupItem.boxWidthList = []; - var boxOffsetList = groupItem.boxOffsetList = []; - var boundList = []; - var bandWidth; - - if (baseAxis.type === 'category') { - bandWidth = baseAxis.getBandWidth(); - } else { - var maxDataCount_1 = 0; - each$6(seriesModels, function (seriesModel) { - maxDataCount_1 = Math.max(maxDataCount_1, seriesModel.getData().count()); - }); - var extent = baseAxis.getExtent(); - bandWidth = Math.abs(extent[1] - extent[0]) / maxDataCount_1; - } - - each$6(seriesModels, function (seriesModel) { - var boxWidthBound = seriesModel.get('boxWidth'); - - if (!isArray(boxWidthBound)) { - boxWidthBound = [boxWidthBound, boxWidthBound]; - } - - boundList.push([parsePercent$1(boxWidthBound[0], bandWidth) || 0, parsePercent$1(boxWidthBound[1], bandWidth) || 0]); - }); - var availableWidth = bandWidth * 0.8 - 2; - var boxGap = availableWidth / seriesCount * 0.3; - var boxWidth = (availableWidth - boxGap * (seriesCount - 1)) / seriesCount; - var base = boxWidth / 2 - availableWidth / 2; - each$6(seriesModels, function (seriesModel, idx) { - boxOffsetList.push(base); - base += boxGap + boxWidth; - boxWidthList.push(Math.min(Math.max(boxWidth, boundList[idx][0]), boundList[idx][1])); - }); - } - /** - * Calculate points location for each series. - */ - - - function layoutSingleSeries(seriesModel, offset, boxWidth) { - var coordSys = seriesModel.coordinateSystem; - var data = seriesModel.getData(); - var halfWidth = boxWidth / 2; - var cDimIdx = seriesModel.get('layout') === 'horizontal' ? 0 : 1; - var vDimIdx = 1 - cDimIdx; - var coordDims = ['x', 'y']; - var cDim = data.mapDimension(coordDims[cDimIdx]); - var vDims = data.mapDimensionsAll(coordDims[vDimIdx]); - - if (cDim == null || vDims.length < 5) { - return; - } - - for (var dataIndex = 0; dataIndex < data.count(); dataIndex++) { - var axisDimVal = data.get(cDim, dataIndex); - var median = getPoint(axisDimVal, vDims[2], dataIndex); - var end1 = getPoint(axisDimVal, vDims[0], dataIndex); - var end2 = getPoint(axisDimVal, vDims[1], dataIndex); - var end4 = getPoint(axisDimVal, vDims[3], dataIndex); - var end5 = getPoint(axisDimVal, vDims[4], dataIndex); - var ends = []; - addBodyEnd(ends, end2, false); - addBodyEnd(ends, end4, true); - ends.push(end1, end2, end5, end4); - layEndLine(ends, end1); - layEndLine(ends, end5); - layEndLine(ends, median); - data.setItemLayout(dataIndex, { - initBaseline: median[vDimIdx], - ends: ends - }); - } - - function getPoint(axisDimVal, dim, dataIndex) { - var val = data.get(dim, dataIndex); - var p = []; - p[cDimIdx] = axisDimVal; - p[vDimIdx] = val; - var point; - - if (isNaN(axisDimVal) || isNaN(val)) { - point = [NaN, NaN]; - } else { - point = coordSys.dataToPoint(p); - point[cDimIdx] += offset; - } - - return point; - } - - function addBodyEnd(ends, point, start) { - var point1 = point.slice(); - var point2 = point.slice(); - point1[cDimIdx] += halfWidth; - point2[cDimIdx] -= halfWidth; - start ? ends.push(point1, point2) : ends.push(point2, point1); - } - - function layEndLine(ends, endCenter) { - var from = endCenter.slice(); - var to = endCenter.slice(); - from[cDimIdx] -= halfWidth; - to[cDimIdx] += halfWidth; - ends.push(from, to); - } - } - - /** - * See: - * <https://en.wikipedia.org/wiki/Box_plot#cite_note-frigge_hoaglin_iglewicz-2> - * <http://stat.ethz.ch/R-manual/R-devel/library/grDevices/html/boxplot.stats.html> - * - * Helper method for preparing data. - * - * @param rawData like - * [ - * [12,232,443], (raw data set for the first box) - * [3843,5545,1232], (raw data set for the second box) - * ... - * ] - * @param opt.boundIQR=1.5 Data less than min bound is outlier. - * default 1.5, means Q1 - 1.5 * (Q3 - Q1). - * If 'none'/0 passed, min bound will not be used. - */ - - function prepareBoxplotData(rawData, opt) { - opt = opt || {}; - var boxData = []; - var outliers = []; - var boundIQR = opt.boundIQR; - var useExtreme = boundIQR === 'none' || boundIQR === 0; - - for (var i = 0; i < rawData.length; i++) { - var ascList = asc(rawData[i].slice()); - var Q1 = quantile(ascList, 0.25); - var Q2 = quantile(ascList, 0.5); - var Q3 = quantile(ascList, 0.75); - var min = ascList[0]; - var max = ascList[ascList.length - 1]; - var bound = (boundIQR == null ? 1.5 : boundIQR) * (Q3 - Q1); - var low = useExtreme ? min : Math.max(min, Q1 - bound); - var high = useExtreme ? max : Math.min(max, Q3 + bound); - var itemNameFormatter = opt.itemNameFormatter; - var itemName = isFunction(itemNameFormatter) ? itemNameFormatter({ - value: i - }) : isString(itemNameFormatter) ? itemNameFormatter.replace('{value}', i + '') : i + ''; - boxData.push([itemName, low, Q1, Q2, Q3, high]); - - for (var j = 0; j < ascList.length; j++) { - var dataItem = ascList[j]; - - if (dataItem < low || dataItem > high) { - var outlier = [itemName, dataItem]; - outliers.push(outlier); - } - } - } - - return { - boxData: boxData, - outliers: outliers - }; - } - - var boxplotTransform = { - type: 'echarts:boxplot', - transform: function transform(params) { - var upstream = params.upstream; - - if (upstream.sourceFormat !== SOURCE_FORMAT_ARRAY_ROWS) { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = makePrintable('source data is not applicable for this boxplot transform. Expect number[][].'); - } - - throwError(errMsg); - } - - var result = prepareBoxplotData(upstream.getRawData(), params.config); - return [{ - dimensions: ['ItemName', 'Low', 'Q1', 'Q2', 'Q3', 'High'], - data: result.boxData - }, { - data: result.outliers - }]; - } - }; - - function install$j(registers) { - registers.registerSeriesModel(BoxplotSeriesModel); - registers.registerChartView(BoxplotView); - registers.registerLayout(boxplotLayout); - registers.registerTransform(boxplotTransform); - } - - var SKIP_PROPS = ['color', 'borderColor']; - - var CandlestickView = - /** @class */ - function (_super) { - __extends(CandlestickView, _super); - - function CandlestickView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CandlestickView.type; - return _this; - } - - CandlestickView.prototype.render = function (seriesModel, ecModel, api) { - // If there is clipPath created in large mode. Remove it. - this.group.removeClipPath(); // Clear previously rendered progressive elements. - - this._progressiveEls = null; - - this._updateDrawMode(seriesModel); - - this._isLargeDraw ? this._renderLarge(seriesModel) : this._renderNormal(seriesModel); - }; - - CandlestickView.prototype.incrementalPrepareRender = function (seriesModel, ecModel, api) { - this._clear(); - - this._updateDrawMode(seriesModel); - }; - - CandlestickView.prototype.incrementalRender = function (params, seriesModel, ecModel, api) { - this._progressiveEls = []; - this._isLargeDraw ? this._incrementalRenderLarge(params, seriesModel) : this._incrementalRenderNormal(params, seriesModel); - }; - - CandlestickView.prototype.eachRendered = function (cb) { - traverseElements(this._progressiveEls || this.group, cb); - }; - - CandlestickView.prototype._updateDrawMode = function (seriesModel) { - var isLargeDraw = seriesModel.pipelineContext.large; - - if (this._isLargeDraw == null || isLargeDraw !== this._isLargeDraw) { - this._isLargeDraw = isLargeDraw; - - this._clear(); - } - }; - - CandlestickView.prototype._renderNormal = function (seriesModel) { - var data = seriesModel.getData(); - var oldData = this._data; - var group = this.group; - var isSimpleBox = data.getLayout('isSimpleBox'); - var needsClip = seriesModel.get('clip', true); - var coord = seriesModel.coordinateSystem; - var clipArea = coord.getArea && coord.getArea(); // There is no old data only when first rendering or switching from - // stream mode to normal mode, where previous elements should be removed. - - if (!this._data) { - group.removeAll(); - } - - data.diff(oldData).add(function (newIdx) { - if (data.hasValue(newIdx)) { - var itemLayout = data.getItemLayout(newIdx); - - if (needsClip && isNormalBoxClipped(clipArea, itemLayout)) { - return; - } - - var el = createNormalBox$1(itemLayout, newIdx, true); - initProps(el, { - shape: { - points: itemLayout.ends - } - }, seriesModel, newIdx); - setBoxCommon(el, data, newIdx, isSimpleBox); - group.add(el); - data.setItemGraphicEl(newIdx, el); - } - }).update(function (newIdx, oldIdx) { - var el = oldData.getItemGraphicEl(oldIdx); // Empty data - - if (!data.hasValue(newIdx)) { - group.remove(el); - return; - } - - var itemLayout = data.getItemLayout(newIdx); - - if (needsClip && isNormalBoxClipped(clipArea, itemLayout)) { - group.remove(el); - return; - } - - if (!el) { - el = createNormalBox$1(itemLayout); - } else { - updateProps(el, { - shape: { - points: itemLayout.ends - } - }, seriesModel, newIdx); - saveOldStyle(el); - } - - setBoxCommon(el, data, newIdx, isSimpleBox); - group.add(el); - data.setItemGraphicEl(newIdx, el); - }).remove(function (oldIdx) { - var el = oldData.getItemGraphicEl(oldIdx); - el && group.remove(el); - }).execute(); - this._data = data; - }; - - CandlestickView.prototype._renderLarge = function (seriesModel) { - this._clear(); - - createLarge$1(seriesModel, this.group); - var clipPath = seriesModel.get('clip', true) ? createClipPath(seriesModel.coordinateSystem, false, seriesModel) : null; - - if (clipPath) { - this.group.setClipPath(clipPath); - } else { - this.group.removeClipPath(); - } - }; - - CandlestickView.prototype._incrementalRenderNormal = function (params, seriesModel) { - var data = seriesModel.getData(); - var isSimpleBox = data.getLayout('isSimpleBox'); - var dataIndex; - - while ((dataIndex = params.next()) != null) { - var itemLayout = data.getItemLayout(dataIndex); - var el = createNormalBox$1(itemLayout); - setBoxCommon(el, data, dataIndex, isSimpleBox); - el.incremental = true; - this.group.add(el); - - this._progressiveEls.push(el); - } - }; - - CandlestickView.prototype._incrementalRenderLarge = function (params, seriesModel) { - createLarge$1(seriesModel, this.group, this._progressiveEls, true); - }; - - CandlestickView.prototype.remove = function (ecModel) { - this._clear(); - }; - - CandlestickView.prototype._clear = function () { - this.group.removeAll(); - this._data = null; - }; - - CandlestickView.type = 'candlestick'; - return CandlestickView; - }(ChartView); - - var NormalBoxPathShape = - /** @class */ - function () { - function NormalBoxPathShape() {} - - return NormalBoxPathShape; - }(); - - var NormalBoxPath = - /** @class */ - function (_super) { - __extends(NormalBoxPath, _super); - - function NormalBoxPath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'normalCandlestickBox'; - return _this; - } - - NormalBoxPath.prototype.getDefaultShape = function () { - return new NormalBoxPathShape(); - }; - - NormalBoxPath.prototype.buildPath = function (ctx, shape) { - var ends = shape.points; - - if (this.__simpleBox) { - ctx.moveTo(ends[4][0], ends[4][1]); - ctx.lineTo(ends[6][0], ends[6][1]); - } else { - ctx.moveTo(ends[0][0], ends[0][1]); - ctx.lineTo(ends[1][0], ends[1][1]); - ctx.lineTo(ends[2][0], ends[2][1]); - ctx.lineTo(ends[3][0], ends[3][1]); - ctx.closePath(); - ctx.moveTo(ends[4][0], ends[4][1]); - ctx.lineTo(ends[5][0], ends[5][1]); - ctx.moveTo(ends[6][0], ends[6][1]); - ctx.lineTo(ends[7][0], ends[7][1]); - } - }; - - return NormalBoxPath; - }(Path); - - function createNormalBox$1(itemLayout, dataIndex, isInit) { - var ends = itemLayout.ends; - return new NormalBoxPath({ - shape: { - points: isInit ? transInit$1(ends, itemLayout) : ends - }, - z2: 100 - }); - } - - function isNormalBoxClipped(clipArea, itemLayout) { - var clipped = true; - - for (var i = 0; i < itemLayout.ends.length; i++) { - // If any point are in the region. - if (clipArea.contain(itemLayout.ends[i][0], itemLayout.ends[i][1])) { - clipped = false; - break; - } - } - - return clipped; - } - - function setBoxCommon(el, data, dataIndex, isSimpleBox) { - var itemModel = data.getItemModel(dataIndex); - el.useStyle(data.getItemVisual(dataIndex, 'style')); - el.style.strokeNoScale = true; - el.__simpleBox = isSimpleBox; - setStatesStylesFromModel(el, itemModel); - } - - function transInit$1(points, itemLayout) { - return map(points, function (point) { - point = point.slice(); - point[1] = itemLayout.initBaseline; - return point; - }); - } - - var LargeBoxPathShape = - /** @class */ - function () { - function LargeBoxPathShape() {} - - return LargeBoxPathShape; - }(); - - var LargeBoxPath = - /** @class */ - function (_super) { - __extends(LargeBoxPath, _super); - - function LargeBoxPath(opts) { - var _this = _super.call(this, opts) || this; - - _this.type = 'largeCandlestickBox'; - return _this; - } - - LargeBoxPath.prototype.getDefaultShape = function () { - return new LargeBoxPathShape(); - }; - - LargeBoxPath.prototype.buildPath = function (ctx, shape) { - // Drawing lines is more efficient than drawing - // a whole line or drawing rects. - var points = shape.points; - - for (var i = 0; i < points.length;) { - if (this.__sign === points[i++]) { - var x = points[i++]; - ctx.moveTo(x, points[i++]); - ctx.lineTo(x, points[i++]); - } else { - i += 3; - } - } - }; - - return LargeBoxPath; - }(Path); - - function createLarge$1(seriesModel, group, progressiveEls, incremental) { - var data = seriesModel.getData(); - var largePoints = data.getLayout('largePoints'); - var elP = new LargeBoxPath({ - shape: { - points: largePoints - }, - __sign: 1, - ignoreCoarsePointer: true - }); - group.add(elP); - var elN = new LargeBoxPath({ - shape: { - points: largePoints - }, - __sign: -1, - ignoreCoarsePointer: true - }); - group.add(elN); - var elDoji = new LargeBoxPath({ - shape: { - points: largePoints - }, - __sign: 0, - ignoreCoarsePointer: true - }); - group.add(elDoji); - setLargeStyle(1, elP, seriesModel); - setLargeStyle(-1, elN, seriesModel); - setLargeStyle(0, elDoji, seriesModel); - - if (incremental) { - elP.incremental = true; - elN.incremental = true; - } - - if (progressiveEls) { - progressiveEls.push(elP, elN); - } - } - - function setLargeStyle(sign, el, seriesModel, data) { - // TODO put in visual? - var borderColor = seriesModel.get(['itemStyle', sign > 0 ? 'borderColor' : 'borderColor0']) // Use color for border color by default. - || seriesModel.get(['itemStyle', sign > 0 ? 'color' : 'color0']); - - if (sign === 0) { - borderColor = seriesModel.get(['itemStyle', 'borderColorDoji']); - } // Color must be excluded. - // Because symbol provide setColor individually to set fill and stroke - - - var itemStyle = seriesModel.getModel('itemStyle').getItemStyle(SKIP_PROPS); - el.useStyle(itemStyle); - el.style.fill = null; - el.style.stroke = borderColor; - } - - var CandlestickSeriesModel = - /** @class */ - function (_super) { - __extends(CandlestickSeriesModel, _super); - - function CandlestickSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CandlestickSeriesModel.type; - _this.defaultValueDimensions = [{ - name: 'open', - defaultTooltip: true - }, { - name: 'close', - defaultTooltip: true - }, { - name: 'lowest', - defaultTooltip: true - }, { - name: 'highest', - defaultTooltip: true - }]; - return _this; - } - /** - * Get dimension for shadow in dataZoom - * @return dimension name - */ - - - CandlestickSeriesModel.prototype.getShadowDim = function () { - return 'open'; - }; - - CandlestickSeriesModel.prototype.brushSelector = function (dataIndex, data, selectors) { - var itemLayout = data.getItemLayout(dataIndex); - return itemLayout && selectors.rect(itemLayout.brushRect); - }; - - CandlestickSeriesModel.type = 'series.candlestick'; - CandlestickSeriesModel.dependencies = ['xAxis', 'yAxis', 'grid']; - CandlestickSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - coordinateSystem: 'cartesian2d', - legendHoverLink: true, - // xAxisIndex: 0, - // yAxisIndex: 0, - layout: null, - clip: true, - itemStyle: { - color: '#eb5454', - color0: '#47b262', - borderColor: '#eb5454', - borderColor0: '#47b262', - borderColorDoji: null, - // borderColor: '#d24040', - // borderColor0: '#398f4f', - borderWidth: 1 - }, - emphasis: { - scale: true, - itemStyle: { - borderWidth: 2 - } - }, - barMaxWidth: null, - barMinWidth: null, - barWidth: null, - large: true, - largeThreshold: 600, - progressive: 3e3, - progressiveThreshold: 1e4, - progressiveChunkMode: 'mod', - animationEasing: 'linear', - animationDuration: 300 - }; - return CandlestickSeriesModel; - }(SeriesModel); - - mixin(CandlestickSeriesModel, WhiskerBoxCommonMixin, true); - - function candlestickPreprocessor(option) { - if (!option || !isArray(option.series)) { - return; - } // Translate 'k' to 'candlestick'. - - - each(option.series, function (seriesItem) { - if (isObject(seriesItem) && seriesItem.type === 'k') { - seriesItem.type = 'candlestick'; - } - }); - } - - var positiveBorderColorQuery = ['itemStyle', 'borderColor']; - var negativeBorderColorQuery = ['itemStyle', 'borderColor0']; - var dojiBorderColorQuery = ['itemStyle', 'borderColorDoji']; - var positiveColorQuery = ['itemStyle', 'color']; - var negativeColorQuery = ['itemStyle', 'color0']; - var candlestickVisual = { - seriesType: 'candlestick', - plan: createRenderPlanner(), - // For legend. - performRawSeries: true, - reset: function (seriesModel, ecModel) { - function getColor(sign, model) { - return model.get(sign > 0 ? positiveColorQuery : negativeColorQuery); - } - - function getBorderColor(sign, model) { - return model.get(sign === 0 ? dojiBorderColorQuery : sign > 0 ? positiveBorderColorQuery : negativeBorderColorQuery); - } // Only visible series has each data be visual encoded - - - if (ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - var isLargeRender = seriesModel.pipelineContext.large; - return !isLargeRender && { - progress: function (params, data) { - var dataIndex; - - while ((dataIndex = params.next()) != null) { - var itemModel = data.getItemModel(dataIndex); - var sign = data.getItemLayout(dataIndex).sign; - var style = itemModel.getItemStyle(); - style.fill = getColor(sign, itemModel); - style.stroke = getBorderColor(sign, itemModel) || style.fill; - var existsStyle = data.ensureUniqueItemVisual(dataIndex, 'style'); - extend(existsStyle, style); - } - } - }; - } - }; - - var candlestickLayout = { - seriesType: 'candlestick', - plan: createRenderPlanner(), - reset: function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - var data = seriesModel.getData(); - var candleWidth = calculateCandleWidth(seriesModel, data); - var cDimIdx = 0; - var vDimIdx = 1; - var coordDims = ['x', 'y']; - var cDimI = data.getDimensionIndex(data.mapDimension(coordDims[cDimIdx])); - var vDimsI = map(data.mapDimensionsAll(coordDims[vDimIdx]), data.getDimensionIndex, data); - var openDimI = vDimsI[0]; - var closeDimI = vDimsI[1]; - var lowestDimI = vDimsI[2]; - var highestDimI = vDimsI[3]; - data.setLayout({ - candleWidth: candleWidth, - // The value is experimented visually. - isSimpleBox: candleWidth <= 1.3 - }); - - if (cDimI < 0 || vDimsI.length < 4) { - return; - } - - return { - progress: seriesModel.pipelineContext.large ? largeProgress : normalProgress - }; - - function normalProgress(params, data) { - var dataIndex; - var store = data.getStore(); - - while ((dataIndex = params.next()) != null) { - var axisDimVal = store.get(cDimI, dataIndex); - var openVal = store.get(openDimI, dataIndex); - var closeVal = store.get(closeDimI, dataIndex); - var lowestVal = store.get(lowestDimI, dataIndex); - var highestVal = store.get(highestDimI, dataIndex); - var ocLow = Math.min(openVal, closeVal); - var ocHigh = Math.max(openVal, closeVal); - var ocLowPoint = getPoint(ocLow, axisDimVal); - var ocHighPoint = getPoint(ocHigh, axisDimVal); - var lowestPoint = getPoint(lowestVal, axisDimVal); - var highestPoint = getPoint(highestVal, axisDimVal); - var ends = []; - addBodyEnd(ends, ocHighPoint, 0); - addBodyEnd(ends, ocLowPoint, 1); - ends.push(subPixelOptimizePoint(highestPoint), subPixelOptimizePoint(ocHighPoint), subPixelOptimizePoint(lowestPoint), subPixelOptimizePoint(ocLowPoint)); - var itemModel = data.getItemModel(dataIndex); - var hasDojiColor = !!itemModel.get(['itemStyle', 'borderColorDoji']); - data.setItemLayout(dataIndex, { - sign: getSign(store, dataIndex, openVal, closeVal, closeDimI, hasDojiColor), - initBaseline: openVal > closeVal ? ocHighPoint[vDimIdx] : ocLowPoint[vDimIdx], - ends: ends, - brushRect: makeBrushRect(lowestVal, highestVal, axisDimVal) - }); - } - - function getPoint(val, axisDimVal) { - var p = []; - p[cDimIdx] = axisDimVal; - p[vDimIdx] = val; - return isNaN(axisDimVal) || isNaN(val) ? [NaN, NaN] : coordSys.dataToPoint(p); - } - - function addBodyEnd(ends, point, start) { - var point1 = point.slice(); - var point2 = point.slice(); - point1[cDimIdx] = subPixelOptimize$1(point1[cDimIdx] + candleWidth / 2, 1, false); - point2[cDimIdx] = subPixelOptimize$1(point2[cDimIdx] - candleWidth / 2, 1, true); - start ? ends.push(point1, point2) : ends.push(point2, point1); - } - - function makeBrushRect(lowestVal, highestVal, axisDimVal) { - var pmin = getPoint(lowestVal, axisDimVal); - var pmax = getPoint(highestVal, axisDimVal); - pmin[cDimIdx] -= candleWidth / 2; - pmax[cDimIdx] -= candleWidth / 2; - return { - x: pmin[0], - y: pmin[1], - width: candleWidth , - height: pmax[1] - pmin[1] - }; - } - - function subPixelOptimizePoint(point) { - point[cDimIdx] = subPixelOptimize$1(point[cDimIdx], 1); - return point; - } - } - - function largeProgress(params, data) { - // Structure: [sign, x, yhigh, ylow, sign, x, yhigh, ylow, ...] - var points = createFloat32Array(params.count * 4); - var offset = 0; - var point; - var tmpIn = []; - var tmpOut = []; - var dataIndex; - var store = data.getStore(); - var hasDojiColor = !!seriesModel.get(['itemStyle', 'borderColorDoji']); - - while ((dataIndex = params.next()) != null) { - var axisDimVal = store.get(cDimI, dataIndex); - var openVal = store.get(openDimI, dataIndex); - var closeVal = store.get(closeDimI, dataIndex); - var lowestVal = store.get(lowestDimI, dataIndex); - var highestVal = store.get(highestDimI, dataIndex); - - if (isNaN(axisDimVal) || isNaN(lowestVal) || isNaN(highestVal)) { - points[offset++] = NaN; - offset += 3; - continue; - } - - points[offset++] = getSign(store, dataIndex, openVal, closeVal, closeDimI, hasDojiColor); - tmpIn[cDimIdx] = axisDimVal; - tmpIn[vDimIdx] = lowestVal; - point = coordSys.dataToPoint(tmpIn, null, tmpOut); - points[offset++] = point ? point[0] : NaN; - points[offset++] = point ? point[1] : NaN; - tmpIn[vDimIdx] = highestVal; - point = coordSys.dataToPoint(tmpIn, null, tmpOut); - points[offset++] = point ? point[1] : NaN; - } - - data.setLayout('largePoints', points); - } - } - }; - /** - * Get the sign of a single data. - * - * @returns 0 for doji with hasDojiColor: true, - * 1 for positive, - * -1 for negative. - */ - - function getSign(store, dataIndex, openVal, closeVal, closeDimI, hasDojiColor) { - var sign; - - if (openVal > closeVal) { - sign = -1; - } else if (openVal < closeVal) { - sign = 1; - } else { - sign = hasDojiColor // When doji color is set, use it instead of color/color0. - ? 0 : dataIndex > 0 // If close === open, compare with close of last record - ? store.get(closeDimI, dataIndex - 1) <= closeVal ? 1 : -1 : // No record of previous, set to be positive - 1; - } - - return sign; - } - - function calculateCandleWidth(seriesModel, data) { - var baseAxis = seriesModel.getBaseAxis(); - var extent; - var bandWidth = baseAxis.type === 'category' ? baseAxis.getBandWidth() : (extent = baseAxis.getExtent(), Math.abs(extent[1] - extent[0]) / data.count()); - var barMaxWidth = parsePercent$1(retrieve2(seriesModel.get('barMaxWidth'), bandWidth), bandWidth); - var barMinWidth = parsePercent$1(retrieve2(seriesModel.get('barMinWidth'), 1), bandWidth); - var barWidth = seriesModel.get('barWidth'); - return barWidth != null ? parsePercent$1(barWidth, bandWidth) // Put max outer to ensure bar visible in spite of overlap. - : Math.max(Math.min(bandWidth / 2, barMaxWidth), barMinWidth); - } - - function install$k(registers) { - registers.registerChartView(CandlestickView); - registers.registerSeriesModel(CandlestickSeriesModel); - registers.registerPreprocessor(candlestickPreprocessor); - registers.registerVisual(candlestickVisual); - registers.registerLayout(candlestickLayout); - } - - function updateRipplePath(rippleGroup, effectCfg) { - var color = effectCfg.rippleEffectColor || effectCfg.color; - rippleGroup.eachChild(function (ripplePath) { - ripplePath.attr({ - z: effectCfg.z, - zlevel: effectCfg.zlevel, - style: { - stroke: effectCfg.brushType === 'stroke' ? color : null, - fill: effectCfg.brushType === 'fill' ? color : null - } - }); - }); - } - - var EffectSymbol = - /** @class */ - function (_super) { - __extends(EffectSymbol, _super); - - function EffectSymbol(data, idx) { - var _this = _super.call(this) || this; - - var symbol = new Symbol(data, idx); - var rippleGroup = new Group(); - - _this.add(symbol); - - _this.add(rippleGroup); - - _this.updateData(data, idx); - - return _this; - } - - EffectSymbol.prototype.stopEffectAnimation = function () { - this.childAt(1).removeAll(); - }; - - EffectSymbol.prototype.startEffectAnimation = function (effectCfg) { - var symbolType = effectCfg.symbolType; - var color = effectCfg.color; - var rippleNumber = effectCfg.rippleNumber; - var rippleGroup = this.childAt(1); - - for (var i = 0; i < rippleNumber; i++) { - // If width/height are set too small (e.g., set to 1) on ios10 - // and macOS Sierra, a circle stroke become a rect, no matter what - // the scale is set. So we set width/height as 2. See #4136. - var ripplePath = createSymbol(symbolType, -1, -1, 2, 2, color); - ripplePath.attr({ - style: { - strokeNoScale: true - }, - z2: 99, - silent: true, - scaleX: 0.5, - scaleY: 0.5 - }); - var delay = -i / rippleNumber * effectCfg.period + effectCfg.effectOffset; - ripplePath.animate('', true).when(effectCfg.period, { - scaleX: effectCfg.rippleScale / 2, - scaleY: effectCfg.rippleScale / 2 - }).delay(delay).start(); - ripplePath.animateStyle(true).when(effectCfg.period, { - opacity: 0 - }).delay(delay).start(); - rippleGroup.add(ripplePath); - } - - updateRipplePath(rippleGroup, effectCfg); - }; - /** - * Update effect symbol - */ - - - EffectSymbol.prototype.updateEffectAnimation = function (effectCfg) { - var oldEffectCfg = this._effectCfg; - var rippleGroup = this.childAt(1); // Must reinitialize effect if following configuration changed - - var DIFFICULT_PROPS = ['symbolType', 'period', 'rippleScale', 'rippleNumber']; - - for (var i = 0; i < DIFFICULT_PROPS.length; i++) { - var propName = DIFFICULT_PROPS[i]; - - if (oldEffectCfg[propName] !== effectCfg[propName]) { - this.stopEffectAnimation(); - this.startEffectAnimation(effectCfg); - return; - } - } - - updateRipplePath(rippleGroup, effectCfg); - }; - /** - * Highlight symbol - */ - - - EffectSymbol.prototype.highlight = function () { - enterEmphasis(this); - }; - /** - * Downplay symbol - */ - - - EffectSymbol.prototype.downplay = function () { - leaveEmphasis(this); - }; - - EffectSymbol.prototype.getSymbolType = function () { - var symbol = this.childAt(0); - return symbol && symbol.getSymbolType(); - }; - /** - * Update symbol properties - */ - - - EffectSymbol.prototype.updateData = function (data, idx) { - var _this = this; - - var seriesModel = data.hostModel; - this.childAt(0).updateData(data, idx); - var rippleGroup = this.childAt(1); - var itemModel = data.getItemModel(idx); - var symbolType = data.getItemVisual(idx, 'symbol'); - var symbolSize = normalizeSymbolSize(data.getItemVisual(idx, 'symbolSize')); - var symbolStyle = data.getItemVisual(idx, 'style'); - var color = symbolStyle && symbolStyle.fill; - var emphasisModel = itemModel.getModel('emphasis'); - rippleGroup.setScale(symbolSize); - rippleGroup.traverse(function (ripplePath) { - ripplePath.setStyle('fill', color); - }); - var symbolOffset = normalizeSymbolOffset(data.getItemVisual(idx, 'symbolOffset'), symbolSize); - - if (symbolOffset) { - rippleGroup.x = symbolOffset[0]; - rippleGroup.y = symbolOffset[1]; - } - - var symbolRotate = data.getItemVisual(idx, 'symbolRotate'); - rippleGroup.rotation = (symbolRotate || 0) * Math.PI / 180 || 0; - var effectCfg = {}; - effectCfg.showEffectOn = seriesModel.get('showEffectOn'); - effectCfg.rippleScale = itemModel.get(['rippleEffect', 'scale']); - effectCfg.brushType = itemModel.get(['rippleEffect', 'brushType']); - effectCfg.period = itemModel.get(['rippleEffect', 'period']) * 1000; - effectCfg.effectOffset = idx / data.count(); - effectCfg.z = seriesModel.getShallow('z') || 0; - effectCfg.zlevel = seriesModel.getShallow('zlevel') || 0; - effectCfg.symbolType = symbolType; - effectCfg.color = color; - effectCfg.rippleEffectColor = itemModel.get(['rippleEffect', 'color']); - effectCfg.rippleNumber = itemModel.get(['rippleEffect', 'number']); - - if (effectCfg.showEffectOn === 'render') { - this._effectCfg ? this.updateEffectAnimation(effectCfg) : this.startEffectAnimation(effectCfg); - this._effectCfg = effectCfg; - } else { - // Not keep old effect config - this._effectCfg = null; - this.stopEffectAnimation(); - - this.onHoverStateChange = function (toState) { - if (toState === 'emphasis') { - if (effectCfg.showEffectOn !== 'render') { - _this.startEffectAnimation(effectCfg); - } - } else if (toState === 'normal') { - if (effectCfg.showEffectOn !== 'render') { - _this.stopEffectAnimation(); - } - } - }; - } - - this._effectCfg = effectCfg; - toggleHoverEmphasis(this, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - }; - - EffectSymbol.prototype.fadeOut = function (cb) { - cb && cb(); - }; - return EffectSymbol; - }(Group); - - var EffectScatterView = - /** @class */ - function (_super) { - __extends(EffectScatterView, _super); - - function EffectScatterView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = EffectScatterView.type; - return _this; - } - - EffectScatterView.prototype.init = function () { - this._symbolDraw = new SymbolDraw(EffectSymbol); - }; - - EffectScatterView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var effectSymbolDraw = this._symbolDraw; - effectSymbolDraw.updateData(data, { - clipShape: this._getClipShape(seriesModel) - }); - this.group.add(effectSymbolDraw.group); - }; - - EffectScatterView.prototype._getClipShape = function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - var clipArea = coordSys && coordSys.getArea && coordSys.getArea(); - return seriesModel.get('clip', true) ? clipArea : null; - }; - - EffectScatterView.prototype.updateTransform = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - this.group.dirty(); - var res = pointsLayout('').reset(seriesModel, ecModel, api); - - if (res.progress) { - res.progress({ - start: 0, - end: data.count(), - count: data.count() - }, data); - } - - this._symbolDraw.updateLayout(); - }; - - EffectScatterView.prototype._updateGroupTransform = function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && coordSys.getRoamTransform) { - this.group.transform = clone$2(coordSys.getRoamTransform()); - this.group.decomposeTransform(); - } - }; - - EffectScatterView.prototype.remove = function (ecModel, api) { - this._symbolDraw && this._symbolDraw.remove(true); - }; - - EffectScatterView.type = 'effectScatter'; - return EffectScatterView; - }(ChartView); - - var EffectScatterSeriesModel = - /** @class */ - function (_super) { - __extends(EffectScatterSeriesModel, _super); - - function EffectScatterSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = EffectScatterSeriesModel.type; - _this.hasSymbolVisual = true; - return _this; - } - - EffectScatterSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesData(null, this, { - useEncodeDefaulter: true - }); - }; - - EffectScatterSeriesModel.prototype.brushSelector = function (dataIndex, data, selectors) { - return selectors.point(data.getItemLayout(dataIndex)); - }; - - EffectScatterSeriesModel.type = 'series.effectScatter'; - EffectScatterSeriesModel.dependencies = ['grid', 'polar']; - EffectScatterSeriesModel.defaultOption = { - coordinateSystem: 'cartesian2d', - // zlevel: 0, - z: 2, - legendHoverLink: true, - effectType: 'ripple', - progressive: 0, - // When to show the effect, option: 'render'|'emphasis' - showEffectOn: 'render', - clip: true, - // Ripple effect config - rippleEffect: { - period: 4, - // Scale of ripple - scale: 2.5, - // Brush type can be fill or stroke - brushType: 'fill', - // Ripple number - number: 3 - }, - universalTransition: { - divideShape: 'clone' - }, - // Cartesian coordinate system - // xAxisIndex: 0, - // yAxisIndex: 0, - // Polar coordinate system - // polarIndex: 0, - // Geo coordinate system - // geoIndex: 0, - // symbol: null, // 图形类型 - symbolSize: 10 // 图形大小,半宽(半径)参数,当图形为方向或菱形则总宽度为symbolSize * 2 - // symbolRotate: null, // 图形旋转控制 - // itemStyle: { - // opacity: 1 - // } - - }; - return EffectScatterSeriesModel; - }(SeriesModel); - - function install$l(registers) { - registers.registerChartView(EffectScatterView); - registers.registerSeriesModel(EffectScatterSeriesModel); - registers.registerLayout(pointsLayout('effectScatter')); - } - - var EffectLine = - /** @class */ - function (_super) { - __extends(EffectLine, _super); - - function EffectLine(lineData, idx, seriesScope) { - var _this = _super.call(this) || this; - - _this.add(_this.createLine(lineData, idx, seriesScope)); - - _this._updateEffectSymbol(lineData, idx); - - return _this; - } - - EffectLine.prototype.createLine = function (lineData, idx, seriesScope) { - return new Line$1(lineData, idx, seriesScope); - }; - - EffectLine.prototype._updateEffectSymbol = function (lineData, idx) { - var itemModel = lineData.getItemModel(idx); - var effectModel = itemModel.getModel('effect'); - var size = effectModel.get('symbolSize'); - var symbolType = effectModel.get('symbol'); - - if (!isArray(size)) { - size = [size, size]; - } - - var lineStyle = lineData.getItemVisual(idx, 'style'); - var color = effectModel.get('color') || lineStyle && lineStyle.stroke; - var symbol = this.childAt(1); - - if (this._symbolType !== symbolType) { - // Remove previous - this.remove(symbol); - symbol = createSymbol(symbolType, -0.5, -0.5, 1, 1, color); - symbol.z2 = 100; - symbol.culling = true; - this.add(symbol); - } // Symbol may be removed if loop is false - - - if (!symbol) { - return; - } // Shadow color is same with color in default - - - symbol.setStyle('shadowColor', color); - symbol.setStyle(effectModel.getItemStyle(['color'])); - symbol.scaleX = size[0]; - symbol.scaleY = size[1]; - symbol.setColor(color); - this._symbolType = symbolType; - this._symbolScale = size; - - this._updateEffectAnimation(lineData, effectModel, idx); - }; - - EffectLine.prototype._updateEffectAnimation = function (lineData, effectModel, idx) { - var symbol = this.childAt(1); - - if (!symbol) { - return; - } - - var points = lineData.getItemLayout(idx); - var period = effectModel.get('period') * 1000; - var loop = effectModel.get('loop'); - var roundTrip = effectModel.get('roundTrip'); - var constantSpeed = effectModel.get('constantSpeed'); - var delayExpr = retrieve(effectModel.get('delay'), function (idx) { - return idx / lineData.count() * period / 3; - }); // Ignore when updating - - symbol.ignore = true; - - this._updateAnimationPoints(symbol, points); - - if (constantSpeed > 0) { - period = this._getLineLength(symbol) / constantSpeed * 1000; - } - - if (period !== this._period || loop !== this._loop || roundTrip !== this._roundTrip) { - symbol.stopAnimation(); - var delayNum = void 0; - - if (isFunction(delayExpr)) { - delayNum = delayExpr(idx); - } else { - delayNum = delayExpr; - } - - if (symbol.__t > 0) { - delayNum = -period * symbol.__t; - } - - this._animateSymbol(symbol, period, delayNum, loop, roundTrip); - } - - this._period = period; - this._loop = loop; - this._roundTrip = roundTrip; - }; - - EffectLine.prototype._animateSymbol = function (symbol, period, delayNum, loop, roundTrip) { - if (period > 0) { - symbol.__t = 0; - var self_1 = this; - var animator = symbol.animate('', loop).when(roundTrip ? period * 2 : period, { - __t: roundTrip ? 2 : 1 - }).delay(delayNum).during(function () { - self_1._updateSymbolPosition(symbol); - }); - - if (!loop) { - animator.done(function () { - self_1.remove(symbol); - }); - } - - animator.start(); - } - }; - - EffectLine.prototype._getLineLength = function (symbol) { - // Not so accurate - return dist(symbol.__p1, symbol.__cp1) + dist(symbol.__cp1, symbol.__p2); - }; - - EffectLine.prototype._updateAnimationPoints = function (symbol, points) { - symbol.__p1 = points[0]; - symbol.__p2 = points[1]; - symbol.__cp1 = points[2] || [(points[0][0] + points[1][0]) / 2, (points[0][1] + points[1][1]) / 2]; - }; - - EffectLine.prototype.updateData = function (lineData, idx, seriesScope) { - this.childAt(0).updateData(lineData, idx, seriesScope); - - this._updateEffectSymbol(lineData, idx); - }; - - EffectLine.prototype._updateSymbolPosition = function (symbol) { - var p1 = symbol.__p1; - var p2 = symbol.__p2; - var cp1 = symbol.__cp1; - var t = symbol.__t < 1 ? symbol.__t : 2 - symbol.__t; - var pos = [symbol.x, symbol.y]; - var lastPos = pos.slice(); - var quadraticAt$1 = quadraticAt; - var quadraticDerivativeAt$1 = quadraticDerivativeAt; - pos[0] = quadraticAt$1(p1[0], cp1[0], p2[0], t); - pos[1] = quadraticAt$1(p1[1], cp1[1], p2[1], t); // Tangent - - var tx = symbol.__t < 1 ? quadraticDerivativeAt$1(p1[0], cp1[0], p2[0], t) : quadraticDerivativeAt$1(p2[0], cp1[0], p1[0], 1 - t); - var ty = symbol.__t < 1 ? quadraticDerivativeAt$1(p1[1], cp1[1], p2[1], t) : quadraticDerivativeAt$1(p2[1], cp1[1], p1[1], 1 - t); - symbol.rotation = -Math.atan2(ty, tx) - Math.PI / 2; // enable continuity trail for 'line', 'rect', 'roundRect' symbolType - - if (this._symbolType === 'line' || this._symbolType === 'rect' || this._symbolType === 'roundRect') { - if (symbol.__lastT !== undefined && symbol.__lastT < symbol.__t) { - symbol.scaleY = dist(lastPos, pos) * 1.05; // make sure the last segment render within endPoint - - if (t === 1) { - pos[0] = lastPos[0] + (pos[0] - lastPos[0]) / 2; - pos[1] = lastPos[1] + (pos[1] - lastPos[1]) / 2; - } - } else if (symbol.__lastT === 1) { - // After first loop, symbol.__t does NOT start with 0, so connect p1 to pos directly. - symbol.scaleY = 2 * dist(p1, pos); - } else { - symbol.scaleY = this._symbolScale[1]; - } - } - - symbol.__lastT = symbol.__t; - symbol.ignore = false; - symbol.x = pos[0]; - symbol.y = pos[1]; - }; - - EffectLine.prototype.updateLayout = function (lineData, idx) { - this.childAt(0).updateLayout(lineData, idx); - var effectModel = lineData.getItemModel(idx).getModel('effect'); - - this._updateEffectAnimation(lineData, effectModel, idx); - }; - - return EffectLine; - }(Group); - - var Polyline$1 = - /** @class */ - function (_super) { - __extends(Polyline$1, _super); - - function Polyline$1(lineData, idx, seriesScope) { - var _this = _super.call(this) || this; - - _this._createPolyline(lineData, idx, seriesScope); - - return _this; - } - - Polyline$1.prototype._createPolyline = function (lineData, idx, seriesScope) { - // let seriesModel = lineData.hostModel; - var points = lineData.getItemLayout(idx); - var line = new Polyline({ - shape: { - points: points - } - }); - this.add(line); - - this._updateCommonStl(lineData, idx, seriesScope); - }; - - Polyline$1.prototype.updateData = function (lineData, idx, seriesScope) { - var seriesModel = lineData.hostModel; - var line = this.childAt(0); - var target = { - shape: { - points: lineData.getItemLayout(idx) - } - }; - updateProps(line, target, seriesModel, idx); - - this._updateCommonStl(lineData, idx, seriesScope); - }; - - Polyline$1.prototype._updateCommonStl = function (lineData, idx, seriesScope) { - var line = this.childAt(0); - var itemModel = lineData.getItemModel(idx); - var emphasisLineStyle = seriesScope && seriesScope.emphasisLineStyle; - var focus = seriesScope && seriesScope.focus; - var blurScope = seriesScope && seriesScope.blurScope; - var emphasisDisabled = seriesScope && seriesScope.emphasisDisabled; - - if (!seriesScope || lineData.hasItemOption) { - var emphasisModel = itemModel.getModel('emphasis'); - emphasisLineStyle = emphasisModel.getModel('lineStyle').getLineStyle(); - emphasisDisabled = emphasisModel.get('disabled'); - focus = emphasisModel.get('focus'); - blurScope = emphasisModel.get('blurScope'); - } - - line.useStyle(lineData.getItemVisual(idx, 'style')); - line.style.fill = null; - line.style.strokeNoScale = true; - var lineEmphasisState = line.ensureState('emphasis'); - lineEmphasisState.style = emphasisLineStyle; - toggleHoverEmphasis(this, focus, blurScope, emphasisDisabled); - }; - - Polyline$1.prototype.updateLayout = function (lineData, idx) { - var polyline = this.childAt(0); - polyline.setShape('points', lineData.getItemLayout(idx)); - }; - return Polyline$1; - }(Group); - - var EffectPolyline = - /** @class */ - function (_super) { - __extends(EffectPolyline, _super); - - function EffectPolyline() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this._lastFrame = 0; - _this._lastFramePercent = 0; - return _this; - } // Override - - - EffectPolyline.prototype.createLine = function (lineData, idx, seriesScope) { - return new Polyline$1(lineData, idx, seriesScope); - }; - - EffectPolyline.prototype._updateAnimationPoints = function (symbol, points) { - this._points = points; - var accLenArr = [0]; - var len = 0; - - for (var i = 1; i < points.length; i++) { - var p1 = points[i - 1]; - var p2 = points[i]; - len += dist(p1, p2); - accLenArr.push(len); - } - - if (len === 0) { - this._length = 0; - return; - } - - for (var i = 0; i < accLenArr.length; i++) { - accLenArr[i] /= len; - } - - this._offsets = accLenArr; - this._length = len; - }; - - EffectPolyline.prototype._getLineLength = function () { - return this._length; - }; - - EffectPolyline.prototype._updateSymbolPosition = function (symbol) { - var t = symbol.__t < 1 ? symbol.__t : 2 - symbol.__t; - var points = this._points; - var offsets = this._offsets; - var len = points.length; - - if (!offsets) { - // Has length 0 - return; - } - - var lastFrame = this._lastFrame; - var frame; - - if (t < this._lastFramePercent) { - // Start from the next frame - // PENDING start from lastFrame ? - var start = Math.min(lastFrame + 1, len - 1); - - for (frame = start; frame >= 0; frame--) { - if (offsets[frame] <= t) { - break; - } - } // PENDING really need to do this ? - - - frame = Math.min(frame, len - 2); - } else { - for (frame = lastFrame; frame < len; frame++) { - if (offsets[frame] > t) { - break; - } - } - - frame = Math.min(frame - 1, len - 2); - } - - var p = (t - offsets[frame]) / (offsets[frame + 1] - offsets[frame]); - var p0 = points[frame]; - var p1 = points[frame + 1]; - symbol.x = p0[0] * (1 - p) + p * p1[0]; - symbol.y = p0[1] * (1 - p) + p * p1[1]; - var tx = symbol.__t < 1 ? p1[0] - p0[0] : p0[0] - p1[0]; - var ty = symbol.__t < 1 ? p1[1] - p0[1] : p0[1] - p1[1]; - symbol.rotation = -Math.atan2(ty, tx) - Math.PI / 2; - this._lastFrame = frame; - this._lastFramePercent = t; - symbol.ignore = false; - }; - return EffectPolyline; - }(EffectLine); - - var LargeLinesPathShape = - /** @class */ - function () { - function LargeLinesPathShape() { - this.polyline = false; - this.curveness = 0; - this.segs = []; - } - - return LargeLinesPathShape; - }(); - - var LargeLinesPath = - /** @class */ - function (_super) { - __extends(LargeLinesPath, _super); - - function LargeLinesPath(opts) { - var _this = _super.call(this, opts) || this; - - _this._off = 0; - _this.hoverDataIdx = -1; - return _this; - } - - LargeLinesPath.prototype.reset = function () { - this.notClear = false; - this._off = 0; - }; - - LargeLinesPath.prototype.getDefaultStyle = function () { - return { - stroke: '#000', - fill: null - }; - }; - - LargeLinesPath.prototype.getDefaultShape = function () { - return new LargeLinesPathShape(); - }; - - LargeLinesPath.prototype.buildPath = function (ctx, shape) { - var segs = shape.segs; - var curveness = shape.curveness; - var i; - - if (shape.polyline) { - for (i = this._off; i < segs.length;) { - var count = segs[i++]; - - if (count > 0) { - ctx.moveTo(segs[i++], segs[i++]); - - for (var k = 1; k < count; k++) { - ctx.lineTo(segs[i++], segs[i++]); - } - } - } - } else { - for (i = this._off; i < segs.length;) { - var x0 = segs[i++]; - var y0 = segs[i++]; - var x1 = segs[i++]; - var y1 = segs[i++]; - ctx.moveTo(x0, y0); - - if (curveness > 0) { - var x2 = (x0 + x1) / 2 - (y0 - y1) * curveness; - var y2 = (y0 + y1) / 2 - (x1 - x0) * curveness; - ctx.quadraticCurveTo(x2, y2, x1, y1); - } else { - ctx.lineTo(x1, y1); - } - } - } - - if (this.incremental) { - this._off = i; - this.notClear = true; - } - }; - - LargeLinesPath.prototype.findDataIndex = function (x, y) { - var shape = this.shape; - var segs = shape.segs; - var curveness = shape.curveness; - var lineWidth = this.style.lineWidth; - - if (shape.polyline) { - var dataIndex = 0; - - for (var i = 0; i < segs.length;) { - var count = segs[i++]; - - if (count > 0) { - var x0 = segs[i++]; - var y0 = segs[i++]; - - for (var k = 1; k < count; k++) { - var x1 = segs[i++]; - var y1 = segs[i++]; - - if (containStroke(x0, y0, x1, y1, lineWidth, x, y)) { - return dataIndex; - } - } - } - - dataIndex++; - } - } else { - var dataIndex = 0; - - for (var i = 0; i < segs.length;) { - var x0 = segs[i++]; - var y0 = segs[i++]; - var x1 = segs[i++]; - var y1 = segs[i++]; - - if (curveness > 0) { - var x2 = (x0 + x1) / 2 - (y0 - y1) * curveness; - var y2 = (y0 + y1) / 2 - (x1 - x0) * curveness; - - if (containStroke$2(x0, y0, x2, y2, x1, y1, lineWidth, x, y)) { - return dataIndex; - } - } else { - if (containStroke(x0, y0, x1, y1, lineWidth, x, y)) { - return dataIndex; - } - } - - dataIndex++; - } - } - - return -1; - }; - - LargeLinesPath.prototype.contain = function (x, y) { - var localPos = this.transformCoordToLocal(x, y); - var rect = this.getBoundingRect(); - x = localPos[0]; - y = localPos[1]; - - if (rect.contain(x, y)) { - // Cache found data index. - var dataIdx = this.hoverDataIdx = this.findDataIndex(x, y); - return dataIdx >= 0; - } - - this.hoverDataIdx = -1; - return false; - }; - - LargeLinesPath.prototype.getBoundingRect = function () { - // Ignore stroke for large symbol draw. - var rect = this._rect; - - if (!rect) { - var shape = this.shape; - var points = shape.segs; - var minX = Infinity; - var minY = Infinity; - var maxX = -Infinity; - var maxY = -Infinity; - - for (var i = 0; i < points.length;) { - var x = points[i++]; - var y = points[i++]; - minX = Math.min(x, minX); - maxX = Math.max(x, maxX); - minY = Math.min(y, minY); - maxY = Math.max(y, maxY); - } - - rect = this._rect = new BoundingRect(minX, minY, maxX, maxY); - } - - return rect; - }; - - return LargeLinesPath; - }(Path); - - var LargeLineDraw = - /** @class */ - function () { - function LargeLineDraw() { - this.group = new Group(); - } - /** - * Update symbols draw by new data - */ - - - LargeLineDraw.prototype.updateData = function (data) { - this._clear(); - - var lineEl = this._create(); - - lineEl.setShape({ - segs: data.getLayout('linesPoints') - }); - - this._setCommon(lineEl, data); - }; - /** - * @override - */ - - LargeLineDraw.prototype.incrementalPrepareUpdate = function (data) { - this.group.removeAll(); - - this._clear(); - }; - /** - * @override - */ - - LargeLineDraw.prototype.incrementalUpdate = function (taskParams, data) { - var lastAdded = this._newAdded[0]; - var linePoints = data.getLayout('linesPoints'); - var oldSegs = lastAdded && lastAdded.shape.segs; // Merging the exists. Each element has 1e4 points. - // Consider the performance balance between too much elements and too much points in one shape(may affect hover optimization) - - if (oldSegs && oldSegs.length < 2e4) { - var oldLen = oldSegs.length; - var newSegs = new Float32Array(oldLen + linePoints.length); // Concat two array - - newSegs.set(oldSegs); - newSegs.set(linePoints, oldLen); - lastAdded.setShape({ - segs: newSegs - }); - } else { - // Clear - this._newAdded = []; - - var lineEl = this._create(); - - lineEl.incremental = true; - lineEl.setShape({ - segs: linePoints - }); - - this._setCommon(lineEl, data); - - lineEl.__startIndex = taskParams.start; - } - }; - /** - * @override - */ - - - LargeLineDraw.prototype.remove = function () { - this._clear(); - }; - - LargeLineDraw.prototype.eachRendered = function (cb) { - this._newAdded[0] && cb(this._newAdded[0]); - }; - - LargeLineDraw.prototype._create = function () { - var lineEl = new LargeLinesPath({ - cursor: 'default', - ignoreCoarsePointer: true - }); - - this._newAdded.push(lineEl); - - this.group.add(lineEl); - return lineEl; - }; - - LargeLineDraw.prototype._setCommon = function (lineEl, data, isIncremental) { - var hostModel = data.hostModel; - lineEl.setShape({ - polyline: hostModel.get('polyline'), - curveness: hostModel.get(['lineStyle', 'curveness']) - }); - lineEl.useStyle(hostModel.getModel('lineStyle').getLineStyle()); - lineEl.style.strokeNoScale = true; - var style = data.getVisual('style'); - - if (style && style.stroke) { - lineEl.setStyle('stroke', style.stroke); - } - - lineEl.setStyle('fill', null); - var ecData = getECData(lineEl); // Enable tooltip - // PENDING May have performance issue when path is extremely large - - ecData.seriesIndex = hostModel.seriesIndex; - lineEl.on('mousemove', function (e) { - ecData.dataIndex = null; - var dataIndex = lineEl.hoverDataIdx; - - if (dataIndex > 0) { - // Provide dataIndex for tooltip - ecData.dataIndex = dataIndex + lineEl.__startIndex; - } - }); - }; - - LargeLineDraw.prototype._clear = function () { - this._newAdded = []; - this.group.removeAll(); - }; - return LargeLineDraw; - }(); - - var linesLayout = { - seriesType: 'lines', - plan: createRenderPlanner(), - reset: function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - - if (!coordSys) { - if ("development" !== 'production') { - error('The lines series must have a coordinate system.'); - } - - return; - } - - var isPolyline = seriesModel.get('polyline'); - var isLarge = seriesModel.pipelineContext.large; - return { - progress: function (params, lineData) { - var lineCoords = []; - - if (isLarge) { - var points = void 0; - var segCount = params.end - params.start; - - if (isPolyline) { - var totalCoordsCount = 0; - - for (var i = params.start; i < params.end; i++) { - totalCoordsCount += seriesModel.getLineCoordsCount(i); - } - - points = new Float32Array(segCount + totalCoordsCount * 2); - } else { - points = new Float32Array(segCount * 4); - } - - var offset = 0; - var pt = []; - - for (var i = params.start; i < params.end; i++) { - var len = seriesModel.getLineCoords(i, lineCoords); - - if (isPolyline) { - points[offset++] = len; - } - - for (var k = 0; k < len; k++) { - pt = coordSys.dataToPoint(lineCoords[k], false, pt); - points[offset++] = pt[0]; - points[offset++] = pt[1]; - } - } - - lineData.setLayout('linesPoints', points); - } else { - for (var i = params.start; i < params.end; i++) { - var itemModel = lineData.getItemModel(i); - var len = seriesModel.getLineCoords(i, lineCoords); - var pts = []; - - if (isPolyline) { - for (var j = 0; j < len; j++) { - pts.push(coordSys.dataToPoint(lineCoords[j])); - } - } else { - pts[0] = coordSys.dataToPoint(lineCoords[0]); - pts[1] = coordSys.dataToPoint(lineCoords[1]); - var curveness = itemModel.get(['lineStyle', 'curveness']); - - if (+curveness) { - pts[2] = [(pts[0][0] + pts[1][0]) / 2 - (pts[0][1] - pts[1][1]) * curveness, (pts[0][1] + pts[1][1]) / 2 - (pts[1][0] - pts[0][0]) * curveness]; - } - } - - lineData.setItemLayout(i, pts); - } - } - } - }; - } - }; - - var LinesView = - /** @class */ - function (_super) { - __extends(LinesView, _super); - - function LinesView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = LinesView.type; - return _this; - } - - LinesView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - - var lineDraw = this._updateLineDraw(data, seriesModel); - - var zlevel = seriesModel.get('zlevel'); - var trailLength = seriesModel.get(['effect', 'trailLength']); - var zr = api.getZr(); // Avoid the drag cause ghost shadow - // FIXME Better way ? - // SVG doesn't support - - var isSvg = zr.painter.getType() === 'svg'; - - if (!isSvg) { - zr.painter.getLayer(zlevel).clear(true); - } // Config layer with motion blur - - - if (this._lastZlevel != null && !isSvg) { - zr.configLayer(this._lastZlevel, { - motionBlur: false - }); - } - - if (this._showEffect(seriesModel) && trailLength > 0) { - if (!isSvg) { - zr.configLayer(zlevel, { - motionBlur: true, - lastFrameAlpha: Math.max(Math.min(trailLength / 10 + 0.9, 1), 0) - }); - } else if ("development" !== 'production') { - console.warn('SVG render mode doesn\'t support lines with trail effect'); - } - } - - lineDraw.updateData(data); - var clipPath = seriesModel.get('clip', true) && createClipPath(seriesModel.coordinateSystem, false, seriesModel); - - if (clipPath) { - this.group.setClipPath(clipPath); - } else { - this.group.removeClipPath(); - } - - this._lastZlevel = zlevel; - this._finished = true; - }; - - LinesView.prototype.incrementalPrepareRender = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - - var lineDraw = this._updateLineDraw(data, seriesModel); - - lineDraw.incrementalPrepareUpdate(data); - - this._clearLayer(api); - - this._finished = false; - }; - - LinesView.prototype.incrementalRender = function (taskParams, seriesModel, ecModel) { - this._lineDraw.incrementalUpdate(taskParams, seriesModel.getData()); - - this._finished = taskParams.end === seriesModel.getData().count(); - }; - - LinesView.prototype.eachRendered = function (cb) { - this._lineDraw && this._lineDraw.eachRendered(cb); - }; - - LinesView.prototype.updateTransform = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var pipelineContext = seriesModel.pipelineContext; - - if (!this._finished || pipelineContext.large || pipelineContext.progressiveRender) { - // TODO Don't have to do update in large mode. Only do it when there are millions of data. - return { - update: true - }; - } else { - // TODO Use same logic with ScatterView. - // Manually update layout - var res = linesLayout.reset(seriesModel, ecModel, api); - - if (res.progress) { - res.progress({ - start: 0, - end: data.count(), - count: data.count() - }, data); - } // Not in large mode - - - this._lineDraw.updateLayout(); - - this._clearLayer(api); - } - }; - - LinesView.prototype._updateLineDraw = function (data, seriesModel) { - var lineDraw = this._lineDraw; - - var hasEffect = this._showEffect(seriesModel); - - var isPolyline = !!seriesModel.get('polyline'); - var pipelineContext = seriesModel.pipelineContext; - var isLargeDraw = pipelineContext.large; - - if ("development" !== 'production') { - if (hasEffect && isLargeDraw) { - console.warn('Large lines not support effect'); - } - } - - if (!lineDraw || hasEffect !== this._hasEffet || isPolyline !== this._isPolyline || isLargeDraw !== this._isLargeDraw) { - if (lineDraw) { - lineDraw.remove(); - } - - lineDraw = this._lineDraw = isLargeDraw ? new LargeLineDraw() : new LineDraw(isPolyline ? hasEffect ? EffectPolyline : Polyline$1 : hasEffect ? EffectLine : Line$1); - this._hasEffet = hasEffect; - this._isPolyline = isPolyline; - this._isLargeDraw = isLargeDraw; - } - - this.group.add(lineDraw.group); - return lineDraw; - }; - - LinesView.prototype._showEffect = function (seriesModel) { - return !!seriesModel.get(['effect', 'show']); - }; - - LinesView.prototype._clearLayer = function (api) { - // Not use motion when dragging or zooming - var zr = api.getZr(); - var isSvg = zr.painter.getType() === 'svg'; - - if (!isSvg && this._lastZlevel != null) { - zr.painter.getLayer(this._lastZlevel).clear(true); - } - }; - - LinesView.prototype.remove = function (ecModel, api) { - this._lineDraw && this._lineDraw.remove(); - this._lineDraw = null; // Clear motion when lineDraw is removed - - this._clearLayer(api); - }; - - LinesView.prototype.dispose = function (ecModel, api) { - this.remove(ecModel, api); - }; - - LinesView.type = 'lines'; - return LinesView; - }(ChartView); - - var Uint32Arr = typeof Uint32Array === 'undefined' ? Array : Uint32Array; - var Float64Arr = typeof Float64Array === 'undefined' ? Array : Float64Array; - - function compatEc2(seriesOpt) { - var data = seriesOpt.data; - - if (data && data[0] && data[0][0] && data[0][0].coord) { - if ("development" !== 'production') { - console.warn('Lines data configuration has been changed to' + ' { coords:[[1,2],[2,3]] }'); - } - - seriesOpt.data = map(data, function (itemOpt) { - var coords = [itemOpt[0].coord, itemOpt[1].coord]; - var target = { - coords: coords - }; - - if (itemOpt[0].name) { - target.fromName = itemOpt[0].name; - } - - if (itemOpt[1].name) { - target.toName = itemOpt[1].name; - } - - return mergeAll([target, itemOpt[0], itemOpt[1]]); - }); - } - } - - var LinesSeriesModel = - /** @class */ - function (_super) { - __extends(LinesSeriesModel, _super); - - function LinesSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = LinesSeriesModel.type; - _this.visualStyleAccessPath = 'lineStyle'; - _this.visualDrawType = 'stroke'; - return _this; - } - - LinesSeriesModel.prototype.init = function (option) { - // The input data may be null/undefined. - option.data = option.data || []; // Not using preprocessor because mergeOption may not have series.type - - compatEc2(option); - - var result = this._processFlatCoordsArray(option.data); - - this._flatCoords = result.flatCoords; - this._flatCoordsOffset = result.flatCoordsOffset; - - if (result.flatCoords) { - option.data = new Float32Array(result.count); - } - - _super.prototype.init.apply(this, arguments); - }; - - LinesSeriesModel.prototype.mergeOption = function (option) { - compatEc2(option); - - if (option.data) { - // Only update when have option data to merge. - var result = this._processFlatCoordsArray(option.data); - - this._flatCoords = result.flatCoords; - this._flatCoordsOffset = result.flatCoordsOffset; - - if (result.flatCoords) { - option.data = new Float32Array(result.count); - } - } - - _super.prototype.mergeOption.apply(this, arguments); - }; - - LinesSeriesModel.prototype.appendData = function (params) { - var result = this._processFlatCoordsArray(params.data); - - if (result.flatCoords) { - if (!this._flatCoords) { - this._flatCoords = result.flatCoords; - this._flatCoordsOffset = result.flatCoordsOffset; - } else { - this._flatCoords = concatArray(this._flatCoords, result.flatCoords); - this._flatCoordsOffset = concatArray(this._flatCoordsOffset, result.flatCoordsOffset); - } - - params.data = new Float32Array(result.count); - } - - this.getRawData().appendData(params.data); - }; - - LinesSeriesModel.prototype._getCoordsFromItemModel = function (idx) { - var itemModel = this.getData().getItemModel(idx); - var coords = itemModel.option instanceof Array ? itemModel.option : itemModel.getShallow('coords'); - - if ("development" !== 'production') { - if (!(coords instanceof Array && coords.length > 0 && coords[0] instanceof Array)) { - throw new Error('Invalid coords ' + JSON.stringify(coords) + '. Lines must have 2d coords array in data item.'); - } - } - - return coords; - }; - - LinesSeriesModel.prototype.getLineCoordsCount = function (idx) { - if (this._flatCoordsOffset) { - return this._flatCoordsOffset[idx * 2 + 1]; - } else { - return this._getCoordsFromItemModel(idx).length; - } - }; - - LinesSeriesModel.prototype.getLineCoords = function (idx, out) { - if (this._flatCoordsOffset) { - var offset = this._flatCoordsOffset[idx * 2]; - var len = this._flatCoordsOffset[idx * 2 + 1]; - - for (var i = 0; i < len; i++) { - out[i] = out[i] || []; - out[i][0] = this._flatCoords[offset + i * 2]; - out[i][1] = this._flatCoords[offset + i * 2 + 1]; - } - - return len; - } else { - var coords = this._getCoordsFromItemModel(idx); - - for (var i = 0; i < coords.length; i++) { - out[i] = out[i] || []; - out[i][0] = coords[i][0]; - out[i][1] = coords[i][1]; - } - - return coords.length; - } - }; - - LinesSeriesModel.prototype._processFlatCoordsArray = function (data) { - var startOffset = 0; - - if (this._flatCoords) { - startOffset = this._flatCoords.length; - } // Stored as a typed array. In format - // Points Count(2) | x | y | x | y | Points Count(3) | x | y | x | y | x | y | - - - if (isNumber(data[0])) { - var len = data.length; // Store offset and len of each segment - - var coordsOffsetAndLenStorage = new Uint32Arr(len); - var coordsStorage = new Float64Arr(len); - var coordsCursor = 0; - var offsetCursor = 0; - var dataCount = 0; - - for (var i = 0; i < len;) { - dataCount++; - var count = data[i++]; // Offset - - coordsOffsetAndLenStorage[offsetCursor++] = coordsCursor + startOffset; // Len - - coordsOffsetAndLenStorage[offsetCursor++] = count; - - for (var k = 0; k < count; k++) { - var x = data[i++]; - var y = data[i++]; - coordsStorage[coordsCursor++] = x; - coordsStorage[coordsCursor++] = y; - - if (i > len) { - if ("development" !== 'production') { - throw new Error('Invalid data format.'); - } - } - } - } - - return { - flatCoordsOffset: new Uint32Array(coordsOffsetAndLenStorage.buffer, 0, offsetCursor), - flatCoords: coordsStorage, - count: dataCount - }; - } - - return { - flatCoordsOffset: null, - flatCoords: null, - count: data.length - }; - }; - - LinesSeriesModel.prototype.getInitialData = function (option, ecModel) { - if ("development" !== 'production') { - var CoordSys = CoordinateSystemManager.get(option.coordinateSystem); - - if (!CoordSys) { - throw new Error('Unknown coordinate system ' + option.coordinateSystem); - } - } - - var lineData = new SeriesData(['value'], this); - lineData.hasItemOption = false; - lineData.initData(option.data, [], function (dataItem, dimName, dataIndex, dimIndex) { - // dataItem is simply coords - if (dataItem instanceof Array) { - return NaN; - } else { - lineData.hasItemOption = true; - var value = dataItem.value; - - if (value != null) { - return value instanceof Array ? value[dimIndex] : value; - } - } - }); - return lineData; - }; - - LinesSeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - var data = this.getData(); - var itemModel = data.getItemModel(dataIndex); - var name = itemModel.get('name'); - - if (name) { - return name; - } - - var fromName = itemModel.get('fromName'); - var toName = itemModel.get('toName'); - var nameArr = []; - fromName != null && nameArr.push(fromName); - toName != null && nameArr.push(toName); - return createTooltipMarkup('nameValue', { - name: nameArr.join(' > ') - }); - }; - - LinesSeriesModel.prototype.preventIncremental = function () { - return !!this.get(['effect', 'show']); - }; - - LinesSeriesModel.prototype.getProgressive = function () { - var progressive = this.option.progressive; - - if (progressive == null) { - return this.option.large ? 1e4 : this.get('progressive'); - } - - return progressive; - }; - - LinesSeriesModel.prototype.getProgressiveThreshold = function () { - var progressiveThreshold = this.option.progressiveThreshold; - - if (progressiveThreshold == null) { - return this.option.large ? 2e4 : this.get('progressiveThreshold'); - } - - return progressiveThreshold; - }; - - LinesSeriesModel.prototype.getZLevelKey = function () { - var effectModel = this.getModel('effect'); - var trailLength = effectModel.get('trailLength'); - return this.getData().count() > this.getProgressiveThreshold() // Each progressive series has individual key. - ? this.id : effectModel.get('show') && trailLength > 0 ? trailLength + '' : ''; - }; - - LinesSeriesModel.type = 'series.lines'; - LinesSeriesModel.dependencies = ['grid', 'polar', 'geo', 'calendar']; - LinesSeriesModel.defaultOption = { - coordinateSystem: 'geo', - // zlevel: 0, - z: 2, - legendHoverLink: true, - // Cartesian coordinate system - xAxisIndex: 0, - yAxisIndex: 0, - symbol: ['none', 'none'], - symbolSize: [10, 10], - // Geo coordinate system - geoIndex: 0, - effect: { - show: false, - period: 4, - constantSpeed: 0, - symbol: 'circle', - symbolSize: 3, - loop: true, - trailLength: 0.2 - }, - large: false, - // Available when large is true - largeThreshold: 2000, - polyline: false, - clip: true, - label: { - show: false, - position: 'end' // distance: 5, - // formatter: 标签文本格式器,同Tooltip.formatter,不支持异步回调 - - }, - lineStyle: { - opacity: 0.5 - } - }; - return LinesSeriesModel; - }(SeriesModel); - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function normalize$3(a) { - if (!(a instanceof Array)) { - a = [a, a]; - } - - return a; - } - - var linesVisual = { - seriesType: 'lines', - reset: function (seriesModel) { - var symbolType = normalize$3(seriesModel.get('symbol')); - var symbolSize = normalize$3(seriesModel.get('symbolSize')); - var data = seriesModel.getData(); - data.setVisual('fromSymbol', symbolType && symbolType[0]); - data.setVisual('toSymbol', symbolType && symbolType[1]); - data.setVisual('fromSymbolSize', symbolSize && symbolSize[0]); - data.setVisual('toSymbolSize', symbolSize && symbolSize[1]); - - function dataEach(data, idx) { - var itemModel = data.getItemModel(idx); - var symbolType = normalize$3(itemModel.getShallow('symbol', true)); - var symbolSize = normalize$3(itemModel.getShallow('symbolSize', true)); - symbolType[0] && data.setItemVisual(idx, 'fromSymbol', symbolType[0]); - symbolType[1] && data.setItemVisual(idx, 'toSymbol', symbolType[1]); - symbolSize[0] && data.setItemVisual(idx, 'fromSymbolSize', symbolSize[0]); - symbolSize[1] && data.setItemVisual(idx, 'toSymbolSize', symbolSize[1]); - } - - return { - dataEach: data.hasItemOption ? dataEach : null - }; - } - }; - - function install$m(registers) { - registers.registerChartView(LinesView); - registers.registerSeriesModel(LinesSeriesModel); - registers.registerLayout(linesLayout); - registers.registerVisual(linesVisual); - } - - var GRADIENT_LEVELS = 256; - - var HeatmapLayer = - /** @class */ - function () { - function HeatmapLayer() { - this.blurSize = 30; - this.pointSize = 20; - this.maxOpacity = 1; - this.minOpacity = 0; - this._gradientPixels = { - inRange: null, - outOfRange: null - }; - var canvas = platformApi.createCanvas(); - this.canvas = canvas; - } - /** - * Renders Heatmap and returns the rendered canvas - * @param data array of data, each has x, y, value - * @param width canvas width - * @param height canvas height - */ - - - HeatmapLayer.prototype.update = function (data, width, height, normalize, colorFunc, isInRange) { - var brush = this._getBrush(); - - var gradientInRange = this._getGradient(colorFunc, 'inRange'); - - var gradientOutOfRange = this._getGradient(colorFunc, 'outOfRange'); - - var r = this.pointSize + this.blurSize; - var canvas = this.canvas; - var ctx = canvas.getContext('2d'); - var len = data.length; - canvas.width = width; - canvas.height = height; - - for (var i = 0; i < len; ++i) { - var p = data[i]; - var x = p[0]; - var y = p[1]; - var value = p[2]; // calculate alpha using value - - var alpha = normalize(value); // draw with the circle brush with alpha - - ctx.globalAlpha = alpha; - ctx.drawImage(brush, x - r, y - r); - } - - if (!canvas.width || !canvas.height) { - // Avoid "Uncaught DOMException: Failed to execute 'getImageData' on - // 'CanvasRenderingContext2D': The source height is 0." - return canvas; - } // colorize the canvas using alpha value and set with gradient - - - var imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); - var pixels = imageData.data; - var offset = 0; - var pixelLen = pixels.length; - var minOpacity = this.minOpacity; - var maxOpacity = this.maxOpacity; - var diffOpacity = maxOpacity - minOpacity; - - while (offset < pixelLen) { - var alpha = pixels[offset + 3] / 256; - var gradientOffset = Math.floor(alpha * (GRADIENT_LEVELS - 1)) * 4; // Simple optimize to ignore the empty data - - if (alpha > 0) { - var gradient = isInRange(alpha) ? gradientInRange : gradientOutOfRange; // Any alpha > 0 will be mapped to [minOpacity, maxOpacity] - - alpha > 0 && (alpha = alpha * diffOpacity + minOpacity); - pixels[offset++] = gradient[gradientOffset]; - pixels[offset++] = gradient[gradientOffset + 1]; - pixels[offset++] = gradient[gradientOffset + 2]; - pixels[offset++] = gradient[gradientOffset + 3] * alpha * 256; - } else { - offset += 4; - } - } - - ctx.putImageData(imageData, 0, 0); - return canvas; - }; - /** - * get canvas of a black circle brush used for canvas to draw later - */ - - - HeatmapLayer.prototype._getBrush = function () { - var brushCanvas = this._brushCanvas || (this._brushCanvas = platformApi.createCanvas()); // set brush size - - var r = this.pointSize + this.blurSize; - var d = r * 2; - brushCanvas.width = d; - brushCanvas.height = d; - var ctx = brushCanvas.getContext('2d'); - ctx.clearRect(0, 0, d, d); // in order to render shadow without the distinct circle, - // draw the distinct circle in an invisible place, - // and use shadowOffset to draw shadow in the center of the canvas - - ctx.shadowOffsetX = d; - ctx.shadowBlur = this.blurSize; // draw the shadow in black, and use alpha and shadow blur to generate - // color in color map - - ctx.shadowColor = '#000'; // draw circle in the left to the canvas - - ctx.beginPath(); - ctx.arc(-r, r, this.pointSize, 0, Math.PI * 2, true); - ctx.closePath(); - ctx.fill(); - return brushCanvas; - }; - /** - * get gradient color map - * @private - */ - - - HeatmapLayer.prototype._getGradient = function (colorFunc, state) { - var gradientPixels = this._gradientPixels; - var pixelsSingleState = gradientPixels[state] || (gradientPixels[state] = new Uint8ClampedArray(256 * 4)); - var color = [0, 0, 0, 0]; - var off = 0; - - for (var i = 0; i < 256; i++) { - colorFunc[state](i / 255, true, color); - pixelsSingleState[off++] = color[0]; - pixelsSingleState[off++] = color[1]; - pixelsSingleState[off++] = color[2]; - pixelsSingleState[off++] = color[3]; - } - - return pixelsSingleState; - }; - - return HeatmapLayer; - }(); - - function getIsInPiecewiseRange(dataExtent, pieceList, selected) { - var dataSpan = dataExtent[1] - dataExtent[0]; - pieceList = map(pieceList, function (piece) { - return { - interval: [(piece.interval[0] - dataExtent[0]) / dataSpan, (piece.interval[1] - dataExtent[0]) / dataSpan] - }; - }); - var len = pieceList.length; - var lastIndex = 0; - return function (val) { - var i; // Try to find in the location of the last found - - for (i = lastIndex; i < len; i++) { - var interval = pieceList[i].interval; - - if (interval[0] <= val && val <= interval[1]) { - lastIndex = i; - break; - } - } - - if (i === len) { - // Not found, back interation - for (i = lastIndex - 1; i >= 0; i--) { - var interval = pieceList[i].interval; - - if (interval[0] <= val && val <= interval[1]) { - lastIndex = i; - break; - } - } - } - - return i >= 0 && i < len && selected[i]; - }; - } - - function getIsInContinuousRange(dataExtent, range) { - var dataSpan = dataExtent[1] - dataExtent[0]; - range = [(range[0] - dataExtent[0]) / dataSpan, (range[1] - dataExtent[0]) / dataSpan]; - return function (val) { - return val >= range[0] && val <= range[1]; - }; - } - - function isGeoCoordSys(coordSys) { - var dimensions = coordSys.dimensions; // Not use coordSys.type === 'geo' because coordSys maybe extended - - return dimensions[0] === 'lng' && dimensions[1] === 'lat'; - } - - var HeatmapView = - /** @class */ - function (_super) { - __extends(HeatmapView, _super); - - function HeatmapView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = HeatmapView.type; - return _this; - } - - HeatmapView.prototype.render = function (seriesModel, ecModel, api) { - var visualMapOfThisSeries; - ecModel.eachComponent('visualMap', function (visualMap) { - visualMap.eachTargetSeries(function (targetSeries) { - if (targetSeries === seriesModel) { - visualMapOfThisSeries = visualMap; - } - }); - }); - - if ("development" !== 'production') { - if (!visualMapOfThisSeries) { - throw new Error('Heatmap must use with visualMap'); - } - } // Clear previously rendered progressive elements. - - - this._progressiveEls = null; - this.group.removeAll(); - var coordSys = seriesModel.coordinateSystem; - - if (coordSys.type === 'cartesian2d' || coordSys.type === 'calendar') { - this._renderOnCartesianAndCalendar(seriesModel, api, 0, seriesModel.getData().count()); - } else if (isGeoCoordSys(coordSys)) { - this._renderOnGeo(coordSys, seriesModel, visualMapOfThisSeries, api); - } - }; - - HeatmapView.prototype.incrementalPrepareRender = function (seriesModel, ecModel, api) { - this.group.removeAll(); - }; - - HeatmapView.prototype.incrementalRender = function (params, seriesModel, ecModel, api) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys) { - // geo does not support incremental rendering? - if (isGeoCoordSys(coordSys)) { - this.render(seriesModel, ecModel, api); - } else { - this._progressiveEls = []; - - this._renderOnCartesianAndCalendar(seriesModel, api, params.start, params.end, true); - } - } - }; - - HeatmapView.prototype.eachRendered = function (cb) { - traverseElements(this._progressiveEls || this.group, cb); - }; - - HeatmapView.prototype._renderOnCartesianAndCalendar = function (seriesModel, api, start, end, incremental) { - var coordSys = seriesModel.coordinateSystem; - var isCartesian2d = isCoordinateSystemType(coordSys, 'cartesian2d'); - var width; - var height; - var xAxisExtent; - var yAxisExtent; - - if (isCartesian2d) { - var xAxis = coordSys.getAxis('x'); - var yAxis = coordSys.getAxis('y'); - - if ("development" !== 'production') { - if (!(xAxis.type === 'category' && yAxis.type === 'category')) { - throw new Error('Heatmap on cartesian must have two category axes'); - } - - if (!(xAxis.onBand && yAxis.onBand)) { - throw new Error('Heatmap on cartesian must have two axes with boundaryGap true'); - } - } // add 0.5px to avoid the gaps - - - width = xAxis.getBandWidth() + .5; - height = yAxis.getBandWidth() + .5; - xAxisExtent = xAxis.scale.getExtent(); - yAxisExtent = yAxis.scale.getExtent(); - } - - var group = this.group; - var data = seriesModel.getData(); - var emphasisStyle = seriesModel.getModel(['emphasis', 'itemStyle']).getItemStyle(); - var blurStyle = seriesModel.getModel(['blur', 'itemStyle']).getItemStyle(); - var selectStyle = seriesModel.getModel(['select', 'itemStyle']).getItemStyle(); - var borderRadius = seriesModel.get(['itemStyle', 'borderRadius']); - var labelStatesModels = getLabelStatesModels(seriesModel); - var emphasisModel = seriesModel.getModel('emphasis'); - var focus = emphasisModel.get('focus'); - var blurScope = emphasisModel.get('blurScope'); - var emphasisDisabled = emphasisModel.get('disabled'); - var dataDims = isCartesian2d ? [data.mapDimension('x'), data.mapDimension('y'), data.mapDimension('value')] : [data.mapDimension('time'), data.mapDimension('value')]; - - for (var idx = start; idx < end; idx++) { - var rect = void 0; - var style = data.getItemVisual(idx, 'style'); - - if (isCartesian2d) { - var dataDimX = data.get(dataDims[0], idx); - var dataDimY = data.get(dataDims[1], idx); // Ignore empty data and out of extent data - - if (isNaN(data.get(dataDims[2], idx)) || isNaN(dataDimX) || isNaN(dataDimY) || dataDimX < xAxisExtent[0] || dataDimX > xAxisExtent[1] || dataDimY < yAxisExtent[0] || dataDimY > yAxisExtent[1]) { - continue; - } - - var point = coordSys.dataToPoint([dataDimX, dataDimY]); - rect = new Rect({ - shape: { - x: point[0] - width / 2, - y: point[1] - height / 2, - width: width, - height: height - }, - style: style - }); - } else { - // Ignore empty data - if (isNaN(data.get(dataDims[1], idx))) { - continue; - } - - rect = new Rect({ - z2: 1, - shape: coordSys.dataToRect([data.get(dataDims[0], idx)]).contentShape, - style: style - }); - } // Optimization for large dataset - - - if (data.hasItemOption) { - var itemModel = data.getItemModel(idx); - var emphasisModel_1 = itemModel.getModel('emphasis'); - emphasisStyle = emphasisModel_1.getModel('itemStyle').getItemStyle(); - blurStyle = itemModel.getModel(['blur', 'itemStyle']).getItemStyle(); - selectStyle = itemModel.getModel(['select', 'itemStyle']).getItemStyle(); // Each item value struct in the data would be firstly - // { - // itemStyle: { borderRadius: [30, 30] }, - // value: [2022, 02, 22] - // } - - borderRadius = itemModel.get(['itemStyle', 'borderRadius']); - focus = emphasisModel_1.get('focus'); - blurScope = emphasisModel_1.get('blurScope'); - emphasisDisabled = emphasisModel_1.get('disabled'); - labelStatesModels = getLabelStatesModels(itemModel); - } - - rect.shape.r = borderRadius; - var rawValue = seriesModel.getRawValue(idx); - var defaultText = '-'; - - if (rawValue && rawValue[2] != null) { - defaultText = rawValue[2] + ''; - } - - setLabelStyle(rect, labelStatesModels, { - labelFetcher: seriesModel, - labelDataIndex: idx, - defaultOpacity: style.opacity, - defaultText: defaultText - }); - rect.ensureState('emphasis').style = emphasisStyle; - rect.ensureState('blur').style = blurStyle; - rect.ensureState('select').style = selectStyle; - toggleHoverEmphasis(rect, focus, blurScope, emphasisDisabled); - rect.incremental = incremental; // PENDING - - if (incremental) { - // Rect must use hover layer if it's incremental. - rect.states.emphasis.hoverLayer = true; - } - - group.add(rect); - data.setItemGraphicEl(idx, rect); - - if (this._progressiveEls) { - this._progressiveEls.push(rect); - } - } - }; - - HeatmapView.prototype._renderOnGeo = function (geo, seriesModel, visualMapModel, api) { - var inRangeVisuals = visualMapModel.targetVisuals.inRange; - var outOfRangeVisuals = visualMapModel.targetVisuals.outOfRange; // if (!visualMapping) { - // throw new Error('Data range must have color visuals'); - // } - - var data = seriesModel.getData(); - var hmLayer = this._hmLayer || this._hmLayer || new HeatmapLayer(); - hmLayer.blurSize = seriesModel.get('blurSize'); - hmLayer.pointSize = seriesModel.get('pointSize'); - hmLayer.minOpacity = seriesModel.get('minOpacity'); - hmLayer.maxOpacity = seriesModel.get('maxOpacity'); - var rect = geo.getViewRect().clone(); - var roamTransform = geo.getRoamTransform(); - rect.applyTransform(roamTransform); // Clamp on viewport - - var x = Math.max(rect.x, 0); - var y = Math.max(rect.y, 0); - var x2 = Math.min(rect.width + rect.x, api.getWidth()); - var y2 = Math.min(rect.height + rect.y, api.getHeight()); - var width = x2 - x; - var height = y2 - y; - var dims = [data.mapDimension('lng'), data.mapDimension('lat'), data.mapDimension('value')]; - var points = data.mapArray(dims, function (lng, lat, value) { - var pt = geo.dataToPoint([lng, lat]); - pt[0] -= x; - pt[1] -= y; - pt.push(value); - return pt; - }); - var dataExtent = visualMapModel.getExtent(); - var isInRange = visualMapModel.type === 'visualMap.continuous' ? getIsInContinuousRange(dataExtent, visualMapModel.option.range) : getIsInPiecewiseRange(dataExtent, visualMapModel.getPieceList(), visualMapModel.option.selected); - hmLayer.update(points, width, height, inRangeVisuals.color.getNormalizer(), { - inRange: inRangeVisuals.color.getColorMapper(), - outOfRange: outOfRangeVisuals.color.getColorMapper() - }, isInRange); - var img = new ZRImage({ - style: { - width: width, - height: height, - x: x, - y: y, - image: hmLayer.canvas - }, - silent: true - }); - this.group.add(img); - }; - - HeatmapView.type = 'heatmap'; - return HeatmapView; - }(ChartView); - - var HeatmapSeriesModel = - /** @class */ - function (_super) { - __extends(HeatmapSeriesModel, _super); - - function HeatmapSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = HeatmapSeriesModel.type; - return _this; - } - - HeatmapSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesData(null, this, { - generateCoord: 'value' - }); - }; - - HeatmapSeriesModel.prototype.preventIncremental = function () { - var coordSysCreator = CoordinateSystemManager.get(this.get('coordinateSystem')); - - if (coordSysCreator && coordSysCreator.dimensions) { - return coordSysCreator.dimensions[0] === 'lng' && coordSysCreator.dimensions[1] === 'lat'; - } - }; - - HeatmapSeriesModel.type = 'series.heatmap'; - HeatmapSeriesModel.dependencies = ['grid', 'geo', 'calendar']; - HeatmapSeriesModel.defaultOption = { - coordinateSystem: 'cartesian2d', - // zlevel: 0, - z: 2, - // Cartesian coordinate system - // xAxisIndex: 0, - // yAxisIndex: 0, - // Geo coordinate system - geoIndex: 0, - blurSize: 30, - pointSize: 20, - maxOpacity: 1, - minOpacity: 0, - select: { - itemStyle: { - borderColor: '#212121' - } - } - }; - return HeatmapSeriesModel; - }(SeriesModel); - - function install$n(registers) { - registers.registerChartView(HeatmapView); - registers.registerSeriesModel(HeatmapSeriesModel); - } - - var BAR_BORDER_WIDTH_QUERY = ['itemStyle', 'borderWidth']; // index: +isHorizontal - - var LAYOUT_ATTRS = [{ - xy: 'x', - wh: 'width', - index: 0, - posDesc: ['left', 'right'] - }, { - xy: 'y', - wh: 'height', - index: 1, - posDesc: ['top', 'bottom'] - }]; - var pathForLineWidth = new Circle(); - - var PictorialBarView = - /** @class */ - function (_super) { - __extends(PictorialBarView, _super); - - function PictorialBarView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = PictorialBarView.type; - return _this; - } - - PictorialBarView.prototype.render = function (seriesModel, ecModel, api) { - var group = this.group; - var data = seriesModel.getData(); - var oldData = this._data; - var cartesian = seriesModel.coordinateSystem; - var baseAxis = cartesian.getBaseAxis(); - var isHorizontal = baseAxis.isHorizontal(); - var coordSysRect = cartesian.master.getRect(); - var opt = { - ecSize: { - width: api.getWidth(), - height: api.getHeight() - }, - seriesModel: seriesModel, - coordSys: cartesian, - coordSysExtent: [[coordSysRect.x, coordSysRect.x + coordSysRect.width], [coordSysRect.y, coordSysRect.y + coordSysRect.height]], - isHorizontal: isHorizontal, - valueDim: LAYOUT_ATTRS[+isHorizontal], - categoryDim: LAYOUT_ATTRS[1 - +isHorizontal] - }; - data.diff(oldData).add(function (dataIndex) { - if (!data.hasValue(dataIndex)) { - return; - } - - var itemModel = getItemModel(data, dataIndex); - var symbolMeta = getSymbolMeta(data, dataIndex, itemModel, opt); - var bar = createBar(data, opt, symbolMeta); - data.setItemGraphicEl(dataIndex, bar); - group.add(bar); - updateCommon$1(bar, opt, symbolMeta); - }).update(function (newIndex, oldIndex) { - var bar = oldData.getItemGraphicEl(oldIndex); - - if (!data.hasValue(newIndex)) { - group.remove(bar); - return; - } - - var itemModel = getItemModel(data, newIndex); - var symbolMeta = getSymbolMeta(data, newIndex, itemModel, opt); - var pictorialShapeStr = getShapeStr(data, symbolMeta); - - if (bar && pictorialShapeStr !== bar.__pictorialShapeStr) { - group.remove(bar); - data.setItemGraphicEl(newIndex, null); - bar = null; - } - - if (bar) { - updateBar(bar, opt, symbolMeta); - } else { - bar = createBar(data, opt, symbolMeta, true); - } - - data.setItemGraphicEl(newIndex, bar); - bar.__pictorialSymbolMeta = symbolMeta; // Add back - - group.add(bar); - updateCommon$1(bar, opt, symbolMeta); - }).remove(function (dataIndex) { - var bar = oldData.getItemGraphicEl(dataIndex); - bar && removeBar(oldData, dataIndex, bar.__pictorialSymbolMeta.animationModel, bar); - }).execute(); - this._data = data; - return this.group; - }; - - PictorialBarView.prototype.remove = function (ecModel, api) { - var group = this.group; - var data = this._data; - - if (ecModel.get('animation')) { - if (data) { - data.eachItemGraphicEl(function (bar) { - removeBar(data, getECData(bar).dataIndex, ecModel, bar); - }); - } - } else { - group.removeAll(); - } - }; - - PictorialBarView.type = 'pictorialBar'; - return PictorialBarView; - }(ChartView); // Set or calculate default value about symbol, and calculate layout info. - - - function getSymbolMeta(data, dataIndex, itemModel, opt) { - var layout = data.getItemLayout(dataIndex); - var symbolRepeat = itemModel.get('symbolRepeat'); - var symbolClip = itemModel.get('symbolClip'); - var symbolPosition = itemModel.get('symbolPosition') || 'start'; - var symbolRotate = itemModel.get('symbolRotate'); - var rotation = (symbolRotate || 0) * Math.PI / 180 || 0; - var symbolPatternSize = itemModel.get('symbolPatternSize') || 2; - var isAnimationEnabled = itemModel.isAnimationEnabled(); - var symbolMeta = { - dataIndex: dataIndex, - layout: layout, - itemModel: itemModel, - symbolType: data.getItemVisual(dataIndex, 'symbol') || 'circle', - style: data.getItemVisual(dataIndex, 'style'), - symbolClip: symbolClip, - symbolRepeat: symbolRepeat, - symbolRepeatDirection: itemModel.get('symbolRepeatDirection'), - symbolPatternSize: symbolPatternSize, - rotation: rotation, - animationModel: isAnimationEnabled ? itemModel : null, - hoverScale: isAnimationEnabled && itemModel.get(['emphasis', 'scale']), - z2: itemModel.getShallow('z', true) || 0 - }; - prepareBarLength(itemModel, symbolRepeat, layout, opt, symbolMeta); - prepareSymbolSize(data, dataIndex, layout, symbolRepeat, symbolClip, symbolMeta.boundingLength, symbolMeta.pxSign, symbolPatternSize, opt, symbolMeta); - prepareLineWidth(itemModel, symbolMeta.symbolScale, rotation, opt, symbolMeta); - var symbolSize = symbolMeta.symbolSize; - var symbolOffset = normalizeSymbolOffset(itemModel.get('symbolOffset'), symbolSize); - prepareLayoutInfo(itemModel, symbolSize, layout, symbolRepeat, symbolClip, symbolOffset, symbolPosition, symbolMeta.valueLineWidth, symbolMeta.boundingLength, symbolMeta.repeatCutLength, opt, symbolMeta); - return symbolMeta; - } // bar length can be negative. - - - function prepareBarLength(itemModel, symbolRepeat, layout, opt, outputSymbolMeta) { - var valueDim = opt.valueDim; - var symbolBoundingData = itemModel.get('symbolBoundingData'); - var valueAxis = opt.coordSys.getOtherAxis(opt.coordSys.getBaseAxis()); - var zeroPx = valueAxis.toGlobalCoord(valueAxis.dataToCoord(0)); - var pxSignIdx = 1 - +(layout[valueDim.wh] <= 0); - var boundingLength; - - if (isArray(symbolBoundingData)) { - var symbolBoundingExtent = [convertToCoordOnAxis(valueAxis, symbolBoundingData[0]) - zeroPx, convertToCoordOnAxis(valueAxis, symbolBoundingData[1]) - zeroPx]; - symbolBoundingExtent[1] < symbolBoundingExtent[0] && symbolBoundingExtent.reverse(); - boundingLength = symbolBoundingExtent[pxSignIdx]; - } else if (symbolBoundingData != null) { - boundingLength = convertToCoordOnAxis(valueAxis, symbolBoundingData) - zeroPx; - } else if (symbolRepeat) { - boundingLength = opt.coordSysExtent[valueDim.index][pxSignIdx] - zeroPx; - } else { - boundingLength = layout[valueDim.wh]; - } - - outputSymbolMeta.boundingLength = boundingLength; - - if (symbolRepeat) { - outputSymbolMeta.repeatCutLength = layout[valueDim.wh]; - } // if 'pxSign' means sign of pixel, it can't be zero, or symbolScale will be zero - // and when borderWidth be settled, the actual linewidth will be NaN - - - outputSymbolMeta.pxSign = boundingLength > 0 ? 1 : -1; - } - - function convertToCoordOnAxis(axis, value) { - return axis.toGlobalCoord(axis.dataToCoord(axis.scale.parse(value))); - } // Support ['100%', '100%'] - - - function prepareSymbolSize(data, dataIndex, layout, symbolRepeat, symbolClip, boundingLength, pxSign, symbolPatternSize, opt, outputSymbolMeta) { - var valueDim = opt.valueDim; - var categoryDim = opt.categoryDim; - var categorySize = Math.abs(layout[categoryDim.wh]); - var symbolSize = data.getItemVisual(dataIndex, 'symbolSize'); - var parsedSymbolSize; - - if (isArray(symbolSize)) { - parsedSymbolSize = symbolSize.slice(); - } else { - if (symbolSize == null) { - // will parse to number below - parsedSymbolSize = ['100%', '100%']; - } else { - parsedSymbolSize = [symbolSize, symbolSize]; - } - } // Note: percentage symbolSize (like '100%') do not consider lineWidth, because it is - // to complicated to calculate real percent value if considering scaled lineWidth. - // So the actual size will bigger than layout size if lineWidth is bigger than zero, - // which can be tolerated in pictorial chart. - - - parsedSymbolSize[categoryDim.index] = parsePercent$1(parsedSymbolSize[categoryDim.index], categorySize); - parsedSymbolSize[valueDim.index] = parsePercent$1(parsedSymbolSize[valueDim.index], symbolRepeat ? categorySize : Math.abs(boundingLength)); - outputSymbolMeta.symbolSize = parsedSymbolSize; // If x or y is less than zero, show reversed shape. - - var symbolScale = outputSymbolMeta.symbolScale = [parsedSymbolSize[0] / symbolPatternSize, parsedSymbolSize[1] / symbolPatternSize]; // Follow convention, 'right' and 'top' is the normal scale. - - symbolScale[valueDim.index] *= (opt.isHorizontal ? -1 : 1) * pxSign; - } - - function prepareLineWidth(itemModel, symbolScale, rotation, opt, outputSymbolMeta) { - // In symbols are drawn with scale, so do not need to care about the case that width - // or height are too small. But symbol use strokeNoScale, where acture lineWidth should - // be calculated. - var valueLineWidth = itemModel.get(BAR_BORDER_WIDTH_QUERY) || 0; - - if (valueLineWidth) { - pathForLineWidth.attr({ - scaleX: symbolScale[0], - scaleY: symbolScale[1], - rotation: rotation - }); - pathForLineWidth.updateTransform(); - valueLineWidth /= pathForLineWidth.getLineScale(); - valueLineWidth *= symbolScale[opt.valueDim.index]; - } - - outputSymbolMeta.valueLineWidth = valueLineWidth || 0; - } - - function prepareLayoutInfo(itemModel, symbolSize, layout, symbolRepeat, symbolClip, symbolOffset, symbolPosition, valueLineWidth, boundingLength, repeatCutLength, opt, outputSymbolMeta) { - var categoryDim = opt.categoryDim; - var valueDim = opt.valueDim; - var pxSign = outputSymbolMeta.pxSign; - var unitLength = Math.max(symbolSize[valueDim.index] + valueLineWidth, 0); - var pathLen = unitLength; // Note: rotation will not effect the layout of symbols, because user may - // want symbols to rotate on its center, which should not be translated - // when rotating. - - if (symbolRepeat) { - var absBoundingLength = Math.abs(boundingLength); - var symbolMargin = retrieve(itemModel.get('symbolMargin'), '15%') + ''; - var hasEndGap = false; - - if (symbolMargin.lastIndexOf('!') === symbolMargin.length - 1) { - hasEndGap = true; - symbolMargin = symbolMargin.slice(0, symbolMargin.length - 1); - } - - var symbolMarginNumeric = parsePercent$1(symbolMargin, symbolSize[valueDim.index]); - var uLenWithMargin = Math.max(unitLength + symbolMarginNumeric * 2, 0); // When symbol margin is less than 0, margin at both ends will be subtracted - // to ensure that all of the symbols will not be overflow the given area. - - var endFix = hasEndGap ? 0 : symbolMarginNumeric * 2; // Both final repeatTimes and final symbolMarginNumeric area calculated based on - // boundingLength. - - var repeatSpecified = isNumeric(symbolRepeat); - var repeatTimes = repeatSpecified ? symbolRepeat : toIntTimes((absBoundingLength + endFix) / uLenWithMargin); // Adjust calculate margin, to ensure each symbol is displayed - // entirely in the given layout area. - - var mDiff = absBoundingLength - repeatTimes * unitLength; - symbolMarginNumeric = mDiff / 2 / (hasEndGap ? repeatTimes : Math.max(repeatTimes - 1, 1)); - uLenWithMargin = unitLength + symbolMarginNumeric * 2; - endFix = hasEndGap ? 0 : symbolMarginNumeric * 2; // Update repeatTimes when not all symbol will be shown. - - if (!repeatSpecified && symbolRepeat !== 'fixed') { - repeatTimes = repeatCutLength ? toIntTimes((Math.abs(repeatCutLength) + endFix) / uLenWithMargin) : 0; - } - - pathLen = repeatTimes * uLenWithMargin - endFix; - outputSymbolMeta.repeatTimes = repeatTimes; - outputSymbolMeta.symbolMargin = symbolMarginNumeric; - } - - var sizeFix = pxSign * (pathLen / 2); - var pathPosition = outputSymbolMeta.pathPosition = []; - pathPosition[categoryDim.index] = layout[categoryDim.wh] / 2; - pathPosition[valueDim.index] = symbolPosition === 'start' ? sizeFix : symbolPosition === 'end' ? boundingLength - sizeFix : boundingLength / 2; // 'center' - - if (symbolOffset) { - pathPosition[0] += symbolOffset[0]; - pathPosition[1] += symbolOffset[1]; - } - - var bundlePosition = outputSymbolMeta.bundlePosition = []; - bundlePosition[categoryDim.index] = layout[categoryDim.xy]; - bundlePosition[valueDim.index] = layout[valueDim.xy]; - var barRectShape = outputSymbolMeta.barRectShape = extend({}, layout); - barRectShape[valueDim.wh] = pxSign * Math.max(Math.abs(layout[valueDim.wh]), Math.abs(pathPosition[valueDim.index] + sizeFix)); - barRectShape[categoryDim.wh] = layout[categoryDim.wh]; - var clipShape = outputSymbolMeta.clipShape = {}; // Consider that symbol may be overflow layout rect. - - clipShape[categoryDim.xy] = -layout[categoryDim.xy]; - clipShape[categoryDim.wh] = opt.ecSize[categoryDim.wh]; - clipShape[valueDim.xy] = 0; - clipShape[valueDim.wh] = layout[valueDim.wh]; - } - - function createPath(symbolMeta) { - var symbolPatternSize = symbolMeta.symbolPatternSize; - var path = createSymbol( // Consider texture img, make a big size. - symbolMeta.symbolType, -symbolPatternSize / 2, -symbolPatternSize / 2, symbolPatternSize, symbolPatternSize); - path.attr({ - culling: true - }); - path.type !== 'image' && path.setStyle({ - strokeNoScale: true - }); - return path; - } - - function createOrUpdateRepeatSymbols(bar, opt, symbolMeta, isUpdate) { - var bundle = bar.__pictorialBundle; - var symbolSize = symbolMeta.symbolSize; - var valueLineWidth = symbolMeta.valueLineWidth; - var pathPosition = symbolMeta.pathPosition; - var valueDim = opt.valueDim; - var repeatTimes = symbolMeta.repeatTimes || 0; - var index = 0; - var unit = symbolSize[opt.valueDim.index] + valueLineWidth + symbolMeta.symbolMargin * 2; - eachPath(bar, function (path) { - path.__pictorialAnimationIndex = index; - path.__pictorialRepeatTimes = repeatTimes; - - if (index < repeatTimes) { - updateAttr(path, null, makeTarget(index), symbolMeta, isUpdate); - } else { - updateAttr(path, null, { - scaleX: 0, - scaleY: 0 - }, symbolMeta, isUpdate, function () { - bundle.remove(path); - }); - } // updateHoverAnimation(path, symbolMeta); - - - index++; - }); - - for (; index < repeatTimes; index++) { - var path = createPath(symbolMeta); - path.__pictorialAnimationIndex = index; - path.__pictorialRepeatTimes = repeatTimes; - bundle.add(path); - var target = makeTarget(index); - updateAttr(path, { - x: target.x, - y: target.y, - scaleX: 0, - scaleY: 0 - }, { - scaleX: target.scaleX, - scaleY: target.scaleY, - rotation: target.rotation - }, symbolMeta, isUpdate); - } - - function makeTarget(index) { - var position = pathPosition.slice(); // (start && pxSign > 0) || (end && pxSign < 0): i = repeatTimes - index - // Otherwise: i = index; - - var pxSign = symbolMeta.pxSign; - var i = index; - - if (symbolMeta.symbolRepeatDirection === 'start' ? pxSign > 0 : pxSign < 0) { - i = repeatTimes - 1 - index; - } - - position[valueDim.index] = unit * (i - repeatTimes / 2 + 0.5) + pathPosition[valueDim.index]; - return { - x: position[0], - y: position[1], - scaleX: symbolMeta.symbolScale[0], - scaleY: symbolMeta.symbolScale[1], - rotation: symbolMeta.rotation - }; - } - } - - function createOrUpdateSingleSymbol(bar, opt, symbolMeta, isUpdate) { - var bundle = bar.__pictorialBundle; - var mainPath = bar.__pictorialMainPath; - - if (!mainPath) { - mainPath = bar.__pictorialMainPath = createPath(symbolMeta); - bundle.add(mainPath); - updateAttr(mainPath, { - x: symbolMeta.pathPosition[0], - y: symbolMeta.pathPosition[1], - scaleX: 0, - scaleY: 0, - rotation: symbolMeta.rotation - }, { - scaleX: symbolMeta.symbolScale[0], - scaleY: symbolMeta.symbolScale[1] - }, symbolMeta, isUpdate); - } else { - updateAttr(mainPath, null, { - x: symbolMeta.pathPosition[0], - y: symbolMeta.pathPosition[1], - scaleX: symbolMeta.symbolScale[0], - scaleY: symbolMeta.symbolScale[1], - rotation: symbolMeta.rotation - }, symbolMeta, isUpdate); - } - } // bar rect is used for label. - - - function createOrUpdateBarRect(bar, symbolMeta, isUpdate) { - var rectShape = extend({}, symbolMeta.barRectShape); - var barRect = bar.__pictorialBarRect; - - if (!barRect) { - barRect = bar.__pictorialBarRect = new Rect({ - z2: 2, - shape: rectShape, - silent: true, - style: { - stroke: 'transparent', - fill: 'transparent', - lineWidth: 0 - } - }); - barRect.disableMorphing = true; - bar.add(barRect); - } else { - updateAttr(barRect, null, { - shape: rectShape - }, symbolMeta, isUpdate); - } - } - - function createOrUpdateClip(bar, opt, symbolMeta, isUpdate) { - // If not clip, symbol will be remove and rebuilt. - if (symbolMeta.symbolClip) { - var clipPath = bar.__pictorialClipPath; - var clipShape = extend({}, symbolMeta.clipShape); - var valueDim = opt.valueDim; - var animationModel = symbolMeta.animationModel; - var dataIndex = symbolMeta.dataIndex; - - if (clipPath) { - updateProps(clipPath, { - shape: clipShape - }, animationModel, dataIndex); - } else { - clipShape[valueDim.wh] = 0; - clipPath = new Rect({ - shape: clipShape - }); - - bar.__pictorialBundle.setClipPath(clipPath); - - bar.__pictorialClipPath = clipPath; - var target = {}; - target[valueDim.wh] = symbolMeta.clipShape[valueDim.wh]; - graphic[isUpdate ? 'updateProps' : 'initProps'](clipPath, { - shape: target - }, animationModel, dataIndex); - } - } - } - - function getItemModel(data, dataIndex) { - var itemModel = data.getItemModel(dataIndex); - itemModel.getAnimationDelayParams = getAnimationDelayParams; - itemModel.isAnimationEnabled = isAnimationEnabled; - return itemModel; - } - - function getAnimationDelayParams(path) { - // The order is the same as the z-order, see `symbolRepeatDiretion`. - return { - index: path.__pictorialAnimationIndex, - count: path.__pictorialRepeatTimes - }; - } - - function isAnimationEnabled() { - // `animation` prop can be set on itemModel in pictorial bar chart. - return this.parentModel.isAnimationEnabled() && !!this.getShallow('animation'); - } - - function createBar(data, opt, symbolMeta, isUpdate) { - // bar is the main element for each data. - var bar = new Group(); // bundle is used for location and clip. - - var bundle = new Group(); - bar.add(bundle); - bar.__pictorialBundle = bundle; - bundle.x = symbolMeta.bundlePosition[0]; - bundle.y = symbolMeta.bundlePosition[1]; - - if (symbolMeta.symbolRepeat) { - createOrUpdateRepeatSymbols(bar, opt, symbolMeta); - } else { - createOrUpdateSingleSymbol(bar, opt, symbolMeta); - } - - createOrUpdateBarRect(bar, symbolMeta, isUpdate); - createOrUpdateClip(bar, opt, symbolMeta, isUpdate); - bar.__pictorialShapeStr = getShapeStr(data, symbolMeta); - bar.__pictorialSymbolMeta = symbolMeta; - return bar; - } - - function updateBar(bar, opt, symbolMeta) { - var animationModel = symbolMeta.animationModel; - var dataIndex = symbolMeta.dataIndex; - var bundle = bar.__pictorialBundle; - updateProps(bundle, { - x: symbolMeta.bundlePosition[0], - y: symbolMeta.bundlePosition[1] - }, animationModel, dataIndex); - - if (symbolMeta.symbolRepeat) { - createOrUpdateRepeatSymbols(bar, opt, symbolMeta, true); - } else { - createOrUpdateSingleSymbol(bar, opt, symbolMeta, true); - } - - createOrUpdateBarRect(bar, symbolMeta, true); - createOrUpdateClip(bar, opt, symbolMeta, true); - } - - function removeBar(data, dataIndex, animationModel, bar) { - // Not show text when animating - var labelRect = bar.__pictorialBarRect; - labelRect && labelRect.removeTextContent(); - var paths = []; - eachPath(bar, function (path) { - paths.push(path); - }); - bar.__pictorialMainPath && paths.push(bar.__pictorialMainPath); // I do not find proper remove animation for clip yet. - - bar.__pictorialClipPath && (animationModel = null); - each(paths, function (path) { - removeElement(path, { - scaleX: 0, - scaleY: 0 - }, animationModel, dataIndex, function () { - bar.parent && bar.parent.remove(bar); - }); - }); - data.setItemGraphicEl(dataIndex, null); - } - - function getShapeStr(data, symbolMeta) { - return [data.getItemVisual(symbolMeta.dataIndex, 'symbol') || 'none', !!symbolMeta.symbolRepeat, !!symbolMeta.symbolClip].join(':'); - } - - function eachPath(bar, cb, context) { - // Do not use Group#eachChild, because it do not support remove. - each(bar.__pictorialBundle.children(), function (el) { - el !== bar.__pictorialBarRect && cb.call(context, el); - }); - } - - function updateAttr(el, immediateAttrs, animationAttrs, symbolMeta, isUpdate, cb) { - immediateAttrs && el.attr(immediateAttrs); // when symbolCip used, only clip path has init animation, otherwise it would be weird effect. - - if (symbolMeta.symbolClip && !isUpdate) { - animationAttrs && el.attr(animationAttrs); - } else { - animationAttrs && graphic[isUpdate ? 'updateProps' : 'initProps'](el, animationAttrs, symbolMeta.animationModel, symbolMeta.dataIndex, cb); - } - } - - function updateCommon$1(bar, opt, symbolMeta) { - var dataIndex = symbolMeta.dataIndex; - var itemModel = symbolMeta.itemModel; // Color must be excluded. - // Because symbol provide setColor individually to set fill and stroke - - var emphasisModel = itemModel.getModel('emphasis'); - var emphasisStyle = emphasisModel.getModel('itemStyle').getItemStyle(); - var blurStyle = itemModel.getModel(['blur', 'itemStyle']).getItemStyle(); - var selectStyle = itemModel.getModel(['select', 'itemStyle']).getItemStyle(); - var cursorStyle = itemModel.getShallow('cursor'); - var focus = emphasisModel.get('focus'); - var blurScope = emphasisModel.get('blurScope'); - var hoverScale = emphasisModel.get('scale'); - eachPath(bar, function (path) { - if (path instanceof ZRImage) { - var pathStyle = path.style; - path.useStyle(extend({ - // TODO other properties like dx, dy ? - image: pathStyle.image, - x: pathStyle.x, - y: pathStyle.y, - width: pathStyle.width, - height: pathStyle.height - }, symbolMeta.style)); - } else { - path.useStyle(symbolMeta.style); - } - - var emphasisState = path.ensureState('emphasis'); - emphasisState.style = emphasisStyle; - - if (hoverScale) { - // NOTE: Must after scale is set after updateAttr - emphasisState.scaleX = path.scaleX * 1.1; - emphasisState.scaleY = path.scaleY * 1.1; - } - - path.ensureState('blur').style = blurStyle; - path.ensureState('select').style = selectStyle; - cursorStyle && (path.cursor = cursorStyle); - path.z2 = symbolMeta.z2; - }); - var barPositionOutside = opt.valueDim.posDesc[+(symbolMeta.boundingLength > 0)]; - var barRect = bar.__pictorialBarRect; - setLabelStyle(barRect, getLabelStatesModels(itemModel), { - labelFetcher: opt.seriesModel, - labelDataIndex: dataIndex, - defaultText: getDefaultLabel(opt.seriesModel.getData(), dataIndex), - inheritColor: symbolMeta.style.fill, - defaultOpacity: symbolMeta.style.opacity, - defaultOutsidePosition: barPositionOutside - }); - toggleHoverEmphasis(bar, focus, blurScope, emphasisModel.get('disabled')); - } - - function toIntTimes(times) { - var roundedTimes = Math.round(times); // Escapse accurate error - - return Math.abs(times - roundedTimes) < 1e-4 ? roundedTimes : Math.ceil(times); - } - - var PictorialBarSeriesModel = - /** @class */ - function (_super) { - __extends(PictorialBarSeriesModel, _super); - - function PictorialBarSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = PictorialBarSeriesModel.type; - _this.hasSymbolVisual = true; - _this.defaultSymbol = 'roundRect'; - return _this; - } - - PictorialBarSeriesModel.prototype.getInitialData = function (option) { - // Disable stack. - option.stack = null; - return _super.prototype.getInitialData.apply(this, arguments); - }; - - PictorialBarSeriesModel.type = 'series.pictorialBar'; - PictorialBarSeriesModel.dependencies = ['grid']; - PictorialBarSeriesModel.defaultOption = inheritDefaultOption(BaseBarSeriesModel.defaultOption, { - symbol: 'circle', - symbolSize: null, - symbolRotate: null, - symbolPosition: null, - symbolOffset: null, - symbolMargin: null, - symbolRepeat: false, - symbolRepeatDirection: 'end', - symbolClip: false, - symbolBoundingData: null, - symbolPatternSize: 400, - barGap: '-100%', - // z can be set in data item, which is z2 actually. - // Disable progressive - progressive: 0, - emphasis: { - // By default pictorialBar do not hover scale. Hover scale is not suitable - // for the case that both has foreground and background. - scale: false - }, - select: { - itemStyle: { - borderColor: '#212121' - } - } - }); - return PictorialBarSeriesModel; - }(BaseBarSeriesModel); - - function install$o(registers) { - registers.registerChartView(PictorialBarView); - registers.registerSeriesModel(PictorialBarSeriesModel); - registers.registerLayout(registers.PRIORITY.VISUAL.LAYOUT, curry(layout, 'pictorialBar')); // Do layout after other overall layout, which can prepare some information. - - registers.registerLayout(registers.PRIORITY.VISUAL.PROGRESSIVE_LAYOUT, createProgressiveLayout('pictorialBar')); - } - - var ThemeRiverView = - /** @class */ - function (_super) { - __extends(ThemeRiverView, _super); - - function ThemeRiverView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ThemeRiverView.type; - _this._layers = []; - return _this; - } - - ThemeRiverView.prototype.render = function (seriesModel, ecModel, api) { - var data = seriesModel.getData(); - var self = this; - var group = this.group; - var layersSeries = seriesModel.getLayerSeries(); - var layoutInfo = data.getLayout('layoutInfo'); - var rect = layoutInfo.rect; - var boundaryGap = layoutInfo.boundaryGap; - group.x = 0; - group.y = rect.y + boundaryGap[0]; - - function keyGetter(item) { - return item.name; - } - - var dataDiffer = new DataDiffer(this._layersSeries || [], layersSeries, keyGetter, keyGetter); - var newLayersGroups = []; - dataDiffer.add(bind(process, this, 'add')).update(bind(process, this, 'update')).remove(bind(process, this, 'remove')).execute(); - - function process(status, idx, oldIdx) { - var oldLayersGroups = self._layers; - - if (status === 'remove') { - group.remove(oldLayersGroups[idx]); - return; - } - - var points0 = []; - var points1 = []; - var style; - var indices = layersSeries[idx].indices; - var j = 0; - - for (; j < indices.length; j++) { - var layout = data.getItemLayout(indices[j]); - var x = layout.x; - var y0 = layout.y0; - var y = layout.y; - points0.push(x, y0); - points1.push(x, y0 + y); - style = data.getItemVisual(indices[j], 'style'); - } - - var polygon; - var textLayout = data.getItemLayout(indices[0]); - var labelModel = seriesModel.getModel('label'); - var margin = labelModel.get('margin'); - var emphasisModel = seriesModel.getModel('emphasis'); - - if (status === 'add') { - var layerGroup = newLayersGroups[idx] = new Group(); - polygon = new ECPolygon({ - shape: { - points: points0, - stackedOnPoints: points1, - smooth: 0.4, - stackedOnSmooth: 0.4, - smoothConstraint: false - }, - z2: 0 - }); - layerGroup.add(polygon); - group.add(layerGroup); - - if (seriesModel.isAnimationEnabled()) { - polygon.setClipPath(createGridClipShape$2(polygon.getBoundingRect(), seriesModel, function () { - polygon.removeClipPath(); - })); - } - } else { - var layerGroup = oldLayersGroups[oldIdx]; - polygon = layerGroup.childAt(0); - group.add(layerGroup); - newLayersGroups[idx] = layerGroup; - updateProps(polygon, { - shape: { - points: points0, - stackedOnPoints: points1 - } - }, seriesModel); - saveOldStyle(polygon); - } - - setLabelStyle(polygon, getLabelStatesModels(seriesModel), { - labelDataIndex: indices[j - 1], - defaultText: data.getName(indices[j - 1]), - inheritColor: style.fill - }, { - normal: { - verticalAlign: 'middle' // align: 'right' - - } - }); - polygon.setTextConfig({ - position: null, - local: true - }); - var labelEl = polygon.getTextContent(); // TODO More label position options. - - if (labelEl) { - labelEl.x = textLayout.x - margin; - labelEl.y = textLayout.y0 + textLayout.y / 2; - } - - polygon.useStyle(style); - data.setItemGraphicEl(idx, polygon); - setStatesStylesFromModel(polygon, seriesModel); - toggleHoverEmphasis(polygon, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - } - - this._layersSeries = layersSeries; - this._layers = newLayersGroups; - }; - - ThemeRiverView.type = 'themeRiver'; - return ThemeRiverView; - }(ChartView); - - function createGridClipShape$2(rect, seriesModel, cb) { - var rectEl = new Rect({ - shape: { - x: rect.x - 10, - y: rect.y - 10, - width: 0, - height: rect.height + 20 - } - }); - initProps(rectEl, { - shape: { - x: rect.x - 50, - width: rect.width + 100, - height: rect.height + 20 - } - }, seriesModel, cb); - return rectEl; - } - - var DATA_NAME_INDEX = 2; - - var ThemeRiverSeriesModel = - /** @class */ - function (_super) { - __extends(ThemeRiverSeriesModel, _super); - - function ThemeRiverSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ThemeRiverSeriesModel.type; - return _this; - } - /** - * @override - */ - - - ThemeRiverSeriesModel.prototype.init = function (option) { - // eslint-disable-next-line - _super.prototype.init.apply(this, arguments); // Put this function here is for the sake of consistency of code style. - // Enable legend selection for each data item - // Use a function instead of direct access because data reference may changed - - - this.legendVisualProvider = new LegendVisualProvider(bind(this.getData, this), bind(this.getRawData, this)); - }; - /** - * If there is no value of a certain point in the time for some event,set it value to 0. - * - * @param {Array} data initial data in the option - * @return {Array} - */ - - - ThemeRiverSeriesModel.prototype.fixData = function (data) { - var rawDataLength = data.length; - /** - * Make sure every layer data get the same keys. - * The value index tells which layer has visited. - * { - * 2014/01/01: -1 - * } - */ - - var timeValueKeys = {}; // grouped data by name - - var groupResult = groupData(data, function (item) { - if (!timeValueKeys.hasOwnProperty(item[0] + '')) { - timeValueKeys[item[0] + ''] = -1; - } - - return item[2]; - }); - var layerData = []; - groupResult.buckets.each(function (items, key) { - layerData.push({ - name: key, - dataList: items - }); - }); - var layerNum = layerData.length; - - for (var k = 0; k < layerNum; ++k) { - var name_1 = layerData[k].name; - - for (var j = 0; j < layerData[k].dataList.length; ++j) { - var timeValue = layerData[k].dataList[j][0] + ''; - timeValueKeys[timeValue] = k; - } - - for (var timeValue in timeValueKeys) { - if (timeValueKeys.hasOwnProperty(timeValue) && timeValueKeys[timeValue] !== k) { - timeValueKeys[timeValue] = k; - data[rawDataLength] = [timeValue, 0, name_1]; - rawDataLength++; - } - } - } - - return data; - }; - /** - * @override - * @param option the initial option that user gave - * @param ecModel the model object for themeRiver option - */ - - - ThemeRiverSeriesModel.prototype.getInitialData = function (option, ecModel) { - var singleAxisModel = this.getReferringComponents('singleAxis', SINGLE_REFERRING).models[0]; - var axisType = singleAxisModel.get('type'); // filter the data item with the value of label is undefined - - var filterData = filter(option.data, function (dataItem) { - return dataItem[2] !== undefined; - }); // ??? TODO design a stage to transfer data for themeRiver and lines? - - var data = this.fixData(filterData || []); - var nameList = []; - var nameMap = this.nameMap = createHashMap(); - var count = 0; - - for (var i = 0; i < data.length; ++i) { - nameList.push(data[i][DATA_NAME_INDEX]); - - if (!nameMap.get(data[i][DATA_NAME_INDEX])) { - nameMap.set(data[i][DATA_NAME_INDEX], count); - count++; - } - } - - var dimensions = prepareSeriesDataSchema(data, { - coordDimensions: ['single'], - dimensionsDefine: [{ - name: 'time', - type: getDimensionTypeByAxis(axisType) - }, { - name: 'value', - type: 'float' - }, { - name: 'name', - type: 'ordinal' - }], - encodeDefine: { - single: 0, - value: 1, - itemName: 2 - } - }).dimensions; - var list = new SeriesData(dimensions, this); - list.initData(data); - return list; - }; - /** - * The raw data is divided into multiple layers and each layer - * has same name. - */ - - - ThemeRiverSeriesModel.prototype.getLayerSeries = function () { - var data = this.getData(); - var lenCount = data.count(); - var indexArr = []; - - for (var i = 0; i < lenCount; ++i) { - indexArr[i] = i; - } - - var timeDim = data.mapDimension('single'); // data group by name - - var groupResult = groupData(indexArr, function (index) { - return data.get('name', index); - }); - var layerSeries = []; - groupResult.buckets.each(function (items, key) { - items.sort(function (index1, index2) { - return data.get(timeDim, index1) - data.get(timeDim, index2); - }); - layerSeries.push({ - name: key, - indices: items - }); - }); - return layerSeries; - }; - /** - * Get data indices for show tooltip content - */ - - - ThemeRiverSeriesModel.prototype.getAxisTooltipData = function (dim, value, baseAxis) { - if (!isArray(dim)) { - dim = dim ? [dim] : []; - } - - var data = this.getData(); - var layerSeries = this.getLayerSeries(); - var indices = []; - var layerNum = layerSeries.length; - var nestestValue; - - for (var i = 0; i < layerNum; ++i) { - var minDist = Number.MAX_VALUE; - var nearestIdx = -1; - var pointNum = layerSeries[i].indices.length; - - for (var j = 0; j < pointNum; ++j) { - var theValue = data.get(dim[0], layerSeries[i].indices[j]); - var dist = Math.abs(theValue - value); - - if (dist <= minDist) { - nestestValue = theValue; - minDist = dist; - nearestIdx = layerSeries[i].indices[j]; - } - } - - indices.push(nearestIdx); - } - - return { - dataIndices: indices, - nestestValue: nestestValue - }; - }; - - ThemeRiverSeriesModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - var data = this.getData(); - var name = data.getName(dataIndex); - var value = data.get(data.mapDimension('value'), dataIndex); - return createTooltipMarkup('nameValue', { - name: name, - value: value - }); - }; - - ThemeRiverSeriesModel.type = 'series.themeRiver'; - ThemeRiverSeriesModel.dependencies = ['singleAxis']; - ThemeRiverSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - colorBy: 'data', - coordinateSystem: 'singleAxis', - // gap in axis's orthogonal orientation - boundaryGap: ['10%', '10%'], - // legendHoverLink: true, - singleAxisIndex: 0, - animationEasing: 'linear', - label: { - margin: 4, - show: true, - position: 'left', - fontSize: 11 - }, - emphasis: { - label: { - show: true - } - } - }; - return ThemeRiverSeriesModel; - }(SeriesModel); - - function themeRiverLayout(ecModel, api) { - ecModel.eachSeriesByType('themeRiver', function (seriesModel) { - var data = seriesModel.getData(); - var single = seriesModel.coordinateSystem; - var layoutInfo = {}; // use the axis boundingRect for view - - var rect = single.getRect(); - layoutInfo.rect = rect; - var boundaryGap = seriesModel.get('boundaryGap'); - var axis = single.getAxis(); - layoutInfo.boundaryGap = boundaryGap; - - if (axis.orient === 'horizontal') { - boundaryGap[0] = parsePercent$1(boundaryGap[0], rect.height); - boundaryGap[1] = parsePercent$1(boundaryGap[1], rect.height); - var height = rect.height - boundaryGap[0] - boundaryGap[1]; - doThemeRiverLayout(data, seriesModel, height); - } else { - boundaryGap[0] = parsePercent$1(boundaryGap[0], rect.width); - boundaryGap[1] = parsePercent$1(boundaryGap[1], rect.width); - var width = rect.width - boundaryGap[0] - boundaryGap[1]; - doThemeRiverLayout(data, seriesModel, width); - } - - data.setLayout('layoutInfo', layoutInfo); - }); - } - /** - * The layout information about themeriver - * - * @param data data in the series - * @param seriesModel the model object of themeRiver series - * @param height value used to compute every series height - */ - - function doThemeRiverLayout(data, seriesModel, height) { - if (!data.count()) { - return; - } - - var coordSys = seriesModel.coordinateSystem; // the data in each layer are organized into a series. - - var layerSeries = seriesModel.getLayerSeries(); // the points in each layer. - - var timeDim = data.mapDimension('single'); - var valueDim = data.mapDimension('value'); - var layerPoints = map(layerSeries, function (singleLayer) { - return map(singleLayer.indices, function (idx) { - var pt = coordSys.dataToPoint(data.get(timeDim, idx)); - pt[1] = data.get(valueDim, idx); - return pt; - }); - }); - var base = computeBaseline(layerPoints); - var baseLine = base.y0; - var ky = height / base.max; // set layout information for each item. - - var n = layerSeries.length; - var m = layerSeries[0].indices.length; - var baseY0; - - for (var j = 0; j < m; ++j) { - baseY0 = baseLine[j] * ky; - data.setItemLayout(layerSeries[0].indices[j], { - layerIndex: 0, - x: layerPoints[0][j][0], - y0: baseY0, - y: layerPoints[0][j][1] * ky - }); - - for (var i = 1; i < n; ++i) { - baseY0 += layerPoints[i - 1][j][1] * ky; - data.setItemLayout(layerSeries[i].indices[j], { - layerIndex: i, - x: layerPoints[i][j][0], - y0: baseY0, - y: layerPoints[i][j][1] * ky - }); - } - } - } - /** - * Compute the baseLine of the rawdata - * Inspired by Lee Byron's paper Stacked Graphs - Geometry & Aesthetics - * - * @param data the points in each layer - */ - - - function computeBaseline(data) { - var layerNum = data.length; - var pointNum = data[0].length; - var sums = []; - var y0 = []; - var max = 0; - - for (var i = 0; i < pointNum; ++i) { - var temp = 0; - - for (var j = 0; j < layerNum; ++j) { - temp += data[j][i][1]; - } - - if (temp > max) { - max = temp; - } - - sums.push(temp); - } - - for (var k = 0; k < pointNum; ++k) { - y0[k] = (max - sums[k]) / 2; - } - - max = 0; - - for (var l = 0; l < pointNum; ++l) { - var sum = sums[l] + y0[l]; - - if (sum > max) { - max = sum; - } - } - - return { - y0: y0, - max: max - }; - } - - function install$p(registers) { - registers.registerChartView(ThemeRiverView); - registers.registerSeriesModel(ThemeRiverSeriesModel); - registers.registerLayout(themeRiverLayout); - registers.registerProcessor(dataFilter('themeRiver')); - } - - var DEFAULT_SECTOR_Z = 2; - var DEFAULT_TEXT_Z = 4; - /** - * Sunburstce of Sunburst including Sector, Label, LabelLine - */ - - var SunburstPiece = - /** @class */ - function (_super) { - __extends(SunburstPiece, _super); - - function SunburstPiece(node, seriesModel, ecModel, api) { - var _this = _super.call(this) || this; - - _this.z2 = DEFAULT_SECTOR_Z; - _this.textConfig = { - inside: true - }; - getECData(_this).seriesIndex = seriesModel.seriesIndex; - var text = new ZRText({ - z2: DEFAULT_TEXT_Z, - silent: node.getModel().get(['label', 'silent']) - }); - - _this.setTextContent(text); - - _this.updateData(true, node, seriesModel, ecModel, api); - - return _this; - } - - SunburstPiece.prototype.updateData = function (firstCreate, node, // state: 'emphasis' | 'normal' | 'highlight' | 'downplay', - seriesModel, ecModel, api) { - this.node = node; - node.piece = this; - seriesModel = seriesModel || this._seriesModel; - ecModel = ecModel || this._ecModel; - var sector = this; - getECData(sector).dataIndex = node.dataIndex; - var itemModel = node.getModel(); - var emphasisModel = itemModel.getModel('emphasis'); - var layout = node.getLayout(); - var sectorShape = extend({}, layout); - sectorShape.label = null; - var normalStyle = node.getVisual('style'); - normalStyle.lineJoin = 'bevel'; - var decal = node.getVisual('decal'); - - if (decal) { - normalStyle.decal = createOrUpdatePatternFromDecal(decal, api); - } - - var cornerRadius = getSectorCornerRadius(itemModel.getModel('itemStyle'), sectorShape, true); - extend(sectorShape, cornerRadius); - each(SPECIAL_STATES, function (stateName) { - var state = sector.ensureState(stateName); - var itemStyleModel = itemModel.getModel([stateName, 'itemStyle']); - state.style = itemStyleModel.getItemStyle(); // border radius - - var cornerRadius = getSectorCornerRadius(itemStyleModel, sectorShape); - - if (cornerRadius) { - state.shape = cornerRadius; - } - }); - - if (firstCreate) { - sector.setShape(sectorShape); - sector.shape.r = layout.r0; - initProps(sector, { - shape: { - r: layout.r - } - }, seriesModel, node.dataIndex); - } else { - // Disable animation for gradient since no interpolation method - // is supported for gradient - updateProps(sector, { - shape: sectorShape - }, seriesModel); - saveOldStyle(sector); - } - - sector.useStyle(normalStyle); - - this._updateLabel(seriesModel); - - var cursorStyle = itemModel.getShallow('cursor'); - cursorStyle && sector.attr('cursor', cursorStyle); - this._seriesModel = seriesModel || this._seriesModel; - this._ecModel = ecModel || this._ecModel; - var focus = emphasisModel.get('focus'); - var focusOrIndices = focus === 'ancestor' ? node.getAncestorsIndices() : focus === 'descendant' ? node.getDescendantIndices() : focus; - toggleHoverEmphasis(this, focusOrIndices, emphasisModel.get('blurScope'), emphasisModel.get('disabled')); - }; - - SunburstPiece.prototype._updateLabel = function (seriesModel) { - var _this = this; - - var itemModel = this.node.getModel(); - var normalLabelModel = itemModel.getModel('label'); - var layout = this.node.getLayout(); - var angle = layout.endAngle - layout.startAngle; - var midAngle = (layout.startAngle + layout.endAngle) / 2; - var dx = Math.cos(midAngle); - var dy = Math.sin(midAngle); - var sector = this; - var label = sector.getTextContent(); - var dataIndex = this.node.dataIndex; - var labelMinAngle = normalLabelModel.get('minAngle') / 180 * Math.PI; - var isNormalShown = normalLabelModel.get('show') && !(labelMinAngle != null && Math.abs(angle) < labelMinAngle); - label.ignore = !isNormalShown; // TODO use setLabelStyle - - each(DISPLAY_STATES, function (stateName) { - var labelStateModel = stateName === 'normal' ? itemModel.getModel('label') : itemModel.getModel([stateName, 'label']); - var isNormal = stateName === 'normal'; - var state = isNormal ? label : label.ensureState(stateName); - var text = seriesModel.getFormattedLabel(dataIndex, stateName); - - if (isNormal) { - text = text || _this.node.name; - } - - state.style = createTextStyle(labelStateModel, {}, null, stateName !== 'normal', true); - - if (text) { - state.style.text = text; - } // Not displaying text when angle is too small - - - var isShown = labelStateModel.get('show'); - - if (isShown != null && !isNormal) { - state.ignore = !isShown; - } - - var labelPosition = getLabelAttr(labelStateModel, 'position'); - var sectorState = isNormal ? sector : sector.states[stateName]; - var labelColor = sectorState.style.fill; - sectorState.textConfig = { - outsideFill: labelStateModel.get('color') === 'inherit' ? labelColor : null, - inside: labelPosition !== 'outside' - }; - var r; - var labelPadding = getLabelAttr(labelStateModel, 'distance') || 0; - var textAlign = getLabelAttr(labelStateModel, 'align'); - - if (labelPosition === 'outside') { - r = layout.r + labelPadding; - textAlign = midAngle > Math.PI / 2 ? 'right' : 'left'; - } else { - if (!textAlign || textAlign === 'center') { - // Put label in the center if it's a circle - if (angle === 2 * Math.PI && layout.r0 === 0) { - r = 0; - } else { - r = (layout.r + layout.r0) / 2; - } - - textAlign = 'center'; - } else if (textAlign === 'left') { - r = layout.r0 + labelPadding; - - if (midAngle > Math.PI / 2) { - textAlign = 'right'; - } - } else if (textAlign === 'right') { - r = layout.r - labelPadding; - - if (midAngle > Math.PI / 2) { - textAlign = 'left'; - } - } - } - - state.style.align = textAlign; - state.style.verticalAlign = getLabelAttr(labelStateModel, 'verticalAlign') || 'middle'; - state.x = r * dx + layout.cx; - state.y = r * dy + layout.cy; - var rotateType = getLabelAttr(labelStateModel, 'rotate'); - var rotate = 0; - - if (rotateType === 'radial') { - rotate = normalizeRadian(-midAngle); - - if (rotate > Math.PI / 2 && rotate < Math.PI * 1.5) { - rotate += Math.PI; - } - } else if (rotateType === 'tangential') { - rotate = Math.PI / 2 - midAngle; - - if (rotate > Math.PI / 2) { - rotate -= Math.PI; - } else if (rotate < -Math.PI / 2) { - rotate += Math.PI; - } - } else if (isNumber(rotateType)) { - rotate = rotateType * Math.PI / 180; - } - - state.rotation = rotate; - }); - - function getLabelAttr(model, name) { - var stateAttr = model.get(name); - - if (stateAttr == null) { - return normalLabelModel.get(name); - } - - return stateAttr; - } - - label.dirtyStyle(); - }; - - return SunburstPiece; - }(Sector); - - var ROOT_TO_NODE_ACTION = 'sunburstRootToNode'; - var HIGHLIGHT_ACTION = 'sunburstHighlight'; - var UNHIGHLIGHT_ACTION = 'sunburstUnhighlight'; - function installSunburstAction(registers) { - registers.registerAction({ - type: ROOT_TO_NODE_ACTION, - update: 'updateView' - }, function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'series', - subType: 'sunburst', - query: payload - }, handleRootToNode); - - function handleRootToNode(model, index) { - var targetInfo = retrieveTargetInfo(payload, [ROOT_TO_NODE_ACTION], model); - - if (targetInfo) { - var originViewRoot = model.getViewRoot(); - - if (originViewRoot) { - payload.direction = aboveViewRoot(originViewRoot, targetInfo.node) ? 'rollUp' : 'drillDown'; - } - - model.resetViewRoot(targetInfo.node); - } - } - }); - registers.registerAction({ - type: HIGHLIGHT_ACTION, - update: 'none' - }, function (payload, ecModel, api) { - // Clone - payload = extend({}, payload); - ecModel.eachComponent({ - mainType: 'series', - subType: 'sunburst', - query: payload - }, handleHighlight); - - function handleHighlight(model) { - var targetInfo = retrieveTargetInfo(payload, [HIGHLIGHT_ACTION], model); - - if (targetInfo) { - payload.dataIndex = targetInfo.node.dataIndex; - } - } - - if ("development" !== 'production') { - deprecateReplaceLog('sunburstHighlight', 'highlight'); - } // Fast forward action - - - api.dispatchAction(extend(payload, { - type: 'highlight' - })); - }); - registers.registerAction({ - type: UNHIGHLIGHT_ACTION, - update: 'updateView' - }, function (payload, ecModel, api) { - payload = extend({}, payload); - - if ("development" !== 'production') { - deprecateReplaceLog('sunburstUnhighlight', 'downplay'); - } - - api.dispatchAction(extend(payload, { - type: 'downplay' - })); - }); - } - - var SunburstView = - /** @class */ - function (_super) { - __extends(SunburstView, _super); - - function SunburstView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SunburstView.type; - return _this; - } - - SunburstView.prototype.render = function (seriesModel, ecModel, api, // @ts-ignore - payload) { - var self = this; - this.seriesModel = seriesModel; - this.api = api; - this.ecModel = ecModel; - var data = seriesModel.getData(); - var virtualRoot = data.tree.root; - var newRoot = seriesModel.getViewRoot(); - var group = this.group; - var renderLabelForZeroData = seriesModel.get('renderLabelForZeroData'); - var newChildren = []; - newRoot.eachNode(function (node) { - newChildren.push(node); - }); - var oldChildren = this._oldChildren || []; - dualTravel(newChildren, oldChildren); - renderRollUp(virtualRoot, newRoot); - - this._initEvents(); - - this._oldChildren = newChildren; - - function dualTravel(newChildren, oldChildren) { - if (newChildren.length === 0 && oldChildren.length === 0) { - return; - } - - new DataDiffer(oldChildren, newChildren, getKey, getKey).add(processNode).update(processNode).remove(curry(processNode, null)).execute(); - - function getKey(node) { - return node.getId(); - } - - function processNode(newIdx, oldIdx) { - var newNode = newIdx == null ? null : newChildren[newIdx]; - var oldNode = oldIdx == null ? null : oldChildren[oldIdx]; - doRenderNode(newNode, oldNode); - } - } - - function doRenderNode(newNode, oldNode) { - if (!renderLabelForZeroData && newNode && !newNode.getValue()) { - // Not render data with value 0 - newNode = null; - } - - if (newNode !== virtualRoot && oldNode !== virtualRoot) { - if (oldNode && oldNode.piece) { - if (newNode) { - // Update - oldNode.piece.updateData(false, newNode, seriesModel, ecModel, api); // For tooltip - - data.setItemGraphicEl(newNode.dataIndex, oldNode.piece); - } else { - // Remove - removeNode(oldNode); - } - } else if (newNode) { - // Add - var piece = new SunburstPiece(newNode, seriesModel, ecModel, api); - group.add(piece); // For tooltip - - data.setItemGraphicEl(newNode.dataIndex, piece); - } - } - } - - function removeNode(node) { - if (!node) { - return; - } - - if (node.piece) { - group.remove(node.piece); - node.piece = null; - } - } - - function renderRollUp(virtualRoot, viewRoot) { - if (viewRoot.depth > 0) { - // Render - if (self.virtualPiece) { - // Update - self.virtualPiece.updateData(false, virtualRoot, seriesModel, ecModel, api); - } else { - // Add - self.virtualPiece = new SunburstPiece(virtualRoot, seriesModel, ecModel, api); - group.add(self.virtualPiece); - } // TODO event scope - - - viewRoot.piece.off('click'); - self.virtualPiece.on('click', function (e) { - self._rootToNode(viewRoot.parentNode); - }); - } else if (self.virtualPiece) { - // Remove - group.remove(self.virtualPiece); - self.virtualPiece = null; - } - } - }; - /** - * @private - */ - - - SunburstView.prototype._initEvents = function () { - var _this = this; - - this.group.off('click'); - this.group.on('click', function (e) { - var targetFound = false; - - var viewRoot = _this.seriesModel.getViewRoot(); - - viewRoot.eachNode(function (node) { - if (!targetFound && node.piece && node.piece === e.target) { - var nodeClick = node.getModel().get('nodeClick'); - - if (nodeClick === 'rootToNode') { - _this._rootToNode(node); - } else if (nodeClick === 'link') { - var itemModel = node.getModel(); - var link = itemModel.get('link'); - - if (link) { - var linkTarget = itemModel.get('target', true) || '_blank'; - windowOpen(link, linkTarget); - } - } - - targetFound = true; - } - }); - }); - }; - /** - * @private - */ - - - SunburstView.prototype._rootToNode = function (node) { - if (node !== this.seriesModel.getViewRoot()) { - this.api.dispatchAction({ - type: ROOT_TO_NODE_ACTION, - from: this.uid, - seriesId: this.seriesModel.id, - targetNode: node - }); - } - }; - /** - * @implement - */ - - - SunburstView.prototype.containPoint = function (point, seriesModel) { - var treeRoot = seriesModel.getData(); - var itemLayout = treeRoot.getItemLayout(0); - - if (itemLayout) { - var dx = point[0] - itemLayout.cx; - var dy = point[1] - itemLayout.cy; - var radius = Math.sqrt(dx * dx + dy * dy); - return radius <= itemLayout.r && radius >= itemLayout.r0; - } - }; - - SunburstView.type = 'sunburst'; - return SunburstView; - }(ChartView); - - var SunburstSeriesModel = - /** @class */ - function (_super) { - __extends(SunburstSeriesModel, _super); - - function SunburstSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SunburstSeriesModel.type; - _this.ignoreStyleOnData = true; - return _this; - } - - SunburstSeriesModel.prototype.getInitialData = function (option, ecModel) { - // Create a virtual root. - var root = { - name: option.name, - children: option.data - }; - completeTreeValue$1(root); - var levelModels = this._levelModels = map(option.levels || [], function (levelDefine) { - return new Model(levelDefine, this, ecModel); - }, this); // Make sure always a new tree is created when setOption, - // in TreemapView, we check whether oldTree === newTree - // to choose mappings approach among old shapes and new shapes. - - var tree = Tree.createTree(root, this, beforeLink); - - function beforeLink(nodeData) { - nodeData.wrapMethod('getItemModel', function (model, idx) { - var node = tree.getNodeByDataIndex(idx); - var levelModel = levelModels[node.depth]; - levelModel && (model.parentModel = levelModel); - return model; - }); - } - - return tree.data; - }; - - SunburstSeriesModel.prototype.optionUpdated = function () { - this.resetViewRoot(); - }; - /* - * @override - */ - - - SunburstSeriesModel.prototype.getDataParams = function (dataIndex) { - var params = _super.prototype.getDataParams.apply(this, arguments); - - var node = this.getData().tree.getNodeByDataIndex(dataIndex); - params.treePathInfo = wrapTreePathInfo(node, this); - return params; - }; - - SunburstSeriesModel.prototype.getLevelModel = function (node) { - return this._levelModels && this._levelModels[node.depth]; - }; - - SunburstSeriesModel.prototype.getViewRoot = function () { - return this._viewRoot; - }; - - SunburstSeriesModel.prototype.resetViewRoot = function (viewRoot) { - viewRoot ? this._viewRoot = viewRoot : viewRoot = this._viewRoot; - var root = this.getRawData().tree.root; - - if (!viewRoot || viewRoot !== root && !root.contains(viewRoot)) { - this._viewRoot = root; - } - }; - - SunburstSeriesModel.prototype.enableAriaDecal = function () { - enableAriaDecalForTree(this); - }; - - SunburstSeriesModel.type = 'series.sunburst'; - SunburstSeriesModel.defaultOption = { - // zlevel: 0, - z: 2, - // 默认全局居中 - center: ['50%', '50%'], - radius: [0, '75%'], - // 默认顺时针 - clockwise: true, - startAngle: 90, - // 最小角度改为0 - minAngle: 0, - // If still show when all data zero. - stillShowZeroSum: true, - // 'rootToNode', 'link', or false - nodeClick: 'rootToNode', - renderLabelForZeroData: false, - label: { - // could be: 'radial', 'tangential', or 'none' - rotate: 'radial', - show: true, - opacity: 1, - // 'left' is for inner side of inside, and 'right' is for outer - // side for inside - align: 'center', - position: 'inside', - distance: 5, - silent: true - }, - itemStyle: { - borderWidth: 1, - borderColor: 'white', - borderType: 'solid', - shadowBlur: 0, - shadowColor: 'rgba(0, 0, 0, 0.2)', - shadowOffsetX: 0, - shadowOffsetY: 0, - opacity: 1 - }, - emphasis: { - focus: 'descendant' - }, - blur: { - itemStyle: { - opacity: 0.2 - }, - label: { - opacity: 0.1 - } - }, - // Animation type can be expansion, scale. - animationType: 'expansion', - animationDuration: 1000, - animationDurationUpdate: 500, - data: [], - - /** - * Sort order. - * - * Valid values: 'desc', 'asc', null, or callback function. - * 'desc' and 'asc' for descend and ascendant order; - * null for not sorting; - * example of callback function: - * function(nodeA, nodeB) { - * return nodeA.getValue() - nodeB.getValue(); - * } - */ - sort: 'desc' - }; - return SunburstSeriesModel; - }(SeriesModel); - - function completeTreeValue$1(dataNode) { - // Postorder travel tree. - // If value of none-leaf node is not set, - // calculate it by suming up the value of all children. - var sum = 0; - each(dataNode.children, function (child) { - completeTreeValue$1(child); - var childValue = child.value; // TODO First value of array must be a number - - isArray(childValue) && (childValue = childValue[0]); - sum += childValue; - }); - var thisValue = dataNode.value; - - if (isArray(thisValue)) { - thisValue = thisValue[0]; - } - - if (thisValue == null || isNaN(thisValue)) { - thisValue = sum; - } // Value should not less than 0. - - - if (thisValue < 0) { - thisValue = 0; - } - - isArray(dataNode.value) ? dataNode.value[0] = thisValue : dataNode.value = thisValue; - } - - var RADIAN$2 = Math.PI / 180; - function sunburstLayout(seriesType, ecModel, api) { - ecModel.eachSeriesByType(seriesType, function (seriesModel) { - var center = seriesModel.get('center'); - var radius = seriesModel.get('radius'); - - if (!isArray(radius)) { - radius = [0, radius]; - } - - if (!isArray(center)) { - center = [center, center]; - } - - var width = api.getWidth(); - var height = api.getHeight(); - var size = Math.min(width, height); - var cx = parsePercent$1(center[0], width); - var cy = parsePercent$1(center[1], height); - var r0 = parsePercent$1(radius[0], size / 2); - var r = parsePercent$1(radius[1], size / 2); - var startAngle = -seriesModel.get('startAngle') * RADIAN$2; - var minAngle = seriesModel.get('minAngle') * RADIAN$2; - var virtualRoot = seriesModel.getData().tree.root; - var treeRoot = seriesModel.getViewRoot(); - var rootDepth = treeRoot.depth; - var sort = seriesModel.get('sort'); - - if (sort != null) { - initChildren$1(treeRoot, sort); - } - - var validDataCount = 0; - each(treeRoot.children, function (child) { - !isNaN(child.getValue()) && validDataCount++; - }); - var sum = treeRoot.getValue(); // Sum may be 0 - - var unitRadian = Math.PI / (sum || validDataCount) * 2; - var renderRollupNode = treeRoot.depth > 0; - var levels = treeRoot.height - (renderRollupNode ? -1 : 1); - var rPerLevel = (r - r0) / (levels || 1); - var clockwise = seriesModel.get('clockwise'); - var stillShowZeroSum = seriesModel.get('stillShowZeroSum'); // In the case some sector angle is smaller than minAngle - // let restAngle = PI2; - // let valueSumLargerThanMinAngle = 0; - - var dir = clockwise ? 1 : -1; - /** - * Render a tree - * @return increased angle - */ - - var renderNode = function (node, startAngle) { - if (!node) { - return; - } - - var endAngle = startAngle; // Render self - - if (node !== virtualRoot) { - // Tree node is virtual, so it doesn't need to be drawn - var value = node.getValue(); - var angle = sum === 0 && stillShowZeroSum ? unitRadian : value * unitRadian; - - if (angle < minAngle) { - angle = minAngle; // restAngle -= minAngle; - } // else { - // valueSumLargerThanMinAngle += value; - // } - - - endAngle = startAngle + dir * angle; - var depth = node.depth - rootDepth - (renderRollupNode ? -1 : 1); - var rStart = r0 + rPerLevel * depth; - var rEnd = r0 + rPerLevel * (depth + 1); - var levelModel = seriesModel.getLevelModel(node); - - if (levelModel) { - var r0_1 = levelModel.get('r0', true); - var r_1 = levelModel.get('r', true); - var radius_1 = levelModel.get('radius', true); - - if (radius_1 != null) { - r0_1 = radius_1[0]; - r_1 = radius_1[1]; - } - - r0_1 != null && (rStart = parsePercent$1(r0_1, size / 2)); - r_1 != null && (rEnd = parsePercent$1(r_1, size / 2)); - } - - node.setLayout({ - angle: angle, - startAngle: startAngle, - endAngle: endAngle, - clockwise: clockwise, - cx: cx, - cy: cy, - r0: rStart, - r: rEnd - }); - } // Render children - - - if (node.children && node.children.length) { - // currentAngle = startAngle; - var siblingAngle_1 = 0; - each(node.children, function (node) { - siblingAngle_1 += renderNode(node, startAngle + siblingAngle_1); - }); - } - - return endAngle - startAngle; - }; // Virtual root node for roll up - - - if (renderRollupNode) { - var rStart = r0; - var rEnd = r0 + rPerLevel; - var angle = Math.PI * 2; - virtualRoot.setLayout({ - angle: angle, - startAngle: startAngle, - endAngle: startAngle + angle, - clockwise: clockwise, - cx: cx, - cy: cy, - r0: rStart, - r: rEnd - }); - } - - renderNode(treeRoot, startAngle); - }); - } - /** - * Init node children by order and update visual - */ - - function initChildren$1(node, sortOrder) { - var children = node.children || []; - node.children = sort$2(children, sortOrder); // Init children recursively - - if (children.length) { - each(node.children, function (child) { - initChildren$1(child, sortOrder); - }); - } - } - /** - * Sort children nodes - * - * @param {TreeNode[]} children children of node to be sorted - * @param {string | function | null} sort sort method - * See SunburstSeries.js for details. - */ - - - function sort$2(children, sortOrder) { - if (isFunction(sortOrder)) { - var sortTargets = map(children, function (child, idx) { - var value = child.getValue(); - return { - params: { - depth: child.depth, - height: child.height, - dataIndex: child.dataIndex, - getValue: function () { - return value; - } - }, - index: idx - }; - }); - sortTargets.sort(function (a, b) { - return sortOrder(a.params, b.params); - }); - return map(sortTargets, function (target) { - return children[target.index]; - }); - } else { - var isAsc_1 = sortOrder === 'asc'; - return children.sort(function (a, b) { - var diff = (a.getValue() - b.getValue()) * (isAsc_1 ? 1 : -1); - return diff === 0 ? (a.dataIndex - b.dataIndex) * (isAsc_1 ? -1 : 1) : diff; - }); - } - } - - function sunburstVisual(ecModel) { - var paletteScope = {}; // Default color strategy - - function pickColor(node, seriesModel, treeHeight) { - // Choose color from palette based on the first level. - var current = node; - - while (current && current.depth > 1) { - current = current.parentNode; - } - - var color = seriesModel.getColorFromPalette(current.name || current.dataIndex + '', paletteScope); - - if (node.depth > 1 && isString(color)) { - // Lighter on the deeper level. - color = lift(color, (node.depth - 1) / (treeHeight - 1) * 0.5); - } - - return color; - } - - ecModel.eachSeriesByType('sunburst', function (seriesModel) { - var data = seriesModel.getData(); - var tree = data.tree; - tree.eachNode(function (node) { - var model = node.getModel(); - var style = model.getModel('itemStyle').getItemStyle(); - - if (!style.fill) { - style.fill = pickColor(node, seriesModel, tree.root.height); - } - - var existsStyle = data.ensureUniqueItemVisual(node.dataIndex, 'style'); - extend(existsStyle, style); - }); - }); - } - - function install$q(registers) { - registers.registerChartView(SunburstView); - registers.registerSeriesModel(SunburstSeriesModel); - registers.registerLayout(curry(sunburstLayout, 'sunburst')); - registers.registerProcessor(curry(dataFilter, 'sunburst')); - registers.registerVisual(sunburstVisual); - installSunburstAction(registers); - } - - // `visual('color') visual('borderColor')` is supported. - - var STYLE_VISUAL_TYPE = { - color: 'fill', - borderColor: 'stroke' - }; - var NON_STYLE_VISUAL_PROPS = { - symbol: 1, - symbolSize: 1, - symbolKeepAspect: 1, - legendIcon: 1, - visualMeta: 1, - liftZ: 1, - decal: 1 - }; - var customInnerStore = makeInner(); - - var CustomSeriesModel = - /** @class */ - function (_super) { - __extends(CustomSeriesModel, _super); - - function CustomSeriesModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CustomSeriesModel.type; - return _this; - } - - CustomSeriesModel.prototype.optionUpdated = function () { - this.currentZLevel = this.get('zlevel', true); - this.currentZ = this.get('z', true); - }; - - CustomSeriesModel.prototype.getInitialData = function (option, ecModel) { - return createSeriesData(null, this); - }; - - CustomSeriesModel.prototype.getDataParams = function (dataIndex, dataType, el) { - var params = _super.prototype.getDataParams.call(this, dataIndex, dataType); - - el && (params.info = customInnerStore(el).info); - return params; - }; - - CustomSeriesModel.type = 'series.custom'; - CustomSeriesModel.dependencies = ['grid', 'polar', 'geo', 'singleAxis', 'calendar']; - CustomSeriesModel.defaultOption = { - coordinateSystem: 'cartesian2d', - // zlevel: 0, - z: 2, - legendHoverLink: true, - // Custom series will not clip by default. - // Some case will use custom series to draw label - // For example https://echarts.apache.org/examples/en/editor.html?c=custom-gantt-flight - clip: false // Cartesian coordinate system - // xAxisIndex: 0, - // yAxisIndex: 0, - // Polar coordinate system - // polarIndex: 0, - // Geo coordinate system - // geoIndex: 0, - - }; - return CustomSeriesModel; - }(SeriesModel); - - function dataToCoordSize(dataSize, dataItem) { - // dataItem is necessary in log axis. - dataItem = dataItem || [0, 0]; - return map(['x', 'y'], function (dim, dimIdx) { - var axis = this.getAxis(dim); - var val = dataItem[dimIdx]; - var halfSize = dataSize[dimIdx] / 2; - return axis.type === 'category' ? axis.getBandWidth() : Math.abs(axis.dataToCoord(val - halfSize) - axis.dataToCoord(val + halfSize)); - }, this); - } - - function cartesianPrepareCustom(coordSys) { - var rect = coordSys.master.getRect(); - return { - coordSys: { - // The name exposed to user is always 'cartesian2d' but not 'grid'. - type: 'cartesian2d', - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height - }, - api: { - coord: function (data) { - // do not provide "out" param - return coordSys.dataToPoint(data); - }, - size: bind(dataToCoordSize, coordSys) - } - }; - } - - function dataToCoordSize$1(dataSize, dataItem) { - dataItem = dataItem || [0, 0]; - return map([0, 1], function (dimIdx) { - var val = dataItem[dimIdx]; - var halfSize = dataSize[dimIdx] / 2; - var p1 = []; - var p2 = []; - p1[dimIdx] = val - halfSize; - p2[dimIdx] = val + halfSize; - p1[1 - dimIdx] = p2[1 - dimIdx] = dataItem[1 - dimIdx]; - return Math.abs(this.dataToPoint(p1)[dimIdx] - this.dataToPoint(p2)[dimIdx]); - }, this); - } - - function geoPrepareCustom(coordSys) { - var rect = coordSys.getBoundingRect(); - return { - coordSys: { - type: 'geo', - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height, - zoom: coordSys.getZoom() - }, - api: { - coord: function (data) { - // do not provide "out" and noRoam param, - // Compatible with this usage: - // echarts.util.map(item.points, api.coord) - return coordSys.dataToPoint(data); - }, - size: bind(dataToCoordSize$1, coordSys) - } - }; - } - - function dataToCoordSize$2(dataSize, dataItem) { - // dataItem is necessary in log axis. - var axis = this.getAxis(); - var val = dataItem instanceof Array ? dataItem[0] : dataItem; - var halfSize = (dataSize instanceof Array ? dataSize[0] : dataSize) / 2; - return axis.type === 'category' ? axis.getBandWidth() : Math.abs(axis.dataToCoord(val - halfSize) - axis.dataToCoord(val + halfSize)); - } - - function singlePrepareCustom(coordSys) { - var rect = coordSys.getRect(); - return { - coordSys: { - type: 'singleAxis', - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height - }, - api: { - coord: function (val) { - // do not provide "out" param - return coordSys.dataToPoint(val); - }, - size: bind(dataToCoordSize$2, coordSys) - } - }; - } - - function dataToCoordSize$3(dataSize, dataItem) { - // dataItem is necessary in log axis. - dataItem = dataItem || [0, 0]; - return map(['Radius', 'Angle'], function (dim, dimIdx) { - var getterName = 'get' + dim + 'Axis'; // TODO: TYPE Check Angle Axis - - var axis = this[getterName](); - var val = dataItem[dimIdx]; - var halfSize = dataSize[dimIdx] / 2; - var result = axis.type === 'category' ? axis.getBandWidth() : Math.abs(axis.dataToCoord(val - halfSize) - axis.dataToCoord(val + halfSize)); - - if (dim === 'Angle') { - result = result * Math.PI / 180; - } - - return result; - }, this); - } - - function polarPrepareCustom(coordSys) { - var radiusAxis = coordSys.getRadiusAxis(); - var angleAxis = coordSys.getAngleAxis(); - var radius = radiusAxis.getExtent(); - radius[0] > radius[1] && radius.reverse(); - return { - coordSys: { - type: 'polar', - cx: coordSys.cx, - cy: coordSys.cy, - r: radius[1], - r0: radius[0] - }, - api: { - coord: function (data) { - var radius = radiusAxis.dataToRadius(data[0]); - var angle = angleAxis.dataToAngle(data[1]); - var coord = coordSys.coordToPoint([radius, angle]); - coord.push(radius, angle * Math.PI / 180); - return coord; - }, - size: bind(dataToCoordSize$3, coordSys) - } - }; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function calendarPrepareCustom(coordSys) { - var rect = coordSys.getRect(); - var rangeInfo = coordSys.getRangeInfo(); - return { - coordSys: { - type: 'calendar', - x: rect.x, - y: rect.y, - width: rect.width, - height: rect.height, - cellWidth: coordSys.getCellWidth(), - cellHeight: coordSys.getCellHeight(), - rangeInfo: { - start: rangeInfo.start, - end: rangeInfo.end, - weeks: rangeInfo.weeks, - dayCount: rangeInfo.allDay - } - }, - api: { - coord: function (data, clamp) { - return coordSys.dataToPoint(data, clamp); - } - } - }; - } - - var deprecatedLogs = {}; - /** - * Whether need to call `convertEC4CompatibleStyle`. - */ - - function isEC4CompatibleStyle(style, elType, hasOwnTextContentOption, hasOwnTextConfig) { - // Since echarts5, `RectText` is separated from its host element and style.text - // does not exist any more. The compat work brings some extra burden on performance. - // So we provide: - // `legacy: true` force make compat. - // `legacy: false`, force do not compat. - // `legacy` not set: auto detect whether legacy. - // But in this case we do not compat (difficult to detect and rare case): - // Becuse custom series and graphic component support "merge", users may firstly - // only set `textStrokeWidth` style or secondly only set `text`. - return style && (style.legacy || style.legacy !== false && !hasOwnTextContentOption && !hasOwnTextConfig && elType !== 'tspan' // Difficult to detect whether legacy for a "text" el. - && (elType === 'text' || hasOwn(style, 'text'))); - } - /** - * `EC4CompatibleStyle` is style that might be in echarts4 format or echarts5 format. - * @param hostStyle The properties might be modified. - * @return If be text el, `textContentStyle` and `textConfig` will not be returned. - * Otherwise a `textContentStyle` and `textConfig` will be created, whose props area - * retried from the `hostStyle`. - */ - - function convertFromEC4CompatibleStyle(hostStyle, elType, isNormal) { - var srcStyle = hostStyle; - var textConfig; - var textContent; - var textContentStyle; - - if (elType === 'text') { - textContentStyle = srcStyle; - } else { - textContentStyle = {}; - hasOwn(srcStyle, 'text') && (textContentStyle.text = srcStyle.text); - hasOwn(srcStyle, 'rich') && (textContentStyle.rich = srcStyle.rich); - hasOwn(srcStyle, 'textFill') && (textContentStyle.fill = srcStyle.textFill); - hasOwn(srcStyle, 'textStroke') && (textContentStyle.stroke = srcStyle.textStroke); - hasOwn(srcStyle, 'fontFamily') && (textContentStyle.fontFamily = srcStyle.fontFamily); - hasOwn(srcStyle, 'fontSize') && (textContentStyle.fontSize = srcStyle.fontSize); - hasOwn(srcStyle, 'fontStyle') && (textContentStyle.fontStyle = srcStyle.fontStyle); - hasOwn(srcStyle, 'fontWeight') && (textContentStyle.fontWeight = srcStyle.fontWeight); - textContent = { - type: 'text', - style: textContentStyle, - // ec4 does not support rectText trigger. - // And when text position is different in normal and emphasis - // => hover text trigger emphasis; - // => text position changed, leave mouse pointer immediately; - // That might cause incorrect state. - silent: true - }; - textConfig = {}; - var hasOwnPos = hasOwn(srcStyle, 'textPosition'); - - if (isNormal) { - textConfig.position = hasOwnPos ? srcStyle.textPosition : 'inside'; - } else { - hasOwnPos && (textConfig.position = srcStyle.textPosition); - } - - hasOwn(srcStyle, 'textPosition') && (textConfig.position = srcStyle.textPosition); - hasOwn(srcStyle, 'textOffset') && (textConfig.offset = srcStyle.textOffset); - hasOwn(srcStyle, 'textRotation') && (textConfig.rotation = srcStyle.textRotation); - hasOwn(srcStyle, 'textDistance') && (textConfig.distance = srcStyle.textDistance); - } - - convertEC4CompatibleRichItem(textContentStyle, hostStyle); - each(textContentStyle.rich, function (richItem) { - convertEC4CompatibleRichItem(richItem, richItem); - }); - return { - textConfig: textConfig, - textContent: textContent - }; - } - /** - * The result will be set to `out`. - */ - - function convertEC4CompatibleRichItem(out, richItem) { - if (!richItem) { - return; - } // (1) For simplicity, make textXXX properties (deprecated since ec5) has - // higher priority. For example, consider in ec4 `borderColor: 5, textBorderColor: 10` - // on a rect means `borderColor: 4` on the rect and `borderColor: 10` on an attached - // richText in ec5. - // (2) `out === richItem` if and only if `out` is text el or rich item. - // So we can overwrite existing props in `out` since textXXX has higher priority. - - - richItem.font = richItem.textFont || richItem.font; - hasOwn(richItem, 'textStrokeWidth') && (out.lineWidth = richItem.textStrokeWidth); - hasOwn(richItem, 'textAlign') && (out.align = richItem.textAlign); - hasOwn(richItem, 'textVerticalAlign') && (out.verticalAlign = richItem.textVerticalAlign); - hasOwn(richItem, 'textLineHeight') && (out.lineHeight = richItem.textLineHeight); - hasOwn(richItem, 'textWidth') && (out.width = richItem.textWidth); - hasOwn(richItem, 'textHeight') && (out.height = richItem.textHeight); - hasOwn(richItem, 'textBackgroundColor') && (out.backgroundColor = richItem.textBackgroundColor); - hasOwn(richItem, 'textPadding') && (out.padding = richItem.textPadding); - hasOwn(richItem, 'textBorderColor') && (out.borderColor = richItem.textBorderColor); - hasOwn(richItem, 'textBorderWidth') && (out.borderWidth = richItem.textBorderWidth); - hasOwn(richItem, 'textBorderRadius') && (out.borderRadius = richItem.textBorderRadius); - hasOwn(richItem, 'textBoxShadowColor') && (out.shadowColor = richItem.textBoxShadowColor); - hasOwn(richItem, 'textBoxShadowBlur') && (out.shadowBlur = richItem.textBoxShadowBlur); - hasOwn(richItem, 'textBoxShadowOffsetX') && (out.shadowOffsetX = richItem.textBoxShadowOffsetX); - hasOwn(richItem, 'textBoxShadowOffsetY') && (out.shadowOffsetY = richItem.textBoxShadowOffsetY); - } - /** - * Convert to pure echarts4 format style. - * `itemStyle` will be modified, added with ec4 style properties from - * `textStyle` and `textConfig`. - * - * [Caveat]: For simplicity, `insideRollback` in ec4 does not compat, where - * `styleEmphasis: {textFill: 'red'}` will remove the normal auto added stroke. - */ - - - function convertToEC4StyleForCustomSerise(itemStl, txStl, txCfg) { - var out = itemStl; // See `custom.ts`, a trick to set extra `textPosition` firstly. - - out.textPosition = out.textPosition || txCfg.position || 'inside'; - txCfg.offset != null && (out.textOffset = txCfg.offset); - txCfg.rotation != null && (out.textRotation = txCfg.rotation); - txCfg.distance != null && (out.textDistance = txCfg.distance); - var isInside = out.textPosition.indexOf('inside') >= 0; - var hostFill = itemStl.fill || '#000'; - convertToEC4RichItem(out, txStl); - var textFillNotSet = out.textFill == null; - - if (isInside) { - if (textFillNotSet) { - out.textFill = txCfg.insideFill || '#fff'; - !out.textStroke && txCfg.insideStroke && (out.textStroke = txCfg.insideStroke); - !out.textStroke && (out.textStroke = hostFill); - out.textStrokeWidth == null && (out.textStrokeWidth = 2); - } - } else { - if (textFillNotSet) { - out.textFill = itemStl.fill || txCfg.outsideFill || '#000'; - } - - !out.textStroke && txCfg.outsideStroke && (out.textStroke = txCfg.outsideStroke); - } - - out.text = txStl.text; - out.rich = txStl.rich; - each(txStl.rich, function (richItem) { - convertToEC4RichItem(richItem, richItem); - }); - return out; - } - - function convertToEC4RichItem(out, richItem) { - if (!richItem) { - return; - } - - hasOwn(richItem, 'fill') && (out.textFill = richItem.fill); - hasOwn(richItem, 'stroke') && (out.textStroke = richItem.fill); - hasOwn(richItem, 'lineWidth') && (out.textStrokeWidth = richItem.lineWidth); - hasOwn(richItem, 'font') && (out.font = richItem.font); - hasOwn(richItem, 'fontStyle') && (out.fontStyle = richItem.fontStyle); - hasOwn(richItem, 'fontWeight') && (out.fontWeight = richItem.fontWeight); - hasOwn(richItem, 'fontSize') && (out.fontSize = richItem.fontSize); - hasOwn(richItem, 'fontFamily') && (out.fontFamily = richItem.fontFamily); - hasOwn(richItem, 'align') && (out.textAlign = richItem.align); - hasOwn(richItem, 'verticalAlign') && (out.textVerticalAlign = richItem.verticalAlign); - hasOwn(richItem, 'lineHeight') && (out.textLineHeight = richItem.lineHeight); - hasOwn(richItem, 'width') && (out.textWidth = richItem.width); - hasOwn(richItem, 'height') && (out.textHeight = richItem.height); - hasOwn(richItem, 'backgroundColor') && (out.textBackgroundColor = richItem.backgroundColor); - hasOwn(richItem, 'padding') && (out.textPadding = richItem.padding); - hasOwn(richItem, 'borderColor') && (out.textBorderColor = richItem.borderColor); - hasOwn(richItem, 'borderWidth') && (out.textBorderWidth = richItem.borderWidth); - hasOwn(richItem, 'borderRadius') && (out.textBorderRadius = richItem.borderRadius); - hasOwn(richItem, 'shadowColor') && (out.textBoxShadowColor = richItem.shadowColor); - hasOwn(richItem, 'shadowBlur') && (out.textBoxShadowBlur = richItem.shadowBlur); - hasOwn(richItem, 'shadowOffsetX') && (out.textBoxShadowOffsetX = richItem.shadowOffsetX); - hasOwn(richItem, 'shadowOffsetY') && (out.textBoxShadowOffsetY = richItem.shadowOffsetY); - hasOwn(richItem, 'textShadowColor') && (out.textShadowColor = richItem.textShadowColor); - hasOwn(richItem, 'textShadowBlur') && (out.textShadowBlur = richItem.textShadowBlur); - hasOwn(richItem, 'textShadowOffsetX') && (out.textShadowOffsetX = richItem.textShadowOffsetX); - hasOwn(richItem, 'textShadowOffsetY') && (out.textShadowOffsetY = richItem.textShadowOffsetY); - } - - function warnDeprecated(deprecated, insteadApproach) { - if ("development" !== 'production') { - var key = deprecated + '^_^' + insteadApproach; - - if (!deprecatedLogs[key]) { - console.warn("[ECharts] DEPRECATED: \"" + deprecated + "\" has been deprecated. " + insteadApproach); - deprecatedLogs[key] = true; - } - } - } - - var LEGACY_TRANSFORM_PROPS_MAP = { - position: ['x', 'y'], - scale: ['scaleX', 'scaleY'], - origin: ['originX', 'originY'] - }; - var LEGACY_TRANSFORM_PROPS = keys(LEGACY_TRANSFORM_PROPS_MAP); - var TRANSFORM_PROPS_MAP = reduce(TRANSFORMABLE_PROPS, function (obj, key) { - obj[key] = 1; - return obj; - }, {}); - var transformPropNamesStr = TRANSFORMABLE_PROPS.join(', '); // '' means root - - var ELEMENT_ANIMATABLE_PROPS = ['', 'style', 'shape', 'extra']; - var transitionInnerStore = makeInner(); - - function getElementAnimationConfig(animationType, el, elOption, parentModel, dataIndex) { - var animationProp = animationType + "Animation"; - var config = getAnimationConfig(animationType, parentModel, dataIndex) || {}; - var userDuring = transitionInnerStore(el).userDuring; // Only set when duration is > 0 and it's need to be animated. - - if (config.duration > 0) { - // For simplicity, if during not specified, the previous during will not work any more. - config.during = userDuring ? bind(duringCall, { - el: el, - userDuring: userDuring - }) : null; - config.setToFinal = true; - config.scope = animationType; - } - - extend(config, elOption[animationProp]); - return config; - } - - function applyUpdateTransition(el, elOption, animatableModel, opts) { - opts = opts || {}; - var dataIndex = opts.dataIndex, - isInit = opts.isInit, - clearStyle = opts.clearStyle; - var hasAnimation = animatableModel.isAnimationEnabled(); // Save the meta info for further morphing. Like apply on the sub morphing elements. - - var store = transitionInnerStore(el); - var styleOpt = elOption.style; - store.userDuring = elOption.during; - var transFromProps = {}; - var propsToSet = {}; - prepareTransformAllPropsFinal(el, elOption, propsToSet); - prepareShapeOrExtraAllPropsFinal('shape', elOption, propsToSet); - prepareShapeOrExtraAllPropsFinal('extra', elOption, propsToSet); - - if (!isInit && hasAnimation) { - prepareTransformTransitionFrom(el, elOption, transFromProps); - prepareShapeOrExtraTransitionFrom('shape', el, elOption, transFromProps); - prepareShapeOrExtraTransitionFrom('extra', el, elOption, transFromProps); - prepareStyleTransitionFrom(el, elOption, styleOpt, transFromProps); - } - - propsToSet.style = styleOpt; - applyPropsDirectly(el, propsToSet, clearStyle); - applyMiscProps(el, elOption); - - if (hasAnimation) { - if (isInit) { - var enterFromProps_1 = {}; - each(ELEMENT_ANIMATABLE_PROPS, function (propName) { - var prop = propName ? elOption[propName] : elOption; - - if (prop && prop.enterFrom) { - if (propName) { - enterFromProps_1[propName] = enterFromProps_1[propName] || {}; - } - - extend(propName ? enterFromProps_1[propName] : enterFromProps_1, prop.enterFrom); - } - }); - var config = getElementAnimationConfig('enter', el, elOption, animatableModel, dataIndex); - - if (config.duration > 0) { - el.animateFrom(enterFromProps_1, config); - } - } else { - applyPropsTransition(el, elOption, dataIndex || 0, animatableModel, transFromProps); - } - } // Store leave to be used in leave transition. - - - updateLeaveTo(el, elOption); - styleOpt ? el.dirty() : el.markRedraw(); - } - function updateLeaveTo(el, elOption) { - // Try merge to previous set leaveTo - var leaveToProps = transitionInnerStore(el).leaveToProps; - - for (var i = 0; i < ELEMENT_ANIMATABLE_PROPS.length; i++) { - var propName = ELEMENT_ANIMATABLE_PROPS[i]; - var prop = propName ? elOption[propName] : elOption; - - if (prop && prop.leaveTo) { - if (!leaveToProps) { - leaveToProps = transitionInnerStore(el).leaveToProps = {}; - } - - if (propName) { - leaveToProps[propName] = leaveToProps[propName] || {}; - } - - extend(propName ? leaveToProps[propName] : leaveToProps, prop.leaveTo); - } - } - } - function applyLeaveTransition(el, elOption, animatableModel, onRemove) { - if (el) { - var parent_1 = el.parent; - var leaveToProps = transitionInnerStore(el).leaveToProps; - - if (leaveToProps) { - // TODO TODO use leave after leaveAnimation in series is introduced - // TODO Data index? - var config = getElementAnimationConfig('update', el, elOption, animatableModel, 0); - - config.done = function () { - parent_1.remove(el); - onRemove && onRemove(); - }; - - el.animateTo(leaveToProps, config); - } else { - parent_1.remove(el); - onRemove && onRemove(); - } - } - } - function isTransitionAll(transition) { - return transition === 'all'; - } - - function applyPropsDirectly(el, // Can be null/undefined - allPropsFinal, clearStyle) { - var styleOpt = allPropsFinal.style; - - if (!el.isGroup && styleOpt) { - if (clearStyle) { - el.useStyle({}); // When style object changed, how to trade the existing animation? - // It is probably complicated and not needed to cover all the cases. - // But still need consider the case: - // (1) When using init animation on `style.opacity`, and before the animation - // ended users triggers an update by mousewhel. At that time the init - // animation should better be continued rather than terminated. - // So after `useStyle` called, we should change the animation target manually - // to continue the effect of the init animation. - // (2) PENDING: If the previous animation targeted at a `val1`, and currently we need - // to update the value to `val2` and no animation declared, should be terminate - // the previous animation or just modify the target of the animation? - // Therotically That will happen not only on `style` but also on `shape` and - // `transfrom` props. But we haven't handle this case at present yet. - // (3) PENDING: Is it proper to visit `animators` and `targetName`? - - var animators = el.animators; - - for (var i = 0; i < animators.length; i++) { - var animator = animators[i]; // targetName is the "topKey". - - if (animator.targetName === 'style') { - animator.changeTarget(el.style); - } - } - } - - el.setStyle(styleOpt); - } - - if (allPropsFinal) { - // Not set style here. - allPropsFinal.style = null; // Set el to the final state firstly. - - allPropsFinal && el.attr(allPropsFinal); - allPropsFinal.style = styleOpt; - } - } - - function applyPropsTransition(el, elOption, dataIndex, model, // Can be null/undefined - transFromProps) { - if (transFromProps) { - var config = getElementAnimationConfig('update', el, elOption, model, dataIndex); - - if (config.duration > 0) { - el.animateFrom(transFromProps, config); - } - } - } - - function applyMiscProps(el, elOption) { - // Merge by default. - hasOwn(elOption, 'silent') && (el.silent = elOption.silent); - hasOwn(elOption, 'ignore') && (el.ignore = elOption.ignore); - - if (el instanceof Displayable) { - hasOwn(elOption, 'invisible') && (el.invisible = elOption.invisible); - } - - if (el instanceof Path) { - hasOwn(elOption, 'autoBatch') && (el.autoBatch = elOption.autoBatch); - } - } // Use it to avoid it be exposed to user. - - - var tmpDuringScope = {}; - var transitionDuringAPI = { - // Usually other props do not need to be changed in animation during. - setTransform: function (key, val) { - if ("development" !== 'production') { - assert(hasOwn(TRANSFORM_PROPS_MAP, key), 'Only ' + transformPropNamesStr + ' available in `setTransform`.'); - } - - tmpDuringScope.el[key] = val; - return this; - }, - getTransform: function (key) { - if ("development" !== 'production') { - assert(hasOwn(TRANSFORM_PROPS_MAP, key), 'Only ' + transformPropNamesStr + ' available in `getTransform`.'); - } - - return tmpDuringScope.el[key]; - }, - setShape: function (key, val) { - if ("development" !== 'production') { - assertNotReserved(key); - } - - var el = tmpDuringScope.el; - var shape = el.shape || (el.shape = {}); - shape[key] = val; - el.dirtyShape && el.dirtyShape(); - return this; - }, - getShape: function (key) { - if ("development" !== 'production') { - assertNotReserved(key); - } - - var shape = tmpDuringScope.el.shape; - - if (shape) { - return shape[key]; - } - }, - setStyle: function (key, val) { - if ("development" !== 'production') { - assertNotReserved(key); - } - - var el = tmpDuringScope.el; - var style = el.style; - - if (style) { - if ("development" !== 'production') { - if (eqNaN(val)) { - warn('style.' + key + ' must not be assigned with NaN.'); - } - } - - style[key] = val; - el.dirtyStyle && el.dirtyStyle(); - } - - return this; - }, - getStyle: function (key) { - if ("development" !== 'production') { - assertNotReserved(key); - } - - var style = tmpDuringScope.el.style; - - if (style) { - return style[key]; - } - }, - setExtra: function (key, val) { - if ("development" !== 'production') { - assertNotReserved(key); - } - - var extra = tmpDuringScope.el.extra || (tmpDuringScope.el.extra = {}); - extra[key] = val; - return this; - }, - getExtra: function (key) { - if ("development" !== 'production') { - assertNotReserved(key); - } - - var extra = tmpDuringScope.el.extra; - - if (extra) { - return extra[key]; - } - } - }; - - function assertNotReserved(key) { - if ("development" !== 'production') { - if (key === 'transition' || key === 'enterFrom' || key === 'leaveTo') { - throw new Error('key must not be "' + key + '"'); - } - } - } - - function duringCall() { - // Do not provide "percent" until some requirements come. - // Because consider thies case: - // enterFrom: {x: 100, y: 30}, transition: 'x'. - // And enter duration is different from update duration. - // Thus it might be confused about the meaning of "percent" in during callback. - var scope = this; - var el = scope.el; - - if (!el) { - return; - } // If el is remove from zr by reason like legend, during still need to called, - // because el will be added back to zr and the prop value should not be incorrect. - - - var latestUserDuring = transitionInnerStore(el).userDuring; - var scopeUserDuring = scope.userDuring; // Ensured a during is only called once in each animation frame. - // If a during is called multiple times in one frame, maybe some users' calculation logic - // might be wrong (not sure whether this usage exists). - // The case of a during might be called twice can be: by default there is a animator for - // 'x', 'y' when init. Before the init animation finished, call `setOption` to start - // another animators for 'style'/'shape'/'extra'. - - if (latestUserDuring !== scopeUserDuring) { - // release - scope.el = scope.userDuring = null; - return; - } - - tmpDuringScope.el = el; // Give no `this` to user in "during" calling. - - scopeUserDuring(transitionDuringAPI); // FIXME: if in future meet the case that some prop will be both modified in `during` and `state`, - // consider the issue that the prop might be incorrect when return to "normal" state. - } - - function prepareShapeOrExtraTransitionFrom(mainAttr, fromEl, elOption, transFromProps) { - var attrOpt = elOption[mainAttr]; - - if (!attrOpt) { - return; - } - - var elPropsInAttr = fromEl[mainAttr]; - var transFromPropsInAttr; - - if (elPropsInAttr) { - var transition = elOption.transition; - var attrTransition = attrOpt.transition; - - if (attrTransition) { - !transFromPropsInAttr && (transFromPropsInAttr = transFromProps[mainAttr] = {}); - - if (isTransitionAll(attrTransition)) { - extend(transFromPropsInAttr, elPropsInAttr); - } else { - var transitionKeys = normalizeToArray(attrTransition); - - for (var i = 0; i < transitionKeys.length; i++) { - var key = transitionKeys[i]; - var elVal = elPropsInAttr[key]; - transFromPropsInAttr[key] = elVal; - } - } - } else if (isTransitionAll(transition) || indexOf(transition, mainAttr) >= 0) { - !transFromPropsInAttr && (transFromPropsInAttr = transFromProps[mainAttr] = {}); - var elPropsInAttrKeys = keys(elPropsInAttr); - - for (var i = 0; i < elPropsInAttrKeys.length; i++) { - var key = elPropsInAttrKeys[i]; - var elVal = elPropsInAttr[key]; - - if (isNonStyleTransitionEnabled(attrOpt[key], elVal)) { - transFromPropsInAttr[key] = elVal; - } - } - } - } - } - - function prepareShapeOrExtraAllPropsFinal(mainAttr, elOption, allProps) { - var attrOpt = elOption[mainAttr]; - - if (!attrOpt) { - return; - } - - var allPropsInAttr = allProps[mainAttr] = {}; - var keysInAttr = keys(attrOpt); - - for (var i = 0; i < keysInAttr.length; i++) { - var key = keysInAttr[i]; // To avoid share one object with different element, and - // to avoid user modify the object inexpectedly, have to clone. - - allPropsInAttr[key] = cloneValue(attrOpt[key]); - } - } - - function prepareTransformTransitionFrom(el, elOption, transFromProps) { - var transition = elOption.transition; - var transitionKeys = isTransitionAll(transition) ? TRANSFORMABLE_PROPS : normalizeToArray(transition || []); - - for (var i = 0; i < transitionKeys.length; i++) { - var key = transitionKeys[i]; - - if (key === 'style' || key === 'shape' || key === 'extra') { - continue; - } - - var elVal = el[key]; - - if ("development" !== 'production') { - checkTransformPropRefer(key, 'el.transition'); - } // Do not clone, animator will perform that clone. - - - transFromProps[key] = elVal; - } - } - - function prepareTransformAllPropsFinal(el, elOption, allProps) { - for (var i = 0; i < LEGACY_TRANSFORM_PROPS.length; i++) { - var legacyName = LEGACY_TRANSFORM_PROPS[i]; - var xyName = LEGACY_TRANSFORM_PROPS_MAP[legacyName]; - var legacyArr = elOption[legacyName]; - - if (legacyArr) { - allProps[xyName[0]] = legacyArr[0]; - allProps[xyName[1]] = legacyArr[1]; - } - } - - for (var i = 0; i < TRANSFORMABLE_PROPS.length; i++) { - var key = TRANSFORMABLE_PROPS[i]; - - if (elOption[key] != null) { - allProps[key] = elOption[key]; - } - } - } - - function prepareStyleTransitionFrom(fromEl, elOption, styleOpt, transFromProps) { - if (!styleOpt) { - return; - } - - var fromElStyle = fromEl.style; - var transFromStyleProps; - - if (fromElStyle) { - var styleTransition = styleOpt.transition; - var elTransition = elOption.transition; - - if (styleTransition && !isTransitionAll(styleTransition)) { - var transitionKeys = normalizeToArray(styleTransition); - !transFromStyleProps && (transFromStyleProps = transFromProps.style = {}); - - for (var i = 0; i < transitionKeys.length; i++) { - var key = transitionKeys[i]; - var elVal = fromElStyle[key]; // Do not clone, see `checkNonStyleTansitionRefer`. - - transFromStyleProps[key] = elVal; - } - } else if (fromEl.getAnimationStyleProps && (isTransitionAll(elTransition) || isTransitionAll(styleTransition) || indexOf(elTransition, 'style') >= 0)) { - var animationProps = fromEl.getAnimationStyleProps(); - var animationStyleProps = animationProps ? animationProps.style : null; - - if (animationStyleProps) { - !transFromStyleProps && (transFromStyleProps = transFromProps.style = {}); - var styleKeys = keys(styleOpt); - - for (var i = 0; i < styleKeys.length; i++) { - var key = styleKeys[i]; - - if (animationStyleProps[key]) { - var elVal = fromElStyle[key]; - transFromStyleProps[key] = elVal; - } - } - } - } - } - } - - function isNonStyleTransitionEnabled(optVal, elVal) { - // The same as `checkNonStyleTansitionRefer`. - return !isArrayLike(optVal) ? optVal != null && isFinite(optVal) : optVal !== elVal; - } - - var checkTransformPropRefer; - - if ("development" !== 'production') { - checkTransformPropRefer = function (key, usedIn) { - if (!hasOwn(TRANSFORM_PROPS_MAP, key)) { - warn('Prop `' + key + '` is not a permitted in `' + usedIn + '`. ' + 'Only `' + keys(TRANSFORM_PROPS_MAP).join('`, `') + '` are permitted.'); - } - }; - } - - var getStateToRestore = makeInner(); - var KEYFRAME_EXCLUDE_KEYS = ['percent', 'easing', 'shape', 'style', 'extra']; - /** - * Stop previous keyframe animation and restore the attributes. - * Avoid new keyframe animation starts with wrong internal state when the percent: 0 is not set. - */ - - function stopPreviousKeyframeAnimationAndRestore(el) { - // Stop previous keyframe animation. - el.stopAnimation('keyframe'); // Restore - - el.attr(getStateToRestore(el)); - } - function applyKeyframeAnimation(el, animationOpts, animatableModel) { - if (!animatableModel.isAnimationEnabled() || !animationOpts) { - return; - } - - if (isArray(animationOpts)) { - each(animationOpts, function (singleAnimationOpts) { - applyKeyframeAnimation(el, singleAnimationOpts, animatableModel); - }); - return; - } - - var keyframes = animationOpts.keyframes; - var duration = animationOpts.duration; - - if (animatableModel && duration == null) { - // Default to use duration of config. - // NOTE: animation config from payload will be ignored because they are mainly for transitions. - var config = getAnimationConfig('enter', animatableModel, 0); - duration = config && config.duration; - } - - if (!keyframes || !duration) { - return; - } - - var stateToRestore = getStateToRestore(el); - each(ELEMENT_ANIMATABLE_PROPS, function (targetPropName) { - if (targetPropName && !el[targetPropName]) { - return; - } - - var animator; - var endFrameIsSet = false; // Sort keyframes by percent. - - keyframes.sort(function (a, b) { - return a.percent - b.percent; - }); - each(keyframes, function (kf) { - // Stop current animation. - var animators = el.animators; - var kfValues = targetPropName ? kf[targetPropName] : kf; - - if ("development" !== 'production') { - if (kf.percent >= 1) { - endFrameIsSet = true; - } - } - - if (!kfValues) { - return; - } - - var propKeys = keys(kfValues); - - if (!targetPropName) { - // PENDING performance? - propKeys = filter(propKeys, function (key) { - return indexOf(KEYFRAME_EXCLUDE_KEYS, key) < 0; - }); - } - - if (!propKeys.length) { - return; - } - - if (!animator) { - animator = el.animate(targetPropName, animationOpts.loop, true); - animator.scope = 'keyframe'; - } - - for (var i = 0; i < animators.length; i++) { - // Stop all other animation that is not keyframe. - if (animators[i] !== animator && animators[i].targetName === animator.targetName) { - animators[i].stopTracks(propKeys); - } - } - - targetPropName && (stateToRestore[targetPropName] = stateToRestore[targetPropName] || {}); - var savedTarget = targetPropName ? stateToRestore[targetPropName] : stateToRestore; - each(propKeys, function (key) { - // Save original value. - savedTarget[key] = ((targetPropName ? el[targetPropName] : el) || {})[key]; - }); - animator.whenWithKeys(duration * kf.percent, kfValues, propKeys, kf.easing); - }); - - if (!animator) { - return; - } - - if ("development" !== 'production') { - if (!endFrameIsSet) { - warn('End frame with percent: 1 is missing in the keyframeAnimation.', true); - } - } - - animator.delay(animationOpts.delay || 0).duration(duration).start(animationOpts.easing); - }); - } - - var EMPHASIS = 'emphasis'; - var NORMAL = 'normal'; - var BLUR = 'blur'; - var SELECT = 'select'; - var STATES = [NORMAL, EMPHASIS, BLUR, SELECT]; - var PATH_ITEM_STYLE = { - normal: ['itemStyle'], - emphasis: [EMPHASIS, 'itemStyle'], - blur: [BLUR, 'itemStyle'], - select: [SELECT, 'itemStyle'] - }; - var PATH_LABEL = { - normal: ['label'], - emphasis: [EMPHASIS, 'label'], - blur: [BLUR, 'label'], - select: [SELECT, 'label'] - }; - var DEFAULT_TRANSITION = ['x', 'y']; // Use prefix to avoid index to be the same as el.name, - // which will cause weird update animation. - - var GROUP_DIFF_PREFIX = 'e\0\0'; - var attachedTxInfoTmp = { - normal: {}, - emphasis: {}, - blur: {}, - select: {} - }; - /** - * To reduce total package size of each coordinate systems, the modules `prepareCustom` - * of each coordinate systems are not required by each coordinate systems directly, but - * required by the module `custom`. - * - * prepareInfoForCustomSeries {Function}: optional - * @return {Object} {coordSys: {...}, api: { - * coord: function (data, clamp) {}, // return point in global. - * size: function (dataSize, dataItem) {} // return size of each axis in coordSys. - * }} - */ - - var prepareCustoms = { - cartesian2d: cartesianPrepareCustom, - geo: geoPrepareCustom, - single: singlePrepareCustom, - polar: polarPrepareCustom, - calendar: calendarPrepareCustom - }; - - function isPath$1(el) { - return el instanceof Path; - } - - function isDisplayable(el) { - return el instanceof Displayable; - } - - function copyElement(sourceEl, targetEl) { - targetEl.copyTransform(sourceEl); - - if (isDisplayable(targetEl) && isDisplayable(sourceEl)) { - targetEl.setStyle(sourceEl.style); - targetEl.z = sourceEl.z; - targetEl.z2 = sourceEl.z2; - targetEl.zlevel = sourceEl.zlevel; - targetEl.invisible = sourceEl.invisible; - targetEl.ignore = sourceEl.ignore; - - if (isPath$1(targetEl) && isPath$1(sourceEl)) { - targetEl.setShape(sourceEl.shape); - } - } - } - - var CustomChartView = - /** @class */ - function (_super) { - __extends(CustomChartView, _super); - - function CustomChartView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CustomChartView.type; - return _this; - } - - CustomChartView.prototype.render = function (customSeries, ecModel, api, payload) { - // Clear previously rendered progressive elements. - this._progressiveEls = null; - var oldData = this._data; - var data = customSeries.getData(); - var group = this.group; - var renderItem = makeRenderItem(customSeries, data, ecModel, api); - - if (!oldData) { - // Previous render is incremental render or first render. - // Needs remove the incremental rendered elements. - group.removeAll(); - } - - data.diff(oldData).add(function (newIdx) { - createOrUpdateItem(api, null, newIdx, renderItem(newIdx, payload), customSeries, group, data); - }).remove(function (oldIdx) { - var el = oldData.getItemGraphicEl(oldIdx); - el && applyLeaveTransition(el, customInnerStore(el).option, customSeries); - }).update(function (newIdx, oldIdx) { - var oldEl = oldData.getItemGraphicEl(oldIdx); - createOrUpdateItem(api, oldEl, newIdx, renderItem(newIdx, payload), customSeries, group, data); - }).execute(); // Do clipping - - var clipPath = customSeries.get('clip', true) ? createClipPath(customSeries.coordinateSystem, false, customSeries) : null; - - if (clipPath) { - group.setClipPath(clipPath); - } else { - group.removeClipPath(); - } - - this._data = data; - }; - - CustomChartView.prototype.incrementalPrepareRender = function (customSeries, ecModel, api) { - this.group.removeAll(); - this._data = null; - }; - - CustomChartView.prototype.incrementalRender = function (params, customSeries, ecModel, api, payload) { - var data = customSeries.getData(); - var renderItem = makeRenderItem(customSeries, data, ecModel, api); - var progressiveEls = this._progressiveEls = []; - - function setIncrementalAndHoverLayer(el) { - if (!el.isGroup) { - el.incremental = true; - el.ensureState('emphasis').hoverLayer = true; - } - } - - for (var idx = params.start; idx < params.end; idx++) { - var el = createOrUpdateItem(null, null, idx, renderItem(idx, payload), customSeries, this.group, data); - - if (el) { - el.traverse(setIncrementalAndHoverLayer); - progressiveEls.push(el); - } - } - }; - - CustomChartView.prototype.eachRendered = function (cb) { - traverseElements(this._progressiveEls || this.group, cb); - }; - - CustomChartView.prototype.filterForExposedEvent = function (eventType, query, targetEl, packedEvent) { - var elementName = query.element; - - if (elementName == null || targetEl.name === elementName) { - return true; - } // Enable to give a name on a group made by `renderItem`, and listen - // events that are triggered by its descendents. - - - while ((targetEl = targetEl.__hostTarget || targetEl.parent) && targetEl !== this.group) { - if (targetEl.name === elementName) { - return true; - } - } - - return false; - }; - - CustomChartView.type = 'custom'; - return CustomChartView; - }(ChartView); - - function createEl(elOption) { - var graphicType = elOption.type; - var el; // Those graphic elements are not shapes. They should not be - // overwritten by users, so do them first. - - if (graphicType === 'path') { - var shape = elOption.shape; // Using pathRect brings convenience to users sacle svg path. - - var pathRect = shape.width != null && shape.height != null ? { - x: shape.x || 0, - y: shape.y || 0, - width: shape.width, - height: shape.height - } : null; - var pathData = getPathData(shape); // Path is also used for icon, so layout 'center' by default. - - el = makePath(pathData, null, pathRect, shape.layout || 'center'); - customInnerStore(el).customPathData = pathData; - } else if (graphicType === 'image') { - el = new ZRImage({}); - customInnerStore(el).customImagePath = elOption.style.image; - } else if (graphicType === 'text') { - el = new ZRText({}); // customInnerStore(el).customText = (elOption.style as TextStyleProps).text; - } else if (graphicType === 'group') { - el = new Group(); - } else if (graphicType === 'compoundPath') { - throw new Error('"compoundPath" is not supported yet.'); - } else { - var Clz = getShapeClass(graphicType); - - if (!Clz) { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = 'graphic type "' + graphicType + '" can not be found.'; - } - - throwError(errMsg); - } - - el = new Clz(); - } - - customInnerStore(el).customGraphicType = graphicType; - el.name = elOption.name; // Compat ec4: the default z2 lift is 1. If changing the number, - // some cases probably be broken: hierarchy layout along z, like circle packing, - // where emphasis only intending to modify color/border rather than lift z2. - - el.z2EmphasisLift = 1; - el.z2SelectLift = 1; - return el; - } - - function updateElNormal( // Can be null/undefined - api, el, dataIndex, elOption, attachedTxInfo, seriesModel, isInit) { - // Stop and restore before update any other attributes. - stopPreviousKeyframeAnimationAndRestore(el); - var txCfgOpt = attachedTxInfo && attachedTxInfo.normal.cfg; - - if (txCfgOpt) { - // PENDING: whether use user object directly rather than clone? - // TODO:5.0 textConfig transition animation? - el.setTextConfig(txCfgOpt); - } // Default transition ['x', 'y'] - - - if (elOption && elOption.transition == null) { - elOption.transition = DEFAULT_TRANSITION; - } // Do some normalization on style. - - - var styleOpt = elOption && elOption.style; - - if (styleOpt) { - if (el.type === 'text') { - var textOptionStyle = styleOpt; // Compatible with ec4: if `textFill` or `textStroke` exists use them. - - hasOwn(textOptionStyle, 'textFill') && (textOptionStyle.fill = textOptionStyle.textFill); - hasOwn(textOptionStyle, 'textStroke') && (textOptionStyle.stroke = textOptionStyle.textStroke); - } - - var decalPattern = void 0; - var decalObj = isPath$1(el) ? styleOpt.decal : null; - - if (api && decalObj) { - decalObj.dirty = true; - decalPattern = createOrUpdatePatternFromDecal(decalObj, api); - } // Always overwrite in case user specify this prop. - - - styleOpt.__decalPattern = decalPattern; - } - - if (isDisplayable(el)) { - if (styleOpt) { - var decalPattern = styleOpt.__decalPattern; - - if (decalPattern) { - styleOpt.decal = decalPattern; - } - } - } - - applyUpdateTransition(el, elOption, seriesModel, { - dataIndex: dataIndex, - isInit: isInit, - clearStyle: true - }); - applyKeyframeAnimation(el, elOption.keyframeAnimation, seriesModel); - } - - function updateElOnState(state, el, elStateOpt, styleOpt, attachedTxInfo) { - var elDisplayable = el.isGroup ? null : el; - var txCfgOpt = attachedTxInfo && attachedTxInfo[state].cfg; // PENDING:5.0 support customize scale change and transition animation? - - if (elDisplayable) { - // By default support auto lift color when hover whether `emphasis` specified. - var stateObj = elDisplayable.ensureState(state); - - if (styleOpt === false) { - var existingEmphasisState = elDisplayable.getState(state); - - if (existingEmphasisState) { - existingEmphasisState.style = null; - } - } else { - // style is needed to enable default emphasis. - stateObj.style = styleOpt || null; - } // If `elOption.styleEmphasis` or `elOption.emphasis.style` is `false`, - // remove hover style. - // If `elOption.textConfig` or `elOption.emphasis.textConfig` is null/undefined, it does not - // make sense. So for simplicity, we do not ditinguish `hasOwnProperty` and null/undefined. - - - if (txCfgOpt) { - stateObj.textConfig = txCfgOpt; - } - - setDefaultStateProxy(elDisplayable); - } - } - - function updateZ$1(el, elOption, seriesModel) { - // Group not support textContent and not support z yet. - if (el.isGroup) { - return; - } - - var elDisplayable = el; - var currentZ = seriesModel.currentZ; - var currentZLevel = seriesModel.currentZLevel; // Always erase. - - elDisplayable.z = currentZ; - elDisplayable.zlevel = currentZLevel; // z2 must not be null/undefined, otherwise sort error may occur. - - var optZ2 = elOption.z2; - optZ2 != null && (elDisplayable.z2 = optZ2 || 0); - - for (var i = 0; i < STATES.length; i++) { - updateZForEachState(elDisplayable, elOption, STATES[i]); - } - } - - function updateZForEachState(elDisplayable, elOption, state) { - var isNormal = state === NORMAL; - var elStateOpt = isNormal ? elOption : retrieveStateOption(elOption, state); - var optZ2 = elStateOpt ? elStateOpt.z2 : null; - var stateObj; - - if (optZ2 != null) { - // Do not `ensureState` until required. - stateObj = isNormal ? elDisplayable : elDisplayable.ensureState(state); - stateObj.z2 = optZ2 || 0; - } - } - - function makeRenderItem(customSeries, data, ecModel, api) { - var renderItem = customSeries.get('renderItem'); - var coordSys = customSeries.coordinateSystem; - var prepareResult = {}; - - if (coordSys) { - if ("development" !== 'production') { - assert(renderItem, 'series.render is required.'); - assert(coordSys.prepareCustoms || prepareCustoms[coordSys.type], 'This coordSys does not support custom series.'); - } // `coordSys.prepareCustoms` is used for external coord sys like bmap. - - - prepareResult = coordSys.prepareCustoms ? coordSys.prepareCustoms(coordSys) : prepareCustoms[coordSys.type](coordSys); - } - - var userAPI = defaults({ - getWidth: api.getWidth, - getHeight: api.getHeight, - getZr: api.getZr, - getDevicePixelRatio: api.getDevicePixelRatio, - value: value, - style: style, - ordinalRawValue: ordinalRawValue, - styleEmphasis: styleEmphasis, - visual: visual, - barLayout: barLayout, - currentSeriesIndices: currentSeriesIndices, - font: font - }, prepareResult.api || {}); - var userParams = { - // The life cycle of context: current round of rendering. - // The global life cycle is probably not necessary, because - // user can store global status by themselves. - context: {}, - seriesId: customSeries.id, - seriesName: customSeries.name, - seriesIndex: customSeries.seriesIndex, - coordSys: prepareResult.coordSys, - dataInsideLength: data.count(), - encode: wrapEncodeDef(customSeries.getData()) - }; // If someday intending to refactor them to a class, should consider do not - // break change: currently these attribute member are encapsulated in a closure - // so that do not need to force user to call these method with a scope. - // Do not support call `api` asynchronously without dataIndexInside input. - - var currDataIndexInside; - var currItemModel; - var currItemStyleModels = {}; - var currLabelModels = {}; - var seriesItemStyleModels = {}; - var seriesLabelModels = {}; - - for (var i = 0; i < STATES.length; i++) { - var stateName = STATES[i]; - seriesItemStyleModels[stateName] = customSeries.getModel(PATH_ITEM_STYLE[stateName]); - seriesLabelModels[stateName] = customSeries.getModel(PATH_LABEL[stateName]); - } - - function getItemModel(dataIndexInside) { - return dataIndexInside === currDataIndexInside ? currItemModel || (currItemModel = data.getItemModel(dataIndexInside)) : data.getItemModel(dataIndexInside); - } - - function getItemStyleModel(dataIndexInside, state) { - return !data.hasItemOption ? seriesItemStyleModels[state] : dataIndexInside === currDataIndexInside ? currItemStyleModels[state] || (currItemStyleModels[state] = getItemModel(dataIndexInside).getModel(PATH_ITEM_STYLE[state])) : getItemModel(dataIndexInside).getModel(PATH_ITEM_STYLE[state]); - } - - function getLabelModel(dataIndexInside, state) { - return !data.hasItemOption ? seriesLabelModels[state] : dataIndexInside === currDataIndexInside ? currLabelModels[state] || (currLabelModels[state] = getItemModel(dataIndexInside).getModel(PATH_LABEL[state])) : getItemModel(dataIndexInside).getModel(PATH_LABEL[state]); - } - - return function (dataIndexInside, payload) { - currDataIndexInside = dataIndexInside; - currItemModel = null; - currItemStyleModels = {}; - currLabelModels = {}; - return renderItem && renderItem(defaults({ - dataIndexInside: dataIndexInside, - dataIndex: data.getRawIndex(dataIndexInside), - // Can be used for optimization when zoom or roam. - actionType: payload ? payload.type : null - }, userParams), userAPI); - }; - /** - * @public - * @param dim by default 0. - * @param dataIndexInside by default `currDataIndexInside`. - */ - - function value(dim, dataIndexInside) { - dataIndexInside == null && (dataIndexInside = currDataIndexInside); - return data.getStore().get(data.getDimensionIndex(dim || 0), dataIndexInside); - } - /** - * @public - * @param dim by default 0. - * @param dataIndexInside by default `currDataIndexInside`. - */ - - - function ordinalRawValue(dim, dataIndexInside) { - dataIndexInside == null && (dataIndexInside = currDataIndexInside); - dim = dim || 0; - var dimInfo = data.getDimensionInfo(dim); - - if (!dimInfo) { - var dimIndex = data.getDimensionIndex(dim); - return dimIndex >= 0 ? data.getStore().get(dimIndex, dataIndexInside) : undefined; - } - - var val = data.get(dimInfo.name, dataIndexInside); - var ordinalMeta = dimInfo && dimInfo.ordinalMeta; - return ordinalMeta ? ordinalMeta.categories[val] : val; - } - /** - * @deprecated The original intention of `api.style` is enable to set itemStyle - * like other series. But it is not necessary and not easy to give a strict definition - * of what it returns. And since echarts5 it needs to be make compat work. So - * deprecates it since echarts5. - * - * By default, `visual` is applied to style (to support visualMap). - * `visual.color` is applied at `fill`. If user want apply visual.color on `stroke`, - * it can be implemented as: - * `api.style({stroke: api.visual('color'), fill: null})`; - * - * [Compat]: since ec5, RectText has been separated from its hosts el. - * so `api.style()` will only return the style from `itemStyle` but not handle `label` - * any more. But `series.label` config is never published in doc. - * We still compat it in `api.style()`. But not encourage to use it and will still not - * to pulish it to doc. - * @public - * @param dataIndexInside by default `currDataIndexInside`. - */ - - - function style(userProps, dataIndexInside) { - if ("development" !== 'production') { - warnDeprecated('api.style', 'Please write literal style directly instead.'); - } - - dataIndexInside == null && (dataIndexInside = currDataIndexInside); - var style = data.getItemVisual(dataIndexInside, 'style'); - var visualColor = style && style.fill; - var opacity = style && style.opacity; - var itemStyle = getItemStyleModel(dataIndexInside, NORMAL).getItemStyle(); - visualColor != null && (itemStyle.fill = visualColor); - opacity != null && (itemStyle.opacity = opacity); - var opt = { - inheritColor: isString(visualColor) ? visualColor : '#000' - }; - var labelModel = getLabelModel(dataIndexInside, NORMAL); // Now that the feature of "auto adjust text fill/stroke" has been migrated to zrender - // since ec5, we should set `isAttached` as `false` here and make compat in - // `convertToEC4StyleForCustomSerise`. - - var textStyle = createTextStyle(labelModel, null, opt, false, true); - textStyle.text = labelModel.getShallow('show') ? retrieve2(customSeries.getFormattedLabel(dataIndexInside, NORMAL), getDefaultLabel(data, dataIndexInside)) : null; - var textConfig = createTextConfig(labelModel, opt, false); - preFetchFromExtra(userProps, itemStyle); - itemStyle = convertToEC4StyleForCustomSerise(itemStyle, textStyle, textConfig); - userProps && applyUserPropsAfter(itemStyle, userProps); - itemStyle.legacy = true; - return itemStyle; - } - /** - * @deprecated The reason see `api.style()` - * @public - * @param dataIndexInside by default `currDataIndexInside`. - */ - - - function styleEmphasis(userProps, dataIndexInside) { - if ("development" !== 'production') { - warnDeprecated('api.styleEmphasis', 'Please write literal style directly instead.'); - } - - dataIndexInside == null && (dataIndexInside = currDataIndexInside); - var itemStyle = getItemStyleModel(dataIndexInside, EMPHASIS).getItemStyle(); - var labelModel = getLabelModel(dataIndexInside, EMPHASIS); - var textStyle = createTextStyle(labelModel, null, null, true, true); - textStyle.text = labelModel.getShallow('show') ? retrieve3(customSeries.getFormattedLabel(dataIndexInside, EMPHASIS), customSeries.getFormattedLabel(dataIndexInside, NORMAL), getDefaultLabel(data, dataIndexInside)) : null; - var textConfig = createTextConfig(labelModel, null, true); - preFetchFromExtra(userProps, itemStyle); - itemStyle = convertToEC4StyleForCustomSerise(itemStyle, textStyle, textConfig); - userProps && applyUserPropsAfter(itemStyle, userProps); - itemStyle.legacy = true; - return itemStyle; - } - - function applyUserPropsAfter(itemStyle, extra) { - for (var key in extra) { - if (hasOwn(extra, key)) { - itemStyle[key] = extra[key]; - } - } - } - - function preFetchFromExtra(extra, itemStyle) { - // A trick to retrieve those props firstly, which are used to - // apply auto inside fill/stroke in `convertToEC4StyleForCustomSerise`. - // (It's not reasonable but only for a degree of compat) - if (extra) { - extra.textFill && (itemStyle.textFill = extra.textFill); - extra.textPosition && (itemStyle.textPosition = extra.textPosition); - } - } - /** - * @public - * @param dataIndexInside by default `currDataIndexInside`. - */ - - - function visual(visualType, dataIndexInside) { - dataIndexInside == null && (dataIndexInside = currDataIndexInside); - - if (hasOwn(STYLE_VISUAL_TYPE, visualType)) { - var style_1 = data.getItemVisual(dataIndexInside, 'style'); - return style_1 ? style_1[STYLE_VISUAL_TYPE[visualType]] : null; - } // Only support these visuals. Other visual might be inner tricky - // for performance (like `style`), do not expose to users. - - - if (hasOwn(NON_STYLE_VISUAL_PROPS, visualType)) { - return data.getItemVisual(dataIndexInside, visualType); - } - } - /** - * @public - * @return If not support, return undefined. - */ - - - function barLayout(opt) { - if (coordSys.type === 'cartesian2d') { - var baseAxis = coordSys.getBaseAxis(); - return getLayoutOnAxis(defaults({ - axis: baseAxis - }, opt)); - } - } - /** - * @public - */ - - - function currentSeriesIndices() { - return ecModel.getCurrentSeriesIndices(); - } - /** - * @public - * @return font string - */ - - - function font(opt) { - return getFont(opt, ecModel); - } - } - - function wrapEncodeDef(data) { - var encodeDef = {}; - each(data.dimensions, function (dimName) { - var dimInfo = data.getDimensionInfo(dimName); - - if (!dimInfo.isExtraCoord) { - var coordDim = dimInfo.coordDim; - var dataDims = encodeDef[coordDim] = encodeDef[coordDim] || []; - dataDims[dimInfo.coordDimIndex] = data.getDimensionIndex(dimName); - } - }); - return encodeDef; - } - - function createOrUpdateItem(api, existsEl, dataIndex, elOption, seriesModel, group, data) { - // [Rule] - // If `renderItem` returns `null`/`undefined`/`false`, remove the previous el if existing. - // (It seems that violate the "merge" principle, but most of users probably intuitively - // regard "return;" as "show nothing element whatever", so make a exception to meet the - // most cases.) - // The rule or "merge" see [STRATEGY_MERGE]. - // If `elOption` is `null`/`undefined`/`false` (when `renderItem` returns nothing). - if (!elOption) { - group.remove(existsEl); - return; - } - - var el = doCreateOrUpdateEl(api, existsEl, dataIndex, elOption, seriesModel, group); - el && data.setItemGraphicEl(dataIndex, el); - el && toggleHoverEmphasis(el, elOption.focus, elOption.blurScope, elOption.emphasisDisabled); - return el; - } - - function doCreateOrUpdateEl(api, existsEl, dataIndex, elOption, seriesModel, group) { - if ("development" !== 'production') { - assert(elOption, 'should not have an null/undefined element setting'); - } - - var toBeReplacedIdx = -1; - var oldEl = existsEl; - - if (existsEl && doesElNeedRecreate(existsEl, elOption, seriesModel) // || ( - // // PENDING: even in one-to-one mapping case, if el is marked as morph, - // // do not sure whether the el will be mapped to another el with different - // // hierarchy in Group tree. So always recreate el rather than reuse the el. - // morphHelper && morphHelper.isOneToOneFrom(el) - // ) - ) { - // Should keep at the original index, otherwise "merge by index" will be incorrect. - toBeReplacedIdx = indexOf(group.childrenRef(), existsEl); - existsEl = null; - } - - var isInit = !existsEl; - var el = existsEl; - - if (!el) { - el = createEl(elOption); - - if (oldEl) { - copyElement(oldEl, el); - } - } else { - // FIMXE:NEXT unified clearState? - // If in some case the performance issue arised, consider - // do not clearState but update cached normal state directly. - el.clearStates(); - } // Need to set morph: false explictly to disable automatically morphing. - - - if (elOption.morph === false) { - el.disableMorphing = true; - } else if (el.disableMorphing) { - el.disableMorphing = false; - } - - attachedTxInfoTmp.normal.cfg = attachedTxInfoTmp.normal.conOpt = attachedTxInfoTmp.emphasis.cfg = attachedTxInfoTmp.emphasis.conOpt = attachedTxInfoTmp.blur.cfg = attachedTxInfoTmp.blur.conOpt = attachedTxInfoTmp.select.cfg = attachedTxInfoTmp.select.conOpt = null; - attachedTxInfoTmp.isLegacy = false; - doCreateOrUpdateAttachedTx(el, dataIndex, elOption, seriesModel, isInit, attachedTxInfoTmp); - doCreateOrUpdateClipPath(el, dataIndex, elOption, seriesModel, isInit); - updateElNormal(api, el, dataIndex, elOption, attachedTxInfoTmp, seriesModel, isInit); // `elOption.info` enables user to mount some info on - // elements and use them in event handlers. - // Update them only when user specified, otherwise, remain. - - hasOwn(elOption, 'info') && (customInnerStore(el).info = elOption.info); - - for (var i = 0; i < STATES.length; i++) { - var stateName = STATES[i]; - - if (stateName !== NORMAL) { - var otherStateOpt = retrieveStateOption(elOption, stateName); - var otherStyleOpt = retrieveStyleOptionOnState(elOption, otherStateOpt, stateName); - updateElOnState(stateName, el, otherStateOpt, otherStyleOpt, attachedTxInfoTmp); - } - } - - updateZ$1(el, elOption, seriesModel); - - if (elOption.type === 'group') { - mergeChildren(api, el, dataIndex, elOption, seriesModel); - } - - if (toBeReplacedIdx >= 0) { - group.replaceAt(el, toBeReplacedIdx); - } else { - group.add(el); - } - - return el; - } // `el` must not be null/undefined. - - - function doesElNeedRecreate(el, elOption, seriesModel) { - var elInner = customInnerStore(el); - var elOptionType = elOption.type; - var elOptionShape = elOption.shape; - var elOptionStyle = elOption.style; - return (// Always create new if universal transition is enabled. - // Because we do transition after render. It needs to know what old element is. Replacement will loose it. - seriesModel.isUniversalTransitionEnabled() // If `elOptionType` is `null`, follow the merge principle. - || elOptionType != null && elOptionType !== elInner.customGraphicType || elOptionType === 'path' && hasOwnPathData(elOptionShape) && getPathData(elOptionShape) !== elInner.customPathData || elOptionType === 'image' && hasOwn(elOptionStyle, 'image') && elOptionStyle.image !== elInner.customImagePath // // FIXME test and remove this restriction? - // || (elOptionType === 'text' - // && hasOwn(elOptionStyle, 'text') - // && (elOptionStyle as TextStyleProps).text !== elInner.customText - // ) - - ); - } - - function doCreateOrUpdateClipPath(el, dataIndex, elOption, seriesModel, isInit) { - // Based on the "merge" principle, if no clipPath provided, - // do nothing. The exists clip will be totally removed only if - // `el.clipPath` is `false`. Otherwise it will be merged/replaced. - var clipPathOpt = elOption.clipPath; - - if (clipPathOpt === false) { - if (el && el.getClipPath()) { - el.removeClipPath(); - } - } else if (clipPathOpt) { - var clipPath = el.getClipPath(); - - if (clipPath && doesElNeedRecreate(clipPath, clipPathOpt, seriesModel)) { - clipPath = null; - } - - if (!clipPath) { - clipPath = createEl(clipPathOpt); - - if ("development" !== 'production') { - assert(isPath$1(clipPath), 'Only any type of `path` can be used in `clipPath`, rather than ' + clipPath.type + '.'); - } - - el.setClipPath(clipPath); - } - - updateElNormal(null, clipPath, dataIndex, clipPathOpt, null, seriesModel, isInit); - } // If not define `clipPath` in option, do nothing unnecessary. - - } - - function doCreateOrUpdateAttachedTx(el, dataIndex, elOption, seriesModel, isInit, attachedTxInfo) { - // Group does not support textContent temporarily until necessary. - if (el.isGroup) { - return; - } // Normal must be called before emphasis, for `isLegacy` detection. - - - processTxInfo(elOption, null, attachedTxInfo); - processTxInfo(elOption, EMPHASIS, attachedTxInfo); // If `elOption.textConfig` or `elOption.textContent` is null/undefined, it does not make sense. - // So for simplicity, if "elOption hasOwnProperty of them but be null/undefined", we do not - // trade them as set to null to el. - // Especially: - // `elOption.textContent: false` means remove textContent. - // `elOption.textContent.emphasis.style: false` means remove the style from emphasis state. - - var txConOptNormal = attachedTxInfo.normal.conOpt; - var txConOptEmphasis = attachedTxInfo.emphasis.conOpt; - var txConOptBlur = attachedTxInfo.blur.conOpt; - var txConOptSelect = attachedTxInfo.select.conOpt; - - if (txConOptNormal != null || txConOptEmphasis != null || txConOptSelect != null || txConOptBlur != null) { - var textContent = el.getTextContent(); - - if (txConOptNormal === false) { - textContent && el.removeTextContent(); - } else { - txConOptNormal = attachedTxInfo.normal.conOpt = txConOptNormal || { - type: 'text' - }; - - if (!textContent) { - textContent = createEl(txConOptNormal); - el.setTextContent(textContent); - } else { - // If in some case the performance issue arised, consider - // do not clearState but update cached normal state directly. - textContent.clearStates(); - } - - updateElNormal(null, textContent, dataIndex, txConOptNormal, null, seriesModel, isInit); - var txConStlOptNormal = txConOptNormal && txConOptNormal.style; - - for (var i = 0; i < STATES.length; i++) { - var stateName = STATES[i]; - - if (stateName !== NORMAL) { - var txConOptOtherState = attachedTxInfo[stateName].conOpt; - updateElOnState(stateName, textContent, txConOptOtherState, retrieveStyleOptionOnState(txConOptNormal, txConOptOtherState, stateName), null); - } - } - - txConStlOptNormal ? textContent.dirty() : textContent.markRedraw(); - } - } - } - - function processTxInfo(elOption, state, attachedTxInfo) { - var stateOpt = !state ? elOption : retrieveStateOption(elOption, state); - var styleOpt = !state ? elOption.style : retrieveStyleOptionOnState(elOption, stateOpt, EMPHASIS); - var elType = elOption.type; - var txCfg = stateOpt ? stateOpt.textConfig : null; - var txConOptNormal = elOption.textContent; - var txConOpt = !txConOptNormal ? null : !state ? txConOptNormal : retrieveStateOption(txConOptNormal, state); - - if (styleOpt && ( // Because emphasis style has little info to detect legacy, - // if normal is legacy, emphasis is trade as legacy. - attachedTxInfo.isLegacy || isEC4CompatibleStyle(styleOpt, elType, !!txCfg, !!txConOpt))) { - attachedTxInfo.isLegacy = true; - var convertResult = convertFromEC4CompatibleStyle(styleOpt, elType, !state); // Explicitly specified `textConfig` and `textContent` has higher priority than - // the ones generated by legacy style. Otherwise if users use them and `api.style` - // at the same time, they not both work and hardly to known why. - - if (!txCfg && convertResult.textConfig) { - txCfg = convertResult.textConfig; - } - - if (!txConOpt && convertResult.textContent) { - txConOpt = convertResult.textContent; - } - } - - if (!state && txConOpt) { - var txConOptNormal_1 = txConOpt; // `textContent: {type: 'text'}`, the "type" is easy to be missing. So we tolerate it. - - !txConOptNormal_1.type && (txConOptNormal_1.type = 'text'); - - if ("development" !== 'production') { - // Do not tolerate incorrcet type for forward compat. - assert(txConOptNormal_1.type === 'text', 'textContent.type must be "text"'); - } - } - - var info = !state ? attachedTxInfo.normal : attachedTxInfo[state]; - info.cfg = txCfg; - info.conOpt = txConOpt; - } - - function retrieveStateOption(elOption, state) { - return !state ? elOption : elOption ? elOption[state] : null; - } - - function retrieveStyleOptionOnState(stateOptionNormal, stateOption, state) { - var style = stateOption && stateOption.style; - - if (style == null && state === EMPHASIS && stateOptionNormal) { - style = stateOptionNormal.styleEmphasis; - } - - return style; - } // Usage: - // (1) By default, `elOption.$mergeChildren` is `'byIndex'`, which indicates - // that the existing children will not be removed, and enables the feature - // that update some of the props of some of the children simply by construct - // the returned children of `renderItem` like: - // `var children = group.children = []; children[3] = {opacity: 0.5};` - // (2) If `elOption.$mergeChildren` is `'byName'`, add/update/remove children - // by child.name. But that might be lower performance. - // (3) If `elOption.$mergeChildren` is `false`, the existing children will be - // replaced totally. - // (4) If `!elOption.children`, following the "merge" principle, nothing will - // happen. - // (5) If `elOption.$mergeChildren` is not `false` neither `'byName'` and the - // `el` is a group, and if any of the new child is null, it means to remove - // the element at the same index, if exists. On the other hand, if the new - // child is and empty object `{}`, it means to keep the element not changed. - // - // For implementation simpleness, do not provide a direct way to remove single - // child (otherwise the total indices of the children array have to be modified). - // User can remove a single child by setting its `ignore` to `true`. - - - function mergeChildren(api, el, dataIndex, elOption, seriesModel) { - var newChildren = elOption.children; - var newLen = newChildren ? newChildren.length : 0; - var mergeChildren = elOption.$mergeChildren; // `diffChildrenByName` has been deprecated. - - var byName = mergeChildren === 'byName' || elOption.diffChildrenByName; - var notMerge = mergeChildren === false; // For better performance on roam update, only enter if necessary. - - if (!newLen && !byName && !notMerge) { - return; - } - - if (byName) { - diffGroupChildren({ - api: api, - oldChildren: el.children() || [], - newChildren: newChildren || [], - dataIndex: dataIndex, - seriesModel: seriesModel, - group: el - }); - return; - } - - notMerge && el.removeAll(); // Mapping children of a group simply by index, which - // might be better performance. - - var index = 0; - - for (; index < newLen; index++) { - var newChild = newChildren[index]; - var oldChild = el.childAt(index); - - if (newChild) { - if (newChild.ignore == null) { - // The old child is set to be ignored if null (see comments - // below). So we need to set ignore to be false back. - newChild.ignore = false; - } - - doCreateOrUpdateEl(api, oldChild, dataIndex, newChild, seriesModel, el); - } else { - if ("development" !== 'production') { - assert(oldChild, 'renderItem should not return a group containing elements' + ' as null/undefined/{} if they do not exist before.'); - } // If the new element option is null, it means to remove the old - // element. But we cannot really remove the element from the group - // directly, because the element order may not be stable when this - // element is added back. So we set the element to be ignored. - - - oldChild.ignore = true; - } - } - - for (var i = el.childCount() - 1; i >= index; i--) { - var child = el.childAt(i); - removeChildFromGroup(el, child, seriesModel); - } - } - - function removeChildFromGroup(group, child, seriesModel) { - // Do not support leave elements that are not mentioned in the latest - // `renderItem` return. Otherwise users may not have a clear and simple - // concept that how to control all of the elements. - child && applyLeaveTransition(child, customInnerStore(group).option, seriesModel); - } - - function diffGroupChildren(context) { - new DataDiffer(context.oldChildren, context.newChildren, getKey, getKey, context).add(processAddUpdate).update(processAddUpdate).remove(processRemove).execute(); - } - - function getKey(item, idx) { - var name = item && item.name; - return name != null ? name : GROUP_DIFF_PREFIX + idx; - } - - function processAddUpdate(newIndex, oldIndex) { - var context = this.context; - var childOption = newIndex != null ? context.newChildren[newIndex] : null; - var child = oldIndex != null ? context.oldChildren[oldIndex] : null; - doCreateOrUpdateEl(context.api, child, context.dataIndex, childOption, context.seriesModel, context.group); - } - - function processRemove(oldIndex) { - var context = this.context; - var child = context.oldChildren[oldIndex]; - child && applyLeaveTransition(child, customInnerStore(child).option, context.seriesModel); - } - /** - * @return SVG Path data. - */ - - - function getPathData(shape) { - // "d" follows the SVG convention. - return shape && (shape.pathData || shape.d); - } - - function hasOwnPathData(shape) { - return shape && (hasOwn(shape, 'pathData') || hasOwn(shape, 'd')); - } - - function install$r(registers) { - registers.registerChartView(CustomChartView); - registers.registerSeriesModel(CustomSeriesModel); - } - - var inner$a = makeInner(); - var clone$3 = clone; - var bind$1 = bind; - /** - * Base axis pointer class in 2D. - */ - - var BaseAxisPointer = - /** @class */ - function () { - function BaseAxisPointer() { - this._dragging = false; - /** - * In px, arbitrary value. Do not set too small, - * no animation is ok for most cases. - */ - - this.animationThreshold = 15; - } - /** - * @implement - */ - - - BaseAxisPointer.prototype.render = function (axisModel, axisPointerModel, api, forceRender) { - var value = axisPointerModel.get('value'); - var status = axisPointerModel.get('status'); // Bind them to `this`, not in closure, otherwise they will not - // be replaced when user calling setOption in not merge mode. - - this._axisModel = axisModel; - this._axisPointerModel = axisPointerModel; - this._api = api; // Optimize: `render` will be called repeatedly during mouse move. - // So it is power consuming if performing `render` each time, - // especially on mobile device. - - if (!forceRender && this._lastValue === value && this._lastStatus === status) { - return; - } - - this._lastValue = value; - this._lastStatus = status; - var group = this._group; - var handle = this._handle; - - if (!status || status === 'hide') { - // Do not clear here, for animation better. - group && group.hide(); - handle && handle.hide(); - return; - } - - group && group.show(); - handle && handle.show(); // Otherwise status is 'show' - - var elOption = {}; - this.makeElOption(elOption, value, axisModel, axisPointerModel, api); // Enable change axis pointer type. - - var graphicKey = elOption.graphicKey; - - if (graphicKey !== this._lastGraphicKey) { - this.clear(api); - } - - this._lastGraphicKey = graphicKey; - var moveAnimation = this._moveAnimation = this.determineAnimation(axisModel, axisPointerModel); - - if (!group) { - group = this._group = new Group(); - this.createPointerEl(group, elOption, axisModel, axisPointerModel); - this.createLabelEl(group, elOption, axisModel, axisPointerModel); - api.getZr().add(group); - } else { - var doUpdateProps = curry(updateProps$1, axisPointerModel, moveAnimation); - this.updatePointerEl(group, elOption, doUpdateProps); - this.updateLabelEl(group, elOption, doUpdateProps, axisPointerModel); - } - - updateMandatoryProps(group, axisPointerModel, true); - - this._renderHandle(value); - }; - /** - * @implement - */ - - - BaseAxisPointer.prototype.remove = function (api) { - this.clear(api); - }; - /** - * @implement - */ - - - BaseAxisPointer.prototype.dispose = function (api) { - this.clear(api); - }; - /** - * @protected - */ - - - BaseAxisPointer.prototype.determineAnimation = function (axisModel, axisPointerModel) { - var animation = axisPointerModel.get('animation'); - var axis = axisModel.axis; - var isCategoryAxis = axis.type === 'category'; - var useSnap = axisPointerModel.get('snap'); // Value axis without snap always do not snap. - - if (!useSnap && !isCategoryAxis) { - return false; - } - - if (animation === 'auto' || animation == null) { - var animationThreshold = this.animationThreshold; - - if (isCategoryAxis && axis.getBandWidth() > animationThreshold) { - return true; - } // It is important to auto animation when snap used. Consider if there is - // a dataZoom, animation will be disabled when too many points exist, while - // it will be enabled for better visual effect when little points exist. - - - if (useSnap) { - var seriesDataCount = getAxisInfo(axisModel).seriesDataCount; - var axisExtent = axis.getExtent(); // Approximate band width - - return Math.abs(axisExtent[0] - axisExtent[1]) / seriesDataCount > animationThreshold; - } - - return false; - } - - return animation === true; - }; - /** - * add {pointer, label, graphicKey} to elOption - * @protected - */ - - - BaseAxisPointer.prototype.makeElOption = function (elOption, value, axisModel, axisPointerModel, api) {// Should be implemenented by sub-class. - }; - /** - * @protected - */ - - - BaseAxisPointer.prototype.createPointerEl = function (group, elOption, axisModel, axisPointerModel) { - var pointerOption = elOption.pointer; - - if (pointerOption) { - var pointerEl = inner$a(group).pointerEl = new graphic[pointerOption.type](clone$3(elOption.pointer)); - group.add(pointerEl); - } - }; - /** - * @protected - */ - - - BaseAxisPointer.prototype.createLabelEl = function (group, elOption, axisModel, axisPointerModel) { - if (elOption.label) { - var labelEl = inner$a(group).labelEl = new ZRText(clone$3(elOption.label)); - group.add(labelEl); - updateLabelShowHide(labelEl, axisPointerModel); - } - }; - /** - * @protected - */ - - - BaseAxisPointer.prototype.updatePointerEl = function (group, elOption, updateProps) { - var pointerEl = inner$a(group).pointerEl; - - if (pointerEl && elOption.pointer) { - pointerEl.setStyle(elOption.pointer.style); - updateProps(pointerEl, { - shape: elOption.pointer.shape - }); - } - }; - /** - * @protected - */ - - - BaseAxisPointer.prototype.updateLabelEl = function (group, elOption, updateProps, axisPointerModel) { - var labelEl = inner$a(group).labelEl; - - if (labelEl) { - labelEl.setStyle(elOption.label.style); - updateProps(labelEl, { - // Consider text length change in vertical axis, animation should - // be used on shape, otherwise the effect will be weird. - // TODOTODO - // shape: elOption.label.shape, - x: elOption.label.x, - y: elOption.label.y - }); - updateLabelShowHide(labelEl, axisPointerModel); - } - }; - /** - * @private - */ - - - BaseAxisPointer.prototype._renderHandle = function (value) { - if (this._dragging || !this.updateHandleTransform) { - return; - } - - var axisPointerModel = this._axisPointerModel; - - var zr = this._api.getZr(); - - var handle = this._handle; - var handleModel = axisPointerModel.getModel('handle'); - var status = axisPointerModel.get('status'); - - if (!handleModel.get('show') || !status || status === 'hide') { - handle && zr.remove(handle); - this._handle = null; - return; - } - - var isInit; - - if (!this._handle) { - isInit = true; - handle = this._handle = createIcon(handleModel.get('icon'), { - cursor: 'move', - draggable: true, - onmousemove: function (e) { - // For mobile device, prevent screen slider on the button. - stop(e.event); - }, - onmousedown: bind$1(this._onHandleDragMove, this, 0, 0), - drift: bind$1(this._onHandleDragMove, this), - ondragend: bind$1(this._onHandleDragEnd, this) - }); - zr.add(handle); - } - - updateMandatoryProps(handle, axisPointerModel, false); // update style - - handle.setStyle(handleModel.getItemStyle(null, ['color', 'borderColor', 'borderWidth', 'opacity', 'shadowColor', 'shadowBlur', 'shadowOffsetX', 'shadowOffsetY'])); // update position - - var handleSize = handleModel.get('size'); - - if (!isArray(handleSize)) { - handleSize = [handleSize, handleSize]; - } - - handle.scaleX = handleSize[0] / 2; - handle.scaleY = handleSize[1] / 2; - createOrUpdate(this, '_doDispatchAxisPointer', handleModel.get('throttle') || 0, 'fixRate'); - - this._moveHandleToValue(value, isInit); - }; - - BaseAxisPointer.prototype._moveHandleToValue = function (value, isInit) { - updateProps$1(this._axisPointerModel, !isInit && this._moveAnimation, this._handle, getHandleTransProps(this.getHandleTransform(value, this._axisModel, this._axisPointerModel))); - }; - - BaseAxisPointer.prototype._onHandleDragMove = function (dx, dy) { - var handle = this._handle; - - if (!handle) { - return; - } - - this._dragging = true; // Persistent for throttle. - - var trans = this.updateHandleTransform(getHandleTransProps(handle), [dx, dy], this._axisModel, this._axisPointerModel); - this._payloadInfo = trans; - handle.stopAnimation(); - handle.attr(getHandleTransProps(trans)); - inner$a(handle).lastProp = null; - - this._doDispatchAxisPointer(); - }; - /** - * Throttled method. - */ - - - BaseAxisPointer.prototype._doDispatchAxisPointer = function () { - var handle = this._handle; - - if (!handle) { - return; - } - - var payloadInfo = this._payloadInfo; - var axisModel = this._axisModel; - - this._api.dispatchAction({ - type: 'updateAxisPointer', - x: payloadInfo.cursorPoint[0], - y: payloadInfo.cursorPoint[1], - tooltipOption: payloadInfo.tooltipOption, - axesInfo: [{ - axisDim: axisModel.axis.dim, - axisIndex: axisModel.componentIndex - }] - }); - }; - - BaseAxisPointer.prototype._onHandleDragEnd = function () { - this._dragging = false; - var handle = this._handle; - - if (!handle) { - return; - } - - var value = this._axisPointerModel.get('value'); // Consider snap or categroy axis, handle may be not consistent with - // axisPointer. So move handle to align the exact value position when - // drag ended. - - - this._moveHandleToValue(value); // For the effect: tooltip will be shown when finger holding on handle - // button, and will be hidden after finger left handle button. - - - this._api.dispatchAction({ - type: 'hideTip' - }); - }; - /** - * @private - */ - - - BaseAxisPointer.prototype.clear = function (api) { - this._lastValue = null; - this._lastStatus = null; - var zr = api.getZr(); - var group = this._group; - var handle = this._handle; - - if (zr && group) { - this._lastGraphicKey = null; - group && zr.remove(group); - handle && zr.remove(handle); - this._group = null; - this._handle = null; - this._payloadInfo = null; - } - - clear(this, '_doDispatchAxisPointer'); - }; - /** - * @protected - */ - - - BaseAxisPointer.prototype.doClear = function () {// Implemented by sub-class if necessary. - }; - - BaseAxisPointer.prototype.buildLabel = function (xy, wh, xDimIndex) { - xDimIndex = xDimIndex || 0; - return { - x: xy[xDimIndex], - y: xy[1 - xDimIndex], - width: wh[xDimIndex], - height: wh[1 - xDimIndex] - }; - }; - - return BaseAxisPointer; - }(); - - function updateProps$1(animationModel, moveAnimation, el, props) { - // Animation optimize. - if (!propsEqual(inner$a(el).lastProp, props)) { - inner$a(el).lastProp = props; - moveAnimation ? updateProps(el, props, animationModel) : (el.stopAnimation(), el.attr(props)); - } - } - - function propsEqual(lastProps, newProps) { - if (isObject(lastProps) && isObject(newProps)) { - var equals_1 = true; - each(newProps, function (item, key) { - equals_1 = equals_1 && propsEqual(lastProps[key], item); - }); - return !!equals_1; - } else { - return lastProps === newProps; - } - } - - function updateLabelShowHide(labelEl, axisPointerModel) { - labelEl[axisPointerModel.get(['label', 'show']) ? 'show' : 'hide'](); - } - - function getHandleTransProps(trans) { - return { - x: trans.x || 0, - y: trans.y || 0, - rotation: trans.rotation || 0 - }; - } - - function updateMandatoryProps(group, axisPointerModel, silent) { - var z = axisPointerModel.get('z'); - var zlevel = axisPointerModel.get('zlevel'); - group && group.traverse(function (el) { - if (el.type !== 'group') { - z != null && (el.z = z); - zlevel != null && (el.zlevel = zlevel); - el.silent = silent; - } - }); - } - - function buildElStyle(axisPointerModel) { - var axisPointerType = axisPointerModel.get('type'); - var styleModel = axisPointerModel.getModel(axisPointerType + 'Style'); - var style; - - if (axisPointerType === 'line') { - style = styleModel.getLineStyle(); - style.fill = null; - } else if (axisPointerType === 'shadow') { - style = styleModel.getAreaStyle(); - style.stroke = null; - } - - return style; - } - /** - * @param {Function} labelPos {align, verticalAlign, position} - */ - - function buildLabelElOption(elOption, axisModel, axisPointerModel, api, labelPos) { - var value = axisPointerModel.get('value'); - var text = getValueLabel(value, axisModel.axis, axisModel.ecModel, axisPointerModel.get('seriesDataIndices'), { - precision: axisPointerModel.get(['label', 'precision']), - formatter: axisPointerModel.get(['label', 'formatter']) - }); - var labelModel = axisPointerModel.getModel('label'); - var paddings = normalizeCssArray$1(labelModel.get('padding') || 0); - var font = labelModel.getFont(); - var textRect = getBoundingRect(text, font); - var position = labelPos.position; - var width = textRect.width + paddings[1] + paddings[3]; - var height = textRect.height + paddings[0] + paddings[2]; // Adjust by align. - - var align = labelPos.align; - align === 'right' && (position[0] -= width); - align === 'center' && (position[0] -= width / 2); - var verticalAlign = labelPos.verticalAlign; - verticalAlign === 'bottom' && (position[1] -= height); - verticalAlign === 'middle' && (position[1] -= height / 2); // Not overflow ec container - - confineInContainer(position, width, height, api); - var bgColor = labelModel.get('backgroundColor'); - - if (!bgColor || bgColor === 'auto') { - bgColor = axisModel.get(['axisLine', 'lineStyle', 'color']); - } - - elOption.label = { - // shape: {x: 0, y: 0, width: width, height: height, r: labelModel.get('borderRadius')}, - x: position[0], - y: position[1], - style: createTextStyle(labelModel, { - text: text, - font: font, - fill: labelModel.getTextColor(), - padding: paddings, - backgroundColor: bgColor - }), - // Label should be over axisPointer. - z2: 10 - }; - } // Do not overflow ec container - - function confineInContainer(position, width, height, api) { - var viewWidth = api.getWidth(); - var viewHeight = api.getHeight(); - position[0] = Math.min(position[0] + width, viewWidth) - width; - position[1] = Math.min(position[1] + height, viewHeight) - height; - position[0] = Math.max(position[0], 0); - position[1] = Math.max(position[1], 0); - } - - function getValueLabel(value, axis, ecModel, seriesDataIndices, opt) { - value = axis.scale.parse(value); - var text = axis.scale.getLabel({ - value: value - }, { - // If `precision` is set, width can be fixed (like '12.00500'), which - // helps to debounce when when moving label. - precision: opt.precision - }); - var formatter = opt.formatter; - - if (formatter) { - var params_1 = { - value: getAxisRawValue(axis, { - value: value - }), - axisDimension: axis.dim, - axisIndex: axis.index, - seriesData: [] - }; - each(seriesDataIndices, function (idxItem) { - var series = ecModel.getSeriesByIndex(idxItem.seriesIndex); - var dataIndex = idxItem.dataIndexInside; - var dataParams = series && series.getDataParams(dataIndex); - dataParams && params_1.seriesData.push(dataParams); - }); - - if (isString(formatter)) { - text = formatter.replace('{value}', text); - } else if (isFunction(formatter)) { - text = formatter(params_1); - } - } - - return text; - } - function getTransformedPosition(axis, value, layoutInfo) { - var transform = create$1(); - rotate(transform, transform, layoutInfo.rotation); - translate(transform, transform, layoutInfo.position); - return applyTransform$1([axis.dataToCoord(value), (layoutInfo.labelOffset || 0) + (layoutInfo.labelDirection || 1) * (layoutInfo.labelMargin || 0)], transform); - } - function buildCartesianSingleLabelElOption(value, elOption, layoutInfo, axisModel, axisPointerModel, api) { - // @ts-ignore - var textLayout = AxisBuilder.innerTextLayout(layoutInfo.rotation, 0, layoutInfo.labelDirection); - layoutInfo.labelMargin = axisPointerModel.get(['label', 'margin']); - buildLabelElOption(elOption, axisModel, axisPointerModel, api, { - position: getTransformedPosition(axisModel.axis, value, layoutInfo), - align: textLayout.textAlign, - verticalAlign: textLayout.textVerticalAlign - }); - } - function makeLineShape(p1, p2, xDimIndex) { - xDimIndex = xDimIndex || 0; - return { - x1: p1[xDimIndex], - y1: p1[1 - xDimIndex], - x2: p2[xDimIndex], - y2: p2[1 - xDimIndex] - }; - } - function makeRectShape(xy, wh, xDimIndex) { - xDimIndex = xDimIndex || 0; - return { - x: xy[xDimIndex], - y: xy[1 - xDimIndex], - width: wh[xDimIndex], - height: wh[1 - xDimIndex] - }; - } - function makeSectorShape(cx, cy, r0, r, startAngle, endAngle) { - return { - cx: cx, - cy: cy, - r0: r0, - r: r, - startAngle: startAngle, - endAngle: endAngle, - clockwise: true - }; - } - - var CartesianAxisPointer = - /** @class */ - function (_super) { - __extends(CartesianAxisPointer, _super); - - function CartesianAxisPointer() { - return _super !== null && _super.apply(this, arguments) || this; - } - /** - * @override - */ - - - CartesianAxisPointer.prototype.makeElOption = function (elOption, value, axisModel, axisPointerModel, api) { - var axis = axisModel.axis; - var grid = axis.grid; - var axisPointerType = axisPointerModel.get('type'); - var otherExtent = getCartesian(grid, axis).getOtherAxis(axis).getGlobalExtent(); - var pixelValue = axis.toGlobalCoord(axis.dataToCoord(value, true)); - - if (axisPointerType && axisPointerType !== 'none') { - var elStyle = buildElStyle(axisPointerModel); - var pointerOption = pointerShapeBuilder[axisPointerType](axis, pixelValue, otherExtent); - pointerOption.style = elStyle; - elOption.graphicKey = pointerOption.type; - elOption.pointer = pointerOption; - } - - var layoutInfo = layout$1(grid.model, axisModel); - buildCartesianSingleLabelElOption( // @ts-ignore - value, elOption, layoutInfo, axisModel, axisPointerModel, api); - }; - /** - * @override - */ - - - CartesianAxisPointer.prototype.getHandleTransform = function (value, axisModel, axisPointerModel) { - var layoutInfo = layout$1(axisModel.axis.grid.model, axisModel, { - labelInside: false - }); // @ts-ignore - - layoutInfo.labelMargin = axisPointerModel.get(['handle', 'margin']); - var pos = getTransformedPosition(axisModel.axis, value, layoutInfo); - return { - x: pos[0], - y: pos[1], - rotation: layoutInfo.rotation + (layoutInfo.labelDirection < 0 ? Math.PI : 0) - }; - }; - /** - * @override - */ - - - CartesianAxisPointer.prototype.updateHandleTransform = function (transform, delta, axisModel, axisPointerModel) { - var axis = axisModel.axis; - var grid = axis.grid; - var axisExtent = axis.getGlobalExtent(true); - var otherExtent = getCartesian(grid, axis).getOtherAxis(axis).getGlobalExtent(); - var dimIndex = axis.dim === 'x' ? 0 : 1; - var currPosition = [transform.x, transform.y]; - currPosition[dimIndex] += delta[dimIndex]; - currPosition[dimIndex] = Math.min(axisExtent[1], currPosition[dimIndex]); - currPosition[dimIndex] = Math.max(axisExtent[0], currPosition[dimIndex]); - var cursorOtherValue = (otherExtent[1] + otherExtent[0]) / 2; - var cursorPoint = [cursorOtherValue, cursorOtherValue]; - cursorPoint[dimIndex] = currPosition[dimIndex]; // Make tooltip do not overlap axisPointer and in the middle of the grid. - - var tooltipOptions = [{ - verticalAlign: 'middle' - }, { - align: 'center' - }]; - return { - x: currPosition[0], - y: currPosition[1], - rotation: transform.rotation, - cursorPoint: cursorPoint, - tooltipOption: tooltipOptions[dimIndex] - }; - }; - - return CartesianAxisPointer; - }(BaseAxisPointer); - - function getCartesian(grid, axis) { - var opt = {}; - opt[axis.dim + 'AxisIndex'] = axis.index; - return grid.getCartesian(opt); - } - - var pointerShapeBuilder = { - line: function (axis, pixelValue, otherExtent) { - var targetShape = makeLineShape([pixelValue, otherExtent[0]], [pixelValue, otherExtent[1]], getAxisDimIndex(axis)); - return { - type: 'Line', - subPixelOptimize: true, - shape: targetShape - }; - }, - shadow: function (axis, pixelValue, otherExtent) { - var bandWidth = Math.max(1, axis.getBandWidth()); - var span = otherExtent[1] - otherExtent[0]; - return { - type: 'Rect', - shape: makeRectShape([pixelValue - bandWidth / 2, otherExtent[0]], [bandWidth, span], getAxisDimIndex(axis)) - }; - } - }; - - function getAxisDimIndex(axis) { - return axis.dim === 'x' ? 0 : 1; - } - - var AxisPointerModel = - /** @class */ - function (_super) { - __extends(AxisPointerModel, _super); - - function AxisPointerModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = AxisPointerModel.type; - return _this; - } - - AxisPointerModel.type = 'axisPointer'; - AxisPointerModel.defaultOption = { - // 'auto' means that show when triggered by tooltip or handle. - show: 'auto', - // zlevel: 0, - z: 50, - type: 'line', - // axispointer triggered by tootip determine snap automatically, - // see `modelHelper`. - snap: false, - triggerTooltip: true, - value: null, - status: null, - link: [], - // Do not set 'auto' here, otherwise global animation: false - // will not effect at this axispointer. - animation: null, - animationDurationUpdate: 200, - lineStyle: { - color: '#B9BEC9', - width: 1, - type: 'dashed' - }, - shadowStyle: { - color: 'rgba(210,219,238,0.2)' - }, - label: { - show: true, - formatter: null, - precision: 'auto', - margin: 3, - color: '#fff', - padding: [5, 7, 5, 7], - backgroundColor: 'auto', - borderColor: null, - borderWidth: 0, - borderRadius: 3 - }, - handle: { - show: false, - // eslint-disable-next-line - icon: 'M10.7,11.9v-1.3H9.3v1.3c-4.9,0.3-8.8,4.4-8.8,9.4c0,5,3.9,9.1,8.8,9.4h1.3c4.9-0.3,8.8-4.4,8.8-9.4C19.5,16.3,15.6,12.2,10.7,11.9z M13.3,24.4H6.7v-1.2h6.6z M13.3,22H6.7v-1.2h6.6z M13.3,19.6H6.7v-1.2h6.6z', - size: 45, - // handle margin is from symbol center to axis, which is stable when circular move. - margin: 50, - // color: '#1b8bbd' - // color: '#2f4554' - color: '#333', - shadowBlur: 3, - shadowColor: '#aaa', - shadowOffsetX: 0, - shadowOffsetY: 2, - // For mobile performance - throttle: 40 - } - }; - return AxisPointerModel; - }(ComponentModel); - - var inner$b = makeInner(); - var each$7 = each; - /** - * @param {string} key - * @param {module:echarts/ExtensionAPI} api - * @param {Function} handler - * param: {string} currTrigger - * param: {Array.<number>} point - */ - - function register(key, api, handler) { - if (env.node) { - return; - } - - var zr = api.getZr(); - inner$b(zr).records || (inner$b(zr).records = {}); - initGlobalListeners(zr, api); - var record = inner$b(zr).records[key] || (inner$b(zr).records[key] = {}); - record.handler = handler; - } - - function initGlobalListeners(zr, api) { - if (inner$b(zr).initialized) { - return; - } - - inner$b(zr).initialized = true; - useHandler('click', curry(doEnter, 'click')); - useHandler('mousemove', curry(doEnter, 'mousemove')); // useHandler('mouseout', onLeave); - - useHandler('globalout', onLeave); - - function useHandler(eventType, cb) { - zr.on(eventType, function (e) { - var dis = makeDispatchAction(api); - each$7(inner$b(zr).records, function (record) { - record && cb(record, e, dis.dispatchAction); - }); - dispatchTooltipFinally(dis.pendings, api); - }); - } - } - - function dispatchTooltipFinally(pendings, api) { - var showLen = pendings.showTip.length; - var hideLen = pendings.hideTip.length; - var actuallyPayload; - - if (showLen) { - actuallyPayload = pendings.showTip[showLen - 1]; - } else if (hideLen) { - actuallyPayload = pendings.hideTip[hideLen - 1]; - } - - if (actuallyPayload) { - actuallyPayload.dispatchAction = null; - api.dispatchAction(actuallyPayload); - } - } - - function onLeave(record, e, dispatchAction) { - record.handler('leave', null, dispatchAction); - } - - function doEnter(currTrigger, record, e, dispatchAction) { - record.handler(currTrigger, e, dispatchAction); - } - - function makeDispatchAction(api) { - var pendings = { - showTip: [], - hideTip: [] - }; // FIXME - // better approach? - // 'showTip' and 'hideTip' can be triggered by axisPointer and tooltip, - // which may be conflict, (axisPointer call showTip but tooltip call hideTip); - // So we have to add "final stage" to merge those dispatched actions. - - var dispatchAction = function (payload) { - var pendingList = pendings[payload.type]; - - if (pendingList) { - pendingList.push(payload); - } else { - payload.dispatchAction = dispatchAction; - api.dispatchAction(payload); - } - }; - - return { - dispatchAction: dispatchAction, - pendings: pendings - }; - } - - function unregister(key, api) { - if (env.node) { - return; - } - - var zr = api.getZr(); - var record = (inner$b(zr).records || {})[key]; - - if (record) { - inner$b(zr).records[key] = null; - } - } - - var AxisPointerView = - /** @class */ - function (_super) { - __extends(AxisPointerView, _super); - - function AxisPointerView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = AxisPointerView.type; - return _this; - } - - AxisPointerView.prototype.render = function (globalAxisPointerModel, ecModel, api) { - var globalTooltipModel = ecModel.getComponent('tooltip'); - var triggerOn = globalAxisPointerModel.get('triggerOn') || globalTooltipModel && globalTooltipModel.get('triggerOn') || 'mousemove|click'; // Register global listener in AxisPointerView to enable - // AxisPointerView to be independent to Tooltip. - - register('axisPointer', api, function (currTrigger, e, dispatchAction) { - // If 'none', it is not controlled by mouse totally. - if (triggerOn !== 'none' && (currTrigger === 'leave' || triggerOn.indexOf(currTrigger) >= 0)) { - dispatchAction({ - type: 'updateAxisPointer', - currTrigger: currTrigger, - x: e && e.offsetX, - y: e && e.offsetY - }); - } - }); - }; - - AxisPointerView.prototype.remove = function (ecModel, api) { - unregister('axisPointer', api); - }; - - AxisPointerView.prototype.dispose = function (ecModel, api) { - unregister('axisPointer', api); - }; - - AxisPointerView.type = 'axisPointer'; - return AxisPointerView; - }(ComponentView); - - /** - * @param finder contains {seriesIndex, dataIndex, dataIndexInside} - * @param ecModel - * @return {point: [x, y], el: ...} point Will not be null. - */ - - function findPointFromSeries(finder, ecModel) { - var point = []; - var seriesIndex = finder.seriesIndex; - var seriesModel; - - if (seriesIndex == null || !(seriesModel = ecModel.getSeriesByIndex(seriesIndex))) { - return { - point: [] - }; - } - - var data = seriesModel.getData(); - var dataIndex = queryDataIndex(data, finder); - - if (dataIndex == null || dataIndex < 0 || isArray(dataIndex)) { - return { - point: [] - }; - } - - var el = data.getItemGraphicEl(dataIndex); - var coordSys = seriesModel.coordinateSystem; - - if (seriesModel.getTooltipPosition) { - point = seriesModel.getTooltipPosition(dataIndex) || []; - } else if (coordSys && coordSys.dataToPoint) { - if (finder.isStacked) { - var baseAxis = coordSys.getBaseAxis(); - var valueAxis = coordSys.getOtherAxis(baseAxis); - var valueAxisDim = valueAxis.dim; - var baseAxisDim = baseAxis.dim; - var baseDataOffset = valueAxisDim === 'x' || valueAxisDim === 'radius' ? 1 : 0; - var baseDim = data.mapDimension(baseAxisDim); - var stackedData = []; - stackedData[baseDataOffset] = data.get(baseDim, dataIndex); - stackedData[1 - baseDataOffset] = data.get(data.getCalculationInfo('stackResultDimension'), dataIndex); - point = coordSys.dataToPoint(stackedData) || []; - } else { - point = coordSys.dataToPoint(data.getValues(map(coordSys.dimensions, function (dim) { - return data.mapDimension(dim); - }), dataIndex)) || []; - } - } else if (el) { - // Use graphic bounding rect - var rect = el.getBoundingRect().clone(); - rect.applyTransform(el.transform); - point = [rect.x + rect.width / 2, rect.y + rect.height / 2]; - } - - return { - point: point, - el: el - }; - } - - var inner$c = makeInner(); - /** - * Basic logic: check all axis, if they do not demand show/highlight, - * then hide/downplay them. - * - * @return content of event obj for echarts.connect. - */ - - function axisTrigger(payload, ecModel, api) { - var currTrigger = payload.currTrigger; - var point = [payload.x, payload.y]; - var finder = payload; - var dispatchAction = payload.dispatchAction || bind(api.dispatchAction, api); - var coordSysAxesInfo = ecModel.getComponent('axisPointer').coordSysAxesInfo; // Pending - // See #6121. But we are not able to reproduce it yet. - - if (!coordSysAxesInfo) { - return; - } - - if (illegalPoint(point)) { - // Used in the default behavior of `connection`: use the sample seriesIndex - // and dataIndex. And also used in the tooltipView trigger. - point = findPointFromSeries({ - seriesIndex: finder.seriesIndex, - // Do not use dataIndexInside from other ec instance. - // FIXME: auto detect it? - dataIndex: finder.dataIndex - }, ecModel).point; - } - - var isIllegalPoint = illegalPoint(point); // Axis and value can be specified when calling dispatchAction({type: 'updateAxisPointer'}). - // Notice: In this case, it is difficult to get the `point` (which is necessary to show - // tooltip, so if point is not given, we just use the point found by sample seriesIndex - // and dataIndex. - - var inputAxesInfo = finder.axesInfo; - var axesInfo = coordSysAxesInfo.axesInfo; - var shouldHide = currTrigger === 'leave' || illegalPoint(point); - var outputPayload = {}; - var showValueMap = {}; - var dataByCoordSys = { - list: [], - map: {} - }; - var updaters = { - showPointer: curry(showPointer, showValueMap), - showTooltip: curry(showTooltip, dataByCoordSys) - }; // Process for triggered axes. - - each(coordSysAxesInfo.coordSysMap, function (coordSys, coordSysKey) { - // If a point given, it must be contained by the coordinate system. - var coordSysContainsPoint = isIllegalPoint || coordSys.containPoint(point); - each(coordSysAxesInfo.coordSysAxesInfo[coordSysKey], function (axisInfo, key) { - var axis = axisInfo.axis; - var inputAxisInfo = findInputAxisInfo(inputAxesInfo, axisInfo); // If no inputAxesInfo, no axis is restricted. - - if (!shouldHide && coordSysContainsPoint && (!inputAxesInfo || inputAxisInfo)) { - var val = inputAxisInfo && inputAxisInfo.value; - - if (val == null && !isIllegalPoint) { - val = axis.pointToData(point); - } - - val != null && processOnAxis(axisInfo, val, updaters, false, outputPayload); - } - }); - }); // Process for linked axes. - - var linkTriggers = {}; - each(axesInfo, function (tarAxisInfo, tarKey) { - var linkGroup = tarAxisInfo.linkGroup; // If axis has been triggered in the previous stage, it should not be triggered by link. - - if (linkGroup && !showValueMap[tarKey]) { - each(linkGroup.axesInfo, function (srcAxisInfo, srcKey) { - var srcValItem = showValueMap[srcKey]; // If srcValItem exist, source axis is triggered, so link to target axis. - - if (srcAxisInfo !== tarAxisInfo && srcValItem) { - var val = srcValItem.value; - linkGroup.mapper && (val = tarAxisInfo.axis.scale.parse(linkGroup.mapper(val, makeMapperParam(srcAxisInfo), makeMapperParam(tarAxisInfo)))); - linkTriggers[tarAxisInfo.key] = val; - } - }); - } - }); - each(linkTriggers, function (val, tarKey) { - processOnAxis(axesInfo[tarKey], val, updaters, true, outputPayload); - }); - updateModelActually(showValueMap, axesInfo, outputPayload); - dispatchTooltipActually(dataByCoordSys, point, payload, dispatchAction); - dispatchHighDownActually(axesInfo, dispatchAction, api); - return outputPayload; - } - - function processOnAxis(axisInfo, newValue, updaters, noSnap, outputFinder) { - var axis = axisInfo.axis; - - if (axis.scale.isBlank() || !axis.containData(newValue)) { - return; - } - - if (!axisInfo.involveSeries) { - updaters.showPointer(axisInfo, newValue); - return; - } // Heavy calculation. So put it after axis.containData checking. - - - var payloadInfo = buildPayloadsBySeries(newValue, axisInfo); - var payloadBatch = payloadInfo.payloadBatch; - var snapToValue = payloadInfo.snapToValue; // Fill content of event obj for echarts.connect. - // By default use the first involved series data as a sample to connect. - - if (payloadBatch[0] && outputFinder.seriesIndex == null) { - extend(outputFinder, payloadBatch[0]); - } // If no linkSource input, this process is for collecting link - // target, where snap should not be accepted. - - - if (!noSnap && axisInfo.snap) { - if (axis.containData(snapToValue) && snapToValue != null) { - newValue = snapToValue; - } - } - - updaters.showPointer(axisInfo, newValue, payloadBatch); // Tooltip should always be snapToValue, otherwise there will be - // incorrect "axis value ~ series value" mapping displayed in tooltip. - - updaters.showTooltip(axisInfo, payloadInfo, snapToValue); - } - - function buildPayloadsBySeries(value, axisInfo) { - var axis = axisInfo.axis; - var dim = axis.dim; - var snapToValue = value; - var payloadBatch = []; - var minDist = Number.MAX_VALUE; - var minDiff = -1; - each(axisInfo.seriesModels, function (series, idx) { - var dataDim = series.getData().mapDimensionsAll(dim); - var seriesNestestValue; - var dataIndices; - - if (series.getAxisTooltipData) { - var result = series.getAxisTooltipData(dataDim, value, axis); - dataIndices = result.dataIndices; - seriesNestestValue = result.nestestValue; - } else { - dataIndices = series.getData().indicesOfNearest(dataDim[0], value, // Add a threshold to avoid find the wrong dataIndex - // when data length is not same. - // false, - axis.type === 'category' ? 0.5 : null); - - if (!dataIndices.length) { - return; - } - - seriesNestestValue = series.getData().get(dataDim[0], dataIndices[0]); - } - - if (seriesNestestValue == null || !isFinite(seriesNestestValue)) { - return; - } - - var diff = value - seriesNestestValue; - var dist = Math.abs(diff); // Consider category case - - if (dist <= minDist) { - if (dist < minDist || diff >= 0 && minDiff < 0) { - minDist = dist; - minDiff = diff; - snapToValue = seriesNestestValue; - payloadBatch.length = 0; - } - - each(dataIndices, function (dataIndex) { - payloadBatch.push({ - seriesIndex: series.seriesIndex, - dataIndexInside: dataIndex, - dataIndex: series.getData().getRawIndex(dataIndex) - }); - }); - } - }); - return { - payloadBatch: payloadBatch, - snapToValue: snapToValue - }; - } - - function showPointer(showValueMap, axisInfo, value, payloadBatch) { - showValueMap[axisInfo.key] = { - value: value, - payloadBatch: payloadBatch - }; - } - - function showTooltip(dataByCoordSys, axisInfo, payloadInfo, value) { - var payloadBatch = payloadInfo.payloadBatch; - var axis = axisInfo.axis; - var axisModel = axis.model; - var axisPointerModel = axisInfo.axisPointerModel; // If no data, do not create anything in dataByCoordSys, - // whose length will be used to judge whether dispatch action. - - if (!axisInfo.triggerTooltip || !payloadBatch.length) { - return; - } - - var coordSysModel = axisInfo.coordSys.model; - var coordSysKey = makeKey(coordSysModel); - var coordSysItem = dataByCoordSys.map[coordSysKey]; - - if (!coordSysItem) { - coordSysItem = dataByCoordSys.map[coordSysKey] = { - coordSysId: coordSysModel.id, - coordSysIndex: coordSysModel.componentIndex, - coordSysType: coordSysModel.type, - coordSysMainType: coordSysModel.mainType, - dataByAxis: [] - }; - dataByCoordSys.list.push(coordSysItem); - } - - coordSysItem.dataByAxis.push({ - axisDim: axis.dim, - axisIndex: axisModel.componentIndex, - axisType: axisModel.type, - axisId: axisModel.id, - value: value, - // Caustion: viewHelper.getValueLabel is actually on "view stage", which - // depends that all models have been updated. So it should not be performed - // here. Considering axisPointerModel used here is volatile, which is hard - // to be retrieve in TooltipView, we prepare parameters here. - valueLabelOpt: { - precision: axisPointerModel.get(['label', 'precision']), - formatter: axisPointerModel.get(['label', 'formatter']) - }, - seriesDataIndices: payloadBatch.slice() - }); - } - - function updateModelActually(showValueMap, axesInfo, outputPayload) { - var outputAxesInfo = outputPayload.axesInfo = []; // Basic logic: If no 'show' required, 'hide' this axisPointer. - - each(axesInfo, function (axisInfo, key) { - var option = axisInfo.axisPointerModel.option; - var valItem = showValueMap[key]; - - if (valItem) { - !axisInfo.useHandle && (option.status = 'show'); - option.value = valItem.value; // For label formatter param and highlight. - - option.seriesDataIndices = (valItem.payloadBatch || []).slice(); - } // When always show (e.g., handle used), remain - // original value and status. - else { - // If hide, value still need to be set, consider - // click legend to toggle axis blank. - !axisInfo.useHandle && (option.status = 'hide'); - } // If status is 'hide', should be no info in payload. - - - option.status === 'show' && outputAxesInfo.push({ - axisDim: axisInfo.axis.dim, - axisIndex: axisInfo.axis.model.componentIndex, - value: option.value - }); - }); - } - - function dispatchTooltipActually(dataByCoordSys, point, payload, dispatchAction) { - // Basic logic: If no showTip required, hideTip will be dispatched. - if (illegalPoint(point) || !dataByCoordSys.list.length) { - dispatchAction({ - type: 'hideTip' - }); - return; - } // In most case only one axis (or event one series is used). It is - // convenient to fetch payload.seriesIndex and payload.dataIndex - // directly. So put the first seriesIndex and dataIndex of the first - // axis on the payload. - - - var sampleItem = ((dataByCoordSys.list[0].dataByAxis[0] || {}).seriesDataIndices || [])[0] || {}; - dispatchAction({ - type: 'showTip', - escapeConnect: true, - x: point[0], - y: point[1], - tooltipOption: payload.tooltipOption, - position: payload.position, - dataIndexInside: sampleItem.dataIndexInside, - dataIndex: sampleItem.dataIndex, - seriesIndex: sampleItem.seriesIndex, - dataByCoordSys: dataByCoordSys.list - }); - } - - function dispatchHighDownActually(axesInfo, dispatchAction, api) { - // FIXME - // highlight status modification should be a stage of main process? - // (Consider confilct (e.g., legend and axisPointer) and setOption) - var zr = api.getZr(); - var highDownKey = 'axisPointerLastHighlights'; - var lastHighlights = inner$c(zr)[highDownKey] || {}; - var newHighlights = inner$c(zr)[highDownKey] = {}; // Update highlight/downplay status according to axisPointer model. - // Build hash map and remove duplicate incidentally. - - each(axesInfo, function (axisInfo, key) { - var option = axisInfo.axisPointerModel.option; - option.status === 'show' && each(option.seriesDataIndices, function (batchItem) { - var key = batchItem.seriesIndex + ' | ' + batchItem.dataIndex; - newHighlights[key] = batchItem; - }); - }); // Diff. - - var toHighlight = []; - var toDownplay = []; - each(lastHighlights, function (batchItem, key) { - !newHighlights[key] && toDownplay.push(batchItem); - }); - each(newHighlights, function (batchItem, key) { - !lastHighlights[key] && toHighlight.push(batchItem); - }); - toDownplay.length && api.dispatchAction({ - type: 'downplay', - escapeConnect: true, - // Not blur others when highlight in axisPointer. - notBlur: true, - batch: toDownplay - }); - toHighlight.length && api.dispatchAction({ - type: 'highlight', - escapeConnect: true, - // Not blur others when highlight in axisPointer. - notBlur: true, - batch: toHighlight - }); - } - - function findInputAxisInfo(inputAxesInfo, axisInfo) { - for (var i = 0; i < (inputAxesInfo || []).length; i++) { - var inputAxisInfo = inputAxesInfo[i]; - - if (axisInfo.axis.dim === inputAxisInfo.axisDim && axisInfo.axis.model.componentIndex === inputAxisInfo.axisIndex) { - return inputAxisInfo; - } - } - } - - function makeMapperParam(axisInfo) { - var axisModel = axisInfo.axis.model; - var item = {}; - var dim = item.axisDim = axisInfo.axis.dim; - item.axisIndex = item[dim + 'AxisIndex'] = axisModel.componentIndex; - item.axisName = item[dim + 'AxisName'] = axisModel.name; - item.axisId = item[dim + 'AxisId'] = axisModel.id; - return item; - } - - function illegalPoint(point) { - return !point || point[0] == null || isNaN(point[0]) || point[1] == null || isNaN(point[1]); - } - - function install$s(registers) { - // CartesianAxisPointer is not supposed to be required here. But consider - // echarts.simple.js and online build tooltip, which only require gridSimple, - // CartesianAxisPointer should be able to required somewhere. - AxisView.registerAxisPointerClass('CartesianAxisPointer', CartesianAxisPointer); - registers.registerComponentModel(AxisPointerModel); - registers.registerComponentView(AxisPointerView); - registers.registerPreprocessor(function (option) { - // Always has a global axisPointerModel for default setting. - if (option) { - (!option.axisPointer || option.axisPointer.length === 0) && (option.axisPointer = {}); - var link = option.axisPointer.link; // Normalize to array to avoid object mergin. But if link - // is not set, remain null/undefined, otherwise it will - // override existent link setting. - - if (link && !isArray(link)) { - option.axisPointer.link = [link]; - } - } - }); // This process should proformed after coordinate systems created - // and series data processed. So put it on statistic processing stage. - - registers.registerProcessor(registers.PRIORITY.PROCESSOR.STATISTIC, function (ecModel, api) { - // Build axisPointerModel, mergin tooltip.axisPointer model for each axis. - // allAxesInfo should be updated when setOption performed. - ecModel.getComponent('axisPointer').coordSysAxesInfo = collect(ecModel, api); - }); // Broadcast to all views. - - registers.registerAction({ - type: 'updateAxisPointer', - event: 'updateAxisPointer', - update: ':updateAxisPointer' - }, axisTrigger); - } - - function install$t(registers) { - use(install$5); - use(install$s); - } - - var PolarAxisPointer = - /** @class */ - function (_super) { - __extends(PolarAxisPointer, _super); - - function PolarAxisPointer() { - return _super !== null && _super.apply(this, arguments) || this; - } - /** - * @override - */ - - - PolarAxisPointer.prototype.makeElOption = function (elOption, value, axisModel, axisPointerModel, api) { - var axis = axisModel.axis; - - if (axis.dim === 'angle') { - this.animationThreshold = Math.PI / 18; - } - - var polar = axis.polar; - var otherAxis = polar.getOtherAxis(axis); - var otherExtent = otherAxis.getExtent(); - var coordValue = axis.dataToCoord(value); - var axisPointerType = axisPointerModel.get('type'); - - if (axisPointerType && axisPointerType !== 'none') { - var elStyle = buildElStyle(axisPointerModel); - var pointerOption = pointerShapeBuilder$1[axisPointerType](axis, polar, coordValue, otherExtent); - pointerOption.style = elStyle; - elOption.graphicKey = pointerOption.type; - elOption.pointer = pointerOption; - } - - var labelMargin = axisPointerModel.get(['label', 'margin']); - var labelPos = getLabelPosition(value, axisModel, axisPointerModel, polar, labelMargin); - buildLabelElOption(elOption, axisModel, axisPointerModel, api, labelPos); - }; - - return PolarAxisPointer; - }(BaseAxisPointer); - - function getLabelPosition(value, axisModel, axisPointerModel, polar, labelMargin) { - var axis = axisModel.axis; - var coord = axis.dataToCoord(value); - var axisAngle = polar.getAngleAxis().getExtent()[0]; - axisAngle = axisAngle / 180 * Math.PI; - var radiusExtent = polar.getRadiusAxis().getExtent(); - var position; - var align; - var verticalAlign; - - if (axis.dim === 'radius') { - var transform = create$1(); - rotate(transform, transform, axisAngle); - translate(transform, transform, [polar.cx, polar.cy]); - position = applyTransform$1([coord, -labelMargin], transform); - var labelRotation = axisModel.getModel('axisLabel').get('rotate') || 0; // @ts-ignore - - var labelLayout = AxisBuilder.innerTextLayout(axisAngle, labelRotation * Math.PI / 180, -1); - align = labelLayout.textAlign; - verticalAlign = labelLayout.textVerticalAlign; - } else { - // angle axis - var r = radiusExtent[1]; - position = polar.coordToPoint([r + labelMargin, coord]); - var cx = polar.cx; - var cy = polar.cy; - align = Math.abs(position[0] - cx) / r < 0.3 ? 'center' : position[0] > cx ? 'left' : 'right'; - verticalAlign = Math.abs(position[1] - cy) / r < 0.3 ? 'middle' : position[1] > cy ? 'top' : 'bottom'; - } - - return { - position: position, - align: align, - verticalAlign: verticalAlign - }; - } - - var pointerShapeBuilder$1 = { - line: function (axis, polar, coordValue, otherExtent) { - return axis.dim === 'angle' ? { - type: 'Line', - shape: makeLineShape(polar.coordToPoint([otherExtent[0], coordValue]), polar.coordToPoint([otherExtent[1], coordValue])) - } : { - type: 'Circle', - shape: { - cx: polar.cx, - cy: polar.cy, - r: coordValue - } - }; - }, - shadow: function (axis, polar, coordValue, otherExtent) { - var bandWidth = Math.max(1, axis.getBandWidth()); - var radian = Math.PI / 180; - return axis.dim === 'angle' ? { - type: 'Sector', - shape: makeSectorShape(polar.cx, polar.cy, otherExtent[0], otherExtent[1], // In ECharts y is negative if angle is positive - (-coordValue - bandWidth / 2) * radian, (-coordValue + bandWidth / 2) * radian) - } : { - type: 'Sector', - shape: makeSectorShape(polar.cx, polar.cy, coordValue - bandWidth / 2, coordValue + bandWidth / 2, 0, Math.PI * 2) - }; - } - }; - - var PolarModel = - /** @class */ - function (_super) { - __extends(PolarModel, _super); - - function PolarModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = PolarModel.type; - return _this; - } - - PolarModel.prototype.findAxisModel = function (axisType) { - var foundAxisModel; - var ecModel = this.ecModel; - ecModel.eachComponent(axisType, function (axisModel) { - if (axisModel.getCoordSysModel() === this) { - foundAxisModel = axisModel; - } - }, this); - return foundAxisModel; - }; - - PolarModel.type = 'polar'; - PolarModel.dependencies = ['radiusAxis', 'angleAxis']; - PolarModel.defaultOption = { - // zlevel: 0, - z: 0, - center: ['50%', '50%'], - radius: '80%' - }; - return PolarModel; - }(ComponentModel); - - var PolarAxisModel = - /** @class */ - function (_super) { - __extends(PolarAxisModel, _super); - - function PolarAxisModel() { - return _super !== null && _super.apply(this, arguments) || this; - } - - PolarAxisModel.prototype.getCoordSysModel = function () { - return this.getReferringComponents('polar', SINGLE_REFERRING).models[0]; - }; - - PolarAxisModel.type = 'polarAxis'; - return PolarAxisModel; - }(ComponentModel); - - mixin(PolarAxisModel, AxisModelCommonMixin); - - var AngleAxisModel = - /** @class */ - function (_super) { - __extends(AngleAxisModel, _super); - - function AngleAxisModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = AngleAxisModel.type; - return _this; - } - - AngleAxisModel.type = 'angleAxis'; - return AngleAxisModel; - }(PolarAxisModel); - - var RadiusAxisModel = - /** @class */ - function (_super) { - __extends(RadiusAxisModel, _super); - - function RadiusAxisModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = RadiusAxisModel.type; - return _this; - } - - RadiusAxisModel.type = 'radiusAxis'; - return RadiusAxisModel; - }(PolarAxisModel); - - var RadiusAxis = - /** @class */ - function (_super) { - __extends(RadiusAxis, _super); - - function RadiusAxis(scale, radiusExtent) { - return _super.call(this, 'radius', scale, radiusExtent) || this; - } - - RadiusAxis.prototype.pointToData = function (point, clamp) { - return this.polar.pointToData(point, clamp)[this.dim === 'radius' ? 0 : 1]; - }; - - return RadiusAxis; - }(Axis); - - RadiusAxis.prototype.dataToRadius = Axis.prototype.dataToCoord; - RadiusAxis.prototype.radiusToData = Axis.prototype.coordToData; - - var inner$d = makeInner(); - - var AngleAxis = - /** @class */ - function (_super) { - __extends(AngleAxis, _super); - - function AngleAxis(scale, angleExtent) { - return _super.call(this, 'angle', scale, angleExtent || [0, 360]) || this; - } - - AngleAxis.prototype.pointToData = function (point, clamp) { - return this.polar.pointToData(point, clamp)[this.dim === 'radius' ? 0 : 1]; - }; - /** - * Only be called in category axis. - * Angle axis uses text height to decide interval - * - * @override - * @return {number} Auto interval for cateogry axis tick and label - */ - - - AngleAxis.prototype.calculateCategoryInterval = function () { - var axis = this; - var labelModel = axis.getLabelModel(); - var ordinalScale = axis.scale; - var ordinalExtent = ordinalScale.getExtent(); // Providing this method is for optimization: - // avoid generating a long array by `getTicks` - // in large category data case. - - var tickCount = ordinalScale.count(); - - if (ordinalExtent[1] - ordinalExtent[0] < 1) { - return 0; - } - - var tickValue = ordinalExtent[0]; - var unitSpan = axis.dataToCoord(tickValue + 1) - axis.dataToCoord(tickValue); - var unitH = Math.abs(unitSpan); // Not precise, just use height as text width - // and each distance from axis line yet. - - var rect = getBoundingRect(tickValue == null ? '' : tickValue + '', labelModel.getFont(), 'center', 'top'); - var maxH = Math.max(rect.height, 7); - var dh = maxH / unitH; // 0/0 is NaN, 1/0 is Infinity. - - isNaN(dh) && (dh = Infinity); - var interval = Math.max(0, Math.floor(dh)); - var cache = inner$d(axis.model); - var lastAutoInterval = cache.lastAutoInterval; - var lastTickCount = cache.lastTickCount; // Use cache to keep interval stable while moving zoom window, - // otherwise the calculated interval might jitter when the zoom - // window size is close to the interval-changing size. - - if (lastAutoInterval != null && lastTickCount != null && Math.abs(lastAutoInterval - interval) <= 1 && Math.abs(lastTickCount - tickCount) <= 1 // Always choose the bigger one, otherwise the critical - // point is not the same when zooming in or zooming out. - && lastAutoInterval > interval) { - interval = lastAutoInterval; - } // Only update cache if cache not used, otherwise the - // changing of interval is too insensitive. - else { - cache.lastTickCount = tickCount; - cache.lastAutoInterval = interval; - } - - return interval; - }; - - return AngleAxis; - }(Axis); - - AngleAxis.prototype.dataToAngle = Axis.prototype.dataToCoord; - AngleAxis.prototype.angleToData = Axis.prototype.coordToData; - - var polarDimensions = ['radius', 'angle']; - - var Polar = - /** @class */ - function () { - function Polar(name) { - this.dimensions = polarDimensions; - this.type = 'polar'; - /** - * x of polar center - */ - - this.cx = 0; - /** - * y of polar center - */ - - this.cy = 0; - this._radiusAxis = new RadiusAxis(); - this._angleAxis = new AngleAxis(); - this.axisPointerEnabled = true; - this.name = name || ''; - this._radiusAxis.polar = this._angleAxis.polar = this; - } - /** - * If contain coord - */ - - - Polar.prototype.containPoint = function (point) { - var coord = this.pointToCoord(point); - return this._radiusAxis.contain(coord[0]) && this._angleAxis.contain(coord[1]); - }; - /** - * If contain data - */ - - - Polar.prototype.containData = function (data) { - return this._radiusAxis.containData(data[0]) && this._angleAxis.containData(data[1]); - }; - - Polar.prototype.getAxis = function (dim) { - var key = '_' + dim + 'Axis'; - return this[key]; - }; - - Polar.prototype.getAxes = function () { - return [this._radiusAxis, this._angleAxis]; - }; - /** - * Get axes by type of scale - */ - - - Polar.prototype.getAxesByScale = function (scaleType) { - var axes = []; - var angleAxis = this._angleAxis; - var radiusAxis = this._radiusAxis; - angleAxis.scale.type === scaleType && axes.push(angleAxis); - radiusAxis.scale.type === scaleType && axes.push(radiusAxis); - return axes; - }; - - Polar.prototype.getAngleAxis = function () { - return this._angleAxis; - }; - - Polar.prototype.getRadiusAxis = function () { - return this._radiusAxis; - }; - - Polar.prototype.getOtherAxis = function (axis) { - var angleAxis = this._angleAxis; - return axis === angleAxis ? this._radiusAxis : angleAxis; - }; - /** - * Base axis will be used on stacking. - * - */ - - - Polar.prototype.getBaseAxis = function () { - return this.getAxesByScale('ordinal')[0] || this.getAxesByScale('time')[0] || this.getAngleAxis(); - }; - - Polar.prototype.getTooltipAxes = function (dim) { - var baseAxis = dim != null && dim !== 'auto' ? this.getAxis(dim) : this.getBaseAxis(); - return { - baseAxes: [baseAxis], - otherAxes: [this.getOtherAxis(baseAxis)] - }; - }; - /** - * Convert a single data item to (x, y) point. - * Parameter data is an array which the first element is radius and the second is angle - */ - - - Polar.prototype.dataToPoint = function (data, clamp) { - return this.coordToPoint([this._radiusAxis.dataToRadius(data[0], clamp), this._angleAxis.dataToAngle(data[1], clamp)]); - }; - /** - * Convert a (x, y) point to data - */ - - - Polar.prototype.pointToData = function (point, clamp) { - var coord = this.pointToCoord(point); - return [this._radiusAxis.radiusToData(coord[0], clamp), this._angleAxis.angleToData(coord[1], clamp)]; - }; - /** - * Convert a (x, y) point to (radius, angle) coord - */ - - - Polar.prototype.pointToCoord = function (point) { - var dx = point[0] - this.cx; - var dy = point[1] - this.cy; - var angleAxis = this.getAngleAxis(); - var extent = angleAxis.getExtent(); - var minAngle = Math.min(extent[0], extent[1]); - var maxAngle = Math.max(extent[0], extent[1]); // Fix fixed extent in polarCreator - // FIXME - - angleAxis.inverse ? minAngle = maxAngle - 360 : maxAngle = minAngle + 360; - var radius = Math.sqrt(dx * dx + dy * dy); - dx /= radius; - dy /= radius; - var radian = Math.atan2(-dy, dx) / Math.PI * 180; // move to angleExtent - - var dir = radian < minAngle ? 1 : -1; - - while (radian < minAngle || radian > maxAngle) { - radian += dir * 360; - } - - return [radius, radian]; - }; - /** - * Convert a (radius, angle) coord to (x, y) point - */ - - - Polar.prototype.coordToPoint = function (coord) { - var radius = coord[0]; - var radian = coord[1] / 180 * Math.PI; - var x = Math.cos(radian) * radius + this.cx; // Inverse the y - - var y = -Math.sin(radian) * radius + this.cy; - return [x, y]; - }; - /** - * Get ring area of cartesian. - * Area will have a contain function to determine if a point is in the coordinate system. - */ - - - Polar.prototype.getArea = function () { - var angleAxis = this.getAngleAxis(); - var radiusAxis = this.getRadiusAxis(); - var radiusExtent = radiusAxis.getExtent().slice(); - radiusExtent[0] > radiusExtent[1] && radiusExtent.reverse(); - var angleExtent = angleAxis.getExtent(); - var RADIAN = Math.PI / 180; - return { - cx: this.cx, - cy: this.cy, - r0: radiusExtent[0], - r: radiusExtent[1], - startAngle: -angleExtent[0] * RADIAN, - endAngle: -angleExtent[1] * RADIAN, - clockwise: angleAxis.inverse, - contain: function (x, y) { - // It's a ring shape. - // Start angle and end angle don't matter - var dx = x - this.cx; - var dy = y - this.cy; // minus a tiny value 1e-4 to avoid being clipped unexpectedly - - var d2 = dx * dx + dy * dy - 1e-4; - var r = this.r; - var r0 = this.r0; - return d2 <= r * r && d2 >= r0 * r0; - } - }; - }; - - Polar.prototype.convertToPixel = function (ecModel, finder, value) { - var coordSys = getCoordSys$2(finder); - return coordSys === this ? this.dataToPoint(value) : null; - }; - - Polar.prototype.convertFromPixel = function (ecModel, finder, pixel) { - var coordSys = getCoordSys$2(finder); - return coordSys === this ? this.pointToData(pixel) : null; - }; - - return Polar; - }(); - - function getCoordSys$2(finder) { - var seriesModel = finder.seriesModel; - var polarModel = finder.polarModel; - return polarModel && polarModel.coordinateSystem || seriesModel && seriesModel.coordinateSystem; - } - - /** - * Resize method bound to the polar - */ - - function resizePolar(polar, polarModel, api) { - var center = polarModel.get('center'); - var width = api.getWidth(); - var height = api.getHeight(); - polar.cx = parsePercent$1(center[0], width); - polar.cy = parsePercent$1(center[1], height); - var radiusAxis = polar.getRadiusAxis(); - var size = Math.min(width, height) / 2; - var radius = polarModel.get('radius'); - - if (radius == null) { - radius = [0, '100%']; - } else if (!isArray(radius)) { - // r0 = 0 - radius = [0, radius]; - } - - var parsedRadius = [parsePercent$1(radius[0], size), parsePercent$1(radius[1], size)]; - radiusAxis.inverse ? radiusAxis.setExtent(parsedRadius[1], parsedRadius[0]) : radiusAxis.setExtent(parsedRadius[0], parsedRadius[1]); - } - /** - * Update polar - */ - - - function updatePolarScale(ecModel, api) { - var polar = this; - var angleAxis = polar.getAngleAxis(); - var radiusAxis = polar.getRadiusAxis(); // Reset scale - - angleAxis.scale.setExtent(Infinity, -Infinity); - radiusAxis.scale.setExtent(Infinity, -Infinity); - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.coordinateSystem === polar) { - var data_1 = seriesModel.getData(); - each(getDataDimensionsOnAxis(data_1, 'radius'), function (dim) { - radiusAxis.scale.unionExtentFromData(data_1, dim); - }); - each(getDataDimensionsOnAxis(data_1, 'angle'), function (dim) { - angleAxis.scale.unionExtentFromData(data_1, dim); - }); - } - }); - niceScaleExtent(angleAxis.scale, angleAxis.model); - niceScaleExtent(radiusAxis.scale, radiusAxis.model); // Fix extent of category angle axis - - if (angleAxis.type === 'category' && !angleAxis.onBand) { - var extent = angleAxis.getExtent(); - var diff = 360 / angleAxis.scale.count(); - angleAxis.inverse ? extent[1] += diff : extent[1] -= diff; - angleAxis.setExtent(extent[0], extent[1]); - } - } - - function isAngleAxisModel(axisModel) { - return axisModel.mainType === 'angleAxis'; - } - /** - * Set common axis properties - */ - - - function setAxis(axis, axisModel) { - axis.type = axisModel.get('type'); - axis.scale = createScaleByModel(axisModel); - axis.onBand = axisModel.get('boundaryGap') && axis.type === 'category'; - axis.inverse = axisModel.get('inverse'); - - if (isAngleAxisModel(axisModel)) { - axis.inverse = axis.inverse !== axisModel.get('clockwise'); - var startAngle = axisModel.get('startAngle'); - axis.setExtent(startAngle, startAngle + (axis.inverse ? -360 : 360)); - } // Inject axis instance - - - axisModel.axis = axis; - axis.model = axisModel; - } - - var polarCreator = { - dimensions: polarDimensions, - create: function (ecModel, api) { - var polarList = []; - ecModel.eachComponent('polar', function (polarModel, idx) { - var polar = new Polar(idx + ''); // Inject resize and update method - - polar.update = updatePolarScale; - var radiusAxis = polar.getRadiusAxis(); - var angleAxis = polar.getAngleAxis(); - var radiusAxisModel = polarModel.findAxisModel('radiusAxis'); - var angleAxisModel = polarModel.findAxisModel('angleAxis'); - setAxis(radiusAxis, radiusAxisModel); - setAxis(angleAxis, angleAxisModel); - resizePolar(polar, polarModel, api); - polarList.push(polar); - polarModel.coordinateSystem = polar; - polar.model = polarModel; - }); // Inject coordinateSystem to series - - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.get('coordinateSystem') === 'polar') { - var polarModel = seriesModel.getReferringComponents('polar', SINGLE_REFERRING).models[0]; - - if ("development" !== 'production') { - if (!polarModel) { - throw new Error('Polar "' + retrieve(seriesModel.get('polarIndex'), seriesModel.get('polarId'), 0) + '" not found'); - } - } - - seriesModel.coordinateSystem = polarModel.coordinateSystem; - } - }); - return polarList; - } - }; - - var elementList$1 = ['axisLine', 'axisLabel', 'axisTick', 'minorTick', 'splitLine', 'minorSplitLine', 'splitArea']; - - function getAxisLineShape(polar, rExtent, angle) { - rExtent[1] > rExtent[0] && (rExtent = rExtent.slice().reverse()); - var start = polar.coordToPoint([rExtent[0], angle]); - var end = polar.coordToPoint([rExtent[1], angle]); - return { - x1: start[0], - y1: start[1], - x2: end[0], - y2: end[1] - }; - } - - function getRadiusIdx(polar) { - var radiusAxis = polar.getRadiusAxis(); - return radiusAxis.inverse ? 0 : 1; - } // Remove the last tick which will overlap the first tick - - - function fixAngleOverlap(list) { - var firstItem = list[0]; - var lastItem = list[list.length - 1]; - - if (firstItem && lastItem && Math.abs(Math.abs(firstItem.coord - lastItem.coord) - 360) < 1e-4) { - list.pop(); - } - } - - var AngleAxisView = - /** @class */ - function (_super) { - __extends(AngleAxisView, _super); - - function AngleAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = AngleAxisView.type; - _this.axisPointerClass = 'PolarAxisPointer'; - return _this; - } - - AngleAxisView.prototype.render = function (angleAxisModel, ecModel) { - this.group.removeAll(); - - if (!angleAxisModel.get('show')) { - return; - } - - var angleAxis = angleAxisModel.axis; - var polar = angleAxis.polar; - var radiusExtent = polar.getRadiusAxis().getExtent(); - var ticksAngles = angleAxis.getTicksCoords(); - var minorTickAngles = angleAxis.getMinorTicksCoords(); - var labels = map(angleAxis.getViewLabels(), function (labelItem) { - labelItem = clone(labelItem); - var scale = angleAxis.scale; - var tickValue = scale.type === 'ordinal' ? scale.getRawOrdinalNumber(labelItem.tickValue) : labelItem.tickValue; - labelItem.coord = angleAxis.dataToCoord(tickValue); - return labelItem; - }); - fixAngleOverlap(labels); - fixAngleOverlap(ticksAngles); - each(elementList$1, function (name) { - if (angleAxisModel.get([name, 'show']) && (!angleAxis.scale.isBlank() || name === 'axisLine')) { - angelAxisElementsBuilders[name](this.group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent, labels); - } - }, this); - }; - - AngleAxisView.type = 'angleAxis'; - return AngleAxisView; - }(AxisView); - - var angelAxisElementsBuilders = { - axisLine: function (group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent) { - var lineStyleModel = angleAxisModel.getModel(['axisLine', 'lineStyle']); // extent id of the axis radius (r0 and r) - - var rId = getRadiusIdx(polar); - var r0Id = rId ? 0 : 1; - var shape; - - if (radiusExtent[r0Id] === 0) { - shape = new Circle({ - shape: { - cx: polar.cx, - cy: polar.cy, - r: radiusExtent[rId] - }, - style: lineStyleModel.getLineStyle(), - z2: 1, - silent: true - }); - } else { - shape = new Ring({ - shape: { - cx: polar.cx, - cy: polar.cy, - r: radiusExtent[rId], - r0: radiusExtent[r0Id] - }, - style: lineStyleModel.getLineStyle(), - z2: 1, - silent: true - }); - } - - shape.style.fill = null; - group.add(shape); - }, - axisTick: function (group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent) { - var tickModel = angleAxisModel.getModel('axisTick'); - var tickLen = (tickModel.get('inside') ? -1 : 1) * tickModel.get('length'); - var radius = radiusExtent[getRadiusIdx(polar)]; - var lines = map(ticksAngles, function (tickAngleItem) { - return new Line({ - shape: getAxisLineShape(polar, [radius, radius + tickLen], tickAngleItem.coord) - }); - }); - group.add(mergePath$1(lines, { - style: defaults(tickModel.getModel('lineStyle').getLineStyle(), { - stroke: angleAxisModel.get(['axisLine', 'lineStyle', 'color']) - }) - })); - }, - minorTick: function (group, angleAxisModel, polar, tickAngles, minorTickAngles, radiusExtent) { - if (!minorTickAngles.length) { - return; - } - - var tickModel = angleAxisModel.getModel('axisTick'); - var minorTickModel = angleAxisModel.getModel('minorTick'); - var tickLen = (tickModel.get('inside') ? -1 : 1) * minorTickModel.get('length'); - var radius = radiusExtent[getRadiusIdx(polar)]; - var lines = []; - - for (var i = 0; i < minorTickAngles.length; i++) { - for (var k = 0; k < minorTickAngles[i].length; k++) { - lines.push(new Line({ - shape: getAxisLineShape(polar, [radius, radius + tickLen], minorTickAngles[i][k].coord) - })); - } - } - - group.add(mergePath$1(lines, { - style: defaults(minorTickModel.getModel('lineStyle').getLineStyle(), defaults(tickModel.getLineStyle(), { - stroke: angleAxisModel.get(['axisLine', 'lineStyle', 'color']) - })) - })); - }, - axisLabel: function (group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent, labels) { - var rawCategoryData = angleAxisModel.getCategories(true); - var commonLabelModel = angleAxisModel.getModel('axisLabel'); - var labelMargin = commonLabelModel.get('margin'); - var triggerEvent = angleAxisModel.get('triggerEvent'); // Use length of ticksAngles because it may remove the last tick to avoid overlapping - - each(labels, function (labelItem, idx) { - var labelModel = commonLabelModel; - var tickValue = labelItem.tickValue; - var r = radiusExtent[getRadiusIdx(polar)]; - var p = polar.coordToPoint([r + labelMargin, labelItem.coord]); - var cx = polar.cx; - var cy = polar.cy; - var labelTextAlign = Math.abs(p[0] - cx) / r < 0.3 ? 'center' : p[0] > cx ? 'left' : 'right'; - var labelTextVerticalAlign = Math.abs(p[1] - cy) / r < 0.3 ? 'middle' : p[1] > cy ? 'top' : 'bottom'; - - if (rawCategoryData && rawCategoryData[tickValue]) { - var rawCategoryItem = rawCategoryData[tickValue]; - - if (isObject(rawCategoryItem) && rawCategoryItem.textStyle) { - labelModel = new Model(rawCategoryItem.textStyle, commonLabelModel, commonLabelModel.ecModel); - } - } - - var textEl = new ZRText({ - silent: AxisBuilder.isLabelSilent(angleAxisModel), - style: createTextStyle(labelModel, { - x: p[0], - y: p[1], - fill: labelModel.getTextColor() || angleAxisModel.get(['axisLine', 'lineStyle', 'color']), - text: labelItem.formattedLabel, - align: labelTextAlign, - verticalAlign: labelTextVerticalAlign - }) - }); - group.add(textEl); // Pack data for mouse event - - if (triggerEvent) { - var eventData = AxisBuilder.makeAxisEventDataBase(angleAxisModel); - eventData.targetType = 'axisLabel'; - eventData.value = labelItem.rawLabel; - getECData(textEl).eventData = eventData; - } - }, this); - }, - splitLine: function (group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent) { - var splitLineModel = angleAxisModel.getModel('splitLine'); - var lineStyleModel = splitLineModel.getModel('lineStyle'); - var lineColors = lineStyleModel.get('color'); - var lineCount = 0; - lineColors = lineColors instanceof Array ? lineColors : [lineColors]; - var splitLines = []; - - for (var i = 0; i < ticksAngles.length; i++) { - var colorIndex = lineCount++ % lineColors.length; - splitLines[colorIndex] = splitLines[colorIndex] || []; - splitLines[colorIndex].push(new Line({ - shape: getAxisLineShape(polar, radiusExtent, ticksAngles[i].coord) - })); - } // Simple optimization - // Batching the lines if color are the same - - - for (var i = 0; i < splitLines.length; i++) { - group.add(mergePath$1(splitLines[i], { - style: defaults({ - stroke: lineColors[i % lineColors.length] - }, lineStyleModel.getLineStyle()), - silent: true, - z: angleAxisModel.get('z') - })); - } - }, - minorSplitLine: function (group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent) { - if (!minorTickAngles.length) { - return; - } - - var minorSplitLineModel = angleAxisModel.getModel('minorSplitLine'); - var lineStyleModel = minorSplitLineModel.getModel('lineStyle'); - var lines = []; - - for (var i = 0; i < minorTickAngles.length; i++) { - for (var k = 0; k < minorTickAngles[i].length; k++) { - lines.push(new Line({ - shape: getAxisLineShape(polar, radiusExtent, minorTickAngles[i][k].coord) - })); - } - } - - group.add(mergePath$1(lines, { - style: lineStyleModel.getLineStyle(), - silent: true, - z: angleAxisModel.get('z') - })); - }, - splitArea: function (group, angleAxisModel, polar, ticksAngles, minorTickAngles, radiusExtent) { - if (!ticksAngles.length) { - return; - } - - var splitAreaModel = angleAxisModel.getModel('splitArea'); - var areaStyleModel = splitAreaModel.getModel('areaStyle'); - var areaColors = areaStyleModel.get('color'); - var lineCount = 0; - areaColors = areaColors instanceof Array ? areaColors : [areaColors]; - var splitAreas = []; - var RADIAN = Math.PI / 180; - var prevAngle = -ticksAngles[0].coord * RADIAN; - var r0 = Math.min(radiusExtent[0], radiusExtent[1]); - var r1 = Math.max(radiusExtent[0], radiusExtent[1]); - var clockwise = angleAxisModel.get('clockwise'); - - for (var i = 1, len = ticksAngles.length; i <= len; i++) { - var coord = i === len ? ticksAngles[0].coord : ticksAngles[i].coord; - var colorIndex = lineCount++ % areaColors.length; - splitAreas[colorIndex] = splitAreas[colorIndex] || []; - splitAreas[colorIndex].push(new Sector({ - shape: { - cx: polar.cx, - cy: polar.cy, - r0: r0, - r: r1, - startAngle: prevAngle, - endAngle: -coord * RADIAN, - clockwise: clockwise - }, - silent: true - })); - prevAngle = -coord * RADIAN; - } // Simple optimization - // Batching the lines if color are the same - - - for (var i = 0; i < splitAreas.length; i++) { - group.add(mergePath$1(splitAreas[i], { - style: defaults({ - fill: areaColors[i % areaColors.length] - }, areaStyleModel.getAreaStyle()), - silent: true - })); - } - } - }; - - var axisBuilderAttrs$2 = ['axisLine', 'axisTickLabel', 'axisName']; - var selfBuilderAttrs$1 = ['splitLine', 'splitArea', 'minorSplitLine']; - - var RadiusAxisView = - /** @class */ - function (_super) { - __extends(RadiusAxisView, _super); - - function RadiusAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = RadiusAxisView.type; - _this.axisPointerClass = 'PolarAxisPointer'; - return _this; - } - - RadiusAxisView.prototype.render = function (radiusAxisModel, ecModel) { - this.group.removeAll(); - - if (!radiusAxisModel.get('show')) { - return; - } - - var oldAxisGroup = this._axisGroup; - var newAxisGroup = this._axisGroup = new Group(); - this.group.add(newAxisGroup); - var radiusAxis = radiusAxisModel.axis; - var polar = radiusAxis.polar; - var angleAxis = polar.getAngleAxis(); - var ticksCoords = radiusAxis.getTicksCoords(); - var minorTicksCoords = radiusAxis.getMinorTicksCoords(); - var axisAngle = angleAxis.getExtent()[0]; - var radiusExtent = radiusAxis.getExtent(); - var layout = layoutAxis(polar, radiusAxisModel, axisAngle); - var axisBuilder = new AxisBuilder(radiusAxisModel, layout); - each(axisBuilderAttrs$2, axisBuilder.add, axisBuilder); - newAxisGroup.add(axisBuilder.getGroup()); - groupTransition(oldAxisGroup, newAxisGroup, radiusAxisModel); - each(selfBuilderAttrs$1, function (name) { - if (radiusAxisModel.get([name, 'show']) && !radiusAxis.scale.isBlank()) { - axisElementBuilders$1[name](this.group, radiusAxisModel, polar, axisAngle, radiusExtent, ticksCoords, minorTicksCoords); - } - }, this); - }; - - RadiusAxisView.type = 'radiusAxis'; - return RadiusAxisView; - }(AxisView); - - var axisElementBuilders$1 = { - splitLine: function (group, radiusAxisModel, polar, axisAngle, radiusExtent, ticksCoords) { - var splitLineModel = radiusAxisModel.getModel('splitLine'); - var lineStyleModel = splitLineModel.getModel('lineStyle'); - var lineColors = lineStyleModel.get('color'); - var lineCount = 0; - lineColors = lineColors instanceof Array ? lineColors : [lineColors]; - var splitLines = []; - - for (var i = 0; i < ticksCoords.length; i++) { - var colorIndex = lineCount++ % lineColors.length; - splitLines[colorIndex] = splitLines[colorIndex] || []; - splitLines[colorIndex].push(new Circle({ - shape: { - cx: polar.cx, - cy: polar.cy, - // ensure circle radius >= 0 - r: Math.max(ticksCoords[i].coord, 0) - } - })); - } // Simple optimization - // Batching the lines if color are the same - - - for (var i = 0; i < splitLines.length; i++) { - group.add(mergePath$1(splitLines[i], { - style: defaults({ - stroke: lineColors[i % lineColors.length], - fill: null - }, lineStyleModel.getLineStyle()), - silent: true - })); - } - }, - minorSplitLine: function (group, radiusAxisModel, polar, axisAngle, radiusExtent, ticksCoords, minorTicksCoords) { - if (!minorTicksCoords.length) { - return; - } - - var minorSplitLineModel = radiusAxisModel.getModel('minorSplitLine'); - var lineStyleModel = minorSplitLineModel.getModel('lineStyle'); - var lines = []; - - for (var i = 0; i < minorTicksCoords.length; i++) { - for (var k = 0; k < minorTicksCoords[i].length; k++) { - lines.push(new Circle({ - shape: { - cx: polar.cx, - cy: polar.cy, - r: minorTicksCoords[i][k].coord - } - })); - } - } - - group.add(mergePath$1(lines, { - style: defaults({ - fill: null - }, lineStyleModel.getLineStyle()), - silent: true - })); - }, - splitArea: function (group, radiusAxisModel, polar, axisAngle, radiusExtent, ticksCoords) { - if (!ticksCoords.length) { - return; - } - - var splitAreaModel = radiusAxisModel.getModel('splitArea'); - var areaStyleModel = splitAreaModel.getModel('areaStyle'); - var areaColors = areaStyleModel.get('color'); - var lineCount = 0; - areaColors = areaColors instanceof Array ? areaColors : [areaColors]; - var splitAreas = []; - var prevRadius = ticksCoords[0].coord; - - for (var i = 1; i < ticksCoords.length; i++) { - var colorIndex = lineCount++ % areaColors.length; - splitAreas[colorIndex] = splitAreas[colorIndex] || []; - splitAreas[colorIndex].push(new Sector({ - shape: { - cx: polar.cx, - cy: polar.cy, - r0: prevRadius, - r: ticksCoords[i].coord, - startAngle: 0, - endAngle: Math.PI * 2 - }, - silent: true - })); - prevRadius = ticksCoords[i].coord; - } // Simple optimization - // Batching the lines if color are the same - - - for (var i = 0; i < splitAreas.length; i++) { - group.add(mergePath$1(splitAreas[i], { - style: defaults({ - fill: areaColors[i % areaColors.length] - }, areaStyleModel.getAreaStyle()), - silent: true - })); - } - } - }; - /** - * @inner - */ - - function layoutAxis(polar, radiusAxisModel, axisAngle) { - return { - position: [polar.cx, polar.cy], - rotation: axisAngle / 180 * Math.PI, - labelDirection: -1, - tickDirection: -1, - nameDirection: 1, - labelRotate: radiusAxisModel.getModel('axisLabel').get('rotate'), - // Over splitLine and splitArea - z2: 1 - }; - } - - function getSeriesStackId$1(seriesModel) { - return seriesModel.get('stack') || '__ec_stack_' + seriesModel.seriesIndex; - } - - function getAxisKey$1(polar, axis) { - return axis.dim + polar.model.componentIndex; - } - - function barLayoutPolar(seriesType, ecModel, api) { - var lastStackCoords = {}; - var barWidthAndOffset = calRadialBar(filter(ecModel.getSeriesByType(seriesType), function (seriesModel) { - return !ecModel.isSeriesFiltered(seriesModel) && seriesModel.coordinateSystem && seriesModel.coordinateSystem.type === 'polar'; - })); - ecModel.eachSeriesByType(seriesType, function (seriesModel) { - // Check series coordinate, do layout for polar only - if (seriesModel.coordinateSystem.type !== 'polar') { - return; - } - - var data = seriesModel.getData(); - var polar = seriesModel.coordinateSystem; - var baseAxis = polar.getBaseAxis(); - var axisKey = getAxisKey$1(polar, baseAxis); - var stackId = getSeriesStackId$1(seriesModel); - var columnLayoutInfo = barWidthAndOffset[axisKey][stackId]; - var columnOffset = columnLayoutInfo.offset; - var columnWidth = columnLayoutInfo.width; - var valueAxis = polar.getOtherAxis(baseAxis); - var cx = seriesModel.coordinateSystem.cx; - var cy = seriesModel.coordinateSystem.cy; - var barMinHeight = seriesModel.get('barMinHeight') || 0; - var barMinAngle = seriesModel.get('barMinAngle') || 0; - lastStackCoords[stackId] = lastStackCoords[stackId] || []; - var valueDim = data.mapDimension(valueAxis.dim); - var baseDim = data.mapDimension(baseAxis.dim); - var stacked = isDimensionStacked(data, valueDim - /* , baseDim */ - ); - var clampLayout = baseAxis.dim !== 'radius' || !seriesModel.get('roundCap', true); - var valueAxisStart = valueAxis.dataToCoord(0); - - for (var idx = 0, len = data.count(); idx < len; idx++) { - var value = data.get(valueDim, idx); - var baseValue = data.get(baseDim, idx); - var sign = value >= 0 ? 'p' : 'n'; - var baseCoord = valueAxisStart; // Because of the barMinHeight, we can not use the value in - // stackResultDimension directly. - // Only ordinal axis can be stacked. - - if (stacked) { - if (!lastStackCoords[stackId][baseValue]) { - lastStackCoords[stackId][baseValue] = { - p: valueAxisStart, - n: valueAxisStart // Negative stack - - }; - } // Should also consider #4243 - - - baseCoord = lastStackCoords[stackId][baseValue][sign]; - } - - var r0 = void 0; - var r = void 0; - var startAngle = void 0; - var endAngle = void 0; // radial sector - - if (valueAxis.dim === 'radius') { - var radiusSpan = valueAxis.dataToCoord(value) - valueAxisStart; - var angle = baseAxis.dataToCoord(baseValue); - - if (Math.abs(radiusSpan) < barMinHeight) { - radiusSpan = (radiusSpan < 0 ? -1 : 1) * barMinHeight; - } - - r0 = baseCoord; - r = baseCoord + radiusSpan; - startAngle = angle - columnOffset; - endAngle = startAngle - columnWidth; - stacked && (lastStackCoords[stackId][baseValue][sign] = r); - } // tangential sector - else { - var angleSpan = valueAxis.dataToCoord(value, clampLayout) - valueAxisStart; - var radius = baseAxis.dataToCoord(baseValue); - - if (Math.abs(angleSpan) < barMinAngle) { - angleSpan = (angleSpan < 0 ? -1 : 1) * barMinAngle; - } - - r0 = radius + columnOffset; - r = r0 + columnWidth; - startAngle = baseCoord; - endAngle = baseCoord + angleSpan; // if the previous stack is at the end of the ring, - // add a round to differentiate it from origin - // let extent = angleAxis.getExtent(); - // let stackCoord = angle; - // if (stackCoord === extent[0] && value > 0) { - // stackCoord = extent[1]; - // } - // else if (stackCoord === extent[1] && value < 0) { - // stackCoord = extent[0]; - // } - - stacked && (lastStackCoords[stackId][baseValue][sign] = endAngle); - } - - data.setItemLayout(idx, { - cx: cx, - cy: cy, - r0: r0, - r: r, - // Consider that positive angle is anti-clockwise, - // while positive radian of sector is clockwise - startAngle: -startAngle * Math.PI / 180, - endAngle: -endAngle * Math.PI / 180, - - /** - * Keep the same logic with bar in catesion: use end value to - * control direction. Notice that if clockwise is true (by - * default), the sector will always draw clockwisely, no matter - * whether endAngle is greater or less than startAngle. - */ - clockwise: startAngle >= endAngle - }); - } - }); - } - /** - * Calculate bar width and offset for radial bar charts - */ - - - function calRadialBar(barSeries) { - // Columns info on each category axis. Key is polar name - var columnsMap = {}; - each(barSeries, function (seriesModel, idx) { - var data = seriesModel.getData(); - var polar = seriesModel.coordinateSystem; - var baseAxis = polar.getBaseAxis(); - var axisKey = getAxisKey$1(polar, baseAxis); - var axisExtent = baseAxis.getExtent(); - var bandWidth = baseAxis.type === 'category' ? baseAxis.getBandWidth() : Math.abs(axisExtent[1] - axisExtent[0]) / data.count(); - var columnsOnAxis = columnsMap[axisKey] || { - bandWidth: bandWidth, - remainedWidth: bandWidth, - autoWidthCount: 0, - categoryGap: '20%', - gap: '30%', - stacks: {} - }; - var stacks = columnsOnAxis.stacks; - columnsMap[axisKey] = columnsOnAxis; - var stackId = getSeriesStackId$1(seriesModel); - - if (!stacks[stackId]) { - columnsOnAxis.autoWidthCount++; - } - - stacks[stackId] = stacks[stackId] || { - width: 0, - maxWidth: 0 - }; - var barWidth = parsePercent$1(seriesModel.get('barWidth'), bandWidth); - var barMaxWidth = parsePercent$1(seriesModel.get('barMaxWidth'), bandWidth); - var barGap = seriesModel.get('barGap'); - var barCategoryGap = seriesModel.get('barCategoryGap'); - - if (barWidth && !stacks[stackId].width) { - barWidth = Math.min(columnsOnAxis.remainedWidth, barWidth); - stacks[stackId].width = barWidth; - columnsOnAxis.remainedWidth -= barWidth; - } - - barMaxWidth && (stacks[stackId].maxWidth = barMaxWidth); - barGap != null && (columnsOnAxis.gap = barGap); - barCategoryGap != null && (columnsOnAxis.categoryGap = barCategoryGap); - }); - var result = {}; - each(columnsMap, function (columnsOnAxis, coordSysName) { - result[coordSysName] = {}; - var stacks = columnsOnAxis.stacks; - var bandWidth = columnsOnAxis.bandWidth; - var categoryGap = parsePercent$1(columnsOnAxis.categoryGap, bandWidth); - var barGapPercent = parsePercent$1(columnsOnAxis.gap, 1); - var remainedWidth = columnsOnAxis.remainedWidth; - var autoWidthCount = columnsOnAxis.autoWidthCount; - var autoWidth = (remainedWidth - categoryGap) / (autoWidthCount + (autoWidthCount - 1) * barGapPercent); - autoWidth = Math.max(autoWidth, 0); // Find if any auto calculated bar exceeded maxBarWidth - - each(stacks, function (column, stack) { - var maxWidth = column.maxWidth; - - if (maxWidth && maxWidth < autoWidth) { - maxWidth = Math.min(maxWidth, remainedWidth); - - if (column.width) { - maxWidth = Math.min(maxWidth, column.width); - } - - remainedWidth -= maxWidth; - column.width = maxWidth; - autoWidthCount--; - } - }); // Recalculate width again - - autoWidth = (remainedWidth - categoryGap) / (autoWidthCount + (autoWidthCount - 1) * barGapPercent); - autoWidth = Math.max(autoWidth, 0); - var widthSum = 0; - var lastColumn; - each(stacks, function (column, idx) { - if (!column.width) { - column.width = autoWidth; - } - - lastColumn = column; - widthSum += column.width * (1 + barGapPercent); - }); - - if (lastColumn) { - widthSum -= lastColumn.width * barGapPercent; - } - - var offset = -widthSum / 2; - each(stacks, function (column, stackId) { - result[coordSysName][stackId] = result[coordSysName][stackId] || { - offset: offset, - width: column.width - }; - offset += column.width * (1 + barGapPercent); - }); - }); - return result; - } - - var angleAxisExtraOption = { - startAngle: 90, - clockwise: true, - splitNumber: 12, - axisLabel: { - rotate: 0 - } - }; - var radiusAxisExtraOption = { - splitNumber: 5 - }; - - var PolarView = - /** @class */ - function (_super) { - __extends(PolarView, _super); - - function PolarView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = PolarView.type; - return _this; - } - - PolarView.type = 'polar'; - return PolarView; - }(ComponentView); - - function install$u(registers) { - use(install$s); - AxisView.registerAxisPointerClass('PolarAxisPointer', PolarAxisPointer); - registers.registerCoordinateSystem('polar', polarCreator); - registers.registerComponentModel(PolarModel); - registers.registerComponentView(PolarView); // Model and view for angleAxis and radiusAxis - - axisModelCreator(registers, 'angle', AngleAxisModel, angleAxisExtraOption); - axisModelCreator(registers, 'radius', RadiusAxisModel, radiusAxisExtraOption); - registers.registerComponentView(AngleAxisView); - registers.registerComponentView(RadiusAxisView); - registers.registerLayout(curry(barLayoutPolar, 'bar')); - } - - function layout$2(axisModel, opt) { - opt = opt || {}; - var single = axisModel.coordinateSystem; - var axis = axisModel.axis; - var layout = {}; - var axisPosition = axis.position; - var orient = axis.orient; - var rect = single.getRect(); - var rectBound = [rect.x, rect.x + rect.width, rect.y, rect.y + rect.height]; - var positionMap = { - horizontal: { - top: rectBound[2], - bottom: rectBound[3] - }, - vertical: { - left: rectBound[0], - right: rectBound[1] - } - }; - layout.position = [orient === 'vertical' ? positionMap.vertical[axisPosition] : rectBound[0], orient === 'horizontal' ? positionMap.horizontal[axisPosition] : rectBound[3]]; - var r = { - horizontal: 0, - vertical: 1 - }; - layout.rotation = Math.PI / 2 * r[orient]; - var directionMap = { - top: -1, - bottom: 1, - right: 1, - left: -1 - }; - layout.labelDirection = layout.tickDirection = layout.nameDirection = directionMap[axisPosition]; - - if (axisModel.get(['axisTick', 'inside'])) { - layout.tickDirection = -layout.tickDirection; - } - - if (retrieve(opt.labelInside, axisModel.get(['axisLabel', 'inside']))) { - layout.labelDirection = -layout.labelDirection; - } - - var labelRotation = opt.rotate; - labelRotation == null && (labelRotation = axisModel.get(['axisLabel', 'rotate'])); - layout.labelRotation = axisPosition === 'top' ? -labelRotation : labelRotation; - layout.z2 = 1; - return layout; - } - - var axisBuilderAttrs$3 = ['axisLine', 'axisTickLabel', 'axisName']; - var selfBuilderAttrs$2 = ['splitArea', 'splitLine']; - - var SingleAxisView = - /** @class */ - function (_super) { - __extends(SingleAxisView, _super); - - function SingleAxisView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SingleAxisView.type; - _this.axisPointerClass = 'SingleAxisPointer'; - return _this; - } - - SingleAxisView.prototype.render = function (axisModel, ecModel, api, payload) { - var group = this.group; - group.removeAll(); - var oldAxisGroup = this._axisGroup; - this._axisGroup = new Group(); - var layout = layout$2(axisModel); - var axisBuilder = new AxisBuilder(axisModel, layout); - each(axisBuilderAttrs$3, axisBuilder.add, axisBuilder); - group.add(this._axisGroup); - group.add(axisBuilder.getGroup()); - each(selfBuilderAttrs$2, function (name) { - if (axisModel.get([name, 'show'])) { - axisElementBuilders$2[name](this, this.group, this._axisGroup, axisModel); - } - }, this); - groupTransition(oldAxisGroup, this._axisGroup, axisModel); - - _super.prototype.render.call(this, axisModel, ecModel, api, payload); - }; - - SingleAxisView.prototype.remove = function () { - rectCoordAxisHandleRemove(this); - }; - - SingleAxisView.type = 'singleAxis'; - return SingleAxisView; - }(AxisView); - - var axisElementBuilders$2 = { - splitLine: function (axisView, group, axisGroup, axisModel) { - var axis = axisModel.axis; - - if (axis.scale.isBlank()) { - return; - } - - var splitLineModel = axisModel.getModel('splitLine'); - var lineStyleModel = splitLineModel.getModel('lineStyle'); - var lineColors = lineStyleModel.get('color'); - lineColors = lineColors instanceof Array ? lineColors : [lineColors]; - var lineWidth = lineStyleModel.get('width'); - var gridRect = axisModel.coordinateSystem.getRect(); - var isHorizontal = axis.isHorizontal(); - var splitLines = []; - var lineCount = 0; - var ticksCoords = axis.getTicksCoords({ - tickModel: splitLineModel - }); - var p1 = []; - var p2 = []; - - for (var i = 0; i < ticksCoords.length; ++i) { - var tickCoord = axis.toGlobalCoord(ticksCoords[i].coord); - - if (isHorizontal) { - p1[0] = tickCoord; - p1[1] = gridRect.y; - p2[0] = tickCoord; - p2[1] = gridRect.y + gridRect.height; - } else { - p1[0] = gridRect.x; - p1[1] = tickCoord; - p2[0] = gridRect.x + gridRect.width; - p2[1] = tickCoord; - } - - var line = new Line({ - shape: { - x1: p1[0], - y1: p1[1], - x2: p2[0], - y2: p2[1] - }, - silent: true - }); - subPixelOptimizeLine$1(line.shape, lineWidth); - var colorIndex = lineCount++ % lineColors.length; - splitLines[colorIndex] = splitLines[colorIndex] || []; - splitLines[colorIndex].push(line); - } - - var lineStyle = lineStyleModel.getLineStyle(['color']); - - for (var i = 0; i < splitLines.length; ++i) { - group.add(mergePath$1(splitLines[i], { - style: defaults({ - stroke: lineColors[i % lineColors.length] - }, lineStyle), - silent: true - })); - } - }, - splitArea: function (axisView, group, axisGroup, axisModel) { - rectCoordAxisBuildSplitArea(axisView, axisGroup, axisModel, axisModel); - } - }; - - var SingleAxisModel = - /** @class */ - function (_super) { - __extends(SingleAxisModel, _super); - - function SingleAxisModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SingleAxisModel.type; - return _this; - } - - SingleAxisModel.prototype.getCoordSysModel = function () { - return this; - }; - - SingleAxisModel.type = 'singleAxis'; - SingleAxisModel.layoutMode = 'box'; - SingleAxisModel.defaultOption = { - left: '5%', - top: '5%', - right: '5%', - bottom: '5%', - type: 'value', - position: 'bottom', - orient: 'horizontal', - axisLine: { - show: true, - lineStyle: { - width: 1, - type: 'solid' - } - }, - // Single coordinate system and single axis is the, - // which is used as the parent tooltip model. - // same model, so we set default tooltip show as true. - tooltip: { - show: true - }, - axisTick: { - show: true, - length: 6, - lineStyle: { - width: 1 - } - }, - axisLabel: { - show: true, - interval: 'auto' - }, - splitLine: { - show: true, - lineStyle: { - type: 'dashed', - opacity: 0.2 - } - } - }; - return SingleAxisModel; - }(ComponentModel); - - mixin(SingleAxisModel, AxisModelCommonMixin.prototype); - - var SingleAxis = - /** @class */ - function (_super) { - __extends(SingleAxis, _super); - - function SingleAxis(dim, scale, coordExtent, axisType, position) { - var _this = _super.call(this, dim, scale, coordExtent) || this; - - _this.type = axisType || 'value'; - _this.position = position || 'bottom'; - return _this; - } - /** - * Judge the orient of the axis. - */ - - - SingleAxis.prototype.isHorizontal = function () { - var position = this.position; - return position === 'top' || position === 'bottom'; - }; - - SingleAxis.prototype.pointToData = function (point, clamp) { - return this.coordinateSystem.pointToData(point)[0]; - }; - - return SingleAxis; - }(Axis); - - var singleDimensions = ['single']; - /** - * Create a single coordinates system. - */ - - var Single = - /** @class */ - function () { - function Single(axisModel, ecModel, api) { - this.type = 'single'; - this.dimension = 'single'; - /** - * Add it just for draw tooltip. - */ - - this.dimensions = singleDimensions; - this.axisPointerEnabled = true; - this.model = axisModel; - - this._init(axisModel, ecModel, api); - } - /** - * Initialize single coordinate system. - */ - - - Single.prototype._init = function (axisModel, ecModel, api) { - var dim = this.dimension; - var axis = new SingleAxis(dim, createScaleByModel(axisModel), [0, 0], axisModel.get('type'), axisModel.get('position')); - var isCategory = axis.type === 'category'; - axis.onBand = isCategory && axisModel.get('boundaryGap'); - axis.inverse = axisModel.get('inverse'); - axis.orient = axisModel.get('orient'); - axisModel.axis = axis; - axis.model = axisModel; - axis.coordinateSystem = this; - this._axis = axis; - }; - /** - * Update axis scale after data processed - */ - - - Single.prototype.update = function (ecModel, api) { - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.coordinateSystem === this) { - var data_1 = seriesModel.getData(); - each(data_1.mapDimensionsAll(this.dimension), function (dim) { - this._axis.scale.unionExtentFromData(data_1, dim); - }, this); - niceScaleExtent(this._axis.scale, this._axis.model); - } - }, this); - }; - /** - * Resize the single coordinate system. - */ - - - Single.prototype.resize = function (axisModel, api) { - this._rect = getLayoutRect({ - left: axisModel.get('left'), - top: axisModel.get('top'), - right: axisModel.get('right'), - bottom: axisModel.get('bottom'), - width: axisModel.get('width'), - height: axisModel.get('height') - }, { - width: api.getWidth(), - height: api.getHeight() - }); - - this._adjustAxis(); - }; - - Single.prototype.getRect = function () { - return this._rect; - }; - - Single.prototype._adjustAxis = function () { - var rect = this._rect; - var axis = this._axis; - var isHorizontal = axis.isHorizontal(); - var extent = isHorizontal ? [0, rect.width] : [0, rect.height]; - var idx = axis.inverse ? 1 : 0; - axis.setExtent(extent[idx], extent[1 - idx]); - - this._updateAxisTransform(axis, isHorizontal ? rect.x : rect.y); - }; - - Single.prototype._updateAxisTransform = function (axis, coordBase) { - var axisExtent = axis.getExtent(); - var extentSum = axisExtent[0] + axisExtent[1]; - var isHorizontal = axis.isHorizontal(); - axis.toGlobalCoord = isHorizontal ? function (coord) { - return coord + coordBase; - } : function (coord) { - return extentSum - coord + coordBase; - }; - axis.toLocalCoord = isHorizontal ? function (coord) { - return coord - coordBase; - } : function (coord) { - return extentSum - coord + coordBase; - }; - }; - /** - * Get axis. - */ - - - Single.prototype.getAxis = function () { - return this._axis; - }; - /** - * Get axis, add it just for draw tooltip. - */ - - - Single.prototype.getBaseAxis = function () { - return this._axis; - }; - - Single.prototype.getAxes = function () { - return [this._axis]; - }; - - Single.prototype.getTooltipAxes = function () { - return { - baseAxes: [this.getAxis()], - // Empty otherAxes - otherAxes: [] - }; - }; - /** - * If contain point. - */ - - - Single.prototype.containPoint = function (point) { - var rect = this.getRect(); - var axis = this.getAxis(); - var orient = axis.orient; - - if (orient === 'horizontal') { - return axis.contain(axis.toLocalCoord(point[0])) && point[1] >= rect.y && point[1] <= rect.y + rect.height; - } else { - return axis.contain(axis.toLocalCoord(point[1])) && point[0] >= rect.y && point[0] <= rect.y + rect.height; - } - }; - - Single.prototype.pointToData = function (point) { - var axis = this.getAxis(); - return [axis.coordToData(axis.toLocalCoord(point[axis.orient === 'horizontal' ? 0 : 1]))]; - }; - /** - * Convert the series data to concrete point. - * Can be [val] | val - */ - - - Single.prototype.dataToPoint = function (val) { - var axis = this.getAxis(); - var rect = this.getRect(); - var pt = []; - var idx = axis.orient === 'horizontal' ? 0 : 1; - - if (val instanceof Array) { - val = val[0]; - } - - pt[idx] = axis.toGlobalCoord(axis.dataToCoord(+val)); - pt[1 - idx] = idx === 0 ? rect.y + rect.height / 2 : rect.x + rect.width / 2; - return pt; - }; - - Single.prototype.convertToPixel = function (ecModel, finder, value) { - var coordSys = getCoordSys$3(finder); - return coordSys === this ? this.dataToPoint(value) : null; - }; - - Single.prototype.convertFromPixel = function (ecModel, finder, pixel) { - var coordSys = getCoordSys$3(finder); - return coordSys === this ? this.pointToData(pixel) : null; - }; - - return Single; - }(); - - function getCoordSys$3(finder) { - var seriesModel = finder.seriesModel; - var singleModel = finder.singleAxisModel; - return singleModel && singleModel.coordinateSystem || seriesModel && seriesModel.coordinateSystem; - } - - /** - * Create single coordinate system and inject it into seriesModel. - */ - - function create$2(ecModel, api) { - var singles = []; - ecModel.eachComponent('singleAxis', function (axisModel, idx) { - var single = new Single(axisModel, ecModel, api); - single.name = 'single_' + idx; - single.resize(axisModel, api); - axisModel.coordinateSystem = single; - singles.push(single); - }); - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.get('coordinateSystem') === 'singleAxis') { - var singleAxisModel = seriesModel.getReferringComponents('singleAxis', SINGLE_REFERRING).models[0]; - seriesModel.coordinateSystem = singleAxisModel && singleAxisModel.coordinateSystem; - } - }); - return singles; - } - - var singleCreator = { - create: create$2, - dimensions: singleDimensions - }; - - var XY = ['x', 'y']; - var WH = ['width', 'height']; - - var SingleAxisPointer = - /** @class */ - function (_super) { - __extends(SingleAxisPointer, _super); - - function SingleAxisPointer() { - return _super !== null && _super.apply(this, arguments) || this; - } - /** - * @override - */ - - - SingleAxisPointer.prototype.makeElOption = function (elOption, value, axisModel, axisPointerModel, api) { - var axis = axisModel.axis; - var coordSys = axis.coordinateSystem; - var otherExtent = getGlobalExtent(coordSys, 1 - getPointDimIndex(axis)); - var pixelValue = coordSys.dataToPoint(value)[0]; - var axisPointerType = axisPointerModel.get('type'); - - if (axisPointerType && axisPointerType !== 'none') { - var elStyle = buildElStyle(axisPointerModel); - var pointerOption = pointerShapeBuilder$2[axisPointerType](axis, pixelValue, otherExtent); - pointerOption.style = elStyle; - elOption.graphicKey = pointerOption.type; - elOption.pointer = pointerOption; - } - - var layoutInfo = layout$2(axisModel); - buildCartesianSingleLabelElOption( // @ts-ignore - value, elOption, layoutInfo, axisModel, axisPointerModel, api); - }; - /** - * @override - */ - - - SingleAxisPointer.prototype.getHandleTransform = function (value, axisModel, axisPointerModel) { - var layoutInfo = layout$2(axisModel, { - labelInside: false - }); // @ts-ignore - - layoutInfo.labelMargin = axisPointerModel.get(['handle', 'margin']); - var position = getTransformedPosition(axisModel.axis, value, layoutInfo); - return { - x: position[0], - y: position[1], - rotation: layoutInfo.rotation + (layoutInfo.labelDirection < 0 ? Math.PI : 0) - }; - }; - /** - * @override - */ - - - SingleAxisPointer.prototype.updateHandleTransform = function (transform, delta, axisModel, axisPointerModel) { - var axis = axisModel.axis; - var coordSys = axis.coordinateSystem; - var dimIndex = getPointDimIndex(axis); - var axisExtent = getGlobalExtent(coordSys, dimIndex); - var currPosition = [transform.x, transform.y]; - currPosition[dimIndex] += delta[dimIndex]; - currPosition[dimIndex] = Math.min(axisExtent[1], currPosition[dimIndex]); - currPosition[dimIndex] = Math.max(axisExtent[0], currPosition[dimIndex]); - var otherExtent = getGlobalExtent(coordSys, 1 - dimIndex); - var cursorOtherValue = (otherExtent[1] + otherExtent[0]) / 2; - var cursorPoint = [cursorOtherValue, cursorOtherValue]; - cursorPoint[dimIndex] = currPosition[dimIndex]; - return { - x: currPosition[0], - y: currPosition[1], - rotation: transform.rotation, - cursorPoint: cursorPoint, - tooltipOption: { - verticalAlign: 'middle' - } - }; - }; - - return SingleAxisPointer; - }(BaseAxisPointer); - - var pointerShapeBuilder$2 = { - line: function (axis, pixelValue, otherExtent) { - var targetShape = makeLineShape([pixelValue, otherExtent[0]], [pixelValue, otherExtent[1]], getPointDimIndex(axis)); - return { - type: 'Line', - subPixelOptimize: true, - shape: targetShape - }; - }, - shadow: function (axis, pixelValue, otherExtent) { - var bandWidth = axis.getBandWidth(); - var span = otherExtent[1] - otherExtent[0]; - return { - type: 'Rect', - shape: makeRectShape([pixelValue - bandWidth / 2, otherExtent[0]], [bandWidth, span], getPointDimIndex(axis)) - }; - } - }; - - function getPointDimIndex(axis) { - return axis.isHorizontal() ? 0 : 1; - } - - function getGlobalExtent(coordSys, dimIndex) { - var rect = coordSys.getRect(); - return [rect[XY[dimIndex]], rect[XY[dimIndex]] + rect[WH[dimIndex]]]; - } - - var SingleView = - /** @class */ - function (_super) { - __extends(SingleView, _super); - - function SingleView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SingleView.type; - return _this; - } - - SingleView.type = 'single'; - return SingleView; - }(ComponentView); - - function install$v(registers) { - use(install$s); - AxisView.registerAxisPointerClass('SingleAxisPointer', SingleAxisPointer); - registers.registerComponentView(SingleView); // Axis - - registers.registerComponentView(SingleAxisView); - registers.registerComponentModel(SingleAxisModel); - axisModelCreator(registers, 'single', SingleAxisModel, SingleAxisModel.defaultOption); - registers.registerCoordinateSystem('single', singleCreator); - } - - var CalendarModel = - /** @class */ - function (_super) { - __extends(CalendarModel, _super); - - function CalendarModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CalendarModel.type; - return _this; - } - /** - * @override - */ - - - CalendarModel.prototype.init = function (option, parentModel, ecModel) { - var inputPositionParams = getLayoutParams(option); - - _super.prototype.init.apply(this, arguments); - - mergeAndNormalizeLayoutParams(option, inputPositionParams); - }; - /** - * @override - */ - - - CalendarModel.prototype.mergeOption = function (option) { - _super.prototype.mergeOption.apply(this, arguments); - - mergeAndNormalizeLayoutParams(this.option, option); - }; - - CalendarModel.prototype.getCellSize = function () { - // Has been normalized - return this.option.cellSize; - }; - - CalendarModel.type = 'calendar'; - CalendarModel.defaultOption = { - // zlevel: 0, - z: 2, - left: 80, - top: 60, - cellSize: 20, - // horizontal vertical - orient: 'horizontal', - // month separate line style - splitLine: { - show: true, - lineStyle: { - color: '#000', - width: 1, - type: 'solid' - } - }, - // rect style temporarily unused emphasis - itemStyle: { - color: '#fff', - borderWidth: 1, - borderColor: '#ccc' - }, - // week text style - dayLabel: { - show: true, - firstDay: 0, - // start end - position: 'start', - margin: '50%', - color: '#000' - }, - // month text style - monthLabel: { - show: true, - // start end - position: 'start', - margin: 5, - // center or left - align: 'center', - formatter: null, - color: '#000' - }, - // year text style - yearLabel: { - show: true, - // top bottom left right - position: null, - margin: 30, - formatter: null, - color: '#ccc', - fontFamily: 'sans-serif', - fontWeight: 'bolder', - fontSize: 20 - } - }; - return CalendarModel; - }(ComponentModel); - - function mergeAndNormalizeLayoutParams(target, raw) { - // Normalize cellSize - var cellSize = target.cellSize; - var cellSizeArr; - - if (!isArray(cellSize)) { - cellSizeArr = target.cellSize = [cellSize, cellSize]; - } else { - cellSizeArr = cellSize; - } - - if (cellSizeArr.length === 1) { - cellSizeArr[1] = cellSizeArr[0]; - } - - var ignoreSize = map([0, 1], function (hvIdx) { - // If user have set `width` or both `left` and `right`, cellSizeArr - // will be automatically set to 'auto', otherwise the default - // setting of cellSizeArr will make `width` setting not work. - if (sizeCalculable(raw, hvIdx)) { - cellSizeArr[hvIdx] = 'auto'; - } - - return cellSizeArr[hvIdx] != null && cellSizeArr[hvIdx] !== 'auto'; - }); - mergeLayoutParam(target, raw, { - type: 'box', - ignoreSize: ignoreSize - }); - } - - var CalendarView = - /** @class */ - function (_super) { - __extends(CalendarView, _super); - - function CalendarView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = CalendarView.type; - return _this; - } - - CalendarView.prototype.render = function (calendarModel, ecModel, api) { - var group = this.group; - group.removeAll(); - var coordSys = calendarModel.coordinateSystem; // range info - - var rangeData = coordSys.getRangeInfo(); - var orient = coordSys.getOrient(); // locale - - var localeModel = ecModel.getLocaleModel(); - - this._renderDayRect(calendarModel, rangeData, group); // _renderLines must be called prior to following function - - - this._renderLines(calendarModel, rangeData, orient, group); - - this._renderYearText(calendarModel, rangeData, orient, group); - - this._renderMonthText(calendarModel, localeModel, orient, group); - - this._renderWeekText(calendarModel, localeModel, rangeData, orient, group); - }; // render day rect - - - CalendarView.prototype._renderDayRect = function (calendarModel, rangeData, group) { - var coordSys = calendarModel.coordinateSystem; - var itemRectStyleModel = calendarModel.getModel('itemStyle').getItemStyle(); - var sw = coordSys.getCellWidth(); - var sh = coordSys.getCellHeight(); - - for (var i = rangeData.start.time; i <= rangeData.end.time; i = coordSys.getNextNDay(i, 1).time) { - var point = coordSys.dataToRect([i], false).tl; // every rect - - var rect = new Rect({ - shape: { - x: point[0], - y: point[1], - width: sw, - height: sh - }, - cursor: 'default', - style: itemRectStyleModel - }); - group.add(rect); - } - }; // render separate line - - - CalendarView.prototype._renderLines = function (calendarModel, rangeData, orient, group) { - var self = this; - var coordSys = calendarModel.coordinateSystem; - var lineStyleModel = calendarModel.getModel(['splitLine', 'lineStyle']).getLineStyle(); - var show = calendarModel.get(['splitLine', 'show']); - var lineWidth = lineStyleModel.lineWidth; - this._tlpoints = []; - this._blpoints = []; - this._firstDayOfMonth = []; - this._firstDayPoints = []; - var firstDay = rangeData.start; - - for (var i = 0; firstDay.time <= rangeData.end.time; i++) { - addPoints(firstDay.formatedDate); - - if (i === 0) { - firstDay = coordSys.getDateInfo(rangeData.start.y + '-' + rangeData.start.m); - } - - var date = firstDay.date; - date.setMonth(date.getMonth() + 1); - firstDay = coordSys.getDateInfo(date); - } - - addPoints(coordSys.getNextNDay(rangeData.end.time, 1).formatedDate); - - function addPoints(date) { - self._firstDayOfMonth.push(coordSys.getDateInfo(date)); - - self._firstDayPoints.push(coordSys.dataToRect([date], false).tl); - - var points = self._getLinePointsOfOneWeek(calendarModel, date, orient); - - self._tlpoints.push(points[0]); - - self._blpoints.push(points[points.length - 1]); - - show && self._drawSplitline(points, lineStyleModel, group); - } // render top/left line - - - show && this._drawSplitline(self._getEdgesPoints(self._tlpoints, lineWidth, orient), lineStyleModel, group); // render bottom/right line - - show && this._drawSplitline(self._getEdgesPoints(self._blpoints, lineWidth, orient), lineStyleModel, group); - }; // get points at both ends - - - CalendarView.prototype._getEdgesPoints = function (points, lineWidth, orient) { - var rs = [points[0].slice(), points[points.length - 1].slice()]; - var idx = orient === 'horizontal' ? 0 : 1; // both ends of the line are extend half lineWidth - - rs[0][idx] = rs[0][idx] - lineWidth / 2; - rs[1][idx] = rs[1][idx] + lineWidth / 2; - return rs; - }; // render split line - - - CalendarView.prototype._drawSplitline = function (points, lineStyle, group) { - var poyline = new Polyline({ - z2: 20, - shape: { - points: points - }, - style: lineStyle - }); - group.add(poyline); - }; // render month line of one week points - - - CalendarView.prototype._getLinePointsOfOneWeek = function (calendarModel, date, orient) { - var coordSys = calendarModel.coordinateSystem; - var parsedDate = coordSys.getDateInfo(date); - var points = []; - - for (var i = 0; i < 7; i++) { - var tmpD = coordSys.getNextNDay(parsedDate.time, i); - var point = coordSys.dataToRect([tmpD.time], false); - points[2 * tmpD.day] = point.tl; - points[2 * tmpD.day + 1] = point[orient === 'horizontal' ? 'bl' : 'tr']; - } - - return points; - }; - - CalendarView.prototype._formatterLabel = function (formatter, params) { - if (isString(formatter) && formatter) { - return formatTplSimple(formatter, params); - } - - if (isFunction(formatter)) { - return formatter(params); - } - - return params.nameMap; - }; - - CalendarView.prototype._yearTextPositionControl = function (textEl, point, orient, position, margin) { - var x = point[0]; - var y = point[1]; - var aligns = ['center', 'bottom']; - - if (position === 'bottom') { - y += margin; - aligns = ['center', 'top']; - } else if (position === 'left') { - x -= margin; - } else if (position === 'right') { - x += margin; - aligns = ['center', 'top']; - } else { - // top - y -= margin; - } - - var rotate = 0; - - if (position === 'left' || position === 'right') { - rotate = Math.PI / 2; - } - - return { - rotation: rotate, - x: x, - y: y, - style: { - align: aligns[0], - verticalAlign: aligns[1] - } - }; - }; // render year - - - CalendarView.prototype._renderYearText = function (calendarModel, rangeData, orient, group) { - var yearLabel = calendarModel.getModel('yearLabel'); - - if (!yearLabel.get('show')) { - return; - } - - var margin = yearLabel.get('margin'); - var pos = yearLabel.get('position'); - - if (!pos) { - pos = orient !== 'horizontal' ? 'top' : 'left'; - } - - var points = [this._tlpoints[this._tlpoints.length - 1], this._blpoints[0]]; - var xc = (points[0][0] + points[1][0]) / 2; - var yc = (points[0][1] + points[1][1]) / 2; - var idx = orient === 'horizontal' ? 0 : 1; - var posPoints = { - top: [xc, points[idx][1]], - bottom: [xc, points[1 - idx][1]], - left: [points[1 - idx][0], yc], - right: [points[idx][0], yc] - }; - var name = rangeData.start.y; - - if (+rangeData.end.y > +rangeData.start.y) { - name = name + '-' + rangeData.end.y; - } - - var formatter = yearLabel.get('formatter'); - var params = { - start: rangeData.start.y, - end: rangeData.end.y, - nameMap: name - }; - - var content = this._formatterLabel(formatter, params); - - var yearText = new ZRText({ - z2: 30, - style: createTextStyle(yearLabel, { - text: content - }) - }); - yearText.attr(this._yearTextPositionControl(yearText, posPoints[pos], orient, pos, margin)); - group.add(yearText); - }; - - CalendarView.prototype._monthTextPositionControl = function (point, isCenter, orient, position, margin) { - var align = 'left'; - var vAlign = 'top'; - var x = point[0]; - var y = point[1]; - - if (orient === 'horizontal') { - y = y + margin; - - if (isCenter) { - align = 'center'; - } - - if (position === 'start') { - vAlign = 'bottom'; - } - } else { - x = x + margin; - - if (isCenter) { - vAlign = 'middle'; - } - - if (position === 'start') { - align = 'right'; - } - } - - return { - x: x, - y: y, - align: align, - verticalAlign: vAlign - }; - }; // render month and year text - - - CalendarView.prototype._renderMonthText = function (calendarModel, localeModel, orient, group) { - var monthLabel = calendarModel.getModel('monthLabel'); - - if (!monthLabel.get('show')) { - return; - } - - var nameMap = monthLabel.get('nameMap'); - var margin = monthLabel.get('margin'); - var pos = monthLabel.get('position'); - var align = monthLabel.get('align'); - var termPoints = [this._tlpoints, this._blpoints]; - - if (!nameMap || isString(nameMap)) { - if (nameMap) { - // case-sensitive - localeModel = getLocaleModel(nameMap) || localeModel; - } // PENDING - // for ZH locale, original form is `一月` but current form is `1月` - - - nameMap = localeModel.get(['time', 'monthAbbr']) || []; - } - - var idx = pos === 'start' ? 0 : 1; - var axis = orient === 'horizontal' ? 0 : 1; - margin = pos === 'start' ? -margin : margin; - var isCenter = align === 'center'; - - for (var i = 0; i < termPoints[idx].length - 1; i++) { - var tmp = termPoints[idx][i].slice(); - var firstDay = this._firstDayOfMonth[i]; - - if (isCenter) { - var firstDayPoints = this._firstDayPoints[i]; - tmp[axis] = (firstDayPoints[axis] + termPoints[0][i + 1][axis]) / 2; - } - - var formatter = monthLabel.get('formatter'); - var name_1 = nameMap[+firstDay.m - 1]; - var params = { - yyyy: firstDay.y, - yy: (firstDay.y + '').slice(2), - MM: firstDay.m, - M: +firstDay.m, - nameMap: name_1 - }; - - var content = this._formatterLabel(formatter, params); - - var monthText = new ZRText({ - z2: 30, - style: extend(createTextStyle(monthLabel, { - text: content - }), this._monthTextPositionControl(tmp, isCenter, orient, pos, margin)) - }); - group.add(monthText); - } - }; - - CalendarView.prototype._weekTextPositionControl = function (point, orient, position, margin, cellSize) { - var align = 'center'; - var vAlign = 'middle'; - var x = point[0]; - var y = point[1]; - var isStart = position === 'start'; - - if (orient === 'horizontal') { - x = x + margin + (isStart ? 1 : -1) * cellSize[0] / 2; - align = isStart ? 'right' : 'left'; - } else { - y = y + margin + (isStart ? 1 : -1) * cellSize[1] / 2; - vAlign = isStart ? 'bottom' : 'top'; - } - - return { - x: x, - y: y, - align: align, - verticalAlign: vAlign - }; - }; // render weeks - - - CalendarView.prototype._renderWeekText = function (calendarModel, localeModel, rangeData, orient, group) { - var dayLabel = calendarModel.getModel('dayLabel'); - - if (!dayLabel.get('show')) { - return; - } - - var coordSys = calendarModel.coordinateSystem; - var pos = dayLabel.get('position'); - var nameMap = dayLabel.get('nameMap'); - var margin = dayLabel.get('margin'); - var firstDayOfWeek = coordSys.getFirstDayOfWeek(); - - if (!nameMap || isString(nameMap)) { - if (nameMap) { - // case-sensitive - localeModel = getLocaleModel(nameMap) || localeModel; - } // Use the first letter of `dayOfWeekAbbr` if `dayOfWeekShort` doesn't exist in the locale file - - - var dayOfWeekShort = localeModel.get(['time', 'dayOfWeekShort']); - nameMap = dayOfWeekShort || map(localeModel.get(['time', 'dayOfWeekAbbr']), function (val) { - return val[0]; - }); - } - - var start = coordSys.getNextNDay(rangeData.end.time, 7 - rangeData.lweek).time; - var cellSize = [coordSys.getCellWidth(), coordSys.getCellHeight()]; - margin = parsePercent$1(margin, Math.min(cellSize[1], cellSize[0])); - - if (pos === 'start') { - start = coordSys.getNextNDay(rangeData.start.time, -(7 + rangeData.fweek)).time; - margin = -margin; - } - - for (var i = 0; i < 7; i++) { - var tmpD = coordSys.getNextNDay(start, i); - var point = coordSys.dataToRect([tmpD.time], false).center; - var day = i; - day = Math.abs((i + firstDayOfWeek) % 7); - var weekText = new ZRText({ - z2: 30, - style: extend(createTextStyle(dayLabel, { - text: nameMap[day] - }), this._weekTextPositionControl(point, orient, pos, margin, cellSize)) - }); - group.add(weekText); - } - }; - - CalendarView.type = 'calendar'; - return CalendarView; - }(ComponentView); - - var PROXIMATE_ONE_DAY = 86400000; - - var Calendar = - /** @class */ - function () { - function Calendar(calendarModel, ecModel, api) { - this.type = 'calendar'; - this.dimensions = Calendar.dimensions; // Required in createListFromData - - this.getDimensionsInfo = Calendar.getDimensionsInfo; - this._model = calendarModel; - } - - Calendar.getDimensionsInfo = function () { - return [{ - name: 'time', - type: 'time' - }, 'value']; - }; - - Calendar.prototype.getRangeInfo = function () { - return this._rangeInfo; - }; - - Calendar.prototype.getModel = function () { - return this._model; - }; - - Calendar.prototype.getRect = function () { - return this._rect; - }; - - Calendar.prototype.getCellWidth = function () { - return this._sw; - }; - - Calendar.prototype.getCellHeight = function () { - return this._sh; - }; - - Calendar.prototype.getOrient = function () { - return this._orient; - }; - /** - * getFirstDayOfWeek - * - * @example - * 0 : start at Sunday - * 1 : start at Monday - * - * @return {number} - */ - - - Calendar.prototype.getFirstDayOfWeek = function () { - return this._firstDayOfWeek; - }; - /** - * get date info - * } - */ - - - Calendar.prototype.getDateInfo = function (date) { - date = parseDate(date); - var y = date.getFullYear(); - var m = date.getMonth() + 1; - var mStr = m < 10 ? '0' + m : '' + m; - var d = date.getDate(); - var dStr = d < 10 ? '0' + d : '' + d; - var day = date.getDay(); - day = Math.abs((day + 7 - this.getFirstDayOfWeek()) % 7); - return { - y: y + '', - m: mStr, - d: dStr, - day: day, - time: date.getTime(), - formatedDate: y + '-' + mStr + '-' + dStr, - date: date - }; - }; - - Calendar.prototype.getNextNDay = function (date, n) { - n = n || 0; - - if (n === 0) { - return this.getDateInfo(date); - } - - date = new Date(this.getDateInfo(date).time); - date.setDate(date.getDate() + n); - return this.getDateInfo(date); - }; - - Calendar.prototype.update = function (ecModel, api) { - this._firstDayOfWeek = +this._model.getModel('dayLabel').get('firstDay'); - this._orient = this._model.get('orient'); - this._lineWidth = this._model.getModel('itemStyle').getItemStyle().lineWidth || 0; - this._rangeInfo = this._getRangeInfo(this._initRangeOption()); - var weeks = this._rangeInfo.weeks || 1; - var whNames = ['width', 'height']; - - var cellSize = this._model.getCellSize().slice(); - - var layoutParams = this._model.getBoxLayoutParams(); - - var cellNumbers = this._orient === 'horizontal' ? [weeks, 7] : [7, weeks]; - each([0, 1], function (idx) { - if (cellSizeSpecified(cellSize, idx)) { - layoutParams[whNames[idx]] = cellSize[idx] * cellNumbers[idx]; - } - }); - var whGlobal = { - width: api.getWidth(), - height: api.getHeight() - }; - var calendarRect = this._rect = getLayoutRect(layoutParams, whGlobal); - each([0, 1], function (idx) { - if (!cellSizeSpecified(cellSize, idx)) { - cellSize[idx] = calendarRect[whNames[idx]] / cellNumbers[idx]; - } - }); - - function cellSizeSpecified(cellSize, idx) { - return cellSize[idx] != null && cellSize[idx] !== 'auto'; - } // Has been calculated out number. - - - this._sw = cellSize[0]; - this._sh = cellSize[1]; - }; - /** - * Convert a time data(time, value) item to (x, y) point. - */ - // TODO Clamp of calendar is not same with cartesian coordinate systems. - // It will return NaN if data exceeds. - - - Calendar.prototype.dataToPoint = function (data, clamp) { - isArray(data) && (data = data[0]); - clamp == null && (clamp = true); - var dayInfo = this.getDateInfo(data); - var range = this._rangeInfo; - var date = dayInfo.formatedDate; // if not in range return [NaN, NaN] - - if (clamp && !(dayInfo.time >= range.start.time && dayInfo.time < range.end.time + PROXIMATE_ONE_DAY)) { - return [NaN, NaN]; - } - - var week = dayInfo.day; - - var nthWeek = this._getRangeInfo([range.start.time, date]).nthWeek; - - if (this._orient === 'vertical') { - return [this._rect.x + week * this._sw + this._sw / 2, this._rect.y + nthWeek * this._sh + this._sh / 2]; - } - - return [this._rect.x + nthWeek * this._sw + this._sw / 2, this._rect.y + week * this._sh + this._sh / 2]; - }; - /** - * Convert a (x, y) point to time data - */ - - - Calendar.prototype.pointToData = function (point) { - var date = this.pointToDate(point); - return date && date.time; - }; - /** - * Convert a time date item to (x, y) four point. - */ - - - Calendar.prototype.dataToRect = function (data, clamp) { - var point = this.dataToPoint(data, clamp); - return { - contentShape: { - x: point[0] - (this._sw - this._lineWidth) / 2, - y: point[1] - (this._sh - this._lineWidth) / 2, - width: this._sw - this._lineWidth, - height: this._sh - this._lineWidth - }, - center: point, - tl: [point[0] - this._sw / 2, point[1] - this._sh / 2], - tr: [point[0] + this._sw / 2, point[1] - this._sh / 2], - br: [point[0] + this._sw / 2, point[1] + this._sh / 2], - bl: [point[0] - this._sw / 2, point[1] + this._sh / 2] - }; - }; - /** - * Convert a (x, y) point to time date - * - * @param {Array} point point - * @return {Object} date - */ - - - Calendar.prototype.pointToDate = function (point) { - var nthX = Math.floor((point[0] - this._rect.x) / this._sw) + 1; - var nthY = Math.floor((point[1] - this._rect.y) / this._sh) + 1; - var range = this._rangeInfo.range; - - if (this._orient === 'vertical') { - return this._getDateByWeeksAndDay(nthY, nthX - 1, range); - } - - return this._getDateByWeeksAndDay(nthX, nthY - 1, range); - }; - - Calendar.prototype.convertToPixel = function (ecModel, finder, value) { - var coordSys = getCoordSys$4(finder); - return coordSys === this ? coordSys.dataToPoint(value) : null; - }; - - Calendar.prototype.convertFromPixel = function (ecModel, finder, pixel) { - var coordSys = getCoordSys$4(finder); - return coordSys === this ? coordSys.pointToData(pixel) : null; - }; - - Calendar.prototype.containPoint = function (point) { - console.warn('Not implemented.'); - return false; - }; - /** - * initRange - * Normalize to an [start, end] array - */ - - - Calendar.prototype._initRangeOption = function () { - var range = this._model.get('range'); - - var normalizedRange; // Convert [1990] to 1990 - - if (isArray(range) && range.length === 1) { - range = range[0]; - } - - if (!isArray(range)) { - var rangeStr = range.toString(); // One year. - - if (/^\d{4}$/.test(rangeStr)) { - normalizedRange = [rangeStr + '-01-01', rangeStr + '-12-31']; - } // One month - - - if (/^\d{4}[\/|-]\d{1,2}$/.test(rangeStr)) { - var start = this.getDateInfo(rangeStr); - var firstDay = start.date; - firstDay.setMonth(firstDay.getMonth() + 1); - var end = this.getNextNDay(firstDay, -1); - normalizedRange = [start.formatedDate, end.formatedDate]; - } // One day - - - if (/^\d{4}[\/|-]\d{1,2}[\/|-]\d{1,2}$/.test(rangeStr)) { - normalizedRange = [rangeStr, rangeStr]; - } - } else { - normalizedRange = range; - } - - if (!normalizedRange) { - if ("development" !== 'production') { - logError('Invalid date range.'); - } // Not handling it. - - - return range; - } - - var tmp = this._getRangeInfo(normalizedRange); - - if (tmp.start.time > tmp.end.time) { - normalizedRange.reverse(); - } - - return normalizedRange; - }; - /** - * range info - * - * @private - * @param {Array} range range ['2017-01-01', '2017-07-08'] - * If range[0] > range[1], they will not be reversed. - * @return {Object} obj - */ - - - Calendar.prototype._getRangeInfo = function (range) { - var parsedRange = [this.getDateInfo(range[0]), this.getDateInfo(range[1])]; - var reversed; - - if (parsedRange[0].time > parsedRange[1].time) { - reversed = true; - parsedRange.reverse(); - } - - var allDay = Math.floor(parsedRange[1].time / PROXIMATE_ONE_DAY) - Math.floor(parsedRange[0].time / PROXIMATE_ONE_DAY) + 1; // Consider case1 (#11677 #10430): - // Set the system timezone as "UK", set the range to `['2016-07-01', '2016-12-31']` - // Consider case2: - // Firstly set system timezone as "Time Zone: America/Toronto", - // ``` - // let first = new Date(1478412000000 - 3600 * 1000 * 2.5); - // let second = new Date(1478412000000); - // let allDays = Math.floor(second / ONE_DAY) - Math.floor(first / ONE_DAY) + 1; - // ``` - // will get wrong result because of DST. So we should fix it. - - var date = new Date(parsedRange[0].time); - var startDateNum = date.getDate(); - var endDateNum = parsedRange[1].date.getDate(); - date.setDate(startDateNum + allDay - 1); // The bias can not over a month, so just compare date. - - var dateNum = date.getDate(); - - if (dateNum !== endDateNum) { - var sign = date.getTime() - parsedRange[1].time > 0 ? 1 : -1; - - while ((dateNum = date.getDate()) !== endDateNum && (date.getTime() - parsedRange[1].time) * sign > 0) { - allDay -= sign; - date.setDate(dateNum - sign); - } - } - - var weeks = Math.floor((allDay + parsedRange[0].day + 6) / 7); - var nthWeek = reversed ? -weeks + 1 : weeks - 1; - reversed && parsedRange.reverse(); - return { - range: [parsedRange[0].formatedDate, parsedRange[1].formatedDate], - start: parsedRange[0], - end: parsedRange[1], - allDay: allDay, - weeks: weeks, - // From 0. - nthWeek: nthWeek, - fweek: parsedRange[0].day, - lweek: parsedRange[1].day - }; - }; - /** - * get date by nthWeeks and week day in range - * - * @private - * @param {number} nthWeek the week - * @param {number} day the week day - * @param {Array} range [d1, d2] - * @return {Object} - */ - - - Calendar.prototype._getDateByWeeksAndDay = function (nthWeek, day, range) { - var rangeInfo = this._getRangeInfo(range); - - if (nthWeek > rangeInfo.weeks || nthWeek === 0 && day < rangeInfo.fweek || nthWeek === rangeInfo.weeks && day > rangeInfo.lweek) { - return null; - } - - var nthDay = (nthWeek - 1) * 7 - rangeInfo.fweek + day; - var date = new Date(rangeInfo.start.time); - date.setDate(+rangeInfo.start.d + nthDay); - return this.getDateInfo(date); - }; - - Calendar.create = function (ecModel, api) { - var calendarList = []; - ecModel.eachComponent('calendar', function (calendarModel) { - var calendar = new Calendar(calendarModel, ecModel, api); - calendarList.push(calendar); - calendarModel.coordinateSystem = calendar; - }); - ecModel.eachSeries(function (calendarSeries) { - if (calendarSeries.get('coordinateSystem') === 'calendar') { - // Inject coordinate system - calendarSeries.coordinateSystem = calendarList[calendarSeries.get('calendarIndex') || 0]; - } - }); - return calendarList; - }; - - Calendar.dimensions = ['time', 'value']; - return Calendar; - }(); - - function getCoordSys$4(finder) { - var calendarModel = finder.calendarModel; - var seriesModel = finder.seriesModel; - var coordSys = calendarModel ? calendarModel.coordinateSystem : seriesModel ? seriesModel.coordinateSystem : null; - return coordSys; - } - - function install$w(registers) { - registers.registerComponentModel(CalendarModel); - registers.registerComponentView(CalendarView); - registers.registerCoordinateSystem('calendar', Calendar); - } - - function setKeyInfoToNewElOption(resultItem, newElOption) { - var existElOption = resultItem.existing; // Set id and type after id assigned. - - newElOption.id = resultItem.keyInfo.id; - !newElOption.type && existElOption && (newElOption.type = existElOption.type); // Set parent id if not specified - - if (newElOption.parentId == null) { - var newElParentOption = newElOption.parentOption; - - if (newElParentOption) { - newElOption.parentId = newElParentOption.id; - } else if (existElOption) { - newElOption.parentId = existElOption.parentId; - } - } // Clear - - - newElOption.parentOption = null; - } - - function isSetLoc(obj, props) { - var isSet; - each(props, function (prop) { - obj[prop] != null && obj[prop] !== 'auto' && (isSet = true); - }); - return isSet; - } - - function mergeNewElOptionToExist(existList, index, newElOption) { - // Update existing options, for `getOption` feature. - var newElOptCopy = extend({}, newElOption); - var existElOption = existList[index]; - var $action = newElOption.$action || 'merge'; - - if ($action === 'merge') { - if (existElOption) { - if ("development" !== 'production') { - var newType = newElOption.type; - assert(!newType || existElOption.type === newType, 'Please set $action: "replace" to change `type`'); - } // We can ensure that newElOptCopy and existElOption are not - // the same object, so `merge` will not change newElOptCopy. - - - merge(existElOption, newElOptCopy, true); // Rigid body, use ignoreSize. - - mergeLayoutParam(existElOption, newElOptCopy, { - ignoreSize: true - }); // Will be used in render. - - copyLayoutParams(newElOption, existElOption); // Copy transition info to new option so it can be used in the transition. - // DO IT AFTER merge - - copyTransitionInfo(newElOption, existElOption); - copyTransitionInfo(newElOption, existElOption, 'shape'); - copyTransitionInfo(newElOption, existElOption, 'style'); - copyTransitionInfo(newElOption, existElOption, 'extra'); // Copy clipPath - - newElOption.clipPath = existElOption.clipPath; - } else { - existList[index] = newElOptCopy; - } - } else if ($action === 'replace') { - existList[index] = newElOptCopy; - } else if ($action === 'remove') { - // null will be cleaned later. - existElOption && (existList[index] = null); - } - } - - var TRANSITION_PROPS_TO_COPY = ['transition', 'enterFrom', 'leaveTo']; - var ROOT_TRANSITION_PROPS_TO_COPY = TRANSITION_PROPS_TO_COPY.concat(['enterAnimation', 'updateAnimation', 'leaveAnimation']); - - function copyTransitionInfo(target, source, targetProp) { - if (targetProp) { - if (!target[targetProp] && source[targetProp]) { - // TODO avoid creating this empty object when there is no transition configuration. - target[targetProp] = {}; - } - - target = target[targetProp]; - source = source[targetProp]; - } - - if (!target || !source) { - return; - } - - var props = targetProp ? TRANSITION_PROPS_TO_COPY : ROOT_TRANSITION_PROPS_TO_COPY; - - for (var i = 0; i < props.length; i++) { - var prop = props[i]; - - if (target[prop] == null && source[prop] != null) { - target[prop] = source[prop]; - } - } - } - - function setLayoutInfoToExist(existItem, newElOption) { - if (!existItem) { - return; - } - - existItem.hv = newElOption.hv = [// Rigid body, don't care about `width`. - isSetLoc(newElOption, ['left', 'right']), // Rigid body, don't care about `height`. - isSetLoc(newElOption, ['top', 'bottom'])]; // Give default group size. Otherwise layout error may occur. - - if (existItem.type === 'group') { - var existingGroupOpt = existItem; - var newGroupOpt = newElOption; - existingGroupOpt.width == null && (existingGroupOpt.width = newGroupOpt.width = 0); - existingGroupOpt.height == null && (existingGroupOpt.height = newGroupOpt.height = 0); - } - } - - var GraphicComponentModel = - /** @class */ - function (_super) { - __extends(GraphicComponentModel, _super); - - function GraphicComponentModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GraphicComponentModel.type; - _this.preventAutoZ = true; - return _this; - } - - GraphicComponentModel.prototype.mergeOption = function (option, ecModel) { - // Prevent default merge to elements - var elements = this.option.elements; - this.option.elements = null; - - _super.prototype.mergeOption.call(this, option, ecModel); - - this.option.elements = elements; - }; - - GraphicComponentModel.prototype.optionUpdated = function (newOption, isInit) { - var thisOption = this.option; - var newList = (isInit ? thisOption : newOption).elements; - var existList = thisOption.elements = isInit ? [] : thisOption.elements; - var flattenedList = []; - - this._flatten(newList, flattenedList, null); - - var mappingResult = mappingToExists(existList, flattenedList, 'normalMerge'); // Clear elOptionsToUpdate - - var elOptionsToUpdate = this._elOptionsToUpdate = []; - each(mappingResult, function (resultItem, index) { - var newElOption = resultItem.newOption; - - if ("development" !== 'production') { - assert(isObject(newElOption) || resultItem.existing, 'Empty graphic option definition'); - } - - if (!newElOption) { - return; - } - - elOptionsToUpdate.push(newElOption); - setKeyInfoToNewElOption(resultItem, newElOption); - mergeNewElOptionToExist(existList, index, newElOption); - setLayoutInfoToExist(existList[index], newElOption); - }, this); // Clean - - thisOption.elements = filter(existList, function (item) { - // $action should be volatile, otherwise option gotten from - // `getOption` will contain unexpected $action. - item && delete item.$action; - return item != null; - }); - }; - /** - * Convert - * [{ - * type: 'group', - * id: 'xx', - * children: [{type: 'circle'}, {type: 'polygon'}] - * }] - * to - * [ - * {type: 'group', id: 'xx'}, - * {type: 'circle', parentId: 'xx'}, - * {type: 'polygon', parentId: 'xx'} - * ] - */ - - - GraphicComponentModel.prototype._flatten = function (optionList, result, parentOption) { - each(optionList, function (option) { - if (!option) { - return; - } - - if (parentOption) { - option.parentOption = parentOption; - } - - result.push(option); - var children = option.children; // here we don't judge if option.type is `group` - // when new option doesn't provide `type`, it will cause that the children can't be updated. - - if (children && children.length) { - this._flatten(children, result, option); - } // Deleting for JSON output, and for not affecting group creation. - - - delete option.children; - }, this); - }; // FIXME - // Pass to view using payload? setOption has a payload? - - - GraphicComponentModel.prototype.useElOptionsToUpdate = function () { - var els = this._elOptionsToUpdate; // Clear to avoid render duplicately when zooming. - - this._elOptionsToUpdate = null; - return els; - }; - - GraphicComponentModel.type = 'graphic'; - GraphicComponentModel.defaultOption = { - elements: [] // parentId: null - - }; - return GraphicComponentModel; - }(ComponentModel); - - var nonShapeGraphicElements = { - // Reserved but not supported in graphic component. - path: null, - compoundPath: null, - // Supported in graphic component. - group: Group, - image: ZRImage, - text: ZRText - }; - var inner$e = makeInner(); // ------------------------ - // View - // ------------------------ - - var GraphicComponentView = - /** @class */ - function (_super) { - __extends(GraphicComponentView, _super); - - function GraphicComponentView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = GraphicComponentView.type; - return _this; - } - - GraphicComponentView.prototype.init = function () { - this._elMap = createHashMap(); - }; - - GraphicComponentView.prototype.render = function (graphicModel, ecModel, api) { - // Having leveraged between use cases and algorithm complexity, a very - // simple layout mechanism is used: - // The size(width/height) can be determined by itself or its parent (not - // implemented yet), but can not by its children. (Top-down travel) - // The location(x/y) can be determined by the bounding rect of itself - // (can including its descendants or not) and the size of its parent. - // (Bottom-up travel) - // When `chart.clear()` or `chart.setOption({...}, true)` with the same id, - // view will be reused. - if (graphicModel !== this._lastGraphicModel) { - this._clear(); - } - - this._lastGraphicModel = graphicModel; - - this._updateElements(graphicModel); - - this._relocate(graphicModel, api); - }; - /** - * Update graphic elements. - */ - - - GraphicComponentView.prototype._updateElements = function (graphicModel) { - var elOptionsToUpdate = graphicModel.useElOptionsToUpdate(); - - if (!elOptionsToUpdate) { - return; - } - - var elMap = this._elMap; - var rootGroup = this.group; - var globalZ = graphicModel.get('z'); - var globalZLevel = graphicModel.get('zlevel'); // Top-down tranverse to assign graphic settings to each elements. - - each(elOptionsToUpdate, function (elOption) { - var id = convertOptionIdName(elOption.id, null); - var elExisting = id != null ? elMap.get(id) : null; - var parentId = convertOptionIdName(elOption.parentId, null); - var targetElParent = parentId != null ? elMap.get(parentId) : rootGroup; - var elType = elOption.type; - var elOptionStyle = elOption.style; - - if (elType === 'text' && elOptionStyle) { - // In top/bottom mode, textVerticalAlign should not be used, which cause - // inaccurately locating. - if (elOption.hv && elOption.hv[1]) { - elOptionStyle.textVerticalAlign = elOptionStyle.textBaseline = elOptionStyle.verticalAlign = elOptionStyle.align = null; - } - } - - var textContentOption = elOption.textContent; - var textConfig = elOption.textConfig; - - if (elOptionStyle && isEC4CompatibleStyle(elOptionStyle, elType, !!textConfig, !!textContentOption)) { - var convertResult = convertFromEC4CompatibleStyle(elOptionStyle, elType, true); - - if (!textConfig && convertResult.textConfig) { - textConfig = elOption.textConfig = convertResult.textConfig; - } - - if (!textContentOption && convertResult.textContent) { - textContentOption = convertResult.textContent; - } - } // Remove unnecessary props to avoid potential problems. - - - var elOptionCleaned = getCleanedElOption(elOption); // For simple, do not support parent change, otherwise reorder is needed. - - if ("development" !== 'production') { - elExisting && assert(targetElParent === elExisting.parent, 'Changing parent is not supported.'); - } - - var $action = elOption.$action || 'merge'; - var isMerge = $action === 'merge'; - var isReplace = $action === 'replace'; - - if (isMerge) { - var isInit = !elExisting; - var el_1 = elExisting; - - if (isInit) { - el_1 = createEl$1(id, targetElParent, elOption.type, elMap); - } else { - el_1 && (inner$e(el_1).isNew = false); // Stop and restore before update any other attributes. - - stopPreviousKeyframeAnimationAndRestore(el_1); - } - - if (el_1) { - applyUpdateTransition(el_1, elOptionCleaned, graphicModel, { - isInit: isInit - }); - updateCommonAttrs(el_1, elOption, globalZ, globalZLevel); - } - } else if (isReplace) { - removeEl(elExisting, elOption, elMap, graphicModel); - var el_2 = createEl$1(id, targetElParent, elOption.type, elMap); - - if (el_2) { - applyUpdateTransition(el_2, elOptionCleaned, graphicModel, { - isInit: true - }); - updateCommonAttrs(el_2, elOption, globalZ, globalZLevel); - } - } else if ($action === 'remove') { - updateLeaveTo(elExisting, elOption); - removeEl(elExisting, elOption, elMap, graphicModel); - } - - var el = elMap.get(id); - - if (el && textContentOption) { - if (isMerge) { - var textContentExisting = el.getTextContent(); - textContentExisting ? textContentExisting.attr(textContentOption) : el.setTextContent(new ZRText(textContentOption)); - } else if (isReplace) { - el.setTextContent(new ZRText(textContentOption)); - } - } - - if (el) { - var clipPathOption = elOption.clipPath; - - if (clipPathOption) { - var clipPathType = clipPathOption.type; - var clipPath = void 0; - var isInit = false; - - if (isMerge) { - var oldClipPath = el.getClipPath(); - isInit = !oldClipPath || inner$e(oldClipPath).type !== clipPathType; - clipPath = isInit ? newEl(clipPathType) : oldClipPath; - } else if (isReplace) { - isInit = true; - clipPath = newEl(clipPathType); - } - - el.setClipPath(clipPath); - applyUpdateTransition(clipPath, clipPathOption, graphicModel, { - isInit: isInit - }); - applyKeyframeAnimation(clipPath, clipPathOption.keyframeAnimation, graphicModel); - } - - var elInner = inner$e(el); - el.setTextConfig(textConfig); - elInner.option = elOption; - setEventData(el, graphicModel, elOption); - setTooltipConfig({ - el: el, - componentModel: graphicModel, - itemName: el.name, - itemTooltipOption: elOption.tooltip - }); - applyKeyframeAnimation(el, elOption.keyframeAnimation, graphicModel); - } - }); - }; - /** - * Locate graphic elements. - */ - - - GraphicComponentView.prototype._relocate = function (graphicModel, api) { - var elOptions = graphicModel.option.elements; - var rootGroup = this.group; - var elMap = this._elMap; - var apiWidth = api.getWidth(); - var apiHeight = api.getHeight(); - var xy = ['x', 'y']; // Top-down to calculate percentage width/height of group - - for (var i = 0; i < elOptions.length; i++) { - var elOption = elOptions[i]; - var id = convertOptionIdName(elOption.id, null); - var el = id != null ? elMap.get(id) : null; - - if (!el || !el.isGroup) { - continue; - } - - var parentEl = el.parent; - var isParentRoot = parentEl === rootGroup; // Like 'position:absolut' in css, default 0. - - var elInner = inner$e(el); - var parentElInner = inner$e(parentEl); - elInner.width = parsePercent$1(elInner.option.width, isParentRoot ? apiWidth : parentElInner.width) || 0; - elInner.height = parsePercent$1(elInner.option.height, isParentRoot ? apiHeight : parentElInner.height) || 0; - } // Bottom-up tranvese all elements (consider ec resize) to locate elements. - - - for (var i = elOptions.length - 1; i >= 0; i--) { - var elOption = elOptions[i]; - var id = convertOptionIdName(elOption.id, null); - var el = id != null ? elMap.get(id) : null; - - if (!el) { - continue; - } - - var parentEl = el.parent; - var parentElInner = inner$e(parentEl); - var containerInfo = parentEl === rootGroup ? { - width: apiWidth, - height: apiHeight - } : { - width: parentElInner.width, - height: parentElInner.height - }; // PENDING - // Currently, when `bounding: 'all'`, the union bounding rect of the group - // does not include the rect of [0, 0, group.width, group.height], which - // is probably weird for users. Should we make a break change for it? - - var layoutPos = {}; - var layouted = positionElement(el, elOption, containerInfo, null, { - hv: elOption.hv, - boundingMode: elOption.bounding - }, layoutPos); - - if (!inner$e(el).isNew && layouted) { - var transition = elOption.transition; - var animatePos = {}; - - for (var k = 0; k < xy.length; k++) { - var key = xy[k]; - var val = layoutPos[key]; - - if (transition && (isTransitionAll(transition) || indexOf(transition, key) >= 0)) { - animatePos[key] = val; - } else { - el[key] = val; - } - } - - updateProps(el, animatePos, graphicModel, 0); - } else { - el.attr(layoutPos); - } - } - }; - /** - * Clear all elements. - */ - - - GraphicComponentView.prototype._clear = function () { - var _this = this; - - var elMap = this._elMap; - elMap.each(function (el) { - removeEl(el, inner$e(el).option, elMap, _this._lastGraphicModel); - }); - this._elMap = createHashMap(); - }; - - GraphicComponentView.prototype.dispose = function () { - this._clear(); - }; - - GraphicComponentView.type = 'graphic'; - return GraphicComponentView; - }(ComponentView); - - function newEl(graphicType) { - if ("development" !== 'production') { - assert(graphicType, 'graphic type MUST be set'); - } - - var Clz = hasOwn(nonShapeGraphicElements, graphicType) // Those graphic elements are not shapes. They should not be - // overwritten by users, so do them first. - ? nonShapeGraphicElements[graphicType] : getShapeClass(graphicType); - - if ("development" !== 'production') { - assert(Clz, "graphic type " + graphicType + " can not be found"); - } - - var el = new Clz({}); - inner$e(el).type = graphicType; - return el; - } - - function createEl$1(id, targetElParent, graphicType, elMap) { - var el = newEl(graphicType); - targetElParent.add(el); - elMap.set(id, el); - inner$e(el).id = id; - inner$e(el).isNew = true; - return el; - } - - function removeEl(elExisting, elOption, elMap, graphicModel) { - var existElParent = elExisting && elExisting.parent; - - if (existElParent) { - elExisting.type === 'group' && elExisting.traverse(function (el) { - removeEl(el, elOption, elMap, graphicModel); - }); - applyLeaveTransition(elExisting, elOption, graphicModel); - elMap.removeKey(inner$e(elExisting).id); - } - } - - function updateCommonAttrs(el, elOption, defaultZ, defaultZlevel) { - if (!el.isGroup) { - each([['cursor', Displayable.prototype.cursor], // We should not support configure z and zlevel in the element level. - // But seems we didn't limit it previously. So here still use it to avoid breaking. - ['zlevel', defaultZlevel || 0], ['z', defaultZ || 0], // z2 must not be null/undefined, otherwise sort error may occur. - ['z2', 0]], function (item) { - var prop = item[0]; - - if (hasOwn(elOption, prop)) { - el[prop] = retrieve2(elOption[prop], item[1]); - } else if (el[prop] == null) { - el[prop] = item[1]; - } - }); - } - - each(keys(elOption), function (key) { - // Assign event handlers. - // PENDING: should enumerate all event names or use pattern matching? - if (key.indexOf('on') === 0) { - var val = elOption[key]; - el[key] = isFunction(val) ? val : null; - } - }); - - if (hasOwn(elOption, 'draggable')) { - el.draggable = elOption.draggable; - } // Other attributes - - - elOption.name != null && (el.name = elOption.name); - elOption.id != null && (el.id = elOption.id); - } // Remove unnecessary props to avoid potential problems. - - - function getCleanedElOption(elOption) { - elOption = extend({}, elOption); - each(['id', 'parentId', '$action', 'hv', 'bounding', 'textContent', 'clipPath'].concat(LOCATION_PARAMS), function (name) { - delete elOption[name]; - }); - return elOption; - } - - function setEventData(el, graphicModel, elOption) { - var eventData = getECData(el).eventData; // Simple optimize for large amount of elements that no need event. - - if (!el.silent && !el.ignore && !eventData) { - eventData = getECData(el).eventData = { - componentType: 'graphic', - componentIndex: graphicModel.componentIndex, - name: el.name - }; - } // `elOption.info` enables user to mount some info on - // elements and use them in event handlers. - - - if (eventData) { - eventData.info = elOption.info; - } - } - - function install$x(registers) { - registers.registerComponentModel(GraphicComponentModel); - registers.registerComponentView(GraphicComponentView); - registers.registerPreprocessor(function (option) { - var graphicOption = option.graphic; // Convert - // {graphic: [{left: 10, type: 'circle'}, ...]} - // or - // {graphic: {left: 10, type: 'circle'}} - // to - // {graphic: [{elements: [{left: 10, type: 'circle'}, ...]}]} - - if (isArray(graphicOption)) { - if (!graphicOption[0] || !graphicOption[0].elements) { - option.graphic = [{ - elements: graphicOption - }]; - } else { - // Only one graphic instance can be instantiated. (We don't - // want that too many views are created in echarts._viewMap.) - option.graphic = [option.graphic[0]]; - } - } else if (graphicOption && !graphicOption.elements) { - option.graphic = [{ - elements: [graphicOption] - }]; - } - }); - } - - var DATA_ZOOM_AXIS_DIMENSIONS = ['x', 'y', 'radius', 'angle', 'single']; // Supported coords. - // FIXME: polar has been broken (but rarely used). - - var SERIES_COORDS = ['cartesian2d', 'polar', 'singleAxis']; - function isCoordSupported(seriesModel) { - var coordType = seriesModel.get('coordinateSystem'); - return indexOf(SERIES_COORDS, coordType) >= 0; - } - function getAxisMainType(axisDim) { - if ("development" !== 'production') { - assert(axisDim); - } - - return axisDim + 'Axis'; - } - /** - * If two dataZoomModels has the same axis controlled, we say that they are 'linked'. - * This function finds all linked dataZoomModels start from the given payload. - */ - - function findEffectedDataZooms(ecModel, payload) { - // Key: `DataZoomAxisDimension` - var axisRecords = createHashMap(); - var effectedModels = []; // Key: uid of dataZoomModel - - var effectedModelMap = createHashMap(); // Find the dataZooms specified by payload. - - ecModel.eachComponent({ - mainType: 'dataZoom', - query: payload - }, function (dataZoomModel) { - if (!effectedModelMap.get(dataZoomModel.uid)) { - addToEffected(dataZoomModel); - } - }); // Start from the given dataZoomModels, travel the graph to find - // all of the linked dataZoom models. - - var foundNewLink; - - do { - foundNewLink = false; - ecModel.eachComponent('dataZoom', processSingle); - } while (foundNewLink); - - function processSingle(dataZoomModel) { - if (!effectedModelMap.get(dataZoomModel.uid) && isLinked(dataZoomModel)) { - addToEffected(dataZoomModel); - foundNewLink = true; - } - } - - function addToEffected(dataZoom) { - effectedModelMap.set(dataZoom.uid, true); - effectedModels.push(dataZoom); - markAxisControlled(dataZoom); - } - - function isLinked(dataZoomModel) { - var isLink = false; - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - var axisIdxArr = axisRecords.get(axisDim); - - if (axisIdxArr && axisIdxArr[axisIndex]) { - isLink = true; - } - }); - return isLink; - } - - function markAxisControlled(dataZoomModel) { - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - (axisRecords.get(axisDim) || axisRecords.set(axisDim, []))[axisIndex] = true; - }); - } - - return effectedModels; - } - /** - * Find the first target coordinate system. - * Available after model built. - * - * @return Like { - * grid: [ - * {model: coord0, axisModels: [axis1, axis3], coordIndex: 1}, - * {model: coord1, axisModels: [axis0, axis2], coordIndex: 0}, - * ... - * ], // cartesians must not be null/undefined. - * polar: [ - * {model: coord0, axisModels: [axis4], coordIndex: 0}, - * ... - * ], // polars must not be null/undefined. - * singleAxis: [ - * {model: coord0, axisModels: [], coordIndex: 0} - * ] - * } - */ - - function collectReferCoordSysModelInfo(dataZoomModel) { - var ecModel = dataZoomModel.ecModel; - var coordSysInfoWrap = { - infoList: [], - infoMap: createHashMap() - }; - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - var axisModel = ecModel.getComponent(getAxisMainType(axisDim), axisIndex); - - if (!axisModel) { - return; - } - - var coordSysModel = axisModel.getCoordSysModel(); - - if (!coordSysModel) { - return; - } - - var coordSysUid = coordSysModel.uid; - var coordSysInfo = coordSysInfoWrap.infoMap.get(coordSysUid); - - if (!coordSysInfo) { - coordSysInfo = { - model: coordSysModel, - axisModels: [] - }; - coordSysInfoWrap.infoList.push(coordSysInfo); - coordSysInfoWrap.infoMap.set(coordSysUid, coordSysInfo); - } - - coordSysInfo.axisModels.push(axisModel); - }); - return coordSysInfoWrap; - } - - var DataZoomAxisInfo = - /** @class */ - function () { - function DataZoomAxisInfo() { - this.indexList = []; - this.indexMap = []; - } - - DataZoomAxisInfo.prototype.add = function (axisCmptIdx) { - // Remove duplication. - if (!this.indexMap[axisCmptIdx]) { - this.indexList.push(axisCmptIdx); - this.indexMap[axisCmptIdx] = true; - } - }; - - return DataZoomAxisInfo; - }(); - - var DataZoomModel = - /** @class */ - function (_super) { - __extends(DataZoomModel, _super); - - function DataZoomModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = DataZoomModel.type; - _this._autoThrottle = true; - _this._noTarget = true; - /** - * It is `[rangeModeForMin, rangeModeForMax]`. - * The optional values for `rangeMode`: - * + `'value'` mode: the axis extent will always be determined by - * `dataZoom.startValue` and `dataZoom.endValue`, despite - * how data like and how `axis.min` and `axis.max` are. - * + `'percent'` mode: `100` represents 100% of the `[dMin, dMax]`, - * where `dMin` is `axis.min` if `axis.min` specified, otherwise `data.extent[0]`, - * and `dMax` is `axis.max` if `axis.max` specified, otherwise `data.extent[1]`. - * Axis extent will be determined by the result of the percent of `[dMin, dMax]`. - * - * For example, when users are using dynamic data (update data periodically via `setOption`), - * if in `'value`' mode, the window will be kept in a fixed value range despite how - * data are appended, while if in `'percent'` mode, whe window range will be changed alone with - * the appended data (suppose `axis.min` and `axis.max` are not specified). - */ - - _this._rangePropMode = ['percent', 'percent']; - return _this; - } - - DataZoomModel.prototype.init = function (option, parentModel, ecModel) { - var inputRawOption = retrieveRawOption(option); - /** - * Suppose a "main process" start at the point that model prepared (that is, - * model initialized or merged or method called in `action`). - * We should keep the `main process` idempotent, that is, given a set of values - * on `option`, we get the same result. - * - * But sometimes, values on `option` will be updated for providing users - * a "final calculated value" (`dataZoomProcessor` will do that). Those value - * should not be the base/input of the `main process`. - * - * So in that case we should save and keep the input of the `main process` - * separately, called `settledOption`. - * - * For example, consider the case: - * (Step_1) brush zoom the grid by `toolbox.dataZoom`, - * where the original input `option.startValue`, `option.endValue` are earsed by - * calculated value. - * (Step)2) click the legend to hide and show a series, - * where the new range is calculated by the earsed `startValue` and `endValue`, - * which brings incorrect result. - */ - - this.settledOption = inputRawOption; - this.mergeDefaultAndTheme(option, ecModel); - - this._doInit(inputRawOption); - }; - - DataZoomModel.prototype.mergeOption = function (newOption) { - var inputRawOption = retrieveRawOption(newOption); // FIX #2591 - - merge(this.option, newOption, true); - merge(this.settledOption, inputRawOption, true); - - this._doInit(inputRawOption); - }; - - DataZoomModel.prototype._doInit = function (inputRawOption) { - var thisOption = this.option; - - this._setDefaultThrottle(inputRawOption); - - this._updateRangeUse(inputRawOption); - - var settledOption = this.settledOption; - each([['start', 'startValue'], ['end', 'endValue']], function (names, index) { - // start/end has higher priority over startValue/endValue if they - // both set, but we should make chart.setOption({endValue: 1000}) - // effective, rather than chart.setOption({endValue: 1000, end: null}). - if (this._rangePropMode[index] === 'value') { - thisOption[names[0]] = settledOption[names[0]] = null; - } // Otherwise do nothing and use the merge result. - - }, this); - - this._resetTarget(); - }; - - DataZoomModel.prototype._resetTarget = function () { - var optionOrient = this.get('orient', true); - var targetAxisIndexMap = this._targetAxisInfoMap = createHashMap(); - - var hasAxisSpecified = this._fillSpecifiedTargetAxis(targetAxisIndexMap); - - if (hasAxisSpecified) { - this._orient = optionOrient || this._makeAutoOrientByTargetAxis(); - } else { - this._orient = optionOrient || 'horizontal'; - - this._fillAutoTargetAxisByOrient(targetAxisIndexMap, this._orient); - } - - this._noTarget = true; - targetAxisIndexMap.each(function (axisInfo) { - if (axisInfo.indexList.length) { - this._noTarget = false; - } - }, this); - }; - - DataZoomModel.prototype._fillSpecifiedTargetAxis = function (targetAxisIndexMap) { - var hasAxisSpecified = false; - each(DATA_ZOOM_AXIS_DIMENSIONS, function (axisDim) { - var refering = this.getReferringComponents(getAxisMainType(axisDim), MULTIPLE_REFERRING); // When user set axisIndex as a empty array, we think that user specify axisIndex - // but do not want use auto mode. Because empty array may be encountered when - // some error occurred. - - if (!refering.specified) { - return; - } - - hasAxisSpecified = true; - var axisInfo = new DataZoomAxisInfo(); - each(refering.models, function (axisModel) { - axisInfo.add(axisModel.componentIndex); - }); - targetAxisIndexMap.set(axisDim, axisInfo); - }, this); - return hasAxisSpecified; - }; - - DataZoomModel.prototype._fillAutoTargetAxisByOrient = function (targetAxisIndexMap, orient) { - var ecModel = this.ecModel; - var needAuto = true; // Find axis that parallel to dataZoom as default. - - if (needAuto) { - var axisDim = orient === 'vertical' ? 'y' : 'x'; - var axisModels = ecModel.findComponents({ - mainType: axisDim + 'Axis' - }); - setParallelAxis(axisModels, axisDim); - } // Find axis that parallel to dataZoom as default. - - - if (needAuto) { - var axisModels = ecModel.findComponents({ - mainType: 'singleAxis', - filter: function (axisModel) { - return axisModel.get('orient', true) === orient; - } - }); - setParallelAxis(axisModels, 'single'); - } - - function setParallelAxis(axisModels, axisDim) { - // At least use the first parallel axis as the target axis. - var axisModel = axisModels[0]; - - if (!axisModel) { - return; - } - - var axisInfo = new DataZoomAxisInfo(); - axisInfo.add(axisModel.componentIndex); - targetAxisIndexMap.set(axisDim, axisInfo); - needAuto = false; // Find parallel axes in the same grid. - - if (axisDim === 'x' || axisDim === 'y') { - var gridModel_1 = axisModel.getReferringComponents('grid', SINGLE_REFERRING).models[0]; - gridModel_1 && each(axisModels, function (axModel) { - if (axisModel.componentIndex !== axModel.componentIndex && gridModel_1 === axModel.getReferringComponents('grid', SINGLE_REFERRING).models[0]) { - axisInfo.add(axModel.componentIndex); - } - }); - } - } - - if (needAuto) { - // If no parallel axis, find the first category axis as default. (Also consider polar). - each(DATA_ZOOM_AXIS_DIMENSIONS, function (axisDim) { - if (!needAuto) { - return; - } - - var axisModels = ecModel.findComponents({ - mainType: getAxisMainType(axisDim), - filter: function (axisModel) { - return axisModel.get('type', true) === 'category'; - } - }); - - if (axisModels[0]) { - var axisInfo = new DataZoomAxisInfo(); - axisInfo.add(axisModels[0].componentIndex); - targetAxisIndexMap.set(axisDim, axisInfo); - needAuto = false; - } - }, this); - } - }; - - DataZoomModel.prototype._makeAutoOrientByTargetAxis = function () { - var dim; // Find the first axis - - this.eachTargetAxis(function (axisDim) { - !dim && (dim = axisDim); - }, this); - return dim === 'y' ? 'vertical' : 'horizontal'; - }; - - DataZoomModel.prototype._setDefaultThrottle = function (inputRawOption) { - // When first time user set throttle, auto throttle ends. - if (inputRawOption.hasOwnProperty('throttle')) { - this._autoThrottle = false; - } - - if (this._autoThrottle) { - var globalOption = this.ecModel.option; - this.option.throttle = globalOption.animation && globalOption.animationDurationUpdate > 0 ? 100 : 20; - } - }; - - DataZoomModel.prototype._updateRangeUse = function (inputRawOption) { - var rangePropMode = this._rangePropMode; - var rangeModeInOption = this.get('rangeMode'); - each([['start', 'startValue'], ['end', 'endValue']], function (names, index) { - var percentSpecified = inputRawOption[names[0]] != null; - var valueSpecified = inputRawOption[names[1]] != null; - - if (percentSpecified && !valueSpecified) { - rangePropMode[index] = 'percent'; - } else if (!percentSpecified && valueSpecified) { - rangePropMode[index] = 'value'; - } else if (rangeModeInOption) { - rangePropMode[index] = rangeModeInOption[index]; - } else if (percentSpecified) { - // percentSpecified && valueSpecified - rangePropMode[index] = 'percent'; - } // else remain its original setting. - - }); - }; - - DataZoomModel.prototype.noTarget = function () { - return this._noTarget; - }; - - DataZoomModel.prototype.getFirstTargetAxisModel = function () { - var firstAxisModel; - this.eachTargetAxis(function (axisDim, axisIndex) { - if (firstAxisModel == null) { - firstAxisModel = this.ecModel.getComponent(getAxisMainType(axisDim), axisIndex); - } - }, this); - return firstAxisModel; - }; - /** - * @param {Function} callback param: axisModel, dimNames, axisIndex, dataZoomModel, ecModel - */ - - - DataZoomModel.prototype.eachTargetAxis = function (callback, context) { - this._targetAxisInfoMap.each(function (axisInfo, axisDim) { - each(axisInfo.indexList, function (axisIndex) { - callback.call(context, axisDim, axisIndex); - }); - }); - }; - /** - * @return If not found, return null/undefined. - */ - - - DataZoomModel.prototype.getAxisProxy = function (axisDim, axisIndex) { - var axisModel = this.getAxisModel(axisDim, axisIndex); - - if (axisModel) { - return axisModel.__dzAxisProxy; - } - }; - /** - * @return If not found, return null/undefined. - */ - - - DataZoomModel.prototype.getAxisModel = function (axisDim, axisIndex) { - if ("development" !== 'production') { - assert(axisDim && axisIndex != null); - } - - var axisInfo = this._targetAxisInfoMap.get(axisDim); - - if (axisInfo && axisInfo.indexMap[axisIndex]) { - return this.ecModel.getComponent(getAxisMainType(axisDim), axisIndex); - } - }; - /** - * If not specified, set to undefined. - */ - - - DataZoomModel.prototype.setRawRange = function (opt) { - var thisOption = this.option; - var settledOption = this.settledOption; - each([['start', 'startValue'], ['end', 'endValue']], function (names) { - // Consider the pair <start, startValue>: - // If one has value and the other one is `null/undefined`, we both set them - // to `settledOption`. This strategy enables the feature to clear the original - // value in `settledOption` to `null/undefined`. - // But if both of them are `null/undefined`, we do not set them to `settledOption` - // and keep `settledOption` with the original value. This strategy enables users to - // only set <end or endValue> but not set <start or startValue> when calling - // `dispatchAction`. - // The pair <end, endValue> is treated in the same way. - if (opt[names[0]] != null || opt[names[1]] != null) { - thisOption[names[0]] = settledOption[names[0]] = opt[names[0]]; - thisOption[names[1]] = settledOption[names[1]] = opt[names[1]]; - } - }, this); - - this._updateRangeUse(opt); - }; - - DataZoomModel.prototype.setCalculatedRange = function (opt) { - var option = this.option; - each(['start', 'startValue', 'end', 'endValue'], function (name) { - option[name] = opt[name]; - }); - }; - - DataZoomModel.prototype.getPercentRange = function () { - var axisProxy = this.findRepresentativeAxisProxy(); - - if (axisProxy) { - return axisProxy.getDataPercentWindow(); - } - }; - /** - * For example, chart.getModel().getComponent('dataZoom').getValueRange('y', 0); - * - * @return [startValue, endValue] value can only be '-' or finite number. - */ - - - DataZoomModel.prototype.getValueRange = function (axisDim, axisIndex) { - if (axisDim == null && axisIndex == null) { - var axisProxy = this.findRepresentativeAxisProxy(); - - if (axisProxy) { - return axisProxy.getDataValueWindow(); - } - } else { - return this.getAxisProxy(axisDim, axisIndex).getDataValueWindow(); - } - }; - /** - * @param axisModel If axisModel given, find axisProxy - * corresponding to the axisModel - */ - - - DataZoomModel.prototype.findRepresentativeAxisProxy = function (axisModel) { - if (axisModel) { - return axisModel.__dzAxisProxy; - } // Find the first hosted axisProxy - - - var firstProxy; - - var axisDimList = this._targetAxisInfoMap.keys(); - - for (var i = 0; i < axisDimList.length; i++) { - var axisDim = axisDimList[i]; - - var axisInfo = this._targetAxisInfoMap.get(axisDim); - - for (var j = 0; j < axisInfo.indexList.length; j++) { - var proxy = this.getAxisProxy(axisDim, axisInfo.indexList[j]); - - if (proxy.hostedBy(this)) { - return proxy; - } - - if (!firstProxy) { - firstProxy = proxy; - } - } - } // If no hosted proxy found, still need to return a proxy. - // This case always happens in toolbox dataZoom, where axes are all hosted by - // other dataZooms. - - - return firstProxy; - }; - - DataZoomModel.prototype.getRangePropMode = function () { - return this._rangePropMode.slice(); - }; - - DataZoomModel.prototype.getOrient = function () { - if ("development" !== 'production') { - // Should not be called before initialized. - assert(this._orient); - } - - return this._orient; - }; - - DataZoomModel.type = 'dataZoom'; - DataZoomModel.dependencies = ['xAxis', 'yAxis', 'radiusAxis', 'angleAxis', 'singleAxis', 'series', 'toolbox']; - DataZoomModel.defaultOption = { - // zlevel: 0, - z: 4, - filterMode: 'filter', - start: 0, - end: 100 - }; - return DataZoomModel; - }(ComponentModel); - /** - * Retrieve those raw params from option, which will be cached separately, - * because they will be overwritten by normalized/calculated values in the main - * process. - */ - - - function retrieveRawOption(option) { - var ret = {}; - each(['start', 'end', 'startValue', 'endValue', 'throttle'], function (name) { - option.hasOwnProperty(name) && (ret[name] = option[name]); - }); - return ret; - } - - var SelectDataZoomModel = - /** @class */ - function (_super) { - __extends(SelectDataZoomModel, _super); - - function SelectDataZoomModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SelectDataZoomModel.type; - return _this; - } - - SelectDataZoomModel.type = 'dataZoom.select'; - return SelectDataZoomModel; - }(DataZoomModel); - - var DataZoomView = - /** @class */ - function (_super) { - __extends(DataZoomView, _super); - - function DataZoomView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = DataZoomView.type; - return _this; - } - - DataZoomView.prototype.render = function (dataZoomModel, ecModel, api, payload) { - this.dataZoomModel = dataZoomModel; - this.ecModel = ecModel; - this.api = api; - }; - - DataZoomView.type = 'dataZoom'; - return DataZoomView; - }(ComponentView); - - var SelectDataZoomView = - /** @class */ - function (_super) { - __extends(SelectDataZoomView, _super); - - function SelectDataZoomView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SelectDataZoomView.type; - return _this; - } - - SelectDataZoomView.type = 'dataZoom.select'; - return SelectDataZoomView; - }(DataZoomView); - - var each$8 = each; - var asc$1 = asc; - /** - * Operate single axis. - * One axis can only operated by one axis operator. - * Different dataZoomModels may be defined to operate the same axis. - * (i.e. 'inside' data zoom and 'slider' data zoom components) - * So dataZoomModels share one axisProxy in that case. - */ - - var AxisProxy = - /** @class */ - function () { - function AxisProxy(dimName, axisIndex, dataZoomModel, ecModel) { - this._dimName = dimName; - this._axisIndex = axisIndex; - this.ecModel = ecModel; - this._dataZoomModel = dataZoomModel; // /** - // * @readOnly - // * @private - // */ - // this.hasSeriesStacked; - } - /** - * Whether the axisProxy is hosted by dataZoomModel. - */ - - - AxisProxy.prototype.hostedBy = function (dataZoomModel) { - return this._dataZoomModel === dataZoomModel; - }; - /** - * @return Value can only be NaN or finite value. - */ - - - AxisProxy.prototype.getDataValueWindow = function () { - return this._valueWindow.slice(); - }; - /** - * @return {Array.<number>} - */ - - - AxisProxy.prototype.getDataPercentWindow = function () { - return this._percentWindow.slice(); - }; - - AxisProxy.prototype.getTargetSeriesModels = function () { - var seriesModels = []; - this.ecModel.eachSeries(function (seriesModel) { - if (isCoordSupported(seriesModel)) { - var axisMainType = getAxisMainType(this._dimName); - var axisModel = seriesModel.getReferringComponents(axisMainType, SINGLE_REFERRING).models[0]; - - if (axisModel && this._axisIndex === axisModel.componentIndex) { - seriesModels.push(seriesModel); - } - } - }, this); - return seriesModels; - }; - - AxisProxy.prototype.getAxisModel = function () { - return this.ecModel.getComponent(this._dimName + 'Axis', this._axisIndex); - }; - - AxisProxy.prototype.getMinMaxSpan = function () { - return clone(this._minMaxSpan); - }; - /** - * Only calculate by given range and this._dataExtent, do not change anything. - */ - - - AxisProxy.prototype.calculateDataWindow = function (opt) { - var dataExtent = this._dataExtent; - var axisModel = this.getAxisModel(); - var scale = axisModel.axis.scale; - - var rangePropMode = this._dataZoomModel.getRangePropMode(); - - var percentExtent = [0, 100]; - var percentWindow = []; - var valueWindow = []; - var hasPropModeValue; - each$8(['start', 'end'], function (prop, idx) { - var boundPercent = opt[prop]; - var boundValue = opt[prop + 'Value']; // Notice: dataZoom is based either on `percentProp` ('start', 'end') or - // on `valueProp` ('startValue', 'endValue'). (They are based on the data extent - // but not min/max of axis, which will be calculated by data window then). - // The former one is suitable for cases that a dataZoom component controls multiple - // axes with different unit or extent, and the latter one is suitable for accurate - // zoom by pixel (e.g., in dataZoomSelect). - // we use `getRangePropMode()` to mark which prop is used. `rangePropMode` is updated - // only when setOption or dispatchAction, otherwise it remains its original value. - // (Why not only record `percentProp` and always map to `valueProp`? Because - // the map `valueProp` -> `percentProp` -> `valueProp` probably not the original - // `valueProp`. consider two axes constrolled by one dataZoom. They have different - // data extent. All of values that are overflow the `dataExtent` will be calculated - // to percent '100%'). - - if (rangePropMode[idx] === 'percent') { - boundPercent == null && (boundPercent = percentExtent[idx]); // Use scale.parse to math round for category or time axis. - - boundValue = scale.parse(linearMap(boundPercent, percentExtent, dataExtent)); - } else { - hasPropModeValue = true; - boundValue = boundValue == null ? dataExtent[idx] : scale.parse(boundValue); // Calculating `percent` from `value` may be not accurate, because - // This calculation can not be inversed, because all of values that - // are overflow the `dataExtent` will be calculated to percent '100%' - - boundPercent = linearMap(boundValue, dataExtent, percentExtent); - } // valueWindow[idx] = round(boundValue); - // percentWindow[idx] = round(boundPercent); - // fallback to extent start/end when parsed value or percent is invalid - - - valueWindow[idx] = boundValue == null || isNaN(boundValue) ? dataExtent[idx] : boundValue; - percentWindow[idx] = boundPercent == null || isNaN(boundPercent) ? percentExtent[idx] : boundPercent; - }); - asc$1(valueWindow); - asc$1(percentWindow); // The windows from user calling of `dispatchAction` might be out of the extent, - // or do not obey the `min/maxSpan`, `min/maxValueSpan`. But we don't restrict window - // by `zoomLock` here, because we see `zoomLock` just as a interaction constraint, - // where API is able to initialize/modify the window size even though `zoomLock` - // specified. - - var spans = this._minMaxSpan; - hasPropModeValue ? restrictSet(valueWindow, percentWindow, dataExtent, percentExtent, false) : restrictSet(percentWindow, valueWindow, percentExtent, dataExtent, true); - - function restrictSet(fromWindow, toWindow, fromExtent, toExtent, toValue) { - var suffix = toValue ? 'Span' : 'ValueSpan'; - sliderMove(0, fromWindow, fromExtent, 'all', spans['min' + suffix], spans['max' + suffix]); - - for (var i = 0; i < 2; i++) { - toWindow[i] = linearMap(fromWindow[i], fromExtent, toExtent, true); - toValue && (toWindow[i] = scale.parse(toWindow[i])); - } - } - - return { - valueWindow: valueWindow, - percentWindow: percentWindow - }; - }; - /** - * Notice: reset should not be called before series.restoreData() is called, - * so it is recommended to be called in "process stage" but not "model init - * stage". - */ - - - AxisProxy.prototype.reset = function (dataZoomModel) { - if (dataZoomModel !== this._dataZoomModel) { - return; - } - - var targetSeries = this.getTargetSeriesModels(); // Culculate data window and data extent, and record them. - - this._dataExtent = calculateDataExtent(this, this._dimName, targetSeries); // `calculateDataWindow` uses min/maxSpan. - - this._updateMinMaxSpan(); - - var dataWindow = this.calculateDataWindow(dataZoomModel.settledOption); - this._valueWindow = dataWindow.valueWindow; - this._percentWindow = dataWindow.percentWindow; // Update axis setting then. - - this._setAxisModel(); - }; - - AxisProxy.prototype.filterData = function (dataZoomModel, api) { - if (dataZoomModel !== this._dataZoomModel) { - return; - } - - var axisDim = this._dimName; - var seriesModels = this.getTargetSeriesModels(); - var filterMode = dataZoomModel.get('filterMode'); - var valueWindow = this._valueWindow; - - if (filterMode === 'none') { - return; - } // FIXME - // Toolbox may has dataZoom injected. And if there are stacked bar chart - // with NaN data, NaN will be filtered and stack will be wrong. - // So we need to force the mode to be set empty. - // In fect, it is not a big deal that do not support filterMode-'filter' - // when using toolbox#dataZoom, utill tooltip#dataZoom support "single axis - // selection" some day, which might need "adapt to data extent on the - // otherAxis", which is disabled by filterMode-'empty'. - // But currently, stack has been fixed to based on value but not index, - // so this is not an issue any more. - // let otherAxisModel = this.getOtherAxisModel(); - // if (dataZoomModel.get('$fromToolbox') - // && otherAxisModel - // && otherAxisModel.hasSeriesStacked - // ) { - // filterMode = 'empty'; - // } - // TODO - // filterMode 'weakFilter' and 'empty' is not optimized for huge data yet. - - - each$8(seriesModels, function (seriesModel) { - var seriesData = seriesModel.getData(); - var dataDims = seriesData.mapDimensionsAll(axisDim); - - if (!dataDims.length) { - return; - } - - if (filterMode === 'weakFilter') { - var store_1 = seriesData.getStore(); - var dataDimIndices_1 = map(dataDims, function (dim) { - return seriesData.getDimensionIndex(dim); - }, seriesData); - seriesData.filterSelf(function (dataIndex) { - var leftOut; - var rightOut; - var hasValue; - - for (var i = 0; i < dataDims.length; i++) { - var value = store_1.get(dataDimIndices_1[i], dataIndex); - var thisHasValue = !isNaN(value); - var thisLeftOut = value < valueWindow[0]; - var thisRightOut = value > valueWindow[1]; - - if (thisHasValue && !thisLeftOut && !thisRightOut) { - return true; - } - - thisHasValue && (hasValue = true); - thisLeftOut && (leftOut = true); - thisRightOut && (rightOut = true); - } // If both left out and right out, do not filter. - - - return hasValue && leftOut && rightOut; - }); - } else { - each$8(dataDims, function (dim) { - if (filterMode === 'empty') { - seriesModel.setData(seriesData = seriesData.map(dim, function (value) { - return !isInWindow(value) ? NaN : value; - })); - } else { - var range = {}; - range[dim] = valueWindow; // console.time('select'); - - seriesData.selectRange(range); // console.timeEnd('select'); - } - }); - } - - each$8(dataDims, function (dim) { - seriesData.setApproximateExtent(valueWindow, dim); - }); - }); - - function isInWindow(value) { - return value >= valueWindow[0] && value <= valueWindow[1]; - } - }; - - AxisProxy.prototype._updateMinMaxSpan = function () { - var minMaxSpan = this._minMaxSpan = {}; - var dataZoomModel = this._dataZoomModel; - var dataExtent = this._dataExtent; - each$8(['min', 'max'], function (minMax) { - var percentSpan = dataZoomModel.get(minMax + 'Span'); - var valueSpan = dataZoomModel.get(minMax + 'ValueSpan'); - valueSpan != null && (valueSpan = this.getAxisModel().axis.scale.parse(valueSpan)); // minValueSpan and maxValueSpan has higher priority than minSpan and maxSpan - - if (valueSpan != null) { - percentSpan = linearMap(dataExtent[0] + valueSpan, dataExtent, [0, 100], true); - } else if (percentSpan != null) { - valueSpan = linearMap(percentSpan, [0, 100], dataExtent, true) - dataExtent[0]; - } - - minMaxSpan[minMax + 'Span'] = percentSpan; - minMaxSpan[minMax + 'ValueSpan'] = valueSpan; - }, this); - }; - - AxisProxy.prototype._setAxisModel = function () { - var axisModel = this.getAxisModel(); - var percentWindow = this._percentWindow; - var valueWindow = this._valueWindow; - - if (!percentWindow) { - return; - } // [0, 500]: arbitrary value, guess axis extent. - - - var precision = getPixelPrecision(valueWindow, [0, 500]); - precision = Math.min(precision, 20); // For value axis, if min/max/scale are not set, we just use the extent obtained - // by series data, which may be a little different from the extent calculated by - // `axisHelper.getScaleExtent`. But the different just affects the experience a - // little when zooming. So it will not be fixed until some users require it strongly. - - var rawExtentInfo = axisModel.axis.scale.rawExtentInfo; - - if (percentWindow[0] !== 0) { - rawExtentInfo.setDeterminedMinMax('min', +valueWindow[0].toFixed(precision)); - } - - if (percentWindow[1] !== 100) { - rawExtentInfo.setDeterminedMinMax('max', +valueWindow[1].toFixed(precision)); - } - - rawExtentInfo.freeze(); - }; - - return AxisProxy; - }(); - - function calculateDataExtent(axisProxy, axisDim, seriesModels) { - var dataExtent = [Infinity, -Infinity]; - each$8(seriesModels, function (seriesModel) { - unionAxisExtentFromData(dataExtent, seriesModel.getData(), axisDim); - }); // It is important to get "consistent" extent when more then one axes is - // controlled by a `dataZoom`, otherwise those axes will not be synchronized - // when zooming. But it is difficult to know what is "consistent", considering - // axes have different type or even different meanings (For example, two - // time axes are used to compare data of the same date in different years). - // So basically dataZoom just obtains extent by series.data (in category axis - // extent can be obtained from axis.data). - // Nevertheless, user can set min/max/scale on axes to make extent of axes - // consistent. - - var axisModel = axisProxy.getAxisModel(); - var rawExtentResult = ensureScaleRawExtentInfo(axisModel.axis.scale, axisModel, dataExtent).calculate(); - return [rawExtentResult.min, rawExtentResult.max]; - } - - var dataZoomProcessor = { - // `dataZoomProcessor` will only be performed in needed series. Consider if - // there is a line series and a pie series, it is better not to update the - // line series if only pie series is needed to be updated. - getTargetSeries: function (ecModel) { - function eachAxisModel(cb) { - ecModel.eachComponent('dataZoom', function (dataZoomModel) { - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - var axisModel = ecModel.getComponent(getAxisMainType(axisDim), axisIndex); - cb(axisDim, axisIndex, axisModel, dataZoomModel); - }); - }); - } // FIXME: it brings side-effect to `getTargetSeries`. - // Prepare axis proxies. - - - eachAxisModel(function (axisDim, axisIndex, axisModel, dataZoomModel) { - // dispose all last axis proxy, in case that some axis are deleted. - axisModel.__dzAxisProxy = null; - }); - var proxyList = []; - eachAxisModel(function (axisDim, axisIndex, axisModel, dataZoomModel) { - // Different dataZooms may constrol the same axis. In that case, - // an axisProxy serves both of them. - if (!axisModel.__dzAxisProxy) { - // Use the first dataZoomModel as the main model of axisProxy. - axisModel.__dzAxisProxy = new AxisProxy(axisDim, axisIndex, dataZoomModel, ecModel); - proxyList.push(axisModel.__dzAxisProxy); - } - }); - var seriesModelMap = createHashMap(); - each(proxyList, function (axisProxy) { - each(axisProxy.getTargetSeriesModels(), function (seriesModel) { - seriesModelMap.set(seriesModel.uid, seriesModel); - }); - }); - return seriesModelMap; - }, - // Consider appendData, where filter should be performed. Because data process is - // in block mode currently, it is not need to worry about that the overallProgress - // execute every frame. - overallReset: function (ecModel, api) { - ecModel.eachComponent('dataZoom', function (dataZoomModel) { - // We calculate window and reset axis here but not in model - // init stage and not after action dispatch handler, because - // reset should be called after seriesData.restoreData. - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - dataZoomModel.getAxisProxy(axisDim, axisIndex).reset(dataZoomModel); - }); // Caution: data zoom filtering is order sensitive when using - // percent range and no min/max/scale set on axis. - // For example, we have dataZoom definition: - // [ - // {xAxisIndex: 0, start: 30, end: 70}, - // {yAxisIndex: 0, start: 20, end: 80} - // ] - // In this case, [20, 80] of y-dataZoom should be based on data - // that have filtered by x-dataZoom using range of [30, 70], - // but should not be based on full raw data. Thus sliding - // x-dataZoom will change both ranges of xAxis and yAxis, - // while sliding y-dataZoom will only change the range of yAxis. - // So we should filter x-axis after reset x-axis immediately, - // and then reset y-axis and filter y-axis. - - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - dataZoomModel.getAxisProxy(axisDim, axisIndex).filterData(dataZoomModel, api); - }); - }); - ecModel.eachComponent('dataZoom', function (dataZoomModel) { - // Fullfill all of the range props so that user - // is able to get them from chart.getOption(). - var axisProxy = dataZoomModel.findRepresentativeAxisProxy(); - - if (axisProxy) { - var percentRange = axisProxy.getDataPercentWindow(); - var valueRange = axisProxy.getDataValueWindow(); - dataZoomModel.setCalculatedRange({ - start: percentRange[0], - end: percentRange[1], - startValue: valueRange[0], - endValue: valueRange[1] - }); - } - }); - } - }; - - function installDataZoomAction(registers) { - registers.registerAction('dataZoom', function (payload, ecModel) { - var effectedModels = findEffectedDataZooms(ecModel, payload); - each(effectedModels, function (dataZoomModel) { - dataZoomModel.setRawRange({ - start: payload.start, - end: payload.end, - startValue: payload.startValue, - endValue: payload.endValue - }); - }); - }); - } - - var installed = false; - function installCommon(registers) { - if (installed) { - return; - } - - installed = true; - registers.registerProcessor(registers.PRIORITY.PROCESSOR.FILTER, dataZoomProcessor); - installDataZoomAction(registers); - registers.registerSubTypeDefaulter('dataZoom', function () { - // Default 'slider' when no type specified. - return 'slider'; - }); - } - - function install$y(registers) { - registers.registerComponentModel(SelectDataZoomModel); - registers.registerComponentView(SelectDataZoomView); - installCommon(registers); - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - var ToolboxFeature = - /** @class */ - function () { - function ToolboxFeature() {} - - return ToolboxFeature; - }(); - var features = {}; - function registerFeature(name, ctor) { - features[name] = ctor; - } - function getFeature(name) { - return features[name]; - } - - var ToolboxModel = - /** @class */ - function (_super) { - __extends(ToolboxModel, _super); - - function ToolboxModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ToolboxModel.type; - return _this; - } - - ToolboxModel.prototype.optionUpdated = function () { - _super.prototype.optionUpdated.apply(this, arguments); - - var ecModel = this.ecModel; - each(this.option.feature, function (featureOpt, featureName) { - var Feature = getFeature(featureName); - - if (Feature) { - if (Feature.getDefaultOption) { - Feature.defaultOption = Feature.getDefaultOption(ecModel); - } - - merge(featureOpt, Feature.defaultOption); - } - }); - }; - - ToolboxModel.type = 'toolbox'; - ToolboxModel.layoutMode = { - type: 'box', - ignoreSize: true - }; - ToolboxModel.defaultOption = { - show: true, - z: 6, - // zlevel: 0, - orient: 'horizontal', - left: 'right', - top: 'top', - // right - // bottom - backgroundColor: 'transparent', - borderColor: '#ccc', - borderRadius: 0, - borderWidth: 0, - padding: 5, - itemSize: 15, - itemGap: 8, - showTitle: true, - iconStyle: { - borderColor: '#666', - color: 'none' - }, - emphasis: { - iconStyle: { - borderColor: '#3E98C5' - } - }, - // textStyle: {}, - // feature - tooltip: { - show: false, - position: 'bottom' - } - }; - return ToolboxModel; - }(ComponentModel); - - /** - * Layout list like component. - * It will box layout each items in group of component and then position the whole group in the viewport - * @param {module:zrender/group/Group} group - * @param {module:echarts/model/Component} componentModel - * @param {module:echarts/ExtensionAPI} - */ - - function layout$3(group, componentModel, api) { - var boxLayoutParams = componentModel.getBoxLayoutParams(); - var padding = componentModel.get('padding'); - var viewportSize = { - width: api.getWidth(), - height: api.getHeight() - }; - var rect = getLayoutRect(boxLayoutParams, viewportSize, padding); - box(componentModel.get('orient'), group, componentModel.get('itemGap'), rect.width, rect.height); - positionElement(group, boxLayoutParams, viewportSize, padding); - } - function makeBackground(rect, componentModel) { - var padding = normalizeCssArray$1(componentModel.get('padding')); - var style = componentModel.getItemStyle(['color', 'opacity']); - style.fill = componentModel.get('backgroundColor'); - rect = new Rect({ - shape: { - x: rect.x - padding[3], - y: rect.y - padding[0], - width: rect.width + padding[1] + padding[3], - height: rect.height + padding[0] + padding[2], - r: componentModel.get('borderRadius') - }, - style: style, - silent: true, - z2: -1 - }); // FIXME - // `subPixelOptimizeRect` may bring some gap between edge of viewpart - // and background rect when setting like `left: 0`, `top: 0`. - // graphic.subPixelOptimizeRect(rect); - - return rect; - } - - var ToolboxView = - /** @class */ - function (_super) { - __extends(ToolboxView, _super); - - function ToolboxView() { - return _super !== null && _super.apply(this, arguments) || this; - } - - ToolboxView.prototype.render = function (toolboxModel, ecModel, api, payload) { - var group = this.group; - group.removeAll(); - - if (!toolboxModel.get('show')) { - return; - } - - var itemSize = +toolboxModel.get('itemSize'); - var isVertical = toolboxModel.get('orient') === 'vertical'; - var featureOpts = toolboxModel.get('feature') || {}; - var features = this._features || (this._features = {}); - var featureNames = []; - each(featureOpts, function (opt, name) { - featureNames.push(name); - }); - new DataDiffer(this._featureNames || [], featureNames).add(processFeature).update(processFeature).remove(curry(processFeature, null)).execute(); // Keep for diff. - - this._featureNames = featureNames; - - function processFeature(newIndex, oldIndex) { - var featureName = featureNames[newIndex]; - var oldName = featureNames[oldIndex]; - var featureOpt = featureOpts[featureName]; - var featureModel = new Model(featureOpt, toolboxModel, toolboxModel.ecModel); - var feature; // FIX#11236, merge feature title from MagicType newOption. TODO: consider seriesIndex ? - - if (payload && payload.newTitle != null && payload.featureName === featureName) { - featureOpt.title = payload.newTitle; - } - - if (featureName && !oldName) { - // Create - if (isUserFeatureName(featureName)) { - feature = { - onclick: featureModel.option.onclick, - featureName: featureName - }; - } else { - var Feature = getFeature(featureName); - - if (!Feature) { - return; - } - - feature = new Feature(); - } - - features[featureName] = feature; - } else { - feature = features[oldName]; // If feature does not exist. - - if (!feature) { - return; - } - } - - feature.uid = getUID('toolbox-feature'); - feature.model = featureModel; - feature.ecModel = ecModel; - feature.api = api; - var isToolboxFeature = feature instanceof ToolboxFeature; - - if (!featureName && oldName) { - isToolboxFeature && feature.dispose && feature.dispose(ecModel, api); - return; - } - - if (!featureModel.get('show') || isToolboxFeature && feature.unusable) { - isToolboxFeature && feature.remove && feature.remove(ecModel, api); - return; - } - - createIconPaths(featureModel, feature, featureName); - - featureModel.setIconStatus = function (iconName, status) { - var option = this.option; - var iconPaths = this.iconPaths; - option.iconStatus = option.iconStatus || {}; - option.iconStatus[iconName] = status; - - if (iconPaths[iconName]) { - (status === 'emphasis' ? enterEmphasis : leaveEmphasis)(iconPaths[iconName]); - } - }; - - if (feature instanceof ToolboxFeature) { - if (feature.render) { - feature.render(featureModel, ecModel, api, payload); - } - } - } - - function createIconPaths(featureModel, feature, featureName) { - var iconStyleModel = featureModel.getModel('iconStyle'); - var iconStyleEmphasisModel = featureModel.getModel(['emphasis', 'iconStyle']); // If one feature has multiple icons, they are organized as - // { - // icon: { - // foo: '', - // bar: '' - // }, - // title: { - // foo: '', - // bar: '' - // } - // } - - var icons = feature instanceof ToolboxFeature && feature.getIcons ? feature.getIcons() : featureModel.get('icon'); - var titles = featureModel.get('title') || {}; - var iconsMap; - var titlesMap; - - if (isString(icons)) { - iconsMap = {}; - iconsMap[featureName] = icons; - } else { - iconsMap = icons; - } - - if (isString(titles)) { - titlesMap = {}; - titlesMap[featureName] = titles; - } else { - titlesMap = titles; - } - - var iconPaths = featureModel.iconPaths = {}; - each(iconsMap, function (iconStr, iconName) { - var path = createIcon(iconStr, {}, { - x: -itemSize / 2, - y: -itemSize / 2, - width: itemSize, - height: itemSize - }); // TODO handling image - - path.setStyle(iconStyleModel.getItemStyle()); - var pathEmphasisState = path.ensureState('emphasis'); - pathEmphasisState.style = iconStyleEmphasisModel.getItemStyle(); // Text position calculation - - var textContent = new ZRText({ - style: { - text: titlesMap[iconName], - align: iconStyleEmphasisModel.get('textAlign'), - borderRadius: iconStyleEmphasisModel.get('textBorderRadius'), - padding: iconStyleEmphasisModel.get('textPadding'), - fill: null - }, - ignore: true - }); - path.setTextContent(textContent); - setTooltipConfig({ - el: path, - componentModel: toolboxModel, - itemName: iconName, - formatterParamsExtra: { - title: titlesMap[iconName] - } - }); - path.__title = titlesMap[iconName]; - path.on('mouseover', function () { - // Should not reuse above hoverStyle, which might be modified. - var hoverStyle = iconStyleEmphasisModel.getItemStyle(); - var defaultTextPosition = isVertical ? toolboxModel.get('right') == null && toolboxModel.get('left') !== 'right' ? 'right' : 'left' : toolboxModel.get('bottom') == null && toolboxModel.get('top') !== 'bottom' ? 'bottom' : 'top'; - textContent.setStyle({ - fill: iconStyleEmphasisModel.get('textFill') || hoverStyle.fill || hoverStyle.stroke || '#000', - backgroundColor: iconStyleEmphasisModel.get('textBackgroundColor') - }); - path.setTextConfig({ - position: iconStyleEmphasisModel.get('textPosition') || defaultTextPosition - }); - textContent.ignore = !toolboxModel.get('showTitle'); // Use enterEmphasis and leaveEmphasis provide by ec. - // There are flags managed by the echarts. - - api.enterEmphasis(this); - }).on('mouseout', function () { - if (featureModel.get(['iconStatus', iconName]) !== 'emphasis') { - api.leaveEmphasis(this); - } - - textContent.hide(); - }); - (featureModel.get(['iconStatus', iconName]) === 'emphasis' ? enterEmphasis : leaveEmphasis)(path); - group.add(path); - path.on('click', bind(feature.onclick, feature, ecModel, api, iconName)); - iconPaths[iconName] = path; - }); - } - - layout$3(group, toolboxModel, api); // Render background after group is layout - // FIXME - - group.add(makeBackground(group.getBoundingRect(), toolboxModel)); // Adjust icon title positions to avoid them out of screen - - isVertical || group.eachChild(function (icon) { - var titleText = icon.__title; // const hoverStyle = icon.hoverStyle; - // TODO simplify code? - - var emphasisState = icon.ensureState('emphasis'); - var emphasisTextConfig = emphasisState.textConfig || (emphasisState.textConfig = {}); - var textContent = icon.getTextContent(); - var emphasisTextState = textContent && textContent.ensureState('emphasis'); // May be background element - - if (emphasisTextState && !isFunction(emphasisTextState) && titleText) { - var emphasisTextStyle = emphasisTextState.style || (emphasisTextState.style = {}); - var rect = getBoundingRect(titleText, ZRText.makeFont(emphasisTextStyle)); - var offsetX = icon.x + group.x; - var offsetY = icon.y + group.y + itemSize; - var needPutOnTop = false; - - if (offsetY + rect.height > api.getHeight()) { - emphasisTextConfig.position = 'top'; - needPutOnTop = true; - } - - var topOffset = needPutOnTop ? -5 - rect.height : itemSize + 10; - - if (offsetX + rect.width / 2 > api.getWidth()) { - emphasisTextConfig.position = ['100%', topOffset]; - emphasisTextStyle.align = 'right'; - } else if (offsetX - rect.width / 2 < 0) { - emphasisTextConfig.position = [0, topOffset]; - emphasisTextStyle.align = 'left'; - } - } - }); - }; - - ToolboxView.prototype.updateView = function (toolboxModel, ecModel, api, payload) { - each(this._features, function (feature) { - feature instanceof ToolboxFeature && feature.updateView && feature.updateView(feature.model, ecModel, api, payload); - }); - }; // updateLayout(toolboxModel, ecModel, api, payload) { - // zrUtil.each(this._features, function (feature) { - // feature.updateLayout && feature.updateLayout(feature.model, ecModel, api, payload); - // }); - // }, - - - ToolboxView.prototype.remove = function (ecModel, api) { - each(this._features, function (feature) { - feature instanceof ToolboxFeature && feature.remove && feature.remove(ecModel, api); - }); - this.group.removeAll(); - }; - - ToolboxView.prototype.dispose = function (ecModel, api) { - each(this._features, function (feature) { - feature instanceof ToolboxFeature && feature.dispose && feature.dispose(ecModel, api); - }); - }; - - ToolboxView.type = 'toolbox'; - return ToolboxView; - }(ComponentView); - - function isUserFeatureName(featureName) { - return featureName.indexOf('my') === 0; - } - - /* global window, document */ - - var SaveAsImage = - /** @class */ - function (_super) { - __extends(SaveAsImage, _super); - - function SaveAsImage() { - return _super !== null && _super.apply(this, arguments) || this; - } - - SaveAsImage.prototype.onclick = function (ecModel, api) { - var model = this.model; - var title = model.get('name') || ecModel.get('title.0.text') || 'echarts'; - var isSvg = api.getZr().painter.getType() === 'svg'; - var type = isSvg ? 'svg' : model.get('type', true) || 'png'; - var url = api.getConnectedDataURL({ - type: type, - backgroundColor: model.get('backgroundColor', true) || ecModel.get('backgroundColor') || '#fff', - connectedBackgroundColor: model.get('connectedBackgroundColor'), - excludeComponents: model.get('excludeComponents'), - pixelRatio: model.get('pixelRatio') - }); - var browser = env.browser; // Chrome, Firefox, New Edge - - if (isFunction(MouseEvent) && (browser.newEdge || !browser.ie && !browser.edge)) { - var $a = document.createElement('a'); - $a.download = title + '.' + type; - $a.target = '_blank'; - $a.href = url; - var evt = new MouseEvent('click', { - // some micro front-end framework, window maybe is a Proxy - view: document.defaultView, - bubbles: true, - cancelable: false - }); - $a.dispatchEvent(evt); - } // IE or old Edge - else { - // @ts-ignore - if (window.navigator.msSaveOrOpenBlob || isSvg) { - var parts = url.split(','); // data:[<mime type>][;charset=<charset>][;base64],<encoded data> - - var base64Encoded = parts[0].indexOf('base64') > -1; - var bstr = isSvg // should decode the svg data uri first - ? decodeURIComponent(parts[1]) : parts[1]; // only `atob` when the data uri is encoded with base64 - // otherwise, like `svg` data uri exported by zrender, - // there will be an error, for it's not encoded with base64. - // (just a url-encoded string through `encodeURIComponent`) - - base64Encoded && (bstr = window.atob(bstr)); - var filename = title + '.' + type; // @ts-ignore - - if (window.navigator.msSaveOrOpenBlob) { - var n = bstr.length; - var u8arr = new Uint8Array(n); - - while (n--) { - u8arr[n] = bstr.charCodeAt(n); - } - - var blob = new Blob([u8arr]); // @ts-ignore - - window.navigator.msSaveOrOpenBlob(blob, filename); - } else { - var frame = document.createElement('iframe'); - document.body.appendChild(frame); - var cw = frame.contentWindow; - var doc = cw.document; - doc.open('image/svg+xml', 'replace'); - doc.write(bstr); - doc.close(); - cw.focus(); - doc.execCommand('SaveAs', true, filename); - document.body.removeChild(frame); - } - } else { - var lang = model.get('lang'); - var html = '' + '<body style="margin:0;">' + '<img src="' + url + '" style="max-width:100%;" title="' + (lang && lang[0] || '') + '" />' + '</body>'; - var tab = window.open(); - tab.document.write(html); - tab.document.title = title; - } - } - }; - - SaveAsImage.getDefaultOption = function (ecModel) { - var defaultOption = { - show: true, - icon: 'M4.7,22.9L29.3,45.5L54.7,23.4M4.6,43.6L4.6,58L53.8,58L53.8,43.6M29.2,45.1L29.2,0', - title: ecModel.getLocaleModel().get(['toolbox', 'saveAsImage', 'title']), - type: 'png', - // Default use option.backgroundColor - // backgroundColor: '#fff', - connectedBackgroundColor: '#fff', - name: '', - excludeComponents: ['toolbox'], - // use current pixel ratio of device by default - // pixelRatio: 1, - lang: ecModel.getLocaleModel().get(['toolbox', 'saveAsImage', 'lang']) - }; - return defaultOption; - }; - - return SaveAsImage; - }(ToolboxFeature); - - var INNER_STACK_KEYWORD = '__ec_magicType_stack__'; - var radioTypes = [['line', 'bar'], ['stack']]; - - var MagicType = - /** @class */ - function (_super) { - __extends(MagicType, _super); - - function MagicType() { - return _super !== null && _super.apply(this, arguments) || this; - } - - MagicType.prototype.getIcons = function () { - var model = this.model; - var availableIcons = model.get('icon'); - var icons = {}; - each(model.get('type'), function (type) { - if (availableIcons[type]) { - icons[type] = availableIcons[type]; - } - }); - return icons; - }; - - MagicType.getDefaultOption = function (ecModel) { - var defaultOption = { - show: true, - type: [], - // Icon group - icon: { - line: 'M4.1,28.9h7.1l9.3-22l7.4,38l9.7-19.7l3,12.8h14.9M4.1,58h51.4', - bar: 'M6.7,22.9h10V48h-10V22.9zM24.9,13h10v35h-10V13zM43.2,2h10v46h-10V2zM3.1,58h53.7', - // eslint-disable-next-line - stack: 'M8.2,38.4l-8.4,4.1l30.6,15.3L60,42.5l-8.1-4.1l-21.5,11L8.2,38.4z M51.9,30l-8.1,4.2l-13.4,6.9l-13.9-6.9L8.2,30l-8.4,4.2l8.4,4.2l22.2,11l21.5-11l8.1-4.2L51.9,30z M51.9,21.7l-8.1,4.2L35.7,30l-5.3,2.8L24.9,30l-8.4-4.1l-8.3-4.2l-8.4,4.2L8.2,30l8.3,4.2l13.9,6.9l13.4-6.9l8.1-4.2l8.1-4.1L51.9,21.7zM30.4,2.2L-0.2,17.5l8.4,4.1l8.3,4.2l8.4,4.2l5.5,2.7l5.3-2.7l8.1-4.2l8.1-4.2l8.1-4.1L30.4,2.2z' // jshint ignore:line - - }, - // `line`, `bar`, `stack`, `tiled` - title: ecModel.getLocaleModel().get(['toolbox', 'magicType', 'title']), - option: {}, - seriesIndex: {} - }; - return defaultOption; - }; - - MagicType.prototype.onclick = function (ecModel, api, type) { - var model = this.model; - var seriesIndex = model.get(['seriesIndex', type]); // Not supported magicType - - if (!seriesOptGenreator[type]) { - return; - } - - var newOption = { - series: [] - }; - - var generateNewSeriesTypes = function (seriesModel) { - var seriesType = seriesModel.subType; - var seriesId = seriesModel.id; - var newSeriesOpt = seriesOptGenreator[type](seriesType, seriesId, seriesModel, model); - - if (newSeriesOpt) { - // PENDING If merge original option? - defaults(newSeriesOpt, seriesModel.option); - newOption.series.push(newSeriesOpt); - } // Modify boundaryGap - - - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && coordSys.type === 'cartesian2d' && (type === 'line' || type === 'bar')) { - var categoryAxis = coordSys.getAxesByScale('ordinal')[0]; - - if (categoryAxis) { - var axisDim = categoryAxis.dim; - var axisType = axisDim + 'Axis'; - var axisModel = seriesModel.getReferringComponents(axisType, SINGLE_REFERRING).models[0]; - var axisIndex = axisModel.componentIndex; - newOption[axisType] = newOption[axisType] || []; - - for (var i = 0; i <= axisIndex; i++) { - newOption[axisType][axisIndex] = newOption[axisType][axisIndex] || {}; - } - - newOption[axisType][axisIndex].boundaryGap = type === 'bar'; - } - } - }; - - each(radioTypes, function (radio) { - if (indexOf(radio, type) >= 0) { - each(radio, function (item) { - model.setIconStatus(item, 'normal'); - }); - } - }); - model.setIconStatus(type, 'emphasis'); - ecModel.eachComponent({ - mainType: 'series', - query: seriesIndex == null ? null : { - seriesIndex: seriesIndex - } - }, generateNewSeriesTypes); - var newTitle; - var currentType = type; // Change title of stack - - if (type === 'stack') { - // use titles in model instead of ecModel - // as stack and tiled appears in pair, just flip them - // no need of checking stack state - newTitle = merge({ - stack: model.option.title.tiled, - tiled: model.option.title.stack - }, model.option.title); - - if (model.get(['iconStatus', type]) !== 'emphasis') { - currentType = 'tiled'; - } - } - - api.dispatchAction({ - type: 'changeMagicType', - currentType: currentType, - newOption: newOption, - newTitle: newTitle, - featureName: 'magicType' - }); - }; - - return MagicType; - }(ToolboxFeature); - - var seriesOptGenreator = { - 'line': function (seriesType, seriesId, seriesModel, model) { - if (seriesType === 'bar') { - return merge({ - id: seriesId, - type: 'line', - // Preserve data related option - data: seriesModel.get('data'), - stack: seriesModel.get('stack'), - markPoint: seriesModel.get('markPoint'), - markLine: seriesModel.get('markLine') - }, model.get(['option', 'line']) || {}, true); - } - }, - 'bar': function (seriesType, seriesId, seriesModel, model) { - if (seriesType === 'line') { - return merge({ - id: seriesId, - type: 'bar', - // Preserve data related option - data: seriesModel.get('data'), - stack: seriesModel.get('stack'), - markPoint: seriesModel.get('markPoint'), - markLine: seriesModel.get('markLine') - }, model.get(['option', 'bar']) || {}, true); - } - }, - 'stack': function (seriesType, seriesId, seriesModel, model) { - var isStack = seriesModel.get('stack') === INNER_STACK_KEYWORD; - - if (seriesType === 'line' || seriesType === 'bar') { - model.setIconStatus('stack', isStack ? 'normal' : 'emphasis'); - return merge({ - id: seriesId, - stack: isStack ? '' : INNER_STACK_KEYWORD - }, model.get(['option', 'stack']) || {}, true); - } - } - }; // TODO: SELF REGISTERED. - - registerAction({ - type: 'changeMagicType', - event: 'magicTypeChanged', - update: 'prepareAndUpdate' - }, function (payload, ecModel) { - ecModel.mergeOption(payload.newOption); - }); - - /* global document */ - - var BLOCK_SPLITER = new Array(60).join('-'); - var ITEM_SPLITER = '\t'; - /** - * Group series into two types - * 1. on category axis, like line, bar - * 2. others, like scatter, pie - */ - - function groupSeries(ecModel) { - var seriesGroupByCategoryAxis = {}; - var otherSeries = []; - var meta = []; - ecModel.eachRawSeries(function (seriesModel) { - var coordSys = seriesModel.coordinateSystem; - - if (coordSys && (coordSys.type === 'cartesian2d' || coordSys.type === 'polar')) { - // TODO: TYPE Consider polar? Include polar may increase unecessary bundle size. - var baseAxis = coordSys.getBaseAxis(); - - if (baseAxis.type === 'category') { - var key = baseAxis.dim + '_' + baseAxis.index; - - if (!seriesGroupByCategoryAxis[key]) { - seriesGroupByCategoryAxis[key] = { - categoryAxis: baseAxis, - valueAxis: coordSys.getOtherAxis(baseAxis), - series: [] - }; - meta.push({ - axisDim: baseAxis.dim, - axisIndex: baseAxis.index - }); - } - - seriesGroupByCategoryAxis[key].series.push(seriesModel); - } else { - otherSeries.push(seriesModel); - } - } else { - otherSeries.push(seriesModel); - } - }); - return { - seriesGroupByCategoryAxis: seriesGroupByCategoryAxis, - other: otherSeries, - meta: meta - }; - } - /** - * Assemble content of series on cateogory axis - * @inner - */ - - - function assembleSeriesWithCategoryAxis(groups) { - var tables = []; - each(groups, function (group, key) { - var categoryAxis = group.categoryAxis; - var valueAxis = group.valueAxis; - var valueAxisDim = valueAxis.dim; - var headers = [' '].concat(map(group.series, function (series) { - return series.name; - })); // @ts-ignore TODO Polar - - var columns = [categoryAxis.model.getCategories()]; - each(group.series, function (series) { - var rawData = series.getRawData(); - columns.push(series.getRawData().mapArray(rawData.mapDimension(valueAxisDim), function (val) { - return val; - })); - }); // Assemble table content - - var lines = [headers.join(ITEM_SPLITER)]; - - for (var i = 0; i < columns[0].length; i++) { - var items = []; - - for (var j = 0; j < columns.length; j++) { - items.push(columns[j][i]); - } - - lines.push(items.join(ITEM_SPLITER)); - } - - tables.push(lines.join('\n')); - }); - return tables.join('\n\n' + BLOCK_SPLITER + '\n\n'); - } - /** - * Assemble content of other series - */ - - - function assembleOtherSeries(series) { - return map(series, function (series) { - var data = series.getRawData(); - var lines = [series.name]; - var vals = []; - data.each(data.dimensions, function () { - var argLen = arguments.length; - var dataIndex = arguments[argLen - 1]; - var name = data.getName(dataIndex); - - for (var i = 0; i < argLen - 1; i++) { - vals[i] = arguments[i]; - } - - lines.push((name ? name + ITEM_SPLITER : '') + vals.join(ITEM_SPLITER)); - }); - return lines.join('\n'); - }).join('\n\n' + BLOCK_SPLITER + '\n\n'); - } - - function getContentFromModel(ecModel) { - var result = groupSeries(ecModel); - return { - value: filter([assembleSeriesWithCategoryAxis(result.seriesGroupByCategoryAxis), assembleOtherSeries(result.other)], function (str) { - return !!str.replace(/[\n\t\s]/g, ''); - }).join('\n\n' + BLOCK_SPLITER + '\n\n'), - meta: result.meta - }; - } - - function trim$1(str) { - return str.replace(/^\s\s*/, '').replace(/\s\s*$/, ''); - } - /** - * If a block is tsv format - */ - - - function isTSVFormat(block) { - // Simple method to find out if a block is tsv format - var firstLine = block.slice(0, block.indexOf('\n')); - - if (firstLine.indexOf(ITEM_SPLITER) >= 0) { - return true; - } - } - - var itemSplitRegex = new RegExp('[' + ITEM_SPLITER + ']+', 'g'); - /** - * @param {string} tsv - * @return {Object} - */ - - function parseTSVContents(tsv) { - var tsvLines = tsv.split(/\n+/g); - var headers = trim$1(tsvLines.shift()).split(itemSplitRegex); - var categories = []; - var series = map(headers, function (header) { - return { - name: header, - data: [] - }; - }); - - for (var i = 0; i < tsvLines.length; i++) { - var items = trim$1(tsvLines[i]).split(itemSplitRegex); - categories.push(items.shift()); - - for (var j = 0; j < items.length; j++) { - series[j] && (series[j].data[i] = items[j]); - } - } - - return { - series: series, - categories: categories - }; - } - - function parseListContents(str) { - var lines = str.split(/\n+/g); - var seriesName = trim$1(lines.shift()); - var data = []; - - for (var i = 0; i < lines.length; i++) { - // if line is empty, ignore it. - // there is a case that a user forgot to delete `\n`. - var line = trim$1(lines[i]); - - if (!line) { - continue; - } - - var items = line.split(itemSplitRegex); - var name_1 = ''; - var value = void 0; - var hasName = false; - - if (isNaN(items[0])) { - // First item is name - hasName = true; - name_1 = items[0]; - items = items.slice(1); - data[i] = { - name: name_1, - value: [] - }; - value = data[i].value; - } else { - value = data[i] = []; - } - - for (var j = 0; j < items.length; j++) { - value.push(+items[j]); - } - - if (value.length === 1) { - hasName ? data[i].value = value[0] : data[i] = value[0]; - } - } - - return { - name: seriesName, - data: data - }; - } - - function parseContents(str, blockMetaList) { - var blocks = str.split(new RegExp('\n*' + BLOCK_SPLITER + '\n*', 'g')); - var newOption = { - series: [] - }; - each(blocks, function (block, idx) { - if (isTSVFormat(block)) { - var result = parseTSVContents(block); - var blockMeta = blockMetaList[idx]; - var axisKey = blockMeta.axisDim + 'Axis'; - - if (blockMeta) { - newOption[axisKey] = newOption[axisKey] || []; - newOption[axisKey][blockMeta.axisIndex] = { - data: result.categories - }; - newOption.series = newOption.series.concat(result.series); - } - } else { - var result = parseListContents(block); - newOption.series.push(result); - } - }); - return newOption; - } - - var DataView = - /** @class */ - function (_super) { - __extends(DataView, _super); - - function DataView() { - return _super !== null && _super.apply(this, arguments) || this; - } - - DataView.prototype.onclick = function (ecModel, api) { - // FIXME: better way? - setTimeout(function () { - api.dispatchAction({ - type: 'hideTip' - }); - }); - var container = api.getDom(); - var model = this.model; - - if (this._dom) { - container.removeChild(this._dom); - } - - var root = document.createElement('div'); // use padding to avoid 5px whitespace - - root.style.cssText = 'position:absolute;top:0;bottom:0;left:0;right:0;padding:5px'; - root.style.backgroundColor = model.get('backgroundColor') || '#fff'; // Create elements - - var header = document.createElement('h4'); - var lang = model.get('lang') || []; - header.innerHTML = lang[0] || model.get('title'); - header.style.cssText = 'margin:10px 20px'; - header.style.color = model.get('textColor'); - var viewMain = document.createElement('div'); - var textarea = document.createElement('textarea'); - viewMain.style.cssText = 'overflow:auto'; - var optionToContent = model.get('optionToContent'); - var contentToOption = model.get('contentToOption'); - var result = getContentFromModel(ecModel); - - if (isFunction(optionToContent)) { - var htmlOrDom = optionToContent(api.getOption()); - - if (isString(htmlOrDom)) { - viewMain.innerHTML = htmlOrDom; - } else if (isDom(htmlOrDom)) { - viewMain.appendChild(htmlOrDom); - } - } else { - // Use default textarea - textarea.readOnly = model.get('readOnly'); - var style = textarea.style; // eslint-disable-next-line max-len - - style.cssText = 'display:block;width:100%;height:100%;font-family:monospace;font-size:14px;line-height:1.6rem;resize:none;box-sizing:border-box;outline:none'; - style.color = model.get('textColor'); - style.borderColor = model.get('textareaBorderColor'); - style.backgroundColor = model.get('textareaColor'); - textarea.value = result.value; - viewMain.appendChild(textarea); - } - - var blockMetaList = result.meta; - var buttonContainer = document.createElement('div'); - buttonContainer.style.cssText = 'position:absolute;bottom:5px;left:0;right:0'; // eslint-disable-next-line max-len - - var buttonStyle = 'float:right;margin-right:20px;border:none;cursor:pointer;padding:2px 5px;font-size:12px;border-radius:3px'; - var closeButton = document.createElement('div'); - var refreshButton = document.createElement('div'); - buttonStyle += ';background-color:' + model.get('buttonColor'); - buttonStyle += ';color:' + model.get('buttonTextColor'); - var self = this; - - function close() { - container.removeChild(root); - self._dom = null; - } - - addEventListener(closeButton, 'click', close); - addEventListener(refreshButton, 'click', function () { - if (contentToOption == null && optionToContent != null || contentToOption != null && optionToContent == null) { - if ("development" !== 'production') { - // eslint-disable-next-line - warn('It seems you have just provided one of `contentToOption` and `optionToContent` functions but missed the other one. Data change is ignored.'); - } - - close(); - return; - } - - var newOption; - - try { - if (isFunction(contentToOption)) { - newOption = contentToOption(viewMain, api.getOption()); - } else { - newOption = parseContents(textarea.value, blockMetaList); - } - } catch (e) { - close(); - throw new Error('Data view format error ' + e); - } - - if (newOption) { - api.dispatchAction({ - type: 'changeDataView', - newOption: newOption - }); - } - - close(); - }); - closeButton.innerHTML = lang[1]; - refreshButton.innerHTML = lang[2]; - refreshButton.style.cssText = closeButton.style.cssText = buttonStyle; - !model.get('readOnly') && buttonContainer.appendChild(refreshButton); - buttonContainer.appendChild(closeButton); - root.appendChild(header); - root.appendChild(viewMain); - root.appendChild(buttonContainer); - viewMain.style.height = container.clientHeight - 80 + 'px'; - container.appendChild(root); - this._dom = root; - }; - - DataView.prototype.remove = function (ecModel, api) { - this._dom && api.getDom().removeChild(this._dom); - }; - - DataView.prototype.dispose = function (ecModel, api) { - this.remove(ecModel, api); - }; - - DataView.getDefaultOption = function (ecModel) { - var defaultOption = { - show: true, - readOnly: false, - optionToContent: null, - contentToOption: null, - // eslint-disable-next-line - icon: 'M17.5,17.3H33 M17.5,17.3H33 M45.4,29.5h-28 M11.5,2v56H51V14.8L38.4,2H11.5z M38.4,2.2v12.7H51 M45.4,41.7h-28', - title: ecModel.getLocaleModel().get(['toolbox', 'dataView', 'title']), - lang: ecModel.getLocaleModel().get(['toolbox', 'dataView', 'lang']), - backgroundColor: '#fff', - textColor: '#000', - textareaColor: '#fff', - textareaBorderColor: '#333', - buttonColor: '#c23531', - buttonTextColor: '#fff' - }; - return defaultOption; - }; - - return DataView; - }(ToolboxFeature); - /** - * @inner - */ - - - function tryMergeDataOption(newData, originalData) { - return map(newData, function (newVal, idx) { - var original = originalData && originalData[idx]; - - if (isObject(original) && !isArray(original)) { - var newValIsObject = isObject(newVal) && !isArray(newVal); - - if (!newValIsObject) { - newVal = { - value: newVal - }; - } // original data has name but new data has no name - - - var shouldDeleteName = original.name != null && newVal.name == null; // Original data has option - - newVal = defaults(newVal, original); - shouldDeleteName && delete newVal.name; - return newVal; - } else { - return newVal; - } - }); - } // TODO: SELF REGISTERED. - - - registerAction({ - type: 'changeDataView', - event: 'dataViewChanged', - update: 'prepareAndUpdate' - }, function (payload, ecModel) { - var newSeriesOptList = []; - each(payload.newOption.series, function (seriesOpt) { - var seriesModel = ecModel.getSeriesByName(seriesOpt.name)[0]; - - if (!seriesModel) { - // New created series - // Geuss the series type - newSeriesOptList.push(extend({ - // Default is scatter - type: 'scatter' - }, seriesOpt)); - } else { - var originalData = seriesModel.get('data'); - newSeriesOptList.push({ - name: seriesOpt.name, - data: tryMergeDataOption(seriesOpt.data, originalData) - }); - } - }); - ecModel.mergeOption(defaults({ - series: newSeriesOptList - }, payload.newOption)); - }); - - var each$9 = each; - var inner$f = makeInner(); - /** - * @param ecModel - * @param newSnapshot key is dataZoomId - */ - - function push(ecModel, newSnapshot) { - var storedSnapshots = getStoreSnapshots(ecModel); // If previous dataZoom can not be found, - // complete an range with current range. - - each$9(newSnapshot, function (batchItem, dataZoomId) { - var i = storedSnapshots.length - 1; - - for (; i >= 0; i--) { - var snapshot = storedSnapshots[i]; - - if (snapshot[dataZoomId]) { - break; - } - } - - if (i < 0) { - // No origin range set, create one by current range. - var dataZoomModel = ecModel.queryComponents({ - mainType: 'dataZoom', - subType: 'select', - id: dataZoomId - })[0]; - - if (dataZoomModel) { - var percentRange = dataZoomModel.getPercentRange(); - storedSnapshots[0][dataZoomId] = { - dataZoomId: dataZoomId, - start: percentRange[0], - end: percentRange[1] - }; - } - } - }); - storedSnapshots.push(newSnapshot); - } - function pop(ecModel) { - var storedSnapshots = getStoreSnapshots(ecModel); - var head = storedSnapshots[storedSnapshots.length - 1]; - storedSnapshots.length > 1 && storedSnapshots.pop(); // Find top for all dataZoom. - - var snapshot = {}; - each$9(head, function (batchItem, dataZoomId) { - for (var i = storedSnapshots.length - 1; i >= 0; i--) { - batchItem = storedSnapshots[i][dataZoomId]; - - if (batchItem) { - snapshot[dataZoomId] = batchItem; - break; - } - } - }); - return snapshot; - } - function clear$1(ecModel) { - inner$f(ecModel).snapshots = null; - } - function count(ecModel) { - return getStoreSnapshots(ecModel).length; - } - /** - * History length of each dataZoom may be different. - * this._history[0] is used to store origin range. - */ - - function getStoreSnapshots(ecModel) { - var store = inner$f(ecModel); - - if (!store.snapshots) { - store.snapshots = [{}]; - } - - return store.snapshots; - } - - var RestoreOption = - /** @class */ - function (_super) { - __extends(RestoreOption, _super); - - function RestoreOption() { - return _super !== null && _super.apply(this, arguments) || this; - } - - RestoreOption.prototype.onclick = function (ecModel, api) { - clear$1(ecModel); - api.dispatchAction({ - type: 'restore', - from: this.uid - }); - }; - - RestoreOption.getDefaultOption = function (ecModel) { - var defaultOption = { - show: true, - // eslint-disable-next-line - icon: 'M3.8,33.4 M47,18.9h9.8V8.7 M56.3,20.1 C52.1,9,40.5,0.6,26.8,2.1C12.6,3.7,1.6,16.2,2.1,30.6 M13,41.1H3.1v10.2 M3.7,39.9c4.2,11.1,15.8,19.5,29.5,18 c14.2-1.6,25.2-14.1,24.7-28.5', - title: ecModel.getLocaleModel().get(['toolbox', 'restore', 'title']) - }; - return defaultOption; - }; - - return RestoreOption; - }(ToolboxFeature); // TODO: SELF REGISTERED. - - - registerAction({ - type: 'restore', - event: 'restore', - update: 'prepareAndUpdate' - }, function (payload, ecModel) { - ecModel.resetOption('recreate'); - }); - - // how to genarialize to more coordinate systems. - - var INCLUDE_FINDER_MAIN_TYPES = ['grid', 'xAxis', 'yAxis', 'geo', 'graph', 'polar', 'radiusAxis', 'angleAxis', 'bmap']; - - var BrushTargetManager = - /** @class */ - function () { - /** - * @param finder contains Index/Id/Name of xAxis/yAxis/geo/grid - * Each can be {number|Array.<number>}. like: {xAxisIndex: [3, 4]} - * @param opt.include include coordinate system types. - */ - function BrushTargetManager(finder, ecModel, opt) { - var _this = this; - - this._targetInfoList = []; - var foundCpts = parseFinder$1(ecModel, finder); - each(targetInfoBuilders, function (builder, type) { - if (!opt || !opt.include || indexOf(opt.include, type) >= 0) { - builder(foundCpts, _this._targetInfoList); - } - }); - } - - BrushTargetManager.prototype.setOutputRanges = function (areas, ecModel) { - this.matchOutputRanges(areas, ecModel, function (area, coordRange, coordSys) { - (area.coordRanges || (area.coordRanges = [])).push(coordRange); // area.coordRange is the first of area.coordRanges - - if (!area.coordRange) { - area.coordRange = coordRange; // In 'category' axis, coord to pixel is not reversible, so we can not - // rebuild range by coordRange accrately, which may bring trouble when - // brushing only one item. So we use __rangeOffset to rebuilding range - // by coordRange. And this it only used in brush component so it is no - // need to be adapted to coordRanges. - - var result = coordConvert[area.brushType](0, coordSys, coordRange); - area.__rangeOffset = { - offset: diffProcessor[area.brushType](result.values, area.range, [1, 1]), - xyMinMax: result.xyMinMax - }; - } - }); - return areas; - }; - - BrushTargetManager.prototype.matchOutputRanges = function (areas, ecModel, cb) { - each(areas, function (area) { - var targetInfo = this.findTargetInfo(area, ecModel); - - if (targetInfo && targetInfo !== true) { - each(targetInfo.coordSyses, function (coordSys) { - var result = coordConvert[area.brushType](1, coordSys, area.range, true); - cb(area, result.values, coordSys, ecModel); - }); - } - }, this); - }; - /** - * the `areas` is `BrushModel.areas`. - * Called in layout stage. - * convert `area.coordRange` to global range and set panelId to `area.range`. - */ - - - BrushTargetManager.prototype.setInputRanges = function (areas, ecModel) { - each(areas, function (area) { - var targetInfo = this.findTargetInfo(area, ecModel); - - if ("development" !== 'production') { - assert(!targetInfo || targetInfo === true || area.coordRange, 'coordRange must be specified when coord index specified.'); - assert(!targetInfo || targetInfo !== true || area.range, 'range must be specified in global brush.'); - } - - area.range = area.range || []; // convert coordRange to global range and set panelId. - - if (targetInfo && targetInfo !== true) { - area.panelId = targetInfo.panelId; // (1) area.range should always be calculate from coordRange but does - // not keep its original value, for the sake of the dataZoom scenario, - // where area.coordRange remains unchanged but area.range may be changed. - // (2) Only support converting one coordRange to pixel range in brush - // component. So do not consider `coordRanges`. - // (3) About __rangeOffset, see comment above. - - var result = coordConvert[area.brushType](0, targetInfo.coordSys, area.coordRange); - var rangeOffset = area.__rangeOffset; - area.range = rangeOffset ? diffProcessor[area.brushType](result.values, rangeOffset.offset, getScales(result.xyMinMax, rangeOffset.xyMinMax)) : result.values; - } - }, this); - }; - - BrushTargetManager.prototype.makePanelOpts = function (api, getDefaultBrushType) { - return map(this._targetInfoList, function (targetInfo) { - var rect = targetInfo.getPanelRect(); - return { - panelId: targetInfo.panelId, - defaultBrushType: getDefaultBrushType ? getDefaultBrushType(targetInfo) : null, - clipPath: makeRectPanelClipPath(rect), - isTargetByCursor: makeRectIsTargetByCursor(rect, api, targetInfo.coordSysModel), - getLinearBrushOtherExtent: makeLinearBrushOtherExtent(rect) - }; - }); - }; - - BrushTargetManager.prototype.controlSeries = function (area, seriesModel, ecModel) { - // Check whether area is bound in coord, and series do not belong to that coord. - // If do not do this check, some brush (like lineX) will controll all axes. - var targetInfo = this.findTargetInfo(area, ecModel); - return targetInfo === true || targetInfo && indexOf(targetInfo.coordSyses, seriesModel.coordinateSystem) >= 0; - }; - /** - * If return Object, a coord found. - * If return true, global found. - * Otherwise nothing found. - */ - - - BrushTargetManager.prototype.findTargetInfo = function (area, ecModel) { - var targetInfoList = this._targetInfoList; - var foundCpts = parseFinder$1(ecModel, area); - - for (var i = 0; i < targetInfoList.length; i++) { - var targetInfo = targetInfoList[i]; - var areaPanelId = area.panelId; - - if (areaPanelId) { - if (targetInfo.panelId === areaPanelId) { - return targetInfo; - } - } else { - for (var j = 0; j < targetInfoMatchers.length; j++) { - if (targetInfoMatchers[j](foundCpts, targetInfo)) { - return targetInfo; - } - } - } - } - - return true; - }; - - return BrushTargetManager; - }(); - - function formatMinMax(minMax) { - minMax[0] > minMax[1] && minMax.reverse(); - return minMax; - } - - function parseFinder$1(ecModel, finder) { - return parseFinder(ecModel, finder, { - includeMainTypes: INCLUDE_FINDER_MAIN_TYPES - }); - } - - var targetInfoBuilders = { - grid: function (foundCpts, targetInfoList) { - var xAxisModels = foundCpts.xAxisModels; - var yAxisModels = foundCpts.yAxisModels; - var gridModels = foundCpts.gridModels; // Remove duplicated. - - var gridModelMap = createHashMap(); - var xAxesHas = {}; - var yAxesHas = {}; - - if (!xAxisModels && !yAxisModels && !gridModels) { - return; - } - - each(xAxisModels, function (axisModel) { - var gridModel = axisModel.axis.grid.model; - gridModelMap.set(gridModel.id, gridModel); - xAxesHas[gridModel.id] = true; - }); - each(yAxisModels, function (axisModel) { - var gridModel = axisModel.axis.grid.model; - gridModelMap.set(gridModel.id, gridModel); - yAxesHas[gridModel.id] = true; - }); - each(gridModels, function (gridModel) { - gridModelMap.set(gridModel.id, gridModel); - xAxesHas[gridModel.id] = true; - yAxesHas[gridModel.id] = true; - }); - gridModelMap.each(function (gridModel) { - var grid = gridModel.coordinateSystem; - var cartesians = []; - each(grid.getCartesians(), function (cartesian, index) { - if (indexOf(xAxisModels, cartesian.getAxis('x').model) >= 0 || indexOf(yAxisModels, cartesian.getAxis('y').model) >= 0) { - cartesians.push(cartesian); - } - }); - targetInfoList.push({ - panelId: 'grid--' + gridModel.id, - gridModel: gridModel, - coordSysModel: gridModel, - // Use the first one as the representitive coordSys. - coordSys: cartesians[0], - coordSyses: cartesians, - getPanelRect: panelRectBuilders.grid, - xAxisDeclared: xAxesHas[gridModel.id], - yAxisDeclared: yAxesHas[gridModel.id] - }); - }); - }, - geo: function (foundCpts, targetInfoList) { - each(foundCpts.geoModels, function (geoModel) { - var coordSys = geoModel.coordinateSystem; - targetInfoList.push({ - panelId: 'geo--' + geoModel.id, - geoModel: geoModel, - coordSysModel: geoModel, - coordSys: coordSys, - coordSyses: [coordSys], - getPanelRect: panelRectBuilders.geo - }); - }); - } - }; - var targetInfoMatchers = [// grid - function (foundCpts, targetInfo) { - var xAxisModel = foundCpts.xAxisModel; - var yAxisModel = foundCpts.yAxisModel; - var gridModel = foundCpts.gridModel; - !gridModel && xAxisModel && (gridModel = xAxisModel.axis.grid.model); - !gridModel && yAxisModel && (gridModel = yAxisModel.axis.grid.model); - return gridModel && gridModel === targetInfo.gridModel; - }, // geo - function (foundCpts, targetInfo) { - var geoModel = foundCpts.geoModel; - return geoModel && geoModel === targetInfo.geoModel; - }]; - var panelRectBuilders = { - grid: function () { - // grid is not Transformable. - return this.coordSys.master.getRect().clone(); - }, - geo: function () { - var coordSys = this.coordSys; - var rect = coordSys.getBoundingRect().clone(); // geo roam and zoom transform - - rect.applyTransform(getTransform(coordSys)); - return rect; - } - }; - var coordConvert = { - lineX: curry(axisConvert, 0), - lineY: curry(axisConvert, 1), - rect: function (to, coordSys, rangeOrCoordRange, clamp) { - var xminymin = to ? coordSys.pointToData([rangeOrCoordRange[0][0], rangeOrCoordRange[1][0]], clamp) : coordSys.dataToPoint([rangeOrCoordRange[0][0], rangeOrCoordRange[1][0]], clamp); - var xmaxymax = to ? coordSys.pointToData([rangeOrCoordRange[0][1], rangeOrCoordRange[1][1]], clamp) : coordSys.dataToPoint([rangeOrCoordRange[0][1], rangeOrCoordRange[1][1]], clamp); - var values = [formatMinMax([xminymin[0], xmaxymax[0]]), formatMinMax([xminymin[1], xmaxymax[1]])]; - return { - values: values, - xyMinMax: values - }; - }, - polygon: function (to, coordSys, rangeOrCoordRange, clamp) { - var xyMinMax = [[Infinity, -Infinity], [Infinity, -Infinity]]; - var values = map(rangeOrCoordRange, function (item) { - var p = to ? coordSys.pointToData(item, clamp) : coordSys.dataToPoint(item, clamp); - xyMinMax[0][0] = Math.min(xyMinMax[0][0], p[0]); - xyMinMax[1][0] = Math.min(xyMinMax[1][0], p[1]); - xyMinMax[0][1] = Math.max(xyMinMax[0][1], p[0]); - xyMinMax[1][1] = Math.max(xyMinMax[1][1], p[1]); - return p; - }); - return { - values: values, - xyMinMax: xyMinMax - }; - } - }; - - function axisConvert(axisNameIndex, to, coordSys, rangeOrCoordRange) { - if ("development" !== 'production') { - assert(coordSys.type === 'cartesian2d', 'lineX/lineY brush is available only in cartesian2d.'); - } - - var axis = coordSys.getAxis(['x', 'y'][axisNameIndex]); - var values = formatMinMax(map([0, 1], function (i) { - return to ? axis.coordToData(axis.toLocalCoord(rangeOrCoordRange[i]), true) : axis.toGlobalCoord(axis.dataToCoord(rangeOrCoordRange[i])); - })); - var xyMinMax = []; - xyMinMax[axisNameIndex] = values; - xyMinMax[1 - axisNameIndex] = [NaN, NaN]; - return { - values: values, - xyMinMax: xyMinMax - }; - } - - var diffProcessor = { - lineX: curry(axisDiffProcessor, 0), - lineY: curry(axisDiffProcessor, 1), - rect: function (values, refer, scales) { - return [[values[0][0] - scales[0] * refer[0][0], values[0][1] - scales[0] * refer[0][1]], [values[1][0] - scales[1] * refer[1][0], values[1][1] - scales[1] * refer[1][1]]]; - }, - polygon: function (values, refer, scales) { - return map(values, function (item, idx) { - return [item[0] - scales[0] * refer[idx][0], item[1] - scales[1] * refer[idx][1]]; - }); - } - }; - - function axisDiffProcessor(axisNameIndex, values, refer, scales) { - return [values[0] - scales[axisNameIndex] * refer[0], values[1] - scales[axisNameIndex] * refer[1]]; - } // We have to process scale caused by dataZoom manually, - // although it might be not accurate. - // Return [0~1, 0~1] - - - function getScales(xyMinMaxCurr, xyMinMaxOrigin) { - var sizeCurr = getSize$1(xyMinMaxCurr); - var sizeOrigin = getSize$1(xyMinMaxOrigin); - var scales = [sizeCurr[0] / sizeOrigin[0], sizeCurr[1] / sizeOrigin[1]]; - isNaN(scales[0]) && (scales[0] = 1); - isNaN(scales[1]) && (scales[1] = 1); - return scales; - } - - function getSize$1(xyMinMax) { - return xyMinMax ? [xyMinMax[0][1] - xyMinMax[0][0], xyMinMax[1][1] - xyMinMax[1][0]] : [NaN, NaN]; - } - - var each$a = each; - var DATA_ZOOM_ID_BASE = makeInternalComponentId('toolbox-dataZoom_'); - - var DataZoomFeature = - /** @class */ - function (_super) { - __extends(DataZoomFeature, _super); - - function DataZoomFeature() { - return _super !== null && _super.apply(this, arguments) || this; - } - - DataZoomFeature.prototype.render = function (featureModel, ecModel, api, payload) { - if (!this._brushController) { - this._brushController = new BrushController(api.getZr()); - - this._brushController.on('brush', bind(this._onBrush, this)).mount(); - } - - updateZoomBtnStatus(featureModel, ecModel, this, payload, api); - updateBackBtnStatus(featureModel, ecModel); - }; - - DataZoomFeature.prototype.onclick = function (ecModel, api, type) { - handlers$1[type].call(this); - }; - - DataZoomFeature.prototype.remove = function (ecModel, api) { - this._brushController && this._brushController.unmount(); - }; - - DataZoomFeature.prototype.dispose = function (ecModel, api) { - this._brushController && this._brushController.dispose(); - }; - - DataZoomFeature.prototype._onBrush = function (eventParam) { - var areas = eventParam.areas; - - if (!eventParam.isEnd || !areas.length) { - return; - } - - var snapshot = {}; - var ecModel = this.ecModel; - - this._brushController.updateCovers([]); // remove cover - - - var brushTargetManager = new BrushTargetManager(makeAxisFinder(this.model), ecModel, { - include: ['grid'] - }); - brushTargetManager.matchOutputRanges(areas, ecModel, function (area, coordRange, coordSys) { - if (coordSys.type !== 'cartesian2d') { - return; - } - - var brushType = area.brushType; - - if (brushType === 'rect') { - setBatch('x', coordSys, coordRange[0]); - setBatch('y', coordSys, coordRange[1]); - } else { - setBatch({ - lineX: 'x', - lineY: 'y' - }[brushType], coordSys, coordRange); - } - }); - push(ecModel, snapshot); - - this._dispatchZoomAction(snapshot); - - function setBatch(dimName, coordSys, minMax) { - var axis = coordSys.getAxis(dimName); - var axisModel = axis.model; - var dataZoomModel = findDataZoom(dimName, axisModel, ecModel); // Restrict range. - - var minMaxSpan = dataZoomModel.findRepresentativeAxisProxy(axisModel).getMinMaxSpan(); - - if (minMaxSpan.minValueSpan != null || minMaxSpan.maxValueSpan != null) { - minMax = sliderMove(0, minMax.slice(), axis.scale.getExtent(), 0, minMaxSpan.minValueSpan, minMaxSpan.maxValueSpan); - } - - dataZoomModel && (snapshot[dataZoomModel.id] = { - dataZoomId: dataZoomModel.id, - startValue: minMax[0], - endValue: minMax[1] - }); - } - - function findDataZoom(dimName, axisModel, ecModel) { - var found; - ecModel.eachComponent({ - mainType: 'dataZoom', - subType: 'select' - }, function (dzModel) { - var has = dzModel.getAxisModel(dimName, axisModel.componentIndex); - has && (found = dzModel); - }); - return found; - } - }; - - DataZoomFeature.prototype._dispatchZoomAction = function (snapshot) { - var batch = []; // Convert from hash map to array. - - each$a(snapshot, function (batchItem, dataZoomId) { - batch.push(clone(batchItem)); - }); - batch.length && this.api.dispatchAction({ - type: 'dataZoom', - from: this.uid, - batch: batch - }); - }; - - DataZoomFeature.getDefaultOption = function (ecModel) { - var defaultOption = { - show: true, - filterMode: 'filter', - // Icon group - icon: { - zoom: 'M0,13.5h26.9 M13.5,26.9V0 M32.1,13.5H58V58H13.5 V32.1', - back: 'M22,1.4L9.9,13.5l12.3,12.3 M10.3,13.5H54.9v44.6 H10.3v-26' - }, - // `zoom`, `back` - title: ecModel.getLocaleModel().get(['toolbox', 'dataZoom', 'title']), - brushStyle: { - borderWidth: 0, - color: 'rgba(210,219,238,0.2)' - } - }; - return defaultOption; - }; - - return DataZoomFeature; - }(ToolboxFeature); - - var handlers$1 = { - zoom: function () { - var nextActive = !this._isZoomActive; - this.api.dispatchAction({ - type: 'takeGlobalCursor', - key: 'dataZoomSelect', - dataZoomSelectActive: nextActive - }); - }, - back: function () { - this._dispatchZoomAction(pop(this.ecModel)); - } - }; - - function makeAxisFinder(dzFeatureModel) { - var setting = { - xAxisIndex: dzFeatureModel.get('xAxisIndex', true), - yAxisIndex: dzFeatureModel.get('yAxisIndex', true), - xAxisId: dzFeatureModel.get('xAxisId', true), - yAxisId: dzFeatureModel.get('yAxisId', true) - }; // If both `xAxisIndex` `xAxisId` not set, it means 'all'. - // If both `yAxisIndex` `yAxisId` not set, it means 'all'. - // Some old cases set like this below to close yAxis control but leave xAxis control: - // `{ feature: { dataZoom: { yAxisIndex: false } }`. - - if (setting.xAxisIndex == null && setting.xAxisId == null) { - setting.xAxisIndex = 'all'; - } - - if (setting.yAxisIndex == null && setting.yAxisId == null) { - setting.yAxisIndex = 'all'; - } - - return setting; - } - - function updateBackBtnStatus(featureModel, ecModel) { - featureModel.setIconStatus('back', count(ecModel) > 1 ? 'emphasis' : 'normal'); - } - - function updateZoomBtnStatus(featureModel, ecModel, view, payload, api) { - var zoomActive = view._isZoomActive; - - if (payload && payload.type === 'takeGlobalCursor') { - zoomActive = payload.key === 'dataZoomSelect' ? payload.dataZoomSelectActive : false; - } - - view._isZoomActive = zoomActive; - featureModel.setIconStatus('zoom', zoomActive ? 'emphasis' : 'normal'); - var brushTargetManager = new BrushTargetManager(makeAxisFinder(featureModel), ecModel, { - include: ['grid'] - }); - var panels = brushTargetManager.makePanelOpts(api, function (targetInfo) { - return targetInfo.xAxisDeclared && !targetInfo.yAxisDeclared ? 'lineX' : !targetInfo.xAxisDeclared && targetInfo.yAxisDeclared ? 'lineY' : 'rect'; - }); - - view._brushController.setPanels(panels).enableBrush(zoomActive && panels.length ? { - brushType: 'auto', - brushStyle: featureModel.getModel('brushStyle').getItemStyle() - } : false); - } - - registerInternalOptionCreator('dataZoom', function (ecModel) { - var toolboxModel = ecModel.getComponent('toolbox', 0); - var featureDataZoomPath = ['feature', 'dataZoom']; - - if (!toolboxModel || toolboxModel.get(featureDataZoomPath) == null) { - return; - } - - var dzFeatureModel = toolboxModel.getModel(featureDataZoomPath); - var dzOptions = []; - var finder = makeAxisFinder(dzFeatureModel); - var finderResult = parseFinder(ecModel, finder); - each$a(finderResult.xAxisModels, function (axisModel) { - return buildInternalOptions(axisModel, 'xAxis', 'xAxisIndex'); - }); - each$a(finderResult.yAxisModels, function (axisModel) { - return buildInternalOptions(axisModel, 'yAxis', 'yAxisIndex'); - }); - - function buildInternalOptions(axisModel, axisMainType, axisIndexPropName) { - var axisIndex = axisModel.componentIndex; - var newOpt = { - type: 'select', - $fromToolbox: true, - // Default to be filter - filterMode: dzFeatureModel.get('filterMode', true) || 'filter', - // Id for merge mapping. - id: DATA_ZOOM_ID_BASE + axisMainType + axisIndex - }; - newOpt[axisIndexPropName] = axisIndex; - dzOptions.push(newOpt); - } - - return dzOptions; - }); - - function install$z(registers) { - registers.registerComponentModel(ToolboxModel); - registers.registerComponentView(ToolboxView); - registerFeature('saveAsImage', SaveAsImage); - registerFeature('magicType', MagicType); - registerFeature('dataView', DataView); - registerFeature('dataZoom', DataZoomFeature); - registerFeature('restore', RestoreOption); - use(install$y); - } - - var TooltipModel = - /** @class */ - function (_super) { - __extends(TooltipModel, _super); - - function TooltipModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TooltipModel.type; - return _this; - } - - TooltipModel.type = 'tooltip'; - TooltipModel.dependencies = ['axisPointer']; - TooltipModel.defaultOption = { - // zlevel: 0, - z: 60, - show: true, - // tooltip main content - showContent: true, - // 'trigger' only works on coordinate system. - // 'item' | 'axis' | 'none' - trigger: 'item', - // 'click' | 'mousemove' | 'none' - triggerOn: 'mousemove|click', - alwaysShowContent: false, - displayMode: 'single', - renderMode: 'auto', - // whether restraint content inside viewRect. - // If renderMode: 'richText', default true. - // If renderMode: 'html', defaut false (for backward compat). - confine: null, - showDelay: 0, - hideDelay: 100, - // Animation transition time, unit is second - transitionDuration: 0.4, - enterable: false, - backgroundColor: '#fff', - // box shadow - shadowBlur: 10, - shadowColor: 'rgba(0, 0, 0, .2)', - shadowOffsetX: 1, - shadowOffsetY: 2, - // tooltip border radius, unit is px, default is 4 - borderRadius: 4, - // tooltip border width, unit is px, default is 0 (no border) - borderWidth: 1, - // Tooltip inside padding, default is 5 for all direction - // Array is allowed to set up, right, bottom, left, same with css - // The default value: See `tooltip/tooltipMarkup.ts#getPaddingFromTooltipModel`. - padding: null, - // Extra css text - extraCssText: '', - // axis indicator, trigger by axis - axisPointer: { - // default is line - // legal values: 'line' | 'shadow' | 'cross' - type: 'line', - // Valid when type is line, appoint tooltip line locate on which line. Optional - // legal values: 'x' | 'y' | 'angle' | 'radius' | 'auto' - // default is 'auto', chose the axis which type is category. - // for multiply y axis, cartesian coord chose x axis, polar chose angle axis - axis: 'auto', - animation: 'auto', - animationDurationUpdate: 200, - animationEasingUpdate: 'exponentialOut', - crossStyle: { - color: '#999', - width: 1, - type: 'dashed', - // TODO formatter - textStyle: {} - } // lineStyle and shadowStyle should not be specified here, - // otherwise it will always override those styles on option.axisPointer. - - }, - textStyle: { - color: '#666', - fontSize: 14 - } - }; - return TooltipModel; - }(ComponentModel); - - /* global document */ - - function shouldTooltipConfine(tooltipModel) { - var confineOption = tooltipModel.get('confine'); - return confineOption != null ? !!confineOption // In richText mode, the outside part can not be visible. - : tooltipModel.get('renderMode') === 'richText'; - } - - function testStyle(styleProps) { - if (!env.domSupported) { - return; - } - - var style = document.documentElement.style; - - for (var i = 0, len = styleProps.length; i < len; i++) { - if (styleProps[i] in style) { - return styleProps[i]; - } - } - } - - var TRANSFORM_VENDOR = testStyle(['transform', 'webkitTransform', 'OTransform', 'MozTransform', 'msTransform']); - var TRANSITION_VENDOR = testStyle(['webkitTransition', 'transition', 'OTransition', 'MozTransition', 'msTransition']); - function toCSSVendorPrefix(styleVendor, styleProp) { - if (!styleVendor) { - return styleProp; - } - - styleProp = toCamelCase(styleProp, true); - var idx = styleVendor.indexOf(styleProp); - styleVendor = idx === -1 ? styleProp : "-" + styleVendor.slice(0, idx) + "-" + styleProp; - return styleVendor.toLowerCase(); - } - function getComputedStyle(el, style) { - var stl = el.currentStyle || document.defaultView && document.defaultView.getComputedStyle(el); - return stl ? style ? stl[style] : stl : null; - } - - /* global document, window */ - - var CSS_TRANSITION_VENDOR = toCSSVendorPrefix(TRANSITION_VENDOR, 'transition'); - var CSS_TRANSFORM_VENDOR = toCSSVendorPrefix(TRANSFORM_VENDOR, 'transform'); // eslint-disable-next-line - - var gCssText = "position:absolute;display:block;border-style:solid;white-space:nowrap;z-index:9999999;" + (env.transform3dSupported ? 'will-change:transform;' : ''); - - function mirrorPos(pos) { - pos = pos === 'left' ? 'right' : pos === 'right' ? 'left' : pos === 'top' ? 'bottom' : 'top'; - return pos; - } - - function assembleArrow(tooltipModel, borderColor, arrowPosition) { - if (!isString(arrowPosition) || arrowPosition === 'inside') { - return ''; - } - - var backgroundColor = tooltipModel.get('backgroundColor'); - var borderWidth = tooltipModel.get('borderWidth'); - borderColor = convertToColorString(borderColor); - var arrowPos = mirrorPos(arrowPosition); - var arrowSize = Math.max(Math.round(borderWidth) * 1.5, 6); - var positionStyle = ''; - var transformStyle = CSS_TRANSFORM_VENDOR + ':'; - var rotateDeg; - - if (indexOf(['left', 'right'], arrowPos) > -1) { - positionStyle += 'top:50%'; - transformStyle += "translateY(-50%) rotate(" + (rotateDeg = arrowPos === 'left' ? -225 : -45) + "deg)"; - } else { - positionStyle += 'left:50%'; - transformStyle += "translateX(-50%) rotate(" + (rotateDeg = arrowPos === 'top' ? 225 : 45) + "deg)"; - } - - var rotateRadian = rotateDeg * Math.PI / 180; - var arrowWH = arrowSize + borderWidth; - var rotatedWH = arrowWH * Math.abs(Math.cos(rotateRadian)) + arrowWH * Math.abs(Math.sin(rotateRadian)); - var arrowOffset = Math.round(((rotatedWH - Math.SQRT2 * borderWidth) / 2 + Math.SQRT2 * borderWidth - (rotatedWH - arrowWH) / 2) * 100) / 100; - positionStyle += ";" + arrowPos + ":-" + arrowOffset + "px"; - var borderStyle = borderColor + " solid " + borderWidth + "px;"; - var styleCss = ["position:absolute;width:" + arrowSize + "px;height:" + arrowSize + "px;z-index:-1;", positionStyle + ";" + transformStyle + ";", "border-bottom:" + borderStyle, "border-right:" + borderStyle, "background-color:" + backgroundColor + ";"]; - return "<div style=\"" + styleCss.join('') + "\"></div>"; - } - - function assembleTransition(duration, onlyFade) { - var transitionCurve = 'cubic-bezier(0.23,1,0.32,1)'; - var transitionOption = " " + duration / 2 + "s " + transitionCurve; - var transitionText = "opacity" + transitionOption + ",visibility" + transitionOption; - - if (!onlyFade) { - transitionOption = " " + duration + "s " + transitionCurve; - transitionText += env.transformSupported ? "," + CSS_TRANSFORM_VENDOR + transitionOption : ",left" + transitionOption + ",top" + transitionOption; - } - - return CSS_TRANSITION_VENDOR + ':' + transitionText; - } - - function assembleTransform(x, y, toString) { - // If using float on style, the final width of the dom might - // keep changing slightly while mouse move. So `toFixed(0)` them. - var x0 = x.toFixed(0) + 'px'; - var y0 = y.toFixed(0) + 'px'; // not support transform, use `left` and `top` instead. - - if (!env.transformSupported) { - return toString ? "top:" + y0 + ";left:" + x0 + ";" : [['top', y0], ['left', x0]]; - } // support transform - - - var is3d = env.transform3dSupported; - var translate = "translate" + (is3d ? '3d' : '') + "(" + x0 + "," + y0 + (is3d ? ',0' : '') + ")"; - return toString ? 'top:0;left:0;' + CSS_TRANSFORM_VENDOR + ':' + translate + ';' : [['top', 0], ['left', 0], [TRANSFORM_VENDOR, translate]]; - } - /** - * @param {Object} textStyle - * @return {string} - * @inner - */ - - - function assembleFont(textStyleModel) { - var cssText = []; - var fontSize = textStyleModel.get('fontSize'); - var color = textStyleModel.getTextColor(); - color && cssText.push('color:' + color); - cssText.push('font:' + textStyleModel.getFont()); - fontSize // @ts-ignore, leave it to the tooltip refactor. - && cssText.push('line-height:' + Math.round(fontSize * 3 / 2) + 'px'); - var shadowColor = textStyleModel.get('textShadowColor'); - var shadowBlur = textStyleModel.get('textShadowBlur') || 0; - var shadowOffsetX = textStyleModel.get('textShadowOffsetX') || 0; - var shadowOffsetY = textStyleModel.get('textShadowOffsetY') || 0; - shadowColor && shadowBlur && cssText.push('text-shadow:' + shadowOffsetX + 'px ' + shadowOffsetY + 'px ' + shadowBlur + 'px ' + shadowColor); - each(['decoration', 'align'], function (name) { - var val = textStyleModel.get(name); - val && cssText.push('text-' + name + ':' + val); - }); - return cssText.join(';'); - } - - function assembleCssText(tooltipModel, enableTransition, onlyFade) { - var cssText = []; - var transitionDuration = tooltipModel.get('transitionDuration'); - var backgroundColor = tooltipModel.get('backgroundColor'); - var shadowBlur = tooltipModel.get('shadowBlur'); - var shadowColor = tooltipModel.get('shadowColor'); - var shadowOffsetX = tooltipModel.get('shadowOffsetX'); - var shadowOffsetY = tooltipModel.get('shadowOffsetY'); - var textStyleModel = tooltipModel.getModel('textStyle'); - var padding = getPaddingFromTooltipModel(tooltipModel, 'html'); - var boxShadow = shadowOffsetX + "px " + shadowOffsetY + "px " + shadowBlur + "px " + shadowColor; - cssText.push('box-shadow:' + boxShadow); // Animation transition. Do not animate when transitionDuration is 0. - - enableTransition && transitionDuration && cssText.push(assembleTransition(transitionDuration, onlyFade)); - - if (backgroundColor) { - cssText.push('background-color:' + backgroundColor); - } // Border style - - - each(['width', 'color', 'radius'], function (name) { - var borderName = 'border-' + name; - var camelCase = toCamelCase(borderName); - var val = tooltipModel.get(camelCase); - val != null && cssText.push(borderName + ':' + val + (name === 'color' ? '' : 'px')); - }); // Text style - - cssText.push(assembleFont(textStyleModel)); // Padding - - if (padding != null) { - cssText.push('padding:' + normalizeCssArray$1(padding).join('px ') + 'px'); - } - - return cssText.join(';') + ';'; - } // If not able to make, do not modify the input `out`. - - - function makeStyleCoord(out, zr, appendToBody, zrX, zrY) { - var zrPainter = zr && zr.painter; - - if (appendToBody) { - var zrViewportRoot = zrPainter && zrPainter.getViewportRoot(); - - if (zrViewportRoot) { - // Some APPs might use scale on body, so we support CSS transform here. - transformLocalCoord(out, zrViewportRoot, document.body, zrX, zrY); - } - } else { - out[0] = zrX; - out[1] = zrY; // xy should be based on canvas root. But tooltipContent is - // the sibling of canvas root. So padding of ec container - // should be considered here. - - var viewportRootOffset = zrPainter && zrPainter.getViewportRootOffset(); - - if (viewportRootOffset) { - out[0] += viewportRootOffset.offsetLeft; - out[1] += viewportRootOffset.offsetTop; - } - } - - out[2] = out[0] / zr.getWidth(); - out[3] = out[1] / zr.getHeight(); - } - - var TooltipHTMLContent = - /** @class */ - function () { - function TooltipHTMLContent(container, api, opt) { - this._show = false; - this._styleCoord = [0, 0, 0, 0]; - this._enterable = true; - this._alwaysShowContent = false; - this._firstShow = true; - this._longHide = true; - - if (env.wxa) { - return null; - } - - var el = document.createElement('div'); // TODO: TYPE - - el.domBelongToZr = true; - this.el = el; - var zr = this._zr = api.getZr(); - var appendToBody = this._appendToBody = opt && opt.appendToBody; - makeStyleCoord(this._styleCoord, zr, appendToBody, api.getWidth() / 2, api.getHeight() / 2); - - if (appendToBody) { - document.body.appendChild(el); - } else { - container.appendChild(el); - } - - this._container = container; // FIXME - // Is it needed to trigger zr event manually if - // the browser do not support `pointer-events: none`. - - var self = this; - - el.onmouseenter = function () { - // clear the timeout in hideLater and keep showing tooltip - if (self._enterable) { - clearTimeout(self._hideTimeout); - self._show = true; - } - - self._inContent = true; - }; - - el.onmousemove = function (e) { - e = e || window.event; - - if (!self._enterable) { - // `pointer-events: none` is set to tooltip content div - // if `enterable` is set as `false`, and `el.onmousemove` - // can not be triggered. But in browser that do not - // support `pointer-events`, we need to do this: - // Try trigger zrender event to avoid mouse - // in and out shape too frequently - var handler = zr.handler; - var zrViewportRoot = zr.painter.getViewportRoot(); - normalizeEvent(zrViewportRoot, e, true); - handler.dispatch('mousemove', e); - } - }; - - el.onmouseleave = function () { - // set `_inContent` to `false` before `hideLater` - self._inContent = false; - - if (self._enterable) { - if (self._show) { - self.hideLater(self._hideDelay); - } - } - }; - } - /** - * Update when tooltip is rendered - */ - - - TooltipHTMLContent.prototype.update = function (tooltipModel) { - // FIXME - // Move this logic to ec main? - var container = this._container; - var position = getComputedStyle(container, 'position'); - var domStyle = container.style; - - if (domStyle.position !== 'absolute' && position !== 'absolute') { - domStyle.position = 'relative'; - } // move tooltip if chart resized - - - var alwaysShowContent = tooltipModel.get('alwaysShowContent'); - alwaysShowContent && this._moveIfResized(); // update alwaysShowContent - - this._alwaysShowContent = alwaysShowContent; // update className - - this.el.className = tooltipModel.get('className') || ''; // Hide the tooltip - // PENDING - // this.hide(); - }; - - TooltipHTMLContent.prototype.show = function (tooltipModel, nearPointColor) { - clearTimeout(this._hideTimeout); - clearTimeout(this._longHideTimeout); - var el = this.el; - var style = el.style; - var styleCoord = this._styleCoord; - - if (!el.innerHTML) { - style.display = 'none'; - } else { - style.cssText = gCssText + assembleCssText(tooltipModel, !this._firstShow, this._longHide) // initial transform - + assembleTransform(styleCoord[0], styleCoord[1], true) + ("border-color:" + convertToColorString(nearPointColor) + ";") + (tooltipModel.get('extraCssText') || '') // If mouse occasionally move over the tooltip, a mouseout event will be - // triggered by canvas, and cause some unexpectable result like dragging - // stop, "unfocusAdjacency". Here `pointer-events: none` is used to solve - // it. Although it is not supported by IE8~IE10, fortunately it is a rare - // scenario. - + (";pointer-events:" + (this._enterable ? 'auto' : 'none')); - } - - this._show = true; - this._firstShow = false; - this._longHide = false; - }; - - TooltipHTMLContent.prototype.setContent = function (content, markers, tooltipModel, borderColor, arrowPosition) { - var el = this.el; - - if (content == null) { - el.innerHTML = ''; - return; - } - - var arrow = ''; - - if (isString(arrowPosition) && tooltipModel.get('trigger') === 'item' && !shouldTooltipConfine(tooltipModel)) { - arrow = assembleArrow(tooltipModel, borderColor, arrowPosition); - } - - if (isString(content)) { - el.innerHTML = content + arrow; - } else if (content) { - // Clear previous - el.innerHTML = ''; - - if (!isArray(content)) { - content = [content]; - } - - for (var i = 0; i < content.length; i++) { - if (isDom(content[i]) && content[i].parentNode !== el) { - el.appendChild(content[i]); - } - } // no arrow if empty - - - if (arrow && el.childNodes.length) { - // no need to create a new parent element, but it's not supported by IE 10 and older. - // const arrowEl = document.createRange().createContextualFragment(arrow); - var arrowEl = document.createElement('div'); - arrowEl.innerHTML = arrow; - el.appendChild(arrowEl); - } - } - }; - - TooltipHTMLContent.prototype.setEnterable = function (enterable) { - this._enterable = enterable; - }; - - TooltipHTMLContent.prototype.getSize = function () { - var el = this.el; - return [el.offsetWidth, el.offsetHeight]; - }; - - TooltipHTMLContent.prototype.moveTo = function (zrX, zrY) { - var styleCoord = this._styleCoord; - makeStyleCoord(styleCoord, this._zr, this._appendToBody, zrX, zrY); - - if (styleCoord[0] != null && styleCoord[1] != null) { - var style_1 = this.el.style; - var transforms = assembleTransform(styleCoord[0], styleCoord[1]); - each(transforms, function (transform) { - style_1[transform[0]] = transform[1]; - }); - } - }; - /** - * when `alwaysShowContent` is true, - * move the tooltip after chart resized - */ - - - TooltipHTMLContent.prototype._moveIfResized = function () { - // The ratio of left to width - var ratioX = this._styleCoord[2]; // The ratio of top to height - - var ratioY = this._styleCoord[3]; - this.moveTo(ratioX * this._zr.getWidth(), ratioY * this._zr.getHeight()); - }; - - TooltipHTMLContent.prototype.hide = function () { - var _this = this; - - var style = this.el.style; - style.visibility = 'hidden'; - style.opacity = '0'; - env.transform3dSupported && (style.willChange = ''); - this._show = false; - this._longHideTimeout = setTimeout(function () { - return _this._longHide = true; - }, 500); - }; - - TooltipHTMLContent.prototype.hideLater = function (time) { - if (this._show && !(this._inContent && this._enterable) && !this._alwaysShowContent) { - if (time) { - this._hideDelay = time; // Set show false to avoid invoke hideLater multiple times - - this._show = false; - this._hideTimeout = setTimeout(bind(this.hide, this), time); - } else { - this.hide(); - } - } - }; - - TooltipHTMLContent.prototype.isShow = function () { - return this._show; - }; - - TooltipHTMLContent.prototype.dispose = function () { - this.el.parentNode.removeChild(this.el); - }; - - return TooltipHTMLContent; - }(); - - var TooltipRichContent = - /** @class */ - function () { - function TooltipRichContent(api) { - this._show = false; - this._styleCoord = [0, 0, 0, 0]; - this._alwaysShowContent = false; - this._enterable = true; - this._zr = api.getZr(); - makeStyleCoord$1(this._styleCoord, this._zr, api.getWidth() / 2, api.getHeight() / 2); - } - /** - * Update when tooltip is rendered - */ - - - TooltipRichContent.prototype.update = function (tooltipModel) { - var alwaysShowContent = tooltipModel.get('alwaysShowContent'); - alwaysShowContent && this._moveIfResized(); // update alwaysShowContent - - this._alwaysShowContent = alwaysShowContent; - }; - - TooltipRichContent.prototype.show = function () { - if (this._hideTimeout) { - clearTimeout(this._hideTimeout); - } - - this.el.show(); - this._show = true; - }; - /** - * Set tooltip content - */ - - - TooltipRichContent.prototype.setContent = function (content, markupStyleCreator, tooltipModel, borderColor, arrowPosition) { - var _this = this; - - if (isObject(content)) { - throwError("development" !== 'production' ? 'Passing DOM nodes as content is not supported in richText tooltip!' : ''); - } - - if (this.el) { - this._zr.remove(this.el); - } - - var textStyleModel = tooltipModel.getModel('textStyle'); - this.el = new ZRText({ - style: { - rich: markupStyleCreator.richTextStyles, - text: content, - lineHeight: 22, - borderWidth: 1, - borderColor: borderColor, - textShadowColor: textStyleModel.get('textShadowColor'), - fill: tooltipModel.get(['textStyle', 'color']), - padding: getPaddingFromTooltipModel(tooltipModel, 'richText'), - verticalAlign: 'top', - align: 'left' - }, - z: tooltipModel.get('z') - }); - each(['backgroundColor', 'borderRadius', 'shadowColor', 'shadowBlur', 'shadowOffsetX', 'shadowOffsetY'], function (propName) { - _this.el.style[propName] = tooltipModel.get(propName); - }); - each(['textShadowBlur', 'textShadowOffsetX', 'textShadowOffsetY'], function (propName) { - _this.el.style[propName] = textStyleModel.get(propName) || 0; - }); - - this._zr.add(this.el); - - var self = this; - this.el.on('mouseover', function () { - // clear the timeout in hideLater and keep showing tooltip - if (self._enterable) { - clearTimeout(self._hideTimeout); - self._show = true; - } - - self._inContent = true; - }); - this.el.on('mouseout', function () { - if (self._enterable) { - if (self._show) { - self.hideLater(self._hideDelay); - } - } - - self._inContent = false; - }); - }; - - TooltipRichContent.prototype.setEnterable = function (enterable) { - this._enterable = enterable; - }; - - TooltipRichContent.prototype.getSize = function () { - var el = this.el; - var bounding = this.el.getBoundingRect(); // bounding rect does not include shadow. For renderMode richText, - // if overflow, it will be cut. So calculate them accurately. - - var shadowOuterSize = calcShadowOuterSize(el.style); - return [bounding.width + shadowOuterSize.left + shadowOuterSize.right, bounding.height + shadowOuterSize.top + shadowOuterSize.bottom]; - }; - - TooltipRichContent.prototype.moveTo = function (x, y) { - var el = this.el; - - if (el) { - var styleCoord = this._styleCoord; - makeStyleCoord$1(styleCoord, this._zr, x, y); - x = styleCoord[0]; - y = styleCoord[1]; - var style = el.style; - var borderWidth = mathMaxWith0(style.borderWidth || 0); - var shadowOuterSize = calcShadowOuterSize(style); // rich text x, y do not include border. - - el.x = x + borderWidth + shadowOuterSize.left; - el.y = y + borderWidth + shadowOuterSize.top; - el.markRedraw(); - } - }; - /** - * when `alwaysShowContent` is true, - * move the tooltip after chart resized - */ - - - TooltipRichContent.prototype._moveIfResized = function () { - // The ratio of left to width - var ratioX = this._styleCoord[2]; // The ratio of top to height - - var ratioY = this._styleCoord[3]; - this.moveTo(ratioX * this._zr.getWidth(), ratioY * this._zr.getHeight()); - }; - - TooltipRichContent.prototype.hide = function () { - if (this.el) { - this.el.hide(); - } - - this._show = false; - }; - - TooltipRichContent.prototype.hideLater = function (time) { - if (this._show && !(this._inContent && this._enterable) && !this._alwaysShowContent) { - if (time) { - this._hideDelay = time; // Set show false to avoid invoke hideLater multiple times - - this._show = false; - this._hideTimeout = setTimeout(bind(this.hide, this), time); - } else { - this.hide(); - } - } - }; - - TooltipRichContent.prototype.isShow = function () { - return this._show; - }; - - TooltipRichContent.prototype.dispose = function () { - this._zr.remove(this.el); - }; - - return TooltipRichContent; - }(); - - function mathMaxWith0(val) { - return Math.max(0, val); - } - - function calcShadowOuterSize(style) { - var shadowBlur = mathMaxWith0(style.shadowBlur || 0); - var shadowOffsetX = mathMaxWith0(style.shadowOffsetX || 0); - var shadowOffsetY = mathMaxWith0(style.shadowOffsetY || 0); - return { - left: mathMaxWith0(shadowBlur - shadowOffsetX), - right: mathMaxWith0(shadowBlur + shadowOffsetX), - top: mathMaxWith0(shadowBlur - shadowOffsetY), - bottom: mathMaxWith0(shadowBlur + shadowOffsetY) - }; - } - - function makeStyleCoord$1(out, zr, zrX, zrY) { - out[0] = zrX; - out[1] = zrY; - out[2] = out[0] / zr.getWidth(); - out[3] = out[1] / zr.getHeight(); - } - - var proxyRect = new Rect({ - shape: { - x: -1, - y: -1, - width: 2, - height: 2 - } - }); - - var TooltipView = - /** @class */ - function (_super) { - __extends(TooltipView, _super); - - function TooltipView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TooltipView.type; - return _this; - } - - TooltipView.prototype.init = function (ecModel, api) { - if (env.node || !api.getDom()) { - return; - } - - var tooltipModel = ecModel.getComponent('tooltip'); - var renderMode = this._renderMode = getTooltipRenderMode(tooltipModel.get('renderMode')); - this._tooltipContent = renderMode === 'richText' ? new TooltipRichContent(api) : new TooltipHTMLContent(api.getDom(), api, { - appendToBody: tooltipModel.get('appendToBody', true) - }); - }; - - TooltipView.prototype.render = function (tooltipModel, ecModel, api) { - if (env.node || !api.getDom()) { - return; - } // Reset - - - this.group.removeAll(); - this._tooltipModel = tooltipModel; - this._ecModel = ecModel; - this._api = api; - var tooltipContent = this._tooltipContent; - tooltipContent.update(tooltipModel); - tooltipContent.setEnterable(tooltipModel.get('enterable')); - - this._initGlobalListener(); - - this._keepShow(); // PENDING - // `mousemove` event will be triggered very frequently when the mouse moves fast, - // which causes that the `updatePosition` function was also called frequently. - // In Chrome with devtools open and Firefox, tooltip looks laggy and shakes. See #14695 #16101 - // To avoid frequent triggering, - // consider throttling it in 50ms when transition is enabled - - - if (this._renderMode !== 'richText' && tooltipModel.get('transitionDuration')) { - createOrUpdate(this, '_updatePosition', 50, 'fixRate'); - } else { - clear(this, '_updatePosition'); - } - }; - - TooltipView.prototype._initGlobalListener = function () { - var tooltipModel = this._tooltipModel; - var triggerOn = tooltipModel.get('triggerOn'); - register('itemTooltip', this._api, bind(function (currTrigger, e, dispatchAction) { - // If 'none', it is not controlled by mouse totally. - if (triggerOn !== 'none') { - if (triggerOn.indexOf(currTrigger) >= 0) { - this._tryShow(e, dispatchAction); - } else if (currTrigger === 'leave') { - this._hide(dispatchAction); - } - } - }, this)); - }; - - TooltipView.prototype._keepShow = function () { - var tooltipModel = this._tooltipModel; - var ecModel = this._ecModel; - var api = this._api; - var triggerOn = tooltipModel.get('triggerOn'); // Try to keep the tooltip show when refreshing - - if (this._lastX != null && this._lastY != null // When user is willing to control tooltip totally using API, - // self.manuallyShowTip({x, y}) might cause tooltip hide, - // which is not expected. - && triggerOn !== 'none' && triggerOn !== 'click') { - var self_1 = this; - clearTimeout(this._refreshUpdateTimeout); - this._refreshUpdateTimeout = setTimeout(function () { - // Show tip next tick after other charts are rendered - // In case highlight action has wrong result - // FIXME - !api.isDisposed() && self_1.manuallyShowTip(tooltipModel, ecModel, api, { - x: self_1._lastX, - y: self_1._lastY, - dataByCoordSys: self_1._lastDataByCoordSys - }); - }); - } - }; - /** - * Show tip manually by - * dispatchAction({ - * type: 'showTip', - * x: 10, - * y: 10 - * }); - * Or - * dispatchAction({ - * type: 'showTip', - * seriesIndex: 0, - * dataIndex or dataIndexInside or name - * }); - * - * TODO Batch - */ - - - TooltipView.prototype.manuallyShowTip = function (tooltipModel, ecModel, api, payload) { - if (payload.from === this.uid || env.node || !api.getDom()) { - return; - } - - var dispatchAction = makeDispatchAction$1(payload, api); // Reset ticket - - this._ticket = ''; // When triggered from axisPointer. - - var dataByCoordSys = payload.dataByCoordSys; - var cmptRef = findComponentReference(payload, ecModel, api); - - if (cmptRef) { - var rect = cmptRef.el.getBoundingRect().clone(); - rect.applyTransform(cmptRef.el.transform); - - this._tryShow({ - offsetX: rect.x + rect.width / 2, - offsetY: rect.y + rect.height / 2, - target: cmptRef.el, - position: payload.position, - // When manully trigger, the mouse is not on the el, so we'd better to - // position tooltip on the bottom of the el and display arrow is possible. - positionDefault: 'bottom' - }, dispatchAction); - } else if (payload.tooltip && payload.x != null && payload.y != null) { - var el = proxyRect; - el.x = payload.x; - el.y = payload.y; - el.update(); - getECData(el).tooltipConfig = { - name: null, - option: payload.tooltip - }; // Manually show tooltip while view is not using zrender elements. - - this._tryShow({ - offsetX: payload.x, - offsetY: payload.y, - target: el - }, dispatchAction); - } else if (dataByCoordSys) { - this._tryShow({ - offsetX: payload.x, - offsetY: payload.y, - position: payload.position, - dataByCoordSys: dataByCoordSys, - tooltipOption: payload.tooltipOption - }, dispatchAction); - } else if (payload.seriesIndex != null) { - if (this._manuallyAxisShowTip(tooltipModel, ecModel, api, payload)) { - return; - } - - var pointInfo = findPointFromSeries(payload, ecModel); - var cx = pointInfo.point[0]; - var cy = pointInfo.point[1]; - - if (cx != null && cy != null) { - this._tryShow({ - offsetX: cx, - offsetY: cy, - target: pointInfo.el, - position: payload.position, - // When manully trigger, the mouse is not on the el, so we'd better to - // position tooltip on the bottom of the el and display arrow is possible. - positionDefault: 'bottom' - }, dispatchAction); - } - } else if (payload.x != null && payload.y != null) { - // FIXME - // should wrap dispatchAction like `axisPointer/globalListener` ? - api.dispatchAction({ - type: 'updateAxisPointer', - x: payload.x, - y: payload.y - }); - - this._tryShow({ - offsetX: payload.x, - offsetY: payload.y, - position: payload.position, - target: api.getZr().findHover(payload.x, payload.y).target - }, dispatchAction); - } - }; - - TooltipView.prototype.manuallyHideTip = function (tooltipModel, ecModel, api, payload) { - var tooltipContent = this._tooltipContent; - - if (this._tooltipModel) { - tooltipContent.hideLater(this._tooltipModel.get('hideDelay')); - } - - this._lastX = this._lastY = this._lastDataByCoordSys = null; - - if (payload.from !== this.uid) { - this._hide(makeDispatchAction$1(payload, api)); - } - }; // Be compatible with previous design, that is, when tooltip.type is 'axis' and - // dispatchAction 'showTip' with seriesIndex and dataIndex will trigger axis pointer - // and tooltip. - - - TooltipView.prototype._manuallyAxisShowTip = function (tooltipModel, ecModel, api, payload) { - var seriesIndex = payload.seriesIndex; - var dataIndex = payload.dataIndex; // @ts-ignore - - var coordSysAxesInfo = ecModel.getComponent('axisPointer').coordSysAxesInfo; - - if (seriesIndex == null || dataIndex == null || coordSysAxesInfo == null) { - return; - } - - var seriesModel = ecModel.getSeriesByIndex(seriesIndex); - - if (!seriesModel) { - return; - } - - var data = seriesModel.getData(); - var tooltipCascadedModel = buildTooltipModel([data.getItemModel(dataIndex), seriesModel, (seriesModel.coordinateSystem || {}).model], this._tooltipModel); - - if (tooltipCascadedModel.get('trigger') !== 'axis') { - return; - } - - api.dispatchAction({ - type: 'updateAxisPointer', - seriesIndex: seriesIndex, - dataIndex: dataIndex, - position: payload.position - }); - return true; - }; - - TooltipView.prototype._tryShow = function (e, dispatchAction) { - var el = e.target; - var tooltipModel = this._tooltipModel; - - if (!tooltipModel) { - return; - } // Save mouse x, mouse y. So we can try to keep showing the tip if chart is refreshed - - - this._lastX = e.offsetX; - this._lastY = e.offsetY; - var dataByCoordSys = e.dataByCoordSys; - - if (dataByCoordSys && dataByCoordSys.length) { - this._showAxisTooltip(dataByCoordSys, e); - } else if (el) { - this._lastDataByCoordSys = null; - var seriesDispatcher_1; - var cmptDispatcher_1; - findEventDispatcher(el, function (target) { - // Always show item tooltip if mouse is on the element with dataIndex - if (getECData(target).dataIndex != null) { - seriesDispatcher_1 = target; - return true; - } // Tooltip provided directly. Like legend. - - - if (getECData(target).tooltipConfig != null) { - cmptDispatcher_1 = target; - return true; - } - }, true); - - if (seriesDispatcher_1) { - this._showSeriesItemTooltip(e, seriesDispatcher_1, dispatchAction); - } else if (cmptDispatcher_1) { - this._showComponentItemTooltip(e, cmptDispatcher_1, dispatchAction); - } else { - this._hide(dispatchAction); - } - } else { - this._lastDataByCoordSys = null; - - this._hide(dispatchAction); - } - }; - - TooltipView.prototype._showOrMove = function (tooltipModel, cb) { - // showDelay is used in this case: tooltip.enterable is set - // as true. User intent to move mouse into tooltip and click - // something. `showDelay` makes it easier to enter the content - // but tooltip do not move immediately. - var delay = tooltipModel.get('showDelay'); - cb = bind(cb, this); - clearTimeout(this._showTimout); - delay > 0 ? this._showTimout = setTimeout(cb, delay) : cb(); - }; - - TooltipView.prototype._showAxisTooltip = function (dataByCoordSys, e) { - var ecModel = this._ecModel; - var globalTooltipModel = this._tooltipModel; - var point = [e.offsetX, e.offsetY]; - var singleTooltipModel = buildTooltipModel([e.tooltipOption], globalTooltipModel); - var renderMode = this._renderMode; - var cbParamsList = []; - var articleMarkup = createTooltipMarkup('section', { - blocks: [], - noHeader: true - }); // Only for legacy: `Serise['formatTooltip']` returns a string. - - var markupTextArrLegacy = []; - var markupStyleCreator = new TooltipMarkupStyleCreator(); - each(dataByCoordSys, function (itemCoordSys) { - each(itemCoordSys.dataByAxis, function (axisItem) { - var axisModel = ecModel.getComponent(axisItem.axisDim + 'Axis', axisItem.axisIndex); - var axisValue = axisItem.value; - - if (!axisModel || axisValue == null) { - return; - } - - var axisValueLabel = getValueLabel(axisValue, axisModel.axis, ecModel, axisItem.seriesDataIndices, axisItem.valueLabelOpt); - var axisSectionMarkup = createTooltipMarkup('section', { - header: axisValueLabel, - noHeader: !trim(axisValueLabel), - sortBlocks: true, - blocks: [] - }); - articleMarkup.blocks.push(axisSectionMarkup); - each(axisItem.seriesDataIndices, function (idxItem) { - var series = ecModel.getSeriesByIndex(idxItem.seriesIndex); - var dataIndex = idxItem.dataIndexInside; - var cbParams = series.getDataParams(dataIndex); // Can't find data. - - if (cbParams.dataIndex < 0) { - return; - } - - cbParams.axisDim = axisItem.axisDim; - cbParams.axisIndex = axisItem.axisIndex; - cbParams.axisType = axisItem.axisType; - cbParams.axisId = axisItem.axisId; - cbParams.axisValue = getAxisRawValue(axisModel.axis, { - value: axisValue - }); - cbParams.axisValueLabel = axisValueLabel; // Pre-create marker style for makers. Users can assemble richText - // text in `formatter` callback and use those markers style. - - cbParams.marker = markupStyleCreator.makeTooltipMarker('item', convertToColorString(cbParams.color), renderMode); - var seriesTooltipResult = normalizeTooltipFormatResult(series.formatTooltip(dataIndex, true, null)); - var frag = seriesTooltipResult.frag; - - if (frag) { - var valueFormatter = buildTooltipModel([series], globalTooltipModel).get('valueFormatter'); - axisSectionMarkup.blocks.push(valueFormatter ? extend({ - valueFormatter: valueFormatter - }, frag) : frag); - } - - if (seriesTooltipResult.text) { - markupTextArrLegacy.push(seriesTooltipResult.text); - } - - cbParamsList.push(cbParams); - }); - }); - }); // In most cases, the second axis is displays upper on the first one. - // So we reverse it to look better. - - articleMarkup.blocks.reverse(); - markupTextArrLegacy.reverse(); - var positionExpr = e.position; - var orderMode = singleTooltipModel.get('order'); - var builtMarkupText = buildTooltipMarkup(articleMarkup, markupStyleCreator, renderMode, orderMode, ecModel.get('useUTC'), singleTooltipModel.get('textStyle')); - builtMarkupText && markupTextArrLegacy.unshift(builtMarkupText); - var blockBreak = renderMode === 'richText' ? '\n\n' : '<br/>'; - var allMarkupText = markupTextArrLegacy.join(blockBreak); - - this._showOrMove(singleTooltipModel, function () { - if (this._updateContentNotChangedOnAxis(dataByCoordSys, cbParamsList)) { - this._updatePosition(singleTooltipModel, positionExpr, point[0], point[1], this._tooltipContent, cbParamsList); - } else { - this._showTooltipContent(singleTooltipModel, allMarkupText, cbParamsList, Math.random() + '', point[0], point[1], positionExpr, null, markupStyleCreator); - } - }); // Do not trigger events here, because this branch only be entered - // from dispatchAction. - - }; - - TooltipView.prototype._showSeriesItemTooltip = function (e, dispatcher, dispatchAction) { - var ecModel = this._ecModel; - var ecData = getECData(dispatcher); // Use dataModel in element if possible - // Used when mouseover on a element like markPoint or edge - // In which case, the data is not main data in series. - - var seriesIndex = ecData.seriesIndex; - var seriesModel = ecModel.getSeriesByIndex(seriesIndex); // For example, graph link. - - var dataModel = ecData.dataModel || seriesModel; - var dataIndex = ecData.dataIndex; - var dataType = ecData.dataType; - var data = dataModel.getData(dataType); - var renderMode = this._renderMode; - var positionDefault = e.positionDefault; - var tooltipModel = buildTooltipModel([data.getItemModel(dataIndex), dataModel, seriesModel && (seriesModel.coordinateSystem || {}).model], this._tooltipModel, positionDefault ? { - position: positionDefault - } : null); - var tooltipTrigger = tooltipModel.get('trigger'); - - if (tooltipTrigger != null && tooltipTrigger !== 'item') { - return; - } - - var params = dataModel.getDataParams(dataIndex, dataType); - var markupStyleCreator = new TooltipMarkupStyleCreator(); // Pre-create marker style for makers. Users can assemble richText - // text in `formatter` callback and use those markers style. - - params.marker = markupStyleCreator.makeTooltipMarker('item', convertToColorString(params.color), renderMode); - var seriesTooltipResult = normalizeTooltipFormatResult(dataModel.formatTooltip(dataIndex, false, dataType)); - var orderMode = tooltipModel.get('order'); - var valueFormatter = tooltipModel.get('valueFormatter'); - var frag = seriesTooltipResult.frag; - var markupText = frag ? buildTooltipMarkup(valueFormatter ? extend({ - valueFormatter: valueFormatter - }, frag) : frag, markupStyleCreator, renderMode, orderMode, ecModel.get('useUTC'), tooltipModel.get('textStyle')) : seriesTooltipResult.text; - var asyncTicket = 'item_' + dataModel.name + '_' + dataIndex; - - this._showOrMove(tooltipModel, function () { - this._showTooltipContent(tooltipModel, markupText, params, asyncTicket, e.offsetX, e.offsetY, e.position, e.target, markupStyleCreator); - }); // FIXME - // duplicated showtip if manuallyShowTip is called from dispatchAction. - - - dispatchAction({ - type: 'showTip', - dataIndexInside: dataIndex, - dataIndex: data.getRawIndex(dataIndex), - seriesIndex: seriesIndex, - from: this.uid - }); - }; - - TooltipView.prototype._showComponentItemTooltip = function (e, el, dispatchAction) { - var ecData = getECData(el); - var tooltipConfig = ecData.tooltipConfig; - var tooltipOpt = tooltipConfig.option || {}; - - if (isString(tooltipOpt)) { - var content = tooltipOpt; - tooltipOpt = { - content: content, - // Fixed formatter - formatter: content - }; - } - - var tooltipModelCascade = [tooltipOpt]; - - var cmpt = this._ecModel.getComponent(ecData.componentMainType, ecData.componentIndex); - - if (cmpt) { - tooltipModelCascade.push(cmpt); - } // In most cases, component tooltip formatter has different params with series tooltip formatter, - // so that they cannot share the same formatter. Since the global tooltip formatter is used for series - // by convention, we do not use it as the default formatter for component. - - - tooltipModelCascade.push({ - formatter: tooltipOpt.content - }); - var positionDefault = e.positionDefault; - var subTooltipModel = buildTooltipModel(tooltipModelCascade, this._tooltipModel, positionDefault ? { - position: positionDefault - } : null); - var defaultHtml = subTooltipModel.get('content'); - var asyncTicket = Math.random() + ''; // PENDING: this case do not support richText style yet. - - var markupStyleCreator = new TooltipMarkupStyleCreator(); // Do not check whether `trigger` is 'none' here, because `trigger` - // only works on coordinate system. In fact, we have not found case - // that requires setting `trigger` nothing on component yet. - - this._showOrMove(subTooltipModel, function () { - // Use formatterParams from element defined in component - // Avoid users modify it. - var formatterParams = clone(subTooltipModel.get('formatterParams') || {}); - - this._showTooltipContent(subTooltipModel, defaultHtml, formatterParams, asyncTicket, e.offsetX, e.offsetY, e.position, el, markupStyleCreator); - }); // If not dispatch showTip, tip may be hide triggered by axis. - - - dispatchAction({ - type: 'showTip', - from: this.uid - }); - }; - - TooltipView.prototype._showTooltipContent = function ( // Use Model<TooltipOption> insteadof TooltipModel because this model may be from series or other options. - // Instead of top level tooltip. - tooltipModel, defaultHtml, params, asyncTicket, x, y, positionExpr, el, markupStyleCreator) { - // Reset ticket - this._ticket = ''; - - if (!tooltipModel.get('showContent') || !tooltipModel.get('show')) { - return; - } - - var tooltipContent = this._tooltipContent; - tooltipContent.setEnterable(tooltipModel.get('enterable')); - var formatter = tooltipModel.get('formatter'); - positionExpr = positionExpr || tooltipModel.get('position'); - var html = defaultHtml; - - var nearPoint = this._getNearestPoint([x, y], params, tooltipModel.get('trigger'), tooltipModel.get('borderColor')); - - var nearPointColor = nearPoint.color; - - if (formatter) { - if (isString(formatter)) { - var useUTC = tooltipModel.ecModel.get('useUTC'); - var params0 = isArray(params) ? params[0] : params; - var isTimeAxis = params0 && params0.axisType && params0.axisType.indexOf('time') >= 0; - html = formatter; - - if (isTimeAxis) { - html = format(params0.axisValue, html, useUTC); - } - - html = formatTpl(html, params, true); - } else if (isFunction(formatter)) { - var callback = bind(function (cbTicket, html) { - if (cbTicket === this._ticket) { - tooltipContent.setContent(html, markupStyleCreator, tooltipModel, nearPointColor, positionExpr); - - this._updatePosition(tooltipModel, positionExpr, x, y, tooltipContent, params, el); - } - }, this); - this._ticket = asyncTicket; - html = formatter(params, asyncTicket, callback); - } else { - html = formatter; - } - } - - tooltipContent.setContent(html, markupStyleCreator, tooltipModel, nearPointColor, positionExpr); - tooltipContent.show(tooltipModel, nearPointColor); - - this._updatePosition(tooltipModel, positionExpr, x, y, tooltipContent, params, el); - }; - - TooltipView.prototype._getNearestPoint = function (point, tooltipDataParams, trigger, borderColor) { - if (trigger === 'axis' || isArray(tooltipDataParams)) { - return { - color: borderColor || (this._renderMode === 'html' ? '#fff' : 'none') - }; - } - - if (!isArray(tooltipDataParams)) { - return { - color: borderColor || tooltipDataParams.color || tooltipDataParams.borderColor - }; - } - }; - - TooltipView.prototype._updatePosition = function (tooltipModel, positionExpr, x, // Mouse x - y, // Mouse y - content, params, el) { - var viewWidth = this._api.getWidth(); - - var viewHeight = this._api.getHeight(); - - positionExpr = positionExpr || tooltipModel.get('position'); - var contentSize = content.getSize(); - var align = tooltipModel.get('align'); - var vAlign = tooltipModel.get('verticalAlign'); - var rect = el && el.getBoundingRect().clone(); - el && rect.applyTransform(el.transform); - - if (isFunction(positionExpr)) { - // Callback of position can be an array or a string specify the position - positionExpr = positionExpr([x, y], params, content.el, rect, { - viewSize: [viewWidth, viewHeight], - contentSize: contentSize.slice() - }); - } - - if (isArray(positionExpr)) { - x = parsePercent$1(positionExpr[0], viewWidth); - y = parsePercent$1(positionExpr[1], viewHeight); - } else if (isObject(positionExpr)) { - var boxLayoutPosition = positionExpr; - boxLayoutPosition.width = contentSize[0]; - boxLayoutPosition.height = contentSize[1]; - var layoutRect = getLayoutRect(boxLayoutPosition, { - width: viewWidth, - height: viewHeight - }); - x = layoutRect.x; - y = layoutRect.y; - align = null; // When positionExpr is left/top/right/bottom, - // align and verticalAlign will not work. - - vAlign = null; - } // Specify tooltip position by string 'top' 'bottom' 'left' 'right' around graphic element - else if (isString(positionExpr) && el) { - var pos = calcTooltipPosition(positionExpr, rect, contentSize, tooltipModel.get('borderWidth')); - x = pos[0]; - y = pos[1]; - } else { - var pos = refixTooltipPosition(x, y, content, viewWidth, viewHeight, align ? null : 20, vAlign ? null : 20); - x = pos[0]; - y = pos[1]; - } - - align && (x -= isCenterAlign(align) ? contentSize[0] / 2 : align === 'right' ? contentSize[0] : 0); - vAlign && (y -= isCenterAlign(vAlign) ? contentSize[1] / 2 : vAlign === 'bottom' ? contentSize[1] : 0); - - if (shouldTooltipConfine(tooltipModel)) { - var pos = confineTooltipPosition(x, y, content, viewWidth, viewHeight); - x = pos[0]; - y = pos[1]; - } - - content.moveTo(x, y); - }; // FIXME - // Should we remove this but leave this to user? - - - TooltipView.prototype._updateContentNotChangedOnAxis = function (dataByCoordSys, cbParamsList) { - var lastCoordSys = this._lastDataByCoordSys; - var lastCbParamsList = this._cbParamsList; - var contentNotChanged = !!lastCoordSys && lastCoordSys.length === dataByCoordSys.length; - contentNotChanged && each(lastCoordSys, function (lastItemCoordSys, indexCoordSys) { - var lastDataByAxis = lastItemCoordSys.dataByAxis || []; - var thisItemCoordSys = dataByCoordSys[indexCoordSys] || {}; - var thisDataByAxis = thisItemCoordSys.dataByAxis || []; - contentNotChanged = contentNotChanged && lastDataByAxis.length === thisDataByAxis.length; - contentNotChanged && each(lastDataByAxis, function (lastItem, indexAxis) { - var thisItem = thisDataByAxis[indexAxis] || {}; - var lastIndices = lastItem.seriesDataIndices || []; - var newIndices = thisItem.seriesDataIndices || []; - contentNotChanged = contentNotChanged && lastItem.value === thisItem.value && lastItem.axisType === thisItem.axisType && lastItem.axisId === thisItem.axisId && lastIndices.length === newIndices.length; - contentNotChanged && each(lastIndices, function (lastIdxItem, j) { - var newIdxItem = newIndices[j]; - contentNotChanged = contentNotChanged && lastIdxItem.seriesIndex === newIdxItem.seriesIndex && lastIdxItem.dataIndex === newIdxItem.dataIndex; - }); // check is cbParams data value changed - - lastCbParamsList && each(lastItem.seriesDataIndices, function (idxItem) { - var seriesIdx = idxItem.seriesIndex; - var cbParams = cbParamsList[seriesIdx]; - var lastCbParams = lastCbParamsList[seriesIdx]; - - if (cbParams && lastCbParams && lastCbParams.data !== cbParams.data) { - contentNotChanged = false; - } - }); - }); - }); - this._lastDataByCoordSys = dataByCoordSys; - this._cbParamsList = cbParamsList; - return !!contentNotChanged; - }; - - TooltipView.prototype._hide = function (dispatchAction) { - // Do not directly hideLater here, because this behavior may be prevented - // in dispatchAction when showTip is dispatched. - // FIXME - // duplicated hideTip if manuallyHideTip is called from dispatchAction. - this._lastDataByCoordSys = null; - dispatchAction({ - type: 'hideTip', - from: this.uid - }); - }; - - TooltipView.prototype.dispose = function (ecModel, api) { - if (env.node || !api.getDom()) { - return; - } - - clear(this, '_updatePosition'); - - this._tooltipContent.dispose(); - - unregister('itemTooltip', api); - }; - - TooltipView.type = 'tooltip'; - return TooltipView; - }(ComponentView); - /** - * From top to bottom. (the last one should be globalTooltipModel); - */ - - - function buildTooltipModel(modelCascade, globalTooltipModel, defaultTooltipOption) { - // Last is always tooltip model. - var ecModel = globalTooltipModel.ecModel; - var resultModel; - - if (defaultTooltipOption) { - resultModel = new Model(defaultTooltipOption, ecModel, ecModel); - resultModel = new Model(globalTooltipModel.option, resultModel, ecModel); - } else { - resultModel = globalTooltipModel; - } - - for (var i = modelCascade.length - 1; i >= 0; i--) { - var tooltipOpt = modelCascade[i]; - - if (tooltipOpt) { - if (tooltipOpt instanceof Model) { - tooltipOpt = tooltipOpt.get('tooltip', true); - } // In each data item tooltip can be simply write: - // { - // value: 10, - // tooltip: 'Something you need to know' - // } - - - if (isString(tooltipOpt)) { - tooltipOpt = { - formatter: tooltipOpt - }; - } - - if (tooltipOpt) { - resultModel = new Model(tooltipOpt, resultModel, ecModel); - } - } - } - - return resultModel; - } - - function makeDispatchAction$1(payload, api) { - return payload.dispatchAction || bind(api.dispatchAction, api); - } - - function refixTooltipPosition(x, y, content, viewWidth, viewHeight, gapH, gapV) { - var size = content.getSize(); - var width = size[0]; - var height = size[1]; - - if (gapH != null) { - // Add extra 2 pixels for this case: - // At present the "values" in default tooltip are using CSS `float: right`. - // When the right edge of the tooltip box is on the right side of the - // viewport, the `float` layout might push the "values" to the second line. - if (x + width + gapH + 2 > viewWidth) { - x -= width + gapH; - } else { - x += gapH; - } - } - - if (gapV != null) { - if (y + height + gapV > viewHeight) { - y -= height + gapV; - } else { - y += gapV; - } - } - - return [x, y]; - } - - function confineTooltipPosition(x, y, content, viewWidth, viewHeight) { - var size = content.getSize(); - var width = size[0]; - var height = size[1]; - x = Math.min(x + width, viewWidth) - width; - y = Math.min(y + height, viewHeight) - height; - x = Math.max(x, 0); - y = Math.max(y, 0); - return [x, y]; - } - - function calcTooltipPosition(position, rect, contentSize, borderWidth) { - var domWidth = contentSize[0]; - var domHeight = contentSize[1]; - var offset = Math.ceil(Math.SQRT2 * borderWidth) + 8; - var x = 0; - var y = 0; - var rectWidth = rect.width; - var rectHeight = rect.height; - - switch (position) { - case 'inside': - x = rect.x + rectWidth / 2 - domWidth / 2; - y = rect.y + rectHeight / 2 - domHeight / 2; - break; - - case 'top': - x = rect.x + rectWidth / 2 - domWidth / 2; - y = rect.y - domHeight - offset; - break; - - case 'bottom': - x = rect.x + rectWidth / 2 - domWidth / 2; - y = rect.y + rectHeight + offset; - break; - - case 'left': - x = rect.x - domWidth - offset; - y = rect.y + rectHeight / 2 - domHeight / 2; - break; - - case 'right': - x = rect.x + rectWidth + offset; - y = rect.y + rectHeight / 2 - domHeight / 2; - } - - return [x, y]; - } - - function isCenterAlign(align) { - return align === 'center' || align === 'middle'; - } - /** - * Find target component by payload like: - * ```js - * { legendId: 'some_id', name: 'xxx' } - * { toolboxIndex: 1, name: 'xxx' } - * { geoName: 'some_name', name: 'xxx' } - * ``` - * PENDING: at present only - * - * If not found, return null/undefined. - */ - - - function findComponentReference(payload, ecModel, api) { - var queryOptionMap = preParseFinder(payload).queryOptionMap; - var componentMainType = queryOptionMap.keys()[0]; - - if (!componentMainType || componentMainType === 'series') { - return; - } - - var queryResult = queryReferringComponents(ecModel, componentMainType, queryOptionMap.get(componentMainType), { - useDefault: false, - enableAll: false, - enableNone: false - }); - var model = queryResult.models[0]; - - if (!model) { - return; - } - - var view = api.getViewOfComponentModel(model); - var el; - view.group.traverse(function (subEl) { - var tooltipConfig = getECData(subEl).tooltipConfig; - - if (tooltipConfig && tooltipConfig.name === payload.name) { - el = subEl; - return true; // stop - } - }); - - if (el) { - return { - componentMainType: componentMainType, - componentIndex: model.componentIndex, - el: el - }; - } - } - - function install$A(registers) { - use(install$s); - registers.registerComponentModel(TooltipModel); - registers.registerComponentView(TooltipView); - /** - * @action - * @property {string} type - * @property {number} seriesIndex - * @property {number} dataIndex - * @property {number} [x] - * @property {number} [y] - */ - - registers.registerAction({ - type: 'showTip', - event: 'showTip', - update: 'tooltip:manuallyShowTip' - }, noop); - registers.registerAction({ - type: 'hideTip', - event: 'hideTip', - update: 'tooltip:manuallyHideTip' - }, noop); - } - - var DEFAULT_TOOLBOX_BTNS = ['rect', 'polygon', 'keep', 'clear']; - function brushPreprocessor(option, isNew) { - var brushComponents = normalizeToArray(option ? option.brush : []); - - if (!brushComponents.length) { - return; - } - - var brushComponentSpecifiedBtns = []; - each(brushComponents, function (brushOpt) { - var tbs = brushOpt.hasOwnProperty('toolbox') ? brushOpt.toolbox : []; - - if (tbs instanceof Array) { - brushComponentSpecifiedBtns = brushComponentSpecifiedBtns.concat(tbs); - } - }); - var toolbox = option && option.toolbox; - - if (isArray(toolbox)) { - toolbox = toolbox[0]; - } - - if (!toolbox) { - toolbox = { - feature: {} - }; - option.toolbox = [toolbox]; - } - - var toolboxFeature = toolbox.feature || (toolbox.feature = {}); - var toolboxBrush = toolboxFeature.brush || (toolboxFeature.brush = {}); - var brushTypes = toolboxBrush.type || (toolboxBrush.type = []); - brushTypes.push.apply(brushTypes, brushComponentSpecifiedBtns); - removeDuplicate(brushTypes); - - if (isNew && !brushTypes.length) { - brushTypes.push.apply(brushTypes, DEFAULT_TOOLBOX_BTNS); - } - } - - function removeDuplicate(arr) { - var map = {}; - each(arr, function (val) { - map[val] = 1; - }); - arr.length = 0; - each(map, function (flag, val) { - arr.push(val); - }); - } - - var each$b = each; - - function hasKeys(obj) { - if (obj) { - for (var name_1 in obj) { - if (obj.hasOwnProperty(name_1)) { - return true; - } - } - } - } - - function createVisualMappings(option, stateList, supplementVisualOption) { - var visualMappings = {}; - each$b(stateList, function (state) { - var mappings = visualMappings[state] = createMappings(); - each$b(option[state], function (visualData, visualType) { - if (!VisualMapping.isValidType(visualType)) { - return; - } - - var mappingOption = { - type: visualType, - visual: visualData - }; - supplementVisualOption && supplementVisualOption(mappingOption, state); - mappings[visualType] = new VisualMapping(mappingOption); // Prepare a alpha for opacity, for some case that opacity - // is not supported, such as rendering using gradient color. - - if (visualType === 'opacity') { - mappingOption = clone(mappingOption); - mappingOption.type = 'colorAlpha'; - mappings.__hidden.__alphaForOpacity = new VisualMapping(mappingOption); - } - }); - }); - return visualMappings; - - function createMappings() { - var Creater = function () {}; // Make sure hidden fields will not be visited by - // object iteration (with hasOwnProperty checking). - - - Creater.prototype.__hidden = Creater.prototype; - var obj = new Creater(); - return obj; - } - } - function replaceVisualOption(thisOption, newOption, keys) { - // Visual attributes merge is not supported, otherwise it - // brings overcomplicated merge logic. See #2853. So if - // newOption has anyone of these keys, all of these keys - // will be reset. Otherwise, all keys remain. - var has; - each(keys, function (key) { - if (newOption.hasOwnProperty(key) && hasKeys(newOption[key])) { - has = true; - } - }); - has && each(keys, function (key) { - if (newOption.hasOwnProperty(key) && hasKeys(newOption[key])) { - thisOption[key] = clone(newOption[key]); - } else { - delete thisOption[key]; - } - }); - } - /** - * @param stateList - * @param visualMappings - * @param list - * @param getValueState param: valueOrIndex, return: state. - * @param scope Scope for getValueState - * @param dimension Concrete dimension, if used. - */ - // ???! handle brush? - - function applyVisual(stateList, visualMappings, data, getValueState, scope, dimension) { - var visualTypesMap = {}; - each(stateList, function (state) { - var visualTypes = VisualMapping.prepareVisualTypes(visualMappings[state]); - visualTypesMap[state] = visualTypes; - }); - var dataIndex; - - function getVisual(key) { - return getItemVisualFromData(data, dataIndex, key); - } - - function setVisual(key, value) { - setItemVisualFromData(data, dataIndex, key, value); - } - - if (dimension == null) { - data.each(eachItem); - } else { - data.each([dimension], eachItem); - } - - function eachItem(valueOrIndex, index) { - dataIndex = dimension == null ? valueOrIndex // First argument is index - : index; - var rawDataItem = data.getRawDataItem(dataIndex); // Consider performance - // @ts-ignore - - if (rawDataItem && rawDataItem.visualMap === false) { - return; - } - - var valueState = getValueState.call(scope, valueOrIndex); - var mappings = visualMappings[valueState]; - var visualTypes = visualTypesMap[valueState]; - - for (var i = 0, len = visualTypes.length; i < len; i++) { - var type = visualTypes[i]; - mappings[type] && mappings[type].applyVisual(valueOrIndex, getVisual, setVisual); - } - } - } - /** - * @param data - * @param stateList - * @param visualMappings <state, Object.<visualType, module:echarts/visual/VisualMapping>> - * @param getValueState param: valueOrIndex, return: state. - * @param dim dimension or dimension index. - */ - - function incrementalApplyVisual(stateList, visualMappings, getValueState, dim) { - var visualTypesMap = {}; - each(stateList, function (state) { - var visualTypes = VisualMapping.prepareVisualTypes(visualMappings[state]); - visualTypesMap[state] = visualTypes; - }); - return { - progress: function progress(params, data) { - var dimIndex; - - if (dim != null) { - dimIndex = data.getDimensionIndex(dim); - } - - function getVisual(key) { - return getItemVisualFromData(data, dataIndex, key); - } - - function setVisual(key, value) { - setItemVisualFromData(data, dataIndex, key, value); - } - - var dataIndex; - var store = data.getStore(); - - while ((dataIndex = params.next()) != null) { - var rawDataItem = data.getRawDataItem(dataIndex); // Consider performance - // @ts-ignore - - if (rawDataItem && rawDataItem.visualMap === false) { - continue; - } - - var value = dim != null ? store.get(dimIndex, dataIndex) : dataIndex; - var valueState = getValueState(value); - var mappings = visualMappings[valueState]; - var visualTypes = visualTypesMap[valueState]; - - for (var i = 0, len = visualTypes.length; i < len; i++) { - var type = visualTypes[i]; - mappings[type] && mappings[type].applyVisual(value, getVisual, setVisual); - } - } - } - }; - } - - function makeBrushCommonSelectorForSeries(area) { - var brushType = area.brushType; // Do not use function binding or curry for performance. - - var selectors = { - point: function (itemLayout) { - return selector[brushType].point(itemLayout, selectors, area); - }, - rect: function (itemLayout) { - return selector[brushType].rect(itemLayout, selectors, area); - } - }; - return selectors; - } - var selector = { - lineX: getLineSelectors(0), - lineY: getLineSelectors(1), - rect: { - point: function (itemLayout, selectors, area) { - return itemLayout && area.boundingRect.contain(itemLayout[0], itemLayout[1]); - }, - rect: function (itemLayout, selectors, area) { - return itemLayout && area.boundingRect.intersect(itemLayout); - } - }, - polygon: { - point: function (itemLayout, selectors, area) { - return itemLayout && area.boundingRect.contain(itemLayout[0], itemLayout[1]) && contain$2(area.range, itemLayout[0], itemLayout[1]); - }, - rect: function (itemLayout, selectors, area) { - var points = area.range; - - if (!itemLayout || points.length <= 1) { - return false; - } - - var x = itemLayout.x; - var y = itemLayout.y; - var width = itemLayout.width; - var height = itemLayout.height; - var p = points[0]; - - if (contain$2(points, x, y) || contain$2(points, x + width, y) || contain$2(points, x, y + height) || contain$2(points, x + width, y + height) || BoundingRect.create(itemLayout).contain(p[0], p[1]) || linePolygonIntersect(x, y, x + width, y, points) || linePolygonIntersect(x, y, x, y + height, points) || linePolygonIntersect(x + width, y, x + width, y + height, points) || linePolygonIntersect(x, y + height, x + width, y + height, points)) { - return true; - } - } - } - }; - - function getLineSelectors(xyIndex) { - var xy = ['x', 'y']; - var wh = ['width', 'height']; - return { - point: function (itemLayout, selectors, area) { - if (itemLayout) { - var range = area.range; - var p = itemLayout[xyIndex]; - return inLineRange(p, range); - } - }, - rect: function (itemLayout, selectors, area) { - if (itemLayout) { - var range = area.range; - var layoutRange = [itemLayout[xy[xyIndex]], itemLayout[xy[xyIndex]] + itemLayout[wh[xyIndex]]]; - layoutRange[1] < layoutRange[0] && layoutRange.reverse(); - return inLineRange(layoutRange[0], range) || inLineRange(layoutRange[1], range) || inLineRange(range[0], layoutRange) || inLineRange(range[1], layoutRange); - } - } - }; - } - - function inLineRange(p, range) { - return range[0] <= p && p <= range[1]; - } - - var STATE_LIST = ['inBrush', 'outOfBrush']; - var DISPATCH_METHOD = '__ecBrushSelect'; - var DISPATCH_FLAG = '__ecInBrushSelectEvent'; - function layoutCovers(ecModel) { - ecModel.eachComponent({ - mainType: 'brush' - }, function (brushModel) { - var brushTargetManager = brushModel.brushTargetManager = new BrushTargetManager(brushModel.option, ecModel); - brushTargetManager.setInputRanges(brushModel.areas, ecModel); - }); - } - /** - * Register the visual encoding if this modules required. - */ - - function brushVisual(ecModel, api, payload) { - var brushSelected = []; - var throttleType; - var throttleDelay; - ecModel.eachComponent({ - mainType: 'brush' - }, function (brushModel) { - payload && payload.type === 'takeGlobalCursor' && brushModel.setBrushOption(payload.key === 'brush' ? payload.brushOption : { - brushType: false - }); - }); - layoutCovers(ecModel); - ecModel.eachComponent({ - mainType: 'brush' - }, function (brushModel, brushIndex) { - var thisBrushSelected = { - brushId: brushModel.id, - brushIndex: brushIndex, - brushName: brushModel.name, - areas: clone(brushModel.areas), - selected: [] - }; // Every brush component exists in event params, convenient - // for user to find by index. - - brushSelected.push(thisBrushSelected); - var brushOption = brushModel.option; - var brushLink = brushOption.brushLink; - var linkedSeriesMap = []; - var selectedDataIndexForLink = []; - var rangeInfoBySeries = []; - var hasBrushExists = false; - - if (!brushIndex) { - // Only the first throttle setting works. - throttleType = brushOption.throttleType; - throttleDelay = brushOption.throttleDelay; - } // Add boundingRect and selectors to range. - - - var areas = map(brushModel.areas, function (area) { - var builder = boundingRectBuilders[area.brushType]; - var selectableArea = defaults({ - boundingRect: builder ? builder(area) : void 0 - }, area); - selectableArea.selectors = makeBrushCommonSelectorForSeries(selectableArea); - return selectableArea; - }); - var visualMappings = createVisualMappings(brushModel.option, STATE_LIST, function (mappingOption) { - mappingOption.mappingMethod = 'fixed'; - }); - isArray(brushLink) && each(brushLink, function (seriesIndex) { - linkedSeriesMap[seriesIndex] = 1; - }); - - function linkOthers(seriesIndex) { - return brushLink === 'all' || !!linkedSeriesMap[seriesIndex]; - } // If no supported brush or no brush on the series, - // all visuals should be in original state. - - - function brushed(rangeInfoList) { - return !!rangeInfoList.length; - } - /** - * Logic for each series: (If the logic has to be modified one day, do it carefully!) - * - * ( brushed ┬ && ┬hasBrushExist ┬ && linkOthers ) => StepA: ┬record, ┬ StepB: ┬visualByRecord. - * !brushed┘ ├hasBrushExist ┤ └nothing,┘ ├visualByRecord. - * └!hasBrushExist┘ └nothing. - * ( !brushed && ┬hasBrushExist ┬ && linkOthers ) => StepA: nothing, StepB: ┬visualByRecord. - * └!hasBrushExist┘ └nothing. - * ( brushed ┬ && !linkOthers ) => StepA: nothing, StepB: ┬visualByCheck. - * !brushed┘ └nothing. - * ( !brushed && !linkOthers ) => StepA: nothing, StepB: nothing. - */ - // Step A - - - ecModel.eachSeries(function (seriesModel, seriesIndex) { - var rangeInfoList = rangeInfoBySeries[seriesIndex] = []; - seriesModel.subType === 'parallel' ? stepAParallel(seriesModel, seriesIndex) : stepAOthers(seriesModel, seriesIndex, rangeInfoList); - }); - - function stepAParallel(seriesModel, seriesIndex) { - var coordSys = seriesModel.coordinateSystem; - hasBrushExists = hasBrushExists || coordSys.hasAxisBrushed(); - linkOthers(seriesIndex) && coordSys.eachActiveState(seriesModel.getData(), function (activeState, dataIndex) { - activeState === 'active' && (selectedDataIndexForLink[dataIndex] = 1); - }); - } - - function stepAOthers(seriesModel, seriesIndex, rangeInfoList) { - if (!seriesModel.brushSelector || brushModelNotControll(brushModel, seriesIndex)) { - return; - } - - each(areas, function (area) { - if (brushModel.brushTargetManager.controlSeries(area, seriesModel, ecModel)) { - rangeInfoList.push(area); - } - - hasBrushExists = hasBrushExists || brushed(rangeInfoList); - }); - - if (linkOthers(seriesIndex) && brushed(rangeInfoList)) { - var data_1 = seriesModel.getData(); - data_1.each(function (dataIndex) { - if (checkInRange(seriesModel, rangeInfoList, data_1, dataIndex)) { - selectedDataIndexForLink[dataIndex] = 1; - } - }); - } - } // Step B - - - ecModel.eachSeries(function (seriesModel, seriesIndex) { - var seriesBrushSelected = { - seriesId: seriesModel.id, - seriesIndex: seriesIndex, - seriesName: seriesModel.name, - dataIndex: [] - }; // Every series exists in event params, convenient - // for user to find series by seriesIndex. - - thisBrushSelected.selected.push(seriesBrushSelected); - var rangeInfoList = rangeInfoBySeries[seriesIndex]; - var data = seriesModel.getData(); - var getValueState = linkOthers(seriesIndex) ? function (dataIndex) { - return selectedDataIndexForLink[dataIndex] ? (seriesBrushSelected.dataIndex.push(data.getRawIndex(dataIndex)), 'inBrush') : 'outOfBrush'; - } : function (dataIndex) { - return checkInRange(seriesModel, rangeInfoList, data, dataIndex) ? (seriesBrushSelected.dataIndex.push(data.getRawIndex(dataIndex)), 'inBrush') : 'outOfBrush'; - }; // If no supported brush or no brush, all visuals are in original state. - - (linkOthers(seriesIndex) ? hasBrushExists : brushed(rangeInfoList)) && applyVisual(STATE_LIST, visualMappings, data, getValueState); - }); - }); - dispatchAction(api, throttleType, throttleDelay, brushSelected, payload); - } - - function dispatchAction(api, throttleType, throttleDelay, brushSelected, payload) { - // This event will not be triggered when `setOpion`, otherwise dead lock may - // triggered when do `setOption` in event listener, which we do not find - // satisfactory way to solve yet. Some considered resolutions: - // (a) Diff with prevoius selected data ant only trigger event when changed. - // But store previous data and diff precisely (i.e., not only by dataIndex, but - // also detect value changes in selected data) might bring complexity or fragility. - // (b) Use spectial param like `silent` to suppress event triggering. - // But such kind of volatile param may be weird in `setOption`. - if (!payload) { - return; - } - - var zr = api.getZr(); - - if (zr[DISPATCH_FLAG]) { - return; - } - - if (!zr[DISPATCH_METHOD]) { - zr[DISPATCH_METHOD] = doDispatch; - } - - var fn = createOrUpdate(zr, DISPATCH_METHOD, throttleDelay, throttleType); - fn(api, brushSelected); - } - - function doDispatch(api, brushSelected) { - if (!api.isDisposed()) { - var zr = api.getZr(); - zr[DISPATCH_FLAG] = true; - api.dispatchAction({ - type: 'brushSelect', - batch: brushSelected - }); - zr[DISPATCH_FLAG] = false; - } - } - - function checkInRange(seriesModel, rangeInfoList, data, dataIndex) { - for (var i = 0, len = rangeInfoList.length; i < len; i++) { - var area = rangeInfoList[i]; - - if (seriesModel.brushSelector(dataIndex, data, area.selectors, area)) { - return true; - } - } - } - - function brushModelNotControll(brushModel, seriesIndex) { - var seriesIndices = brushModel.option.seriesIndex; - return seriesIndices != null && seriesIndices !== 'all' && (isArray(seriesIndices) ? indexOf(seriesIndices, seriesIndex) < 0 : seriesIndex !== seriesIndices); - } - - var boundingRectBuilders = { - rect: function (area) { - return getBoundingRectFromMinMax(area.range); - }, - polygon: function (area) { - var minMax; - var range = area.range; - - for (var i = 0, len = range.length; i < len; i++) { - minMax = minMax || [[Infinity, -Infinity], [Infinity, -Infinity]]; - var rg = range[i]; - rg[0] < minMax[0][0] && (minMax[0][0] = rg[0]); - rg[0] > minMax[0][1] && (minMax[0][1] = rg[0]); - rg[1] < minMax[1][0] && (minMax[1][0] = rg[1]); - rg[1] > minMax[1][1] && (minMax[1][1] = rg[1]); - } - - return minMax && getBoundingRectFromMinMax(minMax); - } - }; - - function getBoundingRectFromMinMax(minMax) { - return new BoundingRect(minMax[0][0], minMax[1][0], minMax[0][1] - minMax[0][0], minMax[1][1] - minMax[1][0]); - } - - var BrushView = - /** @class */ - function (_super) { - __extends(BrushView, _super); - - function BrushView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = BrushView.type; - return _this; - } - - BrushView.prototype.init = function (ecModel, api) { - this.ecModel = ecModel; - this.api = api; - this.model; - (this._brushController = new BrushController(api.getZr())).on('brush', bind(this._onBrush, this)).mount(); - }; - - BrushView.prototype.render = function (brushModel, ecModel, api, payload) { - this.model = brushModel; - - this._updateController(brushModel, ecModel, api, payload); - }; - - BrushView.prototype.updateTransform = function (brushModel, ecModel, api, payload) { - // PENDING: `updateTransform` is a little tricky, whose layout need - // to be calculate mandatorily and other stages will not be performed. - // Take care the correctness of the logic. See #11754 . - layoutCovers(ecModel); - - this._updateController(brushModel, ecModel, api, payload); - }; - - BrushView.prototype.updateVisual = function (brushModel, ecModel, api, payload) { - this.updateTransform(brushModel, ecModel, api, payload); - }; - - BrushView.prototype.updateView = function (brushModel, ecModel, api, payload) { - this._updateController(brushModel, ecModel, api, payload); - }; - - BrushView.prototype._updateController = function (brushModel, ecModel, api, payload) { - // Do not update controller when drawing. - (!payload || payload.$from !== brushModel.id) && this._brushController.setPanels(brushModel.brushTargetManager.makePanelOpts(api)).enableBrush(brushModel.brushOption).updateCovers(brushModel.areas.slice()); - }; // updateLayout: updateController, - // updateVisual: updateController, - - - BrushView.prototype.dispose = function () { - this._brushController.dispose(); - }; - - BrushView.prototype._onBrush = function (eventParam) { - var modelId = this.model.id; - var areas = this.model.brushTargetManager.setOutputRanges(eventParam.areas, this.ecModel); // Action is not dispatched on drag end, because the drag end - // emits the same params with the last drag move event, and - // may have some delay when using touch pad, which makes - // animation not smooth (when using debounce). - - (!eventParam.isEnd || eventParam.removeOnClick) && this.api.dispatchAction({ - type: 'brush', - brushId: modelId, - areas: clone(areas), - $from: modelId - }); - eventParam.isEnd && this.api.dispatchAction({ - type: 'brushEnd', - brushId: modelId, - areas: clone(areas), - $from: modelId - }); - }; - - BrushView.type = 'brush'; - return BrushView; - }(ComponentView); - - var DEFAULT_OUT_OF_BRUSH_COLOR = '#ddd'; - - var BrushModel = - /** @class */ - function (_super) { - __extends(BrushModel, _super); - - function BrushModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = BrushModel.type; - /** - * @readOnly - */ - - _this.areas = []; - /** - * Current brush painting area settings. - * @readOnly - */ - - _this.brushOption = {}; - return _this; - } - - BrushModel.prototype.optionUpdated = function (newOption, isInit) { - var thisOption = this.option; - !isInit && replaceVisualOption(thisOption, newOption, ['inBrush', 'outOfBrush']); - var inBrush = thisOption.inBrush = thisOption.inBrush || {}; // Always give default visual, consider setOption at the second time. - - thisOption.outOfBrush = thisOption.outOfBrush || { - color: DEFAULT_OUT_OF_BRUSH_COLOR - }; - - if (!inBrush.hasOwnProperty('liftZ')) { - // Bigger than the highlight z lift, otherwise it will - // be effected by the highlight z when brush. - inBrush.liftZ = 5; - } - }; - /** - * If `areas` is null/undefined, range state remain. - */ - - - BrushModel.prototype.setAreas = function (areas) { - if ("development" !== 'production') { - assert(isArray(areas)); - each(areas, function (area) { - assert(area.brushType, 'Illegal areas'); - }); - } // If areas is null/undefined, range state remain. - // This helps user to dispatchAction({type: 'brush'}) with no areas - // set but just want to get the current brush select info from a `brush` event. - - - if (!areas) { - return; - } - - this.areas = map(areas, function (area) { - return generateBrushOption(this.option, area); - }, this); - }; - /** - * Set the current painting brush option. - */ - - - BrushModel.prototype.setBrushOption = function (brushOption) { - this.brushOption = generateBrushOption(this.option, brushOption); - this.brushType = this.brushOption.brushType; - }; - - BrushModel.type = 'brush'; - BrushModel.dependencies = ['geo', 'grid', 'xAxis', 'yAxis', 'parallel', 'series']; - BrushModel.defaultOption = { - seriesIndex: 'all', - brushType: 'rect', - brushMode: 'single', - transformable: true, - brushStyle: { - borderWidth: 1, - color: 'rgba(210,219,238,0.3)', - borderColor: '#D2DBEE' - }, - throttleType: 'fixRate', - throttleDelay: 0, - removeOnClick: true, - z: 10000 - }; - return BrushModel; - }(ComponentModel); - - function generateBrushOption(option, brushOption) { - return merge({ - brushType: option.brushType, - brushMode: option.brushMode, - transformable: option.transformable, - brushStyle: new Model(option.brushStyle).getItemStyle(), - removeOnClick: option.removeOnClick, - z: option.z - }, brushOption, true); - } - - var ICON_TYPES = ['rect', 'polygon', 'lineX', 'lineY', 'keep', 'clear']; - - var BrushFeature = - /** @class */ - function (_super) { - __extends(BrushFeature, _super); - - function BrushFeature() { - return _super !== null && _super.apply(this, arguments) || this; - } - - BrushFeature.prototype.render = function (featureModel, ecModel, api) { - var brushType; - var brushMode; - var isBrushed; - ecModel.eachComponent({ - mainType: 'brush' - }, function (brushModel) { - brushType = brushModel.brushType; - brushMode = brushModel.brushOption.brushMode || 'single'; - isBrushed = isBrushed || !!brushModel.areas.length; - }); - this._brushType = brushType; - this._brushMode = brushMode; - each(featureModel.get('type', true), function (type) { - featureModel.setIconStatus(type, (type === 'keep' ? brushMode === 'multiple' : type === 'clear' ? isBrushed : type === brushType) ? 'emphasis' : 'normal'); - }); - }; - - BrushFeature.prototype.updateView = function (featureModel, ecModel, api) { - this.render(featureModel, ecModel, api); - }; - - BrushFeature.prototype.getIcons = function () { - var model = this.model; - var availableIcons = model.get('icon', true); - var icons = {}; - each(model.get('type', true), function (type) { - if (availableIcons[type]) { - icons[type] = availableIcons[type]; - } - }); - return icons; - }; - - BrushFeature.prototype.onclick = function (ecModel, api, type) { - var brushType = this._brushType; - var brushMode = this._brushMode; - - if (type === 'clear') { - // Trigger parallel action firstly - api.dispatchAction({ - type: 'axisAreaSelect', - intervals: [] - }); - api.dispatchAction({ - type: 'brush', - command: 'clear', - // Clear all areas of all brush components. - areas: [] - }); - } else { - api.dispatchAction({ - type: 'takeGlobalCursor', - key: 'brush', - brushOption: { - brushType: type === 'keep' ? brushType : brushType === type ? false : type, - brushMode: type === 'keep' ? brushMode === 'multiple' ? 'single' : 'multiple' : brushMode - } - }); - } - }; - - BrushFeature.getDefaultOption = function (ecModel) { - var defaultOption = { - show: true, - type: ICON_TYPES.slice(), - icon: { - /* eslint-disable */ - rect: 'M7.3,34.7 M0.4,10V-0.2h9.8 M89.6,10V-0.2h-9.8 M0.4,60v10.2h9.8 M89.6,60v10.2h-9.8 M12.3,22.4V10.5h13.1 M33.6,10.5h7.8 M49.1,10.5h7.8 M77.5,22.4V10.5h-13 M12.3,31.1v8.2 M77.7,31.1v8.2 M12.3,47.6v11.9h13.1 M33.6,59.5h7.6 M49.1,59.5 h7.7 M77.5,47.6v11.9h-13', - polygon: 'M55.2,34.9c1.7,0,3.1,1.4,3.1,3.1s-1.4,3.1-3.1,3.1 s-3.1-1.4-3.1-3.1S53.5,34.9,55.2,34.9z M50.4,51c1.7,0,3.1,1.4,3.1,3.1c0,1.7-1.4,3.1-3.1,3.1c-1.7,0-3.1-1.4-3.1-3.1 C47.3,52.4,48.7,51,50.4,51z M55.6,37.1l1.5-7.8 M60.1,13.5l1.6-8.7l-7.8,4 M59,19l-1,5.3 M24,16.1l6.4,4.9l6.4-3.3 M48.5,11.6 l-5.9,3.1 M19.1,12.8L9.7,5.1l1.1,7.7 M13.4,29.8l1,7.3l6.6,1.6 M11.6,18.4l1,6.1 M32.8,41.9 M26.6,40.4 M27.3,40.2l6.1,1.6 M49.9,52.1l-5.6-7.6l-4.9-1.2', - lineX: 'M15.2,30 M19.7,15.6V1.9H29 M34.8,1.9H40.4 M55.3,15.6V1.9H45.9 M19.7,44.4V58.1H29 M34.8,58.1H40.4 M55.3,44.4 V58.1H45.9 M12.5,20.3l-9.4,9.6l9.6,9.8 M3.1,29.9h16.5 M62.5,20.3l9.4,9.6L62.3,39.7 M71.9,29.9H55.4', - lineY: 'M38.8,7.7 M52.7,12h13.2v9 M65.9,26.6V32 M52.7,46.3h13.2v-9 M24.9,12H11.8v9 M11.8,26.6V32 M24.9,46.3H11.8v-9 M48.2,5.1l-9.3-9l-9.4,9.2 M38.9-3.9V12 M48.2,53.3l-9.3,9l-9.4-9.2 M38.9,62.3V46.4', - keep: 'M4,10.5V1h10.3 M20.7,1h6.1 M33,1h6.1 M55.4,10.5V1H45.2 M4,17.3v6.6 M55.6,17.3v6.6 M4,30.5V40h10.3 M20.7,40 h6.1 M33,40h6.1 M55.4,30.5V40H45.2 M21,18.9h62.9v48.6H21V18.9z', - clear: 'M22,14.7l30.9,31 M52.9,14.7L22,45.7 M4.7,16.8V4.2h13.1 M26,4.2h7.8 M41.6,4.2h7.8 M70.3,16.8V4.2H57.2 M4.7,25.9v8.6 M70.3,25.9v8.6 M4.7,43.2v12.6h13.1 M26,55.8h7.8 M41.6,55.8h7.8 M70.3,43.2v12.6H57.2' // jshint ignore:line - - /* eslint-enable */ - - }, - // `rect`, `polygon`, `lineX`, `lineY`, `keep`, `clear` - title: ecModel.getLocaleModel().get(['toolbox', 'brush', 'title']) - }; - return defaultOption; - }; - - return BrushFeature; - }(ToolboxFeature); - - function install$B(registers) { - registers.registerComponentView(BrushView); - registers.registerComponentModel(BrushModel); - registers.registerPreprocessor(brushPreprocessor); - registers.registerVisual(registers.PRIORITY.VISUAL.BRUSH, brushVisual); - registers.registerAction({ - type: 'brush', - event: 'brush', - update: 'updateVisual' - }, function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'brush', - query: payload - }, function (brushModel) { - brushModel.setAreas(payload.areas); - }); - }); - /** - * payload: { - * brushComponents: [ - * { - * brushId, - * brushIndex, - * brushName, - * series: [ - * { - * seriesId, - * seriesIndex, - * seriesName, - * rawIndices: [21, 34, ...] - * }, - * ... - * ] - * }, - * ... - * ] - * } - */ - - registers.registerAction({ - type: 'brushSelect', - event: 'brushSelected', - update: 'none' - }, noop); - registers.registerAction({ - type: 'brushEnd', - event: 'brushEnd', - update: 'none' - }, noop); - registerFeature('brush', BrushFeature); - } - - var TitleModel = - /** @class */ - function (_super) { - __extends(TitleModel, _super); - - function TitleModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TitleModel.type; - _this.layoutMode = { - type: 'box', - ignoreSize: true - }; - return _this; - } - - TitleModel.type = 'title'; - TitleModel.defaultOption = { - // zlevel: 0, - z: 6, - show: true, - text: '', - target: 'blank', - subtext: '', - subtarget: 'blank', - left: 0, - top: 0, - backgroundColor: 'rgba(0,0,0,0)', - borderColor: '#ccc', - borderWidth: 0, - padding: 5, - itemGap: 10, - textStyle: { - fontSize: 18, - fontWeight: 'bold', - color: '#464646' - }, - subtextStyle: { - fontSize: 12, - color: '#6E7079' - } - }; - return TitleModel; - }(ComponentModel); // View - - - var TitleView = - /** @class */ - function (_super) { - __extends(TitleView, _super); - - function TitleView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TitleView.type; - return _this; - } - - TitleView.prototype.render = function (titleModel, ecModel, api) { - this.group.removeAll(); - - if (!titleModel.get('show')) { - return; - } - - var group = this.group; - var textStyleModel = titleModel.getModel('textStyle'); - var subtextStyleModel = titleModel.getModel('subtextStyle'); - var textAlign = titleModel.get('textAlign'); - var textVerticalAlign = retrieve2(titleModel.get('textBaseline'), titleModel.get('textVerticalAlign')); - var textEl = new ZRText({ - style: createTextStyle(textStyleModel, { - text: titleModel.get('text'), - fill: textStyleModel.getTextColor() - }, { - disableBox: true - }), - z2: 10 - }); - var textRect = textEl.getBoundingRect(); - var subText = titleModel.get('subtext'); - var subTextEl = new ZRText({ - style: createTextStyle(subtextStyleModel, { - text: subText, - fill: subtextStyleModel.getTextColor(), - y: textRect.height + titleModel.get('itemGap'), - verticalAlign: 'top' - }, { - disableBox: true - }), - z2: 10 - }); - var link = titleModel.get('link'); - var sublink = titleModel.get('sublink'); - var triggerEvent = titleModel.get('triggerEvent', true); - textEl.silent = !link && !triggerEvent; - subTextEl.silent = !sublink && !triggerEvent; - - if (link) { - textEl.on('click', function () { - windowOpen(link, '_' + titleModel.get('target')); - }); - } - - if (sublink) { - subTextEl.on('click', function () { - windowOpen(sublink, '_' + titleModel.get('subtarget')); - }); - } - - getECData(textEl).eventData = getECData(subTextEl).eventData = triggerEvent ? { - componentType: 'title', - componentIndex: titleModel.componentIndex - } : null; - group.add(textEl); - subText && group.add(subTextEl); // If no subText, but add subTextEl, there will be an empty line. - - var groupRect = group.getBoundingRect(); - var layoutOption = titleModel.getBoxLayoutParams(); - layoutOption.width = groupRect.width; - layoutOption.height = groupRect.height; - var layoutRect = getLayoutRect(layoutOption, { - width: api.getWidth(), - height: api.getHeight() - }, titleModel.get('padding')); // Adjust text align based on position - - if (!textAlign) { - // Align left if title is on the left. center and right is same - textAlign = titleModel.get('left') || titleModel.get('right'); // @ts-ignore - - if (textAlign === 'middle') { - textAlign = 'center'; - } // Adjust layout by text align - - - if (textAlign === 'right') { - layoutRect.x += layoutRect.width; - } else if (textAlign === 'center') { - layoutRect.x += layoutRect.width / 2; - } - } - - if (!textVerticalAlign) { - textVerticalAlign = titleModel.get('top') || titleModel.get('bottom'); // @ts-ignore - - if (textVerticalAlign === 'center') { - textVerticalAlign = 'middle'; - } - - if (textVerticalAlign === 'bottom') { - layoutRect.y += layoutRect.height; - } else if (textVerticalAlign === 'middle') { - layoutRect.y += layoutRect.height / 2; - } - - textVerticalAlign = textVerticalAlign || 'top'; - } - - group.x = layoutRect.x; - group.y = layoutRect.y; - group.markRedraw(); - var alignStyle = { - align: textAlign, - verticalAlign: textVerticalAlign - }; - textEl.setStyle(alignStyle); - subTextEl.setStyle(alignStyle); // Render background - // Get groupRect again because textAlign has been changed - - groupRect = group.getBoundingRect(); - var padding = layoutRect.margin; - var style = titleModel.getItemStyle(['color', 'opacity']); - style.fill = titleModel.get('backgroundColor'); - var rect = new Rect({ - shape: { - x: groupRect.x - padding[3], - y: groupRect.y - padding[0], - width: groupRect.width + padding[1] + padding[3], - height: groupRect.height + padding[0] + padding[2], - r: titleModel.get('borderRadius') - }, - style: style, - subPixelOptimize: true, - silent: true - }); - group.add(rect); - }; - - TitleView.type = 'title'; - return TitleView; - }(ComponentView); - - function install$C(registers) { - registers.registerComponentModel(TitleModel); - registers.registerComponentView(TitleView); - } - - var TimelineModel = - /** @class */ - function (_super) { - __extends(TimelineModel, _super); - - function TimelineModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TimelineModel.type; - _this.layoutMode = 'box'; - return _this; - } - /** - * @override - */ - - - TimelineModel.prototype.init = function (option, parentModel, ecModel) { - this.mergeDefaultAndTheme(option, ecModel); - - this._initData(); - }; - /** - * @override - */ - - - TimelineModel.prototype.mergeOption = function (option) { - _super.prototype.mergeOption.apply(this, arguments); - - this._initData(); - }; - - TimelineModel.prototype.setCurrentIndex = function (currentIndex) { - if (currentIndex == null) { - currentIndex = this.option.currentIndex; - } - - var count = this._data.count(); - - if (this.option.loop) { - currentIndex = (currentIndex % count + count) % count; - } else { - currentIndex >= count && (currentIndex = count - 1); - currentIndex < 0 && (currentIndex = 0); - } - - this.option.currentIndex = currentIndex; - }; - /** - * @return {number} currentIndex - */ - - - TimelineModel.prototype.getCurrentIndex = function () { - return this.option.currentIndex; - }; - /** - * @return {boolean} - */ - - - TimelineModel.prototype.isIndexMax = function () { - return this.getCurrentIndex() >= this._data.count() - 1; - }; - /** - * @param {boolean} state true: play, false: stop - */ - - - TimelineModel.prototype.setPlayState = function (state) { - this.option.autoPlay = !!state; - }; - /** - * @return {boolean} true: play, false: stop - */ - - - TimelineModel.prototype.getPlayState = function () { - return !!this.option.autoPlay; - }; - /** - * @private - */ - - - TimelineModel.prototype._initData = function () { - var thisOption = this.option; - var dataArr = thisOption.data || []; - var axisType = thisOption.axisType; - var names = this._names = []; - var processedDataArr; - - if (axisType === 'category') { - processedDataArr = []; - each(dataArr, function (item, index) { - var value = convertOptionIdName(getDataItemValue(item), ''); - var newItem; - - if (isObject(item)) { - newItem = clone(item); - newItem.value = index; - } else { - newItem = index; - } - - processedDataArr.push(newItem); - names.push(value); - }); - } else { - processedDataArr = dataArr; - } - - var dimType = { - category: 'ordinal', - time: 'time', - value: 'number' - }[axisType] || 'number'; - var data = this._data = new SeriesData([{ - name: 'value', - type: dimType - }], this); - data.initData(processedDataArr, names); - }; - - TimelineModel.prototype.getData = function () { - return this._data; - }; - /** - * @public - * @return {Array.<string>} categoreis - */ - - - TimelineModel.prototype.getCategories = function () { - if (this.get('axisType') === 'category') { - return this._names.slice(); - } - }; - - TimelineModel.type = 'timeline'; - /** - * @protected - */ - - TimelineModel.defaultOption = { - // zlevel: 0, // 一级层叠 - z: 4, - show: true, - axisType: 'time', - realtime: true, - left: '20%', - top: null, - right: '20%', - bottom: 0, - width: null, - height: 40, - padding: 5, - controlPosition: 'left', - autoPlay: false, - rewind: false, - loop: true, - playInterval: 2000, - currentIndex: 0, - itemStyle: {}, - label: { - color: '#000' - }, - data: [] - }; - return TimelineModel; - }(ComponentModel); - - var SliderTimelineModel = - /** @class */ - function (_super) { - __extends(SliderTimelineModel, _super); - - function SliderTimelineModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SliderTimelineModel.type; - return _this; - } - - SliderTimelineModel.type = 'timeline.slider'; - /** - * @protected - */ - - SliderTimelineModel.defaultOption = inheritDefaultOption(TimelineModel.defaultOption, { - backgroundColor: 'rgba(0,0,0,0)', - borderColor: '#ccc', - borderWidth: 0, - orient: 'horizontal', - inverse: false, - tooltip: { - trigger: 'item' // data item may also have tootip attr. - - }, - symbol: 'circle', - symbolSize: 12, - lineStyle: { - show: true, - width: 2, - color: '#DAE1F5' - }, - label: { - position: 'auto', - // When using number, label position is not - // restricted by viewRect. - // positive: right/bottom, negative: left/top - show: true, - interval: 'auto', - rotate: 0, - // formatter: null, - // 其余属性默认使用全局文本样式,详见TEXTSTYLE - color: '#A4B1D7' - }, - itemStyle: { - color: '#A4B1D7', - borderWidth: 1 - }, - checkpointStyle: { - symbol: 'circle', - symbolSize: 15, - color: '#316bf3', - borderColor: '#fff', - borderWidth: 2, - shadowBlur: 2, - shadowOffsetX: 1, - shadowOffsetY: 1, - shadowColor: 'rgba(0, 0, 0, 0.3)', - // borderColor: 'rgba(194,53,49, 0.5)', - animation: true, - animationDuration: 300, - animationEasing: 'quinticInOut' - }, - controlStyle: { - show: true, - showPlayBtn: true, - showPrevBtn: true, - showNextBtn: true, - itemSize: 24, - itemGap: 12, - position: 'left', - playIcon: 'path://M31.6,53C17.5,53,6,41.5,6,27.4S17.5,1.8,31.6,1.8C45.7,1.8,57.2,13.3,57.2,27.4S45.7,53,31.6,53z M31.6,3.3 C18.4,3.3,7.5,14.1,7.5,27.4c0,13.3,10.8,24.1,24.1,24.1C44.9,51.5,55.7,40.7,55.7,27.4C55.7,14.1,44.9,3.3,31.6,3.3z M24.9,21.3 c0-2.2,1.6-3.1,3.5-2l10.5,6.1c1.899,1.1,1.899,2.9,0,4l-10.5,6.1c-1.9,1.1-3.5,0.2-3.5-2V21.3z', - stopIcon: 'path://M30.9,53.2C16.8,53.2,5.3,41.7,5.3,27.6S16.8,2,30.9,2C45,2,56.4,13.5,56.4,27.6S45,53.2,30.9,53.2z M30.9,3.5C17.6,3.5,6.8,14.4,6.8,27.6c0,13.3,10.8,24.1,24.101,24.1C44.2,51.7,55,40.9,55,27.6C54.9,14.4,44.1,3.5,30.9,3.5z M36.9,35.8c0,0.601-0.4,1-0.9,1h-1.3c-0.5,0-0.9-0.399-0.9-1V19.5c0-0.6,0.4-1,0.9-1H36c0.5,0,0.9,0.4,0.9,1V35.8z M27.8,35.8 c0,0.601-0.4,1-0.9,1h-1.3c-0.5,0-0.9-0.399-0.9-1V19.5c0-0.6,0.4-1,0.9-1H27c0.5,0,0.9,0.4,0.9,1L27.8,35.8L27.8,35.8z', - // eslint-disable-next-line max-len - nextIcon: 'M2,18.5A1.52,1.52,0,0,1,.92,18a1.49,1.49,0,0,1,0-2.12L7.81,9.36,1,3.11A1.5,1.5,0,1,1,3,.89l8,7.34a1.48,1.48,0,0,1,.49,1.09,1.51,1.51,0,0,1-.46,1.1L3,18.08A1.5,1.5,0,0,1,2,18.5Z', - // eslint-disable-next-line max-len - prevIcon: 'M10,.5A1.52,1.52,0,0,1,11.08,1a1.49,1.49,0,0,1,0,2.12L4.19,9.64,11,15.89a1.5,1.5,0,1,1-2,2.22L1,10.77A1.48,1.48,0,0,1,.5,9.68,1.51,1.51,0,0,1,1,8.58L9,.92A1.5,1.5,0,0,1,10,.5Z', - prevBtnSize: 18, - nextBtnSize: 18, - color: '#A4B1D7', - borderColor: '#A4B1D7', - borderWidth: 1 - }, - emphasis: { - label: { - show: true, - // 其余属性默认使用全局文本样式,详见TEXTSTYLE - color: '#6f778d' - }, - itemStyle: { - color: '#316BF3' - }, - controlStyle: { - color: '#316BF3', - borderColor: '#316BF3', - borderWidth: 2 - } - }, - progress: { - lineStyle: { - color: '#316BF3' - }, - itemStyle: { - color: '#316BF3' - }, - label: { - color: '#6f778d' - } - }, - data: [] - }); - return SliderTimelineModel; - }(TimelineModel); - - mixin(SliderTimelineModel, DataFormatMixin.prototype); - - var TimelineView = - /** @class */ - function (_super) { - __extends(TimelineView, _super); - - function TimelineView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = TimelineView.type; - return _this; - } - - TimelineView.type = 'timeline'; - return TimelineView; - }(ComponentView); - - /** - * Extend axis 2d - */ - - var TimelineAxis = - /** @class */ - function (_super) { - __extends(TimelineAxis, _super); - - function TimelineAxis(dim, scale, coordExtent, axisType) { - var _this = _super.call(this, dim, scale, coordExtent) || this; - - _this.type = axisType || 'value'; - return _this; - } - /** - * @override - */ - - - TimelineAxis.prototype.getLabelModel = function () { - // Force override - return this.model.getModel('label'); - }; - /** - * @override - */ - - - TimelineAxis.prototype.isHorizontal = function () { - return this.model.get('orient') === 'horizontal'; - }; - - return TimelineAxis; - }(Axis); - - var PI$8 = Math.PI; - var labelDataIndexStore = makeInner(); - - var SliderTimelineView = - /** @class */ - function (_super) { - __extends(SliderTimelineView, _super); - - function SliderTimelineView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SliderTimelineView.type; - return _this; - } - - SliderTimelineView.prototype.init = function (ecModel, api) { - this.api = api; - }; - /** - * @override - */ - - - SliderTimelineView.prototype.render = function (timelineModel, ecModel, api) { - this.model = timelineModel; - this.api = api; - this.ecModel = ecModel; - this.group.removeAll(); - - if (timelineModel.get('show', true)) { - var layoutInfo_1 = this._layout(timelineModel, api); - - var mainGroup_1 = this._createGroup('_mainGroup'); - - var labelGroup = this._createGroup('_labelGroup'); - - var axis_1 = this._axis = this._createAxis(layoutInfo_1, timelineModel); - - timelineModel.formatTooltip = function (dataIndex) { - var name = axis_1.scale.getLabel({ - value: dataIndex - }); - return createTooltipMarkup('nameValue', { - noName: true, - value: name - }); - }; - - each(['AxisLine', 'AxisTick', 'Control', 'CurrentPointer'], function (name) { - this['_render' + name](layoutInfo_1, mainGroup_1, axis_1, timelineModel); - }, this); - - this._renderAxisLabel(layoutInfo_1, labelGroup, axis_1, timelineModel); - - this._position(layoutInfo_1, timelineModel); - } - - this._doPlayStop(); - - this._updateTicksStatus(); - }; - /** - * @override - */ - - - SliderTimelineView.prototype.remove = function () { - this._clearTimer(); - - this.group.removeAll(); - }; - /** - * @override - */ - - - SliderTimelineView.prototype.dispose = function () { - this._clearTimer(); - }; - - SliderTimelineView.prototype._layout = function (timelineModel, api) { - var labelPosOpt = timelineModel.get(['label', 'position']); - var orient = timelineModel.get('orient'); - var viewRect = getViewRect$5(timelineModel, api); - var parsedLabelPos; // Auto label offset. - - if (labelPosOpt == null || labelPosOpt === 'auto') { - parsedLabelPos = orient === 'horizontal' ? viewRect.y + viewRect.height / 2 < api.getHeight() / 2 ? '-' : '+' : viewRect.x + viewRect.width / 2 < api.getWidth() / 2 ? '+' : '-'; - } else if (isString(labelPosOpt)) { - parsedLabelPos = { - horizontal: { - top: '-', - bottom: '+' - }, - vertical: { - left: '-', - right: '+' - } - }[orient][labelPosOpt]; - } else { - // is number - parsedLabelPos = labelPosOpt; - } - - var labelAlignMap = { - horizontal: 'center', - vertical: parsedLabelPos >= 0 || parsedLabelPos === '+' ? 'left' : 'right' - }; - var labelBaselineMap = { - horizontal: parsedLabelPos >= 0 || parsedLabelPos === '+' ? 'top' : 'bottom', - vertical: 'middle' - }; - var rotationMap = { - horizontal: 0, - vertical: PI$8 / 2 - }; // Position - - var mainLength = orient === 'vertical' ? viewRect.height : viewRect.width; - var controlModel = timelineModel.getModel('controlStyle'); - var showControl = controlModel.get('show', true); - var controlSize = showControl ? controlModel.get('itemSize') : 0; - var controlGap = showControl ? controlModel.get('itemGap') : 0; - var sizePlusGap = controlSize + controlGap; // Special label rotate. - - var labelRotation = timelineModel.get(['label', 'rotate']) || 0; - labelRotation = labelRotation * PI$8 / 180; // To radian. - - var playPosition; - var prevBtnPosition; - var nextBtnPosition; - var controlPosition = controlModel.get('position', true); - var showPlayBtn = showControl && controlModel.get('showPlayBtn', true); - var showPrevBtn = showControl && controlModel.get('showPrevBtn', true); - var showNextBtn = showControl && controlModel.get('showNextBtn', true); - var xLeft = 0; - var xRight = mainLength; // position[0] means left, position[1] means middle. - - if (controlPosition === 'left' || controlPosition === 'bottom') { - showPlayBtn && (playPosition = [0, 0], xLeft += sizePlusGap); - showPrevBtn && (prevBtnPosition = [xLeft, 0], xLeft += sizePlusGap); - showNextBtn && (nextBtnPosition = [xRight - controlSize, 0], xRight -= sizePlusGap); - } else { - // 'top' 'right' - showPlayBtn && (playPosition = [xRight - controlSize, 0], xRight -= sizePlusGap); - showPrevBtn && (prevBtnPosition = [0, 0], xLeft += sizePlusGap); - showNextBtn && (nextBtnPosition = [xRight - controlSize, 0], xRight -= sizePlusGap); - } - - var axisExtent = [xLeft, xRight]; - - if (timelineModel.get('inverse')) { - axisExtent.reverse(); - } - - return { - viewRect: viewRect, - mainLength: mainLength, - orient: orient, - rotation: rotationMap[orient], - labelRotation: labelRotation, - labelPosOpt: parsedLabelPos, - labelAlign: timelineModel.get(['label', 'align']) || labelAlignMap[orient], - labelBaseline: timelineModel.get(['label', 'verticalAlign']) || timelineModel.get(['label', 'baseline']) || labelBaselineMap[orient], - // Based on mainGroup. - playPosition: playPosition, - prevBtnPosition: prevBtnPosition, - nextBtnPosition: nextBtnPosition, - axisExtent: axisExtent, - controlSize: controlSize, - controlGap: controlGap - }; - }; - - SliderTimelineView.prototype._position = function (layoutInfo, timelineModel) { - // Position is be called finally, because bounding rect is needed for - // adapt content to fill viewRect (auto adapt offset). - // Timeline may be not all in the viewRect when 'offset' is specified - // as a number, because it is more appropriate that label aligns at - // 'offset' but not the other edge defined by viewRect. - var mainGroup = this._mainGroup; - var labelGroup = this._labelGroup; - var viewRect = layoutInfo.viewRect; - - if (layoutInfo.orient === 'vertical') { - // transform to horizontal, inverse rotate by left-top point. - var m = create$1(); - var rotateOriginX = viewRect.x; - var rotateOriginY = viewRect.y + viewRect.height; - translate(m, m, [-rotateOriginX, -rotateOriginY]); - rotate(m, m, -PI$8 / 2); - translate(m, m, [rotateOriginX, rotateOriginY]); - viewRect = viewRect.clone(); - viewRect.applyTransform(m); - } - - var viewBound = getBound(viewRect); - var mainBound = getBound(mainGroup.getBoundingRect()); - var labelBound = getBound(labelGroup.getBoundingRect()); - var mainPosition = [mainGroup.x, mainGroup.y]; - var labelsPosition = [labelGroup.x, labelGroup.y]; - labelsPosition[0] = mainPosition[0] = viewBound[0][0]; - var labelPosOpt = layoutInfo.labelPosOpt; - - if (labelPosOpt == null || isString(labelPosOpt)) { - // '+' or '-' - var mainBoundIdx = labelPosOpt === '+' ? 0 : 1; - toBound(mainPosition, mainBound, viewBound, 1, mainBoundIdx); - toBound(labelsPosition, labelBound, viewBound, 1, 1 - mainBoundIdx); - } else { - var mainBoundIdx = labelPosOpt >= 0 ? 0 : 1; - toBound(mainPosition, mainBound, viewBound, 1, mainBoundIdx); - labelsPosition[1] = mainPosition[1] + labelPosOpt; - } - - mainGroup.setPosition(mainPosition); - labelGroup.setPosition(labelsPosition); - mainGroup.rotation = labelGroup.rotation = layoutInfo.rotation; - setOrigin(mainGroup); - setOrigin(labelGroup); - - function setOrigin(targetGroup) { - targetGroup.originX = viewBound[0][0] - targetGroup.x; - targetGroup.originY = viewBound[1][0] - targetGroup.y; - } - - function getBound(rect) { - // [[xmin, xmax], [ymin, ymax]] - return [[rect.x, rect.x + rect.width], [rect.y, rect.y + rect.height]]; - } - - function toBound(fromPos, from, to, dimIdx, boundIdx) { - fromPos[dimIdx] += to[dimIdx][boundIdx] - from[dimIdx][boundIdx]; - } - }; - - SliderTimelineView.prototype._createAxis = function (layoutInfo, timelineModel) { - var data = timelineModel.getData(); - var axisType = timelineModel.get('axisType'); - var scale = createScaleByModel$1(timelineModel, axisType); // Customize scale. The `tickValue` is `dataIndex`. - - scale.getTicks = function () { - return data.mapArray(['value'], function (value) { - return { - value: value - }; - }); - }; - - var dataExtent = data.getDataExtent('value'); - scale.setExtent(dataExtent[0], dataExtent[1]); - scale.calcNiceTicks(); - var axis = new TimelineAxis('value', scale, layoutInfo.axisExtent, axisType); - axis.model = timelineModel; - return axis; - }; - - SliderTimelineView.prototype._createGroup = function (key) { - var newGroup = this[key] = new Group(); - this.group.add(newGroup); - return newGroup; - }; - - SliderTimelineView.prototype._renderAxisLine = function (layoutInfo, group, axis, timelineModel) { - var axisExtent = axis.getExtent(); - - if (!timelineModel.get(['lineStyle', 'show'])) { - return; - } - - var line = new Line({ - shape: { - x1: axisExtent[0], - y1: 0, - x2: axisExtent[1], - y2: 0 - }, - style: extend({ - lineCap: 'round' - }, timelineModel.getModel('lineStyle').getLineStyle()), - silent: true, - z2: 1 - }); - group.add(line); - var progressLine = this._progressLine = new Line({ - shape: { - x1: axisExtent[0], - x2: this._currentPointer ? this._currentPointer.x : axisExtent[0], - y1: 0, - y2: 0 - }, - style: defaults({ - lineCap: 'round', - lineWidth: line.style.lineWidth - }, timelineModel.getModel(['progress', 'lineStyle']).getLineStyle()), - silent: true, - z2: 1 - }); - group.add(progressLine); - }; - - SliderTimelineView.prototype._renderAxisTick = function (layoutInfo, group, axis, timelineModel) { - var _this = this; - - var data = timelineModel.getData(); // Show all ticks, despite ignoring strategy. - - var ticks = axis.scale.getTicks(); - this._tickSymbols = []; // The value is dataIndex, see the customized scale. - - each(ticks, function (tick) { - var tickCoord = axis.dataToCoord(tick.value); - var itemModel = data.getItemModel(tick.value); - var itemStyleModel = itemModel.getModel('itemStyle'); - var hoverStyleModel = itemModel.getModel(['emphasis', 'itemStyle']); - var progressStyleModel = itemModel.getModel(['progress', 'itemStyle']); - var symbolOpt = { - x: tickCoord, - y: 0, - onclick: bind(_this._changeTimeline, _this, tick.value) - }; - var el = giveSymbol(itemModel, itemStyleModel, group, symbolOpt); - el.ensureState('emphasis').style = hoverStyleModel.getItemStyle(); - el.ensureState('progress').style = progressStyleModel.getItemStyle(); - enableHoverEmphasis(el); - var ecData = getECData(el); - - if (itemModel.get('tooltip')) { - ecData.dataIndex = tick.value; - ecData.dataModel = timelineModel; - } else { - ecData.dataIndex = ecData.dataModel = null; - } - - _this._tickSymbols.push(el); - }); - }; - - SliderTimelineView.prototype._renderAxisLabel = function (layoutInfo, group, axis, timelineModel) { - var _this = this; - - var labelModel = axis.getLabelModel(); - - if (!labelModel.get('show')) { - return; - } - - var data = timelineModel.getData(); - var labels = axis.getViewLabels(); - this._tickLabels = []; - each(labels, function (labelItem) { - // The tickValue is dataIndex, see the customized scale. - var dataIndex = labelItem.tickValue; - var itemModel = data.getItemModel(dataIndex); - var normalLabelModel = itemModel.getModel('label'); - var hoverLabelModel = itemModel.getModel(['emphasis', 'label']); - var progressLabelModel = itemModel.getModel(['progress', 'label']); - var tickCoord = axis.dataToCoord(labelItem.tickValue); - var textEl = new ZRText({ - x: tickCoord, - y: 0, - rotation: layoutInfo.labelRotation - layoutInfo.rotation, - onclick: bind(_this._changeTimeline, _this, dataIndex), - silent: false, - style: createTextStyle(normalLabelModel, { - text: labelItem.formattedLabel, - align: layoutInfo.labelAlign, - verticalAlign: layoutInfo.labelBaseline - }) - }); - textEl.ensureState('emphasis').style = createTextStyle(hoverLabelModel); - textEl.ensureState('progress').style = createTextStyle(progressLabelModel); - group.add(textEl); - enableHoverEmphasis(textEl); - labelDataIndexStore(textEl).dataIndex = dataIndex; - - _this._tickLabels.push(textEl); - }); - }; - - SliderTimelineView.prototype._renderControl = function (layoutInfo, group, axis, timelineModel) { - var controlSize = layoutInfo.controlSize; - var rotation = layoutInfo.rotation; - var itemStyle = timelineModel.getModel('controlStyle').getItemStyle(); - var hoverStyle = timelineModel.getModel(['emphasis', 'controlStyle']).getItemStyle(); - var playState = timelineModel.getPlayState(); - var inverse = timelineModel.get('inverse', true); - makeBtn(layoutInfo.nextBtnPosition, 'next', bind(this._changeTimeline, this, inverse ? '-' : '+')); - makeBtn(layoutInfo.prevBtnPosition, 'prev', bind(this._changeTimeline, this, inverse ? '+' : '-')); - makeBtn(layoutInfo.playPosition, playState ? 'stop' : 'play', bind(this._handlePlayClick, this, !playState), true); - - function makeBtn(position, iconName, onclick, willRotate) { - if (!position) { - return; - } - - var iconSize = parsePercent(retrieve2(timelineModel.get(['controlStyle', iconName + 'BtnSize']), controlSize), controlSize); - var rect = [0, -iconSize / 2, iconSize, iconSize]; - var btn = makeControlIcon(timelineModel, iconName + 'Icon', rect, { - x: position[0], - y: position[1], - originX: controlSize / 2, - originY: 0, - rotation: willRotate ? -rotation : 0, - rectHover: true, - style: itemStyle, - onclick: onclick - }); - btn.ensureState('emphasis').style = hoverStyle; - group.add(btn); - enableHoverEmphasis(btn); - } - }; - - SliderTimelineView.prototype._renderCurrentPointer = function (layoutInfo, group, axis, timelineModel) { - var data = timelineModel.getData(); - var currentIndex = timelineModel.getCurrentIndex(); - var pointerModel = data.getItemModel(currentIndex).getModel('checkpointStyle'); - var me = this; - var callback = { - onCreate: function (pointer) { - pointer.draggable = true; - pointer.drift = bind(me._handlePointerDrag, me); - pointer.ondragend = bind(me._handlePointerDragend, me); - pointerMoveTo(pointer, me._progressLine, currentIndex, axis, timelineModel, true); - }, - onUpdate: function (pointer) { - pointerMoveTo(pointer, me._progressLine, currentIndex, axis, timelineModel); - } - }; // Reuse when exists, for animation and drag. - - this._currentPointer = giveSymbol(pointerModel, pointerModel, this._mainGroup, {}, this._currentPointer, callback); - }; - - SliderTimelineView.prototype._handlePlayClick = function (nextState) { - this._clearTimer(); - - this.api.dispatchAction({ - type: 'timelinePlayChange', - playState: nextState, - from: this.uid - }); - }; - - SliderTimelineView.prototype._handlePointerDrag = function (dx, dy, e) { - this._clearTimer(); - - this._pointerChangeTimeline([e.offsetX, e.offsetY]); - }; - - SliderTimelineView.prototype._handlePointerDragend = function (e) { - this._pointerChangeTimeline([e.offsetX, e.offsetY], true); - }; - - SliderTimelineView.prototype._pointerChangeTimeline = function (mousePos, trigger) { - var toCoord = this._toAxisCoord(mousePos)[0]; - - var axis = this._axis; - var axisExtent = asc(axis.getExtent().slice()); - toCoord > axisExtent[1] && (toCoord = axisExtent[1]); - toCoord < axisExtent[0] && (toCoord = axisExtent[0]); - this._currentPointer.x = toCoord; - - this._currentPointer.markRedraw(); - - var progressLine = this._progressLine; - - if (progressLine) { - progressLine.shape.x2 = toCoord; - progressLine.dirty(); - } - - var targetDataIndex = this._findNearestTick(toCoord); - - var timelineModel = this.model; - - if (trigger || targetDataIndex !== timelineModel.getCurrentIndex() && timelineModel.get('realtime')) { - this._changeTimeline(targetDataIndex); - } - }; - - SliderTimelineView.prototype._doPlayStop = function () { - var _this = this; - - this._clearTimer(); - - if (this.model.getPlayState()) { - this._timer = setTimeout(function () { - // Do not cache - var timelineModel = _this.model; - - _this._changeTimeline(timelineModel.getCurrentIndex() + (timelineModel.get('rewind', true) ? -1 : 1)); - }, this.model.get('playInterval')); - } - }; - - SliderTimelineView.prototype._toAxisCoord = function (vertex) { - var trans = this._mainGroup.getLocalTransform(); - - return applyTransform$1(vertex, trans, true); - }; - - SliderTimelineView.prototype._findNearestTick = function (axisCoord) { - var data = this.model.getData(); - var dist = Infinity; - var targetDataIndex; - var axis = this._axis; - data.each(['value'], function (value, dataIndex) { - var coord = axis.dataToCoord(value); - var d = Math.abs(coord - axisCoord); - - if (d < dist) { - dist = d; - targetDataIndex = dataIndex; - } - }); - return targetDataIndex; - }; - - SliderTimelineView.prototype._clearTimer = function () { - if (this._timer) { - clearTimeout(this._timer); - this._timer = null; - } - }; - - SliderTimelineView.prototype._changeTimeline = function (nextIndex) { - var currentIndex = this.model.getCurrentIndex(); - - if (nextIndex === '+') { - nextIndex = currentIndex + 1; - } else if (nextIndex === '-') { - nextIndex = currentIndex - 1; - } - - this.api.dispatchAction({ - type: 'timelineChange', - currentIndex: nextIndex, - from: this.uid - }); - }; - - SliderTimelineView.prototype._updateTicksStatus = function () { - var currentIndex = this.model.getCurrentIndex(); - var tickSymbols = this._tickSymbols; - var tickLabels = this._tickLabels; - - if (tickSymbols) { - for (var i = 0; i < tickSymbols.length; i++) { - tickSymbols && tickSymbols[i] && tickSymbols[i].toggleState('progress', i < currentIndex); - } - } - - if (tickLabels) { - for (var i = 0; i < tickLabels.length; i++) { - tickLabels && tickLabels[i] && tickLabels[i].toggleState('progress', labelDataIndexStore(tickLabels[i]).dataIndex <= currentIndex); - } - } - }; - - SliderTimelineView.type = 'timeline.slider'; - return SliderTimelineView; - }(TimelineView); - - function createScaleByModel$1(model, axisType) { - axisType = axisType || model.get('type'); - - if (axisType) { - switch (axisType) { - // Buildin scale - case 'category': - return new OrdinalScale({ - ordinalMeta: model.getCategories(), - extent: [Infinity, -Infinity] - }); - - case 'time': - return new TimeScale({ - locale: model.ecModel.getLocaleModel(), - useUTC: model.ecModel.get('useUTC') - }); - - default: - // default to be value - return new IntervalScale(); - } - } - } - - function getViewRect$5(model, api) { - return getLayoutRect(model.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }, model.get('padding')); - } - - function makeControlIcon(timelineModel, objPath, rect, opts) { - var style = opts.style; - var icon = createIcon(timelineModel.get(['controlStyle', objPath]), opts || {}, new BoundingRect(rect[0], rect[1], rect[2], rect[3])); // TODO createIcon won't use style in opt. - - if (style) { - icon.setStyle(style); - } - - return icon; - } - /** - * Create symbol or update symbol - * opt: basic position and event handlers - */ - - - function giveSymbol(hostModel, itemStyleModel, group, opt, symbol, callback) { - var color = itemStyleModel.get('color'); - - if (!symbol) { - var symbolType = hostModel.get('symbol'); - symbol = createSymbol(symbolType, -1, -1, 2, 2, color); - symbol.setStyle('strokeNoScale', true); - group.add(symbol); - callback && callback.onCreate(symbol); - } else { - symbol.setColor(color); - group.add(symbol); // Group may be new, also need to add. - - callback && callback.onUpdate(symbol); - } // Style - - - var itemStyle = itemStyleModel.getItemStyle(['color']); - symbol.setStyle(itemStyle); // Transform and events. - - opt = merge({ - rectHover: true, - z2: 100 - }, opt, true); - var symbolSize = normalizeSymbolSize(hostModel.get('symbolSize')); - opt.scaleX = symbolSize[0] / 2; - opt.scaleY = symbolSize[1] / 2; - var symbolOffset = normalizeSymbolOffset(hostModel.get('symbolOffset'), symbolSize); - - if (symbolOffset) { - opt.x = (opt.x || 0) + symbolOffset[0]; - opt.y = (opt.y || 0) + symbolOffset[1]; - } - - var symbolRotate = hostModel.get('symbolRotate'); - opt.rotation = (symbolRotate || 0) * Math.PI / 180 || 0; - symbol.attr(opt); // FIXME - // (1) When symbol.style.strokeNoScale is true and updateTransform is not performed, - // getBoundingRect will return wrong result. - // (This is supposed to be resolved in zrender, but it is a little difficult to - // leverage performance and auto updateTransform) - // (2) All of ancesters of symbol do not scale, so we can just updateTransform symbol. - - symbol.updateTransform(); - return symbol; - } - - function pointerMoveTo(pointer, progressLine, dataIndex, axis, timelineModel, noAnimation) { - if (pointer.dragging) { - return; - } - - var pointerModel = timelineModel.getModel('checkpointStyle'); - var toCoord = axis.dataToCoord(timelineModel.getData().get('value', dataIndex)); - - if (noAnimation || !pointerModel.get('animation', true)) { - pointer.attr({ - x: toCoord, - y: 0 - }); - progressLine && progressLine.attr({ - shape: { - x2: toCoord - } - }); - } else { - var animationCfg = { - duration: pointerModel.get('animationDuration', true), - easing: pointerModel.get('animationEasing', true) - }; - pointer.stopAnimation(null, true); - pointer.animateTo({ - x: toCoord, - y: 0 - }, animationCfg); - progressLine && progressLine.animateTo({ - shape: { - x2: toCoord - } - }, animationCfg); - } - } - - function installTimelineAction(registers) { - registers.registerAction({ - type: 'timelineChange', - event: 'timelineChanged', - update: 'prepareAndUpdate' - }, function (payload, ecModel, api) { - var timelineModel = ecModel.getComponent('timeline'); - - if (timelineModel && payload.currentIndex != null) { - timelineModel.setCurrentIndex(payload.currentIndex); - - if (!timelineModel.get('loop', true) && timelineModel.isIndexMax() && timelineModel.getPlayState()) { - timelineModel.setPlayState(false); // The timeline has played to the end, trigger event - - api.dispatchAction({ - type: 'timelinePlayChange', - playState: false, - from: payload.from - }); - } - } // Set normalized currentIndex to payload. - - - ecModel.resetOption('timeline', { - replaceMerge: timelineModel.get('replaceMerge', true) - }); - return defaults({ - currentIndex: timelineModel.option.currentIndex - }, payload); - }); - registers.registerAction({ - type: 'timelinePlayChange', - event: 'timelinePlayChanged', - update: 'update' - }, function (payload, ecModel) { - var timelineModel = ecModel.getComponent('timeline'); - - if (timelineModel && payload.playState != null) { - timelineModel.setPlayState(payload.playState); - } - }); - } - - function timelinePreprocessor(option) { - var timelineOpt = option && option.timeline; - - if (!isArray(timelineOpt)) { - timelineOpt = timelineOpt ? [timelineOpt] : []; - } - - each(timelineOpt, function (opt) { - if (!opt) { - return; - } - - compatibleEC2(opt); - }); - } - - function compatibleEC2(opt) { - var type = opt.type; - var ec2Types = { - 'number': 'value', - 'time': 'time' - }; // Compatible with ec2 - - if (ec2Types[type]) { - opt.axisType = ec2Types[type]; - delete opt.type; - } - - transferItem(opt); - - if (has(opt, 'controlPosition')) { - var controlStyle = opt.controlStyle || (opt.controlStyle = {}); - - if (!has(controlStyle, 'position')) { - controlStyle.position = opt.controlPosition; - } - - if (controlStyle.position === 'none' && !has(controlStyle, 'show')) { - controlStyle.show = false; - delete controlStyle.position; - } - - delete opt.controlPosition; - } - - each(opt.data || [], function (dataItem) { - if (isObject(dataItem) && !isArray(dataItem)) { - if (!has(dataItem, 'value') && has(dataItem, 'name')) { - // In ec2, using name as value. - dataItem.value = dataItem.name; - } - - transferItem(dataItem); - } - }); - } - - function transferItem(opt) { - var itemStyle = opt.itemStyle || (opt.itemStyle = {}); - var itemStyleEmphasis = itemStyle.emphasis || (itemStyle.emphasis = {}); // Transfer label out - - var label = opt.label || opt.label || {}; - var labelNormal = label.normal || (label.normal = {}); - var excludeLabelAttr = { - normal: 1, - emphasis: 1 - }; - each(label, function (value, name) { - if (!excludeLabelAttr[name] && !has(labelNormal, name)) { - labelNormal[name] = value; - } - }); - - if (itemStyleEmphasis.label && !has(label, 'emphasis')) { - label.emphasis = itemStyleEmphasis.label; - delete itemStyleEmphasis.label; - } - } - - function has(obj, attr) { - return obj.hasOwnProperty(attr); - } - - function install$D(registers) { - registers.registerComponentModel(SliderTimelineModel); - registers.registerComponentView(SliderTimelineView); - registers.registerSubTypeDefaulter('timeline', function () { - // Only slider now. - return 'slider'; - }); - installTimelineAction(registers); - registers.registerPreprocessor(timelinePreprocessor); - } - - function checkMarkerInSeries(seriesOpts, markerType) { - if (!seriesOpts) { - return false; - } - - var seriesOptArr = isArray(seriesOpts) ? seriesOpts : [seriesOpts]; - - for (var idx = 0; idx < seriesOptArr.length; idx++) { - if (seriesOptArr[idx] && seriesOptArr[idx][markerType]) { - return true; - } - } - - return false; - } - - function fillLabel(opt) { - defaultEmphasis(opt, 'label', ['show']); - } // { [componentType]: MarkerModel } - - - var inner$g = makeInner(); - - var MarkerModel = - /** @class */ - function (_super) { - __extends(MarkerModel, _super); - - function MarkerModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkerModel.type; - /** - * If marker model is created by self from series - */ - - _this.createdBySelf = false; - return _this; - } - /** - * @overrite - */ - - - MarkerModel.prototype.init = function (option, parentModel, ecModel) { - if ("development" !== 'production') { - if (this.type === 'marker') { - throw new Error('Marker component is abstract component. Use markLine, markPoint, markArea instead.'); - } - } - - this.mergeDefaultAndTheme(option, ecModel); - - this._mergeOption(option, ecModel, false, true); - }; - - MarkerModel.prototype.isAnimationEnabled = function () { - if (env.node) { - return false; - } - - var hostSeries = this.__hostSeries; - return this.getShallow('animation') && hostSeries && hostSeries.isAnimationEnabled(); - }; - /** - * @overrite - */ - - - MarkerModel.prototype.mergeOption = function (newOpt, ecModel) { - this._mergeOption(newOpt, ecModel, false, false); - }; - - MarkerModel.prototype._mergeOption = function (newOpt, ecModel, createdBySelf, isInit) { - var componentType = this.mainType; - - if (!createdBySelf) { - ecModel.eachSeries(function (seriesModel) { - // mainType can be markPoint, markLine, markArea - var markerOpt = seriesModel.get(this.mainType, true); - var markerModel = inner$g(seriesModel)[componentType]; - - if (!markerOpt || !markerOpt.data) { - inner$g(seriesModel)[componentType] = null; - return; - } - - if (!markerModel) { - if (isInit) { - // Default label emphasis `position` and `show` - fillLabel(markerOpt); - } - - each(markerOpt.data, function (item) { - // FIXME Overwrite fillLabel method ? - if (item instanceof Array) { - fillLabel(item[0]); - fillLabel(item[1]); - } else { - fillLabel(item); - } - }); - markerModel = this.createMarkerModelFromSeries(markerOpt, this, ecModel); // markerModel = new ImplementedMarkerModel( - // markerOpt, this, ecModel - // ); - - extend(markerModel, { - mainType: this.mainType, - // Use the same series index and name - seriesIndex: seriesModel.seriesIndex, - name: seriesModel.name, - createdBySelf: true - }); - markerModel.__hostSeries = seriesModel; - } else { - markerModel._mergeOption(markerOpt, ecModel, true); - } - - inner$g(seriesModel)[componentType] = markerModel; - }, this); - } - }; - - MarkerModel.prototype.formatTooltip = function (dataIndex, multipleSeries, dataType) { - var data = this.getData(); - var value = this.getRawValue(dataIndex); - var itemName = data.getName(dataIndex); - return createTooltipMarkup('section', { - header: this.name, - blocks: [createTooltipMarkup('nameValue', { - name: itemName, - value: value, - noName: !itemName, - noValue: value == null - })] - }); - }; - - MarkerModel.prototype.getData = function () { - return this._data; - }; - - MarkerModel.prototype.setData = function (data) { - this._data = data; - }; - - MarkerModel.getMarkerModelFromSeries = function (seriesModel, // Support three types of markers. Strict check. - componentType) { - return inner$g(seriesModel)[componentType]; - }; - - MarkerModel.type = 'marker'; - MarkerModel.dependencies = ['series', 'grid', 'polar', 'geo']; - return MarkerModel; - }(ComponentModel); - - mixin(MarkerModel, DataFormatMixin.prototype); - - var MarkPointModel = - /** @class */ - function (_super) { - __extends(MarkPointModel, _super); - - function MarkPointModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkPointModel.type; - return _this; - } - - MarkPointModel.prototype.createMarkerModelFromSeries = function (markerOpt, masterMarkerModel, ecModel) { - return new MarkPointModel(markerOpt, masterMarkerModel, ecModel); - }; - - MarkPointModel.type = 'markPoint'; - MarkPointModel.defaultOption = { - // zlevel: 0, - z: 5, - symbol: 'pin', - symbolSize: 50, - // symbolRotate: 0, - // symbolOffset: [0, 0] - tooltip: { - trigger: 'item' - }, - label: { - show: true, - position: 'inside' - }, - itemStyle: { - borderWidth: 2 - }, - emphasis: { - label: { - show: true - } - } - }; - return MarkPointModel; - }(MarkerModel); - - function hasXOrY(item) { - return !(isNaN(parseFloat(item.x)) && isNaN(parseFloat(item.y))); - } - - function hasXAndY(item) { - return !isNaN(parseFloat(item.x)) && !isNaN(parseFloat(item.y)); - } - - function markerTypeCalculatorWithExtent(markerType, data, otherDataDim, targetDataDim, otherCoordIndex, targetCoordIndex) { - var coordArr = []; - var stacked = isDimensionStacked(data, targetDataDim - /* , otherDataDim */ - ); - var calcDataDim = stacked ? data.getCalculationInfo('stackResultDimension') : targetDataDim; - var value = numCalculate(data, calcDataDim, markerType); - var dataIndex = data.indicesOfNearest(calcDataDim, value)[0]; - coordArr[otherCoordIndex] = data.get(otherDataDim, dataIndex); - coordArr[targetCoordIndex] = data.get(calcDataDim, dataIndex); - var coordArrValue = data.get(targetDataDim, dataIndex); // Make it simple, do not visit all stacked value to count precision. - - var precision = getPrecision(data.get(targetDataDim, dataIndex)); - precision = Math.min(precision, 20); - - if (precision >= 0) { - coordArr[targetCoordIndex] = +coordArr[targetCoordIndex].toFixed(precision); - } - - return [coordArr, coordArrValue]; - } // TODO Specified percent - - - var markerTypeCalculator = { - min: curry(markerTypeCalculatorWithExtent, 'min'), - max: curry(markerTypeCalculatorWithExtent, 'max'), - average: curry(markerTypeCalculatorWithExtent, 'average'), - median: curry(markerTypeCalculatorWithExtent, 'median') - }; - /** - * Transform markPoint data item to format used in List by do the following - * 1. Calculate statistic like `max`, `min`, `average` - * 2. Convert `item.xAxis`, `item.yAxis` to `item.coord` array - */ - - function dataTransform(seriesModel, item) { - if (!item) { - return; - } - - var data = seriesModel.getData(); - var coordSys = seriesModel.coordinateSystem; - var dims = coordSys && coordSys.dimensions; // 1. If not specify the position with pixel directly - // 2. If `coord` is not a data array. Which uses `xAxis`, - // `yAxis` to specify the coord on each dimension - // parseFloat first because item.x and item.y can be percent string like '20%' - - if (!hasXAndY(item) && !isArray(item.coord) && isArray(dims)) { - var axisInfo = getAxisInfo$1(item, data, coordSys, seriesModel); // Clone the option - // Transform the properties xAxis, yAxis, radiusAxis, angleAxis, geoCoord to value - - item = clone(item); - - if (item.type && markerTypeCalculator[item.type] && axisInfo.baseAxis && axisInfo.valueAxis) { - var otherCoordIndex = indexOf(dims, axisInfo.baseAxis.dim); - var targetCoordIndex = indexOf(dims, axisInfo.valueAxis.dim); - var coordInfo = markerTypeCalculator[item.type](data, axisInfo.baseDataDim, axisInfo.valueDataDim, otherCoordIndex, targetCoordIndex); - item.coord = coordInfo[0]; // Force to use the value of calculated value. - // let item use the value without stack. - - item.value = coordInfo[1]; - } else { - // FIXME Only has one of xAxis and yAxis. - item.coord = [item.xAxis != null ? item.xAxis : item.radiusAxis, item.yAxis != null ? item.yAxis : item.angleAxis]; - } - } // x y is provided - - - if (item.coord == null || !isArray(dims)) { - item.coord = []; - } else { - // Each coord support max, min, average - var coord = item.coord; - - for (var i = 0; i < 2; i++) { - if (markerTypeCalculator[coord[i]]) { - coord[i] = numCalculate(data, data.mapDimension(dims[i]), coord[i]); - } - } - } - - return item; - } - function getAxisInfo$1(item, data, coordSys, seriesModel) { - var ret = {}; - - if (item.valueIndex != null || item.valueDim != null) { - ret.valueDataDim = item.valueIndex != null ? data.getDimension(item.valueIndex) : item.valueDim; - ret.valueAxis = coordSys.getAxis(dataDimToCoordDim(seriesModel, ret.valueDataDim)); - ret.baseAxis = coordSys.getOtherAxis(ret.valueAxis); - ret.baseDataDim = data.mapDimension(ret.baseAxis.dim); - } else { - ret.baseAxis = seriesModel.getBaseAxis(); - ret.valueAxis = coordSys.getOtherAxis(ret.baseAxis); - ret.baseDataDim = data.mapDimension(ret.baseAxis.dim); - ret.valueDataDim = data.mapDimension(ret.valueAxis.dim); - } - - return ret; - } - - function dataDimToCoordDim(seriesModel, dataDim) { - var dimItem = seriesModel.getData().getDimensionInfo(dataDim); - return dimItem && dimItem.coordDim; - } - /** - * Filter data which is out of coordinateSystem range - * [dataFilter description] - */ - - - function dataFilter$1( // Currently only polar and cartesian has containData. - coordSys, item) { - // Always return true if there is no coordSys - return coordSys && coordSys.containData && item.coord && !hasXOrY(item) ? coordSys.containData(item.coord) : true; - } - function zoneFilter( // Currently only polar and cartesian has containData. - coordSys, item1, item2) { - // Always return true if there is no coordSys - return coordSys && coordSys.containZone && item1.coord && item2.coord && !hasXOrY(item1) && !hasXOrY(item2) ? coordSys.containZone(item1.coord, item2.coord) : true; - } - function createMarkerDimValueGetter(inCoordSys, dims) { - return inCoordSys ? function (item, dimName, dataIndex, dimIndex) { - var rawVal = dimIndex < 2 // x, y, radius, angle - ? item.coord && item.coord[dimIndex] : item.value; - return parseDataValue(rawVal, dims[dimIndex]); - } : function (item, dimName, dataIndex, dimIndex) { - return parseDataValue(item.value, dims[dimIndex]); - }; - } - function numCalculate(data, valueDataDim, type) { - if (type === 'average') { - var sum_1 = 0; - var count_1 = 0; - data.each(valueDataDim, function (val, idx) { - if (!isNaN(val)) { - sum_1 += val; - count_1++; - } - }); - return sum_1 / count_1; - } else if (type === 'median') { - return data.getMedian(valueDataDim); - } else { - // max & min - return data.getDataExtent(valueDataDim)[type === 'max' ? 1 : 0]; - } - } - - var inner$h = makeInner(); - - var MarkerView = - /** @class */ - function (_super) { - __extends(MarkerView, _super); - - function MarkerView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkerView.type; - return _this; - } - - MarkerView.prototype.init = function () { - this.markerGroupMap = createHashMap(); - }; - - MarkerView.prototype.render = function (markerModel, ecModel, api) { - var _this = this; - - var markerGroupMap = this.markerGroupMap; - markerGroupMap.each(function (item) { - inner$h(item).keep = false; - }); - ecModel.eachSeries(function (seriesModel) { - var markerModel = MarkerModel.getMarkerModelFromSeries(seriesModel, _this.type); - markerModel && _this.renderSeries(seriesModel, markerModel, ecModel, api); - }); - markerGroupMap.each(function (item) { - !inner$h(item).keep && _this.group.remove(item.group); - }); - }; - - MarkerView.prototype.markKeep = function (drawGroup) { - inner$h(drawGroup).keep = true; - }; - - MarkerView.prototype.toggleBlurSeries = function (seriesModelList, isBlur) { - var _this = this; - - each(seriesModelList, function (seriesModel) { - var markerModel = MarkerModel.getMarkerModelFromSeries(seriesModel, _this.type); - - if (markerModel) { - var data = markerModel.getData(); - data.eachItemGraphicEl(function (el) { - if (el) { - isBlur ? enterBlur(el) : leaveBlur(el); - } - }); - } - }); - }; - - MarkerView.type = 'marker'; - return MarkerView; - }(ComponentView); - - function updateMarkerLayout(mpData, seriesModel, api) { - var coordSys = seriesModel.coordinateSystem; - mpData.each(function (idx) { - var itemModel = mpData.getItemModel(idx); - var point; - var xPx = parsePercent$1(itemModel.get('x'), api.getWidth()); - var yPx = parsePercent$1(itemModel.get('y'), api.getHeight()); - - if (!isNaN(xPx) && !isNaN(yPx)) { - point = [xPx, yPx]; - } // Chart like bar may have there own marker positioning logic - else if (seriesModel.getMarkerPosition) { - // Use the getMarkerPosition - point = seriesModel.getMarkerPosition(mpData.getValues(mpData.dimensions, idx)); - } else if (coordSys) { - var x = mpData.get(coordSys.dimensions[0], idx); - var y = mpData.get(coordSys.dimensions[1], idx); - point = coordSys.dataToPoint([x, y]); - } // Use x, y if has any - - - if (!isNaN(xPx)) { - point[0] = xPx; - } - - if (!isNaN(yPx)) { - point[1] = yPx; - } - - mpData.setItemLayout(idx, point); - }); - } - - var MarkPointView = - /** @class */ - function (_super) { - __extends(MarkPointView, _super); - - function MarkPointView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkPointView.type; - return _this; - } - - MarkPointView.prototype.updateTransform = function (markPointModel, ecModel, api) { - ecModel.eachSeries(function (seriesModel) { - var mpModel = MarkerModel.getMarkerModelFromSeries(seriesModel, 'markPoint'); - - if (mpModel) { - updateMarkerLayout(mpModel.getData(), seriesModel, api); - this.markerGroupMap.get(seriesModel.id).updateLayout(); - } - }, this); - }; - - MarkPointView.prototype.renderSeries = function (seriesModel, mpModel, ecModel, api) { - var coordSys = seriesModel.coordinateSystem; - var seriesId = seriesModel.id; - var seriesData = seriesModel.getData(); - var symbolDrawMap = this.markerGroupMap; - var symbolDraw = symbolDrawMap.get(seriesId) || symbolDrawMap.set(seriesId, new SymbolDraw()); - var mpData = createData(coordSys, seriesModel, mpModel); // FIXME - - mpModel.setData(mpData); - updateMarkerLayout(mpModel.getData(), seriesModel, api); - mpData.each(function (idx) { - var itemModel = mpData.getItemModel(idx); - var symbol = itemModel.getShallow('symbol'); - var symbolSize = itemModel.getShallow('symbolSize'); - var symbolRotate = itemModel.getShallow('symbolRotate'); - var symbolOffset = itemModel.getShallow('symbolOffset'); - var symbolKeepAspect = itemModel.getShallow('symbolKeepAspect'); // TODO: refactor needed: single data item should not support callback function - - if (isFunction(symbol) || isFunction(symbolSize) || isFunction(symbolRotate) || isFunction(symbolOffset)) { - var rawIdx = mpModel.getRawValue(idx); - var dataParams = mpModel.getDataParams(idx); - - if (isFunction(symbol)) { - symbol = symbol(rawIdx, dataParams); - } - - if (isFunction(symbolSize)) { - // FIXME 这里不兼容 ECharts 2.x,2.x 貌似参数是整个数据? - symbolSize = symbolSize(rawIdx, dataParams); - } - - if (isFunction(symbolRotate)) { - symbolRotate = symbolRotate(rawIdx, dataParams); - } - - if (isFunction(symbolOffset)) { - symbolOffset = symbolOffset(rawIdx, dataParams); - } - } - - var style = itemModel.getModel('itemStyle').getItemStyle(); - var color = getVisualFromData(seriesData, 'color'); - - if (!style.fill) { - style.fill = color; - } - - mpData.setItemVisual(idx, { - symbol: symbol, - symbolSize: symbolSize, - symbolRotate: symbolRotate, - symbolOffset: symbolOffset, - symbolKeepAspect: symbolKeepAspect, - style: style - }); - }); // TODO Text are wrong - - symbolDraw.updateData(mpData); - this.group.add(symbolDraw.group); // Set host model for tooltip - // FIXME - - mpData.eachItemGraphicEl(function (el) { - el.traverse(function (child) { - getECData(child).dataModel = mpModel; - }); - }); - this.markKeep(symbolDraw); - symbolDraw.group.silent = mpModel.get('silent') || seriesModel.get('silent'); - }; - - MarkPointView.type = 'markPoint'; - return MarkPointView; - }(MarkerView); - - function createData(coordSys, seriesModel, mpModel) { - var coordDimsInfos; - - if (coordSys) { - coordDimsInfos = map(coordSys && coordSys.dimensions, function (coordDim) { - var info = seriesModel.getData().getDimensionInfo(seriesModel.getData().mapDimension(coordDim)) || {}; // In map series data don't have lng and lat dimension. Fallback to same with coordSys - - return extend(extend({}, info), { - name: coordDim, - // DON'T use ordinalMeta to parse and collect ordinal. - ordinalMeta: null - }); - }); - } else { - coordDimsInfos = [{ - name: 'value', - type: 'float' - }]; - } - - var mpData = new SeriesData(coordDimsInfos, mpModel); - var dataOpt = map(mpModel.get('data'), curry(dataTransform, seriesModel)); - - if (coordSys) { - dataOpt = filter(dataOpt, curry(dataFilter$1, coordSys)); - } - - var dimValueGetter = createMarkerDimValueGetter(!!coordSys, coordDimsInfos); - mpData.initData(dataOpt, null, dimValueGetter); - return mpData; - } - - function install$E(registers) { - registers.registerComponentModel(MarkPointModel); - registers.registerComponentView(MarkPointView); - registers.registerPreprocessor(function (opt) { - if (checkMarkerInSeries(opt.series, 'markPoint')) { - // Make sure markPoint component is enabled - opt.markPoint = opt.markPoint || {}; - } - }); - } - - var MarkLineModel = - /** @class */ - function (_super) { - __extends(MarkLineModel, _super); - - function MarkLineModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkLineModel.type; - return _this; - } - - MarkLineModel.prototype.createMarkerModelFromSeries = function (markerOpt, masterMarkerModel, ecModel) { - return new MarkLineModel(markerOpt, masterMarkerModel, ecModel); - }; - - MarkLineModel.type = 'markLine'; - MarkLineModel.defaultOption = { - // zlevel: 0, - z: 5, - symbol: ['circle', 'arrow'], - symbolSize: [8, 16], - // symbolRotate: 0, - symbolOffset: 0, - precision: 2, - tooltip: { - trigger: 'item' - }, - label: { - show: true, - position: 'end', - distance: 5 - }, - lineStyle: { - type: 'dashed' - }, - emphasis: { - label: { - show: true - }, - lineStyle: { - width: 3 - } - }, - animationEasing: 'linear' - }; - return MarkLineModel; - }(MarkerModel); - - var inner$i = makeInner(); - - var markLineTransform = function (seriesModel, coordSys, mlModel, item) { - var data = seriesModel.getData(); - var itemArray; - - if (!isArray(item)) { - // Special type markLine like 'min', 'max', 'average', 'median' - var mlType = item.type; - - if (mlType === 'min' || mlType === 'max' || mlType === 'average' || mlType === 'median' // In case - // data: [{ - // yAxis: 10 - // }] - || item.xAxis != null || item.yAxis != null) { - var valueAxis = void 0; - var value = void 0; - - if (item.yAxis != null || item.xAxis != null) { - valueAxis = coordSys.getAxis(item.yAxis != null ? 'y' : 'x'); - value = retrieve(item.yAxis, item.xAxis); - } else { - var axisInfo = getAxisInfo$1(item, data, coordSys, seriesModel); - valueAxis = axisInfo.valueAxis; - var valueDataDim = getStackedDimension(data, axisInfo.valueDataDim); - value = numCalculate(data, valueDataDim, mlType); - } - - var valueIndex = valueAxis.dim === 'x' ? 0 : 1; - var baseIndex = 1 - valueIndex; // Normized to 2d data with start and end point - - var mlFrom = clone(item); - var mlTo = { - coord: [] - }; - mlFrom.type = null; - mlFrom.coord = []; - mlFrom.coord[baseIndex] = -Infinity; - mlTo.coord[baseIndex] = Infinity; - var precision = mlModel.get('precision'); - - if (precision >= 0 && isNumber(value)) { - value = +value.toFixed(Math.min(precision, 20)); - } - - mlFrom.coord[valueIndex] = mlTo.coord[valueIndex] = value; - itemArray = [mlFrom, mlTo, { - type: mlType, - valueIndex: item.valueIndex, - // Force to use the value of calculated value. - value: value - }]; - } else { - // Invalid data - if ("development" !== 'production') { - logError('Invalid markLine data.'); - } - - itemArray = []; - } - } else { - itemArray = item; - } - - var normalizedItem = [dataTransform(seriesModel, itemArray[0]), dataTransform(seriesModel, itemArray[1]), extend({}, itemArray[2])]; // Avoid line data type is extended by from(to) data type - - normalizedItem[2].type = normalizedItem[2].type || null; // Merge from option and to option into line option - - merge(normalizedItem[2], normalizedItem[0]); - merge(normalizedItem[2], normalizedItem[1]); - return normalizedItem; - }; - - function isInfinity(val) { - return !isNaN(val) && !isFinite(val); - } // If a markLine has one dim - - - function ifMarkLineHasOnlyDim(dimIndex, fromCoord, toCoord, coordSys) { - var otherDimIndex = 1 - dimIndex; - var dimName = coordSys.dimensions[dimIndex]; - return isInfinity(fromCoord[otherDimIndex]) && isInfinity(toCoord[otherDimIndex]) && fromCoord[dimIndex] === toCoord[dimIndex] && coordSys.getAxis(dimName).containData(fromCoord[dimIndex]); - } - - function markLineFilter(coordSys, item) { - if (coordSys.type === 'cartesian2d') { - var fromCoord = item[0].coord; - var toCoord = item[1].coord; // In case - // { - // markLine: { - // data: [{ yAxis: 2 }] - // } - // } - - if (fromCoord && toCoord && (ifMarkLineHasOnlyDim(1, fromCoord, toCoord, coordSys) || ifMarkLineHasOnlyDim(0, fromCoord, toCoord, coordSys))) { - return true; - } - } - - return dataFilter$1(coordSys, item[0]) && dataFilter$1(coordSys, item[1]); - } - - function updateSingleMarkerEndLayout(data, idx, isFrom, seriesModel, api) { - var coordSys = seriesModel.coordinateSystem; - var itemModel = data.getItemModel(idx); - var point; - var xPx = parsePercent$1(itemModel.get('x'), api.getWidth()); - var yPx = parsePercent$1(itemModel.get('y'), api.getHeight()); - - if (!isNaN(xPx) && !isNaN(yPx)) { - point = [xPx, yPx]; - } else { - // Chart like bar may have there own marker positioning logic - if (seriesModel.getMarkerPosition) { - // Use the getMarkerPosition - point = seriesModel.getMarkerPosition(data.getValues(data.dimensions, idx)); - } else { - var dims = coordSys.dimensions; - var x = data.get(dims[0], idx); - var y = data.get(dims[1], idx); - point = coordSys.dataToPoint([x, y]); - } // Expand line to the edge of grid if value on one axis is Inifnity - // In case - // markLine: { - // data: [{ - // yAxis: 2 - // // or - // type: 'average' - // }] - // } - - - if (isCoordinateSystemType(coordSys, 'cartesian2d')) { - // TODO: TYPE ts@4.1 may still infer it as Axis instead of Axis2D. Not sure if it's a bug - var xAxis = coordSys.getAxis('x'); - var yAxis = coordSys.getAxis('y'); - var dims = coordSys.dimensions; - - if (isInfinity(data.get(dims[0], idx))) { - point[0] = xAxis.toGlobalCoord(xAxis.getExtent()[isFrom ? 0 : 1]); - } else if (isInfinity(data.get(dims[1], idx))) { - point[1] = yAxis.toGlobalCoord(yAxis.getExtent()[isFrom ? 0 : 1]); - } - } // Use x, y if has any - - - if (!isNaN(xPx)) { - point[0] = xPx; - } - - if (!isNaN(yPx)) { - point[1] = yPx; - } - } - - data.setItemLayout(idx, point); - } - - var MarkLineView = - /** @class */ - function (_super) { - __extends(MarkLineView, _super); - - function MarkLineView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkLineView.type; - return _this; - } - - MarkLineView.prototype.updateTransform = function (markLineModel, ecModel, api) { - ecModel.eachSeries(function (seriesModel) { - var mlModel = MarkerModel.getMarkerModelFromSeries(seriesModel, 'markLine'); - - if (mlModel) { - var mlData_1 = mlModel.getData(); - var fromData_1 = inner$i(mlModel).from; - var toData_1 = inner$i(mlModel).to; // Update visual and layout of from symbol and to symbol - - fromData_1.each(function (idx) { - updateSingleMarkerEndLayout(fromData_1, idx, true, seriesModel, api); - updateSingleMarkerEndLayout(toData_1, idx, false, seriesModel, api); - }); // Update layout of line - - mlData_1.each(function (idx) { - mlData_1.setItemLayout(idx, [fromData_1.getItemLayout(idx), toData_1.getItemLayout(idx)]); - }); - this.markerGroupMap.get(seriesModel.id).updateLayout(); - } - }, this); - }; - - MarkLineView.prototype.renderSeries = function (seriesModel, mlModel, ecModel, api) { - var coordSys = seriesModel.coordinateSystem; - var seriesId = seriesModel.id; - var seriesData = seriesModel.getData(); - var lineDrawMap = this.markerGroupMap; - var lineDraw = lineDrawMap.get(seriesId) || lineDrawMap.set(seriesId, new LineDraw()); - this.group.add(lineDraw.group); - var mlData = createList$1(coordSys, seriesModel, mlModel); - var fromData = mlData.from; - var toData = mlData.to; - var lineData = mlData.line; - inner$i(mlModel).from = fromData; - inner$i(mlModel).to = toData; // Line data for tooltip and formatter - - mlModel.setData(lineData); // TODO - // Functionally, `symbolSize` & `symbolOffset` can also be 2D array now. - // But the related logic and type definition are not finished yet. - // Finish it if required - - var symbolType = mlModel.get('symbol'); - var symbolSize = mlModel.get('symbolSize'); - var symbolRotate = mlModel.get('symbolRotate'); - var symbolOffset = mlModel.get('symbolOffset'); // TODO: support callback function like markPoint - - if (!isArray(symbolType)) { - symbolType = [symbolType, symbolType]; - } - - if (!isArray(symbolSize)) { - symbolSize = [symbolSize, symbolSize]; - } - - if (!isArray(symbolRotate)) { - symbolRotate = [symbolRotate, symbolRotate]; - } - - if (!isArray(symbolOffset)) { - symbolOffset = [symbolOffset, symbolOffset]; - } // Update visual and layout of from symbol and to symbol - - - mlData.from.each(function (idx) { - updateDataVisualAndLayout(fromData, idx, true); - updateDataVisualAndLayout(toData, idx, false); - }); // Update visual and layout of line - - lineData.each(function (idx) { - var lineStyle = lineData.getItemModel(idx).getModel('lineStyle').getLineStyle(); // lineData.setItemVisual(idx, { - // color: lineColor || fromData.getItemVisual(idx, 'color') - // }); - - lineData.setItemLayout(idx, [fromData.getItemLayout(idx), toData.getItemLayout(idx)]); - - if (lineStyle.stroke == null) { - lineStyle.stroke = fromData.getItemVisual(idx, 'style').fill; - } - - lineData.setItemVisual(idx, { - fromSymbolKeepAspect: fromData.getItemVisual(idx, 'symbolKeepAspect'), - fromSymbolOffset: fromData.getItemVisual(idx, 'symbolOffset'), - fromSymbolRotate: fromData.getItemVisual(idx, 'symbolRotate'), - fromSymbolSize: fromData.getItemVisual(idx, 'symbolSize'), - fromSymbol: fromData.getItemVisual(idx, 'symbol'), - toSymbolKeepAspect: toData.getItemVisual(idx, 'symbolKeepAspect'), - toSymbolOffset: toData.getItemVisual(idx, 'symbolOffset'), - toSymbolRotate: toData.getItemVisual(idx, 'symbolRotate'), - toSymbolSize: toData.getItemVisual(idx, 'symbolSize'), - toSymbol: toData.getItemVisual(idx, 'symbol'), - style: lineStyle - }); - }); - lineDraw.updateData(lineData); // Set host model for tooltip - // FIXME - - mlData.line.eachItemGraphicEl(function (el) { - getECData(el).dataModel = mlModel; - el.traverse(function (child) { - getECData(child).dataModel = mlModel; - }); - }); - - function updateDataVisualAndLayout(data, idx, isFrom) { - var itemModel = data.getItemModel(idx); - updateSingleMarkerEndLayout(data, idx, isFrom, seriesModel, api); - var style = itemModel.getModel('itemStyle').getItemStyle(); - - if (style.fill == null) { - style.fill = getVisualFromData(seriesData, 'color'); - } - - data.setItemVisual(idx, { - symbolKeepAspect: itemModel.get('symbolKeepAspect'), - // `0` should be considered as a valid value, so use `retrieve2` instead of `||` - symbolOffset: retrieve2(itemModel.get('symbolOffset', true), symbolOffset[isFrom ? 0 : 1]), - symbolRotate: retrieve2(itemModel.get('symbolRotate', true), symbolRotate[isFrom ? 0 : 1]), - // TODO: when 2d array is supported, it should ignore parent - symbolSize: retrieve2(itemModel.get('symbolSize'), symbolSize[isFrom ? 0 : 1]), - symbol: retrieve2(itemModel.get('symbol', true), symbolType[isFrom ? 0 : 1]), - style: style - }); - } - - this.markKeep(lineDraw); - lineDraw.group.silent = mlModel.get('silent') || seriesModel.get('silent'); - }; - - MarkLineView.type = 'markLine'; - return MarkLineView; - }(MarkerView); - - function createList$1(coordSys, seriesModel, mlModel) { - var coordDimsInfos; - - if (coordSys) { - coordDimsInfos = map(coordSys && coordSys.dimensions, function (coordDim) { - var info = seriesModel.getData().getDimensionInfo(seriesModel.getData().mapDimension(coordDim)) || {}; // In map series data don't have lng and lat dimension. Fallback to same with coordSys - - return extend(extend({}, info), { - name: coordDim, - // DON'T use ordinalMeta to parse and collect ordinal. - ordinalMeta: null - }); - }); - } else { - coordDimsInfos = [{ - name: 'value', - type: 'float' - }]; - } - - var fromData = new SeriesData(coordDimsInfos, mlModel); - var toData = new SeriesData(coordDimsInfos, mlModel); // No dimensions - - var lineData = new SeriesData([], mlModel); - var optData = map(mlModel.get('data'), curry(markLineTransform, seriesModel, coordSys, mlModel)); - - if (coordSys) { - optData = filter(optData, curry(markLineFilter, coordSys)); - } - - var dimValueGetter = createMarkerDimValueGetter(!!coordSys, coordDimsInfos); - fromData.initData(map(optData, function (item) { - return item[0]; - }), null, dimValueGetter); - toData.initData(map(optData, function (item) { - return item[1]; - }), null, dimValueGetter); - lineData.initData(map(optData, function (item) { - return item[2]; - })); - lineData.hasItemOption = true; - return { - from: fromData, - to: toData, - line: lineData - }; - } - - function install$F(registers) { - registers.registerComponentModel(MarkLineModel); - registers.registerComponentView(MarkLineView); - registers.registerPreprocessor(function (opt) { - if (checkMarkerInSeries(opt.series, 'markLine')) { - // Make sure markLine component is enabled - opt.markLine = opt.markLine || {}; - } - }); - } - - var MarkAreaModel = - /** @class */ - function (_super) { - __extends(MarkAreaModel, _super); - - function MarkAreaModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkAreaModel.type; - return _this; - } - - MarkAreaModel.prototype.createMarkerModelFromSeries = function (markerOpt, masterMarkerModel, ecModel) { - return new MarkAreaModel(markerOpt, masterMarkerModel, ecModel); - }; - - MarkAreaModel.type = 'markArea'; - MarkAreaModel.defaultOption = { - // zlevel: 0, - // PENDING - z: 1, - tooltip: { - trigger: 'item' - }, - // markArea should fixed on the coordinate system - animation: false, - label: { - show: true, - position: 'top' - }, - itemStyle: { - // color and borderColor default to use color from series - // color: 'auto' - // borderColor: 'auto' - borderWidth: 0 - }, - emphasis: { - label: { - show: true, - position: 'top' - } - } - }; - return MarkAreaModel; - }(MarkerModel); - - var inner$j = makeInner(); - - var markAreaTransform = function (seriesModel, coordSys, maModel, item) { - // item may be null - var item0 = item[0]; - var item1 = item[1]; - - if (!item0 || !item1) { - return; - } - - var lt = dataTransform(seriesModel, item0); - var rb = dataTransform(seriesModel, item1); // FIXME make sure lt is less than rb - - var ltCoord = lt.coord; - var rbCoord = rb.coord; - ltCoord[0] = retrieve(ltCoord[0], -Infinity); - ltCoord[1] = retrieve(ltCoord[1], -Infinity); - rbCoord[0] = retrieve(rbCoord[0], Infinity); - rbCoord[1] = retrieve(rbCoord[1], Infinity); // Merge option into one - - var result = mergeAll([{}, lt, rb]); - result.coord = [lt.coord, rb.coord]; - result.x0 = lt.x; - result.y0 = lt.y; - result.x1 = rb.x; - result.y1 = rb.y; - return result; - }; - - function isInfinity$1(val) { - return !isNaN(val) && !isFinite(val); - } // If a markArea has one dim - - - function ifMarkAreaHasOnlyDim(dimIndex, fromCoord, toCoord, coordSys) { - var otherDimIndex = 1 - dimIndex; - return isInfinity$1(fromCoord[otherDimIndex]) && isInfinity$1(toCoord[otherDimIndex]); - } - - function markAreaFilter(coordSys, item) { - var fromCoord = item.coord[0]; - var toCoord = item.coord[1]; - var item0 = { - coord: fromCoord, - x: item.x0, - y: item.y0 - }; - var item1 = { - coord: toCoord, - x: item.x1, - y: item.y1 - }; - - if (isCoordinateSystemType(coordSys, 'cartesian2d')) { - // In case - // { - // markArea: { - // data: [{ yAxis: 2 }] - // } - // } - if (fromCoord && toCoord && (ifMarkAreaHasOnlyDim(1, fromCoord, toCoord) || ifMarkAreaHasOnlyDim(0, fromCoord, toCoord))) { - return true; - } // Directly returning true may also do the work, - // because markArea will not be shown automatically - // when it's not included in coordinate system. - // But filtering ahead can avoid keeping rendering markArea - // when there are too many of them. - - - return zoneFilter(coordSys, item0, item1); - } - - return dataFilter$1(coordSys, item0) || dataFilter$1(coordSys, item1); - } // dims can be ['x0', 'y0'], ['x1', 'y1'], ['x0', 'y1'], ['x1', 'y0'] - - - function getSingleMarkerEndPoint(data, idx, dims, seriesModel, api) { - var coordSys = seriesModel.coordinateSystem; - var itemModel = data.getItemModel(idx); - var point; - var xPx = parsePercent$1(itemModel.get(dims[0]), api.getWidth()); - var yPx = parsePercent$1(itemModel.get(dims[1]), api.getHeight()); - - if (!isNaN(xPx) && !isNaN(yPx)) { - point = [xPx, yPx]; - } else { - // Chart like bar may have there own marker positioning logic - if (seriesModel.getMarkerPosition) { - // Consider the case that user input the right-bottom point first - // Pick the larger x and y as 'x1' and 'y1' - var pointValue0 = data.getValues(['x0', 'y0'], idx); - var pointValue1 = data.getValues(['x1', 'y1'], idx); - var clampPointValue0 = coordSys.clampData(pointValue0); - var clampPointValue1 = coordSys.clampData(pointValue1); - var pointValue = []; - - if (dims[0] === 'x0') { - pointValue[0] = clampPointValue0[0] > clampPointValue1[0] ? pointValue1[0] : pointValue0[0]; - } else { - pointValue[0] = clampPointValue0[0] > clampPointValue1[0] ? pointValue0[0] : pointValue1[0]; - } - - if (dims[1] === 'y0') { - pointValue[1] = clampPointValue0[1] > clampPointValue1[1] ? pointValue1[1] : pointValue0[1]; - } else { - pointValue[1] = clampPointValue0[1] > clampPointValue1[1] ? pointValue0[1] : pointValue1[1]; - } // Use the getMarkerPosition - - - point = seriesModel.getMarkerPosition(pointValue, dims, true); - } else { - var x = data.get(dims[0], idx); - var y = data.get(dims[1], idx); - var pt = [x, y]; - coordSys.clampData && coordSys.clampData(pt, pt); - point = coordSys.dataToPoint(pt, true); - } - - if (isCoordinateSystemType(coordSys, 'cartesian2d')) { - // TODO: TYPE ts@4.1 may still infer it as Axis instead of Axis2D. Not sure if it's a bug - var xAxis = coordSys.getAxis('x'); - var yAxis = coordSys.getAxis('y'); - var x = data.get(dims[0], idx); - var y = data.get(dims[1], idx); - - if (isInfinity$1(x)) { - point[0] = xAxis.toGlobalCoord(xAxis.getExtent()[dims[0] === 'x0' ? 0 : 1]); - } else if (isInfinity$1(y)) { - point[1] = yAxis.toGlobalCoord(yAxis.getExtent()[dims[1] === 'y0' ? 0 : 1]); - } - } // Use x, y if has any - - - if (!isNaN(xPx)) { - point[0] = xPx; - } - - if (!isNaN(yPx)) { - point[1] = yPx; - } - } - - return point; - } - - var dimPermutations = [['x0', 'y0'], ['x1', 'y0'], ['x1', 'y1'], ['x0', 'y1']]; - - var MarkAreaView = - /** @class */ - function (_super) { - __extends(MarkAreaView, _super); - - function MarkAreaView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = MarkAreaView.type; - return _this; - } - - MarkAreaView.prototype.updateTransform = function (markAreaModel, ecModel, api) { - ecModel.eachSeries(function (seriesModel) { - var maModel = MarkerModel.getMarkerModelFromSeries(seriesModel, 'markArea'); - - if (maModel) { - var areaData_1 = maModel.getData(); - areaData_1.each(function (idx) { - var points = map(dimPermutations, function (dim) { - return getSingleMarkerEndPoint(areaData_1, idx, dim, seriesModel, api); - }); // Layout - - areaData_1.setItemLayout(idx, points); - var el = areaData_1.getItemGraphicEl(idx); - el.setShape('points', points); - }); - } - }, this); - }; - - MarkAreaView.prototype.renderSeries = function (seriesModel, maModel, ecModel, api) { - var coordSys = seriesModel.coordinateSystem; - var seriesId = seriesModel.id; - var seriesData = seriesModel.getData(); - var areaGroupMap = this.markerGroupMap; - var polygonGroup = areaGroupMap.get(seriesId) || areaGroupMap.set(seriesId, { - group: new Group() - }); - this.group.add(polygonGroup.group); - this.markKeep(polygonGroup); - var areaData = createList$2(coordSys, seriesModel, maModel); // Line data for tooltip and formatter - - maModel.setData(areaData); // Update visual and layout of line - - areaData.each(function (idx) { - // Layout - var points = map(dimPermutations, function (dim) { - return getSingleMarkerEndPoint(areaData, idx, dim, seriesModel, api); - }); - var xAxisScale = coordSys.getAxis('x').scale; - var yAxisScale = coordSys.getAxis('y').scale; - var xAxisExtent = xAxisScale.getExtent(); - var yAxisExtent = yAxisScale.getExtent(); - var xPointExtent = [xAxisScale.parse(areaData.get('x0', idx)), xAxisScale.parse(areaData.get('x1', idx))]; - var yPointExtent = [yAxisScale.parse(areaData.get('y0', idx)), yAxisScale.parse(areaData.get('y1', idx))]; - asc(xPointExtent); - asc(yPointExtent); - var overlapped = !(xAxisExtent[0] > xPointExtent[1] || xAxisExtent[1] < xPointExtent[0] || yAxisExtent[0] > yPointExtent[1] || yAxisExtent[1] < yPointExtent[0]); // If none of the area is inside coordSys, allClipped is set to be true - // in layout so that label will not be displayed. See #12591 - - var allClipped = !overlapped; - areaData.setItemLayout(idx, { - points: points, - allClipped: allClipped - }); - var style = areaData.getItemModel(idx).getModel('itemStyle').getItemStyle(); - var color$1 = getVisualFromData(seriesData, 'color'); - - if (!style.fill) { - style.fill = color$1; - - if (isString(style.fill)) { - style.fill = modifyAlpha(style.fill, 0.4); - } - } - - if (!style.stroke) { - style.stroke = color$1; - } // Visual - - - areaData.setItemVisual(idx, 'style', style); - }); - areaData.diff(inner$j(polygonGroup).data).add(function (idx) { - var layout = areaData.getItemLayout(idx); - - if (!layout.allClipped) { - var polygon = new Polygon({ - shape: { - points: layout.points - } - }); - areaData.setItemGraphicEl(idx, polygon); - polygonGroup.group.add(polygon); - } - }).update(function (newIdx, oldIdx) { - var polygon = inner$j(polygonGroup).data.getItemGraphicEl(oldIdx); - var layout = areaData.getItemLayout(newIdx); - - if (!layout.allClipped) { - if (polygon) { - updateProps(polygon, { - shape: { - points: layout.points - } - }, maModel, newIdx); - } else { - polygon = new Polygon({ - shape: { - points: layout.points - } - }); - } - - areaData.setItemGraphicEl(newIdx, polygon); - polygonGroup.group.add(polygon); - } else if (polygon) { - polygonGroup.group.remove(polygon); - } - }).remove(function (idx) { - var polygon = inner$j(polygonGroup).data.getItemGraphicEl(idx); - polygonGroup.group.remove(polygon); - }).execute(); - areaData.eachItemGraphicEl(function (polygon, idx) { - var itemModel = areaData.getItemModel(idx); - var style = areaData.getItemVisual(idx, 'style'); - polygon.useStyle(areaData.getItemVisual(idx, 'style')); - setLabelStyle(polygon, getLabelStatesModels(itemModel), { - labelFetcher: maModel, - labelDataIndex: idx, - defaultText: areaData.getName(idx) || '', - inheritColor: isString(style.fill) ? modifyAlpha(style.fill, 1) : '#000' - }); - setStatesStylesFromModel(polygon, itemModel); - toggleHoverEmphasis(polygon, null, null, itemModel.get(['emphasis', 'disabled'])); - getECData(polygon).dataModel = maModel; - }); - inner$j(polygonGroup).data = areaData; - polygonGroup.group.silent = maModel.get('silent') || seriesModel.get('silent'); - }; - - MarkAreaView.type = 'markArea'; - return MarkAreaView; - }(MarkerView); - - function createList$2(coordSys, seriesModel, maModel) { - var areaData; - var dataDims; - var dims = ['x0', 'y0', 'x1', 'y1']; - - if (coordSys) { - var coordDimsInfos_1 = map(coordSys && coordSys.dimensions, function (coordDim) { - var data = seriesModel.getData(); - var info = data.getDimensionInfo(data.mapDimension(coordDim)) || {}; // In map series data don't have lng and lat dimension. Fallback to same with coordSys - - return extend(extend({}, info), { - name: coordDim, - // DON'T use ordinalMeta to parse and collect ordinal. - ordinalMeta: null - }); - }); - dataDims = map(dims, function (dim, idx) { - return { - name: dim, - type: coordDimsInfos_1[idx % 2].type - }; - }); - areaData = new SeriesData(dataDims, maModel); - } else { - dataDims = [{ - name: 'value', - type: 'float' - }]; - areaData = new SeriesData(dataDims, maModel); - } - - var optData = map(maModel.get('data'), curry(markAreaTransform, seriesModel, coordSys, maModel)); - - if (coordSys) { - optData = filter(optData, curry(markAreaFilter, coordSys)); - } - - var dimValueGetter = coordSys ? function (item, dimName, dataIndex, dimIndex) { - // TODO should convert to ParsedValue? - var rawVal = item.coord[Math.floor(dimIndex / 2)][dimIndex % 2]; - return parseDataValue(rawVal, dataDims[dimIndex]); - } : function (item, dimName, dataIndex, dimIndex) { - return parseDataValue(item.value, dataDims[dimIndex]); - }; - areaData.initData(optData, null, dimValueGetter); - areaData.hasItemOption = true; - return areaData; - } - - function install$G(registers) { - registers.registerComponentModel(MarkAreaModel); - registers.registerComponentView(MarkAreaView); - registers.registerPreprocessor(function (opt) { - if (checkMarkerInSeries(opt.series, 'markArea')) { - // Make sure markArea component is enabled - opt.markArea = opt.markArea || {}; - } - }); - } - - var getDefaultSelectorOptions = function (ecModel, type) { - if (type === 'all') { - return { - type: 'all', - title: ecModel.getLocaleModel().get(['legend', 'selector', 'all']) - }; - } else if (type === 'inverse') { - return { - type: 'inverse', - title: ecModel.getLocaleModel().get(['legend', 'selector', 'inverse']) - }; - } - }; - - var LegendModel = - /** @class */ - function (_super) { - __extends(LegendModel, _super); - - function LegendModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = LegendModel.type; - _this.layoutMode = { - type: 'box', - // legend.width/height are maxWidth/maxHeight actually, - // whereas real width/height is calculated by its content. - // (Setting {left: 10, right: 10} does not make sense). - // So consider the case: - // `setOption({legend: {left: 10});` - // then `setOption({legend: {right: 10});` - // The previous `left` should be cleared by setting `ignoreSize`. - ignoreSize: true - }; - return _this; - } - - LegendModel.prototype.init = function (option, parentModel, ecModel) { - this.mergeDefaultAndTheme(option, ecModel); - option.selected = option.selected || {}; - - this._updateSelector(option); - }; - - LegendModel.prototype.mergeOption = function (option, ecModel) { - _super.prototype.mergeOption.call(this, option, ecModel); - - this._updateSelector(option); - }; - - LegendModel.prototype._updateSelector = function (option) { - var selector = option.selector; - var ecModel = this.ecModel; - - if (selector === true) { - selector = option.selector = ['all', 'inverse']; - } - - if (isArray(selector)) { - each(selector, function (item, index) { - isString(item) && (item = { - type: item - }); - selector[index] = merge(item, getDefaultSelectorOptions(ecModel, item.type)); - }); - } - }; - - LegendModel.prototype.optionUpdated = function () { - this._updateData(this.ecModel); - - var legendData = this._data; // If selectedMode is single, try to select one - - if (legendData[0] && this.get('selectedMode') === 'single') { - var hasSelected = false; // If has any selected in option.selected - - for (var i = 0; i < legendData.length; i++) { - var name_1 = legendData[i].get('name'); - - if (this.isSelected(name_1)) { - // Force to unselect others - this.select(name_1); - hasSelected = true; - break; - } - } // Try select the first if selectedMode is single - - - !hasSelected && this.select(legendData[0].get('name')); - } - }; - - LegendModel.prototype._updateData = function (ecModel) { - var potentialData = []; - var availableNames = []; - ecModel.eachRawSeries(function (seriesModel) { - var seriesName = seriesModel.name; - availableNames.push(seriesName); - var isPotential; - - if (seriesModel.legendVisualProvider) { - var provider = seriesModel.legendVisualProvider; - var names = provider.getAllNames(); - - if (!ecModel.isSeriesFiltered(seriesModel)) { - availableNames = availableNames.concat(names); - } - - if (names.length) { - potentialData = potentialData.concat(names); - } else { - isPotential = true; - } - } else { - isPotential = true; - } - - if (isPotential && isNameSpecified(seriesModel)) { - potentialData.push(seriesModel.name); - } - }); - /** - * @type {Array.<string>} - * @private - */ - - this._availableNames = availableNames; // If legend.data is not specified in option, use availableNames as data, - // which is convenient for user preparing option. - - var rawData = this.get('data') || potentialData; - var legendNameMap = createHashMap(); - var legendData = map(rawData, function (dataItem) { - // Can be string or number - if (isString(dataItem) || isNumber(dataItem)) { - dataItem = { - name: dataItem - }; - } - - if (legendNameMap.get(dataItem.name)) { - // remove legend name duplicate - return null; - } - - legendNameMap.set(dataItem.name, true); - return new Model(dataItem, this, this.ecModel); - }, this); - /** - * @type {Array.<module:echarts/model/Model>} - * @private - */ - - this._data = filter(legendData, function (item) { - return !!item; - }); - }; - - LegendModel.prototype.getData = function () { - return this._data; - }; - - LegendModel.prototype.select = function (name) { - var selected = this.option.selected; - var selectedMode = this.get('selectedMode'); - - if (selectedMode === 'single') { - var data = this._data; - each(data, function (dataItem) { - selected[dataItem.get('name')] = false; - }); - } - - selected[name] = true; - }; - - LegendModel.prototype.unSelect = function (name) { - if (this.get('selectedMode') !== 'single') { - this.option.selected[name] = false; - } - }; - - LegendModel.prototype.toggleSelected = function (name) { - var selected = this.option.selected; // Default is true - - if (!selected.hasOwnProperty(name)) { - selected[name] = true; - } - - this[selected[name] ? 'unSelect' : 'select'](name); - }; - - LegendModel.prototype.allSelect = function () { - var data = this._data; - var selected = this.option.selected; - each(data, function (dataItem) { - selected[dataItem.get('name', true)] = true; - }); - }; - - LegendModel.prototype.inverseSelect = function () { - var data = this._data; - var selected = this.option.selected; - each(data, function (dataItem) { - var name = dataItem.get('name', true); // Initially, default value is true - - if (!selected.hasOwnProperty(name)) { - selected[name] = true; - } - - selected[name] = !selected[name]; - }); - }; - - LegendModel.prototype.isSelected = function (name) { - var selected = this.option.selected; - return !(selected.hasOwnProperty(name) && !selected[name]) && indexOf(this._availableNames, name) >= 0; - }; - - LegendModel.prototype.getOrient = function () { - return this.get('orient') === 'vertical' ? { - index: 1, - name: 'vertical' - } : { - index: 0, - name: 'horizontal' - }; - }; - - LegendModel.type = 'legend.plain'; - LegendModel.dependencies = ['series']; - LegendModel.defaultOption = { - // zlevel: 0, - z: 4, - show: true, - orient: 'horizontal', - left: 'center', - // right: 'center', - top: 0, - // bottom: null, - align: 'auto', - backgroundColor: 'rgba(0,0,0,0)', - borderColor: '#ccc', - borderRadius: 0, - borderWidth: 0, - padding: 5, - itemGap: 10, - itemWidth: 25, - itemHeight: 14, - symbolRotate: 'inherit', - symbolKeepAspect: true, - inactiveColor: '#ccc', - inactiveBorderColor: '#ccc', - inactiveBorderWidth: 'auto', - itemStyle: { - color: 'inherit', - opacity: 'inherit', - borderColor: 'inherit', - borderWidth: 'auto', - borderCap: 'inherit', - borderJoin: 'inherit', - borderDashOffset: 'inherit', - borderMiterLimit: 'inherit' - }, - lineStyle: { - width: 'auto', - color: 'inherit', - inactiveColor: '#ccc', - inactiveWidth: 2, - opacity: 'inherit', - type: 'inherit', - cap: 'inherit', - join: 'inherit', - dashOffset: 'inherit', - miterLimit: 'inherit' - }, - textStyle: { - color: '#333' - }, - selectedMode: true, - selector: false, - selectorLabel: { - show: true, - borderRadius: 10, - padding: [3, 5, 3, 5], - fontSize: 12, - fontFamily: 'sans-serif', - color: '#666', - borderWidth: 1, - borderColor: '#666' - }, - emphasis: { - selectorLabel: { - show: true, - color: '#eee', - backgroundColor: '#666' - } - }, - selectorPosition: 'auto', - selectorItemGap: 7, - selectorButtonGap: 10, - tooltip: { - show: false - } - }; - return LegendModel; - }(ComponentModel); - - var curry$1 = curry; - var each$c = each; - var Group$2 = Group; - - var LegendView = - /** @class */ - function (_super) { - __extends(LegendView, _super); - - function LegendView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = LegendView.type; - _this.newlineDisabled = false; - return _this; - } - - LegendView.prototype.init = function () { - this.group.add(this._contentGroup = new Group$2()); - this.group.add(this._selectorGroup = new Group$2()); - this._isFirstRender = true; - }; - /** - * @protected - */ - - - LegendView.prototype.getContentGroup = function () { - return this._contentGroup; - }; - /** - * @protected - */ - - - LegendView.prototype.getSelectorGroup = function () { - return this._selectorGroup; - }; - /** - * @override - */ - - - LegendView.prototype.render = function (legendModel, ecModel, api) { - var isFirstRender = this._isFirstRender; - this._isFirstRender = false; - this.resetInner(); - - if (!legendModel.get('show', true)) { - return; - } - - var itemAlign = legendModel.get('align'); - var orient = legendModel.get('orient'); - - if (!itemAlign || itemAlign === 'auto') { - itemAlign = legendModel.get('left') === 'right' && orient === 'vertical' ? 'right' : 'left'; - } // selector has been normalized to an array in model - - - var selector = legendModel.get('selector', true); - var selectorPosition = legendModel.get('selectorPosition', true); - - if (selector && (!selectorPosition || selectorPosition === 'auto')) { - selectorPosition = orient === 'horizontal' ? 'end' : 'start'; - } - - this.renderInner(itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition); // Perform layout. - - var positionInfo = legendModel.getBoxLayoutParams(); - var viewportSize = { - width: api.getWidth(), - height: api.getHeight() - }; - var padding = legendModel.get('padding'); - var maxSize = getLayoutRect(positionInfo, viewportSize, padding); - var mainRect = this.layoutInner(legendModel, itemAlign, maxSize, isFirstRender, selector, selectorPosition); // Place mainGroup, based on the calculated `mainRect`. - - var layoutRect = getLayoutRect(defaults({ - width: mainRect.width, - height: mainRect.height - }, positionInfo), viewportSize, padding); - this.group.x = layoutRect.x - mainRect.x; - this.group.y = layoutRect.y - mainRect.y; - this.group.markRedraw(); // Render background after group is layout. - - this.group.add(this._backgroundEl = makeBackground(mainRect, legendModel)); - }; - - LegendView.prototype.resetInner = function () { - this.getContentGroup().removeAll(); - this._backgroundEl && this.group.remove(this._backgroundEl); - this.getSelectorGroup().removeAll(); - }; - - LegendView.prototype.renderInner = function (itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition) { - var contentGroup = this.getContentGroup(); - var legendDrawnMap = createHashMap(); - var selectMode = legendModel.get('selectedMode'); - var excludeSeriesId = []; - ecModel.eachRawSeries(function (seriesModel) { - !seriesModel.get('legendHoverLink') && excludeSeriesId.push(seriesModel.id); - }); - each$c(legendModel.getData(), function (legendItemModel, dataIndex) { - var name = legendItemModel.get('name'); // Use empty string or \n as a newline string - - if (!this.newlineDisabled && (name === '' || name === '\n')) { - var g = new Group$2(); // @ts-ignore - - g.newline = true; - contentGroup.add(g); - return; - } // Representitive series. - - - var seriesModel = ecModel.getSeriesByName(name)[0]; - - if (legendDrawnMap.get(name)) { - // Have been drawn - return; - } // Legend to control series. - - - if (seriesModel) { - var data = seriesModel.getData(); - var lineVisualStyle = data.getVisual('legendLineStyle') || {}; - var legendIcon = data.getVisual('legendIcon'); - /** - * `data.getVisual('style')` may be the color from the register - * in series. For example, for line series, - */ - - var style = data.getVisual('style'); - - var itemGroup = this._createItem(seriesModel, name, dataIndex, legendItemModel, legendModel, itemAlign, lineVisualStyle, style, legendIcon, selectMode, api); - - itemGroup.on('click', curry$1(dispatchSelectAction, name, null, api, excludeSeriesId)).on('mouseover', curry$1(dispatchHighlightAction, seriesModel.name, null, api, excludeSeriesId)).on('mouseout', curry$1(dispatchDownplayAction, seriesModel.name, null, api, excludeSeriesId)); - legendDrawnMap.set(name, true); - } else { - // Legend to control data. In pie and funnel. - ecModel.eachRawSeries(function (seriesModel) { - // In case multiple series has same data name - if (legendDrawnMap.get(name)) { - return; - } - - if (seriesModel.legendVisualProvider) { - var provider = seriesModel.legendVisualProvider; - - if (!provider.containName(name)) { - return; - } - - var idx = provider.indexOfName(name); - var style = provider.getItemVisual(idx, 'style'); - var legendIcon = provider.getItemVisual(idx, 'legendIcon'); - var colorArr = parse(style.fill); // Color may be set to transparent in visualMap when data is out of range. - // Do not show nothing. - - if (colorArr && colorArr[3] === 0) { - colorArr[3] = 0.2; // TODO color is set to 0, 0, 0, 0. Should show correct RGBA - - style = extend(extend({}, style), { - fill: stringify(colorArr, 'rgba') - }); - } - - var itemGroup = this._createItem(seriesModel, name, dataIndex, legendItemModel, legendModel, itemAlign, {}, style, legendIcon, selectMode, api); // FIXME: consider different series has items with the same name. - - - itemGroup.on('click', curry$1(dispatchSelectAction, null, name, api, excludeSeriesId)) // Should not specify the series name, consider legend controls - // more than one pie series. - .on('mouseover', curry$1(dispatchHighlightAction, null, name, api, excludeSeriesId)).on('mouseout', curry$1(dispatchDownplayAction, null, name, api, excludeSeriesId)); - legendDrawnMap.set(name, true); - } - }, this); - } - - if ("development" !== 'production') { - if (!legendDrawnMap.get(name)) { - console.warn(name + ' series not exists. Legend data should be same with series name or data name.'); - } - } - }, this); - - if (selector) { - this._createSelector(selector, legendModel, api, orient, selectorPosition); - } - }; - - LegendView.prototype._createSelector = function (selector, legendModel, api, orient, selectorPosition) { - var selectorGroup = this.getSelectorGroup(); - each$c(selector, function createSelectorButton(selectorItem) { - var type = selectorItem.type; - var labelText = new ZRText({ - style: { - x: 0, - y: 0, - align: 'center', - verticalAlign: 'middle' - }, - onclick: function () { - api.dispatchAction({ - type: type === 'all' ? 'legendAllSelect' : 'legendInverseSelect' - }); - } - }); - selectorGroup.add(labelText); - var labelModel = legendModel.getModel('selectorLabel'); - var emphasisLabelModel = legendModel.getModel(['emphasis', 'selectorLabel']); - setLabelStyle(labelText, { - normal: labelModel, - emphasis: emphasisLabelModel - }, { - defaultText: selectorItem.title - }); - enableHoverEmphasis(labelText); - }); - }; - - LegendView.prototype._createItem = function (seriesModel, name, dataIndex, legendItemModel, legendModel, itemAlign, lineVisualStyle, itemVisualStyle, legendIcon, selectMode, api) { - var drawType = seriesModel.visualDrawType; - var itemWidth = legendModel.get('itemWidth'); - var itemHeight = legendModel.get('itemHeight'); - var isSelected = legendModel.isSelected(name); - var iconRotate = legendItemModel.get('symbolRotate'); - var symbolKeepAspect = legendItemModel.get('symbolKeepAspect'); - var legendIconType = legendItemModel.get('icon'); - legendIcon = legendIconType || legendIcon || 'roundRect'; - var style = getLegendStyle(legendIcon, legendItemModel, lineVisualStyle, itemVisualStyle, drawType, isSelected, api); - var itemGroup = new Group$2(); - var textStyleModel = legendItemModel.getModel('textStyle'); - - if (isFunction(seriesModel.getLegendIcon) && (!legendIconType || legendIconType === 'inherit')) { - // Series has specific way to define legend icon - itemGroup.add(seriesModel.getLegendIcon({ - itemWidth: itemWidth, - itemHeight: itemHeight, - icon: legendIcon, - iconRotate: iconRotate, - itemStyle: style.itemStyle, - lineStyle: style.lineStyle, - symbolKeepAspect: symbolKeepAspect - })); - } else { - // Use default legend icon policy for most series - var rotate = legendIconType === 'inherit' && seriesModel.getData().getVisual('symbol') ? iconRotate === 'inherit' ? seriesModel.getData().getVisual('symbolRotate') : iconRotate : 0; // No rotation for no icon - - itemGroup.add(getDefaultLegendIcon({ - itemWidth: itemWidth, - itemHeight: itemHeight, - icon: legendIcon, - iconRotate: rotate, - itemStyle: style.itemStyle, - lineStyle: style.lineStyle, - symbolKeepAspect: symbolKeepAspect - })); - } - - var textX = itemAlign === 'left' ? itemWidth + 5 : -5; - var textAlign = itemAlign; - var formatter = legendModel.get('formatter'); - var content = name; - - if (isString(formatter) && formatter) { - content = formatter.replace('{name}', name != null ? name : ''); - } else if (isFunction(formatter)) { - content = formatter(name); - } - - var inactiveColor = legendItemModel.get('inactiveColor'); - itemGroup.add(new ZRText({ - style: createTextStyle(textStyleModel, { - text: content, - x: textX, - y: itemHeight / 2, - fill: isSelected ? textStyleModel.getTextColor() : inactiveColor, - align: textAlign, - verticalAlign: 'middle' - }) - })); // Add a invisible rect to increase the area of mouse hover - - var hitRect = new Rect({ - shape: itemGroup.getBoundingRect(), - invisible: true - }); - var tooltipModel = legendItemModel.getModel('tooltip'); - - if (tooltipModel.get('show')) { - setTooltipConfig({ - el: hitRect, - componentModel: legendModel, - itemName: name, - itemTooltipOption: tooltipModel.option - }); - } - - itemGroup.add(hitRect); - itemGroup.eachChild(function (child) { - child.silent = true; - }); - hitRect.silent = !selectMode; - this.getContentGroup().add(itemGroup); - enableHoverEmphasis(itemGroup); // @ts-ignore - - itemGroup.__legendDataIndex = dataIndex; - return itemGroup; - }; - - LegendView.prototype.layoutInner = function (legendModel, itemAlign, maxSize, isFirstRender, selector, selectorPosition) { - var contentGroup = this.getContentGroup(); - var selectorGroup = this.getSelectorGroup(); // Place items in contentGroup. - - box(legendModel.get('orient'), contentGroup, legendModel.get('itemGap'), maxSize.width, maxSize.height); - var contentRect = contentGroup.getBoundingRect(); - var contentPos = [-contentRect.x, -contentRect.y]; - selectorGroup.markRedraw(); - contentGroup.markRedraw(); - - if (selector) { - // Place buttons in selectorGroup - box( // Buttons in selectorGroup always layout horizontally - 'horizontal', selectorGroup, legendModel.get('selectorItemGap', true)); - var selectorRect = selectorGroup.getBoundingRect(); - var selectorPos = [-selectorRect.x, -selectorRect.y]; - var selectorButtonGap = legendModel.get('selectorButtonGap', true); - var orientIdx = legendModel.getOrient().index; - var wh = orientIdx === 0 ? 'width' : 'height'; - var hw = orientIdx === 0 ? 'height' : 'width'; - var yx = orientIdx === 0 ? 'y' : 'x'; - - if (selectorPosition === 'end') { - selectorPos[orientIdx] += contentRect[wh] + selectorButtonGap; - } else { - contentPos[orientIdx] += selectorRect[wh] + selectorButtonGap; - } // Always align selector to content as 'middle' - - - selectorPos[1 - orientIdx] += contentRect[hw] / 2 - selectorRect[hw] / 2; - selectorGroup.x = selectorPos[0]; - selectorGroup.y = selectorPos[1]; - contentGroup.x = contentPos[0]; - contentGroup.y = contentPos[1]; - var mainRect = { - x: 0, - y: 0 - }; - mainRect[wh] = contentRect[wh] + selectorButtonGap + selectorRect[wh]; - mainRect[hw] = Math.max(contentRect[hw], selectorRect[hw]); - mainRect[yx] = Math.min(0, selectorRect[yx] + selectorPos[1 - orientIdx]); - return mainRect; - } else { - contentGroup.x = contentPos[0]; - contentGroup.y = contentPos[1]; - return this.group.getBoundingRect(); - } - }; - /** - * @protected - */ - - - LegendView.prototype.remove = function () { - this.getContentGroup().removeAll(); - this._isFirstRender = true; - }; - - LegendView.type = 'legend.plain'; - return LegendView; - }(ComponentView); - - function getLegendStyle(iconType, legendItemModel, lineVisualStyle, itemVisualStyle, drawType, isSelected, api) { - /** - * Use series style if is inherit; - * elsewise, use legend style - */ - function handleCommonProps(style, visualStyle) { - // If lineStyle.width is 'auto', it is set to be 2 if series has border - if (style.lineWidth === 'auto') { - style.lineWidth = visualStyle.lineWidth > 0 ? 2 : 0; - } - - each$c(style, function (propVal, propName) { - style[propName] === 'inherit' && (style[propName] = visualStyle[propName]); - }); - } // itemStyle - - - var itemStyleModel = legendItemModel.getModel('itemStyle'); - var itemStyle = itemStyleModel.getItemStyle(); - var iconBrushType = iconType.lastIndexOf('empty', 0) === 0 ? 'fill' : 'stroke'; - var decalStyle = itemStyleModel.getShallow('decal'); - itemStyle.decal = !decalStyle || decalStyle === 'inherit' ? itemVisualStyle.decal : createOrUpdatePatternFromDecal(decalStyle, api); - - if (itemStyle.fill === 'inherit') { - /** - * Series with visualDrawType as 'stroke' should have - * series stroke as legend fill - */ - itemStyle.fill = itemVisualStyle[drawType]; - } - - if (itemStyle.stroke === 'inherit') { - /** - * icon type with "emptyXXX" should use fill color - * in visual style - */ - itemStyle.stroke = itemVisualStyle[iconBrushType]; - } - - if (itemStyle.opacity === 'inherit') { - /** - * Use lineStyle.opacity if drawType is stroke - */ - itemStyle.opacity = (drawType === 'fill' ? itemVisualStyle : lineVisualStyle).opacity; - } - - handleCommonProps(itemStyle, itemVisualStyle); // lineStyle - - var legendLineModel = legendItemModel.getModel('lineStyle'); - var lineStyle = legendLineModel.getLineStyle(); - handleCommonProps(lineStyle, lineVisualStyle); // Fix auto color to real color - - itemStyle.fill === 'auto' && (itemStyle.fill = itemVisualStyle.fill); - itemStyle.stroke === 'auto' && (itemStyle.stroke = itemVisualStyle.fill); - lineStyle.stroke === 'auto' && (lineStyle.stroke = itemVisualStyle.fill); - - if (!isSelected) { - var borderWidth = legendItemModel.get('inactiveBorderWidth'); - /** - * Since stroke is set to be inactiveBorderColor, it may occur that - * there is no border in series but border in legend, so we need to - * use border only when series has border if is set to be auto - */ - - var visualHasBorder = itemStyle[iconBrushType]; - itemStyle.lineWidth = borderWidth === 'auto' ? itemVisualStyle.lineWidth > 0 && visualHasBorder ? 2 : 0 : itemStyle.lineWidth; - itemStyle.fill = legendItemModel.get('inactiveColor'); - itemStyle.stroke = legendItemModel.get('inactiveBorderColor'); - lineStyle.stroke = legendLineModel.get('inactiveColor'); - lineStyle.lineWidth = legendLineModel.get('inactiveWidth'); - } - - return { - itemStyle: itemStyle, - lineStyle: lineStyle - }; - } - - function getDefaultLegendIcon(opt) { - var symboType = opt.icon || 'roundRect'; - var icon = createSymbol(symboType, 0, 0, opt.itemWidth, opt.itemHeight, opt.itemStyle.fill, opt.symbolKeepAspect); - icon.setStyle(opt.itemStyle); - icon.rotation = (opt.iconRotate || 0) * Math.PI / 180; - icon.setOrigin([opt.itemWidth / 2, opt.itemHeight / 2]); - - if (symboType.indexOf('empty') > -1) { - icon.style.stroke = icon.style.fill; - icon.style.fill = '#fff'; - icon.style.lineWidth = 2; - } - - return icon; - } - - function dispatchSelectAction(seriesName, dataName, api, excludeSeriesId) { - // downplay before unselect - dispatchDownplayAction(seriesName, dataName, api, excludeSeriesId); - api.dispatchAction({ - type: 'legendToggleSelect', - name: seriesName != null ? seriesName : dataName - }); // highlight after select - // TODO highlight immediately may cause animation loss. - - dispatchHighlightAction(seriesName, dataName, api, excludeSeriesId); - } - - function isUseHoverLayer(api) { - var list = api.getZr().storage.getDisplayList(); - var emphasisState; - var i = 0; - var len = list.length; - - while (i < len && !(emphasisState = list[i].states.emphasis)) { - i++; - } - - return emphasisState && emphasisState.hoverLayer; - } - - function dispatchHighlightAction(seriesName, dataName, api, excludeSeriesId) { - // If element hover will move to a hoverLayer. - if (!isUseHoverLayer(api)) { - api.dispatchAction({ - type: 'highlight', - seriesName: seriesName, - name: dataName, - excludeSeriesId: excludeSeriesId - }); - } - } - - function dispatchDownplayAction(seriesName, dataName, api, excludeSeriesId) { - // If element hover will move to a hoverLayer. - if (!isUseHoverLayer(api)) { - api.dispatchAction({ - type: 'downplay', - seriesName: seriesName, - name: dataName, - excludeSeriesId: excludeSeriesId - }); - } - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function legendFilter(ecModel) { - var legendModels = ecModel.findComponents({ - mainType: 'legend' - }); - - if (legendModels && legendModels.length) { - ecModel.filterSeries(function (series) { - // If in any legend component the status is not selected. - // Because in legend series is assumed selected when it is not in the legend data. - for (var i = 0; i < legendModels.length; i++) { - if (!legendModels[i].isSelected(series.name)) { - return false; - } - } - - return true; - }); - } - } - - function legendSelectActionHandler(methodName, payload, ecModel) { - var selectedMap = {}; - var isToggleSelect = methodName === 'toggleSelected'; - var isSelected; // Update all legend components - - ecModel.eachComponent('legend', function (legendModel) { - if (isToggleSelect && isSelected != null) { - // Force other legend has same selected status - // Or the first is toggled to true and other are toggled to false - // In the case one legend has some item unSelected in option. And if other legend - // doesn't has the item, they will assume it is selected. - legendModel[isSelected ? 'select' : 'unSelect'](payload.name); - } else if (methodName === 'allSelect' || methodName === 'inverseSelect') { - legendModel[methodName](); - } else { - legendModel[methodName](payload.name); - isSelected = legendModel.isSelected(payload.name); - } - - var legendData = legendModel.getData(); - each(legendData, function (model) { - var name = model.get('name'); // Wrap element - - if (name === '\n' || name === '') { - return; - } - - var isItemSelected = legendModel.isSelected(name); - - if (selectedMap.hasOwnProperty(name)) { - // Unselected if any legend is unselected - selectedMap[name] = selectedMap[name] && isItemSelected; - } else { - selectedMap[name] = isItemSelected; - } - }); - }); // Return the event explicitly - - return methodName === 'allSelect' || methodName === 'inverseSelect' ? { - selected: selectedMap - } : { - name: payload.name, - selected: selectedMap - }; - } - - function installLegendAction(registers) { - /** - * @event legendToggleSelect - * @type {Object} - * @property {string} type 'legendToggleSelect' - * @property {string} [from] - * @property {string} name Series name or data item name - */ - registers.registerAction('legendToggleSelect', 'legendselectchanged', curry(legendSelectActionHandler, 'toggleSelected')); - registers.registerAction('legendAllSelect', 'legendselectall', curry(legendSelectActionHandler, 'allSelect')); - registers.registerAction('legendInverseSelect', 'legendinverseselect', curry(legendSelectActionHandler, 'inverseSelect')); - /** - * @event legendSelect - * @type {Object} - * @property {string} type 'legendSelect' - * @property {string} name Series name or data item name - */ - - registers.registerAction('legendSelect', 'legendselected', curry(legendSelectActionHandler, 'select')); - /** - * @event legendUnSelect - * @type {Object} - * @property {string} type 'legendUnSelect' - * @property {string} name Series name or data item name - */ - - registers.registerAction('legendUnSelect', 'legendunselected', curry(legendSelectActionHandler, 'unSelect')); - } - - function install$H(registers) { - registers.registerComponentModel(LegendModel); - registers.registerComponentView(LegendView); - registers.registerProcessor(registers.PRIORITY.PROCESSOR.SERIES_FILTER, legendFilter); - registers.registerSubTypeDefaulter('legend', function () { - return 'plain'; - }); - installLegendAction(registers); - } - - var ScrollableLegendModel = - /** @class */ - function (_super) { - __extends(ScrollableLegendModel, _super); - - function ScrollableLegendModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ScrollableLegendModel.type; - return _this; - } - /** - * @param {number} scrollDataIndex - */ - - - ScrollableLegendModel.prototype.setScrollDataIndex = function (scrollDataIndex) { - this.option.scrollDataIndex = scrollDataIndex; - }; - - ScrollableLegendModel.prototype.init = function (option, parentModel, ecModel) { - var inputPositionParams = getLayoutParams(option); - - _super.prototype.init.call(this, option, parentModel, ecModel); - - mergeAndNormalizeLayoutParams$1(this, option, inputPositionParams); - }; - /** - * @override - */ - - - ScrollableLegendModel.prototype.mergeOption = function (option, ecModel) { - _super.prototype.mergeOption.call(this, option, ecModel); - - mergeAndNormalizeLayoutParams$1(this, this.option, option); - }; - - ScrollableLegendModel.type = 'legend.scroll'; - ScrollableLegendModel.defaultOption = inheritDefaultOption(LegendModel.defaultOption, { - scrollDataIndex: 0, - pageButtonItemGap: 5, - pageButtonGap: null, - pageButtonPosition: 'end', - pageFormatter: '{current}/{total}', - pageIcons: { - horizontal: ['M0,0L12,-10L12,10z', 'M0,0L-12,-10L-12,10z'], - vertical: ['M0,0L20,0L10,-20z', 'M0,0L20,0L10,20z'] - }, - pageIconColor: '#2f4554', - pageIconInactiveColor: '#aaa', - pageIconSize: 15, - pageTextStyle: { - color: '#333' - }, - animationDurationUpdate: 800 - }); - return ScrollableLegendModel; - }(LegendModel); - - function mergeAndNormalizeLayoutParams$1(legendModel, target, raw) { - var orient = legendModel.getOrient(); - var ignoreSize = [1, 1]; - ignoreSize[orient.index] = 0; - mergeLayoutParam(target, raw, { - type: 'box', - ignoreSize: !!ignoreSize - }); - } - - var Group$3 = Group; - var WH$1 = ['width', 'height']; - var XY$1 = ['x', 'y']; - - var ScrollableLegendView = - /** @class */ - function (_super) { - __extends(ScrollableLegendView, _super); - - function ScrollableLegendView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ScrollableLegendView.type; - _this.newlineDisabled = true; - _this._currentIndex = 0; - return _this; - } - - ScrollableLegendView.prototype.init = function () { - _super.prototype.init.call(this); - - this.group.add(this._containerGroup = new Group$3()); - - this._containerGroup.add(this.getContentGroup()); - - this.group.add(this._controllerGroup = new Group$3()); - }; - /** - * @override - */ - - - ScrollableLegendView.prototype.resetInner = function () { - _super.prototype.resetInner.call(this); - - this._controllerGroup.removeAll(); - - this._containerGroup.removeClipPath(); - - this._containerGroup.__rectSize = null; - }; - /** - * @override - */ - - - ScrollableLegendView.prototype.renderInner = function (itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition) { - var self = this; // Render content items. - - _super.prototype.renderInner.call(this, itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition); - - var controllerGroup = this._controllerGroup; // FIXME: support be 'auto' adapt to size number text length, - // e.g., '3/12345' should not overlap with the control arrow button. - - var pageIconSize = legendModel.get('pageIconSize', true); - var pageIconSizeArr = isArray(pageIconSize) ? pageIconSize : [pageIconSize, pageIconSize]; - createPageButton('pagePrev', 0); - var pageTextStyleModel = legendModel.getModel('pageTextStyle'); - controllerGroup.add(new ZRText({ - name: 'pageText', - style: { - // Placeholder to calculate a proper layout. - text: 'xx/xx', - fill: pageTextStyleModel.getTextColor(), - font: pageTextStyleModel.getFont(), - verticalAlign: 'middle', - align: 'center' - }, - silent: true - })); - createPageButton('pageNext', 1); - - function createPageButton(name, iconIdx) { - var pageDataIndexName = name + 'DataIndex'; - var icon = createIcon(legendModel.get('pageIcons', true)[legendModel.getOrient().name][iconIdx], { - // Buttons will be created in each render, so we do not need - // to worry about avoiding using legendModel kept in scope. - onclick: bind(self._pageGo, self, pageDataIndexName, legendModel, api) - }, { - x: -pageIconSizeArr[0] / 2, - y: -pageIconSizeArr[1] / 2, - width: pageIconSizeArr[0], - height: pageIconSizeArr[1] - }); - icon.name = name; - controllerGroup.add(icon); - } - }; - /** - * @override - */ - - - ScrollableLegendView.prototype.layoutInner = function (legendModel, itemAlign, maxSize, isFirstRender, selector, selectorPosition) { - var selectorGroup = this.getSelectorGroup(); - var orientIdx = legendModel.getOrient().index; - var wh = WH$1[orientIdx]; - var xy = XY$1[orientIdx]; - var hw = WH$1[1 - orientIdx]; - var yx = XY$1[1 - orientIdx]; - selector && box( // Buttons in selectorGroup always layout horizontally - 'horizontal', selectorGroup, legendModel.get('selectorItemGap', true)); - var selectorButtonGap = legendModel.get('selectorButtonGap', true); - var selectorRect = selectorGroup.getBoundingRect(); - var selectorPos = [-selectorRect.x, -selectorRect.y]; - var processMaxSize = clone(maxSize); - selector && (processMaxSize[wh] = maxSize[wh] - selectorRect[wh] - selectorButtonGap); - - var mainRect = this._layoutContentAndController(legendModel, isFirstRender, processMaxSize, orientIdx, wh, hw, yx, xy); - - if (selector) { - if (selectorPosition === 'end') { - selectorPos[orientIdx] += mainRect[wh] + selectorButtonGap; - } else { - var offset = selectorRect[wh] + selectorButtonGap; - selectorPos[orientIdx] -= offset; - mainRect[xy] -= offset; - } - - mainRect[wh] += selectorRect[wh] + selectorButtonGap; - selectorPos[1 - orientIdx] += mainRect[yx] + mainRect[hw] / 2 - selectorRect[hw] / 2; - mainRect[hw] = Math.max(mainRect[hw], selectorRect[hw]); - mainRect[yx] = Math.min(mainRect[yx], selectorRect[yx] + selectorPos[1 - orientIdx]); - selectorGroup.x = selectorPos[0]; - selectorGroup.y = selectorPos[1]; - selectorGroup.markRedraw(); - } - - return mainRect; - }; - - ScrollableLegendView.prototype._layoutContentAndController = function (legendModel, isFirstRender, maxSize, orientIdx, wh, hw, yx, xy) { - var contentGroup = this.getContentGroup(); - var containerGroup = this._containerGroup; - var controllerGroup = this._controllerGroup; // Place items in contentGroup. - - box(legendModel.get('orient'), contentGroup, legendModel.get('itemGap'), !orientIdx ? null : maxSize.width, orientIdx ? null : maxSize.height); - box( // Buttons in controller are layout always horizontally. - 'horizontal', controllerGroup, legendModel.get('pageButtonItemGap', true)); - var contentRect = contentGroup.getBoundingRect(); - var controllerRect = controllerGroup.getBoundingRect(); - var showController = this._showController = contentRect[wh] > maxSize[wh]; // In case that the inner elements of contentGroup layout do not based on [0, 0] - - var contentPos = [-contentRect.x, -contentRect.y]; // Remain contentPos when scroll animation perfroming. - // If first rendering, `contentGroup.position` is [0, 0], which - // does not make sense and may cause unexepcted animation if adopted. - - if (!isFirstRender) { - contentPos[orientIdx] = contentGroup[xy]; - } // Layout container group based on 0. - - - var containerPos = [0, 0]; - var controllerPos = [-controllerRect.x, -controllerRect.y]; - var pageButtonGap = retrieve2(legendModel.get('pageButtonGap', true), legendModel.get('itemGap', true)); // Place containerGroup and controllerGroup and contentGroup. - - if (showController) { - var pageButtonPosition = legendModel.get('pageButtonPosition', true); // controller is on the right / bottom. - - if (pageButtonPosition === 'end') { - controllerPos[orientIdx] += maxSize[wh] - controllerRect[wh]; - } // controller is on the left / top. - else { - containerPos[orientIdx] += controllerRect[wh] + pageButtonGap; - } - } // Always align controller to content as 'middle'. - - - controllerPos[1 - orientIdx] += contentRect[hw] / 2 - controllerRect[hw] / 2; - contentGroup.setPosition(contentPos); - containerGroup.setPosition(containerPos); - controllerGroup.setPosition(controllerPos); // Calculate `mainRect` and set `clipPath`. - // mainRect should not be calculated by `this.group.getBoundingRect()` - // for sake of the overflow. - - var mainRect = { - x: 0, - y: 0 - }; // Consider content may be overflow (should be clipped). - - mainRect[wh] = showController ? maxSize[wh] : contentRect[wh]; - mainRect[hw] = Math.max(contentRect[hw], controllerRect[hw]); // `containerRect[yx] + containerPos[1 - orientIdx]` is 0. - - mainRect[yx] = Math.min(0, controllerRect[yx] + controllerPos[1 - orientIdx]); - containerGroup.__rectSize = maxSize[wh]; - - if (showController) { - var clipShape = { - x: 0, - y: 0 - }; - clipShape[wh] = Math.max(maxSize[wh] - controllerRect[wh] - pageButtonGap, 0); - clipShape[hw] = mainRect[hw]; - containerGroup.setClipPath(new Rect({ - shape: clipShape - })); // Consider content may be larger than container, container rect - // can not be obtained from `containerGroup.getBoundingRect()`. - - containerGroup.__rectSize = clipShape[wh]; - } else { - // Do not remove or ignore controller. Keep them set as placeholders. - controllerGroup.eachChild(function (child) { - child.attr({ - invisible: true, - silent: true - }); - }); - } // Content translate animation. - - - var pageInfo = this._getPageInfo(legendModel); - - pageInfo.pageIndex != null && updateProps(contentGroup, { - x: pageInfo.contentPosition[0], - y: pageInfo.contentPosition[1] - }, // When switch from "show controller" to "not show controller", view should be - // updated immediately without animation, otherwise causes weird effect. - showController ? legendModel : null); - - this._updatePageInfoView(legendModel, pageInfo); - - return mainRect; - }; - - ScrollableLegendView.prototype._pageGo = function (to, legendModel, api) { - var scrollDataIndex = this._getPageInfo(legendModel)[to]; - - scrollDataIndex != null && api.dispatchAction({ - type: 'legendScroll', - scrollDataIndex: scrollDataIndex, - legendId: legendModel.id - }); - }; - - ScrollableLegendView.prototype._updatePageInfoView = function (legendModel, pageInfo) { - var controllerGroup = this._controllerGroup; - each(['pagePrev', 'pageNext'], function (name) { - var key = name + 'DataIndex'; - var canJump = pageInfo[key] != null; - var icon = controllerGroup.childOfName(name); - - if (icon) { - icon.setStyle('fill', canJump ? legendModel.get('pageIconColor', true) : legendModel.get('pageIconInactiveColor', true)); - icon.cursor = canJump ? 'pointer' : 'default'; - } - }); - var pageText = controllerGroup.childOfName('pageText'); - var pageFormatter = legendModel.get('pageFormatter'); - var pageIndex = pageInfo.pageIndex; - var current = pageIndex != null ? pageIndex + 1 : 0; - var total = pageInfo.pageCount; - pageText && pageFormatter && pageText.setStyle('text', isString(pageFormatter) ? pageFormatter.replace('{current}', current == null ? '' : current + '').replace('{total}', total == null ? '' : total + '') : pageFormatter({ - current: current, - total: total - })); - }; - /** - * contentPosition: Array.<number>, null when data item not found. - * pageIndex: number, null when data item not found. - * pageCount: number, always be a number, can be 0. - * pagePrevDataIndex: number, null when no previous page. - * pageNextDataIndex: number, null when no next page. - * } - */ - - - ScrollableLegendView.prototype._getPageInfo = function (legendModel) { - var scrollDataIndex = legendModel.get('scrollDataIndex', true); - var contentGroup = this.getContentGroup(); - var containerRectSize = this._containerGroup.__rectSize; - var orientIdx = legendModel.getOrient().index; - var wh = WH$1[orientIdx]; - var xy = XY$1[orientIdx]; - - var targetItemIndex = this._findTargetItemIndex(scrollDataIndex); - - var children = contentGroup.children(); - var targetItem = children[targetItemIndex]; - var itemCount = children.length; - var pCount = !itemCount ? 0 : 1; - var result = { - contentPosition: [contentGroup.x, contentGroup.y], - pageCount: pCount, - pageIndex: pCount - 1, - pagePrevDataIndex: null, - pageNextDataIndex: null - }; - - if (!targetItem) { - return result; - } - - var targetItemInfo = getItemInfo(targetItem); - result.contentPosition[orientIdx] = -targetItemInfo.s; // Strategy: - // (1) Always align based on the left/top most item. - // (2) It is user-friendly that the last item shown in the - // current window is shown at the begining of next window. - // Otherwise if half of the last item is cut by the window, - // it will have no chance to display entirely. - // (3) Consider that item size probably be different, we - // have calculate pageIndex by size rather than item index, - // and we can not get page index directly by division. - // (4) The window is to narrow to contain more than - // one item, we should make sure that the page can be fliped. - - for (var i = targetItemIndex + 1, winStartItemInfo = targetItemInfo, winEndItemInfo = targetItemInfo, currItemInfo = null; i <= itemCount; ++i) { - currItemInfo = getItemInfo(children[i]); - - if ( // Half of the last item is out of the window. - !currItemInfo && winEndItemInfo.e > winStartItemInfo.s + containerRectSize || // If the current item does not intersect with the window, the new page - // can be started at the current item or the last item. - currItemInfo && !intersect(currItemInfo, winStartItemInfo.s)) { - if (winEndItemInfo.i > winStartItemInfo.i) { - winStartItemInfo = winEndItemInfo; - } else { - // e.g., when page size is smaller than item size. - winStartItemInfo = currItemInfo; - } - - if (winStartItemInfo) { - if (result.pageNextDataIndex == null) { - result.pageNextDataIndex = winStartItemInfo.i; - } - - ++result.pageCount; - } - } - - winEndItemInfo = currItemInfo; - } - - for (var i = targetItemIndex - 1, winStartItemInfo = targetItemInfo, winEndItemInfo = targetItemInfo, currItemInfo = null; i >= -1; --i) { - currItemInfo = getItemInfo(children[i]); - - if ( // If the the end item does not intersect with the window started - // from the current item, a page can be settled. - (!currItemInfo || !intersect(winEndItemInfo, currItemInfo.s)) && // e.g., when page size is smaller than item size. - winStartItemInfo.i < winEndItemInfo.i) { - winEndItemInfo = winStartItemInfo; - - if (result.pagePrevDataIndex == null) { - result.pagePrevDataIndex = winStartItemInfo.i; - } - - ++result.pageCount; - ++result.pageIndex; - } - - winStartItemInfo = currItemInfo; - } - - return result; - - function getItemInfo(el) { - if (el) { - var itemRect = el.getBoundingRect(); - var start = itemRect[xy] + el[xy]; - return { - s: start, - e: start + itemRect[wh], - i: el.__legendDataIndex - }; - } - } - - function intersect(itemInfo, winStart) { - return itemInfo.e >= winStart && itemInfo.s <= winStart + containerRectSize; - } - }; - - ScrollableLegendView.prototype._findTargetItemIndex = function (targetDataIndex) { - if (!this._showController) { - return 0; - } - - var index; - var contentGroup = this.getContentGroup(); - var defaultIndex; - contentGroup.eachChild(function (child, idx) { - var legendDataIdx = child.__legendDataIndex; // FIXME - // If the given targetDataIndex (from model) is illegal, - // we use defaultIndex. But the index on the legend model and - // action payload is still illegal. That case will not be - // changed until some scenario requires. - - if (defaultIndex == null && legendDataIdx != null) { - defaultIndex = idx; - } - - if (legendDataIdx === targetDataIndex) { - index = idx; - } - }); - return index != null ? index : defaultIndex; - }; - - ScrollableLegendView.type = 'legend.scroll'; - return ScrollableLegendView; - }(LegendView); - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - function installScrollableLegendAction(registers) { - /** - * @event legendScroll - * @type {Object} - * @property {string} type 'legendScroll' - * @property {string} scrollDataIndex - */ - registers.registerAction('legendScroll', 'legendscroll', function (payload, ecModel) { - var scrollDataIndex = payload.scrollDataIndex; - scrollDataIndex != null && ecModel.eachComponent({ - mainType: 'legend', - subType: 'scroll', - query: payload - }, function (legendModel) { - legendModel.setScrollDataIndex(scrollDataIndex); - }); - }); - } - - function install$I(registers) { - use(install$H); - registers.registerComponentModel(ScrollableLegendModel); - registers.registerComponentView(ScrollableLegendView); - installScrollableLegendAction(registers); - } - - function install$J(registers) { - use(install$H); - use(install$I); - } - - var InsideZoomModel = - /** @class */ - function (_super) { - __extends(InsideZoomModel, _super); - - function InsideZoomModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = InsideZoomModel.type; - return _this; - } - - InsideZoomModel.type = 'dataZoom.inside'; - InsideZoomModel.defaultOption = inheritDefaultOption(DataZoomModel.defaultOption, { - disabled: false, - zoomLock: false, - zoomOnMouseWheel: true, - moveOnMouseMove: true, - moveOnMouseWheel: false, - preventDefaultMouseMove: true - }); - return InsideZoomModel; - }(DataZoomModel); - - var inner$k = makeInner(); - function setViewInfoToCoordSysRecord(api, dataZoomModel, getRange) { - inner$k(api).coordSysRecordMap.each(function (coordSysRecord) { - var dzInfo = coordSysRecord.dataZoomInfoMap.get(dataZoomModel.uid); - - if (dzInfo) { - dzInfo.getRange = getRange; - } - }); - } - function disposeCoordSysRecordIfNeeded(api, dataZoomModel) { - var coordSysRecordMap = inner$k(api).coordSysRecordMap; - var coordSysKeyArr = coordSysRecordMap.keys(); - - for (var i = 0; i < coordSysKeyArr.length; i++) { - var coordSysKey = coordSysKeyArr[i]; - var coordSysRecord = coordSysRecordMap.get(coordSysKey); - var dataZoomInfoMap = coordSysRecord.dataZoomInfoMap; - - if (dataZoomInfoMap) { - var dzUid = dataZoomModel.uid; - var dzInfo = dataZoomInfoMap.get(dzUid); - - if (dzInfo) { - dataZoomInfoMap.removeKey(dzUid); - - if (!dataZoomInfoMap.keys().length) { - disposeCoordSysRecord(coordSysRecordMap, coordSysRecord); - } - } - } - } - } - - function disposeCoordSysRecord(coordSysRecordMap, coordSysRecord) { - if (coordSysRecord) { - coordSysRecordMap.removeKey(coordSysRecord.model.uid); - var controller = coordSysRecord.controller; - controller && controller.dispose(); - } - } - - function createCoordSysRecord(api, coordSysModel) { - // These init props will never change after record created. - var coordSysRecord = { - model: coordSysModel, - containsPoint: curry(containsPoint, coordSysModel), - dispatchAction: curry(dispatchAction$1, api), - dataZoomInfoMap: null, - controller: null - }; // Must not do anything depends on coordSysRecord outside the event handler here, - // because coordSysRecord not completed yet. - - var controller = coordSysRecord.controller = new RoamController(api.getZr()); - each(['pan', 'zoom', 'scrollMove'], function (eventName) { - controller.on(eventName, function (event) { - var batch = []; - coordSysRecord.dataZoomInfoMap.each(function (dzInfo) { - // Check whether the behaviors (zoomOnMouseWheel, moveOnMouseMove, - // moveOnMouseWheel, ...) enabled. - if (!event.isAvailableBehavior(dzInfo.model.option)) { - return; - } - - var method = (dzInfo.getRange || {})[eventName]; - var range = method && method(dzInfo.dzReferCoordSysInfo, coordSysRecord.model.mainType, coordSysRecord.controller, event); - !dzInfo.model.get('disabled', true) && range && batch.push({ - dataZoomId: dzInfo.model.id, - start: range[0], - end: range[1] - }); - }); - batch.length && coordSysRecord.dispatchAction(batch); - }); - }); - return coordSysRecord; - } - /** - * This action will be throttled. - */ - - - function dispatchAction$1(api, batch) { - if (!api.isDisposed()) { - api.dispatchAction({ - type: 'dataZoom', - animation: { - easing: 'cubicOut', - duration: 100 - }, - batch: batch - }); - } - } - - function containsPoint(coordSysModel, e, x, y) { - return coordSysModel.coordinateSystem.containPoint([x, y]); - } - /** - * Merge roamController settings when multiple dataZooms share one roamController. - */ - - - function mergeControllerParams(dataZoomInfoMap) { - var controlType; // DO NOT use reserved word (true, false, undefined) as key literally. Even if encapsulated - // as string, it is probably revert to reserved word by compress tool. See #7411. - - var prefix = 'type_'; - var typePriority = { - 'type_true': 2, - 'type_move': 1, - 'type_false': 0, - 'type_undefined': -1 - }; - var preventDefaultMouseMove = true; - dataZoomInfoMap.each(function (dataZoomInfo) { - var dataZoomModel = dataZoomInfo.model; - var oneType = dataZoomModel.get('disabled', true) ? false : dataZoomModel.get('zoomLock', true) ? 'move' : true; - - if (typePriority[prefix + oneType] > typePriority[prefix + controlType]) { - controlType = oneType; - } // Prevent default move event by default. If one false, do not prevent. Otherwise - // users may be confused why it does not work when multiple insideZooms exist. - - - preventDefaultMouseMove = preventDefaultMouseMove && dataZoomModel.get('preventDefaultMouseMove', true); - }); - return { - controlType: controlType, - opt: { - // RoamController will enable all of these functionalities, - // and the final behavior is determined by its event listener - // provided by each inside zoom. - zoomOnMouseWheel: true, - moveOnMouseMove: true, - moveOnMouseWheel: true, - preventDefaultMouseMove: !!preventDefaultMouseMove - } - }; - } - - function installDataZoomRoamProcessor(registers) { - registers.registerProcessor(registers.PRIORITY.PROCESSOR.FILTER, function (ecModel, api) { - var apiInner = inner$k(api); - var coordSysRecordMap = apiInner.coordSysRecordMap || (apiInner.coordSysRecordMap = createHashMap()); - coordSysRecordMap.each(function (coordSysRecord) { - // `coordSysRecordMap` always exists (because it holds the `roam controller`, which should - // better not re-create each time), but clear `dataZoomInfoMap` each round of the workflow. - coordSysRecord.dataZoomInfoMap = null; - }); - ecModel.eachComponent({ - mainType: 'dataZoom', - subType: 'inside' - }, function (dataZoomModel) { - var dzReferCoordSysWrap = collectReferCoordSysModelInfo(dataZoomModel); - each(dzReferCoordSysWrap.infoList, function (dzCoordSysInfo) { - var coordSysUid = dzCoordSysInfo.model.uid; - var coordSysRecord = coordSysRecordMap.get(coordSysUid) || coordSysRecordMap.set(coordSysUid, createCoordSysRecord(api, dzCoordSysInfo.model)); - var dataZoomInfoMap = coordSysRecord.dataZoomInfoMap || (coordSysRecord.dataZoomInfoMap = createHashMap()); // Notice these props might be changed each time for a single dataZoomModel. - - dataZoomInfoMap.set(dataZoomModel.uid, { - dzReferCoordSysInfo: dzCoordSysInfo, - model: dataZoomModel, - getRange: null - }); - }); - }); // (1) Merge dataZoom settings for each coord sys and set to the roam controller. - // (2) Clear coord sys if not refered by any dataZoom. - - coordSysRecordMap.each(function (coordSysRecord) { - var controller = coordSysRecord.controller; - var firstDzInfo; - var dataZoomInfoMap = coordSysRecord.dataZoomInfoMap; - - if (dataZoomInfoMap) { - var firstDzKey = dataZoomInfoMap.keys()[0]; - - if (firstDzKey != null) { - firstDzInfo = dataZoomInfoMap.get(firstDzKey); - } - } - - if (!firstDzInfo) { - disposeCoordSysRecord(coordSysRecordMap, coordSysRecord); - return; - } - - var controllerParams = mergeControllerParams(dataZoomInfoMap); - controller.enable(controllerParams.controlType, controllerParams.opt); - controller.setPointerChecker(coordSysRecord.containsPoint); - createOrUpdate(coordSysRecord, 'dispatchAction', firstDzInfo.model.get('throttle', true), 'fixRate'); - }); - }); - } - - var InsideZoomView = - /** @class */ - function (_super) { - __extends(InsideZoomView, _super); - - function InsideZoomView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'dataZoom.inside'; - return _this; - } - - InsideZoomView.prototype.render = function (dataZoomModel, ecModel, api) { - _super.prototype.render.apply(this, arguments); - - if (dataZoomModel.noTarget()) { - this._clear(); - - return; - } // Hence the `throttle` util ensures to preserve command order, - // here simply updating range all the time will not cause missing - // any of the the roam change. - - - this.range = dataZoomModel.getPercentRange(); // Reset controllers. - - setViewInfoToCoordSysRecord(api, dataZoomModel, { - pan: bind(getRangeHandlers.pan, this), - zoom: bind(getRangeHandlers.zoom, this), - scrollMove: bind(getRangeHandlers.scrollMove, this) - }); - }; - - InsideZoomView.prototype.dispose = function () { - this._clear(); - - _super.prototype.dispose.apply(this, arguments); - }; - - InsideZoomView.prototype._clear = function () { - disposeCoordSysRecordIfNeeded(this.api, this.dataZoomModel); - this.range = null; - }; - - InsideZoomView.type = 'dataZoom.inside'; - return InsideZoomView; - }(DataZoomView); - - var getRangeHandlers = { - zoom: function (coordSysInfo, coordSysMainType, controller, e) { - var lastRange = this.range; - var range = lastRange.slice(); // Calculate transform by the first axis. - - var axisModel = coordSysInfo.axisModels[0]; - - if (!axisModel) { - return; - } - - var directionInfo = getDirectionInfo[coordSysMainType](null, [e.originX, e.originY], axisModel, controller, coordSysInfo); - var percentPoint = (directionInfo.signal > 0 ? directionInfo.pixelStart + directionInfo.pixelLength - directionInfo.pixel : directionInfo.pixel - directionInfo.pixelStart) / directionInfo.pixelLength * (range[1] - range[0]) + range[0]; - var scale = Math.max(1 / e.scale, 0); - range[0] = (range[0] - percentPoint) * scale + percentPoint; - range[1] = (range[1] - percentPoint) * scale + percentPoint; // Restrict range. - - var minMaxSpan = this.dataZoomModel.findRepresentativeAxisProxy().getMinMaxSpan(); - sliderMove(0, range, [0, 100], 0, minMaxSpan.minSpan, minMaxSpan.maxSpan); - this.range = range; - - if (lastRange[0] !== range[0] || lastRange[1] !== range[1]) { - return range; - } - }, - pan: makeMover(function (range, axisModel, coordSysInfo, coordSysMainType, controller, e) { - var directionInfo = getDirectionInfo[coordSysMainType]([e.oldX, e.oldY], [e.newX, e.newY], axisModel, controller, coordSysInfo); - return directionInfo.signal * (range[1] - range[0]) * directionInfo.pixel / directionInfo.pixelLength; - }), - scrollMove: makeMover(function (range, axisModel, coordSysInfo, coordSysMainType, controller, e) { - var directionInfo = getDirectionInfo[coordSysMainType]([0, 0], [e.scrollDelta, e.scrollDelta], axisModel, controller, coordSysInfo); - return directionInfo.signal * (range[1] - range[0]) * e.scrollDelta; - }) - }; - - function makeMover(getPercentDelta) { - return function (coordSysInfo, coordSysMainType, controller, e) { - var lastRange = this.range; - var range = lastRange.slice(); // Calculate transform by the first axis. - - var axisModel = coordSysInfo.axisModels[0]; - - if (!axisModel) { - return; - } - - var percentDelta = getPercentDelta(range, axisModel, coordSysInfo, coordSysMainType, controller, e); - sliderMove(percentDelta, range, [0, 100], 'all'); - this.range = range; - - if (lastRange[0] !== range[0] || lastRange[1] !== range[1]) { - return range; - } - }; - } - - var getDirectionInfo = { - grid: function (oldPoint, newPoint, axisModel, controller, coordSysInfo) { - var axis = axisModel.axis; - var ret = {}; - var rect = coordSysInfo.model.coordinateSystem.getRect(); - oldPoint = oldPoint || [0, 0]; - - if (axis.dim === 'x') { - ret.pixel = newPoint[0] - oldPoint[0]; - ret.pixelLength = rect.width; - ret.pixelStart = rect.x; - ret.signal = axis.inverse ? 1 : -1; - } else { - // axis.dim === 'y' - ret.pixel = newPoint[1] - oldPoint[1]; - ret.pixelLength = rect.height; - ret.pixelStart = rect.y; - ret.signal = axis.inverse ? -1 : 1; - } - - return ret; - }, - polar: function (oldPoint, newPoint, axisModel, controller, coordSysInfo) { - var axis = axisModel.axis; - var ret = {}; - var polar = coordSysInfo.model.coordinateSystem; - var radiusExtent = polar.getRadiusAxis().getExtent(); - var angleExtent = polar.getAngleAxis().getExtent(); - oldPoint = oldPoint ? polar.pointToCoord(oldPoint) : [0, 0]; - newPoint = polar.pointToCoord(newPoint); - - if (axisModel.mainType === 'radiusAxis') { - ret.pixel = newPoint[0] - oldPoint[0]; // ret.pixelLength = Math.abs(radiusExtent[1] - radiusExtent[0]); - // ret.pixelStart = Math.min(radiusExtent[0], radiusExtent[1]); - - ret.pixelLength = radiusExtent[1] - radiusExtent[0]; - ret.pixelStart = radiusExtent[0]; - ret.signal = axis.inverse ? 1 : -1; - } else { - // 'angleAxis' - ret.pixel = newPoint[1] - oldPoint[1]; // ret.pixelLength = Math.abs(angleExtent[1] - angleExtent[0]); - // ret.pixelStart = Math.min(angleExtent[0], angleExtent[1]); - - ret.pixelLength = angleExtent[1] - angleExtent[0]; - ret.pixelStart = angleExtent[0]; - ret.signal = axis.inverse ? -1 : 1; - } - - return ret; - }, - singleAxis: function (oldPoint, newPoint, axisModel, controller, coordSysInfo) { - var axis = axisModel.axis; - var rect = coordSysInfo.model.coordinateSystem.getRect(); - var ret = {}; - oldPoint = oldPoint || [0, 0]; - - if (axis.orient === 'horizontal') { - ret.pixel = newPoint[0] - oldPoint[0]; - ret.pixelLength = rect.width; - ret.pixelStart = rect.x; - ret.signal = axis.inverse ? 1 : -1; - } else { - // 'vertical' - ret.pixel = newPoint[1] - oldPoint[1]; - ret.pixelLength = rect.height; - ret.pixelStart = rect.y; - ret.signal = axis.inverse ? -1 : 1; - } - - return ret; - } - }; - - function install$K(registers) { - installCommon(registers); - registers.registerComponentModel(InsideZoomModel); - registers.registerComponentView(InsideZoomView); - installDataZoomRoamProcessor(registers); - } - - var SliderZoomModel = - /** @class */ - function (_super) { - __extends(SliderZoomModel, _super); - - function SliderZoomModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SliderZoomModel.type; - return _this; - } - - SliderZoomModel.type = 'dataZoom.slider'; - SliderZoomModel.layoutMode = 'box'; - SliderZoomModel.defaultOption = inheritDefaultOption(DataZoomModel.defaultOption, { - show: true, - // deault value can only be drived in view stage. - right: 'ph', - top: 'ph', - width: 'ph', - height: 'ph', - left: null, - bottom: null, - borderColor: '#d2dbee', - borderRadius: 3, - backgroundColor: 'rgba(47,69,84,0)', - // dataBackgroundColor: '#ddd', - dataBackground: { - lineStyle: { - color: '#d2dbee', - width: 0.5 - }, - areaStyle: { - color: '#d2dbee', - opacity: 0.2 - } - }, - selectedDataBackground: { - lineStyle: { - color: '#8fb0f7', - width: 0.5 - }, - areaStyle: { - color: '#8fb0f7', - opacity: 0.2 - } - }, - // Color of selected window. - fillerColor: 'rgba(135,175,274,0.2)', - handleIcon: 'path://M-9.35,34.56V42m0-40V9.5m-2,0h4a2,2,0,0,1,2,2v21a2,2,0,0,1-2,2h-4a2,2,0,0,1-2-2v-21A2,2,0,0,1-11.35,9.5Z', - // Percent of the slider height - handleSize: '100%', - handleStyle: { - color: '#fff', - borderColor: '#ACB8D1' - }, - moveHandleSize: 7, - moveHandleIcon: 'path://M-320.9-50L-320.9-50c18.1,0,27.1,9,27.1,27.1V85.7c0,18.1-9,27.1-27.1,27.1l0,0c-18.1,0-27.1-9-27.1-27.1V-22.9C-348-41-339-50-320.9-50z M-212.3-50L-212.3-50c18.1,0,27.1,9,27.1,27.1V85.7c0,18.1-9,27.1-27.1,27.1l0,0c-18.1,0-27.1-9-27.1-27.1V-22.9C-239.4-41-230.4-50-212.3-50z M-103.7-50L-103.7-50c18.1,0,27.1,9,27.1,27.1V85.7c0,18.1-9,27.1-27.1,27.1l0,0c-18.1,0-27.1-9-27.1-27.1V-22.9C-130.9-41-121.8-50-103.7-50z', - moveHandleStyle: { - color: '#D2DBEE', - opacity: 0.7 - }, - showDetail: true, - showDataShadow: 'auto', - realtime: true, - zoomLock: false, - textStyle: { - color: '#6E7079' - }, - brushSelect: true, - brushStyle: { - color: 'rgba(135,175,274,0.15)' - }, - emphasis: { - handleStyle: { - borderColor: '#8FB0F7' - }, - moveHandleStyle: { - color: '#8FB0F7' - } - } - }); - return SliderZoomModel; - }(DataZoomModel); - - var Rect$2 = Rect; // Constants - - var DEFAULT_LOCATION_EDGE_GAP = 7; - var DEFAULT_FRAME_BORDER_WIDTH = 1; - var DEFAULT_FILLER_SIZE = 30; - var DEFAULT_MOVE_HANDLE_SIZE = 7; - var HORIZONTAL = 'horizontal'; - var VERTICAL = 'vertical'; - var LABEL_GAP = 5; - var SHOW_DATA_SHADOW_SERIES_TYPE = ['line', 'bar', 'candlestick', 'scatter']; - var REALTIME_ANIMATION_CONFIG = { - easing: 'cubicOut', - duration: 100, - delay: 0 - }; - - var SliderZoomView = - /** @class */ - function (_super) { - __extends(SliderZoomView, _super); - - function SliderZoomView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = SliderZoomView.type; - _this._displayables = {}; - return _this; - } - - SliderZoomView.prototype.init = function (ecModel, api) { - this.api = api; // A unique handler for each dataZoom component - - this._onBrush = bind(this._onBrush, this); - this._onBrushEnd = bind(this._onBrushEnd, this); - }; - - SliderZoomView.prototype.render = function (dataZoomModel, ecModel, api, payload) { - _super.prototype.render.apply(this, arguments); - - createOrUpdate(this, '_dispatchZoomAction', dataZoomModel.get('throttle'), 'fixRate'); - this._orient = dataZoomModel.getOrient(); - - if (dataZoomModel.get('show') === false) { - this.group.removeAll(); - return; - } - - if (dataZoomModel.noTarget()) { - this._clear(); - - this.group.removeAll(); - return; - } // Notice: this._resetInterval() should not be executed when payload.type - // is 'dataZoom', origin this._range should be maintained, otherwise 'pan' - // or 'zoom' info will be missed because of 'throttle' of this.dispatchAction, - - - if (!payload || payload.type !== 'dataZoom' || payload.from !== this.uid) { - this._buildView(); - } - - this._updateView(); - }; - - SliderZoomView.prototype.dispose = function () { - this._clear(); - - _super.prototype.dispose.apply(this, arguments); - }; - - SliderZoomView.prototype._clear = function () { - clear(this, '_dispatchZoomAction'); - var zr = this.api.getZr(); - zr.off('mousemove', this._onBrush); - zr.off('mouseup', this._onBrushEnd); - }; - - SliderZoomView.prototype._buildView = function () { - var thisGroup = this.group; - thisGroup.removeAll(); - this._brushing = false; - this._displayables.brushRect = null; - - this._resetLocation(); - - this._resetInterval(); - - var barGroup = this._displayables.sliderGroup = new Group(); - - this._renderBackground(); - - this._renderHandle(); - - this._renderDataShadow(); - - thisGroup.add(barGroup); - - this._positionGroup(); - }; - - SliderZoomView.prototype._resetLocation = function () { - var dataZoomModel = this.dataZoomModel; - var api = this.api; - var showMoveHandle = dataZoomModel.get('brushSelect'); - var moveHandleSize = showMoveHandle ? DEFAULT_MOVE_HANDLE_SIZE : 0; // If some of x/y/width/height are not specified, - // auto-adapt according to target grid. - - var coordRect = this._findCoordRect(); - - var ecSize = { - width: api.getWidth(), - height: api.getHeight() - }; // Default align by coordinate system rect. - - var positionInfo = this._orient === HORIZONTAL ? { - // Why using 'right', because right should be used in vertical, - // and it is better to be consistent for dealing with position param merge. - right: ecSize.width - coordRect.x - coordRect.width, - top: ecSize.height - DEFAULT_FILLER_SIZE - DEFAULT_LOCATION_EDGE_GAP - moveHandleSize, - width: coordRect.width, - height: DEFAULT_FILLER_SIZE - } : { - right: DEFAULT_LOCATION_EDGE_GAP, - top: coordRect.y, - width: DEFAULT_FILLER_SIZE, - height: coordRect.height - }; // Do not write back to option and replace value 'ph', because - // the 'ph' value should be recalculated when resize. - - var layoutParams = getLayoutParams(dataZoomModel.option); // Replace the placeholder value. - - each(['right', 'top', 'width', 'height'], function (name) { - if (layoutParams[name] === 'ph') { - layoutParams[name] = positionInfo[name]; - } - }); - var layoutRect = getLayoutRect(layoutParams, ecSize); - this._location = { - x: layoutRect.x, - y: layoutRect.y - }; - this._size = [layoutRect.width, layoutRect.height]; - this._orient === VERTICAL && this._size.reverse(); - }; - - SliderZoomView.prototype._positionGroup = function () { - var thisGroup = this.group; - var location = this._location; - var orient = this._orient; // Just use the first axis to determine mapping. - - var targetAxisModel = this.dataZoomModel.getFirstTargetAxisModel(); - var inverse = targetAxisModel && targetAxisModel.get('inverse'); - var sliderGroup = this._displayables.sliderGroup; - var otherAxisInverse = (this._dataShadowInfo || {}).otherAxisInverse; // Transform barGroup. - - sliderGroup.attr(orient === HORIZONTAL && !inverse ? { - scaleY: otherAxisInverse ? 1 : -1, - scaleX: 1 - } : orient === HORIZONTAL && inverse ? { - scaleY: otherAxisInverse ? 1 : -1, - scaleX: -1 - } : orient === VERTICAL && !inverse ? { - scaleY: otherAxisInverse ? -1 : 1, - scaleX: 1, - rotation: Math.PI / 2 - } // Don't use Math.PI, considering shadow direction. - : { - scaleY: otherAxisInverse ? -1 : 1, - scaleX: -1, - rotation: Math.PI / 2 - }); // Position barGroup - - var rect = thisGroup.getBoundingRect([sliderGroup]); - thisGroup.x = location.x - rect.x; - thisGroup.y = location.y - rect.y; - thisGroup.markRedraw(); - }; - - SliderZoomView.prototype._getViewExtent = function () { - return [0, this._size[0]]; - }; - - SliderZoomView.prototype._renderBackground = function () { - var dataZoomModel = this.dataZoomModel; - var size = this._size; - var barGroup = this._displayables.sliderGroup; - var brushSelect = dataZoomModel.get('brushSelect'); - barGroup.add(new Rect$2({ - silent: true, - shape: { - x: 0, - y: 0, - width: size[0], - height: size[1] - }, - style: { - fill: dataZoomModel.get('backgroundColor') - }, - z2: -40 - })); // Click panel, over shadow, below handles. - - var clickPanel = new Rect$2({ - shape: { - x: 0, - y: 0, - width: size[0], - height: size[1] - }, - style: { - fill: 'transparent' - }, - z2: 0, - onclick: bind(this._onClickPanel, this) - }); - var zr = this.api.getZr(); - - if (brushSelect) { - clickPanel.on('mousedown', this._onBrushStart, this); - clickPanel.cursor = 'crosshair'; - zr.on('mousemove', this._onBrush); - zr.on('mouseup', this._onBrushEnd); - } else { - zr.off('mousemove', this._onBrush); - zr.off('mouseup', this._onBrushEnd); - } - - barGroup.add(clickPanel); - }; - - SliderZoomView.prototype._renderDataShadow = function () { - var info = this._dataShadowInfo = this._prepareDataShadowInfo(); - - this._displayables.dataShadowSegs = []; - - if (!info) { - return; - } - - var size = this._size; - var oldSize = this._shadowSize || []; - var seriesModel = info.series; - var data = seriesModel.getRawData(); - var candlestickDim = seriesModel.getShadowDim && seriesModel.getShadowDim(); - var otherDim = candlestickDim && data.getDimensionInfo(candlestickDim) ? seriesModel.getShadowDim() // @see candlestick - : info.otherDim; - - if (otherDim == null) { - return; - } - - var polygonPts = this._shadowPolygonPts; - var polylinePts = this._shadowPolylinePts; // Not re-render if data doesn't change. - - if (data !== this._shadowData || otherDim !== this._shadowDim || size[0] !== oldSize[0] || size[1] !== oldSize[1]) { - var otherDataExtent_1 = data.getDataExtent(otherDim); // Nice extent. - - var otherOffset = (otherDataExtent_1[1] - otherDataExtent_1[0]) * 0.3; - otherDataExtent_1 = [otherDataExtent_1[0] - otherOffset, otherDataExtent_1[1] + otherOffset]; - var otherShadowExtent_1 = [0, size[1]]; - var thisShadowExtent = [0, size[0]]; - var areaPoints_1 = [[size[0], 0], [0, 0]]; - var linePoints_1 = []; - var step_1 = thisShadowExtent[1] / (data.count() - 1); - var thisCoord_1 = 0; // Optimize for large data shadow - - var stride_1 = Math.round(data.count() / size[0]); - var lastIsEmpty_1; - data.each([otherDim], function (value, index) { - if (stride_1 > 0 && index % stride_1) { - thisCoord_1 += step_1; - return; - } // FIXME - // Should consider axis.min/axis.max when drawing dataShadow. - // FIXME - // 应该使用统一的空判断?还是在list里进行空判断? - - - var isEmpty = value == null || isNaN(value) || value === ''; // See #4235. - - var otherCoord = isEmpty ? 0 : linearMap(value, otherDataExtent_1, otherShadowExtent_1, true); // Attempt to draw data shadow precisely when there are empty value. - - if (isEmpty && !lastIsEmpty_1 && index) { - areaPoints_1.push([areaPoints_1[areaPoints_1.length - 1][0], 0]); - linePoints_1.push([linePoints_1[linePoints_1.length - 1][0], 0]); - } else if (!isEmpty && lastIsEmpty_1) { - areaPoints_1.push([thisCoord_1, 0]); - linePoints_1.push([thisCoord_1, 0]); - } - - areaPoints_1.push([thisCoord_1, otherCoord]); - linePoints_1.push([thisCoord_1, otherCoord]); - thisCoord_1 += step_1; - lastIsEmpty_1 = isEmpty; - }); - polygonPts = this._shadowPolygonPts = areaPoints_1; - polylinePts = this._shadowPolylinePts = linePoints_1; - } - - this._shadowData = data; - this._shadowDim = otherDim; - this._shadowSize = [size[0], size[1]]; - var dataZoomModel = this.dataZoomModel; - - function createDataShadowGroup(isSelectedArea) { - var model = dataZoomModel.getModel(isSelectedArea ? 'selectedDataBackground' : 'dataBackground'); - var group = new Group(); - var polygon = new Polygon({ - shape: { - points: polygonPts - }, - segmentIgnoreThreshold: 1, - style: model.getModel('areaStyle').getAreaStyle(), - silent: true, - z2: -20 - }); - var polyline = new Polyline({ - shape: { - points: polylinePts - }, - segmentIgnoreThreshold: 1, - style: model.getModel('lineStyle').getLineStyle(), - silent: true, - z2: -19 - }); - group.add(polygon); - group.add(polyline); - return group; - } // let dataBackgroundModel = dataZoomModel.getModel('dataBackground'); - - - for (var i = 0; i < 3; i++) { - var group = createDataShadowGroup(i === 1); - - this._displayables.sliderGroup.add(group); - - this._displayables.dataShadowSegs.push(group); - } - }; - - SliderZoomView.prototype._prepareDataShadowInfo = function () { - var dataZoomModel = this.dataZoomModel; - var showDataShadow = dataZoomModel.get('showDataShadow'); - - if (showDataShadow === false) { - return; - } // Find a representative series. - - - var result; - var ecModel = this.ecModel; - dataZoomModel.eachTargetAxis(function (axisDim, axisIndex) { - var seriesModels = dataZoomModel.getAxisProxy(axisDim, axisIndex).getTargetSeriesModels(); - each(seriesModels, function (seriesModel) { - if (result) { - return; - } - - if (showDataShadow !== true && indexOf(SHOW_DATA_SHADOW_SERIES_TYPE, seriesModel.get('type')) < 0) { - return; - } - - var thisAxis = ecModel.getComponent(getAxisMainType(axisDim), axisIndex).axis; - var otherDim = getOtherDim(axisDim); - var otherAxisInverse; - var coordSys = seriesModel.coordinateSystem; - - if (otherDim != null && coordSys.getOtherAxis) { - otherAxisInverse = coordSys.getOtherAxis(thisAxis).inverse; - } - - otherDim = seriesModel.getData().mapDimension(otherDim); - result = { - thisAxis: thisAxis, - series: seriesModel, - thisDim: axisDim, - otherDim: otherDim, - otherAxisInverse: otherAxisInverse - }; - }, this); - }, this); - return result; - }; - - SliderZoomView.prototype._renderHandle = function () { - var thisGroup = this.group; - var displayables = this._displayables; - var handles = displayables.handles = [null, null]; - var handleLabels = displayables.handleLabels = [null, null]; - var sliderGroup = this._displayables.sliderGroup; - var size = this._size; - var dataZoomModel = this.dataZoomModel; - var api = this.api; - var borderRadius = dataZoomModel.get('borderRadius') || 0; - var brushSelect = dataZoomModel.get('brushSelect'); - var filler = displayables.filler = new Rect$2({ - silent: brushSelect, - style: { - fill: dataZoomModel.get('fillerColor') - }, - textConfig: { - position: 'inside' - } - }); - sliderGroup.add(filler); // Frame border. - - sliderGroup.add(new Rect$2({ - silent: true, - subPixelOptimize: true, - shape: { - x: 0, - y: 0, - width: size[0], - height: size[1], - r: borderRadius - }, - style: { - // deprecated option - stroke: dataZoomModel.get('dataBackgroundColor') || dataZoomModel.get('borderColor'), - lineWidth: DEFAULT_FRAME_BORDER_WIDTH, - fill: 'rgba(0,0,0,0)' - } - })); // Left and right handle to resize - - each([0, 1], function (handleIndex) { - var iconStr = dataZoomModel.get('handleIcon'); - - if (!symbolBuildProxies[iconStr] && iconStr.indexOf('path://') < 0 && iconStr.indexOf('image://') < 0) { - // Compatitable with the old icon parsers. Which can use a path string without path:// - iconStr = 'path://' + iconStr; - - if ("development" !== 'production') { - deprecateLog('handleIcon now needs \'path://\' prefix when using a path string'); - } - } - - var path = createSymbol(iconStr, -1, 0, 2, 2, null, true); - path.attr({ - cursor: getCursor(this._orient), - draggable: true, - drift: bind(this._onDragMove, this, handleIndex), - ondragend: bind(this._onDragEnd, this), - onmouseover: bind(this._showDataInfo, this, true), - onmouseout: bind(this._showDataInfo, this, false), - z2: 5 - }); - var bRect = path.getBoundingRect(); - var handleSize = dataZoomModel.get('handleSize'); - this._handleHeight = parsePercent$1(handleSize, this._size[1]); - this._handleWidth = bRect.width / bRect.height * this._handleHeight; - path.setStyle(dataZoomModel.getModel('handleStyle').getItemStyle()); - path.style.strokeNoScale = true; - path.rectHover = true; - path.ensureState('emphasis').style = dataZoomModel.getModel(['emphasis', 'handleStyle']).getItemStyle(); - enableHoverEmphasis(path); - var handleColor = dataZoomModel.get('handleColor'); // deprecated option - // Compatitable with previous version - - if (handleColor != null) { - path.style.fill = handleColor; - } - - sliderGroup.add(handles[handleIndex] = path); - var textStyleModel = dataZoomModel.getModel('textStyle'); - thisGroup.add(handleLabels[handleIndex] = new ZRText({ - silent: true, - invisible: true, - style: createTextStyle(textStyleModel, { - x: 0, - y: 0, - text: '', - verticalAlign: 'middle', - align: 'center', - fill: textStyleModel.getTextColor(), - font: textStyleModel.getFont() - }), - z2: 10 - })); - }, this); // Handle to move. Only visible when brushSelect is set true. - - var actualMoveZone = filler; - - if (brushSelect) { - var moveHandleHeight = parsePercent$1(dataZoomModel.get('moveHandleSize'), size[1]); - var moveHandle_1 = displayables.moveHandle = new Rect({ - style: dataZoomModel.getModel('moveHandleStyle').getItemStyle(), - silent: true, - shape: { - r: [0, 0, 2, 2], - y: size[1] - 0.5, - height: moveHandleHeight - } - }); - var iconSize = moveHandleHeight * 0.8; - var moveHandleIcon = displayables.moveHandleIcon = createSymbol(dataZoomModel.get('moveHandleIcon'), -iconSize / 2, -iconSize / 2, iconSize, iconSize, '#fff', true); - moveHandleIcon.silent = true; - moveHandleIcon.y = size[1] + moveHandleHeight / 2 - 0.5; - moveHandle_1.ensureState('emphasis').style = dataZoomModel.getModel(['emphasis', 'moveHandleStyle']).getItemStyle(); - var moveZoneExpandSize = Math.min(size[1] / 2, Math.max(moveHandleHeight, 10)); - actualMoveZone = displayables.moveZone = new Rect({ - invisible: true, - shape: { - y: size[1] - moveZoneExpandSize, - height: moveHandleHeight + moveZoneExpandSize - } - }); - actualMoveZone.on('mouseover', function () { - api.enterEmphasis(moveHandle_1); - }).on('mouseout', function () { - api.leaveEmphasis(moveHandle_1); - }); - sliderGroup.add(moveHandle_1); - sliderGroup.add(moveHandleIcon); - sliderGroup.add(actualMoveZone); - } - - actualMoveZone.attr({ - draggable: true, - cursor: getCursor(this._orient), - drift: bind(this._onDragMove, this, 'all'), - ondragstart: bind(this._showDataInfo, this, true), - ondragend: bind(this._onDragEnd, this), - onmouseover: bind(this._showDataInfo, this, true), - onmouseout: bind(this._showDataInfo, this, false) - }); - }; - - SliderZoomView.prototype._resetInterval = function () { - var range = this._range = this.dataZoomModel.getPercentRange(); - - var viewExtent = this._getViewExtent(); - - this._handleEnds = [linearMap(range[0], [0, 100], viewExtent, true), linearMap(range[1], [0, 100], viewExtent, true)]; - }; - - SliderZoomView.prototype._updateInterval = function (handleIndex, delta) { - var dataZoomModel = this.dataZoomModel; - var handleEnds = this._handleEnds; - - var viewExtend = this._getViewExtent(); - - var minMaxSpan = dataZoomModel.findRepresentativeAxisProxy().getMinMaxSpan(); - var percentExtent = [0, 100]; - sliderMove(delta, handleEnds, viewExtend, dataZoomModel.get('zoomLock') ? 'all' : handleIndex, minMaxSpan.minSpan != null ? linearMap(minMaxSpan.minSpan, percentExtent, viewExtend, true) : null, minMaxSpan.maxSpan != null ? linearMap(minMaxSpan.maxSpan, percentExtent, viewExtend, true) : null); - var lastRange = this._range; - var range = this._range = asc([linearMap(handleEnds[0], viewExtend, percentExtent, true), linearMap(handleEnds[1], viewExtend, percentExtent, true)]); - return !lastRange || lastRange[0] !== range[0] || lastRange[1] !== range[1]; - }; - - SliderZoomView.prototype._updateView = function (nonRealtime) { - var displaybles = this._displayables; - var handleEnds = this._handleEnds; - var handleInterval = asc(handleEnds.slice()); - var size = this._size; - each([0, 1], function (handleIndex) { - // Handles - var handle = displaybles.handles[handleIndex]; - var handleHeight = this._handleHeight; - handle.attr({ - scaleX: handleHeight / 2, - scaleY: handleHeight / 2, - // This is a trick, by adding an extra tiny offset to let the default handle's end point align to the drag window. - // NOTE: It may affect some custom shapes a bit. But we prefer to have better result by default. - x: handleEnds[handleIndex] + (handleIndex ? -1 : 1), - y: size[1] / 2 - handleHeight / 2 - }); - }, this); // Filler - - displaybles.filler.setShape({ - x: handleInterval[0], - y: 0, - width: handleInterval[1] - handleInterval[0], - height: size[1] - }); - var viewExtent = { - x: handleInterval[0], - width: handleInterval[1] - handleInterval[0] - }; // Move handle - - if (displaybles.moveHandle) { - displaybles.moveHandle.setShape(viewExtent); - displaybles.moveZone.setShape(viewExtent); // Force update path on the invisible object - - displaybles.moveZone.getBoundingRect(); - displaybles.moveHandleIcon && displaybles.moveHandleIcon.attr('x', viewExtent.x + viewExtent.width / 2); - } // update clip path of shadow. - - - var dataShadowSegs = displaybles.dataShadowSegs; - var segIntervals = [0, handleInterval[0], handleInterval[1], size[0]]; - - for (var i = 0; i < dataShadowSegs.length; i++) { - var segGroup = dataShadowSegs[i]; - var clipPath = segGroup.getClipPath(); - - if (!clipPath) { - clipPath = new Rect(); - segGroup.setClipPath(clipPath); - } - - clipPath.setShape({ - x: segIntervals[i], - y: 0, - width: segIntervals[i + 1] - segIntervals[i], - height: size[1] - }); - } - - this._updateDataInfo(nonRealtime); - }; - - SliderZoomView.prototype._updateDataInfo = function (nonRealtime) { - var dataZoomModel = this.dataZoomModel; - var displaybles = this._displayables; - var handleLabels = displaybles.handleLabels; - var orient = this._orient; - var labelTexts = ['', '']; // FIXME - // date型,支持formatter,autoformatter(ec2 date.getAutoFormatter) - - if (dataZoomModel.get('showDetail')) { - var axisProxy = dataZoomModel.findRepresentativeAxisProxy(); - - if (axisProxy) { - var axis = axisProxy.getAxisModel().axis; - var range = this._range; - var dataInterval = nonRealtime // See #4434, data and axis are not processed and reset yet in non-realtime mode. - ? axisProxy.calculateDataWindow({ - start: range[0], - end: range[1] - }).valueWindow : axisProxy.getDataValueWindow(); - labelTexts = [this._formatLabel(dataInterval[0], axis), this._formatLabel(dataInterval[1], axis)]; - } - } - - var orderedHandleEnds = asc(this._handleEnds.slice()); - setLabel.call(this, 0); - setLabel.call(this, 1); - - function setLabel(handleIndex) { - // Label - // Text should not transform by barGroup. - // Ignore handlers transform - var barTransform = getTransform(displaybles.handles[handleIndex].parent, this.group); - var direction = transformDirection(handleIndex === 0 ? 'right' : 'left', barTransform); - var offset = this._handleWidth / 2 + LABEL_GAP; - var textPoint = applyTransform$1([orderedHandleEnds[handleIndex] + (handleIndex === 0 ? -offset : offset), this._size[1] / 2], barTransform); - handleLabels[handleIndex].setStyle({ - x: textPoint[0], - y: textPoint[1], - verticalAlign: orient === HORIZONTAL ? 'middle' : direction, - align: orient === HORIZONTAL ? direction : 'center', - text: labelTexts[handleIndex] - }); - } - }; - - SliderZoomView.prototype._formatLabel = function (value, axis) { - var dataZoomModel = this.dataZoomModel; - var labelFormatter = dataZoomModel.get('labelFormatter'); - var labelPrecision = dataZoomModel.get('labelPrecision'); - - if (labelPrecision == null || labelPrecision === 'auto') { - labelPrecision = axis.getPixelPrecision(); - } - - var valueStr = value == null || isNaN(value) ? '' // FIXME Glue code - : axis.type === 'category' || axis.type === 'time' ? axis.scale.getLabel({ - value: Math.round(value) - }) // param of toFixed should less then 20. - : value.toFixed(Math.min(labelPrecision, 20)); - return isFunction(labelFormatter) ? labelFormatter(value, valueStr) : isString(labelFormatter) ? labelFormatter.replace('{value}', valueStr) : valueStr; - }; - /** - * @param showOrHide true: show, false: hide - */ - - - SliderZoomView.prototype._showDataInfo = function (showOrHide) { - // Always show when drgging. - showOrHide = this._dragging || showOrHide; - var displayables = this._displayables; - var handleLabels = displayables.handleLabels; - handleLabels[0].attr('invisible', !showOrHide); - handleLabels[1].attr('invisible', !showOrHide); // Highlight move handle - - displayables.moveHandle && this.api[showOrHide ? 'enterEmphasis' : 'leaveEmphasis'](displayables.moveHandle, 1); - }; - - SliderZoomView.prototype._onDragMove = function (handleIndex, dx, dy, event) { - this._dragging = true; // For mobile device, prevent screen slider on the button. - - stop(event.event); // Transform dx, dy to bar coordination. - - var barTransform = this._displayables.sliderGroup.getLocalTransform(); - - var vertex = applyTransform$1([dx, dy], barTransform, true); - - var changed = this._updateInterval(handleIndex, vertex[0]); - - var realtime = this.dataZoomModel.get('realtime'); - - this._updateView(!realtime); // Avoid dispatch dataZoom repeatly but range not changed, - // which cause bad visual effect when progressive enabled. - - - changed && realtime && this._dispatchZoomAction(true); - }; - - SliderZoomView.prototype._onDragEnd = function () { - this._dragging = false; - - this._showDataInfo(false); // While in realtime mode and stream mode, dispatch action when - // drag end will cause the whole view rerender, which is unnecessary. - - - var realtime = this.dataZoomModel.get('realtime'); - !realtime && this._dispatchZoomAction(false); - }; - - SliderZoomView.prototype._onClickPanel = function (e) { - var size = this._size; - - var localPoint = this._displayables.sliderGroup.transformCoordToLocal(e.offsetX, e.offsetY); - - if (localPoint[0] < 0 || localPoint[0] > size[0] || localPoint[1] < 0 || localPoint[1] > size[1]) { - return; - } - - var handleEnds = this._handleEnds; - var center = (handleEnds[0] + handleEnds[1]) / 2; - - var changed = this._updateInterval('all', localPoint[0] - center); - - this._updateView(); - - changed && this._dispatchZoomAction(false); - }; - - SliderZoomView.prototype._onBrushStart = function (e) { - var x = e.offsetX; - var y = e.offsetY; - this._brushStart = new Point(x, y); - this._brushing = true; - this._brushStartTime = +new Date(); // this._updateBrushRect(x, y); - }; - - SliderZoomView.prototype._onBrushEnd = function (e) { - if (!this._brushing) { - return; - } - - var brushRect = this._displayables.brushRect; - this._brushing = false; - - if (!brushRect) { - return; - } - - brushRect.attr('ignore', true); - var brushShape = brushRect.shape; - var brushEndTime = +new Date(); // console.log(brushEndTime - this._brushStartTime); - - if (brushEndTime - this._brushStartTime < 200 && Math.abs(brushShape.width) < 5) { - // Will treat it as a click - return; - } - - var viewExtend = this._getViewExtent(); - - var percentExtent = [0, 100]; - this._range = asc([linearMap(brushShape.x, viewExtend, percentExtent, true), linearMap(brushShape.x + brushShape.width, viewExtend, percentExtent, true)]); - this._handleEnds = [brushShape.x, brushShape.x + brushShape.width]; - - this._updateView(); - - this._dispatchZoomAction(false); - }; - - SliderZoomView.prototype._onBrush = function (e) { - if (this._brushing) { - // For mobile device, prevent screen slider on the button. - stop(e.event); - - this._updateBrushRect(e.offsetX, e.offsetY); - } - }; - - SliderZoomView.prototype._updateBrushRect = function (mouseX, mouseY) { - var displayables = this._displayables; - var dataZoomModel = this.dataZoomModel; - var brushRect = displayables.brushRect; - - if (!brushRect) { - brushRect = displayables.brushRect = new Rect$2({ - silent: true, - style: dataZoomModel.getModel('brushStyle').getItemStyle() - }); - displayables.sliderGroup.add(brushRect); - } - - brushRect.attr('ignore', false); - var brushStart = this._brushStart; - var sliderGroup = this._displayables.sliderGroup; - var endPoint = sliderGroup.transformCoordToLocal(mouseX, mouseY); - var startPoint = sliderGroup.transformCoordToLocal(brushStart.x, brushStart.y); - var size = this._size; - endPoint[0] = Math.max(Math.min(size[0], endPoint[0]), 0); - brushRect.setShape({ - x: startPoint[0], - y: 0, - width: endPoint[0] - startPoint[0], - height: size[1] - }); - }; - /** - * This action will be throttled. - */ - - - SliderZoomView.prototype._dispatchZoomAction = function (realtime) { - var range = this._range; - this.api.dispatchAction({ - type: 'dataZoom', - from: this.uid, - dataZoomId: this.dataZoomModel.id, - animation: realtime ? REALTIME_ANIMATION_CONFIG : null, - start: range[0], - end: range[1] - }); - }; - - SliderZoomView.prototype._findCoordRect = function () { - // Find the grid corresponding to the first axis referred by dataZoom. - var rect; - var coordSysInfoList = collectReferCoordSysModelInfo(this.dataZoomModel).infoList; - - if (!rect && coordSysInfoList.length) { - var coordSys = coordSysInfoList[0].model.coordinateSystem; - rect = coordSys.getRect && coordSys.getRect(); - } - - if (!rect) { - var width = this.api.getWidth(); - var height = this.api.getHeight(); - rect = { - x: width * 0.2, - y: height * 0.2, - width: width * 0.6, - height: height * 0.6 - }; - } - - return rect; - }; - - SliderZoomView.type = 'dataZoom.slider'; - return SliderZoomView; - }(DataZoomView); - - function getOtherDim(thisDim) { - // FIXME - // 这个逻辑和getOtherAxis里一致,但是写在这里是否不好 - var map = { - x: 'y', - y: 'x', - radius: 'angle', - angle: 'radius' - }; - return map[thisDim]; - } - - function getCursor(orient) { - return orient === 'vertical' ? 'ns-resize' : 'ew-resize'; - } - - function install$L(registers) { - registers.registerComponentModel(SliderZoomModel); - registers.registerComponentView(SliderZoomView); - installCommon(registers); - } - - function install$M(registers) { - use(install$K); - use(install$L); // Do not install './dataZoomSelect', - // since it only work for toolbox dataZoom. - } - - var visualDefault = { - /** - * @public - */ - get: function (visualType, key, isCategory) { - var value = clone((defaultOption$1[visualType] || {})[key]); - return isCategory ? isArray(value) ? value[value.length - 1] : value : value; - } - }; - var defaultOption$1 = { - color: { - active: ['#006edd', '#e0ffff'], - inactive: ['rgba(0,0,0,0)'] - }, - colorHue: { - active: [0, 360], - inactive: [0, 0] - }, - colorSaturation: { - active: [0.3, 1], - inactive: [0, 0] - }, - colorLightness: { - active: [0.9, 0.5], - inactive: [0, 0] - }, - colorAlpha: { - active: [0.3, 1], - inactive: [0, 0] - }, - opacity: { - active: [0.3, 1], - inactive: [0, 0] - }, - symbol: { - active: ['circle', 'roundRect', 'diamond'], - inactive: ['none'] - }, - symbolSize: { - active: [10, 50], - inactive: [0, 0] - } - }; - - var mapVisual$1 = VisualMapping.mapVisual; - var eachVisual = VisualMapping.eachVisual; - var isArray$1 = isArray; - var each$d = each; - var asc$2 = asc; - var linearMap$1 = linearMap; - - var VisualMapModel = - /** @class */ - function (_super) { - __extends(VisualMapModel, _super); - - function VisualMapModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = VisualMapModel.type; - _this.stateList = ['inRange', 'outOfRange']; - _this.replacableOptionKeys = ['inRange', 'outOfRange', 'target', 'controller', 'color']; - _this.layoutMode = { - type: 'box', - ignoreSize: true - }; - /** - * [lowerBound, upperBound] - */ - - _this.dataBound = [-Infinity, Infinity]; - _this.targetVisuals = {}; - _this.controllerVisuals = {}; - return _this; - } - - VisualMapModel.prototype.init = function (option, parentModel, ecModel) { - this.mergeDefaultAndTheme(option, ecModel); - }; - /** - * @protected - */ - - - VisualMapModel.prototype.optionUpdated = function (newOption, isInit) { - var thisOption = this.option; - !isInit && replaceVisualOption(thisOption, newOption, this.replacableOptionKeys); - this.textStyleModel = this.getModel('textStyle'); - this.resetItemSize(); - this.completeVisualOption(); - }; - /** - * @protected - */ - - - VisualMapModel.prototype.resetVisual = function (supplementVisualOption) { - var stateList = this.stateList; - supplementVisualOption = bind(supplementVisualOption, this); - this.controllerVisuals = createVisualMappings(this.option.controller, stateList, supplementVisualOption); - this.targetVisuals = createVisualMappings(this.option.target, stateList, supplementVisualOption); - }; - /** - * @public - */ - - - VisualMapModel.prototype.getItemSymbol = function () { - return null; - }; - /** - * @protected - * @return {Array.<number>} An array of series indices. - */ - - - VisualMapModel.prototype.getTargetSeriesIndices = function () { - var optionSeriesIndex = this.option.seriesIndex; - var seriesIndices = []; - - if (optionSeriesIndex == null || optionSeriesIndex === 'all') { - this.ecModel.eachSeries(function (seriesModel, index) { - seriesIndices.push(index); - }); - } else { - seriesIndices = normalizeToArray(optionSeriesIndex); - } - - return seriesIndices; - }; - /** - * @public - */ - - - VisualMapModel.prototype.eachTargetSeries = function (callback, context) { - each(this.getTargetSeriesIndices(), function (seriesIndex) { - var seriesModel = this.ecModel.getSeriesByIndex(seriesIndex); - - if (seriesModel) { - callback.call(context, seriesModel); - } - }, this); - }; - /** - * @pubilc - */ - - - VisualMapModel.prototype.isTargetSeries = function (seriesModel) { - var is = false; - this.eachTargetSeries(function (model) { - model === seriesModel && (is = true); - }); - return is; - }; - /** - * @example - * this.formatValueText(someVal); // format single numeric value to text. - * this.formatValueText(someVal, true); // format single category value to text. - * this.formatValueText([min, max]); // format numeric min-max to text. - * this.formatValueText([this.dataBound[0], max]); // using data lower bound. - * this.formatValueText([min, this.dataBound[1]]); // using data upper bound. - * - * @param value Real value, or this.dataBound[0 or 1]. - * @param isCategory Only available when value is number. - * @param edgeSymbols Open-close symbol when value is interval. - * @protected - */ - - - VisualMapModel.prototype.formatValueText = function (value, isCategory, edgeSymbols) { - var option = this.option; - var precision = option.precision; - var dataBound = this.dataBound; - var formatter = option.formatter; - var isMinMax; - edgeSymbols = edgeSymbols || ['<', '>']; - - if (isArray(value)) { - value = value.slice(); - isMinMax = true; - } - - var textValue = isCategory ? value // Value is string when isCategory - : isMinMax ? [toFixed(value[0]), toFixed(value[1])] : toFixed(value); - - if (isString(formatter)) { - return formatter.replace('{value}', isMinMax ? textValue[0] : textValue).replace('{value2}', isMinMax ? textValue[1] : textValue); - } else if (isFunction(formatter)) { - return isMinMax ? formatter(value[0], value[1]) : formatter(value); - } - - if (isMinMax) { - if (value[0] === dataBound[0]) { - return edgeSymbols[0] + ' ' + textValue[1]; - } else if (value[1] === dataBound[1]) { - return edgeSymbols[1] + ' ' + textValue[0]; - } else { - return textValue[0] + ' - ' + textValue[1]; - } - } else { - // Format single value (includes category case). - return textValue; - } - - function toFixed(val) { - return val === dataBound[0] ? 'min' : val === dataBound[1] ? 'max' : (+val).toFixed(Math.min(precision, 20)); - } - }; - /** - * @protected - */ - - - VisualMapModel.prototype.resetExtent = function () { - var thisOption = this.option; // Can not calculate data extent by data here. - // Because series and data may be modified in processing stage. - // So we do not support the feature "auto min/max". - - var extent = asc$2([thisOption.min, thisOption.max]); - this._dataExtent = extent; - }; - /** - * PENDING: - * delete this method if no outer usage. - * - * Return Concrete dimension. If null/undefined is returned, no dimension is used. - */ - // getDataDimension(data: SeriesData) { - // const optDim = this.option.dimension; - // if (optDim != null) { - // return data.getDimension(optDim); - // } - // const dimNames = data.dimensions; - // for (let i = dimNames.length - 1; i >= 0; i--) { - // const dimName = dimNames[i]; - // const dimInfo = data.getDimensionInfo(dimName); - // if (!dimInfo.isCalculationCoord) { - // return dimName; - // } - // } - // } - - - VisualMapModel.prototype.getDataDimensionIndex = function (data) { - var optDim = this.option.dimension; - - if (optDim != null) { - return data.getDimensionIndex(optDim); - } - - var dimNames = data.dimensions; - - for (var i = dimNames.length - 1; i >= 0; i--) { - var dimName = dimNames[i]; - var dimInfo = data.getDimensionInfo(dimName); - - if (!dimInfo.isCalculationCoord) { - return dimInfo.storeDimIndex; - } - } - }; - - VisualMapModel.prototype.getExtent = function () { - return this._dataExtent.slice(); - }; - - VisualMapModel.prototype.completeVisualOption = function () { - var ecModel = this.ecModel; - var thisOption = this.option; - var base = { - inRange: thisOption.inRange, - outOfRange: thisOption.outOfRange - }; - var target = thisOption.target || (thisOption.target = {}); - var controller = thisOption.controller || (thisOption.controller = {}); - merge(target, base); // Do not override - - merge(controller, base); // Do not override - - var isCategory = this.isCategory(); - completeSingle.call(this, target); - completeSingle.call(this, controller); - completeInactive.call(this, target, 'inRange', 'outOfRange'); // completeInactive.call(this, target, 'outOfRange', 'inRange'); - - completeController.call(this, controller); - - function completeSingle(base) { - // Compatible with ec2 dataRange.color. - // The mapping order of dataRange.color is: [high value, ..., low value] - // whereas inRange.color and outOfRange.color is [low value, ..., high value] - // Notice: ec2 has no inverse. - if (isArray$1(thisOption.color) // If there has been inRange: {symbol: ...}, adding color is a mistake. - // So adding color only when no inRange defined. - && !base.inRange) { - base.inRange = { - color: thisOption.color.slice().reverse() - }; - } // Compatible with previous logic, always give a default color, otherwise - // simple config with no inRange and outOfRange will not work. - // Originally we use visualMap.color as the default color, but setOption at - // the second time the default color will be erased. So we change to use - // constant DEFAULT_COLOR. - // If user do not want the default color, set inRange: {color: null}. - - - base.inRange = base.inRange || { - color: ecModel.get('gradientColor') - }; - } - - function completeInactive(base, stateExist, stateAbsent) { - var optExist = base[stateExist]; - var optAbsent = base[stateAbsent]; - - if (optExist && !optAbsent) { - optAbsent = base[stateAbsent] = {}; - each$d(optExist, function (visualData, visualType) { - if (!VisualMapping.isValidType(visualType)) { - return; - } - - var defa = visualDefault.get(visualType, 'inactive', isCategory); - - if (defa != null) { - optAbsent[visualType] = defa; // Compatibable with ec2: - // Only inactive color to rgba(0,0,0,0) can not - // make label transparent, so use opacity also. - - if (visualType === 'color' && !optAbsent.hasOwnProperty('opacity') && !optAbsent.hasOwnProperty('colorAlpha')) { - optAbsent.opacity = [0, 0]; - } - } - }); - } - } - - function completeController(controller) { - var symbolExists = (controller.inRange || {}).symbol || (controller.outOfRange || {}).symbol; - var symbolSizeExists = (controller.inRange || {}).symbolSize || (controller.outOfRange || {}).symbolSize; - var inactiveColor = this.get('inactiveColor'); - var itemSymbol = this.getItemSymbol(); - var defaultSymbol = itemSymbol || 'roundRect'; - each$d(this.stateList, function (state) { - var itemSize = this.itemSize; - var visuals = controller[state]; // Set inactive color for controller if no other color - // attr (like colorAlpha) specified. - - if (!visuals) { - visuals = controller[state] = { - color: isCategory ? inactiveColor : [inactiveColor] - }; - } // Consistent symbol and symbolSize if not specified. - - - if (visuals.symbol == null) { - visuals.symbol = symbolExists && clone(symbolExists) || (isCategory ? defaultSymbol : [defaultSymbol]); - } - - if (visuals.symbolSize == null) { - visuals.symbolSize = symbolSizeExists && clone(symbolSizeExists) || (isCategory ? itemSize[0] : [itemSize[0], itemSize[0]]); - } // Filter none - - - visuals.symbol = mapVisual$1(visuals.symbol, function (symbol) { - return symbol === 'none' ? defaultSymbol : symbol; - }); // Normalize symbolSize - - var symbolSize = visuals.symbolSize; - - if (symbolSize != null) { - var max_1 = -Infinity; // symbolSize can be object when categories defined. - - eachVisual(symbolSize, function (value) { - value > max_1 && (max_1 = value); - }); - visuals.symbolSize = mapVisual$1(symbolSize, function (value) { - return linearMap$1(value, [0, max_1], [0, itemSize[0]], true); - }); - } - }, this); - } - }; - - VisualMapModel.prototype.resetItemSize = function () { - this.itemSize = [parseFloat(this.get('itemWidth')), parseFloat(this.get('itemHeight'))]; - }; - - VisualMapModel.prototype.isCategory = function () { - return !!this.option.categories; - }; - /** - * @public - * @abstract - */ - - - VisualMapModel.prototype.setSelected = function (selected) {}; - - VisualMapModel.prototype.getSelected = function () { - return null; - }; - /** - * @public - * @abstract - */ - - - VisualMapModel.prototype.getValueState = function (value) { - return null; - }; - /** - * FIXME - * Do not publish to thirt-part-dev temporarily - * util the interface is stable. (Should it return - * a function but not visual meta?) - * - * @pubilc - * @abstract - * @param getColorVisual - * params: value, valueState - * return: color - * @return {Object} visualMeta - * should includes {stops, outerColors} - * outerColor means [colorBeyondMinValue, colorBeyondMaxValue] - */ - - - VisualMapModel.prototype.getVisualMeta = function (getColorVisual) { - return null; - }; - - VisualMapModel.type = 'visualMap'; - VisualMapModel.dependencies = ['series']; - VisualMapModel.defaultOption = { - show: true, - // zlevel: 0, - z: 4, - seriesIndex: 'all', - min: 0, - max: 200, - left: 0, - right: null, - top: null, - bottom: 0, - itemWidth: null, - itemHeight: null, - inverse: false, - orient: 'vertical', - backgroundColor: 'rgba(0,0,0,0)', - borderColor: '#ccc', - contentColor: '#5793f3', - inactiveColor: '#aaa', - borderWidth: 0, - padding: 5, - // 接受数组分别设定上右下左边距,同css - textGap: 10, - precision: 0, - textStyle: { - color: '#333' // 值域文字颜色 - - } - }; - return VisualMapModel; - }(ComponentModel); - - var DEFAULT_BAR_BOUND = [20, 140]; - - var ContinuousModel = - /** @class */ - function (_super) { - __extends(ContinuousModel, _super); - - function ContinuousModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ContinuousModel.type; - return _this; - } - /** - * @override - */ - - - ContinuousModel.prototype.optionUpdated = function (newOption, isInit) { - _super.prototype.optionUpdated.apply(this, arguments); - - this.resetExtent(); - this.resetVisual(function (mappingOption) { - mappingOption.mappingMethod = 'linear'; - mappingOption.dataExtent = this.getExtent(); - }); - - this._resetRange(); - }; - /** - * @protected - * @override - */ - - - ContinuousModel.prototype.resetItemSize = function () { - _super.prototype.resetItemSize.apply(this, arguments); - - var itemSize = this.itemSize; - (itemSize[0] == null || isNaN(itemSize[0])) && (itemSize[0] = DEFAULT_BAR_BOUND[0]); - (itemSize[1] == null || isNaN(itemSize[1])) && (itemSize[1] = DEFAULT_BAR_BOUND[1]); - }; - /** - * @private - */ - - - ContinuousModel.prototype._resetRange = function () { - var dataExtent = this.getExtent(); - var range = this.option.range; - - if (!range || range.auto) { - // `range` should always be array (so we don't use other - // value like 'auto') for user-friend. (consider getOption). - dataExtent.auto = 1; - this.option.range = dataExtent; - } else if (isArray(range)) { - if (range[0] > range[1]) { - range.reverse(); - } - - range[0] = Math.max(range[0], dataExtent[0]); - range[1] = Math.min(range[1], dataExtent[1]); - } - }; - /** - * @protected - * @override - */ - - - ContinuousModel.prototype.completeVisualOption = function () { - _super.prototype.completeVisualOption.apply(this, arguments); - - each(this.stateList, function (state) { - var symbolSize = this.option.controller[state].symbolSize; - - if (symbolSize && symbolSize[0] !== symbolSize[1]) { - symbolSize[0] = symbolSize[1] / 3; // For good looking. - } - }, this); - }; - /** - * @override - */ - - - ContinuousModel.prototype.setSelected = function (selected) { - this.option.range = selected.slice(); - - this._resetRange(); - }; - /** - * @public - */ - - - ContinuousModel.prototype.getSelected = function () { - var dataExtent = this.getExtent(); - var dataInterval = asc((this.get('range') || []).slice()); // Clamp - - dataInterval[0] > dataExtent[1] && (dataInterval[0] = dataExtent[1]); - dataInterval[1] > dataExtent[1] && (dataInterval[1] = dataExtent[1]); - dataInterval[0] < dataExtent[0] && (dataInterval[0] = dataExtent[0]); - dataInterval[1] < dataExtent[0] && (dataInterval[1] = dataExtent[0]); - return dataInterval; - }; - /** - * @override - */ - - - ContinuousModel.prototype.getValueState = function (value) { - var range = this.option.range; - var dataExtent = this.getExtent(); // When range[0] === dataExtent[0], any value larger than dataExtent[0] maps to 'inRange'. - // range[1] is processed likewise. - - return (range[0] <= dataExtent[0] || range[0] <= value) && (range[1] >= dataExtent[1] || value <= range[1]) ? 'inRange' : 'outOfRange'; - }; - - ContinuousModel.prototype.findTargetDataIndices = function (range) { - var result = []; - this.eachTargetSeries(function (seriesModel) { - var dataIndices = []; - var data = seriesModel.getData(); - data.each(this.getDataDimensionIndex(data), function (value, dataIndex) { - range[0] <= value && value <= range[1] && dataIndices.push(dataIndex); - }, this); - result.push({ - seriesId: seriesModel.id, - dataIndex: dataIndices - }); - }, this); - return result; - }; - /** - * @implement - */ - - - ContinuousModel.prototype.getVisualMeta = function (getColorVisual) { - var oVals = getColorStopValues(this, 'outOfRange', this.getExtent()); - var iVals = getColorStopValues(this, 'inRange', this.option.range.slice()); - var stops = []; - - function setStop(value, valueState) { - stops.push({ - value: value, - color: getColorVisual(value, valueState) - }); - } // Format to: outOfRange -- inRange -- outOfRange. - - - var iIdx = 0; - var oIdx = 0; - var iLen = iVals.length; - var oLen = oVals.length; - - for (; oIdx < oLen && (!iVals.length || oVals[oIdx] <= iVals[0]); oIdx++) { - // If oVal[oIdx] === iVals[iIdx], oVal[oIdx] should be ignored. - if (oVals[oIdx] < iVals[iIdx]) { - setStop(oVals[oIdx], 'outOfRange'); - } - } - - for (var first = 1; iIdx < iLen; iIdx++, first = 0) { - // If range is full, value beyond min, max will be clamped. - // make a singularity - first && stops.length && setStop(iVals[iIdx], 'outOfRange'); - setStop(iVals[iIdx], 'inRange'); - } - - for (var first = 1; oIdx < oLen; oIdx++) { - if (!iVals.length || iVals[iVals.length - 1] < oVals[oIdx]) { - // make a singularity - if (first) { - stops.length && setStop(stops[stops.length - 1].value, 'outOfRange'); - first = 0; - } - - setStop(oVals[oIdx], 'outOfRange'); - } - } - - var stopsLen = stops.length; - return { - stops: stops, - outerColors: [stopsLen ? stops[0].color : 'transparent', stopsLen ? stops[stopsLen - 1].color : 'transparent'] - }; - }; - - ContinuousModel.type = 'visualMap.continuous'; - ContinuousModel.defaultOption = inheritDefaultOption(VisualMapModel.defaultOption, { - align: 'auto', - calculable: false, - hoverLink: true, - realtime: true, - handleIcon: 'path://M-11.39,9.77h0a3.5,3.5,0,0,1-3.5,3.5h-22a3.5,3.5,0,0,1-3.5-3.5h0a3.5,3.5,0,0,1,3.5-3.5h22A3.5,3.5,0,0,1-11.39,9.77Z', - handleSize: '120%', - handleStyle: { - borderColor: '#fff', - borderWidth: 1 - }, - indicatorIcon: 'circle', - indicatorSize: '50%', - indicatorStyle: { - borderColor: '#fff', - borderWidth: 2, - shadowBlur: 2, - shadowOffsetX: 1, - shadowOffsetY: 1, - shadowColor: 'rgba(0,0,0,0.2)' - } // emphasis: { - // handleStyle: { - // shadowBlur: 3, - // shadowOffsetX: 1, - // shadowOffsetY: 1, - // shadowColor: 'rgba(0,0,0,0.2)' - // } - // } - - }); - return ContinuousModel; - }(VisualMapModel); - - function getColorStopValues(visualMapModel, valueState, dataExtent) { - if (dataExtent[0] === dataExtent[1]) { - return dataExtent.slice(); - } // When using colorHue mapping, it is not linear color any more. - // Moreover, canvas gradient seems not to be accurate linear. - // FIXME - // Should be arbitrary value 100? or based on pixel size? - - - var count = 200; - var step = (dataExtent[1] - dataExtent[0]) / count; - var value = dataExtent[0]; - var stopValues = []; - - for (var i = 0; i <= count && value < dataExtent[1]; i++) { - stopValues.push(value); - value += step; - } - - stopValues.push(dataExtent[1]); - return stopValues; - } - - var VisualMapView = - /** @class */ - function (_super) { - __extends(VisualMapView, _super); - - function VisualMapView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = VisualMapView.type; - _this.autoPositionValues = { - left: 1, - right: 1, - top: 1, - bottom: 1 - }; - return _this; - } - - VisualMapView.prototype.init = function (ecModel, api) { - this.ecModel = ecModel; - this.api = api; - }; - /** - * @protected - */ - - - VisualMapView.prototype.render = function (visualMapModel, ecModel, api, payload // TODO: TYPE - ) { - this.visualMapModel = visualMapModel; - - if (visualMapModel.get('show') === false) { - this.group.removeAll(); - return; - } - - this.doRender(visualMapModel, ecModel, api, payload); - }; - /** - * @protected - */ - - - VisualMapView.prototype.renderBackground = function (group) { - var visualMapModel = this.visualMapModel; - var padding = normalizeCssArray$1(visualMapModel.get('padding') || 0); - var rect = group.getBoundingRect(); - group.add(new Rect({ - z2: -1, - silent: true, - shape: { - x: rect.x - padding[3], - y: rect.y - padding[0], - width: rect.width + padding[3] + padding[1], - height: rect.height + padding[0] + padding[2] - }, - style: { - fill: visualMapModel.get('backgroundColor'), - stroke: visualMapModel.get('borderColor'), - lineWidth: visualMapModel.get('borderWidth') - } - })); - }; - /** - * @protected - * @param targetValue can be Infinity or -Infinity - * @param visualCluster Only can be 'color' 'opacity' 'symbol' 'symbolSize' - * @param opts - * @param opts.forceState Specify state, instead of using getValueState method. - * @param opts.convertOpacityToAlpha For color gradient in controller widget. - * @return {*} Visual value. - */ - - - VisualMapView.prototype.getControllerVisual = function (targetValue, visualCluster, opts) { - opts = opts || {}; - var forceState = opts.forceState; - var visualMapModel = this.visualMapModel; - var visualObj = {}; // Default values. - - if (visualCluster === 'color') { - var defaultColor = visualMapModel.get('contentColor'); - visualObj.color = defaultColor; - } - - function getter(key) { - return visualObj[key]; - } - - function setter(key, value) { - visualObj[key] = value; - } - - var mappings = visualMapModel.controllerVisuals[forceState || visualMapModel.getValueState(targetValue)]; - var visualTypes = VisualMapping.prepareVisualTypes(mappings); - each(visualTypes, function (type) { - var visualMapping = mappings[type]; - - if (opts.convertOpacityToAlpha && type === 'opacity') { - type = 'colorAlpha'; - visualMapping = mappings.__alphaForOpacity; - } - - if (VisualMapping.dependsOn(type, visualCluster)) { - visualMapping && visualMapping.applyVisual(targetValue, getter, setter); - } - }); - return visualObj[visualCluster]; - }; - - VisualMapView.prototype.positionGroup = function (group) { - var model = this.visualMapModel; - var api = this.api; - positionElement(group, model.getBoxLayoutParams(), { - width: api.getWidth(), - height: api.getHeight() - }); - }; - - VisualMapView.prototype.doRender = function (visualMapModel, ecModel, api, payload) {}; - - VisualMapView.type = 'visualMap'; - return VisualMapView; - }(ComponentView); - - var paramsSet = [['left', 'right', 'width'], ['top', 'bottom', 'height']]; - /** - * @param visualMapModel - * @param api - * @param itemSize always [short, long] - * @return {string} 'left' or 'right' or 'top' or 'bottom' - */ - - function getItemAlign(visualMapModel, api, itemSize) { - var modelOption = visualMapModel.option; - var itemAlign = modelOption.align; - - if (itemAlign != null && itemAlign !== 'auto') { - return itemAlign; - } // Auto decision align. - - - var ecSize = { - width: api.getWidth(), - height: api.getHeight() - }; - var realIndex = modelOption.orient === 'horizontal' ? 1 : 0; - var reals = paramsSet[realIndex]; - var fakeValue = [0, null, 10]; - var layoutInput = {}; - - for (var i = 0; i < 3; i++) { - layoutInput[paramsSet[1 - realIndex][i]] = fakeValue[i]; - layoutInput[reals[i]] = i === 2 ? itemSize[0] : modelOption[reals[i]]; - } - - var rParam = [['x', 'width', 3], ['y', 'height', 0]][realIndex]; - var rect = getLayoutRect(layoutInput, ecSize, modelOption.padding); - return reals[(rect.margin[rParam[2]] || 0) + rect[rParam[0]] + rect[rParam[1]] * 0.5 < ecSize[rParam[1]] * 0.5 ? 0 : 1]; - } - /** - * Prepare dataIndex for outside usage, where dataIndex means rawIndex, and - * dataIndexInside means filtered index. - */ - // TODO: TYPE more specified payload types. - - function makeHighDownBatch(batch, visualMapModel) { - each(batch || [], function (batchItem) { - if (batchItem.dataIndex != null) { - batchItem.dataIndexInside = batchItem.dataIndex; - batchItem.dataIndex = null; - } - - batchItem.highlightKey = 'visualMap' + (visualMapModel ? visualMapModel.componentIndex : ''); - }); - return batch; - } - - var linearMap$2 = linearMap; - var each$e = each; - var mathMin$a = Math.min; - var mathMax$a = Math.max; // Arbitrary value - - var HOVER_LINK_SIZE = 12; - var HOVER_LINK_OUT = 6; // Notice: - // Any "interval" should be by the order of [low, high]. - // "handle0" (handleIndex === 0) maps to - // low data value: this._dataInterval[0] and has low coord. - // "handle1" (handleIndex === 1) maps to - // high data value: this._dataInterval[1] and has high coord. - // The logic of transform is implemented in this._createBarGroup. - - var ContinuousView = - /** @class */ - function (_super) { - __extends(ContinuousView, _super); - - function ContinuousView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = ContinuousView.type; - _this._shapes = {}; - _this._dataInterval = []; - _this._handleEnds = []; - _this._hoverLinkDataIndices = []; - return _this; - } - - ContinuousView.prototype.doRender = function (visualMapModel, ecModel, api, payload) { - this._api = api; - - if (!payload || payload.type !== 'selectDataRange' || payload.from !== this.uid) { - this._buildView(); - } - }; - - ContinuousView.prototype._buildView = function () { - this.group.removeAll(); - var visualMapModel = this.visualMapModel; - var thisGroup = this.group; - this._orient = visualMapModel.get('orient'); - this._useHandle = visualMapModel.get('calculable'); - - this._resetInterval(); - - this._renderBar(thisGroup); - - var dataRangeText = visualMapModel.get('text'); - - this._renderEndsText(thisGroup, dataRangeText, 0); - - this._renderEndsText(thisGroup, dataRangeText, 1); // Do this for background size calculation. - - - this._updateView(true); // After updating view, inner shapes is built completely, - // and then background can be rendered. - - - this.renderBackground(thisGroup); // Real update view - - this._updateView(); - - this._enableHoverLinkToSeries(); - - this._enableHoverLinkFromSeries(); - - this.positionGroup(thisGroup); - }; - - ContinuousView.prototype._renderEndsText = function (group, dataRangeText, endsIndex) { - if (!dataRangeText) { - return; - } // Compatible with ec2, text[0] map to high value, text[1] map low value. - - - var text = dataRangeText[1 - endsIndex]; - text = text != null ? text + '' : ''; - var visualMapModel = this.visualMapModel; - var textGap = visualMapModel.get('textGap'); - var itemSize = visualMapModel.itemSize; - var barGroup = this._shapes.mainGroup; - - var position = this._applyTransform([itemSize[0] / 2, endsIndex === 0 ? -textGap : itemSize[1] + textGap], barGroup); - - var align = this._applyTransform(endsIndex === 0 ? 'bottom' : 'top', barGroup); - - var orient = this._orient; - var textStyleModel = this.visualMapModel.textStyleModel; - this.group.add(new ZRText({ - style: createTextStyle(textStyleModel, { - x: position[0], - y: position[1], - verticalAlign: orient === 'horizontal' ? 'middle' : align, - align: orient === 'horizontal' ? align : 'center', - text: text - }) - })); - }; - - ContinuousView.prototype._renderBar = function (targetGroup) { - var visualMapModel = this.visualMapModel; - var shapes = this._shapes; - var itemSize = visualMapModel.itemSize; - var orient = this._orient; - var useHandle = this._useHandle; - var itemAlign = getItemAlign(visualMapModel, this.api, itemSize); - - var mainGroup = shapes.mainGroup = this._createBarGroup(itemAlign); - - var gradientBarGroup = new Group(); - mainGroup.add(gradientBarGroup); // Bar - - gradientBarGroup.add(shapes.outOfRange = createPolygon()); - gradientBarGroup.add(shapes.inRange = createPolygon(null, useHandle ? getCursor$1(this._orient) : null, bind(this._dragHandle, this, 'all', false), bind(this._dragHandle, this, 'all', true))); // A border radius clip. - - gradientBarGroup.setClipPath(new Rect({ - shape: { - x: 0, - y: 0, - width: itemSize[0], - height: itemSize[1], - r: 3 - } - })); - var textRect = visualMapModel.textStyleModel.getTextRect('国'); - var textSize = mathMax$a(textRect.width, textRect.height); // Handle - - if (useHandle) { - shapes.handleThumbs = []; - shapes.handleLabels = []; - shapes.handleLabelPoints = []; - - this._createHandle(visualMapModel, mainGroup, 0, itemSize, textSize, orient); - - this._createHandle(visualMapModel, mainGroup, 1, itemSize, textSize, orient); - } - - this._createIndicator(visualMapModel, mainGroup, itemSize, textSize, orient); - - targetGroup.add(mainGroup); - }; - - ContinuousView.prototype._createHandle = function (visualMapModel, mainGroup, handleIndex, itemSize, textSize, orient) { - var onDrift = bind(this._dragHandle, this, handleIndex, false); - var onDragEnd = bind(this._dragHandle, this, handleIndex, true); - var handleSize = parsePercent(visualMapModel.get('handleSize'), itemSize[0]); - var handleThumb = createSymbol(visualMapModel.get('handleIcon'), -handleSize / 2, -handleSize / 2, handleSize, handleSize, null, true); - var cursor = getCursor$1(this._orient); - handleThumb.attr({ - cursor: cursor, - draggable: true, - drift: onDrift, - ondragend: onDragEnd, - onmousemove: function (e) { - stop(e.event); - } - }); - handleThumb.x = itemSize[0] / 2; - handleThumb.useStyle(visualMapModel.getModel('handleStyle').getItemStyle()); - handleThumb.setStyle({ - strokeNoScale: true, - strokeFirst: true - }); - handleThumb.style.lineWidth *= 2; - handleThumb.ensureState('emphasis').style = visualMapModel.getModel(['emphasis', 'handleStyle']).getItemStyle(); - setAsHighDownDispatcher(handleThumb, true); - mainGroup.add(handleThumb); // Text is always horizontal layout but should not be effected by - // transform (orient/inverse). So label is built separately but not - // use zrender/graphic/helper/RectText, and is located based on view - // group (according to handleLabelPoint) but not barGroup. - - var textStyleModel = this.visualMapModel.textStyleModel; - var handleLabel = new ZRText({ - cursor: cursor, - draggable: true, - drift: onDrift, - onmousemove: function (e) { - // For mobile device, prevent screen slider on the button. - stop(e.event); - }, - ondragend: onDragEnd, - style: createTextStyle(textStyleModel, { - x: 0, - y: 0, - text: '' - }) - }); - handleLabel.ensureState('blur').style = { - opacity: 0.1 - }; - handleLabel.stateTransition = { - duration: 200 - }; - this.group.add(handleLabel); - var handleLabelPoint = [handleSize, 0]; - var shapes = this._shapes; - shapes.handleThumbs[handleIndex] = handleThumb; - shapes.handleLabelPoints[handleIndex] = handleLabelPoint; - shapes.handleLabels[handleIndex] = handleLabel; - }; - - ContinuousView.prototype._createIndicator = function (visualMapModel, mainGroup, itemSize, textSize, orient) { - var scale = parsePercent(visualMapModel.get('indicatorSize'), itemSize[0]); - var indicator = createSymbol(visualMapModel.get('indicatorIcon'), -scale / 2, -scale / 2, scale, scale, null, true); - indicator.attr({ - cursor: 'move', - invisible: true, - silent: true, - x: itemSize[0] / 2 - }); - var indicatorStyle = visualMapModel.getModel('indicatorStyle').getItemStyle(); - - if (indicator instanceof ZRImage) { - var pathStyle = indicator.style; - indicator.useStyle(extend({ - // TODO other properties like x, y ? - image: pathStyle.image, - x: pathStyle.x, - y: pathStyle.y, - width: pathStyle.width, - height: pathStyle.height - }, indicatorStyle)); - } else { - indicator.useStyle(indicatorStyle); - } - - mainGroup.add(indicator); - var textStyleModel = this.visualMapModel.textStyleModel; - var indicatorLabel = new ZRText({ - silent: true, - invisible: true, - style: createTextStyle(textStyleModel, { - x: 0, - y: 0, - text: '' - }) - }); - this.group.add(indicatorLabel); - var indicatorLabelPoint = [(orient === 'horizontal' ? textSize / 2 : HOVER_LINK_OUT) + itemSize[0] / 2, 0]; - var shapes = this._shapes; - shapes.indicator = indicator; - shapes.indicatorLabel = indicatorLabel; - shapes.indicatorLabelPoint = indicatorLabelPoint; - this._firstShowIndicator = true; - }; - - ContinuousView.prototype._dragHandle = function (handleIndex, isEnd, // dx is event from ondragend if isEnd is true. It's not used - dx, dy) { - if (!this._useHandle) { - return; - } - - this._dragging = !isEnd; - - if (!isEnd) { - // Transform dx, dy to bar coordination. - var vertex = this._applyTransform([dx, dy], this._shapes.mainGroup, true); - - this._updateInterval(handleIndex, vertex[1]); - - this._hideIndicator(); // Considering realtime, update view should be executed - // before dispatch action. - - - this._updateView(); - } // dragEnd do not dispatch action when realtime. - - - if (isEnd === !this.visualMapModel.get('realtime')) { - // jshint ignore:line - this.api.dispatchAction({ - type: 'selectDataRange', - from: this.uid, - visualMapId: this.visualMapModel.id, - selected: this._dataInterval.slice() - }); - } - - if (isEnd) { - !this._hovering && this._clearHoverLinkToSeries(); - } else if (useHoverLinkOnHandle(this.visualMapModel)) { - this._doHoverLinkToSeries(this._handleEnds[handleIndex], false); - } - }; - - ContinuousView.prototype._resetInterval = function () { - var visualMapModel = this.visualMapModel; - var dataInterval = this._dataInterval = visualMapModel.getSelected(); - var dataExtent = visualMapModel.getExtent(); - var sizeExtent = [0, visualMapModel.itemSize[1]]; - this._handleEnds = [linearMap$2(dataInterval[0], dataExtent, sizeExtent, true), linearMap$2(dataInterval[1], dataExtent, sizeExtent, true)]; - }; - /** - * @private - * @param {(number|string)} handleIndex 0 or 1 or 'all' - * @param {number} dx - * @param {number} dy - */ - - - ContinuousView.prototype._updateInterval = function (handleIndex, delta) { - delta = delta || 0; - var visualMapModel = this.visualMapModel; - var handleEnds = this._handleEnds; - var sizeExtent = [0, visualMapModel.itemSize[1]]; - sliderMove(delta, handleEnds, sizeExtent, handleIndex, // cross is forbidden - 0); - var dataExtent = visualMapModel.getExtent(); // Update data interval. - - this._dataInterval = [linearMap$2(handleEnds[0], sizeExtent, dataExtent, true), linearMap$2(handleEnds[1], sizeExtent, dataExtent, true)]; - }; - - ContinuousView.prototype._updateView = function (forSketch) { - var visualMapModel = this.visualMapModel; - var dataExtent = visualMapModel.getExtent(); - var shapes = this._shapes; - var outOfRangeHandleEnds = [0, visualMapModel.itemSize[1]]; - var inRangeHandleEnds = forSketch ? outOfRangeHandleEnds : this._handleEnds; - - var visualInRange = this._createBarVisual(this._dataInterval, dataExtent, inRangeHandleEnds, 'inRange'); - - var visualOutOfRange = this._createBarVisual(dataExtent, dataExtent, outOfRangeHandleEnds, 'outOfRange'); - - shapes.inRange.setStyle({ - fill: visualInRange.barColor // opacity: visualInRange.opacity - - }).setShape('points', visualInRange.barPoints); - shapes.outOfRange.setStyle({ - fill: visualOutOfRange.barColor // opacity: visualOutOfRange.opacity - - }).setShape('points', visualOutOfRange.barPoints); - - this._updateHandle(inRangeHandleEnds, visualInRange); - }; - - ContinuousView.prototype._createBarVisual = function (dataInterval, dataExtent, handleEnds, forceState) { - var opts = { - forceState: forceState, - convertOpacityToAlpha: true - }; - - var colorStops = this._makeColorGradient(dataInterval, opts); - - var symbolSizes = [this.getControllerVisual(dataInterval[0], 'symbolSize', opts), this.getControllerVisual(dataInterval[1], 'symbolSize', opts)]; - - var barPoints = this._createBarPoints(handleEnds, symbolSizes); - - return { - barColor: new LinearGradient(0, 0, 0, 1, colorStops), - barPoints: barPoints, - handlesColor: [colorStops[0].color, colorStops[colorStops.length - 1].color] - }; - }; - - ContinuousView.prototype._makeColorGradient = function (dataInterval, opts) { - // Considering colorHue, which is not linear, so we have to sample - // to calculate gradient color stops, but not only calculate head - // and tail. - var sampleNumber = 100; // Arbitrary value. - - var colorStops = []; - var step = (dataInterval[1] - dataInterval[0]) / sampleNumber; - colorStops.push({ - color: this.getControllerVisual(dataInterval[0], 'color', opts), - offset: 0 - }); - - for (var i = 1; i < sampleNumber; i++) { - var currValue = dataInterval[0] + step * i; - - if (currValue > dataInterval[1]) { - break; - } - - colorStops.push({ - color: this.getControllerVisual(currValue, 'color', opts), - offset: i / sampleNumber - }); - } - - colorStops.push({ - color: this.getControllerVisual(dataInterval[1], 'color', opts), - offset: 1 - }); - return colorStops; - }; - - ContinuousView.prototype._createBarPoints = function (handleEnds, symbolSizes) { - var itemSize = this.visualMapModel.itemSize; - return [[itemSize[0] - symbolSizes[0], handleEnds[0]], [itemSize[0], handleEnds[0]], [itemSize[0], handleEnds[1]], [itemSize[0] - symbolSizes[1], handleEnds[1]]]; - }; - - ContinuousView.prototype._createBarGroup = function (itemAlign) { - var orient = this._orient; - var inverse = this.visualMapModel.get('inverse'); - return new Group(orient === 'horizontal' && !inverse ? { - scaleX: itemAlign === 'bottom' ? 1 : -1, - rotation: Math.PI / 2 - } : orient === 'horizontal' && inverse ? { - scaleX: itemAlign === 'bottom' ? -1 : 1, - rotation: -Math.PI / 2 - } : orient === 'vertical' && !inverse ? { - scaleX: itemAlign === 'left' ? 1 : -1, - scaleY: -1 - } : { - scaleX: itemAlign === 'left' ? 1 : -1 - }); - }; - - ContinuousView.prototype._updateHandle = function (handleEnds, visualInRange) { - if (!this._useHandle) { - return; - } - - var shapes = this._shapes; - var visualMapModel = this.visualMapModel; - var handleThumbs = shapes.handleThumbs; - var handleLabels = shapes.handleLabels; - var itemSize = visualMapModel.itemSize; - var dataExtent = visualMapModel.getExtent(); - each$e([0, 1], function (handleIndex) { - var handleThumb = handleThumbs[handleIndex]; - handleThumb.setStyle('fill', visualInRange.handlesColor[handleIndex]); - handleThumb.y = handleEnds[handleIndex]; - var val = linearMap$2(handleEnds[handleIndex], [0, itemSize[1]], dataExtent, true); - var symbolSize = this.getControllerVisual(val, 'symbolSize'); - handleThumb.scaleX = handleThumb.scaleY = symbolSize / itemSize[0]; - handleThumb.x = itemSize[0] - symbolSize / 2; // Update handle label position. - - var textPoint = applyTransform$1(shapes.handleLabelPoints[handleIndex], getTransform(handleThumb, this.group)); - handleLabels[handleIndex].setStyle({ - x: textPoint[0], - y: textPoint[1], - text: visualMapModel.formatValueText(this._dataInterval[handleIndex]), - verticalAlign: 'middle', - align: this._orient === 'vertical' ? this._applyTransform('left', shapes.mainGroup) : 'center' - }); - }, this); - }; - - ContinuousView.prototype._showIndicator = function (cursorValue, textValue, rangeSymbol, halfHoverLinkSize) { - var visualMapModel = this.visualMapModel; - var dataExtent = visualMapModel.getExtent(); - var itemSize = visualMapModel.itemSize; - var sizeExtent = [0, itemSize[1]]; - var shapes = this._shapes; - var indicator = shapes.indicator; - - if (!indicator) { - return; - } - - indicator.attr('invisible', false); - var opts = { - convertOpacityToAlpha: true - }; - var color = this.getControllerVisual(cursorValue, 'color', opts); - var symbolSize = this.getControllerVisual(cursorValue, 'symbolSize'); - var y = linearMap$2(cursorValue, dataExtent, sizeExtent, true); - var x = itemSize[0] - symbolSize / 2; - var oldIndicatorPos = { - x: indicator.x, - y: indicator.y - }; // Update handle label position. - - indicator.y = y; - indicator.x = x; - var textPoint = applyTransform$1(shapes.indicatorLabelPoint, getTransform(indicator, this.group)); - var indicatorLabel = shapes.indicatorLabel; - indicatorLabel.attr('invisible', false); - - var align = this._applyTransform('left', shapes.mainGroup); - - var orient = this._orient; - var isHorizontal = orient === 'horizontal'; - indicatorLabel.setStyle({ - text: (rangeSymbol ? rangeSymbol : '') + visualMapModel.formatValueText(textValue), - verticalAlign: isHorizontal ? align : 'middle', - align: isHorizontal ? 'center' : align - }); - var indicatorNewProps = { - x: x, - y: y, - style: { - fill: color - } - }; - var labelNewProps = { - style: { - x: textPoint[0], - y: textPoint[1] - } - }; - - if (visualMapModel.ecModel.isAnimationEnabled() && !this._firstShowIndicator) { - var animationCfg = { - duration: 100, - easing: 'cubicInOut', - additive: true - }; - indicator.x = oldIndicatorPos.x; - indicator.y = oldIndicatorPos.y; - indicator.animateTo(indicatorNewProps, animationCfg); - indicatorLabel.animateTo(labelNewProps, animationCfg); - } else { - indicator.attr(indicatorNewProps); - indicatorLabel.attr(labelNewProps); - } - - this._firstShowIndicator = false; - var handleLabels = this._shapes.handleLabels; - - if (handleLabels) { - for (var i = 0; i < handleLabels.length; i++) { - // Fade out handle labels. - // NOTE: Must use api enter/leave on emphasis/blur/select state. Or the global states manager will change it. - this._api.enterBlur(handleLabels[i]); - } - } - }; - - ContinuousView.prototype._enableHoverLinkToSeries = function () { - var self = this; - - this._shapes.mainGroup.on('mousemove', function (e) { - self._hovering = true; - - if (!self._dragging) { - var itemSize = self.visualMapModel.itemSize; - - var pos = self._applyTransform([e.offsetX, e.offsetY], self._shapes.mainGroup, true, true); // For hover link show when hover handle, which might be - // below or upper than sizeExtent. - - - pos[1] = mathMin$a(mathMax$a(0, pos[1]), itemSize[1]); - - self._doHoverLinkToSeries(pos[1], 0 <= pos[0] && pos[0] <= itemSize[0]); - } - }).on('mouseout', function () { - // When mouse is out of handle, hoverLink still need - // to be displayed when realtime is set as false. - self._hovering = false; - !self._dragging && self._clearHoverLinkToSeries(); - }); - }; - - ContinuousView.prototype._enableHoverLinkFromSeries = function () { - var zr = this.api.getZr(); - - if (this.visualMapModel.option.hoverLink) { - zr.on('mouseover', this._hoverLinkFromSeriesMouseOver, this); - zr.on('mouseout', this._hideIndicator, this); - } else { - this._clearHoverLinkFromSeries(); - } - }; - - ContinuousView.prototype._doHoverLinkToSeries = function (cursorPos, hoverOnBar) { - var visualMapModel = this.visualMapModel; - var itemSize = visualMapModel.itemSize; - - if (!visualMapModel.option.hoverLink) { - return; - } - - var sizeExtent = [0, itemSize[1]]; - var dataExtent = visualMapModel.getExtent(); // For hover link show when hover handle, which might be below or upper than sizeExtent. - - cursorPos = mathMin$a(mathMax$a(sizeExtent[0], cursorPos), sizeExtent[1]); - var halfHoverLinkSize = getHalfHoverLinkSize(visualMapModel, dataExtent, sizeExtent); - var hoverRange = [cursorPos - halfHoverLinkSize, cursorPos + halfHoverLinkSize]; - var cursorValue = linearMap$2(cursorPos, sizeExtent, dataExtent, true); - var valueRange = [linearMap$2(hoverRange[0], sizeExtent, dataExtent, true), linearMap$2(hoverRange[1], sizeExtent, dataExtent, true)]; // Consider data range is out of visualMap range, see test/visualMap-continuous.html, - // where china and india has very large population. - - hoverRange[0] < sizeExtent[0] && (valueRange[0] = -Infinity); - hoverRange[1] > sizeExtent[1] && (valueRange[1] = Infinity); // Do not show indicator when mouse is over handle, - // otherwise labels overlap, especially when dragging. - - if (hoverOnBar) { - if (valueRange[0] === -Infinity) { - this._showIndicator(cursorValue, valueRange[1], '< ', halfHoverLinkSize); - } else if (valueRange[1] === Infinity) { - this._showIndicator(cursorValue, valueRange[0], '> ', halfHoverLinkSize); - } else { - this._showIndicator(cursorValue, cursorValue, '≈ ', halfHoverLinkSize); - } - } // When realtime is set as false, handles, which are in barGroup, - // also trigger hoverLink, which help user to realize where they - // focus on when dragging. (see test/heatmap-large.html) - // When realtime is set as true, highlight will not show when hover - // handle, because the label on handle, which displays a exact value - // but not range, might mislead users. - - - var oldBatch = this._hoverLinkDataIndices; - var newBatch = []; - - if (hoverOnBar || useHoverLinkOnHandle(visualMapModel)) { - newBatch = this._hoverLinkDataIndices = visualMapModel.findTargetDataIndices(valueRange); - } - - var resultBatches = compressBatches(oldBatch, newBatch); - - this._dispatchHighDown('downplay', makeHighDownBatch(resultBatches[0], visualMapModel)); - - this._dispatchHighDown('highlight', makeHighDownBatch(resultBatches[1], visualMapModel)); - }; - - ContinuousView.prototype._hoverLinkFromSeriesMouseOver = function (e) { - var ecData; - findEventDispatcher(e.target, function (target) { - var currECData = getECData(target); - - if (currECData.dataIndex != null) { - ecData = currECData; - return true; - } - }, true); - - if (!ecData) { - return; - } - - var dataModel = this.ecModel.getSeriesByIndex(ecData.seriesIndex); - var visualMapModel = this.visualMapModel; - - if (!visualMapModel.isTargetSeries(dataModel)) { - return; - } - - var data = dataModel.getData(ecData.dataType); - var value = data.getStore().get(visualMapModel.getDataDimensionIndex(data), ecData.dataIndex); - - if (!isNaN(value)) { - this._showIndicator(value, value); - } - }; - - ContinuousView.prototype._hideIndicator = function () { - var shapes = this._shapes; - shapes.indicator && shapes.indicator.attr('invisible', true); - shapes.indicatorLabel && shapes.indicatorLabel.attr('invisible', true); - var handleLabels = this._shapes.handleLabels; - - if (handleLabels) { - for (var i = 0; i < handleLabels.length; i++) { - // Fade out handle labels. - // NOTE: Must use api enter/leave on emphasis/blur/select state. Or the global states manager will change it. - this._api.leaveBlur(handleLabels[i]); - } - } - }; - - ContinuousView.prototype._clearHoverLinkToSeries = function () { - this._hideIndicator(); - - var indices = this._hoverLinkDataIndices; - - this._dispatchHighDown('downplay', makeHighDownBatch(indices, this.visualMapModel)); - - indices.length = 0; - }; - - ContinuousView.prototype._clearHoverLinkFromSeries = function () { - this._hideIndicator(); - - var zr = this.api.getZr(); - zr.off('mouseover', this._hoverLinkFromSeriesMouseOver); - zr.off('mouseout', this._hideIndicator); - }; - - ContinuousView.prototype._applyTransform = function (vertex, element, inverse, global) { - var transform = getTransform(element, global ? null : this.group); - return isArray(vertex) ? applyTransform$1(vertex, transform, inverse) : transformDirection(vertex, transform, inverse); - }; // TODO: TYPE more specified payload types. - - - ContinuousView.prototype._dispatchHighDown = function (type, batch) { - batch && batch.length && this.api.dispatchAction({ - type: type, - batch: batch - }); - }; - /** - * @override - */ - - - ContinuousView.prototype.dispose = function () { - this._clearHoverLinkFromSeries(); - - this._clearHoverLinkToSeries(); - }; - /** - * @override - */ - - - ContinuousView.prototype.remove = function () { - this._clearHoverLinkFromSeries(); - - this._clearHoverLinkToSeries(); - }; - - ContinuousView.type = 'visualMap.continuous'; - return ContinuousView; - }(VisualMapView); - - function createPolygon(points, cursor, onDrift, onDragEnd) { - return new Polygon({ - shape: { - points: points - }, - draggable: !!onDrift, - cursor: cursor, - drift: onDrift, - onmousemove: function (e) { - // For mobile device, prevent screen slider on the button. - stop(e.event); - }, - ondragend: onDragEnd - }); - } - - function getHalfHoverLinkSize(visualMapModel, dataExtent, sizeExtent) { - var halfHoverLinkSize = HOVER_LINK_SIZE / 2; - var hoverLinkDataSize = visualMapModel.get('hoverLinkDataSize'); - - if (hoverLinkDataSize) { - halfHoverLinkSize = linearMap$2(hoverLinkDataSize, dataExtent, sizeExtent, true) / 2; - } - - return halfHoverLinkSize; - } - - function useHoverLinkOnHandle(visualMapModel) { - var hoverLinkOnHandle = visualMapModel.get('hoverLinkOnHandle'); - return !!(hoverLinkOnHandle == null ? visualMapModel.get('realtime') : hoverLinkOnHandle); - } - - function getCursor$1(orient) { - return orient === 'vertical' ? 'ns-resize' : 'ew-resize'; - } - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - - - /** - * AUTO-GENERATED FILE. DO NOT MODIFY. - */ - - /* - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - var visualMapActionInfo = { - type: 'selectDataRange', - event: 'dataRangeSelected', - // FIXME use updateView appears wrong - update: 'update' - }; - var visualMapActionHander = function (payload, ecModel) { - ecModel.eachComponent({ - mainType: 'visualMap', - query: payload - }, function (model) { - model.setSelected(payload.selected); - }); - }; - - var visualMapEncodingHandlers = [{ - createOnAllSeries: true, - reset: function (seriesModel, ecModel) { - var resetDefines = []; - ecModel.eachComponent('visualMap', function (visualMapModel) { - var pipelineContext = seriesModel.pipelineContext; - - if (!visualMapModel.isTargetSeries(seriesModel) || pipelineContext && pipelineContext.large) { - return; - } - - resetDefines.push(incrementalApplyVisual(visualMapModel.stateList, visualMapModel.targetVisuals, bind(visualMapModel.getValueState, visualMapModel), visualMapModel.getDataDimensionIndex(seriesModel.getData()))); - }); - return resetDefines; - } - }, // Only support color. - { - createOnAllSeries: true, - reset: function (seriesModel, ecModel) { - var data = seriesModel.getData(); - var visualMetaList = []; - ecModel.eachComponent('visualMap', function (visualMapModel) { - if (visualMapModel.isTargetSeries(seriesModel)) { - var visualMeta = visualMapModel.getVisualMeta(bind(getColorVisual, null, seriesModel, visualMapModel)) || { - stops: [], - outerColors: [] - }; - var dimIdx = visualMapModel.getDataDimensionIndex(data); - - if (dimIdx >= 0) { - // visualMeta.dimension should be dimension index, but not concrete dimension. - visualMeta.dimension = dimIdx; - visualMetaList.push(visualMeta); - } - } - }); // console.log(JSON.stringify(visualMetaList.map(a => a.stops))); - - seriesModel.getData().setVisual('visualMeta', visualMetaList); - } - }]; // FIXME - // performance and export for heatmap? - // value can be Infinity or -Infinity - - function getColorVisual(seriesModel, visualMapModel, value, valueState) { - var mappings = visualMapModel.targetVisuals[valueState]; - var visualTypes = VisualMapping.prepareVisualTypes(mappings); - var resultVisual = { - color: getVisualFromData(seriesModel.getData(), 'color') // default color. - - }; - - for (var i = 0, len = visualTypes.length; i < len; i++) { - var type = visualTypes[i]; - var mapping = mappings[type === 'opacity' ? '__alphaForOpacity' : type]; - mapping && mapping.applyVisual(value, getVisual, setVisual); - } - - return resultVisual.color; - - function getVisual(key) { - return resultVisual[key]; - } - - function setVisual(key, value) { - resultVisual[key] = value; - } - } - - var each$f = each; - function visualMapPreprocessor(option) { - var visualMap = option && option.visualMap; - - if (!isArray(visualMap)) { - visualMap = visualMap ? [visualMap] : []; - } - - each$f(visualMap, function (opt) { - if (!opt) { - return; - } // rename splitList to pieces - - - if (has$1(opt, 'splitList') && !has$1(opt, 'pieces')) { - opt.pieces = opt.splitList; - delete opt.splitList; - } - - var pieces = opt.pieces; - - if (pieces && isArray(pieces)) { - each$f(pieces, function (piece) { - if (isObject(piece)) { - if (has$1(piece, 'start') && !has$1(piece, 'min')) { - piece.min = piece.start; - } - - if (has$1(piece, 'end') && !has$1(piece, 'max')) { - piece.max = piece.end; - } - } - }); - } - }); - } - - function has$1(obj, name) { - return obj && obj.hasOwnProperty && obj.hasOwnProperty(name); - } - - var installed$1 = false; - function installCommon$1(registers) { - if (installed$1) { - return; - } - - installed$1 = true; - registers.registerSubTypeDefaulter('visualMap', function (option) { - // Compatible with ec2, when splitNumber === 0, continuous visualMap will be used. - return !option.categories && (!(option.pieces ? option.pieces.length > 0 : option.splitNumber > 0) || option.calculable) ? 'continuous' : 'piecewise'; - }); - registers.registerAction(visualMapActionInfo, visualMapActionHander); - each(visualMapEncodingHandlers, function (handler) { - registers.registerVisual(registers.PRIORITY.VISUAL.COMPONENT, handler); - }); - registers.registerPreprocessor(visualMapPreprocessor); - } - - function install$N(registers) { - registers.registerComponentModel(ContinuousModel); - registers.registerComponentView(ContinuousView); - installCommon$1(registers); - } - - var PiecewiseModel = - /** @class */ - function (_super) { - __extends(PiecewiseModel, _super); - - function PiecewiseModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = PiecewiseModel.type; - /** - * The order is always [low, ..., high]. - * [{text: string, interval: Array.<number>}, ...] - */ - - _this._pieceList = []; - return _this; - } - - PiecewiseModel.prototype.optionUpdated = function (newOption, isInit) { - _super.prototype.optionUpdated.apply(this, arguments); - - this.resetExtent(); - - var mode = this._mode = this._determineMode(); - - this._pieceList = []; - - resetMethods[this._mode].call(this, this._pieceList); - - this._resetSelected(newOption, isInit); - - var categories = this.option.categories; - this.resetVisual(function (mappingOption, state) { - if (mode === 'categories') { - mappingOption.mappingMethod = 'category'; - mappingOption.categories = clone(categories); - } else { - mappingOption.dataExtent = this.getExtent(); - mappingOption.mappingMethod = 'piecewise'; - mappingOption.pieceList = map(this._pieceList, function (piece) { - piece = clone(piece); - - if (state !== 'inRange') { - // FIXME - // outOfRange do not support special visual in pieces. - piece.visual = null; - } - - return piece; - }); - } - }); - }; - /** - * @protected - * @override - */ - - - PiecewiseModel.prototype.completeVisualOption = function () { - // Consider this case: - // visualMap: { - // pieces: [{symbol: 'circle', lt: 0}, {symbol: 'rect', gte: 0}] - // } - // where no inRange/outOfRange set but only pieces. So we should make - // default inRange/outOfRange for this case, otherwise visuals that only - // appear in `pieces` will not be taken into account in visual encoding. - var option = this.option; - var visualTypesInPieces = {}; - var visualTypes = VisualMapping.listVisualTypes(); - var isCategory = this.isCategory(); - each(option.pieces, function (piece) { - each(visualTypes, function (visualType) { - if (piece.hasOwnProperty(visualType)) { - visualTypesInPieces[visualType] = 1; - } - }); - }); - each(visualTypesInPieces, function (v, visualType) { - var exists = false; - each(this.stateList, function (state) { - exists = exists || has(option, state, visualType) || has(option.target, state, visualType); - }, this); - !exists && each(this.stateList, function (state) { - (option[state] || (option[state] = {}))[visualType] = visualDefault.get(visualType, state === 'inRange' ? 'active' : 'inactive', isCategory); - }); - }, this); - - function has(obj, state, visualType) { - return obj && obj[state] && obj[state].hasOwnProperty(visualType); - } - - _super.prototype.completeVisualOption.apply(this, arguments); - }; - - PiecewiseModel.prototype._resetSelected = function (newOption, isInit) { - var thisOption = this.option; - var pieceList = this._pieceList; // Selected do not merge but all override. - - var selected = (isInit ? thisOption : newOption).selected || {}; - thisOption.selected = selected; // Consider 'not specified' means true. - - each(pieceList, function (piece, index) { - var key = this.getSelectedMapKey(piece); - - if (!selected.hasOwnProperty(key)) { - selected[key] = true; - } - }, this); - - if (thisOption.selectedMode === 'single') { - // Ensure there is only one selected. - var hasSel_1 = false; - each(pieceList, function (piece, index) { - var key = this.getSelectedMapKey(piece); - - if (selected[key]) { - hasSel_1 ? selected[key] = false : hasSel_1 = true; - } - }, this); - } // thisOption.selectedMode === 'multiple', default: all selected. - - }; - /** - * @public - */ - - - PiecewiseModel.prototype.getItemSymbol = function () { - return this.get('itemSymbol'); - }; - /** - * @public - */ - - - PiecewiseModel.prototype.getSelectedMapKey = function (piece) { - return this._mode === 'categories' ? piece.value + '' : piece.index + ''; - }; - /** - * @public - */ - - - PiecewiseModel.prototype.getPieceList = function () { - return this._pieceList; - }; - /** - * @return {string} - */ - - - PiecewiseModel.prototype._determineMode = function () { - var option = this.option; - return option.pieces && option.pieces.length > 0 ? 'pieces' : this.option.categories ? 'categories' : 'splitNumber'; - }; - /** - * @override - */ - - - PiecewiseModel.prototype.setSelected = function (selected) { - this.option.selected = clone(selected); - }; - /** - * @override - */ - - - PiecewiseModel.prototype.getValueState = function (value) { - var index = VisualMapping.findPieceIndex(value, this._pieceList); - return index != null ? this.option.selected[this.getSelectedMapKey(this._pieceList[index])] ? 'inRange' : 'outOfRange' : 'outOfRange'; - }; - /** - * @public - * @param pieceIndex piece index in visualMapModel.getPieceList() - */ - - - PiecewiseModel.prototype.findTargetDataIndices = function (pieceIndex) { - var result = []; - var pieceList = this._pieceList; - this.eachTargetSeries(function (seriesModel) { - var dataIndices = []; - var data = seriesModel.getData(); - data.each(this.getDataDimensionIndex(data), function (value, dataIndex) { - // Should always base on model pieceList, because it is order sensitive. - var pIdx = VisualMapping.findPieceIndex(value, pieceList); - pIdx === pieceIndex && dataIndices.push(dataIndex); - }, this); - result.push({ - seriesId: seriesModel.id, - dataIndex: dataIndices - }); - }, this); - return result; - }; - /** - * @private - * @param piece piece.value or piece.interval is required. - * @return Can be Infinity or -Infinity - */ - - - PiecewiseModel.prototype.getRepresentValue = function (piece) { - var representValue; - - if (this.isCategory()) { - representValue = piece.value; - } else { - if (piece.value != null) { - representValue = piece.value; - } else { - var pieceInterval = piece.interval || []; - representValue = pieceInterval[0] === -Infinity && pieceInterval[1] === Infinity ? 0 : (pieceInterval[0] + pieceInterval[1]) / 2; - } - } - - return representValue; - }; - - PiecewiseModel.prototype.getVisualMeta = function (getColorVisual) { - // Do not support category. (category axis is ordinal, numerical) - if (this.isCategory()) { - return; - } - - var stops = []; - var outerColors = ['', '']; - var visualMapModel = this; - - function setStop(interval, valueState) { - var representValue = visualMapModel.getRepresentValue({ - interval: interval - }); // Not category - - if (!valueState) { - valueState = visualMapModel.getValueState(representValue); - } - - var color = getColorVisual(representValue, valueState); - - if (interval[0] === -Infinity) { - outerColors[0] = color; - } else if (interval[1] === Infinity) { - outerColors[1] = color; - } else { - stops.push({ - value: interval[0], - color: color - }, { - value: interval[1], - color: color - }); - } - } // Suplement - - - var pieceList = this._pieceList.slice(); - - if (!pieceList.length) { - pieceList.push({ - interval: [-Infinity, Infinity] - }); - } else { - var edge = pieceList[0].interval[0]; - edge !== -Infinity && pieceList.unshift({ - interval: [-Infinity, edge] - }); - edge = pieceList[pieceList.length - 1].interval[1]; - edge !== Infinity && pieceList.push({ - interval: [edge, Infinity] - }); - } - - var curr = -Infinity; - each(pieceList, function (piece) { - var interval = piece.interval; - - if (interval) { - // Fulfill gap. - interval[0] > curr && setStop([curr, interval[0]], 'outOfRange'); - setStop(interval.slice()); - curr = interval[1]; - } - }, this); - return { - stops: stops, - outerColors: outerColors - }; - }; - - PiecewiseModel.type = 'visualMap.piecewise'; - PiecewiseModel.defaultOption = inheritDefaultOption(VisualMapModel.defaultOption, { - selected: null, - minOpen: false, - maxOpen: false, - align: 'auto', - itemWidth: 20, - itemHeight: 14, - itemSymbol: 'roundRect', - pieces: null, - categories: null, - splitNumber: 5, - selectedMode: 'multiple', - itemGap: 10, - hoverLink: true // Enable hover highlight. - - }); - return PiecewiseModel; - }(VisualMapModel); - /** - * Key is this._mode - * @type {Object} - * @this {module:echarts/component/viusalMap/PiecewiseMode} - */ - - var resetMethods = { - splitNumber: function (outPieceList) { - var thisOption = this.option; - var precision = Math.min(thisOption.precision, 20); - var dataExtent = this.getExtent(); - var splitNumber = thisOption.splitNumber; - splitNumber = Math.max(parseInt(splitNumber, 10), 1); - thisOption.splitNumber = splitNumber; - var splitStep = (dataExtent[1] - dataExtent[0]) / splitNumber; // Precision auto-adaption - - while (+splitStep.toFixed(precision) !== splitStep && precision < 5) { - precision++; - } - - thisOption.precision = precision; - splitStep = +splitStep.toFixed(precision); - - if (thisOption.minOpen) { - outPieceList.push({ - interval: [-Infinity, dataExtent[0]], - close: [0, 0] - }); - } - - for (var index = 0, curr = dataExtent[0]; index < splitNumber; curr += splitStep, index++) { - var max = index === splitNumber - 1 ? dataExtent[1] : curr + splitStep; - outPieceList.push({ - interval: [curr, max], - close: [1, 1] - }); - } - - if (thisOption.maxOpen) { - outPieceList.push({ - interval: [dataExtent[1], Infinity], - close: [0, 0] - }); - } - - reformIntervals(outPieceList); - each(outPieceList, function (piece, index) { - piece.index = index; - piece.text = this.formatValueText(piece.interval); - }, this); - }, - categories: function (outPieceList) { - var thisOption = this.option; - each(thisOption.categories, function (cate) { - // FIXME category模式也使用pieceList,但在visualMapping中不是使用pieceList。 - // 是否改一致。 - outPieceList.push({ - text: this.formatValueText(cate, true), - value: cate - }); - }, this); // See "Order Rule". - - normalizeReverse(thisOption, outPieceList); - }, - pieces: function (outPieceList) { - var thisOption = this.option; - each(thisOption.pieces, function (pieceListItem, index) { - if (!isObject(pieceListItem)) { - pieceListItem = { - value: pieceListItem - }; - } - - var item = { - text: '', - index: index - }; - - if (pieceListItem.label != null) { - item.text = pieceListItem.label; - } - - if (pieceListItem.hasOwnProperty('value')) { - var value = item.value = pieceListItem.value; - item.interval = [value, value]; - item.close = [1, 1]; - } else { - // `min` `max` is legacy option. - // `lt` `gt` `lte` `gte` is recommended. - var interval = item.interval = []; - var close_1 = item.close = [0, 0]; - var closeList = [1, 0, 1]; - var infinityList = [-Infinity, Infinity]; - var useMinMax = []; - - for (var lg = 0; lg < 2; lg++) { - var names = [['gte', 'gt', 'min'], ['lte', 'lt', 'max']][lg]; - - for (var i = 0; i < 3 && interval[lg] == null; i++) { - interval[lg] = pieceListItem[names[i]]; - close_1[lg] = closeList[i]; - useMinMax[lg] = i === 2; - } - - interval[lg] == null && (interval[lg] = infinityList[lg]); - } - - useMinMax[0] && interval[1] === Infinity && (close_1[0] = 0); - useMinMax[1] && interval[0] === -Infinity && (close_1[1] = 0); - - if ("development" !== 'production') { - if (interval[0] > interval[1]) { - console.warn('Piece ' + index + 'is illegal: ' + interval + ' lower bound should not greater then uppper bound.'); - } - } - - if (interval[0] === interval[1] && close_1[0] && close_1[1]) { - // Consider: [{min: 5, max: 5, visual: {...}}, {min: 0, max: 5}], - // we use value to lift the priority when min === max - item.value = interval[0]; - } - } - - item.visual = VisualMapping.retrieveVisuals(pieceListItem); - outPieceList.push(item); - }, this); // See "Order Rule". - - normalizeReverse(thisOption, outPieceList); // Only pieces - - reformIntervals(outPieceList); - each(outPieceList, function (piece) { - var close = piece.close; - var edgeSymbols = [['<', '≤'][close[1]], ['>', '≥'][close[0]]]; - piece.text = piece.text || this.formatValueText(piece.value != null ? piece.value : piece.interval, false, edgeSymbols); - }, this); - } - }; - - function normalizeReverse(thisOption, pieceList) { - var inverse = thisOption.inverse; - - if (thisOption.orient === 'vertical' ? !inverse : inverse) { - pieceList.reverse(); - } - } - - var PiecewiseVisualMapView = - /** @class */ - function (_super) { - __extends(PiecewiseVisualMapView, _super); - - function PiecewiseVisualMapView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = PiecewiseVisualMapView.type; - return _this; - } - - PiecewiseVisualMapView.prototype.doRender = function () { - var thisGroup = this.group; - thisGroup.removeAll(); - var visualMapModel = this.visualMapModel; - var textGap = visualMapModel.get('textGap'); - var textStyleModel = visualMapModel.textStyleModel; - var textFont = textStyleModel.getFont(); - var textFill = textStyleModel.getTextColor(); - - var itemAlign = this._getItemAlign(); - - var itemSize = visualMapModel.itemSize; - - var viewData = this._getViewData(); - - var endsText = viewData.endsText; - var showLabel = retrieve(visualMapModel.get('showLabel', true), !endsText); - endsText && this._renderEndsText(thisGroup, endsText[0], itemSize, showLabel, itemAlign); - each(viewData.viewPieceList, function (item) { - var piece = item.piece; - var itemGroup = new Group(); - itemGroup.onclick = bind(this._onItemClick, this, piece); - - this._enableHoverLink(itemGroup, item.indexInModelPieceList); // TODO Category - - - var representValue = visualMapModel.getRepresentValue(piece); - - this._createItemSymbol(itemGroup, representValue, [0, 0, itemSize[0], itemSize[1]]); - - if (showLabel) { - var visualState = this.visualMapModel.getValueState(representValue); - itemGroup.add(new ZRText({ - style: { - x: itemAlign === 'right' ? -textGap : itemSize[0] + textGap, - y: itemSize[1] / 2, - text: piece.text, - verticalAlign: 'middle', - align: itemAlign, - font: textFont, - fill: textFill, - opacity: visualState === 'outOfRange' ? 0.5 : 1 - } - })); - } - - thisGroup.add(itemGroup); - }, this); - endsText && this._renderEndsText(thisGroup, endsText[1], itemSize, showLabel, itemAlign); - box(visualMapModel.get('orient'), thisGroup, visualMapModel.get('itemGap')); - this.renderBackground(thisGroup); - this.positionGroup(thisGroup); - }; - - PiecewiseVisualMapView.prototype._enableHoverLink = function (itemGroup, pieceIndex) { - var _this = this; - - itemGroup.on('mouseover', function () { - return onHoverLink('highlight'); - }).on('mouseout', function () { - return onHoverLink('downplay'); - }); - - var onHoverLink = function (method) { - var visualMapModel = _this.visualMapModel; // TODO: TYPE More detailed action types - - visualMapModel.option.hoverLink && _this.api.dispatchAction({ - type: method, - batch: makeHighDownBatch(visualMapModel.findTargetDataIndices(pieceIndex), visualMapModel) - }); - }; - }; - - PiecewiseVisualMapView.prototype._getItemAlign = function () { - var visualMapModel = this.visualMapModel; - var modelOption = visualMapModel.option; - - if (modelOption.orient === 'vertical') { - return getItemAlign(visualMapModel, this.api, visualMapModel.itemSize); - } else { - // horizontal, most case left unless specifying right. - var align = modelOption.align; - - if (!align || align === 'auto') { - align = 'left'; - } - - return align; - } - }; - - PiecewiseVisualMapView.prototype._renderEndsText = function (group, text, itemSize, showLabel, itemAlign) { - if (!text) { - return; - } - - var itemGroup = new Group(); - var textStyleModel = this.visualMapModel.textStyleModel; - itemGroup.add(new ZRText({ - style: createTextStyle(textStyleModel, { - x: showLabel ? itemAlign === 'right' ? itemSize[0] : 0 : itemSize[0] / 2, - y: itemSize[1] / 2, - verticalAlign: 'middle', - align: showLabel ? itemAlign : 'center', - text: text - }) - })); - group.add(itemGroup); - }; - /** - * @private - * @return {Object} {peiceList, endsText} The order is the same as screen pixel order. - */ - - - PiecewiseVisualMapView.prototype._getViewData = function () { - var visualMapModel = this.visualMapModel; - var viewPieceList = map(visualMapModel.getPieceList(), function (piece, index) { - return { - piece: piece, - indexInModelPieceList: index - }; - }); - var endsText = visualMapModel.get('text'); // Consider orient and inverse. - - var orient = visualMapModel.get('orient'); - var inverse = visualMapModel.get('inverse'); // Order of model pieceList is always [low, ..., high] - - if (orient === 'horizontal' ? inverse : !inverse) { - viewPieceList.reverse(); - } // Origin order of endsText is [high, low] - else if (endsText) { - endsText = endsText.slice().reverse(); - } - - return { - viewPieceList: viewPieceList, - endsText: endsText - }; - }; - - PiecewiseVisualMapView.prototype._createItemSymbol = function (group, representValue, shapeParam) { - group.add(createSymbol( // symbol will be string - this.getControllerVisual(representValue, 'symbol'), shapeParam[0], shapeParam[1], shapeParam[2], shapeParam[3], // color will be string - this.getControllerVisual(representValue, 'color'))); - }; - - PiecewiseVisualMapView.prototype._onItemClick = function (piece) { - var visualMapModel = this.visualMapModel; - var option = visualMapModel.option; - var selectedMode = option.selectedMode; - - if (!selectedMode) { - return; - } - - var selected = clone(option.selected); - var newKey = visualMapModel.getSelectedMapKey(piece); - - if (selectedMode === 'single' || selectedMode === true) { - selected[newKey] = true; - each(selected, function (o, key) { - selected[key] = key === newKey; - }); - } else { - selected[newKey] = !selected[newKey]; - } - - this.api.dispatchAction({ - type: 'selectDataRange', - from: this.uid, - visualMapId: this.visualMapModel.id, - selected: selected - }); - }; - - PiecewiseVisualMapView.type = 'visualMap.piecewise'; - return PiecewiseVisualMapView; - }(VisualMapView); - - function install$O(registers) { - registers.registerComponentModel(PiecewiseModel); - registers.registerComponentView(PiecewiseVisualMapView); - installCommon$1(registers); - } - - function install$P(registers) { - use(install$N); - use(install$O); // Do not install './dataZoomSelect', - // since it only work for toolbox dataZoom. - } - - var DEFAULT_OPTION = { - label: { - enabled: true - }, - decal: { - show: false - } - }; - var inner$l = makeInner(); - var decalPaletteScope = {}; - function ariaVisual(ecModel, api) { - var ariaModel = ecModel.getModel('aria'); // See "area enabled" detection code in `GlobalModel.ts`. - - if (!ariaModel.get('enabled')) { - return; - } - - var defaultOption = clone(DEFAULT_OPTION); - merge(defaultOption.label, ecModel.getLocaleModel().get('aria'), false); - merge(ariaModel.option, defaultOption, false); - setDecal(); - setLabel(); - - function setDecal() { - var decalModel = ariaModel.getModel('decal'); - var useDecal = decalModel.get('show'); - - if (useDecal) { - // Each type of series use one scope. - // Pie and funnel are using different scopes. - var paletteScopeGroupByType_1 = createHashMap(); - ecModel.eachSeries(function (seriesModel) { - if (seriesModel.isColorBySeries()) { - return; - } - - var decalScope = paletteScopeGroupByType_1.get(seriesModel.type); - - if (!decalScope) { - decalScope = {}; - paletteScopeGroupByType_1.set(seriesModel.type, decalScope); - } - - inner$l(seriesModel).scope = decalScope; - }); - ecModel.eachRawSeries(function (seriesModel) { - if (ecModel.isSeriesFiltered(seriesModel)) { - return; - } - - if (isFunction(seriesModel.enableAriaDecal)) { - // Let series define how to use decal palette on data - seriesModel.enableAriaDecal(); - return; - } - - var data = seriesModel.getData(); - - if (!seriesModel.isColorBySeries()) { - var dataAll_1 = seriesModel.getRawData(); - var idxMap_1 = {}; - var decalScope_1 = inner$l(seriesModel).scope; - data.each(function (idx) { - var rawIdx = data.getRawIndex(idx); - idxMap_1[rawIdx] = idx; - }); - var dataCount_1 = dataAll_1.count(); - dataAll_1.each(function (rawIdx) { - var idx = idxMap_1[rawIdx]; - var name = dataAll_1.getName(rawIdx) || rawIdx + ''; - var paletteDecal = getDecalFromPalette(seriesModel.ecModel, name, decalScope_1, dataCount_1); - var specifiedDecal = data.getItemVisual(idx, 'decal'); - data.setItemVisual(idx, 'decal', mergeDecal(specifiedDecal, paletteDecal)); - }); - } else { - var paletteDecal = getDecalFromPalette(seriesModel.ecModel, seriesModel.name, decalPaletteScope, ecModel.getSeriesCount()); - var specifiedDecal = data.getVisual('decal'); - data.setVisual('decal', mergeDecal(specifiedDecal, paletteDecal)); - } - - function mergeDecal(specifiedDecal, paletteDecal) { - // Merge decal from palette to decal from itemStyle. - // User do not need to specify all of the decal props. - var resultDecal = specifiedDecal ? extend(extend({}, paletteDecal), specifiedDecal) : paletteDecal; - resultDecal.dirty = true; - return resultDecal; - } - }); - } - } - - function setLabel() { - var labelLocale = ecModel.getLocaleModel().get('aria'); - var labelModel = ariaModel.getModel('label'); - labelModel.option = defaults(labelModel.option, labelLocale); - - if (!labelModel.get('enabled')) { - return; - } - - var dom = api.getZr().dom; - - if (labelModel.get('description')) { - dom.setAttribute('aria-label', labelModel.get('description')); - return; - } - - var seriesCnt = ecModel.getSeriesCount(); - var maxDataCnt = labelModel.get(['data', 'maxCount']) || 10; - var maxSeriesCnt = labelModel.get(['series', 'maxCount']) || 10; - var displaySeriesCnt = Math.min(seriesCnt, maxSeriesCnt); - var ariaLabel; - - if (seriesCnt < 1) { - // No series, no aria label - return; - } else { - var title = getTitle(); - - if (title) { - var withTitle = labelModel.get(['general', 'withTitle']); - ariaLabel = replace(withTitle, { - title: title - }); - } else { - ariaLabel = labelModel.get(['general', 'withoutTitle']); - } - - var seriesLabels_1 = []; - var prefix = seriesCnt > 1 ? labelModel.get(['series', 'multiple', 'prefix']) : labelModel.get(['series', 'single', 'prefix']); - ariaLabel += replace(prefix, { - seriesCount: seriesCnt - }); - ecModel.eachSeries(function (seriesModel, idx) { - if (idx < displaySeriesCnt) { - var seriesLabel = void 0; - var seriesName = seriesModel.get('name'); - var withName = seriesName ? 'withName' : 'withoutName'; - seriesLabel = seriesCnt > 1 ? labelModel.get(['series', 'multiple', withName]) : labelModel.get(['series', 'single', withName]); - seriesLabel = replace(seriesLabel, { - seriesId: seriesModel.seriesIndex, - seriesName: seriesModel.get('name'), - seriesType: getSeriesTypeName(seriesModel.subType) - }); - var data = seriesModel.getData(); - - if (data.count() > maxDataCnt) { - // Show part of data - var partialLabel = labelModel.get(['data', 'partialData']); - seriesLabel += replace(partialLabel, { - displayCnt: maxDataCnt - }); - } else { - seriesLabel += labelModel.get(['data', 'allData']); - } - - var middleSeparator_1 = labelModel.get(['data', 'separator', 'middle']); - var endSeparator_1 = labelModel.get(['data', 'separator', 'end']); - var dataLabels = []; - - for (var i = 0; i < data.count(); i++) { - if (i < maxDataCnt) { - var name_1 = data.getName(i); - var value = data.getValues(i); - var dataLabel = labelModel.get(['data', name_1 ? 'withName' : 'withoutName']); - dataLabels.push(replace(dataLabel, { - name: name_1, - value: value.join(middleSeparator_1) - })); - } - } - - seriesLabel += dataLabels.join(middleSeparator_1) + endSeparator_1; - seriesLabels_1.push(seriesLabel); - } - }); - var separatorModel = labelModel.getModel(['series', 'multiple', 'separator']); - var middleSeparator = separatorModel.get('middle'); - var endSeparator = separatorModel.get('end'); - ariaLabel += seriesLabels_1.join(middleSeparator) + endSeparator; - dom.setAttribute('aria-label', ariaLabel); - } - } - - function replace(str, keyValues) { - if (!isString(str)) { - return str; - } - - var result = str; - each(keyValues, function (value, key) { - result = result.replace(new RegExp('\\{\\s*' + key + '\\s*\\}', 'g'), value); - }); - return result; - } - - function getTitle() { - var title = ecModel.get('title'); - - if (title && title.length) { - title = title[0]; - } - - return title && title.text; - } - - function getSeriesTypeName(type) { - return ecModel.getLocaleModel().get(['series', 'typeNames'])[type] || '自定义图'; - } - } - - function ariaPreprocessor(option) { - if (!option || !option.aria) { - return; - } - - var aria = option.aria; // aria.show is deprecated and should use aria.enabled instead - - if (aria.show != null) { - aria.enabled = aria.show; - } - - aria.label = aria.label || {}; // move description, general, series, data to be under aria.label - - each(['description', 'general', 'series', 'data'], function (name) { - if (aria[name] != null) { - aria.label[name] = aria[name]; - } - }); - } - - function install$Q(registers) { - registers.registerPreprocessor(ariaPreprocessor); - registers.registerVisual(registers.PRIORITY.VISUAL.ARIA, ariaVisual); - } - - var RELATIONAL_EXPRESSION_OP_ALIAS_MAP = { - value: 'eq', - // PENDING: not good for literal semantic? - '<': 'lt', - '<=': 'lte', - '>': 'gt', - '>=': 'gte', - '=': 'eq', - '!=': 'ne', - '<>': 'ne' // Might be misleading for sake of the difference between '==' and '===', - // so don't support them. - // '==': 'eq', - // '===': 'seq', - // '!==': 'sne' - // PENDING: Whether support some common alias "ge", "le", "neq"? - // ge: 'gte', - // le: 'lte', - // neq: 'ne', - - }; // type RelationalExpressionOpEvaluate = (tarVal: unknown, condVal: unknown) => boolean; - - var RegExpEvaluator = - /** @class */ - function () { - function RegExpEvaluator(rVal) { - // Support condVal: RegExp | string - var condValue = this._condVal = isString(rVal) ? new RegExp(rVal) : isRegExp(rVal) ? rVal : null; - - if (condValue == null) { - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = makePrintable('Illegal regexp', rVal, 'in'); - } - - throwError(errMsg); - } - } - - RegExpEvaluator.prototype.evaluate = function (lVal) { - var type = typeof lVal; - return isString(type) ? this._condVal.test(lVal) : isNumber(type) ? this._condVal.test(lVal + '') : false; - }; - - return RegExpEvaluator; - }(); - - var ConstConditionInternal = - /** @class */ - function () { - function ConstConditionInternal() {} - - ConstConditionInternal.prototype.evaluate = function () { - return this.value; - }; - - return ConstConditionInternal; - }(); - - var AndConditionInternal = - /** @class */ - function () { - function AndConditionInternal() {} - - AndConditionInternal.prototype.evaluate = function () { - var children = this.children; - - for (var i = 0; i < children.length; i++) { - if (!children[i].evaluate()) { - return false; - } - } - - return true; - }; - - return AndConditionInternal; - }(); - - var OrConditionInternal = - /** @class */ - function () { - function OrConditionInternal() {} - - OrConditionInternal.prototype.evaluate = function () { - var children = this.children; - - for (var i = 0; i < children.length; i++) { - if (children[i].evaluate()) { - return true; - } - } - - return false; - }; - - return OrConditionInternal; - }(); - - var NotConditionInternal = - /** @class */ - function () { - function NotConditionInternal() {} - - NotConditionInternal.prototype.evaluate = function () { - return !this.child.evaluate(); - }; - - return NotConditionInternal; - }(); - - var RelationalConditionInternal = - /** @class */ - function () { - function RelationalConditionInternal() {} - - RelationalConditionInternal.prototype.evaluate = function () { - var needParse = !!this.valueParser; // Call getValue with no `this`. - - var getValue = this.getValue; - var tarValRaw = getValue(this.valueGetterParam); - var tarValParsed = needParse ? this.valueParser(tarValRaw) : null; // Relational cond follow "and" logic internally. - - for (var i = 0; i < this.subCondList.length; i++) { - if (!this.subCondList[i].evaluate(needParse ? tarValParsed : tarValRaw)) { - return false; - } - } - - return true; - }; - - return RelationalConditionInternal; - }(); - - function parseOption(exprOption, getters) { - if (exprOption === true || exprOption === false) { - var cond = new ConstConditionInternal(); - cond.value = exprOption; - return cond; - } - - var errMsg = ''; - - if (!isObjectNotArray(exprOption)) { - if ("development" !== 'production') { - errMsg = makePrintable('Illegal config. Expect a plain object but actually', exprOption); - } - - throwError(errMsg); - } - - if (exprOption.and) { - return parseAndOrOption('and', exprOption, getters); - } else if (exprOption.or) { - return parseAndOrOption('or', exprOption, getters); - } else if (exprOption.not) { - return parseNotOption(exprOption, getters); - } - - return parseRelationalOption(exprOption, getters); - } - - function parseAndOrOption(op, exprOption, getters) { - var subOptionArr = exprOption[op]; - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = makePrintable('"and"/"or" condition should only be `' + op + ': [...]` and must not be empty array.', 'Illegal condition:', exprOption); - } - - if (!isArray(subOptionArr)) { - throwError(errMsg); - } - - if (!subOptionArr.length) { - throwError(errMsg); - } - - var cond = op === 'and' ? new AndConditionInternal() : new OrConditionInternal(); - cond.children = map(subOptionArr, function (subOption) { - return parseOption(subOption, getters); - }); - - if (!cond.children.length) { - throwError(errMsg); - } - - return cond; - } - - function parseNotOption(exprOption, getters) { - var subOption = exprOption.not; - var errMsg = ''; - - if ("development" !== 'production') { - errMsg = makePrintable('"not" condition should only be `not: {}`.', 'Illegal condition:', exprOption); - } - - if (!isObjectNotArray(subOption)) { - throwError(errMsg); - } - - var cond = new NotConditionInternal(); - cond.child = parseOption(subOption, getters); - - if (!cond.child) { - throwError(errMsg); - } - - return cond; - } - - function parseRelationalOption(exprOption, getters) { - var errMsg = ''; - var valueGetterParam = getters.prepareGetValue(exprOption); - var subCondList = []; - var exprKeys = keys(exprOption); - var parserName = exprOption.parser; - var valueParser = parserName ? getRawValueParser(parserName) : null; - - for (var i = 0; i < exprKeys.length; i++) { - var keyRaw = exprKeys[i]; - - if (keyRaw === 'parser' || getters.valueGetterAttrMap.get(keyRaw)) { - continue; - } - - var op = hasOwn(RELATIONAL_EXPRESSION_OP_ALIAS_MAP, keyRaw) ? RELATIONAL_EXPRESSION_OP_ALIAS_MAP[keyRaw] : keyRaw; - var condValueRaw = exprOption[keyRaw]; - var condValueParsed = valueParser ? valueParser(condValueRaw) : condValueRaw; - var evaluator = createFilterComparator(op, condValueParsed) || op === 'reg' && new RegExpEvaluator(condValueParsed); - - if (!evaluator) { - if ("development" !== 'production') { - errMsg = makePrintable('Illegal relational operation: "' + keyRaw + '" in condition:', exprOption); - } - - throwError(errMsg); - } - - subCondList.push(evaluator); - } - - if (!subCondList.length) { - if ("development" !== 'production') { - errMsg = makePrintable('Relational condition must have at least one operator.', 'Illegal condition:', exprOption); - } // No relational operator always disabled in case of dangers result. - - - throwError(errMsg); - } - - var cond = new RelationalConditionInternal(); - cond.valueGetterParam = valueGetterParam; - cond.valueParser = valueParser; - cond.getValue = getters.getValue; - cond.subCondList = subCondList; - return cond; - } - - function isObjectNotArray(val) { - return isObject(val) && !isArrayLike(val); - } - - var ConditionalExpressionParsed = - /** @class */ - function () { - function ConditionalExpressionParsed(exprOption, getters) { - this._cond = parseOption(exprOption, getters); - } - - ConditionalExpressionParsed.prototype.evaluate = function () { - return this._cond.evaluate(); - }; - - return ConditionalExpressionParsed; - }(); - function parseConditionalExpression(exprOption, getters) { - return new ConditionalExpressionParsed(exprOption, getters); - } - - var filterTransform = { - type: 'echarts:filter', - // PENDING: enhance to filter by index rather than create new data - transform: function (params) { - // [Caveat] Fail-Fast: - // Do not return the whole dataset unless user config indicates it explicitly. - // For example, if no condition is specified by mistake, returning an empty result - // is better than returning the entire raw source for the user to find the mistake. - var upstream = params.upstream; - var rawItem; - var condition = parseConditionalExpression(params.config, { - valueGetterAttrMap: createHashMap({ - dimension: true - }), - prepareGetValue: function (exprOption) { - var errMsg = ''; - var dimLoose = exprOption.dimension; - - if (!hasOwn(exprOption, 'dimension')) { - if ("development" !== 'production') { - errMsg = makePrintable('Relation condition must has prop "dimension" specified.', 'Illegal condition:', exprOption); - } - - throwError(errMsg); - } - - var dimInfo = upstream.getDimensionInfo(dimLoose); - - if (!dimInfo) { - if ("development" !== 'production') { - errMsg = makePrintable('Can not find dimension info via: ' + dimLoose + '.\n', 'Existing dimensions: ', upstream.cloneAllDimensionInfo(), '.\n', 'Illegal condition:', exprOption, '.\n'); - } - - throwError(errMsg); - } - - return { - dimIdx: dimInfo.index - }; - }, - getValue: function (param) { - return upstream.retrieveValueFromItem(rawItem, param.dimIdx); - } - }); - var resultData = []; - - for (var i = 0, len = upstream.count(); i < len; i++) { - rawItem = upstream.getRawDataItem(i); - - if (condition.evaluate()) { - resultData.push(rawItem); - } - } - - return { - data: resultData - }; - } - }; - - var sampleLog = ''; - - if ("development" !== 'production') { - sampleLog = ['Valid config is like:', '{ dimension: "age", order: "asc" }', 'or [{ dimension: "age", order: "asc"], { dimension: "date", order: "desc" }]'].join(' '); - } - - var sortTransform = { - type: 'echarts:sort', - transform: function (params) { - var upstream = params.upstream; - var config = params.config; - var errMsg = ''; // Normalize - // const orderExprList: OrderExpression[] = isArray(config[0]) - // ? config as OrderExpression[] - // : [config as OrderExpression]; - - var orderExprList = normalizeToArray(config); - - if (!orderExprList.length) { - if ("development" !== 'production') { - errMsg = 'Empty `config` in sort transform.'; - } - - throwError(errMsg); - } - - var orderDefList = []; - each(orderExprList, function (orderExpr) { - var dimLoose = orderExpr.dimension; - var order = orderExpr.order; - var parserName = orderExpr.parser; - var incomparable = orderExpr.incomparable; - - if (dimLoose == null) { - if ("development" !== 'production') { - errMsg = 'Sort transform config must has "dimension" specified.' + sampleLog; - } - - throwError(errMsg); - } - - if (order !== 'asc' && order !== 'desc') { - if ("development" !== 'production') { - errMsg = 'Sort transform config must has "order" specified.' + sampleLog; - } - - throwError(errMsg); - } - - if (incomparable && incomparable !== 'min' && incomparable !== 'max') { - var errMsg_1 = ''; - - if ("development" !== 'production') { - errMsg_1 = 'incomparable must be "min" or "max" rather than "' + incomparable + '".'; - } - - throwError(errMsg_1); - } - - if (order !== 'asc' && order !== 'desc') { - var errMsg_2 = ''; - - if ("development" !== 'production') { - errMsg_2 = 'order must be "asc" or "desc" rather than "' + order + '".'; - } - - throwError(errMsg_2); - } - - var dimInfo = upstream.getDimensionInfo(dimLoose); - - if (!dimInfo) { - if ("development" !== 'production') { - errMsg = makePrintable('Can not find dimension info via: ' + dimLoose + '.\n', 'Existing dimensions: ', upstream.cloneAllDimensionInfo(), '.\n', 'Illegal config:', orderExpr, '.\n'); - } - - throwError(errMsg); - } - - var parser = parserName ? getRawValueParser(parserName) : null; - - if (parserName && !parser) { - if ("development" !== 'production') { - errMsg = makePrintable('Invalid parser name ' + parserName + '.\n', 'Illegal config:', orderExpr, '.\n'); - } - - throwError(errMsg); - } - - orderDefList.push({ - dimIdx: dimInfo.index, - parser: parser, - comparator: new SortOrderComparator(order, incomparable) - }); - }); // TODO: support it? - - var sourceFormat = upstream.sourceFormat; - - if (sourceFormat !== SOURCE_FORMAT_ARRAY_ROWS && sourceFormat !== SOURCE_FORMAT_OBJECT_ROWS) { - if ("development" !== 'production') { - errMsg = 'sourceFormat "' + sourceFormat + '" is not supported yet'; - } - - throwError(errMsg); - } // Other upstream format are all array. - - - var resultData = []; - - for (var i = 0, len = upstream.count(); i < len; i++) { - resultData.push(upstream.getRawDataItem(i)); - } - - resultData.sort(function (item0, item1) { - for (var i = 0; i < orderDefList.length; i++) { - var orderDef = orderDefList[i]; - var val0 = upstream.retrieveValueFromItem(item0, orderDef.dimIdx); - var val1 = upstream.retrieveValueFromItem(item1, orderDef.dimIdx); - - if (orderDef.parser) { - val0 = orderDef.parser(val0); - val1 = orderDef.parser(val1); - } - - var result = orderDef.comparator.evaluate(val0, val1); - - if (result !== 0) { - return result; - } - } - - return 0; - }); - return { - data: resultData - }; - } - }; - - function install$R(registers) { - registers.registerTransform(filterTransform); - registers.registerTransform(sortTransform); - } - - var DatasetModel = - /** @class */ - function (_super) { - __extends(DatasetModel, _super); - - function DatasetModel() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'dataset'; - return _this; - } - - DatasetModel.prototype.init = function (option, parentModel, ecModel) { - _super.prototype.init.call(this, option, parentModel, ecModel); - - this._sourceManager = new SourceManager(this); - disableTransformOptionMerge(this); - }; - - DatasetModel.prototype.mergeOption = function (newOption, ecModel) { - _super.prototype.mergeOption.call(this, newOption, ecModel); - - disableTransformOptionMerge(this); - }; - - DatasetModel.prototype.optionUpdated = function () { - this._sourceManager.dirty(); - }; - - DatasetModel.prototype.getSourceManager = function () { - return this._sourceManager; - }; - - DatasetModel.type = 'dataset'; - DatasetModel.defaultOption = { - seriesLayoutBy: SERIES_LAYOUT_BY_COLUMN - }; - return DatasetModel; - }(ComponentModel); - - var DatasetView = - /** @class */ - function (_super) { - __extends(DatasetView, _super); - - function DatasetView() { - var _this = _super !== null && _super.apply(this, arguments) || this; - - _this.type = 'dataset'; - return _this; - } - - DatasetView.type = 'dataset'; - return DatasetView; - }(ComponentView); - - function install$S(registers) { - registers.registerComponentModel(DatasetModel); - registers.registerComponentView(DatasetView); - } - - var CMD$4 = PathProxy.CMD; - function aroundEqual(a, b) { - return Math.abs(a - b) < 1e-5; - } - function pathToBezierCurves(path) { - var data = path.data; - var len = path.len(); - var bezierArrayGroups = []; - var currentSubpath; - var xi = 0; - var yi = 0; - var x0 = 0; - var y0 = 0; - function createNewSubpath(x, y) { - if (currentSubpath && currentSubpath.length > 2) { - bezierArrayGroups.push(currentSubpath); - } - currentSubpath = [x, y]; - } - function addLine(x0, y0, x1, y1) { - if (!(aroundEqual(x0, x1) && aroundEqual(y0, y1))) { - currentSubpath.push(x0, y0, x1, y1, x1, y1); - } - } - function addArc(startAngle, endAngle, cx, cy, rx, ry) { - var delta = Math.abs(endAngle - startAngle); - var len = Math.tan(delta / 4) * 4 / 3; - var dir = endAngle < startAngle ? -1 : 1; - var c1 = Math.cos(startAngle); - var s1 = Math.sin(startAngle); - var c2 = Math.cos(endAngle); - var s2 = Math.sin(endAngle); - var x1 = c1 * rx + cx; - var y1 = s1 * ry + cy; - var x4 = c2 * rx + cx; - var y4 = s2 * ry + cy; - var hx = rx * len * dir; - var hy = ry * len * dir; - currentSubpath.push(x1 - hx * s1, y1 + hy * c1, x4 + hx * s2, y4 - hy * c2, x4, y4); - } - var x1; - var y1; - var x2; - var y2; - for (var i = 0; i < len;) { - var cmd = data[i++]; - var isFirst = i === 1; - if (isFirst) { - xi = data[i]; - yi = data[i + 1]; - x0 = xi; - y0 = yi; - if (cmd === CMD$4.L || cmd === CMD$4.C || cmd === CMD$4.Q) { - currentSubpath = [x0, y0]; - } - } - switch (cmd) { - case CMD$4.M: - xi = x0 = data[i++]; - yi = y0 = data[i++]; - createNewSubpath(x0, y0); - break; - case CMD$4.L: - x1 = data[i++]; - y1 = data[i++]; - addLine(xi, yi, x1, y1); - xi = x1; - yi = y1; - break; - case CMD$4.C: - currentSubpath.push(data[i++], data[i++], data[i++], data[i++], xi = data[i++], yi = data[i++]); - break; - case CMD$4.Q: - x1 = data[i++]; - y1 = data[i++]; - x2 = data[i++]; - y2 = data[i++]; - currentSubpath.push(xi + 2 / 3 * (x1 - xi), yi + 2 / 3 * (y1 - yi), x2 + 2 / 3 * (x1 - x2), y2 + 2 / 3 * (y1 - y2), x2, y2); - xi = x2; - yi = y2; - break; - case CMD$4.A: - var cx = data[i++]; - var cy = data[i++]; - var rx = data[i++]; - var ry = data[i++]; - var startAngle = data[i++]; - var endAngle = data[i++] + startAngle; - i += 1; - var anticlockwise = !data[i++]; - x1 = Math.cos(startAngle) * rx + cx; - y1 = Math.sin(startAngle) * ry + cy; - if (isFirst) { - x0 = x1; - y0 = y1; - createNewSubpath(x0, y0); - } - else { - addLine(xi, yi, x1, y1); - } - xi = Math.cos(endAngle) * rx + cx; - yi = Math.sin(endAngle) * ry + cy; - var step = (anticlockwise ? -1 : 1) * Math.PI / 2; - for (var angle = startAngle; anticlockwise ? angle > endAngle : angle < endAngle; angle += step) { - var nextAngle = anticlockwise ? Math.max(angle + step, endAngle) - : Math.min(angle + step, endAngle); - addArc(angle, nextAngle, cx, cy, rx, ry); - } - break; - case CMD$4.R: - x0 = xi = data[i++]; - y0 = yi = data[i++]; - x1 = x0 + data[i++]; - y1 = y0 + data[i++]; - createNewSubpath(x1, y0); - addLine(x1, y0, x1, y1); - addLine(x1, y1, x0, y1); - addLine(x0, y1, x0, y0); - addLine(x0, y0, x1, y0); - break; - case CMD$4.Z: - currentSubpath && addLine(xi, yi, x0, y0); - xi = x0; - yi = y0; - break; - } - } - if (currentSubpath && currentSubpath.length > 2) { - bezierArrayGroups.push(currentSubpath); - } - return bezierArrayGroups; - } - function adpativeBezier(x0, y0, x1, y1, x2, y2, x3, y3, out, scale) { - if (aroundEqual(x0, x1) && aroundEqual(y0, y1) && aroundEqual(x2, x3) && aroundEqual(y2, y3)) { - out.push(x3, y3); - return; - } - var PIXEL_DISTANCE = 2 / scale; - var PIXEL_DISTANCE_SQR = PIXEL_DISTANCE * PIXEL_DISTANCE; - var dx = x3 - x0; - var dy = y3 - y0; - var d = Math.sqrt(dx * dx + dy * dy); - dx /= d; - dy /= d; - var dx1 = x1 - x0; - var dy1 = y1 - y0; - var dx2 = x2 - x3; - var dy2 = y2 - y3; - var cp1LenSqr = dx1 * dx1 + dy1 * dy1; - var cp2LenSqr = dx2 * dx2 + dy2 * dy2; - if (cp1LenSqr < PIXEL_DISTANCE_SQR && cp2LenSqr < PIXEL_DISTANCE_SQR) { - out.push(x3, y3); - return; - } - var projLen1 = dx * dx1 + dy * dy1; - var projLen2 = -dx * dx2 - dy * dy2; - var d1Sqr = cp1LenSqr - projLen1 * projLen1; - var d2Sqr = cp2LenSqr - projLen2 * projLen2; - if (d1Sqr < PIXEL_DISTANCE_SQR && projLen1 >= 0 - && d2Sqr < PIXEL_DISTANCE_SQR && projLen2 >= 0) { - out.push(x3, y3); - return; - } - var tmpSegX = []; - var tmpSegY = []; - cubicSubdivide(x0, x1, x2, x3, 0.5, tmpSegX); - cubicSubdivide(y0, y1, y2, y3, 0.5, tmpSegY); - adpativeBezier(tmpSegX[0], tmpSegY[0], tmpSegX[1], tmpSegY[1], tmpSegX[2], tmpSegY[2], tmpSegX[3], tmpSegY[3], out, scale); - adpativeBezier(tmpSegX[4], tmpSegY[4], tmpSegX[5], tmpSegY[5], tmpSegX[6], tmpSegY[6], tmpSegX[7], tmpSegY[7], out, scale); - } - function pathToPolygons(path, scale) { - var bezierArrayGroups = pathToBezierCurves(path); - var polygons = []; - scale = scale || 1; - for (var i = 0; i < bezierArrayGroups.length; i++) { - var beziers = bezierArrayGroups[i]; - var polygon = []; - var x0 = beziers[0]; - var y0 = beziers[1]; - polygon.push(x0, y0); - for (var k = 2; k < beziers.length;) { - var x1 = beziers[k++]; - var y1 = beziers[k++]; - var x2 = beziers[k++]; - var y2 = beziers[k++]; - var x3 = beziers[k++]; - var y3 = beziers[k++]; - adpativeBezier(x0, y0, x1, y1, x2, y2, x3, y3, polygon, scale); - x0 = x3; - y0 = y3; - } - polygons.push(polygon); - } - return polygons; - } - - function getDividingGrids(dimSize, rowDim, count) { - var rowSize = dimSize[rowDim]; - var columnSize = dimSize[1 - rowDim]; - var ratio = Math.abs(rowSize / columnSize); - var rowCount = Math.ceil(Math.sqrt(ratio * count)); - var columnCount = Math.floor(count / rowCount); - if (columnCount === 0) { - columnCount = 1; - rowCount = count; - } - var grids = []; - for (var i = 0; i < rowCount; i++) { - grids.push(columnCount); - } - var currentCount = rowCount * columnCount; - var remained = count - currentCount; - if (remained > 0) { - for (var i = 0; i < remained; i++) { - grids[i % rowCount] += 1; - } - } - return grids; - } - function divideSector(sectorShape, count, outShapes) { - var r0 = sectorShape.r0; - var r = sectorShape.r; - var startAngle = sectorShape.startAngle; - var endAngle = sectorShape.endAngle; - var angle = Math.abs(endAngle - startAngle); - var arcLen = angle * r; - var deltaR = r - r0; - var isAngleRow = arcLen > Math.abs(deltaR); - var grids = getDividingGrids([arcLen, deltaR], isAngleRow ? 0 : 1, count); - var rowSize = (isAngleRow ? angle : deltaR) / grids.length; - for (var row = 0; row < grids.length; row++) { - var columnSize = (isAngleRow ? deltaR : angle) / grids[row]; - for (var column = 0; column < grids[row]; column++) { - var newShape = {}; - if (isAngleRow) { - newShape.startAngle = startAngle + rowSize * row; - newShape.endAngle = startAngle + rowSize * (row + 1); - newShape.r0 = r0 + columnSize * column; - newShape.r = r0 + columnSize * (column + 1); - } - else { - newShape.startAngle = startAngle + columnSize * column; - newShape.endAngle = startAngle + columnSize * (column + 1); - newShape.r0 = r0 + rowSize * row; - newShape.r = r0 + rowSize * (row + 1); - } - newShape.clockwise = sectorShape.clockwise; - newShape.cx = sectorShape.cx; - newShape.cy = sectorShape.cy; - outShapes.push(newShape); - } - } - } - function divideRect(rectShape, count, outShapes) { - var width = rectShape.width; - var height = rectShape.height; - var isHorizontalRow = width > height; - var grids = getDividingGrids([width, height], isHorizontalRow ? 0 : 1, count); - var rowSizeDim = isHorizontalRow ? 'width' : 'height'; - var columnSizeDim = isHorizontalRow ? 'height' : 'width'; - var rowDim = isHorizontalRow ? 'x' : 'y'; - var columnDim = isHorizontalRow ? 'y' : 'x'; - var rowSize = rectShape[rowSizeDim] / grids.length; - for (var row = 0; row < grids.length; row++) { - var columnSize = rectShape[columnSizeDim] / grids[row]; - for (var column = 0; column < grids[row]; column++) { - var newShape = {}; - newShape[rowDim] = row * rowSize; - newShape[columnDim] = column * columnSize; - newShape[rowSizeDim] = rowSize; - newShape[columnSizeDim] = columnSize; - newShape.x += rectShape.x; - newShape.y += rectShape.y; - outShapes.push(newShape); - } - } - } - function crossProduct2d$1(x1, y1, x2, y2) { - return x1 * y2 - x2 * y1; - } - function lineLineIntersect$1(a1x, a1y, a2x, a2y, b1x, b1y, b2x, b2y) { - var mx = a2x - a1x; - var my = a2y - a1y; - var nx = b2x - b1x; - var ny = b2y - b1y; - var nmCrossProduct = crossProduct2d$1(nx, ny, mx, my); - if (Math.abs(nmCrossProduct) < 1e-6) { - return null; - } - var b1a1x = a1x - b1x; - var b1a1y = a1y - b1y; - var p = crossProduct2d$1(b1a1x, b1a1y, nx, ny) / nmCrossProduct; - if (p < 0 || p > 1) { - return null; - } - return new Point(p * mx + a1x, p * my + a1y); - } - function projPtOnLine(pt, lineA, lineB) { - var dir = new Point(); - Point.sub(dir, lineB, lineA); - dir.normalize(); - var dir2 = new Point(); - Point.sub(dir2, pt, lineA); - var len = dir2.dot(dir); - return len; - } - function addToPoly(poly, pt) { - var last = poly[poly.length - 1]; - if (last && last[0] === pt[0] && last[1] === pt[1]) { - return; - } - poly.push(pt); - } - function splitPolygonByLine(points, lineA, lineB) { - var len = points.length; - var intersections = []; - for (var i = 0; i < len; i++) { - var p0 = points[i]; - var p1 = points[(i + 1) % len]; - var intersectionPt = lineLineIntersect$1(p0[0], p0[1], p1[0], p1[1], lineA.x, lineA.y, lineB.x, lineB.y); - if (intersectionPt) { - intersections.push({ - projPt: projPtOnLine(intersectionPt, lineA, lineB), - pt: intersectionPt, - idx: i - }); - } - } - if (intersections.length < 2) { - return [{ points: points }, { points: points }]; - } - intersections.sort(function (a, b) { - return a.projPt - b.projPt; - }); - var splitPt0 = intersections[0]; - var splitPt1 = intersections[intersections.length - 1]; - if (splitPt1.idx < splitPt0.idx) { - var tmp = splitPt0; - splitPt0 = splitPt1; - splitPt1 = tmp; - } - var splitPt0Arr = [splitPt0.pt.x, splitPt0.pt.y]; - var splitPt1Arr = [splitPt1.pt.x, splitPt1.pt.y]; - var newPolyA = [splitPt0Arr]; - var newPolyB = [splitPt1Arr]; - for (var i = splitPt0.idx + 1; i <= splitPt1.idx; i++) { - addToPoly(newPolyA, points[i].slice()); - } - addToPoly(newPolyA, splitPt1Arr); - addToPoly(newPolyA, splitPt0Arr); - for (var i = splitPt1.idx + 1; i <= splitPt0.idx + len; i++) { - addToPoly(newPolyB, points[i % len].slice()); - } - addToPoly(newPolyB, splitPt0Arr); - addToPoly(newPolyB, splitPt1Arr); - return [{ - points: newPolyA - }, { - points: newPolyB - }]; - } - function binaryDividePolygon(polygonShape) { - var points = polygonShape.points; - var min = []; - var max = []; - fromPoints(points, min, max); - var boundingRect = new BoundingRect(min[0], min[1], max[0] - min[0], max[1] - min[1]); - var width = boundingRect.width; - var height = boundingRect.height; - var x = boundingRect.x; - var y = boundingRect.y; - var pt0 = new Point(); - var pt1 = new Point(); - if (width > height) { - pt0.x = pt1.x = x + width / 2; - pt0.y = y; - pt1.y = y + height; - } - else { - pt0.y = pt1.y = y + height / 2; - pt0.x = x; - pt1.x = x + width; - } - return splitPolygonByLine(points, pt0, pt1); - } - function binaryDivideRecursive(divider, shape, count, out) { - if (count === 1) { - out.push(shape); - } - else { - var mid = Math.floor(count / 2); - var sub = divider(shape); - binaryDivideRecursive(divider, sub[0], mid, out); - binaryDivideRecursive(divider, sub[1], count - mid, out); - } - return out; - } - function clone$4(path, count) { - var paths = []; - for (var i = 0; i < count; i++) { - paths.push(clonePath(path)); - } - return paths; - } - function copyPathProps(source, target) { - target.setStyle(source.style); - target.z = source.z; - target.z2 = source.z2; - target.zlevel = source.zlevel; - } - function polygonConvert(points) { - var out = []; - for (var i = 0; i < points.length;) { - out.push([points[i++], points[i++]]); - } - return out; - } - function split(path, count) { - var outShapes = []; - var shape = path.shape; - var OutShapeCtor; - switch (path.type) { - case 'rect': - divideRect(shape, count, outShapes); - OutShapeCtor = Rect; - break; - case 'sector': - divideSector(shape, count, outShapes); - OutShapeCtor = Sector; - break; - case 'circle': - divideSector({ - r0: 0, r: shape.r, startAngle: 0, endAngle: Math.PI * 2, - cx: shape.cx, cy: shape.cy - }, count, outShapes); - OutShapeCtor = Sector; - break; - default: - var m = path.getComputedTransform(); - var scale = m ? Math.sqrt(Math.max(m[0] * m[0] + m[1] * m[1], m[2] * m[2] + m[3] * m[3])) : 1; - var polygons = map(pathToPolygons(path.getUpdatedPathProxy(), scale), function (poly) { return polygonConvert(poly); }); - var polygonCount = polygons.length; - if (polygonCount === 0) { - binaryDivideRecursive(binaryDividePolygon, { - points: polygons[0] - }, count, outShapes); - } - else if (polygonCount === count) { - for (var i = 0; i < polygonCount; i++) { - outShapes.push({ - points: polygons[i] - }); - } - } - else { - var totalArea_1 = 0; - var items = map(polygons, function (poly) { - var min = []; - var max = []; - fromPoints(poly, min, max); - var area = (max[1] - min[1]) * (max[0] - min[0]); - totalArea_1 += area; - return { poly: poly, area: area }; - }); - items.sort(function (a, b) { return b.area - a.area; }); - var left = count; - for (var i = 0; i < polygonCount; i++) { - var item = items[i]; - if (left <= 0) { - break; - } - var selfCount = i === polygonCount - 1 - ? left - : Math.ceil(item.area / totalArea_1 * count); - if (selfCount < 0) { - continue; - } - binaryDivideRecursive(binaryDividePolygon, { - points: item.poly - }, selfCount, outShapes); - left -= selfCount; - } - } - OutShapeCtor = Polygon; - break; - } - if (!OutShapeCtor) { - return clone$4(path, count); - } - var out = []; - for (var i = 0; i < outShapes.length; i++) { - var subPath = new OutShapeCtor(); - subPath.setShape(outShapes[i]); - copyPathProps(path, subPath); - out.push(subPath); - } - return out; - } - - function alignSubpath(subpath1, subpath2) { - var len1 = subpath1.length; - var len2 = subpath2.length; - if (len1 === len2) { - return [subpath1, subpath2]; - } - var tmpSegX = []; - var tmpSegY = []; - var shorterPath = len1 < len2 ? subpath1 : subpath2; - var shorterLen = Math.min(len1, len2); - var diff = Math.abs(len2 - len1) / 6; - var shorterBezierCount = (shorterLen - 2) / 6; - var eachCurveSubDivCount = Math.ceil(diff / shorterBezierCount) + 1; - var newSubpath = [shorterPath[0], shorterPath[1]]; - var remained = diff; - for (var i = 2; i < shorterLen;) { - var x0 = shorterPath[i - 2]; - var y0 = shorterPath[i - 1]; - var x1 = shorterPath[i++]; - var y1 = shorterPath[i++]; - var x2 = shorterPath[i++]; - var y2 = shorterPath[i++]; - var x3 = shorterPath[i++]; - var y3 = shorterPath[i++]; - if (remained <= 0) { - newSubpath.push(x1, y1, x2, y2, x3, y3); - continue; - } - var actualSubDivCount = Math.min(remained, eachCurveSubDivCount - 1) + 1; - for (var k = 1; k <= actualSubDivCount; k++) { - var p = k / actualSubDivCount; - cubicSubdivide(x0, x1, x2, x3, p, tmpSegX); - cubicSubdivide(y0, y1, y2, y3, p, tmpSegY); - x0 = tmpSegX[3]; - y0 = tmpSegY[3]; - newSubpath.push(tmpSegX[1], tmpSegY[1], tmpSegX[2], tmpSegY[2], x0, y0); - x1 = tmpSegX[5]; - y1 = tmpSegY[5]; - x2 = tmpSegX[6]; - y2 = tmpSegY[6]; - } - remained -= actualSubDivCount - 1; - } - return shorterPath === subpath1 ? [newSubpath, subpath2] : [subpath1, newSubpath]; - } - function createSubpath(lastSubpathSubpath, otherSubpath) { - var len = lastSubpathSubpath.length; - var lastX = lastSubpathSubpath[len - 2]; - var lastY = lastSubpathSubpath[len - 1]; - var newSubpath = []; - for (var i = 0; i < otherSubpath.length;) { - newSubpath[i++] = lastX; - newSubpath[i++] = lastY; - } - return newSubpath; - } - function alignBezierCurves(array1, array2) { - var _a; - var lastSubpath1; - var lastSubpath2; - var newArray1 = []; - var newArray2 = []; - for (var i = 0; i < Math.max(array1.length, array2.length); i++) { - var subpath1 = array1[i]; - var subpath2 = array2[i]; - var newSubpath1 = void 0; - var newSubpath2 = void 0; - if (!subpath1) { - newSubpath1 = createSubpath(lastSubpath1 || subpath2, subpath2); - newSubpath2 = subpath2; - } - else if (!subpath2) { - newSubpath2 = createSubpath(lastSubpath2 || subpath1, subpath1); - newSubpath1 = subpath1; - } - else { - _a = alignSubpath(subpath1, subpath2), newSubpath1 = _a[0], newSubpath2 = _a[1]; - lastSubpath1 = newSubpath1; - lastSubpath2 = newSubpath2; - } - newArray1.push(newSubpath1); - newArray2.push(newSubpath2); - } - return [newArray1, newArray2]; - } - function centroid$1(array) { - var signedArea = 0; - var cx = 0; - var cy = 0; - var len = array.length; - for (var i = 0, j = len - 2; i < len; j = i, i += 2) { - var x0 = array[j]; - var y0 = array[j + 1]; - var x1 = array[i]; - var y1 = array[i + 1]; - var a = x0 * y1 - x1 * y0; - signedArea += a; - cx += (x0 + x1) * a; - cy += (y0 + y1) * a; - } - if (signedArea === 0) { - return [array[0] || 0, array[1] || 0]; - } - return [cx / signedArea / 3, cy / signedArea / 3, signedArea]; - } - function findBestRingOffset(fromSubBeziers, toSubBeziers, fromCp, toCp) { - var bezierCount = (fromSubBeziers.length - 2) / 6; - var bestScore = Infinity; - var bestOffset = 0; - var len = fromSubBeziers.length; - var len2 = len - 2; - for (var offset = 0; offset < bezierCount; offset++) { - var cursorOffset = offset * 6; - var score = 0; - for (var k = 0; k < len; k += 2) { - var idx = k === 0 ? cursorOffset : ((cursorOffset + k - 2) % len2 + 2); - var x0 = fromSubBeziers[idx] - fromCp[0]; - var y0 = fromSubBeziers[idx + 1] - fromCp[1]; - var x1 = toSubBeziers[k] - toCp[0]; - var y1 = toSubBeziers[k + 1] - toCp[1]; - var dx = x1 - x0; - var dy = y1 - y0; - score += dx * dx + dy * dy; - } - if (score < bestScore) { - bestScore = score; - bestOffset = offset; - } - } - return bestOffset; - } - function reverse(array) { - var newArr = []; - var len = array.length; - for (var i = 0; i < len; i += 2) { - newArr[i] = array[len - i - 2]; - newArr[i + 1] = array[len - i - 1]; - } - return newArr; - } - function findBestMorphingRotation(fromArr, toArr, searchAngleIteration, searchAngleRange) { - var result = []; - var fromNeedsReverse; - for (var i = 0; i < fromArr.length; i++) { - var fromSubpathBezier = fromArr[i]; - var toSubpathBezier = toArr[i]; - var fromCp = centroid$1(fromSubpathBezier); - var toCp = centroid$1(toSubpathBezier); - if (fromNeedsReverse == null) { - fromNeedsReverse = fromCp[2] < 0 !== toCp[2] < 0; - } - var newFromSubpathBezier = []; - var newToSubpathBezier = []; - var bestAngle = 0; - var bestScore = Infinity; - var tmpArr = []; - var len = fromSubpathBezier.length; - if (fromNeedsReverse) { - fromSubpathBezier = reverse(fromSubpathBezier); - } - var offset = findBestRingOffset(fromSubpathBezier, toSubpathBezier, fromCp, toCp) * 6; - var len2 = len - 2; - for (var k = 0; k < len2; k += 2) { - var idx = (offset + k) % len2 + 2; - newFromSubpathBezier[k + 2] = fromSubpathBezier[idx] - fromCp[0]; - newFromSubpathBezier[k + 3] = fromSubpathBezier[idx + 1] - fromCp[1]; - } - newFromSubpathBezier[0] = fromSubpathBezier[offset] - fromCp[0]; - newFromSubpathBezier[1] = fromSubpathBezier[offset + 1] - fromCp[1]; - if (searchAngleIteration > 0) { - var step = searchAngleRange / searchAngleIteration; - for (var angle = -searchAngleRange / 2; angle <= searchAngleRange / 2; angle += step) { - var sa = Math.sin(angle); - var ca = Math.cos(angle); - var score = 0; - for (var k = 0; k < fromSubpathBezier.length; k += 2) { - var x0 = newFromSubpathBezier[k]; - var y0 = newFromSubpathBezier[k + 1]; - var x1 = toSubpathBezier[k] - toCp[0]; - var y1 = toSubpathBezier[k + 1] - toCp[1]; - var newX1 = x1 * ca - y1 * sa; - var newY1 = x1 * sa + y1 * ca; - tmpArr[k] = newX1; - tmpArr[k + 1] = newY1; - var dx = newX1 - x0; - var dy = newY1 - y0; - score += dx * dx + dy * dy; - } - if (score < bestScore) { - bestScore = score; - bestAngle = angle; - for (var m = 0; m < tmpArr.length; m++) { - newToSubpathBezier[m] = tmpArr[m]; - } - } - } - } - else { - for (var i_1 = 0; i_1 < len; i_1 += 2) { - newToSubpathBezier[i_1] = toSubpathBezier[i_1] - toCp[0]; - newToSubpathBezier[i_1 + 1] = toSubpathBezier[i_1 + 1] - toCp[1]; - } - } - result.push({ - from: newFromSubpathBezier, - to: newToSubpathBezier, - fromCp: fromCp, - toCp: toCp, - rotation: -bestAngle - }); - } - return result; - } - function isCombineMorphing(path) { - return path.__isCombineMorphing; - } - var SAVED_METHOD_PREFIX = '__mOriginal_'; - function saveAndModifyMethod(obj, methodName, modifiers) { - var savedMethodName = SAVED_METHOD_PREFIX + methodName; - var originalMethod = obj[savedMethodName] || obj[methodName]; - if (!obj[savedMethodName]) { - obj[savedMethodName] = obj[methodName]; - } - var replace = modifiers.replace; - var after = modifiers.after; - var before = modifiers.before; - obj[methodName] = function () { - var args = arguments; - var res; - before && before.apply(this, args); - if (replace) { - res = replace.apply(this, args); - } - else { - res = originalMethod.apply(this, args); - } - after && after.apply(this, args); - return res; - }; - } - function restoreMethod(obj, methodName) { - var savedMethodName = SAVED_METHOD_PREFIX + methodName; - if (obj[savedMethodName]) { - obj[methodName] = obj[savedMethodName]; - obj[savedMethodName] = null; - } - } - function applyTransformOnBeziers(bezierCurves, mm) { - for (var i = 0; i < bezierCurves.length; i++) { - var subBeziers = bezierCurves[i]; - for (var k = 0; k < subBeziers.length;) { - var x = subBeziers[k]; - var y = subBeziers[k + 1]; - subBeziers[k++] = mm[0] * x + mm[2] * y + mm[4]; - subBeziers[k++] = mm[1] * x + mm[3] * y + mm[5]; - } - } - } - function prepareMorphPath(fromPath, toPath) { - var fromPathProxy = fromPath.getUpdatedPathProxy(); - var toPathProxy = toPath.getUpdatedPathProxy(); - var _a = alignBezierCurves(pathToBezierCurves(fromPathProxy), pathToBezierCurves(toPathProxy)), fromBezierCurves = _a[0], toBezierCurves = _a[1]; - var fromPathTransform = fromPath.getComputedTransform(); - var toPathTransform = toPath.getComputedTransform(); - function updateIdentityTransform() { - this.transform = null; - } - fromPathTransform && applyTransformOnBeziers(fromBezierCurves, fromPathTransform); - toPathTransform && applyTransformOnBeziers(toBezierCurves, toPathTransform); - saveAndModifyMethod(toPath, 'updateTransform', { replace: updateIdentityTransform }); - toPath.transform = null; - var morphingData = findBestMorphingRotation(fromBezierCurves, toBezierCurves, 10, Math.PI); - var tmpArr = []; - saveAndModifyMethod(toPath, 'buildPath', { replace: function (path) { - var t = toPath.__morphT; - var onet = 1 - t; - var newCp = []; - for (var i = 0; i < morphingData.length; i++) { - var item = morphingData[i]; - var from = item.from; - var to = item.to; - var angle = item.rotation * t; - var fromCp = item.fromCp; - var toCp = item.toCp; - var sa = Math.sin(angle); - var ca = Math.cos(angle); - lerp(newCp, fromCp, toCp, t); - for (var m = 0; m < from.length; m += 2) { - var x0_1 = from[m]; - var y0_1 = from[m + 1]; - var x1 = to[m]; - var y1 = to[m + 1]; - var x = x0_1 * onet + x1 * t; - var y = y0_1 * onet + y1 * t; - tmpArr[m] = (x * ca - y * sa) + newCp[0]; - tmpArr[m + 1] = (x * sa + y * ca) + newCp[1]; - } - var x0 = tmpArr[0]; - var y0 = tmpArr[1]; - path.moveTo(x0, y0); - for (var m = 2; m < from.length;) { - var x1 = tmpArr[m++]; - var y1 = tmpArr[m++]; - var x2 = tmpArr[m++]; - var y2 = tmpArr[m++]; - var x3 = tmpArr[m++]; - var y3 = tmpArr[m++]; - if (x0 === x1 && y0 === y1 && x2 === x3 && y2 === y3) { - path.lineTo(x3, y3); - } - else { - path.bezierCurveTo(x1, y1, x2, y2, x3, y3); - } - x0 = x3; - y0 = y3; - } - } - } }); - } - function morphPath(fromPath, toPath, animationOpts) { - if (!fromPath || !toPath) { - return toPath; - } - var oldDone = animationOpts.done; - var oldDuring = animationOpts.during; - prepareMorphPath(fromPath, toPath); - toPath.__morphT = 0; - function restoreToPath() { - restoreMethod(toPath, 'buildPath'); - restoreMethod(toPath, 'updateTransform'); - toPath.__morphT = -1; - toPath.createPathProxy(); - toPath.dirtyShape(); - } - toPath.animateTo({ - __morphT: 1 - }, defaults({ - during: function (p) { - toPath.dirtyShape(); - oldDuring && oldDuring(p); - }, - done: function () { - restoreToPath(); - oldDone && oldDone(); - } - }, animationOpts)); - return toPath; - } - function hilbert(x, y, minX, minY, maxX, maxY) { - var bits = 16; - x = (maxX === minX) ? 0 : Math.round(32767 * (x - minX) / (maxX - minX)); - y = (maxY === minY) ? 0 : Math.round(32767 * (y - minY) / (maxY - minY)); - var d = 0; - var tmp; - for (var s = (1 << bits) / 2; s > 0; s /= 2) { - var rx = 0; - var ry = 0; - if ((x & s) > 0) { - rx = 1; - } - if ((y & s) > 0) { - ry = 1; - } - d += s * s * ((3 * rx) ^ ry); - if (ry === 0) { - if (rx === 1) { - x = s - 1 - x; - y = s - 1 - y; - } - tmp = x; - x = y; - y = tmp; - } - } - return d; - } - function sortPaths(pathList) { - var xMin = Infinity; - var yMin = Infinity; - var xMax = -Infinity; - var yMax = -Infinity; - var cps = map(pathList, function (path) { - var rect = path.getBoundingRect(); - var m = path.getComputedTransform(); - var x = rect.x + rect.width / 2 + (m ? m[4] : 0); - var y = rect.y + rect.height / 2 + (m ? m[5] : 0); - xMin = Math.min(x, xMin); - yMin = Math.min(y, yMin); - xMax = Math.max(x, xMax); - yMax = Math.max(y, yMax); - return [x, y]; - }); - var items = map(cps, function (cp, idx) { - return { - cp: cp, - z: hilbert(cp[0], cp[1], xMin, yMin, xMax, yMax), - path: pathList[idx] - }; - }); - return items.sort(function (a, b) { return a.z - b.z; }).map(function (item) { return item.path; }); - } - function defaultDividePath(param) { - return split(param.path, param.count); - } - function createEmptyReturn() { - return { - fromIndividuals: [], - toIndividuals: [], - count: 0 - }; - } - function combineMorph(fromList, toPath, animationOpts) { - var fromPathList = []; - function addFromPath(fromList) { - for (var i = 0; i < fromList.length; i++) { - var from = fromList[i]; - if (isCombineMorphing(from)) { - addFromPath(from.childrenRef()); - } - else if (from instanceof Path) { - fromPathList.push(from); - } - } - } - addFromPath(fromList); - var separateCount = fromPathList.length; - if (!separateCount) { - return createEmptyReturn(); - } - var dividePath = animationOpts.dividePath || defaultDividePath; - var toSubPathList = dividePath({ - path: toPath, count: separateCount - }); - if (toSubPathList.length !== separateCount) { - console.error('Invalid morphing: unmatched splitted path'); - return createEmptyReturn(); - } - fromPathList = sortPaths(fromPathList); - toSubPathList = sortPaths(toSubPathList); - var oldDone = animationOpts.done; - var oldDuring = animationOpts.during; - var individualDelay = animationOpts.individualDelay; - var identityTransform = new Transformable(); - for (var i = 0; i < separateCount; i++) { - var from = fromPathList[i]; - var to = toSubPathList[i]; - to.parent = toPath; - to.copyTransform(identityTransform); - if (!individualDelay) { - prepareMorphPath(from, to); - } - } - toPath.__isCombineMorphing = true; - toPath.childrenRef = function () { - return toSubPathList; - }; - function addToSubPathListToZr(zr) { - for (var i = 0; i < toSubPathList.length; i++) { - toSubPathList[i].addSelfToZr(zr); - } - } - saveAndModifyMethod(toPath, 'addSelfToZr', { - after: function (zr) { - addToSubPathListToZr(zr); - } - }); - saveAndModifyMethod(toPath, 'removeSelfFromZr', { - after: function (zr) { - for (var i = 0; i < toSubPathList.length; i++) { - toSubPathList[i].removeSelfFromZr(zr); - } - } - }); - function restoreToPath() { - toPath.__isCombineMorphing = false; - toPath.__morphT = -1; - toPath.childrenRef = null; - restoreMethod(toPath, 'addSelfToZr'); - restoreMethod(toPath, 'removeSelfFromZr'); - } - var toLen = toSubPathList.length; - if (individualDelay) { - var animating_1 = toLen; - var eachDone = function () { - animating_1--; - if (animating_1 === 0) { - restoreToPath(); - oldDone && oldDone(); - } - }; - for (var i = 0; i < toLen; i++) { - var indivdualAnimationOpts = individualDelay ? defaults({ - delay: (animationOpts.delay || 0) + individualDelay(i, toLen, fromPathList[i], toSubPathList[i]), - done: eachDone - }, animationOpts) : animationOpts; - morphPath(fromPathList[i], toSubPathList[i], indivdualAnimationOpts); - } - } - else { - toPath.__morphT = 0; - toPath.animateTo({ - __morphT: 1 - }, defaults({ - during: function (p) { - for (var i = 0; i < toLen; i++) { - var child = toSubPathList[i]; - child.__morphT = toPath.__morphT; - child.dirtyShape(); - } - oldDuring && oldDuring(p); - }, - done: function () { - restoreToPath(); - for (var i = 0; i < fromList.length; i++) { - restoreMethod(fromList[i], 'updateTransform'); - } - oldDone && oldDone(); - } - }, animationOpts)); - } - if (toPath.__zr) { - addToSubPathListToZr(toPath.__zr); - } - return { - fromIndividuals: fromPathList, - toIndividuals: toSubPathList, - count: toLen - }; - } - function separateMorph(fromPath, toPathList, animationOpts) { - var toLen = toPathList.length; - var fromPathList = []; - var dividePath = animationOpts.dividePath || defaultDividePath; - function addFromPath(fromList) { - for (var i = 0; i < fromList.length; i++) { - var from = fromList[i]; - if (isCombineMorphing(from)) { - addFromPath(from.childrenRef()); - } - else if (from instanceof Path) { - fromPathList.push(from); - } - } - } - if (isCombineMorphing(fromPath)) { - addFromPath(fromPath.childrenRef()); - var fromLen = fromPathList.length; - if (fromLen < toLen) { - var k = 0; - for (var i = fromLen; i < toLen; i++) { - fromPathList.push(clonePath(fromPathList[k++ % fromLen])); - } - } - fromPathList.length = toLen; - } - else { - fromPathList = dividePath({ path: fromPath, count: toLen }); - var fromPathTransform = fromPath.getComputedTransform(); - for (var i = 0; i < fromPathList.length; i++) { - fromPathList[i].setLocalTransform(fromPathTransform); - } - if (fromPathList.length !== toLen) { - console.error('Invalid morphing: unmatched splitted path'); - return createEmptyReturn(); - } - } - fromPathList = sortPaths(fromPathList); - toPathList = sortPaths(toPathList); - var individualDelay = animationOpts.individualDelay; - for (var i = 0; i < toLen; i++) { - var indivdualAnimationOpts = individualDelay ? defaults({ - delay: (animationOpts.delay || 0) + individualDelay(i, toLen, fromPathList[i], toPathList[i]) - }, animationOpts) : animationOpts; - morphPath(fromPathList[i], toPathList[i], indivdualAnimationOpts); - } - return { - fromIndividuals: fromPathList, - toIndividuals: toPathList, - count: toPathList.length - }; - } - - function isMultiple(elements) { - return isArray(elements[0]); - } - - function prepareMorphBatches(one, many) { - var batches = []; - var batchCount = one.length; - - for (var i = 0; i < batchCount; i++) { - batches.push({ - one: one[i], - many: [] - }); - } - - for (var i = 0; i < many.length; i++) { - var len = many[i].length; - var k = void 0; - - for (k = 0; k < len; k++) { - batches[k % batchCount].many.push(many[i][k]); - } - } - - var off = 0; // If one has more paths than each one of many. average them. - - for (var i = batchCount - 1; i >= 0; i--) { - if (!batches[i].many.length) { - var moveFrom = batches[off].many; - - if (moveFrom.length <= 1) { - // Not enough - // Start from the first one. - if (off) { - off = 0; - } else { - return batches; - } - } - - var len = moveFrom.length; - var mid = Math.ceil(len / 2); - batches[i].many = moveFrom.slice(mid, len); - batches[off].many = moveFrom.slice(0, mid); - off++; - } - } - - return batches; - } - - var pathDividers = { - clone: function (params) { - var ret = []; // Fitting the alpha - - var approxOpacity = 1 - Math.pow(1 - params.path.style.opacity, 1 / params.count); - - for (var i = 0; i < params.count; i++) { - var cloned = clonePath(params.path); - cloned.setStyle('opacity', approxOpacity); - ret.push(cloned); - } - - return ret; - }, - // Use the default divider - split: null - }; - function applyMorphAnimation(from, to, divideShape, seriesModel, dataIndex, animateOtherProps) { - if (!from.length || !to.length) { - return; - } - - var updateAnimationCfg = getAnimationConfig('update', seriesModel, dataIndex); - - if (!(updateAnimationCfg && updateAnimationCfg.duration > 0)) { - return; - } - - var animationDelay = seriesModel.getModel('universalTransition').get('delay'); - var animationCfg = Object.assign({ - // Need to setToFinal so the further calculation based on the style can be correct. - // Like emphasis color. - setToFinal: true - }, updateAnimationCfg); - var many; - var one; - - if (isMultiple(from)) { - // manyToOne - many = from; - one = to; - } - - if (isMultiple(to)) { - // oneToMany - many = to; - one = from; - } - - function morphOneBatch(batch, fromIsMany, animateIndex, animateCount, forceManyOne) { - var batchMany = batch.many; - var batchOne = batch.one; - - if (batchMany.length === 1 && !forceManyOne) { - // Is one to one - var batchFrom = fromIsMany ? batchMany[0] : batchOne; - var batchTo = fromIsMany ? batchOne : batchMany[0]; - - if (isCombineMorphing(batchFrom)) { - // Keep doing combine animation. - morphOneBatch({ - many: [batchFrom], - one: batchTo - }, true, animateIndex, animateCount, true); - } else { - var individualAnimationCfg = animationDelay ? defaults({ - delay: animationDelay(animateIndex, animateCount) - }, animationCfg) : animationCfg; - morphPath(batchFrom, batchTo, individualAnimationCfg); - animateOtherProps(batchFrom, batchTo, batchFrom, batchTo, individualAnimationCfg); - } - } else { - var separateAnimationCfg = defaults({ - dividePath: pathDividers[divideShape], - individualDelay: animationDelay && function (idx, count, fromPath, toPath) { - return animationDelay(idx + animateIndex, animateCount); - } - }, animationCfg); - - var _a = fromIsMany ? combineMorph(batchMany, batchOne, separateAnimationCfg) : separateMorph(batchOne, batchMany, separateAnimationCfg), - fromIndividuals = _a.fromIndividuals, - toIndividuals = _a.toIndividuals; - - var count = fromIndividuals.length; - - for (var k = 0; k < count; k++) { - var individualAnimationCfg = animationDelay ? defaults({ - delay: animationDelay(k, count) - }, animationCfg) : animationCfg; - animateOtherProps(fromIndividuals[k], toIndividuals[k], fromIsMany ? batchMany[k] : batch.one, fromIsMany ? batch.one : batchMany[k], individualAnimationCfg); - } - } - } - - var fromIsMany = many ? many === from // Is one to one. If the path number not match. also needs do merge and separate morphing. - : from.length > to.length; - var morphBatches = many ? prepareMorphBatches(one, many) : prepareMorphBatches(fromIsMany ? to : from, [fromIsMany ? from : to]); - var animateCount = 0; - - for (var i = 0; i < morphBatches.length; i++) { - animateCount += morphBatches[i].many.length; - } - - var animateIndex = 0; - - for (var i = 0; i < morphBatches.length; i++) { - morphOneBatch(morphBatches[i], fromIsMany, animateIndex, animateCount); - animateIndex += morphBatches[i].many.length; - } - } - function getPathList(elements) { - if (!elements) { - return []; - } - - if (isArray(elements)) { - var pathList_1 = []; - - for (var i = 0; i < elements.length; i++) { - pathList_1.push(getPathList(elements[i])); - } - - return pathList_1; - } - - var pathList = []; - elements.traverse(function (el) { - if (el instanceof Path && !el.disableMorphing && !el.invisible && !el.ignore) { - pathList.push(el); - } - }); - return pathList; - } - - var DATA_COUNT_THRESHOLD = 1e4; - var getUniversalTransitionGlobalStore = makeInner(); - - function getGroupIdDimension(data) { - var dimensions = data.dimensions; - - for (var i = 0; i < dimensions.length; i++) { - var dimInfo = data.getDimensionInfo(dimensions[i]); - - if (dimInfo && dimInfo.otherDims.itemGroupId === 0) { - return dimensions[i]; - } - } - } - - function flattenDataDiffItems(list) { - var items = []; - each(list, function (seriesInfo) { - var data = seriesInfo.data; - - if (data.count() > DATA_COUNT_THRESHOLD) { - if ("development" !== 'production') { - warn('Universal transition is disabled on large data > 10k.'); - } - - return; - } - - var indices = data.getIndices(); - var groupDim = getGroupIdDimension(data); - - for (var dataIndex = 0; dataIndex < indices.length; dataIndex++) { - items.push({ - dataGroupId: seriesInfo.dataGroupId, - data: data, - dim: seriesInfo.dim || groupDim, - divide: seriesInfo.divide, - dataIndex: dataIndex - }); - } - }); - return items; - } - - function fadeInElement(newEl, newSeries, newIndex) { - newEl.traverse(function (el) { - if (el instanceof Path) { - // TODO use fade in animation for target element. - initProps(el, { - style: { - opacity: 0 - } - }, newSeries, { - dataIndex: newIndex, - isFrom: true - }); - } - }); - } - - function removeEl$1(el) { - if (el.parent) { - // Bake parent transform to element. - // So it can still have proper transform to transition after it's removed. - var computedTransform = el.getComputedTransform(); - el.setLocalTransform(computedTransform); - el.parent.remove(el); - } - } - - function stopAnimation(el) { - el.stopAnimation(); - - if (el.isGroup) { - el.traverse(function (child) { - child.stopAnimation(); - }); - } - } - - function animateElementStyles(el, dataIndex, seriesModel) { - var animationConfig = getAnimationConfig('update', seriesModel, dataIndex); - animationConfig && el.traverse(function (child) { - if (child instanceof Displayable) { - var oldStyle = getOldStyle(child); - - if (oldStyle) { - child.animateFrom({ - style: oldStyle - }, animationConfig); - } - } - }); - } - - function isAllIdSame(oldDiffItems, newDiffItems) { - var len = oldDiffItems.length; - - if (len !== newDiffItems.length) { - return false; - } - - for (var i = 0; i < len; i++) { - var oldItem = oldDiffItems[i]; - var newItem = newDiffItems[i]; - - if (oldItem.data.getId(oldItem.dataIndex) !== newItem.data.getId(newItem.dataIndex)) { - return false; - } - } - - return true; - } - - function transitionBetween(oldList, newList, api) { - var oldDiffItems = flattenDataDiffItems(oldList); - var newDiffItems = flattenDataDiffItems(newList); - - function updateMorphingPathProps(from, to, rawFrom, rawTo, animationCfg) { - if (rawFrom || from) { - to.animateFrom({ - style: rawFrom && rawFrom !== from ? // dividingMethod like clone may override the style(opacity) - // So extend it to raw style. - extend(extend({}, rawFrom.style), from.style) : from.style - }, animationCfg); - } - } - - function findKeyDim(items) { - for (var i = 0; i < items.length; i++) { - if (items[i].dim) { - return items[i].dim; - } - } - } - - var oldKeyDim = findKeyDim(oldDiffItems); - var newKeyDim = findKeyDim(newDiffItems); - var hasMorphAnimation = false; - - function createKeyGetter(isOld, onlyGetId) { - return function (diffItem) { - var data = diffItem.data; - var dataIndex = diffItem.dataIndex; // TODO if specified dim - - if (onlyGetId) { - return data.getId(dataIndex); - } // Use group id as transition key by default. - // So we can achieve multiple to multiple animation like drilldown / up naturally. - // If group id not exits. Use id instead. If so, only one to one transition will be applied. - - - var dataGroupId = diffItem.dataGroupId; // If specified key dimension(itemGroupId by default). Use this same dimension from other data. - // PENDING: If only use key dimension of newData. - - var keyDim = isOld ? oldKeyDim || newKeyDim : newKeyDim || oldKeyDim; - var dimInfo = keyDim && data.getDimensionInfo(keyDim); - var dimOrdinalMeta = dimInfo && dimInfo.ordinalMeta; - - if (dimInfo) { - // Get from encode.itemGroupId. - var key = data.get(dimInfo.name, dataIndex); - - if (dimOrdinalMeta) { - return dimOrdinalMeta.categories[key] || key + ''; - } - - return key + ''; - } // Get groupId from raw item. { groupId: '' } - - - var itemVal = data.getRawDataItem(dataIndex); - - if (itemVal && itemVal.groupId) { - return itemVal.groupId + ''; - } - - return dataGroupId || data.getId(dataIndex); - }; - } // Use id if it's very likely to be an one to one animation - // It's more robust than groupId - // TODO Check if key dimension is specified. - - - var useId = isAllIdSame(oldDiffItems, newDiffItems); - var isElementStillInChart = {}; - - if (!useId) { - // We may have different diff strategy with basicTransition if we use other dimension as key. - // If so, we can't simply check if oldEl is same with newEl. We need a map to check if oldEl is still being used in the new chart. - // We can't use the elements that already being morphed. Let it keep it's original basic transition. - for (var i = 0; i < newDiffItems.length; i++) { - var newItem = newDiffItems[i]; - var el = newItem.data.getItemGraphicEl(newItem.dataIndex); - - if (el) { - isElementStillInChart[el.id] = true; - } - } - } - - function updateOneToOne(newIndex, oldIndex) { - var oldItem = oldDiffItems[oldIndex]; - var newItem = newDiffItems[newIndex]; - var newSeries = newItem.data.hostModel; // TODO Mark this elements is morphed and don't morph them anymore - - var oldEl = oldItem.data.getItemGraphicEl(oldItem.dataIndex); - var newEl = newItem.data.getItemGraphicEl(newItem.dataIndex); // Can't handle same elements. - - if (oldEl === newEl) { - newEl && animateElementStyles(newEl, newItem.dataIndex, newSeries); - return; - } - - if ( // We can't use the elements that already being morphed - oldEl && isElementStillInChart[oldEl.id]) { - return; - } - - if (newEl) { - // TODO: If keep animating the group in case - // some of the elements don't want to be morphed. - // TODO Label? - stopAnimation(newEl); - - if (oldEl) { - stopAnimation(oldEl); // If old element is doing leaving animation. stop it and remove it immediately. - - removeEl$1(oldEl); - hasMorphAnimation = true; - applyMorphAnimation(getPathList(oldEl), getPathList(newEl), newItem.divide, newSeries, newIndex, updateMorphingPathProps); - } else { - fadeInElement(newEl, newSeries, newIndex); - } - } // else keep oldEl leaving animation. - - } - - new DataDiffer(oldDiffItems, newDiffItems, createKeyGetter(true, useId), createKeyGetter(false, useId), null, 'multiple').update(updateOneToOne).updateManyToOne(function (newIndex, oldIndices) { - var newItem = newDiffItems[newIndex]; - var newData = newItem.data; - var newSeries = newData.hostModel; - var newEl = newData.getItemGraphicEl(newItem.dataIndex); - var oldElsList = filter(map(oldIndices, function (idx) { - return oldDiffItems[idx].data.getItemGraphicEl(oldDiffItems[idx].dataIndex); - }), function (oldEl) { - return oldEl && oldEl !== newEl && !isElementStillInChart[oldEl.id]; - }); - - if (newEl) { - stopAnimation(newEl); - - if (oldElsList.length) { - // If old element is doing leaving animation. stop it and remove it immediately. - each(oldElsList, function (oldEl) { - stopAnimation(oldEl); - removeEl$1(oldEl); - }); - hasMorphAnimation = true; - applyMorphAnimation(getPathList(oldElsList), getPathList(newEl), newItem.divide, newSeries, newIndex, updateMorphingPathProps); - } else { - fadeInElement(newEl, newSeries, newItem.dataIndex); - } - } // else keep oldEl leaving animation. - - }).updateOneToMany(function (newIndices, oldIndex) { - var oldItem = oldDiffItems[oldIndex]; - var oldEl = oldItem.data.getItemGraphicEl(oldItem.dataIndex); // We can't use the elements that already being morphed - - if (oldEl && isElementStillInChart[oldEl.id]) { - return; - } - - var newElsList = filter(map(newIndices, function (idx) { - return newDiffItems[idx].data.getItemGraphicEl(newDiffItems[idx].dataIndex); - }), function (el) { - return el && el !== oldEl; - }); - var newSeris = newDiffItems[newIndices[0]].data.hostModel; - - if (newElsList.length) { - each(newElsList, function (newEl) { - return stopAnimation(newEl); - }); - - if (oldEl) { - stopAnimation(oldEl); // If old element is doing leaving animation. stop it and remove it immediately. - - removeEl$1(oldEl); - hasMorphAnimation = true; - applyMorphAnimation(getPathList(oldEl), getPathList(newElsList), oldItem.divide, // Use divide on old. - newSeris, newIndices[0], updateMorphingPathProps); - } else { - each(newElsList, function (newEl) { - return fadeInElement(newEl, newSeris, newIndices[0]); - }); - } - } // else keep oldEl leaving animation. - - }).updateManyToMany(function (newIndices, oldIndices) { - // If two data are same and both have groupId. - // Normally they should be diff by id. - new DataDiffer(oldIndices, newIndices, function (rawIdx) { - return oldDiffItems[rawIdx].data.getId(oldDiffItems[rawIdx].dataIndex); - }, function (rawIdx) { - return newDiffItems[rawIdx].data.getId(newDiffItems[rawIdx].dataIndex); - }).update(function (newIndex, oldIndex) { - // Use the original index - updateOneToOne(newIndices[newIndex], oldIndices[oldIndex]); - }).execute(); - }).execute(); - - if (hasMorphAnimation) { - each(newList, function (_a) { - var data = _a.data; - var seriesModel = data.hostModel; - var view = seriesModel && api.getViewOfSeriesModel(seriesModel); - var animationCfg = getAnimationConfig('update', seriesModel, 0); // use 0 index. - - if (view && seriesModel.isAnimationEnabled() && animationCfg && animationCfg.duration > 0) { - view.group.traverse(function (el) { - if (el instanceof Path && !el.animators.length) { - // We can't accept there still exists element that has no animation - // if universalTransition is enabled - el.animateFrom({ - style: { - opacity: 0 - } - }, animationCfg); - } - }); - } - }); - } - } - - function getSeriesTransitionKey(series) { - var seriesKey = series.getModel('universalTransition').get('seriesKey'); - - if (!seriesKey) { - // Use series id by default. - return series.id; - } - - return seriesKey; - } - - function convertArraySeriesKeyToString(seriesKey) { - if (isArray(seriesKey)) { - // Order independent. - return seriesKey.sort().join(','); - } - - return seriesKey; - } - - function getDivideShapeFromData(data) { - if (data.hostModel) { - return data.hostModel.getModel('universalTransition').get('divideShape'); - } - } - - function findTransitionSeriesBatches(globalStore, params) { - var updateBatches = createHashMap(); - var oldDataMap = createHashMap(); // Map that only store key in array seriesKey. - // Which is used to query the old data when transition from one to multiple series. - - var oldDataMapForSplit = createHashMap(); - each(globalStore.oldSeries, function (series, idx) { - var oldDataGroupId = globalStore.oldDataGroupIds[idx]; - var oldData = globalStore.oldData[idx]; - var transitionKey = getSeriesTransitionKey(series); - var transitionKeyStr = convertArraySeriesKeyToString(transitionKey); - oldDataMap.set(transitionKeyStr, { - dataGroupId: oldDataGroupId, - data: oldData - }); - - if (isArray(transitionKey)) { - // Same key can't in different array seriesKey. - each(transitionKey, function (key) { - oldDataMapForSplit.set(key, { - key: transitionKeyStr, - dataGroupId: oldDataGroupId, - data: oldData - }); - }); - } - }); - - function checkTransitionSeriesKeyDuplicated(transitionKeyStr) { - if (updateBatches.get(transitionKeyStr)) { - warn("Duplicated seriesKey in universalTransition " + transitionKeyStr); - } - } - - each(params.updatedSeries, function (series) { - if (series.isUniversalTransitionEnabled() && series.isAnimationEnabled()) { - var newDataGroupId = series.get('dataGroupId'); - var newData = series.getData(); - var transitionKey = getSeriesTransitionKey(series); - var transitionKeyStr = convertArraySeriesKeyToString(transitionKey); // Only transition between series with same id. - - var oldData = oldDataMap.get(transitionKeyStr); // string transition key is the best match. - - if (oldData) { - if ("development" !== 'production') { - checkTransitionSeriesKeyDuplicated(transitionKeyStr); - } // TODO check if data is same? - - - updateBatches.set(transitionKeyStr, { - oldSeries: [{ - dataGroupId: oldData.dataGroupId, - divide: getDivideShapeFromData(oldData.data), - data: oldData.data - }], - newSeries: [{ - dataGroupId: newDataGroupId, - divide: getDivideShapeFromData(newData), - data: newData - }] - }); - } else { - // Transition from multiple series. - if (isArray(transitionKey)) { - if ("development" !== 'production') { - checkTransitionSeriesKeyDuplicated(transitionKeyStr); - } - - var oldSeries_1 = []; - each(transitionKey, function (key) { - var oldData = oldDataMap.get(key); - - if (oldData.data) { - oldSeries_1.push({ - dataGroupId: oldData.dataGroupId, - divide: getDivideShapeFromData(oldData.data), - data: oldData.data - }); - } - }); - - if (oldSeries_1.length) { - updateBatches.set(transitionKeyStr, { - oldSeries: oldSeries_1, - newSeries: [{ - dataGroupId: newDataGroupId, - data: newData, - divide: getDivideShapeFromData(newData) - }] - }); - } - } else { - // Try transition to multiple series. - var oldData_1 = oldDataMapForSplit.get(transitionKey); - - if (oldData_1) { - var batch = updateBatches.get(oldData_1.key); - - if (!batch) { - batch = { - oldSeries: [{ - dataGroupId: oldData_1.dataGroupId, - data: oldData_1.data, - divide: getDivideShapeFromData(oldData_1.data) - }], - newSeries: [] - }; - updateBatches.set(oldData_1.key, batch); - } - - batch.newSeries.push({ - dataGroupId: newDataGroupId, - data: newData, - divide: getDivideShapeFromData(newData) - }); - } - } - } - } - }); - return updateBatches; - } - - function querySeries(series, finder) { - for (var i = 0; i < series.length; i++) { - var found = finder.seriesIndex != null && finder.seriesIndex === series[i].seriesIndex || finder.seriesId != null && finder.seriesId === series[i].id; - - if (found) { - return i; - } - } - } - - function transitionSeriesFromOpt(transitionOpt, globalStore, params, api) { - var from = []; - var to = []; - each(normalizeToArray(transitionOpt.from), function (finder) { - var idx = querySeries(globalStore.oldSeries, finder); - - if (idx >= 0) { - from.push({ - dataGroupId: globalStore.oldDataGroupIds[idx], - data: globalStore.oldData[idx], - // TODO can specify divideShape in transition. - divide: getDivideShapeFromData(globalStore.oldData[idx]), - dim: finder.dimension - }); - } - }); - each(normalizeToArray(transitionOpt.to), function (finder) { - var idx = querySeries(params.updatedSeries, finder); - - if (idx >= 0) { - var data = params.updatedSeries[idx].getData(); - to.push({ - dataGroupId: globalStore.oldDataGroupIds[idx], - data: data, - divide: getDivideShapeFromData(data), - dim: finder.dimension - }); - } - }); - - if (from.length > 0 && to.length > 0) { - transitionBetween(from, to, api); - } - } - - function installUniversalTransition(registers) { - registers.registerUpdateLifecycle('series:beforeupdate', function (ecMOdel, api, params) { - each(normalizeToArray(params.seriesTransition), function (transOpt) { - each(normalizeToArray(transOpt.to), function (finder) { - var series = params.updatedSeries; - - for (var i = 0; i < series.length; i++) { - if (finder.seriesIndex != null && finder.seriesIndex === series[i].seriesIndex || finder.seriesId != null && finder.seriesId === series[i].id) { - series[i][SERIES_UNIVERSAL_TRANSITION_PROP] = true; - } - } - }); - }); - }); - registers.registerUpdateLifecycle('series:transition', function (ecModel, api, params) { - // TODO api provide an namespace that can save stuff per instance - var globalStore = getUniversalTransitionGlobalStore(api); // TODO multiple to multiple series. - - if (globalStore.oldSeries && params.updatedSeries && params.optionChanged) { - // Use give transition config if its' give; - var transitionOpt = params.seriesTransition; - - if (transitionOpt) { - each(normalizeToArray(transitionOpt), function (opt) { - transitionSeriesFromOpt(opt, globalStore, params, api); - }); - } else { - // Else guess from series based on transition series key. - var updateBatches_1 = findTransitionSeriesBatches(globalStore, params); - each(updateBatches_1.keys(), function (key) { - var batch = updateBatches_1.get(key); - transitionBetween(batch.oldSeries, batch.newSeries, api); - }); - } // Reset - - - each(params.updatedSeries, function (series) { - // Reset; - if (series[SERIES_UNIVERSAL_TRANSITION_PROP]) { - series[SERIES_UNIVERSAL_TRANSITION_PROP] = false; - } - }); - } // Save all series of current update. Not only the updated one. - - - var allSeries = ecModel.getSeries(); - var savedSeries = globalStore.oldSeries = []; - var savedDataGroupIds = globalStore.oldDataGroupIds = []; - var savedData = globalStore.oldData = []; - - for (var i = 0; i < allSeries.length; i++) { - var data = allSeries[i].getData(); // Only save the data that can have transition. - // Avoid large data costing too much extra memory - - if (data.count() < DATA_COUNT_THRESHOLD) { - savedSeries.push(allSeries[i]); - savedDataGroupIds.push(allSeries[i].get('dataGroupId')); - savedData.push(data); - } - } - }); - } - - // Render engines - // ----------------- - // Render via Canvas. - // echarts.init(dom, null, { renderer: 'canvas' }) - - use([install$1]); // Render via SVG. - // echarts.init(dom, null, { renderer: 'svg' }) - - use([install]); // ---------------- - // Charts (series) - // ---------------- - // All of the series types, for example: - // chart.setOption({ - // series: [{ - // type: 'line' // or 'bar', 'pie', ... - // }] - // }); - - use([install$2, install$3, install$4, install$6, install$8, install$a, install$b, install$c, install$d, install$e, install$f, install$h, install$i, install$j, install$k, install$l, install$m, install$n, install$o, install$p, install$q, install$r]); // ------------------- - // Coordinate systems - // ------------------- - // All of the axis modules have been included in the - // coordinate system module below, do not need to - // make extra import. - // `cartesian` coordinate system. For some historical - // reasons, it is named as grid, for example: - // chart.setOption({ - // grid: {...}, - // xAxis: {...}, - // yAxis: {...}, - // series: [{...}] - // }); - - use(install$t); // `polar` coordinate system, for example: - // chart.setOption({ - // polar: {...}, - // radiusAxis: {...}, - // angleAxis: {...}, - // series: [{ - // coordinateSystem: 'polar' - // }] - // }); - - use(install$u); // `geo` coordinate system, for example: - // chart.setOption({ - // geo: {...}, - // series: [{ - // coordinateSystem: 'geo' - // }] - // }); - - use(install$9); // `singleAxis` coordinate system (notice, it is a coordinate system - // with only one axis, work for chart like theme river), for example: - // chart.setOption({ - // singleAxis: {...} - // series: [{type: 'themeRiver', ...}] - // }); - - use(install$v); // `parallel` coordinate system, only work for parallel series, for example: - // chart.setOption({ - // parallel: {...}, - // parallelAxis: [{...}, ...], - // series: [{ - // type: 'parallel' - // }] - // }); - - use(install$g); // `calendar` coordinate system. for example, - // chart.setOptionp({ - // calendar: {...}, - // series: [{ - // coordinateSystem: 'calendar' - // }] - // ); - - use(install$w); // ------------------ - // Other components - // ------------------ - // `graphic` component, for example: - // chart.setOption({ - // graphic: {...} - // }); - - use(install$x); // `toolbox` component, for example: - // chart.setOption({ - // toolbox: {...} - // }); - - use(install$z); // `tooltip` component, for example: - // chart.setOption({ - // tooltip: {...} - // }); - - use(install$A); // `axisPointer` component, for example: - // chart.setOption({ - // tooltip: {axisPointer: {...}, ...} - // }); - // Or - // chart.setOption({ - // axisPointer: {...} - // }); - - use(install$s); // `brush` component, for example: - // chart.setOption({ - // brush: {...} - // }); - // Or - // chart.setOption({ - // tooltip: {feature: {brush: {...}} - // }) - - use(install$B); // `title` component, for example: - // chart.setOption({ - // title: {...} - // }); - - use(install$C); // `timeline` component, for example: - // chart.setOption({ - // timeline: {...} - // }); - - use(install$D); // `markPoint` component, for example: - // chart.setOption({ - // series: [{markPoint: {...}}] - // }); - - use(install$E); // `markLine` component, for example: - // chart.setOption({ - // series: [{markLine: {...}}] - // }); - - use(install$F); // `markArea` component, for example: - // chart.setOption({ - // series: [{markArea: {...}}] - // }); - - use(install$G); // `legend` component not scrollable. for example: - // chart.setOption({ - // legend: {...} - // }); - - use(install$J); // `dataZoom` component including both `dataZoomInside` and `dataZoomSlider`. - - use(install$M); // `dataZoom` component providing drag, pinch, wheel behaviors - // inside coodinate system, for example: - // chart.setOption({ - // dataZoom: {type: 'inside'} - // }); - - use(install$K); // `dataZoom` component providing a slider bar, for example: - // chart.setOption({ - // dataZoom: {type: 'slider'} - // }); - - use(install$L); // `visualMap` component including both `visualMapContinuous` and `visualMapPiecewise`. - - use(install$P); // `visualMap` component providing continuous bar, for example: - // chart.setOption({ - // visualMap: {type: 'continuous'} - // }); - - use(install$N); // `visualMap` component providing pieces bar, for example: - // chart.setOption({ - // visualMap: {type: 'piecewise'} - // }); - - use(install$O); // `aria` component providing aria, for example: - // chart.setOption({ - // aria: {...} - // }); - - use(install$Q); // dataset transform - // chart.setOption({ - // dataset: { - // transform: [] - // } - // }); - - use(install$R); - use(install$S); // universal transition - // chart.setOption({ - // series: { - // universalTransition: { enabled: true } - // } - // }) - - use(installUniversalTransition); // label layout - // chart.setOption({ - // series: { - // labelLayout: { hideOverlap: true } - // } - // }) - - use(installLabelLayout); - - exports.Axis = Axis; - exports.ChartView = ChartView; - exports.ComponentModel = ComponentModel; - exports.ComponentView = ComponentView; - exports.List = SeriesData; - exports.Model = Model; - exports.PRIORITY = PRIORITY; - exports.SeriesModel = SeriesModel; - exports.color = color; - exports.connect = connect; - exports.dataTool = dataTool; - exports.dependencies = dependencies; - exports.disConnect = disConnect; - exports.disconnect = disconnect; - exports.dispose = dispose$1; - exports.env = env; - exports.extendChartView = extendChartView; - exports.extendComponentModel = extendComponentModel; - exports.extendComponentView = extendComponentView; - exports.extendSeriesModel = extendSeriesModel; - exports.format = format$1; - exports.getCoordinateSystemDimensions = getCoordinateSystemDimensions; - exports.getInstanceByDom = getInstanceByDom; - exports.getInstanceById = getInstanceById; - exports.getMap = getMap; - exports.graphic = graphic$1; - exports.helper = helper; - exports.init = init$1; - exports.innerDrawElementOnCanvas = brushSingle; - exports.matrix = matrix; - exports.number = number; - exports.parseGeoJSON = parseGeoJSON; - exports.parseGeoJson = parseGeoJSON; - exports.registerAction = registerAction; - exports.registerCoordinateSystem = registerCoordinateSystem; - exports.registerLayout = registerLayout; - exports.registerLoading = registerLoading; - exports.registerLocale = registerLocale; - exports.registerMap = registerMap; - exports.registerPostInit = registerPostInit; - exports.registerPostUpdate = registerPostUpdate; - exports.registerPreprocessor = registerPreprocessor; - exports.registerProcessor = registerProcessor; - exports.registerTheme = registerTheme; - exports.registerTransform = registerTransform; - exports.registerUpdateLifecycle = registerUpdateLifecycle; - exports.registerVisual = registerVisual; - exports.setCanvasCreator = setCanvasCreator; - exports.setPlatformAPI = setPlatformAPI; - exports.throttle = throttle; - exports.time = time; - exports.use = use; - exports.util = util$1; - exports.vector = vector; - exports.version = version$1; - exports.zrUtil = util; - exports.zrender = zrender; - - Object.defineProperty(exports, '__esModule', { value: true }); - -}))); -//# sourceMappingURL=echarts.js.map diff --git a/public/static/common/js/echartsOptions.js b/public/static/common/js/echartsOptions.js deleted file mode 100644 index 625f576..0000000 --- a/public/static/common/js/echartsOptions.js +++ /dev/null @@ -1,1050 +0,0 @@ -import { i18n } from './index.js' -//初始化图表options -// GIS -let PRPDGIS = ''; - -// GIL -let PRPDGIL = ''; -let GIL1 = ''; -let GIL2 = ''; -let GIL3 = ''; -let GIL4 = ''; - -// 电缆局放 -let CPD = '', CPD1 = ''; -export const loadOption = () => { - CPD = { - backgroundColor: '#ffffff', - title: { - // text: '幅值趋势图' , - left: 'center', - top: 5, - textStyle: { - fontSize: 16, - color: '#000000' - } - }, - // animation: false, - tooltip: { - show: false, - }, - color: ["#35C822", '#1f8ffd', '#fd1f44', '#ff9933', '#660099'], - legend: { - left: 'center', - top: 10, - data: ['脉冲最大值', '脉冲数量'], - textStyle: { - color: "#92A3A8", - fontsize: 5 - }, - // selected: { - // "Discharge phase": false, - // "放电相位": false, - // "фаза разряда": false, - // }, - type: 'scroll', - width: '50%' - }, - tooltip: { - trigger: 'axis', - //trigger: 'item', - axisPointer: { - type: 'cross', - animation: false, - label: { - backgroundColor: '#92A3A8' - } - } - - }, - xAxis: [{ - type: 'category', - data:[], - axisLine: { - lineStyle: { - color: '#92A3A8' - } - }, - axisLabel: { //调整x轴的lable - textStyle: { - color: '#92A3A8' - } - }, - offset: -20, - - }], - yAxis: [{ - type: 'value', - min: -80, - max: 0, - name: 'dBm', - nameLocation: 'end', - splitNumber: 1, - // nameGap:'30', - // interval:10, - axisLine: { - lineStyle: { color: '#000000', opacity: 0.1 } - }, - axisPointer: { - lineStyle: { color: '#000000' }, - lineStyle: { color: '#000000' }, - }, - }, - { - type: 'value', - min: 0, - max: 3600, - position: 'right', - name: '', - nameLocation: 'end', - splitNumber: 1, - // nameGap:'30', - // interval:10, - axisLine: { - lineStyle: { color: '#000000', opacity: 0.1 } - }, - axisPointer: { - lineStyle: { color: '#000000' }, - lineStyle: { color: '#000000' }, - }, - }], - dataZoom: [{ - show: false, - realtime: true, - start: 0, - end: 100, - // backgroundColor:'#d' - textStyle: { - color: "#92A3A8" - }, - }, { - type: 'inside', - realtime: true, - start: 5, - end: 85 - }], - axisPointer: { - show: true, - snap: true, - lineStyle: { - type: 'dashed', - }, - label: { - show: true, - margin: 6, - backgroundColor: '#556', - color: '#fff' - }, - }, - grid: { - left: 28, - right: 42, - bottom: 18, - top: 30, - x: 0, - y: 0, - x2: 0, - y2: 0, - // borderWidth: 1, - show: false, - containLabel: true - }, - series: [{ - name: '脉冲最大值', - type: 'line', - data:[], - //data: [['9:01:01', -20], ['9:01:02', -22], ['9:01:03', -24], ['9:01:04', -26], ['9:01:05', -20], ['9:01:06', -28]], - smooth: true, - showSymbol: false, - symbolSize: 1, - animation: false, - itemStyle: { - color: "#1f8ffd", - }, - yAxisIndex: 0, - areaStyle: { - color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{ - offset: 0, - color: '#1f8ffd' - }, { - offset: 0.8, - color: 'rgba(87,114,199, 0)' - }], false), - shadowColor: 'rgba(0, 0, 0, 0.1)', - shadowBlur: 10, - origin: 'start', - }, - // symbol: 'circle', - itemStyle: { - color: '#1f8ffd', - borderColor: 'rgba(31, 143, 253,0.57)', - borderWidth: 12 - }, - }, - { - name: '脉冲数量', - type: 'line', - data:[], - //data: [['9:01:01', 3100], ['9:01:02', 3000], ['9:01:03', 3105], ['9:01:04', 3108], ['9:01:05', 3105], ['9:01:06', 2580]], - smooth: true, - showSymbol: false, - symbolSize: 1, - showAllSymbol: true, - animation: false, - itemStyle: { - color: "#fd1f44", - }, - yAxisIndex: 1, - areaStyle: { - color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{ - offset: 0, - color: '#fd1f44' - }, { - offset: 0.8, - color: 'rgba(87,114,199, 0)' - }], false), - shadowColor: 'rgba(0, 0, 0, 0.1)', - shadowBlur: 10, - origin: 'start', - }, - symbol: 'circle', - itemStyle: { - color: '#fd1f44', - borderColor: 'rgba(253, 31, 68,0.57)', - borderWidth: 12 - }, - }] - }; - GIL1 = { - backgroundColor: '#000d13', - title: { - text: '飞行', - left: 'center', - top: 5, - textStyle: { - fontSize: 16, - color: 'rgba(255, 255, 255, 0.7)' - } - }, - // animation: false, - visualMap: [{ - type: 'continuous', - seriesIndex: 0, - type: 'piecewise', - // dimension :2, - pieces: [ - /*{min: 16,color: '#820404'}, - {min: 11, max: 15,color: '#f00'}, - {min: 6, max: 10,color: '#ff9900'}, - {min: 0,max: 5,color: '#ffff00'}*/ - { min: 15, color: '#d94e5d' }, - { min: 11, max: 14, color: '#e28b4a' }, - { min: 8, max: 10, color: '#eac736' }, - { min: 6, max: 7, color: '#9db578' }, - { min: 4, max: 5, color: '#50a3ba' }, - { min: 0, max: 3, color: '#0082df' } - ], - formatter: function (value, value2) { - if (value2 == "Infinity") { - return (value - 1) + "+"; - } - return value2; - // return ""; - }, - itemWidth: 10, - orient: "vertical", - // inRange: { - // color: ['#313695','#9db578', '#fdae61', '#a50026'] - // }, - // splitNumber: 5, - textStyle: { - show: false, - color: '#fff', - }, - // itemHeight:80, - right: 0, - top: 30, - // min:1, - // max:15, - calculable: true, - hoverLink: false, - // inverse:true, - // itemWidth:2, - textGap: 1, - orient: "vertical", - // inRange: { - // color: ['#0082df','#eac736', '#d94e5d', '#ee0000'] - // }, - // inRange: { - // color: ['#ffff00', '#ffcc00', '#ff9900','#ff6600','#ff3300','#ff0000'] - // }, - }], - tooltip: { - show: false, - }, - toolbox: { - show: true, - orient: 'horizontal', - top: 0, - right: 0, - showTitle: false, - iconStyle: { - borderColor: '#fff' - }, - emphasis: { - show: false, - iconStyle: { - borderColor: '#fff' - } - }, - feature: { - mark: { show: true }, - saveAsImage: { show: true } - } - }, - xAxis: [{ - type: 'value', - min: 0, - max: 1000, - // splitNumber:4, - // interval:90, - name: '时间' + '(ms)', - nameLocation: 'center', - nameGap: '20', - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - axisPointer: { - lineStyle: { color: '#fff' }, - label: { - precision: 0 - } - }, - }], - yAxis: [{ - type: 'value', - min: 0, - max: 100, - // splitNumber:4, - name: 'mV', - nameLocation: 'end', - // nameGap:'30', - // interval:10, - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - axisPointer: { - lineStyle: { color: '#fff' }, - lineStyle: { color: '#fff' }, - }, - }], - axisPointer: { - show: false, - snap: true, - lineStyle: { - type: 'dashed', - }, - label: { - show: true, - margin: 6, - backgroundColor: '#556', - textStyle: { - color: '#fff' - } - }, - }, - grid: { - left: 60, - right: 60, - bottom: 40, - top: 40, - x: 0, - y: 0, - x2: 0, - y2: 0, - borderWidth: 1 - }, - series: [{ - name: 'price-area', - type: 'scatter', - symbolSize: 2, - progressive: 0, - animation: false, - large: true, // 大规模散点图 - itemStyle: { - color: "#0082df", - }, - largeThreshold: 10, - - }, { - name: 'price-area', - type: 'scatter', - symbolSize: 2, - progressive: 0, - animation: false, - large: true, // 大规模散点图 - itemStyle: { - color: "#50a3ba", - }, - largeThreshold: 10, - - }, { - name: 'price-area', - type: 'scatter', - symbolSize: 2, - progressive: 0, - animation: false, - large: true, // 大规模散点图 - itemStyle: { - color: "#9db578", - }, - largeThreshold: 10, - - }, { - name: 'price-area', - type: 'scatter', - symbolSize: 2, - progressive: 0, - animation: false, - large: true, // 大规模散点图 - itemStyle: { - color: "#eac736", - }, - largeThreshold: 10, - - }, { - name: 'price-area', - type: 'scatter', - symbolSize: 2, - progressive: 0, - animation: false, - large: true, // 大规模散点图 - itemStyle: { - color: "#e28b4a", - }, - largeThreshold: 10, - - }, { - name: 'price-area', - type: 'scatter', - symbolSize: 2, - progressive: 0, - animation: false, - large: true, // 大规模散点图 - itemStyle: { - color: "#d94e5d", - }, - largeThreshold: 10, - }] - }; - GIL2 = { - backgroundColor: '#000d13', - title: { - text: '相位', - left: 'center', - top: 5, - textStyle: { - fontSize: 16, - color: 'rgba(255, 255, 255, 0.7)' - } - }, - // animation: false, - tooltip: { - show: false, - }, - toolbox: { - show: true, - orient: 'horizontal', - top: 0, - right: 0, - showTitle: false, - iconStyle: { - borderColor: '#fff' - }, - emphasis: { - show: false, - iconStyle: { - borderColor: '#fff' - } - }, - feature: { - mark: { show: true }, - saveAsImage: { show: true } - } - }, - xAxis: [{ - type: 'value', - min: 0, - max: 1000, - // splitNumber:4, - // interval:90, - name: '时间' + '(ms)', - nameLocation: 'center', - nameGap: '20', - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - axisPointer: { - lineStyle: { color: '#fff' }, - label: { - precision: 0 - } - }, - }], - yAxis: [{ - type: 'value', - min: 0, - max: 100, - name: 'mV', - nameLocation: 'end', - // splitNumber:4, - // nameGap:'30', - // interval:10, - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - axisPointer: { - lineStyle: { color: '#fff' }, - lineStyle: { color: '#fff' }, - }, - }], - axisPointer: { - show: false, - snap: true, - lineStyle: { - type: 'dashed', - }, - label: { - show: true, - margin: 6, - backgroundColor: '#556', - textStyle: { - color: '#fff' - } - }, - }, - grid: { - left: 70, - right: 50, - bottom: 40, - top: 40, - x: 0, - y: 0, - x2: 0, - y2: 0, - borderWidth: 1 - }, - series: [{ - name: 'line', - type: 'line', - data: [], - smooth: true, - showSymbol: false, - symbolSize: 1, - animation: false, - itemStyle: { - color: "#5772c7", - }, - areaStyle: { - color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{ - offset: 0, - color: 'rgba(87,114,199, 0.3)' - }, { - offset: 0.8, - color: 'rgba(87,114,199, 0)' - }], false), - shadowColor: 'rgba(0, 0, 0, 0.1)', - shadowBlur: 10, - origin: 'start', - }, - symbol: 'circle', - itemStyle: { - color: 'rgb(87,114,199)', - borderColor: 'rgba(87,114,199,0.27)', - borderWidth: 12 - }, - }] - }; - GIL3 = { - title: [ - { - textBaseline: "middle", - left: '15%', - top: "10%", - text: " mV", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - { - textBaseline: "middle", - left: '15%', - top: "60%", - text: " mV", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - { - textBaseline: "middle", - right: '10', - top: "35%", - text: "", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - { - textBaseline: "middle", - right: '10', - top: "85%", - text: "", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - ], - xAxis: [ - { - type: "value", - min: '0', - max: '100', - interval: 900000000, - // show:false, - position: 'top', - gridIndex: 0, - axisLabel: { - showMaxLabel: true, - color: '#fff' - }, - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - }, - { - type: "value", - show: false, - gridIndex: 1, - }, - { - type: "value", - min: '0', - max: '100', - interval: 900000000, - // show:false, - position: 'top', - gridIndex: 2, - axisLabel: { - showMaxLabel: true, - color: '#fff' - }, - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - }, - { - type: "value", - show: false, - gridIndex: 3, - }, - ], - yAxis: [ - { - type: "category", - show: false, - axisLine: { - onZero: false - }, - gridIndex: 0, - }, - { - type: "value", - gridIndex: 1, - // min:'0', - // max:'150', - show: false - }, - { - type: "category", - show: false, - gridIndex: 2, - }, - { - type: "value", - // min:'0', - // max:'150', - gridIndex: 3, - show: false - }, - ], - grid: [ - { - height: "15%", - top: "20%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2, - }, - { - height: "15%", - top: "35%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2 - }, - { - height: "15%", - top: "70%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2 - }, - { - height: "15%", - top: "85%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2 - }, - ], - visualMap: [ - { - type: "continuous", - seriesIndex: 0, - min: 0, - max: 100, - show: false, - itemWidth: 5, - orient: "vertical", - dimension: 0, - inRange: { - color: ['#00ee00', '#eeee00', '#ee0000', '#4e0211'] - }, - }, - { - type: "continuous", - seriesIndex: 2, - min: 0, - max: 100, - show: false, - itemWidth: 5, - dimension: 0, - orient: "horizontal", - inRange: { - color: ['#00ee00', '#eeee00', '#ee0000', '#4e0211'] - }, - }, - ], - series: [ - { - type: "bar", - xAxisIndex: 0, - yAxisIndex: 0, - barWidth: '100%', - data: [], - }, - { - type: "line", - xAxisIndex: 1, - yAxisIndex: 1, - symbol: "none", - smooth: true, - showSymbol: false, - lineStyle: { - color: "#00d4c7" - }, - areaStyle: { - - color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{ - offset: 0, - color: 'rgba(0, 136, 212, 0.8)' - }, { - offset: 0.8, - color: 'rgba(0, 136, 212, 0.3)' - }], false), - shadowColor: 'rgba(0, 0, 0, 0.1)', - shadowBlur: 10 - - }, - data: [], - }, - { - type: "bar", - xAxisIndex: 2, - yAxisIndex: 2, - barWidth: '100%', - data: [], - }, - { - type: "line", - xAxisIndex: 3, - yAxisIndex: 3, - symbol: "none", - smooth: true, - showSymbol: false, - lineStyle: { - color: "#00d4c7" - }, - areaStyle: { - color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{ - offset: 0, - color: 'rgba(0, 136, 212, 0.8)' - }, { - offset: 0.8, - color: 'rgba(0, 136, 212, 0.3)' - }], false), - shadowColor: 'rgba(0, 0, 0, 0.1)', - shadowBlur: 10 - }, - data: [], - }, - ], - } - GIL4 = { - title: [ - { - textBaseline: "middle", - left: '15%', - top: "10%", - text: "1 mV", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - { - textBaseline: "middle", - left: '15%', - top: "60%", - text: "2 mV", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - { - textBaseline: "middle", - right: '10', - top: "35%", - text: "", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - { - textBaseline: "middle", - right: '10', - top: "85%", - text: "", - textStyle: { - color: '#fff', - fontSize: '12px' - } - }, - ], - xAxis: [ - { - type: "value", - min: '0', - max: '100', - interval: 900000000, - // inverse:true, - // show:false, - position: 'top', - gridIndex: 0, - axisLabel: { - showMaxLabel: true, - color: '#fff' - }, - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - }, - { - type: "value", - show: false, - gridIndex: 1, - }, - { - type: "value", - min: '0', - max: '100', - interval: 900000000, - // inverse:true, - // show:false, - position: 'top', - gridIndex: 2, - axisLabel: { - showMaxLabel: true, - color: '#fff' - }, - axisLine: { - lineStyle: { color: '#fff', opacity: 0.1 } - }, - }, - { - type: "value", - show: false, - gridIndex: 3, - }, - ], - yAxis: [ - { - type: "category", - show: false, - axisLine: { - onZero: false - }, - gridIndex: 0, - }, - { - type: "value", - gridIndex: 1, - // inverse:true, - // min:'0', - // max:'150', - show: false - }, - { - type: "category", - show: false, - gridIndex: 2, - }, - { - type: "value", - gridIndex: 3, - // inverse:true, - // min:'0', - // max:'150', - show: false - }, - ], - grid: [ - { - height: "15%", - top: "20%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2, - }, - { - height: "15%", - top: "35%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2 - }, - { - height: "15%", - top: "70%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2 - }, - { - height: "15%", - top: "85%", - right: "60", - show: true, - borderColor: "#eee", - borderWidth: 2 - }, - ], - visualMap: [ - { - type: "continuous", - seriesIndex: 0, - min: 0, - max: 100, - show: false, - itemWidth: 5, - orient: "vertical", - dimension: 0, - inRange: { - // color: ['#00ee00','#eeee00', '#ee0000', '#4e0211'] - color: ['#00d4c7', '#0092f6', '#7e075f', '#ee0000'] - }, - }, - { - type: "continuous", - seriesIndex: 2, - min: 0, - max: 100, - show: false, - itemWidth: 5, - dimension: 0, - orient: "horizontal", - inRange: { - // color: ['#00ee00','#eeee00', '#ee0000', '#4e0211'] - color: ['#00d4c7', '#0092f6', '#7e075f', '#ee0000'] - }, - }, - ], - series: [ - { - type: "bar", - xAxisIndex: 0, - yAxisIndex: 0, - itemStyle: { - color: "#7e075f" - }, - barWidth: '100%', - data: [], - }, - { - type: "line", - xAxisIndex: 1, - yAxisIndex: 1, - symbol: "none", - smooth: true, - showSymbol: false, - lineStyle: { - color: "#00d4c7" - }, - areaStyle: { - color: "#0092f6" - }, - data: [], - }, - { - type: "bar", - xAxisIndex: 2, - yAxisIndex: 2, - itemStyle: { - color: "#7e075f" - }, - barWidth: '100%', - data: [], - }, - { - type: "line", - xAxisIndex: 3, - yAxisIndex: 3, - symbol: "none", - smooth: true, - showSymbol: false, - lineStyle: { - color: "#00d4c7" - }, - areaStyle: { - color: "#0092f6" - }, - data: [], - }, - ], - - }; -} -export { PRPDGIS, PRPDGIL, GIL1, GIL2, GIL3, GIL4, CPD, CPD1 } \ No newline at end of file diff --git a/public/static/common/js/history.js b/public/static/common/js/history.js deleted file mode 100644 index 4ba259a..0000000 --- a/public/static/common/js/history.js +++ /dev/null @@ -1,211 +0,0 @@ -import { drawAxis, drawAxis2D } from './drawHistory.js'; -let renderer, scene, camera, controls, renderer2D, scene2D, camera2D, controls2D; - -let axis = new THREE.Group(); -let axis2D = new THREE.Group(); -let initAxis = 80;//正值最大值 -let minNum = 0;//负值最大值 - -let planes = [];//截断面 - -let maxRange = 100; - -const lut = new THREE.Lut(); -lut.addColorMap('axis', [[0.0, 0x00ee00], [0.25, 0xeeee00], [0.75, 0xee0000], [1.0, 0x4e0211]],); -lut.setColorMap('axis', 500) - -let meshList = new THREE.Group();//图例柱子 -meshList.name = 'chartsMesh'; -const dummy = new THREE.Object3D(); - -let pointsList = new THREE.Group();//图例散点 -pointsList.name = 'chartsPoint'; - -// 周期,相位 柱子数 -let period = 50, phase = 128, count = period * phase; -const geometry = new THREE.BoxGeometry(0.6, 1, 2.5); -geometry.computeVertexNormals(); - -//非光泽表面的材质,没有镜面高光 -//const initInstancedMesh = new THREE.InstancedMesh(geometry, new THREE.MeshLambertMaterial(), count); -//最佳性能材质 -const initInstancedMesh = new THREE.InstancedMesh(geometry, new THREE.MeshBasicMaterial(), count); -initInstancedMesh.name = 'initInstancedMesh'; - -/* prpd初始Instanced点 */ -let initInstancedPoints,//prpd实例 - pointPositions, pointColors,//PRPS位置及颜色的集合 - pointsGeometry = new THREE.BufferGeometry(), - pointCount; - - -init(); -init2D(); -render(); -render2D(); -onWindowResize() -//prps or prpd 切换 -$('#prps').click(function () { - $('#box3D1').toggle(); - $('#box2D1').toggle(); - $(this).toggleClass('ready') - $('#prpd').toggleClass('ready') -}) - -$('#prpd').click(function () { - $('#box2D1').toggle(); - $('#box3D1').toggle(); - $(this).toggleClass('ready') - $('#prps').toggleClass('ready') -}) -$(window).resize(function () { - window.requestAnimationFrame(function () { - onWindowResize() - }) -}) -//prps初始化 -function init() { - let box = $('#box3D1'); - let width = box.width(); - let height = box.height(); - renderer = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true, antialias: true, preserveDrawingBuffer: true, });//preserveDrawingBuffer保存three.js canvas画布上的信息 - renderer.localClippingEnabled = true - let constant = (360 / (initAxis)) * initAxis + 1 - planes = [ - new THREE.Plane(new THREE.Vector3(0, -1, 0), constant), //y轴正值截取 - ] - renderer.clippingPlanes = planes - renderer.setSize(width, height);//设置渲染区域尺寸 - renderer.setClearColor(0xffffff, 1); //设置背景颜色 - //renderer.setClearColor(0xb9d3ff, 1); //设置背景颜色 - renderer.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 - box?.append(renderer.domElement); //body元素中插入canvas对象 - scene = new THREE.Scene(); - scene.add(meshList, axis, pointsList); - - /* var point = new THREE.PointLight(0xffffff); - point.position.set(600, 600, 600); //点光源位置 - scene.add(point); //点光源添加到场景中 - var ambient = new THREE.AmbientLight(0x444444); - scene.add(ambient); */ - - drawAxis(initAxis, minNum, true) - - var k = width / height; //窗口宽高比 - var s = 420; //三维场景显示范围控制系数,系数越大,显示的范围越大 - //创建相机对象 - camera = new THREE.OrthographicCamera(-s * k, s * k, s, -s, 1, 1600); - camera.position.set(500, 200, 500); //设置相机位置 - camera.lookAt(scene.position); //设置相机方向(指向的场景对象) - controls = new THREE.OrbitControls(camera, renderer.domElement); - // 设置左右方向的最大、最小角度限制为 90 度和 0 度 - controls.minAzimuthAngle = 0; - controls.maxAzimuthAngle = Math.PI / 2; - // 设置上下方向的最大、最小角度限制为 90 度和 0 度 - controls.minPolarAngle = 0; - controls.maxPolarAngle = Math.PI / 2; - //设置放大缩小上下限 - controls.minZoom = 0.5; - controls.maxZoom = 2; - //controls.enablePan = false; - //设置控制器中心点 - controls.target.set(25, 0, 180); - renderer.render(scene, camera); - - //controls.addEventListener('change', render); -} -//prpd初始化 -function init2D() { - let box = $('#box2D1'); - let width = box.width(); - let height = box.height(); - renderer2D = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true, antialias: true, preserveDrawingBuffer: true, });//preserveDrawingBuffer保存three.js canvas画布上的信息\ - renderer2D.localClippingEnabled = true - let constant = (360 / (initAxis)) * initAxis + 1 - planes = [ - new THREE.Plane(new THREE.Vector3(0, -1, 0), constant), //y轴正值截取 - ] - renderer2D.clippingPlanes = planes - renderer2D.setSize(width, height);//设置渲染区域尺寸 - renderer2D.setClearColor(0xffffff, 1); //设置背景颜色 - renderer2D.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 - box?.append(renderer2D.domElement); //body元素中插入canvas对象 - scene2D = new THREE.Scene(); - const List = pointsList.clone() - scene2D.add(axis2D, List, pointsList) - drawAxis2D(initAxis, minNum) - // 辅助坐标系 参数250表示坐标系大小,可以根据场景大小去设置 - // let axisHelper = new THREE.AxesHelper(250); - // scene2D.add(axisHelper); - var k = width / height; //窗口宽高比 - var s = 380; //三维场景显示范围控制系数,系数越大,显示的范围越大 - //创建相机对象 - camera2D = new THREE.OrthographicCamera(-s * k - 200, s * k - 200, s + 160, -s + 160, 1, 1200); - camera2D.position.set(600, 0, 0); //设置相机位置 - camera2D.lookAt(scene2D.position); //设置相机方向(指向的场景对象) - controls2D = new THREE.OrbitControls(camera2D, renderer2D.domElement); - controls2D.enablePan = false//右键位移禁用 - controls2D.enableZoom = false//放大缩小禁用 - //设置放大缩小上下限 - controls2D.minZoom = 0.5; - controls2D.maxZoom = 2; - controls2D.enableRotate = false - renderer2D.render(scene2D, camera2D); - controls2D.addEventListener('change', render2D); -} -function onWindowResize() { - let h = $(window).height(); - let w = $(window).width(); - $(".loading-box").css('height', h * 0.95) - let box = $('#box3D1'); - let width = box.width(); - - let height = box.height(); - let k = width / height; //窗口宽高比 - let s = 350; //三维场景显示范围控制系数,系数越大,显示的范围越大 - - camera.left = -s * k - 20; - camera.right = s * k - 20; - camera.top = s + (minNum ? -80 : 60); - camera.bottom = -s + (minNum ? -80 : 60); - camera.aspect = k; - camera.updateProjectionMatrix(); - renderer.setSize(width, height); - renderer.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 - - let s2D = 270; - camera2D.left = -s2D * k - 200; - camera2D.right = s2D * k - 200; - camera2D.top = s2D + (minNum ? -30 : 155); - camera2D.bottom = -s2D + (minNum ? -30 : 155); - camera2D.aspect = k; - camera2D.updateProjectionMatrix(); - renderer2D.setSize(width, height); - renderer2D.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 -} -function render() { - // 获取摄像机的视锥体 - const frustum = new THREE.Frustum(); - const cameraViewProjectionMatrix = new THREE.Matrix4(); - camera.updateMatrixWorld(); // 确保摄像机的世界矩阵已更新 - cameraViewProjectionMatrix.multiplyMatrices(camera.projectionMatrix, camera.matrixWorldInverse); - frustum.setFromProjectionMatrix(cameraViewProjectionMatrix); - - // 遍历场景中的每个对象,并根据视锥体进行剔除 - scene.traverse((object) => { - if (object.isMesh) { - object.visible = frustum.intersectsObject(object); - } - }); - // 更新帧率显示器 - //stats.update(); - renderer.render(scene, camera); - requestAnimationFrame(render) -} -function render2D() { - //scene.rotateY(0.001);//每次绕y轴旋转0.01弧度 - requestAnimationFrame(render2D) - renderer2D.render(scene2D, camera2D); -} - -export { scene, axis, meshList, pointsList, axis2D, scene2D } \ No newline at end of file diff --git a/public/static/common/js/index.js b/public/static/common/js/index.js deleted file mode 100644 index d064e42..0000000 --- a/public/static/common/js/index.js +++ /dev/null @@ -1,1464 +0,0 @@ -import { drawAxis, drawAxis2D, draw2DText } from './draw.js'; -import { loadOption, GIL1, GIL2, GIL3, GIL4, CPD, CPD1 } from './echartsOptions.js'; -let renderer, scene, camera, controls, renderer2D, scene2D, camera2D, controls2D; -//let socket = null; -let axis = new THREE.Group(); -let axis2D = new THREE.Group(); -let initAxis = 80;//正值最大值 -let maxNum = [];//上一次的最大值 -let minNum = 0;//负值最大值 -let planes = [];//截断面 -let i18n; -let viewType = 1;//窗口显示状态1:初始状态,PRPSorPRPD只显示一个且可以切换, 2:半屏状态:3:其他窗口半屏状态 -const paramsStr = window.location.search -const params = new URLSearchParams(paramsStr) -let viewIndex = params.get('viewIndex') || 0;//视窗序号 -let productId = params.get('productId') || 2;//产品id -let deviceId = params.get('deviceId') || '50100001_1';//设备id -let win;//窗口序列 -let isReconnect = false; -let socket; -let token = 'a04cc4a938c376496c36264973e471a3' -let url = `ws://192.168.1.23:8844/messaging/${token}` -//const stats = new Stats(); -//document.body.appendChild(stats.dom); -let box3D = document.getElementById("box3D2"); -let box2D = document.getElementById("box2D2"); -let box2D5 = document.getElementById('box2D5'); -let box2D6 = document.getElementById('box2D6'); -let box3Dchart = ''; -let box2Dchart = ''; -let box2D5chart = ''; -let box2D6chart = ''; - -let maxRange = 100; - -let clearHistory = 30; -// //打点统计次数颜色 -let realTime = { - channelFiltering: 0,//频段过滤 //必填 - switch: 1,//开关:1、启动、0、关闭 //必填 - mapCategory: 3,//图谱类型:1、PRPS、2、PRPD、3、PRPS/PRPD //必填 - monitorType: 1,// GIS 1 GIL故障定位 2 GIL局放 3 电缆 4 //必填 - cumulativeDot: 1,//累计打点时长 默认1秒,-1则无限累计 //必填 - correlationDenoising: 0,//关联降噪1、启动、0、关闭 //不必填 - adaptiveNoise: 0,//自适应降噪1、启动、0、关闭 //不必填 - groundNoise: 0,//手动降噪 //不必填 - migrationPhase: 0,//偏移相位//不必填 - waveSwitch: 0,//录播开关:1、开始、0、停止 //不必填 - atlasType: 1, -} -//电缆局放,1 PRPS/PRPD 2 折现图 -let atlasType = 1; -// monitorType GIS 1 GIL故障定位 2 GIL局放 3 电缆 4 -// atlasType 1 PRPS/PRPD 2 飞行模式/脉冲数据 3 连续模式 -const lut = new THREE.Lut(); -lut.addColorMap('axis', [[0.0, 0x00ee00], [0.25, 0xeeee00], [0.75, 0xee0000], [1.0, 0x4e0211]],); -lut.setColorMap('axis', 500) -let meshList = new THREE.Group();//图例柱子 -const dummy = new THREE.Object3D(); - -meshList.name = 'chartsMesh'; -let pointsList = new THREE.Group();//图例散点 -pointsList.name = 'chartsPoint'; -let axisTextGroup = new THREE.Group(); -axisTextGroup.name = 'axisText'; -let axisTextGroup2D = new THREE.Group(); -axisTextGroup2D.name = 'axisText'; -// 周期,相位 柱子数 -let period = 50, phase = 128, count = period * phase; -const geometry = new THREE.BoxGeometry(0.6, 1, 2.5); -geometry.computeVertexNormals() -//非光泽表面的材质,没有镜面高光 -//const initInstancedMesh = new THREE.InstancedMesh(geometry, new THREE.MeshLambertMaterial(), count); -//最佳性能材质 -const initInstancedMesh = new THREE.InstancedMesh(geometry, new THREE.MeshBasicMaterial(), count); -initInstancedMesh.name = 'initInstancedMesh'; -/* prpd初始Instanced点 */ -let initInstancedPoints,//prpd实例 - pointPositions, pointColors,//PRPS位置及颜色的集合 - pointsGeometry = new THREE.BufferGeometry(), - pointCount; -//pointsGeometry = new THREE.BoxGeometry(0, 0.8, (360 / initAxis) * 0.8,),// prpd位置属性数组 - -//pointsGeometry.computeVertexNormals() -let counter = 1; - -//菜单展示隐藏 -$('#show').click(function () { - $('.LinkANC').toggle(); - $('#associated-noise-box').toggle() - $('.eventList').toggle(); - $('#record-btn-box').toggle(); - $('.refreshtitle').toggle(); - $('#total').toggle(); - $('.noisy-slide-box').toggle(); - $('#addPointNumDiv').toggle(); - $(this).toggleClass('ready') -}) -//prps or prpd 切换 -$('#prps').click(function () { - updateParams('mapCategory', $(this), 1) - $('#box3D1').toggle(); - $('#box2D1').toggle(); - $('#prpd').toggleClass('ready') -}) - -$('#prpd').click(function () { - updateParams('mapCategory', $(this), 2) - $('#box2D1').toggle(); - $('#box3D1').toggle(); - $('#prps').toggleClass('ready') -}) -//暂停or开启 -$('#start').click(function () { - updateParams('switch', $(this)) -}) -//图谱放大缩小 -$('.scale').click(function () { - if ($(this).hasClass('big')) { - maxRange = maxRange + 25; - $('.small').removeClass('ready') - if (maxRange == 100) { - $(this).toggleClass('ready') - } - } else { - maxRange = maxRange - 25; - $('.big').removeClass('ready') - if (maxRange == 25) { - $(this).toggleClass('ready') - } - } - drawAxis((maxRange / 100) * initAxis, 0); - clearPoints() - let constant = (360 / (maxRange)) * maxRange + 1 - planes[0].constant = constant -}) -//自适应降噪 -$('#adaptive-noise').click(function () { - if (!realTime.adaptiveNoise) {//groundNoise - $(".noisy-slide").slider({ value: -80 }); - $(".noisy-slide span").attr('data-content', '底噪' + ' ' + -80); - $(".noisy-slide").slider({ disabled: true }); - realTime.groundNoise = 0; - } else { - $(".noisy-slide").slider({ disabled: false }); - } - updateParams('adaptiveNoise', $(this)) -}) -//关联传感器降噪 -$('#associated-noise').click(function () { - updateParams('correlationDenoising', $(this)) -}) -// 录播按钮点击 -$('.record-btn-box').click(function () { - let $this = $(this).children(); - if (socket) { - //初始状态:没暂停clss且录播状态为停止的时候添加该class - if (!$this.hasClass('pause') && !realTime.waveSwitch) { - $this.addClass('pause') - } - $this.toggleClass('live') - $this.toggleClass('pause') - } - updateParams('waveSwitch', null) -}) -$('.refresh-btn').click(function () { - $('#total').slideToggle() -}) -//累计打点时间输入 -$("#total-time").on("input", function () { // 使用 "input" 事件可以实现实时过滤 - var currentValue = $(this).val(); - var filteredValue = currentValue.replace(/[^0-9]/g, ''); // 使用正则表达式过滤非数字字符 - if (currentValue !== filteredValue) { - $(this).val(filteredValue); // 如果有变化,则更新输入框的值 - } -}); -$('#total-time').blur(function () { - const value = $(this).val() || 1 - updateParams('cumulativeDot', null, value) - if (value == 1) { - if (!$('.refresh-btn').hasClass('ready')) { - $('.refresh-btn').addClass('ready') - } - } else { - $('.refresh-btn').removeClass('ready') - } -}) -//频段切换 -$('#channelFiltering').on('change', function () { - let val = $(this).val(); - updateParams('channelFiltering', null, val) -}) - -//更新socket参数 -function updateParams(prams, that, value) { - if (socket) { - realTime[prams] = value || (realTime[prams] ? 0 : 1); - that && that.toggleClass('ready') - sendMessage() - } -} -loadOption(); -init(); -init2D(); -render(); -render2D(); -initTrendChart() -function initTrendChart() { - box2D6chart = echarts.init(box2D6) - setTimeout(() => { - box2D6chart.setOption(CPD); - }) -} -const generateArray = () => { - const newArray = []; - const step = 360 / phase; - for (let i = 1; i <= phase; i++) { - newArray.push([counter, (360 - parseInt(i * step)), getRandomInt(10, 65)]); - } - counter = (counter % period) + 1; - return newArray; -} -const generateArray1 = () => { - const newArray = []; - for (let i = 0; i < 4000; i++) { - newArray.push([getRandomInt(0, 360), getRandomInt(0, 80), getRandomInt(1, 15)]) - } - return newArray -} -function initPointsFun() { - if (initInstancedPoints) { - pointsList.remove(initInstancedPoints); - initInstancedPoints.material.dispose()//map(i => { i.dispose() }) - initInstancedPoints.geometry.dispose()//map(i => { i.disponse() }) - initInstancedPoints = null; - } - let maxYPointsMap = [800, 10000, 10000, 7000]; - let maxYPoints = maxYPointsMap[realTime.monitorType - 1]; - pointCount = phase * maxYPoints; - pointPositions = new Float32Array(pointCount * 3) // x, y, z - pointColors = new Float32Array(pointCount * 3) // r, g, b - // 初始化点的位置和颜色 - for (let i = 0; i < pointCount; i++) { - const x = i - const y = 9999 - const z = 10 - - pointPositions[i * 3] = x - pointPositions[i * 3 + 1] = y - pointPositions[i * 3 + 2] = z - - pointColors[i * 3] = Math.random() - pointColors[i * 3 + 1] = Math.random() - pointColors[i * 3 + 2] = Math.random() - } - pointsGeometry.setAttribute('position', new THREE.BufferAttribute(pointPositions, 3)) - pointsGeometry.setAttribute('color', new THREE.BufferAttribute(pointColors, 3)) - // 创建 ShaderMateria - const material = new THREE.PointsMaterial({ size: 2, vertexColors: true }) - // 创建 Points 对象 - initInstancedPoints = new THREE.Points(pointsGeometry, material) - initInstancedPoints.name = 'initInstancedPoints'; - // const object1 = meshList.getObjectByName('initInstancedMesh'); - // if (!object1) { - pointsList.add(initInstancedPoints); - //} -} -initCharts() -function initCharts(type) { - if (type == 'open') { - meshList.visible = true; - return; - } else if (type == 'close') { - meshList.visible = false; - let k = 1, s = 1; - for (let j = 0; j < period; j++) { - for (let i = 0; i < phase; i++) { - initInstancedMesh.getMatrixAt(k, dummy.matrix); - dummy.position.setFromMatrixPosition(dummy.matrix); - dummy.scale.setFromMatrixScale(dummy.matrix); - dummy.position.y = 99999; - dummy.scale.y = Math.abs(1); - const color = lut.getColor(Math.abs(0) / initAxis); - initInstancedMesh.setColorAt(k, color); - dummy.updateMatrix(); - initInstancedMesh.setMatrixAt(k, dummy.matrix); - k++; - } - } - initInstancedMesh.instanceMatrix.needsUpdate = true; - initInstancedMesh.instanceColor.needsUpdate = true; - for (let i = 0; i < pointCount; i++) { - pointPositions[i * 3 + 1] = 9999999 // Y//移到y轴高出,隐藏多余的点 - } - // 通知 Three.js 更新 BufferGeometry - pointsGeometry.attributes.position.needsUpdate = true - return; - } - let k = 0, initList = []; - for (let i = 0; i < period; i++) { - initList.push(generateArray()); - } - for (let j = 0; j < initList.length; j++) { - for (let i = 0; i < initList[j].length; i++) { - const [a, b, c] = initList[j][i]; - dummy.position.set(j, 99999, b); - dummy.scale.y = 0; - dummy.updateMatrix(); - initInstancedMesh.setMatrixAt(k, dummy.matrix); - - const color = lut.getColor(Math.abs(c) / initAxis); - initInstancedMesh.setColorAt(k, color); - k++; - } - } - initInstancedMesh.instanceMatrix.needsUpdate = true; - initInstancedMesh.instanceColor.needsUpdate = true; - const object = meshList.getObjectByName('initInstancedMesh'); - if (!object) { - meshList.add(initInstancedMesh); - } - initPointsFun() -} -function drawPRPD(list, clear) { - if (clear) { - if (initInstancedPoints) { - for (let i = 0; i < pointCount; i++) { - pointPositions[i * 3 + 1] = 9999999 // Y//移到y轴高出,隐藏多余的点 - } - // 通知 Three.js 更新 BufferGeometry - pointsGeometry.attributes.position.needsUpdate = true - } - } - const colors = [ - { - color: new THREE.Color(0x50A3BA), - num: 5, - }, - { - color: new THREE.Color(0x9DB578), - num: 7, - }, - { - color: new THREE.Color(0xEAC736), - num: 10, - }, - { - color: new THREE.Color(0xE28B4A), - num: 14, - }, - ]; - const range = realTime.monitorType == 1 ? maxRange : false; - const max = range ? initAxis * (range / 100) : initAxis; - list.forEach((item, index) => { - // 更新颜色 - const color = item[2] <= 3 ? new THREE.Color(0x0082df) : (item[2] >= 15 ? new THREE.Color(0xD94E5D) : colors.find(i => item[2] <= i.num).color); - pointPositions[index * 3] = 10 // X - pointPositions[index * 3 + 1] = item[1] > max ? max : item[1] // Y - pointPositions[index * 3 + 2] = 360 - item[0] // Z - pointColors[index * 3] = color.r //R - pointColors[index * 3 + 1] = color.g //G - pointColors[index * 3 + 2] = color.b //B - }) - for (let i = list.length - 1; i < pointCount; i++) { - pointPositions[i * 3 + 1] = 9999999 // Y//移到y轴高出,隐藏多余的点 - } - // 通知 Three.js 更新 BufferGeometry - pointsGeometry.attributes.position.needsUpdate = true - pointsGeometry.attributes.color.needsUpdate = true -} -function drawPRPS(data) { - let k = 0; - let list = JSON.parse(JSON.stringify(data)) - list = list.map((i) => ({ - h: i[2], - color: lut.getColor(Math.abs((i[2]) / initAxis).toFixed(2)), - })) - for (let j = 0; j < period; j++) { - for (let i = 0; i < list.length; i++) { - initInstancedMesh.getMatrixAt(k, dummy.matrix); - dummy.position.setFromMatrixPosition(dummy.matrix); - dummy.scale.setFromMatrixScale(dummy.matrix); - if (dummy.position.x > 0) { - dummy.position.x -= 1; - } else { - let { h, color } = list[i]; - dummy.position.x = 49; - dummy.position.y = h == 0 ? 99999 : h / 2; - dummy.scale.y = Math.abs(h); - initInstancedMesh.setColorAt(k, color); - } - dummy.updateMatrix(); - initInstancedMesh.setMatrixAt(k, dummy.matrix); - k++; - } - } - initInstancedMesh.instanceMatrix.needsUpdate = true; - initInstancedMesh.instanceColor.needsUpdate = true; -} -// 在页面加载完成后执行WebSocket连接的代码 -document.addEventListener('DOMContentLoaded', function () { - initSocket() - // aaa() - //bbb() -}); -function sendMessage() { - let ws = new WebSocket(url); // 替换成你的WebSocket服务器地址 - // 当WebSocket连接打开时,执行此函数 - ws.onopen = function (event) { - ws.send(JSON.stringify({ - "type": "sub", - "topic": `/device-message-sender/${productId}/${deviceId}`, - "parameter": { - "messageType": "INVOKE_FUNCTION", - "inputs": realTime, - "functionId": "requestPrpsData", - "headers": { - "async": true - } - }, - "id": "request-id" - })) - } -} -//滑动条初始化 -function LoadSlide() { - // 相位偏移滑动条 - $(".cdf-slide").slider({ - orientation: "horizontal", - animate: "slow", - min: -180, - max: 180, - step: 1, - value: 0, - create: function (event, ui) { - $(this).children('.ui-slider-handle').attr('data-content', '相位偏移' + ` 0 ` + '度') - }, - slide: function (event, ui) { - $(this).children('.ui-slider-handle').attr('data-content', '相位偏移' + ` ${ui.value} ` + '度'); - $(this).attr('data-price', ui.value); - - }, - change: function (event, ui) { - updateParams('migrationPhase', null, ui.value) - } - }); - - // 底噪滑动条 - $(".noisy-slide.GIS").slider({ - orientation: "vertical", - animate: "slow", - min: -80, - max: 0, - step: 1, - value: -80, - create: function (event, ui) { - $(this).children('.ui-slider-handle').attr('data-content', '底噪' + ` -80`) - }, - slide(event, ui) { - $(this).children('.ui-slider-handle').attr('data-content', '底噪' + ` ${ui.value}`); - }, - change: function (event, ui) { - updateParams('groundNoise', null, 80 + ui.value) - } - }); - - $(".noisy-slide.GIL").slider({ - orientation: "vertical", - animate: "slow", - min: 0, - max: 80, - step: 1, - value: 0, - create: function (event, ui) { - $(this).children('.ui-slider-handle').attr('data-content', '底噪' + ` 0`) - }, - slide(event, ui) { - $(this).children('.ui-slider-handle').attr('data-content', '底噪' + ` ${ui.value}`); - }, - change: function (event, ui) { - $(this).attr('data-price', ui.value); - if ($(this).parent().parent().parent().hasClass('realdata')) { - realTime.groundNoise = ui.value; - if (socket != '' && socket != null && !$('#adaptive-noise').hasClass("ready")) { - socket.emit('realTimeSet', realTime); - } - } - - } - }); - -} -//更新趋势图谱 -function updataHistory(timestamp, realData) { - let time = timestampToTime(timestamp), - data = [[time, (realData.max).toFixed(2)]], - data1 = [[time, realData.count]], - option = box2D6chart.getOption(); - if (option.xAxis[0].data.length > clearHistory) { - box2D6chart.setOption({ - xAxis: [{ - type: 'category', - data: [time], - }], - series: [ - { seriesIndex: 0, data }, - { seriesIndex: 1, data: data1 } - ] - }) - } else { - box2D6chart.setOption({ - xAxis: [{ - type: 'category', - data: [...box2D6chart.getOption().xAxis[0].data, time], - }] - }) - box2D6chart.appendData({ seriesIndex: 0, data }) - box2D6chart.appendData({ seriesIndex: 1, data: data1 }) - box2D6chart.resize() - } -} -function initSocket() { - socket = new WebSocket(url); // 替换成你的WebSocket服务器地址 - - // 当WebSocket连接打开时,执行此函数 - socket.onopen = function (event) { - console.log('WebSocket连接已打开'); - socket.send(JSON.stringify({ - "type": "sub", - "topic": `/device/${productId}/${deviceId}/message/property/report`, - "parameter": { - "messageType": "READ_PROPERTY", - }, - "id": "request-id" - })); - $('#start').addClass('ready') - $('.record-btn').addClass('ready') - $('#total-time').prop('disabled', false) - - sendMessage() - LoadSlide() - }; - - // 当接收到服务器发送的消息时,执行此函数 - socket.onmessage = function (event) { - - const data = JSON.parse(event.data); - const { payload: { properties: { realData }, timestamp } } = data - - if (realData?.lists2d?.length) { - drawEcharts1(null, realData.lists2d); - $('#maxValue').html(`${(realData.max).toFixed(2)}`); - $('#averageValue').html(`${(realData.avg).toFixed(2)}`); - $('#impulseQuantity').html(`${realData.count}`); - updataHistory(timestamp, realData) - } - if (realData?.lists3d?.length) { - drawEcharts1(realData.lists3d); - } - - }; - - // 当WebSocket连接关闭时,执行此函数 - socket.onclose = function (event) { - console.log('WebSocket连接已关闭'); - }; - - // 当发生错误时,执行此函数 - socket.onerror = function (error) { - console.error('WebSocket错误:', error); - }; -} -// 时间戳:1637244864707 -/* 时间戳转换为时间 */ -function timestampToTime(timestamp) { - timestamp = timestamp ? timestamp : null; - let date = new Date(timestamp);//时间戳为10位需*1000,时间戳为13位的话不需乘1000 - let Y = date.getFullYear() + '-'; - let M = (date.getMonth() + 1 < 10 ? '0' + (date.getMonth() + 1) : date.getMonth() + 1) + '-'; - let D = (date.getDate() < 10 ? '0' + date.getDate() : date.getDate()) + ' '; - let h = (date.getHours() < 10 ? '0' + date.getHours() : date.getHours()) + ':'; - let m = (date.getMinutes() < 10 ? '0' + date.getMinutes() : date.getMinutes()) + ':'; - let s = date.getSeconds() < 10 ? '0' + date.getSeconds() : date.getSeconds(); - return Y + M + D + h + m + s; -} - -function bbb() { - setTimeout(() => { - drawPRPD(generateArray1()) - const now = new Date(); - const hours = now.getHours(); - const minutes = now.getMinutes(); - const seconds = now.getSeconds(); - //console.log(box2D6chart.series) - const data1 = [[`${hours}:${minutes}:${seconds}`, getRandomInt(-20, -40)], ...box2D6chart.getOption().series[0].data] - //const data2 = [[`${hours}:${minutes}:${seconds}`, getRandomInt(2000, 4400)],...box2D7chart.getOption().series[0].data] - box2D6chart.setOption({ series: [{ type: 'line', data: data1 }] }); - bbb() - }, 1000) -} -function aaa() { - setTimeout(() => { - drawPRPS(generateArray()); - aaa(); - }, 20); -} -//aaa(); -// 生成指定范围内的随机整数 -function getRandomInt(min, max) { - const randomNumber = Math.random() * (max - min) + min; - const roundedNumber = randomNumber.toFixed(0); - return parseFloat(roundedNumber); -} - -//电缆局放折线图更新提交 -$('#form-submit').click(function () { - if ($('#realdata').attr('data-id')) { - socket.emit('getFrequencyData', { wl: $('#form-wl').val(), fltr: $('#form-fltr').val() }); - } -}) -//dom加载完成时 -$(window).ready(function () { - $(window).scrollTop(0, 0); - let maxHeight1 = 500, maxHeight = 273 - let h = $(window).height(); - let w = $(window).width(); - $(".loading-box").css('height', h * 0.75) - $(".loading-box-mini").css('height', h * 0.25) - $('.noisy-slide').css('height', h * 0.4) - - $('.toggle--checkbox').attr('checked', 'false'); - $('.record-btn').removeClass('ready'); - $('#total-time').prop('disabled', true) - - - let isLoad = { command: 'load', win: win } - // title 国际化 - window.parent.postMessage(isLoad, '*'); -}) -// 页面放大缩小时 -$(window).resize(function () { - window.requestAnimationFrame(function () { - onWindowResize() - }) -}) -$(window).resize() -//电缆局放时Three图谱和echarts图谱相互切换 -function cpdSwitchChart(type, callBack) { - if (type == 1) { - $('#box3D1').show(); - $('#box2D5').hide(); - $('#box2D1').show(); - $('#form-line').fadeOut(); - $('.cdf-slide-box').fadeIn(); - } else { - $('#form-line').fadeIn(); - $('.cdf-slide-box').fadeOut(); - $('#box3D1').hide(); - $('#box2D5').show(); - $('#box2D1').hide(); - if (echarts.getInstanceByDom(box2D5) == null) { - box2D5chart = echarts.init(box2D5) - } - - setTimeout(() => { - box2D5chart.setOption(CPD); - }) - } - if (callBack) { callBack() } - //onWindowResize() -} -//GIL模式时THREE图谱和echarts图谱相互切换 -function gilSwitchChart(type, callBack) { - if (type == 1) { - $('#box3D1').show(); - $('#box3D2').hide(); - $('#box2D1').show(); - $('#box2D2').hide(); - } else { - $('#box3D1').hide(); - $('#box3D2').show(); - $('#box2D1').hide(); - $('#box2D2').show(); - if (echarts.getInstanceByDom(box3D) == null) { - box3Dchart = echarts.init(box3D) - } - if (echarts.getInstanceByDom(box2D) == null) { - box2Dchart = echarts.init(box2D) - } - - box3Dchart.setOption(type == 2 ? GIL1 : GIL3, true); - box2Dchart.setOption(type == 2 ? GIL2 : GIL4, true); - - } - if (callBack) { callBack() } - //onWindowResize() - -} -//清空Three中的Point点位 -function clearPoints(type) { - // drawPRPS([], true); - drawPRPD([], true); -} -//电缆局放图谱更改 -$('#model-tab-dl button').click(function () { - let $this = $(this); - atlasType = $this.val(); - $this.addClass('active').siblings().removeClass('active'); - cpdSwitchChart(atlasType) - if (atlasType == 1) { - if ($('#realdata').attr('data-id')) { - if (socket != '' && socket != null) { - socket.emit('restartRealTime'); - } - } - } else if (atlasType == 2) { - - if ($('#realdata').attr('data-id')) { - if (socket != '' && socket != null) { - // socket.disconnect(); - socket.emit('getFrequencyData', { wl: $('#form-wl').val(), fltr: $('#form-fltr').val() }); - } - } - } - $(window).resize() -}) -// GIL 图谱更改 -$('#model-tab button').click(function () { - let $this = $(this); - realTime.atlasType = $this.val(); - $this.addClass('active').siblings().removeClass('active'); - gilSwitchChart($this.val()); - if ($this.val() == 1) { - $('.cdf-slide-box').fadeIn() - $('.noisy-slide-box').fadeIn() - rms = []; - peak = []; - p50 = []; - p100 = []; - } else if ($this.val() == 2) { - box2Dchart.setOption(GIL2, true); - $('.cdf-slide-box').fadeIn() - $('.noisy-slide-box').fadeIn() - rms = []; - peak = []; - p50 = []; - p100 = []; - } else { - $('.cdf-slide-box').fadeOut() - $('.noisy-slide-box').fadeOut() - box2Dchart.setOption(GIL4, true); - } - socket.emit('message', realTime); - $(window).resize() -}) -//量程切换 -// 实时数据量程修改 -$('#original').on('change', function () { - let val = ($(this).val() / 100) * initAxis; - console.log("🚀 val:", val) - - if (maxRange != val) { - clearPoints(); - let $this = $(this); - const max = ($this.val() / 100) * initAxis - $this.parent().addClass('is-checked').parent().addClass('is-checked').siblings().removeClass('is-checked').children().removeClass('is-checked') - $("[name='range']:checked").attr("checked", false); - $("[name='range'][value='" + $this.val() + "']").attr("checked", true); - //initAxis = val; - drawAxis(val, 0); - clearPoints() - maxRange = $this.val(); - let constant = (360 / (max)) * max + 1 - planes[0].constant = constant - } -}) -// 打开事件列表 -$('.event-btn').on('click', function () { - if ($(this).hasClass('ready')) { - var isFull = !!(document.webkitIsFullScreen || document.mozFullScreen || - document.msFullscreenElement || document.fullscreenElement - ); - - if (isFull) { - var dblChoseAlert = simpleAlert({ - "content": i18n.message.exitFullScreen + ` !`, - "buttons": { - "确定": function () { - dblChoseAlert.close(); - } - } - }) - } else { - let deviceNum = $('#realdata').attr('data-deviceNum'), - channelNum = $('#realdata').attr('data-chanel'), - name = $('#realdata').attr('data-name'); - - let data = { command: 'eventList', win: win, deviceNum: deviceNum, channelNum: channelNum, name: name }; - window.parent.postMessage(data, '*'); - } - } -}) -//全屏 -$(".charts").on("dblclick", function () { - if (viewIndex == 88) { - window.parent.postMessage({ command: 'closeFullScreen', index: viewIndex }, '*'); - } else { - window.parent.postMessage({ command: 'dblClick', index: viewIndex }, '*'); - } -}) -$("#prps-img").on('click', function () { - // 获取渲染器renderer的canvas,然后调用todataurl方法获取数据URI - let dataUrl = renderer.domElement.toDataURL(); - console.log("🚀 renderer.domElement:", renderer.domElement) - // 将DataURL转换为Blob对象 - let blob = dataURLtoBlob(dataUrl); - - // 创建下载链接 - let link = document.createElement('a'); - link.setAttribute('download', 'PRPS.png'); - let url = URL.createObjectURL(blob); - console.log("🚀 url:", url) - link.href = url//.replace('http', 'https'); - - // 将下载链接添加到HTML页面 - document.body.appendChild(link); - - // 触发下载链接的单击事件 - link.click(); -}) -function dataURLtoBlob(dataUrl) { - const arr = dataUrl.split(','); - const mime = arr[0].match(/:(.*?);/)[1]; - const bstr = atob(arr[1]); - let n = bstr.length; - const u8arr = new Uint8Array(n); - - while (n--) { - u8arr[n] = bstr.charCodeAt(n); - } - - return new Blob([u8arr], { type: mime }); -} -$("#prpd-img").on('click', function () { - // 获取渲染器renderer的canvas,然后调用todataurl方法获取数据URI - - let dataUrl = renderer2D.domElement.toDataURL('image/png'); - // 将DataURL转换为Blob对象 - let blob = dataURLtoBlob(dataUrl); - - // 创建下载链接 - let link = document.createElement('a'); - link.setAttribute('download', 'PRPD.png'); - link.href = URL.createObjectURL(blob); - - // 将下载链接添加到HTML页面 - document.body.appendChild(link); - - // 触发下载链接的单击事件 - link.click(); -}) -//主页面传递事件 -window.addEventListener('message', function (e) { - let data = e.data; - window.parent.postMessage(data, '*'); - if (data.command == 'i18n') { - viewIndex = viewIndex ? viewIndex : data.index; - $('.phaseOffset').html('相位偏移') - $('.LinkNC').attr('title', '关联噪声传感器降噪') - $('.LinkANC').attr('title', '自适应降噪') - $('#record-btn-box').attr('title', '录波') - $('#toggle-box').attr('title', '开启/暂停') - $('.noise').html('底噪') - // $('.addPointNum').html(CumulativeRBI) - $('.eventList').attr('title', '事件列表') - $('.refreshtitle').attr('title', '刷新') - - $('.channelNum .name').html('监测点' + ":") - $('.maxValue .name').html('脉冲最大值' + ":") - // $('.pc-tip .name').html(discharge + ':') - // $('.avgValue .name').html(data.i18n.content.realTime.gil.averageValue + ":") - $('.pulseNumber .name').html('脉冲数量' + ":") - //onWindowResize() - } else if (data.command == 'setView') { - viewType = data.viewType; - onWindowResize() - } else if (data.command == 'open') { - - $('#box3D1').show(); - $('#box3D2').hide(); - let { id, name, equipmentId, channelNum, monitorType, ipAddress } = data; - $("#range100").click(); - $('#model-tab-dl-1').click(); - $('#realdata').attr('data-id', id) - $('#realdata').attr('data-name', name) - $('#realdata').attr('data-deviceNum', equipmentId) - $('#realdata').attr('data-chanel', channelNum) - atlasType = 1; - initChangeType(monitorType) - //不一致 切换图谱 - if (realTime.monitorType != monitorType) { - realTime.monitorType = monitorType; - changeEcharts(monitorType); - initPointsFun(); - } - initCharts('open'); - if ($('#realdata').children('.echarts-btn').first().children('.toggle--knob').children('.toggle--checkbox').is(':checked')) { - // console.log("kai----"+data.win) - // console.log(data) - // console.log($('#toggle--knob1')) - $('#toggle--knob1').click() - } - win = data.win; - initRealTime(); - } else if (data.command == 'close') { - if ($('#realdata').attr('data-id') == data.id) { - $('#realdata').attr('data-id', '') - $('#realdata').attr('data-name', '') - $('#realdata').attr('data-deviceNum', '') - $('#realdata').attr('data-chanel', '') - $(".noisy-slide").slider({ disabled: false }); - if (!$('#realdata').children('.echarts-btn').first().children('.toggle--knob').children('.toggle--checkbox').is(':checked')) { - $('#toggle--knob1').click() - } - $('#channelNum').html('&nbsp;&nbsp;'); - $('#maxValue').html("0"); - $('#averageValue').html("0"); - $('#impulseQuantity').html("0"); - // $('#pc-number').html('0'); - $('#clear').click(); - $('#model-tab-dl-1').click() - $('#associated-noise').addClass('disabled'); - $('#adaptive-noise').addClass('disabled'); - clearPoints() - win = undefined; - initCharts('close'); - } - } else if (data.command == 'historyPRPD') { - - } else if (data.command == 'historyPRPS') { - - } -}) -function changeEcharts(type) { - upDrawAxis(0, 0, type) - if (type == 1) { - $('.unit').html('dBm') - $(".noisy-slide.GIL").fadeOut(() => { - $(".noisy-slide.GIS").fadeIn(); - }); - // $('.pc-tip').hide(); - $('#model-tab-dl').hide(); - - } else if (type == 2 || type == 3) { - realTime.atlasType = 1; - $('#model-tab button:first-child').addClass('active').siblings().removeClass('active') - $('.unit').html('mV') - - $(".noisy-slide.GIS").fadeOut(() => { - $(".noisy-slide.GIL").fadeIn(); - }); - //$('.pc-tip').hide(); - $('#model-tab-dl').hide(); - } else if (type == 4) { - atlasType = 1; - $('#model-tab-dl button:first-child').addClass('active').siblings().removeClass('active') - $('.unit').html('mV'); - //$('.pc-tip').show(); - $('#model-tab-dl').show() - } -} -function initRealTime() { - $('.associated-noise').removeClass('ready'); - $('.adaptive-noise').removeClass('ready'); - $("#channelFiltering").val(0); - if (realTime.monitorType == 1) { - $(".noisy-slide").slider({ value: -80 }); - $(".noisy-slide span").attr('data-content', i18n.content.realTime.gis.noise1 + ' ' + -80); - } else if (realTime.monitorType == 2 || realTime.monitorType == 3) { - $(".noisy-slide").slider({ value: 0 }); - $(".noisy-slide span").attr('data-content', i18n.content.realTime.gis.noise1 + ' ' + 0); - } - - - realTime.channelFiltering = 0; - realTime.cumulativeDot = 1; - realTime.correlationDenoising = 0; - realTime.adaptiveNoise = 0; - realTime.groundNoise = 0; -} -function downloadPng(type) { - let dataUrl = ''; - // 获取渲染器renderer的canvas,然后调用todataurl方法获取数据URI - if (type) { - dataUrl = renderer.domElement.toDataURL('image/png'); - } else { - dataUrl = renderer2D.domElement.toDataURL('image/png'); - } - - // 将DataURL转换为Blob对象 - let blob = new Blob([dataUrl], { type: 'image/png' }); - - // 创建下载链接 - let link = document.createElement('a'); - link.setAttribute('download', 'my_scene_image.png'); - link.href = URL.createObjectURL(blob); - - // 将下载链接添加到HTML页面 - document.body.appendChild(link); - - // 触发下载链接的单击事件 - link.click(); -} -//初始化绘制坐标轴文字 -function drawAxisText(textMap, type) { - let setTextMap = ''; - if (textMap) { - setTextMap = textMap; - } else { - let { - content: { - analyzeDiagnosis: { - analyzeDiagnosis: { amplitude } - }, - }, - tip: { period, phase, } } = i18n; - let inifo = { - amplitude, - period, - phase, - }; - setTextMap = inifo; - } - if (type == 2 || type == 3) { - setTextMap.amplitude = 'mV'; - } else if (type == 4) { - setTextMap.amplitude = 'mV'; - } - let infoNum = [ - {}, { - w: 60, - h: 32, - cw: 9, - ch: 8, - size: 26.3, - }, {}, { - w: 125, - h: 50, - size: 38.3, - }, { - w: 125, - h: 50, - size: 38.3, - }, - { - w: 130, - h: 50, - size: 38.3, - }, {}, {}, { - w: 165, - h: 122, - size: 68.3, - }, - ] - let axisText = [{ - type: 'text', - lang: true, - content: 'amplitude', - rotate: 90, - color: '#000000', - size: 30, - font: 'normal Bold 500px Arial,sans-serif', - xyz: [-60, minNum ? 0 : 120, 382], - }, - { - type: 'text', - lang: true, - content: 'period', - xyz: [150, -180, 420], - xyz: [minNum ? 250 : 150, minNum ? -220 : -130, 420], - color: '#000000', - size: 30, - font: 'normal Bold 500px Arial,sans-serif', - }, - { - type: 'text', - lang: true, - content: 'phase', - color: '#000000', - size: 30, - font: 'normal Bold 500px Arial,sans-serif', - xyz: [500, -110, 140], - },]; - let axis2dText = [{ - type: 'text', - lang: true, - content: 'amplitude', - rotate: 10, - color: '#000000', - size: 30, - font: 'normal Bold 500px Arial,sans-serif', - xyz: [-20, minNum ? 0 : 180, 405], - }, - { - type: 'text', - lang: true, - content: 'phase', - color: '#000000', - size: 30, - font: 'normal Bold 500px Arial,sans-serif', - xyz: [0, minNum ? -265 : -87, 180], - },]; - let setList = (list) => { - let newList = []; - for (let i of list) { - let str = setTextMap[i.content]; - i.content = [str]; - let { w, h, cw, ch, size } = infoNum[str.length - 1] - i.w = i.rotate ? h : w; - i.h = i.rotate ? w : h; - i.cw = i.rotate ? ch : cw; - i.ch = i.rotate ? cw : ch; - i.size = size; - if (i.rotate) { - i.xyz[2] = 445 + size / 2; - } - newList.push(draw2DText(i)) - } - return newList - } - axisTextGroup.children.length && axisTextGroup.remove(...axisTextGroup.children) - axisTextGroup.add(...setList(axisText)) - - axisTextGroup2D.children.length && axisTextGroup2D.remove(...axisTextGroup2D.children) - axisTextGroup2D.add(...setList(axis2dText)) -} -//根据坐标轴类型更新坐标轴 -function upDrawAxis(max, min, type) { - if (!type) { - initAxis = max; minNum = min; drawAxis(initAxis, minNum); - return; - } - if (type == 1) { - initAxis = 80; minNum = 0; - drawAxis(initAxis, minNum); - } else if (type == 3 || type == 2) { - initAxis = 500; minNum = 0; - drawAxis(initAxis, minNum) - } else if (type == 4) { - initAxis = 2000; minNum = -2000; - drawAxis(initAxis, minNum); - } - drawAxisText(0, type); - onWindowResize() - planes[0].constant = (360 / (initAxis)) * initAxis + 1 -} -renderer.domElement.addEventListener('webglcontextlost', handleContextLost, false); -function handleContextLost(event) { - event.preventDefault(); - init(); -} -renderer2D.domElement.addEventListener('webglcontextlost', handleContextLost2, false); -function handleContextLost2(event) { - event.preventDefault(); - init2D(); -} -function handleVisibilityChange() { - if (document.hidden) { - // 页面被最小化或切换应用程序,暂停渲染并清除 InstancedMesh 相关资源 - instancedMesh.dispose(); - } else { - // 页面重新显示,重新创建 WebGL 上下文并重新设置 InstancedMesh - init(); - } -} -//prps初始化 -function init() { - let box = $('#box3D1'); - let width = box.width(); - let height = box.height(); - renderer = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true, antialias: true, preserveDrawingBuffer: true, });//preserveDrawingBuffer保存three.js canvas画布上的信息 - renderer.localClippingEnabled = true - let constant = (360 / (initAxis)) * initAxis + 1 - planes = [ - new THREE.Plane(new THREE.Vector3(0, -1, 0), constant), //y轴正值截取 - ] - renderer.clippingPlanes = planes - renderer.setSize(width, height);//设置渲染区域尺寸 - renderer.setClearColor(0xffffff, 1); //设置背景颜色 - //renderer.setClearColor(0xb9d3ff, 1); //设置背景颜色 - renderer.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 - box?.append(renderer.domElement); //body元素中插入canvas对象 - scene = new THREE.Scene(); - scene.add(meshList, axis, pointsList, axisTextGroup); - - /* var point = new THREE.PointLight(0xffffff); - point.position.set(600, 600, 600); //点光源位置 - scene.add(point); //点光源添加到场景中 - var ambient = new THREE.AmbientLight(0x444444); - scene.add(ambient); */ - - drawAxis(initAxis, minNum, true) - - var k = width / height; //窗口宽高比 - var s = 420; //三维场景显示范围控制系数,系数越大,显示的范围越大 - //创建相机对象 - camera = new THREE.OrthographicCamera(-s * k, s * k, s, -s, 1, 1600); - camera.position.set(500, 200, 500); //设置相机位置 - camera.lookAt(scene.position); //设置相机方向(指向的场景对象) - controls = new THREE.OrbitControls(camera, renderer.domElement); - // 设置左右方向的最大、最小角度限制为 90 度和 0 度 - controls.minAzimuthAngle = 0; - controls.maxAzimuthAngle = Math.PI / 2; - // 设置上下方向的最大、最小角度限制为 90 度和 0 度 - controls.minPolarAngle = 0; - controls.maxPolarAngle = Math.PI / 2; - //设置放大缩小上下限 - controls.minZoom = 0.5; - controls.maxZoom = 2; - //controls.enablePan = false; - //设置控制器中心点 - controls.target.set(25, 0, 180); - renderer.render(scene, camera); - - //controls.addEventListener('change', render); -} -//prpd初始化 -function init2D() { - let box = $('#box2D1'); - let width = box.width(); - let height = box.height(); - renderer2D = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true, antialias: true, preserveDrawingBuffer: true, });//preserveDrawingBuffer保存three.js canvas画布上的信息\ - renderer2D.localClippingEnabled = true - let constant = (360 / (initAxis)) * initAxis + 1 - planes = [ - new THREE.Plane(new THREE.Vector3(0, -1, 0), constant), //y轴正值截取 - ] - renderer2D.clippingPlanes = planes - renderer2D.setSize(width, height);//设置渲染区域尺寸 - renderer2D.setClearColor(0xffffff, 1); //设置背景颜色 - renderer2D.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 - box?.append(renderer2D.domElement); //body元素中插入canvas对象 - scene2D = new THREE.Scene(); - const List = pointsList.clone() - scene2D.add(axis2D, List, axisTextGroup2D, pointsList) - drawAxis2D(initAxis, minNum) - // 辅助坐标系 参数250表示坐标系大小,可以根据场景大小去设置 - // let axisHelper = new THREE.AxesHelper(250); - // scene2D.add(axisHelper); - var k = width / height; //窗口宽高比 - var s = 380; //三维场景显示范围控制系数,系数越大,显示的范围越大 - //创建相机对象 - camera2D = new THREE.OrthographicCamera(-s * k - 200, s * k - 200, s + 160, -s + 160, 1, 1200); - camera2D.position.set(600, 0, 0); //设置相机位置 - camera2D.lookAt(scene2D.position); //设置相机方向(指向的场景对象) - controls2D = new THREE.OrbitControls(camera2D, renderer2D.domElement); - controls2D.enablePan = false//右键位移禁用 - controls2D.enableZoom = false//放大缩小禁用 - //设置放大缩小上下限 - controls2D.minZoom = 0.5; - controls2D.maxZoom = 2; - controls2D.enableRotate = false - renderer2D.render(scene2D, camera2D); - controls2D.addEventListener('change', render2D); -} -function onWindowResize() { - if (viewType == 1 || viewType == 2) {//全功能PrpsAndPrpd - let h = $(window).height(); - let w = $(window).width(); - $(".loading-box").css('height', h * 0.65) - $(".loading-box-mini").css('height', h * 0.35) - $('.noisy-slide').css('height', h * 0.4) - $('#realdata').show(); - $('.noisy-slide-box').show(); - $('#echarts-text').show(); - $('#addPointNumDiv').show(); - $('#box3D1').show(); - $('#box2D1').hide(); - $('#box2D6').show(); - } else if (viewType == 3) {//实时数据仅展示prpd - let h = $(window).height(); - let w = $(window).width(); - $(".loading-box").css('height', h * 0.9) - $(".loading-box").css('width', w * 0.9) - $('#realdata').hide(); - $('.noisy-slide-box').hide(); - $('#echarts-text').hide(); - $('#addPointNumDiv').hide(); - $('#box3D1').hide(); - $('#box2D1').show(); - $('#box2D6').hide(); - } else if (viewType == 4) {//历史趋势仅展示prps - let h = $(window).height(); - let w = $(window).width(); - $(".loading-box").css('height', h * 0.9) - $(".loading-box").css('width', w * 0.9) - $('#realdata').hide(); - $('.noisy-slide-box').hide(); - $('#echarts-text').hide(); - $('#addPointNumDiv').hide(); - $('#box3D1').hide(); - $('#box2D1').show(); - $('#box2D6').hide(); - } else if (viewType == 5) {//历史趋势仅展示prpd - let h = $(window).height(); - let w = $(window).width(); - $(".loading-box").css('height', h * 0.9) - $(".loading-box").css('width', w * 0.9) - $('#realdata').hide(); - $('.noisy-slide-box').hide(); - $('#echarts-text').hide(); - $('#addPointNumDiv').hide(); - $('#box3D1').hide(); - $('#box2D1').show(); - $('#box2D6').hide(); - } - - if (echarts.getInstanceByDom(box3D) != null) { - box3Dchart.resize(); - } - if (echarts.getInstanceByDom(box2D) != null) { - box2Dchart.resize(); - } - if (echarts.getInstanceByDom(box2D5) != null) { - box2D5chart.resize(); - } - setTimeout(() => { - box2D6chart.resize(); - }) - - let box = $('#box3D1'); - let width = box.width(); - - let height = box.height(); - let k = width / height; //窗口宽高比 - let s = 350; //三维场景显示范围控制系数,系数越大,显示的范围越大 - - camera.left = -s * k - 20; - camera.right = s * k - 20; - camera.top = s + (minNum ? -80 : 60); - camera.bottom = -s + (minNum ? -80 : 60); - camera.aspect = k; - camera.updateProjectionMatrix(); - renderer.setSize(width, height); - renderer.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 - - let s2D = 270; - camera2D.left = -s2D * k - 200; - camera2D.right = s2D * k - 200; - camera2D.top = s2D + (minNum ? -30 : 155); - camera2D.bottom = -s2D + (minNum ? -30 : 155); - camera2D.aspect = k; - camera2D.updateProjectionMatrix(); - renderer2D.setSize(width, height); - renderer2D.setPixelRatio(window.devicePixelRatio)//设备像素比,优化渲染效果 -} -function render() { - // 获取摄像机的视锥体 - const frustum = new THREE.Frustum(); - const cameraViewProjectionMatrix = new THREE.Matrix4(); - camera.updateMatrixWorld(); // 确保摄像机的世界矩阵已更新 - cameraViewProjectionMatrix.multiplyMatrices(camera.projectionMatrix, camera.matrixWorldInverse); - frustum.setFromProjectionMatrix(cameraViewProjectionMatrix); - - // 遍历场景中的每个对象,并根据视锥体进行剔除 - scene.traverse((object) => { - if (object.isMesh) { - object.visible = frustum.intersectsObject(object); - } - }); - // 更新帧率显示器 - //stats.update(); - renderer.render(scene, camera); - requestAnimationFrame(render) -} -function render2D() { - //scene.rotateY(0.001);//每次绕y轴旋转0.01弧度 - requestAnimationFrame(render2D) - renderer2D.render(scene2D, camera2D); -} -function initChangeType(monitorType) { - gilSwitchChart(1) - cpdSwitchChart(1) - upDrawAxis(0, 0, monitorType) - if (monitorType == 1) { - //$('.avgValue').fadeOut() - - $('#model-tab').fadeOut(() => { - $('#channelFiltering').fadeIn() - $('.pm-slide-box').fadeIn() - $('#associated-noise-box').fadeIn() - // $('#record-btn-box').fadeIn() - $('#toggle-box').fadeIn() - $('.noisy-slide-box').fadeIn() - $('.radiogroup').fadeIn() - $('.cdf-slide-box').fadeIn() - $('.eventList').fadeIn() - $('.refreshtitle').fadeIn() - - $('#associated-noise').removeClass('disabled'); - $('#adaptive-noise').removeClass('disabled'); - $(".noisy-slide").slider({ disabled: false }); - $("#channelFiltering").val(0); - - $('.monitorType').html('UHF') - }) - } else if (monitorType == 2 || monitorType == 3) { - $('#channelFiltering').fadeOut(() => { - $('#model-tab').fadeIn() - if (realTime.atlasType == 1) { - $('#model-tab button:first-child').addClass('active').siblings().removeClass('active') - } else if (realTime.atlasType == 2) { - $('#model-tab button:nth-child(2)').addClass('active').siblings().removeClass('active') - } else { - $('#model-tab button:last-child').addClass('active').siblings().removeClass('active') - $('.cdf-slide-box').fadeOut() - $('.noisy-slide-box').fadeOut() - } - }) - $('.pm-slide-box').fadeOut() - $('#associated-noise-box').fadeOut() - // $('#record-btn-box').fadeOut() - // $('#toggle-box').fadeOut() - $('.radiogroup').fadeOut() - // $('.eventList').fadeOut() - $('.refreshtitle').fadeOut() - - //$('.avgValue').fadeIn() - $('.avgValue .name').html(i18n.content.realTime.gil.averageValue + ":") - - $('.monitorType').html('AE') - $('#line-btn-box').hide(); - $('#form-line').hide(); - } else if (monitorType == 4) { - $('#channelFiltering').fadeOut() - $('.pm-slide-box').fadeOut() - $('#associated-noise-box').fadeOut() - // $('#record-btn-box').fadeOut() - // $('#toggle-box').fadeOut() - $('.noisy-slide-box').fadeOut() - $('.radiogroup').fadeOut() - $('.refreshtitle').fadeOut() - $('#model-tab').fadeOut() - - $('.cdf-slide-box').fadeIn() - $('.eventList').fadeIn() - //$('.avgValue').fadeIn() - $('.avgValue .name').html(i18n.content.realTime.gis.averageValue + ":") - - $('.monitorType').html('HFCT') - } -} -// 图表绘制 -function drawEcharts1(list1, list2) { - // monitorType GIS 1 GIL故障定位 2 GIL局放 3 电缆 4 - // atlasType 1 PRPS/PRPD 2 飞行模式/脉冲数据 3 连续模式 - if (list1?.length) { - drawPRPS(list1) - } - if (list2?.length) { - drawPRPD(list2) - } -} -export { scene, axis, meshList, pointsList, axis2D, scene2D, i18n, } \ No newline at end of file diff --git a/public/static/common/js/jquery-ui.min.js b/public/static/common/js/jquery-ui.min.js deleted file mode 100644 index 25398a1..0000000 --- a/public/static/common/js/jquery-ui.min.js +++ /dev/null @@ -1,13 +0,0 @@ -/*! jQuery UI - v1.12.1 - 2016-09-14 -* http://jqueryui.com -* Includes: widget.js, position.js, data.js, disable-selection.js, effect.js, effects/effect-blind.js, effects/effect-bounce.js, effects/effect-clip.js, effects/effect-drop.js, effects/effect-explode.js, effects/effect-fade.js, effects/effect-fold.js, effects/effect-highlight.js, effects/effect-puff.js, effects/effect-pulsate.js, effects/effect-scale.js, effects/effect-shake.js, effects/effect-size.js, effects/effect-slide.js, effects/effect-transfer.js, focusable.js, form-reset-mixin.js, jquery-1-7.js, keycode.js, labels.js, scroll-parent.js, tabbable.js, unique-id.js, widgets/accordion.js, widgets/autocomplete.js, widgets/button.js, widgets/checkboxradio.js, widgets/controlgroup.js, widgets/datepicker.js, widgets/dialog.js, widgets/draggable.js, widgets/droppable.js, widgets/menu.js, widgets/mouse.js, widgets/progressbar.js, widgets/resizable.js, widgets/selectable.js, widgets/selectmenu.js, widgets/slider.js, widgets/sortable.js, widgets/spinner.js, widgets/tabs.js, widgets/tooltip.js -* Copyright jQuery Foundation and other contributors; Licensed MIT */ - -(function(t){"function"==typeof define&&define.amd?define(["jquery"],t):t(jQuery)})(function(t){function e(t){for(var e=t.css("visibility");"inherit"===e;)t=t.parent(),e=t.css("visibility");return"hidden"!==e}function i(t){for(var e,i;t.length&&t[0]!==document;){if(e=t.css("position"),("absolute"===e||"relative"===e||"fixed"===e)&&(i=parseInt(t.css("zIndex"),10),!isNaN(i)&&0!==i))return i;t=t.parent()}return 0}function s(){this._curInst=null,this._keyEvent=!1,this._disabledInputs=[],this._datepickerShowing=!1,this._inDialog=!1,this._mainDivId="ui-datepicker-div",this._inlineClass="ui-datepicker-inline",this._appendClass="ui-datepicker-append",this._triggerClass="ui-datepicker-trigger",this._dialogClass="ui-datepicker-dialog",this._disableClass="ui-datepicker-disabled",this._unselectableClass="ui-datepicker-unselectable",this._currentClass="ui-datepicker-current-day",this._dayOverClass="ui-datepicker-days-cell-over",this.regional=[],this.regional[""]={closeText:"Done",prevText:"Prev",nextText:"Next",currentText:"Today",monthNames:["January","February","March","April","May","June","July","August","September","October","November","December"],monthNamesShort:["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"],dayNames:["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"],dayNamesShort:["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],dayNamesMin:["Su","Mo","Tu","We","Th","Fr","Sa"],weekHeader:"Wk",dateFormat:"mm/dd/yy",firstDay:0,isRTL:!1,showMonthAfterYear:!1,yearSuffix:""},this._defaults={showOn:"focus",showAnim:"fadeIn",showOptions:{},defaultDate:null,appendText:"",buttonText:"...",buttonImage:"",buttonImageOnly:!1,hideIfNoPrevNext:!1,navigationAsDateFormat:!1,gotoCurrent:!1,changeMonth:!1,changeYear:!1,yearRange:"c-10:c+10",showOtherMonths:!1,selectOtherMonths:!1,showWeek:!1,calculateWeek:this.iso8601Week,shortYearCutoff:"+10",minDate:null,maxDate:null,duration:"fast",beforeShowDay:null,beforeShow:null,onSelect:null,onChangeMonthYear:null,onClose:null,numberOfMonths:1,showCurrentAtPos:0,stepMonths:1,stepBigMonths:12,altField:"",altFormat:"",constrainInput:!0,showButtonPanel:!1,autoSize:!1,disabled:!1},t.extend(this._defaults,this.regional[""]),this.regional.en=t.extend(!0,{},this.regional[""]),this.regional["en-US"]=t.extend(!0,{},this.regional.en),this.dpDiv=n(t("<div id='"+this._mainDivId+"' class='ui-datepicker ui-widget ui-widget-content ui-helper-clearfix ui-corner-all'></div>"))}function n(e){var i="button, .ui-datepicker-prev, .ui-datepicker-next, .ui-datepicker-calendar td a";return e.on("mouseout",i,function(){t(this).removeClass("ui-state-hover"),-1!==this.className.indexOf("ui-datepicker-prev")&&t(this).removeClass("ui-datepicker-prev-hover"),-1!==this.className.indexOf("ui-datepicker-next")&&t(this).removeClass("ui-datepicker-next-hover")}).on("mouseover",i,o)}function o(){t.datepicker._isDisabledDatepicker(m.inline?m.dpDiv.parent()[0]:m.input[0])||(t(this).parents(".ui-datepicker-calendar").find("a").removeClass("ui-state-hover"),t(this).addClass("ui-state-hover"),-1!==this.className.indexOf("ui-datepicker-prev")&&t(this).addClass("ui-datepicker-prev-hover"),-1!==this.className.indexOf("ui-datepicker-next")&&t(this).addClass("ui-datepicker-next-hover"))}function a(e,i){t.extend(e,i);for(var s in i)null==i[s]&&(e[s]=i[s]);return e}function r(t){return function(){var e=this.element.val();t.apply(this,arguments),this._refresh(),e!==this.element.val()&&this._trigger("change")}}t.ui=t.ui||{},t.ui.version="1.12.1";var h=0,l=Array.prototype.slice;t.cleanData=function(e){return function(i){var s,n,o;for(o=0;null!=(n=i[o]);o++)try{s=t._data(n,"events"),s&&s.remove&&t(n).triggerHandler("remove")}catch(a){}e(i)}}(t.cleanData),t.widget=function(e,i,s){var n,o,a,r={},h=e.split(".")[0];e=e.split(".")[1];var l=h+"-"+e;return s||(s=i,i=t.Widget),t.isArray(s)&&(s=t.extend.apply(null,[{}].concat(s))),t.expr[":"][l.toLowerCase()]=function(e){return!!t.data(e,l)},t[h]=t[h]||{},n=t[h][e],o=t[h][e]=function(t,e){return this._createWidget?(arguments.length&&this._createWidget(t,e),void 0):new o(t,e)},t.extend(o,n,{version:s.version,_proto:t.extend({},s),_childConstructors:[]}),a=new i,a.options=t.widget.extend({},a.options),t.each(s,function(e,s){return t.isFunction(s)?(r[e]=function(){function t(){return i.prototype[e].apply(this,arguments)}function n(t){return i.prototype[e].apply(this,t)}return function(){var e,i=this._super,o=this._superApply;return this._super=t,this._superApply=n,e=s.apply(this,arguments),this._super=i,this._superApply=o,e}}(),void 0):(r[e]=s,void 0)}),o.prototype=t.widget.extend(a,{widgetEventPrefix:n?a.widgetEventPrefix||e:e},r,{constructor:o,namespace:h,widgetName:e,widgetFullName:l}),n?(t.each(n._childConstructors,function(e,i){var s=i.prototype;t.widget(s.namespace+"."+s.widgetName,o,i._proto)}),delete n._childConstructors):i._childConstructors.push(o),t.widget.bridge(e,o),o},t.widget.extend=function(e){for(var i,s,n=l.call(arguments,1),o=0,a=n.length;a>o;o++)for(i in n[o])s=n[o][i],n[o].hasOwnProperty(i)&&void 0!==s&&(e[i]=t.isPlainObject(s)?t.isPlainObject(e[i])?t.widget.extend({},e[i],s):t.widget.extend({},s):s);return e},t.widget.bridge=function(e,i){var s=i.prototype.widgetFullName||e;t.fn[e]=function(n){var o="string"==typeof n,a=l.call(arguments,1),r=this;return o?this.length||"instance"!==n?this.each(function(){var i,o=t.data(this,s);return"instance"===n?(r=o,!1):o?t.isFunction(o[n])&&"_"!==n.charAt(0)?(i=o[n].apply(o,a),i!==o&&void 0!==i?(r=i&&i.jquery?r.pushStack(i.get()):i,!1):void 0):t.error("no such method '"+n+"' for "+e+" widget instance"):t.error("cannot call methods on "+e+" prior to initialization; "+"attempted to call method '"+n+"'")}):r=void 0:(a.length&&(n=t.widget.extend.apply(null,[n].concat(a))),this.each(function(){var e=t.data(this,s);e?(e.option(n||{}),e._init&&e._init()):t.data(this,s,new i(n,this))})),r}},t.Widget=function(){},t.Widget._childConstructors=[],t.Widget.prototype={widgetName:"widget",widgetEventPrefix:"",defaultElement:"<div>",options:{classes:{},disabled:!1,create:null},_createWidget:function(e,i){i=t(i||this.defaultElement||this)[0],this.element=t(i),this.uuid=h++,this.eventNamespace="."+this.widgetName+this.uuid,this.bindings=t(),this.hoverable=t(),this.focusable=t(),this.classesElementLookup={},i!==this&&(t.data(i,this.widgetFullName,this),this._on(!0,this.element,{remove:function(t){t.target===i&&this.destroy()}}),this.document=t(i.style?i.ownerDocument:i.document||i),this.window=t(this.document[0].defaultView||this.document[0].parentWindow)),this.options=t.widget.extend({},this.options,this._getCreateOptions(),e),this._create(),this.options.disabled&&this._setOptionDisabled(this.options.disabled),this._trigger("create",null,this._getCreateEventData()),this._init()},_getCreateOptions:function(){return{}},_getCreateEventData:t.noop,_create:t.noop,_init:t.noop,destroy:function(){var e=this;this._destroy(),t.each(this.classesElementLookup,function(t,i){e._removeClass(i,t)}),this.element.off(this.eventNamespace).removeData(this.widgetFullName),this.widget().off(this.eventNamespace).removeAttr("aria-disabled"),this.bindings.off(this.eventNamespace)},_destroy:t.noop,widget:function(){return this.element},option:function(e,i){var s,n,o,a=e;if(0===arguments.length)return t.widget.extend({},this.options);if("string"==typeof e)if(a={},s=e.split("."),e=s.shift(),s.length){for(n=a[e]=t.widget.extend({},this.options[e]),o=0;s.length-1>o;o++)n[s[o]]=n[s[o]]||{},n=n[s[o]];if(e=s.pop(),1===arguments.length)return void 0===n[e]?null:n[e];n[e]=i}else{if(1===arguments.length)return void 0===this.options[e]?null:this.options[e];a[e]=i}return this._setOptions(a),this},_setOptions:function(t){var e;for(e in t)this._setOption(e,t[e]);return this},_setOption:function(t,e){return"classes"===t&&this._setOptionClasses(e),this.options[t]=e,"disabled"===t&&this._setOptionDisabled(e),this},_setOptionClasses:function(e){var i,s,n;for(i in e)n=this.classesElementLookup[i],e[i]!==this.options.classes[i]&&n&&n.length&&(s=t(n.get()),this._removeClass(n,i),s.addClass(this._classes({element:s,keys:i,classes:e,add:!0})))},_setOptionDisabled:function(t){this._toggleClass(this.widget(),this.widgetFullName+"-disabled",null,!!t),t&&(this._removeClass(this.hoverable,null,"ui-state-hover"),this._removeClass(this.focusable,null,"ui-state-focus"))},enable:function(){return this._setOptions({disabled:!1})},disable:function(){return this._setOptions({disabled:!0})},_classes:function(e){function i(i,o){var a,r;for(r=0;i.length>r;r++)a=n.classesElementLookup[i[r]]||t(),a=e.add?t(t.unique(a.get().concat(e.element.get()))):t(a.not(e.element).get()),n.classesElementLookup[i[r]]=a,s.push(i[r]),o&&e.classes[i[r]]&&s.push(e.classes[i[r]])}var s=[],n=this;return e=t.extend({element:this.element,classes:this.options.classes||{}},e),this._on(e.element,{remove:"_untrackClassesElement"}),e.keys&&i(e.keys.match(/\S+/g)||[],!0),e.extra&&i(e.extra.match(/\S+/g)||[]),s.join(" ")},_untrackClassesElement:function(e){var i=this;t.each(i.classesElementLookup,function(s,n){-1!==t.inArray(e.target,n)&&(i.classesElementLookup[s]=t(n.not(e.target).get()))})},_removeClass:function(t,e,i){return this._toggleClass(t,e,i,!1)},_addClass:function(t,e,i){return this._toggleClass(t,e,i,!0)},_toggleClass:function(t,e,i,s){s="boolean"==typeof s?s:i;var n="string"==typeof t||null===t,o={extra:n?e:i,keys:n?t:e,element:n?this.element:t,add:s};return o.element.toggleClass(this._classes(o),s),this},_on:function(e,i,s){var n,o=this;"boolean"!=typeof e&&(s=i,i=e,e=!1),s?(i=n=t(i),this.bindings=this.bindings.add(i)):(s=i,i=this.element,n=this.widget()),t.each(s,function(s,a){function r(){return e||o.options.disabled!==!0&&!t(this).hasClass("ui-state-disabled")?("string"==typeof a?o[a]:a).apply(o,arguments):void 0}"string"!=typeof a&&(r.guid=a.guid=a.guid||r.guid||t.guid++);var h=s.match(/^([\w:-]*)\s*(.*)$/),l=h[1]+o.eventNamespace,c=h[2];c?n.on(l,c,r):i.on(l,r)})},_off:function(e,i){i=(i||"").split(" ").join(this.eventNamespace+" ")+this.eventNamespace,e.off(i).off(i),this.bindings=t(this.bindings.not(e).get()),this.focusable=t(this.focusable.not(e).get()),this.hoverable=t(this.hoverable.not(e).get())},_delay:function(t,e){function i(){return("string"==typeof t?s[t]:t).apply(s,arguments)}var s=this;return setTimeout(i,e||0)},_hoverable:function(e){this.hoverable=this.hoverable.add(e),this._on(e,{mouseenter:function(e){this._addClass(t(e.currentTarget),null,"ui-state-hover")},mouseleave:function(e){this._removeClass(t(e.currentTarget),null,"ui-state-hover")}})},_focusable:function(e){this.focusable=this.focusable.add(e),this._on(e,{focusin:function(e){this._addClass(t(e.currentTarget),null,"ui-state-focus")},focusout:function(e){this._removeClass(t(e.currentTarget),null,"ui-state-focus")}})},_trigger:function(e,i,s){var n,o,a=this.options[e];if(s=s||{},i=t.Event(i),i.type=(e===this.widgetEventPrefix?e:this.widgetEventPrefix+e).toLowerCase(),i.target=this.element[0],o=i.originalEvent)for(n in o)n in i||(i[n]=o[n]);return this.element.trigger(i,s),!(t.isFunction(a)&&a.apply(this.element[0],[i].concat(s))===!1||i.isDefaultPrevented())}},t.each({show:"fadeIn",hide:"fadeOut"},function(e,i){t.Widget.prototype["_"+e]=function(s,n,o){"string"==typeof n&&(n={effect:n});var a,r=n?n===!0||"number"==typeof n?i:n.effect||i:e;n=n||{},"number"==typeof n&&(n={duration:n}),a=!t.isEmptyObject(n),n.complete=o,n.delay&&s.delay(n.delay),a&&t.effects&&t.effects.effect[r]?s[e](n):r!==e&&s[r]?s[r](n.duration,n.easing,o):s.queue(function(i){t(this)[e](),o&&o.call(s[0]),i()})}}),t.widget,function(){function e(t,e,i){return[parseFloat(t[0])*(u.test(t[0])?e/100:1),parseFloat(t[1])*(u.test(t[1])?i/100:1)]}function i(e,i){return parseInt(t.css(e,i),10)||0}function s(e){var i=e[0];return 9===i.nodeType?{width:e.width(),height:e.height(),offset:{top:0,left:0}}:t.isWindow(i)?{width:e.width(),height:e.height(),offset:{top:e.scrollTop(),left:e.scrollLeft()}}:i.preventDefault?{width:0,height:0,offset:{top:i.pageY,left:i.pageX}}:{width:e.outerWidth(),height:e.outerHeight(),offset:e.offset()}}var n,o=Math.max,a=Math.abs,r=/left|center|right/,h=/top|center|bottom/,l=/[\+\-]\d+(\.[\d]+)?%?/,c=/^\w+/,u=/%$/,d=t.fn.position;t.position={scrollbarWidth:function(){if(void 0!==n)return n;var e,i,s=t("<div style='display:block;position:absolute;width:50px;height:50px;overflow:hidden;'><div style='height:100px;width:auto;'></div></div>"),o=s.children()[0];return t("body").append(s),e=o.offsetWidth,s.css("overflow","scroll"),i=o.offsetWidth,e===i&&(i=s[0].clientWidth),s.remove(),n=e-i},getScrollInfo:function(e){var i=e.isWindow||e.isDocument?"":e.element.css("overflow-x"),s=e.isWindow||e.isDocument?"":e.element.css("overflow-y"),n="scroll"===i||"auto"===i&&e.width<e.element[0].scrollWidth,o="scroll"===s||"auto"===s&&e.height<e.element[0].scrollHeight;return{width:o?t.position.scrollbarWidth():0,height:n?t.position.scrollbarWidth():0}},getWithinInfo:function(e){var i=t(e||window),s=t.isWindow(i[0]),n=!!i[0]&&9===i[0].nodeType,o=!s&&!n;return{element:i,isWindow:s,isDocument:n,offset:o?t(e).offset():{left:0,top:0},scrollLeft:i.scrollLeft(),scrollTop:i.scrollTop(),width:i.outerWidth(),height:i.outerHeight()}}},t.fn.position=function(n){if(!n||!n.of)return d.apply(this,arguments);n=t.extend({},n);var u,p,f,g,m,_,v=t(n.of),b=t.position.getWithinInfo(n.within),y=t.position.getScrollInfo(b),w=(n.collision||"flip").split(" "),k={};return _=s(v),v[0].preventDefault&&(n.at="left top"),p=_.width,f=_.height,g=_.offset,m=t.extend({},g),t.each(["my","at"],function(){var t,e,i=(n[this]||"").split(" ");1===i.length&&(i=r.test(i[0])?i.concat(["center"]):h.test(i[0])?["center"].concat(i):["center","center"]),i[0]=r.test(i[0])?i[0]:"center",i[1]=h.test(i[1])?i[1]:"center",t=l.exec(i[0]),e=l.exec(i[1]),k[this]=[t?t[0]:0,e?e[0]:0],n[this]=[c.exec(i[0])[0],c.exec(i[1])[0]]}),1===w.length&&(w[1]=w[0]),"right"===n.at[0]?m.left+=p:"center"===n.at[0]&&(m.left+=p/2),"bottom"===n.at[1]?m.top+=f:"center"===n.at[1]&&(m.top+=f/2),u=e(k.at,p,f),m.left+=u[0],m.top+=u[1],this.each(function(){var s,r,h=t(this),l=h.outerWidth(),c=h.outerHeight(),d=i(this,"marginLeft"),_=i(this,"marginTop"),x=l+d+i(this,"marginRight")+y.width,C=c+_+i(this,"marginBottom")+y.height,D=t.extend({},m),I=e(k.my,h.outerWidth(),h.outerHeight());"right"===n.my[0]?D.left-=l:"center"===n.my[0]&&(D.left-=l/2),"bottom"===n.my[1]?D.top-=c:"center"===n.my[1]&&(D.top-=c/2),D.left+=I[0],D.top+=I[1],s={marginLeft:d,marginTop:_},t.each(["left","top"],function(e,i){t.ui.position[w[e]]&&t.ui.position[w[e]][i](D,{targetWidth:p,targetHeight:f,elemWidth:l,elemHeight:c,collisionPosition:s,collisionWidth:x,collisionHeight:C,offset:[u[0]+I[0],u[1]+I[1]],my:n.my,at:n.at,within:b,elem:h})}),n.using&&(r=function(t){var e=g.left-D.left,i=e+p-l,s=g.top-D.top,r=s+f-c,u={target:{element:v,left:g.left,top:g.top,width:p,height:f},element:{element:h,left:D.left,top:D.top,width:l,height:c},horizontal:0>i?"left":e>0?"right":"center",vertical:0>r?"top":s>0?"bottom":"middle"};l>p&&p>a(e+i)&&(u.horizontal="center"),c>f&&f>a(s+r)&&(u.vertical="middle"),u.important=o(a(e),a(i))>o(a(s),a(r))?"horizontal":"vertical",n.using.call(this,t,u)}),h.offset(t.extend(D,{using:r}))})},t.ui.position={fit:{left:function(t,e){var i,s=e.within,n=s.isWindow?s.scrollLeft:s.offset.left,a=s.width,r=t.left-e.collisionPosition.marginLeft,h=n-r,l=r+e.collisionWidth-a-n;e.collisionWidth>a?h>0&&0>=l?(i=t.left+h+e.collisionWidth-a-n,t.left+=h-i):t.left=l>0&&0>=h?n:h>l?n+a-e.collisionWidth:n:h>0?t.left+=h:l>0?t.left-=l:t.left=o(t.left-r,t.left)},top:function(t,e){var i,s=e.within,n=s.isWindow?s.scrollTop:s.offset.top,a=e.within.height,r=t.top-e.collisionPosition.marginTop,h=n-r,l=r+e.collisionHeight-a-n;e.collisionHeight>a?h>0&&0>=l?(i=t.top+h+e.collisionHeight-a-n,t.top+=h-i):t.top=l>0&&0>=h?n:h>l?n+a-e.collisionHeight:n:h>0?t.top+=h:l>0?t.top-=l:t.top=o(t.top-r,t.top)}},flip:{left:function(t,e){var i,s,n=e.within,o=n.offset.left+n.scrollLeft,r=n.width,h=n.isWindow?n.scrollLeft:n.offset.left,l=t.left-e.collisionPosition.marginLeft,c=l-h,u=l+e.collisionWidth-r-h,d="left"===e.my[0]?-e.elemWidth:"right"===e.my[0]?e.elemWidth:0,p="left"===e.at[0]?e.targetWidth:"right"===e.at[0]?-e.targetWidth:0,f=-2*e.offset[0];0>c?(i=t.left+d+p+f+e.collisionWidth-r-o,(0>i||a(c)>i)&&(t.left+=d+p+f)):u>0&&(s=t.left-e.collisionPosition.marginLeft+d+p+f-h,(s>0||u>a(s))&&(t.left+=d+p+f))},top:function(t,e){var i,s,n=e.within,o=n.offset.top+n.scrollTop,r=n.height,h=n.isWindow?n.scrollTop:n.offset.top,l=t.top-e.collisionPosition.marginTop,c=l-h,u=l+e.collisionHeight-r-h,d="top"===e.my[1],p=d?-e.elemHeight:"bottom"===e.my[1]?e.elemHeight:0,f="top"===e.at[1]?e.targetHeight:"bottom"===e.at[1]?-e.targetHeight:0,g=-2*e.offset[1];0>c?(s=t.top+p+f+g+e.collisionHeight-r-o,(0>s||a(c)>s)&&(t.top+=p+f+g)):u>0&&(i=t.top-e.collisionPosition.marginTop+p+f+g-h,(i>0||u>a(i))&&(t.top+=p+f+g))}},flipfit:{left:function(){t.ui.position.flip.left.apply(this,arguments),t.ui.position.fit.left.apply(this,arguments)},top:function(){t.ui.position.flip.top.apply(this,arguments),t.ui.position.fit.top.apply(this,arguments)}}}}(),t.ui.position,t.extend(t.expr[":"],{data:t.expr.createPseudo?t.expr.createPseudo(function(e){return function(i){return!!t.data(i,e)}}):function(e,i,s){return!!t.data(e,s[3])}}),t.fn.extend({disableSelection:function(){var t="onselectstart"in document.createElement("div")?"selectstart":"mousedown";return function(){return this.on(t+".ui-disableSelection",function(t){t.preventDefault()})}}(),enableSelection:function(){return this.off(".ui-disableSelection")}});var c="ui-effects-",u="ui-effects-style",d="ui-effects-animated",p=t;t.effects={effect:{}},function(t,e){function i(t,e,i){var s=u[e.type]||{};return null==t?i||!e.def?null:e.def:(t=s.floor?~~t:parseFloat(t),isNaN(t)?e.def:s.mod?(t+s.mod)%s.mod:0>t?0:t>s.max?s.max:t)}function s(i){var s=l(),n=s._rgba=[];return i=i.toLowerCase(),f(h,function(t,o){var a,r=o.re.exec(i),h=r&&o.parse(r),l=o.space||"rgba";return h?(a=s[l](h),s[c[l].cache]=a[c[l].cache],n=s._rgba=a._rgba,!1):e}),n.length?("0,0,0,0"===n.join()&&t.extend(n,o.transparent),s):o[i]}function n(t,e,i){return i=(i+1)%1,1>6*i?t+6*(e-t)*i:1>2*i?e:2>3*i?t+6*(e-t)*(2/3-i):t}var o,a="backgroundColor borderBottomColor borderLeftColor borderRightColor borderTopColor color columnRuleColor outlineColor textDecorationColor textEmphasisColor",r=/^([\-+])=\s*(\d+\.?\d*)/,h=[{re:/rgba?\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*,\s*(\d{1,3})\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,parse:function(t){return[t[1],t[2],t[3],t[4]]}},{re:/rgba?\(\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,parse:function(t){return[2.55*t[1],2.55*t[2],2.55*t[3],t[4]]}},{re:/#([a-f0-9]{2})([a-f0-9]{2})([a-f0-9]{2})/,parse:function(t){return[parseInt(t[1],16),parseInt(t[2],16),parseInt(t[3],16)]}},{re:/#([a-f0-9])([a-f0-9])([a-f0-9])/,parse:function(t){return[parseInt(t[1]+t[1],16),parseInt(t[2]+t[2],16),parseInt(t[3]+t[3],16)]}},{re:/hsla?\(\s*(\d+(?:\.\d+)?)\s*,\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,space:"hsla",parse:function(t){return[t[1],t[2]/100,t[3]/100,t[4]]}}],l=t.Color=function(e,i,s,n){return new t.Color.fn.parse(e,i,s,n)},c={rgba:{props:{red:{idx:0,type:"byte"},green:{idx:1,type:"byte"},blue:{idx:2,type:"byte"}}},hsla:{props:{hue:{idx:0,type:"degrees"},saturation:{idx:1,type:"percent"},lightness:{idx:2,type:"percent"}}}},u={"byte":{floor:!0,max:255},percent:{max:1},degrees:{mod:360,floor:!0}},d=l.support={},p=t("<p>")[0],f=t.each;p.style.cssText="background-color:rgba(1,1,1,.5)",d.rgba=p.style.backgroundColor.indexOf("rgba")>-1,f(c,function(t,e){e.cache="_"+t,e.props.alpha={idx:3,type:"percent",def:1}}),l.fn=t.extend(l.prototype,{parse:function(n,a,r,h){if(n===e)return this._rgba=[null,null,null,null],this;(n.jquery||n.nodeType)&&(n=t(n).css(a),a=e);var u=this,d=t.type(n),p=this._rgba=[];return a!==e&&(n=[n,a,r,h],d="array"),"string"===d?this.parse(s(n)||o._default):"array"===d?(f(c.rgba.props,function(t,e){p[e.idx]=i(n[e.idx],e)}),this):"object"===d?(n instanceof l?f(c,function(t,e){n[e.cache]&&(u[e.cache]=n[e.cache].slice())}):f(c,function(e,s){var o=s.cache;f(s.props,function(t,e){if(!u[o]&&s.to){if("alpha"===t||null==n[t])return;u[o]=s.to(u._rgba)}u[o][e.idx]=i(n[t],e,!0)}),u[o]&&0>t.inArray(null,u[o].slice(0,3))&&(u[o][3]=1,s.from&&(u._rgba=s.from(u[o])))}),this):e},is:function(t){var i=l(t),s=!0,n=this;return f(c,function(t,o){var a,r=i[o.cache];return r&&(a=n[o.cache]||o.to&&o.to(n._rgba)||[],f(o.props,function(t,i){return null!=r[i.idx]?s=r[i.idx]===a[i.idx]:e})),s}),s},_space:function(){var t=[],e=this;return f(c,function(i,s){e[s.cache]&&t.push(i)}),t.pop()},transition:function(t,e){var s=l(t),n=s._space(),o=c[n],a=0===this.alpha()?l("transparent"):this,r=a[o.cache]||o.to(a._rgba),h=r.slice();return s=s[o.cache],f(o.props,function(t,n){var o=n.idx,a=r[o],l=s[o],c=u[n.type]||{};null!==l&&(null===a?h[o]=l:(c.mod&&(l-a>c.mod/2?a+=c.mod:a-l>c.mod/2&&(a-=c.mod)),h[o]=i((l-a)*e+a,n)))}),this[n](h)},blend:function(e){if(1===this._rgba[3])return this;var i=this._rgba.slice(),s=i.pop(),n=l(e)._rgba;return l(t.map(i,function(t,e){return(1-s)*n[e]+s*t}))},toRgbaString:function(){var e="rgba(",i=t.map(this._rgba,function(t,e){return null==t?e>2?1:0:t});return 1===i[3]&&(i.pop(),e="rgb("),e+i.join()+")"},toHslaString:function(){var e="hsla(",i=t.map(this.hsla(),function(t,e){return null==t&&(t=e>2?1:0),e&&3>e&&(t=Math.round(100*t)+"%"),t});return 1===i[3]&&(i.pop(),e="hsl("),e+i.join()+")"},toHexString:function(e){var i=this._rgba.slice(),s=i.pop();return e&&i.push(~~(255*s)),"#"+t.map(i,function(t){return t=(t||0).toString(16),1===t.length?"0"+t:t}).join("")},toString:function(){return 0===this._rgba[3]?"transparent":this.toRgbaString()}}),l.fn.parse.prototype=l.fn,c.hsla.to=function(t){if(null==t[0]||null==t[1]||null==t[2])return[null,null,null,t[3]];var e,i,s=t[0]/255,n=t[1]/255,o=t[2]/255,a=t[3],r=Math.max(s,n,o),h=Math.min(s,n,o),l=r-h,c=r+h,u=.5*c;return e=h===r?0:s===r?60*(n-o)/l+360:n===r?60*(o-s)/l+120:60*(s-n)/l+240,i=0===l?0:.5>=u?l/c:l/(2-c),[Math.round(e)%360,i,u,null==a?1:a]},c.hsla.from=function(t){if(null==t[0]||null==t[1]||null==t[2])return[null,null,null,t[3]];var e=t[0]/360,i=t[1],s=t[2],o=t[3],a=.5>=s?s*(1+i):s+i-s*i,r=2*s-a;return[Math.round(255*n(r,a,e+1/3)),Math.round(255*n(r,a,e)),Math.round(255*n(r,a,e-1/3)),o]},f(c,function(s,n){var o=n.props,a=n.cache,h=n.to,c=n.from;l.fn[s]=function(s){if(h&&!this[a]&&(this[a]=h(this._rgba)),s===e)return this[a].slice();var n,r=t.type(s),u="array"===r||"object"===r?s:arguments,d=this[a].slice();return f(o,function(t,e){var s=u["object"===r?t:e.idx];null==s&&(s=d[e.idx]),d[e.idx]=i(s,e)}),c?(n=l(c(d)),n[a]=d,n):l(d)},f(o,function(e,i){l.fn[e]||(l.fn[e]=function(n){var o,a=t.type(n),h="alpha"===e?this._hsla?"hsla":"rgba":s,l=this[h](),c=l[i.idx];return"undefined"===a?c:("function"===a&&(n=n.call(this,c),a=t.type(n)),null==n&&i.empty?this:("string"===a&&(o=r.exec(n),o&&(n=c+parseFloat(o[2])*("+"===o[1]?1:-1))),l[i.idx]=n,this[h](l)))})})}),l.hook=function(e){var i=e.split(" ");f(i,function(e,i){t.cssHooks[i]={set:function(e,n){var o,a,r="";if("transparent"!==n&&("string"!==t.type(n)||(o=s(n)))){if(n=l(o||n),!d.rgba&&1!==n._rgba[3]){for(a="backgroundColor"===i?e.parentNode:e;(""===r||"transparent"===r)&&a&&a.style;)try{r=t.css(a,"backgroundColor"),a=a.parentNode}catch(h){}n=n.blend(r&&"transparent"!==r?r:"_default")}n=n.toRgbaString()}try{e.style[i]=n}catch(h){}}},t.fx.step[i]=function(e){e.colorInit||(e.start=l(e.elem,i),e.end=l(e.end),e.colorInit=!0),t.cssHooks[i].set(e.elem,e.start.transition(e.end,e.pos))}})},l.hook(a),t.cssHooks.borderColor={expand:function(t){var e={};return f(["Top","Right","Bottom","Left"],function(i,s){e["border"+s+"Color"]=t}),e}},o=t.Color.names={aqua:"#00ffff",black:"#000000",blue:"#0000ff",fuchsia:"#ff00ff",gray:"#808080",green:"#008000",lime:"#00ff00",maroon:"#800000",navy:"#000080",olive:"#808000",purple:"#800080",red:"#ff0000",silver:"#c0c0c0",teal:"#008080",white:"#ffffff",yellow:"#ffff00",transparent:[null,null,null,0],_default:"#ffffff"}}(p),function(){function e(e){var i,s,n=e.ownerDocument.defaultView?e.ownerDocument.defaultView.getComputedStyle(e,null):e.currentStyle,o={};if(n&&n.length&&n[0]&&n[n[0]])for(s=n.length;s--;)i=n[s],"string"==typeof n[i]&&(o[t.camelCase(i)]=n[i]);else for(i in n)"string"==typeof n[i]&&(o[i]=n[i]);return o}function i(e,i){var s,o,a={};for(s in i)o=i[s],e[s]!==o&&(n[s]||(t.fx.step[s]||!isNaN(parseFloat(o)))&&(a[s]=o));return a}var s=["add","remove","toggle"],n={border:1,borderBottom:1,borderColor:1,borderLeft:1,borderRight:1,borderTop:1,borderWidth:1,margin:1,padding:1};t.each(["borderLeftStyle","borderRightStyle","borderBottomStyle","borderTopStyle"],function(e,i){t.fx.step[i]=function(t){("none"!==t.end&&!t.setAttr||1===t.pos&&!t.setAttr)&&(p.style(t.elem,i,t.end),t.setAttr=!0)}}),t.fn.addBack||(t.fn.addBack=function(t){return this.add(null==t?this.prevObject:this.prevObject.filter(t))}),t.effects.animateClass=function(n,o,a,r){var h=t.speed(o,a,r);return this.queue(function(){var o,a=t(this),r=a.attr("class")||"",l=h.children?a.find("*").addBack():a;l=l.map(function(){var i=t(this);return{el:i,start:e(this)}}),o=function(){t.each(s,function(t,e){n[e]&&a[e+"Class"](n[e])})},o(),l=l.map(function(){return this.end=e(this.el[0]),this.diff=i(this.start,this.end),this}),a.attr("class",r),l=l.map(function(){var e=this,i=t.Deferred(),s=t.extend({},h,{queue:!1,complete:function(){i.resolve(e)}});return this.el.animate(this.diff,s),i.promise()}),t.when.apply(t,l.get()).done(function(){o(),t.each(arguments,function(){var e=this.el;t.each(this.diff,function(t){e.css(t,"")})}),h.complete.call(a[0])})})},t.fn.extend({addClass:function(e){return function(i,s,n,o){return s?t.effects.animateClass.call(this,{add:i},s,n,o):e.apply(this,arguments)}}(t.fn.addClass),removeClass:function(e){return function(i,s,n,o){return arguments.length>1?t.effects.animateClass.call(this,{remove:i},s,n,o):e.apply(this,arguments)}}(t.fn.removeClass),toggleClass:function(e){return function(i,s,n,o,a){return"boolean"==typeof s||void 0===s?n?t.effects.animateClass.call(this,s?{add:i}:{remove:i},n,o,a):e.apply(this,arguments):t.effects.animateClass.call(this,{toggle:i},s,n,o)}}(t.fn.toggleClass),switchClass:function(e,i,s,n,o){return t.effects.animateClass.call(this,{add:i,remove:e},s,n,o)}})}(),function(){function e(e,i,s,n){return t.isPlainObject(e)&&(i=e,e=e.effect),e={effect:e},null==i&&(i={}),t.isFunction(i)&&(n=i,s=null,i={}),("number"==typeof i||t.fx.speeds[i])&&(n=s,s=i,i={}),t.isFunction(s)&&(n=s,s=null),i&&t.extend(e,i),s=s||i.duration,e.duration=t.fx.off?0:"number"==typeof s?s:s in t.fx.speeds?t.fx.speeds[s]:t.fx.speeds._default,e.complete=n||i.complete,e}function i(e){return!e||"number"==typeof e||t.fx.speeds[e]?!0:"string"!=typeof e||t.effects.effect[e]?t.isFunction(e)?!0:"object"!=typeof e||e.effect?!1:!0:!0}function s(t,e){var i=e.outerWidth(),s=e.outerHeight(),n=/^rect\((-?\d*\.?\d*px|-?\d+%|auto),?\s*(-?\d*\.?\d*px|-?\d+%|auto),?\s*(-?\d*\.?\d*px|-?\d+%|auto),?\s*(-?\d*\.?\d*px|-?\d+%|auto)\)$/,o=n.exec(t)||["",0,i,s,0];return{top:parseFloat(o[1])||0,right:"auto"===o[2]?i:parseFloat(o[2]),bottom:"auto"===o[3]?s:parseFloat(o[3]),left:parseFloat(o[4])||0}}t.expr&&t.expr.filters&&t.expr.filters.animated&&(t.expr.filters.animated=function(e){return function(i){return!!t(i).data(d)||e(i)}}(t.expr.filters.animated)),t.uiBackCompat!==!1&&t.extend(t.effects,{save:function(t,e){for(var i=0,s=e.length;s>i;i++)null!==e[i]&&t.data(c+e[i],t[0].style[e[i]])},restore:function(t,e){for(var i,s=0,n=e.length;n>s;s++)null!==e[s]&&(i=t.data(c+e[s]),t.css(e[s],i))},setMode:function(t,e){return"toggle"===e&&(e=t.is(":hidden")?"show":"hide"),e},createWrapper:function(e){if(e.parent().is(".ui-effects-wrapper"))return e.parent();var i={width:e.outerWidth(!0),height:e.outerHeight(!0),"float":e.css("float")},s=t("<div></div>").addClass("ui-effects-wrapper").css({fontSize:"100%",background:"transparent",border:"none",margin:0,padding:0}),n={width:e.width(),height:e.height()},o=document.activeElement;try{o.id}catch(a){o=document.body}return e.wrap(s),(e[0]===o||t.contains(e[0],o))&&t(o).trigger("focus"),s=e.parent(),"static"===e.css("position")?(s.css({position:"relative"}),e.css({position:"relative"})):(t.extend(i,{position:e.css("position"),zIndex:e.css("z-index")}),t.each(["top","left","bottom","right"],function(t,s){i[s]=e.css(s),isNaN(parseInt(i[s],10))&&(i[s]="auto")}),e.css({position:"relative",top:0,left:0,right:"auto",bottom:"auto"})),e.css(n),s.css(i).show()},removeWrapper:function(e){var i=document.activeElement;return e.parent().is(".ui-effects-wrapper")&&(e.parent().replaceWith(e),(e[0]===i||t.contains(e[0],i))&&t(i).trigger("focus")),e}}),t.extend(t.effects,{version:"1.12.1",define:function(e,i,s){return s||(s=i,i="effect"),t.effects.effect[e]=s,t.effects.effect[e].mode=i,s},scaledDimensions:function(t,e,i){if(0===e)return{height:0,width:0,outerHeight:0,outerWidth:0};var s="horizontal"!==i?(e||100)/100:1,n="vertical"!==i?(e||100)/100:1;return{height:t.height()*n,width:t.width()*s,outerHeight:t.outerHeight()*n,outerWidth:t.outerWidth()*s}},clipToBox:function(t){return{width:t.clip.right-t.clip.left,height:t.clip.bottom-t.clip.top,left:t.clip.left,top:t.clip.top}},unshift:function(t,e,i){var s=t.queue();e>1&&s.splice.apply(s,[1,0].concat(s.splice(e,i))),t.dequeue()},saveStyle:function(t){t.data(u,t[0].style.cssText)},restoreStyle:function(t){t[0].style.cssText=t.data(u)||"",t.removeData(u)},mode:function(t,e){var i=t.is(":hidden");return"toggle"===e&&(e=i?"show":"hide"),(i?"hide"===e:"show"===e)&&(e="none"),e},getBaseline:function(t,e){var i,s;switch(t[0]){case"top":i=0;break;case"middle":i=.5;break;case"bottom":i=1;break;default:i=t[0]/e.height}switch(t[1]){case"left":s=0;break;case"center":s=.5;break;case"right":s=1;break;default:s=t[1]/e.width}return{x:s,y:i}},createPlaceholder:function(e){var i,s=e.css("position"),n=e.position();return e.css({marginTop:e.css("marginTop"),marginBottom:e.css("marginBottom"),marginLeft:e.css("marginLeft"),marginRight:e.css("marginRight")}).outerWidth(e.outerWidth()).outerHeight(e.outerHeight()),/^(static|relative)/.test(s)&&(s="absolute",i=t("<"+e[0].nodeName+">").insertAfter(e).css({display:/^(inline|ruby)/.test(e.css("display"))?"inline-block":"block",visibility:"hidden",marginTop:e.css("marginTop"),marginBottom:e.css("marginBottom"),marginLeft:e.css("marginLeft"),marginRight:e.css("marginRight"),"float":e.css("float")}).outerWidth(e.outerWidth()).outerHeight(e.outerHeight()).addClass("ui-effects-placeholder"),e.data(c+"placeholder",i)),e.css({position:s,left:n.left,top:n.top}),i},removePlaceholder:function(t){var e=c+"placeholder",i=t.data(e);i&&(i.remove(),t.removeData(e))},cleanUp:function(e){t.effects.restoreStyle(e),t.effects.removePlaceholder(e)},setTransition:function(e,i,s,n){return n=n||{},t.each(i,function(t,i){var o=e.cssUnit(i);o[0]>0&&(n[i]=o[0]*s+o[1])}),n}}),t.fn.extend({effect:function(){function i(e){function i(){r.removeData(d),t.effects.cleanUp(r),"hide"===s.mode&&r.hide(),a()}function a(){t.isFunction(h)&&h.call(r[0]),t.isFunction(e)&&e()}var r=t(this);s.mode=c.shift(),t.uiBackCompat===!1||o?"none"===s.mode?(r[l](),a()):n.call(r[0],s,i):(r.is(":hidden")?"hide"===l:"show"===l)?(r[l](),a()):n.call(r[0],s,a)}var s=e.apply(this,arguments),n=t.effects.effect[s.effect],o=n.mode,a=s.queue,r=a||"fx",h=s.complete,l=s.mode,c=[],u=function(e){var i=t(this),s=t.effects.mode(i,l)||o;i.data(d,!0),c.push(s),o&&("show"===s||s===o&&"hide"===s)&&i.show(),o&&"none"===s||t.effects.saveStyle(i),t.isFunction(e)&&e()};return t.fx.off||!n?l?this[l](s.duration,h):this.each(function(){h&&h.call(this)}):a===!1?this.each(u).each(i):this.queue(r,u).queue(r,i)},show:function(t){return function(s){if(i(s))return t.apply(this,arguments);var n=e.apply(this,arguments);return n.mode="show",this.effect.call(this,n) -}}(t.fn.show),hide:function(t){return function(s){if(i(s))return t.apply(this,arguments);var n=e.apply(this,arguments);return n.mode="hide",this.effect.call(this,n)}}(t.fn.hide),toggle:function(t){return function(s){if(i(s)||"boolean"==typeof s)return t.apply(this,arguments);var n=e.apply(this,arguments);return n.mode="toggle",this.effect.call(this,n)}}(t.fn.toggle),cssUnit:function(e){var i=this.css(e),s=[];return t.each(["em","px","%","pt"],function(t,e){i.indexOf(e)>0&&(s=[parseFloat(i),e])}),s},cssClip:function(t){return t?this.css("clip","rect("+t.top+"px "+t.right+"px "+t.bottom+"px "+t.left+"px)"):s(this.css("clip"),this)},transfer:function(e,i){var s=t(this),n=t(e.to),o="fixed"===n.css("position"),a=t("body"),r=o?a.scrollTop():0,h=o?a.scrollLeft():0,l=n.offset(),c={top:l.top-r,left:l.left-h,height:n.innerHeight(),width:n.innerWidth()},u=s.offset(),d=t("<div class='ui-effects-transfer'></div>").appendTo("body").addClass(e.className).css({top:u.top-r,left:u.left-h,height:s.innerHeight(),width:s.innerWidth(),position:o?"fixed":"absolute"}).animate(c,e.duration,e.easing,function(){d.remove(),t.isFunction(i)&&i()})}}),t.fx.step.clip=function(e){e.clipInit||(e.start=t(e.elem).cssClip(),"string"==typeof e.end&&(e.end=s(e.end,e.elem)),e.clipInit=!0),t(e.elem).cssClip({top:e.pos*(e.end.top-e.start.top)+e.start.top,right:e.pos*(e.end.right-e.start.right)+e.start.right,bottom:e.pos*(e.end.bottom-e.start.bottom)+e.start.bottom,left:e.pos*(e.end.left-e.start.left)+e.start.left})}}(),function(){var e={};t.each(["Quad","Cubic","Quart","Quint","Expo"],function(t,i){e[i]=function(e){return Math.pow(e,t+2)}}),t.extend(e,{Sine:function(t){return 1-Math.cos(t*Math.PI/2)},Circ:function(t){return 1-Math.sqrt(1-t*t)},Elastic:function(t){return 0===t||1===t?t:-Math.pow(2,8*(t-1))*Math.sin((80*(t-1)-7.5)*Math.PI/15)},Back:function(t){return t*t*(3*t-2)},Bounce:function(t){for(var e,i=4;((e=Math.pow(2,--i))-1)/11>t;);return 1/Math.pow(4,3-i)-7.5625*Math.pow((3*e-2)/22-t,2)}}),t.each(e,function(e,i){t.easing["easeIn"+e]=i,t.easing["easeOut"+e]=function(t){return 1-i(1-t)},t.easing["easeInOut"+e]=function(t){return.5>t?i(2*t)/2:1-i(-2*t+2)/2}})}();var f=t.effects;t.effects.define("blind","hide",function(e,i){var s={up:["bottom","top"],vertical:["bottom","top"],down:["top","bottom"],left:["right","left"],horizontal:["right","left"],right:["left","right"]},n=t(this),o=e.direction||"up",a=n.cssClip(),r={clip:t.extend({},a)},h=t.effects.createPlaceholder(n);r.clip[s[o][0]]=r.clip[s[o][1]],"show"===e.mode&&(n.cssClip(r.clip),h&&h.css(t.effects.clipToBox(r)),r.clip=a),h&&h.animate(t.effects.clipToBox(r),e.duration,e.easing),n.animate(r,{queue:!1,duration:e.duration,easing:e.easing,complete:i})}),t.effects.define("bounce",function(e,i){var s,n,o,a=t(this),r=e.mode,h="hide"===r,l="show"===r,c=e.direction||"up",u=e.distance,d=e.times||5,p=2*d+(l||h?1:0),f=e.duration/p,g=e.easing,m="up"===c||"down"===c?"top":"left",_="up"===c||"left"===c,v=0,b=a.queue().length;for(t.effects.createPlaceholder(a),o=a.css(m),u||(u=a["top"===m?"outerHeight":"outerWidth"]()/3),l&&(n={opacity:1},n[m]=o,a.css("opacity",0).css(m,_?2*-u:2*u).animate(n,f,g)),h&&(u/=Math.pow(2,d-1)),n={},n[m]=o;d>v;v++)s={},s[m]=(_?"-=":"+=")+u,a.animate(s,f,g).animate(n,f,g),u=h?2*u:u/2;h&&(s={opacity:0},s[m]=(_?"-=":"+=")+u,a.animate(s,f,g)),a.queue(i),t.effects.unshift(a,b,p+1)}),t.effects.define("clip","hide",function(e,i){var s,n={},o=t(this),a=e.direction||"vertical",r="both"===a,h=r||"horizontal"===a,l=r||"vertical"===a;s=o.cssClip(),n.clip={top:l?(s.bottom-s.top)/2:s.top,right:h?(s.right-s.left)/2:s.right,bottom:l?(s.bottom-s.top)/2:s.bottom,left:h?(s.right-s.left)/2:s.left},t.effects.createPlaceholder(o),"show"===e.mode&&(o.cssClip(n.clip),n.clip=s),o.animate(n,{queue:!1,duration:e.duration,easing:e.easing,complete:i})}),t.effects.define("drop","hide",function(e,i){var s,n=t(this),o=e.mode,a="show"===o,r=e.direction||"left",h="up"===r||"down"===r?"top":"left",l="up"===r||"left"===r?"-=":"+=",c="+="===l?"-=":"+=",u={opacity:0};t.effects.createPlaceholder(n),s=e.distance||n["top"===h?"outerHeight":"outerWidth"](!0)/2,u[h]=l+s,a&&(n.css(u),u[h]=c+s,u.opacity=1),n.animate(u,{queue:!1,duration:e.duration,easing:e.easing,complete:i})}),t.effects.define("explode","hide",function(e,i){function s(){b.push(this),b.length===u*d&&n()}function n(){p.css({visibility:"visible"}),t(b).remove(),i()}var o,a,r,h,l,c,u=e.pieces?Math.round(Math.sqrt(e.pieces)):3,d=u,p=t(this),f=e.mode,g="show"===f,m=p.show().css("visibility","hidden").offset(),_=Math.ceil(p.outerWidth()/d),v=Math.ceil(p.outerHeight()/u),b=[];for(o=0;u>o;o++)for(h=m.top+o*v,c=o-(u-1)/2,a=0;d>a;a++)r=m.left+a*_,l=a-(d-1)/2,p.clone().appendTo("body").wrap("<div></div>").css({position:"absolute",visibility:"visible",left:-a*_,top:-o*v}).parent().addClass("ui-effects-explode").css({position:"absolute",overflow:"hidden",width:_,height:v,left:r+(g?l*_:0),top:h+(g?c*v:0),opacity:g?0:1}).animate({left:r+(g?0:l*_),top:h+(g?0:c*v),opacity:g?1:0},e.duration||500,e.easing,s)}),t.effects.define("fade","toggle",function(e,i){var s="show"===e.mode;t(this).css("opacity",s?0:1).animate({opacity:s?1:0},{queue:!1,duration:e.duration,easing:e.easing,complete:i})}),t.effects.define("fold","hide",function(e,i){var s=t(this),n=e.mode,o="show"===n,a="hide"===n,r=e.size||15,h=/([0-9]+)%/.exec(r),l=!!e.horizFirst,c=l?["right","bottom"]:["bottom","right"],u=e.duration/2,d=t.effects.createPlaceholder(s),p=s.cssClip(),f={clip:t.extend({},p)},g={clip:t.extend({},p)},m=[p[c[0]],p[c[1]]],_=s.queue().length;h&&(r=parseInt(h[1],10)/100*m[a?0:1]),f.clip[c[0]]=r,g.clip[c[0]]=r,g.clip[c[1]]=0,o&&(s.cssClip(g.clip),d&&d.css(t.effects.clipToBox(g)),g.clip=p),s.queue(function(i){d&&d.animate(t.effects.clipToBox(f),u,e.easing).animate(t.effects.clipToBox(g),u,e.easing),i()}).animate(f,u,e.easing).animate(g,u,e.easing).queue(i),t.effects.unshift(s,_,4)}),t.effects.define("highlight","show",function(e,i){var s=t(this),n={backgroundColor:s.css("backgroundColor")};"hide"===e.mode&&(n.opacity=0),t.effects.saveStyle(s),s.css({backgroundImage:"none",backgroundColor:e.color||"#ffff99"}).animate(n,{queue:!1,duration:e.duration,easing:e.easing,complete:i})}),t.effects.define("size",function(e,i){var s,n,o,a=t(this),r=["fontSize"],h=["borderTopWidth","borderBottomWidth","paddingTop","paddingBottom"],l=["borderLeftWidth","borderRightWidth","paddingLeft","paddingRight"],c=e.mode,u="effect"!==c,d=e.scale||"both",p=e.origin||["middle","center"],f=a.css("position"),g=a.position(),m=t.effects.scaledDimensions(a),_=e.from||m,v=e.to||t.effects.scaledDimensions(a,0);t.effects.createPlaceholder(a),"show"===c&&(o=_,_=v,v=o),n={from:{y:_.height/m.height,x:_.width/m.width},to:{y:v.height/m.height,x:v.width/m.width}},("box"===d||"both"===d)&&(n.from.y!==n.to.y&&(_=t.effects.setTransition(a,h,n.from.y,_),v=t.effects.setTransition(a,h,n.to.y,v)),n.from.x!==n.to.x&&(_=t.effects.setTransition(a,l,n.from.x,_),v=t.effects.setTransition(a,l,n.to.x,v))),("content"===d||"both"===d)&&n.from.y!==n.to.y&&(_=t.effects.setTransition(a,r,n.from.y,_),v=t.effects.setTransition(a,r,n.to.y,v)),p&&(s=t.effects.getBaseline(p,m),_.top=(m.outerHeight-_.outerHeight)*s.y+g.top,_.left=(m.outerWidth-_.outerWidth)*s.x+g.left,v.top=(m.outerHeight-v.outerHeight)*s.y+g.top,v.left=(m.outerWidth-v.outerWidth)*s.x+g.left),a.css(_),("content"===d||"both"===d)&&(h=h.concat(["marginTop","marginBottom"]).concat(r),l=l.concat(["marginLeft","marginRight"]),a.find("*[width]").each(function(){var i=t(this),s=t.effects.scaledDimensions(i),o={height:s.height*n.from.y,width:s.width*n.from.x,outerHeight:s.outerHeight*n.from.y,outerWidth:s.outerWidth*n.from.x},a={height:s.height*n.to.y,width:s.width*n.to.x,outerHeight:s.height*n.to.y,outerWidth:s.width*n.to.x};n.from.y!==n.to.y&&(o=t.effects.setTransition(i,h,n.from.y,o),a=t.effects.setTransition(i,h,n.to.y,a)),n.from.x!==n.to.x&&(o=t.effects.setTransition(i,l,n.from.x,o),a=t.effects.setTransition(i,l,n.to.x,a)),u&&t.effects.saveStyle(i),i.css(o),i.animate(a,e.duration,e.easing,function(){u&&t.effects.restoreStyle(i)})})),a.animate(v,{queue:!1,duration:e.duration,easing:e.easing,complete:function(){var e=a.offset();0===v.opacity&&a.css("opacity",_.opacity),u||(a.css("position","static"===f?"relative":f).offset(e),t.effects.saveStyle(a)),i()}})}),t.effects.define("scale",function(e,i){var s=t(this),n=e.mode,o=parseInt(e.percent,10)||(0===parseInt(e.percent,10)?0:"effect"!==n?0:100),a=t.extend(!0,{from:t.effects.scaledDimensions(s),to:t.effects.scaledDimensions(s,o,e.direction||"both"),origin:e.origin||["middle","center"]},e);e.fade&&(a.from.opacity=1,a.to.opacity=0),t.effects.effect.size.call(this,a,i)}),t.effects.define("puff","hide",function(e,i){var s=t.extend(!0,{},e,{fade:!0,percent:parseInt(e.percent,10)||150});t.effects.effect.scale.call(this,s,i)}),t.effects.define("pulsate","show",function(e,i){var s=t(this),n=e.mode,o="show"===n,a="hide"===n,r=o||a,h=2*(e.times||5)+(r?1:0),l=e.duration/h,c=0,u=1,d=s.queue().length;for((o||!s.is(":visible"))&&(s.css("opacity",0).show(),c=1);h>u;u++)s.animate({opacity:c},l,e.easing),c=1-c;s.animate({opacity:c},l,e.easing),s.queue(i),t.effects.unshift(s,d,h+1)}),t.effects.define("shake",function(e,i){var s=1,n=t(this),o=e.direction||"left",a=e.distance||20,r=e.times||3,h=2*r+1,l=Math.round(e.duration/h),c="up"===o||"down"===o?"top":"left",u="up"===o||"left"===o,d={},p={},f={},g=n.queue().length;for(t.effects.createPlaceholder(n),d[c]=(u?"-=":"+=")+a,p[c]=(u?"+=":"-=")+2*a,f[c]=(u?"-=":"+=")+2*a,n.animate(d,l,e.easing);r>s;s++)n.animate(p,l,e.easing).animate(f,l,e.easing);n.animate(p,l,e.easing).animate(d,l/2,e.easing).queue(i),t.effects.unshift(n,g,h+1)}),t.effects.define("slide","show",function(e,i){var s,n,o=t(this),a={up:["bottom","top"],down:["top","bottom"],left:["right","left"],right:["left","right"]},r=e.mode,h=e.direction||"left",l="up"===h||"down"===h?"top":"left",c="up"===h||"left"===h,u=e.distance||o["top"===l?"outerHeight":"outerWidth"](!0),d={};t.effects.createPlaceholder(o),s=o.cssClip(),n=o.position()[l],d[l]=(c?-1:1)*u+n,d.clip=o.cssClip(),d.clip[a[h][1]]=d.clip[a[h][0]],"show"===r&&(o.cssClip(d.clip),o.css(l,d[l]),d.clip=s,d[l]=n),o.animate(d,{queue:!1,duration:e.duration,easing:e.easing,complete:i})});var f;t.uiBackCompat!==!1&&(f=t.effects.define("transfer",function(e,i){t(this).transfer(e,i)})),t.ui.focusable=function(i,s){var n,o,a,r,h,l=i.nodeName.toLowerCase();return"area"===l?(n=i.parentNode,o=n.name,i.href&&o&&"map"===n.nodeName.toLowerCase()?(a=t("img[usemap='#"+o+"']"),a.length>0&&a.is(":visible")):!1):(/^(input|select|textarea|button|object)$/.test(l)?(r=!i.disabled,r&&(h=t(i).closest("fieldset")[0],h&&(r=!h.disabled))):r="a"===l?i.href||s:s,r&&t(i).is(":visible")&&e(t(i)))},t.extend(t.expr[":"],{focusable:function(e){return t.ui.focusable(e,null!=t.attr(e,"tabindex"))}}),t.ui.focusable,t.fn.form=function(){return"string"==typeof this[0].form?this.closest("form"):t(this[0].form)},t.ui.formResetMixin={_formResetHandler:function(){var e=t(this);setTimeout(function(){var i=e.data("ui-form-reset-instances");t.each(i,function(){this.refresh()})})},_bindFormResetHandler:function(){if(this.form=this.element.form(),this.form.length){var t=this.form.data("ui-form-reset-instances")||[];t.length||this.form.on("reset.ui-form-reset",this._formResetHandler),t.push(this),this.form.data("ui-form-reset-instances",t)}},_unbindFormResetHandler:function(){if(this.form.length){var e=this.form.data("ui-form-reset-instances");e.splice(t.inArray(this,e),1),e.length?this.form.data("ui-form-reset-instances",e):this.form.removeData("ui-form-reset-instances").off("reset.ui-form-reset")}}},"1.7"===t.fn.jquery.substring(0,3)&&(t.each(["Width","Height"],function(e,i){function s(e,i,s,o){return t.each(n,function(){i-=parseFloat(t.css(e,"padding"+this))||0,s&&(i-=parseFloat(t.css(e,"border"+this+"Width"))||0),o&&(i-=parseFloat(t.css(e,"margin"+this))||0)}),i}var n="Width"===i?["Left","Right"]:["Top","Bottom"],o=i.toLowerCase(),a={innerWidth:t.fn.innerWidth,innerHeight:t.fn.innerHeight,outerWidth:t.fn.outerWidth,outerHeight:t.fn.outerHeight};t.fn["inner"+i]=function(e){return void 0===e?a["inner"+i].call(this):this.each(function(){t(this).css(o,s(this,e)+"px")})},t.fn["outer"+i]=function(e,n){return"number"!=typeof e?a["outer"+i].call(this,e):this.each(function(){t(this).css(o,s(this,e,!0,n)+"px")})}}),t.fn.addBack=function(t){return this.add(null==t?this.prevObject:this.prevObject.filter(t))}),t.ui.keyCode={BACKSPACE:8,COMMA:188,DELETE:46,DOWN:40,END:35,ENTER:13,ESCAPE:27,HOME:36,LEFT:37,PAGE_DOWN:34,PAGE_UP:33,PERIOD:190,RIGHT:39,SPACE:32,TAB:9,UP:38},t.ui.escapeSelector=function(){var t=/([!"#$%&'()*+,.\/:;<=>?@[\]^`{|}~])/g;return function(e){return e.replace(t,"\\$1")}}(),t.fn.labels=function(){var e,i,s,n,o;return this[0].labels&&this[0].labels.length?this.pushStack(this[0].labels):(n=this.eq(0).parents("label"),s=this.attr("id"),s&&(e=this.eq(0).parents().last(),o=e.add(e.length?e.siblings():this.siblings()),i="label[for='"+t.ui.escapeSelector(s)+"']",n=n.add(o.find(i).addBack(i))),this.pushStack(n))},t.fn.scrollParent=function(e){var i=this.css("position"),s="absolute"===i,n=e?/(auto|scroll|hidden)/:/(auto|scroll)/,o=this.parents().filter(function(){var e=t(this);return s&&"static"===e.css("position")?!1:n.test(e.css("overflow")+e.css("overflow-y")+e.css("overflow-x"))}).eq(0);return"fixed"!==i&&o.length?o:t(this[0].ownerDocument||document)},t.extend(t.expr[":"],{tabbable:function(e){var i=t.attr(e,"tabindex"),s=null!=i;return(!s||i>=0)&&t.ui.focusable(e,s)}}),t.fn.extend({uniqueId:function(){var t=0;return function(){return this.each(function(){this.id||(this.id="ui-id-"+ ++t)})}}(),removeUniqueId:function(){return this.each(function(){/^ui-id-\d+$/.test(this.id)&&t(this).removeAttr("id")})}}),t.widget("ui.accordion",{version:"1.12.1",options:{active:0,animate:{},classes:{"ui-accordion-header":"ui-corner-top","ui-accordion-header-collapsed":"ui-corner-all","ui-accordion-content":"ui-corner-bottom"},collapsible:!1,event:"click",header:"> li > :first-child, > :not(li):even",heightStyle:"auto",icons:{activeHeader:"ui-icon-triangle-1-s",header:"ui-icon-triangle-1-e"},activate:null,beforeActivate:null},hideProps:{borderTopWidth:"hide",borderBottomWidth:"hide",paddingTop:"hide",paddingBottom:"hide",height:"hide"},showProps:{borderTopWidth:"show",borderBottomWidth:"show",paddingTop:"show",paddingBottom:"show",height:"show"},_create:function(){var e=this.options;this.prevShow=this.prevHide=t(),this._addClass("ui-accordion","ui-widget ui-helper-reset"),this.element.attr("role","tablist"),e.collapsible||e.active!==!1&&null!=e.active||(e.active=0),this._processPanels(),0>e.active&&(e.active+=this.headers.length),this._refresh()},_getCreateEventData:function(){return{header:this.active,panel:this.active.length?this.active.next():t()}},_createIcons:function(){var e,i,s=this.options.icons;s&&(e=t("<span>"),this._addClass(e,"ui-accordion-header-icon","ui-icon "+s.header),e.prependTo(this.headers),i=this.active.children(".ui-accordion-header-icon"),this._removeClass(i,s.header)._addClass(i,null,s.activeHeader)._addClass(this.headers,"ui-accordion-icons"))},_destroyIcons:function(){this._removeClass(this.headers,"ui-accordion-icons"),this.headers.children(".ui-accordion-header-icon").remove()},_destroy:function(){var t;this.element.removeAttr("role"),this.headers.removeAttr("role aria-expanded aria-selected aria-controls tabIndex").removeUniqueId(),this._destroyIcons(),t=this.headers.next().css("display","").removeAttr("role aria-hidden aria-labelledby").removeUniqueId(),"content"!==this.options.heightStyle&&t.css("height","")},_setOption:function(t,e){return"active"===t?(this._activate(e),void 0):("event"===t&&(this.options.event&&this._off(this.headers,this.options.event),this._setupEvents(e)),this._super(t,e),"collapsible"!==t||e||this.options.active!==!1||this._activate(0),"icons"===t&&(this._destroyIcons(),e&&this._createIcons()),void 0)},_setOptionDisabled:function(t){this._super(t),this.element.attr("aria-disabled",t),this._toggleClass(null,"ui-state-disabled",!!t),this._toggleClass(this.headers.add(this.headers.next()),null,"ui-state-disabled",!!t)},_keydown:function(e){if(!e.altKey&&!e.ctrlKey){var i=t.ui.keyCode,s=this.headers.length,n=this.headers.index(e.target),o=!1;switch(e.keyCode){case i.RIGHT:case i.DOWN:o=this.headers[(n+1)%s];break;case i.LEFT:case i.UP:o=this.headers[(n-1+s)%s];break;case i.SPACE:case i.ENTER:this._eventHandler(e);break;case i.HOME:o=this.headers[0];break;case i.END:o=this.headers[s-1]}o&&(t(e.target).attr("tabIndex",-1),t(o).attr("tabIndex",0),t(o).trigger("focus"),e.preventDefault())}},_panelKeyDown:function(e){e.keyCode===t.ui.keyCode.UP&&e.ctrlKey&&t(e.currentTarget).prev().trigger("focus")},refresh:function(){var e=this.options;this._processPanels(),e.active===!1&&e.collapsible===!0||!this.headers.length?(e.active=!1,this.active=t()):e.active===!1?this._activate(0):this.active.length&&!t.contains(this.element[0],this.active[0])?this.headers.length===this.headers.find(".ui-state-disabled").length?(e.active=!1,this.active=t()):this._activate(Math.max(0,e.active-1)):e.active=this.headers.index(this.active),this._destroyIcons(),this._refresh()},_processPanels:function(){var t=this.headers,e=this.panels;this.headers=this.element.find(this.options.header),this._addClass(this.headers,"ui-accordion-header ui-accordion-header-collapsed","ui-state-default"),this.panels=this.headers.next().filter(":not(.ui-accordion-content-active)").hide(),this._addClass(this.panels,"ui-accordion-content","ui-helper-reset ui-widget-content"),e&&(this._off(t.not(this.headers)),this._off(e.not(this.panels)))},_refresh:function(){var e,i=this.options,s=i.heightStyle,n=this.element.parent();this.active=this._findActive(i.active),this._addClass(this.active,"ui-accordion-header-active","ui-state-active")._removeClass(this.active,"ui-accordion-header-collapsed"),this._addClass(this.active.next(),"ui-accordion-content-active"),this.active.next().show(),this.headers.attr("role","tab").each(function(){var e=t(this),i=e.uniqueId().attr("id"),s=e.next(),n=s.uniqueId().attr("id");e.attr("aria-controls",n),s.attr("aria-labelledby",i)}).next().attr("role","tabpanel"),this.headers.not(this.active).attr({"aria-selected":"false","aria-expanded":"false",tabIndex:-1}).next().attr({"aria-hidden":"true"}).hide(),this.active.length?this.active.attr({"aria-selected":"true","aria-expanded":"true",tabIndex:0}).next().attr({"aria-hidden":"false"}):this.headers.eq(0).attr("tabIndex",0),this._createIcons(),this._setupEvents(i.event),"fill"===s?(e=n.height(),this.element.siblings(":visible").each(function(){var i=t(this),s=i.css("position");"absolute"!==s&&"fixed"!==s&&(e-=i.outerHeight(!0))}),this.headers.each(function(){e-=t(this).outerHeight(!0)}),this.headers.next().each(function(){t(this).height(Math.max(0,e-t(this).innerHeight()+t(this).height()))}).css("overflow","auto")):"auto"===s&&(e=0,this.headers.next().each(function(){var i=t(this).is(":visible");i||t(this).show(),e=Math.max(e,t(this).css("height","").height()),i||t(this).hide()}).height(e))},_activate:function(e){var i=this._findActive(e)[0];i!==this.active[0]&&(i=i||this.active[0],this._eventHandler({target:i,currentTarget:i,preventDefault:t.noop}))},_findActive:function(e){return"number"==typeof e?this.headers.eq(e):t()},_setupEvents:function(e){var i={keydown:"_keydown"};e&&t.each(e.split(" "),function(t,e){i[e]="_eventHandler"}),this._off(this.headers.add(this.headers.next())),this._on(this.headers,i),this._on(this.headers.next(),{keydown:"_panelKeyDown"}),this._hoverable(this.headers),this._focusable(this.headers)},_eventHandler:function(e){var i,s,n=this.options,o=this.active,a=t(e.currentTarget),r=a[0]===o[0],h=r&&n.collapsible,l=h?t():a.next(),c=o.next(),u={oldHeader:o,oldPanel:c,newHeader:h?t():a,newPanel:l};e.preventDefault(),r&&!n.collapsible||this._trigger("beforeActivate",e,u)===!1||(n.active=h?!1:this.headers.index(a),this.active=r?t():a,this._toggle(u),this._removeClass(o,"ui-accordion-header-active","ui-state-active"),n.icons&&(i=o.children(".ui-accordion-header-icon"),this._removeClass(i,null,n.icons.activeHeader)._addClass(i,null,n.icons.header)),r||(this._removeClass(a,"ui-accordion-header-collapsed")._addClass(a,"ui-accordion-header-active","ui-state-active"),n.icons&&(s=a.children(".ui-accordion-header-icon"),this._removeClass(s,null,n.icons.header)._addClass(s,null,n.icons.activeHeader)),this._addClass(a.next(),"ui-accordion-content-active")))},_toggle:function(e){var i=e.newPanel,s=this.prevShow.length?this.prevShow:e.oldPanel;this.prevShow.add(this.prevHide).stop(!0,!0),this.prevShow=i,this.prevHide=s,this.options.animate?this._animate(i,s,e):(s.hide(),i.show(),this._toggleComplete(e)),s.attr({"aria-hidden":"true"}),s.prev().attr({"aria-selected":"false","aria-expanded":"false"}),i.length&&s.length?s.prev().attr({tabIndex:-1,"aria-expanded":"false"}):i.length&&this.headers.filter(function(){return 0===parseInt(t(this).attr("tabIndex"),10)}).attr("tabIndex",-1),i.attr("aria-hidden","false").prev().attr({"aria-selected":"true","aria-expanded":"true",tabIndex:0})},_animate:function(t,e,i){var s,n,o,a=this,r=0,h=t.css("box-sizing"),l=t.length&&(!e.length||t.index()<e.index()),c=this.options.animate||{},u=l&&c.down||c,d=function(){a._toggleComplete(i)};return"number"==typeof u&&(o=u),"string"==typeof u&&(n=u),n=n||u.easing||c.easing,o=o||u.duration||c.duration,e.length?t.length?(s=t.show().outerHeight(),e.animate(this.hideProps,{duration:o,easing:n,step:function(t,e){e.now=Math.round(t)}}),t.hide().animate(this.showProps,{duration:o,easing:n,complete:d,step:function(t,i){i.now=Math.round(t),"height"!==i.prop?"content-box"===h&&(r+=i.now):"content"!==a.options.heightStyle&&(i.now=Math.round(s-e.outerHeight()-r),r=0)}}),void 0):e.animate(this.hideProps,o,n,d):t.animate(this.showProps,o,n,d)},_toggleComplete:function(t){var e=t.oldPanel,i=e.prev();this._removeClass(e,"ui-accordion-content-active"),this._removeClass(i,"ui-accordion-header-active")._addClass(i,"ui-accordion-header-collapsed"),e.length&&(e.parent()[0].className=e.parent()[0].className),this._trigger("activate",null,t)}}),t.ui.safeActiveElement=function(t){var e;try{e=t.activeElement}catch(i){e=t.body}return e||(e=t.body),e.nodeName||(e=t.body),e},t.widget("ui.menu",{version:"1.12.1",defaultElement:"<ul>",delay:300,options:{icons:{submenu:"ui-icon-caret-1-e"},items:"> *",menus:"ul",position:{my:"left top",at:"right top"},role:"menu",blur:null,focus:null,select:null},_create:function(){this.activeMenu=this.element,this.mouseHandled=!1,this.element.uniqueId().attr({role:this.options.role,tabIndex:0}),this._addClass("ui-menu","ui-widget ui-widget-content"),this._on({"mousedown .ui-menu-item":function(t){t.preventDefault()},"click .ui-menu-item":function(e){var i=t(e.target),s=t(t.ui.safeActiveElement(this.document[0]));!this.mouseHandled&&i.not(".ui-state-disabled").length&&(this.select(e),e.isPropagationStopped()||(this.mouseHandled=!0),i.has(".ui-menu").length?this.expand(e):!this.element.is(":focus")&&s.closest(".ui-menu").length&&(this.element.trigger("focus",[!0]),this.active&&1===this.active.parents(".ui-menu").length&&clearTimeout(this.timer)))},"mouseenter .ui-menu-item":function(e){if(!this.previousFilter){var i=t(e.target).closest(".ui-menu-item"),s=t(e.currentTarget);i[0]===s[0]&&(this._removeClass(s.siblings().children(".ui-state-active"),null,"ui-state-active"),this.focus(e,s))}},mouseleave:"collapseAll","mouseleave .ui-menu":"collapseAll",focus:function(t,e){var i=this.active||this.element.find(this.options.items).eq(0);e||this.focus(t,i)},blur:function(e){this._delay(function(){var i=!t.contains(this.element[0],t.ui.safeActiveElement(this.document[0]));i&&this.collapseAll(e)})},keydown:"_keydown"}),this.refresh(),this._on(this.document,{click:function(t){this._closeOnDocumentClick(t)&&this.collapseAll(t),this.mouseHandled=!1}})},_destroy:function(){var e=this.element.find(".ui-menu-item").removeAttr("role aria-disabled"),i=e.children(".ui-menu-item-wrapper").removeUniqueId().removeAttr("tabIndex role aria-haspopup");this.element.removeAttr("aria-activedescendant").find(".ui-menu").addBack().removeAttr("role aria-labelledby aria-expanded aria-hidden aria-disabled tabIndex").removeUniqueId().show(),i.children().each(function(){var e=t(this);e.data("ui-menu-submenu-caret")&&e.remove()})},_keydown:function(e){var i,s,n,o,a=!0;switch(e.keyCode){case t.ui.keyCode.PAGE_UP:this.previousPage(e);break;case t.ui.keyCode.PAGE_DOWN:this.nextPage(e);break;case t.ui.keyCode.HOME:this._move("first","first",e);break;case t.ui.keyCode.END:this._move("last","last",e);break;case t.ui.keyCode.UP:this.previous(e);break;case t.ui.keyCode.DOWN:this.next(e);break;case t.ui.keyCode.LEFT:this.collapse(e);break;case t.ui.keyCode.RIGHT:this.active&&!this.active.is(".ui-state-disabled")&&this.expand(e);break;case t.ui.keyCode.ENTER:case t.ui.keyCode.SPACE:this._activate(e);break;case t.ui.keyCode.ESCAPE:this.collapse(e);break;default:a=!1,s=this.previousFilter||"",o=!1,n=e.keyCode>=96&&105>=e.keyCode?""+(e.keyCode-96):String.fromCharCode(e.keyCode),clearTimeout(this.filterTimer),n===s?o=!0:n=s+n,i=this._filterMenuItems(n),i=o&&-1!==i.index(this.active.next())?this.active.nextAll(".ui-menu-item"):i,i.length||(n=String.fromCharCode(e.keyCode),i=this._filterMenuItems(n)),i.length?(this.focus(e,i),this.previousFilter=n,this.filterTimer=this._delay(function(){delete this.previousFilter},1e3)):delete this.previousFilter}a&&e.preventDefault()},_activate:function(t){this.active&&!this.active.is(".ui-state-disabled")&&(this.active.children("[aria-haspopup='true']").length?this.expand(t):this.select(t))},refresh:function(){var e,i,s,n,o,a=this,r=this.options.icons.submenu,h=this.element.find(this.options.menus);this._toggleClass("ui-menu-icons",null,!!this.element.find(".ui-icon").length),s=h.filter(":not(.ui-menu)").hide().attr({role:this.options.role,"aria-hidden":"true","aria-expanded":"false"}).each(function(){var e=t(this),i=e.prev(),s=t("<span>").data("ui-menu-submenu-caret",!0);a._addClass(s,"ui-menu-icon","ui-icon "+r),i.attr("aria-haspopup","true").prepend(s),e.attr("aria-labelledby",i.attr("id"))}),this._addClass(s,"ui-menu","ui-widget ui-widget-content ui-front"),e=h.add(this.element),i=e.find(this.options.items),i.not(".ui-menu-item").each(function(){var e=t(this);a._isDivider(e)&&a._addClass(e,"ui-menu-divider","ui-widget-content")}),n=i.not(".ui-menu-item, .ui-menu-divider"),o=n.children().not(".ui-menu").uniqueId().attr({tabIndex:-1,role:this._itemRole()}),this._addClass(n,"ui-menu-item")._addClass(o,"ui-menu-item-wrapper"),i.filter(".ui-state-disabled").attr("aria-disabled","true"),this.active&&!t.contains(this.element[0],this.active[0])&&this.blur()},_itemRole:function(){return{menu:"menuitem",listbox:"option"}[this.options.role]},_setOption:function(t,e){if("icons"===t){var i=this.element.find(".ui-menu-icon");this._removeClass(i,null,this.options.icons.submenu)._addClass(i,null,e.submenu)}this._super(t,e)},_setOptionDisabled:function(t){this._super(t),this.element.attr("aria-disabled",t+""),this._toggleClass(null,"ui-state-disabled",!!t)},focus:function(t,e){var i,s,n;this.blur(t,t&&"focus"===t.type),this._scrollIntoView(e),this.active=e.first(),s=this.active.children(".ui-menu-item-wrapper"),this._addClass(s,null,"ui-state-active"),this.options.role&&this.element.attr("aria-activedescendant",s.attr("id")),n=this.active.parent().closest(".ui-menu-item").children(".ui-menu-item-wrapper"),this._addClass(n,null,"ui-state-active"),t&&"keydown"===t.type?this._close():this.timer=this._delay(function(){this._close()},this.delay),i=e.children(".ui-menu"),i.length&&t&&/^mouse/.test(t.type)&&this._startOpening(i),this.activeMenu=e.parent(),this._trigger("focus",t,{item:e})},_scrollIntoView:function(e){var i,s,n,o,a,r;this._hasScroll()&&(i=parseFloat(t.css(this.activeMenu[0],"borderTopWidth"))||0,s=parseFloat(t.css(this.activeMenu[0],"paddingTop"))||0,n=e.offset().top-this.activeMenu.offset().top-i-s,o=this.activeMenu.scrollTop(),a=this.activeMenu.height(),r=e.outerHeight(),0>n?this.activeMenu.scrollTop(o+n):n+r>a&&this.activeMenu.scrollTop(o+n-a+r))},blur:function(t,e){e||clearTimeout(this.timer),this.active&&(this._removeClass(this.active.children(".ui-menu-item-wrapper"),null,"ui-state-active"),this._trigger("blur",t,{item:this.active}),this.active=null)},_startOpening:function(t){clearTimeout(this.timer),"true"===t.attr("aria-hidden")&&(this.timer=this._delay(function(){this._close(),this._open(t)},this.delay))},_open:function(e){var i=t.extend({of:this.active},this.options.position);clearTimeout(this.timer),this.element.find(".ui-menu").not(e.parents(".ui-menu")).hide().attr("aria-hidden","true"),e.show().removeAttr("aria-hidden").attr("aria-expanded","true").position(i)},collapseAll:function(e,i){clearTimeout(this.timer),this.timer=this._delay(function(){var s=i?this.element:t(e&&e.target).closest(this.element.find(".ui-menu"));s.length||(s=this.element),this._close(s),this.blur(e),this._removeClass(s.find(".ui-state-active"),null,"ui-state-active"),this.activeMenu=s},this.delay)},_close:function(t){t||(t=this.active?this.active.parent():this.element),t.find(".ui-menu").hide().attr("aria-hidden","true").attr("aria-expanded","false")},_closeOnDocumentClick:function(e){return!t(e.target).closest(".ui-menu").length},_isDivider:function(t){return!/[^\-\u2014\u2013\s]/.test(t.text())},collapse:function(t){var e=this.active&&this.active.parent().closest(".ui-menu-item",this.element);e&&e.length&&(this._close(),this.focus(t,e))},expand:function(t){var e=this.active&&this.active.children(".ui-menu ").find(this.options.items).first();e&&e.length&&(this._open(e.parent()),this._delay(function(){this.focus(t,e)}))},next:function(t){this._move("next","first",t)},previous:function(t){this._move("prev","last",t)},isFirstItem:function(){return this.active&&!this.active.prevAll(".ui-menu-item").length},isLastItem:function(){return this.active&&!this.active.nextAll(".ui-menu-item").length},_move:function(t,e,i){var s;this.active&&(s="first"===t||"last"===t?this.active["first"===t?"prevAll":"nextAll"](".ui-menu-item").eq(-1):this.active[t+"All"](".ui-menu-item").eq(0)),s&&s.length&&this.active||(s=this.activeMenu.find(this.options.items)[e]()),this.focus(i,s)},nextPage:function(e){var i,s,n;return this.active?(this.isLastItem()||(this._hasScroll()?(s=this.active.offset().top,n=this.element.height(),this.active.nextAll(".ui-menu-item").each(function(){return i=t(this),0>i.offset().top-s-n}),this.focus(e,i)):this.focus(e,this.activeMenu.find(this.options.items)[this.active?"last":"first"]())),void 0):(this.next(e),void 0)},previousPage:function(e){var i,s,n;return this.active?(this.isFirstItem()||(this._hasScroll()?(s=this.active.offset().top,n=this.element.height(),this.active.prevAll(".ui-menu-item").each(function(){return i=t(this),i.offset().top-s+n>0}),this.focus(e,i)):this.focus(e,this.activeMenu.find(this.options.items).first())),void 0):(this.next(e),void 0)},_hasScroll:function(){return this.element.outerHeight()<this.element.prop("scrollHeight")},select:function(e){this.active=this.active||t(e.target).closest(".ui-menu-item");var i={item:this.active};this.active.has(".ui-menu").length||this.collapseAll(e,!0),this._trigger("select",e,i)},_filterMenuItems:function(e){var i=e.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g,"\\$&"),s=RegExp("^"+i,"i");return this.activeMenu.find(this.options.items).filter(".ui-menu-item").filter(function(){return s.test(t.trim(t(this).children(".ui-menu-item-wrapper").text()))})}}),t.widget("ui.autocomplete",{version:"1.12.1",defaultElement:"<input>",options:{appendTo:null,autoFocus:!1,delay:300,minLength:1,position:{my:"left top",at:"left bottom",collision:"none"},source:null,change:null,close:null,focus:null,open:null,response:null,search:null,select:null},requestIndex:0,pending:0,_create:function(){var e,i,s,n=this.element[0].nodeName.toLowerCase(),o="textarea"===n,a="input"===n; -this.isMultiLine=o||!a&&this._isContentEditable(this.element),this.valueMethod=this.element[o||a?"val":"text"],this.isNewMenu=!0,this._addClass("ui-autocomplete-input"),this.element.attr("autocomplete","off"),this._on(this.element,{keydown:function(n){if(this.element.prop("readOnly"))return e=!0,s=!0,i=!0,void 0;e=!1,s=!1,i=!1;var o=t.ui.keyCode;switch(n.keyCode){case o.PAGE_UP:e=!0,this._move("previousPage",n);break;case o.PAGE_DOWN:e=!0,this._move("nextPage",n);break;case o.UP:e=!0,this._keyEvent("previous",n);break;case o.DOWN:e=!0,this._keyEvent("next",n);break;case o.ENTER:this.menu.active&&(e=!0,n.preventDefault(),this.menu.select(n));break;case o.TAB:this.menu.active&&this.menu.select(n);break;case o.ESCAPE:this.menu.element.is(":visible")&&(this.isMultiLine||this._value(this.term),this.close(n),n.preventDefault());break;default:i=!0,this._searchTimeout(n)}},keypress:function(s){if(e)return e=!1,(!this.isMultiLine||this.menu.element.is(":visible"))&&s.preventDefault(),void 0;if(!i){var n=t.ui.keyCode;switch(s.keyCode){case n.PAGE_UP:this._move("previousPage",s);break;case n.PAGE_DOWN:this._move("nextPage",s);break;case n.UP:this._keyEvent("previous",s);break;case n.DOWN:this._keyEvent("next",s)}}},input:function(t){return s?(s=!1,t.preventDefault(),void 0):(this._searchTimeout(t),void 0)},focus:function(){this.selectedItem=null,this.previous=this._value()},blur:function(t){return this.cancelBlur?(delete this.cancelBlur,void 0):(clearTimeout(this.searching),this.close(t),this._change(t),void 0)}}),this._initSource(),this.menu=t("<ul>").appendTo(this._appendTo()).menu({role:null}).hide().menu("instance"),this._addClass(this.menu.element,"ui-autocomplete","ui-front"),this._on(this.menu.element,{mousedown:function(e){e.preventDefault(),this.cancelBlur=!0,this._delay(function(){delete this.cancelBlur,this.element[0]!==t.ui.safeActiveElement(this.document[0])&&this.element.trigger("focus")})},menufocus:function(e,i){var s,n;return this.isNewMenu&&(this.isNewMenu=!1,e.originalEvent&&/^mouse/.test(e.originalEvent.type))?(this.menu.blur(),this.document.one("mousemove",function(){t(e.target).trigger(e.originalEvent)}),void 0):(n=i.item.data("ui-autocomplete-item"),!1!==this._trigger("focus",e,{item:n})&&e.originalEvent&&/^key/.test(e.originalEvent.type)&&this._value(n.value),s=i.item.attr("aria-label")||n.value,s&&t.trim(s).length&&(this.liveRegion.children().hide(),t("<div>").text(s).appendTo(this.liveRegion)),void 0)},menuselect:function(e,i){var s=i.item.data("ui-autocomplete-item"),n=this.previous;this.element[0]!==t.ui.safeActiveElement(this.document[0])&&(this.element.trigger("focus"),this.previous=n,this._delay(function(){this.previous=n,this.selectedItem=s})),!1!==this._trigger("select",e,{item:s})&&this._value(s.value),this.term=this._value(),this.close(e),this.selectedItem=s}}),this.liveRegion=t("<div>",{role:"status","aria-live":"assertive","aria-relevant":"additions"}).appendTo(this.document[0].body),this._addClass(this.liveRegion,null,"ui-helper-hidden-accessible"),this._on(this.window,{beforeunload:function(){this.element.removeAttr("autocomplete")}})},_destroy:function(){clearTimeout(this.searching),this.element.removeAttr("autocomplete"),this.menu.element.remove(),this.liveRegion.remove()},_setOption:function(t,e){this._super(t,e),"source"===t&&this._initSource(),"appendTo"===t&&this.menu.element.appendTo(this._appendTo()),"disabled"===t&&e&&this.xhr&&this.xhr.abort()},_isEventTargetInWidget:function(e){var i=this.menu.element[0];return e.target===this.element[0]||e.target===i||t.contains(i,e.target)},_closeOnClickOutside:function(t){this._isEventTargetInWidget(t)||this.close()},_appendTo:function(){var e=this.options.appendTo;return e&&(e=e.jquery||e.nodeType?t(e):this.document.find(e).eq(0)),e&&e[0]||(e=this.element.closest(".ui-front, dialog")),e.length||(e=this.document[0].body),e},_initSource:function(){var e,i,s=this;t.isArray(this.options.source)?(e=this.options.source,this.source=function(i,s){s(t.ui.autocomplete.filter(e,i.term))}):"string"==typeof this.options.source?(i=this.options.source,this.source=function(e,n){s.xhr&&s.xhr.abort(),s.xhr=t.ajax({url:i,data:e,dataType:"json",success:function(t){n(t)},error:function(){n([])}})}):this.source=this.options.source},_searchTimeout:function(t){clearTimeout(this.searching),this.searching=this._delay(function(){var e=this.term===this._value(),i=this.menu.element.is(":visible"),s=t.altKey||t.ctrlKey||t.metaKey||t.shiftKey;(!e||e&&!i&&!s)&&(this.selectedItem=null,this.search(null,t))},this.options.delay)},search:function(t,e){return t=null!=t?t:this._value(),this.term=this._value(),t.length<this.options.minLength?this.close(e):this._trigger("search",e)!==!1?this._search(t):void 0},_search:function(t){this.pending++,this._addClass("ui-autocomplete-loading"),this.cancelSearch=!1,this.source({term:t},this._response())},_response:function(){var e=++this.requestIndex;return t.proxy(function(t){e===this.requestIndex&&this.__response(t),this.pending--,this.pending||this._removeClass("ui-autocomplete-loading")},this)},__response:function(t){t&&(t=this._normalize(t)),this._trigger("response",null,{content:t}),!this.options.disabled&&t&&t.length&&!this.cancelSearch?(this._suggest(t),this._trigger("open")):this._close()},close:function(t){this.cancelSearch=!0,this._close(t)},_close:function(t){this._off(this.document,"mousedown"),this.menu.element.is(":visible")&&(this.menu.element.hide(),this.menu.blur(),this.isNewMenu=!0,this._trigger("close",t))},_change:function(t){this.previous!==this._value()&&this._trigger("change",t,{item:this.selectedItem})},_normalize:function(e){return e.length&&e[0].label&&e[0].value?e:t.map(e,function(e){return"string"==typeof e?{label:e,value:e}:t.extend({},e,{label:e.label||e.value,value:e.value||e.label})})},_suggest:function(e){var i=this.menu.element.empty();this._renderMenu(i,e),this.isNewMenu=!0,this.menu.refresh(),i.show(),this._resizeMenu(),i.position(t.extend({of:this.element},this.options.position)),this.options.autoFocus&&this.menu.next(),this._on(this.document,{mousedown:"_closeOnClickOutside"})},_resizeMenu:function(){var t=this.menu.element;t.outerWidth(Math.max(t.width("").outerWidth()+1,this.element.outerWidth()))},_renderMenu:function(e,i){var s=this;t.each(i,function(t,i){s._renderItemData(e,i)})},_renderItemData:function(t,e){return this._renderItem(t,e).data("ui-autocomplete-item",e)},_renderItem:function(e,i){return t("<li>").append(t("<div>").text(i.label)).appendTo(e)},_move:function(t,e){return this.menu.element.is(":visible")?this.menu.isFirstItem()&&/^previous/.test(t)||this.menu.isLastItem()&&/^next/.test(t)?(this.isMultiLine||this._value(this.term),this.menu.blur(),void 0):(this.menu[t](e),void 0):(this.search(null,e),void 0)},widget:function(){return this.menu.element},_value:function(){return this.valueMethod.apply(this.element,arguments)},_keyEvent:function(t,e){(!this.isMultiLine||this.menu.element.is(":visible"))&&(this._move(t,e),e.preventDefault())},_isContentEditable:function(t){if(!t.length)return!1;var e=t.prop("contentEditable");return"inherit"===e?this._isContentEditable(t.parent()):"true"===e}}),t.extend(t.ui.autocomplete,{escapeRegex:function(t){return t.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g,"\\$&")},filter:function(e,i){var s=RegExp(t.ui.autocomplete.escapeRegex(i),"i");return t.grep(e,function(t){return s.test(t.label||t.value||t)})}}),t.widget("ui.autocomplete",t.ui.autocomplete,{options:{messages:{noResults:"No search results.",results:function(t){return t+(t>1?" results are":" result is")+" available, use up and down arrow keys to navigate."}}},__response:function(e){var i;this._superApply(arguments),this.options.disabled||this.cancelSearch||(i=e&&e.length?this.options.messages.results(e.length):this.options.messages.noResults,this.liveRegion.children().hide(),t("<div>").text(i).appendTo(this.liveRegion))}}),t.ui.autocomplete;var g=/ui-corner-([a-z]){2,6}/g;t.widget("ui.controlgroup",{version:"1.12.1",defaultElement:"<div>",options:{direction:"horizontal",disabled:null,onlyVisible:!0,items:{button:"input[type=button], input[type=submit], input[type=reset], button, a",controlgroupLabel:".ui-controlgroup-label",checkboxradio:"input[type='checkbox'], input[type='radio']",selectmenu:"select",spinner:".ui-spinner-input"}},_create:function(){this._enhance()},_enhance:function(){this.element.attr("role","toolbar"),this.refresh()},_destroy:function(){this._callChildMethod("destroy"),this.childWidgets.removeData("ui-controlgroup-data"),this.element.removeAttr("role"),this.options.items.controlgroupLabel&&this.element.find(this.options.items.controlgroupLabel).find(".ui-controlgroup-label-contents").contents().unwrap()},_initWidgets:function(){var e=this,i=[];t.each(this.options.items,function(s,n){var o,a={};return n?"controlgroupLabel"===s?(o=e.element.find(n),o.each(function(){var e=t(this);e.children(".ui-controlgroup-label-contents").length||e.contents().wrapAll("<span class='ui-controlgroup-label-contents'></span>")}),e._addClass(o,null,"ui-widget ui-widget-content ui-state-default"),i=i.concat(o.get()),void 0):(t.fn[s]&&(a=e["_"+s+"Options"]?e["_"+s+"Options"]("middle"):{classes:{}},e.element.find(n).each(function(){var n=t(this),o=n[s]("instance"),r=t.widget.extend({},a);if("button"!==s||!n.parent(".ui-spinner").length){o||(o=n[s]()[s]("instance")),o&&(r.classes=e._resolveClassesValues(r.classes,o)),n[s](r);var h=n[s]("widget");t.data(h[0],"ui-controlgroup-data",o?o:n[s]("instance")),i.push(h[0])}})),void 0):void 0}),this.childWidgets=t(t.unique(i)),this._addClass(this.childWidgets,"ui-controlgroup-item")},_callChildMethod:function(e){this.childWidgets.each(function(){var i=t(this),s=i.data("ui-controlgroup-data");s&&s[e]&&s[e]()})},_updateCornerClass:function(t,e){var i="ui-corner-top ui-corner-bottom ui-corner-left ui-corner-right ui-corner-all",s=this._buildSimpleOptions(e,"label").classes.label;this._removeClass(t,null,i),this._addClass(t,null,s)},_buildSimpleOptions:function(t,e){var i="vertical"===this.options.direction,s={classes:{}};return s.classes[e]={middle:"",first:"ui-corner-"+(i?"top":"left"),last:"ui-corner-"+(i?"bottom":"right"),only:"ui-corner-all"}[t],s},_spinnerOptions:function(t){var e=this._buildSimpleOptions(t,"ui-spinner");return e.classes["ui-spinner-up"]="",e.classes["ui-spinner-down"]="",e},_buttonOptions:function(t){return this._buildSimpleOptions(t,"ui-button")},_checkboxradioOptions:function(t){return this._buildSimpleOptions(t,"ui-checkboxradio-label")},_selectmenuOptions:function(t){var e="vertical"===this.options.direction;return{width:e?"auto":!1,classes:{middle:{"ui-selectmenu-button-open":"","ui-selectmenu-button-closed":""},first:{"ui-selectmenu-button-open":"ui-corner-"+(e?"top":"tl"),"ui-selectmenu-button-closed":"ui-corner-"+(e?"top":"left")},last:{"ui-selectmenu-button-open":e?"":"ui-corner-tr","ui-selectmenu-button-closed":"ui-corner-"+(e?"bottom":"right")},only:{"ui-selectmenu-button-open":"ui-corner-top","ui-selectmenu-button-closed":"ui-corner-all"}}[t]}},_resolveClassesValues:function(e,i){var s={};return t.each(e,function(n){var o=i.options.classes[n]||"";o=t.trim(o.replace(g,"")),s[n]=(o+" "+e[n]).replace(/\s+/g," ")}),s},_setOption:function(t,e){return"direction"===t&&this._removeClass("ui-controlgroup-"+this.options.direction),this._super(t,e),"disabled"===t?(this._callChildMethod(e?"disable":"enable"),void 0):(this.refresh(),void 0)},refresh:function(){var e,i=this;this._addClass("ui-controlgroup ui-controlgroup-"+this.options.direction),"horizontal"===this.options.direction&&this._addClass(null,"ui-helper-clearfix"),this._initWidgets(),e=this.childWidgets,this.options.onlyVisible&&(e=e.filter(":visible")),e.length&&(t.each(["first","last"],function(t,s){var n=e[s]().data("ui-controlgroup-data");if(n&&i["_"+n.widgetName+"Options"]){var o=i["_"+n.widgetName+"Options"](1===e.length?"only":s);o.classes=i._resolveClassesValues(o.classes,n),n.element[n.widgetName](o)}else i._updateCornerClass(e[s](),s)}),this._callChildMethod("refresh"))}}),t.widget("ui.checkboxradio",[t.ui.formResetMixin,{version:"1.12.1",options:{disabled:null,label:null,icon:!0,classes:{"ui-checkboxradio-label":"ui-corner-all","ui-checkboxradio-icon":"ui-corner-all"}},_getCreateOptions:function(){var e,i,s=this,n=this._super()||{};return this._readType(),i=this.element.labels(),this.label=t(i[i.length-1]),this.label.length||t.error("No label found for checkboxradio widget"),this.originalLabel="",this.label.contents().not(this.element[0]).each(function(){s.originalLabel+=3===this.nodeType?t(this).text():this.outerHTML}),this.originalLabel&&(n.label=this.originalLabel),e=this.element[0].disabled,null!=e&&(n.disabled=e),n},_create:function(){var t=this.element[0].checked;this._bindFormResetHandler(),null==this.options.disabled&&(this.options.disabled=this.element[0].disabled),this._setOption("disabled",this.options.disabled),this._addClass("ui-checkboxradio","ui-helper-hidden-accessible"),this._addClass(this.label,"ui-checkboxradio-label","ui-button ui-widget"),"radio"===this.type&&this._addClass(this.label,"ui-checkboxradio-radio-label"),this.options.label&&this.options.label!==this.originalLabel?this._updateLabel():this.originalLabel&&(this.options.label=this.originalLabel),this._enhance(),t&&(this._addClass(this.label,"ui-checkboxradio-checked","ui-state-active"),this.icon&&this._addClass(this.icon,null,"ui-state-hover")),this._on({change:"_toggleClasses",focus:function(){this._addClass(this.label,null,"ui-state-focus ui-visual-focus")},blur:function(){this._removeClass(this.label,null,"ui-state-focus ui-visual-focus")}})},_readType:function(){var e=this.element[0].nodeName.toLowerCase();this.type=this.element[0].type,"input"===e&&/radio|checkbox/.test(this.type)||t.error("Can't create checkboxradio on element.nodeName="+e+" and element.type="+this.type)},_enhance:function(){this._updateIcon(this.element[0].checked)},widget:function(){return this.label},_getRadioGroup:function(){var e,i=this.element[0].name,s="input[name='"+t.ui.escapeSelector(i)+"']";return i?(e=this.form.length?t(this.form[0].elements).filter(s):t(s).filter(function(){return 0===t(this).form().length}),e.not(this.element)):t([])},_toggleClasses:function(){var e=this.element[0].checked;this._toggleClass(this.label,"ui-checkboxradio-checked","ui-state-active",e),this.options.icon&&"checkbox"===this.type&&this._toggleClass(this.icon,null,"ui-icon-check ui-state-checked",e)._toggleClass(this.icon,null,"ui-icon-blank",!e),"radio"===this.type&&this._getRadioGroup().each(function(){var e=t(this).checkboxradio("instance");e&&e._removeClass(e.label,"ui-checkboxradio-checked","ui-state-active")})},_destroy:function(){this._unbindFormResetHandler(),this.icon&&(this.icon.remove(),this.iconSpace.remove())},_setOption:function(t,e){return"label"!==t||e?(this._super(t,e),"disabled"===t?(this._toggleClass(this.label,null,"ui-state-disabled",e),this.element[0].disabled=e,void 0):(this.refresh(),void 0)):void 0},_updateIcon:function(e){var i="ui-icon ui-icon-background ";this.options.icon?(this.icon||(this.icon=t("<span>"),this.iconSpace=t("<span> </span>"),this._addClass(this.iconSpace,"ui-checkboxradio-icon-space")),"checkbox"===this.type?(i+=e?"ui-icon-check ui-state-checked":"ui-icon-blank",this._removeClass(this.icon,null,e?"ui-icon-blank":"ui-icon-check")):i+="ui-icon-blank",this._addClass(this.icon,"ui-checkboxradio-icon",i),e||this._removeClass(this.icon,null,"ui-icon-check ui-state-checked"),this.icon.prependTo(this.label).after(this.iconSpace)):void 0!==this.icon&&(this.icon.remove(),this.iconSpace.remove(),delete this.icon)},_updateLabel:function(){var t=this.label.contents().not(this.element[0]);this.icon&&(t=t.not(this.icon[0])),this.iconSpace&&(t=t.not(this.iconSpace[0])),t.remove(),this.label.append(this.options.label)},refresh:function(){var t=this.element[0].checked,e=this.element[0].disabled;this._updateIcon(t),this._toggleClass(this.label,"ui-checkboxradio-checked","ui-state-active",t),null!==this.options.label&&this._updateLabel(),e!==this.options.disabled&&this._setOptions({disabled:e})}}]),t.ui.checkboxradio,t.widget("ui.button",{version:"1.12.1",defaultElement:"<button>",options:{classes:{"ui-button":"ui-corner-all"},disabled:null,icon:null,iconPosition:"beginning",label:null,showLabel:!0},_getCreateOptions:function(){var t,e=this._super()||{};return this.isInput=this.element.is("input"),t=this.element[0].disabled,null!=t&&(e.disabled=t),this.originalLabel=this.isInput?this.element.val():this.element.html(),this.originalLabel&&(e.label=this.originalLabel),e},_create:function(){!this.option.showLabel&!this.options.icon&&(this.options.showLabel=!0),null==this.options.disabled&&(this.options.disabled=this.element[0].disabled||!1),this.hasTitle=!!this.element.attr("title"),this.options.label&&this.options.label!==this.originalLabel&&(this.isInput?this.element.val(this.options.label):this.element.html(this.options.label)),this._addClass("ui-button","ui-widget"),this._setOption("disabled",this.options.disabled),this._enhance(),this.element.is("a")&&this._on({keyup:function(e){e.keyCode===t.ui.keyCode.SPACE&&(e.preventDefault(),this.element[0].click?this.element[0].click():this.element.trigger("click"))}})},_enhance:function(){this.element.is("button")||this.element.attr("role","button"),this.options.icon&&(this._updateIcon("icon",this.options.icon),this._updateTooltip())},_updateTooltip:function(){this.title=this.element.attr("title"),this.options.showLabel||this.title||this.element.attr("title",this.options.label)},_updateIcon:function(e,i){var s="iconPosition"!==e,n=s?this.options.iconPosition:i,o="top"===n||"bottom"===n;this.icon?s&&this._removeClass(this.icon,null,this.options.icon):(this.icon=t("<span>"),this._addClass(this.icon,"ui-button-icon","ui-icon"),this.options.showLabel||this._addClass("ui-button-icon-only")),s&&this._addClass(this.icon,null,i),this._attachIcon(n),o?(this._addClass(this.icon,null,"ui-widget-icon-block"),this.iconSpace&&this.iconSpace.remove()):(this.iconSpace||(this.iconSpace=t("<span> </span>"),this._addClass(this.iconSpace,"ui-button-icon-space")),this._removeClass(this.icon,null,"ui-wiget-icon-block"),this._attachIconSpace(n))},_destroy:function(){this.element.removeAttr("role"),this.icon&&this.icon.remove(),this.iconSpace&&this.iconSpace.remove(),this.hasTitle||this.element.removeAttr("title")},_attachIconSpace:function(t){this.icon[/^(?:end|bottom)/.test(t)?"before":"after"](this.iconSpace)},_attachIcon:function(t){this.element[/^(?:end|bottom)/.test(t)?"append":"prepend"](this.icon)},_setOptions:function(t){var e=void 0===t.showLabel?this.options.showLabel:t.showLabel,i=void 0===t.icon?this.options.icon:t.icon;e||i||(t.showLabel=!0),this._super(t)},_setOption:function(t,e){"icon"===t&&(e?this._updateIcon(t,e):this.icon&&(this.icon.remove(),this.iconSpace&&this.iconSpace.remove())),"iconPosition"===t&&this._updateIcon(t,e),"showLabel"===t&&(this._toggleClass("ui-button-icon-only",null,!e),this._updateTooltip()),"label"===t&&(this.isInput?this.element.val(e):(this.element.html(e),this.icon&&(this._attachIcon(this.options.iconPosition),this._attachIconSpace(this.options.iconPosition)))),this._super(t,e),"disabled"===t&&(this._toggleClass(null,"ui-state-disabled",e),this.element[0].disabled=e,e&&this.element.blur())},refresh:function(){var t=this.element.is("input, button")?this.element[0].disabled:this.element.hasClass("ui-button-disabled");t!==this.options.disabled&&this._setOptions({disabled:t}),this._updateTooltip()}}),t.uiBackCompat!==!1&&(t.widget("ui.button",t.ui.button,{options:{text:!0,icons:{primary:null,secondary:null}},_create:function(){this.options.showLabel&&!this.options.text&&(this.options.showLabel=this.options.text),!this.options.showLabel&&this.options.text&&(this.options.text=this.options.showLabel),this.options.icon||!this.options.icons.primary&&!this.options.icons.secondary?this.options.icon&&(this.options.icons.primary=this.options.icon):this.options.icons.primary?this.options.icon=this.options.icons.primary:(this.options.icon=this.options.icons.secondary,this.options.iconPosition="end"),this._super()},_setOption:function(t,e){return"text"===t?(this._super("showLabel",e),void 0):("showLabel"===t&&(this.options.text=e),"icon"===t&&(this.options.icons.primary=e),"icons"===t&&(e.primary?(this._super("icon",e.primary),this._super("iconPosition","beginning")):e.secondary&&(this._super("icon",e.secondary),this._super("iconPosition","end"))),this._superApply(arguments),void 0)}}),t.fn.button=function(e){return function(){return!this.length||this.length&&"INPUT"!==this[0].tagName||this.length&&"INPUT"===this[0].tagName&&"checkbox"!==this.attr("type")&&"radio"!==this.attr("type")?e.apply(this,arguments):(t.ui.checkboxradio||t.error("Checkboxradio widget missing"),0===arguments.length?this.checkboxradio({icon:!1}):this.checkboxradio.apply(this,arguments))}}(t.fn.button),t.fn.buttonset=function(){return t.ui.controlgroup||t.error("Controlgroup widget missing"),"option"===arguments[0]&&"items"===arguments[1]&&arguments[2]?this.controlgroup.apply(this,[arguments[0],"items.button",arguments[2]]):"option"===arguments[0]&&"items"===arguments[1]?this.controlgroup.apply(this,[arguments[0],"items.button"]):("object"==typeof arguments[0]&&arguments[0].items&&(arguments[0].items={button:arguments[0].items}),this.controlgroup.apply(this,arguments))}),t.ui.button,t.extend(t.ui,{datepicker:{version:"1.12.1"}});var m;t.extend(s.prototype,{markerClassName:"hasDatepicker",maxRows:4,_widgetDatepicker:function(){return this.dpDiv},setDefaults:function(t){return a(this._defaults,t||{}),this},_attachDatepicker:function(e,i){var s,n,o;s=e.nodeName.toLowerCase(),n="div"===s||"span"===s,e.id||(this.uuid+=1,e.id="dp"+this.uuid),o=this._newInst(t(e),n),o.settings=t.extend({},i||{}),"input"===s?this._connectDatepicker(e,o):n&&this._inlineDatepicker(e,o)},_newInst:function(e,i){var s=e[0].id.replace(/([^A-Za-z0-9_\-])/g,"\\\\$1");return{id:s,input:e,selectedDay:0,selectedMonth:0,selectedYear:0,drawMonth:0,drawYear:0,inline:i,dpDiv:i?n(t("<div class='"+this._inlineClass+" ui-datepicker ui-widget ui-widget-content ui-helper-clearfix ui-corner-all'></div>")):this.dpDiv}},_connectDatepicker:function(e,i){var s=t(e);i.append=t([]),i.trigger=t([]),s.hasClass(this.markerClassName)||(this._attachments(s,i),s.addClass(this.markerClassName).on("keydown",this._doKeyDown).on("keypress",this._doKeyPress).on("keyup",this._doKeyUp),this._autoSize(i),t.data(e,"datepicker",i),i.settings.disabled&&this._disableDatepicker(e))},_attachments:function(e,i){var s,n,o,a=this._get(i,"appendText"),r=this._get(i,"isRTL");i.append&&i.append.remove(),a&&(i.append=t("<span class='"+this._appendClass+"'>"+a+"</span>"),e[r?"before":"after"](i.append)),e.off("focus",this._showDatepicker),i.trigger&&i.trigger.remove(),s=this._get(i,"showOn"),("focus"===s||"both"===s)&&e.on("focus",this._showDatepicker),("button"===s||"both"===s)&&(n=this._get(i,"buttonText"),o=this._get(i,"buttonImage"),i.trigger=t(this._get(i,"buttonImageOnly")?t("<img/>").addClass(this._triggerClass).attr({src:o,alt:n,title:n}):t("<button type='button'></button>").addClass(this._triggerClass).html(o?t("<img/>").attr({src:o,alt:n,title:n}):n)),e[r?"before":"after"](i.trigger),i.trigger.on("click",function(){return t.datepicker._datepickerShowing&&t.datepicker._lastInput===e[0]?t.datepicker._hideDatepicker():t.datepicker._datepickerShowing&&t.datepicker._lastInput!==e[0]?(t.datepicker._hideDatepicker(),t.datepicker._showDatepicker(e[0])):t.datepicker._showDatepicker(e[0]),!1}))},_autoSize:function(t){if(this._get(t,"autoSize")&&!t.inline){var e,i,s,n,o=new Date(2009,11,20),a=this._get(t,"dateFormat");a.match(/[DM]/)&&(e=function(t){for(i=0,s=0,n=0;t.length>n;n++)t[n].length>i&&(i=t[n].length,s=n);return s},o.setMonth(e(this._get(t,a.match(/MM/)?"monthNames":"monthNamesShort"))),o.setDate(e(this._get(t,a.match(/DD/)?"dayNames":"dayNamesShort"))+20-o.getDay())),t.input.attr("size",this._formatDate(t,o).length)}},_inlineDatepicker:function(e,i){var s=t(e);s.hasClass(this.markerClassName)||(s.addClass(this.markerClassName).append(i.dpDiv),t.data(e,"datepicker",i),this._setDate(i,this._getDefaultDate(i),!0),this._updateDatepicker(i),this._updateAlternate(i),i.settings.disabled&&this._disableDatepicker(e),i.dpDiv.css("display","block"))},_dialogDatepicker:function(e,i,s,n,o){var r,h,l,c,u,d=this._dialogInst;return d||(this.uuid+=1,r="dp"+this.uuid,this._dialogInput=t("<input type='text' id='"+r+"' style='position: absolute; top: -100px; width: 0px;'/>"),this._dialogInput.on("keydown",this._doKeyDown),t("body").append(this._dialogInput),d=this._dialogInst=this._newInst(this._dialogInput,!1),d.settings={},t.data(this._dialogInput[0],"datepicker",d)),a(d.settings,n||{}),i=i&&i.constructor===Date?this._formatDate(d,i):i,this._dialogInput.val(i),this._pos=o?o.length?o:[o.pageX,o.pageY]:null,this._pos||(h=document.documentElement.clientWidth,l=document.documentElement.clientHeight,c=document.documentElement.scrollLeft||document.body.scrollLeft,u=document.documentElement.scrollTop||document.body.scrollTop,this._pos=[h/2-100+c,l/2-150+u]),this._dialogInput.css("left",this._pos[0]+20+"px").css("top",this._pos[1]+"px"),d.settings.onSelect=s,this._inDialog=!0,this.dpDiv.addClass(this._dialogClass),this._showDatepicker(this._dialogInput[0]),t.blockUI&&t.blockUI(this.dpDiv),t.data(this._dialogInput[0],"datepicker",d),this},_destroyDatepicker:function(e){var i,s=t(e),n=t.data(e,"datepicker");s.hasClass(this.markerClassName)&&(i=e.nodeName.toLowerCase(),t.removeData(e,"datepicker"),"input"===i?(n.append.remove(),n.trigger.remove(),s.removeClass(this.markerClassName).off("focus",this._showDatepicker).off("keydown",this._doKeyDown).off("keypress",this._doKeyPress).off("keyup",this._doKeyUp)):("div"===i||"span"===i)&&s.removeClass(this.markerClassName).empty(),m===n&&(m=null))},_enableDatepicker:function(e){var i,s,n=t(e),o=t.data(e,"datepicker");n.hasClass(this.markerClassName)&&(i=e.nodeName.toLowerCase(),"input"===i?(e.disabled=!1,o.trigger.filter("button").each(function(){this.disabled=!1}).end().filter("img").css({opacity:"1.0",cursor:""})):("div"===i||"span"===i)&&(s=n.children("."+this._inlineClass),s.children().removeClass("ui-state-disabled"),s.find("select.ui-datepicker-month, select.ui-datepicker-year").prop("disabled",!1)),this._disabledInputs=t.map(this._disabledInputs,function(t){return t===e?null:t}))},_disableDatepicker:function(e){var i,s,n=t(e),o=t.data(e,"datepicker");n.hasClass(this.markerClassName)&&(i=e.nodeName.toLowerCase(),"input"===i?(e.disabled=!0,o.trigger.filter("button").each(function(){this.disabled=!0}).end().filter("img").css({opacity:"0.5",cursor:"default"})):("div"===i||"span"===i)&&(s=n.children("."+this._inlineClass),s.children().addClass("ui-state-disabled"),s.find("select.ui-datepicker-month, select.ui-datepicker-year").prop("disabled",!0)),this._disabledInputs=t.map(this._disabledInputs,function(t){return t===e?null:t}),this._disabledInputs[this._disabledInputs.length]=e)},_isDisabledDatepicker:function(t){if(!t)return!1;for(var e=0;this._disabledInputs.length>e;e++)if(this._disabledInputs[e]===t)return!0;return!1},_getInst:function(e){try{return t.data(e,"datepicker")}catch(i){throw"Missing instance data for this datepicker"}},_optionDatepicker:function(e,i,s){var n,o,r,h,l=this._getInst(e);return 2===arguments.length&&"string"==typeof i?"defaults"===i?t.extend({},t.datepicker._defaults):l?"all"===i?t.extend({},l.settings):this._get(l,i):null:(n=i||{},"string"==typeof i&&(n={},n[i]=s),l&&(this._curInst===l&&this._hideDatepicker(),o=this._getDateDatepicker(e,!0),r=this._getMinMaxDate(l,"min"),h=this._getMinMaxDate(l,"max"),a(l.settings,n),null!==r&&void 0!==n.dateFormat&&void 0===n.minDate&&(l.settings.minDate=this._formatDate(l,r)),null!==h&&void 0!==n.dateFormat&&void 0===n.maxDate&&(l.settings.maxDate=this._formatDate(l,h)),"disabled"in n&&(n.disabled?this._disableDatepicker(e):this._enableDatepicker(e)),this._attachments(t(e),l),this._autoSize(l),this._setDate(l,o),this._updateAlternate(l),this._updateDatepicker(l)),void 0)},_changeDatepicker:function(t,e,i){this._optionDatepicker(t,e,i)},_refreshDatepicker:function(t){var e=this._getInst(t);e&&this._updateDatepicker(e)},_setDateDatepicker:function(t,e){var i=this._getInst(t);i&&(this._setDate(i,e),this._updateDatepicker(i),this._updateAlternate(i))},_getDateDatepicker:function(t,e){var i=this._getInst(t);return i&&!i.inline&&this._setDateFromField(i,e),i?this._getDate(i):null},_doKeyDown:function(e){var i,s,n,o=t.datepicker._getInst(e.target),a=!0,r=o.dpDiv.is(".ui-datepicker-rtl");if(o._keyEvent=!0,t.datepicker._datepickerShowing)switch(e.keyCode){case 9:t.datepicker._hideDatepicker(),a=!1;break;case 13:return n=t("td."+t.datepicker._dayOverClass+":not(."+t.datepicker._currentClass+")",o.dpDiv),n[0]&&t.datepicker._selectDay(e.target,o.selectedMonth,o.selectedYear,n[0]),i=t.datepicker._get(o,"onSelect"),i?(s=t.datepicker._formatDate(o),i.apply(o.input?o.input[0]:null,[s,o])):t.datepicker._hideDatepicker(),!1;case 27:t.datepicker._hideDatepicker();break;case 33:t.datepicker._adjustDate(e.target,e.ctrlKey?-t.datepicker._get(o,"stepBigMonths"):-t.datepicker._get(o,"stepMonths"),"M");break;case 34:t.datepicker._adjustDate(e.target,e.ctrlKey?+t.datepicker._get(o,"stepBigMonths"):+t.datepicker._get(o,"stepMonths"),"M");break;case 35:(e.ctrlKey||e.metaKey)&&t.datepicker._clearDate(e.target),a=e.ctrlKey||e.metaKey;break;case 36:(e.ctrlKey||e.metaKey)&&t.datepicker._gotoToday(e.target),a=e.ctrlKey||e.metaKey;break;case 37:(e.ctrlKey||e.metaKey)&&t.datepicker._adjustDate(e.target,r?1:-1,"D"),a=e.ctrlKey||e.metaKey,e.originalEvent.altKey&&t.datepicker._adjustDate(e.target,e.ctrlKey?-t.datepicker._get(o,"stepBigMonths"):-t.datepicker._get(o,"stepMonths"),"M");break;case 38:(e.ctrlKey||e.metaKey)&&t.datepicker._adjustDate(e.target,-7,"D"),a=e.ctrlKey||e.metaKey;break;case 39:(e.ctrlKey||e.metaKey)&&t.datepicker._adjustDate(e.target,r?-1:1,"D"),a=e.ctrlKey||e.metaKey,e.originalEvent.altKey&&t.datepicker._adjustDate(e.target,e.ctrlKey?+t.datepicker._get(o,"stepBigMonths"):+t.datepicker._get(o,"stepMonths"),"M");break;case 40:(e.ctrlKey||e.metaKey)&&t.datepicker._adjustDate(e.target,7,"D"),a=e.ctrlKey||e.metaKey;break;default:a=!1}else 36===e.keyCode&&e.ctrlKey?t.datepicker._showDatepicker(this):a=!1;a&&(e.preventDefault(),e.stopPropagation())},_doKeyPress:function(e){var i,s,n=t.datepicker._getInst(e.target);return t.datepicker._get(n,"constrainInput")?(i=t.datepicker._possibleChars(t.datepicker._get(n,"dateFormat")),s=String.fromCharCode(null==e.charCode?e.keyCode:e.charCode),e.ctrlKey||e.metaKey||" ">s||!i||i.indexOf(s)>-1):void 0},_doKeyUp:function(e){var i,s=t.datepicker._getInst(e.target);if(s.input.val()!==s.lastVal)try{i=t.datepicker.parseDate(t.datepicker._get(s,"dateFormat"),s.input?s.input.val():null,t.datepicker._getFormatConfig(s)),i&&(t.datepicker._setDateFromField(s),t.datepicker._updateAlternate(s),t.datepicker._updateDatepicker(s))}catch(n){}return!0},_showDatepicker:function(e){if(e=e.target||e,"input"!==e.nodeName.toLowerCase()&&(e=t("input",e.parentNode)[0]),!t.datepicker._isDisabledDatepicker(e)&&t.datepicker._lastInput!==e){var s,n,o,r,h,l,c;s=t.datepicker._getInst(e),t.datepicker._curInst&&t.datepicker._curInst!==s&&(t.datepicker._curInst.dpDiv.stop(!0,!0),s&&t.datepicker._datepickerShowing&&t.datepicker._hideDatepicker(t.datepicker._curInst.input[0])),n=t.datepicker._get(s,"beforeShow"),o=n?n.apply(e,[e,s]):{},o!==!1&&(a(s.settings,o),s.lastVal=null,t.datepicker._lastInput=e,t.datepicker._setDateFromField(s),t.datepicker._inDialog&&(e.value=""),t.datepicker._pos||(t.datepicker._pos=t.datepicker._findPos(e),t.datepicker._pos[1]+=e.offsetHeight),r=!1,t(e).parents().each(function(){return r|="fixed"===t(this).css("position"),!r}),h={left:t.datepicker._pos[0],top:t.datepicker._pos[1]},t.datepicker._pos=null,s.dpDiv.empty(),s.dpDiv.css({position:"absolute",display:"block",top:"-1000px"}),t.datepicker._updateDatepicker(s),h=t.datepicker._checkOffset(s,h,r),s.dpDiv.css({position:t.datepicker._inDialog&&t.blockUI?"static":r?"fixed":"absolute",display:"none",left:h.left+"px",top:h.top+"px"}),s.inline||(l=t.datepicker._get(s,"showAnim"),c=t.datepicker._get(s,"duration"),s.dpDiv.css("z-index",i(t(e))+1),t.datepicker._datepickerShowing=!0,t.effects&&t.effects.effect[l]?s.dpDiv.show(l,t.datepicker._get(s,"showOptions"),c):s.dpDiv[l||"show"](l?c:null),t.datepicker._shouldFocusInput(s)&&s.input.trigger("focus"),t.datepicker._curInst=s)) -}},_updateDatepicker:function(e){this.maxRows=4,m=e,e.dpDiv.empty().append(this._generateHTML(e)),this._attachHandlers(e);var i,s=this._getNumberOfMonths(e),n=s[1],a=17,r=e.dpDiv.find("."+this._dayOverClass+" a");r.length>0&&o.apply(r.get(0)),e.dpDiv.removeClass("ui-datepicker-multi-2 ui-datepicker-multi-3 ui-datepicker-multi-4").width(""),n>1&&e.dpDiv.addClass("ui-datepicker-multi-"+n).css("width",a*n+"em"),e.dpDiv[(1!==s[0]||1!==s[1]?"add":"remove")+"Class"]("ui-datepicker-multi"),e.dpDiv[(this._get(e,"isRTL")?"add":"remove")+"Class"]("ui-datepicker-rtl"),e===t.datepicker._curInst&&t.datepicker._datepickerShowing&&t.datepicker._shouldFocusInput(e)&&e.input.trigger("focus"),e.yearshtml&&(i=e.yearshtml,setTimeout(function(){i===e.yearshtml&&e.yearshtml&&e.dpDiv.find("select.ui-datepicker-year:first").replaceWith(e.yearshtml),i=e.yearshtml=null},0))},_shouldFocusInput:function(t){return t.input&&t.input.is(":visible")&&!t.input.is(":disabled")&&!t.input.is(":focus")},_checkOffset:function(e,i,s){var n=e.dpDiv.outerWidth(),o=e.dpDiv.outerHeight(),a=e.input?e.input.outerWidth():0,r=e.input?e.input.outerHeight():0,h=document.documentElement.clientWidth+(s?0:t(document).scrollLeft()),l=document.documentElement.clientHeight+(s?0:t(document).scrollTop());return i.left-=this._get(e,"isRTL")?n-a:0,i.left-=s&&i.left===e.input.offset().left?t(document).scrollLeft():0,i.top-=s&&i.top===e.input.offset().top+r?t(document).scrollTop():0,i.left-=Math.min(i.left,i.left+n>h&&h>n?Math.abs(i.left+n-h):0),i.top-=Math.min(i.top,i.top+o>l&&l>o?Math.abs(o+r):0),i},_findPos:function(e){for(var i,s=this._getInst(e),n=this._get(s,"isRTL");e&&("hidden"===e.type||1!==e.nodeType||t.expr.filters.hidden(e));)e=e[n?"previousSibling":"nextSibling"];return i=t(e).offset(),[i.left,i.top]},_hideDatepicker:function(e){var i,s,n,o,a=this._curInst;!a||e&&a!==t.data(e,"datepicker")||this._datepickerShowing&&(i=this._get(a,"showAnim"),s=this._get(a,"duration"),n=function(){t.datepicker._tidyDialog(a)},t.effects&&(t.effects.effect[i]||t.effects[i])?a.dpDiv.hide(i,t.datepicker._get(a,"showOptions"),s,n):a.dpDiv["slideDown"===i?"slideUp":"fadeIn"===i?"fadeOut":"hide"](i?s:null,n),i||n(),this._datepickerShowing=!1,o=this._get(a,"onClose"),o&&o.apply(a.input?a.input[0]:null,[a.input?a.input.val():"",a]),this._lastInput=null,this._inDialog&&(this._dialogInput.css({position:"absolute",left:"0",top:"-100px"}),t.blockUI&&(t.unblockUI(),t("body").append(this.dpDiv))),this._inDialog=!1)},_tidyDialog:function(t){t.dpDiv.removeClass(this._dialogClass).off(".ui-datepicker-calendar")},_checkExternalClick:function(e){if(t.datepicker._curInst){var i=t(e.target),s=t.datepicker._getInst(i[0]);(i[0].id!==t.datepicker._mainDivId&&0===i.parents("#"+t.datepicker._mainDivId).length&&!i.hasClass(t.datepicker.markerClassName)&&!i.closest("."+t.datepicker._triggerClass).length&&t.datepicker._datepickerShowing&&(!t.datepicker._inDialog||!t.blockUI)||i.hasClass(t.datepicker.markerClassName)&&t.datepicker._curInst!==s)&&t.datepicker._hideDatepicker()}},_adjustDate:function(e,i,s){var n=t(e),o=this._getInst(n[0]);this._isDisabledDatepicker(n[0])||(this._adjustInstDate(o,i+("M"===s?this._get(o,"showCurrentAtPos"):0),s),this._updateDatepicker(o))},_gotoToday:function(e){var i,s=t(e),n=this._getInst(s[0]);this._get(n,"gotoCurrent")&&n.currentDay?(n.selectedDay=n.currentDay,n.drawMonth=n.selectedMonth=n.currentMonth,n.drawYear=n.selectedYear=n.currentYear):(i=new Date,n.selectedDay=i.getDate(),n.drawMonth=n.selectedMonth=i.getMonth(),n.drawYear=n.selectedYear=i.getFullYear()),this._notifyChange(n),this._adjustDate(s)},_selectMonthYear:function(e,i,s){var n=t(e),o=this._getInst(n[0]);o["selected"+("M"===s?"Month":"Year")]=o["draw"+("M"===s?"Month":"Year")]=parseInt(i.options[i.selectedIndex].value,10),this._notifyChange(o),this._adjustDate(n)},_selectDay:function(e,i,s,n){var o,a=t(e);t(n).hasClass(this._unselectableClass)||this._isDisabledDatepicker(a[0])||(o=this._getInst(a[0]),o.selectedDay=o.currentDay=t("a",n).html(),o.selectedMonth=o.currentMonth=i,o.selectedYear=o.currentYear=s,this._selectDate(e,this._formatDate(o,o.currentDay,o.currentMonth,o.currentYear)))},_clearDate:function(e){var i=t(e);this._selectDate(i,"")},_selectDate:function(e,i){var s,n=t(e),o=this._getInst(n[0]);i=null!=i?i:this._formatDate(o),o.input&&o.input.val(i),this._updateAlternate(o),s=this._get(o,"onSelect"),s?s.apply(o.input?o.input[0]:null,[i,o]):o.input&&o.input.trigger("change"),o.inline?this._updateDatepicker(o):(this._hideDatepicker(),this._lastInput=o.input[0],"object"!=typeof o.input[0]&&o.input.trigger("focus"),this._lastInput=null)},_updateAlternate:function(e){var i,s,n,o=this._get(e,"altField");o&&(i=this._get(e,"altFormat")||this._get(e,"dateFormat"),s=this._getDate(e),n=this.formatDate(i,s,this._getFormatConfig(e)),t(o).val(n))},noWeekends:function(t){var e=t.getDay();return[e>0&&6>e,""]},iso8601Week:function(t){var e,i=new Date(t.getTime());return i.setDate(i.getDate()+4-(i.getDay()||7)),e=i.getTime(),i.setMonth(0),i.setDate(1),Math.floor(Math.round((e-i)/864e5)/7)+1},parseDate:function(e,i,s){if(null==e||null==i)throw"Invalid arguments";if(i="object"==typeof i?""+i:i+"",""===i)return null;var n,o,a,r,h=0,l=(s?s.shortYearCutoff:null)||this._defaults.shortYearCutoff,c="string"!=typeof l?l:(new Date).getFullYear()%100+parseInt(l,10),u=(s?s.dayNamesShort:null)||this._defaults.dayNamesShort,d=(s?s.dayNames:null)||this._defaults.dayNames,p=(s?s.monthNamesShort:null)||this._defaults.monthNamesShort,f=(s?s.monthNames:null)||this._defaults.monthNames,g=-1,m=-1,_=-1,v=-1,b=!1,y=function(t){var i=e.length>n+1&&e.charAt(n+1)===t;return i&&n++,i},w=function(t){var e=y(t),s="@"===t?14:"!"===t?20:"y"===t&&e?4:"o"===t?3:2,n="y"===t?s:1,o=RegExp("^\\d{"+n+","+s+"}"),a=i.substring(h).match(o);if(!a)throw"Missing number at position "+h;return h+=a[0].length,parseInt(a[0],10)},k=function(e,s,n){var o=-1,a=t.map(y(e)?n:s,function(t,e){return[[e,t]]}).sort(function(t,e){return-(t[1].length-e[1].length)});if(t.each(a,function(t,e){var s=e[1];return i.substr(h,s.length).toLowerCase()===s.toLowerCase()?(o=e[0],h+=s.length,!1):void 0}),-1!==o)return o+1;throw"Unknown name at position "+h},x=function(){if(i.charAt(h)!==e.charAt(n))throw"Unexpected literal at position "+h;h++};for(n=0;e.length>n;n++)if(b)"'"!==e.charAt(n)||y("'")?x():b=!1;else switch(e.charAt(n)){case"d":_=w("d");break;case"D":k("D",u,d);break;case"o":v=w("o");break;case"m":m=w("m");break;case"M":m=k("M",p,f);break;case"y":g=w("y");break;case"@":r=new Date(w("@")),g=r.getFullYear(),m=r.getMonth()+1,_=r.getDate();break;case"!":r=new Date((w("!")-this._ticksTo1970)/1e4),g=r.getFullYear(),m=r.getMonth()+1,_=r.getDate();break;case"'":y("'")?x():b=!0;break;default:x()}if(i.length>h&&(a=i.substr(h),!/^\s+/.test(a)))throw"Extra/unparsed characters found in date: "+a;if(-1===g?g=(new Date).getFullYear():100>g&&(g+=(new Date).getFullYear()-(new Date).getFullYear()%100+(c>=g?0:-100)),v>-1)for(m=1,_=v;;){if(o=this._getDaysInMonth(g,m-1),o>=_)break;m++,_-=o}if(r=this._daylightSavingAdjust(new Date(g,m-1,_)),r.getFullYear()!==g||r.getMonth()+1!==m||r.getDate()!==_)throw"Invalid date";return r},ATOM:"yy-mm-dd",COOKIE:"D, dd M yy",ISO_8601:"yy-mm-dd",RFC_822:"D, d M y",RFC_850:"DD, dd-M-y",RFC_1036:"D, d M y",RFC_1123:"D, d M yy",RFC_2822:"D, d M yy",RSS:"D, d M y",TICKS:"!",TIMESTAMP:"@",W3C:"yy-mm-dd",_ticksTo1970:1e7*60*60*24*(718685+Math.floor(492.5)-Math.floor(19.7)+Math.floor(4.925)),formatDate:function(t,e,i){if(!e)return"";var s,n=(i?i.dayNamesShort:null)||this._defaults.dayNamesShort,o=(i?i.dayNames:null)||this._defaults.dayNames,a=(i?i.monthNamesShort:null)||this._defaults.monthNamesShort,r=(i?i.monthNames:null)||this._defaults.monthNames,h=function(e){var i=t.length>s+1&&t.charAt(s+1)===e;return i&&s++,i},l=function(t,e,i){var s=""+e;if(h(t))for(;i>s.length;)s="0"+s;return s},c=function(t,e,i,s){return h(t)?s[e]:i[e]},u="",d=!1;if(e)for(s=0;t.length>s;s++)if(d)"'"!==t.charAt(s)||h("'")?u+=t.charAt(s):d=!1;else switch(t.charAt(s)){case"d":u+=l("d",e.getDate(),2);break;case"D":u+=c("D",e.getDay(),n,o);break;case"o":u+=l("o",Math.round((new Date(e.getFullYear(),e.getMonth(),e.getDate()).getTime()-new Date(e.getFullYear(),0,0).getTime())/864e5),3);break;case"m":u+=l("m",e.getMonth()+1,2);break;case"M":u+=c("M",e.getMonth(),a,r);break;case"y":u+=h("y")?e.getFullYear():(10>e.getFullYear()%100?"0":"")+e.getFullYear()%100;break;case"@":u+=e.getTime();break;case"!":u+=1e4*e.getTime()+this._ticksTo1970;break;case"'":h("'")?u+="'":d=!0;break;default:u+=t.charAt(s)}return u},_possibleChars:function(t){var e,i="",s=!1,n=function(i){var s=t.length>e+1&&t.charAt(e+1)===i;return s&&e++,s};for(e=0;t.length>e;e++)if(s)"'"!==t.charAt(e)||n("'")?i+=t.charAt(e):s=!1;else switch(t.charAt(e)){case"d":case"m":case"y":case"@":i+="0123456789";break;case"D":case"M":return null;case"'":n("'")?i+="'":s=!0;break;default:i+=t.charAt(e)}return i},_get:function(t,e){return void 0!==t.settings[e]?t.settings[e]:this._defaults[e]},_setDateFromField:function(t,e){if(t.input.val()!==t.lastVal){var i=this._get(t,"dateFormat"),s=t.lastVal=t.input?t.input.val():null,n=this._getDefaultDate(t),o=n,a=this._getFormatConfig(t);try{o=this.parseDate(i,s,a)||n}catch(r){s=e?"":s}t.selectedDay=o.getDate(),t.drawMonth=t.selectedMonth=o.getMonth(),t.drawYear=t.selectedYear=o.getFullYear(),t.currentDay=s?o.getDate():0,t.currentMonth=s?o.getMonth():0,t.currentYear=s?o.getFullYear():0,this._adjustInstDate(t)}},_getDefaultDate:function(t){return this._restrictMinMax(t,this._determineDate(t,this._get(t,"defaultDate"),new Date))},_determineDate:function(e,i,s){var n=function(t){var e=new Date;return e.setDate(e.getDate()+t),e},o=function(i){try{return t.datepicker.parseDate(t.datepicker._get(e,"dateFormat"),i,t.datepicker._getFormatConfig(e))}catch(s){}for(var n=(i.toLowerCase().match(/^c/)?t.datepicker._getDate(e):null)||new Date,o=n.getFullYear(),a=n.getMonth(),r=n.getDate(),h=/([+\-]?[0-9]+)\s*(d|D|w|W|m|M|y|Y)?/g,l=h.exec(i);l;){switch(l[2]||"d"){case"d":case"D":r+=parseInt(l[1],10);break;case"w":case"W":r+=7*parseInt(l[1],10);break;case"m":case"M":a+=parseInt(l[1],10),r=Math.min(r,t.datepicker._getDaysInMonth(o,a));break;case"y":case"Y":o+=parseInt(l[1],10),r=Math.min(r,t.datepicker._getDaysInMonth(o,a))}l=h.exec(i)}return new Date(o,a,r)},a=null==i||""===i?s:"string"==typeof i?o(i):"number"==typeof i?isNaN(i)?s:n(i):new Date(i.getTime());return a=a&&"Invalid Date"==""+a?s:a,a&&(a.setHours(0),a.setMinutes(0),a.setSeconds(0),a.setMilliseconds(0)),this._daylightSavingAdjust(a)},_daylightSavingAdjust:function(t){return t?(t.setHours(t.getHours()>12?t.getHours()+2:0),t):null},_setDate:function(t,e,i){var s=!e,n=t.selectedMonth,o=t.selectedYear,a=this._restrictMinMax(t,this._determineDate(t,e,new Date));t.selectedDay=t.currentDay=a.getDate(),t.drawMonth=t.selectedMonth=t.currentMonth=a.getMonth(),t.drawYear=t.selectedYear=t.currentYear=a.getFullYear(),n===t.selectedMonth&&o===t.selectedYear||i||this._notifyChange(t),this._adjustInstDate(t),t.input&&t.input.val(s?"":this._formatDate(t))},_getDate:function(t){var e=!t.currentYear||t.input&&""===t.input.val()?null:this._daylightSavingAdjust(new Date(t.currentYear,t.currentMonth,t.currentDay));return e},_attachHandlers:function(e){var i=this._get(e,"stepMonths"),s="#"+e.id.replace(/\\\\/g,"\\");e.dpDiv.find("[data-handler]").map(function(){var e={prev:function(){t.datepicker._adjustDate(s,-i,"M")},next:function(){t.datepicker._adjustDate(s,+i,"M")},hide:function(){t.datepicker._hideDatepicker()},today:function(){t.datepicker._gotoToday(s)},selectDay:function(){return t.datepicker._selectDay(s,+this.getAttribute("data-month"),+this.getAttribute("data-year"),this),!1},selectMonth:function(){return t.datepicker._selectMonthYear(s,this,"M"),!1},selectYear:function(){return t.datepicker._selectMonthYear(s,this,"Y"),!1}};t(this).on(this.getAttribute("data-event"),e[this.getAttribute("data-handler")])})},_generateHTML:function(t){var e,i,s,n,o,a,r,h,l,c,u,d,p,f,g,m,_,v,b,y,w,k,x,C,D,I,T,P,M,S,H,z,O,A,N,W,E,F,L,R=new Date,B=this._daylightSavingAdjust(new Date(R.getFullYear(),R.getMonth(),R.getDate())),Y=this._get(t,"isRTL"),j=this._get(t,"showButtonPanel"),q=this._get(t,"hideIfNoPrevNext"),K=this._get(t,"navigationAsDateFormat"),U=this._getNumberOfMonths(t),V=this._get(t,"showCurrentAtPos"),$=this._get(t,"stepMonths"),X=1!==U[0]||1!==U[1],G=this._daylightSavingAdjust(t.currentDay?new Date(t.currentYear,t.currentMonth,t.currentDay):new Date(9999,9,9)),Q=this._getMinMaxDate(t,"min"),J=this._getMinMaxDate(t,"max"),Z=t.drawMonth-V,te=t.drawYear;if(0>Z&&(Z+=12,te--),J)for(e=this._daylightSavingAdjust(new Date(J.getFullYear(),J.getMonth()-U[0]*U[1]+1,J.getDate())),e=Q&&Q>e?Q:e;this._daylightSavingAdjust(new Date(te,Z,1))>e;)Z--,0>Z&&(Z=11,te--);for(t.drawMonth=Z,t.drawYear=te,i=this._get(t,"prevText"),i=K?this.formatDate(i,this._daylightSavingAdjust(new Date(te,Z-$,1)),this._getFormatConfig(t)):i,s=this._canAdjustMonth(t,-1,te,Z)?"<a class='ui-datepicker-prev ui-corner-all' data-handler='prev' data-event='click' title='"+i+"'><span class='ui-icon ui-icon-circle-triangle-"+(Y?"e":"w")+"'>"+i+"</span></a>":q?"":"<a class='ui-datepicker-prev ui-corner-all ui-state-disabled' title='"+i+"'><span class='ui-icon ui-icon-circle-triangle-"+(Y?"e":"w")+"'>"+i+"</span></a>",n=this._get(t,"nextText"),n=K?this.formatDate(n,this._daylightSavingAdjust(new Date(te,Z+$,1)),this._getFormatConfig(t)):n,o=this._canAdjustMonth(t,1,te,Z)?"<a class='ui-datepicker-next ui-corner-all' data-handler='next' data-event='click' title='"+n+"'><span class='ui-icon ui-icon-circle-triangle-"+(Y?"w":"e")+"'>"+n+"</span></a>":q?"":"<a class='ui-datepicker-next ui-corner-all ui-state-disabled' title='"+n+"'><span class='ui-icon ui-icon-circle-triangle-"+(Y?"w":"e")+"'>"+n+"</span></a>",a=this._get(t,"currentText"),r=this._get(t,"gotoCurrent")&&t.currentDay?G:B,a=K?this.formatDate(a,r,this._getFormatConfig(t)):a,h=t.inline?"":"<button type='button' class='ui-datepicker-close ui-state-default ui-priority-primary ui-corner-all' data-handler='hide' data-event='click'>"+this._get(t,"closeText")+"</button>",l=j?"<div class='ui-datepicker-buttonpane ui-widget-content'>"+(Y?h:"")+(this._isInRange(t,r)?"<button type='button' class='ui-datepicker-current ui-state-default ui-priority-secondary ui-corner-all' data-handler='today' data-event='click'>"+a+"</button>":"")+(Y?"":h)+"</div>":"",c=parseInt(this._get(t,"firstDay"),10),c=isNaN(c)?0:c,u=this._get(t,"showWeek"),d=this._get(t,"dayNames"),p=this._get(t,"dayNamesMin"),f=this._get(t,"monthNames"),g=this._get(t,"monthNamesShort"),m=this._get(t,"beforeShowDay"),_=this._get(t,"showOtherMonths"),v=this._get(t,"selectOtherMonths"),b=this._getDefaultDate(t),y="",k=0;U[0]>k;k++){for(x="",this.maxRows=4,C=0;U[1]>C;C++){if(D=this._daylightSavingAdjust(new Date(te,Z,t.selectedDay)),I=" ui-corner-all",T="",X){if(T+="<div class='ui-datepicker-group",U[1]>1)switch(C){case 0:T+=" ui-datepicker-group-first",I=" ui-corner-"+(Y?"right":"left");break;case U[1]-1:T+=" ui-datepicker-group-last",I=" ui-corner-"+(Y?"left":"right");break;default:T+=" ui-datepicker-group-middle",I=""}T+="'>"}for(T+="<div class='ui-datepicker-header ui-widget-header ui-helper-clearfix"+I+"'>"+(/all|left/.test(I)&&0===k?Y?o:s:"")+(/all|right/.test(I)&&0===k?Y?s:o:"")+this._generateMonthYearHeader(t,Z,te,Q,J,k>0||C>0,f,g)+"</div><table class='ui-datepicker-calendar'><thead>"+"<tr>",P=u?"<th class='ui-datepicker-week-col'>"+this._get(t,"weekHeader")+"</th>":"",w=0;7>w;w++)M=(w+c)%7,P+="<th scope='col'"+((w+c+6)%7>=5?" class='ui-datepicker-week-end'":"")+">"+"<span title='"+d[M]+"'>"+p[M]+"</span></th>";for(T+=P+"</tr></thead><tbody>",S=this._getDaysInMonth(te,Z),te===t.selectedYear&&Z===t.selectedMonth&&(t.selectedDay=Math.min(t.selectedDay,S)),H=(this._getFirstDayOfMonth(te,Z)-c+7)%7,z=Math.ceil((H+S)/7),O=X?this.maxRows>z?this.maxRows:z:z,this.maxRows=O,A=this._daylightSavingAdjust(new Date(te,Z,1-H)),N=0;O>N;N++){for(T+="<tr>",W=u?"<td class='ui-datepicker-week-col'>"+this._get(t,"calculateWeek")(A)+"</td>":"",w=0;7>w;w++)E=m?m.apply(t.input?t.input[0]:null,[A]):[!0,""],F=A.getMonth()!==Z,L=F&&!v||!E[0]||Q&&Q>A||J&&A>J,W+="<td class='"+((w+c+6)%7>=5?" ui-datepicker-week-end":"")+(F?" ui-datepicker-other-month":"")+(A.getTime()===D.getTime()&&Z===t.selectedMonth&&t._keyEvent||b.getTime()===A.getTime()&&b.getTime()===D.getTime()?" "+this._dayOverClass:"")+(L?" "+this._unselectableClass+" ui-state-disabled":"")+(F&&!_?"":" "+E[1]+(A.getTime()===G.getTime()?" "+this._currentClass:"")+(A.getTime()===B.getTime()?" ui-datepicker-today":""))+"'"+(F&&!_||!E[2]?"":" title='"+E[2].replace(/'/g,"&#39;")+"'")+(L?"":" data-handler='selectDay' data-event='click' data-month='"+A.getMonth()+"' data-year='"+A.getFullYear()+"'")+">"+(F&&!_?"&#xa0;":L?"<span class='ui-state-default'>"+A.getDate()+"</span>":"<a class='ui-state-default"+(A.getTime()===B.getTime()?" ui-state-highlight":"")+(A.getTime()===G.getTime()?" ui-state-active":"")+(F?" ui-priority-secondary":"")+"' href='#'>"+A.getDate()+"</a>")+"</td>",A.setDate(A.getDate()+1),A=this._daylightSavingAdjust(A);T+=W+"</tr>"}Z++,Z>11&&(Z=0,te++),T+="</tbody></table>"+(X?"</div>"+(U[0]>0&&C===U[1]-1?"<div class='ui-datepicker-row-break'></div>":""):""),x+=T}y+=x}return y+=l,t._keyEvent=!1,y},_generateMonthYearHeader:function(t,e,i,s,n,o,a,r){var h,l,c,u,d,p,f,g,m=this._get(t,"changeMonth"),_=this._get(t,"changeYear"),v=this._get(t,"showMonthAfterYear"),b="<div class='ui-datepicker-title'>",y="";if(o||!m)y+="<span class='ui-datepicker-month'>"+a[e]+"</span>";else{for(h=s&&s.getFullYear()===i,l=n&&n.getFullYear()===i,y+="<select class='ui-datepicker-month' data-handler='selectMonth' data-event='change'>",c=0;12>c;c++)(!h||c>=s.getMonth())&&(!l||n.getMonth()>=c)&&(y+="<option value='"+c+"'"+(c===e?" selected='selected'":"")+">"+r[c]+"</option>");y+="</select>"}if(v||(b+=y+(!o&&m&&_?"":"&#xa0;")),!t.yearshtml)if(t.yearshtml="",o||!_)b+="<span class='ui-datepicker-year'>"+i+"</span>";else{for(u=this._get(t,"yearRange").split(":"),d=(new Date).getFullYear(),p=function(t){var e=t.match(/c[+\-].*/)?i+parseInt(t.substring(1),10):t.match(/[+\-].*/)?d+parseInt(t,10):parseInt(t,10);return isNaN(e)?d:e},f=p(u[0]),g=Math.max(f,p(u[1]||"")),f=s?Math.max(f,s.getFullYear()):f,g=n?Math.min(g,n.getFullYear()):g,t.yearshtml+="<select class='ui-datepicker-year' data-handler='selectYear' data-event='change'>";g>=f;f++)t.yearshtml+="<option value='"+f+"'"+(f===i?" selected='selected'":"")+">"+f+"</option>";t.yearshtml+="</select>",b+=t.yearshtml,t.yearshtml=null}return b+=this._get(t,"yearSuffix"),v&&(b+=(!o&&m&&_?"":"&#xa0;")+y),b+="</div>"},_adjustInstDate:function(t,e,i){var s=t.selectedYear+("Y"===i?e:0),n=t.selectedMonth+("M"===i?e:0),o=Math.min(t.selectedDay,this._getDaysInMonth(s,n))+("D"===i?e:0),a=this._restrictMinMax(t,this._daylightSavingAdjust(new Date(s,n,o)));t.selectedDay=a.getDate(),t.drawMonth=t.selectedMonth=a.getMonth(),t.drawYear=t.selectedYear=a.getFullYear(),("M"===i||"Y"===i)&&this._notifyChange(t)},_restrictMinMax:function(t,e){var i=this._getMinMaxDate(t,"min"),s=this._getMinMaxDate(t,"max"),n=i&&i>e?i:e;return s&&n>s?s:n},_notifyChange:function(t){var e=this._get(t,"onChangeMonthYear");e&&e.apply(t.input?t.input[0]:null,[t.selectedYear,t.selectedMonth+1,t])},_getNumberOfMonths:function(t){var e=this._get(t,"numberOfMonths");return null==e?[1,1]:"number"==typeof e?[1,e]:e},_getMinMaxDate:function(t,e){return this._determineDate(t,this._get(t,e+"Date"),null)},_getDaysInMonth:function(t,e){return 32-this._daylightSavingAdjust(new Date(t,e,32)).getDate()},_getFirstDayOfMonth:function(t,e){return new Date(t,e,1).getDay()},_canAdjustMonth:function(t,e,i,s){var n=this._getNumberOfMonths(t),o=this._daylightSavingAdjust(new Date(i,s+(0>e?e:n[0]*n[1]),1));return 0>e&&o.setDate(this._getDaysInMonth(o.getFullYear(),o.getMonth())),this._isInRange(t,o)},_isInRange:function(t,e){var i,s,n=this._getMinMaxDate(t,"min"),o=this._getMinMaxDate(t,"max"),a=null,r=null,h=this._get(t,"yearRange");return h&&(i=h.split(":"),s=(new Date).getFullYear(),a=parseInt(i[0],10),r=parseInt(i[1],10),i[0].match(/[+\-].*/)&&(a+=s),i[1].match(/[+\-].*/)&&(r+=s)),(!n||e.getTime()>=n.getTime())&&(!o||e.getTime()<=o.getTime())&&(!a||e.getFullYear()>=a)&&(!r||r>=e.getFullYear())},_getFormatConfig:function(t){var e=this._get(t,"shortYearCutoff");return e="string"!=typeof e?e:(new Date).getFullYear()%100+parseInt(e,10),{shortYearCutoff:e,dayNamesShort:this._get(t,"dayNamesShort"),dayNames:this._get(t,"dayNames"),monthNamesShort:this._get(t,"monthNamesShort"),monthNames:this._get(t,"monthNames")}},_formatDate:function(t,e,i,s){e||(t.currentDay=t.selectedDay,t.currentMonth=t.selectedMonth,t.currentYear=t.selectedYear);var n=e?"object"==typeof e?e:this._daylightSavingAdjust(new Date(s,i,e)):this._daylightSavingAdjust(new Date(t.currentYear,t.currentMonth,t.currentDay));return this.formatDate(this._get(t,"dateFormat"),n,this._getFormatConfig(t))}}),t.fn.datepicker=function(e){if(!this.length)return this;t.datepicker.initialized||(t(document).on("mousedown",t.datepicker._checkExternalClick),t.datepicker.initialized=!0),0===t("#"+t.datepicker._mainDivId).length&&t("body").append(t.datepicker.dpDiv);var i=Array.prototype.slice.call(arguments,1);return"string"!=typeof e||"isDisabled"!==e&&"getDate"!==e&&"widget"!==e?"option"===e&&2===arguments.length&&"string"==typeof arguments[1]?t.datepicker["_"+e+"Datepicker"].apply(t.datepicker,[this[0]].concat(i)):this.each(function(){"string"==typeof e?t.datepicker["_"+e+"Datepicker"].apply(t.datepicker,[this].concat(i)):t.datepicker._attachDatepicker(this,e)}):t.datepicker["_"+e+"Datepicker"].apply(t.datepicker,[this[0]].concat(i))},t.datepicker=new s,t.datepicker.initialized=!1,t.datepicker.uuid=(new Date).getTime(),t.datepicker.version="1.12.1",t.datepicker,t.ui.ie=!!/msie [\w.]+/.exec(navigator.userAgent.toLowerCase());var _=!1;t(document).on("mouseup",function(){_=!1}),t.widget("ui.mouse",{version:"1.12.1",options:{cancel:"input, textarea, button, select, option",distance:1,delay:0},_mouseInit:function(){var e=this;this.element.on("mousedown."+this.widgetName,function(t){return e._mouseDown(t)}).on("click."+this.widgetName,function(i){return!0===t.data(i.target,e.widgetName+".preventClickEvent")?(t.removeData(i.target,e.widgetName+".preventClickEvent"),i.stopImmediatePropagation(),!1):void 0}),this.started=!1},_mouseDestroy:function(){this.element.off("."+this.widgetName),this._mouseMoveDelegate&&this.document.off("mousemove."+this.widgetName,this._mouseMoveDelegate).off("mouseup."+this.widgetName,this._mouseUpDelegate)},_mouseDown:function(e){if(!_){this._mouseMoved=!1,this._mouseStarted&&this._mouseUp(e),this._mouseDownEvent=e;var i=this,s=1===e.which,n="string"==typeof this.options.cancel&&e.target.nodeName?t(e.target).closest(this.options.cancel).length:!1;return s&&!n&&this._mouseCapture(e)?(this.mouseDelayMet=!this.options.delay,this.mouseDelayMet||(this._mouseDelayTimer=setTimeout(function(){i.mouseDelayMet=!0},this.options.delay)),this._mouseDistanceMet(e)&&this._mouseDelayMet(e)&&(this._mouseStarted=this._mouseStart(e)!==!1,!this._mouseStarted)?(e.preventDefault(),!0):(!0===t.data(e.target,this.widgetName+".preventClickEvent")&&t.removeData(e.target,this.widgetName+".preventClickEvent"),this._mouseMoveDelegate=function(t){return i._mouseMove(t)},this._mouseUpDelegate=function(t){return i._mouseUp(t)},this.document.on("mousemove."+this.widgetName,this._mouseMoveDelegate).on("mouseup."+this.widgetName,this._mouseUpDelegate),e.preventDefault(),_=!0,!0)):!0}},_mouseMove:function(e){if(this._mouseMoved){if(t.ui.ie&&(!document.documentMode||9>document.documentMode)&&!e.button)return this._mouseUp(e);if(!e.which)if(e.originalEvent.altKey||e.originalEvent.ctrlKey||e.originalEvent.metaKey||e.originalEvent.shiftKey)this.ignoreMissingWhich=!0;else if(!this.ignoreMissingWhich)return this._mouseUp(e)}return(e.which||e.button)&&(this._mouseMoved=!0),this._mouseStarted?(this._mouseDrag(e),e.preventDefault()):(this._mouseDistanceMet(e)&&this._mouseDelayMet(e)&&(this._mouseStarted=this._mouseStart(this._mouseDownEvent,e)!==!1,this._mouseStarted?this._mouseDrag(e):this._mouseUp(e)),!this._mouseStarted)},_mouseUp:function(e){this.document.off("mousemove."+this.widgetName,this._mouseMoveDelegate).off("mouseup."+this.widgetName,this._mouseUpDelegate),this._mouseStarted&&(this._mouseStarted=!1,e.target===this._mouseDownEvent.target&&t.data(e.target,this.widgetName+".preventClickEvent",!0),this._mouseStop(e)),this._mouseDelayTimer&&(clearTimeout(this._mouseDelayTimer),delete this._mouseDelayTimer),this.ignoreMissingWhich=!1,_=!1,e.preventDefault()},_mouseDistanceMet:function(t){return Math.max(Math.abs(this._mouseDownEvent.pageX-t.pageX),Math.abs(this._mouseDownEvent.pageY-t.pageY))>=this.options.distance},_mouseDelayMet:function(){return this.mouseDelayMet},_mouseStart:function(){},_mouseDrag:function(){},_mouseStop:function(){},_mouseCapture:function(){return!0}}),t.ui.plugin={add:function(e,i,s){var n,o=t.ui[e].prototype;for(n in s)o.plugins[n]=o.plugins[n]||[],o.plugins[n].push([i,s[n]])},call:function(t,e,i,s){var n,o=t.plugins[e];if(o&&(s||t.element[0].parentNode&&11!==t.element[0].parentNode.nodeType))for(n=0;o.length>n;n++)t.options[o[n][0]]&&o[n][1].apply(t.element,i)}},t.ui.safeBlur=function(e){e&&"body"!==e.nodeName.toLowerCase()&&t(e).trigger("blur")},t.widget("ui.draggable",t.ui.mouse,{version:"1.12.1",widgetEventPrefix:"drag",options:{addClasses:!0,appendTo:"parent",axis:!1,connectToSortable:!1,containment:!1,cursor:"auto",cursorAt:!1,grid:!1,handle:!1,helper:"original",iframeFix:!1,opacity:!1,refreshPositions:!1,revert:!1,revertDuration:500,scope:"default",scroll:!0,scrollSensitivity:20,scrollSpeed:20,snap:!1,snapMode:"both",snapTolerance:20,stack:!1,zIndex:!1,drag:null,start:null,stop:null},_create:function(){"original"===this.options.helper&&this._setPositionRelative(),this.options.addClasses&&this._addClass("ui-draggable"),this._setHandleClassName(),this._mouseInit()},_setOption:function(t,e){this._super(t,e),"handle"===t&&(this._removeHandleClassName(),this._setHandleClassName())},_destroy:function(){return(this.helper||this.element).is(".ui-draggable-dragging")?(this.destroyOnClear=!0,void 0):(this._removeHandleClassName(),this._mouseDestroy(),void 0)},_mouseCapture:function(e){var i=this.options;return this.helper||i.disabled||t(e.target).closest(".ui-resizable-handle").length>0?!1:(this.handle=this._getHandle(e),this.handle?(this._blurActiveElement(e),this._blockFrames(i.iframeFix===!0?"iframe":i.iframeFix),!0):!1)},_blockFrames:function(e){this.iframeBlocks=this.document.find(e).map(function(){var e=t(this);return t("<div>").css("position","absolute").appendTo(e.parent()).outerWidth(e.outerWidth()).outerHeight(e.outerHeight()).offset(e.offset())[0]})},_unblockFrames:function(){this.iframeBlocks&&(this.iframeBlocks.remove(),delete this.iframeBlocks)},_blurActiveElement:function(e){var i=t.ui.safeActiveElement(this.document[0]),s=t(e.target);s.closest(i).length||t.ui.safeBlur(i)},_mouseStart:function(e){var i=this.options;return this.helper=this._createHelper(e),this._addClass(this.helper,"ui-draggable-dragging"),this._cacheHelperProportions(),t.ui.ddmanager&&(t.ui.ddmanager.current=this),this._cacheMargins(),this.cssPosition=this.helper.css("position"),this.scrollParent=this.helper.scrollParent(!0),this.offsetParent=this.helper.offsetParent(),this.hasFixedAncestor=this.helper.parents().filter(function(){return"fixed"===t(this).css("position")}).length>0,this.positionAbs=this.element.offset(),this._refreshOffsets(e),this.originalPosition=this.position=this._generatePosition(e,!1),this.originalPageX=e.pageX,this.originalPageY=e.pageY,i.cursorAt&&this._adjustOffsetFromHelper(i.cursorAt),this._setContainment(),this._trigger("start",e)===!1?(this._clear(),!1):(this._cacheHelperProportions(),t.ui.ddmanager&&!i.dropBehaviour&&t.ui.ddmanager.prepareOffsets(this,e),this._mouseDrag(e,!0),t.ui.ddmanager&&t.ui.ddmanager.dragStart(this,e),!0)},_refreshOffsets:function(t){this.offset={top:this.positionAbs.top-this.margins.top,left:this.positionAbs.left-this.margins.left,scroll:!1,parent:this._getParentOffset(),relative:this._getRelativeOffset()},this.offset.click={left:t.pageX-this.offset.left,top:t.pageY-this.offset.top}},_mouseDrag:function(e,i){if(this.hasFixedAncestor&&(this.offset.parent=this._getParentOffset()),this.position=this._generatePosition(e,!0),this.positionAbs=this._convertPositionTo("absolute"),!i){var s=this._uiHash();if(this._trigger("drag",e,s)===!1)return this._mouseUp(new t.Event("mouseup",e)),!1;this.position=s.position}return this.helper[0].style.left=this.position.left+"px",this.helper[0].style.top=this.position.top+"px",t.ui.ddmanager&&t.ui.ddmanager.drag(this,e),!1},_mouseStop:function(e){var i=this,s=!1;return t.ui.ddmanager&&!this.options.dropBehaviour&&(s=t.ui.ddmanager.drop(this,e)),this.dropped&&(s=this.dropped,this.dropped=!1),"invalid"===this.options.revert&&!s||"valid"===this.options.revert&&s||this.options.revert===!0||t.isFunction(this.options.revert)&&this.options.revert.call(this.element,s)?t(this.helper).animate(this.originalPosition,parseInt(this.options.revertDuration,10),function(){i._trigger("stop",e)!==!1&&i._clear()}):this._trigger("stop",e)!==!1&&this._clear(),!1},_mouseUp:function(e){return this._unblockFrames(),t.ui.ddmanager&&t.ui.ddmanager.dragStop(this,e),this.handleElement.is(e.target)&&this.element.trigger("focus"),t.ui.mouse.prototype._mouseUp.call(this,e)},cancel:function(){return this.helper.is(".ui-draggable-dragging")?this._mouseUp(new t.Event("mouseup",{target:this.element[0]})):this._clear(),this},_getHandle:function(e){return this.options.handle?!!t(e.target).closest(this.element.find(this.options.handle)).length:!0},_setHandleClassName:function(){this.handleElement=this.options.handle?this.element.find(this.options.handle):this.element,this._addClass(this.handleElement,"ui-draggable-handle")},_removeHandleClassName:function(){this._removeClass(this.handleElement,"ui-draggable-handle")},_createHelper:function(e){var i=this.options,s=t.isFunction(i.helper),n=s?t(i.helper.apply(this.element[0],[e])):"clone"===i.helper?this.element.clone().removeAttr("id"):this.element;return n.parents("body").length||n.appendTo("parent"===i.appendTo?this.element[0].parentNode:i.appendTo),s&&n[0]===this.element[0]&&this._setPositionRelative(),n[0]===this.element[0]||/(fixed|absolute)/.test(n.css("position"))||n.css("position","absolute"),n},_setPositionRelative:function(){/^(?:r|a|f)/.test(this.element.css("position"))||(this.element[0].style.position="relative")},_adjustOffsetFromHelper:function(e){"string"==typeof e&&(e=e.split(" ")),t.isArray(e)&&(e={left:+e[0],top:+e[1]||0}),"left"in e&&(this.offset.click.left=e.left+this.margins.left),"right"in e&&(this.offset.click.left=this.helperProportions.width-e.right+this.margins.left),"top"in e&&(this.offset.click.top=e.top+this.margins.top),"bottom"in e&&(this.offset.click.top=this.helperProportions.height-e.bottom+this.margins.top)},_isRootNode:function(t){return/(html|body)/i.test(t.tagName)||t===this.document[0]},_getParentOffset:function(){var e=this.offsetParent.offset(),i=this.document[0];return"absolute"===this.cssPosition&&this.scrollParent[0]!==i&&t.contains(this.scrollParent[0],this.offsetParent[0])&&(e.left+=this.scrollParent.scrollLeft(),e.top+=this.scrollParent.scrollTop()),this._isRootNode(this.offsetParent[0])&&(e={top:0,left:0}),{top:e.top+(parseInt(this.offsetParent.css("borderTopWidth"),10)||0),left:e.left+(parseInt(this.offsetParent.css("borderLeftWidth"),10)||0)}},_getRelativeOffset:function(){if("relative"!==this.cssPosition)return{top:0,left:0};var t=this.element.position(),e=this._isRootNode(this.scrollParent[0]);return{top:t.top-(parseInt(this.helper.css("top"),10)||0)+(e?0:this.scrollParent.scrollTop()),left:t.left-(parseInt(this.helper.css("left"),10)||0)+(e?0:this.scrollParent.scrollLeft())} -},_cacheMargins:function(){this.margins={left:parseInt(this.element.css("marginLeft"),10)||0,top:parseInt(this.element.css("marginTop"),10)||0,right:parseInt(this.element.css("marginRight"),10)||0,bottom:parseInt(this.element.css("marginBottom"),10)||0}},_cacheHelperProportions:function(){this.helperProportions={width:this.helper.outerWidth(),height:this.helper.outerHeight()}},_setContainment:function(){var e,i,s,n=this.options,o=this.document[0];return this.relativeContainer=null,n.containment?"window"===n.containment?(this.containment=[t(window).scrollLeft()-this.offset.relative.left-this.offset.parent.left,t(window).scrollTop()-this.offset.relative.top-this.offset.parent.top,t(window).scrollLeft()+t(window).width()-this.helperProportions.width-this.margins.left,t(window).scrollTop()+(t(window).height()||o.body.parentNode.scrollHeight)-this.helperProportions.height-this.margins.top],void 0):"document"===n.containment?(this.containment=[0,0,t(o).width()-this.helperProportions.width-this.margins.left,(t(o).height()||o.body.parentNode.scrollHeight)-this.helperProportions.height-this.margins.top],void 0):n.containment.constructor===Array?(this.containment=n.containment,void 0):("parent"===n.containment&&(n.containment=this.helper[0].parentNode),i=t(n.containment),s=i[0],s&&(e=/(scroll|auto)/.test(i.css("overflow")),this.containment=[(parseInt(i.css("borderLeftWidth"),10)||0)+(parseInt(i.css("paddingLeft"),10)||0),(parseInt(i.css("borderTopWidth"),10)||0)+(parseInt(i.css("paddingTop"),10)||0),(e?Math.max(s.scrollWidth,s.offsetWidth):s.offsetWidth)-(parseInt(i.css("borderRightWidth"),10)||0)-(parseInt(i.css("paddingRight"),10)||0)-this.helperProportions.width-this.margins.left-this.margins.right,(e?Math.max(s.scrollHeight,s.offsetHeight):s.offsetHeight)-(parseInt(i.css("borderBottomWidth"),10)||0)-(parseInt(i.css("paddingBottom"),10)||0)-this.helperProportions.height-this.margins.top-this.margins.bottom],this.relativeContainer=i),void 0):(this.containment=null,void 0)},_convertPositionTo:function(t,e){e||(e=this.position);var i="absolute"===t?1:-1,s=this._isRootNode(this.scrollParent[0]);return{top:e.top+this.offset.relative.top*i+this.offset.parent.top*i-("fixed"===this.cssPosition?-this.offset.scroll.top:s?0:this.offset.scroll.top)*i,left:e.left+this.offset.relative.left*i+this.offset.parent.left*i-("fixed"===this.cssPosition?-this.offset.scroll.left:s?0:this.offset.scroll.left)*i}},_generatePosition:function(t,e){var i,s,n,o,a=this.options,r=this._isRootNode(this.scrollParent[0]),h=t.pageX,l=t.pageY;return r&&this.offset.scroll||(this.offset.scroll={top:this.scrollParent.scrollTop(),left:this.scrollParent.scrollLeft()}),e&&(this.containment&&(this.relativeContainer?(s=this.relativeContainer.offset(),i=[this.containment[0]+s.left,this.containment[1]+s.top,this.containment[2]+s.left,this.containment[3]+s.top]):i=this.containment,t.pageX-this.offset.click.left<i[0]&&(h=i[0]+this.offset.click.left),t.pageY-this.offset.click.top<i[1]&&(l=i[1]+this.offset.click.top),t.pageX-this.offset.click.left>i[2]&&(h=i[2]+this.offset.click.left),t.pageY-this.offset.click.top>i[3]&&(l=i[3]+this.offset.click.top)),a.grid&&(n=a.grid[1]?this.originalPageY+Math.round((l-this.originalPageY)/a.grid[1])*a.grid[1]:this.originalPageY,l=i?n-this.offset.click.top>=i[1]||n-this.offset.click.top>i[3]?n:n-this.offset.click.top>=i[1]?n-a.grid[1]:n+a.grid[1]:n,o=a.grid[0]?this.originalPageX+Math.round((h-this.originalPageX)/a.grid[0])*a.grid[0]:this.originalPageX,h=i?o-this.offset.click.left>=i[0]||o-this.offset.click.left>i[2]?o:o-this.offset.click.left>=i[0]?o-a.grid[0]:o+a.grid[0]:o),"y"===a.axis&&(h=this.originalPageX),"x"===a.axis&&(l=this.originalPageY)),{top:l-this.offset.click.top-this.offset.relative.top-this.offset.parent.top+("fixed"===this.cssPosition?-this.offset.scroll.top:r?0:this.offset.scroll.top),left:h-this.offset.click.left-this.offset.relative.left-this.offset.parent.left+("fixed"===this.cssPosition?-this.offset.scroll.left:r?0:this.offset.scroll.left)}},_clear:function(){this._removeClass(this.helper,"ui-draggable-dragging"),this.helper[0]===this.element[0]||this.cancelHelperRemoval||this.helper.remove(),this.helper=null,this.cancelHelperRemoval=!1,this.destroyOnClear&&this.destroy()},_trigger:function(e,i,s){return s=s||this._uiHash(),t.ui.plugin.call(this,e,[i,s,this],!0),/^(drag|start|stop)/.test(e)&&(this.positionAbs=this._convertPositionTo("absolute"),s.offset=this.positionAbs),t.Widget.prototype._trigger.call(this,e,i,s)},plugins:{},_uiHash:function(){return{helper:this.helper,position:this.position,originalPosition:this.originalPosition,offset:this.positionAbs}}}),t.ui.plugin.add("draggable","connectToSortable",{start:function(e,i,s){var n=t.extend({},i,{item:s.element});s.sortables=[],t(s.options.connectToSortable).each(function(){var i=t(this).sortable("instance");i&&!i.options.disabled&&(s.sortables.push(i),i.refreshPositions(),i._trigger("activate",e,n))})},stop:function(e,i,s){var n=t.extend({},i,{item:s.element});s.cancelHelperRemoval=!1,t.each(s.sortables,function(){var t=this;t.isOver?(t.isOver=0,s.cancelHelperRemoval=!0,t.cancelHelperRemoval=!1,t._storedCSS={position:t.placeholder.css("position"),top:t.placeholder.css("top"),left:t.placeholder.css("left")},t._mouseStop(e),t.options.helper=t.options._helper):(t.cancelHelperRemoval=!0,t._trigger("deactivate",e,n))})},drag:function(e,i,s){t.each(s.sortables,function(){var n=!1,o=this;o.positionAbs=s.positionAbs,o.helperProportions=s.helperProportions,o.offset.click=s.offset.click,o._intersectsWith(o.containerCache)&&(n=!0,t.each(s.sortables,function(){return this.positionAbs=s.positionAbs,this.helperProportions=s.helperProportions,this.offset.click=s.offset.click,this!==o&&this._intersectsWith(this.containerCache)&&t.contains(o.element[0],this.element[0])&&(n=!1),n})),n?(o.isOver||(o.isOver=1,s._parent=i.helper.parent(),o.currentItem=i.helper.appendTo(o.element).data("ui-sortable-item",!0),o.options._helper=o.options.helper,o.options.helper=function(){return i.helper[0]},e.target=o.currentItem[0],o._mouseCapture(e,!0),o._mouseStart(e,!0,!0),o.offset.click.top=s.offset.click.top,o.offset.click.left=s.offset.click.left,o.offset.parent.left-=s.offset.parent.left-o.offset.parent.left,o.offset.parent.top-=s.offset.parent.top-o.offset.parent.top,s._trigger("toSortable",e),s.dropped=o.element,t.each(s.sortables,function(){this.refreshPositions()}),s.currentItem=s.element,o.fromOutside=s),o.currentItem&&(o._mouseDrag(e),i.position=o.position)):o.isOver&&(o.isOver=0,o.cancelHelperRemoval=!0,o.options._revert=o.options.revert,o.options.revert=!1,o._trigger("out",e,o._uiHash(o)),o._mouseStop(e,!0),o.options.revert=o.options._revert,o.options.helper=o.options._helper,o.placeholder&&o.placeholder.remove(),i.helper.appendTo(s._parent),s._refreshOffsets(e),i.position=s._generatePosition(e,!0),s._trigger("fromSortable",e),s.dropped=!1,t.each(s.sortables,function(){this.refreshPositions()}))})}}),t.ui.plugin.add("draggable","cursor",{start:function(e,i,s){var n=t("body"),o=s.options;n.css("cursor")&&(o._cursor=n.css("cursor")),n.css("cursor",o.cursor)},stop:function(e,i,s){var n=s.options;n._cursor&&t("body").css("cursor",n._cursor)}}),t.ui.plugin.add("draggable","opacity",{start:function(e,i,s){var n=t(i.helper),o=s.options;n.css("opacity")&&(o._opacity=n.css("opacity")),n.css("opacity",o.opacity)},stop:function(e,i,s){var n=s.options;n._opacity&&t(i.helper).css("opacity",n._opacity)}}),t.ui.plugin.add("draggable","scroll",{start:function(t,e,i){i.scrollParentNotHidden||(i.scrollParentNotHidden=i.helper.scrollParent(!1)),i.scrollParentNotHidden[0]!==i.document[0]&&"HTML"!==i.scrollParentNotHidden[0].tagName&&(i.overflowOffset=i.scrollParentNotHidden.offset())},drag:function(e,i,s){var n=s.options,o=!1,a=s.scrollParentNotHidden[0],r=s.document[0];a!==r&&"HTML"!==a.tagName?(n.axis&&"x"===n.axis||(s.overflowOffset.top+a.offsetHeight-e.pageY<n.scrollSensitivity?a.scrollTop=o=a.scrollTop+n.scrollSpeed:e.pageY-s.overflowOffset.top<n.scrollSensitivity&&(a.scrollTop=o=a.scrollTop-n.scrollSpeed)),n.axis&&"y"===n.axis||(s.overflowOffset.left+a.offsetWidth-e.pageX<n.scrollSensitivity?a.scrollLeft=o=a.scrollLeft+n.scrollSpeed:e.pageX-s.overflowOffset.left<n.scrollSensitivity&&(a.scrollLeft=o=a.scrollLeft-n.scrollSpeed))):(n.axis&&"x"===n.axis||(e.pageY-t(r).scrollTop()<n.scrollSensitivity?o=t(r).scrollTop(t(r).scrollTop()-n.scrollSpeed):t(window).height()-(e.pageY-t(r).scrollTop())<n.scrollSensitivity&&(o=t(r).scrollTop(t(r).scrollTop()+n.scrollSpeed))),n.axis&&"y"===n.axis||(e.pageX-t(r).scrollLeft()<n.scrollSensitivity?o=t(r).scrollLeft(t(r).scrollLeft()-n.scrollSpeed):t(window).width()-(e.pageX-t(r).scrollLeft())<n.scrollSensitivity&&(o=t(r).scrollLeft(t(r).scrollLeft()+n.scrollSpeed)))),o!==!1&&t.ui.ddmanager&&!n.dropBehaviour&&t.ui.ddmanager.prepareOffsets(s,e)}}),t.ui.plugin.add("draggable","snap",{start:function(e,i,s){var n=s.options;s.snapElements=[],t(n.snap.constructor!==String?n.snap.items||":data(ui-draggable)":n.snap).each(function(){var e=t(this),i=e.offset();this!==s.element[0]&&s.snapElements.push({item:this,width:e.outerWidth(),height:e.outerHeight(),top:i.top,left:i.left})})},drag:function(e,i,s){var n,o,a,r,h,l,c,u,d,p,f=s.options,g=f.snapTolerance,m=i.offset.left,_=m+s.helperProportions.width,v=i.offset.top,b=v+s.helperProportions.height;for(d=s.snapElements.length-1;d>=0;d--)h=s.snapElements[d].left-s.margins.left,l=h+s.snapElements[d].width,c=s.snapElements[d].top-s.margins.top,u=c+s.snapElements[d].height,h-g>_||m>l+g||c-g>b||v>u+g||!t.contains(s.snapElements[d].item.ownerDocument,s.snapElements[d].item)?(s.snapElements[d].snapping&&s.options.snap.release&&s.options.snap.release.call(s.element,e,t.extend(s._uiHash(),{snapItem:s.snapElements[d].item})),s.snapElements[d].snapping=!1):("inner"!==f.snapMode&&(n=g>=Math.abs(c-b),o=g>=Math.abs(u-v),a=g>=Math.abs(h-_),r=g>=Math.abs(l-m),n&&(i.position.top=s._convertPositionTo("relative",{top:c-s.helperProportions.height,left:0}).top),o&&(i.position.top=s._convertPositionTo("relative",{top:u,left:0}).top),a&&(i.position.left=s._convertPositionTo("relative",{top:0,left:h-s.helperProportions.width}).left),r&&(i.position.left=s._convertPositionTo("relative",{top:0,left:l}).left)),p=n||o||a||r,"outer"!==f.snapMode&&(n=g>=Math.abs(c-v),o=g>=Math.abs(u-b),a=g>=Math.abs(h-m),r=g>=Math.abs(l-_),n&&(i.position.top=s._convertPositionTo("relative",{top:c,left:0}).top),o&&(i.position.top=s._convertPositionTo("relative",{top:u-s.helperProportions.height,left:0}).top),a&&(i.position.left=s._convertPositionTo("relative",{top:0,left:h}).left),r&&(i.position.left=s._convertPositionTo("relative",{top:0,left:l-s.helperProportions.width}).left)),!s.snapElements[d].snapping&&(n||o||a||r||p)&&s.options.snap.snap&&s.options.snap.snap.call(s.element,e,t.extend(s._uiHash(),{snapItem:s.snapElements[d].item})),s.snapElements[d].snapping=n||o||a||r||p)}}),t.ui.plugin.add("draggable","stack",{start:function(e,i,s){var n,o=s.options,a=t.makeArray(t(o.stack)).sort(function(e,i){return(parseInt(t(e).css("zIndex"),10)||0)-(parseInt(t(i).css("zIndex"),10)||0)});a.length&&(n=parseInt(t(a[0]).css("zIndex"),10)||0,t(a).each(function(e){t(this).css("zIndex",n+e)}),this.css("zIndex",n+a.length))}}),t.ui.plugin.add("draggable","zIndex",{start:function(e,i,s){var n=t(i.helper),o=s.options;n.css("zIndex")&&(o._zIndex=n.css("zIndex")),n.css("zIndex",o.zIndex)},stop:function(e,i,s){var n=s.options;n._zIndex&&t(i.helper).css("zIndex",n._zIndex)}}),t.ui.draggable,t.widget("ui.resizable",t.ui.mouse,{version:"1.12.1",widgetEventPrefix:"resize",options:{alsoResize:!1,animate:!1,animateDuration:"slow",animateEasing:"swing",aspectRatio:!1,autoHide:!1,classes:{"ui-resizable-se":"ui-icon ui-icon-gripsmall-diagonal-se"},containment:!1,ghost:!1,grid:!1,handles:"e,s,se",helper:!1,maxHeight:null,maxWidth:null,minHeight:10,minWidth:10,zIndex:90,resize:null,start:null,stop:null},_num:function(t){return parseFloat(t)||0},_isNumber:function(t){return!isNaN(parseFloat(t))},_hasScroll:function(e,i){if("hidden"===t(e).css("overflow"))return!1;var s=i&&"left"===i?"scrollLeft":"scrollTop",n=!1;return e[s]>0?!0:(e[s]=1,n=e[s]>0,e[s]=0,n)},_create:function(){var e,i=this.options,s=this;this._addClass("ui-resizable"),t.extend(this,{_aspectRatio:!!i.aspectRatio,aspectRatio:i.aspectRatio,originalElement:this.element,_proportionallyResizeElements:[],_helper:i.helper||i.ghost||i.animate?i.helper||"ui-resizable-helper":null}),this.element[0].nodeName.match(/^(canvas|textarea|input|select|button|img)$/i)&&(this.element.wrap(t("<div class='ui-wrapper' style='overflow: hidden;'></div>").css({position:this.element.css("position"),width:this.element.outerWidth(),height:this.element.outerHeight(),top:this.element.css("top"),left:this.element.css("left")})),this.element=this.element.parent().data("ui-resizable",this.element.resizable("instance")),this.elementIsWrapper=!0,e={marginTop:this.originalElement.css("marginTop"),marginRight:this.originalElement.css("marginRight"),marginBottom:this.originalElement.css("marginBottom"),marginLeft:this.originalElement.css("marginLeft")},this.element.css(e),this.originalElement.css("margin",0),this.originalResizeStyle=this.originalElement.css("resize"),this.originalElement.css("resize","none"),this._proportionallyResizeElements.push(this.originalElement.css({position:"static",zoom:1,display:"block"})),this.originalElement.css(e),this._proportionallyResize()),this._setupHandles(),i.autoHide&&t(this.element).on("mouseenter",function(){i.disabled||(s._removeClass("ui-resizable-autohide"),s._handles.show())}).on("mouseleave",function(){i.disabled||s.resizing||(s._addClass("ui-resizable-autohide"),s._handles.hide())}),this._mouseInit()},_destroy:function(){this._mouseDestroy();var e,i=function(e){t(e).removeData("resizable").removeData("ui-resizable").off(".resizable").find(".ui-resizable-handle").remove()};return this.elementIsWrapper&&(i(this.element),e=this.element,this.originalElement.css({position:e.css("position"),width:e.outerWidth(),height:e.outerHeight(),top:e.css("top"),left:e.css("left")}).insertAfter(e),e.remove()),this.originalElement.css("resize",this.originalResizeStyle),i(this.originalElement),this},_setOption:function(t,e){switch(this._super(t,e),t){case"handles":this._removeHandles(),this._setupHandles();break;default:}},_setupHandles:function(){var e,i,s,n,o,a=this.options,r=this;if(this.handles=a.handles||(t(".ui-resizable-handle",this.element).length?{n:".ui-resizable-n",e:".ui-resizable-e",s:".ui-resizable-s",w:".ui-resizable-w",se:".ui-resizable-se",sw:".ui-resizable-sw",ne:".ui-resizable-ne",nw:".ui-resizable-nw"}:"e,s,se"),this._handles=t(),this.handles.constructor===String)for("all"===this.handles&&(this.handles="n,e,s,w,se,sw,ne,nw"),s=this.handles.split(","),this.handles={},i=0;s.length>i;i++)e=t.trim(s[i]),n="ui-resizable-"+e,o=t("<div>"),this._addClass(o,"ui-resizable-handle "+n),o.css({zIndex:a.zIndex}),this.handles[e]=".ui-resizable-"+e,this.element.append(o);this._renderAxis=function(e){var i,s,n,o;e=e||this.element;for(i in this.handles)this.handles[i].constructor===String?this.handles[i]=this.element.children(this.handles[i]).first().show():(this.handles[i].jquery||this.handles[i].nodeType)&&(this.handles[i]=t(this.handles[i]),this._on(this.handles[i],{mousedown:r._mouseDown})),this.elementIsWrapper&&this.originalElement[0].nodeName.match(/^(textarea|input|select|button)$/i)&&(s=t(this.handles[i],this.element),o=/sw|ne|nw|se|n|s/.test(i)?s.outerHeight():s.outerWidth(),n=["padding",/ne|nw|n/.test(i)?"Top":/se|sw|s/.test(i)?"Bottom":/^e$/.test(i)?"Right":"Left"].join(""),e.css(n,o),this._proportionallyResize()),this._handles=this._handles.add(this.handles[i])},this._renderAxis(this.element),this._handles=this._handles.add(this.element.find(".ui-resizable-handle")),this._handles.disableSelection(),this._handles.on("mouseover",function(){r.resizing||(this.className&&(o=this.className.match(/ui-resizable-(se|sw|ne|nw|n|e|s|w)/i)),r.axis=o&&o[1]?o[1]:"se")}),a.autoHide&&(this._handles.hide(),this._addClass("ui-resizable-autohide"))},_removeHandles:function(){this._handles.remove()},_mouseCapture:function(e){var i,s,n=!1;for(i in this.handles)s=t(this.handles[i])[0],(s===e.target||t.contains(s,e.target))&&(n=!0);return!this.options.disabled&&n},_mouseStart:function(e){var i,s,n,o=this.options,a=this.element;return this.resizing=!0,this._renderProxy(),i=this._num(this.helper.css("left")),s=this._num(this.helper.css("top")),o.containment&&(i+=t(o.containment).scrollLeft()||0,s+=t(o.containment).scrollTop()||0),this.offset=this.helper.offset(),this.position={left:i,top:s},this.size=this._helper?{width:this.helper.width(),height:this.helper.height()}:{width:a.width(),height:a.height()},this.originalSize=this._helper?{width:a.outerWidth(),height:a.outerHeight()}:{width:a.width(),height:a.height()},this.sizeDiff={width:a.outerWidth()-a.width(),height:a.outerHeight()-a.height()},this.originalPosition={left:i,top:s},this.originalMousePosition={left:e.pageX,top:e.pageY},this.aspectRatio="number"==typeof o.aspectRatio?o.aspectRatio:this.originalSize.width/this.originalSize.height||1,n=t(".ui-resizable-"+this.axis).css("cursor"),t("body").css("cursor","auto"===n?this.axis+"-resize":n),this._addClass("ui-resizable-resizing"),this._propagate("start",e),!0},_mouseDrag:function(e){var i,s,n=this.originalMousePosition,o=this.axis,a=e.pageX-n.left||0,r=e.pageY-n.top||0,h=this._change[o];return this._updatePrevProperties(),h?(i=h.apply(this,[e,a,r]),this._updateVirtualBoundaries(e.shiftKey),(this._aspectRatio||e.shiftKey)&&(i=this._updateRatio(i,e)),i=this._respectSize(i,e),this._updateCache(i),this._propagate("resize",e),s=this._applyChanges(),!this._helper&&this._proportionallyResizeElements.length&&this._proportionallyResize(),t.isEmptyObject(s)||(this._updatePrevProperties(),this._trigger("resize",e,this.ui()),this._applyChanges()),!1):!1},_mouseStop:function(e){this.resizing=!1;var i,s,n,o,a,r,h,l=this.options,c=this;return this._helper&&(i=this._proportionallyResizeElements,s=i.length&&/textarea/i.test(i[0].nodeName),n=s&&this._hasScroll(i[0],"left")?0:c.sizeDiff.height,o=s?0:c.sizeDiff.width,a={width:c.helper.width()-o,height:c.helper.height()-n},r=parseFloat(c.element.css("left"))+(c.position.left-c.originalPosition.left)||null,h=parseFloat(c.element.css("top"))+(c.position.top-c.originalPosition.top)||null,l.animate||this.element.css(t.extend(a,{top:h,left:r})),c.helper.height(c.size.height),c.helper.width(c.size.width),this._helper&&!l.animate&&this._proportionallyResize()),t("body").css("cursor","auto"),this._removeClass("ui-resizable-resizing"),this._propagate("stop",e),this._helper&&this.helper.remove(),!1},_updatePrevProperties:function(){this.prevPosition={top:this.position.top,left:this.position.left},this.prevSize={width:this.size.width,height:this.size.height}},_applyChanges:function(){var t={};return this.position.top!==this.prevPosition.top&&(t.top=this.position.top+"px"),this.position.left!==this.prevPosition.left&&(t.left=this.position.left+"px"),this.size.width!==this.prevSize.width&&(t.width=this.size.width+"px"),this.size.height!==this.prevSize.height&&(t.height=this.size.height+"px"),this.helper.css(t),t},_updateVirtualBoundaries:function(t){var e,i,s,n,o,a=this.options;o={minWidth:this._isNumber(a.minWidth)?a.minWidth:0,maxWidth:this._isNumber(a.maxWidth)?a.maxWidth:1/0,minHeight:this._isNumber(a.minHeight)?a.minHeight:0,maxHeight:this._isNumber(a.maxHeight)?a.maxHeight:1/0},(this._aspectRatio||t)&&(e=o.minHeight*this.aspectRatio,s=o.minWidth/this.aspectRatio,i=o.maxHeight*this.aspectRatio,n=o.maxWidth/this.aspectRatio,e>o.minWidth&&(o.minWidth=e),s>o.minHeight&&(o.minHeight=s),o.maxWidth>i&&(o.maxWidth=i),o.maxHeight>n&&(o.maxHeight=n)),this._vBoundaries=o},_updateCache:function(t){this.offset=this.helper.offset(),this._isNumber(t.left)&&(this.position.left=t.left),this._isNumber(t.top)&&(this.position.top=t.top),this._isNumber(t.height)&&(this.size.height=t.height),this._isNumber(t.width)&&(this.size.width=t.width)},_updateRatio:function(t){var e=this.position,i=this.size,s=this.axis;return this._isNumber(t.height)?t.width=t.height*this.aspectRatio:this._isNumber(t.width)&&(t.height=t.width/this.aspectRatio),"sw"===s&&(t.left=e.left+(i.width-t.width),t.top=null),"nw"===s&&(t.top=e.top+(i.height-t.height),t.left=e.left+(i.width-t.width)),t},_respectSize:function(t){var e=this._vBoundaries,i=this.axis,s=this._isNumber(t.width)&&e.maxWidth&&e.maxWidth<t.width,n=this._isNumber(t.height)&&e.maxHeight&&e.maxHeight<t.height,o=this._isNumber(t.width)&&e.minWidth&&e.minWidth>t.width,a=this._isNumber(t.height)&&e.minHeight&&e.minHeight>t.height,r=this.originalPosition.left+this.originalSize.width,h=this.originalPosition.top+this.originalSize.height,l=/sw|nw|w/.test(i),c=/nw|ne|n/.test(i);return o&&(t.width=e.minWidth),a&&(t.height=e.minHeight),s&&(t.width=e.maxWidth),n&&(t.height=e.maxHeight),o&&l&&(t.left=r-e.minWidth),s&&l&&(t.left=r-e.maxWidth),a&&c&&(t.top=h-e.minHeight),n&&c&&(t.top=h-e.maxHeight),t.width||t.height||t.left||!t.top?t.width||t.height||t.top||!t.left||(t.left=null):t.top=null,t},_getPaddingPlusBorderDimensions:function(t){for(var e=0,i=[],s=[t.css("borderTopWidth"),t.css("borderRightWidth"),t.css("borderBottomWidth"),t.css("borderLeftWidth")],n=[t.css("paddingTop"),t.css("paddingRight"),t.css("paddingBottom"),t.css("paddingLeft")];4>e;e++)i[e]=parseFloat(s[e])||0,i[e]+=parseFloat(n[e])||0;return{height:i[0]+i[2],width:i[1]+i[3]}},_proportionallyResize:function(){if(this._proportionallyResizeElements.length)for(var t,e=0,i=this.helper||this.element;this._proportionallyResizeElements.length>e;e++)t=this._proportionallyResizeElements[e],this.outerDimensions||(this.outerDimensions=this._getPaddingPlusBorderDimensions(t)),t.css({height:i.height()-this.outerDimensions.height||0,width:i.width()-this.outerDimensions.width||0})},_renderProxy:function(){var e=this.element,i=this.options;this.elementOffset=e.offset(),this._helper?(this.helper=this.helper||t("<div style='overflow:hidden;'></div>"),this._addClass(this.helper,this._helper),this.helper.css({width:this.element.outerWidth(),height:this.element.outerHeight(),position:"absolute",left:this.elementOffset.left+"px",top:this.elementOffset.top+"px",zIndex:++i.zIndex}),this.helper.appendTo("body").disableSelection()):this.helper=this.element},_change:{e:function(t,e){return{width:this.originalSize.width+e}},w:function(t,e){var i=this.originalSize,s=this.originalPosition;return{left:s.left+e,width:i.width-e}},n:function(t,e,i){var s=this.originalSize,n=this.originalPosition;return{top:n.top+i,height:s.height-i}},s:function(t,e,i){return{height:this.originalSize.height+i}},se:function(e,i,s){return t.extend(this._change.s.apply(this,arguments),this._change.e.apply(this,[e,i,s]))},sw:function(e,i,s){return t.extend(this._change.s.apply(this,arguments),this._change.w.apply(this,[e,i,s]))},ne:function(e,i,s){return t.extend(this._change.n.apply(this,arguments),this._change.e.apply(this,[e,i,s]))},nw:function(e,i,s){return t.extend(this._change.n.apply(this,arguments),this._change.w.apply(this,[e,i,s]))}},_propagate:function(e,i){t.ui.plugin.call(this,e,[i,this.ui()]),"resize"!==e&&this._trigger(e,i,this.ui())},plugins:{},ui:function(){return{originalElement:this.originalElement,element:this.element,helper:this.helper,position:this.position,size:this.size,originalSize:this.originalSize,originalPosition:this.originalPosition}}}),t.ui.plugin.add("resizable","animate",{stop:function(e){var i=t(this).resizable("instance"),s=i.options,n=i._proportionallyResizeElements,o=n.length&&/textarea/i.test(n[0].nodeName),a=o&&i._hasScroll(n[0],"left")?0:i.sizeDiff.height,r=o?0:i.sizeDiff.width,h={width:i.size.width-r,height:i.size.height-a},l=parseFloat(i.element.css("left"))+(i.position.left-i.originalPosition.left)||null,c=parseFloat(i.element.css("top"))+(i.position.top-i.originalPosition.top)||null;i.element.animate(t.extend(h,c&&l?{top:c,left:l}:{}),{duration:s.animateDuration,easing:s.animateEasing,step:function(){var s={width:parseFloat(i.element.css("width")),height:parseFloat(i.element.css("height")),top:parseFloat(i.element.css("top")),left:parseFloat(i.element.css("left"))};n&&n.length&&t(n[0]).css({width:s.width,height:s.height}),i._updateCache(s),i._propagate("resize",e)}})}}),t.ui.plugin.add("resizable","containment",{start:function(){var e,i,s,n,o,a,r,h=t(this).resizable("instance"),l=h.options,c=h.element,u=l.containment,d=u instanceof t?u.get(0):/parent/.test(u)?c.parent().get(0):u;d&&(h.containerElement=t(d),/document/.test(u)||u===document?(h.containerOffset={left:0,top:0},h.containerPosition={left:0,top:0},h.parentData={element:t(document),left:0,top:0,width:t(document).width(),height:t(document).height()||document.body.parentNode.scrollHeight}):(e=t(d),i=[],t(["Top","Right","Left","Bottom"]).each(function(t,s){i[t]=h._num(e.css("padding"+s))}),h.containerOffset=e.offset(),h.containerPosition=e.position(),h.containerSize={height:e.innerHeight()-i[3],width:e.innerWidth()-i[1]},s=h.containerOffset,n=h.containerSize.height,o=h.containerSize.width,a=h._hasScroll(d,"left")?d.scrollWidth:o,r=h._hasScroll(d)?d.scrollHeight:n,h.parentData={element:d,left:s.left,top:s.top,width:a,height:r}))},resize:function(e){var i,s,n,o,a=t(this).resizable("instance"),r=a.options,h=a.containerOffset,l=a.position,c=a._aspectRatio||e.shiftKey,u={top:0,left:0},d=a.containerElement,p=!0;d[0]!==document&&/static/.test(d.css("position"))&&(u=h),l.left<(a._helper?h.left:0)&&(a.size.width=a.size.width+(a._helper?a.position.left-h.left:a.position.left-u.left),c&&(a.size.height=a.size.width/a.aspectRatio,p=!1),a.position.left=r.helper?h.left:0),l.top<(a._helper?h.top:0)&&(a.size.height=a.size.height+(a._helper?a.position.top-h.top:a.position.top),c&&(a.size.width=a.size.height*a.aspectRatio,p=!1),a.position.top=a._helper?h.top:0),n=a.containerElement.get(0)===a.element.parent().get(0),o=/relative|absolute/.test(a.containerElement.css("position")),n&&o?(a.offset.left=a.parentData.left+a.position.left,a.offset.top=a.parentData.top+a.position.top):(a.offset.left=a.element.offset().left,a.offset.top=a.element.offset().top),i=Math.abs(a.sizeDiff.width+(a._helper?a.offset.left-u.left:a.offset.left-h.left)),s=Math.abs(a.sizeDiff.height+(a._helper?a.offset.top-u.top:a.offset.top-h.top)),i+a.size.width>=a.parentData.width&&(a.size.width=a.parentData.width-i,c&&(a.size.height=a.size.width/a.aspectRatio,p=!1)),s+a.size.height>=a.parentData.height&&(a.size.height=a.parentData.height-s,c&&(a.size.width=a.size.height*a.aspectRatio,p=!1)),p||(a.position.left=a.prevPosition.left,a.position.top=a.prevPosition.top,a.size.width=a.prevSize.width,a.size.height=a.prevSize.height)},stop:function(){var e=t(this).resizable("instance"),i=e.options,s=e.containerOffset,n=e.containerPosition,o=e.containerElement,a=t(e.helper),r=a.offset(),h=a.outerWidth()-e.sizeDiff.width,l=a.outerHeight()-e.sizeDiff.height;e._helper&&!i.animate&&/relative/.test(o.css("position"))&&t(this).css({left:r.left-n.left-s.left,width:h,height:l}),e._helper&&!i.animate&&/static/.test(o.css("position"))&&t(this).css({left:r.left-n.left-s.left,width:h,height:l})}}),t.ui.plugin.add("resizable","alsoResize",{start:function(){var e=t(this).resizable("instance"),i=e.options;t(i.alsoResize).each(function(){var e=t(this);e.data("ui-resizable-alsoresize",{width:parseFloat(e.width()),height:parseFloat(e.height()),left:parseFloat(e.css("left")),top:parseFloat(e.css("top"))})})},resize:function(e,i){var s=t(this).resizable("instance"),n=s.options,o=s.originalSize,a=s.originalPosition,r={height:s.size.height-o.height||0,width:s.size.width-o.width||0,top:s.position.top-a.top||0,left:s.position.left-a.left||0};t(n.alsoResize).each(function(){var e=t(this),s=t(this).data("ui-resizable-alsoresize"),n={},o=e.parents(i.originalElement[0]).length?["width","height"]:["width","height","top","left"];t.each(o,function(t,e){var i=(s[e]||0)+(r[e]||0);i&&i>=0&&(n[e]=i||null)}),e.css(n)})},stop:function(){t(this).removeData("ui-resizable-alsoresize")}}),t.ui.plugin.add("resizable","ghost",{start:function(){var e=t(this).resizable("instance"),i=e.size;e.ghost=e.originalElement.clone(),e.ghost.css({opacity:.25,display:"block",position:"relative",height:i.height,width:i.width,margin:0,left:0,top:0}),e._addClass(e.ghost,"ui-resizable-ghost"),t.uiBackCompat!==!1&&"string"==typeof e.options.ghost&&e.ghost.addClass(this.options.ghost),e.ghost.appendTo(e.helper)},resize:function(){var e=t(this).resizable("instance");e.ghost&&e.ghost.css({position:"relative",height:e.size.height,width:e.size.width})},stop:function(){var e=t(this).resizable("instance");e.ghost&&e.helper&&e.helper.get(0).removeChild(e.ghost.get(0))}}),t.ui.plugin.add("resizable","grid",{resize:function(){var e,i=t(this).resizable("instance"),s=i.options,n=i.size,o=i.originalSize,a=i.originalPosition,r=i.axis,h="number"==typeof s.grid?[s.grid,s.grid]:s.grid,l=h[0]||1,c=h[1]||1,u=Math.round((n.width-o.width)/l)*l,d=Math.round((n.height-o.height)/c)*c,p=o.width+u,f=o.height+d,g=s.maxWidth&&p>s.maxWidth,m=s.maxHeight&&f>s.maxHeight,_=s.minWidth&&s.minWidth>p,v=s.minHeight&&s.minHeight>f;s.grid=h,_&&(p+=l),v&&(f+=c),g&&(p-=l),m&&(f-=c),/^(se|s|e)$/.test(r)?(i.size.width=p,i.size.height=f):/^(ne)$/.test(r)?(i.size.width=p,i.size.height=f,i.position.top=a.top-d):/^(sw)$/.test(r)?(i.size.width=p,i.size.height=f,i.position.left=a.left-u):((0>=f-c||0>=p-l)&&(e=i._getPaddingPlusBorderDimensions(this)),f-c>0?(i.size.height=f,i.position.top=a.top-d):(f=c-e.height,i.size.height=f,i.position.top=a.top+o.height-f),p-l>0?(i.size.width=p,i.position.left=a.left-u):(p=l-e.width,i.size.width=p,i.position.left=a.left+o.width-p))}}),t.ui.resizable,t.widget("ui.dialog",{version:"1.12.1",options:{appendTo:"body",autoOpen:!0,buttons:[],classes:{"ui-dialog":"ui-corner-all","ui-dialog-titlebar":"ui-corner-all"},closeOnEscape:!0,closeText:"Close",draggable:!0,hide:null,height:"auto",maxHeight:null,maxWidth:null,minHeight:150,minWidth:150,modal:!1,position:{my:"center",at:"center",of:window,collision:"fit",using:function(e){var i=t(this).css(e).offset().top;0>i&&t(this).css("top",e.top-i)}},resizable:!0,show:null,title:null,width:300,beforeClose:null,close:null,drag:null,dragStart:null,dragStop:null,focus:null,open:null,resize:null,resizeStart:null,resizeStop:null},sizeRelatedOptions:{buttons:!0,height:!0,maxHeight:!0,maxWidth:!0,minHeight:!0,minWidth:!0,width:!0},resizableRelatedOptions:{maxHeight:!0,maxWidth:!0,minHeight:!0,minWidth:!0},_create:function(){this.originalCss={display:this.element[0].style.display,width:this.element[0].style.width,minHeight:this.element[0].style.minHeight,maxHeight:this.element[0].style.maxHeight,height:this.element[0].style.height},this.originalPosition={parent:this.element.parent(),index:this.element.parent().children().index(this.element)},this.originalTitle=this.element.attr("title"),null==this.options.title&&null!=this.originalTitle&&(this.options.title=this.originalTitle),this.options.disabled&&(this.options.disabled=!1),this._createWrapper(),this.element.show().removeAttr("title").appendTo(this.uiDialog),this._addClass("ui-dialog-content","ui-widget-content"),this._createTitlebar(),this._createButtonPane(),this.options.draggable&&t.fn.draggable&&this._makeDraggable(),this.options.resizable&&t.fn.resizable&&this._makeResizable(),this._isOpen=!1,this._trackFocus()},_init:function(){this.options.autoOpen&&this.open()},_appendTo:function(){var e=this.options.appendTo;return e&&(e.jquery||e.nodeType)?t(e):this.document.find(e||"body").eq(0)},_destroy:function(){var t,e=this.originalPosition;this._untrackInstance(),this._destroyOverlay(),this.element.removeUniqueId().css(this.originalCss).detach(),this.uiDialog.remove(),this.originalTitle&&this.element.attr("title",this.originalTitle),t=e.parent.children().eq(e.index),t.length&&t[0]!==this.element[0]?t.before(this.element):e.parent.append(this.element)},widget:function(){return this.uiDialog -},disable:t.noop,enable:t.noop,close:function(e){var i=this;this._isOpen&&this._trigger("beforeClose",e)!==!1&&(this._isOpen=!1,this._focusedElement=null,this._destroyOverlay(),this._untrackInstance(),this.opener.filter(":focusable").trigger("focus").length||t.ui.safeBlur(t.ui.safeActiveElement(this.document[0])),this._hide(this.uiDialog,this.options.hide,function(){i._trigger("close",e)}))},isOpen:function(){return this._isOpen},moveToTop:function(){this._moveToTop()},_moveToTop:function(e,i){var s=!1,n=this.uiDialog.siblings(".ui-front:visible").map(function(){return+t(this).css("z-index")}).get(),o=Math.max.apply(null,n);return o>=+this.uiDialog.css("z-index")&&(this.uiDialog.css("z-index",o+1),s=!0),s&&!i&&this._trigger("focus",e),s},open:function(){var e=this;return this._isOpen?(this._moveToTop()&&this._focusTabbable(),void 0):(this._isOpen=!0,this.opener=t(t.ui.safeActiveElement(this.document[0])),this._size(),this._position(),this._createOverlay(),this._moveToTop(null,!0),this.overlay&&this.overlay.css("z-index",this.uiDialog.css("z-index")-1),this._show(this.uiDialog,this.options.show,function(){e._focusTabbable(),e._trigger("focus")}),this._makeFocusTarget(),this._trigger("open"),void 0)},_focusTabbable:function(){var t=this._focusedElement;t||(t=this.element.find("[autofocus]")),t.length||(t=this.element.find(":tabbable")),t.length||(t=this.uiDialogButtonPane.find(":tabbable")),t.length||(t=this.uiDialogTitlebarClose.filter(":tabbable")),t.length||(t=this.uiDialog),t.eq(0).trigger("focus")},_keepFocus:function(e){function i(){var e=t.ui.safeActiveElement(this.document[0]),i=this.uiDialog[0]===e||t.contains(this.uiDialog[0],e);i||this._focusTabbable()}e.preventDefault(),i.call(this),this._delay(i)},_createWrapper:function(){this.uiDialog=t("<div>").hide().attr({tabIndex:-1,role:"dialog"}).appendTo(this._appendTo()),this._addClass(this.uiDialog,"ui-dialog","ui-widget ui-widget-content ui-front"),this._on(this.uiDialog,{keydown:function(e){if(this.options.closeOnEscape&&!e.isDefaultPrevented()&&e.keyCode&&e.keyCode===t.ui.keyCode.ESCAPE)return e.preventDefault(),this.close(e),void 0;if(e.keyCode===t.ui.keyCode.TAB&&!e.isDefaultPrevented()){var i=this.uiDialog.find(":tabbable"),s=i.filter(":first"),n=i.filter(":last");e.target!==n[0]&&e.target!==this.uiDialog[0]||e.shiftKey?e.target!==s[0]&&e.target!==this.uiDialog[0]||!e.shiftKey||(this._delay(function(){n.trigger("focus")}),e.preventDefault()):(this._delay(function(){s.trigger("focus")}),e.preventDefault())}},mousedown:function(t){this._moveToTop(t)&&this._focusTabbable()}}),this.element.find("[aria-describedby]").length||this.uiDialog.attr({"aria-describedby":this.element.uniqueId().attr("id")})},_createTitlebar:function(){var e;this.uiDialogTitlebar=t("<div>"),this._addClass(this.uiDialogTitlebar,"ui-dialog-titlebar","ui-widget-header ui-helper-clearfix"),this._on(this.uiDialogTitlebar,{mousedown:function(e){t(e.target).closest(".ui-dialog-titlebar-close")||this.uiDialog.trigger("focus")}}),this.uiDialogTitlebarClose=t("<button type='button'></button>").button({label:t("<a>").text(this.options.closeText).html(),icon:"ui-icon-closethick",showLabel:!1}).appendTo(this.uiDialogTitlebar),this._addClass(this.uiDialogTitlebarClose,"ui-dialog-titlebar-close"),this._on(this.uiDialogTitlebarClose,{click:function(t){t.preventDefault(),this.close(t)}}),e=t("<span>").uniqueId().prependTo(this.uiDialogTitlebar),this._addClass(e,"ui-dialog-title"),this._title(e),this.uiDialogTitlebar.prependTo(this.uiDialog),this.uiDialog.attr({"aria-labelledby":e.attr("id")})},_title:function(t){this.options.title?t.text(this.options.title):t.html("&#160;")},_createButtonPane:function(){this.uiDialogButtonPane=t("<div>"),this._addClass(this.uiDialogButtonPane,"ui-dialog-buttonpane","ui-widget-content ui-helper-clearfix"),this.uiButtonSet=t("<div>").appendTo(this.uiDialogButtonPane),this._addClass(this.uiButtonSet,"ui-dialog-buttonset"),this._createButtons()},_createButtons:function(){var e=this,i=this.options.buttons;return this.uiDialogButtonPane.remove(),this.uiButtonSet.empty(),t.isEmptyObject(i)||t.isArray(i)&&!i.length?(this._removeClass(this.uiDialog,"ui-dialog-buttons"),void 0):(t.each(i,function(i,s){var n,o;s=t.isFunction(s)?{click:s,text:i}:s,s=t.extend({type:"button"},s),n=s.click,o={icon:s.icon,iconPosition:s.iconPosition,showLabel:s.showLabel,icons:s.icons,text:s.text},delete s.click,delete s.icon,delete s.iconPosition,delete s.showLabel,delete s.icons,"boolean"==typeof s.text&&delete s.text,t("<button></button>",s).button(o).appendTo(e.uiButtonSet).on("click",function(){n.apply(e.element[0],arguments)})}),this._addClass(this.uiDialog,"ui-dialog-buttons"),this.uiDialogButtonPane.appendTo(this.uiDialog),void 0)},_makeDraggable:function(){function e(t){return{position:t.position,offset:t.offset}}var i=this,s=this.options;this.uiDialog.draggable({cancel:".ui-dialog-content, .ui-dialog-titlebar-close",handle:".ui-dialog-titlebar",containment:"document",start:function(s,n){i._addClass(t(this),"ui-dialog-dragging"),i._blockFrames(),i._trigger("dragStart",s,e(n))},drag:function(t,s){i._trigger("drag",t,e(s))},stop:function(n,o){var a=o.offset.left-i.document.scrollLeft(),r=o.offset.top-i.document.scrollTop();s.position={my:"left top",at:"left"+(a>=0?"+":"")+a+" "+"top"+(r>=0?"+":"")+r,of:i.window},i._removeClass(t(this),"ui-dialog-dragging"),i._unblockFrames(),i._trigger("dragStop",n,e(o))}})},_makeResizable:function(){function e(t){return{originalPosition:t.originalPosition,originalSize:t.originalSize,position:t.position,size:t.size}}var i=this,s=this.options,n=s.resizable,o=this.uiDialog.css("position"),a="string"==typeof n?n:"n,e,s,w,se,sw,ne,nw";this.uiDialog.resizable({cancel:".ui-dialog-content",containment:"document",alsoResize:this.element,maxWidth:s.maxWidth,maxHeight:s.maxHeight,minWidth:s.minWidth,minHeight:this._minHeight(),handles:a,start:function(s,n){i._addClass(t(this),"ui-dialog-resizing"),i._blockFrames(),i._trigger("resizeStart",s,e(n))},resize:function(t,s){i._trigger("resize",t,e(s))},stop:function(n,o){var a=i.uiDialog.offset(),r=a.left-i.document.scrollLeft(),h=a.top-i.document.scrollTop();s.height=i.uiDialog.height(),s.width=i.uiDialog.width(),s.position={my:"left top",at:"left"+(r>=0?"+":"")+r+" "+"top"+(h>=0?"+":"")+h,of:i.window},i._removeClass(t(this),"ui-dialog-resizing"),i._unblockFrames(),i._trigger("resizeStop",n,e(o))}}).css("position",o)},_trackFocus:function(){this._on(this.widget(),{focusin:function(e){this._makeFocusTarget(),this._focusedElement=t(e.target)}})},_makeFocusTarget:function(){this._untrackInstance(),this._trackingInstances().unshift(this)},_untrackInstance:function(){var e=this._trackingInstances(),i=t.inArray(this,e);-1!==i&&e.splice(i,1)},_trackingInstances:function(){var t=this.document.data("ui-dialog-instances");return t||(t=[],this.document.data("ui-dialog-instances",t)),t},_minHeight:function(){var t=this.options;return"auto"===t.height?t.minHeight:Math.min(t.minHeight,t.height)},_position:function(){var t=this.uiDialog.is(":visible");t||this.uiDialog.show(),this.uiDialog.position(this.options.position),t||this.uiDialog.hide()},_setOptions:function(e){var i=this,s=!1,n={};t.each(e,function(t,e){i._setOption(t,e),t in i.sizeRelatedOptions&&(s=!0),t in i.resizableRelatedOptions&&(n[t]=e)}),s&&(this._size(),this._position()),this.uiDialog.is(":data(ui-resizable)")&&this.uiDialog.resizable("option",n)},_setOption:function(e,i){var s,n,o=this.uiDialog;"disabled"!==e&&(this._super(e,i),"appendTo"===e&&this.uiDialog.appendTo(this._appendTo()),"buttons"===e&&this._createButtons(),"closeText"===e&&this.uiDialogTitlebarClose.button({label:t("<a>").text(""+this.options.closeText).html()}),"draggable"===e&&(s=o.is(":data(ui-draggable)"),s&&!i&&o.draggable("destroy"),!s&&i&&this._makeDraggable()),"position"===e&&this._position(),"resizable"===e&&(n=o.is(":data(ui-resizable)"),n&&!i&&o.resizable("destroy"),n&&"string"==typeof i&&o.resizable("option","handles",i),n||i===!1||this._makeResizable()),"title"===e&&this._title(this.uiDialogTitlebar.find(".ui-dialog-title")))},_size:function(){var t,e,i,s=this.options;this.element.show().css({width:"auto",minHeight:0,maxHeight:"none",height:0}),s.minWidth>s.width&&(s.width=s.minWidth),t=this.uiDialog.css({height:"auto",width:s.width}).outerHeight(),e=Math.max(0,s.minHeight-t),i="number"==typeof s.maxHeight?Math.max(0,s.maxHeight-t):"none","auto"===s.height?this.element.css({minHeight:e,maxHeight:i,height:"auto"}):this.element.height(Math.max(0,s.height-t)),this.uiDialog.is(":data(ui-resizable)")&&this.uiDialog.resizable("option","minHeight",this._minHeight())},_blockFrames:function(){this.iframeBlocks=this.document.find("iframe").map(function(){var e=t(this);return t("<div>").css({position:"absolute",width:e.outerWidth(),height:e.outerHeight()}).appendTo(e.parent()).offset(e.offset())[0]})},_unblockFrames:function(){this.iframeBlocks&&(this.iframeBlocks.remove(),delete this.iframeBlocks)},_allowInteraction:function(e){return t(e.target).closest(".ui-dialog").length?!0:!!t(e.target).closest(".ui-datepicker").length},_createOverlay:function(){if(this.options.modal){var e=!0;this._delay(function(){e=!1}),this.document.data("ui-dialog-overlays")||this._on(this.document,{focusin:function(t){e||this._allowInteraction(t)||(t.preventDefault(),this._trackingInstances()[0]._focusTabbable())}}),this.overlay=t("<div>").appendTo(this._appendTo()),this._addClass(this.overlay,null,"ui-widget-overlay ui-front"),this._on(this.overlay,{mousedown:"_keepFocus"}),this.document.data("ui-dialog-overlays",(this.document.data("ui-dialog-overlays")||0)+1)}},_destroyOverlay:function(){if(this.options.modal&&this.overlay){var t=this.document.data("ui-dialog-overlays")-1;t?this.document.data("ui-dialog-overlays",t):(this._off(this.document,"focusin"),this.document.removeData("ui-dialog-overlays")),this.overlay.remove(),this.overlay=null}}}),t.uiBackCompat!==!1&&t.widget("ui.dialog",t.ui.dialog,{options:{dialogClass:""},_createWrapper:function(){this._super(),this.uiDialog.addClass(this.options.dialogClass)},_setOption:function(t,e){"dialogClass"===t&&this.uiDialog.removeClass(this.options.dialogClass).addClass(e),this._superApply(arguments)}}),t.ui.dialog,t.widget("ui.droppable",{version:"1.12.1",widgetEventPrefix:"drop",options:{accept:"*",addClasses:!0,greedy:!1,scope:"default",tolerance:"intersect",activate:null,deactivate:null,drop:null,out:null,over:null},_create:function(){var e,i=this.options,s=i.accept;this.isover=!1,this.isout=!0,this.accept=t.isFunction(s)?s:function(t){return t.is(s)},this.proportions=function(){return arguments.length?(e=arguments[0],void 0):e?e:e={width:this.element[0].offsetWidth,height:this.element[0].offsetHeight}},this._addToManager(i.scope),i.addClasses&&this._addClass("ui-droppable")},_addToManager:function(e){t.ui.ddmanager.droppables[e]=t.ui.ddmanager.droppables[e]||[],t.ui.ddmanager.droppables[e].push(this)},_splice:function(t){for(var e=0;t.length>e;e++)t[e]===this&&t.splice(e,1)},_destroy:function(){var e=t.ui.ddmanager.droppables[this.options.scope];this._splice(e)},_setOption:function(e,i){if("accept"===e)this.accept=t.isFunction(i)?i:function(t){return t.is(i)};else if("scope"===e){var s=t.ui.ddmanager.droppables[this.options.scope];this._splice(s),this._addToManager(i)}this._super(e,i)},_activate:function(e){var i=t.ui.ddmanager.current;this._addActiveClass(),i&&this._trigger("activate",e,this.ui(i))},_deactivate:function(e){var i=t.ui.ddmanager.current;this._removeActiveClass(),i&&this._trigger("deactivate",e,this.ui(i))},_over:function(e){var i=t.ui.ddmanager.current;i&&(i.currentItem||i.element)[0]!==this.element[0]&&this.accept.call(this.element[0],i.currentItem||i.element)&&(this._addHoverClass(),this._trigger("over",e,this.ui(i)))},_out:function(e){var i=t.ui.ddmanager.current;i&&(i.currentItem||i.element)[0]!==this.element[0]&&this.accept.call(this.element[0],i.currentItem||i.element)&&(this._removeHoverClass(),this._trigger("out",e,this.ui(i)))},_drop:function(e,i){var s=i||t.ui.ddmanager.current,n=!1;return s&&(s.currentItem||s.element)[0]!==this.element[0]?(this.element.find(":data(ui-droppable)").not(".ui-draggable-dragging").each(function(){var i=t(this).droppable("instance");return i.options.greedy&&!i.options.disabled&&i.options.scope===s.options.scope&&i.accept.call(i.element[0],s.currentItem||s.element)&&v(s,t.extend(i,{offset:i.element.offset()}),i.options.tolerance,e)?(n=!0,!1):void 0}),n?!1:this.accept.call(this.element[0],s.currentItem||s.element)?(this._removeActiveClass(),this._removeHoverClass(),this._trigger("drop",e,this.ui(s)),this.element):!1):!1},ui:function(t){return{draggable:t.currentItem||t.element,helper:t.helper,position:t.position,offset:t.positionAbs}},_addHoverClass:function(){this._addClass("ui-droppable-hover")},_removeHoverClass:function(){this._removeClass("ui-droppable-hover")},_addActiveClass:function(){this._addClass("ui-droppable-active")},_removeActiveClass:function(){this._removeClass("ui-droppable-active")}});var v=t.ui.intersect=function(){function t(t,e,i){return t>=e&&e+i>t}return function(e,i,s,n){if(!i.offset)return!1;var o=(e.positionAbs||e.position.absolute).left+e.margins.left,a=(e.positionAbs||e.position.absolute).top+e.margins.top,r=o+e.helperProportions.width,h=a+e.helperProportions.height,l=i.offset.left,c=i.offset.top,u=l+i.proportions().width,d=c+i.proportions().height;switch(s){case"fit":return o>=l&&u>=r&&a>=c&&d>=h;case"intersect":return o+e.helperProportions.width/2>l&&u>r-e.helperProportions.width/2&&a+e.helperProportions.height/2>c&&d>h-e.helperProportions.height/2;case"pointer":return t(n.pageY,c,i.proportions().height)&&t(n.pageX,l,i.proportions().width);case"touch":return(a>=c&&d>=a||h>=c&&d>=h||c>a&&h>d)&&(o>=l&&u>=o||r>=l&&u>=r||l>o&&r>u);default:return!1}}}();t.ui.ddmanager={current:null,droppables:{"default":[]},prepareOffsets:function(e,i){var s,n,o=t.ui.ddmanager.droppables[e.options.scope]||[],a=i?i.type:null,r=(e.currentItem||e.element).find(":data(ui-droppable)").addBack();t:for(s=0;o.length>s;s++)if(!(o[s].options.disabled||e&&!o[s].accept.call(o[s].element[0],e.currentItem||e.element))){for(n=0;r.length>n;n++)if(r[n]===o[s].element[0]){o[s].proportions().height=0;continue t}o[s].visible="none"!==o[s].element.css("display"),o[s].visible&&("mousedown"===a&&o[s]._activate.call(o[s],i),o[s].offset=o[s].element.offset(),o[s].proportions({width:o[s].element[0].offsetWidth,height:o[s].element[0].offsetHeight}))}},drop:function(e,i){var s=!1;return t.each((t.ui.ddmanager.droppables[e.options.scope]||[]).slice(),function(){this.options&&(!this.options.disabled&&this.visible&&v(e,this,this.options.tolerance,i)&&(s=this._drop.call(this,i)||s),!this.options.disabled&&this.visible&&this.accept.call(this.element[0],e.currentItem||e.element)&&(this.isout=!0,this.isover=!1,this._deactivate.call(this,i)))}),s},dragStart:function(e,i){e.element.parentsUntil("body").on("scroll.droppable",function(){e.options.refreshPositions||t.ui.ddmanager.prepareOffsets(e,i)})},drag:function(e,i){e.options.refreshPositions&&t.ui.ddmanager.prepareOffsets(e,i),t.each(t.ui.ddmanager.droppables[e.options.scope]||[],function(){if(!this.options.disabled&&!this.greedyChild&&this.visible){var s,n,o,a=v(e,this,this.options.tolerance,i),r=!a&&this.isover?"isout":a&&!this.isover?"isover":null;r&&(this.options.greedy&&(n=this.options.scope,o=this.element.parents(":data(ui-droppable)").filter(function(){return t(this).droppable("instance").options.scope===n}),o.length&&(s=t(o[0]).droppable("instance"),s.greedyChild="isover"===r)),s&&"isover"===r&&(s.isover=!1,s.isout=!0,s._out.call(s,i)),this[r]=!0,this["isout"===r?"isover":"isout"]=!1,this["isover"===r?"_over":"_out"].call(this,i),s&&"isout"===r&&(s.isout=!1,s.isover=!0,s._over.call(s,i)))}})},dragStop:function(e,i){e.element.parentsUntil("body").off("scroll.droppable"),e.options.refreshPositions||t.ui.ddmanager.prepareOffsets(e,i)}},t.uiBackCompat!==!1&&t.widget("ui.droppable",t.ui.droppable,{options:{hoverClass:!1,activeClass:!1},_addActiveClass:function(){this._super(),this.options.activeClass&&this.element.addClass(this.options.activeClass)},_removeActiveClass:function(){this._super(),this.options.activeClass&&this.element.removeClass(this.options.activeClass)},_addHoverClass:function(){this._super(),this.options.hoverClass&&this.element.addClass(this.options.hoverClass)},_removeHoverClass:function(){this._super(),this.options.hoverClass&&this.element.removeClass(this.options.hoverClass)}}),t.ui.droppable,t.widget("ui.progressbar",{version:"1.12.1",options:{classes:{"ui-progressbar":"ui-corner-all","ui-progressbar-value":"ui-corner-left","ui-progressbar-complete":"ui-corner-right"},max:100,value:0,change:null,complete:null},min:0,_create:function(){this.oldValue=this.options.value=this._constrainedValue(),this.element.attr({role:"progressbar","aria-valuemin":this.min}),this._addClass("ui-progressbar","ui-widget ui-widget-content"),this.valueDiv=t("<div>").appendTo(this.element),this._addClass(this.valueDiv,"ui-progressbar-value","ui-widget-header"),this._refreshValue()},_destroy:function(){this.element.removeAttr("role aria-valuemin aria-valuemax aria-valuenow"),this.valueDiv.remove()},value:function(t){return void 0===t?this.options.value:(this.options.value=this._constrainedValue(t),this._refreshValue(),void 0)},_constrainedValue:function(t){return void 0===t&&(t=this.options.value),this.indeterminate=t===!1,"number"!=typeof t&&(t=0),this.indeterminate?!1:Math.min(this.options.max,Math.max(this.min,t))},_setOptions:function(t){var e=t.value;delete t.value,this._super(t),this.options.value=this._constrainedValue(e),this._refreshValue()},_setOption:function(t,e){"max"===t&&(e=Math.max(this.min,e)),this._super(t,e)},_setOptionDisabled:function(t){this._super(t),this.element.attr("aria-disabled",t),this._toggleClass(null,"ui-state-disabled",!!t)},_percentage:function(){return this.indeterminate?100:100*(this.options.value-this.min)/(this.options.max-this.min)},_refreshValue:function(){var e=this.options.value,i=this._percentage();this.valueDiv.toggle(this.indeterminate||e>this.min).width(i.toFixed(0)+"%"),this._toggleClass(this.valueDiv,"ui-progressbar-complete",null,e===this.options.max)._toggleClass("ui-progressbar-indeterminate",null,this.indeterminate),this.indeterminate?(this.element.removeAttr("aria-valuenow"),this.overlayDiv||(this.overlayDiv=t("<div>").appendTo(this.valueDiv),this._addClass(this.overlayDiv,"ui-progressbar-overlay"))):(this.element.attr({"aria-valuemax":this.options.max,"aria-valuenow":e}),this.overlayDiv&&(this.overlayDiv.remove(),this.overlayDiv=null)),this.oldValue!==e&&(this.oldValue=e,this._trigger("change")),e===this.options.max&&this._trigger("complete")}}),t.widget("ui.selectable",t.ui.mouse,{version:"1.12.1",options:{appendTo:"body",autoRefresh:!0,distance:0,filter:"*",tolerance:"touch",selected:null,selecting:null,start:null,stop:null,unselected:null,unselecting:null},_create:function(){var e=this;this._addClass("ui-selectable"),this.dragged=!1,this.refresh=function(){e.elementPos=t(e.element[0]).offset(),e.selectees=t(e.options.filter,e.element[0]),e._addClass(e.selectees,"ui-selectee"),e.selectees.each(function(){var i=t(this),s=i.offset(),n={left:s.left-e.elementPos.left,top:s.top-e.elementPos.top};t.data(this,"selectable-item",{element:this,$element:i,left:n.left,top:n.top,right:n.left+i.outerWidth(),bottom:n.top+i.outerHeight(),startselected:!1,selected:i.hasClass("ui-selected"),selecting:i.hasClass("ui-selecting"),unselecting:i.hasClass("ui-unselecting")})})},this.refresh(),this._mouseInit(),this.helper=t("<div>"),this._addClass(this.helper,"ui-selectable-helper")},_destroy:function(){this.selectees.removeData("selectable-item"),this._mouseDestroy()},_mouseStart:function(e){var i=this,s=this.options;this.opos=[e.pageX,e.pageY],this.elementPos=t(this.element[0]).offset(),this.options.disabled||(this.selectees=t(s.filter,this.element[0]),this._trigger("start",e),t(s.appendTo).append(this.helper),this.helper.css({left:e.pageX,top:e.pageY,width:0,height:0}),s.autoRefresh&&this.refresh(),this.selectees.filter(".ui-selected").each(function(){var s=t.data(this,"selectable-item");s.startselected=!0,e.metaKey||e.ctrlKey||(i._removeClass(s.$element,"ui-selected"),s.selected=!1,i._addClass(s.$element,"ui-unselecting"),s.unselecting=!0,i._trigger("unselecting",e,{unselecting:s.element}))}),t(e.target).parents().addBack().each(function(){var s,n=t.data(this,"selectable-item");return n?(s=!e.metaKey&&!e.ctrlKey||!n.$element.hasClass("ui-selected"),i._removeClass(n.$element,s?"ui-unselecting":"ui-selected")._addClass(n.$element,s?"ui-selecting":"ui-unselecting"),n.unselecting=!s,n.selecting=s,n.selected=s,s?i._trigger("selecting",e,{selecting:n.element}):i._trigger("unselecting",e,{unselecting:n.element}),!1):void 0}))},_mouseDrag:function(e){if(this.dragged=!0,!this.options.disabled){var i,s=this,n=this.options,o=this.opos[0],a=this.opos[1],r=e.pageX,h=e.pageY;return o>r&&(i=r,r=o,o=i),a>h&&(i=h,h=a,a=i),this.helper.css({left:o,top:a,width:r-o,height:h-a}),this.selectees.each(function(){var i=t.data(this,"selectable-item"),l=!1,c={};i&&i.element!==s.element[0]&&(c.left=i.left+s.elementPos.left,c.right=i.right+s.elementPos.left,c.top=i.top+s.elementPos.top,c.bottom=i.bottom+s.elementPos.top,"touch"===n.tolerance?l=!(c.left>r||o>c.right||c.top>h||a>c.bottom):"fit"===n.tolerance&&(l=c.left>o&&r>c.right&&c.top>a&&h>c.bottom),l?(i.selected&&(s._removeClass(i.$element,"ui-selected"),i.selected=!1),i.unselecting&&(s._removeClass(i.$element,"ui-unselecting"),i.unselecting=!1),i.selecting||(s._addClass(i.$element,"ui-selecting"),i.selecting=!0,s._trigger("selecting",e,{selecting:i.element}))):(i.selecting&&((e.metaKey||e.ctrlKey)&&i.startselected?(s._removeClass(i.$element,"ui-selecting"),i.selecting=!1,s._addClass(i.$element,"ui-selected"),i.selected=!0):(s._removeClass(i.$element,"ui-selecting"),i.selecting=!1,i.startselected&&(s._addClass(i.$element,"ui-unselecting"),i.unselecting=!0),s._trigger("unselecting",e,{unselecting:i.element}))),i.selected&&(e.metaKey||e.ctrlKey||i.startselected||(s._removeClass(i.$element,"ui-selected"),i.selected=!1,s._addClass(i.$element,"ui-unselecting"),i.unselecting=!0,s._trigger("unselecting",e,{unselecting:i.element})))))}),!1}},_mouseStop:function(e){var i=this;return this.dragged=!1,t(".ui-unselecting",this.element[0]).each(function(){var s=t.data(this,"selectable-item");i._removeClass(s.$element,"ui-unselecting"),s.unselecting=!1,s.startselected=!1,i._trigger("unselected",e,{unselected:s.element})}),t(".ui-selecting",this.element[0]).each(function(){var s=t.data(this,"selectable-item");i._removeClass(s.$element,"ui-selecting")._addClass(s.$element,"ui-selected"),s.selecting=!1,s.selected=!0,s.startselected=!0,i._trigger("selected",e,{selected:s.element})}),this._trigger("stop",e),this.helper.remove(),!1}}),t.widget("ui.selectmenu",[t.ui.formResetMixin,{version:"1.12.1",defaultElement:"<select>",options:{appendTo:null,classes:{"ui-selectmenu-button-open":"ui-corner-top","ui-selectmenu-button-closed":"ui-corner-all"},disabled:null,icons:{button:"ui-icon-triangle-1-s"},position:{my:"left top",at:"left bottom",collision:"none"},width:!1,change:null,close:null,focus:null,open:null,select:null},_create:function(){var e=this.element.uniqueId().attr("id");this.ids={element:e,button:e+"-button",menu:e+"-menu"},this._drawButton(),this._drawMenu(),this._bindFormResetHandler(),this._rendered=!1,this.menuItems=t()},_drawButton:function(){var e,i=this,s=this._parseOption(this.element.find("option:selected"),this.element[0].selectedIndex);this.labels=this.element.labels().attr("for",this.ids.button),this._on(this.labels,{click:function(t){this.button.focus(),t.preventDefault()}}),this.element.hide(),this.button=t("<span>",{tabindex:this.options.disabled?-1:0,id:this.ids.button,role:"combobox","aria-expanded":"false","aria-autocomplete":"list","aria-owns":this.ids.menu,"aria-haspopup":"true",title:this.element.attr("title")}).insertAfter(this.element),this._addClass(this.button,"ui-selectmenu-button ui-selectmenu-button-closed","ui-button ui-widget"),e=t("<span>").appendTo(this.button),this._addClass(e,"ui-selectmenu-icon","ui-icon "+this.options.icons.button),this.buttonItem=this._renderButtonItem(s).appendTo(this.button),this.options.width!==!1&&this._resizeButton(),this._on(this.button,this._buttonEvents),this.button.one("focusin",function(){i._rendered||i._refreshMenu()})},_drawMenu:function(){var e=this;this.menu=t("<ul>",{"aria-hidden":"true","aria-labelledby":this.ids.button,id:this.ids.menu}),this.menuWrap=t("<div>").append(this.menu),this._addClass(this.menuWrap,"ui-selectmenu-menu","ui-front"),this.menuWrap.appendTo(this._appendTo()),this.menuInstance=this.menu.menu({classes:{"ui-menu":"ui-corner-bottom"},role:"listbox",select:function(t,i){t.preventDefault(),e._setSelection(),e._select(i.item.data("ui-selectmenu-item"),t)},focus:function(t,i){var s=i.item.data("ui-selectmenu-item");null!=e.focusIndex&&s.index!==e.focusIndex&&(e._trigger("focus",t,{item:s}),e.isOpen||e._select(s,t)),e.focusIndex=s.index,e.button.attr("aria-activedescendant",e.menuItems.eq(s.index).attr("id"))}}).menu("instance"),this.menuInstance._off(this.menu,"mouseleave"),this.menuInstance._closeOnDocumentClick=function(){return!1},this.menuInstance._isDivider=function(){return!1}},refresh:function(){this._refreshMenu(),this.buttonItem.replaceWith(this.buttonItem=this._renderButtonItem(this._getSelectedItem().data("ui-selectmenu-item")||{})),null===this.options.width&&this._resizeButton()},_refreshMenu:function(){var t,e=this.element.find("option");this.menu.empty(),this._parseOptions(e),this._renderMenu(this.menu,this.items),this.menuInstance.refresh(),this.menuItems=this.menu.find("li").not(".ui-selectmenu-optgroup").find(".ui-menu-item-wrapper"),this._rendered=!0,e.length&&(t=this._getSelectedItem(),this.menuInstance.focus(null,t),this._setAria(t.data("ui-selectmenu-item")),this._setOption("disabled",this.element.prop("disabled")))},open:function(t){this.options.disabled||(this._rendered?(this._removeClass(this.menu.find(".ui-state-active"),null,"ui-state-active"),this.menuInstance.focus(null,this._getSelectedItem())):this._refreshMenu(),this.menuItems.length&&(this.isOpen=!0,this._toggleAttr(),this._resizeMenu(),this._position(),this._on(this.document,this._documentClick),this._trigger("open",t)))},_position:function(){this.menuWrap.position(t.extend({of:this.button},this.options.position))},close:function(t){this.isOpen&&(this.isOpen=!1,this._toggleAttr(),this.range=null,this._off(this.document),this._trigger("close",t))},widget:function(){return this.button},menuWidget:function(){return this.menu},_renderButtonItem:function(e){var i=t("<span>");return this._setText(i,e.label),this._addClass(i,"ui-selectmenu-text"),i},_renderMenu:function(e,i){var s=this,n="";t.each(i,function(i,o){var a;o.optgroup!==n&&(a=t("<li>",{text:o.optgroup}),s._addClass(a,"ui-selectmenu-optgroup","ui-menu-divider"+(o.element.parent("optgroup").prop("disabled")?" ui-state-disabled":"")),a.appendTo(e),n=o.optgroup),s._renderItemData(e,o)})},_renderItemData:function(t,e){return this._renderItem(t,e).data("ui-selectmenu-item",e)},_renderItem:function(e,i){var s=t("<li>"),n=t("<div>",{title:i.element.attr("title")});return i.disabled&&this._addClass(s,null,"ui-state-disabled"),this._setText(n,i.label),s.append(n).appendTo(e)},_setText:function(t,e){e?t.text(e):t.html("&#160;")},_move:function(t,e){var i,s,n=".ui-menu-item";this.isOpen?i=this.menuItems.eq(this.focusIndex).parent("li"):(i=this.menuItems.eq(this.element[0].selectedIndex).parent("li"),n+=":not(.ui-state-disabled)"),s="first"===t||"last"===t?i["first"===t?"prevAll":"nextAll"](n).eq(-1):i[t+"All"](n).eq(0),s.length&&this.menuInstance.focus(e,s)},_getSelectedItem:function(){return this.menuItems.eq(this.element[0].selectedIndex).parent("li")},_toggle:function(t){this[this.isOpen?"close":"open"](t)},_setSelection:function(){var t;this.range&&(window.getSelection?(t=window.getSelection(),t.removeAllRanges(),t.addRange(this.range)):this.range.select(),this.button.focus())},_documentClick:{mousedown:function(e){this.isOpen&&(t(e.target).closest(".ui-selectmenu-menu, #"+t.ui.escapeSelector(this.ids.button)).length||this.close(e))}},_buttonEvents:{mousedown:function(){var t;window.getSelection?(t=window.getSelection(),t.rangeCount&&(this.range=t.getRangeAt(0))):this.range=document.selection.createRange()},click:function(t){this._setSelection(),this._toggle(t)},keydown:function(e){var i=!0;switch(e.keyCode){case t.ui.keyCode.TAB:case t.ui.keyCode.ESCAPE:this.close(e),i=!1;break;case t.ui.keyCode.ENTER:this.isOpen&&this._selectFocusedItem(e);break;case t.ui.keyCode.UP:e.altKey?this._toggle(e):this._move("prev",e);break;case t.ui.keyCode.DOWN:e.altKey?this._toggle(e):this._move("next",e);break;case t.ui.keyCode.SPACE:this.isOpen?this._selectFocusedItem(e):this._toggle(e);break;case t.ui.keyCode.LEFT:this._move("prev",e);break;case t.ui.keyCode.RIGHT:this._move("next",e);break;case t.ui.keyCode.HOME:case t.ui.keyCode.PAGE_UP:this._move("first",e);break;case t.ui.keyCode.END:case t.ui.keyCode.PAGE_DOWN:this._move("last",e);break;default:this.menu.trigger(e),i=!1}i&&e.preventDefault()}},_selectFocusedItem:function(t){var e=this.menuItems.eq(this.focusIndex).parent("li");e.hasClass("ui-state-disabled")||this._select(e.data("ui-selectmenu-item"),t)},_select:function(t,e){var i=this.element[0].selectedIndex;this.element[0].selectedIndex=t.index,this.buttonItem.replaceWith(this.buttonItem=this._renderButtonItem(t)),this._setAria(t),this._trigger("select",e,{item:t}),t.index!==i&&this._trigger("change",e,{item:t}),this.close(e)},_setAria:function(t){var e=this.menuItems.eq(t.index).attr("id");this.button.attr({"aria-labelledby":e,"aria-activedescendant":e}),this.menu.attr("aria-activedescendant",e)},_setOption:function(t,e){if("icons"===t){var i=this.button.find("span.ui-icon");this._removeClass(i,null,this.options.icons.button)._addClass(i,null,e.button)}this._super(t,e),"appendTo"===t&&this.menuWrap.appendTo(this._appendTo()),"width"===t&&this._resizeButton()},_setOptionDisabled:function(t){this._super(t),this.menuInstance.option("disabled",t),this.button.attr("aria-disabled",t),this._toggleClass(this.button,null,"ui-state-disabled",t),this.element.prop("disabled",t),t?(this.button.attr("tabindex",-1),this.close()):this.button.attr("tabindex",0)},_appendTo:function(){var e=this.options.appendTo;return e&&(e=e.jquery||e.nodeType?t(e):this.document.find(e).eq(0)),e&&e[0]||(e=this.element.closest(".ui-front, dialog")),e.length||(e=this.document[0].body),e},_toggleAttr:function(){this.button.attr("aria-expanded",this.isOpen),this._removeClass(this.button,"ui-selectmenu-button-"+(this.isOpen?"closed":"open"))._addClass(this.button,"ui-selectmenu-button-"+(this.isOpen?"open":"closed"))._toggleClass(this.menuWrap,"ui-selectmenu-open",null,this.isOpen),this.menu.attr("aria-hidden",!this.isOpen)},_resizeButton:function(){var t=this.options.width;return t===!1?(this.button.css("width",""),void 0):(null===t&&(t=this.element.show().outerWidth(),this.element.hide()),this.button.outerWidth(t),void 0)},_resizeMenu:function(){this.menu.outerWidth(Math.max(this.button.outerWidth(),this.menu.width("").outerWidth()+1))},_getCreateOptions:function(){var t=this._super();return t.disabled=this.element.prop("disabled"),t},_parseOptions:function(e){var i=this,s=[];e.each(function(e,n){s.push(i._parseOption(t(n),e))}),this.items=s},_parseOption:function(t,e){var i=t.parent("optgroup");return{element:t,index:e,value:t.val(),label:t.text(),optgroup:i.attr("label")||"",disabled:i.prop("disabled")||t.prop("disabled")}},_destroy:function(){this._unbindFormResetHandler(),this.menuWrap.remove(),this.button.remove(),this.element.show(),this.element.removeUniqueId(),this.labels.attr("for",this.ids.element)}}]),t.widget("ui.slider",t.ui.mouse,{version:"1.12.1",widgetEventPrefix:"slide",options:{animate:!1,classes:{"ui-slider":"ui-corner-all","ui-slider-handle":"ui-corner-all","ui-slider-range":"ui-corner-all ui-widget-header"},distance:0,max:100,min:0,orientation:"horizontal",range:!1,step:1,value:0,values:null,change:null,slide:null,start:null,stop:null},numPages:5,_create:function(){this._keySliding=!1,this._mouseSliding=!1,this._animateOff=!0,this._handleIndex=null,this._detectOrientation(),this._mouseInit(),this._calculateNewMax(),this._addClass("ui-slider ui-slider-"+this.orientation,"ui-widget ui-widget-content"),this._refresh(),this._animateOff=!1 -},_refresh:function(){this._createRange(),this._createHandles(),this._setupEvents(),this._refreshValue()},_createHandles:function(){var e,i,s=this.options,n=this.element.find(".ui-slider-handle"),o="<span tabindex='0'></span>",a=[];for(i=s.values&&s.values.length||1,n.length>i&&(n.slice(i).remove(),n=n.slice(0,i)),e=n.length;i>e;e++)a.push(o);this.handles=n.add(t(a.join("")).appendTo(this.element)),this._addClass(this.handles,"ui-slider-handle","ui-state-default"),this.handle=this.handles.eq(0),this.handles.each(function(e){t(this).data("ui-slider-handle-index",e).attr("tabIndex",0)})},_createRange:function(){var e=this.options;e.range?(e.range===!0&&(e.values?e.values.length&&2!==e.values.length?e.values=[e.values[0],e.values[0]]:t.isArray(e.values)&&(e.values=e.values.slice(0)):e.values=[this._valueMin(),this._valueMin()]),this.range&&this.range.length?(this._removeClass(this.range,"ui-slider-range-min ui-slider-range-max"),this.range.css({left:"",bottom:""})):(this.range=t("<div>").appendTo(this.element),this._addClass(this.range,"ui-slider-range")),("min"===e.range||"max"===e.range)&&this._addClass(this.range,"ui-slider-range-"+e.range)):(this.range&&this.range.remove(),this.range=null)},_setupEvents:function(){this._off(this.handles),this._on(this.handles,this._handleEvents),this._hoverable(this.handles),this._focusable(this.handles)},_destroy:function(){this.handles.remove(),this.range&&this.range.remove(),this._mouseDestroy()},_mouseCapture:function(e){var i,s,n,o,a,r,h,l,c=this,u=this.options;return u.disabled?!1:(this.elementSize={width:this.element.outerWidth(),height:this.element.outerHeight()},this.elementOffset=this.element.offset(),i={x:e.pageX,y:e.pageY},s=this._normValueFromMouse(i),n=this._valueMax()-this._valueMin()+1,this.handles.each(function(e){var i=Math.abs(s-c.values(e));(n>i||n===i&&(e===c._lastChangedValue||c.values(e)===u.min))&&(n=i,o=t(this),a=e)}),r=this._start(e,a),r===!1?!1:(this._mouseSliding=!0,this._handleIndex=a,this._addClass(o,null,"ui-state-active"),o.trigger("focus"),h=o.offset(),l=!t(e.target).parents().addBack().is(".ui-slider-handle"),this._clickOffset=l?{left:0,top:0}:{left:e.pageX-h.left-o.width()/2,top:e.pageY-h.top-o.height()/2-(parseInt(o.css("borderTopWidth"),10)||0)-(parseInt(o.css("borderBottomWidth"),10)||0)+(parseInt(o.css("marginTop"),10)||0)},this.handles.hasClass("ui-state-hover")||this._slide(e,a,s),this._animateOff=!0,!0))},_mouseStart:function(){return!0},_mouseDrag:function(t){var e={x:t.pageX,y:t.pageY},i=this._normValueFromMouse(e);return this._slide(t,this._handleIndex,i),!1},_mouseStop:function(t){return this._removeClass(this.handles,null,"ui-state-active"),this._mouseSliding=!1,this._stop(t,this._handleIndex),this._change(t,this._handleIndex),this._handleIndex=null,this._clickOffset=null,this._animateOff=!1,!1},_detectOrientation:function(){this.orientation="vertical"===this.options.orientation?"vertical":"horizontal"},_normValueFromMouse:function(t){var e,i,s,n,o;return"horizontal"===this.orientation?(e=this.elementSize.width,i=t.x-this.elementOffset.left-(this._clickOffset?this._clickOffset.left:0)):(e=this.elementSize.height,i=t.y-this.elementOffset.top-(this._clickOffset?this._clickOffset.top:0)),s=i/e,s>1&&(s=1),0>s&&(s=0),"vertical"===this.orientation&&(s=1-s),n=this._valueMax()-this._valueMin(),o=this._valueMin()+s*n,this._trimAlignValue(o)},_uiHash:function(t,e,i){var s={handle:this.handles[t],handleIndex:t,value:void 0!==e?e:this.value()};return this._hasMultipleValues()&&(s.value=void 0!==e?e:this.values(t),s.values=i||this.values()),s},_hasMultipleValues:function(){return this.options.values&&this.options.values.length},_start:function(t,e){return this._trigger("start",t,this._uiHash(e))},_slide:function(t,e,i){var s,n,o=this.value(),a=this.values();this._hasMultipleValues()&&(n=this.values(e?0:1),o=this.values(e),2===this.options.values.length&&this.options.range===!0&&(i=0===e?Math.min(n,i):Math.max(n,i)),a[e]=i),i!==o&&(s=this._trigger("slide",t,this._uiHash(e,i,a)),s!==!1&&(this._hasMultipleValues()?this.values(e,i):this.value(i)))},_stop:function(t,e){this._trigger("stop",t,this._uiHash(e))},_change:function(t,e){this._keySliding||this._mouseSliding||(this._lastChangedValue=e,this._trigger("change",t,this._uiHash(e)))},value:function(t){return arguments.length?(this.options.value=this._trimAlignValue(t),this._refreshValue(),this._change(null,0),void 0):this._value()},values:function(e,i){var s,n,o;if(arguments.length>1)return this.options.values[e]=this._trimAlignValue(i),this._refreshValue(),this._change(null,e),void 0;if(!arguments.length)return this._values();if(!t.isArray(arguments[0]))return this._hasMultipleValues()?this._values(e):this.value();for(s=this.options.values,n=arguments[0],o=0;s.length>o;o+=1)s[o]=this._trimAlignValue(n[o]),this._change(null,o);this._refreshValue()},_setOption:function(e,i){var s,n=0;switch("range"===e&&this.options.range===!0&&("min"===i?(this.options.value=this._values(0),this.options.values=null):"max"===i&&(this.options.value=this._values(this.options.values.length-1),this.options.values=null)),t.isArray(this.options.values)&&(n=this.options.values.length),this._super(e,i),e){case"orientation":this._detectOrientation(),this._removeClass("ui-slider-horizontal ui-slider-vertical")._addClass("ui-slider-"+this.orientation),this._refreshValue(),this.options.range&&this._refreshRange(i),this.handles.css("horizontal"===i?"bottom":"left","");break;case"value":this._animateOff=!0,this._refreshValue(),this._change(null,0),this._animateOff=!1;break;case"values":for(this._animateOff=!0,this._refreshValue(),s=n-1;s>=0;s--)this._change(null,s);this._animateOff=!1;break;case"step":case"min":case"max":this._animateOff=!0,this._calculateNewMax(),this._refreshValue(),this._animateOff=!1;break;case"range":this._animateOff=!0,this._refresh(),this._animateOff=!1}},_setOptionDisabled:function(t){this._super(t),this._toggleClass(null,"ui-state-disabled",!!t)},_value:function(){var t=this.options.value;return t=this._trimAlignValue(t)},_values:function(t){var e,i,s;if(arguments.length)return e=this.options.values[t],e=this._trimAlignValue(e);if(this._hasMultipleValues()){for(i=this.options.values.slice(),s=0;i.length>s;s+=1)i[s]=this._trimAlignValue(i[s]);return i}return[]},_trimAlignValue:function(t){if(this._valueMin()>=t)return this._valueMin();if(t>=this._valueMax())return this._valueMax();var e=this.options.step>0?this.options.step:1,i=(t-this._valueMin())%e,s=t-i;return 2*Math.abs(i)>=e&&(s+=i>0?e:-e),parseFloat(s.toFixed(5))},_calculateNewMax:function(){var t=this.options.max,e=this._valueMin(),i=this.options.step,s=Math.round((t-e)/i)*i;t=s+e,t>this.options.max&&(t-=i),this.max=parseFloat(t.toFixed(this._precision()))},_precision:function(){var t=this._precisionOf(this.options.step);return null!==this.options.min&&(t=Math.max(t,this._precisionOf(this.options.min))),t},_precisionOf:function(t){var e=""+t,i=e.indexOf(".");return-1===i?0:e.length-i-1},_valueMin:function(){return this.options.min},_valueMax:function(){return this.max},_refreshRange:function(t){"vertical"===t&&this.range.css({width:"",left:""}),"horizontal"===t&&this.range.css({height:"",bottom:""})},_refreshValue:function(){var e,i,s,n,o,a=this.options.range,r=this.options,h=this,l=this._animateOff?!1:r.animate,c={};this._hasMultipleValues()?this.handles.each(function(s){i=100*((h.values(s)-h._valueMin())/(h._valueMax()-h._valueMin())),c["horizontal"===h.orientation?"left":"bottom"]=i+"%",t(this).stop(1,1)[l?"animate":"css"](c,r.animate),h.options.range===!0&&("horizontal"===h.orientation?(0===s&&h.range.stop(1,1)[l?"animate":"css"]({left:i+"%"},r.animate),1===s&&h.range[l?"animate":"css"]({width:i-e+"%"},{queue:!1,duration:r.animate})):(0===s&&h.range.stop(1,1)[l?"animate":"css"]({bottom:i+"%"},r.animate),1===s&&h.range[l?"animate":"css"]({height:i-e+"%"},{queue:!1,duration:r.animate}))),e=i}):(s=this.value(),n=this._valueMin(),o=this._valueMax(),i=o!==n?100*((s-n)/(o-n)):0,c["horizontal"===this.orientation?"left":"bottom"]=i+"%",this.handle.stop(1,1)[l?"animate":"css"](c,r.animate),"min"===a&&"horizontal"===this.orientation&&this.range.stop(1,1)[l?"animate":"css"]({width:i+"%"},r.animate),"max"===a&&"horizontal"===this.orientation&&this.range.stop(1,1)[l?"animate":"css"]({width:100-i+"%"},r.animate),"min"===a&&"vertical"===this.orientation&&this.range.stop(1,1)[l?"animate":"css"]({height:i+"%"},r.animate),"max"===a&&"vertical"===this.orientation&&this.range.stop(1,1)[l?"animate":"css"]({height:100-i+"%"},r.animate))},_handleEvents:{keydown:function(e){var i,s,n,o,a=t(e.target).data("ui-slider-handle-index");switch(e.keyCode){case t.ui.keyCode.HOME:case t.ui.keyCode.END:case t.ui.keyCode.PAGE_UP:case t.ui.keyCode.PAGE_DOWN:case t.ui.keyCode.UP:case t.ui.keyCode.RIGHT:case t.ui.keyCode.DOWN:case t.ui.keyCode.LEFT:if(e.preventDefault(),!this._keySliding&&(this._keySliding=!0,this._addClass(t(e.target),null,"ui-state-active"),i=this._start(e,a),i===!1))return}switch(o=this.options.step,s=n=this._hasMultipleValues()?this.values(a):this.value(),e.keyCode){case t.ui.keyCode.HOME:n=this._valueMin();break;case t.ui.keyCode.END:n=this._valueMax();break;case t.ui.keyCode.PAGE_UP:n=this._trimAlignValue(s+(this._valueMax()-this._valueMin())/this.numPages);break;case t.ui.keyCode.PAGE_DOWN:n=this._trimAlignValue(s-(this._valueMax()-this._valueMin())/this.numPages);break;case t.ui.keyCode.UP:case t.ui.keyCode.RIGHT:if(s===this._valueMax())return;n=this._trimAlignValue(s+o);break;case t.ui.keyCode.DOWN:case t.ui.keyCode.LEFT:if(s===this._valueMin())return;n=this._trimAlignValue(s-o)}this._slide(e,a,n)},keyup:function(e){var i=t(e.target).data("ui-slider-handle-index");this._keySliding&&(this._keySliding=!1,this._stop(e,i),this._change(e,i),this._removeClass(t(e.target),null,"ui-state-active"))}}}),t.widget("ui.sortable",t.ui.mouse,{version:"1.12.1",widgetEventPrefix:"sort",ready:!1,options:{appendTo:"parent",axis:!1,connectWith:!1,containment:!1,cursor:"auto",cursorAt:!1,dropOnEmpty:!0,forcePlaceholderSize:!1,forceHelperSize:!1,grid:!1,handle:!1,helper:"original",items:"> *",opacity:!1,placeholder:!1,revert:!1,scroll:!0,scrollSensitivity:20,scrollSpeed:20,scope:"default",tolerance:"intersect",zIndex:1e3,activate:null,beforeStop:null,change:null,deactivate:null,out:null,over:null,receive:null,remove:null,sort:null,start:null,stop:null,update:null},_isOverAxis:function(t,e,i){return t>=e&&e+i>t},_isFloating:function(t){return/left|right/.test(t.css("float"))||/inline|table-cell/.test(t.css("display"))},_create:function(){this.containerCache={},this._addClass("ui-sortable"),this.refresh(),this.offset=this.element.offset(),this._mouseInit(),this._setHandleClassName(),this.ready=!0},_setOption:function(t,e){this._super(t,e),"handle"===t&&this._setHandleClassName()},_setHandleClassName:function(){var e=this;this._removeClass(this.element.find(".ui-sortable-handle"),"ui-sortable-handle"),t.each(this.items,function(){e._addClass(this.instance.options.handle?this.item.find(this.instance.options.handle):this.item,"ui-sortable-handle")})},_destroy:function(){this._mouseDestroy();for(var t=this.items.length-1;t>=0;t--)this.items[t].item.removeData(this.widgetName+"-item");return this},_mouseCapture:function(e,i){var s=null,n=!1,o=this;return this.reverting?!1:this.options.disabled||"static"===this.options.type?!1:(this._refreshItems(e),t(e.target).parents().each(function(){return t.data(this,o.widgetName+"-item")===o?(s=t(this),!1):void 0}),t.data(e.target,o.widgetName+"-item")===o&&(s=t(e.target)),s?!this.options.handle||i||(t(this.options.handle,s).find("*").addBack().each(function(){this===e.target&&(n=!0)}),n)?(this.currentItem=s,this._removeCurrentsFromItems(),!0):!1:!1)},_mouseStart:function(e,i,s){var n,o,a=this.options;if(this.currentContainer=this,this.refreshPositions(),this.helper=this._createHelper(e),this._cacheHelperProportions(),this._cacheMargins(),this.scrollParent=this.helper.scrollParent(),this.offset=this.currentItem.offset(),this.offset={top:this.offset.top-this.margins.top,left:this.offset.left-this.margins.left},t.extend(this.offset,{click:{left:e.pageX-this.offset.left,top:e.pageY-this.offset.top},parent:this._getParentOffset(),relative:this._getRelativeOffset()}),this.helper.css("position","absolute"),this.cssPosition=this.helper.css("position"),this.originalPosition=this._generatePosition(e),this.originalPageX=e.pageX,this.originalPageY=e.pageY,a.cursorAt&&this._adjustOffsetFromHelper(a.cursorAt),this.domPosition={prev:this.currentItem.prev()[0],parent:this.currentItem.parent()[0]},this.helper[0]!==this.currentItem[0]&&this.currentItem.hide(),this._createPlaceholder(),a.containment&&this._setContainment(),a.cursor&&"auto"!==a.cursor&&(o=this.document.find("body"),this.storedCursor=o.css("cursor"),o.css("cursor",a.cursor),this.storedStylesheet=t("<style>*{ cursor: "+a.cursor+" !important; }</style>").appendTo(o)),a.opacity&&(this.helper.css("opacity")&&(this._storedOpacity=this.helper.css("opacity")),this.helper.css("opacity",a.opacity)),a.zIndex&&(this.helper.css("zIndex")&&(this._storedZIndex=this.helper.css("zIndex")),this.helper.css("zIndex",a.zIndex)),this.scrollParent[0]!==this.document[0]&&"HTML"!==this.scrollParent[0].tagName&&(this.overflowOffset=this.scrollParent.offset()),this._trigger("start",e,this._uiHash()),this._preserveHelperProportions||this._cacheHelperProportions(),!s)for(n=this.containers.length-1;n>=0;n--)this.containers[n]._trigger("activate",e,this._uiHash(this));return t.ui.ddmanager&&(t.ui.ddmanager.current=this),t.ui.ddmanager&&!a.dropBehaviour&&t.ui.ddmanager.prepareOffsets(this,e),this.dragging=!0,this._addClass(this.helper,"ui-sortable-helper"),this._mouseDrag(e),!0},_mouseDrag:function(e){var i,s,n,o,a=this.options,r=!1;for(this.position=this._generatePosition(e),this.positionAbs=this._convertPositionTo("absolute"),this.lastPositionAbs||(this.lastPositionAbs=this.positionAbs),this.options.scroll&&(this.scrollParent[0]!==this.document[0]&&"HTML"!==this.scrollParent[0].tagName?(this.overflowOffset.top+this.scrollParent[0].offsetHeight-e.pageY<a.scrollSensitivity?this.scrollParent[0].scrollTop=r=this.scrollParent[0].scrollTop+a.scrollSpeed:e.pageY-this.overflowOffset.top<a.scrollSensitivity&&(this.scrollParent[0].scrollTop=r=this.scrollParent[0].scrollTop-a.scrollSpeed),this.overflowOffset.left+this.scrollParent[0].offsetWidth-e.pageX<a.scrollSensitivity?this.scrollParent[0].scrollLeft=r=this.scrollParent[0].scrollLeft+a.scrollSpeed:e.pageX-this.overflowOffset.left<a.scrollSensitivity&&(this.scrollParent[0].scrollLeft=r=this.scrollParent[0].scrollLeft-a.scrollSpeed)):(e.pageY-this.document.scrollTop()<a.scrollSensitivity?r=this.document.scrollTop(this.document.scrollTop()-a.scrollSpeed):this.window.height()-(e.pageY-this.document.scrollTop())<a.scrollSensitivity&&(r=this.document.scrollTop(this.document.scrollTop()+a.scrollSpeed)),e.pageX-this.document.scrollLeft()<a.scrollSensitivity?r=this.document.scrollLeft(this.document.scrollLeft()-a.scrollSpeed):this.window.width()-(e.pageX-this.document.scrollLeft())<a.scrollSensitivity&&(r=this.document.scrollLeft(this.document.scrollLeft()+a.scrollSpeed))),r!==!1&&t.ui.ddmanager&&!a.dropBehaviour&&t.ui.ddmanager.prepareOffsets(this,e)),this.positionAbs=this._convertPositionTo("absolute"),this.options.axis&&"y"===this.options.axis||(this.helper[0].style.left=this.position.left+"px"),this.options.axis&&"x"===this.options.axis||(this.helper[0].style.top=this.position.top+"px"),i=this.items.length-1;i>=0;i--)if(s=this.items[i],n=s.item[0],o=this._intersectsWithPointer(s),o&&s.instance===this.currentContainer&&n!==this.currentItem[0]&&this.placeholder[1===o?"next":"prev"]()[0]!==n&&!t.contains(this.placeholder[0],n)&&("semi-dynamic"===this.options.type?!t.contains(this.element[0],n):!0)){if(this.direction=1===o?"down":"up","pointer"!==this.options.tolerance&&!this._intersectsWithSides(s))break;this._rearrange(e,s),this._trigger("change",e,this._uiHash());break}return this._contactContainers(e),t.ui.ddmanager&&t.ui.ddmanager.drag(this,e),this._trigger("sort",e,this._uiHash()),this.lastPositionAbs=this.positionAbs,!1},_mouseStop:function(e,i){if(e){if(t.ui.ddmanager&&!this.options.dropBehaviour&&t.ui.ddmanager.drop(this,e),this.options.revert){var s=this,n=this.placeholder.offset(),o=this.options.axis,a={};o&&"x"!==o||(a.left=n.left-this.offset.parent.left-this.margins.left+(this.offsetParent[0]===this.document[0].body?0:this.offsetParent[0].scrollLeft)),o&&"y"!==o||(a.top=n.top-this.offset.parent.top-this.margins.top+(this.offsetParent[0]===this.document[0].body?0:this.offsetParent[0].scrollTop)),this.reverting=!0,t(this.helper).animate(a,parseInt(this.options.revert,10)||500,function(){s._clear(e)})}else this._clear(e,i);return!1}},cancel:function(){if(this.dragging){this._mouseUp(new t.Event("mouseup",{target:null})),"original"===this.options.helper?(this.currentItem.css(this._storedCSS),this._removeClass(this.currentItem,"ui-sortable-helper")):this.currentItem.show();for(var e=this.containers.length-1;e>=0;e--)this.containers[e]._trigger("deactivate",null,this._uiHash(this)),this.containers[e].containerCache.over&&(this.containers[e]._trigger("out",null,this._uiHash(this)),this.containers[e].containerCache.over=0)}return this.placeholder&&(this.placeholder[0].parentNode&&this.placeholder[0].parentNode.removeChild(this.placeholder[0]),"original"!==this.options.helper&&this.helper&&this.helper[0].parentNode&&this.helper.remove(),t.extend(this,{helper:null,dragging:!1,reverting:!1,_noFinalSort:null}),this.domPosition.prev?t(this.domPosition.prev).after(this.currentItem):t(this.domPosition.parent).prepend(this.currentItem)),this},serialize:function(e){var i=this._getItemsAsjQuery(e&&e.connected),s=[];return e=e||{},t(i).each(function(){var i=(t(e.item||this).attr(e.attribute||"id")||"").match(e.expression||/(.+)[\-=_](.+)/);i&&s.push((e.key||i[1]+"[]")+"="+(e.key&&e.expression?i[1]:i[2]))}),!s.length&&e.key&&s.push(e.key+"="),s.join("&")},toArray:function(e){var i=this._getItemsAsjQuery(e&&e.connected),s=[];return e=e||{},i.each(function(){s.push(t(e.item||this).attr(e.attribute||"id")||"")}),s},_intersectsWith:function(t){var e=this.positionAbs.left,i=e+this.helperProportions.width,s=this.positionAbs.top,n=s+this.helperProportions.height,o=t.left,a=o+t.width,r=t.top,h=r+t.height,l=this.offset.click.top,c=this.offset.click.left,u="x"===this.options.axis||s+l>r&&h>s+l,d="y"===this.options.axis||e+c>o&&a>e+c,p=u&&d;return"pointer"===this.options.tolerance||this.options.forcePointerForContainers||"pointer"!==this.options.tolerance&&this.helperProportions[this.floating?"width":"height"]>t[this.floating?"width":"height"]?p:e+this.helperProportions.width/2>o&&a>i-this.helperProportions.width/2&&s+this.helperProportions.height/2>r&&h>n-this.helperProportions.height/2},_intersectsWithPointer:function(t){var e,i,s="x"===this.options.axis||this._isOverAxis(this.positionAbs.top+this.offset.click.top,t.top,t.height),n="y"===this.options.axis||this._isOverAxis(this.positionAbs.left+this.offset.click.left,t.left,t.width),o=s&&n;return o?(e=this._getDragVerticalDirection(),i=this._getDragHorizontalDirection(),this.floating?"right"===i||"down"===e?2:1:e&&("down"===e?2:1)):!1},_intersectsWithSides:function(t){var e=this._isOverAxis(this.positionAbs.top+this.offset.click.top,t.top+t.height/2,t.height),i=this._isOverAxis(this.positionAbs.left+this.offset.click.left,t.left+t.width/2,t.width),s=this._getDragVerticalDirection(),n=this._getDragHorizontalDirection();return this.floating&&n?"right"===n&&i||"left"===n&&!i:s&&("down"===s&&e||"up"===s&&!e)},_getDragVerticalDirection:function(){var t=this.positionAbs.top-this.lastPositionAbs.top;return 0!==t&&(t>0?"down":"up")},_getDragHorizontalDirection:function(){var t=this.positionAbs.left-this.lastPositionAbs.left;return 0!==t&&(t>0?"right":"left")},refresh:function(t){return this._refreshItems(t),this._setHandleClassName(),this.refreshPositions(),this},_connectWith:function(){var t=this.options;return t.connectWith.constructor===String?[t.connectWith]:t.connectWith},_getItemsAsjQuery:function(e){function i(){r.push(this)}var s,n,o,a,r=[],h=[],l=this._connectWith();if(l&&e)for(s=l.length-1;s>=0;s--)for(o=t(l[s],this.document[0]),n=o.length-1;n>=0;n--)a=t.data(o[n],this.widgetFullName),a&&a!==this&&!a.options.disabled&&h.push([t.isFunction(a.options.items)?a.options.items.call(a.element):t(a.options.items,a.element).not(".ui-sortable-helper").not(".ui-sortable-placeholder"),a]);for(h.push([t.isFunction(this.options.items)?this.options.items.call(this.element,null,{options:this.options,item:this.currentItem}):t(this.options.items,this.element).not(".ui-sortable-helper").not(".ui-sortable-placeholder"),this]),s=h.length-1;s>=0;s--)h[s][0].each(i);return t(r)},_removeCurrentsFromItems:function(){var e=this.currentItem.find(":data("+this.widgetName+"-item)");this.items=t.grep(this.items,function(t){for(var i=0;e.length>i;i++)if(e[i]===t.item[0])return!1;return!0})},_refreshItems:function(e){this.items=[],this.containers=[this];var i,s,n,o,a,r,h,l,c=this.items,u=[[t.isFunction(this.options.items)?this.options.items.call(this.element[0],e,{item:this.currentItem}):t(this.options.items,this.element),this]],d=this._connectWith();if(d&&this.ready)for(i=d.length-1;i>=0;i--)for(n=t(d[i],this.document[0]),s=n.length-1;s>=0;s--)o=t.data(n[s],this.widgetFullName),o&&o!==this&&!o.options.disabled&&(u.push([t.isFunction(o.options.items)?o.options.items.call(o.element[0],e,{item:this.currentItem}):t(o.options.items,o.element),o]),this.containers.push(o));for(i=u.length-1;i>=0;i--)for(a=u[i][1],r=u[i][0],s=0,l=r.length;l>s;s++)h=t(r[s]),h.data(this.widgetName+"-item",a),c.push({item:h,instance:a,width:0,height:0,left:0,top:0})},refreshPositions:function(e){this.floating=this.items.length?"x"===this.options.axis||this._isFloating(this.items[0].item):!1,this.offsetParent&&this.helper&&(this.offset.parent=this._getParentOffset());var i,s,n,o;for(i=this.items.length-1;i>=0;i--)s=this.items[i],s.instance!==this.currentContainer&&this.currentContainer&&s.item[0]!==this.currentItem[0]||(n=this.options.toleranceElement?t(this.options.toleranceElement,s.item):s.item,e||(s.width=n.outerWidth(),s.height=n.outerHeight()),o=n.offset(),s.left=o.left,s.top=o.top);if(this.options.custom&&this.options.custom.refreshContainers)this.options.custom.refreshContainers.call(this);else for(i=this.containers.length-1;i>=0;i--)o=this.containers[i].element.offset(),this.containers[i].containerCache.left=o.left,this.containers[i].containerCache.top=o.top,this.containers[i].containerCache.width=this.containers[i].element.outerWidth(),this.containers[i].containerCache.height=this.containers[i].element.outerHeight();return this},_createPlaceholder:function(e){e=e||this;var i,s=e.options;s.placeholder&&s.placeholder.constructor!==String||(i=s.placeholder,s.placeholder={element:function(){var s=e.currentItem[0].nodeName.toLowerCase(),n=t("<"+s+">",e.document[0]);return e._addClass(n,"ui-sortable-placeholder",i||e.currentItem[0].className)._removeClass(n,"ui-sortable-helper"),"tbody"===s?e._createTrPlaceholder(e.currentItem.find("tr").eq(0),t("<tr>",e.document[0]).appendTo(n)):"tr"===s?e._createTrPlaceholder(e.currentItem,n):"img"===s&&n.attr("src",e.currentItem.attr("src")),i||n.css("visibility","hidden"),n},update:function(t,n){(!i||s.forcePlaceholderSize)&&(n.height()||n.height(e.currentItem.innerHeight()-parseInt(e.currentItem.css("paddingTop")||0,10)-parseInt(e.currentItem.css("paddingBottom")||0,10)),n.width()||n.width(e.currentItem.innerWidth()-parseInt(e.currentItem.css("paddingLeft")||0,10)-parseInt(e.currentItem.css("paddingRight")||0,10)))}}),e.placeholder=t(s.placeholder.element.call(e.element,e.currentItem)),e.currentItem.after(e.placeholder),s.placeholder.update(e,e.placeholder)},_createTrPlaceholder:function(e,i){var s=this;e.children().each(function(){t("<td>&#160;</td>",s.document[0]).attr("colspan",t(this).attr("colspan")||1).appendTo(i)})},_contactContainers:function(e){var i,s,n,o,a,r,h,l,c,u,d=null,p=null;for(i=this.containers.length-1;i>=0;i--)if(!t.contains(this.currentItem[0],this.containers[i].element[0]))if(this._intersectsWith(this.containers[i].containerCache)){if(d&&t.contains(this.containers[i].element[0],d.element[0]))continue;d=this.containers[i],p=i}else this.containers[i].containerCache.over&&(this.containers[i]._trigger("out",e,this._uiHash(this)),this.containers[i].containerCache.over=0);if(d)if(1===this.containers.length)this.containers[p].containerCache.over||(this.containers[p]._trigger("over",e,this._uiHash(this)),this.containers[p].containerCache.over=1);else{for(n=1e4,o=null,c=d.floating||this._isFloating(this.currentItem),a=c?"left":"top",r=c?"width":"height",u=c?"pageX":"pageY",s=this.items.length-1;s>=0;s--)t.contains(this.containers[p].element[0],this.items[s].item[0])&&this.items[s].item[0]!==this.currentItem[0]&&(h=this.items[s].item.offset()[a],l=!1,e[u]-h>this.items[s][r]/2&&(l=!0),n>Math.abs(e[u]-h)&&(n=Math.abs(e[u]-h),o=this.items[s],this.direction=l?"up":"down"));if(!o&&!this.options.dropOnEmpty)return;if(this.currentContainer===this.containers[p])return this.currentContainer.containerCache.over||(this.containers[p]._trigger("over",e,this._uiHash()),this.currentContainer.containerCache.over=1),void 0;o?this._rearrange(e,o,null,!0):this._rearrange(e,null,this.containers[p].element,!0),this._trigger("change",e,this._uiHash()),this.containers[p]._trigger("change",e,this._uiHash(this)),this.currentContainer=this.containers[p],this.options.placeholder.update(this.currentContainer,this.placeholder),this.containers[p]._trigger("over",e,this._uiHash(this)),this.containers[p].containerCache.over=1}},_createHelper:function(e){var i=this.options,s=t.isFunction(i.helper)?t(i.helper.apply(this.element[0],[e,this.currentItem])):"clone"===i.helper?this.currentItem.clone():this.currentItem;return s.parents("body").length||t("parent"!==i.appendTo?i.appendTo:this.currentItem[0].parentNode)[0].appendChild(s[0]),s[0]===this.currentItem[0]&&(this._storedCSS={width:this.currentItem[0].style.width,height:this.currentItem[0].style.height,position:this.currentItem.css("position"),top:this.currentItem.css("top"),left:this.currentItem.css("left")}),(!s[0].style.width||i.forceHelperSize)&&s.width(this.currentItem.width()),(!s[0].style.height||i.forceHelperSize)&&s.height(this.currentItem.height()),s},_adjustOffsetFromHelper:function(e){"string"==typeof e&&(e=e.split(" ")),t.isArray(e)&&(e={left:+e[0],top:+e[1]||0}),"left"in e&&(this.offset.click.left=e.left+this.margins.left),"right"in e&&(this.offset.click.left=this.helperProportions.width-e.right+this.margins.left),"top"in e&&(this.offset.click.top=e.top+this.margins.top),"bottom"in e&&(this.offset.click.top=this.helperProportions.height-e.bottom+this.margins.top)},_getParentOffset:function(){this.offsetParent=this.helper.offsetParent();var e=this.offsetParent.offset();return"absolute"===this.cssPosition&&this.scrollParent[0]!==this.document[0]&&t.contains(this.scrollParent[0],this.offsetParent[0])&&(e.left+=this.scrollParent.scrollLeft(),e.top+=this.scrollParent.scrollTop()),(this.offsetParent[0]===this.document[0].body||this.offsetParent[0].tagName&&"html"===this.offsetParent[0].tagName.toLowerCase()&&t.ui.ie)&&(e={top:0,left:0}),{top:e.top+(parseInt(this.offsetParent.css("borderTopWidth"),10)||0),left:e.left+(parseInt(this.offsetParent.css("borderLeftWidth"),10)||0)}},_getRelativeOffset:function(){if("relative"===this.cssPosition){var t=this.currentItem.position();return{top:t.top-(parseInt(this.helper.css("top"),10)||0)+this.scrollParent.scrollTop(),left:t.left-(parseInt(this.helper.css("left"),10)||0)+this.scrollParent.scrollLeft()}}return{top:0,left:0}},_cacheMargins:function(){this.margins={left:parseInt(this.currentItem.css("marginLeft"),10)||0,top:parseInt(this.currentItem.css("marginTop"),10)||0}},_cacheHelperProportions:function(){this.helperProportions={width:this.helper.outerWidth(),height:this.helper.outerHeight()}},_setContainment:function(){var e,i,s,n=this.options;"parent"===n.containment&&(n.containment=this.helper[0].parentNode),("document"===n.containment||"window"===n.containment)&&(this.containment=[0-this.offset.relative.left-this.offset.parent.left,0-this.offset.relative.top-this.offset.parent.top,"document"===n.containment?this.document.width():this.window.width()-this.helperProportions.width-this.margins.left,("document"===n.containment?this.document.height()||document.body.parentNode.scrollHeight:this.window.height()||this.document[0].body.parentNode.scrollHeight)-this.helperProportions.height-this.margins.top]),/^(document|window|parent)$/.test(n.containment)||(e=t(n.containment)[0],i=t(n.containment).offset(),s="hidden"!==t(e).css("overflow"),this.containment=[i.left+(parseInt(t(e).css("borderLeftWidth"),10)||0)+(parseInt(t(e).css("paddingLeft"),10)||0)-this.margins.left,i.top+(parseInt(t(e).css("borderTopWidth"),10)||0)+(parseInt(t(e).css("paddingTop"),10)||0)-this.margins.top,i.left+(s?Math.max(e.scrollWidth,e.offsetWidth):e.offsetWidth)-(parseInt(t(e).css("borderLeftWidth"),10)||0)-(parseInt(t(e).css("paddingRight"),10)||0)-this.helperProportions.width-this.margins.left,i.top+(s?Math.max(e.scrollHeight,e.offsetHeight):e.offsetHeight)-(parseInt(t(e).css("borderTopWidth"),10)||0)-(parseInt(t(e).css("paddingBottom"),10)||0)-this.helperProportions.height-this.margins.top])},_convertPositionTo:function(e,i){i||(i=this.position);var s="absolute"===e?1:-1,n="absolute"!==this.cssPosition||this.scrollParent[0]!==this.document[0]&&t.contains(this.scrollParent[0],this.offsetParent[0])?this.scrollParent:this.offsetParent,o=/(html|body)/i.test(n[0].tagName);return{top:i.top+this.offset.relative.top*s+this.offset.parent.top*s-("fixed"===this.cssPosition?-this.scrollParent.scrollTop():o?0:n.scrollTop())*s,left:i.left+this.offset.relative.left*s+this.offset.parent.left*s-("fixed"===this.cssPosition?-this.scrollParent.scrollLeft():o?0:n.scrollLeft())*s}},_generatePosition:function(e){var i,s,n=this.options,o=e.pageX,a=e.pageY,r="absolute"!==this.cssPosition||this.scrollParent[0]!==this.document[0]&&t.contains(this.scrollParent[0],this.offsetParent[0])?this.scrollParent:this.offsetParent,h=/(html|body)/i.test(r[0].tagName);return"relative"!==this.cssPosition||this.scrollParent[0]!==this.document[0]&&this.scrollParent[0]!==this.offsetParent[0]||(this.offset.relative=this._getRelativeOffset()),this.originalPosition&&(this.containment&&(e.pageX-this.offset.click.left<this.containment[0]&&(o=this.containment[0]+this.offset.click.left),e.pageY-this.offset.click.top<this.containment[1]&&(a=this.containment[1]+this.offset.click.top),e.pageX-this.offset.click.left>this.containment[2]&&(o=this.containment[2]+this.offset.click.left),e.pageY-this.offset.click.top>this.containment[3]&&(a=this.containment[3]+this.offset.click.top)),n.grid&&(i=this.originalPageY+Math.round((a-this.originalPageY)/n.grid[1])*n.grid[1],a=this.containment?i-this.offset.click.top>=this.containment[1]&&i-this.offset.click.top<=this.containment[3]?i:i-this.offset.click.top>=this.containment[1]?i-n.grid[1]:i+n.grid[1]:i,s=this.originalPageX+Math.round((o-this.originalPageX)/n.grid[0])*n.grid[0],o=this.containment?s-this.offset.click.left>=this.containment[0]&&s-this.offset.click.left<=this.containment[2]?s:s-this.offset.click.left>=this.containment[0]?s-n.grid[0]:s+n.grid[0]:s)),{top:a-this.offset.click.top-this.offset.relative.top-this.offset.parent.top+("fixed"===this.cssPosition?-this.scrollParent.scrollTop():h?0:r.scrollTop()),left:o-this.offset.click.left-this.offset.relative.left-this.offset.parent.left+("fixed"===this.cssPosition?-this.scrollParent.scrollLeft():h?0:r.scrollLeft())}},_rearrange:function(t,e,i,s){i?i[0].appendChild(this.placeholder[0]):e.item[0].parentNode.insertBefore(this.placeholder[0],"down"===this.direction?e.item[0]:e.item[0].nextSibling),this.counter=this.counter?++this.counter:1;var n=this.counter; -this._delay(function(){n===this.counter&&this.refreshPositions(!s)})},_clear:function(t,e){function i(t,e,i){return function(s){i._trigger(t,s,e._uiHash(e))}}this.reverting=!1;var s,n=[];if(!this._noFinalSort&&this.currentItem.parent().length&&this.placeholder.before(this.currentItem),this._noFinalSort=null,this.helper[0]===this.currentItem[0]){for(s in this._storedCSS)("auto"===this._storedCSS[s]||"static"===this._storedCSS[s])&&(this._storedCSS[s]="");this.currentItem.css(this._storedCSS),this._removeClass(this.currentItem,"ui-sortable-helper")}else this.currentItem.show();for(this.fromOutside&&!e&&n.push(function(t){this._trigger("receive",t,this._uiHash(this.fromOutside))}),!this.fromOutside&&this.domPosition.prev===this.currentItem.prev().not(".ui-sortable-helper")[0]&&this.domPosition.parent===this.currentItem.parent()[0]||e||n.push(function(t){this._trigger("update",t,this._uiHash())}),this!==this.currentContainer&&(e||(n.push(function(t){this._trigger("remove",t,this._uiHash())}),n.push(function(t){return function(e){t._trigger("receive",e,this._uiHash(this))}}.call(this,this.currentContainer)),n.push(function(t){return function(e){t._trigger("update",e,this._uiHash(this))}}.call(this,this.currentContainer)))),s=this.containers.length-1;s>=0;s--)e||n.push(i("deactivate",this,this.containers[s])),this.containers[s].containerCache.over&&(n.push(i("out",this,this.containers[s])),this.containers[s].containerCache.over=0);if(this.storedCursor&&(this.document.find("body").css("cursor",this.storedCursor),this.storedStylesheet.remove()),this._storedOpacity&&this.helper.css("opacity",this._storedOpacity),this._storedZIndex&&this.helper.css("zIndex","auto"===this._storedZIndex?"":this._storedZIndex),this.dragging=!1,e||this._trigger("beforeStop",t,this._uiHash()),this.placeholder[0].parentNode.removeChild(this.placeholder[0]),this.cancelHelperRemoval||(this.helper[0]!==this.currentItem[0]&&this.helper.remove(),this.helper=null),!e){for(s=0;n.length>s;s++)n[s].call(this,t);this._trigger("stop",t,this._uiHash())}return this.fromOutside=!1,!this.cancelHelperRemoval},_trigger:function(){t.Widget.prototype._trigger.apply(this,arguments)===!1&&this.cancel()},_uiHash:function(e){var i=e||this;return{helper:i.helper,placeholder:i.placeholder||t([]),position:i.position,originalPosition:i.originalPosition,offset:i.positionAbs,item:i.currentItem,sender:e?e.element:null}}}),t.widget("ui.spinner",{version:"1.12.1",defaultElement:"<input>",widgetEventPrefix:"spin",options:{classes:{"ui-spinner":"ui-corner-all","ui-spinner-down":"ui-corner-br","ui-spinner-up":"ui-corner-tr"},culture:null,icons:{down:"ui-icon-triangle-1-s",up:"ui-icon-triangle-1-n"},incremental:!0,max:null,min:null,numberFormat:null,page:10,step:1,change:null,spin:null,start:null,stop:null},_create:function(){this._setOption("max",this.options.max),this._setOption("min",this.options.min),this._setOption("step",this.options.step),""!==this.value()&&this._value(this.element.val(),!0),this._draw(),this._on(this._events),this._refresh(),this._on(this.window,{beforeunload:function(){this.element.removeAttr("autocomplete")}})},_getCreateOptions:function(){var e=this._super(),i=this.element;return t.each(["min","max","step"],function(t,s){var n=i.attr(s);null!=n&&n.length&&(e[s]=n)}),e},_events:{keydown:function(t){this._start(t)&&this._keydown(t)&&t.preventDefault()},keyup:"_stop",focus:function(){this.previous=this.element.val()},blur:function(t){return this.cancelBlur?(delete this.cancelBlur,void 0):(this._stop(),this._refresh(),this.previous!==this.element.val()&&this._trigger("change",t),void 0)},mousewheel:function(t,e){if(e){if(!this.spinning&&!this._start(t))return!1;this._spin((e>0?1:-1)*this.options.step,t),clearTimeout(this.mousewheelTimer),this.mousewheelTimer=this._delay(function(){this.spinning&&this._stop(t)},100),t.preventDefault()}},"mousedown .ui-spinner-button":function(e){function i(){var e=this.element[0]===t.ui.safeActiveElement(this.document[0]);e||(this.element.trigger("focus"),this.previous=s,this._delay(function(){this.previous=s}))}var s;s=this.element[0]===t.ui.safeActiveElement(this.document[0])?this.previous:this.element.val(),e.preventDefault(),i.call(this),this.cancelBlur=!0,this._delay(function(){delete this.cancelBlur,i.call(this)}),this._start(e)!==!1&&this._repeat(null,t(e.currentTarget).hasClass("ui-spinner-up")?1:-1,e)},"mouseup .ui-spinner-button":"_stop","mouseenter .ui-spinner-button":function(e){return t(e.currentTarget).hasClass("ui-state-active")?this._start(e)===!1?!1:(this._repeat(null,t(e.currentTarget).hasClass("ui-spinner-up")?1:-1,e),void 0):void 0},"mouseleave .ui-spinner-button":"_stop"},_enhance:function(){this.uiSpinner=this.element.attr("autocomplete","off").wrap("<span>").parent().append("<a></a><a></a>")},_draw:function(){this._enhance(),this._addClass(this.uiSpinner,"ui-spinner","ui-widget ui-widget-content"),this._addClass("ui-spinner-input"),this.element.attr("role","spinbutton"),this.buttons=this.uiSpinner.children("a").attr("tabIndex",-1).attr("aria-hidden",!0).button({classes:{"ui-button":""}}),this._removeClass(this.buttons,"ui-corner-all"),this._addClass(this.buttons.first(),"ui-spinner-button ui-spinner-up"),this._addClass(this.buttons.last(),"ui-spinner-button ui-spinner-down"),this.buttons.first().button({icon:this.options.icons.up,showLabel:!1}),this.buttons.last().button({icon:this.options.icons.down,showLabel:!1}),this.buttons.height()>Math.ceil(.5*this.uiSpinner.height())&&this.uiSpinner.height()>0&&this.uiSpinner.height(this.uiSpinner.height())},_keydown:function(e){var i=this.options,s=t.ui.keyCode;switch(e.keyCode){case s.UP:return this._repeat(null,1,e),!0;case s.DOWN:return this._repeat(null,-1,e),!0;case s.PAGE_UP:return this._repeat(null,i.page,e),!0;case s.PAGE_DOWN:return this._repeat(null,-i.page,e),!0}return!1},_start:function(t){return this.spinning||this._trigger("start",t)!==!1?(this.counter||(this.counter=1),this.spinning=!0,!0):!1},_repeat:function(t,e,i){t=t||500,clearTimeout(this.timer),this.timer=this._delay(function(){this._repeat(40,e,i)},t),this._spin(e*this.options.step,i)},_spin:function(t,e){var i=this.value()||0;this.counter||(this.counter=1),i=this._adjustValue(i+t*this._increment(this.counter)),this.spinning&&this._trigger("spin",e,{value:i})===!1||(this._value(i),this.counter++)},_increment:function(e){var i=this.options.incremental;return i?t.isFunction(i)?i(e):Math.floor(e*e*e/5e4-e*e/500+17*e/200+1):1},_precision:function(){var t=this._precisionOf(this.options.step);return null!==this.options.min&&(t=Math.max(t,this._precisionOf(this.options.min))),t},_precisionOf:function(t){var e=""+t,i=e.indexOf(".");return-1===i?0:e.length-i-1},_adjustValue:function(t){var e,i,s=this.options;return e=null!==s.min?s.min:0,i=t-e,i=Math.round(i/s.step)*s.step,t=e+i,t=parseFloat(t.toFixed(this._precision())),null!==s.max&&t>s.max?s.max:null!==s.min&&s.min>t?s.min:t},_stop:function(t){this.spinning&&(clearTimeout(this.timer),clearTimeout(this.mousewheelTimer),this.counter=0,this.spinning=!1,this._trigger("stop",t))},_setOption:function(t,e){var i,s,n;return"culture"===t||"numberFormat"===t?(i=this._parse(this.element.val()),this.options[t]=e,this.element.val(this._format(i)),void 0):(("max"===t||"min"===t||"step"===t)&&"string"==typeof e&&(e=this._parse(e)),"icons"===t&&(s=this.buttons.first().find(".ui-icon"),this._removeClass(s,null,this.options.icons.up),this._addClass(s,null,e.up),n=this.buttons.last().find(".ui-icon"),this._removeClass(n,null,this.options.icons.down),this._addClass(n,null,e.down)),this._super(t,e),void 0)},_setOptionDisabled:function(t){this._super(t),this._toggleClass(this.uiSpinner,null,"ui-state-disabled",!!t),this.element.prop("disabled",!!t),this.buttons.button(t?"disable":"enable")},_setOptions:r(function(t){this._super(t)}),_parse:function(t){return"string"==typeof t&&""!==t&&(t=window.Globalize&&this.options.numberFormat?Globalize.parseFloat(t,10,this.options.culture):+t),""===t||isNaN(t)?null:t},_format:function(t){return""===t?"":window.Globalize&&this.options.numberFormat?Globalize.format(t,this.options.numberFormat,this.options.culture):t},_refresh:function(){this.element.attr({"aria-valuemin":this.options.min,"aria-valuemax":this.options.max,"aria-valuenow":this._parse(this.element.val())})},isValid:function(){var t=this.value();return null===t?!1:t===this._adjustValue(t)},_value:function(t,e){var i;""!==t&&(i=this._parse(t),null!==i&&(e||(i=this._adjustValue(i)),t=this._format(i))),this.element.val(t),this._refresh()},_destroy:function(){this.element.prop("disabled",!1).removeAttr("autocomplete role aria-valuemin aria-valuemax aria-valuenow"),this.uiSpinner.replaceWith(this.element)},stepUp:r(function(t){this._stepUp(t)}),_stepUp:function(t){this._start()&&(this._spin((t||1)*this.options.step),this._stop())},stepDown:r(function(t){this._stepDown(t)}),_stepDown:function(t){this._start()&&(this._spin((t||1)*-this.options.step),this._stop())},pageUp:r(function(t){this._stepUp((t||1)*this.options.page)}),pageDown:r(function(t){this._stepDown((t||1)*this.options.page)}),value:function(t){return arguments.length?(r(this._value).call(this,t),void 0):this._parse(this.element.val())},widget:function(){return this.uiSpinner}}),t.uiBackCompat!==!1&&t.widget("ui.spinner",t.ui.spinner,{_enhance:function(){this.uiSpinner=this.element.attr("autocomplete","off").wrap(this._uiSpinnerHtml()).parent().append(this._buttonHtml())},_uiSpinnerHtml:function(){return"<span>"},_buttonHtml:function(){return"<a></a><a></a>"}}),t.ui.spinner,t.widget("ui.tabs",{version:"1.12.1",delay:300,options:{active:null,classes:{"ui-tabs":"ui-corner-all","ui-tabs-nav":"ui-corner-all","ui-tabs-panel":"ui-corner-bottom","ui-tabs-tab":"ui-corner-top"},collapsible:!1,event:"click",heightStyle:"content",hide:null,show:null,activate:null,beforeActivate:null,beforeLoad:null,load:null},_isLocal:function(){var t=/#.*$/;return function(e){var i,s;i=e.href.replace(t,""),s=location.href.replace(t,"");try{i=decodeURIComponent(i)}catch(n){}try{s=decodeURIComponent(s)}catch(n){}return e.hash.length>1&&i===s}}(),_create:function(){var e=this,i=this.options;this.running=!1,this._addClass("ui-tabs","ui-widget ui-widget-content"),this._toggleClass("ui-tabs-collapsible",null,i.collapsible),this._processTabs(),i.active=this._initialActive(),t.isArray(i.disabled)&&(i.disabled=t.unique(i.disabled.concat(t.map(this.tabs.filter(".ui-state-disabled"),function(t){return e.tabs.index(t)}))).sort()),this.active=this.options.active!==!1&&this.anchors.length?this._findActive(i.active):t(),this._refresh(),this.active.length&&this.load(i.active)},_initialActive:function(){var e=this.options.active,i=this.options.collapsible,s=location.hash.substring(1);return null===e&&(s&&this.tabs.each(function(i,n){return t(n).attr("aria-controls")===s?(e=i,!1):void 0}),null===e&&(e=this.tabs.index(this.tabs.filter(".ui-tabs-active"))),(null===e||-1===e)&&(e=this.tabs.length?0:!1)),e!==!1&&(e=this.tabs.index(this.tabs.eq(e)),-1===e&&(e=i?!1:0)),!i&&e===!1&&this.anchors.length&&(e=0),e},_getCreateEventData:function(){return{tab:this.active,panel:this.active.length?this._getPanelForTab(this.active):t()}},_tabKeydown:function(e){var i=t(t.ui.safeActiveElement(this.document[0])).closest("li"),s=this.tabs.index(i),n=!0;if(!this._handlePageNav(e)){switch(e.keyCode){case t.ui.keyCode.RIGHT:case t.ui.keyCode.DOWN:s++;break;case t.ui.keyCode.UP:case t.ui.keyCode.LEFT:n=!1,s--;break;case t.ui.keyCode.END:s=this.anchors.length-1;break;case t.ui.keyCode.HOME:s=0;break;case t.ui.keyCode.SPACE:return e.preventDefault(),clearTimeout(this.activating),this._activate(s),void 0;case t.ui.keyCode.ENTER:return e.preventDefault(),clearTimeout(this.activating),this._activate(s===this.options.active?!1:s),void 0;default:return}e.preventDefault(),clearTimeout(this.activating),s=this._focusNextTab(s,n),e.ctrlKey||e.metaKey||(i.attr("aria-selected","false"),this.tabs.eq(s).attr("aria-selected","true"),this.activating=this._delay(function(){this.option("active",s)},this.delay))}},_panelKeydown:function(e){this._handlePageNav(e)||e.ctrlKey&&e.keyCode===t.ui.keyCode.UP&&(e.preventDefault(),this.active.trigger("focus"))},_handlePageNav:function(e){return e.altKey&&e.keyCode===t.ui.keyCode.PAGE_UP?(this._activate(this._focusNextTab(this.options.active-1,!1)),!0):e.altKey&&e.keyCode===t.ui.keyCode.PAGE_DOWN?(this._activate(this._focusNextTab(this.options.active+1,!0)),!0):void 0},_findNextTab:function(e,i){function s(){return e>n&&(e=0),0>e&&(e=n),e}for(var n=this.tabs.length-1;-1!==t.inArray(s(),this.options.disabled);)e=i?e+1:e-1;return e},_focusNextTab:function(t,e){return t=this._findNextTab(t,e),this.tabs.eq(t).trigger("focus"),t},_setOption:function(t,e){return"active"===t?(this._activate(e),void 0):(this._super(t,e),"collapsible"===t&&(this._toggleClass("ui-tabs-collapsible",null,e),e||this.options.active!==!1||this._activate(0)),"event"===t&&this._setupEvents(e),"heightStyle"===t&&this._setupHeightStyle(e),void 0)},_sanitizeSelector:function(t){return t?t.replace(/[!"$%&'()*+,.\/:;<=>?@\[\]\^`{|}~]/g,"\\$&"):""},refresh:function(){var e=this.options,i=this.tablist.children(":has(a[href])");e.disabled=t.map(i.filter(".ui-state-disabled"),function(t){return i.index(t)}),this._processTabs(),e.active!==!1&&this.anchors.length?this.active.length&&!t.contains(this.tablist[0],this.active[0])?this.tabs.length===e.disabled.length?(e.active=!1,this.active=t()):this._activate(this._findNextTab(Math.max(0,e.active-1),!1)):e.active=this.tabs.index(this.active):(e.active=!1,this.active=t()),this._refresh()},_refresh:function(){this._setOptionDisabled(this.options.disabled),this._setupEvents(this.options.event),this._setupHeightStyle(this.options.heightStyle),this.tabs.not(this.active).attr({"aria-selected":"false","aria-expanded":"false",tabIndex:-1}),this.panels.not(this._getPanelForTab(this.active)).hide().attr({"aria-hidden":"true"}),this.active.length?(this.active.attr({"aria-selected":"true","aria-expanded":"true",tabIndex:0}),this._addClass(this.active,"ui-tabs-active","ui-state-active"),this._getPanelForTab(this.active).show().attr({"aria-hidden":"false"})):this.tabs.eq(0).attr("tabIndex",0)},_processTabs:function(){var e=this,i=this.tabs,s=this.anchors,n=this.panels;this.tablist=this._getList().attr("role","tablist"),this._addClass(this.tablist,"ui-tabs-nav","ui-helper-reset ui-helper-clearfix ui-widget-header"),this.tablist.on("mousedown"+this.eventNamespace,"> li",function(e){t(this).is(".ui-state-disabled")&&e.preventDefault()}).on("focus"+this.eventNamespace,".ui-tabs-anchor",function(){t(this).closest("li").is(".ui-state-disabled")&&this.blur()}),this.tabs=this.tablist.find("> li:has(a[href])").attr({role:"tab",tabIndex:-1}),this._addClass(this.tabs,"ui-tabs-tab","ui-state-default"),this.anchors=this.tabs.map(function(){return t("a",this)[0]}).attr({role:"presentation",tabIndex:-1}),this._addClass(this.anchors,"ui-tabs-anchor"),this.panels=t(),this.anchors.each(function(i,s){var n,o,a,r=t(s).uniqueId().attr("id"),h=t(s).closest("li"),l=h.attr("aria-controls");e._isLocal(s)?(n=s.hash,a=n.substring(1),o=e.element.find(e._sanitizeSelector(n))):(a=h.attr("aria-controls")||t({}).uniqueId()[0].id,n="#"+a,o=e.element.find(n),o.length||(o=e._createPanel(a),o.insertAfter(e.panels[i-1]||e.tablist)),o.attr("aria-live","polite")),o.length&&(e.panels=e.panels.add(o)),l&&h.data("ui-tabs-aria-controls",l),h.attr({"aria-controls":a,"aria-labelledby":r}),o.attr("aria-labelledby",r)}),this.panels.attr("role","tabpanel"),this._addClass(this.panels,"ui-tabs-panel","ui-widget-content"),i&&(this._off(i.not(this.tabs)),this._off(s.not(this.anchors)),this._off(n.not(this.panels)))},_getList:function(){return this.tablist||this.element.find("ol, ul").eq(0)},_createPanel:function(e){return t("<div>").attr("id",e).data("ui-tabs-destroy",!0)},_setOptionDisabled:function(e){var i,s,n;for(t.isArray(e)&&(e.length?e.length===this.anchors.length&&(e=!0):e=!1),n=0;s=this.tabs[n];n++)i=t(s),e===!0||-1!==t.inArray(n,e)?(i.attr("aria-disabled","true"),this._addClass(i,null,"ui-state-disabled")):(i.removeAttr("aria-disabled"),this._removeClass(i,null,"ui-state-disabled"));this.options.disabled=e,this._toggleClass(this.widget(),this.widgetFullName+"-disabled",null,e===!0)},_setupEvents:function(e){var i={};e&&t.each(e.split(" "),function(t,e){i[e]="_eventHandler"}),this._off(this.anchors.add(this.tabs).add(this.panels)),this._on(!0,this.anchors,{click:function(t){t.preventDefault()}}),this._on(this.anchors,i),this._on(this.tabs,{keydown:"_tabKeydown"}),this._on(this.panels,{keydown:"_panelKeydown"}),this._focusable(this.tabs),this._hoverable(this.tabs)},_setupHeightStyle:function(e){var i,s=this.element.parent();"fill"===e?(i=s.height(),i-=this.element.outerHeight()-this.element.height(),this.element.siblings(":visible").each(function(){var e=t(this),s=e.css("position");"absolute"!==s&&"fixed"!==s&&(i-=e.outerHeight(!0))}),this.element.children().not(this.panels).each(function(){i-=t(this).outerHeight(!0)}),this.panels.each(function(){t(this).height(Math.max(0,i-t(this).innerHeight()+t(this).height()))}).css("overflow","auto")):"auto"===e&&(i=0,this.panels.each(function(){i=Math.max(i,t(this).height("").height())}).height(i))},_eventHandler:function(e){var i=this.options,s=this.active,n=t(e.currentTarget),o=n.closest("li"),a=o[0]===s[0],r=a&&i.collapsible,h=r?t():this._getPanelForTab(o),l=s.length?this._getPanelForTab(s):t(),c={oldTab:s,oldPanel:l,newTab:r?t():o,newPanel:h};e.preventDefault(),o.hasClass("ui-state-disabled")||o.hasClass("ui-tabs-loading")||this.running||a&&!i.collapsible||this._trigger("beforeActivate",e,c)===!1||(i.active=r?!1:this.tabs.index(o),this.active=a?t():o,this.xhr&&this.xhr.abort(),l.length||h.length||t.error("jQuery UI Tabs: Mismatching fragment identifier."),h.length&&this.load(this.tabs.index(o),e),this._toggle(e,c))},_toggle:function(e,i){function s(){o.running=!1,o._trigger("activate",e,i)}function n(){o._addClass(i.newTab.closest("li"),"ui-tabs-active","ui-state-active"),a.length&&o.options.show?o._show(a,o.options.show,s):(a.show(),s())}var o=this,a=i.newPanel,r=i.oldPanel;this.running=!0,r.length&&this.options.hide?this._hide(r,this.options.hide,function(){o._removeClass(i.oldTab.closest("li"),"ui-tabs-active","ui-state-active"),n()}):(this._removeClass(i.oldTab.closest("li"),"ui-tabs-active","ui-state-active"),r.hide(),n()),r.attr("aria-hidden","true"),i.oldTab.attr({"aria-selected":"false","aria-expanded":"false"}),a.length&&r.length?i.oldTab.attr("tabIndex",-1):a.length&&this.tabs.filter(function(){return 0===t(this).attr("tabIndex")}).attr("tabIndex",-1),a.attr("aria-hidden","false"),i.newTab.attr({"aria-selected":"true","aria-expanded":"true",tabIndex:0})},_activate:function(e){var i,s=this._findActive(e);s[0]!==this.active[0]&&(s.length||(s=this.active),i=s.find(".ui-tabs-anchor")[0],this._eventHandler({target:i,currentTarget:i,preventDefault:t.noop}))},_findActive:function(e){return e===!1?t():this.tabs.eq(e)},_getIndex:function(e){return"string"==typeof e&&(e=this.anchors.index(this.anchors.filter("[href$='"+t.ui.escapeSelector(e)+"']"))),e},_destroy:function(){this.xhr&&this.xhr.abort(),this.tablist.removeAttr("role").off(this.eventNamespace),this.anchors.removeAttr("role tabIndex").removeUniqueId(),this.tabs.add(this.panels).each(function(){t.data(this,"ui-tabs-destroy")?t(this).remove():t(this).removeAttr("role tabIndex aria-live aria-busy aria-selected aria-labelledby aria-hidden aria-expanded")}),this.tabs.each(function(){var e=t(this),i=e.data("ui-tabs-aria-controls");i?e.attr("aria-controls",i).removeData("ui-tabs-aria-controls"):e.removeAttr("aria-controls")}),this.panels.show(),"content"!==this.options.heightStyle&&this.panels.css("height","")},enable:function(e){var i=this.options.disabled;i!==!1&&(void 0===e?i=!1:(e=this._getIndex(e),i=t.isArray(i)?t.map(i,function(t){return t!==e?t:null}):t.map(this.tabs,function(t,i){return i!==e?i:null})),this._setOptionDisabled(i))},disable:function(e){var i=this.options.disabled;if(i!==!0){if(void 0===e)i=!0;else{if(e=this._getIndex(e),-1!==t.inArray(e,i))return;i=t.isArray(i)?t.merge([e],i).sort():[e]}this._setOptionDisabled(i)}},load:function(e,i){e=this._getIndex(e);var s=this,n=this.tabs.eq(e),o=n.find(".ui-tabs-anchor"),a=this._getPanelForTab(n),r={tab:n,panel:a},h=function(t,e){"abort"===e&&s.panels.stop(!1,!0),s._removeClass(n,"ui-tabs-loading"),a.removeAttr("aria-busy"),t===s.xhr&&delete s.xhr};this._isLocal(o[0])||(this.xhr=t.ajax(this._ajaxSettings(o,i,r)),this.xhr&&"canceled"!==this.xhr.statusText&&(this._addClass(n,"ui-tabs-loading"),a.attr("aria-busy","true"),this.xhr.done(function(t,e,n){setTimeout(function(){a.html(t),s._trigger("load",i,r),h(n,e)},1)}).fail(function(t,e){setTimeout(function(){h(t,e)},1)})))},_ajaxSettings:function(e,i,s){var n=this;return{url:e.attr("href").replace(/#.*$/,""),beforeSend:function(e,o){return n._trigger("beforeLoad",i,t.extend({jqXHR:e,ajaxSettings:o},s))}}},_getPanelForTab:function(e){var i=t(e).attr("aria-controls");return this.element.find(this._sanitizeSelector("#"+i))}}),t.uiBackCompat!==!1&&t.widget("ui.tabs",t.ui.tabs,{_processTabs:function(){this._superApply(arguments),this._addClass(this.tabs,"ui-tab")}}),t.ui.tabs,t.widget("ui.tooltip",{version:"1.12.1",options:{classes:{"ui-tooltip":"ui-corner-all ui-widget-shadow"},content:function(){var e=t(this).attr("title")||"";return t("<a>").text(e).html()},hide:!0,items:"[title]:not([disabled])",position:{my:"left top+15",at:"left bottom",collision:"flipfit flip"},show:!0,track:!1,close:null,open:null},_addDescribedBy:function(e,i){var s=(e.attr("aria-describedby")||"").split(/\s+/);s.push(i),e.data("ui-tooltip-id",i).attr("aria-describedby",t.trim(s.join(" ")))},_removeDescribedBy:function(e){var i=e.data("ui-tooltip-id"),s=(e.attr("aria-describedby")||"").split(/\s+/),n=t.inArray(i,s);-1!==n&&s.splice(n,1),e.removeData("ui-tooltip-id"),s=t.trim(s.join(" ")),s?e.attr("aria-describedby",s):e.removeAttr("aria-describedby")},_create:function(){this._on({mouseover:"open",focusin:"open"}),this.tooltips={},this.parents={},this.liveRegion=t("<div>").attr({role:"log","aria-live":"assertive","aria-relevant":"additions"}).appendTo(this.document[0].body),this._addClass(this.liveRegion,null,"ui-helper-hidden-accessible"),this.disabledTitles=t([])},_setOption:function(e,i){var s=this;this._super(e,i),"content"===e&&t.each(this.tooltips,function(t,e){s._updateContent(e.element)})},_setOptionDisabled:function(t){this[t?"_disable":"_enable"]()},_disable:function(){var e=this;t.each(this.tooltips,function(i,s){var n=t.Event("blur");n.target=n.currentTarget=s.element[0],e.close(n,!0)}),this.disabledTitles=this.disabledTitles.add(this.element.find(this.options.items).addBack().filter(function(){var e=t(this);return e.is("[title]")?e.data("ui-tooltip-title",e.attr("title")).removeAttr("title"):void 0}))},_enable:function(){this.disabledTitles.each(function(){var e=t(this);e.data("ui-tooltip-title")&&e.attr("title",e.data("ui-tooltip-title"))}),this.disabledTitles=t([])},open:function(e){var i=this,s=t(e?e.target:this.element).closest(this.options.items);s.length&&!s.data("ui-tooltip-id")&&(s.attr("title")&&s.data("ui-tooltip-title",s.attr("title")),s.data("ui-tooltip-open",!0),e&&"mouseover"===e.type&&s.parents().each(function(){var e,s=t(this);s.data("ui-tooltip-open")&&(e=t.Event("blur"),e.target=e.currentTarget=this,i.close(e,!0)),s.attr("title")&&(s.uniqueId(),i.parents[this.id]={element:this,title:s.attr("title")},s.attr("title",""))}),this._registerCloseHandlers(e,s),this._updateContent(s,e))},_updateContent:function(t,e){var i,s=this.options.content,n=this,o=e?e.type:null;return"string"==typeof s||s.nodeType||s.jquery?this._open(e,t,s):(i=s.call(t[0],function(i){n._delay(function(){t.data("ui-tooltip-open")&&(e&&(e.type=o),this._open(e,t,i))})}),i&&this._open(e,t,i),void 0)},_open:function(e,i,s){function n(t){l.of=t,a.is(":hidden")||a.position(l)}var o,a,r,h,l=t.extend({},this.options.position);if(s){if(o=this._find(i))return o.tooltip.find(".ui-tooltip-content").html(s),void 0;i.is("[title]")&&(e&&"mouseover"===e.type?i.attr("title",""):i.removeAttr("title")),o=this._tooltip(i),a=o.tooltip,this._addDescribedBy(i,a.attr("id")),a.find(".ui-tooltip-content").html(s),this.liveRegion.children().hide(),h=t("<div>").html(a.find(".ui-tooltip-content").html()),h.removeAttr("name").find("[name]").removeAttr("name"),h.removeAttr("id").find("[id]").removeAttr("id"),h.appendTo(this.liveRegion),this.options.track&&e&&/^mouse/.test(e.type)?(this._on(this.document,{mousemove:n}),n(e)):a.position(t.extend({of:i},this.options.position)),a.hide(),this._show(a,this.options.show),this.options.track&&this.options.show&&this.options.show.delay&&(r=this.delayedShow=setInterval(function(){a.is(":visible")&&(n(l.of),clearInterval(r))},t.fx.interval)),this._trigger("open",e,{tooltip:a})}},_registerCloseHandlers:function(e,i){var s={keyup:function(e){if(e.keyCode===t.ui.keyCode.ESCAPE){var s=t.Event(e);s.currentTarget=i[0],this.close(s,!0)}}};i[0]!==this.element[0]&&(s.remove=function(){this._removeTooltip(this._find(i).tooltip)}),e&&"mouseover"!==e.type||(s.mouseleave="close"),e&&"focusin"!==e.type||(s.focusout="close"),this._on(!0,i,s)},close:function(e){var i,s=this,n=t(e?e.currentTarget:this.element),o=this._find(n);return o?(i=o.tooltip,o.closing||(clearInterval(this.delayedShow),n.data("ui-tooltip-title")&&!n.attr("title")&&n.attr("title",n.data("ui-tooltip-title")),this._removeDescribedBy(n),o.hiding=!0,i.stop(!0),this._hide(i,this.options.hide,function(){s._removeTooltip(t(this))}),n.removeData("ui-tooltip-open"),this._off(n,"mouseleave focusout keyup"),n[0]!==this.element[0]&&this._off(n,"remove"),this._off(this.document,"mousemove"),e&&"mouseleave"===e.type&&t.each(this.parents,function(e,i){t(i.element).attr("title",i.title),delete s.parents[e]}),o.closing=!0,this._trigger("close",e,{tooltip:i}),o.hiding||(o.closing=!1)),void 0):(n.removeData("ui-tooltip-open"),void 0)},_tooltip:function(e){var i=t("<div>").attr("role","tooltip"),s=t("<div>").appendTo(i),n=i.uniqueId().attr("id");return this._addClass(s,"ui-tooltip-content"),this._addClass(i,"ui-tooltip","ui-widget ui-widget-content"),i.appendTo(this._appendTo(e)),this.tooltips[n]={element:e,tooltip:i}},_find:function(t){var e=t.data("ui-tooltip-id");return e?this.tooltips[e]:null},_removeTooltip:function(t){t.remove(),delete this.tooltips[t.attr("id")]},_appendTo:function(t){var e=t.closest(".ui-front, dialog");return e.length||(e=this.document[0].body),e},_destroy:function(){var e=this;t.each(this.tooltips,function(i,s){var n=t.Event("blur"),o=s.element;n.target=n.currentTarget=o[0],e.close(n,!0),t("#"+i).remove(),o.data("ui-tooltip-title")&&(o.attr("title")||o.attr("title",o.data("ui-tooltip-title")),o.removeData("ui-tooltip-title"))}),this.liveRegion.remove()}}),t.uiBackCompat!==!1&&t.widget("ui.tooltip",t.ui.tooltip,{options:{tooltipClass:null},_tooltip:function(){var t=this._superApply(arguments);return this.options.tooltipClass&&t.tooltip.addClass(this.options.tooltipClass),t}}),t.ui.tooltip}); \ No newline at end of file diff --git a/public/static/common/js/jquery.min.js b/public/static/common/js/jquery.min.js deleted file mode 100644 index 4d9b3a2..0000000 --- a/public/static/common/js/jquery.min.js +++ /dev/null @@ -1,2 +0,0 @@ -/*! jQuery v3.3.1 | (c) JS Foundation and other contributors | jquery.org/license */ -!function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(!e.document)throw new Error("jQuery requires a window with a document");return t(e)}:t(e)}("undefined"!=typeof window?window:this,function(e,t){"use strict";var n=[],r=e.document,i=Object.getPrototypeOf,o=n.slice,a=n.concat,s=n.push,u=n.indexOf,l={},c=l.toString,f=l.hasOwnProperty,p=f.toString,d=p.call(Object),h={},g=function e(t){return"function"==typeof t&&"number"!=typeof t.nodeType},y=function e(t){return null!=t&&t===t.window},v={type:!0,src:!0,noModule:!0};function m(e,t,n){var i,o=(t=t||r).createElement("script");if(o.text=e,n)for(i in v)n[i]&&(o[i]=n[i]);t.head.appendChild(o).parentNode.removeChild(o)}function x(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?l[c.call(e)]||"object":typeof e}var b="3.3.1",w=function(e,t){return new w.fn.init(e,t)},T=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g;w.fn=w.prototype={jquery:"3.3.1",constructor:w,length:0,toArray:function(){return o.call(this)},get:function(e){return null==e?o.call(this):e<0?this[e+this.length]:this[e]},pushStack:function(e){var t=w.merge(this.constructor(),e);return t.prevObject=this,t},each:function(e){return w.each(this,e)},map:function(e){return this.pushStack(w.map(this,function(t,n){return e.call(t,n,t)}))},slice:function(){return this.pushStack(o.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(e){var t=this.length,n=+e+(e<0?t:0);return this.pushStack(n>=0&&n<t?[this[n]]:[])},end:function(){return this.prevObject||this.constructor()},push:s,sort:n.sort,splice:n.splice},w.extend=w.fn.extend=function(){var e,t,n,r,i,o,a=arguments[0]||{},s=1,u=arguments.length,l=!1;for("boolean"==typeof a&&(l=a,a=arguments[s]||{},s++),"object"==typeof a||g(a)||(a={}),s===u&&(a=this,s--);s<u;s++)if(null!=(e=arguments[s]))for(t in e)n=a[t],a!==(r=e[t])&&(l&&r&&(w.isPlainObject(r)||(i=Array.isArray(r)))?(i?(i=!1,o=n&&Array.isArray(n)?n:[]):o=n&&w.isPlainObject(n)?n:{},a[t]=w.extend(l,o,r)):void 0!==r&&(a[t]=r));return a},w.extend({expando:"jQuery"+("3.3.1"+Math.random()).replace(/\D/g,""),isReady:!0,error:function(e){throw new Error(e)},noop:function(){},isPlainObject:function(e){var t,n;return!(!e||"[object Object]"!==c.call(e))&&(!(t=i(e))||"function"==typeof(n=f.call(t,"constructor")&&t.constructor)&&p.call(n)===d)},isEmptyObject:function(e){var t;for(t in e)return!1;return!0},globalEval:function(e){m(e)},each:function(e,t){var n,r=0;if(C(e)){for(n=e.length;r<n;r++)if(!1===t.call(e[r],r,e[r]))break}else for(r in e)if(!1===t.call(e[r],r,e[r]))break;return e},trim:function(e){return null==e?"":(e+"").replace(T,"")},makeArray:function(e,t){var n=t||[];return null!=e&&(C(Object(e))?w.merge(n,"string"==typeof e?[e]:e):s.call(n,e)),n},inArray:function(e,t,n){return null==t?-1:u.call(t,e,n)},merge:function(e,t){for(var n=+t.length,r=0,i=e.length;r<n;r++)e[i++]=t[r];return e.length=i,e},grep:function(e,t,n){for(var r,i=[],o=0,a=e.length,s=!n;o<a;o++)(r=!t(e[o],o))!==s&&i.push(e[o]);return i},map:function(e,t,n){var r,i,o=0,s=[];if(C(e))for(r=e.length;o<r;o++)null!=(i=t(e[o],o,n))&&s.push(i);else for(o in e)null!=(i=t(e[o],o,n))&&s.push(i);return a.apply([],s)},guid:1,support:h}),"function"==typeof Symbol&&(w.fn[Symbol.iterator]=n[Symbol.iterator]),w.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(e,t){l["[object "+t+"]"]=t.toLowerCase()});function C(e){var t=!!e&&"length"in e&&e.length,n=x(e);return!g(e)&&!y(e)&&("array"===n||0===t||"number"==typeof t&&t>0&&t-1 in e)}var E=function(e){var t,n,r,i,o,a,s,u,l,c,f,p,d,h,g,y,v,m,x,b="sizzle"+1*new Date,w=e.document,T=0,C=0,E=ae(),k=ae(),S=ae(),D=function(e,t){return e===t&&(f=!0),0},N={}.hasOwnProperty,A=[],j=A.pop,q=A.push,L=A.push,H=A.slice,O=function(e,t){for(var n=0,r=e.length;n<r;n++)if(e[n]===t)return n;return-1},P="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",M="[\\x20\\t\\r\\n\\f]",R="(?:\\\\.|[\\w-]|[^\0-\\xa0])+",I="\\["+M+"*("+R+")(?:"+M+"*([*^$|!~]?=)"+M+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+R+"))|)"+M+"*\\]",W=":("+R+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+I+")*)|.*)\\)|)",$=new RegExp(M+"+","g"),B=new RegExp("^"+M+"+|((?:^|[^\\\\])(?:\\\\.)*)"+M+"+$","g"),F=new RegExp("^"+M+"*,"+M+"*"),_=new RegExp("^"+M+"*([>+~]|"+M+")"+M+"*"),z=new RegExp("="+M+"*([^\\]'\"]*?)"+M+"*\\]","g"),X=new RegExp(W),U=new RegExp("^"+R+"$"),V={ID:new RegExp("^#("+R+")"),CLASS:new RegExp("^\\.("+R+")"),TAG:new RegExp("^("+R+"|[*])"),ATTR:new RegExp("^"+I),PSEUDO:new RegExp("^"+W),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+M+"*(even|odd|(([+-]|)(\\d*)n|)"+M+"*(?:([+-]|)"+M+"*(\\d+)|))"+M+"*\\)|)","i"),bool:new RegExp("^(?:"+P+")$","i"),needsContext:new RegExp("^"+M+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+M+"*((?:-\\d)?\\d*)"+M+"*\\)|)(?=[^-]|$)","i")},G=/^(?:input|select|textarea|button)$/i,Y=/^h\d$/i,Q=/^[^{]+\{\s*\[native \w/,J=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,K=/[+~]/,Z=new RegExp("\\\\([\\da-f]{1,6}"+M+"?|("+M+")|.)","ig"),ee=function(e,t,n){var r="0x"+t-65536;return r!==r||n?t:r<0?String.fromCharCode(r+65536):String.fromCharCode(r>>10|55296,1023&r|56320)},te=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,ne=function(e,t){return t?"\0"===e?"\ufffd":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e},re=function(){p()},ie=me(function(e){return!0===e.disabled&&("form"in e||"label"in e)},{dir:"parentNode",next:"legend"});try{L.apply(A=H.call(w.childNodes),w.childNodes),A[w.childNodes.length].nodeType}catch(e){L={apply:A.length?function(e,t){q.apply(e,H.call(t))}:function(e,t){var n=e.length,r=0;while(e[n++]=t[r++]);e.length=n-1}}}function oe(e,t,r,i){var o,s,l,c,f,h,v,m=t&&t.ownerDocument,T=t?t.nodeType:9;if(r=r||[],"string"!=typeof e||!e||1!==T&&9!==T&&11!==T)return r;if(!i&&((t?t.ownerDocument||t:w)!==d&&p(t),t=t||d,g)){if(11!==T&&(f=J.exec(e)))if(o=f[1]){if(9===T){if(!(l=t.getElementById(o)))return r;if(l.id===o)return r.push(l),r}else if(m&&(l=m.getElementById(o))&&x(t,l)&&l.id===o)return r.push(l),r}else{if(f[2])return L.apply(r,t.getElementsByTagName(e)),r;if((o=f[3])&&n.getElementsByClassName&&t.getElementsByClassName)return L.apply(r,t.getElementsByClassName(o)),r}if(n.qsa&&!S[e+" "]&&(!y||!y.test(e))){if(1!==T)m=t,v=e;else if("object"!==t.nodeName.toLowerCase()){(c=t.getAttribute("id"))?c=c.replace(te,ne):t.setAttribute("id",c=b),s=(h=a(e)).length;while(s--)h[s]="#"+c+" "+ve(h[s]);v=h.join(","),m=K.test(e)&&ge(t.parentNode)||t}if(v)try{return L.apply(r,m.querySelectorAll(v)),r}catch(e){}finally{c===b&&t.removeAttribute("id")}}}return u(e.replace(B,"$1"),t,r,i)}function ae(){var e=[];function t(n,i){return e.push(n+" ")>r.cacheLength&&delete t[e.shift()],t[n+" "]=i}return t}function se(e){return e[b]=!0,e}function ue(e){var t=d.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function le(e,t){var n=e.split("|"),i=n.length;while(i--)r.attrHandle[n[i]]=t}function ce(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&e.sourceIndex-t.sourceIndex;if(r)return r;if(n)while(n=n.nextSibling)if(n===t)return-1;return e?1:-1}function fe(e){return function(t){return"input"===t.nodeName.toLowerCase()&&t.type===e}}function pe(e){return function(t){var n=t.nodeName.toLowerCase();return("input"===n||"button"===n)&&t.type===e}}function de(e){return function(t){return"form"in t?t.parentNode&&!1===t.disabled?"label"in t?"label"in t.parentNode?t.parentNode.disabled===e:t.disabled===e:t.isDisabled===e||t.isDisabled!==!e&&ie(t)===e:t.disabled===e:"label"in t&&t.disabled===e}}function he(e){return se(function(t){return t=+t,se(function(n,r){var i,o=e([],n.length,t),a=o.length;while(a--)n[i=o[a]]&&(n[i]=!(r[i]=n[i]))})})}function ge(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}n=oe.support={},o=oe.isXML=function(e){var t=e&&(e.ownerDocument||e).documentElement;return!!t&&"HTML"!==t.nodeName},p=oe.setDocument=function(e){var t,i,a=e?e.ownerDocument||e:w;return a!==d&&9===a.nodeType&&a.documentElement?(d=a,h=d.documentElement,g=!o(d),w!==d&&(i=d.defaultView)&&i.top!==i&&(i.addEventListener?i.addEventListener("unload",re,!1):i.attachEvent&&i.attachEvent("onunload",re)),n.attributes=ue(function(e){return e.className="i",!e.getAttribute("className")}),n.getElementsByTagName=ue(function(e){return e.appendChild(d.createComment("")),!e.getElementsByTagName("*").length}),n.getElementsByClassName=Q.test(d.getElementsByClassName),n.getById=ue(function(e){return h.appendChild(e).id=b,!d.getElementsByName||!d.getElementsByName(b).length}),n.getById?(r.filter.ID=function(e){var t=e.replace(Z,ee);return function(e){return e.getAttribute("id")===t}},r.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&g){var n=t.getElementById(e);return n?[n]:[]}}):(r.filter.ID=function(e){var t=e.replace(Z,ee);return function(e){var n="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return n&&n.value===t}},r.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&g){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),r.find.TAG=n.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):n.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"===e){while(n=o[i++])1===n.nodeType&&r.push(n);return r}return o},r.find.CLASS=n.getElementsByClassName&&function(e,t){if("undefined"!=typeof t.getElementsByClassName&&g)return t.getElementsByClassName(e)},v=[],y=[],(n.qsa=Q.test(d.querySelectorAll))&&(ue(function(e){h.appendChild(e).innerHTML="<a id='"+b+"'></a><select id='"+b+"-\r\\' msallowcapture=''><option selected=''></option></select>",e.querySelectorAll("[msallowcapture^='']").length&&y.push("[*^$]="+M+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||y.push("\\["+M+"*(?:value|"+P+")"),e.querySelectorAll("[id~="+b+"-]").length||y.push("~="),e.querySelectorAll(":checked").length||y.push(":checked"),e.querySelectorAll("a#"+b+"+*").length||y.push(".#.+[+~]")}),ue(function(e){e.innerHTML="<a href='' disabled='disabled'></a><select disabled='disabled'><option/></select>";var t=d.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&y.push("name"+M+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&y.push(":enabled",":disabled"),h.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&y.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),y.push(",.*:")})),(n.matchesSelector=Q.test(m=h.matches||h.webkitMatchesSelector||h.mozMatchesSelector||h.oMatchesSelector||h.msMatchesSelector))&&ue(function(e){n.disconnectedMatch=m.call(e,"*"),m.call(e,"[s!='']:x"),v.push("!=",W)}),y=y.length&&new RegExp(y.join("|")),v=v.length&&new RegExp(v.join("|")),t=Q.test(h.compareDocumentPosition),x=t||Q.test(h.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,r=t&&t.parentNode;return e===r||!(!r||1!==r.nodeType||!(n.contains?n.contains(r):e.compareDocumentPosition&&16&e.compareDocumentPosition(r)))}:function(e,t){if(t)while(t=t.parentNode)if(t===e)return!0;return!1},D=t?function(e,t){if(e===t)return f=!0,0;var r=!e.compareDocumentPosition-!t.compareDocumentPosition;return r||(1&(r=(e.ownerDocument||e)===(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!n.sortDetached&&t.compareDocumentPosition(e)===r?e===d||e.ownerDocument===w&&x(w,e)?-1:t===d||t.ownerDocument===w&&x(w,t)?1:c?O(c,e)-O(c,t):0:4&r?-1:1)}:function(e,t){if(e===t)return f=!0,0;var n,r=0,i=e.parentNode,o=t.parentNode,a=[e],s=[t];if(!i||!o)return e===d?-1:t===d?1:i?-1:o?1:c?O(c,e)-O(c,t):0;if(i===o)return ce(e,t);n=e;while(n=n.parentNode)a.unshift(n);n=t;while(n=n.parentNode)s.unshift(n);while(a[r]===s[r])r++;return r?ce(a[r],s[r]):a[r]===w?-1:s[r]===w?1:0},d):d},oe.matches=function(e,t){return oe(e,null,null,t)},oe.matchesSelector=function(e,t){if((e.ownerDocument||e)!==d&&p(e),t=t.replace(z,"='$1']"),n.matchesSelector&&g&&!S[t+" "]&&(!v||!v.test(t))&&(!y||!y.test(t)))try{var r=m.call(e,t);if(r||n.disconnectedMatch||e.document&&11!==e.document.nodeType)return r}catch(e){}return oe(t,d,null,[e]).length>0},oe.contains=function(e,t){return(e.ownerDocument||e)!==d&&p(e),x(e,t)},oe.attr=function(e,t){(e.ownerDocument||e)!==d&&p(e);var i=r.attrHandle[t.toLowerCase()],o=i&&N.call(r.attrHandle,t.toLowerCase())?i(e,t,!g):void 0;return void 0!==o?o:n.attributes||!g?e.getAttribute(t):(o=e.getAttributeNode(t))&&o.specified?o.value:null},oe.escape=function(e){return(e+"").replace(te,ne)},oe.error=function(e){throw new Error("Syntax error, unrecognized expression: "+e)},oe.uniqueSort=function(e){var t,r=[],i=0,o=0;if(f=!n.detectDuplicates,c=!n.sortStable&&e.slice(0),e.sort(D),f){while(t=e[o++])t===e[o]&&(i=r.push(o));while(i--)e.splice(r[i],1)}return c=null,e},i=oe.getText=function(e){var t,n="",r=0,o=e.nodeType;if(o){if(1===o||9===o||11===o){if("string"==typeof e.textContent)return e.textContent;for(e=e.firstChild;e;e=e.nextSibling)n+=i(e)}else if(3===o||4===o)return e.nodeValue}else while(t=e[r++])n+=i(t);return n},(r=oe.selectors={cacheLength:50,createPseudo:se,match:V,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(Z,ee),e[3]=(e[3]||e[4]||e[5]||"").replace(Z,ee),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||oe.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&oe.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return V.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&X.test(n)&&(t=a(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(Z,ee).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=E[e+" "];return t||(t=new RegExp("(^|"+M+")"+e+"("+M+"|$)"))&&E(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(e,t,n){return function(r){var i=oe.attr(r,e);return null==i?"!="===t:!t||(i+="","="===t?i===n:"!="===t?i!==n:"^="===t?n&&0===i.indexOf(n):"*="===t?n&&i.indexOf(n)>-1:"$="===t?n&&i.slice(-n.length)===n:"~="===t?(" "+i.replace($," ")+" ").indexOf(n)>-1:"|="===t&&(i===n||i.slice(0,n.length+1)===n+"-"))}},CHILD:function(e,t,n,r,i){var o="nth"!==e.slice(0,3),a="last"!==e.slice(-4),s="of-type"===t;return 1===r&&0===i?function(e){return!!e.parentNode}:function(t,n,u){var l,c,f,p,d,h,g=o!==a?"nextSibling":"previousSibling",y=t.parentNode,v=s&&t.nodeName.toLowerCase(),m=!u&&!s,x=!1;if(y){if(o){while(g){p=t;while(p=p[g])if(s?p.nodeName.toLowerCase()===v:1===p.nodeType)return!1;h=g="only"===e&&!h&&"nextSibling"}return!0}if(h=[a?y.firstChild:y.lastChild],a&&m){x=(d=(l=(c=(f=(p=y)[b]||(p[b]={}))[p.uniqueID]||(f[p.uniqueID]={}))[e]||[])[0]===T&&l[1])&&l[2],p=d&&y.childNodes[d];while(p=++d&&p&&p[g]||(x=d=0)||h.pop())if(1===p.nodeType&&++x&&p===t){c[e]=[T,d,x];break}}else if(m&&(x=d=(l=(c=(f=(p=t)[b]||(p[b]={}))[p.uniqueID]||(f[p.uniqueID]={}))[e]||[])[0]===T&&l[1]),!1===x)while(p=++d&&p&&p[g]||(x=d=0)||h.pop())if((s?p.nodeName.toLowerCase()===v:1===p.nodeType)&&++x&&(m&&((c=(f=p[b]||(p[b]={}))[p.uniqueID]||(f[p.uniqueID]={}))[e]=[T,x]),p===t))break;return(x-=i)===r||x%r==0&&x/r>=0}}},PSEUDO:function(e,t){var n,i=r.pseudos[e]||r.setFilters[e.toLowerCase()]||oe.error("unsupported pseudo: "+e);return i[b]?i(t):i.length>1?(n=[e,e,"",t],r.setFilters.hasOwnProperty(e.toLowerCase())?se(function(e,n){var r,o=i(e,t),a=o.length;while(a--)e[r=O(e,o[a])]=!(n[r]=o[a])}):function(e){return i(e,0,n)}):i}},pseudos:{not:se(function(e){var t=[],n=[],r=s(e.replace(B,"$1"));return r[b]?se(function(e,t,n,i){var o,a=r(e,null,i,[]),s=e.length;while(s--)(o=a[s])&&(e[s]=!(t[s]=o))}):function(e,i,o){return t[0]=e,r(t,null,o,n),t[0]=null,!n.pop()}}),has:se(function(e){return function(t){return oe(e,t).length>0}}),contains:se(function(e){return e=e.replace(Z,ee),function(t){return(t.textContent||t.innerText||i(t)).indexOf(e)>-1}}),lang:se(function(e){return U.test(e||"")||oe.error("unsupported lang: "+e),e=e.replace(Z,ee).toLowerCase(),function(t){var n;do{if(n=g?t.lang:t.getAttribute("xml:lang")||t.getAttribute("lang"))return(n=n.toLowerCase())===e||0===n.indexOf(e+"-")}while((t=t.parentNode)&&1===t.nodeType);return!1}}),target:function(t){var n=e.location&&e.location.hash;return n&&n.slice(1)===t.id},root:function(e){return e===h},focus:function(e){return e===d.activeElement&&(!d.hasFocus||d.hasFocus())&&!!(e.type||e.href||~e.tabIndex)},enabled:de(!1),disabled:de(!0),checked:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&!!e.checked||"option"===t&&!!e.selected},selected:function(e){return e.parentNode&&e.parentNode.selectedIndex,!0===e.selected},empty:function(e){for(e=e.firstChild;e;e=e.nextSibling)if(e.nodeType<6)return!1;return!0},parent:function(e){return!r.pseudos.empty(e)},header:function(e){return Y.test(e.nodeName)},input:function(e){return G.test(e.nodeName)},button:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&"button"===e.type||"button"===t},text:function(e){var t;return"input"===e.nodeName.toLowerCase()&&"text"===e.type&&(null==(t=e.getAttribute("type"))||"text"===t.toLowerCase())},first:he(function(){return[0]}),last:he(function(e,t){return[t-1]}),eq:he(function(e,t,n){return[n<0?n+t:n]}),even:he(function(e,t){for(var n=0;n<t;n+=2)e.push(n);return e}),odd:he(function(e,t){for(var n=1;n<t;n+=2)e.push(n);return e}),lt:he(function(e,t,n){for(var r=n<0?n+t:n;--r>=0;)e.push(r);return e}),gt:he(function(e,t,n){for(var r=n<0?n+t:n;++r<t;)e.push(r);return e})}}).pseudos.nth=r.pseudos.eq;for(t in{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})r.pseudos[t]=fe(t);for(t in{submit:!0,reset:!0})r.pseudos[t]=pe(t);function ye(){}ye.prototype=r.filters=r.pseudos,r.setFilters=new ye,a=oe.tokenize=function(e,t){var n,i,o,a,s,u,l,c=k[e+" "];if(c)return t?0:c.slice(0);s=e,u=[],l=r.preFilter;while(s){n&&!(i=F.exec(s))||(i&&(s=s.slice(i[0].length)||s),u.push(o=[])),n=!1,(i=_.exec(s))&&(n=i.shift(),o.push({value:n,type:i[0].replace(B," ")}),s=s.slice(n.length));for(a in r.filter)!(i=V[a].exec(s))||l[a]&&!(i=l[a](i))||(n=i.shift(),o.push({value:n,type:a,matches:i}),s=s.slice(n.length));if(!n)break}return t?s.length:s?oe.error(e):k(e,u).slice(0)};function ve(e){for(var t=0,n=e.length,r="";t<n;t++)r+=e[t].value;return r}function me(e,t,n){var r=t.dir,i=t.next,o=i||r,a=n&&"parentNode"===o,s=C++;return t.first?function(t,n,i){while(t=t[r])if(1===t.nodeType||a)return e(t,n,i);return!1}:function(t,n,u){var l,c,f,p=[T,s];if(u){while(t=t[r])if((1===t.nodeType||a)&&e(t,n,u))return!0}else while(t=t[r])if(1===t.nodeType||a)if(f=t[b]||(t[b]={}),c=f[t.uniqueID]||(f[t.uniqueID]={}),i&&i===t.nodeName.toLowerCase())t=t[r]||t;else{if((l=c[o])&&l[0]===T&&l[1]===s)return p[2]=l[2];if(c[o]=p,p[2]=e(t,n,u))return!0}return!1}}function xe(e){return e.length>1?function(t,n,r){var i=e.length;while(i--)if(!e[i](t,n,r))return!1;return!0}:e[0]}function be(e,t,n){for(var r=0,i=t.length;r<i;r++)oe(e,t[r],n);return n}function we(e,t,n,r,i){for(var o,a=[],s=0,u=e.length,l=null!=t;s<u;s++)(o=e[s])&&(n&&!n(o,r,i)||(a.push(o),l&&t.push(s)));return a}function Te(e,t,n,r,i,o){return r&&!r[b]&&(r=Te(r)),i&&!i[b]&&(i=Te(i,o)),se(function(o,a,s,u){var l,c,f,p=[],d=[],h=a.length,g=o||be(t||"*",s.nodeType?[s]:s,[]),y=!e||!o&&t?g:we(g,p,e,s,u),v=n?i||(o?e:h||r)?[]:a:y;if(n&&n(y,v,s,u),r){l=we(v,d),r(l,[],s,u),c=l.length;while(c--)(f=l[c])&&(v[d[c]]=!(y[d[c]]=f))}if(o){if(i||e){if(i){l=[],c=v.length;while(c--)(f=v[c])&&l.push(y[c]=f);i(null,v=[],l,u)}c=v.length;while(c--)(f=v[c])&&(l=i?O(o,f):p[c])>-1&&(o[l]=!(a[l]=f))}}else v=we(v===a?v.splice(h,v.length):v),i?i(null,a,v,u):L.apply(a,v)})}function Ce(e){for(var t,n,i,o=e.length,a=r.relative[e[0].type],s=a||r.relative[" "],u=a?1:0,c=me(function(e){return e===t},s,!0),f=me(function(e){return O(t,e)>-1},s,!0),p=[function(e,n,r){var i=!a&&(r||n!==l)||((t=n).nodeType?c(e,n,r):f(e,n,r));return t=null,i}];u<o;u++)if(n=r.relative[e[u].type])p=[me(xe(p),n)];else{if((n=r.filter[e[u].type].apply(null,e[u].matches))[b]){for(i=++u;i<o;i++)if(r.relative[e[i].type])break;return Te(u>1&&xe(p),u>1&&ve(e.slice(0,u-1).concat({value:" "===e[u-2].type?"*":""})).replace(B,"$1"),n,u<i&&Ce(e.slice(u,i)),i<o&&Ce(e=e.slice(i)),i<o&&ve(e))}p.push(n)}return xe(p)}function Ee(e,t){var n=t.length>0,i=e.length>0,o=function(o,a,s,u,c){var f,h,y,v=0,m="0",x=o&&[],b=[],w=l,C=o||i&&r.find.TAG("*",c),E=T+=null==w?1:Math.random()||.1,k=C.length;for(c&&(l=a===d||a||c);m!==k&&null!=(f=C[m]);m++){if(i&&f){h=0,a||f.ownerDocument===d||(p(f),s=!g);while(y=e[h++])if(y(f,a||d,s)){u.push(f);break}c&&(T=E)}n&&((f=!y&&f)&&v--,o&&x.push(f))}if(v+=m,n&&m!==v){h=0;while(y=t[h++])y(x,b,a,s);if(o){if(v>0)while(m--)x[m]||b[m]||(b[m]=j.call(u));b=we(b)}L.apply(u,b),c&&!o&&b.length>0&&v+t.length>1&&oe.uniqueSort(u)}return c&&(T=E,l=w),x};return n?se(o):o}return s=oe.compile=function(e,t){var n,r=[],i=[],o=S[e+" "];if(!o){t||(t=a(e)),n=t.length;while(n--)(o=Ce(t[n]))[b]?r.push(o):i.push(o);(o=S(e,Ee(i,r))).selector=e}return o},u=oe.select=function(e,t,n,i){var o,u,l,c,f,p="function"==typeof e&&e,d=!i&&a(e=p.selector||e);if(n=n||[],1===d.length){if((u=d[0]=d[0].slice(0)).length>2&&"ID"===(l=u[0]).type&&9===t.nodeType&&g&&r.relative[u[1].type]){if(!(t=(r.find.ID(l.matches[0].replace(Z,ee),t)||[])[0]))return n;p&&(t=t.parentNode),e=e.slice(u.shift().value.length)}o=V.needsContext.test(e)?0:u.length;while(o--){if(l=u[o],r.relative[c=l.type])break;if((f=r.find[c])&&(i=f(l.matches[0].replace(Z,ee),K.test(u[0].type)&&ge(t.parentNode)||t))){if(u.splice(o,1),!(e=i.length&&ve(u)))return L.apply(n,i),n;break}}}return(p||s(e,d))(i,t,!g,n,!t||K.test(e)&&ge(t.parentNode)||t),n},n.sortStable=b.split("").sort(D).join("")===b,n.detectDuplicates=!!f,p(),n.sortDetached=ue(function(e){return 1&e.compareDocumentPosition(d.createElement("fieldset"))}),ue(function(e){return e.innerHTML="<a href='#'></a>","#"===e.firstChild.getAttribute("href")})||le("type|href|height|width",function(e,t,n){if(!n)return e.getAttribute(t,"type"===t.toLowerCase()?1:2)}),n.attributes&&ue(function(e){return e.innerHTML="<input/>",e.firstChild.setAttribute("value",""),""===e.firstChild.getAttribute("value")})||le("value",function(e,t,n){if(!n&&"input"===e.nodeName.toLowerCase())return e.defaultValue}),ue(function(e){return null==e.getAttribute("disabled")})||le(P,function(e,t,n){var r;if(!n)return!0===e[t]?t.toLowerCase():(r=e.getAttributeNode(t))&&r.specified?r.value:null}),oe}(e);w.find=E,w.expr=E.selectors,w.expr[":"]=w.expr.pseudos,w.uniqueSort=w.unique=E.uniqueSort,w.text=E.getText,w.isXMLDoc=E.isXML,w.contains=E.contains,w.escapeSelector=E.escape;var k=function(e,t,n){var r=[],i=void 0!==n;while((e=e[t])&&9!==e.nodeType)if(1===e.nodeType){if(i&&w(e).is(n))break;r.push(e)}return r},S=function(e,t){for(var n=[];e;e=e.nextSibling)1===e.nodeType&&e!==t&&n.push(e);return n},D=w.expr.match.needsContext;function N(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()}var A=/^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function j(e,t,n){return g(t)?w.grep(e,function(e,r){return!!t.call(e,r,e)!==n}):t.nodeType?w.grep(e,function(e){return e===t!==n}):"string"!=typeof t?w.grep(e,function(e){return u.call(t,e)>-1!==n}):w.filter(t,e,n)}w.filter=function(e,t,n){var r=t[0];return n&&(e=":not("+e+")"),1===t.length&&1===r.nodeType?w.find.matchesSelector(r,e)?[r]:[]:w.find.matches(e,w.grep(t,function(e){return 1===e.nodeType}))},w.fn.extend({find:function(e){var t,n,r=this.length,i=this;if("string"!=typeof e)return this.pushStack(w(e).filter(function(){for(t=0;t<r;t++)if(w.contains(i[t],this))return!0}));for(n=this.pushStack([]),t=0;t<r;t++)w.find(e,i[t],n);return r>1?w.uniqueSort(n):n},filter:function(e){return this.pushStack(j(this,e||[],!1))},not:function(e){return this.pushStack(j(this,e||[],!0))},is:function(e){return!!j(this,"string"==typeof e&&D.test(e)?w(e):e||[],!1).length}});var q,L=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/;(w.fn.init=function(e,t,n){var i,o;if(!e)return this;if(n=n||q,"string"==typeof e){if(!(i="<"===e[0]&&">"===e[e.length-1]&&e.length>=3?[null,e,null]:L.exec(e))||!i[1]&&t)return!t||t.jquery?(t||n).find(e):this.constructor(t).find(e);if(i[1]){if(t=t instanceof w?t[0]:t,w.merge(this,w.parseHTML(i[1],t&&t.nodeType?t.ownerDocument||t:r,!0)),A.test(i[1])&&w.isPlainObject(t))for(i in t)g(this[i])?this[i](t[i]):this.attr(i,t[i]);return this}return(o=r.getElementById(i[2]))&&(this[0]=o,this.length=1),this}return e.nodeType?(this[0]=e,this.length=1,this):g(e)?void 0!==n.ready?n.ready(e):e(w):w.makeArray(e,this)}).prototype=w.fn,q=w(r);var H=/^(?:parents|prev(?:Until|All))/,O={children:!0,contents:!0,next:!0,prev:!0};w.fn.extend({has:function(e){var t=w(e,this),n=t.length;return this.filter(function(){for(var e=0;e<n;e++)if(w.contains(this,t[e]))return!0})},closest:function(e,t){var n,r=0,i=this.length,o=[],a="string"!=typeof e&&w(e);if(!D.test(e))for(;r<i;r++)for(n=this[r];n&&n!==t;n=n.parentNode)if(n.nodeType<11&&(a?a.index(n)>-1:1===n.nodeType&&w.find.matchesSelector(n,e))){o.push(n);break}return this.pushStack(o.length>1?w.uniqueSort(o):o)},index:function(e){return e?"string"==typeof e?u.call(w(e),this[0]):u.call(this,e.jquery?e[0]:e):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(e,t){return this.pushStack(w.uniqueSort(w.merge(this.get(),w(e,t))))},addBack:function(e){return this.add(null==e?this.prevObject:this.prevObject.filter(e))}});function P(e,t){while((e=e[t])&&1!==e.nodeType);return e}w.each({parent:function(e){var t=e.parentNode;return t&&11!==t.nodeType?t:null},parents:function(e){return k(e,"parentNode")},parentsUntil:function(e,t,n){return k(e,"parentNode",n)},next:function(e){return P(e,"nextSibling")},prev:function(e){return P(e,"previousSibling")},nextAll:function(e){return k(e,"nextSibling")},prevAll:function(e){return k(e,"previousSibling")},nextUntil:function(e,t,n){return k(e,"nextSibling",n)},prevUntil:function(e,t,n){return k(e,"previousSibling",n)},siblings:function(e){return S((e.parentNode||{}).firstChild,e)},children:function(e){return S(e.firstChild)},contents:function(e){return N(e,"iframe")?e.contentDocument:(N(e,"template")&&(e=e.content||e),w.merge([],e.childNodes))}},function(e,t){w.fn[e]=function(n,r){var i=w.map(this,t,n);return"Until"!==e.slice(-5)&&(r=n),r&&"string"==typeof r&&(i=w.filter(r,i)),this.length>1&&(O[e]||w.uniqueSort(i),H.test(e)&&i.reverse()),this.pushStack(i)}});var M=/[^\x20\t\r\n\f]+/g;function R(e){var t={};return w.each(e.match(M)||[],function(e,n){t[n]=!0}),t}w.Callbacks=function(e){e="string"==typeof e?R(e):w.extend({},e);var t,n,r,i,o=[],a=[],s=-1,u=function(){for(i=i||e.once,r=t=!0;a.length;s=-1){n=a.shift();while(++s<o.length)!1===o[s].apply(n[0],n[1])&&e.stopOnFalse&&(s=o.length,n=!1)}e.memory||(n=!1),t=!1,i&&(o=n?[]:"")},l={add:function(){return o&&(n&&!t&&(s=o.length-1,a.push(n)),function t(n){w.each(n,function(n,r){g(r)?e.unique&&l.has(r)||o.push(r):r&&r.length&&"string"!==x(r)&&t(r)})}(arguments),n&&!t&&u()),this},remove:function(){return w.each(arguments,function(e,t){var n;while((n=w.inArray(t,o,n))>-1)o.splice(n,1),n<=s&&s--}),this},has:function(e){return e?w.inArray(e,o)>-1:o.length>0},empty:function(){return o&&(o=[]),this},disable:function(){return i=a=[],o=n="",this},disabled:function(){return!o},lock:function(){return i=a=[],n||t||(o=n=""),this},locked:function(){return!!i},fireWith:function(e,n){return i||(n=[e,(n=n||[]).slice?n.slice():n],a.push(n),t||u()),this},fire:function(){return l.fireWith(this,arguments),this},fired:function(){return!!r}};return l};function I(e){return e}function W(e){throw e}function $(e,t,n,r){var i;try{e&&g(i=e.promise)?i.call(e).done(t).fail(n):e&&g(i=e.then)?i.call(e,t,n):t.apply(void 0,[e].slice(r))}catch(e){n.apply(void 0,[e])}}w.extend({Deferred:function(t){var n=[["notify","progress",w.Callbacks("memory"),w.Callbacks("memory"),2],["resolve","done",w.Callbacks("once memory"),w.Callbacks("once memory"),0,"resolved"],["reject","fail",w.Callbacks("once memory"),w.Callbacks("once memory"),1,"rejected"]],r="pending",i={state:function(){return r},always:function(){return o.done(arguments).fail(arguments),this},"catch":function(e){return i.then(null,e)},pipe:function(){var e=arguments;return w.Deferred(function(t){w.each(n,function(n,r){var i=g(e[r[4]])&&e[r[4]];o[r[1]](function(){var e=i&&i.apply(this,arguments);e&&g(e.promise)?e.promise().progress(t.notify).done(t.resolve).fail(t.reject):t[r[0]+"With"](this,i?[e]:arguments)})}),e=null}).promise()},then:function(t,r,i){var o=0;function a(t,n,r,i){return function(){var s=this,u=arguments,l=function(){var e,l;if(!(t<o)){if((e=r.apply(s,u))===n.promise())throw new TypeError("Thenable self-resolution");l=e&&("object"==typeof e||"function"==typeof e)&&e.then,g(l)?i?l.call(e,a(o,n,I,i),a(o,n,W,i)):(o++,l.call(e,a(o,n,I,i),a(o,n,W,i),a(o,n,I,n.notifyWith))):(r!==I&&(s=void 0,u=[e]),(i||n.resolveWith)(s,u))}},c=i?l:function(){try{l()}catch(e){w.Deferred.exceptionHook&&w.Deferred.exceptionHook(e,c.stackTrace),t+1>=o&&(r!==W&&(s=void 0,u=[e]),n.rejectWith(s,u))}};t?c():(w.Deferred.getStackHook&&(c.stackTrace=w.Deferred.getStackHook()),e.setTimeout(c))}}return w.Deferred(function(e){n[0][3].add(a(0,e,g(i)?i:I,e.notifyWith)),n[1][3].add(a(0,e,g(t)?t:I)),n[2][3].add(a(0,e,g(r)?r:W))}).promise()},promise:function(e){return null!=e?w.extend(e,i):i}},o={};return w.each(n,function(e,t){var a=t[2],s=t[5];i[t[1]]=a.add,s&&a.add(function(){r=s},n[3-e][2].disable,n[3-e][3].disable,n[0][2].lock,n[0][3].lock),a.add(t[3].fire),o[t[0]]=function(){return o[t[0]+"With"](this===o?void 0:this,arguments),this},o[t[0]+"With"]=a.fireWith}),i.promise(o),t&&t.call(o,o),o},when:function(e){var t=arguments.length,n=t,r=Array(n),i=o.call(arguments),a=w.Deferred(),s=function(e){return function(n){r[e]=this,i[e]=arguments.length>1?o.call(arguments):n,--t||a.resolveWith(r,i)}};if(t<=1&&($(e,a.done(s(n)).resolve,a.reject,!t),"pending"===a.state()||g(i[n]&&i[n].then)))return a.then();while(n--)$(i[n],s(n),a.reject);return a.promise()}});var B=/^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/;w.Deferred.exceptionHook=function(t,n){e.console&&e.console.warn&&t&&B.test(t.name)&&e.console.warn("jQuery.Deferred exception: "+t.message,t.stack,n)},w.readyException=function(t){e.setTimeout(function(){throw t})};var F=w.Deferred();w.fn.ready=function(e){return F.then(e)["catch"](function(e){w.readyException(e)}),this},w.extend({isReady:!1,readyWait:1,ready:function(e){(!0===e?--w.readyWait:w.isReady)||(w.isReady=!0,!0!==e&&--w.readyWait>0||F.resolveWith(r,[w]))}}),w.ready.then=F.then;function _(){r.removeEventListener("DOMContentLoaded",_),e.removeEventListener("load",_),w.ready()}"complete"===r.readyState||"loading"!==r.readyState&&!r.documentElement.doScroll?e.setTimeout(w.ready):(r.addEventListener("DOMContentLoaded",_),e.addEventListener("load",_));var z=function(e,t,n,r,i,o,a){var s=0,u=e.length,l=null==n;if("object"===x(n)){i=!0;for(s in n)z(e,t,s,n[s],!0,o,a)}else if(void 0!==r&&(i=!0,g(r)||(a=!0),l&&(a?(t.call(e,r),t=null):(l=t,t=function(e,t,n){return l.call(w(e),n)})),t))for(;s<u;s++)t(e[s],n,a?r:r.call(e[s],s,t(e[s],n)));return i?e:l?t.call(e):u?t(e[0],n):o},X=/^-ms-/,U=/-([a-z])/g;function V(e,t){return t.toUpperCase()}function G(e){return e.replace(X,"ms-").replace(U,V)}var Y=function(e){return 1===e.nodeType||9===e.nodeType||!+e.nodeType};function Q(){this.expando=w.expando+Q.uid++}Q.uid=1,Q.prototype={cache:function(e){var t=e[this.expando];return t||(t={},Y(e)&&(e.nodeType?e[this.expando]=t:Object.defineProperty(e,this.expando,{value:t,configurable:!0}))),t},set:function(e,t,n){var r,i=this.cache(e);if("string"==typeof t)i[G(t)]=n;else for(r in t)i[G(r)]=t[r];return i},get:function(e,t){return void 0===t?this.cache(e):e[this.expando]&&e[this.expando][G(t)]},access:function(e,t,n){return void 0===t||t&&"string"==typeof t&&void 0===n?this.get(e,t):(this.set(e,t,n),void 0!==n?n:t)},remove:function(e,t){var n,r=e[this.expando];if(void 0!==r){if(void 0!==t){n=(t=Array.isArray(t)?t.map(G):(t=G(t))in r?[t]:t.match(M)||[]).length;while(n--)delete r[t[n]]}(void 0===t||w.isEmptyObject(r))&&(e.nodeType?e[this.expando]=void 0:delete e[this.expando])}},hasData:function(e){var t=e[this.expando];return void 0!==t&&!w.isEmptyObject(t)}};var J=new Q,K=new Q,Z=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,ee=/[A-Z]/g;function te(e){return"true"===e||"false"!==e&&("null"===e?null:e===+e+""?+e:Z.test(e)?JSON.parse(e):e)}function ne(e,t,n){var r;if(void 0===n&&1===e.nodeType)if(r="data-"+t.replace(ee,"-$&").toLowerCase(),"string"==typeof(n=e.getAttribute(r))){try{n=te(n)}catch(e){}K.set(e,t,n)}else n=void 0;return n}w.extend({hasData:function(e){return K.hasData(e)||J.hasData(e)},data:function(e,t,n){return K.access(e,t,n)},removeData:function(e,t){K.remove(e,t)},_data:function(e,t,n){return J.access(e,t,n)},_removeData:function(e,t){J.remove(e,t)}}),w.fn.extend({data:function(e,t){var n,r,i,o=this[0],a=o&&o.attributes;if(void 0===e){if(this.length&&(i=K.get(o),1===o.nodeType&&!J.get(o,"hasDataAttrs"))){n=a.length;while(n--)a[n]&&0===(r=a[n].name).indexOf("data-")&&(r=G(r.slice(5)),ne(o,r,i[r]));J.set(o,"hasDataAttrs",!0)}return i}return"object"==typeof e?this.each(function(){K.set(this,e)}):z(this,function(t){var n;if(o&&void 0===t){if(void 0!==(n=K.get(o,e)))return n;if(void 0!==(n=ne(o,e)))return n}else this.each(function(){K.set(this,e,t)})},null,t,arguments.length>1,null,!0)},removeData:function(e){return this.each(function(){K.remove(this,e)})}}),w.extend({queue:function(e,t,n){var r;if(e)return t=(t||"fx")+"queue",r=J.get(e,t),n&&(!r||Array.isArray(n)?r=J.access(e,t,w.makeArray(n)):r.push(n)),r||[]},dequeue:function(e,t){t=t||"fx";var n=w.queue(e,t),r=n.length,i=n.shift(),o=w._queueHooks(e,t),a=function(){w.dequeue(e,t)};"inprogress"===i&&(i=n.shift(),r--),i&&("fx"===t&&n.unshift("inprogress"),delete o.stop,i.call(e,a,o)),!r&&o&&o.empty.fire()},_queueHooks:function(e,t){var n=t+"queueHooks";return J.get(e,n)||J.access(e,n,{empty:w.Callbacks("once memory").add(function(){J.remove(e,[t+"queue",n])})})}}),w.fn.extend({queue:function(e,t){var n=2;return"string"!=typeof e&&(t=e,e="fx",n--),arguments.length<n?w.queue(this[0],e):void 0===t?this:this.each(function(){var n=w.queue(this,e,t);w._queueHooks(this,e),"fx"===e&&"inprogress"!==n[0]&&w.dequeue(this,e)})},dequeue:function(e){return this.each(function(){w.dequeue(this,e)})},clearQueue:function(e){return this.queue(e||"fx",[])},promise:function(e,t){var n,r=1,i=w.Deferred(),o=this,a=this.length,s=function(){--r||i.resolveWith(o,[o])};"string"!=typeof e&&(t=e,e=void 0),e=e||"fx";while(a--)(n=J.get(o[a],e+"queueHooks"))&&n.empty&&(r++,n.empty.add(s));return s(),i.promise(t)}});var re=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,ie=new RegExp("^(?:([+-])=|)("+re+")([a-z%]*)$","i"),oe=["Top","Right","Bottom","Left"],ae=function(e,t){return"none"===(e=t||e).style.display||""===e.style.display&&w.contains(e.ownerDocument,e)&&"none"===w.css(e,"display")},se=function(e,t,n,r){var i,o,a={};for(o in t)a[o]=e.style[o],e.style[o]=t[o];i=n.apply(e,r||[]);for(o in t)e.style[o]=a[o];return i};function ue(e,t,n,r){var i,o,a=20,s=r?function(){return r.cur()}:function(){return w.css(e,t,"")},u=s(),l=n&&n[3]||(w.cssNumber[t]?"":"px"),c=(w.cssNumber[t]||"px"!==l&&+u)&&ie.exec(w.css(e,t));if(c&&c[3]!==l){u/=2,l=l||c[3],c=+u||1;while(a--)w.style(e,t,c+l),(1-o)*(1-(o=s()/u||.5))<=0&&(a=0),c/=o;c*=2,w.style(e,t,c+l),n=n||[]}return n&&(c=+c||+u||0,i=n[1]?c+(n[1]+1)*n[2]:+n[2],r&&(r.unit=l,r.start=c,r.end=i)),i}var le={};function ce(e){var t,n=e.ownerDocument,r=e.nodeName,i=le[r];return i||(t=n.body.appendChild(n.createElement(r)),i=w.css(t,"display"),t.parentNode.removeChild(t),"none"===i&&(i="block"),le[r]=i,i)}function fe(e,t){for(var n,r,i=[],o=0,a=e.length;o<a;o++)(r=e[o]).style&&(n=r.style.display,t?("none"===n&&(i[o]=J.get(r,"display")||null,i[o]||(r.style.display="")),""===r.style.display&&ae(r)&&(i[o]=ce(r))):"none"!==n&&(i[o]="none",J.set(r,"display",n)));for(o=0;o<a;o++)null!=i[o]&&(e[o].style.display=i[o]);return e}w.fn.extend({show:function(){return fe(this,!0)},hide:function(){return fe(this)},toggle:function(e){return"boolean"==typeof e?e?this.show():this.hide():this.each(function(){ae(this)?w(this).show():w(this).hide()})}});var pe=/^(?:checkbox|radio)$/i,de=/<([a-z][^\/\0>\x20\t\r\n\f]+)/i,he=/^$|^module$|\/(?:java|ecma)script/i,ge={option:[1,"<select multiple='multiple'>","</select>"],thead:[1,"<table>","</table>"],col:[2,"<table><colgroup>","</colgroup></table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:[0,"",""]};ge.optgroup=ge.option,ge.tbody=ge.tfoot=ge.colgroup=ge.caption=ge.thead,ge.th=ge.td;function ye(e,t){var n;return n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[],void 0===t||t&&N(e,t)?w.merge([e],n):n}function ve(e,t){for(var n=0,r=e.length;n<r;n++)J.set(e[n],"globalEval",!t||J.get(t[n],"globalEval"))}var me=/<|&#?\w+;/;function xe(e,t,n,r,i){for(var o,a,s,u,l,c,f=t.createDocumentFragment(),p=[],d=0,h=e.length;d<h;d++)if((o=e[d])||0===o)if("object"===x(o))w.merge(p,o.nodeType?[o]:o);else if(me.test(o)){a=a||f.appendChild(t.createElement("div")),s=(de.exec(o)||["",""])[1].toLowerCase(),u=ge[s]||ge._default,a.innerHTML=u[1]+w.htmlPrefilter(o)+u[2],c=u[0];while(c--)a=a.lastChild;w.merge(p,a.childNodes),(a=f.firstChild).textContent=""}else p.push(t.createTextNode(o));f.textContent="",d=0;while(o=p[d++])if(r&&w.inArray(o,r)>-1)i&&i.push(o);else if(l=w.contains(o.ownerDocument,o),a=ye(f.appendChild(o),"script"),l&&ve(a),n){c=0;while(o=a[c++])he.test(o.type||"")&&n.push(o)}return f}!function(){var e=r.createDocumentFragment().appendChild(r.createElement("div")),t=r.createElement("input");t.setAttribute("type","radio"),t.setAttribute("checked","checked"),t.setAttribute("name","t"),e.appendChild(t),h.checkClone=e.cloneNode(!0).cloneNode(!0).lastChild.checked,e.innerHTML="<textarea>x</textarea>",h.noCloneChecked=!!e.cloneNode(!0).lastChild.defaultValue}();var be=r.documentElement,we=/^key/,Te=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,Ce=/^([^.]*)(?:\.(.+)|)/;function Ee(){return!0}function ke(){return!1}function Se(){try{return r.activeElement}catch(e){}}function De(e,t,n,r,i,o){var a,s;if("object"==typeof t){"string"!=typeof n&&(r=r||n,n=void 0);for(s in t)De(e,s,n,r,t[s],o);return e}if(null==r&&null==i?(i=n,r=n=void 0):null==i&&("string"==typeof n?(i=r,r=void 0):(i=r,r=n,n=void 0)),!1===i)i=ke;else if(!i)return e;return 1===o&&(a=i,(i=function(e){return w().off(e),a.apply(this,arguments)}).guid=a.guid||(a.guid=w.guid++)),e.each(function(){w.event.add(this,t,i,r,n)})}w.event={global:{},add:function(e,t,n,r,i){var o,a,s,u,l,c,f,p,d,h,g,y=J.get(e);if(y){n.handler&&(n=(o=n).handler,i=o.selector),i&&w.find.matchesSelector(be,i),n.guid||(n.guid=w.guid++),(u=y.events)||(u=y.events={}),(a=y.handle)||(a=y.handle=function(t){return"undefined"!=typeof w&&w.event.triggered!==t.type?w.event.dispatch.apply(e,arguments):void 0}),l=(t=(t||"").match(M)||[""]).length;while(l--)d=g=(s=Ce.exec(t[l])||[])[1],h=(s[2]||"").split(".").sort(),d&&(f=w.event.special[d]||{},d=(i?f.delegateType:f.bindType)||d,f=w.event.special[d]||{},c=w.extend({type:d,origType:g,data:r,handler:n,guid:n.guid,selector:i,needsContext:i&&w.expr.match.needsContext.test(i),namespace:h.join(".")},o),(p=u[d])||((p=u[d]=[]).delegateCount=0,f.setup&&!1!==f.setup.call(e,r,h,a)||e.addEventListener&&e.addEventListener(d,a)),f.add&&(f.add.call(e,c),c.handler.guid||(c.handler.guid=n.guid)),i?p.splice(p.delegateCount++,0,c):p.push(c),w.event.global[d]=!0)}},remove:function(e,t,n,r,i){var o,a,s,u,l,c,f,p,d,h,g,y=J.hasData(e)&&J.get(e);if(y&&(u=y.events)){l=(t=(t||"").match(M)||[""]).length;while(l--)if(s=Ce.exec(t[l])||[],d=g=s[1],h=(s[2]||"").split(".").sort(),d){f=w.event.special[d]||{},p=u[d=(r?f.delegateType:f.bindType)||d]||[],s=s[2]&&new RegExp("(^|\\.)"+h.join("\\.(?:.*\\.|)")+"(\\.|$)"),a=o=p.length;while(o--)c=p[o],!i&&g!==c.origType||n&&n.guid!==c.guid||s&&!s.test(c.namespace)||r&&r!==c.selector&&("**"!==r||!c.selector)||(p.splice(o,1),c.selector&&p.delegateCount--,f.remove&&f.remove.call(e,c));a&&!p.length&&(f.teardown&&!1!==f.teardown.call(e,h,y.handle)||w.removeEvent(e,d,y.handle),delete u[d])}else for(d in u)w.event.remove(e,d+t[l],n,r,!0);w.isEmptyObject(u)&&J.remove(e,"handle events")}},dispatch:function(e){var t=w.event.fix(e),n,r,i,o,a,s,u=new Array(arguments.length),l=(J.get(this,"events")||{})[t.type]||[],c=w.event.special[t.type]||{};for(u[0]=t,n=1;n<arguments.length;n++)u[n]=arguments[n];if(t.delegateTarget=this,!c.preDispatch||!1!==c.preDispatch.call(this,t)){s=w.event.handlers.call(this,t,l),n=0;while((o=s[n++])&&!t.isPropagationStopped()){t.currentTarget=o.elem,r=0;while((a=o.handlers[r++])&&!t.isImmediatePropagationStopped())t.rnamespace&&!t.rnamespace.test(a.namespace)||(t.handleObj=a,t.data=a.data,void 0!==(i=((w.event.special[a.origType]||{}).handle||a.handler).apply(o.elem,u))&&!1===(t.result=i)&&(t.preventDefault(),t.stopPropagation()))}return c.postDispatch&&c.postDispatch.call(this,t),t.result}},handlers:function(e,t){var n,r,i,o,a,s=[],u=t.delegateCount,l=e.target;if(u&&l.nodeType&&!("click"===e.type&&e.button>=1))for(;l!==this;l=l.parentNode||this)if(1===l.nodeType&&("click"!==e.type||!0!==l.disabled)){for(o=[],a={},n=0;n<u;n++)void 0===a[i=(r=t[n]).selector+" "]&&(a[i]=r.needsContext?w(i,this).index(l)>-1:w.find(i,this,null,[l]).length),a[i]&&o.push(r);o.length&&s.push({elem:l,handlers:o})}return l=this,u<t.length&&s.push({elem:l,handlers:t.slice(u)}),s},addProp:function(e,t){Object.defineProperty(w.Event.prototype,e,{enumerable:!0,configurable:!0,get:g(t)?function(){if(this.originalEvent)return t(this.originalEvent)}:function(){if(this.originalEvent)return this.originalEvent[e]},set:function(t){Object.defineProperty(this,e,{enumerable:!0,configurable:!0,writable:!0,value:t})}})},fix:function(e){return e[w.expando]?e:new w.Event(e)},special:{load:{noBubble:!0},focus:{trigger:function(){if(this!==Se()&&this.focus)return this.focus(),!1},delegateType:"focusin"},blur:{trigger:function(){if(this===Se()&&this.blur)return this.blur(),!1},delegateType:"focusout"},click:{trigger:function(){if("checkbox"===this.type&&this.click&&N(this,"input"))return this.click(),!1},_default:function(e){return N(e.target,"a")}},beforeunload:{postDispatch:function(e){void 0!==e.result&&e.originalEvent&&(e.originalEvent.returnValue=e.result)}}}},w.removeEvent=function(e,t,n){e.removeEventListener&&e.removeEventListener(t,n)},w.Event=function(e,t){if(!(this instanceof w.Event))return new w.Event(e,t);e&&e.type?(this.originalEvent=e,this.type=e.type,this.isDefaultPrevented=e.defaultPrevented||void 0===e.defaultPrevented&&!1===e.returnValue?Ee:ke,this.target=e.target&&3===e.target.nodeType?e.target.parentNode:e.target,this.currentTarget=e.currentTarget,this.relatedTarget=e.relatedTarget):this.type=e,t&&w.extend(this,t),this.timeStamp=e&&e.timeStamp||Date.now(),this[w.expando]=!0},w.Event.prototype={constructor:w.Event,isDefaultPrevented:ke,isPropagationStopped:ke,isImmediatePropagationStopped:ke,isSimulated:!1,preventDefault:function(){var e=this.originalEvent;this.isDefaultPrevented=Ee,e&&!this.isSimulated&&e.preventDefault()},stopPropagation:function(){var e=this.originalEvent;this.isPropagationStopped=Ee,e&&!this.isSimulated&&e.stopPropagation()},stopImmediatePropagation:function(){var e=this.originalEvent;this.isImmediatePropagationStopped=Ee,e&&!this.isSimulated&&e.stopImmediatePropagation(),this.stopPropagation()}},w.each({altKey:!0,bubbles:!0,cancelable:!0,changedTouches:!0,ctrlKey:!0,detail:!0,eventPhase:!0,metaKey:!0,pageX:!0,pageY:!0,shiftKey:!0,view:!0,"char":!0,charCode:!0,key:!0,keyCode:!0,button:!0,buttons:!0,clientX:!0,clientY:!0,offsetX:!0,offsetY:!0,pointerId:!0,pointerType:!0,screenX:!0,screenY:!0,targetTouches:!0,toElement:!0,touches:!0,which:function(e){var t=e.button;return null==e.which&&we.test(e.type)?null!=e.charCode?e.charCode:e.keyCode:!e.which&&void 0!==t&&Te.test(e.type)?1&t?1:2&t?3:4&t?2:0:e.which}},w.event.addProp),w.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(e,t){w.event.special[e]={delegateType:t,bindType:t,handle:function(e){var n,r=this,i=e.relatedTarget,o=e.handleObj;return i&&(i===r||w.contains(r,i))||(e.type=o.origType,n=o.handler.apply(this,arguments),e.type=t),n}}}),w.fn.extend({on:function(e,t,n,r){return De(this,e,t,n,r)},one:function(e,t,n,r){return De(this,e,t,n,r,1)},off:function(e,t,n){var r,i;if(e&&e.preventDefault&&e.handleObj)return r=e.handleObj,w(e.delegateTarget).off(r.namespace?r.origType+"."+r.namespace:r.origType,r.selector,r.handler),this;if("object"==typeof e){for(i in e)this.off(i,t,e[i]);return this}return!1!==t&&"function"!=typeof t||(n=t,t=void 0),!1===n&&(n=ke),this.each(function(){w.event.remove(this,e,n,t)})}});var Ne=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([a-z][^\/\0>\x20\t\r\n\f]*)[^>]*)\/>/gi,Ae=/<script|<style|<link/i,je=/checked\s*(?:[^=]|=\s*.checked.)/i,qe=/^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g;function Le(e,t){return N(e,"table")&&N(11!==t.nodeType?t:t.firstChild,"tr")?w(e).children("tbody")[0]||e:e}function He(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function Oe(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Pe(e,t){var n,r,i,o,a,s,u,l;if(1===t.nodeType){if(J.hasData(e)&&(o=J.access(e),a=J.set(t,o),l=o.events)){delete a.handle,a.events={};for(i in l)for(n=0,r=l[i].length;n<r;n++)w.event.add(t,i,l[i][n])}K.hasData(e)&&(s=K.access(e),u=w.extend({},s),K.set(t,u))}}function Me(e,t){var n=t.nodeName.toLowerCase();"input"===n&&pe.test(e.type)?t.checked=e.checked:"input"!==n&&"textarea"!==n||(t.defaultValue=e.defaultValue)}function Re(e,t,n,r){t=a.apply([],t);var i,o,s,u,l,c,f=0,p=e.length,d=p-1,y=t[0],v=g(y);if(v||p>1&&"string"==typeof y&&!h.checkClone&&je.test(y))return e.each(function(i){var o=e.eq(i);v&&(t[0]=y.call(this,i,o.html())),Re(o,t,n,r)});if(p&&(i=xe(t,e[0].ownerDocument,!1,e,r),o=i.firstChild,1===i.childNodes.length&&(i=o),o||r)){for(u=(s=w.map(ye(i,"script"),He)).length;f<p;f++)l=i,f!==d&&(l=w.clone(l,!0,!0),u&&w.merge(s,ye(l,"script"))),n.call(e[f],l,f);if(u)for(c=s[s.length-1].ownerDocument,w.map(s,Oe),f=0;f<u;f++)l=s[f],he.test(l.type||"")&&!J.access(l,"globalEval")&&w.contains(c,l)&&(l.src&&"module"!==(l.type||"").toLowerCase()?w._evalUrl&&w._evalUrl(l.src):m(l.textContent.replace(qe,""),c,l))}return e}function Ie(e,t,n){for(var r,i=t?w.filter(t,e):e,o=0;null!=(r=i[o]);o++)n||1!==r.nodeType||w.cleanData(ye(r)),r.parentNode&&(n&&w.contains(r.ownerDocument,r)&&ve(ye(r,"script")),r.parentNode.removeChild(r));return e}w.extend({htmlPrefilter:function(e){return e.replace(Ne,"<$1></$2>")},clone:function(e,t,n){var r,i,o,a,s=e.cloneNode(!0),u=w.contains(e.ownerDocument,e);if(!(h.noCloneChecked||1!==e.nodeType&&11!==e.nodeType||w.isXMLDoc(e)))for(a=ye(s),r=0,i=(o=ye(e)).length;r<i;r++)Me(o[r],a[r]);if(t)if(n)for(o=o||ye(e),a=a||ye(s),r=0,i=o.length;r<i;r++)Pe(o[r],a[r]);else Pe(e,s);return(a=ye(s,"script")).length>0&&ve(a,!u&&ye(e,"script")),s},cleanData:function(e){for(var t,n,r,i=w.event.special,o=0;void 0!==(n=e[o]);o++)if(Y(n)){if(t=n[J.expando]){if(t.events)for(r in t.events)i[r]?w.event.remove(n,r):w.removeEvent(n,r,t.handle);n[J.expando]=void 0}n[K.expando]&&(n[K.expando]=void 0)}}}),w.fn.extend({detach:function(e){return Ie(this,e,!0)},remove:function(e){return Ie(this,e)},text:function(e){return z(this,function(e){return void 0===e?w.text(this):this.empty().each(function(){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(this.textContent=e)})},null,e,arguments.length)},append:function(){return Re(this,arguments,function(e){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||Le(this,e).appendChild(e)})},prepend:function(){return Re(this,arguments,function(e){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var t=Le(this,e);t.insertBefore(e,t.firstChild)}})},before:function(){return Re(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this)})},after:function(){return Re(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this.nextSibling)})},empty:function(){for(var e,t=0;null!=(e=this[t]);t++)1===e.nodeType&&(w.cleanData(ye(e,!1)),e.textContent="");return this},clone:function(e,t){return e=null!=e&&e,t=null==t?e:t,this.map(function(){return w.clone(this,e,t)})},html:function(e){return z(this,function(e){var t=this[0]||{},n=0,r=this.length;if(void 0===e&&1===t.nodeType)return t.innerHTML;if("string"==typeof e&&!Ae.test(e)&&!ge[(de.exec(e)||["",""])[1].toLowerCase()]){e=w.htmlPrefilter(e);try{for(;n<r;n++)1===(t=this[n]||{}).nodeType&&(w.cleanData(ye(t,!1)),t.innerHTML=e);t=0}catch(e){}}t&&this.empty().append(e)},null,e,arguments.length)},replaceWith:function(){var e=[];return Re(this,arguments,function(t){var n=this.parentNode;w.inArray(this,e)<0&&(w.cleanData(ye(this)),n&&n.replaceChild(t,this))},e)}}),w.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(e,t){w.fn[e]=function(e){for(var n,r=[],i=w(e),o=i.length-1,a=0;a<=o;a++)n=a===o?this:this.clone(!0),w(i[a])[t](n),s.apply(r,n.get());return this.pushStack(r)}});var We=new RegExp("^("+re+")(?!px)[a-z%]+$","i"),$e=function(t){var n=t.ownerDocument.defaultView;return n&&n.opener||(n=e),n.getComputedStyle(t)},Be=new RegExp(oe.join("|"),"i");!function(){function t(){if(c){l.style.cssText="position:absolute;left:-11111px;width:60px;margin-top:1px;padding:0;border:0",c.style.cssText="position:relative;display:block;box-sizing:border-box;overflow:scroll;margin:auto;border:1px;padding:1px;width:60%;top:1%",be.appendChild(l).appendChild(c);var t=e.getComputedStyle(c);i="1%"!==t.top,u=12===n(t.marginLeft),c.style.right="60%",s=36===n(t.right),o=36===n(t.width),c.style.position="absolute",a=36===c.offsetWidth||"absolute",be.removeChild(l),c=null}}function n(e){return Math.round(parseFloat(e))}var i,o,a,s,u,l=r.createElement("div"),c=r.createElement("div");c.style&&(c.style.backgroundClip="content-box",c.cloneNode(!0).style.backgroundClip="",h.clearCloneStyle="content-box"===c.style.backgroundClip,w.extend(h,{boxSizingReliable:function(){return t(),o},pixelBoxStyles:function(){return t(),s},pixelPosition:function(){return t(),i},reliableMarginLeft:function(){return t(),u},scrollboxSize:function(){return t(),a}}))}();function Fe(e,t,n){var r,i,o,a,s=e.style;return(n=n||$e(e))&&(""!==(a=n.getPropertyValue(t)||n[t])||w.contains(e.ownerDocument,e)||(a=w.style(e,t)),!h.pixelBoxStyles()&&We.test(a)&&Be.test(t)&&(r=s.width,i=s.minWidth,o=s.maxWidth,s.minWidth=s.maxWidth=s.width=a,a=n.width,s.width=r,s.minWidth=i,s.maxWidth=o)),void 0!==a?a+"":a}function _e(e,t){return{get:function(){if(!e())return(this.get=t).apply(this,arguments);delete this.get}}}var ze=/^(none|table(?!-c[ea]).+)/,Xe=/^--/,Ue={position:"absolute",visibility:"hidden",display:"block"},Ve={letterSpacing:"0",fontWeight:"400"},Ge=["Webkit","Moz","ms"],Ye=r.createElement("div").style;function Qe(e){if(e in Ye)return e;var t=e[0].toUpperCase()+e.slice(1),n=Ge.length;while(n--)if((e=Ge[n]+t)in Ye)return e}function Je(e){var t=w.cssProps[e];return t||(t=w.cssProps[e]=Qe(e)||e),t}function Ke(e,t,n){var r=ie.exec(t);return r?Math.max(0,r[2]-(n||0))+(r[3]||"px"):t}function Ze(e,t,n,r,i,o){var a="width"===t?1:0,s=0,u=0;if(n===(r?"border":"content"))return 0;for(;a<4;a+=2)"margin"===n&&(u+=w.css(e,n+oe[a],!0,i)),r?("content"===n&&(u-=w.css(e,"padding"+oe[a],!0,i)),"margin"!==n&&(u-=w.css(e,"border"+oe[a]+"Width",!0,i))):(u+=w.css(e,"padding"+oe[a],!0,i),"padding"!==n?u+=w.css(e,"border"+oe[a]+"Width",!0,i):s+=w.css(e,"border"+oe[a]+"Width",!0,i));return!r&&o>=0&&(u+=Math.max(0,Math.ceil(e["offset"+t[0].toUpperCase()+t.slice(1)]-o-u-s-.5))),u}function et(e,t,n){var r=$e(e),i=Fe(e,t,r),o="border-box"===w.css(e,"boxSizing",!1,r),a=o;if(We.test(i)){if(!n)return i;i="auto"}return a=a&&(h.boxSizingReliable()||i===e.style[t]),("auto"===i||!parseFloat(i)&&"inline"===w.css(e,"display",!1,r))&&(i=e["offset"+t[0].toUpperCase()+t.slice(1)],a=!0),(i=parseFloat(i)||0)+Ze(e,t,n||(o?"border":"content"),a,r,i)+"px"}w.extend({cssHooks:{opacity:{get:function(e,t){if(t){var n=Fe(e,"opacity");return""===n?"1":n}}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{},style:function(e,t,n,r){if(e&&3!==e.nodeType&&8!==e.nodeType&&e.style){var i,o,a,s=G(t),u=Xe.test(t),l=e.style;if(u||(t=Je(s)),a=w.cssHooks[t]||w.cssHooks[s],void 0===n)return a&&"get"in a&&void 0!==(i=a.get(e,!1,r))?i:l[t];"string"==(o=typeof n)&&(i=ie.exec(n))&&i[1]&&(n=ue(e,t,i),o="number"),null!=n&&n===n&&("number"===o&&(n+=i&&i[3]||(w.cssNumber[s]?"":"px")),h.clearCloneStyle||""!==n||0!==t.indexOf("background")||(l[t]="inherit"),a&&"set"in a&&void 0===(n=a.set(e,n,r))||(u?l.setProperty(t,n):l[t]=n))}},css:function(e,t,n,r){var i,o,a,s=G(t);return Xe.test(t)||(t=Je(s)),(a=w.cssHooks[t]||w.cssHooks[s])&&"get"in a&&(i=a.get(e,!0,n)),void 0===i&&(i=Fe(e,t,r)),"normal"===i&&t in Ve&&(i=Ve[t]),""===n||n?(o=parseFloat(i),!0===n||isFinite(o)?o||0:i):i}}),w.each(["height","width"],function(e,t){w.cssHooks[t]={get:function(e,n,r){if(n)return!ze.test(w.css(e,"display"))||e.getClientRects().length&&e.getBoundingClientRect().width?et(e,t,r):se(e,Ue,function(){return et(e,t,r)})},set:function(e,n,r){var i,o=$e(e),a="border-box"===w.css(e,"boxSizing",!1,o),s=r&&Ze(e,t,r,a,o);return a&&h.scrollboxSize()===o.position&&(s-=Math.ceil(e["offset"+t[0].toUpperCase()+t.slice(1)]-parseFloat(o[t])-Ze(e,t,"border",!1,o)-.5)),s&&(i=ie.exec(n))&&"px"!==(i[3]||"px")&&(e.style[t]=n,n=w.css(e,t)),Ke(e,n,s)}}}),w.cssHooks.marginLeft=_e(h.reliableMarginLeft,function(e,t){if(t)return(parseFloat(Fe(e,"marginLeft"))||e.getBoundingClientRect().left-se(e,{marginLeft:0},function(){return e.getBoundingClientRect().left}))+"px"}),w.each({margin:"",padding:"",border:"Width"},function(e,t){w.cssHooks[e+t]={expand:function(n){for(var r=0,i={},o="string"==typeof n?n.split(" "):[n];r<4;r++)i[e+oe[r]+t]=o[r]||o[r-2]||o[0];return i}},"margin"!==e&&(w.cssHooks[e+t].set=Ke)}),w.fn.extend({css:function(e,t){return z(this,function(e,t,n){var r,i,o={},a=0;if(Array.isArray(t)){for(r=$e(e),i=t.length;a<i;a++)o[t[a]]=w.css(e,t[a],!1,r);return o}return void 0!==n?w.style(e,t,n):w.css(e,t)},e,t,arguments.length>1)}});function tt(e,t,n,r,i){return new tt.prototype.init(e,t,n,r,i)}w.Tween=tt,tt.prototype={constructor:tt,init:function(e,t,n,r,i,o){this.elem=e,this.prop=n,this.easing=i||w.easing._default,this.options=t,this.start=this.now=this.cur(),this.end=r,this.unit=o||(w.cssNumber[n]?"":"px")},cur:function(){var e=tt.propHooks[this.prop];return e&&e.get?e.get(this):tt.propHooks._default.get(this)},run:function(e){var t,n=tt.propHooks[this.prop];return this.options.duration?this.pos=t=w.easing[this.easing](e,this.options.duration*e,0,1,this.options.duration):this.pos=t=e,this.now=(this.end-this.start)*t+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),n&&n.set?n.set(this):tt.propHooks._default.set(this),this}},tt.prototype.init.prototype=tt.prototype,tt.propHooks={_default:{get:function(e){var t;return 1!==e.elem.nodeType||null!=e.elem[e.prop]&&null==e.elem.style[e.prop]?e.elem[e.prop]:(t=w.css(e.elem,e.prop,""))&&"auto"!==t?t:0},set:function(e){w.fx.step[e.prop]?w.fx.step[e.prop](e):1!==e.elem.nodeType||null==e.elem.style[w.cssProps[e.prop]]&&!w.cssHooks[e.prop]?e.elem[e.prop]=e.now:w.style(e.elem,e.prop,e.now+e.unit)}}},tt.propHooks.scrollTop=tt.propHooks.scrollLeft={set:function(e){e.elem.nodeType&&e.elem.parentNode&&(e.elem[e.prop]=e.now)}},w.easing={linear:function(e){return e},swing:function(e){return.5-Math.cos(e*Math.PI)/2},_default:"swing"},w.fx=tt.prototype.init,w.fx.step={};var nt,rt,it=/^(?:toggle|show|hide)$/,ot=/queueHooks$/;function at(){rt&&(!1===r.hidden&&e.requestAnimationFrame?e.requestAnimationFrame(at):e.setTimeout(at,w.fx.interval),w.fx.tick())}function st(){return e.setTimeout(function(){nt=void 0}),nt=Date.now()}function ut(e,t){var n,r=0,i={height:e};for(t=t?1:0;r<4;r+=2-t)i["margin"+(n=oe[r])]=i["padding"+n]=e;return t&&(i.opacity=i.width=e),i}function lt(e,t,n){for(var r,i=(pt.tweeners[t]||[]).concat(pt.tweeners["*"]),o=0,a=i.length;o<a;o++)if(r=i[o].call(n,t,e))return r}function ct(e,t,n){var r,i,o,a,s,u,l,c,f="width"in t||"height"in t,p=this,d={},h=e.style,g=e.nodeType&&ae(e),y=J.get(e,"fxshow");n.queue||(null==(a=w._queueHooks(e,"fx")).unqueued&&(a.unqueued=0,s=a.empty.fire,a.empty.fire=function(){a.unqueued||s()}),a.unqueued++,p.always(function(){p.always(function(){a.unqueued--,w.queue(e,"fx").length||a.empty.fire()})}));for(r in t)if(i=t[r],it.test(i)){if(delete t[r],o=o||"toggle"===i,i===(g?"hide":"show")){if("show"!==i||!y||void 0===y[r])continue;g=!0}d[r]=y&&y[r]||w.style(e,r)}if((u=!w.isEmptyObject(t))||!w.isEmptyObject(d)){f&&1===e.nodeType&&(n.overflow=[h.overflow,h.overflowX,h.overflowY],null==(l=y&&y.display)&&(l=J.get(e,"display")),"none"===(c=w.css(e,"display"))&&(l?c=l:(fe([e],!0),l=e.style.display||l,c=w.css(e,"display"),fe([e]))),("inline"===c||"inline-block"===c&&null!=l)&&"none"===w.css(e,"float")&&(u||(p.done(function(){h.display=l}),null==l&&(c=h.display,l="none"===c?"":c)),h.display="inline-block")),n.overflow&&(h.overflow="hidden",p.always(function(){h.overflow=n.overflow[0],h.overflowX=n.overflow[1],h.overflowY=n.overflow[2]})),u=!1;for(r in d)u||(y?"hidden"in y&&(g=y.hidden):y=J.access(e,"fxshow",{display:l}),o&&(y.hidden=!g),g&&fe([e],!0),p.done(function(){g||fe([e]),J.remove(e,"fxshow");for(r in d)w.style(e,r,d[r])})),u=lt(g?y[r]:0,r,p),r in y||(y[r]=u.start,g&&(u.end=u.start,u.start=0))}}function ft(e,t){var n,r,i,o,a;for(n in e)if(r=G(n),i=t[r],o=e[n],Array.isArray(o)&&(i=o[1],o=e[n]=o[0]),n!==r&&(e[r]=o,delete e[n]),(a=w.cssHooks[r])&&"expand"in a){o=a.expand(o),delete e[r];for(n in o)n in e||(e[n]=o[n],t[n]=i)}else t[r]=i}function pt(e,t,n){var r,i,o=0,a=pt.prefilters.length,s=w.Deferred().always(function(){delete u.elem}),u=function(){if(i)return!1;for(var t=nt||st(),n=Math.max(0,l.startTime+l.duration-t),r=1-(n/l.duration||0),o=0,a=l.tweens.length;o<a;o++)l.tweens[o].run(r);return s.notifyWith(e,[l,r,n]),r<1&&a?n:(a||s.notifyWith(e,[l,1,0]),s.resolveWith(e,[l]),!1)},l=s.promise({elem:e,props:w.extend({},t),opts:w.extend(!0,{specialEasing:{},easing:w.easing._default},n),originalProperties:t,originalOptions:n,startTime:nt||st(),duration:n.duration,tweens:[],createTween:function(t,n){var r=w.Tween(e,l.opts,t,n,l.opts.specialEasing[t]||l.opts.easing);return l.tweens.push(r),r},stop:function(t){var n=0,r=t?l.tweens.length:0;if(i)return this;for(i=!0;n<r;n++)l.tweens[n].run(1);return t?(s.notifyWith(e,[l,1,0]),s.resolveWith(e,[l,t])):s.rejectWith(e,[l,t]),this}}),c=l.props;for(ft(c,l.opts.specialEasing);o<a;o++)if(r=pt.prefilters[o].call(l,e,c,l.opts))return g(r.stop)&&(w._queueHooks(l.elem,l.opts.queue).stop=r.stop.bind(r)),r;return w.map(c,lt,l),g(l.opts.start)&&l.opts.start.call(e,l),l.progress(l.opts.progress).done(l.opts.done,l.opts.complete).fail(l.opts.fail).always(l.opts.always),w.fx.timer(w.extend(u,{elem:e,anim:l,queue:l.opts.queue})),l}w.Animation=w.extend(pt,{tweeners:{"*":[function(e,t){var n=this.createTween(e,t);return ue(n.elem,e,ie.exec(t),n),n}]},tweener:function(e,t){g(e)?(t=e,e=["*"]):e=e.match(M);for(var n,r=0,i=e.length;r<i;r++)n=e[r],pt.tweeners[n]=pt.tweeners[n]||[],pt.tweeners[n].unshift(t)},prefilters:[ct],prefilter:function(e,t){t?pt.prefilters.unshift(e):pt.prefilters.push(e)}}),w.speed=function(e,t,n){var r=e&&"object"==typeof e?w.extend({},e):{complete:n||!n&&t||g(e)&&e,duration:e,easing:n&&t||t&&!g(t)&&t};return w.fx.off?r.duration=0:"number"!=typeof r.duration&&(r.duration in w.fx.speeds?r.duration=w.fx.speeds[r.duration]:r.duration=w.fx.speeds._default),null!=r.queue&&!0!==r.queue||(r.queue="fx"),r.old=r.complete,r.complete=function(){g(r.old)&&r.old.call(this),r.queue&&w.dequeue(this,r.queue)},r},w.fn.extend({fadeTo:function(e,t,n,r){return this.filter(ae).css("opacity",0).show().end().animate({opacity:t},e,n,r)},animate:function(e,t,n,r){var i=w.isEmptyObject(e),o=w.speed(t,n,r),a=function(){var t=pt(this,w.extend({},e),o);(i||J.get(this,"finish"))&&t.stop(!0)};return a.finish=a,i||!1===o.queue?this.each(a):this.queue(o.queue,a)},stop:function(e,t,n){var r=function(e){var t=e.stop;delete e.stop,t(n)};return"string"!=typeof e&&(n=t,t=e,e=void 0),t&&!1!==e&&this.queue(e||"fx",[]),this.each(function(){var t=!0,i=null!=e&&e+"queueHooks",o=w.timers,a=J.get(this);if(i)a[i]&&a[i].stop&&r(a[i]);else for(i in a)a[i]&&a[i].stop&&ot.test(i)&&r(a[i]);for(i=o.length;i--;)o[i].elem!==this||null!=e&&o[i].queue!==e||(o[i].anim.stop(n),t=!1,o.splice(i,1));!t&&n||w.dequeue(this,e)})},finish:function(e){return!1!==e&&(e=e||"fx"),this.each(function(){var t,n=J.get(this),r=n[e+"queue"],i=n[e+"queueHooks"],o=w.timers,a=r?r.length:0;for(n.finish=!0,w.queue(this,e,[]),i&&i.stop&&i.stop.call(this,!0),t=o.length;t--;)o[t].elem===this&&o[t].queue===e&&(o[t].anim.stop(!0),o.splice(t,1));for(t=0;t<a;t++)r[t]&&r[t].finish&&r[t].finish.call(this);delete n.finish})}}),w.each(["toggle","show","hide"],function(e,t){var n=w.fn[t];w.fn[t]=function(e,r,i){return null==e||"boolean"==typeof e?n.apply(this,arguments):this.animate(ut(t,!0),e,r,i)}}),w.each({slideDown:ut("show"),slideUp:ut("hide"),slideToggle:ut("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(e,t){w.fn[e]=function(e,n,r){return this.animate(t,e,n,r)}}),w.timers=[],w.fx.tick=function(){var e,t=0,n=w.timers;for(nt=Date.now();t<n.length;t++)(e=n[t])()||n[t]!==e||n.splice(t--,1);n.length||w.fx.stop(),nt=void 0},w.fx.timer=function(e){w.timers.push(e),w.fx.start()},w.fx.interval=13,w.fx.start=function(){rt||(rt=!0,at())},w.fx.stop=function(){rt=null},w.fx.speeds={slow:600,fast:200,_default:400},w.fn.delay=function(t,n){return t=w.fx?w.fx.speeds[t]||t:t,n=n||"fx",this.queue(n,function(n,r){var i=e.setTimeout(n,t);r.stop=function(){e.clearTimeout(i)}})},function(){var e=r.createElement("input"),t=r.createElement("select").appendChild(r.createElement("option"));e.type="checkbox",h.checkOn=""!==e.value,h.optSelected=t.selected,(e=r.createElement("input")).value="t",e.type="radio",h.radioValue="t"===e.value}();var dt,ht=w.expr.attrHandle;w.fn.extend({attr:function(e,t){return z(this,w.attr,e,t,arguments.length>1)},removeAttr:function(e){return this.each(function(){w.removeAttr(this,e)})}}),w.extend({attr:function(e,t,n){var r,i,o=e.nodeType;if(3!==o&&8!==o&&2!==o)return"undefined"==typeof e.getAttribute?w.prop(e,t,n):(1===o&&w.isXMLDoc(e)||(i=w.attrHooks[t.toLowerCase()]||(w.expr.match.bool.test(t)?dt:void 0)),void 0!==n?null===n?void w.removeAttr(e,t):i&&"set"in i&&void 0!==(r=i.set(e,n,t))?r:(e.setAttribute(t,n+""),n):i&&"get"in i&&null!==(r=i.get(e,t))?r:null==(r=w.find.attr(e,t))?void 0:r)},attrHooks:{type:{set:function(e,t){if(!h.radioValue&&"radio"===t&&N(e,"input")){var n=e.value;return e.setAttribute("type",t),n&&(e.value=n),t}}}},removeAttr:function(e,t){var n,r=0,i=t&&t.match(M);if(i&&1===e.nodeType)while(n=i[r++])e.removeAttribute(n)}}),dt={set:function(e,t,n){return!1===t?w.removeAttr(e,n):e.setAttribute(n,n),n}},w.each(w.expr.match.bool.source.match(/\w+/g),function(e,t){var n=ht[t]||w.find.attr;ht[t]=function(e,t,r){var i,o,a=t.toLowerCase();return r||(o=ht[a],ht[a]=i,i=null!=n(e,t,r)?a:null,ht[a]=o),i}});var gt=/^(?:input|select|textarea|button)$/i,yt=/^(?:a|area)$/i;w.fn.extend({prop:function(e,t){return z(this,w.prop,e,t,arguments.length>1)},removeProp:function(e){return this.each(function(){delete this[w.propFix[e]||e]})}}),w.extend({prop:function(e,t,n){var r,i,o=e.nodeType;if(3!==o&&8!==o&&2!==o)return 1===o&&w.isXMLDoc(e)||(t=w.propFix[t]||t,i=w.propHooks[t]),void 0!==n?i&&"set"in i&&void 0!==(r=i.set(e,n,t))?r:e[t]=n:i&&"get"in i&&null!==(r=i.get(e,t))?r:e[t]},propHooks:{tabIndex:{get:function(e){var t=w.find.attr(e,"tabindex");return t?parseInt(t,10):gt.test(e.nodeName)||yt.test(e.nodeName)&&e.href?0:-1}}},propFix:{"for":"htmlFor","class":"className"}}),h.optSelected||(w.propHooks.selected={get:function(e){var t=e.parentNode;return t&&t.parentNode&&t.parentNode.selectedIndex,null},set:function(e){var t=e.parentNode;t&&(t.selectedIndex,t.parentNode&&t.parentNode.selectedIndex)}}),w.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){w.propFix[this.toLowerCase()]=this});function vt(e){return(e.match(M)||[]).join(" ")}function mt(e){return e.getAttribute&&e.getAttribute("class")||""}function xt(e){return Array.isArray(e)?e:"string"==typeof e?e.match(M)||[]:[]}w.fn.extend({addClass:function(e){var t,n,r,i,o,a,s,u=0;if(g(e))return this.each(function(t){w(this).addClass(e.call(this,t,mt(this)))});if((t=xt(e)).length)while(n=this[u++])if(i=mt(n),r=1===n.nodeType&&" "+vt(i)+" "){a=0;while(o=t[a++])r.indexOf(" "+o+" ")<0&&(r+=o+" ");i!==(s=vt(r))&&n.setAttribute("class",s)}return this},removeClass:function(e){var t,n,r,i,o,a,s,u=0;if(g(e))return this.each(function(t){w(this).removeClass(e.call(this,t,mt(this)))});if(!arguments.length)return this.attr("class","");if((t=xt(e)).length)while(n=this[u++])if(i=mt(n),r=1===n.nodeType&&" "+vt(i)+" "){a=0;while(o=t[a++])while(r.indexOf(" "+o+" ")>-1)r=r.replace(" "+o+" "," ");i!==(s=vt(r))&&n.setAttribute("class",s)}return this},toggleClass:function(e,t){var n=typeof e,r="string"===n||Array.isArray(e);return"boolean"==typeof t&&r?t?this.addClass(e):this.removeClass(e):g(e)?this.each(function(n){w(this).toggleClass(e.call(this,n,mt(this),t),t)}):this.each(function(){var t,i,o,a;if(r){i=0,o=w(this),a=xt(e);while(t=a[i++])o.hasClass(t)?o.removeClass(t):o.addClass(t)}else void 0!==e&&"boolean"!==n||((t=mt(this))&&J.set(this,"__className__",t),this.setAttribute&&this.setAttribute("class",t||!1===e?"":J.get(this,"__className__")||""))})},hasClass:function(e){var t,n,r=0;t=" "+e+" ";while(n=this[r++])if(1===n.nodeType&&(" "+vt(mt(n))+" ").indexOf(t)>-1)return!0;return!1}});var bt=/\r/g;w.fn.extend({val:function(e){var t,n,r,i=this[0];{if(arguments.length)return r=g(e),this.each(function(n){var i;1===this.nodeType&&(null==(i=r?e.call(this,n,w(this).val()):e)?i="":"number"==typeof i?i+="":Array.isArray(i)&&(i=w.map(i,function(e){return null==e?"":e+""})),(t=w.valHooks[this.type]||w.valHooks[this.nodeName.toLowerCase()])&&"set"in t&&void 0!==t.set(this,i,"value")||(this.value=i))});if(i)return(t=w.valHooks[i.type]||w.valHooks[i.nodeName.toLowerCase()])&&"get"in t&&void 0!==(n=t.get(i,"value"))?n:"string"==typeof(n=i.value)?n.replace(bt,""):null==n?"":n}}}),w.extend({valHooks:{option:{get:function(e){var t=w.find.attr(e,"value");return null!=t?t:vt(w.text(e))}},select:{get:function(e){var t,n,r,i=e.options,o=e.selectedIndex,a="select-one"===e.type,s=a?null:[],u=a?o+1:i.length;for(r=o<0?u:a?o:0;r<u;r++)if(((n=i[r]).selected||r===o)&&!n.disabled&&(!n.parentNode.disabled||!N(n.parentNode,"optgroup"))){if(t=w(n).val(),a)return t;s.push(t)}return s},set:function(e,t){var n,r,i=e.options,o=w.makeArray(t),a=i.length;while(a--)((r=i[a]).selected=w.inArray(w.valHooks.option.get(r),o)>-1)&&(n=!0);return n||(e.selectedIndex=-1),o}}}}),w.each(["radio","checkbox"],function(){w.valHooks[this]={set:function(e,t){if(Array.isArray(t))return e.checked=w.inArray(w(e).val(),t)>-1}},h.checkOn||(w.valHooks[this].get=function(e){return null===e.getAttribute("value")?"on":e.value})}),h.focusin="onfocusin"in e;var wt=/^(?:focusinfocus|focusoutblur)$/,Tt=function(e){e.stopPropagation()};w.extend(w.event,{trigger:function(t,n,i,o){var a,s,u,l,c,p,d,h,v=[i||r],m=f.call(t,"type")?t.type:t,x=f.call(t,"namespace")?t.namespace.split("."):[];if(s=h=u=i=i||r,3!==i.nodeType&&8!==i.nodeType&&!wt.test(m+w.event.triggered)&&(m.indexOf(".")>-1&&(m=(x=m.split(".")).shift(),x.sort()),c=m.indexOf(":")<0&&"on"+m,t=t[w.expando]?t:new w.Event(m,"object"==typeof t&&t),t.isTrigger=o?2:3,t.namespace=x.join("."),t.rnamespace=t.namespace?new RegExp("(^|\\.)"+x.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,t.result=void 0,t.target||(t.target=i),n=null==n?[t]:w.makeArray(n,[t]),d=w.event.special[m]||{},o||!d.trigger||!1!==d.trigger.apply(i,n))){if(!o&&!d.noBubble&&!y(i)){for(l=d.delegateType||m,wt.test(l+m)||(s=s.parentNode);s;s=s.parentNode)v.push(s),u=s;u===(i.ownerDocument||r)&&v.push(u.defaultView||u.parentWindow||e)}a=0;while((s=v[a++])&&!t.isPropagationStopped())h=s,t.type=a>1?l:d.bindType||m,(p=(J.get(s,"events")||{})[t.type]&&J.get(s,"handle"))&&p.apply(s,n),(p=c&&s[c])&&p.apply&&Y(s)&&(t.result=p.apply(s,n),!1===t.result&&t.preventDefault());return t.type=m,o||t.isDefaultPrevented()||d._default&&!1!==d._default.apply(v.pop(),n)||!Y(i)||c&&g(i[m])&&!y(i)&&((u=i[c])&&(i[c]=null),w.event.triggered=m,t.isPropagationStopped()&&h.addEventListener(m,Tt),i[m](),t.isPropagationStopped()&&h.removeEventListener(m,Tt),w.event.triggered=void 0,u&&(i[c]=u)),t.result}},simulate:function(e,t,n){var r=w.extend(new w.Event,n,{type:e,isSimulated:!0});w.event.trigger(r,null,t)}}),w.fn.extend({trigger:function(e,t){return this.each(function(){w.event.trigger(e,t,this)})},triggerHandler:function(e,t){var n=this[0];if(n)return w.event.trigger(e,t,n,!0)}}),h.focusin||w.each({focus:"focusin",blur:"focusout"},function(e,t){var n=function(e){w.event.simulate(t,e.target,w.event.fix(e))};w.event.special[t]={setup:function(){var r=this.ownerDocument||this,i=J.access(r,t);i||r.addEventListener(e,n,!0),J.access(r,t,(i||0)+1)},teardown:function(){var r=this.ownerDocument||this,i=J.access(r,t)-1;i?J.access(r,t,i):(r.removeEventListener(e,n,!0),J.remove(r,t))}}});var Ct=e.location,Et=Date.now(),kt=/\?/;w.parseXML=function(t){var n;if(!t||"string"!=typeof t)return null;try{n=(new e.DOMParser).parseFromString(t,"text/xml")}catch(e){n=void 0}return n&&!n.getElementsByTagName("parsererror").length||w.error("Invalid XML: "+t),n};var St=/\[\]$/,Dt=/\r?\n/g,Nt=/^(?:submit|button|image|reset|file)$/i,At=/^(?:input|select|textarea|keygen)/i;function jt(e,t,n,r){var i;if(Array.isArray(t))w.each(t,function(t,i){n||St.test(e)?r(e,i):jt(e+"["+("object"==typeof i&&null!=i?t:"")+"]",i,n,r)});else if(n||"object"!==x(t))r(e,t);else for(i in t)jt(e+"["+i+"]",t[i],n,r)}w.param=function(e,t){var n,r=[],i=function(e,t){var n=g(t)?t():t;r[r.length]=encodeURIComponent(e)+"="+encodeURIComponent(null==n?"":n)};if(Array.isArray(e)||e.jquery&&!w.isPlainObject(e))w.each(e,function(){i(this.name,this.value)});else for(n in e)jt(n,e[n],t,i);return r.join("&")},w.fn.extend({serialize:function(){return w.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var e=w.prop(this,"elements");return e?w.makeArray(e):this}).filter(function(){var e=this.type;return this.name&&!w(this).is(":disabled")&&At.test(this.nodeName)&&!Nt.test(e)&&(this.checked||!pe.test(e))}).map(function(e,t){var n=w(this).val();return null==n?null:Array.isArray(n)?w.map(n,function(e){return{name:t.name,value:e.replace(Dt,"\r\n")}}):{name:t.name,value:n.replace(Dt,"\r\n")}}).get()}});var qt=/%20/g,Lt=/#.*$/,Ht=/([?&])_=[^&]*/,Ot=/^(.*?):[ \t]*([^\r\n]*)$/gm,Pt=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,Mt=/^(?:GET|HEAD)$/,Rt=/^\/\//,It={},Wt={},$t="*/".concat("*"),Bt=r.createElement("a");Bt.href=Ct.href;function Ft(e){return function(t,n){"string"!=typeof t&&(n=t,t="*");var r,i=0,o=t.toLowerCase().match(M)||[];if(g(n))while(r=o[i++])"+"===r[0]?(r=r.slice(1)||"*",(e[r]=e[r]||[]).unshift(n)):(e[r]=e[r]||[]).push(n)}}function _t(e,t,n,r){var i={},o=e===Wt;function a(s){var u;return i[s]=!0,w.each(e[s]||[],function(e,s){var l=s(t,n,r);return"string"!=typeof l||o||i[l]?o?!(u=l):void 0:(t.dataTypes.unshift(l),a(l),!1)}),u}return a(t.dataTypes[0])||!i["*"]&&a("*")}function zt(e,t){var n,r,i=w.ajaxSettings.flatOptions||{};for(n in t)void 0!==t[n]&&((i[n]?e:r||(r={}))[n]=t[n]);return r&&w.extend(!0,e,r),e}function Xt(e,t,n){var r,i,o,a,s=e.contents,u=e.dataTypes;while("*"===u[0])u.shift(),void 0===r&&(r=e.mimeType||t.getResponseHeader("Content-Type"));if(r)for(i in s)if(s[i]&&s[i].test(r)){u.unshift(i);break}if(u[0]in n)o=u[0];else{for(i in n){if(!u[0]||e.converters[i+" "+u[0]]){o=i;break}a||(a=i)}o=o||a}if(o)return o!==u[0]&&u.unshift(o),n[o]}function Ut(e,t,n,r){var i,o,a,s,u,l={},c=e.dataTypes.slice();if(c[1])for(a in e.converters)l[a.toLowerCase()]=e.converters[a];o=c.shift();while(o)if(e.responseFields[o]&&(n[e.responseFields[o]]=t),!u&&r&&e.dataFilter&&(t=e.dataFilter(t,e.dataType)),u=o,o=c.shift())if("*"===o)o=u;else if("*"!==u&&u!==o){if(!(a=l[u+" "+o]||l["* "+o]))for(i in l)if((s=i.split(" "))[1]===o&&(a=l[u+" "+s[0]]||l["* "+s[0]])){!0===a?a=l[i]:!0!==l[i]&&(o=s[0],c.unshift(s[1]));break}if(!0!==a)if(a&&e["throws"])t=a(t);else try{t=a(t)}catch(e){return{state:"parsererror",error:a?e:"No conversion from "+u+" to "+o}}}return{state:"success",data:t}}w.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:Ct.href,type:"GET",isLocal:Pt.test(Ct.protocol),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":$t,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":JSON.parse,"text xml":w.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(e,t){return t?zt(zt(e,w.ajaxSettings),t):zt(w.ajaxSettings,e)},ajaxPrefilter:Ft(It),ajaxTransport:Ft(Wt),ajax:function(t,n){"object"==typeof t&&(n=t,t=void 0),n=n||{};var i,o,a,s,u,l,c,f,p,d,h=w.ajaxSetup({},n),g=h.context||h,y=h.context&&(g.nodeType||g.jquery)?w(g):w.event,v=w.Deferred(),m=w.Callbacks("once memory"),x=h.statusCode||{},b={},T={},C="canceled",E={readyState:0,getResponseHeader:function(e){var t;if(c){if(!s){s={};while(t=Ot.exec(a))s[t[1].toLowerCase()]=t[2]}t=s[e.toLowerCase()]}return null==t?null:t},getAllResponseHeaders:function(){return c?a:null},setRequestHeader:function(e,t){return null==c&&(e=T[e.toLowerCase()]=T[e.toLowerCase()]||e,b[e]=t),this},overrideMimeType:function(e){return null==c&&(h.mimeType=e),this},statusCode:function(e){var t;if(e)if(c)E.always(e[E.status]);else for(t in e)x[t]=[x[t],e[t]];return this},abort:function(e){var t=e||C;return i&&i.abort(t),k(0,t),this}};if(v.promise(E),h.url=((t||h.url||Ct.href)+"").replace(Rt,Ct.protocol+"//"),h.type=n.method||n.type||h.method||h.type,h.dataTypes=(h.dataType||"*").toLowerCase().match(M)||[""],null==h.crossDomain){l=r.createElement("a");try{l.href=h.url,l.href=l.href,h.crossDomain=Bt.protocol+"//"+Bt.host!=l.protocol+"//"+l.host}catch(e){h.crossDomain=!0}}if(h.data&&h.processData&&"string"!=typeof h.data&&(h.data=w.param(h.data,h.traditional)),_t(It,h,n,E),c)return E;(f=w.event&&h.global)&&0==w.active++&&w.event.trigger("ajaxStart"),h.type=h.type.toUpperCase(),h.hasContent=!Mt.test(h.type),o=h.url.replace(Lt,""),h.hasContent?h.data&&h.processData&&0===(h.contentType||"").indexOf("application/x-www-form-urlencoded")&&(h.data=h.data.replace(qt,"+")):(d=h.url.slice(o.length),h.data&&(h.processData||"string"==typeof h.data)&&(o+=(kt.test(o)?"&":"?")+h.data,delete h.data),!1===h.cache&&(o=o.replace(Ht,"$1"),d=(kt.test(o)?"&":"?")+"_="+Et+++d),h.url=o+d),h.ifModified&&(w.lastModified[o]&&E.setRequestHeader("If-Modified-Since",w.lastModified[o]),w.etag[o]&&E.setRequestHeader("If-None-Match",w.etag[o])),(h.data&&h.hasContent&&!1!==h.contentType||n.contentType)&&E.setRequestHeader("Content-Type",h.contentType),E.setRequestHeader("Accept",h.dataTypes[0]&&h.accepts[h.dataTypes[0]]?h.accepts[h.dataTypes[0]]+("*"!==h.dataTypes[0]?", "+$t+"; q=0.01":""):h.accepts["*"]);for(p in h.headers)E.setRequestHeader(p,h.headers[p]);if(h.beforeSend&&(!1===h.beforeSend.call(g,E,h)||c))return E.abort();if(C="abort",m.add(h.complete),E.done(h.success),E.fail(h.error),i=_t(Wt,h,n,E)){if(E.readyState=1,f&&y.trigger("ajaxSend",[E,h]),c)return E;h.async&&h.timeout>0&&(u=e.setTimeout(function(){E.abort("timeout")},h.timeout));try{c=!1,i.send(b,k)}catch(e){if(c)throw e;k(-1,e)}}else k(-1,"No Transport");function k(t,n,r,s){var l,p,d,b,T,C=n;c||(c=!0,u&&e.clearTimeout(u),i=void 0,a=s||"",E.readyState=t>0?4:0,l=t>=200&&t<300||304===t,r&&(b=Xt(h,E,r)),b=Ut(h,b,E,l),l?(h.ifModified&&((T=E.getResponseHeader("Last-Modified"))&&(w.lastModified[o]=T),(T=E.getResponseHeader("etag"))&&(w.etag[o]=T)),204===t||"HEAD"===h.type?C="nocontent":304===t?C="notmodified":(C=b.state,p=b.data,l=!(d=b.error))):(d=C,!t&&C||(C="error",t<0&&(t=0))),E.status=t,E.statusText=(n||C)+"",l?v.resolveWith(g,[p,C,E]):v.rejectWith(g,[E,C,d]),E.statusCode(x),x=void 0,f&&y.trigger(l?"ajaxSuccess":"ajaxError",[E,h,l?p:d]),m.fireWith(g,[E,C]),f&&(y.trigger("ajaxComplete",[E,h]),--w.active||w.event.trigger("ajaxStop")))}return E},getJSON:function(e,t,n){return w.get(e,t,n,"json")},getScript:function(e,t){return w.get(e,void 0,t,"script")}}),w.each(["get","post"],function(e,t){w[t]=function(e,n,r,i){return g(n)&&(i=i||r,r=n,n=void 0),w.ajax(w.extend({url:e,type:t,dataType:i,data:n,success:r},w.isPlainObject(e)&&e))}}),w._evalUrl=function(e){return w.ajax({url:e,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,"throws":!0})},w.fn.extend({wrapAll:function(e){var t;return this[0]&&(g(e)&&(e=e.call(this[0])),t=w(e,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&t.insertBefore(this[0]),t.map(function(){var e=this;while(e.firstElementChild)e=e.firstElementChild;return e}).append(this)),this},wrapInner:function(e){return g(e)?this.each(function(t){w(this).wrapInner(e.call(this,t))}):this.each(function(){var t=w(this),n=t.contents();n.length?n.wrapAll(e):t.append(e)})},wrap:function(e){var t=g(e);return this.each(function(n){w(this).wrapAll(t?e.call(this,n):e)})},unwrap:function(e){return this.parent(e).not("body").each(function(){w(this).replaceWith(this.childNodes)}),this}}),w.expr.pseudos.hidden=function(e){return!w.expr.pseudos.visible(e)},w.expr.pseudos.visible=function(e){return!!(e.offsetWidth||e.offsetHeight||e.getClientRects().length)},w.ajaxSettings.xhr=function(){try{return new e.XMLHttpRequest}catch(e){}};var Vt={0:200,1223:204},Gt=w.ajaxSettings.xhr();h.cors=!!Gt&&"withCredentials"in Gt,h.ajax=Gt=!!Gt,w.ajaxTransport(function(t){var n,r;if(h.cors||Gt&&!t.crossDomain)return{send:function(i,o){var a,s=t.xhr();if(s.open(t.type,t.url,t.async,t.username,t.password),t.xhrFields)for(a in t.xhrFields)s[a]=t.xhrFields[a];t.mimeType&&s.overrideMimeType&&s.overrideMimeType(t.mimeType),t.crossDomain||i["X-Requested-With"]||(i["X-Requested-With"]="XMLHttpRequest");for(a in i)s.setRequestHeader(a,i[a]);n=function(e){return function(){n&&(n=r=s.onload=s.onerror=s.onabort=s.ontimeout=s.onreadystatechange=null,"abort"===e?s.abort():"error"===e?"number"!=typeof s.status?o(0,"error"):o(s.status,s.statusText):o(Vt[s.status]||s.status,s.statusText,"text"!==(s.responseType||"text")||"string"!=typeof s.responseText?{binary:s.response}:{text:s.responseText},s.getAllResponseHeaders()))}},s.onload=n(),r=s.onerror=s.ontimeout=n("error"),void 0!==s.onabort?s.onabort=r:s.onreadystatechange=function(){4===s.readyState&&e.setTimeout(function(){n&&r()})},n=n("abort");try{s.send(t.hasContent&&t.data||null)}catch(e){if(n)throw e}},abort:function(){n&&n()}}}),w.ajaxPrefilter(function(e){e.crossDomain&&(e.contents.script=!1)}),w.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(e){return w.globalEval(e),e}}}),w.ajaxPrefilter("script",function(e){void 0===e.cache&&(e.cache=!1),e.crossDomain&&(e.type="GET")}),w.ajaxTransport("script",function(e){if(e.crossDomain){var t,n;return{send:function(i,o){t=w("<script>").prop({charset:e.scriptCharset,src:e.url}).on("load error",n=function(e){t.remove(),n=null,e&&o("error"===e.type?404:200,e.type)}),r.head.appendChild(t[0])},abort:function(){n&&n()}}}});var Yt=[],Qt=/(=)\?(?=&|$)|\?\?/;w.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Yt.pop()||w.expando+"_"+Et++;return this[e]=!0,e}}),w.ajaxPrefilter("json jsonp",function(t,n,r){var i,o,a,s=!1!==t.jsonp&&(Qt.test(t.url)?"url":"string"==typeof t.data&&0===(t.contentType||"").indexOf("application/x-www-form-urlencoded")&&Qt.test(t.data)&&"data");if(s||"jsonp"===t.dataTypes[0])return i=t.jsonpCallback=g(t.jsonpCallback)?t.jsonpCallback():t.jsonpCallback,s?t[s]=t[s].replace(Qt,"$1"+i):!1!==t.jsonp&&(t.url+=(kt.test(t.url)?"&":"?")+t.jsonp+"="+i),t.converters["script json"]=function(){return a||w.error(i+" was not called"),a[0]},t.dataTypes[0]="json",o=e[i],e[i]=function(){a=arguments},r.always(function(){void 0===o?w(e).removeProp(i):e[i]=o,t[i]&&(t.jsonpCallback=n.jsonpCallback,Yt.push(i)),a&&g(o)&&o(a[0]),a=o=void 0}),"script"}),h.createHTMLDocument=function(){var e=r.implementation.createHTMLDocument("").body;return e.innerHTML="<form></form><form></form>",2===e.childNodes.length}(),w.parseHTML=function(e,t,n){if("string"!=typeof e)return[];"boolean"==typeof t&&(n=t,t=!1);var i,o,a;return t||(h.createHTMLDocument?((i=(t=r.implementation.createHTMLDocument("")).createElement("base")).href=r.location.href,t.head.appendChild(i)):t=r),o=A.exec(e),a=!n&&[],o?[t.createElement(o[1])]:(o=xe([e],t,a),a&&a.length&&w(a).remove(),w.merge([],o.childNodes))},w.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return s>-1&&(r=vt(e.slice(s)),e=e.slice(0,s)),g(t)?(n=t,t=void 0):t&&"object"==typeof t&&(i="POST"),a.length>0&&w.ajax({url:e,type:i||"GET",dataType:"html",data:t}).done(function(e){o=arguments,a.html(r?w("<div>").append(w.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},w.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){w.fn[t]=function(e){return this.on(t,e)}}),w.expr.pseudos.animated=function(e){return w.grep(w.timers,function(t){return e===t.elem}).length},w.offset={setOffset:function(e,t,n){var r,i,o,a,s,u,l,c=w.css(e,"position"),f=w(e),p={};"static"===c&&(e.style.position="relative"),s=f.offset(),o=w.css(e,"top"),u=w.css(e,"left"),(l=("absolute"===c||"fixed"===c)&&(o+u).indexOf("auto")>-1)?(a=(r=f.position()).top,i=r.left):(a=parseFloat(o)||0,i=parseFloat(u)||0),g(t)&&(t=t.call(e,n,w.extend({},s))),null!=t.top&&(p.top=t.top-s.top+a),null!=t.left&&(p.left=t.left-s.left+i),"using"in t?t.using.call(e,p):f.css(p)}},w.fn.extend({offset:function(e){if(arguments.length)return void 0===e?this:this.each(function(t){w.offset.setOffset(this,e,t)});var t,n,r=this[0];if(r)return r.getClientRects().length?(t=r.getBoundingClientRect(),n=r.ownerDocument.defaultView,{top:t.top+n.pageYOffset,left:t.left+n.pageXOffset}):{top:0,left:0}},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===w.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===w.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=w(e).offset()).top+=w.css(e,"borderTopWidth",!0),i.left+=w.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-w.css(r,"marginTop",!0),left:t.left-i.left-w.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===w.css(e,"position"))e=e.offsetParent;return e||be})}}),w.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(e,t){var n="pageYOffset"===t;w.fn[e]=function(r){return z(this,function(e,r,i){var o;if(y(e)?o=e:9===e.nodeType&&(o=e.defaultView),void 0===i)return o?o[t]:e[r];o?o.scrollTo(n?o.pageXOffset:i,n?i:o.pageYOffset):e[r]=i},e,r,arguments.length)}}),w.each(["top","left"],function(e,t){w.cssHooks[t]=_e(h.pixelPosition,function(e,n){if(n)return n=Fe(e,t),We.test(n)?w(e).position()[t]+"px":n})}),w.each({Height:"height",Width:"width"},function(e,t){w.each({padding:"inner"+e,content:t,"":"outer"+e},function(n,r){w.fn[r]=function(i,o){var a=arguments.length&&(n||"boolean"!=typeof i),s=n||(!0===i||!0===o?"margin":"border");return z(this,function(t,n,i){var o;return y(t)?0===r.indexOf("outer")?t["inner"+e]:t.document.documentElement["client"+e]:9===t.nodeType?(o=t.documentElement,Math.max(t.body["scroll"+e],o["scroll"+e],t.body["offset"+e],o["offset"+e],o["client"+e])):void 0===i?w.css(t,n,s):w.style(t,n,i,s)},t,a?i:void 0,a)}})}),w.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,t){w.fn[t]=function(e,n){return arguments.length>0?this.on(t,null,e,n):this.trigger(t)}}),w.fn.extend({hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),w.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)}}),w.proxy=function(e,t){var n,r,i;if("string"==typeof t&&(n=e[t],t=e,e=n),g(e))return r=o.call(arguments,2),i=function(){return e.apply(t||this,r.concat(o.call(arguments)))},i.guid=e.guid=e.guid||w.guid++,i},w.holdReady=function(e){e?w.readyWait++:w.ready(!0)},w.isArray=Array.isArray,w.parseJSON=JSON.parse,w.nodeName=N,w.isFunction=g,w.isWindow=y,w.camelCase=G,w.type=x,w.now=Date.now,w.isNumeric=function(e){var t=w.type(e);return("number"===t||"string"===t)&&!isNaN(e-parseFloat(e))},"function"==typeof define&&define.amd&&define("jquery",[],function(){return w});var Jt=e.jQuery,Kt=e.$;return w.noConflict=function(t){return e.$===w&&(e.$=Kt),t&&e.jQuery===w&&(e.jQuery=Jt),w},t||(e.jQuery=e.$=w),w}); diff --git a/public/static/common/js/simpleAlert.js b/public/static/common/js/simpleAlert.js deleted file mode 100644 index 87ee391..0000000 --- a/public/static/common/js/simpleAlert.js +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Created by Dasate on 2017/9/14. - * QQ361899429 - */ -var simpleAlert = function (opts) { - //设置默认参数 - var opt = { - "closeAll": false, - "content": "", - "buttons": {} - } - //合并参数 - var option = $.extend(opt, opts); - //事件 - var dialog = {} - var $simpleAlert = $('<div class="simpleAlert">'); - var $shelter = $('<div class="simpleAlertShelter">'); - var $simpleAlertBody = $('<div class="simpleAlertBody">'); - var $simpleAlertBodyClose = $('<img class="simpleAlertBodyClose" src="img/close.png" height="14" width="14"/>'); - var $simpleAlertBodyContent = $('<p class="simpleAlertBodyContent">' + option.content + '</p>'); - dialog.init = function () { - $simpleAlertBody.append($simpleAlertBodyClose).append($simpleAlertBodyContent); - var num = 0; - var only = false; - var onlyArr = []; - for (var i = 0; i < 2; i++) { - for (var key in option.buttons) { - switch (i) { - case 0: - onlyArr.push(key); - break; - case 1: - if (onlyArr.length <= 1) { - only = true; - } else { - only = false; - } - num++; - var $btn = $('<button class="simpleAlertBtn simpleAlertBtn' + num + '">' + key + '</button>') - // console.log(key) - - $btn.bind("click", option.buttons[key=='OK'?'确定':key=='Yes'?'是':key=='No'?'否':key=='Cancel'?'取消':key]); - if (only) { - $btn.addClass("onlyOne") - } - $simpleAlertBody.append($btn); - break; - } - - } - } - $simpleAlert.append($shelter).append($simpleAlertBody); - $("body").append($simpleAlert); - $simpleAlertBody.show().animate({"marginTop":"-128px","opacity":"1"},300); - } - //右上角关闭按键事件 - $simpleAlertBodyClose.bind("click", function () { - option.closeAll=false; - dialog.close(); - }) - dialog.close = function () { - if(option.closeAll){ - $(".simpleAlert").remove() - }else { - $simpleAlertBody.animate({"marginTop": "-188px", "opacity": "0"}, 200, function () { - $(".simpleAlert").last().remove() - }); - } - } - dialog.init(); - return dialog; -} \ No newline at end of file diff --git a/public/static/common/js/three/Lut.js b/public/static/common/js/three/Lut.js deleted file mode 100644 index d7797ed..0000000 --- a/public/static/common/js/three/Lut.js +++ /dev/null @@ -1,200 +0,0 @@ -(function(){ - class Lut { - - constructor( colormap, count = 32 ) { - - this.isLut = true; - - this.lut = []; - this.map = []; - this.n = 0; - this.minV = 0; - this.maxV = 1; - - this.setColorMap( colormap, count ); - - } - - set( value ) { - - if ( value.isLut === true ) { - - this.copy( value ); - - } - - return this; - - } - - setMin( min ) { - - this.minV = min; - - return this; - - } - - setMax( max ) { - - this.maxV = max; - - return this; - - } - - setColorMap( colormap, count = 32 ) { - - this.map = ColorMapKeywords[ colormap ] || ColorMapKeywords.rainbow; - this.n = count; - - const step = 1.0 / this.n; - const minColor = new THREE.Color(); - const maxColor = new THREE.Color(); - - this.lut.length = 0; - - // sample at 0 - - this.lut.push( new THREE.Color( this.map[ 0 ][ 1 ] ) ); - - // sample at 1/n, ..., (n-1)/n - - for ( let i = 1; i < count; i ++ ) { - - const alpha = i * step; - - for ( let j = 0; j < this.map.length - 1; j ++ ) { - - if ( alpha > this.map[ j ][ 0 ] && alpha <= this.map[ j + 1 ][ 0 ] ) { - - const min = this.map[ j ][ 0 ]; - const max = this.map[ j + 1 ][ 0 ]; - - minColor.setHex( this.map[ j ][ 1 ], THREE.LinearSRGBColorSpace ); - maxColor.setHex( this.map[ j + 1 ][ 1 ], THREE.LinearSRGBColorSpace ); - - const color = new THREE.Color().lerpColors( minColor, maxColor, ( alpha - min ) / ( max - min ) ); - - this.lut.push( color ); - - } - - } - - } - - // sample at 1 - - this.lut.push( new THREE.Color( this.map[ this.map.length - 1 ][ 1 ] ) ); - - return this; - - } - - copy( lut ) { - - this.lut = lut.lut; - this.map = lut.map; - this.n = lut.n; - this.minV = lut.minV; - this.maxV = lut.maxV; - - return this; - - } - - getColor( alpha ) { - - alpha = THREE.MathUtils.clamp( alpha, this.minV, this.maxV ); - - alpha = ( alpha - this.minV ) / ( this.maxV - this.minV ); - - const colorPosition = Math.round( alpha * this.n ); - - return this.lut[ colorPosition ]; - - } - - addColorMap( name, arrayOfColors ) { - - ColorMapKeywords[ name ] = arrayOfColors; - - return this; - - } - - createCanvas() { - - const canvas = document.createElement( 'canvas' ); - canvas.width = 1; - canvas.height = this.n; - - this.updateCanvas( canvas ); - - return canvas; - - } - - updateCanvas( canvas ) { - - const ctx = canvas.getContext( '2d', { alpha: false } ); - - const imageData = ctx.getImageData( 0, 0, 1, this.n ); - - const data = imageData.data; - - let k = 0; - - const step = 1.0 / this.n; - - const minColor = new THREE.Color(); - const maxColor = new THREE.Color(); - const finalColor = new THREE.Color(); - - for ( let i = 1; i >= 0; i -= step ) { - - for ( let j = this.map.length - 1; j >= 0; j -- ) { - - if ( i < this.map[ j ][ 0 ] && i >= this.map[ j - 1 ][ 0 ] ) { - - const min = this.map[ j - 1 ][ 0 ]; - const max = this.map[ j ][ 0 ]; - - minColor.setHex( this.map[ j - 1 ][ 1 ], THREE.LinearSRGBColorSpace ); - maxColor.setHex( this.map[ j ][ 1 ], THREE.LinearSRGBColorSpace ); - - finalColor.lerpColors( minColor, maxColor, ( i - min ) / ( max - min ) ); - - data[ k * 4 ] = Math.round( finalColor.r * 255 ); - data[ k * 4 + 1 ] = Math.round( finalColor.g * 255 ); - data[ k * 4 + 2 ] = Math.round( finalColor.b * 255 ); - data[ k * 4 + 3 ] = 255; - - k += 1; - - } - - } - - } - - ctx.putImageData( imageData, 0, 0 ); - - return canvas; - - } - - } - - const ColorMapKeywords = { - - 'rainbow': [[ 0.0, 0x0000FF ], [ 0.2, 0x00FFFF ], [ 0.5, 0x00FF00 ], [ 0.8, 0xFFFF00 ], [ 1.0, 0xFF0000 ]], - 'cooltowarm': [[ 0.0, 0x3C4EC2 ], [ 0.2, 0x9BBCFF ], [ 0.5, 0xDCDCDC ], [ 0.8, 0xF6A385 ], [ 1.0, 0xB40426 ]], - 'blackbody': [[ 0.0, 0x000000 ], [ 0.2, 0x780000 ], [ 0.5, 0xE63200 ], [ 0.8, 0xFFFF00 ], [ 1.0, 0xFFFFFF ]], - 'grayscale': [[ 0.0, 0x000000 ], [ 0.2, 0x404040 ], [ 0.5, 0x7F7F80 ], [ 0.8, 0xBFBFBF ], [ 1.0, 0xFFFFFF ]] - - }; - THREE.Lut = Lut; -})() - diff --git a/public/static/common/js/three/OrbitControls.js b/public/static/common/js/three/OrbitControls.js deleted file mode 100644 index 38eb76e..0000000 --- a/public/static/common/js/three/OrbitControls.js +++ /dev/null @@ -1,1065 +0,0 @@ -( function () { - - // This set of controls performs orbiting, dollying (zooming), and panning. - // Unlike TrackballControls, it maintains the "up" direction object.up (+Y by default). - // - // Orbit - left mouse / touch: one-finger move - // Zoom - middle mouse, or mousewheel / touch: two-finger spread or squish - // Pan - right mouse, or left mouse + ctrl/meta/shiftKey, or arrow keys / touch: two-finger move - - const _changeEvent = { - type: 'change' - }; - const _startEvent = { - type: 'start' - }; - const _endEvent = { - type: 'end' - }; - class OrbitControls extends THREE.EventDispatcher { - - constructor( object, domElement ) { - - super(); - this.object = object; - this.domElement = domElement; - this.domElement.style.touchAction = 'none'; // disable touch scroll - - // Set to false to disable this control - this.enabled = true; - - // "target" sets the location of focus, where the object orbits around - this.target = new THREE.Vector3(); - - // How far you can dolly in and out ( PerspectiveCamera only ) - this.minDistance = 0; - this.maxDistance = Infinity; - - // How far you can zoom in and out ( OrthographicCamera only ) - this.minZoom = 0; - this.maxZoom = Infinity; - - // How far you can orbit vertically, upper and lower limits. - // Range is 0 to Math.PI radians. - this.minPolarAngle = 0; // radians - this.maxPolarAngle = Math.PI; // radians - - // How far you can orbit horizontally, upper and lower limits. - // If set, the interval [ min, max ] must be a sub-interval of [ - 2 PI, 2 PI ], with ( max - min < 2 PI ) - this.minAzimuthAngle = - Infinity; // radians - this.maxAzimuthAngle = Infinity; // radians - - // Set to true to enable damping (inertia) - // If damping is enabled, you must call controls.update() in your animation loop - this.enableDamping = false; - this.dampingFactor = 0.05; - - // This option actually enables dollying in and out; left as "zoom" for backwards compatibility. - // Set to false to disable zooming - this.enableZoom = true; - this.zoomSpeed = 1.0; - - // Set to false to disable rotating - this.enableRotate = true; - this.rotateSpeed = 1.0; - - // Set to false to disable panning - this.enablePan = true; - this.panSpeed = 1.0; - this.screenSpacePanning = true; // if false, pan orthogonal to world-space direction camera.up - this.keyPanSpeed = 7.0; // pixels moved per arrow key push - - // Set to true to automatically rotate around the target - // If auto-rotate is enabled, you must call controls.update() in your animation loop - this.autoRotate = false; - this.autoRotateSpeed = 2.0; // 30 seconds per orbit when fps is 60 - - // The four arrow keys - this.keys = { - LEFT: 'ArrowLeft', - UP: 'ArrowUp', - RIGHT: 'ArrowRight', - BOTTOM: 'ArrowDown' - }; - - // Mouse buttons - this.mouseButtons = { - LEFT: THREE.MOUSE.ROTATE, - MIDDLE: THREE.MOUSE.DOLLY, - RIGHT: THREE.MOUSE.PAN - }; - - // Touch fingers - this.touches = { - ONE: THREE.TOUCH.ROTATE, - TWO: THREE.TOUCH.DOLLY_PAN - }; - - // for reset - this.target0 = this.target.clone(); - this.position0 = this.object.position.clone(); - this.zoom0 = this.object.zoom; - - // the target DOM element for key events - this._domElementKeyEvents = null; - - // - // public methods - // - - this.getPolarAngle = function () { - - return spherical.phi; - - }; - - this.getAzimuthalAngle = function () { - - return spherical.theta; - - }; - - this.getDistance = function () { - - return this.object.position.distanceTo( this.target ); - - }; - - this.listenToKeyEvents = function ( domElement ) { - - domElement.addEventListener( 'keydown', onKeyDown ); - this._domElementKeyEvents = domElement; - - }; - - this.saveState = function () { - - scope.target0.copy( scope.target ); - scope.position0.copy( scope.object.position ); - scope.zoom0 = scope.object.zoom; - - }; - - this.reset = function () { - - scope.target.copy( scope.target0 ); - scope.object.position.copy( scope.position0 ); - scope.object.zoom = scope.zoom0; - scope.object.updateProjectionMatrix(); - scope.dispatchEvent( _changeEvent ); - scope.update(); - state = STATE.NONE; - - }; - - // this method is exposed, but perhaps it would be better if we can make it private... - this.update = function () { - - const offset = new THREE.Vector3(); - - // so camera.up is the orbit axis - const quat = new THREE.Quaternion().setFromUnitVectors( object.up, new THREE.Vector3( 0, 1, 0 ) ); - const quatInverse = quat.clone().invert(); - const lastPosition = new THREE.Vector3(); - const lastQuaternion = new THREE.Quaternion(); - const twoPI = 2 * Math.PI; - return function update() { - - const position = scope.object.position; - offset.copy( position ).sub( scope.target ); - - // rotate offset to "y-axis-is-up" space - offset.applyQuaternion( quat ); - - // angle from z-axis around y-axis - spherical.setFromVector3( offset ); - if ( scope.autoRotate && state === STATE.NONE ) { - - rotateLeft( getAutoRotationAngle() ); - - } - - if ( scope.enableDamping ) { - - spherical.theta += sphericalDelta.theta * scope.dampingFactor; - spherical.phi += sphericalDelta.phi * scope.dampingFactor; - - } else { - - spherical.theta += sphericalDelta.theta; - spherical.phi += sphericalDelta.phi; - - } - - // restrict theta to be between desired limits - - let min = scope.minAzimuthAngle; - let max = scope.maxAzimuthAngle; - if ( isFinite( min ) && isFinite( max ) ) { - - if ( min < - Math.PI ) min += twoPI; else if ( min > Math.PI ) min -= twoPI; - if ( max < - Math.PI ) max += twoPI; else if ( max > Math.PI ) max -= twoPI; - if ( min <= max ) { - - spherical.theta = Math.max( min, Math.min( max, spherical.theta ) ); - - } else { - - spherical.theta = spherical.theta > ( min + max ) / 2 ? Math.max( min, spherical.theta ) : Math.min( max, spherical.theta ); - - } - - } - - // restrict phi to be between desired limits - spherical.phi = Math.max( scope.minPolarAngle, Math.min( scope.maxPolarAngle, spherical.phi ) ); - spherical.makeSafe(); - spherical.radius *= scale; - - // restrict radius to be between desired limits - spherical.radius = Math.max( scope.minDistance, Math.min( scope.maxDistance, spherical.radius ) ); - - // move target to panned location - - if ( scope.enableDamping === true ) { - - scope.target.addScaledVector( panOffset, scope.dampingFactor ); - - } else { - - scope.target.add( panOffset ); - - } - - offset.setFromSpherical( spherical ); - - // rotate offset back to "camera-up-vector-is-up" space - offset.applyQuaternion( quatInverse ); - position.copy( scope.target ).add( offset ); - scope.object.lookAt( scope.target ); - if ( scope.enableDamping === true ) { - - sphericalDelta.theta *= 1 - scope.dampingFactor; - sphericalDelta.phi *= 1 - scope.dampingFactor; - panOffset.multiplyScalar( 1 - scope.dampingFactor ); - - } else { - - sphericalDelta.set( 0, 0, 0 ); - panOffset.set( 0, 0, 0 ); - - } - - scale = 1; - - // update condition is: - // min(camera displacement, camera rotation in radians)^2 > EPS - // using small-angle approximation cos(x/2) = 1 - x^2 / 8 - - if ( zoomChanged || lastPosition.distanceToSquared( scope.object.position ) > EPS || 8 * ( 1 - lastQuaternion.dot( scope.object.quaternion ) ) > EPS ) { - - scope.dispatchEvent( _changeEvent ); - lastPosition.copy( scope.object.position ); - lastQuaternion.copy( scope.object.quaternion ); - zoomChanged = false; - return true; - - } - - return false; - - }; - - }(); - this.dispose = function () { - - scope.domElement.removeEventListener( 'contextmenu', onContextMenu ); - scope.domElement.removeEventListener( 'pointerdown', onPointerDown ); - scope.domElement.removeEventListener( 'pointercancel', onPointerCancel ); - scope.domElement.removeEventListener( 'wheel', onMouseWheel ); - scope.domElement.removeEventListener( 'pointermove', onPointerMove ); - scope.domElement.removeEventListener( 'pointerup', onPointerUp ); - if ( scope._domElementKeyEvents !== null ) { - - scope._domElementKeyEvents.removeEventListener( 'keydown', onKeyDown ); - - } - - //scope.dispatchEvent( { type: 'dispose' } ); // should this be added here? - - }; - - // - // internals - // - - const scope = this; - const STATE = { - NONE: - 1, - ROTATE: 0, - DOLLY: 1, - PAN: 2, - TOUCH_ROTATE: 3, - TOUCH_PAN: 4, - TOUCH_DOLLY_PAN: 5, - TOUCH_DOLLY_ROTATE: 6 - }; - let state = STATE.NONE; - const EPS = 0.000001; - - // current position in spherical coordinates - const spherical = new THREE.Spherical(); - const sphericalDelta = new THREE.Spherical(); - let scale = 1; - const panOffset = new THREE.Vector3(); - let zoomChanged = false; - const rotateStart = new THREE.Vector2(); - const rotateEnd = new THREE.Vector2(); - const rotateDelta = new THREE.Vector2(); - const panStart = new THREE.Vector2(); - const panEnd = new THREE.Vector2(); - const panDelta = new THREE.Vector2(); - const dollyStart = new THREE.Vector2(); - const dollyEnd = new THREE.Vector2(); - const dollyDelta = new THREE.Vector2(); - const pointers = []; - const pointerPositions = {}; - function getAutoRotationAngle() { - - return 2 * Math.PI / 60 / 60 * scope.autoRotateSpeed; - - } - - function getZoomScale() { - - return Math.pow( 0.95, scope.zoomSpeed ); - - } - - function rotateLeft( angle ) { - - sphericalDelta.theta -= angle; - - } - - function rotateUp( angle ) { - - sphericalDelta.phi -= angle; - - } - - const panLeft = function () { - - const v = new THREE.Vector3(); - return function panLeft( distance, objectMatrix ) { - - v.setFromMatrixColumn( objectMatrix, 0 ); // get X column of objectMatrix - v.multiplyScalar( - distance ); - panOffset.add( v ); - - }; - - }(); - const panUp = function () { - - const v = new THREE.Vector3(); - return function panUp( distance, objectMatrix ) { - - if ( scope.screenSpacePanning === true ) { - - v.setFromMatrixColumn( objectMatrix, 1 ); - - } else { - - v.setFromMatrixColumn( objectMatrix, 0 ); - v.crossVectors( scope.object.up, v ); - - } - - v.multiplyScalar( distance ); - panOffset.add( v ); - - }; - - }(); - - // deltaX and deltaY are in pixels; right and down are positive - const pan = function () { - - const offset = new THREE.Vector3(); - return function pan( deltaX, deltaY ) { - - const element = scope.domElement; - if ( scope.object.isPerspectiveCamera ) { - - // perspective - const position = scope.object.position; - offset.copy( position ).sub( scope.target ); - let targetDistance = offset.length(); - - // half of the fov is center to top of screen - targetDistance *= Math.tan( scope.object.fov / 2 * Math.PI / 180.0 ); - - // we use only clientHeight here so aspect ratio does not distort speed - panLeft( 2 * deltaX * targetDistance / element.clientHeight, scope.object.matrix ); - panUp( 2 * deltaY * targetDistance / element.clientHeight, scope.object.matrix ); - - } else if ( scope.object.isOrthographicCamera ) { - - // orthographic - panLeft( deltaX * ( scope.object.right - scope.object.left ) / scope.object.zoom / element.clientWidth, scope.object.matrix ); - panUp( deltaY * ( scope.object.top - scope.object.bottom ) / scope.object.zoom / element.clientHeight, scope.object.matrix ); - - } else { - - // camera neither orthographic nor perspective - console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - pan disabled.' ); - scope.enablePan = false; - - } - - }; - - }(); - function dollyOut( dollyScale ) { - - if ( scope.object.isPerspectiveCamera ) { - - scale /= dollyScale; - - } else if ( scope.object.isOrthographicCamera ) { - - scope.object.zoom = Math.max( scope.minZoom, Math.min( scope.maxZoom, scope.object.zoom * dollyScale ) ); - scope.object.updateProjectionMatrix(); - zoomChanged = true; - - } else { - - console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled.' ); - scope.enableZoom = false; - - } - - } - - function dollyIn( dollyScale ) { - - if ( scope.object.isPerspectiveCamera ) { - - scale *= dollyScale; - - } else if ( scope.object.isOrthographicCamera ) { - - scope.object.zoom = Math.max( scope.minZoom, Math.min( scope.maxZoom, scope.object.zoom / dollyScale ) ); - scope.object.updateProjectionMatrix(); - zoomChanged = true; - - } else { - - console.warn( 'WARNING: OrbitControls.js encountered an unknown camera type - dolly/zoom disabled.' ); - scope.enableZoom = false; - - } - - } - - // - // event callbacks - update the object state - // - - function handleMouseDownRotate( event ) { - - rotateStart.set( event.clientX, event.clientY ); - - } - - function handleMouseDownDolly( event ) { - - dollyStart.set( event.clientX, event.clientY ); - - } - - function handleMouseDownPan( event ) { - - panStart.set( event.clientX, event.clientY ); - - } - - function handleMouseMoveRotate( event ) { - - rotateEnd.set( event.clientX, event.clientY ); - rotateDelta.subVectors( rotateEnd, rotateStart ).multiplyScalar( scope.rotateSpeed ); - const element = scope.domElement; - rotateLeft( 2 * Math.PI * rotateDelta.x / element.clientHeight ); // yes, height - - rotateUp( 2 * Math.PI * rotateDelta.y / element.clientHeight ); - rotateStart.copy( rotateEnd ); - scope.update(); - - } - - function handleMouseMoveDolly( event ) { - - dollyEnd.set( event.clientX, event.clientY ); - dollyDelta.subVectors( dollyEnd, dollyStart ); - if ( dollyDelta.y > 0 ) { - - dollyOut( getZoomScale() ); - - } else if ( dollyDelta.y < 0 ) { - - dollyIn( getZoomScale() ); - - } - - dollyStart.copy( dollyEnd ); - scope.update(); - - } - - function handleMouseMovePan( event ) { - - panEnd.set( event.clientX, event.clientY ); - panDelta.subVectors( panEnd, panStart ).multiplyScalar( scope.panSpeed ); - pan( panDelta.x, panDelta.y ); - panStart.copy( panEnd ); - scope.update(); - - } - - function handleMouseWheel( event ) { - - if ( event.deltaY < 0 ) { - - dollyIn( getZoomScale() ); - - } else if ( event.deltaY > 0 ) { - - dollyOut( getZoomScale() ); - - } - - scope.update(); - - } - - function handleKeyDown( event ) { - - let needsUpdate = false; - switch ( event.code ) { - - case scope.keys.UP: - pan( 0, scope.keyPanSpeed ); - needsUpdate = true; - break; - case scope.keys.BOTTOM: - pan( 0, - scope.keyPanSpeed ); - needsUpdate = true; - break; - case scope.keys.LEFT: - pan( scope.keyPanSpeed, 0 ); - needsUpdate = true; - break; - case scope.keys.RIGHT: - pan( - scope.keyPanSpeed, 0 ); - needsUpdate = true; - break; - - } - - if ( needsUpdate ) { - - // prevent the browser from scrolling on cursor keys - event.preventDefault(); - scope.update(); - - } - - } - - function handleTouchStartRotate() { - - if ( pointers.length === 1 ) { - - rotateStart.set( pointers[ 0 ].pageX, pointers[ 0 ].pageY ); - - } else { - - const x = 0.5 * ( pointers[ 0 ].pageX + pointers[ 1 ].pageX ); - const y = 0.5 * ( pointers[ 0 ].pageY + pointers[ 1 ].pageY ); - rotateStart.set( x, y ); - - } - - } - - function handleTouchStartPan() { - - if ( pointers.length === 1 ) { - - panStart.set( pointers[ 0 ].pageX, pointers[ 0 ].pageY ); - - } else { - - const x = 0.5 * ( pointers[ 0 ].pageX + pointers[ 1 ].pageX ); - const y = 0.5 * ( pointers[ 0 ].pageY + pointers[ 1 ].pageY ); - panStart.set( x, y ); - - } - - } - - function handleTouchStartDolly() { - - const dx = pointers[ 0 ].pageX - pointers[ 1 ].pageX; - const dy = pointers[ 0 ].pageY - pointers[ 1 ].pageY; - const distance = Math.sqrt( dx * dx + dy * dy ); - dollyStart.set( 0, distance ); - - } - - function handleTouchStartDollyPan() { - - if ( scope.enableZoom ) handleTouchStartDolly(); - if ( scope.enablePan ) handleTouchStartPan(); - - } - - function handleTouchStartDollyRotate() { - - if ( scope.enableZoom ) handleTouchStartDolly(); - if ( scope.enableRotate ) handleTouchStartRotate(); - - } - - function handleTouchMoveRotate( event ) { - - if ( pointers.length == 1 ) { - - rotateEnd.set( event.pageX, event.pageY ); - - } else { - - const position = getSecondPointerPosition( event ); - const x = 0.5 * ( event.pageX + position.x ); - const y = 0.5 * ( event.pageY + position.y ); - rotateEnd.set( x, y ); - - } - - rotateDelta.subVectors( rotateEnd, rotateStart ).multiplyScalar( scope.rotateSpeed ); - const element = scope.domElement; - rotateLeft( 2 * Math.PI * rotateDelta.x / element.clientHeight ); // yes, height - - rotateUp( 2 * Math.PI * rotateDelta.y / element.clientHeight ); - rotateStart.copy( rotateEnd ); - - } - - function handleTouchMovePan( event ) { - - if ( pointers.length === 1 ) { - - panEnd.set( event.pageX, event.pageY ); - - } else { - - const position = getSecondPointerPosition( event ); - const x = 0.5 * ( event.pageX + position.x ); - const y = 0.5 * ( event.pageY + position.y ); - panEnd.set( x, y ); - - } - - panDelta.subVectors( panEnd, panStart ).multiplyScalar( scope.panSpeed ); - pan( panDelta.x, panDelta.y ); - panStart.copy( panEnd ); - - } - - function handleTouchMoveDolly( event ) { - - const position = getSecondPointerPosition( event ); - const dx = event.pageX - position.x; - const dy = event.pageY - position.y; - const distance = Math.sqrt( dx * dx + dy * dy ); - dollyEnd.set( 0, distance ); - dollyDelta.set( 0, Math.pow( dollyEnd.y / dollyStart.y, scope.zoomSpeed ) ); - dollyOut( dollyDelta.y ); - dollyStart.copy( dollyEnd ); - - } - - function handleTouchMoveDollyPan( event ) { - - if ( scope.enableZoom ) handleTouchMoveDolly( event ); - if ( scope.enablePan ) handleTouchMovePan( event ); - - } - - function handleTouchMoveDollyRotate( event ) { - - if ( scope.enableZoom ) handleTouchMoveDolly( event ); - if ( scope.enableRotate ) handleTouchMoveRotate( event ); - - } - - // - // event handlers - FSM: listen for events and reset state - // - - function onPointerDown( event ) { - - if ( scope.enabled === false ) return; - if ( pointers.length === 0 ) { - - scope.domElement.setPointerCapture( event.pointerId ); - scope.domElement.addEventListener( 'pointermove', onPointerMove ); - scope.domElement.addEventListener( 'pointerup', onPointerUp ); - - } - - // - - addPointer( event ); - if ( event.pointerType === 'touch' ) { - - onTouchStart( event ); - - } else { - - onMouseDown( event ); - - } - - } - - function onPointerMove( event ) { - - if ( scope.enabled === false ) return; - if ( event.pointerType === 'touch' ) { - - onTouchMove( event ); - - } else { - - onMouseMove( event ); - - } - - } - - function onPointerUp( event ) { - - removePointer( event ); - if ( pointers.length === 0 ) { - - scope.domElement.releasePointerCapture( event.pointerId ); - scope.domElement.removeEventListener( 'pointermove', onPointerMove ); - scope.domElement.removeEventListener( 'pointerup', onPointerUp ); - - } - - scope.dispatchEvent( _endEvent ); - state = STATE.NONE; - - } - - function onPointerCancel( event ) { - - removePointer( event ); - - } - - function onMouseDown( event ) { - - let mouseAction; - switch ( event.button ) { - - case 0: - mouseAction = scope.mouseButtons.LEFT; - break; - case 1: - mouseAction = scope.mouseButtons.MIDDLE; - break; - case 2: - mouseAction = scope.mouseButtons.RIGHT; - break; - default: - mouseAction = - 1; - - } - - switch ( mouseAction ) { - - case THREE.MOUSE.DOLLY: - if ( scope.enableZoom === false ) return; - handleMouseDownDolly( event ); - state = STATE.DOLLY; - break; - case THREE.MOUSE.ROTATE: - if ( event.ctrlKey || event.metaKey || event.shiftKey ) { - - if ( scope.enablePan === false ) return; - handleMouseDownPan( event ); - state = STATE.PAN; - - } else { - - if ( scope.enableRotate === false ) return; - handleMouseDownRotate( event ); - state = STATE.ROTATE; - - } - - break; - case THREE.MOUSE.PAN: - if ( event.ctrlKey || event.metaKey || event.shiftKey ) { - - if ( scope.enableRotate === false ) return; - handleMouseDownRotate( event ); - state = STATE.ROTATE; - - } else { - - if ( scope.enablePan === false ) return; - handleMouseDownPan( event ); - state = STATE.PAN; - - } - - break; - default: - state = STATE.NONE; - - } - - if ( state !== STATE.NONE ) { - - scope.dispatchEvent( _startEvent ); - - } - - } - - function onMouseMove( event ) { - - switch ( state ) { - - case STATE.ROTATE: - if ( scope.enableRotate === false ) return; - handleMouseMoveRotate( event ); - break; - case STATE.DOLLY: - if ( scope.enableZoom === false ) return; - handleMouseMoveDolly( event ); - break; - case STATE.PAN: - if ( scope.enablePan === false ) return; - handleMouseMovePan( event ); - break; - - } - - } - - function onMouseWheel( event ) { - - if ( scope.enabled === false || scope.enableZoom === false || state !== STATE.NONE ) return; - event.preventDefault(); - scope.dispatchEvent( _startEvent ); - handleMouseWheel( event ); - scope.dispatchEvent( _endEvent ); - - } - - function onKeyDown( event ) { - - if ( scope.enabled === false || scope.enablePan === false ) return; - handleKeyDown( event ); - - } - - function onTouchStart( event ) { - - trackPointer( event ); - switch ( pointers.length ) { - - case 1: - switch ( scope.touches.ONE ) { - - case THREE.TOUCH.ROTATE: - if ( scope.enableRotate === false ) return; - handleTouchStartRotate(); - state = STATE.TOUCH_ROTATE; - break; - case THREE.TOUCH.PAN: - if ( scope.enablePan === false ) return; - handleTouchStartPan(); - state = STATE.TOUCH_PAN; - break; - default: - state = STATE.NONE; - - } - - break; - case 2: - switch ( scope.touches.TWO ) { - - case THREE.TOUCH.DOLLY_PAN: - if ( scope.enableZoom === false && scope.enablePan === false ) return; - handleTouchStartDollyPan(); - state = STATE.TOUCH_DOLLY_PAN; - break; - case THREE.TOUCH.DOLLY_ROTATE: - if ( scope.enableZoom === false && scope.enableRotate === false ) return; - handleTouchStartDollyRotate(); - state = STATE.TOUCH_DOLLY_ROTATE; - break; - default: - state = STATE.NONE; - - } - - break; - default: - state = STATE.NONE; - - } - - if ( state !== STATE.NONE ) { - - scope.dispatchEvent( _startEvent ); - - } - - } - - function onTouchMove( event ) { - - trackPointer( event ); - switch ( state ) { - - case STATE.TOUCH_ROTATE: - if ( scope.enableRotate === false ) return; - handleTouchMoveRotate( event ); - scope.update(); - break; - case STATE.TOUCH_PAN: - if ( scope.enablePan === false ) return; - handleTouchMovePan( event ); - scope.update(); - break; - case STATE.TOUCH_DOLLY_PAN: - if ( scope.enableZoom === false && scope.enablePan === false ) return; - handleTouchMoveDollyPan( event ); - scope.update(); - break; - case STATE.TOUCH_DOLLY_ROTATE: - if ( scope.enableZoom === false && scope.enableRotate === false ) return; - handleTouchMoveDollyRotate( event ); - scope.update(); - break; - default: - state = STATE.NONE; - - } - - } - - function onContextMenu( event ) { - - if ( scope.enabled === false ) return; - event.preventDefault(); - - } - - function addPointer( event ) { - - pointers.push( event ); - - } - - function removePointer( event ) { - - delete pointerPositions[ event.pointerId ]; - for ( let i = 0; i < pointers.length; i ++ ) { - - if ( pointers[ i ].pointerId == event.pointerId ) { - - pointers.splice( i, 1 ); - return; - - } - - } - - } - - function trackPointer( event ) { - - let position = pointerPositions[ event.pointerId ]; - if ( position === undefined ) { - - position = new THREE.Vector2(); - pointerPositions[ event.pointerId ] = position; - - } - - position.set( event.pageX, event.pageY ); - - } - - function getSecondPointerPosition( event ) { - - const pointer = event.pointerId === pointers[ 0 ].pointerId ? pointers[ 1 ] : pointers[ 0 ]; - return pointerPositions[ pointer.pointerId ]; - - } - - // - - scope.domElement.addEventListener( 'contextmenu', onContextMenu ); - scope.domElement.addEventListener( 'pointerdown', onPointerDown ); - scope.domElement.addEventListener( 'pointercancel', onPointerCancel ); - scope.domElement.addEventListener( 'wheel', onMouseWheel, { - passive: false - } ); - - // force an update at start - - this.update(); - - } - - } - - // This set of controls performs orbiting, dollying (zooming), and panning. - // Unlike TrackballControls, it maintains the "up" direction object.up (+Y by default). - // This is very similar to OrbitControls, another set of touch behavior - // - // Orbit - right mouse, or left mouse + ctrl/meta/shiftKey / touch: two-finger rotate - // Zoom - middle mouse, or mousewheel / touch: two-finger spread or squish - // Pan - left mouse, or arrow keys / touch: one-finger move - - class MapControls extends OrbitControls { - - constructor( object, domElement ) { - - super( object, domElement ); - this.screenSpacePanning = false; // pan orthogonal to world-space direction camera.up - - this.mouseButtons.LEFT = THREE.MOUSE.PAN; - this.mouseButtons.RIGHT = THREE.MOUSE.ROTATE; - this.touches.ONE = THREE.TOUCH.PAN; - this.touches.TWO = THREE.TOUCH.DOLLY_ROTATE; - - } - - } - - THREE.MapControls = MapControls; - THREE.OrbitControls = OrbitControls; - -} )(); diff --git a/public/static/common/js/three/stats.module.js b/public/static/common/js/three/stats.module.js deleted file mode 100644 index 0d372ba..0000000 --- a/public/static/common/js/three/stats.module.js +++ /dev/null @@ -1,167 +0,0 @@ -var Stats = function () { - - var mode = 0; - - var container = document.createElement( 'div' ); - container.style.cssText = 'position:fixed;top:0;left:0;cursor:pointer;opacity:0.9;z-index:10000'; - container.addEventListener( 'click', function ( event ) { - - event.preventDefault(); - showPanel( ++ mode % container.children.length ); - - }, false ); - - // - - function addPanel( panel ) { - - container.appendChild( panel.dom ); - return panel; - - } - - function showPanel( id ) { - - for ( var i = 0; i < container.children.length; i ++ ) { - - container.children[ i ].style.display = i === id ? 'block' : 'none'; - - } - - mode = id; - - } - - // - - var beginTime = ( performance || Date ).now(), prevTime = beginTime, frames = 0; - - var fpsPanel = addPanel( new Stats.Panel( 'FPS', '#0ff', '#002' ) ); - var msPanel = addPanel( new Stats.Panel( 'MS', '#0f0', '#020' ) ); - - if ( self.performance && self.performance.memory ) { - - var memPanel = addPanel( new Stats.Panel( 'MB', '#f08', '#201' ) ); - - } - - showPanel( 0 ); - - return { - - REVISION: 16, - - dom: container, - - addPanel: addPanel, - showPanel: showPanel, - - begin: function () { - - beginTime = ( performance || Date ).now(); - - }, - - end: function () { - - frames ++; - - var time = ( performance || Date ).now(); - - msPanel.update( time - beginTime, 200 ); - - if ( time >= prevTime + 1000 ) { - - fpsPanel.update( ( frames * 1000 ) / ( time - prevTime ), 100 ); - - prevTime = time; - frames = 0; - - if ( memPanel ) { - - var memory = performance.memory; - memPanel.update( memory.usedJSHeapSize / 1048576, memory.jsHeapSizeLimit / 1048576 ); - - } - - } - - return time; - - }, - - update: function () { - - beginTime = this.end(); - - }, - - // Backwards Compatibility - - domElement: container, - setMode: showPanel - - }; - -}; - -Stats.Panel = function ( name, fg, bg ) { - - var min = Infinity, max = 0, round = Math.round; - var PR = round( window.devicePixelRatio || 1 ); - - var WIDTH = 80 * PR, HEIGHT = 48 * PR, - TEXT_X = 3 * PR, TEXT_Y = 2 * PR, - GRAPH_X = 3 * PR, GRAPH_Y = 15 * PR, - GRAPH_WIDTH = 74 * PR, GRAPH_HEIGHT = 30 * PR; - - var canvas = document.createElement( 'canvas' ); - canvas.width = WIDTH; - canvas.height = HEIGHT; - canvas.style.cssText = 'width:80px;height:48px'; - - var context = canvas.getContext( '2d' ); - context.font = 'bold ' + ( 9 * PR ) + 'px Helvetica,Arial,sans-serif'; - context.textBaseline = 'top'; - - context.fillStyle = bg; - context.fillRect( 0, 0, WIDTH, HEIGHT ); - - context.fillStyle = fg; - context.fillText( name, TEXT_X, TEXT_Y ); - context.fillRect( GRAPH_X, GRAPH_Y, GRAPH_WIDTH, GRAPH_HEIGHT ); - - context.fillStyle = bg; - context.globalAlpha = 0.9; - context.fillRect( GRAPH_X, GRAPH_Y, GRAPH_WIDTH, GRAPH_HEIGHT ); - - return { - - dom: canvas, - - update: function ( value, maxValue ) { - - min = Math.min( min, value ); - max = Math.max( max, value ); - - context.fillStyle = bg; - context.globalAlpha = 1; - context.fillRect( 0, 0, WIDTH, GRAPH_Y ); - context.fillStyle = fg; - context.fillText( round( value ) + ' ' + name + ' (' + round( min ) + '-' + round( max ) + ')', TEXT_X, TEXT_Y ); - - context.drawImage( canvas, GRAPH_X + PR, GRAPH_Y, GRAPH_WIDTH - PR, GRAPH_HEIGHT, GRAPH_X, GRAPH_Y, GRAPH_WIDTH - PR, GRAPH_HEIGHT ); - - context.fillRect( GRAPH_X + GRAPH_WIDTH - PR, GRAPH_Y, PR, GRAPH_HEIGHT ); - - context.fillStyle = bg; - context.globalAlpha = 0.9; - context.fillRect( GRAPH_X + GRAPH_WIDTH - PR, GRAPH_Y, PR, round( ( 1 - ( value / maxValue ) ) * GRAPH_HEIGHT ) ); - - } - - }; - -}; - -export default Stats; diff --git a/public/static/common/js/three/three.min.js b/public/static/common/js/three/three.min.js deleted file mode 100644 index dc6e61f..0000000 --- a/public/static/common/js/three/three.min.js +++ /dev/null @@ -1,6 +0,0 @@ -/** - * @license - * Copyright 2010-2022 Three.js Authors - * SPDX-License-Identifier: MIT - */ -!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).THREE={})}(this,(function(t){"use strict";const e="146",i=100,n=300,r=301,s=302,a=303,o=304,l=306,c=1e3,h=1001,u=1002,d=1003,p=1004,m=1005,f=1006,g=1007,v=1008,x=1009,_=1012,y=1014,M=1015,b=1016,S=1020,w=1023,T=1026,A=1027,E=33776,C=33777,L=33778,R=33779,P=35840,D=35841,I=35842,N=35843,O=37492,z=37496,U=37808,B=37809,F=37810,k=37811,G=37812,V=37813,H=37814,W=37815,j=37816,q=37817,X=37818,Y=37819,Z=37820,J=37821,K=36492,$=2300,Q=2301,tt=2302,et=2400,it=2401,nt=2402,rt=2500,st=2501,at=3e3,ot=3001,lt="srgb",ct="srgb-linear",ht=7680,ut=35044,dt="300 es",pt=1035;class mt{addEventListener(t,e){void 0===this._listeners&&(this._listeners={});const i=this._listeners;void 0===i[t]&&(i[t]=[]),-1===i[t].indexOf(e)&&i[t].push(e)}hasEventListener(t,e){if(void 0===this._listeners)return!1;const i=this._listeners;return void 0!==i[t]&&-1!==i[t].indexOf(e)}removeEventListener(t,e){if(void 0===this._listeners)return;const i=this._listeners[t];if(void 0!==i){const t=i.indexOf(e);-1!==t&&i.splice(t,1)}}dispatchEvent(t){if(void 0===this._listeners)return;const e=this._listeners[t.type];if(void 0!==e){t.target=this;const i=e.slice(0);for(let e=0,n=i.length;e<n;e++)i[e].call(this,t);t.target=null}}}const ft=["00","01","02","03","04","05","06","07","08","09","0a","0b","0c","0d","0e","0f","10","11","12","13","14","15","16","17","18","19","1a","1b","1c","1d","1e","1f","20","21","22","23","24","25","26","27","28","29","2a","2b","2c","2d","2e","2f","30","31","32","33","34","35","36","37","38","39","3a","3b","3c","3d","3e","3f","40","41","42","43","44","45","46","47","48","49","4a","4b","4c","4d","4e","4f","50","51","52","53","54","55","56","57","58","59","5a","5b","5c","5d","5e","5f","60","61","62","63","64","65","66","67","68","69","6a","6b","6c","6d","6e","6f","70","71","72","73","74","75","76","77","78","79","7a","7b","7c","7d","7e","7f","80","81","82","83","84","85","86","87","88","89","8a","8b","8c","8d","8e","8f","90","91","92","93","94","95","96","97","98","99","9a","9b","9c","9d","9e","9f","a0","a1","a2","a3","a4","a5","a6","a7","a8","a9","aa","ab","ac","ad","ae","af","b0","b1","b2","b3","b4","b5","b6","b7","b8","b9","ba","bb","bc","bd","be","bf","c0","c1","c2","c3","c4","c5","c6","c7","c8","c9","ca","cb","cc","cd","ce","cf","d0","d1","d2","d3","d4","d5","d6","d7","d8","d9","da","db","dc","dd","de","df","e0","e1","e2","e3","e4","e5","e6","e7","e8","e9","ea","eb","ec","ed","ee","ef","f0","f1","f2","f3","f4","f5","f6","f7","f8","f9","fa","fb","fc","fd","fe","ff"];let gt=1234567;const vt=Math.PI/180,xt=180/Math.PI;function _t(){const t=4294967295*Math.random()|0,e=4294967295*Math.random()|0,i=4294967295*Math.random()|0,n=4294967295*Math.random()|0;return(ft[255&t]+ft[t>>8&255]+ft[t>>16&255]+ft[t>>24&255]+"-"+ft[255&e]+ft[e>>8&255]+"-"+ft[e>>16&15|64]+ft[e>>24&255]+"-"+ft[63&i|128]+ft[i>>8&255]+"-"+ft[i>>16&255]+ft[i>>24&255]+ft[255&n]+ft[n>>8&255]+ft[n>>16&255]+ft[n>>24&255]).toLowerCase()}function yt(t,e,i){return Math.max(e,Math.min(i,t))}function Mt(t,e){return(t%e+e)%e}function bt(t,e,i){return(1-i)*t+i*e}function St(t){return 0==(t&t-1)&&0!==t}function wt(t){return Math.pow(2,Math.ceil(Math.log(t)/Math.LN2))}function Tt(t){return Math.pow(2,Math.floor(Math.log(t)/Math.LN2))}function At(t,e){switch(e.constructor){case Float32Array:return t;case Uint16Array:return t/65535;case Uint8Array:return t/255;case Int16Array:return Math.max(t/32767,-1);case Int8Array:return Math.max(t/127,-1);default:throw new Error("Invalid component type.")}}function Et(t,e){switch(e.constructor){case Float32Array:return t;case Uint16Array:return Math.round(65535*t);case Uint8Array:return Math.round(255*t);case Int16Array:return Math.round(32767*t);case Int8Array:return Math.round(127*t);default:throw new Error("Invalid component type.")}}var Ct=Object.freeze({__proto__:null,DEG2RAD:vt,RAD2DEG:xt,generateUUID:_t,clamp:yt,euclideanModulo:Mt,mapLinear:function(t,e,i,n,r){return n+(t-e)*(r-n)/(i-e)},inverseLerp:function(t,e,i){return t!==e?(i-t)/(e-t):0},lerp:bt,damp:function(t,e,i,n){return bt(t,e,1-Math.exp(-i*n))},pingpong:function(t,e=1){return e-Math.abs(Mt(t,2*e)-e)},smoothstep:function(t,e,i){return t<=e?0:t>=i?1:(t=(t-e)/(i-e))*t*(3-2*t)},smootherstep:function(t,e,i){return t<=e?0:t>=i?1:(t=(t-e)/(i-e))*t*t*(t*(6*t-15)+10)},randInt:function(t,e){return t+Math.floor(Math.random()*(e-t+1))},randFloat:function(t,e){return t+Math.random()*(e-t)},randFloatSpread:function(t){return t*(.5-Math.random())},seededRandom:function(t){void 0!==t&&(gt=t);let e=gt+=1831565813;return e=Math.imul(e^e>>>15,1|e),e^=e+Math.imul(e^e>>>7,61|e),((e^e>>>14)>>>0)/4294967296},degToRad:function(t){return t*vt},radToDeg:function(t){return t*xt},isPowerOfTwo:St,ceilPowerOfTwo:wt,floorPowerOfTwo:Tt,setQuaternionFromProperEuler:function(t,e,i,n,r){const s=Math.cos,a=Math.sin,o=s(i/2),l=a(i/2),c=s((e+n)/2),h=a((e+n)/2),u=s((e-n)/2),d=a((e-n)/2),p=s((n-e)/2),m=a((n-e)/2);switch(r){case"XYX":t.set(o*h,l*u,l*d,o*c);break;case"YZY":t.set(l*d,o*h,l*u,o*c);break;case"ZXZ":t.set(l*u,l*d,o*h,o*c);break;case"XZX":t.set(o*h,l*m,l*p,o*c);break;case"YXY":t.set(l*p,o*h,l*m,o*c);break;case"ZYZ":t.set(l*m,l*p,o*h,o*c);break;default:console.warn("THREE.MathUtils: .setQuaternionFromProperEuler() encountered an unknown order: "+r)}},normalize:Et,denormalize:At});class Lt{constructor(t=0,e=0){Lt.prototype.isVector2=!0,this.x=t,this.y=e}get width(){return this.x}set width(t){this.x=t}get height(){return this.y}set height(t){this.y=t}set(t,e){return this.x=t,this.y=e,this}setScalar(t){return this.x=t,this.y=t,this}setX(t){return this.x=t,this}setY(t){return this.y=t,this}setComponent(t,e){switch(t){case 0:this.x=e;break;case 1:this.y=e;break;default:throw new Error("index is out of range: "+t)}return this}getComponent(t){switch(t){case 0:return this.x;case 1:return this.y;default:throw new Error("index is out of range: "+t)}}clone(){return new this.constructor(this.x,this.y)}copy(t){return this.x=t.x,this.y=t.y,this}add(t){return this.x+=t.x,this.y+=t.y,this}addScalar(t){return this.x+=t,this.y+=t,this}addVectors(t,e){return this.x=t.x+e.x,this.y=t.y+e.y,this}addScaledVector(t,e){return this.x+=t.x*e,this.y+=t.y*e,this}sub(t){return this.x-=t.x,this.y-=t.y,this}subScalar(t){return this.x-=t,this.y-=t,this}subVectors(t,e){return this.x=t.x-e.x,this.y=t.y-e.y,this}multiply(t){return this.x*=t.x,this.y*=t.y,this}multiplyScalar(t){return this.x*=t,this.y*=t,this}divide(t){return this.x/=t.x,this.y/=t.y,this}divideScalar(t){return this.multiplyScalar(1/t)}applyMatrix3(t){const e=this.x,i=this.y,n=t.elements;return this.x=n[0]*e+n[3]*i+n[6],this.y=n[1]*e+n[4]*i+n[7],this}min(t){return this.x=Math.min(this.x,t.x),this.y=Math.min(this.y,t.y),this}max(t){return this.x=Math.max(this.x,t.x),this.y=Math.max(this.y,t.y),this}clamp(t,e){return this.x=Math.max(t.x,Math.min(e.x,this.x)),this.y=Math.max(t.y,Math.min(e.y,this.y)),this}clampScalar(t,e){return this.x=Math.max(t,Math.min(e,this.x)),this.y=Math.max(t,Math.min(e,this.y)),this}clampLength(t,e){const i=this.length();return this.divideScalar(i||1).multiplyScalar(Math.max(t,Math.min(e,i)))}floor(){return this.x=Math.floor(this.x),this.y=Math.floor(this.y),this}ceil(){return this.x=Math.ceil(this.x),this.y=Math.ceil(this.y),this}round(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this}roundToZero(){return this.x=this.x<0?Math.ceil(this.x):Math.floor(this.x),this.y=this.y<0?Math.ceil(this.y):Math.floor(this.y),this}negate(){return this.x=-this.x,this.y=-this.y,this}dot(t){return this.x*t.x+this.y*t.y}cross(t){return this.x*t.y-this.y*t.x}lengthSq(){return this.x*this.x+this.y*this.y}length(){return Math.sqrt(this.x*this.x+this.y*this.y)}manhattanLength(){return Math.abs(this.x)+Math.abs(this.y)}normalize(){return this.divideScalar(this.length()||1)}angle(){return Math.atan2(-this.y,-this.x)+Math.PI}distanceTo(t){return Math.sqrt(this.distanceToSquared(t))}distanceToSquared(t){const e=this.x-t.x,i=this.y-t.y;return e*e+i*i}manhattanDistanceTo(t){return Math.abs(this.x-t.x)+Math.abs(this.y-t.y)}setLength(t){return this.normalize().multiplyScalar(t)}lerp(t,e){return this.x+=(t.x-this.x)*e,this.y+=(t.y-this.y)*e,this}lerpVectors(t,e,i){return this.x=t.x+(e.x-t.x)*i,this.y=t.y+(e.y-t.y)*i,this}equals(t){return t.x===this.x&&t.y===this.y}fromArray(t,e=0){return this.x=t[e],this.y=t[e+1],this}toArray(t=[],e=0){return t[e]=this.x,t[e+1]=this.y,t}fromBufferAttribute(t,e){return this.x=t.getX(e),this.y=t.getY(e),this}rotateAround(t,e){const i=Math.cos(e),n=Math.sin(e),r=this.x-t.x,s=this.y-t.y;return this.x=r*i-s*n+t.x,this.y=r*n+s*i+t.y,this}random(){return this.x=Math.random(),this.y=Math.random(),this}*[Symbol.iterator](){yield this.x,yield this.y}}class Rt{constructor(){Rt.prototype.isMatrix3=!0,this.elements=[1,0,0,0,1,0,0,0,1]}set(t,e,i,n,r,s,a,o,l){const c=this.elements;return c[0]=t,c[1]=n,c[2]=a,c[3]=e,c[4]=r,c[5]=o,c[6]=i,c[7]=s,c[8]=l,this}identity(){return this.set(1,0,0,0,1,0,0,0,1),this}copy(t){const e=this.elements,i=t.elements;return e[0]=i[0],e[1]=i[1],e[2]=i[2],e[3]=i[3],e[4]=i[4],e[5]=i[5],e[6]=i[6],e[7]=i[7],e[8]=i[8],this}extractBasis(t,e,i){return t.setFromMatrix3Column(this,0),e.setFromMatrix3Column(this,1),i.setFromMatrix3Column(this,2),this}setFromMatrix4(t){const e=t.elements;return this.set(e[0],e[4],e[8],e[1],e[5],e[9],e[2],e[6],e[10]),this}multiply(t){return this.multiplyMatrices(this,t)}premultiply(t){return this.multiplyMatrices(t,this)}multiplyMatrices(t,e){const i=t.elements,n=e.elements,r=this.elements,s=i[0],a=i[3],o=i[6],l=i[1],c=i[4],h=i[7],u=i[2],d=i[5],p=i[8],m=n[0],f=n[3],g=n[6],v=n[1],x=n[4],_=n[7],y=n[2],M=n[5],b=n[8];return r[0]=s*m+a*v+o*y,r[3]=s*f+a*x+o*M,r[6]=s*g+a*_+o*b,r[1]=l*m+c*v+h*y,r[4]=l*f+c*x+h*M,r[7]=l*g+c*_+h*b,r[2]=u*m+d*v+p*y,r[5]=u*f+d*x+p*M,r[8]=u*g+d*_+p*b,this}multiplyScalar(t){const e=this.elements;return e[0]*=t,e[3]*=t,e[6]*=t,e[1]*=t,e[4]*=t,e[7]*=t,e[2]*=t,e[5]*=t,e[8]*=t,this}determinant(){const t=this.elements,e=t[0],i=t[1],n=t[2],r=t[3],s=t[4],a=t[5],o=t[6],l=t[7],c=t[8];return e*s*c-e*a*l-i*r*c+i*a*o+n*r*l-n*s*o}invert(){const t=this.elements,e=t[0],i=t[1],n=t[2],r=t[3],s=t[4],a=t[5],o=t[6],l=t[7],c=t[8],h=c*s-a*l,u=a*o-c*r,d=l*r-s*o,p=e*h+i*u+n*d;if(0===p)return this.set(0,0,0,0,0,0,0,0,0);const m=1/p;return t[0]=h*m,t[1]=(n*l-c*i)*m,t[2]=(a*i-n*s)*m,t[3]=u*m,t[4]=(c*e-n*o)*m,t[5]=(n*r-a*e)*m,t[6]=d*m,t[7]=(i*o-l*e)*m,t[8]=(s*e-i*r)*m,this}transpose(){let t;const e=this.elements;return t=e[1],e[1]=e[3],e[3]=t,t=e[2],e[2]=e[6],e[6]=t,t=e[5],e[5]=e[7],e[7]=t,this}getNormalMatrix(t){return this.setFromMatrix4(t).invert().transpose()}transposeIntoArray(t){const e=this.elements;return t[0]=e[0],t[1]=e[3],t[2]=e[6],t[3]=e[1],t[4]=e[4],t[5]=e[7],t[6]=e[2],t[7]=e[5],t[8]=e[8],this}setUvTransform(t,e,i,n,r,s,a){const o=Math.cos(r),l=Math.sin(r);return this.set(i*o,i*l,-i*(o*s+l*a)+s+t,-n*l,n*o,-n*(-l*s+o*a)+a+e,0,0,1),this}scale(t,e){const i=this.elements;return i[0]*=t,i[3]*=t,i[6]*=t,i[1]*=e,i[4]*=e,i[7]*=e,this}rotate(t){const e=Math.cos(t),i=Math.sin(t),n=this.elements,r=n[0],s=n[3],a=n[6],o=n[1],l=n[4],c=n[7];return n[0]=e*r+i*o,n[3]=e*s+i*l,n[6]=e*a+i*c,n[1]=-i*r+e*o,n[4]=-i*s+e*l,n[7]=-i*a+e*c,this}translate(t,e){const i=this.elements;return i[0]+=t*i[2],i[3]+=t*i[5],i[6]+=t*i[8],i[1]+=e*i[2],i[4]+=e*i[5],i[7]+=e*i[8],this}equals(t){const e=this.elements,i=t.elements;for(let t=0;t<9;t++)if(e[t]!==i[t])return!1;return!0}fromArray(t,e=0){for(let i=0;i<9;i++)this.elements[i]=t[i+e];return this}toArray(t=[],e=0){const i=this.elements;return t[e]=i[0],t[e+1]=i[1],t[e+2]=i[2],t[e+3]=i[3],t[e+4]=i[4],t[e+5]=i[5],t[e+6]=i[6],t[e+7]=i[7],t[e+8]=i[8],t}clone(){return(new this.constructor).fromArray(this.elements)}}function Pt(t){for(let e=t.length-1;e>=0;--e)if(t[e]>=65535)return!0;return!1}const Dt={Int8Array:Int8Array,Uint8Array:Uint8Array,Uint8ClampedArray:Uint8ClampedArray,Int16Array:Int16Array,Uint16Array:Uint16Array,Int32Array:Int32Array,Uint32Array:Uint32Array,Float32Array:Float32Array,Float64Array:Float64Array};function It(t,e){return new Dt[t](e)}function Nt(t){return document.createElementNS("http://www.w3.org/1999/xhtml",t)}function Ot(t){return t<.04045?.0773993808*t:Math.pow(.9478672986*t+.0521327014,2.4)}function zt(t){return t<.0031308?12.92*t:1.055*Math.pow(t,.41666)-.055}const Ut={[lt]:{[ct]:Ot},[ct]:{[lt]:zt}},Bt={legacyMode:!0,get workingColorSpace(){return ct},set workingColorSpace(t){console.warn("THREE.ColorManagement: .workingColorSpace is readonly.")},convert:function(t,e,i){if(this.legacyMode||e===i||!e||!i)return t;if(Ut[e]&&void 0!==Ut[e][i]){const n=Ut[e][i];return t.r=n(t.r),t.g=n(t.g),t.b=n(t.b),t}throw new Error("Unsupported color space conversion.")},fromWorkingColorSpace:function(t,e){return this.convert(t,this.workingColorSpace,e)},toWorkingColorSpace:function(t,e){return this.convert(t,e,this.workingColorSpace)}},Ft={aliceblue:15792383,antiquewhite:16444375,aqua:65535,aquamarine:8388564,azure:15794175,beige:16119260,bisque:16770244,black:0,blanchedalmond:16772045,blue:255,blueviolet:9055202,brown:10824234,burlywood:14596231,cadetblue:6266528,chartreuse:8388352,chocolate:13789470,coral:16744272,cornflowerblue:6591981,cornsilk:16775388,crimson:14423100,cyan:65535,darkblue:139,darkcyan:35723,darkgoldenrod:12092939,darkgray:11119017,darkgreen:25600,darkgrey:11119017,darkkhaki:12433259,darkmagenta:9109643,darkolivegreen:5597999,darkorange:16747520,darkorchid:10040012,darkred:9109504,darksalmon:15308410,darkseagreen:9419919,darkslateblue:4734347,darkslategray:3100495,darkslategrey:3100495,darkturquoise:52945,darkviolet:9699539,deeppink:16716947,deepskyblue:49151,dimgray:6908265,dimgrey:6908265,dodgerblue:2003199,firebrick:11674146,floralwhite:16775920,forestgreen:2263842,fuchsia:16711935,gainsboro:14474460,ghostwhite:16316671,gold:16766720,goldenrod:14329120,gray:8421504,green:32768,greenyellow:11403055,grey:8421504,honeydew:15794160,hotpink:16738740,indianred:13458524,indigo:4915330,ivory:16777200,khaki:15787660,lavender:15132410,lavenderblush:16773365,lawngreen:8190976,lemonchiffon:16775885,lightblue:11393254,lightcoral:15761536,lightcyan:14745599,lightgoldenrodyellow:16448210,lightgray:13882323,lightgreen:9498256,lightgrey:13882323,lightpink:16758465,lightsalmon:16752762,lightseagreen:2142890,lightskyblue:8900346,lightslategray:7833753,lightslategrey:7833753,lightsteelblue:11584734,lightyellow:16777184,lime:65280,limegreen:3329330,linen:16445670,magenta:16711935,maroon:8388608,mediumaquamarine:6737322,mediumblue:205,mediumorchid:12211667,mediumpurple:9662683,mediumseagreen:3978097,mediumslateblue:8087790,mediumspringgreen:64154,mediumturquoise:4772300,mediumvioletred:13047173,midnightblue:1644912,mintcream:16121850,mistyrose:16770273,moccasin:16770229,navajowhite:16768685,navy:128,oldlace:16643558,olive:8421376,olivedrab:7048739,orange:16753920,orangered:16729344,orchid:14315734,palegoldenrod:15657130,palegreen:10025880,paleturquoise:11529966,palevioletred:14381203,papayawhip:16773077,peachpuff:16767673,peru:13468991,pink:16761035,plum:14524637,powderblue:11591910,purple:8388736,rebeccapurple:6697881,red:16711680,rosybrown:12357519,royalblue:4286945,saddlebrown:9127187,salmon:16416882,sandybrown:16032864,seagreen:3050327,seashell:16774638,sienna:10506797,silver:12632256,skyblue:8900331,slateblue:6970061,slategray:7372944,slategrey:7372944,snow:16775930,springgreen:65407,steelblue:4620980,tan:13808780,teal:32896,thistle:14204888,tomato:16737095,turquoise:4251856,violet:15631086,wheat:16113331,white:16777215,whitesmoke:16119285,yellow:16776960,yellowgreen:10145074},kt={r:0,g:0,b:0},Gt={h:0,s:0,l:0},Vt={h:0,s:0,l:0};function Ht(t,e,i){return i<0&&(i+=1),i>1&&(i-=1),i<1/6?t+6*(e-t)*i:i<.5?e:i<2/3?t+6*(e-t)*(2/3-i):t}function Wt(t,e){return e.r=t.r,e.g=t.g,e.b=t.b,e}class jt{constructor(t,e,i){return this.isColor=!0,this.r=1,this.g=1,this.b=1,void 0===e&&void 0===i?this.set(t):this.setRGB(t,e,i)}set(t){return t&&t.isColor?this.copy(t):"number"==typeof t?this.setHex(t):"string"==typeof t&&this.setStyle(t),this}setScalar(t){return this.r=t,this.g=t,this.b=t,this}setHex(t,e="srgb"){return t=Math.floor(t),this.r=(t>>16&255)/255,this.g=(t>>8&255)/255,this.b=(255&t)/255,Bt.toWorkingColorSpace(this,e),this}setRGB(t,e,i,n="srgb-linear"){return this.r=t,this.g=e,this.b=i,Bt.toWorkingColorSpace(this,n),this}setHSL(t,e,i,n="srgb-linear"){if(t=Mt(t,1),e=yt(e,0,1),i=yt(i,0,1),0===e)this.r=this.g=this.b=i;else{const n=i<=.5?i*(1+e):i+e-i*e,r=2*i-n;this.r=Ht(r,n,t+1/3),this.g=Ht(r,n,t),this.b=Ht(r,n,t-1/3)}return Bt.toWorkingColorSpace(this,n),this}setStyle(t,e="srgb"){function i(e){void 0!==e&&parseFloat(e)<1&&console.warn("THREE.Color: Alpha component of "+t+" will be ignored.")}let n;if(n=/^((?:rgb|hsl)a?)\(([^\)]*)\)/.exec(t)){let t;const r=n[1],s=n[2];switch(r){case"rgb":case"rgba":if(t=/^\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*(?:,\s*(\d*\.?\d+)\s*)?$/.exec(s))return this.r=Math.min(255,parseInt(t[1],10))/255,this.g=Math.min(255,parseInt(t[2],10))/255,this.b=Math.min(255,parseInt(t[3],10))/255,Bt.toWorkingColorSpace(this,e),i(t[4]),this;if(t=/^\s*(\d+)\%\s*,\s*(\d+)\%\s*,\s*(\d+)\%\s*(?:,\s*(\d*\.?\d+)\s*)?$/.exec(s))return this.r=Math.min(100,parseInt(t[1],10))/100,this.g=Math.min(100,parseInt(t[2],10))/100,this.b=Math.min(100,parseInt(t[3],10))/100,Bt.toWorkingColorSpace(this,e),i(t[4]),this;break;case"hsl":case"hsla":if(t=/^\s*(\d*\.?\d+)\s*,\s*(\d*\.?\d+)\%\s*,\s*(\d*\.?\d+)\%\s*(?:,\s*(\d*\.?\d+)\s*)?$/.exec(s)){const n=parseFloat(t[1])/360,r=parseFloat(t[2])/100,s=parseFloat(t[3])/100;return i(t[4]),this.setHSL(n,r,s,e)}}}else if(n=/^\#([A-Fa-f\d]+)$/.exec(t)){const t=n[1],i=t.length;if(3===i)return this.r=parseInt(t.charAt(0)+t.charAt(0),16)/255,this.g=parseInt(t.charAt(1)+t.charAt(1),16)/255,this.b=parseInt(t.charAt(2)+t.charAt(2),16)/255,Bt.toWorkingColorSpace(this,e),this;if(6===i)return this.r=parseInt(t.charAt(0)+t.charAt(1),16)/255,this.g=parseInt(t.charAt(2)+t.charAt(3),16)/255,this.b=parseInt(t.charAt(4)+t.charAt(5),16)/255,Bt.toWorkingColorSpace(this,e),this}return t&&t.length>0?this.setColorName(t,e):this}setColorName(t,e="srgb"){const i=Ft[t.toLowerCase()];return void 0!==i?this.setHex(i,e):console.warn("THREE.Color: Unknown color "+t),this}clone(){return new this.constructor(this.r,this.g,this.b)}copy(t){return this.r=t.r,this.g=t.g,this.b=t.b,this}copySRGBToLinear(t){return this.r=Ot(t.r),this.g=Ot(t.g),this.b=Ot(t.b),this}copyLinearToSRGB(t){return this.r=zt(t.r),this.g=zt(t.g),this.b=zt(t.b),this}convertSRGBToLinear(){return this.copySRGBToLinear(this),this}convertLinearToSRGB(){return this.copyLinearToSRGB(this),this}getHex(t="srgb"){return Bt.fromWorkingColorSpace(Wt(this,kt),t),yt(255*kt.r,0,255)<<16^yt(255*kt.g,0,255)<<8^yt(255*kt.b,0,255)<<0}getHexString(t="srgb"){return("000000"+this.getHex(t).toString(16)).slice(-6)}getHSL(t,e="srgb-linear"){Bt.fromWorkingColorSpace(Wt(this,kt),e);const i=kt.r,n=kt.g,r=kt.b,s=Math.max(i,n,r),a=Math.min(i,n,r);let o,l;const c=(a+s)/2;if(a===s)o=0,l=0;else{const t=s-a;switch(l=c<=.5?t/(s+a):t/(2-s-a),s){case i:o=(n-r)/t+(n<r?6:0);break;case n:o=(r-i)/t+2;break;case r:o=(i-n)/t+4}o/=6}return t.h=o,t.s=l,t.l=c,t}getRGB(t,e="srgb-linear"){return Bt.fromWorkingColorSpace(Wt(this,kt),e),t.r=kt.r,t.g=kt.g,t.b=kt.b,t}getStyle(t="srgb"){return Bt.fromWorkingColorSpace(Wt(this,kt),t),t!==lt?`color(${t} ${kt.r} ${kt.g} ${kt.b})`:`rgb(${255*kt.r|0},${255*kt.g|0},${255*kt.b|0})`}offsetHSL(t,e,i){return this.getHSL(Gt),Gt.h+=t,Gt.s+=e,Gt.l+=i,this.setHSL(Gt.h,Gt.s,Gt.l),this}add(t){return this.r+=t.r,this.g+=t.g,this.b+=t.b,this}addColors(t,e){return this.r=t.r+e.r,this.g=t.g+e.g,this.b=t.b+e.b,this}addScalar(t){return this.r+=t,this.g+=t,this.b+=t,this}sub(t){return this.r=Math.max(0,this.r-t.r),this.g=Math.max(0,this.g-t.g),this.b=Math.max(0,this.b-t.b),this}multiply(t){return this.r*=t.r,this.g*=t.g,this.b*=t.b,this}multiplyScalar(t){return this.r*=t,this.g*=t,this.b*=t,this}lerp(t,e){return this.r+=(t.r-this.r)*e,this.g+=(t.g-this.g)*e,this.b+=(t.b-this.b)*e,this}lerpColors(t,e,i){return this.r=t.r+(e.r-t.r)*i,this.g=t.g+(e.g-t.g)*i,this.b=t.b+(e.b-t.b)*i,this}lerpHSL(t,e){this.getHSL(Gt),t.getHSL(Vt);const i=bt(Gt.h,Vt.h,e),n=bt(Gt.s,Vt.s,e),r=bt(Gt.l,Vt.l,e);return this.setHSL(i,n,r),this}equals(t){return t.r===this.r&&t.g===this.g&&t.b===this.b}fromArray(t,e=0){return this.r=t[e],this.g=t[e+1],this.b=t[e+2],this}toArray(t=[],e=0){return t[e]=this.r,t[e+1]=this.g,t[e+2]=this.b,t}fromBufferAttribute(t,e){return this.r=t.getX(e),this.g=t.getY(e),this.b=t.getZ(e),this}toJSON(){return this.getHex()}*[Symbol.iterator](){yield this.r,yield this.g,yield this.b}}let qt;jt.NAMES=Ft;class Xt{static getDataURL(t){if(/^data:/i.test(t.src))return t.src;if("undefined"==typeof HTMLCanvasElement)return t.src;let e;if(t instanceof HTMLCanvasElement)e=t;else{void 0===qt&&(qt=Nt("canvas")),qt.width=t.width,qt.height=t.height;const i=qt.getContext("2d");t instanceof ImageData?i.putImageData(t,0,0):i.drawImage(t,0,0,t.width,t.height),e=qt}return e.width>2048||e.height>2048?(console.warn("THREE.ImageUtils.getDataURL: Image converted to jpg for performance reasons",t),e.toDataURL("image/jpeg",.6)):e.toDataURL("image/png")}static sRGBToLinear(t){if("undefined"!=typeof HTMLImageElement&&t instanceof HTMLImageElement||"undefined"!=typeof HTMLCanvasElement&&t instanceof HTMLCanvasElement||"undefined"!=typeof ImageBitmap&&t instanceof ImageBitmap){const e=Nt("canvas");e.width=t.width,e.height=t.height;const i=e.getContext("2d");i.drawImage(t,0,0,t.width,t.height);const n=i.getImageData(0,0,t.width,t.height),r=n.data;for(let t=0;t<r.length;t++)r[t]=255*Ot(r[t]/255);return i.putImageData(n,0,0),e}if(t.data){const e=t.data.slice(0);for(let t=0;t<e.length;t++)e instanceof Uint8Array||e instanceof Uint8ClampedArray?e[t]=Math.floor(255*Ot(e[t]/255)):e[t]=Ot(e[t]);return{data:e,width:t.width,height:t.height}}return console.warn("THREE.ImageUtils.sRGBToLinear(): Unsupported image type. No color space conversion applied."),t}}class Yt{constructor(t=null){this.isSource=!0,this.uuid=_t(),this.data=t,this.version=0}set needsUpdate(t){!0===t&&this.version++}toJSON(t){const e=void 0===t||"string"==typeof t;if(!e&&void 0!==t.images[this.uuid])return t.images[this.uuid];const i={uuid:this.uuid,url:""},n=this.data;if(null!==n){let t;if(Array.isArray(n)){t=[];for(let e=0,i=n.length;e<i;e++)n[e].isDataTexture?t.push(Zt(n[e].image)):t.push(Zt(n[e]))}else t=Zt(n);i.url=t}return e||(t.images[this.uuid]=i),i}}function Zt(t){return"undefined"!=typeof HTMLImageElement&&t instanceof HTMLImageElement||"undefined"!=typeof HTMLCanvasElement&&t instanceof HTMLCanvasElement||"undefined"!=typeof ImageBitmap&&t instanceof ImageBitmap?Xt.getDataURL(t):t.data?{data:Array.from(t.data),width:t.width,height:t.height,type:t.data.constructor.name}:(console.warn("THREE.Texture: Unable to serialize Texture."),{})}let Jt=0;class Kt extends mt{constructor(t=Kt.DEFAULT_IMAGE,e=Kt.DEFAULT_MAPPING,i=1001,n=1001,r=1006,s=1008,a=1023,o=1009,l=1,c=3e3){super(),this.isTexture=!0,Object.defineProperty(this,"id",{value:Jt++}),this.uuid=_t(),this.name="",this.source=new Yt(t),this.mipmaps=[],this.mapping=e,this.wrapS=i,this.wrapT=n,this.magFilter=r,this.minFilter=s,this.anisotropy=l,this.format=a,this.internalFormat=null,this.type=o,this.offset=new Lt(0,0),this.repeat=new Lt(1,1),this.center=new Lt(0,0),this.rotation=0,this.matrixAutoUpdate=!0,this.matrix=new Rt,this.generateMipmaps=!0,this.premultiplyAlpha=!1,this.flipY=!0,this.unpackAlignment=4,this.encoding=c,this.userData={},this.version=0,this.onUpdate=null,this.isRenderTargetTexture=!1,this.needsPMREMUpdate=!1}get image(){return this.source.data}set image(t){this.source.data=t}updateMatrix(){this.matrix.setUvTransform(this.offset.x,this.offset.y,this.repeat.x,this.repeat.y,this.rotation,this.center.x,this.center.y)}clone(){return(new this.constructor).copy(this)}copy(t){return this.name=t.name,this.source=t.source,this.mipmaps=t.mipmaps.slice(0),this.mapping=t.mapping,this.wrapS=t.wrapS,this.wrapT=t.wrapT,this.magFilter=t.magFilter,this.minFilter=t.minFilter,this.anisotropy=t.anisotropy,this.format=t.format,this.internalFormat=t.internalFormat,this.type=t.type,this.offset.copy(t.offset),this.repeat.copy(t.repeat),this.center.copy(t.center),this.rotation=t.rotation,this.matrixAutoUpdate=t.matrixAutoUpdate,this.matrix.copy(t.matrix),this.generateMipmaps=t.generateMipmaps,this.premultiplyAlpha=t.premultiplyAlpha,this.flipY=t.flipY,this.unpackAlignment=t.unpackAlignment,this.encoding=t.encoding,this.userData=JSON.parse(JSON.stringify(t.userData)),this.needsUpdate=!0,this}toJSON(t){const e=void 0===t||"string"==typeof t;if(!e&&void 0!==t.textures[this.uuid])return t.textures[this.uuid];const i={metadata:{version:4.5,type:"Texture",generator:"Texture.toJSON"},uuid:this.uuid,name:this.name,image:this.source.toJSON(t).uuid,mapping:this.mapping,repeat:[this.repeat.x,this.repeat.y],offset:[this.offset.x,this.offset.y],center:[this.center.x,this.center.y],rotation:this.rotation,wrap:[this.wrapS,this.wrapT],format:this.format,type:this.type,encoding:this.encoding,minFilter:this.minFilter,magFilter:this.magFilter,anisotropy:this.anisotropy,flipY:this.flipY,premultiplyAlpha:this.premultiplyAlpha,unpackAlignment:this.unpackAlignment};return"{}"!==JSON.stringify(this.userData)&&(i.userData=this.userData),e||(t.textures[this.uuid]=i),i}dispose(){this.dispatchEvent({type:"dispose"})}transformUv(t){if(this.mapping!==n)return t;if(t.applyMatrix3(this.matrix),t.x<0||t.x>1)switch(this.wrapS){case c:t.x=t.x-Math.floor(t.x);break;case h:t.x=t.x<0?0:1;break;case u:1===Math.abs(Math.floor(t.x)%2)?t.x=Math.ceil(t.x)-t.x:t.x=t.x-Math.floor(t.x)}if(t.y<0||t.y>1)switch(this.wrapT){case c:t.y=t.y-Math.floor(t.y);break;case h:t.y=t.y<0?0:1;break;case u:1===Math.abs(Math.floor(t.y)%2)?t.y=Math.ceil(t.y)-t.y:t.y=t.y-Math.floor(t.y)}return this.flipY&&(t.y=1-t.y),t}set needsUpdate(t){!0===t&&(this.version++,this.source.needsUpdate=!0)}}Kt.DEFAULT_IMAGE=null,Kt.DEFAULT_MAPPING=n;class $t{constructor(t=0,e=0,i=0,n=1){$t.prototype.isVector4=!0,this.x=t,this.y=e,this.z=i,this.w=n}get width(){return this.z}set width(t){this.z=t}get height(){return this.w}set height(t){this.w=t}set(t,e,i,n){return this.x=t,this.y=e,this.z=i,this.w=n,this}setScalar(t){return this.x=t,this.y=t,this.z=t,this.w=t,this}setX(t){return this.x=t,this}setY(t){return this.y=t,this}setZ(t){return this.z=t,this}setW(t){return this.w=t,this}setComponent(t,e){switch(t){case 0:this.x=e;break;case 1:this.y=e;break;case 2:this.z=e;break;case 3:this.w=e;break;default:throw new Error("index is out of range: "+t)}return this}getComponent(t){switch(t){case 0:return this.x;case 1:return this.y;case 2:return this.z;case 3:return this.w;default:throw new Error("index is out of range: "+t)}}clone(){return new this.constructor(this.x,this.y,this.z,this.w)}copy(t){return this.x=t.x,this.y=t.y,this.z=t.z,this.w=void 0!==t.w?t.w:1,this}add(t){return this.x+=t.x,this.y+=t.y,this.z+=t.z,this.w+=t.w,this}addScalar(t){return this.x+=t,this.y+=t,this.z+=t,this.w+=t,this}addVectors(t,e){return this.x=t.x+e.x,this.y=t.y+e.y,this.z=t.z+e.z,this.w=t.w+e.w,this}addScaledVector(t,e){return this.x+=t.x*e,this.y+=t.y*e,this.z+=t.z*e,this.w+=t.w*e,this}sub(t){return this.x-=t.x,this.y-=t.y,this.z-=t.z,this.w-=t.w,this}subScalar(t){return this.x-=t,this.y-=t,this.z-=t,this.w-=t,this}subVectors(t,e){return this.x=t.x-e.x,this.y=t.y-e.y,this.z=t.z-e.z,this.w=t.w-e.w,this}multiply(t){return this.x*=t.x,this.y*=t.y,this.z*=t.z,this.w*=t.w,this}multiplyScalar(t){return this.x*=t,this.y*=t,this.z*=t,this.w*=t,this}applyMatrix4(t){const e=this.x,i=this.y,n=this.z,r=this.w,s=t.elements;return this.x=s[0]*e+s[4]*i+s[8]*n+s[12]*r,this.y=s[1]*e+s[5]*i+s[9]*n+s[13]*r,this.z=s[2]*e+s[6]*i+s[10]*n+s[14]*r,this.w=s[3]*e+s[7]*i+s[11]*n+s[15]*r,this}divideScalar(t){return this.multiplyScalar(1/t)}setAxisAngleFromQuaternion(t){this.w=2*Math.acos(t.w);const e=Math.sqrt(1-t.w*t.w);return e<1e-4?(this.x=1,this.y=0,this.z=0):(this.x=t.x/e,this.y=t.y/e,this.z=t.z/e),this}setAxisAngleFromRotationMatrix(t){let e,i,n,r;const s=.01,a=.1,o=t.elements,l=o[0],c=o[4],h=o[8],u=o[1],d=o[5],p=o[9],m=o[2],f=o[6],g=o[10];if(Math.abs(c-u)<s&&Math.abs(h-m)<s&&Math.abs(p-f)<s){if(Math.abs(c+u)<a&&Math.abs(h+m)<a&&Math.abs(p+f)<a&&Math.abs(l+d+g-3)<a)return this.set(1,0,0,0),this;e=Math.PI;const t=(l+1)/2,o=(d+1)/2,v=(g+1)/2,x=(c+u)/4,_=(h+m)/4,y=(p+f)/4;return t>o&&t>v?t<s?(i=0,n=.707106781,r=.707106781):(i=Math.sqrt(t),n=x/i,r=_/i):o>v?o<s?(i=.707106781,n=0,r=.707106781):(n=Math.sqrt(o),i=x/n,r=y/n):v<s?(i=.707106781,n=.707106781,r=0):(r=Math.sqrt(v),i=_/r,n=y/r),this.set(i,n,r,e),this}let v=Math.sqrt((f-p)*(f-p)+(h-m)*(h-m)+(u-c)*(u-c));return Math.abs(v)<.001&&(v=1),this.x=(f-p)/v,this.y=(h-m)/v,this.z=(u-c)/v,this.w=Math.acos((l+d+g-1)/2),this}min(t){return this.x=Math.min(this.x,t.x),this.y=Math.min(this.y,t.y),this.z=Math.min(this.z,t.z),this.w=Math.min(this.w,t.w),this}max(t){return this.x=Math.max(this.x,t.x),this.y=Math.max(this.y,t.y),this.z=Math.max(this.z,t.z),this.w=Math.max(this.w,t.w),this}clamp(t,e){return this.x=Math.max(t.x,Math.min(e.x,this.x)),this.y=Math.max(t.y,Math.min(e.y,this.y)),this.z=Math.max(t.z,Math.min(e.z,this.z)),this.w=Math.max(t.w,Math.min(e.w,this.w)),this}clampScalar(t,e){return this.x=Math.max(t,Math.min(e,this.x)),this.y=Math.max(t,Math.min(e,this.y)),this.z=Math.max(t,Math.min(e,this.z)),this.w=Math.max(t,Math.min(e,this.w)),this}clampLength(t,e){const i=this.length();return this.divideScalar(i||1).multiplyScalar(Math.max(t,Math.min(e,i)))}floor(){return this.x=Math.floor(this.x),this.y=Math.floor(this.y),this.z=Math.floor(this.z),this.w=Math.floor(this.w),this}ceil(){return this.x=Math.ceil(this.x),this.y=Math.ceil(this.y),this.z=Math.ceil(this.z),this.w=Math.ceil(this.w),this}round(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this.z=Math.round(this.z),this.w=Math.round(this.w),this}roundToZero(){return this.x=this.x<0?Math.ceil(this.x):Math.floor(this.x),this.y=this.y<0?Math.ceil(this.y):Math.floor(this.y),this.z=this.z<0?Math.ceil(this.z):Math.floor(this.z),this.w=this.w<0?Math.ceil(this.w):Math.floor(this.w),this}negate(){return this.x=-this.x,this.y=-this.y,this.z=-this.z,this.w=-this.w,this}dot(t){return this.x*t.x+this.y*t.y+this.z*t.z+this.w*t.w}lengthSq(){return this.x*this.x+this.y*this.y+this.z*this.z+this.w*this.w}length(){return Math.sqrt(this.x*this.x+this.y*this.y+this.z*this.z+this.w*this.w)}manhattanLength(){return Math.abs(this.x)+Math.abs(this.y)+Math.abs(this.z)+Math.abs(this.w)}normalize(){return this.divideScalar(this.length()||1)}setLength(t){return this.normalize().multiplyScalar(t)}lerp(t,e){return this.x+=(t.x-this.x)*e,this.y+=(t.y-this.y)*e,this.z+=(t.z-this.z)*e,this.w+=(t.w-this.w)*e,this}lerpVectors(t,e,i){return this.x=t.x+(e.x-t.x)*i,this.y=t.y+(e.y-t.y)*i,this.z=t.z+(e.z-t.z)*i,this.w=t.w+(e.w-t.w)*i,this}equals(t){return t.x===this.x&&t.y===this.y&&t.z===this.z&&t.w===this.w}fromArray(t,e=0){return this.x=t[e],this.y=t[e+1],this.z=t[e+2],this.w=t[e+3],this}toArray(t=[],e=0){return t[e]=this.x,t[e+1]=this.y,t[e+2]=this.z,t[e+3]=this.w,t}fromBufferAttribute(t,e){return this.x=t.getX(e),this.y=t.getY(e),this.z=t.getZ(e),this.w=t.getW(e),this}random(){return this.x=Math.random(),this.y=Math.random(),this.z=Math.random(),this.w=Math.random(),this}*[Symbol.iterator](){yield this.x,yield this.y,yield this.z,yield this.w}}class Qt extends mt{constructor(t=1,e=1,i={}){super(),this.isWebGLRenderTarget=!0,this.width=t,this.height=e,this.depth=1,this.scissor=new $t(0,0,t,e),this.scissorTest=!1,this.viewport=new $t(0,0,t,e);const n={width:t,height:e,depth:1};this.texture=new Kt(n,i.mapping,i.wrapS,i.wrapT,i.magFilter,i.minFilter,i.format,i.type,i.anisotropy,i.encoding),this.texture.isRenderTargetTexture=!0,this.texture.flipY=!1,this.texture.generateMipmaps=void 0!==i.generateMipmaps&&i.generateMipmaps,this.texture.internalFormat=void 0!==i.internalFormat?i.internalFormat:null,this.texture.minFilter=void 0!==i.minFilter?i.minFilter:f,this.depthBuffer=void 0===i.depthBuffer||i.depthBuffer,this.stencilBuffer=void 0!==i.stencilBuffer&&i.stencilBuffer,this.depthTexture=void 0!==i.depthTexture?i.depthTexture:null,this.samples=void 0!==i.samples?i.samples:0}setSize(t,e,i=1){this.width===t&&this.height===e&&this.depth===i||(this.width=t,this.height=e,this.depth=i,this.texture.image.width=t,this.texture.image.height=e,this.texture.image.depth=i,this.dispose()),this.viewport.set(0,0,t,e),this.scissor.set(0,0,t,e)}clone(){return(new this.constructor).copy(this)}copy(t){this.width=t.width,this.height=t.height,this.depth=t.depth,this.viewport.copy(t.viewport),this.texture=t.texture.clone(),this.texture.isRenderTargetTexture=!0;const e=Object.assign({},t.texture.image);return this.texture.source=new Yt(e),this.depthBuffer=t.depthBuffer,this.stencilBuffer=t.stencilBuffer,null!==t.depthTexture&&(this.depthTexture=t.depthTexture.clone()),this.samples=t.samples,this}dispose(){this.dispatchEvent({type:"dispose"})}}class te extends Kt{constructor(t=null,e=1,i=1,n=1){super(null),this.isDataArrayTexture=!0,this.image={data:t,width:e,height:i,depth:n},this.magFilter=d,this.minFilter=d,this.wrapR=h,this.generateMipmaps=!1,this.flipY=!1,this.unpackAlignment=1}}class ee extends Kt{constructor(t=null,e=1,i=1,n=1){super(null),this.isData3DTexture=!0,this.image={data:t,width:e,height:i,depth:n},this.magFilter=d,this.minFilter=d,this.wrapR=h,this.generateMipmaps=!1,this.flipY=!1,this.unpackAlignment=1}}class ie{constructor(t=0,e=0,i=0,n=1){this.isQuaternion=!0,this._x=t,this._y=e,this._z=i,this._w=n}static slerpFlat(t,e,i,n,r,s,a){let o=i[n+0],l=i[n+1],c=i[n+2],h=i[n+3];const u=r[s+0],d=r[s+1],p=r[s+2],m=r[s+3];if(0===a)return t[e+0]=o,t[e+1]=l,t[e+2]=c,void(t[e+3]=h);if(1===a)return t[e+0]=u,t[e+1]=d,t[e+2]=p,void(t[e+3]=m);if(h!==m||o!==u||l!==d||c!==p){let t=1-a;const e=o*u+l*d+c*p+h*m,i=e>=0?1:-1,n=1-e*e;if(n>Number.EPSILON){const r=Math.sqrt(n),s=Math.atan2(r,e*i);t=Math.sin(t*s)/r,a=Math.sin(a*s)/r}const r=a*i;if(o=o*t+u*r,l=l*t+d*r,c=c*t+p*r,h=h*t+m*r,t===1-a){const t=1/Math.sqrt(o*o+l*l+c*c+h*h);o*=t,l*=t,c*=t,h*=t}}t[e]=o,t[e+1]=l,t[e+2]=c,t[e+3]=h}static multiplyQuaternionsFlat(t,e,i,n,r,s){const a=i[n],o=i[n+1],l=i[n+2],c=i[n+3],h=r[s],u=r[s+1],d=r[s+2],p=r[s+3];return t[e]=a*p+c*h+o*d-l*u,t[e+1]=o*p+c*u+l*h-a*d,t[e+2]=l*p+c*d+a*u-o*h,t[e+3]=c*p-a*h-o*u-l*d,t}get x(){return this._x}set x(t){this._x=t,this._onChangeCallback()}get y(){return this._y}set y(t){this._y=t,this._onChangeCallback()}get z(){return this._z}set z(t){this._z=t,this._onChangeCallback()}get w(){return this._w}set w(t){this._w=t,this._onChangeCallback()}set(t,e,i,n){return this._x=t,this._y=e,this._z=i,this._w=n,this._onChangeCallback(),this}clone(){return new this.constructor(this._x,this._y,this._z,this._w)}copy(t){return this._x=t.x,this._y=t.y,this._z=t.z,this._w=t.w,this._onChangeCallback(),this}setFromEuler(t,e){const i=t._x,n=t._y,r=t._z,s=t._order,a=Math.cos,o=Math.sin,l=a(i/2),c=a(n/2),h=a(r/2),u=o(i/2),d=o(n/2),p=o(r/2);switch(s){case"XYZ":this._x=u*c*h+l*d*p,this._y=l*d*h-u*c*p,this._z=l*c*p+u*d*h,this._w=l*c*h-u*d*p;break;case"YXZ":this._x=u*c*h+l*d*p,this._y=l*d*h-u*c*p,this._z=l*c*p-u*d*h,this._w=l*c*h+u*d*p;break;case"ZXY":this._x=u*c*h-l*d*p,this._y=l*d*h+u*c*p,this._z=l*c*p+u*d*h,this._w=l*c*h-u*d*p;break;case"ZYX":this._x=u*c*h-l*d*p,this._y=l*d*h+u*c*p,this._z=l*c*p-u*d*h,this._w=l*c*h+u*d*p;break;case"YZX":this._x=u*c*h+l*d*p,this._y=l*d*h+u*c*p,this._z=l*c*p-u*d*h,this._w=l*c*h-u*d*p;break;case"XZY":this._x=u*c*h-l*d*p,this._y=l*d*h-u*c*p,this._z=l*c*p+u*d*h,this._w=l*c*h+u*d*p;break;default:console.warn("THREE.Quaternion: .setFromEuler() encountered an unknown order: "+s)}return!1!==e&&this._onChangeCallback(),this}setFromAxisAngle(t,e){const i=e/2,n=Math.sin(i);return this._x=t.x*n,this._y=t.y*n,this._z=t.z*n,this._w=Math.cos(i),this._onChangeCallback(),this}setFromRotationMatrix(t){const e=t.elements,i=e[0],n=e[4],r=e[8],s=e[1],a=e[5],o=e[9],l=e[2],c=e[6],h=e[10],u=i+a+h;if(u>0){const t=.5/Math.sqrt(u+1);this._w=.25/t,this._x=(c-o)*t,this._y=(r-l)*t,this._z=(s-n)*t}else if(i>a&&i>h){const t=2*Math.sqrt(1+i-a-h);this._w=(c-o)/t,this._x=.25*t,this._y=(n+s)/t,this._z=(r+l)/t}else if(a>h){const t=2*Math.sqrt(1+a-i-h);this._w=(r-l)/t,this._x=(n+s)/t,this._y=.25*t,this._z=(o+c)/t}else{const t=2*Math.sqrt(1+h-i-a);this._w=(s-n)/t,this._x=(r+l)/t,this._y=(o+c)/t,this._z=.25*t}return this._onChangeCallback(),this}setFromUnitVectors(t,e){let i=t.dot(e)+1;return i<Number.EPSILON?(i=0,Math.abs(t.x)>Math.abs(t.z)?(this._x=-t.y,this._y=t.x,this._z=0,this._w=i):(this._x=0,this._y=-t.z,this._z=t.y,this._w=i)):(this._x=t.y*e.z-t.z*e.y,this._y=t.z*e.x-t.x*e.z,this._z=t.x*e.y-t.y*e.x,this._w=i),this.normalize()}angleTo(t){return 2*Math.acos(Math.abs(yt(this.dot(t),-1,1)))}rotateTowards(t,e){const i=this.angleTo(t);if(0===i)return this;const n=Math.min(1,e/i);return this.slerp(t,n),this}identity(){return this.set(0,0,0,1)}invert(){return this.conjugate()}conjugate(){return this._x*=-1,this._y*=-1,this._z*=-1,this._onChangeCallback(),this}dot(t){return this._x*t._x+this._y*t._y+this._z*t._z+this._w*t._w}lengthSq(){return this._x*this._x+this._y*this._y+this._z*this._z+this._w*this._w}length(){return Math.sqrt(this._x*this._x+this._y*this._y+this._z*this._z+this._w*this._w)}normalize(){let t=this.length();return 0===t?(this._x=0,this._y=0,this._z=0,this._w=1):(t=1/t,this._x=this._x*t,this._y=this._y*t,this._z=this._z*t,this._w=this._w*t),this._onChangeCallback(),this}multiply(t){return this.multiplyQuaternions(this,t)}premultiply(t){return this.multiplyQuaternions(t,this)}multiplyQuaternions(t,e){const i=t._x,n=t._y,r=t._z,s=t._w,a=e._x,o=e._y,l=e._z,c=e._w;return this._x=i*c+s*a+n*l-r*o,this._y=n*c+s*o+r*a-i*l,this._z=r*c+s*l+i*o-n*a,this._w=s*c-i*a-n*o-r*l,this._onChangeCallback(),this}slerp(t,e){if(0===e)return this;if(1===e)return this.copy(t);const i=this._x,n=this._y,r=this._z,s=this._w;let a=s*t._w+i*t._x+n*t._y+r*t._z;if(a<0?(this._w=-t._w,this._x=-t._x,this._y=-t._y,this._z=-t._z,a=-a):this.copy(t),a>=1)return this._w=s,this._x=i,this._y=n,this._z=r,this;const o=1-a*a;if(o<=Number.EPSILON){const t=1-e;return this._w=t*s+e*this._w,this._x=t*i+e*this._x,this._y=t*n+e*this._y,this._z=t*r+e*this._z,this.normalize(),this._onChangeCallback(),this}const l=Math.sqrt(o),c=Math.atan2(l,a),h=Math.sin((1-e)*c)/l,u=Math.sin(e*c)/l;return this._w=s*h+this._w*u,this._x=i*h+this._x*u,this._y=n*h+this._y*u,this._z=r*h+this._z*u,this._onChangeCallback(),this}slerpQuaternions(t,e,i){return this.copy(t).slerp(e,i)}random(){const t=Math.random(),e=Math.sqrt(1-t),i=Math.sqrt(t),n=2*Math.PI*Math.random(),r=2*Math.PI*Math.random();return this.set(e*Math.cos(n),i*Math.sin(r),i*Math.cos(r),e*Math.sin(n))}equals(t){return t._x===this._x&&t._y===this._y&&t._z===this._z&&t._w===this._w}fromArray(t,e=0){return this._x=t[e],this._y=t[e+1],this._z=t[e+2],this._w=t[e+3],this._onChangeCallback(),this}toArray(t=[],e=0){return t[e]=this._x,t[e+1]=this._y,t[e+2]=this._z,t[e+3]=this._w,t}fromBufferAttribute(t,e){return this._x=t.getX(e),this._y=t.getY(e),this._z=t.getZ(e),this._w=t.getW(e),this}_onChange(t){return this._onChangeCallback=t,this}_onChangeCallback(){}*[Symbol.iterator](){yield this._x,yield this._y,yield this._z,yield this._w}}class ne{constructor(t=0,e=0,i=0){ne.prototype.isVector3=!0,this.x=t,this.y=e,this.z=i}set(t,e,i){return void 0===i&&(i=this.z),this.x=t,this.y=e,this.z=i,this}setScalar(t){return this.x=t,this.y=t,this.z=t,this}setX(t){return this.x=t,this}setY(t){return this.y=t,this}setZ(t){return this.z=t,this}setComponent(t,e){switch(t){case 0:this.x=e;break;case 1:this.y=e;break;case 2:this.z=e;break;default:throw new Error("index is out of range: "+t)}return this}getComponent(t){switch(t){case 0:return this.x;case 1:return this.y;case 2:return this.z;default:throw new Error("index is out of range: "+t)}}clone(){return new this.constructor(this.x,this.y,this.z)}copy(t){return this.x=t.x,this.y=t.y,this.z=t.z,this}add(t){return this.x+=t.x,this.y+=t.y,this.z+=t.z,this}addScalar(t){return this.x+=t,this.y+=t,this.z+=t,this}addVectors(t,e){return this.x=t.x+e.x,this.y=t.y+e.y,this.z=t.z+e.z,this}addScaledVector(t,e){return this.x+=t.x*e,this.y+=t.y*e,this.z+=t.z*e,this}sub(t){return this.x-=t.x,this.y-=t.y,this.z-=t.z,this}subScalar(t){return this.x-=t,this.y-=t,this.z-=t,this}subVectors(t,e){return this.x=t.x-e.x,this.y=t.y-e.y,this.z=t.z-e.z,this}multiply(t){return this.x*=t.x,this.y*=t.y,this.z*=t.z,this}multiplyScalar(t){return this.x*=t,this.y*=t,this.z*=t,this}multiplyVectors(t,e){return this.x=t.x*e.x,this.y=t.y*e.y,this.z=t.z*e.z,this}applyEuler(t){return this.applyQuaternion(se.setFromEuler(t))}applyAxisAngle(t,e){return this.applyQuaternion(se.setFromAxisAngle(t,e))}applyMatrix3(t){const e=this.x,i=this.y,n=this.z,r=t.elements;return this.x=r[0]*e+r[3]*i+r[6]*n,this.y=r[1]*e+r[4]*i+r[7]*n,this.z=r[2]*e+r[5]*i+r[8]*n,this}applyNormalMatrix(t){return this.applyMatrix3(t).normalize()}applyMatrix4(t){const e=this.x,i=this.y,n=this.z,r=t.elements,s=1/(r[3]*e+r[7]*i+r[11]*n+r[15]);return this.x=(r[0]*e+r[4]*i+r[8]*n+r[12])*s,this.y=(r[1]*e+r[5]*i+r[9]*n+r[13])*s,this.z=(r[2]*e+r[6]*i+r[10]*n+r[14])*s,this}applyQuaternion(t){const e=this.x,i=this.y,n=this.z,r=t.x,s=t.y,a=t.z,o=t.w,l=o*e+s*n-a*i,c=o*i+a*e-r*n,h=o*n+r*i-s*e,u=-r*e-s*i-a*n;return this.x=l*o+u*-r+c*-a-h*-s,this.y=c*o+u*-s+h*-r-l*-a,this.z=h*o+u*-a+l*-s-c*-r,this}project(t){return this.applyMatrix4(t.matrixWorldInverse).applyMatrix4(t.projectionMatrix)}unproject(t){return this.applyMatrix4(t.projectionMatrixInverse).applyMatrix4(t.matrixWorld)}transformDirection(t){const e=this.x,i=this.y,n=this.z,r=t.elements;return this.x=r[0]*e+r[4]*i+r[8]*n,this.y=r[1]*e+r[5]*i+r[9]*n,this.z=r[2]*e+r[6]*i+r[10]*n,this.normalize()}divide(t){return this.x/=t.x,this.y/=t.y,this.z/=t.z,this}divideScalar(t){return this.multiplyScalar(1/t)}min(t){return this.x=Math.min(this.x,t.x),this.y=Math.min(this.y,t.y),this.z=Math.min(this.z,t.z),this}max(t){return this.x=Math.max(this.x,t.x),this.y=Math.max(this.y,t.y),this.z=Math.max(this.z,t.z),this}clamp(t,e){return this.x=Math.max(t.x,Math.min(e.x,this.x)),this.y=Math.max(t.y,Math.min(e.y,this.y)),this.z=Math.max(t.z,Math.min(e.z,this.z)),this}clampScalar(t,e){return this.x=Math.max(t,Math.min(e,this.x)),this.y=Math.max(t,Math.min(e,this.y)),this.z=Math.max(t,Math.min(e,this.z)),this}clampLength(t,e){const i=this.length();return this.divideScalar(i||1).multiplyScalar(Math.max(t,Math.min(e,i)))}floor(){return this.x=Math.floor(this.x),this.y=Math.floor(this.y),this.z=Math.floor(this.z),this}ceil(){return this.x=Math.ceil(this.x),this.y=Math.ceil(this.y),this.z=Math.ceil(this.z),this}round(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this.z=Math.round(this.z),this}roundToZero(){return this.x=this.x<0?Math.ceil(this.x):Math.floor(this.x),this.y=this.y<0?Math.ceil(this.y):Math.floor(this.y),this.z=this.z<0?Math.ceil(this.z):Math.floor(this.z),this}negate(){return this.x=-this.x,this.y=-this.y,this.z=-this.z,this}dot(t){return this.x*t.x+this.y*t.y+this.z*t.z}lengthSq(){return this.x*this.x+this.y*this.y+this.z*this.z}length(){return Math.sqrt(this.x*this.x+this.y*this.y+this.z*this.z)}manhattanLength(){return Math.abs(this.x)+Math.abs(this.y)+Math.abs(this.z)}normalize(){return this.divideScalar(this.length()||1)}setLength(t){return this.normalize().multiplyScalar(t)}lerp(t,e){return this.x+=(t.x-this.x)*e,this.y+=(t.y-this.y)*e,this.z+=(t.z-this.z)*e,this}lerpVectors(t,e,i){return this.x=t.x+(e.x-t.x)*i,this.y=t.y+(e.y-t.y)*i,this.z=t.z+(e.z-t.z)*i,this}cross(t){return this.crossVectors(this,t)}crossVectors(t,e){const i=t.x,n=t.y,r=t.z,s=e.x,a=e.y,o=e.z;return this.x=n*o-r*a,this.y=r*s-i*o,this.z=i*a-n*s,this}projectOnVector(t){const e=t.lengthSq();if(0===e)return this.set(0,0,0);const i=t.dot(this)/e;return this.copy(t).multiplyScalar(i)}projectOnPlane(t){return re.copy(this).projectOnVector(t),this.sub(re)}reflect(t){return this.sub(re.copy(t).multiplyScalar(2*this.dot(t)))}angleTo(t){const e=Math.sqrt(this.lengthSq()*t.lengthSq());if(0===e)return Math.PI/2;const i=this.dot(t)/e;return Math.acos(yt(i,-1,1))}distanceTo(t){return Math.sqrt(this.distanceToSquared(t))}distanceToSquared(t){const e=this.x-t.x,i=this.y-t.y,n=this.z-t.z;return e*e+i*i+n*n}manhattanDistanceTo(t){return Math.abs(this.x-t.x)+Math.abs(this.y-t.y)+Math.abs(this.z-t.z)}setFromSpherical(t){return this.setFromSphericalCoords(t.radius,t.phi,t.theta)}setFromSphericalCoords(t,e,i){const n=Math.sin(e)*t;return this.x=n*Math.sin(i),this.y=Math.cos(e)*t,this.z=n*Math.cos(i),this}setFromCylindrical(t){return this.setFromCylindricalCoords(t.radius,t.theta,t.y)}setFromCylindricalCoords(t,e,i){return this.x=t*Math.sin(e),this.y=i,this.z=t*Math.cos(e),this}setFromMatrixPosition(t){const e=t.elements;return this.x=e[12],this.y=e[13],this.z=e[14],this}setFromMatrixScale(t){const e=this.setFromMatrixColumn(t,0).length(),i=this.setFromMatrixColumn(t,1).length(),n=this.setFromMatrixColumn(t,2).length();return this.x=e,this.y=i,this.z=n,this}setFromMatrixColumn(t,e){return this.fromArray(t.elements,4*e)}setFromMatrix3Column(t,e){return this.fromArray(t.elements,3*e)}setFromEuler(t){return this.x=t._x,this.y=t._y,this.z=t._z,this}equals(t){return t.x===this.x&&t.y===this.y&&t.z===this.z}fromArray(t,e=0){return this.x=t[e],this.y=t[e+1],this.z=t[e+2],this}toArray(t=[],e=0){return t[e]=this.x,t[e+1]=this.y,t[e+2]=this.z,t}fromBufferAttribute(t,e){return this.x=t.getX(e),this.y=t.getY(e),this.z=t.getZ(e),this}random(){return this.x=Math.random(),this.y=Math.random(),this.z=Math.random(),this}randomDirection(){const t=2*(Math.random()-.5),e=Math.random()*Math.PI*2,i=Math.sqrt(1-t**2);return this.x=i*Math.cos(e),this.y=i*Math.sin(e),this.z=t,this}*[Symbol.iterator](){yield this.x,yield this.y,yield this.z}}const re=new ne,se=new ie;class ae{constructor(t=new ne(1/0,1/0,1/0),e=new ne(-1/0,-1/0,-1/0)){this.isBox3=!0,this.min=t,this.max=e}set(t,e){return this.min.copy(t),this.max.copy(e),this}setFromArray(t){let e=1/0,i=1/0,n=1/0,r=-1/0,s=-1/0,a=-1/0;for(let o=0,l=t.length;o<l;o+=3){const l=t[o],c=t[o+1],h=t[o+2];l<e&&(e=l),c<i&&(i=c),h<n&&(n=h),l>r&&(r=l),c>s&&(s=c),h>a&&(a=h)}return this.min.set(e,i,n),this.max.set(r,s,a),this}setFromBufferAttribute(t){let e=1/0,i=1/0,n=1/0,r=-1/0,s=-1/0,a=-1/0;for(let o=0,l=t.count;o<l;o++){const l=t.getX(o),c=t.getY(o),h=t.getZ(o);l<e&&(e=l),c<i&&(i=c),h<n&&(n=h),l>r&&(r=l),c>s&&(s=c),h>a&&(a=h)}return this.min.set(e,i,n),this.max.set(r,s,a),this}setFromPoints(t){this.makeEmpty();for(let e=0,i=t.length;e<i;e++)this.expandByPoint(t[e]);return this}setFromCenterAndSize(t,e){const i=le.copy(e).multiplyScalar(.5);return this.min.copy(t).sub(i),this.max.copy(t).add(i),this}setFromObject(t,e=!1){return this.makeEmpty(),this.expandByObject(t,e)}clone(){return(new this.constructor).copy(this)}copy(t){return this.min.copy(t.min),this.max.copy(t.max),this}makeEmpty(){return this.min.x=this.min.y=this.min.z=1/0,this.max.x=this.max.y=this.max.z=-1/0,this}isEmpty(){return this.max.x<this.min.x||this.max.y<this.min.y||this.max.z<this.min.z}getCenter(t){return this.isEmpty()?t.set(0,0,0):t.addVectors(this.min,this.max).multiplyScalar(.5)}getSize(t){return this.isEmpty()?t.set(0,0,0):t.subVectors(this.max,this.min)}expandByPoint(t){return this.min.min(t),this.max.max(t),this}expandByVector(t){return this.min.sub(t),this.max.add(t),this}expandByScalar(t){return this.min.addScalar(-t),this.max.addScalar(t),this}expandByObject(t,e=!1){t.updateWorldMatrix(!1,!1);const i=t.geometry;if(void 0!==i)if(e&&null!=i.attributes&&void 0!==i.attributes.position){const e=i.attributes.position;for(let i=0,n=e.count;i<n;i++)le.fromBufferAttribute(e,i).applyMatrix4(t.matrixWorld),this.expandByPoint(le)}else null===i.boundingBox&&i.computeBoundingBox(),ce.copy(i.boundingBox),ce.applyMatrix4(t.matrixWorld),this.union(ce);const n=t.children;for(let t=0,i=n.length;t<i;t++)this.expandByObject(n[t],e);return this}containsPoint(t){return!(t.x<this.min.x||t.x>this.max.x||t.y<this.min.y||t.y>this.max.y||t.z<this.min.z||t.z>this.max.z)}containsBox(t){return this.min.x<=t.min.x&&t.max.x<=this.max.x&&this.min.y<=t.min.y&&t.max.y<=this.max.y&&this.min.z<=t.min.z&&t.max.z<=this.max.z}getParameter(t,e){return e.set((t.x-this.min.x)/(this.max.x-this.min.x),(t.y-this.min.y)/(this.max.y-this.min.y),(t.z-this.min.z)/(this.max.z-this.min.z))}intersectsBox(t){return!(t.max.x<this.min.x||t.min.x>this.max.x||t.max.y<this.min.y||t.min.y>this.max.y||t.max.z<this.min.z||t.min.z>this.max.z)}intersectsSphere(t){return this.clampPoint(t.center,le),le.distanceToSquared(t.center)<=t.radius*t.radius}intersectsPlane(t){let e,i;return t.normal.x>0?(e=t.normal.x*this.min.x,i=t.normal.x*this.max.x):(e=t.normal.x*this.max.x,i=t.normal.x*this.min.x),t.normal.y>0?(e+=t.normal.y*this.min.y,i+=t.normal.y*this.max.y):(e+=t.normal.y*this.max.y,i+=t.normal.y*this.min.y),t.normal.z>0?(e+=t.normal.z*this.min.z,i+=t.normal.z*this.max.z):(e+=t.normal.z*this.max.z,i+=t.normal.z*this.min.z),e<=-t.constant&&i>=-t.constant}intersectsTriangle(t){if(this.isEmpty())return!1;this.getCenter(ge),ve.subVectors(this.max,ge),he.subVectors(t.a,ge),ue.subVectors(t.b,ge),de.subVectors(t.c,ge),pe.subVectors(ue,he),me.subVectors(de,ue),fe.subVectors(he,de);let e=[0,-pe.z,pe.y,0,-me.z,me.y,0,-fe.z,fe.y,pe.z,0,-pe.x,me.z,0,-me.x,fe.z,0,-fe.x,-pe.y,pe.x,0,-me.y,me.x,0,-fe.y,fe.x,0];return!!ye(e,he,ue,de,ve)&&(e=[1,0,0,0,1,0,0,0,1],!!ye(e,he,ue,de,ve)&&(xe.crossVectors(pe,me),e=[xe.x,xe.y,xe.z],ye(e,he,ue,de,ve)))}clampPoint(t,e){return e.copy(t).clamp(this.min,this.max)}distanceToPoint(t){return le.copy(t).clamp(this.min,this.max).sub(t).length()}getBoundingSphere(t){return this.getCenter(t.center),t.radius=.5*this.getSize(le).length(),t}intersect(t){return this.min.max(t.min),this.max.min(t.max),this.isEmpty()&&this.makeEmpty(),this}union(t){return this.min.min(t.min),this.max.max(t.max),this}applyMatrix4(t){return this.isEmpty()||(oe[0].set(this.min.x,this.min.y,this.min.z).applyMatrix4(t),oe[1].set(this.min.x,this.min.y,this.max.z).applyMatrix4(t),oe[2].set(this.min.x,this.max.y,this.min.z).applyMatrix4(t),oe[3].set(this.min.x,this.max.y,this.max.z).applyMatrix4(t),oe[4].set(this.max.x,this.min.y,this.min.z).applyMatrix4(t),oe[5].set(this.max.x,this.min.y,this.max.z).applyMatrix4(t),oe[6].set(this.max.x,this.max.y,this.min.z).applyMatrix4(t),oe[7].set(this.max.x,this.max.y,this.max.z).applyMatrix4(t),this.setFromPoints(oe)),this}translate(t){return this.min.add(t),this.max.add(t),this}equals(t){return t.min.equals(this.min)&&t.max.equals(this.max)}}const oe=[new ne,new ne,new ne,new ne,new ne,new ne,new ne,new ne],le=new ne,ce=new ae,he=new ne,ue=new ne,de=new ne,pe=new ne,me=new ne,fe=new ne,ge=new ne,ve=new ne,xe=new ne,_e=new ne;function ye(t,e,i,n,r){for(let s=0,a=t.length-3;s<=a;s+=3){_e.fromArray(t,s);const a=r.x*Math.abs(_e.x)+r.y*Math.abs(_e.y)+r.z*Math.abs(_e.z),o=e.dot(_e),l=i.dot(_e),c=n.dot(_e);if(Math.max(-Math.max(o,l,c),Math.min(o,l,c))>a)return!1}return!0}const Me=new ae,be=new ne,Se=new ne;class we{constructor(t=new ne,e=-1){this.center=t,this.radius=e}set(t,e){return this.center.copy(t),this.radius=e,this}setFromPoints(t,e){const i=this.center;void 0!==e?i.copy(e):Me.setFromPoints(t).getCenter(i);let n=0;for(let e=0,r=t.length;e<r;e++)n=Math.max(n,i.distanceToSquared(t[e]));return this.radius=Math.sqrt(n),this}copy(t){return this.center.copy(t.center),this.radius=t.radius,this}isEmpty(){return this.radius<0}makeEmpty(){return this.center.set(0,0,0),this.radius=-1,this}containsPoint(t){return t.distanceToSquared(this.center)<=this.radius*this.radius}distanceToPoint(t){return t.distanceTo(this.center)-this.radius}intersectsSphere(t){const e=this.radius+t.radius;return t.center.distanceToSquared(this.center)<=e*e}intersectsBox(t){return t.intersectsSphere(this)}intersectsPlane(t){return Math.abs(t.distanceToPoint(this.center))<=this.radius}clampPoint(t,e){const i=this.center.distanceToSquared(t);return e.copy(t),i>this.radius*this.radius&&(e.sub(this.center).normalize(),e.multiplyScalar(this.radius).add(this.center)),e}getBoundingBox(t){return this.isEmpty()?(t.makeEmpty(),t):(t.set(this.center,this.center),t.expandByScalar(this.radius),t)}applyMatrix4(t){return this.center.applyMatrix4(t),this.radius=this.radius*t.getMaxScaleOnAxis(),this}translate(t){return this.center.add(t),this}expandByPoint(t){if(this.isEmpty())return this.center.copy(t),this.radius=0,this;be.subVectors(t,this.center);const e=be.lengthSq();if(e>this.radius*this.radius){const t=Math.sqrt(e),i=.5*(t-this.radius);this.center.addScaledVector(be,i/t),this.radius+=i}return this}union(t){return t.isEmpty()?this:this.isEmpty()?(this.copy(t),this):(!0===this.center.equals(t.center)?this.radius=Math.max(this.radius,t.radius):(Se.subVectors(t.center,this.center).setLength(t.radius),this.expandByPoint(be.copy(t.center).add(Se)),this.expandByPoint(be.copy(t.center).sub(Se))),this)}equals(t){return t.center.equals(this.center)&&t.radius===this.radius}clone(){return(new this.constructor).copy(this)}}const Te=new ne,Ae=new ne,Ee=new ne,Ce=new ne,Le=new ne,Re=new ne,Pe=new ne;class De{constructor(t=new ne,e=new ne(0,0,-1)){this.origin=t,this.direction=e}set(t,e){return this.origin.copy(t),this.direction.copy(e),this}copy(t){return this.origin.copy(t.origin),this.direction.copy(t.direction),this}at(t,e){return e.copy(this.direction).multiplyScalar(t).add(this.origin)}lookAt(t){return this.direction.copy(t).sub(this.origin).normalize(),this}recast(t){return this.origin.copy(this.at(t,Te)),this}closestPointToPoint(t,e){e.subVectors(t,this.origin);const i=e.dot(this.direction);return i<0?e.copy(this.origin):e.copy(this.direction).multiplyScalar(i).add(this.origin)}distanceToPoint(t){return Math.sqrt(this.distanceSqToPoint(t))}distanceSqToPoint(t){const e=Te.subVectors(t,this.origin).dot(this.direction);return e<0?this.origin.distanceToSquared(t):(Te.copy(this.direction).multiplyScalar(e).add(this.origin),Te.distanceToSquared(t))}distanceSqToSegment(t,e,i,n){Ae.copy(t).add(e).multiplyScalar(.5),Ee.copy(e).sub(t).normalize(),Ce.copy(this.origin).sub(Ae);const r=.5*t.distanceTo(e),s=-this.direction.dot(Ee),a=Ce.dot(this.direction),o=-Ce.dot(Ee),l=Ce.lengthSq(),c=Math.abs(1-s*s);let h,u,d,p;if(c>0)if(h=s*o-a,u=s*a-o,p=r*c,h>=0)if(u>=-p)if(u<=p){const t=1/c;h*=t,u*=t,d=h*(h+s*u+2*a)+u*(s*h+u+2*o)+l}else u=r,h=Math.max(0,-(s*u+a)),d=-h*h+u*(u+2*o)+l;else u=-r,h=Math.max(0,-(s*u+a)),d=-h*h+u*(u+2*o)+l;else u<=-p?(h=Math.max(0,-(-s*r+a)),u=h>0?-r:Math.min(Math.max(-r,-o),r),d=-h*h+u*(u+2*o)+l):u<=p?(h=0,u=Math.min(Math.max(-r,-o),r),d=u*(u+2*o)+l):(h=Math.max(0,-(s*r+a)),u=h>0?r:Math.min(Math.max(-r,-o),r),d=-h*h+u*(u+2*o)+l);else u=s>0?-r:r,h=Math.max(0,-(s*u+a)),d=-h*h+u*(u+2*o)+l;return i&&i.copy(this.direction).multiplyScalar(h).add(this.origin),n&&n.copy(Ee).multiplyScalar(u).add(Ae),d}intersectSphere(t,e){Te.subVectors(t.center,this.origin);const i=Te.dot(this.direction),n=Te.dot(Te)-i*i,r=t.radius*t.radius;if(n>r)return null;const s=Math.sqrt(r-n),a=i-s,o=i+s;return a<0&&o<0?null:a<0?this.at(o,e):this.at(a,e)}intersectsSphere(t){return this.distanceSqToPoint(t.center)<=t.radius*t.radius}distanceToPlane(t){const e=t.normal.dot(this.direction);if(0===e)return 0===t.distanceToPoint(this.origin)?0:null;const i=-(this.origin.dot(t.normal)+t.constant)/e;return i>=0?i:null}intersectPlane(t,e){const i=this.distanceToPlane(t);return null===i?null:this.at(i,e)}intersectsPlane(t){const e=t.distanceToPoint(this.origin);if(0===e)return!0;return t.normal.dot(this.direction)*e<0}intersectBox(t,e){let i,n,r,s,a,o;const l=1/this.direction.x,c=1/this.direction.y,h=1/this.direction.z,u=this.origin;return l>=0?(i=(t.min.x-u.x)*l,n=(t.max.x-u.x)*l):(i=(t.max.x-u.x)*l,n=(t.min.x-u.x)*l),c>=0?(r=(t.min.y-u.y)*c,s=(t.max.y-u.y)*c):(r=(t.max.y-u.y)*c,s=(t.min.y-u.y)*c),i>s||r>n?null:((r>i||isNaN(i))&&(i=r),(s<n||isNaN(n))&&(n=s),h>=0?(a=(t.min.z-u.z)*h,o=(t.max.z-u.z)*h):(a=(t.max.z-u.z)*h,o=(t.min.z-u.z)*h),i>o||a>n?null:((a>i||i!=i)&&(i=a),(o<n||n!=n)&&(n=o),n<0?null:this.at(i>=0?i:n,e)))}intersectsBox(t){return null!==this.intersectBox(t,Te)}intersectTriangle(t,e,i,n,r){Le.subVectors(e,t),Re.subVectors(i,t),Pe.crossVectors(Le,Re);let s,a=this.direction.dot(Pe);if(a>0){if(n)return null;s=1}else{if(!(a<0))return null;s=-1,a=-a}Ce.subVectors(this.origin,t);const o=s*this.direction.dot(Re.crossVectors(Ce,Re));if(o<0)return null;const l=s*this.direction.dot(Le.cross(Ce));if(l<0)return null;if(o+l>a)return null;const c=-s*Ce.dot(Pe);return c<0?null:this.at(c/a,r)}applyMatrix4(t){return this.origin.applyMatrix4(t),this.direction.transformDirection(t),this}equals(t){return t.origin.equals(this.origin)&&t.direction.equals(this.direction)}clone(){return(new this.constructor).copy(this)}}class Ie{constructor(){Ie.prototype.isMatrix4=!0,this.elements=[1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1]}set(t,e,i,n,r,s,a,o,l,c,h,u,d,p,m,f){const g=this.elements;return g[0]=t,g[4]=e,g[8]=i,g[12]=n,g[1]=r,g[5]=s,g[9]=a,g[13]=o,g[2]=l,g[6]=c,g[10]=h,g[14]=u,g[3]=d,g[7]=p,g[11]=m,g[15]=f,this}identity(){return this.set(1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1),this}clone(){return(new Ie).fromArray(this.elements)}copy(t){const e=this.elements,i=t.elements;return e[0]=i[0],e[1]=i[1],e[2]=i[2],e[3]=i[3],e[4]=i[4],e[5]=i[5],e[6]=i[6],e[7]=i[7],e[8]=i[8],e[9]=i[9],e[10]=i[10],e[11]=i[11],e[12]=i[12],e[13]=i[13],e[14]=i[14],e[15]=i[15],this}copyPosition(t){const e=this.elements,i=t.elements;return e[12]=i[12],e[13]=i[13],e[14]=i[14],this}setFromMatrix3(t){const e=t.elements;return this.set(e[0],e[3],e[6],0,e[1],e[4],e[7],0,e[2],e[5],e[8],0,0,0,0,1),this}extractBasis(t,e,i){return t.setFromMatrixColumn(this,0),e.setFromMatrixColumn(this,1),i.setFromMatrixColumn(this,2),this}makeBasis(t,e,i){return this.set(t.x,e.x,i.x,0,t.y,e.y,i.y,0,t.z,e.z,i.z,0,0,0,0,1),this}extractRotation(t){const e=this.elements,i=t.elements,n=1/Ne.setFromMatrixColumn(t,0).length(),r=1/Ne.setFromMatrixColumn(t,1).length(),s=1/Ne.setFromMatrixColumn(t,2).length();return e[0]=i[0]*n,e[1]=i[1]*n,e[2]=i[2]*n,e[3]=0,e[4]=i[4]*r,e[5]=i[5]*r,e[6]=i[6]*r,e[7]=0,e[8]=i[8]*s,e[9]=i[9]*s,e[10]=i[10]*s,e[11]=0,e[12]=0,e[13]=0,e[14]=0,e[15]=1,this}makeRotationFromEuler(t){const e=this.elements,i=t.x,n=t.y,r=t.z,s=Math.cos(i),a=Math.sin(i),o=Math.cos(n),l=Math.sin(n),c=Math.cos(r),h=Math.sin(r);if("XYZ"===t.order){const t=s*c,i=s*h,n=a*c,r=a*h;e[0]=o*c,e[4]=-o*h,e[8]=l,e[1]=i+n*l,e[5]=t-r*l,e[9]=-a*o,e[2]=r-t*l,e[6]=n+i*l,e[10]=s*o}else if("YXZ"===t.order){const t=o*c,i=o*h,n=l*c,r=l*h;e[0]=t+r*a,e[4]=n*a-i,e[8]=s*l,e[1]=s*h,e[5]=s*c,e[9]=-a,e[2]=i*a-n,e[6]=r+t*a,e[10]=s*o}else if("ZXY"===t.order){const t=o*c,i=o*h,n=l*c,r=l*h;e[0]=t-r*a,e[4]=-s*h,e[8]=n+i*a,e[1]=i+n*a,e[5]=s*c,e[9]=r-t*a,e[2]=-s*l,e[6]=a,e[10]=s*o}else if("ZYX"===t.order){const t=s*c,i=s*h,n=a*c,r=a*h;e[0]=o*c,e[4]=n*l-i,e[8]=t*l+r,e[1]=o*h,e[5]=r*l+t,e[9]=i*l-n,e[2]=-l,e[6]=a*o,e[10]=s*o}else if("YZX"===t.order){const t=s*o,i=s*l,n=a*o,r=a*l;e[0]=o*c,e[4]=r-t*h,e[8]=n*h+i,e[1]=h,e[5]=s*c,e[9]=-a*c,e[2]=-l*c,e[6]=i*h+n,e[10]=t-r*h}else if("XZY"===t.order){const t=s*o,i=s*l,n=a*o,r=a*l;e[0]=o*c,e[4]=-h,e[8]=l*c,e[1]=t*h+r,e[5]=s*c,e[9]=i*h-n,e[2]=n*h-i,e[6]=a*c,e[10]=r*h+t}return e[3]=0,e[7]=0,e[11]=0,e[12]=0,e[13]=0,e[14]=0,e[15]=1,this}makeRotationFromQuaternion(t){return this.compose(ze,t,Ue)}lookAt(t,e,i){const n=this.elements;return ke.subVectors(t,e),0===ke.lengthSq()&&(ke.z=1),ke.normalize(),Be.crossVectors(i,ke),0===Be.lengthSq()&&(1===Math.abs(i.z)?ke.x+=1e-4:ke.z+=1e-4,ke.normalize(),Be.crossVectors(i,ke)),Be.normalize(),Fe.crossVectors(ke,Be),n[0]=Be.x,n[4]=Fe.x,n[8]=ke.x,n[1]=Be.y,n[5]=Fe.y,n[9]=ke.y,n[2]=Be.z,n[6]=Fe.z,n[10]=ke.z,this}multiply(t){return this.multiplyMatrices(this,t)}premultiply(t){return this.multiplyMatrices(t,this)}multiplyMatrices(t,e){const i=t.elements,n=e.elements,r=this.elements,s=i[0],a=i[4],o=i[8],l=i[12],c=i[1],h=i[5],u=i[9],d=i[13],p=i[2],m=i[6],f=i[10],g=i[14],v=i[3],x=i[7],_=i[11],y=i[15],M=n[0],b=n[4],S=n[8],w=n[12],T=n[1],A=n[5],E=n[9],C=n[13],L=n[2],R=n[6],P=n[10],D=n[14],I=n[3],N=n[7],O=n[11],z=n[15];return r[0]=s*M+a*T+o*L+l*I,r[4]=s*b+a*A+o*R+l*N,r[8]=s*S+a*E+o*P+l*O,r[12]=s*w+a*C+o*D+l*z,r[1]=c*M+h*T+u*L+d*I,r[5]=c*b+h*A+u*R+d*N,r[9]=c*S+h*E+u*P+d*O,r[13]=c*w+h*C+u*D+d*z,r[2]=p*M+m*T+f*L+g*I,r[6]=p*b+m*A+f*R+g*N,r[10]=p*S+m*E+f*P+g*O,r[14]=p*w+m*C+f*D+g*z,r[3]=v*M+x*T+_*L+y*I,r[7]=v*b+x*A+_*R+y*N,r[11]=v*S+x*E+_*P+y*O,r[15]=v*w+x*C+_*D+y*z,this}multiplyScalar(t){const e=this.elements;return e[0]*=t,e[4]*=t,e[8]*=t,e[12]*=t,e[1]*=t,e[5]*=t,e[9]*=t,e[13]*=t,e[2]*=t,e[6]*=t,e[10]*=t,e[14]*=t,e[3]*=t,e[7]*=t,e[11]*=t,e[15]*=t,this}determinant(){const t=this.elements,e=t[0],i=t[4],n=t[8],r=t[12],s=t[1],a=t[5],o=t[9],l=t[13],c=t[2],h=t[6],u=t[10],d=t[14];return t[3]*(+r*o*h-n*l*h-r*a*u+i*l*u+n*a*d-i*o*d)+t[7]*(+e*o*d-e*l*u+r*s*u-n*s*d+n*l*c-r*o*c)+t[11]*(+e*l*h-e*a*d-r*s*h+i*s*d+r*a*c-i*l*c)+t[15]*(-n*a*c-e*o*h+e*a*u+n*s*h-i*s*u+i*o*c)}transpose(){const t=this.elements;let e;return e=t[1],t[1]=t[4],t[4]=e,e=t[2],t[2]=t[8],t[8]=e,e=t[6],t[6]=t[9],t[9]=e,e=t[3],t[3]=t[12],t[12]=e,e=t[7],t[7]=t[13],t[13]=e,e=t[11],t[11]=t[14],t[14]=e,this}setPosition(t,e,i){const n=this.elements;return t.isVector3?(n[12]=t.x,n[13]=t.y,n[14]=t.z):(n[12]=t,n[13]=e,n[14]=i),this}invert(){const t=this.elements,e=t[0],i=t[1],n=t[2],r=t[3],s=t[4],a=t[5],o=t[6],l=t[7],c=t[8],h=t[9],u=t[10],d=t[11],p=t[12],m=t[13],f=t[14],g=t[15],v=h*f*l-m*u*l+m*o*d-a*f*d-h*o*g+a*u*g,x=p*u*l-c*f*l-p*o*d+s*f*d+c*o*g-s*u*g,_=c*m*l-p*h*l+p*a*d-s*m*d-c*a*g+s*h*g,y=p*h*o-c*m*o-p*a*u+s*m*u+c*a*f-s*h*f,M=e*v+i*x+n*_+r*y;if(0===M)return this.set(0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0);const b=1/M;return t[0]=v*b,t[1]=(m*u*r-h*f*r-m*n*d+i*f*d+h*n*g-i*u*g)*b,t[2]=(a*f*r-m*o*r+m*n*l-i*f*l-a*n*g+i*o*g)*b,t[3]=(h*o*r-a*u*r-h*n*l+i*u*l+a*n*d-i*o*d)*b,t[4]=x*b,t[5]=(c*f*r-p*u*r+p*n*d-e*f*d-c*n*g+e*u*g)*b,t[6]=(p*o*r-s*f*r-p*n*l+e*f*l+s*n*g-e*o*g)*b,t[7]=(s*u*r-c*o*r+c*n*l-e*u*l-s*n*d+e*o*d)*b,t[8]=_*b,t[9]=(p*h*r-c*m*r-p*i*d+e*m*d+c*i*g-e*h*g)*b,t[10]=(s*m*r-p*a*r+p*i*l-e*m*l-s*i*g+e*a*g)*b,t[11]=(c*a*r-s*h*r-c*i*l+e*h*l+s*i*d-e*a*d)*b,t[12]=y*b,t[13]=(c*m*n-p*h*n+p*i*u-e*m*u-c*i*f+e*h*f)*b,t[14]=(p*a*n-s*m*n-p*i*o+e*m*o+s*i*f-e*a*f)*b,t[15]=(s*h*n-c*a*n+c*i*o-e*h*o-s*i*u+e*a*u)*b,this}scale(t){const e=this.elements,i=t.x,n=t.y,r=t.z;return e[0]*=i,e[4]*=n,e[8]*=r,e[1]*=i,e[5]*=n,e[9]*=r,e[2]*=i,e[6]*=n,e[10]*=r,e[3]*=i,e[7]*=n,e[11]*=r,this}getMaxScaleOnAxis(){const t=this.elements,e=t[0]*t[0]+t[1]*t[1]+t[2]*t[2],i=t[4]*t[4]+t[5]*t[5]+t[6]*t[6],n=t[8]*t[8]+t[9]*t[9]+t[10]*t[10];return Math.sqrt(Math.max(e,i,n))}makeTranslation(t,e,i){return this.set(1,0,0,t,0,1,0,e,0,0,1,i,0,0,0,1),this}makeRotationX(t){const e=Math.cos(t),i=Math.sin(t);return this.set(1,0,0,0,0,e,-i,0,0,i,e,0,0,0,0,1),this}makeRotationY(t){const e=Math.cos(t),i=Math.sin(t);return this.set(e,0,i,0,0,1,0,0,-i,0,e,0,0,0,0,1),this}makeRotationZ(t){const e=Math.cos(t),i=Math.sin(t);return this.set(e,-i,0,0,i,e,0,0,0,0,1,0,0,0,0,1),this}makeRotationAxis(t,e){const i=Math.cos(e),n=Math.sin(e),r=1-i,s=t.x,a=t.y,o=t.z,l=r*s,c=r*a;return this.set(l*s+i,l*a-n*o,l*o+n*a,0,l*a+n*o,c*a+i,c*o-n*s,0,l*o-n*a,c*o+n*s,r*o*o+i,0,0,0,0,1),this}makeScale(t,e,i){return this.set(t,0,0,0,0,e,0,0,0,0,i,0,0,0,0,1),this}makeShear(t,e,i,n,r,s){return this.set(1,i,r,0,t,1,s,0,e,n,1,0,0,0,0,1),this}compose(t,e,i){const n=this.elements,r=e._x,s=e._y,a=e._z,o=e._w,l=r+r,c=s+s,h=a+a,u=r*l,d=r*c,p=r*h,m=s*c,f=s*h,g=a*h,v=o*l,x=o*c,_=o*h,y=i.x,M=i.y,b=i.z;return n[0]=(1-(m+g))*y,n[1]=(d+_)*y,n[2]=(p-x)*y,n[3]=0,n[4]=(d-_)*M,n[5]=(1-(u+g))*M,n[6]=(f+v)*M,n[7]=0,n[8]=(p+x)*b,n[9]=(f-v)*b,n[10]=(1-(u+m))*b,n[11]=0,n[12]=t.x,n[13]=t.y,n[14]=t.z,n[15]=1,this}decompose(t,e,i){const n=this.elements;let r=Ne.set(n[0],n[1],n[2]).length();const s=Ne.set(n[4],n[5],n[6]).length(),a=Ne.set(n[8],n[9],n[10]).length();this.determinant()<0&&(r=-r),t.x=n[12],t.y=n[13],t.z=n[14],Oe.copy(this);const o=1/r,l=1/s,c=1/a;return Oe.elements[0]*=o,Oe.elements[1]*=o,Oe.elements[2]*=o,Oe.elements[4]*=l,Oe.elements[5]*=l,Oe.elements[6]*=l,Oe.elements[8]*=c,Oe.elements[9]*=c,Oe.elements[10]*=c,e.setFromRotationMatrix(Oe),i.x=r,i.y=s,i.z=a,this}makePerspective(t,e,i,n,r,s){const a=this.elements,o=2*r/(e-t),l=2*r/(i-n),c=(e+t)/(e-t),h=(i+n)/(i-n),u=-(s+r)/(s-r),d=-2*s*r/(s-r);return a[0]=o,a[4]=0,a[8]=c,a[12]=0,a[1]=0,a[5]=l,a[9]=h,a[13]=0,a[2]=0,a[6]=0,a[10]=u,a[14]=d,a[3]=0,a[7]=0,a[11]=-1,a[15]=0,this}makeOrthographic(t,e,i,n,r,s){const a=this.elements,o=1/(e-t),l=1/(i-n),c=1/(s-r),h=(e+t)*o,u=(i+n)*l,d=(s+r)*c;return a[0]=2*o,a[4]=0,a[8]=0,a[12]=-h,a[1]=0,a[5]=2*l,a[9]=0,a[13]=-u,a[2]=0,a[6]=0,a[10]=-2*c,a[14]=-d,a[3]=0,a[7]=0,a[11]=0,a[15]=1,this}equals(t){const e=this.elements,i=t.elements;for(let t=0;t<16;t++)if(e[t]!==i[t])return!1;return!0}fromArray(t,e=0){for(let i=0;i<16;i++)this.elements[i]=t[i+e];return this}toArray(t=[],e=0){const i=this.elements;return t[e]=i[0],t[e+1]=i[1],t[e+2]=i[2],t[e+3]=i[3],t[e+4]=i[4],t[e+5]=i[5],t[e+6]=i[6],t[e+7]=i[7],t[e+8]=i[8],t[e+9]=i[9],t[e+10]=i[10],t[e+11]=i[11],t[e+12]=i[12],t[e+13]=i[13],t[e+14]=i[14],t[e+15]=i[15],t}}const Ne=new ne,Oe=new Ie,ze=new ne(0,0,0),Ue=new ne(1,1,1),Be=new ne,Fe=new ne,ke=new ne,Ge=new Ie,Ve=new ie;class He{constructor(t=0,e=0,i=0,n=He.DefaultOrder){this.isEuler=!0,this._x=t,this._y=e,this._z=i,this._order=n}get x(){return this._x}set x(t){this._x=t,this._onChangeCallback()}get y(){return this._y}set y(t){this._y=t,this._onChangeCallback()}get z(){return this._z}set z(t){this._z=t,this._onChangeCallback()}get order(){return this._order}set order(t){this._order=t,this._onChangeCallback()}set(t,e,i,n=this._order){return this._x=t,this._y=e,this._z=i,this._order=n,this._onChangeCallback(),this}clone(){return new this.constructor(this._x,this._y,this._z,this._order)}copy(t){return this._x=t._x,this._y=t._y,this._z=t._z,this._order=t._order,this._onChangeCallback(),this}setFromRotationMatrix(t,e=this._order,i=!0){const n=t.elements,r=n[0],s=n[4],a=n[8],o=n[1],l=n[5],c=n[9],h=n[2],u=n[6],d=n[10];switch(e){case"XYZ":this._y=Math.asin(yt(a,-1,1)),Math.abs(a)<.9999999?(this._x=Math.atan2(-c,d),this._z=Math.atan2(-s,r)):(this._x=Math.atan2(u,l),this._z=0);break;case"YXZ":this._x=Math.asin(-yt(c,-1,1)),Math.abs(c)<.9999999?(this._y=Math.atan2(a,d),this._z=Math.atan2(o,l)):(this._y=Math.atan2(-h,r),this._z=0);break;case"ZXY":this._x=Math.asin(yt(u,-1,1)),Math.abs(u)<.9999999?(this._y=Math.atan2(-h,d),this._z=Math.atan2(-s,l)):(this._y=0,this._z=Math.atan2(o,r));break;case"ZYX":this._y=Math.asin(-yt(h,-1,1)),Math.abs(h)<.9999999?(this._x=Math.atan2(u,d),this._z=Math.atan2(o,r)):(this._x=0,this._z=Math.atan2(-s,l));break;case"YZX":this._z=Math.asin(yt(o,-1,1)),Math.abs(o)<.9999999?(this._x=Math.atan2(-c,l),this._y=Math.atan2(-h,r)):(this._x=0,this._y=Math.atan2(a,d));break;case"XZY":this._z=Math.asin(-yt(s,-1,1)),Math.abs(s)<.9999999?(this._x=Math.atan2(u,l),this._y=Math.atan2(a,r)):(this._x=Math.atan2(-c,d),this._y=0);break;default:console.warn("THREE.Euler: .setFromRotationMatrix() encountered an unknown order: "+e)}return this._order=e,!0===i&&this._onChangeCallback(),this}setFromQuaternion(t,e,i){return Ge.makeRotationFromQuaternion(t),this.setFromRotationMatrix(Ge,e,i)}setFromVector3(t,e=this._order){return this.set(t.x,t.y,t.z,e)}reorder(t){return Ve.setFromEuler(this),this.setFromQuaternion(Ve,t)}equals(t){return t._x===this._x&&t._y===this._y&&t._z===this._z&&t._order===this._order}fromArray(t){return this._x=t[0],this._y=t[1],this._z=t[2],void 0!==t[3]&&(this._order=t[3]),this._onChangeCallback(),this}toArray(t=[],e=0){return t[e]=this._x,t[e+1]=this._y,t[e+2]=this._z,t[e+3]=this._order,t}_onChange(t){return this._onChangeCallback=t,this}_onChangeCallback(){}*[Symbol.iterator](){yield this._x,yield this._y,yield this._z,yield this._order}toVector3(){console.error("THREE.Euler: .toVector3() has been removed. Use Vector3.setFromEuler() instead")}}He.DefaultOrder="XYZ",He.RotationOrders=["XYZ","YZX","ZXY","XZY","YXZ","ZYX"];class We{constructor(){this.mask=1}set(t){this.mask=(1<<t|0)>>>0}enable(t){this.mask|=1<<t|0}enableAll(){this.mask=-1}toggle(t){this.mask^=1<<t|0}disable(t){this.mask&=~(1<<t|0)}disableAll(){this.mask=0}test(t){return 0!=(this.mask&t.mask)}isEnabled(t){return 0!=(this.mask&(1<<t|0))}}let je=0;const qe=new ne,Xe=new ie,Ye=new Ie,Ze=new ne,Je=new ne,Ke=new ne,$e=new ie,Qe=new ne(1,0,0),ti=new ne(0,1,0),ei=new ne(0,0,1),ii={type:"added"},ni={type:"removed"};class ri extends mt{constructor(){super(),this.isObject3D=!0,Object.defineProperty(this,"id",{value:je++}),this.uuid=_t(),this.name="",this.type="Object3D",this.parent=null,this.children=[],this.up=ri.DefaultUp.clone();const t=new ne,e=new He,i=new ie,n=new ne(1,1,1);e._onChange((function(){i.setFromEuler(e,!1)})),i._onChange((function(){e.setFromQuaternion(i,void 0,!1)})),Object.defineProperties(this,{position:{configurable:!0,enumerable:!0,value:t},rotation:{configurable:!0,enumerable:!0,value:e},quaternion:{configurable:!0,enumerable:!0,value:i},scale:{configurable:!0,enumerable:!0,value:n},modelViewMatrix:{value:new Ie},normalMatrix:{value:new Rt}}),this.matrix=new Ie,this.matrixWorld=new Ie,this.matrixAutoUpdate=ri.DefaultMatrixAutoUpdate,this.matrixWorldNeedsUpdate=!1,this.matrixWorldAutoUpdate=ri.DefaultMatrixWorldAutoUpdate,this.layers=new We,this.visible=!0,this.castShadow=!1,this.receiveShadow=!1,this.frustumCulled=!0,this.renderOrder=0,this.animations=[],this.userData={}}onBeforeRender(){}onAfterRender(){}applyMatrix4(t){this.matrixAutoUpdate&&this.updateMatrix(),this.matrix.premultiply(t),this.matrix.decompose(this.position,this.quaternion,this.scale)}applyQuaternion(t){return this.quaternion.premultiply(t),this}setRotationFromAxisAngle(t,e){this.quaternion.setFromAxisAngle(t,e)}setRotationFromEuler(t){this.quaternion.setFromEuler(t,!0)}setRotationFromMatrix(t){this.quaternion.setFromRotationMatrix(t)}setRotationFromQuaternion(t){this.quaternion.copy(t)}rotateOnAxis(t,e){return Xe.setFromAxisAngle(t,e),this.quaternion.multiply(Xe),this}rotateOnWorldAxis(t,e){return Xe.setFromAxisAngle(t,e),this.quaternion.premultiply(Xe),this}rotateX(t){return this.rotateOnAxis(Qe,t)}rotateY(t){return this.rotateOnAxis(ti,t)}rotateZ(t){return this.rotateOnAxis(ei,t)}translateOnAxis(t,e){return qe.copy(t).applyQuaternion(this.quaternion),this.position.add(qe.multiplyScalar(e)),this}translateX(t){return this.translateOnAxis(Qe,t)}translateY(t){return this.translateOnAxis(ti,t)}translateZ(t){return this.translateOnAxis(ei,t)}localToWorld(t){return t.applyMatrix4(this.matrixWorld)}worldToLocal(t){return t.applyMatrix4(Ye.copy(this.matrixWorld).invert())}lookAt(t,e,i){t.isVector3?Ze.copy(t):Ze.set(t,e,i);const n=this.parent;this.updateWorldMatrix(!0,!1),Je.setFromMatrixPosition(this.matrixWorld),this.isCamera||this.isLight?Ye.lookAt(Je,Ze,this.up):Ye.lookAt(Ze,Je,this.up),this.quaternion.setFromRotationMatrix(Ye),n&&(Ye.extractRotation(n.matrixWorld),Xe.setFromRotationMatrix(Ye),this.quaternion.premultiply(Xe.invert()))}add(t){if(arguments.length>1){for(let t=0;t<arguments.length;t++)this.add(arguments[t]);return this}return t===this?(console.error("THREE.Object3D.add: object can't be added as a child of itself.",t),this):(t&&t.isObject3D?(null!==t.parent&&t.parent.remove(t),t.parent=this,this.children.push(t),t.dispatchEvent(ii)):console.error("THREE.Object3D.add: object not an instance of THREE.Object3D.",t),this)}remove(t){if(arguments.length>1){for(let t=0;t<arguments.length;t++)this.remove(arguments[t]);return this}const e=this.children.indexOf(t);return-1!==e&&(t.parent=null,this.children.splice(e,1),t.dispatchEvent(ni)),this}removeFromParent(){const t=this.parent;return null!==t&&t.remove(this),this}clear(){for(let t=0;t<this.children.length;t++){const e=this.children[t];e.parent=null,e.dispatchEvent(ni)}return this.children.length=0,this}attach(t){return this.updateWorldMatrix(!0,!1),Ye.copy(this.matrixWorld).invert(),null!==t.parent&&(t.parent.updateWorldMatrix(!0,!1),Ye.multiply(t.parent.matrixWorld)),t.applyMatrix4(Ye),this.add(t),t.updateWorldMatrix(!1,!0),this}getObjectById(t){return this.getObjectByProperty("id",t)}getObjectByName(t){return this.getObjectByProperty("name",t)}getObjectByProperty(t,e){if(this[t]===e)return this;for(let i=0,n=this.children.length;i<n;i++){const n=this.children[i].getObjectByProperty(t,e);if(void 0!==n)return n}}getWorldPosition(t){return this.updateWorldMatrix(!0,!1),t.setFromMatrixPosition(this.matrixWorld)}getWorldQuaternion(t){return this.updateWorldMatrix(!0,!1),this.matrixWorld.decompose(Je,t,Ke),t}getWorldScale(t){return this.updateWorldMatrix(!0,!1),this.matrixWorld.decompose(Je,$e,t),t}getWorldDirection(t){this.updateWorldMatrix(!0,!1);const e=this.matrixWorld.elements;return t.set(e[8],e[9],e[10]).normalize()}raycast(){}traverse(t){t(this);const e=this.children;for(let i=0,n=e.length;i<n;i++)e[i].traverse(t)}traverseVisible(t){if(!1===this.visible)return;t(this);const e=this.children;for(let i=0,n=e.length;i<n;i++)e[i].traverseVisible(t)}traverseAncestors(t){const e=this.parent;null!==e&&(t(e),e.traverseAncestors(t))}updateMatrix(){this.matrix.compose(this.position,this.quaternion,this.scale),this.matrixWorldNeedsUpdate=!0}updateMatrixWorld(t){this.matrixAutoUpdate&&this.updateMatrix(),(this.matrixWorldNeedsUpdate||t)&&(null===this.parent?this.matrixWorld.copy(this.matrix):this.matrixWorld.multiplyMatrices(this.parent.matrixWorld,this.matrix),this.matrixWorldNeedsUpdate=!1,t=!0);const e=this.children;for(let i=0,n=e.length;i<n;i++){const n=e[i];!0!==n.matrixWorldAutoUpdate&&!0!==t||n.updateMatrixWorld(t)}}updateWorldMatrix(t,e){const i=this.parent;if(!0===t&&null!==i&&!0===i.matrixWorldAutoUpdate&&i.updateWorldMatrix(!0,!1),this.matrixAutoUpdate&&this.updateMatrix(),null===this.parent?this.matrixWorld.copy(this.matrix):this.matrixWorld.multiplyMatrices(this.parent.matrixWorld,this.matrix),!0===e){const t=this.children;for(let e=0,i=t.length;e<i;e++){const i=t[e];!0===i.matrixWorldAutoUpdate&&i.updateWorldMatrix(!1,!0)}}}toJSON(t){const e=void 0===t||"string"==typeof t,i={};e&&(t={geometries:{},materials:{},textures:{},images:{},shapes:{},skeletons:{},animations:{},nodes:{}},i.metadata={version:4.5,type:"Object",generator:"Object3D.toJSON"});const n={};function r(e,i){return void 0===e[i.uuid]&&(e[i.uuid]=i.toJSON(t)),i.uuid}if(n.uuid=this.uuid,n.type=this.type,""!==this.name&&(n.name=this.name),!0===this.castShadow&&(n.castShadow=!0),!0===this.receiveShadow&&(n.receiveShadow=!0),!1===this.visible&&(n.visible=!1),!1===this.frustumCulled&&(n.frustumCulled=!1),0!==this.renderOrder&&(n.renderOrder=this.renderOrder),"{}"!==JSON.stringify(this.userData)&&(n.userData=this.userData),n.layers=this.layers.mask,n.matrix=this.matrix.toArray(),!1===this.matrixAutoUpdate&&(n.matrixAutoUpdate=!1),this.isInstancedMesh&&(n.type="InstancedMesh",n.count=this.count,n.instanceMatrix=this.instanceMatrix.toJSON(),null!==this.instanceColor&&(n.instanceColor=this.instanceColor.toJSON())),this.isScene)this.background&&(this.background.isColor?n.background=this.background.toJSON():this.background.isTexture&&(n.background=this.background.toJSON(t).uuid)),this.environment&&this.environment.isTexture&&!0!==this.environment.isRenderTargetTexture&&(n.environment=this.environment.toJSON(t).uuid);else if(this.isMesh||this.isLine||this.isPoints){n.geometry=r(t.geometries,this.geometry);const e=this.geometry.parameters;if(void 0!==e&&void 0!==e.shapes){const i=e.shapes;if(Array.isArray(i))for(let e=0,n=i.length;e<n;e++){const n=i[e];r(t.shapes,n)}else r(t.shapes,i)}}if(this.isSkinnedMesh&&(n.bindMode=this.bindMode,n.bindMatrix=this.bindMatrix.toArray(),void 0!==this.skeleton&&(r(t.skeletons,this.skeleton),n.skeleton=this.skeleton.uuid)),void 0!==this.material)if(Array.isArray(this.material)){const e=[];for(let i=0,n=this.material.length;i<n;i++)e.push(r(t.materials,this.material[i]));n.material=e}else n.material=r(t.materials,this.material);if(this.children.length>0){n.children=[];for(let e=0;e<this.children.length;e++)n.children.push(this.children[e].toJSON(t).object)}if(this.animations.length>0){n.animations=[];for(let e=0;e<this.animations.length;e++){const i=this.animations[e];n.animations.push(r(t.animations,i))}}if(e){const e=s(t.geometries),n=s(t.materials),r=s(t.textures),a=s(t.images),o=s(t.shapes),l=s(t.skeletons),c=s(t.animations),h=s(t.nodes);e.length>0&&(i.geometries=e),n.length>0&&(i.materials=n),r.length>0&&(i.textures=r),a.length>0&&(i.images=a),o.length>0&&(i.shapes=o),l.length>0&&(i.skeletons=l),c.length>0&&(i.animations=c),h.length>0&&(i.nodes=h)}return i.object=n,i;function s(t){const e=[];for(const i in t){const n=t[i];delete n.metadata,e.push(n)}return e}}clone(t){return(new this.constructor).copy(this,t)}copy(t,e=!0){if(this.name=t.name,this.up.copy(t.up),this.position.copy(t.position),this.rotation.order=t.rotation.order,this.quaternion.copy(t.quaternion),this.scale.copy(t.scale),this.matrix.copy(t.matrix),this.matrixWorld.copy(t.matrixWorld),this.matrixAutoUpdate=t.matrixAutoUpdate,this.matrixWorldNeedsUpdate=t.matrixWorldNeedsUpdate,this.matrixWorldAutoUpdate=t.matrixWorldAutoUpdate,this.layers.mask=t.layers.mask,this.visible=t.visible,this.castShadow=t.castShadow,this.receiveShadow=t.receiveShadow,this.frustumCulled=t.frustumCulled,this.renderOrder=t.renderOrder,this.userData=JSON.parse(JSON.stringify(t.userData)),!0===e)for(let e=0;e<t.children.length;e++){const i=t.children[e];this.add(i.clone())}return this}}ri.DefaultUp=new ne(0,1,0),ri.DefaultMatrixAutoUpdate=!0,ri.DefaultMatrixWorldAutoUpdate=!0;const si=new ne,ai=new ne,oi=new ne,li=new ne,ci=new ne,hi=new ne,ui=new ne,di=new ne,pi=new ne,mi=new ne;class fi{constructor(t=new ne,e=new ne,i=new ne){this.a=t,this.b=e,this.c=i}static getNormal(t,e,i,n){n.subVectors(i,e),si.subVectors(t,e),n.cross(si);const r=n.lengthSq();return r>0?n.multiplyScalar(1/Math.sqrt(r)):n.set(0,0,0)}static getBarycoord(t,e,i,n,r){si.subVectors(n,e),ai.subVectors(i,e),oi.subVectors(t,e);const s=si.dot(si),a=si.dot(ai),o=si.dot(oi),l=ai.dot(ai),c=ai.dot(oi),h=s*l-a*a;if(0===h)return r.set(-2,-1,-1);const u=1/h,d=(l*o-a*c)*u,p=(s*c-a*o)*u;return r.set(1-d-p,p,d)}static containsPoint(t,e,i,n){return this.getBarycoord(t,e,i,n,li),li.x>=0&&li.y>=0&&li.x+li.y<=1}static getUV(t,e,i,n,r,s,a,o){return this.getBarycoord(t,e,i,n,li),o.set(0,0),o.addScaledVector(r,li.x),o.addScaledVector(s,li.y),o.addScaledVector(a,li.z),o}static isFrontFacing(t,e,i,n){return si.subVectors(i,e),ai.subVectors(t,e),si.cross(ai).dot(n)<0}set(t,e,i){return this.a.copy(t),this.b.copy(e),this.c.copy(i),this}setFromPointsAndIndices(t,e,i,n){return this.a.copy(t[e]),this.b.copy(t[i]),this.c.copy(t[n]),this}setFromAttributeAndIndices(t,e,i,n){return this.a.fromBufferAttribute(t,e),this.b.fromBufferAttribute(t,i),this.c.fromBufferAttribute(t,n),this}clone(){return(new this.constructor).copy(this)}copy(t){return this.a.copy(t.a),this.b.copy(t.b),this.c.copy(t.c),this}getArea(){return si.subVectors(this.c,this.b),ai.subVectors(this.a,this.b),.5*si.cross(ai).length()}getMidpoint(t){return t.addVectors(this.a,this.b).add(this.c).multiplyScalar(1/3)}getNormal(t){return fi.getNormal(this.a,this.b,this.c,t)}getPlane(t){return t.setFromCoplanarPoints(this.a,this.b,this.c)}getBarycoord(t,e){return fi.getBarycoord(t,this.a,this.b,this.c,e)}getUV(t,e,i,n,r){return fi.getUV(t,this.a,this.b,this.c,e,i,n,r)}containsPoint(t){return fi.containsPoint(t,this.a,this.b,this.c)}isFrontFacing(t){return fi.isFrontFacing(this.a,this.b,this.c,t)}intersectsBox(t){return t.intersectsTriangle(this)}closestPointToPoint(t,e){const i=this.a,n=this.b,r=this.c;let s,a;ci.subVectors(n,i),hi.subVectors(r,i),di.subVectors(t,i);const o=ci.dot(di),l=hi.dot(di);if(o<=0&&l<=0)return e.copy(i);pi.subVectors(t,n);const c=ci.dot(pi),h=hi.dot(pi);if(c>=0&&h<=c)return e.copy(n);const u=o*h-c*l;if(u<=0&&o>=0&&c<=0)return s=o/(o-c),e.copy(i).addScaledVector(ci,s);mi.subVectors(t,r);const d=ci.dot(mi),p=hi.dot(mi);if(p>=0&&d<=p)return e.copy(r);const m=d*l-o*p;if(m<=0&&l>=0&&p<=0)return a=l/(l-p),e.copy(i).addScaledVector(hi,a);const f=c*p-d*h;if(f<=0&&h-c>=0&&d-p>=0)return ui.subVectors(r,n),a=(h-c)/(h-c+(d-p)),e.copy(n).addScaledVector(ui,a);const g=1/(f+m+u);return s=m*g,a=u*g,e.copy(i).addScaledVector(ci,s).addScaledVector(hi,a)}equals(t){return t.a.equals(this.a)&&t.b.equals(this.b)&&t.c.equals(this.c)}}let gi=0;class vi extends mt{constructor(){super(),this.isMaterial=!0,Object.defineProperty(this,"id",{value:gi++}),this.uuid=_t(),this.name="",this.type="Material",this.blending=1,this.side=0,this.vertexColors=!1,this.opacity=1,this.transparent=!1,this.blendSrc=204,this.blendDst=205,this.blendEquation=i,this.blendSrcAlpha=null,this.blendDstAlpha=null,this.blendEquationAlpha=null,this.depthFunc=3,this.depthTest=!0,this.depthWrite=!0,this.stencilWriteMask=255,this.stencilFunc=519,this.stencilRef=0,this.stencilFuncMask=255,this.stencilFail=ht,this.stencilZFail=ht,this.stencilZPass=ht,this.stencilWrite=!1,this.clippingPlanes=null,this.clipIntersection=!1,this.clipShadows=!1,this.shadowSide=null,this.colorWrite=!0,this.precision=null,this.polygonOffset=!1,this.polygonOffsetFactor=0,this.polygonOffsetUnits=0,this.dithering=!1,this.alphaToCoverage=!1,this.premultipliedAlpha=!1,this.visible=!0,this.toneMapped=!0,this.userData={},this.version=0,this._alphaTest=0}get alphaTest(){return this._alphaTest}set alphaTest(t){this._alphaTest>0!=t>0&&this.version++,this._alphaTest=t}onBuild(){}onBeforeRender(){}onBeforeCompile(){}customProgramCacheKey(){return this.onBeforeCompile.toString()}setValues(t){if(void 0!==t)for(const e in t){const i=t[e];if(void 0===i){console.warn("THREE.Material: '"+e+"' parameter is undefined.");continue}const n=this[e];void 0!==n?n&&n.isColor?n.set(i):n&&n.isVector3&&i&&i.isVector3?n.copy(i):this[e]=i:console.warn("THREE."+this.type+": '"+e+"' is not a property of this material.")}}toJSON(t){const e=void 0===t||"string"==typeof t;e&&(t={textures:{},images:{}});const i={metadata:{version:4.5,type:"Material",generator:"Material.toJSON"}};function n(t){const e=[];for(const i in t){const n=t[i];delete n.metadata,e.push(n)}return e}if(i.uuid=this.uuid,i.type=this.type,""!==this.name&&(i.name=this.name),this.color&&this.color.isColor&&(i.color=this.color.getHex()),void 0!==this.roughness&&(i.roughness=this.roughness),void 0!==this.metalness&&(i.metalness=this.metalness),void 0!==this.sheen&&(i.sheen=this.sheen),this.sheenColor&&this.sheenColor.isColor&&(i.sheenColor=this.sheenColor.getHex()),void 0!==this.sheenRoughness&&(i.sheenRoughness=this.sheenRoughness),this.emissive&&this.emissive.isColor&&(i.emissive=this.emissive.getHex()),this.emissiveIntensity&&1!==this.emissiveIntensity&&(i.emissiveIntensity=this.emissiveIntensity),this.specular&&this.specular.isColor&&(i.specular=this.specular.getHex()),void 0!==this.specularIntensity&&(i.specularIntensity=this.specularIntensity),this.specularColor&&this.specularColor.isColor&&(i.specularColor=this.specularColor.getHex()),void 0!==this.shininess&&(i.shininess=this.shininess),void 0!==this.clearcoat&&(i.clearcoat=this.clearcoat),void 0!==this.clearcoatRoughness&&(i.clearcoatRoughness=this.clearcoatRoughness),this.clearcoatMap&&this.clearcoatMap.isTexture&&(i.clearcoatMap=this.clearcoatMap.toJSON(t).uuid),this.clearcoatRoughnessMap&&this.clearcoatRoughnessMap.isTexture&&(i.clearcoatRoughnessMap=this.clearcoatRoughnessMap.toJSON(t).uuid),this.clearcoatNormalMap&&this.clearcoatNormalMap.isTexture&&(i.clearcoatNormalMap=this.clearcoatNormalMap.toJSON(t).uuid,i.clearcoatNormalScale=this.clearcoatNormalScale.toArray()),void 0!==this.iridescence&&(i.iridescence=this.iridescence),void 0!==this.iridescenceIOR&&(i.iridescenceIOR=this.iridescenceIOR),void 0!==this.iridescenceThicknessRange&&(i.iridescenceThicknessRange=this.iridescenceThicknessRange),this.iridescenceMap&&this.iridescenceMap.isTexture&&(i.iridescenceMap=this.iridescenceMap.toJSON(t).uuid),this.iridescenceThicknessMap&&this.iridescenceThicknessMap.isTexture&&(i.iridescenceThicknessMap=this.iridescenceThicknessMap.toJSON(t).uuid),this.map&&this.map.isTexture&&(i.map=this.map.toJSON(t).uuid),this.matcap&&this.matcap.isTexture&&(i.matcap=this.matcap.toJSON(t).uuid),this.alphaMap&&this.alphaMap.isTexture&&(i.alphaMap=this.alphaMap.toJSON(t).uuid),this.lightMap&&this.lightMap.isTexture&&(i.lightMap=this.lightMap.toJSON(t).uuid,i.lightMapIntensity=this.lightMapIntensity),this.aoMap&&this.aoMap.isTexture&&(i.aoMap=this.aoMap.toJSON(t).uuid,i.aoMapIntensity=this.aoMapIntensity),this.bumpMap&&this.bumpMap.isTexture&&(i.bumpMap=this.bumpMap.toJSON(t).uuid,i.bumpScale=this.bumpScale),this.normalMap&&this.normalMap.isTexture&&(i.normalMap=this.normalMap.toJSON(t).uuid,i.normalMapType=this.normalMapType,i.normalScale=this.normalScale.toArray()),this.displacementMap&&this.displacementMap.isTexture&&(i.displacementMap=this.displacementMap.toJSON(t).uuid,i.displacementScale=this.displacementScale,i.displacementBias=this.displacementBias),this.roughnessMap&&this.roughnessMap.isTexture&&(i.roughnessMap=this.roughnessMap.toJSON(t).uuid),this.metalnessMap&&this.metalnessMap.isTexture&&(i.metalnessMap=this.metalnessMap.toJSON(t).uuid),this.emissiveMap&&this.emissiveMap.isTexture&&(i.emissiveMap=this.emissiveMap.toJSON(t).uuid),this.specularMap&&this.specularMap.isTexture&&(i.specularMap=this.specularMap.toJSON(t).uuid),this.specularIntensityMap&&this.specularIntensityMap.isTexture&&(i.specularIntensityMap=this.specularIntensityMap.toJSON(t).uuid),this.specularColorMap&&this.specularColorMap.isTexture&&(i.specularColorMap=this.specularColorMap.toJSON(t).uuid),this.envMap&&this.envMap.isTexture&&(i.envMap=this.envMap.toJSON(t).uuid,void 0!==this.combine&&(i.combine=this.combine)),void 0!==this.envMapIntensity&&(i.envMapIntensity=this.envMapIntensity),void 0!==this.reflectivity&&(i.reflectivity=this.reflectivity),void 0!==this.refractionRatio&&(i.refractionRatio=this.refractionRatio),this.gradientMap&&this.gradientMap.isTexture&&(i.gradientMap=this.gradientMap.toJSON(t).uuid),void 0!==this.transmission&&(i.transmission=this.transmission),this.transmissionMap&&this.transmissionMap.isTexture&&(i.transmissionMap=this.transmissionMap.toJSON(t).uuid),void 0!==this.thickness&&(i.thickness=this.thickness),this.thicknessMap&&this.thicknessMap.isTexture&&(i.thicknessMap=this.thicknessMap.toJSON(t).uuid),void 0!==this.attenuationDistance&&this.attenuationDistance!==1/0&&(i.attenuationDistance=this.attenuationDistance),void 0!==this.attenuationColor&&(i.attenuationColor=this.attenuationColor.getHex()),void 0!==this.size&&(i.size=this.size),null!==this.shadowSide&&(i.shadowSide=this.shadowSide),void 0!==this.sizeAttenuation&&(i.sizeAttenuation=this.sizeAttenuation),1!==this.blending&&(i.blending=this.blending),0!==this.side&&(i.side=this.side),this.vertexColors&&(i.vertexColors=!0),this.opacity<1&&(i.opacity=this.opacity),!0===this.transparent&&(i.transparent=this.transparent),i.depthFunc=this.depthFunc,i.depthTest=this.depthTest,i.depthWrite=this.depthWrite,i.colorWrite=this.colorWrite,i.stencilWrite=this.stencilWrite,i.stencilWriteMask=this.stencilWriteMask,i.stencilFunc=this.stencilFunc,i.stencilRef=this.stencilRef,i.stencilFuncMask=this.stencilFuncMask,i.stencilFail=this.stencilFail,i.stencilZFail=this.stencilZFail,i.stencilZPass=this.stencilZPass,void 0!==this.rotation&&0!==this.rotation&&(i.rotation=this.rotation),!0===this.polygonOffset&&(i.polygonOffset=!0),0!==this.polygonOffsetFactor&&(i.polygonOffsetFactor=this.polygonOffsetFactor),0!==this.polygonOffsetUnits&&(i.polygonOffsetUnits=this.polygonOffsetUnits),void 0!==this.linewidth&&1!==this.linewidth&&(i.linewidth=this.linewidth),void 0!==this.dashSize&&(i.dashSize=this.dashSize),void 0!==this.gapSize&&(i.gapSize=this.gapSize),void 0!==this.scale&&(i.scale=this.scale),!0===this.dithering&&(i.dithering=!0),this.alphaTest>0&&(i.alphaTest=this.alphaTest),!0===this.alphaToCoverage&&(i.alphaToCoverage=this.alphaToCoverage),!0===this.premultipliedAlpha&&(i.premultipliedAlpha=this.premultipliedAlpha),!0===this.wireframe&&(i.wireframe=this.wireframe),this.wireframeLinewidth>1&&(i.wireframeLinewidth=this.wireframeLinewidth),"round"!==this.wireframeLinecap&&(i.wireframeLinecap=this.wireframeLinecap),"round"!==this.wireframeLinejoin&&(i.wireframeLinejoin=this.wireframeLinejoin),!0===this.flatShading&&(i.flatShading=this.flatShading),!1===this.visible&&(i.visible=!1),!1===this.toneMapped&&(i.toneMapped=!1),!1===this.fog&&(i.fog=!1),"{}"!==JSON.stringify(this.userData)&&(i.userData=this.userData),e){const e=n(t.textures),r=n(t.images);e.length>0&&(i.textures=e),r.length>0&&(i.images=r)}return i}clone(){return(new this.constructor).copy(this)}copy(t){this.name=t.name,this.blending=t.blending,this.side=t.side,this.vertexColors=t.vertexColors,this.opacity=t.opacity,this.transparent=t.transparent,this.blendSrc=t.blendSrc,this.blendDst=t.blendDst,this.blendEquation=t.blendEquation,this.blendSrcAlpha=t.blendSrcAlpha,this.blendDstAlpha=t.blendDstAlpha,this.blendEquationAlpha=t.blendEquationAlpha,this.depthFunc=t.depthFunc,this.depthTest=t.depthTest,this.depthWrite=t.depthWrite,this.stencilWriteMask=t.stencilWriteMask,this.stencilFunc=t.stencilFunc,this.stencilRef=t.stencilRef,this.stencilFuncMask=t.stencilFuncMask,this.stencilFail=t.stencilFail,this.stencilZFail=t.stencilZFail,this.stencilZPass=t.stencilZPass,this.stencilWrite=t.stencilWrite;const e=t.clippingPlanes;let i=null;if(null!==e){const t=e.length;i=new Array(t);for(let n=0;n!==t;++n)i[n]=e[n].clone()}return this.clippingPlanes=i,this.clipIntersection=t.clipIntersection,this.clipShadows=t.clipShadows,this.shadowSide=t.shadowSide,this.colorWrite=t.colorWrite,this.precision=t.precision,this.polygonOffset=t.polygonOffset,this.polygonOffsetFactor=t.polygonOffsetFactor,this.polygonOffsetUnits=t.polygonOffsetUnits,this.dithering=t.dithering,this.alphaTest=t.alphaTest,this.alphaToCoverage=t.alphaToCoverage,this.premultipliedAlpha=t.premultipliedAlpha,this.visible=t.visible,this.toneMapped=t.toneMapped,this.userData=JSON.parse(JSON.stringify(t.userData)),this}dispose(){this.dispatchEvent({type:"dispose"})}set needsUpdate(t){!0===t&&this.version++}}class xi extends vi{constructor(t){super(),this.isMeshBasicMaterial=!0,this.type="MeshBasicMaterial",this.color=new jt(16777215),this.map=null,this.lightMap=null,this.lightMapIntensity=1,this.aoMap=null,this.aoMapIntensity=1,this.specularMap=null,this.alphaMap=null,this.envMap=null,this.combine=0,this.reflectivity=1,this.refractionRatio=.98,this.wireframe=!1,this.wireframeLinewidth=1,this.wireframeLinecap="round",this.wireframeLinejoin="round",this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.map=t.map,this.lightMap=t.lightMap,this.lightMapIntensity=t.lightMapIntensity,this.aoMap=t.aoMap,this.aoMapIntensity=t.aoMapIntensity,this.specularMap=t.specularMap,this.alphaMap=t.alphaMap,this.envMap=t.envMap,this.combine=t.combine,this.reflectivity=t.reflectivity,this.refractionRatio=t.refractionRatio,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.wireframeLinecap=t.wireframeLinecap,this.wireframeLinejoin=t.wireframeLinejoin,this.fog=t.fog,this}}const _i=new ne,yi=new Lt;class Mi{constructor(t,e,i){if(Array.isArray(t))throw new TypeError("THREE.BufferAttribute: array should be a Typed Array.");this.isBufferAttribute=!0,this.name="",this.array=t,this.itemSize=e,this.count=void 0!==t?t.length/e:0,this.normalized=!0===i,this.usage=ut,this.updateRange={offset:0,count:-1},this.version=0}onUploadCallback(){}set needsUpdate(t){!0===t&&this.version++}setUsage(t){return this.usage=t,this}copy(t){return this.name=t.name,this.array=new t.array.constructor(t.array),this.itemSize=t.itemSize,this.count=t.count,this.normalized=t.normalized,this.usage=t.usage,this}copyAt(t,e,i){t*=this.itemSize,i*=e.itemSize;for(let n=0,r=this.itemSize;n<r;n++)this.array[t+n]=e.array[i+n];return this}copyArray(t){return this.array.set(t),this}applyMatrix3(t){if(2===this.itemSize)for(let e=0,i=this.count;e<i;e++)yi.fromBufferAttribute(this,e),yi.applyMatrix3(t),this.setXY(e,yi.x,yi.y);else if(3===this.itemSize)for(let e=0,i=this.count;e<i;e++)_i.fromBufferAttribute(this,e),_i.applyMatrix3(t),this.setXYZ(e,_i.x,_i.y,_i.z);return this}applyMatrix4(t){for(let e=0,i=this.count;e<i;e++)_i.fromBufferAttribute(this,e),_i.applyMatrix4(t),this.setXYZ(e,_i.x,_i.y,_i.z);return this}applyNormalMatrix(t){for(let e=0,i=this.count;e<i;e++)_i.fromBufferAttribute(this,e),_i.applyNormalMatrix(t),this.setXYZ(e,_i.x,_i.y,_i.z);return this}transformDirection(t){for(let e=0,i=this.count;e<i;e++)_i.fromBufferAttribute(this,e),_i.transformDirection(t),this.setXYZ(e,_i.x,_i.y,_i.z);return this}set(t,e=0){return this.array.set(t,e),this}getX(t){let e=this.array[t*this.itemSize];return this.normalized&&(e=At(e,this.array)),e}setX(t,e){return this.normalized&&(e=Et(e,this.array)),this.array[t*this.itemSize]=e,this}getY(t){let e=this.array[t*this.itemSize+1];return this.normalized&&(e=At(e,this.array)),e}setY(t,e){return this.normalized&&(e=Et(e,this.array)),this.array[t*this.itemSize+1]=e,this}getZ(t){let e=this.array[t*this.itemSize+2];return this.normalized&&(e=At(e,this.array)),e}setZ(t,e){return this.normalized&&(e=Et(e,this.array)),this.array[t*this.itemSize+2]=e,this}getW(t){let e=this.array[t*this.itemSize+3];return this.normalized&&(e=At(e,this.array)),e}setW(t,e){return this.normalized&&(e=Et(e,this.array)),this.array[t*this.itemSize+3]=e,this}setXY(t,e,i){return t*=this.itemSize,this.normalized&&(e=Et(e,this.array),i=Et(i,this.array)),this.array[t+0]=e,this.array[t+1]=i,this}setXYZ(t,e,i,n){return t*=this.itemSize,this.normalized&&(e=Et(e,this.array),i=Et(i,this.array),n=Et(n,this.array)),this.array[t+0]=e,this.array[t+1]=i,this.array[t+2]=n,this}setXYZW(t,e,i,n,r){return t*=this.itemSize,this.normalized&&(e=Et(e,this.array),i=Et(i,this.array),n=Et(n,this.array),r=Et(r,this.array)),this.array[t+0]=e,this.array[t+1]=i,this.array[t+2]=n,this.array[t+3]=r,this}onUpload(t){return this.onUploadCallback=t,this}clone(){return new this.constructor(this.array,this.itemSize).copy(this)}toJSON(){const t={itemSize:this.itemSize,type:this.array.constructor.name,array:Array.from(this.array),normalized:this.normalized};return""!==this.name&&(t.name=this.name),this.usage!==ut&&(t.usage=this.usage),0===this.updateRange.offset&&-1===this.updateRange.count||(t.updateRange=this.updateRange),t}copyColorsArray(){console.error("THREE.BufferAttribute: copyColorsArray() was removed in r144.")}copyVector2sArray(){console.error("THREE.BufferAttribute: copyVector2sArray() was removed in r144.")}copyVector3sArray(){console.error("THREE.BufferAttribute: copyVector3sArray() was removed in r144.")}copyVector4sArray(){console.error("THREE.BufferAttribute: copyVector4sArray() was removed in r144.")}}class bi extends Mi{constructor(t,e,i){super(new Uint16Array(t),e,i)}}class Si extends Mi{constructor(t,e,i){super(new Uint32Array(t),e,i)}}class wi extends Mi{constructor(t,e,i){super(new Float32Array(t),e,i)}}let Ti=0;const Ai=new Ie,Ei=new ri,Ci=new ne,Li=new ae,Ri=new ae,Pi=new ne;class Di extends mt{constructor(){super(),this.isBufferGeometry=!0,Object.defineProperty(this,"id",{value:Ti++}),this.uuid=_t(),this.name="",this.type="BufferGeometry",this.index=null,this.attributes={},this.morphAttributes={},this.morphTargetsRelative=!1,this.groups=[],this.boundingBox=null,this.boundingSphere=null,this.drawRange={start:0,count:1/0},this.userData={}}getIndex(){return this.index}setIndex(t){return Array.isArray(t)?this.index=new(Pt(t)?Si:bi)(t,1):this.index=t,this}getAttribute(t){return this.attributes[t]}setAttribute(t,e){return this.attributes[t]=e,this}deleteAttribute(t){return delete this.attributes[t],this}hasAttribute(t){return void 0!==this.attributes[t]}addGroup(t,e,i=0){this.groups.push({start:t,count:e,materialIndex:i})}clearGroups(){this.groups=[]}setDrawRange(t,e){this.drawRange.start=t,this.drawRange.count=e}applyMatrix4(t){const e=this.attributes.position;void 0!==e&&(e.applyMatrix4(t),e.needsUpdate=!0);const i=this.attributes.normal;if(void 0!==i){const e=(new Rt).getNormalMatrix(t);i.applyNormalMatrix(e),i.needsUpdate=!0}const n=this.attributes.tangent;return void 0!==n&&(n.transformDirection(t),n.needsUpdate=!0),null!==this.boundingBox&&this.computeBoundingBox(),null!==this.boundingSphere&&this.computeBoundingSphere(),this}applyQuaternion(t){return Ai.makeRotationFromQuaternion(t),this.applyMatrix4(Ai),this}rotateX(t){return Ai.makeRotationX(t),this.applyMatrix4(Ai),this}rotateY(t){return Ai.makeRotationY(t),this.applyMatrix4(Ai),this}rotateZ(t){return Ai.makeRotationZ(t),this.applyMatrix4(Ai),this}translate(t,e,i){return Ai.makeTranslation(t,e,i),this.applyMatrix4(Ai),this}scale(t,e,i){return Ai.makeScale(t,e,i),this.applyMatrix4(Ai),this}lookAt(t){return Ei.lookAt(t),Ei.updateMatrix(),this.applyMatrix4(Ei.matrix),this}center(){return this.computeBoundingBox(),this.boundingBox.getCenter(Ci).negate(),this.translate(Ci.x,Ci.y,Ci.z),this}setFromPoints(t){const e=[];for(let i=0,n=t.length;i<n;i++){const n=t[i];e.push(n.x,n.y,n.z||0)}return this.setAttribute("position",new wi(e,3)),this}computeBoundingBox(){null===this.boundingBox&&(this.boundingBox=new ae);const t=this.attributes.position,e=this.morphAttributes.position;if(t&&t.isGLBufferAttribute)return console.error('THREE.BufferGeometry.computeBoundingBox(): GLBufferAttribute requires a manual bounding box. Alternatively set "mesh.frustumCulled" to "false".',this),void this.boundingBox.set(new ne(-1/0,-1/0,-1/0),new ne(1/0,1/0,1/0));if(void 0!==t){if(this.boundingBox.setFromBufferAttribute(t),e)for(let t=0,i=e.length;t<i;t++){const i=e[t];Li.setFromBufferAttribute(i),this.morphTargetsRelative?(Pi.addVectors(this.boundingBox.min,Li.min),this.boundingBox.expandByPoint(Pi),Pi.addVectors(this.boundingBox.max,Li.max),this.boundingBox.expandByPoint(Pi)):(this.boundingBox.expandByPoint(Li.min),this.boundingBox.expandByPoint(Li.max))}}else this.boundingBox.makeEmpty();(isNaN(this.boundingBox.min.x)||isNaN(this.boundingBox.min.y)||isNaN(this.boundingBox.min.z))&&console.error('THREE.BufferGeometry.computeBoundingBox(): Computed min/max have NaN values. The "position" attribute is likely to have NaN values.',this)}computeBoundingSphere(){null===this.boundingSphere&&(this.boundingSphere=new we);const t=this.attributes.position,e=this.morphAttributes.position;if(t&&t.isGLBufferAttribute)return console.error('THREE.BufferGeometry.computeBoundingSphere(): GLBufferAttribute requires a manual bounding sphere. Alternatively set "mesh.frustumCulled" to "false".',this),void this.boundingSphere.set(new ne,1/0);if(t){const i=this.boundingSphere.center;if(Li.setFromBufferAttribute(t),e)for(let t=0,i=e.length;t<i;t++){const i=e[t];Ri.setFromBufferAttribute(i),this.morphTargetsRelative?(Pi.addVectors(Li.min,Ri.min),Li.expandByPoint(Pi),Pi.addVectors(Li.max,Ri.max),Li.expandByPoint(Pi)):(Li.expandByPoint(Ri.min),Li.expandByPoint(Ri.max))}Li.getCenter(i);let n=0;for(let e=0,r=t.count;e<r;e++)Pi.fromBufferAttribute(t,e),n=Math.max(n,i.distanceToSquared(Pi));if(e)for(let r=0,s=e.length;r<s;r++){const s=e[r],a=this.morphTargetsRelative;for(let e=0,r=s.count;e<r;e++)Pi.fromBufferAttribute(s,e),a&&(Ci.fromBufferAttribute(t,e),Pi.add(Ci)),n=Math.max(n,i.distanceToSquared(Pi))}this.boundingSphere.radius=Math.sqrt(n),isNaN(this.boundingSphere.radius)&&console.error('THREE.BufferGeometry.computeBoundingSphere(): Computed radius is NaN. The "position" attribute is likely to have NaN values.',this)}}computeTangents(){const t=this.index,e=this.attributes;if(null===t||void 0===e.position||void 0===e.normal||void 0===e.uv)return void console.error("THREE.BufferGeometry: .computeTangents() failed. Missing required attributes (index, position, normal or uv)");const i=t.array,n=e.position.array,r=e.normal.array,s=e.uv.array,a=n.length/3;!1===this.hasAttribute("tangent")&&this.setAttribute("tangent",new Mi(new Float32Array(4*a),4));const o=this.getAttribute("tangent").array,l=[],c=[];for(let t=0;t<a;t++)l[t]=new ne,c[t]=new ne;const h=new ne,u=new ne,d=new ne,p=new Lt,m=new Lt,f=new Lt,g=new ne,v=new ne;function x(t,e,i){h.fromArray(n,3*t),u.fromArray(n,3*e),d.fromArray(n,3*i),p.fromArray(s,2*t),m.fromArray(s,2*e),f.fromArray(s,2*i),u.sub(h),d.sub(h),m.sub(p),f.sub(p);const r=1/(m.x*f.y-f.x*m.y);isFinite(r)&&(g.copy(u).multiplyScalar(f.y).addScaledVector(d,-m.y).multiplyScalar(r),v.copy(d).multiplyScalar(m.x).addScaledVector(u,-f.x).multiplyScalar(r),l[t].add(g),l[e].add(g),l[i].add(g),c[t].add(v),c[e].add(v),c[i].add(v))}let _=this.groups;0===_.length&&(_=[{start:0,count:i.length}]);for(let t=0,e=_.length;t<e;++t){const e=_[t],n=e.start;for(let t=n,r=n+e.count;t<r;t+=3)x(i[t+0],i[t+1],i[t+2])}const y=new ne,M=new ne,b=new ne,S=new ne;function w(t){b.fromArray(r,3*t),S.copy(b);const e=l[t];y.copy(e),y.sub(b.multiplyScalar(b.dot(e))).normalize(),M.crossVectors(S,e);const i=M.dot(c[t])<0?-1:1;o[4*t]=y.x,o[4*t+1]=y.y,o[4*t+2]=y.z,o[4*t+3]=i}for(let t=0,e=_.length;t<e;++t){const e=_[t],n=e.start;for(let t=n,r=n+e.count;t<r;t+=3)w(i[t+0]),w(i[t+1]),w(i[t+2])}}computeVertexNormals(){const t=this.index,e=this.getAttribute("position");if(void 0!==e){let i=this.getAttribute("normal");if(void 0===i)i=new Mi(new Float32Array(3*e.count),3),this.setAttribute("normal",i);else for(let t=0,e=i.count;t<e;t++)i.setXYZ(t,0,0,0);const n=new ne,r=new ne,s=new ne,a=new ne,o=new ne,l=new ne,c=new ne,h=new ne;if(t)for(let u=0,d=t.count;u<d;u+=3){const d=t.getX(u+0),p=t.getX(u+1),m=t.getX(u+2);n.fromBufferAttribute(e,d),r.fromBufferAttribute(e,p),s.fromBufferAttribute(e,m),c.subVectors(s,r),h.subVectors(n,r),c.cross(h),a.fromBufferAttribute(i,d),o.fromBufferAttribute(i,p),l.fromBufferAttribute(i,m),a.add(c),o.add(c),l.add(c),i.setXYZ(d,a.x,a.y,a.z),i.setXYZ(p,o.x,o.y,o.z),i.setXYZ(m,l.x,l.y,l.z)}else for(let t=0,a=e.count;t<a;t+=3)n.fromBufferAttribute(e,t+0),r.fromBufferAttribute(e,t+1),s.fromBufferAttribute(e,t+2),c.subVectors(s,r),h.subVectors(n,r),c.cross(h),i.setXYZ(t+0,c.x,c.y,c.z),i.setXYZ(t+1,c.x,c.y,c.z),i.setXYZ(t+2,c.x,c.y,c.z);this.normalizeNormals(),i.needsUpdate=!0}}merge(){return console.error("THREE.BufferGeometry.merge() has been removed. Use THREE.BufferGeometryUtils.mergeBufferGeometries() instead."),this}normalizeNormals(){const t=this.attributes.normal;for(let e=0,i=t.count;e<i;e++)Pi.fromBufferAttribute(t,e),Pi.normalize(),t.setXYZ(e,Pi.x,Pi.y,Pi.z)}toNonIndexed(){function t(t,e){const i=t.array,n=t.itemSize,r=t.normalized,s=new i.constructor(e.length*n);let a=0,o=0;for(let r=0,l=e.length;r<l;r++){a=t.isInterleavedBufferAttribute?e[r]*t.data.stride+t.offset:e[r]*n;for(let t=0;t<n;t++)s[o++]=i[a++]}return new Mi(s,n,r)}if(null===this.index)return console.warn("THREE.BufferGeometry.toNonIndexed(): BufferGeometry is already non-indexed."),this;const e=new Di,i=this.index.array,n=this.attributes;for(const r in n){const s=t(n[r],i);e.setAttribute(r,s)}const r=this.morphAttributes;for(const n in r){const s=[],a=r[n];for(let e=0,n=a.length;e<n;e++){const n=t(a[e],i);s.push(n)}e.morphAttributes[n]=s}e.morphTargetsRelative=this.morphTargetsRelative;const s=this.groups;for(let t=0,i=s.length;t<i;t++){const i=s[t];e.addGroup(i.start,i.count,i.materialIndex)}return e}toJSON(){const t={metadata:{version:4.5,type:"BufferGeometry",generator:"BufferGeometry.toJSON"}};if(t.uuid=this.uuid,t.type=this.type,""!==this.name&&(t.name=this.name),Object.keys(this.userData).length>0&&(t.userData=this.userData),void 0!==this.parameters){const e=this.parameters;for(const i in e)void 0!==e[i]&&(t[i]=e[i]);return t}t.data={attributes:{}};const e=this.index;null!==e&&(t.data.index={type:e.array.constructor.name,array:Array.prototype.slice.call(e.array)});const i=this.attributes;for(const e in i){const n=i[e];t.data.attributes[e]=n.toJSON(t.data)}const n={};let r=!1;for(const e in this.morphAttributes){const i=this.morphAttributes[e],s=[];for(let e=0,n=i.length;e<n;e++){const n=i[e];s.push(n.toJSON(t.data))}s.length>0&&(n[e]=s,r=!0)}r&&(t.data.morphAttributes=n,t.data.morphTargetsRelative=this.morphTargetsRelative);const s=this.groups;s.length>0&&(t.data.groups=JSON.parse(JSON.stringify(s)));const a=this.boundingSphere;return null!==a&&(t.data.boundingSphere={center:a.center.toArray(),radius:a.radius}),t}clone(){return(new this.constructor).copy(this)}copy(t){this.index=null,this.attributes={},this.morphAttributes={},this.groups=[],this.boundingBox=null,this.boundingSphere=null;const e={};this.name=t.name;const i=t.index;null!==i&&this.setIndex(i.clone(e));const n=t.attributes;for(const t in n){const i=n[t];this.setAttribute(t,i.clone(e))}const r=t.morphAttributes;for(const t in r){const i=[],n=r[t];for(let t=0,r=n.length;t<r;t++)i.push(n[t].clone(e));this.morphAttributes[t]=i}this.morphTargetsRelative=t.morphTargetsRelative;const s=t.groups;for(let t=0,e=s.length;t<e;t++){const e=s[t];this.addGroup(e.start,e.count,e.materialIndex)}const a=t.boundingBox;null!==a&&(this.boundingBox=a.clone());const o=t.boundingSphere;return null!==o&&(this.boundingSphere=o.clone()),this.drawRange.start=t.drawRange.start,this.drawRange.count=t.drawRange.count,this.userData=t.userData,void 0!==t.parameters&&(this.parameters=Object.assign({},t.parameters)),this}dispose(){this.dispatchEvent({type:"dispose"})}}const Ii=new Ie,Ni=new De,Oi=new we,zi=new ne,Ui=new ne,Bi=new ne,Fi=new ne,ki=new ne,Gi=new ne,Vi=new ne,Hi=new ne,Wi=new ne,ji=new Lt,qi=new Lt,Xi=new Lt,Yi=new ne,Zi=new ne;class Ji extends ri{constructor(t=new Di,e=new xi){super(),this.isMesh=!0,this.type="Mesh",this.geometry=t,this.material=e,this.updateMorphTargets()}copy(t,e){return super.copy(t,e),void 0!==t.morphTargetInfluences&&(this.morphTargetInfluences=t.morphTargetInfluences.slice()),void 0!==t.morphTargetDictionary&&(this.morphTargetDictionary=Object.assign({},t.morphTargetDictionary)),this.material=t.material,this.geometry=t.geometry,this}updateMorphTargets(){const t=this.geometry.morphAttributes,e=Object.keys(t);if(e.length>0){const i=t[e[0]];if(void 0!==i){this.morphTargetInfluences=[],this.morphTargetDictionary={};for(let t=0,e=i.length;t<e;t++){const e=i[t].name||String(t);this.morphTargetInfluences.push(0),this.morphTargetDictionary[e]=t}}}}raycast(t,e){const i=this.geometry,n=this.material,r=this.matrixWorld;if(void 0===n)return;if(null===i.boundingSphere&&i.computeBoundingSphere(),Oi.copy(i.boundingSphere),Oi.applyMatrix4(r),!1===t.ray.intersectsSphere(Oi))return;if(Ii.copy(r).invert(),Ni.copy(t.ray).applyMatrix4(Ii),null!==i.boundingBox&&!1===Ni.intersectsBox(i.boundingBox))return;let s;const a=i.index,o=i.attributes.position,l=i.morphAttributes.position,c=i.morphTargetsRelative,h=i.attributes.uv,u=i.attributes.uv2,d=i.groups,p=i.drawRange;if(null!==a)if(Array.isArray(n))for(let i=0,r=d.length;i<r;i++){const r=d[i],m=n[r.materialIndex];for(let i=Math.max(r.start,p.start),n=Math.min(a.count,Math.min(r.start+r.count,p.start+p.count));i<n;i+=3){const n=a.getX(i),d=a.getX(i+1),p=a.getX(i+2);s=Ki(this,m,t,Ni,o,l,c,h,u,n,d,p),s&&(s.faceIndex=Math.floor(i/3),s.face.materialIndex=r.materialIndex,e.push(s))}}else{for(let i=Math.max(0,p.start),r=Math.min(a.count,p.start+p.count);i<r;i+=3){const r=a.getX(i),d=a.getX(i+1),p=a.getX(i+2);s=Ki(this,n,t,Ni,o,l,c,h,u,r,d,p),s&&(s.faceIndex=Math.floor(i/3),e.push(s))}}else if(void 0!==o)if(Array.isArray(n))for(let i=0,r=d.length;i<r;i++){const r=d[i],a=n[r.materialIndex];for(let i=Math.max(r.start,p.start),n=Math.min(o.count,Math.min(r.start+r.count,p.start+p.count));i<n;i+=3){s=Ki(this,a,t,Ni,o,l,c,h,u,i,i+1,i+2),s&&(s.faceIndex=Math.floor(i/3),s.face.materialIndex=r.materialIndex,e.push(s))}}else{for(let i=Math.max(0,p.start),r=Math.min(o.count,p.start+p.count);i<r;i+=3){s=Ki(this,n,t,Ni,o,l,c,h,u,i,i+1,i+2),s&&(s.faceIndex=Math.floor(i/3),e.push(s))}}}}function Ki(t,e,i,n,r,s,a,o,l,c,h,u){zi.fromBufferAttribute(r,c),Ui.fromBufferAttribute(r,h),Bi.fromBufferAttribute(r,u);const d=t.morphTargetInfluences;if(s&&d){Vi.set(0,0,0),Hi.set(0,0,0),Wi.set(0,0,0);for(let t=0,e=s.length;t<e;t++){const e=d[t],i=s[t];0!==e&&(Fi.fromBufferAttribute(i,c),ki.fromBufferAttribute(i,h),Gi.fromBufferAttribute(i,u),a?(Vi.addScaledVector(Fi,e),Hi.addScaledVector(ki,e),Wi.addScaledVector(Gi,e)):(Vi.addScaledVector(Fi.sub(zi),e),Hi.addScaledVector(ki.sub(Ui),e),Wi.addScaledVector(Gi.sub(Bi),e)))}zi.add(Vi),Ui.add(Hi),Bi.add(Wi)}t.isSkinnedMesh&&(t.boneTransform(c,zi),t.boneTransform(h,Ui),t.boneTransform(u,Bi));const p=function(t,e,i,n,r,s,a,o){let l;if(l=1===e.side?n.intersectTriangle(a,s,r,!0,o):n.intersectTriangle(r,s,a,2!==e.side,o),null===l)return null;Zi.copy(o),Zi.applyMatrix4(t.matrixWorld);const c=i.ray.origin.distanceTo(Zi);return c<i.near||c>i.far?null:{distance:c,point:Zi.clone(),object:t}}(t,e,i,n,zi,Ui,Bi,Yi);if(p){o&&(ji.fromBufferAttribute(o,c),qi.fromBufferAttribute(o,h),Xi.fromBufferAttribute(o,u),p.uv=fi.getUV(Yi,zi,Ui,Bi,ji,qi,Xi,new Lt)),l&&(ji.fromBufferAttribute(l,c),qi.fromBufferAttribute(l,h),Xi.fromBufferAttribute(l,u),p.uv2=fi.getUV(Yi,zi,Ui,Bi,ji,qi,Xi,new Lt));const t={a:c,b:h,c:u,normal:new ne,materialIndex:0};fi.getNormal(zi,Ui,Bi,t.normal),p.face=t}return p}class $i extends Di{constructor(t=1,e=1,i=1,n=1,r=1,s=1){super(),this.type="BoxGeometry",this.parameters={width:t,height:e,depth:i,widthSegments:n,heightSegments:r,depthSegments:s};const a=this;n=Math.floor(n),r=Math.floor(r),s=Math.floor(s);const o=[],l=[],c=[],h=[];let u=0,d=0;function p(t,e,i,n,r,s,p,m,f,g,v){const x=s/f,_=p/g,y=s/2,M=p/2,b=m/2,S=f+1,w=g+1;let T=0,A=0;const E=new ne;for(let s=0;s<w;s++){const a=s*_-M;for(let o=0;o<S;o++){const u=o*x-y;E[t]=u*n,E[e]=a*r,E[i]=b,l.push(E.x,E.y,E.z),E[t]=0,E[e]=0,E[i]=m>0?1:-1,c.push(E.x,E.y,E.z),h.push(o/f),h.push(1-s/g),T+=1}}for(let t=0;t<g;t++)for(let e=0;e<f;e++){const i=u+e+S*t,n=u+e+S*(t+1),r=u+(e+1)+S*(t+1),s=u+(e+1)+S*t;o.push(i,n,s),o.push(n,r,s),A+=6}a.addGroup(d,A,v),d+=A,u+=T}p("z","y","x",-1,-1,i,e,t,s,r,0),p("z","y","x",1,-1,i,e,-t,s,r,1),p("x","z","y",1,1,t,i,e,n,s,2),p("x","z","y",1,-1,t,i,-e,n,s,3),p("x","y","z",1,-1,t,e,i,n,r,4),p("x","y","z",-1,-1,t,e,-i,n,r,5),this.setIndex(o),this.setAttribute("position",new wi(l,3)),this.setAttribute("normal",new wi(c,3)),this.setAttribute("uv",new wi(h,2))}static fromJSON(t){return new $i(t.width,t.height,t.depth,t.widthSegments,t.heightSegments,t.depthSegments)}}function Qi(t){const e={};for(const i in t){e[i]={};for(const n in t[i]){const r=t[i][n];r&&(r.isColor||r.isMatrix3||r.isMatrix4||r.isVector2||r.isVector3||r.isVector4||r.isTexture||r.isQuaternion)?e[i][n]=r.clone():Array.isArray(r)?e[i][n]=r.slice():e[i][n]=r}}return e}function tn(t){const e={};for(let i=0;i<t.length;i++){const n=Qi(t[i]);for(const t in n)e[t]=n[t]}return e}const en={clone:Qi,merge:tn};class nn extends vi{constructor(t){super(),this.isShaderMaterial=!0,this.type="ShaderMaterial",this.defines={},this.uniforms={},this.uniformsGroups=[],this.vertexShader="void main() {\n\tgl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );\n}",this.fragmentShader="void main() {\n\tgl_FragColor = vec4( 1.0, 0.0, 0.0, 1.0 );\n}",this.linewidth=1,this.wireframe=!1,this.wireframeLinewidth=1,this.fog=!1,this.lights=!1,this.clipping=!1,this.extensions={derivatives:!1,fragDepth:!1,drawBuffers:!1,shaderTextureLOD:!1},this.defaultAttributeValues={color:[1,1,1],uv:[0,0],uv2:[0,0]},this.index0AttributeName=void 0,this.uniformsNeedUpdate=!1,this.glslVersion=null,void 0!==t&&this.setValues(t)}copy(t){return super.copy(t),this.fragmentShader=t.fragmentShader,this.vertexShader=t.vertexShader,this.uniforms=Qi(t.uniforms),this.uniformsGroups=function(t){const e=[];for(let i=0;i<t.length;i++)e.push(t[i].clone());return e}(t.uniformsGroups),this.defines=Object.assign({},t.defines),this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.fog=t.fog,this.lights=t.lights,this.clipping=t.clipping,this.extensions=Object.assign({},t.extensions),this.glslVersion=t.glslVersion,this}toJSON(t){const e=super.toJSON(t);e.glslVersion=this.glslVersion,e.uniforms={};for(const i in this.uniforms){const n=this.uniforms[i].value;n&&n.isTexture?e.uniforms[i]={type:"t",value:n.toJSON(t).uuid}:n&&n.isColor?e.uniforms[i]={type:"c",value:n.getHex()}:n&&n.isVector2?e.uniforms[i]={type:"v2",value:n.toArray()}:n&&n.isVector3?e.uniforms[i]={type:"v3",value:n.toArray()}:n&&n.isVector4?e.uniforms[i]={type:"v4",value:n.toArray()}:n&&n.isMatrix3?e.uniforms[i]={type:"m3",value:n.toArray()}:n&&n.isMatrix4?e.uniforms[i]={type:"m4",value:n.toArray()}:e.uniforms[i]={value:n}}Object.keys(this.defines).length>0&&(e.defines=this.defines),e.vertexShader=this.vertexShader,e.fragmentShader=this.fragmentShader;const i={};for(const t in this.extensions)!0===this.extensions[t]&&(i[t]=!0);return Object.keys(i).length>0&&(e.extensions=i),e}}class rn extends ri{constructor(){super(),this.isCamera=!0,this.type="Camera",this.matrixWorldInverse=new Ie,this.projectionMatrix=new Ie,this.projectionMatrixInverse=new Ie}copy(t,e){return super.copy(t,e),this.matrixWorldInverse.copy(t.matrixWorldInverse),this.projectionMatrix.copy(t.projectionMatrix),this.projectionMatrixInverse.copy(t.projectionMatrixInverse),this}getWorldDirection(t){this.updateWorldMatrix(!0,!1);const e=this.matrixWorld.elements;return t.set(-e[8],-e[9],-e[10]).normalize()}updateMatrixWorld(t){super.updateMatrixWorld(t),this.matrixWorldInverse.copy(this.matrixWorld).invert()}updateWorldMatrix(t,e){super.updateWorldMatrix(t,e),this.matrixWorldInverse.copy(this.matrixWorld).invert()}clone(){return(new this.constructor).copy(this)}}class sn extends rn{constructor(t=50,e=1,i=.1,n=2e3){super(),this.isPerspectiveCamera=!0,this.type="PerspectiveCamera",this.fov=t,this.zoom=1,this.near=i,this.far=n,this.focus=10,this.aspect=e,this.view=null,this.filmGauge=35,this.filmOffset=0,this.updateProjectionMatrix()}copy(t,e){return super.copy(t,e),this.fov=t.fov,this.zoom=t.zoom,this.near=t.near,this.far=t.far,this.focus=t.focus,this.aspect=t.aspect,this.view=null===t.view?null:Object.assign({},t.view),this.filmGauge=t.filmGauge,this.filmOffset=t.filmOffset,this}setFocalLength(t){const e=.5*this.getFilmHeight()/t;this.fov=2*xt*Math.atan(e),this.updateProjectionMatrix()}getFocalLength(){const t=Math.tan(.5*vt*this.fov);return.5*this.getFilmHeight()/t}getEffectiveFOV(){return 2*xt*Math.atan(Math.tan(.5*vt*this.fov)/this.zoom)}getFilmWidth(){return this.filmGauge*Math.min(this.aspect,1)}getFilmHeight(){return this.filmGauge/Math.max(this.aspect,1)}setViewOffset(t,e,i,n,r,s){this.aspect=t/e,null===this.view&&(this.view={enabled:!0,fullWidth:1,fullHeight:1,offsetX:0,offsetY:0,width:1,height:1}),this.view.enabled=!0,this.view.fullWidth=t,this.view.fullHeight=e,this.view.offsetX=i,this.view.offsetY=n,this.view.width=r,this.view.height=s,this.updateProjectionMatrix()}clearViewOffset(){null!==this.view&&(this.view.enabled=!1),this.updateProjectionMatrix()}updateProjectionMatrix(){const t=this.near;let e=t*Math.tan(.5*vt*this.fov)/this.zoom,i=2*e,n=this.aspect*i,r=-.5*n;const s=this.view;if(null!==this.view&&this.view.enabled){const t=s.fullWidth,a=s.fullHeight;r+=s.offsetX*n/t,e-=s.offsetY*i/a,n*=s.width/t,i*=s.height/a}const a=this.filmOffset;0!==a&&(r+=t*a/this.getFilmWidth()),this.projectionMatrix.makePerspective(r,r+n,e,e-i,t,this.far),this.projectionMatrixInverse.copy(this.projectionMatrix).invert()}toJSON(t){const e=super.toJSON(t);return e.object.fov=this.fov,e.object.zoom=this.zoom,e.object.near=this.near,e.object.far=this.far,e.object.focus=this.focus,e.object.aspect=this.aspect,null!==this.view&&(e.object.view=Object.assign({},this.view)),e.object.filmGauge=this.filmGauge,e.object.filmOffset=this.filmOffset,e}}const an=90;class on extends ri{constructor(t,e,i){super(),this.type="CubeCamera",this.renderTarget=i;const n=new sn(an,1,t,e);n.layers=this.layers,n.up.set(0,-1,0),n.lookAt(new ne(1,0,0)),this.add(n);const r=new sn(an,1,t,e);r.layers=this.layers,r.up.set(0,-1,0),r.lookAt(new ne(-1,0,0)),this.add(r);const s=new sn(an,1,t,e);s.layers=this.layers,s.up.set(0,0,1),s.lookAt(new ne(0,1,0)),this.add(s);const a=new sn(an,1,t,e);a.layers=this.layers,a.up.set(0,0,-1),a.lookAt(new ne(0,-1,0)),this.add(a);const o=new sn(an,1,t,e);o.layers=this.layers,o.up.set(0,-1,0),o.lookAt(new ne(0,0,1)),this.add(o);const l=new sn(an,1,t,e);l.layers=this.layers,l.up.set(0,-1,0),l.lookAt(new ne(0,0,-1)),this.add(l)}update(t,e){null===this.parent&&this.updateMatrixWorld();const i=this.renderTarget,[n,r,s,a,o,l]=this.children,c=t.getRenderTarget(),h=t.toneMapping,u=t.xr.enabled;t.toneMapping=0,t.xr.enabled=!1;const d=i.texture.generateMipmaps;i.texture.generateMipmaps=!1,t.setRenderTarget(i,0),t.render(e,n),t.setRenderTarget(i,1),t.render(e,r),t.setRenderTarget(i,2),t.render(e,s),t.setRenderTarget(i,3),t.render(e,a),t.setRenderTarget(i,4),t.render(e,o),i.texture.generateMipmaps=d,t.setRenderTarget(i,5),t.render(e,l),t.setRenderTarget(c),t.toneMapping=h,t.xr.enabled=u,i.texture.needsPMREMUpdate=!0}}class ln extends Kt{constructor(t,e,i,n,s,a,o,l,c,h){super(t=void 0!==t?t:[],e=void 0!==e?e:r,i,n,s,a,o,l,c,h),this.isCubeTexture=!0,this.flipY=!1}get images(){return this.image}set images(t){this.image=t}}class cn extends Qt{constructor(t=1,e={}){super(t,t,e),this.isWebGLCubeRenderTarget=!0;const i={width:t,height:t,depth:1},n=[i,i,i,i,i,i];this.texture=new ln(n,e.mapping,e.wrapS,e.wrapT,e.magFilter,e.minFilter,e.format,e.type,e.anisotropy,e.encoding),this.texture.isRenderTargetTexture=!0,this.texture.generateMipmaps=void 0!==e.generateMipmaps&&e.generateMipmaps,this.texture.minFilter=void 0!==e.minFilter?e.minFilter:f}fromEquirectangularTexture(t,e){this.texture.type=e.type,this.texture.encoding=e.encoding,this.texture.generateMipmaps=e.generateMipmaps,this.texture.minFilter=e.minFilter,this.texture.magFilter=e.magFilter;const i={uniforms:{tEquirect:{value:null}},vertexShader:"\n\n\t\t\t\tvarying vec3 vWorldDirection;\n\n\t\t\t\tvec3 transformDirection( in vec3 dir, in mat4 matrix ) {\n\n\t\t\t\t\treturn normalize( ( matrix * vec4( dir, 0.0 ) ).xyz );\n\n\t\t\t\t}\n\n\t\t\t\tvoid main() {\n\n\t\t\t\t\tvWorldDirection = transformDirection( position, modelMatrix );\n\n\t\t\t\t\t#include <begin_vertex>\n\t\t\t\t\t#include <project_vertex>\n\n\t\t\t\t}\n\t\t\t",fragmentShader:"\n\n\t\t\t\tuniform sampler2D tEquirect;\n\n\t\t\t\tvarying vec3 vWorldDirection;\n\n\t\t\t\t#include <common>\n\n\t\t\t\tvoid main() {\n\n\t\t\t\t\tvec3 direction = normalize( vWorldDirection );\n\n\t\t\t\t\tvec2 sampleUV = equirectUv( direction );\n\n\t\t\t\t\tgl_FragColor = texture2D( tEquirect, sampleUV );\n\n\t\t\t\t}\n\t\t\t"},n=new $i(5,5,5),r=new nn({name:"CubemapFromEquirect",uniforms:Qi(i.uniforms),vertexShader:i.vertexShader,fragmentShader:i.fragmentShader,side:1,blending:0});r.uniforms.tEquirect.value=e;const s=new Ji(n,r),a=e.minFilter;e.minFilter===v&&(e.minFilter=f);return new on(1,10,this).update(t,s),e.minFilter=a,s.geometry.dispose(),s.material.dispose(),this}clear(t,e,i,n){const r=t.getRenderTarget();for(let r=0;r<6;r++)t.setRenderTarget(this,r),t.clear(e,i,n);t.setRenderTarget(r)}}const hn=new ne,un=new ne,dn=new Rt;class pn{constructor(t=new ne(1,0,0),e=0){this.isPlane=!0,this.normal=t,this.constant=e}set(t,e){return this.normal.copy(t),this.constant=e,this}setComponents(t,e,i,n){return this.normal.set(t,e,i),this.constant=n,this}setFromNormalAndCoplanarPoint(t,e){return this.normal.copy(t),this.constant=-e.dot(this.normal),this}setFromCoplanarPoints(t,e,i){const n=hn.subVectors(i,e).cross(un.subVectors(t,e)).normalize();return this.setFromNormalAndCoplanarPoint(n,t),this}copy(t){return this.normal.copy(t.normal),this.constant=t.constant,this}normalize(){const t=1/this.normal.length();return this.normal.multiplyScalar(t),this.constant*=t,this}negate(){return this.constant*=-1,this.normal.negate(),this}distanceToPoint(t){return this.normal.dot(t)+this.constant}distanceToSphere(t){return this.distanceToPoint(t.center)-t.radius}projectPoint(t,e){return e.copy(this.normal).multiplyScalar(-this.distanceToPoint(t)).add(t)}intersectLine(t,e){const i=t.delta(hn),n=this.normal.dot(i);if(0===n)return 0===this.distanceToPoint(t.start)?e.copy(t.start):null;const r=-(t.start.dot(this.normal)+this.constant)/n;return r<0||r>1?null:e.copy(i).multiplyScalar(r).add(t.start)}intersectsLine(t){const e=this.distanceToPoint(t.start),i=this.distanceToPoint(t.end);return e<0&&i>0||i<0&&e>0}intersectsBox(t){return t.intersectsPlane(this)}intersectsSphere(t){return t.intersectsPlane(this)}coplanarPoint(t){return t.copy(this.normal).multiplyScalar(-this.constant)}applyMatrix4(t,e){const i=e||dn.getNormalMatrix(t),n=this.coplanarPoint(hn).applyMatrix4(t),r=this.normal.applyMatrix3(i).normalize();return this.constant=-n.dot(r),this}translate(t){return this.constant-=t.dot(this.normal),this}equals(t){return t.normal.equals(this.normal)&&t.constant===this.constant}clone(){return(new this.constructor).copy(this)}}const mn=new we,fn=new ne;class gn{constructor(t=new pn,e=new pn,i=new pn,n=new pn,r=new pn,s=new pn){this.planes=[t,e,i,n,r,s]}set(t,e,i,n,r,s){const a=this.planes;return a[0].copy(t),a[1].copy(e),a[2].copy(i),a[3].copy(n),a[4].copy(r),a[5].copy(s),this}copy(t){const e=this.planes;for(let i=0;i<6;i++)e[i].copy(t.planes[i]);return this}setFromProjectionMatrix(t){const e=this.planes,i=t.elements,n=i[0],r=i[1],s=i[2],a=i[3],o=i[4],l=i[5],c=i[6],h=i[7],u=i[8],d=i[9],p=i[10],m=i[11],f=i[12],g=i[13],v=i[14],x=i[15];return e[0].setComponents(a-n,h-o,m-u,x-f).normalize(),e[1].setComponents(a+n,h+o,m+u,x+f).normalize(),e[2].setComponents(a+r,h+l,m+d,x+g).normalize(),e[3].setComponents(a-r,h-l,m-d,x-g).normalize(),e[4].setComponents(a-s,h-c,m-p,x-v).normalize(),e[5].setComponents(a+s,h+c,m+p,x+v).normalize(),this}intersectsObject(t){const e=t.geometry;return null===e.boundingSphere&&e.computeBoundingSphere(),mn.copy(e.boundingSphere).applyMatrix4(t.matrixWorld),this.intersectsSphere(mn)}intersectsSprite(t){return mn.center.set(0,0,0),mn.radius=.7071067811865476,mn.applyMatrix4(t.matrixWorld),this.intersectsSphere(mn)}intersectsSphere(t){const e=this.planes,i=t.center,n=-t.radius;for(let t=0;t<6;t++){if(e[t].distanceToPoint(i)<n)return!1}return!0}intersectsBox(t){const e=this.planes;for(let i=0;i<6;i++){const n=e[i];if(fn.x=n.normal.x>0?t.max.x:t.min.x,fn.y=n.normal.y>0?t.max.y:t.min.y,fn.z=n.normal.z>0?t.max.z:t.min.z,n.distanceToPoint(fn)<0)return!1}return!0}containsPoint(t){const e=this.planes;for(let i=0;i<6;i++)if(e[i].distanceToPoint(t)<0)return!1;return!0}clone(){return(new this.constructor).copy(this)}}function vn(){let t=null,e=!1,i=null,n=null;function r(e,s){i(e,s),n=t.requestAnimationFrame(r)}return{start:function(){!0!==e&&null!==i&&(n=t.requestAnimationFrame(r),e=!0)},stop:function(){t.cancelAnimationFrame(n),e=!1},setAnimationLoop:function(t){i=t},setContext:function(e){t=e}}}function xn(t,e){const i=e.isWebGL2,n=new WeakMap;return{get:function(t){return t.isInterleavedBufferAttribute&&(t=t.data),n.get(t)},remove:function(e){e.isInterleavedBufferAttribute&&(e=e.data);const i=n.get(e);i&&(t.deleteBuffer(i.buffer),n.delete(e))},update:function(e,r){if(e.isGLBufferAttribute){const t=n.get(e);return void((!t||t.version<e.version)&&n.set(e,{buffer:e.buffer,type:e.type,bytesPerElement:e.elementSize,version:e.version}))}e.isInterleavedBufferAttribute&&(e=e.data);const s=n.get(e);void 0===s?n.set(e,function(e,n){const r=e.array,s=e.usage,a=t.createBuffer();let o;if(t.bindBuffer(n,a),t.bufferData(n,r,s),e.onUploadCallback(),r instanceof Float32Array)o=5126;else if(r instanceof Uint16Array)if(e.isFloat16BufferAttribute){if(!i)throw new Error("THREE.WebGLAttributes: Usage of Float16BufferAttribute requires WebGL2.");o=5131}else o=5123;else if(r instanceof Int16Array)o=5122;else if(r instanceof Uint32Array)o=5125;else if(r instanceof Int32Array)o=5124;else if(r instanceof Int8Array)o=5120;else if(r instanceof Uint8Array)o=5121;else{if(!(r instanceof Uint8ClampedArray))throw new Error("THREE.WebGLAttributes: Unsupported buffer data format: "+r);o=5121}return{buffer:a,type:o,bytesPerElement:r.BYTES_PER_ELEMENT,version:e.version}}(e,r)):s.version<e.version&&(!function(e,n,r){const s=n.array,a=n.updateRange;t.bindBuffer(r,e),-1===a.count?t.bufferSubData(r,0,s):(i?t.bufferSubData(r,a.offset*s.BYTES_PER_ELEMENT,s,a.offset,a.count):t.bufferSubData(r,a.offset*s.BYTES_PER_ELEMENT,s.subarray(a.offset,a.offset+a.count)),a.count=-1)}(s.buffer,e,r),s.version=e.version)}}}class _n extends Di{constructor(t=1,e=1,i=1,n=1){super(),this.type="PlaneGeometry",this.parameters={width:t,height:e,widthSegments:i,heightSegments:n};const r=t/2,s=e/2,a=Math.floor(i),o=Math.floor(n),l=a+1,c=o+1,h=t/a,u=e/o,d=[],p=[],m=[],f=[];for(let t=0;t<c;t++){const e=t*u-s;for(let i=0;i<l;i++){const n=i*h-r;p.push(n,-e,0),m.push(0,0,1),f.push(i/a),f.push(1-t/o)}}for(let t=0;t<o;t++)for(let e=0;e<a;e++){const i=e+l*t,n=e+l*(t+1),r=e+1+l*(t+1),s=e+1+l*t;d.push(i,n,s),d.push(n,r,s)}this.setIndex(d),this.setAttribute("position",new wi(p,3)),this.setAttribute("normal",new wi(m,3)),this.setAttribute("uv",new wi(f,2))}static fromJSON(t){return new _n(t.width,t.height,t.widthSegments,t.heightSegments)}}const yn={alphamap_fragment:"#ifdef USE_ALPHAMAP\n\tdiffuseColor.a *= texture2D( alphaMap, vUv ).g;\n#endif",alphamap_pars_fragment:"#ifdef USE_ALPHAMAP\n\tuniform sampler2D alphaMap;\n#endif",alphatest_fragment:"#ifdef USE_ALPHATEST\n\tif ( diffuseColor.a < alphaTest ) discard;\n#endif",alphatest_pars_fragment:"#ifdef USE_ALPHATEST\n\tuniform float alphaTest;\n#endif",aomap_fragment:"#ifdef USE_AOMAP\n\tfloat ambientOcclusion = ( texture2D( aoMap, vUv2 ).r - 1.0 ) * aoMapIntensity + 1.0;\n\treflectedLight.indirectDiffuse *= ambientOcclusion;\n\t#if defined( USE_ENVMAP ) && defined( STANDARD )\n\t\tfloat dotNV = saturate( dot( geometry.normal, geometry.viewDir ) );\n\t\treflectedLight.indirectSpecular *= computeSpecularOcclusion( dotNV, ambientOcclusion, material.roughness );\n\t#endif\n#endif",aomap_pars_fragment:"#ifdef USE_AOMAP\n\tuniform sampler2D aoMap;\n\tuniform float aoMapIntensity;\n#endif",begin_vertex:"vec3 transformed = vec3( position );",beginnormal_vertex:"vec3 objectNormal = vec3( normal );\n#ifdef USE_TANGENT\n\tvec3 objectTangent = vec3( tangent.xyz );\n#endif",bsdfs:"vec3 BRDF_Lambert( const in vec3 diffuseColor ) {\n\treturn RECIPROCAL_PI * diffuseColor;\n}\nvec3 F_Schlick( const in vec3 f0, const in float f90, const in float dotVH ) {\n\tfloat fresnel = exp2( ( - 5.55473 * dotVH - 6.98316 ) * dotVH );\n\treturn f0 * ( 1.0 - fresnel ) + ( f90 * fresnel );\n}\nfloat F_Schlick( const in float f0, const in float f90, const in float dotVH ) {\n\tfloat fresnel = exp2( ( - 5.55473 * dotVH - 6.98316 ) * dotVH );\n\treturn f0 * ( 1.0 - fresnel ) + ( f90 * fresnel );\n}\nvec3 Schlick_to_F0( const in vec3 f, const in float f90, const in float dotVH ) {\n\t\tfloat x = clamp( 1.0 - dotVH, 0.0, 1.0 );\n\t\tfloat x2 = x * x;\n\t\tfloat x5 = clamp( x * x2 * x2, 0.0, 0.9999 );\n\t\treturn ( f - vec3( f90 ) * x5 ) / ( 1.0 - x5 );\n}\nfloat V_GGX_SmithCorrelated( const in float alpha, const in float dotNL, const in float dotNV ) {\n\tfloat a2 = pow2( alpha );\n\tfloat gv = dotNL * sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNV ) );\n\tfloat gl = dotNV * sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNL ) );\n\treturn 0.5 / max( gv + gl, EPSILON );\n}\nfloat D_GGX( const in float alpha, const in float dotNH ) {\n\tfloat a2 = pow2( alpha );\n\tfloat denom = pow2( dotNH ) * ( a2 - 1.0 ) + 1.0;\n\treturn RECIPROCAL_PI * a2 / pow2( denom );\n}\nvec3 BRDF_GGX( const in vec3 lightDir, const in vec3 viewDir, const in vec3 normal, const in vec3 f0, const in float f90, const in float roughness ) {\n\tfloat alpha = pow2( roughness );\n\tvec3 halfDir = normalize( lightDir + viewDir );\n\tfloat dotNL = saturate( dot( normal, lightDir ) );\n\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\tfloat dotNH = saturate( dot( normal, halfDir ) );\n\tfloat dotVH = saturate( dot( viewDir, halfDir ) );\n\tvec3 F = F_Schlick( f0, f90, dotVH );\n\tfloat V = V_GGX_SmithCorrelated( alpha, dotNL, dotNV );\n\tfloat D = D_GGX( alpha, dotNH );\n\treturn F * ( V * D );\n}\n#ifdef USE_IRIDESCENCE\n\tvec3 BRDF_GGX_Iridescence( const in vec3 lightDir, const in vec3 viewDir, const in vec3 normal, const in vec3 f0, const in float f90, const in float iridescence, const in vec3 iridescenceFresnel, const in float roughness ) {\n\t\tfloat alpha = pow2( roughness );\n\t\tvec3 halfDir = normalize( lightDir + viewDir );\n\t\tfloat dotNL = saturate( dot( normal, lightDir ) );\n\t\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\t\tfloat dotNH = saturate( dot( normal, halfDir ) );\n\t\tfloat dotVH = saturate( dot( viewDir, halfDir ) );\n\t\tvec3 F = mix( F_Schlick( f0, f90, dotVH ), iridescenceFresnel, iridescence );\n\t\tfloat V = V_GGX_SmithCorrelated( alpha, dotNL, dotNV );\n\t\tfloat D = D_GGX( alpha, dotNH );\n\t\treturn F * ( V * D );\n\t}\n#endif\nvec2 LTC_Uv( const in vec3 N, const in vec3 V, const in float roughness ) {\n\tconst float LUT_SIZE = 64.0;\n\tconst float LUT_SCALE = ( LUT_SIZE - 1.0 ) / LUT_SIZE;\n\tconst float LUT_BIAS = 0.5 / LUT_SIZE;\n\tfloat dotNV = saturate( dot( N, V ) );\n\tvec2 uv = vec2( roughness, sqrt( 1.0 - dotNV ) );\n\tuv = uv * LUT_SCALE + LUT_BIAS;\n\treturn uv;\n}\nfloat LTC_ClippedSphereFormFactor( const in vec3 f ) {\n\tfloat l = length( f );\n\treturn max( ( l * l + f.z ) / ( l + 1.0 ), 0.0 );\n}\nvec3 LTC_EdgeVectorFormFactor( const in vec3 v1, const in vec3 v2 ) {\n\tfloat x = dot( v1, v2 );\n\tfloat y = abs( x );\n\tfloat a = 0.8543985 + ( 0.4965155 + 0.0145206 * y ) * y;\n\tfloat b = 3.4175940 + ( 4.1616724 + y ) * y;\n\tfloat v = a / b;\n\tfloat theta_sintheta = ( x > 0.0 ) ? v : 0.5 * inversesqrt( max( 1.0 - x * x, 1e-7 ) ) - v;\n\treturn cross( v1, v2 ) * theta_sintheta;\n}\nvec3 LTC_Evaluate( const in vec3 N, const in vec3 V, const in vec3 P, const in mat3 mInv, const in vec3 rectCoords[ 4 ] ) {\n\tvec3 v1 = rectCoords[ 1 ] - rectCoords[ 0 ];\n\tvec3 v2 = rectCoords[ 3 ] - rectCoords[ 0 ];\n\tvec3 lightNormal = cross( v1, v2 );\n\tif( dot( lightNormal, P - rectCoords[ 0 ] ) < 0.0 ) return vec3( 0.0 );\n\tvec3 T1, T2;\n\tT1 = normalize( V - N * dot( V, N ) );\n\tT2 = - cross( N, T1 );\n\tmat3 mat = mInv * transposeMat3( mat3( T1, T2, N ) );\n\tvec3 coords[ 4 ];\n\tcoords[ 0 ] = mat * ( rectCoords[ 0 ] - P );\n\tcoords[ 1 ] = mat * ( rectCoords[ 1 ] - P );\n\tcoords[ 2 ] = mat * ( rectCoords[ 2 ] - P );\n\tcoords[ 3 ] = mat * ( rectCoords[ 3 ] - P );\n\tcoords[ 0 ] = normalize( coords[ 0 ] );\n\tcoords[ 1 ] = normalize( coords[ 1 ] );\n\tcoords[ 2 ] = normalize( coords[ 2 ] );\n\tcoords[ 3 ] = normalize( coords[ 3 ] );\n\tvec3 vectorFormFactor = vec3( 0.0 );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 0 ], coords[ 1 ] );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 1 ], coords[ 2 ] );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 2 ], coords[ 3 ] );\n\tvectorFormFactor += LTC_EdgeVectorFormFactor( coords[ 3 ], coords[ 0 ] );\n\tfloat result = LTC_ClippedSphereFormFactor( vectorFormFactor );\n\treturn vec3( result );\n}\nfloat G_BlinnPhong_Implicit( ) {\n\treturn 0.25;\n}\nfloat D_BlinnPhong( const in float shininess, const in float dotNH ) {\n\treturn RECIPROCAL_PI * ( shininess * 0.5 + 1.0 ) * pow( dotNH, shininess );\n}\nvec3 BRDF_BlinnPhong( const in vec3 lightDir, const in vec3 viewDir, const in vec3 normal, const in vec3 specularColor, const in float shininess ) {\n\tvec3 halfDir = normalize( lightDir + viewDir );\n\tfloat dotNH = saturate( dot( normal, halfDir ) );\n\tfloat dotVH = saturate( dot( viewDir, halfDir ) );\n\tvec3 F = F_Schlick( specularColor, 1.0, dotVH );\n\tfloat G = G_BlinnPhong_Implicit( );\n\tfloat D = D_BlinnPhong( shininess, dotNH );\n\treturn F * ( G * D );\n}\n#if defined( USE_SHEEN )\nfloat D_Charlie( float roughness, float dotNH ) {\n\tfloat alpha = pow2( roughness );\n\tfloat invAlpha = 1.0 / alpha;\n\tfloat cos2h = dotNH * dotNH;\n\tfloat sin2h = max( 1.0 - cos2h, 0.0078125 );\n\treturn ( 2.0 + invAlpha ) * pow( sin2h, invAlpha * 0.5 ) / ( 2.0 * PI );\n}\nfloat V_Neubelt( float dotNV, float dotNL ) {\n\treturn saturate( 1.0 / ( 4.0 * ( dotNL + dotNV - dotNL * dotNV ) ) );\n}\nvec3 BRDF_Sheen( const in vec3 lightDir, const in vec3 viewDir, const in vec3 normal, vec3 sheenColor, const in float sheenRoughness ) {\n\tvec3 halfDir = normalize( lightDir + viewDir );\n\tfloat dotNL = saturate( dot( normal, lightDir ) );\n\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\tfloat dotNH = saturate( dot( normal, halfDir ) );\n\tfloat D = D_Charlie( sheenRoughness, dotNH );\n\tfloat V = V_Neubelt( dotNV, dotNL );\n\treturn sheenColor * ( D * V );\n}\n#endif",iridescence_fragment:"#ifdef USE_IRIDESCENCE\n\tconst mat3 XYZ_TO_REC709 = mat3(\n\t\t 3.2404542, -0.9692660,\t0.0556434,\n\t\t-1.5371385,\t1.8760108, -0.2040259,\n\t\t-0.4985314,\t0.0415560,\t1.0572252\n\t);\n\tvec3 Fresnel0ToIor( vec3 fresnel0 ) {\n\t\tvec3 sqrtF0 = sqrt( fresnel0 );\n\t\treturn ( vec3( 1.0 ) + sqrtF0 ) / ( vec3( 1.0 ) - sqrtF0 );\n\t}\n\tvec3 IorToFresnel0( vec3 transmittedIor, float incidentIor ) {\n\t\treturn pow2( ( transmittedIor - vec3( incidentIor ) ) / ( transmittedIor + vec3( incidentIor ) ) );\n\t}\n\tfloat IorToFresnel0( float transmittedIor, float incidentIor ) {\n\t\treturn pow2( ( transmittedIor - incidentIor ) / ( transmittedIor + incidentIor ));\n\t}\n\tvec3 evalSensitivity( float OPD, vec3 shift ) {\n\t\tfloat phase = 2.0 * PI * OPD * 1.0e-9;\n\t\tvec3 val = vec3( 5.4856e-13, 4.4201e-13, 5.2481e-13 );\n\t\tvec3 pos = vec3( 1.6810e+06, 1.7953e+06, 2.2084e+06 );\n\t\tvec3 var = vec3( 4.3278e+09, 9.3046e+09, 6.6121e+09 );\n\t\tvec3 xyz = val * sqrt( 2.0 * PI * var ) * cos( pos * phase + shift ) * exp( - pow2( phase ) * var );\n\t\txyz.x += 9.7470e-14 * sqrt( 2.0 * PI * 4.5282e+09 ) * cos( 2.2399e+06 * phase + shift[ 0 ] ) * exp( - 4.5282e+09 * pow2( phase ) );\n\t\txyz /= 1.0685e-7;\n\t\tvec3 rgb = XYZ_TO_REC709 * xyz;\n\t\treturn rgb;\n\t}\n\tvec3 evalIridescence( float outsideIOR, float eta2, float cosTheta1, float thinFilmThickness, vec3 baseF0 ) {\n\t\tvec3 I;\n\t\tfloat iridescenceIOR = mix( outsideIOR, eta2, smoothstep( 0.0, 0.03, thinFilmThickness ) );\n\t\tfloat sinTheta2Sq = pow2( outsideIOR / iridescenceIOR ) * ( 1.0 - pow2( cosTheta1 ) );\n\t\tfloat cosTheta2Sq = 1.0 - sinTheta2Sq;\n\t\tif ( cosTheta2Sq < 0.0 ) {\n\t\t\t return vec3( 1.0 );\n\t\t}\n\t\tfloat cosTheta2 = sqrt( cosTheta2Sq );\n\t\tfloat R0 = IorToFresnel0( iridescenceIOR, outsideIOR );\n\t\tfloat R12 = F_Schlick( R0, 1.0, cosTheta1 );\n\t\tfloat R21 = R12;\n\t\tfloat T121 = 1.0 - R12;\n\t\tfloat phi12 = 0.0;\n\t\tif ( iridescenceIOR < outsideIOR ) phi12 = PI;\n\t\tfloat phi21 = PI - phi12;\n\t\tvec3 baseIOR = Fresnel0ToIor( clamp( baseF0, 0.0, 0.9999 ) );\t\tvec3 R1 = IorToFresnel0( baseIOR, iridescenceIOR );\n\t\tvec3 R23 = F_Schlick( R1, 1.0, cosTheta2 );\n\t\tvec3 phi23 = vec3( 0.0 );\n\t\tif ( baseIOR[ 0 ] < iridescenceIOR ) phi23[ 0 ] = PI;\n\t\tif ( baseIOR[ 1 ] < iridescenceIOR ) phi23[ 1 ] = PI;\n\t\tif ( baseIOR[ 2 ] < iridescenceIOR ) phi23[ 2 ] = PI;\n\t\tfloat OPD = 2.0 * iridescenceIOR * thinFilmThickness * cosTheta2;\n\t\tvec3 phi = vec3( phi21 ) + phi23;\n\t\tvec3 R123 = clamp( R12 * R23, 1e-5, 0.9999 );\n\t\tvec3 r123 = sqrt( R123 );\n\t\tvec3 Rs = pow2( T121 ) * R23 / ( vec3( 1.0 ) - R123 );\n\t\tvec3 C0 = R12 + Rs;\n\t\tI = C0;\n\t\tvec3 Cm = Rs - T121;\n\t\tfor ( int m = 1; m <= 2; ++ m ) {\n\t\t\tCm *= r123;\n\t\t\tvec3 Sm = 2.0 * evalSensitivity( float( m ) * OPD, float( m ) * phi );\n\t\t\tI += Cm * Sm;\n\t\t}\n\t\treturn max( I, vec3( 0.0 ) );\n\t}\n#endif",bumpmap_pars_fragment:"#ifdef USE_BUMPMAP\n\tuniform sampler2D bumpMap;\n\tuniform float bumpScale;\n\tvec2 dHdxy_fwd() {\n\t\tvec2 dSTdx = dFdx( vUv );\n\t\tvec2 dSTdy = dFdy( vUv );\n\t\tfloat Hll = bumpScale * texture2D( bumpMap, vUv ).x;\n\t\tfloat dBx = bumpScale * texture2D( bumpMap, vUv + dSTdx ).x - Hll;\n\t\tfloat dBy = bumpScale * texture2D( bumpMap, vUv + dSTdy ).x - Hll;\n\t\treturn vec2( dBx, dBy );\n\t}\n\tvec3 perturbNormalArb( vec3 surf_pos, vec3 surf_norm, vec2 dHdxy, float faceDirection ) {\n\t\tvec3 vSigmaX = dFdx( surf_pos.xyz );\n\t\tvec3 vSigmaY = dFdy( surf_pos.xyz );\n\t\tvec3 vN = surf_norm;\n\t\tvec3 R1 = cross( vSigmaY, vN );\n\t\tvec3 R2 = cross( vN, vSigmaX );\n\t\tfloat fDet = dot( vSigmaX, R1 ) * faceDirection;\n\t\tvec3 vGrad = sign( fDet ) * ( dHdxy.x * R1 + dHdxy.y * R2 );\n\t\treturn normalize( abs( fDet ) * surf_norm - vGrad );\n\t}\n#endif",clipping_planes_fragment:"#if NUM_CLIPPING_PLANES > 0\n\tvec4 plane;\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < UNION_CLIPPING_PLANES; i ++ ) {\n\t\tplane = clippingPlanes[ i ];\n\t\tif ( dot( vClipPosition, plane.xyz ) > plane.w ) discard;\n\t}\n\t#pragma unroll_loop_end\n\t#if UNION_CLIPPING_PLANES < NUM_CLIPPING_PLANES\n\t\tbool clipped = true;\n\t\t#pragma unroll_loop_start\n\t\tfor ( int i = UNION_CLIPPING_PLANES; i < NUM_CLIPPING_PLANES; i ++ ) {\n\t\t\tplane = clippingPlanes[ i ];\n\t\t\tclipped = ( dot( vClipPosition, plane.xyz ) > plane.w ) && clipped;\n\t\t}\n\t\t#pragma unroll_loop_end\n\t\tif ( clipped ) discard;\n\t#endif\n#endif",clipping_planes_pars_fragment:"#if NUM_CLIPPING_PLANES > 0\n\tvarying vec3 vClipPosition;\n\tuniform vec4 clippingPlanes[ NUM_CLIPPING_PLANES ];\n#endif",clipping_planes_pars_vertex:"#if NUM_CLIPPING_PLANES > 0\n\tvarying vec3 vClipPosition;\n#endif",clipping_planes_vertex:"#if NUM_CLIPPING_PLANES > 0\n\tvClipPosition = - mvPosition.xyz;\n#endif",color_fragment:"#if defined( USE_COLOR_ALPHA )\n\tdiffuseColor *= vColor;\n#elif defined( USE_COLOR )\n\tdiffuseColor.rgb *= vColor;\n#endif",color_pars_fragment:"#if defined( USE_COLOR_ALPHA )\n\tvarying vec4 vColor;\n#elif defined( USE_COLOR )\n\tvarying vec3 vColor;\n#endif",color_pars_vertex:"#if defined( USE_COLOR_ALPHA )\n\tvarying vec4 vColor;\n#elif defined( USE_COLOR ) || defined( USE_INSTANCING_COLOR )\n\tvarying vec3 vColor;\n#endif",color_vertex:"#if defined( USE_COLOR_ALPHA )\n\tvColor = vec4( 1.0 );\n#elif defined( USE_COLOR ) || defined( USE_INSTANCING_COLOR )\n\tvColor = vec3( 1.0 );\n#endif\n#ifdef USE_COLOR\n\tvColor *= color;\n#endif\n#ifdef USE_INSTANCING_COLOR\n\tvColor.xyz *= instanceColor.xyz;\n#endif",common:"#define PI 3.141592653589793\n#define PI2 6.283185307179586\n#define PI_HALF 1.5707963267948966\n#define RECIPROCAL_PI 0.3183098861837907\n#define RECIPROCAL_PI2 0.15915494309189535\n#define EPSILON 1e-6\n#ifndef saturate\n#define saturate( a ) clamp( a, 0.0, 1.0 )\n#endif\n#define whiteComplement( a ) ( 1.0 - saturate( a ) )\nfloat pow2( const in float x ) { return x*x; }\nvec3 pow2( const in vec3 x ) { return x*x; }\nfloat pow3( const in float x ) { return x*x*x; }\nfloat pow4( const in float x ) { float x2 = x*x; return x2*x2; }\nfloat max3( const in vec3 v ) { return max( max( v.x, v.y ), v.z ); }\nfloat average( const in vec3 v ) { return dot( v, vec3( 0.3333333 ) ); }\nhighp float rand( const in vec2 uv ) {\n\tconst highp float a = 12.9898, b = 78.233, c = 43758.5453;\n\thighp float dt = dot( uv.xy, vec2( a,b ) ), sn = mod( dt, PI );\n\treturn fract( sin( sn ) * c );\n}\n#ifdef HIGH_PRECISION\n\tfloat precisionSafeLength( vec3 v ) { return length( v ); }\n#else\n\tfloat precisionSafeLength( vec3 v ) {\n\t\tfloat maxComponent = max3( abs( v ) );\n\t\treturn length( v / maxComponent ) * maxComponent;\n\t}\n#endif\nstruct IncidentLight {\n\tvec3 color;\n\tvec3 direction;\n\tbool visible;\n};\nstruct ReflectedLight {\n\tvec3 directDiffuse;\n\tvec3 directSpecular;\n\tvec3 indirectDiffuse;\n\tvec3 indirectSpecular;\n};\nstruct GeometricContext {\n\tvec3 position;\n\tvec3 normal;\n\tvec3 viewDir;\n#ifdef USE_CLEARCOAT\n\tvec3 clearcoatNormal;\n#endif\n};\nvec3 transformDirection( in vec3 dir, in mat4 matrix ) {\n\treturn normalize( ( matrix * vec4( dir, 0.0 ) ).xyz );\n}\nvec3 inverseTransformDirection( in vec3 dir, in mat4 matrix ) {\n\treturn normalize( ( vec4( dir, 0.0 ) * matrix ).xyz );\n}\nmat3 transposeMat3( const in mat3 m ) {\n\tmat3 tmp;\n\ttmp[ 0 ] = vec3( m[ 0 ].x, m[ 1 ].x, m[ 2 ].x );\n\ttmp[ 1 ] = vec3( m[ 0 ].y, m[ 1 ].y, m[ 2 ].y );\n\ttmp[ 2 ] = vec3( m[ 0 ].z, m[ 1 ].z, m[ 2 ].z );\n\treturn tmp;\n}\nfloat luminance( const in vec3 rgb ) {\n\tconst vec3 weights = vec3( 0.2126729, 0.7151522, 0.0721750 );\n\treturn dot( weights, rgb );\n}\nbool isPerspectiveMatrix( mat4 m ) {\n\treturn m[ 2 ][ 3 ] == - 1.0;\n}\nvec2 equirectUv( in vec3 dir ) {\n\tfloat u = atan( dir.z, dir.x ) * RECIPROCAL_PI2 + 0.5;\n\tfloat v = asin( clamp( dir.y, - 1.0, 1.0 ) ) * RECIPROCAL_PI + 0.5;\n\treturn vec2( u, v );\n}",cube_uv_reflection_fragment:"#ifdef ENVMAP_TYPE_CUBE_UV\n\t#define cubeUV_minMipLevel 4.0\n\t#define cubeUV_minTileSize 16.0\n\tfloat getFace( vec3 direction ) {\n\t\tvec3 absDirection = abs( direction );\n\t\tfloat face = - 1.0;\n\t\tif ( absDirection.x > absDirection.z ) {\n\t\t\tif ( absDirection.x > absDirection.y )\n\t\t\t\tface = direction.x > 0.0 ? 0.0 : 3.0;\n\t\t\telse\n\t\t\t\tface = direction.y > 0.0 ? 1.0 : 4.0;\n\t\t} else {\n\t\t\tif ( absDirection.z > absDirection.y )\n\t\t\t\tface = direction.z > 0.0 ? 2.0 : 5.0;\n\t\t\telse\n\t\t\t\tface = direction.y > 0.0 ? 1.0 : 4.0;\n\t\t}\n\t\treturn face;\n\t}\n\tvec2 getUV( vec3 direction, float face ) {\n\t\tvec2 uv;\n\t\tif ( face == 0.0 ) {\n\t\t\tuv = vec2( direction.z, direction.y ) / abs( direction.x );\n\t\t} else if ( face == 1.0 ) {\n\t\t\tuv = vec2( - direction.x, - direction.z ) / abs( direction.y );\n\t\t} else if ( face == 2.0 ) {\n\t\t\tuv = vec2( - direction.x, direction.y ) / abs( direction.z );\n\t\t} else if ( face == 3.0 ) {\n\t\t\tuv = vec2( - direction.z, direction.y ) / abs( direction.x );\n\t\t} else if ( face == 4.0 ) {\n\t\t\tuv = vec2( - direction.x, direction.z ) / abs( direction.y );\n\t\t} else {\n\t\t\tuv = vec2( direction.x, direction.y ) / abs( direction.z );\n\t\t}\n\t\treturn 0.5 * ( uv + 1.0 );\n\t}\n\tvec3 bilinearCubeUV( sampler2D envMap, vec3 direction, float mipInt ) {\n\t\tfloat face = getFace( direction );\n\t\tfloat filterInt = max( cubeUV_minMipLevel - mipInt, 0.0 );\n\t\tmipInt = max( mipInt, cubeUV_minMipLevel );\n\t\tfloat faceSize = exp2( mipInt );\n\t\tvec2 uv = getUV( direction, face ) * ( faceSize - 2.0 ) + 1.0;\n\t\tif ( face > 2.0 ) {\n\t\t\tuv.y += faceSize;\n\t\t\tface -= 3.0;\n\t\t}\n\t\tuv.x += face * faceSize;\n\t\tuv.x += filterInt * 3.0 * cubeUV_minTileSize;\n\t\tuv.y += 4.0 * ( exp2( CUBEUV_MAX_MIP ) - faceSize );\n\t\tuv.x *= CUBEUV_TEXEL_WIDTH;\n\t\tuv.y *= CUBEUV_TEXEL_HEIGHT;\n\t\t#ifdef texture2DGradEXT\n\t\t\treturn texture2DGradEXT( envMap, uv, vec2( 0.0 ), vec2( 0.0 ) ).rgb;\n\t\t#else\n\t\t\treturn texture2D( envMap, uv ).rgb;\n\t\t#endif\n\t}\n\t#define cubeUV_r0 1.0\n\t#define cubeUV_v0 0.339\n\t#define cubeUV_m0 - 2.0\n\t#define cubeUV_r1 0.8\n\t#define cubeUV_v1 0.276\n\t#define cubeUV_m1 - 1.0\n\t#define cubeUV_r4 0.4\n\t#define cubeUV_v4 0.046\n\t#define cubeUV_m4 2.0\n\t#define cubeUV_r5 0.305\n\t#define cubeUV_v5 0.016\n\t#define cubeUV_m5 3.0\n\t#define cubeUV_r6 0.21\n\t#define cubeUV_v6 0.0038\n\t#define cubeUV_m6 4.0\n\tfloat roughnessToMip( float roughness ) {\n\t\tfloat mip = 0.0;\n\t\tif ( roughness >= cubeUV_r1 ) {\n\t\t\tmip = ( cubeUV_r0 - roughness ) * ( cubeUV_m1 - cubeUV_m0 ) / ( cubeUV_r0 - cubeUV_r1 ) + cubeUV_m0;\n\t\t} else if ( roughness >= cubeUV_r4 ) {\n\t\t\tmip = ( cubeUV_r1 - roughness ) * ( cubeUV_m4 - cubeUV_m1 ) / ( cubeUV_r1 - cubeUV_r4 ) + cubeUV_m1;\n\t\t} else if ( roughness >= cubeUV_r5 ) {\n\t\t\tmip = ( cubeUV_r4 - roughness ) * ( cubeUV_m5 - cubeUV_m4 ) / ( cubeUV_r4 - cubeUV_r5 ) + cubeUV_m4;\n\t\t} else if ( roughness >= cubeUV_r6 ) {\n\t\t\tmip = ( cubeUV_r5 - roughness ) * ( cubeUV_m6 - cubeUV_m5 ) / ( cubeUV_r5 - cubeUV_r6 ) + cubeUV_m5;\n\t\t} else {\n\t\t\tmip = - 2.0 * log2( 1.16 * roughness );\t\t}\n\t\treturn mip;\n\t}\n\tvec4 textureCubeUV( sampler2D envMap, vec3 sampleDir, float roughness ) {\n\t\tfloat mip = clamp( roughnessToMip( roughness ), cubeUV_m0, CUBEUV_MAX_MIP );\n\t\tfloat mipF = fract( mip );\n\t\tfloat mipInt = floor( mip );\n\t\tvec3 color0 = bilinearCubeUV( envMap, sampleDir, mipInt );\n\t\tif ( mipF == 0.0 ) {\n\t\t\treturn vec4( color0, 1.0 );\n\t\t} else {\n\t\t\tvec3 color1 = bilinearCubeUV( envMap, sampleDir, mipInt + 1.0 );\n\t\t\treturn vec4( mix( color0, color1, mipF ), 1.0 );\n\t\t}\n\t}\n#endif",defaultnormal_vertex:"vec3 transformedNormal = objectNormal;\n#ifdef USE_INSTANCING\n\tmat3 m = mat3( instanceMatrix );\n\ttransformedNormal /= vec3( dot( m[ 0 ], m[ 0 ] ), dot( m[ 1 ], m[ 1 ] ), dot( m[ 2 ], m[ 2 ] ) );\n\ttransformedNormal = m * transformedNormal;\n#endif\ntransformedNormal = normalMatrix * transformedNormal;\n#ifdef FLIP_SIDED\n\ttransformedNormal = - transformedNormal;\n#endif\n#ifdef USE_TANGENT\n\tvec3 transformedTangent = ( modelViewMatrix * vec4( objectTangent, 0.0 ) ).xyz;\n\t#ifdef FLIP_SIDED\n\t\ttransformedTangent = - transformedTangent;\n\t#endif\n#endif",displacementmap_pars_vertex:"#ifdef USE_DISPLACEMENTMAP\n\tuniform sampler2D displacementMap;\n\tuniform float displacementScale;\n\tuniform float displacementBias;\n#endif",displacementmap_vertex:"#ifdef USE_DISPLACEMENTMAP\n\ttransformed += normalize( objectNormal ) * ( texture2D( displacementMap, vUv ).x * displacementScale + displacementBias );\n#endif",emissivemap_fragment:"#ifdef USE_EMISSIVEMAP\n\tvec4 emissiveColor = texture2D( emissiveMap, vUv );\n\ttotalEmissiveRadiance *= emissiveColor.rgb;\n#endif",emissivemap_pars_fragment:"#ifdef USE_EMISSIVEMAP\n\tuniform sampler2D emissiveMap;\n#endif",encodings_fragment:"gl_FragColor = linearToOutputTexel( gl_FragColor );",encodings_pars_fragment:"vec4 LinearToLinear( in vec4 value ) {\n\treturn value;\n}\nvec4 LinearTosRGB( in vec4 value ) {\n\treturn vec4( mix( pow( value.rgb, vec3( 0.41666 ) ) * 1.055 - vec3( 0.055 ), value.rgb * 12.92, vec3( lessThanEqual( value.rgb, vec3( 0.0031308 ) ) ) ), value.a );\n}",envmap_fragment:"#ifdef USE_ENVMAP\n\t#ifdef ENV_WORLDPOS\n\t\tvec3 cameraToFrag;\n\t\tif ( isOrthographic ) {\n\t\t\tcameraToFrag = normalize( vec3( - viewMatrix[ 0 ][ 2 ], - viewMatrix[ 1 ][ 2 ], - viewMatrix[ 2 ][ 2 ] ) );\n\t\t} else {\n\t\t\tcameraToFrag = normalize( vWorldPosition - cameraPosition );\n\t\t}\n\t\tvec3 worldNormal = inverseTransformDirection( normal, viewMatrix );\n\t\t#ifdef ENVMAP_MODE_REFLECTION\n\t\t\tvec3 reflectVec = reflect( cameraToFrag, worldNormal );\n\t\t#else\n\t\t\tvec3 reflectVec = refract( cameraToFrag, worldNormal, refractionRatio );\n\t\t#endif\n\t#else\n\t\tvec3 reflectVec = vReflect;\n\t#endif\n\t#ifdef ENVMAP_TYPE_CUBE\n\t\tvec4 envColor = textureCube( envMap, vec3( flipEnvMap * reflectVec.x, reflectVec.yz ) );\n\t#else\n\t\tvec4 envColor = vec4( 0.0 );\n\t#endif\n\t#ifdef ENVMAP_BLENDING_MULTIPLY\n\t\toutgoingLight = mix( outgoingLight, outgoingLight * envColor.xyz, specularStrength * reflectivity );\n\t#elif defined( ENVMAP_BLENDING_MIX )\n\t\toutgoingLight = mix( outgoingLight, envColor.xyz, specularStrength * reflectivity );\n\t#elif defined( ENVMAP_BLENDING_ADD )\n\t\toutgoingLight += envColor.xyz * specularStrength * reflectivity;\n\t#endif\n#endif",envmap_common_pars_fragment:"#ifdef USE_ENVMAP\n\tuniform float envMapIntensity;\n\tuniform float flipEnvMap;\n\t#ifdef ENVMAP_TYPE_CUBE\n\t\tuniform samplerCube envMap;\n\t#else\n\t\tuniform sampler2D envMap;\n\t#endif\n\t\n#endif",envmap_pars_fragment:"#ifdef USE_ENVMAP\n\tuniform float reflectivity;\n\t#if defined( USE_BUMPMAP ) || defined( USE_NORMALMAP ) || defined( PHONG ) || defined( LAMBERT )\n\t\t#define ENV_WORLDPOS\n\t#endif\n\t#ifdef ENV_WORLDPOS\n\t\tvarying vec3 vWorldPosition;\n\t\tuniform float refractionRatio;\n\t#else\n\t\tvarying vec3 vReflect;\n\t#endif\n#endif",envmap_pars_vertex:"#ifdef USE_ENVMAP\n\t#if defined( USE_BUMPMAP ) || defined( USE_NORMALMAP ) || defined( PHONG ) || defined( LAMBERT )\n\t\t#define ENV_WORLDPOS\n\t#endif\n\t#ifdef ENV_WORLDPOS\n\t\t\n\t\tvarying vec3 vWorldPosition;\n\t#else\n\t\tvarying vec3 vReflect;\n\t\tuniform float refractionRatio;\n\t#endif\n#endif",envmap_physical_pars_fragment:"#if defined( USE_ENVMAP )\n\tvec3 getIBLIrradiance( const in vec3 normal ) {\n\t\t#if defined( ENVMAP_TYPE_CUBE_UV )\n\t\t\tvec3 worldNormal = inverseTransformDirection( normal, viewMatrix );\n\t\t\tvec4 envMapColor = textureCubeUV( envMap, worldNormal, 1.0 );\n\t\t\treturn PI * envMapColor.rgb * envMapIntensity;\n\t\t#else\n\t\t\treturn vec3( 0.0 );\n\t\t#endif\n\t}\n\tvec3 getIBLRadiance( const in vec3 viewDir, const in vec3 normal, const in float roughness ) {\n\t\t#if defined( ENVMAP_TYPE_CUBE_UV )\n\t\t\tvec3 reflectVec = reflect( - viewDir, normal );\n\t\t\treflectVec = normalize( mix( reflectVec, normal, roughness * roughness) );\n\t\t\treflectVec = inverseTransformDirection( reflectVec, viewMatrix );\n\t\t\tvec4 envMapColor = textureCubeUV( envMap, reflectVec, roughness );\n\t\t\treturn envMapColor.rgb * envMapIntensity;\n\t\t#else\n\t\t\treturn vec3( 0.0 );\n\t\t#endif\n\t}\n#endif",envmap_vertex:"#ifdef USE_ENVMAP\n\t#ifdef ENV_WORLDPOS\n\t\tvWorldPosition = worldPosition.xyz;\n\t#else\n\t\tvec3 cameraToVertex;\n\t\tif ( isOrthographic ) {\n\t\t\tcameraToVertex = normalize( vec3( - viewMatrix[ 0 ][ 2 ], - viewMatrix[ 1 ][ 2 ], - viewMatrix[ 2 ][ 2 ] ) );\n\t\t} else {\n\t\t\tcameraToVertex = normalize( worldPosition.xyz - cameraPosition );\n\t\t}\n\t\tvec3 worldNormal = inverseTransformDirection( transformedNormal, viewMatrix );\n\t\t#ifdef ENVMAP_MODE_REFLECTION\n\t\t\tvReflect = reflect( cameraToVertex, worldNormal );\n\t\t#else\n\t\t\tvReflect = refract( cameraToVertex, worldNormal, refractionRatio );\n\t\t#endif\n\t#endif\n#endif",fog_vertex:"#ifdef USE_FOG\n\tvFogDepth = - mvPosition.z;\n#endif",fog_pars_vertex:"#ifdef USE_FOG\n\tvarying float vFogDepth;\n#endif",fog_fragment:"#ifdef USE_FOG\n\t#ifdef FOG_EXP2\n\t\tfloat fogFactor = 1.0 - exp( - fogDensity * fogDensity * vFogDepth * vFogDepth );\n\t#else\n\t\tfloat fogFactor = smoothstep( fogNear, fogFar, vFogDepth );\n\t#endif\n\tgl_FragColor.rgb = mix( gl_FragColor.rgb, fogColor, fogFactor );\n#endif",fog_pars_fragment:"#ifdef USE_FOG\n\tuniform vec3 fogColor;\n\tvarying float vFogDepth;\n\t#ifdef FOG_EXP2\n\t\tuniform float fogDensity;\n\t#else\n\t\tuniform float fogNear;\n\t\tuniform float fogFar;\n\t#endif\n#endif",gradientmap_pars_fragment:"#ifdef USE_GRADIENTMAP\n\tuniform sampler2D gradientMap;\n#endif\nvec3 getGradientIrradiance( vec3 normal, vec3 lightDirection ) {\n\tfloat dotNL = dot( normal, lightDirection );\n\tvec2 coord = vec2( dotNL * 0.5 + 0.5, 0.0 );\n\t#ifdef USE_GRADIENTMAP\n\t\treturn vec3( texture2D( gradientMap, coord ).r );\n\t#else\n\t\tvec2 fw = fwidth( coord ) * 0.5;\n\t\treturn mix( vec3( 0.7 ), vec3( 1.0 ), smoothstep( 0.7 - fw.x, 0.7 + fw.x, coord.x ) );\n\t#endif\n}",lightmap_fragment:"#ifdef USE_LIGHTMAP\n\tvec4 lightMapTexel = texture2D( lightMap, vUv2 );\n\tvec3 lightMapIrradiance = lightMapTexel.rgb * lightMapIntensity;\n\treflectedLight.indirectDiffuse += lightMapIrradiance;\n#endif",lightmap_pars_fragment:"#ifdef USE_LIGHTMAP\n\tuniform sampler2D lightMap;\n\tuniform float lightMapIntensity;\n#endif",lights_lambert_fragment:"LambertMaterial material;\nmaterial.diffuseColor = diffuseColor.rgb;\nmaterial.specularStrength = specularStrength;",lights_lambert_pars_fragment:"varying vec3 vViewPosition;\nstruct LambertMaterial {\n\tvec3 diffuseColor;\n\tfloat specularStrength;\n};\nvoid RE_Direct_Lambert( const in IncidentLight directLight, const in GeometricContext geometry, const in LambertMaterial material, inout ReflectedLight reflectedLight ) {\n\tfloat dotNL = saturate( dot( geometry.normal, directLight.direction ) );\n\tvec3 irradiance = dotNL * directLight.color;\n\treflectedLight.directDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\nvoid RE_IndirectDiffuse_Lambert( const in vec3 irradiance, const in GeometricContext geometry, const in LambertMaterial material, inout ReflectedLight reflectedLight ) {\n\treflectedLight.indirectDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\n#define RE_Direct\t\t\t\tRE_Direct_Lambert\n#define RE_IndirectDiffuse\t\tRE_IndirectDiffuse_Lambert\n#define Material_LightProbeLOD( material )\t(0)",lights_pars_begin:"uniform bool receiveShadow;\nuniform vec3 ambientLightColor;\nuniform vec3 lightProbe[ 9 ];\nvec3 shGetIrradianceAt( in vec3 normal, in vec3 shCoefficients[ 9 ] ) {\n\tfloat x = normal.x, y = normal.y, z = normal.z;\n\tvec3 result = shCoefficients[ 0 ] * 0.886227;\n\tresult += shCoefficients[ 1 ] * 2.0 * 0.511664 * y;\n\tresult += shCoefficients[ 2 ] * 2.0 * 0.511664 * z;\n\tresult += shCoefficients[ 3 ] * 2.0 * 0.511664 * x;\n\tresult += shCoefficients[ 4 ] * 2.0 * 0.429043 * x * y;\n\tresult += shCoefficients[ 5 ] * 2.0 * 0.429043 * y * z;\n\tresult += shCoefficients[ 6 ] * ( 0.743125 * z * z - 0.247708 );\n\tresult += shCoefficients[ 7 ] * 2.0 * 0.429043 * x * z;\n\tresult += shCoefficients[ 8 ] * 0.429043 * ( x * x - y * y );\n\treturn result;\n}\nvec3 getLightProbeIrradiance( const in vec3 lightProbe[ 9 ], const in vec3 normal ) {\n\tvec3 worldNormal = inverseTransformDirection( normal, viewMatrix );\n\tvec3 irradiance = shGetIrradianceAt( worldNormal, lightProbe );\n\treturn irradiance;\n}\nvec3 getAmbientLightIrradiance( const in vec3 ambientLightColor ) {\n\tvec3 irradiance = ambientLightColor;\n\treturn irradiance;\n}\nfloat getDistanceAttenuation( const in float lightDistance, const in float cutoffDistance, const in float decayExponent ) {\n\t#if defined ( PHYSICALLY_CORRECT_LIGHTS )\n\t\tfloat distanceFalloff = 1.0 / max( pow( lightDistance, decayExponent ), 0.01 );\n\t\tif ( cutoffDistance > 0.0 ) {\n\t\t\tdistanceFalloff *= pow2( saturate( 1.0 - pow4( lightDistance / cutoffDistance ) ) );\n\t\t}\n\t\treturn distanceFalloff;\n\t#else\n\t\tif ( cutoffDistance > 0.0 && decayExponent > 0.0 ) {\n\t\t\treturn pow( saturate( - lightDistance / cutoffDistance + 1.0 ), decayExponent );\n\t\t}\n\t\treturn 1.0;\n\t#endif\n}\nfloat getSpotAttenuation( const in float coneCosine, const in float penumbraCosine, const in float angleCosine ) {\n\treturn smoothstep( coneCosine, penumbraCosine, angleCosine );\n}\n#if NUM_DIR_LIGHTS > 0\n\tstruct DirectionalLight {\n\t\tvec3 direction;\n\t\tvec3 color;\n\t};\n\tuniform DirectionalLight directionalLights[ NUM_DIR_LIGHTS ];\n\tvoid getDirectionalLightInfo( const in DirectionalLight directionalLight, const in GeometricContext geometry, out IncidentLight light ) {\n\t\tlight.color = directionalLight.color;\n\t\tlight.direction = directionalLight.direction;\n\t\tlight.visible = true;\n\t}\n#endif\n#if NUM_POINT_LIGHTS > 0\n\tstruct PointLight {\n\t\tvec3 position;\n\t\tvec3 color;\n\t\tfloat distance;\n\t\tfloat decay;\n\t};\n\tuniform PointLight pointLights[ NUM_POINT_LIGHTS ];\n\tvoid getPointLightInfo( const in PointLight pointLight, const in GeometricContext geometry, out IncidentLight light ) {\n\t\tvec3 lVector = pointLight.position - geometry.position;\n\t\tlight.direction = normalize( lVector );\n\t\tfloat lightDistance = length( lVector );\n\t\tlight.color = pointLight.color;\n\t\tlight.color *= getDistanceAttenuation( lightDistance, pointLight.distance, pointLight.decay );\n\t\tlight.visible = ( light.color != vec3( 0.0 ) );\n\t}\n#endif\n#if NUM_SPOT_LIGHTS > 0\n\tstruct SpotLight {\n\t\tvec3 position;\n\t\tvec3 direction;\n\t\tvec3 color;\n\t\tfloat distance;\n\t\tfloat decay;\n\t\tfloat coneCos;\n\t\tfloat penumbraCos;\n\t};\n\tuniform SpotLight spotLights[ NUM_SPOT_LIGHTS ];\n\tvoid getSpotLightInfo( const in SpotLight spotLight, const in GeometricContext geometry, out IncidentLight light ) {\n\t\tvec3 lVector = spotLight.position - geometry.position;\n\t\tlight.direction = normalize( lVector );\n\t\tfloat angleCos = dot( light.direction, spotLight.direction );\n\t\tfloat spotAttenuation = getSpotAttenuation( spotLight.coneCos, spotLight.penumbraCos, angleCos );\n\t\tif ( spotAttenuation > 0.0 ) {\n\t\t\tfloat lightDistance = length( lVector );\n\t\t\tlight.color = spotLight.color * spotAttenuation;\n\t\t\tlight.color *= getDistanceAttenuation( lightDistance, spotLight.distance, spotLight.decay );\n\t\t\tlight.visible = ( light.color != vec3( 0.0 ) );\n\t\t} else {\n\t\t\tlight.color = vec3( 0.0 );\n\t\t\tlight.visible = false;\n\t\t}\n\t}\n#endif\n#if NUM_RECT_AREA_LIGHTS > 0\n\tstruct RectAreaLight {\n\t\tvec3 color;\n\t\tvec3 position;\n\t\tvec3 halfWidth;\n\t\tvec3 halfHeight;\n\t};\n\tuniform sampler2D ltc_1;\tuniform sampler2D ltc_2;\n\tuniform RectAreaLight rectAreaLights[ NUM_RECT_AREA_LIGHTS ];\n#endif\n#if NUM_HEMI_LIGHTS > 0\n\tstruct HemisphereLight {\n\t\tvec3 direction;\n\t\tvec3 skyColor;\n\t\tvec3 groundColor;\n\t};\n\tuniform HemisphereLight hemisphereLights[ NUM_HEMI_LIGHTS ];\n\tvec3 getHemisphereLightIrradiance( const in HemisphereLight hemiLight, const in vec3 normal ) {\n\t\tfloat dotNL = dot( normal, hemiLight.direction );\n\t\tfloat hemiDiffuseWeight = 0.5 * dotNL + 0.5;\n\t\tvec3 irradiance = mix( hemiLight.groundColor, hemiLight.skyColor, hemiDiffuseWeight );\n\t\treturn irradiance;\n\t}\n#endif",lights_toon_fragment:"ToonMaterial material;\nmaterial.diffuseColor = diffuseColor.rgb;",lights_toon_pars_fragment:"varying vec3 vViewPosition;\nstruct ToonMaterial {\n\tvec3 diffuseColor;\n};\nvoid RE_Direct_Toon( const in IncidentLight directLight, const in GeometricContext geometry, const in ToonMaterial material, inout ReflectedLight reflectedLight ) {\n\tvec3 irradiance = getGradientIrradiance( geometry.normal, directLight.direction ) * directLight.color;\n\treflectedLight.directDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\nvoid RE_IndirectDiffuse_Toon( const in vec3 irradiance, const in GeometricContext geometry, const in ToonMaterial material, inout ReflectedLight reflectedLight ) {\n\treflectedLight.indirectDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\n#define RE_Direct\t\t\t\tRE_Direct_Toon\n#define RE_IndirectDiffuse\t\tRE_IndirectDiffuse_Toon\n#define Material_LightProbeLOD( material )\t(0)",lights_phong_fragment:"BlinnPhongMaterial material;\nmaterial.diffuseColor = diffuseColor.rgb;\nmaterial.specularColor = specular;\nmaterial.specularShininess = shininess;\nmaterial.specularStrength = specularStrength;",lights_phong_pars_fragment:"varying vec3 vViewPosition;\nstruct BlinnPhongMaterial {\n\tvec3 diffuseColor;\n\tvec3 specularColor;\n\tfloat specularShininess;\n\tfloat specularStrength;\n};\nvoid RE_Direct_BlinnPhong( const in IncidentLight directLight, const in GeometricContext geometry, const in BlinnPhongMaterial material, inout ReflectedLight reflectedLight ) {\n\tfloat dotNL = saturate( dot( geometry.normal, directLight.direction ) );\n\tvec3 irradiance = dotNL * directLight.color;\n\treflectedLight.directDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n\treflectedLight.directSpecular += irradiance * BRDF_BlinnPhong( directLight.direction, geometry.viewDir, geometry.normal, material.specularColor, material.specularShininess ) * material.specularStrength;\n}\nvoid RE_IndirectDiffuse_BlinnPhong( const in vec3 irradiance, const in GeometricContext geometry, const in BlinnPhongMaterial material, inout ReflectedLight reflectedLight ) {\n\treflectedLight.indirectDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\n#define RE_Direct\t\t\t\tRE_Direct_BlinnPhong\n#define RE_IndirectDiffuse\t\tRE_IndirectDiffuse_BlinnPhong\n#define Material_LightProbeLOD( material )\t(0)",lights_physical_fragment:"PhysicalMaterial material;\nmaterial.diffuseColor = diffuseColor.rgb * ( 1.0 - metalnessFactor );\nvec3 dxy = max( abs( dFdx( geometryNormal ) ), abs( dFdy( geometryNormal ) ) );\nfloat geometryRoughness = max( max( dxy.x, dxy.y ), dxy.z );\nmaterial.roughness = max( roughnessFactor, 0.0525 );material.roughness += geometryRoughness;\nmaterial.roughness = min( material.roughness, 1.0 );\n#ifdef IOR\n\tmaterial.ior = ior;\n\t#ifdef SPECULAR\n\t\tfloat specularIntensityFactor = specularIntensity;\n\t\tvec3 specularColorFactor = specularColor;\n\t\t#ifdef USE_SPECULARINTENSITYMAP\n\t\t\tspecularIntensityFactor *= texture2D( specularIntensityMap, vUv ).a;\n\t\t#endif\n\t\t#ifdef USE_SPECULARCOLORMAP\n\t\t\tspecularColorFactor *= texture2D( specularColorMap, vUv ).rgb;\n\t\t#endif\n\t\tmaterial.specularF90 = mix( specularIntensityFactor, 1.0, metalnessFactor );\n\t#else\n\t\tfloat specularIntensityFactor = 1.0;\n\t\tvec3 specularColorFactor = vec3( 1.0 );\n\t\tmaterial.specularF90 = 1.0;\n\t#endif\n\tmaterial.specularColor = mix( min( pow2( ( material.ior - 1.0 ) / ( material.ior + 1.0 ) ) * specularColorFactor, vec3( 1.0 ) ) * specularIntensityFactor, diffuseColor.rgb, metalnessFactor );\n#else\n\tmaterial.specularColor = mix( vec3( 0.04 ), diffuseColor.rgb, metalnessFactor );\n\tmaterial.specularF90 = 1.0;\n#endif\n#ifdef USE_CLEARCOAT\n\tmaterial.clearcoat = clearcoat;\n\tmaterial.clearcoatRoughness = clearcoatRoughness;\n\tmaterial.clearcoatF0 = vec3( 0.04 );\n\tmaterial.clearcoatF90 = 1.0;\n\t#ifdef USE_CLEARCOATMAP\n\t\tmaterial.clearcoat *= texture2D( clearcoatMap, vUv ).x;\n\t#endif\n\t#ifdef USE_CLEARCOAT_ROUGHNESSMAP\n\t\tmaterial.clearcoatRoughness *= texture2D( clearcoatRoughnessMap, vUv ).y;\n\t#endif\n\tmaterial.clearcoat = saturate( material.clearcoat );\tmaterial.clearcoatRoughness = max( material.clearcoatRoughness, 0.0525 );\n\tmaterial.clearcoatRoughness += geometryRoughness;\n\tmaterial.clearcoatRoughness = min( material.clearcoatRoughness, 1.0 );\n#endif\n#ifdef USE_IRIDESCENCE\n\tmaterial.iridescence = iridescence;\n\tmaterial.iridescenceIOR = iridescenceIOR;\n\t#ifdef USE_IRIDESCENCEMAP\n\t\tmaterial.iridescence *= texture2D( iridescenceMap, vUv ).r;\n\t#endif\n\t#ifdef USE_IRIDESCENCE_THICKNESSMAP\n\t\tmaterial.iridescenceThickness = (iridescenceThicknessMaximum - iridescenceThicknessMinimum) * texture2D( iridescenceThicknessMap, vUv ).g + iridescenceThicknessMinimum;\n\t#else\n\t\tmaterial.iridescenceThickness = iridescenceThicknessMaximum;\n\t#endif\n#endif\n#ifdef USE_SHEEN\n\tmaterial.sheenColor = sheenColor;\n\t#ifdef USE_SHEENCOLORMAP\n\t\tmaterial.sheenColor *= texture2D( sheenColorMap, vUv ).rgb;\n\t#endif\n\tmaterial.sheenRoughness = clamp( sheenRoughness, 0.07, 1.0 );\n\t#ifdef USE_SHEENROUGHNESSMAP\n\t\tmaterial.sheenRoughness *= texture2D( sheenRoughnessMap, vUv ).a;\n\t#endif\n#endif",lights_physical_pars_fragment:"struct PhysicalMaterial {\n\tvec3 diffuseColor;\n\tfloat roughness;\n\tvec3 specularColor;\n\tfloat specularF90;\n\t#ifdef USE_CLEARCOAT\n\t\tfloat clearcoat;\n\t\tfloat clearcoatRoughness;\n\t\tvec3 clearcoatF0;\n\t\tfloat clearcoatF90;\n\t#endif\n\t#ifdef USE_IRIDESCENCE\n\t\tfloat iridescence;\n\t\tfloat iridescenceIOR;\n\t\tfloat iridescenceThickness;\n\t\tvec3 iridescenceFresnel;\n\t\tvec3 iridescenceF0;\n\t#endif\n\t#ifdef USE_SHEEN\n\t\tvec3 sheenColor;\n\t\tfloat sheenRoughness;\n\t#endif\n\t#ifdef IOR\n\t\tfloat ior;\n\t#endif\n\t#ifdef USE_TRANSMISSION\n\t\tfloat transmission;\n\t\tfloat transmissionAlpha;\n\t\tfloat thickness;\n\t\tfloat attenuationDistance;\n\t\tvec3 attenuationColor;\n\t#endif\n};\nvec3 clearcoatSpecular = vec3( 0.0 );\nvec3 sheenSpecular = vec3( 0.0 );\nfloat IBLSheenBRDF( const in vec3 normal, const in vec3 viewDir, const in float roughness ) {\n\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\tfloat r2 = roughness * roughness;\n\tfloat a = roughness < 0.25 ? -339.2 * r2 + 161.4 * roughness - 25.9 : -8.48 * r2 + 14.3 * roughness - 9.95;\n\tfloat b = roughness < 0.25 ? 44.0 * r2 - 23.7 * roughness + 3.26 : 1.97 * r2 - 3.27 * roughness + 0.72;\n\tfloat DG = exp( a * dotNV + b ) + ( roughness < 0.25 ? 0.0 : 0.1 * ( roughness - 0.25 ) );\n\treturn saturate( DG * RECIPROCAL_PI );\n}\nvec2 DFGApprox( const in vec3 normal, const in vec3 viewDir, const in float roughness ) {\n\tfloat dotNV = saturate( dot( normal, viewDir ) );\n\tconst vec4 c0 = vec4( - 1, - 0.0275, - 0.572, 0.022 );\n\tconst vec4 c1 = vec4( 1, 0.0425, 1.04, - 0.04 );\n\tvec4 r = roughness * c0 + c1;\n\tfloat a004 = min( r.x * r.x, exp2( - 9.28 * dotNV ) ) * r.x + r.y;\n\tvec2 fab = vec2( - 1.04, 1.04 ) * a004 + r.zw;\n\treturn fab;\n}\nvec3 EnvironmentBRDF( const in vec3 normal, const in vec3 viewDir, const in vec3 specularColor, const in float specularF90, const in float roughness ) {\n\tvec2 fab = DFGApprox( normal, viewDir, roughness );\n\treturn specularColor * fab.x + specularF90 * fab.y;\n}\n#ifdef USE_IRIDESCENCE\nvoid computeMultiscatteringIridescence( const in vec3 normal, const in vec3 viewDir, const in vec3 specularColor, const in float specularF90, const in float iridescence, const in vec3 iridescenceF0, const in float roughness, inout vec3 singleScatter, inout vec3 multiScatter ) {\n#else\nvoid computeMultiscattering( const in vec3 normal, const in vec3 viewDir, const in vec3 specularColor, const in float specularF90, const in float roughness, inout vec3 singleScatter, inout vec3 multiScatter ) {\n#endif\n\tvec2 fab = DFGApprox( normal, viewDir, roughness );\n\t#ifdef USE_IRIDESCENCE\n\t\tvec3 Fr = mix( specularColor, iridescenceF0, iridescence );\n\t#else\n\t\tvec3 Fr = specularColor;\n\t#endif\n\tvec3 FssEss = Fr * fab.x + specularF90 * fab.y;\n\tfloat Ess = fab.x + fab.y;\n\tfloat Ems = 1.0 - Ess;\n\tvec3 Favg = Fr + ( 1.0 - Fr ) * 0.047619;\tvec3 Fms = FssEss * Favg / ( 1.0 - Ems * Favg );\n\tsingleScatter += FssEss;\n\tmultiScatter += Fms * Ems;\n}\n#if NUM_RECT_AREA_LIGHTS > 0\n\tvoid RE_Direct_RectArea_Physical( const in RectAreaLight rectAreaLight, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight ) {\n\t\tvec3 normal = geometry.normal;\n\t\tvec3 viewDir = geometry.viewDir;\n\t\tvec3 position = geometry.position;\n\t\tvec3 lightPos = rectAreaLight.position;\n\t\tvec3 halfWidth = rectAreaLight.halfWidth;\n\t\tvec3 halfHeight = rectAreaLight.halfHeight;\n\t\tvec3 lightColor = rectAreaLight.color;\n\t\tfloat roughness = material.roughness;\n\t\tvec3 rectCoords[ 4 ];\n\t\trectCoords[ 0 ] = lightPos + halfWidth - halfHeight;\t\trectCoords[ 1 ] = lightPos - halfWidth - halfHeight;\n\t\trectCoords[ 2 ] = lightPos - halfWidth + halfHeight;\n\t\trectCoords[ 3 ] = lightPos + halfWidth + halfHeight;\n\t\tvec2 uv = LTC_Uv( normal, viewDir, roughness );\n\t\tvec4 t1 = texture2D( ltc_1, uv );\n\t\tvec4 t2 = texture2D( ltc_2, uv );\n\t\tmat3 mInv = mat3(\n\t\t\tvec3( t1.x, 0, t1.y ),\n\t\t\tvec3(\t\t0, 1,\t\t0 ),\n\t\t\tvec3( t1.z, 0, t1.w )\n\t\t);\n\t\tvec3 fresnel = ( material.specularColor * t2.x + ( vec3( 1.0 ) - material.specularColor ) * t2.y );\n\t\treflectedLight.directSpecular += lightColor * fresnel * LTC_Evaluate( normal, viewDir, position, mInv, rectCoords );\n\t\treflectedLight.directDiffuse += lightColor * material.diffuseColor * LTC_Evaluate( normal, viewDir, position, mat3( 1.0 ), rectCoords );\n\t}\n#endif\nvoid RE_Direct_Physical( const in IncidentLight directLight, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight ) {\n\tfloat dotNL = saturate( dot( geometry.normal, directLight.direction ) );\n\tvec3 irradiance = dotNL * directLight.color;\n\t#ifdef USE_CLEARCOAT\n\t\tfloat dotNLcc = saturate( dot( geometry.clearcoatNormal, directLight.direction ) );\n\t\tvec3 ccIrradiance = dotNLcc * directLight.color;\n\t\tclearcoatSpecular += ccIrradiance * BRDF_GGX( directLight.direction, geometry.viewDir, geometry.clearcoatNormal, material.clearcoatF0, material.clearcoatF90, material.clearcoatRoughness );\n\t#endif\n\t#ifdef USE_SHEEN\n\t\tsheenSpecular += irradiance * BRDF_Sheen( directLight.direction, geometry.viewDir, geometry.normal, material.sheenColor, material.sheenRoughness );\n\t#endif\n\t#ifdef USE_IRIDESCENCE\n\t\treflectedLight.directSpecular += irradiance * BRDF_GGX_Iridescence( directLight.direction, geometry.viewDir, geometry.normal, material.specularColor, material.specularF90, material.iridescence, material.iridescenceFresnel, material.roughness );\n\t#else\n\t\treflectedLight.directSpecular += irradiance * BRDF_GGX( directLight.direction, geometry.viewDir, geometry.normal, material.specularColor, material.specularF90, material.roughness );\n\t#endif\n\treflectedLight.directDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\nvoid RE_IndirectDiffuse_Physical( const in vec3 irradiance, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight ) {\n\treflectedLight.indirectDiffuse += irradiance * BRDF_Lambert( material.diffuseColor );\n}\nvoid RE_IndirectSpecular_Physical( const in vec3 radiance, const in vec3 irradiance, const in vec3 clearcoatRadiance, const in GeometricContext geometry, const in PhysicalMaterial material, inout ReflectedLight reflectedLight) {\n\t#ifdef USE_CLEARCOAT\n\t\tclearcoatSpecular += clearcoatRadiance * EnvironmentBRDF( geometry.clearcoatNormal, geometry.viewDir, material.clearcoatF0, material.clearcoatF90, material.clearcoatRoughness );\n\t#endif\n\t#ifdef USE_SHEEN\n\t\tsheenSpecular += irradiance * material.sheenColor * IBLSheenBRDF( geometry.normal, geometry.viewDir, material.sheenRoughness );\n\t#endif\n\tvec3 singleScattering = vec3( 0.0 );\n\tvec3 multiScattering = vec3( 0.0 );\n\tvec3 cosineWeightedIrradiance = irradiance * RECIPROCAL_PI;\n\t#ifdef USE_IRIDESCENCE\n\t\tcomputeMultiscatteringIridescence( geometry.normal, geometry.viewDir, material.specularColor, material.specularF90, material.iridescence, material.iridescenceFresnel, material.roughness, singleScattering, multiScattering );\n\t#else\n\t\tcomputeMultiscattering( geometry.normal, geometry.viewDir, material.specularColor, material.specularF90, material.roughness, singleScattering, multiScattering );\n\t#endif\n\tvec3 totalScattering = singleScattering + multiScattering;\n\tvec3 diffuse = material.diffuseColor * ( 1.0 - max( max( totalScattering.r, totalScattering.g ), totalScattering.b ) );\n\treflectedLight.indirectSpecular += radiance * singleScattering;\n\treflectedLight.indirectSpecular += multiScattering * cosineWeightedIrradiance;\n\treflectedLight.indirectDiffuse += diffuse * cosineWeightedIrradiance;\n}\n#define RE_Direct\t\t\t\tRE_Direct_Physical\n#define RE_Direct_RectArea\t\tRE_Direct_RectArea_Physical\n#define RE_IndirectDiffuse\t\tRE_IndirectDiffuse_Physical\n#define RE_IndirectSpecular\t\tRE_IndirectSpecular_Physical\nfloat computeSpecularOcclusion( const in float dotNV, const in float ambientOcclusion, const in float roughness ) {\n\treturn saturate( pow( dotNV + ambientOcclusion, exp2( - 16.0 * roughness - 1.0 ) ) - 1.0 + ambientOcclusion );\n}",lights_fragment_begin:"\nGeometricContext geometry;\ngeometry.position = - vViewPosition;\ngeometry.normal = normal;\ngeometry.viewDir = ( isOrthographic ) ? vec3( 0, 0, 1 ) : normalize( vViewPosition );\n#ifdef USE_CLEARCOAT\n\tgeometry.clearcoatNormal = clearcoatNormal;\n#endif\n#ifdef USE_IRIDESCENCE\n\tfloat dotNVi = saturate( dot( normal, geometry.viewDir ) );\n\tif ( material.iridescenceThickness == 0.0 ) {\n\t\tmaterial.iridescence = 0.0;\n\t} else {\n\t\tmaterial.iridescence = saturate( material.iridescence );\n\t}\n\tif ( material.iridescence > 0.0 ) {\n\t\tmaterial.iridescenceFresnel = evalIridescence( 1.0, material.iridescenceIOR, dotNVi, material.iridescenceThickness, material.specularColor );\n\t\tmaterial.iridescenceF0 = Schlick_to_F0( material.iridescenceFresnel, 1.0, dotNVi );\n\t}\n#endif\nIncidentLight directLight;\n#if ( NUM_POINT_LIGHTS > 0 ) && defined( RE_Direct )\n\tPointLight pointLight;\n\t#if defined( USE_SHADOWMAP ) && NUM_POINT_LIGHT_SHADOWS > 0\n\tPointLightShadow pointLightShadow;\n\t#endif\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_POINT_LIGHTS; i ++ ) {\n\t\tpointLight = pointLights[ i ];\n\t\tgetPointLightInfo( pointLight, geometry, directLight );\n\t\t#if defined( USE_SHADOWMAP ) && ( UNROLLED_LOOP_INDEX < NUM_POINT_LIGHT_SHADOWS )\n\t\tpointLightShadow = pointLightShadows[ i ];\n\t\tdirectLight.color *= all( bvec2( directLight.visible, receiveShadow ) ) ? getPointShadow( pointShadowMap[ i ], pointLightShadow.shadowMapSize, pointLightShadow.shadowBias, pointLightShadow.shadowRadius, vPointShadowCoord[ i ], pointLightShadow.shadowCameraNear, pointLightShadow.shadowCameraFar ) : 1.0;\n\t\t#endif\n\t\tRE_Direct( directLight, geometry, material, reflectedLight );\n\t}\n\t#pragma unroll_loop_end\n#endif\n#if ( NUM_SPOT_LIGHTS > 0 ) && defined( RE_Direct )\n\tSpotLight spotLight;\n\tvec4 spotColor;\n\tvec3 spotLightCoord;\n\tbool inSpotLightMap;\n\t#if defined( USE_SHADOWMAP ) && NUM_SPOT_LIGHT_SHADOWS > 0\n\tSpotLightShadow spotLightShadow;\n\t#endif\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_SPOT_LIGHTS; i ++ ) {\n\t\tspotLight = spotLights[ i ];\n\t\tgetSpotLightInfo( spotLight, geometry, directLight );\n\t\t#if ( UNROLLED_LOOP_INDEX < NUM_SPOT_LIGHT_SHADOWS_WITH_MAPS )\n\t\t#define SPOT_LIGHT_MAP_INDEX UNROLLED_LOOP_INDEX\n\t\t#elif ( UNROLLED_LOOP_INDEX < NUM_SPOT_LIGHT_SHADOWS )\n\t\t#define SPOT_LIGHT_MAP_INDEX NUM_SPOT_LIGHT_MAPS\n\t\t#else\n\t\t#define SPOT_LIGHT_MAP_INDEX ( UNROLLED_LOOP_INDEX - NUM_SPOT_LIGHT_SHADOWS + NUM_SPOT_LIGHT_SHADOWS_WITH_MAPS )\n\t\t#endif\n\t\t#if ( SPOT_LIGHT_MAP_INDEX < NUM_SPOT_LIGHT_MAPS )\n\t\t\tspotLightCoord = vSpotLightCoord[ i ].xyz / vSpotLightCoord[ i ].w;\n\t\t\tinSpotLightMap = all( lessThan( abs( spotLightCoord * 2. - 1. ), vec3( 1.0 ) ) );\n\t\t\tspotColor = texture2D( spotLightMap[ SPOT_LIGHT_MAP_INDEX ], spotLightCoord.xy );\n\t\t\tdirectLight.color = inSpotLightMap ? directLight.color * spotColor.rgb : directLight.color;\n\t\t#endif\n\t\t#undef SPOT_LIGHT_MAP_INDEX\n\t\t#if defined( USE_SHADOWMAP ) && ( UNROLLED_LOOP_INDEX < NUM_SPOT_LIGHT_SHADOWS )\n\t\tspotLightShadow = spotLightShadows[ i ];\n\t\tdirectLight.color *= all( bvec2( directLight.visible, receiveShadow ) ) ? getShadow( spotShadowMap[ i ], spotLightShadow.shadowMapSize, spotLightShadow.shadowBias, spotLightShadow.shadowRadius, vSpotLightCoord[ i ] ) : 1.0;\n\t\t#endif\n\t\tRE_Direct( directLight, geometry, material, reflectedLight );\n\t}\n\t#pragma unroll_loop_end\n#endif\n#if ( NUM_DIR_LIGHTS > 0 ) && defined( RE_Direct )\n\tDirectionalLight directionalLight;\n\t#if defined( USE_SHADOWMAP ) && NUM_DIR_LIGHT_SHADOWS > 0\n\tDirectionalLightShadow directionalLightShadow;\n\t#endif\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_DIR_LIGHTS; i ++ ) {\n\t\tdirectionalLight = directionalLights[ i ];\n\t\tgetDirectionalLightInfo( directionalLight, geometry, directLight );\n\t\t#if defined( USE_SHADOWMAP ) && ( UNROLLED_LOOP_INDEX < NUM_DIR_LIGHT_SHADOWS )\n\t\tdirectionalLightShadow = directionalLightShadows[ i ];\n\t\tdirectLight.color *= all( bvec2( directLight.visible, receiveShadow ) ) ? getShadow( directionalShadowMap[ i ], directionalLightShadow.shadowMapSize, directionalLightShadow.shadowBias, directionalLightShadow.shadowRadius, vDirectionalShadowCoord[ i ] ) : 1.0;\n\t\t#endif\n\t\tRE_Direct( directLight, geometry, material, reflectedLight );\n\t}\n\t#pragma unroll_loop_end\n#endif\n#if ( NUM_RECT_AREA_LIGHTS > 0 ) && defined( RE_Direct_RectArea )\n\tRectAreaLight rectAreaLight;\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_RECT_AREA_LIGHTS; i ++ ) {\n\t\trectAreaLight = rectAreaLights[ i ];\n\t\tRE_Direct_RectArea( rectAreaLight, geometry, material, reflectedLight );\n\t}\n\t#pragma unroll_loop_end\n#endif\n#if defined( RE_IndirectDiffuse )\n\tvec3 iblIrradiance = vec3( 0.0 );\n\tvec3 irradiance = getAmbientLightIrradiance( ambientLightColor );\n\tirradiance += getLightProbeIrradiance( lightProbe, geometry.normal );\n\t#if ( NUM_HEMI_LIGHTS > 0 )\n\t\t#pragma unroll_loop_start\n\t\tfor ( int i = 0; i < NUM_HEMI_LIGHTS; i ++ ) {\n\t\t\tirradiance += getHemisphereLightIrradiance( hemisphereLights[ i ], geometry.normal );\n\t\t}\n\t\t#pragma unroll_loop_end\n\t#endif\n#endif\n#if defined( RE_IndirectSpecular )\n\tvec3 radiance = vec3( 0.0 );\n\tvec3 clearcoatRadiance = vec3( 0.0 );\n#endif",lights_fragment_maps:"#if defined( RE_IndirectDiffuse )\n\t#ifdef USE_LIGHTMAP\n\t\tvec4 lightMapTexel = texture2D( lightMap, vUv2 );\n\t\tvec3 lightMapIrradiance = lightMapTexel.rgb * lightMapIntensity;\n\t\tirradiance += lightMapIrradiance;\n\t#endif\n\t#if defined( USE_ENVMAP ) && defined( STANDARD ) && defined( ENVMAP_TYPE_CUBE_UV )\n\t\tiblIrradiance += getIBLIrradiance( geometry.normal );\n\t#endif\n#endif\n#if defined( USE_ENVMAP ) && defined( RE_IndirectSpecular )\n\tradiance += getIBLRadiance( geometry.viewDir, geometry.normal, material.roughness );\n\t#ifdef USE_CLEARCOAT\n\t\tclearcoatRadiance += getIBLRadiance( geometry.viewDir, geometry.clearcoatNormal, material.clearcoatRoughness );\n\t#endif\n#endif",lights_fragment_end:"#if defined( RE_IndirectDiffuse )\n\tRE_IndirectDiffuse( irradiance, geometry, material, reflectedLight );\n#endif\n#if defined( RE_IndirectSpecular )\n\tRE_IndirectSpecular( radiance, iblIrradiance, clearcoatRadiance, geometry, material, reflectedLight );\n#endif",logdepthbuf_fragment:"#if defined( USE_LOGDEPTHBUF ) && defined( USE_LOGDEPTHBUF_EXT )\n\tgl_FragDepthEXT = vIsPerspective == 0.0 ? gl_FragCoord.z : log2( vFragDepth ) * logDepthBufFC * 0.5;\n#endif",logdepthbuf_pars_fragment:"#if defined( USE_LOGDEPTHBUF ) && defined( USE_LOGDEPTHBUF_EXT )\n\tuniform float logDepthBufFC;\n\tvarying float vFragDepth;\n\tvarying float vIsPerspective;\n#endif",logdepthbuf_pars_vertex:"#ifdef USE_LOGDEPTHBUF\n\t#ifdef USE_LOGDEPTHBUF_EXT\n\t\tvarying float vFragDepth;\n\t\tvarying float vIsPerspective;\n\t#else\n\t\tuniform float logDepthBufFC;\n\t#endif\n#endif",logdepthbuf_vertex:"#ifdef USE_LOGDEPTHBUF\n\t#ifdef USE_LOGDEPTHBUF_EXT\n\t\tvFragDepth = 1.0 + gl_Position.w;\n\t\tvIsPerspective = float( isPerspectiveMatrix( projectionMatrix ) );\n\t#else\n\t\tif ( isPerspectiveMatrix( projectionMatrix ) ) {\n\t\t\tgl_Position.z = log2( max( EPSILON, gl_Position.w + 1.0 ) ) * logDepthBufFC - 1.0;\n\t\t\tgl_Position.z *= gl_Position.w;\n\t\t}\n\t#endif\n#endif",map_fragment:"#ifdef USE_MAP\n\tvec4 sampledDiffuseColor = texture2D( map, vUv );\n\t#ifdef DECODE_VIDEO_TEXTURE\n\t\tsampledDiffuseColor = vec4( mix( pow( sampledDiffuseColor.rgb * 0.9478672986 + vec3( 0.0521327014 ), vec3( 2.4 ) ), sampledDiffuseColor.rgb * 0.0773993808, vec3( lessThanEqual( sampledDiffuseColor.rgb, vec3( 0.04045 ) ) ) ), sampledDiffuseColor.w );\n\t#endif\n\tdiffuseColor *= sampledDiffuseColor;\n#endif",map_pars_fragment:"#ifdef USE_MAP\n\tuniform sampler2D map;\n#endif",map_particle_fragment:"#if defined( USE_MAP ) || defined( USE_ALPHAMAP )\n\tvec2 uv = ( uvTransform * vec3( gl_PointCoord.x, 1.0 - gl_PointCoord.y, 1 ) ).xy;\n#endif\n#ifdef USE_MAP\n\tdiffuseColor *= texture2D( map, uv );\n#endif\n#ifdef USE_ALPHAMAP\n\tdiffuseColor.a *= texture2D( alphaMap, uv ).g;\n#endif",map_particle_pars_fragment:"#if defined( USE_MAP ) || defined( USE_ALPHAMAP )\n\tuniform mat3 uvTransform;\n#endif\n#ifdef USE_MAP\n\tuniform sampler2D map;\n#endif\n#ifdef USE_ALPHAMAP\n\tuniform sampler2D alphaMap;\n#endif",metalnessmap_fragment:"float metalnessFactor = metalness;\n#ifdef USE_METALNESSMAP\n\tvec4 texelMetalness = texture2D( metalnessMap, vUv );\n\tmetalnessFactor *= texelMetalness.b;\n#endif",metalnessmap_pars_fragment:"#ifdef USE_METALNESSMAP\n\tuniform sampler2D metalnessMap;\n#endif",morphcolor_vertex:"#if defined( USE_MORPHCOLORS ) && defined( MORPHTARGETS_TEXTURE )\n\tvColor *= morphTargetBaseInfluence;\n\tfor ( int i = 0; i < MORPHTARGETS_COUNT; i ++ ) {\n\t\t#if defined( USE_COLOR_ALPHA )\n\t\t\tif ( morphTargetInfluences[ i ] != 0.0 ) vColor += getMorph( gl_VertexID, i, 2 ) * morphTargetInfluences[ i ];\n\t\t#elif defined( USE_COLOR )\n\t\t\tif ( morphTargetInfluences[ i ] != 0.0 ) vColor += getMorph( gl_VertexID, i, 2 ).rgb * morphTargetInfluences[ i ];\n\t\t#endif\n\t}\n#endif",morphnormal_vertex:"#ifdef USE_MORPHNORMALS\n\tobjectNormal *= morphTargetBaseInfluence;\n\t#ifdef MORPHTARGETS_TEXTURE\n\t\tfor ( int i = 0; i < MORPHTARGETS_COUNT; i ++ ) {\n\t\t\tif ( morphTargetInfluences[ i ] != 0.0 ) objectNormal += getMorph( gl_VertexID, i, 1 ).xyz * morphTargetInfluences[ i ];\n\t\t}\n\t#else\n\t\tobjectNormal += morphNormal0 * morphTargetInfluences[ 0 ];\n\t\tobjectNormal += morphNormal1 * morphTargetInfluences[ 1 ];\n\t\tobjectNormal += morphNormal2 * morphTargetInfluences[ 2 ];\n\t\tobjectNormal += morphNormal3 * morphTargetInfluences[ 3 ];\n\t#endif\n#endif",morphtarget_pars_vertex:"#ifdef USE_MORPHTARGETS\n\tuniform float morphTargetBaseInfluence;\n\t#ifdef MORPHTARGETS_TEXTURE\n\t\tuniform float morphTargetInfluences[ MORPHTARGETS_COUNT ];\n\t\tuniform sampler2DArray morphTargetsTexture;\n\t\tuniform ivec2 morphTargetsTextureSize;\n\t\tvec4 getMorph( const in int vertexIndex, const in int morphTargetIndex, const in int offset ) {\n\t\t\tint texelIndex = vertexIndex * MORPHTARGETS_TEXTURE_STRIDE + offset;\n\t\t\tint y = texelIndex / morphTargetsTextureSize.x;\n\t\t\tint x = texelIndex - y * morphTargetsTextureSize.x;\n\t\t\tivec3 morphUV = ivec3( x, y, morphTargetIndex );\n\t\t\treturn texelFetch( morphTargetsTexture, morphUV, 0 );\n\t\t}\n\t#else\n\t\t#ifndef USE_MORPHNORMALS\n\t\t\tuniform float morphTargetInfluences[ 8 ];\n\t\t#else\n\t\t\tuniform float morphTargetInfluences[ 4 ];\n\t\t#endif\n\t#endif\n#endif",morphtarget_vertex:"#ifdef USE_MORPHTARGETS\n\ttransformed *= morphTargetBaseInfluence;\n\t#ifdef MORPHTARGETS_TEXTURE\n\t\tfor ( int i = 0; i < MORPHTARGETS_COUNT; i ++ ) {\n\t\t\tif ( morphTargetInfluences[ i ] != 0.0 ) transformed += getMorph( gl_VertexID, i, 0 ).xyz * morphTargetInfluences[ i ];\n\t\t}\n\t#else\n\t\ttransformed += morphTarget0 * morphTargetInfluences[ 0 ];\n\t\ttransformed += morphTarget1 * morphTargetInfluences[ 1 ];\n\t\ttransformed += morphTarget2 * morphTargetInfluences[ 2 ];\n\t\ttransformed += morphTarget3 * morphTargetInfluences[ 3 ];\n\t\t#ifndef USE_MORPHNORMALS\n\t\t\ttransformed += morphTarget4 * morphTargetInfluences[ 4 ];\n\t\t\ttransformed += morphTarget5 * morphTargetInfluences[ 5 ];\n\t\t\ttransformed += morphTarget6 * morphTargetInfluences[ 6 ];\n\t\t\ttransformed += morphTarget7 * morphTargetInfluences[ 7 ];\n\t\t#endif\n\t#endif\n#endif",normal_fragment_begin:"float faceDirection = gl_FrontFacing ? 1.0 : - 1.0;\n#ifdef FLAT_SHADED\n\tvec3 fdx = dFdx( vViewPosition );\n\tvec3 fdy = dFdy( vViewPosition );\n\tvec3 normal = normalize( cross( fdx, fdy ) );\n#else\n\tvec3 normal = normalize( vNormal );\n\t#ifdef DOUBLE_SIDED\n\t\tnormal = normal * faceDirection;\n\t#endif\n\t#ifdef USE_TANGENT\n\t\tvec3 tangent = normalize( vTangent );\n\t\tvec3 bitangent = normalize( vBitangent );\n\t\t#ifdef DOUBLE_SIDED\n\t\t\ttangent = tangent * faceDirection;\n\t\t\tbitangent = bitangent * faceDirection;\n\t\t#endif\n\t\t#if defined( TANGENTSPACE_NORMALMAP ) || defined( USE_CLEARCOAT_NORMALMAP )\n\t\t\tmat3 vTBN = mat3( tangent, bitangent, normal );\n\t\t#endif\n\t#endif\n#endif\nvec3 geometryNormal = normal;",normal_fragment_maps:"#ifdef OBJECTSPACE_NORMALMAP\n\tnormal = texture2D( normalMap, vUv ).xyz * 2.0 - 1.0;\n\t#ifdef FLIP_SIDED\n\t\tnormal = - normal;\n\t#endif\n\t#ifdef DOUBLE_SIDED\n\t\tnormal = normal * faceDirection;\n\t#endif\n\tnormal = normalize( normalMatrix * normal );\n#elif defined( TANGENTSPACE_NORMALMAP )\n\tvec3 mapN = texture2D( normalMap, vUv ).xyz * 2.0 - 1.0;\n\tmapN.xy *= normalScale;\n\t#ifdef USE_TANGENT\n\t\tnormal = normalize( vTBN * mapN );\n\t#else\n\t\tnormal = perturbNormal2Arb( - vViewPosition, normal, mapN, faceDirection );\n\t#endif\n#elif defined( USE_BUMPMAP )\n\tnormal = perturbNormalArb( - vViewPosition, normal, dHdxy_fwd(), faceDirection );\n#endif",normal_pars_fragment:"#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n\t#ifdef USE_TANGENT\n\t\tvarying vec3 vTangent;\n\t\tvarying vec3 vBitangent;\n\t#endif\n#endif",normal_pars_vertex:"#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n\t#ifdef USE_TANGENT\n\t\tvarying vec3 vTangent;\n\t\tvarying vec3 vBitangent;\n\t#endif\n#endif",normal_vertex:"#ifndef FLAT_SHADED\n\tvNormal = normalize( transformedNormal );\n\t#ifdef USE_TANGENT\n\t\tvTangent = normalize( transformedTangent );\n\t\tvBitangent = normalize( cross( vNormal, vTangent ) * tangent.w );\n\t#endif\n#endif",normalmap_pars_fragment:"#ifdef USE_NORMALMAP\n\tuniform sampler2D normalMap;\n\tuniform vec2 normalScale;\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\n\tuniform mat3 normalMatrix;\n#endif\n#if ! defined ( USE_TANGENT ) && ( defined ( TANGENTSPACE_NORMALMAP ) || defined ( USE_CLEARCOAT_NORMALMAP ) )\n\tvec3 perturbNormal2Arb( vec3 eye_pos, vec3 surf_norm, vec3 mapN, float faceDirection ) {\n\t\tvec3 q0 = dFdx( eye_pos.xyz );\n\t\tvec3 q1 = dFdy( eye_pos.xyz );\n\t\tvec2 st0 = dFdx( vUv.st );\n\t\tvec2 st1 = dFdy( vUv.st );\n\t\tvec3 N = surf_norm;\n\t\tvec3 q1perp = cross( q1, N );\n\t\tvec3 q0perp = cross( N, q0 );\n\t\tvec3 T = q1perp * st0.x + q0perp * st1.x;\n\t\tvec3 B = q1perp * st0.y + q0perp * st1.y;\n\t\tfloat det = max( dot( T, T ), dot( B, B ) );\n\t\tfloat scale = ( det == 0.0 ) ? 0.0 : faceDirection * inversesqrt( det );\n\t\treturn normalize( T * ( mapN.x * scale ) + B * ( mapN.y * scale ) + N * mapN.z );\n\t}\n#endif",clearcoat_normal_fragment_begin:"#ifdef USE_CLEARCOAT\n\tvec3 clearcoatNormal = geometryNormal;\n#endif",clearcoat_normal_fragment_maps:"#ifdef USE_CLEARCOAT_NORMALMAP\n\tvec3 clearcoatMapN = texture2D( clearcoatNormalMap, vUv ).xyz * 2.0 - 1.0;\n\tclearcoatMapN.xy *= clearcoatNormalScale;\n\t#ifdef USE_TANGENT\n\t\tclearcoatNormal = normalize( vTBN * clearcoatMapN );\n\t#else\n\t\tclearcoatNormal = perturbNormal2Arb( - vViewPosition, clearcoatNormal, clearcoatMapN, faceDirection );\n\t#endif\n#endif",clearcoat_pars_fragment:"#ifdef USE_CLEARCOATMAP\n\tuniform sampler2D clearcoatMap;\n#endif\n#ifdef USE_CLEARCOAT_ROUGHNESSMAP\n\tuniform sampler2D clearcoatRoughnessMap;\n#endif\n#ifdef USE_CLEARCOAT_NORMALMAP\n\tuniform sampler2D clearcoatNormalMap;\n\tuniform vec2 clearcoatNormalScale;\n#endif",iridescence_pars_fragment:"#ifdef USE_IRIDESCENCEMAP\n\tuniform sampler2D iridescenceMap;\n#endif\n#ifdef USE_IRIDESCENCE_THICKNESSMAP\n\tuniform sampler2D iridescenceThicknessMap;\n#endif",output_fragment:"#ifdef OPAQUE\ndiffuseColor.a = 1.0;\n#endif\n#ifdef USE_TRANSMISSION\ndiffuseColor.a *= material.transmissionAlpha + 0.1;\n#endif\ngl_FragColor = vec4( outgoingLight, diffuseColor.a );",packing:"vec3 packNormalToRGB( const in vec3 normal ) {\n\treturn normalize( normal ) * 0.5 + 0.5;\n}\nvec3 unpackRGBToNormal( const in vec3 rgb ) {\n\treturn 2.0 * rgb.xyz - 1.0;\n}\nconst float PackUpscale = 256. / 255.;const float UnpackDownscale = 255. / 256.;\nconst vec3 PackFactors = vec3( 256. * 256. * 256., 256. * 256., 256. );\nconst vec4 UnpackFactors = UnpackDownscale / vec4( PackFactors, 1. );\nconst float ShiftRight8 = 1. / 256.;\nvec4 packDepthToRGBA( const in float v ) {\n\tvec4 r = vec4( fract( v * PackFactors ), v );\n\tr.yzw -= r.xyz * ShiftRight8;\treturn r * PackUpscale;\n}\nfloat unpackRGBAToDepth( const in vec4 v ) {\n\treturn dot( v, UnpackFactors );\n}\nvec2 packDepthToRG( in highp float v ) {\n\treturn packDepthToRGBA( v ).yx;\n}\nfloat unpackRGToDepth( const in highp vec2 v ) {\n\treturn unpackRGBAToDepth( vec4( v.xy, 0.0, 0.0 ) );\n}\nvec4 pack2HalfToRGBA( vec2 v ) {\n\tvec4 r = vec4( v.x, fract( v.x * 255.0 ), v.y, fract( v.y * 255.0 ) );\n\treturn vec4( r.x - r.y / 255.0, r.y, r.z - r.w / 255.0, r.w );\n}\nvec2 unpackRGBATo2Half( vec4 v ) {\n\treturn vec2( v.x + ( v.y / 255.0 ), v.z + ( v.w / 255.0 ) );\n}\nfloat viewZToOrthographicDepth( const in float viewZ, const in float near, const in float far ) {\n\treturn ( viewZ + near ) / ( near - far );\n}\nfloat orthographicDepthToViewZ( const in float linearClipZ, const in float near, const in float far ) {\n\treturn linearClipZ * ( near - far ) - near;\n}\nfloat viewZToPerspectiveDepth( const in float viewZ, const in float near, const in float far ) {\n\treturn ( ( near + viewZ ) * far ) / ( ( far - near ) * viewZ );\n}\nfloat perspectiveDepthToViewZ( const in float invClipZ, const in float near, const in float far ) {\n\treturn ( near * far ) / ( ( far - near ) * invClipZ - far );\n}",premultiplied_alpha_fragment:"#ifdef PREMULTIPLIED_ALPHA\n\tgl_FragColor.rgb *= gl_FragColor.a;\n#endif",project_vertex:"vec4 mvPosition = vec4( transformed, 1.0 );\n#ifdef USE_INSTANCING\n\tmvPosition = instanceMatrix * mvPosition;\n#endif\nmvPosition = modelViewMatrix * mvPosition;\ngl_Position = projectionMatrix * mvPosition;",dithering_fragment:"#ifdef DITHERING\n\tgl_FragColor.rgb = dithering( gl_FragColor.rgb );\n#endif",dithering_pars_fragment:"#ifdef DITHERING\n\tvec3 dithering( vec3 color ) {\n\t\tfloat grid_position = rand( gl_FragCoord.xy );\n\t\tvec3 dither_shift_RGB = vec3( 0.25 / 255.0, -0.25 / 255.0, 0.25 / 255.0 );\n\t\tdither_shift_RGB = mix( 2.0 * dither_shift_RGB, -2.0 * dither_shift_RGB, grid_position );\n\t\treturn color + dither_shift_RGB;\n\t}\n#endif",roughnessmap_fragment:"float roughnessFactor = roughness;\n#ifdef USE_ROUGHNESSMAP\n\tvec4 texelRoughness = texture2D( roughnessMap, vUv );\n\troughnessFactor *= texelRoughness.g;\n#endif",roughnessmap_pars_fragment:"#ifdef USE_ROUGHNESSMAP\n\tuniform sampler2D roughnessMap;\n#endif",shadowmap_pars_fragment:"#if NUM_SPOT_LIGHT_COORDS > 0\n\tvarying vec4 vSpotLightCoord[ NUM_SPOT_LIGHT_COORDS ];\n#endif\n#if NUM_SPOT_LIGHT_MAPS > 0\n\tuniform sampler2D spotLightMap[ NUM_SPOT_LIGHT_MAPS ];\n#endif\n#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\t\tuniform sampler2D directionalShadowMap[ NUM_DIR_LIGHT_SHADOWS ];\n\t\tvarying vec4 vDirectionalShadowCoord[ NUM_DIR_LIGHT_SHADOWS ];\n\t\tstruct DirectionalLightShadow {\n\t\t\tfloat shadowBias;\n\t\t\tfloat shadowNormalBias;\n\t\t\tfloat shadowRadius;\n\t\t\tvec2 shadowMapSize;\n\t\t};\n\t\tuniform DirectionalLightShadow directionalLightShadows[ NUM_DIR_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\t\tuniform sampler2D spotShadowMap[ NUM_SPOT_LIGHT_SHADOWS ];\n\t\tstruct SpotLightShadow {\n\t\t\tfloat shadowBias;\n\t\t\tfloat shadowNormalBias;\n\t\t\tfloat shadowRadius;\n\t\t\tvec2 shadowMapSize;\n\t\t};\n\t\tuniform SpotLightShadow spotLightShadows[ NUM_SPOT_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\t\tuniform sampler2D pointShadowMap[ NUM_POINT_LIGHT_SHADOWS ];\n\t\tvarying vec4 vPointShadowCoord[ NUM_POINT_LIGHT_SHADOWS ];\n\t\tstruct PointLightShadow {\n\t\t\tfloat shadowBias;\n\t\t\tfloat shadowNormalBias;\n\t\t\tfloat shadowRadius;\n\t\t\tvec2 shadowMapSize;\n\t\t\tfloat shadowCameraNear;\n\t\t\tfloat shadowCameraFar;\n\t\t};\n\t\tuniform PointLightShadow pointLightShadows[ NUM_POINT_LIGHT_SHADOWS ];\n\t#endif\n\tfloat texture2DCompare( sampler2D depths, vec2 uv, float compare ) {\n\t\treturn step( compare, unpackRGBAToDepth( texture2D( depths, uv ) ) );\n\t}\n\tvec2 texture2DDistribution( sampler2D shadow, vec2 uv ) {\n\t\treturn unpackRGBATo2Half( texture2D( shadow, uv ) );\n\t}\n\tfloat VSMShadow (sampler2D shadow, vec2 uv, float compare ){\n\t\tfloat occlusion = 1.0;\n\t\tvec2 distribution = texture2DDistribution( shadow, uv );\n\t\tfloat hard_shadow = step( compare , distribution.x );\n\t\tif (hard_shadow != 1.0 ) {\n\t\t\tfloat distance = compare - distribution.x ;\n\t\t\tfloat variance = max( 0.00000, distribution.y * distribution.y );\n\t\t\tfloat softness_probability = variance / (variance + distance * distance );\t\t\tsoftness_probability = clamp( ( softness_probability - 0.3 ) / ( 0.95 - 0.3 ), 0.0, 1.0 );\t\t\tocclusion = clamp( max( hard_shadow, softness_probability ), 0.0, 1.0 );\n\t\t}\n\t\treturn occlusion;\n\t}\n\tfloat getShadow( sampler2D shadowMap, vec2 shadowMapSize, float shadowBias, float shadowRadius, vec4 shadowCoord ) {\n\t\tfloat shadow = 1.0;\n\t\tshadowCoord.xyz /= shadowCoord.w;\n\t\tshadowCoord.z += shadowBias;\n\t\tbvec4 inFrustumVec = bvec4 ( shadowCoord.x >= 0.0, shadowCoord.x <= 1.0, shadowCoord.y >= 0.0, shadowCoord.y <= 1.0 );\n\t\tbool inFrustum = all( inFrustumVec );\n\t\tbvec2 frustumTestVec = bvec2( inFrustum, shadowCoord.z <= 1.0 );\n\t\tbool frustumTest = all( frustumTestVec );\n\t\tif ( frustumTest ) {\n\t\t#if defined( SHADOWMAP_TYPE_PCF )\n\t\t\tvec2 texelSize = vec2( 1.0 ) / shadowMapSize;\n\t\t\tfloat dx0 = - texelSize.x * shadowRadius;\n\t\t\tfloat dy0 = - texelSize.y * shadowRadius;\n\t\t\tfloat dx1 = + texelSize.x * shadowRadius;\n\t\t\tfloat dy1 = + texelSize.y * shadowRadius;\n\t\t\tfloat dx2 = dx0 / 2.0;\n\t\t\tfloat dy2 = dy0 / 2.0;\n\t\t\tfloat dx3 = dx1 / 2.0;\n\t\t\tfloat dy3 = dy1 / 2.0;\n\t\t\tshadow = (\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx0, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx1, dy0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx2, dy2 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy2 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx3, dy2 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx0, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx2, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy, shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx3, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx1, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx2, dy3 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy3 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx3, dy3 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx0, dy1 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( 0.0, dy1 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, shadowCoord.xy + vec2( dx1, dy1 ), shadowCoord.z )\n\t\t\t) * ( 1.0 / 17.0 );\n\t\t#elif defined( SHADOWMAP_TYPE_PCF_SOFT )\n\t\t\tvec2 texelSize = vec2( 1.0 ) / shadowMapSize;\n\t\t\tfloat dx = texelSize.x;\n\t\t\tfloat dy = texelSize.y;\n\t\t\tvec2 uv = shadowCoord.xy;\n\t\t\tvec2 f = fract( uv * shadowMapSize + 0.5 );\n\t\t\tuv -= f * texelSize;\n\t\t\tshadow = (\n\t\t\t\ttexture2DCompare( shadowMap, uv, shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, uv + vec2( dx, 0.0 ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, uv + vec2( 0.0, dy ), shadowCoord.z ) +\n\t\t\t\ttexture2DCompare( shadowMap, uv + texelSize, shadowCoord.z ) +\n\t\t\t\tmix( texture2DCompare( shadowMap, uv + vec2( -dx, 0.0 ), shadowCoord.z ),\n\t\t\t\t\t texture2DCompare( shadowMap, uv + vec2( 2.0 * dx, 0.0 ), shadowCoord.z ),\n\t\t\t\t\t f.x ) +\n\t\t\t\tmix( texture2DCompare( shadowMap, uv + vec2( -dx, dy ), shadowCoord.z ),\n\t\t\t\t\t texture2DCompare( shadowMap, uv + vec2( 2.0 * dx, dy ), shadowCoord.z ),\n\t\t\t\t\t f.x ) +\n\t\t\t\tmix( texture2DCompare( shadowMap, uv + vec2( 0.0, -dy ), shadowCoord.z ),\n\t\t\t\t\t texture2DCompare( shadowMap, uv + vec2( 0.0, 2.0 * dy ), shadowCoord.z ),\n\t\t\t\t\t f.y ) +\n\t\t\t\tmix( texture2DCompare( shadowMap, uv + vec2( dx, -dy ), shadowCoord.z ),\n\t\t\t\t\t texture2DCompare( shadowMap, uv + vec2( dx, 2.0 * dy ), shadowCoord.z ),\n\t\t\t\t\t f.y ) +\n\t\t\t\tmix( mix( texture2DCompare( shadowMap, uv + vec2( -dx, -dy ), shadowCoord.z ),\n\t\t\t\t\t\t\ttexture2DCompare( shadowMap, uv + vec2( 2.0 * dx, -dy ), shadowCoord.z ),\n\t\t\t\t\t\t\tf.x ),\n\t\t\t\t\t mix( texture2DCompare( shadowMap, uv + vec2( -dx, 2.0 * dy ), shadowCoord.z ),\n\t\t\t\t\t\t\ttexture2DCompare( shadowMap, uv + vec2( 2.0 * dx, 2.0 * dy ), shadowCoord.z ),\n\t\t\t\t\t\t\tf.x ),\n\t\t\t\t\t f.y )\n\t\t\t) * ( 1.0 / 9.0 );\n\t\t#elif defined( SHADOWMAP_TYPE_VSM )\n\t\t\tshadow = VSMShadow( shadowMap, shadowCoord.xy, shadowCoord.z );\n\t\t#else\n\t\t\tshadow = texture2DCompare( shadowMap, shadowCoord.xy, shadowCoord.z );\n\t\t#endif\n\t\t}\n\t\treturn shadow;\n\t}\n\tvec2 cubeToUV( vec3 v, float texelSizeY ) {\n\t\tvec3 absV = abs( v );\n\t\tfloat scaleToCube = 1.0 / max( absV.x, max( absV.y, absV.z ) );\n\t\tabsV *= scaleToCube;\n\t\tv *= scaleToCube * ( 1.0 - 2.0 * texelSizeY );\n\t\tvec2 planar = v.xy;\n\t\tfloat almostATexel = 1.5 * texelSizeY;\n\t\tfloat almostOne = 1.0 - almostATexel;\n\t\tif ( absV.z >= almostOne ) {\n\t\t\tif ( v.z > 0.0 )\n\t\t\t\tplanar.x = 4.0 - v.x;\n\t\t} else if ( absV.x >= almostOne ) {\n\t\t\tfloat signX = sign( v.x );\n\t\t\tplanar.x = v.z * signX + 2.0 * signX;\n\t\t} else if ( absV.y >= almostOne ) {\n\t\t\tfloat signY = sign( v.y );\n\t\t\tplanar.x = v.x + 2.0 * signY + 2.0;\n\t\t\tplanar.y = v.z * signY - 2.0;\n\t\t}\n\t\treturn vec2( 0.125, 0.25 ) * planar + vec2( 0.375, 0.75 );\n\t}\n\tfloat getPointShadow( sampler2D shadowMap, vec2 shadowMapSize, float shadowBias, float shadowRadius, vec4 shadowCoord, float shadowCameraNear, float shadowCameraFar ) {\n\t\tvec2 texelSize = vec2( 1.0 ) / ( shadowMapSize * vec2( 4.0, 2.0 ) );\n\t\tvec3 lightToPosition = shadowCoord.xyz;\n\t\tfloat dp = ( length( lightToPosition ) - shadowCameraNear ) / ( shadowCameraFar - shadowCameraNear );\t\tdp += shadowBias;\n\t\tvec3 bd3D = normalize( lightToPosition );\n\t\t#if defined( SHADOWMAP_TYPE_PCF ) || defined( SHADOWMAP_TYPE_PCF_SOFT ) || defined( SHADOWMAP_TYPE_VSM )\n\t\t\tvec2 offset = vec2( - 1, 1 ) * shadowRadius * texelSize.y;\n\t\t\treturn (\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xyy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yyy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xyx, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yyx, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xxy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yxy, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.xxx, texelSize.y ), dp ) +\n\t\t\t\ttexture2DCompare( shadowMap, cubeToUV( bd3D + offset.yxx, texelSize.y ), dp )\n\t\t\t) * ( 1.0 / 9.0 );\n\t\t#else\n\t\t\treturn texture2DCompare( shadowMap, cubeToUV( bd3D, texelSize.y ), dp );\n\t\t#endif\n\t}\n#endif",shadowmap_pars_vertex:"#if NUM_SPOT_LIGHT_COORDS > 0\n\tuniform mat4 spotLightMatrix[ NUM_SPOT_LIGHT_COORDS ];\n\tvarying vec4 vSpotLightCoord[ NUM_SPOT_LIGHT_COORDS ];\n#endif\n#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\t\tuniform mat4 directionalShadowMatrix[ NUM_DIR_LIGHT_SHADOWS ];\n\t\tvarying vec4 vDirectionalShadowCoord[ NUM_DIR_LIGHT_SHADOWS ];\n\t\tstruct DirectionalLightShadow {\n\t\t\tfloat shadowBias;\n\t\t\tfloat shadowNormalBias;\n\t\t\tfloat shadowRadius;\n\t\t\tvec2 shadowMapSize;\n\t\t};\n\t\tuniform DirectionalLightShadow directionalLightShadows[ NUM_DIR_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\t\tstruct SpotLightShadow {\n\t\t\tfloat shadowBias;\n\t\t\tfloat shadowNormalBias;\n\t\t\tfloat shadowRadius;\n\t\t\tvec2 shadowMapSize;\n\t\t};\n\t\tuniform SpotLightShadow spotLightShadows[ NUM_SPOT_LIGHT_SHADOWS ];\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\t\tuniform mat4 pointShadowMatrix[ NUM_POINT_LIGHT_SHADOWS ];\n\t\tvarying vec4 vPointShadowCoord[ NUM_POINT_LIGHT_SHADOWS ];\n\t\tstruct PointLightShadow {\n\t\t\tfloat shadowBias;\n\t\t\tfloat shadowNormalBias;\n\t\t\tfloat shadowRadius;\n\t\t\tvec2 shadowMapSize;\n\t\t\tfloat shadowCameraNear;\n\t\t\tfloat shadowCameraFar;\n\t\t};\n\t\tuniform PointLightShadow pointLightShadows[ NUM_POINT_LIGHT_SHADOWS ];\n\t#endif\n#endif",shadowmap_vertex:"#if defined( USE_SHADOWMAP ) || ( NUM_SPOT_LIGHT_COORDS > 0 )\n\t#if NUM_DIR_LIGHT_SHADOWS > 0 || NUM_SPOT_LIGHT_COORDS > 0 || NUM_POINT_LIGHT_SHADOWS > 0\n\t\tvec3 shadowWorldNormal = inverseTransformDirection( transformedNormal, viewMatrix );\n\t\tvec4 shadowWorldPosition;\n\t#endif\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_DIR_LIGHT_SHADOWS; i ++ ) {\n\t\tshadowWorldPosition = worldPosition + vec4( shadowWorldNormal * directionalLightShadows[ i ].shadowNormalBias, 0 );\n\t\tvDirectionalShadowCoord[ i ] = directionalShadowMatrix[ i ] * shadowWorldPosition;\n\t}\n\t#pragma unroll_loop_end\n\t#endif\n\t#if NUM_SPOT_LIGHT_COORDS > 0\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_SPOT_LIGHT_COORDS; i ++ ) {\n\t\tshadowWorldPosition = worldPosition;\n\t\t#if ( defined( USE_SHADOWMAP ) && UNROLLED_LOOP_INDEX < NUM_SPOT_LIGHT_SHADOWS )\n\t\t\tshadowWorldPosition.xyz += shadowWorldNormal * spotLightShadows[ i ].shadowNormalBias;\n\t\t#endif\n\t\tvSpotLightCoord[ i ] = spotLightMatrix[ i ] * shadowWorldPosition;\n\t}\n\t#pragma unroll_loop_end\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_POINT_LIGHT_SHADOWS; i ++ ) {\n\t\tshadowWorldPosition = worldPosition + vec4( shadowWorldNormal * pointLightShadows[ i ].shadowNormalBias, 0 );\n\t\tvPointShadowCoord[ i ] = pointShadowMatrix[ i ] * shadowWorldPosition;\n\t}\n\t#pragma unroll_loop_end\n\t#endif\n#endif",shadowmask_pars_fragment:"float getShadowMask() {\n\tfloat shadow = 1.0;\n\t#ifdef USE_SHADOWMAP\n\t#if NUM_DIR_LIGHT_SHADOWS > 0\n\tDirectionalLightShadow directionalLight;\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_DIR_LIGHT_SHADOWS; i ++ ) {\n\t\tdirectionalLight = directionalLightShadows[ i ];\n\t\tshadow *= receiveShadow ? getShadow( directionalShadowMap[ i ], directionalLight.shadowMapSize, directionalLight.shadowBias, directionalLight.shadowRadius, vDirectionalShadowCoord[ i ] ) : 1.0;\n\t}\n\t#pragma unroll_loop_end\n\t#endif\n\t#if NUM_SPOT_LIGHT_SHADOWS > 0\n\tSpotLightShadow spotLight;\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_SPOT_LIGHT_SHADOWS; i ++ ) {\n\t\tspotLight = spotLightShadows[ i ];\n\t\tshadow *= receiveShadow ? getShadow( spotShadowMap[ i ], spotLight.shadowMapSize, spotLight.shadowBias, spotLight.shadowRadius, vSpotLightCoord[ i ] ) : 1.0;\n\t}\n\t#pragma unroll_loop_end\n\t#endif\n\t#if NUM_POINT_LIGHT_SHADOWS > 0\n\tPointLightShadow pointLight;\n\t#pragma unroll_loop_start\n\tfor ( int i = 0; i < NUM_POINT_LIGHT_SHADOWS; i ++ ) {\n\t\tpointLight = pointLightShadows[ i ];\n\t\tshadow *= receiveShadow ? getPointShadow( pointShadowMap[ i ], pointLight.shadowMapSize, pointLight.shadowBias, pointLight.shadowRadius, vPointShadowCoord[ i ], pointLight.shadowCameraNear, pointLight.shadowCameraFar ) : 1.0;\n\t}\n\t#pragma unroll_loop_end\n\t#endif\n\t#endif\n\treturn shadow;\n}",skinbase_vertex:"#ifdef USE_SKINNING\n\tmat4 boneMatX = getBoneMatrix( skinIndex.x );\n\tmat4 boneMatY = getBoneMatrix( skinIndex.y );\n\tmat4 boneMatZ = getBoneMatrix( skinIndex.z );\n\tmat4 boneMatW = getBoneMatrix( skinIndex.w );\n#endif",skinning_pars_vertex:"#ifdef USE_SKINNING\n\tuniform mat4 bindMatrix;\n\tuniform mat4 bindMatrixInverse;\n\tuniform highp sampler2D boneTexture;\n\tuniform int boneTextureSize;\n\tmat4 getBoneMatrix( const in float i ) {\n\t\tfloat j = i * 4.0;\n\t\tfloat x = mod( j, float( boneTextureSize ) );\n\t\tfloat y = floor( j / float( boneTextureSize ) );\n\t\tfloat dx = 1.0 / float( boneTextureSize );\n\t\tfloat dy = 1.0 / float( boneTextureSize );\n\t\ty = dy * ( y + 0.5 );\n\t\tvec4 v1 = texture2D( boneTexture, vec2( dx * ( x + 0.5 ), y ) );\n\t\tvec4 v2 = texture2D( boneTexture, vec2( dx * ( x + 1.5 ), y ) );\n\t\tvec4 v3 = texture2D( boneTexture, vec2( dx * ( x + 2.5 ), y ) );\n\t\tvec4 v4 = texture2D( boneTexture, vec2( dx * ( x + 3.5 ), y ) );\n\t\tmat4 bone = mat4( v1, v2, v3, v4 );\n\t\treturn bone;\n\t}\n#endif",skinning_vertex:"#ifdef USE_SKINNING\n\tvec4 skinVertex = bindMatrix * vec4( transformed, 1.0 );\n\tvec4 skinned = vec4( 0.0 );\n\tskinned += boneMatX * skinVertex * skinWeight.x;\n\tskinned += boneMatY * skinVertex * skinWeight.y;\n\tskinned += boneMatZ * skinVertex * skinWeight.z;\n\tskinned += boneMatW * skinVertex * skinWeight.w;\n\ttransformed = ( bindMatrixInverse * skinned ).xyz;\n#endif",skinnormal_vertex:"#ifdef USE_SKINNING\n\tmat4 skinMatrix = mat4( 0.0 );\n\tskinMatrix += skinWeight.x * boneMatX;\n\tskinMatrix += skinWeight.y * boneMatY;\n\tskinMatrix += skinWeight.z * boneMatZ;\n\tskinMatrix += skinWeight.w * boneMatW;\n\tskinMatrix = bindMatrixInverse * skinMatrix * bindMatrix;\n\tobjectNormal = vec4( skinMatrix * vec4( objectNormal, 0.0 ) ).xyz;\n\t#ifdef USE_TANGENT\n\t\tobjectTangent = vec4( skinMatrix * vec4( objectTangent, 0.0 ) ).xyz;\n\t#endif\n#endif",specularmap_fragment:"float specularStrength;\n#ifdef USE_SPECULARMAP\n\tvec4 texelSpecular = texture2D( specularMap, vUv );\n\tspecularStrength = texelSpecular.r;\n#else\n\tspecularStrength = 1.0;\n#endif",specularmap_pars_fragment:"#ifdef USE_SPECULARMAP\n\tuniform sampler2D specularMap;\n#endif",tonemapping_fragment:"#if defined( TONE_MAPPING )\n\tgl_FragColor.rgb = toneMapping( gl_FragColor.rgb );\n#endif",tonemapping_pars_fragment:"#ifndef saturate\n#define saturate( a ) clamp( a, 0.0, 1.0 )\n#endif\nuniform float toneMappingExposure;\nvec3 LinearToneMapping( vec3 color ) {\n\treturn toneMappingExposure * color;\n}\nvec3 ReinhardToneMapping( vec3 color ) {\n\tcolor *= toneMappingExposure;\n\treturn saturate( color / ( vec3( 1.0 ) + color ) );\n}\nvec3 OptimizedCineonToneMapping( vec3 color ) {\n\tcolor *= toneMappingExposure;\n\tcolor = max( vec3( 0.0 ), color - 0.004 );\n\treturn pow( ( color * ( 6.2 * color + 0.5 ) ) / ( color * ( 6.2 * color + 1.7 ) + 0.06 ), vec3( 2.2 ) );\n}\nvec3 RRTAndODTFit( vec3 v ) {\n\tvec3 a = v * ( v + 0.0245786 ) - 0.000090537;\n\tvec3 b = v * ( 0.983729 * v + 0.4329510 ) + 0.238081;\n\treturn a / b;\n}\nvec3 ACESFilmicToneMapping( vec3 color ) {\n\tconst mat3 ACESInputMat = mat3(\n\t\tvec3( 0.59719, 0.07600, 0.02840 ),\t\tvec3( 0.35458, 0.90834, 0.13383 ),\n\t\tvec3( 0.04823, 0.01566, 0.83777 )\n\t);\n\tconst mat3 ACESOutputMat = mat3(\n\t\tvec3(\t1.60475, -0.10208, -0.00327 ),\t\tvec3( -0.53108,\t1.10813, -0.07276 ),\n\t\tvec3( -0.07367, -0.00605,\t1.07602 )\n\t);\n\tcolor *= toneMappingExposure / 0.6;\n\tcolor = ACESInputMat * color;\n\tcolor = RRTAndODTFit( color );\n\tcolor = ACESOutputMat * color;\n\treturn saturate( color );\n}\nvec3 CustomToneMapping( vec3 color ) { return color; }",transmission_fragment:"#ifdef USE_TRANSMISSION\n\tmaterial.transmission = transmission;\n\tmaterial.transmissionAlpha = 1.0;\n\tmaterial.thickness = thickness;\n\tmaterial.attenuationDistance = attenuationDistance;\n\tmaterial.attenuationColor = attenuationColor;\n\t#ifdef USE_TRANSMISSIONMAP\n\t\tmaterial.transmission *= texture2D( transmissionMap, vUv ).r;\n\t#endif\n\t#ifdef USE_THICKNESSMAP\n\t\tmaterial.thickness *= texture2D( thicknessMap, vUv ).g;\n\t#endif\n\tvec3 pos = vWorldPosition;\n\tvec3 v = normalize( cameraPosition - pos );\n\tvec3 n = inverseTransformDirection( normal, viewMatrix );\n\tvec4 transmission = getIBLVolumeRefraction(\n\t\tn, v, material.roughness, material.diffuseColor, material.specularColor, material.specularF90,\n\t\tpos, modelMatrix, viewMatrix, projectionMatrix, material.ior, material.thickness,\n\t\tmaterial.attenuationColor, material.attenuationDistance );\n\tmaterial.transmissionAlpha = mix( material.transmissionAlpha, transmission.a, material.transmission );\n\ttotalDiffuse = mix( totalDiffuse, transmission.rgb, material.transmission );\n#endif",transmission_pars_fragment:"#ifdef USE_TRANSMISSION\n\tuniform float transmission;\n\tuniform float thickness;\n\tuniform float attenuationDistance;\n\tuniform vec3 attenuationColor;\n\t#ifdef USE_TRANSMISSIONMAP\n\t\tuniform sampler2D transmissionMap;\n\t#endif\n\t#ifdef USE_THICKNESSMAP\n\t\tuniform sampler2D thicknessMap;\n\t#endif\n\tuniform vec2 transmissionSamplerSize;\n\tuniform sampler2D transmissionSamplerMap;\n\tuniform mat4 modelMatrix;\n\tuniform mat4 projectionMatrix;\n\tvarying vec3 vWorldPosition;\n\tvec3 getVolumeTransmissionRay( const in vec3 n, const in vec3 v, const in float thickness, const in float ior, const in mat4 modelMatrix ) {\n\t\tvec3 refractionVector = refract( - v, normalize( n ), 1.0 / ior );\n\t\tvec3 modelScale;\n\t\tmodelScale.x = length( vec3( modelMatrix[ 0 ].xyz ) );\n\t\tmodelScale.y = length( vec3( modelMatrix[ 1 ].xyz ) );\n\t\tmodelScale.z = length( vec3( modelMatrix[ 2 ].xyz ) );\n\t\treturn normalize( refractionVector ) * thickness * modelScale;\n\t}\n\tfloat applyIorToRoughness( const in float roughness, const in float ior ) {\n\t\treturn roughness * clamp( ior * 2.0 - 2.0, 0.0, 1.0 );\n\t}\n\tvec4 getTransmissionSample( const in vec2 fragCoord, const in float roughness, const in float ior ) {\n\t\tfloat framebufferLod = log2( transmissionSamplerSize.x ) * applyIorToRoughness( roughness, ior );\n\t\t#ifdef texture2DLodEXT\n\t\t\treturn texture2DLodEXT( transmissionSamplerMap, fragCoord.xy, framebufferLod );\n\t\t#else\n\t\t\treturn texture2D( transmissionSamplerMap, fragCoord.xy, framebufferLod );\n\t\t#endif\n\t}\n\tvec3 applyVolumeAttenuation( const in vec3 radiance, const in float transmissionDistance, const in vec3 attenuationColor, const in float attenuationDistance ) {\n\t\tif ( isinf( attenuationDistance ) ) {\n\t\t\treturn radiance;\n\t\t} else {\n\t\t\tvec3 attenuationCoefficient = -log( attenuationColor ) / attenuationDistance;\n\t\t\tvec3 transmittance = exp( - attenuationCoefficient * transmissionDistance );\t\t\treturn transmittance * radiance;\n\t\t}\n\t}\n\tvec4 getIBLVolumeRefraction( const in vec3 n, const in vec3 v, const in float roughness, const in vec3 diffuseColor,\n\t\tconst in vec3 specularColor, const in float specularF90, const in vec3 position, const in mat4 modelMatrix,\n\t\tconst in mat4 viewMatrix, const in mat4 projMatrix, const in float ior, const in float thickness,\n\t\tconst in vec3 attenuationColor, const in float attenuationDistance ) {\n\t\tvec3 transmissionRay = getVolumeTransmissionRay( n, v, thickness, ior, modelMatrix );\n\t\tvec3 refractedRayExit = position + transmissionRay;\n\t\tvec4 ndcPos = projMatrix * viewMatrix * vec4( refractedRayExit, 1.0 );\n\t\tvec2 refractionCoords = ndcPos.xy / ndcPos.w;\n\t\trefractionCoords += 1.0;\n\t\trefractionCoords /= 2.0;\n\t\tvec4 transmittedLight = getTransmissionSample( refractionCoords, roughness, ior );\n\t\tvec3 attenuatedColor = applyVolumeAttenuation( transmittedLight.rgb, length( transmissionRay ), attenuationColor, attenuationDistance );\n\t\tvec3 F = EnvironmentBRDF( n, v, specularColor, specularF90, roughness );\n\t\treturn vec4( ( 1.0 - F ) * attenuatedColor * diffuseColor, transmittedLight.a );\n\t}\n#endif",uv_pars_fragment:"#if ( defined( USE_UV ) && ! defined( UVS_VERTEX_ONLY ) )\n\tvarying vec2 vUv;\n#endif",uv_pars_vertex:"#ifdef USE_UV\n\t#ifdef UVS_VERTEX_ONLY\n\t\tvec2 vUv;\n\t#else\n\t\tvarying vec2 vUv;\n\t#endif\n\tuniform mat3 uvTransform;\n#endif",uv_vertex:"#ifdef USE_UV\n\tvUv = ( uvTransform * vec3( uv, 1 ) ).xy;\n#endif",uv2_pars_fragment:"#if defined( USE_LIGHTMAP ) || defined( USE_AOMAP )\n\tvarying vec2 vUv2;\n#endif",uv2_pars_vertex:"#if defined( USE_LIGHTMAP ) || defined( USE_AOMAP )\n\tattribute vec2 uv2;\n\tvarying vec2 vUv2;\n\tuniform mat3 uv2Transform;\n#endif",uv2_vertex:"#if defined( USE_LIGHTMAP ) || defined( USE_AOMAP )\n\tvUv2 = ( uv2Transform * vec3( uv2, 1 ) ).xy;\n#endif",worldpos_vertex:"#if defined( USE_ENVMAP ) || defined( DISTANCE ) || defined ( USE_SHADOWMAP ) || defined ( USE_TRANSMISSION ) || NUM_SPOT_LIGHT_COORDS > 0\n\tvec4 worldPosition = vec4( transformed, 1.0 );\n\t#ifdef USE_INSTANCING\n\t\tworldPosition = instanceMatrix * worldPosition;\n\t#endif\n\tworldPosition = modelMatrix * worldPosition;\n#endif",background_vert:"varying vec2 vUv;\nuniform mat3 uvTransform;\nvoid main() {\n\tvUv = ( uvTransform * vec3( uv, 1 ) ).xy;\n\tgl_Position = vec4( position.xy, 1.0, 1.0 );\n}",background_frag:"uniform sampler2D t2D;\nvarying vec2 vUv;\nvoid main() {\n\tgl_FragColor = texture2D( t2D, vUv );\n\t#ifdef DECODE_VIDEO_TEXTURE\n\t\tgl_FragColor = vec4( mix( pow( gl_FragColor.rgb * 0.9478672986 + vec3( 0.0521327014 ), vec3( 2.4 ) ), gl_FragColor.rgb * 0.0773993808, vec3( lessThanEqual( gl_FragColor.rgb, vec3( 0.04045 ) ) ) ), gl_FragColor.w );\n\t#endif\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}",backgroundCube_vert:"varying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvWorldDirection = transformDirection( position, modelMatrix );\n\t#include <begin_vertex>\n\t#include <project_vertex>\n\tgl_Position.z = gl_Position.w;\n}",backgroundCube_frag:"#ifdef ENVMAP_TYPE_CUBE\n\tuniform samplerCube envMap;\n#elif defined( ENVMAP_TYPE_CUBE_UV )\n\tuniform sampler2D envMap;\n#endif\nuniform float flipEnvMap;\nuniform float backgroundBlurriness;\nvarying vec3 vWorldDirection;\n#include <cube_uv_reflection_fragment>\nvoid main() {\n\t#ifdef ENVMAP_TYPE_CUBE\n\t\tvec4 texColor = textureCube( envMap, vec3( flipEnvMap * vWorldDirection.x, vWorldDirection.yz ) );\n\t#elif defined( ENVMAP_TYPE_CUBE_UV )\n\t\tvec4 texColor = textureCubeUV( envMap, vWorldDirection, backgroundBlurriness );\n\t#else\n\t\tvec4 texColor = vec4( 0.0, 0.0, 0.0, 1.0 );\n\t#endif\n\tgl_FragColor = texColor;\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}",cube_vert:"varying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvWorldDirection = transformDirection( position, modelMatrix );\n\t#include <begin_vertex>\n\t#include <project_vertex>\n\tgl_Position.z = gl_Position.w;\n}",cube_frag:"uniform samplerCube tCube;\nuniform float tFlip;\nuniform float opacity;\nvarying vec3 vWorldDirection;\nvoid main() {\n\tvec4 texColor = textureCube( tCube, vec3( tFlip * vWorldDirection.x, vWorldDirection.yz ) );\n\tgl_FragColor = texColor;\n\tgl_FragColor.a *= opacity;\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}",depth_vert:"#include <common>\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvarying vec2 vHighPrecisionZW;\nvoid main() {\n\t#include <uv_vertex>\n\t#include <skinbase_vertex>\n\t#ifdef USE_DISPLACEMENTMAP\n\t\t#include <beginnormal_vertex>\n\t\t#include <morphnormal_vertex>\n\t\t#include <skinnormal_vertex>\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvHighPrecisionZW = gl_Position.zw;\n}",depth_frag:"#if DEPTH_PACKING == 3200\n\tuniform float opacity;\n#endif\n#include <common>\n#include <packing>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvarying vec2 vHighPrecisionZW;\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( 1.0 );\n\t#if DEPTH_PACKING == 3200\n\t\tdiffuseColor.a = opacity;\n\t#endif\n\t#include <map_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <logdepthbuf_fragment>\n\tfloat fragCoordZ = 0.5 * vHighPrecisionZW[0] / vHighPrecisionZW[1] + 0.5;\n\t#if DEPTH_PACKING == 3200\n\t\tgl_FragColor = vec4( vec3( 1.0 - fragCoordZ ), opacity );\n\t#elif DEPTH_PACKING == 3201\n\t\tgl_FragColor = packDepthToRGBA( fragCoordZ );\n\t#endif\n}",distanceRGBA_vert:"#define DISTANCE\nvarying vec3 vWorldPosition;\n#include <common>\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <skinbase_vertex>\n\t#ifdef USE_DISPLACEMENTMAP\n\t\t#include <beginnormal_vertex>\n\t\t#include <morphnormal_vertex>\n\t\t#include <skinnormal_vertex>\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <worldpos_vertex>\n\t#include <clipping_planes_vertex>\n\tvWorldPosition = worldPosition.xyz;\n}",distanceRGBA_frag:"#define DISTANCE\nuniform vec3 referencePosition;\nuniform float nearDistance;\nuniform float farDistance;\nvarying vec3 vWorldPosition;\n#include <common>\n#include <packing>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main () {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( 1.0 );\n\t#include <map_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\tfloat dist = length( vWorldPosition - referencePosition );\n\tdist = ( dist - nearDistance ) / ( farDistance - nearDistance );\n\tdist = saturate( dist );\n\tgl_FragColor = packDepthToRGBA( dist );\n}",equirect_vert:"varying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvWorldDirection = transformDirection( position, modelMatrix );\n\t#include <begin_vertex>\n\t#include <project_vertex>\n}",equirect_frag:"uniform sampler2D tEquirect;\nvarying vec3 vWorldDirection;\n#include <common>\nvoid main() {\n\tvec3 direction = normalize( vWorldDirection );\n\tvec2 sampleUV = equirectUv( direction );\n\tgl_FragColor = texture2D( tEquirect, sampleUV );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n}",linedashed_vert:"uniform float scale;\nattribute float lineDistance;\nvarying float vLineDistance;\n#include <common>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\tvLineDistance = scale * lineDistance;\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <fog_vertex>\n}",linedashed_frag:"uniform vec3 diffuse;\nuniform float opacity;\nuniform float dashSize;\nuniform float totalSize;\nvarying float vLineDistance;\n#include <common>\n#include <color_pars_fragment>\n#include <fog_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tif ( mod( vLineDistance, totalSize ) > dashSize ) {\n\t\tdiscard;\n\t}\n\tvec3 outgoingLight = vec3( 0.0 );\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <color_fragment>\n\toutgoingLight = diffuseColor.rgb;\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n}",meshbasic_vert:"#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <envmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#if defined ( USE_ENVMAP ) || defined ( USE_SKINNING )\n\t\t#include <beginnormal_vertex>\n\t\t#include <morphnormal_vertex>\n\t\t#include <skinbase_vertex>\n\t\t#include <skinnormal_vertex>\n\t\t#include <defaultnormal_vertex>\n\t#endif\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <worldpos_vertex>\n\t#include <envmap_vertex>\n\t#include <fog_vertex>\n}",meshbasic_frag:"uniform vec3 diffuse;\nuniform float opacity;\n#ifndef FLAT_SHADED\n\tvarying vec3 vNormal;\n#endif\n#include <common>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_pars_fragment>\n#include <fog_pars_fragment>\n#include <specularmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <specularmap_fragment>\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\t#ifdef USE_LIGHTMAP\n\t\tvec4 lightMapTexel = texture2D( lightMap, vUv2 );\n\t\treflectedLight.indirectDiffuse += lightMapTexel.rgb * lightMapIntensity * RECIPROCAL_PI;\n\t#else\n\t\treflectedLight.indirectDiffuse += vec3( 1.0 );\n\t#endif\n\t#include <aomap_fragment>\n\treflectedLight.indirectDiffuse *= diffuseColor.rgb;\n\tvec3 outgoingLight = reflectedLight.indirectDiffuse;\n\t#include <envmap_fragment>\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}",meshlambert_vert:"#define LAMBERT\nvarying vec3 vViewPosition;\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <envmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <normal_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <normal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvViewPosition = - mvPosition.xyz;\n\t#include <worldpos_vertex>\n\t#include <envmap_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}",meshlambert_frag:"#define LAMBERT\nuniform vec3 diffuse;\nuniform vec3 emissive;\nuniform float opacity;\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_pars_fragment>\n#include <fog_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <normal_pars_fragment>\n#include <lights_lambert_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <specularmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <specularmap_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\t#include <emissivemap_fragment>\n\t#include <lights_lambert_fragment>\n\t#include <lights_fragment_begin>\n\t#include <lights_fragment_maps>\n\t#include <lights_fragment_end>\n\t#include <aomap_fragment>\n\tvec3 outgoingLight = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse + totalEmissiveRadiance;\n\t#include <envmap_fragment>\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}",meshmatcap_vert:"#define MATCAP\nvarying vec3 vViewPosition;\n#include <common>\n#include <uv_pars_vertex>\n#include <color_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <fog_pars_vertex>\n#include <normal_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <normal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <fog_vertex>\n\tvViewPosition = - mvPosition.xyz;\n}",meshmatcap_frag:"#define MATCAP\nuniform vec3 diffuse;\nuniform float opacity;\nuniform sampler2D matcap;\nvarying vec3 vViewPosition;\n#include <common>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <fog_pars_fragment>\n#include <normal_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\tvec3 viewDir = normalize( vViewPosition );\n\tvec3 x = normalize( vec3( viewDir.z, 0.0, - viewDir.x ) );\n\tvec3 y = cross( viewDir, x );\n\tvec2 uv = vec2( dot( x, normal ), dot( y, normal ) ) * 0.495 + 0.5;\n\t#ifdef USE_MATCAP\n\t\tvec4 matcapColor = texture2D( matcap, uv );\n\t#else\n\t\tvec4 matcapColor = vec4( vec3( mix( 0.2, 0.8, uv.y ) ), 1.0 );\n\t#endif\n\tvec3 outgoingLight = diffuseColor.rgb * matcapColor.rgb;\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}",meshnormal_vert:"#define NORMAL\n#if defined( FLAT_SHADED ) || defined( USE_BUMPMAP ) || defined( TANGENTSPACE_NORMALMAP )\n\tvarying vec3 vViewPosition;\n#endif\n#include <common>\n#include <uv_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <normal_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <normal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n#if defined( FLAT_SHADED ) || defined( USE_BUMPMAP ) || defined( TANGENTSPACE_NORMALMAP )\n\tvViewPosition = - mvPosition.xyz;\n#endif\n}",meshnormal_frag:"#define NORMAL\nuniform float opacity;\n#if defined( FLAT_SHADED ) || defined( USE_BUMPMAP ) || defined( TANGENTSPACE_NORMALMAP )\n\tvarying vec3 vViewPosition;\n#endif\n#include <packing>\n#include <uv_pars_fragment>\n#include <normal_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\t#include <logdepthbuf_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\tgl_FragColor = vec4( packNormalToRGB( normal ), opacity );\n\t#ifdef OPAQUE\n\t\tgl_FragColor.a = 1.0;\n\t#endif\n}",meshphong_vert:"#define PHONG\nvarying vec3 vViewPosition;\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <envmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <normal_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <normal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvViewPosition = - mvPosition.xyz;\n\t#include <worldpos_vertex>\n\t#include <envmap_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}",meshphong_frag:"#define PHONG\nuniform vec3 diffuse;\nuniform vec3 emissive;\nuniform vec3 specular;\nuniform float shininess;\nuniform float opacity;\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_pars_fragment>\n#include <fog_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <normal_pars_fragment>\n#include <lights_phong_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <specularmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <specularmap_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\t#include <emissivemap_fragment>\n\t#include <lights_phong_fragment>\n\t#include <lights_fragment_begin>\n\t#include <lights_fragment_maps>\n\t#include <lights_fragment_end>\n\t#include <aomap_fragment>\n\tvec3 outgoingLight = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse + reflectedLight.directSpecular + reflectedLight.indirectSpecular + totalEmissiveRadiance;\n\t#include <envmap_fragment>\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}",meshphysical_vert:"#define STANDARD\nvarying vec3 vViewPosition;\n#ifdef USE_TRANSMISSION\n\tvarying vec3 vWorldPosition;\n#endif\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <normal_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <normal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvViewPosition = - mvPosition.xyz;\n\t#include <worldpos_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n#ifdef USE_TRANSMISSION\n\tvWorldPosition = worldPosition.xyz;\n#endif\n}",meshphysical_frag:"#define STANDARD\n#ifdef PHYSICAL\n\t#define IOR\n\t#define SPECULAR\n#endif\nuniform vec3 diffuse;\nuniform vec3 emissive;\nuniform float roughness;\nuniform float metalness;\nuniform float opacity;\n#ifdef IOR\n\tuniform float ior;\n#endif\n#ifdef SPECULAR\n\tuniform float specularIntensity;\n\tuniform vec3 specularColor;\n\t#ifdef USE_SPECULARINTENSITYMAP\n\t\tuniform sampler2D specularIntensityMap;\n\t#endif\n\t#ifdef USE_SPECULARCOLORMAP\n\t\tuniform sampler2D specularColorMap;\n\t#endif\n#endif\n#ifdef USE_CLEARCOAT\n\tuniform float clearcoat;\n\tuniform float clearcoatRoughness;\n#endif\n#ifdef USE_IRIDESCENCE\n\tuniform float iridescence;\n\tuniform float iridescenceIOR;\n\tuniform float iridescenceThicknessMinimum;\n\tuniform float iridescenceThicknessMaximum;\n#endif\n#ifdef USE_SHEEN\n\tuniform vec3 sheenColor;\n\tuniform float sheenRoughness;\n\t#ifdef USE_SHEENCOLORMAP\n\t\tuniform sampler2D sheenColorMap;\n\t#endif\n\t#ifdef USE_SHEENROUGHNESSMAP\n\t\tuniform sampler2D sheenRoughnessMap;\n\t#endif\n#endif\nvarying vec3 vViewPosition;\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <bsdfs>\n#include <iridescence_fragment>\n#include <cube_uv_reflection_fragment>\n#include <envmap_common_pars_fragment>\n#include <envmap_physical_pars_fragment>\n#include <fog_pars_fragment>\n#include <lights_pars_begin>\n#include <normal_pars_fragment>\n#include <lights_physical_pars_fragment>\n#include <transmission_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <clearcoat_pars_fragment>\n#include <iridescence_pars_fragment>\n#include <roughnessmap_pars_fragment>\n#include <metalnessmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <roughnessmap_fragment>\n\t#include <metalnessmap_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\t#include <clearcoat_normal_fragment_begin>\n\t#include <clearcoat_normal_fragment_maps>\n\t#include <emissivemap_fragment>\n\t#include <lights_physical_fragment>\n\t#include <lights_fragment_begin>\n\t#include <lights_fragment_maps>\n\t#include <lights_fragment_end>\n\t#include <aomap_fragment>\n\tvec3 totalDiffuse = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse;\n\tvec3 totalSpecular = reflectedLight.directSpecular + reflectedLight.indirectSpecular;\n\t#include <transmission_fragment>\n\tvec3 outgoingLight = totalDiffuse + totalSpecular + totalEmissiveRadiance;\n\t#ifdef USE_SHEEN\n\t\tfloat sheenEnergyComp = 1.0 - 0.157 * max3( material.sheenColor );\n\t\toutgoingLight = outgoingLight * sheenEnergyComp + sheenSpecular;\n\t#endif\n\t#ifdef USE_CLEARCOAT\n\t\tfloat dotNVcc = saturate( dot( geometry.clearcoatNormal, geometry.viewDir ) );\n\t\tvec3 Fcc = F_Schlick( material.clearcoatF0, material.clearcoatF90, dotNVcc );\n\t\toutgoingLight = outgoingLight * ( 1.0 - material.clearcoat * Fcc ) + clearcoatSpecular * material.clearcoat;\n\t#endif\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}",meshtoon_vert:"#define TOON\nvarying vec3 vViewPosition;\n#include <common>\n#include <uv_pars_vertex>\n#include <uv2_pars_vertex>\n#include <displacementmap_pars_vertex>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <normal_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\t#include <uv2_vertex>\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <normal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <displacementmap_vertex>\n\t#include <project_vertex>\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\tvViewPosition = - mvPosition.xyz;\n\t#include <worldpos_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}",meshtoon_frag:"#define TOON\nuniform vec3 diffuse;\nuniform vec3 emissive;\nuniform float opacity;\n#include <common>\n#include <packing>\n#include <dithering_pars_fragment>\n#include <color_pars_fragment>\n#include <uv_pars_fragment>\n#include <uv2_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <aomap_pars_fragment>\n#include <lightmap_pars_fragment>\n#include <emissivemap_pars_fragment>\n#include <gradientmap_pars_fragment>\n#include <fog_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <normal_pars_fragment>\n#include <lights_toon_pars_fragment>\n#include <shadowmap_pars_fragment>\n#include <bumpmap_pars_fragment>\n#include <normalmap_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\tReflectedLight reflectedLight = ReflectedLight( vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ), vec3( 0.0 ) );\n\tvec3 totalEmissiveRadiance = emissive;\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <color_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\t#include <normal_fragment_begin>\n\t#include <normal_fragment_maps>\n\t#include <emissivemap_fragment>\n\t#include <lights_toon_fragment>\n\t#include <lights_fragment_begin>\n\t#include <lights_fragment_maps>\n\t#include <lights_fragment_end>\n\t#include <aomap_fragment>\n\tvec3 outgoingLight = reflectedLight.directDiffuse + reflectedLight.indirectDiffuse + totalEmissiveRadiance;\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n\t#include <dithering_fragment>\n}",points_vert:"uniform float size;\nuniform float scale;\n#include <common>\n#include <color_pars_vertex>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <color_vertex>\n\t#include <morphcolor_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <project_vertex>\n\tgl_PointSize = size;\n\t#ifdef USE_SIZEATTENUATION\n\t\tbool isPerspective = isPerspectiveMatrix( projectionMatrix );\n\t\tif ( isPerspective ) gl_PointSize *= ( scale / - mvPosition.z );\n\t#endif\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <worldpos_vertex>\n\t#include <fog_vertex>\n}",points_frag:"uniform vec3 diffuse;\nuniform float opacity;\n#include <common>\n#include <color_pars_fragment>\n#include <map_particle_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <fog_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec3 outgoingLight = vec3( 0.0 );\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_particle_fragment>\n\t#include <color_fragment>\n\t#include <alphatest_fragment>\n\toutgoingLight = diffuseColor.rgb;\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n\t#include <premultiplied_alpha_fragment>\n}",shadow_vert:"#include <common>\n#include <fog_pars_vertex>\n#include <morphtarget_pars_vertex>\n#include <skinning_pars_vertex>\n#include <shadowmap_pars_vertex>\nvoid main() {\n\t#include <beginnormal_vertex>\n\t#include <morphnormal_vertex>\n\t#include <skinbase_vertex>\n\t#include <skinnormal_vertex>\n\t#include <defaultnormal_vertex>\n\t#include <begin_vertex>\n\t#include <morphtarget_vertex>\n\t#include <skinning_vertex>\n\t#include <project_vertex>\n\t#include <worldpos_vertex>\n\t#include <shadowmap_vertex>\n\t#include <fog_vertex>\n}",shadow_frag:"uniform vec3 color;\nuniform float opacity;\n#include <common>\n#include <packing>\n#include <fog_pars_fragment>\n#include <bsdfs>\n#include <lights_pars_begin>\n#include <shadowmap_pars_fragment>\n#include <shadowmask_pars_fragment>\nvoid main() {\n\tgl_FragColor = vec4( color, opacity * ( 1.0 - getShadowMask() ) );\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}",sprite_vert:"uniform float rotation;\nuniform vec2 center;\n#include <common>\n#include <uv_pars_vertex>\n#include <fog_pars_vertex>\n#include <logdepthbuf_pars_vertex>\n#include <clipping_planes_pars_vertex>\nvoid main() {\n\t#include <uv_vertex>\n\tvec4 mvPosition = modelViewMatrix * vec4( 0.0, 0.0, 0.0, 1.0 );\n\tvec2 scale;\n\tscale.x = length( vec3( modelMatrix[ 0 ].x, modelMatrix[ 0 ].y, modelMatrix[ 0 ].z ) );\n\tscale.y = length( vec3( modelMatrix[ 1 ].x, modelMatrix[ 1 ].y, modelMatrix[ 1 ].z ) );\n\t#ifndef USE_SIZEATTENUATION\n\t\tbool isPerspective = isPerspectiveMatrix( projectionMatrix );\n\t\tif ( isPerspective ) scale *= - mvPosition.z;\n\t#endif\n\tvec2 alignedPosition = ( position.xy - ( center - vec2( 0.5 ) ) ) * scale;\n\tvec2 rotatedPosition;\n\trotatedPosition.x = cos( rotation ) * alignedPosition.x - sin( rotation ) * alignedPosition.y;\n\trotatedPosition.y = sin( rotation ) * alignedPosition.x + cos( rotation ) * alignedPosition.y;\n\tmvPosition.xy += rotatedPosition;\n\tgl_Position = projectionMatrix * mvPosition;\n\t#include <logdepthbuf_vertex>\n\t#include <clipping_planes_vertex>\n\t#include <fog_vertex>\n}",sprite_frag:"uniform vec3 diffuse;\nuniform float opacity;\n#include <common>\n#include <uv_pars_fragment>\n#include <map_pars_fragment>\n#include <alphamap_pars_fragment>\n#include <alphatest_pars_fragment>\n#include <fog_pars_fragment>\n#include <logdepthbuf_pars_fragment>\n#include <clipping_planes_pars_fragment>\nvoid main() {\n\t#include <clipping_planes_fragment>\n\tvec3 outgoingLight = vec3( 0.0 );\n\tvec4 diffuseColor = vec4( diffuse, opacity );\n\t#include <logdepthbuf_fragment>\n\t#include <map_fragment>\n\t#include <alphamap_fragment>\n\t#include <alphatest_fragment>\n\toutgoingLight = diffuseColor.rgb;\n\t#include <output_fragment>\n\t#include <tonemapping_fragment>\n\t#include <encodings_fragment>\n\t#include <fog_fragment>\n}"},Mn={common:{diffuse:{value:new jt(16777215)},opacity:{value:1},map:{value:null},uvTransform:{value:new Rt},uv2Transform:{value:new Rt},alphaMap:{value:null},alphaTest:{value:0}},specularmap:{specularMap:{value:null}},envmap:{envMap:{value:null},flipEnvMap:{value:-1},reflectivity:{value:1},ior:{value:1.5},refractionRatio:{value:.98}},aomap:{aoMap:{value:null},aoMapIntensity:{value:1}},lightmap:{lightMap:{value:null},lightMapIntensity:{value:1}},emissivemap:{emissiveMap:{value:null}},bumpmap:{bumpMap:{value:null},bumpScale:{value:1}},normalmap:{normalMap:{value:null},normalScale:{value:new Lt(1,1)}},displacementmap:{displacementMap:{value:null},displacementScale:{value:1},displacementBias:{value:0}},roughnessmap:{roughnessMap:{value:null}},metalnessmap:{metalnessMap:{value:null}},gradientmap:{gradientMap:{value:null}},fog:{fogDensity:{value:25e-5},fogNear:{value:1},fogFar:{value:2e3},fogColor:{value:new jt(16777215)}},lights:{ambientLightColor:{value:[]},lightProbe:{value:[]},directionalLights:{value:[],properties:{direction:{},color:{}}},directionalLightShadows:{value:[],properties:{shadowBias:{},shadowNormalBias:{},shadowRadius:{},shadowMapSize:{}}},directionalShadowMap:{value:[]},directionalShadowMatrix:{value:[]},spotLights:{value:[],properties:{color:{},position:{},direction:{},distance:{},coneCos:{},penumbraCos:{},decay:{}}},spotLightShadows:{value:[],properties:{shadowBias:{},shadowNormalBias:{},shadowRadius:{},shadowMapSize:{}}},spotLightMap:{value:[]},spotShadowMap:{value:[]},spotLightMatrix:{value:[]},pointLights:{value:[],properties:{color:{},position:{},decay:{},distance:{}}},pointLightShadows:{value:[],properties:{shadowBias:{},shadowNormalBias:{},shadowRadius:{},shadowMapSize:{},shadowCameraNear:{},shadowCameraFar:{}}},pointShadowMap:{value:[]},pointShadowMatrix:{value:[]},hemisphereLights:{value:[],properties:{direction:{},skyColor:{},groundColor:{}}},rectAreaLights:{value:[],properties:{color:{},position:{},width:{},height:{}}},ltc_1:{value:null},ltc_2:{value:null}},points:{diffuse:{value:new jt(16777215)},opacity:{value:1},size:{value:1},scale:{value:1},map:{value:null},alphaMap:{value:null},alphaTest:{value:0},uvTransform:{value:new Rt}},sprite:{diffuse:{value:new jt(16777215)},opacity:{value:1},center:{value:new Lt(.5,.5)},rotation:{value:0},map:{value:null},alphaMap:{value:null},alphaTest:{value:0},uvTransform:{value:new Rt}}},bn={basic:{uniforms:tn([Mn.common,Mn.specularmap,Mn.envmap,Mn.aomap,Mn.lightmap,Mn.fog]),vertexShader:yn.meshbasic_vert,fragmentShader:yn.meshbasic_frag},lambert:{uniforms:tn([Mn.common,Mn.specularmap,Mn.envmap,Mn.aomap,Mn.lightmap,Mn.emissivemap,Mn.bumpmap,Mn.normalmap,Mn.displacementmap,Mn.fog,Mn.lights,{emissive:{value:new jt(0)}}]),vertexShader:yn.meshlambert_vert,fragmentShader:yn.meshlambert_frag},phong:{uniforms:tn([Mn.common,Mn.specularmap,Mn.envmap,Mn.aomap,Mn.lightmap,Mn.emissivemap,Mn.bumpmap,Mn.normalmap,Mn.displacementmap,Mn.fog,Mn.lights,{emissive:{value:new jt(0)},specular:{value:new jt(1118481)},shininess:{value:30}}]),vertexShader:yn.meshphong_vert,fragmentShader:yn.meshphong_frag},standard:{uniforms:tn([Mn.common,Mn.envmap,Mn.aomap,Mn.lightmap,Mn.emissivemap,Mn.bumpmap,Mn.normalmap,Mn.displacementmap,Mn.roughnessmap,Mn.metalnessmap,Mn.fog,Mn.lights,{emissive:{value:new jt(0)},roughness:{value:1},metalness:{value:0},envMapIntensity:{value:1}}]),vertexShader:yn.meshphysical_vert,fragmentShader:yn.meshphysical_frag},toon:{uniforms:tn([Mn.common,Mn.aomap,Mn.lightmap,Mn.emissivemap,Mn.bumpmap,Mn.normalmap,Mn.displacementmap,Mn.gradientmap,Mn.fog,Mn.lights,{emissive:{value:new jt(0)}}]),vertexShader:yn.meshtoon_vert,fragmentShader:yn.meshtoon_frag},matcap:{uniforms:tn([Mn.common,Mn.bumpmap,Mn.normalmap,Mn.displacementmap,Mn.fog,{matcap:{value:null}}]),vertexShader:yn.meshmatcap_vert,fragmentShader:yn.meshmatcap_frag},points:{uniforms:tn([Mn.points,Mn.fog]),vertexShader:yn.points_vert,fragmentShader:yn.points_frag},dashed:{uniforms:tn([Mn.common,Mn.fog,{scale:{value:1},dashSize:{value:1},totalSize:{value:2}}]),vertexShader:yn.linedashed_vert,fragmentShader:yn.linedashed_frag},depth:{uniforms:tn([Mn.common,Mn.displacementmap]),vertexShader:yn.depth_vert,fragmentShader:yn.depth_frag},normal:{uniforms:tn([Mn.common,Mn.bumpmap,Mn.normalmap,Mn.displacementmap,{opacity:{value:1}}]),vertexShader:yn.meshnormal_vert,fragmentShader:yn.meshnormal_frag},sprite:{uniforms:tn([Mn.sprite,Mn.fog]),vertexShader:yn.sprite_vert,fragmentShader:yn.sprite_frag},background:{uniforms:{uvTransform:{value:new Rt},t2D:{value:null}},vertexShader:yn.background_vert,fragmentShader:yn.background_frag},backgroundCube:{uniforms:{envMap:{value:null},flipEnvMap:{value:-1},backgroundBlurriness:{value:0}},vertexShader:yn.backgroundCube_vert,fragmentShader:yn.backgroundCube_frag},cube:{uniforms:{tCube:{value:null},tFlip:{value:-1},opacity:{value:1}},vertexShader:yn.cube_vert,fragmentShader:yn.cube_frag},equirect:{uniforms:{tEquirect:{value:null}},vertexShader:yn.equirect_vert,fragmentShader:yn.equirect_frag},distanceRGBA:{uniforms:tn([Mn.common,Mn.displacementmap,{referencePosition:{value:new ne},nearDistance:{value:1},farDistance:{value:1e3}}]),vertexShader:yn.distanceRGBA_vert,fragmentShader:yn.distanceRGBA_frag},shadow:{uniforms:tn([Mn.lights,Mn.fog,{color:{value:new jt(0)},opacity:{value:1}}]),vertexShader:yn.shadow_vert,fragmentShader:yn.shadow_frag}};function Sn(t,e,i,n,r,s,a){const o=new jt(0);let c,h,u=!0===s?0:1,d=null,p=0,m=null;function f(t,e){n.buffers.color.setClear(t.r,t.g,t.b,e,a)}return{getClearColor:function(){return o},setClearColor:function(t,e=1){o.set(t),u=e,f(o,u)},getClearAlpha:function(){return u},setClearAlpha:function(t){u=t,f(o,u)},render:function(n,s){let a=!1,g=!0===s.isScene?s.background:null;if(g&&g.isTexture){g=(s.backgroundBlurriness>0?i:e).get(g)}const v=t.xr,x=v.getSession&&v.getSession();x&&"additive"===x.environmentBlendMode&&(g=null),null===g?f(o,u):g&&g.isColor&&(f(g,1),a=!0),(t.autoClear||a)&&t.clear(t.autoClearColor,t.autoClearDepth,t.autoClearStencil),g&&(g.isCubeTexture||g.mapping===l)?(void 0===h&&(h=new Ji(new $i(1,1,1),new nn({name:"BackgroundCubeMaterial",uniforms:Qi(bn.backgroundCube.uniforms),vertexShader:bn.backgroundCube.vertexShader,fragmentShader:bn.backgroundCube.fragmentShader,side:1,depthTest:!1,depthWrite:!1,fog:!1})),h.geometry.deleteAttribute("normal"),h.geometry.deleteAttribute("uv"),h.onBeforeRender=function(t,e,i){this.matrixWorld.copyPosition(i.matrixWorld)},Object.defineProperty(h.material,"envMap",{get:function(){return this.uniforms.envMap.value}}),r.update(h)),h.material.uniforms.envMap.value=g,h.material.uniforms.flipEnvMap.value=g.isCubeTexture&&!1===g.isRenderTargetTexture?-1:1,h.material.uniforms.backgroundBlurriness.value=s.backgroundBlurriness,d===g&&p===g.version&&m===t.toneMapping||(h.material.needsUpdate=!0,d=g,p=g.version,m=t.toneMapping),h.layers.enableAll(),n.unshift(h,h.geometry,h.material,0,0,null)):g&&g.isTexture&&(void 0===c&&(c=new Ji(new _n(2,2),new nn({name:"BackgroundMaterial",uniforms:Qi(bn.background.uniforms),vertexShader:bn.background.vertexShader,fragmentShader:bn.background.fragmentShader,side:0,depthTest:!1,depthWrite:!1,fog:!1})),c.geometry.deleteAttribute("normal"),Object.defineProperty(c.material,"map",{get:function(){return this.uniforms.t2D.value}}),r.update(c)),c.material.uniforms.t2D.value=g,!0===g.matrixAutoUpdate&&g.updateMatrix(),c.material.uniforms.uvTransform.value.copy(g.matrix),d===g&&p===g.version&&m===t.toneMapping||(c.material.needsUpdate=!0,d=g,p=g.version,m=t.toneMapping),c.layers.enableAll(),n.unshift(c,c.geometry,c.material,0,0,null))}}}function wn(t,e,i,n){const r=t.getParameter(34921),s=n.isWebGL2?null:e.get("OES_vertex_array_object"),a=n.isWebGL2||null!==s,o={},l=p(null);let c=l,h=!1;function u(e){return n.isWebGL2?t.bindVertexArray(e):s.bindVertexArrayOES(e)}function d(e){return n.isWebGL2?t.deleteVertexArray(e):s.deleteVertexArrayOES(e)}function p(t){const e=[],i=[],n=[];for(let t=0;t<r;t++)e[t]=0,i[t]=0,n[t]=0;return{geometry:null,program:null,wireframe:!1,newAttributes:e,enabledAttributes:i,attributeDivisors:n,object:t,attributes:{},index:null}}function m(){const t=c.newAttributes;for(let e=0,i=t.length;e<i;e++)t[e]=0}function f(t){g(t,0)}function g(i,r){const s=c.newAttributes,a=c.enabledAttributes,o=c.attributeDivisors;if(s[i]=1,0===a[i]&&(t.enableVertexAttribArray(i),a[i]=1),o[i]!==r){(n.isWebGL2?t:e.get("ANGLE_instanced_arrays"))[n.isWebGL2?"vertexAttribDivisor":"vertexAttribDivisorANGLE"](i,r),o[i]=r}}function v(){const e=c.newAttributes,i=c.enabledAttributes;for(let n=0,r=i.length;n<r;n++)i[n]!==e[n]&&(t.disableVertexAttribArray(n),i[n]=0)}function x(e,i,r,s,a,o){!0!==n.isWebGL2||5124!==r&&5125!==r?t.vertexAttribPointer(e,i,r,s,a,o):t.vertexAttribIPointer(e,i,r,a,o)}function _(){y(),h=!0,c!==l&&(c=l,u(c.object))}function y(){l.geometry=null,l.program=null,l.wireframe=!1}return{setup:function(r,l,d,_,y){let M=!1;if(a){const e=function(e,i,r){const a=!0===r.wireframe;let l=o[e.id];void 0===l&&(l={},o[e.id]=l);let c=l[i.id];void 0===c&&(c={},l[i.id]=c);let h=c[a];void 0===h&&(h=p(n.isWebGL2?t.createVertexArray():s.createVertexArrayOES()),c[a]=h);return h}(_,d,l);c!==e&&(c=e,u(c.object)),M=function(t,e,i,n){const r=c.attributes,s=e.attributes;let a=0;const o=i.getAttributes();for(const e in o){if(o[e].location>=0){const i=r[e];let n=s[e];if(void 0===n&&("instanceMatrix"===e&&t.instanceMatrix&&(n=t.instanceMatrix),"instanceColor"===e&&t.instanceColor&&(n=t.instanceColor)),void 0===i)return!0;if(i.attribute!==n)return!0;if(n&&i.data!==n.data)return!0;a++}}return c.attributesNum!==a||c.index!==n}(r,_,d,y),M&&function(t,e,i,n){const r={},s=e.attributes;let a=0;const o=i.getAttributes();for(const e in o){if(o[e].location>=0){let i=s[e];void 0===i&&("instanceMatrix"===e&&t.instanceMatrix&&(i=t.instanceMatrix),"instanceColor"===e&&t.instanceColor&&(i=t.instanceColor));const n={};n.attribute=i,i&&i.data&&(n.data=i.data),r[e]=n,a++}}c.attributes=r,c.attributesNum=a,c.index=n}(r,_,d,y)}else{const t=!0===l.wireframe;c.geometry===_.id&&c.program===d.id&&c.wireframe===t||(c.geometry=_.id,c.program=d.id,c.wireframe=t,M=!0)}null!==y&&i.update(y,34963),(M||h)&&(h=!1,function(r,s,a,o){if(!1===n.isWebGL2&&(r.isInstancedMesh||o.isInstancedBufferGeometry)&&null===e.get("ANGLE_instanced_arrays"))return;m();const l=o.attributes,c=a.getAttributes(),h=s.defaultAttributeValues;for(const e in c){const n=c[e];if(n.location>=0){let s=l[e];if(void 0===s&&("instanceMatrix"===e&&r.instanceMatrix&&(s=r.instanceMatrix),"instanceColor"===e&&r.instanceColor&&(s=r.instanceColor)),void 0!==s){const e=s.normalized,a=s.itemSize,l=i.get(s);if(void 0===l)continue;const c=l.buffer,h=l.type,u=l.bytesPerElement;if(s.isInterleavedBufferAttribute){const i=s.data,l=i.stride,d=s.offset;if(i.isInstancedInterleavedBuffer){for(let t=0;t<n.locationSize;t++)g(n.location+t,i.meshPerAttribute);!0!==r.isInstancedMesh&&void 0===o._maxInstanceCount&&(o._maxInstanceCount=i.meshPerAttribute*i.count)}else for(let t=0;t<n.locationSize;t++)f(n.location+t);t.bindBuffer(34962,c);for(let t=0;t<n.locationSize;t++)x(n.location+t,a/n.locationSize,h,e,l*u,(d+a/n.locationSize*t)*u)}else{if(s.isInstancedBufferAttribute){for(let t=0;t<n.locationSize;t++)g(n.location+t,s.meshPerAttribute);!0!==r.isInstancedMesh&&void 0===o._maxInstanceCount&&(o._maxInstanceCount=s.meshPerAttribute*s.count)}else for(let t=0;t<n.locationSize;t++)f(n.location+t);t.bindBuffer(34962,c);for(let t=0;t<n.locationSize;t++)x(n.location+t,a/n.locationSize,h,e,a*u,a/n.locationSize*t*u)}}else if(void 0!==h){const i=h[e];if(void 0!==i)switch(i.length){case 2:t.vertexAttrib2fv(n.location,i);break;case 3:t.vertexAttrib3fv(n.location,i);break;case 4:t.vertexAttrib4fv(n.location,i);break;default:t.vertexAttrib1fv(n.location,i)}}}}v()}(r,l,d,_),null!==y&&t.bindBuffer(34963,i.get(y).buffer))},reset:_,resetDefaultState:y,dispose:function(){_();for(const t in o){const e=o[t];for(const t in e){const i=e[t];for(const t in i)d(i[t].object),delete i[t];delete e[t]}delete o[t]}},releaseStatesOfGeometry:function(t){if(void 0===o[t.id])return;const e=o[t.id];for(const t in e){const i=e[t];for(const t in i)d(i[t].object),delete i[t];delete e[t]}delete o[t.id]},releaseStatesOfProgram:function(t){for(const e in o){const i=o[e];if(void 0===i[t.id])continue;const n=i[t.id];for(const t in n)d(n[t].object),delete n[t];delete i[t.id]}},initAttributes:m,enableAttribute:f,disableUnusedAttributes:v}}function Tn(t,e,i,n){const r=n.isWebGL2;let s;this.setMode=function(t){s=t},this.render=function(e,n){t.drawArrays(s,e,n),i.update(n,s,1)},this.renderInstances=function(n,a,o){if(0===o)return;let l,c;if(r)l=t,c="drawArraysInstanced";else if(l=e.get("ANGLE_instanced_arrays"),c="drawArraysInstancedANGLE",null===l)return void console.error("THREE.WebGLBufferRenderer: using THREE.InstancedBufferGeometry but hardware does not support extension ANGLE_instanced_arrays.");l[c](s,n,a,o),i.update(a,s,o)}}function An(t,e,i){let n;function r(e){if("highp"===e){if(t.getShaderPrecisionFormat(35633,36338).precision>0&&t.getShaderPrecisionFormat(35632,36338).precision>0)return"highp";e="mediump"}return"mediump"===e&&t.getShaderPrecisionFormat(35633,36337).precision>0&&t.getShaderPrecisionFormat(35632,36337).precision>0?"mediump":"lowp"}const s="undefined"!=typeof WebGL2RenderingContext&&t instanceof WebGL2RenderingContext||"undefined"!=typeof WebGL2ComputeRenderingContext&&t instanceof WebGL2ComputeRenderingContext;let a=void 0!==i.precision?i.precision:"highp";const o=r(a);o!==a&&(console.warn("THREE.WebGLRenderer:",a,"not supported, using",o,"instead."),a=o);const l=s||e.has("WEBGL_draw_buffers"),c=!0===i.logarithmicDepthBuffer,h=t.getParameter(34930),u=t.getParameter(35660),d=t.getParameter(3379),p=t.getParameter(34076),m=t.getParameter(34921),f=t.getParameter(36347),g=t.getParameter(36348),v=t.getParameter(36349),x=u>0,_=s||e.has("OES_texture_float");return{isWebGL2:s,drawBuffers:l,getMaxAnisotropy:function(){if(void 0!==n)return n;if(!0===e.has("EXT_texture_filter_anisotropic")){const i=e.get("EXT_texture_filter_anisotropic");n=t.getParameter(i.MAX_TEXTURE_MAX_ANISOTROPY_EXT)}else n=0;return n},getMaxPrecision:r,precision:a,logarithmicDepthBuffer:c,maxTextures:h,maxVertexTextures:u,maxTextureSize:d,maxCubemapSize:p,maxAttributes:m,maxVertexUniforms:f,maxVaryings:g,maxFragmentUniforms:v,vertexTextures:x,floatFragmentTextures:_,floatVertexTextures:x&&_,maxSamples:s?t.getParameter(36183):0}}function En(t){const e=this;let i=null,n=0,r=!1,s=!1;const a=new pn,o=new Rt,l={value:null,needsUpdate:!1};function c(){l.value!==i&&(l.value=i,l.needsUpdate=n>0),e.numPlanes=n,e.numIntersection=0}function h(t,i,n,r){const s=null!==t?t.length:0;let c=null;if(0!==s){if(c=l.value,!0!==r||null===c){const e=n+4*s,r=i.matrixWorldInverse;o.getNormalMatrix(r),(null===c||c.length<e)&&(c=new Float32Array(e));for(let e=0,i=n;e!==s;++e,i+=4)a.copy(t[e]).applyMatrix4(r,o),a.normal.toArray(c,i),c[i+3]=a.constant}l.value=c,l.needsUpdate=!0}return e.numPlanes=s,e.numIntersection=0,c}this.uniform=l,this.numPlanes=0,this.numIntersection=0,this.init=function(t,e,s){const a=0!==t.length||e||0!==n||r;return r=e,i=h(t,s,0),n=t.length,a},this.beginShadows=function(){s=!0,h(null)},this.endShadows=function(){s=!1,c()},this.setState=function(e,a,o){const u=e.clippingPlanes,d=e.clipIntersection,p=e.clipShadows,m=t.get(e);if(!r||null===u||0===u.length||s&&!p)s?h(null):c();else{const t=s?0:n,e=4*t;let r=m.clippingState||null;l.value=r,r=h(u,a,e,o);for(let t=0;t!==e;++t)r[t]=i[t];m.clippingState=r,this.numIntersection=d?this.numPlanes:0,this.numPlanes+=t}}}function Cn(t){let e=new WeakMap;function i(t,e){return e===a?t.mapping=r:e===o&&(t.mapping=s),t}function n(t){const i=t.target;i.removeEventListener("dispose",n);const r=e.get(i);void 0!==r&&(e.delete(i),r.dispose())}return{get:function(r){if(r&&r.isTexture&&!1===r.isRenderTargetTexture){const s=r.mapping;if(s===a||s===o){if(e.has(r)){return i(e.get(r).texture,r.mapping)}{const s=r.image;if(s&&s.height>0){const a=new cn(s.height/2);return a.fromEquirectangularTexture(t,r),e.set(r,a),r.addEventListener("dispose",n),i(a.texture,r.mapping)}return null}}}return r},dispose:function(){e=new WeakMap}}}bn.physical={uniforms:tn([bn.standard.uniforms,{clearcoat:{value:0},clearcoatMap:{value:null},clearcoatRoughness:{value:0},clearcoatRoughnessMap:{value:null},clearcoatNormalScale:{value:new Lt(1,1)},clearcoatNormalMap:{value:null},iridescence:{value:0},iridescenceMap:{value:null},iridescenceIOR:{value:1.3},iridescenceThicknessMinimum:{value:100},iridescenceThicknessMaximum:{value:400},iridescenceThicknessMap:{value:null},sheen:{value:0},sheenColor:{value:new jt(0)},sheenColorMap:{value:null},sheenRoughness:{value:1},sheenRoughnessMap:{value:null},transmission:{value:0},transmissionMap:{value:null},transmissionSamplerSize:{value:new Lt},transmissionSamplerMap:{value:null},thickness:{value:0},thicknessMap:{value:null},attenuationDistance:{value:0},attenuationColor:{value:new jt(0)},specularIntensity:{value:1},specularIntensityMap:{value:null},specularColor:{value:new jt(1,1,1)},specularColorMap:{value:null}}]),vertexShader:yn.meshphysical_vert,fragmentShader:yn.meshphysical_frag};class Ln extends rn{constructor(t=-1,e=1,i=1,n=-1,r=.1,s=2e3){super(),this.isOrthographicCamera=!0,this.type="OrthographicCamera",this.zoom=1,this.view=null,this.left=t,this.right=e,this.top=i,this.bottom=n,this.near=r,this.far=s,this.updateProjectionMatrix()}copy(t,e){return super.copy(t,e),this.left=t.left,this.right=t.right,this.top=t.top,this.bottom=t.bottom,this.near=t.near,this.far=t.far,this.zoom=t.zoom,this.view=null===t.view?null:Object.assign({},t.view),this}setViewOffset(t,e,i,n,r,s){null===this.view&&(this.view={enabled:!0,fullWidth:1,fullHeight:1,offsetX:0,offsetY:0,width:1,height:1}),this.view.enabled=!0,this.view.fullWidth=t,this.view.fullHeight=e,this.view.offsetX=i,this.view.offsetY=n,this.view.width=r,this.view.height=s,this.updateProjectionMatrix()}clearViewOffset(){null!==this.view&&(this.view.enabled=!1),this.updateProjectionMatrix()}updateProjectionMatrix(){const t=(this.right-this.left)/(2*this.zoom),e=(this.top-this.bottom)/(2*this.zoom),i=(this.right+this.left)/2,n=(this.top+this.bottom)/2;let r=i-t,s=i+t,a=n+e,o=n-e;if(null!==this.view&&this.view.enabled){const t=(this.right-this.left)/this.view.fullWidth/this.zoom,e=(this.top-this.bottom)/this.view.fullHeight/this.zoom;r+=t*this.view.offsetX,s=r+t*this.view.width,a-=e*this.view.offsetY,o=a-e*this.view.height}this.projectionMatrix.makeOrthographic(r,s,a,o,this.near,this.far),this.projectionMatrixInverse.copy(this.projectionMatrix).invert()}toJSON(t){const e=super.toJSON(t);return e.object.zoom=this.zoom,e.object.left=this.left,e.object.right=this.right,e.object.top=this.top,e.object.bottom=this.bottom,e.object.near=this.near,e.object.far=this.far,null!==this.view&&(e.object.view=Object.assign({},this.view)),e}}const Rn=[.125,.215,.35,.446,.526,.582],Pn=20,Dn=new Ln,In=new jt;let Nn=null;const On=(1+Math.sqrt(5))/2,zn=1/On,Un=[new ne(1,1,1),new ne(-1,1,1),new ne(1,1,-1),new ne(-1,1,-1),new ne(0,On,zn),new ne(0,On,-zn),new ne(zn,0,On),new ne(-zn,0,On),new ne(On,zn,0),new ne(-On,zn,0)];class Bn{constructor(t){this._renderer=t,this._pingPongRenderTarget=null,this._lodMax=0,this._cubeSize=0,this._lodPlanes=[],this._sizeLods=[],this._sigmas=[],this._blurMaterial=null,this._cubemapMaterial=null,this._equirectMaterial=null,this._compileMaterial(this._blurMaterial)}fromScene(t,e=0,i=.1,n=100){Nn=this._renderer.getRenderTarget(),this._setSize(256);const r=this._allocateTargets();return r.depthBuffer=!0,this._sceneToCubeUV(t,i,n,r),e>0&&this._blur(r,0,0,e),this._applyPMREM(r),this._cleanup(r),r}fromEquirectangular(t,e=null){return this._fromTexture(t,e)}fromCubemap(t,e=null){return this._fromTexture(t,e)}compileCubemapShader(){null===this._cubemapMaterial&&(this._cubemapMaterial=Vn(),this._compileMaterial(this._cubemapMaterial))}compileEquirectangularShader(){null===this._equirectMaterial&&(this._equirectMaterial=Gn(),this._compileMaterial(this._equirectMaterial))}dispose(){this._dispose(),null!==this._cubemapMaterial&&this._cubemapMaterial.dispose(),null!==this._equirectMaterial&&this._equirectMaterial.dispose()}_setSize(t){this._lodMax=Math.floor(Math.log2(t)),this._cubeSize=Math.pow(2,this._lodMax)}_dispose(){null!==this._blurMaterial&&this._blurMaterial.dispose(),null!==this._pingPongRenderTarget&&this._pingPongRenderTarget.dispose();for(let t=0;t<this._lodPlanes.length;t++)this._lodPlanes[t].dispose()}_cleanup(t){this._renderer.setRenderTarget(Nn),t.scissorTest=!1,kn(t,0,0,t.width,t.height)}_fromTexture(t,e){t.mapping===r||t.mapping===s?this._setSize(0===t.image.length?16:t.image[0].width||t.image[0].image.width):this._setSize(t.image.width/4),Nn=this._renderer.getRenderTarget();const i=e||this._allocateTargets();return this._textureToCubeUV(t,i),this._applyPMREM(i),this._cleanup(i),i}_allocateTargets(){const t=3*Math.max(this._cubeSize,112),e=4*this._cubeSize,i={magFilter:f,minFilter:f,generateMipmaps:!1,type:b,format:w,encoding:at,depthBuffer:!1},n=Fn(t,e,i);if(null===this._pingPongRenderTarget||this._pingPongRenderTarget.width!==t){null!==this._pingPongRenderTarget&&this._dispose(),this._pingPongRenderTarget=Fn(t,e,i);const{_lodMax:n}=this;({sizeLods:this._sizeLods,lodPlanes:this._lodPlanes,sigmas:this._sigmas}=function(t){const e=[],i=[],n=[];let r=t;const s=t-4+1+Rn.length;for(let a=0;a<s;a++){const s=Math.pow(2,r);i.push(s);let o=1/s;a>t-4?o=Rn[a-t+4-1]:0===a&&(o=0),n.push(o);const l=1/(s-2),c=-l,h=1+l,u=[c,c,h,c,h,h,c,c,h,h,c,h],d=6,p=6,m=3,f=2,g=1,v=new Float32Array(m*p*d),x=new Float32Array(f*p*d),_=new Float32Array(g*p*d);for(let t=0;t<d;t++){const e=t%3*2/3-1,i=t>2?0:-1,n=[e,i,0,e+2/3,i,0,e+2/3,i+1,0,e,i,0,e+2/3,i+1,0,e,i+1,0];v.set(n,m*p*t),x.set(u,f*p*t);const r=[t,t,t,t,t,t];_.set(r,g*p*t)}const y=new Di;y.setAttribute("position",new Mi(v,m)),y.setAttribute("uv",new Mi(x,f)),y.setAttribute("faceIndex",new Mi(_,g)),e.push(y),r>4&&r--}return{lodPlanes:e,sizeLods:i,sigmas:n}}(n)),this._blurMaterial=function(t,e,i){const n=new Float32Array(Pn),r=new ne(0,1,0);return new nn({name:"SphericalGaussianBlur",defines:{n:Pn,CUBEUV_TEXEL_WIDTH:1/e,CUBEUV_TEXEL_HEIGHT:1/i,CUBEUV_MAX_MIP:`${t}.0`},uniforms:{envMap:{value:null},samples:{value:1},weights:{value:n},latitudinal:{value:!1},dTheta:{value:0},mipInt:{value:0},poleAxis:{value:r}},vertexShader:Hn(),fragmentShader:"\n\n\t\t\tprecision mediump float;\n\t\t\tprecision mediump int;\n\n\t\t\tvarying vec3 vOutputDirection;\n\n\t\t\tuniform sampler2D envMap;\n\t\t\tuniform int samples;\n\t\t\tuniform float weights[ n ];\n\t\t\tuniform bool latitudinal;\n\t\t\tuniform float dTheta;\n\t\t\tuniform float mipInt;\n\t\t\tuniform vec3 poleAxis;\n\n\t\t\t#define ENVMAP_TYPE_CUBE_UV\n\t\t\t#include <cube_uv_reflection_fragment>\n\n\t\t\tvec3 getSample( float theta, vec3 axis ) {\n\n\t\t\t\tfloat cosTheta = cos( theta );\n\t\t\t\t// Rodrigues' axis-angle rotation\n\t\t\t\tvec3 sampleDirection = vOutputDirection * cosTheta\n\t\t\t\t\t+ cross( axis, vOutputDirection ) * sin( theta )\n\t\t\t\t\t+ axis * dot( axis, vOutputDirection ) * ( 1.0 - cosTheta );\n\n\t\t\t\treturn bilinearCubeUV( envMap, sampleDirection, mipInt );\n\n\t\t\t}\n\n\t\t\tvoid main() {\n\n\t\t\t\tvec3 axis = latitudinal ? poleAxis : cross( poleAxis, vOutputDirection );\n\n\t\t\t\tif ( all( equal( axis, vec3( 0.0 ) ) ) ) {\n\n\t\t\t\t\taxis = vec3( vOutputDirection.z, 0.0, - vOutputDirection.x );\n\n\t\t\t\t}\n\n\t\t\t\taxis = normalize( axis );\n\n\t\t\t\tgl_FragColor = vec4( 0.0, 0.0, 0.0, 1.0 );\n\t\t\t\tgl_FragColor.rgb += weights[ 0 ] * getSample( 0.0, axis );\n\n\t\t\t\tfor ( int i = 1; i < n; i++ ) {\n\n\t\t\t\t\tif ( i >= samples ) {\n\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\t}\n\n\t\t\t\t\tfloat theta = dTheta * float( i );\n\t\t\t\t\tgl_FragColor.rgb += weights[ i ] * getSample( -1.0 * theta, axis );\n\t\t\t\t\tgl_FragColor.rgb += weights[ i ] * getSample( theta, axis );\n\n\t\t\t\t}\n\n\t\t\t}\n\t\t",blending:0,depthTest:!1,depthWrite:!1})}(n,t,e)}return n}_compileMaterial(t){const e=new Ji(this._lodPlanes[0],t);this._renderer.compile(e,Dn)}_sceneToCubeUV(t,e,i,n){const r=new sn(90,1,e,i),s=[1,-1,1,1,1,1],a=[1,1,1,-1,-1,-1],o=this._renderer,l=o.autoClear,c=o.toneMapping;o.getClearColor(In),o.toneMapping=0,o.autoClear=!1;const h=new xi({name:"PMREM.Background",side:1,depthWrite:!1,depthTest:!1}),u=new Ji(new $i,h);let d=!1;const p=t.background;p?p.isColor&&(h.color.copy(p),t.background=null,d=!0):(h.color.copy(In),d=!0);for(let e=0;e<6;e++){const i=e%3;0===i?(r.up.set(0,s[e],0),r.lookAt(a[e],0,0)):1===i?(r.up.set(0,0,s[e]),r.lookAt(0,a[e],0)):(r.up.set(0,s[e],0),r.lookAt(0,0,a[e]));const l=this._cubeSize;kn(n,i*l,e>2?l:0,l,l),o.setRenderTarget(n),d&&o.render(u,r),o.render(t,r)}u.geometry.dispose(),u.material.dispose(),o.toneMapping=c,o.autoClear=l,t.background=p}_textureToCubeUV(t,e){const i=this._renderer,n=t.mapping===r||t.mapping===s;n?(null===this._cubemapMaterial&&(this._cubemapMaterial=Vn()),this._cubemapMaterial.uniforms.flipEnvMap.value=!1===t.isRenderTargetTexture?-1:1):null===this._equirectMaterial&&(this._equirectMaterial=Gn());const a=n?this._cubemapMaterial:this._equirectMaterial,o=new Ji(this._lodPlanes[0],a);a.uniforms.envMap.value=t;const l=this._cubeSize;kn(e,0,0,3*l,2*l),i.setRenderTarget(e),i.render(o,Dn)}_applyPMREM(t){const e=this._renderer,i=e.autoClear;e.autoClear=!1;for(let e=1;e<this._lodPlanes.length;e++){const i=Math.sqrt(this._sigmas[e]*this._sigmas[e]-this._sigmas[e-1]*this._sigmas[e-1]),n=Un[(e-1)%Un.length];this._blur(t,e-1,e,i,n)}e.autoClear=i}_blur(t,e,i,n,r){const s=this._pingPongRenderTarget;this._halfBlur(t,s,e,i,n,"latitudinal",r),this._halfBlur(s,t,i,i,n,"longitudinal",r)}_halfBlur(t,e,i,n,r,s,a){const o=this._renderer,l=this._blurMaterial;"latitudinal"!==s&&"longitudinal"!==s&&console.error("blur direction must be either latitudinal or longitudinal!");const c=new Ji(this._lodPlanes[n],l),h=l.uniforms,u=this._sizeLods[i]-1,d=isFinite(r)?Math.PI/(2*u):2*Math.PI/39,p=r/d,m=isFinite(r)?1+Math.floor(3*p):Pn;m>Pn&&console.warn(`sigmaRadians, ${r}, is too large and will clip, as it requested ${m} samples when the maximum is set to 20`);const f=[];let g=0;for(let t=0;t<Pn;++t){const e=t/p,i=Math.exp(-e*e/2);f.push(i),0===t?g+=i:t<m&&(g+=2*i)}for(let t=0;t<f.length;t++)f[t]=f[t]/g;h.envMap.value=t.texture,h.samples.value=m,h.weights.value=f,h.latitudinal.value="latitudinal"===s,a&&(h.poleAxis.value=a);const{_lodMax:v}=this;h.dTheta.value=d,h.mipInt.value=v-i;const x=this._sizeLods[n];kn(e,3*x*(n>v-4?n-v+4:0),4*(this._cubeSize-x),3*x,2*x),o.setRenderTarget(e),o.render(c,Dn)}}function Fn(t,e,i){const n=new Qt(t,e,i);return n.texture.mapping=l,n.texture.name="PMREM.cubeUv",n.scissorTest=!0,n}function kn(t,e,i,n,r){t.viewport.set(e,i,n,r),t.scissor.set(e,i,n,r)}function Gn(){return new nn({name:"EquirectangularToCubeUV",uniforms:{envMap:{value:null}},vertexShader:Hn(),fragmentShader:"\n\n\t\t\tprecision mediump float;\n\t\t\tprecision mediump int;\n\n\t\t\tvarying vec3 vOutputDirection;\n\n\t\t\tuniform sampler2D envMap;\n\n\t\t\t#include <common>\n\n\t\t\tvoid main() {\n\n\t\t\t\tvec3 outputDirection = normalize( vOutputDirection );\n\t\t\t\tvec2 uv = equirectUv( outputDirection );\n\n\t\t\t\tgl_FragColor = vec4( texture2D ( envMap, uv ).rgb, 1.0 );\n\n\t\t\t}\n\t\t",blending:0,depthTest:!1,depthWrite:!1})}function Vn(){return new nn({name:"CubemapToCubeUV",uniforms:{envMap:{value:null},flipEnvMap:{value:-1}},vertexShader:Hn(),fragmentShader:"\n\n\t\t\tprecision mediump float;\n\t\t\tprecision mediump int;\n\n\t\t\tuniform float flipEnvMap;\n\n\t\t\tvarying vec3 vOutputDirection;\n\n\t\t\tuniform samplerCube envMap;\n\n\t\t\tvoid main() {\n\n\t\t\t\tgl_FragColor = textureCube( envMap, vec3( flipEnvMap * vOutputDirection.x, vOutputDirection.yz ) );\n\n\t\t\t}\n\t\t",blending:0,depthTest:!1,depthWrite:!1})}function Hn(){return"\n\n\t\tprecision mediump float;\n\t\tprecision mediump int;\n\n\t\tattribute float faceIndex;\n\n\t\tvarying vec3 vOutputDirection;\n\n\t\t// RH coordinate system; PMREM face-indexing convention\n\t\tvec3 getDirection( vec2 uv, float face ) {\n\n\t\t\tuv = 2.0 * uv - 1.0;\n\n\t\t\tvec3 direction = vec3( uv, 1.0 );\n\n\t\t\tif ( face == 0.0 ) {\n\n\t\t\t\tdirection = direction.zyx; // ( 1, v, u ) pos x\n\n\t\t\t} else if ( face == 1.0 ) {\n\n\t\t\t\tdirection = direction.xzy;\n\t\t\t\tdirection.xz *= -1.0; // ( -u, 1, -v ) pos y\n\n\t\t\t} else if ( face == 2.0 ) {\n\n\t\t\t\tdirection.x *= -1.0; // ( -u, v, 1 ) pos z\n\n\t\t\t} else if ( face == 3.0 ) {\n\n\t\t\t\tdirection = direction.zyx;\n\t\t\t\tdirection.xz *= -1.0; // ( -1, v, -u ) neg x\n\n\t\t\t} else if ( face == 4.0 ) {\n\n\t\t\t\tdirection = direction.xzy;\n\t\t\t\tdirection.xy *= -1.0; // ( -u, -1, v ) neg y\n\n\t\t\t} else if ( face == 5.0 ) {\n\n\t\t\t\tdirection.z *= -1.0; // ( u, v, -1 ) neg z\n\n\t\t\t}\n\n\t\t\treturn direction;\n\n\t\t}\n\n\t\tvoid main() {\n\n\t\t\tvOutputDirection = getDirection( uv, faceIndex );\n\t\t\tgl_Position = vec4( position, 1.0 );\n\n\t\t}\n\t"}function Wn(t){let e=new WeakMap,i=null;function n(t){const i=t.target;i.removeEventListener("dispose",n);const r=e.get(i);void 0!==r&&(e.delete(i),r.dispose())}return{get:function(l){if(l&&l.isTexture){const c=l.mapping,h=c===a||c===o,u=c===r||c===s;if(h||u){if(l.isRenderTargetTexture&&!0===l.needsPMREMUpdate){l.needsPMREMUpdate=!1;let n=e.get(l);return null===i&&(i=new Bn(t)),n=h?i.fromEquirectangular(l,n):i.fromCubemap(l,n),e.set(l,n),n.texture}if(e.has(l))return e.get(l).texture;{const r=l.image;if(h&&r&&r.height>0||u&&r&&function(t){let e=0;const i=6;for(let n=0;n<i;n++)void 0!==t[n]&&e++;return e===i}(r)){null===i&&(i=new Bn(t));const r=h?i.fromEquirectangular(l):i.fromCubemap(l);return e.set(l,r),l.addEventListener("dispose",n),r.texture}return null}}}return l},dispose:function(){e=new WeakMap,null!==i&&(i.dispose(),i=null)}}}function jn(t){const e={};function i(i){if(void 0!==e[i])return e[i];let n;switch(i){case"WEBGL_depth_texture":n=t.getExtension("WEBGL_depth_texture")||t.getExtension("MOZ_WEBGL_depth_texture")||t.getExtension("WEBKIT_WEBGL_depth_texture");break;case"EXT_texture_filter_anisotropic":n=t.getExtension("EXT_texture_filter_anisotropic")||t.getExtension("MOZ_EXT_texture_filter_anisotropic")||t.getExtension("WEBKIT_EXT_texture_filter_anisotropic");break;case"WEBGL_compressed_texture_s3tc":n=t.getExtension("WEBGL_compressed_texture_s3tc")||t.getExtension("MOZ_WEBGL_compressed_texture_s3tc")||t.getExtension("WEBKIT_WEBGL_compressed_texture_s3tc");break;case"WEBGL_compressed_texture_pvrtc":n=t.getExtension("WEBGL_compressed_texture_pvrtc")||t.getExtension("WEBKIT_WEBGL_compressed_texture_pvrtc");break;default:n=t.getExtension(i)}return e[i]=n,n}return{has:function(t){return null!==i(t)},init:function(t){t.isWebGL2?i("EXT_color_buffer_float"):(i("WEBGL_depth_texture"),i("OES_texture_float"),i("OES_texture_half_float"),i("OES_texture_half_float_linear"),i("OES_standard_derivatives"),i("OES_element_index_uint"),i("OES_vertex_array_object"),i("ANGLE_instanced_arrays")),i("OES_texture_float_linear"),i("EXT_color_buffer_half_float"),i("WEBGL_multisampled_render_to_texture")},get:function(t){const e=i(t);return null===e&&console.warn("THREE.WebGLRenderer: "+t+" extension not supported."),e}}}function qn(t,e,i,n){const r={},s=new WeakMap;function a(t){const o=t.target;null!==o.index&&e.remove(o.index);for(const t in o.attributes)e.remove(o.attributes[t]);o.removeEventListener("dispose",a),delete r[o.id];const l=s.get(o);l&&(e.remove(l),s.delete(o)),n.releaseStatesOfGeometry(o),!0===o.isInstancedBufferGeometry&&delete o._maxInstanceCount,i.memory.geometries--}function o(t){const i=[],n=t.index,r=t.attributes.position;let a=0;if(null!==n){const t=n.array;a=n.version;for(let e=0,n=t.length;e<n;e+=3){const n=t[e+0],r=t[e+1],s=t[e+2];i.push(n,r,r,s,s,n)}}else{const t=r.array;a=r.version;for(let e=0,n=t.length/3-1;e<n;e+=3){const t=e+0,n=e+1,r=e+2;i.push(t,n,n,r,r,t)}}const o=new(Pt(i)?Si:bi)(i,1);o.version=a;const l=s.get(t);l&&e.remove(l),s.set(t,o)}return{get:function(t,e){return!0===r[e.id]||(e.addEventListener("dispose",a),r[e.id]=!0,i.memory.geometries++),e},update:function(t){const i=t.attributes;for(const t in i)e.update(i[t],34962);const n=t.morphAttributes;for(const t in n){const i=n[t];for(let t=0,n=i.length;t<n;t++)e.update(i[t],34962)}},getWireframeAttribute:function(t){const e=s.get(t);if(e){const i=t.index;null!==i&&e.version<i.version&&o(t)}else o(t);return s.get(t)}}}function Xn(t,e,i,n){const r=n.isWebGL2;let s,a,o;this.setMode=function(t){s=t},this.setIndex=function(t){a=t.type,o=t.bytesPerElement},this.render=function(e,n){t.drawElements(s,n,a,e*o),i.update(n,s,1)},this.renderInstances=function(n,l,c){if(0===c)return;let h,u;if(r)h=t,u="drawElementsInstanced";else if(h=e.get("ANGLE_instanced_arrays"),u="drawElementsInstancedANGLE",null===h)return void console.error("THREE.WebGLIndexedBufferRenderer: using THREE.InstancedBufferGeometry but hardware does not support extension ANGLE_instanced_arrays.");h[u](s,l,a,n*o,c),i.update(l,s,c)}}function Yn(t){const e={frame:0,calls:0,triangles:0,points:0,lines:0};return{memory:{geometries:0,textures:0},render:e,programs:null,autoReset:!0,reset:function(){e.frame++,e.calls=0,e.triangles=0,e.points=0,e.lines=0},update:function(t,i,n){switch(e.calls++,i){case 4:e.triangles+=n*(t/3);break;case 1:e.lines+=n*(t/2);break;case 3:e.lines+=n*(t-1);break;case 2:e.lines+=n*t;break;case 0:e.points+=n*t;break;default:console.error("THREE.WebGLInfo: Unknown draw mode:",i)}}}}function Zn(t,e){return t[0]-e[0]}function Jn(t,e){return Math.abs(e[1])-Math.abs(t[1])}function Kn(t,e,i){const n={},r=new Float32Array(8),s=new WeakMap,a=new $t,o=[];for(let t=0;t<8;t++)o[t]=[t,0];return{update:function(l,c,h,u){const d=l.morphTargetInfluences;if(!0===e.isWebGL2){const p=c.morphAttributes.position||c.morphAttributes.normal||c.morphAttributes.color,m=void 0!==p?p.length:0;let f=s.get(c);if(void 0===f||f.count!==m){void 0!==f&&f.texture.dispose();const x=void 0!==c.morphAttributes.position,_=void 0!==c.morphAttributes.normal,y=void 0!==c.morphAttributes.color,b=c.morphAttributes.position||[],S=c.morphAttributes.normal||[],w=c.morphAttributes.color||[];let T=0;!0===x&&(T=1),!0===_&&(T=2),!0===y&&(T=3);let A=c.attributes.position.count*T,E=1;A>e.maxTextureSize&&(E=Math.ceil(A/e.maxTextureSize),A=e.maxTextureSize);const C=new Float32Array(A*E*4*m),L=new te(C,A,E,m);L.type=M,L.needsUpdate=!0;const R=4*T;for(let D=0;D<m;D++){const I=b[D],N=S[D],O=w[D],z=A*E*4*D;for(let U=0;U<I.count;U++){const B=U*R;!0===x&&(a.fromBufferAttribute(I,U),C[z+B+0]=a.x,C[z+B+1]=a.y,C[z+B+2]=a.z,C[z+B+3]=0),!0===_&&(a.fromBufferAttribute(N,U),C[z+B+4]=a.x,C[z+B+5]=a.y,C[z+B+6]=a.z,C[z+B+7]=0),!0===y&&(a.fromBufferAttribute(O,U),C[z+B+8]=a.x,C[z+B+9]=a.y,C[z+B+10]=a.z,C[z+B+11]=4===O.itemSize?a.w:1)}}function P(){L.dispose(),s.delete(c),c.removeEventListener("dispose",P)}f={count:m,texture:L,size:new Lt(A,E)},s.set(c,f),c.addEventListener("dispose",P)}let g=0;for(let F=0;F<d.length;F++)g+=d[F];const v=c.morphTargetsRelative?1:1-g;u.getUniforms().setValue(t,"morphTargetBaseInfluence",v),u.getUniforms().setValue(t,"morphTargetInfluences",d),u.getUniforms().setValue(t,"morphTargetsTexture",f.texture,i),u.getUniforms().setValue(t,"morphTargetsTextureSize",f.size)}else{const k=void 0===d?0:d.length;let G=n[c.id];if(void 0===G||G.length!==k){G=[];for(let q=0;q<k;q++)G[q]=[q,0];n[c.id]=G}for(let X=0;X<k;X++){const Y=G[X];Y[0]=X,Y[1]=d[X]}G.sort(Jn);for(let Z=0;Z<8;Z++)Z<k&&G[Z][1]?(o[Z][0]=G[Z][0],o[Z][1]=G[Z][1]):(o[Z][0]=Number.MAX_SAFE_INTEGER,o[Z][1]=0);o.sort(Zn);const V=c.morphAttributes.position,H=c.morphAttributes.normal;let W=0;for(let J=0;J<8;J++){const K=o[J],$=K[0],Q=K[1];$!==Number.MAX_SAFE_INTEGER&&Q?(V&&c.getAttribute("morphTarget"+J)!==V[$]&&c.setAttribute("morphTarget"+J,V[$]),H&&c.getAttribute("morphNormal"+J)!==H[$]&&c.setAttribute("morphNormal"+J,H[$]),r[J]=Q,W+=Q):(V&&!0===c.hasAttribute("morphTarget"+J)&&c.deleteAttribute("morphTarget"+J),H&&!0===c.hasAttribute("morphNormal"+J)&&c.deleteAttribute("morphNormal"+J),r[J]=0)}const j=c.morphTargetsRelative?1:1-W;u.getUniforms().setValue(t,"morphTargetBaseInfluence",j),u.getUniforms().setValue(t,"morphTargetInfluences",r)}}}}function $n(t,e,i,n){let r=new WeakMap;function s(t){const e=t.target;e.removeEventListener("dispose",s),i.remove(e.instanceMatrix),null!==e.instanceColor&&i.remove(e.instanceColor)}return{update:function(t){const a=n.render.frame,o=t.geometry,l=e.get(t,o);return r.get(l)!==a&&(e.update(l),r.set(l,a)),t.isInstancedMesh&&(!1===t.hasEventListener("dispose",s)&&t.addEventListener("dispose",s),i.update(t.instanceMatrix,34962),null!==t.instanceColor&&i.update(t.instanceColor,34962)),l},dispose:function(){r=new WeakMap}}}const Qn=new Kt,tr=new te,er=new ee,ir=new ln,nr=[],rr=[],sr=new Float32Array(16),ar=new Float32Array(9),or=new Float32Array(4);function lr(t,e,i){const n=t[0];if(n<=0||n>0)return t;const r=e*i;let s=nr[r];if(void 0===s&&(s=new Float32Array(r),nr[r]=s),0!==e){n.toArray(s,0);for(let n=1,r=0;n!==e;++n)r+=i,t[n].toArray(s,r)}return s}function cr(t,e){if(t.length!==e.length)return!1;for(let i=0,n=t.length;i<n;i++)if(t[i]!==e[i])return!1;return!0}function hr(t,e){for(let i=0,n=e.length;i<n;i++)t[i]=e[i]}function ur(t,e){let i=rr[e];void 0===i&&(i=new Int32Array(e),rr[e]=i);for(let n=0;n!==e;++n)i[n]=t.allocateTextureUnit();return i}function dr(t,e){const i=this.cache;i[0]!==e&&(t.uniform1f(this.addr,e),i[0]=e)}function pr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y||(t.uniform2f(this.addr,e.x,e.y),i[0]=e.x,i[1]=e.y);else{if(cr(i,e))return;t.uniform2fv(this.addr,e),hr(i,e)}}function mr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y&&i[2]===e.z||(t.uniform3f(this.addr,e.x,e.y,e.z),i[0]=e.x,i[1]=e.y,i[2]=e.z);else if(void 0!==e.r)i[0]===e.r&&i[1]===e.g&&i[2]===e.b||(t.uniform3f(this.addr,e.r,e.g,e.b),i[0]=e.r,i[1]=e.g,i[2]=e.b);else{if(cr(i,e))return;t.uniform3fv(this.addr,e),hr(i,e)}}function fr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y&&i[2]===e.z&&i[3]===e.w||(t.uniform4f(this.addr,e.x,e.y,e.z,e.w),i[0]=e.x,i[1]=e.y,i[2]=e.z,i[3]=e.w);else{if(cr(i,e))return;t.uniform4fv(this.addr,e),hr(i,e)}}function gr(t,e){const i=this.cache,n=e.elements;if(void 0===n){if(cr(i,e))return;t.uniformMatrix2fv(this.addr,!1,e),hr(i,e)}else{if(cr(i,n))return;or.set(n),t.uniformMatrix2fv(this.addr,!1,or),hr(i,n)}}function vr(t,e){const i=this.cache,n=e.elements;if(void 0===n){if(cr(i,e))return;t.uniformMatrix3fv(this.addr,!1,e),hr(i,e)}else{if(cr(i,n))return;ar.set(n),t.uniformMatrix3fv(this.addr,!1,ar),hr(i,n)}}function xr(t,e){const i=this.cache,n=e.elements;if(void 0===n){if(cr(i,e))return;t.uniformMatrix4fv(this.addr,!1,e),hr(i,e)}else{if(cr(i,n))return;sr.set(n),t.uniformMatrix4fv(this.addr,!1,sr),hr(i,n)}}function _r(t,e){const i=this.cache;i[0]!==e&&(t.uniform1i(this.addr,e),i[0]=e)}function yr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y||(t.uniform2i(this.addr,e.x,e.y),i[0]=e.x,i[1]=e.y);else{if(cr(i,e))return;t.uniform2iv(this.addr,e),hr(i,e)}}function Mr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y&&i[2]===e.z||(t.uniform3i(this.addr,e.x,e.y,e.z),i[0]=e.x,i[1]=e.y,i[2]=e.z);else{if(cr(i,e))return;t.uniform3iv(this.addr,e),hr(i,e)}}function br(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y&&i[2]===e.z&&i[3]===e.w||(t.uniform4i(this.addr,e.x,e.y,e.z,e.w),i[0]=e.x,i[1]=e.y,i[2]=e.z,i[3]=e.w);else{if(cr(i,e))return;t.uniform4iv(this.addr,e),hr(i,e)}}function Sr(t,e){const i=this.cache;i[0]!==e&&(t.uniform1ui(this.addr,e),i[0]=e)}function wr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y||(t.uniform2ui(this.addr,e.x,e.y),i[0]=e.x,i[1]=e.y);else{if(cr(i,e))return;t.uniform2uiv(this.addr,e),hr(i,e)}}function Tr(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y&&i[2]===e.z||(t.uniform3ui(this.addr,e.x,e.y,e.z),i[0]=e.x,i[1]=e.y,i[2]=e.z);else{if(cr(i,e))return;t.uniform3uiv(this.addr,e),hr(i,e)}}function Ar(t,e){const i=this.cache;if(void 0!==e.x)i[0]===e.x&&i[1]===e.y&&i[2]===e.z&&i[3]===e.w||(t.uniform4ui(this.addr,e.x,e.y,e.z,e.w),i[0]=e.x,i[1]=e.y,i[2]=e.z,i[3]=e.w);else{if(cr(i,e))return;t.uniform4uiv(this.addr,e),hr(i,e)}}function Er(t,e,i){const n=this.cache,r=i.allocateTextureUnit();n[0]!==r&&(t.uniform1i(this.addr,r),n[0]=r),i.setTexture2D(e||Qn,r)}function Cr(t,e,i){const n=this.cache,r=i.allocateTextureUnit();n[0]!==r&&(t.uniform1i(this.addr,r),n[0]=r),i.setTexture3D(e||er,r)}function Lr(t,e,i){const n=this.cache,r=i.allocateTextureUnit();n[0]!==r&&(t.uniform1i(this.addr,r),n[0]=r),i.setTextureCube(e||ir,r)}function Rr(t,e,i){const n=this.cache,r=i.allocateTextureUnit();n[0]!==r&&(t.uniform1i(this.addr,r),n[0]=r),i.setTexture2DArray(e||tr,r)}function Pr(t,e){t.uniform1fv(this.addr,e)}function Dr(t,e){const i=lr(e,this.size,2);t.uniform2fv(this.addr,i)}function Ir(t,e){const i=lr(e,this.size,3);t.uniform3fv(this.addr,i)}function Nr(t,e){const i=lr(e,this.size,4);t.uniform4fv(this.addr,i)}function Or(t,e){const i=lr(e,this.size,4);t.uniformMatrix2fv(this.addr,!1,i)}function zr(t,e){const i=lr(e,this.size,9);t.uniformMatrix3fv(this.addr,!1,i)}function Ur(t,e){const i=lr(e,this.size,16);t.uniformMatrix4fv(this.addr,!1,i)}function Br(t,e){t.uniform1iv(this.addr,e)}function Fr(t,e){t.uniform2iv(this.addr,e)}function kr(t,e){t.uniform3iv(this.addr,e)}function Gr(t,e){t.uniform4iv(this.addr,e)}function Vr(t,e){t.uniform1uiv(this.addr,e)}function Hr(t,e){t.uniform2uiv(this.addr,e)}function Wr(t,e){t.uniform3uiv(this.addr,e)}function jr(t,e){t.uniform4uiv(this.addr,e)}function qr(t,e,i){const n=this.cache,r=e.length,s=ur(i,r);cr(n,s)||(t.uniform1iv(this.addr,s),hr(n,s));for(let t=0;t!==r;++t)i.setTexture2D(e[t]||Qn,s[t])}function Xr(t,e,i){const n=this.cache,r=e.length,s=ur(i,r);cr(n,s)||(t.uniform1iv(this.addr,s),hr(n,s));for(let t=0;t!==r;++t)i.setTexture3D(e[t]||er,s[t])}function Yr(t,e,i){const n=this.cache,r=e.length,s=ur(i,r);cr(n,s)||(t.uniform1iv(this.addr,s),hr(n,s));for(let t=0;t!==r;++t)i.setTextureCube(e[t]||ir,s[t])}function Zr(t,e,i){const n=this.cache,r=e.length,s=ur(i,r);cr(n,s)||(t.uniform1iv(this.addr,s),hr(n,s));for(let t=0;t!==r;++t)i.setTexture2DArray(e[t]||tr,s[t])}class Jr{constructor(t,e,i){this.id=t,this.addr=i,this.cache=[],this.setValue=function(t){switch(t){case 5126:return dr;case 35664:return pr;case 35665:return mr;case 35666:return fr;case 35674:return gr;case 35675:return vr;case 35676:return xr;case 5124:case 35670:return _r;case 35667:case 35671:return yr;case 35668:case 35672:return Mr;case 35669:case 35673:return br;case 5125:return Sr;case 36294:return wr;case 36295:return Tr;case 36296:return Ar;case 35678:case 36198:case 36298:case 36306:case 35682:return Er;case 35679:case 36299:case 36307:return Cr;case 35680:case 36300:case 36308:case 36293:return Lr;case 36289:case 36303:case 36311:case 36292:return Rr}}(e.type)}}class Kr{constructor(t,e,i){this.id=t,this.addr=i,this.cache=[],this.size=e.size,this.setValue=function(t){switch(t){case 5126:return Pr;case 35664:return Dr;case 35665:return Ir;case 35666:return Nr;case 35674:return Or;case 35675:return zr;case 35676:return Ur;case 5124:case 35670:return Br;case 35667:case 35671:return Fr;case 35668:case 35672:return kr;case 35669:case 35673:return Gr;case 5125:return Vr;case 36294:return Hr;case 36295:return Wr;case 36296:return jr;case 35678:case 36198:case 36298:case 36306:case 35682:return qr;case 35679:case 36299:case 36307:return Xr;case 35680:case 36300:case 36308:case 36293:return Yr;case 36289:case 36303:case 36311:case 36292:return Zr}}(e.type)}}class $r{constructor(t){this.id=t,this.seq=[],this.map={}}setValue(t,e,i){const n=this.seq;for(let r=0,s=n.length;r!==s;++r){const s=n[r];s.setValue(t,e[s.id],i)}}}const Qr=/(\w+)(\])?(\[|\.)?/g;function ts(t,e){t.seq.push(e),t.map[e.id]=e}function es(t,e,i){const n=t.name,r=n.length;for(Qr.lastIndex=0;;){const s=Qr.exec(n),a=Qr.lastIndex;let o=s[1];const l="]"===s[2],c=s[3];if(l&&(o|=0),void 0===c||"["===c&&a+2===r){ts(i,void 0===c?new Jr(o,t,e):new Kr(o,t,e));break}{let t=i.map[o];void 0===t&&(t=new $r(o),ts(i,t)),i=t}}}class is{constructor(t,e){this.seq=[],this.map={};const i=t.getProgramParameter(e,35718);for(let n=0;n<i;++n){const i=t.getActiveUniform(e,n);es(i,t.getUniformLocation(e,i.name),this)}}setValue(t,e,i,n){const r=this.map[e];void 0!==r&&r.setValue(t,i,n)}setOptional(t,e,i){const n=e[i];void 0!==n&&this.setValue(t,i,n)}static upload(t,e,i,n){for(let r=0,s=e.length;r!==s;++r){const s=e[r],a=i[s.id];!1!==a.needsUpdate&&s.setValue(t,a.value,n)}}static seqWithValue(t,e){const i=[];for(let n=0,r=t.length;n!==r;++n){const r=t[n];r.id in e&&i.push(r)}return i}}function ns(t,e,i){const n=t.createShader(e);return t.shaderSource(n,i),t.compileShader(n),n}let rs=0;function ss(t,e,i){const n=t.getShaderParameter(e,35713),r=t.getShaderInfoLog(e).trim();if(n&&""===r)return"";const s=/ERROR: 0:(\d+)/.exec(r);if(s){const n=parseInt(s[1]);return i.toUpperCase()+"\n\n"+r+"\n\n"+function(t,e){const i=t.split("\n"),n=[],r=Math.max(e-6,0),s=Math.min(e+6,i.length);for(let t=r;t<s;t++){const r=t+1;n.push(`${r===e?">":" "} ${r}: ${i[t]}`)}return n.join("\n")}(t.getShaderSource(e),n)}return r}function as(t,e){const i=function(t){switch(t){case at:return["Linear","( value )"];case ot:return["sRGB","( value )"];default:return console.warn("THREE.WebGLProgram: Unsupported encoding:",t),["Linear","( value )"]}}(e);return"vec4 "+t+"( vec4 value ) { return LinearTo"+i[0]+i[1]+"; }"}function os(t,e){let i;switch(e){case 1:i="Linear";break;case 2:i="Reinhard";break;case 3:i="OptimizedCineon";break;case 4:i="ACESFilmic";break;case 5:i="Custom";break;default:console.warn("THREE.WebGLProgram: Unsupported toneMapping:",e),i="Linear"}return"vec3 "+t+"( vec3 color ) { return "+i+"ToneMapping( color ); }"}function ls(t){return""!==t}function cs(t,e){const i=e.numSpotLightShadows+e.numSpotLightMaps-e.numSpotLightShadowsWithMaps;return t.replace(/NUM_DIR_LIGHTS/g,e.numDirLights).replace(/NUM_SPOT_LIGHTS/g,e.numSpotLights).replace(/NUM_SPOT_LIGHT_MAPS/g,e.numSpotLightMaps).replace(/NUM_SPOT_LIGHT_COORDS/g,i).replace(/NUM_RECT_AREA_LIGHTS/g,e.numRectAreaLights).replace(/NUM_POINT_LIGHTS/g,e.numPointLights).replace(/NUM_HEMI_LIGHTS/g,e.numHemiLights).replace(/NUM_DIR_LIGHT_SHADOWS/g,e.numDirLightShadows).replace(/NUM_SPOT_LIGHT_SHADOWS_WITH_MAPS/g,e.numSpotLightShadowsWithMaps).replace(/NUM_SPOT_LIGHT_SHADOWS/g,e.numSpotLightShadows).replace(/NUM_POINT_LIGHT_SHADOWS/g,e.numPointLightShadows)}function hs(t,e){return t.replace(/NUM_CLIPPING_PLANES/g,e.numClippingPlanes).replace(/UNION_CLIPPING_PLANES/g,e.numClippingPlanes-e.numClipIntersection)}const us=/^[ \t]*#include +<([\w\d./]+)>/gm;function ds(t){return t.replace(us,ps)}function ps(t,e){const i=yn[e];if(void 0===i)throw new Error("Can not resolve #include <"+e+">");return ds(i)}const ms=/#pragma unroll_loop_start\s+for\s*\(\s*int\s+i\s*=\s*(\d+)\s*;\s*i\s*<\s*(\d+)\s*;\s*i\s*\+\+\s*\)\s*{([\s\S]+?)}\s+#pragma unroll_loop_end/g;function fs(t){return t.replace(ms,gs)}function gs(t,e,i,n){let r="";for(let t=parseInt(e);t<parseInt(i);t++)r+=n.replace(/\[\s*i\s*\]/g,"[ "+t+" ]").replace(/UNROLLED_LOOP_INDEX/g,t);return r}function vs(t){let e="precision "+t.precision+" float;\nprecision "+t.precision+" int;";return"highp"===t.precision?e+="\n#define HIGH_PRECISION":"mediump"===t.precision?e+="\n#define MEDIUM_PRECISION":"lowp"===t.precision&&(e+="\n#define LOW_PRECISION"),e}function xs(t,e,i,n){const a=t.getContext(),o=i.defines;let c=i.vertexShader,h=i.fragmentShader;const u=function(t){let e="SHADOWMAP_TYPE_BASIC";return 1===t.shadowMapType?e="SHADOWMAP_TYPE_PCF":2===t.shadowMapType?e="SHADOWMAP_TYPE_PCF_SOFT":3===t.shadowMapType&&(e="SHADOWMAP_TYPE_VSM"),e}(i),d=function(t){let e="ENVMAP_TYPE_CUBE";if(t.envMap)switch(t.envMapMode){case r:case s:e="ENVMAP_TYPE_CUBE";break;case l:e="ENVMAP_TYPE_CUBE_UV"}return e}(i),p=function(t){let e="ENVMAP_MODE_REFLECTION";t.envMap&&t.envMapMode===s&&(e="ENVMAP_MODE_REFRACTION");return e}(i),m=function(t){let e="ENVMAP_BLENDING_NONE";if(t.envMap)switch(t.combine){case 0:e="ENVMAP_BLENDING_MULTIPLY";break;case 1:e="ENVMAP_BLENDING_MIX";break;case 2:e="ENVMAP_BLENDING_ADD"}return e}(i),f=function(t){const e=t.envMapCubeUVHeight;if(null===e)return null;const i=Math.log2(e)-2,n=1/e;return{texelWidth:1/(3*Math.max(Math.pow(2,i),112)),texelHeight:n,maxMip:i}}(i),g=i.isWebGL2?"":function(t){return[t.extensionDerivatives||t.envMapCubeUVHeight||t.bumpMap||t.tangentSpaceNormalMap||t.clearcoatNormalMap||t.flatShading||"physical"===t.shaderID?"#extension GL_OES_standard_derivatives : enable":"",(t.extensionFragDepth||t.logarithmicDepthBuffer)&&t.rendererExtensionFragDepth?"#extension GL_EXT_frag_depth : enable":"",t.extensionDrawBuffers&&t.rendererExtensionDrawBuffers?"#extension GL_EXT_draw_buffers : require":"",(t.extensionShaderTextureLOD||t.envMap||t.transmission)&&t.rendererExtensionShaderTextureLod?"#extension GL_EXT_shader_texture_lod : enable":""].filter(ls).join("\n")}(i),v=function(t){const e=[];for(const i in t){const n=t[i];!1!==n&&e.push("#define "+i+" "+n)}return e.join("\n")}(o),x=a.createProgram();let _,y,M=i.glslVersion?"#version "+i.glslVersion+"\n":"";i.isRawShaderMaterial?(_=[v].filter(ls).join("\n"),_.length>0&&(_+="\n"),y=[g,v].filter(ls).join("\n"),y.length>0&&(y+="\n")):(_=[vs(i),"#define SHADER_NAME "+i.shaderName,v,i.instancing?"#define USE_INSTANCING":"",i.instancingColor?"#define USE_INSTANCING_COLOR":"",i.supportsVertexTextures?"#define VERTEX_TEXTURES":"",i.useFog&&i.fog?"#define USE_FOG":"",i.useFog&&i.fogExp2?"#define FOG_EXP2":"",i.map?"#define USE_MAP":"",i.envMap?"#define USE_ENVMAP":"",i.envMap?"#define "+p:"",i.lightMap?"#define USE_LIGHTMAP":"",i.aoMap?"#define USE_AOMAP":"",i.emissiveMap?"#define USE_EMISSIVEMAP":"",i.bumpMap?"#define USE_BUMPMAP":"",i.normalMap?"#define USE_NORMALMAP":"",i.normalMap&&i.objectSpaceNormalMap?"#define OBJECTSPACE_NORMALMAP":"",i.normalMap&&i.tangentSpaceNormalMap?"#define TANGENTSPACE_NORMALMAP":"",i.clearcoatMap?"#define USE_CLEARCOATMAP":"",i.clearcoatRoughnessMap?"#define USE_CLEARCOAT_ROUGHNESSMAP":"",i.clearcoatNormalMap?"#define USE_CLEARCOAT_NORMALMAP":"",i.iridescenceMap?"#define USE_IRIDESCENCEMAP":"",i.iridescenceThicknessMap?"#define USE_IRIDESCENCE_THICKNESSMAP":"",i.displacementMap&&i.supportsVertexTextures?"#define USE_DISPLACEMENTMAP":"",i.specularMap?"#define USE_SPECULARMAP":"",i.specularIntensityMap?"#define USE_SPECULARINTENSITYMAP":"",i.specularColorMap?"#define USE_SPECULARCOLORMAP":"",i.roughnessMap?"#define USE_ROUGHNESSMAP":"",i.metalnessMap?"#define USE_METALNESSMAP":"",i.alphaMap?"#define USE_ALPHAMAP":"",i.transmission?"#define USE_TRANSMISSION":"",i.transmissionMap?"#define USE_TRANSMISSIONMAP":"",i.thicknessMap?"#define USE_THICKNESSMAP":"",i.sheenColorMap?"#define USE_SHEENCOLORMAP":"",i.sheenRoughnessMap?"#define USE_SHEENROUGHNESSMAP":"",i.vertexTangents?"#define USE_TANGENT":"",i.vertexColors?"#define USE_COLOR":"",i.vertexAlphas?"#define USE_COLOR_ALPHA":"",i.vertexUvs?"#define USE_UV":"",i.uvsVertexOnly?"#define UVS_VERTEX_ONLY":"",i.flatShading?"#define FLAT_SHADED":"",i.skinning?"#define USE_SKINNING":"",i.morphTargets?"#define USE_MORPHTARGETS":"",i.morphNormals&&!1===i.flatShading?"#define USE_MORPHNORMALS":"",i.morphColors&&i.isWebGL2?"#define USE_MORPHCOLORS":"",i.morphTargetsCount>0&&i.isWebGL2?"#define MORPHTARGETS_TEXTURE":"",i.morphTargetsCount>0&&i.isWebGL2?"#define MORPHTARGETS_TEXTURE_STRIDE "+i.morphTextureStride:"",i.morphTargetsCount>0&&i.isWebGL2?"#define MORPHTARGETS_COUNT "+i.morphTargetsCount:"",i.doubleSided?"#define DOUBLE_SIDED":"",i.flipSided?"#define FLIP_SIDED":"",i.shadowMapEnabled?"#define USE_SHADOWMAP":"",i.shadowMapEnabled?"#define "+u:"",i.sizeAttenuation?"#define USE_SIZEATTENUATION":"",i.logarithmicDepthBuffer?"#define USE_LOGDEPTHBUF":"",i.logarithmicDepthBuffer&&i.rendererExtensionFragDepth?"#define USE_LOGDEPTHBUF_EXT":"","uniform mat4 modelMatrix;","uniform mat4 modelViewMatrix;","uniform mat4 projectionMatrix;","uniform mat4 viewMatrix;","uniform mat3 normalMatrix;","uniform vec3 cameraPosition;","uniform bool isOrthographic;","#ifdef USE_INSTANCING","\tattribute mat4 instanceMatrix;","#endif","#ifdef USE_INSTANCING_COLOR","\tattribute vec3 instanceColor;","#endif","attribute vec3 position;","attribute vec3 normal;","attribute vec2 uv;","#ifdef USE_TANGENT","\tattribute vec4 tangent;","#endif","#if defined( USE_COLOR_ALPHA )","\tattribute vec4 color;","#elif defined( USE_COLOR )","\tattribute vec3 color;","#endif","#if ( defined( USE_MORPHTARGETS ) && ! defined( MORPHTARGETS_TEXTURE ) )","\tattribute vec3 morphTarget0;","\tattribute vec3 morphTarget1;","\tattribute vec3 morphTarget2;","\tattribute vec3 morphTarget3;","\t#ifdef USE_MORPHNORMALS","\t\tattribute vec3 morphNormal0;","\t\tattribute vec3 morphNormal1;","\t\tattribute vec3 morphNormal2;","\t\tattribute vec3 morphNormal3;","\t#else","\t\tattribute vec3 morphTarget4;","\t\tattribute vec3 morphTarget5;","\t\tattribute vec3 morphTarget6;","\t\tattribute vec3 morphTarget7;","\t#endif","#endif","#ifdef USE_SKINNING","\tattribute vec4 skinIndex;","\tattribute vec4 skinWeight;","#endif","\n"].filter(ls).join("\n"),y=[g,vs(i),"#define SHADER_NAME "+i.shaderName,v,i.useFog&&i.fog?"#define USE_FOG":"",i.useFog&&i.fogExp2?"#define FOG_EXP2":"",i.map?"#define USE_MAP":"",i.matcap?"#define USE_MATCAP":"",i.envMap?"#define USE_ENVMAP":"",i.envMap?"#define "+d:"",i.envMap?"#define "+p:"",i.envMap?"#define "+m:"",f?"#define CUBEUV_TEXEL_WIDTH "+f.texelWidth:"",f?"#define CUBEUV_TEXEL_HEIGHT "+f.texelHeight:"",f?"#define CUBEUV_MAX_MIP "+f.maxMip+".0":"",i.lightMap?"#define USE_LIGHTMAP":"",i.aoMap?"#define USE_AOMAP":"",i.emissiveMap?"#define USE_EMISSIVEMAP":"",i.bumpMap?"#define USE_BUMPMAP":"",i.normalMap?"#define USE_NORMALMAP":"",i.normalMap&&i.objectSpaceNormalMap?"#define OBJECTSPACE_NORMALMAP":"",i.normalMap&&i.tangentSpaceNormalMap?"#define TANGENTSPACE_NORMALMAP":"",i.clearcoat?"#define USE_CLEARCOAT":"",i.clearcoatMap?"#define USE_CLEARCOATMAP":"",i.clearcoatRoughnessMap?"#define USE_CLEARCOAT_ROUGHNESSMAP":"",i.clearcoatNormalMap?"#define USE_CLEARCOAT_NORMALMAP":"",i.iridescence?"#define USE_IRIDESCENCE":"",i.iridescenceMap?"#define USE_IRIDESCENCEMAP":"",i.iridescenceThicknessMap?"#define USE_IRIDESCENCE_THICKNESSMAP":"",i.specularMap?"#define USE_SPECULARMAP":"",i.specularIntensityMap?"#define USE_SPECULARINTENSITYMAP":"",i.specularColorMap?"#define USE_SPECULARCOLORMAP":"",i.roughnessMap?"#define USE_ROUGHNESSMAP":"",i.metalnessMap?"#define USE_METALNESSMAP":"",i.alphaMap?"#define USE_ALPHAMAP":"",i.alphaTest?"#define USE_ALPHATEST":"",i.sheen?"#define USE_SHEEN":"",i.sheenColorMap?"#define USE_SHEENCOLORMAP":"",i.sheenRoughnessMap?"#define USE_SHEENROUGHNESSMAP":"",i.transmission?"#define USE_TRANSMISSION":"",i.transmissionMap?"#define USE_TRANSMISSIONMAP":"",i.thicknessMap?"#define USE_THICKNESSMAP":"",i.decodeVideoTexture?"#define DECODE_VIDEO_TEXTURE":"",i.vertexTangents?"#define USE_TANGENT":"",i.vertexColors||i.instancingColor?"#define USE_COLOR":"",i.vertexAlphas?"#define USE_COLOR_ALPHA":"",i.vertexUvs?"#define USE_UV":"",i.uvsVertexOnly?"#define UVS_VERTEX_ONLY":"",i.gradientMap?"#define USE_GRADIENTMAP":"",i.flatShading?"#define FLAT_SHADED":"",i.doubleSided?"#define DOUBLE_SIDED":"",i.flipSided?"#define FLIP_SIDED":"",i.shadowMapEnabled?"#define USE_SHADOWMAP":"",i.shadowMapEnabled?"#define "+u:"",i.premultipliedAlpha?"#define PREMULTIPLIED_ALPHA":"",i.physicallyCorrectLights?"#define PHYSICALLY_CORRECT_LIGHTS":"",i.logarithmicDepthBuffer?"#define USE_LOGDEPTHBUF":"",i.logarithmicDepthBuffer&&i.rendererExtensionFragDepth?"#define USE_LOGDEPTHBUF_EXT":"","uniform mat4 viewMatrix;","uniform vec3 cameraPosition;","uniform bool isOrthographic;",0!==i.toneMapping?"#define TONE_MAPPING":"",0!==i.toneMapping?yn.tonemapping_pars_fragment:"",0!==i.toneMapping?os("toneMapping",i.toneMapping):"",i.dithering?"#define DITHERING":"",i.opaque?"#define OPAQUE":"",yn.encodings_pars_fragment,as("linearToOutputTexel",i.outputEncoding),i.useDepthPacking?"#define DEPTH_PACKING "+i.depthPacking:"","\n"].filter(ls).join("\n")),c=ds(c),c=cs(c,i),c=hs(c,i),h=ds(h),h=cs(h,i),h=hs(h,i),c=fs(c),h=fs(h),i.isWebGL2&&!0!==i.isRawShaderMaterial&&(M="#version 300 es\n",_=["precision mediump sampler2DArray;","#define attribute in","#define varying out","#define texture2D texture"].join("\n")+"\n"+_,y=["#define varying in",i.glslVersion===dt?"":"layout(location = 0) out highp vec4 pc_fragColor;",i.glslVersion===dt?"":"#define gl_FragColor pc_fragColor","#define gl_FragDepthEXT gl_FragDepth","#define texture2D texture","#define textureCube texture","#define texture2DProj textureProj","#define texture2DLodEXT textureLod","#define texture2DProjLodEXT textureProjLod","#define textureCubeLodEXT textureLod","#define texture2DGradEXT textureGrad","#define texture2DProjGradEXT textureProjGrad","#define textureCubeGradEXT textureGrad"].join("\n")+"\n"+y);const b=M+y+h,S=ns(a,35633,M+_+c),w=ns(a,35632,b);if(a.attachShader(x,S),a.attachShader(x,w),void 0!==i.index0AttributeName?a.bindAttribLocation(x,0,i.index0AttributeName):!0===i.morphTargets&&a.bindAttribLocation(x,0,"position"),a.linkProgram(x),t.debug.checkShaderErrors){const t=a.getProgramInfoLog(x).trim(),e=a.getShaderInfoLog(S).trim(),i=a.getShaderInfoLog(w).trim();let n=!0,r=!0;if(!1===a.getProgramParameter(x,35714)){n=!1;const e=ss(a,S,"vertex"),i=ss(a,w,"fragment");console.error("THREE.WebGLProgram: Shader Error "+a.getError()+" - VALIDATE_STATUS "+a.getProgramParameter(x,35715)+"\n\nProgram Info Log: "+t+"\n"+e+"\n"+i)}else""!==t?console.warn("THREE.WebGLProgram: Program Info Log:",t):""!==e&&""!==i||(r=!1);r&&(this.diagnostics={runnable:n,programLog:t,vertexShader:{log:e,prefix:_},fragmentShader:{log:i,prefix:y}})}let T,A;return a.deleteShader(S),a.deleteShader(w),this.getUniforms=function(){return void 0===T&&(T=new is(a,x)),T},this.getAttributes=function(){return void 0===A&&(A=function(t,e){const i={},n=t.getProgramParameter(e,35721);for(let r=0;r<n;r++){const n=t.getActiveAttrib(e,r),s=n.name;let a=1;35674===n.type&&(a=2),35675===n.type&&(a=3),35676===n.type&&(a=4),i[s]={type:n.type,location:t.getAttribLocation(e,s),locationSize:a}}return i}(a,x)),A},this.destroy=function(){n.releaseStatesOfProgram(this),a.deleteProgram(x),this.program=void 0},this.name=i.shaderName,this.id=rs++,this.cacheKey=e,this.usedTimes=1,this.program=x,this.vertexShader=S,this.fragmentShader=w,this}let _s=0;class ys{constructor(){this.shaderCache=new Map,this.materialCache=new Map}update(t){const e=t.vertexShader,i=t.fragmentShader,n=this._getShaderStage(e),r=this._getShaderStage(i),s=this._getShaderCacheForMaterial(t);return!1===s.has(n)&&(s.add(n),n.usedTimes++),!1===s.has(r)&&(s.add(r),r.usedTimes++),this}remove(t){const e=this.materialCache.get(t);for(const t of e)t.usedTimes--,0===t.usedTimes&&this.shaderCache.delete(t.code);return this.materialCache.delete(t),this}getVertexShaderID(t){return this._getShaderStage(t.vertexShader).id}getFragmentShaderID(t){return this._getShaderStage(t.fragmentShader).id}dispose(){this.shaderCache.clear(),this.materialCache.clear()}_getShaderCacheForMaterial(t){const e=this.materialCache;let i=e.get(t);return void 0===i&&(i=new Set,e.set(t,i)),i}_getShaderStage(t){const e=this.shaderCache;let i=e.get(t);return void 0===i&&(i=new Ms(t),e.set(t,i)),i}}class Ms{constructor(t){this.id=_s++,this.code=t,this.usedTimes=0}}function bs(t,e,i,n,r,s,a){const o=new We,c=new ys,h=[],u=r.isWebGL2,d=r.logarithmicDepthBuffer,p=r.vertexTextures;let m=r.precision;const f={MeshDepthMaterial:"depth",MeshDistanceMaterial:"distanceRGBA",MeshNormalMaterial:"normal",MeshBasicMaterial:"basic",MeshLambertMaterial:"lambert",MeshPhongMaterial:"phong",MeshToonMaterial:"toon",MeshStandardMaterial:"physical",MeshPhysicalMaterial:"physical",MeshMatcapMaterial:"matcap",LineBasicMaterial:"basic",LineDashedMaterial:"dashed",PointsMaterial:"points",ShadowMaterial:"shadow",SpriteMaterial:"sprite"};return{getParameters:function(s,o,h,g,v){const x=g.fog,_=v.geometry,y=s.isMeshStandardMaterial?g.environment:null,M=(s.isMeshStandardMaterial?i:e).get(s.envMap||y),b=M&&M.mapping===l?M.image.height:null,S=f[s.type];null!==s.precision&&(m=r.getMaxPrecision(s.precision),m!==s.precision&&console.warn("THREE.WebGLProgram.getParameters:",s.precision,"not supported, using",m,"instead."));const w=_.morphAttributes.position||_.morphAttributes.normal||_.morphAttributes.color,T=void 0!==w?w.length:0;let A,E,C,L,R=0;if(void 0!==_.morphAttributes.position&&(R=1),void 0!==_.morphAttributes.normal&&(R=2),void 0!==_.morphAttributes.color&&(R=3),S){const t=bn[S];A=t.vertexShader,E=t.fragmentShader}else A=s.vertexShader,E=s.fragmentShader,c.update(s),C=c.getVertexShaderID(s),L=c.getFragmentShaderID(s);const P=t.getRenderTarget(),D=s.alphaTest>0,I=s.clearcoat>0,N=s.iridescence>0;return{isWebGL2:u,shaderID:S,shaderName:s.type,vertexShader:A,fragmentShader:E,defines:s.defines,customVertexShaderID:C,customFragmentShaderID:L,isRawShaderMaterial:!0===s.isRawShaderMaterial,glslVersion:s.glslVersion,precision:m,instancing:!0===v.isInstancedMesh,instancingColor:!0===v.isInstancedMesh&&null!==v.instanceColor,supportsVertexTextures:p,outputEncoding:null===P?t.outputEncoding:!0===P.isXRRenderTarget?P.texture.encoding:at,map:!!s.map,matcap:!!s.matcap,envMap:!!M,envMapMode:M&&M.mapping,envMapCubeUVHeight:b,lightMap:!!s.lightMap,aoMap:!!s.aoMap,emissiveMap:!!s.emissiveMap,bumpMap:!!s.bumpMap,normalMap:!!s.normalMap,objectSpaceNormalMap:1===s.normalMapType,tangentSpaceNormalMap:0===s.normalMapType,decodeVideoTexture:!!s.map&&!0===s.map.isVideoTexture&&s.map.encoding===ot,clearcoat:I,clearcoatMap:I&&!!s.clearcoatMap,clearcoatRoughnessMap:I&&!!s.clearcoatRoughnessMap,clearcoatNormalMap:I&&!!s.clearcoatNormalMap,iridescence:N,iridescenceMap:N&&!!s.iridescenceMap,iridescenceThicknessMap:N&&!!s.iridescenceThicknessMap,displacementMap:!!s.displacementMap,roughnessMap:!!s.roughnessMap,metalnessMap:!!s.metalnessMap,specularMap:!!s.specularMap,specularIntensityMap:!!s.specularIntensityMap,specularColorMap:!!s.specularColorMap,opaque:!1===s.transparent&&1===s.blending,alphaMap:!!s.alphaMap,alphaTest:D,gradientMap:!!s.gradientMap,sheen:s.sheen>0,sheenColorMap:!!s.sheenColorMap,sheenRoughnessMap:!!s.sheenRoughnessMap,transmission:s.transmission>0,transmissionMap:!!s.transmissionMap,thicknessMap:!!s.thicknessMap,combine:s.combine,vertexTangents:!!s.normalMap&&!!_.attributes.tangent,vertexColors:s.vertexColors,vertexAlphas:!0===s.vertexColors&&!!_.attributes.color&&4===_.attributes.color.itemSize,vertexUvs:!!(s.map||s.bumpMap||s.normalMap||s.specularMap||s.alphaMap||s.emissiveMap||s.roughnessMap||s.metalnessMap||s.clearcoatMap||s.clearcoatRoughnessMap||s.clearcoatNormalMap||s.iridescenceMap||s.iridescenceThicknessMap||s.displacementMap||s.transmissionMap||s.thicknessMap||s.specularIntensityMap||s.specularColorMap||s.sheenColorMap||s.sheenRoughnessMap),uvsVertexOnly:!(s.map||s.bumpMap||s.normalMap||s.specularMap||s.alphaMap||s.emissiveMap||s.roughnessMap||s.metalnessMap||s.clearcoatNormalMap||s.iridescenceMap||s.iridescenceThicknessMap||s.transmission>0||s.transmissionMap||s.thicknessMap||s.specularIntensityMap||s.specularColorMap||s.sheen>0||s.sheenColorMap||s.sheenRoughnessMap||!s.displacementMap),fog:!!x,useFog:!0===s.fog,fogExp2:x&&x.isFogExp2,flatShading:!!s.flatShading,sizeAttenuation:s.sizeAttenuation,logarithmicDepthBuffer:d,skinning:!0===v.isSkinnedMesh,morphTargets:void 0!==_.morphAttributes.position,morphNormals:void 0!==_.morphAttributes.normal,morphColors:void 0!==_.morphAttributes.color,morphTargetsCount:T,morphTextureStride:R,numDirLights:o.directional.length,numPointLights:o.point.length,numSpotLights:o.spot.length,numSpotLightMaps:o.spotLightMap.length,numRectAreaLights:o.rectArea.length,numHemiLights:o.hemi.length,numDirLightShadows:o.directionalShadowMap.length,numPointLightShadows:o.pointShadowMap.length,numSpotLightShadows:o.spotShadowMap.length,numSpotLightShadowsWithMaps:o.numSpotLightShadowsWithMaps,numClippingPlanes:a.numPlanes,numClipIntersection:a.numIntersection,dithering:s.dithering,shadowMapEnabled:t.shadowMap.enabled&&h.length>0,shadowMapType:t.shadowMap.type,toneMapping:s.toneMapped?t.toneMapping:0,physicallyCorrectLights:t.physicallyCorrectLights,premultipliedAlpha:s.premultipliedAlpha,doubleSided:2===s.side,flipSided:1===s.side,useDepthPacking:!!s.depthPacking,depthPacking:s.depthPacking||0,index0AttributeName:s.index0AttributeName,extensionDerivatives:s.extensions&&s.extensions.derivatives,extensionFragDepth:s.extensions&&s.extensions.fragDepth,extensionDrawBuffers:s.extensions&&s.extensions.drawBuffers,extensionShaderTextureLOD:s.extensions&&s.extensions.shaderTextureLOD,rendererExtensionFragDepth:u||n.has("EXT_frag_depth"),rendererExtensionDrawBuffers:u||n.has("WEBGL_draw_buffers"),rendererExtensionShaderTextureLod:u||n.has("EXT_shader_texture_lod"),customProgramCacheKey:s.customProgramCacheKey()}},getProgramCacheKey:function(e){const i=[];if(e.shaderID?i.push(e.shaderID):(i.push(e.customVertexShaderID),i.push(e.customFragmentShaderID)),void 0!==e.defines)for(const t in e.defines)i.push(t),i.push(e.defines[t]);return!1===e.isRawShaderMaterial&&(!function(t,e){t.push(e.precision),t.push(e.outputEncoding),t.push(e.envMapMode),t.push(e.envMapCubeUVHeight),t.push(e.combine),t.push(e.vertexUvs),t.push(e.fogExp2),t.push(e.sizeAttenuation),t.push(e.morphTargetsCount),t.push(e.morphAttributeCount),t.push(e.numDirLights),t.push(e.numPointLights),t.push(e.numSpotLights),t.push(e.numSpotLightMaps),t.push(e.numHemiLights),t.push(e.numRectAreaLights),t.push(e.numDirLightShadows),t.push(e.numPointLightShadows),t.push(e.numSpotLightShadows),t.push(e.numSpotLightShadowsWithMaps),t.push(e.shadowMapType),t.push(e.toneMapping),t.push(e.numClippingPlanes),t.push(e.numClipIntersection),t.push(e.depthPacking)}(i,e),function(t,e){o.disableAll(),e.isWebGL2&&o.enable(0);e.supportsVertexTextures&&o.enable(1);e.instancing&&o.enable(2);e.instancingColor&&o.enable(3);e.map&&o.enable(4);e.matcap&&o.enable(5);e.envMap&&o.enable(6);e.lightMap&&o.enable(7);e.aoMap&&o.enable(8);e.emissiveMap&&o.enable(9);e.bumpMap&&o.enable(10);e.normalMap&&o.enable(11);e.objectSpaceNormalMap&&o.enable(12);e.tangentSpaceNormalMap&&o.enable(13);e.clearcoat&&o.enable(14);e.clearcoatMap&&o.enable(15);e.clearcoatRoughnessMap&&o.enable(16);e.clearcoatNormalMap&&o.enable(17);e.iridescence&&o.enable(18);e.iridescenceMap&&o.enable(19);e.iridescenceThicknessMap&&o.enable(20);e.displacementMap&&o.enable(21);e.specularMap&&o.enable(22);e.roughnessMap&&o.enable(23);e.metalnessMap&&o.enable(24);e.gradientMap&&o.enable(25);e.alphaMap&&o.enable(26);e.alphaTest&&o.enable(27);e.vertexColors&&o.enable(28);e.vertexAlphas&&o.enable(29);e.vertexUvs&&o.enable(30);e.vertexTangents&&o.enable(31);e.uvsVertexOnly&&o.enable(32);t.push(o.mask),o.disableAll(),e.fog&&o.enable(0);e.useFog&&o.enable(1);e.flatShading&&o.enable(2);e.logarithmicDepthBuffer&&o.enable(3);e.skinning&&o.enable(4);e.morphTargets&&o.enable(5);e.morphNormals&&o.enable(6);e.morphColors&&o.enable(7);e.premultipliedAlpha&&o.enable(8);e.shadowMapEnabled&&o.enable(9);e.physicallyCorrectLights&&o.enable(10);e.doubleSided&&o.enable(11);e.flipSided&&o.enable(12);e.useDepthPacking&&o.enable(13);e.dithering&&o.enable(14);e.specularIntensityMap&&o.enable(15);e.specularColorMap&&o.enable(16);e.transmission&&o.enable(17);e.transmissionMap&&o.enable(18);e.thicknessMap&&o.enable(19);e.sheen&&o.enable(20);e.sheenColorMap&&o.enable(21);e.sheenRoughnessMap&&o.enable(22);e.decodeVideoTexture&&o.enable(23);e.opaque&&o.enable(24);t.push(o.mask)}(i,e),i.push(t.outputEncoding)),i.push(e.customProgramCacheKey),i.join()},getUniforms:function(t){const e=f[t.type];let i;if(e){const t=bn[e];i=en.clone(t.uniforms)}else i=t.uniforms;return i},acquireProgram:function(e,i){let n;for(let t=0,e=h.length;t<e;t++){const e=h[t];if(e.cacheKey===i){n=e,++n.usedTimes;break}}return void 0===n&&(n=new xs(t,i,e,s),h.push(n)),n},releaseProgram:function(t){if(0==--t.usedTimes){const e=h.indexOf(t);h[e]=h[h.length-1],h.pop(),t.destroy()}},releaseShaderCache:function(t){c.remove(t)},programs:h,dispose:function(){c.dispose()}}}function Ss(){let t=new WeakMap;return{get:function(e){let i=t.get(e);return void 0===i&&(i={},t.set(e,i)),i},remove:function(e){t.delete(e)},update:function(e,i,n){t.get(e)[i]=n},dispose:function(){t=new WeakMap}}}function ws(t,e){return t.groupOrder!==e.groupOrder?t.groupOrder-e.groupOrder:t.renderOrder!==e.renderOrder?t.renderOrder-e.renderOrder:t.material.id!==e.material.id?t.material.id-e.material.id:t.z!==e.z?t.z-e.z:t.id-e.id}function Ts(t,e){return t.groupOrder!==e.groupOrder?t.groupOrder-e.groupOrder:t.renderOrder!==e.renderOrder?t.renderOrder-e.renderOrder:t.z!==e.z?e.z-t.z:t.id-e.id}function As(){const t=[];let e=0;const i=[],n=[],r=[];function s(i,n,r,s,a,o){let l=t[e];return void 0===l?(l={id:i.id,object:i,geometry:n,material:r,groupOrder:s,renderOrder:i.renderOrder,z:a,group:o},t[e]=l):(l.id=i.id,l.object=i,l.geometry=n,l.material=r,l.groupOrder=s,l.renderOrder=i.renderOrder,l.z=a,l.group=o),e++,l}return{opaque:i,transmissive:n,transparent:r,init:function(){e=0,i.length=0,n.length=0,r.length=0},push:function(t,e,a,o,l,c){const h=s(t,e,a,o,l,c);a.transmission>0?n.push(h):!0===a.transparent?r.push(h):i.push(h)},unshift:function(t,e,a,o,l,c){const h=s(t,e,a,o,l,c);a.transmission>0?n.unshift(h):!0===a.transparent?r.unshift(h):i.unshift(h)},finish:function(){for(let i=e,n=t.length;i<n;i++){const e=t[i];if(null===e.id)break;e.id=null,e.object=null,e.geometry=null,e.material=null,e.group=null}},sort:function(t,e){i.length>1&&i.sort(t||ws),n.length>1&&n.sort(e||Ts),r.length>1&&r.sort(e||Ts)}}}function Es(){let t=new WeakMap;return{get:function(e,i){const n=t.get(e);let r;return void 0===n?(r=new As,t.set(e,[r])):i>=n.length?(r=new As,n.push(r)):r=n[i],r},dispose:function(){t=new WeakMap}}}function Cs(){const t={};return{get:function(e){if(void 0!==t[e.id])return t[e.id];let i;switch(e.type){case"DirectionalLight":i={direction:new ne,color:new jt};break;case"SpotLight":i={position:new ne,direction:new ne,color:new jt,distance:0,coneCos:0,penumbraCos:0,decay:0};break;case"PointLight":i={position:new ne,color:new jt,distance:0,decay:0};break;case"HemisphereLight":i={direction:new ne,skyColor:new jt,groundColor:new jt};break;case"RectAreaLight":i={color:new jt,position:new ne,halfWidth:new ne,halfHeight:new ne}}return t[e.id]=i,i}}}let Ls=0;function Rs(t,e){return(e.castShadow?2:0)-(t.castShadow?2:0)+(e.map?1:0)-(t.map?1:0)}function Ps(t,e){const i=new Cs,n=function(){const t={};return{get:function(e){if(void 0!==t[e.id])return t[e.id];let i;switch(e.type){case"DirectionalLight":case"SpotLight":i={shadowBias:0,shadowNormalBias:0,shadowRadius:1,shadowMapSize:new Lt};break;case"PointLight":i={shadowBias:0,shadowNormalBias:0,shadowRadius:1,shadowMapSize:new Lt,shadowCameraNear:1,shadowCameraFar:1e3}}return t[e.id]=i,i}}}(),r={version:0,hash:{directionalLength:-1,pointLength:-1,spotLength:-1,rectAreaLength:-1,hemiLength:-1,numDirectionalShadows:-1,numPointShadows:-1,numSpotShadows:-1,numSpotMaps:-1},ambient:[0,0,0],probe:[],directional:[],directionalShadow:[],directionalShadowMap:[],directionalShadowMatrix:[],spot:[],spotLightMap:[],spotShadow:[],spotShadowMap:[],spotLightMatrix:[],rectArea:[],rectAreaLTC1:null,rectAreaLTC2:null,point:[],pointShadow:[],pointShadowMap:[],pointShadowMatrix:[],hemi:[],numSpotLightShadowsWithMaps:0};for(let t=0;t<9;t++)r.probe.push(new ne);const s=new ne,a=new Ie,o=new Ie;return{setup:function(s,a){let o=0,l=0,c=0;for(let t=0;t<9;t++)r.probe[t].set(0,0,0);let h=0,u=0,d=0,p=0,m=0,f=0,g=0,v=0,x=0,_=0;s.sort(Rs);const y=!0!==a?Math.PI:1;for(let t=0,e=s.length;t<e;t++){const e=s[t],a=e.color,M=e.intensity,b=e.distance,S=e.shadow&&e.shadow.map?e.shadow.map.texture:null;if(e.isAmbientLight)o+=a.r*M*y,l+=a.g*M*y,c+=a.b*M*y;else if(e.isLightProbe)for(let t=0;t<9;t++)r.probe[t].addScaledVector(e.sh.coefficients[t],M);else if(e.isDirectionalLight){const t=i.get(e);if(t.color.copy(e.color).multiplyScalar(e.intensity*y),e.castShadow){const t=e.shadow,i=n.get(e);i.shadowBias=t.bias,i.shadowNormalBias=t.normalBias,i.shadowRadius=t.radius,i.shadowMapSize=t.mapSize,r.directionalShadow[h]=i,r.directionalShadowMap[h]=S,r.directionalShadowMatrix[h]=e.shadow.matrix,f++}r.directional[h]=t,h++}else if(e.isSpotLight){const t=i.get(e);t.position.setFromMatrixPosition(e.matrixWorld),t.color.copy(a).multiplyScalar(M*y),t.distance=b,t.coneCos=Math.cos(e.angle),t.penumbraCos=Math.cos(e.angle*(1-e.penumbra)),t.decay=e.decay,r.spot[d]=t;const s=e.shadow;if(e.map&&(r.spotLightMap[x]=e.map,x++,s.updateMatrices(e),e.castShadow&&_++),r.spotLightMatrix[d]=s.matrix,e.castShadow){const t=n.get(e);t.shadowBias=s.bias,t.shadowNormalBias=s.normalBias,t.shadowRadius=s.radius,t.shadowMapSize=s.mapSize,r.spotShadow[d]=t,r.spotShadowMap[d]=S,v++}d++}else if(e.isRectAreaLight){const t=i.get(e);t.color.copy(a).multiplyScalar(M),t.halfWidth.set(.5*e.width,0,0),t.halfHeight.set(0,.5*e.height,0),r.rectArea[p]=t,p++}else if(e.isPointLight){const t=i.get(e);if(t.color.copy(e.color).multiplyScalar(e.intensity*y),t.distance=e.distance,t.decay=e.decay,e.castShadow){const t=e.shadow,i=n.get(e);i.shadowBias=t.bias,i.shadowNormalBias=t.normalBias,i.shadowRadius=t.radius,i.shadowMapSize=t.mapSize,i.shadowCameraNear=t.camera.near,i.shadowCameraFar=t.camera.far,r.pointShadow[u]=i,r.pointShadowMap[u]=S,r.pointShadowMatrix[u]=e.shadow.matrix,g++}r.point[u]=t,u++}else if(e.isHemisphereLight){const t=i.get(e);t.skyColor.copy(e.color).multiplyScalar(M*y),t.groundColor.copy(e.groundColor).multiplyScalar(M*y),r.hemi[m]=t,m++}}p>0&&(e.isWebGL2||!0===t.has("OES_texture_float_linear")?(r.rectAreaLTC1=Mn.LTC_FLOAT_1,r.rectAreaLTC2=Mn.LTC_FLOAT_2):!0===t.has("OES_texture_half_float_linear")?(r.rectAreaLTC1=Mn.LTC_HALF_1,r.rectAreaLTC2=Mn.LTC_HALF_2):console.error("THREE.WebGLRenderer: Unable to use RectAreaLight. Missing WebGL extensions.")),r.ambient[0]=o,r.ambient[1]=l,r.ambient[2]=c;const M=r.hash;M.directionalLength===h&&M.pointLength===u&&M.spotLength===d&&M.rectAreaLength===p&&M.hemiLength===m&&M.numDirectionalShadows===f&&M.numPointShadows===g&&M.numSpotShadows===v&&M.numSpotMaps===x||(r.directional.length=h,r.spot.length=d,r.rectArea.length=p,r.point.length=u,r.hemi.length=m,r.directionalShadow.length=f,r.directionalShadowMap.length=f,r.pointShadow.length=g,r.pointShadowMap.length=g,r.spotShadow.length=v,r.spotShadowMap.length=v,r.directionalShadowMatrix.length=f,r.pointShadowMatrix.length=g,r.spotLightMatrix.length=v+x-_,r.spotLightMap.length=x,r.numSpotLightShadowsWithMaps=_,M.directionalLength=h,M.pointLength=u,M.spotLength=d,M.rectAreaLength=p,M.hemiLength=m,M.numDirectionalShadows=f,M.numPointShadows=g,M.numSpotShadows=v,M.numSpotMaps=x,r.version=Ls++)},setupView:function(t,e){let i=0,n=0,l=0,c=0,h=0;const u=e.matrixWorldInverse;for(let e=0,d=t.length;e<d;e++){const d=t[e];if(d.isDirectionalLight){const t=r.directional[i];t.direction.setFromMatrixPosition(d.matrixWorld),s.setFromMatrixPosition(d.target.matrixWorld),t.direction.sub(s),t.direction.transformDirection(u),i++}else if(d.isSpotLight){const t=r.spot[l];t.position.setFromMatrixPosition(d.matrixWorld),t.position.applyMatrix4(u),t.direction.setFromMatrixPosition(d.matrixWorld),s.setFromMatrixPosition(d.target.matrixWorld),t.direction.sub(s),t.direction.transformDirection(u),l++}else if(d.isRectAreaLight){const t=r.rectArea[c];t.position.setFromMatrixPosition(d.matrixWorld),t.position.applyMatrix4(u),o.identity(),a.copy(d.matrixWorld),a.premultiply(u),o.extractRotation(a),t.halfWidth.set(.5*d.width,0,0),t.halfHeight.set(0,.5*d.height,0),t.halfWidth.applyMatrix4(o),t.halfHeight.applyMatrix4(o),c++}else if(d.isPointLight){const t=r.point[n];t.position.setFromMatrixPosition(d.matrixWorld),t.position.applyMatrix4(u),n++}else if(d.isHemisphereLight){const t=r.hemi[h];t.direction.setFromMatrixPosition(d.matrixWorld),t.direction.transformDirection(u),h++}}},state:r}}function Ds(t,e){const i=new Ps(t,e),n=[],r=[];return{init:function(){n.length=0,r.length=0},state:{lightsArray:n,shadowsArray:r,lights:i},setupLights:function(t){i.setup(n,t)},setupLightsView:function(t){i.setupView(n,t)},pushLight:function(t){n.push(t)},pushShadow:function(t){r.push(t)}}}function Is(t,e){let i=new WeakMap;return{get:function(n,r=0){const s=i.get(n);let a;return void 0===s?(a=new Ds(t,e),i.set(n,[a])):r>=s.length?(a=new Ds(t,e),s.push(a)):a=s[r],a},dispose:function(){i=new WeakMap}}}class Ns extends vi{constructor(t){super(),this.isMeshDepthMaterial=!0,this.type="MeshDepthMaterial",this.depthPacking=3200,this.map=null,this.alphaMap=null,this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.wireframe=!1,this.wireframeLinewidth=1,this.setValues(t)}copy(t){return super.copy(t),this.depthPacking=t.depthPacking,this.map=t.map,this.alphaMap=t.alphaMap,this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this}}class Os extends vi{constructor(t){super(),this.isMeshDistanceMaterial=!0,this.type="MeshDistanceMaterial",this.referencePosition=new ne,this.nearDistance=1,this.farDistance=1e3,this.map=null,this.alphaMap=null,this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.setValues(t)}copy(t){return super.copy(t),this.referencePosition.copy(t.referencePosition),this.nearDistance=t.nearDistance,this.farDistance=t.farDistance,this.map=t.map,this.alphaMap=t.alphaMap,this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this}}function zs(t,e,i){let n=new gn;const r=new Lt,s=new Lt,a=new $t,o=new Ns({depthPacking:3201}),l=new Os,c={},h=i.maxTextureSize,u={0:1,1:0,2:2},p=new nn({defines:{VSM_SAMPLES:8},uniforms:{shadow_pass:{value:null},resolution:{value:new Lt},radius:{value:4}},vertexShader:"void main() {\n\tgl_Position = vec4( position, 1.0 );\n}",fragmentShader:"uniform sampler2D shadow_pass;\nuniform vec2 resolution;\nuniform float radius;\n#include <packing>\nvoid main() {\n\tconst float samples = float( VSM_SAMPLES );\n\tfloat mean = 0.0;\n\tfloat squared_mean = 0.0;\n\tfloat uvStride = samples <= 1.0 ? 0.0 : 2.0 / ( samples - 1.0 );\n\tfloat uvStart = samples <= 1.0 ? 0.0 : - 1.0;\n\tfor ( float i = 0.0; i < samples; i ++ ) {\n\t\tfloat uvOffset = uvStart + i * uvStride;\n\t\t#ifdef HORIZONTAL_PASS\n\t\t\tvec2 distribution = unpackRGBATo2Half( texture2D( shadow_pass, ( gl_FragCoord.xy + vec2( uvOffset, 0.0 ) * radius ) / resolution ) );\n\t\t\tmean += distribution.x;\n\t\t\tsquared_mean += distribution.y * distribution.y + distribution.x * distribution.x;\n\t\t#else\n\t\t\tfloat depth = unpackRGBAToDepth( texture2D( shadow_pass, ( gl_FragCoord.xy + vec2( 0.0, uvOffset ) * radius ) / resolution ) );\n\t\t\tmean += depth;\n\t\t\tsquared_mean += depth * depth;\n\t\t#endif\n\t}\n\tmean = mean / samples;\n\tsquared_mean = squared_mean / samples;\n\tfloat std_dev = sqrt( squared_mean - mean * mean );\n\tgl_FragColor = pack2HalfToRGBA( vec2( mean, std_dev ) );\n}"}),m=p.clone();m.defines.HORIZONTAL_PASS=1;const f=new Di;f.setAttribute("position",new Mi(new Float32Array([-1,-1,.5,3,-1,.5,-1,3,.5]),3));const g=new Ji(f,p),v=this;function x(i,n){const s=e.update(g);p.defines.VSM_SAMPLES!==i.blurSamples&&(p.defines.VSM_SAMPLES=i.blurSamples,m.defines.VSM_SAMPLES=i.blurSamples,p.needsUpdate=!0,m.needsUpdate=!0),null===i.mapPass&&(i.mapPass=new Qt(r.x,r.y)),p.uniforms.shadow_pass.value=i.map.texture,p.uniforms.resolution.value=i.mapSize,p.uniforms.radius.value=i.radius,t.setRenderTarget(i.mapPass),t.clear(),t.renderBufferDirect(n,null,s,p,g,null),m.uniforms.shadow_pass.value=i.mapPass.texture,m.uniforms.resolution.value=i.mapSize,m.uniforms.radius.value=i.radius,t.setRenderTarget(i.map),t.clear(),t.renderBufferDirect(n,null,s,m,g,null)}function _(e,i,n,r,s,a){let h=null;const d=!0===n.isPointLight?e.customDistanceMaterial:e.customDepthMaterial;if(h=void 0!==d?d:!0===n.isPointLight?l:o,t.localClippingEnabled&&!0===i.clipShadows&&Array.isArray(i.clippingPlanes)&&0!==i.clippingPlanes.length||i.displacementMap&&0!==i.displacementScale||i.alphaMap&&i.alphaTest>0){const t=h.uuid,e=i.uuid;let n=c[t];void 0===n&&(n={},c[t]=n);let r=n[e];void 0===r&&(r=h.clone(),n[e]=r),h=r}return h.visible=i.visible,h.wireframe=i.wireframe,h.side=3===a?null!==i.shadowSide?i.shadowSide:i.side:null!==i.shadowSide?i.shadowSide:u[i.side],h.alphaMap=i.alphaMap,h.alphaTest=i.alphaTest,h.clipShadows=i.clipShadows,h.clippingPlanes=i.clippingPlanes,h.clipIntersection=i.clipIntersection,h.displacementMap=i.displacementMap,h.displacementScale=i.displacementScale,h.displacementBias=i.displacementBias,h.wireframeLinewidth=i.wireframeLinewidth,h.linewidth=i.linewidth,!0===n.isPointLight&&!0===h.isMeshDistanceMaterial&&(h.referencePosition.setFromMatrixPosition(n.matrixWorld),h.nearDistance=r,h.farDistance=s),h}function y(i,r,s,a,o){if(!1===i.visible)return;if(i.layers.test(r.layers)&&(i.isMesh||i.isLine||i.isPoints)&&(i.castShadow||i.receiveShadow&&3===o)&&(!i.frustumCulled||n.intersectsObject(i))){i.modelViewMatrix.multiplyMatrices(s.matrixWorldInverse,i.matrixWorld);const n=e.update(i),r=i.material;if(Array.isArray(r)){const e=n.groups;for(let l=0,c=e.length;l<c;l++){const c=e[l],h=r[c.materialIndex];if(h&&h.visible){const e=_(i,h,a,s.near,s.far,o);t.renderBufferDirect(s,null,n,e,i,c)}}}else if(r.visible){const e=_(i,r,a,s.near,s.far,o);t.renderBufferDirect(s,null,n,e,i,null)}}const l=i.children;for(let t=0,e=l.length;t<e;t++)y(l[t],r,s,a,o)}this.enabled=!1,this.autoUpdate=!0,this.needsUpdate=!1,this.type=1,this.render=function(e,i,o){if(!1===v.enabled)return;if(!1===v.autoUpdate&&!1===v.needsUpdate)return;if(0===e.length)return;const l=t.getRenderTarget(),c=t.getActiveCubeFace(),u=t.getActiveMipmapLevel(),p=t.state;p.setBlending(0),p.buffers.color.setClear(1,1,1,1),p.buffers.depth.setTest(!0),p.setScissorTest(!1);for(let l=0,c=e.length;l<c;l++){const c=e[l],u=c.shadow;if(void 0===u){console.warn("THREE.WebGLShadowMap:",c,"has no shadow.");continue}if(!1===u.autoUpdate&&!1===u.needsUpdate)continue;r.copy(u.mapSize);const m=u.getFrameExtents();if(r.multiply(m),s.copy(u.mapSize),(r.x>h||r.y>h)&&(r.x>h&&(s.x=Math.floor(h/m.x),r.x=s.x*m.x,u.mapSize.x=s.x),r.y>h&&(s.y=Math.floor(h/m.y),r.y=s.y*m.y,u.mapSize.y=s.y)),null===u.map){const t=3!==this.type?{minFilter:d,magFilter:d}:{};u.map=new Qt(r.x,r.y,t),u.map.texture.name=c.name+".shadowMap",u.camera.updateProjectionMatrix()}t.setRenderTarget(u.map),t.clear();const f=u.getViewportCount();for(let t=0;t<f;t++){const e=u.getViewport(t);a.set(s.x*e.x,s.y*e.y,s.x*e.z,s.y*e.w),p.viewport(a),u.updateMatrices(c,t),n=u.getFrustum(),y(i,o,u.camera,c,this.type)}!0!==u.isPointLightShadow&&3===this.type&&x(u,o),u.needsUpdate=!1}v.needsUpdate=!1,t.setRenderTarget(l,c,u)}}function Us(t,e,n){const r=n.isWebGL2;const s=new function(){let e=!1;const i=new $t;let n=null;const r=new $t(0,0,0,0);return{setMask:function(i){n===i||e||(t.colorMask(i,i,i,i),n=i)},setLocked:function(t){e=t},setClear:function(e,n,s,a,o){!0===o&&(e*=a,n*=a,s*=a),i.set(e,n,s,a),!1===r.equals(i)&&(t.clearColor(e,n,s,a),r.copy(i))},reset:function(){e=!1,n=null,r.set(-1,0,0,0)}}},a=new function(){let e=!1,i=null,n=null,r=null;return{setTest:function(t){t?G(2929):V(2929)},setMask:function(n){i===n||e||(t.depthMask(n),i=n)},setFunc:function(e){if(n!==e){switch(e){case 0:t.depthFunc(512);break;case 1:t.depthFunc(519);break;case 2:t.depthFunc(513);break;case 3:default:t.depthFunc(515);break;case 4:t.depthFunc(514);break;case 5:t.depthFunc(518);break;case 6:t.depthFunc(516);break;case 7:t.depthFunc(517)}n=e}},setLocked:function(t){e=t},setClear:function(e){r!==e&&(t.clearDepth(e),r=e)},reset:function(){e=!1,i=null,n=null,r=null}}},o=new function(){let e=!1,i=null,n=null,r=null,s=null,a=null,o=null,l=null,c=null;return{setTest:function(t){e||(t?G(2960):V(2960))},setMask:function(n){i===n||e||(t.stencilMask(n),i=n)},setFunc:function(e,i,a){n===e&&r===i&&s===a||(t.stencilFunc(e,i,a),n=e,r=i,s=a)},setOp:function(e,i,n){a===e&&o===i&&l===n||(t.stencilOp(e,i,n),a=e,o=i,l=n)},setLocked:function(t){e=t},setClear:function(e){c!==e&&(t.clearStencil(e),c=e)},reset:function(){e=!1,i=null,n=null,r=null,s=null,a=null,o=null,l=null,c=null}}},l=new WeakMap,c=new WeakMap;let h={},u={},d=new WeakMap,p=[],m=null,f=!1,g=null,v=null,x=null,_=null,y=null,M=null,b=null,S=!1,w=null,T=null,A=null,E=null,C=null;const L=t.getParameter(35661);let R=!1,P=0;const D=t.getParameter(7938);-1!==D.indexOf("WebGL")?(P=parseFloat(/^WebGL (\d)/.exec(D)[1]),R=P>=1):-1!==D.indexOf("OpenGL ES")&&(P=parseFloat(/^OpenGL ES (\d)/.exec(D)[1]),R=P>=2);let I=null,N={};const O=t.getParameter(3088),z=t.getParameter(2978),U=(new $t).fromArray(O),B=(new $t).fromArray(z);function F(e,i,n){const r=new Uint8Array(4),s=t.createTexture();t.bindTexture(e,s),t.texParameteri(e,10241,9728),t.texParameteri(e,10240,9728);for(let e=0;e<n;e++)t.texImage2D(i+e,0,6408,1,1,0,6408,5121,r);return s}const k={};function G(e){!0!==h[e]&&(t.enable(e),h[e]=!0)}function V(e){!1!==h[e]&&(t.disable(e),h[e]=!1)}k[3553]=F(3553,3553,1),k[34067]=F(34067,34069,6),s.setClear(0,0,0,1),a.setClear(1),o.setClear(0),G(2929),a.setFunc(3),q(!1),X(1),G(2884),j(0);const H={[i]:32774,101:32778,102:32779};if(r)H[103]=32775,H[104]=32776;else{const t=e.get("EXT_blend_minmax");null!==t&&(H[103]=t.MIN_EXT,H[104]=t.MAX_EXT)}const W={200:0,201:1,202:768,204:770,210:776,208:774,206:772,203:769,205:771,209:775,207:773};function j(e,n,r,s,a,o,l,c){if(0!==e){if(!1===f&&(G(3042),f=!0),5===e)a=a||n,o=o||r,l=l||s,n===v&&a===y||(t.blendEquationSeparate(H[n],H[a]),v=n,y=a),r===x&&s===_&&o===M&&l===b||(t.blendFuncSeparate(W[r],W[s],W[o],W[l]),x=r,_=s,M=o,b=l),g=e,S=null;else if(e!==g||c!==S){if(v===i&&y===i||(t.blendEquation(32774),v=i,y=i),c)switch(e){case 1:t.blendFuncSeparate(1,771,1,771);break;case 2:t.blendFunc(1,1);break;case 3:t.blendFuncSeparate(0,769,0,1);break;case 4:t.blendFuncSeparate(0,768,0,770);break;default:console.error("THREE.WebGLState: Invalid blending: ",e)}else switch(e){case 1:t.blendFuncSeparate(770,771,1,771);break;case 2:t.blendFunc(770,1);break;case 3:t.blendFuncSeparate(0,769,0,1);break;case 4:t.blendFunc(0,768);break;default:console.error("THREE.WebGLState: Invalid blending: ",e)}x=null,_=null,M=null,b=null,g=e,S=c}}else!0===f&&(V(3042),f=!1)}function q(e){w!==e&&(e?t.frontFace(2304):t.frontFace(2305),w=e)}function X(e){0!==e?(G(2884),e!==T&&(1===e?t.cullFace(1029):2===e?t.cullFace(1028):t.cullFace(1032))):V(2884),T=e}function Y(e,i,n){e?(G(32823),E===i&&C===n||(t.polygonOffset(i,n),E=i,C=n)):V(32823)}return{buffers:{color:s,depth:a,stencil:o},enable:G,disable:V,bindFramebuffer:function(e,i){return u[e]!==i&&(t.bindFramebuffer(e,i),u[e]=i,r&&(36009===e&&(u[36160]=i),36160===e&&(u[36009]=i)),!0)},drawBuffers:function(i,r){let s=p,a=!1;if(i)if(s=d.get(r),void 0===s&&(s=[],d.set(r,s)),i.isWebGLMultipleRenderTargets){const t=i.texture;if(s.length!==t.length||36064!==s[0]){for(let e=0,i=t.length;e<i;e++)s[e]=36064+e;s.length=t.length,a=!0}}else 36064!==s[0]&&(s[0]=36064,a=!0);else 1029!==s[0]&&(s[0]=1029,a=!0);a&&(n.isWebGL2?t.drawBuffers(s):e.get("WEBGL_draw_buffers").drawBuffersWEBGL(s))},useProgram:function(e){return m!==e&&(t.useProgram(e),m=e,!0)},setBlending:j,setMaterial:function(t,e){2===t.side?V(2884):G(2884);let i=1===t.side;e&&(i=!i),q(i),1===t.blending&&!1===t.transparent?j(0):j(t.blending,t.blendEquation,t.blendSrc,t.blendDst,t.blendEquationAlpha,t.blendSrcAlpha,t.blendDstAlpha,t.premultipliedAlpha),a.setFunc(t.depthFunc),a.setTest(t.depthTest),a.setMask(t.depthWrite),s.setMask(t.colorWrite);const n=t.stencilWrite;o.setTest(n),n&&(o.setMask(t.stencilWriteMask),o.setFunc(t.stencilFunc,t.stencilRef,t.stencilFuncMask),o.setOp(t.stencilFail,t.stencilZFail,t.stencilZPass)),Y(t.polygonOffset,t.polygonOffsetFactor,t.polygonOffsetUnits),!0===t.alphaToCoverage?G(32926):V(32926)},setFlipSided:q,setCullFace:X,setLineWidth:function(e){e!==A&&(R&&t.lineWidth(e),A=e)},setPolygonOffset:Y,setScissorTest:function(t){t?G(3089):V(3089)},activeTexture:function(e){void 0===e&&(e=33984+L-1),I!==e&&(t.activeTexture(e),I=e)},bindTexture:function(e,i,n){void 0===n&&(n=null===I?33984+L-1:I);let r=N[n];void 0===r&&(r={type:void 0,texture:void 0},N[n]=r),r.type===e&&r.texture===i||(I!==n&&(t.activeTexture(n),I=n),t.bindTexture(e,i||k[e]),r.type=e,r.texture=i)},unbindTexture:function(){const e=N[I];void 0!==e&&void 0!==e.type&&(t.bindTexture(e.type,null),e.type=void 0,e.texture=void 0)},compressedTexImage2D:function(){try{t.compressedTexImage2D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},compressedTexImage3D:function(){try{t.compressedTexImage3D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},texImage2D:function(){try{t.texImage2D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},texImage3D:function(){try{t.texImage3D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},updateUBOMapping:function(e,i){let n=c.get(i);void 0===n&&(n=new WeakMap,c.set(i,n));let r=n.get(e);void 0===r&&(r=t.getUniformBlockIndex(i,e.name),n.set(e,r))},uniformBlockBinding:function(e,i){const n=c.get(i).get(e);l.get(e)!==n&&(t.uniformBlockBinding(i,n,e.__bindingPointIndex),l.set(e,n))},texStorage2D:function(){try{t.texStorage2D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},texStorage3D:function(){try{t.texStorage3D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},texSubImage2D:function(){try{t.texSubImage2D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},texSubImage3D:function(){try{t.texSubImage3D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},compressedTexSubImage2D:function(){try{t.compressedTexSubImage2D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},compressedTexSubImage3D:function(){try{t.compressedTexSubImage3D.apply(t,arguments)}catch(t){console.error("THREE.WebGLState:",t)}},scissor:function(e){!1===U.equals(e)&&(t.scissor(e.x,e.y,e.z,e.w),U.copy(e))},viewport:function(e){!1===B.equals(e)&&(t.viewport(e.x,e.y,e.z,e.w),B.copy(e))},reset:function(){t.disable(3042),t.disable(2884),t.disable(2929),t.disable(32823),t.disable(3089),t.disable(2960),t.disable(32926),t.blendEquation(32774),t.blendFunc(1,0),t.blendFuncSeparate(1,0,1,0),t.colorMask(!0,!0,!0,!0),t.clearColor(0,0,0,0),t.depthMask(!0),t.depthFunc(513),t.clearDepth(1),t.stencilMask(4294967295),t.stencilFunc(519,0,4294967295),t.stencilOp(7680,7680,7680),t.clearStencil(0),t.cullFace(1029),t.frontFace(2305),t.polygonOffset(0,0),t.activeTexture(33984),t.bindFramebuffer(36160,null),!0===r&&(t.bindFramebuffer(36009,null),t.bindFramebuffer(36008,null)),t.useProgram(null),t.lineWidth(1),t.scissor(0,0,t.canvas.width,t.canvas.height),t.viewport(0,0,t.canvas.width,t.canvas.height),h={},I=null,N={},u={},d=new WeakMap,p=[],m=null,f=!1,g=null,v=null,x=null,_=null,y=null,M=null,b=null,S=!1,w=null,T=null,A=null,E=null,C=null,U.set(0,0,t.canvas.width,t.canvas.height),B.set(0,0,t.canvas.width,t.canvas.height),s.reset(),a.reset(),o.reset()}}}function Bs(t,e,i,n,r,s,a){const o=r.isWebGL2,l=r.maxTextures,E=r.maxCubemapSize,C=r.maxTextureSize,L=r.maxSamples,R=e.has("WEBGL_multisampled_render_to_texture")?e.get("WEBGL_multisampled_render_to_texture"):null,P=/OculusBrowser/g.test("undefined"==typeof navigator?"":navigator.userAgent),D=new WeakMap;let I;const N=new WeakMap;let O=!1;try{O="undefined"!=typeof OffscreenCanvas&&null!==new OffscreenCanvas(1,1).getContext("2d")}catch(t){}function z(t,e){return O?new OffscreenCanvas(t,e):Nt("canvas")}function U(t,e,i,n){let r=1;if((t.width>n||t.height>n)&&(r=n/Math.max(t.width,t.height)),r<1||!0===e){if("undefined"!=typeof HTMLImageElement&&t instanceof HTMLImageElement||"undefined"!=typeof HTMLCanvasElement&&t instanceof HTMLCanvasElement||"undefined"!=typeof ImageBitmap&&t instanceof ImageBitmap){const n=e?Tt:Math.floor,s=n(r*t.width),a=n(r*t.height);void 0===I&&(I=z(s,a));const o=i?z(s,a):I;o.width=s,o.height=a;return o.getContext("2d").drawImage(t,0,0,s,a),console.warn("THREE.WebGLRenderer: Texture has been resized from ("+t.width+"x"+t.height+") to ("+s+"x"+a+")."),o}return"data"in t&&console.warn("THREE.WebGLRenderer: Image in DataTexture is too big ("+t.width+"x"+t.height+")."),t}return t}function B(t){return St(t.width)&&St(t.height)}function F(t,e){return t.generateMipmaps&&e&&t.minFilter!==d&&t.minFilter!==f}function k(e){t.generateMipmap(e)}function G(i,n,r,s,a=!1){if(!1===o)return n;if(null!==i){if(void 0!==t[i])return t[i];console.warn("THREE.WebGLRenderer: Attempt to use non-existing WebGL internal format '"+i+"'")}let l=n;return 6403===n&&(5126===r&&(l=33326),5131===r&&(l=33325),5121===r&&(l=33321)),33319===n&&(5126===r&&(l=33328),5131===r&&(l=33327),5121===r&&(l=33323)),6408===n&&(5126===r&&(l=34836),5131===r&&(l=34842),5121===r&&(l=s===ot&&!1===a?35907:32856),32819===r&&(l=32854),32820===r&&(l=32855)),33325!==l&&33326!==l&&33327!==l&&33328!==l&&34842!==l&&34836!==l||e.get("EXT_color_buffer_float"),l}function V(t,e,i){return!0===F(t,i)||t.isFramebufferTexture&&t.minFilter!==d&&t.minFilter!==f?Math.log2(Math.max(e.width,e.height))+1:void 0!==t.mipmaps&&t.mipmaps.length>0?t.mipmaps.length:t.isCompressedTexture&&Array.isArray(t.image)?e.mipmaps.length:1}function H(t){return t===d||t===p||t===m?9728:9729}function W(t){const e=t.target;e.removeEventListener("dispose",W),function(t){const e=n.get(t);if(void 0===e.__webglInit)return;const i=t.source,r=N.get(i);if(r){const n=r[e.__cacheKey];n.usedTimes--,0===n.usedTimes&&q(t),0===Object.keys(r).length&&N.delete(i)}n.remove(t)}(e),e.isVideoTexture&&D.delete(e)}function j(e){const i=e.target;i.removeEventListener("dispose",j),function(e){const i=e.texture,r=n.get(e),s=n.get(i);void 0!==s.__webglTexture&&(t.deleteTexture(s.__webglTexture),a.memory.textures--);e.depthTexture&&e.depthTexture.dispose();if(e.isWebGLCubeRenderTarget)for(let e=0;e<6;e++)t.deleteFramebuffer(r.__webglFramebuffer[e]),r.__webglDepthbuffer&&t.deleteRenderbuffer(r.__webglDepthbuffer[e]);else{if(t.deleteFramebuffer(r.__webglFramebuffer),r.__webglDepthbuffer&&t.deleteRenderbuffer(r.__webglDepthbuffer),r.__webglMultisampledFramebuffer&&t.deleteFramebuffer(r.__webglMultisampledFramebuffer),r.__webglColorRenderbuffer)for(let e=0;e<r.__webglColorRenderbuffer.length;e++)r.__webglColorRenderbuffer[e]&&t.deleteRenderbuffer(r.__webglColorRenderbuffer[e]);r.__webglDepthRenderbuffer&&t.deleteRenderbuffer(r.__webglDepthRenderbuffer)}if(e.isWebGLMultipleRenderTargets)for(let e=0,r=i.length;e<r;e++){const r=n.get(i[e]);r.__webglTexture&&(t.deleteTexture(r.__webglTexture),a.memory.textures--),n.remove(i[e])}n.remove(i),n.remove(e)}(i)}function q(e){const i=n.get(e);t.deleteTexture(i.__webglTexture);const r=e.source;delete N.get(r)[i.__cacheKey],a.memory.textures--}let X=0;function Y(t,e){const r=n.get(t);if(t.isVideoTexture&&function(t){const e=a.render.frame;D.get(t)!==e&&(D.set(t,e),t.update())}(t),!1===t.isRenderTargetTexture&&t.version>0&&r.__version!==t.version){const i=t.image;if(null===i)console.warn("THREE.WebGLRenderer: Texture marked for update but no image data found.");else{if(!1!==i.complete)return void Q(r,t,e);console.warn("THREE.WebGLRenderer: Texture marked for update but image is incomplete")}}i.bindTexture(3553,r.__webglTexture,33984+e)}const Z={[c]:10497,[h]:33071,[u]:33648},J={[d]:9728,[p]:9984,[m]:9986,[f]:9729,[g]:9985,[v]:9987};function K(i,s,a){if(a?(t.texParameteri(i,10242,Z[s.wrapS]),t.texParameteri(i,10243,Z[s.wrapT]),32879!==i&&35866!==i||t.texParameteri(i,32882,Z[s.wrapR]),t.texParameteri(i,10240,J[s.magFilter]),t.texParameteri(i,10241,J[s.minFilter])):(t.texParameteri(i,10242,33071),t.texParameteri(i,10243,33071),32879!==i&&35866!==i||t.texParameteri(i,32882,33071),s.wrapS===h&&s.wrapT===h||console.warn("THREE.WebGLRenderer: Texture is not power of two. Texture.wrapS and Texture.wrapT should be set to THREE.ClampToEdgeWrapping."),t.texParameteri(i,10240,H(s.magFilter)),t.texParameteri(i,10241,H(s.minFilter)),s.minFilter!==d&&s.minFilter!==f&&console.warn("THREE.WebGLRenderer: Texture is not power of two. Texture.minFilter should be set to THREE.NearestFilter or THREE.LinearFilter.")),!0===e.has("EXT_texture_filter_anisotropic")){const a=e.get("EXT_texture_filter_anisotropic");if(s.type===M&&!1===e.has("OES_texture_float_linear"))return;if(!1===o&&s.type===b&&!1===e.has("OES_texture_half_float_linear"))return;(s.anisotropy>1||n.get(s).__currentAnisotropy)&&(t.texParameterf(i,a.TEXTURE_MAX_ANISOTROPY_EXT,Math.min(s.anisotropy,r.getMaxAnisotropy())),n.get(s).__currentAnisotropy=s.anisotropy)}}function $(e,i){let n=!1;void 0===e.__webglInit&&(e.__webglInit=!0,i.addEventListener("dispose",W));const r=i.source;let s=N.get(r);void 0===s&&(s={},N.set(r,s));const o=function(t){const e=[];return e.push(t.wrapS),e.push(t.wrapT),e.push(t.wrapR||0),e.push(t.magFilter),e.push(t.minFilter),e.push(t.anisotropy),e.push(t.internalFormat),e.push(t.format),e.push(t.type),e.push(t.generateMipmaps),e.push(t.premultiplyAlpha),e.push(t.flipY),e.push(t.unpackAlignment),e.push(t.encoding),e.join()}(i);if(o!==e.__cacheKey){void 0===s[o]&&(s[o]={texture:t.createTexture(),usedTimes:0},a.memory.textures++,n=!0),s[o].usedTimes++;const r=s[e.__cacheKey];void 0!==r&&(s[e.__cacheKey].usedTimes--,0===r.usedTimes&&q(i)),e.__cacheKey=o,e.__webglTexture=s[o].texture}return n}function Q(e,r,a){let l=3553;(r.isDataArrayTexture||r.isCompressedArrayTexture)&&(l=35866),r.isData3DTexture&&(l=32879);const c=$(e,r),u=r.source;i.bindTexture(l,e.__webglTexture,33984+a);const p=n.get(u);if(u.version!==p.__version||!0===c){i.activeTexture(33984+a),t.pixelStorei(37440,r.flipY),t.pixelStorei(37441,r.premultiplyAlpha),t.pixelStorei(3317,r.unpackAlignment),t.pixelStorei(37443,0);const e=function(t){return!o&&(t.wrapS!==h||t.wrapT!==h||t.minFilter!==d&&t.minFilter!==f)}(r)&&!1===B(r.image);let n=U(r.image,e,!1,C);n=st(r,n);const m=B(n)||o,g=s.convert(r.format,r.encoding);let v,x=s.convert(r.type),b=G(r.internalFormat,g,x,r.encoding,r.isVideoTexture);K(l,r,m);const E=r.mipmaps,L=o&&!0!==r.isVideoTexture,R=void 0===p.__version||!0===c,P=V(r,n,m);if(r.isDepthTexture)b=6402,o?b=r.type===M?36012:r.type===y?33190:r.type===S?35056:33189:r.type===M&&console.error("WebGLRenderer: Floating point depth texture requires WebGL2."),r.format===T&&6402===b&&r.type!==_&&r.type!==y&&(console.warn("THREE.WebGLRenderer: Use UnsignedShortType or UnsignedIntType for DepthFormat DepthTexture."),r.type=y,x=s.convert(r.type)),r.format===A&&6402===b&&(b=34041,r.type!==S&&(console.warn("THREE.WebGLRenderer: Use UnsignedInt248Type for DepthStencilFormat DepthTexture."),r.type=S,x=s.convert(r.type))),R&&(L?i.texStorage2D(3553,1,b,n.width,n.height):i.texImage2D(3553,0,b,n.width,n.height,0,g,x,null));else if(r.isDataTexture)if(E.length>0&&m){L&&R&&i.texStorage2D(3553,P,b,E[0].width,E[0].height);for(let t=0,e=E.length;t<e;t++)v=E[t],L?i.texSubImage2D(3553,t,0,0,v.width,v.height,g,x,v.data):i.texImage2D(3553,t,b,v.width,v.height,0,g,x,v.data);r.generateMipmaps=!1}else L?(R&&i.texStorage2D(3553,P,b,n.width,n.height),i.texSubImage2D(3553,0,0,0,n.width,n.height,g,x,n.data)):i.texImage2D(3553,0,b,n.width,n.height,0,g,x,n.data);else if(r.isCompressedTexture)if(r.isCompressedArrayTexture){L&&R&&i.texStorage3D(35866,P,b,E[0].width,E[0].height,n.depth);for(let t=0,e=E.length;t<e;t++)v=E[t],r.format!==w?null!==g?L?i.compressedTexSubImage3D(35866,t,0,0,0,v.width,v.height,n.depth,g,v.data,0,0):i.compressedTexImage3D(35866,t,b,v.width,v.height,n.depth,0,v.data,0,0):console.warn("THREE.WebGLRenderer: Attempt to load unsupported compressed texture format in .uploadTexture()"):L?i.texSubImage3D(35866,t,0,0,0,v.width,v.height,n.depth,g,x,v.data):i.texImage3D(35866,t,b,v.width,v.height,n.depth,0,g,x,v.data)}else{L&&R&&i.texStorage2D(3553,P,b,E[0].width,E[0].height);for(let t=0,e=E.length;t<e;t++)v=E[t],r.format!==w?null!==g?L?i.compressedTexSubImage2D(3553,t,0,0,v.width,v.height,g,v.data):i.compressedTexImage2D(3553,t,b,v.width,v.height,0,v.data):console.warn("THREE.WebGLRenderer: Attempt to load unsupported compressed texture format in .uploadTexture()"):L?i.texSubImage2D(3553,t,0,0,v.width,v.height,g,x,v.data):i.texImage2D(3553,t,b,v.width,v.height,0,g,x,v.data)}else if(r.isDataArrayTexture)L?(R&&i.texStorage3D(35866,P,b,n.width,n.height,n.depth),i.texSubImage3D(35866,0,0,0,0,n.width,n.height,n.depth,g,x,n.data)):i.texImage3D(35866,0,b,n.width,n.height,n.depth,0,g,x,n.data);else if(r.isData3DTexture)L?(R&&i.texStorage3D(32879,P,b,n.width,n.height,n.depth),i.texSubImage3D(32879,0,0,0,0,n.width,n.height,n.depth,g,x,n.data)):i.texImage3D(32879,0,b,n.width,n.height,n.depth,0,g,x,n.data);else if(r.isFramebufferTexture){if(R)if(L)i.texStorage2D(3553,P,b,n.width,n.height);else{let t=n.width,e=n.height;for(let n=0;n<P;n++)i.texImage2D(3553,n,b,t,e,0,g,x,null),t>>=1,e>>=1}}else if(E.length>0&&m){L&&R&&i.texStorage2D(3553,P,b,E[0].width,E[0].height);for(let t=0,e=E.length;t<e;t++)v=E[t],L?i.texSubImage2D(3553,t,0,0,g,x,v):i.texImage2D(3553,t,b,g,x,v);r.generateMipmaps=!1}else L?(R&&i.texStorage2D(3553,P,b,n.width,n.height),i.texSubImage2D(3553,0,0,0,g,x,n)):i.texImage2D(3553,0,b,g,x,n);F(r,m)&&k(l),p.__version=u.version,r.onUpdate&&r.onUpdate(r)}e.__version=r.version}function tt(e,r,a,o,l){const c=s.convert(a.format,a.encoding),h=s.convert(a.type),u=G(a.internalFormat,c,h,a.encoding);n.get(r).__hasExternalTextures||(32879===l||35866===l?i.texImage3D(l,0,u,r.width,r.height,r.depth,0,c,h,null):i.texImage2D(l,0,u,r.width,r.height,0,c,h,null)),i.bindFramebuffer(36160,e),rt(r)?R.framebufferTexture2DMultisampleEXT(36160,o,l,n.get(a).__webglTexture,0,nt(r)):(3553===l||l>=34069&&l<=34074)&&t.framebufferTexture2D(36160,o,l,n.get(a).__webglTexture,0),i.bindFramebuffer(36160,null)}function et(e,i,n){if(t.bindRenderbuffer(36161,e),i.depthBuffer&&!i.stencilBuffer){let r=33189;if(n||rt(i)){const e=i.depthTexture;e&&e.isDepthTexture&&(e.type===M?r=36012:e.type===y&&(r=33190));const n=nt(i);rt(i)?R.renderbufferStorageMultisampleEXT(36161,n,r,i.width,i.height):t.renderbufferStorageMultisample(36161,n,r,i.width,i.height)}else t.renderbufferStorage(36161,r,i.width,i.height);t.framebufferRenderbuffer(36160,36096,36161,e)}else if(i.depthBuffer&&i.stencilBuffer){const r=nt(i);n&&!1===rt(i)?t.renderbufferStorageMultisample(36161,r,35056,i.width,i.height):rt(i)?R.renderbufferStorageMultisampleEXT(36161,r,35056,i.width,i.height):t.renderbufferStorage(36161,34041,i.width,i.height),t.framebufferRenderbuffer(36160,33306,36161,e)}else{const e=!0===i.isWebGLMultipleRenderTargets?i.texture:[i.texture];for(let r=0;r<e.length;r++){const a=e[r],o=s.convert(a.format,a.encoding),l=s.convert(a.type),c=G(a.internalFormat,o,l,a.encoding),h=nt(i);n&&!1===rt(i)?t.renderbufferStorageMultisample(36161,h,c,i.width,i.height):rt(i)?R.renderbufferStorageMultisampleEXT(36161,h,c,i.width,i.height):t.renderbufferStorage(36161,c,i.width,i.height)}}t.bindRenderbuffer(36161,null)}function it(e){const r=n.get(e),s=!0===e.isWebGLCubeRenderTarget;if(e.depthTexture&&!r.__autoAllocateDepthBuffer){if(s)throw new Error("target.depthTexture not supported in Cube render targets");!function(e,r){if(r&&r.isWebGLCubeRenderTarget)throw new Error("Depth Texture with cube render targets is not supported");if(i.bindFramebuffer(36160,e),!r.depthTexture||!r.depthTexture.isDepthTexture)throw new Error("renderTarget.depthTexture must be an instance of THREE.DepthTexture");n.get(r.depthTexture).__webglTexture&&r.depthTexture.image.width===r.width&&r.depthTexture.image.height===r.height||(r.depthTexture.image.width=r.width,r.depthTexture.image.height=r.height,r.depthTexture.needsUpdate=!0),Y(r.depthTexture,0);const s=n.get(r.depthTexture).__webglTexture,a=nt(r);if(r.depthTexture.format===T)rt(r)?R.framebufferTexture2DMultisampleEXT(36160,36096,3553,s,0,a):t.framebufferTexture2D(36160,36096,3553,s,0);else{if(r.depthTexture.format!==A)throw new Error("Unknown depthTexture format");rt(r)?R.framebufferTexture2DMultisampleEXT(36160,33306,3553,s,0,a):t.framebufferTexture2D(36160,33306,3553,s,0)}}(r.__webglFramebuffer,e)}else if(s){r.__webglDepthbuffer=[];for(let n=0;n<6;n++)i.bindFramebuffer(36160,r.__webglFramebuffer[n]),r.__webglDepthbuffer[n]=t.createRenderbuffer(),et(r.__webglDepthbuffer[n],e,!1)}else i.bindFramebuffer(36160,r.__webglFramebuffer),r.__webglDepthbuffer=t.createRenderbuffer(),et(r.__webglDepthbuffer,e,!1);i.bindFramebuffer(36160,null)}function nt(t){return Math.min(L,t.samples)}function rt(t){const i=n.get(t);return o&&t.samples>0&&!0===e.has("WEBGL_multisampled_render_to_texture")&&!1!==i.__useRenderToTexture}function st(t,i){const n=t.encoding,r=t.format,s=t.type;return!0===t.isCompressedTexture||!0===t.isVideoTexture||t.format===pt||n!==at&&(n===ot?!1===o?!0===e.has("EXT_sRGB")&&r===w?(t.format=pt,t.minFilter=f,t.generateMipmaps=!1):i=Xt.sRGBToLinear(i):r===w&&s===x||console.warn("THREE.WebGLTextures: sRGB encoded textures have to use RGBAFormat and UnsignedByteType."):console.error("THREE.WebGLTextures: Unsupported texture encoding:",n)),i}this.allocateTextureUnit=function(){const t=X;return t>=l&&console.warn("THREE.WebGLTextures: Trying to use "+t+" texture units while this GPU supports only "+l),X+=1,t},this.resetTextureUnits=function(){X=0},this.setTexture2D=Y,this.setTexture2DArray=function(t,e){const r=n.get(t);t.version>0&&r.__version!==t.version?Q(r,t,e):i.bindTexture(35866,r.__webglTexture,33984+e)},this.setTexture3D=function(t,e){const r=n.get(t);t.version>0&&r.__version!==t.version?Q(r,t,e):i.bindTexture(32879,r.__webglTexture,33984+e)},this.setTextureCube=function(e,r){const a=n.get(e);e.version>0&&a.__version!==e.version?function(e,r,a){if(6!==r.image.length)return;const l=$(e,r),c=r.source;i.bindTexture(34067,e.__webglTexture,33984+a);const h=n.get(c);if(c.version!==h.__version||!0===l){i.activeTexture(33984+a),t.pixelStorei(37440,r.flipY),t.pixelStorei(37441,r.premultiplyAlpha),t.pixelStorei(3317,r.unpackAlignment),t.pixelStorei(37443,0);const e=r.isCompressedTexture||r.image[0].isCompressedTexture,n=r.image[0]&&r.image[0].isDataTexture,u=[];for(let t=0;t<6;t++)u[t]=e||n?n?r.image[t].image:r.image[t]:U(r.image[t],!1,!0,E),u[t]=st(r,u[t]);const d=u[0],p=B(d)||o,m=s.convert(r.format,r.encoding),f=s.convert(r.type),g=G(r.internalFormat,m,f,r.encoding),v=o&&!0!==r.isVideoTexture,x=void 0===h.__version||!0===l;let _,y=V(r,d,p);if(K(34067,r,p),e){v&&x&&i.texStorage2D(34067,y,g,d.width,d.height);for(let t=0;t<6;t++){_=u[t].mipmaps;for(let e=0;e<_.length;e++){const n=_[e];r.format!==w?null!==m?v?i.compressedTexSubImage2D(34069+t,e,0,0,n.width,n.height,m,n.data):i.compressedTexImage2D(34069+t,e,g,n.width,n.height,0,n.data):console.warn("THREE.WebGLRenderer: Attempt to load unsupported compressed texture format in .setTextureCube()"):v?i.texSubImage2D(34069+t,e,0,0,n.width,n.height,m,f,n.data):i.texImage2D(34069+t,e,g,n.width,n.height,0,m,f,n.data)}}}else{_=r.mipmaps,v&&x&&(_.length>0&&y++,i.texStorage2D(34067,y,g,u[0].width,u[0].height));for(let t=0;t<6;t++)if(n){v?i.texSubImage2D(34069+t,0,0,0,u[t].width,u[t].height,m,f,u[t].data):i.texImage2D(34069+t,0,g,u[t].width,u[t].height,0,m,f,u[t].data);for(let e=0;e<_.length;e++){const n=_[e].image[t].image;v?i.texSubImage2D(34069+t,e+1,0,0,n.width,n.height,m,f,n.data):i.texImage2D(34069+t,e+1,g,n.width,n.height,0,m,f,n.data)}}else{v?i.texSubImage2D(34069+t,0,0,0,m,f,u[t]):i.texImage2D(34069+t,0,g,m,f,u[t]);for(let e=0;e<_.length;e++){const n=_[e];v?i.texSubImage2D(34069+t,e+1,0,0,m,f,n.image[t]):i.texImage2D(34069+t,e+1,g,m,f,n.image[t])}}}F(r,p)&&k(34067),h.__version=c.version,r.onUpdate&&r.onUpdate(r)}e.__version=r.version}(a,e,r):i.bindTexture(34067,a.__webglTexture,33984+r)},this.rebindTextures=function(t,e,i){const r=n.get(t);void 0!==e&&tt(r.__webglFramebuffer,t,t.texture,36064,3553),void 0!==i&&it(t)},this.setupRenderTarget=function(e){const l=e.texture,c=n.get(e),h=n.get(l);e.addEventListener("dispose",j),!0!==e.isWebGLMultipleRenderTargets&&(void 0===h.__webglTexture&&(h.__webglTexture=t.createTexture()),h.__version=l.version,a.memory.textures++);const u=!0===e.isWebGLCubeRenderTarget,d=!0===e.isWebGLMultipleRenderTargets,p=B(e)||o;if(u){c.__webglFramebuffer=[];for(let e=0;e<6;e++)c.__webglFramebuffer[e]=t.createFramebuffer()}else{if(c.__webglFramebuffer=t.createFramebuffer(),d)if(r.drawBuffers){const i=e.texture;for(let e=0,r=i.length;e<r;e++){const r=n.get(i[e]);void 0===r.__webglTexture&&(r.__webglTexture=t.createTexture(),a.memory.textures++)}}else console.warn("THREE.WebGLRenderer: WebGLMultipleRenderTargets can only be used with WebGL2 or WEBGL_draw_buffers extension.");if(o&&e.samples>0&&!1===rt(e)){const n=d?l:[l];c.__webglMultisampledFramebuffer=t.createFramebuffer(),c.__webglColorRenderbuffer=[],i.bindFramebuffer(36160,c.__webglMultisampledFramebuffer);for(let i=0;i<n.length;i++){const r=n[i];c.__webglColorRenderbuffer[i]=t.createRenderbuffer(),t.bindRenderbuffer(36161,c.__webglColorRenderbuffer[i]);const a=s.convert(r.format,r.encoding),o=s.convert(r.type),l=G(r.internalFormat,a,o,r.encoding,!0===e.isXRRenderTarget),h=nt(e);t.renderbufferStorageMultisample(36161,h,l,e.width,e.height),t.framebufferRenderbuffer(36160,36064+i,36161,c.__webglColorRenderbuffer[i])}t.bindRenderbuffer(36161,null),e.depthBuffer&&(c.__webglDepthRenderbuffer=t.createRenderbuffer(),et(c.__webglDepthRenderbuffer,e,!0)),i.bindFramebuffer(36160,null)}}if(u){i.bindTexture(34067,h.__webglTexture),K(34067,l,p);for(let t=0;t<6;t++)tt(c.__webglFramebuffer[t],e,l,36064,34069+t);F(l,p)&&k(34067),i.unbindTexture()}else if(d){const t=e.texture;for(let r=0,s=t.length;r<s;r++){const s=t[r],a=n.get(s);i.bindTexture(3553,a.__webglTexture),K(3553,s,p),tt(c.__webglFramebuffer,e,s,36064+r,3553),F(s,p)&&k(3553)}i.unbindTexture()}else{let t=3553;(e.isWebGL3DRenderTarget||e.isWebGLArrayRenderTarget)&&(o?t=e.isWebGL3DRenderTarget?32879:35866:console.error("THREE.WebGLTextures: THREE.Data3DTexture and THREE.DataArrayTexture only supported with WebGL2.")),i.bindTexture(t,h.__webglTexture),K(t,l,p),tt(c.__webglFramebuffer,e,l,36064,t),F(l,p)&&k(t),i.unbindTexture()}e.depthBuffer&&it(e)},this.updateRenderTargetMipmap=function(t){const e=B(t)||o,r=!0===t.isWebGLMultipleRenderTargets?t.texture:[t.texture];for(let s=0,a=r.length;s<a;s++){const a=r[s];if(F(a,e)){const e=t.isWebGLCubeRenderTarget?34067:3553,r=n.get(a).__webglTexture;i.bindTexture(e,r),k(e),i.unbindTexture()}}},this.updateMultisampleRenderTarget=function(e){if(o&&e.samples>0&&!1===rt(e)){const r=e.isWebGLMultipleRenderTargets?e.texture:[e.texture],s=e.width,a=e.height;let o=16384;const l=[],c=e.stencilBuffer?33306:36096,h=n.get(e),u=!0===e.isWebGLMultipleRenderTargets;if(u)for(let e=0;e<r.length;e++)i.bindFramebuffer(36160,h.__webglMultisampledFramebuffer),t.framebufferRenderbuffer(36160,36064+e,36161,null),i.bindFramebuffer(36160,h.__webglFramebuffer),t.framebufferTexture2D(36009,36064+e,3553,null,0);i.bindFramebuffer(36008,h.__webglMultisampledFramebuffer),i.bindFramebuffer(36009,h.__webglFramebuffer);for(let i=0;i<r.length;i++){l.push(36064+i),e.depthBuffer&&l.push(c);const d=void 0!==h.__ignoreDepthValues&&h.__ignoreDepthValues;if(!1===d&&(e.depthBuffer&&(o|=256),e.stencilBuffer&&(o|=1024)),u&&t.framebufferRenderbuffer(36008,36064,36161,h.__webglColorRenderbuffer[i]),!0===d&&(t.invalidateFramebuffer(36008,[c]),t.invalidateFramebuffer(36009,[c])),u){const e=n.get(r[i]).__webglTexture;t.framebufferTexture2D(36009,36064,3553,e,0)}t.blitFramebuffer(0,0,s,a,0,0,s,a,o,9728),P&&t.invalidateFramebuffer(36008,l)}if(i.bindFramebuffer(36008,null),i.bindFramebuffer(36009,null),u)for(let e=0;e<r.length;e++){i.bindFramebuffer(36160,h.__webglMultisampledFramebuffer),t.framebufferRenderbuffer(36160,36064+e,36161,h.__webglColorRenderbuffer[e]);const s=n.get(r[e]).__webglTexture;i.bindFramebuffer(36160,h.__webglFramebuffer),t.framebufferTexture2D(36009,36064+e,3553,s,0)}i.bindFramebuffer(36009,h.__webglMultisampledFramebuffer)}},this.setupDepthRenderbuffer=it,this.setupFrameBufferTexture=tt,this.useMultisampledRTT=rt}function Fs(t,e,i){const n=i.isWebGL2;return{convert:function(i,r=null){let s;if(i===x)return 5121;if(1017===i)return 32819;if(1018===i)return 32820;if(1010===i)return 5120;if(1011===i)return 5122;if(i===_)return 5123;if(1013===i)return 5124;if(i===y)return 5125;if(i===M)return 5126;if(i===b)return n?5131:(s=e.get("OES_texture_half_float"),null!==s?s.HALF_FLOAT_OES:null);if(1021===i)return 6406;if(i===w)return 6408;if(1024===i)return 6409;if(1025===i)return 6410;if(i===T)return 6402;if(i===A)return 34041;if(1028===i)return 6403;if(1022===i)return console.warn("THREE.WebGLRenderer: THREE.RGBFormat has been removed. Use THREE.RGBAFormat instead. https://github.com/mrdoob/three.js/pull/23228"),6408;if(i===pt)return s=e.get("EXT_sRGB"),null!==s?s.SRGB_ALPHA_EXT:null;if(1029===i)return 36244;if(1030===i)return 33319;if(1031===i)return 33320;if(1033===i)return 36249;if(i===E||i===C||i===L||i===R)if(r===ot){if(s=e.get("WEBGL_compressed_texture_s3tc_srgb"),null===s)return null;if(i===E)return s.COMPRESSED_SRGB_S3TC_DXT1_EXT;if(i===C)return s.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT;if(i===L)return s.COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT;if(i===R)return s.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT}else{if(s=e.get("WEBGL_compressed_texture_s3tc"),null===s)return null;if(i===E)return s.COMPRESSED_RGB_S3TC_DXT1_EXT;if(i===C)return s.COMPRESSED_RGBA_S3TC_DXT1_EXT;if(i===L)return s.COMPRESSED_RGBA_S3TC_DXT3_EXT;if(i===R)return s.COMPRESSED_RGBA_S3TC_DXT5_EXT}if(i===P||i===D||i===I||i===N){if(s=e.get("WEBGL_compressed_texture_pvrtc"),null===s)return null;if(i===P)return s.COMPRESSED_RGB_PVRTC_4BPPV1_IMG;if(i===D)return s.COMPRESSED_RGB_PVRTC_2BPPV1_IMG;if(i===I)return s.COMPRESSED_RGBA_PVRTC_4BPPV1_IMG;if(i===N)return s.COMPRESSED_RGBA_PVRTC_2BPPV1_IMG}if(36196===i)return s=e.get("WEBGL_compressed_texture_etc1"),null!==s?s.COMPRESSED_RGB_ETC1_WEBGL:null;if(i===O||i===z){if(s=e.get("WEBGL_compressed_texture_etc"),null===s)return null;if(i===O)return r===ot?s.COMPRESSED_SRGB8_ETC2:s.COMPRESSED_RGB8_ETC2;if(i===z)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC:s.COMPRESSED_RGBA8_ETC2_EAC}if(i===U||i===B||i===F||i===k||i===G||i===V||i===H||i===W||i===j||i===q||i===X||i===Y||i===Z||i===J){if(s=e.get("WEBGL_compressed_texture_astc"),null===s)return null;if(i===U)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR:s.COMPRESSED_RGBA_ASTC_4x4_KHR;if(i===B)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_5x4_KHR:s.COMPRESSED_RGBA_ASTC_5x4_KHR;if(i===F)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_5x5_KHR:s.COMPRESSED_RGBA_ASTC_5x5_KHR;if(i===k)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_6x5_KHR:s.COMPRESSED_RGBA_ASTC_6x5_KHR;if(i===G)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_6x6_KHR:s.COMPRESSED_RGBA_ASTC_6x6_KHR;if(i===V)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_8x5_KHR:s.COMPRESSED_RGBA_ASTC_8x5_KHR;if(i===H)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_8x6_KHR:s.COMPRESSED_RGBA_ASTC_8x6_KHR;if(i===W)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_8x8_KHR:s.COMPRESSED_RGBA_ASTC_8x8_KHR;if(i===j)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_10x5_KHR:s.COMPRESSED_RGBA_ASTC_10x5_KHR;if(i===q)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_10x6_KHR:s.COMPRESSED_RGBA_ASTC_10x6_KHR;if(i===X)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_10x8_KHR:s.COMPRESSED_RGBA_ASTC_10x8_KHR;if(i===Y)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_10x10_KHR:s.COMPRESSED_RGBA_ASTC_10x10_KHR;if(i===Z)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_12x10_KHR:s.COMPRESSED_RGBA_ASTC_12x10_KHR;if(i===J)return r===ot?s.COMPRESSED_SRGB8_ALPHA8_ASTC_12x12_KHR:s.COMPRESSED_RGBA_ASTC_12x12_KHR}if(i===K){if(s=e.get("EXT_texture_compression_bptc"),null===s)return null;if(i===K)return r===ot?s.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT:s.COMPRESSED_RGBA_BPTC_UNORM_EXT}return i===S?n?34042:(s=e.get("WEBGL_depth_texture"),null!==s?s.UNSIGNED_INT_24_8_WEBGL:null):void 0!==t[i]?t[i]:null}}}class ks extends sn{constructor(t=[]){super(),this.isArrayCamera=!0,this.cameras=t}}class Gs extends ri{constructor(){super(),this.isGroup=!0,this.type="Group"}}const Vs={type:"move"};class Hs{constructor(){this._targetRay=null,this._grip=null,this._hand=null}getHandSpace(){return null===this._hand&&(this._hand=new Gs,this._hand.matrixAutoUpdate=!1,this._hand.visible=!1,this._hand.joints={},this._hand.inputState={pinching:!1}),this._hand}getTargetRaySpace(){return null===this._targetRay&&(this._targetRay=new Gs,this._targetRay.matrixAutoUpdate=!1,this._targetRay.visible=!1,this._targetRay.hasLinearVelocity=!1,this._targetRay.linearVelocity=new ne,this._targetRay.hasAngularVelocity=!1,this._targetRay.angularVelocity=new ne),this._targetRay}getGripSpace(){return null===this._grip&&(this._grip=new Gs,this._grip.matrixAutoUpdate=!1,this._grip.visible=!1,this._grip.hasLinearVelocity=!1,this._grip.linearVelocity=new ne,this._grip.hasAngularVelocity=!1,this._grip.angularVelocity=new ne),this._grip}dispatchEvent(t){return null!==this._targetRay&&this._targetRay.dispatchEvent(t),null!==this._grip&&this._grip.dispatchEvent(t),null!==this._hand&&this._hand.dispatchEvent(t),this}disconnect(t){return this.dispatchEvent({type:"disconnected",data:t}),null!==this._targetRay&&(this._targetRay.visible=!1),null!==this._grip&&(this._grip.visible=!1),null!==this._hand&&(this._hand.visible=!1),this}update(t,e,i){let n=null,r=null,s=null;const a=this._targetRay,o=this._grip,l=this._hand;if(t&&"visible-blurred"!==e.session.visibilityState){if(l&&t.hand){s=!0;for(const n of t.hand.values()){const t=e.getJointPose(n,i);if(void 0===l.joints[n.jointName]){const t=new Gs;t.matrixAutoUpdate=!1,t.visible=!1,l.joints[n.jointName]=t,l.add(t)}const r=l.joints[n.jointName];null!==t&&(r.matrix.fromArray(t.transform.matrix),r.matrix.decompose(r.position,r.rotation,r.scale),r.jointRadius=t.radius),r.visible=null!==t}const n=l.joints["index-finger-tip"],r=l.joints["thumb-tip"],a=n.position.distanceTo(r.position),o=.02,c=.005;l.inputState.pinching&&a>o+c?(l.inputState.pinching=!1,this.dispatchEvent({type:"pinchend",handedness:t.handedness,target:this})):!l.inputState.pinching&&a<=o-c&&(l.inputState.pinching=!0,this.dispatchEvent({type:"pinchstart",handedness:t.handedness,target:this}))}else null!==o&&t.gripSpace&&(r=e.getPose(t.gripSpace,i),null!==r&&(o.matrix.fromArray(r.transform.matrix),o.matrix.decompose(o.position,o.rotation,o.scale),r.linearVelocity?(o.hasLinearVelocity=!0,o.linearVelocity.copy(r.linearVelocity)):o.hasLinearVelocity=!1,r.angularVelocity?(o.hasAngularVelocity=!0,o.angularVelocity.copy(r.angularVelocity)):o.hasAngularVelocity=!1));null!==a&&(n=e.getPose(t.targetRaySpace,i),null===n&&null!==r&&(n=r),null!==n&&(a.matrix.fromArray(n.transform.matrix),a.matrix.decompose(a.position,a.rotation,a.scale),n.linearVelocity?(a.hasLinearVelocity=!0,a.linearVelocity.copy(n.linearVelocity)):a.hasLinearVelocity=!1,n.angularVelocity?(a.hasAngularVelocity=!0,a.angularVelocity.copy(n.angularVelocity)):a.hasAngularVelocity=!1,this.dispatchEvent(Vs)))}return null!==a&&(a.visible=null!==n),null!==o&&(o.visible=null!==r),null!==l&&(l.visible=null!==s),this}}class Ws extends Kt{constructor(t,e,i,n,r,s,a,o,l,c){if((c=void 0!==c?c:T)!==T&&c!==A)throw new Error("DepthTexture format must be either THREE.DepthFormat or THREE.DepthStencilFormat");void 0===i&&c===T&&(i=y),void 0===i&&c===A&&(i=S),super(null,n,r,s,a,o,c,i,l),this.isDepthTexture=!0,this.image={width:t,height:e},this.magFilter=void 0!==a?a:d,this.minFilter=void 0!==o?o:d,this.flipY=!1,this.generateMipmaps=!1}}class js extends mt{constructor(t,e){super();const i=this;let n=null,r=1,s=null,a="local-floor",o=null,l=null,c=null,h=null,u=null,d=null;const p=e.getContextAttributes();let m=null,f=null;const g=[],v=[],_=new sn;_.layers.enable(1),_.viewport=new $t;const M=new sn;M.layers.enable(2),M.viewport=new $t;const b=[_,M],E=new ks;E.layers.enable(1),E.layers.enable(2);let C=null,L=null;function R(t){const e=v.indexOf(t.inputSource);if(-1===e)return;const i=g[e];void 0!==i&&i.dispatchEvent({type:t.type,data:t.inputSource})}function P(){n.removeEventListener("select",R),n.removeEventListener("selectstart",R),n.removeEventListener("selectend",R),n.removeEventListener("squeeze",R),n.removeEventListener("squeezestart",R),n.removeEventListener("squeezeend",R),n.removeEventListener("end",P),n.removeEventListener("inputsourceschange",D);for(let t=0;t<g.length;t++){const e=v[t];null!==e&&(v[t]=null,g[t].disconnect(e))}C=null,L=null,t.setRenderTarget(m),u=null,h=null,c=null,n=null,f=null,U.stop(),i.isPresenting=!1,i.dispatchEvent({type:"sessionend"})}function D(t){for(let e=0;e<t.removed.length;e++){const i=t.removed[e],n=v.indexOf(i);n>=0&&(v[n]=null,g[n].dispatchEvent({type:"disconnected",data:i}))}for(let e=0;e<t.added.length;e++){const i=t.added[e];let n=v.indexOf(i);if(-1===n){for(let t=0;t<g.length;t++){if(t>=v.length){v.push(i),n=t;break}if(null===v[t]){v[t]=i,n=t;break}}if(-1===n)break}const r=g[n];r&&r.dispatchEvent({type:"connected",data:i})}}this.cameraAutoUpdate=!0,this.enabled=!1,this.isPresenting=!1,this.getController=function(t){let e=g[t];return void 0===e&&(e=new Hs,g[t]=e),e.getTargetRaySpace()},this.getControllerGrip=function(t){let e=g[t];return void 0===e&&(e=new Hs,g[t]=e),e.getGripSpace()},this.getHand=function(t){let e=g[t];return void 0===e&&(e=new Hs,g[t]=e),e.getHandSpace()},this.setFramebufferScaleFactor=function(t){r=t,!0===i.isPresenting&&console.warn("THREE.WebXRManager: Cannot change framebuffer scale while presenting.")},this.setReferenceSpaceType=function(t){a=t,!0===i.isPresenting&&console.warn("THREE.WebXRManager: Cannot change reference space type while presenting.")},this.getReferenceSpace=function(){return o||s},this.setReferenceSpace=function(t){o=t},this.getBaseLayer=function(){return null!==h?h:u},this.getBinding=function(){return c},this.getFrame=function(){return d},this.getSession=function(){return n},this.setSession=async function(l){if(n=l,null!==n){if(m=t.getRenderTarget(),n.addEventListener("select",R),n.addEventListener("selectstart",R),n.addEventListener("selectend",R),n.addEventListener("squeeze",R),n.addEventListener("squeezestart",R),n.addEventListener("squeezeend",R),n.addEventListener("end",P),n.addEventListener("inputsourceschange",D),!0!==p.xrCompatible&&await e.makeXRCompatible(),void 0===n.renderState.layers||!1===t.capabilities.isWebGL2){const i={antialias:void 0!==n.renderState.layers||p.antialias,alpha:p.alpha,depth:p.depth,stencil:p.stencil,framebufferScaleFactor:r};u=new XRWebGLLayer(n,e,i),n.updateRenderState({baseLayer:u}),f=new Qt(u.framebufferWidth,u.framebufferHeight,{format:w,type:x,encoding:t.outputEncoding,stencilBuffer:p.stencil})}else{let i=null,s=null,a=null;p.depth&&(a=p.stencil?35056:33190,i=p.stencil?A:T,s=p.stencil?S:y);const o={colorFormat:32856,depthFormat:a,scaleFactor:r};c=new XRWebGLBinding(n,e),h=c.createProjectionLayer(o),n.updateRenderState({layers:[h]}),f=new Qt(h.textureWidth,h.textureHeight,{format:w,type:x,depthTexture:new Ws(h.textureWidth,h.textureHeight,s,void 0,void 0,void 0,void 0,void 0,void 0,i),stencilBuffer:p.stencil,encoding:t.outputEncoding,samples:p.antialias?4:0});t.properties.get(f).__ignoreDepthValues=h.ignoreDepthValues}f.isXRRenderTarget=!0,this.setFoveation(1),o=null,s=await n.requestReferenceSpace(a),U.setContext(n),U.start(),i.isPresenting=!0,i.dispatchEvent({type:"sessionstart"})}};const I=new ne,N=new ne;function O(t,e){null===e?t.matrixWorld.copy(t.matrix):t.matrixWorld.multiplyMatrices(e.matrixWorld,t.matrix),t.matrixWorldInverse.copy(t.matrixWorld).invert()}this.updateCamera=function(t){if(null===n)return;E.near=M.near=_.near=t.near,E.far=M.far=_.far=t.far,C===E.near&&L===E.far||(n.updateRenderState({depthNear:E.near,depthFar:E.far}),C=E.near,L=E.far);const e=t.parent,i=E.cameras;O(E,e);for(let t=0;t<i.length;t++)O(i[t],e);E.matrixWorld.decompose(E.position,E.quaternion,E.scale),t.matrix.copy(E.matrix),t.matrix.decompose(t.position,t.quaternion,t.scale);const r=t.children;for(let t=0,e=r.length;t<e;t++)r[t].updateMatrixWorld(!0);2===i.length?function(t,e,i){I.setFromMatrixPosition(e.matrixWorld),N.setFromMatrixPosition(i.matrixWorld);const n=I.distanceTo(N),r=e.projectionMatrix.elements,s=i.projectionMatrix.elements,a=r[14]/(r[10]-1),o=r[14]/(r[10]+1),l=(r[9]+1)/r[5],c=(r[9]-1)/r[5],h=(r[8]-1)/r[0],u=(s[8]+1)/s[0],d=a*h,p=a*u,m=n/(-h+u),f=m*-h;e.matrixWorld.decompose(t.position,t.quaternion,t.scale),t.translateX(f),t.translateZ(m),t.matrixWorld.compose(t.position,t.quaternion,t.scale),t.matrixWorldInverse.copy(t.matrixWorld).invert();const g=a+m,v=o+m,x=d-f,_=p+(n-f),y=l*o/v*g,M=c*o/v*g;t.projectionMatrix.makePerspective(x,_,y,M,g,v)}(E,_,M):E.projectionMatrix.copy(_.projectionMatrix)},this.getCamera=function(){return E},this.getFoveation=function(){return null!==h?h.fixedFoveation:null!==u?u.fixedFoveation:void 0},this.setFoveation=function(t){null!==h&&(h.fixedFoveation=t),null!==u&&void 0!==u.fixedFoveation&&(u.fixedFoveation=t)};let z=null;const U=new vn;U.setAnimationLoop((function(e,i){if(l=i.getViewerPose(o||s),d=i,null!==l){const e=l.views;null!==u&&(t.setRenderTargetFramebuffer(f,u.framebuffer),t.setRenderTarget(f));let i=!1;e.length!==E.cameras.length&&(E.cameras.length=0,i=!0);for(let n=0;n<e.length;n++){const r=e[n];let s=null;if(null!==u)s=u.getViewport(r);else{const e=c.getViewSubImage(h,r);s=e.viewport,0===n&&(t.setRenderTargetTextures(f,e.colorTexture,h.ignoreDepthValues?void 0:e.depthStencilTexture),t.setRenderTarget(f))}let a=b[n];void 0===a&&(a=new sn,a.layers.enable(n),a.viewport=new $t,b[n]=a),a.matrix.fromArray(r.transform.matrix),a.projectionMatrix.fromArray(r.projectionMatrix),a.viewport.set(s.x,s.y,s.width,s.height),0===n&&E.matrix.copy(a.matrix),!0===i&&E.cameras.push(a)}}for(let t=0;t<g.length;t++){const e=v[t],n=g[t];null!==e&&void 0!==n&&n.update(e,i,o||s)}z&&z(e,i),d=null})),this.setAnimationLoop=function(t){z=t},this.dispose=function(){}}}function qs(t,e){function i(i,n){i.opacity.value=n.opacity,n.color&&i.diffuse.value.copy(n.color),n.emissive&&i.emissive.value.copy(n.emissive).multiplyScalar(n.emissiveIntensity),n.map&&(i.map.value=n.map),n.alphaMap&&(i.alphaMap.value=n.alphaMap),n.bumpMap&&(i.bumpMap.value=n.bumpMap,i.bumpScale.value=n.bumpScale,1===n.side&&(i.bumpScale.value*=-1)),n.displacementMap&&(i.displacementMap.value=n.displacementMap,i.displacementScale.value=n.displacementScale,i.displacementBias.value=n.displacementBias),n.emissiveMap&&(i.emissiveMap.value=n.emissiveMap),n.normalMap&&(i.normalMap.value=n.normalMap,i.normalScale.value.copy(n.normalScale),1===n.side&&i.normalScale.value.negate()),n.specularMap&&(i.specularMap.value=n.specularMap),n.alphaTest>0&&(i.alphaTest.value=n.alphaTest);const r=e.get(n).envMap;if(r&&(i.envMap.value=r,i.flipEnvMap.value=r.isCubeTexture&&!1===r.isRenderTargetTexture?-1:1,i.reflectivity.value=n.reflectivity,i.ior.value=n.ior,i.refractionRatio.value=n.refractionRatio),n.lightMap){i.lightMap.value=n.lightMap;const e=!0!==t.physicallyCorrectLights?Math.PI:1;i.lightMapIntensity.value=n.lightMapIntensity*e}let s,a;n.aoMap&&(i.aoMap.value=n.aoMap,i.aoMapIntensity.value=n.aoMapIntensity),n.map?s=n.map:n.specularMap?s=n.specularMap:n.displacementMap?s=n.displacementMap:n.normalMap?s=n.normalMap:n.bumpMap?s=n.bumpMap:n.roughnessMap?s=n.roughnessMap:n.metalnessMap?s=n.metalnessMap:n.alphaMap?s=n.alphaMap:n.emissiveMap?s=n.emissiveMap:n.clearcoatMap?s=n.clearcoatMap:n.clearcoatNormalMap?s=n.clearcoatNormalMap:n.clearcoatRoughnessMap?s=n.clearcoatRoughnessMap:n.iridescenceMap?s=n.iridescenceMap:n.iridescenceThicknessMap?s=n.iridescenceThicknessMap:n.specularIntensityMap?s=n.specularIntensityMap:n.specularColorMap?s=n.specularColorMap:n.transmissionMap?s=n.transmissionMap:n.thicknessMap?s=n.thicknessMap:n.sheenColorMap?s=n.sheenColorMap:n.sheenRoughnessMap&&(s=n.sheenRoughnessMap),void 0!==s&&(s.isWebGLRenderTarget&&(s=s.texture),!0===s.matrixAutoUpdate&&s.updateMatrix(),i.uvTransform.value.copy(s.matrix)),n.aoMap?a=n.aoMap:n.lightMap&&(a=n.lightMap),void 0!==a&&(a.isWebGLRenderTarget&&(a=a.texture),!0===a.matrixAutoUpdate&&a.updateMatrix(),i.uv2Transform.value.copy(a.matrix))}return{refreshFogUniforms:function(t,e){t.fogColor.value.copy(e.color),e.isFog?(t.fogNear.value=e.near,t.fogFar.value=e.far):e.isFogExp2&&(t.fogDensity.value=e.density)},refreshMaterialUniforms:function(t,n,r,s,a){n.isMeshBasicMaterial||n.isMeshLambertMaterial?i(t,n):n.isMeshToonMaterial?(i(t,n),function(t,e){e.gradientMap&&(t.gradientMap.value=e.gradientMap)}(t,n)):n.isMeshPhongMaterial?(i(t,n),function(t,e){t.specular.value.copy(e.specular),t.shininess.value=Math.max(e.shininess,1e-4)}(t,n)):n.isMeshStandardMaterial?(i(t,n),function(t,i){t.roughness.value=i.roughness,t.metalness.value=i.metalness,i.roughnessMap&&(t.roughnessMap.value=i.roughnessMap);i.metalnessMap&&(t.metalnessMap.value=i.metalnessMap);e.get(i).envMap&&(t.envMapIntensity.value=i.envMapIntensity)}(t,n),n.isMeshPhysicalMaterial&&function(t,e,i){t.ior.value=e.ior,e.sheen>0&&(t.sheenColor.value.copy(e.sheenColor).multiplyScalar(e.sheen),t.sheenRoughness.value=e.sheenRoughness,e.sheenColorMap&&(t.sheenColorMap.value=e.sheenColorMap),e.sheenRoughnessMap&&(t.sheenRoughnessMap.value=e.sheenRoughnessMap));e.clearcoat>0&&(t.clearcoat.value=e.clearcoat,t.clearcoatRoughness.value=e.clearcoatRoughness,e.clearcoatMap&&(t.clearcoatMap.value=e.clearcoatMap),e.clearcoatRoughnessMap&&(t.clearcoatRoughnessMap.value=e.clearcoatRoughnessMap),e.clearcoatNormalMap&&(t.clearcoatNormalScale.value.copy(e.clearcoatNormalScale),t.clearcoatNormalMap.value=e.clearcoatNormalMap,1===e.side&&t.clearcoatNormalScale.value.negate()));e.iridescence>0&&(t.iridescence.value=e.iridescence,t.iridescenceIOR.value=e.iridescenceIOR,t.iridescenceThicknessMinimum.value=e.iridescenceThicknessRange[0],t.iridescenceThicknessMaximum.value=e.iridescenceThicknessRange[1],e.iridescenceMap&&(t.iridescenceMap.value=e.iridescenceMap),e.iridescenceThicknessMap&&(t.iridescenceThicknessMap.value=e.iridescenceThicknessMap));e.transmission>0&&(t.transmission.value=e.transmission,t.transmissionSamplerMap.value=i.texture,t.transmissionSamplerSize.value.set(i.width,i.height),e.transmissionMap&&(t.transmissionMap.value=e.transmissionMap),t.thickness.value=e.thickness,e.thicknessMap&&(t.thicknessMap.value=e.thicknessMap),t.attenuationDistance.value=e.attenuationDistance,t.attenuationColor.value.copy(e.attenuationColor));t.specularIntensity.value=e.specularIntensity,t.specularColor.value.copy(e.specularColor),e.specularIntensityMap&&(t.specularIntensityMap.value=e.specularIntensityMap);e.specularColorMap&&(t.specularColorMap.value=e.specularColorMap)}(t,n,a)):n.isMeshMatcapMaterial?(i(t,n),function(t,e){e.matcap&&(t.matcap.value=e.matcap)}(t,n)):n.isMeshDepthMaterial?i(t,n):n.isMeshDistanceMaterial?(i(t,n),function(t,e){t.referencePosition.value.copy(e.referencePosition),t.nearDistance.value=e.nearDistance,t.farDistance.value=e.farDistance}(t,n)):n.isMeshNormalMaterial?i(t,n):n.isLineBasicMaterial?(function(t,e){t.diffuse.value.copy(e.color),t.opacity.value=e.opacity}(t,n),n.isLineDashedMaterial&&function(t,e){t.dashSize.value=e.dashSize,t.totalSize.value=e.dashSize+e.gapSize,t.scale.value=e.scale}(t,n)):n.isPointsMaterial?function(t,e,i,n){t.diffuse.value.copy(e.color),t.opacity.value=e.opacity,t.size.value=e.size*i,t.scale.value=.5*n,e.map&&(t.map.value=e.map);e.alphaMap&&(t.alphaMap.value=e.alphaMap);e.alphaTest>0&&(t.alphaTest.value=e.alphaTest);let r;e.map?r=e.map:e.alphaMap&&(r=e.alphaMap);void 0!==r&&(!0===r.matrixAutoUpdate&&r.updateMatrix(),t.uvTransform.value.copy(r.matrix))}(t,n,r,s):n.isSpriteMaterial?function(t,e){t.diffuse.value.copy(e.color),t.opacity.value=e.opacity,t.rotation.value=e.rotation,e.map&&(t.map.value=e.map);e.alphaMap&&(t.alphaMap.value=e.alphaMap);e.alphaTest>0&&(t.alphaTest.value=e.alphaTest);let i;e.map?i=e.map:e.alphaMap&&(i=e.alphaMap);void 0!==i&&(!0===i.matrixAutoUpdate&&i.updateMatrix(),t.uvTransform.value.copy(i.matrix))}(t,n):n.isShadowMaterial?(t.color.value.copy(n.color),t.opacity.value=n.opacity):n.isShaderMaterial&&(n.uniformsNeedUpdate=!1)}}}function Xs(t,e,i,n){let r={},s={},a=[];const o=i.isWebGL2?t.getParameter(35375):0;function l(t,e,i){const n=t.value;if(void 0===i[e])return i[e]="number"==typeof n?n:n.clone(),!0;if("number"==typeof n){if(i[e]!==n)return i[e]=n,!0}else{const t=i[e];if(!1===t.equals(n))return t.copy(n),!0}return!1}function c(t){const e=t.value,i={boundary:0,storage:0};return"number"==typeof e?(i.boundary=4,i.storage=4):e.isVector2?(i.boundary=8,i.storage=8):e.isVector3||e.isColor?(i.boundary=16,i.storage=12):e.isVector4?(i.boundary=16,i.storage=16):e.isMatrix3?(i.boundary=48,i.storage=48):e.isMatrix4?(i.boundary=64,i.storage=64):e.isTexture?console.warn("THREE.WebGLRenderer: Texture samplers can not be part of an uniforms group."):console.warn("THREE.WebGLRenderer: Unsupported uniform value type.",e),i}function h(e){const i=e.target;i.removeEventListener("dispose",h);const n=a.indexOf(i.__bindingPointIndex);a.splice(n,1),t.deleteBuffer(r[i.id]),delete r[i.id],delete s[i.id]}return{bind:function(t,e){const i=e.program;n.uniformBlockBinding(t,i)},update:function(i,u){let d=r[i.id];void 0===d&&(!function(t){const e=t.uniforms;let i=0;const n=16;let r=0;for(let t=0,s=e.length;t<s;t++){const s=e[t],a=c(s);if(s.__data=new Float32Array(a.storage/Float32Array.BYTES_PER_ELEMENT),s.__offset=i,t>0){r=i%n;const t=n-r;0!==r&&t-a.boundary<0&&(i+=n-r,s.__offset=i)}i+=a.storage}r=i%n,r>0&&(i+=n-r);t.__size=i,t.__cache={}}(i),d=function(e){const i=function(){for(let t=0;t<o;t++)if(-1===a.indexOf(t))return a.push(t),t;return console.error("THREE.WebGLRenderer: Maximum number of simultaneously usable uniforms groups reached."),0}();e.__bindingPointIndex=i;const n=t.createBuffer(),r=e.__size,s=e.usage;return t.bindBuffer(35345,n),t.bufferData(35345,r,s),t.bindBuffer(35345,null),t.bindBufferBase(35345,i,n),n}(i),r[i.id]=d,i.addEventListener("dispose",h));const p=u.program;n.updateUBOMapping(i,p);const m=e.render.frame;s[i.id]!==m&&(!function(e){const i=r[e.id],n=e.uniforms,s=e.__cache;t.bindBuffer(35345,i);for(let e=0,i=n.length;e<i;e++){const i=n[e];if(!0===l(i,e,s)){const e=i.value,n=i.__offset;"number"==typeof e?(i.__data[0]=e,t.bufferSubData(35345,n,i.__data)):(i.value.isMatrix3?(i.__data[0]=i.value.elements[0],i.__data[1]=i.value.elements[1],i.__data[2]=i.value.elements[2],i.__data[3]=i.value.elements[0],i.__data[4]=i.value.elements[3],i.__data[5]=i.value.elements[4],i.__data[6]=i.value.elements[5],i.__data[7]=i.value.elements[0],i.__data[8]=i.value.elements[6],i.__data[9]=i.value.elements[7],i.__data[10]=i.value.elements[8],i.__data[11]=i.value.elements[0]):e.toArray(i.__data),t.bufferSubData(35345,n,i.__data))}}t.bindBuffer(35345,null)}(i),s[i.id]=m)},dispose:function(){for(const e in r)t.deleteBuffer(r[e]);a=[],r={},s={}}}}function Ys(t={}){this.isWebGLRenderer=!0;const e=void 0!==t.canvas?t.canvas:function(){const t=Nt("canvas");return t.style.display="block",t}(),i=void 0!==t.context?t.context:null,n=void 0===t.depth||t.depth,r=void 0===t.stencil||t.stencil,s=void 0!==t.antialias&&t.antialias,a=void 0===t.premultipliedAlpha||t.premultipliedAlpha,o=void 0!==t.preserveDrawingBuffer&&t.preserveDrawingBuffer,l=void 0!==t.powerPreference?t.powerPreference:"default",c=void 0!==t.failIfMajorPerformanceCaveat&&t.failIfMajorPerformanceCaveat;let h;h=null!==i?i.getContextAttributes().alpha:void 0!==t.alpha&&t.alpha;let u=null,d=null;const p=[],m=[];this.domElement=e,this.debug={checkShaderErrors:!0},this.autoClear=!0,this.autoClearColor=!0,this.autoClearDepth=!0,this.autoClearStencil=!0,this.sortObjects=!0,this.clippingPlanes=[],this.localClippingEnabled=!1,this.outputEncoding=at,this.physicallyCorrectLights=!1,this.toneMapping=0,this.toneMappingExposure=1,Object.defineProperties(this,{gammaFactor:{get:function(){return console.warn("THREE.WebGLRenderer: .gammaFactor has been removed."),2},set:function(){console.warn("THREE.WebGLRenderer: .gammaFactor has been removed.")}}});const f=this;let g=!1,_=0,y=0,S=null,T=-1,A=null;const E=new $t,C=new $t;let L=null,R=e.width,P=e.height,D=1,I=null,N=null;const O=new $t(0,0,R,P),z=new $t(0,0,R,P);let U=!1;const B=new gn;let F=!1,k=!1,G=null;const V=new Ie,H=new Lt,W=new ne,j={background:null,fog:null,environment:null,overrideMaterial:null,isScene:!0};function q(){return null===S?D:1}let X,Y,Z,J,K,$,Q,tt,et,it,nt,rt,st,ot,lt,ct,ht,ut,dt,pt,mt,ft,gt,vt,xt=i;function _t(t,i){for(let n=0;n<t.length;n++){const r=t[n],s=e.getContext(r,i);if(null!==s)return s}return null}try{const t={alpha:!0,depth:n,stencil:r,antialias:s,premultipliedAlpha:a,preserveDrawingBuffer:o,powerPreference:l,failIfMajorPerformanceCaveat:c};if("setAttribute"in e&&e.setAttribute("data-engine","three.js r146"),e.addEventListener("webglcontextlost",bt,!1),e.addEventListener("webglcontextrestored",St,!1),e.addEventListener("webglcontextcreationerror",wt,!1),null===xt){const e=["webgl2","webgl","experimental-webgl"];if(!0===f.isWebGL1Renderer&&e.shift(),xt=_t(e,t),null===xt)throw _t(e)?new Error("Error creating WebGL context with your selected attributes."):new Error("Error creating WebGL context.")}void 0===xt.getShaderPrecisionFormat&&(xt.getShaderPrecisionFormat=function(){return{rangeMin:1,rangeMax:1,precision:1}})}catch(t){throw console.error("THREE.WebGLRenderer: "+t.message),t}function yt(){X=new jn(xt),Y=new An(xt,X,t),X.init(Y),ft=new Fs(xt,X,Y),Z=new Us(xt,X,Y),J=new Yn,K=new Ss,$=new Bs(xt,X,Z,K,Y,ft,J),Q=new Cn(f),tt=new Wn(f),et=new xn(xt,Y),gt=new wn(xt,X,et,Y),it=new qn(xt,et,J,gt),nt=new $n(xt,it,et,J),dt=new Kn(xt,Y,$),ct=new En(K),rt=new bs(f,Q,tt,X,Y,gt,ct),st=new qs(f,K),ot=new Es,lt=new Is(X,Y),ut=new Sn(f,Q,tt,Z,nt,h,a),ht=new zs(f,nt,Y),vt=new Xs(xt,J,Y,Z),pt=new Tn(xt,X,J,Y),mt=new Xn(xt,X,J,Y),J.programs=rt.programs,f.capabilities=Y,f.extensions=X,f.properties=K,f.renderLists=ot,f.shadowMap=ht,f.state=Z,f.info=J}yt();const Mt=new js(f,xt);function bt(t){t.preventDefault(),console.log("THREE.WebGLRenderer: Context Lost."),g=!0}function St(){console.log("THREE.WebGLRenderer: Context Restored."),g=!1;const t=J.autoReset,e=ht.enabled,i=ht.autoUpdate,n=ht.needsUpdate,r=ht.type;yt(),J.autoReset=t,ht.enabled=e,ht.autoUpdate=i,ht.needsUpdate=n,ht.type=r}function wt(t){console.error("THREE.WebGLRenderer: A WebGL context could not be created. Reason: ",t.statusMessage)}function At(t){const e=t.target;e.removeEventListener("dispose",At),function(t){(function(t){const e=K.get(t).programs;void 0!==e&&(e.forEach((function(t){rt.releaseProgram(t)})),t.isShaderMaterial&&rt.releaseShaderCache(t))})(t),K.remove(t)}(e)}this.xr=Mt,this.getContext=function(){return xt},this.getContextAttributes=function(){return xt.getContextAttributes()},this.forceContextLoss=function(){const t=X.get("WEBGL_lose_context");t&&t.loseContext()},this.forceContextRestore=function(){const t=X.get("WEBGL_lose_context");t&&t.restoreContext()},this.getPixelRatio=function(){return D},this.setPixelRatio=function(t){void 0!==t&&(D=t,this.setSize(R,P,!1))},this.getSize=function(t){return t.set(R,P)},this.setSize=function(t,i,n){Mt.isPresenting?console.warn("THREE.WebGLRenderer: Can't change size while VR device is presenting."):(R=t,P=i,e.width=Math.floor(t*D),e.height=Math.floor(i*D),!1!==n&&(e.style.width=t+"px",e.style.height=i+"px"),this.setViewport(0,0,t,i))},this.getDrawingBufferSize=function(t){return t.set(R*D,P*D).floor()},this.setDrawingBufferSize=function(t,i,n){R=t,P=i,D=n,e.width=Math.floor(t*n),e.height=Math.floor(i*n),this.setViewport(0,0,t,i)},this.getCurrentViewport=function(t){return t.copy(E)},this.getViewport=function(t){return t.copy(O)},this.setViewport=function(t,e,i,n){t.isVector4?O.set(t.x,t.y,t.z,t.w):O.set(t,e,i,n),Z.viewport(E.copy(O).multiplyScalar(D).floor())},this.getScissor=function(t){return t.copy(z)},this.setScissor=function(t,e,i,n){t.isVector4?z.set(t.x,t.y,t.z,t.w):z.set(t,e,i,n),Z.scissor(C.copy(z).multiplyScalar(D).floor())},this.getScissorTest=function(){return U},this.setScissorTest=function(t){Z.setScissorTest(U=t)},this.setOpaqueSort=function(t){I=t},this.setTransparentSort=function(t){N=t},this.getClearColor=function(t){return t.copy(ut.getClearColor())},this.setClearColor=function(){ut.setClearColor.apply(ut,arguments)},this.getClearAlpha=function(){return ut.getClearAlpha()},this.setClearAlpha=function(){ut.setClearAlpha.apply(ut,arguments)},this.clear=function(t=!0,e=!0,i=!0){let n=0;t&&(n|=16384),e&&(n|=256),i&&(n|=1024),xt.clear(n)},this.clearColor=function(){this.clear(!0,!1,!1)},this.clearDepth=function(){this.clear(!1,!0,!1)},this.clearStencil=function(){this.clear(!1,!1,!0)},this.dispose=function(){e.removeEventListener("webglcontextlost",bt,!1),e.removeEventListener("webglcontextrestored",St,!1),e.removeEventListener("webglcontextcreationerror",wt,!1),ot.dispose(),lt.dispose(),K.dispose(),Q.dispose(),tt.dispose(),nt.dispose(),gt.dispose(),vt.dispose(),rt.dispose(),Mt.dispose(),Mt.removeEventListener("sessionstart",Ct),Mt.removeEventListener("sessionend",Rt),G&&(G.dispose(),G=null),Pt.stop()},this.renderBufferDirect=function(t,e,i,n,r,s){null===e&&(e=j);const a=r.isMesh&&r.matrixWorld.determinant()<0,o=function(t,e,i,n,r){!0!==e.isScene&&(e=j);$.resetTextureUnits();const s=e.fog,a=n.isMeshStandardMaterial?e.environment:null,o=null===S?f.outputEncoding:!0===S.isXRRenderTarget?S.texture.encoding:at,l=(n.isMeshStandardMaterial?tt:Q).get(n.envMap||a),c=!0===n.vertexColors&&!!i.attributes.color&&4===i.attributes.color.itemSize,h=!!n.normalMap&&!!i.attributes.tangent,u=!!i.morphAttributes.position,p=!!i.morphAttributes.normal,m=!!i.morphAttributes.color,g=n.toneMapped?f.toneMapping:0,v=i.morphAttributes.position||i.morphAttributes.normal||i.morphAttributes.color,x=void 0!==v?v.length:0,_=K.get(n),y=d.state.lights;if(!0===F&&(!0===k||t!==A)){const e=t===A&&n.id===T;ct.setState(n,t,e)}let M=!1;n.version===_.__version?_.needsLights&&_.lightsStateVersion!==y.state.version||_.outputEncoding!==o||r.isInstancedMesh&&!1===_.instancing?M=!0:r.isInstancedMesh||!0!==_.instancing?r.isSkinnedMesh&&!1===_.skinning?M=!0:r.isSkinnedMesh||!0!==_.skinning?_.envMap!==l||!0===n.fog&&_.fog!==s?M=!0:void 0===_.numClippingPlanes||_.numClippingPlanes===ct.numPlanes&&_.numIntersection===ct.numIntersection?(_.vertexAlphas!==c||_.vertexTangents!==h||_.morphTargets!==u||_.morphNormals!==p||_.morphColors!==m||_.toneMapping!==g||!0===Y.isWebGL2&&_.morphTargetsCount!==x)&&(M=!0):M=!0:M=!0:M=!0:(M=!0,_.__version=n.version);let b=_.currentProgram;!0===M&&(b=Ut(n,e,r));let w=!1,E=!1,C=!1;const L=b.getUniforms(),R=_.uniforms;Z.useProgram(b.program)&&(w=!0,E=!0,C=!0);n.id!==T&&(T=n.id,E=!0);if(w||A!==t){if(L.setValue(xt,"projectionMatrix",t.projectionMatrix),Y.logarithmicDepthBuffer&&L.setValue(xt,"logDepthBufFC",2/(Math.log(t.far+1)/Math.LN2)),A!==t&&(A=t,E=!0,C=!0),n.isShaderMaterial||n.isMeshPhongMaterial||n.isMeshToonMaterial||n.isMeshStandardMaterial||n.envMap){const e=L.map.cameraPosition;void 0!==e&&e.setValue(xt,W.setFromMatrixPosition(t.matrixWorld))}(n.isMeshPhongMaterial||n.isMeshToonMaterial||n.isMeshLambertMaterial||n.isMeshBasicMaterial||n.isMeshStandardMaterial||n.isShaderMaterial)&&L.setValue(xt,"isOrthographic",!0===t.isOrthographicCamera),(n.isMeshPhongMaterial||n.isMeshToonMaterial||n.isMeshLambertMaterial||n.isMeshBasicMaterial||n.isMeshStandardMaterial||n.isShaderMaterial||n.isShadowMaterial||r.isSkinnedMesh)&&L.setValue(xt,"viewMatrix",t.matrixWorldInverse)}if(r.isSkinnedMesh){L.setOptional(xt,r,"bindMatrix"),L.setOptional(xt,r,"bindMatrixInverse");const t=r.skeleton;t&&(Y.floatVertexTextures?(null===t.boneTexture&&t.computeBoneTexture(),L.setValue(xt,"boneTexture",t.boneTexture,$),L.setValue(xt,"boneTextureSize",t.boneTextureSize)):console.warn("THREE.WebGLRenderer: SkinnedMesh can only be used with WebGL 2. With WebGL 1 OES_texture_float and vertex textures support is required."))}const I=i.morphAttributes;(void 0!==I.position||void 0!==I.normal||void 0!==I.color&&!0===Y.isWebGL2)&&dt.update(r,i,n,b);(E||_.receiveShadow!==r.receiveShadow)&&(_.receiveShadow=r.receiveShadow,L.setValue(xt,"receiveShadow",r.receiveShadow));n.isMeshGouraudMaterial&&null!==n.envMap&&(R.envMap.value=l,R.flipEnvMap.value=l.isCubeTexture&&!1===l.isRenderTargetTexture?-1:1);E&&(L.setValue(xt,"toneMappingExposure",f.toneMappingExposure),_.needsLights&&(O=C,(N=R).ambientLightColor.needsUpdate=O,N.lightProbe.needsUpdate=O,N.directionalLights.needsUpdate=O,N.directionalLightShadows.needsUpdate=O,N.pointLights.needsUpdate=O,N.pointLightShadows.needsUpdate=O,N.spotLights.needsUpdate=O,N.spotLightShadows.needsUpdate=O,N.rectAreaLights.needsUpdate=O,N.hemisphereLights.needsUpdate=O),s&&!0===n.fog&&st.refreshFogUniforms(R,s),st.refreshMaterialUniforms(R,n,D,P,G),is.upload(xt,_.uniformsList,R,$));var N,O;n.isShaderMaterial&&!0===n.uniformsNeedUpdate&&(is.upload(xt,_.uniformsList,R,$),n.uniformsNeedUpdate=!1);n.isSpriteMaterial&&L.setValue(xt,"center",r.center);if(L.setValue(xt,"modelViewMatrix",r.modelViewMatrix),L.setValue(xt,"normalMatrix",r.normalMatrix),L.setValue(xt,"modelMatrix",r.matrixWorld),n.isShaderMaterial||n.isRawShaderMaterial){const t=n.uniformsGroups;for(let e=0,i=t.length;e<i;e++)if(Y.isWebGL2){const i=t[e];vt.update(i,b),vt.bind(i,b)}else console.warn("THREE.WebGLRenderer: Uniform Buffer Objects can only be used with WebGL 2.")}return b}(t,e,i,n,r);Z.setMaterial(n,a);let l=i.index;const c=i.attributes.position;if(null===l){if(void 0===c||0===c.count)return}else if(0===l.count)return;let h,u=1;!0===n.wireframe&&(l=it.getWireframeAttribute(i),u=2),gt.setup(r,n,o,i,l);let p=pt;null!==l&&(h=et.get(l),p=mt,p.setIndex(h));const m=null!==l?l.count:c.count,g=i.drawRange.start*u,v=i.drawRange.count*u,x=null!==s?s.start*u:0,_=null!==s?s.count*u:1/0,y=Math.max(g,x),M=Math.min(m,g+v,x+_)-1,b=Math.max(0,M-y+1);if(0!==b){if(r.isMesh)!0===n.wireframe?(Z.setLineWidth(n.wireframeLinewidth*q()),p.setMode(1)):p.setMode(4);else if(r.isLine){let t=n.linewidth;void 0===t&&(t=1),Z.setLineWidth(t*q()),r.isLineSegments?p.setMode(1):r.isLineLoop?p.setMode(2):p.setMode(3)}else r.isPoints?p.setMode(0):r.isSprite&&p.setMode(4);if(r.isInstancedMesh)p.renderInstances(y,b,r.count);else if(i.isInstancedBufferGeometry){const t=Math.min(i.instanceCount,i._maxInstanceCount);p.renderInstances(y,b,t)}else p.render(y,b)}},this.compile=function(t,e){function i(t,e,i){!0===t.transparent&&2===t.side?(t.side=1,t.needsUpdate=!0,Ut(t,e,i),t.side=0,t.needsUpdate=!0,Ut(t,e,i),t.side=2):Ut(t,e,i)}d=lt.get(t),d.init(),m.push(d),t.traverseVisible((function(t){t.isLight&&t.layers.test(e.layers)&&(d.pushLight(t),t.castShadow&&d.pushShadow(t))})),d.setupLights(f.physicallyCorrectLights),t.traverse((function(e){const n=e.material;if(n)if(Array.isArray(n))for(let r=0;r<n.length;r++){i(n[r],t,e)}else i(n,t,e)})),m.pop(),d=null};let Et=null;function Ct(){Pt.stop()}function Rt(){Pt.start()}const Pt=new vn;function Dt(t,e,i,n){if(!1===t.visible)return;if(t.layers.test(e.layers))if(t.isGroup)i=t.renderOrder;else if(t.isLOD)!0===t.autoUpdate&&t.update(e);else if(t.isLight)d.pushLight(t),t.castShadow&&d.pushShadow(t);else if(t.isSprite){if(!t.frustumCulled||B.intersectsSprite(t)){n&&W.setFromMatrixPosition(t.matrixWorld).applyMatrix4(V);const e=nt.update(t),r=t.material;r.visible&&u.push(t,e,r,i,W.z,null)}}else if((t.isMesh||t.isLine||t.isPoints)&&(t.isSkinnedMesh&&t.skeleton.frame!==J.render.frame&&(t.skeleton.update(),t.skeleton.frame=J.render.frame),!t.frustumCulled||B.intersectsObject(t))){n&&W.setFromMatrixPosition(t.matrixWorld).applyMatrix4(V);const e=nt.update(t),r=t.material;if(Array.isArray(r)){const n=e.groups;for(let s=0,a=n.length;s<a;s++){const a=n[s],o=r[a.materialIndex];o&&o.visible&&u.push(t,e,o,i,W.z,a)}}else r.visible&&u.push(t,e,r,i,W.z,null)}const r=t.children;for(let t=0,s=r.length;t<s;t++)Dt(r[t],e,i,n)}function It(t,e,i,n){const r=t.opaque,a=t.transmissive,o=t.transparent;d.setupLightsView(i),a.length>0&&function(t,e,i){const n=Y.isWebGL2;null===G&&(G=new Qt(1,1,{generateMipmaps:!0,type:X.has("EXT_color_buffer_half_float")?b:x,minFilter:v,samples:n&&!0===s?4:0}));f.getDrawingBufferSize(H),n?G.setSize(H.x,H.y):G.setSize(Tt(H.x),Tt(H.y));const r=f.getRenderTarget();f.setRenderTarget(G),f.clear();const a=f.toneMapping;f.toneMapping=0,Ot(t,e,i),f.toneMapping=a,$.updateMultisampleRenderTarget(G),$.updateRenderTargetMipmap(G),f.setRenderTarget(r)}(r,e,i),n&&Z.viewport(E.copy(n)),r.length>0&&Ot(r,e,i),a.length>0&&Ot(a,e,i),o.length>0&&Ot(o,e,i),Z.buffers.depth.setTest(!0),Z.buffers.depth.setMask(!0),Z.buffers.color.setMask(!0),Z.setPolygonOffset(!1)}function Ot(t,e,i){const n=!0===e.isScene?e.overrideMaterial:null;for(let r=0,s=t.length;r<s;r++){const s=t[r],a=s.object,o=s.geometry,l=null===n?s.material:n,c=s.group;a.layers.test(i.layers)&&zt(a,e,i,o,l,c)}}function zt(t,e,i,n,r,s){t.onBeforeRender(f,e,i,n,r,s),t.modelViewMatrix.multiplyMatrices(i.matrixWorldInverse,t.matrixWorld),t.normalMatrix.getNormalMatrix(t.modelViewMatrix),r.onBeforeRender(f,e,i,n,t,s),!0===r.transparent&&2===r.side?(r.side=1,r.needsUpdate=!0,f.renderBufferDirect(i,e,n,r,t,s),r.side=0,r.needsUpdate=!0,f.renderBufferDirect(i,e,n,r,t,s),r.side=2):f.renderBufferDirect(i,e,n,r,t,s),t.onAfterRender(f,e,i,n,r,s)}function Ut(t,e,i){!0!==e.isScene&&(e=j);const n=K.get(t),r=d.state.lights,s=d.state.shadowsArray,a=r.state.version,o=rt.getParameters(t,r.state,s,e,i),l=rt.getProgramCacheKey(o);let c=n.programs;n.environment=t.isMeshStandardMaterial?e.environment:null,n.fog=e.fog,n.envMap=(t.isMeshStandardMaterial?tt:Q).get(t.envMap||n.environment),void 0===c&&(t.addEventListener("dispose",At),c=new Map,n.programs=c);let h=c.get(l);if(void 0!==h){if(n.currentProgram===h&&n.lightsStateVersion===a)return Bt(t,o),h}else o.uniforms=rt.getUniforms(t),t.onBuild(i,o,f),t.onBeforeCompile(o,f),h=rt.acquireProgram(o,l),c.set(l,h),n.uniforms=o.uniforms;const u=n.uniforms;(t.isShaderMaterial||t.isRawShaderMaterial)&&!0!==t.clipping||(u.clippingPlanes=ct.uniform),Bt(t,o),n.needsLights=function(t){return t.isMeshLambertMaterial||t.isMeshToonMaterial||t.isMeshPhongMaterial||t.isMeshStandardMaterial||t.isShadowMaterial||t.isShaderMaterial&&!0===t.lights}(t),n.lightsStateVersion=a,n.needsLights&&(u.ambientLightColor.value=r.state.ambient,u.lightProbe.value=r.state.probe,u.directionalLights.value=r.state.directional,u.directionalLightShadows.value=r.state.directionalShadow,u.spotLights.value=r.state.spot,u.spotLightShadows.value=r.state.spotShadow,u.rectAreaLights.value=r.state.rectArea,u.ltc_1.value=r.state.rectAreaLTC1,u.ltc_2.value=r.state.rectAreaLTC2,u.pointLights.value=r.state.point,u.pointLightShadows.value=r.state.pointShadow,u.hemisphereLights.value=r.state.hemi,u.directionalShadowMap.value=r.state.directionalShadowMap,u.directionalShadowMatrix.value=r.state.directionalShadowMatrix,u.spotShadowMap.value=r.state.spotShadowMap,u.spotLightMatrix.value=r.state.spotLightMatrix,u.spotLightMap.value=r.state.spotLightMap,u.pointShadowMap.value=r.state.pointShadowMap,u.pointShadowMatrix.value=r.state.pointShadowMatrix);const p=h.getUniforms(),m=is.seqWithValue(p.seq,u);return n.currentProgram=h,n.uniformsList=m,h}function Bt(t,e){const i=K.get(t);i.outputEncoding=e.outputEncoding,i.instancing=e.instancing,i.skinning=e.skinning,i.morphTargets=e.morphTargets,i.morphNormals=e.morphNormals,i.morphColors=e.morphColors,i.morphTargetsCount=e.morphTargetsCount,i.numClippingPlanes=e.numClippingPlanes,i.numIntersection=e.numClipIntersection,i.vertexAlphas=e.vertexAlphas,i.vertexTangents=e.vertexTangents,i.toneMapping=e.toneMapping}Pt.setAnimationLoop((function(t){Et&&Et(t)})),"undefined"!=typeof self&&Pt.setContext(self),this.setAnimationLoop=function(t){Et=t,Mt.setAnimationLoop(t),null===t?Pt.stop():Pt.start()},Mt.addEventListener("sessionstart",Ct),Mt.addEventListener("sessionend",Rt),this.render=function(t,e){if(void 0!==e&&!0!==e.isCamera)return void console.error("THREE.WebGLRenderer.render: camera is not an instance of THREE.Camera.");if(!0===g)return;!0===t.matrixWorldAutoUpdate&&t.updateMatrixWorld(),null===e.parent&&!0===e.matrixWorldAutoUpdate&&e.updateMatrixWorld(),!0===Mt.enabled&&!0===Mt.isPresenting&&(!0===Mt.cameraAutoUpdate&&Mt.updateCamera(e),e=Mt.getCamera()),!0===t.isScene&&t.onBeforeRender(f,t,e,S),d=lt.get(t,m.length),d.init(),m.push(d),V.multiplyMatrices(e.projectionMatrix,e.matrixWorldInverse),B.setFromProjectionMatrix(V),k=this.localClippingEnabled,F=ct.init(this.clippingPlanes,k,e),u=ot.get(t,p.length),u.init(),p.push(u),Dt(t,e,0,f.sortObjects),u.finish(),!0===f.sortObjects&&u.sort(I,N),!0===F&&ct.beginShadows();const i=d.state.shadowsArray;if(ht.render(i,t,e),!0===F&&ct.endShadows(),!0===this.info.autoReset&&this.info.reset(),ut.render(u,t),d.setupLights(f.physicallyCorrectLights),e.isArrayCamera){const i=e.cameras;for(let e=0,n=i.length;e<n;e++){const n=i[e];It(u,t,n,n.viewport)}}else It(u,t,e);null!==S&&($.updateMultisampleRenderTarget(S),$.updateRenderTargetMipmap(S)),!0===t.isScene&&t.onAfterRender(f,t,e),gt.resetDefaultState(),T=-1,A=null,m.pop(),d=m.length>0?m[m.length-1]:null,p.pop(),u=p.length>0?p[p.length-1]:null},this.getActiveCubeFace=function(){return _},this.getActiveMipmapLevel=function(){return y},this.getRenderTarget=function(){return S},this.setRenderTargetTextures=function(t,e,i){K.get(t.texture).__webglTexture=e,K.get(t.depthTexture).__webglTexture=i;const n=K.get(t);n.__hasExternalTextures=!0,n.__hasExternalTextures&&(n.__autoAllocateDepthBuffer=void 0===i,n.__autoAllocateDepthBuffer||!0===X.has("WEBGL_multisampled_render_to_texture")&&(console.warn("THREE.WebGLRenderer: Render-to-texture extension was disabled because an external texture was provided"),n.__useRenderToTexture=!1))},this.setRenderTargetFramebuffer=function(t,e){const i=K.get(t);i.__webglFramebuffer=e,i.__useDefaultFramebuffer=void 0===e},this.setRenderTarget=function(t,e=0,i=0){S=t,_=e,y=i;let n=!0,r=null,s=!1,a=!1;if(t){const i=K.get(t);void 0!==i.__useDefaultFramebuffer?(Z.bindFramebuffer(36160,null),n=!1):void 0===i.__webglFramebuffer?$.setupRenderTarget(t):i.__hasExternalTextures&&$.rebindTextures(t,K.get(t.texture).__webglTexture,K.get(t.depthTexture).__webglTexture);const o=t.texture;(o.isData3DTexture||o.isDataArrayTexture||o.isCompressedArrayTexture)&&(a=!0);const l=K.get(t).__webglFramebuffer;t.isWebGLCubeRenderTarget?(r=l[e],s=!0):r=Y.isWebGL2&&t.samples>0&&!1===$.useMultisampledRTT(t)?K.get(t).__webglMultisampledFramebuffer:l,E.copy(t.viewport),C.copy(t.scissor),L=t.scissorTest}else E.copy(O).multiplyScalar(D).floor(),C.copy(z).multiplyScalar(D).floor(),L=U;if(Z.bindFramebuffer(36160,r)&&Y.drawBuffers&&n&&Z.drawBuffers(t,r),Z.viewport(E),Z.scissor(C),Z.setScissorTest(L),s){const n=K.get(t.texture);xt.framebufferTexture2D(36160,36064,34069+e,n.__webglTexture,i)}else if(a){const n=K.get(t.texture),r=e||0;xt.framebufferTextureLayer(36160,36064,n.__webglTexture,i||0,r)}T=-1},this.readRenderTargetPixels=function(t,e,i,n,r,s,a){if(!t||!t.isWebGLRenderTarget)return void console.error("THREE.WebGLRenderer.readRenderTargetPixels: renderTarget is not THREE.WebGLRenderTarget.");let o=K.get(t).__webglFramebuffer;if(t.isWebGLCubeRenderTarget&&void 0!==a&&(o=o[a]),o){Z.bindFramebuffer(36160,o);try{const a=t.texture,o=a.format,l=a.type;if(o!==w&&ft.convert(o)!==xt.getParameter(35739))return void console.error("THREE.WebGLRenderer.readRenderTargetPixels: renderTarget is not in RGBA or implementation defined format.");const c=l===b&&(X.has("EXT_color_buffer_half_float")||Y.isWebGL2&&X.has("EXT_color_buffer_float"));if(!(l===x||ft.convert(l)===xt.getParameter(35738)||l===M&&(Y.isWebGL2||X.has("OES_texture_float")||X.has("WEBGL_color_buffer_float"))||c))return void console.error("THREE.WebGLRenderer.readRenderTargetPixels: renderTarget is not in UnsignedByteType or implementation defined type.");e>=0&&e<=t.width-n&&i>=0&&i<=t.height-r&&xt.readPixels(e,i,n,r,ft.convert(o),ft.convert(l),s)}finally{const t=null!==S?K.get(S).__webglFramebuffer:null;Z.bindFramebuffer(36160,t)}}},this.copyFramebufferToTexture=function(t,e,i=0){const n=Math.pow(2,-i),r=Math.floor(e.image.width*n),s=Math.floor(e.image.height*n);$.setTexture2D(e,0),xt.copyTexSubImage2D(3553,i,0,0,t.x,t.y,r,s),Z.unbindTexture()},this.copyTextureToTexture=function(t,e,i,n=0){const r=e.image.width,s=e.image.height,a=ft.convert(i.format),o=ft.convert(i.type);$.setTexture2D(i,0),xt.pixelStorei(37440,i.flipY),xt.pixelStorei(37441,i.premultiplyAlpha),xt.pixelStorei(3317,i.unpackAlignment),e.isDataTexture?xt.texSubImage2D(3553,n,t.x,t.y,r,s,a,o,e.image.data):e.isCompressedTexture?xt.compressedTexSubImage2D(3553,n,t.x,t.y,e.mipmaps[0].width,e.mipmaps[0].height,a,e.mipmaps[0].data):xt.texSubImage2D(3553,n,t.x,t.y,a,o,e.image),0===n&&i.generateMipmaps&&xt.generateMipmap(3553),Z.unbindTexture()},this.copyTextureToTexture3D=function(t,e,i,n,r=0){if(f.isWebGL1Renderer)return void console.warn("THREE.WebGLRenderer.copyTextureToTexture3D: can only be used with WebGL2.");const s=t.max.x-t.min.x+1,a=t.max.y-t.min.y+1,o=t.max.z-t.min.z+1,l=ft.convert(n.format),c=ft.convert(n.type);let h;if(n.isData3DTexture)$.setTexture3D(n,0),h=32879;else{if(!n.isDataArrayTexture)return void console.warn("THREE.WebGLRenderer.copyTextureToTexture3D: only supports THREE.DataTexture3D and THREE.DataTexture2DArray.");$.setTexture2DArray(n,0),h=35866}xt.pixelStorei(37440,n.flipY),xt.pixelStorei(37441,n.premultiplyAlpha),xt.pixelStorei(3317,n.unpackAlignment);const u=xt.getParameter(3314),d=xt.getParameter(32878),p=xt.getParameter(3316),m=xt.getParameter(3315),g=xt.getParameter(32877),v=i.isCompressedTexture?i.mipmaps[0]:i.image;xt.pixelStorei(3314,v.width),xt.pixelStorei(32878,v.height),xt.pixelStorei(3316,t.min.x),xt.pixelStorei(3315,t.min.y),xt.pixelStorei(32877,t.min.z),i.isDataTexture||i.isData3DTexture?xt.texSubImage3D(h,r,e.x,e.y,e.z,s,a,o,l,c,v.data):i.isCompressedArrayTexture?(console.warn("THREE.WebGLRenderer.copyTextureToTexture3D: untested support for compressed srcTexture."),xt.compressedTexSubImage3D(h,r,e.x,e.y,e.z,s,a,o,l,v.data)):xt.texSubImage3D(h,r,e.x,e.y,e.z,s,a,o,l,c,v),xt.pixelStorei(3314,u),xt.pixelStorei(32878,d),xt.pixelStorei(3316,p),xt.pixelStorei(3315,m),xt.pixelStorei(32877,g),0===r&&n.generateMipmaps&&xt.generateMipmap(h),Z.unbindTexture()},this.initTexture=function(t){t.isCubeTexture?$.setTextureCube(t,0):t.isData3DTexture?$.setTexture3D(t,0):t.isDataArrayTexture||t.isCompressedArrayTexture?$.setTexture2DArray(t,0):$.setTexture2D(t,0),Z.unbindTexture()},this.resetState=function(){_=0,y=0,S=null,Z.reset(),gt.reset()},"undefined"!=typeof __THREE_DEVTOOLS__&&__THREE_DEVTOOLS__.dispatchEvent(new CustomEvent("observe",{detail:this}))}class Zs extends Ys{}Zs.prototype.isWebGL1Renderer=!0;class Js{constructor(t,e=25e-5){this.isFogExp2=!0,this.name="",this.color=new jt(t),this.density=e}clone(){return new Js(this.color,this.density)}toJSON(){return{type:"FogExp2",color:this.color.getHex(),density:this.density}}}class Ks{constructor(t,e=1,i=1e3){this.isFog=!0,this.name="",this.color=new jt(t),this.near=e,this.far=i}clone(){return new Ks(this.color,this.near,this.far)}toJSON(){return{type:"Fog",color:this.color.getHex(),near:this.near,far:this.far}}}class $s extends ri{constructor(){super(),this.isScene=!0,this.type="Scene",this.background=null,this.environment=null,this.fog=null,this.backgroundBlurriness=0,this.overrideMaterial=null,"undefined"!=typeof __THREE_DEVTOOLS__&&__THREE_DEVTOOLS__.dispatchEvent(new CustomEvent("observe",{detail:this}))}copy(t,e){return super.copy(t,e),null!==t.background&&(this.background=t.background.clone()),null!==t.environment&&(this.environment=t.environment.clone()),null!==t.fog&&(this.fog=t.fog.clone()),this.backgroundBlurriness=t.backgroundBlurriness,null!==t.overrideMaterial&&(this.overrideMaterial=t.overrideMaterial.clone()),this.matrixAutoUpdate=t.matrixAutoUpdate,this}toJSON(t){const e=super.toJSON(t);return null!==this.fog&&(e.object.fog=this.fog.toJSON()),this.backgroundBlurriness>0&&(e.backgroundBlurriness=this.backgroundBlurriness),e}get autoUpdate(){return console.warn("THREE.Scene: autoUpdate was renamed to matrixWorldAutoUpdate in r144."),this.matrixWorldAutoUpdate}set autoUpdate(t){console.warn("THREE.Scene: autoUpdate was renamed to matrixWorldAutoUpdate in r144."),this.matrixWorldAutoUpdate=t}}class Qs{constructor(t,e){this.isInterleavedBuffer=!0,this.array=t,this.stride=e,this.count=void 0!==t?t.length/e:0,this.usage=ut,this.updateRange={offset:0,count:-1},this.version=0,this.uuid=_t()}onUploadCallback(){}set needsUpdate(t){!0===t&&this.version++}setUsage(t){return this.usage=t,this}copy(t){return this.array=new t.array.constructor(t.array),this.count=t.count,this.stride=t.stride,this.usage=t.usage,this}copyAt(t,e,i){t*=this.stride,i*=e.stride;for(let n=0,r=this.stride;n<r;n++)this.array[t+n]=e.array[i+n];return this}set(t,e=0){return this.array.set(t,e),this}clone(t){void 0===t.arrayBuffers&&(t.arrayBuffers={}),void 0===this.array.buffer._uuid&&(this.array.buffer._uuid=_t()),void 0===t.arrayBuffers[this.array.buffer._uuid]&&(t.arrayBuffers[this.array.buffer._uuid]=this.array.slice(0).buffer);const e=new this.array.constructor(t.arrayBuffers[this.array.buffer._uuid]),i=new this.constructor(e,this.stride);return i.setUsage(this.usage),i}onUpload(t){return this.onUploadCallback=t,this}toJSON(t){return void 0===t.arrayBuffers&&(t.arrayBuffers={}),void 0===this.array.buffer._uuid&&(this.array.buffer._uuid=_t()),void 0===t.arrayBuffers[this.array.buffer._uuid]&&(t.arrayBuffers[this.array.buffer._uuid]=Array.from(new Uint32Array(this.array.buffer))),{uuid:this.uuid,buffer:this.array.buffer._uuid,type:this.array.constructor.name,stride:this.stride}}}const ta=new ne;class ea{constructor(t,e,i,n=!1){this.isInterleavedBufferAttribute=!0,this.name="",this.data=t,this.itemSize=e,this.offset=i,this.normalized=!0===n}get count(){return this.data.count}get array(){return this.data.array}set needsUpdate(t){this.data.needsUpdate=t}applyMatrix4(t){for(let e=0,i=this.data.count;e<i;e++)ta.fromBufferAttribute(this,e),ta.applyMatrix4(t),this.setXYZ(e,ta.x,ta.y,ta.z);return this}applyNormalMatrix(t){for(let e=0,i=this.count;e<i;e++)ta.fromBufferAttribute(this,e),ta.applyNormalMatrix(t),this.setXYZ(e,ta.x,ta.y,ta.z);return this}transformDirection(t){for(let e=0,i=this.count;e<i;e++)ta.fromBufferAttribute(this,e),ta.transformDirection(t),this.setXYZ(e,ta.x,ta.y,ta.z);return this}setX(t,e){return this.normalized&&(e=Et(e,this.array)),this.data.array[t*this.data.stride+this.offset]=e,this}setY(t,e){return this.normalized&&(e=Et(e,this.array)),this.data.array[t*this.data.stride+this.offset+1]=e,this}setZ(t,e){return this.normalized&&(e=Et(e,this.array)),this.data.array[t*this.data.stride+this.offset+2]=e,this}setW(t,e){return this.normalized&&(e=Et(e,this.array)),this.data.array[t*this.data.stride+this.offset+3]=e,this}getX(t){let e=this.data.array[t*this.data.stride+this.offset];return this.normalized&&(e=At(e,this.array)),e}getY(t){let e=this.data.array[t*this.data.stride+this.offset+1];return this.normalized&&(e=At(e,this.array)),e}getZ(t){let e=this.data.array[t*this.data.stride+this.offset+2];return this.normalized&&(e=At(e,this.array)),e}getW(t){let e=this.data.array[t*this.data.stride+this.offset+3];return this.normalized&&(e=At(e,this.array)),e}setXY(t,e,i){return t=t*this.data.stride+this.offset,this.normalized&&(e=Et(e,this.array),i=Et(i,this.array)),this.data.array[t+0]=e,this.data.array[t+1]=i,this}setXYZ(t,e,i,n){return t=t*this.data.stride+this.offset,this.normalized&&(e=Et(e,this.array),i=Et(i,this.array),n=Et(n,this.array)),this.data.array[t+0]=e,this.data.array[t+1]=i,this.data.array[t+2]=n,this}setXYZW(t,e,i,n,r){return t=t*this.data.stride+this.offset,this.normalized&&(e=Et(e,this.array),i=Et(i,this.array),n=Et(n,this.array),r=Et(r,this.array)),this.data.array[t+0]=e,this.data.array[t+1]=i,this.data.array[t+2]=n,this.data.array[t+3]=r,this}clone(t){if(void 0===t){console.log("THREE.InterleavedBufferAttribute.clone(): Cloning an interleaved buffer attribute will deinterleave buffer data.");const t=[];for(let e=0;e<this.count;e++){const i=e*this.data.stride+this.offset;for(let e=0;e<this.itemSize;e++)t.push(this.data.array[i+e])}return new Mi(new this.array.constructor(t),this.itemSize,this.normalized)}return void 0===t.interleavedBuffers&&(t.interleavedBuffers={}),void 0===t.interleavedBuffers[this.data.uuid]&&(t.interleavedBuffers[this.data.uuid]=this.data.clone(t)),new ea(t.interleavedBuffers[this.data.uuid],this.itemSize,this.offset,this.normalized)}toJSON(t){if(void 0===t){console.log("THREE.InterleavedBufferAttribute.toJSON(): Serializing an interleaved buffer attribute will deinterleave buffer data.");const t=[];for(let e=0;e<this.count;e++){const i=e*this.data.stride+this.offset;for(let e=0;e<this.itemSize;e++)t.push(this.data.array[i+e])}return{itemSize:this.itemSize,type:this.array.constructor.name,array:t,normalized:this.normalized}}return void 0===t.interleavedBuffers&&(t.interleavedBuffers={}),void 0===t.interleavedBuffers[this.data.uuid]&&(t.interleavedBuffers[this.data.uuid]=this.data.toJSON(t)),{isInterleavedBufferAttribute:!0,itemSize:this.itemSize,data:this.data.uuid,offset:this.offset,normalized:this.normalized}}}class ia extends vi{constructor(t){super(),this.isSpriteMaterial=!0,this.type="SpriteMaterial",this.color=new jt(16777215),this.map=null,this.alphaMap=null,this.rotation=0,this.sizeAttenuation=!0,this.transparent=!0,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.map=t.map,this.alphaMap=t.alphaMap,this.rotation=t.rotation,this.sizeAttenuation=t.sizeAttenuation,this.fog=t.fog,this}}let na;const ra=new ne,sa=new ne,aa=new ne,oa=new Lt,la=new Lt,ca=new Ie,ha=new ne,ua=new ne,da=new ne,pa=new Lt,ma=new Lt,fa=new Lt;class ga extends ri{constructor(t){if(super(),this.isSprite=!0,this.type="Sprite",void 0===na){na=new Di;const t=new Float32Array([-.5,-.5,0,0,0,.5,-.5,0,1,0,.5,.5,0,1,1,-.5,.5,0,0,1]),e=new Qs(t,5);na.setIndex([0,1,2,0,2,3]),na.setAttribute("position",new ea(e,3,0,!1)),na.setAttribute("uv",new ea(e,2,3,!1))}this.geometry=na,this.material=void 0!==t?t:new ia,this.center=new Lt(.5,.5)}raycast(t,e){null===t.camera&&console.error('THREE.Sprite: "Raycaster.camera" needs to be set in order to raycast against sprites.'),sa.setFromMatrixScale(this.matrixWorld),ca.copy(t.camera.matrixWorld),this.modelViewMatrix.multiplyMatrices(t.camera.matrixWorldInverse,this.matrixWorld),aa.setFromMatrixPosition(this.modelViewMatrix),t.camera.isPerspectiveCamera&&!1===this.material.sizeAttenuation&&sa.multiplyScalar(-aa.z);const i=this.material.rotation;let n,r;0!==i&&(r=Math.cos(i),n=Math.sin(i));const s=this.center;va(ha.set(-.5,-.5,0),aa,s,sa,n,r),va(ua.set(.5,-.5,0),aa,s,sa,n,r),va(da.set(.5,.5,0),aa,s,sa,n,r),pa.set(0,0),ma.set(1,0),fa.set(1,1);let a=t.ray.intersectTriangle(ha,ua,da,!1,ra);if(null===a&&(va(ua.set(-.5,.5,0),aa,s,sa,n,r),ma.set(0,1),a=t.ray.intersectTriangle(ha,da,ua,!1,ra),null===a))return;const o=t.ray.origin.distanceTo(ra);o<t.near||o>t.far||e.push({distance:o,point:ra.clone(),uv:fi.getUV(ra,ha,ua,da,pa,ma,fa,new Lt),face:null,object:this})}copy(t,e){return super.copy(t,e),void 0!==t.center&&this.center.copy(t.center),this.material=t.material,this}}function va(t,e,i,n,r,s){oa.subVectors(t,i).addScalar(.5).multiply(n),void 0!==r?(la.x=s*oa.x-r*oa.y,la.y=r*oa.x+s*oa.y):la.copy(oa),t.copy(e),t.x+=la.x,t.y+=la.y,t.applyMatrix4(ca)}const xa=new ne,_a=new ne;class ya extends ri{constructor(){super(),this._currentLevel=0,this.type="LOD",Object.defineProperties(this,{levels:{enumerable:!0,value:[]},isLOD:{value:!0}}),this.autoUpdate=!0}copy(t){super.copy(t,!1);const e=t.levels;for(let t=0,i=e.length;t<i;t++){const i=e[t];this.addLevel(i.object.clone(),i.distance)}return this.autoUpdate=t.autoUpdate,this}addLevel(t,e=0){e=Math.abs(e);const i=this.levels;let n;for(n=0;n<i.length&&!(e<i[n].distance);n++);return i.splice(n,0,{distance:e,object:t}),this.add(t),this}getCurrentLevel(){return this._currentLevel}getObjectForDistance(t){const e=this.levels;if(e.length>0){let i,n;for(i=1,n=e.length;i<n&&!(t<e[i].distance);i++);return e[i-1].object}return null}raycast(t,e){if(this.levels.length>0){xa.setFromMatrixPosition(this.matrixWorld);const i=t.ray.origin.distanceTo(xa);this.getObjectForDistance(i).raycast(t,e)}}update(t){const e=this.levels;if(e.length>1){xa.setFromMatrixPosition(t.matrixWorld),_a.setFromMatrixPosition(this.matrixWorld);const i=xa.distanceTo(_a)/t.zoom;let n,r;for(e[0].object.visible=!0,n=1,r=e.length;n<r&&i>=e[n].distance;n++)e[n-1].object.visible=!1,e[n].object.visible=!0;for(this._currentLevel=n-1;n<r;n++)e[n].object.visible=!1}}toJSON(t){const e=super.toJSON(t);!1===this.autoUpdate&&(e.object.autoUpdate=!1),e.object.levels=[];const i=this.levels;for(let t=0,n=i.length;t<n;t++){const n=i[t];e.object.levels.push({object:n.object.uuid,distance:n.distance})}return e}}const Ma=new ne,ba=new $t,Sa=new $t,wa=new ne,Ta=new Ie;class Aa extends Ji{constructor(t,e){super(t,e),this.isSkinnedMesh=!0,this.type="SkinnedMesh",this.bindMode="attached",this.bindMatrix=new Ie,this.bindMatrixInverse=new Ie}copy(t,e){return super.copy(t,e),this.bindMode=t.bindMode,this.bindMatrix.copy(t.bindMatrix),this.bindMatrixInverse.copy(t.bindMatrixInverse),this.skeleton=t.skeleton,this}bind(t,e){this.skeleton=t,void 0===e&&(this.updateMatrixWorld(!0),this.skeleton.calculateInverses(),e=this.matrixWorld),this.bindMatrix.copy(e),this.bindMatrixInverse.copy(e).invert()}pose(){this.skeleton.pose()}normalizeSkinWeights(){const t=new $t,e=this.geometry.attributes.skinWeight;for(let i=0,n=e.count;i<n;i++){t.fromBufferAttribute(e,i);const n=1/t.manhattanLength();n!==1/0?t.multiplyScalar(n):t.set(1,0,0,0),e.setXYZW(i,t.x,t.y,t.z,t.w)}}updateMatrixWorld(t){super.updateMatrixWorld(t),"attached"===this.bindMode?this.bindMatrixInverse.copy(this.matrixWorld).invert():"detached"===this.bindMode?this.bindMatrixInverse.copy(this.bindMatrix).invert():console.warn("THREE.SkinnedMesh: Unrecognized bindMode: "+this.bindMode)}boneTransform(t,e){const i=this.skeleton,n=this.geometry;ba.fromBufferAttribute(n.attributes.skinIndex,t),Sa.fromBufferAttribute(n.attributes.skinWeight,t),Ma.copy(e).applyMatrix4(this.bindMatrix),e.set(0,0,0);for(let t=0;t<4;t++){const n=Sa.getComponent(t);if(0!==n){const r=ba.getComponent(t);Ta.multiplyMatrices(i.bones[r].matrixWorld,i.boneInverses[r]),e.addScaledVector(wa.copy(Ma).applyMatrix4(Ta),n)}}return e.applyMatrix4(this.bindMatrixInverse)}}class Ea extends ri{constructor(){super(),this.isBone=!0,this.type="Bone"}}class Ca extends Kt{constructor(t=null,e=1,i=1,n,r,s,a,o,l=1003,c=1003,h,u){super(null,s,a,o,l,c,n,r,h,u),this.isDataTexture=!0,this.image={data:t,width:e,height:i},this.generateMipmaps=!1,this.flipY=!1,this.unpackAlignment=1}}const La=new Ie,Ra=new Ie;class Pa{constructor(t=[],e=[]){this.uuid=_t(),this.bones=t.slice(0),this.boneInverses=e,this.boneMatrices=null,this.boneTexture=null,this.boneTextureSize=0,this.frame=-1,this.init()}init(){const t=this.bones,e=this.boneInverses;if(this.boneMatrices=new Float32Array(16*t.length),0===e.length)this.calculateInverses();else if(t.length!==e.length){console.warn("THREE.Skeleton: Number of inverse bone matrices does not match amount of bones."),this.boneInverses=[];for(let t=0,e=this.bones.length;t<e;t++)this.boneInverses.push(new Ie)}}calculateInverses(){this.boneInverses.length=0;for(let t=0,e=this.bones.length;t<e;t++){const e=new Ie;this.bones[t]&&e.copy(this.bones[t].matrixWorld).invert(),this.boneInverses.push(e)}}pose(){for(let t=0,e=this.bones.length;t<e;t++){const e=this.bones[t];e&&e.matrixWorld.copy(this.boneInverses[t]).invert()}for(let t=0,e=this.bones.length;t<e;t++){const e=this.bones[t];e&&(e.parent&&e.parent.isBone?(e.matrix.copy(e.parent.matrixWorld).invert(),e.matrix.multiply(e.matrixWorld)):e.matrix.copy(e.matrixWorld),e.matrix.decompose(e.position,e.quaternion,e.scale))}}update(){const t=this.bones,e=this.boneInverses,i=this.boneMatrices,n=this.boneTexture;for(let n=0,r=t.length;n<r;n++){const r=t[n]?t[n].matrixWorld:Ra;La.multiplyMatrices(r,e[n]),La.toArray(i,16*n)}null!==n&&(n.needsUpdate=!0)}clone(){return new Pa(this.bones,this.boneInverses)}computeBoneTexture(){let t=Math.sqrt(4*this.bones.length);t=wt(t),t=Math.max(t,4);const e=new Float32Array(t*t*4);e.set(this.boneMatrices);const i=new Ca(e,t,t,w,M);return i.needsUpdate=!0,this.boneMatrices=e,this.boneTexture=i,this.boneTextureSize=t,this}getBoneByName(t){for(let e=0,i=this.bones.length;e<i;e++){const i=this.bones[e];if(i.name===t)return i}}dispose(){null!==this.boneTexture&&(this.boneTexture.dispose(),this.boneTexture=null)}fromJSON(t,e){this.uuid=t.uuid;for(let i=0,n=t.bones.length;i<n;i++){const n=t.bones[i];let r=e[n];void 0===r&&(console.warn("THREE.Skeleton: No bone found with UUID:",n),r=new Ea),this.bones.push(r),this.boneInverses.push((new Ie).fromArray(t.boneInverses[i]))}return this.init(),this}toJSON(){const t={metadata:{version:4.5,type:"Skeleton",generator:"Skeleton.toJSON"},bones:[],boneInverses:[]};t.uuid=this.uuid;const e=this.bones,i=this.boneInverses;for(let n=0,r=e.length;n<r;n++){const r=e[n];t.bones.push(r.uuid);const s=i[n];t.boneInverses.push(s.toArray())}return t}}class Da extends Mi{constructor(t,e,i,n=1){super(t,e,i),this.isInstancedBufferAttribute=!0,this.meshPerAttribute=n}copy(t){return super.copy(t),this.meshPerAttribute=t.meshPerAttribute,this}toJSON(){const t=super.toJSON();return t.meshPerAttribute=this.meshPerAttribute,t.isInstancedBufferAttribute=!0,t}}const Ia=new Ie,Na=new Ie,Oa=[],za=new Ie,Ua=new Ji;class Ba extends Ji{constructor(t,e,i){super(t,e),this.isInstancedMesh=!0,this.instanceMatrix=new Da(new Float32Array(16*i),16),this.instanceColor=null,this.count=i,this.frustumCulled=!1;for(let t=0;t<i;t++)this.setMatrixAt(t,za)}copy(t,e){return super.copy(t,e),this.instanceMatrix.copy(t.instanceMatrix),null!==t.instanceColor&&(this.instanceColor=t.instanceColor.clone()),this.count=t.count,this}getColorAt(t,e){e.fromArray(this.instanceColor.array,3*t)}getMatrixAt(t,e){e.fromArray(this.instanceMatrix.array,16*t)}raycast(t,e){const i=this.matrixWorld,n=this.count;if(Ua.geometry=this.geometry,Ua.material=this.material,void 0!==Ua.material)for(let r=0;r<n;r++){this.getMatrixAt(r,Ia),Na.multiplyMatrices(i,Ia),Ua.matrixWorld=Na,Ua.raycast(t,Oa);for(let t=0,i=Oa.length;t<i;t++){const i=Oa[t];i.instanceId=r,i.object=this,e.push(i)}Oa.length=0}}setColorAt(t,e){null===this.instanceColor&&(this.instanceColor=new Da(new Float32Array(3*this.instanceMatrix.count),3)),e.toArray(this.instanceColor.array,3*t)}setMatrixAt(t,e){e.toArray(this.instanceMatrix.array,16*t)}updateMorphTargets(){}dispose(){this.dispatchEvent({type:"dispose"})}}class Fa extends vi{constructor(t){super(),this.isLineBasicMaterial=!0,this.type="LineBasicMaterial",this.color=new jt(16777215),this.linewidth=1,this.linecap="round",this.linejoin="round",this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.linewidth=t.linewidth,this.linecap=t.linecap,this.linejoin=t.linejoin,this.fog=t.fog,this}}const ka=new ne,Ga=new ne,Va=new Ie,Ha=new De,Wa=new we;class ja extends ri{constructor(t=new Di,e=new Fa){super(),this.isLine=!0,this.type="Line",this.geometry=t,this.material=e,this.updateMorphTargets()}copy(t,e){return super.copy(t,e),this.material=t.material,this.geometry=t.geometry,this}computeLineDistances(){const t=this.geometry;if(null===t.index){const e=t.attributes.position,i=[0];for(let t=1,n=e.count;t<n;t++)ka.fromBufferAttribute(e,t-1),Ga.fromBufferAttribute(e,t),i[t]=i[t-1],i[t]+=ka.distanceTo(Ga);t.setAttribute("lineDistance",new wi(i,1))}else console.warn("THREE.Line.computeLineDistances(): Computation only possible with non-indexed BufferGeometry.");return this}raycast(t,e){const i=this.geometry,n=this.matrixWorld,r=t.params.Line.threshold,s=i.drawRange;if(null===i.boundingSphere&&i.computeBoundingSphere(),Wa.copy(i.boundingSphere),Wa.applyMatrix4(n),Wa.radius+=r,!1===t.ray.intersectsSphere(Wa))return;Va.copy(n).invert(),Ha.copy(t.ray).applyMatrix4(Va);const a=r/((this.scale.x+this.scale.y+this.scale.z)/3),o=a*a,l=new ne,c=new ne,h=new ne,u=new ne,d=this.isLineSegments?2:1,p=i.index,m=i.attributes.position;if(null!==p){for(let i=Math.max(0,s.start),n=Math.min(p.count,s.start+s.count)-1;i<n;i+=d){const n=p.getX(i),r=p.getX(i+1);l.fromBufferAttribute(m,n),c.fromBufferAttribute(m,r);if(Ha.distanceSqToSegment(l,c,u,h)>o)continue;u.applyMatrix4(this.matrixWorld);const s=t.ray.origin.distanceTo(u);s<t.near||s>t.far||e.push({distance:s,point:h.clone().applyMatrix4(this.matrixWorld),index:i,face:null,faceIndex:null,object:this})}}else{for(let i=Math.max(0,s.start),n=Math.min(m.count,s.start+s.count)-1;i<n;i+=d){l.fromBufferAttribute(m,i),c.fromBufferAttribute(m,i+1);if(Ha.distanceSqToSegment(l,c,u,h)>o)continue;u.applyMatrix4(this.matrixWorld);const n=t.ray.origin.distanceTo(u);n<t.near||n>t.far||e.push({distance:n,point:h.clone().applyMatrix4(this.matrixWorld),index:i,face:null,faceIndex:null,object:this})}}}updateMorphTargets(){const t=this.geometry.morphAttributes,e=Object.keys(t);if(e.length>0){const i=t[e[0]];if(void 0!==i){this.morphTargetInfluences=[],this.morphTargetDictionary={};for(let t=0,e=i.length;t<e;t++){const e=i[t].name||String(t);this.morphTargetInfluences.push(0),this.morphTargetDictionary[e]=t}}}}}const qa=new ne,Xa=new ne;class Ya extends ja{constructor(t,e){super(t,e),this.isLineSegments=!0,this.type="LineSegments"}computeLineDistances(){const t=this.geometry;if(null===t.index){const e=t.attributes.position,i=[];for(let t=0,n=e.count;t<n;t+=2)qa.fromBufferAttribute(e,t),Xa.fromBufferAttribute(e,t+1),i[t]=0===t?0:i[t-1],i[t+1]=i[t]+qa.distanceTo(Xa);t.setAttribute("lineDistance",new wi(i,1))}else console.warn("THREE.LineSegments.computeLineDistances(): Computation only possible with non-indexed BufferGeometry.");return this}}class Za extends ja{constructor(t,e){super(t,e),this.isLineLoop=!0,this.type="LineLoop"}}class Ja extends vi{constructor(t){super(),this.isPointsMaterial=!0,this.type="PointsMaterial",this.color=new jt(16777215),this.map=null,this.alphaMap=null,this.size=1,this.sizeAttenuation=!0,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.map=t.map,this.alphaMap=t.alphaMap,this.size=t.size,this.sizeAttenuation=t.sizeAttenuation,this.fog=t.fog,this}}const Ka=new Ie,$a=new De,Qa=new we,to=new ne;class eo extends ri{constructor(t=new Di,e=new Ja){super(),this.isPoints=!0,this.type="Points",this.geometry=t,this.material=e,this.updateMorphTargets()}copy(t,e){return super.copy(t,e),this.material=t.material,this.geometry=t.geometry,this}raycast(t,e){const i=this.geometry,n=this.matrixWorld,r=t.params.Points.threshold,s=i.drawRange;if(null===i.boundingSphere&&i.computeBoundingSphere(),Qa.copy(i.boundingSphere),Qa.applyMatrix4(n),Qa.radius+=r,!1===t.ray.intersectsSphere(Qa))return;Ka.copy(n).invert(),$a.copy(t.ray).applyMatrix4(Ka);const a=r/((this.scale.x+this.scale.y+this.scale.z)/3),o=a*a,l=i.index,c=i.attributes.position;if(null!==l){for(let i=Math.max(0,s.start),r=Math.min(l.count,s.start+s.count);i<r;i++){const r=l.getX(i);to.fromBufferAttribute(c,r),io(to,r,o,n,t,e,this)}}else{for(let i=Math.max(0,s.start),r=Math.min(c.count,s.start+s.count);i<r;i++)to.fromBufferAttribute(c,i),io(to,i,o,n,t,e,this)}}updateMorphTargets(){const t=this.geometry.morphAttributes,e=Object.keys(t);if(e.length>0){const i=t[e[0]];if(void 0!==i){this.morphTargetInfluences=[],this.morphTargetDictionary={};for(let t=0,e=i.length;t<e;t++){const e=i[t].name||String(t);this.morphTargetInfluences.push(0),this.morphTargetDictionary[e]=t}}}}}function io(t,e,i,n,r,s,a){const o=$a.distanceSqToPoint(t);if(o<i){const i=new ne;$a.closestPointToPoint(t,i),i.applyMatrix4(n);const l=r.ray.origin.distanceTo(i);if(l<r.near||l>r.far)return;s.push({distance:l,distanceToRay:Math.sqrt(o),point:i,index:e,face:null,object:a})}}class no extends Kt{constructor(t,e,i,n,r,s,a,o,l,c,h,u){super(null,s,a,o,l,c,n,r,h,u),this.isCompressedTexture=!0,this.image={width:e,height:i},this.mipmaps=t,this.flipY=!1,this.generateMipmaps=!1}}class ro{constructor(){this.type="Curve",this.arcLengthDivisions=200}getPoint(){return console.warn("THREE.Curve: .getPoint() not implemented."),null}getPointAt(t,e){const i=this.getUtoTmapping(t);return this.getPoint(i,e)}getPoints(t=5){const e=[];for(let i=0;i<=t;i++)e.push(this.getPoint(i/t));return e}getSpacedPoints(t=5){const e=[];for(let i=0;i<=t;i++)e.push(this.getPointAt(i/t));return e}getLength(){const t=this.getLengths();return t[t.length-1]}getLengths(t=this.arcLengthDivisions){if(this.cacheArcLengths&&this.cacheArcLengths.length===t+1&&!this.needsUpdate)return this.cacheArcLengths;this.needsUpdate=!1;const e=[];let i,n=this.getPoint(0),r=0;e.push(0);for(let s=1;s<=t;s++)i=this.getPoint(s/t),r+=i.distanceTo(n),e.push(r),n=i;return this.cacheArcLengths=e,e}updateArcLengths(){this.needsUpdate=!0,this.getLengths()}getUtoTmapping(t,e){const i=this.getLengths();let n=0;const r=i.length;let s;s=e||t*i[r-1];let a,o=0,l=r-1;for(;o<=l;)if(n=Math.floor(o+(l-o)/2),a=i[n]-s,a<0)o=n+1;else{if(!(a>0)){l=n;break}l=n-1}if(n=l,i[n]===s)return n/(r-1);const c=i[n];return(n+(s-c)/(i[n+1]-c))/(r-1)}getTangent(t,e){const i=1e-4;let n=t-i,r=t+i;n<0&&(n=0),r>1&&(r=1);const s=this.getPoint(n),a=this.getPoint(r),o=e||(s.isVector2?new Lt:new ne);return o.copy(a).sub(s).normalize(),o}getTangentAt(t,e){const i=this.getUtoTmapping(t);return this.getTangent(i,e)}computeFrenetFrames(t,e){const i=new ne,n=[],r=[],s=[],a=new ne,o=new Ie;for(let e=0;e<=t;e++){const i=e/t;n[e]=this.getTangentAt(i,new ne)}r[0]=new ne,s[0]=new ne;let l=Number.MAX_VALUE;const c=Math.abs(n[0].x),h=Math.abs(n[0].y),u=Math.abs(n[0].z);c<=l&&(l=c,i.set(1,0,0)),h<=l&&(l=h,i.set(0,1,0)),u<=l&&i.set(0,0,1),a.crossVectors(n[0],i).normalize(),r[0].crossVectors(n[0],a),s[0].crossVectors(n[0],r[0]);for(let e=1;e<=t;e++){if(r[e]=r[e-1].clone(),s[e]=s[e-1].clone(),a.crossVectors(n[e-1],n[e]),a.length()>Number.EPSILON){a.normalize();const t=Math.acos(yt(n[e-1].dot(n[e]),-1,1));r[e].applyMatrix4(o.makeRotationAxis(a,t))}s[e].crossVectors(n[e],r[e])}if(!0===e){let e=Math.acos(yt(r[0].dot(r[t]),-1,1));e/=t,n[0].dot(a.crossVectors(r[0],r[t]))>0&&(e=-e);for(let i=1;i<=t;i++)r[i].applyMatrix4(o.makeRotationAxis(n[i],e*i)),s[i].crossVectors(n[i],r[i])}return{tangents:n,normals:r,binormals:s}}clone(){return(new this.constructor).copy(this)}copy(t){return this.arcLengthDivisions=t.arcLengthDivisions,this}toJSON(){const t={metadata:{version:4.5,type:"Curve",generator:"Curve.toJSON"}};return t.arcLengthDivisions=this.arcLengthDivisions,t.type=this.type,t}fromJSON(t){return this.arcLengthDivisions=t.arcLengthDivisions,this}}class so extends ro{constructor(t=0,e=0,i=1,n=1,r=0,s=2*Math.PI,a=!1,o=0){super(),this.isEllipseCurve=!0,this.type="EllipseCurve",this.aX=t,this.aY=e,this.xRadius=i,this.yRadius=n,this.aStartAngle=r,this.aEndAngle=s,this.aClockwise=a,this.aRotation=o}getPoint(t,e){const i=e||new Lt,n=2*Math.PI;let r=this.aEndAngle-this.aStartAngle;const s=Math.abs(r)<Number.EPSILON;for(;r<0;)r+=n;for(;r>n;)r-=n;r<Number.EPSILON&&(r=s?0:n),!0!==this.aClockwise||s||(r===n?r=-n:r-=n);const a=this.aStartAngle+t*r;let o=this.aX+this.xRadius*Math.cos(a),l=this.aY+this.yRadius*Math.sin(a);if(0!==this.aRotation){const t=Math.cos(this.aRotation),e=Math.sin(this.aRotation),i=o-this.aX,n=l-this.aY;o=i*t-n*e+this.aX,l=i*e+n*t+this.aY}return i.set(o,l)}copy(t){return super.copy(t),this.aX=t.aX,this.aY=t.aY,this.xRadius=t.xRadius,this.yRadius=t.yRadius,this.aStartAngle=t.aStartAngle,this.aEndAngle=t.aEndAngle,this.aClockwise=t.aClockwise,this.aRotation=t.aRotation,this}toJSON(){const t=super.toJSON();return t.aX=this.aX,t.aY=this.aY,t.xRadius=this.xRadius,t.yRadius=this.yRadius,t.aStartAngle=this.aStartAngle,t.aEndAngle=this.aEndAngle,t.aClockwise=this.aClockwise,t.aRotation=this.aRotation,t}fromJSON(t){return super.fromJSON(t),this.aX=t.aX,this.aY=t.aY,this.xRadius=t.xRadius,this.yRadius=t.yRadius,this.aStartAngle=t.aStartAngle,this.aEndAngle=t.aEndAngle,this.aClockwise=t.aClockwise,this.aRotation=t.aRotation,this}}class ao extends so{constructor(t,e,i,n,r,s){super(t,e,i,i,n,r,s),this.isArcCurve=!0,this.type="ArcCurve"}}function oo(){let t=0,e=0,i=0,n=0;function r(r,s,a,o){t=r,e=a,i=-3*r+3*s-2*a-o,n=2*r-2*s+a+o}return{initCatmullRom:function(t,e,i,n,s){r(e,i,s*(i-t),s*(n-e))},initNonuniformCatmullRom:function(t,e,i,n,s,a,o){let l=(e-t)/s-(i-t)/(s+a)+(i-e)/a,c=(i-e)/a-(n-e)/(a+o)+(n-i)/o;l*=a,c*=a,r(e,i,l,c)},calc:function(r){const s=r*r;return t+e*r+i*s+n*(s*r)}}}const lo=new ne,co=new oo,ho=new oo,uo=new oo;class po extends ro{constructor(t=[],e=!1,i="centripetal",n=.5){super(),this.isCatmullRomCurve3=!0,this.type="CatmullRomCurve3",this.points=t,this.closed=e,this.curveType=i,this.tension=n}getPoint(t,e=new ne){const i=e,n=this.points,r=n.length,s=(r-(this.closed?0:1))*t;let a,o,l=Math.floor(s),c=s-l;this.closed?l+=l>0?0:(Math.floor(Math.abs(l)/r)+1)*r:0===c&&l===r-1&&(l=r-2,c=1),this.closed||l>0?a=n[(l-1)%r]:(lo.subVectors(n[0],n[1]).add(n[0]),a=lo);const h=n[l%r],u=n[(l+1)%r];if(this.closed||l+2<r?o=n[(l+2)%r]:(lo.subVectors(n[r-1],n[r-2]).add(n[r-1]),o=lo),"centripetal"===this.curveType||"chordal"===this.curveType){const t="chordal"===this.curveType?.5:.25;let e=Math.pow(a.distanceToSquared(h),t),i=Math.pow(h.distanceToSquared(u),t),n=Math.pow(u.distanceToSquared(o),t);i<1e-4&&(i=1),e<1e-4&&(e=i),n<1e-4&&(n=i),co.initNonuniformCatmullRom(a.x,h.x,u.x,o.x,e,i,n),ho.initNonuniformCatmullRom(a.y,h.y,u.y,o.y,e,i,n),uo.initNonuniformCatmullRom(a.z,h.z,u.z,o.z,e,i,n)}else"catmullrom"===this.curveType&&(co.initCatmullRom(a.x,h.x,u.x,o.x,this.tension),ho.initCatmullRom(a.y,h.y,u.y,o.y,this.tension),uo.initCatmullRom(a.z,h.z,u.z,o.z,this.tension));return i.set(co.calc(c),ho.calc(c),uo.calc(c)),i}copy(t){super.copy(t),this.points=[];for(let e=0,i=t.points.length;e<i;e++){const i=t.points[e];this.points.push(i.clone())}return this.closed=t.closed,this.curveType=t.curveType,this.tension=t.tension,this}toJSON(){const t=super.toJSON();t.points=[];for(let e=0,i=this.points.length;e<i;e++){const i=this.points[e];t.points.push(i.toArray())}return t.closed=this.closed,t.curveType=this.curveType,t.tension=this.tension,t}fromJSON(t){super.fromJSON(t),this.points=[];for(let e=0,i=t.points.length;e<i;e++){const i=t.points[e];this.points.push((new ne).fromArray(i))}return this.closed=t.closed,this.curveType=t.curveType,this.tension=t.tension,this}}function mo(t,e,i,n,r){const s=.5*(n-e),a=.5*(r-i),o=t*t;return(2*i-2*n+s+a)*(t*o)+(-3*i+3*n-2*s-a)*o+s*t+i}function fo(t,e,i,n){return function(t,e){const i=1-t;return i*i*e}(t,e)+function(t,e){return 2*(1-t)*t*e}(t,i)+function(t,e){return t*t*e}(t,n)}function go(t,e,i,n,r){return function(t,e){const i=1-t;return i*i*i*e}(t,e)+function(t,e){const i=1-t;return 3*i*i*t*e}(t,i)+function(t,e){return 3*(1-t)*t*t*e}(t,n)+function(t,e){return t*t*t*e}(t,r)}class vo extends ro{constructor(t=new Lt,e=new Lt,i=new Lt,n=new Lt){super(),this.isCubicBezierCurve=!0,this.type="CubicBezierCurve",this.v0=t,this.v1=e,this.v2=i,this.v3=n}getPoint(t,e=new Lt){const i=e,n=this.v0,r=this.v1,s=this.v2,a=this.v3;return i.set(go(t,n.x,r.x,s.x,a.x),go(t,n.y,r.y,s.y,a.y)),i}copy(t){return super.copy(t),this.v0.copy(t.v0),this.v1.copy(t.v1),this.v2.copy(t.v2),this.v3.copy(t.v3),this}toJSON(){const t=super.toJSON();return t.v0=this.v0.toArray(),t.v1=this.v1.toArray(),t.v2=this.v2.toArray(),t.v3=this.v3.toArray(),t}fromJSON(t){return super.fromJSON(t),this.v0.fromArray(t.v0),this.v1.fromArray(t.v1),this.v2.fromArray(t.v2),this.v3.fromArray(t.v3),this}}class xo extends ro{constructor(t=new ne,e=new ne,i=new ne,n=new ne){super(),this.isCubicBezierCurve3=!0,this.type="CubicBezierCurve3",this.v0=t,this.v1=e,this.v2=i,this.v3=n}getPoint(t,e=new ne){const i=e,n=this.v0,r=this.v1,s=this.v2,a=this.v3;return i.set(go(t,n.x,r.x,s.x,a.x),go(t,n.y,r.y,s.y,a.y),go(t,n.z,r.z,s.z,a.z)),i}copy(t){return super.copy(t),this.v0.copy(t.v0),this.v1.copy(t.v1),this.v2.copy(t.v2),this.v3.copy(t.v3),this}toJSON(){const t=super.toJSON();return t.v0=this.v0.toArray(),t.v1=this.v1.toArray(),t.v2=this.v2.toArray(),t.v3=this.v3.toArray(),t}fromJSON(t){return super.fromJSON(t),this.v0.fromArray(t.v0),this.v1.fromArray(t.v1),this.v2.fromArray(t.v2),this.v3.fromArray(t.v3),this}}class _o extends ro{constructor(t=new Lt,e=new Lt){super(),this.isLineCurve=!0,this.type="LineCurve",this.v1=t,this.v2=e}getPoint(t,e=new Lt){const i=e;return 1===t?i.copy(this.v2):(i.copy(this.v2).sub(this.v1),i.multiplyScalar(t).add(this.v1)),i}getPointAt(t,e){return this.getPoint(t,e)}getTangent(t,e){const i=e||new Lt;return i.copy(this.v2).sub(this.v1).normalize(),i}copy(t){return super.copy(t),this.v1.copy(t.v1),this.v2.copy(t.v2),this}toJSON(){const t=super.toJSON();return t.v1=this.v1.toArray(),t.v2=this.v2.toArray(),t}fromJSON(t){return super.fromJSON(t),this.v1.fromArray(t.v1),this.v2.fromArray(t.v2),this}}class yo extends ro{constructor(t=new ne,e=new ne){super(),this.isLineCurve3=!0,this.type="LineCurve3",this.v1=t,this.v2=e}getPoint(t,e=new ne){const i=e;return 1===t?i.copy(this.v2):(i.copy(this.v2).sub(this.v1),i.multiplyScalar(t).add(this.v1)),i}getPointAt(t,e){return this.getPoint(t,e)}copy(t){return super.copy(t),this.v1.copy(t.v1),this.v2.copy(t.v2),this}toJSON(){const t=super.toJSON();return t.v1=this.v1.toArray(),t.v2=this.v2.toArray(),t}fromJSON(t){return super.fromJSON(t),this.v1.fromArray(t.v1),this.v2.fromArray(t.v2),this}}class Mo extends ro{constructor(t=new Lt,e=new Lt,i=new Lt){super(),this.isQuadraticBezierCurve=!0,this.type="QuadraticBezierCurve",this.v0=t,this.v1=e,this.v2=i}getPoint(t,e=new Lt){const i=e,n=this.v0,r=this.v1,s=this.v2;return i.set(fo(t,n.x,r.x,s.x),fo(t,n.y,r.y,s.y)),i}copy(t){return super.copy(t),this.v0.copy(t.v0),this.v1.copy(t.v1),this.v2.copy(t.v2),this}toJSON(){const t=super.toJSON();return t.v0=this.v0.toArray(),t.v1=this.v1.toArray(),t.v2=this.v2.toArray(),t}fromJSON(t){return super.fromJSON(t),this.v0.fromArray(t.v0),this.v1.fromArray(t.v1),this.v2.fromArray(t.v2),this}}class bo extends ro{constructor(t=new ne,e=new ne,i=new ne){super(),this.isQuadraticBezierCurve3=!0,this.type="QuadraticBezierCurve3",this.v0=t,this.v1=e,this.v2=i}getPoint(t,e=new ne){const i=e,n=this.v0,r=this.v1,s=this.v2;return i.set(fo(t,n.x,r.x,s.x),fo(t,n.y,r.y,s.y),fo(t,n.z,r.z,s.z)),i}copy(t){return super.copy(t),this.v0.copy(t.v0),this.v1.copy(t.v1),this.v2.copy(t.v2),this}toJSON(){const t=super.toJSON();return t.v0=this.v0.toArray(),t.v1=this.v1.toArray(),t.v2=this.v2.toArray(),t}fromJSON(t){return super.fromJSON(t),this.v0.fromArray(t.v0),this.v1.fromArray(t.v1),this.v2.fromArray(t.v2),this}}class So extends ro{constructor(t=[]){super(),this.isSplineCurve=!0,this.type="SplineCurve",this.points=t}getPoint(t,e=new Lt){const i=e,n=this.points,r=(n.length-1)*t,s=Math.floor(r),a=r-s,o=n[0===s?s:s-1],l=n[s],c=n[s>n.length-2?n.length-1:s+1],h=n[s>n.length-3?n.length-1:s+2];return i.set(mo(a,o.x,l.x,c.x,h.x),mo(a,o.y,l.y,c.y,h.y)),i}copy(t){super.copy(t),this.points=[];for(let e=0,i=t.points.length;e<i;e++){const i=t.points[e];this.points.push(i.clone())}return this}toJSON(){const t=super.toJSON();t.points=[];for(let e=0,i=this.points.length;e<i;e++){const i=this.points[e];t.points.push(i.toArray())}return t}fromJSON(t){super.fromJSON(t),this.points=[];for(let e=0,i=t.points.length;e<i;e++){const i=t.points[e];this.points.push((new Lt).fromArray(i))}return this}}var wo=Object.freeze({__proto__:null,ArcCurve:ao,CatmullRomCurve3:po,CubicBezierCurve:vo,CubicBezierCurve3:xo,EllipseCurve:so,LineCurve:_o,LineCurve3:yo,QuadraticBezierCurve:Mo,QuadraticBezierCurve3:bo,SplineCurve:So});class To extends ro{constructor(){super(),this.type="CurvePath",this.curves=[],this.autoClose=!1}add(t){this.curves.push(t)}closePath(){const t=this.curves[0].getPoint(0),e=this.curves[this.curves.length-1].getPoint(1);t.equals(e)||this.curves.push(new _o(e,t))}getPoint(t,e){const i=t*this.getLength(),n=this.getCurveLengths();let r=0;for(;r<n.length;){if(n[r]>=i){const t=n[r]-i,s=this.curves[r],a=s.getLength(),o=0===a?0:1-t/a;return s.getPointAt(o,e)}r++}return null}getLength(){const t=this.getCurveLengths();return t[t.length-1]}updateArcLengths(){this.needsUpdate=!0,this.cacheLengths=null,this.getCurveLengths()}getCurveLengths(){if(this.cacheLengths&&this.cacheLengths.length===this.curves.length)return this.cacheLengths;const t=[];let e=0;for(let i=0,n=this.curves.length;i<n;i++)e+=this.curves[i].getLength(),t.push(e);return this.cacheLengths=t,t}getSpacedPoints(t=40){const e=[];for(let i=0;i<=t;i++)e.push(this.getPoint(i/t));return this.autoClose&&e.push(e[0]),e}getPoints(t=12){const e=[];let i;for(let n=0,r=this.curves;n<r.length;n++){const s=r[n],a=s.isEllipseCurve?2*t:s.isLineCurve||s.isLineCurve3?1:s.isSplineCurve?t*s.points.length:t,o=s.getPoints(a);for(let t=0;t<o.length;t++){const n=o[t];i&&i.equals(n)||(e.push(n),i=n)}}return this.autoClose&&e.length>1&&!e[e.length-1].equals(e[0])&&e.push(e[0]),e}copy(t){super.copy(t),this.curves=[];for(let e=0,i=t.curves.length;e<i;e++){const i=t.curves[e];this.curves.push(i.clone())}return this.autoClose=t.autoClose,this}toJSON(){const t=super.toJSON();t.autoClose=this.autoClose,t.curves=[];for(let e=0,i=this.curves.length;e<i;e++){const i=this.curves[e];t.curves.push(i.toJSON())}return t}fromJSON(t){super.fromJSON(t),this.autoClose=t.autoClose,this.curves=[];for(let e=0,i=t.curves.length;e<i;e++){const i=t.curves[e];this.curves.push((new wo[i.type]).fromJSON(i))}return this}}class Ao extends To{constructor(t){super(),this.type="Path",this.currentPoint=new Lt,t&&this.setFromPoints(t)}setFromPoints(t){this.moveTo(t[0].x,t[0].y);for(let e=1,i=t.length;e<i;e++)this.lineTo(t[e].x,t[e].y);return this}moveTo(t,e){return this.currentPoint.set(t,e),this}lineTo(t,e){const i=new _o(this.currentPoint.clone(),new Lt(t,e));return this.curves.push(i),this.currentPoint.set(t,e),this}quadraticCurveTo(t,e,i,n){const r=new Mo(this.currentPoint.clone(),new Lt(t,e),new Lt(i,n));return this.curves.push(r),this.currentPoint.set(i,n),this}bezierCurveTo(t,e,i,n,r,s){const a=new vo(this.currentPoint.clone(),new Lt(t,e),new Lt(i,n),new Lt(r,s));return this.curves.push(a),this.currentPoint.set(r,s),this}splineThru(t){const e=[this.currentPoint.clone()].concat(t),i=new So(e);return this.curves.push(i),this.currentPoint.copy(t[t.length-1]),this}arc(t,e,i,n,r,s){const a=this.currentPoint.x,o=this.currentPoint.y;return this.absarc(t+a,e+o,i,n,r,s),this}absarc(t,e,i,n,r,s){return this.absellipse(t,e,i,i,n,r,s),this}ellipse(t,e,i,n,r,s,a,o){const l=this.currentPoint.x,c=this.currentPoint.y;return this.absellipse(t+l,e+c,i,n,r,s,a,o),this}absellipse(t,e,i,n,r,s,a,o){const l=new so(t,e,i,n,r,s,a,o);if(this.curves.length>0){const t=l.getPoint(0);t.equals(this.currentPoint)||this.lineTo(t.x,t.y)}this.curves.push(l);const c=l.getPoint(1);return this.currentPoint.copy(c),this}copy(t){return super.copy(t),this.currentPoint.copy(t.currentPoint),this}toJSON(){const t=super.toJSON();return t.currentPoint=this.currentPoint.toArray(),t}fromJSON(t){return super.fromJSON(t),this.currentPoint.fromArray(t.currentPoint),this}}class Eo extends Di{constructor(t=[new Lt(0,-.5),new Lt(.5,0),new Lt(0,.5)],e=12,i=0,n=2*Math.PI){super(),this.type="LatheGeometry",this.parameters={points:t,segments:e,phiStart:i,phiLength:n},e=Math.floor(e),n=yt(n,0,2*Math.PI);const r=[],s=[],a=[],o=[],l=[],c=1/e,h=new ne,u=new Lt,d=new ne,p=new ne,m=new ne;let f=0,g=0;for(let e=0;e<=t.length-1;e++)switch(e){case 0:f=t[e+1].x-t[e].x,g=t[e+1].y-t[e].y,d.x=1*g,d.y=-f,d.z=0*g,m.copy(d),d.normalize(),o.push(d.x,d.y,d.z);break;case t.length-1:o.push(m.x,m.y,m.z);break;default:f=t[e+1].x-t[e].x,g=t[e+1].y-t[e].y,d.x=1*g,d.y=-f,d.z=0*g,p.copy(d),d.x+=m.x,d.y+=m.y,d.z+=m.z,d.normalize(),o.push(d.x,d.y,d.z),m.copy(p)}for(let r=0;r<=e;r++){const d=i+r*c*n,p=Math.sin(d),m=Math.cos(d);for(let i=0;i<=t.length-1;i++){h.x=t[i].x*p,h.y=t[i].y,h.z=t[i].x*m,s.push(h.x,h.y,h.z),u.x=r/e,u.y=i/(t.length-1),a.push(u.x,u.y);const n=o[3*i+0]*p,c=o[3*i+1],d=o[3*i+0]*m;l.push(n,c,d)}}for(let i=0;i<e;i++)for(let e=0;e<t.length-1;e++){const n=e+i*t.length,s=n,a=n+t.length,o=n+t.length+1,l=n+1;r.push(s,a,l),r.push(o,l,a)}this.setIndex(r),this.setAttribute("position",new wi(s,3)),this.setAttribute("uv",new wi(a,2)),this.setAttribute("normal",new wi(l,3))}static fromJSON(t){return new Eo(t.points,t.segments,t.phiStart,t.phiLength)}}class Co extends Eo{constructor(t=1,e=1,i=4,n=8){const r=new Ao;r.absarc(0,-e/2,t,1.5*Math.PI,0),r.absarc(0,e/2,t,0,.5*Math.PI),super(r.getPoints(i),n),this.type="CapsuleGeometry",this.parameters={radius:t,height:e,capSegments:i,radialSegments:n}}static fromJSON(t){return new Co(t.radius,t.length,t.capSegments,t.radialSegments)}}class Lo extends Di{constructor(t=1,e=8,i=0,n=2*Math.PI){super(),this.type="CircleGeometry",this.parameters={radius:t,segments:e,thetaStart:i,thetaLength:n},e=Math.max(3,e);const r=[],s=[],a=[],o=[],l=new ne,c=new Lt;s.push(0,0,0),a.push(0,0,1),o.push(.5,.5);for(let r=0,h=3;r<=e;r++,h+=3){const u=i+r/e*n;l.x=t*Math.cos(u),l.y=t*Math.sin(u),s.push(l.x,l.y,l.z),a.push(0,0,1),c.x=(s[h]/t+1)/2,c.y=(s[h+1]/t+1)/2,o.push(c.x,c.y)}for(let t=1;t<=e;t++)r.push(t,t+1,0);this.setIndex(r),this.setAttribute("position",new wi(s,3)),this.setAttribute("normal",new wi(a,3)),this.setAttribute("uv",new wi(o,2))}static fromJSON(t){return new Lo(t.radius,t.segments,t.thetaStart,t.thetaLength)}}class Ro extends Di{constructor(t=1,e=1,i=1,n=8,r=1,s=!1,a=0,o=2*Math.PI){super(),this.type="CylinderGeometry",this.parameters={radiusTop:t,radiusBottom:e,height:i,radialSegments:n,heightSegments:r,openEnded:s,thetaStart:a,thetaLength:o};const l=this;n=Math.floor(n),r=Math.floor(r);const c=[],h=[],u=[],d=[];let p=0;const m=[],f=i/2;let g=0;function v(i){const r=p,s=new Lt,m=new ne;let v=0;const x=!0===i?t:e,_=!0===i?1:-1;for(let t=1;t<=n;t++)h.push(0,f*_,0),u.push(0,_,0),d.push(.5,.5),p++;const y=p;for(let t=0;t<=n;t++){const e=t/n*o+a,i=Math.cos(e),r=Math.sin(e);m.x=x*r,m.y=f*_,m.z=x*i,h.push(m.x,m.y,m.z),u.push(0,_,0),s.x=.5*i+.5,s.y=.5*r*_+.5,d.push(s.x,s.y),p++}for(let t=0;t<n;t++){const e=r+t,n=y+t;!0===i?c.push(n,n+1,e):c.push(n+1,n,e),v+=3}l.addGroup(g,v,!0===i?1:2),g+=v}!function(){const s=new ne,v=new ne;let x=0;const _=(e-t)/i;for(let l=0;l<=r;l++){const c=[],g=l/r,x=g*(e-t)+t;for(let t=0;t<=n;t++){const e=t/n,r=e*o+a,l=Math.sin(r),m=Math.cos(r);v.x=x*l,v.y=-g*i+f,v.z=x*m,h.push(v.x,v.y,v.z),s.set(l,_,m).normalize(),u.push(s.x,s.y,s.z),d.push(e,1-g),c.push(p++)}m.push(c)}for(let t=0;t<n;t++)for(let e=0;e<r;e++){const i=m[e][t],n=m[e+1][t],r=m[e+1][t+1],s=m[e][t+1];c.push(i,n,s),c.push(n,r,s),x+=6}l.addGroup(g,x,0),g+=x}(),!1===s&&(t>0&&v(!0),e>0&&v(!1)),this.setIndex(c),this.setAttribute("position",new wi(h,3)),this.setAttribute("normal",new wi(u,3)),this.setAttribute("uv",new wi(d,2))}static fromJSON(t){return new Ro(t.radiusTop,t.radiusBottom,t.height,t.radialSegments,t.heightSegments,t.openEnded,t.thetaStart,t.thetaLength)}}class Po extends Ro{constructor(t=1,e=1,i=8,n=1,r=!1,s=0,a=2*Math.PI){super(0,t,e,i,n,r,s,a),this.type="ConeGeometry",this.parameters={radius:t,height:e,radialSegments:i,heightSegments:n,openEnded:r,thetaStart:s,thetaLength:a}}static fromJSON(t){return new Po(t.radius,t.height,t.radialSegments,t.heightSegments,t.openEnded,t.thetaStart,t.thetaLength)}}class Do extends Di{constructor(t=[],e=[],i=1,n=0){super(),this.type="PolyhedronGeometry",this.parameters={vertices:t,indices:e,radius:i,detail:n};const r=[],s=[];function a(t,e,i,n){const r=n+1,s=[];for(let n=0;n<=r;n++){s[n]=[];const a=t.clone().lerp(i,n/r),o=e.clone().lerp(i,n/r),l=r-n;for(let t=0;t<=l;t++)s[n][t]=0===t&&n===r?a:a.clone().lerp(o,t/l)}for(let t=0;t<r;t++)for(let e=0;e<2*(r-t)-1;e++){const i=Math.floor(e/2);e%2==0?(o(s[t][i+1]),o(s[t+1][i]),o(s[t][i])):(o(s[t][i+1]),o(s[t+1][i+1]),o(s[t+1][i]))}}function o(t){r.push(t.x,t.y,t.z)}function l(e,i){const n=3*e;i.x=t[n+0],i.y=t[n+1],i.z=t[n+2]}function c(t,e,i,n){n<0&&1===t.x&&(s[e]=t.x-1),0===i.x&&0===i.z&&(s[e]=n/2/Math.PI+.5)}function h(t){return Math.atan2(t.z,-t.x)}!function(t){const i=new ne,n=new ne,r=new ne;for(let s=0;s<e.length;s+=3)l(e[s+0],i),l(e[s+1],n),l(e[s+2],r),a(i,n,r,t)}(n),function(t){const e=new ne;for(let i=0;i<r.length;i+=3)e.x=r[i+0],e.y=r[i+1],e.z=r[i+2],e.normalize().multiplyScalar(t),r[i+0]=e.x,r[i+1]=e.y,r[i+2]=e.z}(i),function(){const t=new ne;for(let i=0;i<r.length;i+=3){t.x=r[i+0],t.y=r[i+1],t.z=r[i+2];const n=h(t)/2/Math.PI+.5,a=(e=t,Math.atan2(-e.y,Math.sqrt(e.x*e.x+e.z*e.z))/Math.PI+.5);s.push(n,1-a)}var e;(function(){const t=new ne,e=new ne,i=new ne,n=new ne,a=new Lt,o=new Lt,l=new Lt;for(let u=0,d=0;u<r.length;u+=9,d+=6){t.set(r[u+0],r[u+1],r[u+2]),e.set(r[u+3],r[u+4],r[u+5]),i.set(r[u+6],r[u+7],r[u+8]),a.set(s[d+0],s[d+1]),o.set(s[d+2],s[d+3]),l.set(s[d+4],s[d+5]),n.copy(t).add(e).add(i).divideScalar(3);const p=h(n);c(a,d+0,t,p),c(o,d+2,e,p),c(l,d+4,i,p)}})(),function(){for(let t=0;t<s.length;t+=6){const e=s[t+0],i=s[t+2],n=s[t+4],r=Math.max(e,i,n),a=Math.min(e,i,n);r>.9&&a<.1&&(e<.2&&(s[t+0]+=1),i<.2&&(s[t+2]+=1),n<.2&&(s[t+4]+=1))}}()}(),this.setAttribute("position",new wi(r,3)),this.setAttribute("normal",new wi(r.slice(),3)),this.setAttribute("uv",new wi(s,2)),0===n?this.computeVertexNormals():this.normalizeNormals()}static fromJSON(t){return new Do(t.vertices,t.indices,t.radius,t.details)}}class Io extends Do{constructor(t=1,e=0){const i=(1+Math.sqrt(5))/2,n=1/i;super([-1,-1,-1,-1,-1,1,-1,1,-1,-1,1,1,1,-1,-1,1,-1,1,1,1,-1,1,1,1,0,-n,-i,0,-n,i,0,n,-i,0,n,i,-n,-i,0,-n,i,0,n,-i,0,n,i,0,-i,0,-n,i,0,-n,-i,0,n,i,0,n],[3,11,7,3,7,15,3,15,13,7,19,17,7,17,6,7,6,15,17,4,8,17,8,10,17,10,6,8,0,16,8,16,2,8,2,10,0,12,1,0,1,18,0,18,16,6,10,2,6,2,13,6,13,15,2,16,18,2,18,3,2,3,13,18,1,9,18,9,11,18,11,3,4,14,12,4,12,0,4,0,8,11,9,5,11,5,19,11,19,7,19,5,14,19,14,4,19,4,17,1,12,14,1,14,5,1,5,9],t,e),this.type="DodecahedronGeometry",this.parameters={radius:t,detail:e}}static fromJSON(t){return new Io(t.radius,t.detail)}}const No=new ne,Oo=new ne,zo=new ne,Uo=new fi;class Bo extends Di{constructor(t=null,e=1){if(super(),this.type="EdgesGeometry",this.parameters={geometry:t,thresholdAngle:e},null!==t){const i=4,n=Math.pow(10,i),r=Math.cos(vt*e),s=t.getIndex(),a=t.getAttribute("position"),o=s?s.count:a.count,l=[0,0,0],c=["a","b","c"],h=new Array(3),u={},d=[];for(let t=0;t<o;t+=3){s?(l[0]=s.getX(t),l[1]=s.getX(t+1),l[2]=s.getX(t+2)):(l[0]=t,l[1]=t+1,l[2]=t+2);const{a:e,b:i,c:o}=Uo;if(e.fromBufferAttribute(a,l[0]),i.fromBufferAttribute(a,l[1]),o.fromBufferAttribute(a,l[2]),Uo.getNormal(zo),h[0]=`${Math.round(e.x*n)},${Math.round(e.y*n)},${Math.round(e.z*n)}`,h[1]=`${Math.round(i.x*n)},${Math.round(i.y*n)},${Math.round(i.z*n)}`,h[2]=`${Math.round(o.x*n)},${Math.round(o.y*n)},${Math.round(o.z*n)}`,h[0]!==h[1]&&h[1]!==h[2]&&h[2]!==h[0])for(let t=0;t<3;t++){const e=(t+1)%3,i=h[t],n=h[e],s=Uo[c[t]],a=Uo[c[e]],o=`${i}_${n}`,p=`${n}_${i}`;p in u&&u[p]?(zo.dot(u[p].normal)<=r&&(d.push(s.x,s.y,s.z),d.push(a.x,a.y,a.z)),u[p]=null):o in u||(u[o]={index0:l[t],index1:l[e],normal:zo.clone()})}}for(const t in u)if(u[t]){const{index0:e,index1:i}=u[t];No.fromBufferAttribute(a,e),Oo.fromBufferAttribute(a,i),d.push(No.x,No.y,No.z),d.push(Oo.x,Oo.y,Oo.z)}this.setAttribute("position",new wi(d,3))}}}class Fo extends Ao{constructor(t){super(t),this.uuid=_t(),this.type="Shape",this.holes=[]}getPointsHoles(t){const e=[];for(let i=0,n=this.holes.length;i<n;i++)e[i]=this.holes[i].getPoints(t);return e}extractPoints(t){return{shape:this.getPoints(t),holes:this.getPointsHoles(t)}}copy(t){super.copy(t),this.holes=[];for(let e=0,i=t.holes.length;e<i;e++){const i=t.holes[e];this.holes.push(i.clone())}return this}toJSON(){const t=super.toJSON();t.uuid=this.uuid,t.holes=[];for(let e=0,i=this.holes.length;e<i;e++){const i=this.holes[e];t.holes.push(i.toJSON())}return t}fromJSON(t){super.fromJSON(t),this.uuid=t.uuid,this.holes=[];for(let e=0,i=t.holes.length;e<i;e++){const i=t.holes[e];this.holes.push((new Ao).fromJSON(i))}return this}}const ko=function(t,e,i=2){const n=e&&e.length,r=n?e[0]*i:t.length;let s=Go(t,0,r,i,!0);const a=[];if(!s||s.next===s.prev)return a;let o,l,c,h,u,d,p;if(n&&(s=function(t,e,i,n){const r=[];let s,a,o,l,c;for(s=0,a=e.length;s<a;s++)o=e[s]*n,l=s<a-1?e[s+1]*n:t.length,c=Go(t,o,l,n,!1),c===c.next&&(c.steiner=!0),r.push($o(c));for(r.sort(Yo),s=0;s<r.length;s++)i=Zo(r[s],i);return i}(t,e,s,i)),t.length>80*i){o=c=t[0],l=h=t[1];for(let e=i;e<r;e+=i)u=t[e],d=t[e+1],u<o&&(o=u),d<l&&(l=d),u>c&&(c=u),d>h&&(h=d);p=Math.max(c-o,h-l),p=0!==p?32767/p:0}return Ho(s,a,i,o,l,p,0),a};function Go(t,e,i,n,r){let s,a;if(r===function(t,e,i,n){let r=0;for(let s=e,a=i-n;s<i;s+=n)r+=(t[a]-t[s])*(t[s+1]+t[a+1]),a=s;return r}(t,e,i,n)>0)for(s=e;s<i;s+=n)a=ll(s,t[s],t[s+1],a);else for(s=i-n;s>=e;s-=n)a=ll(s,t[s],t[s+1],a);return a&&il(a,a.next)&&(cl(a),a=a.next),a}function Vo(t,e){if(!t)return t;e||(e=t);let i,n=t;do{if(i=!1,n.steiner||!il(n,n.next)&&0!==el(n.prev,n,n.next))n=n.next;else{if(cl(n),n=e=n.prev,n===n.next)break;i=!0}}while(i||n!==e);return e}function Ho(t,e,i,n,r,s,a){if(!t)return;!a&&s&&function(t,e,i,n){let r=t;do{0===r.z&&(r.z=Ko(r.x,r.y,e,i,n)),r.prevZ=r.prev,r.nextZ=r.next,r=r.next}while(r!==t);r.prevZ.nextZ=null,r.prevZ=null,function(t){let e,i,n,r,s,a,o,l,c=1;do{for(i=t,t=null,s=null,a=0;i;){for(a++,n=i,o=0,e=0;e<c&&(o++,n=n.nextZ,n);e++);for(l=c;o>0||l>0&&n;)0!==o&&(0===l||!n||i.z<=n.z)?(r=i,i=i.nextZ,o--):(r=n,n=n.nextZ,l--),s?s.nextZ=r:t=r,r.prevZ=s,s=r;i=n}s.nextZ=null,c*=2}while(a>1)}(r)}(t,n,r,s);let o,l,c=t;for(;t.prev!==t.next;)if(o=t.prev,l=t.next,s?jo(t,n,r,s):Wo(t))e.push(o.i/i|0),e.push(t.i/i|0),e.push(l.i/i|0),cl(t),t=l.next,c=l.next;else if((t=l)===c){a?1===a?Ho(t=qo(Vo(t),e,i),e,i,n,r,s,2):2===a&&Xo(t,e,i,n,r,s):Ho(Vo(t),e,i,n,r,s,1);break}}function Wo(t){const e=t.prev,i=t,n=t.next;if(el(e,i,n)>=0)return!1;const r=e.x,s=i.x,a=n.x,o=e.y,l=i.y,c=n.y,h=r<s?r<a?r:a:s<a?s:a,u=o<l?o<c?o:c:l<c?l:c,d=r>s?r>a?r:a:s>a?s:a,p=o>l?o>c?o:c:l>c?l:c;let m=n.next;for(;m!==e;){if(m.x>=h&&m.x<=d&&m.y>=u&&m.y<=p&&Qo(r,o,s,l,a,c,m.x,m.y)&&el(m.prev,m,m.next)>=0)return!1;m=m.next}return!0}function jo(t,e,i,n){const r=t.prev,s=t,a=t.next;if(el(r,s,a)>=0)return!1;const o=r.x,l=s.x,c=a.x,h=r.y,u=s.y,d=a.y,p=o<l?o<c?o:c:l<c?l:c,m=h<u?h<d?h:d:u<d?u:d,f=o>l?o>c?o:c:l>c?l:c,g=h>u?h>d?h:d:u>d?u:d,v=Ko(p,m,e,i,n),x=Ko(f,g,e,i,n);let _=t.prevZ,y=t.nextZ;for(;_&&_.z>=v&&y&&y.z<=x;){if(_.x>=p&&_.x<=f&&_.y>=m&&_.y<=g&&_!==r&&_!==a&&Qo(o,h,l,u,c,d,_.x,_.y)&&el(_.prev,_,_.next)>=0)return!1;if(_=_.prevZ,y.x>=p&&y.x<=f&&y.y>=m&&y.y<=g&&y!==r&&y!==a&&Qo(o,h,l,u,c,d,y.x,y.y)&&el(y.prev,y,y.next)>=0)return!1;y=y.nextZ}for(;_&&_.z>=v;){if(_.x>=p&&_.x<=f&&_.y>=m&&_.y<=g&&_!==r&&_!==a&&Qo(o,h,l,u,c,d,_.x,_.y)&&el(_.prev,_,_.next)>=0)return!1;_=_.prevZ}for(;y&&y.z<=x;){if(y.x>=p&&y.x<=f&&y.y>=m&&y.y<=g&&y!==r&&y!==a&&Qo(o,h,l,u,c,d,y.x,y.y)&&el(y.prev,y,y.next)>=0)return!1;y=y.nextZ}return!0}function qo(t,e,i){let n=t;do{const r=n.prev,s=n.next.next;!il(r,s)&&nl(r,n,n.next,s)&&al(r,s)&&al(s,r)&&(e.push(r.i/i|0),e.push(n.i/i|0),e.push(s.i/i|0),cl(n),cl(n.next),n=t=s),n=n.next}while(n!==t);return Vo(n)}function Xo(t,e,i,n,r,s){let a=t;do{let t=a.next.next;for(;t!==a.prev;){if(a.i!==t.i&&tl(a,t)){let o=ol(a,t);return a=Vo(a,a.next),o=Vo(o,o.next),Ho(a,e,i,n,r,s,0),void Ho(o,e,i,n,r,s,0)}t=t.next}a=a.next}while(a!==t)}function Yo(t,e){return t.x-e.x}function Zo(t,e){const i=function(t,e){let i,n=e,r=-1/0;const s=t.x,a=t.y;do{if(a<=n.y&&a>=n.next.y&&n.next.y!==n.y){const t=n.x+(a-n.y)*(n.next.x-n.x)/(n.next.y-n.y);if(t<=s&&t>r&&(r=t,i=n.x<n.next.x?n:n.next,t===s))return i}n=n.next}while(n!==e);if(!i)return null;const o=i,l=i.x,c=i.y;let h,u=1/0;n=i;do{s>=n.x&&n.x>=l&&s!==n.x&&Qo(a<c?s:r,a,l,c,a<c?r:s,a,n.x,n.y)&&(h=Math.abs(a-n.y)/(s-n.x),al(n,t)&&(h<u||h===u&&(n.x>i.x||n.x===i.x&&Jo(i,n)))&&(i=n,u=h)),n=n.next}while(n!==o);return i}(t,e);if(!i)return e;const n=ol(i,t);return Vo(n,n.next),Vo(i,i.next)}function Jo(t,e){return el(t.prev,t,e.prev)<0&&el(e.next,t,t.next)<0}function Ko(t,e,i,n,r){return(t=1431655765&((t=858993459&((t=252645135&((t=16711935&((t=(t-i)*r|0)|t<<8))|t<<4))|t<<2))|t<<1))|(e=1431655765&((e=858993459&((e=252645135&((e=16711935&((e=(e-n)*r|0)|e<<8))|e<<4))|e<<2))|e<<1))<<1}function $o(t){let e=t,i=t;do{(e.x<i.x||e.x===i.x&&e.y<i.y)&&(i=e),e=e.next}while(e!==t);return i}function Qo(t,e,i,n,r,s,a,o){return(r-a)*(e-o)>=(t-a)*(s-o)&&(t-a)*(n-o)>=(i-a)*(e-o)&&(i-a)*(s-o)>=(r-a)*(n-o)}function tl(t,e){return t.next.i!==e.i&&t.prev.i!==e.i&&!function(t,e){let i=t;do{if(i.i!==t.i&&i.next.i!==t.i&&i.i!==e.i&&i.next.i!==e.i&&nl(i,i.next,t,e))return!0;i=i.next}while(i!==t);return!1}(t,e)&&(al(t,e)&&al(e,t)&&function(t,e){let i=t,n=!1;const r=(t.x+e.x)/2,s=(t.y+e.y)/2;do{i.y>s!=i.next.y>s&&i.next.y!==i.y&&r<(i.next.x-i.x)*(s-i.y)/(i.next.y-i.y)+i.x&&(n=!n),i=i.next}while(i!==t);return n}(t,e)&&(el(t.prev,t,e.prev)||el(t,e.prev,e))||il(t,e)&&el(t.prev,t,t.next)>0&&el(e.prev,e,e.next)>0)}function el(t,e,i){return(e.y-t.y)*(i.x-e.x)-(e.x-t.x)*(i.y-e.y)}function il(t,e){return t.x===e.x&&t.y===e.y}function nl(t,e,i,n){const r=sl(el(t,e,i)),s=sl(el(t,e,n)),a=sl(el(i,n,t)),o=sl(el(i,n,e));return r!==s&&a!==o||(!(0!==r||!rl(t,i,e))||(!(0!==s||!rl(t,n,e))||(!(0!==a||!rl(i,t,n))||!(0!==o||!rl(i,e,n)))))}function rl(t,e,i){return e.x<=Math.max(t.x,i.x)&&e.x>=Math.min(t.x,i.x)&&e.y<=Math.max(t.y,i.y)&&e.y>=Math.min(t.y,i.y)}function sl(t){return t>0?1:t<0?-1:0}function al(t,e){return el(t.prev,t,t.next)<0?el(t,e,t.next)>=0&&el(t,t.prev,e)>=0:el(t,e,t.prev)<0||el(t,t.next,e)<0}function ol(t,e){const i=new hl(t.i,t.x,t.y),n=new hl(e.i,e.x,e.y),r=t.next,s=e.prev;return t.next=e,e.prev=t,i.next=r,r.prev=i,n.next=i,i.prev=n,s.next=n,n.prev=s,n}function ll(t,e,i,n){const r=new hl(t,e,i);return n?(r.next=n.next,r.prev=n,n.next.prev=r,n.next=r):(r.prev=r,r.next=r),r}function cl(t){t.next.prev=t.prev,t.prev.next=t.next,t.prevZ&&(t.prevZ.nextZ=t.nextZ),t.nextZ&&(t.nextZ.prevZ=t.prevZ)}function hl(t,e,i){this.i=t,this.x=e,this.y=i,this.prev=null,this.next=null,this.z=0,this.prevZ=null,this.nextZ=null,this.steiner=!1}class ul{static area(t){const e=t.length;let i=0;for(let n=e-1,r=0;r<e;n=r++)i+=t[n].x*t[r].y-t[r].x*t[n].y;return.5*i}static isClockWise(t){return ul.area(t)<0}static triangulateShape(t,e){const i=[],n=[],r=[];dl(t),pl(i,t);let s=t.length;e.forEach(dl);for(let t=0;t<e.length;t++)n.push(s),s+=e[t].length,pl(i,e[t]);const a=ko(i,n);for(let t=0;t<a.length;t+=3)r.push(a.slice(t,t+3));return r}}function dl(t){const e=t.length;e>2&&t[e-1].equals(t[0])&&t.pop()}function pl(t,e){for(let i=0;i<e.length;i++)t.push(e[i].x),t.push(e[i].y)}class ml extends Di{constructor(t=new Fo([new Lt(.5,.5),new Lt(-.5,.5),new Lt(-.5,-.5),new Lt(.5,-.5)]),e={}){super(),this.type="ExtrudeGeometry",this.parameters={shapes:t,options:e},t=Array.isArray(t)?t:[t];const i=this,n=[],r=[];for(let e=0,i=t.length;e<i;e++){s(t[e])}function s(t){const s=[],a=void 0!==e.curveSegments?e.curveSegments:12,o=void 0!==e.steps?e.steps:1,l=void 0!==e.depth?e.depth:1;let c=void 0===e.bevelEnabled||e.bevelEnabled,h=void 0!==e.bevelThickness?e.bevelThickness:.2,u=void 0!==e.bevelSize?e.bevelSize:h-.1,d=void 0!==e.bevelOffset?e.bevelOffset:0,p=void 0!==e.bevelSegments?e.bevelSegments:3;const m=e.extrudePath,f=void 0!==e.UVGenerator?e.UVGenerator:fl;let g,v,x,_,y,M=!1;m&&(g=m.getSpacedPoints(o),M=!0,c=!1,v=m.computeFrenetFrames(o,!1),x=new ne,_=new ne,y=new ne),c||(p=0,h=0,u=0,d=0);const b=t.extractPoints(a);let S=b.shape;const w=b.holes;if(!ul.isClockWise(S)){S=S.reverse();for(let t=0,e=w.length;t<e;t++){const e=w[t];ul.isClockWise(e)&&(w[t]=e.reverse())}}const T=ul.triangulateShape(S,w),A=S;for(let t=0,e=w.length;t<e;t++){const e=w[t];S=S.concat(e)}function E(t,e,i){return e||console.error("THREE.ExtrudeGeometry: vec does not exist"),e.clone().multiplyScalar(i).add(t)}const C=S.length,L=T.length;function R(t,e,i){let n,r,s;const a=t.x-e.x,o=t.y-e.y,l=i.x-t.x,c=i.y-t.y,h=a*a+o*o,u=a*c-o*l;if(Math.abs(u)>Number.EPSILON){const u=Math.sqrt(h),d=Math.sqrt(l*l+c*c),p=e.x-o/u,m=e.y+a/u,f=((i.x-c/d-p)*c-(i.y+l/d-m)*l)/(a*c-o*l);n=p+a*f-t.x,r=m+o*f-t.y;const g=n*n+r*r;if(g<=2)return new Lt(n,r);s=Math.sqrt(g/2)}else{let t=!1;a>Number.EPSILON?l>Number.EPSILON&&(t=!0):a<-Number.EPSILON?l<-Number.EPSILON&&(t=!0):Math.sign(o)===Math.sign(c)&&(t=!0),t?(n=-o,r=a,s=Math.sqrt(h)):(n=a,r=o,s=Math.sqrt(h/2))}return new Lt(n/s,r/s)}const P=[];for(let t=0,e=A.length,i=e-1,n=t+1;t<e;t++,i++,n++)i===e&&(i=0),n===e&&(n=0),P[t]=R(A[t],A[i],A[n]);const D=[];let I,N=P.concat();for(let t=0,e=w.length;t<e;t++){const e=w[t];I=[];for(let t=0,i=e.length,n=i-1,r=t+1;t<i;t++,n++,r++)n===i&&(n=0),r===i&&(r=0),I[t]=R(e[t],e[n],e[r]);D.push(I),N=N.concat(I)}for(let t=0;t<p;t++){const e=t/p,i=h*Math.cos(e*Math.PI/2),n=u*Math.sin(e*Math.PI/2)+d;for(let t=0,e=A.length;t<e;t++){const e=E(A[t],P[t],n);U(e.x,e.y,-i)}for(let t=0,e=w.length;t<e;t++){const e=w[t];I=D[t];for(let t=0,r=e.length;t<r;t++){const r=E(e[t],I[t],n);U(r.x,r.y,-i)}}}const O=u+d;for(let t=0;t<C;t++){const e=c?E(S[t],N[t],O):S[t];M?(_.copy(v.normals[0]).multiplyScalar(e.x),x.copy(v.binormals[0]).multiplyScalar(e.y),y.copy(g[0]).add(_).add(x),U(y.x,y.y,y.z)):U(e.x,e.y,0)}for(let t=1;t<=o;t++)for(let e=0;e<C;e++){const i=c?E(S[e],N[e],O):S[e];M?(_.copy(v.normals[t]).multiplyScalar(i.x),x.copy(v.binormals[t]).multiplyScalar(i.y),y.copy(g[t]).add(_).add(x),U(y.x,y.y,y.z)):U(i.x,i.y,l/o*t)}for(let t=p-1;t>=0;t--){const e=t/p,i=h*Math.cos(e*Math.PI/2),n=u*Math.sin(e*Math.PI/2)+d;for(let t=0,e=A.length;t<e;t++){const e=E(A[t],P[t],n);U(e.x,e.y,l+i)}for(let t=0,e=w.length;t<e;t++){const e=w[t];I=D[t];for(let t=0,r=e.length;t<r;t++){const r=E(e[t],I[t],n);M?U(r.x,r.y+g[o-1].y,g[o-1].x+i):U(r.x,r.y,l+i)}}}function z(t,e){let i=t.length;for(;--i>=0;){const n=i;let r=i-1;r<0&&(r=t.length-1);for(let t=0,i=o+2*p;t<i;t++){const i=C*t,s=C*(t+1);F(e+n+i,e+r+i,e+r+s,e+n+s)}}}function U(t,e,i){s.push(t),s.push(e),s.push(i)}function B(t,e,r){k(t),k(e),k(r);const s=n.length/3,a=f.generateTopUV(i,n,s-3,s-2,s-1);G(a[0]),G(a[1]),G(a[2])}function F(t,e,r,s){k(t),k(e),k(s),k(e),k(r),k(s);const a=n.length/3,o=f.generateSideWallUV(i,n,a-6,a-3,a-2,a-1);G(o[0]),G(o[1]),G(o[3]),G(o[1]),G(o[2]),G(o[3])}function k(t){n.push(s[3*t+0]),n.push(s[3*t+1]),n.push(s[3*t+2])}function G(t){r.push(t.x),r.push(t.y)}!function(){const t=n.length/3;if(c){let t=0,e=C*t;for(let t=0;t<L;t++){const i=T[t];B(i[2]+e,i[1]+e,i[0]+e)}t=o+2*p,e=C*t;for(let t=0;t<L;t++){const i=T[t];B(i[0]+e,i[1]+e,i[2]+e)}}else{for(let t=0;t<L;t++){const e=T[t];B(e[2],e[1],e[0])}for(let t=0;t<L;t++){const e=T[t];B(e[0]+C*o,e[1]+C*o,e[2]+C*o)}}i.addGroup(t,n.length/3-t,0)}(),function(){const t=n.length/3;let e=0;z(A,e),e+=A.length;for(let t=0,i=w.length;t<i;t++){const i=w[t];z(i,e),e+=i.length}i.addGroup(t,n.length/3-t,1)}()}this.setAttribute("position",new wi(n,3)),this.setAttribute("uv",new wi(r,2)),this.computeVertexNormals()}toJSON(){const t=super.toJSON();return function(t,e,i){if(i.shapes=[],Array.isArray(t))for(let e=0,n=t.length;e<n;e++){const n=t[e];i.shapes.push(n.uuid)}else i.shapes.push(t.uuid);i.options=Object.assign({},e),void 0!==e.extrudePath&&(i.options.extrudePath=e.extrudePath.toJSON());return i}(this.parameters.shapes,this.parameters.options,t)}static fromJSON(t,e){const i=[];for(let n=0,r=t.shapes.length;n<r;n++){const r=e[t.shapes[n]];i.push(r)}const n=t.options.extrudePath;return void 0!==n&&(t.options.extrudePath=(new wo[n.type]).fromJSON(n)),new ml(i,t.options)}}const fl={generateTopUV:function(t,e,i,n,r){const s=e[3*i],a=e[3*i+1],o=e[3*n],l=e[3*n+1],c=e[3*r],h=e[3*r+1];return[new Lt(s,a),new Lt(o,l),new Lt(c,h)]},generateSideWallUV:function(t,e,i,n,r,s){const a=e[3*i],o=e[3*i+1],l=e[3*i+2],c=e[3*n],h=e[3*n+1],u=e[3*n+2],d=e[3*r],p=e[3*r+1],m=e[3*r+2],f=e[3*s],g=e[3*s+1],v=e[3*s+2];return Math.abs(o-h)<Math.abs(a-c)?[new Lt(a,1-l),new Lt(c,1-u),new Lt(d,1-m),new Lt(f,1-v)]:[new Lt(o,1-l),new Lt(h,1-u),new Lt(p,1-m),new Lt(g,1-v)]}};class gl extends Do{constructor(t=1,e=0){const i=(1+Math.sqrt(5))/2;super([-1,i,0,1,i,0,-1,-i,0,1,-i,0,0,-1,i,0,1,i,0,-1,-i,0,1,-i,i,0,-1,i,0,1,-i,0,-1,-i,0,1],[0,11,5,0,5,1,0,1,7,0,7,10,0,10,11,1,5,9,5,11,4,11,10,2,10,7,6,7,1,8,3,9,4,3,4,2,3,2,6,3,6,8,3,8,9,4,9,5,2,4,11,6,2,10,8,6,7,9,8,1],t,e),this.type="IcosahedronGeometry",this.parameters={radius:t,detail:e}}static fromJSON(t){return new gl(t.radius,t.detail)}}class vl extends Do{constructor(t=1,e=0){super([1,0,0,-1,0,0,0,1,0,0,-1,0,0,0,1,0,0,-1],[0,2,4,0,4,3,0,3,5,0,5,2,1,2,5,1,5,3,1,3,4,1,4,2],t,e),this.type="OctahedronGeometry",this.parameters={radius:t,detail:e}}static fromJSON(t){return new vl(t.radius,t.detail)}}class xl extends Di{constructor(t=.5,e=1,i=8,n=1,r=0,s=2*Math.PI){super(),this.type="RingGeometry",this.parameters={innerRadius:t,outerRadius:e,thetaSegments:i,phiSegments:n,thetaStart:r,thetaLength:s},i=Math.max(3,i);const a=[],o=[],l=[],c=[];let h=t;const u=(e-t)/(n=Math.max(1,n)),d=new ne,p=new Lt;for(let t=0;t<=n;t++){for(let t=0;t<=i;t++){const n=r+t/i*s;d.x=h*Math.cos(n),d.y=h*Math.sin(n),o.push(d.x,d.y,d.z),l.push(0,0,1),p.x=(d.x/e+1)/2,p.y=(d.y/e+1)/2,c.push(p.x,p.y)}h+=u}for(let t=0;t<n;t++){const e=t*(i+1);for(let t=0;t<i;t++){const n=t+e,r=n,s=n+i+1,o=n+i+2,l=n+1;a.push(r,s,l),a.push(s,o,l)}}this.setIndex(a),this.setAttribute("position",new wi(o,3)),this.setAttribute("normal",new wi(l,3)),this.setAttribute("uv",new wi(c,2))}static fromJSON(t){return new xl(t.innerRadius,t.outerRadius,t.thetaSegments,t.phiSegments,t.thetaStart,t.thetaLength)}}class _l extends Di{constructor(t=new Fo([new Lt(0,.5),new Lt(-.5,-.5),new Lt(.5,-.5)]),e=12){super(),this.type="ShapeGeometry",this.parameters={shapes:t,curveSegments:e};const i=[],n=[],r=[],s=[];let a=0,o=0;if(!1===Array.isArray(t))l(t);else for(let e=0;e<t.length;e++)l(t[e]),this.addGroup(a,o,e),a+=o,o=0;function l(t){const a=n.length/3,l=t.extractPoints(e);let c=l.shape;const h=l.holes;!1===ul.isClockWise(c)&&(c=c.reverse());for(let t=0,e=h.length;t<e;t++){const e=h[t];!0===ul.isClockWise(e)&&(h[t]=e.reverse())}const u=ul.triangulateShape(c,h);for(let t=0,e=h.length;t<e;t++){const e=h[t];c=c.concat(e)}for(let t=0,e=c.length;t<e;t++){const e=c[t];n.push(e.x,e.y,0),r.push(0,0,1),s.push(e.x,e.y)}for(let t=0,e=u.length;t<e;t++){const e=u[t],n=e[0]+a,r=e[1]+a,s=e[2]+a;i.push(n,r,s),o+=3}}this.setIndex(i),this.setAttribute("position",new wi(n,3)),this.setAttribute("normal",new wi(r,3)),this.setAttribute("uv",new wi(s,2))}toJSON(){const t=super.toJSON();return function(t,e){if(e.shapes=[],Array.isArray(t))for(let i=0,n=t.length;i<n;i++){const n=t[i];e.shapes.push(n.uuid)}else e.shapes.push(t.uuid);return e}(this.parameters.shapes,t)}static fromJSON(t,e){const i=[];for(let n=0,r=t.shapes.length;n<r;n++){const r=e[t.shapes[n]];i.push(r)}return new _l(i,t.curveSegments)}}class yl extends Di{constructor(t=1,e=32,i=16,n=0,r=2*Math.PI,s=0,a=Math.PI){super(),this.type="SphereGeometry",this.parameters={radius:t,widthSegments:e,heightSegments:i,phiStart:n,phiLength:r,thetaStart:s,thetaLength:a},e=Math.max(3,Math.floor(e)),i=Math.max(2,Math.floor(i));const o=Math.min(s+a,Math.PI);let l=0;const c=[],h=new ne,u=new ne,d=[],p=[],m=[],f=[];for(let d=0;d<=i;d++){const g=[],v=d/i;let x=0;0==d&&0==s?x=.5/e:d==i&&o==Math.PI&&(x=-.5/e);for(let i=0;i<=e;i++){const o=i/e;h.x=-t*Math.cos(n+o*r)*Math.sin(s+v*a),h.y=t*Math.cos(s+v*a),h.z=t*Math.sin(n+o*r)*Math.sin(s+v*a),p.push(h.x,h.y,h.z),u.copy(h).normalize(),m.push(u.x,u.y,u.z),f.push(o+x,1-v),g.push(l++)}c.push(g)}for(let t=0;t<i;t++)for(let n=0;n<e;n++){const e=c[t][n+1],r=c[t][n],a=c[t+1][n],l=c[t+1][n+1];(0!==t||s>0)&&d.push(e,r,l),(t!==i-1||o<Math.PI)&&d.push(r,a,l)}this.setIndex(d),this.setAttribute("position",new wi(p,3)),this.setAttribute("normal",new wi(m,3)),this.setAttribute("uv",new wi(f,2))}static fromJSON(t){return new yl(t.radius,t.widthSegments,t.heightSegments,t.phiStart,t.phiLength,t.thetaStart,t.thetaLength)}}class Ml extends Do{constructor(t=1,e=0){super([1,1,1,-1,-1,1,-1,1,-1,1,-1,-1],[2,1,0,0,3,2,1,3,0,2,3,1],t,e),this.type="TetrahedronGeometry",this.parameters={radius:t,detail:e}}static fromJSON(t){return new Ml(t.radius,t.detail)}}class bl extends Di{constructor(t=1,e=.4,i=8,n=6,r=2*Math.PI){super(),this.type="TorusGeometry",this.parameters={radius:t,tube:e,radialSegments:i,tubularSegments:n,arc:r},i=Math.floor(i),n=Math.floor(n);const s=[],a=[],o=[],l=[],c=new ne,h=new ne,u=new ne;for(let s=0;s<=i;s++)for(let d=0;d<=n;d++){const p=d/n*r,m=s/i*Math.PI*2;h.x=(t+e*Math.cos(m))*Math.cos(p),h.y=(t+e*Math.cos(m))*Math.sin(p),h.z=e*Math.sin(m),a.push(h.x,h.y,h.z),c.x=t*Math.cos(p),c.y=t*Math.sin(p),u.subVectors(h,c).normalize(),o.push(u.x,u.y,u.z),l.push(d/n),l.push(s/i)}for(let t=1;t<=i;t++)for(let e=1;e<=n;e++){const i=(n+1)*t+e-1,r=(n+1)*(t-1)+e-1,a=(n+1)*(t-1)+e,o=(n+1)*t+e;s.push(i,r,o),s.push(r,a,o)}this.setIndex(s),this.setAttribute("position",new wi(a,3)),this.setAttribute("normal",new wi(o,3)),this.setAttribute("uv",new wi(l,2))}static fromJSON(t){return new bl(t.radius,t.tube,t.radialSegments,t.tubularSegments,t.arc)}}class Sl extends Di{constructor(t=1,e=.4,i=64,n=8,r=2,s=3){super(),this.type="TorusKnotGeometry",this.parameters={radius:t,tube:e,tubularSegments:i,radialSegments:n,p:r,q:s},i=Math.floor(i),n=Math.floor(n);const a=[],o=[],l=[],c=[],h=new ne,u=new ne,d=new ne,p=new ne,m=new ne,f=new ne,g=new ne;for(let a=0;a<=i;++a){const x=a/i*r*Math.PI*2;v(x,r,s,t,d),v(x+.01,r,s,t,p),f.subVectors(p,d),g.addVectors(p,d),m.crossVectors(f,g),g.crossVectors(m,f),m.normalize(),g.normalize();for(let t=0;t<=n;++t){const r=t/n*Math.PI*2,s=-e*Math.cos(r),p=e*Math.sin(r);h.x=d.x+(s*g.x+p*m.x),h.y=d.y+(s*g.y+p*m.y),h.z=d.z+(s*g.z+p*m.z),o.push(h.x,h.y,h.z),u.subVectors(h,d).normalize(),l.push(u.x,u.y,u.z),c.push(a/i),c.push(t/n)}}for(let t=1;t<=i;t++)for(let e=1;e<=n;e++){const i=(n+1)*(t-1)+(e-1),r=(n+1)*t+(e-1),s=(n+1)*t+e,o=(n+1)*(t-1)+e;a.push(i,r,o),a.push(r,s,o)}function v(t,e,i,n,r){const s=Math.cos(t),a=Math.sin(t),o=i/e*t,l=Math.cos(o);r.x=n*(2+l)*.5*s,r.y=n*(2+l)*a*.5,r.z=n*Math.sin(o)*.5}this.setIndex(a),this.setAttribute("position",new wi(o,3)),this.setAttribute("normal",new wi(l,3)),this.setAttribute("uv",new wi(c,2))}static fromJSON(t){return new Sl(t.radius,t.tube,t.tubularSegments,t.radialSegments,t.p,t.q)}}class wl extends Di{constructor(t=new bo(new ne(-1,-1,0),new ne(-1,1,0),new ne(1,1,0)),e=64,i=1,n=8,r=!1){super(),this.type="TubeGeometry",this.parameters={path:t,tubularSegments:e,radius:i,radialSegments:n,closed:r};const s=t.computeFrenetFrames(e,r);this.tangents=s.tangents,this.normals=s.normals,this.binormals=s.binormals;const a=new ne,o=new ne,l=new Lt;let c=new ne;const h=[],u=[],d=[],p=[];function m(r){c=t.getPointAt(r/e,c);const l=s.normals[r],d=s.binormals[r];for(let t=0;t<=n;t++){const e=t/n*Math.PI*2,r=Math.sin(e),s=-Math.cos(e);o.x=s*l.x+r*d.x,o.y=s*l.y+r*d.y,o.z=s*l.z+r*d.z,o.normalize(),u.push(o.x,o.y,o.z),a.x=c.x+i*o.x,a.y=c.y+i*o.y,a.z=c.z+i*o.z,h.push(a.x,a.y,a.z)}}!function(){for(let t=0;t<e;t++)m(t);m(!1===r?e:0),function(){for(let t=0;t<=e;t++)for(let i=0;i<=n;i++)l.x=t/e,l.y=i/n,d.push(l.x,l.y)}(),function(){for(let t=1;t<=e;t++)for(let e=1;e<=n;e++){const i=(n+1)*(t-1)+(e-1),r=(n+1)*t+(e-1),s=(n+1)*t+e,a=(n+1)*(t-1)+e;p.push(i,r,a),p.push(r,s,a)}}()}(),this.setIndex(p),this.setAttribute("position",new wi(h,3)),this.setAttribute("normal",new wi(u,3)),this.setAttribute("uv",new wi(d,2))}toJSON(){const t=super.toJSON();return t.path=this.parameters.path.toJSON(),t}static fromJSON(t){return new wl((new wo[t.path.type]).fromJSON(t.path),t.tubularSegments,t.radius,t.radialSegments,t.closed)}}class Tl extends Di{constructor(t=null){if(super(),this.type="WireframeGeometry",this.parameters={geometry:t},null!==t){const e=[],i=new Set,n=new ne,r=new ne;if(null!==t.index){const s=t.attributes.position,a=t.index;let o=t.groups;0===o.length&&(o=[{start:0,count:a.count,materialIndex:0}]);for(let t=0,l=o.length;t<l;++t){const l=o[t],c=l.start;for(let t=c,o=c+l.count;t<o;t+=3)for(let o=0;o<3;o++){const l=a.getX(t+o),c=a.getX(t+(o+1)%3);n.fromBufferAttribute(s,l),r.fromBufferAttribute(s,c),!0===Al(n,r,i)&&(e.push(n.x,n.y,n.z),e.push(r.x,r.y,r.z))}}}else{const s=t.attributes.position;for(let t=0,a=s.count/3;t<a;t++)for(let a=0;a<3;a++){const o=3*t+a,l=3*t+(a+1)%3;n.fromBufferAttribute(s,o),r.fromBufferAttribute(s,l),!0===Al(n,r,i)&&(e.push(n.x,n.y,n.z),e.push(r.x,r.y,r.z))}}this.setAttribute("position",new wi(e,3))}}}function Al(t,e,i){const n=`${t.x},${t.y},${t.z}-${e.x},${e.y},${e.z}`,r=`${e.x},${e.y},${e.z}-${t.x},${t.y},${t.z}`;return!0!==i.has(n)&&!0!==i.has(r)&&(i.add(n),i.add(r),!0)}var El=Object.freeze({__proto__:null,BoxGeometry:$i,CapsuleGeometry:Co,CircleGeometry:Lo,ConeGeometry:Po,CylinderGeometry:Ro,DodecahedronGeometry:Io,EdgesGeometry:Bo,ExtrudeGeometry:ml,IcosahedronGeometry:gl,LatheGeometry:Eo,OctahedronGeometry:vl,PlaneGeometry:_n,PolyhedronGeometry:Do,RingGeometry:xl,ShapeGeometry:_l,SphereGeometry:yl,TetrahedronGeometry:Ml,TorusGeometry:bl,TorusKnotGeometry:Sl,TubeGeometry:wl,WireframeGeometry:Tl});class Cl extends vi{constructor(t){super(),this.isShadowMaterial=!0,this.type="ShadowMaterial",this.color=new jt(0),this.transparent=!0,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.fog=t.fog,this}}class Ll extends nn{constructor(t){super(t),this.isRawShaderMaterial=!0,this.type="RawShaderMaterial"}}class Rl extends vi{constructor(t){super(),this.isMeshStandardMaterial=!0,this.defines={STANDARD:""},this.type="MeshStandardMaterial",this.color=new jt(16777215),this.roughness=1,this.metalness=0,this.map=null,this.lightMap=null,this.lightMapIntensity=1,this.aoMap=null,this.aoMapIntensity=1,this.emissive=new jt(0),this.emissiveIntensity=1,this.emissiveMap=null,this.bumpMap=null,this.bumpScale=1,this.normalMap=null,this.normalMapType=0,this.normalScale=new Lt(1,1),this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.roughnessMap=null,this.metalnessMap=null,this.alphaMap=null,this.envMap=null,this.envMapIntensity=1,this.wireframe=!1,this.wireframeLinewidth=1,this.wireframeLinecap="round",this.wireframeLinejoin="round",this.flatShading=!1,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.defines={STANDARD:""},this.color.copy(t.color),this.roughness=t.roughness,this.metalness=t.metalness,this.map=t.map,this.lightMap=t.lightMap,this.lightMapIntensity=t.lightMapIntensity,this.aoMap=t.aoMap,this.aoMapIntensity=t.aoMapIntensity,this.emissive.copy(t.emissive),this.emissiveMap=t.emissiveMap,this.emissiveIntensity=t.emissiveIntensity,this.bumpMap=t.bumpMap,this.bumpScale=t.bumpScale,this.normalMap=t.normalMap,this.normalMapType=t.normalMapType,this.normalScale.copy(t.normalScale),this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.roughnessMap=t.roughnessMap,this.metalnessMap=t.metalnessMap,this.alphaMap=t.alphaMap,this.envMap=t.envMap,this.envMapIntensity=t.envMapIntensity,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.wireframeLinecap=t.wireframeLinecap,this.wireframeLinejoin=t.wireframeLinejoin,this.flatShading=t.flatShading,this.fog=t.fog,this}}class Pl extends Rl{constructor(t){super(),this.isMeshPhysicalMaterial=!0,this.defines={STANDARD:"",PHYSICAL:""},this.type="MeshPhysicalMaterial",this.clearcoatMap=null,this.clearcoatRoughness=0,this.clearcoatRoughnessMap=null,this.clearcoatNormalScale=new Lt(1,1),this.clearcoatNormalMap=null,this.ior=1.5,Object.defineProperty(this,"reflectivity",{get:function(){return yt(2.5*(this.ior-1)/(this.ior+1),0,1)},set:function(t){this.ior=(1+.4*t)/(1-.4*t)}}),this.iridescenceMap=null,this.iridescenceIOR=1.3,this.iridescenceThicknessRange=[100,400],this.iridescenceThicknessMap=null,this.sheenColor=new jt(0),this.sheenColorMap=null,this.sheenRoughness=1,this.sheenRoughnessMap=null,this.transmissionMap=null,this.thickness=0,this.thicknessMap=null,this.attenuationDistance=1/0,this.attenuationColor=new jt(1,1,1),this.specularIntensity=1,this.specularIntensityMap=null,this.specularColor=new jt(1,1,1),this.specularColorMap=null,this._sheen=0,this._clearcoat=0,this._iridescence=0,this._transmission=0,this.setValues(t)}get sheen(){return this._sheen}set sheen(t){this._sheen>0!=t>0&&this.version++,this._sheen=t}get clearcoat(){return this._clearcoat}set clearcoat(t){this._clearcoat>0!=t>0&&this.version++,this._clearcoat=t}get iridescence(){return this._iridescence}set iridescence(t){this._iridescence>0!=t>0&&this.version++,this._iridescence=t}get transmission(){return this._transmission}set transmission(t){this._transmission>0!=t>0&&this.version++,this._transmission=t}copy(t){return super.copy(t),this.defines={STANDARD:"",PHYSICAL:""},this.clearcoat=t.clearcoat,this.clearcoatMap=t.clearcoatMap,this.clearcoatRoughness=t.clearcoatRoughness,this.clearcoatRoughnessMap=t.clearcoatRoughnessMap,this.clearcoatNormalMap=t.clearcoatNormalMap,this.clearcoatNormalScale.copy(t.clearcoatNormalScale),this.ior=t.ior,this.iridescence=t.iridescence,this.iridescenceMap=t.iridescenceMap,this.iridescenceIOR=t.iridescenceIOR,this.iridescenceThicknessRange=[...t.iridescenceThicknessRange],this.iridescenceThicknessMap=t.iridescenceThicknessMap,this.sheen=t.sheen,this.sheenColor.copy(t.sheenColor),this.sheenColorMap=t.sheenColorMap,this.sheenRoughness=t.sheenRoughness,this.sheenRoughnessMap=t.sheenRoughnessMap,this.transmission=t.transmission,this.transmissionMap=t.transmissionMap,this.thickness=t.thickness,this.thicknessMap=t.thicknessMap,this.attenuationDistance=t.attenuationDistance,this.attenuationColor.copy(t.attenuationColor),this.specularIntensity=t.specularIntensity,this.specularIntensityMap=t.specularIntensityMap,this.specularColor.copy(t.specularColor),this.specularColorMap=t.specularColorMap,this}}class Dl extends vi{constructor(t){super(),this.isMeshPhongMaterial=!0,this.type="MeshPhongMaterial",this.color=new jt(16777215),this.specular=new jt(1118481),this.shininess=30,this.map=null,this.lightMap=null,this.lightMapIntensity=1,this.aoMap=null,this.aoMapIntensity=1,this.emissive=new jt(0),this.emissiveIntensity=1,this.emissiveMap=null,this.bumpMap=null,this.bumpScale=1,this.normalMap=null,this.normalMapType=0,this.normalScale=new Lt(1,1),this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.specularMap=null,this.alphaMap=null,this.envMap=null,this.combine=0,this.reflectivity=1,this.refractionRatio=.98,this.wireframe=!1,this.wireframeLinewidth=1,this.wireframeLinecap="round",this.wireframeLinejoin="round",this.flatShading=!1,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.specular.copy(t.specular),this.shininess=t.shininess,this.map=t.map,this.lightMap=t.lightMap,this.lightMapIntensity=t.lightMapIntensity,this.aoMap=t.aoMap,this.aoMapIntensity=t.aoMapIntensity,this.emissive.copy(t.emissive),this.emissiveMap=t.emissiveMap,this.emissiveIntensity=t.emissiveIntensity,this.bumpMap=t.bumpMap,this.bumpScale=t.bumpScale,this.normalMap=t.normalMap,this.normalMapType=t.normalMapType,this.normalScale.copy(t.normalScale),this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.specularMap=t.specularMap,this.alphaMap=t.alphaMap,this.envMap=t.envMap,this.combine=t.combine,this.reflectivity=t.reflectivity,this.refractionRatio=t.refractionRatio,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.wireframeLinecap=t.wireframeLinecap,this.wireframeLinejoin=t.wireframeLinejoin,this.flatShading=t.flatShading,this.fog=t.fog,this}}class Il extends vi{constructor(t){super(),this.isMeshToonMaterial=!0,this.defines={TOON:""},this.type="MeshToonMaterial",this.color=new jt(16777215),this.map=null,this.gradientMap=null,this.lightMap=null,this.lightMapIntensity=1,this.aoMap=null,this.aoMapIntensity=1,this.emissive=new jt(0),this.emissiveIntensity=1,this.emissiveMap=null,this.bumpMap=null,this.bumpScale=1,this.normalMap=null,this.normalMapType=0,this.normalScale=new Lt(1,1),this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.alphaMap=null,this.wireframe=!1,this.wireframeLinewidth=1,this.wireframeLinecap="round",this.wireframeLinejoin="round",this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.map=t.map,this.gradientMap=t.gradientMap,this.lightMap=t.lightMap,this.lightMapIntensity=t.lightMapIntensity,this.aoMap=t.aoMap,this.aoMapIntensity=t.aoMapIntensity,this.emissive.copy(t.emissive),this.emissiveMap=t.emissiveMap,this.emissiveIntensity=t.emissiveIntensity,this.bumpMap=t.bumpMap,this.bumpScale=t.bumpScale,this.normalMap=t.normalMap,this.normalMapType=t.normalMapType,this.normalScale.copy(t.normalScale),this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.alphaMap=t.alphaMap,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.wireframeLinecap=t.wireframeLinecap,this.wireframeLinejoin=t.wireframeLinejoin,this.fog=t.fog,this}}class Nl extends vi{constructor(t){super(),this.isMeshNormalMaterial=!0,this.type="MeshNormalMaterial",this.bumpMap=null,this.bumpScale=1,this.normalMap=null,this.normalMapType=0,this.normalScale=new Lt(1,1),this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.wireframe=!1,this.wireframeLinewidth=1,this.flatShading=!1,this.setValues(t)}copy(t){return super.copy(t),this.bumpMap=t.bumpMap,this.bumpScale=t.bumpScale,this.normalMap=t.normalMap,this.normalMapType=t.normalMapType,this.normalScale.copy(t.normalScale),this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.flatShading=t.flatShading,this}}class Ol extends vi{constructor(t){super(),this.isMeshLambertMaterial=!0,this.type="MeshLambertMaterial",this.color=new jt(16777215),this.map=null,this.lightMap=null,this.lightMapIntensity=1,this.aoMap=null,this.aoMapIntensity=1,this.emissive=new jt(0),this.emissiveIntensity=1,this.emissiveMap=null,this.bumpMap=null,this.bumpScale=1,this.normalMap=null,this.normalMapType=0,this.normalScale=new Lt(1,1),this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.specularMap=null,this.alphaMap=null,this.envMap=null,this.combine=0,this.reflectivity=1,this.refractionRatio=.98,this.wireframe=!1,this.wireframeLinewidth=1,this.wireframeLinecap="round",this.wireframeLinejoin="round",this.flatShading=!1,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.color.copy(t.color),this.map=t.map,this.lightMap=t.lightMap,this.lightMapIntensity=t.lightMapIntensity,this.aoMap=t.aoMap,this.aoMapIntensity=t.aoMapIntensity,this.emissive.copy(t.emissive),this.emissiveMap=t.emissiveMap,this.emissiveIntensity=t.emissiveIntensity,this.bumpMap=t.bumpMap,this.bumpScale=t.bumpScale,this.normalMap=t.normalMap,this.normalMapType=t.normalMapType,this.normalScale.copy(t.normalScale),this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.specularMap=t.specularMap,this.alphaMap=t.alphaMap,this.envMap=t.envMap,this.combine=t.combine,this.reflectivity=t.reflectivity,this.refractionRatio=t.refractionRatio,this.wireframe=t.wireframe,this.wireframeLinewidth=t.wireframeLinewidth,this.wireframeLinecap=t.wireframeLinecap,this.wireframeLinejoin=t.wireframeLinejoin,this.flatShading=t.flatShading,this.fog=t.fog,this}}class zl extends vi{constructor(t){super(),this.isMeshMatcapMaterial=!0,this.defines={MATCAP:""},this.type="MeshMatcapMaterial",this.color=new jt(16777215),this.matcap=null,this.map=null,this.bumpMap=null,this.bumpScale=1,this.normalMap=null,this.normalMapType=0,this.normalScale=new Lt(1,1),this.displacementMap=null,this.displacementScale=1,this.displacementBias=0,this.alphaMap=null,this.flatShading=!1,this.fog=!0,this.setValues(t)}copy(t){return super.copy(t),this.defines={MATCAP:""},this.color.copy(t.color),this.matcap=t.matcap,this.map=t.map,this.bumpMap=t.bumpMap,this.bumpScale=t.bumpScale,this.normalMap=t.normalMap,this.normalMapType=t.normalMapType,this.normalScale.copy(t.normalScale),this.displacementMap=t.displacementMap,this.displacementScale=t.displacementScale,this.displacementBias=t.displacementBias,this.alphaMap=t.alphaMap,this.flatShading=t.flatShading,this.fog=t.fog,this}}class Ul extends Fa{constructor(t){super(),this.isLineDashedMaterial=!0,this.type="LineDashedMaterial",this.scale=1,this.dashSize=3,this.gapSize=1,this.setValues(t)}copy(t){return super.copy(t),this.scale=t.scale,this.dashSize=t.dashSize,this.gapSize=t.gapSize,this}}function Bl(t,e,i){return kl(t)?new t.constructor(t.subarray(e,void 0!==i?i:t.length)):t.slice(e,i)}function Fl(t,e,i){return!t||!i&&t.constructor===e?t:"number"==typeof e.BYTES_PER_ELEMENT?new e(t):Array.prototype.slice.call(t)}function kl(t){return ArrayBuffer.isView(t)&&!(t instanceof DataView)}function Gl(t){const e=t.length,i=new Array(e);for(let t=0;t!==e;++t)i[t]=t;return i.sort((function(e,i){return t[e]-t[i]})),i}function Vl(t,e,i){const n=t.length,r=new t.constructor(n);for(let s=0,a=0;a!==n;++s){const n=i[s]*e;for(let i=0;i!==e;++i)r[a++]=t[n+i]}return r}function Hl(t,e,i,n){let r=1,s=t[0];for(;void 0!==s&&void 0===s[n];)s=t[r++];if(void 0===s)return;let a=s[n];if(void 0!==a)if(Array.isArray(a))do{a=s[n],void 0!==a&&(e.push(s.time),i.push.apply(i,a)),s=t[r++]}while(void 0!==s);else if(void 0!==a.toArray)do{a=s[n],void 0!==a&&(e.push(s.time),a.toArray(i,i.length)),s=t[r++]}while(void 0!==s);else do{a=s[n],void 0!==a&&(e.push(s.time),i.push(a)),s=t[r++]}while(void 0!==s)}var Wl=Object.freeze({__proto__:null,arraySlice:Bl,convertArray:Fl,isTypedArray:kl,getKeyframeOrder:Gl,sortedArray:Vl,flattenJSON:Hl,subclip:function(t,e,i,n,r=30){const s=t.clone();s.name=e;const a=[];for(let t=0;t<s.tracks.length;++t){const e=s.tracks[t],o=e.getValueSize(),l=[],c=[];for(let t=0;t<e.times.length;++t){const s=e.times[t]*r;if(!(s<i||s>=n)){l.push(e.times[t]);for(let i=0;i<o;++i)c.push(e.values[t*o+i])}}0!==l.length&&(e.times=Fl(l,e.times.constructor),e.values=Fl(c,e.values.constructor),a.push(e))}s.tracks=a;let o=1/0;for(let t=0;t<s.tracks.length;++t)o>s.tracks[t].times[0]&&(o=s.tracks[t].times[0]);for(let t=0;t<s.tracks.length;++t)s.tracks[t].shift(-1*o);return s.resetDuration(),s},makeClipAdditive:function(t,e=0,i=t,n=30){n<=0&&(n=30);const r=i.tracks.length,s=e/n;for(let e=0;e<r;++e){const n=i.tracks[e],r=n.ValueTypeName;if("bool"===r||"string"===r)continue;const a=t.tracks.find((function(t){return t.name===n.name&&t.ValueTypeName===r}));if(void 0===a)continue;let o=0;const l=n.getValueSize();n.createInterpolant.isInterpolantFactoryMethodGLTFCubicSpline&&(o=l/3);let c=0;const h=a.getValueSize();a.createInterpolant.isInterpolantFactoryMethodGLTFCubicSpline&&(c=h/3);const u=n.times.length-1;let d;if(s<=n.times[0]){const t=o,e=l-o;d=Bl(n.values,t,e)}else if(s>=n.times[u]){const t=u*l+o,e=t+l-o;d=Bl(n.values,t,e)}else{const t=n.createInterpolant(),e=o,i=l-o;t.evaluate(s),d=Bl(t.resultBuffer,e,i)}if("quaternion"===r){(new ie).fromArray(d).normalize().conjugate().toArray(d)}const p=a.times.length;for(let t=0;t<p;++t){const e=t*h+c;if("quaternion"===r)ie.multiplyQuaternionsFlat(a.values,e,d,0,a.values,e);else{const t=h-2*c;for(let i=0;i<t;++i)a.values[e+i]-=d[i]}}}return t.blendMode=st,t}});class jl{constructor(t,e,i,n){this.parameterPositions=t,this._cachedIndex=0,this.resultBuffer=void 0!==n?n:new e.constructor(i),this.sampleValues=e,this.valueSize=i,this.settings=null,this.DefaultSettings_={}}evaluate(t){const e=this.parameterPositions;let i=this._cachedIndex,n=e[i],r=e[i-1];t:{e:{let s;i:{n:if(!(t<n)){for(let s=i+2;;){if(void 0===n){if(t<r)break n;return i=e.length,this._cachedIndex=i,this.copySampleValue_(i-1)}if(i===s)break;if(r=n,n=e[++i],t<n)break e}s=e.length;break i}if(t>=r)break t;{const a=e[1];t<a&&(i=2,r=a);for(let s=i-2;;){if(void 0===r)return this._cachedIndex=0,this.copySampleValue_(0);if(i===s)break;if(n=r,r=e[--i-1],t>=r)break e}s=i,i=0}}for(;i<s;){const n=i+s>>>1;t<e[n]?s=n:i=n+1}if(n=e[i],r=e[i-1],void 0===r)return this._cachedIndex=0,this.copySampleValue_(0);if(void 0===n)return i=e.length,this._cachedIndex=i,this.copySampleValue_(i-1)}this._cachedIndex=i,this.intervalChanged_(i,r,n)}return this.interpolate_(i,r,t,n)}getSettings_(){return this.settings||this.DefaultSettings_}copySampleValue_(t){const e=this.resultBuffer,i=this.sampleValues,n=this.valueSize,r=t*n;for(let t=0;t!==n;++t)e[t]=i[r+t];return e}interpolate_(){throw new Error("call to abstract method")}intervalChanged_(){}}class ql extends jl{constructor(t,e,i,n){super(t,e,i,n),this._weightPrev=-0,this._offsetPrev=-0,this._weightNext=-0,this._offsetNext=-0,this.DefaultSettings_={endingStart:et,endingEnd:et}}intervalChanged_(t,e,i){const n=this.parameterPositions;let r=t-2,s=t+1,a=n[r],o=n[s];if(void 0===a)switch(this.getSettings_().endingStart){case it:r=t,a=2*e-i;break;case nt:r=n.length-2,a=e+n[r]-n[r+1];break;default:r=t,a=i}if(void 0===o)switch(this.getSettings_().endingEnd){case it:s=t,o=2*i-e;break;case nt:s=1,o=i+n[1]-n[0];break;default:s=t-1,o=e}const l=.5*(i-e),c=this.valueSize;this._weightPrev=l/(e-a),this._weightNext=l/(o-i),this._offsetPrev=r*c,this._offsetNext=s*c}interpolate_(t,e,i,n){const r=this.resultBuffer,s=this.sampleValues,a=this.valueSize,o=t*a,l=o-a,c=this._offsetPrev,h=this._offsetNext,u=this._weightPrev,d=this._weightNext,p=(i-e)/(n-e),m=p*p,f=m*p,g=-u*f+2*u*m-u*p,v=(1+u)*f+(-1.5-2*u)*m+(-.5+u)*p+1,x=(-1-d)*f+(1.5+d)*m+.5*p,_=d*f-d*m;for(let t=0;t!==a;++t)r[t]=g*s[c+t]+v*s[l+t]+x*s[o+t]+_*s[h+t];return r}}class Xl extends jl{constructor(t,e,i,n){super(t,e,i,n)}interpolate_(t,e,i,n){const r=this.resultBuffer,s=this.sampleValues,a=this.valueSize,o=t*a,l=o-a,c=(i-e)/(n-e),h=1-c;for(let t=0;t!==a;++t)r[t]=s[l+t]*h+s[o+t]*c;return r}}class Yl extends jl{constructor(t,e,i,n){super(t,e,i,n)}interpolate_(t){return this.copySampleValue_(t-1)}}class Zl{constructor(t,e,i,n){if(void 0===t)throw new Error("THREE.KeyframeTrack: track name is undefined");if(void 0===e||0===e.length)throw new Error("THREE.KeyframeTrack: no keyframes in track named "+t);this.name=t,this.times=Fl(e,this.TimeBufferType),this.values=Fl(i,this.ValueBufferType),this.setInterpolation(n||this.DefaultInterpolation)}static toJSON(t){const e=t.constructor;let i;if(e.toJSON!==this.toJSON)i=e.toJSON(t);else{i={name:t.name,times:Fl(t.times,Array),values:Fl(t.values,Array)};const e=t.getInterpolation();e!==t.DefaultInterpolation&&(i.interpolation=e)}return i.type=t.ValueTypeName,i}InterpolantFactoryMethodDiscrete(t){return new Yl(this.times,this.values,this.getValueSize(),t)}InterpolantFactoryMethodLinear(t){return new Xl(this.times,this.values,this.getValueSize(),t)}InterpolantFactoryMethodSmooth(t){return new ql(this.times,this.values,this.getValueSize(),t)}setInterpolation(t){let e;switch(t){case $:e=this.InterpolantFactoryMethodDiscrete;break;case Q:e=this.InterpolantFactoryMethodLinear;break;case tt:e=this.InterpolantFactoryMethodSmooth}if(void 0===e){const e="unsupported interpolation for "+this.ValueTypeName+" keyframe track named "+this.name;if(void 0===this.createInterpolant){if(t===this.DefaultInterpolation)throw new Error(e);this.setInterpolation(this.DefaultInterpolation)}return console.warn("THREE.KeyframeTrack:",e),this}return this.createInterpolant=e,this}getInterpolation(){switch(this.createInterpolant){case this.InterpolantFactoryMethodDiscrete:return $;case this.InterpolantFactoryMethodLinear:return Q;case this.InterpolantFactoryMethodSmooth:return tt}}getValueSize(){return this.values.length/this.times.length}shift(t){if(0!==t){const e=this.times;for(let i=0,n=e.length;i!==n;++i)e[i]+=t}return this}scale(t){if(1!==t){const e=this.times;for(let i=0,n=e.length;i!==n;++i)e[i]*=t}return this}trim(t,e){const i=this.times,n=i.length;let r=0,s=n-1;for(;r!==n&&i[r]<t;)++r;for(;-1!==s&&i[s]>e;)--s;if(++s,0!==r||s!==n){r>=s&&(s=Math.max(s,1),r=s-1);const t=this.getValueSize();this.times=Bl(i,r,s),this.values=Bl(this.values,r*t,s*t)}return this}validate(){let t=!0;const e=this.getValueSize();e-Math.floor(e)!=0&&(console.error("THREE.KeyframeTrack: Invalid value size in track.",this),t=!1);const i=this.times,n=this.values,r=i.length;0===r&&(console.error("THREE.KeyframeTrack: Track is empty.",this),t=!1);let s=null;for(let e=0;e!==r;e++){const n=i[e];if("number"==typeof n&&isNaN(n)){console.error("THREE.KeyframeTrack: Time is not a valid number.",this,e,n),t=!1;break}if(null!==s&&s>n){console.error("THREE.KeyframeTrack: Out of order keys.",this,e,n,s),t=!1;break}s=n}if(void 0!==n&&kl(n))for(let e=0,i=n.length;e!==i;++e){const i=n[e];if(isNaN(i)){console.error("THREE.KeyframeTrack: Value is not a valid number.",this,e,i),t=!1;break}}return t}optimize(){const t=Bl(this.times),e=Bl(this.values),i=this.getValueSize(),n=this.getInterpolation()===tt,r=t.length-1;let s=1;for(let a=1;a<r;++a){let r=!1;const o=t[a];if(o!==t[a+1]&&(1!==a||o!==t[0]))if(n)r=!0;else{const t=a*i,n=t-i,s=t+i;for(let a=0;a!==i;++a){const i=e[t+a];if(i!==e[n+a]||i!==e[s+a]){r=!0;break}}}if(r){if(a!==s){t[s]=t[a];const n=a*i,r=s*i;for(let t=0;t!==i;++t)e[r+t]=e[n+t]}++s}}if(r>0){t[s]=t[r];for(let t=r*i,n=s*i,a=0;a!==i;++a)e[n+a]=e[t+a];++s}return s!==t.length?(this.times=Bl(t,0,s),this.values=Bl(e,0,s*i)):(this.times=t,this.values=e),this}clone(){const t=Bl(this.times,0),e=Bl(this.values,0),i=new(0,this.constructor)(this.name,t,e);return i.createInterpolant=this.createInterpolant,i}}Zl.prototype.TimeBufferType=Float32Array,Zl.prototype.ValueBufferType=Float32Array,Zl.prototype.DefaultInterpolation=Q;class Jl extends Zl{}Jl.prototype.ValueTypeName="bool",Jl.prototype.ValueBufferType=Array,Jl.prototype.DefaultInterpolation=$,Jl.prototype.InterpolantFactoryMethodLinear=void 0,Jl.prototype.InterpolantFactoryMethodSmooth=void 0;class Kl extends Zl{}Kl.prototype.ValueTypeName="color";class $l extends Zl{}$l.prototype.ValueTypeName="number";class Ql extends jl{constructor(t,e,i,n){super(t,e,i,n)}interpolate_(t,e,i,n){const r=this.resultBuffer,s=this.sampleValues,a=this.valueSize,o=(i-e)/(n-e);let l=t*a;for(let t=l+a;l!==t;l+=4)ie.slerpFlat(r,0,s,l-a,s,l,o);return r}}class tc extends Zl{InterpolantFactoryMethodLinear(t){return new Ql(this.times,this.values,this.getValueSize(),t)}}tc.prototype.ValueTypeName="quaternion",tc.prototype.DefaultInterpolation=Q,tc.prototype.InterpolantFactoryMethodSmooth=void 0;class ec extends Zl{}ec.prototype.ValueTypeName="string",ec.prototype.ValueBufferType=Array,ec.prototype.DefaultInterpolation=$,ec.prototype.InterpolantFactoryMethodLinear=void 0,ec.prototype.InterpolantFactoryMethodSmooth=void 0;class ic extends Zl{}ic.prototype.ValueTypeName="vector";class nc{constructor(t,e=-1,i,n=2500){this.name=t,this.tracks=i,this.duration=e,this.blendMode=n,this.uuid=_t(),this.duration<0&&this.resetDuration()}static parse(t){const e=[],i=t.tracks,n=1/(t.fps||1);for(let t=0,r=i.length;t!==r;++t)e.push(rc(i[t]).scale(n));const r=new this(t.name,t.duration,e,t.blendMode);return r.uuid=t.uuid,r}static toJSON(t){const e=[],i=t.tracks,n={name:t.name,duration:t.duration,tracks:e,uuid:t.uuid,blendMode:t.blendMode};for(let t=0,n=i.length;t!==n;++t)e.push(Zl.toJSON(i[t]));return n}static CreateFromMorphTargetSequence(t,e,i,n){const r=e.length,s=[];for(let t=0;t<r;t++){let a=[],o=[];a.push((t+r-1)%r,t,(t+1)%r),o.push(0,1,0);const l=Gl(a);a=Vl(a,1,l),o=Vl(o,1,l),n||0!==a[0]||(a.push(r),o.push(o[0])),s.push(new $l(".morphTargetInfluences["+e[t].name+"]",a,o).scale(1/i))}return new this(t,-1,s)}static findByName(t,e){let i=t;if(!Array.isArray(t)){const e=t;i=e.geometry&&e.geometry.animations||e.animations}for(let t=0;t<i.length;t++)if(i[t].name===e)return i[t];return null}static CreateClipsFromMorphTargetSequences(t,e,i){const n={},r=/^([\w-]*?)([\d]+)$/;for(let e=0,i=t.length;e<i;e++){const i=t[e],s=i.name.match(r);if(s&&s.length>1){const t=s[1];let e=n[t];e||(n[t]=e=[]),e.push(i)}}const s=[];for(const t in n)s.push(this.CreateFromMorphTargetSequence(t,n[t],e,i));return s}static parseAnimation(t,e){if(!t)return console.error("THREE.AnimationClip: No animation in JSONLoader data."),null;const i=function(t,e,i,n,r){if(0!==i.length){const s=[],a=[];Hl(i,s,a,n),0!==s.length&&r.push(new t(e,s,a))}},n=[],r=t.name||"default",s=t.fps||30,a=t.blendMode;let o=t.length||-1;const l=t.hierarchy||[];for(let t=0;t<l.length;t++){const r=l[t].keys;if(r&&0!==r.length)if(r[0].morphTargets){const t={};let e;for(e=0;e<r.length;e++)if(r[e].morphTargets)for(let i=0;i<r[e].morphTargets.length;i++)t[r[e].morphTargets[i]]=-1;for(const i in t){const t=[],s=[];for(let n=0;n!==r[e].morphTargets.length;++n){const n=r[e];t.push(n.time),s.push(n.morphTarget===i?1:0)}n.push(new $l(".morphTargetInfluence["+i+"]",t,s))}o=t.length*s}else{const s=".bones["+e[t].name+"]";i(ic,s+".position",r,"pos",n),i(tc,s+".quaternion",r,"rot",n),i(ic,s+".scale",r,"scl",n)}}if(0===n.length)return null;return new this(r,o,n,a)}resetDuration(){let t=0;for(let e=0,i=this.tracks.length;e!==i;++e){const i=this.tracks[e];t=Math.max(t,i.times[i.times.length-1])}return this.duration=t,this}trim(){for(let t=0;t<this.tracks.length;t++)this.tracks[t].trim(0,this.duration);return this}validate(){let t=!0;for(let e=0;e<this.tracks.length;e++)t=t&&this.tracks[e].validate();return t}optimize(){for(let t=0;t<this.tracks.length;t++)this.tracks[t].optimize();return this}clone(){const t=[];for(let e=0;e<this.tracks.length;e++)t.push(this.tracks[e].clone());return new this.constructor(this.name,this.duration,t,this.blendMode)}toJSON(){return this.constructor.toJSON(this)}}function rc(t){if(void 0===t.type)throw new Error("THREE.KeyframeTrack: track type undefined, can not parse");const e=function(t){switch(t.toLowerCase()){case"scalar":case"double":case"float":case"number":case"integer":return $l;case"vector":case"vector2":case"vector3":case"vector4":return ic;case"color":return Kl;case"quaternion":return tc;case"bool":case"boolean":return Jl;case"string":return ec}throw new Error("THREE.KeyframeTrack: Unsupported typeName: "+t)}(t.type);if(void 0===t.times){const e=[],i=[];Hl(t.keys,e,i,"value"),t.times=e,t.values=i}return void 0!==e.parse?e.parse(t):new e(t.name,t.times,t.values,t.interpolation)}const sc={enabled:!1,files:{},add:function(t,e){!1!==this.enabled&&(this.files[t]=e)},get:function(t){if(!1!==this.enabled)return this.files[t]},remove:function(t){delete this.files[t]},clear:function(){this.files={}}};class ac{constructor(t,e,i){const n=this;let r,s=!1,a=0,o=0;const l=[];this.onStart=void 0,this.onLoad=t,this.onProgress=e,this.onError=i,this.itemStart=function(t){o++,!1===s&&void 0!==n.onStart&&n.onStart(t,a,o),s=!0},this.itemEnd=function(t){a++,void 0!==n.onProgress&&n.onProgress(t,a,o),a===o&&(s=!1,void 0!==n.onLoad&&n.onLoad())},this.itemError=function(t){void 0!==n.onError&&n.onError(t)},this.resolveURL=function(t){return r?r(t):t},this.setURLModifier=function(t){return r=t,this},this.addHandler=function(t,e){return l.push(t,e),this},this.removeHandler=function(t){const e=l.indexOf(t);return-1!==e&&l.splice(e,2),this},this.getHandler=function(t){for(let e=0,i=l.length;e<i;e+=2){const i=l[e],n=l[e+1];if(i.global&&(i.lastIndex=0),i.test(t))return n}return null}}}const oc=new ac;class lc{constructor(t){this.manager=void 0!==t?t:oc,this.crossOrigin="anonymous",this.withCredentials=!1,this.path="",this.resourcePath="",this.requestHeader={}}load(){}loadAsync(t,e){const i=this;return new Promise((function(n,r){i.load(t,n,e,r)}))}parse(){}setCrossOrigin(t){return this.crossOrigin=t,this}setWithCredentials(t){return this.withCredentials=t,this}setPath(t){return this.path=t,this}setResourcePath(t){return this.resourcePath=t,this}setRequestHeader(t){return this.requestHeader=t,this}}const cc={};class hc extends Error{constructor(t,e){super(t),this.response=e}}class uc extends lc{constructor(t){super(t)}load(t,e,i,n){void 0===t&&(t=""),void 0!==this.path&&(t=this.path+t),t=this.manager.resolveURL(t);const r=sc.get(t);if(void 0!==r)return this.manager.itemStart(t),setTimeout((()=>{e&&e(r),this.manager.itemEnd(t)}),0),r;if(void 0!==cc[t])return void cc[t].push({onLoad:e,onProgress:i,onError:n});cc[t]=[],cc[t].push({onLoad:e,onProgress:i,onError:n});const s=new Request(t,{headers:new Headers(this.requestHeader),credentials:this.withCredentials?"include":"same-origin"}),a=this.mimeType,o=this.responseType;fetch(s).then((e=>{if(200===e.status||0===e.status){if(0===e.status&&console.warn("THREE.FileLoader: HTTP Status 0 received."),"undefined"==typeof ReadableStream||void 0===e.body||void 0===e.body.getReader)return e;const i=cc[t],n=e.body.getReader(),r=e.headers.get("Content-Length"),s=r?parseInt(r):0,a=0!==s;let o=0;const l=new ReadableStream({start(t){!function e(){n.read().then((({done:n,value:r})=>{if(n)t.close();else{o+=r.byteLength;const n=new ProgressEvent("progress",{lengthComputable:a,loaded:o,total:s});for(let t=0,e=i.length;t<e;t++){const e=i[t];e.onProgress&&e.onProgress(n)}t.enqueue(r),e()}}))}()}});return new Response(l)}throw new hc(`fetch for "${e.url}" responded with ${e.status}: ${e.statusText}`,e)})).then((t=>{switch(o){case"arraybuffer":return t.arrayBuffer();case"blob":return t.blob();case"document":return t.text().then((t=>(new DOMParser).parseFromString(t,a)));case"json":return t.json();default:if(void 0===a)return t.text();{const e=/charset="?([^;"\s]*)"?/i.exec(a),i=e&&e[1]?e[1].toLowerCase():void 0,n=new TextDecoder(i);return t.arrayBuffer().then((t=>n.decode(t)))}}})).then((e=>{sc.add(t,e);const i=cc[t];delete cc[t];for(let t=0,n=i.length;t<n;t++){const n=i[t];n.onLoad&&n.onLoad(e)}})).catch((e=>{const i=cc[t];if(void 0===i)throw this.manager.itemError(t),e;delete cc[t];for(let t=0,n=i.length;t<n;t++){const n=i[t];n.onError&&n.onError(e)}this.manager.itemError(t)})).finally((()=>{this.manager.itemEnd(t)})),this.manager.itemStart(t)}setResponseType(t){return this.responseType=t,this}setMimeType(t){return this.mimeType=t,this}}class dc extends lc{constructor(t){super(t)}load(t,e,i,n){void 0!==this.path&&(t=this.path+t),t=this.manager.resolveURL(t);const r=this,s=sc.get(t);if(void 0!==s)return r.manager.itemStart(t),setTimeout((function(){e&&e(s),r.manager.itemEnd(t)}),0),s;const a=Nt("img");function o(){c(),sc.add(t,this),e&&e(this),r.manager.itemEnd(t)}function l(e){c(),n&&n(e),r.manager.itemError(t),r.manager.itemEnd(t)}function c(){a.removeEventListener("load",o,!1),a.removeEventListener("error",l,!1)}return a.addEventListener("load",o,!1),a.addEventListener("error",l,!1),"data:"!==t.slice(0,5)&&void 0!==this.crossOrigin&&(a.crossOrigin=this.crossOrigin),r.manager.itemStart(t),a.src=t,a}}class pc extends ri{constructor(t,e=1){super(),this.isLight=!0,this.type="Light",this.color=new jt(t),this.intensity=e}dispose(){}copy(t,e){return super.copy(t,e),this.color.copy(t.color),this.intensity=t.intensity,this}toJSON(t){const e=super.toJSON(t);return e.object.color=this.color.getHex(),e.object.intensity=this.intensity,void 0!==this.groundColor&&(e.object.groundColor=this.groundColor.getHex()),void 0!==this.distance&&(e.object.distance=this.distance),void 0!==this.angle&&(e.object.angle=this.angle),void 0!==this.decay&&(e.object.decay=this.decay),void 0!==this.penumbra&&(e.object.penumbra=this.penumbra),void 0!==this.shadow&&(e.object.shadow=this.shadow.toJSON()),e}}class mc extends pc{constructor(t,e,i){super(t,i),this.isHemisphereLight=!0,this.type="HemisphereLight",this.position.copy(ri.DefaultUp),this.updateMatrix(),this.groundColor=new jt(e)}copy(t,e){return super.copy(t,e),this.groundColor.copy(t.groundColor),this}}const fc=new Ie,gc=new ne,vc=new ne;class xc{constructor(t){this.camera=t,this.bias=0,this.normalBias=0,this.radius=1,this.blurSamples=8,this.mapSize=new Lt(512,512),this.map=null,this.mapPass=null,this.matrix=new Ie,this.autoUpdate=!0,this.needsUpdate=!1,this._frustum=new gn,this._frameExtents=new Lt(1,1),this._viewportCount=1,this._viewports=[new $t(0,0,1,1)]}getViewportCount(){return this._viewportCount}getFrustum(){return this._frustum}updateMatrices(t){const e=this.camera,i=this.matrix;gc.setFromMatrixPosition(t.matrixWorld),e.position.copy(gc),vc.setFromMatrixPosition(t.target.matrixWorld),e.lookAt(vc),e.updateMatrixWorld(),fc.multiplyMatrices(e.projectionMatrix,e.matrixWorldInverse),this._frustum.setFromProjectionMatrix(fc),i.set(.5,0,0,.5,0,.5,0,.5,0,0,.5,.5,0,0,0,1),i.multiply(fc)}getViewport(t){return this._viewports[t]}getFrameExtents(){return this._frameExtents}dispose(){this.map&&this.map.dispose(),this.mapPass&&this.mapPass.dispose()}copy(t){return this.camera=t.camera.clone(),this.bias=t.bias,this.radius=t.radius,this.mapSize.copy(t.mapSize),this}clone(){return(new this.constructor).copy(this)}toJSON(){const t={};return 0!==this.bias&&(t.bias=this.bias),0!==this.normalBias&&(t.normalBias=this.normalBias),1!==this.radius&&(t.radius=this.radius),512===this.mapSize.x&&512===this.mapSize.y||(t.mapSize=this.mapSize.toArray()),t.camera=this.camera.toJSON(!1).object,delete t.camera.matrix,t}}class _c extends xc{constructor(){super(new sn(50,1,.5,500)),this.isSpotLightShadow=!0,this.focus=1}updateMatrices(t){const e=this.camera,i=2*xt*t.angle*this.focus,n=this.mapSize.width/this.mapSize.height,r=t.distance||e.far;i===e.fov&&n===e.aspect&&r===e.far||(e.fov=i,e.aspect=n,e.far=r,e.updateProjectionMatrix()),super.updateMatrices(t)}copy(t){return super.copy(t),this.focus=t.focus,this}}class yc extends pc{constructor(t,e,i=0,n=Math.PI/3,r=0,s=1){super(t,e),this.isSpotLight=!0,this.type="SpotLight",this.position.copy(ri.DefaultUp),this.updateMatrix(),this.target=new ri,this.distance=i,this.angle=n,this.penumbra=r,this.decay=s,this.map=null,this.shadow=new _c}get power(){return this.intensity*Math.PI}set power(t){this.intensity=t/Math.PI}dispose(){this.shadow.dispose()}copy(t,e){return super.copy(t,e),this.distance=t.distance,this.angle=t.angle,this.penumbra=t.penumbra,this.decay=t.decay,this.target=t.target.clone(),this.shadow=t.shadow.clone(),this}}const Mc=new Ie,bc=new ne,Sc=new ne;class wc extends xc{constructor(){super(new sn(90,1,.5,500)),this.isPointLightShadow=!0,this._frameExtents=new Lt(4,2),this._viewportCount=6,this._viewports=[new $t(2,1,1,1),new $t(0,1,1,1),new $t(3,1,1,1),new $t(1,1,1,1),new $t(3,0,1,1),new $t(1,0,1,1)],this._cubeDirections=[new ne(1,0,0),new ne(-1,0,0),new ne(0,0,1),new ne(0,0,-1),new ne(0,1,0),new ne(0,-1,0)],this._cubeUps=[new ne(0,1,0),new ne(0,1,0),new ne(0,1,0),new ne(0,1,0),new ne(0,0,1),new ne(0,0,-1)]}updateMatrices(t,e=0){const i=this.camera,n=this.matrix,r=t.distance||i.far;r!==i.far&&(i.far=r,i.updateProjectionMatrix()),bc.setFromMatrixPosition(t.matrixWorld),i.position.copy(bc),Sc.copy(i.position),Sc.add(this._cubeDirections[e]),i.up.copy(this._cubeUps[e]),i.lookAt(Sc),i.updateMatrixWorld(),n.makeTranslation(-bc.x,-bc.y,-bc.z),Mc.multiplyMatrices(i.projectionMatrix,i.matrixWorldInverse),this._frustum.setFromProjectionMatrix(Mc)}}class Tc extends pc{constructor(t,e,i=0,n=1){super(t,e),this.isPointLight=!0,this.type="PointLight",this.distance=i,this.decay=n,this.shadow=new wc}get power(){return 4*this.intensity*Math.PI}set power(t){this.intensity=t/(4*Math.PI)}dispose(){this.shadow.dispose()}copy(t,e){return super.copy(t,e),this.distance=t.distance,this.decay=t.decay,this.shadow=t.shadow.clone(),this}}class Ac extends xc{constructor(){super(new Ln(-5,5,5,-5,.5,500)),this.isDirectionalLightShadow=!0}}class Ec extends pc{constructor(t,e){super(t,e),this.isDirectionalLight=!0,this.type="DirectionalLight",this.position.copy(ri.DefaultUp),this.updateMatrix(),this.target=new ri,this.shadow=new Ac}dispose(){this.shadow.dispose()}copy(t){return super.copy(t),this.target=t.target.clone(),this.shadow=t.shadow.clone(),this}}class Cc extends pc{constructor(t,e){super(t,e),this.isAmbientLight=!0,this.type="AmbientLight"}}class Lc extends pc{constructor(t,e,i=10,n=10){super(t,e),this.isRectAreaLight=!0,this.type="RectAreaLight",this.width=i,this.height=n}get power(){return this.intensity*this.width*this.height*Math.PI}set power(t){this.intensity=t/(this.width*this.height*Math.PI)}copy(t){return super.copy(t),this.width=t.width,this.height=t.height,this}toJSON(t){const e=super.toJSON(t);return e.object.width=this.width,e.object.height=this.height,e}}class Rc{constructor(){this.isSphericalHarmonics3=!0,this.coefficients=[];for(let t=0;t<9;t++)this.coefficients.push(new ne)}set(t){for(let e=0;e<9;e++)this.coefficients[e].copy(t[e]);return this}zero(){for(let t=0;t<9;t++)this.coefficients[t].set(0,0,0);return this}getAt(t,e){const i=t.x,n=t.y,r=t.z,s=this.coefficients;return e.copy(s[0]).multiplyScalar(.282095),e.addScaledVector(s[1],.488603*n),e.addScaledVector(s[2],.488603*r),e.addScaledVector(s[3],.488603*i),e.addScaledVector(s[4],i*n*1.092548),e.addScaledVector(s[5],n*r*1.092548),e.addScaledVector(s[6],.315392*(3*r*r-1)),e.addScaledVector(s[7],i*r*1.092548),e.addScaledVector(s[8],.546274*(i*i-n*n)),e}getIrradianceAt(t,e){const i=t.x,n=t.y,r=t.z,s=this.coefficients;return e.copy(s[0]).multiplyScalar(.886227),e.addScaledVector(s[1],1.023328*n),e.addScaledVector(s[2],1.023328*r),e.addScaledVector(s[3],1.023328*i),e.addScaledVector(s[4],.858086*i*n),e.addScaledVector(s[5],.858086*n*r),e.addScaledVector(s[6],.743125*r*r-.247708),e.addScaledVector(s[7],.858086*i*r),e.addScaledVector(s[8],.429043*(i*i-n*n)),e}add(t){for(let e=0;e<9;e++)this.coefficients[e].add(t.coefficients[e]);return this}addScaledSH(t,e){for(let i=0;i<9;i++)this.coefficients[i].addScaledVector(t.coefficients[i],e);return this}scale(t){for(let e=0;e<9;e++)this.coefficients[e].multiplyScalar(t);return this}lerp(t,e){for(let i=0;i<9;i++)this.coefficients[i].lerp(t.coefficients[i],e);return this}equals(t){for(let e=0;e<9;e++)if(!this.coefficients[e].equals(t.coefficients[e]))return!1;return!0}copy(t){return this.set(t.coefficients)}clone(){return(new this.constructor).copy(this)}fromArray(t,e=0){const i=this.coefficients;for(let n=0;n<9;n++)i[n].fromArray(t,e+3*n);return this}toArray(t=[],e=0){const i=this.coefficients;for(let n=0;n<9;n++)i[n].toArray(t,e+3*n);return t}static getBasisAt(t,e){const i=t.x,n=t.y,r=t.z;e[0]=.282095,e[1]=.488603*n,e[2]=.488603*r,e[3]=.488603*i,e[4]=1.092548*i*n,e[5]=1.092548*n*r,e[6]=.315392*(3*r*r-1),e[7]=1.092548*i*r,e[8]=.546274*(i*i-n*n)}}class Pc extends pc{constructor(t=new Rc,e=1){super(void 0,e),this.isLightProbe=!0,this.sh=t}copy(t){return super.copy(t),this.sh.copy(t.sh),this}fromJSON(t){return this.intensity=t.intensity,this.sh.fromArray(t.sh),this}toJSON(t){const e=super.toJSON(t);return e.object.sh=this.sh.toArray(),e}}class Dc extends lc{constructor(t){super(t),this.textures={}}load(t,e,i,n){const r=this,s=new uc(r.manager);s.setPath(r.path),s.setRequestHeader(r.requestHeader),s.setWithCredentials(r.withCredentials),s.load(t,(function(i){try{e(r.parse(JSON.parse(i)))}catch(e){n?n(e):console.error(e),r.manager.itemError(t)}}),i,n)}parse(t){const e=this.textures;function i(t){return void 0===e[t]&&console.warn("THREE.MaterialLoader: Undefined texture",t),e[t]}const n=Dc.createMaterialFromType(t.type);if(void 0!==t.uuid&&(n.uuid=t.uuid),void 0!==t.name&&(n.name=t.name),void 0!==t.color&&void 0!==n.color&&n.color.setHex(t.color),void 0!==t.roughness&&(n.roughness=t.roughness),void 0!==t.metalness&&(n.metalness=t.metalness),void 0!==t.sheen&&(n.sheen=t.sheen),void 0!==t.sheenColor&&(n.sheenColor=(new jt).setHex(t.sheenColor)),void 0!==t.sheenRoughness&&(n.sheenRoughness=t.sheenRoughness),void 0!==t.emissive&&void 0!==n.emissive&&n.emissive.setHex(t.emissive),void 0!==t.specular&&void 0!==n.specular&&n.specular.setHex(t.specular),void 0!==t.specularIntensity&&(n.specularIntensity=t.specularIntensity),void 0!==t.specularColor&&void 0!==n.specularColor&&n.specularColor.setHex(t.specularColor),void 0!==t.shininess&&(n.shininess=t.shininess),void 0!==t.clearcoat&&(n.clearcoat=t.clearcoat),void 0!==t.clearcoatRoughness&&(n.clearcoatRoughness=t.clearcoatRoughness),void 0!==t.iridescence&&(n.iridescence=t.iridescence),void 0!==t.iridescenceIOR&&(n.iridescenceIOR=t.iridescenceIOR),void 0!==t.iridescenceThicknessRange&&(n.iridescenceThicknessRange=t.iridescenceThicknessRange),void 0!==t.transmission&&(n.transmission=t.transmission),void 0!==t.thickness&&(n.thickness=t.thickness),void 0!==t.attenuationDistance&&(n.attenuationDistance=t.attenuationDistance),void 0!==t.attenuationColor&&void 0!==n.attenuationColor&&n.attenuationColor.setHex(t.attenuationColor),void 0!==t.fog&&(n.fog=t.fog),void 0!==t.flatShading&&(n.flatShading=t.flatShading),void 0!==t.blending&&(n.blending=t.blending),void 0!==t.combine&&(n.combine=t.combine),void 0!==t.side&&(n.side=t.side),void 0!==t.shadowSide&&(n.shadowSide=t.shadowSide),void 0!==t.opacity&&(n.opacity=t.opacity),void 0!==t.transparent&&(n.transparent=t.transparent),void 0!==t.alphaTest&&(n.alphaTest=t.alphaTest),void 0!==t.depthTest&&(n.depthTest=t.depthTest),void 0!==t.depthWrite&&(n.depthWrite=t.depthWrite),void 0!==t.colorWrite&&(n.colorWrite=t.colorWrite),void 0!==t.stencilWrite&&(n.stencilWrite=t.stencilWrite),void 0!==t.stencilWriteMask&&(n.stencilWriteMask=t.stencilWriteMask),void 0!==t.stencilFunc&&(n.stencilFunc=t.stencilFunc),void 0!==t.stencilRef&&(n.stencilRef=t.stencilRef),void 0!==t.stencilFuncMask&&(n.stencilFuncMask=t.stencilFuncMask),void 0!==t.stencilFail&&(n.stencilFail=t.stencilFail),void 0!==t.stencilZFail&&(n.stencilZFail=t.stencilZFail),void 0!==t.stencilZPass&&(n.stencilZPass=t.stencilZPass),void 0!==t.wireframe&&(n.wireframe=t.wireframe),void 0!==t.wireframeLinewidth&&(n.wireframeLinewidth=t.wireframeLinewidth),void 0!==t.wireframeLinecap&&(n.wireframeLinecap=t.wireframeLinecap),void 0!==t.wireframeLinejoin&&(n.wireframeLinejoin=t.wireframeLinejoin),void 0!==t.rotation&&(n.rotation=t.rotation),1!==t.linewidth&&(n.linewidth=t.linewidth),void 0!==t.dashSize&&(n.dashSize=t.dashSize),void 0!==t.gapSize&&(n.gapSize=t.gapSize),void 0!==t.scale&&(n.scale=t.scale),void 0!==t.polygonOffset&&(n.polygonOffset=t.polygonOffset),void 0!==t.polygonOffsetFactor&&(n.polygonOffsetFactor=t.polygonOffsetFactor),void 0!==t.polygonOffsetUnits&&(n.polygonOffsetUnits=t.polygonOffsetUnits),void 0!==t.dithering&&(n.dithering=t.dithering),void 0!==t.alphaToCoverage&&(n.alphaToCoverage=t.alphaToCoverage),void 0!==t.premultipliedAlpha&&(n.premultipliedAlpha=t.premultipliedAlpha),void 0!==t.visible&&(n.visible=t.visible),void 0!==t.toneMapped&&(n.toneMapped=t.toneMapped),void 0!==t.userData&&(n.userData=t.userData),void 0!==t.vertexColors&&("number"==typeof t.vertexColors?n.vertexColors=t.vertexColors>0:n.vertexColors=t.vertexColors),void 0!==t.uniforms)for(const e in t.uniforms){const r=t.uniforms[e];switch(n.uniforms[e]={},r.type){case"t":n.uniforms[e].value=i(r.value);break;case"c":n.uniforms[e].value=(new jt).setHex(r.value);break;case"v2":n.uniforms[e].value=(new Lt).fromArray(r.value);break;case"v3":n.uniforms[e].value=(new ne).fromArray(r.value);break;case"v4":n.uniforms[e].value=(new $t).fromArray(r.value);break;case"m3":n.uniforms[e].value=(new Rt).fromArray(r.value);break;case"m4":n.uniforms[e].value=(new Ie).fromArray(r.value);break;default:n.uniforms[e].value=r.value}}if(void 0!==t.defines&&(n.defines=t.defines),void 0!==t.vertexShader&&(n.vertexShader=t.vertexShader),void 0!==t.fragmentShader&&(n.fragmentShader=t.fragmentShader),void 0!==t.glslVersion&&(n.glslVersion=t.glslVersion),void 0!==t.extensions)for(const e in t.extensions)n.extensions[e]=t.extensions[e];if(void 0!==t.size&&(n.size=t.size),void 0!==t.sizeAttenuation&&(n.sizeAttenuation=t.sizeAttenuation),void 0!==t.map&&(n.map=i(t.map)),void 0!==t.matcap&&(n.matcap=i(t.matcap)),void 0!==t.alphaMap&&(n.alphaMap=i(t.alphaMap)),void 0!==t.bumpMap&&(n.bumpMap=i(t.bumpMap)),void 0!==t.bumpScale&&(n.bumpScale=t.bumpScale),void 0!==t.normalMap&&(n.normalMap=i(t.normalMap)),void 0!==t.normalMapType&&(n.normalMapType=t.normalMapType),void 0!==t.normalScale){let e=t.normalScale;!1===Array.isArray(e)&&(e=[e,e]),n.normalScale=(new Lt).fromArray(e)}return void 0!==t.displacementMap&&(n.displacementMap=i(t.displacementMap)),void 0!==t.displacementScale&&(n.displacementScale=t.displacementScale),void 0!==t.displacementBias&&(n.displacementBias=t.displacementBias),void 0!==t.roughnessMap&&(n.roughnessMap=i(t.roughnessMap)),void 0!==t.metalnessMap&&(n.metalnessMap=i(t.metalnessMap)),void 0!==t.emissiveMap&&(n.emissiveMap=i(t.emissiveMap)),void 0!==t.emissiveIntensity&&(n.emissiveIntensity=t.emissiveIntensity),void 0!==t.specularMap&&(n.specularMap=i(t.specularMap)),void 0!==t.specularIntensityMap&&(n.specularIntensityMap=i(t.specularIntensityMap)),void 0!==t.specularColorMap&&(n.specularColorMap=i(t.specularColorMap)),void 0!==t.envMap&&(n.envMap=i(t.envMap)),void 0!==t.envMapIntensity&&(n.envMapIntensity=t.envMapIntensity),void 0!==t.reflectivity&&(n.reflectivity=t.reflectivity),void 0!==t.refractionRatio&&(n.refractionRatio=t.refractionRatio),void 0!==t.lightMap&&(n.lightMap=i(t.lightMap)),void 0!==t.lightMapIntensity&&(n.lightMapIntensity=t.lightMapIntensity),void 0!==t.aoMap&&(n.aoMap=i(t.aoMap)),void 0!==t.aoMapIntensity&&(n.aoMapIntensity=t.aoMapIntensity),void 0!==t.gradientMap&&(n.gradientMap=i(t.gradientMap)),void 0!==t.clearcoatMap&&(n.clearcoatMap=i(t.clearcoatMap)),void 0!==t.clearcoatRoughnessMap&&(n.clearcoatRoughnessMap=i(t.clearcoatRoughnessMap)),void 0!==t.clearcoatNormalMap&&(n.clearcoatNormalMap=i(t.clearcoatNormalMap)),void 0!==t.clearcoatNormalScale&&(n.clearcoatNormalScale=(new Lt).fromArray(t.clearcoatNormalScale)),void 0!==t.iridescenceMap&&(n.iridescenceMap=i(t.iridescenceMap)),void 0!==t.iridescenceThicknessMap&&(n.iridescenceThicknessMap=i(t.iridescenceThicknessMap)),void 0!==t.transmissionMap&&(n.transmissionMap=i(t.transmissionMap)),void 0!==t.thicknessMap&&(n.thicknessMap=i(t.thicknessMap)),void 0!==t.sheenColorMap&&(n.sheenColorMap=i(t.sheenColorMap)),void 0!==t.sheenRoughnessMap&&(n.sheenRoughnessMap=i(t.sheenRoughnessMap)),n}setTextures(t){return this.textures=t,this}static createMaterialFromType(t){return new{ShadowMaterial:Cl,SpriteMaterial:ia,RawShaderMaterial:Ll,ShaderMaterial:nn,PointsMaterial:Ja,MeshPhysicalMaterial:Pl,MeshStandardMaterial:Rl,MeshPhongMaterial:Dl,MeshToonMaterial:Il,MeshNormalMaterial:Nl,MeshLambertMaterial:Ol,MeshDepthMaterial:Ns,MeshDistanceMaterial:Os,MeshBasicMaterial:xi,MeshMatcapMaterial:zl,LineDashedMaterial:Ul,LineBasicMaterial:Fa,Material:vi}[t]}}class Ic{static decodeText(t){if("undefined"!=typeof TextDecoder)return(new TextDecoder).decode(t);let e="";for(let i=0,n=t.length;i<n;i++)e+=String.fromCharCode(t[i]);try{return decodeURIComponent(escape(e))}catch(t){return e}}static extractUrlBase(t){const e=t.lastIndexOf("/");return-1===e?"./":t.slice(0,e+1)}static resolveURL(t,e){return"string"!=typeof t||""===t?"":(/^https?:\/\//i.test(e)&&/^\//.test(t)&&(e=e.replace(/(^https?:\/\/[^\/]+).*/i,"$1")),/^(https?:)?\/\//i.test(t)||/^data:.*,.*$/i.test(t)||/^blob:.*$/i.test(t)?t:e+t)}}class Nc extends Di{constructor(){super(),this.isInstancedBufferGeometry=!0,this.type="InstancedBufferGeometry",this.instanceCount=1/0}copy(t){return super.copy(t),this.instanceCount=t.instanceCount,this}clone(){return(new this.constructor).copy(this)}toJSON(){const t=super.toJSON(this);return t.instanceCount=this.instanceCount,t.isInstancedBufferGeometry=!0,t}}class Oc extends lc{constructor(t){super(t)}load(t,e,i,n){const r=this,s=new uc(r.manager);s.setPath(r.path),s.setRequestHeader(r.requestHeader),s.setWithCredentials(r.withCredentials),s.load(t,(function(i){try{e(r.parse(JSON.parse(i)))}catch(e){n?n(e):console.error(e),r.manager.itemError(t)}}),i,n)}parse(t){const e={},i={};function n(t,n){if(void 0!==e[n])return e[n];const r=t.interleavedBuffers[n],s=function(t,e){if(void 0!==i[e])return i[e];const n=t.arrayBuffers[e],r=new Uint32Array(n).buffer;return i[e]=r,r}(t,r.buffer),a=It(r.type,s),o=new Qs(a,r.stride);return o.uuid=r.uuid,e[n]=o,o}const r=t.isInstancedBufferGeometry?new Nc:new Di,s=t.data.index;if(void 0!==s){const t=It(s.type,s.array);r.setIndex(new Mi(t,1))}const a=t.data.attributes;for(const e in a){const i=a[e];let s;if(i.isInterleavedBufferAttribute){const e=n(t.data,i.data);s=new ea(e,i.itemSize,i.offset,i.normalized)}else{const t=It(i.type,i.array);s=new(i.isInstancedBufferAttribute?Da:Mi)(t,i.itemSize,i.normalized)}void 0!==i.name&&(s.name=i.name),void 0!==i.usage&&s.setUsage(i.usage),void 0!==i.updateRange&&(s.updateRange.offset=i.updateRange.offset,s.updateRange.count=i.updateRange.count),r.setAttribute(e,s)}const o=t.data.morphAttributes;if(o)for(const e in o){const i=o[e],s=[];for(let e=0,r=i.length;e<r;e++){const r=i[e];let a;if(r.isInterleavedBufferAttribute){const e=n(t.data,r.data);a=new ea(e,r.itemSize,r.offset,r.normalized)}else{const t=It(r.type,r.array);a=new Mi(t,r.itemSize,r.normalized)}void 0!==r.name&&(a.name=r.name),s.push(a)}r.morphAttributes[e]=s}t.data.morphTargetsRelative&&(r.morphTargetsRelative=!0);const l=t.data.groups||t.data.drawcalls||t.data.offsets;if(void 0!==l)for(let t=0,e=l.length;t!==e;++t){const e=l[t];r.addGroup(e.start,e.count,e.materialIndex)}const c=t.data.boundingSphere;if(void 0!==c){const t=new ne;void 0!==c.center&&t.fromArray(c.center),r.boundingSphere=new we(t,c.radius)}return t.name&&(r.name=t.name),t.userData&&(r.userData=t.userData),r}}const zc={UVMapping:n,CubeReflectionMapping:r,CubeRefractionMapping:s,EquirectangularReflectionMapping:a,EquirectangularRefractionMapping:o,CubeUVReflectionMapping:l},Uc={RepeatWrapping:c,ClampToEdgeWrapping:h,MirroredRepeatWrapping:u},Bc={NearestFilter:d,NearestMipmapNearestFilter:p,NearestMipmapLinearFilter:m,LinearFilter:f,LinearMipmapNearestFilter:g,LinearMipmapLinearFilter:v};let Fc;class kc{static getContext(){return void 0===Fc&&(Fc=new(window.AudioContext||window.webkitAudioContext)),Fc}static setContext(t){Fc=t}}const Gc=new Ie,Vc=new Ie,Hc=new Ie;class Wc{constructor(t=!0){this.autoStart=t,this.startTime=0,this.oldTime=0,this.elapsedTime=0,this.running=!1}start(){this.startTime=jc(),this.oldTime=this.startTime,this.elapsedTime=0,this.running=!0}stop(){this.getElapsedTime(),this.running=!1,this.autoStart=!1}getElapsedTime(){return this.getDelta(),this.elapsedTime}getDelta(){let t=0;if(this.autoStart&&!this.running)return this.start(),0;if(this.running){const e=jc();t=(e-this.oldTime)/1e3,this.oldTime=e,this.elapsedTime+=t}return t}}function jc(){return("undefined"==typeof performance?Date:performance).now()}const qc=new ne,Xc=new ie,Yc=new ne,Zc=new ne;class Jc extends ri{constructor(t){super(),this.type="Audio",this.listener=t,this.context=t.context,this.gain=this.context.createGain(),this.gain.connect(t.getInput()),this.autoplay=!1,this.buffer=null,this.detune=0,this.loop=!1,this.loopStart=0,this.loopEnd=0,this.offset=0,this.duration=void 0,this.playbackRate=1,this.isPlaying=!1,this.hasPlaybackControl=!0,this.source=null,this.sourceType="empty",this._startedAt=0,this._progress=0,this._connected=!1,this.filters=[]}getOutput(){return this.gain}setNodeSource(t){return this.hasPlaybackControl=!1,this.sourceType="audioNode",this.source=t,this.connect(),this}setMediaElementSource(t){return this.hasPlaybackControl=!1,this.sourceType="mediaNode",this.source=this.context.createMediaElementSource(t),this.connect(),this}setMediaStreamSource(t){return this.hasPlaybackControl=!1,this.sourceType="mediaStreamNode",this.source=this.context.createMediaStreamSource(t),this.connect(),this}setBuffer(t){return this.buffer=t,this.sourceType="buffer",this.autoplay&&this.play(),this}play(t=0){if(!0===this.isPlaying)return void console.warn("THREE.Audio: Audio is already playing.");if(!1===this.hasPlaybackControl)return void console.warn("THREE.Audio: this Audio has no playback control.");this._startedAt=this.context.currentTime+t;const e=this.context.createBufferSource();return e.buffer=this.buffer,e.loop=this.loop,e.loopStart=this.loopStart,e.loopEnd=this.loopEnd,e.onended=this.onEnded.bind(this),e.start(this._startedAt,this._progress+this.offset,this.duration),this.isPlaying=!0,this.source=e,this.setDetune(this.detune),this.setPlaybackRate(this.playbackRate),this.connect()}pause(){if(!1!==this.hasPlaybackControl)return!0===this.isPlaying&&(this._progress+=Math.max(this.context.currentTime-this._startedAt,0)*this.playbackRate,!0===this.loop&&(this._progress=this._progress%(this.duration||this.buffer.duration)),this.source.stop(),this.source.onended=null,this.isPlaying=!1),this;console.warn("THREE.Audio: this Audio has no playback control.")}stop(){if(!1!==this.hasPlaybackControl)return this._progress=0,this.source.stop(),this.source.onended=null,this.isPlaying=!1,this;console.warn("THREE.Audio: this Audio has no playback control.")}connect(){if(this.filters.length>0){this.source.connect(this.filters[0]);for(let t=1,e=this.filters.length;t<e;t++)this.filters[t-1].connect(this.filters[t]);this.filters[this.filters.length-1].connect(this.getOutput())}else this.source.connect(this.getOutput());return this._connected=!0,this}disconnect(){if(this.filters.length>0){this.source.disconnect(this.filters[0]);for(let t=1,e=this.filters.length;t<e;t++)this.filters[t-1].disconnect(this.filters[t]);this.filters[this.filters.length-1].disconnect(this.getOutput())}else this.source.disconnect(this.getOutput());return this._connected=!1,this}getFilters(){return this.filters}setFilters(t){return t||(t=[]),!0===this._connected?(this.disconnect(),this.filters=t.slice(),this.connect()):this.filters=t.slice(),this}setDetune(t){if(this.detune=t,void 0!==this.source.detune)return!0===this.isPlaying&&this.source.detune.setTargetAtTime(this.detune,this.context.currentTime,.01),this}getDetune(){return this.detune}getFilter(){return this.getFilters()[0]}setFilter(t){return this.setFilters(t?[t]:[])}setPlaybackRate(t){if(!1!==this.hasPlaybackControl)return this.playbackRate=t,!0===this.isPlaying&&this.source.playbackRate.setTargetAtTime(this.playbackRate,this.context.currentTime,.01),this;console.warn("THREE.Audio: this Audio has no playback control.")}getPlaybackRate(){return this.playbackRate}onEnded(){this.isPlaying=!1}getLoop(){return!1===this.hasPlaybackControl?(console.warn("THREE.Audio: this Audio has no playback control."),!1):this.loop}setLoop(t){if(!1!==this.hasPlaybackControl)return this.loop=t,!0===this.isPlaying&&(this.source.loop=this.loop),this;console.warn("THREE.Audio: this Audio has no playback control.")}setLoopStart(t){return this.loopStart=t,this}setLoopEnd(t){return this.loopEnd=t,this}getVolume(){return this.gain.gain.value}setVolume(t){return this.gain.gain.setTargetAtTime(t,this.context.currentTime,.01),this}}const Kc=new ne,$c=new ie,Qc=new ne,th=new ne;class eh{constructor(t,e,i){let n,r,s;switch(this.binding=t,this.valueSize=i,e){case"quaternion":n=this._slerp,r=this._slerpAdditive,s=this._setAdditiveIdentityQuaternion,this.buffer=new Float64Array(6*i),this._workIndex=5;break;case"string":case"bool":n=this._select,r=this._select,s=this._setAdditiveIdentityOther,this.buffer=new Array(5*i);break;default:n=this._lerp,r=this._lerpAdditive,s=this._setAdditiveIdentityNumeric,this.buffer=new Float64Array(5*i)}this._mixBufferRegion=n,this._mixBufferRegionAdditive=r,this._setIdentity=s,this._origIndex=3,this._addIndex=4,this.cumulativeWeight=0,this.cumulativeWeightAdditive=0,this.useCount=0,this.referenceCount=0}accumulate(t,e){const i=this.buffer,n=this.valueSize,r=t*n+n;let s=this.cumulativeWeight;if(0===s){for(let t=0;t!==n;++t)i[r+t]=i[t];s=e}else{s+=e;const t=e/s;this._mixBufferRegion(i,r,0,t,n)}this.cumulativeWeight=s}accumulateAdditive(t){const e=this.buffer,i=this.valueSize,n=i*this._addIndex;0===this.cumulativeWeightAdditive&&this._setIdentity(),this._mixBufferRegionAdditive(e,n,0,t,i),this.cumulativeWeightAdditive+=t}apply(t){const e=this.valueSize,i=this.buffer,n=t*e+e,r=this.cumulativeWeight,s=this.cumulativeWeightAdditive,a=this.binding;if(this.cumulativeWeight=0,this.cumulativeWeightAdditive=0,r<1){const t=e*this._origIndex;this._mixBufferRegion(i,n,t,1-r,e)}s>0&&this._mixBufferRegionAdditive(i,n,this._addIndex*e,1,e);for(let t=e,r=e+e;t!==r;++t)if(i[t]!==i[t+e]){a.setValue(i,n);break}}saveOriginalState(){const t=this.binding,e=this.buffer,i=this.valueSize,n=i*this._origIndex;t.getValue(e,n);for(let t=i,r=n;t!==r;++t)e[t]=e[n+t%i];this._setIdentity(),this.cumulativeWeight=0,this.cumulativeWeightAdditive=0}restoreOriginalState(){const t=3*this.valueSize;this.binding.setValue(this.buffer,t)}_setAdditiveIdentityNumeric(){const t=this._addIndex*this.valueSize,e=t+this.valueSize;for(let i=t;i<e;i++)this.buffer[i]=0}_setAdditiveIdentityQuaternion(){this._setAdditiveIdentityNumeric(),this.buffer[this._addIndex*this.valueSize+3]=1}_setAdditiveIdentityOther(){const t=this._origIndex*this.valueSize,e=this._addIndex*this.valueSize;for(let i=0;i<this.valueSize;i++)this.buffer[e+i]=this.buffer[t+i]}_select(t,e,i,n,r){if(n>=.5)for(let n=0;n!==r;++n)t[e+n]=t[i+n]}_slerp(t,e,i,n){ie.slerpFlat(t,e,t,e,t,i,n)}_slerpAdditive(t,e,i,n,r){const s=this._workIndex*r;ie.multiplyQuaternionsFlat(t,s,t,e,t,i),ie.slerpFlat(t,e,t,e,t,s,n)}_lerp(t,e,i,n,r){const s=1-n;for(let a=0;a!==r;++a){const r=e+a;t[r]=t[r]*s+t[i+a]*n}}_lerpAdditive(t,e,i,n,r){for(let s=0;s!==r;++s){const r=e+s;t[r]=t[r]+t[i+s]*n}}}const ih="\\[\\]\\.:\\/",nh=new RegExp("[\\[\\]\\.:\\/]","g"),rh="[^\\[\\]\\.:\\/]",sh="[^"+ih.replace("\\.","")+"]",ah=new RegExp("^"+/((?:WC+[\/:])*)/.source.replace("WC",rh)+/(WCOD+)?/.source.replace("WCOD",sh)+/(?:\.(WC+)(?:\[(.+)\])?)?/.source.replace("WC",rh)+/\.(WC+)(?:\[(.+)\])?/.source.replace("WC",rh)+"$"),oh=["material","materials","bones","map"];class lh{constructor(t,e,i){this.path=e,this.parsedPath=i||lh.parseTrackName(e),this.node=lh.findNode(t,this.parsedPath.nodeName)||t,this.rootNode=t,this.getValue=this._getValue_unbound,this.setValue=this._setValue_unbound}static create(t,e,i){return t&&t.isAnimationObjectGroup?new lh.Composite(t,e,i):new lh(t,e,i)}static sanitizeNodeName(t){return t.replace(/\s/g,"_").replace(nh,"")}static parseTrackName(t){const e=ah.exec(t);if(null===e)throw new Error("PropertyBinding: Cannot parse trackName: "+t);const i={nodeName:e[2],objectName:e[3],objectIndex:e[4],propertyName:e[5],propertyIndex:e[6]},n=i.nodeName&&i.nodeName.lastIndexOf(".");if(void 0!==n&&-1!==n){const t=i.nodeName.substring(n+1);-1!==oh.indexOf(t)&&(i.nodeName=i.nodeName.substring(0,n),i.objectName=t)}if(null===i.propertyName||0===i.propertyName.length)throw new Error("PropertyBinding: can not parse propertyName from trackName: "+t);return i}static findNode(t,e){if(void 0===e||""===e||"."===e||-1===e||e===t.name||e===t.uuid)return t;if(t.skeleton){const i=t.skeleton.getBoneByName(e);if(void 0!==i)return i}if(t.children){const i=function(t){for(let n=0;n<t.length;n++){const r=t[n];if(r.name===e||r.uuid===e)return r;const s=i(r.children);if(s)return s}return null},n=i(t.children);if(n)return n}return null}_getValue_unavailable(){}_setValue_unavailable(){}_getValue_direct(t,e){t[e]=this.targetObject[this.propertyName]}_getValue_array(t,e){const i=this.resolvedProperty;for(let n=0,r=i.length;n!==r;++n)t[e++]=i[n]}_getValue_arrayElement(t,e){t[e]=this.resolvedProperty[this.propertyIndex]}_getValue_toArray(t,e){this.resolvedProperty.toArray(t,e)}_setValue_direct(t,e){this.targetObject[this.propertyName]=t[e]}_setValue_direct_setNeedsUpdate(t,e){this.targetObject[this.propertyName]=t[e],this.targetObject.needsUpdate=!0}_setValue_direct_setMatrixWorldNeedsUpdate(t,e){this.targetObject[this.propertyName]=t[e],this.targetObject.matrixWorldNeedsUpdate=!0}_setValue_array(t,e){const i=this.resolvedProperty;for(let n=0,r=i.length;n!==r;++n)i[n]=t[e++]}_setValue_array_setNeedsUpdate(t,e){const i=this.resolvedProperty;for(let n=0,r=i.length;n!==r;++n)i[n]=t[e++];this.targetObject.needsUpdate=!0}_setValue_array_setMatrixWorldNeedsUpdate(t,e){const i=this.resolvedProperty;for(let n=0,r=i.length;n!==r;++n)i[n]=t[e++];this.targetObject.matrixWorldNeedsUpdate=!0}_setValue_arrayElement(t,e){this.resolvedProperty[this.propertyIndex]=t[e]}_setValue_arrayElement_setNeedsUpdate(t,e){this.resolvedProperty[this.propertyIndex]=t[e],this.targetObject.needsUpdate=!0}_setValue_arrayElement_setMatrixWorldNeedsUpdate(t,e){this.resolvedProperty[this.propertyIndex]=t[e],this.targetObject.matrixWorldNeedsUpdate=!0}_setValue_fromArray(t,e){this.resolvedProperty.fromArray(t,e)}_setValue_fromArray_setNeedsUpdate(t,e){this.resolvedProperty.fromArray(t,e),this.targetObject.needsUpdate=!0}_setValue_fromArray_setMatrixWorldNeedsUpdate(t,e){this.resolvedProperty.fromArray(t,e),this.targetObject.matrixWorldNeedsUpdate=!0}_getValue_unbound(t,e){this.bind(),this.getValue(t,e)}_setValue_unbound(t,e){this.bind(),this.setValue(t,e)}bind(){let t=this.node;const e=this.parsedPath,i=e.objectName,n=e.propertyName;let r=e.propertyIndex;if(t||(t=lh.findNode(this.rootNode,e.nodeName)||this.rootNode,this.node=t),this.getValue=this._getValue_unavailable,this.setValue=this._setValue_unavailable,!t)return void console.error("THREE.PropertyBinding: Trying to update node for track: "+this.path+" but it wasn't found.");if(i){let n=e.objectIndex;switch(i){case"materials":if(!t.material)return void console.error("THREE.PropertyBinding: Can not bind to material as node does not have a material.",this);if(!t.material.materials)return void console.error("THREE.PropertyBinding: Can not bind to material.materials as node.material does not have a materials array.",this);t=t.material.materials;break;case"bones":if(!t.skeleton)return void console.error("THREE.PropertyBinding: Can not bind to bones as node does not have a skeleton.",this);t=t.skeleton.bones;for(let e=0;e<t.length;e++)if(t[e].name===n){n=e;break}break;case"map":if("map"in t){t=t.map;break}if(!t.material)return void console.error("THREE.PropertyBinding: Can not bind to material as node does not have a material.",this);if(!t.material.map)return void console.error("THREE.PropertyBinding: Can not bind to material.map as node.material does not have a map.",this);t=t.material.map;break;default:if(void 0===t[i])return void console.error("THREE.PropertyBinding: Can not bind to objectName of node undefined.",this);t=t[i]}if(void 0!==n){if(void 0===t[n])return void console.error("THREE.PropertyBinding: Trying to bind to objectIndex of objectName, but is undefined.",this,t);t=t[n]}}const s=t[n];if(void 0===s){const i=e.nodeName;return void console.error("THREE.PropertyBinding: Trying to update property for track: "+i+"."+n+" but it wasn't found.",t)}let a=this.Versioning.None;this.targetObject=t,void 0!==t.needsUpdate?a=this.Versioning.NeedsUpdate:void 0!==t.matrixWorldNeedsUpdate&&(a=this.Versioning.MatrixWorldNeedsUpdate);let o=this.BindingType.Direct;if(void 0!==r){if("morphTargetInfluences"===n){if(!t.geometry)return void console.error("THREE.PropertyBinding: Can not bind to morphTargetInfluences because node does not have a geometry.",this);if(!t.geometry.morphAttributes)return void console.error("THREE.PropertyBinding: Can not bind to morphTargetInfluences because node does not have a geometry.morphAttributes.",this);void 0!==t.morphTargetDictionary[r]&&(r=t.morphTargetDictionary[r])}o=this.BindingType.ArrayElement,this.resolvedProperty=s,this.propertyIndex=r}else void 0!==s.fromArray&&void 0!==s.toArray?(o=this.BindingType.HasFromToArray,this.resolvedProperty=s):Array.isArray(s)?(o=this.BindingType.EntireArray,this.resolvedProperty=s):this.propertyName=n;this.getValue=this.GetterByBindingType[o],this.setValue=this.SetterByBindingTypeAndVersioning[o][a]}unbind(){this.node=null,this.getValue=this._getValue_unbound,this.setValue=this._setValue_unbound}}lh.Composite=class{constructor(t,e,i){const n=i||lh.parseTrackName(e);this._targetGroup=t,this._bindings=t.subscribe_(e,n)}getValue(t,e){this.bind();const i=this._targetGroup.nCachedObjects_,n=this._bindings[i];void 0!==n&&n.getValue(t,e)}setValue(t,e){const i=this._bindings;for(let n=this._targetGroup.nCachedObjects_,r=i.length;n!==r;++n)i[n].setValue(t,e)}bind(){const t=this._bindings;for(let e=this._targetGroup.nCachedObjects_,i=t.length;e!==i;++e)t[e].bind()}unbind(){const t=this._bindings;for(let e=this._targetGroup.nCachedObjects_,i=t.length;e!==i;++e)t[e].unbind()}},lh.prototype.BindingType={Direct:0,EntireArray:1,ArrayElement:2,HasFromToArray:3},lh.prototype.Versioning={None:0,NeedsUpdate:1,MatrixWorldNeedsUpdate:2},lh.prototype.GetterByBindingType=[lh.prototype._getValue_direct,lh.prototype._getValue_array,lh.prototype._getValue_arrayElement,lh.prototype._getValue_toArray],lh.prototype.SetterByBindingTypeAndVersioning=[[lh.prototype._setValue_direct,lh.prototype._setValue_direct_setNeedsUpdate,lh.prototype._setValue_direct_setMatrixWorldNeedsUpdate],[lh.prototype._setValue_array,lh.prototype._setValue_array_setNeedsUpdate,lh.prototype._setValue_array_setMatrixWorldNeedsUpdate],[lh.prototype._setValue_arrayElement,lh.prototype._setValue_arrayElement_setNeedsUpdate,lh.prototype._setValue_arrayElement_setMatrixWorldNeedsUpdate],[lh.prototype._setValue_fromArray,lh.prototype._setValue_fromArray_setNeedsUpdate,lh.prototype._setValue_fromArray_setMatrixWorldNeedsUpdate]];class ch{constructor(t,e,i=null,n=e.blendMode){this._mixer=t,this._clip=e,this._localRoot=i,this.blendMode=n;const r=e.tracks,s=r.length,a=new Array(s),o={endingStart:et,endingEnd:et};for(let t=0;t!==s;++t){const e=r[t].createInterpolant(null);a[t]=e,e.settings=o}this._interpolantSettings=o,this._interpolants=a,this._propertyBindings=new Array(s),this._cacheIndex=null,this._byClipCacheIndex=null,this._timeScaleInterpolant=null,this._weightInterpolant=null,this.loop=2201,this._loopCount=-1,this._startTime=null,this.time=0,this.timeScale=1,this._effectiveTimeScale=1,this.weight=1,this._effectiveWeight=1,this.repetitions=1/0,this.paused=!1,this.enabled=!0,this.clampWhenFinished=!1,this.zeroSlopeAtStart=!0,this.zeroSlopeAtEnd=!0}play(){return this._mixer._activateAction(this),this}stop(){return this._mixer._deactivateAction(this),this.reset()}reset(){return this.paused=!1,this.enabled=!0,this.time=0,this._loopCount=-1,this._startTime=null,this.stopFading().stopWarping()}isRunning(){return this.enabled&&!this.paused&&0!==this.timeScale&&null===this._startTime&&this._mixer._isActiveAction(this)}isScheduled(){return this._mixer._isActiveAction(this)}startAt(t){return this._startTime=t,this}setLoop(t,e){return this.loop=t,this.repetitions=e,this}setEffectiveWeight(t){return this.weight=t,this._effectiveWeight=this.enabled?t:0,this.stopFading()}getEffectiveWeight(){return this._effectiveWeight}fadeIn(t){return this._scheduleFading(t,0,1)}fadeOut(t){return this._scheduleFading(t,1,0)}crossFadeFrom(t,e,i){if(t.fadeOut(e),this.fadeIn(e),i){const i=this._clip.duration,n=t._clip.duration,r=n/i,s=i/n;t.warp(1,r,e),this.warp(s,1,e)}return this}crossFadeTo(t,e,i){return t.crossFadeFrom(this,e,i)}stopFading(){const t=this._weightInterpolant;return null!==t&&(this._weightInterpolant=null,this._mixer._takeBackControlInterpolant(t)),this}setEffectiveTimeScale(t){return this.timeScale=t,this._effectiveTimeScale=this.paused?0:t,this.stopWarping()}getEffectiveTimeScale(){return this._effectiveTimeScale}setDuration(t){return this.timeScale=this._clip.duration/t,this.stopWarping()}syncWith(t){return this.time=t.time,this.timeScale=t.timeScale,this.stopWarping()}halt(t){return this.warp(this._effectiveTimeScale,0,t)}warp(t,e,i){const n=this._mixer,r=n.time,s=this.timeScale;let a=this._timeScaleInterpolant;null===a&&(a=n._lendControlInterpolant(),this._timeScaleInterpolant=a);const o=a.parameterPositions,l=a.sampleValues;return o[0]=r,o[1]=r+i,l[0]=t/s,l[1]=e/s,this}stopWarping(){const t=this._timeScaleInterpolant;return null!==t&&(this._timeScaleInterpolant=null,this._mixer._takeBackControlInterpolant(t)),this}getMixer(){return this._mixer}getClip(){return this._clip}getRoot(){return this._localRoot||this._mixer._root}_update(t,e,i,n){if(!this.enabled)return void this._updateWeight(t);const r=this._startTime;if(null!==r){const n=(t-r)*i;n<0||0===i?e=0:(this._startTime=null,e=i*n)}e*=this._updateTimeScale(t);const s=this._updateTime(e),a=this._updateWeight(t);if(a>0){const t=this._interpolants,e=this._propertyBindings;if(this.blendMode===st)for(let i=0,n=t.length;i!==n;++i)t[i].evaluate(s),e[i].accumulateAdditive(a);else for(let i=0,r=t.length;i!==r;++i)t[i].evaluate(s),e[i].accumulate(n,a)}}_updateWeight(t){let e=0;if(this.enabled){e=this.weight;const i=this._weightInterpolant;if(null!==i){const n=i.evaluate(t)[0];e*=n,t>i.parameterPositions[1]&&(this.stopFading(),0===n&&(this.enabled=!1))}}return this._effectiveWeight=e,e}_updateTimeScale(t){let e=0;if(!this.paused){e=this.timeScale;const i=this._timeScaleInterpolant;if(null!==i){e*=i.evaluate(t)[0],t>i.parameterPositions[1]&&(this.stopWarping(),0===e?this.paused=!0:this.timeScale=e)}}return this._effectiveTimeScale=e,e}_updateTime(t){const e=this._clip.duration,i=this.loop;let n=this.time+t,r=this._loopCount;const s=2202===i;if(0===t)return-1===r?n:s&&1==(1&r)?e-n:n;if(2200===i){-1===r&&(this._loopCount=0,this._setEndings(!0,!0,!1));t:{if(n>=e)n=e;else{if(!(n<0)){this.time=n;break t}n=0}this.clampWhenFinished?this.paused=!0:this.enabled=!1,this.time=n,this._mixer.dispatchEvent({type:"finished",action:this,direction:t<0?-1:1})}}else{if(-1===r&&(t>=0?(r=0,this._setEndings(!0,0===this.repetitions,s)):this._setEndings(0===this.repetitions,!0,s)),n>=e||n<0){const i=Math.floor(n/e);n-=e*i,r+=Math.abs(i);const a=this.repetitions-r;if(a<=0)this.clampWhenFinished?this.paused=!0:this.enabled=!1,n=t>0?e:0,this.time=n,this._mixer.dispatchEvent({type:"finished",action:this,direction:t>0?1:-1});else{if(1===a){const e=t<0;this._setEndings(e,!e,s)}else this._setEndings(!1,!1,s);this._loopCount=r,this.time=n,this._mixer.dispatchEvent({type:"loop",action:this,loopDelta:i})}}else this.time=n;if(s&&1==(1&r))return e-n}return n}_setEndings(t,e,i){const n=this._interpolantSettings;i?(n.endingStart=it,n.endingEnd=it):(n.endingStart=t?this.zeroSlopeAtStart?it:et:nt,n.endingEnd=e?this.zeroSlopeAtEnd?it:et:nt)}_scheduleFading(t,e,i){const n=this._mixer,r=n.time;let s=this._weightInterpolant;null===s&&(s=n._lendControlInterpolant(),this._weightInterpolant=s);const a=s.parameterPositions,o=s.sampleValues;return a[0]=r,o[0]=e,a[1]=r+t,o[1]=i,this}}const hh=new Float32Array(1);class uh{constructor(t){this.value=t}clone(){return new uh(void 0===this.value.clone?this.value:this.value.clone())}}let dh=0;function ph(t,e){return t.distance-e.distance}function mh(t,e,i,n){if(t.layers.test(e.layers)&&t.raycast(e,i),!0===n){const n=t.children;for(let t=0,r=n.length;t<r;t++)mh(n[t],e,i,!0)}}const fh=new Lt;const gh=new ne,vh=new ne;const xh=new ne;const _h=new ne,yh=new Ie,Mh=new Ie;function bh(t){const e=[];!0===t.isBone&&e.push(t);for(let i=0;i<t.children.length;i++)e.push.apply(e,bh(t.children[i]));return e}const Sh=new ne,wh=new jt,Th=new jt;const Ah=new ne,Eh=new ne,Ch=new ne;const Lh=new ne,Rh=new rn;function Ph(t,e,i,n,r,s,a){Lh.set(r,s,a).unproject(n);const o=e[t];if(void 0!==o){const t=i.getAttribute("position");for(let e=0,i=o.length;e<i;e++)t.setXYZ(o[e],Lh.x,Lh.y,Lh.z)}}const Dh=new ae;const Ih=new ne;let Nh,Oh;const zh=Uh();function Uh(){const t=new ArrayBuffer(4),e=new Float32Array(t),i=new Uint32Array(t),n=new Uint32Array(512),r=new Uint32Array(512);for(let t=0;t<256;++t){const e=t-127;e<-27?(n[t]=0,n[256|t]=32768,r[t]=24,r[256|t]=24):e<-14?(n[t]=1024>>-e-14,n[256|t]=1024>>-e-14|32768,r[t]=-e-1,r[256|t]=-e-1):e<=15?(n[t]=e+15<<10,n[256|t]=e+15<<10|32768,r[t]=13,r[256|t]=13):e<128?(n[t]=31744,n[256|t]=64512,r[t]=24,r[256|t]=24):(n[t]=31744,n[256|t]=64512,r[t]=13,r[256|t]=13)}const s=new Uint32Array(2048),a=new Uint32Array(64),o=new Uint32Array(64);for(let t=1;t<1024;++t){let e=t<<13,i=0;for(;0==(8388608&e);)e<<=1,i-=8388608;e&=-8388609,i+=947912704,s[t]=e|i}for(let t=1024;t<2048;++t)s[t]=939524096+(t-1024<<13);for(let t=1;t<31;++t)a[t]=t<<23;a[31]=1199570944,a[32]=2147483648;for(let t=33;t<63;++t)a[t]=2147483648+(t-32<<23);a[63]=3347054592;for(let t=1;t<64;++t)32!==t&&(o[t]=1024);return{floatView:e,uint32View:i,baseTable:n,shiftTable:r,mantissaTable:s,exponentTable:a,offsetTable:o}}var Bh=Object.freeze({__proto__:null,toHalfFloat:function(t){Math.abs(t)>65504&&console.warn("THREE.DataUtils.toHalfFloat(): Value out of range."),t=yt(t,-65504,65504),zh.floatView[0]=t;const e=zh.uint32View[0],i=e>>23&511;return zh.baseTable[i]+((8388607&e)>>zh.shiftTable[i])},fromHalfFloat:function(t){const e=t>>10;return zh.uint32View[0]=zh.mantissaTable[zh.offsetTable[e]+(1023&t)]+zh.exponentTable[e],zh.floatView[0]}});"undefined"!=typeof __THREE_DEVTOOLS__&&__THREE_DEVTOOLS__.dispatchEvent(new CustomEvent("register",{detail:{revision:e}})),"undefined"!=typeof window&&(window.__THREE__?console.warn("WARNING: Multiple instances of Three.js being imported."):window.__THREE__=e),t.ACESFilmicToneMapping=4,t.AddEquation=i,t.AddOperation=2,t.AdditiveAnimationBlendMode=st,t.AdditiveBlending=2,t.AlphaFormat=1021,t.AlwaysDepth=1,t.AlwaysStencilFunc=519,t.AmbientLight=Cc,t.AmbientLightProbe=class extends Pc{constructor(t,e=1){super(void 0,e),this.isAmbientLightProbe=!0;const i=(new jt).set(t);this.sh.coefficients[0].set(i.r,i.g,i.b).multiplyScalar(2*Math.sqrt(Math.PI))}},t.AnimationClip=nc,t.AnimationLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=this,s=new uc(this.manager);s.setPath(this.path),s.setRequestHeader(this.requestHeader),s.setWithCredentials(this.withCredentials),s.load(t,(function(i){try{e(r.parse(JSON.parse(i)))}catch(e){n?n(e):console.error(e),r.manager.itemError(t)}}),i,n)}parse(t){const e=[];for(let i=0;i<t.length;i++){const n=nc.parse(t[i]);e.push(n)}return e}},t.AnimationMixer=class extends mt{constructor(t){super(),this._root=t,this._initMemoryManager(),this._accuIndex=0,this.time=0,this.timeScale=1}_bindAction(t,e){const i=t._localRoot||this._root,n=t._clip.tracks,r=n.length,s=t._propertyBindings,a=t._interpolants,o=i.uuid,l=this._bindingsByRootAndName;let c=l[o];void 0===c&&(c={},l[o]=c);for(let t=0;t!==r;++t){const r=n[t],l=r.name;let h=c[l];if(void 0!==h)++h.referenceCount,s[t]=h;else{if(h=s[t],void 0!==h){null===h._cacheIndex&&(++h.referenceCount,this._addInactiveBinding(h,o,l));continue}const n=e&&e._propertyBindings[t].binding.parsedPath;h=new eh(lh.create(i,l,n),r.ValueTypeName,r.getValueSize()),++h.referenceCount,this._addInactiveBinding(h,o,l),s[t]=h}a[t].resultBuffer=h.buffer}}_activateAction(t){if(!this._isActiveAction(t)){if(null===t._cacheIndex){const e=(t._localRoot||this._root).uuid,i=t._clip.uuid,n=this._actionsByClip[i];this._bindAction(t,n&&n.knownActions[0]),this._addInactiveAction(t,i,e)}const e=t._propertyBindings;for(let t=0,i=e.length;t!==i;++t){const i=e[t];0==i.useCount++&&(this._lendBinding(i),i.saveOriginalState())}this._lendAction(t)}}_deactivateAction(t){if(this._isActiveAction(t)){const e=t._propertyBindings;for(let t=0,i=e.length;t!==i;++t){const i=e[t];0==--i.useCount&&(i.restoreOriginalState(),this._takeBackBinding(i))}this._takeBackAction(t)}}_initMemoryManager(){this._actions=[],this._nActiveActions=0,this._actionsByClip={},this._bindings=[],this._nActiveBindings=0,this._bindingsByRootAndName={},this._controlInterpolants=[],this._nActiveControlInterpolants=0;const t=this;this.stats={actions:{get total(){return t._actions.length},get inUse(){return t._nActiveActions}},bindings:{get total(){return t._bindings.length},get inUse(){return t._nActiveBindings}},controlInterpolants:{get total(){return t._controlInterpolants.length},get inUse(){return t._nActiveControlInterpolants}}}}_isActiveAction(t){const e=t._cacheIndex;return null!==e&&e<this._nActiveActions}_addInactiveAction(t,e,i){const n=this._actions,r=this._actionsByClip;let s=r[e];if(void 0===s)s={knownActions:[t],actionByRoot:{}},t._byClipCacheIndex=0,r[e]=s;else{const e=s.knownActions;t._byClipCacheIndex=e.length,e.push(t)}t._cacheIndex=n.length,n.push(t),s.actionByRoot[i]=t}_removeInactiveAction(t){const e=this._actions,i=e[e.length-1],n=t._cacheIndex;i._cacheIndex=n,e[n]=i,e.pop(),t._cacheIndex=null;const r=t._clip.uuid,s=this._actionsByClip,a=s[r],o=a.knownActions,l=o[o.length-1],c=t._byClipCacheIndex;l._byClipCacheIndex=c,o[c]=l,o.pop(),t._byClipCacheIndex=null;delete a.actionByRoot[(t._localRoot||this._root).uuid],0===o.length&&delete s[r],this._removeInactiveBindingsForAction(t)}_removeInactiveBindingsForAction(t){const e=t._propertyBindings;for(let t=0,i=e.length;t!==i;++t){const i=e[t];0==--i.referenceCount&&this._removeInactiveBinding(i)}}_lendAction(t){const e=this._actions,i=t._cacheIndex,n=this._nActiveActions++,r=e[n];t._cacheIndex=n,e[n]=t,r._cacheIndex=i,e[i]=r}_takeBackAction(t){const e=this._actions,i=t._cacheIndex,n=--this._nActiveActions,r=e[n];t._cacheIndex=n,e[n]=t,r._cacheIndex=i,e[i]=r}_addInactiveBinding(t,e,i){const n=this._bindingsByRootAndName,r=this._bindings;let s=n[e];void 0===s&&(s={},n[e]=s),s[i]=t,t._cacheIndex=r.length,r.push(t)}_removeInactiveBinding(t){const e=this._bindings,i=t.binding,n=i.rootNode.uuid,r=i.path,s=this._bindingsByRootAndName,a=s[n],o=e[e.length-1],l=t._cacheIndex;o._cacheIndex=l,e[l]=o,e.pop(),delete a[r],0===Object.keys(a).length&&delete s[n]}_lendBinding(t){const e=this._bindings,i=t._cacheIndex,n=this._nActiveBindings++,r=e[n];t._cacheIndex=n,e[n]=t,r._cacheIndex=i,e[i]=r}_takeBackBinding(t){const e=this._bindings,i=t._cacheIndex,n=--this._nActiveBindings,r=e[n];t._cacheIndex=n,e[n]=t,r._cacheIndex=i,e[i]=r}_lendControlInterpolant(){const t=this._controlInterpolants,e=this._nActiveControlInterpolants++;let i=t[e];return void 0===i&&(i=new Xl(new Float32Array(2),new Float32Array(2),1,hh),i.__cacheIndex=e,t[e]=i),i}_takeBackControlInterpolant(t){const e=this._controlInterpolants,i=t.__cacheIndex,n=--this._nActiveControlInterpolants,r=e[n];t.__cacheIndex=n,e[n]=t,r.__cacheIndex=i,e[i]=r}clipAction(t,e,i){const n=e||this._root,r=n.uuid;let s="string"==typeof t?nc.findByName(n,t):t;const a=null!==s?s.uuid:t,o=this._actionsByClip[a];let l=null;if(void 0===i&&(i=null!==s?s.blendMode:rt),void 0!==o){const t=o.actionByRoot[r];if(void 0!==t&&t.blendMode===i)return t;l=o.knownActions[0],null===s&&(s=l._clip)}if(null===s)return null;const c=new ch(this,s,e,i);return this._bindAction(c,l),this._addInactiveAction(c,a,r),c}existingAction(t,e){const i=e||this._root,n=i.uuid,r="string"==typeof t?nc.findByName(i,t):t,s=r?r.uuid:t,a=this._actionsByClip[s];return void 0!==a&&a.actionByRoot[n]||null}stopAllAction(){const t=this._actions;for(let e=this._nActiveActions-1;e>=0;--e)t[e].stop();return this}update(t){t*=this.timeScale;const e=this._actions,i=this._nActiveActions,n=this.time+=t,r=Math.sign(t),s=this._accuIndex^=1;for(let a=0;a!==i;++a){e[a]._update(n,t,r,s)}const a=this._bindings,o=this._nActiveBindings;for(let t=0;t!==o;++t)a[t].apply(s);return this}setTime(t){this.time=0;for(let t=0;t<this._actions.length;t++)this._actions[t].time=0;return this.update(t)}getRoot(){return this._root}uncacheClip(t){const e=this._actions,i=t.uuid,n=this._actionsByClip,r=n[i];if(void 0!==r){const t=r.knownActions;for(let i=0,n=t.length;i!==n;++i){const n=t[i];this._deactivateAction(n);const r=n._cacheIndex,s=e[e.length-1];n._cacheIndex=null,n._byClipCacheIndex=null,s._cacheIndex=r,e[r]=s,e.pop(),this._removeInactiveBindingsForAction(n)}delete n[i]}}uncacheRoot(t){const e=t.uuid,i=this._actionsByClip;for(const t in i){const n=i[t].actionByRoot[e];void 0!==n&&(this._deactivateAction(n),this._removeInactiveAction(n))}const n=this._bindingsByRootAndName[e];if(void 0!==n)for(const t in n){const e=n[t];e.restoreOriginalState(),this._removeInactiveBinding(e)}}uncacheAction(t,e){const i=this.existingAction(t,e);null!==i&&(this._deactivateAction(i),this._removeInactiveAction(i))}},t.AnimationObjectGroup=class{constructor(){this.isAnimationObjectGroup=!0,this.uuid=_t(),this._objects=Array.prototype.slice.call(arguments),this.nCachedObjects_=0;const t={};this._indicesByUUID=t;for(let e=0,i=arguments.length;e!==i;++e)t[arguments[e].uuid]=e;this._paths=[],this._parsedPaths=[],this._bindings=[],this._bindingsIndicesByPath={};const e=this;this.stats={objects:{get total(){return e._objects.length},get inUse(){return this.total-e.nCachedObjects_}},get bindingsPerObject(){return e._bindings.length}}}add(){const t=this._objects,e=this._indicesByUUID,i=this._paths,n=this._parsedPaths,r=this._bindings,s=r.length;let a,o=t.length,l=this.nCachedObjects_;for(let c=0,h=arguments.length;c!==h;++c){const h=arguments[c],u=h.uuid;let d=e[u];if(void 0===d){d=o++,e[u]=d,t.push(h);for(let t=0,e=s;t!==e;++t)r[t].push(new lh(h,i[t],n[t]))}else if(d<l){a=t[d];const o=--l,c=t[o];e[c.uuid]=d,t[d]=c,e[u]=o,t[o]=h;for(let t=0,e=s;t!==e;++t){const e=r[t],s=e[o];let a=e[d];e[d]=s,void 0===a&&(a=new lh(h,i[t],n[t])),e[o]=a}}else t[d]!==a&&console.error("THREE.AnimationObjectGroup: Different objects with the same UUID detected. Clean the caches or recreate your infrastructure when reloading scenes.")}this.nCachedObjects_=l}remove(){const t=this._objects,e=this._indicesByUUID,i=this._bindings,n=i.length;let r=this.nCachedObjects_;for(let s=0,a=arguments.length;s!==a;++s){const a=arguments[s],o=a.uuid,l=e[o];if(void 0!==l&&l>=r){const s=r++,c=t[s];e[c.uuid]=l,t[l]=c,e[o]=s,t[s]=a;for(let t=0,e=n;t!==e;++t){const e=i[t],n=e[s],r=e[l];e[l]=n,e[s]=r}}}this.nCachedObjects_=r}uncache(){const t=this._objects,e=this._indicesByUUID,i=this._bindings,n=i.length;let r=this.nCachedObjects_,s=t.length;for(let a=0,o=arguments.length;a!==o;++a){const o=arguments[a].uuid,l=e[o];if(void 0!==l)if(delete e[o],l<r){const a=--r,o=t[a],c=--s,h=t[c];e[o.uuid]=l,t[l]=o,e[h.uuid]=a,t[a]=h,t.pop();for(let t=0,e=n;t!==e;++t){const e=i[t],n=e[a],r=e[c];e[l]=n,e[a]=r,e.pop()}}else{const r=--s,a=t[r];r>0&&(e[a.uuid]=l),t[l]=a,t.pop();for(let t=0,e=n;t!==e;++t){const e=i[t];e[l]=e[r],e.pop()}}}this.nCachedObjects_=r}subscribe_(t,e){const i=this._bindingsIndicesByPath;let n=i[t];const r=this._bindings;if(void 0!==n)return r[n];const s=this._paths,a=this._parsedPaths,o=this._objects,l=o.length,c=this.nCachedObjects_,h=new Array(l);n=r.length,i[t]=n,s.push(t),a.push(e),r.push(h);for(let i=c,n=o.length;i!==n;++i){const n=o[i];h[i]=new lh(n,t,e)}return h}unsubscribe_(t){const e=this._bindingsIndicesByPath,i=e[t];if(void 0!==i){const n=this._paths,r=this._parsedPaths,s=this._bindings,a=s.length-1,o=s[a];e[t[a]]=i,s[i]=o,s.pop(),r[i]=r[a],r.pop(),n[i]=n[a],n.pop()}}},t.AnimationUtils=Wl,t.ArcCurve=ao,t.ArrayCamera=ks,t.ArrowHelper=class extends ri{constructor(t=new ne(0,0,1),e=new ne(0,0,0),i=1,n=16776960,r=.2*i,s=.2*r){super(),this.type="ArrowHelper",void 0===Nh&&(Nh=new Di,Nh.setAttribute("position",new wi([0,0,0,0,1,0],3)),Oh=new Ro(0,.5,1,5,1),Oh.translate(0,-.5,0)),this.position.copy(e),this.line=new ja(Nh,new Fa({color:n,toneMapped:!1})),this.line.matrixAutoUpdate=!1,this.add(this.line),this.cone=new Ji(Oh,new xi({color:n,toneMapped:!1})),this.cone.matrixAutoUpdate=!1,this.add(this.cone),this.setDirection(t),this.setLength(i,r,s)}setDirection(t){if(t.y>.99999)this.quaternion.set(0,0,0,1);else if(t.y<-.99999)this.quaternion.set(1,0,0,0);else{Ih.set(t.z,0,-t.x).normalize();const e=Math.acos(t.y);this.quaternion.setFromAxisAngle(Ih,e)}}setLength(t,e=.2*t,i=.2*e){this.line.scale.set(1,Math.max(1e-4,t-e),1),this.line.updateMatrix(),this.cone.scale.set(i,e,i),this.cone.position.y=t,this.cone.updateMatrix()}setColor(t){this.line.material.color.set(t),this.cone.material.color.set(t)}copy(t){return super.copy(t,!1),this.line.copy(t.line),this.cone.copy(t.cone),this}dispose(){this.line.geometry.dispose(),this.line.material.dispose(),this.cone.geometry.dispose(),this.cone.material.dispose()}},t.Audio=Jc,t.AudioAnalyser=class{constructor(t,e=2048){this.analyser=t.context.createAnalyser(),this.analyser.fftSize=e,this.data=new Uint8Array(this.analyser.frequencyBinCount),t.getOutput().connect(this.analyser)}getFrequencyData(){return this.analyser.getByteFrequencyData(this.data),this.data}getAverageFrequency(){let t=0;const e=this.getFrequencyData();for(let i=0;i<e.length;i++)t+=e[i];return t/e.length}},t.AudioContext=kc,t.AudioListener=class extends ri{constructor(){super(),this.type="AudioListener",this.context=kc.getContext(),this.gain=this.context.createGain(),this.gain.connect(this.context.destination),this.filter=null,this.timeDelta=0,this._clock=new Wc}getInput(){return this.gain}removeFilter(){return null!==this.filter&&(this.gain.disconnect(this.filter),this.filter.disconnect(this.context.destination),this.gain.connect(this.context.destination),this.filter=null),this}getFilter(){return this.filter}setFilter(t){return null!==this.filter?(this.gain.disconnect(this.filter),this.filter.disconnect(this.context.destination)):this.gain.disconnect(this.context.destination),this.filter=t,this.gain.connect(this.filter),this.filter.connect(this.context.destination),this}getMasterVolume(){return this.gain.gain.value}setMasterVolume(t){return this.gain.gain.setTargetAtTime(t,this.context.currentTime,.01),this}updateMatrixWorld(t){super.updateMatrixWorld(t);const e=this.context.listener,i=this.up;if(this.timeDelta=this._clock.getDelta(),this.matrixWorld.decompose(qc,Xc,Yc),Zc.set(0,0,-1).applyQuaternion(Xc),e.positionX){const t=this.context.currentTime+this.timeDelta;e.positionX.linearRampToValueAtTime(qc.x,t),e.positionY.linearRampToValueAtTime(qc.y,t),e.positionZ.linearRampToValueAtTime(qc.z,t),e.forwardX.linearRampToValueAtTime(Zc.x,t),e.forwardY.linearRampToValueAtTime(Zc.y,t),e.forwardZ.linearRampToValueAtTime(Zc.z,t),e.upX.linearRampToValueAtTime(i.x,t),e.upY.linearRampToValueAtTime(i.y,t),e.upZ.linearRampToValueAtTime(i.z,t)}else e.setPosition(qc.x,qc.y,qc.z),e.setOrientation(Zc.x,Zc.y,Zc.z,i.x,i.y,i.z)}},t.AudioLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=this,s=new uc(this.manager);s.setResponseType("arraybuffer"),s.setPath(this.path),s.setRequestHeader(this.requestHeader),s.setWithCredentials(this.withCredentials),s.load(t,(function(i){try{const t=i.slice(0);kc.getContext().decodeAudioData(t,(function(t){e(t)}))}catch(e){n?n(e):console.error(e),r.manager.itemError(t)}}),i,n)}},t.AxesHelper=class extends Ya{constructor(t=1){const e=[0,0,0,t,0,0,0,0,0,0,t,0,0,0,0,0,0,t],i=new Di;i.setAttribute("position",new wi(e,3)),i.setAttribute("color",new wi([1,0,0,1,.6,0,0,1,0,.6,1,0,0,0,1,0,.6,1],3));super(i,new Fa({vertexColors:!0,toneMapped:!1})),this.type="AxesHelper"}setColors(t,e,i){const n=new jt,r=this.geometry.attributes.color.array;return n.set(t),n.toArray(r,0),n.toArray(r,3),n.set(e),n.toArray(r,6),n.toArray(r,9),n.set(i),n.toArray(r,12),n.toArray(r,15),this.geometry.attributes.color.needsUpdate=!0,this}dispose(){this.geometry.dispose(),this.material.dispose()}},t.BackSide=1,t.BasicDepthPacking=3200,t.BasicShadowMap=0,t.Bone=Ea,t.BooleanKeyframeTrack=Jl,t.Box2=class{constructor(t=new Lt(1/0,1/0),e=new Lt(-1/0,-1/0)){this.isBox2=!0,this.min=t,this.max=e}set(t,e){return this.min.copy(t),this.max.copy(e),this}setFromPoints(t){this.makeEmpty();for(let e=0,i=t.length;e<i;e++)this.expandByPoint(t[e]);return this}setFromCenterAndSize(t,e){const i=fh.copy(e).multiplyScalar(.5);return this.min.copy(t).sub(i),this.max.copy(t).add(i),this}clone(){return(new this.constructor).copy(this)}copy(t){return this.min.copy(t.min),this.max.copy(t.max),this}makeEmpty(){return this.min.x=this.min.y=1/0,this.max.x=this.max.y=-1/0,this}isEmpty(){return this.max.x<this.min.x||this.max.y<this.min.y}getCenter(t){return this.isEmpty()?t.set(0,0):t.addVectors(this.min,this.max).multiplyScalar(.5)}getSize(t){return this.isEmpty()?t.set(0,0):t.subVectors(this.max,this.min)}expandByPoint(t){return this.min.min(t),this.max.max(t),this}expandByVector(t){return this.min.sub(t),this.max.add(t),this}expandByScalar(t){return this.min.addScalar(-t),this.max.addScalar(t),this}containsPoint(t){return!(t.x<this.min.x||t.x>this.max.x||t.y<this.min.y||t.y>this.max.y)}containsBox(t){return this.min.x<=t.min.x&&t.max.x<=this.max.x&&this.min.y<=t.min.y&&t.max.y<=this.max.y}getParameter(t,e){return e.set((t.x-this.min.x)/(this.max.x-this.min.x),(t.y-this.min.y)/(this.max.y-this.min.y))}intersectsBox(t){return!(t.max.x<this.min.x||t.min.x>this.max.x||t.max.y<this.min.y||t.min.y>this.max.y)}clampPoint(t,e){return e.copy(t).clamp(this.min,this.max)}distanceToPoint(t){return fh.copy(t).clamp(this.min,this.max).sub(t).length()}intersect(t){return this.min.max(t.min),this.max.min(t.max),this}union(t){return this.min.min(t.min),this.max.max(t.max),this}translate(t){return this.min.add(t),this.max.add(t),this}equals(t){return t.min.equals(this.min)&&t.max.equals(this.max)}},t.Box3=ae,t.Box3Helper=class extends Ya{constructor(t,e=16776960){const i=new Uint16Array([0,1,1,2,2,3,3,0,4,5,5,6,6,7,7,4,0,4,1,5,2,6,3,7]),n=new Di;n.setIndex(new Mi(i,1)),n.setAttribute("position",new wi([1,1,1,-1,1,1,-1,-1,1,1,-1,1,1,1,-1,-1,1,-1,-1,-1,-1,1,-1,-1],3)),super(n,new Fa({color:e,toneMapped:!1})),this.box=t,this.type="Box3Helper",this.geometry.computeBoundingSphere()}updateMatrixWorld(t){const e=this.box;e.isEmpty()||(e.getCenter(this.position),e.getSize(this.scale),this.scale.multiplyScalar(.5),super.updateMatrixWorld(t))}dispose(){this.geometry.dispose(),this.material.dispose()}},t.BoxBufferGeometry=class extends $i{constructor(t,e,i,n,r,s){console.warn("THREE.BoxBufferGeometry has been renamed to THREE.BoxGeometry."),super(t,e,i,n,r,s)}},t.BoxGeometry=$i,t.BoxHelper=class extends Ya{constructor(t,e=16776960){const i=new Uint16Array([0,1,1,2,2,3,3,0,4,5,5,6,6,7,7,4,0,4,1,5,2,6,3,7]),n=new Float32Array(24),r=new Di;r.setIndex(new Mi(i,1)),r.setAttribute("position",new Mi(n,3)),super(r,new Fa({color:e,toneMapped:!1})),this.object=t,this.type="BoxHelper",this.matrixAutoUpdate=!1,this.update()}update(t){if(void 0!==t&&console.warn("THREE.BoxHelper: .update() has no longer arguments."),void 0!==this.object&&Dh.setFromObject(this.object),Dh.isEmpty())return;const e=Dh.min,i=Dh.max,n=this.geometry.attributes.position,r=n.array;r[0]=i.x,r[1]=i.y,r[2]=i.z,r[3]=e.x,r[4]=i.y,r[5]=i.z,r[6]=e.x,r[7]=e.y,r[8]=i.z,r[9]=i.x,r[10]=e.y,r[11]=i.z,r[12]=i.x,r[13]=i.y,r[14]=e.z,r[15]=e.x,r[16]=i.y,r[17]=e.z,r[18]=e.x,r[19]=e.y,r[20]=e.z,r[21]=i.x,r[22]=e.y,r[23]=e.z,n.needsUpdate=!0,this.geometry.computeBoundingSphere()}setFromObject(t){return this.object=t,this.update(),this}copy(t,e){return super.copy(t,e),this.object=t.object,this}dispose(){this.geometry.dispose(),this.material.dispose()}},t.BufferAttribute=Mi,t.BufferGeometry=Di,t.BufferGeometryLoader=Oc,t.ByteType=1010,t.Cache=sc,t.Camera=rn,t.CameraHelper=class extends Ya{constructor(t){const e=new Di,i=new Fa({color:16777215,vertexColors:!0,toneMapped:!1}),n=[],r=[],s={};function a(t,e){o(t),o(e)}function o(t){n.push(0,0,0),r.push(0,0,0),void 0===s[t]&&(s[t]=[]),s[t].push(n.length/3-1)}a("n1","n2"),a("n2","n4"),a("n4","n3"),a("n3","n1"),a("f1","f2"),a("f2","f4"),a("f4","f3"),a("f3","f1"),a("n1","f1"),a("n2","f2"),a("n3","f3"),a("n4","f4"),a("p","n1"),a("p","n2"),a("p","n3"),a("p","n4"),a("u1","u2"),a("u2","u3"),a("u3","u1"),a("c","t"),a("p","c"),a("cn1","cn2"),a("cn3","cn4"),a("cf1","cf2"),a("cf3","cf4"),e.setAttribute("position",new wi(n,3)),e.setAttribute("color",new wi(r,3)),super(e,i),this.type="CameraHelper",this.camera=t,this.camera.updateProjectionMatrix&&this.camera.updateProjectionMatrix(),this.matrix=t.matrixWorld,this.matrixAutoUpdate=!1,this.pointMap=s,this.update();const l=new jt(16755200),c=new jt(16711680),h=new jt(43775),u=new jt(16777215),d=new jt(3355443);this.setColors(l,c,h,u,d)}setColors(t,e,i,n,r){const s=this.geometry.getAttribute("color");s.setXYZ(0,t.r,t.g,t.b),s.setXYZ(1,t.r,t.g,t.b),s.setXYZ(2,t.r,t.g,t.b),s.setXYZ(3,t.r,t.g,t.b),s.setXYZ(4,t.r,t.g,t.b),s.setXYZ(5,t.r,t.g,t.b),s.setXYZ(6,t.r,t.g,t.b),s.setXYZ(7,t.r,t.g,t.b),s.setXYZ(8,t.r,t.g,t.b),s.setXYZ(9,t.r,t.g,t.b),s.setXYZ(10,t.r,t.g,t.b),s.setXYZ(11,t.r,t.g,t.b),s.setXYZ(12,t.r,t.g,t.b),s.setXYZ(13,t.r,t.g,t.b),s.setXYZ(14,t.r,t.g,t.b),s.setXYZ(15,t.r,t.g,t.b),s.setXYZ(16,t.r,t.g,t.b),s.setXYZ(17,t.r,t.g,t.b),s.setXYZ(18,t.r,t.g,t.b),s.setXYZ(19,t.r,t.g,t.b),s.setXYZ(20,t.r,t.g,t.b),s.setXYZ(21,t.r,t.g,t.b),s.setXYZ(22,t.r,t.g,t.b),s.setXYZ(23,t.r,t.g,t.b),s.setXYZ(24,e.r,e.g,e.b),s.setXYZ(25,e.r,e.g,e.b),s.setXYZ(26,e.r,e.g,e.b),s.setXYZ(27,e.r,e.g,e.b),s.setXYZ(28,e.r,e.g,e.b),s.setXYZ(29,e.r,e.g,e.b),s.setXYZ(30,e.r,e.g,e.b),s.setXYZ(31,e.r,e.g,e.b),s.setXYZ(32,i.r,i.g,i.b),s.setXYZ(33,i.r,i.g,i.b),s.setXYZ(34,i.r,i.g,i.b),s.setXYZ(35,i.r,i.g,i.b),s.setXYZ(36,i.r,i.g,i.b),s.setXYZ(37,i.r,i.g,i.b),s.setXYZ(38,n.r,n.g,n.b),s.setXYZ(39,n.r,n.g,n.b),s.setXYZ(40,r.r,r.g,r.b),s.setXYZ(41,r.r,r.g,r.b),s.setXYZ(42,r.r,r.g,r.b),s.setXYZ(43,r.r,r.g,r.b),s.setXYZ(44,r.r,r.g,r.b),s.setXYZ(45,r.r,r.g,r.b),s.setXYZ(46,r.r,r.g,r.b),s.setXYZ(47,r.r,r.g,r.b),s.setXYZ(48,r.r,r.g,r.b),s.setXYZ(49,r.r,r.g,r.b),s.needsUpdate=!0}update(){const t=this.geometry,e=this.pointMap;Rh.projectionMatrixInverse.copy(this.camera.projectionMatrixInverse),Ph("c",e,t,Rh,0,0,-1),Ph("t",e,t,Rh,0,0,1),Ph("n1",e,t,Rh,-1,-1,-1),Ph("n2",e,t,Rh,1,-1,-1),Ph("n3",e,t,Rh,-1,1,-1),Ph("n4",e,t,Rh,1,1,-1),Ph("f1",e,t,Rh,-1,-1,1),Ph("f2",e,t,Rh,1,-1,1),Ph("f3",e,t,Rh,-1,1,1),Ph("f4",e,t,Rh,1,1,1),Ph("u1",e,t,Rh,.7,1.1,-1),Ph("u2",e,t,Rh,-.7,1.1,-1),Ph("u3",e,t,Rh,0,2,-1),Ph("cf1",e,t,Rh,-1,0,1),Ph("cf2",e,t,Rh,1,0,1),Ph("cf3",e,t,Rh,0,-1,1),Ph("cf4",e,t,Rh,0,1,1),Ph("cn1",e,t,Rh,-1,0,-1),Ph("cn2",e,t,Rh,1,0,-1),Ph("cn3",e,t,Rh,0,-1,-1),Ph("cn4",e,t,Rh,0,1,-1),t.getAttribute("position").needsUpdate=!0}dispose(){this.geometry.dispose(),this.material.dispose()}},t.CanvasTexture=class extends Kt{constructor(t,e,i,n,r,s,a,o,l){super(t,e,i,n,r,s,a,o,l),this.isCanvasTexture=!0,this.needsUpdate=!0}},t.CapsuleBufferGeometry=class extends Co{constructor(t,e,i,n){console.warn("THREE.CapsuleBufferGeometry has been renamed to THREE.CapsuleGeometry."),super(t,e,i,n)}},t.CapsuleGeometry=Co,t.CatmullRomCurve3=po,t.CineonToneMapping=3,t.CircleBufferGeometry=class extends Lo{constructor(t,e,i,n){console.warn("THREE.CircleBufferGeometry has been renamed to THREE.CircleGeometry."),super(t,e,i,n)}},t.CircleGeometry=Lo,t.ClampToEdgeWrapping=h,t.Clock=Wc,t.Color=jt,t.ColorKeyframeTrack=Kl,t.ColorManagement=Bt,t.CompressedArrayTexture=class extends no{constructor(t,e,i,n,r,s){super(t,e,i,r,s),this.isCompressedArrayTexture=!0,this.image.depth=n,this.wrapR=h}},t.CompressedTexture=no,t.CompressedTextureLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=this,s=[],a=new no,o=new uc(this.manager);o.setPath(this.path),o.setResponseType("arraybuffer"),o.setRequestHeader(this.requestHeader),o.setWithCredentials(r.withCredentials);let l=0;function c(c){o.load(t[c],(function(t){const i=r.parse(t,!0);s[c]={width:i.width,height:i.height,format:i.format,mipmaps:i.mipmaps},l+=1,6===l&&(1===i.mipmapCount&&(a.minFilter=f),a.image=s,a.format=i.format,a.needsUpdate=!0,e&&e(a))}),i,n)}if(Array.isArray(t))for(let e=0,i=t.length;e<i;++e)c(e);else o.load(t,(function(t){const i=r.parse(t,!0);if(i.isCubemap){const t=i.mipmaps.length/i.mipmapCount;for(let e=0;e<t;e++){s[e]={mipmaps:[]};for(let t=0;t<i.mipmapCount;t++)s[e].mipmaps.push(i.mipmaps[e*i.mipmapCount+t]),s[e].format=i.format,s[e].width=i.width,s[e].height=i.height}a.image=s}else a.image.width=i.width,a.image.height=i.height,a.mipmaps=i.mipmaps;1===i.mipmapCount&&(a.minFilter=f),a.format=i.format,a.needsUpdate=!0,e&&e(a)}),i,n);return a}},t.ConeBufferGeometry=class extends Po{constructor(t,e,i,n,r,s,a){console.warn("THREE.ConeBufferGeometry has been renamed to THREE.ConeGeometry."),super(t,e,i,n,r,s,a)}},t.ConeGeometry=Po,t.CubeCamera=on,t.CubeReflectionMapping=r,t.CubeRefractionMapping=s,t.CubeTexture=ln,t.CubeTextureLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=new ln,s=new dc(this.manager);s.setCrossOrigin(this.crossOrigin),s.setPath(this.path);let a=0;function o(i){s.load(t[i],(function(t){r.images[i]=t,a++,6===a&&(r.needsUpdate=!0,e&&e(r))}),void 0,n)}for(let e=0;e<t.length;++e)o(e);return r}},t.CubeUVReflectionMapping=l,t.CubicBezierCurve=vo,t.CubicBezierCurve3=xo,t.CubicInterpolant=ql,t.CullFaceBack=1,t.CullFaceFront=2,t.CullFaceFrontBack=3,t.CullFaceNone=0,t.Curve=ro,t.CurvePath=To,t.CustomBlending=5,t.CustomToneMapping=5,t.CylinderBufferGeometry=class extends Ro{constructor(t,e,i,n,r,s,a,o){console.warn("THREE.CylinderBufferGeometry has been renamed to THREE.CylinderGeometry."),super(t,e,i,n,r,s,a,o)}},t.CylinderGeometry=Ro,t.Cylindrical=class{constructor(t=1,e=0,i=0){return this.radius=t,this.theta=e,this.y=i,this}set(t,e,i){return this.radius=t,this.theta=e,this.y=i,this}copy(t){return this.radius=t.radius,this.theta=t.theta,this.y=t.y,this}setFromVector3(t){return this.setFromCartesianCoords(t.x,t.y,t.z)}setFromCartesianCoords(t,e,i){return this.radius=Math.sqrt(t*t+i*i),this.theta=Math.atan2(t,i),this.y=e,this}clone(){return(new this.constructor).copy(this)}},t.Data3DTexture=ee,t.DataArrayTexture=te,t.DataTexture=Ca,t.DataTexture2DArray=class extends te{constructor(t,e,i,n){console.warn("THREE.DataTexture2DArray has been renamed to DataArrayTexture."),super(t,e,i,n)}},t.DataTexture3D=class extends ee{constructor(t,e,i,n){console.warn("THREE.DataTexture3D has been renamed to Data3DTexture."),super(t,e,i,n)}},t.DataTextureLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=this,s=new Ca,a=new uc(this.manager);return a.setResponseType("arraybuffer"),a.setRequestHeader(this.requestHeader),a.setPath(this.path),a.setWithCredentials(r.withCredentials),a.load(t,(function(t){const i=r.parse(t);i&&(void 0!==i.image?s.image=i.image:void 0!==i.data&&(s.image.width=i.width,s.image.height=i.height,s.image.data=i.data),s.wrapS=void 0!==i.wrapS?i.wrapS:h,s.wrapT=void 0!==i.wrapT?i.wrapT:h,s.magFilter=void 0!==i.magFilter?i.magFilter:f,s.minFilter=void 0!==i.minFilter?i.minFilter:f,s.anisotropy=void 0!==i.anisotropy?i.anisotropy:1,void 0!==i.encoding&&(s.encoding=i.encoding),void 0!==i.flipY&&(s.flipY=i.flipY),void 0!==i.format&&(s.format=i.format),void 0!==i.type&&(s.type=i.type),void 0!==i.mipmaps&&(s.mipmaps=i.mipmaps,s.minFilter=v),1===i.mipmapCount&&(s.minFilter=f),void 0!==i.generateMipmaps&&(s.generateMipmaps=i.generateMipmaps),s.needsUpdate=!0,e&&e(s,i))}),i,n),s}},t.DataUtils=Bh,t.DecrementStencilOp=7683,t.DecrementWrapStencilOp=34056,t.DefaultLoadingManager=oc,t.DepthFormat=T,t.DepthStencilFormat=A,t.DepthTexture=Ws,t.DirectionalLight=Ec,t.DirectionalLightHelper=class extends ri{constructor(t,e,i){super(),this.light=t,this.light.updateMatrixWorld(),this.matrix=t.matrixWorld,this.matrixAutoUpdate=!1,this.color=i,void 0===e&&(e=1);let n=new Di;n.setAttribute("position",new wi([-e,e,0,e,e,0,e,-e,0,-e,-e,0,-e,e,0],3));const r=new Fa({fog:!1,toneMapped:!1});this.lightPlane=new ja(n,r),this.add(this.lightPlane),n=new Di,n.setAttribute("position",new wi([0,0,0,0,0,1],3)),this.targetLine=new ja(n,r),this.add(this.targetLine),this.update()}dispose(){this.lightPlane.geometry.dispose(),this.lightPlane.material.dispose(),this.targetLine.geometry.dispose(),this.targetLine.material.dispose()}update(){Ah.setFromMatrixPosition(this.light.matrixWorld),Eh.setFromMatrixPosition(this.light.target.matrixWorld),Ch.subVectors(Eh,Ah),this.lightPlane.lookAt(Eh),void 0!==this.color?(this.lightPlane.material.color.set(this.color),this.targetLine.material.color.set(this.color)):(this.lightPlane.material.color.copy(this.light.color),this.targetLine.material.color.copy(this.light.color)),this.targetLine.lookAt(Eh),this.targetLine.scale.z=Ch.length()}},t.DiscreteInterpolant=Yl,t.DodecahedronBufferGeometry=class extends Io{constructor(t,e){console.warn("THREE.DodecahedronBufferGeometry has been renamed to THREE.DodecahedronGeometry."),super(t,e)}},t.DodecahedronGeometry=Io,t.DoubleSide=2,t.DstAlphaFactor=206,t.DstColorFactor=208,t.DynamicCopyUsage=35050,t.DynamicDrawUsage=35048,t.DynamicReadUsage=35049,t.EdgesGeometry=Bo,t.EllipseCurve=so,t.EqualDepth=4,t.EqualStencilFunc=514,t.EquirectangularReflectionMapping=a,t.EquirectangularRefractionMapping=o,t.Euler=He,t.EventDispatcher=mt,t.ExtrudeBufferGeometry=class extends ml{constructor(t,e){console.warn("THREE.ExtrudeBufferGeometry has been renamed to THREE.ExtrudeGeometry."),super(t,e)}},t.ExtrudeGeometry=ml,t.FileLoader=uc,t.Float16BufferAttribute=class extends Mi{constructor(t,e,i){super(new Uint16Array(t),e,i),this.isFloat16BufferAttribute=!0}},t.Float32BufferAttribute=wi,t.Float64BufferAttribute=class extends Mi{constructor(t,e,i){super(new Float64Array(t),e,i)}},t.FloatType=M,t.Fog=Ks,t.FogExp2=Js,t.FramebufferTexture=class extends Kt{constructor(t,e,i){super({width:t,height:e}),this.isFramebufferTexture=!0,this.format=i,this.magFilter=d,this.minFilter=d,this.generateMipmaps=!1,this.needsUpdate=!0}},t.FrontSide=0,t.Frustum=gn,t.GLBufferAttribute=class{constructor(t,e,i,n,r){this.isGLBufferAttribute=!0,this.buffer=t,this.type=e,this.itemSize=i,this.elementSize=n,this.count=r,this.version=0}set needsUpdate(t){!0===t&&this.version++}setBuffer(t){return this.buffer=t,this}setType(t,e){return this.type=t,this.elementSize=e,this}setItemSize(t){return this.itemSize=t,this}setCount(t){return this.count=t,this}},t.GLSL1="100",t.GLSL3=dt,t.GreaterDepth=6,t.GreaterEqualDepth=5,t.GreaterEqualStencilFunc=518,t.GreaterStencilFunc=516,t.GridHelper=class extends Ya{constructor(t=10,e=10,i=4473924,n=8947848){i=new jt(i),n=new jt(n);const r=e/2,s=t/e,a=t/2,o=[],l=[];for(let t=0,c=0,h=-a;t<=e;t++,h+=s){o.push(-a,0,h,a,0,h),o.push(h,0,-a,h,0,a);const e=t===r?i:n;e.toArray(l,c),c+=3,e.toArray(l,c),c+=3,e.toArray(l,c),c+=3,e.toArray(l,c),c+=3}const c=new Di;c.setAttribute("position",new wi(o,3)),c.setAttribute("color",new wi(l,3));super(c,new Fa({vertexColors:!0,toneMapped:!1})),this.type="GridHelper"}dispose(){this.geometry.dispose(),this.material.dispose()}},t.Group=Gs,t.HalfFloatType=b,t.HemisphereLight=mc,t.HemisphereLightHelper=class extends ri{constructor(t,e,i){super(),this.light=t,this.light.updateMatrixWorld(),this.matrix=t.matrixWorld,this.matrixAutoUpdate=!1,this.color=i;const n=new vl(e);n.rotateY(.5*Math.PI),this.material=new xi({wireframe:!0,fog:!1,toneMapped:!1}),void 0===this.color&&(this.material.vertexColors=!0);const r=n.getAttribute("position"),s=new Float32Array(3*r.count);n.setAttribute("color",new Mi(s,3)),this.add(new Ji(n,this.material)),this.update()}dispose(){this.children[0].geometry.dispose(),this.children[0].material.dispose()}update(){const t=this.children[0];if(void 0!==this.color)this.material.color.set(this.color);else{const e=t.geometry.getAttribute("color");wh.copy(this.light.color),Th.copy(this.light.groundColor);for(let t=0,i=e.count;t<i;t++){const n=t<i/2?wh:Th;e.setXYZ(t,n.r,n.g,n.b)}e.needsUpdate=!0}t.lookAt(Sh.setFromMatrixPosition(this.light.matrixWorld).negate())}},t.HemisphereLightProbe=class extends Pc{constructor(t,e,i=1){super(void 0,i),this.isHemisphereLightProbe=!0;const n=(new jt).set(t),r=(new jt).set(e),s=new ne(n.r,n.g,n.b),a=new ne(r.r,r.g,r.b),o=Math.sqrt(Math.PI),l=o*Math.sqrt(.75);this.sh.coefficients[0].copy(s).add(a).multiplyScalar(o),this.sh.coefficients[1].copy(s).sub(a).multiplyScalar(l)}},t.IcosahedronBufferGeometry=class extends gl{constructor(t,e){console.warn("THREE.IcosahedronBufferGeometry has been renamed to THREE.IcosahedronGeometry."),super(t,e)}},t.IcosahedronGeometry=gl,t.ImageBitmapLoader=class extends lc{constructor(t){super(t),this.isImageBitmapLoader=!0,"undefined"==typeof createImageBitmap&&console.warn("THREE.ImageBitmapLoader: createImageBitmap() not supported."),"undefined"==typeof fetch&&console.warn("THREE.ImageBitmapLoader: fetch() not supported."),this.options={premultiplyAlpha:"none"}}setOptions(t){return this.options=t,this}load(t,e,i,n){void 0===t&&(t=""),void 0!==this.path&&(t=this.path+t),t=this.manager.resolveURL(t);const r=this,s=sc.get(t);if(void 0!==s)return r.manager.itemStart(t),setTimeout((function(){e&&e(s),r.manager.itemEnd(t)}),0),s;const a={};a.credentials="anonymous"===this.crossOrigin?"same-origin":"include",a.headers=this.requestHeader,fetch(t,a).then((function(t){return t.blob()})).then((function(t){return createImageBitmap(t,Object.assign(r.options,{colorSpaceConversion:"none"}))})).then((function(i){sc.add(t,i),e&&e(i),r.manager.itemEnd(t)})).catch((function(e){n&&n(e),r.manager.itemError(t),r.manager.itemEnd(t)})),r.manager.itemStart(t)}},t.ImageLoader=dc,t.ImageUtils=Xt,t.ImmediateRenderObject=function(){console.error("THREE.ImmediateRenderObject has been removed.")},t.IncrementStencilOp=7682,t.IncrementWrapStencilOp=34055,t.InstancedBufferAttribute=Da,t.InstancedBufferGeometry=Nc,t.InstancedInterleavedBuffer=class extends Qs{constructor(t,e,i=1){super(t,e),this.isInstancedInterleavedBuffer=!0,this.meshPerAttribute=i}copy(t){return super.copy(t),this.meshPerAttribute=t.meshPerAttribute,this}clone(t){const e=super.clone(t);return e.meshPerAttribute=this.meshPerAttribute,e}toJSON(t){const e=super.toJSON(t);return e.isInstancedInterleavedBuffer=!0,e.meshPerAttribute=this.meshPerAttribute,e}},t.InstancedMesh=Ba,t.Int16BufferAttribute=class extends Mi{constructor(t,e,i){super(new Int16Array(t),e,i)}},t.Int32BufferAttribute=class extends Mi{constructor(t,e,i){super(new Int32Array(t),e,i)}},t.Int8BufferAttribute=class extends Mi{constructor(t,e,i){super(new Int8Array(t),e,i)}},t.IntType=1013,t.InterleavedBuffer=Qs,t.InterleavedBufferAttribute=ea,t.Interpolant=jl,t.InterpolateDiscrete=$,t.InterpolateLinear=Q,t.InterpolateSmooth=tt,t.InvertStencilOp=5386,t.KeepStencilOp=ht,t.KeyframeTrack=Zl,t.LOD=ya,t.LatheBufferGeometry=class extends Eo{constructor(t,e,i,n){console.warn("THREE.LatheBufferGeometry has been renamed to THREE.LatheGeometry."),super(t,e,i,n)}},t.LatheGeometry=Eo,t.Layers=We,t.LessDepth=2,t.LessEqualDepth=3,t.LessEqualStencilFunc=515,t.LessStencilFunc=513,t.Light=pc,t.LightProbe=Pc,t.Line=ja,t.Line3=class{constructor(t=new ne,e=new ne){this.start=t,this.end=e}set(t,e){return this.start.copy(t),this.end.copy(e),this}copy(t){return this.start.copy(t.start),this.end.copy(t.end),this}getCenter(t){return t.addVectors(this.start,this.end).multiplyScalar(.5)}delta(t){return t.subVectors(this.end,this.start)}distanceSq(){return this.start.distanceToSquared(this.end)}distance(){return this.start.distanceTo(this.end)}at(t,e){return this.delta(e).multiplyScalar(t).add(this.start)}closestPointToPointParameter(t,e){gh.subVectors(t,this.start),vh.subVectors(this.end,this.start);const i=vh.dot(vh);let n=vh.dot(gh)/i;return e&&(n=yt(n,0,1)),n}closestPointToPoint(t,e,i){const n=this.closestPointToPointParameter(t,e);return this.delta(i).multiplyScalar(n).add(this.start)}applyMatrix4(t){return this.start.applyMatrix4(t),this.end.applyMatrix4(t),this}equals(t){return t.start.equals(this.start)&&t.end.equals(this.end)}clone(){return(new this.constructor).copy(this)}},t.LineBasicMaterial=Fa,t.LineCurve=_o,t.LineCurve3=yo,t.LineDashedMaterial=Ul,t.LineLoop=Za,t.LineSegments=Ya,t.LinearEncoding=at,t.LinearFilter=f,t.LinearInterpolant=Xl,t.LinearMipMapLinearFilter=1008,t.LinearMipMapNearestFilter=1007,t.LinearMipmapLinearFilter=v,t.LinearMipmapNearestFilter=g,t.LinearSRGBColorSpace=ct,t.LinearToneMapping=1,t.Loader=lc,t.LoaderUtils=Ic,t.LoadingManager=ac,t.LoopOnce=2200,t.LoopPingPong=2202,t.LoopRepeat=2201,t.LuminanceAlphaFormat=1025,t.LuminanceFormat=1024,t.MOUSE={LEFT:0,MIDDLE:1,RIGHT:2,ROTATE:0,DOLLY:1,PAN:2},t.Material=vi,t.MaterialLoader=Dc,t.MathUtils=Ct,t.Matrix3=Rt,t.Matrix4=Ie,t.MaxEquation=104,t.Mesh=Ji,t.MeshBasicMaterial=xi,t.MeshDepthMaterial=Ns,t.MeshDistanceMaterial=Os,t.MeshLambertMaterial=Ol,t.MeshMatcapMaterial=zl,t.MeshNormalMaterial=Nl,t.MeshPhongMaterial=Dl,t.MeshPhysicalMaterial=Pl,t.MeshStandardMaterial=Rl,t.MeshToonMaterial=Il,t.MinEquation=103,t.MirroredRepeatWrapping=u,t.MixOperation=1,t.MultiplyBlending=4,t.MultiplyOperation=0,t.NearestFilter=d,t.NearestMipMapLinearFilter=1005,t.NearestMipMapNearestFilter=1004,t.NearestMipmapLinearFilter=m,t.NearestMipmapNearestFilter=p,t.NeverDepth=0,t.NeverStencilFunc=512,t.NoBlending=0,t.NoColorSpace="",t.NoToneMapping=0,t.NormalAnimationBlendMode=rt,t.NormalBlending=1,t.NotEqualDepth=7,t.NotEqualStencilFunc=517,t.NumberKeyframeTrack=$l,t.Object3D=ri,t.ObjectLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=this,s=""===this.path?Ic.extractUrlBase(t):this.path;this.resourcePath=this.resourcePath||s;const a=new uc(this.manager);a.setPath(this.path),a.setRequestHeader(this.requestHeader),a.setWithCredentials(this.withCredentials),a.load(t,(function(i){let s=null;try{s=JSON.parse(i)}catch(e){return void 0!==n&&n(e),void console.error("THREE:ObjectLoader: Can't parse "+t+".",e.message)}const a=s.metadata;void 0!==a&&void 0!==a.type&&"geometry"!==a.type.toLowerCase()?r.parse(s,e):console.error("THREE.ObjectLoader: Can't load "+t)}),i,n)}async loadAsync(t,e){const i=""===this.path?Ic.extractUrlBase(t):this.path;this.resourcePath=this.resourcePath||i;const n=new uc(this.manager);n.setPath(this.path),n.setRequestHeader(this.requestHeader),n.setWithCredentials(this.withCredentials);const r=await n.loadAsync(t,e),s=JSON.parse(r),a=s.metadata;if(void 0===a||void 0===a.type||"geometry"===a.type.toLowerCase())throw new Error("THREE.ObjectLoader: Can't load "+t);return await this.parseAsync(s)}parse(t,e){const i=this.parseAnimations(t.animations),n=this.parseShapes(t.shapes),r=this.parseGeometries(t.geometries,n),s=this.parseImages(t.images,(function(){void 0!==e&&e(l)})),a=this.parseTextures(t.textures,s),o=this.parseMaterials(t.materials,a),l=this.parseObject(t.object,r,o,a,i),c=this.parseSkeletons(t.skeletons,l);if(this.bindSkeletons(l,c),void 0!==e){let t=!1;for(const e in s)if(s[e].data instanceof HTMLImageElement){t=!0;break}!1===t&&e(l)}return l}async parseAsync(t){const e=this.parseAnimations(t.animations),i=this.parseShapes(t.shapes),n=this.parseGeometries(t.geometries,i),r=await this.parseImagesAsync(t.images),s=this.parseTextures(t.textures,r),a=this.parseMaterials(t.materials,s),o=this.parseObject(t.object,n,a,s,e),l=this.parseSkeletons(t.skeletons,o);return this.bindSkeletons(o,l),o}parseShapes(t){const e={};if(void 0!==t)for(let i=0,n=t.length;i<n;i++){const n=(new Fo).fromJSON(t[i]);e[n.uuid]=n}return e}parseSkeletons(t,e){const i={},n={};if(e.traverse((function(t){t.isBone&&(n[t.uuid]=t)})),void 0!==t)for(let e=0,r=t.length;e<r;e++){const r=(new Pa).fromJSON(t[e],n);i[r.uuid]=r}return i}parseGeometries(t,e){const i={};if(void 0!==t){const n=new Oc;for(let r=0,s=t.length;r<s;r++){let s;const a=t[r];switch(a.type){case"BufferGeometry":case"InstancedBufferGeometry":s=n.parse(a);break;default:a.type in El?s=El[a.type].fromJSON(a,e):console.warn(`THREE.ObjectLoader: Unsupported geometry type "${a.type}"`)}s.uuid=a.uuid,void 0!==a.name&&(s.name=a.name),!0===s.isBufferGeometry&&void 0!==a.userData&&(s.userData=a.userData),i[a.uuid]=s}}return i}parseMaterials(t,e){const i={},n={};if(void 0!==t){const r=new Dc;r.setTextures(e);for(let e=0,s=t.length;e<s;e++){const s=t[e];void 0===i[s.uuid]&&(i[s.uuid]=r.parse(s)),n[s.uuid]=i[s.uuid]}}return n}parseAnimations(t){const e={};if(void 0!==t)for(let i=0;i<t.length;i++){const n=t[i],r=nc.parse(n);e[r.uuid]=r}return e}parseImages(t,e){const i=this,n={};let r;function s(t){if("string"==typeof t){const e=t;return function(t){return i.manager.itemStart(t),r.load(t,(function(){i.manager.itemEnd(t)}),void 0,(function(){i.manager.itemError(t),i.manager.itemEnd(t)}))}(/^(\/\/)|([a-z]+:(\/\/)?)/i.test(e)?e:i.resourcePath+e)}return t.data?{data:It(t.type,t.data),width:t.width,height:t.height}:null}if(void 0!==t&&t.length>0){const i=new ac(e);r=new dc(i),r.setCrossOrigin(this.crossOrigin);for(let e=0,i=t.length;e<i;e++){const i=t[e],r=i.url;if(Array.isArray(r)){const t=[];for(let e=0,i=r.length;e<i;e++){const i=s(r[e]);null!==i&&(i instanceof HTMLImageElement?t.push(i):t.push(new Ca(i.data,i.width,i.height)))}n[i.uuid]=new Yt(t)}else{const t=s(i.url);n[i.uuid]=new Yt(t)}}}return n}async parseImagesAsync(t){const e=this,i={};let n;async function r(t){if("string"==typeof t){const i=t,r=/^(\/\/)|([a-z]+:(\/\/)?)/i.test(i)?i:e.resourcePath+i;return await n.loadAsync(r)}return t.data?{data:It(t.type,t.data),width:t.width,height:t.height}:null}if(void 0!==t&&t.length>0){n=new dc(this.manager),n.setCrossOrigin(this.crossOrigin);for(let e=0,n=t.length;e<n;e++){const n=t[e],s=n.url;if(Array.isArray(s)){const t=[];for(let e=0,i=s.length;e<i;e++){const i=s[e],n=await r(i);null!==n&&(n instanceof HTMLImageElement?t.push(n):t.push(new Ca(n.data,n.width,n.height)))}i[n.uuid]=new Yt(t)}else{const t=await r(n.url);i[n.uuid]=new Yt(t)}}}return i}parseTextures(t,e){function i(t,e){return"number"==typeof t?t:(console.warn("THREE.ObjectLoader.parseTexture: Constant should be in numeric form.",t),e[t])}const n={};if(void 0!==t)for(let r=0,s=t.length;r<s;r++){const s=t[r];void 0===s.image&&console.warn('THREE.ObjectLoader: No "image" specified for',s.uuid),void 0===e[s.image]&&console.warn("THREE.ObjectLoader: Undefined image",s.image);const a=e[s.image],o=a.data;let l;Array.isArray(o)?(l=new ln,6===o.length&&(l.needsUpdate=!0)):(l=o&&o.data?new Ca:new Kt,o&&(l.needsUpdate=!0)),l.source=a,l.uuid=s.uuid,void 0!==s.name&&(l.name=s.name),void 0!==s.mapping&&(l.mapping=i(s.mapping,zc)),void 0!==s.offset&&l.offset.fromArray(s.offset),void 0!==s.repeat&&l.repeat.fromArray(s.repeat),void 0!==s.center&&l.center.fromArray(s.center),void 0!==s.rotation&&(l.rotation=s.rotation),void 0!==s.wrap&&(l.wrapS=i(s.wrap[0],Uc),l.wrapT=i(s.wrap[1],Uc)),void 0!==s.format&&(l.format=s.format),void 0!==s.type&&(l.type=s.type),void 0!==s.encoding&&(l.encoding=s.encoding),void 0!==s.minFilter&&(l.minFilter=i(s.minFilter,Bc)),void 0!==s.magFilter&&(l.magFilter=i(s.magFilter,Bc)),void 0!==s.anisotropy&&(l.anisotropy=s.anisotropy),void 0!==s.flipY&&(l.flipY=s.flipY),void 0!==s.premultiplyAlpha&&(l.premultiplyAlpha=s.premultiplyAlpha),void 0!==s.unpackAlignment&&(l.unpackAlignment=s.unpackAlignment),void 0!==s.userData&&(l.userData=s.userData),n[s.uuid]=l}return n}parseObject(t,e,i,n,r){let s,a,o;function l(t){return void 0===e[t]&&console.warn("THREE.ObjectLoader: Undefined geometry",t),e[t]}function c(t){if(void 0!==t){if(Array.isArray(t)){const e=[];for(let n=0,r=t.length;n<r;n++){const r=t[n];void 0===i[r]&&console.warn("THREE.ObjectLoader: Undefined material",r),e.push(i[r])}return e}return void 0===i[t]&&console.warn("THREE.ObjectLoader: Undefined material",t),i[t]}}function h(t){return void 0===n[t]&&console.warn("THREE.ObjectLoader: Undefined texture",t),n[t]}switch(t.type){case"Scene":s=new $s,void 0!==t.background&&(Number.isInteger(t.background)?s.background=new jt(t.background):s.background=h(t.background)),void 0!==t.environment&&(s.environment=h(t.environment)),void 0!==t.fog&&("Fog"===t.fog.type?s.fog=new Ks(t.fog.color,t.fog.near,t.fog.far):"FogExp2"===t.fog.type&&(s.fog=new Js(t.fog.color,t.fog.density))),void 0!==t.backgroundBlurriness&&(s.backgroundBlurriness=t.backgroundBlurriness);break;case"PerspectiveCamera":s=new sn(t.fov,t.aspect,t.near,t.far),void 0!==t.focus&&(s.focus=t.focus),void 0!==t.zoom&&(s.zoom=t.zoom),void 0!==t.filmGauge&&(s.filmGauge=t.filmGauge),void 0!==t.filmOffset&&(s.filmOffset=t.filmOffset),void 0!==t.view&&(s.view=Object.assign({},t.view));break;case"OrthographicCamera":s=new Ln(t.left,t.right,t.top,t.bottom,t.near,t.far),void 0!==t.zoom&&(s.zoom=t.zoom),void 0!==t.view&&(s.view=Object.assign({},t.view));break;case"AmbientLight":s=new Cc(t.color,t.intensity);break;case"DirectionalLight":s=new Ec(t.color,t.intensity);break;case"PointLight":s=new Tc(t.color,t.intensity,t.distance,t.decay);break;case"RectAreaLight":s=new Lc(t.color,t.intensity,t.width,t.height);break;case"SpotLight":s=new yc(t.color,t.intensity,t.distance,t.angle,t.penumbra,t.decay);break;case"HemisphereLight":s=new mc(t.color,t.groundColor,t.intensity);break;case"LightProbe":s=(new Pc).fromJSON(t);break;case"SkinnedMesh":a=l(t.geometry),o=c(t.material),s=new Aa(a,o),void 0!==t.bindMode&&(s.bindMode=t.bindMode),void 0!==t.bindMatrix&&s.bindMatrix.fromArray(t.bindMatrix),void 0!==t.skeleton&&(s.skeleton=t.skeleton);break;case"Mesh":a=l(t.geometry),o=c(t.material),s=new Ji(a,o);break;case"InstancedMesh":a=l(t.geometry),o=c(t.material);const e=t.count,i=t.instanceMatrix,n=t.instanceColor;s=new Ba(a,o,e),s.instanceMatrix=new Da(new Float32Array(i.array),16),void 0!==n&&(s.instanceColor=new Da(new Float32Array(n.array),n.itemSize));break;case"LOD":s=new ya;break;case"Line":s=new ja(l(t.geometry),c(t.material));break;case"LineLoop":s=new Za(l(t.geometry),c(t.material));break;case"LineSegments":s=new Ya(l(t.geometry),c(t.material));break;case"PointCloud":case"Points":s=new eo(l(t.geometry),c(t.material));break;case"Sprite":s=new ga(c(t.material));break;case"Group":s=new Gs;break;case"Bone":s=new Ea;break;default:s=new ri}if(s.uuid=t.uuid,void 0!==t.name&&(s.name=t.name),void 0!==t.matrix?(s.matrix.fromArray(t.matrix),void 0!==t.matrixAutoUpdate&&(s.matrixAutoUpdate=t.matrixAutoUpdate),s.matrixAutoUpdate&&s.matrix.decompose(s.position,s.quaternion,s.scale)):(void 0!==t.position&&s.position.fromArray(t.position),void 0!==t.rotation&&s.rotation.fromArray(t.rotation),void 0!==t.quaternion&&s.quaternion.fromArray(t.quaternion),void 0!==t.scale&&s.scale.fromArray(t.scale)),void 0!==t.castShadow&&(s.castShadow=t.castShadow),void 0!==t.receiveShadow&&(s.receiveShadow=t.receiveShadow),t.shadow&&(void 0!==t.shadow.bias&&(s.shadow.bias=t.shadow.bias),void 0!==t.shadow.normalBias&&(s.shadow.normalBias=t.shadow.normalBias),void 0!==t.shadow.radius&&(s.shadow.radius=t.shadow.radius),void 0!==t.shadow.mapSize&&s.shadow.mapSize.fromArray(t.shadow.mapSize),void 0!==t.shadow.camera&&(s.shadow.camera=this.parseObject(t.shadow.camera))),void 0!==t.visible&&(s.visible=t.visible),void 0!==t.frustumCulled&&(s.frustumCulled=t.frustumCulled),void 0!==t.renderOrder&&(s.renderOrder=t.renderOrder),void 0!==t.userData&&(s.userData=t.userData),void 0!==t.layers&&(s.layers.mask=t.layers),void 0!==t.children){const a=t.children;for(let t=0;t<a.length;t++)s.add(this.parseObject(a[t],e,i,n,r))}if(void 0!==t.animations){const e=t.animations;for(let t=0;t<e.length;t++){const i=e[t];s.animations.push(r[i])}}if("LOD"===t.type){void 0!==t.autoUpdate&&(s.autoUpdate=t.autoUpdate);const e=t.levels;for(let t=0;t<e.length;t++){const i=e[t],n=s.getObjectByProperty("uuid",i.object);void 0!==n&&s.addLevel(n,i.distance)}}return s}bindSkeletons(t,e){0!==Object.keys(e).length&&t.traverse((function(t){if(!0===t.isSkinnedMesh&&void 0!==t.skeleton){const i=e[t.skeleton];void 0===i?console.warn("THREE.ObjectLoader: No skeleton found with UUID:",t.skeleton):t.bind(i,t.bindMatrix)}}))}},t.ObjectSpaceNormalMap=1,t.OctahedronBufferGeometry=class extends vl{constructor(t,e){console.warn("THREE.OctahedronBufferGeometry has been renamed to THREE.OctahedronGeometry."),super(t,e)}},t.OctahedronGeometry=vl,t.OneFactor=201,t.OneMinusDstAlphaFactor=207,t.OneMinusDstColorFactor=209,t.OneMinusSrcAlphaFactor=205,t.OneMinusSrcColorFactor=203,t.OrthographicCamera=Ln,t.PCFShadowMap=1,t.PCFSoftShadowMap=2,t.PMREMGenerator=Bn,t.Path=Ao,t.PerspectiveCamera=sn,t.Plane=pn,t.PlaneBufferGeometry=class extends _n{constructor(t,e,i,n){console.warn("THREE.PlaneBufferGeometry has been renamed to THREE.PlaneGeometry."),super(t,e,i,n)}},t.PlaneGeometry=_n,t.PlaneHelper=class extends ja{constructor(t,e=1,i=16776960){const n=i,r=new Di;r.setAttribute("position",new wi([1,-1,0,-1,1,0,-1,-1,0,1,1,0,-1,1,0,-1,-1,0,1,-1,0,1,1,0],3)),r.computeBoundingSphere(),super(r,new Fa({color:n,toneMapped:!1})),this.type="PlaneHelper",this.plane=t,this.size=e;const s=new Di;s.setAttribute("position",new wi([1,1,0,-1,1,0,-1,-1,0,1,1,0,-1,-1,0,1,-1,0],3)),s.computeBoundingSphere(),this.add(new Ji(s,new xi({color:n,opacity:.2,transparent:!0,depthWrite:!1,toneMapped:!1})))}updateMatrixWorld(t){this.position.set(0,0,0),this.scale.set(.5*this.size,.5*this.size,1),this.lookAt(this.plane.normal),this.translateZ(-this.plane.constant),super.updateMatrixWorld(t)}dispose(){this.geometry.dispose(),this.material.dispose(),this.children[0].geometry.dispose(),this.children[0].material.dispose()}},t.PointLight=Tc,t.PointLightHelper=class extends Ji{constructor(t,e,i){super(new yl(e,4,2),new xi({wireframe:!0,fog:!1,toneMapped:!1})),this.light=t,this.light.updateMatrixWorld(),this.color=i,this.type="PointLightHelper",this.matrix=this.light.matrixWorld,this.matrixAutoUpdate=!1,this.update()}dispose(){this.geometry.dispose(),this.material.dispose()}update(){void 0!==this.color?this.material.color.set(this.color):this.material.color.copy(this.light.color)}},t.Points=eo,t.PointsMaterial=Ja,t.PolarGridHelper=class extends Ya{constructor(t=10,e=16,i=8,n=64,r=4473924,s=8947848){r=new jt(r),s=new jt(s);const a=[],o=[];if(e>1)for(let i=0;i<e;i++){const n=i/e*(2*Math.PI),l=Math.sin(n)*t,c=Math.cos(n)*t;a.push(0,0,0),a.push(l,0,c);const h=1&i?r:s;o.push(h.r,h.g,h.b),o.push(h.r,h.g,h.b)}for(let e=0;e<i;e++){const l=1&e?r:s,c=t-t/i*e;for(let t=0;t<n;t++){let e=t/n*(2*Math.PI),i=Math.sin(e)*c,r=Math.cos(e)*c;a.push(i,0,r),o.push(l.r,l.g,l.b),e=(t+1)/n*(2*Math.PI),i=Math.sin(e)*c,r=Math.cos(e)*c,a.push(i,0,r),o.push(l.r,l.g,l.b)}}const l=new Di;l.setAttribute("position",new wi(a,3)),l.setAttribute("color",new wi(o,3));super(l,new Fa({vertexColors:!0,toneMapped:!1})),this.type="PolarGridHelper"}dispose(){this.geometry.dispose(),this.material.dispose()}},t.PolyhedronBufferGeometry=class extends Do{constructor(t,e,i,n){console.warn("THREE.PolyhedronBufferGeometry has been renamed to THREE.PolyhedronGeometry."),super(t,e,i,n)}},t.PolyhedronGeometry=Do,t.PositionalAudio=class extends Jc{constructor(t){super(t),this.panner=this.context.createPanner(),this.panner.panningModel="HRTF",this.panner.connect(this.gain)}disconnect(){super.disconnect(),this.panner.disconnect(this.gain)}getOutput(){return this.panner}getRefDistance(){return this.panner.refDistance}setRefDistance(t){return this.panner.refDistance=t,this}getRolloffFactor(){return this.panner.rolloffFactor}setRolloffFactor(t){return this.panner.rolloffFactor=t,this}getDistanceModel(){return this.panner.distanceModel}setDistanceModel(t){return this.panner.distanceModel=t,this}getMaxDistance(){return this.panner.maxDistance}setMaxDistance(t){return this.panner.maxDistance=t,this}setDirectionalCone(t,e,i){return this.panner.coneInnerAngle=t,this.panner.coneOuterAngle=e,this.panner.coneOuterGain=i,this}updateMatrixWorld(t){if(super.updateMatrixWorld(t),!0===this.hasPlaybackControl&&!1===this.isPlaying)return;this.matrixWorld.decompose(Kc,$c,Qc),th.set(0,0,1).applyQuaternion($c);const e=this.panner;if(e.positionX){const t=this.context.currentTime+this.listener.timeDelta;e.positionX.linearRampToValueAtTime(Kc.x,t),e.positionY.linearRampToValueAtTime(Kc.y,t),e.positionZ.linearRampToValueAtTime(Kc.z,t),e.orientationX.linearRampToValueAtTime(th.x,t),e.orientationY.linearRampToValueAtTime(th.y,t),e.orientationZ.linearRampToValueAtTime(th.z,t)}else e.setPosition(Kc.x,Kc.y,Kc.z),e.setOrientation(th.x,th.y,th.z)}},t.PropertyBinding=lh,t.PropertyMixer=eh,t.QuadraticBezierCurve=Mo,t.QuadraticBezierCurve3=bo,t.Quaternion=ie,t.QuaternionKeyframeTrack=tc,t.QuaternionLinearInterpolant=Ql,t.REVISION=e,t.RGBADepthPacking=3201,t.RGBAFormat=w,t.RGBAIntegerFormat=1033,t.RGBA_ASTC_10x10_Format=Y,t.RGBA_ASTC_10x5_Format=j,t.RGBA_ASTC_10x6_Format=q,t.RGBA_ASTC_10x8_Format=X,t.RGBA_ASTC_12x10_Format=Z,t.RGBA_ASTC_12x12_Format=J,t.RGBA_ASTC_4x4_Format=U,t.RGBA_ASTC_5x4_Format=B,t.RGBA_ASTC_5x5_Format=F,t.RGBA_ASTC_6x5_Format=k,t.RGBA_ASTC_6x6_Format=G,t.RGBA_ASTC_8x5_Format=V,t.RGBA_ASTC_8x6_Format=H,t.RGBA_ASTC_8x8_Format=W,t.RGBA_BPTC_Format=K,t.RGBA_ETC2_EAC_Format=z,t.RGBA_PVRTC_2BPPV1_Format=N,t.RGBA_PVRTC_4BPPV1_Format=I,t.RGBA_S3TC_DXT1_Format=C,t.RGBA_S3TC_DXT3_Format=L,t.RGBA_S3TC_DXT5_Format=R,t.RGBFormat=1022,t.RGB_ETC1_Format=36196,t.RGB_ETC2_Format=O,t.RGB_PVRTC_2BPPV1_Format=D,t.RGB_PVRTC_4BPPV1_Format=P,t.RGB_S3TC_DXT1_Format=E,t.RGFormat=1030,t.RGIntegerFormat=1031,t.RawShaderMaterial=Ll,t.Ray=De,t.Raycaster=class{constructor(t,e,i=0,n=1/0){this.ray=new De(t,e),this.near=i,this.far=n,this.camera=null,this.layers=new We,this.params={Mesh:{},Line:{threshold:1},LOD:{},Points:{threshold:1},Sprite:{}}}set(t,e){this.ray.set(t,e)}setFromCamera(t,e){e.isPerspectiveCamera?(this.ray.origin.setFromMatrixPosition(e.matrixWorld),this.ray.direction.set(t.x,t.y,.5).unproject(e).sub(this.ray.origin).normalize(),this.camera=e):e.isOrthographicCamera?(this.ray.origin.set(t.x,t.y,(e.near+e.far)/(e.near-e.far)).unproject(e),this.ray.direction.set(0,0,-1).transformDirection(e.matrixWorld),this.camera=e):console.error("THREE.Raycaster: Unsupported camera type: "+e.type)}intersectObject(t,e=!0,i=[]){return mh(t,this,i,e),i.sort(ph),i}intersectObjects(t,e=!0,i=[]){for(let n=0,r=t.length;n<r;n++)mh(t[n],this,i,e);return i.sort(ph),i}},t.RectAreaLight=Lc,t.RedFormat=1028,t.RedIntegerFormat=1029,t.ReinhardToneMapping=2,t.RepeatWrapping=c,t.ReplaceStencilOp=7681,t.ReverseSubtractEquation=102,t.RingBufferGeometry=class extends xl{constructor(t,e,i,n,r,s){console.warn("THREE.RingBufferGeometry has been renamed to THREE.RingGeometry."),super(t,e,i,n,r,s)}},t.RingGeometry=xl,t.SRGBColorSpace=lt,t.Scene=$s,t.ShaderChunk=yn,t.ShaderLib=bn,t.ShaderMaterial=nn,t.ShadowMaterial=Cl,t.Shape=Fo,t.ShapeBufferGeometry=class extends _l{constructor(t,e){console.warn("THREE.ShapeBufferGeometry has been renamed to THREE.ShapeGeometry."),super(t,e)}},t.ShapeGeometry=_l,t.ShapePath=class{constructor(){this.type="ShapePath",this.color=new jt,this.subPaths=[],this.currentPath=null}moveTo(t,e){return this.currentPath=new Ao,this.subPaths.push(this.currentPath),this.currentPath.moveTo(t,e),this}lineTo(t,e){return this.currentPath.lineTo(t,e),this}quadraticCurveTo(t,e,i,n){return this.currentPath.quadraticCurveTo(t,e,i,n),this}bezierCurveTo(t,e,i,n,r,s){return this.currentPath.bezierCurveTo(t,e,i,n,r,s),this}splineThru(t){return this.currentPath.splineThru(t),this}toShapes(t){function e(t,e){const i=e.length;let n=!1;for(let r=i-1,s=0;s<i;r=s++){let i=e[r],a=e[s],o=a.x-i.x,l=a.y-i.y;if(Math.abs(l)>Number.EPSILON){if(l<0&&(i=e[s],o=-o,a=e[r],l=-l),t.y<i.y||t.y>a.y)continue;if(t.y===i.y){if(t.x===i.x)return!0}else{const e=l*(t.x-i.x)-o*(t.y-i.y);if(0===e)return!0;if(e<0)continue;n=!n}}else{if(t.y!==i.y)continue;if(a.x<=t.x&&t.x<=i.x||i.x<=t.x&&t.x<=a.x)return!0}}return n}const i=ul.isClockWise,n=this.subPaths;if(0===n.length)return[];let r,s,a;const o=[];if(1===n.length)return s=n[0],a=new Fo,a.curves=s.curves,o.push(a),o;let l=!i(n[0].getPoints());l=t?!l:l;const c=[],h=[];let u,d,p=[],m=0;h[m]=void 0,p[m]=[];for(let e=0,a=n.length;e<a;e++)s=n[e],u=s.getPoints(),r=i(u),r=t?!r:r,r?(!l&&h[m]&&m++,h[m]={s:new Fo,p:u},h[m].s.curves=s.curves,l&&m++,p[m]=[]):p[m].push({h:s,p:u[0]});if(!h[0])return function(t){const e=[];for(let i=0,n=t.length;i<n;i++){const n=t[i],r=new Fo;r.curves=n.curves,e.push(r)}return e}(n);if(h.length>1){let t=!1,i=0;for(let t=0,e=h.length;t<e;t++)c[t]=[];for(let n=0,r=h.length;n<r;n++){const r=p[n];for(let s=0;s<r.length;s++){const a=r[s];let o=!0;for(let r=0;r<h.length;r++)e(a.p,h[r].p)&&(n!==r&&i++,o?(o=!1,c[r].push(a)):t=!0);o&&c[n].push(a)}}i>0&&!1===t&&(p=c)}for(let t=0,e=h.length;t<e;t++){a=h[t].s,o.push(a),d=p[t];for(let t=0,e=d.length;t<e;t++)a.holes.push(d[t].h)}return o}},t.ShapeUtils=ul,t.ShortType=1011,t.Skeleton=Pa,t.SkeletonHelper=class extends Ya{constructor(t){const e=bh(t),i=new Di,n=[],r=[],s=new jt(0,0,1),a=new jt(0,1,0);for(let t=0;t<e.length;t++){const i=e[t];i.parent&&i.parent.isBone&&(n.push(0,0,0),n.push(0,0,0),r.push(s.r,s.g,s.b),r.push(a.r,a.g,a.b))}i.setAttribute("position",new wi(n,3)),i.setAttribute("color",new wi(r,3));super(i,new Fa({vertexColors:!0,depthTest:!1,depthWrite:!1,toneMapped:!1,transparent:!0})),this.isSkeletonHelper=!0,this.type="SkeletonHelper",this.root=t,this.bones=e,this.matrix=t.matrixWorld,this.matrixAutoUpdate=!1}updateMatrixWorld(t){const e=this.bones,i=this.geometry,n=i.getAttribute("position");Mh.copy(this.root.matrixWorld).invert();for(let t=0,i=0;t<e.length;t++){const r=e[t];r.parent&&r.parent.isBone&&(yh.multiplyMatrices(Mh,r.matrixWorld),_h.setFromMatrixPosition(yh),n.setXYZ(i,_h.x,_h.y,_h.z),yh.multiplyMatrices(Mh,r.parent.matrixWorld),_h.setFromMatrixPosition(yh),n.setXYZ(i+1,_h.x,_h.y,_h.z),i+=2)}i.getAttribute("position").needsUpdate=!0,super.updateMatrixWorld(t)}dispose(){this.geometry.dispose(),this.material.dispose()}},t.SkinnedMesh=Aa,t.Source=Yt,t.Sphere=we,t.SphereBufferGeometry=class extends yl{constructor(t,e,i,n,r,s,a){console.warn("THREE.SphereBufferGeometry has been renamed to THREE.SphereGeometry."),super(t,e,i,n,r,s,a)}},t.SphereGeometry=yl,t.Spherical=class{constructor(t=1,e=0,i=0){return this.radius=t,this.phi=e,this.theta=i,this}set(t,e,i){return this.radius=t,this.phi=e,this.theta=i,this}copy(t){return this.radius=t.radius,this.phi=t.phi,this.theta=t.theta,this}makeSafe(){const t=1e-6;return this.phi=Math.max(t,Math.min(Math.PI-t,this.phi)),this}setFromVector3(t){return this.setFromCartesianCoords(t.x,t.y,t.z)}setFromCartesianCoords(t,e,i){return this.radius=Math.sqrt(t*t+e*e+i*i),0===this.radius?(this.theta=0,this.phi=0):(this.theta=Math.atan2(t,i),this.phi=Math.acos(yt(e/this.radius,-1,1))),this}clone(){return(new this.constructor).copy(this)}},t.SphericalHarmonics3=Rc,t.SplineCurve=So,t.SpotLight=yc,t.SpotLightHelper=class extends ri{constructor(t,e){super(),this.light=t,this.light.updateMatrixWorld(),this.matrix=t.matrixWorld,this.matrixAutoUpdate=!1,this.color=e;const i=new Di,n=[0,0,0,0,0,1,0,0,0,1,0,1,0,0,0,-1,0,1,0,0,0,0,1,1,0,0,0,0,-1,1];for(let t=0,e=1,i=32;t<i;t++,e++){const r=t/i*Math.PI*2,s=e/i*Math.PI*2;n.push(Math.cos(r),Math.sin(r),1,Math.cos(s),Math.sin(s),1)}i.setAttribute("position",new wi(n,3));const r=new Fa({fog:!1,toneMapped:!1});this.cone=new Ya(i,r),this.add(this.cone),this.update()}dispose(){this.cone.geometry.dispose(),this.cone.material.dispose()}update(){this.light.updateMatrixWorld();const t=this.light.distance?this.light.distance:1e3,e=t*Math.tan(this.light.angle);this.cone.scale.set(e,e,t),xh.setFromMatrixPosition(this.light.target.matrixWorld),this.cone.lookAt(xh),void 0!==this.color?this.cone.material.color.set(this.color):this.cone.material.color.copy(this.light.color)}},t.Sprite=ga,t.SpriteMaterial=ia,t.SrcAlphaFactor=204,t.SrcAlphaSaturateFactor=210,t.SrcColorFactor=202,t.StaticCopyUsage=35046,t.StaticDrawUsage=ut,t.StaticReadUsage=35045,t.StereoCamera=class{constructor(){this.type="StereoCamera",this.aspect=1,this.eyeSep=.064,this.cameraL=new sn,this.cameraL.layers.enable(1),this.cameraL.matrixAutoUpdate=!1,this.cameraR=new sn,this.cameraR.layers.enable(2),this.cameraR.matrixAutoUpdate=!1,this._cache={focus:null,fov:null,aspect:null,near:null,far:null,zoom:null,eyeSep:null}}update(t){const e=this._cache;if(e.focus!==t.focus||e.fov!==t.fov||e.aspect!==t.aspect*this.aspect||e.near!==t.near||e.far!==t.far||e.zoom!==t.zoom||e.eyeSep!==this.eyeSep){e.focus=t.focus,e.fov=t.fov,e.aspect=t.aspect*this.aspect,e.near=t.near,e.far=t.far,e.zoom=t.zoom,e.eyeSep=this.eyeSep,Hc.copy(t.projectionMatrix);const i=e.eyeSep/2,n=i*e.near/e.focus,r=e.near*Math.tan(vt*e.fov*.5)/e.zoom;let s,a;Vc.elements[12]=-i,Gc.elements[12]=i,s=-r*e.aspect+n,a=r*e.aspect+n,Hc.elements[0]=2*e.near/(a-s),Hc.elements[8]=(a+s)/(a-s),this.cameraL.projectionMatrix.copy(Hc),s=-r*e.aspect-n,a=r*e.aspect-n,Hc.elements[0]=2*e.near/(a-s),Hc.elements[8]=(a+s)/(a-s),this.cameraR.projectionMatrix.copy(Hc)}this.cameraL.matrixWorld.copy(t.matrixWorld).multiply(Vc),this.cameraR.matrixWorld.copy(t.matrixWorld).multiply(Gc)}},t.StreamCopyUsage=35042,t.StreamDrawUsage=35040,t.StreamReadUsage=35041,t.StringKeyframeTrack=ec,t.SubtractEquation=101,t.SubtractiveBlending=3,t.TOUCH={ROTATE:0,PAN:1,DOLLY_PAN:2,DOLLY_ROTATE:3},t.TangentSpaceNormalMap=0,t.TetrahedronBufferGeometry=class extends Ml{constructor(t,e){console.warn("THREE.TetrahedronBufferGeometry has been renamed to THREE.TetrahedronGeometry."),super(t,e)}},t.TetrahedronGeometry=Ml,t.Texture=Kt,t.TextureLoader=class extends lc{constructor(t){super(t)}load(t,e,i,n){const r=new Kt,s=new dc(this.manager);return s.setCrossOrigin(this.crossOrigin),s.setPath(this.path),s.load(t,(function(t){r.image=t,r.needsUpdate=!0,void 0!==e&&e(r)}),i,n),r}},t.TorusBufferGeometry=class extends bl{constructor(t,e,i,n,r){console.warn("THREE.TorusBufferGeometry has been renamed to THREE.TorusGeometry."),super(t,e,i,n,r)}},t.TorusGeometry=bl,t.TorusKnotBufferGeometry=class extends Sl{constructor(t,e,i,n,r,s){console.warn("THREE.TorusKnotBufferGeometry has been renamed to THREE.TorusKnotGeometry."),super(t,e,i,n,r,s)}},t.TorusKnotGeometry=Sl,t.Triangle=fi,t.TriangleFanDrawMode=2,t.TriangleStripDrawMode=1,t.TrianglesDrawMode=0,t.TubeBufferGeometry=class extends wl{constructor(t,e,i,n,r){console.warn("THREE.TubeBufferGeometry has been renamed to THREE.TubeGeometry."),super(t,e,i,n,r)}},t.TubeGeometry=wl,t.UVMapping=n,t.Uint16BufferAttribute=bi,t.Uint32BufferAttribute=Si,t.Uint8BufferAttribute=class extends Mi{constructor(t,e,i){super(new Uint8Array(t),e,i)}},t.Uint8ClampedBufferAttribute=class extends Mi{constructor(t,e,i){super(new Uint8ClampedArray(t),e,i)}},t.Uniform=uh,t.UniformsGroup=class extends mt{constructor(){super(),this.isUniformsGroup=!0,Object.defineProperty(this,"id",{value:dh++}),this.name="",this.usage=ut,this.uniforms=[]}add(t){return this.uniforms.push(t),this}remove(t){const e=this.uniforms.indexOf(t);return-1!==e&&this.uniforms.splice(e,1),this}setName(t){return this.name=t,this}setUsage(t){return this.usage=t,this}dispose(){return this.dispatchEvent({type:"dispose"}),this}copy(t){this.name=t.name,this.usage=t.usage;const e=t.uniforms;this.uniforms.length=0;for(let t=0,i=e.length;t<i;t++)this.uniforms.push(e[t].clone());return this}clone(){return(new this.constructor).copy(this)}},t.UniformsLib=Mn,t.UniformsUtils=en,t.UnsignedByteType=x,t.UnsignedInt248Type=S,t.UnsignedIntType=y,t.UnsignedShort4444Type=1017,t.UnsignedShort5551Type=1018,t.UnsignedShortType=_,t.VSMShadowMap=3,t.Vector2=Lt,t.Vector3=ne,t.Vector4=$t,t.VectorKeyframeTrack=ic,t.VideoTexture=class extends Kt{constructor(t,e,i,n,r,s,a,o,l){super(t,e,i,n,r,s,a,o,l),this.isVideoTexture=!0,this.minFilter=void 0!==s?s:f,this.magFilter=void 0!==r?r:f,this.generateMipmaps=!1;const c=this;"requestVideoFrameCallback"in t&&t.requestVideoFrameCallback((function e(){c.needsUpdate=!0,t.requestVideoFrameCallback(e)}))}clone(){return new this.constructor(this.image).copy(this)}update(){const t=this.image;!1==="requestVideoFrameCallback"in t&&t.readyState>=t.HAVE_CURRENT_DATA&&(this.needsUpdate=!0)}},t.WebGL1Renderer=Zs,t.WebGL3DRenderTarget=class extends Qt{constructor(t=1,e=1,i=1){super(t,e),this.isWebGL3DRenderTarget=!0,this.depth=i,this.texture=new ee(null,t,e,i),this.texture.isRenderTargetTexture=!0}},t.WebGLArrayRenderTarget=class extends Qt{constructor(t=1,e=1,i=1){super(t,e),this.isWebGLArrayRenderTarget=!0,this.depth=i,this.texture=new te(null,t,e,i),this.texture.isRenderTargetTexture=!0}},t.WebGLCubeRenderTarget=cn,t.WebGLMultipleRenderTargets=class extends Qt{constructor(t=1,e=1,i=1,n={}){super(t,e,n),this.isWebGLMultipleRenderTargets=!0;const r=this.texture;this.texture=[];for(let t=0;t<i;t++)this.texture[t]=r.clone(),this.texture[t].isRenderTargetTexture=!0}setSize(t,e,i=1){if(this.width!==t||this.height!==e||this.depth!==i){this.width=t,this.height=e,this.depth=i;for(let n=0,r=this.texture.length;n<r;n++)this.texture[n].image.width=t,this.texture[n].image.height=e,this.texture[n].image.depth=i;this.dispose()}return this.viewport.set(0,0,t,e),this.scissor.set(0,0,t,e),this}copy(t){this.dispose(),this.width=t.width,this.height=t.height,this.depth=t.depth,this.viewport.set(0,0,this.width,this.height),this.scissor.set(0,0,this.width,this.height),this.depthBuffer=t.depthBuffer,this.stencilBuffer=t.stencilBuffer,null!==t.depthTexture&&(this.depthTexture=t.depthTexture.clone()),this.texture.length=0;for(let e=0,i=t.texture.length;e<i;e++)this.texture[e]=t.texture[e].clone(),this.texture[e].isRenderTargetTexture=!0;return this}},t.WebGLMultisampleRenderTarget=class extends Qt{constructor(t,e,i){console.error('THREE.WebGLMultisampleRenderTarget has been removed. Use a normal render target and set the "samples" property to greater 0 to enable multisampling.'),super(t,e,i),this.samples=4}},t.WebGLRenderTarget=Qt,t.WebGLRenderer=Ys,t.WebGLUtils=Fs,t.WireframeGeometry=Tl,t.WrapAroundEnding=nt,t.ZeroCurvatureEnding=et,t.ZeroFactor=200,t.ZeroSlopeEnding=it,t.ZeroStencilOp=0,t._SRGBAFormat=pt,t.sRGBEncoding=ot,Object.defineProperty(t,"__esModule",{value:!0})})); diff --git a/public/static/history.html b/public/static/history.html deleted file mode 100644 index b7b4ece..0000000 --- a/public/static/history.html +++ /dev/null @@ -1,91 +0,0 @@ -<!DOCTYPE html> -<html lang="en"> - -<head> - <meta charset="UTF-8"> - <meta http-equiv="X-UA-Compatible" content="IE=edge"> - <meta name="viewport" content="width=device-width, initial-scale=1.0"> - <title>Document</title> - - <link type="text/css" rel="stylesheet" href="./common/css/common.css"> - <style> - * { - padding: 0; - margin: 0; - } - - #chartDiv { - position: relative; - } - - body { - background-color: #fff; - } - </style> -</head> - -<body> - <!-- 第一个 --> - <div id="realdata" class="realdata"> - <div id="echarts-btn" class="col echarts-btn menu-top"> - - <span class="btn" title="PRPS"> - <span id="prps" class="prps ready"></span> - </span> - - <span class="btn" title="PRPD"> - <span id="prpd" class="prpd"></span> - </span> - - <span class="btn" title="开启"> - <span id="start" class="start"></span> - </span> - - <span class="btn refreshtitle " title="累计"> - <span class="refresh-btn ready"></span> - </span> - - </div> - </div> - - <div id="chartDiv" class="col echarts-chart charts"> - <div id="box3D1" class="loading-box"></div> - <div id="box2D1" class="loading-box" style="display: none;"></div> - <div class="noisy-slide-box"> - <span class="slide-name noise" name="noise">底噪</span> - <div class="noisy-slide slide GIS" data-price="0"></div> - <div class="noisy-slide slide GIL" data-price="0" style="display:none"></div> - </div> - <div id="addPointNumDiv"> - <div class="cdf-slide-box"> - <span class="slide-name phaseOffset" name="phaseOffset">相位偏移</span> - <span class="cdf-slide slide" data-price="0"></span> - </div> - </div> - <div class="zoom"> - <span class="btn" title="放大"> - <span id="scale" class="scale big ready"></span> - </span> - <span class="btn" title="缩小"> - <span id="scale" class="scale small"></span> - </span> - </div> - </div> - - <div id="box2D6" class="loading-box-mini"></div> - - <script type="text/javascript" src="./common/js/jquery.min.js"></script> - <!-- <script type="text/javascript" src="./common/js/jquery-ui.min.js"></script> - <script type="text/javascript" src="./common/js/simpleAlert.js"></script> --> - - <script type="text/javascript" src="./common/js/three/three.min.js"></script> - <script type="text/javascript" src="./common/js/three/OrbitControls.js"></script> - <script type="text/javascript" src="./common/js/three/Lut.js"></script> - - <!--<script type="text/javascript" src="./common/js/echarts/echarts.js"></script> --> - <!-- 使不支持es模块的浏览器支持es --> - <!-- <script type="text/javascript" async src="https://unpkg.com/es-module-shims@1.3.6/dist/es-module-shims.js"></script> --> - <script type="module" src="./common/js/history.js"></script> -</body> - -</html> \ No newline at end of file diff --git a/public/static/index.html b/public/static/index.html deleted file mode 100644 index 7239135..0000000 --- a/public/static/index.html +++ /dev/null @@ -1,137 +0,0 @@ -<!DOCTYPE html> -<html lang="en"> - -<head> - <meta charset="UTF-8"> - <meta http-equiv="X-UA-Compatible" content="IE=edge"> - <meta name="viewport" content="width=device-width, initial-scale=1.0"> - <title>Document</title> - - <link type="text/css" rel="stylesheet" href="./common/css/common.css"> - <style> - * { - padding: 0; - margin: 0; - } - - #chartDiv { - position: relative; - } - - body { - background-color: #fff; - } - </style> -</head> - -<body> - <!-- 第一个 --> - <div id="realdata" class="realdata"> - <div id="echarts-btn" class="col echarts-btn menu-top"> - <select id="channelFiltering" class="selectStyle"> - <option value="0" class="selectFrequency">选择频段</option> - <option value="1" class="lowband">低频</option> - <option value="2" class="fullband">全频</option> - <option value="3" class="narrowband">窄频</option> - <option value="4" class="highband">高频</option> - </select> - - <span class="btn" title="PRPS"> - <span id="prps" class="prps ready"></span> - </span> - - <span class="btn" title="PRPD"> - <span id="prpd" class="prpd"></span> - </span> - - <span class="btn" title="开启"> - <span id="start" class="start"></span> - </span> - <span class="btn LinkANC" title="自适应降噪"> - <span id="adaptive-noise" class="adaptive-noise"></span> - </span> - <span id="associated-noise-box" class="btn LinkNC" title="关联噪声传感器降噪"> - <span id="associated-noise" class="associated-noise"></span> - </span> - <span class="btn eventList" title="事件列表"> - <span class="event-btn"></span> - </span> - - <span id="record-btn-box" class="btn record-btn-box" title="录波"> - <span class="record-btn"></span> - </span> - - <span id="total" class="btn" style="width: 70px;display:none" > - <input id="total-time" type="number" placeholder="累计时间(s)" /> - </span> - - <span class="btn refreshtitle " title="累计"> - <span class="refresh-btn ready"></span> - </span> - - <span id="menu-right" class="btn" title="展开"> - <span id="show" class="show"></span> - </span> - </div> - </div> - - <div id="chartDiv" class="col echarts-chart charts"> - <div id="box3D1" class="loading-box"></div> - <div id="box3D2" class="loading-box" style="display: none;"></div> - <div id="box2D2" class="loading-box" style="display: none;"></div> - <div id="box2D5" class="loading-box" style="display: none;width:100%;height: 100%;"></div> - <div id="box2D1" class="loading-box" style="display: none;"></div> - <div class="noisy-slide-box"> - <span class="slide-name noise" name="noise">底噪</span> - <div class="noisy-slide slide GIS" data-price="0"></div> - <div class="noisy-slide slide GIL" data-price="0" style="display:none"></div> - </div> - <div id="echarts-text" class="echarts-text text-bottom"> - <span class="maxValue" name="maxValue"> - <span class="name">脉冲最大值:</span> - <span id="maxValue">0</span> - <span class="unit">dBm</span> - </span> - <span class="avgValue" name="averageValue"> - <span class="name">脉冲平均值:</span> - <span id="averageValue">0</span> - <span class="unit">dBm</span> - </span> - <span class="pulseNumber" name="impulseQuantity"> - <span class="name">脉冲数量:</span> - <span id="impulseQuantity">0</span> - </span> - </div> - <div id="addPointNumDiv"> - <div class="cdf-slide-box"> - <span class="slide-name phaseOffset" name="phaseOffset">相位偏移</span> - <span class="cdf-slide slide" data-price="0"></span> - </div> - </div> - <div class="zoom"> - <span class="btn" title="放大"> - <span id="scale" class="scale big ready"></span> - </span> - <span class="btn" title="缩小"> - <span id="scale" class="scale small"></span> - </span> - </div> - </div> - - <div id="box2D6" class="loading-box-mini"></div> - - <script type="text/javascript" src="./common/js/jquery.min.js"></script> - <script type="text/javascript" src="./common/js/jquery-ui.min.js"></script> - <!--<script type="text/javascript" src="./common/js/simpleAlert.js"></script> --> - - <script type="text/javascript" src="./common/js/three/three.min.js"></script> - <script type="text/javascript" src="./common/js/three/OrbitControls.js"></script> - <script type="text/javascript" src="./common/js/three/Lut.js"></script> - - <script type="text/javascript" src="./common/js/echarts/echarts.js"></script> - <!-- 使不支持es模块的浏览器支持es --> - <!-- <script type="text/javascript" async src="https://unpkg.com/es-module-shims@1.3.6/dist/es-module-shims.js"></script> --> - <script type="module" src="./common/js/index.js"></script> -</body> - -</html> \ No newline at end of file diff --git a/public/static/threePublic.zip b/public/static/threePublic.zip new file mode 100644 index 0000000000000000000000000000000000000000..bf6ff8e21563b6917351af620a1b157eeae2533a GIT binary patch literal 1021983 zcma%?Q;;ZIx24OrZM$~awzbQ)ZQHhO+qP}nw)^~j(YJ9UIv+CgVa)ZiBEHNq=3Mep zz#u39|GHkSrqurD%fH{i|9KkPT3g%N(98d~BIy4mGIVtOPay<AMdXs&d-v+P`ag@O zY6AeE{_nznTbS;jnrkg@+oKWp-(A1nLFNI01(l=tE%a&^ePYv2QK%wo0C17d)P7P^ z*YLvx=mbx4cfvk#gkBQfHJf-=&<s{qoI+ZQ(wj<s_OaA@r&HT`_fw`pLRaH+9PF`b zZ*u$7<Mdtb+ea8%+Z-RLd}Fn#M5`;AGx|z~m_(}?A1>dIa_5PRA_;X4W9O6u#)$`* zNBTso<{AV`yHnPT6OpJ$#s{*q7u*jTT~+ghWgfRmh7LCR2WK=cg`ATjZ^qmZ@Et^? zU)+Zqqqn$dt^LJ`aQh^5{p6!`-V6+b+R>k}@Z-gGE7o}k^#gWw`b)%z&$hJU_drGs z68k8o9vWA57;e8PC6Y%G7Cg5~rJcR=(jF->C<RPZ3{#3J1E)F?SbP%uW(;CJjQEyN zoOE-%+<N?r^Q^`C4ETIU))Sym5S}BGeeckM?s9aZ6#=YxR9&x^M|$5cdaL{!nptYC zx|GT;X$)>qSxDK}_YZcSp3`lEUOEO%@ji4xiOLEuO4gp8wELV>_j<X#B8<g#j;d%> zs5W*?Bd$g$Qx$NI$vb73ybreUgsrdv=j2vJuyX?(q>^!NY_T8!Ewyt&>g+!#vXk7} zsu(j7eNB35MI*^+M4%M_i1sLlMOwxC>liYN+#REM(_>$<>0JSoR<wEf$x<C6ZopzV z;^F|LlsPI1kgp|=x2J^3f4z+xk5ERt4116|r9Q*)ucZ}gO`$hx*(J~p7w;k8i7EB7 z-jC^bd)+R{(Q~^z?=xrdd+%*idp$B8&6TTC7m=!!3Pcc6&51ztSwsyOL$dO7v|53f z#>>7qSxTPiEMUl10>1Jw1T2_@{m3;%e-ikdtnHP=uAe{13*9*D%~M@Km$%$2B*q1? zoK_;vd=$wRl_}JKJMGyMAL!mpqIw>2`-7F=Iju6^1tA^D>qXnlgwlan%Oe=5RrLiz z0;fsPU7D=oN$aC!VVSb$8|lp>Ox^uESBw@c)f0mMN>zffbJd8&(+`PPPieu$cLl7C zo5dH^?>%|IT6vTbJao{FvOrr<*R`DIfC7~r6|>HJ%raY%$6jx+epE$smI88jMsqK? z34jW4pp;<TOb_%k<A^iue(gYq4w`Q<GUtIY1SdRPe*<m+bc*^Svu%QCnfJW%Dckyd z97f9N9A7w~&b+_5@K`rTD5Ay#22&MtPhN!%NwS;z&=S~JDQv90Z)|=3NKC_(Di<<l zSERUAbS2XBdR`tR;t!8|_n?*3!8TS>T4nB3Ml-6!PT^^ya5fpA897-tSSQP5(<v|k zVDuH!4Wuca))SUrBkWLpw)N*fVF(qh1e23d;;=Y%o){rfv1W&{qE&9G$!XI3MoTM_ zDx^s|vcgJ9Yp#Zz`7>}zwDgH60zwaef1Ehlw37fgm^Rv_7nyn(HuJaf4!}5t&m!pn zkm;&>QS)Uv*5rdZ1G}!8)TEOk$uFFSQ9MX1F~OD!G^pfS6_&n33+$R?FDr$K*7k>$ z)mBQ9ilgx=*7yhGGLB^U+9PdUZ{tFed$Hq0teI#hJzZkV4mfY{_sN8zZUf{-sM8F# zqs9D{ue1%J@q}_$Q=9TH#x$tk9SLHQpVPQ2Hw52)7V~$C$v`X=DJWeuC*Nw;2_mwY zXt14V@GZ-i1K6DooqRCBEXm?4=ObLxkRJ^ZvXRH4Vgcwk0{*K9w$HHS6IGz?oXE#+ zwa&tp4Sav$$;ygAo65*6VsKbRJ4tHhzQA~7UM|y95e=@O#$F-cQvgvgsr$V<e99fP zr)#yC@+`Zok4EU{<T4f)r-E9F)ga9?*fW#qKhwnvAO*7(6K|C#04%!ac0`?JZ81~G zxQb6YsiLE-|F=J+&5wo=42kiv$Wf-r8VPzuK_H|ir;SU7b5y^*&%WBV2q!jZKDkm& z4tKMT*jw#5ATYlD`gpT$Xg$Mbc?54RK!yNDrWjbFXk(GtYK&?T7iPNGtQpjcX^JE$ zf;->wn9tSWb=dG+m-0y}yys2Fv8m|qLorjnv4a!r!ljytWf)XickQK1Nu$aIe@3KA zj^9+6FdP+D-GLADvkCb%HQMeZcQ#f`oh~L4Ih{=u`?bhm8@)*>9^Ha#tS9(F$>w!Z zkKimP1_V!E+eC&Mfuf;u(gPacVR_7pm%rkIc^g%DaU9>htdm}-9bvr<0UNQ4=q5%q znt?2<%nnGkRtH@Qjj+m=pgRKSq8=1_EfKKeb(u!mr|3E}O%}(oNy8TKLAWCapMeFi zduz;#oiyK+40qOAIL5T7TTp!Yik0K4x8WbezuEZ+^U}53w<HeT;#5}%aufqboA@aE z6+7(9c%|kIREZ0MqQIlrq=Di}(4>+n8AD;`eX<)g**glD(6mu_x52>NX~EL~9yC_s zLJG%5kP;#|KCVSqPR;!2wtQW0d;6)iJ?`&~zN9H1&!fHW*Ymx;YT$JrH5<;2BEjY{ zDz<v2is)^?V9la84TWryTAmk0tcWI@l5MpfM*u>M#=zix%$JO0u4SQ{4z@&rD?74; z6{HnwXWod=N*u3Th$I-&wm85iI$n<e2<IvEKChFJu|D?iBV6cw-=;Iw#%}nR(i)X! z^|=#i0VJH5Qg4plamMC2b!HqH$h(i4%p`a*9yvU5x=7Ut7duf6rUdBp-7F0_ZKa}X zw6Bi7y_-CL&<j9mH#$~CJnC*cToaDJWWy^X+U#5>oKf~0oK)Anei0Nx=_a#*YVFim zSdlV}ioyAvrci{1QOVTwaj&p17Y|(}Y=m@O3Z|HuhgmB5S*N%o4#jO{#@G^8VOi_Q zYBQ^Ygte$WWzzm0^;wqitdbla1j^z4zSa((8>AKKMjE{}?$q&_#lzekKa&K~oxajx zCPDm;kAdQTWD=mmoK{YX1VUCh%v(JR9}BKWPqyy5KlPiY-gbjEAlXQKt)uM9KpJIw zAkmQ|{Fdon$;>>j(9_yU-ZOIb7aF5>FY3N=)FAwkx`>;}=VqC`mN9|M*25{O6mcWy zaW6T=%}C-Fi$sQbUx_sbk@+^v@K?nddbsCKXk$U#U#botBVhjtJ3*lE<r0WJfh4gG zdNbj*W#_rU%~eZLnJNYEarFxF^5y@5@}hp9Ep-oPOOZ)2djZ_PLO;Qniw5grQh1lV zaL!O@wAN6#)=N)|*(J6M(^`|aHp$f|jO3qrQ-?e4^0RVaY6}_{+fbc6gdZ;%EQCoE zN77*HCIOtN2j@SRkPitm(;RRX<wGXcYtG8xw;(uAQqNd=$A%{VT<o`7OU#3@QU3sK z@o6;tP(YbmlfJ6KA(>rh>>axkB|vb)0!b3i2gVo;z$20XLcwL$;jwk|y2=LDzCXkF z`FxxH?ee;urvE)Lw1;US%b*eJNZhWYDY~TQP;y%^iy*J+Icqr!%MpO&6@?ZAn%@mx zmJh#ajc%`k(%~YnsMOnFdYFz<wA3l;{1YH`h+DBYw@n1ONc_Z;wKJO3MXZcOdXv;9 z?)H{9w1~S$3N`k7<M*X3x0t@|>dDO_U}Wk+Uyc*#s?CwmIHha;-O!wf<s>DWgO6UJ z?qbz#IOsTR8lWq4hxwAHf-x`dwHP&Ryfw5zmNjeNRc|QfVqhQ;&1=KE-!7;T(8{O< z-`;JLdsQ|2vUWX5cl<|MX>Bl+hl2jC$=i+_0axt>^aMNv@0%Y!v6407WalZ(hvr^` zxRiiU{+%HQpUYqe*qLHT?vEHN^}{hB?%7?KmnVUuacF3?J*D5Y7B*(gx#CCc*+g(C z_Nz+#Y@*xOfbREs0>~?0kq*29gw*S7LfP1)*WY%Fss^aY3-<^Sb=29bMS=0$Z{nav z4DOS}D=8NPhL)l*_r**wikP!gYCw_>UUk1W5-@(jIDpP^r~Br3#Lk)id<Jbu%pFZV zC#5k6JjQrLKl8D(Q9z(f%4q{IRixqC5N(u&+538a+p(Xgl3hgze+&)3FW|pee`UhI zfYn&kwtm>=8n}S)TtVJKQB&C)xr<%0b2J@O|AJ`@nxquO#ANk|1qzOWtB_l@sdSLV z;MwEOmozg$lE{U4v>GQ^-=5ALewyr}gf`abcqc0Axc}bsY5#ox>im51(c$}iz47P- z%wD!>;n0rBCc%A<D&DV2mqjuw35!#X1off?@Ny^~s%>hV!PJ4^=BlU@2(oaiquL8- zLJY*B9rJrABp=s_VsWXLIGwV>b`Gfky7^R65t(%#e%|VMedM#VADD}dynGmBN_fjf z+ugrcqVv7eW9X>7d*v8}=EW9z4r#Lj(aKv-XkzWIW-cO)_ysDVqE}0p9o!kBOv1oz zhpd>>xWZbgh-%d3F8CU$flfq)nMxCB*{Tys196a#>qV8|e@HFd7{8;iKw9rJ6K+v0 zTFMbkH1vJre<5@Wml1P@{RHYL0^$s|By3nktyH4Y71S+R?$F4xZ9so6s_UL)U+Phe zOKOc9AZD~3I<K4Ex_KahdnCvYSvj{WaA6UNZFP83<x>i`QFpKLwW~BTV1cz>d>>NO z;X`h`)xu}_bMvQuIO*}<tR|OxTok)I1`?bv#*zkHO=cqCRa&wuvVY`!nUDZa<=x`v zf6&k3;;mfsti%w?9^vu6s-=@`L>7#>Nt1^bnR6J6C1teO&ccK)Yuv~{4{EebX{Z%q zs25mB5YLHgDg)V&O5RN)ZY)HpgFAwvt=zllr)ZZ8Vn;g<R#3qm)Inu4XDv`QUm@I? z*KN}EB{#*=bYr_y8nqeQl@E<xf#VeNwVY9D^ME`#f<F75pK=PD<vC1!tZJ&9wuDQO zuZq-I{u<RZF@m1E8EJOj$H6);6%9X>ark|V;D{hh<mAx~n5ZwYa#4P1sD5btVZuo( zk>#!90BUhEdqiR4&Bw2N*x5G-)|_M3smdC7!FjlG{Y2o&K`Bk#CGU6yC2Dki<9_mT z*t{=tM+!euAfdoqf=RmLndY$vH#Uqu8kOHl-;Ul(RS&gH=ted!kD_)Zj=DNol|*Gg zGe#+S>vNQA`+#jUqD;7RjH1^1OV0<92n9-dfJn7ptez0hfVDh7=6bu|sE$ZyU+!$6 z<lf5qvOaU(3jDbWe9vM$_Hx4w2Y9I&JhZ(7>9X=bMzDI0g=@18qXFaKeQ&kf6Vvc& z9Vfp~6CTzbrdIO~bs*Wvt*<})D)CC*fLbUTFxnw1+Si;?N_z-(*1#*Lg+2qNMdy<X z0F6`w%cur5R<q)S>Mr7S4eWa~>}I11AY3T%O$lZ^BcJWF6eG>hwLl_U;~(318GcTn zuYRH+NEVw-BPZ!~D6XsfkL$J3s&=x&3nqnAMOn9{q2TnVPh2~_6dlvcqNaeCeLz;h zi>B=YawltAM<!_?IiUEWj6+avcA|yzii|HZ<iNbo?;!Hg%qX{?Rq9po>uUy*dZNWU zcf~Sxk?GvF^rjD{re`ygKve0;uV5_uz+ztbG5jVoxUIuw`<vP0#4gS3YDqK_ple!| zz}zG)@mw@z*HY~W?*vLu6K1Mu$`e9g2SCr|2XeVtaZRNvAT%qUUGb^YrJ+qhqIP@! z)OBK{NnYWMUHmszeRM^u(SoEB)myIe3SZp?4Z^gY3WINl*vV9aWC}1<r1X$tEb$g- zo!IILn2OG~N2iD2CcIS~f(ER*9~crk5S5jMu3vr>U&jr!yTb&XfC>Us3G1`M;i!OP znuf?d8_^2IU!@bOnW-<PR={h8GXmBs`w)=V`}PO8EvAo%MAfLLzt!(OtwNXliT4F3 zvNg@z6w^06l6`7`SSQnDsZap*;-d=bDl2yMjw(weZP=V#sT^q^U{72WT3B1XI_elX zlWUJgs&ry?ejwkb^smwC!y7eV2X-aL(?RQclY%_dRMR{fm4HGfEuO@?>sCGxHUxbG zG5w#?oXy*E9mU)hC;jp6j!<qNr{h|z@Hi||YBrua4lH(EpN5;E*2T=TBPc)I&Db0p z8Rg~5isDvHbUgYNE2m7WI*))6+%;S=upsJ#uWB4?y48EJjezWroA!2tv%{@D2UpA% z%W7Q37U%j2MEO@99Fw4<ODY)%=0NvAj4NgvaEPxtB2q`$(-)1<6HTb*OOlR@GF#_< zDe8$dYWAb3A+<ASK3d1Ij`YdgpO1SneBO_@Zn4`gm-`!fy5E<5^xkhVop!{L$)PI8 z`fkaZ(q0RNcO|g?KO-c8L<%QOj4GDbBafE0xjs*Crny=_+n4CJT|V!3-tvD3BDv0b zgd?+b80Q6eOK(N1dqpx^h6JQy3O4Cvje@&Q>pLpAl;!7t6j%K68{_i2`2;>_DMZuV zeIxjosiiNdk|4cil^<nltyKq<w>=wCLR8x{b5mPJQ^df!9@OMIW&aXS(y_`3v!L?z zF^IXb%5c7nIb4!P6okJNNOnIBI~2<bT)aJeAXh7g)(RH9l<beJQ(*#F%I!s}z6+-a zl1_Rwql4N=N3J6?|4E?9?w9()@>Zwk4QQrcHP7@w{0`4~DN<b##j_`fd-rr?w%5cR zwd>Ql^jo>K|Ilho2{)1;$)c!>VTV2|#bbmS^w4e0vXRZ;Bd}<;!~#vMM`Be2hf~dS zd^&g>rSCDT+LA042)r_j+kf!QQjd@uzmfzvzs-(rf1A(65Ce|Kfc!bEO`%{AeGrlV zN+3l%H>;Nl&+vlOE9{EFs&(^Yohw#mI~ha2dX^Lm`+=~vP}dO!1DkV<wkxtZESrda za;*aWxJ+`AagyeP)#Z}s;iLkOVi})bPQWsf;3*v2y3a1tkqUx}$DDmI(-xG>3j@2C z-XfTZEC$|vH}1Z}`s4vYEcBBYy7`#>9)A>~)j%|^urox3SV&OWpHN5OR;w@W2#%<( z9cPYw^zeuQUv3|a>9oJ97h<7(yYagH1^kcD5(1#Xn@<hVgOxEI6ae7ipFHya$Rf>* z^^J@j{*^@*t7+P7v7&s#$MMuLMl}f(i$6}Nfj!K#I055?XAlk$f`%K$GC{=k*5u)D zOGeU>Q*=b?+><5{kBdS9w~)ra+TXN?s&XP|#lJj!pbAExl@$e`)9R}&fIIB&CKpit zdEPw@dWgu1?Cj)f`+4|K@xWl^Tad(y;GVQu-T==345V36ckF?;D$K}{VA02Hxy0tu z^4VxT7^*xdsk!XaR)nI1Or5w!%tgM&I?k+qb}8E0nnu4VfgBHZ8@kSUAQE~b5kWEX z>oa3u8zm=u5d>9W@;B}mr(OPdy~V+@yo=uH#s|<aQedC_f+Fm?1mPM7{7DXPdzG)6 z{=kwK=ln=VNx~PGD$Z3-2*gh6mupy8lV2(n#+;{MYpbBx&7_9#98=c3i62SYM5)2& zi(~a8Cfza|G|*4XUdL2<HeA)SE-cU<yhAP;r63=V34d6mXeF?dm{GJ3lOhnz?TS{v zX5pIK2vCJ0+c+?3*Ai7bfyUnv>$W9NGr1}c<mVa?ha73!Ghzl!5}Zehi6ybB%&Vy; zJ2V->b@(A*LN-saMpv;;gc=A1AON{U;`m~wPR=zQ5H_cbkvVv)+Nxj~-fUelHG>01 z!9Qfea1U7+VW45wqv`ILUeK)ftj73p@R_wksM~)dtomD~6k`=XGfg?YKwMTz$U|(O zd8MhP;P)m*nYX7HaX8Di?$J4f`9<4WnegSb?HU8&XceT4B+5k_2l%Tk%=@=Q;yb&} zQgE%ytC0q+)$7dF{zl-Ye>c2^9CypUsm@ryr!b|o1z1rCkV3w;I~Pueh0DZgHZq~D z8TkPBI;MdJjt7!%mB9AG;uDyMwIDg|@=7F7iYgZ>X(w8Fy?!i1mR$O9+C+8RH&!Hw zUgz;L(M*zqY?x_}zRk}gh&KYaTtq60x*EuK2Sn7+a+2}@96Tj7z={z(+alcGMT_XO zXA%}Ig2X#dr^)=6D?b4z5@>b7k0;$dX{t?bJ6xmzLa`3KZoi=h;7@m0mz8{0`g%U7 z>^-hOn}!VS)e$mru+HIbD^m>YWM-Nmm9apvpbEw?-LUa44R6ql8t*9S<g9}%TWiZl zjvb6pJ|a-@k)eI;w9cVqzCM*h=q>CBdl^94b-AMv&f-iPA&UhmCu3!(>KEQ}DM9r; zrtvZK?h#GH1Ic8NKJ#7&GPr`adTyXT8^v1)_ce5AlmIE00bG2%V*zQ}x8!tGA=*); z9q7`yXUA5E9KQV|;hug!Cp0<^gk?x+=2&+eIC7WAm;P}i858l`Yc{^2<3_ci6Owe; z&m2UL91YKW5jVvAG^}q2A}bvM3qm^dH3N3z+x#3u7P+!6$m!k4bx3l&zj9)Z@|i=i zS<T=;BLg-FhTry}gpa%bOD1CB>@*Q84{BNp9_Pp8p4Cefs}wb0<Rq{a!DFbOHzQ=A z%c{=Y<f*Q9Up>2oCXz2dp6B};-<Rh02QDtoHzM$7*a#Tt>sX)?sA9ZglbU-7j>P8Z z6&G>Pn$<IW(Z}s-?XJ(~V=mv%$II(&2nLW{qn=ybdSL^N6uj?TSs1IC8{^;aD6~kY z#-8ZNfP}BW<hyw}DBz~C!N1np;I{MqVgNi4<U%knZXzF=qt9im4t)(|_0{^h1?8@) z%s2rj$s^uDeh1<j1}2N_3l#=KG|u>3Xf{Z8iT>odEm&m&sdm%XIMTqm@^~&mL>L_( ztjKVT`KL|hUrdXbFdhSk)tl*(RG9Pro`|$WeJSaYXQZPJX9lhTjIo83EgtmhCUyKf zM*1B4<Cqc8Qy3Z8>RwH)3T#y)O6xk*VY<C-Q(%<FGTz5n{7x{qWtec1?%dNx$OLE} zK3O3PiRb#Z<9qBMA`gDK_v#1H%8R;4Az2!gB-8FdPQ?cQmF29C%W8~kiBv8Bbe;&P z0RWKy_r8*YvA)&6NWQEg^-os`kvA6bH*^LHTHG?8hy&ifEONDT{RjaA5Qkt?WVX(< zgb<E{fWxCIPZNPy9EjN7VPd$4gqPMK!f@2?XG${WSL9=RS>~ddA4#~8T86E&va+(r z!--{(g|byZ)0%F_8$3UYH4`Y;)J!3+k$9(ZC=;d_d#BCoHCsNa=G<X<hatxo(m^B9 zU8?er&|}3~%_OfPEr(^G`f|&%#@vQ7Y_j8Bm42fJkx4g{S$bk4%y7KfBqpXLk&wAg zGA*TMtfUOo)&jD`ani{(Rb=5=y;+TYcM<8XONt~Xgf+GzI4mW`DJE5A0>uTWx*hK< z_XD(tsDt<>f(9a8t`*XG3}y~K{iTKG2_UkViYZql=5N{e5=VKXn`6jYjd_eq$s(oO zK)Ukc23G1c*s4{n0l%H3$*3{|Ix}^aYAGfy_Cf)Ky~CGpch>&48GgC1KJ!kZ(&}<$ zYx*mik{^&EsUlz=UiwbOpA!1TV$w-e3jO|OBPRlW1>4m|Q98fJtQ4vIoA{gxF4UY5 zmy)4Av1@k$IzK#%tVQcVztaj#(hebq&;{(}7#FliP-RXajoH#KV%K3>V00&S6UP!u zMbB;l|IEKU7j;?H708N0BhLr+oY82JGa$;=ZBO)g9Sun-UqRA=LlnwT1{p4>Y*^Im zINbWIZY7`9<>G@=cP-U1bBVw#%HbX~A>}rD146DY1oZ@}rvqvzFg0c(k6B5QqeeXg z&YKu1pKl7#<-?#-0PN{IlDTw`Up^2Fc@eZMhI$h%NS2xh$Yh?{?JwiAiHbM^>yb9W z-$Pe_+uWq3KnbpQv~TH#)KyE*nntjrfkdCaS;i?vr@P3CgxjX(YmO0zm3pS|I_fO4 zRo*#}pzm{QQnO?vI>bxM)jkxbq2G6+c0h332lO*Eip>#Xl3(wbucEI%_Pit?hV!#4 zlJUfDxJzh0BKYQ@jJ3t$%2w*)d6RnY0<p@+7NN8*5`K|7>8Yx={)zz%N-UxFp+NDw zM32N?Q&?&bj-}72l;*6>jUdYJvmT<EjA;xTpLy3T_PyW${t(^*63fwdP>UqmS*`q8 z2c?A$8Q1%({CG&*##Qv}dCm_!ZSNkC3C+F)f@C;fA||_26&P}!127LIW?%&=Ohqp5 zTv{maLwlb=APx;^p!iNh6$67FFvqPcE{VQVAgi&|NUMMJ*JA=z*lrqhGKt+l0*%}x z4{k0hi8^ly#-D0uW!MG-Z|?aaZ+&U&mTgozkTg5dPKoYS4P_T$3v9dHKl%wV;;H1; z4%?h$J0wm<eg34YkZd%gE%hBx+qR-(m`bh#h4&)HnanO5kA`BCaUrD=G;yp^C;wD0 zjo<9tY!H~z08^`6K>xWPsicYQlIx8HE($d2Jn8;Z-X-+Vv(6JesS-A4@r8TQE`AOU z>5`NkMFFYuMJUJ)BB)P$XFC7`w)C$(C%gdqw)O^rXJNmlE*8%V&pAVL4hqHYo0M3N z5QSWd6)4R(RAzG_&Y{Zht70xUh)1ZQH6_){F0U8&b4Xx!PjaNx&#l&0bdHn_Lwes< z9v{x9`dRd7$&5o!R6My6pRw{76pfO`Uxs9P@9{MmERyOVOsri#pgrZZTlP~=Dc&&L z_1s;-Q(g`J{geq?=Q@KlV{-B?y-_S{{DHK9H6Y<l&5N~Exgiy+qe+A(<;@gRmUJqX zH=8`EjpHffK@wQ_%#~Ql^g~GeRm6tOcF`Qlh(Iu`b|06Ol>H7oG4FX!pRMR+80|mn zw>(PxyGU||$RPMGX?6=D^m5WRt_-d?XjzE?IrpriK{-rV;>fNT!TXPHxxX??yZevA z&B)7m6)*D5Vii0NjX_O;i~G}*zumOwbV5KSd4iif+$Ym(NBL5Hq5$MZ7U)VI7ou3& zccF_tKZm-tzMmf#a=Sm5kMwHDyu5MFaFim>UGfb<p|JFN?clKL6-ip_4MULDXI1jU zqT^q$_`VO@=)EVFAvkEL;147G*lHOvFWnEMWx5XrVuX7tgqHXfIQo}}WKKC14QN;y zYU9cyvAP|5!3K+byDQOojpb#O=#YBVLANFzqyg9}RB`alr$U0ebg`e@rV6`(ewx27 zxQaSu`zC@8cW=j&K(pe2F2<~ck60<#83j7{->LNG>IoaS8&;aJC9vc+Rm*%Qc^A3L z+uhFLyHq9E!Iu-aF0hPQ%}z1UD$9p2ai<72yslbnkErSRo(*UPKvkW?B29_hz8?pH zOI6T;oNjLCv}<CCkF!~o4>etQF=B2$N3>PcrVvG;B{X>~^jy_ydbX|Xry7IN2d)!0 z6V<NY9hWcLr`>gJDtGCUvQ<^g=m&~=3d@!)VQ({c5-x%EPyg)9hM|Me1$pOvj{cQ^ zk_YpOv1nws>P5_)t<p|5R{Se>)#p6%BBa%#l$CRlfMH!on5gWm5GW>(R-9f$V}?oB zAP_J=w+%a(sT1DR7`5cZ_8r3x{J^{spY-Cl-9i0*Zr<~5Y5niY)hDY;J&u10MhJMK z2|(}qsg!MCQv0+4Wi)uj*l6~(d^7J{VJ)LTswC@Yfx)Usm#Y-U)rZk)*!R8cn6zJW z2V{R0mBZuNUU(1xqUgxUIW<W4lBak1Xg_i(Mc~kNbcT5&zTyd4YUjsEWo2xy&*SXs zs_)yy$D?k<6|IqrN^90dUm1!17Hr2V<KBbtPBx6Ru^?G7@O6c93g2B<5;r9LRwbs- z&@m`!Xjj_<PTyK7&Kkg0Ktz|tT(<<CxWD)=bKEe_`!=Fl+}%X3e9$=86y|~i@PS=Y z(cz$5k~AvVipmUg0}(rV!lEgvHYWC&X2qbs2fH_zaQ;4&ERWF?A7?FXMMxX-Fr70( zGviWx40kqmc6i9tUGaEWiwJq_?wG_UIGiwivDvV9v1d3MrpQ{OC_55SAh~L`e5V15 zwgz+jTm0q@;${k4jlm<&Cg|f*5sOVsaajiff|f8E3}R3zR0>v4$9bPjKJ=Bt*e0(c z4PGf7t8&!yn`46xK1+6|({&%pxQI&gw;k~6wCJt{-lUr5#>GXjOZ2=pfooPehhsz! zsLo=5#NK?YguS4fViYfl=7Sl2dW`o|Q%T#J7%VjJ6@$uRaCWT#>q-R7`$&L#rDR2k z3k~XGd$*{e-Kp8`^;tK2OF~3j=KW3UG0h_!2m(=8Iy!p$J{bLYFq>+xIB}FL>lG!( za;VXvp^$yXIr#w^Ws@F=lwiL2NVNf_aS;e}ljT*@DK^sFvx7Y#PX<r#%@~t)-srnm zl-BxKr*Ev6Y=*mxDXxIqTk|ry3R8wO8D7a%_(CyG0#=VBV5JbW;TtS3Ar3t{Z!e&p zTg)T}Q$~+|J?f^_B#4g2tl@W}ljDc%&&4EP@B0hA?a$Bo@@22K_xJg;X>a0uV_qZH z>M38o%;c1W7K|A};DZKGL?_P9UXw<{5)Aqs;ielyB_!|6kS&<R+Zqb@k|oAPxFZ1A zsV=LK-`I+TxWb%_5cX$*;4I$WB>f6CRbz*Kj0buNlR2}*;(WfSlMIPRQ)MEiHV9s! zHK4Z&NR!=nrmpwz=j!hFA^K{rxBK%nxgx0apnusFLAi9f=xA#?(mHG{!E=}H!@+PQ zF<m!)ctAPs-dlDLdTbvF<A`)yz00|%u*kuwR=%KXgx@sy9HAI8_M<{GS%+@)128%M zi_-hI-zo~CvA;yL3S<C<_=7#Y8+J2w0myn@h$^Y+=>xqu?R;avo(7WH+)Y-i+L;J7 zSqYCE^&M*;4)=1qlJwDonW_!mih=yZrCA!LOlRi3YMC0ET%?w^hQ+aDHuNlci*6z^ zUihQMWLSklbYpg@dklkegFK``LPAVFy6K$x6M%RRsms!(B0x35th?adUn94ku_TRz zgTd<!`_F82o~(chX;{gPzt6L2kI64<zSSz8U?2?vIp`osoxWJUcY~(Lr|ZYP{~6i* z-@|DCWzNjp+VsE7nf)u<>5E&SI{imM3<m&!{=fXdzl27%t~OS-`bKniHm29ApL~hm z*tzQliI0LbqEe3HvK+#H$gtx%tjx|CLW@)=L>$sm8cm2o)mexfBBGA#xN}apDPtw& zC`}>r#fM&RY~Ox9zU+2yKOJ>*U%k6)UY)yZK0io0+c_l%RrU!G#54Rx+kw~KPZPGl zwy3)+*#W|WJGnOeD}=`|^%lW_;4a;#ZEF~c3{qETGy{M=rF^$YuWXdwec05wR{v6C zc9jz+eGIuV3*l@ILZ97s>*cZ%43*B84U6fSQt>Ux2!h?4pMX(fY!_J?QzVhlPg&Y2 zZ|rm94>F#w-#8LvE-K9vjeXKySWj$lFP?HCInzI!PK@U9=XhDtiO+9JSABosHz5Z8 zs-yx`>ILNGnr;wajQcQJ3-R5lF0Fx>r?ho-ZS~R{FXe7XA)Js=QBqEu+Ldyc0iDh# zFV--M^+SxU9%K;4<g7G2W>eFvq?pqDc@cVgt0mt$lAn&xN<{uAuCe-?3~j0P>r=f9 z$*x>QThss_O6NGskZ}bqlJf*Pcgvmu9B^`uKSSq0g9Sl3z|F*^LQU9sxpz#8cir7l zHB9>n=Y_)~!|N;JXyztc>C)PZ-`r}tRNy?@CK>Wn&;Bu;mynkEjQE<cCOMKX9He3J zc}n>iHEY`|!0=}2uZz<y#9L-GeW?DUM1>nU#lBZv{fc@XT1M?zwUUAOBenVJaBwab zUuqs<Go^1+9F5o%b@=Sj3ispu@I!tfU*DEd$N)kji{;iZth3;hSeWK5;y+_ZA`l0` z5kUgvpt>Q6R0IYdc;G6K)IZna=Yo0nsOc1UO&QJ##_z702vFhf2Q=RO9s7g-{adnY z6|Hyi+Li&8Y0tjOeB%Tc*pFcQ`vn^*nj7~T8@?_q_q|Da5L;}AZ94_KTAR5T(u>^z z&r{YrB_+<K$ow~GTy=)Il~F)i!7VyEx@kRsVl&FRC?=Pi(s21H|7U0CQB@?FyxBw! z<kV4?Q^)SyTdc2U%D3&OxV5!47o^jP{tKwlh)gH*20hD$4AhAjn3h^`wbj%tU^kz` zB{Y^G@&>nnFxTA}{B0YkGtV!y<eK7yC{}?!T;rv^tLB9plvf@H?n+(eoLN{ViKCY@ zpc(sDw6%qq5%SkzqHdm~#+HFGU`)#u!*s~C-J{-(0dyO#pl2-o*cV1co~+RhT!opz z8#$3EF`kVl3~ok{(!~PGF<Xq*ss^w5zOL#HpvOm1wX#2$<0c$VR>auZXQmLPpfgI; zNjsNMr`~qqrb_&u=xK$zV3)bT+=1OHL{Jqh17ZK8%V>Vb0-iTksaLTA_2rVRUmE=d z<`^Y}oG8EuX%k(GOA2fwlOdasJ*y2jIXL0qMnbxvZeIJcA~eH=V!A;Aa3elz?KGtl zM6}6*Y)ERR-(jbduFkeJ<2~4H)=e;e)Y!GeKGR1oh-x`x5+=(qe@{y~6r(vc;7V<G z-`e630YuwK1ipeM*TgNvNJ#7GMZE+Q@{0ncB}2NP?>kAqWv1U-LX(~?`Uo7WyK)Lb zdJJF^kF7RoTnzc-vGb8CHzzxQZig5{3u(L0a^=0C{(;?K<k@CE(Ju#Q0aEdLZ^I9z z1+&^-#Yrfp44|Kg3~M(F<lDhSIcbAkD~LwNs&0PCBi4?5HeZ(-aOoK~oRm7~zFLbe z-6vYlZ%k{Mx@P6(PcYJLl(hrncEcg8$B!Zz9CVxr8fbMf*5?w>;%;ipxlYt(!*<dH z3y03gQ(!DqU$w1D$~VSXH^)!KQ=@k7`sKN({|k+i{@?g!_avPK?O(9M@e{pMRXK$o zeTi)JNQS`>s?FwiC*A33UDCzpvS|**A~(M9_$9_zmv>^#fG~R8tl#>q_Dkvlk@s7* zvyj=PUP*a!s{LhLWyv*Bm8W}L+{n*8_ccScDwffXU4^07DDvezZ4$s2L8{kkLPfHc zh;~C@e~i(+5U=N^9KTD2AbLr5_;lJ0KJUBiXVbmA;%l+ckW|^w%9@Ti#|Uk}exD6u zoB`ww-sHHo+bkTd=liVk?;aFMW-q#=L>Ad<tN#5dtte#pN3}C*hCWnQm+Hr?t2D8N z`Me}}8P!T?XlQs91I3uz=pwnE;?GavA#9nPIgJQ3ZY`?YOLzSFUJMh3%HA2)3~mrx z5@Ym90Nz^XoThS~l5|e3j~2QzK5hnzveNuc4Mrw2*p7}wi^8>fw^bd<VRbsAnwDHc znk|M245g;T0is*bE+tte4FaA54ul**a5a%1CTOL!;NzjSCt?vJoO{m|xHhLLbuCz` z1SUl$vs{PGyv+v-X&;G2@?pu=&u8j#4Jo2F9`))LpltXf86|2LiNZv$JEn7QcIV5$ zTW6OyZuwqkqg{c?N`-gQb>faqy(&M~+n{#BV_o<XyX00|DTrkZQcx1;;KO)E_x;Dt zJw-6iaSmU_Cz~SQ_z89+o2jzXZU;A#1N5#xJj(7dVa34PP?2_FUh~U~OUsf_%e!Uu zH|SA~wQTD=XMK(=iEU$RBGEHz2N`Sw2^slW@f9g2HXhpA+Tic6;Al|LD?t#Vc&iVA zOgS*eW;EV3u9mw*7^(ROlqB2L7c%$FPH)eTsY+>^O6_RM&_^cGO56o&<Kv9q<ML7H zGcW}+*k3Io@&L8Gk|$Oc4DI6U&(2cYg3xcA?efsp@TV0&J_*OjL=1Bs;!im+WJQ?K z>eSNCJ9rgpm|T0y-3kPA)?;v;g@2<S{xG-B-IOm&F?F2XXnU5L34IJpHOy3Nj;($F zHB~cv%#^e*!Qr&n)8vtdSQkwkJFT5?>>;Nmfai+<@~-y_%ns_gq<e@zFeQF&@Ese$ zyLqdxJ#eLy5?i<ncJ_4+Dii6xzs<`^77Oq+oK5;M=T!>vtNZ3qvZW~z-h5d-r{A)K zjYZ?XlmJV2nvHT)I7ogben=-PK5DVysiaJ2f&4aG_s~D1L1!uXKGCF*zkm(8L4<kv zs7a2Dl`*W!Yl_5vpTNwvE0YtK8~m}bd)J62@Ys6w=_!3QhcximHc_qiSrkh?zR@+> zrH+b;4&O$;1uggsqhmoZM^4mFardf3?wH@4n+scMRFc0mxnsDJtv*2Xp6v3=dhYUf zHZyb>&&BIa=U(eQki4JMjmqAomolmEy`~E95XUmd@_^2DiQw~|gYEBC#ua<Il{_h3 zO<*d2dVU8;FKJ3j%FRc!-d#-Uj$Q{hD|#F)KOYA`6(J|=5XjRCkvmbz*baoG;T$17 zMre094Fy5|>TF4c1y|drM2?t0e9Cd%@aKZIDDY=)NMOH<j5$NoObB|e*UNOC9c-W~ z^<zcYmaHB<!EFNN*0Bprj}R{+pT%+8i$ntpRun}FMe`(NTW6wVL_zFx_*-AEqw%r# zdr&3&S2DYcp|iE%3ptmSyq*e%k2AURkK14{yowiZFSBI-esMb7PISOmD`1+mSi%6P zS{nAFPoc=62!Q5ssR0toO<`SreyS)ySH0$w=O&&ejknX26187%Ox<d6_JLoWPSdts zOrft_tC~tDaKM0~BJzrVyN4*P(+&<Xm?}SoRMkx+en7SabWUQkW;*uogQ~Q?Tg!;S zM1sy>_{_lNVyPbi($<@&20<E0FOG{k(4NkS4bdsr9_s{om9Xe$HNbJA5EH6m$fgH2 ze#I@|N_>_l^3{6u1ukC+C6i$hsNRC_zOhff*2sq1k>|Jl)mi*G_?S;v*1)E;Q)*dO zd<%B(NwvpUgc=yC6?EZIVJBI-f>YUyPV^ITqkzAE*HAS6$Z%Q|fE^o5_?WbD{wjBA zW_caM?^JU$ODqbVD6v)XJLzBoFw=AsDbZA=*9DooH|Po0g#hh%T(*GPv&$F*0e{^= zPyW6ozj$Q!ml7yH1oI8FF3Vn%UWHA$*A6g48}~CDCa?j@O-{ig0Awy1b?{u(@Lp|A zw33AuK;U{``ovRri`w$06=e>b-rP%dnJ!2Ia|-5;30N3klvFj_AwW;nAk0n*KuAEN zTu4v^-z)h-LY;T;SFgVke%Rh#L}de)$S?kX^nL$t6Y;NeF8_X_<zL=F+T6uhSl>zi z|8NsbAE8iQAOHZ{|8$F~{+kMYBYiukf68fXY|R~wb!|;d{$(({mA$30jWPImh4SKr z(1!i@k3;%UJi1d-bn>f;S8K9S@|^})5o@lK(adU}Vw&U1yo{54S%>`WHB-Wd#@3`e z#!KkpY0{z?O^Ko~qJv;Y@f`_}Q$e-JdXA|h<GaVVAMbyBW<R@TAGe;9lJTNqx3MD$ z#JdFK((|$)co-4yyg!Dx(DU5VSO4ClR%O4=>wqb2szC$8&~?NxGB#BgpOYzt5g?|^ zI-=?t(<rE^r^dnQ2l=FeLsky}`@maU>$&OaUu4;kws={?0Lx|~RVHDHh{lH4vp-ks zloT{CSy@}xC9q!cBT_t3>j9t-F81A8I5|S~B5j6M`gx2+8|ncMuu;@c7ga<V^ACDp z5}-~vR14>z93=`0l8qCO;}D=BI%tuMLQ2pPDjkKR7%;ABU~o@cHd0F`*<8|?Q8I$h zdZMu3JiZ6<hQ~dx?1M-ow24PW4!0x3Pd2x?Z{%K{yjY9nVa!}Od$^Hbf)Qr6)J=CG z4Yt(oJzNDjJ8BvSfDV2)fs+RjePQO!zj>8SHV2-#AshPgW-zt7VQdZASv9UgVVmxv z6M3Rw(F;9Rtxh#WZ>rEZ)^SgAsKL*f391kaxD+p18+vd3z{bp%F|={yxM*Sm7Z2r$ zojGlw1j!btoJKmk|521g*f3zBET&KtGKa{PB}1KTZAY=Rc44&f$+X<umlW=F5IU?I zH|d%=jFLGu*48`RfI7Wtm+w4n7;nDLh*ayOMh_j|`!#=QftHmQ-^SE7yMp063_-pP zE|l?Q--6<cJZnH|Y`J}zmB$sj{Q?$Yx23996GN+sfD<5I<u9N9(M2=0)jbCR+K&Ba zuq0_V7qurr`47v_`@($s$d!<12>ltdLY{@tzyPsfj!F-%1b9VHF`KU6hbIIp7XaH0 zD#%~v_;~+SN|jJ3NyuUg&E!^R=aHd2b8uZq{s-m9Y!@>?^N){Q$8T}9U8%r$Gsg=t z(=w()mifSXGv81l`5VZyIZcIQIl9t<@9gz=6u1Z7X`sY_lm*9*<Coec&sD5gX){mR z3AtT%Ft0qHGH<W0a+iZv@YUmVpbLY$TNOu*hl#=fwt?)gK{2yo2OiDJ(;c06mN6_O z<wyupbp#lRm{lh(y6o$NQ+yY$g%a8Eh0p{t(KV3=`;St%$`6tCRD~mLU$L0WAIn#h z+m8TgHVZE=YfAGTA5OyJ{Ru7S%~vM)7|?+j8d_fKMzN>TwI_U6$Uhx&TVoeL<|tY} zM_We6<5L5S0`0sAd}~{4T+_+FH4|P7ylPu%X75<iWFY4Stpzk}i?d?jj~=!7cpAd+ zu4J!dh0?1RTE0WYZw}^q`IZK1-=o8ApA$cA;nEghFeFpdwKdn!S*oXTgiP}fNmpC& zLI?!7DA^w+C7h3;y}WMdWivi5mNgo@8TZ=Q3rjl2@IlW*hy)N|inJ9m2L3*BdL_h| zpn^^ykZ6U7`<RQ6mY`z4wTmJgx1rE+%3`)EvXUYaXQ7sJDT=x%t1e&A|3STfn|1*Y z{dV*CcZN~^e=^MG|B0|`SZJlQW>Cpm`IT5|OL7s4Y^O|!)fY+VCRLA74RPh3KNEa@ zQhjZ-Qo;sX|AsJobq4j(CE>V#Ax!K52jZZ?&#1rEs~uxKbBky8*4wxD)O+XD`^JNM z0-k*2HdYLQM4^^AdM;!@P7B2=m*aVb)J*57$)qNt-jR1gikJ%9FyDViWO-3EqoJ8~ zH$M`QL~@Tgoue^d+QwK*kZKC?LmQ8TI|BZlC#gC9#4otUwF7l+AvzPoZUaTl*<Zlu z&de-SY4xPIa$`bfrA3rle-JL&&!86v%x)eB^s9!9jrx_kZo19kEVR@f#!9hG6tpEd zX<Z(WIVu$lRM5OeB}*3o3Nl|UBUKK8gP#a#(k>nb{q#7Z_`xe)iwJNNIIz>*+NC|7 zW_t&k2=ru+Jr-xwn>TkZmE5T7xh;(VLW`uIG(l@@%vf_9=Xln+5!{6+j@j~=lP4?D zK9He=MjE=buP&Rojk{BS`^qI<uiuXExPMp>@dssI&y98Yc*D5ypHDbn=5&Sz53@~? zddaY5FuExp0-EO$7QKvp)anp}43|ok9RtoFC&oVz8&DzUcWM3yVqRDMGSU46tDWjG z4w~jX<$ZfmbLUOt!y3&U@}!EVQsF?1Ab()<$%+-LH42g?prM<&G!p7t+RWHm3!A?y zm#@lUTL)7zUO43zS`_IIfig!^xL#+1*SJz?tF3Rflszr%(_nsbCwI~Va8X=bHPHjM zPfEAz4!i4`KR*tA19L^JMA9Xw-^<ZEDwjU~Fvn-rcUb_Z_tJ3!3Do2$r+@L$PkDJ( zL*GgEj;6AbdH+<6B_}nE-^-!J%CQ;aue<Ii&|!j3xR4P2;k+?FtyJKZ&(ZPv<~ZvO z)MFB|+xJn>6G6tsbxT%up~{~oPQI>ec)6)*V`o0RlLDm3v3k!3CX(;leRFg{2xtEu zuw&jx`1w(PxKgvevg77lJKt0B04hG)PMdhHJU`P@bs-E797t{K1F3EP@;X->J+1h* zlo=J=*x_AI-z4{gXO41;ie+03_VIR;BHN2k4fL(e_@k(bhuV#Hew2cruObC2ogL%t zc$`X5D5pqAV6?++OLA*u9eX&@ZewH9J{r%sPJbL*6eOAsdcgQg9IEo%_0))QkkiXF z`n;F?$@J<a&;icE-BX|TXO$s2X~oUDrQ|#FJd_*=aTE<br)jegY|*L%zQaA8r_{#u z`HLN_#>dU4phHgHD^ax^ngp(u8CJSJxtALB5_j;T#Ap&-kBCw{xeL*-eXth;dkVkJ z*~tiyXCs+2E@EN3vN2FW6^R-e=y5wAx`L06oLe4-EC}R?8ACEdNlTXtp{;C!aDZX` zA?S<~LMRLnKk3O=U6c6jruKCQY{C(LH@iAM+Sq41hS@dk+COXugb^Yj-zl;|X?pP* zTH3XwfIlpTNPq<kHJu~5>95n*uT2NZ>9#mt!VLz)X^~2Duy31(HOG5-N6q7Z{<i%a zz`Va_*gmiU03vVz0C4{sfEl|O+c^CTTFq|G%F5fXuP{t~bHEFPz={FP3ycJ`91D^X zYzu@8NpN{il8{J93ozTHwA%2rj0B_v3%A_2uirJ#KaZb}uB#odRi3Zwx!tqhSB^6~ z<dR}i!9j632mk<aW#v>vb=R-Hwh(wP@6%;O;F|zEh*VI15sacR0DWa;B^0%%rY>>U zxie*ai@hI$BWV~`6az3L*Dsu*4Q3b##wUaQBOZD!6cG8A=$Yj({tmY*#75CX&}85+ z`!W6Qu&H9*N`C-&2XF*Ip){1sa`8eF>QBhcAiCEeoti}8L}6rGmvix%v!!$&)|zpW ztA3O^RI$YEo&)FbUbFEUWS2nRs*yi|_%z1~Gh&DHL%~4T4kP+z0>yLF*xGW2(?q+^ zDiVQl*HOC3<H9v3#I4sY-?w$kS>TYn?lRw;b_T}VFai3=4<li=U&WtYar9%ic?x4o zwJ+`*&ymgP%FPS?ww{toy$o6=+(^`8M^}DF+143l_ijxmPZ`ERX=f1DAFLF{s0qR^ zGz<XZbL}OfGt<7=WJ|xMban7@*>yYQeTMU6Lv6EpXjT~^^IP4V88CFVSyFZmbeRfq zCcZJn^GcnUNXvN-B`lj*7Xw$+PRhNaEp@AeOA6!jgy2bQcgwN`UeKEM>+P!!WjO}= z+k}N``BBEhFTat;!Ws(S!b=$hfPS2S5vWt+XIQ<zPK^xe)wo+!1x|-hB(5KVnZ)h% z{Xq+**@NOZ%g=SweiZ~nRSu!(hxc@!>ex(&)Y`%R=HM}gUKv|9GY|@Rx5Ulb1%wg? z{UDacy7J|=`#~CDJV}cyJ1;?gk_Xj02vmKs1Y?E~owHBvQ_|4Dtd;<Y=kcEPn8V-Z zFlxcx(%LQdLmlDsm8f8jPJC8T`zcuXkW^r83ya{VAE0J%Z=;MHi~XbnImWIM9E2Gh zFLO)CKhZIg+y|rYrc?m$+U!w#JP%3+G!ute;BrHsQ-(ZA`8konZbup_{P{CUd&M_H z1rN$Jy*T{k3-4*Ct1Ems;(8m6IcM<>ZUfob83Q?fH)@q3w#nBJT4<`#)0OZpYzmIl zy3<RqWr1mzyhpmMKitY%gH}thVKma3<wS2nJUg>|)b)y*TF}7;zU(&Jd{g7=_Qp(4 zp*r#<G;IMyF~6lul+vcYOO#{6`@8^(F;+950t&h>=R=Nlmuu`jYV!oz+b2$hGb~1o zn7CgvdAjg5P#t<N54+~g_Mi%54xtr<6AE(RYwInn3Xb#wxXI)qIbABbq<Ju|-siC5 zN)NK%uKxEbBqCUnFJ5OnrV?$``<<W#fBTqnVYak6^@Pg)VH+F2{^{4@#@1W0ViR2c zIRzxVJaMQud;;xiZ<A_rM+tl(T-P3>=U4*y=y#7I4nPp2Sw)^wE|?gYUcVnmMau=V zwENF9m9-WF5$fY6BaFw0kkTww1mcCI{kv0%OC5A8S(c30O1Ss<Nz2X`?f$M^96%5U zlf)Cn1fmw~8bS3OW%r=KiG@bVOsR>N^63QLSs$bh_?vMX%E7kk_eB|}yl^}}c7y@q zDc2D3y);c;Q#E<c*+UE;qeh>KtG_y==Qt6jzVvYruLOPmMFm(vu#>e{+!J!efu}-( z`@*}{9p@|-EM?`>m2&Le0g}Q0VDBA*MG2I2-DTUhZQHiJ7x%Jl+cx*IZQHhOn{W5M zefoBH#5tok>4=`xphh*xFaN}!b=+#DdA~+~nBI0lOUm>@z&7}P)VzmoafJ)UY!J%5 zo7^x0E^}pAz@|x8qc6jpw_DINReFUx$dePU%E4jJvBiTLxi-`=EaQQ?{V?A=9V1NI z`&uTa!d$z%ZIp|QP!PEc&MwVWhrYd@#pTkffQ!7BI5COu7F}XSr?X=#!>$6r5D!{o zGiFX6?0;CCo{_cANMDTXkn)NN7%IbiR^P97l>}S1{aQY0^~94sPY;7}H@RSe*cS@7 zOwR>1rJ7+87Q|{_zlc(3?1xp;lFF?uSo%F{fnbb$3w;|1O&KEWi9?PS!imIzmX9lL zc-p!7X>D8mVG>Y{2gZfGjCtg;^*H+Ro&ROXy<gucnXC=5%^)qD!~2jYO+UGW#6xI) zuhi3D#O%4CqE<xR(2e+lvtqR6gnpVfZ^PSK#GwXw+sJSg)Lq7{IWvkOS#?j4pMGEl zlBtr)fwGW|X4$$EDgScDMlh3-_&5)6=eo8h5wgBG0a|@HaPjt1XN5h(R6a#XiUrnT z6NHBY<@<8cSiA1vQl6wd=W9#1y(WQ5WXjD?uwPN6pGkHlTi{U>AExhVKzwWUEeTv> z#=OQGSw@LY+&v!X>?vSiEEKyX5n-4e*D$w%=U!8`XIt36r~tVJ7&6>zGFzlE&%$~; zcD6TZZdK|cJWx_^7gTXK*8dd{Qr0%5Rd0M~sj__tnA0&8t2($dYL`UKW>v0_2qtAm z1`PEZNk3Ov09BnWpgV<A=FBiu?3TF~T)<NzWp!VCo8ST|q(($mFqwE7RtR+2YDnQt ziL}Obw_#Jp(nM}tGC@im**e-p+yoohG5%S*6KaHZvCJ*9rViQd<LCiFGm^`WNn|6C z=j%&WmIrCYO~YgQ40_#0u7#+>a<F>wgR!GkeD$g1PAGRsou@+<|LfuWT7(N57;8wp zGwHSaQQs2|ki9_NnR?R4Ng@`bQSiA^N}U5SdmgC1pvD4|?S`J4<$X<3SF1}Wdz?x= z-PiW&tgGAph)LAjJE_e1s7G4yUK!UcL89ug1x8PMfv9gVCAMhLPN$bp6X2}+@uDSg zJ&H|Gk<*ILi<pe`gB_oS{CRrE5y7_E6*5HEzd2IWdCJ-^4NfkcL*?TZ>%ryiaU-0r zzMYs(c#RK`@AgdUHbesC$2=E)q^&p$HDp5I!ueb6$ZyErRSl}&IiLt3Y)plE>usa? zfp^M0^mFlr4f@^Qvz1w^PP^ZcChg@+sEOS~)7nGT;pDtka%}G9yOh$}_Z8M6ls<Bh z9K*rorI4dK5+Q?F!jfH$_kO-T)*z5(>kYxB@Dl`0h0W>wSLOf);!iwdR<DhxL&t5Q zldoP61@&_Ipgx+0C05X0Z-%O-=wigj<0a!t6%_F$o2)N?l#sYVWt7G`9F-PoH?RgE zM0mu_dj}c$<j*Om5z7V5g4ZrTdwW)OIcs&=j`pu#BN+<1VB?C>3|CSoQs{vm`)zPo zw#Y#N`nBHTdm6sDVG|$7UD%h7lLj3R;9>(oT#8sQ#<$aYmT@G?WFhgcxjg>*5V?(- zCg`+l>kgaur-0QP9S!aAD|wd}cpE*&Q4~b@V<-i^>L;@Zq5#1Z#P$`eyKnsVWgDmO zzG+vLuYu#4*$D#Om%eAqF|-9hqm0o+bUt?d{k&+a$t~q?ZPKgGDLN|;>*!&3+v(e# zJA7lCL;JNdBwD?5<vaOjfj!XHPkbF$%nH$6Efa2kGf6clUJb{n!srYk!-j1AS^>0~ zqdNgNb-g%>lbbz^F<zw5%qN|2$czPB_kfXbPACfpiGJuX??tXxlPwy1l_arn+guD8 zHQp{lIU$@)UrE^^L?1x9@$n)x-7_C~#`7*gt7)@f6;O2Z2)IFm{1aK|pjWdQWWpSC z63p_-utLS3dFd)rXR>nbkL$ZPx5JsVB)r8?v^=!5srnOwvC&Z$Y!E{rLpEJqi<3Es znOh9uV-Sz|SB&F%_(}AFsr3B3+SkYF{fazo$|PfUDt~=`IT+Pnb>9BsIoKJ5jGI(J zOk5?e^>A)kml7bKj@{@O`|c|zdX2g{5x<f5-U!mEHZIF&85C1TQS~vr*#Q%1Bsf&^ z$)n76v{Zg=4v!$pGrYF?-1ko}gEt&r0h>mNwC=jV=`h+Ssz;$kcQbdF&kPu*;_)%D z-WVWnZ+(CpLP2;bd;h`gMnkG5EK+4=rE<Z2yV`>V*89xbT!kcOr+?3Mo~ie4^p4Jb z0W31B61}5s&&YPj8+eF1!nx97O^}NPe!`_a(zYVi>A(=H(iQA6T_P_|3FT=oiYPWz zcuZ{|db}-8eE9af2J+WlxCP})ypFv)o6%y%QT(3Dy<C)&4e|97{$QXaHWD;_Ry#h9 zE#UtgXR9XScMV}-*sXZk+uH4weGK^0EgJE!1YmAa33g~#V)D6}qZb^E@Q4^R*!|Mi zQZA9}*@P#I71LM}#P0Sr;HR^IMF?aruv!;OfvpRpb?rP7?>Ua;)8`#S_a9py;^c&v z&Mc63`niAS2AA#5&&zo+Gi|NQ-N)RhB%SCSJGv*p?z=a7F}Sh|SyJ{;1;b6d$Q@j} zGfu1{v_X<AtG=ObKqx9Q?lSFEa$I`@ciOArrWJFuUT8sIg~<o}#ARwNGO6WAxs88Z zE_o!wn>(6^C-nP)drY_zCO%e?;muDSYhAV=^@X_IY4-B-6AzJX);v?I2&n$8gyo+- z=8hL|Q6lS{+N3H69dw7sO{H7Me}c5r+=lY)SdmJT95rF8XHjk(f7fY1+<)O(5?C<# zMq?}F&0K{xvMox_hTPJo8&J3iX2V050>pRL5<@_-f5XH}FCas1Oc-U`p%a|CcJuO* zm(SZdU)Lh<n|id}d-~oM&!>QOHMd88*&75G3Pg#HXQY43UwZwNMKi?JEmLKXjXm`D zCeGO!I4{+EiO>w|o-pNyVp{G+I_py_%zlAm%+x1QLm{%rYf0f?^`ZQ8X-ZHIZ91qI zt7LK)R`_UmD;r^AFeWY{?<ixuM}x-6?$yQY{Bua80IU3spTipu)xcH9)Np{tta>N& z_bfojG)Zmq>1K3af`ZQa-5Fp&h0wLXvylt~;sUpOG9&mD>UM5Y(-NM<_~|8y+vVLB z)tSnv;IO=D#o9%zSxY~#%pi4!C=Q2UfZXLxDm*L4ck2Y4_{&+wvNz<RycfrDgUgQr z?~c3Ty9s^C#RM&YZiPV<)NN@-%QEVhA-{IP1JUGbT*M>4>a4)i#|oGZn9(bwB|V|D z@TthBL{Ln*7gX3(XBHN!JS(DH3zu=B{Y(_{(C}h(x3BAf;YLz@Q`T8{i6%E|d2|W# zMX*+1m&m?&Ko+RFPWp+hRK+B#M5rGGowx9;D}<>f>@3r}+6_0kTLW+5NNxg$MIM1f zhao%$w|dxKMu(B*HG$XTu%(qfKiYd@P!|E#8EDbb*Jkbo)1{2A=(4)2{DFPE!yjsq zT5+Ls>8R|l!))-%%oH>&BO&uSUr*d|ieHT9>&{j{WF3jWoMdFr4ExAwvn-f2d@h0v z`^RUMcMw(^JDnn`6<^5t^HEY1&Q;luV}pP{m~_TA2?ds#&|R(ye-8>Qru!`?Km#CS zq>Dx{<2PzN8Ft$eq6)K@DL9~lX#AlVqR>nRH(vz9Qr4HdY-bO#=Fg=B5%w{+3|;j4 zptMX$@3plXT$i<RQE>C)I(!UgTgrosue0EX9(^xnEmqhcSSu@-mc0IOLEuf;T9!fO z6$TYfreBdQ5U<qw7$ehN<I?nEPNG7G;i-q|$$o~-rnYG{fA{Vzc`lQ=#H`g5ifh`` z-|Y)(T-#QXyM*!9EkfS*C||NME`MZ}ur9dUQ>6LxDs>@IMja#2gWRyHJJ4N*>o^s! zv+-HWa+ubyp0ziT@Z*LydT0{>!8+~xZu5G!_7xa<T6Xnf?WU8&vp#xLA_5p7Xm}B7 z?PYE8Y)yS<lT4LbGlSoN2Ra;{h}MEPJ#~=+X5vzn12!074YoX4R~DuVaJ%!Q{zj6T z)L}2q)9*F*Yur&Z|6K|M>GZ{(r`)yc>{<T7O*7M7I-u0dGWb`A9s&|3_*xf%J`xn? zd7|tHXtO3gU3cIjH~qIo{K-IoyKY<#1jxkak)9GfmXnkk#TvBf){EK8YduNpz$>u5 zR1a2*m9mDH{j9wAU2nmmrvC7;d#QZ=LYCc?oUrC%^=-h)p>@#BCwg`S!FjxJ#$r(D z?$F`8NXu7>p3ML!;Ah5h2tI>Ewjc7x>kTh2-Is|D(V3@Ex+}2wEMJy+ag||y{E_fC z_TUHMOT)&jt$goJO<_%?cx|Sn76E6hJ7wYg2@12B0<Uy$5E}UjrW{cZAih~QL(DOj z^t(-g011&O*}i}kukoD0jMI_9#aP+GmNp<{I-9k*r4p8sziRUo8m4J}IY0r;G%kx) zM&E_{3gq5$30g%`wvh85>TYFf%T+Q@&XkM}ZN|y&%`^bZu&nz1m1j6!wL`4G$SBHZ z;cUb3$n-;M*q7;ZcKRN}aT!)XG18!9Cluh(TPD!~ZJ1bv#3&lS`sq2^rx4EiG2e^B z7_8u;<?jsf16$t?6l*L*JysuN&#tepzLzzuj7B_?v7AM7YzLrR&vtKDHvj%+?;y_p zJCf>=VZWS6g9J9xkh#5wo@cW`S1ibQcB5f`&`PJLc5lX7_H&I7qrGo<X3rhbkitGj zDuwAbDRip3ZZUIUx*nQetglny?!V)5n=O=shsO``!Yq~Yf2wsE$=Iucx4Z12{9|!J zy4m8g!~V`|=X3aQ7MPYkUU$2-l9xjbwY9~Op}YeeolnUQ#+P>7kr+;Ov%=?UiGOM_ zw<^`sCvq@io)<$pMd#UZ9%#d`p%+@HxfTA<(6hEykQWEm1jM+{UB&oRT$rA|!DA5M z-A{r9@RXWMmuM;q^j)CK1V=92C49MOMF!dn3XJIRD&`lV_B}bV6!43|Lg|lR@BHP{ z8`avMxv-#j7>}g$0F6CiYuYXI?%Z$t@-q&rA<Md+#`4h<$U=Eww`~;(#dXy?L&%2( zrJTfL2V~a)Ys5lzf|pm%O2V+FOI7jfCubmqdty7HV4hL7rNqgx6ex^2bF&9)t^9Bu z`yK>|JdeebL1h+1IW#E;`|ONt`!!SC9$_>*MwA{RvRRLBF-U+-6M>S=pl^v5q>Qf@ zgi#-ROOEIT8eL;cxVJ#Ep{lUn9<2bJoNI*1R1x##$`W<xk0%tepXB8OVb?Q_W`VO9 zihHwKzhuW`yEzuox;0NY9EicrUly5LlNP=C7#L4PVU=g^Liv&lHSMaC<;$c8Tb^Am zu%dn@u42mBzLF*&Z9&Lc$^AjX%x+}nDnTEx@by#2V6Cb4Jk}=3{5OE$BjA<0t*X&; z!kIyDT~x{YrawIBoMo3f@yK4K?AaM?yN-*$?><g`<H62M9WeB9K9XahZZ)ckdHe?7 z>u6d!FW8}JR=;cvH^J2(2P!#KBZ*n=p9;U%!VbfS4hAkdnT2?Nok@=+BQAHKyETcZ zMW&ek*xS8dEhVD$)rwl{J}4NNK7xijIb3K|hJp#V8;F@buy<UI<2wp%#CfZQ>~pIg z9jgHMlf~*KXN$9>Z&J<R$N+-^iItEbwXYb4A{vxXf#w;l&rMT5EJdDZ!r5x#keNx0 zrqT0rD6I+yI@XK_QsVPcPCV2jN$FA^4eL~jjBk;}a|`$;)6VRi37Sd?XFbJ^9H)g? z7q5#NT6vHI9rETTAy&lQ!!4lX7OlD}UBoiIGwtT+<*&?^bX+0zgbOMD^p_z(4djsM zz5&6qv?+mg8K-l*(w*<l9c2L&){l)^i-ECKq2Yli6PzoeP$8v()9g<V(?(@Yo-l^` z$`x#e*j6?KSNxC`{-Q&iqd1F|53qPjp9~jSjT=DKD~>Pyxr0^bi#xE|V5AT8pS%dj zuXH6VSevS}M1cYs5&-(M)@f()<bfvucRU9ePi7w1)i<#NHyOauts-!1_RB1%n0%xX zDy~KV=<SRL)WZ1-x^DnUGw8DQUPW)AQmgs$A`I3B2SQ?%d1)tv8fmf=yrA^`lg5D~ zdLzzj1^NJ0@ZJ&jXj8wM3n0B$3p{P*A*Z~yh>xM3T(L<Yj>-GWkOUDmnM)^YXU}nu z=2wS9stQGOHiHktI}i*iubjK8i}_JXttNlC%IBt{d7&Hw9Ve5*PrdOjVrsWB2}rS9 z!x4@9H-M2W-5G>>o*ouD6^L}ltcUey#+Ug}LP)$taIZn;jCTfooB9%-ARjPfly5bT zq*5bQWTF+{aV<24qnC``Ru9;KBI`PM7rGOf;vZVlXrImPk;9Fu;<?(#vKs1jp><kO zK$t&I7JA={<BCCHSx!OXpE4-N(k@m2+og9kQC_kzL$5SsH8@{xIS*^3Km<MQv_$;i zswv;_I?6@|gan@p9VU^k`6q5l%1xMMS(Q3G{1iX175lV=+nhYetrn>eQNaRIouA$t z*YO0w^%okqr7x_hwgf~_eu1`nO+|z)>(v~(^z?u7lgjCn^4<+@NX#Yk*At1(P=b_$ znEBI?)=^&!gtT|k7Va*r7T36oTk0K~U9oyZ6&5lMYJ>8%H!@n_tiAQYvdp!6PA#my zmiGGu{f#eDmt_AQ%Ig(C#Nd?50P+Xg#o-q#VaeRfi0EzYS`<1<21wznHy?2R@w%?$ z8b~}uQ;Zz+f%YD5J7h}JChH|f{(b`It|lgB2q186v`g}jU}F{6o8gPUHD?jXe-Lhy zeei*B;cSkyb&<{+i3W43`9<Q^%r<l5Dq4G|fB7{T1B4N;XGN@&PzJ6FaztIE=rbfH z{h2vvwc>K1AT2|Xbgt~*FRn)7)L#vzJbOQ9md*)KbC}UC^1=)PHrnjV`CE^OKqP>O zMbt};EEKs$RHEp8WcN{~81swYR#O1jg`U+rJ0s(0It&rLGmPF2jpNEMo59<w1Z9S0 zRmex}cUBQ<ztf~N=un)*mVsUujuxLaH%TI7E?h+V@jRd~_%b_sDne*O!_PH==ap1U zk`6~)Q;YC#WOZR_OAiWFM?jp5qSlVf088(-nBcjk0~H}z5xnKXX=b3ZqRiN!%*6g5 zXNT^JcFD&^|G3vTj>{f77ihXgRzjj3gHsgZH@pik|K2bY(km$VIt@U#c6W=-1&v7E z{Y^P}b?J8OnXsRaigLMa3E3{$0XPikAN<I{y*f?hTrK_Y9|MLl8`|kOLz+m9DUvQ< z)`8dxQXDyar6=_DuN7a4p~!X3251|9@*kuLu_Fn{2#V&fl&F#B4g4>a?t|N3TU$yn zCnSMYl};J!OwqCmvdEU=^ew6KG{Zf&SSg6lyIbvbIXBL&7k>O%#Mkg`gq;F1WXsH3 z@H6`~B$fy2@FQJPJ$p1I2sk$@BkV+@=U6%=aQEZW;>2Q*17|Gb?h#fJJ?;x5b4#94 zB%%_jQJtGHJkvMkCjF(IdQYfGx!m#CFJ6T_f;GM~#{GaraK5^X0!L5+r@CI=MBg7U ze{DW+wNwnepW;nYpmHYKJu1&{?@r?IKw=Q+^gFjcPq6M)r<d;zugmw0oOCn*v3WEK z_i+cQ&X%zi7leBczv+4hK_}-oXbpQ!X;~-8z_moOz{h92aoc9)z8M~jHt3@l&{O^> zNTywkCEIrPO4}1wps_m-S-@+v857@759U9*N^_@76Mt@Ph&Sy36`PcOmU62vY(+mY z#gDsDSWmR00b{UMLsY(diYl4&A;~j}LKB<+fd*F`{7j(PBWn!3_?18%^z+iOIqF^y z+To#aE(KJZnvsxV7#twF`k#H_|K<ZBBt3<E!U6z@`8zly{#Qdo{l7MGWA}fI4pHo1 zUTC)dI9*00IR#SGJ|YG9u}HYKBvd#Z&`}a@5lTom_!SZ0(m~7<5FjP=Jh@MMZtw8l z<lba4Kk%MRXM6JR%yc=PY=^s7mkCJ?3jE!>D5H{51aI(pb4A+yRlO9#+Li`-SXD>_ zzZk`Q_4nIkR28p>DpJ8YgD7P@ssde)OZd9&0=7J*`ghVHnJiiCeVz?oL=Xd4?==Xm zHgzgAY}I#~Z|z|JN{KLK<p2uQC&`2IFeruAfb=$wa%(_{=~X81$O+Bd-OZQB9?rNz ztQ}byH3!*dJ>Dret#(M>DJDkjv);Snf#?yflMpIqwEiaA06R&7%#Q#OEO$Eus!UPL zAXw;6zpCyfUOis06h}@Bv88p{M_gQLHo0cs>s`*yZCp}8H!{$5OVcu!ko&$rL(U;X zNbX2Ijd^%Sd4NGj$q)11HsJCrdQ|PSluGh7653PQ=AFg~q>eekBnEt~IT5Y|3~|3` zq1xgwGmK;IT)+63mhEcB@bMT+1!8FlRf)&6RcKL4qT5hyS#i22TG9vA>@e=kMs`#( z2j|=u&qdh<RMQBve9K&(!$>lJxy&*fiqF(emkSmd#8rp{LUVI%yZ!P9r22#fKHWm> zk4Fh`+pxU=8!Uxit6UI7Wd5Gma2vbZJHwjp?fB^Y++Jl+W`!FT1W;Tu4`m0@Hh&ui z&|R2Al$%U48o+(4|D2ipx$u{TOuIi4IHwUvjuCK+NvP#xQrZ42p#g09W!b_w979)s zJ83+)imVqmDBaA7dXa$qO{Tcum%5pdfMjbep1)2^^NDYQurYvtsSKmQw%Hng<og@~ z7PM=X+~g~Xc*DgTIka24PlQbt&w!!0-m&aEUWDKw<S%1yTk13ZZPeX#15IU*NvQ3+ z&#?yf5TPv?At~GJrr&)BDal^3&D4jS{#LpZ<mYjw4BUC4QA+iQ1L*N!YJqr%oAR|& zXG+K3j&7;nRTI;&1269a5onc!=Xy4L^|~YdwQjATDLmU(arSVk9RON;`&qj%p%ii* z_@!9=bakegA+S%OHqJ6yswcw(ZXKCh{S_{pk=^~scxGRI?q6C;&}s2Ukip>FeoF>< zrf&EHSoYZXKoT>;E>VvO<X2hWwpP}Wzb$TEix*EMR|x>JS@(SPyEr$1i>1V`;j1aC z{a@_DUxZ()bG~P|^Wy-mM%f&MkVhRWzX0o3HwCBG&3R0&=>Rv`wP&w<42rZElEEcK zm>=ynm?wHJOQ!3nI56#n;aB<F@mCx1K9WBI_;hvPr4oM&hC)G6eg6tclU?qkFLy?$ zJ>@+bZeHUlBjC;2zs8}sYW?|)W(6FgVr&w8?$89s^}-g1Ar0q-_sfZ{<>#eXNB2CP zCprcRCavZnLKW<lD>l3<6`Rtm#mS5)DqkjJd6`;-`9`Qxnc~d1J@qi9N_nQzt=3Tm zpQ*Wq8%c-%yEmXp{!EQZAWGElz+9d1$k0is^Xd!;56X&}wd3o~?8j3zD=^9Ig+hR{ zeWaaWf*rRLhs>srY^^WfC0Xl2q!b0=seGBb?WV38(Zho~;lLfjGkdI=2CU}|k@q~N zgPvOTdi0G_2ES(tWV0vZ#rM9i9)_i#dZCEmec^x&{MoWYF!pCB$`T^;qE74t!I_03 zKD!u7{ayV9WuDLGj~{%Vt@g1VLmm&1H4g$@jw4Dk?YgU3J)7UIWXL1~%WPC8q3_x> z2am>ISIi8J1H>O#dKvEmr}zr#4o_~J#9Es$?OEznbdxMYB2AACPc!XnQy?Tbi=_^P z8$2{Xe>Ds3<?dUmqswNr=YW_1(U!OIat}UCPWsorgM7Wmt-uHrZJ5Bt0o|BN{+?7$ z^_I)Ydr|B;j@eFgLpTA`6mLyj$5plid&C~R5+6_YzkBw@p7u|4#W7@^bNgHYsg=bu zd1D3Zrp&ApZq_5JJ4pA`c?hWDAwSdx2@$J197HO0pM*bB$T6^h{TC@xwO*tn<4)2l zjGqlpbW1u5h3Ekrp{<}5t=_l-9i5LvRO)v^QFL@}q>Bf=nD9~Czxjk<A~9)G-4)zm z=6Na}N*iN4gV`SC+4}5VerB^~w>@7rGNYCjJ9(}<DRxduV?LEr0TywcoG8zahS^SK z>puat*Y|3Eb!eVSx)0c}#E&E96ZHU0xk1cn1R`VBH)4tGwDDNUF(4X+lP7S9WsC|J zx`E^%2fnO_jtj^`oV*#OZEHd<q)`kL60ZfPw<DLrx~;V*ip+7WNz;raG57#9-*qX$ zIO`TJqmNGb=H*f$UPcV2>JG$MnD+F@eUHShikrKOiZkB-%6*6LtFpu52v65OA4B_d z?eaj~&rfW7+1Zg+wc!v5DS5lSPb^bFk(t}3wMVU}%=nYw)c@^pDCayhSZ+@#3&6y{ zFx8WNn>Bq|yKldy$@O3i1_q<Ab%1uk?z{)XalB*C`Ps~{>u-o#QgRasqD=#hfGovU zJ+v&=E_)!trnQCTAW`O{<g1bsi(k%I=hCmd#;7q#@~g>;2}(^>XUrpm!tr$p5g!ba z5kFzfU3k4LyU|Sd&S>CUOL}zAj-sL+8eQAZuq%7n^Q!%l`c=u=LizMvSc!5D+dm>D zl~VKbNQCtg0d(yszM(BAvIBGKH{gzLW63iM;{;o2`}34&+VlEzwMPrXc55i5Nr#E^ zEI$8F48(TV2G{I5sY|%|@J2rwM|hT;FuI<tN9<vi5lezco@J*-ZVN8v*HXU_vuJ`l zEd#8wQPSyBg>|n;Mf|!DY%w3yw1UaE=~UVTjio90)!pxU<awPEYQD1j=4Lfz{m|WS zOZjN906SO79iwMV$wFyyht;nGi46M=tNm?DI{7kp-@oe*5oh0ZP?kvSapSNqV;|(k zy0Kxu1W?`_Twm&qU%%zKl49lV4j1SS^GH0pgJ25?0(@EeO{d#;{8X&JE05ZUI{;^U z>Reccl(5~V=C6XBeUjM&*KK0Dnchl!hBlmA0;0ENec9alwVA|T8b+$R$6+mDTHoQK zG%47rO&$d@t!G0D({r|?;YX3ym^lumu7#I-LVvShL?!q(c1iB(nyyjx>YeUlalLFM zpa1d_(h~x5U(`m29`V)gSYBT4HYU$T7#B3QZ#*x=DKO;tK)GQs%R>2Kv*rb;L^bdb zKr9C%$YIib=-wkwWUS^OPL*tK>b6K61>y9p4ZAg0_`^Eejjf(3HJEh%CMH)@)6jW| z{&~8RL{=r0(TCRjK9dHr$yk&qsHa*JN;Hn-q4GEdSEHh6bA~xtr0?TvX$RXKH{^q7 zwC`_5Mozrz^erO_-9lB+nu^dnGoX2f4TJC*e`)M8#g=*E(Nm6qvf$!b>J@Auc?i8^ zlho$NnB(EI;Lh!8djBpJ&)U(E2}X%Zr)Y}wt-kJ>fLH~BCpkzve=<Y1_Gx@I@5`{R z(0uAw+>Dp4#(`^T(42_){+tq{$j{-;`z!`tDJb$9M77rHT!Y0*Eqm{1y?@Z<xa8eU zt7!te{{2~lN76!i0f@g3ss~1OEhUs~;VF(IQmt|MY#v8}I<%WfPO7AKs0=21s6pOL z0@y9X`zVOTT*5jB|HNO|oGY~S$rDxTYC64Rtw-!BM`bac`ib*Y=H>3Fh5m?Nlg1!> zMiHEY0McGH>Rcv)NNW4IBJtT2U(BuAh(*C-F3_YtQze$P^oE0fPxA`xAy5MB1zUJ@ zxu0(TYYEP^v*+U+uL;=aTDV1qKtA<w#pI%HH~#VzxP+9@Z7G?Ad|!K7A5_;Q$lAUJ zW-1u_D>TUcTRcQf|4W7h#+o%vur}6(*=`M*4#>qq>L@R+3ax@RK>OI)2+u+`D^ic{ zQp0vfhE9KI0T}2MTI*Y2sl=_my&HE4v<|ww<>vR;of#{X<>ck%dWTA}BkpLA%;if^ z8M?~b)JhMPb(K}BYSA}~^VO!WcVg<!PC`a$dj6u{Omhc)QK;I2=O%ixjRV#_t8MjV z0m4R1Jr>FX%WS?kqeFDui9oKnWw@Jf@QHu}?M+0B08D|kc#Nv@T>UUBv%<U4wd5Yi z{Xzw8TL5RoaNHAJAM~Qb%6i0sd`edTy4BjIjL+BH3@NrfxVG6vZ1OcrH~TK?8N2^o z)t$XPtR6pK3=e}w;fStax-J{6irlH>@uQz+oT`CJS<%fN162%M`pRW(LD+&Dha|?+ z;L3utqSbA73M30{_UKn<NgGo}|DW5o!qVP{1GC6ywk1~2S@#(i<*c~b?}a#H{O<lI zN13wmLsfofJO@*@Uce6ScW#7%a@|Qt;?T7oLPPgih2KXnY|rxVjlaIj)#shFx%fM8 zZ)>1>YX6!ykeD!W_Ktw(ex8gW$35L1h{HddGtI9X?3HuR=OtQZp`AOa&Gd4H)<e63 z3$A#YCf>N4&CTtGBFElXZZRw2K|;}$u^TPF_hs!qY?NwLGj@q&$Fy=&Y>cD+3^Z`O zJVVIdO`cl1v1qP2sl^UfSGan;)UoP;3Y`_tD34i4@{SoCPHfvFp{2fGE}@m28P^gX zOo+-VGV#sA?Eu9_`_?Y`-N<~dG279e=8xP9u-_mg&<k%*Il98`uZoVKAfWq4bu;L? z$sN@8&wp~2jq_YtA*$#bq%jU;gmiy&+V{7*qph-5H}J2O*RKb7Dd#;{sT@I;0gTV+ zStl%vwCWwcEM_Brm(b@EM`TqL*758=JYwODN6KPQ-ylUD;(|^8u{a(~9hdy@m`>Ic zt)zxm5*MBAs^9-j17S)q3kj32k*lJ-I>Wk1HkeK-Eq1Y)u`4k~iG1M9_o8YDWCV-q zP4Ad3w*>)gJLfjpfpI&WZV?0wAMPvNz*xUNa<?`nrlHXO5RVIRRQ3_r%V(LtF}nQX zo412o@%+9=avcAqOXYE@-vA*cW4pSi2_Y+(`I7p*1jCDI_VEzaM1n{MmC<t9qCs1; zTNi1uh~+XMsUzEEr7rzA;Z@1=l3*qbu-Z-D`E5WeX^TE3_2a+Ngm`hH(6yE(fyj@q z8`r$rZmF_IzhoPZ$C0%(09s)U&X>FxXYfa0wkzYY$3AR9UNQsmVu)s;&d0RJxg{kH zCatyj-A$JiATxKdN1YWlrhPc>^B_kGuC`*2IXuc}bMi$`7IR-c)^6T<qL0&<3L*3y z#t~<EGBdm53hnP{7NZR>>`)wJrWEipa7=?<|5`+oB&z50N(~2k@(#p4ou@;`D)i(+ zsnJR^aDDv^^Yw4!v@O69Kh?89W(P6NshJ~z#adce34xu)>A6}ihU?0@)8A_L-vM=W zaYv_{zHQ`GG>aho@7Rto{(4>$YX?}d+oQPMLnP%8j<|E<e`eZS!@JMKF8u)>{h=|? zhYoJ+z3AMAY5kWX$1$T>G#4_{+cXONe!Iq)-pLhQvyA1oW(PgPT0PMIR#ep4ce!;j zHQ{iNOI!PMJyDJ)L5$~c9ZJH2wn?GrRG-B-B=fzGQN(B5*PjIF+Ud+p$m1}q`z3h( znW)*+K$#aF7k_rLH_Y-)*hcL(AK0NjO0I0HftEs!7@xs{U#?B#gh=k@GAS)E!YDej zdZ9W#vjr}{*76(LDtX3kH}+&Kyt6hfl;wPLexMv><c&%XL!wBW)bGR6R2Lp_zi1pV zlrvFn!ar2q$M9v=bje<`t2C7!TW$Gf2|=4VB_W`;3|7ZSS+G3;$9oGMqx(G_kKSu> zaCX{K!v5`a#x!k-Trr~)>pXGBu3EjTiEo1>P`@{%YOy3;avSE<P+BFJ{z_BBcjj7> z6IxDUOv_Ia$nUg0*Q`q0+Fo#3S<V`P<T!v0Z~b#s#9^xr+Y({z_iXFPi7&y4(EHcx zn-PtqL7UZ;B;Q+6)1>j2%@B51yI_jv%-6Ux6E}74@Q8NH0ngIQWsRcWPd=fGZ!23@ z!eh+*S&3f`Fp69I0+H=TY*uSyrg?5JsgVfvb>>2L7dtwiw9i#H0kT}G?XGnX*VUsa z+xDovlZt38VpYq@bF8Z;K}Q;En@|M+-RneXU-`lq>f4G-vB~e$J-eZ1M(b1MkHl&k z*Q-JU4b4^Tt%wY0qjqOK{0{%RbP28l)=;=lh~RJIoe_}vlsZW{_*4<*CHKAT3tA&* zPYWrp8b_h6P@#Te^+9sBM6;Yr*9Nar<tJZ;6O`?+pFr^inoLuHVN|Z!v4zEh$@u}K z_r~?L@3u_u%?dvl&LOTPUC1;zvmFpm{xR08;cnY2VbS;5A-Rl1N%Q_ZjOvBHR^kmM zANMm-pu8AYTN&q@l2O^AK^cYF1O=R5#t@t@aBBEeTZ-KPF~myN+Fodbc=R@BQg+@* z9V@;UA&Yi5-2yU-l?Z8WC2xC*yE$@4If@k~@)Xwlaz0i~pfupWsU>-Vk58h}&hlSM zNO$RUZf=Bieh|LvIOF^*lVl{e^I@?zt4!Y;%DTYW=H(<y-Ey(qp3OYS?BFE#6)E0| zOOOxrqn?WZi58*QE`V2V%ZOP5d(_Ez<zCPd1?Na*rdx1hg=~f2v^dcxWnS5n@AI>} zTb8d)L9h=4>5KQKzg*Bj4XN|f`zCcO)UP=tg0EQ;Tteo&G?74Bgl?tSEnFB-slBvC z8V;QIokGHDUwd-N=pv(Tk3^oSzZ!c$_1LW#Klig%L0%kWDo7(1-jvPU#(?U6$!y}t zEPwet2qF_xk7J=87N$;unRw;E=+X7tMT{xsEKIGyqg-ZR`sJgy2ulFsy+|K$c)1o~ z)_g!?^D9r7wZ(3WZ-)N1J7<i_Mp?Ijp#uyXv}Y}3(bwWO_GJ#vgW`TP+Pw6DRae{s z%Yxm1$3(YI6NcJr_LU@IHtaF?aeF$^Yg7Bwxe9w=UN58lX3+S)w>XN#pdGh1-k)#6 z4Xl|@BSldD<?p;v$~+=*T8Pcpknr?$gXO%N`{a|gM+%O?%kn$GF%qu*I492@ey$m) zn>th0f|4Q2R?QF!>YX^tVtt|5l4w2Gln8fu)LWa7cvFzWbHAe#Q*rM7DmDz_vMUj- zpneUs5)mHE`r?+WBW59Ob;a3WH{dIkJ@W?MRK?hT^-&AF$rLuEk2h}N<~zt$rE?&q zk<H%!q2I30T4hLydxaiMq>(p|s)jx3&epInjA6TWI_H%uIh7FYcJg;v1gkVI2|mfr zEA&fk_1!QMhK1t!=?!gk3Xa+<%KP~pJo=jKQ%VSh%jNH?i}sP57s@JtC2H6Z6@Lls zI{4M^>d*pu*m!;rVrMacw>d#zFqV!v9At6yFj~E6yY5Mo10#s9Cg~}egT?QtUd)pR z9efz(IIdW$TyP=z(>yhn0xa4M9&_^Jk766gYB1vkZ~mo^q_Tv>A_@2lQCFj#7eEyc zhjQ29BfP&Ih1zD2yAJtQPssV6Jp9D3)HH^T&a|3UX5q*eOWs(px~GR3gADWGhu;C> zj_6Fd%^CUIQhQE}v=$nAKUg*ziVk8Jz{3^t9pXmb4BXifNLF%@$CaHC<?#Yx<J_ny z+pu)Dg6um7dl;v~(#02M1+pR_j(wQQPQfYWG&fq75Wm+q#Zba1mPcV-=Ds8K5H2)+ zxy^~#KT1eVZIE1M#Tkhf7+%jCBfvpj2G2=|qS<l-l!tui5L2p%7l;Yg)@+6z2Aw^| zZZO6_u8Ed(3#TYJ0f)uvFEXaT>>bW|;ay(t>tS3$Hw+xVixj#$E7&Rk*MHP&v*IO= z?4d5J4cO!aZi(RQJ<+rz8=fKa>Yzjs_|InjmQKV|`;G9&f&=j8+VZjGVkNW<zDB{v zQjW=Hrb2&F6d!8+5G!U4pa!>V2ODXil}W$~o<_A`63tQV_4igklYY+95iQiA#i2Up zdB1=L(^Cf^wPf_|ZOfwq6%~K#TmC~pl|W)=(Q)GWQ5q2lL@>TZe5*R1NAT?1yBZ4p zn~$#qZ*k~AayStOppLQ9HvV2`S$fkHCHHrDFN3GXMgZB0r288OYSHh-<G58tFJ8QO zUh<0)rPI8xGZNUr=kpIgv!2xeyuwLx1nrJ*KXf!3#9P0!6(`?96n|uf^qOggso*lS z-+||?yo&9OJw>2U61hy)L=pbIQu9Wj$AH@0P{++gZ6Gq)8mV&8R_>Ul>5?@VLv;_G zaj-&@!cv;f?2c&3lr5`eD0cWpP%)W*Z;|<w{E-cYjcmGq*;flraL@EWO8}ky>j^=0 zGfbQ8gONdF=g%;fB&epYoEM=aK(YKdQSxcAO<{BUc?$7k_FnjNCFs~5GB5ZOYc5rv zarVtv_NtgCK&bo3PjJK#mc$U&OrTUyt_aiI%gFnkhJe#lkULhlwfwT*UhtaZ5*{S_ z5oXjlrovgn80`x4#JuKw2E>6pNNzh0cry@aIw^hbCfZP#Ee=EoS(|Gwy7K5v=r8eB z8p2o`s8?YC^2UoVp6HdfgHCDS)~ew2lXv0sK{Wn=kN6og11(7q!8dF8=uD=UTPwk` z)?oX$GdJ6?epdO{r1L1#_G9IV(feNec4&GTj-iVscIZbi5az*J{fIHb_wh+j8|2(O z-+dpBZGOmq{BtqKK+j?4uB=7nWBk}t%jVbKfGhtQhXq5I@FNBjyKi0S%kLQ3_Y(qY zKfMIn^80Hij#1DiH)WgyfHC(XC>L_s2d}bQ6n6u};D_&J9hRG(b&TC61FdwyHT+yR zc=~I9TVPdeHVm-z0o$u<$7y$^l?~84+x6X7a+g}Bxc{UHeX6brRZH&nRaDFpv(*Tc z&S<4@{jXx1SnvLNjM2FXBN!dXNb*0KL1K08;7wj`-dus8H<~%*#5qU!{==!D8(G5G zgp4UqIMfA2^4peZ4iPEeU^6Znt%9VP5{^luNtQMyZ!}J4xU+f>TWj(7wFDX%w}1!% zutd)@>cKlef6hbw)gVnPih_UyXyJ`+Km2Ct@%+Z;EGR^a2C9*Go-}mzD<^>kDL447 z*^}UKZM0DPK*{=A%1b8!haaCIW~`n+K|k=|(ac&<C6gx$jOFDKN)I4tm=<&*+TVWr z`BxRkeT<we^PLi?Ml#4kFJwm{^|B`zgh|jW{SiCX3+NtOWv(jqe(f#AWRaDtII7?( zL(F?+0UI5(LA^$c@#jy^I3fo|Xn$3=#T`U1PPtS@1nWJT?GN~W^$Kt8`g0rs0|2=H zy?5}>taktB%)!6S%PwGU9Z!xD_6-dTcAp;YAc5w|1=0dBC8jMRwSkLBq;-g;0v;Bm z#*rhHFmaUc6O~YT9FZ541C@}Ksi5VH*BKZXDojE|l0;tLHAqiR&Pf9nPgM`m82tRA zy+s8L9b7EJg_JBjOmnK{Kej%SQJ9pilKW-7VZETUl$n`$%OjKmD?Wu1D(Vwym6I;7 zP^_pb{lE^3Q%I;(4v!B}wN+QC{HL0L>IM7XrJxK*N!OaC5X41a!6-OR(Em`Sou{9X zVP~XKrD>mIW?*J&Xq{7FU}9*UVr!gYV_jfnXr^OmU}91V_#@3W7bLM%5TLV{>I&pk zDTzr+`X_h$xVyLMwbkkQD7JU!|M~N$+dsZLKAuh?*v8&OPtq{hMz4-(5+rqT|FB>( zAYt;82NZ#yk+P?8P)!o@ICvANwa$HL)&K}B%}3PGzdJw2pD#g3U;k6e(Q&UlQaV6s z;kJSfu8c-v30{hJnx0k)OiU+(Ny5=T*-_ELP|`<H!p8@W1_hYuUiv<g`5qc2D_bz% zv@8!XCp|SmU#}8!`QXCxC|zAAAuToQ{LHWd1Z8ewQl`$Kj)9e((TR<b(b<8K#mU)$ zg^{h+$+3aO)uoYvt<fpq%y~?nZc28F+W$;e@RW6^3_S^lpj|nVX+lYPGveM6wRgLD zeDDg5c+<xJRJQL5%_z&`$v<XJJklx+fdSM%?Y0Q|uov0a0*_$`9Wb%^gTF?$*QyU~ z7kQuS>{6D%v8OlLS0JK{WR(7ue_-JXOb-Soikw#$N0`c{tL<RTRYTBkueagn;fU!C zO<Rius~LAe3w_jHCF~)1ozk`jOr=&_?Dx+CR4l(TJ2lj`8f$0AobqYI<;O5FH8}%= zA<)<)(nV5PyVdfCDErZ*1Qp`?x!aU$s0dE%_20Ftbj_RZkI*k8+g*Ewd6js}bIrH4 z_~<PGwm%}&Rq5iss^dXWvZ&z9W@Kmx3h`d`zn7;WU-)L+tkKHtdb%Sl$6PZ_2yHT2 zJ}px@RkP)v$G2Qn_lRc_?r-j1pmk#Tr4zO5J=bd3tf`qD%|(U92<1e@$g&KOZ;HXi zw7YdO?<*}<e%yVplyKlVI;~Xqywm2c+UNVHlX;uzo{|-=D^#hqf@QU7txZ<+yAZI_ zxYGxu06@qt)9t=AgHD4wkooREISSjZ%|pbmzl8jgSn<rd`5jHU?4R!=B#BbXxyc;S z%CIeUbFHR2nW?pj-n?d)q<fw#&4!%h6}{qw(;v#Xo{_KhGR~$wA`fO-4o@lbfF>5c zU?ySYP1QVJvuNbOQA2X7edsQ-kTL-ND<AH0G__9toe%$Ps{Vh^p!~;F{eLO(KamoQ zu1qaXO?J%xPKo+P1{McqRyHRGMrTF_R!0B-ObP#sCYIiRh6(ili-|KdX!U=|3737* zrmuhJgoRTB^5|#$>j!Y0Ar<63^!LoIQ-?n~Fz|t-Jo-pngtqOq+cT~jf&s_v<&RgJ zO!rU<T6l={NaHf70}krGSH8>mMg<`1Wjx~gPh<bV%;Ho;K!+;qjm;y<JIzP;-RL+Z zOjMcxQ^O#aNjY8Slh2$iCv$>SP&1d_!xkapNFnzgh?(hf$Ith#pH*x(oFo?2!we5K z9~WaI_BdI}IVsEIIsIg&d?DqbzRC1(P!JW8T$+p~$NsJvX1&Z{s?D03eXaW~Q_Zl< zW7?l}6FJqB#4d*yUDemohN4f;&o7{rgPHlmluNzm3fYbENgb`ZIK?pK_;^V2)xnN4 z|G5<Um13TX^(WtL{SK57U^sehRX2Q7r;h5zT1S)EDrs-hqz{Xw2(@1eYSP+kPwMnR zV8?RDwMYQ|ke*~ZzG#G8d)Gm+Jp9wYt35OJ;yrs52uz|!(r;$6(&e(cIt>#>|D4B+ zWr>~-YpsxLGv7>(r%3SOJ3Y<U@Lp&#ZYiVSA1WMsmB)Mqb)=fMH*D{}KG1S{K$Z61 zJMjZE4<x21<MEMCEC>eYmsI3VaFqm+{Wngu1dAA4|J{8N1pDVY!tR%|qsjm3i2tcN zf|1>U)sD@c=|6NtJwvTiBLjmSD?2kY1AFLyCWPRr^YA|dg!Dgjg!sSYL-Re`&%(df z5dza9|G<L>dL6|*1Ni4B`xnjuS&VPAB681*njRc0-|X&cU%!!?NH1o%+v+<(XGqJ~ zcYLiMOeW>T7&tAohy#es-&N*$G7b!gT8LX5*<%CG%29_`VsT|l(1S5s&Uo4Kk#07r zFv(B8o10TplTt^!u>%RSC1l$iZJiE#leQuLE$_~Fe&7=m2xe_Lem2+Vph9}x<`OiG z&v9KI-j4fKpW{)t>Qvts&@&r$S_=;MJCekd4kq7~jaQ5TcUo6NkEg}76S|x}^#ye2 zFzKVYW|ZcF4}Wsbu1(Y?T4Ueq@JEh<OUm!FNs*nNdK^XM^$(>8J`L0^mYmr-?*vro zD1g|)`6#z()cR}}-&M{G{2N4lKL5apMAqLa!Jls@i4H$oM=xvadyGb#;}ufUQQx_) zOs{(;w_9gA9hM;@Wd5*n=vdz8E+=W#M@EZgluxS*%LT2Ck2M$Jfy36UpF%vyY8-gG z)+#6&w|RZR$z@T2;<f(4+QB2W!O>+gz!G&(YcdRepjUMM-&F!)_UX(sBa+mp&LF1i z>R<8tMHsXUiK%aTuJe#}#LM(96vI225A`lT*^uvV84I?2n);u^XX!#SZIM*W@Z_4f zKYB(i!SZFVqJxg%xMpz6ld$_NT;;qs=i`YpKj{s_f84q}3HLc97d*{tLuT$1)_g*m znfaSE>&jC-SUZs<yUID;oo{2pj+C2UQIBV6J*<b`^PRHCgR`mRW^C`KuHfuFQdmFw zr*UFWdQ#+9IR!iDd#E;w{TvnT|E>Ntf!>+4r7UBXI_BfK%B?Qb@ic1Q$xgyA$&t*N zc6(8GvbjR6P7;b#dMg-D|5Gnod|F2Jpz&{@Z1;9H^B@Vj7YJ_uC!hq&6}SEMS5U$I zEx7Rh$(o{riIJ_t|G8TIzgp^lqL#8}W4B{=V*G!kbar%hVf>3urbY%kCKkW{N=nv$ zN=o2=kW%<xlJXf%I8ggvlk#uHWuHSALx|e4y=gzmeM!*kWq17JLG2BXdoe89rhaXA zZSYoiB{_}EhUT>^*(r4Q)g!;M0f_~f!=TtYGrw~=U^wU*@Uv?Ox+jKzEH8!1L#c#Z zjn4w+80qA@m}RHfx`{AgOAfnym-BZ>Dg&37a5HtH-K(_#&*!RQlri>v*zgUGTf5JH zGu^<12ph4g%EzwZqgj9Q5dPKI!6|}VRZVvnxd_?zqa`w-_2|(%f!jBEBg)4O?l}Fm z>TEL|&eY6NF}B=5iffy_04cjN%GjOdseONc#q5cfQdg!8O`k@a?(vc(ulF7Ub+9Yl zLfUiFu&(z~SqF&Us*Er*I1o3DOPi6~W`a~2bno=!sCkk!3-aGo3>~I~r2fALC&|AI z&i}VW{J$akKQcsJn3))uS^uLc(rCx#%4E;PP~XVn$mr<wU&@{TKbj)Rf$@Y4+O?ps zL9Ki)4?s*q%XqM0N;0H^K7Bu!IX3J7P)x(_?HnO0n;woMj5oD`01tiXKNnSxyiS$a z=tp{O18spvBhBQT@~axR{^WO1LoYvky0+xMN!<EH_V}cI8{i`mZy>(|BT<8}z0>$9 zRd0(VWLte^F(zr}zZ4z&MfV(p|2kw@G)6p^-!_>!{iM3tb8P`FW%*w>6{F3`HWFvv z@tt}jemc2l#zj~t4AtK@4c<(~a%RY%!EcTcWE<Pt2kDi_55FxDX)Wh20O?&pn0wIQ zE^#I~UyN4TILXIX_bbV?XHuLxo#n_`bcsh^9j>j%CR%1L-8F`bZD>0UnvKu59QoYO zxT&H7>9<loyJsvsw;G1QymsXTd640Sxm+86`y8i8<{^zQEX|l?D+?k2w*XZvOG2dm zn>ii+mKlWqGC<93%>K93_@7#8*gH5nur>c}JN%<%$x>^_=JNN!%F5)>P#g5$66nA` zD>aP%kwB6EWddz3(!izu_vn=4@<blpq_;c5|1YhGILFb2A;zNp<;)Ina$L3DUvO^k zJ>BPBPbLl4WwESyL)jX<$|Ax<V&P4)L}kK=f-I1qb2qj=3c6$cN<LhX#FVOuCdXvm z#TgQ4BUQFoSU@wJ<ahcncE<XjEUm8nWh3Y;=o)QnAMBu$miGpiG?kKjj~?DbpJ_4^ z{71mlAUVaFEt(Ya8fX|jtQHZAnQJ9dS%r&RsX6PVL=TTE!-@XsQRD+?V?E=lzb;s^ zDPAOGRoVGi)UD2qgF_+tkTTlJppI*nrjH?^AxH7jHIo}Bt6Bwgp5tag(&_cdYZUg} zFbSQ;Wy-%IlS5}w*EUy%BC<@;kc5cpA(ho>_Y9(xoH-ed>Xrp5UwVn`3&Hv6JnVe5 zCbXIU3>yK_hx%6+4;1~d|NWmPzoomWrvB+fT37{k+p)t=O8DcUA=g1ND`A^$Or+Ob zaCl-21)jlU5;c=z8Ydy)lAy#LdBss8zAF?+Ml%$TA_v`rolt1QmTEI+;*W^tRzgPH z-X2NZ?qwf~U{ZX;)tHKZQ2$40b#kcp+J?v9H&0=H56r<9n0<0X&79USn^ZU24tk<& z*V3TDU&VpJKR>2{8j&3Q_w+#FVSHTc+t;-tgk8dSJNpMf(=Xj?H++sUyuPX=AA^|} zsW!GgFLhcLwpQuaiLuMr%00NAyzG_xGxB}u{mp7;mKGNGCb~I)Dml<GxgYLLDo)sy zquum)%G&Ji<Mxg1)t`3ta$TVu9X~65Y-AZTUhc;NCdk{=Wc2V#^*SfT>D{g~W!3mE zq@rQ;_&FV}jAz@&j_uDAUvfJpOqzJv$lg4E;@{S|pKT^0YV2?LTb3Q)F*j<@PNvCZ z`Tsxm&M`>VU|Z8|+qP}@ZrirG+qUiAZJWDo+qP}n?&&)*=ggdQ=idJlF;!7fQ4wE$ zQIS#guC<<AnJ;n%pGj+gLbR8n>97@N_;n2BwVT5}({V#~OFfszd5DolSLCw@p0~$) zk7cc%9NXfr_KffR*MRp1m%%|X&`fujSb{J}3BF)tjH7z;ip=&;7qU{coBtviax?@( z!THfzmmjUg|1Y)H)WO8j{QoQM|FcT##LU3z==6^~sLsxT)zN{a#;LZ3g_+HT?cbr! zf4MkB{tu-!`)`%jMP_Tm^S>`u_@@~E0d+L#Jgj>9ac*|DAKYWBz+KI`ajRrj&1Pwr z1{d&+?1X#Lh7rP3b4L_<Ob=M<Zvm&{Uq!9JNqM7`g43fwk66+{Xq6iDeOkqb^ZY7D z-GH_!36q_IOO&iQr#Ov&SAkIhoSm!>fG-f9j6p0BBGm}A`OrZtWgw)4^OKr|iIb>B z29cRXhLIg3LCRJUBZX(m8RN8eiDNj_iBmZ|X@Ef5sbfKww$;*ljP1fCUZoaHD6FP+ z3U=vDfA;oo`8<0{rWm7M#!zaNEIzlTMDR8ywyL*2dHPGWEqo55AAcM<D>Nl0UprA( zs34wt&!CnOJI_3b2Zh@|(|tlhqKQ;*Yty(Hl60T!MwCQq;(=!$?Q>P}Sa+Ixe<&27 zSEFQ?{`GR|@EIFvCbK^+;WF~f$W3F{2EEG+{Hn*8M(e!g(>9RejV$F)R?2+*Chx#0 zgCd`n;WBeIHrViPLi3yw9h3u7l)#IU+b~%piRYowdIApitAIn;HE29~pU1Mh!Lx=@ zp>U<w?7f`0XKo~xQF$*s&hQg?E~bY1kobG(WO$HTX_u#I;Da3doIjFvcL|vSw2Ft- zIDvXnlQnbwB3)E}=-qp5yws=}!AOg3oR(aNZS9ufWMz0C@b2$LUs8U+?Bs66_5Rs; zpLj&8o8yo}J;mCoT_n%_xBODSvgG&(ygiwwy@eu7>FG?9daycyt=7I<mu2t-@7X=h zb8fLqsrp{Tr1W07%CgO2Q^<VIL+fzI6D-e*T2&TW{Aknp<V|&0Pv`f&s&2Adi4#ee z;Vp2IVpaDzihD`W$aEc&=5foOecvOxJ2Mb)$`A`J%vc6BORw>7&9(GBcTjKtq~;h4 z+OH?GeA@Hh-^nAnu(6&;w{=%f_el=#doR~^d-wIIa#0M5dtb}<n5mdH-j*<<F;XAE zWsp2NpV~{Dq%45{mc))ej4BoXu=39zas4^r_W$Zi;Ar6TKT)~2GORSR8HSBOtq@cp z(j>vuxq?Va<MR3RtnSIhv_EZuyd-QGvg;-8jKKBSra&ZQn&XbuMDjf05b0MGD_NKg z=2;?xkm!bRJ`LA%Khpk?ym~6v22i?59ksk^@=nS|C!c%vvuF3S=M_=a9_`%0gmIiW ziSr6$vp%>{EQx}fZ%gx8gTC&;9d!#+TRQYE3}qccLkb8EU`L2k0>|uzUD5<{I1>8{ zFRFHMJ;dWIASzyeX`g>C#Sgy?4Q#27IRK#7*e{XZ>F@GExIreX(Y{vZ4rHravfUp1 zW6~bM+oj|Dix|+x2XzFxK+qpxbeEq)Uar=B1!QVb1e7m1gkK-7Z!xXswU{cCu1A!a zBv*}462D>|xK$K2%G_CWpW06p7Mq|EFrZNqL%`WWxd0FijpAYmVX+jOT75+-Oj@6I z7*qiQDp;`WkE#eWD}ogZ7GWdUoOvRU;3%6C4xXx_l}iVZXx*<N{DTh`&;i2@DRu`k zgeZJM1Z^6rT4*CAFtFfT0+8~B8D{>H`_~zMeZ)xzo4U7G^QCzl5<>RTx#%dlmX@;% z49>7M45VZ8A))!lTgo<z-!Z0^rv&=-gT<yGY{~DG0dTb$M;x-`XSwQdC0>QV%>II+ zhtJ3qrUEoIgn9+@T8tP|%~%y&g}MA@{SO`_YB}Yxj*M%|7zC7bQ>0(}7AXfq2Mp6> zn7+soC=5;jsoZR5dTr>BbDa~8;YFVQDF=dr%roC~+}TJdD30hw)F6?2<dg!pQ3RJU z!=#6OW^o5pcL(?#zld5sPn)x$$ow7AQzj#>bmh^Z{OA2Ocy@Dqx;wKZ5=-=pawfSf zk=jFKD`s#neIU+<s~b#Ww3nR1k&-3m0Pk($+@?6kvB$P|VY#h7MAV3UYcGCvUkB_J zcm@V9?E+s`4%a!tH5gi=6p)4DCXF_&`Lehci}n@ik3nk$@ckiapVAw5>0+O{(}IUh zmm6JLlj=3miE+{QCLxB}T1m*exzp313&agf#!DN$0wYz!#k`Y23?_OPEf<^}Z5Gxg zYYL<mgOgE@dj#&F%qP?ZE?P&>#bQJtK7u<{XK$)1!Aso`KS@^|@bqPZ1=g8OUdY~@ zT?iq0J%HRk%h19(zLDZ-_ri-$d0Id3uENWxd3%}dZc8Z0;a=PBVx1@7kIi~*AJpxN z7;SqMy1itBTWuHhS%2XF?Rze0;k1ev9?7KUn*EY+?NHe;&Jx-<ycWThIdRY}QNLas z`z)Cw-JOG$#ieWXD(Q0!*U-hjia%9zEd2O9TPR+fui_(7Ar4+8st9mD-&omz#Be;R zOytc?-i^ogsxwY?4W1P{Oocc+Ogis;ahikdn`T$+v8t4C!BB+V)0kb)xgIhH&z{p- z-WX-s?TusqgV-I?ycmE{b=~iw{4%r7z8{uecMHzJrYN!NY}$_K>}hJMC23^CB0S4D z6dVPor8c_C?!`!6SYHcoH)jv^o*YQwL$45nGX&$vNNO6d-#Z%9oVlI2Nk5tEy(CT( zUliihKPt%FuKAD}Po{`%eFP7`-d5mO<Mg7u4ZV$~yZ^Okz%kz`_#gU4^xx>)|7K|a zP&vRKl`BS?_z#s+PHtS@PT2YpIaDptyhJ1f@~buO+`z5H_IWg9rsJMj8m%c<R=QiW z3Pu6rasJRCOs4#O@)(h$4Ryg4l%!zFdD>hJHW$$G38ats_Gj|xd!OrPpJ|;HjRu{} z#5fWJhyfSuv%O3C2gJeq>sUCIto_ORipBuiqOw&2QDh)wS$Bw@t6>WWbpKsg&ya8) ze*&L`X`NnG4k$$WLp6PcD53*+h<s{WEeb#Pa|pB>1lC`308>wofqwApeurG4bBHwv zKfU%o=@grK2Txbk_h<HY>(u<Z5WCPXkZ8byUmo!Sf?#&Fbbar$EFeIU2u)J}bgU}7 zc0gzmWB^$}$(|~jxHT%(^1WD<y5C5-s$~v@CUk{!BPtB}Dnu5nH^%-SHY~c)Ei)hF zp0{y;6(*|yKmxt`@`wu*6m4LNr;i@ue?t~5IAi}X!;(Y>ACM%Vmir+^r96SE<BI^z zV%*DnY5+qF#q7Y-q>!eO8|{IE1=j!+DpKT?_`wa==lOgyr5or7Ki1DiwG;C5(*@)` z&FOyamDT4T02t;$MirT}i!2TopHvj7NBk`U@*70TO2=DKydt)z8abeGz=0c(Wug)F zfaJCSc#Y~i?nhL%UF_AYmndg0ph{Y_YEqvhi<SUBHz+bKmqp&sPjs!$Ay~yg|BkiP zjl)jrN6N_ceuv=6DgNl$t3})%>pZC17)L_aYeoNz&<?W&d)qb{R_qAKzp8V3)z|m{ z8KurXD(bAg85;jEi#wrl(y~27zp?Q+PEy<&sIAJM1MR)UBh86&Tv15iewepK=00`% z&7^tv8c;yi<$dsAaTy+?k#u#X?>O{g;@nO^C(wY#fQJBqo@GPV{9#S!niAZ85_L?M zytS|CVT`|xNyQM0XJgHk*7$0N+@|X?uis3FB$0M0W~3#_p$tUEmooYe93F4mQmzs6 ze8r@_8b`RVF$$&ynL1-*3tHFz8}VkXmnALq9SPOf&S_+UlyOBn*L)?LY!Y(ZSYtWe zxQ?xg@q?%fS)19%cn!u3-Y%pD%NG^p+K<zo10A0o4^_@N7rXt;KaIA<W&2G1I`&wZ z-?Nah=$xO#L@wXysDa(vs{ND4CEQ4rR^U$2*ZWfSZJ}XxeDkB8nDQ>T^Re>$-W%!4 z_*gHx>CV~dXa!ybd6LG<ywj=APS3`(o}4t^PWyl)deOS%duJ`n(hBL3adPyiPsWU? z#a78WxmBriZ`kp=L=-Qxj8P{}R#tGeQ=&^g-P%#u6<z|yY$UxfC8(~k$BqM&;j2C` zJRW$c<5jZ7_IYBN*sRM|`}noya#oqt>buabDu(mf@q>Oo1_j>HB?ObS5jb^dv(tTe zWX%XRtC_X9J^dOlSVyIsT#xMP$EP~b76!=fIq++`N)Lm#)_$kMOYSJu@fNA%gH}b* zKwEf4We^H$uT589tBei@-9W8~`OVd)>a8q=HXFA!5d8q8|6Si#V>`M77Sp-Cp<U$j z9$MRe6nM8^Fz@EKj9qrOh-B}9P%LZJ-R9c~{HhJDARlYD&yC){svHB7B-6}KX)F!( zpHQTuox^V@6Iw^h-!`;1zl}_28ULd|zEu4!kGzBiM@xZ#o-a61)3IDfMlwSbFjIpX zMU$@x0T``sj145g8Z7`!p*2qBEvOC(696OSM*&%$N7PQEh>d_?7)Q%5Eny5TVQ#^F zS^eeR-NEcS^*Qx=)q1@3dNm=zdmOx0?IZou$a&$)eYsaz(+c1hUw4K#XtA^SC%ca# zu0vtUugw6r3y30=Gs_+bAXS6^)C3h&_5G=V4-$Z5=5bg#zZZ@lrAu#9JJTzl9ug{N z-@7m7tH(2Ibpg0yj2RQQq*kt(jmDWP=a@UgiM(Q@(d_a2{>o6xIhuLJdyPF57vTzs zuFD!FF5b!xYA&Wv>j|46*a8@A)rJ@pv;%0|=XU<3u~%jl=CAOIoPe8*XBP%tEpeX# zAPkwKGE0)jAvjndBCFN*B}0Gspqy~_2mu1bMJ;24rGK!A3txy}9*2EU-=fNNYbwHi zM;&`S1TTEw<S5tbVv7CbHNMe$kqBYDs}kI8Z<rBGQ+llJB~;ySXQ3lyWex^L_7rBB zMhG9I_rEjMK6lDmo@gal?3JDkwT&uB%0kWnKA!~8TUM5I56|BuG&3ob^i#fyeEgXZ z>Gb<+!)hh3S5~Vedx@-mOJHM~>8Hv%to9HTu=hA>sh2w;;#bWu3v(d5pS%)8`^pnJ z5TZ04{(sxZWp8=YEE(V(ScEbj&s~#tk6a(6#;i~AT4Ut->Rk^YvYpi$FA9%4Tv}Z# zr#DyMIcBhVQ_=d+OkR}I$u7fSz#S?ES?Et^065fymf@o6hH_@{H`8YI_($S%N+y1{ zKrvtxJ+k+&qf82tIF=K>WQRE*HMC<n>6q;@i#&=0o503`thH9jlCl<6wF`uZrEYU^ z0jK7JDNui|Wqs!#GrBs;0~|q<2hRf(99b0K7yRrC$6CV#*?6b-<eU5T^Zd$b092GW zKgpiXqXQsQezoz5+%xk1qNmI+u{DD8jz6f<^r`e*Y9wm8y7^*8O<fRrzE1$)Xbuqy zzM429Hp=`cBWA^(z-u39Y%E9M<!#c&YbHXtu-&t}ARLbIZap8IBn;%KxXHCF61On- zH;XkCo`RoQDHN&xOJ=BFuOI)2qvI>VZo}A=rH*J92D2pnd{gE2GGb`|02txZ5lv*g zyhvD#MMqwTh0Ul=)3iFa<{??vbtk=&-Prmy_><?oBF30LM;nB6Rnd*7b0jQiVE1Tt zIll<4WPeK8Ff>6+PB|~HW?q6RE~j`Gkl*a?Qa+m+88O%G+{fN^_q!B%59Z%CAHBg> z)6lYtlW0lR240l@og$K8&fL~q+@j7mRqlvShl_zh1V_v0dE^c7cOD6$p{{luF$k^a zHO`K~FPG7=wHnf2rywp@5^k}$ZBXXfL}SWs#ZLZI3dsq@#9YtoH>_o0HIyAA-`6Z+ zWuiWXE*J~NS0tGZe~K=YDYn<Rm5k-v<PK|b%A{&{*Wsyq*7qAdcd3*FYT%aXbdkP0 z@I&;U($S@`4nRlESUtI;U{G99x?<kj0A`D!Np%Fi3N))2RP~tk>Xn|gRq<(zBPYmI zdgbBu<Xv&l+g^tw5c^nGVD&jIjilW&AN;hwlukjn1#Fq^6r)Du5JHwCqjfsm6~(kj z9Q92mU}0i2RaQxh_yQmL@(>$tI5>N^^JRh{^T_GPMqQHH@7r@JjGZR)0GXHJ7|_ZX zqGC!r6+i8QCxx{DmWo=a9=Lg-7iv+*uXiRH8}2IzSeDe)*hHf*Q;x_%b6^ko03yE@ z*3r)#C>#nC<9Jk1a1vWqR9Lg7SmSC-b^E-0Mz*vSy4~*wahEc&JrI`=EQrnrBIkjJ z6l4)?;K=8)3UC|LYN{?WG`qe`5h$+Nm;*JtTfPm8L|eGDj3_CzvB;9@{Bvy23Fult zg>1;Xx}4&0bg{bCwn#lELLH1#|A+jgk}`dv0dNG{@$#0}{rL%Rp`YMx&Kd1%X3T@2 zj(W^GQz1I;+1uoIxHeAKN>H`Z`VRVv1TLbOikYE2yXeni`lE*B1XNQm)LJv!bF?J^ zAXHZ&s)hQ%#j9>U(u+n_8|nTf<eqdPKS8b*2^pD9$FZ+cVor6LwWFPDBk=P&UmKnh z<5zsc?HkyZFAEVfbAppuN#rXG#c`fi6g#BieT>XR-_3)!VAJBQfx9Au{fD1U)aVl3 zMF+>tm@^cpb{8=|-qW6~<tnC&xh)nkmnK{K5l0Iy70Vww84;LelJ$-;!e#zhi1X6h zj37UqRi)@)UGF;(Ui(l1IjTu}1xlWu4Z;O|%dT*A9adTPdz*aKCjfvpRl0P}Qv#6P z!2cQ!TLmlyjDD8RzkXi-MvMPF9RA;E@z0h5W5Bu+dj?<$AD<>-e!z2p90WCjX3MdQ z?B9;lIeg!DhvWAP0L#^S-QMp{S=_E$9q#u}m+Q~?eBLh)m=IlFuXmsCKc9R)K3?yC z2Kat|eLi1r`x3M`pMOVVe^=}IO3=AIeQ+|l9)?(tFIlg)9q+_=hq~hO{PJBKdZlKc z(rPu>ihy-gtH`PM+N>zD^fNz-XJtmK&Tg_hTmYpmp3q~nf7<K9y!R}b4*b)@02t4| zz-z*W@Cgs~Mo(~)6MM}^f9;Ex{S5=d)XY8(7)amx)AXL|9~~VxIJ!DG$xBn!>3CEj z@?O2&;(dHoLeJ;>ISf>#-Q{%q(aKyRo6X_!FtRHI4u7%jYNur5v33O<iywgc@%4Dl zE`8hHY)dYi%5baxDwAt1{qaBBIqYqJ5@9{KKjz=+*V~hSHi*st*nOx6C<YIF3_&6l zPe{bN(RggEXuav#uyfTtz8uP+>BUaE^>7`(KAPAHkzXUL>FK_b)aK1r+a+yQt9^#h zSi{oeBp=cLa#R$2X}%*pHp`hBQLCE2k)pOX^8*T{m&5;77XQ<n;Pv`?d_9>$KCyYe z&rI8C-x#&vIGO19*TR7QPWe~2M|=BKPbaSnYe0H3D03{Whi#Xe-1K2zCOvkxa0u>r z>czh8spr9-^Es4dYHPYc93ngCIvj*JgN(x2Av0s5<fgkvRYXmxoFw}D=7%Hq^N0F= zVe{QE7W6dzy+JybYSXXHw#VP)+{*(!1J3XHdupF_TNce5gvwlPG-l5FCqZV*<J)i7 zKA&mS`lmw&Z_z1g=ISV0Yc1c0@zJtFuFVbEozJxl-*qb$<Jw)8O3yyf9kyAhoxfN( zh^^Mw&Z|q*blP#zF=m-hN1HjSm9_+3K_Zg%{#>w@ZJZWqW42xd$Zq9ifw^;Ol4W=o zFn7JbsvL@%WsaY4Qaf>1+6X9#+l*#}9s1`%H!!jLdf(q~9y07VVj<#tZ~0_;IiFTC zzb`ul@E~wWjk}&PRqqs0(+78Kyx*M81AVe<x;|gM6gQ7EejN7CNyFz>yg=n1Y_C@a zKk;*}wy1AyK3dWT8&@&;pMjZ}c!GJ~dq02qJj#M&o}Zo_*G*C8LH-ju0sy?L%&Y&@ z;s1R6WUT*yjuwvpG(i2&mZ`}9a-D(OZ$}9mQ`>(IfB>MDpI5(NyZgTQA(=&d|I*O* zbK`#wFtW9AbTas{>#}gX)Uvi+XG8e<rT^`Z54K`K8tM_L$K^WWXF$P#0$K+@9tFT? zMT({`6;fDX6aDdwODUO5F_(-qy2Jx5Zgn3+KaG3s#-qr*4Nc4fo%#D<!)oIu;rXP1 z@hs%ZRZN6>xjmR(-Vc@KE3Ap{%Wo%bRx++W{K_wp;WMD^h*eb0-sAz#nN?H~1+-J@ zmBrYQMKC>_Z(x!(3u3Zl7G-U%%Xn<5mn*1FLR&EV@`~C$-`3JBi0k}t;k?ux39}Dk zHI+L1Rf-RTu!9lS()=?$ud6h)EsYoao5QyyyZ^Z{s7}Epb(Fc+LX-=HC20bcNjd0x zQ3y5OmO&R4wg}Zw1@jY$_9zr!ZLj>MTgiR$L`gz-oGO0}7Fn3MCvI;;3Pjk@VeQEP zP7vKe-~RYXAiiN5nB`WystOK@+B9w)`osk@1Sv~gqmUVLzGOB{cv7S)7O3*0f_!=n zeskoJ7h`4nqqO6A5h6`{*8+>Dwda==LL*&HaGO<eC)Sgm;7T$`(^f@YyGZD3Hm$(U zVPe*BJ<8wb(go4zfR1t&Uzb2>(PYdOd+qv}3PrU!u6`i=#>vk$Vqj6)F#n_ku@p|8 zn35B9%zp1$aXO?Sf_iRIx>P)BbWQvjER<<Np}HpwqhieV5SJ~A6YjKJ8K$Vsm3RPl zSdZ168X~o@9(~TnFze)?^$rz&0z6J#5UA4~I9n?{!mq~Gx<Lp`uT`G{uLp^D+-Z2# zc~S*9UE3L%9oZBMjz>XSC!psz{N<x{R}5tPMDcn%!Z|Ny_0C}P=-yny22t$Uvt`&k zR;@7A=8BLRO<Lh~b}S&vW&|rGy0T#TNdu@Qnu`y}%#5OU;pFZfZ_R_6P7SPnwg{s? zVf{6hy4hfYEuV^Zd3GZyh#HS?Ng&ANJun;lfk&Do#mk)okJCDapXq9&=Z@&9rVykq z_`WqCHyzSGFYzs(Ib=5UoFfmMqHuTdQSBuARMR!fL9=M^moL9zJzgN^r^{S5(1%CS zy6JLYsudhD=JUl@OIf2NOhQ=d41uQ{qf@Yx)^Dr|jzkB^Vlwa5sE<tl?r1o*22{#! zAidD(0PUM_YLph9PuXATi|@N>P~-%a*NyNZ+erpK^JX<sx=jlrf+8DG=5w%2H+iQP zoK-?oG(9J5Ih+$|tw9|^2zAO!TAP)sL3$5n1=*1)TNm6?BQXiMJYSI|9u;<&6^u=s z;2fh#=#OQE>RwAD*iu`z(cEvZ^TvrJK(rkBse2K5twke`zF|EIq2hz*;85`rS7MjH zFuWY7pPssIst=0wmJCZ?u`Wcbcp-T?R5-0GvM-@=?dr4`=+_C<)*KkW&>(9%Jcv%7 zOtW^9rsV263O;Bf>o`Czd7<xWE9p7_TI^A#a&o}!+D4Im>Ut$p`ItZ+Ij|WWg>?J2 z1#2eW9y4MYZ|k-ZSId4vMvQdAC9S&G6e6X7WnUl(y=u<~$;lSkQP#7eAo>3}-N8G2 z`jvnIu&y5i36PxSG2^q=2Oq7di?=ewyey!|ZgN={g@5&EO<LIxwNU5$S_W>pmu&z{ z{<lYkwa^*-#UF2=#9@?b(er_iulngoz$B7Y4^7Si0;XO`ZbtpDl$>BBBHJ=OvUsAc z>~O%7G^6T%o3LjaK-KTCb?k%n@l|UJ!%i{27i%spr*@lz3gwyIjaB1qSrT|cnHye; zWbbp4FKd*REhV+r8Y5yy{Zlu&S)zVSbUSUE58VT5Z3bKu0XnNA8TD%{j*PA4{6b+X zXn!%&hgc)yzC!SC7^N;HjJW_aoQM@elQ55B*$Z(2EEUTqC>WJk*$yoG)g2fLaExz+ zz-rkBh!`1gQ>hY5!9~j$pd#OJdQ2S@-vR3t>LeRDt3Hl~?Fy&+{5zwvj#69LCAILm zee1KG$7C5@>g&p!evy<N)w{&k7(tqxl61_rG)29{mxPW1StG8x8!L&_zjIqB->vDH zZFRx4cKOFW1xdNM6i#fGd||mn&s<;Q|3-!FWI<NWegca5PlAa4FM;JJJUBVnT3MMm z`~-;kW#6^7z0)Bd-Kv9+jy<H{hPlklorC;^1s+f>P*@pIE$(0oT{=(#3Mm>z@kxpQ z*tiHNS#&_25|HY_5JD|HhLz93EF%D1re00aSi{oTG5~Atz_?;8lNv|hr<4>zwosv& zo4%)vLhKkB7{-JW$<;A*q(=A`e#nRpE%*h7PEKZQiiU7%l4F4=>i7_dNSDt0GX)k_ zZrwF+NY6rqES~Iv1Edua&#_ceVIx2YmW{qI=?JYI$$+4Z>O=HMjg$O{wB%?f49*Mc z77GT_$Z#W4hE!9zP!;6>J&8h@_z)NO!L1bZpYa(2K-^<W9d~NMwiE^c;FTNz0P$a@ z$JoKZ^=D!>vEFTw#ok;=zCu-~7thb@bh#m8TPTSjFCP=15E@Dn@=Mn$hHv{kme+KS z0V6;Xfjz7x%>5lm`2{`ws>G1+Xx$NB%sMRJOWx71-c6qvEx8XAsIsO&*17LHk36TN z9vuejbatb5%m+x~b7jwBND{-9u;y^pkf@Sv+Th+}99sp#{g1$cf9IKNnGlkiIo0!% z9VW<9b-TvmDK1#$w;ezbcpIb9!fC>WFr}MtRavwFl6j*<l~F%HlBFU1`rN-Q5If8( zqb@c;8!~B>$EblB|5C3Eqa;a)585>lQ6&*bQ$`U)F#!=p;am<R4jLc^4?q+#Al3+3 zkV({L<7OfXroq{3svDtdy<(X_O8FZ&#Z7Q<v&9S4F;xTam1Hp3RE5>K0TiQv>Vq>8 zmWwfP4SEz9+7|niEBH8Hs5VkdFQ%Q6s<OC9Tdzh@79tdM+JI7@y8;)ACzrP9Q2pl; zQna{en__Erqxb8IxE+n2-xjs{Jv!Xgc!z)aYcq0qMr%-ME72_!?ES|9V589u&?C|f zd@Iq3cPy^zYjd5A-!?Ahxg(Au*ICImjUc?mREVfQEcCs6kf@fAL^ZADDKT}(52V$~ zyWtFpQKc3K_b{+0*8Ud*&En-4;{paoK3*W+7r{>Z+^%F(0iFnFDm6-TL3XDngQGFi z*XHn6fyVcf)&8o`_gmffB+>ZK3`%;xzwQ}{cYT%>Sk~$@3L1y#5<ap?=~7O?KsqKZ zN~l8{cj3NHJgwFnx1qjo$8z6oM|fU68%~!^qFt~@ht6~(BhK2&i278I{8_Dw|C8XE z>btmX>yhI7Xwzv$!(-fqHb=d3Hoz~@sAJZ{)GJi$6_F%Sx>>TzS9ju%0pP#$`!2zP zllGw#{|t~9g%IF^R{#+X`h7=>p66(lXCJ|WiqINdh4D}y6&Jd!6yifPoL07onSkGe zD{I;q<50Rd+69Hm0TByI!BqlN1hpgx<_7_6T_fAc4}o(EPlP_Cs&))UA5F7oEbHAu zV?-`aTIwyqjFU99^RFOCy09z~5SjZzS<SGWua%u*^ypmtB5{R&PgNxae2|aN8~z(` zfttvf64FqwClEGIh?>!&vL19rK|(%=mAch#mR=k$#^*tQ_1>Jlu&m2ie_#A7FGHwJ zVrFFNWK`8C8xaQgox=j^r1cF1AYOk)pcw-Xa)6(p)qGV<r4ofQ-O)KHg<7SAWC-jV zdRBY<3dux6G|H)kWWt~Dy*9>YGk)$~pXPel<ohUTZESUIth+<%zA^f|G5WfrXmMyi z7=fXoTwd$7uIp<e$M<SmS6>q`2yl^B4c2JezFb@HW6RfrH(8*3ZJ&gY9{{EXm}8lA z?`|$`-><$-a%Wrn?y71Lzgns04IcdiyPW@xkG{*MqwQ$3t$zU5%tOC#iS5XO+Hz~V ztCTOJx?1msewt|Q(c}eBN*S^oA5|viC7e^lV6DY#R__CJrY-`>ew<2UXG!~|(Z`2% z4pg}xJ~tZfplEM#shR<-F-h!eqJrHqa1sA|t&H7~S&?pJ(A+)t`o#=C3VD5zb+4iA zmz<ullaWmA-H<9CeYKGXKmI05g5q_en=7zz=T34iAhe_D1S(8SvTw`I=4)AWvkHMI z0=&eLp=VGdh(QoGQ$9BKhH)4LLL<!nWXs$1a%cCnXG%(EE9*DDRPGglzQ&&=_Nupg zdJG`^n)PVa4XfHCFw&-Br<b56Dww@Zkkv!2tv9TDO*Z(hXh~@>?UoNZ0AMpPw;Nrj z{@~EZ93ZD7t{tHSrIOC|`mc{7m*G{Dj3H%!PmO~E|6aB17uRl~`tH=%%PF+gI(OTu z+rxyBQ8q^l63&HmR@W=L3{Tq;zoOMJt6b2L$$+pAr=5h!{$bA(n47~**ZFX1SccMG z|9Be|D?>wE*x$jA)*Psd6^S=C2d2dNbRUFvjZV^xlo)c{V=v(hP`iQp2<oXX>~!w2 zUHFj>oRQ#Rl~uiRqa>FK+Akp6KfeLxMcKyTkfAk9iPAN&^B&R(-*|nks<Azgj<_N4 z_`N8Q7*zt7K0$ndT0$<Gp)}9~D=Vu7+e$af<(_5%eF&(VkF_zi$@S-z0`~Xl=gVbz z?;co<5PaY~Ckn!xSWQs@mlCSNQVZc<r|QK!v_6-5zF?8nmETaQwc;W>-{gKT6Qaw( zr~>uD*E3QUs;yROm_c?-!2BIhwV!#1$xfH|`^FU(rEH|xeW_K}6Jqgu`unf(&qxKV zd#i4CfOqoup_t!-80=1)Og>?mIPvlo=~z^Xx&4schEImp!uS_q5f4eWpa63C0XF*O zQ#D4&xFDtvK--YYU-Q7W?jhF|>n|=7{wL}RU{M_<!JJhMDP2;dkWI_|t_-lBVXh6k z<-!lO@z`pA*-p8f!e{9HGw%336cB{)Wt+#*MY>c_*g*HwU`r@4U~RmAPIf^2P496* zY%ch?dm+vh4rMSVkt0p&cDHOgVQebod0!4#Pl$G&UJDmiqc_y;<S@uE-xPr+ERQO9 z$Q>bOz`AHJm$(qX>aSp{Cea1jLxwdwJZfnm-57<J(N__-r$a3+LL^w!`YTAz>XDrO z{Mg4o8e`nft7ux}Z>3OWNqONYPePjCXEZAz`UppZ{_ul@A&waJF}G<qqy&WsEzqNu zw5NgKg1#m-C}&*Iq+pA;5Nyx7f!8_eQ)s`rp&NIUZD*fAOm%VkK2=j0s8o5ffA>U; zeJmD8&HxIqIOv*zT;0B`Kj?38@KR)Xl@p75Tt9UXK;^%nR8Uay6GAX)C6yvDJN4At zU>W*(NgO-c>OCL<CG6b#3G0v<RV}g(>CWT|=P)T(u`&;Qj7ijmK(XNXz!WQ=SVNG? zXnh5%LcD#M`)f;-NK?gIbU1y!lyE@2AjLO<*t91I7%Yp=khYjrpJ+=+fZ8iL@9b8@ z9joOv(Kp%Kq4Zl5^BC|VJccJmFtI2)Az^={$|s{&Nl1*n4QsY2RF<GY0}FFU`jRa! zQGso<KkHg+pccp4Ulz1J{Dc3v9(#Q*F_VEVu|b8$8f-VGSrRN$Op?*0_JSnVDCGwM z5EYPLMKYUIp@Llc0pg}0g)x(+y%}a$!L43dhhNDWZ0a<1Hm^(nG4Kw`wj{%?H36;V zYaDN;1)2{xg)b68R=Zs4WQ~7GoCFaK<9-DYVuPgQ$Wei!vPNYf57gV>yWUPK?`~1Z z;X}WWI>K98Rz0sr?>v^ndiF8phF{8FX474)=B=poIh@qJHLY5z-mI+DG3M63Ap>m7 zKiXIw`gHud^kqZSTVhA+-wHH?&A>K>W#FIkeB~wH?0AUwHNreFr_uTB-ZY+#m_^N6 z?Ksk|ew5GTo3D5XK|6fI#|l72eFC4s$nz<(lUwBB-rw#(5^_1Kse8XAf@_jER!ht8 zpf^36KI%MVwf!AT-P^Hop-xoVXw43las6)8Q3@Is+J}R6VX+6P`u)lk>zb~9r~yiM zPR9G)KRz2Py?F@D$E{E!v5Hj^P2jl{@QDJ(CXRvIGutmirhz{a(j0s7kvN8K^M?_8 zPg#11;3!o~vJegm;a;YvWE}{*WPRZ^&sv3g)|3(?oMo9vCtHsJFE0q<_m%@fFqLZK zz@nXCEKj$wSy)Ry1Q%0gK%Vl-Qb@|;dSDEFLo@hvtlC9H0#^8xjEdRVLJ(X@$yCsl zpb-D1<a6ky1}+pSM^0Y)->s*v>?&O6uqhW9zkvaN>TfJind~uFq8R6w#sT};-rEJ6 zlll6IyvkHN@}+hILGA~c6D3<1_nM{ltU)I&sNS1!p`?0f=#K+LF^tRBUN-V9n@Nb) z-lfbQvj7GQ$3U$iT0auvl>Sqk_Fe&%oBpW&b{Pik6NQND4&-<%L<qBaZb|%h$nVQy zY=ml|K1h`2N&>S`^A5OLq3`<>iIcSVl3##``ujrFE9TJ>_GziQqzyV*R*{w`Mx!uF zZV8x2&JqOj>(^OeKy2Sq;eOl=&#F0Ww%hwcnD^G9v^>BZ$X9Kz=d3pG8?~0g5hNB^ znEq9tV^i(JrSQfXDcuntN5Sh*gptxQE-$Nrl3p9(?7ySIxBCJ_Tr}dgdNJQE<sS2y zpTdBBCt|8)bqp<i%~h0TTuV+md6)IMJOI~3$p&*(a9KwnF$^|*`W)BNiiQFm<Gk6s zOV-}3sLj!9@r$7LVlNT7t~mDKkRsg*_gcR5%*1KYWfOti@Y7&?h?iYl>2F;-6rFrY z2FZ_+To*`02F3Qq{tSE`i0=8;kqyHA`uyiA`G3ZVNdHpSm>8KGI5_=p{0IT?A1)s9 zfg%qF&yWBHJ`De7Z{!b6{KxhG>;0!9y6AlO#2HLt__#lE|KMwY#GYI!h!d^zk79_q zB{kn)w}cE+Q>a*yma^bs<~_ZNWAWP#Md%~lZPJMU!!J03M6jB&PA1g@b~oISZ2aBu z-RDZ@d~G4Y#`g24alxxGJ##2KTllMVoXO=}R>yMEc*k)Hnsg{zhv8z;tGF_j;G_<V zw|?K~I}-Xyp#hcu=o5dWVRtf<r%$5cMPhMC+{@mD7<OSZhv$^9Pk#jGacNF@C2|?J zQxl|u_*gEW)PG<Iqxcly=OD^^(F6$eT{;mBfk1pO3XiBM(nJU*AFD}C_{i#Zq1dRI zwRYbqmE=V_Wg#E}uj+ilcw&zRs*yd!Kt|DVY-#lV<YldA$7CbYNn-)-j&wk+U{{FW zv@<PTmslPA8zoa*zmQOfo5?XBYo}+P;8aANTLO?$zAWhh;BF7V1H%c!5P_~lVIkVX zT@^kcje_W=WO_mtpBIm25VveQhPetHS3UAR2|x^?g7kDyACf=#(j`hJG>hO$Krn?X z0CnD+Q+nOr7Bl?{e%{TU6h6LDH_?>Z9)B%kFmEFBPu3_#jTPDumo%6m#ZW0IlSQu( z07NK{0zN!Ix$-GooTr#`goSIr!Hm=$z-?JBFl1xC58P&OjPrXI$3<gl6GoN=@SB@1 zfCSn-!dNIvB7pc11zHg2f+&DMP5kla(aOr&nkH}I*HVI$yL61*1qs2^!^Ew^Nqy;^ zel@ZQ=FS;-AO|Q6JzQW7ythSg<jlNmOUS+0yzo4{x4IPIcIP41(i90~YC@p`88-wl zJ2b$YV0s>R%mm0hvn$9mep*`gUr>0t>0fq7y{kLB#K%*(oV?uLUSu4ObQ~T>54ESO zr_3e0%i%jCr}ygj-18-|ivbV_U^0N9B^(eZzoTBF*1%Xte-TuMX^g!oQ0qo$F?4L* z5opH*92&DqdWuthQd|yU%36VCSpnNS?PrQp^_K<LE)qINC6B4g|5gq!)s*0dLfb3K zb=d8z-G|p{k%v8dd=s33^6WJPX}J<~3_2SW*bwL-%i<DBEd`sG@;l@?L+pyPLjy?+ z#+v`5&$ued<TnO@00v8qA@)h2DAZG)15g$6TblW=9|C8iVDSh8yj=rJuFJw%&|mc- zO#SO~SKLe<^-54Adw?k#U|x+VRl}1Kf?)ws2<f_11A?J~M+8$6!8Q#JP;<cS1~{W} zbPJGc@mR4Oz-)RT(|B348FHT8GDrgj_<D7hYIu%tk4gi8)Ilpem&TZtP<z4fMM_5G zdA}1Djq!Y^>18`SkQ|-BEiy|wN3th4OK?Dd@&iF<fBJ(oBx886nqx}^a7s;J6RT?g zWd>vD(nVIPCfYC_T;Vz*IJKSt9M&7}n!-Tu-U3QPDr+#U>W<pii*f8~FT2MC@;$aA zg|Fs5fv$jrgkp*GA#?+Hgzc`z4cL8I+!aMsshH8Gy(einZpa4Ku3ul1x9ugyLw5u= zLS<Vr$|ds@_=UV0j^92PZHrR1oI0N0k<rcBs~)$3(c9lAYv6LWJnz?SIhPccqYl`% z+Fc&6g9Qe*tP@}7^~cS0zBh7&Ue-nSeA{l9k(Rby?e?#Fghuy8Db+r=YfF~Ctxr=4 z>Q_nZY(95%UaM}Wf#oUNZ62>?d++4FYhHU<(Z{~Ew_nxl*`IFb%W~FVAN`Z)Tib1S zLygbd#(v+=;pDVlw1lBiM=#UOFLK_0Cb6@ENf+BwL1Te@<<6jjjowL;Dk`znqdm$S zSmihy0D>h8BO4#z91^Et>Uqv&`)$~nFB)-b1!Mr`FZvZa;3BUf?wz)<7}Xjk`EH1p zq>$f~=Ko3@k?dfZ2dn@__K53f!Dk5PvRXA&$b2-I1DSz&n`-1IPfe?)L`yek@nANn z##HC`$iwAlIq4u|x`(I4w)?G1So$%AwSU2Mm?@b3bqZ$AS0E5@am4;3^#{6t12Ta< zB1E?Zi$0VhE9sUn0ov!)fNL|Uz)38&T{yWHBph(RD`C`rlSXF~rVNi6d^5&DJsmZe zSsz%z!x*!))Z)&Z!pjUeqF=9-^^>_aM2u<cT5D0?u!Hdz)VFlN#UCdFKXShz=5m_L z)tXs)e~cx}WC*R2VzVL&DXsjy(fsN_c^MKH+EJ40I7-8(3FBmffJXCM!>A2kge8G4 zn@Z&&yco1&Iv=xNQ^#HGY~L$xmG*y`o;zK<eSDoOEY!d#sbtJkW*Vu*IBL0^-L5<_ z`B&&I@m<P?KWF}8=fJKnzI8D(Z;#h3fy!mr^HC~Gt}RCBj4!W921Ka;_*sF1>HGIg zT$`AgmL1XpmqFBv@``gY-@i!u!cyItkIRc(L=TVKRlrRc(nW1nIwOK4%NjSn;!3`4 zscQ&nY_yuIr22y>Y_OLN%C5AvKq&Uj;H`b8`tTL^2SEgvb}SAK4z(zKe+|9ZSQOi{ zi!rjf+I!l(Ej-t0j4_c7f#+{ShDJ&l!oSmyY{7@y>@G>BJ6R{<6_M0O_U^}Ky#9u1 zP{9EV5K=Ip)M(zx+G(&pqR<A~&5{6A%MpPti66uu$M66=^=$11mDG+dV0FbLfaQxR z-?guevCD|j=kz6?q7HptdY+VTSHU;dBx2K)os`CBkF#OCI^7p2K%6LM9Jzd5*?heH zOQ2)ThE?i4W{8F)GJ74p_>3PCj|#agYPO);#BGm*L?b*lL@pj`DSaUpLQ!5VBBpT5 ziK0BzW)B^jVxnNnDLK4n!5hnDn50w(#-A>;Cc$j(Of(}U4h)PN?DU9`>W~dQ&cn=1 z)@lak0c9yJ1FDar;R4MvksDC_he}l|GGBKF*Xa)9kywG-XfxHVKm+?>k<EI^u##9H zPgdl-CGyNPlTs9#o1;`r)LFm+LhD9n<u<>9Jsw)cZ3GN1u6e9{765=t2aSGj3b|5I z%%jG8;Y5v^2xWY>I%rig5wk@IP{?>`k*&^^-nVkh!+Ewi|EYE8BN12xrW#6S`7ygs zkeuW<1d*eSq0bl)K`OT*>lHi*1zAXd9?9`DzyXh+njEn{ab+o!i-A=uTGWDAnGLc5 zppE*R%)~10grQ!7Vyc9vjl(Yvnr`=8@X1{E6Q>JN^Y%Q|1F$21Sq;#JF_}mUZ0^(< zH5o4Xc8H`#HdHxBuI?GKD-xK%W)wZ^5y}PTC&V(FYic|SQGXzY)&g@Xwx`s+6Zcca zdW+$wh7ot^Z&7RQ(>B6dU&qHitiB7NpiX7T(rtCVrJt`4Y#&)$)=Y~{U<=!pyT&r! zHVEo8j1abhRgTA+U)3yAU|;VCMjK(Vl45mns|(d%iNOR<diSf#m7bo#al_%8Hb&d3 zvaMpvQpA9_=jRAa+87t+3XVF_!n{4SlVIwnAzVTuPI#*gcE_e=I@&?nmFyX)Q7n*? z#|C?#XJ3pjatWf7k{W+M51t^%*p&(n2nB4Vn;B*BHYv2j7QjjY8N-rCeKNopAa2AY zbJ9*4Q58}Ds`J^qm`mL1wTWoNi~@mdVd&H*T*DcTZHuD5=ZIRYr#2$As-R(fwVfN{ z-%D{*e^KdT>aX|<2y3Me#UG5km<ry<xe=dTtFQR_5+?8=HJ-52ku%V}jIJ(S)GMUR zt!vlE;oc)qgx7TP`vx+FgUx0%hme{Cb4Ufe>?a@zyf2Jd{qbx%UWS&aZEH;Q?~~x! za=r=I!1Z38?wk2G-k#x`vOgZq4ofPbVh{mc#yU}rdixmO4AMo&xg=86GFeq%uVA}= zhjoJn4s3PK%_!RzDWr-Gb2~k#VD9D<t2A8V%6PapWT$A+t*A5K&ooc(#yeRlKE2Eq z3L=@Xk2;VVav^ybdBGSvC4&FPg_?&d&ld<PZo-mqD=`H^sW$;+8gmUC6qthcGWQSu z3hLpu5FlG}gI9f#Txct;l;(#!ub3sXhwll$PqV2(Dss5oAGW>21`JD$%d>Y6iYZ!Y ziY*_=uInVTY(R9X=Aa9N4D)x)+w{CR*FY(*OcL_GP)+>WAK)O|{Z$}EG=7;#N6@qH z{wJ{$bxqg=*bz%h0^Zn)zhMsS#+){q%PR;NnUV@!GoQ{>!17P=O%e0LL?byozK~%z zByK9@3-GT`p(5}RUTlKtSWLmZXyiAJk`M^+Ed&-aVBHJAzr(k=*c;w*u<mSBC@ey5 zWDOXVJq<$M>2y5;EjXw<X0$wCe-$P{8RdB>w036yD&>i&`mXUO32|vuf!A}iHMY$A z$BFEU<hI!$ej!z~3Sfm76m!1;*pN!>ivfxlfZxEq<FJV(Ii9h_^P&Z!W)IV9KycG@ zd5R%QxqX%NDn9z8(}a}r?keEMP~ko}{`&5(ZR)e?shM}fY?vvvW>c^NoQ8;Xx=S*y z=@68>a*JyjXp45gOMsiO{&i7PM#lCGK}o6}ps_#C>&oUpfn97ZAu;Pw&TP(1xEKi4 z44tc*3-jRp=eCuu9#$2moK0T6F5k08)^jMmF>)8YnO!Qk*Ly&Bu}au+J1=*(RQ2+4 z6JEepCmN=!H-18cy3^2Up$ViN90abp(Q3fBW>CoNfD)4->5xMYlLkr7-R<q9c{ofb zGs>XoG5<k2Cnj+ijeB%&<!+2(%8d=JGEf^R*PpYgo9@Ipv0WO9Q?b^1(OE3OvvegR z4#PkPyxcy_tk0@6%v0DAk{$mB5?PTI?<K;7x+cVF9efNw-a2nlkXk}ZfE+rBfO{4a zrQaSoTQS%{{?aMbnAIGov66YgB`eNdwL!)G0?GVuJ~%OJ37d~TflM%{WzSK`3M(N# zmkGTFWlEU~q&*5w)B4hZis&DMY?Uk)PvFhrTTDN?UMTQhaATMH`Zfc?4-Vd=x0DIs zdE4+nU;7U3C7>;KwvqiQJm)D6G^@_LtHp78n~ukYrExje0w>P3kHg`N?vKr*9ABT$ zP<pG5mj^}!o~A8NoBcq!YVU`mrRtovukq*V?=O{Kl`Y@BrwR4*CG?rC-#Ib(S&L%a z#Et>7vSI#Hu`Ge2lcPN7wD@}0p7k{fwnO$ju1`ld0@uy+HPP?0Y$8)3;w!2t1E#?a zyA<Q9vXiHDG|krfipf0Hq7xI@e4ZaGpiWYp*8<~C^xene=^;Y&BjEd4^|3DcdeLi# zMY-WaI_tozboga*vQ~QD?OsT3t~7SbfI2<Gc!_B;ruwddv?{co`gHCAAC1|OJSxA0 zDa;u0Bap>)JR%|=`f!#Yr0;GKRF?<~B)!9tF9Yn!O9E-@af&nsO>bT%=#mWLRZf&w znj@Y1yHs|x7w}s26(tkI{KQlhZ2O(NqPWQ}#`?+;p5#w!l#Kj2U_CLp8VRp*jnCsM zrmXX#;u{t*v+Xxr$={gra69Lq+aXHK@OJPnh26U1s{2*_JiQ`$3mSwG9#!{QPIrAu zQ0;;ShWjeoa8R)#|JLd7=Y~qJ=g;+vm>nZb@Yo>pHkv_gv9-th<<7y~|6$vdev@qE z%3X;!?~KLbjC?OxX|j;UY1DV2SeAuNzorXl6W5LUDg2yE&`NpRuWX}Dh9QoPF(V=M z$kwQo!~rGXq|SX-D&pfyNPh_!_RQ<=R<DLI0;n&7#D*2~o@f>H5(gXqL1w6Z9-3)R zB@wSxr?`iKycK;ml5zz;LH&YU>rY3DR_pWOJ5zxK@#n;XJ4qEx*DpL1<0Z+S6o$Yh z?3TpIV40r*1b;4Oip0{IA^nydFqnZf*hXabU-8SX1IN$@6;&S2rneK5Z!5Igog@SC zS~=Y-!T>{>w-*O67vS$h6^Gx|%8d=(+^O|W;kUvsJP{55JC_YG+G#Fh6>H%?^FpPq ztH7WENeOw9Sqc^`d-~Oj=?Pnc6Zvq6COfFTS8_j2A*8fM*bagFDhS>4JyTP?YK)r% zyDBuEUK(MfM)HhmwS0!8;kAEZ74a3>lSIN&j_|DGWG#yYd<cHrB`D@|@~i^Qr63ob z3&)pypF}y*j?TO@{`CpEg7u{xb-p#gQ>dH;Mt_=<LPK5Upu+gH%jqBSnt{z0b56n8 zK7vU2JMd2!-hym5i*;)7St++Qv$3C-G8*$yopFyv>UuA*Az3P_C6ab&v(1%_-4~D0 zn!DulT7|L<b<RAhnPMUSyts36eoxY!LG6q^Bi3B&-_DoA;nb5P2JZSLB@4&}>bXDu zcq8-#5c4h~d&!82fNk&l&25Md^UQYv0`uh{sUHktWSO?Zq#|dA@jIhWMfz}{;15?Z zaB+C-*G#f=<Ww*%Wc>bZ#FlA>L;Vp2Vj9#JQqYO|g(rO?QL;&5WcYEXKqY0m)%!HP z_>-^tQyNYk^?Ble@HSD#S7}o4K`M)*J2m~^89+*O>sz3s8w9Ng?Ile$IAw(ofhWdg z>WPGr`Ct-%KYK1j0XfyHHA&Rec|e1b&_NI8w(5S&V(rio9>+FoSLkv#`2aY5+J=SJ zV71%(=rrv(XXv0)1fS{<c`DuBM<=(td_Et^t2#gX&*eU>_V06aJl{_%mwo)d#_)Ye z6n}jnQoJ<4MKVkrhF>oIm_((V(y<YeP<9&IrKt=s-BdC<BG7kt-d<F*>AoRr5uiks z=V*%p!(?~!yhmEfeYovEx;E>2`rf-v*AVWVkzZSX4QFIj^EJP2X<_4ky4;7m&Jezf z@ol-i@pVpS{$lpTxxIU9KO%yv^1169@~kGZD*YUfWzT+DT$I~h@p=rbT>h$i-NVPo z5^*B~frqY+J3Se~Y&E22BSzOwNsc_26Z5#lv|u29zg5gMbq7#CCn`z&K+_<hBB>EL z@M0oxn})S>lww)p7WN;-2fUrXOpl${s;ks8u+gq#mBRWh{(FG7yIwm#-B9s3r_r$g zwj*?l{?as~Gw#z-O{*?kv+Zrpg-xEdN2_~xCCJ*AP-JhD^>nMlZWb~Wjd6<sY*0eu z2vrc^tx~8sT7u^ff{Y)4fj*LxpekAhTNCG^pF=xYXom`H#za$!eGqQ3jcVzielAGV zt_$nkOQ+mQf&vkjxAu~l$PB4UugxZn)4=P9qPXk9e6%wFBwYugy_)y%qY8?`adXo) z6(x^Sb<|+n&I9xs;W+&!#;{#6jCLzwtmEgTxbUa2ocpj@Eh)?UXg=-to#za`Z@cp` zr5bdbj~6j5-&b5KLq?j=#|&+1<>v`SQse#=ixte3Be*dN8c4~?$T2)i!028oWrG=! zwnnu4qA-U~ozuKBxx#?HzB=%|{{dh?pTFr`yj!z^C_maLOvi7Zy8guPwm!PDb>*GS z=l-_w<3Dfx`N|$;bh<HJOdgtCLPOxAj|Q8UUk1n}MHa&>I!eVh-ncA^*^{~w0g+}E zf>qPytu2`{&q|{q2{n*lS<z{3B-9gha>a=!bTRazl<*6c9gxC0aiXH_Rch+dm`S&t zt!@=4&(3z4G6E@XaZWGhu++R9@L4;#l(9NMOe2p3WeqwTK$nh^RoB4KmsHtVA$eSr zf@hB{l*y9P4OeVe1@R8A=IywBaz2c1L}6t_OaHJ1@)u+J9W`2-Qe6NdUGAY40A(9x z5IgSmF2avW7ea*B7CfdXbU&_-3oelny|BhumpvI@j4GIK??_gP<{~z$H*XiBnV!tG zZ1V70j^(*|kvSPq1ixfpY+WBIa4y<HK*wUeiqCw%1UU*cC5w-g@w7!(3Y=^47FaMJ zMPX~@&PD}b)u37}5an$2Vm69ftwo<5r-)@JE|2`kx7qlKc(M)(Er3SJirE{x39Su7 zaRdMR^=}KRf4kF#ci8M&i){KAovC3>(q>}%>e*w>e){?$<X^1p+gY(^yES!sJI5L_ z{9EXt(L1}pK7xJqcN-r*bu)DOWeHSk#0;sxFT?(Apjdz^sTIF!N<9_!z2-|UxD%lo z6)nx&I@<Y?B+5YR{XvH7W_xlikqKQD+(#F;nGBe{0WCl?$^4h15I)=2M1l}+A*nEU zNiE`+CK&cMD5DwCw_>-pC!Q}!BG?oPPA+wD7j|T6#u%_h34t)<xc0YaHm|((_sBTd zzvOo8cI?a|H#_0lqW3yVC?g2btG@$2LK6Q@{G<HDjTsP^-+9+Du&^!HVWPVJ=BJ>L zZ$9_Me=l9_^UzeUeG4h-OU&R+&7P)1DCQMqzRVV&F^u?faV$D%V^~#j5xt0r#$QD& zvxnL;s(yyo7-Is`$sU%~f(c2KQo=79(VW~m#@!lfY+@s*inc<GGf~!(zC5J;kZ&FS zc0?53I2aP9ZWKX{n?(=@u=}d910+O-CmSzP+b&TdE4Hml)4%QLBTD&vdcNXb%XtPh z&D(XMjZ&TycjznbDdX=;Aei&Naty1pcf04>D>QiMp5ULRXM@GTGAeEyb|Pqeb97?7 zuF9JC3^3YUeK|>%JA8quKel?$Se#{pC>v`>c^G#N$L&KC6W?qQ!>3q`oSxP7iWN3I zbAqP|Q}0Ho#A_LQ_cAqzU`bZws)C8;HY3+Q+xODOUf18KgjnXk;;w6>R9U}G-?zV$ zPNR>;F8PF@2+i1aa4jlTM{oUEML@#5RbHF|c^<POcZnuk96Y=dk-u!5;q|pv-Wdj` z!_)QPbdty9iv+(aBjHF@bUvf={`!|dO!NZ;=`8SlL6mjkUOw86Gj#AC?ZGZT+QGQ5 zy<`*vjE!P6P(0U1U<NW7i+knKhdaJ$fs%%F17??J7<12p(5Uh4zqJ@7{6_|DG~}W$ zK_GF;ezn2LWX*K>t=|wK8SVuq$ZcAo%~qL#bfeXXag-HLZJom+v77HsELFwvz(^d= zmY-wkvgHD4q2CkS@o{mg)^N$6b7kWi?qQLmQ<?(8R9RD#PE+FV#FuZfv}~hLcH&F& z6`>P_X=)|&BEQ79LZ|gnr;`nkjm+(}Zo02gIH#6k4e?4LMjAq6P18YP$2yJ6pc6kD z1+9#)j*627R1B8l<v}U~9Kr&xFY{^%CWAxw1#^?NK>Wv=T`+e56oYxn!*8{Ke(StR zGkcOsOwN07!e;sUz5MGwVoo0H=mIX9dU8}pGcBoFilkt+oNlsYpL9OAiqW#If1EfR zxTtwGL!Mm6x;4szMkXxi7|2Roy$v2WZq<6*pBta2hUXgMum0`yMcMRatCMc4dD6EU zgddAZo+sViwLDSQvR<oSazUw6J|IF4e{E&+Nz<hOBU|xRYistyECaNH4616^WKr9g zpz<vQ<sg!-(P=MT->b-~x5kny`woAFlQ7<uS&Z;{beAGN8e}dDbDnl8wuXzD709yh zde9|`;^rVA+qal3s*!g>^0bbS%fhc3z+}<N4z@O<xC-0!eT*15IC1>MiSMl+8K2+( zxbyo?3hOlDBA<`lg)i!NB8;#!a^XVMx}Xv-TtG=lPSQ+-;g;on;4ssb1@_}*;2C&t zDGlk$HAYomi979--DUXB5ULl#H`V6KQGUd5jWrLW&=_QPbY5T8wKv4L1b|^wvfva} zm~pCa5u7{<K*QM?48TD9L$HwOGqHubJE8}|Y~>Um4OxZ{Qv5M+k6^2Yg&X(^PK15Y zNkqkAn}!n4R0Vm<4`j75C_09Y@rJ(=-tub;B@RK<`LNjz>xmhBX%WAv<8?9=$~%BS z&tgP|%i3Ww1>&h~Wi`&?+{kJM(#~ie>T)T=0YqYpGFXtIvqD>DBa|JM{j9?{t)fjU zhfW$Tha*YyVgQy)ifG1|`lcMHzAJo;ufIV(3U0Ts5@rhm^K?y*^m4k)hg44&>Oj(_ zVp=1wJd6R*Wgv$L$}AucSs=IF1$F;$0oN9K@n&>4Q+b&wI24fK_{?1a&V>MPU3m{G zE&`>_B?-WHpeyi8`RsFOMS+vid9y-GuH%^K=222r2p<>-G48D<9O)UjLP&_J1q-HV zx)4z0dU^+<+r>a97Ced{5sUDOtuTw4!2(}`=td9027Cw)Vu=7iQ>y_+dXQQHT3aaK zq2dhmkV7td;#Cw4@+L$|me)>&{v%>qtWcZ_q8ga7G(sJ+Mt3|FC}UM~xRr>m^x~+X z6R@_|Rub~r?x#Qq>uS|vqbA0Ec6~v?n8sQl(&IS!q_!5(0||}QW<~Qnq@Z^O%Y!g& zkHYXq0pC*9q~+@P3@AJneOI?7H9CE8W`WnT|Ey1UG3u4~@KRyHsz)`%T7}KoysDZ- zy^MB)y4)x*ybG}4?bcE|>O^_utQCXNOj&ypyMOxj0x3p-kYr{Tb~UzO^Ij3wf<<dH zf!dTKeCtBfF|DH9f%XTB&P705jb?Q@)XKA?Gs{-^ieIQTVMBCJmX<Q|2McZ4omw(l zS=>b{&|wCL%`Fz1)0m_35@VL~lJtecq7-!=O&^3Mj3He`cma%*{V3gB;e=`h`tlFz zYJ5}Qk_WQ-2a;ZzEX3e%5@MPx%c$Q+I-RDv9f7Ir&fgSn{g@(M)|$$Bji4J&!cEI% z7<0AHaMN&u#vI{Z+mxQXoW}m2P2~wn{X?g|Cj0hLDeb}5G~L;#*Tivn+orl<zlqEh zzq7kQQ@Au>3ZEA@#c4LD;F$4W{Iy8G7R4_*4QleCkkJp-2TqgPc+)y?;5@czHkE;& zQ+W=XhWVV+Eqia%H2`xNZCGtuW>(HUmUNqJ$RymL;jYQ{JH`haVVcSuW2R2%<Owr< zQm0R1I@u(eY?{E|1?9V0)W@+Lt7%Q8u3;Lg5lzx)2(l$tHyt^fM`&cncAmw(nl>|| ze9fDhCZpmpB>g6>L|C6J@##k{%y1#ba~Jem+_@G?lvVPOFS^oUuEsx_R)r~3yx(~C zwT&y6uYU5x)ldGk@$$c1f955xEopB3(74Z{0y#sKBelziBIL9a#Y0cRRAs#u3@v_* zQ3{w<tD#oZIWVe@>t2;}o%pB41hMeYTX<M349(7sBQsr}(g_Vr7}`fTknU<l8*ja^ z@$AQpFn$2VLXN~hAmdULGFEqqvZN2n&>#nKQmcNl->dqjnCkkf4o<pF?|=N$>%V@w zbkc9QlgR(^Xm0nj`sK%P?foydK7XHkzxC^<u72_s4CeZqpYCoRB=v^g)J6ZegRLw7 zy7}AJJz6?XktK+z5r*#Y7I&T>9@T4~Z2aVRH%eEHlJtp!HLKkgl!eCe2@iQS@b7-e z^UYBM|5D~|2#yEz*&jDw`C{Yc7jMJJ{4;A!Q!DI5M|$le_PAl%b{IMUFU+Aee51f_ z=Co|J-GVvI28+9y)6wnbbo7Si^!m^LwsHCGU9k@_Gj5+j@-;YzD10i+SL)dh(j1M= zGd}>uCW1mj4-iIi@+s%@EKqJ9$yPuW685$xo)4CFcH`4OZ~mf$PwcwOs!#5D5);g- zpdY3V!O*O(GXNd{f1fIL`0A&xZvOe_w~bu7gU)WqW#4w>vhT*o<yUXr&H=UK)!hte zY`X!C-55`M@>7_ko8wqJP+2?4E6zA1k(twiWai97Det{GA;mp*^m0~#_gawo1r-34 z`8IkZeJ5OReEu_#SvH>k#mx-EdyMCbR`eVquSr<ih6LIsDSO<$2_uIm<w|R8j<D&T zS=hK43)(&yt4xn=8`O3((VJWE|5B{9U;gFVUw^ap_LbX1n=u=}O9)i6Z#}Atwf9t^ z5hq0#-A`G1FOr3g6fSIR2-qV)ne3Q{nKC-I<^Hz@DROV{qUP43!5+WbQ|z8txY+#Y z$!mXkV>_w;2G&uuj0#<uelqG4?niee-EY41vyCS{W1${c?Clj53$Z*pQQEg<#JpTa z%xP98#1vv;)xWXmq76~BF<;#g6I*Zm@%pR(x$)PxVVk-6St}-Y3;F;a-_6RvnTnlv zE#|mxt8ADwv^6$`pDr`~br^SdZUVm!>Fa?4NpyJ|;F6QN<iTCg$wyBzkqA}%^(Qty zd<iJW9CJC6N=asWsVVMlFJut+!%{>IHL<SUQJbY2TxvJ;;2H5-@7(SYhwHl;F_hZH zh`H1*M$CL|H4^v9w>MF<ZAn_=*bLVv`&)nLnyT3(NN0BN519<x?ItS3l&<x}rl~Yi z-e9qbN)~<x8KH@Sg7^+1eN%|`B88c6Qw#XIq>-h5kx_Yow(<d)$w8KFqCiXElN`g1 z=Bj_9XO>yiFo$w|-UFIyH;}2EVwREpv46`FX0Fa*=HC>X0FN#0H;H)?PAz`YbodLK z-^uviV{`gCn$y>{S#&IzJ#*|#a4Zd%m+5#54^5WSWYE`fQH)&lALTeI!P-jPTEUY? ze0BsA(uAKPM@I|7n?c(O0(-yh(M|zLuQJk2@PrlP?(-w!^$Tq%Pfs&*f+SzN2A-H% zHMT1}=vzgP_u6)880<nf$p#U>>OcqMqM~{m9gi^M-hjML=#jfMbYn$u%joN@0HFi? z91a;58s7S!rvS4Lg!qJ2T7M*4aNZZ{i3jY(BMZ7mvabu=6q~RCsC-p|A6dt<h)llr zN6~$w^*%<k6&|t-OyNU8DaVLi3G$Qvtc<!~zt5%tAP(F@PBGJ%tP-WsnAnjb6xUN? zPtjLGm95EaO*iH-s_h>@Vw~IEsBQ&_5dAxW@5l4I11ZIYY^#MJo+`j)oUzZ2_`tG& z?26L@#^1A(>Izoj0K3pTrzWpA|8X^`QC4yP`FL^0?1ZW9x>T1xgTh;0x`A~<D<7tQ z*IQy}lsK=5#VP~Lu#fK(%qB8?6K$l<5NknHtpy{P$WQJ-LY-b2@x>&tspSR<#fQA9 z9no)(Op17A*pc#HIc}(H2x1*XT!C&6kTyaYsv^VEd2%c3`56}4E|5aJlvmp8HE0dz z!3XJvXVIQO&%jK6L`~q4?*n*hP6uB#r<c{@@|K`FVU`~&fHOFr7d2J~tg?3LB0Z3k zjIQCmL`?3#cE5mu7;Pd9yMPzYjuT+RFeacAg|xSNmXZ8i^7p%ayZhYs^_Qaa-cWx% z!9Q%qaGX2|ZUJw&A?94pXw4`*qSyMz%`LO2*DhOW<V)<3dWol1Nvm#c)o#)+<<HK5 zO2rDdlwUd_fR0m?=$q=1V>fz?bDAoyX)ZxG1G+_sFM^1a!lhk$h9@|MelD!^IZxt{ z2MoWov@{TY<>4S-0eSg5_;OEB`4;^h-Tdt@H~#z~dZpoOOP#p41RBh@mb)Q_6->HS zswzMIILktykczx=xwf`eH=cJA`BkBAs#i>g@<9>j5fk;Qh&wPj(a<GK75(qT7^Tqm zo%aoNCkN*_qKC6-f=@Xm={&`n6&~Mq7KNQfCsoioSp+xnv0xXjudlD8{NCxv=QWBq z%AKa`O2PYIgD4%&o6xClQ@Zv)E<1lNTnm1Hh3jTvrm13Sb!(?>R@%8=?B+Btrb9m2 zp{AuB2**{`0a(Owt}YHPps2P`FCWe&Ml^aM?PAQ-b?<P^n}@p?R=5PCnq|#rJ~&EK z6;NUj(#s&EHD%T$?xz1^OzM&htyrOU7j$Gf;wN{iCJkqGk>!x$A%`}nB*T%d?m~@` zMb-Ah%ML&L_{Bc%4r!tE2qWJ&5zp}@XmgMkDp7fr-n)T@;$C0jHwzIqf5WG5>LsPs zgwN58&toJzzIUVE)ngz`o<%Tv=H)<i!y=_vNj9n@V^F?uw<Rl+Gn+N0$eM)dcU6`K zu%3w2uzgiKy7ndS4Bd}PVptf-{|9hyMt-re{}myhM4s)jFYiy|bDMIAXp>q@2EFoy zpl5^opz}Sqr_G_q*$k-ny}W>cf^>LcMM;E;pG*}{vG>9Xv%`6cW3PVMj?Dg{KiFGj z(xWOn6P|3SjP%53dMW57VjEou;$9z3(szI+dIz!Nj(BD;+m6`g#E9p&BR9J}k-c4u zco&@BU**2lU9wzS^3bJ#yW9D(CE&fqxy8>m+k`W};!=}Lz2r9F>z&L~#NgGFgbV&j zK9DUukA$DXg?0E-pfJ9!zWpmZR&VCw`T7cJb7h@3Kcb5q5X*$&Q_%#9c9Eo)%^}%@ z1MiYg71z~3ly_0#aXyeAaq&XcD7NMK>U%J08$ecvi{#_y0mp>!w|g!?qDTn~5DQ7j z7Up6qghVb|Li9qCIV>z#5SqNgRAKd9SSd_~5xz;?iuV|GAn&A%&X8$dS|kWj@IL33 zwfv~2gtnvtIGG`$yb~yU!q>_#tk?`)etdUf(ABPV#ji}`yBoBOW8ad;G1*x4uQJqR zeab~?X==DmJ9RNMgHv{|7c|jO%>?9mr>2|*?cZO*Fyx_(-{ezF92*Pped8`hO$gho z7;l8GH7<s&i;GFm^K?zN7Sj{COf02c#EXGJqv_AY4-b8XiHLvT0|F)E!Vf(dUN&xQ zZsLZU<KSe*k?>8mJ9+c%iZ2s4-2riXX`3Fv6t9G<5#Q*{fi(pa!Ey{lshP&k-RD3Y zPXBMF&t58|&7WWc$(526@QtibK|^8>!;v507^9=Rv(T13;zwQbF0pORrp9Jxv-l!^ zF|~Xf<o0EG>zTS0n<L)_R@wz0VB>3V{iwwcdWJa(m}sVE+d`D@My1>Rp87EKKwYT& z|Izk(;KsMtnh8A^_3pQB$N1e7;xA6+;=^?RvpfIK?reUnZR;!Bm%sWQ`TxCp@&D|? zo}YhDyRbBV`z`^{SxRLk$)v&0@^~$5VOY?&g%OHi)+}CE?7U|<c2BIyW?-%@->Jb( z`d`hWtSL8qLegVykGB0CZJlv;rq{CHE+hd!LhzZXjDSyK2<6e9WQ-1FwlSzo8cq+X zfw-7)PgP~>du3I$UKWjI0zGmJE(8VLMzltnLb=hR-z;Vtgp@H%HRRjLJ(e&}yhg|D zb!Z#2mw&JZv8e7~Pu5b3dSt*@X)3bNiFDGzv{7@Wr74KXnB#oCsU{j`L?Kca(Qn-u zDKf^;V6hWtD;&0DDGFgY{FWV_BE}iTAVNPyiZPT2-SA?RQ6Li~?P4Ebk2#W`4AB8r zntUHk<r&vel`Bv(W#R<oZVIO8F;t=uH)R)nm{F*wNY$4FP*KHzAIR@@-9;;&lx|UQ zdj+nB)>E7?E69{CPsUBkSHme1gXJoK%0k##PSQB1XH+{;nBp-`N>Tjwsqo25Y1q4n z(Pu(B%byr;92gt_=Ggc>V;GJV=BBqi+RC!gZU_p-`eb`Fm#)9r7y&xcYoniOH*7}~ zt|&-Yx?)dV3b{;GF-Q_dPD^@xz<P#GRWc~rL4m&yb04-=qO3`|ceGZUO?=1~s(|u@ zF&mW6@g23`j$;&uC^#ASU|fO9qDJqK;mY`U_T=da>9pQSmig|w^`<6lLfyC*_VP-a ztbt;H7=eG2HQ^r&-%f?uMP#KN{sMK5VrMO8+QJl$H79BW`-73u5%^bBG^h(^taA?9 z;lc$Khx~X%BW{o(*^c+2Z&_(hmMU;61f5dLz-by53|OesMhq&_R7KZ=?jWOxZ5;}# zve0NW#^L|QLLJ{a$8%%!kE0qCxerfL7RDEhWlJbZI)qa>A1R)pRn*i}DvjXYdlhb~ zq>}~T2>!R|L^T-0*)-)N`Kkrra8AzH{G4p3FnEfoHCQ|+=10%e;{mrVVN|Ew3bNy} z1Wc=B7s4Y~!up1QWz@6h%E(Z%nA63s;m54@mPh!M1x9x=0TwgeW^ZRm*GPk3&L$%e zKO3Tt7j^6r1A+W%hRFYhobKY|71*Wyhk}aP_&<WAeK;6x0IO=?r8Z|49+|g62oeJl ztpEm25m%_dcpj;GG{Jo(3lnLgW@4~rOsHC-FrPdmW34(uuQ3oyAa=LX;)((^0ybPm z%t;mxeh}k=GCx~bSa^gk+;U7p$*)KxS{%fk_N-2-?2!&Aa}7YxfdilujGJF3@t+2N znXl>62O;So7MjF-{Bi&b$zKx_BEk!yVxJM+gXNihR=+UfxY4J%Mgu|N%hH2f0DwX5 zk$zenP8>LcJDLTG*qFmsZfIzY@ovWNA;N|MJ4nhg23KPdaC?vG*yBd~&C-^9NO1}N z1pOqYCpG5QVW30!e;v9yg#Xtk;4l2YJ_&!FNWw1AHiiPnf8w6pxo#MA;kghCq0>vj zQW%5GX~b9^Q}KQZAvOzC0mH6Q53OCNdj#iClkXf9qq|%97xnK3$rk+IC{`DjxJ5B1 zsWCrq$Z~V4rUxq5Y!rbtN`PU~gn%*W?b(E^IWb>In%t&m4pgo=Sp+s&0t}N52p9*w zJv$(49+)pA9o(j84pgprun6p62{25$OTf6>+q1i5&Aa9cNq2A4GY2Zyyt@eOZUY#@ znzux1$sYqJ?rNJo*9|7#JyHwqG5YUvW+C)=PRU`{KoJ8~2-FFzWkuJP@G=!St_f#N zT@T%M@|84-q;E$zOfRBKMM$wYI~|fB4LQYqDT=?~RVfl+aTrzFow9tPCR7#R5DQ`Z z5z3+sP8Kj;7+GC06-S@Gcu!G5w6}iV>F+6S@zll&Y`8LJ7DdMtDS%7Y!cG~un^!O# z;UiUhC*Q@c9q($_#tFZ>hC{>|g$58lZbm4C#NzKG1|fJ_Hn3UtNP)Et>mv`k=?a8< z7J2x0_1E(S>8OvApel(w;|i{!{kjUL-92vzefUuqXDd$aaZhayG_-=YP#9MKq64Sl zgEQ%Yb3nj3Xv0y?;^dP0(DBo=7iNw;bo|(bv**s74JiKg1yGYQgVHhlHNQq5c3Ni! z`JD|(pN0}C*I95!2W2H#FiG0JgAL{jQRf8wUBGLF@SR=Oh4-&jyeSk3_)gD7i>R^@ zJ7Qtog*rQ9r|~if1+?j+&Kl-QwjsqL$=EXyR`MO4XbJRxSQ0r#;v1j3LtdC8a7u=l z8T9)}nlqO;Iy0kL2R48%j=Jmv;VOk&M<6gG3wS?4uI@QV_%6vl=jRuyHPm>KBk?t5 z3>|*#%qe+sm4G2f!kD254cpD~A?OHUCVP(XMCTDG^dD=08XVz`z}!UY0?TGefJn_n zbX{&rcm70zcQV6nzZ2Dj3YUOJz8H1zJrjXBKq-eUC-tl?o{y<7Chg&ZoD}J<xR2+d zhQ4#4SLe)XNoil;3g`v70PGazY+-DnAGX`{g)xkjLGij>`BzN<N#942DtM`)St$f= zoB~5IdnPQK@+zDC^Sls6*_VzUp5a9)G?;yof6XER7zoVczPe~^j+{rWeZWavkNd~b zwM})4r(NDg8EMC>BcdNft602F6JBp+*(@7HN8}_lgK#lRI)gl#3WyfaLLfNF{$OMZ zX+kD7IB7%!w@_1OdO3<yTA^k|H!hXR=hH&yj(V$h^k9BGC*$=3%L3(9c8ITu<sk0^ zEX-kiP<%ybeqm>69W{)(xc6NQxQ$7$AN0=4AL0_mAO}WJw;b=Ob>8w7#aJB?5#+l` zn@`pk2l669ut09T3-+yt2WC&56f<@yVAe&AFJR}WQ}Yz0TX85Zvn>#DCuml247<L# zVr4;)!AhXflJxVoGNInOkdDZjgJpvj9weaZKP>1ErR$8xRa5=Lu`>b+&sa$iR2w}D z9*Uziq>qZ;H8em8M)<Vr?afpri=afOWM?#FQYnm|Do&1XgIbL8wFo7_;Eq)RS?*TE zSnVxI>{3q2(v1q2=_tO6(n?aiMkPc^SzXbsNMA&Ag}B7@_qP>N574|<z9%yI-paL& z>zM^;jBU^Bvu3V$x<=GbqYTe$ZKRnGcfpiFF+F8pDbJt**^-dC;3djI)QmXJKkg^t zVzL^Q@_-w~i!*!+HAMpVFqf~ULB+J@L{o>TCF`ae-xh1KFj-i%AnthJ%c|UOS=X8o ztmpu%-(rqh%EL?^C<bo5x&vzpc-;MH^#)1vXeS07pTlzuJWTGb-r1<PlkSY%g2|UA z&Nx>3@eb6k7F3rk!~?TK(Ds}w^V5ZoiZ=sQbyF!MLz2*o58)dP7yyDsd5TjjI5uj_ zt!A4tMO9K)@MvFKuLO7jMWe?tMH}FXmm&;sgmATQIr^>Iz6Oz3VKmCfJJ0sH`t}&G z%x?;7lvfwO4U2|I@g3IKg_R~-0@E}f@cTwd|KvGL9l;%7q!x_8rm>QbxP@l#r$RYh ziyq;ZyhgMU$7hN-P&7s8(e6ePHc2$+_e-!UOT|usxV=u}h|vsD%%kpC;^4mf@RY$W z0<aAcd(I{UkHkGvbBf^8Jd1HB&WFulC2qH)UZDspHp{TH7P46R@#QQ6<wWV|THMBX zQpJ3Xp@XkpQsDkx(ku3is}^gnL^#RC4A5~zo#0@9-IptB`m&jp=4d8J<tV~uN}8hy z*`^3UX#}Hz0Vy2hNeM7fQUdgq95Eh2ov%LvjHNQNH|`>b414*MMQu^XQiz!3%mQL1 z@o~DH?b?y_+Vrx98{lvzyD(VT>9C77Q&B279E`wMf;SR0gApn@GUcr*Im4U-&RqCO zr#+Llh=XZPlF1aiAf<t9Gs9gn_$IAi1B}Ih&?aZ++%Q9-Lwv~`@D)=3{#bE&N{pZ< zQ15p#@I(}W){!MTc|(~8oY3SUa(qk{^sIqtn=9bQO@VFN!Z!R~CzGd{Iv2-uCv51; zL9?Q1+tipfNtN)!W{;CFx>`_3o=y6W`xw4eMb>%sl2;ygDb83n=%Td?4MBw?ySlX$ zDxLwG`?T)&VKZGPH-n13s2yr#>nne+c}f=OT(6s=#rSaX<V>TprW^7Yo15N4H(66d zF{&1yG_nOYN23M~g%3UBAb<)V1ZI^U1@)mK-WWZ4<n%*FW(rR@<KsTCkwuw!!5m+H zf-iY8Xsxi-DzjZ)=tY;4kkdie7wN&b7Wdl88eOb3-Wj0>tM=~Y%sT9K<Vkf>T#22M zAWxlnaOU_07A4Oe|2BXf`+ntclREQwpSj`V!ZQqHfx+V#S><sIoKk(f)rngd#~gce z(c1iNVJ=;GH2toKkN0SLenH-&zyq5!>V)!DMzZ21{ZU-$K*nrbv{h+Jvo+kx#IfKT z!Q`<j-OEtcE~V<CikCt0JX#)76ut<3zm?W5+>`c`8igYnK!^BsH!`0^^m-(#HvC4K zxKykyZTwamfr(%h@{u@qSOp%zRfaPxzh~&J{tdQY@)M>|UWA-lF#M@VFT+}(zYRJ< z|H;J5Ysq4`*cnnkF?wH%tbF%Jtq|#HjU#d_#c7t)8-6r;v~CF)q-v>vVq+?FSiKY> z4l?Yykr5B^e9pC-uThZ57R|B(Y0QCEx%DjXr<pV0izn$8wXuZ4>Q%@OQ3{3`29#`7 z?dj*X&B2Ji%2)cxdP0s+ROex|qo{b1!i;Q&&j_Lx3?=F0aldBFn<N_SZsk#s)k9iE zr~_7uwinkHVH`Nvx;L#~C=#3^M#0^&*@ay}y#NZR!!?Kole%7m{Oi1Wx{*n%Oy5^n z6B@r(vJ-(=Rl?Nh)#Rf41u#0a%*aRXKswIk^OP{5L(9HqKVX9b!)!tpPg2+-I<l$h zHqDr~xRnpcjrB3@C5rDNwp!GYWyxh8o;(^noTL{q=~x=#)+?3)A3XCyFepy4DFsph z5~@THwU^a-G!uiGN;aBN3>F?Gpny-M=t!rB)5q@}Gh|4eK8{X!ix*=;7f%xFAJkAS zI0M^EsPl+R15I!yj^IR`MkkVWQ0t;hK{k#9xby6Q2-}eH>?c~f(3Mm~g?c1xUF>vW z9wYK51rSj-MoVK<uDP+Qsy*EcY_hM}7>a%{02-nfc=d4VrX!=~4&)wACX^O!zsibY z`>4zE|D1mJyg?y8scFEfBXm-rrxtX=VU&J%jQ*S#>r0tp-My*RNiuXQk_G7T=0G?l za8OS<>mZg;uvS4q<l{g)1FUm&i&hYV3O=%43(A-ngTH_F;YLl8L_6t?C@ZJhP<Rn% zxTa5>K%NFV=x7$>@gcFdET|xZ!(hq^?4wE29bF1BrcjOE-(AA9#jzmihb>T`89P}` zZ?;)buh*+)w<c<MKTU2O>_J}dhs)7<&;;+qtXzas<RX<S3hPzLDPdib#t@Sp28&}g z8l7zGf!`~=nimJlS^W`?eT9Sw__-M6WA_}KnEbjh%r+Yb?lZeKS;MJbT2gC!O!gAZ z@13=9m<8kV*;4r!HdCDBM;Nhp36>E>absGsE|BOnBQ&S`76o!%3eCxdr`*i*I`e#i zYVj0!`uJV@f#a>fejTAQx)i1{3<`NbJ*d!TMaJ-@CAGXoR}NR*;mJ<QVU%{Z7-!V% zt&TEU>@NF^(-y%FfEc<DX?*(3jOe|I^&4Bvmhf%FB-dL;84TGF5=RHP+Wk=qBGExm z>!DkdpcBynO$h7r><V4&;&nzh(F`3Hwb!~~0QMSr%gn+BbvfdouDs8E*HLeyp^<4v zbj7B(F(iv|o{>Y6&iGRD7BQqHwO9gmy#PiS3FS!)`iu*)m8GXJU=FPh{0kr|ek-}L z5Ub>OZg;EzhTS(vBn**ZjIy<|Ejs{=`q5N)Z}TqO%CaI9eLB{|*>^^as9#KIVoIj5 z;@YQ;g@fUBp|)$Ta2C)|l(G#MnzV~nZv6Nz0a`Zp@?%{-%)omo?mdIz7-Sv&oP>WA zs$9cYt}qa1Ft+txx31xjyJ|B9*Y5G2%u+C3X#tWzExHpeN4?e%g(oUJqlCAXh!4n8 z&tkO951)g$7TVBc*r=kcE=2G-5P!>saQrD0XHzN{A7qW4Q1+1=ju6m-(9>n`EW?LM zA8?80QIa$}T#%Q6T2E6xRj3qVm6B9tHt1ql#hEO~??Slc#iv`h_58@tGNSse(Dg}u zK~Tq{P982Gx%0wiqsiJzB#yOZNor1<*zy?o)Hd#6RDE$#P2fw9KwM|j(q|g=fwO?n z4h=fy3t(Bmb=siiu49-g1w%UI7-z-g62D++F$;Cr$)naqp?P(Pk7g@zpI}I*2am=t zFXPA<6B%p8)Ge*qet9($X&hf8Sg?s{;qOI1)T`pbX>(oZ2&rO;i0r~5Q6<G9VG+Sr zD203iOsKG=tSena@LpL;t~lRqAQf|z_IEVq8AkXxn<bWV$S5YSCgYE~cs;u^Lab+` z8r;XMQuk`FfMIJ&K3&=xxOwm*R~%X!aZ$lCqu<rwaA5uhZJ{wS9yEhdDAtZ(`|lu+ z8AngnNn1R0cP}~vDGZg2nUsv7rtRJ1uzGZXh8JaX<IatwCdx}IVQ)EV>mtUuxII8; z5gU}{$B5}Pex%(NEDbjuGK4mR23dP)%MT{#dpB;kJJA#fmiUW~ai_GWMbmuJ0vFpO z|BJ8ZN_EEw#>`5O{Il3e7RS0F>{IDjmbS*26F;Jr{OP4YFhLG#iQGh3hHHEUcgO~f zu#HBEnyLtFOSj)fQ!{E9^=|Sg=oVo;Ntef>-spofW9_7sje#yYcI52z8106%G{&e= zO~<gH^Q1=<&R?g@Wy3u*oZCfloPK4>#&pL_T?bwPtB@9rEypOBK}Ep&2Fwe)sOuE; zil4Mn4nWETy0KTQMWex-uo}!a_1+?)iwI+mP-KM|3k*G`r${XKT?LLDv{2E_h*;#H z^~*72Z>$I*0)JF7x@B;j3_D_r<zATfqPq?roV-iz5DOCFBu$Jq>cRa3u|JE~X;ce? zewB6*1#CgRtE0-`kE(6R2D&;*+9=tV>W)?xxk5$BPGsfDxJVcX$;v2%qxD!iA6L6k z($8@Ph~KrK3;)7K3CZ<V*vH%H3nkl_QN(m@@)zufSq58zQP6q7**bC|_JRX5f;a#a z%aO=9;Bha?oQq&9Y!6zwwDo1sm0&GQQHpI^Mdu!h6P76Ekx}7ZnDj<PZyh0*4iJN^ zl_s6eD6o|lSwrtd?fpJGG9<%3nvev=J|kq&Gh?(OO^k?!Mi4Y#>mh+<SzHtZy6+>9 z)yRhYPk0kKqk{<^TPInD^#QnO2CGrJm}K;LPD_wnv`V&j8wLfl#^wZG2!`};?x#q} zZE>^6nH?iYxCS>Pte_4ret7!mx5up77<mYK&3t)zWfVu7jYaGI4k^O*Zrg(UaKr`$ zWEsN<6_)||vi2f;nkl3s8kTPbF7<g%+>@^Y&^X(8cw+=2X83AyF>2^C-DCh12-$sr zr<3A%5z~}~dWI}ESSXRIO-2t)*(gqsb<w1Y7(R9Pl0u|nKrD!&$G9YI6Q}5zENmFI z=rqr4l;0mMA(2R8m8VH<SSb-bGZi5cvTT46B(!cPs4{BXE1}Y;8iAhM!Ai0QXqE`1 z2d`udH<Yd2wV|+&!4N<I6$7@TV#uxn4q!g*(Ntg=KLBmg3th0+vgv|MW3m%YlV}!Z zfNa92vb?*Xv#kYP=#Fe;L$Z<5vyFAcH8FIO*LZ_KDF-)}n%u!Gz^qRT2Py;2yvo~k z(*^~_$=#%%XCx)|RwZAwY+*g}u|TO#cV(n%fx*w=;4x9VN&YIA3MG(6^h~e@xe}*s zR*9IP3&-7v!3DwBrFU$-02=rVdQzD01w)PpT*YGJ!Pvf_#t7Iq=K3vdTa!0{?d4-> zPDLAjjQ04NG>?&0pS0SA3~$=;#zq3^^zm<w*6ohudDvQE?PVnC;YAdyD@HG@UHhz5 zG)Z?i_x=TUYzwXuEDUvMU@_@Cjh68jo}uPzWEcibPlQ<HL5O>woQMZ~1T+=rurLnF z%+O;Cm|JI9husnK!VX(=7z(j}#Rx${VF_(%2~Cb6LBwrQ7ZzbmB%h>lN&vhV(Tq*S z!h$-<ox5clKFpX8;tf*|z(IBFW@g@DBE*dHq>u4R!sSpIPL#nzP~{+t><dV0!vg-5 zn@>?tZ901g*2=pQhHH!2sGK2wdJ(#9hvKCZpPeEKj(9z1<OH#&YJ%Z@3|#Yw50}2u zgpUZCqp#Rnv4aOEw}r=mo%;ui=$Bx*-HAyme;q2Sv-JE>*cn8gr<=9~ZeQ^<rhsoy zC+S7Q(FnwUoKYp{^8vN&S4fPX)+&nR=u$#Zu{x7O4iZ{Qua=UOWGN-i{-JM5SQDsq zMwmt0q+`AVGiLc<og#($OnMGg>HA5)InyIZ5B872Fg(UirDURKA)ZVc36q73M+vLN z>NynXRU5BO;043>9_frFi=RQvDn88863{aUf5^rpx&aT7?a1|QE^ul&wH}u-j80}a z+HZm6y2^$#kw9@{<+UBwe(rN$oZwilc^bAZN}nty5#9JM-k8yaYg)l;CCfc}(YEkt zf_(idO%nT<OFX!h%wm7Mr42GSA_UeisOH3b8Is&pe_9v=tLdWAZkW)Z!u4BQ{B}Ws zWruHMq{^TbWzwWo7rzUKcY8%fNk};R{Kshm`z6B0I+Z!PNfYMW@wybP`Eu;n%j`@V zL(|nWE?J`~JQ+a&az%solMzwR|7e712^1BJ(QxVVLxPyKZFe{g%6eztf}!zAY}FY# z?0-84kkE8n3;4_VLO4@hPY#7D(vKaUpktEb;u1=ccrTQ%CV#{T$|5K8xTlw5S|lL~ z2&VJhaAOLN{~QpT5^QX`x0JApwS+aND$iGp!HO%~zBs4Y83SBt#8j~ujiMf4J3q)+ zc)btb#=f(`#;mhptS=#{2T@hlTw@-uY>yf#<MY~nte!`;Dl`fejKI-iP!ta>`$3A2 zo?_^Wb&$jW5_}sL-HlrjTHQPy7Qp88pes&C0jx6^G{aBS%4#$UH5y!lE6h(B0VoV2 zFt?0;+-t2^amh`3C~!y2<T&6aU-ocEQSc_%BlYNbuT6f^2xd;Tib1{EJFqIa!IY=& z*);Ojop9>nFWFdO1QL?ndr*gEhTM|w3mX11$9wJ43~yyGTcsK^__R7+@)B!7K%vk! zMiTVOUBP?f>_ps)^QfW$k$-u`0^BIRxy#$1RZ9OTlvZ6mO4N@`UMvII*QrwQ-<6=b zZqaMOD0|*1>k0BMMbRSnL2;=Wj8J{|RukpRt&}Kde7^48Q&o=&`zn4eP5F4*9%-K^ zv6DIwv<4}@=}p&KZ4F?K9j9S@OBlFOo+-=6<i(k>_yya-`fy>ssibtQEQ~BpAyy|$ z4^lklhDD#{P%KQ*&wCNCy~b(MQ~uw=!I8^!P<<DMC(uKq5TIw7PKmmUs1?eyl^)B` z?cm7Svo-QPWXT*4G@QN(dtn+;v`RE|sh*PYncN{|KTG=91yV;%v|3}G(uN>Sn`GO_ zOR*&E3%NePX2(lm^c-F8z@lr5fy^Yr>)}Qw%dpGAi0HJ5{TB!%8Qv+RdB7*3nFz?+ zi&kS%+$Inpr_P`)to0bsw<YvPj_-mOimcwK&?*^QNl4X~ud$(QkS%bSni}f&taEvA z)CS!KI;UW3MM)9zu{2yphafcZS|lo>_z0hTwBhLUvjyWkZO~tk4|&J~0_mXxIxaek zv(1OVO2cdj$m8<zG2yywGEf&4NiL(g>r#+ojyod^?MS?MBO?VtwZ*_Kcx2oK`3b&J zxLbDc9w2cJrRfV$NN(aBd<(HW=*e3i9m6d0u#7Dcc3^c%dpJ@rjFS$IB7E;m&)7w@ z&WFo{D+6Dg#5g0s00A`Fq_Zl_I%Iq2!$h)8!(i6%jRiVO;Be3FC~KwhB2fgv@SLH5 z9MN5h#^l4leGHm-aPqFN6L{e_GkWl@@rirxy62nURKuJG)fCtT`AwzKQj{WQnHGg1 zUs#WtR>SDSzBWmx*R$uwNO~V-OpJXU>W(UV_^1w(<gZJClpj7edhE=pQL-hCPL7fh zBOAr6@o!EX9Q!%}8O4oaRMDfHeqlwWkaaR|Otmi=n0)<$+-KsH+;$aBEk?^B`dcC$ z@!CWuU_Nw|{y~Dg@$jF{m`&QM2Ng-SUG2uEjsnDbM=sw^a7yt)j=UMpWKvj0-gex$ zDOEL%g5i~HFoNG^c06$rQo1v2isy;rOa?@e3yNT)+kC_~7#2kLlx;ljIc-&I#*w&> zZODm`Yk?+X&_h<5Mx988Qx*+b9D<EPY~Ex$GP}Gb+HsJVhej@M2h&SP-rBrxxKjJJ z;w5GIl#bT4H?7It0Xi8=uc=vbxYNFhdK)(|txvivi@vxPHA^(yMFSk|J=i(ETA;Q> zgQ^UF;XQ1`Z$<H{xaFnC?K(=pW_+X?EC3iicTOZLeLO~sLOP4s=V_S?;3BVP$i3{* zl+I4`6`;-Fc$z}<jG@Z?JGDD=qisshM$smoGzSZg4OvcoPaJU)MvKJBEh*$*jH14A zB~3Fg!7g|r$c{_2+9vmEaS0G=3|it$2R4h*h>u2SO(n7mb$hF2BR_3fm54Yl?l-ky z0mFnpfR{%WsxAcfBaE{%yo28_xWj^vxTcp}5@)QEQ@6o{=(Nb3rkYz1O|e)@OG9+? zrhs>`?OEpj1sEso%3`Q^ge4=93e0L$i;G@@Or-Px6HSNq1LP25TSPG|g&D!IM0{<L zZ<6wT$PNPT%01H<tUML#uMoS@2k`RxmKY29a`cA!)JyPRkjHUb*FrNS4(U8k*sGx) z9uIh=Bk3+1Sp!od8XE;>rcWL}J&UpMGJ4~9#J0U4@@b1a8{%ypv^>qyn^BB8H1fEX zf}-}tT9>S<(@h=9K9?8sV!nNjC;Wn^iV%>zw5iv|d8us;#SRvgIC6#P&+0}Hz5r!< z)=H4FtPNGBns0F!Cp_Y;+I-f7FXV_3lkw8cJg+m)8=1nUcHBT%-AprzvO?Xo)N42U zyp?_4%_arES<wHDV%mFD_m_SeWgboMACd>BE+P^vK(Q`BKa9+bf~VoX$F-V$sbFqX z1@z$oCb4x>SvBoC$GiP}NZbZ1L7q;~2P^e<ob{0xI_Es{y(4=4&Cr>2xSJ93D#g$~ z?ZuAzZg@|~U!WyOJH&{PTT~t%?WrHQh2Rh4I{(O?ui2ONzZG_3kmb6dF~Iyu@%#bP z>CQMG;sk&N#R`?W@p{~2KdGgG?dRu|oB>e4pe`ZK@k<~QkpJYk9O~QEKFb$PFUcFH zfk@b0ZQy5P=FV{mIqLc_Fw;sPS)Pod@pB34{Nm~>9!iLt=^_P|y1bNwB`ejcc~Jd; zdQM5xia<$Cf~ax-YG#9qW+k%u!8RmbZIUBWmIyv1QEg{S3?7ouz1cLzLuLn5F~;2? z@%Qn^N#`^diYjRya59q7b%)llWT(gQ=J3PO;<r!E49G?S+eF?fJdjB}6YYxUrix8G zCeQ0M%MCCjj5;+fJ!KWw=zXnTSV4)00=1b}&3AEhdkoV2Ao1-W;&7)DB&-{PD&hf_ z!u{?og?9dipgGH~m88*_K3w*<vaIq|Q)mlXCi7J(iATA)b}K|0RdxC;@^*3b{eXH? zOKb=>VDq?x+i;%%)0A0^d8e}E22m+JPNahbFBEizy0cL+_m7&<D|GYmp0shp==c5& z^<DTDVaMk1>pXuiyq)S##w!uzVP^Pe7aG*B#Tkd~Vz@=bNG?)1GF5xo_9vz?Y?Km| zoKv2mk!N44<@+~6T_MLIM|Gi@W&Tku)WoInfZ_`lpO`8wDdfmQW}1*Wtp@mNN3kNQ zc;vLg_Yy$W#5WhnOo;mY2nLBCsp40Do8{~@sT>w$wos1%owGml2-?3#+Tt<fBl2y) zSo+DzX9a6Uj04mIEubbkKu>|nXg{bMac#5|V_q)SS}otyVq>R<i%R9!?`nMWp1QqK zinzfzNQKloBBZc5JaeZMTN84ff=1|{Lw7ywdTNa&U=zoRpM*?uL~q&kCXrzicCw)9 z_SlViX24awS}7+s5onT@qF$E<JYLr5d^$+Q+nG*M#cuAf-F&^p^_oCsta*=AU2^A( zwq&EHOEar6YS0oJI>gXUBw%+0FBlby`MI>D|Jg)_!!#;dGI-Jb;gWYSuukdbagW?s z3qvSrb;WK`2=l5*AJSIQ2GFC61#5m(t=vX5IUUAdB{-r#Hh%|%?#$jP6<<Y9la9g7 zj~mGj^*@?UXToI&Z@}t0W^&;e-)e`hCZa{*!g42B3_CdWyG4n$ly8J~{JrQU+x0=d z*x8geD{0jc^hX;7*|OI2G(^Kr#%95=GYsw!=WTZs1CEkSQnjnjA?Zd7T{Kjzn~#aq z5w&wa0zZ453BeGAh$8`_GNQ*<0~wx=-pw@A{&Q(tvNDC#a}h>fb7anx61S&1uh?JS z&M`p&)){~Cx|?#RDwLp{L!!iybUAbSEJ>`;>(eC7scMC#W6NTOLzjU<Sq9mHkeZdy zZNlCW1xe=OS_r)buT7sIXi3OU0j1CSB}g|_ski2{#NKXSd;g2A&)>g(<<)ENzqR%2 zr^H4@VUA0uagIyPV4*{5N1G3>e(}?dx1S7JX_93tVVpX+P@5%HG*M!}D?V=YB{yah z_O_U~dikeUuYA1m?9&@B{ma!W|F-q=(=uxtozDqDr%j$7Mf_5*_1gQJzk6omwJ$b4 z|NX{~f3)?N-|f6qWWQd1Ve6d_OB-fERTTZ;(`z5SS+al>kG8znTm!YSs%*BP$9Ks5 zA!E>)6Bl)?GDT64Bg%^aLTpgpl>?mB2zF7Mlf=CbH3qj!+B#2Bdkgk@Ir5^#-dd94 z!PKZz)KoijW|urKh@v-{(GT~cgA0H;Q6%3-=So$N16oJzCGHH0I-V*V^atZX_wquv z+KN3L^=jMXh2yHx+wAjucMTmj<Bg|(w)y8bH=h0Q`kNnbU4BO$(^$kVKGx1!jt#d3 zUipFxe7N<>yPMDc_1ZIk+IauhS3iAj<D(}FHMq#;>(6bx^ue_seZKMfyBj}#cjKcc zI8K6CjaNT;PwoQ$_1W{A&pmPV${()3{=~Jnf4K4NTbsXl@9HN%RR-P7S3lTz`R$GO zUIkD-JBlbKg$HSDMe_)$nI|!-T1BsGin8XND>B<~SY0k<Q=N$CD<KtSF8T4)Ws0SY zZBg!!huJf#EN5{0wr_BrJ&o>Onkr@g@_mx^(wt!-$FUPGr(u63ZjGTB0RuzJA8_?` za#hoH<#{1gjykHYw9V^s?9k?H0ZLysSwE`IHQ7*M^TiWZx%UbY1Wo&*lXzXP5Zasr zlbZUv#aXQT6A0dXy&e+p<B;p4qh9}B5=f6?K%sLKE@-OAvJmE6JkLQ2E&LfDSiJTy zC{T#>42+k135vaOPRG>2L#IySnU4n!nOwNRI@j_ue7E0Z-4NCmt}d)>PE>C%?eE2y zVfsuK;oB*USO8jfL!r<eL>8ZJm##Ky!h!Drk00MgfS+yeGw2^yxI@ZSs}Y1dyeKK` z>7Lt4Sq>yxFi`Jx&Fp?X)+m6YbF^mM3UkfY@~d%wgsYWW^zlR8W8RwK<KH7Vd%A3$ zu6R|7wpROwicvvd8guq`=cHSWQ5((l!Mbsb&XJw`uH2@kl5}-mwQPtO(r6b0gvBTV zlC{~d!?}^mcGL&mi?di-Uc9#M&l(fI-BldAp~2@OMlmx!m4!^phXCi9%k9b>@|bkA zuo_~ephl2tR#vI)Ncp~FfD&>>m$vCWUlWHUb0hW{x^*&coS3V#b@SBSJTadyJz^;w zE%8B$b)5U<&ne8jn?I%SpGa8Nzk(TzVmRzoF&qGC6rNc%?6p9}6m|nGk!<}}4_2@+ z$O;;IM+xePFivuBKnDNdjEJp8afj4_fOL&kW<|SI$8uh74mS#yo5KxK^I$_qv><va z;h&gCFO9nR^%tc_XwVVF$BNv|h2w2g=z2z##?vClQW2BoYi-cqwp(t{QO~e}2ucS1 zrC{^Q^WJH+P>R&3oAU}ZD*UZOa8@LLa+6z0C`vZPkd+a<y&zd2l8*|v*a;ZldQg<G z0CJn@w>hd^iv1w@euTc#u@rZ5HWKTH`Qlu}GNP*vd72DRHzpUyBEWvg-ryMByIw4T z3)i7IFPJ0gVw|Vw%)N>+3A}*t_E^OMRx6?2YH(k0U>CyNPH|)LQ(Q5ISx=NVY&1KZ zQuGYljuf!aYk+yD5rcr4Nn0x<g2$S1J&fEr<2%YFe3*g{L*b9394wjMh7y63N9LJR z6^}bQe0xtqD{KZcWGph0%MCJokPJ%##0NSdmd?t>z|Z0kK@>k5<{VS77scd~o&h0V zq{y2{w4LI4LDZ?sightW9w1Y*ikYV$?yDD9y#a)@i&CK5P*WkChKHs^TP0>UDk0Ep zN8cy`SFj~{%8AY?xQC@fZPSSz!e0Q3Ky<%+FC0qmJv<b*uS*pI)fB}9Au3G^A2)}Z z$=0mw=1$qVb`vwYwjSIYj3Z-MhsADqzCDc3*j1{SGn7S!c993@ExU?xwj>|IiuA-$ z{%x4=WGTlWUat}(lj{>(Z~XE4tN*$AuP<Ev{PnGmuIv#$i+sgI+6pk1+*mh+O<{eH zzJM0+QB!cDlZ5%?1O?nM!g$c{qp)c1YzWCWS+dI>vL9t)G#%!O{G(}Au=;*fyh0zq zZ1r*srNSL_0^pkUA9tSeb!VR=rGq$Z%`e*~P_Z*cRJlvCLD_so1mU+P+lam|`^{%3 zjP|RaT;6!$=T|>}fq(9?f;w)~afDkSU9%s?gD-gJ{ZP=D(g}@uBgh9POt=Y?Zj}RW ziGyZB`CvERdST<)k8g5xyDQu+%wp0X!UWM1INc&9y8807LPl4IXIJ>>!XFnUUHIc- zr3-&t#B|}$j<{j0Y`yl$)lXl&UASr7K-mcwk=;Q@;jsb#>mXcqs7qRPP`J{?LWuM! zRgCZS;SzF!qjo3s3nnuO2gZuJeOz%A9hR<vWUUZX?FcU*)vX0Nj!4G0(((Gr1c!Bi z-xK0@!;D7(#T#6_;S`@NDL!FCpKzd0I>je#=o1cfxA*}Y`T+;}LGk;5Q|+ML$^oYp zS?!=nj70E=E|=gDCoaL`z+HmJ(YXYVLvRTm`?oVQ`1H?P?_Ak<>DA2_et(+?KHPoL zRhQi9g$N93%_JYzg4H7T2tC{*^l*>R!#zR|_Xv3+S8cEdIqM*2HFoADo4@?&wa@?0 zSBP49M3IJV7msO@n@EHb)^-|(HX13dWuuJJ@^yh9GjY+zH>I_0+;Ziiz@U*xrxF}z z24o}bmyy|0nXM#~neOC=fB`7DANzrKX<bSfr4pGc(aH;S(NZAVE{I76(UKsb*gh!c zk_;#37$7I(uS1t~L=k*JMHE@zc{}>-kDIT2@prpAGrS-rw<Bw9;_R&=t*qKawK}A< zMu8!Q%zIbYeG3^ghQ8J8)=*2@t)bVoQ$w$6qsBIC`lBbWKk@W!m7T(Z5Qz`W<^suv z&(VL+!qv%bGC$Jlf?A`zy26$e!SS@F?f6~_#G1ZL{D)yR>f;#N$1vn-`oNXhjy$V- zv5+yL*Q}i}sThWxPSjDM3cNt6HNaW5Q1l=dT@=fGRE$3#+C|GL=qQDfVFy=BIl3v@ zXd$#aCn}+}@^D_Y-XyvtMDRr<BD<H1Q%1Zj?h_7@J(B86X%sz55Abk8wE`{u1A}j> zW@?ZowVkHYdoUH-ZHm_MHwdLo7Gv?3+Tfddug7%Wu$p4y!jz2fX9}J}?a{($iUmS# zAv&58E6Nn6R_{wMU;W}WW~ZCa{c+=?pEK(D1V6B^0jkHBQS#+-;UHhZx02#43-La= z8F0gh6e9AgN-Zb6PD_>ekV4Ot%shT}X8Por(~xyA8a?Qm?@lNAkrZto{~V=BWi>23 z=8SQK?^8fvrZC~&U{s{IB<Q~}SvqRx_N2J^XBD&#j4-aW7ScG%HvKVu%h;NilO{s< zx|3izUd2}?#AA{*o_hL5b8fH|gjSaY?Pmv+lLvmJcxxrup-2t0gHc3=`IsB81=dq7 z(gOoOe2OX`-8>`;+k6eXOTCqYkxz$g+iR8_Z@Smx+jUrXM41XMxv0%JT-r80WWnU- z@d~PIV_UptBK8`Wv^#2g@(3aOXU}twwtoH8#*1%kJ^86Bux+=DF5#reRP#z7$0(9V zbK)0X6XiYZl3K&LxggHsfhtt8R*61lg-XTZb^X4daV=CSYUE?aer4=+hVE*=c^TbM zwS2ho_LG~xecj2aLFHdvz4Bt|(z<%(4;wH2Wb4_Ful?jnz#Sk_b>-UoZ*IN+%Z+C) zUwh_H*Dn9NnWd$mG4F6NR0Fxm&l=<M#JpA3WSh@f)M|Ls0Pej-<>)Ya<PDT=?4~qf ziDPUwc3r%|#VsPCoY^i0_on#99<&YmZP@X4z!}@PQ(@F+7ed2cud^QrGZU1kgWlEj zP1BrF(>rn0NHNFo0HIw@mUQ6~bE$YBV=&4>Pwg|2F7{XLpOvn8B5s|!sS^k627$Ju zAl;K<KZ<uImZRJmst$(Bf85aZo-Ud(5H~Krvh~WlYL%%~EBfTYg6sXXL$|39WFMsW ztL^zUOn}A^e8a$x5Eeaf{LU<5h*$sin~nEgz4o_XUH#-wJU+!Fg1{jRoFa>M_~e3u zZk)qKFG<NwHz=x5#dze{lZOF6ee?WnU9=)ssOt@xMjZ^;DYKCo;DONLU=$zfY7|SK zH%m9REuE3PT`^&gQiG^e6vnY%WQ|#!!N==dutOVcLx4SB4)(kSwxPfpL{5r23~hbN zOl<fPCpd9vB~IFjYK72bfGA&JI#!yyril_*K`s;)B+uN}!)XgE+~T!ecZ0I`@*;a+ z=Ov>_wRL~^G<u)^^&@llWNxtW?1#cr>*^;zvGhQGBl#T3=J<#&mzp?sDQM;Lz|S+L zEXw`EX`wc2>>j0@npFEfg|f}p>-Y_D-Y^!O&2+Gk*=Pl8>?1O%1@e2t_&#BLpESN7 zFuot0-+|ToTCHIe_i+NkU(I(y%4i@Z7lM-Q*2P&VT6WzXLtO0k%|=NYpi>2sFwHDy z5A#p@t_YbnVSD<3Y0sWHc19U2QhLx#DyWCcH(HB#b;ZevIK>eMBDQ6TxKUResfYs^ zaZ+NNn+%o864gvtWCY+dY$b3RB<f0K**b}0*2t;jI>LFj?vR`p&d>vAXu}zL;B>`y zQS88|VwqO)Nl)=16)*5jL+-UtJ`EAqKK_*0;~J_1Dw)cwhFq<Yh8pH!Z=y0vGYuQY zu*P!_U)Lp>YsKa1Yw!Hqf+E`m772j!Eyx~XEIio)ey5%+lry<D##_aEI6Fwm+G9NT z3QT1*y@O=Et&m$TVY}-NAAZIM={v^5?&$xfM#s_hFnXR++z2yTe>3(~lKV`iqC_{( zmZ)msNQ^o~RQS|iW3(chZZ6PQA&Q6}pj=s_c9@$uyOb5D$_pMe3R<tW`^Md^x%JI4 zn`N;Bqy8RP#SWUq?Dk7BXsn(Bf3A3>1SALHyd&*hhRclKOU5tM7~-sLm@}MEwxwhC zuEBp{D(B{n`F3`Xf-8$mY`BTVt01Z=hM+lU>1I;Z-Qxz#Iwku`)#$vJ3#$%{^0!@b zV|@U@-|T9(`QK!Qq!h7n23?Fke;@c@kgcFg1=_AYdgA(z-sSAAKVH86%fB!cs9RwE zn`-`BYBxhYD(NUbaW29fSVUBCwOd8cj5$m;H}RbWd2_Qik}VwL6_KG{GpiG|0ROkj zd-1+HQIx?M@dkbs`uVEo-x1Pz{oV}La~I{l9m>BUl_0yI_wR`5Z(%s=xMg9;zEy8! zxkp@FMC>$PG>a0(9=Ibkxq)t5cCIU5MBYKuvt~vH2hiA5#wW!&9m9}scQr>{BIt4v zHjE>5(=tvr2Q8yHC^C%$u7qtPhycrT40oF^C=T+#4rLTIVRB>KW^XOqP%oK+C|RGn zHH89`lPj~ZCC#yq3)fO5iu`=1!Qk5f9BRfprv2NDX}@<&f(S%qLtdBUA*vbzyTZO$ zXWhaw+xqNZH-7RihPU|L4>y1FhilLL$@gdhK1)4DFLTr%5sy80Lww3~EK6Hsvm!&R zTk&=@K)`D9L^Dn_01@vz8ylazK=+)p`bypvky2d;<**lbX%5CPo;%077l((O5U6+H z)QL`UwO>>jlCN4(9_3k0C+<0K>Rt4C9>~kAo@?PZ-SiIzVH+PaXifKaZXckTeYnLv zIAh14D(ymL0`%C6)!WyR>E5<o!DU#vh<3n<Du`B%f*D)1GG0H32s4motYxFzC>N}Z zj?(BZ;;UCxdHUka_gHZCljk=7&zqaSz=JMcM*sEMv-$TY|Le2oZWSV0gT>v`!)>7n z>cMX4fpNzF?^iPWE$+yzr-vPBLXZR!1A*0JobXb`#%r(nsN%paqzYIb=?(CS+eRCr zAG@KB2CSlrNh14h6W>U03)i?I?ol8W{C~8BT&`Yz#YZj&Zz;LNy_+DH+e9+dmm6g$ zx0A6b^xQ6@xoH-&6RX*L@832)dwcV>|Jb_xH<NtO1v;<ZLJ9(agDi>LVP_XagiBpH zTjK!e$JuFoSo78|dEK?$(3J8J{|fuH17X=TU?ER-5P7Aa0G@wl^XDHH31H&;`7NA% z6v%t1c<yTYFVfXNJfj{Dnmm?%vQ}w~PaLQbA8g!uWE(x)o#={PY(N+36kVbmyE#z{ zMjPX|XLqvjxFfR>cr4{JevYDRmU5^@dClE3xb5cE0c;caXc5(wi{dn_RHS<S>5b2S z@8NT3vTcm!-=DaZ%&rxutwASDZ-muxPf!`UJ)|T1w0j;WJ0yj!>h0^)E!sJy_~b+A zuF1<8ys@JqdHMCYjb~rmxN>>xqbpZG`|RrF7dJlpbnC-+Fm~S$U*CN1_Z!c>u=$%m zY`pf{&F4S3_QIQJUH{3uTd!QXWn2NnOm8opkp*wc{*4zl#_O&JP@hr{Oxf7~?_ec+ zL$m>g-U$z#Lmm8}cXuDQ;sbt=KIby7B6ty6f?&?a@M_B#8Ya~KzCGQHRzj1V0aH>u z2?y?kUz4AGT9P2{l~vgqq8o)y^l!tO6V$Tpg8LtGxqakPpcW}Dd?`9{Srh)fg`DMo zsO77smK)vkZj=YzApdcCCmQ#Y@|SNFe&euMSNDo@?{2>F^3_kC;Ei<SvnMv*`RV3= z{0+33t+y`227B$VA8dT_)V068cJ0mQMVYsr+IsnE*owE_#s4;6`4~3jYajf9b5SOI z=L&qkRVxRNZg$*|>7wD?<Bfy&OchWob@xgoqqQweK-sNd<k;1n*Y|dP)pi!ExBGE= zZ->2wk$WS329S;V!HI)YH@?8c`YN4|T?v(rVaA8162*liOAsip{q@hAKYw=fx3Awa zBqo5mXqPGqACuL&4j`Td>)Sw!vVom6+j|4zrW5tM4;I5#i6-u@AG~W~hhrNRKaW68 z>>4CbXa$H$SK-MVhabCY%dEP))+Xu`p60$exov~H1$~zn_B{u@&~NpO@5-8SFRrFC zTv-%4&E~lYv5{=MLJOo#HA!7m+mMEFso>5-`^VK!u57;e&c?GJZ~W@rJ@T&VQoJl* z(iM9U1(yG>&z|3S>4U3R{;>J#2b(|taO=ubn8MfPWD0I!AN=O(XTRQj?$hhfyu@+j z0U}^H+IsIf4g+sB)QOj(N@IK+#3*iT^OqlA|M}mnGHsCOh80y}<I2a^{`Llzla59b zR)WvBrh}+zB`=44s{F;DwqE=IdI;67|N0eAebG@XjnRx(P+@+<PSvT<jxZM9urwdN z<3o%+VCGX9avJZ419~TgZ#gS6I@*c*qDp4OE8yUVD)O`_;6CF;(fHrk_?OQ&{^i}x z*WWO!BC?H&^b?)%0sT{&bzk`yaAQ^@{2TT7x8V|lwOE#sNDaiZL0FwIFdJi>Pr9^^ z4`EiS=oN&C72HIOcBT**g%8{O;NQ32`I*^~xEG@hF@BUny~=QSsyXpy0L`;e4aveB zHT@LVoDMUTplM}KSr`ZUF{<M)>rM~97cBiTWHFHdB}Yz@XT^wJLC~5}L!?Yug#>+B z#YCE2P!O6`P^8%nUm%)mi}@IzDrOm|CEpvQ#$CLfiN*B8nR(PV`6Bz}Vxu)&&UuJm zT@RueAZtx5P;Dtrvz%a|LvmnH62_><h?L#9(}}VOI?~2?!b?jLIgc+!<Z(mMD>JK& zv&T;#n?C(*N2@B3S4n_dla%$490K-Lv?<pKB3XztY|chWPmEdIe@LU$Y25c_jHB8{ zX^UO;jX8o{#2xQwp<PsYG+UW60Rx_|{q5Dw|9F$Rf;~9dqU#X6bANSTN$cu@HYSp4 zY(=n##!Jw+<iZJ4zEUTEQUevw(WuGs>G1;D2rvuIm{a-lz&O+o2XKagtq0^mZh_^d ztT#n$Bpfyh$^kk$X;b3D8nlRgoko=rVkILr)rmrw`K>Gey7`+o{4?V>PAI}e4(Sb( z$`@V4j;E=~_w)kv;QBK!Z2sayrVw+ha*A#K<TboJ|HSiGKY0Nd{MKuK*?jR2*FOBw z)+_I_H~}f-0*=A<#og~<<K-7Oo_=BL${V0-b6LDnd-*jbfH)r^Q;mquk2pQHs8IAv z$Rh7QzxnLTct4S@>?&bu^LNi&``~Geg!#pfH=h07^<TYJ1n~O1N2G(F==ar6-ecWg ziciIgyyEa4b9RmbO&8gTIUuoDjFc@YPdp*T!v!+@Y`mbgrcpPH=^dh*VI}`)vl4r> zC0uvZXmh{XstdIX`<+|BMy44sSmOd$RpE_|S$eGiq%R8@Dl4!b;hMEyQ;^GPPG!z} zS-p{8F$F+w*stR7HyU{1xji3pY#OpCV9?z)2n%z@^9(aFy1WJ_&pjBlLf}Uf7PF0! z%!lZ8dJ!t)BPvwZjU+^GhC@v&;xWT(b1ZZNLLFa%z&l&~UH~8{4Tm+1p^l=$nF{bU zelZzjq$uTur{r}{gr1WG`XoR6<9)TzdJ?Jp(Q_EU|32Q!Q~mV386agPqL^W&@7Uio z;5>{wE?}|DA{o9$->2=U(aP8>di+-46*&*ny(|KWlOu6asQ4hvDI6*`yvMjH5IlAX z3+%COckDPG9E;PbddzK_(_w~lq75DE3sr+)Q3+GdWgwhKrJZnIls4sR%yPIQzkICQ z4~%PXJG|p!1O~gIUKVd<BIy*Dq#!7h2IP<oM&VjnRPlN*TF;$2Q~`h2@kJUFc%gu( z?ZoBZp)-ms_wO$tUa)3{e#=muDu!QzWqxs!2CGLtrSnWI(V(aF>|jxAUNAvnMwcy_ z3XJWQ1vriVnJq%#;3n&Kus}?Z^JR2?+eQUBduGi8!JKc4I*iFzGdnt3m|2gB%{Nud zt;z!*sU`{y9K|=5#O~q+O^-17ftM{M-`Ff>sz&GRD$3NTn4#EbitF9vX`ZdX%k7y1 zA41r-b{i4Yb_*d2QnJy7^1)G*4@!{djk%}g1HEel94mT5xIP1VcfpP(9`wU)LT_|N zm?JLLCF9RymkP-Q6fdfYZ)8Vvx7jeunR~V692cUVL3h=Dd+tGNqcuoLGCcVv?!{>6 z*<c6Go##t}ceZsVAW61Z?+@&q6oA&aq{Rfymk$YR1`}lTC{+DF)II%g+eWfK-vjzT zu()@Xd{$(UlE2dwNqUU~q-kO2iU%0M11(V&V@hO6%8u_f?{9xIv!AoOl;t3=?ForY zac5?CXJ=+-XTPkJTIS3YhIRv|)g|QgK6sEH<wyFYE<|Lv&4G<N&(M8D&gv|bnYU;t zCm`GoY!E;<B|?_ZzIEF6n{TKn&jN7@DhKr+=mLNZq0->s8GKrFhdH>Y6#)ISF_J(P zMEl|l-RQljTAw4i1L;{-WG%+Rs!I+yME;(-gI3cb4m34+Fl-eNV2-;fip9HA^~p?@ z!dSVXFbx=(1q%9h>#SogGgG_(4EVNyh{v67AbyOLy6y_=dL&PM+uy0MS~p#76hVEa zjX}RwE8C@PWi4SHdaBz(9q#4EU4LE+R~9#K0tp(<350xQ`?zk~(t%*aN(As)+7@Jw zzuUD!C{~eqGV4uliEdw9)FnF<o6`?<`ouizenyjE^#;a*-@LrIcv%)@{Z5|bDrxm- z>$f;Vq<12A1&e?02T>2YvuSn(mB6HA0<V+F5q$0M<E~^mPuWXXZT{;zyIbU~0Nh8w zF~B6T00fv{*QumW*e(f3UZx<0L?BKM6<C}o0{>dqRxtuJuCn}ELSQ0&B_oM`l{df2 zY7lAy)rgM|7=DpXZx{~*W|6KiK$#ZHay2s)sD2|}0?8?Xt%@h!-{vJ4LaN;u@}7*b z5Ip}Huj)g913;W4;=drUu8aG-vJ}}T`1)u9Ut^a2G@WIwD#I}*?PDzuR86Lfv~Fwc znmCh3jL4##mZ)TuRg7s^$+0F<)@h-+Ktzr>fwErBi}Eh3RZ-#*0!)sPX@`c)s>zER z!%q^dgp;cCWmB$1rIDlX-_+(lU2Tx4Sz6!l_K3&F5*5=dbdexaE%fWMm}RS4a9uJS zzhaBbb)DXeHbZpt#R=6}rZXE^vLzEArf{=JZ*ox#OgkDQp+k$xGQ9>3PKA6Ov-(<O z$|AqbbZL(EC6$Ry3EFpcZ8RmGj2sCOK+{Do__7aWA%!1JEGl(2v&-TUXh@wXmoZib zDa1))QE0MMmo!FoiH}g4>4}p<X;n>q2qX?-x3LLCH5JzO@L2p0P*UyTfC&gN?!T_f zyisD6j1KV2qhx72$~8h7vB|Qk0%zWcKB>ncqX96@&AlbS@o{XMlv%#iqDMx^d>JAL zlXbvAayh#;N=gbonRt4HLFKDD7nU(Tgvy`}j8$8Mrh`4OOh0KNbrh`4@y5@l)eTDc z*hGfVdb()xR1+C_CBbW2o2#-ejXls`MB;i;HmxZN8I(YGg=%fe8t)r*WD1|eWIWOK zB@A7$ow^Y7MFea`PmUb9r_}qb0OKf_IhJM9`lp-Aqb*4+%2jqh%kG?xaH49kC|f34 zqP4k!&R4TMT}cI{9_%;~|A_=9UlWfHC2PhU{kE*{t+r<QYda+hCxnWj7wIjHiiPSc zAQBm(pwdAun}yXojAWN?Q@UFj3z3lMCZ;h9aO$Ar&gXM!XN+iaH7W3FKiyYRqr?ZQ z@I(m#Ousi7h1*7e$!;Y{CQiR30kn)6K@f$auXyB9RPsg!#C%y4jg`ba$463;Jv!ND zI$Us$P#%uQ9vRDU6Wjqa+ycb1+@w2bHR40RYL<nIg*$b?M1f#z(;dguQ={9oA1In_ zzI{*y)n-{?ni~Po4@{E8BWO2~mi1(!sV#Lk;JHzlsF|RJ&Z^ofTbZ^u%AyHa9lC1| zF|jVom9`Xa2fNH>`FiR2)T2a6{-G*`fOEW7t{eiy>h6z#S4aD{@CcA<T~~$3iOd-~ z?n!*?)3T+5goirVc+9A=K0Qvz-O<AB60C6aYMWjmLi-a&l6BpOiR`wO*TN`@-dnja z#|TC)g)U=0TiHe<A>ojfwut4c7LRpZnn8)SL6T0>-q&U0j7yS)H$gd_;#v!5@N%U1 zze;b@52e!ulQbVP9Rav|%Ny3678oQL`AFQc8aZ)kB*&5WOe2jBNwYe=5>ke3kK-0T zG;y8E1FP2+X{tZM<bz{JppE>AURem2YTqg?GTwEgm?XmVFggUO%_a60$v(1iKe0$L zL5UCdrPw*LhmcH8gs_tY=>EJ)tMopFrm3uPBZ(7hnJP=Ci)uZeTW$$8vC{V{tJk`> zCOJId1eobQCgE1XS`}*{fF#2Q<T1qzxxAZMgSEp0Y2-`?Xz8|UdEWKejb{ySK(z18 zsMfge+=s8r`&1j`1Ld?RC5U=WI-3=lp@R&h5fU>lcE6+xs70hnS2I(Tgu2HA_~H=& z+cZru1+zL*jR~pE0(u0?DcSeLqRF$>N({r0)!2(pbVc&wRyaP?MAnx<5B7*-O@Wz3 zOgr{Xw$o!KiW!rMO~v$+E`TDlVqG6|(n2OzMFT7fc29e#J;@_QrIoHytlx51vCV|X z<@#HYvH;6j$Pfkf;IZ`DEbMZbwxzBCTJb|=GMS2_90-9Qv8FU*IMK$!Sye^2TV!c# z47;$mOd{Pu5%UEpv00WkZp4cFxFHWju}`E~=4`^e3>E#eZV^kD)yt-Nao=YBd?)%J zahoi@#iRZ3u4^#E(I^@YF(Y~HYpzx@()rkccPf{`X3GZAuCnNEk+#vxMO`j&!AsUe z{o4lD>a3>XaK0gnM(6U51U!EtkDg&x@C(1;_o-aY>lOO>>biKa)Z>HE?^SBZ3Vz}D zrc%>2Z~eJf>N&oTMnfufJRI@7=2l*F$V(lCG57MR(z?kYpVzCl-?X*L2bh9Ok>yfJ zTykiF>;Q%{R@dUbdi;f9!RM`=K>IUj_2=Jup<PQ%lBf)#qEv)E3~}K$kNy4dX`(*1 zEIwNPot@4~T6u_S3NHR06%|<rYr46PK}5fb)22CFXYb1M>#M%F@Ocm|*ixE>w-kAS zYpP?OyWR!0K(TSapD=Kn^xvZcZ<n4rk}{5Vqy7^2>?~>14nh0tIu^9Cr;A7uMi>J^ zvQ@6Mdu$9p7dw9b8p-2pk=4~(TDH+o2hr6as@!E6DQ0wqH#{QY%A?R^Ei_6h?&Smt zy(Ti|B1V0%bbV@G(2>#C)(g6;^2&N4?x^G|y7V=Y>^^mqdmz&<bSJWZk0ss9i_=B4 z*g>s8IR9rqA%UlvJTvY3ysqsfd`5HE?YojF^oiI7>Ua$m0tB;kL`|2gsb5+VX#|Dm z!*3-~{NXSr4=yKOiA@c0nSoVC2lBVyqP`?KY;a1fzhjixnN8J`sfrp)?7TB}!*SAh zMm7HWY3?p8S+{wyNBMl^+!#t0X9Lq|b)HIo@U{erjZ!Y3)VaqZoumcXIO9yc&J0uS zgM9^T(BP^Tl^G2r-Ox>2iZbo9xx&jOaT`u8&Y&gfkm=u=$K9x>mYJL75ou9G!XK+( z96TUGcsWCnr?_k5Y7KN3<+_+fAJz@-+ktk#3oWP9tZH$8qv1lsvK!V}g+uH9UbF{> zqu1LRw&fd~8he>S^O$8k=o=so&cNT5^$eU5yd%Pzo87^UgE#AncW^lO&1ZJQ`k}eW zfiHo?dwVK-qr#qgttC?eZS&XkYD(8!;6t2@qvjs^=Z{YJEIZ_p?1TTx`(URNN05m( ze@~~-)7nAHkr+1XEBMoocW{^|eLvjG*q>$TYPoX~LB;ZEo6#j8jcGzV>#zGj_Zgm* z2JQnqK844R^ZDH0QH&qePiRh?N#xMh5~*df<zZCVL~ql#PANX|pKcaD)jk&a8!RSn zRekmh`SqLs_Gl-BtAQLGNoW%a$JmLUJ`E$p{FTd?ZKcmJ(2+ogY1$idr0DQZAVq(2 zK~ilAA|r(c2G^V6glGkTG|V3-5u3X=I-?)g%wVJAR-KPO&HKH6Pvt-ho4oX$p53JZ zJ}xwR5@S09@NOjb<r&fQmJrTYVsEV_YS7B1cPAma4h;t^DTe{(^dyqMc6Nph^jU4c zH|V)E(xO_3qj!W9m)Pl*chI|nip=4;g@N$;dSMApbC<W%MbxKadZJ2*J!E$8B+%sI ziX<qu(z~yN)?4TGIHC6%(*At%U#Z&jk)$kORaW56og!O4Gz%O$=hrLf-%oIkhSLI& zAJAFwnZ;PLJx9iW6h|I50r;`Q3!?CN?e6x#@ne9KPGD+2KPkkA#~2L!bQ%r(hz4CW z#lM4$OD@vTE+psHElKtVYnNb=B-lo7{r_<H-#}gdI4;SH%L6P5Z9|uT#KRUm(KLc1 zS$D9CT-XSJIc*%yY6yF*YM6^pf>mr$@d9`2_U$Md0-`C3{)Qn~Je-66$X|Q-M|5-| zz^aEOZ-09C5AD4@pH5+@g$H1?C9T7Su~F1i*)(eQ3N_*@R|O6(H3buDY{ba$4x`7b z9t;ahpi$GHr^wn3!AW&|7S~d$lDYkiA}nE`gaIw_il~D8${_~Q3yhY@6R=L;*DmDs zgx)qbR!mspFdhoenUTI6=CiwvY+16bGz;YhgP7z9?V;w1i14G3KMM(ix*a)6LEyK9 zBTu`>m{?(9REhUv@4_ETd$-l2ht))Bnyn^?m0&`jDF_WR*c;C{*aP$PS1-h^M<bMa z;E$1sbJ_6Xicd(g<mHu0i~Y10DlIW-CiV8mT;C^(hhdx#qF00Hg>bkaZ76r;mUAqd zjhE`HTbJoa2QU5Dg-551;HC}4e?r+4%J=jht1q`18I(6@UkHAe;L(C4Rt3+j=L-`( zI=gs7o6n23u~-_LW|2kl&fh$<&a}bLh4s5*-y@QH({V@_qKmBdcr`YRiZ_u+h;5pC zom(9=x6g%rmN@6gz5}$$eMK_a!FTlQG=Tq#;lFw`{tJfx;vsy_)pKef#Pt|^rnvF~ z$cmCa5NLR{nOI*Mrd`E1N%qA9WM4c$HeqC6DYEBXEr&GEf@j;D*^^2gc<#)Sz<c81 zF?8z3!FWM&FVAr52Uka245Byu3OZ$OLz{reP*>|_p`g5=ur#L#xK(tXrKq#@itILX zP63ceUwvjUzR2e?4n7dnnRi18BD!ux?OC{7LNIR3!L{g9+={FRO;C^F&+cyUP#nrP z*yQ+8Ib7!NRrp%|l9yh{V*keMmLsnC2^kkZgMJ^U`=8ivv9}gw+~uzy%+lr%lBX2~ zTh!8YVw>ussUy;M?AuUY|Aa&#TKW85*I}T^F|Pz%I}vUHua1HY_PzI8O8OU)=e9h{ zK1wnD0+OuG?gl}7`ZP)=WV$s=@iO117=LANV|sf;FL(7@@QiVW<$|i$2OK$D7kBaM z$F!1e1oYfQW*GfIHz&e?AQkh+k+>nYfl<H|PmvLDQQknoE9n2hx6fwL)xB_<Y4UOy zG%PE;YLre_`-QT6F2@M!9JmpU9b%3wJwxswK9jB5M0D;8xImP*D43c)-EakJxo+QM z-76?~5V6ZIy~~S&rrN+Z@2x>h=J^U}K-1<c+^CHQ-Cd`}(3O}u@bnGvvhVOf$~p|C z>|3iv++5c1n{OnH3ZojqfdR_bE=3({ci%t7<HDr>|FEa*-;Hf~rgB@bo5SCwd^Yy# z5_>q8&D+-)JyvclG=Ew1&B3%O9hVj}z-g8jJ{-~$vxd@SH<vn#_bxpL(Bxr5&TiFi zXEOPc)~RNRSaiUKVAcu?U7kzaJ8E*JpR-$in6qA~@Gzm66N-I8$##SiPAKsSjcr0= zv>l{0=9I>U67xZ<wG`Sg&(K~>)0TGLtCJv5H9!^Lm8yl5uAGL9`j@aJ!ulH<A~L4! z#0>nZEQ>5%efuezgS!8;Zc)R(7x5@@j@FP{i@u=zo45yh2M>0@(UCH&ujdw_0ikw( z6ivcCiH5`Bz(}P0ZG>39Akn4gPlL~P)`0c@8Fw2hQ-wd1z527-CCd-6t&>MK2yYpw zoo%d?<B5nFGwEF!23~US?6a-e;IVW%jM<~<bl9>EI*&JHhR%D7rf@#Lasp^-#i~{c zgYca<Whhix41{=+rP=z?0kM_cJUc+r0H}jx<j3;5W%k>d=A~_xg<S|5<%h<UU_lq4 z<o_!My<&DTx?5!8NXwA32pA1v#9PA`55POVzqI<|7cL_r4-)UaPgOVYc-skpplfFr zZ`5daF^Irqn(^+`S92bW<%0?v#e_fEr~*F=6)r$KDo1O~QxS{D7wkHTCV&+ClV|)6 ziXwoN_(+B24_I1atdW`W5S1B7_4<r4oht`ouA5}x_ip&-n6+wpD#s%m9=R&YABz7> zrIqshG?CTm27b#jZO3GH&;#Ql<s#i=xCLr39uFpy!Du*nA8Z{uaZKceTe6#5{0mM= zxIMU&)PL=ar3R`tdbqXY4X;c^RapmA2=1ZidY~VoeuNgMbghxG7E~&9)U(~FH^2jA z^}9ImPOm?vBW3B}SzPibA{tO2^NRT8zADn5FAKDw^IJD3dy2*q2BHSV>i1NHNv;Ro z>8akpTK0@VD&fWs=XI8eqtm2c+lO-eg#`IT$TM_8f58~v2}Szw7ez=;M4$wJE1&*E zd-v+Cas5x6p#V`O9MSdQ@1+I3WRTkQo){}^-jimY*lLPtT5L{?1OY2{cqb<hyUpnI zm}W0k$3gFjgQ2xC5Zt36!UaZMfQ3Sn^l5P-8gt5)h|BryS6SPFZ=y~OwvUxk?MF%K zIH^w|A{QPJ_oO(Bh;^=4$O@RDRyFE)A9uVL^SxZz`Pq@W;JACKuIRWO>77;JAym+d zt0UDNF7zvm##STRtRPQ9--X|uwNO^Zt%ZL7m|9qI!;V_W(AxvGP$hpR=d3H--L#<+ zBu9Q7)az}k;Na+W_bw91@Avs~Lk)?5TS1Cn4J0x@mypg0tVYt#T1HR+2hOn%MfUOB z&|qvILl+cv4v7+^U@^rX*%ZM!8vNsi?q%s|c>z=cx@I6ade1BJ=<;gd4+8ju08Zd3 z0Bve``5|0MA09tcbwH7|7f)IooR7OE$B)-*Phk(6c!aUvrd$I#b$U1aiSezx_C^JP zaoV{A&}D`P(}EqGW({pWOmEA47G-q}+*n{77Ne--v6~vAdq&gBlqDiTCU#5}90U+X zl%tU-K~MlO`3(snKJnI*>9J0agJMefxc*AQx<~W<@cO@g{73Y*Y@@rfzM&?@qu^)H z%|;%TTi*O!<F@&A^9ETZT?xwInIdmB2Nu9Z37vEKQXURiNk|T0RXqAOkjn#;<5<I- zR(*D5z%)SCN9cjdET{{*Y&5AEmSo#XfTcuwsP89MM?0LRMbVcjfqNmhl5bhjvvf5p zQ1afUZ|z~UWel*y==PznE*`nigv^00dLAn_8bbw;H&7xHMyzBfD}^nTyx3|UF4W?W zH&R{^!Q_NXX_ZviG+9{qaq2^V+M?+;_L-PfS3*a>X}Unu^UGMU-}N^{?jl`gn+%aM zB-FqN5t;zBZ+CDzP~Uc`6LFPY>K@3rcRpAJ`T`xWsv1h3Dho|q2iiVRo{cSwe>QZ> zLG^od1wK#MEfKbAjZ)tw^cyPgFkmA;xUewUf0?#RjLn=wr)hZ#{*35jJxcXD3BMkP z^{B(cY?F@n;~@1%da6^w17N{x*Jl;{)1GSr|Hp#}+}0Kz+cEQDzxt0q#)}OWKhQNg zzHK7@jOmZA5l8zh&-agadAH?==8Eg~Ci-bCpTfu;Y;##>@r`qGD^(Q2+c92y>_EFi zk-o4#J}`t2w@;^*F}1oKVVtQ-^pM`%g-XdWCF8LNRm_uBH~??=mN9_@7sCKW_#$Ap zA$OzxlQ9E-5+OREJW(i#i*k!6c~FSKKU`s2l&g%U82dcRBAg*II&bnDmD+KHJ|+4Z z7Nmho{)2lrYdV(v()FMorKXat!WnnHYmq&Ap53ZowM(6~+}e0PNic)11&u6F#TSIj zfD2myt`>2#>0J3z@BEO9xqX<J!z3XXu?2yZkwUo?T*ZgshcaK$MnoQB)rE&)GBqxA z9(ag6T?N?iJ|}J3U*wdwhPc#E)@1g3cF?*yu*VV}Lkh1;P(Nq`$#<XwCs9DS>kRbx zU7b#E=%U+H20I7x-b3guS0ed)co>)xg1*;f^)|&zL_z+hnFy2e+bXUpQi(2{Bs=_X z@ygQv#)oNKn01p~RJczOz|`%obQvVQG`G1TY>R~zBYl^NR)7Ar0()>q07i?i))>#2 zrEN;#b(0s}i3&_+O*i80B*FpI64X){-m%(Dk=XIU%}N>LN?OT6C8EIHyFUQhZC1CS zb$CfjT`lxs5i%4)z6>wccU`i2VqMgy8T@1V(YoRkkDMy&Rx=?=8~8WPRx_QkX^RFR z`{uZRJg_D-nn{PPxLELt9#o@PcfwfZU=5%pfOMv1#r#=B$C`+)0G%#0nyKH-TmxCQ z7I1%28!<W&34)>B{vQAU0RR8&y=!+I$B`iVeb3Q<D9YaHhS=SBP#W8SK$$N|yvx@i zXv-W7mZ<J703|lMda4=(S`^Na9~#RV%ks|1V|ncL*uG<1JG-{zJ06>Q*+1s8N$QjT z!p(@tN93cb8vv!1_fFd)x~n21Gb1A-BO@atWa27+h6mZ?RIi7S=l*5~xr*%-LqcBl z)R*+86C~(Oy(y6d62>wDI*DCQ6_9LDdY`0XFy(O(+3tq)hzB82UAX1-Gd2CRsyMTi zbZ)|2)RiRAX-G+&45E(rHny#7<cw6eSQtVNu#NgH@@UfJ*V}c+cm!c@j=6|rN0ygO z`5Avv_D9U!_pybO^Ffd#*YXjhamJ}DX11a*8buxFRYnDw(AG3q<;V@H5Xy-d1rZ%n zdudw~W&<14BGgzLu)u!Tk!6AOs8hPuyHc+j(>|XHG%aZ&xC)5s>oLBFNlxNJJEe1) zwVKO{ZGpsOQPvxQb(;niJsBNyWYwGpWMLyCXJvnvCe7)(aj=!tt=3aydb6=H2{M#3 zK*3}5SR+>t`8cvCXFQ)6cLZtgtnp22T5_khA_1u)b)$Ev6IG?z%#=_@k03=21A{pv zHWHH%Q(@5(C?;(&NRn=}GeEJa3+a>PXwZvC?HJ@-A=IG1*-84Hk<Cg^%e?e+OM&NC zbP8_%TBVs+zjygc!~qBM{0F}XI5?p@$@s}1AKa4AO<JB%Cp0gz{)X&|S@|`&ozqiE zr>9d04JBxcag)u8VgcRUm~X=8=Eg!(eL6cigpyJ}Z*H8hifjO00gn?-yu`oVj2}s5 zeY=xbvAsf`r)*488)bK(Zi<I6L}};pj_Di*@9N>P7+koRvuA8o8ukr`1sqn_w|5K< zmq9`&Nh`o5bnb>ZBPk`84Rht(rL*TRJwqHtu!bj}vcU<4-G4QZ7{G-~EqAFwDW|$g z>??bH5~Ut~Z#GM9v_{uiMbI^HDh@Ff|Fp%%=jMUzvkp~6`Ce}nLBeX7>)i~~Qn_?d z-O_Zy2NpdR4j^nTQfU$^a&d(apHvvOeWIQGm}nLz0WiW;WZ+`ZoreKYDEnZkRw$=X zh%EaZ2mV|W^?L;2?ZG>ngu%P!44<+f6=mbNixw~0=uq84J;(aOD4{uePwX{jFwd0_ zLg<XzV9o=fo{4QB{aTFlFnNU+{K6-#p%hs7gxAQ92auhZfb2v$vIQ5}MC}JY6ORtu zk89-I75|SO(|_=5xa6kosb%zvFt|SIa6S`^ab+5_-j9XU@~U;+Mk1B`gHjDXho>BT zFbIbmI<1T*ZhJke;bgf5AHAGfYHBHUa3v)VW1+$glR+IK7ldJz!Or(0lN8!LNE;6` zTt^$J>Bk`)h-+Bt#w6!Q$?Q{R#)=LfXv$2rpP3*-JV&{3;~XK}a0T6QjBx*du=JRc zPVhMzt>E+j7kM!yr5>U*@%<4WZW}&*(!t~yN26IUg}9gy92`$c7`do*i;-tVU|MRr zkt|x)R{k=}2D~)<X0N+qFFE4x@c{#n8|y{&#d_6Eiea9O7Og`Z*+KhWirmLSP*3`H zjGeF8R}usPTIvu+LYJK90qBzJxD`wtANIBnB%YU7Q0R)nzNEhOO1W?$Wvw?)%T4;e zl&mE$3;j~wk5f$w8ZU6XB*KO)Yy9vrQ8A7@oC;vBV)mT8XqRO#%)tibJ<c#zEB<-$ zMdy%*K=kpN{35`tdj?i*`kbwBfmakT2ZymclkI5p5`Tw07<w+a>jEXdX0Zh5on+6l z*H9gQNApin-rTK-bVg@lDEK@Ui5|#Ng0ZnrW`aHsZ~!uX`QJWz<JaRmzr6ST-+lD` z-~Q=?Z;xO5-rWyhyZ7eX+l60Nu^dR8!W6#$Pv02-=EwJ5`sMhg4+IKPcN2xqgP*>2 z|6krSXq0=!M8;RL2)Gaag6(jTGaeq^JQi&i0)(`J!n4Y0mg5}Agq}7mEBOgG<a|_0 zIvm4St*yAYyxYGbz9;!`RM!egNwApyFztt|v;q9oz*U7Y>|N&qg<K(!Gr&401|jX1 z$$x%Q#ivr8PfVt9m8d}K$i=_I=QQVq{6e-_uc=lIBn2aIBdDsRf{v1wp}ubre+vy` zE(j1^gHt@>V*#|8-a?1`beQ6G`ZA)4lIMgB3Zp#l6en|`w8UavWoYpT3)*`uXjxGo zckt;nrvw-aq_8T^Rp^WI7PaT!6E3kog)Za#oVv}_)?pgdV`MXNybLa<;u7PT@*)*% zt=8y<FFx=6{{8W*KN-LJ*8Ml$zxVd*ckjF;1=|mQ{NSChkH7ihz3;u_3q-SGf@cq$ zV>|?D*WCdXm&-ea+~N#xO(D{1*riD%pl<^Rbi^y0=-O$oR;9lFri}^|&cU&|ddW}` zID&NT<4x0E2*bGj2{xGOo3h20`?+p%T3kVZ(nibiId+Vr^U$DfaYhbrDK5-CmS7zC zo}7JdbA4sZD+m<u$~hFYv>*6j#15nX*b32-4M!e<>g5sLZWS6BFLbekszXk|fr(G2 zKyZ$wXQn^TWwaYJ-B(HJU(=xKLlWfv2mc_xkkk+w(Sx^t@!*$Vz5PY8lek%~_7lo| zzjm$8^Yi(KAZYNMi8GS&ck`Ut?6E77-i91sXH>%)vwF*SVAe*$PdT9QG@AkQ_A&js zB&|koqh(E&devsvYbKnioS|)-$BkT9<dZ3^GIEHAr8)ASY+vp=F$NY`(U=mK#xSsI zxmj?_93JiRK{p!Frm#Y5OBPzf063s&=wUy8e92Xz;X{fa9mB>Ou)&MFT{@a{=eE;4 zY=wp?SHLW=mcWj*E^Du+=fg*F7lyX5m`oN4mIi;cZda-XTG@y)I)O)*3&3UysB~&) zLWIZ3*y~>It3yQv*^=sn(<=A%)GiF_V=D8&gCiM|{xq$*zEic3tB+SotP!C$1jf$# zxl}$d*Wk=M8wk}keSvo07&~z*=?!pnF#Ev}Qjn%JTswmS@aAC9OFIrz3UmN*D@}G) zuCqMZ5IP7XToZr6L1x$+HDF7evn5M~YqT%{n}}k0G7z!>V%zTwA>r8;vd|VBt#(^B z$5cpGvziB`>VXyRm7&-Vl`gr~nztM0-D}weVDj@2spb}X7BN4gcCrC+kcwfZW`J4Z zz_e(X)xzeP444JRmP^huXUM+n$b<<R%v3rkm+D3l4G7ip&M+ppI1Rbjr^DvCV+-)S zR-8#<$O0FJwV0%2k3mVv5QLS>QtnBH<cMqPB#RtM|A6VrHhZCwi(9x3zgZ*Wv~8OT z?9-IJkhx;RXS%rpqF>8=;49N>(ay?kY5QUHQA~PkJ8C<t<NDsN5EyJ_Zy!iz_>^$N z)t2@J*zKg-5y8dqyqL5R3r_xhWs${g&H&8FGDaM#9YVH4()mJ2_N0F+&9nYChM-{X z$-8O4o9)uEf)4RyP8JBK&VoSDt!{EF?Ic&yy`*;)x&}t;c{vxYQw5<iZls$Zz5nKe zU;oQTFTZyG!(V^&(|7UB7n8j~CT2yX$V8y@F5Ve#36A<NNnf<Q{zd%Ty!YepkN@o* zxauN)wZyh%LrlsqZf?hEuhq%6!8}B9fK=sYllIPL)XO$udy_2Uv$;hGi6IvpR2T() zg-}<)MCTzwU%wD?zJ=j22&4;`RM;zu`!60`6j=41LJ(XGRN9ZD@o!!qzx%zrAG|sK z_CG3Uhg{1*S3Pc`h37C7hqwKpv87xNx<c!!D=TZYdx<5|RWP5$q$~2r(&WRR?Q}0y z6A;pMJ`y_mvCd)u%I`&B#^8~`#QCD;{>gnl*J7cz^+wJtVk;g*K@FT%K-=8&3M$Kb zw8%OVoa?leRzE!rL5DRG69&w=^aV1mCy>mJnP4+KwrG%JYAtTHTb)|V!>2WlRXA^2 zi=Nv`Fv}0Dof+3vTD~-zUZlcgjJF59WKlnXPtubVP3pB`IO-*}1{srPEHL=AVBM?F z&&|y>qT73m(fr(g6o3N3tFIW<XKv#b!Yr1s1>X-8!}YU1;|#fM8<(3vZ94k%PhQkn z9)Bol(%)R!ijgPWTOt6I^ul(97jmtV-FSiL&7$OCPAmvj3SNMJ;Bbh>Iryp4Cbnv> zE~9q$aG=|B-jRgYJ%|eELQwZH-Q$(x4NW{uhL<%5yR9fTWG*OE+9-=)11sO|=u@iF z&CtQlQ6x3s&#Kj%bgd%ZifTz|L-dg^KvZsj)z^@FhQ?wImbEn_6AZ$64KZ>$U@567 zBTe*Z&3}6w2PgEr_9&bsK}zXlPv{L0Eng^Ty#y42D2Pc+`?PDn2<JL2>!X=jB#N~h z5kEb{mU<)IIFxL%&hhR2xc0dq!$aNO^?P&`^><S+Z`t#~()!rpk}ahIhCm&1N;@Fn zMi*$@HfiL=NJF-Ubp`Yp*xIb)^^?#p)9GA+LZOq?RlZmKqM>1CoZZiUm`?J7l`drg zaVe6x1}fO~%jLu7;1_()vKMmO2&D&-*BfLPb+1&yRjS(Kum0%XuYP7P!XL->TrWz~ z@6$QQ+uBI*l);S7(T*839yn)!raR3HZ*AP#J=6HKd!I8xRYx-6B@0SbLel}cT*WI3 zDJm;?IRj?R4%(tjc`;I;3MeTx+=XzSG+?^4*=^G9x)JU4@gC18+lg-W#g=9ltXuFa z&LKx4%a2{*uVlYg8}wD!Aum5MCzedS)htA)?~0<8KY0$bTmvVqIIpdh<Ha5L@|+LB zPIc5KyAp_7zSkm%SO=7|B@JGv^3H_SwBFuMx+xm+$cP~qvs-j^8fXaOppdu7l?MU6 z)`aymN>hzi^@LKK6{%%lW2)x0OdrI{wkhbvXmuWvhcapjrnEO3P>Z8R23vqSPa{ID z7RjP2fuR*@ETmY)u@Qn${@hK)H0kWL@&1@RvY4&|SlME9+a-A<aCV1Vi;-X~`<=44 zXbjNKAO^TI2??3Rp4~xlAR^As&B=dis*APBq?H0wrlpUrazQQ~zARP!TDBOl(W83> zQQI5^DK;$$Vqqbz(<#zn@Kwq-#VojA1Ll`gLI|88Wo(j@03R6PLi@fPlBUP%sw4}f z;?1RXwp@trthXHc$~RAoW_-%%*Fbj&9?%>Y5#2^pMQpa!hpPxAT3(nQ=nK2p0Slvx zie~^z3+b$l)!OHr+M4OO&%#GL)?zvKg8N*uc~{Xq*<rVJ%wt{2b+HkkdJoY~wfG#V zsnfNwnv_Q&n6AI7dNgjA)$W+KhZK`U2;7OayLBG(jp*gb=padGLB0Z1@JWqEDNEth z2Uxm2VzbigWxLiCs?{3N#3`kwj8^RQ0ti~FKFO2Y6<q&Cg<}1uZ2;<3x}YGz5Rdn0 zH@T(aU+}|rR}E%;WcwQ2*6^@89qF7=q7s`h>##3#!tlH97<6D5JS@~)0V$Fyjtlo0 zxh`d_RdlE-6LSmFVJsBLy+>OKM-xYrRYPM^+52kHi<+W-vX^vrh7S2fANrZTz$_`; zGBPaCzskOGwq$My5a2=&7P)G4HRSm|o)E|Vk=db(&Z7dr+e+tM8Wl!QAVWnewMJ0z z<-x8T&rpC~cj&#n4yUd?uC!l=e;f}{IS4@qBDQ2%hVXzLcez%3By`7tLU1n|CJ#ad zZbJCGituZ;IS5&V<@8>I4emOu!|M$;9kq_AwU>2j%6d?9k=-|1d#T1w%I*zMJl!^x z29?dnq?P3{5BY(W@#$K@)b28so2%Qts$HsPGZlMMkBw@r&Zm6kC?w#?l1(f&LA>+8 zZdv<ziH&<wDl;;e54k6tH5>cNgUmOgF{7pHG1d=yiv?`Do)wh(w%W%V1EG973NS5b zF2Pby`7`=-6@g;Zn2>GO+i-|(m*hT(0h4DyApktpd5ZoVk7_NrFVLsk`Tg=sZ@kB4 zK~Py2%0R;2=<spxpz)977XRfTuc2b(1&0IOkKsl+%*z%H0v3Px6v^<^aG0MST@mCB zOo~1Q>fnPt-Pu9ajmkh%c@5JaPUty?{(vO~LUCevbv(`2u`((jjZUtzqmD<6j5z7s z`tY>5M4q=<XM*xW2ZE#_%f37XLN}7vDHy@slE}rhsQ|x%Id`o#X{Rg#Bt5(1c1&1U zZUk)9O>-eil#(@!ggDc~rXkU~-vAJqJ42qlKmkz8!0j!|wSDp_rkZ)@eqb0971w#A zY`x57OVI#0;Esx&0eDD2H)$=4FmKwZ$0o6o?4IMiyp<wavsUp+^SXPXT(z36o|P#} zL9c;xM3qOv_e7Jup#T7VK!U$3ywhpd*I<7d@N0S8FEDl7v?b1gS#)Ekr@|-${bYkJ zzlCftETqlr`9kb<hBB?;BJ8Twl(DKeKK_jeOG~1u8q3}@=Y#;nBc4&}{u}R)?|gmy z^4IVG_9y4h>c`Tf!emIo$J{w0h2v(x3*W@N+T2OIxJ4kFjif@@L&7?O!K&~7=Kb-H z|4u(tG&EsYC$?z>%2`d$p<4K~2df%?_qFk_zVB7lw{DVFRn^CP*tKvSe{~i4BO3qu z$9MnmcjLF;eej!~{^^5n-~Hg9#y@&XPjPt@>y>xZd6<O1d*@f<?|$dOH~)3~>RS)~ z{qM#vec&NZ#SDo1okS-ukBdSulS;UwYJf<ZMUi!2U)3dJvwid?I*BTV<m*{yzI_Iq z5L|jVb3iu9r4N)`L>V0(g&+Q5{Nc~;z4`X|yT2I!@x%My`#I3(&U>)sxcB|v(IW!7 z|4-kz|NfifSAKW*{hyCtfA#Kre|Pu&-#z%{FUGI`H;oS5(kMk}H;;EQo~BUw@w@_~ zzhd+@NfdIDnUVm6)g=MT93r20c?xNV6mpi+yC600zwy@H_rCMs|NiRuTmLG$`{4I) zkN@zEI?Cr6L6>{qdg<Pez7;)_XFG#v{I@S_5W!7E8o1T`^$!JqJ^0!GeDKE~+`aR! z1VAWtJrCagIXxGU_uv2Jy>I;y;04S#UneGodw{RMdGEEKO&IjeJRWSNomp5U85N2o ziMEEr!Q$-f?(S}D_e3kpH)pS1on7dH_AbQe>}SB)a3DsgnJ1f;KXi!`7QjO_70g6z zB>p85O&TpVECoVWjmz0Q-54syF=$;^+wLCxUtUqz>}Bn^7ZroJQ*kN1JA(gL6j2_j zIp$zAi%54*?A7czHHDbioh^B{L-7CRCP@B(sR}GbTUnmIAjTr@)fVmdSRFT_@=qul z93iCG(<SKkrbOqtdTgr$@1a@`U3=WN*FnDwxa`fVSe5g3#7bGMF1-5q8YJHA?k(|$ zPte1Wb&gGhPCD7P*rF$T#kTBz1cM$Tq{>#F3O$=;H<ih6XV7K0-XtH;yD#A@$Ei2u zM~(Q%Bo{=7ucpREfD{Sa3br7z?Sv3Hq#zidRwdrg)ehe3?S4dN0U=miO(2hPgJXMh zHx#t&XsBYLs)*$w>ADmXn(Kf0GDjCk#i^JR0>q*loHF5H?pOV^5Yr{GABkDHnfCoC z&Tv;X8;t1MrX3DiRpvKAeT<8uoEa+9s0&!-K&D60LsS-$A+U*+8^$@kEm)lkm6Txf z2Fwc~cmT&a)=qaTvR}n~&!*jkYYYx2Y>C&^ye?G{FKo5;>x3=1{?U^6o+eC*m91>I zqE&@>J!Pw0f7By2%QkMms4Ubb#H;~%Y~KnC!DrT#^GaSXFcYaH_Bow><Ba9#!jhjI z0BV~c<mxTYd8DkGJfkoFND(ldSu@mT+SdydQ=6j7u2Y7VA@g?X%P4XLtF%C^sJV<u zS5ZiXW5QTI^IYC4J)i?Vf(jA%2d*T&jceIo<p5WvT?mPK6^|#)F)H3cvwdClU0H=* z&kg+mZnp$`q3Le!gP}30v`IEzgO$PSUgA7mjx5Tsm+<u(m(&5pWlE()j-TC$QW2x> z^2sh(U2+DolxX}Vb~JorTtQIFV9ddbMU1*#)?YYE7)O#6wim93TSC|Xs5^!dBmfkk zzieOWP~g2d3`T$lZ^~SttN_MK2=dkDNoOZ7(pzwnst$<Lp|%vH{ae{h*HECnbBkdn zHV}#L)L63XkBogliE}dzIkPx}Y1SNvm&Nqt5o&A(o5~#)Gx(|J<wE>G2gT|$(IFZ0 zAyld?6*49)MWz#Q<-8-M-Jc2jxRZKH9&%=+;G+A#eE*{#{L81Gi>Sk&C?k1-Gt5oO zii=k6y0nNjm~j~jIWH~qb>O|ST$3K6jsC<QQPSB0x9i!CP*i76XFHfu>MB|5=o%&l zb6Yqy81zQh*e*Y|9xM@@Xyw+$v>%yh^!!D<$L^rot+9?ECbdbkCx2JpYeW#c0Q>?P z(N>afZVlPBMHn!A=OuxFM(st%sU#Z{f?kfA{KdY(Mpby|z??G@=;$@WC*cJ`>(zcx z`y(@&$rXu@`7r>tdg(AFv4MvwsVKq;axe^<2ATxeeLjv;-0kGx(<slS{Inls;0))q zbf@oJ7(pOU^@0(Q{7w)p!GSb3d};&e$O9co(2>16u*fXuOu)H8RwuZOz7f;=aFSDw zGrTqsB=6+bH+zsKHDIJ0Kt;-0Fj6fbFuw^Sw+WoP85+a5)te@HA(<HQ<bJemAn0mJ z;S9Mg%od}vYh!rt$IB6l`7hd`r@i3Q+{2)fTI*VUQ!C}fK?CqkAr^ePaSVj<;F4ZT zD9x5|oN3uztP2=zvz_rUoPwhqscFE`dOwRaw7-D}7=}%!$`@4l^rVHV*QUQT`b!=V z+NW~)m$<<J$|Z(;$T3;G@0XaDbu+X!uX_6lcO|U%;Buq*Z`)NJJjlG=h~U4;kUas_ zn+h@)HRr1KfC9pW6w<muglZ7Q8$>R*8a>@u(RnI5Petdc=)6~y7W=)gF!3>cy|VCi zyYQ4XJF-(Wp&4c;SICL_nm$h{SBw1_bpQ2*&6&8@*}-hoqOEzlhlieOU@&PFX#Thd zfIFD9k<G1hP-RqiRQ13*6$rKB2suZhbyVw^R5RkG?Mbybd^x@V25L*P&PK5}kq~Hv zK$4^Y6_`Bgh>1LuKRaLdF0kJJ=l}81oBuw3{on4s`z90p(0<IWu`SrV5v?|3SZKw+ zBZ~#l!Y|NDs6xOLHiU<%4T#zjk0*?`r&?<2lXMJepPdPgVc9t;<O3)~sq!&Hx$6hU zQ45KmCDSQdZ|lvd;9K5c-(HFi^kRR3;fMRlrr1T@N=OTwrIBvg1z^>)bUGUJx*MTq zDi)0QAemBZO=EgKPtS$K4)bL^qLL%;Ih7n?$r4^|W3_Fs+BR0(#%kNaYTF|!=~mkw zQOOaO3{<N^wbr3)PhtyRd=V>r5~(15K+~VZ8sH875WVz`ig4%_r<N8A`y+4KgFN!d zE*r833E}{vwP%m-U;Kr5_vA8oj>1bw)(u~f7&9K|NTV4A-f1)=g+}h(s;8e59i?2w zcjaQw2{;vs(F73;EcJ=j5v8*?*jm%z%Sm;c@l+8Pg}9Guv$Yzi_6eZsy@y`c<t`wk zCQJR?eWg}MwreO={<&iBb!*Ubm@7m@M~dt%%BDNpgwdiRtNmPh_%l-5@2P2v;hlwE zy_H49y%|DM`NhdAuN)))UWc9-C9g*1uhdSg<hAZhc@&@^p92pC#IEHL2^HwUDr0x* zT0{1u!5ZCgr>IR`bnm(M=G%Agyma@@zm9+X?)V39LE?wk-@g0)&+ond_v4p;KK|8r zAH4dt`(JxkJV$Wt0Co4>e<pXT@q2IFf8(w3EAMF7yLVoFgj=oL<1pof<X(M=NXn5L z7YL!h*-7)HUSq+^Kun5n2tt8-Ot~{@-oZ#%UO`ncI*iQ{oLhQR#zH~Vb@p<zrJNLK zt1?eF551T3ftPW&TgN;(SS7zF6O~<hp}7b5SG?z=lIQtiwBS58w`BHbSYN=fDw?!O zk7mn_dKE)f>Rmk9TX+h1W&eD{xC$$N@EnDee1o6T%@!8g+a{2+@~<s*^3`@~I5JQB zk;beD3F;U6?>cAI!2@vaQ2~WqcM{kAUq%<)N_*WL<CA!qU~b3*1NXp8zQv<5<3~Xq zM|w?Rm_s-gnqGSQ0j}^WK80uTCQi(-6Cyfk3L!E6)ep$WSbDeq!)ro>eDuLj?)~JO z<hyEW47#2=V&F4H+Oxb5595MLII3J7C&ZhSoaXQB<ls&jXi`Eu_jSV>%7gFxe*F4B z-uw19$G`rk2qP9nAf3PQM=gl`y{lTe((az$vv8M!JvwD=b_BM`eOeVzYg2cK{Ld%$ zM_n1U{n8K7q01YWVmp}xdZTzJ_-3x!y<@XH817dMY}L^h9`9>vt+o0p5v|u0htRB0 zwE?g498{And?bLC;3cI4^0IZWJY2Or(N^Oul!<#kdTIRK@3;eAHloxxr@>;Svm9yw z#1B^sPw@gbegU5q42q<lv_!>VS5+^ZBna?vx=Z10v9hq0??yK}B^F@~!UiWu$^%jE zM{i*Y^OdyS1FP2EcmHnut=A4m#);6ZBA@i9(IM9xfCevh<k7IHE1-Da?VqaXa``|` zJHZufitgKb;8oB#-a(H8(2uT*d-ZU7C$j{|P;}tXKG%pmcA1K&Bwrnv#R%V&(?^ol z_dH`S<(c{fuD1p`;Gc!|ORW*vDw1QrsdST5I0TERujs9Y)oY9a7E%5ig2At4EA*pm zvKxkGwmVc=$_E}ZNk|Sb6{z-1Ik4oYI<<$YbOb3|WDg{BX_Tw^rgC%+Fv3++)rO>r zs^+Hx3(P4sJ6$&c>Y6)UoG}Q_+<BWX90vEC;23jOa9=m+fn&<Y!G8j^;y+%+!{ES! zFK)z{dgEa8r^RunAW}q?=j9Y){h+MpJWE{_@Sy4+jn#66Hs^AOJA{+?!-YQ}hv-Hg zkV|lYEA`Uwv<Lt@gp3XwLGz=pzBc~O-<ELJ(QntlO&}yt`SqGla9^P-sAR5Ao6ID# ziBdW?qV$M~8t(o6H{<ty0g?G%e#IgC_*Xx8*p#1quc?}#;l0a5vbtN{>4;k)vs+}k zflC}@Q>l>Is?SzOD<$<bV;IszI8{S}=o6%9wI&v%=_jLy*;EWm%1<F3r&%*TagD;E zC0IMF?{Il0QRnTXJ6!NNa0Z%bPJ6J3Nj}40Tb8tbr?=Bh6v3<xs)(;=#NuMM+>v!O z?5v!4!Et+VRlamjj^@<W-?<s55L=I&CV_>$QiT%rU5uTy=J5HLmoFnZ<vl*pPWiik zD=pH{jc;VqDKm5HR!CwJZqIWsU?$+Vt#zpjnOF~uXBZL9EiFgff4G%IZB}j6iNzRU z42$;2c59-0vvJSEfea%9BDT}bO~`Mu71MqDxHH^|dr_O!fY%xFY#4~14{<6B2q)fc zMfEdTzewToL>zVKJPr$qmc^SOG@^7^M38DP$z|#~4;gO(@1%(V$>aE9Iz7q);P;WL zZp}oecSIK;0bd_sH^g306c`RE`(B>G&25O#r~I%yRG|iqFj}hYFc5o?<apG@?u;<* zG#;EGK9tk61ug}3&1gY`kA!r5L=E~t87Us{4@_WHSr4OomK12AZ*^S)A2MW_Q?u^b zbaj^!qEer6x~ZhNPVLked?)y@PA6dMlu%x|n8gSk#p3MjW;)#3X+xT;+3gq7PAAU0 zv;C}_G>7SSkwKVNnBHg4J$>p+7p~p7c>dCji|4LBb58X3#KMxgjKN<&bLzr{%U8}{ zdPabJ4k5JU7nAksPX7Xm!Mc(ADB7)E&aR+H64m=)LG-G&t?|)|Lgq(*fyv3#Ls9PJ zvVNkaoboLIubw40<kxn5GdWd=NsFAn)34)2d&m;fh_+Ig0^6{4vm3qBe-;FNFB2%| zb?Y&J9la8&?N%B(KJXfcqMhoXs#g)k@#AX^y7Rx<%hqN*qm85#RrjQ*49&<s9*u<{ zo6&p#8McpVEjQVt7VZ!T3rz?<cBGaDEy=;AhDKar>Sf-rhx^0EY%9emz!qPM@qW@B zS6l?ahfj?aYnFN@22(?9r9K4Ig>K($GFz=}i&ZWko$nVAZYArli23RADMqIC*_Es+ z5Th4(vX|MhW8j4*q^2G!GfOhVFm!lHL3%wOq_;BDdxWwTBT7%QF~r&8C{wxg@4{lv zyX~}V;ZY+1sGg*feh3&0NQNNCCGwLePqGQBK77t$4;1SO3?<V8xk~wIr`$Zo_)mSx zmJ3M;Xy<WDa5*?Uf7HIB5-_tsV)aNx0P+($;}wk)fzUjXF^N(q40FmS3u_+95Y@sY zR0|%e`4Uv~lTgjus6qilLSc!dA4CAVSzAhm*K0!G>rEfUQlKgD>3dn+;ZruOtZSkc z<ifJqzX?Gt2Nwh=g?o|Qs^>iIFF*|R0}?>xdcVA}Q6xN0ue{<Ae4eR?_Ufi44Q~d` zdbnmNst1wnke62>4Vz7b4b5*^?~|+)TP;efh1*{U9jG5ZR{sgo87kdj-GnD0C+6`I z>7jti4mveD-H$0}J41fb6!z+<#BU*FT|d68=<cAgUzDuld>kdsxd=Pr05;7428sG< zAHk1p{G`<(T4p3z3J&kGO-X;4HGsh9uK5O!)(6y*Yif8>V)E30ZL4{L=86VQ{umcy z+bld$sBBL&qqiMioDdtz5I?9w!XK$XYD$I#@>Ng={v0H-tIw3o<d0=8{~1o@Kj+B| zf2sucC$a$lIm+)(+X@_%%AYd19Gcu!vK2ldc|46QcenN>lKEP?%{Rv($-ABQpF#N@ zEr;z_Oz7r2{S~%3m5(lPpHFhEh&wk|;7TGlW|_&(4hs~<h+VtaC~Z*PDR8<3L&XhM zxDnmZWo#pp1X$HAg3ZMZryEwCKIjEEd#LH&@ql&BSkjVw6AUVaeRL(+)gU>^lFf5w zm781%)r=N`^WLB7WG<WsIc0i9J2emw?AaRF1sDkIGG}~*FTN;oF8DYXCgE)QI7`C> z`pxy5SGH4~xZ4c;)WHs|_Me=t($H?(!H*npYFO`ArG?P^g4vTUej-eRoBl9^OxJmz z+TI!3!8)&?ePbd@Sh$|6Cv)n>^1${=7}<Ca2=Odfc$7WH$F*SN@(nk<BCSWv3piQc zTRq4K&WGki$dJoh=C*agwY0b`^cub3Tu2JYFlN<NB!H}bT%~6#-NKr!+RrQ9p5jTX z^~DyJFnmH%^2ws;)uhKZsq}Cz2d&+enbFPgF)I=4fEl8BWO%zf3#vQj3qV;N3$=CO zP;KRXeNyqzf^Wu!sc|9Ot$P%A+FTXVHH7n7F}jCAS~7d5YN9*4Q_ri<b*Gf5!tMe> zpP&io3~^e@_;|FxlNVV&AKBs9&6@>fUS4oscauUJ=xsHe>WBCIa6_UKk37a{Hmh0Z zKxD96fc=N#6G0r+Rfi-Fi@;}kUWEqxt53MW3kPYiMYJiJY+;b~IUV;zW@Rs7jBjv- zSbul<1g=KShaS2tlkBCzjipaO!s#%jez6dE0k|U*!rFe>&C1j)`>;`xTS!7qw(?US zoA9OF9x6s1@zQn$2@nREX66Ja<oJiVpO7l2KKuUTz@w_fXD+cAnJ18HzM51amQWu~ zsInPrb>I^q)V|{Vx*X{jWj}z~%(M$T%0dGVqK8j288pNB+Kx3V8fwKT)pp7z8sa}z z$)?`*0TL~xVpDw%EbmK+qltvLmVM!7(M(k^0gh>Sm2i#{^uLEos}`pz)^pyMX{FQT zu9S5n*j_CVnA*$bxz=-K7nq_uH773|hLbB40QEV9B2Y;S_*lHWP|3>+6XEn$fWb}T z=1<e=G;sy0X5~ri$3tdivl9452DO4n$$GNHWQop)#Ij`|AIB;K7?}E4ra@U6rn`W9 zNEx)?Q_i9mD93q&tF0lBC&@fm=Y;|m?>~2i-`xaHHZBae@@yA<=g;MNmU|wn=&@{r z)YRq|QfPFteitJI713@YQ?utuF~B5BwF7L>Khf=9^}c>OGr?gu_9grh+Sq#>Qn9fQ z8GB0i`pnc*vEg4ZJQu`P472Ssh%;zdfY)aUolO@ra^&^MC&j#YRBAdXV`8T%*oUTB zmzwTWKk(p~R$Emuz6){N;mi#SCDMm_r7f@SmRT}F1ppflIcvf&nvP~J`6sZNe4O0` zAx{!Po6U(dJR1;;4s9oq)G(4g{e8Rf;f)}k9RpVV@GYJ)bA~aMl)OBmS%Y`HVb7+i z%^IaMF;wi~@6|sYYU6Z94e6?Fh(3W(Q`&z*0RH~!Ka3jXMhzp1xOpQmI|64pOpHA& zJBJe3Q&|@#!A)yjm`2Rouv3Y9_0eKqQeY~nuRceU{_0b@1D>qx^3VSc%Swsln8vR1 ziET!;@HX0$cd%$|ENCm7JJ2B_@8eWf7Hb|&vTRI(3t2Y8_>MS_7N(m=hq4^`Q;2x# zp-rHLPp4&aitPgL%d{%Sf0%O^hEK6M2byP@bOIN^n0e<vwpyR`R;%3nRAwALwD}2a zG!NDOG@T70I4&Qb1;P2MM+HMvXm?B4v8wM&48ZoUgAJzjfP+d8QvoBtExC@hh3{4C zU75vf9?lihUdDvPK7(_v(Qc)RIh~NI+?zH}w&S!9!3e-zT3zflNX`2vBilk}SpaOD z%=vb%6x)Q$=nTiS_sQ94PjR~!Z<_0Bf>_GChylc529<b6FL7YHt4~%Cb7^1G!T{U5 zu_Kbea!Q5U2UvN~&VpPFo`0%I&b$Fs1aq<s4SW<D0+}TLCJYe{4`Q5Zs9b_!$+Goo zsHbOLUb&Z!-%Az@^S-v05W~gIWzTsXKG0tOjaF4mc~a9g@}VnXuwrbw5Zhl8B34F{ zFo|UQ(RNl0=>o9K=~jqIpG1N{!&I!?Ba|~q*NRl}e6f*syS=2<PlmIDv?${K=IrzN z<9nam?rzV1o@e3290+HK;1Fc39yg-4C7<Hf3%%r4(!&L*Eq_=HBB2z3$2<@bhCmh+ zjPJuJfaIU2&wu6OxoD*)7BNU)M`4s845#FavU1RS(RuV+&2A<6hQNT)I{s!ytWB`u z%`VJ+{_`guUznc-qKOe|rv2vg@vXS%<mq78q>wUA2y`%N%G#QWRL!mUR?>`{BsH3I z%>tsyY@1l~!3H^jLvDQ@{@?Cp?b*ja*XblrJpOomVm|3U@x<pox3Q5deBm!oZ2aZt z;!gYV`NtnmlDUNw3$sPu5hFTFqW<~fGZ*IP<`;r>%9|c>>&6X3wWgpq`|axsK`*}u zN2`fr71jw19D&&@mU&Ce)6}d(EU)rRjQ{rK@vq;#`~E+S-}{HCesSiJ^5hdTRM;b- zv=cHn#EOpBX+;HtUOr#ott*NKwb)Dh30uu?<XMI?-^;krO^ZP<9$iR_;e^*0FgzA4 zh!j>vwq=UU3lg#_RarO!qWb$#gB>C%DkE?)IEYI>Dc+#c#lk`U|M<rEhd-7fAb<FW z@rOUV|4-kz|NfifSAO@=oBuq1<@+urbS8&rb!X5OGjJJC0oEd!KH^7j5z6tmUc2|h z-`@MjcOJa`^LyWW=l=WOsK6wHnw>}Ha1eg`<?(m_r{o*Rs6faX5yV!f7DRXNyoZ}% zeEt33jbHkK=*_+Fz2c5j^x*AZJox2TZ+}s+-_5ezDrbqMAVG@Zofg$}*09^Y=}vVj ziU3|OEyZRsWT-j+P`!=FqL3~9`I~6j`7QS4ZazC$3kYqPv%Id>?aC{~kvA=us<@9h zB*=IF?@M1ZU~j*E?|=Q2-U`0@=<6RsuA_Ipe*X<fkOU0&kMG|5x9^FE<Ck9v1UUWP zZ|~mu*Ly$tZvvQVB!0(uQ{%t?_6n@QI+N1)-QV54^TxebUzL;M^|uAfh_?uJ_mBU5 z{Hr%1Ap&yY-8;V$I88e-OTxXMynFApUsG#EIcmxM_kQ`O4_=YN_+<2g_>WRLee~1! z#_zrUrw_h$?>m1lzKs9yeSv+-33%{n<M5p~)PdlK0hX(fy#@%kDpIU7iM3=y*z&!M zdwhwBhFj37ZD~on+A6V0T|U+Cp6Yhll&mXp+Q6WV?u08-by-tQ4CD}bEYc+XTc4e8 zb>hCzd{%Y_1Jocf`aNjrtYE{_5B=EjWj*3UnTQT`dg<UwJlrZm!o-#iNl!&>6m2?I z&Ie}Rm(fRHYGdY!FJU9=z6X<lkWoUNhREVOp@ME5@y0O8gH-^-2u(g;Nrq=60mKN2 zB8nWYV6<TaZPFUDSYYmd=bPgXe>47v*K80xDNy7Nhq+X5!IY-HJ86bkXKk8<&wM$` z)rsaY>0Sd7z8PJd-Y@xOC@8o^N?)VBnt-h>6{2ngaf`8OqNJNZU742cW(4UakgEPg zT2h0g`{(!WzW1HGAO2A+1rPrC10m*ZI81C6pp{*v33YCFFJM4-OA>pUu#e%RBQwO5 zj>hS6u#87(r_|)e`DH>}51*>Y0w4^Ea8zL@G%ZZ!VA2Qp96H{tRZ--|oDH(d(`YFf zX3=(>--M{76opvy27`A)d{lfAE{u~~$|NM?Q$Acka=9QlATonVNfhKD&DE!RxsuV~ zONIKR`7b(p5r{(s>3~8J`Qcwp7lYnD6D-tNIw<M9f%BHk)zBhJzR^h_89`+>CR8u$ zsK3qJd*vS=y!5irgecnR45)HHdG~;+RuUDzmnB%bD)a|?$@OeBNDrOTf+s%Wq?qh> zCp5^30l#5U48=^xDoYJ_KyxdCyXeJ|^ImRHQIMI};Q<)G@)O7yf+^%c99NBmM}0~r z_oCcY(XW$2>rHxV(fAi%yZ7TCiGAPw|MA|vJ1@h(sN@?P1s|Sn(wk~{qGbx%9WsvH zz4IFNBk56vlf~P#`+4{%ZP`|q{WBKTScIKAO$b1jMSgcVf^mlPv(yR^(l|=ZH<du4 zjRbb!fRh55ChQN-w-B^wS-XewA;QK=Rua{Tv6-N$g&S?_rgknQM$<-Q&%Me_WvXd> zT<zk{i1|s8IN-EG20ns*2Vktg5h2fT5zK56oT?##$J4Vqd7)HGVJWNQLUxEWDmH<S z+(bXE&UDSNRyY(zWU~0^o3Gw`;}0bgP)xC`D9_z4Sp(NOnyf*oZ?~GKg3y`?95lg3 zb<x|pD#I)qil04o_VRN#u3Wu*WrggrfTkvjbhF*ly`8+a7|l1Vrxb7LD`hXP^A}(O zT?yhcZvyz-mmbd3*<LwX{jvnJ^B$3Nnrx)<y=6Y{48;r`<k<icqBbH7ZDpMgaZ<Wz zkySZ6H?m|dL=`<I)Ux4Tt2<C>-|Pbf1X|+FJVO>{NPCfVPmj##UAlbf++s8*yS#Gk z`h{~hPM^PY){x=Kg;VD*-MDt{E7umI1)ctUHac}HOS@pY-hi|P(QYb~?zvf<rY)-U zx$_q;+&F#i#?^Bx*DhZ@cUIK;1;b2O)tSo|FJ8W+S!$K4T;)8~i00NB`jHx@(9`SA z(=%BwgYvb{%+1Z!)*1>qoMW7c2k@r7GaS~i2t541va}CRx6@&g(_{R4Ar*hbi}|^= zNB+~AB0ykD_??pG0kLj94#G<oyKw%}xygkdpE1;eJm*m`f483PloK3;9lmTtT4d6g z>9U<X5!;gsqQi0iOdfYrp&t{=X6@%G-@O&e3>G`_rnRS3t>~#}E?T5#OAKy70WL7e zc?G$^Kt~dC6hJxRa0=yUAp~h|jH0s#JlpJL?bsIM(EdFkcQGfzZrT-$dYpdngFS4$ z_-7IvG<~F?wh|21Ex|?{4A6}hssZ)#0bAH~FS-<8l379^`-0f<p6_p@{ZwswQR^rK zP<?Mkl0mf$TQA=&%t*3pl8^Mukt(Q~Ffs*43;LxYEbijb=&4vGBH<I{BEdtWhNxHk zSuhq7m15aKv*h~z5W=hCZQPrNEop?7*>Ew&k@Q~X)Mu}^Q&3qif^|iFtz+qiEMz;X z1ew3}M1cK+Jgp%)SLYHk4?-%LJ`GDx<kE5mg+}B8LiFF|#tCEY3=xgFkZ|-RpK{#c zuy@p(?x<J5g%5fuPrb}w?x;^2DI~Nc;eYVb;%RnM=w5iRP*FOZ=sO$4ofHyn%^$oh zwEMCuf4J(Y@sp9PgKcXa1PypJ7V1=^WdLvuS)W}86ynd0&^3ibO1kUlP@|Fvc?DE9 zBe3UDcF_nH*t2o51^zij#!}6rpAM!V3e=*q4Yfr@$<WkOPg<KTvEAAp2<=HMet?}8 zNr67;Y{j`)9$jKds|3;P<csP|&LN&^=wXSk%9Bz;%GE)~PpdyiC@xi{BV)LJC=5V_ z=Ri7@&4Ke9l1<flXVHm!iHrPnc#NpzrECStERv?vV2i0rC`Jn@sJ^JFrkguC%f2N5 zFA3g?H#X3b7ZQI-e=(B)?0{kes;Xcd(dqRKp^HO^!1XB0>DSZocG?@Q!xe!76gPah zE`+;S83g0!3B;^zk2Ds8Lc${Fs~^vy$XCB4ZZmUMGVrN{IRz=*Aek80)dc~1BbYlv z22esTp@PPwYW7kwNNP?sG$Nrp!yJ#+!F?r8`?JvZ&wy9v)7X5lL?kg2Ce_kfl&K)x zHI}XnKwIeg2j8NBV32}|pf7e4aM_O0!Ef3zd%C*%^Fy(c_67yIKoeh~_ctDwi8W6s z5-Rzd*f|#fA-)Pm5|pJ@Qq>0_29D{ct4YLJpJ<y_bi%s%1>`tj+}ixYV6Wyiykt3* zbLG=TN}84xZMrHZeSS*_C9r=&YlO-rw}1V(Czka_g!8htmHU(^=ksgv=Hzv)VwxOD zr$STBk@PCSAXk-A1eg0lk^$AweiL_Y!f{#OmqX<>JUfZcO-EGvN087$G=X9xR+^ru zUHpnwo*-S2%r~x>6;f6vazu?v79wC>G}~a+A`MtJ$2mGU740c@a?+-BLGTYiJ*u;n zWd$Cw$^`(VH?EDeoF0HetQCjS4gd#i9Vb${mSSp1u*yIZBlwH?vEa)vBO6XO9A?`! zU1SyZA#3{Q=@|4-v5vuGpd#*s!VLSOtsydZK{Y0-J%RpQ$qKw9v}hV#q5`0@4O4m} zO}Y&l1~(n4XlNX*GaK!?*XGOZ=L4B>p(&ISx0I+3g@W2~2(8j+4=+DX^@Or_6a-&} zn-N0kn-Wnu_nl4YoMbV2;*f&O*cio#qrOPqAC-c!VoDuN=l5*n4+YC4-M^eX+Hn5x zC}bJ(58XZ2;{UJ^m6GbYt%URkox>PPo7s%(ReUQ>doaymaX<%_xZjP=Z^NmR+>;p1 z+bk6*y~SQz%!nT32QH#@nEThnV_#mB?00p$uHoLr*-LR994e!Hz+(3g!YUd_nFrzu zTP+qoY_zEyG<V9Nj7!*7qEjH_q6BNF5{0k<TSq~I@avx-jkxfX%_|#EBm?V_riv(I zS}e9F8#{0ui)}y~2kL9014+-2Kp=n|&pSN`r9awkXT3voy0o(zmDOp)0KjUqYZ2$s zEx0@-Up;D%BdISmY>WEMaoo%6K}5x;lV!PB%Yap|+v}_Ia0W8BCibN%)(8z)r<G`3 zqiO@4GNKu*_f4_J3tV|W$u;-L)e(|l^QN{c+e3m%>{T&sw$%uF<e=Bcs;W6XQo33L zk6;v6sNQj{z@LuQDhFS75$jP_O``=%X?z4!VqBAaN;Fl%6VMz@fg~JYjovBBkeneA zv%WyqmemEpglp#e3GuTLJ;6B8s&$INxe|9lXR9(@kcKUG^9628@)Q!KTT4KtLI>~_ z=x=8`LrKX-gx^W2249q3Iw;bj5eXf#lWb+ZE+gawEixBdO)+~bDpoi3x^O)L&aPYA z*$ZNS7`L)~b2jNWzqB&j%{s;Gb4h#l)RptIXYd^0DqX7UZ^|?2g|oACzyV$dFj<nz z0vEH=P2x%+8~fcVUX-qBW!3M)(T0mN1(eQ#J~-qL^5oVz`z(gCpwRiO<H3T?*P?_v zzNOOOpikdgSdJB#ou&PODTR`1TP)jxdD2|Sp>7`UwyYAgPCzDmA(}f|7!d&lXe_{o zDeOqa;|{H_D5q4%T6R;#zU0hS(z%Il0;0LcOxqz;$Z)T9<3>Cj=Gu(_?&><O1uf%4 zU^#>ZN>5!j<fj)nHLr5lqmjUn9ezpQV^5DMIAY+yWrNi@-**|a{#aBKgu;TWy7(HB zNt2SjDD#>pZ|3uYU3+O4>juXVx%#42m~J`~*9o2OX0(xK+gL}M?G)}>#Xe#fc*~Z3 z2qd`jY9Yqg1sZ<eM1PSu^Hq*@p5tE$(OPi!Ws?gZ&3Kdoc7iu*W|`t)7qO~J<uQa^ zEj<iZOz-Tf>Dj?6mnVfz?eb~3dJ>nXdV{T4SFh06eizoeDo4z+)}d^5+J?`nTILIK z-AuwK(ASD=C+{RpxN}}Bry4r8Y{doHo9ce8m|IF6HR)>flO?~69NK)J4W@j<>aDXT zkljg-{%vh0XvD%7;5y+vs?S8|Eyht+-tiS3B?D`T!yh@E^?J(R*kN~h+%D%3@quWl zo>R4c)=w0%ty+~)wxV7+No36)x68WqRMNGx$wl?UL&RZ-f6|0*39jJh(6fnTT!wpT z9A<BYcW>_^zwT3!m0$^dEfe@~Hy1pM*F6^1?;N(e{ZX_lv_M_B;)9##MTRYdwHaM{ zjt%2naEIgVHw{v3f5EXLLcK1?<9nd^tUfw-cechmj1RLRO~at~++qz>xp#IS7E0I= zm&5w#T-QfH63CPYABd}N2!Izc@|`3M3m?d?SpF8tYGK2Xt1bYar^Gl)nKWg?`Dc-l zXSFYN101;WvL$E(k4tY7K#OH{%U%nnx<%l84Nue4@lfn+iev~;LZotnZlQ6sncfl{ zo8eZ&n}oCjc~NTj0Of%^wjZsE=W7mAqcxgwGx&+w&bIKug+y#1#13CT;Aq#VvRcq7 zA8QhnQf{;XI)rXcc}|$n8jRyS?bW9<y{yk<lnPN$BGw>I8irsIeRm1d0Zf(hyr1kU z1qNa^VdI(Cm35(2!Sdcg0l#~)$*O=S)rA05eqi4A%J=e9ZQV|WTUl2+&5^4ZjA4#` zXf-h>Lw;#uvV~j}o!yz|=0<h7EpkhN`D1064GP&P^MFAHZ%=B#%I%8XEW8z_3I?Q_ zK?<QoC}(lQd5XU#>LP3mJ+D0p#@)(sj$Z)>KG;{;?tB912>Z4v_iN}QS(m#r^eodY zvUNrCd{siPscoIkZLx3SN(snYX*a<kq~%w+D>4pN>L#M)FEo9wi2ZJo_btcm6im0k zRs+GDx{abv>FPjdvs)lSx4|tcr5h#OC9)Nrma(ojRa3|&t8sOcPA|^Mpxz-{-EFLR zF>1A1`|HMJY-6U~rF6I>n{dkt-h)WMV;KG}u@}@Ibhw2tdZ#QJBW-Xfle(;kbo#6f zh)Ol>sV>RJBDo0SlKCb?Q88CF9ToFxXH{9CZd%{48Vq|8`dTxt@+#2M7wIx-q$Kc# zc$DPyCf*TpLhvXB3acw)*zWFbi-oR|uQ5W?tmW-Y4SgE4j=I}=b!m5>w`Kgw56ADn z*Ui8qO|~xW_f>PO>zDgc3VgH;f5v}!`TiSkSwL6f6fEBR(SW|_zv}Lo1ugBb5&e}N zah$f#iUlF<HWZ>p)E(s3R2B<@fB`RJgr~98YYg}1mMCDvadTh+X|HYuMb^%`?X^wy zu?0^2vW#tLxG_Kw?+T4?;BDH411oanpxuVpw=~;%whN9FN&i-wXMH<_8N<}lM`D7r z<93ntc7{o9DGUoo9FZN}mcJf)v2XcXMKR?AUK{L{)uZXA-N7ZW3*Hhd$El%UqPA#t zz1EF~ar1?|DR`-wc54m6P&0fbU<K{gJQkU|NfN+qg95QQpI}|q4~I5mlhFxm7JZQv z6|cy4x|jB6b>Ig$Rx6o4)?){YJAZ=HWmWWU?GY1Y4O?q31XUBJi&TJL;*BT<{|Zsn zO>U)~<Vw1i^sa(sg>kdgaK)4_RdBm(3I?aFeVa_W$L^rRTUx*bW-%6Vo}@!sF}UE8 z`%M*^H!VoN|LnY|Zzj;5w9B{_nh(!g`sqGjeYzZ;F3=(lB9}!zHTk=Nncrp8UiMP< z$ZA|muvO9%ZO23oxVCAby#xh6`syFYFTeKDPyhX+|NgU&Ui#MEJHLVp^Y8q2{LcTf z5Ycw4*4$0nH`8GgJhGaExdv*8Wtg#x8~{{dA7(q9ttKqq+0OKZ;z4su%*!6k%_gc0 zR5agg$Mv}e{%<CXV?ak>2tS8(Ns6A}s%AGWV2U-t(j(x9&AvSL4q@4j!T0irNZNQL z*hC>DQ90sDF%2Av+Baiq3pCub->`RTnbgaV{hA!i-IAdH;ETfF?@@>}x(<B*jrYfQ zzAo73qo4ig!MFY&vkMF!eHn+eIcMibP3egp5AGNk39Ona-q;5(jlcW$_|>=Wz4X<4 zZ~gTCzrE^K>$;4Xdbz#Cs(U+w!2uT?7JkI7fBd7j{K|D8Kne}?rw_h;_s)0j-g))z zotMUUetGZrzaM}16#)~2;d}hj2jac>_rdR89skQO$M5{%!5`lh-|xNooqKQo!R>}~ zhRmqf`dp|FaB}vv5a)?9vJ1GS(2kv?x)7ZcYUIeZK-cOfVwvJ16vCMpVie?M5jWxN zDcrkp-FO6S9`Bxqhy~`hFWhSva?N`xRyp=eO3l7lHbCi#doFIFgUKK(XCRqCsnx*> zHmEC!n(B#ZjAu50JCO3Mu=+N<n7+zH4e}ed4OFaNTd)JQD{wg+YGk~sX|)8IaC&EB zBgrd`DjqJUJ|lO`B3x^WC(WAB$>b8(u%wN(FbBjG#~bP|TUue^6Ae5T%tpY`w(u$r z0DHAp+OKrnRiy->94>!NnxHdB@}xj-eM4sZcIE+SE;Rv0r6uoWnVP^r0+jX%YcvX* zuH7lsbqlssUuVN$YbxdHw}PnFzga(^f1z$5XH?h2At%?T>nvqT_)N)|k}A&EssJ1h z-v0UcH(#T5*X2<buM^kOQ@!3A#p;aC!0$+CS7h53I#xtWUep$Ao`(omVTA^N__kW{ z#<GUW8XJ)*<g<MhUo$Z%zB(_}5Nft@-s^!gT;t6uOTl12#1_R-KwRYUt>hwU2p4wr z#AP}I1g}&CaLRhCa$v3Y=YP3)0N%6P<1rxG7x~1mU{PnG2!({K5V;jc=e15Y7`d{- zDvXOe*tP=3VbO5Bovqe){@Imh-!P&4w~`TNnSLWbrNPM1?gpa=&Zc#&HS#XkbK)}I z5hKrj;ej#t84m|C)CH97^>QvPeh$5a9<D(AAvS23CBMu&hg>AcJgyC}K4eAHtT_YW z_UuClF^N1|GaX2nQjM~+KBmdJpHca%U2?>vp$da+$*VfnqRl@0Wl=QCJGy*;@!R)? zQ^MKLwb_Byi&o!~h^|#*G}#Q@f?al<j|$kLt6Po^L_+F#^1*d!s%jj#<Ap>l$1}|o zRPaLS5<3jLO`em`1X)?BElo`U+quaT?k>2Ie6lVn>ki#z`Ny1b<q4+3IMo+y1bxTm zh^9$VE6r}ZX&-_>q14ho81j}FxGtDF>IB*f;}x(~?_1!QDlyD37l3f$j70pik<2Q) zWm*&h<LvzBzwjCSD&E4bt+}x9_!s8C@cFqf7<}M8A=|NCT%FL;5{R<43f!t-_Cl~^ z2$t+r72u+V8H}8-^(gqJxwJ@z*V64osB8GV5zYUlIq!jSCdNZvQvR=SFn;0cm+DW* zd^Fk~Z3nx2_S(e@SWa$(@d3#R^@M)y14e|roUh~^!lh364}b*)+9S?MjNrKLIS3#? z)4{6AiajPd)&eNE8Pg#-_~)MA-n<O%6CR~NEIReEIi-)284g(L+3yxVO_X!c^{_Ap z1g0AI+*S_HnRtip?|cakY~&?b@{I0faX0A_0gHydQ2zW5<O>4p1zbK6L*61p8Q$Z} zfow1x(_ELOvst_nF2Ez$vNp&R-c91Js3#jlS*Np;Lmc9dfomkLgECEo9D;X#)Ge*a zk#kq!{>!=E8iZ)ZLt(s0gT|+I8-vNWvf62{MfkK>)MwtE_LX=9Loj2Nk~a{bjC{Wq z$uBS*owlm9^>AklY3!F}4En&dX+*N&L;&aj?}U|*E<E2D=a%vK^i$Fj_jChHGPgXi zm)uRq*}V~S|3)8Q;s7!CXzXv?cp*nDsxqewLY5w*j+cZxwz?#{Wwr6Lc77b9w*fjT zQj`<Tf^gL8JSmXL4H}I64O=R_>xM21(jzD>FjNCfQ@9}q-WouI)<Y0qHi`jdnR_me z;Y9W}k!6wb#iGp{i{|UTx98;T@pqik!J<??s?V{sz!qN>@@7<*I<6~NWxFZxGwZ|w zQdX)4!D2{r#EU$O?3DF!!Y=Ed2_(`~zc2SMh_~Jj!5fQ#Sf718-JX^Hd&SVsNSADp zq$3Q??J$TW)nc^Dh_>d#I=X!zT`sn)fEgB?cEJTrVxC5Xi0NTV4KXFen&gT`m}Q=H zgopIt&`YkCrEHMv)<L#2B+}N;pYr23%|!B-;-Nq>_n<B4Z5NsgWqx=5JP#GO?+RfL z+i)Ui3bs0J1|i<Uq-;xvNgGbBBn+Mdka&>+ke$Q%imu5sKWBgf(n5BgbC+2j$Tp;Z zYD3_@QkZot7NsuV22tm~!Be3`okS7@gp<8-=G2*I&)v9q>MP>a6LWKOOFS~{)pO5? z$FqNZyS^GXUpUqLtGVVEZmb=j`Rf<!tN(R>?Xj7^-k;sn_*CN5r!QZ+_St!`nL&u4 zsEGSTQz(Jy206Rvu(fefq?>&Rg#v+m_GB_2fhmQv<_3k2QEnoj?g+@v^Mwp-ORG<T z72S4%t9};T%5$MLes&(c!UZmq8B`b4-q;&D2d>#%?6iBSt>!7e6d-Ky1%wo`l_U2E z#?pMWa}^+LU}@b(jAyh=e}oEPQ+CNr?3)QD$AHVN$D`U}4O5e9M^V&jEH+pb1=fNG zf<BmgXs?(GTzSFewyvtJ&0tXUVZHX({aVPKY_3E;P#k%Ub5b~Y$Pf(|6)DnLYdeV{ z&JNz}l>va%uy?_eWaZ1xfN=ft#R~?7Y7qq;jrZydNnJMtlUd<-w4?<C|9D<kK7G|R zJLya=`;fF^W`<!Di!;#^L~{K2Qs^xXr#dgvV<AUgJ$#t_@4Q&mQD>GZfY4ewm=#-Q z8N&<N(9UgeDkU#~+mL!o1DCf;Ua2eAYxcxd?{8B$Rm+amz#D7i@wgwD)TdblAvU)g zCiYIlkD}Ay_t+awXr=~UKX2&!;O?RUZw>1_1&z2W9$k|`h(4x9m>L(M#w$HBDIa!~ zx-xGm{n%@?^d~sa?+;J+GBm3~{JBKe?$tb1Zxm`1H~@A<fhEC_<r29iP#BKm%e)Dz z^K0PJz#g1fv&D#R!n5kYk@vO5UM=x;08TdJE?8`U0$%|OH;;gz>CbyQvdUc7Wi)l% zk+QvsF(wG@fUFrtGqfcBf~p_GjhSidcGJ9Zr6Uv!@!lseza}xDYJ#kuujv@90@1M* zMF*2m-`GnE{E0oV*B}|<%k;{AtpjiHj@ckQ!kqnEGGgg$@HvrRC;s!`T(f;UAlFV@ zj+Jh?7ksKU&Z!q;S7$wi9Tuw8UdvtNIQ{nY>S##E?$b~dg{wG1LVZVH`(E7N+|ij< zlwiZ6@=h5B7j$B+H;w+n<d?DD*(fNUP#KPBHd?Up$lmWAgz}@{Q++ghU<Idi1V$Aj zaJ@b<iiyvN)|rF*mPoaocDudACe;W?rCkpeDrg=+C}TmEAq*0J<P!ertk=tSDcmsn z;BCZ|nQ%RvW?D1>0(cEkmAB$Og$MC+J9!>8Kr!#uqAPg<9u8gfB%&L#Efpj(?GJVY zZQ~u-exkLyt0IcgR<hE7@7x!a7jn@OYlo)7fMDx9Dk6n<#{kiZQ;E_~aZwB@sr!)% zU<vJ%v9l;nS={MhSTNjA?`~lrBo(ww-J8+%xe74^AVY#g>w;A4t!M>q|H>9rt$m)R z92~HE2Ba?NF1(j)g40Xg`)l!sOlEf_5iK}HS#L$KQa31(Z)Iy>`1Dt?GoQ0iO-OiA zoh&2nZRn-UntC<?<V2I9SeNXV&TORctQtkJo5JPrIwzSmOEE$oAzn03we0K~^6_^g z&^k@?owrt-NwNnZyoO7u<bbxt{NtVs)L_EuwfHz2s^iw;F@4LA)Gi=?s8#M7WSUb> zWQ&Y)b(^NWQW#IwUI}}!CLd=GvWIVbxIr?6@4*{16*C@yiU%2WYV?KRBszYCN%RmJ zb+vS&5ILKl+9^)$zjojWRZj3DGUe%`&6QG0e-iUt6Z23^TJml}RYzmJX^0hMyvaj& zxQRN&C_aQCq81);)*gi6rk}i%JEP6ypi?j7zd~Wsc5Lyxg&DNiZ%mlQe)G2J@6LO- zW!^J)FD?qYn~iH3rFelHN!?a-dIw^Kq8|~(=$F}((mu<RL|4=M354#7(rtNql@6ui zY~?>GGpMJzGN>zc?0{N+a0jXeI;_DTKv(A^TG<l2F%`@dc7JfzvLUu;*x#}~mY*7! zH!^dLG=zj<+@l3{#ap#$xz=j6?6U%9=g%-gPq9{lM81Y2q4PuO^j|EWm<XQs&uqo{ z^4uD17L{EM++pBeBkbOKNl_pY?=u9K9Ere$i8Zl53PJFO+@GZ-INAw#BIBtVCYabt z`$Lu53t6>jP3Qp&O1oUEtwEghKHjTQonfgF7_759G#6v7EuO-`td#a(E2F6ik}KO~ zyLH)xLdTc2nk#e;c3LT`jWEO_SOTuereCHP^=aF!g^jX(>0jrEDn?v9fRHzN3ao;n zBoR|-gr`a<OpuYK%a&rPZNX6Kphp?p3h7(rou(#A00z!elfHkV%1=C_WnJ5f7|b4q zY;w*-SuZVOvqNO6`S~pEYXaNG^CJ>iHQ$U_f?+1=xAz01pxB(9r?VZio^%*gAWy|O zEYwYvI1Ubn*b>4j7=&m<3*3?{=Q}Kiwg?GhQqo~g;qbWXGR8(x7S-Y@yRjWC@0~8y z#y|Y9W*;wkh_%}-@Tyqb1U|@pB#Jy`x)T>2UV*CCd2H&9omvA@C`btQz5z3s&$4rS z@it71`9($DMTjU4yto+E;-b6p!d~ZAt>KDi`Ld=Rkl_K0c*{e*5JY|d{ojt?`$Z73 zcxB;bXxV8FKj$#4&%G1oT!AUnxcM0j%pg-!k*ME{2SJfQQ(v(dxIpomPcB;n=!*vn zy6Hd%<gJ=I-oTqV<sNC;MNV2(T5Vx-wZ^qhw@xqCL^ZLSHUdv@uxi!Ur7(o5z{O|H zN53uVX8_XT3HyPk(^4@{*j7k%U!*G5wCW8vh>0{QTDhcMDrNN$A#`7kwAS;8!qzc5 z^;PkmsmT<0(xq;hgA|~0>(&$Qd75xb(eutM3Rb3RBDO@&!%PgW9)5rT*$UX$xePdA zUupTWV%)%;_66mtNC7$LYz$9DiP)lSL-a<3W8~yrXawbcBeV!P@lqGrsGam{M=>b2 z7+J1sR?d}=*+iT*h`cI{h2A2d(UV$&5JG_kVXH~Pok6GMEQ`kKbd0KkL$O*=@nTbg zt#K37D{qzuhRGcwduaHOgtB?hu^F7Q-<sKtrs*n~bR`Wr-Ii}bS6|3ydIn-xNo4Bx zsbM|kW3a27Tk3N-gH$nC($shHe173`5CYuNKNfj!4P_Z!Q`a0&<^5Lf<VpIwKK*sP z!i?KP-SYI*XXK}%JlomaqTNB<*=kvBEYsaS4zE?UX%2d=x(<_$<Eq4=eQaAL@JlAl z=TV)3gLOqb$XlSDCO`uV*P=KMFYiv_fy?4ty4?(fv3w{lmD4&dw!sNzF}iI$o5*dk zFwO#&3ri>NqpRrY5p-e2?B^p0%$7omJ{W;Pl4H!ien$+6G>++mT<*QIGfaC?KS{dk zBsp3SRk1F!s!-A=w3@7AGckEN#%=E=0_ir~<fY6h`urJ)CJ*?dYg;=-v`bmBigX*Z zuw>hUX2>a!1_}$U(<wJ3-;`Kkn=KQ|>J|uz(%($F4MGh8M!<F{nD1873$)a8&Zuw? z98B$Yhz(^S{H6V3+1man#phXb?yR;>ZUq4m#5blrX(qv1o+4q)%p^jJuC6g~ebC)w zh6X_`%aKk2+g@7Usd$Jl$QWVH>0vz`lzBc%)W??<c_9q(as6Y#^JG`609-($zcXe* za0!nBAlxJ#NKLgK%U`y7s8BjSXyH;yv#<;C#Y(r>t%-35r^89FO+hu%-AE}_2e@FB zU$1emrrB+w_RFRO!x2oht=UDeeGdAj9(}=^1Mb^8(0;9;>dgdhk~vO|Ruw>Ry*Kq? zZtGpYhSnw-rmnHqfgE4*<KIfNO)l)|$6zTm#0Xgo`6RBe7OGFL(qgFIde%bq2?U68 z6kS#)Vq26;mw9Z*L1_d9!lJdp*(enRTOO-t|6=f|Y`&|9J&o6C(RQ>5Np`lSch&9q zMg3!+q8h{3+imuEZ*T8*=SBEaLtLkNMTP+g(HK{2i;qAfUjKFfg%@4`IPonOpIYTX zOu4F|GzOK;rA(Q?SGOP0=MXwf&0~B>*aJCqtE`Mo{)YX+MF^i^b~EV%5*G`6OhzpK z9WJ{5V|jXpAGp-q@AP)M3HM(1%P4*6AZ6gC+-r#<155`F9AMmXJ#e*9Tky=8)cQyw zZf%A-sU%*Y679nscxN2}+93O4Hm?d<BmTg@=H)(nK<Xl7{nR<zgc#FtSX8#p0VqZ~ z^`fshId1KR70@r>3x8Yj*;+$IP?~FK0WoK8eoxS-0E8lqf%+b>0P%W*7O1$O$eGcy zxjxzAPx-c*bV&JNcPkYFyIz-yB`=@sB^}yP%mi0iJXL_4FG*KbLs(iVi{hy|XF;Ac zP4?tG+*QAZldsPE-H@6gT1*7z*ENQQ)ry&!umVwvmf9?IHT~7jWjBGiAq~$Id6HWy ztgarSn;eyA_u;Cvcf+Ju@|albjC1}w*P;#j0;EjUvSl%vpPT#7>!zT*fqWq?hDwPu zFN<MMiYwc7mY!LlT9?p(^1+f1<La+mrw(7*P;Whb*)MA8&M>B?6Sg_cYK%3<tvAhO zMg~$K+3yn;$2rXmA(gLetaA?5g*A^$Fn*odH22~WY%kOPMuzU%$c+`JIgYXxFsk*) zu&5NVCI79*cKjvhJM-4Hb3ZSe(vl5U4RWpDdei|-wn~;+eLj@&6W^)uYxtvO78J~7 z2OJ5Pn^v8-YnFeJ`p8?uiuYNtmb5Q4FtI3MN;^UuXbvl@+7o!TQaYiWc03IRCD7g5 z33yt7(JSeKMFK5NkSeAR29{p>0_C<`541_zrl>LehfLNfZB~W;Rn{2E2Qmdb#H^N^ z3D2)eEZioUH>}@Scgg6PnXSncuu%9L3%gZzSRFZ7phswJIMq$pZ3`j=YG-@OXd&-J zpi&!1&gu+jh5f|e;8}PWlc3OGlyPop&bhZV7rZ>KM9D3Suwh?Z1PmI5PK0*ac5k`G zs0cn~HBZ{$n$g$3d1h{dthehXAkDX~zn_oQ^ZcvAZUaNYA>)2*SD@`1LE&#N+{b-- zpLAA^;1*i#uqNxSyp$ijR@1^ipwh4GBP#XLlT|O^HoaU6lwamjHeAsnVc9|^$pwLk z`BO;7vop+ml0zWTBnNM2wL_EhdL1>Kh9#kd^nrkrD9pj0Bq;4+!zB<!8BBaSJp!8w z8l*wm3S`ifR$%BU44B(+83Rmn7QVvss#C){dm>Gt;HN<a|Fa)7*9_f&dTotCi1P4Z z+8u?GLTGnfHEvgZbSQL|jQ3ji(izU;B5CLpGdyG%dKkj%u#D|@<pyt<WoUfwrRYD{ z@_3m43<bN=BN1e^@&ub6H(QJxE~Jcbi?DH};5*XxCgZq4zsv-a@APuhP*L$pv9G%k zA!gQ%kEeJiNR^|1uiUkT5h+asf~#)kfKJ8zcBZWZ5Ey%fuDyAFdsABcU9`aD{$m{h zt+caWDx}-#Vkt;n%!@Fo8#<AZiJ}(+UHa&TwgsNoFd9Jda&pT!P3$^luovsPraVET zZI>d+4|ROX(kf&)LSzgXRxip)V-ZT5#;cFTx)Dwy(G`H~;{j9}R6S$`D~S+%>fY#9 z(i1{r7I&S`7p^wQ3A?}ut0+2&2;j=C!u>@g;!1+Ci5fs4tRwid*r=8r?Q|Wr*t9Rh ze0EKA%aWn0LWNX}d%N+dFj)|ZknKe5MM|;+yJ!Q^CD%dJm5}!{#B9n0wO(2Yxn+wz z%#N70WSS>B^3;ul49+dF#d@YAn;?6{Jzqeo4@f|qqoo25K57;_foG{;$?R9yt!QLa zqOvadSVC6&DVnDZYLN_F_L#LFn3eCs;h_0vKm022*i(t%gSHS-rJsu|Ra+iyR)j~9 z9zzxVHs_4b1!W;w_qNlxj|W|@fyA~^JmoxZvMu<c9b1ooVEDW`n3n3jdy@5_bdbsX z8CLLwE*C(m2vNy5W-Sh#Iffy9ji^MGflmp$l<5}1c3F27rkDd#wf1Pz)HKl%VWv1~ z<e&0W$C7@+r)oAGVVB^cDbcUYN$hjekCC6@+<stY0yutAjk_LS1b51M7g)D;!dhVU z+#N03!MV0;W`qXYm1==*o#QHK@4Y>T2>3XA$Z9B0*xHVL#_`d$GM0qTaV^uTKgKXq z*qcDxl~DUgl?2;y(Qo;NpTaY)SOl@%bW1G0Wt}@d!NTj)*YCmc3HDr<&><@=->g;H zYL!`(!TToL&@R#1Ix={Z!y2^^P@1#86=+EB?XYUtSxbFI>cJQSWkbzprk&U`5D)9l z_lI@fkVZ5=HzO8$v2AyJbW40%okKO}s*pLIY16TcP*EG^e9<}<Op9~dgW+gpnB%cU zF5Pa~!Gm~M(7Td&mFncCZ5`Q^4>BOE9$i{n0L~v(HT2F3rF*w%A8>OKhZj)P5jXp! z1)A!lSIRNv<8;i>hDkO{u@Gznp<Oo8yco)sN&72UN(R85WH`YMQY?T_))@q-JEY~N zf4qZ7M-rCLSTmYOyS;oj2VNz&^|Alj{>+R#EeIKn^d+Uxrc-V<Yc<P}ZTV!hTB|i8 z_yu=~)*9HQT1{=xhy>{{+4j+?+n7<z{eNZ{sq}8LJ;I=0Av?L;teuiu21q@KJ4%@i z9*N&eZwgkt9ly9AD0~76i;acY&Bgs8w4Js~8ym2_Ft*gR?~anf@c4`x6}{`lNK3;( z56@X_+xvEsgFmE_9R5+lRx)BV=u&_tX{u8}<IxgV$Lc6SEN)JfQ(xIpLUAF{rd>VP zY}h&jgamflL<ii|;#0!0*u&>3Rh&4FOlm;`m*zE`Re}eAi_k+g?H5TtBy*!QfeCs( z+SBzu8fAG9=c9?7mXEgRCf{xfMvlXeEqoTEI=MDL?f^p@@ym#T8ABv)ODs9H*Y^x1 z&&Yi_C1j3~&z-h}RB!NX+b|YIAfKe1AX@CqDW_i~GO(paIN+DH{XnC=$mrYg%>+)~ z>b5(HcD|R1r<*dN7^uR}kus1LYWofkSEQ3vq|IIH_*m@d*Xiri@pjr9p)I7dm2_@u z2V*>y19v_+h!ArdZih?P7<M&Kp61of*&X1vPxGh>9Ku!8byfnyXNBH8=eoKD>X{d! ziBerQEDjK3SJ8!r-FxTkZnG~4$PFECXYfFV1qvs~HF=LX(yO#eaSjbZZ-I1MUZ=3m zjh+sjfI3$tZ11<ua4-10*op^sdU|!;1K+g5(HH4mI6fe~Gmc>4_kk#aHUK8P=gw2@ zowV1zA~p$>r4FGi1qobdWTTnRGty5-?5R@nSYz2+t<&?ia^EKXD(6pm;PQNd|2V;a zJif&JB(nZ36t37HSTSVfVt!Gux?7ZTM2143rR8eeO?L{jiBK1k=A3wt%MeDziwveZ z-`uqoSA|Wsz(t3_O9Z+tcO$3C)pP?O8l3XfNouV$)b_zq4uS6T1^DL#{PQ^cV>eE_ zXXhYUF$Rdhi2<{{8;QBjQxHu08mvvMk>LfxRLrq1lwdvK<HDB<rlR=+Rdk}Hq65@{ zFR5V*RdpYzp6RN<=hUPVfvU?=aRmm*yZcOP)G$^_T#<6~f=5NclQO8i<wpgufQ?|5 ztm+;!k*zpR+1--2Y}$x+9*1^TR!lG)^@+-gAD^b;1weX2RJ`S%$m^O3$m^7j#OPIh zqV)B?OYO%e)IIO@dcL~u$0yZ|cha(3$+fHwA}*hAL?eE^4HPQ*ZdckZ^92SYIuB_S z=g{Rrz{h#V2Sz4o;D1dtPhB|=Vb55+icnPzCM5yW30>!j33Z<EiZ<}SxkkjQFR43! zoR`*hJ|3>qhK9Pt1NhJCr~DDBQIcm~S2SNz(Y%6e;C~XOspzq2zV(E@FX`r45!~Zl zO~hN3*DjyEycj(ji{%kdtak=*r4sxYgwhT>U$LXFMWbXW)2V#<)P?hBZ(!C6@$H#Q z=vkd?3>PD47&)YfZxoSEd{t>?Sh+8syL#>XnNt@Gf?+nmQWPu<zmnIDL;!vWdVh#+ zc<bccd*<@Ri<d9mxN`OKm6dg{My*?Mbk`eL7PeVJ|BS(B`c~ZANdQtP9^Gu0&as8^ zzHa3&UYEUHdG^%V%g^2LD+9k-8nD$GXp$?44U%F5%?kc&Zmr=%Uf$R!lHpejuz64c zf8B+@UIvf-GSK=b@8sGe|7lH*&W+*r08TVic(JR>!Ca>a0*KSuUKqgFm#2ts{D^gj zq`H6!C#?N1Eu~ibLeiZZ0pjf7Y-dUNQck%f?reQ_zHS^w8xFX$t3W`{2pOz_ho>@6 zLi-y*N@w$F8M54{zU97e42nhV))yJvn2~%}>>!PnBj<#R<JPWQu0$|_<5>>28F`{g zr58<9Vr%aR8P($<rUxW@6M7^RiDNY=i(+ir?|Jc=R|=m)vOs)tNg++TvcRNa61va< zQo419NNNj9KQerP#x~kTC%7{wV6~&ag04y_V)xnL)3f1BQQ*d+?m<=9V_a!MiSMqW zXb>gkJ^Y2U{(ijl`r6b2$=cYz*v_(E5<`wuaLHJ!h_2{;;D%E&q|3k*q+FW>VWkBh zMLSr#%7a?cROx~OUd>|5Hf%k2{=$VDr{RLj%C*Z^&k16z4)Wxd&iy*vLmLcv*?1C@ zkmD(~PK-H}*H)~^>Wzr7n&iw{PMYz1DzZIdrp0M6EJ_UKKKzs@mfQ#tg6KmP_(9OX z3S&=YfTIUlq4SVpVHUCgLnDgWALnUSfUeMs^UWlRw=wCkyprZn^1$s@<gCFN8?bGM z$wQc+;DmZMk9SYwd{RHOf0nyS-~pgOy&_j9^$e|xDmxgznQjt3o*rGqET%%;ym2EQ z4)ZheG0H(X+tjnxl?$iNUqa!;4KSEKb|_Xb^1VZoCZ%x(tIImNNr}y=N#hJoj@RVv zJ}G46=z#v~GPh@!aHyz0ngY?XBy_7_1W_N0D90G3bxu~KiTLG`REjBU*}5WNv}U?f z0IyNk!GjD7FD^oU^>uToDlZqndDNnXOHErL!$%-0gfJk-n)WE7tH8{#iYcwF8Z2BO zV#v3Tfkf!Mi*9m#Lw+PmVEj23KIO5(rOTJjiJ4o;Axyl(nR5#B3q6R9V`VV}voFmI zhmNAhf?Ck!!kEf6ah0Q_Mvp5tltcRJ+`|;ASH>ZM9w9Cv*mFr!nQ%N6>}!#;<rvHi zZ`fW2A<Bh_6*9aae?CnrAkzn(U@}X(k5|rJI(z=oGxp5W6AafJw6T+G>&+w?z`;p< zT}?RYG%3kG(~4H`v{rmat)Vy4m}Q_8&`ZgbPGrjZB-`ke6Clh)#}y94pTrc{wwo4k zcAxflvK@6@O`wMHfU`f*eX;eNm5s3-@74VseIfu(wMvmjf`OzU+C%e@()`Ru4b284 z?$ajjoP?NG@<?zq-n!0m+=lU)t>+k%K1wy!-KlGV6J&!KkMBx4$1f3nNe1TKNL6f3 znqnQ|(B?wd^5iEtT^ixH(YztgY48}>G?5)>3dTw{<DNav;Jv=Cnz}Ag0h)B126<=O zIFQiA)@`xXOS*DW$jrJoZt&;KQmNnDl+?d0k_CW$;vsFPXoOreHv(jzUd84Sh_!E2 z=u(xn)H}D^b@Nc<WbOrY9WqfD5qzb~pfCJ2_d5HOHAG&XLw}>wu&6m9y`6oPm~C%D zBtArDH&x%nyVnQG1Bh28&^|(blgw69586%|Iw3tzW5lWlpDf6>Ojuj^(|LW}etq5H zD7E#JAlBu%snP<mg5gm4u>9odrdWz$!4+Uvvmu70WNsdKi4y$F1{s7Mmw@VSmYnwS zV9-MrR0!nhBv+h~0`%wlj7xYR1<!VdP}Ah!<NZf+J9LKUlFrwc_)#3wjNk%W$p+a9 zksOYS)&qXIrd?!ZNVHB;6mdQZm_33)ChaN4_tsv%#L%@p_ZI4bQiZy;2~vF}vePHZ zu`5lXXb{-Sc-P+ZSbHnU98eqnd%ZGI;`H_m#R+m$z9M$h5>$0VH{B9tp_{$e{X4ut zvxlBer=-tJ!ku{3<ub5cp;$8OHLbhgJ(YHLhM4(I-zd^Z4JKUoN!y{Ffza`~u#&+I z6%*u6!!spav?$%MwE}pyg<gMJM}zf{%O2N6Nzo}fBogrYB`JH4k}-`tI5l9TXdCe8 zFdICI9(Y!tsCgNh!bLZB`@3=8jbcHf;ZSS=+bL!_z@|QWG4148k!=j4>#^X9N1gU6 zvtZ;U9KxGc(sq*#;!ZjonWCG$tPR3suodf7TLPhz%u;x0%>e-p0ypVnXiGF?i2cb; z8n7-)Wk~2kZl<M&6_viHu(vutOtuA{QUtzJD)#gWn3t9{2Aj8K<V+W4-i5h^N1KRm zRb%Bq0hJlbfQqMeWmeLg&Y}^kvLWEQw~_AA5ecm@@=XJA3e>ux_TBoxOkL+}gIDBW z^dQ=r<R3XyS`09KB9yRiWte9-byN+b0J7&xr6s@w0a}V7(^G~zE*f)V$PA}`d4{4| zf?dQWx{Cw`AkYdAYbnA&rIw|h5ZeRIUId9jVw=9#T~rOXz3%$|G>8+#0N+@7_-Wra zFqk~kl;5uqPqugrmGC$-#<Ou(p(O|%7{w)*&CI+v*hjZ7p}nf6NML9@B&t~(;stLM z09Ma02>ZAV&W04I6}&*BZAegxPTb^D3wf1LYA0}~v1r*n;@M3C{!6eFk<To6gc04M zf?U^y@<6cx)IH4(<T0XTt`L_@Y<alpj4_}*w<mPCAJR<`0+Luzsu0($I6{i35&;lw z%Hk!iKcO|kB}+JZ<3ps(0jQH!P-jXCO@rck=}oFKui(R-Z7{o$45HA@n58Vogu0+q zxHe&ki6HoSII_(s%z#&7KRvvG^B~suBN;oVH(PvShKpbzLAUL&4+%4X9S_dg%iDZ( z7$kv+hUM4u*m72WkuMl*ZE8D;Dlpl$_2*|7`9H@Ra?tHJD|%!@${u9OBy#a_A&0kt zv81T!UA7e`JkIvn@_bb&%eL72tWj4bYfG%|okM~wJ!K7;B3zU5fsGP`zzkhzx%Tao z(?s<1OO@OzggMCq?S(+j;F)*%%_`4O>fCL1K;cMINhNbl+H+0;<D?r6mDlza$zwR0 zq4#>YuLc@{Zi)&$>t!>V50SuXoeaThs}-iXPcnQqjE?Q5iMgta39aRr8wE`T#p2eK zWtGmIIZpUT;BY$%ufsZ;3Ov#b>Lz@+276#l+J>sf(VxulHP!uM_f`bbLl4pMN-)*` z<HyTR*MsC{<9E`_Miij4fnRHciAwowKTfJ%rH87fBhdA#3MN_(XFA+$R`@M|S7l~h z*vs|p?zmt}rFMl&*~Pr1k9lZ{fq-WUzcuz&5CRC#Z{9X(^3K3hDV>|FgYaBaWyZ;p zExCeCyJgmA-=JGI?HsakTRl0Gn^>0Z5e7D+ET@_tc`TEH#C<tqS`U9R>ga*${=z~d z=)UPu|41{BEv}w4_jX9rC@kPP$2Ucmf`q$y!~)y=fNv1`&*`wPysWnhW~760t(4R< z$Ec{_n6IbDIus|nGc>dZAAESQ=a6)yAoX<%4LY)EtiEcm5^6}WZhA$*B}%K6K=W%Q z7210@bODFs0la*`83>$9<XySNDDDgO4m1cb#oHKipnOtq2F7Bktgfc0tT~z~W1u6< zlRd7>^6iopA=r2=R)+(2;^aj<gbi2pYwp284e$FK6F)NqQ&y@1+-Nd$f`U7NG)4!* zveSvOtw>35Q>Jc@q=JQ0*f%|^IS@b&(*Zn&y-SM*ofT!)Aat)6mLQiY;Vba7<~A!T zqc2i(;urWqPy}>&GqzJH^cE|#Uj_|-7*Z(gsrr<d(M>Lt?bnMg4k?o+AOVuNh<Iy9 z0TQ~MAxz%Wt>8XoifGr<?C!z`tb5ZOu8xfM!-Q?q35~4#4UawM*X$$a>Y3pltylKN z9TN}P5enwA508G#pU&wJ2?6x#GENXHT}ZOeLF^aS9iED6%#ET69g%>3U-XO0?wFfM zQ`K|5te)$to<L2SCu#x1#i#8^*=xC8A{y46kpMK{Rh$v_|0_8*w9dU;>JvoZZ0(>2 zVh&<_>zxX@IP7yxmU%AJN@*quo_<>CydTE-M8!FtH7fd~u2xDge{Ps#IIEPaA@P$C zEv=-oY%i;&=X5p)sdF}2D<2V6r#Uz&uhF^OR%#fN?T}aFjvZsmD8F-MMU^!m5YUN% zVNeF*6id~o7g%=2EJy0v2f{5zw*xg14H#bjz=NQ05cFi9P(yTz#j;yJmSKN_mr?+O zt*rWuw{|e$fX%ujqujDbxRCDvBSD~qTSr=;dnaB+NefD9Ru43*M66lMDz9tLESkUz zb5pFEf#OG48Uxtm%Sr7q%U&st>l%ld!G@B+`vI<Z`Q1BvYbRo}hXGt%zdM(c0|QGh zoQ?1?_kHjLA9j}Y5F1zQZXGmE>>1}ReD)S`rY!1O7VXM(D!(XS^!jU~3k7^qeGwbm zBC%+=9~`9V<!tFhP&lQ-yeM{A`iE5}!^ACp+<V6IFLON3mi$2q`J?eVAG3Yb0vG_g z?wXZ+)PV+0&j-PJ0&z@eEa)ocXXk?g$Fu%Wu)JWr`aocM1sw<g?iihAICmH^b3~a5 z9zoy>GLU-ZYB%G|JE~mFc`L#;KJ_DP1I=Kap|;7^OgA09gtikz&eVz!!*c*dh*woI zO{mZ(JyGmhCr=fE|0E_$pr$fs6^A7ir-bGXGsa$}c$A9EQJ0S4+K$+R2uY{|-&-CN z>$*59+kTb`jF*?#i402TC8!p1e%d{9yat4VANCR8>XdR4pdsADg?Pz+eO<pExv#mC zKseIE0Yeb*oeNM-#w0`Eq5=3;_lI=G(8<K)Fq75l7q3@Q!}h(94;?XM!B+<Sk}Z@u zcAT|aQk1OF7|0uL{Sc^C2l}ZQqq#dh6z$1UU5RL3Av&<8LBaJgm@Kgh$T{f=r2XTs zvATjWKdTDV!!c8Amh_P>Mu*21AS8`Q;T;PYA{Q-Jbb&*fjD{2I>jm0&g17m$<nf<s zjg5d$59`1hYIA1_FXWlAy=M$b$g(X*-u#eO8|gZqO1DrI#fTjxHRz<%KFpvyu3HfN z2)ypm>m_QlEIj&7D^(D<j3aBd4x8{%U=q>`Z^qF$N+%YA@gv32fZ9Nz=y#er5T1J( z|K-OU;&)3sP%+Ob_58Ygew{rJ4!84u;>c<VC@lq|pc3><)N{>Ypxp9zjF|y^`3GS~ zeh$85Riz6pm4p{jK|@w)nO(!s9VT^?x~YYC&t1z@D~w5MO8*}K00960?7eGqB*%3i z`gt8O|DlFbxVt%x#*36?fFO;*U`Slf>kdFtBNzq>-36dH-CgafZajv<4X$iilq{LL z5t6KzWog6f)p~78t93}EL`gs94ra(t{tGwHBi~uojhUfEJJ`V?x~nozW}eJEdCtl6 z7#=(ZFcrvC1}B-e4@G6=PO3Z|;onipa0xTOjJ7?04q}ZdV|>2-F{EhsT|et5rlG_O z2-Dh95K>GKb%?V`v}HdR;+X*Uf-bfod&OtJYNgc%j$nP54GOE3+NWk_W@;vYLYlRE z+yd$<_e?dJ#Ka!zGc_Lg_Y)l#z|^C{UlWgMkF4UIyHWFe#03sXl#gGvc7_ogltP@D z+2}m8ek`KXBH!GrWBxg?rtxEB67H}USG!rvebC|iWnaRlb6Is;xyhvt8ICulliw3L zQ$FTi?uDp+?A#p9ZdqYz#$C!G1TQ;~e<3xC4=vDvij8qJ3W{H-2yEmBG?Ee4WGEw- zl1SYyX@nC$nYw{VDT?V)$lzG^nsFj-mB_<f^3gQF!KN6Iw#aSa^Ll!W_cysp4vFIu z&tkzyVgM$^0U*~!NF*#r63ab+GA(%v#wZ2K6MWolR98ex81dXr>bEG1$+~cm$(g7( zWH>!#03*(=VFvp;TGEl@Fa%USY7et~K{Gf?#K4il=jjOM)fga(W*Avb8#YNvq@^nH z${HVb%PB5nVPh*-w9BGIBNm^gFK7zSJzp>|fJb>y>M5l2&_P<Lm_8<%Y1>>FvTw*v zw(Y#ehHFuiJuOwVvM4!B@)L|qVS4vKlk9DPrUfO@1~G?IN#H)KZ*G5N^lHI`(mprZ zHg=ubY`_QBYQFO1V(DHkq3RWEH3q!5p0F2-SdxcnCy}wVM4!<Uk~8V)xMDlF%j_^@ zcBps=1%=51QN<y+aFvgd^v&qkBA&IP%SLn(!-Yn#t4LqqLTh5#okY7iIMCWv(qVy8 zo!sk%6aR>M;m*$nFht<G&tcrM=r9#q8%-absa7R<rK$(6&RvGN;MagLZ;~j*Fqa6Y zhn3Vu@EC)WdPq#Pi;<<CglRwz)Z#s*Dmz6^t;Hk1hNN-Yv*Hw2?Fw9%rq6<jG|{5m z-<7qXk&Fma{c=hcN=;(#T_esN^rGBr%kz1%7J;=Oja$-1^K3!){I=k#(54FlI~b1J z5y$b1G9;5l@LZ9x{tW7=4i|$b5X2|O&|slBfll4%PsCV739med8>}oNVFu4TMYW9< z>)Bvf7(q_qCy1f&N6Mk}hMNcFN|&bw>zb;vRqT}!@r>RmPl-NEIJn!$M$L_5FF`2M zI*5y#NBvtIUabz*11cJz@BsKCanPW+HW-S~+TQG=9U+aAepE>2<^laDc(zwZr*w`^ zeE>3tK_?InP#!~~h(;tGU%Z(O$8Ssl77vnw2z6Yv;|{FfbTH7aNN_sEY@Z8>wxRyc z^p4<ZD51wrltBywu^szUn58C)L9N(~uD`Or5#6}Cfr+8|k?~GLwu4wCad85)726^s z9X$2C6e-X}xV$UGb997mnFHO{ub1r1m@19SI@qu7jk-|}6+TIX>l+9O8<YTY$)SHJ zGyDR!xLRn<;_Mn5VFJ$0D6N<`sIU;#47yPZGj_;}>~-V)R&*6YL{KocL7akkLi+el zBEI%Or<ZaxJd+#|8A9&A5J%;xI50RsT>s<g%cqzra+k9qYM3jROJl;yBsi?sWB`(b zB-B|kgu@iWcDAfEN&v2;jsneLSAp^DQxr-x3CdvoQP@E^Rf-{CPViv+``irzQ-Yi6 zD&(Yn?(8`9M1LZqlVf~N2Wtw<0>_dK(vgXKb<gb$tC{@H%s3snw-C+z{u-YvYaP5; z?nPZOhsKgG%9apJGUOhEWdKJ9?HdY3NLig8WvcIC^N&6_tX?)ixhYb!_xqacovWu< zHGb?xslZe%O4acK`i@nMOp(8@UW+bF(rIz(2!?Nv_0gB!pc9X@K#H`hZ`2;K_XMM= z?`wNe(#O@_7U!OpBOkFr22$1Hb;L`1lox^xpsOP=x0HwGxo|{K+%B`?%j+7H6(=RS zQ73u$loa^D?to)kj49e2&CYqd2-RK!h?DhF6|X`e=~s|mNd=VP9rnfID^fufP6(~= zniRXBq;0C&0{wfyBmoOYVX8avFm8>Tc<!LlQ03`T2_)^ebL@BA)j(Vj>w;%Wz+W!S zu&-nw5f$)w(%w=XgmfDKL81`ZT}r-p8Tc4Sp9D%ZRSX%tv8Bw8zsO>nR2OnYVcq2k zW7L}ifi!urYWI}>U^mbuBa%o-*Aix8(tg?-_4MlA&5n%@mZaEd3w%wR+DRAPelO7y zLswXERF`FY&^(6a5@*8_GI&5cCKsdn0xY0wl&?<wVU{c&ZqV$r<&jAHl{DQTN5I}) zp*ohMTzHhKV^<lvnD$-&?J0LeE6kNtm&bxm6p6J4cAK)aZ3VhZ5(vN3@?LrW#4K2} z4-^!?x2Z5tTR9yR^8`BOD$`cr=|*-f6L|347|Hg7R98^TS+$lG755a4Oc)P!Y!Z!F z=-cd8HEyrR`Q6|eCxtO+ukvJ1kl6z=hT0Q&^EeX{jI8~ndub`4m#m+BhX5jwrLYPK z{T{%3=qaauw78q~v1=*7SVK4=kf8e~#3oI=8u(_l$=nsKQ?P{sh^k0htV{OC#}P-{ zmQDzD>!}(qvI5v;h^+um9JbP^AWLL@<oqB-n)?hH3-{S*I1Uu}6uHkv1LH}%o3`&( zK!Hx<C8YDr=?^>-=sv6(TPM#3)@#G1>NIeOFQeXZ4>dk%58QfLKh|)`f|~58o5?oR zjB{m*LuJe-6fLF4PK!+h>@QS^OZD=rCe*vhiEBmzA5amBSEiQF)9pzerPAfJ2)wBP zGQdx6o~Ctppcb!}Rj)j2WQrPjGb}pd6?yuUy5>p}$<RZ737VXqhDS@A3CK|$`M&^# zg(8(y1%lM?3qF+dYq{zh@0L-#cSa>Pk~b<*s$@8}@^trg`xJdrquMw~_)1S_p4%}k z>vmWJ$0-k_Z6FGraA7NW<Uv;07qpPxCpnHGOk6YCsihdQqPD}CJq!Rlo7xO>dd{+? zoSxg-Q2+(8WjU}FttW|Z_8YWWqzk@YhS{A(@vb2mL%_Ns!}$KAVb&u@s)7R0JvB)C zPc@^bKzsQVS>KUW1=$?4qD~C3J?!cwy|bx?p+|tpD~=L%Cy83lgDks?O5;%>nuc3l zx&x#kXlur&$-3_#g9JR-zgDz1RB`n<NeYELiFSN5TSR!w)9VWaH+PfQVWDmc3~WJ; z7_g}7phX_*6yYQ_Zd5x>GiM3M3{znFLjnH=5T=B)7m@D*COzpVem5pdd=;WBe3M_6 z@`|g-(_Ic1#<w9a<6_doQR+Uh6?BKA_NIOUpJRwn3v6X8>IwcS9%eccx?S>Q;*!dc zXF*OGDDTBQ#@$v#Nti54Z*y0a0<>%##{;&_B&!<m#Fx+xTwr4yx>Za22SW557IpJt zL7riB9fbEB_Ir7$*RZZK`HIn9HRxDyVUim|?-tUy`nV9vpbi%)7o3a5&y%x@b-ica zv6S7zP`wGRV5p#b`+~-MTB9^ZHn#<jpqYwQAa5&M5Z@n3FkaX3P6{$LL1Ie0&(`m$ z9=xSq(BO3{G`lV33xyp1<i1X#k4^iD0U_`vR5<H*v$&&Q1guq*k>tfm9H7Qptek!S zw-3Ma<+ES@{Nb<PKKtsAKKRXV&ffXjgSWn66;ggzt$x<Q_%E$t_R3(8<STKJ2$2HV z!L{oz+RUJlDu{aBqA0DyBvdTeTRS5VKe2zUqR(5w_Ym%69*5mo<m;2LJBxgMWcdcW zF`=M@N<U9Uy!7-?CI-8)q4d;I0Z8bVR|C++?P>!iZl=tqOQ7LpOj!jWr@1hy4efGr zJY3>w?pg&4j;VrpjJi8V9Lqf18-)`&(urCr8}m|0D{?FbQWs15QG#iAD}3<IZy)~V z9i=XD`QLygjy-G4@H>W>#Y=u}a=9`Jy*vwVeX2rwkrv)6g^%szXure01qcv)3UIVP zk*?T<6-9!39jFyxZ0tf+5}kpjV*^FLLzRb}afk`Mm#+OX$9Wiv3c{(<k$cb{8^aQ} z9}QODsOs&+POsKpKpR+<el@htO&iM)LBb4sGDV%X@WXo<lHNMf&en~5PMTz_S!(>I z3Ytc-`gr&CmzA1=Y90MKm=%f)pwx&#!zZm_bS<3&E66yzn$}U(G*2b#39n;#G|?I| z6-L{+rqrh%kIbB($xdqH?3Zr7x_W#2+VUOo|Ic4reRcJk7~*X5>>|BhS-$b=^16OA zKl{w=vwy5=uU^|;zPcf*uH9I<y?TB1#>VnB_ztmbTQjo^B=~ZM62{oBA8Y;L><s9L zT#J197Tz_Z*%?3Lcv@To%Z674H{YqFk%Q$ha?yTdfHo7wTJGamqqV&ap&W$JxxF1! z*yW>OWp+?uAYxw#;k-c_IN4>x)DHBNP6bp?hignc3cHV|j@^D}EsN#I|225>!2EZz z(#BL6OKoKYx`jIA!wrXo;ogu{Z_+KN0f1KVK5Qi{p*TjrR(XvcqzXK+){%MOEIKa_ z>~m((LhMD^DqhSN&|6tCv_MzbT90#@)Rpxfi8b<dVJiF0d`PCfU;7a(6^&R>hT$M5 zHt}08UtMZbX)qw#SYXiIMZK4)NiMpr@r%azy#;@Qfx{h8e!OsNz?<oo-wW}u%o)t+ zgwYY(mrJ6@$$^jod~=rR^8Im!g7<`63gla302XH}UsRf+mxtI<)txXl8Z0OgU0Cqe z0%Q3T?8fr0$7-R9W$+?CkUW1o%ZByx_)?v$2PFi0I$3)}me*L1^+WYqCp|P=N??~! z6(PwNNf%7khjBin5N=|Auj2GZD((6JWO*tmt$!o{I?LNd!Izbrdc-y9ezcG$pqxEK zy^DdWwVUg+^HCA+#d!*f(p_}z0RQwDGo@g4FnJj5NA-IVx{2xR+8{@h`@jLEnoa(E zWJNZhVTykVX`SOXMs7`x2RM!fF?p!TW}xI3i?t5o!h*Cjb(HMhO@~u(6}Q*Tj;4xs zp0N$pFdMZG7I#!{L{UVUkoUV;e_uwChKxp{R+6BLsRiB`66?s>?_rn?Fsz;K;=btu z0VdpNKfQA>91e<w>1lziqD6geW%>SeagZH}6Metc-cMghJ4<sj^Pl<bY(T#sy$0%M zxD9oBCOWhuN4;>6b~;IaM`yCbP`9IfyH7QzygBhTiv3iJF0sUONvxD(s|3e`f+Bnz z`DF&=4!AI5J?+K0*xt^=U0gX+)UI^O-D<s_rTuzsvDN_Si?v0EiQpB!s65%LT{Lcy zdG>`$h3HEg*RS!WJdJYLvk!m$owM)%_UxVa&c6GT3zfc~8QcwnRv*7y7^7zey1-qs z+z_GBrUcTA9G4EV#2~F|jn-q$BpYjR#CDP>=^v(f)<;VSw&oX)r=)UA5+-;f7$REN zOVLGWT<PXac!rlXxb>OEJQbek88PX)XCxDt{wPZk>9Ff>oHh*)6@v!)CUA~Puj!J} z8RQJP)kzN1c5*8{PP%BQ27ujXi=}P&Mw){>J=D37tyZ@k{Fy%kO*l)*YS&zNp{8;= zKXO{|dIsTv46~gpdcSBg{;G)0(a5KpHAy*#@nGse?B*`)=qdD@uYqHY6_a4G5(I?l zpQ3{Zss_tWQbu(N2=^3RHDse9bq0e4SgZw(JMH(A9IyywYWlb?o#~SO8$v$sMpP5w zx2jc&k$qav(PDjm>_UP`Bop*&X`#%fwyj5-TU8AbacMnA_g_ty?x7GmSwB$HD^wrU zPs4L;g7)e6_!f;uR<izHx?kQBkA+=pGK={@CIPnx-Uu6f2i|V?MnY5`fSVs;R9}-@ zy+BRU^|(J0EA3>(fWD+yX+y{`_j$lpOp0&x3xb6RQM*3lnKx`JQO+(o>t7~m4w8!M z-v$F3QV}szYPP3~HJr`d!gRu&l>@<I*SU!?c@E9%4FpknyMZN?+-l21NW80W1Y|?H zXjl`az>T@@Q2jCrhapaK$jn+p5{D;P)$nqpp5vmVHajV3$drkNoB&Jm7Ykpu>rRxT z1h7k%tw$GQ&LRxV3vjQvzSv1eAXcURBzqx~qu?4>5I53YMh80ps}KSfM{vwUok7ZX zD)w@LP|--WSi^(gnK7a}S2<X70jpZ3f=pXy`o+?II%G?6S4JdH#ftf}N^6j$RjZe( z$BCkDS~^TP)D(nZM~QNYVpcu7$2T%^=NFW*C)h};V&GMuC#}J#IH+5o8i7TlDGc)* zydx$(PqFJ0o&jha+LbeI+RGa6nL8@vc}OP(v(OWm8*oWeSHFQK2PHasmNE=38!=F3 zl`FkJVG?5I1nu21-WnI|Z1u@9u4g_AYV^^YB-u2uNLSLs5?)gv<<&Gff&gwX`;&D+ z3t~2g9AFeUt@2Lf&?s(Lkeg7ft%~84jF)99Gv_IMD9Zq3Yg(|HZRr<u8&XzvBO3&3 zLKYSO&g9fTSHer_?5$ru`1a3?edCz)ZsI~)?sk`uG1-dF-v8CvTfYF=;;mmi_{y6P ze(;r!W&UGMFDvoQIgsE2`3yBX%4{#p1uN2llGMWyS1P&L0Hw(jyk!JOgX^e&6J$2) z)k8!HJBv+2@0yY9eA5Bt1rfo`k4A~39#dicKBV|G=w(&1W*o1mrHk9y9u+c~xS5Rg z<~_Hhz8131us{66$hm_FtY^~0Cb0@DpQ0!}6TQHV?m7YPtvvMR$7-vtVY=6*b_fz@ zs<=8j;5k!SWxgACI?`uPt<VHxK8uIG5X%=Z+jK%)PxBH25N7HCWl2}m$}`DoBoHr7 z9U6597$-Xrg$gGF5x%8-1@IQWIdAXbn|tolIeI+j!eW050A_lxMowsT*?ob6J%~iG zB%WBW4cy>MDsB|fn1yww;7Kd>x`BD0bu`aq;wvP8;FLSf>uD#oW?<$$s$|TUH8@w| za5#oQf1Mg>2h2oLj>u<~Y)0m&WF<6a>l}yU_Su*b6jN}-P`Vk-mRexgQ)O&UjVIiU zlc4E*;|~!)dj#WXATK_vlAwgVECXZ?9tY_dyeP#)0B@(xQg*r==^VD^a?`}Ak{>z$ zGU{To%OLUsps!iSf@Xu={Ym98in9FG!nNR#RysfuKb>I}87u_<;x2gW)HnEvbx{_) zEWHEv6=ZvbC?zDk9y>!@ff*t#SiRtMSPv?^kIJxQaEK%#`lchLwn?b{Zf7JfoCfjT zvSO0S_o2NaFuoYv7AUp3n<Bs-$QbpjT}jsh+eT$LkGlXkEtqU}ykfUjZ(Lcs@nV3v zs2eSb<)nY152A1Pt1#lxGKZ4Mj|H)M6T?Sf&dQQ2Ki~tC_AS93KBvx*FoUW$d{~PC z^Gkk9vOQT}fk2OkwRypzkY$WT#eoT8$h7H*mP{OkBG4JN1lWymMe|L#t{rR1!n=AA zs3|aDDF)<6igL&)uXZ)pg?pjNaei>iOv(s4TgeOIW(tZgsj?HvoD*z$4PJ{4mcyeX zK`7b5WiTBn1xz)9Wpy9Rh>mPb%$NEKVTvscv!09*`Uv1SA^O#*z&`r|*IwFBi-SaY zYRg#+2ul8OQLX<jF+7y|_e554vtu|EUiL6-pPO{|xa*59HvfDi7R@qxpNdJl3eMa0 zlI0;k0Io_>G-X7yUMOXev4*@MS27L|dDA_eY?T{q35O-H?+v5@N2NmDF6wEp$p%L7 z35Ic8sd7>u98FMYku4Vqp(GQ&;4DyZk-1C-XoSY*3ZCZLhVd}Xhojin$jeS^yJIRZ zMkp=&O8aljn>C-tBpMr|oD}$)5a@z=6(?URUf>=x1NVR%a8(|PdA?VwL68bwx9aA- zb7&K=R!F7bX%u{%OBDlBH<!-}7FSRyFoGas3V(L_;2bNIlGI4cWx`{{C3&`fU(Lx~ znv%P~pQ+a5RNO6@35i)%WAkok0%&`8<<S->5#%l5QxfP_)nhv^bw0chNkro-3>hCS zsE4>w;8_sbs7N-_9**duf~eF<M(iMXctspGt$ubC8VxiJkOUwF)2LE17)u}3weB*O z0*ifS%U>wx(q4f*T(_&j3qy!Gw2q=#jq`+hq=UEd$J`_amZ6jKankSL!}8V40vW|% ziY8K<!N>4t!I?MO)Lkjn`0b@RCfHd*b?qkBa2Ko`;YBWN%{y4h%IE!TxB}g_Hgl-u zbB{SNZkLs}FfcV6J{hupk#&<+0=k%b4P5cjI&jp#+lR;=LiQ}Oek)R&0)(Wg*;=SO za-;dkuAVPT{=-Tv13C*OtSgc8mZ&N72t0g3%y^ghaLvdBd2#rla3JH==pQoTy~k~u zb0Bwj>Ab@QFYkJJ#hV-_?GZ#h5c_R*zBx1ZY;%6**(e(n(?ufY%-wD#B05Y}G$jGY zDM@YugX2EDm>wq4?6XlX?FuwvD{#RKGhfmgnEoE11HTSHuHp@-kd$C8sRIR-fC5BO z$DbE{C`A=~Fb;ajFit1gB=!T3?1RqIn@p3Q>5~2$4J+XE(P|pmmv#wvE-b|sc|wQH z_T>kzs?lppxpZNW4w!vlIw30xPlgQSlooPcWOQVVC>WTUkdRs0?Xtq2@RIZoKRw$z zI*vo$X;-^~eL<RVlwfiT@b*aCW~2-L0J98fG!2%m8I5C@t;y5}H?SY(4`i3eQED#c zVTK27{VYX@bT8mFQp#bkcHJAC_9zGYLDnYhLC$zADd|{Pf--9bBqqA>D2?-PkE(6~ zGZ<IhnT9vPYx{jzrAq%Jm!(}$Ovc5;&=Qq_F$^h70m<7Xnfnr1@|iM-7{f7|k<gm3 zSMW?UEr2odi)R3A07Ey;*ae34oq~kN80E6t%`gljdR!g}==&a|WzeV@9v+<WgaISD zC5Vu?4VXH^r*kh(yB==_sIB+#Pfu`&{AGTtbgJ6u-c>$NaAC;7RVn>z*fp%we6C@% zY*=Q?kLeFn$n&>MAqRyWstH_0h?<5JoiddQ*+44t2w9s1t2|)bLt2$sbHzUOzUX#A z?1{0yVJ4w3qx290HZHJunvDwGLnQc3bq3(=Q92CMzDZ6M_*7AU`3YjSTszIfxShaa z<&-Oh;GPp184`!L3;rV9a%X(I<;?~70X!~`9K?n3)Sd`K-hGDo3Hyhg6PH!W=W*<f z*epgYS*C`@laTi|l#uV5Hm1coM!Qd^hBF&U*V|hHrVdui-OvvL2~_Y;(4Oamzxn0a z+uxQdo(Erf^XvydIs4yV{I~bNcJ_~7eDK2`+<*7$XaDjYc=*<L@4x@Ov$wu~_Um8W zfA`J%@BY(+Z~yG!_x@58R>0isgl_0_*I0<vY|*Rym-q;z!koe-?H5T-ygRVZS#Nyo zTHCTu8;*I}d+J>?3Mu=v>e=$400MnZjr3USNXO!1LgiVBi#ATLvWkt-Xe6K{Q^t(S ze6@Bj-v)mviIj3fR#jgce$tm<nV_+&PvbpfdpYDFwGaZJRJ}H%(yBV;mRa}ILl)+4 zyiIXDOa1eE@qXIoaAGAIlCq8niLngqV+^kfYU#**#8+v$EnT>nJ^nThEcTc-F0C!O zrj#1hU$1;#L4YG%x-#7gxKb(_fE0inRDE6p!X7GXXwEo3GdKo%FUr5B45aDzOM%!U zQtKEtyAH`2rZI#jug1_r<VJ7Wz{VFOW#DXnN2VBOzG#6&YY$)5Z<m6TM%lC5N551g zP0czFOB_DD>pL~5(*4}c_E$k)CWO!0=da<))^rFwD>V~C`MkP0L;~}aW>qF6%F9)B z3B7O*A?Srm7%7669~Gmr%Y?XWPj=YpJ3HfIdlPlpL7y1)^FlBY=MDpXKC(2@Uf)pv zXWM=VN-<lm?(+pv`LT>fmCb3mtm+^Y7h|8@QhoKSl=6UO`lxE{ths{S703M#B;B&d zs4PdI?|73(?`(SIt!LSyRZ6V4SJMzjBi(8sfaehiLa1Ismu}kuy1u3oy2R0I$?l8S z<Sl_9VR2EU`+XS)mZH^2`Ke)nfl#aIkTVyJ&_}Ho_eF7(pJCM!)6{ZyLG>-D6nnxp zQSgyu<qDeJsj_fSTB7~u(k)PKuBW@*5Gy4=@;;de?PcV@OV+TOGrXcLFB51!TzCTY z6nsJftK9CJ-fXIm=f9dLYYMf%+!S7+U`yvis&Ts=0QIp#g)ij@0bj5Xzz1WnOoQ6P zZ|{6A%cyYPL-(G0e!926=QraYXTl#$`e4}c3tCn!5IU&BASTkRLZi$yruZl@=h~M3 zp0GK&GlVDNgs%nN*<sH9UWI5fc8W4gL`7W{jBhd+md)BtDEY~p$vN{KSU$o@^As)Q z0d@AvpOYe`M1R6Q`e9G`Xj7f1(jdXra!lx(XUP}FC5dO)=Ncn9(#u5FR)9K>k+!Zn z^kp|P`u_dism+V22B^QcBv!zY;SqQitAue2i5DCpkn&?zQfXLn&p6mwETP(}No=rk zq1g<Zuc`_Q22!4zW#M4lwu8?E_5keX#3Bmdi^_uJu_YgW;R|F^tc*KK5jr;JK?!!@ zbAAEBGu|2}59<+JBdyG+KQ#O}Ofa8!6=Om3{7Jm{DrUek(Noh9DC7$&@R95O^LqAJ z@(In4QL;quhcG9=k%xQuNOhd_7dO>zk?A<@Ra;d&9^EUcE?C!v;PDU17GxHTd6wu} z9z6A_;{hsT<@4=7X{Ss}C13txKvV8VsRzpRcU}$gDes^8sjod!kL+$u(3|l+s@wRY zOOi)KSg@nxO8JX?0_tP25y~>U3^UC!lqw{^_@<4e<g9EH&6Xbw@F|nG!egsa{fH6Q z!<~2@9d}EW<qOeFqeR%Q##lg8G>H;>LZ0rts>a<DFx>;oJ;EIn=K1r@^Gl7KRPfAl z*2Cd@%GN{J3>4XuG1fJY!kC0CPAyhGFawjE?qxP&6LhIcl>pf>AlV+_98EybTUO^# zA~vzaW-ipLDR-xpbj#yELmS`IQC9C_<vNZNSGj7&Os!rvjzM+E1kZOBHt@TSHmQhO zG>HJ^C8~osn;}#5b6|Lac`xJCf=GV?G6^~8p|L1h`=OyF=7a|Yt~A*u^GmiBM8o0= zs)XBGAzYANJ{ysi3i<5$60}b^!NHB{+EN;i%n1lEWDYXGirHK^1O*~5wqwj+1Z#qJ zTANI9P)FU*v=u8@w&NHo?AmFJ=USDa@%Xr?kfN$nSw498OCNssAJEN2$mnPPAZ7Qz z`qJ4u?>T}uu7K(ko!(kxS2d`4O<s)38dWhUj4BJ27Mg?$?T2lHDU#u3+iAgs=J+!N zqR6=(4@;5rq!PLJfA`LVZ@%~7YhONl=X-SiKK$NaQjnOlx4-z{TPnhgryG$e4NGiG zWGK@%h3xzAOJ9HZn|IEB`rZfs`^TlAHmPt+I|wYzY()=#`u2l={;^mE5C7+{9)9!J zXYc-n=;DX}?FS!x<>zA|^)f)?<?bk#VQ?ON=ci}i_{X#N-#q)qPtU&klLv49*9SlO z@x%Y~RS%T3ospn`2%%%)a(8eLFU_?6NNIiC4H1Evj@N?d^@G3t#o3$Rz5kovJ^bcR z{_Xv*-GBFQ?!Wuh`|rMa_U`|9@XKGGed8@rCI;|lZ@v$fKX1PO!7sjg_7DH#>}S9I z;CJtd?+?E7HxIt^TX)f82TSj{>9rqM+<y!oC7-HTi=N>_sWNW@i;+M%umuajosgY2 zj5J)c69PM-5k^QOQ0k&rrK8y{chd_hZ64q9Mb3h1S*Z>{*%NqLF7lSL9i=8w+hK>o z!1Odjvmc%vo}rSTV*+FSNE{bL(Z<`vMWP59+lj4blQNn0Iq8;ct*`^A8yD5p$)8uJ zuddS1XM9#=hM{iuT4ZB>BkLqB3KCIq^(PRK$Ow;SXLO0cB|gW=(Qnjsz7ZIrWFo)z ztFyoVd!&AEpS}H^v#)*o>>Gdc;g`QIPA6xID4e1>LE#wt3T3uCbiwhWNOU<tYex!a z4gMNge%IY^e*Q4A&Bo*p3(ko%FYu`j#i>hFh2vDE%6!Vl#h)CL@u}MI$SkSiXuNiC z;>OOfGY@d^m?#*znL({XpkB&TdNh6D_lYBh>nP$*Z>*@X=$vc>lK^Kxn7?IYyOhWd z4XZ0T`1h^pCW!89z#kLaw-R4LT-|>kcFcpH{}@>1cYgTccfTa~Jh6=gnWNeFm$bR` zK$)RZJt-V4S&J{Y#_e9(acU{}e}L56mE54Us+wO>AKD!$5=y#TdXhONqGBrJ;^tBR zR-O$K$RnM0!j1~^W1W6!XKU<kGC$*e`QT$YferF(Uo=sqhjbM<p+6XWh>lO%zf*tB zsqxA%e-p7%PlASp6NXJ7D4u=&TMz&GpT#mK<M_$V;!P3qTqpiyM&jX_3dx4X@WTWB z@XKF)@J&|=0D_Pe<#B3NdH_EJexQ&UZ{Ln>Ul9V>eWCcFV30a<L@y>o6+-zSZ68qb z7(s6Zp#FYD7w<{On@J~OhW%0rgGjI)sd%e&RZ#JPirF2w+`rBJ@dNN$X^`P#CbPiQ z70$A#bZ+p1$&}`LGS{CSHrlcE^KMx&vbMTr)qV(^V3{2d9Gqa@w}4NA23BgnZ4|Mh z0)!?>HREbI{P;v}2cKKg9SjC1k`yf*lEO;&s%$2Md0(wSJ5G5o(vGY^ZmT9Pyzn4O zTNPB9R{H>ty@G+y;{?#*rQ|+RJ{a=kfqk-(m6$@PZd+)m_B{&$uG;|0syfY7IRTDw zbhd`;>#bFIq@;Zi=fh%Zzl*w9Q+-P#fJF5E;wTLYgN(NhtSk?nF=3ZY2q7(p0daC8 z6r<AQu_^QnJW`-ICU*87GtL!6zh~;j>2ye%eCIc3zxqd?prc#LWX4)138=C#Gn@VJ ze|;y|EP207)~-cL3@tQE43_kJu;!RajgyX$<N>|gq^w?o?sM|4+SVq5+j2^`xwzrh z1C>m#p0cWgE=I_bnZF=}I#lJK8`+?=V!mruyp4_MnwPWTFzcC>{UbuW+cJeY<C$QF z;?Xdxm7TGe$@OYCi-!tQW!TW%5RY?PGzKHm)z9jTcExA_(cxi9z$H8cNx@LcO}y_) z$pq)s>(`tF7{w5GqRbW8N%rDV7o*^}Tu7J}BN%MijeY<HH`{;`4DX|I+lt0;$j)P( zObXvRRwPppINkywCZAx;j4zt+`bEWqK&0M5JUrOQ@F25;*24+rvH+By1^#bvF^lf> zwPm)hvRsUu@S)*u3<^krFKHTx2dd3=yt15_8M^6bk6m;po3eLdVK)yn7C4AI*%91z z06mCQu5$HQkS04tjIKpF7e41yl26CgyA!N;Cs?nW_7j;XoyGmQSRdxuUC0Xe0zcEG z7I3y57=mJ4#{_{9=K|=($8}w2I+|-nQ}!4xIr#CW0*J|2pyz`*%m(M{=AE%1cag@O ztDob6s>Dn>sqiHe$(Ky*ICXW-CL`(aC1dE9tUg<F$(zJYW@IlKF63u2ayKoqju1ek zF5b|tU$oy&9O4o=V@V~v<%h(^2*>Lj7b;sSZB_ZxvBS!<VQiF=Ki&BLFdi;~-;MLe z=&#if9FfN4P0`FiqIxwa{On}Qg(17)SS0B70r+e)n)Pau8Hi4X>xRIGLf*-WHFry5 zCD=$u+P;o4n3veq&RL0RZ(qI#F;-V7NzV}AZjTp&iCj*?k5f8sLa}mO>jobe3R8OC z%F>U?ftfSx&UhGP8AAO|sEHrj31f-4OC}6*@=nkS8`}f95Uo`ifK=A75+ewnWF};) zD48Bsg%idUTT#0k7ey20o_N?MA9nKLL`7AsN`t2gWE7^{F$C*R$c03@c}}a+)2bCJ zCpgdpA)*b^qKNzZb(su!KOG*7c3bVNH_fkrB?3u8-X1Rt78p`9>yF?%03hn8S&<Ee zxY<S!F_)cEWefhAfXQ}el;gvxRQ#6}Q=gqypr9`u&~AdR98=aO2=OrOBv~<ibku3j z&CEVC{i(V6&zQKx++Tt})f&6C&yx~ueW;V*vIahS<0^A-r)*uuV;z^ew-l>oX?$g8 zK;;Km*s|X0UdZ>Ytd9m~b~$S(M&SZ}bJ65D?hU#$Q1}4B(o2RYmg+PSPhBwEeVvb) zr`2u{q-u5ssgizng3r{WQw>Em)#7=s!~LeB6AcSY3&KRz1tf7q+KqC<&$<nB<Ift6 zV+TTT=gQ%F0}2c(1Q(6W>NU7??kB`#%hbus297N)EzFSNKE+Q+Q;XlEvX<Vno`tfz zS`!>9$jihiDxW|YIaAm-@|aFXock`~jgTgV0TY9RsjPP9i+t4YhjNh8+ablrG#?Eh zvRY@O<ON!n25WdNtR4zUXm8Xd7d<En$fq$W%ukSWPn8WQaRgz*SW)UhNn&li`60^G zsa1zHE`sjCJIno2{0<n^A+9OrG?u7PGC}{S?JS?XkIP)+)T^ueD$AJtm8Gpd%ZDwK zqBwE#WxP39L8?56r|@iS<`Tk3m2wdA#t257K?;+S8!Z=wSIDx&I6|Mye6vZ*xGDG3 zIvI<g8JP<JK9vcb%;hsKol{tQ_?XBTyRZl;6R)j*mvy5js%q(!F7@E`B74tPre7FR zP+>THA+T`00V4yth~b3;#L}$XNxH)rHW~n#vZ-wdd&bgmbs`uZ$ikH?{s`VF$@8Rx z6jz-Bp%N?RF;O))kd3I+%@H~y`xN$cO!&1p1(QJ?#@!^8xY~RSsGLKXmC+Dg6%rZ) z_=A8u$S&kR#aNu^%;#0cQ<mk+CdDZ2B;xGIPVA!SsS&nGv5?0J=t9>%77l&#FF$4X z6Bq*$jX238g99WR8@OUBf}YY6|Lmzn(8GT6TvAOM2OJ7r&G6}#4T5)Hp<GLohC0DS zOOv3Amb^1DUR11eEn92y0B3t?p49Coe8S?M{7e!ou1URRF0QWDxi$ASd|88-LZfa6 z*7Yz?_xHh2K>A(cT5gJ86x?<+7(ghYP%LNxuaiJ9mO_Omf-E->RJ5M7_FMXLv7lUP zAdk|Lvw+}ci(Vgf_LGJ)8tz@4`)zDr`>@lzFYm%I5jRUz+p>^X$h9mRR8nU<;g35} zn(Os~87i1ant~gPvi~rQKzLsGgwqNE%&?)<MVV4a`Weyh07eX~@K^n#x>XyICf`6# zVOr30j(tSAhwhTqA36{h-Lm>%WNo2aK+ST%>&5PQHmK@&#WE=qfXW-YMRBUjTEjt= zHfI<Q)96e1KP^X%N2ZHV-@mMqUA&zXqh7^;a(rJoqDcotbA4$ks)vT;a7<E)tQe*j zR-|g2lyRhVm=idD*#dNVORpUy7-&FqnDQid8%>L={AS!0{zX$Ru#UNs(ZW@uVs`Ny zjH4lB%N#bNql2s<B|)KL(~(w3_s|s>w62Swl)q<<g<n9Y7MDtmuGrhsuS#MKo&}~! zg9&KWM1KV1w*+kb7McPc#@!Kz`=;(r2B$FLtSdNz$Mm3-#4HYI5dR+~Q8}|OHLowm z%-!)Mm}@~!nW=D+OJkhM2|ij)s8UbM?l|}M+>-s!oT!plg!_`0gi!r1;M!~GDYO8& zXdh^ZYWdPtxA!0)MqO)wlv1g|+W2U{o9)J3bq~*~U_i~r%~w`l+FrS~w(|1!mDOv@ za1lK}LuV1Zr<w?GolGo{@KCCbPS#Tou?E%YW<nad5rcjljGK~vlHU*~Fw9dZr68u4 z$OJ`uNibktU3OC=0E}YVcfnAlv)fhQvC`2&l62KCAt2R^nV^TGfqLJ`j{52cNCIea zL-qv%Gou>SmE?`CphxY4imu^%Jg5N{{t`PueaA+8sDmsOjpm40fJ(t~jkV1Mpu-Cg z#gl)J2IemSZvGa-7e9G<_^mmk=*5G&hj`O8zAqklJY^p)Y~f8SPX^t%ozw}NXhuZs zSXLU+_QxNv>o(Buf%$@fzku-NGS_=Sk{k^jTVC*&kFf@D78d+zY<jp}2;oU1^R&U1 zu6kdD7diKj`?hH>Hn}3VxP8_3X3dyX1DQ}(?(_?Ppe84gh`gE}+Y=?NlKQAW2cs-w zqhN@A+`PMac2p!e%6NO(F`&M7<>vM4P<Fiy*KajF5vNw$X>p7Cc2%4&8-Un4cE{S% z-2DyGTI2hc@odrRtW_AgEabKt+DW-Da2{*2OX394vD1p=`udB&`HzIudO&k17T57f zGt%AT=sxlWvvee2LVnQ%(%SA7FB0%Y1mkre>x=fO1f}w*4@z1`MGImCbFy@iZhsEr zin&kknDyWzzN!U<QO4suPbtKUY90=3Av2HKob@T`!!6Zeig61JSlGW7kFq5%W;Zfv zp~b!7BiLHVG^Rxb?ldqsc^kbH&AOxD^^o@42cJ{RBO|Mz#j=ujoRW7+ORn>sw$YEp z#~C>SEgki#d9ey{Jm8E)WU=x9`LNvtGx~5-DlR;(x`?%saa&)x7D*C47r4Gg9mCzM zIGBd+zg|r53R*PXiv<zNr;EIek6XKOk+fbfo<~@-t>}8Tn+lZht}pa0ywT4Bj{aW3 z9g{~{3qj1aqLqUj@*M#yz1;8QS=w=+z+r{2yIFsK3hPJlE+{QO^Q?f2a6sDXic{*c z;1ccvLXhCd@k8Z5f<9oQ7tA|iL7SK^uY<+&DH%Rrpo>bauMol%w2P_r>UP2_%0`0S z@`CT_+-gHrJ!nUPViLR-!}4mF=bi`6*0OtDl7$Kq2LXJEMC6Ub6$8gK-2@aj3=)db zhOl^(oreTn5wjv#Nbm_1lME59MnX=DcvAZvtZ1(BwC^b61YvC%Gt(1|gw6`84PK$N z=kDc~PiRjpDaZCxI{Zvxi{PYk5UBSx;5zFbsdd3`Z{N$;3RS^qBt+8FBG5+K7wlfY zos=Id7cbUTyTNA7_Ecm2EbqnL^j`9!X1aro@iD;UvdLfWaBy_{R-irVqF4(l8BZ{N zO+RTTMG@yGGTR>lR#2HH77$Y%-g+9=qJ@&-4&ZGE8p5n@x-F<BliM&p)X3~Hk*^Gh zWLM+6m7{LjOF`1wE6EV-e~(ylhVJeQh73{~ayG*q%;K5k;9j@Wa;ZgH;B2f~bK14o zn54VW1J+yxheTz1As3Ii^~Bccf^HdWU@9Z=r3Gdp$iBci_{VyaVumzFt*XFMRe_$% z3DDAw?DK(u6@p#h#sGBERc#`tq>yZ2z_bFVuNGm^V};grWCNRKDb@))>rTmaF_Td& z@N!oY1{rXQVk+SzWnxpMeTbY0CCFlBH))R`3=>Ww$*WXf9M!w=E~H3LR4_Y?8I9Gt z%zPJ|X9>^+R8g!;BO?G<FoE@%y8y<mJBc0A_M5EiQ1h_#5?KX#Ny!NiH{?Te#hbMz zW0Em98Eo1-KEWdrFUPE1hx{ibK@jBNUVg`Dg8WG|lf}Omy&`v<B6oD_m-)8kD2X~5 zBD#mc6M3HW9Dt$-X*cl`O$9;M6lc?kARdSrr1MF!gI6y*9CS-P6jKZ_?oLVK+OwzX z7MR3{9y>FrO=XC}9zo#MyR@p<yCdoeYLV$%AzEeBJ2+6ASP@s)Q3}p@n9QdGeG>vU zQa=F{?xH%$FiyKbXRylVl|0^uJK7@cBwI**m}3!)7!aS(5Ww|p03vX_&zJFpv(mC_ zy<XBuF<{(el?KF+Ug6OTULeZ`EKbT@OLRxDSP}P_n6*E_2yt(`WQakh7qn>@Dta_u zR+!dFiXpVplUkcfAJ0qf2|+$3a5^ZC+to$);uDtLR^kB)JxkH1WFbDs6~)&(TYhDA zZ)YlnR7=DpSpy+O%aa}mMFBbL<2aOkgEugPv!hHHvWCeteGU%+^$rqvDsfUE!FfBo z<ws~3X8Zfy#BRDyzRRv3q<`ycW7L%q0~j3*K7xf&J_0<a1Ix(nH);y2dW{OUC*5s` z%aZoeZk&rl4k+k>lG6n5-2pnaBUQIVDh(RjwBYOzMHb!M(HXZB@c2cRa1Re%K48Sz zf!iiegh*4E(L{|aEfbfaL^g7P;V@UoZacyVG~22xwX&5%iNZw}dgzb>5v@VM6nN6f z(;#v!a-b9_qU@$WCc~HlbY2veN}Ptw`GAHVT>3y!4P20rrq}>rm}T9eX<7m{142|c z8fP1U21rKG)tVj_Pc|DXh||M@V;L`C+E|vBpi2exwxFT5qB<lu(h6(~Ovn_12dlU! zjO?g0@kTL7Z;<ZD@phE&6%QlT9hy{##Q-BG#%0l2*Da7?HnM9>_JOZa3^Z*s83}vb zMngS)*-q0M_3FI_>6GP>j;P&D4r0iEn;Ub2J0L9}=Ws`I6b1l@3Cfn`O|f3~fn7M! zfp%d@cExFK=1P8)E$#}xf%6QaBVv*38BdbFBle3+&6QwxfH7pfA3<+MHM)=R$PA1Y z%!ZL)8}djb_lBJ1uHbu^ymXSow4IQb2BMc9*aS!LS4G+*x$h~OTCo<Vrk+<P>r*IZ z8W05ol|emT5jOo&#|N>0f(~V}Aio6oJHSwjX%V{;$sz2na&XX#9b*Drwxy*S)mGTf z;w-NQz)Pz+Yx<+%BF3++F^h%xZhy6qW)Ahtyr1IkQG8O^#HgQ%9>6816vHhTL!3py zMDByZ;fT3yZNJsiZNN;To)xyBOG+tWqCw=r&gSvv)R`P1D}D^;2CfpFynVrujt0ST z+QgCsFlRdfs{oTk3GD}|*vR7OG&W-0*mn&vmG=CqbpYayrrbNAa1zsABuP(FQo)a8 z`{MBKl2g7oTv5E}!4ca}i=IbWz%)aUD=`?5M+I|~0-Z+c_!ORb<cNqA1_{@#6@xuu z`t|y#a1Q1ZCV8wTFv$v}A?$P<=m0WQLtxIvMaHJ#PW=q7?&Z<g#GD?W6Oa2R5&4m2 zaEP}dT;g74ZyyL>B$%Duv<r7PQQep`Gq-eL7R18YX$pRd`(pDH)Gve}dJMjKK>J6) za2^Uqso^Xfym7a=ohL;J#9?qEFx%rcV8(e&EVf!lFEFN;^s%VyrQEagIp3}!kowfw zN7#eP6TmaCvM%YXXvaCv4hwNGX!~x{EbVecim>SGO0Y0tp{Q=DnAWV^52TL{4siUz zCBsx&PmJ0y@nSK`n@CW|#2A-@7n02?+Y2W~pN96`25M|@Z&RXCd0P?h?SUE?g$K0L z?a_sV1;*2310T{P#Z+Gh5Ycz5qOt4i)SfLM5M4P*hR({oSbCd0e(_Dlf|SkAIAQu+ zXGXAvv4XFjZlG;a_H-m5F_rEdS`g}MtFyD0rdP!jGWyp?`Ci;k8hj^dQX&Q0IXC15 zmS80`tBZ-0fO&wCLl0S2$Z-KALNHdb@p#wu^#v$1tfPLqcVgad$X>TGBeREOSs#fj z=?XzeJ05i7ek`3vpXB-x<C<vsz}(uN+Bb<hgHZR%8ePWfiy-L!J&ae|8mATHu$!B; z5+g0se9JVu+A1|Wa!0xeZ?`O8)W>1fWDQG3g(ln$fd)X1*0da5tQZG@Qr~26uEZng z@zw(V9HJ4bswM?};A12)0;hRaZVnFOf_UWJ(XJ3?>A^(>e2l7=I#i;o{e;SYG*sS0 z`bSLWOWK==&QGZ3{%DlFhP@c8ZEvGelFc-Ns>qL0^J~v|NzW$Mg=YQkNmv_N(wTsE z%v=_ld>i2$*h<FQ|LWTD!ubtp@Pbcj`fx<j%&{<TNo@lgj%sd@7_vk$o+;#_l^7Z# znwFu3vM1(f@;3$UBFz<8TXeQ1Uh_sRh}0nUO!}bh)21q{WMG}IW-3UqOZwSi7JY8+ z87t8n#+OsPNXM3(P4zp9vhs!0tke0Un>H*g7<h#$L%f~Ck?Ec`frj!ucp?x2D@rx0 zO|R&tU2Zv(<+8;;i>&h4vUYovqsfFd2)kd*2Z8T|HByl80rMTQ#~}Rdvn`@VgXFnP zWhyZYWhUctaoFqu6vO3m!!=N5iqjviV>n=U>5{=#Gjc#QD~Xda{z{{xMZb!%@Hd?{ zw}>XY1_urhJ8Y`2&-D)D8yUDyh?OHPLBSa=6@ORO`IMWAm8D!Of{O?1X-ak!dD2Z` zt%X=?6?2g#bYP!HsD)QNwOCAoM1opFtkXJG1HC%w{6NJJ!7jIe-^;|ojtUG|#!yZw z1K<s80El4RXo2maw3M)Jw(AYuHOM&;TDk|2|6s_wIm1@8(JUqV!WWWHP-<I4JBb{F zqxjL>3ia<Op*<?$kldp2qnQq_CK^6!x%yW<i^O84g#|Uit3iWIK_A7F*V4Wq2SxPR zOEwB#bC@7%A3?zXkJA1hIXE*dP<utE-|D69JS(!jAzGp){i#tg-A#A%I6s;0^q-lZ zpL=G$ez<g4v_AW6qZJo}<L5nH1{h~#dj{Uou3PTFT1cT$cQhJ&Iif(+8x3uM112_5 z@%7>WF3Hn<Aq6B05ZK0p3nES2>DV-)=2&ryqu8_TF43FNAJ?~AA9aXv|Dbc&h%qFQ z4K*hxi17KSh$184yo8aF|9HevenK|Cfug6C#FYoaJpvWD=EbCiD@9x&B1N00mvs_( zkV>5gq8c66ySS&F7QFQa@n0?fC8@vT2i&Dn`wQZPRx-R11J0Ol(I@4LXmE853t=f; zY=*_=m6~Lq{;9e7XJ;BsWoBhU1gA<8`}l^j)6`Z~7HqSYp#dh*n5<MYb^ln5W0lxU z9j8O`NXE5hkT+{ZFJ(tS`1wfjyT{eEWxS__g1e4r$6Aa#=iqd(m5&3TBb6tDn%>a$ z%l9%0#kNiQ$C6Z3Mx<AQt7Fdk(a2RVN<n=>!O9ne$O;ie2yqazA_VWH%7iw#6ep0o zyfs_u?55dsY}r5}_j<$FDUgsvUEm4&_*BO*mwXEJMq|uN$@Eo2Y^i4Iu=OOPa&&am zI+|}~`Tq3A?P=m}v3xQ8DS@yl#naK#hJx~%VWUdKG<pgM#}qS6Q_Ly%J5#1X#8YgN z3%PU?GO5D;n3I$3sa7f{sew0;hu~VXJK8Uz(J<{2IcGke($EU9Qess3A}&tIVvL<r zCPeOx(ieLjSJEi$*Psh20u?4d=OI1Jp8}TiBrZU9>5nnN^#I@%HEQqi*d%-vbXx2P zEU?OF2h!Myb>Sc8oTp*|W<XtV4w4)~sKQ1TKZ_9_Vdk0K^xzyN+RR;^fUv@>PQaPP zM)Yw6I;cRXC$W59A9a#K`9$D(6V-w0Hk|&mHZ$M?LDpf&P=_;lc?$L_Bl-X>_VhxJ zC?0#tG5Ay+jCTF9V=}N}ojH4h_3{g78+$5OBdx!iW7nU1+}PXdoQJxkj~FDu@pfZo z)I+bBJ6UbfEH5A4g!~MFN}>fW@A0u}PB++VGLaR@McJklZnqa*OpN;2cmI&|b&itb zUa|+x9aSN)oG!rZa7izq3qSWf)SW>D5?#Q-Qnlo))juJo$|f-oQG0?}v@3-b%-#Zu zIG2va9F1S`d|{}LeT!jI;ZW3#ic;=J>ukW8fE|sH1$MxJac6pG)aT1u2F)LGMqA(< z?t%!aYctkUib*9u4m{pvGd2voDG)#sGAUFrR9(BC$<+n#A2jju2dGw#@(c<(G_c#I zX5E7EIKU4;!*MT4qCX3@Z!h`s%1eRDo)ZK0%H)1cpQmmAwdvB}fwvdv3zM?Uek)FX zl!bDd5~y1y5Mb`V3*|0lju=okwz+ZM*`8&p45_ay>uXuSxK#<=r-%dS<>X{RlaeOI z?paXOk#CxjFH48u$Ej+*0@jsi0!9dG8BIF`Ks`5UjC^UlF7rOtFKyptN9r2wcGGqR zmU;YFvL1?Lb-dL#Jsl-}aiYBd%Qk9RX=pq&AYo*D^~<iijog*9?$JKYUj$9hl%X+p zX9q%X1-DwbN%9zJ`q#o9S8oQ+!a(zyILqhL=CA`)!gPcUfLqS<_@wn5Wr%$KG}=6F zMkiaySBtgbh6OA4@UjSE)g2`}E`?`FaT}O_#a>F%vCvszmSf?CKL}N7aRO1xI7Lli zDebA~<HT!Y4Df35Rz9F28Lo`-0wZS`;h*u*I_Lbhw-V)5!V}4U)M`sWMs@Hzsj*vT zO$R=gA+4M#@B>?2yzmE4N1QvLDTJbfbpOB-zvem-<Y$B>%GVTPsXYy~R<gkf=HwCM zX+(1~v-8d9a-4QX(OSO^g`#D!j)f8hgaAzPLo`Lj%GcIbR&T7Y!q-56jzs)#FPWAR z(ai81T(LDg7Oaqt!X>1<S239^3-Oc?#g45-d2?tx471z_G}PuCs<-H^#3DMX^XXW8 zdX{K36MV+m**ibH|J%Rr#e;|c`rppp|CU99Y2R`N*bCZ{NHAn@wTA#}&$Q-Sb2ZD9 zY^|RTQ;-P(>9KUe4y(){g)X}wodOBAUYe;ToV*ftURus=n=i?PVq1~?1_Y*79PZbw z>UN%563TB`*=pUgU$9%mv`1Uf3WW}&q<663`9eP~kWNi2X0<UMB&AR~Kg^Oad_2K& z;&j9VF*~20rSr(jv<p*vPe#b&+V%ZW+Nn2?apB^UDCu>zc`t9J9gq$0Nl}vfgum^0 z4NM>_7e&cPL?w4HOmD?_?&*;0Kripv5!mnLk~G&H6{UMsr)0&u35Lfo#u*HPYrJ=5 zMFAnKJ27!53T=6Plb^75!`sr%S#Ib0w$KlOAzjpiJ4D(E?yj`smzG7(x$FcxyT%|O zJG>LB%*2RTN}ehpPxj$J%bCEjeddZ8Xy2ra%0zU9f+gC(I!QOwl@28hDCvr#icS_H z9xo49mGZLX1e5I;l>@jxKcVCl-oHmO`6TJ%;}J$atQf@>`;vA{f@USK@$!t9^0P1@ zf-jMC65?M?!J-t7K3~21Dac%DN7eQNdaI{0vpf>-b^2v^jZ&gantcv6Vyy_twsSbv z*OJ{AuW4dR%!wbx+1zri!=9gRY=}7vpi8THp5^u0oou9{6G^+IKds5_AV9#sKdqSt zIBTn)tp!w6L_Dk!)kNJ|fVyK}t+HZrt<G<nZ?~M_!*(0IU+iWK&Wxv^96Fu6JYlp1 zn_`k3nN&|Ge+$T_S}wKb8Tfd<r-XQ`%uqrQB8)ghKfyc-5}?0;y>(^jQnz^}VPiJl z284%n1beAtSTX_E)o{6b)9j@Kv}&`A4>cUzNab0=jG4yR`(8>znE_=t?RQ9^+3Svq z0}?D7-m+BJ$acG&UaN9!uDlJ&+%T6_P&DRT7JEW4(V0%f{=NUZcOHE6J-KQG&F;bt z`?|#cgP*?r;Gchd_STO+{O&)Ted#YA{_BtKzxUT?U->#e`RY#|e)HD~78(~u4}S2K zhhKbewBJ3^Yb@(_u2_)uLUo#4v??ukIvbf^B{m+b!Gz)2ejhf;@p?rnAGP?Bu5CXl z17>mVoOMB3q4MW*Ol@E~e%L*V?NL<q#~kq3-T+|e1O)r?x<K9F<7|R_U{Wad*Bj+l zC&};Zn?Jq(?wb#O^yLqJ`d514N!~7eHUTR)Hd;F$U~&M#2<;Dk`RlW<{)m>LvFbTE zxwF?5^OJO){s^osluv?5P65G4$O!jV5o<@;p&`S)H#gB^^gj1}E-D5fHp3Ge02wmE zs@Y5TIbS81y<waccSMSnymu`*)PwL?P=Z!kdz0Q%8sU|!o8?P0Tb!7n4L`@$>}AkT zNW(0{=CAEZ!!~*s?TVKk9K};I9Pk*%ydcZn!9lz<)B2;W=(?^S?ZvQ<TG2I7#pfIb zpUjsb!W<ZZ<>1f9EvY052>lwM(414kT9aRg(?`ft&;SumB@rm-HV|f*9&MdKtVgiC z-p?Vo`j~0b1y{^+G7(8KylfXRk>u2^#*z=J8d-jMue~RUwA%?b%-4v(Ge)V5+a^y^ zg9aenHC>>TyozI)7=AY~iglKVQnaV6ZpQ~n+|l+$V330G?xLeytZ2cK4F*XcJwUia z7QmZ1T!NxeC2a$#8sNf75ykruNdT-=@LOvf61|5nVblaao~<>L+@7T^5&c`nK}mbh zd#W6ahsp1!D23E3$kRm!S$6lad+5y!+#Ngf;*{uS`zRLD?F=y4+`Y>uYn^&cyskO5 zVDXE`A;Wm_xP=#zAdiAghxPkXRC{hWf4(NZ)S+Wjv~b2b{hJFJLj%K<=7AMW@e}*W zF+{iqSzH_+9JnmJ#K)wUIqxpJMVfxh9&N8#*)DSH)d>t~#SpucbgVhmanQ`3`Gm#> z;in%`BlLHAD9u-S)42Q|zkLK#(FtQ-9MFPcw*3`Zxb3Jl8=fgh@hYQ_2?|8vQG%St zB)pTycHj+@&llo2ZQiO$WvdStSv8o~G3j1UL7gG#gImdHWZV>Z8eis~WfT6=7>`{H zHUF6SwRqD2605OiRI5Idje-tGJ04q;WV-5dWITx*q*zY|HORnvm>I*^sXAl$I#n*5 z<?PXLBFQR^Um!x0?J%vb9$op-I00{D^<*4d<Pxh#=+Y$o1eKNaE%b6s4!+w-k={#; z^HJx@T30a~)Ba(0S4KmD02rbj2owc1(G(dYAeYiPbrovAeqwi&6M%grb-X>TFj{jK zt$RrLgf#Iba|f{Mdc34j<xYAsvW&IH)h5cR*rC0S)I=*1%U3OyRvO0Y#i|vASZhp& z;Py^0862+Dw%{dSgU1tTXACVbRytSfgkZn<m+<*{+oxJ%I<R_7POF;#6#re0i{vxU zM6cYwRw|N$nymEYrXvI{)jqYiw^uW^n4=5}w5XHW4Ku;?$KsVn0D8MNUAfqF?5V(u zWV=>yymh+{e0E)FIw*h3Ts4iB>L7MY;u~au@Hq;x_wpebatva0p%Y7=gW6D;HFE6f zfmhBhCcCo?aD+r44REkxl*gQ^LC_ERCXNmhP`6+kUZO(xRNl|y!9m)dz9kBaqq&4^ zo!UFZdmCKA(EdiY*2Aj+gX3vNok2c!B^$M6m9d%?8IzujqRQ8)VKlX*ET*<1(Bb~$ z7b9@WE)L=}pm15y6&J?Xa4EVZ>7-rVs#7_ztEnKHPEg{~i8wBObK>tpc@VxKGcySy zP1drmnp&JaKK~C!ho#dfN3Cj7@yd~2PGv9x`KWXXm0>PB^Y;zTOBL2BIkC9(Mt;dK z-8DjA2N(nns<j1s7rsOpzHyL~`QZw0=$KdB8fp^(M)?8}^lfbAO8mXj-}Nx~xR~2C z`>|DuJBf<Dy$Ntpfr7og(n^&>XCGL0r4rO~1ko~{%re44OI(->1{VU$@l_ZoAG%Zm zcr~nXSBYoTr$PpCb{Xf%GZVWPBp&N$30Gq{4QZu@$;X|(u&vgld>th02!-#qyWtNb zU=d~wqqKmT(M}d9>z&LR;x_L+R~851#$2;(#FJrV#g~De$+0w2m73(Gs7`dOC4{cz z9vW)Ug9h+E|Aj427qkhgn7;o47L5yR*9gDW3@PhCcxo7+EtuA5@Xj$WTX5~oROG*D zGX725NxvW~7Fe{)gSajJtY>?}Bgn6H71%FAaD93GY9kuq=rV{u@-$dlVeS|pk9oQ~ z0=B115Tkfsj2kr-tw=ey-nhB3wz4YDvo6+$WNcUuq(};CDFjC_=%5_eI-qdcM=p$L z-qSFsIPU<v)mEm@*%26tAEX1(-UbY1{i^It!5+=rtQhM_HsYZeqh@3@qgQE8qPf-# zN`p}8DSrD@V-ag(Xnpo<RnVj4xSb3r)?wBgbW<`2eOia?WGJGKcNj>vJH)F7N+!gK z%C1Q)4%tdOLgQ@q2#>M9=oKzUD6O%lM4!3JEB$U#6w){e6F_h*LAYS-#z!!JI88YH zq7O%TI;5mQPe+AZ0_G%Xl*taJg;fB<LnFGD*Q2%dr=rWt>uc-H=(V+tmu|kY5xus2 z`}XpUjkVSF=;rNc<>rkmYa44fZ-`%4qvacSpytbKH?9ccn<8tS91n6BM3$qweG19n z)M}D_;4D(u>p{{^_tLf)UVncC`+py}9}IRaSovOBz{D0D2D_IIDGMA_^~TmRt=iGk zd=9crpv(Cpo1_=Cr}2&Yq2HpX2->#t_eIkH^SVstgxPIji+AG_m>h~_btFU8V_^(Q zQlJZ;^8M?KC^V$pKBIyyFc}RH#~P2ag2F@9{&WV-y`)9ql~=A`zq7q^^ZKouH&$<K zY~NVEzPf$u_UhF&NU>ED(9)!!I)+XB9^W8yCt?^E7!hIK0?PG@orK#t^pv?^7(_;I zJbg;_FIcv>h#kT2;4Ac{QH7>MX$3bjtWn2Ot_?2tkIACgQdm|K^cNIx{w#2nC8GXj z$;~aZ824|{bsjzGx0U!q>C!=fJfdJ&&}Ce4Ne=>@rVvC%Oy-WD=t$o<Wtf=kqa8vq zgi0+qgN&fHbX5tvGojGxQ2&B-_w2^INtdj>o{rQ9pe>NLuxY=M%`QYW4lYo0O6+Ce zw+)J|+0Uvsr@D|Fhu5g=3b&4-(N8sm3niAm7ea=<caszAIeT5v>*r6!Uf8|KM#7dI zJ<Y-@cM<eyO@P&u6-<2v2eDnS#C#R;)rc};I-|o!{6>3s((KV!SGy-(wA73***~kw z8i1W=wk0o*L8gsSuzCIA<q0i5U0(SFq@sbDCPYC3<2AxeOZ?g&9xRHN&%qn<<<h0# zy%AMF&Rx~sOt%*8l4z~!jl%UU(6AgASNQzlA1eG#TrcRT;B{kD*EeU;eC!OzcVRd+ zJnuGEKew^HzH#T;>h{fBbfzsun^dlb8wzv{5GxvfyoMSZ`3v@%{C+j=rQMSn8@tu= z6mvD#hVeedneJve{ubis<t!IR){6fKb)X8w5&ep42dYsBAor5`B?#(1B#o<~f~*~P zm*oRUybNU(tl}W<WJjt=`Uw_H`qxd$<mx`SqhLTk*KIiZk%Xn681USO8m6Nw#J=0u z*)=gEf8LHe>8QX2^}uZQ!D)a}>|v-Z!T7&2OuMaX%a>QL8PlrY__GeCOyWqcZ>l`H zHA!%x9ZGKt-||c_A2{ul`-%{j9bstkl#VW16gg$0s5LdnM_N!^S(?KNsUi{F74i>F ziX6k%5fuJ<sPn56=CM6|w$-p$0${f(5v2?|BuFSmEYM_?xBwwb0Hez{NSm{B&1i0Z zZoV14!4BSYbMvQ7criB%ueS7?O)R^0y2VWjN_UWSAwo7cEe`)w$m|85<w*aay;k6O z`2ybufnm!R@MQUdJddEt+A!(a{E2mcWK@%Gz~-ChLoP#C80@Rixw>q`<y`8TqcqBZ zvAbd>o#*PO$?rff8U|T>L5-?K$XQSiTvq0zoFNFQD|qleIx?t!gUVejuRPYA6EG!( z$tFI^<k`uQCH`+>8>JHv!oS92DZ;TAjx7?$hr0&jC9KjlV{9{{qL>91vR#S?vLoj6 zhF{X_L&KfW>q6+`TtCUNpeq#AznA1@IDachfz5DHH=dWvoi8vrmEbgz!s!(u><T`$ zKNNf!H4<R;zF-491<SIP!H_1T2#5nl(c4KOc$_9N&YN#w8w*jZ)oSvh!5?B>zHz$c zFthF_6>{D&$k{5nhI!vhz4v0G^azO?*($!TQ?aH!1e3FrSjWUrklql4OGx%C^BnS8 zugi?mRKkd@26lH;sDaUaV8Za^X_+aEInx5sTU8eY2Q38BkSQwW@v`Xp8a!y=Exnyq zL_u)++0@ud;sP^!qPz@?PaGae0hygJLbA+Z0IMIam7_l&CHV<O62>4!#Sn#?CXa~m zkQS5{ZJ4R*<gXHmH3Xm)abSGyg0e-7a7fw<jwOazWWL>+8&JsE4+Jy4xz%VDph5Bc z%L>XaET)~Onu?o2C$rtxn}pfWEsR{EqLlJ#HGF2Q1p;YNb5a5bx~Qg22E{|ogWY+v zA8q+G)Ef?qrf8Zf918aY9WOJU1Jlh8l4^)ckQ5LxI~9v$cw(5w+R_Ed5rj)ES_C8t z_G*K2LWpSbM1y!HLl**WggAKwNx23v-GbOS0pOTCs{w+`;YC-RHwx<ZaHh~hQHcLm z{Yz1;W>*f%7GBp4FpGY1YbG~w&bI8hGL_T9Qxlw}N1OmuZ~+!XEe9_lk?u)M6$;-J z8&Gh!5OEmvM;NRy?Jya}z1~Up_L4m5y8#P%8tcaFAqrumT9e4n!@{L5iQBJg!H8|N z)>#94WMdbND~Y`em!ghiv-ZNnNB=>V9{#cD+D(;fzGqN4Xz61`&-5m7cF(`prU`Bd z<Sqbb$W~Ff>$saIap#0W&UTC(${YYY2ncXb2V-4kC#7)Da#C0_OFG<?m8R4H1nm?> zy5E;N<~RbMbHNQ12WDnsPG|&%n57+pP;E_lMItjsfgvkcVWpk$2A^Ov8H&8%OU@Mv z7XhS06>Xu09d{aY<?^jyg#w`n5fhBZW8zh)w&*`f?WFpQKu)fO$z2V*Uz2gfc{(vq z0|KH<3y{<xSY7&>lboKGki!t=d-7ytSFFgp6COzzStNRX$=*_d-HLAqqvD|chGgkg zA5KH2!J6za&qXkN`R8t8VH(9B-UiI+)%mZG3LH+KqB*2}$dr0FIa%xPWjm4J&bpoR z2v1K`fFVwt<b`tSrCgQ*ZB5!AjD`Uz=Wepr5u7@xKkEV$)N50&c`ea5o+&O!v5*h& zwi5+FA}4gghPhEx5E2DKNC7N8wg3e0KP|0@^8zlins_}q5-sdQ_^+6<rz$8oijI7T zkyRj=2?;_4N>5h%5!_`0$y5-AnL#GAZbuhIUZhBP@nKDHct%M|%V*h?DU(r77THy% zsws{}?e9T%q|tyb&jJ_JOKWT<T{7<Y;$DU~-i+$P5Xy9BVRLwd)03ztCNVWCvM5OR zdkqvz`$W&oUaXQ14-X+Yv|%)S-J6O9)6()%+7Z$W&<sHrhRI7}L5SamOvOjOT2SV3 zFX(YEbkeKiRd5M#e+s%Q_i$D3L0<xW04u+`eA<G83hF6ijmPg%raO~K>ZnVIlQPL6 zB$Z9u;0^@X(rn-|su!u(`=ZzMvH?&1hrl3Ww`{;Q0ZpxsVj)83xQy_@Rdog1dd4ax z`Cc23SpbDN92`CqLyFgGK7cq*0gR)_dpFUWA$0`YC%!t$=F5DqJ8h0((LKOSmJ_Z5 z&o4ZCtOjx#I@Jm4L>7>_V3{DQ&CT5HnaqR$16+%ydWj}4W(IBcys&dkT3|5q61qIb zLvC4<J7n`}D48kxR#ODsRtYGI$@yaPglK?jT<SdA!1P)9uNl{Jw6NOBWw00YdtWkI zP)otBq)35bS*UwYZnhv`l7z_w5m_^Js-gjfS5-2D*dwsw*-c1)g@hH&f4XP`E=~=X zRa>aEySPNF-CKfkDBjYOc*GD(2JQ1XCiti@LiWzDhRS6D^A%9v@UWy7H?j*wIvmBr zEOohbxMQNtxx64)BT^vL;u<hR)(^oSZIxBMENT+}R8rrsxlXgKOtaK<E7J|`73CtM zW;`fb9wTPLnQS^9Z;X15tO=h^7UIP%%!9>_z`M0qF3G0|_tK?uMjSq38w$3luFn(= zzftNcs}BKPI&hZS!RXQg*v56q$JG_TaUFOeD>x(XdAkITPP=ie6$Vd;9N7-;jGKEq zaGw@U%>sk?d^FRtgn)||F{B=xOUTf0oqENd^Y2VT9K8w)B83v0%Qj*jIX6$f)~Ni> zDWq9)#5(Im-P2b#)QH6@*-qNqVyzGZG2@PMSzgb9M1;4#AQCBt0_VF>n_(B!bhH=n z>tzyCV$$nFCh;Qa9wve-MMpCMU_Z;j7f>8gL|9FUDJ&&%8`lI81yVrOAVSYvldkq) zXyVPh`29>XdUi|faUnKsjR$)9QGq&AlnuNY*}p|=BuNTz^yM;+s2Lf*E^oPO1oi=# zIpXW2sPA$m>badBA|5Q8p)&c|Wjks)2y(!JGd}t50_w{BYn8Zcl(?+RS&MX2Je5Xw zSOCizPcApMR2@sAgl39tl(!QcAJ+Tyr)N2Vns{qx+6h}I>H6oeQsSAm>w}g<zFG7u z@ruB7b4$Lxp#w;*AJ@AXnNJws6$_pmGV<R1eN*;k%R2NeWW27B0pE;1Dr~$?*mzxh z3!uXpU2<$g*?iq^8_xk?6x>nBt{cxb72K_0)9|UNQ?7zO^&f?J4f(0l($W}(h3!8? z5v5^wV0L`#K?5*v!JB&mmpR!Uc;bt?WZX0j&5Y%r_IalSFU;tbYZRBFi2x5c-7018 zB$Pnk%#W#+yOd6lhz>qg6XfqO4`HI)O|%>>O`zwketn<E3PergQ;WtMRlS6nfzgD6 z7Q5O;ejQA;=U7WX`_mB6o-9>@{KX(X>1J^Uo!(+0U>SS1-V+=~5#{j_U*J#cmutvs zN7K@Ys$7AHx*TQyzycc105DX8JnhB#396?YMIxKsxakjc|JL?U1l<EWyr#sHa-N@- z*%7R>56uIvNDye-(v&!`;$ZW$mOJ9m66dB%s^~dW%Qe_Q6=*ic@+&#_6EC0*fmh1Z zao8s8=7-xtBR1yhZu8u8EtmKAvcv@I`kfww5>qgr1RH(dD%VCt-^t*4E*zkC`Vq#D z^{g(I?A}$1!8c&4s$vp@M??Q4Ish9Ma`&Shaw8k2;EM$kC2$L|G--j|pb}hb_V;QA z+N&56NiE<5N^j6dzp;DyE9tP?LR~z{4cJSns1ODQKC^Y`h#(sc(+nTMs5dPrC{G7C z<>qGR7Ik64=Hic2ANpL~0t-FFYmWl6Awe55|BNg`udNnm{U!sYz<H*2d|WH*m`#P0 zbF52A42+p3{!#~_3<)S17>Ff_3}X(?MPiN&l{Uo87rojR;**e}aKPIrbHle+`l%pW zxN|OD0;vmMT?dE%TnIl|uMXh}uBOLA3Vv2DGVZ@#^VaOytDRpdtN*(7z|ydiP^T7< z(KXgMj6IadgpYDkokmF;g>kC_n_bEi3Y^N>(TG2S@Kc#)l(hC+&8W7YWHm7Hg7fMP z@q-WGnbVe2kAODW<dmawu;&Z<!WC*Gg0=<|Q7@P<-nif#c6|5~nU$P?{C0!Yc~%YC zzb1a`;GbZ@T|vXRB!2DC-(Xp7XxHTL9sUw5eqt0qk;P9+i|^-YNA?-McIa=gtk_IZ zozMB?cnA)v9HpK=r{<4UU=Iw8b4A<C+2#ShVEg8n&<s)sNo{E9DEs0`=W!vL4^En3 z@HT64jGhtgntyRBQ3AyTfdQ-gytY+3^yA}W+Bx!0G8y6pf86cXB=ZHX-lALs(+u>p zexlwWv8NlihO@}UD~wLbDWY^hA+6$4{1TV7t=7$;N#*HdTI~FEU2GwY&>qr@C<iL6 zk@nl&Q74f#3P<Z9+ht(|wA+(xJ|rQ9z`F5swrRec^C!!AyAatG`M!5A%?rL^4y*>% zsQf<LRF4DA$yW=JQGR?w@?<jCk9i6SWEaU*T;_GArA=P;c1}|gzt(X!>QtcN-#6~Q zZfbkrBOYz?HWDWfuGiX@Ysr+d6S^p@c^rMZIqQT}{PI4~dAX>&>%7tJft`+!MC)># zE_~I|#`)@W&RGHWx~yVrm9WlcX}?SaT9{H&x}ja;3rpD#hl@pTLvJG4zf^Vv&hbU( zig5K9w_;RNY&qS!q2eS;yWRb`3$P30iUDQ?S(T+FtDfK){Kq8y)Jkk`ByWK|U35!~ zv!%mj-+39iCIP60MxFKPKdb-g(WOQmoBzUE=L_hr_vz^dDIA^4*q&aF3f$(bTM7k6 zE>_p&=8#bW{r51tmK{N!JuhSgUgh?@)e9fEj8~AoS0&!T{I+g%2@syr9EkNQ>n-t7 z(eUe<7x4})#_M=i=7-)l6~-FTbJ2`>*MELvNF2j0JSoItmw*>k6DA1)JnAuDG{&rO zg%9D}^tz%O+Eg$?>(1*TTNW)z7PW=dmFr1S*N{!QkXLul*z<H!;HD>_6BIW}Z_day zInD-6kp`e!=r90`m3#jKL>`vF?(42P4@-N%1Y5E^<d6%Lg9ZJYZ5bMq6&eNg2w(8t z)s6bP+(HDvEwbSEWk1vxcYx5|ennsQoxWI|kbMX>y^&?zVLG7LaxhKxTzz%!+F;GP zFKK6D)JHcU^(zJNYX`$#m+u~Bkn5dD`u@QEKe*>^U}!37_Uy@Wl%+<DLgzSsA*uo7 zniyS8LI|b%@dZO~1VHl;R*5R(#XK7g1moRBd-@Zk&03~5Im1)sm=+2|C6N6xJLL0+ zi#&bGy2DiZOxiv6{rO$qlB|9#u97;mfTx|sxViE3+F71VkA`V?nyt=Gquo*aZZd?} z1aJaQJx|^9T`~bNpm(fu01FzSb=k(Cf|hNv40KT{@eTH1R)V!{nLNIL^bFK?-L<r; za|h(Tec<>&egKu;@V+U#Ebs1{;NxT-3{<wnDo)M}0A=KBbn+J?ar!3#ggyx%^hp4r z2?Gdi+_|;7eP#99+V!=K)!U$3Yyr)NS8MCrVjvsKYd7e{_V)Jb%C=#vqF;+wb~GMR zJvWJAUV-f5z_ru`A1p1CXj2!&(}@G8hLKR<i;TyjY+XD#Rsj_BOd0DC!)cJo9CN(~ z;2WZ2xHf#OH>Ob#{B9Lu&W3dxc{X=9Y(c|rMevPAdB7<vy5<#~wTqrQ44vmuP=Yb^ z%+OO19&x}#c>2#a<EeYgQ=gxi`s3}bOZCmxmi3_V!l$QO!=xBGAPuX;nvKM#KpkyA z5^U^KAVUhe*Jbm`Dyt`RT$GUr2eyjZ!#ac|#NQx+0C~^ylk4c>Z|0y>rCOgh(s>E& zs~3xILDEiIiO4WLm;qHTsBN(xUftgI#0x%s-F}$|F-0RNy%WEPO>R=Da(f0CQ`Svd zg2wgh;UYLS)PZOR=7i%Qh{@P73<rTQ;YpWtYUO|~F)mu@I3QR^Ah-b3cnZ6&0pMFD zfF`;0F0~4z$UlM#;$+#WJLUUT)-TH<C7~tZqj7C5kRZZ!vn@D+PVn-nDLB@BWXcQg zBG6eBB_;`e>*oe<$77-eT(_{GGu$<)L0c(dmsL9e15MJe#j?VL70^g8Me2KX>17z6 zERH`+xxO!s%v4CyRz*$_Ii?pp-};9&NG7IMMf(Pej>0y)fUxA=<(2dn26jmYy=YPZ zgYf+x*28epAVk6;0hbZZogcX)(Zu1=PU_w3f^QuPo{D$uQ*bxKAwU~X0k^Z#z3167 z7?g4a4uiTVWSx{aJP>t3V!+GAwAgNU?};;0jr`K3X316Jd}FKidY1O<wQZH62D+iv z*QSEMrIT0aGdM&t)p>LgAn8#yqnZ)bM-2ZdycA&%1HQdz)r=c--5(9bA^=(VuJhu> zx>kS~BWSqm1B|WJf@kV^5FaLlI^^(8uLJJtUIO0b7!0Te>$pZyjcn6&bb-L<1Z$b% zz|-LOn#P;UYLm#KP4wr;F=CCX5$&PJn6b_3wY^%ygtX#9y4;k}lL7#-BF0!mbp$^b z%)SCvu#Xxj#$&;pX4xRh+U-$}mIqpSBvDxBpw>+WhN@X}u?p&kI~oIIC{oyr=A0XC z3b(5Xv|f%H@D|fm)g}NBXtp%E0=?r}j&kYon|JTWrD8%1Hyxe?Uu9;{8h$4&!0}=Y zDT$1|`<j?CB7O2kZXBa^J?;zH0}x)PUPw1lJ^V}sHj&@T6Me7P?wPb`q)VoFFGT$X zn_+CKcNE-7{e+Jy$>#%SfHC<^u^em1$Hz6Ro*=&*6!|&9hrbN!2^-Y&Rj^toCmI}Q z8s_<skEiwO)oG{An;XQXniro*5l`;jYewbLO0_J5$f*(o96rL8eZ~k~){!cEbI}o) zM!Th)Z5hvOmSMbY+EC&2EN6vjC<Z(1R|ck3x+m8{Tqt}3#-xEq6+ILv#2K7`8}ud; zMMJo@FvamwLu#EqW(f_CR^;#M31>`E(M_-Y2?pRBqM8-FrM{mG{Z+30Zupi1MRxjv ziSB`5-N=Thyz^9+*j}{c&x0~M@Gjk`ia6r&%chB8e^n{?)FmP6P%ZwwWv6OqoB)qN zaKG5?d4s^qV+0e(S0>7$zxXQTsSAj57rEu|qJ1Iq5I_<b{2!^5mh^ONrxN*02ymKK zTT;9SyDsrV*Pgh+F!UmAvgjHZ5vN14WpLXomGREArjp|FiD5zzSYKn6J?9~K<wo<` zQlf%3c#G2DI;$5Xs}%75&R}PoXd@k@5JPz{YN_@dEA`B`UR@0C^N{pv0TPV$nEAAg z)!hjd%x(pd@Fr1kqW{;Z7;4eW$9C&+rrhcK#f#OmY7fa|dtB!&W%_gtqj0e0*zRs^ ztBJ9u46y-Twgms8vS&@ZTVT1{WmZjI*OzKLFgGg~u@)|`EFPa1T`su35D-oL&_UQ~ z0DD-x5=(1)Am_pvJ!Ec6A8w%L22lvkMuensE91G#`@P69tO)x?25>3huM;VobOAp4 zlA4rKI7lob6=fUB?ognIs)A*`T*gGWYsq9jCL0Gcin1D5yJ$9~n)*%Hm}{nHRkfRl zyrjN9Q=_W73I_IFFOjc&13HdlxhnN)JVTyl4dGga{d2&qTB4+moKHw6YB$pYv7Hr; zN<HX;QMLrW{NgI>v^v;3Uwn)!v|0_63C6bZucD^fSePZ+PcDZ)xg7rFa(KeaVK6zq zdF5tw3yz40LgvdP;PTx~h}R-sb-UsnCa+|#BGQ_}*@4m7A*d{62{uU+Z%)%I=7#iZ z=2IvTH&j<F4TC6eaJbxNy0tm$A0Tc4I1a5xGh0EzWRNA}?v4zq6S9>pAT|(v^C&-} z&^?tNai9_=6y>ZJ?x-AE3#h5fUYdrSy5|yD1ES;21mYTkeUBWT2|_+IK|JR{2p&g3 zOVhq*-+Ss0^)T*AgGb<Y*J<(9Rf~b~LX|3qW1NvqckbD^@H`q11Y}+hEPit4d~)V| za^{@NAW3X$NKLTLWYRf9SPizR_ueLEjZq0yD67m?(<~scmfcNso*MgYCgxZ$mu?-M z_9gT}W`8MvcPIFc^1iV2F8+`z3C7)(gT9^~qumfwDia@84$<Q=2aH*f`FK^wplT3s zj6%PHTN(7?3PjF4I!JI`bi@z1yi<n65KfX65*Cmoj|*BYVj&%-*{I;2;LQmI>y`N) zx-EQVXk+NNYMt#2cnI*Qu1nxt2UH3c`Bhm*4Hzc+G$5CAP;XR-ZuihIN?7qZPVI?+ zVk~FN(bQ4%T60DJPKK#*L_xe^@5RTu7skb=DlW#t<#kt{j-K&RQrCR|lky_#rXBQ# zFrJ6PkfWt}6767~wjGt+U<ZosNHP#1I0O}m#Q`bUIs6=#>)~;+>qoc~OCif;_$+H_ z?8vpT(_)uh8T1t6Pw%K*f|BHpsO$mXsnMFPasE1)UaRHT6=ew}J3y2%={bA&HHpr- zpydMH5!4O>B?2mXh>I@fLc)NhG65mQ!B!h19@2F`l*Sm>xd%U^6f`3=@@FIhW+6^t zM#nI-4@vJ}07=yZX@iJX7&UPR4bC}rd+YUL$BFs&0;S!vA1*}p5@FoAH8<s&YpT_^ z<pSE|W!#kOS?QrZVTC?ng+5`09)%TJ&Xf2YtdLj_-7-GsiE}hnj8i$E6U!0#e2yAy zh|ehzsrjHYYdg%2QPHPIq)2@5{hvSl)$cy|?U&BJ`ad81<i}@!_qU{9WQ#n@tEWB6 z4>|i+JQ^NAXd*GeV&?0xSqk1Q`(i1Lc3WaYPV?)AQbLkyWO#5ZJCY~<AUj&rFV_D& zr_{apt`NTz_EvALuU)%&19SJzCR3kXv>tEYc;)))?X?wp`HZTy@zU+p)$R5F`}PJd zBiTZIzQOfx-dNq<cxmnS6{s0aNAruU0=ob1zdrceU!1-Doe#hKHWl4IoKcQPby<J= za5nUC&VFcxpIpYYh@VgL3<UH*s@&-LXi5*4yjG*<GzLf;Y=-&Fh8pOnXUoBAsN@$a zpiVhp<;wWrogY7Z=Wp-7`;`a({NoS)_3zGp{KK=Kee>+i_j!FJ5NG1%UKDcwG8T%w zlcKm3r_o@h<oRsL^SP4e^Wo>5xz-Ml<++-*N4sfzc~~FJG)2qe-<<e2FE}5|+)f<3 zLtGV8k-Vstw@>iqY5Awle+M(srD$G!!5^~@V|fW5V|jc(&xK{>InVknJOe*{|G~F@ z_Md4Ut_ZgCFoyKLk1->FNyAX2ULQcA!AwJ9_);_nMFw;5corTH<vU#sEc)l6z+8jp zieFAPHxIu3-ou~#^@AV1_iyii?f!3m_wbuPdHDVBfB5D9_v}mmeD?m^AMXh}jJx&= zH>g~mWy2yktI>lWeC5Hn-V);$EA!#^-wDn#xReCuS2Wus(55nc6-*o<;4QvEqgsHW z&c;G7i5N7$axd|$i`j#@U(}_*#c1A`fAMt=95rT4$I-Sb#B0ZR2e*r2e^0SuRKI}O zue+ksu6PHPW3X^HDqiOIZSnq(`F+QD4<}Ze-_Ii)daq0LmE}D=p&%v35whH%<dnvv z%MDMox8Cmt!sP>^!9~jsTQbJx1<fen{F`cvqjPr=CT}QacPM6eC}ww8N`&`Bv%Y#- zR0Tyl^j~~^PHgOOz&Y$&|I#JAr*^}F24jGRI*ikp7BI_~1(Gidgg+BO%;l$}70(pO z!WY!?Q3mi{Mqty?@?sfccE#IREVY*xDF?_b1YIh{3f2SOVd`a3Z;tCAHa~jwDY$bQ z?%bqs=O(%?P==m35zj|6L9D_{;L~iD4QC;M=R1JsW!yTKqS@B7yvgcQm&I=hDp+x@ zwBl50Md-X}@wq_P;e(x-Vjj0Xbw^CM;xHf4B?>`Gi01)_U7aiM>YUxxxhK%oIkT(h z0I2p$eF_LEVEB@N)_{PjbtJ2ydH9IvN7)Ebo`62iG1)L_{$=bcyfK#SG8bHKgw~%B zeuuG?b>enb)HfO|fw)DN{q3+E1lb!%fB+9$@Qu_RY~ygSfu{jLInEVzf_j%!ojSC7 z$rQDoX8A4ca~CQm5`oT#flfUt(Emu}QNf+paOWq5>n@hk(@@5;I}aoK!~go>cfTP- zjfda+%lp6i`orJ6^YLV1T}~BJmnXfrl2<KfR9C6C$Eh+@I~<9ag3qAvCYI{#9EyEn z4dWw|{e)fq$MWqwNq<B-kf+2~b4Gc|i6SF9FQNG?p<Fr^w-SfAKUYH#U;$lb?vZyC zW9o@=Y?8%(#$mC87zp!gvRlaHB438&+M*pFF{`48uECfD`=Kh#DJng_!bdYj3i4!R z&xJgmFC0Ul9{&2T&ffYfmq4M~^5aR492H0zmC8oLApKb%?RL_`w3GO{kS0{J2+&{# zSCpab7@}Bf(2~I{t|3$L0YzrbB6E2DTB?srM3Jnf2+%A-n$e(UX6@z(*c?Ke)!=4* zc=HT!o*>Q{AmzHjv{3sq^g|4vwyzo?JRt)%)eCqu$B*WWXY=yeyajqzmw3i40S}+W zhxW72;4|ydAK{~UclAE_>bK5*_LmR8_}<xDKMStm`|p1B-`@Y)2fzE~+4p})g+KV^ z-#z@tU;lq^e$fUX4z9DefBL~+{O11q-xJ^Ozx(F>cmL_ZTi^cR%`ct(!}r8@1oPbw zzx=hcxBrRYoqg#S_uu^&g7@GDZ$A9{e}R)6D@yRvkXSC-<C);&lknr?*>LIb<Kww- z>G0#@`Ecp*<Ku8)>#4f~H^sbdX#@xPh2R#wm1p4OdMiu&LsmK-i@kax{vC_`dLsTE zi#>ZH{vC^bdos`0L=Z^2NgLnPcJWo?AnV1)^7<srrgxKjX_CWL?DVGw)S|8W8XDV9 z^d!m2LU5)vGxMyYx?1b+rG2s^SNktd4vn9XC)%uJ?&*%+x!Z>`v;&V9^#k01C#sAX z(%F~Zdie9defX{ae*gWS2@Vt2++Wl36-!X8#j~HiP3uzp6kDG3*;Mz|>WwRFH(sRQ z$~<&?NcN%P?{m>C{=I}_ebzdLwhw2ie^SMN%#(7=ld)|{^iX`(EDm!B^9OTx;uY@z zwS^1V^*<lQc~TdHZvqUn46+QMAn@H4U<li8E^?TjPQaFi|2VU!u(S-G<HazVpMyAm zd8{*-hb(I6=fbwaYS}#JUZi=QcT-Fo2RSSyIO_!!mjI1|u(I&&rO;c@ZyEK@F_X~h zx#m(<sG84Nn9mtMNG%Rg@;?>RYwW6V6AYh{G>s;%?3dezbJbV@Y`cshRI0>Gs8VOn z#m!s*HxfTM$-opMe&&MskvK9)jfbZ(a&npCNv-3jfhIg#4#eJEfNCv8namY@6t_`^ zYYS(J6DdQoDy<urF8bYUUv0%^q|!R^S!4<pWfxJu`kG)ZgxK!i)P7WK?_@W8V(oN* zIJvj2=e%jPu2|<Hb;WAjsv^5O8EWh+)Q<^*O0PjRsu&brWY!T`4fDE+k)c@qiPws{ zxBPcO$dkWdR5%P=G6pUf)wSUaD?d>!V<})ju`^e471%Dj3jBC$WuE*k+7r;p*!N_Q zt=4#CP5*yZJ<so5xqU7rG!mvVCG>^gi?U}@>P=eXQO}?~7QG>rPEMrb5-E&{^cVy~ z6@uKrZh!|ti69UF1f|TuB0|6cvr6o!oE7S<W`L?gm)DVs!6`Z|`={%Lba2l6e2jE( z=hwBVnG5UNJ~~}nwHi&ahL5TAGT|5+uK9po>={;Yc{(qTp|X@&p~ditGOJ9P6)tQ& z4J_JoDyP(%o?Jn7;p3y2y6_)M!DLD{reIj2U^0uFlA&FkRBxXsnVfDa;#sL^l4{Jy zQ8ty$YN@iRY+ik3Q+RIw!zi4}FjJ{;D#K2w5^0*tl}={a|MV)SGHi!cPGxBRXjD$p z`{Pa;tsf+P<^3^<=g0N5|LJ*YqSQ-45>T(=W8+nPY`j_*wX|1=Rcj?H^s9DO@T>oy zyen93BL<>hkblU*2OBm}<D+{C&BeF$51h~j_K+Crh3p=3Yf=iPhd`kvz4Q=z?V*H{ z^mDv~{6#b)OEa1oxeHF9(1XDqjYb;n=xN4UjrJ#zxIb}y_CAq6Bz%BjoH(-K69`En zdi5n<)SH~<2t%>F`}FYr1{|8tKkx6}gM)&hO#9*1uR-nNyuJE$e|6~`n)c#rdwxB2 zT2I@{#nh=dbrrU<=WPxTXQ|(S9>PnXVHL9c!16xd<{dEI4A-DdKd7Y#0mn4aYnb=F z^a5)4dPBHZpzg7o?-8SrC0zNgqn6~3voq!c18!km$~1z<hTk3`q@`4U-cXUN8H%Aq zANaq`tg2?mnqEeH#f1wNt>G%y2-iWcU#zPo&nU9$YBcFS5DM&Laz1<2_h{dJq-aL3 z4T*EV3sizLMeRtOC#9@0g_M7E9#Eiv<AjXJ;o1pQe7>F;JDJjv+qb6hU~TlLtoS*- zbLm4%(^uv86o6y%ed#f0CITO1dVU_W2gZmKk4Nf(G5+*?g&g)Qh==6Fto`*~33h_1 z4dHjgelzS^{{Hyz<wgW?;=Pl>2B7xO*u`_cg78E(_X@U+y1h*xOpyg``2RF}i?N5t zB{^j*7<Vhivs($w4Bt<L6jw*>hJnoqgGXZ)?;h~EAdA=(*s;rovP_`*Dik@<NRHEj z6wPViV4g#6W_A};kZ{{`xS#^zkXhp@_YE61t{wGkQavq+ww@B93Mw%4lsHsS!P>Pb zg-JD%c_dWBn2|c_E_EcN{td9zc;qoz4Ms-lsJqmWka{MZ(dF%8EujC0VE#n#{5phK z$}b4uobeA}KB{d3kGYCGHmU@GW`uE!=HOcM7^PtHmQ&1{@+NL3L!yb=h1AE8)S<yI z<(!>7Ls#12BA?vB+|rPiEY?zof$5;w9=gv9s?N7hA(+~@F+w!EF>0<(hfczNRr9_z znmGxl8-);-+Xt&j2hR`8ri1gBNXz?wL7GH__S;4K=f@_~9<FTJ2m<U`YEs`B%#Q#7 z0RR8&y=i+J$CWVpeV)hvq2)M3Py#`MvfoT7(;kYFW8KkaQBGp>)hjlECfOE%M!EqK zaH9J-lf<!;IL^#uwUfzmlSyWY<9R2G*W|~%A|-#if8m~`_Ns1xl)Pka@Yo{IU3KbI z)u~gbPMrlGhfC8@ex<kFNrJBdx$>8B52m=2wQh(M!S|eDtmI0XBkg()e{%S%k*+HY z>oz(Q1iam(wRnL7h_C|1dLEC6Hb&MWz)W0H9TIOAHH||wr;uRU)~9=_^l{RpJWN3~ zNY923AEBS_1?9N{qw1u2t78OOGMC8+8p}r|LdO_ij!R^zQFOwoX)J^>zBFsPY6t#8 zeby_C^hU-&;5A&1?Z{@hyjp6cf^y4a(^?3Hfpp=kv2~$iSqHXN)`l!*ye7$l&>-?r z+>7VF#slM)Q2WKzojfUaF#gc`Lfur4T{Ak(gt2V?_;^$UI*^TVRvPuvbucXhasp|| zmVafRqh<ZoZ%e&$TMAWoS_xwOh%Sz4NUX&dZs5dn0C{OSVvU6fxg4-LECJ96wY;+S z^puUA3XX|x<W$M*02FtqfJ*4ZN9--q7ER)LF-Wjs8@uSyXigH-k|e8H@?ba4Eo65k z8iw;r><pNEazg8E0f2X#6pI-&m!hMdS87{#ImW@Y=vQ-AW?Zr?)qFV1Qvs5U+`H;< zR;yWorAPnv#<j_~j6tVwz5V~({?_xjg}?lpTW|kj|A)WY*c_&vHon9?{Pwqhb?dEf z9=!S|D7euYj5g}gt#@9z^`{^0|N8p}zj^uA8~=u}s}KI^rCaa*@b)WT-v5VpZ@vBV z{TE*E(~xyBXKS(lpZ|RD+%Kg-zzs3ni1G@i?SydIf8pl`-~BaW@$%R1eDe?cFMoFv zt-sRV_LZbX_80e+s=5)t<vbgSecrxch-yYKwyH}88Jv4N8C=Z<apycKbt#cNDYaJy z{Dg-w+U{hVaVL7Llf{FRi?pp9w-u`wFeH?qtP4v~Q{0uGUy9_l`RY<6T|1swiXIeS zSC*oO#Mh^nqW@7>%SQ|Ye6B1-pHPvwcwIqr*DsvCaDDCE(`UI|cp;iTSy#W#TfYE- zPv~DyaKqvkpz%rl>x%Ub@M_YJo9bEw3AJge43QA+2WfB7Dd7BO@m5{D_}4r7m36&? zi?iw9;m8H3UfDN4bU&rvajY=;a%HnH1RmAPotV;nw0W|ZDT7M3Ws?AA37Wz1bd*uk zPVW7;aVL$7p#1BQu#v%5&?j9>$wHAC5x$oV*5cjdOgxB<yvh_g^?D&*K!sywK@8wU zK`r(E&Uv=XO16lmGR)vdFpxHsYD8-T3Q7t++v>!FD8_NH%8KNELiCk4P-c;N{ey65 zGtTSR-2H_yY~h(xW(~7L((V80-`{)g$9!Bt9F3F83;S>W{?;2W9en>c-XZ42`z#9? zorVxGQ^T%vQSusNl#IDr(o>}3=w2VCOVNT@$JfW=hkUnSzB7JcAlY6zK<S_UjvEnx zMME==wz-?76jBoon|l=A6vLzrvJ8`x<s6PUDw1uC=_#NOM@Cb_v4ut_>Dj+k;9neL z{TxG>a!f<3I6gkt9(m8!hF$ZSlgcS>V;ThcL{I>LUTej6B*Ov*p}?~b_vai1!DU%A z>$Kg21m;_d2RK|DO}R71P#L)hFIxh{jHnzefbvlI1*{}9<`EO<a3n;46+AD6E6zu4 zz_68EPDe@S3R=g&qhd5$0&#p)h+_+)72tW5^QSJACAu<Gvf`riU|3Ts?I%~`-fmo6 zA?^mWDH>Tkb29=ECI#?ft7M06j;YOLJMH0H6K4SHWRy4A1mmsW;Zu;V;ZH+BtIdXr zFh6YfbB2Q7>~gd%7iJ8&30zd;220Kmk7-o>tcO%kti>`oI`p^ThoMq>;Z`+jO!ss% zZ7OL>UXAZ^!n@gSay2t&8x<nfU{Dw1b{j{#29i+YLY=fGv)&>S^QwL{7<x^Xwq{mo zYE$@|MzC$%dF}Q6cfSK-GPAd&{aN^2&7y<vzIpIp-vJqy!}i%wxVEVJ?mz9n_|oma zf3q&2M+giKLIn%KXngp+-|s*71GHFu;q^Pe`@{Y(zYfLOL{$S-2C!IN5pi_u&9Co& z>zlV<{q4awe{tvaF99kXq&j1JSOzUye(#6R-T9X<T3`_xSI}Kz#I-@~S6+ZE^v-i% zy7Pm7fG4+q`O?9czOw)P@1Ot=32!~O-u&_Y%m0S(&Yf*8%tbghu=9TVHwWMNqB|)g zjEg$*=4y<O?bDJdpN7k)UiptAJ@fch(lgr*wUVSwTfb)er~;E?FbjuRx!2e$9+<tX zFjf`DW`(g`0dE4byp@nJ!C}@h$EYno7|=BXdpLT?pCEH3p4Tfe#M{Jr)SWG1ay@um zP&R|v=uv~ECGGPxoIHkhP5dkwOTA;1^v}l<uZgI5)EA&Y1NOZj#%2XIR=gW6R@AF_ zH#%8S@8n&QKU;sVqECmj3WCN8)qH-n%*Merc2-nyX!<%$_#)ke8>*%`tG8t!!o+A& z+sP7UXFmVPlGl;QL|vmzq?Bq3dBUZTdq)}SO4pm&2}I%c&8<C|9u`aJVewwjL#tx* zYmLj_<>jxeo3@fN#&w%hLS~!wF5T{iz~a1RogiRaXyG;6R5xtpG-uB|Qs4>z*1${P z!!~U`UU1q2%3Fv?=_VmhA=*L@58JIWVM}S*>{}*GXI>(QDdn4A!@p}QAyQVys_ClU zF7mjY4hv7+KWTZ3feJ8Z0WiE6r^Y)^S_4<st4DkFXve1^e+sd-S;ZY|lha+PN8yrF zGlC<nBXxEu)ggG8KJ7cU@Il>GGsw4eK#*qz=iOMp8!tG#eLAkLM%)%C>5$~!yx^p$ z9@iY=2=i)Rlf%4v;RsVCG$~}5@$>J#_oZLmdgmwmU-<=I;Jo^iTW`Fjj)ibm1)%CK zC)$7Q8+<n0fBq%Rebdng%Ht7dd5iYH^^Jp{{J~|v@2fHgF$w8$jNz7=8Z27#7Qi4| z8<17i>@<7a``Q;C#VLEkg^%2y{Xc!>y`R1fr|bQ%{C@wnFWr9o`};5aUfjOD_!6J| zRpAT60&|H)X>S+WY{S<rOam2nI-Hx1^Uq2>G8^m^cRLv6x#K)lmP{Q-O02X8`&6%m zw<>g7Fj?s(zg8%|&b3F883bQ`$Z14WUmJEE%@`4Az)w-us7x-49~1Nmji~JjLX%vE zI3s1iIl^xAXG%8mmO&FwMo$9Urls5DQitcQKIT&Fba14rl4;cCP_S|~gqrTPKi+=# zHNH(d_|EGGFTLiE3GSR`n{E51PmyxcV_Q$Ul@rwI>Rpc2^!=LL`Hb@9hQ)wrXS8lS z!;4W|)%B7+>msx!pp@;A!KFo2!YfL(1?xxj(OyY`9hi0O0&3M<9Q@(!{jdFG|JUET z^M~K;|MAP-#!*SdUl8}2m4(pZE=Jn{YjQCkS+DJiPgu~;%T14c2I;5gjw!$9I{pnL z;`XQo%V}yci!gQ~VWum1Wpc&2)P9u7HCj5s7{S~kQm%|;j);95udi#69*to#iP9gf zN)i@*JYnEhg3n<)tcMpTU)<2kej_?%8_^Ahe*@s#_0@KCqhveM3y8bL(RXYh>o+R) zCQli`%gf@pB^%FA|9$@}Uon}SDSz(2{_cBUeR==qFTwV6=ecj(dIP@v*TMI{dGOuW z@jiq+Q=$C+&wmLL@BRzV@4x!T{da%%-dA6O4Gar><EyZNG1&LM^QZl{{^dOqfC0|_ z%UHl@OD;P;jab53x9u;v2^V>mrKlJodyxpjQbQk9KH)F|iRIfv&cG=+@gpW5_u9tW z8j|@@u7B|*HBkG1_}{nRe)-P3|8elr&kkPr)_Xtt!NGH1#2^+XVSAtsUVQG>+rPjO z`}vPy77o7md%hhyc=2EN|MNSyIKB!^$NtxT1pt-1uuEUplTknJ@mbFc30XcE6IPNt zHsLizw5^6zkhiQkaj(Eg2&-N4;_kLRYBu^+y8FX8qCo@BEc2kEWYFDxcrs7t*c5wb z9IKeBG1F|hR6rFFc|$e#*D2&v%`(pfr!ve4&TU0>^p2h>C`BmcoCV?1xH_CHh{3K* z?kS10s`aUUN&!uu9GuBEH37cT$-Ar5qPiuzS8teFYIVxFxx8IGWjSAGV!t=FCaF`` z)S9^O=&&@xFjbzirHwWyV|+FCUZsWbG}6X(H2`%S;hn&m@Xfk-lkPRXO2<`?$T4;v z^_&x6Scpm6raB3bnO~O3n$O4fb9=FhsSBk-B7>veZjz(*Rn*Ull-#DchFw$<fTWib z{%OTnYNGhZRe7V>D4LTSFO9JAYHYu<FU0qDFz;~<C*~234Ku<RD5EBs2;t+0<>N=_ zv92oKQ<w%4wrVhD0>~JFj3tn<1Tsb-@Oai|XlZ+7t%P%J_e!;GPwqO@aQV^*g+-<@ zn>u(L@Z~uC($x*rQle7m!N-}bPI<Gub|dXq&o5GnJ+eUCfq~Ue%>hJhqd$K5G|0O# zi+9=^CJv$ws|LP+0|&iYj2?|<*xF?#T8d`)Hhkt<%?I5capa&!wM{N6p^-?Iv@Nc9 zR1ef2`}N1nIL-R}$gW-@UE|Km6pcF_Mebqzt%c1lLL+Bcvv1JYd>T+2%uU0!EFJ$v z%v9fM4Ka<pxqks0uyzr7fqclj?*D%Kii=O%Rr9yUE+*v>24oSwU_bBK(h1IA!DFg- z5U0y-H@y;rZeJtu5#KzZRVGS2{N>pw>Ik<6S%Abz5zkJ9vlG$m{BW-A2|*F?^o;d< zHZfOO%{x3y2ud(zJEc+8<wq0LiRncd>o_6(s*(KSEey8`$4zfeX^c-%p@+qmgFixw zFD9)M)q5cwC%NP!Q9xCy`sakm6X*xNA^}K{a|-0V9-)wAG0;JO-MzY+Ox$^xG~Iq4 z0koJe63K%FZKqv>Ju~KBt<!NE4hF@rGvK2eUPBa7C+ltdG7E2{uH~pd%u7+g<jk2@ zZ%RuCSGHa)s064mMQ_a=7?fC>^HJR2R@`X~J2*D@8WmqIf!1Vqk8xlwadRcR<e4t< z28^kgEP0zYV0>ya_q*SKeH)x9Erbk~tL$))TsTAYuppm`WIIoiDCs1-M$Ah5&}MMb zHNl}R;iPMTL+(wr(V|E>FlxHlImc9BEDMai0wanTtS*yF;TXlnsRu1I?33-<D)o7& zUMkZqMf+cUdH?l)*#F7%`_F%6|JUE&f9WUYwUQkC_P7^!VZ_N<2q&PRi_eh8M4s>E z<!I^YehQ6SSl<%TB&iFLpn_Y%tG*vmgI=a!rkk9J-1rFg8mA54N%sb8I~ka-ONWzu z`h*WGeQk=bi|V0>(|M6>xEdE$K70MsE9aj)YxIcx0Hq`CNJpO-EvknC$Xo!*)N@M< zGYtrV<IqDAVS|<=w)hs(tC^Q;K|NiIl!&ZJJ8|Ng&&anZ2Gep`aD3G4{>$GV{PPcQ z|Hpr@=9RB#(Wc{kJzA|tPlPMI@B<3xiat*qbP!EC%u8NastP8T2muNLcW}A&##^v- zTI8=l=Kr_e`1-APUcsy7Z+-K<AN}n1i*Fvh_O<<=fBE*ep7-I=M1@t2_!;ZhxYhY1 zNHEB0APrR~zXA8nO&3V9mkwGxzT#hgS}_O{<V>>UzZ9dwGC;&*^4?cpI{5DE`!Bq@ z|Mpv^#yl#9n|QNRkI4P-+Xt`y7GChXJHPt_N9PX1FL4fWy-d7uF^Ml!Xgp0J-+aje z;TbWRAI&660ch?~4n=W^awv*@ltKwfy!cTQe7<CYbtp%2;i*fKt((M2KKhH~c)#d! za``dny<$$LDKxP!s}+4QLj;D=G`k5%O#RgvlJX|>ScS`c1V-?QiV@UNyzXuk|2vJ> zN;!L^XbQMZaBK&>J$&c6_rCqy!Ee5O=Z|07|M@S&HE9zaO@SK~98Xy=pB7?1Ekr96 z$}|;>*ea?xP(U!wO4(2zeDCWZeTU1Xc%pQ(^q2_PM&+C_tz1x1U?&JH2AXWs#QPK4 zY>Tb2x{fVc@RoSzPv1ZI;lJ&__R6g{U#wVb3z2D@9tg+Oa<3K4&sbJxOZ7l>Cd*je z{CMJo{MfO(LLX(5Ulq}-yt*IiYa)lJ;1MO3{gp#kW#C^qXcKwKl@eYeguQ7co2eKZ zu7|y}O((o*%I+j0826;YLJuXShe@)9bTY=gPJ|CR=bjMlO(R}R9M-4HhV?Eayb?X4 z{eXj5m<km9bVaNq5fq#Mc)jb207oK>UDF=}SKoCpqq(sU{AnPtyg(v!xDB3GC;7|c zvEDpUyTK?$Q+3r+T{kpmzLrhSsgfE}wMOO&8~d~I9}-v88s<4_`-~xRitVMHPSnc= z6u^WshdmST#ziYn`vc25gH)=hHfkt9hq9;_M@qvXp~_5N=JqK)=KBYCPZ8iL0o)RS z3mx<KP~RL4-Le#jLpoBscWlzA@{mT6Rf>eaXc4Nk2vsVCd#pb&f8MqBz}V8J+|R0t znHLgt{y6epQFa&}O~iCk0dZ%}F$OSDQ5Ss)iex+5Oa^;N(hFBZLO0_daO4A6piY<r z4sZaA6P~Ar89ThH7F#P~19*cF^8L34!Hoc5er_IdO3KdkzJqx=Vr_Z;oa*mU4x6yp z=Ay;wvnG{k-t(h*(5bq*haY-h%8t$Y@H^efce-QmbZ6d4r3O0*Dc~m-Y+s$WWWSim z+pySHF;y?{9xO)-t9*>!`tWthj_vE#C@|F6XPs(ItTl?}r>Odr+u|6jVY5}o@U%GF z>7;!+f~%m)ImI^~&vJO=lNHqsH3HAb2{GU$bvT)_XmTPe9OQAYz?)&=JYK~Q$wxUG zU}2cZNhN8-YCPn1F@BTkpcDH7ImG96?yc#)j?8_ln6BL>hC&l<tM3!r7rmRCqI-Ja z=5XZZ%7?CGBPdH1=Z=Tr)PKkkDb-s1*{FD^!$;EY=$exhJ3%!5ER`i;;aClhD<y{t zsxSmc+fR?In<)3RxB7Yn4Ms4xBF7Nbfj&8WFZ_GTj4gWmtv?;S_B#w4`mKLwo%Vgf ze9}(d663|V?8T@gWwn#N#CU-+>t|45R0gpO;C)u8Q^UK$N<x4J#&OzPBTsc<v1R5h zQO%18UuI@M^{zCl9|xa3rN$rs;NEr}GySUSr;Hb|p@^Xeo95I5lX+s8wr&9UXH=YI zL`64TV{>XCc4$OeS2yXWW9H7**uY6;!H3!D!7Iqd+zKN73_@gEa0QWmS`eogBHbr| zghp8-n^)5a)d+7k{;5%mBg+MsMh#ya=Pm&j$I0^CFJ7JfUCAwUCIbKuFJtF)%ywf^ zIoLelcbqJ=k?mdIG^Y}!J1lgZ{smO1?Dd!|-9$mrE|gU*VHw$^ow($U7yazw2QR;T z=Z|lGL>BL;Bon=pe#oqGvaCkgiw~4RES4rVl3C|IK!$O5+(7+Yw11vF9IsG67ww-X z564N=&qe!Z5Swy7qkb;hKTjT#3(0XOM@cj&2qcTg&je~P3R2J_3|%X5u-ALrVl4J5 zf9o6n?N<2*x@J$7uGv4=wR+UWdDPW-)a7@y^`>-Ohg)-Zv(_l<z_~@8S=&g5t*EM( z<=wcGKAT*L+iBdB2~R}dIOEF4ml@6}J;5@Ai%mi*i;G$OriEuLY$YrVLkYW8P5{bc zHZ^N#m94VYxa37%@;sbk%qS2_GG;_PEYCBhVca8@ae6A2@t`2L!vz);B<OGzk(ca! z;Vk@$++@|!oIGRId0Q?jmHwkPBKh=0ggys`zXrQ?@+y8L%7v3AbAyNB_mpTNm<buw zI9=KUGGW?O8PcRjdp-=bH%$(x$yuJ<<!xqslNAh|q+1-CnQ)}2Y|XIBnXWw-J!3qA zIlQVec0J&w0yDEv@pKO29yznBG>cR6w#6WClL@M<)Mu0ni~#DKUC!1M>+EGfKqb}X zYZHv*qiVtLrDVtuOk(Z{19EFtyYj5L@v1#HxJVZ40YD0;d=}A8mZ#4m+vyk)A+R;{ zbXi`e^Xf!!9FzdB1fUjS!<zlMrojNn$Bk|e`3Njx8(2y2Rxs6y9F-`^^DM_Ssl5m` z-y|;*9yPuyPYd#=0-o7}=i*_u6%F<>wxWRV{j4+ImNqfuw#SNBimV>sd~PQe8V_MN z;7^D)N7wNz4IHc{LO9G~s<H*ML+7?+%=xUx-eG`QU0H!}CZGES-@m_Mktc#j)dS!m zvuFb{@bnRShPW=#OsKti^}-plh_G~Pr}Vzv4D8$Z?(F5YbLTHzq<tN^p)GQx#Y&DA zH)la6>eS+Tw5i6-RZzq*4Dj}04N)B==A&D0{Py5`zn9@`=n^T*2B?o%p96a5|7D6Q z&a)Qlm)C5hioLimGI4898qH}%_4Ng*Yt;{ow@vpgnILhdz2HJ+v+hZ!Z)@cqLgv=u zwwAQjfjBH~Ye`!R+3N6HA^8RtaPNwk4LWM#kClO+kG#&4X7XQIu}qF|+H!X!G7gM< zAEDk^MFMk8=iPYWy8EPQJ_(usFq-8d!jL~v-&m7x7kIyUlZnWX6Hy3WwV-)bb5^mW zriY<oNWznMvfD4IH?QMC3C1H#r{-t6C>TT#a^^yQ`1moVmavrz*`+qIEJHUEoIgUX z+K0^r4hulF@JPiio5e!{Z3OP8fbfhVF2UDGWnVfpRd@j{Ptiy-*vQT4X9r1YF^B}& zragb~!naVl``YjJU;56S*T1y?)pvn+hr2bfCZM!qdnk|L5Q^=UC>{1!{5#$Ft3<A5 zi_J+bHo3*?M{lwc{7K@#hllR`ENjS^jh}x6XJhiv9AV(YVG_iKvZptc;4ZCQhe~(q zd$&VRy)PeeH&%+M(2o56?8XDLS>_hQuF<fU*MnmaVTfBDlER$L&=xPBkFT4pIUV%& ztix{XVt%hAXt4=0c!Gy9+KwB3L38t!l7QZWttXXRv4k~pyJl|55?t9%p?BsI3Uxmv z`!jw5OBb6^J$g6_2Qa}8ujBWXNuUyDeAn#LVGd4-5EfGrhEEg=9#JezK@?LGgfBWA zni!mgUOikq_!Tqj=Cqa>H&7xreB@jnto^bMslGJTl~~=@B$8$VRf9PA$XDm`DH$rC z3$+F^ryMDBS_4Vd)pSO4I^jA$HO_{n)?;hSv@8rO8x7>CLO|yjMqJhSxH!KbLR)ER z`b=Ul^mzo-IZ3e&#LbCgsdE%rBI8$MhL%|>%%K`Hd|DFC7n_=XwFbyq)21G-zvy9Y z@;SU7wt31KR(JajZNiKG{;N2iajLec=Q@+jBau%a6{_FsE`vM-(P|<Isf^_(HdhCR ziw^h;2M$&~pra_5<;PVtH>(xR%?fbAPU>NJMCk<4&PStXs0y2nhe`zm9{M?ZEtd=v zerV74O9xE_@sF{u+tR+!nv&5X`mR}5>1n~YXpXG!lCl@e#*1U)#jpv)uQ^ognsD42 z`IZ6(Zl^t9=;-Pw;+fbBhiWe`v%0r&0jDetDikmP7ut1dFEbCGgn3L3Sa6<c&aBG{ z5qUu!umavZH$7z7OYY9k{{7b5Z{K=X#5$KaMSNTARYL6cWMnkIg9eEh7js4do3QMW zEVIKtv`61P@#b?q`R=6`Uqu|Ci7|k}{(HXfo;AExUVG9a8L2m5ll*>0`#F51pNUa^ zCPw%vx128b`wkg%8^acI9c15=id8HNdFIBIF?U@1q)MlS@|7q*fgRzHhemX6yma*} z&SPMCh~;U_>pQR9dF`jSU;EzvSKo>*#uqD2q9G0!9u(eGV2=*XU7$Y`11e06qcDLW zLXL|^B#1@X_ws3eAHx=ThAW{$A3hP9^fG0zrw?T~E5`7nM7|++(<74a$-5)p|IUPZ zh?rXM^WgoNkY&ho_-MhlrV6w*RgkTz0&GncT+8w^o={G^8CMjXCU{n9;H+taW`*jE zlvPVBE2eQ;WOD(f3`~2|_$!kA%1a>sPwkV*=ZmU)-ttqGX##flTK1!Rg|g&d=3mVx zfY93jf`G)cdBo#J{=f%6`o+PoetPirA0Pbu*ZV(wz8>BA(+}SJ>Njq^`Q!cHygAzc z;wykK;D(2$cmDO@+kXsC<aLVFjjm3pW<0LE!b|4<h<nO|)5I7XD<5IoRXl8<8~PYq z%Jn~&J!Q%OY<E96RU>kU?ZW013Xf`*F-|>pVb?6>8xuB|kYn+55!j}SzBX;-wW*`7 zDUPdys8z98BfQ$ZTaAu`?1FzrEvF(7E<9k(YH!=a18WM!*j)o#Fn~?n%@3VDU_OL% zpS^Gvm5)*vKkW|a%tcj{AiuIRT!+{`CoI#D=lqe0ccScFHs2rB?LeuS<FI~NevYI? zHX1p+`q-$<6e$W9Q8yk(*=~||;(lqBD$TP|3_p{H+15nMPuVVYm)b)62F23v<88q) z+oT((z|DO&-CUVxHRrm9$F1zrh7v5Fp-UU<en!+buXc1vXqB9OG@%4xb3cuGWgeE5 zjTx12R-HC+gWFC99Fh@g4wAgbhWGfb(a*C%HW>GlhFj>AtQ~I_hT(vE+?S^Nk7fB8 z4By6?ER;om4Us`Kq`%$Y#+@_<dbSf&Z0B)*Cv8zqY%Z3J2Jou1;F+}Ocj9rpDgEQn zMFHAs4KmB&?(>&6(dy)8l=TOgg5L69qanl>YKRT64G{V}lsctOl?}%aGt&l++O(<c z(W5SIV<K4jM5ZsnkdPudH@aZTX+scta^U6XB@RY+m#7~q^e2{Fie?bnjLwO3y`AQR zF~`>=`jBr5qP|t8sjEAAQtV`%_EPkanb7Vk;1{KXq#L!9tr)W*4YFu6Ve)Q7r-y?m z?E#PKM@5pSNrAtT)=r!cin;;<&$1jti9+493%%M+qJENZp>05~1r(bOB4~Wr!E8*J z+a3!FTm^_CBRh2{1)Whzi}teK3<nPswou{)y9^n9S_sBRT}zT^!<oP$Zz)pSNjiOe zG~B$Lj*`x${vhq9&n6oUmk8Wsv|@7Iic5kkNL)7aVc|wSt<%-q*^YM;TMWh`x))Fk z{WxV85LBco3Rs@INS!J>vE;SO48NhCSa#;3>|+=k)+mX;y>eS-A^dU5?T6JM{|vYF zsXT6{IM2(fb=(t<)etNkjXGJrHpu!#09EdU4Zzq|j@7u`Y9QF6W|;vL(r&z+l(ueF z4K=Q+YXCebo3NcRF)$YrBw;kfRFDj=Vtg@NNE9*mlzVN=5oehN1Wi{kij#j6{baw5 zS-`gC@NFiocnD(Crif=u)krddrQX|4+BO7{1Ie7ipjbjsmiZ6*jv%s59-^ST1NUmn zrVt)Es4k!J0Ai^+H8J7VUm$H0&ijCN;ZkM^;)SpZnSi?SRm2!aKB;CdBffxSXcNYy z=>b@m&t5!p?&7D=GZ1HbCkCz^Z*5_(i4_lYvTgCf@LV|FYbL9*=E;}*@S}Vs>^0}# z#yjv`k%c#Io^i1p0<BFq(G(+)irBrer*Cq9ELJc@E8#v#V(@!q^ByCeE0a}8I#>Bv zW!ke#t5mUeEol9`Zs6E4!af<C*KOFu$XtZ>q5>&;Y=a}^svKCP|4z0Uch)E{quI+o zsk4&7nhq67k<s}ZUUCa=7~1=1U94z3BvBd!P3w&n2OVUVt*q0@_K@8z(h<9`N;exM zeW3>(u~+NmqIfCt+LuRAr?I5jg?1Vi4EUVmtn}1+U6)ULBsvKzW)ZI)B_~*-ZNyv& zpyj*4W$*Bdc6_P{+^5XxhTn>q4Q-A`&@@+@XctBzwznyXzO5yL0S>GRA`j?qp_jLw z0iQdUAs1xQl4xWhRXfC)-NH=&HG4S(n6UkThF3+iud%@oljcs`PsEupI3BAYJ_-qr zW+&R}#BQQdIe}QJJgJ~Dy|A>*kP`;pfv=@*2Y4!C%&2Rrn*fyqPRj{*2T5VvFarJ8 zs@RyZ5BLe<$xK)ZmY4c*3ti2R@zE(bCdg;9JO%XP4)d3lPJf5|IRWo&{UksYP6<n> zm2Dpv&ocg9Mq60L?R9xG0feh%molAKW!qKehUQYMT4~M>P`b!uICX2!@phY1bTsQJ zrwHC6KP;NCTc46DxYCD37v-(r3(C*}WPqoTE0cs&O;PLDN;w69BE=I4;`VN?$pW93 zQUgFhoeU;=9^mdXkf$m3m*?G^a}xj<$KQ%ro2cm>!@PnB1y2T70gyi0`R9g!(0bEU z$b1G7+k_e&+&_7bDY7oAMqrO2Nxhx4(l!a8cxDp@s-(8@%(d}+pe#&P8|PCvsX|fH zQKu|ROPw2D-P*20mD7kAu2PM2D^EZQR1!&PVp&TQW+@W=BsbG0tCc5QCA~Yx-Kq2{ zL<KC%USU|r95Cd~YL^Ngwp|U4LnoyDF5TTxN}!=+Fwoa_(v5rVfP1K?2`U>=htNz! zTw$VI$SU9b)wE6}6jdN3=8&4JI%(f@$mzOqRQGfvp291{9xf6K81hMmpzWoD9jV09 zvB{8Mc$MLZaF(Tgp(iQCnJ3Gw0Bv?HI6JJ-T1LS?q+XF!{TaYSsqb`ujmId`2Vd}z zU@YxkV)LBP{3D!6^ev}JH_~Wj{c$a%z7QSdi>?d9Vi3h!aMl&Wtaxd7=iakUtDh;k z*Vo%mJ-Ps_j1C%XUjW03oq>}QF?o6e&y4s|Fg=)#JcYRm<A8P6#Oc-C|1W0z^ijp! z&&!g)n94k`<t`Mr+Y}8U23(+hhq_Sd0$WrJ|MxOdYx$6fgE;Eh3ZpJ@tAk0^_T-`r zUJ4)qJUHbAHS#H$I!upxupADKSP`ulHTiIc4P6>axq}CnQeia9FFGA6<VnXn!f0Mk zT*=+1Q&$+89IZ0N8MNvdfv_jk9Zy@=_cRZPTXcEoGB20IszMr#GQrCz>}}2Bj?&zc zt*cD@&=Sg^SC_LwvMwcht!7!?zM7rK_g?NLcSEn*3?^>wXpkq-(DAqMGdABienu3J zfS9MaAP)PQ$o>9|k)EFRFDH4O1UY#(QX{7WVot#b@0MM;@ZPs&d+)fQY<p@>h#zxQ z6<0K`*2}r$Av3yomNpf3Hi27ylPP?-);0mDaheggjTv0~im8-Up@6Ev8(u$nuB-f% zceV7~<XU0Lm*X5L1zX9v(p14YLu}vI5XE=n{03L4lO~sbw-DWucD?ojnm_eDbQqO# zv9LHWaB~G{ApkC0Kk*E!d*)SnHm3+X@&}xew{WG@JXQRY1&5W4p|@$jFAubKuLT?| zaW&3?N0s$nmuL`{p_4(FN><1)1%0FAUfRWEO*3>jnVEG7Nz+j$%lZxpiOM9ro0Nn_ zLm@N>Z>PmN0S=^f1zG`axR<AJ4v7bY{O>+38l~@Mb2k#8qusbOw6$eY;I>B>3X^BC z>~L^w@$Lm**BEM$qOMoa{1kL2zlJncrfH70SK^MFwmL?~r=5A*R_F`B-+>5kA|$WK z@tX|WnRGceQ_KJfBy_S{Hcr5&tSP#zQuQ(}q|ia8K-aHS!8yg^qo7(eEx1<?G!daN zFIMpYdU2OjK3KBKL=@g5d%v{TyPRkJ1eY0Bt=?=1$8{`+k!b(~f4^57PnnjJn}z1E zRho`!Ed%T6h;>NYwLnQ@@rLwLN`k6XN_9Q4*CvBcB?A?Nmqji6ZtzS3=jCqNi#b~7 zMt39X0$ajU>|WFzwsxpCNJ%jsZIfOH`C>cT9OH8fl0ty46C2T57NvtCDn2(v84ClV zppb_m-cWICmi@M3VREN|Avw``tKp4Tm*RD77PT%XxNgo$iBP{dX}_>%=TRdhl7e$Q z61BSPi~Os3ZPu|i*K+JlnbJ#|Bp$r@ulxTg&2wq#C9JQhM+d+8Hd^t1|Hu39{_OvK z=NtP!eE#;YU;IDM{XNw?OD2ji5`Cl2eIesR;8*>^dLwxZy=jpB7`2Swtb&gT@{vU` zu5xq^{z)mIPGwo8a5Hud{j8e|cCvO5-*v7}a@;l1PndRT{Kw}h5Z$)Y?aKPN^g{KC z{ptGi!qSm#KJmg*%U@ijc;Er;mxfDcZMbQbv#dDrLjX%nIPpWmlm^ee4Qc<xJX^gF z?T;k&avvT8-t7Yqs5#xM47V+t)str<zgiPph}DR*lNdc*3vLj4|FJeBRP=oQro$%L z`!X%os`f7Yc)Lj@3KGxoE^th>7~F5dm4K6&O-@;}bnj9qsJ+XCUb=}@MhMkE%=x3W zt+7|nLUc@k6g{UsI1mHJuEDyf6YX1Z>RHEn>y4N0eCgdttg?Pqq_}o4lHY)>NPxHP zzw@2j-~QG9Z{ED~pFid5Wz%UbNzo2dFG=vqXqmQL*e{?-<~W5-pak|937@z|W+)u8 zeijx;p_^I=bms*z^QOg(y(Ts!0CY<32Wk+>p4t>Rc`#+j`o+i(Sud>NF-Q(Yh+n@} zkNAHXA)XR_!nURJ<=`e}yj;PMVUZa>Ou<##voz>Xe2;HZx)yJ2v$Dz6Gl@V>#;+b_ z2{Im}#`kB8?>CI^c0j=k5yy2}Y+NwPtg5H0#?$la>3QSn74`Ir@pMH!T``_Mp`JdW zups)RKvwnWnP_r6tMXV@Slf&b#KRMrk+bkKF;Y+{Mv#IP(x#PqbA`g@91T(^-im@0 z$|zk5DHQrbK?*6SOCg1#6_CFs0P3zkSmw?-rq>UZy)LM_T5MWEq>zs&8cGSV48Yh0 zAcV?A25?deoMixHJU+FkYjf1>Jb<n<BJee6?EyTLGE6xi5v?LRz6|vO)f6@3B}J$J z!VPE$O}HWZ6bo)im;u)7<|E?pd7u%s|Ev_n5coRgb%*1~LWDZpsx~M0R^lB-41*8y zWK8H;C}5%hLIVEXf#A+VuLFV)Rt>EgmlerR=~@aoW<%TT{pP>wbI{l-&*?r#kgX~b z_xqi(HhSsTqdx9MWmEPHj@2@?J+VkI=`?nT9BoTFbYhAgO(^q0b;*(eX$!l-d)Ez& z5YGE@bfYBo2gZbDQDgmv+wO#!4a}-tV7dW76C^%P#;qIns|n*$JtpFhIG+R4%DQOX zf))^YK~G4EX`zTkKk+luovBAN7x14B{&ODx*~Nc8jsNW6Kab--&*DE%qgR})*}^BQ z_|NC?pC|C20seCp|0(dFHT);We^zFw;x#3f$xWg}pj%6Okq*OPyjabYFJ&t$PRWEv zk;-<YTG+i1Z(H)-ZZMY-sY7;oqub?;ZkIQ@U9r*K6s_!DK~diVIjlz>)Y{MjHZ*`? zE-XZ=HpEC0^lf+6YC=Z+owf<ehoYK*hmS^c7z(%PoYIa9(HNs)a6cHZY~1kdgtI{_ zDGChs;pS4<is64aNaZ6A^8t+cOtM`^xv(Btc2~623KT853}()vsI$$#S!gWCB|#EP zUu^d<vo$C@tCMUk;(#qS0zCMHBobAFQ;f+P2K+I?pTm6IZk6d8G12lG(am(??q-`( zg^w`d_NY&josTHi2qp`wjq&+dJRege5jz{A0O5uaLKs@85~WP0g1|}(SdpDEetB(+ z)4^;NAzC@E;lM|se_W#+IR4K?c2U{WvZ4YK&4C!KSsIogl+eV4Vb~>=$GUNZBVya) z_%8??0|=R!D_dh(V2)ZCQHhZQgk~9|MIVNEpzY9bI7A5uSm<C=l#_QHZpvE-UNb%- zWBV8?!-6sIsjWw_;1?o1f78%nv?U^1T=Y08a|nSRjH7GjN*?0QGDRJu$}U%-<>}R< zrxle|oHBtqF*IvMnVb%TX3yCxO97x=1E1V1sr!f<IapN<MDgDeDOpGz#pNDxlMkv8 zYy4#{T7<ra&=A0y1~504xeM%s2aLavQX_7+(N>^FEL(+a8d+hRWVj|E807aV5(4j0 zLaw-y;O;tKTkm#ucb;v1ZcxP6W^3#J%lLVGc0S!cMH{~|XhVs6i5}HpL;y%Yx4+v- zC+(&KdQMj)1RTu=i++rL?R#XbBJ!PJv}G#5;*2a+Sg!9$53g7-%Lf&0a9gl?NSP+v z>7+LV8Z(Q@+=C?lG#+Yvawe*!oGo)<76qfZ<^udjujdvU51vfsn)Q<pHa_{uzd0#n zA$ZYv=xnq85CLf}Hs=;k&OP|xT(b$VdYSIxEDs+i3KpX`_h@Z>VeXUHj{ncb?4$JU zL~T7edu>kqn0<6!3?Qu*?7>>UlMY1n_5WGaubr5;_ix8s9M$P-bYnu1(-q~AeUJ47 zKSIDLF&{V7qK2}J)I~YEIkl47{M<YU1T)c`y!sd7qAmcBs6=@IdS5SnU$4soryPWO zr_|4uu=Gy*NFNycARa_}JpTm_A>HpG>Z%GIC3<|6=)o#$wWJa(>KUgAflrI_Cl(4V z95R@ev!i~k)mrZX|D4&LK^pqP3<eCtN3$9|b0{l@u|8*aQuHd$^LT9a9G);b)VCTm zYBWZ=apA9N1|z4{+#f}D!24Ez>^u>!4xhyq`Xm3vQ}n{zN73DWmGq^?vno4!p&t%v zhh^Ut-)VH>VsNh4PE6G0`KAjcieYD<AMfp?AXC?B>J_4^fhD7)#Y<z>b8``MEdawZ z!m1LCK7rZA(7KHs({<6sYt~r6i}PtA1F$t<hi>iEG){6<)crw+sS-0|i>it=JAqUj zP)1l7iPGyG^RVc42g^3_n7A^i8A*DLqSiCg4<SSO7%!x>fP+DSOgsj@IvBuUZ_@gL zr6vjrvPPE8Yvf}%>h_JiJe}(i6vS+?JyY<NmCh%<$j*BsSF@9-p|3{GO<*Es5c&BG zY2z|9lZ2H6ji+riC5e_Fu|tX*lMaRG(jYIA$2wU&sP&y}U!v@>v;Yv;p;_F|p%~)M z2YP&H*YJGP`^I!J$*9eQgxz&_lEuiVZaZodcXXh1IJH`za-mgq(8D#OWJ%_nh-H{j zQY36EsGn&KI+q8o=2E@z6_#Yr94eo?I#+Qd4Lsd)&57w@5p0VBB5E#Zz8W_bAM`Xn zz<tpnBI?wm$LrBfJ^FN=$ZQwg%Ff%xjY|?(KEb<H(p%1SW&)Z^dJRnd1nP)ju*14K zE;XK}6y5m%FIG2C?k6Bu_EE6xmj-wXsbY1v&R2D7s$P8rJJtPQ=jy^e>sD~{FDYaT z0s-uJ%K^eY(imR&APM86J%Z<tHiA2MNrfNyvU&XOru+lPcefnhk0!JIbQQuML3~$B z@l~j-PJ>FghyYYq{XYD+FyOMz8f1)0pecA48hDl!s~)=62QqZ!Z{-N#aTsa9`aS;O zD;|EPPE0vfC+>Q#)=KA!<m1Y%geB$5jfBpWd^W(Dg|9)^b6M1I!z=T^lOI$oiPOJK zT-R8wv~3qXIze565tEIrBZ6~L(3ZvND%BnPbx$6`LY}})TnHroPhg7UDazbCi6y{b zX+Xu_u@10(cZ|3A3q^i(DUz7eRCGz~X&UOhS7(wBiawtxqtB1tlGT&?#MFa!ZwBde zQw<Vtm(vUtZ=@dp6@7R^cJ;p#-+SWEhVOl-i_PUwcSk{kzfjPJL_r@W_qn$;v{FVx z`4sCUpGJ&iW3B2!Tk#ZS3cF0T)KWLK=AnY`2P0T18v||mR!*kwt`fA0Gfd(Rcgn1$ zg82W>RB-PX^Pd?VR0Wlknql<&dUl2qOc_s}_PxOLC_L>n<Fc}V{qnQqRKH0sZD>Z( zP1|h@%C<$n2!PWXtzxMNKjpq0>z7ZRD(R)#hzv(8+q7wrtqmySslLJ%<}4yq;7wr} zO^GXIN{OhQ4L1S#Zqm(i3~OBskO1IOH)vFY^|D@p0mX$!p=I+btrrl_QbQc5D@PR} zKw0`@x{l%l!wK2-Tk}Gy$N<b|T#FtxJmO_dp-5YQV0PDbxA|_e+le@*N0P%RVVok0 zq2NXSmYr{rFe76Ou5vL?7$I^UjLO3yR>V0LlOFgQ3SnU56d-dRhU&P=^hXgCO^a>k zZ_Q2D>Xzkus0ZG03Mh%1aMPt?Y+F&W@V^%UEo1h8&&Y?IqhFA@jR=7u%$$x#JA*-g zX?}ihZ?Ca;vXSN6^H;CTuYLMc^H{oQ><qe{k6jjW*L{X}x}6Isx<tki8Lx!<oXkK> z_BORUE5)%gt5vP8+3GI0I&o1{^jLL3Mh$?!N9~#RPoLB?!zKT4ox70a+rW#4ok7~~ zFe{a4sEoTveVVlN3lUPYO%Z{Q#^$=%JfiwQlgRaKrzl}^!hA`YoY<!jsAt=pgzQZ^ zA#)Q6|7=HtT2IZrTm^YO+X+W<7w;VQTNdV2GEb;c6|7YhvE#4`4C?Z&2*_Fy`$P_1 zqdzPx7(sh|e8B;M4a2#b>BcSstTOHQiJj>kk*k&tgi&P5jfO>oejmtMjRE$?F<d(I z1~T2|QMPeR+B7^SZb${(e|Hs%L~Cy1w5yEt`<r16$3htIc_BJ*aGxo5&|jFXl$S8P z4R>ln`Zex7nT}0PiiXA9q-Z!DGxsp@mPIj=OVy)zH%r@555wtTjFu?bO@Lj&TQh!- z6<G~q9?0T|IOTz<iJ&2JDO#;zjvtD$!qGed>#cModT6wwa}`8()JOeuCvInZr#r*k zd9;d2v7%+0PR#-wvH!D9HGYaVAR@k(J&pq=3mul-UUsX(#%z#@UC<aXBZfO!9?}Ci zE6vP!qjM&sa9Jee7}>N?)+f5`Wo^t`ClXqrzz_VT6H%FGvP}zCO6lydvBxPBT!R-1 z(HF4@BZ(ayAFu%~BaHejaO5Dsjs!{DwjQ07_RHBnFn%$HzNcXF144&m*o1=ET!lWi zhIx)gy-9~Dn1UdSVM^Wj(ohFGC)^T5ZW$T<sTH?&68C&G7y&(FK(<wU+dhrw2f+j7 zlg7nM7tcCEEOWB5BhwTq&h&f;e!+vM2c%9k6CTNl@%xV!Z~i02E{Ixm=)IuLYSp@= z_RZDzsI~Z_CGpc=iW0hr_~!=d^$$-*x`q<~EnpWRI2zDZ=}WA>(t+Kjvc#uFaPP?_ zFUM^}xv~sKe0!V%MVA)K)0U$_@dt}y3X`6Q>x|5lwN|0AE;P2i1my>(9Y$P)m({4Q zi8NyF6ACO_9&vcjA`hcVK>Z`SA;MROA{>Z5Oe&${c9)efqI3I)Y=?#e*#Lrt&Cr}? zwuqZvs36`2yih?>2;2YR^ZW1qjKw9Evy?zaa`ybHOs;`qlL#?HkD_Jk0QPxDlHsHr zT9zp|`lK;+tq~_YRlmpfZxE3dOb#bxqUr=6R(21Iot;iFQJqR+hFd1Z$Kujf(Us0# zJT5?(#*D8jsi4?J6Wm49q8KJojkAafHQnL}BvtaG16keXZ9$x<&#)zN<`a&P*=V7O z0e{U!ml~~c$vFqZe>fT$;R7M0_y&lmSql&HA{SZ1h&mK<&5$_BdIV(k!kOzA&R)2_ zcJAr3jg^4Mg^kk9qYnEK27_?TTdKS+Ro4|4>N0ErQ2C;La>NBAR>XYt7zn6)Nt(BT z$_P`+SOc3~fm8;P)48+FCI$-!>9pBwEHpm(KNc1m%}@Nz@#C}Xr^ud^$*7f(79zTT z*h@xzSf3!3;CTVXMh-GYDKACDw<vC*D*S!|(IsxNi9%nV?e#dEIM&6{lc^;sA&yGQ zDDqOoLIqBEQ2KZbtXbfW4RX}8NuE&_KoH1!Bsn1fj7AK?0VVJ2bWnEcwF!bDb;C=L zZyJ<`_+t+~^a&WkkDY8b@y|n__{1lK^Jmi7Ziv7Y#zZz@@SEiBBwyfR3SKOFFI0k` zn*9_JG#*&Tf#!mE(`Ynwku90<7aq(?uEq7PCS7#!S?AebpV81}fKaa0?dNpDK#$Xp zEi5d|xNpVrB2-bFG3eEOf29L*z!7K_Iz+h(sGxlS&0(XDJE!s3H>*#L=z%3@P$=Hm zzZSSGQsIns-mWFR-3OYDO`x3ujshtyefyaNhW^&gS#w)?v`k{e59tp8d7L7Z4}s%Z z)1y;<@yfggvJh-L=@v-$LGeIAuZ|0BD7;jSQa7}+PT_5WtZZw;yncxgld2|W857@w zAt%ksLMafei|<M=@>CpUI$Va!;HSndP;|lsbqy4<klA5@F{ecQ6DYtBVgIH_CE+UZ zLn(R`Sd=&jXzdt7U?~U+iQg>qgkn?Rr#cL?np6=i?-`g>y?E8EK2tHP@&ha99Jx+n z%2DLK{o%l51O;7X)m7qX{c0jyN@xmP8kxGJ;3$DpB&VS#)s5h(fz(&#M>o3u%r%VR zCZyPP`^CbwD!UE>6s{F5J57{YOXH`+&;viJo59HG=BMm}y`+Ij*<nn+$oE3x93AJ| zLyT!$_~#T{EE3^r-f?s?n|8p1X6#djUtO|#f65OH{nax0UFu0$j$ccpUR25P{Ahw4 zFE~g*j`t;bUzRtd`33KZ`#|$1_l;ywOO_yKH3;(RjlHyhB1oIAW<+XI-R+kmsT$Gx zsoZRR%EQL2Jq$85q!dO^*{{a>)!2EJFGNN<7a_PV(4iS52rsixpm3m=$?84s@067T z9JyP9Q%()D?u&flDN={Mlyt8$j95*#?xKoCpNe!SGN-65?+eM2!5_!5JApY_7>4*g z_NxiGQ&n?ng)8CN0u5*C5gtGkQgt<8bvRbZ5h!ibUg>H))V8a85h#hn;gvMpYJ5!^ zl)%u$ehwq2M+;^*%+6Uol9g||owMb2QRJqlUby|u-%P=tuCOLuGbem7fi<Bhe76b) z^XG!2aKLZVZ~&8%kGB3VpHp#_nYWz^lDu^s4xWQ~X3di?fDS};4+CA82I_(aTXK+= zCIs=IJM47ku3(msKBb5_c=;dqpL^%vYu~^9)}JbLPQyXknJ>Co25h=7F5mFJa0mxy zI5nq55d(45(9WU|h8Ekv(ubR{th)2!{VY|IpTA3ba6+#AE&ZK4_~-BK|JUo7iUoe* zD&2zJ{%^OtaoRx;cGV=$vpi34z|l&4b4t!zx0-1WVBY5#HKaW_p!c&5s#N+geElZ; z4<pz=38M)A;b5M$uK#y&OQ`i1(MERx*Zo4{A;St_w@I&0;_IgUS{?*3K2@!c!Swoz z7#nXF8OJUa9-Oop!-=RichV}(v9h4boUF3wR)LrB82wzDC0D!7#ck!}io4ns47zhu z#wiCfIe6`N@BQ@mxBvdlgYUe4`|U4Vg;kcLv^dXsj^Lf>zJPR}zV9{Lv~S?4vATd_ z%>qb)4?IA=7*xhKx8QpNWuT!Swo~Ayy;j0irE=UltTvT(1{Cs-s3rsq2rGUHRM%7T zd92GAt2#=EW!O+8+jvelVQEFScu9XG9&r8_UzX@a2yt=Brq^M+5{ey~Vx&O=PN`Ck zW1b%zU9$=C1q(R*Fl~<{KiukMSq}YnM0~05$JXYsy!W=h-~jFGd0fW^J}waO;<+rk z56tl6bXDb0iv08!>=rmc-J{f<CHgnus@h+)D?OseWl@h7R1_di1VEfD1#vPA!a9*m zFz_>bRTG^65DaxTIE-$;Y69_!(kH#ri{a8<5S%@1Xb(ti&zZNqlk&x)a~&%zgVvoR zqs64*B!_&O|19#Klk{f~{sV@+cXGB~382XUnhc=H0Gbv6UEf^70!dC@I`GbXcYgb$ z{h$8x{_Ee~fAhZ%-u{P!SO53a+v(}#*)+-jjJDH?zD~Etm_0tq9>c+(Zi?L_Vl;05 z`o-IC{SpMBx#r{z$!?WRfMT>R+JF7|JForp_G{nU|LR)@Kl;V(SH67k&0p@nfQa0F z>t7Cj^yB>>{^}MbecJ!VSN321?d@0pasT<RScB<*I3ZNDzP><e>1e&VK$`Z2Yw+XF zpMGFMyY<d1x8C^XB!J)fE!1I|4N%mmeDM{5+?&6D@73pTfBRS12f|x*Lnfkf#-Oa% zdibp^o2M|iWpe?!b=Qs2wYKqAzwmeIUDvXbWGef~qWc6bG~A>lJ-mSz&-Bx7N?W_} zARg%lBD<=KEPl8mHhiwPg;51T_LkpFM#%86+?*~r7b<CQ;Po=8rcxpinAI4&dL;jP zB8Y)rjZv&e@^7%PeP0L!dGL*|?!WemLu*WB)+l+n=m7;jJ!=n@hl^3T>B~>|V#rt; z;=CiO=iX_(OqP&@W>iK|LTuxDPj`1xMUCc$TE6*C*!&4YA;^5;3lV=c1h-(^w0^<+ zYpav>5;SsF(;o7~5picHWIZg)o){dbl87LIp-DsXRGBWe7;e^f>l#d5x9%DE;dMeg z+0GNXci>cnXvVh_lch1!j{%ypSp)E0$<%y#x4jU#M*#-|t8_C^TA}VK3y0`p(Z(ZU z+sxZTcTVswtWbjT^XOF6bp{aH_MY1#$=z*If9Kkk?ozy4q<3xaY|tWkG;X(R>+%Ft z$1|!|tQjb_=$Sf)c`$yEX|(XLPX92E@t7QNA)^ejmJlOs^~s5?Sq(>(@!%K~saIFu z8aCs3(_?2PjK-*gsq)gTL2cHnD-&V&i&I`21En5=Rc;#LDBYm#(jc>su_}pw8at$r zv6Qx&ei|F5sQ=f0ZF6C~$X@h(wFEG{`Q&e$u8ofWzNa3?-0ltXtaDjmX-FS|>sD6= zQGKpk_^&DovqAV*R)$heaHoZ9eC|-c{G?EZOGrRPAwQubd`%(DXGGpr_&8pAXReo) zHCn(&kB6f*afQWjg%I;q#8R?<h05c|ZPGlg4)a}K<*v$!^IS_D%Ua@K*OKQm9&Y&V z*FU<4oI#DtRWyj7zD<0rty1EyURFLb_kfmc4ON{2B8TLia`G@PJ=CUeHk*c;E!%E2 zeWHm(>5YSR-|+KS?)>r1YKz9pS!cYR_3kro8^p5-7Y=<cRdwgU{aj*uZfnP*`CMXE zeBg~{6I@yJ??@LBWn8&hUtfJegCiuhwbrl+b?T&XBa;6x297T2t)|#$Zb?{mSs>AJ z?EM>m>l((NfVJV~Tt6Kp9deB%e?nZ5t+=-v7s81Et-ML*LpKR>$lUA<S-nAqf~r&$ zqt#UpEyu`+e10d((`SJN#~n)7-LNJyN27PCa-|<GW|4y#O}j=9i&;l~<#aV1^oJrD zFuHiKO-76Jxk@7%X<FNk;NgZ*Wup<)PFO|AmT@EP=?5bH1d7`QCTeAm-DyAx-(EA7 zc%Jl=c+eSZD<L%b$+m*fu*r>4bE6LbF5<sq`aWLV&;~<Ak#6^rHoNtwg|G*01V?XO zck9}HA+?lIpfPGrE;F8@%xG~6I8GUbFsI}W*JtB=q#QEKzZRX_iZ*buZOm@~$t7Fx zrVZa0V9eU=ASLNdlB-&Ka2C6a6e+z_(7gecn=zYc3a-D~$+KCDuU-eTbA&EbD3XRX z;D+!zT6A768ZXCA&9PZ??9?1<b+yr*2DPr5wB8^9)t>o=+8u}LP@_gmBuC1bktS#) zJ=>FJh~vl9!zsp(3=@#CU{a^!D6gX|B%UnN6AOyhbFafL1;FfMxDbVux8K=lj6F2Q z6VMpDXpBqH7+0cka{^~2_m97D)*xpkW{Uri!6|-WOVba8y-vXuCl?{aCl}*xeZ+Vy zy2QQTJQl4T)81ztUV6$c@RY+cy#kVtxkK+2WS=;@B#(ZW3nLF;979}Hy?rY1c2D4B zU>GR1BM~z3`@s0L+t&bl6WuT6aqfVWf_Q2w5OUO%$)h33mm(LPFh|*Hho$gn&hDyl zm1_}@xWZq+6pmaZgrl0X!yx&TLQ?o;PQ-HTVJWFMjOL`Rrec?o45gnHDKUf9EX&(z zFCO4RAGUVT%Nh;Cg0(b-9+LgISE<VqgWcp(elgjO@mq8z*@`j66b({T>xt*H0)#XH zlGw<RZ>!!+WOa0F!DX#!@sY9lUDZ-kt=l3AIvf_oFi$vrnratqEYvX0qo{)3@CmAC zdV^#;$)lvbozxW&N~T1%xpX#@8{*dR*=NVN8}dXqG#lYyHSNo?Jt+q%1W+0hxJsik zx6TPsNB0}mbc;-odH^2v8iNk$*GnP5wZOzbN|#@AP0D1t=u-IYAOGug#_+h2sq-(6 zKRLLIyl+x0X^1;$hcYf1V%IW<5)&4y?6zBJv5I~(6-p35W&aQVa_fz+{h#N)xc~Bh z8DGARU+~q}@BH!~@oMy?ukHWw%eVgcql5qW>8&^3%A4>$U&Mb-;y({sF!S5*{OkVn zKY(&gB~RUc^^N^^|LN9S-;CBlS{46YE2)O5kB?NX9xYz;t1VKs#mZ{+=;SrO+DW^b z=R)9E{5l4|w3FD6m~r0=!(ZqA$u;raa~#p}nImT42bl<a%Q<A{sas2gvkG$6dO+ol z=c-~uJN_ken{>*Dws*6<8){PS<yCduk<!yNp`sejs$nuK<m##^cWa5|T~T}bJ9KRm z9vr^vftq4S!kzb`R|?lWKi3(m){!#_JMTT->4Xd4^xd>VL&;@Qa)%jI=Htf|N*xvH z?N@(WX6t5i`17W!`sf^<Og;=Ae}1yEdy=A?5<AAZl_Qj>r0MW3=Qq?li?Mv1T8mDt zMVBS^lf0Fna=huXxG6N_^2NQ;i?a^RWt8rNk!ycDUc5&vQs@XP=X)9^*{tRa4>TKA zFL^d6))~8Xf?6)89XBpXP$lKGvv>&B)zrKz+$CF`X`^oIL2VSZrqe*ho_9gH+FpQv zd&Hp11#MKSqc@q3FcfRpcc+rz9}bv1vE~&ui8b<^;G(mXwH_X7jJ*LEhboQ3l^o6~ zE?7q(4~><8u2&@UO8gv;K9vYQ6T!(s09G$ma87mBa|>9$6pe?u+#?blWUly^8d$Ny zP4&HXFLzX4R?Y@0+F1{5!VP0|XO*iGk8V9wejF;#1FMTmBLaViQ3l5o&8kmpg9EkG zkXmVIxP<5fX4XwUO}?}%gPLS9Xt;Vc?rkTYy{fT*@W(0h-PnAmmJnGpQN3jm1PkL+ zQ{iPN;d1$q*AzZ*)M^bk)7F_JPj~6?wK9OC7d=}U%+}=O)HRmo_r<!JDd?wbD1X>y z#s+BXyJ}`k%>*YH|G2zuir#;27`Jl_58$Ato=+Nu$vr<r6X6l4==DuWjs}@dkX+b; zX0C-rt0b67l1iCOB7fr`xlaPp4pLFzAwhGDExC8aPPQj`Se<lRaWCz{IGRBkKXT$w zc!O1!hJg)M-8Kd`SaqKm*kA=`X_}nHg^M#`au@%7m@^smgWO5JFW=K;sSjXB7i`ky zl0K7k@d=3l_)!aXZMfM^chh!KVt_W{Tn6wjatB4Lux0q=gSOIrc^4Xv;z5&xZ-cgY zQ=`c#7x3Tl5z!o1qB$-_Gib=B@GiIR<3S~yL!*Knv7aB}U^@*3(1tbr?DG!!{?LF} zqNP-z-UgHXOw|!hAG`;@KU5-kNohyMr5(BH)6gXI&`JX0>7u?KnY!+UfPODT^Fc@q zw4_rlqLYYfTdkG$Vj#qt_oW{KcvVlN&2Uv$+?eq)MS=`fTk8WIS4Okgf9IFMBY~}b zTFG11kpO;AkiFu(bq`|fe1H*kw6s(qFWoDN$&r1ufN_`@JJDaF=1dNMV^%qI@Ed2< zV6gOVAFl_=h<lJlX}6zeI+QAUCdnmY6`c@TRurvXFqG9JOmf$U{kBNfa0bIwR|d*f z*(#!CE5`NZ*t^f^n~@QPeCtF;3^`)-Cq^rTp}Y|1H}J}x<7<OR68FY3ofC!i#OHP` z6Lb9?<lVlAa9Gk6WA}7BY(hE}F6Imqw&zwc7JmW*lALz)9N77N2%0lxHFq(TjXk!% ze$8Fv<c=o_ZzqFKb+XO4vqp|*PJG)Zsa=uw1`FKnT&m0}Gy}ohDvEruL{)|rXSS?* zG94B7iclv{4E{#mF0yEmY-1F?bCm7k>Q0`(NxIXnlwl5g&Xv<sH8V#WPt^<O{z>;v zZ2#eWF}h0>M7Nq04@GEq*&Tb;kq}me{j{<FC$A<myONHK=fKz5PbwTrOt1gO!)(6z z2!=rDWck`4>zmnFK9$Gqlr;keL8+347(hMHZ^*Mu{g9I&JryO`De|$%qM9Ju?Ntfc z9M6n``_{Qxe{eN6>|1H=O8DXWOX6?ceNFY3_>VE+mS3hF5QH~rK8Ik?XR@P}CsuM2 zS-{=bj#Z&7Mn}sFCAOsF(l?{U<uXTt@uFL+dj9?38SC4`j>+I`nF#-yK`3~M7w188 zF~^~9G*gcZh-$*{B9?Q9up!zGdd*M-cWF9okcZC2vMpO?Q)UILowPbJh7KP`8)6DS zZ(Iq-7)M&8w1g3>Z*J5%Vz>h0Hp^CABDaZ(CTpw%l!Zr1m&FEvIu;TY!5w+*6dBt^ zgad(52+gcUiw+=m8E`@AQYi7UeK&djw~e}D<;kHitDM^+XL+6?=se7kQBgczHm49* zxnzi3R4sEODU-oGPRSKx6VDV(!LHzCSFm5#6&|yJ^_d@OX*M1zU!0}ya{ryX2-o%S zTsz7#yaFAKI~B_?|0u1%oUr|1R$$|yDb^qra#kUpP;;+PewgJY`uSlkFPkW)SYCDo zZ+Y3T%a)gs+=PU#v<EUX3bb)Y`$IM`61cR975VbWNhFSn1W6x+d;lUsMV-?i1a!^; zGT6(aZk+anS4q(TN`uvs8z>aQZ3dJ3nokL?^N<k|skqR?1>OTzZL^5z4PZhBOTtVW zIztXKM45!bW1V;#S;-*5G!v8|nBtL%oPbn@o>MRJ7?KWV3VO1Y<^?9*q8>}rXqnrR zQa43#YR~huokWisciR3a0md?oo}BLqDf0|kyl(Z|ir2*N2MHYf^Ee-$nNYAb%nOQa zy<oq9!CIwpP!Y^YnhN&AkbO;pQZ*-6Gbux08W76MyFO*dOS-Kdx0Qq+9tpYBZ%)-I zn}V$_8GWcDU0%jK6UD$ZZotED_|1gQvMAlQ(-;YH82`BUHvXtig5ZqRQAVRE8;+uE zJglcs+x~6vVJsXvP+C1KJ6BTIY*a1m;JE1`XdLRM8VJmMa5<slVHtU}HWgA<PfXh7 zP6Z~ftZS`J3J$SZ^Xu4QQOgFfisq(lG+5bgeY!#9Q+WlO-~u`e;B1I8?rJ<m_*!EE zg9B^p$S$9~c;?*2Pw8hy=?3hrOxp<PRJaWH%hEywL5PH%@57Xq^}B3tKN15l9aV8E zl4Vy<=cJK^UR+^0q|aO+h6XK?`S3~291=rwjEk}4m7As5DkclmMOZs)6MdAW@TDM9 z6kx6luGS;Ei0L<0{?6bQo}O~&MUmpfY58#0!TUTr3+|MZpmeY2<|qPaK+ZfRFS@$2 z{G@g3Xk5eFLRIEej!#ZCF|E0k*wha#YQzpTzM2DBPGeK8_oJ<OS*x{^cG_Yb%z1M& zw$3)_aO_=^;6R!6FgNk}3=Sm5ZW^}ikqvX^Azj?T!U5LxVqH2HHIy54*n{b;sU}K{ zlFHJ8+M%MYWT4Y+4H%LQPqVJ}#*TrH9lM;VJCA|vBPk26X4G-6c1qV0*vcC=dydp< zp>P~@oxqsXWMgzyrGCJIw&vzTqePdAB-+&bAo<A2@e^kG0p#rYw9jQqClZ{~V20QG z{_1#iK5ZpE47P!q0XXWf^fA$Ow3clR_D~)97|sia45_WGJvJK+X|Ku9d+Io5nG`8a zdxJdP9HKs~!w{nwW1CX)twuztArbt=OIOdWo<(a9s*mAEsU9FeS>Q>#wwL8M*gUu0 zCTCwbR;N8O2*6)hF;BK*^fupyru*ZZJl(Rr9!MrTX&>6dcqf3aM5bWCvRUZsI2($t z7`+yQQI9??G9WHC7SNZ7D%~&M-amVaYGYPcRIYITbpc(Z_9S=etlRITY{GC{_D;0{ zI6ftSWSg|cV(JRzU)RN=cmTB#Konv8brF6&CVU%NzCAB{IDZ}n_Tt*vIlx&|d9nvI zih;pCH%w8PwK?WQ=`AAYPP~WnN7F>p4>;@r1#zC}<53|=z?>vKWD-F#EzkurOdU$Y zwXznSTf09xy|Q+0tsZ^m+||b~J$W_y%*vH3D;KYxJG&NLx)QBk0=f9=xl0$}*JIJj z#iy|5-<`X7rXD3Jx~s!@f*cB+K($?$lDr$#Bym7m5wleElNLs~f$sIThw*k2ZKG<c zhol06Yd0-$Vhd8ib<%D+Ad)Jus@FFVowSJ-(B964ctj3=(oE=-@ha^ZuhO2{&d*-B zeD$g8mrnm*wxi(Fi`U`J<2X3i&!2njYJ@32gjLLqWV{6UZzhyE5dKwpV0&>7sB<^& z43i}h*~558>DI|sW@Yv9vuCcKUpamD{Pne~Pn|z|{qmJdm(O0IabK2fNa(<4FdOz` zX{RGC*zxyiVBJZ+YLwu7*?yVNu+#L6lT&-flxbvFQ3VYm(oPHqh#4_@7cX78uyP*X zu$kSA9EK}Nk)YZB0-3dNL<iBq>pQUGJGkmELaS%e?Q|dp(0==O$@oItw?`gC=wgs{ z$=l6(sTIa&$*xGaB+<&HyN?7UKzSvt_CvNoB4}E(X>e)NP)X`EZw4CU)Vv|sWU_#- z&%}dRC`O$|xZq<sbc+(ey2af#uO!>3$HA1qz`pWG#N80q7gPi}C=WoZ6BmUH%(RsZ zfb;Mj3Z`bq602BXL)20cl~{L;ix0U9FW6kPE1SZkI=xnB*cREn+(KxwF_^<d%d#|H zR^VDz5vDef?l(M~jWAMBeS_ybg1)9g0Z$b+gjpsrYuM9zgAME%Uld_88{&cp8@aim z)$8&yjPvc@r1#bf8H-5|ki47O#v~Z=K#ZY5J{w-p5jhe}UD!t50H}z-q3Crih)E`O zQFs0yOBH$Z?3!Z^SXm8kG@>VqAzm-Y@gx{T2{=Y26hwvASK}de<=PVctTksX5;=bS z!jo%PQ7SNItH5i7q_q?0gCaV1?AY<+XopB!SxU)TVMjoP2TvhvT(|=h5*~RmS;%`q zq#2u{Fa=RUgnSyGpX+glXN`*>V_cYI?6F~TVcrB?iw9BC<5M4ER3Lf)J=(<JzOrqC z!w4d4+rtZ)m~UFdQ(*mBZ=2|x#IkPDNp_(-(dpqp07eQ|i%YE?XTczLyk3wdiik3k z&$QCK)k!dIHRtKwO=ForJk-SoNP`4i&Ix5OgZYn9Xf4pK3a@BU79yIEZPV?wxH~=0 z0cHvj!-nTkq7N&85rux$r36LgD8D*3G|X}JBl?V?y`Jl#g-r2-pdr{m(@k1&=KWZ0 zaTZRte7hy!^XN<m8tW0sQBW|(>GOmB81IrtqtVRG4QL%eQv9MGtQ?k0#%f+*9zlH1 z)N=~X6N#n>p15w4Xw7Wr*|0x@2+>x{-_kCKI5Qh49&yOmjTvP9GaE>ZnBtx0ov#*i zBxl-E+x)mLCWHAZjkUFm@amIR36^f_*6NDf<PNECz;O*X@J@Lzp)f>rs6#3SS)QAq z1A~o-BoHwZ0xyy1p7<~&42GjNn)w%W#3C7%045{ARxu(%i2xGlSza>)kmwKOrA??i z1{C7FgV(nGe$pcv2b9}zkO6t_hCs86B;QMmu4FZUch=bz0zb%-!-Ob&nvFp^GRU35 z*<fkVXbs_stFB!S$m+q{8mzsSwgyY(<7^2Hkhm`G^)3rpk;{ktJc3(^p-?tAkyQ?s z5wq>9AXi@X^aPHLu}DZC92zO31our8TdWsO9;X&e&%CKLxJep?U_<M0?#+gS<rXrW zYry{!y!W2LMKeRsX57(53J8c@b!wv?F+4kUfyfU}V^S%InQ9F$DFH*~Fz^D+fO{ip zPWbhwm}Ar!F|z6Pv?(Q8BcSE<S=~+UTsX^EpKY_l(@$K%o_~Hc%savPptn!voswcE z2~hAiI-eby6m$ZgdrcYV5bc`m#blyD<?7R1$(TytotTIKi!G@ka@@&AORN@j69JFK z^Re?>1hk@e9Ja@PFQZ#g0sdi-iWy9>*U93xY2=!Aw}YnZ(@?M+pG{kR9VlgV0vZhV z!R-r{bGxFtbjp9G?BH=eYN1tSqFJT2m0CI{CSGXquv?!9WtVM<#SHtQTt+~(vRU{t z@-`T~Oo-OyM^-1zs0ATh*_r_(^_*YR-M(Z?sux1liIuE4QK)HA^I@%}iDPZr0m6V$ zPhk`0dYDcy>5D*E^D^8Ox&dcK48s@&EF>-ya6?wI9$FEyR7f_VQI`=sj|+%KF{>Uf z0j~G(va#;0C7>d6<zBNT)+!bzZm=%}PWH;e@TNwzHIg7OeP(VBgcU$!EQ^h0G0_B< zvxwV35|~p9lYc6hcAz}~I~Qp{e%L*Camxj5F6Nu3jK~$;B;QVi|AYGefKa0;O$Tmr zQb+;f*-8gO%i!z@&aNUIHWB7crJeCMDGA0twud%W>qd^4j(uM=W&C5TDfnf0w?Ac; z?uug4C8IU$SOy|;%jX&atZLn9;>1py?16p;->lPb_LnI!S~g0Qr!p`LmF6<AAeALE zuppIXGZ2L;;%yrxjHugTo_c{Iv||`rIZl%@8q}zyBAt?eW=2=)QRs&m#Z2v5@Mems zpEer(y_@=J=4*;FRe{#ca+Q(RKg{{3<z7AiBIf7&IsYF(h-l#hu$jPVxF=&d%OJ-e zMpWofP{=40$6VYis40YFI9{||%K4(Cwq^Wqb^7MJ=$J72|LebYpVPfbFPTwOH4kJ| zfCQLsp^lmEcu4ot9@5p{fcz6UN<5aJD;~ZF;lu0Waj^rxX(Mrpt-?k#P*&5bS6G7H z*KChB3dA_>Cn(<%8k}K|&R#Pi?E{ta<rV`2Xxv5{_dn6BA%Pt;3Rw;Lpty29np<dC z85QX)<97XFOq9IfM;vl+DiQ_OMC4RyC4zq9gnv%AkKslQIo(>A#v9?YER?SeI<=%@ zOB_k3asB!Zo@wDNIdxVF9QentVpxM4@-LSdlaYi~$D9#e%4(O&6YlMcbo~d_BpCDY z(~GusW3`af6QB&7{5C!>PlMsc;OVobSe%&fh-0czQ06A6F2kc+NoVOCgVqvVdB)p7 z<$NbOiwBys;$n28VN;<7CT@0)l|-U4dadq&67zSEzSI$GrLz~03nnDuGIVs~m>noN zn}fQQ@v>wK44_AWvr(o&AQT4l^ORpoz}MyYfk)qgfih{_AjD3941GeRz+PNT-WDhA zDP{0ybJ@hRxwRCAMe;a;KTj?+RK|V;{QFt)Z$tRs2;VHym(V^sW0IwT9Kf@IJsQoB zug*+_IrKUJZpP{=07#2Z;k5zwwRVhKc5w>nfkT~6I#=Ty@3OF4#=02L)6%tph}iAf z%U=P75sBMvDu)wm(5l-4S5fU)nC4lhvl+K;h`s(Q86}`%#{goRmmMp|&gZGzv}gSZ ztB0_W*W^p;vDRR~hip2Q>}UAFz=S%1q5V$E9x%n(aDaECGhUgk4DbViV<~F>jrs1T zRZ<#W1P~}gOe3d5{OGqgt+vz;qg8U9lqM=E3=fC`8xoHVWkQ8lFqF0;+F;hO(TF~i zkOg|04GX?_RzSr;UED^_4+m*yp6p~kCN5$ne?X>1dI1F%pXl}*3M5_;q4q093!pp& z$rP{HnExe%OQeI2$g2;bVPkc1hnaR^f{GL`qfj%2bfln?E3fXmtS|4H@Jd<(<n)Zt z8u3)N?U7eY8cC?u+Q~9;na#CPX9uG2m@m2Lw$W%+T=CI3Q=*l5oo2;=Ew0E3iEeyD zB#4x&2Y5-*k6S4j0jTkrm4kq=leie;5VZL+ZkQH33i^h9{UjV99c0*M4o9}2;elSY zsheGOIgYz4?mPvVg0<4Y*d86al~cn(H-`mwBd6RUJQQBDxMGdVOTOj9Xh`<brg%Ei z_t>DJ=t*@QPnY}!YIgPTw^KYI!>DC4jF>P!VbF;|sxgJ2*EfrGyoP{*;Cs+w*zadK zUhtB*zafm?&>0!8HGz{!J3lcd2E%J;`yeSzbTktLj0LwOHi!olaXc|V)Ya}<k7jTy zpOH-P@YQ*TpV1hQ%IThPY}kI-Mb8RKC*C}}7@b)5EbpA#O)YR7s$cXS=iQdN-2<N? z%VFVbv$!L>g$($jl<lqOz7{7LK7#p`OFO6~Di(V^dCHPKTyeyastzSY1w<SBTw#kZ zQ>Kr!0HQp_(vg4OP=XMCZSZP$1&<O<JcocQk#%H&Ue!rPpNk~ks<?mETPDo}^9~?B z@*YdWbP9#nng`Fg)<OpEo{u&xqYt#^7+_a~6%ib3l7Wz$9#<5<VS%KbW+Q<;87+Z& zniCcSVBfSylJKxOCc_&x$0J+JW}UZ(u?24jBRk-RkH#Bp3boCwF^|<QYEO_n=DD2< zP7f3|lrYRq<_ti~K4<Z!lWQ%jTEX*}y+~m*gYhd84l@SyS&!HkkiutlyIJqDr2*uC z4tbW*#!NOtb+$C3OEX!Sc8?8QFlVy8-lZOc-!LxDw#bHo-A&m<*Od9Q@>hAfy`4xc zjyUnsg$stmMETEal~Rw2`xd0&mEv(x1f3RQG&s#AY98l+$*7sQ${DCjOoWe>ByKSP zG?L~3mL0kfzV2=nbOk^jx|mMn4Au<)O7!QksN)ztGHXga=|vFLn}*B2Kt^&kcZf@0 z=%K62N%il0U~SXBB^V!B4}nB;zp-TD`TesvyW(279&N@2y940dw}yGn0bh>`Ai~XI zJKYs*Kw9u><IqB>eL1WO2VfI9V5zncsKn8W0&r|5055K5{eiX*rPQ$4gr-H1R`KG! zN1V!FQNAIUb1f*i8s2|Py|iG08`>U)F2dm(P3A^PYFU^MEe%|g*1^a~ksEf%<6(e? z??nsX#=Swoc6XAOG`1Uc>{i3lzcm_-Q<1Tjbl+%ii|n+%m$uOMlEhx($LRp1ZDJZI zlz`~WUI_B8AH{YG3Su-#{vTUgNiIW!I6eJ>ju#|Fz-w&##-19`v_V`JcEHGRH~^6i zk`B<MQyBi?B$icMOS@?YZTsrfjy*&y@6!bUkG5G;w<Jz76Wh8Zb1K#&X<;l5q1KW~ z*|ZOXo5B&Z?Pr$^Y?s8Ll(cT}_s6o_43Mf0t}!R0OqJy%c5#49#@rU~H}JZjPmAO_ zBY?}fY+1*4GU}q*=90U=PIjzxg2(M8B~LK+VZv3)>ryc8GUXinR$uXcfG=l|>)NFK zg)iuG@cr{QZI10|sZ|UXIcDcRd*Q6!A=RmsoU{@&4UX7}42G&eJ|a4L|GeUI_eWb8 zL5CKn@M|*xD!Wq6wh$2K)kagu4ICD>DsDupQrSS<5m;rRRYqtezhVjB$`lyyaZp%o zQxX}$1$WbAFQT&{uFpJ~qn%!BA-2&34$ag7iKnxytI<XyXymQL1zF3nA%OHdaVyac z4>F1`#9RUz9TO@Bjtxt0FITEk&|jzTOV(@R4b0Jzj;w)@^5SMf_P(@*wHs6gjqKbV z$%3orC2q&KJ=j~r#cYtaSmna7tj*<qSsgbS3oLCo#I;#FN8`smBS#r~#gMSeQ-`=4 zVrj`Gg>&Tr4W7w#Os9}wjy=#B)1^Xb<JrXxN%XN{>1BZ?g3>9`(Llb7cZo!xbqfYD z=AOm7%@N-0*rS9>6>*knz^(%b4D&)7dhc;HHuN5DGgl$bL~fBNzY_{|Xw?S0(azTB z6nj32#SxAbBkcsX;7(zTZ}7Bho!m-p7|M>%Sx>O+3mFXbB|1GSQa{FHv6gnAfQV?7 z%v!h`jD=126`|AKi@z|~MxJJQIw0j9&IZn@Drj^M7^ZOZr=@p#d}*kdfOzj5B^aqd zr<DL$*45$Q&%CEh02Td|Lva?-)p`YTG;{)UnMg!mSdN;}!w*MJ72L2?Ao+VR4UB&! za>r@`^!|D|`oC-C2+q#llTow?0)-gQw81)xaKTHxs~K8^;6bJFY<_x7Kkvy=gldM4 zBb7&y`-Hz5{XKqs_0q+)t5;UeUA($>{J7S=GN)wcwue_9vCc2#+IE=J3-Q0I^9wg~ z`Vrdj$nywJ(}rkUQ~o^ufJkgekv`m{9cxR7e#hNi%0vcB3M63O7=_hGL<G$Z1vl%d zk#x7tn%s=o3Pee#!bBw&{z)Opc~;NGRdsG#GRGdsG>Hf0(@G+PxInOb$*sa3#y8${ z(?IeK$xrhl*cR1uMl&G4Evd!`Y)aQX2IC{G7T<elS!;fyEfgFpEf&7yL4F{GAQp0@ z3r#26PFo@(shAPD>lQmOnp#i?R=S;H-~=0WIg=pa@Vq-c<0~&~AIB>9w_%}T_=x)> z+`<hg&X^hjDB&!ewB-#^lIK}I2QqRmQKylZOt|tb6f|)kDDZi2YzQX>ivs7Q)OB1t zNI`9JxGAy@xtK;{CRAzQi-_)_QjpY#gLE_PQ0N--=p_AsP3i$8;r!g~s%tpQX^{S| z#v+8KAVG>%5#5hkxBJy$IfDioN?OAjDH}>+C_M96&ui-3mN-4C=K@kP5Q4Bim#2dw z>1<h?P@TaQ2@q6L1PU$S9=k=;<r!XOp+S?Tv#jTxltoyDF3$o}g_?yHkAYr?va~T= zC(+wCDI~AbPV~}08+nU)Lx$^T=@2TIsWdBV<l?JUBEDieXhSFfg?1?s?bw4{s<_m^ z9}~Nz4PC0^js&ty&vKZ~JIaW#7Q!@VDnX5ko9fnp{yKz()rPz4ACe9x4e=`P$9&&T zY*`*8Z7mWq2uD3gtB;t4c9oG1&D~)!aJPF)xfI^{W#jdbVq6@yFcuY^S50_wtAHan zT|93nRj#;S7y4X+B4cZ?JI+0Tr=ob^ZiEH_!tLFHm_^07*HR~AaZ2P9DP4KH8>BS@ ze@om&`Rb8q)mD^dNJO%{<J_lj1%z%@>U2=Gs#~^Th>zKK7FUwb4PnO6d_)dIJy4I> z`zS`8QeEmwG>4?SdQE#mnGpb!`JzK*Pv!l@<bIk^q)tpdSY3G%WAQA>O)DAo(b6Pz zVIyW#*x*>mNmCMS8WJlto7${dXTzj#R_;)ES#sl6x^=DB=RGU!*`mE%UqbJ~ou9Yw z6J2;_MGyr@U*C3hd&iHqEm$gSpa448w3FSm1w=vTTH3eyC@cd_$kmsL3Y1QZ%b4&e zUWLO4+gW%1ap-*~$teR6y1b7i;R*kBx%->2fQJEi442LZR;E^;X*cdGe4?V2;}qZx zEGn>cXf7LMTX+PRGM7yrL%Jy@CW^!}ufi&@5C{{14ka=Y05g}Nh(AIXnQRND0Gb9E z0}_nY0Rwut)k$0UeXxhv&oKBob^4uABB1!p5ID0yLKhWU`<Z7z4JgtQ!)x=DY$uty z_-7j~5lBNFgUwsCvtdZx(hNylKzN<i*y-tE`YID-;zDoH%eXyXLwQr*s$^QHS48*k zc-#8@At9l3|2JE+3`*pU%Cc32?#|6P)|AccMp8Obv>SzPk#UMJcAVUM8P-Ob7$ws+ zL@Q)@#SE(}eQ-oXl4#fiJ_tmUv^&z=tU`Z5w;RD%?gg`TKWkN!Pt6OAHI&*D<#t1g zm{wwKFOuw49kJP{Gl$aV8^?*1b9=Ii*HGdglXGQSN!#P+#S#^GEUS5bG@8k_w#Ym= zGi;LudS{eFigqiP>@Xe<vbmJ<jkQVb8<DnaYuIT#Q&?K_b#C=qZAP(HkV$;y?5taG zIPl92tq7Zdqg2M#rOYM{E`nVH>acq-ZvuK*xqJ@yT#n_U_?n2|!l$)P+PhJLzadhE z6_G|4lsFgC2|9gPL8eK*($b(7o%XW4ZQ<dl3YYyys3WxIu!Jt4uB0AEcRoR&FY=Zz z)Jv}{$;@0vl!m#Gt8zSc3?6!B5C$>(u5N5^iA4v*Z0DsCL<}E^3+B3F?)z9~YK3^# z6~oprYMo>|!9Xv&DQ<2TVmP5U8jV@CArbQtbcTc$Ld1}2<X*=}T<XzosZJb@x|c4G z5*=kmd3V?wh?FnQczX7%NMAV{zB@<TaEP)B)J%wBn?o7Y&kg3cZvK`YEtd74sHu&$ zLeT;rUXD&K+FtpGYuPHG;3^+3+i}Z=?Q69YX<59Lu~$%{ZSz*i@+W-eFY#HYdN)ls zo1mgZG915zV`5L%_460-UkR0dOkGE|<kvESy+o@!N$W<m0bl8^JTwZd;Di02c`TYJ z6d{`Zf}sQec#XC;c)T_>)>M-$U>>8WhxxULzOZwEhzCcX4Ywq^B|b(0Yhimkk1uMf z2!;bf?dYhkj^*<v^GC9!a6zJY+7+eMBQ&I+-LU$h{Ehu)s<EcvpC28hMZJ$M&MI_} zRp_IOuhNPNZXm6$N*vl1Pq4~4l0V_i>Op362B)dpY^81%pmkNZxq&DEU9gI(_6>OQ z(S3iV7&o$wW%?JS6BTOLFh?~PC6~0;sasM_fmv2P^h%2vunLQ3ZdsffvmE|51$<I( zq3%D<(+p0DRHz|#W%R+4R!t19AxA~N6K6REPugj)7=xjk2Wf|7eY}G~Uy-CBkwxQ) zkXVdVHp(<b4GuQf>12Dg87!oHKV&WCakXquZn&nxQQe{$cQSQ8$8jV{Hb10ecQ0ur zMG@zQ%MTu5jR7LXq_p3oGom#{{2GVQeY+*rV{Tfv=#gY3^zmH8cSq6`7jd}gF#xy` z8&}|EfJTFBXDc%8t?_~ypBWO3F&{wP<~Rr9#%Y@N%+4if^b@JxqsrKj<UJ1G(L5*Y zIe?hy#!I}G4eZiFZrcwuYtC5B!fDcZ<e>}iXW_8v6bo9}%V9S-mq5Z8^+z>wOQt0j zxx<=V&ZsJ<NH_W@N`ZNr>?VsfyX)p`*l)PrVQ0(RD@y0fZ;d-x56_K$YSdIiqE?Cw ze6d8PyL5Ld*9PN5zf8}Ob{_Bb47Gzh9n$3jwGiHbhy38RA3JKZHk1eB2I*3<jSbWL z9N%wf{G_3RpdJe>MO_zG<{*V?TWu4d>ATMyYERs93fl$&B{O^oxnf;Ap!znEk6{8v zG-%ew12s4-D!*OE0MnU5e|TPQ)=h_B?ZkWYCLJ?R+1@y7aH8$Ct6PY>9Sagq=TbBi zZ?<ISR7|Csv}b~CTotyXTtt%rsq737Ak%2iI04iwA;S_VJdptz!Ko53ws_(QBfhXf z!0bCs<#gs!lG-*EjY_4T*`_S&5wv?Qq;NoEx&M$}UTT|+Q$7w~A*_owEa7ZJ*_-&c zzB*NH^<{fdCsQI?B4Fe$#WfMOGcLQJH+vqaVFlczK`<mvW5QK+N9d$ymRcbgyd%JL z?3(x~b-@!L9UoKA0Z#!n%7uOK9DvFJ(ZD<yTE*`=>$&U;`$PAIQx+sass6}4@~rt_ zhe=z8qq%lEXuPkw5!K?<+JJ7U?qXfJvM9uqKab3dOH!e_`c8WcuEaG6)`GCysM3X$ zUG}7fltDsgH$6K%DQY0<`m5lDezycO)GZiCF(1QP1#Ab7awhkhBwA)#G{E`gBm^`z zm(LnIXuXc6tkx5D5ifhLUB@jIo+=u$NhOY+sW1`Bs_O7G+)ny=LYE>z(l|WX((d8w z-GrJ<QH_vKI5L(Dlt$fB;eGUdrt-_;kR^C?CNFMa{o*-|%sSC4$TBYs-6E18Z_KR5 zw{9V2v2#B5e?BcRz};1nLP)kT<a141mQ%UaPj&HVpQj4rIBVqPD)9(M=}FubNM{QX zbgcl<XgJ7bWD*G0^jq<g@Po${EGMIqnJC&#S1&Ph<Vu#;JzN~t8X#kiUW%t(w4Tk! z?CT^bsF0WQ!Z=wThotFZLa78*m~S#*+#n*}kRe#u@<t2u8LbB`y<ALoDH($mp#`lL zQzJ`;c&zX+?ngA91kP}MlY|XOG@<}fK&`);W%o;#C3KsrBYS9ym-jf9<S*|Jg@t!r z;L9E`%14jPe4)#kNFt{B0ol&>;_aj{tNKn(j^5Z%OCpKZCI!I708#QKk|(k%Oyk-@ zTWNx{VM=}0hPlCpV;Y`eII6mz<(3S8JX#^4l<%lBx*KMxlk;bk;5@)%g3boRh?$a= zsy53?71sq;gVpqL#+n3m%gnRDVlpS5xgtcQOStjMavT`3t-1L|P~O>yHp(O{Hm0Rb zz;I1s+yDm-y(Ac{xH~<ajg}(8pX<?Qb`l!;c1kzvcy0zx^Zx+=0RR8&y~}bVNwy%i zzGGJZ5Sdlo0E!?;fcXp-GcS|Ls^Z*yO_J5s)ok8m00@#%iB}<jBvDy3n}y6IGs(2m zEM%IswzJbhrj0&jOn3AfIxl~FfM9mrxp!oV>P`X??#JDaA3uKl_<3M>Zk&MbVBCA3 z)W>vBi!NKX0gBBf6+H7+bDmUkH8gcjl1vyG>EHNkp?F%v`i%^LC6$=k@G+hVe}4v8 zL6d_K$1h}h@-mG@Oz~i^F}$%fhcY(Sn9D}fw>B(n-U0+g4rpd+9@4M{RtcsU$44a} zV#2Oh55~%K)(9cWUURDI8?~b9e`KJdxYWjXbFVwGDqbj~G^j+~F!N6x!QuK=Pc}9x zJFREzjTiF0#MCg<YR$ID1bXP@Z;RfACOdi;+~EIhf}`aaZ8PVowaUA~pE$S`F<jep zUSED2oUt#E!?{~GTsRmJ1NU=-^T0~R#-JBz1Z$o%HdKi~UFu8#-zRug>ORUsIRHPF zp4Jy>BJH>^9lgGaP*;m+6*2ChB~R%L&5B<9=rk~kI?SChAB71h`WQG-Iz@e=S>a6{ z*+2jXou2V}f;_I7-3+tyDQ>J7aM6k`A!L)=1~DNDFrbj`8bt+hF;tt4Cp8bj=7$lc zB8ZKUgIoUf*;z?J5|>QFn94Xk(3ekd#}dC}N&P96aq8Q3t>8Cx`}x<B^YYC>vXQhO zZ8j<gK^1t=7J$m)&kGSwb}IGfOGeX|+O>vF|AqkXTPkf|tw#!@9>EMzCNHd$#`$2} z%?9Scx}z)O(Urfp_;A>=Pmv+tvJ}65=t%P3Pm^Q!A$NhYfwntASjeSe9Vv`o)jN4K z8pCd1lQ>MY&M1v)Mt(-{rWLJx9XxfPT%Wi-t?CF2_ao|&{p$Lcb&Z2C4rclA6{We` z!Q<9j`;o^JokY}7?rmNIqqlTw4m?+fri<Glgz*zZ0j>rTZRX}AUDnj>Wn~ZJBjV9o zU-3$tt+0j~s7<Be>Y*?7;~JL4J0y%5>~m;o|A3o<LUi&vHA3rYq<IyABQ=*Cr;0fB z6)EfrAG6=CuCA@igf7rF{1g&FinzYMK5a%@u!FC#w0g?*-C$)(=%<)0QjimVO0H0B ztYQ#HCNCY@+s#Xr=aJ%-=@1~7dV3)kT^G^H8#USh`RgIyDlwI|a$$95?$pX?cc~pU zoeX)A&ZCe8BJ0Sg{|q7PV9_71n=X|!edboFSx}$Kq~(UWQ~b~!cikENhTF*{g5XU( zYkGmXlz^qhQi>#?%yU9zx%0+e>(@7G21Az4>^xxQ6>L1p!mHT#C$sL5ZM!VnX4gKe zu3*!KMKgQ0EkKv@V~hRcIZV_6Iuxswynxd>w&JY+Wj2vXy1XdV5zaK@XtnqWh8&9T z@PXhibnC&Co!vPOG>FDM;T|7_0Uq7X%d)VaP^=?wkwtf!=uV*;UVU$g^h)!Gc4RNO zzG?mA$3uF6a;8gLuyXBZ!#NIOwf>%rqbQ%DLd6NH=&BZ7rC#X-f}_Ku5NIeH-m#=n zAV<GI^i=5z#@}*WeOOYvN;Bfqq%8^44u@heUW%J3frh-cTXy;lx5+~lq<_^$J;lyS z&3kp0bmT{!4lO(vtkfyaP~WiRi0aNzSuW^>A&zJv)H|Pog5I?Pbt50<y%kmw8QH(| zk`?1#N?2AWXBY}XyDzCu#>lF|NGnQ#&c#L4%1SREOR05AhD?}g-HlWw!OTtcG$q!G znVp6;LC;7QjY=m<9fp(d<{|@;EazT7HI<L321QonI)gRrH5uRD6;l&HgO~XESRyOf zs-l)pq7;a~QoPoLsEgGav4x;t2TK6o{HW6X-OvD+t8U>(JMW-31kPu4(+Hr{k^#Wm zDDZEyVKG?T44%tXH^A>Jc;xP&IKM${m0=OB?r^LB2`Q?V4ORg2!7f^JIoy(JXjz9D zt}KICw+ooM?fVs`dZ5cyj72faB`_|T9Y(9K(ua#>?H^_5%WKcF^IE`JiG3mtNkTWL zOE(g_%dXuUhFk1bUALy%ttwShZ_xXBbi=SFT&PwJ#@g&|)7fxay`hB_D4=U^i#LFK zoArx%xe7ckX-#en<P}uvk1rJ(mH;hH!KIxPh|dP|l>lP7Rp7A__Rp(;l%lIc{b_Zm zKh*+-r6(h;icha_?Pl@JDjgTkRz$p8cIAe#@RD{{nxLTg%B@TJ8#{2rTn5E&*@2&e z(~rYBB<~!V!hduQQC1E<HcNRbk5W>M;AzB>=#<61xqsCb-)=5z%&kaH<nF0Rf`mz& zI*k!^*&h!Pu5IK9bRm%wmYIH_`NN_3&G7Xu2d{t{Ci5i1V2N7=v&pNjIh&%s;{a&{ zhPfHV&{YEUc&g%$VfbBE$rZEctrDy1oc<7K6Q)80R_Zt1OycH3E#Q}2pG!^LH_uw( zR9}WmOrI$!5OdEGjb8mwW*}$R|CqdbeRQz5-(Z~mh;3)eq=VHjbu{<DVAG!Eh-Egi z%w(jYg4<+Wj2OZ;^E$i&P6`GVMzl=x0jYM3EZ<yaN?}DJGx3(X`I#Cpx6$LvCYtRE zbB&4!^Eh98k6pK&xp^LaqoJ_xI2V)=qsfl{F>619-XGsEP(S>W>1iV{rMu~*Xq(TG z3=_HAb?zpuD-yo+UUGJ|?e3g+zV_9%D_eT(r0ngMSzOg=la2k<b-NTa54@7DIf#^Y zA3D-XSZ9x-Meg{JbnJ5X;EdO$ew#ZxRysBN!?)$C=W*SfAn=XhB$1EU`58kEwmLg* zZ#I*y&68@#oUPTR4=ZWwN?ii-=;v3j&+7V;KY6l~Y=(6`djbFE8qlV<HHGv$vAKZu z4WZoGaov&x9lK`PlGC_*<+HcD_x=9!cQ1CI?Z0?;c=XeY{daHPzJ9a+_UK^$@RVYm ztJQ}dex4pB)+d<Cp#R|&ptDx+jv+*nS|e#>-9+xejFQZ+Aofeks*wC2uV-n&)W_CJ zU*Cw4)l^EgFVD`-YTIr}$FwIW4>CB<U`ev=JOi<c_7a_V!@K_qbzMrIS)uI;yQdYP zF%Gn>ik!VH<B27ulFrT7X%>oapspx_SJc<pK6IX_LHYSuq`5FT8Y%v-uo;Oi5@^Q) zau4<jN3M*-*rj;zWsk0l*#glrarl}Bqg~;w#F@f}u8q;6<r-&|Lkk`<Ph%rB4%}sh zYveRJ%5;^*N6S8#EWU|=Nj?PB%w`Kt&rs*c^<P(sA`LcV;+T{`-n|Qy{D9}UVeI_Y z38ax?NYH0Vp{6IgX0mip5FqYhAV1uYYRncNf_W?>(`y-1xG#Kn&9{_1Ec5yE&1~8B z6ZFa4ZBoD4j#bfXJ5uRTB_pn?rdDocT`CX&2vBHw7YCUJL}6a1dn*5Sk}Qz)Y^nRT zhy{|i@2Ke&sWl1eCVN^8>&MRQkl6-K#t7=JbAsZ;QbSWR4wAgJ*4otP>vp9Q$Fr^1 zQM>?iA1%|9*J{lWw)ErEfZIB92YOw19aZ6<*mJEPZ+9BWT2h-_#Reey@K0;QR+87s zY*enJz2lh(YA3Cq$Hk~#OKOcUOn){ihQLx@PY?4M`SXG3$#Lg^$OfFulI$k27;0aY zQ;u;s?{%GQmm)p(O$-eb9aV#F9>!i5Jzc!O8u+nldy7ISQbp9JU_mwR9)edVu!V^z zP5E=}eMEvV8XG%Zo&E6=^4owdZ<Y_)SO*W?7z7p1Y9<=#X;JQ~n)@6F@>Ihlw-XRY z#MpuX*H76Hm7z8Wo@K<4dC+jDb`eifuKTDJ_M+}8(tWVO0Sh+2*;#Q#u@@v1;nSqX z-zaWh%@fM|^Qki8^n2r6uBe8eOZgfs0BOvZ^GzqSlC|n9LFUq{z<r)$th7}-jD;(p zT14&xucmbTRCGl>@XT7BWe^eABjeb0;8B{uGOUQ7r>j=4JBvMH+l?RA2v(otuA|5^ zM6?`j7O|<Xb<?A&yKi0ViGS#PiaylB_m0MO;c%*a)1M{`c-ZvG6S-b9xpN12p9?&t z%vUAIkQ$ryMw42c&8FN~18d_-?se;p8#2v?z_y*27SwHruL^iinov@;ta4G+>1+j# z#~VmA==P&R{c0YW^)T+CNTw>q02)FT=L+W-Pn2_<X@%onHfQ<T4pvHHGmg)kAapz8 zXEj!FX*si=mGNUtrw!Hh*(E0KVkw06jk?1WMH!}eJ{sg@nJmWh<oyUvi<fwO!C!c? z@53H>ew)OPPBG<~OEVDz5DzWqEkZXMmyV03iQRo~1-I*U<&{Jp)+Zbl^PgjE^9GMA z+MJZ|V-7#Z?M)zd_{YSrzHTPgrseGXCYLEJTGDloL8={$&+9c!uz=W6eg|Ys&QvNu zF-*+#pR<>r)HVAg<oD%vf>0<{ajcZpA^GRvgHbX2yk7tRv0i&!pW4v%`S+tWamfF4 zV*3t#t*3C1AYX2%IW5S;l+(&5bsiD_YOp?~z^g@~Gu&99EDFEchFxl?-;JB*ZCG3t zqm3ig1?>2>w0G-j$EwvJ5C*!;sj_v3-ZXu*&(?qfl#ctD-*mR9AD<WrTiF;hYETQ# z-|#9RfQ6soeDGDf4iKAC6lJd$<6br}odFR63OI%X40T7<;d4cWF*?-sr*s!JPS&Kt za)dJu+AZZ*ZVZv#=`>rkzF~5HqR^fk1F0-dmX!SM2;{oH_e_VZ&<%4)6w3v7EjjW? zE#)h-<8~88FJ$%n<;1R}Rd&r}Dn4aR)C$6svx2NdvuI4_tEj`Fi+JFaoogUuM2dE` zW@lP@#GR9RD)zCDErX`uUQV|U)32ya<E1IBeDzh^Jx*)fprZnu^{~AcfFAS4HQ_cD zb`odA5i&C%d_rQ4o!=oJK9_*yU=73*Xa!F$0=R~cg36RXD%vU^=@Fl`L4G!?CABGM zXCR$zQtOUqv+=ODoSJp_Z%1{1Gok1VHmX@T-|GRjZXS_MPI?f}6@a=1wS=a$a^g+V zdyn2il!_uBmGi0GYa8rL*}Kj5nW`f;8kWA|;u&6u?^r~7eD8#gbDL!NsGr`CKHZe3 z))DGy(uN}xxsqnxvfe@43mf)^Q{8sp`|!8*rSrG-rDJUf)EnZf&>c&UYN(eEygc7e zlXb>o>c1C4c$%p9o6SmCL@WU6LmZS)5tL8`<VbvCI;}b!a4LRNK~>w{#-p*twQ!E$ z0Pa6+pEzhXh4HYCswWwHDp+#DO&TDov>IM&yssl$A7{FtQ52Ty6Tgv3_j#Kl^mKG0 zzR@@ooAfg!@H*lIws52ysGkD^IqyV-FHd872bxA3*K;=%J)_iT%(n2dj|l_n=Yc*i zAFvmZ<1%#KQd!vQ=5UmU*PhfPQY(H(7~^aSFou5ghLqYmmJ%lvpl-?Sr7h9uG_b~B z>pC3cu121ckAvZkyFZ7w=H|568^E~(2pK($R6GSwpQZ$e6o;yQSl#xtf2fqX$R^=s zEA)ilARahe8G2Ld@^PJ*AOb48XwXyHX2W2aZfn&6wcb3bx0q1E=Age%mq*mHp4sh? zN~mO%D99@<zm#a#@mY*qohyA-qyo1UtK=WlrQOJ~1MpG3p*J1)fju%74nsT}g$C3a zI1CqQm8EG4^&8Q$VJ9Mvw3SklNII^-0}{_<j7d}zlo`|H__(vxOzzzS`Rn*Te0%WV zgffoA1Ht1il6*vowwaK7Kr>-0|Ldp_95lU4$zgkgVlqMFIO<5iHu3cvdi899R!?;< z*QC~M?>liGC#Aw=wm{R#*#!zF#i%!)QeWxn{CqmeP)nlP?Q#&Ksj&&6J0CH%R0;`B zFh@3?E$=Wj(Ooi&jGhFn2zO9=1DlyeVfYwm@#1B61&rv5ht_h4P!}mib`dz}&=<nK zf48yFt+nme7EWsm0kvqvico91h-!@|Il`>@bhCXEDe6-niTH1CprTJCI#u2!GU`8W z-vp+_i(6C|jq-C~>K}5Q>_!Jniqa!?S*X#B!J|ku-v-sDXe*dwD&29=ZWjm4TR&yk zOGrRXf3)#n%Bgf^;|sTOMs_tx=p`h{if6E24jA2dvkKTl^JzFUT+l@cEY884)sGiy zftk=B#%yJRc~mhE+u=AmN5Xdjhfj6@@34z4=<6aPH2J(3&UXG4+9Cz|avxo?;TyED z9u(){r`YCtvIs?ks^nJc)K~Pzqr=%$2Z^k~-gKU)8yhuL+SVp8FX{&i?@9RqJ@sc7 z%c8@+G5TM$!j3CHPhsb~>gAJJa!O_=(LlKX5nDZ3W#LunSPx)~)MYXc$MGrV8m=*l zrH2*jf*EnZc2*c$94kQzUs3F83G?ZAJy8w{%a~Vv#|ap}&tHuXQ8VNQ2qiS{@E>%r z2P#<yk$u5`gc>?cJ@=h!=UQDl9wR?_W!b>0*>3PL(<LrA0)=vom<SOXcQnLuQ1tTp zBVT~Uk9Z}Nb#ZegR7T1nxBSZi!mYdB*!T&rGeZCvq4Et(bL17_M!Gu9CxZ++tBJdr z)M~K~;m2ft@jp0>idVH>j=AF_2FVph#r`UN%hM(KKqMh4hkZkj%A`*Jo@S>FomiIN zPR4`9xp;776w?<Ink$c#<kfg43YE)T-G}rjZEbIiATx>!t1C%>4sn4VDd&w}=D-O^ zHsP%58EkAnqE88O%)Xdk7H6s?88M&D(fBXC7g3iJS^KKu`6B+jQ&GO-mcK#xWcg0p z_A&N61zA;YKSbCYu5q9+R)=vkvC~cLA^rp3xGlXNb$pBh*3Txq?kI7oJQh)OkdIJ= zTu(rLf@0q!&U1LQwiX)87$}7;bx{Hq=qkGG-q24F4)NbM{@dXn9aM^hfkBi--rTl6 zv<?M4rrT5Z>~c(aEe(kih4YW@Y2!0x`_H=-@Vgc8yA|-eoyF>Oe={+4@mUw2b@3TM zw~tH%G~ND?_udUp#dm>zWhlkam$rDq;g6A1rmM<y+jfScAt?zvSMU<LW9_~}vS{io z=*7Z%F~S;Ms?l|7j284_K`$1zWU0DrcMXXVQ8^f(h2GhjE{vc2svxVe)Ay%Ii5Yl_ zf*1@m@;>>$|Br;>G%<;L@gW<~CG=R8`7^&pPGOIBG~AnxOFZHC=Oirjte>5JeFP{U z0q%$Jcc{f}N+$tJ;V=1gTz5e<V&a0G_V{`QG??WT!vh5$Kb=UmUrurH9OHGyiI1AB zjTUY|iwaLX-SF@ic{n<Cqx4w-enDk>UBD<xX$T^RjbtOKGfLHyQwI`%sEd<jkVY{` zcE+qWkReiDJ0viCM^xE1G^XDD5>6(8YH}?-rO>)NgVr4X0y_rXBm9y)N!lySti6XU zZJg<9Y2y>s_+-V#9i@hBDXyypHiQ-fMC-6uch>s_aOsi)tpd48Y-U%GcSdoA@-zJZ z3dH7xQD;k|S8Zs8PEc2Po3dyJ=E!>0zykGa!y`=CT5sp;4;py>!v}BxT<0g-ChpqX z{2Z16`s&V$!E7C^TTuO4w$M$u%;lGHy`|Xj#dHSe45}|lzbQC8TGO0gSi2pj{T%{U z`pQFeDy<zN)OxK~OnU<iSGY%CuwPz@l28dH4(Usz1i9HYleh3E@>awrC?H}{MHz}L z!iDf_tR3h8wu2{y@U7H;5wF|Le=YLq-hBE23q3QxB4x8F=7!px_E>+w|B=sv7&O7O z@;S71HqCQ3w&w?pI#9vi{iXA6|K*$ScMlH^-@Q2a_9#vKC%~uvZmFk8nk;>d_4%Yp zeNGeWhk1v`9`y>p;~QtJY2y6E_iSK1$@d0XS*D5e6W^=jRH_eF>M0%4_%VH@=NKSK zimR{O3)FS-3-cY{l=JSJ;wm4gylgn1X%c%HD~+@fmVV2JvY>QWl1fE#Po+uhsg(AL z1@Ctw>AtCC>;=||<p)orS;J+<+)UMJn7mqgVp3A2ZcbF1D=%wR8sxDODB8axb>g5~ zZ}ki5AG!ji$|ne)S$7MmeJ4VrSJq{zF*Fl6`ldH^tfBC&dlQ@nH^W#aG(<=v#tGDx zdpVl=x20IdzX?c#y8%_o=x0+;UrQylb}6+qP535umQ153OjX*Ez|&AD$8vP9mL~KE zf6Kc4i(a%mgKGMCV@PF#z$d&Bq_QF46J-4<Z2<U7I+rSjubvHPDXVW(k9w&P!Ao5& zZHq^zHp9kuz?)NK(A51=k56aW;5lbG#Y?m_37>bl`+MWz1mh+#_pyJePGr;Xrdhwp zk!aoD)M@yd4NM95H+3BW0y21b_Al7=c~jW{A?9lU-UES!y&v(oyE`FA?8EWd>=H%i zZ?P)h`qy_4zip6?d5V^)SX*W1Kp*01(Ve5|;(#G0**WY?IC>BxDE$Y5=ja*9Suvpc z7;l{FVH^)M$mP2#Ol^n<U5o{VO@rKmNuZ`fxo5^B@p;BJy1nsaLFNw$uq7a|i(&%p zVR#ehuIfz5fK{_FRy2#{p;)69WHggM0U#W5r?r{X$prZ8<gUEE+t{Ys7&p(c+ePR~ zb4=Eqn1lu`8nL#FPPYMl{7Ha}yXb#SS8kJqQu(}4ScD0qiAs`-*=&+-Y+PPmwleB( zYdk&QP$S%U0f>EdxW5j4mMTB6J+M{^;Jx02BEn)altFe$32ktiX!@ZKnDB>}(AP<+ zNWh*XgEG}Y7`R=4>_iA%yNAia;oamJN*vAP$AhEqU;l8F{J8t}?d~fSAd=T_lfBok zo*x_?ynY40zD;&t{e(6D;o#Ntrb?QVUrnYs2-1*4-|@mjQ<LfgD@6#;5|E|`!yBE? zQ5SrUo8gG!_~z4LQR2jwqP!UvY`Rcl)nIHOtD@}AlGmHvqwnACy?z0`-bs!vZ_s8U z!9v+MNJR=^_d~R=HMSOfaKI0n409kM?STj1&-GKXvGL>IbFfhj<EAWjQp&TaQ!h67 zi{!zOY&Pb?wjxfa7v>)b-wW<<ADZxOB&w>bw8Quvp+>syg?!Z>yiaZ4+`5^T#!|N` z^kNSuVUIRDa>A87K8$zmLff>wcKJK{r}8)UBE1EgSq5=DEdrD&uVqiMsXLU0NVWWZ zn_;aV<uWE(O?mQ0>K&4F8gGnTbR%^bBNerj)2*eKZ94Q5PL+@*Z(;^-xEv~QXYX{8 zv=?5VRX+*i{(yo(Na)L(ig}?e$!yXuW8zre1Wm_7ys4la(ekFUl&RpxQnM*w9DV)( zarU&?mqVW=iO-V6XG!9Ak_4<VjLNP;X`!})g#|!qJ{7KI*+C!IfF1^xoYA!3@qInM zwL_e`Wso;;lu})cf4arSKnule;WuC+PR3rSc#zfaeu}<}_0x`r6?w%RRRkOO)6fD} zq44PmTsErh9R$|aVBW_#a%^prhcb>bh_FG_adnb)h+o??G(&_x&&mrWA<@o+Xn!Fu z9Jgd$8Ur1%8G{8teT`U`l@fT3xUs!!x{X!J9IC76h-7tD1){~rwMesdQ-`9QjR!@) zhWniLTtY-WjfiBa51<cy#N75zljHl%<o-#Ll2b|h<nvDRd8hfj(|p=aqYdlW`Yee! z=*ry`vruvc{R%DT>D#t2))COBY*GjKy{?o;0v&|n4D-}hIRSGi^d|>#5|wq%9lvdz z2Saj=s+W2eTe(kGsP!F12Zm~L)Cl|yImQ$BqM2C0!_ZOo`SI^5DdMu8+Q-AJ|9WH& zjB+j{rfW5b_9;$Ym)o@}BVDaobB;Gh_{!#Ft}NxlbUcG6I*FtMTJrnshzi3tRyf2( zk<*J|@vG%Jg6SvGgA?nc-rY9Lzz@adnV!#?L5nTYH#^Bazo6~~LRmVixS7wU*=u7M z8RW~ukL}wKYokTFei09J7b`(3!y`%K19%+Icnkih@MNu+l&LeSz~K4&fB)D2?w|gL z|M*Y;`~Ub)|I>f?r~m%{`Y->>e+pDYYNatUycxqGu<|6@$p7<y{?GsAKl~?XBzeuV zN*8G!40^`zZFz&+bN#uoWZj~YW}^P8++&2UXkT1A;oro5j(?%~T&Ck~M%82IBiQ=i z5FzZ)b$>)n_H{&cV-E@20MX9^6}S(PQEye8Oj|joSBHA(#N?5Z{qc8spv|z*7kL9C zFWKf4yW$`9@7SHEfNA-Re+UUvrO1hYg(<mC-sg*djwukmggq0`(vpdg7|hIuTx`3A z&@N&}7nPu+Pt}7|z8phqC!U<|^6eYZ3K0CjSk$(Ob7}#MIGcH!gRQOcK!@IPQ2TFL zSjR{bCqe1k?9tvFudr#F$*1SxBCJA}@p80RqKnR_Hcj{^6kc);k77vC<<$sC3QN5S z(zD^Ga;uN%!RsCvK8}VfH4<hF<7&8-O(@IA>JG>Sjow8l>i+EQoy(QBHc=g%Ng%%q z5EA2C4=VX*i<{Kqj!fZ(IMfYG^#13RC(;3Db#5bhlaq#6wpsLbU3xy=n{h%A?0~f? zho@r17#6M*Oq+HZA(Be;9b&=Ep=#});>UaK4YFT<{b5!NT2@s>1Ey9u8{~FGw2N`^ z0dE`HOGx{6xsSzIP}8`-?1J4+;?v1eGjY!Y4kk;8ymU%97|rUnmY@9F!zz|tiEo)` zL3)Y2q(Dl#r0#@mfG-U0H??^=0)*R@TX3x297qwZdA?J9v61L?(8VVmvT$A-EnUuD z_gp7HNf^|!lB(IidGz6{b_>JPMjhHM>TMan|2s<}e+!cC+=*7*b}M*}yCh!gW|!@O zR!e@_s<@sN?b_kHS-GUq#mry$-s}P>CBD<R4in-K!4)b!B5*yDlz=g+XA5sur1o)f z5{w+xRmG^vuK~02!G70M&Vg(^zrj?!fq7=@6s*OL*TEYkj?+qFKo4;ZbK;U3rL*b& zd~AAlViMZeedmgC`k*Y!VaMf<S9_Tlr1o@YQ>#ng;(>HHN`SI-J;k!8D&)G$_uN2N zvda#6%X9diDDmy}NU?oz<x+WT>25on@oPQxpR_PhiQ8z(Td0@qRSk$Koww>1+gSqt z&gm&xOzyKcpx&&lS9=%?*u44T--r&VF8e%cX^mDLi+BZ~N7>Q>t|3IEfUCIwY?$;} z$@^K!d&}#@F2`}&boH{6c^B~cMa;DtIsF&0P+MPw;r@O&cuwM-tzlD<!Ig_TWqAVX zAlC+b_%kOJG4>Wv35k)y8ckt<V+yB-I_0G3qyO7{RQxih;Bym%(bY*}RTu^81jn0i za0;}wwWbZQ-e}Fn-xgP}>ECat2=4E~*hYU(Y}y|wGsVMs7cZqGmjf_!j$UYm0;7vT zuT-f52XPUu)6!P$UTG+7#4=ooz_3O$ETYq6=dO#iQ;2uJlKD4209NW0uqB?Lr^imw z7Uu262t>MFwSoGB>E|<~(2>H>vOz=?esmz_YRaGuetGi49f!>9s!jthGrHfs0SPoT zPxhlAG&~+Sjmk($uObQ`H=qrB5Q=B`kuk1PU+G0}luxZiU?I)SkO~S#1M`XfXgnDY z#^-2_lu+EG0kXken79v6I(e1PXh*B-C$C2dC-H@&B>S&E&nFnQVAPWoZ6YfnPxiFe zKqaQAByl^gUF}{KC7K;9=*x-mM}=dlyrqZ^{GL(?bW1$*$WEp4_5#*ZGx-@#R-*F5 z%z>EdkO#SNAkWB;7Q<_h&3K~wh8n~D>68NhatfN+#S}JZT{~n878Z{Pi9$wA19kBo zWi_;e20BMn#f;VO*b!7ke|eP+X{^Z+#}VCANMQOJz^j%9wfbjIl9|=AjI2Pq_-~fA zuC8jKTF>v<83@X{U(Xg8!?i~~UL;7*3-m(PxL_xxRshtB=bK3e7mk^qCIKTfloY~I zR~fR$B!?iL@CGDcb(IZ=i_QFbbl8}S{H8PHrUu<qt%fIR7l0f9E0q6L3V&pG5=fHo zTw(bMfgw)ocBg!s6T&)$=fL_c`so!ZXPh=iij2nlARR9dSg63%qdBA~x=v`k9E3a_ z630fFLgombg_*T%kjGdAHy__u-_=G+uwd$V6-sH3m`|sB-bTrlCgVw<22eHrq=GmS zvW^2Feh@cKns_ie@{n`qjytv+0Z}*<A5e5p(KJm5LnT&)<84|40`a>zFW_(j)Kf|X zlxQe8;NLlhC%9ebedP@kL!|4bL*={&TS_^eHp5~eLZ1SG4na8H@1N(pqyBEcpQP3! z7er-3i-1TE^I5Y>Cx#!|2uO`f8$`Xf=j|N}Iu()2c<HJwU7|JF!E~SNh=Dy<)bN^T z8Vt&vv~paedl)lfeLX@)E7X%vtugwNMKUihB3)c6ojA&;NVcG4ke{R1FQX4r-AT;+ zCdZQFq^z!z8x^al;O2Ye`bVyiCdU|Y@4BIK`Hn|}g#g1n9@FejRD@9j4a5NBg-p@v zYCc`T9>~39S;o4<Ch@=lCqaEVlPqV4Q~k(`(y2=*z+&h?JujsV2O9NfG^Q-ZJ<5~G zK8O?set=P@$#Htxmps5STGryx_Um>IV|T+AXpoY9$ch1-rvW|Ofkta7jWSABpj4M% zgp?O@b^I({3=!pE)E33v57W>p<$b<Dabjd~M*WTwY^QoOEInz~1XSkKE0N}!PFb6; zu29@8z*L7b(FOb~13e)(ZTtSoO6~j34o$H*%=V5hODEeA?&wN6kqtcD0C9z*MJ$D6 zK=+yFMheER;5%kYO(M(K=4&HTF4wSH%X{w2*L4Ligs90~s$xh~>yZ{G>vj{al;?L; z_sV9ZZbM4VAHP5eDpzu0YpR$R#}Iezk6Y%@>5)B&3VpOxqwR9kdXBlKdpw!u{am)| z#Dnoo#Q+?9Spax6E|7>IaveTmQp;wmlG=@JxMO9&mSA$bk^0rEXDu?k>m{;-Qpue& ze^cwo&_`<ju%an?Tx7MS8609*m>}7_LO@Y+`$ub*fft@M9Zq^uk>*(4gBVbiBlwgR z9ArZGxWch5%gXZc!C9*(l6soy$7P#T5|edPwn2wKDDn?x+(U>{%b?`?^L@zY`;gD~ zA-6Dk$M?H9&C~w+Btx(2ff;`eB3K{Bj@>B7EHTp!21x&;N>tR5A{Zo1$;EZ|mw7gt z!Hc`eWrnvKlLEn@tMJ|aR_o{TE_NZQ<9KPE^>ZMVLCHX|fgwfjllBB^Hy?~ITMgCL zywn{k$Mb1V`bCcv)uLcOD95N{M!8op160owY7kp!r9`Vy$B$GffS=Ocr2g#i`TEx0 z0OJf|Dma>xZ$cvNZ#`)^*G+GCpC9bLdbj`P@ZiPkSHM%-`T9NAe&QWvL@upRui9V9 zL^<8~$G?_qzrp{%+Bl~`r@r!<!(L!C@d?sI#^O68y+1|;KaPG1)FZc9L!92}>oeaQ zVGa|w!Q%H9)_qv=vCa>FbAxa;uCXPzq@NT=+JDY}=-PcI4>2o;m=|=5^K&B9Y@}Yc zPnH9w;Uvjh`Xg}ZSp`^r=R?)jw!J|%oP6Qj^~DDr4W^h)F@FLeqbc2Qa(hjN>)1m8 zK?*i@%vh6kgOqKjEDvx;_a92CVQO&*1riHvAUrFg($Hdof)!Nvd79KUR@ZbVFnnCc zmq*}mz58K|*Oa~9e46zZXl@5=d5CK}?;|!S$X{0K^6h&MT3e4{NDn%#M_bgdM_aA? z_a9py6IdU4h9cEjUe17CH;_5nbPG%NOFbiBs-BRVO2avzaN-%v+-wG%0(fb6EG#!a z14dhr2CL^>-s)_F8VZ597|+VdSY0#VJh&4c&d!sFw5-gVzu7TVu*{&#A^`hsm93aW zv(v<OGwAjb(AL*jB8#xW-+e6K{N7`!$dGHS&X<j)dK}kFK}Gsr-S2B}tttOfgt=~a zZo}d)Uz-)bX7|qx%{b=FVj0R381-6(_rNrM-Euqt8Ay!e_@0wv|CY(C>=h16)halB zIc+|`0|W>xDSJP;<;J5->{WnUY0D|1gX<0kF8elWB!^wIu`UCGGmU`6c#Ot?t_0B* z7?pbI8X2GshYAm#`p<_&zdy+B=lD@L+DU5nH~&!c=+U?NV@Q;qDu9n!sH%al3Y(K) z1;8cSRW<zf=JFOSGeu?TtSVO-J7Zavf)uh_LkyQ!-}u8<8<k<JX%fF3jI)`UqYX^k zi=m|*gUgtgP^7lMm#LDv`^wb?sXk5C<b}VO>g`7IZE=-MCAdnG(*-&p;tZRUwZMsL zl6t41lQtN!Pc1##NlV;|W+eQNBzF{>3)Enj%-0{*s)JU?Lqs3L(W6(Cfd{GaI^v!j z&ZZ}^<@8&H0Td0Gv&-lfl+nDM$$(tGRDq}!to0Tq)Fmt}VX*L*n_b<mMYc}EVpQiZ z*%f9N)CC&Nq_e5!8*eo0Ygfi1$`o_WQT0fG1g?+(uKcOb${rj>kdaP>QQ}FZ7)|E% zNE=a5oW??u!#Hz5a^^J_b}=u{9dJRkTxh-y&rRxHD1+Uxl6Pgzq#F|pGmK!Y5bcc{ z!1x2ja9kQqQ6iH=foCB1M%8g5@UpAcn(?{f3tw1fR4mrtkSJK78=`P|mQi?K68TwR zBNICcH3Tb*yNC-+mKvN<&&KE%W`xp&`vs+j7)=?v0pxK=#r6HTDy9sMh7&8|Kl~pb z=oB5sa|4rq(QzJ_)Pj>l8Okm2I_uI1|0wmht;0Z1^y+H-a^ugYeYpNe8u&WNkr(Ea zCD_I^e)8mNYgOTgfUd2%D3wvsJAe8swq|`0ykYyb3E?n1%ROR1%Zq6Y=^B6f{xrfH zQ{;>H^7VG7#^2W9n}txy&qnqRRst0hxP$E8xQs8QlMAE8W7bNYF-2;uCGCi|($x8e zGD@zS2dPiUzW9l4_Q$hmr(3aTBf2oRwFymNC5JFonLu;Ov-`3@$B;qNhmlQ3XkJSB zy!AK=FX?-hpdBlenqXw!3Z7=G0{|G=+0T|1BWQLxkE&b-<kJ^cHr&QE*|gz0L^l|p ztBnJ`T5ImbtM;aPgWs)J2>0KSvTL?bMPQd7PwN1}YE@46<~NJ!2+c_`RQ#Iz;CNLd zIGyLz8UX{gA8cZx4;M777`3-GU{=p5ce=RU1IH}sroNlcE_2{To1_)%Y+6C^mv5}= z4ieU!L10E-d#_b)Peu7HdeP_BUJJz9L2A`cVtY=;lnAnvB+4;zMpl+{ZDd`}hl!Cq zv-7<CLbAw{T8u7vfz`InCOyDg`VxgOJT?Jq5v?PXnwPRuld8H*H#YK7>#}%{naGNa zwAJ`);{}6w#~^?`z0=x)<qN{}DI<eL>^f`?G}h?mTJ*2@az&{u2`ijkV=^7%y%HOp zYRFMA*0uvD8Pt-=)>vMQFA<eOD~L&Vvk#V{$jB&@>9CNS*dvg~W6P7v|5(f}9Fbry zcSJBiTU>L}+3I4)Q0n9c;lNs3^C8*t_2<3ZZpxC4dc9__TPhi-*8|eFgc9Mc_*J!r z#BS{()L@kwF4{#ww=A;B__E%1q*jYld>GI260rF1=CJLN_2VQysEYB#M?#u|Ve+-= z$=5i2zEo)J9S`%2y6cs;a4rR?cXp4R9^wn%>u<L#i*rEx91f$ft@jvID;S5xaT!jm zw`f`i)OKi70KvQ6?AtK`c&bekdd;siS|adRL-ZCHI`xj|+a8bR=HF+fY7ti()oM8c zp*7@h?Ah}13~accjL<IFWq>A4sTv`$dKlX)Rsy}+|8Do_;7|MS4qknG@ao{`r|qEd z0ebm)o{xUcERafQMpTR14HP;J-?x*swIZr0)m4rQ%+n3cMRkDOXS;C3+(7lUSgIm) zsk7flB^M>D8dEDVtgWr_G*Sb0%cev8%97`~o|(N0orIweShBqQwvCsgOw!BUKfgQN z{T3(bX#czYw?K8;z^`PTLyK>z{F2b2$D5lE+m9c2?%#WOZ}aiv_O|OO0W4!Z!7M9p z4w{aZTxCK{FuZ!ElCXIj55HUKtN^&8c<rM)S8_zQNzqZ<^34GNQ`ZrS$#IqB4^(gs z|84M(ssjybtm56PN~2Yj+`V0r)0QQdd6KMqw}Fym<{R&p&~y$CiUZi1*ZW0DciY%9 zj{E4~<^H?3``^LfH~z8yG_C)&|MA{+1HK)v|8?U8f3};QiU&uD^BXJwweNmAZZ)yY zTBGq4|EvGyx8uLA{dN5W-=|Nh)Km4}R~y^7Bz`V0ijnd)N%8i@9ss+BBwnmF_Z0a& zo8|UVa-K^*9r|9X`#j*q`DB*&Q-T4jCD}#!rJRvvD2OTu2h~n^!HzNs2u8(_F04tG zxg7EDU&kX(_eL;YPn9x&Ocu&{x6DZnJ2-ruJbJL%ZnF8RPB*KX$IA3(k!RCpG8~UE z(hAz9!a=mN+3u`wZlUjpd|lscyT6jn`@w^B^Fgg?!7p>flr|<gLXCPk$$C~3Q1PhU zNjsf%Ypd1ncugkmD+{znMI6*$TsZpq+M~_Q&5C#HIDk#76%=Iwli~%V{D{cnirg95 zewQcd%=AdzCWz?+1x{mtuzBanH%b^Ej`1AxGsYVkZair?b2=tt3l*!JEN$$R!S$QF zmZ`oq%#72h_2{?O^^|UGZJ6(yTN^mpcCF3F8xJ<xourd0>{(9$YmcH2Y0~tln4E#! zhm)psJVxt39PN1tFmYYb;XC(v80o_CHH(mrxXlYutuI+#QPBkpMwA)WMF7L9yaoU+ z^1L`Z%PEFOcRIerBw5_KgpIvNu-B;qml>Qc%P|_QbCCkhve{^kS~b_Q{g7-|h!P4o zB<5x6yhaxW8how&_~9l{GOp1EQXk;3xuhse?*<jkNQQEt7$8L5UWf9Wdy;&)^`Num zi%xj$w4-@^E5FKnGBMk>972a7Djmo^(1y&)!HBQouxl;1kQIoBIu7T7O79Q=_yeHW zTB)SB9F6G*t=2I`FP7D+!{WFdJ$@A4cLHdBH)4|wVy3VP&E=sVYR9ViY(4|&_C5L; zOo{o|Q0=#tw3D4D_`O}bVbGyCxW>I{pG2H**6RGSbAry*?FQ=8V#T=B7MH3naE}UX zE@|jKmx!e`78nnzVQ_(mT)=%_5k&|vdL;8H(+Md3x7mlR>`jZwj8I0wsg%SAT=Mzh zQBr@NEhu+Ya+rO<Swkr9(ox)c*4ujA0(tdf-fb1*jl4I^IuAM<UzT}qy)RJKOM<eF zw6&h~(TR0Pw}S4`1x+=sGK`K1j9^is<+G+nSa{fd=;~eUOY#@m;&z81r1r9nqwy5> z>G55+i!&tY8HpTZN=Ewu`ji@`!>)iX|2Wme)V}xd(Y<z?O;#wXiNYp__O3U!sRqLA z+}k`|3I?_T{M-O$YtsXBNE}h}-YYv!4f~Wmap)Pb1YmKe7FuJ*9LSUa9G+}ly7kUW zxXjfmZ_UO(V0iYutjy~Tx`wFzC5&)|9|FC;ljy3fQHRsv))J2@{P=5G4d`EJQR6`V zC6;o`8_3&#)y5m%v@Y2M{4=L6<6btwwiFm6YHctoGL_*-lDE!V$z}^<mBarqWi0Ck z?ZZHDfq3Hq@fT?;cI<NWXsY1;6}utv-J;)h^;T(Ce~Ao*3gT#B2%W8anDRFpl%(x! z!SAozoBy~)vPsY+7AL7-Yw5sY;f~hzNt%#+D_l53^rX34>Ll~4iF>i3E=cO4Fx1x~ z`nJJxK*QQ-NqjY-(8r>pM!XXJ<o)$N9>~OrTTqViO4i$(bRol1qw#9T4ZlCnl_jsy zU=J|4EFF)W4mOb+j>HmNLLP#osNu~i1Pe?U_Ebe#Dd$i~-1U*kBG$+OpVZn=vXy+< zZg=iIR$w+<3~#uE2NlTSq3vQF9#`DC!W*9~&rna7CI9)~|8FDUj{o`J|3{^&&nA=U z_zDBL<sdf^CE_@)TFG0)FkwlGV#afN#tw&kGGXZ{F9vk>H6<w4EM>F>Bm?*<8uWue z{vlI7o0O*+!Aqyg8FY(aiiHcw>o7lfmx$sStojanv)<=Fc=`e)<<s@?+4>AN-yu*z z0sFZ!#V&KlT84n5@O<|bv?pq=lC+BCu#-^c%knPP{!UWAewPo+;7Zb6Y_EdF)^ZDL z!x%!&N-Gf&!nVLv$I)8lipA{27wooE)Z9H<^hAa2f@geRj+g#eg;X+aG?F|^z5&8B z1xIJr6@}Y3^^~f#?gzW|=*%)ku&CDAYOr#rs&Z$A%J<?`w^pe7FkbcklB&@mZZ6+Z zW9a%DkoD=P+WK@<RZP(Dh$=Sffl^-!Nm}<8Xdbr(FF+XJ?Wnc2>3;6X0hv@i2mAZ` z$;12i)Byk>bT*j<<;i@4f@abszJ+XqZ@S%xSbE7}LhlY5My#=Q3EztA*J7Zdf2X)p zrgHP+1`8t3eL5z@U0kS?R#M4{2US%4E&N}1Fz#+V?)4sLj~?F7A8b9y+Wp?y)`MQJ zoo_zu_Rk(Z%-dQ2LEh;-yth$Kdm9<v8kZZ&w+7mH($XB!_FE}uuo=+ZzA)&EJnK7- zF|JnLzmA5}-tIIA`lZ^f^Xq2fdNh2GR&@NzGB#;IGZQ?ITdeBBl_$P;t~K$H!BCC~ zfb9r@wou>0L*+oM<`!n`1GJH&Ii;$LRuLa)>img&jk$X0!W;H7klLUI??x127x6V@ z<p%m!4YCR2iX|BF2BE;h(;W4Q2MSL8_zR`z`>23pVxC9M<U!{IeL%(-<$*KxTsNbI z*AGT#Xcn<(ChHG3txCZ3H!CYWgtzOR2dlQxc>r%8tyua2Y}oDln=6)XZ$7wH@%zwo zHRNElROO_BW|{ja)l?w(Uj}EJ%ZBAo`Le<xf|r6`1!LQ>#m#{qU~vz~K7a#*oKpSM zWSFDgZGtpEWth5k_=r@ly1|P@bwb_B7#kVaWYuaJMM?G^)}27*HL8=kXn4d6O2D{R zfbL2g*uKjY;A=%ASN8Qey2wUU1TXQC+qZ#DC<b-yUhCsUEN~r7WS&vM9^%T{=VKv` z2PVALc3@7ZkV=Wm(4ys}r5<@V>OE;n6{FvdK3F$CIOo>kFNhcT*Ndu*ANKF(rj7^a zCnzi8km7+4?1Ee_a*<A5k|KAY@uy^LXz-Le@(^L${VCYCTiXsxD5UG=s+88w5I0Xq z($K%#CwTqJFyP(S@)aJVvMQA4n9e4on<|>Fo###7Q2Ll>7523&13Rpk<yYuY?BG0V zpVD%tGU+-G-0s8^2zww9SIdCl2lMZRPw`XFT80?b0gy!+0JyHGBZb8mOUIzPiRGY` z01MGd-Xl93p1uDY#}D-5z##vm6khLT<&00g7*(~v!(f@zQDOEQo|b>3SHIo8+aM^d zRtpuW!=fY{O<*E0a8ZWWitEaXj&3k0>jVagoDZ@o9MLcGD;!WhMd71<ck>EB3D9!h zolP^Id05Ro3{Aw02cs0hsUAA)A7wv&4To5l@m8BEyyCIeKLqv(Xhdo}Vm@$y1|>SN z(SeK#rjJ>A>uj3mzfzc4e$c=)%zvjHFH_?c3%l9h**;gP=YfUq=`5NmClY)Dl37aP z8GN@*##6_D+<%J(<!NAIu1Yy(!D(nItja~~6VocgMXcjn(4>J)4UJJ5v!uqZ1)qmT zBx%I3L}T$eSr_Q9_*^LxQms<3pDOi28YmpF6zVEc(pZ=}m;h9L<DNTvaO!W$JPM(& z)Q}ob(C0AapTlx|4$E<yupArMIJ%KxE^$^W)P9{}NS`Tb0?t{txk0JM9m6~zVTzss zGIe%_K2SOIW`VI9bFE5sb`Q^k=gQqlDjfZwl|5>te&yUQuubt*)M2Z(UNajq{vo$J zbeXBu8ew`HsdWXinTdhHW@}q$wzgA$n!+#mzE*#F=db-ngW_Vq#!>U+fHi1EX(XP2 z%<(kAcHr?9d0(foU>bf5c=+o)H=dGnb<6o}F)ghOZ6OjB5F3%R@m@B}2XspcubYVr z&iEpeKNJ>`pzd%eeu0W}zF##MTCcYV5#}sJH;2vS9CndOJ2o-=#=Wo&4?{S;Zs#cW z_sl&na6i%C2z|PDXZ6j7cN4sp7^kUr)lDN5Q8U7k`G6G<ieK}+vZO;L<y{S)ZW~cW z@bNaslh8)}_>bQ_xm!B{%xpXFK{82n^6T0kYd{Ns%!ZTgnq0xwzM)5hnR)Vro}61x z?$VQA=40#GUu*R6%dN-TwIE+kK0>wO_eU>Z)H!fX5ch?{*#v3Wp2EzVUuh!MWeF)7 zFk`pLfT*ii8Ein~<GAM>ryBM=9KCt*?oYdK-|fCQ0B*a3!LcbeHsxB!e=tC3jgqUM z;Xmg!j!uGgFM(zKDRblb!*h%Z)|4=u2bM{^Vj3T7WOq;Ha3$-KmN#~-hCXfv1fb&@ zdCj%M8IN+{=V-pd8~ZF8WZ*iPO`KRZo)Y6X6-=AnmVsl9;R`&;%lLuM3%FlLlR+K& z(6m};3dOiyFa;BR-IR0Lu6d3#WqzHwT$sHy5O*(f$rK<3-C&K>rY;y|@HKxmBkZr> zpVEPg9G#y6E7K9?Vww{wRhb%!0@hJF<sFwYT`PY;Cf;{W26}-ADb}enUJ%>AQ77bO zM$%ogh!4AHetu(6Xg&7&tHHeI>hFUQzK^JBOu|hIgSv^6V}fw}{=`?5gu@OG%xd5+ z_THb^3o>Gi!qf;0&tha`3y_M<hV_ZtrZ{UH`si=JC8s=&FWHrrt+P&l|0T*fnXQEy ztWk-UC@&p%ta;9<P3@_^X}XvZLM`TQ-mFy22wL){+IqyB5#Px_G7T;=b<RJ`K7NAX zqWM9S)W>)QHW*J^(b>ToEqZ@a{hl&Q!(NU;?UG_>O1x9UL|C<xg#S`PjB?(^c%8NC znk+zL=<1!?#cVi$`uvD8hVj+m80`;W>!SAr7tXL3BxhFx9%j?``ShT_Q~NW(EczT? z8rM6&X(6)6tQ9R2`I!Xb&Ju*5aFlup22T3)F`N}<fl9qP^Ux7@{Urg%bSaXmTM(*= z4c?%JwG^*<_=zg;k2E=yy4S6jI=`+_=s*a-xj(udc37Vi*6>7I?@TTmJCO}rHyGGb z!hU3xG;=5AtOqC&(|lRw)s#qS{S(Xnx#mc<cc{Os?P%<JX1aoAjns6kI**%--3@KZ z2c(v7%1MTz?qZPd+=VkOEc-=Tj8JE`-W`m4@3$#lMK2qy!_IU*N;$01HjL%G7_E~M zB7HQuQV%dzP`U-bYc6wG!<asJ=edq<P4hCn2W2jc{_KLjUE~A^zoLoLIcb~LnH~a2 z7Jm_}0rYKTO?783`H+Gy?>_lv0|)xVkz7|8yfe)ED-5f>smDcMWnlFEpLb-gxZvLa zpiM6N>JUSq)t(RdcAlv+pqoHMSM-)z-@P4xGF_m4ulA9lV$U_zLvMe(hE2q)MCear z_dUW6=#Tr>c8c^A?3O!8pU)cuOH#z6;o6Q_Q)-Q&u}7_i{#LVTU0g!ft&kORWHhR1 zwL76>)^j5rtRxmV-enbHK~es2v?o$QB3Y>@DS-BUP##7ak9nS)dcn*}D4cJg-S(+h z8dX7lX)E;Ems+iRD$GBcf=xsbfX`Bh$Q9Ii=b%861u~(_Jb!PIJrJ^fMNVrK?|zv> z4}qTIZ!faJ*#JR6zP~zpzE!-pUPa`c+RK;g{r+D^3;0A&59IZ;6Zj`tjD9}r(S8QQ zMy3jXYNXP!ieMxS1aw@3eG=CN)`2DQEgT<z!v3AqZHO-`ioYZzosbU0HfQNql*j&h zpf4(leh=sP(knZx>P9h|&vL(XSyB2h2hPy<OCMIk9t?`oslt^)@f4=6)})pGvYB-3 z>o}_jlz16^LlfRi78Nh?Rq_Wk504h0L?qaBzp`m8(T~2lxRA;hm6fr?MfA;ZD3ynm zm9fMy`lc+Ua#>j!OO(+!hlht!`w$%?%+yqi?|a##m;uW=Lv3A@J!-Y)4f17=A(7gz zB2Y0bz7^Dd%;nn+4oYO}mS$6E3`i0;SyJ!W54PP?Ntmls9rr-iiw=}zPXJW;@@zbc zzdTk|cToo0;fRdOjwev`Pa?NT6%5zt*6<DPbj>$-1vNT%OY5+GsuYHL{+^_plPDJW zv;&gHlGh5o-oaOu`FI6Ou$hoJ1&e_i;|#~#aP$ZQU3(@m58tA(933uPN`Ycd5ku-~ za836SPE0c_MW~HDcTgXBF9?pjF-Y}ZA|n0rf;`#Sf^%~p{c*7X+lu2I&#BwSDOrs| zXYzi6Dh@d~qe~mgs+ZZ6?r|pBtN>ykka-(M{VT|&jf_3fT!Cgb*VV~6LbuK5Nit73 z10K>!-hhNta^UpS$b6lWmv5eh23S(qnlV1G(&Qv&K9X<>5*zCz{x<c}=9dO3U;LZZ z$>kF};LWtxp^Et=%Q*CYnq}SS%7=CFKABGk(U%;2CFu?_D1?st`QXe)@MSUTk1t<C zJ9UKBluE83$8bojdPmS|Bx+fY`Dvgf<)(L}l@<=i$?-?;=`opicrjb}ElR&LjsS)| zTz=%S8*;|uofw$)vnYF`U5UPQOwIixS{+z-XsLanMwSi!9Z?#)Ou~}p{Fhp1L42c? zeyP#_InP-cnEKj5u_SR6Gijizu^@pgpIUM~zOgsn)Y*n<<kl9MdON6U-CN^nj8R)< zoigg3q0ZEz)X$b3pDjB+TXx*qvg1XzzyyUC`Cy`a*YK{u<n$Hm!!fp`SR~yhYypep zYlSL#SfT^Hc)hoaxOn$w_wDXWx~XNp%hGF8#r|$`DDkEa`VHSm;=?a2^TFBu<c8M1 z|I@oyyD#^xW{wRMw?XB7eXO7dJvdSA>8-IlzG9SN<|&HbNd^n>Cxd^6+0~CMqrx}d z)E%#sD=V~?(ghe#B=%Tm2vP57^D^-)z^@$$+;SBN^|k_r6`>AAlb6}mA#6`hM`_3{ zocz6uV$iot-Fx^u3@1myv4oD$R2oaQ)C145e2STIR2g65^3x_`s5t6FvLn5sdT{2$ zn3P`pTc^TXZpHxk;~&kBmD>Gt<k?ReM`}8lOhi2eKYoQRB^!jX2S~{ROZv|gAr8b2 zB3e}Vx?O&n)YqLSEgUD+xN4Atn0NI`T?L?6j2iIX0~7dhnoZcrh#N&_pZ7~cZ@3@k z6z>#6H{iMJl+hi4`u5<@FZcPZFvTFQXSj{BxQk2VsapXN#Ql3>8UsO#CtsS1x&YWK zDK^^JNb$2kt#5?9*RXey^3jz**k+#LRf(ktiS%9}D%Fuo80JC+nL6D(eY#?2bl}_Y zZPV`Q@_fL{daOwECwJ(p(B?wX=Ax1|5$Zx{^QT+UrZBP<iBr+qPfTl4c|~Ho9nmb7 z5a&-qoc>TN=hrZUPouR@BpPaQ)g024WN0ZG`2@XPQD1EZ<Q9`$ZAad;@y)_vuNAD- zZTKl#b?yjXpb7H~GSj5KkODtR0ox|QIeP!;+4u_S?R)$?AUS}<qL><wI_JESTNY(H zi-!w}?Y!|FQtP!Q+ECX9or>$=XqI-g3tqesFITseNJ}8jz}U4=vmFdJKFEKCBl*BX zL3&=)^?*Q>Q`gqDhBLY)Q~0lNbQe{lvqcJLd<SO<k?K_^2wX^pB$6#iWMVCYYYicD z+e?bugkdf*avbNTfF)UC6#kZKXMq40741m&A(pI-Xs_Fmz2wG*qPk<n9ROwH2u_WS zkX^8K6~li;+2~wV!R_1za3`(~*VINk>PJWAeI4O6+E$(|6e{%RAsA1gs+0BN7Erf) z+H||rl!lyhjV_^%Cr%Y-_i~EsqljBxi&pN*$xoBz;J)c~!oFZ!+)#PZ^t)u&zUti6 zc4ukhH&tJBqRle|?ZkkR3V*j#MLFGQG#catNwTr(f&$Q41wu!I=-89uj7b14R)Duq z;1-b{EmrB#LicFl^vFjO(`3vHAFwSiDcqTVQR%3O(OAcbMu3BLpu2Se&Z#Zr`_@U* zjh!auJ*M^oYcEWVO{;c?YKxHJ%NTDcdZ<1vqk{1FW`M}@Z^5d7bKj!$G`tC1TAp`= zTU4NVx<&bm$lc`93K{NTvy)_({^GXAo8gAc^38mE8J|98A~-Q~sz@=AJEr-Dt%Zp3 zZ59Nq+o%;-w{@{~`*rhQhutHE-oQ4o$TGeQRC@yiD6y3BPo=lZU`+YquDTmc-Zwzu zLQ6YyDX3lsMt1q?&Rpax3`t$DS29}?w?(RcsR&53Hn#LT_tOR&LDjF)wyJE|Rd^j@ zZ^PB8JA<j#&ICKY019Ks2I%NMQ}Qf1FA{N~qI@c=IkEi(`QY0Kc7|X>#H`1J&h>C` z>(of7oaH7&A>*`h5v&baE$fPe)_tU}BaO#wf(3DRBKlt&U5El{7L|2~I!a^Yly0{q zn4J~FRq0?jIl6S$snuyI?y2%Qz0dPW-ggRZd7vrwAEiqo164Qy&?TI85^$=M2F-aQ zco8x%L4ljlfnLTJS{P`C3UGo^iW&$AzHk6>7)IfL6dI5RQ;w6f%+9IdlMRY0)pP94 z&l8`ZHi9)5Q|MGsBZe<~^l2)hO4_>trXQ1g&s%L{&2i|OkaJ}5;Hyy>j9HCWu=aHl zXkM>l(5fat4_0mJnxS~$u3Z*GTw@uBp#sEUm@OdPP#ga8$AWIZWV@^6FPHM$=QC9d z2Ka*80B?p%Sj&Za;_?`uyI`^zv|}+m6d>~}ouJs?q)~tf_so?N7+2#4Sd7LCfA@C~ z^;t39@Um&R0sXO-u9qMMC&R}HPecSMVfN-_$coQ1Bk02Z;cwB)u@%q4Tt!sNX=w?z zc%hYJ%u`44UqSMok}?QywMpJ1na$B=(B7q2A*IRSxHM)$6%{I-Ja=5CK|Nf`hwvhT zhyV>6U^kOQ<JyZ#(o&Sd<HTCRTK)K`6eB9c2l!K;s#}W9#%-htOv=UBxmAbt`9?lV zgp!ynf*lr);`P`Ps@1*gW#Zhh(ty|O^ECprWJzz@jlW=z9I9-QQoj;!@}ABZYaO%I z0Xs{6ooXK#%u!h6ml*AWN+qQ7wNB%urQYl=#F~SPe$*6t-Aq2p!GO`hiaR83iZMJL zPY`=KMxriC%=>0`48fmD{!*NeM^ig9^NaC$(KESUKs-A{-wqHe&NDQaR59%)<ATS7 z7qKIhhQ`x=L)YtIKxPcgdE1I0v>i>el6Jg0ZR`0!U*#i8$YzK$O>(WnvHgrm@T|DX zEyK}rmQRwJf?HFu`1<3V_uK63Or5IHlcf-rFr^&3qjQ5Dz;Kti!2srG3KC~F!=%Q_ zNU6+*Dq(d#Xqrb7@LM7M9OfV&g<2QyfWFT}a}BVPLl$8#vJY&{>RPh$&|PoT&nlec zHIV`)qo2%LUBMW-lMPbXp|i`{>JUZAb4!r7R^Hf3_J|$(3U>7uA9jx0&Gw0O)v$h) zmT7G$&bmA*?y*0>of=TRSoX{3kXRKqa*2j;MpW%u)XA6YY)~spN!2HasbQ5A4d1Rx zrzZ@2Jk3TWhs&geIfv3#f-Gy(ALB87hDOI!5L2M_^Ad!aem3HSntm0EY)mfEC0oi- zx))5qL;VrXP43Hf|00ms1hKFbQc0cDCh&QWg~Bx1tTkLTAl<Bj24qz}DmYCWjd=O! zIJ58Be83ZerkIx*<<L@_6Qv^p>Np#~d;-V_I6bmfQrB6@GiH>~p<XfV4LAWoDaj9L z;yF7nT53+QZX*KuU?f}|`x-9h4G|UWlu43k!`E7>%ECVD$cb0bS~e5gfkXMP-4Xxq zJG{2=D}QsfPal>Rb;3H^Yl{7~@lfxtHuAj1<>6TG-MPTBcRo0>m}B}7$U>I%>9&kk z4M=z%ERvpJ%Qf!YbR8X7Jp=+42IXxgYvI&!mKX2MMQ#L77j%lPSJ}r^Sm2v(FEgy8 zT=#<(Qs)cwZaq><sUDu<*-1;@YtVkR(46Sict+s`0khmIk&nr%H5n5xM@Eqknn7I_ zN-RAX)Pv<f!eEfsiv~=>X>S0`Fz5@tLbK(C>8~EpvN$D}6bloU<gjG~dOTcy(lbj= zozLt863J+}68u#%6MOd2m@;y;fS*2lAeUZ1kdJbCW=w~g`bx|)8N2q7>d8LxvG3Jf zR$!j%jW}Ts8I7A~qUoz4GD(_Lk@=0JiCcw2wXR5^{hhlPJu!GM{$)U3s+&j|oSX$S zgE99bvRF8xF|b?ojDd{{CXNM8E2n)vfbE6p*t3fzr(V3$y?|BiUe}_Ah%G=<$E>)I z_pKlN;tN%LA;r~NZDq6Bv|dw%Yt1Cii2&d+CPMEoP5{C&bs6t9*AjfnKS`Mn1uA3M zS-z2ol|7!{l1}}~Nx(WF!Bi=W?s)Friw_v`Xfw4WsqLhlx>A(A{gimg#j63Pwa)!$ zG92IGi(ekA9voAl6BzS}Qmz|{!~nq+<=dvb>?N^M9aAd6p5DvJuZSM~h*Cly1qKj4 z1e+MVoQ|O>a&%df?aj?=j*^a<*nk8-DmjObrLzmzK+Z4l)oTVr7M1-XOMaQ>(}fa` z_=*sg2<00X=F@Y@Nk<EgKpDv9>_%GsoDOi-iApl%$>7A-faG+HxdMw2ovcI%L2Ibg z3R=XoWCWB}%ayD+k~tjml*IOiVpX9ch*(-nGdF%A(<YJX)d!^=9wlR?9sy}pu|niq zmF#ZAj?54J{_7V=95#CE*oW6bpt=+}2<BdIWn@&%8e1HK-EI2D!E@pZm3`&7Vzht; z8+q$SX>tlkh>PFkU?V@($%Yd*qcM_-eenZ&bBRZcaZeE;m?UuC4eR>Wq2cY-?jGpg z0#Y^9I2j#GBVZjNb}igEGh~aZcVGqx8DF*82N}&-IPlR&FTTGSwL}Us54JTh4zy?9 zFhHm3vH8`sfS$OmUA$$r#|{tdE9jr*Nt<M%VoT6V3b#)r0zDT;e=ir3izjJUL3fL{ zgW?}dajxLb2lfObT$Mf#S3QSm(G1|@^>*;`PpAsrgB@??nQfEZJ2uge)#NuzE9WC* ztkn1`6|KDEm@u;^mFmC6E8syO^Ky&F_zsPsZ7lIaTWYjL$Xm67_jqcz>5&(!rt6lu zW`|8mY3YhG0D!v;EHT&?rW|J|AC4&zD+7SnO4ne0VDowd8-LweM%+OpZ#{{6_#QTN zEq>yQ+W?%V$^f#UJ4E1QM{ufgi&j0K1l+_5VR~e<5@pYiA^){f(U1AgafA62AbkR8 zq%dqKOOoc;!yvX{57b5@ie}+4$|I|igpk1Jnug>4)Y-eL+v}{-UVFv%L?Gk3?&-+< zpetyHVpDxG0U<%u?DX9}i9j`E{?>9zwIqW@RYL6uNDs`BZC)@Qoec`5UlTq;hXa(- zID94)nvZ4<2u?168`>!3AD-nw$bvvr519_?JRB0Qin|-b?%$4EOq8KcNG6{p@diQ% zX}M+>m_=oFIc7<fSBe7$Z%)(|1tSlSan_{yd4t*E$7b(OBG{?Ih2|4+bi_<45JfbM zY{co^S<)es0K22e8Yj~N1~XfDD>xQ+mKI87+*$RZ%?$4{c4YY#{?)FcnmZPw4OpRi z=SepBI@&CLLf*V|YqCH7b#nyMk)JnD(RCmzsDX{UKda#eIy%(<w4mVY%h4M?f-ZQE z^p$ruROWs+ffdx)r$Fl=9I&(bY#am~Ga-*qMw0!Tk=snXXx;d^)ke>7^qeucYC*zJ zT=O;u(LVo>e~KG7uL;(+#c##9H(x}(8AKB8Db)e<4pbVHu~;$AQ7<0i9MS6(`*^MK z>mSAc+ZcB+q6(^!nC(E161fGw2^D#M`%CGG!9>>%iEm42tug{IRWaEA9M$7<RFBV5 zJywY7p*=f!{l6@($TUV)D_y``*{mT{t+~6Gw{5U`k+fp~tIk}|56ljn&@O|F(JvzM zqAA2|)Gx-&)GjCN)h{Trs42(H*Dt1SB}_Tt&DJ%S9ZGilGxzz-eLi!a)nf|j8w(Xc z2rUK%M!n2NREdf;Om8=IX^~@?htZ`hh2O+;;{pDvyI#(J&nY*&_{9;J$={<pz-&72 z&BoKH<TVa;*io=_NpUUKaWB=_R;rWtbb0aXcCS7#FoSmm!i)$n`_rVRwchAI&2JbH z_#uHY>!bR}JRvIqD~>8X-rz4qA0rb-A*oALwz#27WI!LUR@6g{yRa<q-W;=FsQer) zI)0)aUH6k7H8fbATlXR(ZQyJ?hM65L@H!N4AG;V@5+J3z1_o#S4g&#_aS|posI$^h z7HY|7-a=i~Rou?2OzjA<(@1uarlC?lzrw{lgq4oac^F1tgU`nLMhuM#@yYI+gQT8O zyBa8}Hm3O*E{QR7`5C3`9@#e)Mqi4^1O!m@ibAeFQPpa9bE}a&=RowW>Is&a#J*I{ z<t?dV2>kVk$?bQ6SI)bi22L(`J#ZSG2RM6p%^3yCAY^Hq$3VF=c%uD~6jZTlP<k!W zZZClwUE|*kU{wsRZ=5V)0!vzXpD$7~P{f{oMP}P+;&WHYU~O4{An;ZkAFXdxrWCn) zt7chGp4>d1em+5)31ML%O&*^t!<|PZm0;({o+?pvIG<2PLcWFO%TG}Yno*M`y_Aw5 zHeS^LJC{k_Xw^4BFx_Bza>F)bYXqnhJ5aS*@L!c3<)cyTQ{c<|=HYRdIgNrd%CD?_ z%9hNPor1ORG!-pLl9bmISF_7|3Gywv25u%R!f@^_NKS^yV4I*H4xZO*dGB3sGJ8jK zqBA63>k;_Z^`FplbN4P84|Yd=5L1W7Ttn)It;cG`dZPgNw}ob6P>+fIFi*Vey=68A zSlm5$JPmYFVs?<Ee7WcXpJB(9R3M$j^9rNEzTRR}0W~gDk4`s?N#W^@W!3mN^xWoz z^+tGyWri2OpR}*X74Qh6niig7)^6^6hQ<_n8e`;Rj3-*EoF1o`PJGg&-*)`b9;`TS z^xzqfF*OrsPQ_KlhB+D^T#(`ZFu@E8eb3%xZ=zNBMR@=apwFfWf4VCb0TdRw$OnC- zZxPCAaK82C9WK>!_S2~CCiGe*1N83(NG={^WvS-E;)q^R5U9}gg-Tc6im<5#m1&^> zFh-8<b}@1TVeEb>``~0=UXb-hq^RmFHvt<?KBceWjb=DdryIVR0e+ktA=MknAG@Bk z@rvjPUlpydM^WInz>C9q+Ejp#ixbPb$nCq+smL^A5|0X;TD3)|6ax^ja-h$*C{*Ll zs|6lMU8NaYEZ-H9j$)i-$Yx;F{gtbQUiHn>8Wk*UYNe4XhG|Mj60Hf|{CfSWN_|zs zLX&gPq{3Nmpx!^9#jWI;lwui<{d;PSw5UpNkp22A8It8A)L;-j?(?aV3J~2o5S#k> zCvDwQ7eAO{Dxdj+Vbhb{`ML~a#WoGr)p4Qa02G@zITf=k<4P-5VM}1q4N`R_zVw`7 z*8%|Wm0*-eek{DzkNpz(sUai5+ns_k&CdA`jtc*(G2wR^RjnBX;~|Rj*E*=G`PJys zL>^t!H$947Ou3y+GCE6>ifo#n!_pR>)WlUK-{3ed-&B!9?x*Yuxc7|u81#TH*DXqV znT@h@_TFHM^++lCs34-1D$|{ojIpZ>%Wtmkf>|;qZ!v%vKOUfCB$;P_Aft9};>dqf z_q$5FAWuLKj{n@KeSc`btvkmu#~TG^v9z}{NxA9uRAE0apDkVrg?K>ii?%&n#}0Rs zX7CSgjJ<5eg`BO;beOZ+MlxsvTK7j}iGhSl8m5*$(3PZeo7C~E-JMwlucdG@G%g$g z3I|w_<j!cz-)Ft|nDQw@#&$*;qtS;P6(T*=3koS}<AtxurBaOoZQ$B?eI~usuNuiW z7;Vf)T5CQkFN(8S-4yh%+hsIZTHQMi@%AL38TWGKbKZf^cLbmB2yT5tiOH13dy5j} zp?6mHp*YWw>N(61I>#d(M|7H|v@^g7U`j{2AaMn33Oz#m##_#Ut_!#HL&}@EZ6*9V zC+(feuc&WJp-(1R=N_k_tWD3mS$(rf|21m%i494r1e)r@^mQ?EDG=>Oc`=@q8|}>p z_a8slTt`=&bBuaa^w!@L`8p}x);|>G9K)|==3IwT95;Ae`||$7$6IGxC>ea&Z{N!v zwCU?v|8Z~YK7D=g@P4<O@zcC_uhZkQ{d@4?ulM?A_qjl?fA7)5O`ZM#vp}%(7&fPr z?@sgGUVp3GrONrEM`uu(zV>_VjGKSl@7?b`;-~HWk$ipFeb{>-Pw#i(KknaV_uiw& zT&?pc&$vLhpLHH*cBf49z25x?4|o`lyZr|_4=&Fh-+Msf)t8U5UY5z|+j&pQ=J}%s zj~>w1?A{~jEPcJ-&mMFcLKnK7<#wCTGhmet9#k*PW&-``Y~9~tXq~fr_wRFa58BzL z3<eO~c_5f+_a0{4%ZK+KpFP^L)1CsT?9qcx$HoAG${%h%WT^M=J-9FU?%nIbJcy^K zQL>(6##z`G_B6iE9snKl!26FLbsjR|bkBemdbJaqU@=PYE-W)zzYF@-&*AVuL9@=M zAVuSC#WYt=6LMf{X#iLzwUr8((&{kC<Z7m~_9bRmbuC1p$?-H}6+~?Wd!hIK%|5WT zSKpaum}u+UY*-8ysb1(HZOnQX^^Jf0u^4S&7U|l{qBk8Q+f9DTzRwHF#93yeavd*v z&b&5kkhR@ukqv0u>Vu>Ur;=zbtin3qcE^9r7^tz&lwavbu|ZI;Bf1~ioVp;pmaVKm z?lLdA%)4s58=cMey^YPzM(5rJIcTi|!x{D0Q7v1p7o&B|1ijwNMjx_rBjivy$iir* z)f$d}Ee3;(Hottd{=?x$f7~lKe$2ZYyKfFQ_6WdRa*L#M_y}jX^L&FjCP_wRG0R_1 z@~O+cbh6q~?7h};=it}W)rj^x8x=!E2xJl2^qxPTD?(`l9aX($ZCILYg+=x=#1(*t zosdL2nCaK^S<n!dQ-Er{dAI0EIj`i=1?;F7;{mLLj_DJ-QJfX$(k0qdK}J8v?YH=l zZxka~S18YsJE)*n_AXP*0={K{TLseO-sYxFIl2alKCor-J?`inM3f7h08~KoMJ1l@ zhd{q(H6PGyO<_3>huEb)J64@i#TQslLqElmC<%SY26zvJ@Gx#?S(5F#26-AA%xN<r z3+c<ERJ}5ugm@b~gG~|pZ&kLuZPhjEd<2t7_jzUr=*26}g~>PSN^1rXH!gsS!J%e- zfQE%FTpdrOT4Z<{-5M|mNvUsWVA3kfrB#{#aI}{u<g)hJc<A%N;Pb)YRtE!P^wWtL z{n$#GS9ykPf@GdIun1=}RHLBW)}(n*?_SM^j+SP^W#KXD2eE>3tuP$)_4C*APKPf= z3zKPIZC^1r4FfrJ(qLr<0z58Gl=l8fWk=OVTc=CSsGmbv&>G`<La|NsF{fwY%QFn^ z4FIJ!J}a+-WQO<#s6`+T`hG&DNTFBRFn^g%>{|{?=CMr$yj&_|<MZ-|!y{yC%DQ1g zANpF!YrUD^ZMSd<>AZ&Oe`Zv{jSZ$V*p|E@p+$VZI6KQJ6q*yYAr=Cv1kRH0VDl%{ zg}hm}N_u&(dA1V%rCZ(5mm>t>+v#{1nK0I^Q-7)#1b(Opn&i`8%-{;W_b$5W-M8)q zV5d$`{lsV%HCVL_kyrs2ohpVlf@&+g0;tZ#87$%-Fs-obk8q02H~TOy`j&GZPQTd2 z7!7J<<w{dg7l)+_S1N{%H7*w9WMwZvJ1qOrnBBgO*)joFmq`9Jkc^l2J!oEKl6Y$2 zg`dwTpk}@9(U3|mNX~_k*1VuWPov{Yt2lWCU`WO8H9ktOBMw&zXPz1bTA`>`0!tQ? z@J4Hfy^%UUVQ`Ut2R0FR{7TLP;rF*piKQpO>(jAXN;WN#t7FW^E2KaLE>4NjHM=8A z;6PF;CVxmG<8!cLFM!X+^q=M7&+_o6Cg|Oz{R(y^%mRZ<;qdhjZ};}!ef#?D%iW`Q zuiqYgckpWW1uD_mJ36D7bYV#x|4;AUzWxzY<JL0VP2P?#%Xs0}&;ITH-jOcN9*fn* z|6%{9{pauYUcdO^<twW2K3@>~o)1SARepH&53gST_$ndo{reHDA-My1`t)hC_x<jx z@Ai}31Y?54Y9Iad1{$OxAq3pC`?NuFJU*(h{}v|u-HY9yVEW%Z`$@Wo#e38--7mi& zj2cT#$CtIw4C^z)`pmFy@BMSk8sBe;5>2Y3wdAg1Qg=Cu8O1+kf<**TK!|zC@kqt3 z1h4DScOsv%Uvx3chooOu=GP|DAymTBRHS9Ms@a`1d;En8fOT90vA;j>kpp*$asTj3 zHpGAtd^4oVy~O*4EP0X5C`9$6$E~eq^7wx1kz1-5yibmDyvKRmX|<``J*ULpMV`Y{ z?2UV4Ds``Q5B_bpI!>t{v%zRSOP&E28Wq&ueJuCzLF+Nc(Q>h>8FVnfk21bA@kc<T zO=TcF;*<tt#WJ&Nn~_OP9-yj;1C7`8MyzVnz;(NTfOzcy6RO|$?fQtA>ku*aMkJ{D zs2kB!pN7+G57!@n-z96}Y82500Y2&gRy$kwn@M}~UI%9QkxeK<YETXl4zt@<RAA5X zwLKKG$=$Ud%kD|foHcl(#~T}AI<J)GLF_Z_RqT|bZv5r~YGH7L=WJWEr0}t_{e+Sy z(Z-K?;j@L!v&D|uucOhmoj7wCJ+5ECR!XevIjM)jG@G;EzR$`FI1xeVeAFBF^H@op zuZ=Dv$PRC3v!aOICOQ~VxEYndRV!r<37d5uKONN~sxNjok!X0n_Y)o9Ei{9e65@V4 z2KN>0zn<32Z_}BkSi{6xb&K>1gI-#9n<3RuNTOxJ;?v~gHAWK^HCQ9sd)&sdK{-zD z_6fg-RttAqw;5#INsaA2oo+`51H+9<vC&SG0Li&_nSMW^pNYE^qTO~%SRHqtW7p#n zakTS>q7&+(N=7N}2ByEV2d_d~!D5XPnYVl`g7e-)`_1e5cj>dsDq?-LnTP48^Yo8@ z6!QszQ;!si(|kxI&17DTW<fvWj*?6pnsr%pVV;RHHf(z@h9Gpk%}0ssH;vo(Cdwt_ z^Q&L`(RY5FW|Kt8kji71SP_c0>`4p936NFASQs^eIy?^Z4Otq<E|13i2z#M*=LpV^ z5QO2>g@4<2rBE`Ze=Q3Sn`5LP)E?=5n<4v}(aMvpTQIFS`P;T7W)skN(D<Aa+0#s0 zM%*CO^xoGGb(Eve2i?yH-OmTz)irY*g`Yxl!3>?A&!JPVkpi>9sp#sS?ZYyCF8Vp) z8!zW2tWG#Lkp(^-7jkvUN1^(Wf@<qEjyg%30CLpzLGfN)TfAiRvCa}0K^@pm(qxMQ zj1m@!FDkBJ?%lVR1=~Q|8sGUHS=t6O5jQ=(S5DjVZ;1K*hHxs0jnTJ$ec^;Z)1r`f z7_mj`P%Nz509{877Gv%A;-o3U-nJ?0$%M;=KTgT!v-?P#v<i3@>o<2M_(zAcRKcG^ zz$zaqhJ~m3oxuTOin)I1mJU5U#*<`#R(}a8L!P>~M+i|FCZNc^8C)eT?YdDeSKVj1 zCKFBQ%19=`gtkn+!N>6!msE9@52X@YL%2l<LsTS^_*Q*zUn)j(+pX8awE6@XN4W6_ zn77A^`ow}?R`)~45dXxkSStP7_vN~Z9(AumlGHQ)K3^!!Tr?dn5rKEF4s|4%F5&+L zFfuB43@<9;3NIbKbLjlt#!svymS!1qDqND#xIdkar}f&w=tDLr`h<kg@kn^D%d`9g zsr}Ozi;oXsd&;K8xIAGc7Y5D&WPT_xZx)NV)Y*Q4DL@7blR=P$;nd4Xz~>Wqs^a;Y zS4K{UV>`TdQjU2C)|5}d9srV>kEyPy%)1AM=Xt9l={PJp=<xB99y@Q)G=KthD1dzW z8Ri-6`lt4@G}%1$t5TBNi##|HXX<pR{Ma~wm&tmDSs+)%X@E6R8H9*4U2U6|jLP#n zG+3hJ5Jno6^N}Q68`Itg09x(_yJSbV;*ok$Rpl_mEx4Pkz*R(T7co1mL%w9&WfC`b z^D~?+E7So>SZLWLR|3lf4JR6p7jjDvV2Uo)JOtx?>W2hD>t#iP+2mABFo%qKpDWW4 znb(%Ploha+fq<qLjxHdGEqlC|&9Q&7h&{A$APAXL2}~t%#eqnvhG6cl;OQKR1q9k+ zfWlV2aaTZ@yOCp(Vlf~aU>HV)0;D@G>B}DU`+Piw^8z}F)Pei4dsYMH<r0o8Ll~bN z7|)7nIWy~pXKNphCv5c15XQaUe1ZeX(C52E+n$kMiF^3t#Uh4p>DM8LDC3+~vjqk6 z!u(yV2hz%)CL4ZH)UgYWww0iCOa|@@$?z4uP$|Gv0_(ATjjK6)-l=%Lb$u<$e!91! zHt#O(a(+-wyf@57bC`?;x!zLR;=6Dt0GWQPi9yuwR;C|y3H^|ItrUe(Vu#}rI`A)9 zu_Wr!a~;dUDkjpgmbWY++O9;DUM)K>Q-my0f>i1~td=~DI4VAz&IT${r`zt^U;y9e zS)Vd@xRtDTR+WhM(nXPeA7E2$bR1~OJ{&nywr$QD(G!jb(!}Nnk2{V4?VgHWN6tgE zPkD2CqugTEb!Q2U4t)bY;6-JwvV)Y)7KWh%FW}gVQT4;n`4tmvE5E&g#pc`$P|dlC z)?W9BA%-V@X<Jf8$RT(X^v$6W+TLwaAsrH&c6B-)_2Ah;_A})4?z}dgd24#uX*z>$ z20i@3yO>$BuS;U@6&0nfy3+h;R!wZ`_q<B!>I}DfI#Lgt!4-qufL5CawalxEwzM{T zX@k0y>zMdzjF#O>$dqkgXIJBd)4Tzl+8Hpc`-RlyLHx?r42wd`C|vY-0+iWLsI%8j z(D?`{{_O;)c^lc<2FFO%?QnLr_4IS^RXu3pI=0rTZY0!ugD6J_bi@6p(Fyn(oVU2W zCojk4OzC`Clo}wPK)oxg!cq|EQ5d}-e6<`9B0wUc>FmE$owUvOm5vY(HEq!ny5v@1 zj!gE&GpR%|ObEA`QL9aO%D#G}W3{U579_9geGXY!;(&Tegyh3bY4zNoO|xE=69XA! zj4KB^0Bz&LfnL5v)YZ+c4)i0cHh@l}ww3%I@dklqvH0Df^}!wkNIRnk|J*@vmAO(_ zYjQ|9qApdI#E*S{XYWP!wO%_Gk_<oB@yD^uHC1goz8W{c_8tLR+nYo&Bh=XAf*<u% zPH2(Flff7$WiTF3JOp2dEP~lcd($564f%MSkvJsu6o@}{2IuH1na}teHDKPE4@AA+ zWMsn+;@jO!r%euR6A$O+wQx*D(X8L5#;5cv<scU;V75_(imE%?xR9VePlNZO<aLYr z`#Ceqh&gTm-KdRyh@>RkJc-qS#}$>V{A0d~?*?XWX;lM{zkq~RAs1uGw>O5PdT(a- zu6ksqKB3EwD)jg1bU@=0{E8Q?p>1INaQ<s1q}^SCxH;@1miEc`Z%2bVM+PS++ORgw zG3;QL4>`$`5K)zcZ?6w^Z@|DBvOU08b;$y|TKm&T&N>_iv-j@Z^Zh@)dl&AFC9E`5 z)87f^7b9mG)u!@y<?P@9lI>~B#2nsRwK(z}<)3}$Wl7FAipsYZ+-?@RqmQN6mh`KW zUiymOyw94V<JvCTe!}syM+TzLoL_t9*9nLD(_spxc~*|e?xK}En_EgQ30@EF%;^rE z0~t`nM?_+o7~31XmLKufVG=-*wY21{NBM3%cT9%z5!KWFqk0|Q0==&2y^J-;jW}x6 zMF;1Z50P1ba;2rSeLkGb7Cb0R0C4=`u8p)k(te<!7S*CWidz@ZxZy!~eHPnb<#JJ_ zUbx1Px$d1CY$H@CO+IqOef)JDxFP;f()Id?vY@KlSJNX`UjH$NvmD?9MP0E}SV&+@ zRF;y9awe;hs0HL|2LOxV1ua@iK|m%MDlW!g;R)Gf8P9nFE5vA}y@!HTLIJuZp*5yH zW3W_5cKp0nI-BiI%f&6Xy0+uWBnn@6Tp$Zig@ZmPKS19_7h~27_R%K&bE%?*iM0)- zmxSYsQ`I;MjA@zc^@ryi_hmk_<BCZA`YzAYIcKDRq8bSGM2<^zNsyIQy>7&&#2!`E z%!CCEVymW#(hy&9q-9w^<N>TJ1|giObkMe{TiSJ-$;Te^*mi6~#!pNxcc}r+UTc!g zwihPRjX0HBlxfr#dVNQw>t46&Krzg&UcmW7k5_&>$1VLVIT(>y>B82+t|-FBug0zT z5w-;eUVOtNX*g~W(d*X55JHK!#R&)}BFzH#x<N#$l6pkQ;lQEW%lqt#G%NO+D^aNF ziaxqCdZR>a-(rx$#6P`K6!7lP@=Jn-TO|YJ-@$yI3Kyo}+8UW2T;qcNxx6SaeG6SI zr=(*sGy%0kg}f0fJb_t+WlACHoCD{1TvWC;lbC?)tVI>mG@?{?uO4yp22lZZG}`14 z78BqEEjICq7!c@ToQ?H0gIvblrA{ZnkME{*Uz=Uw4c`>b{o~Pk&s-tqWh*%x>$^-i zM07dn^(=eOUS9Yq^AEsh2V`%l_87{ZGoJPj7bVa^sh#WXexYn39W!PFuDKqKW%=-M zBQs_azhP%L-(9IM3;xnpPul89M?LA_6Gn?3xnO3N3_vEF(TM3b6h~!Wuo8CYM$_D< zLicoUj0lMu7Ty3cjghh1<w_1vj^=!kvEiqNMTwPpzUA7OaK(4EK2mTlagHd7u}mK# zN<*_qfV9-V?*zQ#?nkC!;UWEt6?}z=kPJ6TjzzAM1UXNdl6cGfhTfq_36BWmlWV%G zb-jY*ns({+tY5go<|51iEBD&3aAlyZaq0@$r$HiEygBDi>{C@ZR$`dPJy}C_4+l-V zr$M$@k|wwZJ9g5NPrJj!L)RQBU`8452i`S5dmEHwksudC$$VA}*uH$4|59-LJhmlJ zdT@P5p!>q{IGj<&?hklev1<T+gK7Ygx-|DTZyCk*d#glGA;rE_m#5ny2I-8m#E!(O zf~Ixe-7bc{UrTE5P^>C#e+TQf(*Wo~MP=Fn_)(V`>b2q@KV}P3fk2T%Ji<#6e*x|I zPg*4!qw6D@>~sOm*%~5#g<*1=8rnhMJ*(ZZj%IcZPB!d;KenKto>UgOUGLhLg6)?r za%`asi7}gZQ}po|<|;+^cwu{do`S0i@n&Vg6Vhb79_<N~r30^$&xv}Eo$;zTF`CS+ zGp<FaZ!v0X-y%ObTA(yy8Zg(}>i7Wf@!((v2<@ho18%aAsfum7f#fk%oBpwjXEQ#z z$@Vt`T$rt|Kc>^ch%*<oeqWO0yLo;mY#>1DwqKleg*Q(q(YCK=I2a{*ul0_Ux1vpb zlWC@sr?zEb-Xl*GJF$Zv`QX4otS@=uXnelr)~#vS=JHyV{lX(;Pp)O%DiTVBg88BT zB*`RSrQTpfe4&#ln!r5`0OX_;AS*4}jb!h;N)|?4&<biPwMZ(dBw^FSXr$5{NiZb& zX;CMHV2f={r=ybKQid`w^Puv~2yY;*C^cg%c|EdTU1XR%CRV)*1#Bs7<Z589eHyf7 z+h&#MX<H^JHFpJ(?o=+Zn+b1oC8AS1dA$4z12krg{<v0NS}yzhSGBs_uGtIBLy^Ub zjjf?{pik3Bmpw)Q3WF5V7h-TXszOx3xC%jo=EJXm2Wh85o8a|uMQ$58zN6&tvvqkK zJ#bU`r|O$hFQoRpkL*DXSz3Asna5UPWKXxO7OcGaj=(5w3DqxeX4032UGH1vhDk>? zz7-L13XVZdm7bLB+v#9bX~g=k_jH8lcm~U%Uyq!G=mh8z@qdeFqT8K+qJAhbE6*wm zSgzo|aWFL*&o~x4qTf4+-0m9Q*b9Ba4Y<0uUUGi~?}(K}U6~w@sDiGH;14Ui=_azl z2MJ0N7QF(GA_C8Y!4biBPYOJW(2}cv#a4!C^6^768C|ywy7fB@?&yX2b@?8b-1Jiu znpf@ED{_YA{iJjc#L~I@p=l5?4{Jk>dkntJW*4ntF{<CU*5V4jgVG}zbASAu<*r)~ zr$Nzd%y=<&sBVTJLLyQ%E&IyCt>^E|HXeDqv&P{UCR~+o=2Hy)+$A$*XVgG=#v!@& zY8;iBj}I*!uuT3*%F$GfaOc87gE@T1#}Qm00a)>(8ZjVFpH8=D+6!Kpb*DkyyH;!D z`uv;yU|!DdCQp*PTIBuz*t?eAHjWtl3;G|{xkv&8rEZXmfWSyo8!!;NLF1bncqL1A zx3#o_R<>gV`S0axIGo3>q`XeiCGqa;JUATAkVAf?zk=>a0MxTlx?lr(;RObdc=TXz zfGTq9%dX7q8a9;_Rr{*dK1lJUYcs78advF6SH6uwzVtynGRsPrTGxjRTSPT>4sp5L z5O419ZYqpCa-$e@b?G4OQmzl~K9E05JFyFI9|CVz%IJI%JlRwYT?F?j`vq!X%R25* z%Hr)ygqZewFOvQFpFB1gcMoMPCc1i@tY-qCGPWQBaEct=!U3=Bz37$r&&!v+av(rD zsRk+ugUF8->&Ps;fIssr2BkXW7ZX*N?BUlfS!&Q%l<>5$RIZQiGhM!BwH^km6{>Fc z_cb5v(s~%2Y<5kzna4{GH;gu44_5$&Zv2`3<E4AN<!U~&%#U<!jg!}h{`Fzxg*oV( zfD<KhJO)v1n9*O^afB`=WtI_2A5{eFH4|3kypGH<EPp)nZnydOV@u1hYrkt!EldqQ z1wDcNc|_qu)EEl)3q4WZ*TtiF4xyJ3)8Q?Bj`4;`Ac)l<FYS<md+3a!m@OPT`{k9z z(;Ed(ieRijx{<dVXBQHSmIKa7g1)1`<5msIDr9wM7^4s|jg_D~P>H1XdGKD_5O0?r zjfC|?Bdi<ASjcWycnK@f5vr$Yb@bte-c9GX)so!hN{#>Of6|D2s1rvEtD!9V|K1am zQYa@|ptMFAErZ9fKlhL~l5DjSe~<b(C=O`iBCsNiiS@n2?vWFfOb-qp&+Fc0&u3Gh zX^g_)(`<1#FWAr@9BEi5L8G)SxDq)PHFJGN33VN`j8E-4wAE|GKjNJkO)213je|Dq zIGsyK(?^@Him0-n-G?2_FzmV@041OZfuotIU*|vjxUM;g2RFsh$v~>xyQd+|+&}fB zMwVAlHEf|rZPpVC0d9@dPw**cvVKFOiQh{<^GVF)HwhP=Rh5h#B-~A%-K$NrvKm$L zv?X9~o2u1fS;*AT;&<%GQn2dfe1Y8?7sZ5lQ8#xmK?X`=6vWIKVp>?D95HqYh|B1X zVNjlQJP>+3eO|sD=XGl=>RsAihTCj6e36Vc(ISIQ0livL_V5TPbu%!!#gHdZ%m5<3 zE$S4-W#Hafxq8iDW^5V{jZoOGkt?09%kl}lITeJVILKy}*)C%M9IYTd<IVI=(3gsD z1f?9&D=4?|BM~~g{~8<~i74B*6CDO~p_-)LPOlEVo%<raA3rK06&;R1-CXnKfs<eA zgz09!;bej0?U$Qr{N?T``tm8~Sl`DD()GK)R*MwfTXUQVS8ITW@p1!Q4|DTie=;Y5 zAh5qq9U|bWPuoYoufwXNHzFemv7-@YdW=NE5AY%*ODlNk=j=z2=o<%!o#)c_bjjJM zSKnf0ve;_zEKq^gX~4IaGMsJG&@}~LP4Df|ZC^|Atjsu%vU^cuUOWW*DnD+`8uq2L z&eiTS?^`9%R-yA<r!pI%_e1zy9O82^fD{RoSsn)_;xBLH**3iq&AKNDTG?6b?LA5S zp9ox2+f9y6(>a_zDCW_`wTD&?4X(Ry5P%gF?uYr3j;6z+N14=r>v0SXoVJbm@ENNx z&qC`*Sv4mZhtco#Kr_Z1jU~d#l#v%y6rL|1$rx<BetHupB;%1CrDn&L(@aEB@9IMJ zY=H<!`2!D59VfIlhWh*y%lRjk^G_`2y^|IlzadL&!I|xgg>y$?vk9ACiceXb!CzeR z6<?cL7f*T07x>|_Y8Kvw4^6vEgzqEV`kA<wHFF2fjG%PBtccM0Y^<F^l+V*;Q4$KE zn7m5r_>fb-JG?teZYyR&;04N(>urOKdAXP@pU7jjVJ|n?cYF`DqW)$LoP3XxzoE?U zK+gL$n{x+j<3b$Ij-E{l#e*gz{1W(c_M9CYons{wWb2yS7X<5Y`oz(@LxtvaJtYvq z>$zKG+63BXsV1UfvypuD<jUt$3Gm0#{TG9xB#_9mxPjdWN7pA1<@#<~e;ka86TG5> zopDm#QR1naTBHKI-%O^UiBeUifw3&HOcutE-=9sEz|-rIvJ-Bvf4;*dbai@ue*706 zZ!tIB-@jhXudgUZz~!hDHl2V_YVvI`uO>D{=RC!e)M{?jY6P$w?a<3j>a*pl1L`gu za3~IgTX(KdV#JkTYY#0}ZEO}}huvt}8gj+Xz$+Asz<1{hPIrs{y4=9_ebtQghJbFJ z@KkX&fqiMhhI~pMLLu_#-)jyZnj`~%n)ITBPGRxYx_W@KjeYn$>e!X*J|d~{EeAb` z{J;0;NWfd4JKGM3sz!SG+nH=MXSWKF+??^h@7D9xy1oV(U$0M~Oc;8juoU4*=%AnQ zyaIW~7w^3lhSHH$o^Q(VMnJAS20dSPkg`BH+L6C1**1i<DiRg)HJ!r~T6N_&r@0F+ z$;J#(i@|w5#Y*<qThAE}p*O~z@*2e<KiK7Cl=m3B3LlfVA-C$DdYRBw$6`LKCSe8i zr~rwI8%0=$-aoh;R|;7#OHww)IN`Y=v1K=;U80e2O>bm(^^l-x8PDn&mI&sVZ>OFl zNxGUmmHJsN6Mftjf-__{%3CKaldCRq&eH2KC%bm+2Yn0j44qm>xbssI@uBHkg02(( z*@hVAI3LskhWg7Pkx?PD1@?DtHl9R53^k?0$?>#MrVdHOe45awa~LTuSO*J)kF^(B z&TwyAxC4aPMNon!Ev8sh30mh8dWU5O!z^9D(U4mxOgB#Wiu=&yMtkPp3xD#0oUE#x zxy4r3mS7!!;?Qf0TW4PLywzfyP}F?1otjEj9#99*=NxK++PCZ}>zXl3BDraLZsjJN zE;8c|d?$>oiDaigS#z)|PgU?V`U0AJv)Z5$G~*APMPql=3M@aMOD$`oJ6vNTt09k3 za~p)lqLUicOtNQ=IUzkN4%K2(ZE8ngsA9gkwq7&LHcF&09#C3$ub`^sMZ68+k0VKg z(w(_U0_3ACTew;e)aQW@(x5|8POrZ9r48$nmD~4>v3xOU{u=?=HcwgE&BVEMb$-xw zYag|_tm5woCJ1^N6CvZWioY?>V?S<VSR^FRt+KY%DXr6kKf{;oU#+D(7HY_r(Nt{e z6)@Rt1v@Q6&_st@)8a7x7UbaTyE~f6GwDau2a)|yHI;XBVM)?<v{gn8Khl8>L2F%s z)&<ILHMeO<-On#qruJJw4|1Sk7d0EYb#eHaN)IjUY_L+tdjzRK#;cF%K0o7mA4m}j zDAp+%V!!?e00960?7eGuB*%3i_}%C5KV-|atDvr~Zjkow*#gl}KR~i~`ZWzuvJAqF zL{$PvcA>IdS=9jC#5o*4WsN+NceJ)Vk}b<)Yiym7WR1?)G9}CZ#lvn=pZpg#?js^@ zys`?-mb~jl9*K?0jOUFTH*VagGcLD!*gdAQ8eAI<KtIzq0|^U)O&##9+!cxzbU0&3 zPyjDID^`89=tdU_GA6;I&|F7w3W%(JJR1X5uEBs5T6$ERpkJDq$f}uBr1_v4&y<x7 zlS9QfY|tx#KWY>msz$M*7SN)`5kn&)VSi2P7K5?%^%L3vb8y5(=y@0sp=m+%N*4v| zLT;X|2$DBp0?DRsEr1VwhZd-D*{THS2%P=B>eh;c`PWb4=B*!rbF%p*VE5c{MjT(@ zX|1E7pgpN=M%GqXBLpctQg1{Cb!cXZf)Qp?D58xPP_lrK#GX%1BSV8Aa0T3u539$; z`e46Mq~Y3N!1>W!y3I*dVuPAXv=afNFSx)=Y-AU}4(s_O!A2&k5|rK=4?v%WmGLIL zz>bBdDN-~bo(l&AfWp5JI>ed8+kwY!E`hrqMWQ1=-bNgI;Bgo*+E$Q|*Bzlqla9BT z$g~=RbbHNFaK+eEA_iab!N>tlQ?=#NBIB+X;^7jblt?9!vKhT14Y?@hHkCv+c%Cyn zxq{OD^}9E3-F<^4${?~O`Ea(Du10YXrxUoh*(jgs^6)f_LO<RK*7IK^sH1!gJUK*e zX-!g7PmHq3C&ayU`Nfx-MbeS&F$azze(C?}6?wkl@gjQXk!kb{NQ}k=^M{!xH1MED z0mM?1JA#vTK?jjDK~!~4aZriv7U=3tb`a4`*VrDMjBLHk$D-Cz9};(KyiV#$?Vq}C zxW;jl#@=61%)Ob;(zO_bJu-n2DpR5oc;X2L&U9O_k@4NJEO60HRgE+1;!F-M71#6r zquE%yiGU`1uLSW1MP!zR<yDoWX;djL&?WI3g)X=S?lg)|CS(mEVj5ah_(bBIS~eaF zyc`jux|NJ%$bF0EvhMYjz!;Njy<wBy0Q9QabUd4?Qyt_A1IY%WGL#gt#NG{-Oe3Ef z&vu7p-~RRjM|dK${g|R5Ehi%xhme-Jn4rUZ27)B@rk*1`dOk4^kFJj0+Tnd{(>Zo^ zP)HlpUCNfh%I+#z+g-Yp<b^P9IRIV@)rjKoy2Ns*=vkKatpO#RhK|YVu;K8<(bZ#& zELV2UlmI32DlX)z8@7tOylP!QM!?PZic4S6DWtm1;@m`X&WLJRVC^PQ_axGClS0dU zCM~6gP*6a-8G#G;s!6>uZw>~vBl9rd<Xr4E!H>QHXgee+RYVf)o(i6g?l2bYn#T>_ zU{2^aCS?StO<+S2^$3kn@Dw59ME-(!H-Iy}qx%4cLD;F_j?rW729Vaa73u&c)4Akg znG)UPvgx8*wk%(^wqX(!*|<YmUDD+4p3FmncRA83{krNj&S&a6S$>rxpyLD~u>QiZ z>oV4uKTzGYiQX!k9{hCLrAUZpXjx-Cx7m#NSAEP!C|mhg&rv$?m;%HXlD=@!eT($u zVZ#I8^3dGvsx)CMoi+7TJ>|eHGVs2SbaX@s<rTlLKAm5S>}PL|SQCT3+EeWL=4&jb z-5yHm^Bq05jWIY${1Pg80w(jC&B_d815Vr{h#K2$1Sf>fk)T9ku6;aP2VqxBUfA)b zMlM`X1%d|Vm_I9wNI{EY*4!e`2$46Z9QdcY4}v~&Phy9R8>57@Z7*O84zRnor@H9? zTojTnA%~ODOmRRJ(EwHA*ba+iCvpISTc8;vps?_P;^%94F<`Mh3BXW#2bd*xx&UyT zW^_daP;A&|$roV~;xwL?39uj$aH9}&mjPo$5WIe(H69Zd=3Cm^$S;9WdO1=p?B)F; zlI_N$?j2LnBnmQa5mRKxiE~De33J5QqXyrbpO#zzX+W00Au5apwxhNw1kV!<uU>}U z^1oVDhpi3`j$;GBt6Ky0J>TuOPR$`@Wiij$9bZE`TA!v~jmtuGZX-{qPqVW6X>{%s zpcQwROp+mKO$|gcjE(_azi|i@1<_#T6@3FA71L!C9o}2J7r@SCcXIIDs3s>V#wO0$ z(Zx0&rkP7o3m0aarugLe_?Z3A;jaPy18a5a!ZgcRS))D!Qit3uKZ?gBwlbfxe*x-u z+@w*}#Mp;J-<R1+Jm_*zq1~<#t^lt}t((~>vY6Oueq*ER(l|dER{0>S%jt}K2w*4} zesP*lz~xBs_c<X=sa*y_n7+=4<v^7Nxt(E|0uj)lkGUAIb>s`<E!};x7;8xYl&l)Q z@%XuU3}~dy8>$>>d;h)u*~Sk`UfX7yn`$nZUn2(5Bdu*y+B>p(XXGbMOpDj)aGifb zBy2I<nZ}QN!0&5Q^rSN&y_mId&-rOQiO-Ie9>j{mmWdRG-Rl?UDgQbe6UGnxU*9rg z-PT-8k~L6;2aA@}!)z!U;X&3?vLZ|Ujp!gLxzdIY+Si0uS`nZ<@D6co%H&g$&PGBc z(MLJxSy&{>P$XRehlK=Ky~fB~A{GfF@uUd#nfi+0(Wv+^*EFv=De6v8Cu!WJN-ruy z@t7Nspo$<r!(zV}4O)`RrB`>{!`mV2>L511PsKs*wjHtZBazc#L(@`mi%4X$vzMd( za5gXoM1C!|6O59g4$T$^a+>0=Zl1T678v=*Wb!<|sEHqnu|o@v<#{;3k2kWE(1)0v zRV&q`m^J2?S_=wTL%wj<dd24z(ob1ik2#M0s80KXYLK<u!?@MNH}rRdkDo+-a9SK| zy{SIqomGg7P9YN>SB|8wSj4K=m4$%qx8j6c#g*<jR9$h`z$)zeMpwIOTQ&E*hH1uj zs#Ja@4L(i24_hhHRFkb14VFA9cs^dooh)c5W^r?7v4L^0Ppf}|=|zkvB@WE((b$2d z`!sBwvGPRB@or`*!&)A(Xkgw57gJZjA$m6EnY5_^=d7SdU#_1JVG>64ekc))4MMu! zwCvL8W5v`Q5d?=ba&(0wR`#ZjZd1UOyGZCQ=9H$jIYwt|tWoA^bU2!S8hmWwVrM1# zNNcJZ0JtR@*lwI1sF6*&I!tlB2ayd7o2RaeQ|FTU`mJBNvkpuPf)kUzP9Nk)dMbd) z+T19OgZ{byg3yqTP60;h=Kpn=Xsn49GV<SNz{eK>tHpGo4S?}PhjWI?dnYv-PDuW7 zheRrk#t~gZt0N-~F)4)DBFIQ4O$Crp7TF{}(m5cLxc&UBNw_2_ZiK9pwRMYe#SVPn z`0R)nnK&lcdfUdsKc8)<Y`LukrsF<ipf`{=2)Pejo6Ce&d5RgQ#R~Z(?5D9}hTtu- zXwG6ZKrni|e>^nqJ_jCqH2OvI;B)A7nhXE`F5cT9j4`SoH(@*bRzXa8;X4OJxsV}l z<gx(Hnjhv3r}~TAXU~B%ItI5u36E7U{J|M^USkB1c-kg8DRoo9Z?7|mL#8jxL$oOJ zJLjZtq=-l><3zjW)l1r6@l33Twq^;>b<Wk~*kR5);A7iXhY($e+xS@MMo9nSk77!` zPOH74sR}cZ?gl>F0b%RP-qPl;?ifNy3P*oAeb)DUY?;V&+vO&@bv&j`Coe`E;&wvr zuWk*WTem;Y==AD#<%z!+bAom2Bn7pBb8F^3NePKN^{7}VDG%}_xfg`_Hku-p_1$W^ ziHbwQ4Jm5UxR;Qe!y~<a=7!t|1xw*3LLC9GshMavN+$)LzenKAPwg9R-NfI!bx}_E zK#!S=*<!eH5+-AAsk^Rt!x*$U895Z-(x(Bg6UpaXfKdXMR3rEW?S8^q(z3^|$KPCo z7(Wler_k!-B|lFgxNx9#udySipimm_)-B5XzI$&Iqay8K(1Y9gNio6rW8?y&7O%3- zCRApJ>LSFe=^g0vhN){)Hegwp+y(gp3mF<=*b$7?CBpT<3-X2yglEPw3WdnFQB1cT z4k#gih)XmTJf42@H=q64*Peat_k~v~@-x}f&;7}>&wu{u@BIA-|M;Ij{4c-F;Us?Y z-VZ<e@gIKh{$GCh!$0}xyMLw2{^f6f_@&=<2c3+@fK!ZG2(sMVfP>?wqp&sZ@xcb> zIyv_R`8!TH){JKYcd}UAg%PEh6)P3><6AID5KPt``kVmW-aO4PVuUVzGvLi-k~&x# z$B~d^93njwCt8#JVUSuO(Oc+S9E2ys#2u4IlnDjHh3a5d`ADY{L0_(FHU)=fn`fo$ z7G1UdFmjoVA!Z_(Q2@9l<5;r8*}8UBBicB`jb~?aF^borTY@UMyb`@GDfp0Koz54t zwcWwch7e^<p)B0l00to5XGV|ITBZrUUg^|0PAEMt!r|&jxu9*YJ6V&c4<T{YGU0wO zJ<Z@oIJCfX>4}}2%qK6vaEIblK%VWRq8Qn`q7)*7Nx!&O!aF;~N5x4ms8;G*I%moi zwfR#3BXbAYJ(PEl=IJ|^3p2s^mC1Y-hm1vj9ft|`Vc_2cDdZr*`rV=!)I@GL>A7kP zgO5^}&DjVe7$`uDC)HS8C{9F|PH$vMbyGVh8qy<Xkd7vnI7!W|$TjtUSH|C~nZD!S zFGaHp<BB>#`^-A(mZ|8~Q);aCkMU@D3on=j4#45Dj5RrZa${Zc{eb>m%eL^3?W|{S z3{EazH77|U^f67*Xd0sGse0Hn5RM4%Oy%R3*dVvG`X{|s5?l>2NL*)KM6MeFmyQxH z6VXIg!K5m7c56)^dtURhn){{eN`~}_Q3Mkqry8;*6oD<g*5nX84r4*cI2#pW*G|iE z2T!UYCKZCe##IT7P=2B_?XbP2GZ&qJ%P4UcSiRU)n6q|uBwuWiUFY;Vi|c8;4n-Q0 z*B1x9>qZ3@%Q4-%f^!v?F02i7M?%9l9dxhhz6GY?G}L0;I~v_*qKRhVaQ&N<>ew{V zQA{0;?2uT@n9!CZN?E3FPtJ;3>n*pZc8l9;ElGUWP6l!7iHNI#^*wRq!hvD3173;v zwyVY9oRU{Do=6B=QcQdsT$VgE;@C1IRHz~RIiJ?{`fi#d<6_L;W};!k_AV!u2g<d< zFogwhbQD19A3nUf{^^Gg!@MPNG>^oxgk(7oreS?DsV40fv$H%8kf|v`QR|K!#ji~- zfM7l>kIPY8L~ZU+^t}7|u$WGZJMfw4%XzAS^Hc-psRkBGHGpAg=vs5_{w<EIw?!MT z1-WL>P80t6)d~OcnQ|~a;J@E206+Q1&Em0Y{eF2|3?D$>@LB<IoKO1)*U0`(y(#OP z>RqKaruKbl(T8epPdAiL^S2({=3h6!bp_&xy04K0@ZUsKOnd6hr_18#-k!^(aAEqi zVNc@LOFsv!DI``^pxePNDRG?Q7EaV9;jrF8sP0I1`TEKF5Q)H!vC%F^?@+c?g>MYh z?mM+}+h2zeF|bYw+H|SDHL9nul`PXiEU>#OohBnM*rP^1v0Fk28riZIpt#)`MTAjV zo~LO!r=91t^K7)U?@@{ya4V=LwZ7Aet?v^Rjy2sD5l^+K$fG`5-G&=9(M%V@Emu>z zS(*k#D7jv007er@HyWvgG`AxNdL3#a6T>t&A+6YogUI33vFiy58={6G98QA^cIjVt zfWGBZWPHIp2)5VHrx<pF#RX{{V|salcY&IBhLNTg0<LGb<hAVT0o6zKKB7sb4_)e( z)OF6QB<Lbmkg*eSQL8~8-kvN*A@xcVRx64QVmeUvxWb@LYU`Q7r0~v&jS>Y`=8b6| zMOF#fl)1(>b#A&iKW)!X+w;@*xssx>e(8`l<x6wYqTTaT4(aS@cmV=}`%4go6DjWs zAMKz#c==TwPmSLm=Hr)_Nt&nMfb^f<!Cx!7)qnri`h(9te6aq8`f~Z#+K1)n-NXEN zX|3J8xb*Vn6*;HP^#^yd^<TMv``X=WxDqy_+7``o+rEM4+K9cuNT_4PDAp~kM%9=R zD0Q8*1Ew7(Y}^y`8V|d23JD6(kFDc;A>YR6r^jVI%ZGRJarCQEayf6|K@ULn0iqLl zl@|C+u9WuOw)ur>1qS3K=kV}JSW-YGNFT?SH>3aTW-U{_rLBGQk!hn<@eLHIvHq;v zHOq528&1nh9F6YmOJa3PJ?~5Ng3Vd9=A~Hx{tFiIm&si!T&|9Xf|MyHI%<Yu6q_7S zn7KJajA7aHRSU4sErQ8eqN9ugNjsh@Qh|b&UiRuMR}uiNG%lE-9``aR?d-$)pgQW2 z-V#HHxb4K?&^)kmWCsQs8$Hhd{cPj><ty=lLJ-!EGtOvd%decVxB2v>DY2n*+Bgbd zuecNkodPHBYY&{3+WMyOQeYn7E8i3O_niK8Sw~T?x=?gS{Wofbkm{z{E}0efPTnUk z>&_WE(JJj|C%D_R81rn+$3w4*X4wWH+UTWaNcxl05e%iQ*+hYf2k7IdB6`q$ArS48 zu{UO_vU<iBIBnWX6f*dQH99Jeu1ixU5<6Zt)V`}N>qrSb&Fa~BTuqEz^;O7qI;p3_ z6RWFMx2wtU1e-y#Amoaib`*aNZ0kk=nm$#P8wHlMr3<z+n_go}=ETo9xG(Uly9f-p zDxtDm=OyGEfjXiO2VB5r_Nxk6d&T+U^$BviW}i_5Y}I&i>(FR-E4ZBK_Arlx=030) zcGZV*tKo|=?}GV9&B<XXx+kXZ9ZxEC#W)0Qo=jYU*CYT^58t=d?R31nl=%i<PsHJ? z`_iWfMv;=a8+1FbEZPo|-JqG5e$F9x3y8Fw@(JezFVeCXT*@x;gIs6v2wkrl($}79 zjek&yx3G4LxT-VS2KhT}P2a)?jo+N$>Uu9lb5EMdESBvQ6uG8KAVUIEDnLv~pQ%_N zg->u*D<9K!6VQE9&1y>PT!0k-UBV%6^fagKVFsygxQYRe=PL>#LYpD6UG>remJ$RK z)Px%&=PnKeVhGlt(Wh96ZSif1%1*D<_x0SI&%S|aGF~4>+UM-rxc9VMU^OWBbiz@D z3nrW9*kDLiI!-I4NpCGOSZ9<{dR*jF-D1C*P-;1+c3Y=erHnLm`5@~zs#Z?xVz{>? z*1@9xs=n|Flvwd5i9YDg8?s^c^Rw33v(Np}NALX(r(ONv2jBkaJ3s#LAHVa#`@j3_ z>tFid2j6=7y+3>Q-hca-AARAY_r73nHLBdR|M87yU;BM3tv)~f+@E~(Phb7$8(;Y7 z+uwZphoApn@BOClm{JV~_rxjX<xMq5m#m_Ib3dQd#lXyb3_l+jv_mgIckRh#X8!)_ zWEZ><*>)uQM2<qT2$Li0MofP7N>mfMyI_~X!G;TwwJG5cS&aj&@`Fr!LyT?xdteYI z?2o~2*raaoHnOyT^qTHUQ_A%8qNxgSjwo>Ims3bZ?;{zelt2vAZ+K_%;Z6hV1P$Y+ zg7s9e04btKJfBUE6dOe+Jsikjd*=5`5qO6AGzdr~xENI!@;TO|9A=AaPJF>)Uf>1_ z(1SpDrFUjEjC;3Mmr#h%1m>-ggbY3iJ*K)T^QgZvAwNE+@n(Db*ys#5ewJ}b7CzbF zbB={s_{GP;6;ea7SAxW7=0uVWX;s)$#3HKE4b!BtF6#%J6*tLXcua*z0WH}I+fX9s zL^JJm2p3pqM@|r;`<Qq2>F2-x(Qo~uqWmBH;1BRf@~D35Hi?qicGZC59eu=wlvpBN zq#^TpUK18y^l2KlqmeLZVeWkn9<5;F=7UAHkPR2Zp~g1$C{`b2%xUbw<x0}TpkX*4 z;jV*;1Zqvy$`MFTA47yrkq~e9gu@2&{bffS7GcZ464DtaX|nE*&mgo_M&}iEAKbA3 z*in?U=;!K7(5_;96ySzszFQ8}S)%Px!5X_BtK$VEythVKh3XEt99jJW-ynk~`8r1` zot;BJiP?@vyw|eCLYNq!4l@vP4bR~gpS+;V%{Jqv`i^L=6NJFyw8Bjq-MkKdqchrK zry5+Y{jnmFAciPRA;e+ZRvtOpzA9`AnyRbpsKDx=;N`tYZkZS^TMvrtUft?)vN|`J zwvd%ZV{MUy-8(eoyh9M{$7lxh{tt`5;{kBX9lMPafYWO)rQ(0aQye~5%J82e7B|ZU z-Bj+2pmNbiiMGMkd|Y<5-J|2*Oohn?(l1SKgW~d>@lKvtNqd|`<&d&`=^hyJlL=bv z9VOtC%L2)ajq|(iM!ev2aOh>*$VwTXn%W(se__4@x-jlsYs|)|T$cQ`9v2q`cKkpA z7Q6>0Wi?T>T#VRB8Fi(TsUf3Q)|~<*k9rNhsXs1Wj5@=@Ku!>AFi3LSG_h4~Q*7ss z#*&V~fMiw;^rFLH27!%IYU@Pr+Rwi7-KSsrrUd@e_rLtncmL|?kKTX!xxf4H%YXgs zbANAd&R_bbVxgXX@w<wb`rx0w^X$w2px7zktG@bYPyg}%LYB%|XgyB5%#Euou<g=v zGG_@VC-EBAeXalcEFacw<ARuEf<nknMEY*UtCeA_p>()RkAaF)J`NYKx|&^DR%f?_ zh4yUiWdbcQ1W_^#;-Y~y;oc1_$Wd7rdYtO&ctFYd7GoDn&&N_s+1ZoE^vb}@m4YlF zOzc2_hT_O^Z%b%-^yff7<9u)_8@!G?J;QmLPfM_%L7SeU|4KJH1ju`9#5(%DX~~g) zb|e1R@Hjr$vC<ZvW_%#I(7oBN<hb5_SG2`)g><*#hpK3tB@q<@c(;Rw6Mm51fQiB) zb&G)qKR?$H&GI?`#5caR=r0kebp*Z&`G>`wpm#k&@g2pv_NR6ZPDKHbMJd>SGH|;& zLXeEhA+3ZpQkkpmf!ZX7;1zj`h3Rw~GiS6^Eyf0E83r-P`D6663GzD!KS)mRXd`e& zXWBQBwl@^FEw_fcMT)wl^7EbA!rU};PjmD)-ssH!PYlz12X4!bCo*D72)QT5-7VYL zwl*yG_NQ<q)Fr?#RLB=ZknAYVGc7+9817i_%D_ZUYR}kK;DIOUyNE|MDelw9DAwA~ zCLRo(4GWxnfj@S1d3xe+F+~r)J^zI$DmO^a`Q;aal@P}}W59~<?^zc`_L{H=J$e<b zK{p><?$`CJm#{t%ma?cX|2+O$zKS(BNIu<Drx&u22gpEnvK%ZYr|3L|<!<EYDhoz7 zXjldosx`oRuRKPhBgfIyV_K7ZsywbI1=_=~sTn^<WYD8)+7C3&o@q*aqM=V<=%!|N z*<O#-wi#AOHH%kp%i0~>*Fg?X4JyFk*id$grn0Me8#Ui$lx{Cw+^pze3ZF=3!fF4Y zfbdT6xr;yB#!NGkxldPsh*|vi*7COV^Ofy|!v6Te5(oiVQTh|zY;bAIhbBde?yEbk zZdXUe<OT$ciKQ9@#%Kb0U~pNtbR*jw7@}jk4tZ)1+NhO}$ex4UD(kKI(QV{_AS}Wd zGZ>gb$fyz)C=hu_;kp}k$ndc047$3?O<CfMIH6Bjuc`bv-PCIhO+5jurCUn8GBoek z_M`hmt~$#d{r(l{{;ls+;Ehd@52^{sD8XdB?xVC7?!PG9LNnhYBl95FgI;&ieWxl% z?N+C?L|Fm3NBi2RXU0T?twYqSAP$fs?#oLnJNR+)pkPn_K}BkB%r?hKs4z#DPVCXa zzrgd7EM+*`c)WvG!Vbv(yoLG!P?t@?3>QL&Rn>5}B3g)5C_wL^!cZ@u=7S7<Y|bx7 zSM2SnRWttdyKvo7ZFk6oT{VQ`5}*Cd8GHl9q`Hr*M@1930;3q7q!{*as(RW#00RKF zdctietN@;;1855@n$+H>r~*~I6%%f28Si?pEJ5AkI0p*@vb#j9p+ltP3=Bj=RL`nQ zSbl=e3-ANHO$hXRMXnICu(Q0ge&=2<`%KY#jNU4UYz12@k?38~l1T+_6*b}CayMHC zl>mjpQJ6DOy@4<v>=-z7COO9F6g9I^31WG5Sg76)Icmu#ap$%-tIC*3DdfTUkhrZ1 zh)r-wb*-2~EdELt8Sq>oKt8CBj4U^6-)AeXet|n*P2w@t38Lfq3USnbdyKMqLux<o z0_8B5=Ck7ptYfPr(zdfAl*%L{oh@)+4F|dnt0%Q7%EN=`?0MoN&1#nl(8lQjh<Gv` z<Ot4?w^KkO%tLQJ`YfxQdzigme+(@c*jwa*5LH10ugz*5qUzMqDTS>hz=C2g)4C)0 zg1CB(3074BEyDuiDENtieiG66*32T<qMb~%!s`&P-OEHZz-&=*k;}oUIHt~h<<5QO z&VA+ZTDw404>&e{*>`F)`IBD7G|oDi(u%Hi|K7&Nt?RefA8u@Z_O_a{4F;khn{P}{ zhL9bt1s81dW28vD{PKu@aK-)sI3Ant`7TA`z&~z-CX~W&_?L$1`i)m0%qG?Z-o&}L zmih~yN$S6fmQ|1hTYp&*QXpA8auslS|F{~-gUozo5(ayNKJQYnV6TC#(!cT1;d0BJ znH+hSO86Sc!?8ME%etweF0<mX24RmF+17u%W#$pyb}Q#It%ZEyOM}gtxaD|mDW!K< z(wxSLfi_!+rJ}<cPB`qaGY2ru9X8g@RwF4hC9tAvtd=VF{%5gRy5B{XOP2*w)I8i5 zxMaGg&Rh`9>9o0aa^V6&01CS%)4PZU>)a}g8gE<_DN&uw=%6+5<RY53ZE?wM%!Mmo z=sls<&v|XuDWFVL!B9`Xr38(MlozBWMqGS?3lEq98pj#n#Ht~}mc$!V((Yr}KT3`Q z7sXoZAMn1$eTIk{g3!lIA7qK?ge$8b*22`z0(2}upEEACv%aggxkiftId!o{v3)OR z=KPH9d09Y?O7yV+&QFR%#mlVUc$KhzL-8)N+BwP_c16n^V8&FcskG%t7Ab9+nT6U$ zo@L>Zx21U(*kQuGA|KDMJsT^WzwLAeCT^2_d@#RqG1xC`-G~%+x>+1g4L8-nEp|6| zL-yY|DEg1UWdt`NOQP73q)mZ_--d1RwvFc5?RJ+eUxols*s+1yN6BNpN(;}FOm{9# z0H%6;ZZ)?_Y{7liH5CQRXWlt`A<o)uuK&s=ab*uFF8Ko7ogTEFNVFN3#|wt$hd4DW z5+z>lq%8zNGoo7z;F=~IkrRrz?pJSAhlkY&+jDm-pGF=7o!aAK1X29+wpBo;IK_xp zPn_&t1Hpt?ndxK>x#vyif<QD|qVbBLw_<QdvGrH9Kt2W=r<iQa&eIeLCK-~Gbj=0G zNxHI2PBgeoi8u$CY%slQu}Ll5RAh$XEEjGqoU#+C9AZ=7$Jo@XnXPZ$3MiR(5#Tr& zV+#hG#ovcP(O_)1$#lUT+HJCTk|VqAn?NK6hUFF-mn<0!>zX=hgJHo-Z&>guI;^li zEE?+%9X@4W@WSjeg%;OU%}3g1?A0E0E*%ZfOSQvp<!qEXs<IvMLB>=yl{7y<7YmA* zj&9*c`DDPOMq33GF0r5$y_lsdm~kI4hJEh-nwsS0h{~=4ljn*5DE7_P0#S^-<RkB{ z(E6qi7!cFcZe1wn3kXbIG~1MJwE9bUR=H;^9ih6kJy#>gX?oFhb#<OLThQu1**d^Z zuAfhf{VGTu91Re@z=0MTNddG5QAiB}D8(Q&m<OFWWCkHrPQ8&q>;OmwLvi(?!pEX% zXd0}4up6~^w>%s(zPaUAnjt{7o&yF{F$o0;1H5vEAUF%@)boeG6Vo4x>6>EnW<Q)e zd$gwJ%o|i3=6r^2WGZc43HH$<wU*~fOA>e6IStR>)2_WBtkN8sna}o%)?&N~TnCTK zI%Ff7YGA0Sg-Z{LX<j;Pg@IOT2FwuobBNvlZ1r7rybb!3+2L;V{*0aalyO)*_L5P& z5URM)1bwrdMs`_^G=D5IXc~m5XQSO2SjpB?j0}k0T1EetI^fb(PIWMrDF%T7#Sx#; zH1n>1ZJ2^;w_W)WJ@w90AJ(!e>#6^*Wm}cWY~wegt`%V;ZMPM*522wYBZE1$afM-s z`Zny_ae{H>rju@uv^{MeJK3`)qi6UMqUaq6=v?XsrI*0oV9;Fem>fb{$Jf+;0v*qZ z{$r94%2};{<#2)3Kgdp5{{ly_;jkEriut6aZ?imW7#eg&*4QnyY||C^pf^1at9|_^ zScc-?cGwIy3kqm9YCsDl!jYCZ8_I?JmM*;Q03Y7QgmapZHGPj!7%rcrUklXj!oPSz zL-6Iu)y!;dk3u(^@y3u*FRTL*{nDQ`3RVOrSgUqX0}<5z>_#dykmpG0g%5(*(8OTG z>qhvVHbblrMq3K|-OR82(uY|5WDup32ZBF2x6U9xnGX-QiaOw@2WPExQsmx6Ugp7r zSX`<oCM>nmzBK5t^pcx;J_<B<NV*W{v{g@wF4TTeIen?$0fj}@%}`k`u^@lTrB@p0 z4i&*@+2Ca0%pBjTuFt?I+B!aGma@=EL&tbsIO*K1k^TaAepz~D9?pxxQHr)mt<+AL zUsI~HTGl|#C2f3t7d8DGf@Tv;lqzQ1D1_7;V=o8U$Z9KfONdRV#1yLyXF+4NIOqhr zIA}BvcgVcgMICy#7y-0}VQ78EsE{H4Wl^%rcB`^eeib{N;~m=y$&O<{@F^0rkpvBK zmMlfIQVlgZ)810@I699;e{NxSZee$pX!OPjnfzsNF%iygc2!oj<|29t{6J22deE)a z_@~+#Lmv;-?mMKT=XWu25!Ox#i?{*xU(mMhzOx0Tx5{m)6hvJfluOuzgDBxqQl6$m zd+fdw+<fa|O3{;O+UR)H$pGSjO2zYo0p2|-P8I`7b}3s90i&vOm#$`(fw85&r}Kf^ z0<OKpX$0_C&=pi*9~ug1s3R68+NoOr;AmTnscvuQiU>{b8_OBr>63Yg+t)s;w%x<~ z5ANMxe*oLgb@Qyo0p|)YK~!NBASmRf*URE{#HgR{!(5UL&di%D+uIQgGv*7hy1`(r zMe0`T7%%w6-n6ydQ9n=cl3$JK*YPVTKfwHS^ci35R?}&9*xJ@cr_jlbdw1?%+q|`T z4XiWa$GyAjcQ>W+>;9x1Q2Mw^N-3y+2&_+CBMZ?o;z9L?i8zkQ@Yowl3nPZo`jw&` zkQRwZ%R+)5At(yU$P3gRlLnMyljgu+lIyeyi)2%RvP+pGNs9VEIHD$(nOk<hE+B_; z6!Oa{g}fy|F7~)gHdW?S%dP;@VaTm~(j<dzQGljsv;nJ88kXaGYCrD{tE22ff@Z+u zR&fR++gMDm<w{iTh{t8ovg*9eCP8>?Dpmi{XCUUtqVrJYF-FO%{}x*Lcr!_i7)3(F zx!!=Ruc~V}UZs?sKX)-JY;gT>D)T(UNm4*yiF@)+!%cUb#*$Hi6I8iLjH)&&5Qzs6 zT^X$*Ij%S`V}_K7^Mv-7sp1|V^wCr^#MhR$=ZUjz6ZFu{JjVWAflSl}gR3pMcZvrm zZzTMQ{$^6m>p~4t(NOrzf6lWdmh{P2e#8PLG^}TJMqj4$tJe8d>-?&9rmI$ac`4iC z2<zKD3J-_QyBG%sL!n!X8~Cb*tvW+5VpSa;l)14!3x-oqbIJ^{3&$4460N~$_^@m0 zz)WBBViHoB%Nr1oLA{^+JR9<WD*L{Up0b(OkLs&5^jDTBmbS@cSSh5WsO$=+5T%+z zyAMx9$HwivI4=6Qv@$<CXT8@MDT+Oy0%}72M-cI{KO1TmP7511@Y`xb>_c)^Ql)OG ztxksp1yy}BuMgC36E#YELIVY5y`az5#iJ_LJO)t);v}|P_`$c|(-FS%VUJ+7gT>IJ z9by2eqp_vVVIA)qzImYw@m-XSF!SU%WOCj(B(4hVl>AidPaQ8!sBYLXKQl7~%<y7$ z1S4_duLwB21xSs-Qg~q%Q91^%esFS7U{FLK=rRO4J&DAH!Vbts_@=sWVCtV9*&0C* zP>U=u2wn=Wcxx#V925%{1K?d3Zr3Z`)b1JDT&TV4Cue9;6ppr8)*HGYhEHv1(JvZp zkJ5{wWIx4pBQEvi50BXhuuGxNtvx+`JQo0KFf4-?BOj60-#|D|V80B8c71q<V}-Cd zZCk;gytALWA+S9M@&+IFkhgPRI2&99p2T&k!*vS#F&H}{8mCB&)96;Ti%Z$^b~m|5 zAqj9Bq0n;rRkNbap-VuD+lD?M&@H2BWkgPaw$rySB62=D%9nhi@O(pm?qucQ_||A6 zAhkB??5EMiDiM<QZhjX&s((6}`j1~|P*PH9uh(vjOm(toDUay*Jy0Ds*Yg|jbaLcA z?csD-%P-=rI+(A~*$!KfM1kNarMN*-?QsBeJ}qq;nPNy8u<1TnJ1q_r|1d&!9K!CP zI11vU6n!Dc(GwUIN6u`~=nx{%0j!-kT30&R<7JB5b1NE46%Vl4q1zN3pNWMKOJy64 z5iS#lXh+>_Mrs;&)m6RT0p^@?V=2Z2b9bQO4{!vCKw`avCOM8zKCCO?s<6Cbx(5LG zoTAQ*CncP&yTx#dJw&J>81>w3t2xJSp-z-lRdODgLW$qWr(qHiW<o;^@9IY(D%nA~ z_!gJlgpCF3ra!cu)$wK^f2OQGc}@!>hByN})Mx~sIO1T<4IWVBrpFL#3JvZj1zs)4 z{8^zkVGY^}MxwY2EK(DoRdh=k%>&5#E&U^n!#fb0?27vL(xvD^_*|cRs=Tuu^u{w) zY`8+rf+X#S<!{i1vP9K0US244Sfv?d2daK90(e-svH<SD6UU$0A(h<;f4R3OxPaIi zGlC@S%;IB>jQ~KM95O)qF&;kNFck`3KQYK&&pA~h{f0zKER-f*c;&5XN|?sChEcJf z+uz!gd3qu&(qmHxii2|n9tNg#6{ZbwNS*fM4Zx!qFKpvA(Z*}s#$9Jj@ivk$h6ftH zD7913Le8#gF6(O^<r4|58%DDAH3iIufCg(7hP#p~u)M9Yg@zievqRxO?aboT#R@_B z4yu}?VAXsX<U`oI^1d}5zyZXOG3(;jXK=$47ka;Sms6n#VmB?%avnM9T-bOnY&^TL zkup@NKi1aP6h!xONHw~q*a~os$H<2>5XH@E=u)krmMlI7vft%gtq_bF5rLycs1pUx zyf+HSYOyB6ZYm~5hqgeKEMzw7WQQ{{Ea@N2M(QG<PJ9%7=~TE~T^0111E;2_aR^D} zb|HxPF;|gM3fV@PC6_q-c3sUTeYGvrbznoa;QC5$;74`CxiwN$QlUE+ciCZjr^pc( zr%Md<vRTn9UClL+pri(-HvM6Us<fNLf?C){ort>{vRL@`b)+M<kbEmuZkrZ{U0nX` z*81|5i)(i3ZhNb0Zuj1LaAW=9>-QeqxwiRm^RxHYZ$1Qk|7@Vpe)#a_`lla0jB2+S z%1M(G0?VX2!i-w$5T&x+5~C%E;40H@IDv&D)&()3(FjwVCzL5{Zx2MPz`y#R%s@3_ zge9y?s;^a~`2Udo3vTmzr`(}8TR!3>fsEjN_~71WHf)tOx6;L|^{}P>TT9F%h23?L zDz)d-qW6~bPAytqtM))O%ya2vty?3-%BaaW?e#J{1bP!Vi=y;)(59rK+?-?fqJeZD zG9W9UMYgKVj=X|2=N3~UMJ2{y<^j}4&6`4y!Lhwp?$0KCtUB#?vAm7=>@@}lWKGiM z(ZKOxo)p6ZwJxNbm?0hYnzpq?0B<6$7bwRKh%EB{iAzVdloE8{oH-6S9t5i2hxy=W zIDFSDNnVU}Ov3t(LD46Aom#ld9#lv5!+tfK9gbQ(i4g!d%4IDiw|&T=*ozHBKRlp2 zV6AqTAGfb`#2&J;L^<FtURQ&A@u}!nK>_^toxF{M@J8oAgS>|a#|mE)y=^CYQ{08Z z;F~DAwtsKH!Oa`ZX8ICPRHW>MfDlE9iY&^*K=JnhXHV?9D!Ar=TfLbNXGJEid^}hq zWLHxiL}Wf-XyQvt^|#>g^U6GzQ@IOEb|#P9y)bAi3aAN~>c4<D;Jt_kSF@l<dr+!I z7C@(lWe?86o31XyHe>RSXYL`;%RZr+_(ZoAldoHhrI1H}vWufD{zE$;hAR9=P~c2E z=}~b~3~nGPI>X53Y$SE0a|9y?Qw=8p&TQqVQ2PrVup+prh@-m|xyi(agyL)PU%d%G z_D-9-c}SDcIKM^qlWJz6{S*U84awt15vagpH<$s=&tQXTK|~ZQ=Fl5f6>$@OQ1WWn z0c-0dfFSra!U&;*X4N32vcFa%I`-7EaF068)w~6Qhx#XoQ|(9`f(WFeW?h1AOv?Ro zln>p#`7<OCQxkW@BM*iT>k*&V`Pif*;3O2<wnIkdvU`O%`?1=A3wQKvMe(~i!Rh+q zEg^8gtN33;jtFTmw;uPV^GRrX0g~X<O5>awfkH5sCUDOev*5^$Z*SGc3)$UGw^D3P zUv%PI#|dtu`$!~^ZCx7-)cwF-Q`N1()`pSLHDlzNpTot&mTQHL&(#h&w^COys-fpd zUvEIpbV@5x+RINqgoqj>SrJN<J``@;93Y)vzkBo6-8W*x(ksvCfP1a&PsyD@GRmx{ z<+n5@aRZV(A>KHpyOJe9&M_OkG;Dj-ecBY@Uu^qunIz2?W7f3V3mT)HVf1xwk{=04 zn{%Br_Yfqu54fNqxKzlpesY0PEr}kkMnQ=}GW%Cx2+cs#g|yANqLR8X!zN`t^G?so zj7PG3jpOE{=VOjKntArth?2Bm>Mo3gMB3C6kfuV9_e+QQxO*@?94;+&%w(KK)OLE( zud2zQmu+>s-EBeSSXYNSV^*6$b)2S|gxnuGoQeT1OVG#$iR}VdNLlS=NJW-B1EV$S z(tX*Zms`5w?O;9Z2^(3BU_S;W&blUxVG)hhMWa)@C!#{=hOMCso^@-`*rPLxZ4Gjr zhShdg+U$#tJzj}b2W?kvJ}=GREUB3^Z<a;p0aTvz0fA9(4TA(_sxeH(<b9y~duv)6 zV^D+IU3Zi1nbpIiXOxYHpe^F?yQobVRlHC3%3{cgvSERD@a>M%9!lyUt9D(N`y;D% zE4alD=1eN5rXW~0541h?@WkD`WLLez-tK12cDnIhY-R$H%Jq*fe+^JrATE`0m8k@r ze01Uf2osDwp&onL@)bWG_65QnbspTL<70w>gUVX30Y!bZ^d%y0Pz@a9wJmIgn7YjA zT&#B6OcAG$#aZ)7QCNfkW6Yp(PWsAe;XsoHWE=BGleG>e7DGKp*cg+(P6P(24QPvo zTT#-<7eOpSawnzgA=!0z{#iR-fI`tA79dehlD$pPA0HG$bs^J&ZQfIQ9cikh;QJJK znjU6PDD?Kbw65=}@f4@#^s=SMaq?X%e45c9_o{8OlR}e}Htw1hSR%uu(d|wLw9i`9 ze#@pzQ;(|q32vxcfg(O1-iJit*G7Yl{BiM#PR4I}{C!2ZAigP-33ZM>KshTVcOXNN zM+vLQ@~ua|sw<;~FAtQ}BAOJKvfyCI!)^JLTupRebup6rNk<c7@VT^(8@f}F<|`b9 zC=l{4;-k-(h)>-y!&)6`V9`kc^QZ;yTwlu+%p81<Th5&8F_8S(n8c+s^o)U0iZ~sL zV9aVx0MSa*nhVexBa2f?fA+j9#TQla64&u8hUwtAh!X}3Z<~|AnH3nkHX*KAOkGMx zv1cp=C6-xXJcksSwBhX453ovB|EO`=K)*ODFr82z-E^C6_fS&|sD7cQ+7c_Z26cUm zbGPLs*p{>kt&1(EQ@^4<!_lNQUa*&4)Oc}fwQj6*gKer#^jdpT?G|L;Gb%uXDcIHy z|Invqhl5|Rjjk0ek?8|_X@S=cUYpUjnIHlNw`HxEniDishTt;*XAVxYhSje@V%3@i z0IGccxGK^A<uIRiF$;NdoMXyKz*27asL<@zqfvE)%^XxmP@O7b7EH|23MM6YV3uGg zf+oadiKhy%?SwXxA7C@oJ_mxn()~Y{ukf?n;Y8#IiUel6u!d^9lWGPg8+M*uV^GpF z2`}~7FhM2a%w>apmjU%~D{=#CBHJ_*dCwmW&Wg)mJbJ3F2yIc?6{q%~OP?upT5KC# zKs7OEB306<x$#8rzDBjKzUwu@#NVA&a0sZjCS~!s&?v@Xt=#k2H9Z;lf~<H{eu<eR zbkae*r!J<NQ$#bAtHz5*;JRxHEvTxg6*1h1k1S%9qB}IkH%nHPt^W3`Rm_NKh1@r; zdANCrCdRs(o03C;gJH=%A)Nyl9;+A92b!n;vA1tiq7<4<Gk2|rbtN-fa)R57-iv*H zN;?Yb7(7N!QD<$%N{!0`QlpNS@VY=rWd|rH-FgDE=_y_aB6jdC;+H$@IZ+HqB7WL| z1Si{A?+C-fQ{G#-l8Sm*!9SiaQh(^#Y+*}4hRMv#CW8ktDTWC)-t5c?l4467&{m(` zvQ7S!$ipe7N~gP4OM|A>u%RK_c(|XPbnIu@&>wr-*e;!BE(GR9_a5AO<JR44w;ir! zUY+az=lYFJVUx7rjLt%JqQeiZqxA4O{=A8y62IMw0SL*mh*V^%HQmHuRmCmr+}8Nq z*7)4k_-t8PZ{E72z^jJ(@b<Oq>$e|1Sbsz9$;)p)`Bb}mac$}C)64tXVvhAmibWol z)2RaWY#)Slnw}74xKkdNBkN60rzAX|%+#@qA&L)coEahh)HusEp9^jYD`8MYFgKhn zq8UY^Kw3NoGlqXz1bQFG5eSC(sOLrlk5XxppdIT?;T5;6W;R55hu!r13aj7LrCciA z;3J0)mdr761>jlIg-a4FE_w{;YJCoXq${C78N>X@skOp5Na%5>F2Xj3(wrtl;h-VD zx1no`hO&+h0nYF`<WIZBC4&~oGY@K5>8flL#17&&g^RG28rma0(k4}YU|zTV9{m<{ zj&(gB{o7wR4vCLBK6DMZ?~}1v%PX#E`Q+YgI-Z%8wr?Ckx86Ihc*g`!f_{3Y$4;Wu z#%z}g;tXo~l`6~;3|I{<UwSocn693k{+iPV+bTsX%6`O~c-cYNMLB3{{VsCyejVJ3 z(OSg$!+*UG2KRdo1&EaD>CGGYMo`j!c?I~Uq3KhwFlwvy=pXNs*o!^+!ykaY<^fO? zi51AVCVq6?a&b5nqrTcI={D$lye?>oi%C!H!#nt<9kDoRm^D0h*Y?EUZn@JX$D#us zI5Y9QLPS9Kc?@@`xD`0Jck-Q1wzCWW>BD~p@E@r|gXd+ne-#k7S{Bh-Iy^jtcZ4Nu z2|b7o3~Kct@(5<`9YtRs=h&?=HOgv>{DxZFnf?pqmO7<fE8Rfk-1bHxw8|L12;^fJ zBMI-<=wc1gL`;2tX2drUXMs1^a&J#vn<4X*vA68deUf7Orw55(f<SAl9H~0OW$41_ zL^ZWtcQh?^MNuI)@;cWTVD^CoM?&zn+=3KLm{Gb;P*{W9N#T!o-!*IG9S&z<YJlYE z^H!eR_%0@_(iNHSbGsoXY#ujcq(jpB0JOT)W;bPQ;h$no@m1$%pwhz~<j=vP)v5&q z@&w+Iz@SGR1%ccxuqIFo7lyD)ZD-R(%Tu=Y^qv#zy+BoWAu#tVsZqpSV3-=|!myVN zCJ&ufqJ@MCQV8WjPXpc=W(>TN5zAJ1OrXyA&}jq4<8`WU;F5zCHpj!Z*qye^fM&Bi zENkU?{^V0zuWq02?(RPM)Ob=2X6i!G@Aq5XppxeZ&^m1q@nPaWo0db8)laK#6dG!* z-AOSX=KZ1_xs*8}B2m`o3aS06Q(|FN!kXIo#du?NftX&Kwy!J^K(|_~Lew9isY}^1 zf8Gv3tsj`}YIYH-)^LAsFH?eJsJsF|x;Ct<p6X${m91XQu7s^b-0O|Y`ywKH-zc-u zSlYjvT1AlwD4jf=%E9wpN|82cH9A)a&jdZmzz{ilGfhD#PSG)BQfMt9fnlTnfouYW zUDo>aEd0W*;=77`q(E{w0|snZ5QDsm7T`^1e;R@s%6u_%72_I)yydA#(oR-k<T{4G z(BG{japhc;crHpj7bPwpw!l*ov+yk=w)Z8e!PP_4W+;2=`h$@^A!zANXs~0xwW%@R zL<+p|<Y!+&_bE_(PbV{oqA+)i{ONV>U4K+3GuX@DnAOwjFvA~-@>^U%9pt0!>;Vq3 z&RSTdMPZU4$O^Sr9v&70i26mbGn?8&yZUW1uh-@RsOT@Iux6~On<_ZWYpELDEGN^G z#oLmr5nBiP3ET$#p>s*nS2lty$j3#3b43;ac-imp%lS-WHQ({4Q-Eh>i|`9?90M+7 zJi+-KxS_~%aXd9XBD_i9$_aa_UrvW75V6fvAj1&YEZL0W=JdG<RFqQ0JPw}T6(itM zE5g?v0ImJ(0lto7eR^wg+}~@H{raVtR&`4hQ2cu_-p)46tl)QdfohLRsHXm#^%3{U z;0C^fsmgGHCKM#DD`eXMs~I3MM`{mLM;IJpvD(1C+#u{Xjo|ug<TUW>>Y+(De9@W{ zY<M;XSR03^*_Z+=3$q58JX{oQ1DIsd1CPWa4BdeO3LUn^8id`4Yb)NFvpaxfWBnT) z8PS}`Mqgws>9Wc^;{?)ZN<!#{XBBZ%I;<1-g;ynk>}t$3*Fc@2F?Q8QO<hP10|GMh z&kBWaH|Zs{-3`CRtO03U569)0Qu2kKHs)KmKbwHqMyygylMR%DVy3P5-5^N;Z8R*C z)0M4c)*oKAB^Y9MfBI%vA24gB)<9vXw#DMun0&h7SxHf}`URe*hxsGtt6IeUNj_F} zO%DaJ+lr`+N<cF(%8(o&&=y4=_Tj`=8~~Cj2)(SS&SQ+iImv#bsuUsL7luZ|9HW3; zKap8f@CAblH>%8k#5K(RGq94I5~%$ZQRy(=MZ+cv`aN{}Y_1v|2;hL7FPKfxuvC#? zQvB7tNFxd#;yzv&QsYfoT_v5i&@BkU=q+PH#WuwT<eTw3iYtf1cwaSA^oGQb@WXtL zlYYrUO97)_yk1Ufut!0>2sQ%I0X2C!omDfTn7<nzlR6@c{u*|s3Slc8O9r=^KTLsN z885|(_eCza{OOpK!J##WAgdCN&5fXZt{g1oe`<VBnXWx#f;KDG)ehF_Y2uP9h8-0( zl+kIyHFN>F5!`fv#HmdSY`EIY@RPp#0ZtbhIJ=!_Dgo;*c&NpOE}5=1`O77dSueXa z+5^wolL)oETTP2zwlSUU)*XucT`Ix`(FpZSk@d-j1$qHNq#aAz<S)mndQ^#0V>Rmx zP5vn9x<18`n^m`6{$);B+yThB7_Lm57u;+tbHGG6XwM2?s0oD%1RMo<DMx`wSYfX9 zoO2t3U`_Dd0yFLcgNxH7+r@(pX-j0EQt)#%kJn(Rp?I7l5$R3sg2a=nQP^K{z&22e ziUp%t^7OJU5+`9$=@~GE(Y3eA?GDYT0~6{HUFo>AJR<cB!*$A^sgT0SVVrpkZUD5% z4%EW)!cYr%4D#<ta<bEbeLp)aXn)Zs;wp2AE9gLHK9$6T)aZm%S16y^(LL`#Y74g$ z&&AH6h$f~9*whY4Eh{LcJNLCj;T}(RF-pg71UK>#N|z?paJU<04>0u58SYg*Uimhe zKr1Xtu^bZQ+(#Zx9J2uz0*B%OaT9dRb;E>&E%S*cI~*iwXR>+p|1#mokMffmy`|_7 zg_~go4H_q^TFV9uHSIRwY!qNG1VJPXOqh3Lkr|TE`k)xJZ4LrfT~6{T+8nh#d9dX$ z?4HB>5#2z1?l!@yir4Cn8*(q@G+TAW_G*?=HVq8D+K(IREKvW1%qToOBv8b8bcN0z zH?<%<n(e}NSAuVn@v7xK^rPpZ54V0%4%<$}%b8)}y<=k99nt-=O{2J)EyMBT)W%rS zYw+$al<o1A31V^Fb#J3ufwSVRduib;lEBJF+A)2J1|^<{5b=n9U%bfXBuPG>huV&s z2kM>B*3JKdf#6d1NnVYM0+hPkWm_(=D&q@?E_xLY2Y=^;yQwEe^qK>8zpk&s;fD?# zXSslQ90LCYa^Rd@MLB*5WaZPD6KpV!9rT9q1GG?TAsu)YE<DD54z92uKkNFlarYxU zX3bWD<(){O0H}19I0upy7nxg5s2+++3+)FjZ3F{B`67@Ycy7UxjZnk5H98X~Vf+`E zl#v*<y&wa^9V$j%EuVt1xWRdBUaF+DoWf3fV#qxj0EcaYEp<tM1tQS1qlTZnEFIIA zgqZ^}zFlf)IkFrxP<aIDcm63P%F>%F{&f9Q$M_})agz(F<OST`Eeum`y1^N_Vg|u? zJEhtY@aRgM7?~=<Jy;+2(R&O$(c8PJmJ}pJMj7-Kzf}UeQDrrX(DzOxPeReGL6OFR zgi-Jgj0*;$b7vV`HBhSOT_Zxv6+7Hy@FEWZA~W=HL?5T}g*-z8j1kvRiG=1YcOZ&^ z^@ZiOkOqnLZPs`uTrF4cr8VYS3;&WYF1Nf=oCrEmmn5oreBWz6cc#_n37w#t9Ev<Z z1uf<+lvkb&f3l=2{V&IE1YX^d<M`DWS=kH?5KRAobD$nqzzFlwU=a`;!n|XGVu%*P zk-O}$O;8p(R!sAOxXw-#`Y;_DK@QfBN$n)6>1qP+?Szb+#1X7JWh~zD<W7Ur6M+&( z9VvVy*e&3h=4xV4km0`arYJ>m0R(Jqv6<w=b?w-JFXf)@G%PpZ!(+dgzGYjVx1EH( zDbmnQIn8-ZIkd!YOdTTfKobc#M|Mf{wGdrP4~TWs-k*RBWOxLb<I$J3qxmkA@)4r_ zpmVA&XOjG>`zbUgbzHM4`>+;MS(}gguDUOSM?GIrCq=)S3`kyq$1|}RU_GL^N>d+i z%2-I<oOMzJ9EjEl=KE+uf@8s81qZgaZ5cE=2HPBtZ;?B#7TVAvfYa1-Kj?7<@dg+T z2F31dKWkgltbTP!hFKU~(gZ*sTv)=YZ$Wm5o;t=#QlYWfs<m6}mm_psNf$z6`s@iP zGS`o%6NvaeKIJL~fkY+5knyD0D~~%6T7E)~wVX?q6u#Y}4~5^lb<;$ufE^lsRM){3 zQx^&StQ_3L*~^OFmuv@KJbd|y-{8fIr#pUwYSQYgW|oY8HFV;CU$iS`7sr@EaNE8| z(uh>p)r0b9U6s^BJ1;-svY6=WK36!sY~CAvDY$M2<uWcuQ(ntjZP*}7Ep?k}wN7_d zoNjTn6z1^B6nXZRYRx*QtA9?#1W%2?%Nsg$q4}XJZWneiAmTE~BNb&^y=v=Syts6> zk?m;N7*4P5scD<WPM3C^QJcRajjC0)jXEeMn^<54BW;g^w#DL=gVUuHq?oza?K!33 z(H+2y6~*v6anbkGsXY%YPGQ{*o{|1c@a1VIUV-_<pbo!JS5q`KQc1l{RFOp5uPDFS z$$G0A5HEO$3GF^APHHXyccpc}TlEeQ_eroO;rgqhW<=NuhMD$PJ&vicsd7+vDoWB~ zU!`4i{Bg<6VU6_Z7y4T-KOuUIJ_4Jq)9vgR`a9W09v9@>q-f<=Gk1}p7aF$h2eM;p zZ%*)k8ekS-QArUgytTnZPkX8-{*ALe-Lb%4>b_Hzqju{T`l>xKQxF@eJ4$z0?RUZc z@}$jgI@y+*(>7dx`}wpDqgxX7)IXwXTXW;;^?4|ObNk(M``xpJ0w_lcyI%e!^eKL% zNb?9nn;8AchD|H2N*L#Aa)RP~LPnP|MF&*&SyboTuHw}nK5Rw34L$|;WN`7*Z`g+4 z@!P%95x47pRY54VX|ZuqV`v(VA6kyy!SFCTSg4LcbEBfzYq|n8)c4*RC(_6eVLm}b zfec)Nrr4u5bdC#0F5V68a<1gZ==W|@4>6(Quf{8S&TuXp_&9;TsQ_s}mcJEpt=AE2 z8isVO-v?wI=V~*njY^*WfPFMbuQPgyUcglGzN!bz8HDkMyiZ+<0kB9{GA_WJ4zJsy ziEd@wfVhtfoAObnydh?1(SB^E5IUY3m#*FFxTWafQPv(7d(+EEk}prI@#WoWI;{>b zAC-gY!R3Pj-Y<3i)#5;9TB1e&YCY={GW>!N{v7sc?)IP|lTWg<nqWa9)r#cFgdT5t zzbe|(*M9&jq7c<Hk+Pt2Cc!qk?k8`6a*#+1JePZGZYsU~9pxc@*TM%-d|4j=Umr^9 z23-~zeoxcHdRATJL1E+$@Hj4FPQP7i9{U8%xk(9q8jcikh%=;J8x7RSd{`K;Go)SZ zsG|6=92_%B^DVx^sppw{i!v0TJ6EK2UGr#?kB!S$#}QE`=cp}w2V}SFir?p2oKZ(D zcZe*bLTZdj>PS`S=RKN>a;bP?@5VlCs*^tY;SI2}_3(B)DIY7eN}w&DhO{^cpX-Gp zONFALKygqK7=VjJzeix;tBz!nGG$1LAZq<1l8-4iYEq*=0UXg&G~89+oK|A}Ddwg> zR6AzlK;3k!qot1X6Hh^>4KgJsH*J_cEi#IfbDf|H1G+i2lF%boI1Y2y!~~d^SV@XH ztoXUnBUnKXaBADA=}t>fIx5(d<eD;~aGW{EwS<q^PHlM+<KVcoqmO;utWMKF5^?xQ z?CYswnyjroR6&3+EmrhGG5NrSksy*KnLGR*RoYJLu!=iugTIr37*rdCtM%J02X)4J z)aHSMW*tW6!tzFJ!W^WBsw&<kAGnopkBjlPUFwsoTg-+DC()J^BJL0h!Q&}(1p<)S z8;!n)xH55vS*yZHefJOr7w_^L!)iAlo@(;u9epu+8$0ndarEsp?9r*It9|3yUZB++ zr8%!g_jM2hTim6HYv8;~ncg_I0SyHVWuKh-l;XO&SgY#?ad;5)mZB@_U^dl=fa@;p zbqJZiM^5E~8D4d!g+kluQGxm9F#ZfT;Zy{hP#wIshn&uE2(VE%P|IS*V*3Dwt1|U$ z59A3gLY$Uo6Nn?_z&{$OJPJ=`?cE#>^7_UwN3%-XLM#%#X#e7>8Ae?1j0obH;4)IU zFGaX0I<+yBV_Obi1W=Yo7j>aA&A3`i#=pH3WD(%+9D7*DNm2XzpgIyng*Vtbe>dHj zXb)R0sL|S93YiggN5}JbQyuHb3v+Fx{-MpVMdbkII7NZYW*Cu8(Gh#1(dix=)fXct zRTQo0y@tK3DI2I=)K2l#l(wAL7+h#^wXm|DM9?F!-2tN%Ler}k0qYq^%vo2Uc?QWP zn9q31u{W@t$z^(8{?pYmhu`P&r_aNsEp+|`EANNq@ljLnZ5-;H4X%?MWxzaT<efZY z^=Y;iDJPQ=thxAubtven$cu&}Od0u;PKs%WXXW$UX<YSKAk7@uNzk8NghblB-=lO~ zOOQGo?%53a8zX%HI%hdS+5#TEM+S6l!SL&RUSPy2L>1G(9d;20p&LjOr-k8aYy=On zX}s_g@ljnPojW;RR~ReIXO*nuFH+p#GldqhC~#`R@)##lGs1YjK8}eK%f<y3_bxlJ zykwGe8{jFz++n77Q0Rngp>?~PEtlG0l|2{$G*TnO`bwJ>Ku<fRwwQM_Uu#0Oqt7m9 zL&4BsJ8Jt+iwQ(zJIu#Biq;n7zlq@nkg}OpK4Xj(us{v?g!Tw2c3=IevmBX5$T4HY zHVmlp9|H{=sMhpNWUrX2ON9lwHdHW;SU)G!QEPwb9oOUOj3H~10lh=4_Yukjo$)#F zDamypN~iVmZTm-s*0VOIhxJbhk5Y(`(HtdH=6+%mX;_@w>Q-6|%)#YHA@c)Lc_UJo z{h0&HfOE=^d{{m%K)sSrQ4d0<HFWdtE7tuHMzPtgLbGeWykQ2Vg850EflCW5;i|OM z(Ou$nR3|}G5Y1o;67msiLEf1S_Q4^!BxRGEr3uI8cqNObjCb-;uI>t|DadyLRZx+! zmn&{ZF(C2yR^dtA#3nk@_eCSiOKhy;*!a`HMBzp+kZ~n|o}o5ixzx*u!gTl~!h4wq zIq214fQ;-$+pZ{(lzS(}04aLRVnJyJ)|NK=a}qm8UH&86OeS7o&}*XThR5hi+G#6a zXJHHz5gqG*=vKzjqfUT8=0RP~IflX$GVsD&D}bLsX^nj6omo8%D}%*ZTO@YrWY><% zTJev^C!Op#Q^Tm#u7iODb@gDOPnvWg98ZQPOWmN#2B~(ORjrLp7ZP|iU!d+mRb#qn zsXT7hoM(>4_L#J0w+Jd5|E>s%O{F!fVZEcc6lcd1_#@Nd1)JZko!Izwkq`j0B#}}Q zyoo=4-sE6#bSLREj0<y$h43(=@5$g{+_cU7Fkg%YLFRDoyLd;T10F6ZAM4O%j%eOM z=iZd&4jq)ZS*|t*U=5h+TMcSflO`5+0(^l|mfN^~Vnl`%+(cE|I6-#qx@r^lo8{*| ziH@DjUr$QN-n_>r6oq~5hR^~{0fcI*`;*cUGg>I=j*?&^N=_rQDk@h+yb?gw)KC;T zHm}tCb_7$L#J3Xo`evy^-UX5KgQN0i8=B8&CT)o1)fQyrZEa_y0YZZywFLB*GSy7V z{sG*vTXtO+dAJE3y$VpodXH&9spA6mo7v5KcMxhu<-+b^XX=turRgpennSa1Zr-^~ zba6C2;}p!)=T8HQX2@mGG{_zV%<s|e`b_=8sH-tJV{xK$5xCU3B=}w!8t(>XWXO^3 zz*6jx8iayw*E`+@OvYn02-?Y9Je0AHEk1c#YBV<0f$%*?1}7H3_yGoSh&!I7U)}3& zQy4Q@6QZOLrmbbI*LEkDucGPf+atevSwG;_jI>}D+N;FS0uA$N^wTZvr)EDI*bz+e zsxMMmro81bG?k7@m5IcAA&-KtSv`|_^3tffL~+(HDbjukNc$yyTh&)~08z*oLsC(S zP6Qws%=#wXrbPf#28|;rIB6=TErb(xlNPxt#YA9J`RMpLW;E}vnx;5eCR#5tP6H=| zL{NU@ba<GK>_XsB2G41SEZ0LY82??g0(ZZ2r)nPBk1ZVk8ghIuFSkWu3BptuV0hHx zfT1d;^H?l|ZZwcCPB4-p7&v!xxI-?;A=Iu7Jp&yU4LX|dm*>SuA;ua4Qh0XT86n0r zML|A85;|{MxAw9flgyQ-uw!T?h`DM?qnMf;<rw4c70odzS>dXT%dPQkRpVQ75gl(S zXGxN?b~Dq;bhLh9AT*q1uT{@>TM&&#hn2H7I~TK7K80_ZE+tx19li+}T~`-My8zFe z)C=%vYr@$sta>X^DxRbvQCLr-XXYP6_y7yNX>MSAagIxFrY@qAgP4h5_w-*~W)>0< zR5(q|TGAMw0z7MSBo{ed25=-gpQ1y!;OM*5T7tz?;7qp)z9I3i5fT88krMM&)@DMW zdn&RuImu1e;!XI257Q+#K>?g)E+$7!b85;&9JbhY*%3JY;pAsVM%8Y&PSkVx&YjBx zb>}(g9Uk`Tnw$=n5@4cn<b%PD>hLhHF+yAd&Qat*#88J49Vld`PjBOxO1{&YFG-RY z=0djgPT0m!k7_$c1_95-iw#5|)XIbhqGh%0nY<>43eZ+879b+=TuX?PD`8W-m%yEq z8@;H(gRTbpEucR89LOf(`Htwzg4BdZ(9?fJiqxb&WH@^ON@;36K5qksT!zt$v3Wch zv5#C-`)N)--RX;hM=DiGA}bgivHI9Q%*W=~Z?~9RDF%EzwL;au+s^GmBnA_lnpffR z+-z?_HkBF+C>CB-(?JeQ(s5rj@ti5whzOW;oLFFq#1iT+vd`X_f{ryz`$b*0{gPW( zwkgQ-QrD*wV*Bgr^e=2cbrT=eWQN@GEn{w=FG(L&uanu3&Wy1_(dkKq%aZ9>20G1N z+g&mUHl%uO3n+t@+$rQ!>b2CZ$}QG&u_NAw+_CdnZ-HJ4y#Ba)kv7G=)(H7vU3!_5 zwcetSrjst%1TzxeKbb8&BRgoYb$z%18`!}9G22&0c)o=QR1=Tqn0+sD06K@BCpnmx zaWN*aVAvwD@UmLt)uF@^CSW&-ZG$c9wAD&j!#RcQ4<Bm<3AcrSS!`bjcp-hP_K=ed zkhRr{(=Uf#HaOz)%ldUs{}thvp+OKisK&gp`I8cJzi7q~o=9qvqv#zV0L_Sj0`tZu zacuF5z4Ye*WClJL{V|98Gk$ObKzjJ!=su_;2X1q+xL5!g{K7&#bz*59@;Vx550b)l z^sOB|etEe&irKm8m^NOt;o$twz5c{$%U1C1mMIgQf`08_7$#_m;Vn#Lv#SjqZ5}9& z^Pn0ITJadDkx~0I<=IC5WYG9+1Pdpuy_2iZWO`>?!qlk>8DF1NhxhX#gt%}lH7ROn zzaq)(ug_G+SpM$pFi@zld7xe={wXHzd#mY+4)egHz|_Ls$nd(Loz<}`nug~SS!RY3 ze2<e$*6pzw3UM-QG)DGdNm$!sIM#r1MKzN8$?R%i`mZ+ZpimqW@=)GZ)pa*3KoIqZ zTv7corZo)At{TTQSB0p9NP$_5)pJmsX;f=NAmoq$xo&4dt9ReFH&6emXVPwoLA3r= zhinMPbB$e)oyD}yn0%(7<l9nuX|z}5b?h?RYD)R8I-T==Xpo3pa`!blyRYhWW*}%u z6(y)#JtZbJDCKFr5xXvvdB<g9eL;76ikfKnD+Ie%RXt$_3?>Yg(rS_FDRR80-Ed8y zE#^{*fF|8z&j_IgJ*<-vPC({<1&IK_N1nU!GKg9?C33HeOoykUE%ciln`0IhWM*=C z158oW`v8d{XbGGk4ej2jMvu!;2{(>C4r~vqd{)T9E>iOVw-0HO=sQzu4?vT`$R7td z*@63Le}ZOd`LJoG<Js=en6yRQYMZ^Pb)Xi`6V&5qs}NRng&a2z;dE*XFL}D&QF>QA zx}apFRB*&mi#mGVowqhNg((yTdfIX9__ci}mVXn%a2*xK+7E3~!Q!3yN+)XwhM<6N z<R#uk_o5@pg~fhvWMZZ-fq2(3=@FrQN`wT4ZCSCmr$AQ+NuxY_dtntTBao?MZzA$8 zVNYO7a!~Grb>VnYRuhPgpP>rnsD#K2WLAl$W2PNe2=_TWxB{IzohK!Rb|A~JVqgN7 ziTj&0))hCh=xt0+!A-mCH9|LFn*22dh7bVnzMF~OPW|wwb80`a(_rCP%2ja1Et!^& zIf<s_v94i7@~SQEIKWCg-Dt<t?)!|zLTWj#CvChV`ix)vMLiypg-6Pmqf!5%lg4p9 zibq3&VHfb4Pezw=ZHDOw;M4YShXJ_t%^}$?`k@EyS^<G(oSiOs5^{~%+OnNm_pWpj zw)URVE}DV`IzdE(;hGr0c(Qdef8dba6)AxtOrY*^U6a91(b!<JJ3of$ppbudr$y{d zA^H{c*#%#Ls}YM-W6CPgtZ8K=K*HvhysfD7dTHYc0UXcODYi4k?&#il^V;1vu3f*q z{_xKF=9~9!Zpght-UlRe5%a)`(9a$GE{ahU#*=n6f+1`a!@V1rf65()vjN<t^(!VR zOWa0UmKL-!Fq2MOKG(%iQR4^Dx45JR=E`h}=a?(U=wxbe&fd~Y4ake-meDs=mTQ}G zg7#@10Enz<Mq#J<0cfTp?>UXn%7K*wV%+9f0yOPIpgv+ZIyi5KOor9$Og^+nMKPur zF}bv91_w#_+@^~Pn(_LMy0n(DENkBFIBD-L!}n>UclpkfYoO>wd`=pvtG1lTfN<f7 zKv7m}l$lKJoXlQ6j|g@i5$rr7*rE}^jIaL7%kenGksw)onfwhRL6>2dsFhZKy)mg~ z<N0xoNY(*mNvWS<J>lUU>C;Q3-rTk#BXpR(b?auk^|&mK%rsjzWrLWy&BvOpTKiLx zRk$Ysm5jq=LN4Xw((wZ&NVyU@86!bNYf@DqkkiSq$_Gvx9G-e1UsK;+8*kAfUPS0| z3Kyu8;H$UN>*O4Es_5=SL3e$aY))<DN5x?Mu@lqGQ8tWDv-^<R4dX;}Y-s$v0~eAV zU*<s!g}Q35A7hYiY+BsWS2RK!!0LRv(PdV}-GW{6VfwVfm26cnf207p>Ngf4oxLt1 zlK)7NKx)06ll~^&&%tR=A95?mtdfSn&WwZeDVP=b1;cHuQ@}nkeU>_y`h7)jcgvx= zaRYf)XOALy#RMyNklUiabo`EvDZ)`Y*lD4dkb6|6*3uoqoqVK76`+h9Mkjls8v<sC z#>p|!v|!}?3_m}^&(H7$&hQ`=h$i{B_EbOGhghd6?`R^+jxHkKEi{Mp6L0Phkl^1} zBOT#ID#93;c%qSLqJl+_(!W4lv{x3x0Z6N`#n{DDm-NK6o;b|=8rUdQYCnRh>1_nf zD)-w(NUg++3T=+9Ae1x&D&N=%jb(dqGlEqcGa!L{)QE_Dpg}f1mf7xjW8cXJRe{#% zPQ%5qLg|_mIX}O={J$+PXWdMVaaZk}6GtjcQeBQZ#!)cfMqaRgDkc|j+ExSGVK>qT zIfsG4&}BQUDaa??cWOUb<S+tDasSi{1wtP8RTS*wg}}X7`^~Gw2zuc{yX~UIuhEoH zaNCZd!LP|tE}^NNmpOXHX2>}|%Fd6n^P}uc0jhsFI8q*))PeR)ikwt@{FpQ62B<q_ z2GD;aw^^80%gbzJ1%k<Nx^N6>Yf6;(tKS_-5HXCd{VoFO8tsbLmkgl`L$eWdQ5uIM zjo+@>_7gSI@c_;k#|h6W&nf(Q0!t=q;r#>9_1gVgr*Iwf)?ndFz8pHAQfD4Kow=uy zr}<e=JB;hew^I);p-f|;3vL`<Z>qCpi02FIb}OkUCfgQ~HSpgM7!UN!&kRYSP^tF! zhlPQmuP7v?n5o*yS~_ao9J?aOQ3h((|KBajfG7U5TocNZJ7+~66o=}d1-07VzZN~5 z;1C)pqJV)p8oN~M+He>Piy={tQx5)X&hRXf)(6I3knDAG9a2!<V-7{C<9|cDq_k#P z_sLl40iOv;8!bK;7lV%*4Uza!;9BS>Tua%G4fT`n*pL+eCP6$Gwr^1sE_A`de29Y* z;tv`_sDBf1iP(XX)ZI&}!SJJRn;Iky(1W`yFt8b40BU63_7!?6Tm^$o0Y&wznC#{K zP@UqNdScCl1|)R1pk!CjJDQ041n6AT<7#>UmyDLjy=Z`b$9(p5JHZF5(}~o1A-WuK zX<DSJkRn+L??y3&0B~xDS9>}j9!G0@>E{+$N!*qp!cdc%`*_!~hE8d1<!rdso8^8v zeaW)|Erx248_-}I=7ddmygE+)U{4aPSR*y6{e6H^w615rqRN*wUbpRg-bWiC$0*Q< z^|1U6KkOpKQU>M>SX|&aatO;efdSc)vFz$1aB?RbqT8?<MgO224sNhHAjVTxpVD!Z z9XL_tYtweQlmgjYlbBOKG5DhKG-LZ&)7vxRibMpT`6c5y!|P%@fn}>s9fSnxrhehp zwK5NM5Olx<_Fi0!EDG_@t||2IiW42!M=UZXba>Ngs2!^+s`T6x@H#3sjbHcR;5AtY z65JFvZdNveH@T4dCbhRBF_Hw<NEh)pD3a^?&{SLijZTG?9kuNSyu}U*4#H!55Ilbv zpJ1WgpBTfbtC=Qh-lWxr0+1wud3Bv90c=zu+r`gPvrR*pr_-h>fqy3PX=Lo_sMgH> zh!~2pU{se`;`7P0VyhU{gK5|m_@xY;9r0@0F|G{_OFbFka~e6j{)4)N>q3a;h~rXx zSHGmmChvIro!_h&o@#Av?E+t0Gr<bT6JE3zQ8hJa7^6BS-4^nbc3|++p<mt_3gXqc z+@c4&>`>nyillO_)NQRL30vD4k{Sq%c}u?Bc4(*kV&5y6@7Q4N7*s1Cs$FCv5DIhg z{1A>!r>-iFDwXL$?%orp+8?ef$J`=hY&is30OkV5=c6*N;u+><3vQo<HS@IGdXxl- zJNa0)t9Oq?FH?|2;<Uw6*J&xj=lWq#|IlDxKNA1C!Hv>OqJy2d4bq<N_#3n%cVx%k zer~IYgfRrcV3Oc6tQc|}@5D$9cJ$nLK#%5T7(gch%ACZBw!jYEX;HQ&f>>F<EA6Y` zxgY7&qs9TtHaUedUGnBtykWn19n!wtm&5e(1vpcxV{L+ta&&^juLYWi&C}`9d4Z@g zN#Vm!7Z1)=<>#vMb5;47RONJ*+<S2Ajazrw(+Hx&ynW^2Lv==0)9J9dhjuG;%(Pg8 z{@~i?x?k>rI=}Vz%{L$1+uXdp{&4fN_k)7K7q?d2Tc&onwO7TqcI90<&Gtr78N!yL z8TrZTm2HI&pjxGLys@~|gJM_h_I_a_R)10x51@TLFml6mktmRZ2!WPw*uh?QIwW?| zjx<yp&9J(%IqYD8_Bx^gK&`#<7@}Dz81fvn<V(UqLvWxV5_rz)kL|rtC*wZ!yoKQc z3cO~M2_%aY{2$B-^G8w)VflLfxafPYAtWO3vm&-F`D*U%iJutqLUP}e{aTjh!QE9v zo<~C;aq!^qiT}R)u83-EB}8gaZu|o=R0!plL5hX;5+>=x7>rKhuyFj?`Up}34Nuw} ztOEuN{cmjt>>7e`H*U~~Cmn-$k<nmpVN`1DeHH$h;bG(1w?XNf22w64F^VjFi`JpC z$!hdY)C`IRfP;mH`-z_PB?^vXx>~dN)iVpVq>7nII0kinwQVN=rfN2A<5~sT5w+`j zG@DWnx9JYx$WFGQsPpZca6Stb4f1wPoCa}k=4f|=J5!xjJN@0A4zXJhE(XZ8+RPYj zo$Vfh)?&)uuJK!R8dWXzXR0r#^NCvhDsYQ`f~D70eh6P63BpL{6zP&<U+#>;X_PVp z)#e}(b2XfxT9JZuHl3_u^AV&+HQ`0!rP>097n7gRM=GqDTgXogutT5`phjS+BS)y1 z*O5Pa<Q?)_Eer4p9at{#w6Ec~JJGpb@M=r)K#`Jaz>?H!xVPN-vt*Z!{TW08U_f>< zby6eeqD@yJ87W2ehoD?G1OXnYqDkt1x4?CpQ~<$CyLtc7Cj^vq#HX8*re@;NM5U%J z4!8@e(y>T>h9pra3xr*DRZ-r!>=!9uTB4h3xpAnMpwDJ8u}g889(X5;&0p(8`hab1 z&MB!(bmDF3I7GW$qyuRhiz|ZEA-Ii12}leA9w9qz8b8YvH91trcE9kIPHNN<aw%l8 z{E_g%k(7qjoWcQUbzFhn6;Eh?18w#UpmnmkO3R7t7sGE5&%q>!8qt0>AmXz6Fn#9? z7ll}$Zr8FyW~Ir9@ZN+3mTk!|;pXT#Z(0xC-QVg26|Ad%yQO<+EqTo-mI`@N^A>U7 zlN8CWX@V%<OHI&D3UG`gi3LMmtSgo|zsSCl{Itofz5C8fF_p~T=-L#+9Y*+UK$Gog ztu4vauNshVTj&GYwKO8PT;A41>yex-zYF+V4~f(~Iy~qO;&CUFV{rt?d!SuY6}G6n zO|5_dTlPkgI9f4&k1Y6KhsO3sTh65kgWT@vTAFj-6a%74&c$fkD-Q6YVsO85M`^x; z5E7;f9R__+X08YnbTe2}=QUI<Tu7D{6s9de2Y_TSBohbw;Lhxz{6yQ_RBQz)*r&uC zeq_JlI@s~K#azf|E}Yz=2>pz}oDkFEQoBUgJ>0T&qNVI66D5S*7qnrWU4oQCY>s&w zL1Tvc^oze3!qdSCcTIz`z83CW40kSuI~T((>ajvPqZ=Ghly@ByjM25>kc{daVZHBS z=X_)*jAm1#Fx*x&GFUWvJ_e(^)`yIM_j84OUX+~2s1Lp26@u9AVC#)((z_gjzFRB6 zD^{-S#Nh?wn*uT2REP{99BkVgSajULVZ#j%MqSpL5pjavU!cD+b}H;LwQ=5<<O7IQ z7{-F9qCNjUA#%NA3HLO<VB+07V@gJu^(Gobnhh^{Iv|w}bcdlx*@rihiO3I0*C8`U z8Qyk-(G*RT+!m4y0UCiQY~&;aoOyu58HQ2McTuF`V@?#ku1AZyNG6&~p_<og1gD~7 z3KENpIm%u;vhhHa1@2mznohZk>;f(n1T#c&2<HIJXu2F-eBhyxUY@e1)p?y=g43}F zb=79PGxLI>2%DW$;x!2ek#r(n5Qi|Zi6}bmCW!etkij-~e}-f@rxAIE`vSD38eK5% zsDV7ZcFSra35mtrn?0JBHRYs(#<Y!!G`zxhsf=`z(3u8eWG+liBAalr+ifU{+v(;E z-b78*<P*V#$cjhTEZ(i0#dCMwxHIPr<aEWy9etNINWp31)eeT9IUW0&#Thyy{Ny$K z&$$BooO3?soS!3X$!%`4m$lJi8(j;RvKt5GsHh7qOd?Jj`nN1Lg)?@{hA7x%<=PU# zqt2#Y)_VFofBlp9e&5eCQ;yVRK=<2v`o%wh@8SOD8W62}w>oxO-<{0rgP^}@Ifd+3 z5ef&&9o5x`U;mq@|M(ZGw`c$HZ(_*~$JOv;AHsUKo__6fkwGH>6o`IBm3a2o-+KC` zKTNj;<vt62zW+ZP%R#_NRqTU*`qO7$_z&q~D5-#MfABj$`IEooa=}4u{uC_$Dg89M zh{wUzdiH<+?Xy4l&PV_4cb|Us$9~ta_`Ph)D+7!A{h$2txACvf{r$5)`rFoay7OPH zs>3s^)o)<&*3&P%|Mcr$|L{kD^X$vPsNeyv$1C>HKYiutfBO1jP&v%^%l;<%!Y@2B zSW@WnbHDfO5B}nVAH4tU^I!S!{eOx;jm9@|%6|QOAAbJt5@pqmi6)~<KmGIHefq^e zj}@Jkm<>i0{fF=W<UhZ^*rYj?NfHwR0LsdB{<yeS-#SFHk!V0+@zZaA1xEGNAAk7! zzn8+Mr(gX3Prm$}r(gW0Q&^&qO9E=XuZvzmBLG&ObugB32_yN*U;QWoY+97$TJQoV z*R}3ra(LV`|MkP){D(*piqyVg3;x~LpZ(}t$>QpUj6?d#SO5N_AOAQ~V5$x^Xy(uV zF!Av)9~;URFQtg*cvvC*`_A{C{@H(w6zQu|L>;ly64CJA#b@8}>7dA`Jdoe|_S4UQ zHBy8=KmFR@M!tcpb~r5XwD^bbfAoznJpI!zJ^S<DORP|hR5dn?eejsTBj@`ce*cHD z1<Xf}iW7k6H~#SH*WZtnP&bqj4&>R7zWD5q{y5f2KHEoJ|KYbj`o>>>_|>n(CRD<K zSgIR0uI@+Q`v$De*S{BA!vn=zJ}4g}wtVn|AO7SozVhsUeE-@1{136=&qlkmNexT+ z?2o_w?2rEn$|QG9t_I-j4s~)D6UgP@?HFL`HVp~*lV@M~?gu}3?}HzFOM&r&_y5yB zzxTB#xW7;T`Mp2=@W1@F>Ql}8f4%pc{^+2YYBr6wgmwPqM_)+MmT-gPYjn4Z(qhF- z?Ul!n5G_I^VI)8t02Q$355M<Ah4^YXPpBs#{HJO-@aZq#e_{jkpGM{-<~F}ksPEBY ze<N^OwdQF({myTF_~q|Csjd9z)W8vN`*VN%;SYcN+2_90FkT+)hyVS*b2GOF&<52u zOA~@Xt+5BD`4E}Y$pY=w<i;OiBMson!2mJqUw-t}RH|y^O<Un2;6g0SF2y)mN?)T$ z%ij?%q~`X|?|oVQFG9>9^R6xb!4JNz=KS>c|KJIyPq?8L^C#c<hhz(4@t%x$MNFqp zFu~7hV)N;0@j}fo7VwR(5W%*A(>TAEpWn;R@8!>{(SqaJ6_Hklq2@`K)RIx5YaqdB z3%(k0Jqo-{c`^Zl3}s$IMc;}8w>wzFVTu8_`k<)$lX6VPG_FlnbqFFv(0Jow(O51G z^W9?D>ST@WtfU8HC(frKX4};U)*1jHg(i7^JE6q$h0PZ)n)Y;OWPRv*J0Ir9K+CCD ze-4!7=NJ#@7DldImX_b@(}ozczYvu9si8g}7_yB!`ShTBSdQ9e@ExZccks~TZGgYP z2Mfj5vSmN^^<A6IYE%(U;`%LsE1qr?c}C7t41YQrQ8GqUZv%m}88MiY6l(VUqK(?S z(O$XVZZQwrH9Tu;sS~%^L@pHn@lLEW*p<q`BnF{5-d0kdXE7I>s%45N{hjzeqe+9s zxtcAzgIUWA#xM)nrE1=h+%x5ff7Yc$mv&wkV^F3F0N6}0pgl3TQ&%td{F_iCgL<#f z(Z~0sa!9~4&@s4W?uI$YaPt9U)k^7qm|Y#ix@%xYI7)af$_)jX(bY2P^=aW+;3Bjz zwQ;u<=}^qYL9<jsX^Rhbn5u2fH73T{3rjP<*jkE$&>faW67e>=?`%y-5ALSZ!nc@| zy{R6tu!;3*tV!ZLqjle85s{-Cttrw#L?L<_2g4c6JoIx^jt13Hmuta43BBMz<k(WU zV^5l-nhvZyAi*eHGJuogcp^!VG;DG@yu@uKK}q#vFE2eY(s=nmiHt8JSBiwiePjau z!1+oR-NJ#&8XmonaU!fAH`WXVm66MojxGZaQZiLzftA!lisiU3W0R7V@Op>4Qn6aa z<KjVngeh3HTD_^hg#qREOP7R4<4toZc+zR!{bwy$Ohda8h6?`Ei?(3sS%ON)r(sp( zGeTm7#yyKwz?JDaS$G?s$d-^#12j@A9BDD~%G+BOCE|@zGDjceMw-wi$w4b+W^yrD zEnBr14d$OejH%*HlhcjcoWttKHr)n-Df+C|+}S#_OxF%4yuh`|nX{p8A}kCa&S2gD zs5n7dP>0?>>fnPQX444;=OEkYR?3iFeQsiD)vXx}^;W%d9CXQyLWWqvU4bKv^vSK! zAKf(F0iVMM#s2#7xZQgD?I&-)U4OErfUg<_z6O52{q`wnCikO+AhI)~(UIW!e*gdg z|Nrd0Yi}IMmLT?hfc_7udwZvftx}Pc`dMtYK#^28=Sm_+q`GfkUM`ueN|HIMs;tQ@ zK1vb?_QLS)+Q2Sg7_hP9U0@8u1NgbH2V>8U7yD<X@7?(qj`N6!6OXJaQtI9wK!t9x zDl_85>%@t3PMn7c(F0wooqfSKgO!^v2Z7HWy$>Y7k!KYLizJKJ*z8dJo$(o^i-m7- zJH23~KxNT}E_j7vhaNZ3?q|c(@!5)!mh<RY<}KVl*W1qz{NwdHie=8^dc|Vw0avrr z6txTrmxyK1Iw>{Oql`(aYg0nznCuE>ulR*0DdOWu!13??(m{NbI;o-ylJlt4j>qG1 z$G%4jL&YU$YGm$A_u(5l;WYl^KmA{R_&@&D-~G)$hpQyp-g^!1dkyY;4ekv!xI0^z z%@H+pJ6kJrM23t7<PCkA_7w@LYCH<z71D}n@6m8{7Jzk8NSkTa1U3D@=B<KQLkf^K z+n4%l3`+#y4PAmC;8F#DFdZ-xoF3J#n()RELZM@wq`|GoTPf3J_4=1fOH1<IiJ~>T z`75}~TE0KJlCK7sc$})0r^jjY{wJTb68q0m`||}^9+c&~Ba90dM1Jb(=DYVFeDYw~ zsYX?vt8SlHon(1QyaOAC@ABUGtdo2q9>3}<L9X9PmY4|3<nZx4C;IJxl8+E$)E&#z zBNZ5F2e*BS45@~2<;{GEqbbrUEK5J+TNs)8cs-`j^NKPz0lMrdEDR$+TQ3rDZD5#D z-GgK5q#=qU3k*ZL#poJCbpQp%7Kly;<$c?a0)s!Re<Rg)iV6=Jz#BMAVxTv2@_q3Z zT)%#7r`=%hTTJ_M?X;VDlh;59OQ3VW3Z-S2=XtH{HoZx$&eDjj94&$(Wou-paSuu0 zVsbxzee9P~OTW}!j+8BV=*~9sxJmzRJBkbRrYv4&3=#Kijk)GJnkbdP&eU0S^1ek- z=J)iDO<zka;jAaZkZinKmv%bL6=##}!YVY;)xZ;9Aw)_OphR_ecJ6WNKT00J^n3mG zLE^$T8{NEws4V5yaQl4Rc*h-+$a90PVc?bKgQcY#_t|a|)q5`*@8V*zHH0G<gc3!W z<hG(G_0cs@mQb4-5rWkC__b+?4N?~0kE+z<Q0&a)VFde=j=A#WzINfhsA&>sQOoyW zRW@?P<wNwDdk$IRpIC6gon;6VBTGU^m<l*P3Ic>-d35~H<3hDV>z*So*TcG$BGDZk zbx|<_Z8IMb$EZ}xd>_Id1tSOX$IRkGwJl)S3sX97`;P&~02cCRP(COXYw=+MZP-Q? z^uPrYf=2NmzR57MHugdkiwy*}N2LTc+!`M4cdt6?f8Ppoq8?6asDPpAwJ;GR%&rF( zp+=rx5-r2U2<$m2H4?t_I^^<F(|Y5dfs^nN_rb-Gn$CDD5Sgdzm2k@%PKkM1e7Evm z!hJ8{zL#)sm-3i|!bHokoD7Il)n+=zhNz%624%u6f&d(*Rx|5vvFA94euj&Jow5KA z%cA@S4b)2|9Fl9V{%y`+jB{jRphi#sSP?}irn!w}PN^P;C;2e1R96grgIvP-6-Me( zyBC7|Qb#ZW5=05+NOwbpB7;n!7n~C*ZL)N1V2FAH+@Hsp33{&(B|U*tnrG6u&T||C z?Q*WbJm)HIKTJevb?_mj2qEQ_fdSjpMgl*!DT>wV!C7^dk26eG+_nN(bPYK6V*9Y} zZS;A&f=%kkt?p79u)$;Zfd`1{s5zL~0fmO$blmh1$ga4P-lxnj)DqYSlt9${hR^;P z!FaK?(c#SA2OX~reAqZa9hOhi(Zg5?ghs_&U{EP2Z3GPa2zIxc0x144Qp;vM^-~|w z>?ve9&A5Xmi|IFMb*6qdO%?)<qmFrh%)WJU|IE~P)upYPnh!;F6&q=@`_cy4RYVfu z`8j|%o<oO%25@!cZA}7jq=8)>ajddI5h13WeQvknI#C1#?@09L2#zcc@Cd{+2HXp2 zZb3pgNu(mVBEAq-9SYmF6^VYCB}aHIaMVhU$ORL=ol=Gn_{v_AM=b*tAC2=7yiFBr zj=8O}QvJ-vT}56C8~c%aTV^K^=%2X!BgNQ}JGOvHX*mX2(7SF?04D<RvZ{%qR0BoC zui)}eI{<3J4H#CKjJrUV@ZeQEUdc(Xyx!PpYA!E8q~J@%52#hdL_<LSD8eYP9tvwA zrYUx?2LXL;#M3Uv8-<ptj3{--K+e_Lk&%y~uL-RLXN|XO*v#6eZ9A+Tzr2BoF@^yV z#?ZqGBY)~zM++?oanF6jBjFxLr-6X@Cpuy(wQ0LqQFi&9=(kuQF-iAvl4?d$WfzK0 z7z<CuC*)lNFIS6s1UF^444j7Eu6=!n5nrIod>5EPmK_|h56ugvqM&}4A@6vk5x<MY z6piTuP2=TK{K|PI_F(hDLh=m=dWDJ6=%5aEC_qWoJ(6Mt^6L3)i*uhy2FG2+Ft2$z zo+JYxVv0)`JWSTlvhGW01-Ua5=^ELBGYGYdfm>C7AVXqB_1#TmawSNjL8Ud7fchy{ zHr><Ll!Ej$gB(pThO<n)9_L+T4peXWgupg%<72M{YG4!()e*n~XEuo>LLw0ZUG5E^ z=Cx%X+-B8~p=HQI$=O_5I2lKgQ5~^7uezZN)BM;@;KTs3hiC@G*c&s6f7pXFjDBm0 z113pPysjc120C(g<H!i)vd~hZlEmx=FO7xThQ~_MfXou-l?6{BbQMAsN-*3lpfhLb z1;*Tk1_-))q{xma#7&3iu(ob<lJ6AF%H~jGM6~cIpkhZ0?K-}|k7_f<Bt%`c#{xZ= z;6{L{GX>k?h18vh;(z#u@@E_8!KSWVuyXq%EWAv0q(*$9My2LoFvf{VU4y}%aoZl7 zyvk^bq?rqaJv+39YZOV`9jXFbJ?a-C><>r?Md>VyRkq*3+^C0#8h7W6nQ2g6^TPT8 zEC5)HcwpOkPcqb_^)V>i%v;V0B1*`)S%qYLI40XQNZ_VxixfZ<%eUv0Yu>@;-n_wP z!-Px=PHFCV-n-fw?&wi|Wg9z&exPyiikuKdw}B&--F|2*;oFWMS6|U|G`_^>U8^)H zFfQn&eCJkq1WG=qDu>Kb4dRV%H_Q7?r@$h1Z^2JYr!P<8We1D2sDTdsX*tbwoL|=l z!!1@?oDd0S_~Dk_ecFs9iZ`OX(0y?%PU2`=PR^&&Bv?k3y5g?NVxZ${q22(qcT)J` zwf<!~fmrEdQl6s@0eF2%3#8}|7`Y-fX;DmZV%kuER+It4b08psfRQ<YZXsKOd>LgY zRkf(W_4FL?;_$~2@c_Fh$pSG@2LSREQG4wp9o=Lk@{NkR&NMr&3lMwV@xz+ATV)5% z-K{!@Nx3mt#&nmC5wbYxkAjVNq*#@7NQwyLM0i0&L3J>2;4Z<HXodcX62!80@~jxM zk^>oP5b=<U)x?wJ2PV6=JASq+C#TjC@)A0S)DcRBZIQC9t@uF$igb#g$}b1WkpOvQ zgfuyL<QmDT>^QSA>oKo@R@X#Y6w4V&K{qckoH}Z!q+Zdl1aNaoIkc}2EKSUH1z$84 zJZSM^dm41Z<qwg^b*Pz0&PAI7B@vlWPVySWS6nK{Kr2S6&D0FsO$PF+V&)b_V2OL! zp9k%;@t}WHORa-Zh>2f6J0HG;S8D7S?pW1raIi3#1dk3+Ngo(FByETz_yoAn2_H$T zl3Ic;?LJ`q9r|PMD<rjN{#R4{DyHRdb3JM`jGGTd5+_pWn1T+Git2h?L1UIjSHS$q zLnK(v9s4o}kZO-8ub!fu&b5=IvXyhT1y{^@uOr}wK}&ijGr4>*PV;{Jh2Bhof07uO z*>S-|R!Bn}ta1x8xwzvTBBK%$jc*BdNR^phqPgJr!}404<eqPSL0k)RJZK5glnbhu zz0a1Ge0W_#`=#28LRf;PLBUqy9ILYTu#@+M&5+>uX@`u6uI_5V#N8t9-f)$gp+qA# zi_M1W-6L3gV0kLjC4#<BuzQ9~-vf=bG=XQGaj<Pll#tMxUdUWP0l5Y;5U>DFfyyn= z;^a#fwtk86jR9F`!_$^F)5ysZxNwVm<0j*j`&C0_;pe#ve4C!$3c7H?j;uT7l9wI@ z>H+f8e(F#M6>QpEyiG7U7lj~7fkX94U|$OviV8YO8A)U?-ql`)Mr&5?y_c6znt3-3 z9v<TU1o3PtS7GYn?xHE?iLlV@Jerv#A$LPwk%T*Gno6B?u+0A<qm3OtLe-b2*wnfP zqa`#NJLsxb7IB~KuIAxPxn>fpZSpt2#(s$O&qh~JjK=yehe9S_(FXmXv>PPRG)zH- z(;J|F@Meypqt2c(w0$-6HhqZhom6aW0Ba>kF&#pcR2H7jf+<XM$r8yi%til|@N$$1 zo!5Zi55$Ym<(rKW#X_Lk$9)10N)n8CSL3fF$W!QkFCaZPl-BY@0nv4u7NVO0DDU)= z)kp(vr-)bG)sRIhyP7g@?(Dm$YY@MGmSsJ}-<7b4YHv8+9iJal@1@F3zVd3ySH3#1 zQjeqlLC|Bk<Jsli1WOONsfyS7#==`q_ve57LBe^gwtb|uANI+I7(kk;gVZ0QVmq{z z<dOtAtL40p91WvIKv((bf`g5Lgv|%2$P^?7JckPoHDO?TQ&yP6?(!_}o}pO{UPAa< zY<V9cfQ^AbXAIG=)K1pm?oYKt7ZjB8&T;QY?L`QdIz#I9%2-VZ__zzAiV$i_9Wwz? zR(2Wfk=EtPVz)w9L_J=nrdY1P;WR)9Lnf0q?v9!P&G&b+AzGMr1pRd0#`N;7B%&Xb zpZKPPs`Z<4T)!E-Vd&2dFn&B8)!_sB_N`w(-Hg6b1A^P_At@F?52O$ju8EKQtRF{! zeP`y-vUkxQL^6!IC$Hq2+OCQ&UQEb)d2hd<2BDT7?Ws`tb{vuG;DScq7qyc|87WgB z{iD{dktH1(2?1AEjU#u2!M<S7@-|9sTRjpx4%0*bn&`4D)bt(auabUons*~9FCayH zR-UD!O5C3bJs^$%U7g9y0h5N1WtzV51ly#{Pf*UV)gTFwO&f}yu;D7U@p>^nG_qxj zfJgI%a9zcU8O{~Lx9UXo1IRWUnGAv3?doL?C+QuBaqxQ`-&pFrsZB{S79t3V9I>t8 z9!BG7rw|r*NY!`_8U>$N+$16A8wylIS0cWtlF6nUC3u1)2KW-ih|1q}_g{fqyd^U3 zKpjT*I*^EV4ON}3cZ>1Fw^5kEc4GC4wmnuji5mu}%hpf{Q-HMa<2fk@1s=M?i|J9! zx;hFK`s!IehBG9^?|wRrYg#~;T&5Rr0jsr^Eq&1nl-#*bmYO^7*ot11qEiL&$W~XL z=1wl<A{I?*EClK3ffl)U<g04`k}|S?jui!Y>F*9-g=p@+0dHT7i0m|L4t~WW^)u>r zMD>LjC$~7vhq%46PXji<liv9^g?}isL7ER6xbqx3GqSB%BH~G&aWztn^x2AtI8)R$ zHx;hzJSvjWoK-qhdP~)wjU?qY+_rB-anJI^uj9RzH!x_VB@0_qH86a#>~$+S3i6sA z2^EWCb%4`>tlr?HP?ULJ7#o=+&iWFsKzsM&bFjYOwx%N|lvJDDk+pV*7@mph>5(12 zYZC^33Kd?+sUIM65G#Q<MM<uh^L}=bQW`C|M#lYSPE+J{<%9(zb#||$vF1>1QXS8i z%V(v1y+IVzqY0D~Ra(h(;NJQ=EqL#m4!RTh3xIn^DZQV5-7@d32|Bms+9VB57_|73 zCNyYqCDSw~Y<!Ljig`j)Z3w0H<56Pm3MiiaflDYsq3x9RZ**u5kh;X^xH_0R0>X~^ z9rGup?0&W!YOG^vHe{f~J@kz}_k)~WJJCud<#g{5;e#xjb6e!+3m~lM7;5Oru%jFu zpS;IvK`?_V=^{HW(a9@%jMqorn<uE_^zWr;5*^MpoJT!}vRE0JMso&q>{oH<iK&}< zE##bqf^Jgg*z+Nphp@DSI4y7T=eYZ1(6^VR(adX#olO`Gh6HF#VjYOkZa*li_OG&u zXSic#QSu0pb2+$yc{d01P1Ke%3pg~!x1?(!FKkAIl*w4mCm8YyR9o1Bph1uuQqT)V zS<xu&d%*-tJxz2IPuZW5y(#a*ysaW%$v_dcyX9JP3zh~02}X6F!1p=Txzk>mnF~z; zehRk8!!RQhIa*yd7f}q!_W5{pKHePme8}EwIfZMUK`=TaigA(c8DLk52_-{QcyvAJ z^$bMRhUx@}Lj1|u)uUedYE9;KH3L*%x3`n0O4@^}Ez88A)-?c$?W5I}B3VVDzGi=V z4!S7Is9k2JH^ypB%4hG4Qv!VYQMsi^1CSL|bFwXnpoW&wunc(x3Ck3tGC|$bLMxm+ zTq&k61{`a_`O@;7r8jMIZrNs>&#2WT*>2laH6}gUyk8vzXab^+vg~SvnAcaZ9}y#6 zjZwpgUihTkS2$2+-OSjDXbX6B3MT7J=%?ytTTJ`wfHsM(b&{$g;#A>ly&!#UbD?H+ zqwFe_c^j5h7R4R5>T2>sb5>mgPvod(KL%$CqZ#sruMPKeeoQ1>;S4IS=-D>?@U#F6 z{wWU*FP^i^G&h-+e@zlVvtnT2mmpnqJmux|>n$+|ybZZMbL*j8$A92w#&L;v$A1v# zBC(ISvhMlj7qNB~UN0OmX}YOW4E+P1)H|GSz>Lniv3hsBDBrNgdC`9mSQ^aN?RjRO zKWs9_n_1I}G0DRTBH@|+w4|Mn{b4;?6AC%%cEV5=-FnNMJL_I|SbgsG$`E$22vooX zdWS9PmsXKW)@(4L-eltoxRNa{hgyp%iiEJ47RErMCrGRz4~iqt2g~7&AC=Q!d?tQJ zR%hU@&D$-Ot4IiXNJ*awlq;({J5d)Q+kPmt*6(xnWsk?<IIe0LvJ*B9*0sIILkJPl z&tGMG#TqzbHl1?sc4EBPDzYaW6V**)@etAgHf1p<Ja9Zc1Op$WFEe{bV-VO&1TI5G zbvS`sSZhVNwAYmZ+rD^{OAD?pyr!5K=9U;14@X>mV4ij}#R0#>t|AgyJT^(RRD&1$ zZf6cwe4`=>LhU8-v#oP+2Mgnb4W?0004^Ot7@sJXxISoptev_b_F<|q_!#M8$_O-z z9*a616f)}p-H06^Q>X!I3un<Q&y&Z6HCrbZ^FW%yN_&n<asH!*sWvP{8*JtAqR4%+ z>Zs|QJ37ae5FWJ2PPMi?!P19L%w5oN+ECMX_pV!d6c*I;YvWa1mg#6$Uc0Q;8ZJgm z4l%()^_VMoHjplxRk@9Lm{+eU3Uc<@)Fz~&08&$BB!A1+L-yESiF@XZ6rk?K)<&}- zUztdGu4$%wMI+v=$|*7=#e|Qyl~$}3vh_Q}pd5HD1v0u4(XocH%@Kn9&;+(IH&kjb zgL%&g0zT;VBj3P1?aOU^xF=|cg8{YZa0Kfee9nw1zEZ7_fB@%Viw)IKMES@m%tj$d z=s<i|k~-@FvWvVpuk;NI-=w2^H+O<P+7M)1ZJ@OR;0TwD%L&;sp<AU0D>6ofi+VLo z%d)skdWtpE{p3NEz?wr56P-ZR<^I4`PdRi$o>(V-Cr(p<d9-A(5e(8hIzr)-3&U`2 z9@eVre1N?uGl~kJgo4Si%YhqIKVUA7tSJSln~sElVB6RFil=JWs4jXc49J*fchhnA zENR+(eC?vD)L|ONK){9l3kb~4AO0h!{s(_BP_Lh6<Fleyxt*iHH&3-0k1WSfTZs$_ z4TS`Nwm%`(!N9!sG<2>_fTU#3YH|FY>qvX+CiIB}rEsU`bGyAZ*b@-MRg*WX!K``D zq+W7}r0j((W!KL$Jdl${aVlC${(fZ*Qa(zuO5xA=9`7X+>&UPxm~uyLcU8@XFY@hj zHmrZcSz2vh4i#UgB-OF77Nq9)0o(yH-o?m8JKL9RXu5L!KJMWAxP$NG4&ErTK1OX; zva#A_`?R$hzkKl7+Wn0+Nc8f{2lv-M+57^(e)`$w(&hvF`o-hTPal1bUmri-c=X9S zemyz);^eav9p72~KK|_f7gS+$>C^jboAmY5FP1iz@oV<karcBF_P+S^<P-e*`Ge0s z-=x;R`1H}^$B!Ey`~gW-=(-JiVh=LA6X&PZNEk0(52ocjf09n%5^&#=mwji9Cnvqn zKk3rf-e<?3oHPzvb!EjQL`Lq8l*d#LOib?d8FeR9Oj|ftvNb&&^yJ`uWY+f-@;!yT z1%*^7o>r7pKoK3a0kf>3S=Ot(QheN?s4?Qz5o_$Ggc?tj{2T<-2cdemHq^eMaH2Ey zN}QJ4-d+yQn?GH9m{ctQ3o4o)p+QX0b&DAtlb>oY1z9qdmB1t+Nz=43e^=JZEDah| z+bMllV;9P@73Ub=N&DbkA&>H2@6+Q?Kg9#$vroQA(<M4VKK-Ki`Db*3^t#I_ok3ap zS@y{TdwwhG(EE(eq|Y9t4?cUqUyr{yrqlAX`=5W29@FWw+)Y2{GxXyxPCmZRhc=on z+SA$@tnD#32`(+C7#hK(pLy`Wbclm)cKpr6(Kp{ce*3W)%A%^?CBS}kmk3yteP<>j zFjxv+>cFetanpP2L5_o|2Vv?_rtlx3LllzL0g|h#anPihica)#QT8C)-7U5scSQ&K z`!R-T;=2AQbC?fMkLy6G-!n5{3)BX`y&Eq?gedGUAIwE`<J)LTThq^>*@KPrXG-@F zNYTIjjEFu&4ym`tU_gX5MMpb=nK1!&`o(GOJIwc4`?i~ov(uuSL<=x*AN{Jy;JAPc zQ;=n*uk)?zj$%yaI0>>?<5fw|Pdy40&?J2N4p|eo2O><DORnBxmna!=V|nS5rCO$C z3D1u8#~*+3C|1V7nQ6d8K?!o^*oia2bH6R|plx_H%;+Kv<=#uB@1@fBQt1uzh3uUG zML@d0QQUIK8KS<a4w(!T`c|)(Lkx%h1h1X2(z8O*qKx8ua&*y`SvGQTMol0&G66@f zCY=I%tF9gAYX1^2<6M87oxq;^xV_w7qN{x{MFW?6^h2ioVgiHHtsmY^LZaacQI2{} zUEuNPD9rl4?1Mir*ClK;4YxVHrcnVJaSq#Wm_x=(K2eY<x_-nL7@Q`PV!n5*UJ(H- zLgu<lkZlxkJs%7O{i913a{YlQw7Z+n6)GJ*S^Hig*Wsh@57)P!ym<Po<-vn1yv0VU z=u<--jpz0z60ZKqHX%!X;bnM$ku9EOeT6oQRdHT+vv2UnFGP3vaf)FTIn17Y-BTw$ zUGYX<QeApsk1+a1%lLv~S9a<Wwed|R!g-9;v8e-&@u{HB1-i^OYx&{wtc4V{G~WO2 zfBN5l_#ghuAO8LS`1}9ruRoB9b96RU&fc(ysTCJD`~7@`5yGufho7kvXb0l))$}2w zQWlxQT#a4K#)S#SWCpI19+wlFb<_YkKF|9-XjaZ-cYM_%@PU;QT!=2UKl$YT{m(!C z+_O&t-)gMY?p}GXCid0DdzG43rC;mVzVd4o&{(Rb`{fEC@$dfaU;Xem|M&0y#b5sL z&;I)N|Kx9e_|O04@Bi{||Nfu;ORo}i<JeUC;otm|-~ahvO{=5|{oUXG3tHme|EvGv zhyUk){{8>@e?rm!??3zDFaPVm|BHY3yZ`R5|Ng)EXTSg3|6YOk;m`m4cmMh?{_g+% z=fC^k{^CFW)xZAXFaFIB|MtKA-JkzozyEjtK`aqQr@qMciY6p+OIyiA8Xv*y<Lp(g z7@hOdDW(1tlU4#hi!1tltbX#hSupN8mDEoM<lVXY!IkcsN(Az*>BrsK{kZ2;(nGsz zh8F8bw_j8lpr|yK)Vpv|(`!hdnD)Oe)gQRUMo)=GyKT0g`yN!Lwypv`&cImDuf#9h z(jS3o^F;=Eu5692I(8Q6A(wtC!nvSyq65X4&?$I0s>5iI!jZeEz+FOiCUs7y=yD4v zyRr3T>*?0s=JUhnn_nyJUEKd|)jGJl=+Gc=t2@q5lj`D>lM1!?{K6Y_o0AH}D-70m zKF~q7KZvD^20&MXJ_Gx}NfxbB_rUwID8_*iiGo#Szkd3psi1fkRl5ieUxfZ5Pz`k2 zkeYU5`{@q;c8tNHTdQ2Kjej1i`L{sTsR+W;qUu`1q4d0$H;V2#aNUYG?PZu#;vCFE zlMmZyU(hSM_y*a4W3U|<osXsG)WvBV-W%J8FI*!B>>U)s2wm%EdB0~n!Cip*&%iK2 zVaAttHn!LIzTeq|N?#$D0x2pQ>Q{rC<%kiwWvJO67f*^ywW(6D>8BVEPy>CeWe?%a za!n71AY<moV1VP#lQx#QI9>6r7I(k>TCa*@^nIvUmP%X7*r$LU)<#Ia$tz>VD!r5R z%RTZ#(XYs~Pq%GtIS2)uyAHMTvjQG!eL*SF3D8y5wbpIqWF!!n`UhrMfS;RXS(MGj zRv!$M`T!|JI5gSiDF_mJJ2M1-Rt$>Js30h40`$fLgGZ0>;DaSpD7!Xh*`nFtcMXgk z+i0u^tcO6Jka2U7GHi;74>^XN!YJ~;SYBHCN54P^5QxSjD~Pm1oMsE955?_uVJ{(V z^U}^uvaxWDx#VQIv}5~);>E%NU?J+h>41#b!JVH0g-Tg4^m}1t-pOFc6-)>LGDm?< zNty;!&(dd2>KPhL>VZLVlY<)zal2T`^jsG+omm@=(C!hfSuwLUW^YnxnNiX!9XJ(+ zY(`lTsRN{G5$lZJv#uv^7pPAiIby1UWjqAO_&w0vGet(I1U2QH$_yO;PNO9PYayrY zO7L{ARLBR~3uoi;%ixIM=W8VfLND`yDM`wlrgK`#9;?aTrPON*UD3l<Z81z~#t?KS zd0~4fdZJo1jt;BI$6-89JI2rk7(NQVz*`G-f9U{FHHfthpGjSv<@!3DXg6N({z0sY z&bu40^WeaX<RN|XuD#_JGlo;4u%EVw&at8%@NVEI?h<=A!LAwKdXHrjEoz{Z*p2Qk zAjTTYPlrXR;RK7PlME@D-?YqW-Dvw$GU1k!=jj;jc*p7m=!?LhH2^B~3TT~6kqa^R zlTKgVRU+L8Q<cFxP>n$tWnvkrw_%`7ZN_WC*>3xN)=8&WsiI#7FtIS$=NeGR9YGBk zfM`o+>(FPBB9>_YuB+ITQnB)Xa$b#bVvmAVID(o-K(~%4UZ0tM#nA--wE`wdyWOJH z(H~_$&^<bLRt<S?!vuR?hiLiRe;%B_`cr#w%L15So}l=z>1n&_ru~egkxLw=7haB2 zukNU*P{SLmuHQu<_?uFbJ`|)yep*odo%9(Ip#eC%bx>L3%4uL|RPANe2!s@!7SCCP z3h+Z67-@yi70;%aY7_t4+mA5Fbp`dNWCCYi;S1D7u@Dvsh=CtDNGnK4{eemmO&}er z(c!=H@*s>i2=kbYUGAaaMB{|%mSJ$_G=co&ZHKw`M$@6g+&*-AX&<}YZ3UDQCZQvL zghdPrb%G~6USrA;<G3wv;IrM&(47L})(^+HV-~FD60Nbd;Io===^BYfGDyI0njbo! z(3i+5ebsKazg(oRn4cMVqDyo&Fik0ZY+wm}p9D_aySoaP)2gEIt38xMddVr;?Dr=G zf^$+bPc}|ZD|AEAG1%Jjz_SAbNem>RhHqp}jU*@i^wj6Igd`+Fp!gv*+%`bQZAADs zBRIK`>>8L11sFWhX}?|^r_K{5RQtLo2ZG`NG%6kqRIUJ6sU5sNI3_-$K`ekax-Mxn zaXI|F*YIngZXq54*@~z8dGCNm<JFE{8JgLzRZ^&)-EvMaV?^zAxO>R!t9Zp{&os&} z<p4dI^UDJ0?m-N`)Ojoi<EarEqJ4u8@LLt$a?fn+f$=0~<et_zyr0?l+@g!><sY9b zj*rlP{_Q?h!SR;++5<I<OzT>+$+RAvq|7xdjyw9Qc6;_%1+nDm>wA;>DwD7Z4IC!q z*U4Q+=6C(TZMpCLa+9b+hC`eH%YgVIwPR5W2&m98J%?ZG#H{;Cqx3PlhZTSX3j`E} zHS5+%_T?yH0$R>^;0aqDiBM+wP#sV?8HhB+I<~N<#^=`47?_o(kQ0_Hx>|!1HMCvD zt-u)qKFl~JZT(an9;Tq$pXU3Qo<bDD9&bL|*n0N0Sg;VRYV3;@zXbEm;)j}!{pD7& zd=M=Qg7inLwl9k{FhO{Zmvi-YB*|kvg{_ak(nApm<T70qp21P5LTW2Br9QdVE9CHk zoOZ~X&x)hPDia!~jn2F~Sr6j0R)DETqElI{7K={fC?rN@=&82gh3UEZ5O^I7hd<vS zpw?XDC{cw_Ka?&C8oMAz_`-#&(TP(wO$&Xs%VdtalbX7rh9_QM=u-n25tUCIh*YO8 zNzZe4qCh+>hE<_9b-y^Rfn^9}W5LtR$+|e_rf3wkN~=jU*u^$_H^wij#~8MbKjy?S zsyP>yQPFxlqxZF#)}05{x^=kLod?;vb=X$UQMsInA38Dyj?-T{P=4Qh;Wz*UEE|~c z&ADh8dVZcIqpbCH3r1O~>j=a0Jd|O%jx@|uds0{X>i#@!->++Xp4yZ9^Degtt8W%t z8?oGYy)$DYgV%5LYtB{Yx^pc&m#trWF7Wf*t@E^9*0o*E({?$jS$=Qvd7Aw79~J${ zX)(O{`j5b|rmB@8=(VF6KnWZl2STF(hQmdlol2LDi;dI;>^r024*jarIfkkTg5S}N z13Np#r@*_(sKssW)LgjDWoPnkTrReDw<Ff0l&(o(+N2%NjG?9W26+0oQqbkr4eDov zJ>t;X;h#SZ{)mN*QRRG)p59_v^MRHyezOG-=3G}*ZLVci<?2^ey+jg>;Jw+x;dE-& z`btf}%g$Muoiu{ev$jr9bW&Gz5)@^IeJYn;!=F!Q%4e;%nj(LpQNwiOb*|U;)F^xE zbv+SP&yTB6imQ|yP0`gta(6xrlT&TtSDS=%=Vv8>kE>C7i<5Wi>fdDHJ?eTDz1)Kj zn8~S0D^5<}R`|+&Ib|8+#24C{%-PnYuB~A0>k0i&fy;5+`cbhfzLD;(s`1wAV|$ai zFcS{&n;|CO4Wg~o3B*t@&-<D48a#rk&f;P(yU6+w(y2WtUa68PdIn{~doOkudquZe z{4P6Q-2L|JMKceJ0PtP_5IV`4OS&vXzkw<rXWJ&x;%d^kA_HJ&dFxj<u-D_pIB6z$ zNF^2cNsRN0tR1j5#8**Bb72bGo11%lH!PNBFH+Qf@}95?P3HztM#@3iKP^wSQFa6c zLUOdMx8l}trgYs(mX`t=H3xvnGyqrk=jZ`T$;<IB%mKm3*&2Yi`?aCF)q$j?wr+1; zEw|J-TdJ$-?|rv6-3{rY=uf#jE=F?q^Sk)5!Y{=h=EW@0!hYsrdI(|)-JoBtWkID% z)!lGXDQ!iq3I<o4@Nm`*C;E=i39*XJd22=QqW9ADsttKaueMK`jen$8^(!SkC$DvW zSTLeh{A$SXn-&eG&=W+6WKsRQ<TdCyftzbas)uc|%h%vIW9Nk>u%ud^)jAz}3}=p3 zBl3#@80mF5-7w2K$W3I%;n6*GFJ=k~(J3H6#flG1BMn@@7^E^YX_~3cj8Kep52BaF zu7P}n(zX)!YMl)OI7&KwgAlk>GwWg9FzP?uS5GYmpLj88n(qyeoeisIyk@JmhjmjB z3dM9NIRTCPc{SE1yiTEMcD#-tzi(q(JMpQPC@3{Fz!tmkK(QkCql~RUZm6=KAH-PN zpp@eLmz|Q-ulDG?I&1EOhtq+WAoyb>a45GT!jGorw57}u+{p$oA&Q35U(G00<Fa@O zlDhupzPLx?WQPvc@9>8%QH;`VKAyl!{k0{Ux+mVl?`^LPHy}#UlbUi6$S8%d^tv+} z>Org!lTJ};o1S=O;cYh^>4NL(Yox@VsOi;3{s_K%Wd;g=OLT39Quyl${xVZh49E9Q z(m~#*nNU9;(=Vq4SoG=laE<J}MUylATH2q{`%4nwxxjby%bugaJ>N_F`6&(sey!mT z4Fg|+cfh^lw8~f`hvgpW-@)NJMc>^58o(*iw}vL@inCu|m70!Z{gAq7&l+8t>{T}V zRD5hA#iI7XY+KF3@1CW-;?fewIc!kKx<y|S+G9bC%2V2tODzwhMF8-7P<bI{LtZmA z`eP`N5=Q{wGmbp#?Cq$18i037wcRMRO#v}X?;`0GK=~G3acYDO327w`NB4%{)QQaU z!!x??jA36ibpS?p24+pW*Wu$S@oWrTwHTN6ALp1F1t_RNzlVf!@T!q$Qo15*?IKyI zsYfrdCR1C3849f{yCEvQ3xE@+vwy*B#w}~$@btA&gca7$Tm_R?*|D`Cn)^<wPLQUo zNE_H(s_R$O=X$7i>=C$BSwQA-Q|hpNyBhMO5hsgxhy<sFOMC?(CCCRdU&u&x5wSMB z6WrlfYHm$2qBXW#Fmk=L3+~!-fli|93jEZVtzR9~&!p}}OEL;UeQQzfc^T?CmlPbZ zDf%1-hrt@0>4=wwvto3X9%tjc+i>|2Xn9Hlw%T0H92etpF__T=o?;v`9HQ#igXWTI z6|<A^{7@B%p93Za#1TMU%nsNP!Ec0o1?~WOh`TI<!vQ!)d2n9OhxKzK<;b99h=*}B z)N`fB?<R+2!e!oS<~^u2je&Rk#JIQnk`Nt;VR$dl%^5xPd_5q?Tqgn}yX--To5jcY znN>Y$nolQA{k-fy&rUNC8y3~~-!^|+ee}y_`=f^ozg+--e6x0jnyNlA3DtduC<3T- zC<$fV5vjNEAz5L?JQKO0RJ%_x$pKoja}uzU|Fj42s;T>FJW^GIx?Vs-)eF6ry3GD^ zAOw*KKbq02QqlsG<r-^N@1<NAmr~d<hFxYNoA{`oj<YvtpZTUN#`xo9cKK`k_xm?$ zSY>{-@Y`l{|F`#kTP@xD;<xRrb?_0)`)P#FoyHjsrB2c5Qmv9!7_NyKwKi&w!M1>A zed)?<)K9xvbFsAuF$$94-eh3i<mF=nVS8j`-k>D|-JRT)lkWzy%VIVW^wtio=tV;$ z#LCiAb0B{>g0};LeP!<@%OHJ!MN;`aUkKO3ykNFff>4t9N?3Btc`TfQ(2QRgeMfD! zV@270p0~QpF<D5E6heQ{MYv(mv(c&Jy>;io9U13qmNRlN^8mj&XeH{ued&e^(Ds)O z797Tco@yW9gF3*+YTz#;?`T-K*M6iSA#63{w;7Nram#TL_4ttnM!<Pov0>eEEfkUa zk$S8xYcJDs*lg^oh6IJej67b{hCqis-w=pG*AJq6+kp(gzJE13>9sYMqcxVN)VLq5 zaeqpU2hkc2rquX2TI1s>H9m>f_~b@AB%V|c!{8`(@kyb!to5%S_{v?ygOEx90^@+M zI>cm@QHqc`v}gMMKmKjiSvk17pg5$C`d>Z3pLklkXLXGdMxM}G(vF|bq->M&Hs`Wk zmEm4EXSA=pu?a0z4WBhMZ9gGxwb60e(YE_(HQs`|9dWNAC*-+=%-2v&_a$C07=VzH zoC+)~L^Uv`NBb@qh1drI6hFLw;E(F&x~d!LsE8FcT@;F-T744D{tOjT9A<c@;fFCY zOM5K?X6blWm^romCEbU8buC?|opMy*nrykEZ^$U?gJ`6z*LWhfjOQ10)rpvO(<m3t z?f}?e3PUdRi*gLc9my#Swk+(w?v|#9esNX$DVRPC@?n!2YcU)VRtEx+X5~e;2C;O; z&%qiZX<R+cd*7@78x|rrx66R{d1mm>!AekW$A&-<I;VSbZrTuuE*P^Knc_7+;t||6 zlXJgS2WH}dF*D@Au}@iKU-9rw?o_DLafTdq1N6qqcwC3F6q4=Dub)2Iq!>XMiOngZ z@ov8ujSr&Ox^F@(9V8oc1ISaPHapnV&<`g#5{8+rlGF|S)UJn5b7dRgx#JdM@t5Xl z;<!)@O*-T@oR}g9GYjgF69@5)O2mS~(Z0riab+<=16#~M@MMe)!jb(%^7L_;VOTc~ z{K)|~S_Tdqg3=td+z=5o-wbp2+pldcZ3`N$ro&J~>Nz1HUc;57M@6m)mX2v)f_R|I zoqz}nMp1YJ;3Aj=ln&?|<aq3!_jRa46EkMm69Gr^iow%52gH%sgm1qNW4i@W14Vqg z6=zqDM94r3XXSmCzW33--$(ntX%4DN;WmKNhKJvz)tO@1*aTHE0}zf>aSICOE9zYr z;>Ez7Ijm`yGF0iH;A;I|DrhaN*`sWn7(u5flEHcR46;6`L11Vn6Op^E`!YyJ8|gUR zHC=iwub0noLeyS?b`@PK%3cS<Ss!uhZMA#Ul4YgpzX)@<&H=_DvKL^h$@x32o_Hip z-&7At5nra4UuVT1Rp5Z5zVE{Fq<u)xR0)GNE5tx=tKcp=uaIYLT4>?5FxOSHVM~Ra zGJI@SbyKkI2fcJ5?ng9b9H1J4U`FE0=~&)w6S0C~M%?1!g(ucVBmgrmucvBVAJs;X z7awmrPU$bmHFw106NUNbYCp!)-Mh={EQB+fhy-s-y46mDb@axJmFHO}fV8{)Y`(Rc zXPtKEULUbSbl=7umda{+6jc<n`j<+y$2Nbk93fq-@O0trZ!@fytlD{S4L)>RBaSUQ z;vhdz>mVoz$Z~kNvH9)cVH6V}&Qv*814S^c#6*l$WI?Zo5-Y}>S>2gCzb)ckczS%D ze4x`{Dvh^+mIl#u8~sj8NH1lcU1V|M71Nn$_@@@WbI)hZzel0rH&1<M`m*C;gBWQS zkA6EBQ23l7s)BcHm}xBQ#jkwGAl;`OLK1aMAQZf9gi{cMgCP>Zb7CL^fc90qNd+9Z z)5kp4W`n1&2}{odybG5?jfI6~xHgJ#ZA1LX5%~l#W-2yC$$*Mw`V-34P-Z%IpA+rd zfB;o+4pG06N9aRNYv7mdp|{o_pB3k)XHZh{31bAWTG8SRhom)Os!c3$glYq<sV#At z!AYL$Zc(I5#VAYZh{08mb{m=zp$0XZrlVI9EJ$OIeFxG37zEKW$G}y#lJjxiSJl$K zRY(X23dQMA^mQH7cQgjx1m+^wA$HY~Gy@wo0$!ZL#E7`j4rWI7z<crHy?F6nyqH70 zKr==>Fu`V!PtFy3j+=g!*#>@=SDK`v^`8DA-jiYn-_lp}4tMW5#DOFm)5Oc@<Gwd% z*#+aPIvzf5f6-=t12T~q0bv_<^Gt+oVrtic;t$&Q8}eM-+uYqdeEfXv>E_|{&8KTy z&kpyto^GnT%cjWA^R4aYTYKLh?mXXK-`w5Zet!6P>&f2cb11ch{vS2PIQfq${rR-Q z8*6)OyL)TvzfuLim<n}oZEtH=Q5IG36KdZp`|Z~5i?t`5bf~76Tt<QY`niI;_3fr8 zzg%1X>nGcf)`0TfuZOw*&DwJS^I%G|+iKdeoNe{NC-o4wpYCiw+k6JaJ|?Ij;%@!= z=_Y;~W?i*y(sJ@JUjNba7rWooO_xO!GslkaKP_LU9uEY2?yPvUic><?)T<h=rQce< z1pd<SoZixOc50A8G#IXYSSc(c01zsKmFT%Bi0d4d!^g5iK2AjX;}?;wXdH>Oq+f~R zh!SB@d~7dXwKnA`oLQr?05ns~(gjdt&@^Yj31mtFJ5nq@B?~)hB}eLv7=abn8qz`v zu6?8^d;TgrA|(q2$yMZck_;ehF`TWK?c}0(sScK5R;lx)KWXVB7DH+S0}6Nj9WF=e z8mBEfde1R2*HcP7^9!(V`vn|$12Kv=R>p*J5E561jfSJKo0jDSf+*8TUy`l|xzm24 z<P%^RFTq$E*jvSooMv#|LU|0FE?Y^Od0U<S5Sl%?1JL#rg+(&UxP7p|x7XftAnu2U z(6g-}O#TkR(^=NHci<d7e6gdb=H><!c5;&Gf@|wokn{HZ<NfsBtF?Q-S-SVd;lbSx zDS?krbCWLZIV+JiVW{7!`NY7;F!3N@H7S&NE1}zr4K@9Rnm9kfyJcf>#z^!#!`z=` zi(<`tkyP<|+y!bQulD?A^hdw=F2!S?A`0g{Uyyao84Yi&f2g&^!6rkF*bH2HUyoWS z4D?Wd3ozfoUR3`G!bt0?2>;}~Zv&+6(?lM~_`B2QN;CDrr5xvaW3}L)J%yv-G+WOg z3Alfe;k>jNbWHC|U-)9eDKCP%BKt8Hx-<$fHP*>NYT=!n%n$J7WFCma>;_%T1jIpp zl@FT^+?JUkRxsDh{pLFU^=F+QH+#jPMcLm!{BRlXf2{5fek7~($4wKYu=DF|%vs|V zTMtP3fw3rIm4Sw;ZQDWl;K%XRCFm>btM~Quz)HhO6p0EeotsNP=46q)=T>4B@yasE zdpCfD(d?tZxb8Q5_Tt&T8<qJ*s`DMY5D(SFgY9r(Q@pT6F<c5iNA?&m@EYB8c#&4x zGQUo<7FKMAI=}+5{0w`L)z{PHJmg64Nry|=$_wqo!%aNlccJ0OYcHPc9V+tuY-98J z=JUftKn<_;IhKtRXX<Q_zoDu#=Z?JY5MN*t@oPTJL6wM@>?kIvGaW?Ww*DFEZc9$% zg_6B?@~f<m)*4*de;no{z+!cY3Gk&cg-IOUq@0`0D=-sDK3z&qMmjr$oiZ)R&52-; zm1{iiXT9H)kNfB3o4}8n$2q$FwNKy$I72M7lI73bpvPi!qL!+uAJ{;@TtJ;RIteXZ z5j8DA(G;u(4cNqB-VtSZ&S~h=HYA%cdHGr4dVLWo-&b?CNfp78!gH^#9ZdU6oonzN zHO>yu*S8(6mf+`ZRabbI!@hkd<{7@%<a&YGa1`v_i8^ElbGNCgsTxmDwdI5l=xI7s z=Tq5?Z{K)lHo?gjv?FUfTXWI3bd(zcN8gEMkbn$BWX|JVFnlS%<z%4PG<s}}MasM= z^YH`<0U2qgt>id%eUU0+4}~EkDu1=a9R5@aLh~Z8&eJ{sQVgOGTkm!(2^oS}P$<eN z73Q5xg=1MdSlsQNW$?U=p4Nwzeyt5)^l#>A)vQt$Y`<$_V76BwC#pmKWL<n`pm1~U z!{)*b2I#WNbK>nl*5N*QT(!iMEp=Y1O$vHr5*;NhI83qsm0yW$+A({V;v`031OU!q z$q_LOin}Fo8t@gy(q`#eKsN>O{1K06LY)TJ(_v$5MCl|YMDCqqf-ySGQKt&ite$3a z5TJ5YFG>FP2$(wc(qT@T{5IS=os^1sZRjN}hWOj>5OkQ?g&t0iIz1zuX0F;?!RXYs zD91y3Ty3}q`y(5#RXZi<fb)w?N;s_AC5+es@j(rNL0fSG+m(e(=abb~`fr=uCMvl) z;;3uvEq0T)zvS4Tq^~Bd=5QO_cU1!i;bmstr;tst(rtR{+~r2q7S@x|795q3Q#f;S zoWn<IW2!d&as|!Vjp?i-UP-U}(Spc!?Kns(i3^R`>ji0bSeJq*7_TB|HRy*fio6Fm zg~!GCEa}@NLIs$@rBl4!$cJ#FmLT>pdKw!Tv)JwPp^_Hz>deC&IDC*S8+dJZeaj3K zP<uIIW?a3$;?~Hzc->$SZUDc$GTt#M-o|>IxM^bGqh=*yOo2Yw+}FF9a~YhQLSqQq z2;ZshW}tW6ORJYvllvdm0>XA)Lg=9WBys|2OD_ex`w|tJ)mhQ^lk^2mZ%9BhdO*uh z3kF3ht>>B2<qhf;V@}sFMM;HR5#DFFlSf%MrOU{4OojA&u9%0(Ao2c+h!*=@rO9bg z^eDe#rsN?tsfv*5JMWXCuXFQ(5)|7CfWli0ZDI^|!!BcvvoT!A4RwA*jF85ip-Wc} zuIVRuqu_00J{V-ZT)}}Hj3;pVj@9gg<+I83>-5XxGMVz|#X`m;x&Sg2MC0DQdup|~ zH^i|d>RF>DHd^4;>7&fISZ@Zd&@oGh?JzzA)7=)NFo9Mh_CjLc-&;+ZVmB%F((JRE z5M%5WBw&1%ETJ1sj2<~{BliGC4Jh_1B<kng95x)@(-I*~3}HYX$jfhgXC~2ukJdqu z7#+(t$-UX^R~T}DdIW6CF<ews=VVC9_*cPt68GWMKIVNT>w`iV@l7B3auoKOMC$~# zMZc$}0yfg&AWpOK2I{3E{DaQRqqSEnwXL|ioj()0omhjvDd%YKRT(t*0{B~UN+5pI z-1ju;-T}PY$4|5J6cSgclaJ0$T&4OduRqYn{JL@{X}8<f#@|BnUeFPY5AhlKgu7lU z+M|M6i@bW{xMx%cq<k$()0-(?A5~iyTE(Moir+`ow!`4UYCbb1<j}>CnHPnR*`J={ zan+8kpDifYaO(U^el(`ZBnyXT{BBB-8*#6v^Re<(9q>btij`yp=W4Ctu|_kP;W@5w zrtr&}Vbu+3D~qO{57#!?&JD<JdY$c4$=PU5*%WgclV~do=`Vh7=i;N09tyIL5uNlv zd{&-Tp4B8Qif+qiO~bJH!5*B;0Hf}khcV=KzD5l;zISLb*J)3eWBk$}2sbH>JSUTv zmn3#TK@=C!u?T&kzdK4UaTq^0=uBj{nB@xPi`Sa+&|ewRa+d5Oc7$&m?u$;h6Q1Dk zQ|DZXr&pPs07I)-u<lt|42$y$uG2vf83WN7fqES+*YL)JWoHN0Nj8Qh<Fijtg0?$m z?^K>!M4NZsZXwDU&^Hw1yfb!VbLaWyI`q4S9PhD`JaVFHE;1|2qHH!;qwq^zNflIV zg>3+qLMt>Lw3m>CPCxOrmJX4Bk@jQVv9I^>tdn1hel;!I3xrrBk+W$^EB0=mcCP`@ zoHBIw?X2klhEki0zcHn6-`F?g_Bd~ARQ&eMZFrYr^EEtz5KxM*?v3y&j+KZ01setl z4>|^S&P*K0rW=f;fzuzKS*>8!t7Ew;=LKK)GCu?tSa8cbq-e;VC#=01j!lGC46akx zJWi-`IG#J%@Z!VeHjQFe#~7KaiGx3}t9bpqJ(5e{KjDcqg}s658s1~u8&%)i6U~NK z*a<V?+|KHYzeO3|SRsM%k^sY+ema2|2Ika8jZ!g6jH0$O&!1rua<I^@)ZuUf#@{Hx z=VM!1QPBc~&*CDpmh7tTtMbJ-@3%qcfB}o8drf8oFmSjr4DMEp)-=?nbCtPhV?HsG z2;2^Eg76@U&*1b8iXaF%b`zxAy*LH6&x_|z>eaoBY^sU*Fm3uOzLO<n2d631KXdNI z<#{;u<AP^p8eJ1Jds8|An)lVlXn5K<m{Q8_zt!bw<>FK)S~R=r_Ro8nxk#-J>d0~g zD*`g~W5_;rGz$54zxSH%1TSQzDOz9wv@(^Q`)XaP%K9e^!r%>(fS_rKkmmk*&kuyA z1q6U$>iW61P-%sJYz`n>=sMb1JPw>16@+?G<M8gq_sy^N;UN^;>iH4hkPN^pvSH$Y z4^gMQ_;ga-JAe=itQ0^_oOqIEv|`|qZ7d=#`fP$iXqY%f*T{<O;<RC5J!I=FNe!FA z%dBJG$H=ju6>e`S@!KM}p<zkj3(e-ov<k5QMLP>o#PTLmE;vr7BRfat(Gk)Dny2Q= z5V(J6BrZa=){Eg#buR1uCnH~P!JUy;jlw}q)vUuUAgmxpxWrpMjp^F1P`_5E-PzN8 zC}IowFVidYGQAQnpRWD-@Y#!}kI<`S3w74xiO@lH_57msW3|s62`^(IwlFkd;Q4AA z3<_TRx?~aLUTp~qiafOl%|FK%bHaO~jpPW;8=akLv2j6jyn<pA&!I2nT{;hF*h=zH zyN|PmT4J0lTuz>~DEYJ<N5?9K4PhDhJY?N7s=k{GlF|VcD+cOWt`@5VWqlk($Aa(( z8(He%eu<K6od0vwRy-#cFawENR$m@*f%QHlhzat`>klIeqij=xZNs}3mWb6P6pB_H zL)dl5M--{1QH%G1TM=Kjs7$P~!MbQ|dfVbsvHBJz=|GGi0DOAIpu=S&Ap-QYb&_zl z;-%G8c!kxIu;{hl<@%hbxNIM-!Y-0pDYYcQLz?j5F;8Xi9FXgh^&M389=7HcIn!DS zhI0=Z_lXSnVULYGP4;h$;=bNk+5pRXEq{|*gJ7c74~aTMA_KOuj&LdPQS1nrDfz3; zw!gmFFMPD=ueAD&tOi{V>_%dXAUOd)3)J^F8r}jUqhXO!lC*rB4QeSA93M~skuF>y zS#+X{0&@7Pq(8cgcU2$1Jhfp|SAStc_yQvZ;piHzMjXrUCKL4Z|HW5O6x#&zc=hPZ z#nZ3+p$EwBvWsdmA)c9}qJTowPiOM=0^kW^4#-C?!xEiI(PE_trdwkMyl|%mqhUq) z7FrjiU|kSU;8~Kd0I$DJiT*<UOOCVgWtP!xbXU8Lyo<?7dp(TxfQLy9>8wxjtUp(s zWT}Qu!e=u5^_0yjKXn^dey(fkV>HgGLxA)j><>^khc!*ci2h7mx-FFmQdrlm*~|f} zHM8XABjnwP=uBToLw9r)U<~SOaB6E!Ic_k85t7FE>gD4$#ouDL%Hv7w*E!|Q$hNMz zyOTP1s}V9vNUrHB$JRN#31&aU;9_Y%VmCD|_?|9!MV1K=8o{@k|0Rb`*(*@N@0u`h z;7%=@o<K@S^byp9**ZzIXF=9JZMQH&QKOX%GIhQruhkEXG^{rNb=#{)K*3)TtOQJi zW-wtU!!R{%BMR3G;R%kjesS56FkZ_a@Tt%iib5MN>R&zc^%)!QICXRnRVnz|gS<jT zwB+$19qFR@y~kgorLWema?=(1(bM0e6(+L6#8jBnRyZy5p6NP#@9~#t>2|xFG}+*p zGK(qESr&4&5$Lk=Mu$d4%V<}H;TE81qA~4(i(PQmpO@vN*((^O@v%Z1w_IbP9{p=D z9FV_uXUs-_$VB{p0~CZrKhGWT4|)R_{F=E1OiA7ZhAVI@uwOkne-mE%nNogCJE~dg ze{D$1dZN&}7S)RGUaXuHbS4TqJ)$8dI;#NwvbMSO)^Y%`{Hn4`HqYwayMdNX7evX# z)^XFjklv2o?%TCEaCr-e<Y!|0$BO$y&}(P{>05Y_2$x|d0FB9sr#y`r^EsI|T+oeY zMU2FP@i^z+s4xhi9SI18YT8j!nV^tznqMe!g@)A@!ex-DfvNmfMV7WEvUbn`G`6&t zEQV&<*647+jJ59x`OlS*|IAn!V|+W6h{Kv{0&WGhpC93UE3(PTcs(i+R1?Zqt>k4k zafYIP+dT0Iw1Q;utK<#9;B;jVT#Z)E&|b1>>3WAm<O<c)QBCUaX32mQ!~@<ES$?yc z+_@92)eJI;Y2*?g<B6qaf-jYDi<-Hp5#ssY*mEV{r!m*v7wZ9IWi?p5n4CFKcd$Vn z8fJ^8J0@~RMDv)oN_KEscv<~%P9K;(b%0Q0SsY(|Mff~S8WbrT&Kq+gXb&wuq7pgG zc8z*de>!;P;mj&RI0BP3j(X`eG7V5|Hmw_F3x-eK49_#W@tKvA5#-T|=Fpmw&)B-x zdOzs_lh!+TY$_W^paa_~c8)@sj`j?k?qF^8)+aZCvxm1hPJh1VH;i1@FyFU^5F;!+ z+-&D<HNB>qe`@Jz<(bwo$k<NX>A>+h#Z*IMH_+5^C4Z{Go>R?<7+=c_uO&v8Kx|OA zHxg33M}!OH=|`Yqm~hRtF$Sg*n!v^p)_gKJF8cPH&deOMXCVK&_(5)9J1INrPTlzr z*;ewJBilSR=0_S)J!>0rf<PaMfff%}W-<2UC1)!Hl)}xiw}-*?5PbyqU1ho2y>u)z z;_6M?WyG6R*>kPiY`SCfFx!sL%g@#D3QCKfhq_L<P59H93Dt#%REPPP=jkPuX!?yf zwo9D()FF5N*a8WT8&Wf~rM4JhVbiPSprC9~SLJG=R1JAmCmE)$DAh8mql*I~>!T9a zRc@~uM*(9eb!{s@wRLNp4BWUfb!M_)dbNGxMzM8vrQ2fMY`w<l?rFeD%{xa-HRqim z^1!{8xIf3#!1w9Xvg=X-n;18;uiXhT+@#!&;Y^@mpL7>WM6`SMFj^b-v_GXt^PNum zMe52lw+rR~DM<;I;UuEbSUej!t|Z<^1F)RW)pOy&tqysN(~WY3^QhS%V4frO+lND_ zv**_D+^|Q18qF&`Pc#r7GrRq>nZ9w>@(vO0eI8dmxrc)NVI8#9wi-i#PB;k6fhg8I zR%hOj4G@oYdt=*qu#2%~i?KT0^}68U2Kqq7#TScZreuUBxu{X+V9Vq2mEt8kP6zw{ z=~oMFr;!&!fI+n>`aQ6Tve5+{TLxmo<{a%+@stgUl+yrCS3Sa0Vr^#&onza+mpt0l z;5~1siNnwgGM)=p+8QQL@*yUd!|?GH#okqnY(Jt;!y$wBdJegZAO}2^E%Q?dtB-0l zsV=nITQd5Y5*y(LVK8bC!bamrD6kaU2!%O-b2@sBKQV`+fF+1b5d(v5s{Qn3ZImNS zxM8pf7-PI%;*3qyUk4-vtH3lBsvZWWKPQO+<E`WunGK25_9^3>u|8BI-(`BwIwN3= z4cC?m_gdt?31guOkSxKaeuS!;2}d7u6Cs4az}T9YH$z9kKTgZ9%e<#Ud7R`|3+?35 z`PhMsx%$fV63mQY?r=gCN5tTvC)hf>h;t9xkc?}5KE&I4&Ys|~QAn{70AT{=Fd#e- zQ^ONb;&qF0l8@l>%s3z2ewLPK{;`wxv#~8J54w4Rc+_AzVTZf?9KWM(XY8u2YK^zu zLm+e_(_RsA|1dOSK!Y$i!H469-<TpQTbrNzC&Nr*Y~3Fe(bh^MA2pAs8(oF{C&4d2 znKU&lh-`9t9B3^_8%RYk&-LnRvJ};mgpK%9AWO$nQO9&NucAh0Q%39@oQyg<MKvb2 z3;oZhpypi6%jU4>@GscxyJxgSX3gmw*SuU)*4@GYaqJx=Uxf_SPsZH9xda?GoU!de zTE2WCSX@0`g`eJ}L^wwyt8C;-IE`7<$K9D~9HClCjkpwT^IK9t;~dw?V!r$p-E9&c z;+}Y*q>i+&4#4OZ463)iDXUd*IvLZag78K#g}lbv;WpH$=o>?KhH2K0OwXO(yaR3e zyK1N2=G<zWunCIj8fM8fofRm_!KbFnrm~%`-Z%#H=8#q;5aahTH@>*IQdd>>{02xe z1wq^?#guk(fakfQ5z9WzIWGa;7WVZm_!~v`ewAgTWR#CGeKUeNknu@h2^}UDmXanY zvrq*wZz<a}rB8zRYaI0r`9NLOm9IR>bsmOF(c6AHQ6g2tjcN^_Ily_-9sqw`WwZf1 z2)&i`c<wviT8K%mlV%{S3WgB`T{=$;rI=+kRy+AREA*SLVg-^k=@(;nj_9>@k8R?1 zszKdWgp9bi%_5ZCbU^NK-7(JMUV`XCKLQg1oBM6}kDrW<KptW?{{Z7@++e%@^c(h8 zyVcuJK{`+&5II2&4GPtWMn4Y}gptoe!W$??oi&{sdFR1jN_`neVUDyBU@=73xII5i zlhX*Ln7NVq=Y9~qwl_1Kq(MI#Kscu$ydmhf!FiiG^gk)uKf&|&J=#ano+c+m`=1b* z-#ue>>mBkl+Y9<aOk`xB%PeGtYU}`5eDTfi<P8iL39n#+`;i@9vB*t_m|N=A?l?Wo zXliXKeh0{?8Mubw`5J<oGvH>_1=OBVPxP#xau_xz+=ZnUPIfEW5efkmjp<KxlR8bD z!4e&%MYVl?JjEcxTL`9Vi^*}Rz;x4!RnoZOHB5iUb81X~Z<B!vYp0p2^tj)gCz|sO zDvTzXz}<6?Fe#w0t~l?Wsd>wWDWvfj7MG$pMJHjQ*v;|+_IllP+;l`{e6!F(Il@;0 z)SHj~>3LRy@308~gb()h#DJ;TR<Ch@x>{U_;8lG+w5zDCN^fgxgVj)NWqRotl62v= zo&yNJwh?_S)?gDI`lAQSn1p8e20&XsT{W1tf)_>n81B*3`ZpS2*dEETYfzFR%j{}v zFEf<3_h!Xnq+f?kM%<W20bEx*GlbmN{rpw7SD>x44|HL{M+DI6pZJ+D{o}I5)|6-V z+DdOaFzt<MCe55qH}<8&*BjQGH~vWRzJe@bAq2>3WNzT2Hvnyq{ZL(EFiF_wLRPhg zRX$;$0oRnoGxI#~ZF6HLQ7xiI!Vkry%G8~VMi(;1ysR(Pzje+>|F@^G;Hm>pd&T5J zo|dk9g1hce?dOyR#(0^3FjtTwZC<}`Q>`69mKfYP>G`;5VAOsJqn=i{kc%e8-NF=D z<m!EEU|wWxir^|jaQo&rCJWA#07th9jLivmetuE0O;j%qqoEy<V`puQ?^eWwaRv|! zhLN3|C@hiC`OV{B+bFK)k#5(vS<9Q(A8Qo3SsmKLQ%onK$rC1f;5e}jbOCR;V>3a! z9p7#*t<7AaxZN%&NHLSBdBZc$7vuPbscPnqZtStcp{0VWBXk%unHJAi@z2jEL$O=k zG>F^~L9ZPh|BQPFClH=PAa~D-%MjItm7oc+28mZ!8*KIK#%r^rbiaFY8l^;vP+UIY zW+XWpqQ|GI^^*i$v*Gbovu<d7)dh${kvehUhm_*Vgv4Ru#gFk1m>cLxaS*<JmiICq zvbl3YV!+eAiL+hd#2y43MV(uOYwY~QSvm4z9BThXpvowI3-T*Q5t}MQB!NE0`q>mR zE3BH>1Tn)upJ1x(VQDQ#zaaV0wP*bXY{^2j^s!1zaeZjoAMcon90YjsKsey0S7f1f zM%eh;3n1(9D429b{rF(skmQ5(WkysKT(Bv@x11sk&>?rsSHUu&D`NBYb;~rZN1){Z ziabcKsU@I4x-`Dl0uuL^Yeoipew_Ea(Sci5FB0}qcI6;p)_+XVPp9WQIRKeqr1+eK zA8Yp-^^OnBdd|ybypr$OPEP0hp;U)c(x*_qfK|if5it!ua3FUQ%BY*$kS{ay7--Y8 zI8@vg7Ft{JqUSJg5Yr|ZV8FB#H=NczK_J6+92QPMCwZm_4GyWMm>Fm%SyK#e(#yKA zzqNfKw<AFK$m|JstD#yJ(_Q>DfjiA10xddl)q_MN$+&YzJ-Cj-Aj{dxMc_HOLz`#M z%U0>jyX|BXonI6~Gyw!T;uugmeTAQDWxlqbrW0S#&WH6WGEG`WpZ5!LjrfT)$t)yd zs}S#6uh?*U716=B#Ljk^bF(2My)wByoT%W;(P?K{10cDx%1kXP70{;exGXLePu9N7 z6f-^cRdrj#<U4$kWJ*lzDOQ`yw~`}Fhg6MCWb((5ed`D}Kcpksf>3N1Y5&Lp2n;(X zR_f<?E{@XEO!ZOG#RBwoNSqd0oYg?2(No8FugK5_)q$u}sVE}Rr(Ag$1dj1eXDrhI z*@!j6^x|1BR3IL?_;aKcQY<@;i$WxVqLGZ!VWRu*z^l4tmJLnQgHZ}RkRV&IJnTiQ z6Ov<co#4XP&tTpRv`jOklNfT%kWkzzo|NKaw54-gC30-D+E3CH!JcJRHyvdh{8*~< z<a1G;7WlgoWgm}T?TNG9mfZpi=v7g%ZJ)c^&e53r+O#FKwGlAX+Pt$}jr#ex+2}ZV zAEej``a^=&9u?NM%eKJZ_LmN{IufN>NAcOTg=Pz_j+iWFAEUwZcR4r5DOsYsQMNG} zqNU%K8tG-hY7V)ufV{Bhq633;<U*JkhY#IjR}&UtlrU@=KP+EsF@_QFuno(@%J<Ck z<!+RN>tJIcZZgyOh`~8g_4P~fD6hUzjLJrFIkbpub)skLT^|^k1|U=Lx~Hl^5(z7= z;bn!MMLA<Qye)s);1Je&&>{PnQji$zDk#>?;N_JFk0Z84Y)f@lgx*!<=0I_4)Ev9s zL>n?+Q6Ftq){e`CszZ|@Yp)nJtnOwLaQJ&O6x_hn9cW3GSqbanGSd++g$t)ExPa!G ziVr5kDoP(0NcjhMx_B~LM{kX$DE4b~3D94)H2bd3t>WGI5iyrENzAWx-=7vK741Mr zVg84kx29<!48i_V!-(tmT{Rta!=bIf&NPO_FlzwKyd!oU{EYP&bFsiRBVHf6s}M_w z7F!RzzD#yNHq-LxL$^?!d#0VQxm`5%XU=SGGyEl-#`oo~gBgAB-OynkyadF>iDC9( z!R?yy!QfuASsLK=af~;Mv280Tr56Cz;SoCSPp>F;rsHmPG0X@H9q+XRKpo664a{tv z@*UQQ+jMJ+U?_%oJ)=Oz#Gy4Eg5_#Ft{t{4!UZ;C44G5HsKwmJJR;6M?8!eA(Je6a zv<e!pe07$exWV)1l8M3`fX;9u9IQWFe}?sxG7m?wSR9oz>7tHw3biBjrK4KfVeb`% zMA1ds$<~;?6(MgdxWATJ+8^f}MzI>Eqv}i@4^1Vk^ntxKxuDS5)oE8tERl983`%o( zA>o9=LbTP@NX5~-zkq?@m3(d6^2<C}h`UYt<vv~jYYJXa-kJRB6MUsOfpmgFEOeT| zVU{U=Po2=m#=V!pN!yg;eAv@j8))%fmHJXwub+}|;WkN6b))60QN`}Wn+>zTrb_6@ ztVouaXCg4}RiT}DMa#1k?8sfwt^jou*C~emiCYP=#7-#0wop&c(h<y}11KNr#cU_d zb?NlkO||IT%N>nImAuTdkpqQW&b4qoC_ruoJ@10YYXryHT3^xs^V2gBupt#Z9E?yh zL?4!&^~42`NbRReDQe$+$+@{%ZpQq{c$adSC5i+VR44+&M^@TS#qoJH9>PtnTLpCV zx=K5*9&;s``UpP250JrvcZSv!lU8;%?RjnCC~>r<11`eDR1H5G_$6ZL<!hkT&$xjA zy=sd9LQXLX)(7Wl55JlUz`lME(Q^|1Zcz9cspEuNKS@A78S7~^GCnDnxfkg=(W(>n z)D8Y4Ja_2|^wETZ(})IKc65qyf|yF{q2xIFn1RVVQ9H$YT2}<m^{jvp8%MU#k(E7k z95y5wESRvvIC{KjmvsY-T-RHqIl};4##j!Wyh|cKZO{UesVphY)Hw$_yW+$^k{v=9 zctbs69ne1WOg)dm0KGJ@;C>*#$}q5LdZj4H(I6ilEglWht0OpJ&^^8=O$#u=t)u`= z{bi2AiUCEP#twAZQh>Obu)qDHYlBOu-;_$CXF#1QS@<mPDvmH4U^LV-ihY5qL4vLI zd~macFs4ouUgxNx^ogcfrZG->tX9RnntGm{=G9nV>d@{cfibp!@os|b*>pubcfZTW zXHSaDtb{WM%>SB=VhBeqFsp!oE|im#`N5o=)WaEOH|S0#;n)e-Az6zIyG&jyL^#UV zoXT*54w;;utXWfo52jhyMyz?6MGsARB?F$@jCioJM)zZ_hy{^?p|QWRtoJa9+pi{_ zBs5+1z>&qs3}{fOGtI2zm9Qsqa-)*NxV4BgBrFywS`V*2N;DVP&ku~rUXZw*X|f%W zt^e?*p55V%6MURPq<hfTjf%ea3dY1{5Zd$Fi7XZ|v@vxfri?yZPn)4R#!iy=*hCMw zFz6wZCBZQ1)km||#$nby#HiA2^9eQ?;(@0d6Jb;tf%ZeEVUuDr?vJeX**Z3bRdCxT zeZ>b<ROMd1a`DwIM5}CGez=SQ&jEVVG^%w@<~L}aLNFcn6^sC2HI-XuR)|pM25<;o z>u3aHj&HPtX$L_zf<!jF2oR|Vfd&~?&^{GT#~_l0UrH<@FKeWb^@5LOK4lvjo3yhw z+J<Y;%k8P(x&CfUwap;xX7_FsqzSH$UF^T66rzU;%MD)e1(3H&!kituXw<9TLQ_Cf zD$}(fiK*5j7QMJ5$t!SxzHf=KP-cqbx&enbd~?vwKp`WxY&@#|9XZnh2K!^c!X&PO zQ%swA`2Yv4nNb`w%qnRBvrrWW3Ab(CteHpr?N!}uoH<yHxaEg5@R4o`ns|3cGgPbU z(2A-{NpxEz&p&W?`zYBQb|GRChh>GhOj_iHh^!PHEfy(@>uWGtikpv#aqAQ;JJQXB z|JE7<(fh|3mg^unf^B%zA$<qEYU#SOK1X<7>&igzZVCX_L&*WgSVzI4KyGg5web}= z>OE5Y<6e4di)bzme=p)+bJ8l-vE+E4xI!SCPIu3jvx8fQ-5K$|j+PhQ;oN-3@VG_f zL{>uVMAnep7ZE}6vPeH!amvExl^xI?8%`nkym=E86jmU2H1vh}Dgp{Xy?8Z$KsYEG z1x!IRI*|+Rz^=`&YkuP@ee0c<^`6u$R#;M#joVo}3o*SOv{l#`i0olG4Vi>0^G3lY zEJNaWv{}YK$nfi06(g@x!&F<?bBC;OloikM1fWd8D3P5HwNBQ_sd%fN<JskOKV;bP zEuTYm2TY)5^Hy!{pdURG>%*WSjLSU6i|44o){4d8{WkDilJTUp#>$WX<a0vqd&a}L zqcmAKq)z%xP&FDDDO6dKlAq#e;;Oh(`BV}62H{e4oES52h=kZQgnyAZH>Xzwn)4VX z%-ax;w+_f;qbUV!vD=De&zu-c_OSr!J8<mKR5oql8iauDr>7gmUmYIlGz^cl5bR8c zgM?7wwNiu7fW29j_1My)X-A-1-l2zU<!L2`u;4{>F;hakpK(gMBcr%e_(poyh;N3_ z!$t1`zwDzvShm!cH@n7L6KC)a9vb}KI>Gp-XNty^z>sK7DmDBKJbnUCcgN}YJn$uw zkk)#=BrWrl9>BMz(d|GgQKyzmD(8u$=upLxTT)U$_~sx?z&M0-hQez<-F~sVdARxQ z=Ci%SXKPP4cfnGNiZ;4Ra(D^(3VO$V^8i;ssJ|U=$zX=;4-CCP-{IJ3un<U3z%t9c zJ|Dr?Q{=O`8fI3$8V|DJdE<jWIB@io9(i*Vp@$%?qvo@UyNTeO1;cB(?h>FV4M@(J zBD6%|MA@izX4Z9yQx<Pa+LXjK@}6!kA}{ia1e0y#Z@JT?`m{3{=vWkT4hz#B-EfNX z-jyI(RGD_7S{dq~^vu_vx{O2Yh~2jt3Zm}ga<EX+TrE*W`iY!C%~g|n=G`{c_T2&m z0<3@NR1iB@tSNik(%m36C!A2KE3>A6hzwtRD1@q#?s*02>ha0}F9V&3YO^NLRGYps z(!)nqnx!8jrRW$8z&p?|17w{G#x17XaRk{4DhC;825Pi$&0~^vbb>msgC~inFYa(y zUBTgz1M=9-JQ1nRnRMt*Hz*k{*0#9{Bf|y?LZKr+Ul!%dc1(%sUE<1*!+Kf;E?jRq zk0Z$HRQ0ZH-RF_YV68_^B1Ff~Hl<KREyqyuU9hsLz%vLbf_85@o<t!*)F6n~z6f71 zFB5)SQ!wZX&>dl3MWHk_n8sqm0OIceN*x1`-RClk-EMlYvSE762LF7M2uqdI)u)^E zCkJN%RWOf3B5!Cgw}8+n3agOQwpo3=j}r^l8XlXWg*D)#NJUL0nuMFy!s!YtO-Z!n zVxYc#i6D+se5SF0{6qrA0tyd&c}oX2@mdLz(8!Mn<bnBEZ5)qyT;bXiKqz(xn*O7` zVZ2>et^HPUIp7n-<_OMf_{@_!4X(s9ayyYzz&=Fy#-^Cr8@aJ<j)*Wj?m!-JwW9_( zSe{Ay02v|ZGM&KA>nVDoXv?``u9K=bFS~*mM79q@hXmJ0aGE{_Z=SJx?jF_DV-G#c z1%V2#X;u-hSmsaU*Px08ql1j@uXDo^)<-kW!6%+7ukKi8TU>QG*YYL8?e-ZttA{1Y z%pvO{%WKk95b*dNjj4$o;AX$0c!E~E$SCc;WOKp}_o=2NP9ERr)D*eE0H^T`iP!aG z6L!0lGTNH*I1q%;D_)6fT}}cFwmFt>X!dikFe!@uI3KC=raZ6CT9Bx9!GTd~(~wte z9ioK+I-<~3&UsGWwGND4q4wegqY||!o>odGedyvx#-2xjf4<wCYb{BJF^Z=;Rwv#> z8z(8q+S~>ury;3@bTJ14IRe5f#fPGK;#gQ0D(3DYS4<skb9Dd?Gb8%Q1CSrJUy+^3 z$$5WW0azP9Nhxo(d&_8CEQ_ICNi&C2Q!<iD)A2l)(i+*&?pb5*z)>M)4Nl$A$Asr9 z9}o>65p_K(E%hz6WAVJ;{S>}2Q8uqO*t{<;$vU~Sv?XBB)H-O_W8hKk{!@igFtcc* z+Pr3_s*~&=;7qaGt=oN`0%ML88wvO4Cm)M}=&KLaAlJqUUleWN;1F=26@%tNE7{n7 zdbqZ?_k8Qoi@nXmUu}NhN*Z2hpLo)(*#V7kr}(+$>eb?2N_BlmPrF7oiZ$8P%`Ql0 zf;UPuz4UZ}3xx6lZL;*N%as7Nnz%owSA3CeEZO#Xacw5KL4K7Fo118+qI$0>8G*2N zLmrOgJqNpHW~f(e7WDvQwvcrY3jfDhW}(ZlS*}WOKK-M#iL13H!^~f&uvH-63;mGz z)n8Z4&KQj%3^G$GMgLOO0X4ybT6*#RsJ6Mi7t2Xn3j+R?o7U&djBD=<yw8iinhLTA zN3Z&!D(W_Kn>z#D38|KbqfRaO3or<<vs;sM+l#6zXo3Fx2zer=Wg+o(@3YrOeqetU z!-isrE^~0ig+xC!d^X%XR093N)7?zSG_I^JZXfBc`7TR3(O_CT(Du9GRiZW;ga>b; z??fWdfv`f2LUYr9`tU>sOwGtf0>n<ep^&>22L~mRAOUT&VO~<6D`7!1HF<6{yA|vv z+nDo%Y%ftzyNBxg8BwUV!Q_NP&}b`p(bdRpF#WkXwi@?PP;kHjpwNMM(wKgf1`h@v zU!#FkM30DFN2Sh*m;m&#A?r>s&PPunt)R^z>0GzOALzc*Q(Qc3n#&aa9fNT{I~;|( z7_`shiJCZU3n$vN8r_ztt{(3~9bSIWfSD>&ww5~$Od8pB5l+#5Isx@q=sQzaQmjgE zN`3f<_c!YPjr?<d_Iv&}O*T8c@HhD@ty!XN{mKb%@v5&*puSfEpP6EuGw&5*AHC!H zH`z>#7h_{?jTOVq{z^v9fEv*KSAj_)b=xnh2SZ=xvkW1Zg-y7isMPPZoh{;PkY94+ zGdc-nZJ}6NSr(1>-UIA=>sn|9QF_3dj{^k=`hr?U?RB$@PB%(&0k2EX4NxX%j8$sM z61M6>%WI3#@zG?Hh>^@s5INV`wYcWQH+olA2_XaGnQ2DObHEx~ZQ323oOH1=2Cc1z z*1)sI!M1CZ$~>lEi)J6Akx=HtndDJkaU1%AM&!hXu!j+c-^OQ0B(i|htiI^Rw?=ib zuyTY2(F+z|9dTeb0&o}KSalQbKe|MsJFFJja31NoAO(>Ar2~AU3g{y>2GCY@lE&5A zRbDl0z=;+=Qs@3SJ1xoy=IQ0PZHUj`Xh-`)T_FqB7D4Bhj$RApekZwqebh3~Kzjqd zxN69!TfGi&5`zq$8YtK=7YIJ`VNf{%LUtI9cc=B$T+4Qn<@*l^qQ0C}Ak_@K<aUPC zZln?7Eo3W-6W0?(*R=U7#Y<s{1-}QG-lZBe#Nkpan_i@OAMd6hbRS*z(tHVb^>szQ zQ&o5CBtawP5fmN^oZ|j~2cH`=`l!X)J5?G%>Cn<a@cw$RfiJnMOSY;Ma`#4M%!%=9 zR7<8c$t|fcHir0C<BQ_A(M;2B^z_yvQ0TDQn_C6`Q^^}$C5Cpqu9;jgCX)OgQ|x?8 zDrvO>>#@=JkNtcC`W@g+?Mp-w=dE*k=!r#F%jsL%x#$mq*z6*8;NI8d!yNY3N8$=u z`T$Wqk0e6>km&NI5On3pCJFFZSUpAJvP-s!k3GR8sA}^Q$eQ4;6IQ&gn@ETV$D6*Z zV-^)Qa{`-xw-O33ocFF}9HxkO0YJaHx<YaRBtd-#V#Hs!U`rhh`GKJ;`UfqY>J`N2 zE-$iPX!Ia%jxgmmMPyV9J<ztEZOYC4xF|QTQ1ydY?O$`Dy5{@@9C$I_7+==4IH^R- zNowIE@UOB7n&c3<TW%$iLeejyjmy5p3J3%rE;sp{6UO#nY5y{t)bba4Mk1kf^e5(a zqlZ}Qvgo%1glrcpf!NHoU1yy5H0~5IC29z5wNL0qc>APj!PVXCq1wnm_$paK4<rk6 zm4RaalY63rvCuBD{)sIHkScV|S!&rv>beFWdBr6Vwnmww2V|N=X#lE)!hrvYkp%s9 zN(W`teK9%X0v(grQ&8HX*IepT68|G`tVRW;KM{Vh=5gc^TIBLBYM-Um_T_MgjF=`( z*aEdPWzLR$Rc!i@W)G#NV|R4T+Fsa0YI-;POuMy~!}_)=PUbu2n@0@IY;^2buZZnx zM|>CEFgIz&bq{H0$G@`^sdHM1;zuf9m!vqCi8UkwU()<^2=+k~O&6_vwAoXaoS}4W z6Pb{ur+|;~)lm9X#8{iI?A!%+<{BE)43&<Jg`$yu=y4>1<#LLP=UxqMfMd%BPM_8& z#uj3qR#!n+=zDLxM~@qlQUWV&tr_=eo^xUZ--b)w)(@uea8%GdS&1Mv#?NUQEY_TB zLld_~7y_rR3bQ;nsFCYq-8ELa0R%YYZ_;3|wy@o%E-OF1iP~D`G`d~Ynu%mT+b9zS z7*_w*Q`Nx7CmwA}g<5gJ#)aWIL|DBR4o_<fib{<Mjz0Q#d-zRJg=Pkfjelwm$^-3) z2*Uy(`?fYl@#8Y9vQ9+pW{4+VrS1`x>%?x$;e)}y)o!lN%gly4McQ{+3?0UHm|cxE z)${``ONuFYeUr3MXkk#`GeMZtnWcjkqP#$5&=7?sRU3!6vSOHq23|B~RY6p+pdzga zh^~!E2zW!d5G%T;XAj-+$Ve0t9LhE@5>)$3YO4LER{1>UnqPf&9UmJYzY%iLt3wCM z4JM77Fd%Cnl2t2&zj&u$J>QcM9Qgnz89M>LyGi2?2+!cKadqTFASSxM4%fc@rNXpU zLO&d+U`CwnnyYfe0}a$m!y;<AIaDnEpul6I>F{F~;fELn>T7cAbZhj$<C>#WDAZFt zvXU*9xGJjex26^e3{5yt30`X)>JA+?Tw4b+q~>^^o)HH?-Q8Q;d$D`Cv-xae>)F?A zDB!)sfN=D7x#sQWhI)BFe5t#6$aC|8b0LAjgLP){E??I5Jdzy;n@4onsH4K4t{YTD z4yEE#rBI^)q8yv-5g3s@ZJ~WTF5ttdFVRa<`tsJsZtTYUHr8%r((a(<K7mE7JD}6s z9~{V$E^<WW6TFDJ;1C#Ou<V4GopHMmmtx`yiBxA0>2k*UAWKd&FyM<XwV*A2L0mAi zyXEBC>a}CSj1m^1U<%0$0@Ase=QQnj%!_51Dy@sIQ@k0`nHqUCCK-nzdf+o{(}{*# zJ~~7i#@Tb~yv6+0p`#{8bIzETfrj_oO{eW`CylA(Tn06@*XG~sJ$=I0hu2B37&tFY z9ydJg4x;a7q|5AO7kM|^fy_kDA?(UQ@<N?Qmr9~3E`yg~LC>iN9FQi%6N2QZ1|{Xq z?4A{10oWje6Ta2DIBk&W#n6Oz#dVv~QO@fLz3cY#48r262ijsfAC53vNijGQovcZ4 zBtP#oPpIwOH~8eOdw0X`$r~UHd4+z;h#Vh&xQsbWW@i({1k_6MNx&W<m#OQaMw6d@ zxUANB)K9xvv(av#0W2^s3+fndQmS~xFv8WP4(A&!9Wm1`i%VpNHp{XoBhp@j0WMZ$ z7mYDZhr+U8qzM;mMUfPJ3Z2V_qOxYRUOQPoFX2*Jm%<><$!|D-Lc|g1ZW0?ez`E;| zu2s^ZnnkVI%xP(CQMgaYe0q}ReYnEQ)&A;3uBl=WO(5u|@PP1Y#-?=@>=pE{M-wn} zhF+{#L~+XBGW+44zRyhgh}36plJ?6i?M-wLQ@o=s=$^A~H1IJdBBF0A-e9L$+U1oV zX`rBhX2EI|?rGZN;>BpBwcMm_UtfFn?b>c5s24$4T+z0qS*QTPoUd=*B$l|DB-4>N zd=Pv{z+v(d0wPLO2TA;#u~v%i{I)R$v;ZL_q5grdDZ+0jJN*pg7F3b=o8c}bZE;6T zW_NDy+m1e$uiKW|Au!ZXUZsOkUva*Z;vDZgaW|?X1if!C%r;+)!vnB{jr2-UV?Nl5 zA?DnfxlOt(CE(5@$`Pfh42BLdjHcG$aFQeg8hdJM$gayCXztc9YL4pxwia<X(A9;L zO+@T7FO8Oz8gf|;eK$bI_#2kqT_d_b<KcgGtp!*Az+ri#$*-l!!Is+d0My2zwVN}q zC~>^577r5x>^qkbOU6&a=yc?y4<1#tf?%nSg>4b!Te|_v8u~owjta(osvVW>l&CQW z59rp_-(cf6P7#%a!azy3E4Lhq97b$TrI9z5_ryr9A#!U8{IcL&4BOb;dA_+0+;9xz zYEh~O@fPD)HKkkn>UtWI`%R+CF0Z;K-<oaORUm4I7Vij?%%N|cs<+M?;N{hc!|ePU z-4m1Qelc`d@fHz1p(dkD?<4kJMnH{Fi2y-jU4nDqF0Khjai{5yM>W8`UI=dCO7s$j z+}2~@LrEy{nhp2~Lytu3$WPX`OIeGE%o#Vj=0ZeO;)mVz-4o2Z#Dyu18QyYT`%4n# zkvva6JKd+VsrbS{oz$mM<7GB^i#~4kn$p@M=px)K`#mrb)bWJ#!D}a4RLgQUaD5b- zcBJZ+(oo}tE0wBkkApUhgnrwZ60JPGc8b8$x=u<}dDE|CdRwDb=ai;?t-1v>^5Ce< zi!vWiR+pESmSPh58#xd$(D(>!_X%vvRx%?nmUDV8o8jdH)C(ZAlAY&U+t0W5zSk+4 zwx1t9-g>gP`8*P{j@QZF7u2qgbE8l@HdD$Ae`Uty-AwR`xMlrrq6sjWbn1p!2ZTnQ z^c~G^)h|>a`YCk4bDR&s88GIqX5{?Wl+DWplU-P&Q{5v{HFb1qO>nwtT(=CINXz(+ zns=6^$xIy~-@>&>B^ISPBGXuqncJv&{B&doGmFtUYRVxDkhBhJFv?`zjAD|b<b&3D zPRT=^b$^i))D)sqRKs=8QKNx2r6uEKAta54d$?KkHJe$PMAWvULLPe<nR)(nlIFe~ zoRhxQ<L(%lOveOYci`kpg`||I!v_M5Lb;No`Z~)5l?k%Wxfx(hCOSLJ1=k{sXGBhu zUtf0UThL!%C<M-D1``LOV`h5FH`E+Per+AI7d_wnx;@USvAA17?!kX;`XO%oo&4Um z{y_u@ocA#G^iyAS%Y~5{`}`z&&drd(eKR?1i#L@q2=?Yj*NN|^Qnz3Ha!tz7;Xjr~ zy8rHJI!sSdsmCWyJqUj!UuTfPYe1enc!z8viG@jV+A{{P*&-HM9a{HQj5+dqxM6yN zLx_4Stplf9fX7nZG=S5W%+^5HZ%%CopR_dY7QqMuraw%fz0bqhmHcF1L_+*(b_v*4 zwT^k_@Y|@i;~e#L;SWo9;o1gZg{NTI;=Lf-vMvOOP9Qx^w-^i{!g~Qy*U3qSidG{B zVdilW9T%kceB4mTayc~FWBaGT&YTT`6K@_?%pNTS4&tKLZ5NsV!GYN-Q&=!3-g&I9 zV7|WH+I_M1<nYPb_uDV_6mLR>!c>g)BYY5y&~t&TKwKLvLc`a*IMc>VHXmZlw#v=N zZ0m+&UjJt8ISiS~)DJn}olg|qWv?ZU6NX-h+rzbfpS((2Iu7ge^in?oyW4R<7s8Q{ z)fmL7mIFCaG@kEncKp#v;z4RH-B&Y<=mc)K0Q1x}YrquQ{7P^d`cbm#_uys<?`P>V zw?7tejY=!Ldg%AFQ?<8s%{uM8xku`qL!t_kY7GxTj6EVI*>bd(Wg1?L^LY>N+J}c_ zdWl;26X!}do~s;5)i);W>#m6=46t^^%O<R@SWK&q6@^seFzj%Ryh6}J=pVneBPAh5 zmJW2Z^5rZxV3^j+=xpW0QaW2UJ-#w?>)JXQLERLbz7-=E%Z8t?&f&HMJedys&XGH> zaV4dgx&UeMBk+~b^*&mZ>SLMo$gR3!%8-YLt}knAOqh~yNmN{M>cv}~Go6x+hr|%C zopttLvZJ%UPWaKM9@FjIYUaPSnUQ|FiKn;|?{wZjT>hyheb+_?xzUu1dh<@WePd&S zw&C``zOYyE`n8T<^)ufwVmf}Z?V@9z+-$4pTDMxYpL(TiSeLp*n!cY<2S05+w9y3K z*`_-542Oruh@%DM{{&<8J@uq*WB@QvhX60x@FFjZAqI5e+U)a6Tpxz^gp&8@EkW2V zAn>rhC?+X$lAI}S2JVfqqU{Z6#j~`Bd08Nsl<HIWe1O4NDZB+;8#J{Y(}IyZ9>JY+ ziK)&_<$SEKqmLE$RFP%9i=38~?UG1FZ~3CzsQM7CsEe!GU)y$K#VMDWt5Q>tAIXyZ z2qL`@U=|6Rj#kkW=QZCQfP$z3gA*kRW<$u4?HVvKI$$O~gxh>@zm+_=FPdiksN7~j zFS?$*{2_!iD)SzU8drjfH(Gk$YPY}M<%4v@sxU9tv$jzGiZur2M(#8#{-`2_V`yu~ zLNLT@CaU2qT=KQ5ySx3&jQK+f3G-rit^FsJr#I$y{OQ&!q&c~#=?DR+y1K@SzdqaL zydd1R;KM*W`b#S)A;+xbu4|{j^@6{^{@~NRgeX&Ou=|taSHI^PJJ*kPUcu<VKW>+5 zdlZGAoh#7$4usbTH#@&KE=C2XJBI#WyOZ<!l$INc;`-W_4+4`s*(M{US<#j=#|HaV zG4NU8+qCu#AlztIXg!+iQ6m*+qD6Ulfo%?0+gzq4WiU#|`ElMy$ryuYm<Xqk$GWV> z>Zw-ZlE-<sD0>xpA$ucL{md>h*{*AApsW7jAL_6BOK8|GWJiP9wnUp(On9-n*e#06 z7c5)qY4tQEzFy<&C)<zKp0qel2xdlu?nkdC_57{l^tr`!naD-m#@gQ6?%vw^uP7FP zl4ZLuWo0e@HCSQLFhbx4Js`;wq6pG&t>%`)<GU4ovrE@kXZh(_A56G(J{Wz46}&L_ zSoV#Yzm__yixD+HwIRqb1Ch!6cKqk28S}zxMGTVfwcVTS2KPF_zxYf}B^T<YCBzRJ z>(>Z(+77rxi$}nHkgltdV&ymnMN#yRlFOX)6GCKmh^t9vk4H?SjyNa`v7%^$Dk_7; z+kj#Nwynz<ypwKkAN;{b$&<XRsuLFk;MP<>)SunrWPAzA*vAkYol<T#*LEK-B<IMj zsf`M?3-C5BDJC^qGpGlBf|#VIWkw0$+X)7M2k_6f_qNtI!O0uzLo|7^g6OFMkI%?h z5AtrH`Zw+Ma%h@Uf0mT{6Dq37PfJuDcZ<=aBzx-1VOCaW`AD?~0e@6?b!RF9teaJ1 z1yjLuV_?ps=E-O!-?CZn{q_>tRYRp;@Y`Q3tXM<C0)vBF=xRB-*$5dtSnhsK_CFu# z(U~?>AHQcH#qk(ot)w^<Z594R&*E6s1^`KNHXe^Ui;I_+m+ch$+b+t}MLojBCu(A! z?QY&veda1JhJ8gKOducNgVPbJ+aO+1KfQ$Y!)3zdSADoFbBONWg4&fv0l7$a%5(?w z$|+F%{{R30|NrcLYjYb%lHm7^(0^!Y<_@3*fuua1xg|~MBvG;%BZ&@?9^W39AvB35 z*%lsLG$4v&^S@u^qpC9N(EugQ&E75Hun2V5BdaPaD=VL7Ah<#go*fi>2M>zp&kptu zdc`k$hd=KBd|3SQ?A^O(Zx8og9u)iUiWmEDclSUKXMVjZp1u7ysQK5ux4XTfsxe+u z_30Agg#x5C%qN6#bl7BJ09@f!Vtvu68Q$c40h)+&;1MCNFNl00#{sIAKyx^%$Mw8q zvi40pHnVl$q|>p2iBkNvy8Z#Pm2~TQ6&w1PXSpL5nadk6IXB-@Zw!wdzVX)qQ_Xdq zDProUi#R5gFIUIbf#74^)O=4u&t7}^0wiXG*(in-9LKw;o$1n0dJK3^!Prya^z6M( zm;*s{XRyED;VIdcFD5pBH@yhzxOS<s|M~mx(K?0A%DU=jx~`~gXJ^L_#;&xVk}%8_ z)mDM7d|~IjO_%~7h#1C(%5K6jZ%+w%=a81$lI!%vypaChQDQg_Ha@hW=7nfngqgQ8 z^4ox96BMMNvPbVyz_^jG4IAY%O~Xd-R)CEgxfN{Hr|NW}M;@6j6S})l@8;*WE<kAS z3QU6V9NfmPjlm#%+Jjb>72#G4zEgik$LTim1!@Hucxu@G9CZjl?de{7MMv%Ej!v`? zwQ6ZiFuI-w6<U>>yA$S~fDbwg3rArBc&VfChw7Thj_)pZ6edUSPefxRpw_%n7Y9+G zDUA`vDo9dkh;uc*oL{qrJaCMCb8V<VcsoaxT*vV3=jOOOsYcVQRCd2%d-6S<K!ts9 zoKg-)**%j!7QpUc+z+t8n6SGUSdu0JQklkoaoN4bF6IOb?HUhBiZOv`j^UqaOPs1> zDbX%^<5!>!HN&$j6&J&_-_;~JZunpmJ3GZD`pTqMh@A(rk;8Cl0S3c%p3S5DkMKzb zHjSg!*-r};t2M3&Dkr<LzX2P1MIC~f%Hx;-Pmx2{GUtqLYxE}CZh>#U+O#X~JXI&& zd9e1uT&^7HYw4~HE7*;eQ<K{dFbwr%)nH&>Kn=<hvnH8rcK7|B$pY%x1-H)T2aNG6 zpt!?;UjYMp@lG}`zK5sgOG~W4b~ulIJ-%HYugp*H_(iv7<wEO@qY}=VF`Pdi_~7|q z&m0oN4IR7Z#*^9zWefUq+F^4Yg3yQ%_vYiY2C2g2%n<82-7q=wiQ_+mC}*^FAW8^2 z4SA@J6uieUAhV|eTH$;{EAk5+$2-EggHn;qexOMra|hUk!9tBa-K)2wrGQTgy(l!B z?Yj-bMf-9%NA5?s$3p_?I>8aBITHACCuF%3vfK$-Ru-}Vb=g1s@#VYsyL)fUBqP8( ztc(4zY%YM%9?|yda7Mol8bli27rp>PX+!72_r0MHp^ne*{mXHX`Keiq$K|a4dnHVL zhcewVMVEy_DqJ*DLNH&&)GqYC?;faP276zAn$Jo!4>sM<9ZZi(ri@_WLp6L2<C2#x z_4be^KnJejOq-mc7d7Gr(+6l<_G|M8NMpFam|rejEMjQVKNVw94+RNzmccNg4aNOT zm<tR23C#-Y1q+!9k|PlZO-L1>d78}BZl1&SqoBOr%E}pyoNc}6zI@7yfIghqM&CXP z4l(;U>FwK=td*=0&A38bMs25Wb|-zch@u@yux)zY5cX7tUenwD>;;)8;1ZY<w@ZuB za}BN?sV0u8?PM}i3O~Ue*Nw;N<SZnx9&8HS_<O}>I{1)5;h!n<YynZ62~q&6=pr@2 zO>t(n^kg`?UPBjC*4^N%L~c@4p-@gt6bb>#V3wc;!yMWG`xl_1F$DU20hExka5}6^ zwra1b<!Dj)qJ3~KKbe|?r~zvdbKklKdQE_1ynX}fbo#-N%xM^1oEwILVkq<poy((Z z(?<CTlse`BoStHIMqmjt9KJd4ka(DRMk^n?YUqWKk8P|752#BeGK8g*QMqXG4C6wS zg9+krih~CB1&-kJ%5V^9mxF7YO1cfKu9<^+PHJK_@dy0uQ}ZzK3rtf0+W|ubkv^E$ z;%qh@BL;z`Es7cN!j$MJxk)_j>#w$7XiFsa;7z=RY;m4#CgbVMO98DXZd1^<Ojl}Y zA}WN|)gtUW=X*|(I)BQM*sqnOxK^TVZ7Fs@yljQ$$bL)R;ax?Mj7U6VhyHZFEeqq( zFtJ{f7_YOzx5W92H>Pp3KS9bC9?#U}E3joSrQR|TEL19q6lshXC3*hvOwwvlUk)MV z<eB~R^yyPhuT~)lL`QWz6{Lcba7=yG4p`c8ExdMO5EUfQ<=pJAAviLA0#2Yo&hhM; zNH<6!R6D?jAhTZ0J+n3FSrPpJq5KK&wlcJRdP4o6X+dF)H-ZT;Ws6A#77``JB;cN$ zY_sQ%1*IRL+Hl|)GBX7VM7J+9;0fHw+{y?jW2^}0)DS<@XP012;FjgPmbSF5%V!ZT z{X*d~6|rn4{p#a6we8qRLZN|tIk2Y~*uoen%*Gn}xugr^oT;=&G}mH1;rpi{$avIa zqkUhGjV=4d8?xf$b;C;LW;L5)RN3jc8ak4WQU@^_2tDb4rbGb|mo3XjK5yZidFGoZ zhOZ;rAvm_61IUu{Hju*BZJh`#+#oJUM|51Q)`y;n+}L@X(X{CY+v&oI&JNv#Q5Ean zLTGI@s?x?RfAS0kmE;_m*nc4Ig<@(ln1Bw6j2s2gPi*`!Unm8YBW1v7J->z|S6$%L zl*<=4WuKErxl85c0eV_=$c1zy9&QEo<~81AMV;WuR?#^dP0M*Ff;YhzDmXDco}qTY zb{iURu3`)ha-`6W<>i`%xzmGu1B3#!fn5{(TnCd#F^|kiBw2I;naqmAnq12v{@!Rh zIq$2vx^ID_LEC9T?Yw+}WB2Ba>;Ce1W*D-=iwe@!;~`5!_qTgCbT@y%=@{=Cj+1w9 z`M|aQ$UC|F?zXCd?XaVE_?T{*{FIaWiS{_j3<ZnhQdFPn(s^e~UrsOat{oO+WB>H! z*fzC>_knEfaN2`cjC5QVHr5Ty<*&^}4T)9HC+5uDg3NTT46m%7g*zeXosjfSNV>X^ zG?Cac@Pa18BC+8q4()OAY&5b<dRAAXVS^^7$evx4qtWyVR=p;Dd&1O9k`U9LI&sbK zq>%yp{Qyp4EX0VdkTyK73mMA;^F6?}Hz%ck#9{u~THQWgcSjIv0w+?6B|E}okY=OS z6OL~J0oBIMZgti}Z7OBUE-tFkrO$ft4TyF47{h2K=k>>Gg7&@zbAL3S{-}dgb~PB# z3CQ;^iHh8EETZG&kbXgNl?~(Ij|d_M{^!{2I^1Hu|7UP7qZ^IQZE6U>1KXB){D3Zd zBvs}f>`r#gXd{tw;><e+qNN72A`U2pv$^Y4vmbGi|L}s#0Co+W^cbmnDJ4b^<Yg8X z?*ZW7a9Yug#0?LlJE}2wKo4OsFe~OJYIfm-$kEkBg}ZzpMVpl95ggnGYhR9#LeiW@ zN|I;c9#yLbv&cSv;iZ4D3RoNxVhLDQh}^a_QKgKqR;E$2C_ujPL~k&32jB?kn5Sp1 zGFqj-FmUnW!_RflNDRJsSImjUN=;_nw9Muio-iJgKfE7G^46p#l~U^fA3z9}9LGxY zU1#SR=9PF0oGB+D12o+76FM0D+PGi8xo_}cO0-oF2pqqz9Q9NwWZ0LKvC6b1#bm<Z zY{44|yZtT+kV0;vLO$-#98=QLk(HYTr({yVnHhiZZ4dHce)qt-OE|2+oWovfpPt4e zu6~@Hz@v>`@on&`-|x4NJ6Kl)3cYe6{s`zX_>2xG1x<Q<R^rqwfckH7i6KbmF*#A= zMZ)&2KwU&l2boa%xv4b8G0@LkG(OlEIC!-vWRRZFjowPlZm!lWufwLF@$FVrL%M*8 zlC2=r+EroN<-z*ECtl_aFQSe|``kOk;bY$C@l4(%cl+~-j}|z>LK?O7JacEVYtOxY z;7;dN4u=ynu~j+AQY&#BCWpbza7ZDfiYx(mQ_ag<vr}Ir!_d%i?47Y|Obv`_$%ck0 zZf-jeYG#jH*`r3b@62@WZlHHJ(A(WW3(5joyaFlXpZ|nzOTA)mGXM55fB(4-er*03 ze-!!>3)~Zhd4sb_<#8i&VjZ&A9ka!EQ%<4A;j;etClNstqX^UcpWp7je6{x$bCGo% zWe2$^yfriX@_)TIoACXseN)nuc|=v&1R*rEblT<<nsA$H1xfwh5THA%ho6o~qDBS; zcq3%ngQ_-nmc{TAy=5jf?O*oJ+}nNs(>nv}|H9+||Gx3>8yz<a$3Oc+nFYLC1qUmY zD}mxUH*zT`rq^3~-MB%sM!q?}l*=0v%N|L9aaoZ#^SzSTdBk6vXVb|Tw+aOiF_Ym_ zlsfV=Ymo~c4CR7?ZNB;W;Ls7G0Y;lJGbhlIEOp7~cx~X-R<NK@6i5zPSJw?T?IRF6 zs{&gNMmBV&1sID%cLuIIG^6UFoKi89qRh5$ZZFYVZhex=vaLb0J5hn=9=3qSWwt{N z2%n_Tes@%htf!Qd83HvoHPF~``H3a$tOD%6$D`aZxfEjds=RKjQ$cz40(_p$CB3;c zR}q+Q%J^D`ngZxkRCrSS`MdAF{muX%F2+_k2a$f{bYzb57Z;1khc3y7$`Srj)_^Z4 z+NUbe7p{F?ce)R@D^9isZTsjbWQz1mn5Ec&IoP~e2Lk$O&=Kn!%wQ~X2cX(P0jP)O zbUAqS^4+uNuU|rJfS+Ez!^38$ux6AOWdp_#bpP(4yOg17K+5n(J@j8S<MgJywEp!S zn0$~CV!2a*;=r)W{ueF`{xl5k-v9%wH<oSS)RnO7=wJcC#PAbC!$0o*%dy4k*ZXfX zLwk?HM`{Sw#-D)XRxkZi(-;4u1_dS<j(5_WVT=V(9vT$ofP_7k4my-zBgV{BTfIM3 z>c4*vfdviKFnsUDbcmUGj*i!)$s^oOBAZ8y4@FG`N=;Wh4$X-09@zqzof4byj}m6< z-XekO#~|BgqBo&kXSg;?N11>acTG2hJ7GUzp*<}}r;AZ3Vqmij+5#OEaQnsn=LPJ3 zOVa~$353v6I&`XHOztIu2&t&Uzsp-vBvyZNBXoQBTp$^h;phk>!SB}A(D}Csie7@J zB9c7odOJ5g5KSD5s7&7E+1`dv@bY3VN^_1$L6LdEgt1rr-JF?_r`OCK$al{Hnprul z7g`Gu%nylrF)G^)X1K<$%*DYDMjmiS+x$99U{_(ydr@n3tp5R9se^>#y5lXPl>!%q zd!c={wrCCwx?-}YFZ3xVo=BaW^yRTuu9kd@YUp`G;X#nOEY3#dIm#_4v>E!JNV~~s zlxk|oIZjiPx;a3^?5b)p6$xAAOvHgch_O%=zDan+Aa(FrWpv{r%S+V5=K`ijrvb`L zABUn>HA7=T{gVCs6AukedRq)R{r7d@tg(5znCvWJo|~>@%`Z1{v!JU<0aosf8J_Yb z*IUV%tNLeaxjAf(RJr4@)9pN<HIJ)~a6FiYl>8^Q?vM6+4B1iY;7p3fAaNmZfuX2M zoefWt0>m_#!--7_%wUXkrF5CswZxZ1$No-s8);^s9|>0{4UrZMrVfNuSB~VHPTcpe z*gnQV%IS@G)ln2tIeh^>vKoVCaRO$uBpjwIiIlS*I#D|pv^sXQ*{>%)m2C<k=+f&E zBE}%KDi${99xh)ZPc8O)aIvBn@f7T8QA`3u`rm)33_rji!(0yWJEaDEBB8&W)gOIm zZw);6k(ZCJzE|9(MQ+Z|Y>KWtfm@Ym3Ng#011F}c8E1w?L%JN`3*g<j{yRuK#?=^f zh^NzLo&hS4#;Z;!fx_@*;s1eTcbtOZ(%o+uqxreZ*w98H+QH$LvdasAm|e~p8$2M( z=u&&H++4O-bQ}Au!Rw%)D}n=BR0;yvvLt`4uD5L8lqv-;$y)_$G?q?C8<F4}{@<3Y z2F^>g7CbNX{Hsj}<;-ZqQHrQ(SjB1#QKFkn*Su$J&S#FoyDBIgxASG&(!41@QCM4h zrlLS&nZm!lUne0k-ELOjwb8!Yx7$wK_4E<dzTNt<s3+O_yYh-|+kEP=@ju?-!9!rd z?p#*5BG1jepbp>CLCZ3)M;NDB7XJWS#_%9soBWZs%x?x2tR+H^q1h_3$LnWM7$-Ga zdUD;W*=k9fcEsU;soocL9ZARXP`p6Zpmm5wOBy)o&PvMaQ_uUTk9y$PL$dFB@MT2Z z^TG$t6+7j*tz$c7SHxnzXI)mNUMXon{P3$GMJ&zAtc&!z#8r`2AAa(8_d)4QzLThz zCGi4=k#FwXVy^fvAftvXmnEugz%Q~uF;9c5z93&rioDkJAfEa~`O$NYpu0z@H(Bu1 zlS$=NeIQs{AdfwT9x#O)wC71Zs#^?&ug|WbbBL=7g)7WRVu*=hy&stopr=_M-9^t( zQb)BpyALLiKjrvVbUV7LC=wM;&p^1uo7P=*4y|ILAETGT7>qDL1Lb*?FFKl)k9`Ky zMWJo0@WSX<hDrSig1nfjlsd*ne8D2_wG~cu;0fszt%P$=UK7z;Q>D@(b`nd*1{N?5 z--ps76fX6%?+WxlV^h^EWMh^3^#lfv=mdtC#lVbxfXO+r%0sfZ9aC;GHZAK`{queg z%!Ag@Z5!GQ%(~!*vg-P#2t=p@=@HT0Sffz8R{zrZ)B*iuXEZsdFC^pa^s`Vs!h1EV zHZ6n?HXjVuw(ePBE7o)2)tvU&Nn4>wv_)neHQFZza)5yvayi$s+P?0~oTFIWUO&;D z9L;AypY0Av(o(NFjri$aU0gqPdQ&bM#5kr9$9)L?1{A@RHiTquum|>_nWe<U)f3r? zw}dixXtcs8%F+A+HD{A4ICX=i_LwJv9Qc8CDAB9FFuT3%`@r*h`H9m8$xeOjr+2G> zB~z=z7z;Z7Y^BYC5BXlhv<uK}@=Uv?&<=ExM5XEcQ9LUFl1J#0#P51;)$J#h3yy%W z3?6)XFaYGdraynDKL*wLOF5@6DBh>9g#HiQFilUV3v<^UgGC&f(RvKER|mA*;j1wY z$256f@yPE*i5^|1svr*i?>3hmj2HL{7uGD1vi(JJaul7C8$Z-Jq;kMQOX%#x#t2Nr z3-XjZGxLk+kwRx+c8Qn8bRiZ4f-wsb{_np-WHwB}hR{HM^OWo@>D5ThXirZR69**3 zx-mil?dCW{v-BH9wPB51mQ}%uI1a-s*1U$NLVUht8RaJTYr**X*~%yW?9a>$TbD)Z zv@iEY5&6PxtgS@+TQP|_votnAY<UiU0nCIlIGI&0Q4RTmq9w$H!W*1TwOv387mo0u zI?kZ)<7g$*sZgKXD2gJf7K{|k*G_Fffc)@<t9l~XZ5n#V-GpxP-kmZSK#-rpDmBzg zm-!N`u!?6s*bmI|*zoG+=+mzJRzj{0IJ2=K?4tAh=sEvgwhRIjA%in)oNKB#nfmA{ zYwg&3MU?FVDU~UMPAy5Y_IrN8+CjGmJZxR);tWhz9S9&UUUvp@OGck=LI$Gjvh9wY zCwRCPV+|znNRnS46bJ%LN@JVU%`byX?q1XujMG!Q01=nPnFt-%7&xti){RA#ZE$#m zjksiWq9kS!2BVh4S9pqq#+s1Wo#Wqc@q3Dw(%`0uH`bmrZ>I|v2g1nYLSjnJlKvaM z%#pjCHRH5@4J+^wZe9IUa}J*|v*hc&W`2y91SCGNzo3o9*v4oAQp5;<b_NKkctsii zX`TH96iz>*cC8Vj^DL>Ydw<`0yQus<JQ5%1xQpT|xcZWBhGlWkr#!66<Huf{egfwh zgrDY(o){m$|43U&9T!~-MbE}#qIjaH(e1ftIBUhu6W;gH)&nmF)a<Bp$~@>f*svkd zuQ0jbFm^xjike0a$eN~Z_E>4pe)X8c7T5T-IlSrNw%?Yb5zVX+XsE-d)dXN|Q3f>N zWo&qhaA&rqxV<k&0Y3JY(PKw>l!8N0p^lv;bbXhlNUf&JB#f`3yttF{<ES16TO*Yn z_eQJnSm4Uu{9gF8Uc4}u@tG_dXw>|RRz0E6ow=__Q-X3S#h7p5U3FQNb1QuoQYvfh z5Mq{}DX*k(0tAfd)aEPPDpz5Gu&X-t!AJD-<|u&0&N$!$p?TL(SN!q{y)LM{Q#1PH zL2#Utf$O<VzEKDfW&lk419dUTk7_&L2kydm)2k*s@_uQHNVIwfS+Fs38i|(s0XC*Y z#1_qn$h<N{&h6cep4d0twKdt*9xG8n?qqJxD1#!BD&qQJ?aRfix;5I|1fnprDuRS9 zV^+PUL!X$=)|`{4;F$nYP&%ey;&4#ilj;nZ&4Jp31DljG!5;xXIT<F&w7Q+9sGEyA zd0%zcD>|apQ)GKbeB}rs_S8smIi3K8^I}Vw6JX{<n7q!G<>9^Lfly}xMO1>TgDt45 zBk{1WF2N58;x5m1i~&>s;SK?zueL2%iKTl`H)e^x>H?EN$QUhTg`YZ#wTl4{ZBTU4 zo!Ej0(oL%^!$u&$IeS?{vQo~1=Y%QfH;k8FI-RW0uZFuq*^sx>;7vKdFg*At3IJoz zPhtSvX(~iC0n5gJzIkG}HipG|($CRJ4k+Xy8J;@A(jluO$r>HWANXJsGJ$^EEB^GS z<0!f0(cAK^c>`{^iI<&6VDDKr!!z@D<5<2|KVE{In*tIvAPhJi&9Cu#ST&c^3A)aq zSTvA$wlb|QDR8)f|3;G|w=m5YPqn(>NM&%K9#%Wdp6u$RPg$`xvD8NwdyUcdV;fbq z;~kL?$iA|xZQ1vJV^g7rMJx81%Z$WAb9dR0Eh!HN%VUH}?`0;ruOXj6Xot&lk}#LG zB`gV|@uL7O;`a1w3Y5tgD|+6|*urAtyqM{x<;#reA$&+lN4_gfmn-$sLL+FZ=Mk@$ zm|jO2ZTkre0|}YNAssLe<OI8Xfngq>Jh5m^Y%hM*F!2VFbNul=0E1r>i${#vp3TEj zJW>aN+GW-N7BTT`WOhG_3D*5ogt-pz#s1kDFhtb+e=FqnRJV7?O-dHscsdx0>ru)V zbi@2W$oWAa=Z8$blL$dSq|yw1MDw0jPOz(MQBBy5M{XLAf!|bXhWEmrpXgrkK{pdd zqLqko<Y%70({utZo37Oyis8N!6i9EbXy7B7Kv#p)ymrH9*k1@LRDYtYZzig5CaNpb zX}sGi+HODkjzyA5x@gHsU*#4U&^`VITO(rL5do!PNZ!PR35PI?u0A<CijaX$@>1%c z6MU7BfbWwbWF@$IIu@SnsG{&89y~=!Ry!(JVORfe9;S-;lEFxdDvKK8Lawn#lyR4F zc{gAA2$wzSi+k&ZURK*hPdvpAta?gA`Tqyb=m>z&uZ9;iBug%6wRDwm0!EblZbZ=e zDOHQ$9%a(t<ontiW_q#+lWY1bGAB#f69#=%X5?$IBg20PCj#N7F__0zaX6z$80J0! z0VL4Q2&`V}iFsms_caAPLhb}lC|8?(FOk#q_EH<}m|G1V<f_ckX0LdBtR6j9zn&ve zMS=~1I_W!`ge)LVhz(JT5nEMNyc0tNJi7dKt~&%kA62OA<sCUt*!EOYKVtxHGm1-O zSzzKOAZJ6J1T^r`KE}XwT~G+Gg~R=?!sMCv2k!XFF<L=A1NKiZ-|p_c{h@gK^Xu1m z1xDqaf%5OwY#R5HXC<{N0`+x<9}L+l59R;IJfDiOj|%>a>huGk1PCipej}yEPgTvs z{oQ@T>YSC}Ud;B9WXTwxYP(4RsGYe9xhpCTGn!&6M@33whDw#%vx2g4c8}=`yLk5d z{Tnnf0-5baeQqEJhk@FV?RhG}1_ps3DG{0(o1zf^1a=6TQ$hrJGyrL;R3Z=wIO*Wp za+$gg+0Je<l?JY8YU*T&TD6zR!@`X<)1m<rwi-#go#q6n@JaFcthveMoPbX*f1<rs z@L9cV=z=%Xq{O$8N?sXM#e-K+7k=DEVeCi1tK(+m=8C7tmRHmw(}#st1Q1IxWftln z&{(cT3}P!<G<!t+?0fs1Wr;I`(WM(f=-LLDK#0uaIB{h}{DC7`G;Bj%U?3xJ+^dV4 zm(!WETXdTDYx4vG*%qXT1;ZB%<1-qPaY#j3cESxzJe0tHOL9Qm2Y<Hw&Zvh;q6O7` z-j}5WfY}L~mAEGOKzfe%Wr`<9w%usq$##dqC%FF5YB8$D&wu}2xl}vohg>7|j`vC2 z*pmduE?RsDxjP|wCuZ+VHnAFcNmhOMa67`L005DMr&|vtsjG)=y6lW12ikF(AFwSC zxoXgHPm09^QIYsv>xWBLC%|N?Fif_xYBtVi@y{2lIHAccE=Q-qrATTLb~#KK?>CGX zc4RMyFh?@11*AK8A^Bx-mwHreYWW4x*DwwOiWw?!j>Cp*R4K)j>TWCmfDBPhz=YtL z)dqYonD<aHU}z8V<URrf21O!uLp|w&7UEGj-fyBw#m0k6)*+f)H!dP8h9sl_q9^6a z2>hR%Ystw-U%nv3VLjms5K0+*KLSg2Dr8E0oSBAnGGsgcqhDbYhE>hewBVp^m1rul z)o|Cqj-+UuFxfaDOZ3&9M_`kfYPtyF75W2^vh=E|i#&#qXj|A1YW>7Eze(*ey&qMy zvTU#yiZ7OQ@`8FC>?nRO!yN^0b;=sC(^$krT_)5v;dS)IVg{k<M%O*Mkqf~kIz|AK zF4-C|09X=EwaBnnBx6c4Qvvg|ET%@%WzNcwY*aHJzb>J)Xl2cG@$mM?+uGU;l5D~3 zZ|@5TJA9*G99l*JqGU#GSQf~z6C3FG#u1b-quLyJk082az#66GpTVSLV+UoX^{9sV z3%YGT&{I$l?%_F@@7e{ogJ{z;P7UC)!hvzLhzSo7Q4OU`-I>B6VCRq~m4pjxVFvJE z3^Hh0l$ps~<xy|}x(Xy_xZl0GBsdUUvEihk0VJV7;1byfqfIN75<GQ*QbLtXh^H(R ztIMcd4K1jW73kDPfW$q}j^;Skyc@s@VkJzFo4nzX=;(CWZ7wD;S(?{!gZnDCHi9^F zQ8O=Fa>08p7ykJCM%yl`bl`sB{rR=wh-EvM*`%Ga3MFCL=80>f5$2S9&}(_-*?P*1 zN$i0-5h_Up!nEf{90Jo8goYDrwq~{lUL0?(m?!2~UmH+7WfgxFi>Y|7ws0Uh({VIY ze3PFs7fRDX8Xg-dO6dyc>7?(4sB`UpEJZ5#pb%&!!0jTOEMT@Uz-@9Y8e_xu#i|B@ zUG3FHmdsC~H1=$pLEl2_VHeF}%SF>VxJYD=z1mn;@s0|=E#C@13y6SX7=l?jIS)+q zt@9BOSCFxd7tK6!ThmpnOOk0@J-4pZpj|rsuo{s#ucAPSD1a~4zv)psrh-m{h`Ns_ zX1;+mk_Jm`Ky9`@bCa>HkYu^ayj@XfrQN@jruC=o{c(F<3rgoWvFqR)v>h1)WEDPv ziGZ{xOR0Qfi-MrMy=E>s9$lh8dkoU`EO8dP?5fi!T(<QcgGKdMOAyeJ>BOl&cvFcA z`5KY6{@PYmqfmM5G9+T>IY7d0J%X45)o_1eyLAt?qw<{e(3MEF$v<JQPNueX_Afef zzpVr_s3-vX8?S`xOmLTkPU((c@Vy=*<xd-Pf&W-hBRxY~&P)FrC5)s`8@eRIL4z>E zWcTPcqk<2meyFEYT_xLs5BLgGbKj?$`yQGDm9zph*;;PyX&RxihdZkA^o-pkpGu_g z3($fYx*7i~r0yRXl6D_aAk)&nHA$E)FP$R6zx(V{5hZ?pZRt`>lc7!F$4Hcz9=i4C zpWg?9$kj<Q>sTY2EH}9Iy}^QCNR*ddVQxorxkifYstR_9b`S}I%OK}zt;P}}xu7M0 zI3NRba=tK}CHi86ci?DxW!NbW?RZd)(BIHp>_^q9Ig1Qaz=r)l;^sF@0A!<vbR5;J zcyF3~RZ=i$8QDjpk1e$nbN;z=9;4Nl&%R^6NCUhW&1-2rcdymz-WNPZH)YH*JM6J? zs~nB8O+{?j3cOarjNdok!gx6bAP6gmn~|L(MN4bpzN7|`Es-~F<cT6PMEI;=t>|{K z=E5-xM$&$fL_uk5JT%`&!K`J`#BeZlbu>4T`b84*m_;{>>r99YDe4z{lhNq{-84Fn zFc<@gpUC3E<)zFnzc>-T3@t9o#&Q%wxEuPc>zD>HBbJMPu-&>wL8OGG3nlWfAWvxx z>(H4g$i~94GF0W~i#a8!0H-n8kkAa{wA+X?sy4^*6VXsmKy8C^985uI3UHL8W=a9c zAofXciKlod9?kR|SX^6+{cm|8r9+@NKefPyqgBO(iWPe{)N9Q#BU&Pz#3&YD^rAo- zrfp~Vq-af=zL?2<1M$lmurh*)Q&E9Ru-ZGl@ORNdg@TD<g=kHmvYbvZ6eKVbMzc9P zS_i77X7}%RiQ@<66_*7=+*qJq=s<TA)kc#}1~0`RDUcF6E1WNk`~k~!%BEUqmI)E% zCODNBH3piZ>FbL#aWR{{oVabQ{hUnWSO)5F;P@jv$?xCC(H{4Ioz|0X=Y6N8#!#k& z^>+^>Y-7+rqIDE)Mmp;dAOAj`jyal{L!p6d<pQ12!1a&36n*v_h{{Dc;+=eLBrv*o zx<)3|Y1K3kc7x0>L75Igj)5bZRU$|y!Zxcg2C|Gf2tzeE3Xwr~3hkP@4IMy~Bb+FW zhk>&htUtAqTK&MjorK0dkc`;yu4RYgRb*wzw_z%mi5)^Om-i9kr61$C=5yB>D~bt# zL`m}Hre`8jQZz=;<Ov|95w!iD7V^|K^s|Qq5HbpJSX^RG>1`~8SqsI0w)xMgO}AB; z3@|96Nx`Rau_rh!SPZ8}6?2<i&i;{U4~&(BJ!G`#a-O#X=0v{GhlBNJvAOn&=CqoW zvwBMUFFGw67abmR4N|^v;Zy&O+~{B-$;V7NMo~F<4ZB!HhNT{whB$WObkUd%NZzn~ zM#<=0lyeT~hHqeVV@zrONK*%!ABtfUW?xq`@)FiT{ZzuO=-hgih#I@X@wLKXr1o_b zoR-$jMnzE6FYYy=D`}R2ZUAFIoWCjJmkgm5_8p4_t46!}P+e>O^qb_3<wX=N)(^@w zkx?n>N=K$(B7T^T=f+V^Dbx3?`0(gsF~vu!@HghlhPyO&lx~&++v5%HgeWf6BM!w= zho?5dQ*t=HLYLku(&cd2lB&eqV6zy17AqL8l41T=+bOr;$r@~PLmy!gx0JqED6S@d zok)^<e~LugPS)eAFYxj-!AWHq4C3LA6Yj}Z%XF@$YI4oS)vzunUuD_eP*b;EIcpYo zB!un|iVmd}EY1#Lvh)K?i)eaPBmo^IEuP<Gt|SKw-P~ix2K=K~cMuS|A0ROMk40_9 zNs_RqqF_B9FXko1d@cTa(M)G^ftph=@YoG!!zRX)ph1|O*}Urq3%cDD3sx47ij!>t zv+XQ_(GE(-tuj_91j35oFDD6$__2j0{#)@FOKr+h@c@syQ20@?xmK(z53DW7;un>M z!ceYwm}-mPLUYI1-nPhlintEq@^3(wIjca}@5A_Fnq^`;*j<W>rUd0LqCk-Su(*|$ z=(Zw}2|iz^w5--V4J{?j_tW{7wUn|<mW^ZPBBol2JoV4PGa(UgJul3X)+1bB8&aii z_-cY82}pQ{dDEye$$Eb>{ni+B5R!;OHZCy)86wSex_=f6W&J@UbZvQ!s8+}^^$zH^ zDAVrMtW|W5q6?kLXA$*;!Yr#y4egS0b-g|{uuDoP!+0z$(E)aL;(3=Jd-IRmzfn*8 zFaM3}Vzy_{E@|FAv}<!4!d{Vu;iQ=?j`l+wJ!7k_PR>VkdneSj>BrOZ$tBt<tB-Qd zkMZ_X`Yd&X*%$2?ZdWum*#shC1vUw5*zgLSjlfqgNx&9G&eL;h=KZ>wXX55r=x1Iv z7g`t+Ln;9EW!bd>Lm0wQ>dXUEM^M_%QgbD?frvF}r`1s~?3@vaeeNqPlV3B9W4V*Z zf{9GU`bBIaM$cx}5V|o}e%#Eehr}k17gJ3o?$z5&qKnd1d+qK&YC4me;K;aCA=?uX zmVr#;4BEN3-v?emauU?+;EqyKwNyQUhpIL>dGh`SJKYkb7(Vk=yK$gz;Y@6|ogIIm ztL?b!j(KkcXR8If(H9lYM(VjTVv61Z5YgnXFdM`_p2?JX2z-5h5&8#t4;i*)!Fu2C zP~pf>a*V!UROq2C13aDf922&dN2ba#xPqj{0UrhwT7Kw&5&H)|&V^Jp`Xt||9V)c? zsI}GfJ|5S@ELpzQJ~n*X1O&~^Tpo2RD)ZtRE-v`E{IpwByxJXe>j&}d^|9XSL8d4` zZxEZI^2*F5YOEoWyxkcWrHVlBaz`K(AOWqeadSbgllc_in{k;|VwoB@X~;8=J$+82 zNYf5O9oj1ewn;Ojwg6ANKlVcfc8eB9*G-Ws0eI5;T)i*s3Z*vcxSs4%0GyzddwB+Y z&ZEuj*eG((h9qS#oH}`4vDO7paR9&87xQ%p9PJZ+_kV2$q3_e@oS3vO2rkK19)+@% zP~twMkt35{o-|#ky%rD>EMq8A3RmpM3>h;#!H5UgH(Fem#P^!rJRI=^1n>-SPlS&j zZmtb_Q4d#D2L#I)<QDii5I+<rhBIa<GbVK!=(QR2h;_cIIZ;(pRAqDR6Eyke!6;V8 z^rwI*lc-IDey9PY=-&{}^SVcAb|_-_V67ju&T8t5sm;|rVHcMHt{50|4T_pXsl{ni zR&f)cMz2?$!(Yu~7*R+@<a|Xv_~h+4w~6Ttpz`=(_fag20~15Svz;Bc+t4nQ#|=O1 zMgx9DH<1+Z`As|Y(W;^XxD%k8M1kBEmSsYxWgILtY6oIAOWIanwtW<J_3+_}-bnOE z*SD^$JMj=gwNpe!W3lFKnA;UBodo|Xe*IE|rb)mNxmP8MDPtI)?_fLapXC}z)wXFI zB5u#xwC8of&dtIHyhyPj`DvvPO<z2#BJ5?_TuAWmOQIb6uskZtR?aUaj+394=ns>b zf+80c(|c_7#r8L8R#7=biFY9jnF_n@qF^Q60HmL9Qc<<k@8ZauMI5(%Z}@2kEm8tj zu*Lc}x<H3EU_t_s^J9oc?t;g-i;<rqLynU!jZm{E>hEHd7vb8dn&@=0n!SJUgzXKp z)5W#1kQ`5H&2J(yWa^MxXVRGyFJA1xJ=Ak6E}cHeh!$lG9YIVwlG=8oIoH_U(t{IG zd(Lyw00(ba3d~O2vxhryPN12;oK<J_r+xtB-tPOK-n}$U!Syn_-YQZYQDPTs&FQfl zX`QMl)iUc<Wr-@yb(6p-D^(4YAsMxy#1bawyxU|1RW68Rp5mcB+H49l%!#+iaRaTF zCQ}Y?L-vTOmiD2+#HO%fVX&^84=Yg9IC>&DOq$KruSm}M^W+1h-SpyS2=T&|)8bBb zfbH_ZMOjW0H|8*vlq%x0&zzp97ofy9fX=H=^BvfhxaFw?Csm{CFAg0QmYq)>sj&QH z)s>{EQ$>%kZ0EXj9MrgOsT5L?o3)7GU|Q4Q_H4+?DL9JKSlpKvn6J{UO$Rm^83}f_ z6H+$vn_C{QRwof`D2t%a7b6=`fPm8&{oamBz`GOq_=Lprfh~9=c?2oz(;fW^&Wo^v zC}U?)Vf4r5TPiNP<ZtHq&<(H!j+-U!dR$(1Gc=`H^Q7Iqp-0-*Go?I8kOVp`u;G-9 zz`S6+nj{KsmUO|5J*FN9=;wjav&Qw`?M~xQb5+j`-gUX(c)x=@VPh-4d7Pf~;X2xN zxLJn=6!1BjRpp26M3bAVG%1<3=4b;9r`K-V(c{nWNI%`YZTB3mBhkN)vYCEo%~g&< zb{0N}11>x#hp5T8a#Mq*v7-;if%yx>>dXu#5qgWH?p8jYX+FGW!pBT?d3iay_H!MX zN{X{#J31W5obMldGAJ@ET&QwhH2*Y`?D&*_AYoqOs`F1G(jEXH)gHDbj9`{@Qwv{4 zz-=dTLOdZqm9Zn@JR{2^7UZK9oXP+RNqP~TBvGVyUv6qmAU1U64ONqjz4NC~CiV_e zx~jU|){oe$FLloh!kM)mUTl>?!%PMahrSWh+YYOg`bjpsnX@TO8IW%{hsf>KKq2F} zHh6;0Gl81hPlE6iSPA5taMW2+?+3RcwYSR$729HG5O$InlYmT~9F7znKrAb(A3g*5 zy)|eTc^nw!8EaMdN)|_(F47J8a(0tbgWC>@TZdi@_d%$<E}mwjkHyv!@h7w{BsP&O z<p3=sovA5B5GlHlfLd-kG(*Fgqzv`lH|1o3etGqT;;>JLRnHj_eOHX@(Wtf=P@Pde zW{S3cK>hd<Bs`V_!vyeQ{@SOjia()FH@C+?&S<A4vKgVtO64|B7_(XujN(XPb)Q1y zuXP-WLAIO)ZuU}_fc(t4qzC`FlTLM^<EY&J*zzt+xNach*f>r(G0E4Px;|PHzp5v7 zb5WsmQ21&tv1@?@2%qM&YOGH)%PHA_;{D>;2}JXG{K+)l00FD5vwslnki4ryV~pho z*HDJzk1X{4#D$lS)VU*!KRK#Y{s0o5`qZ}vB+eUPtB_XO#n03REguXa3d~P?;e*r# zCmBL8=RY8!5rN_?<dMojtT)7HLdPa{EOtkcc9(=@N3E_;Xwf4fU$$lTGzT;@<bZNk zU~R8gJ(*2ERFgjHa;?i3-iSH3VY=bQv^U4#^HT#vDbjVIof?TAsdqWDS!IqNh-;8- z7%Zg$1jx!^9l)$S^X$wJeK1ZvLt|(*K?TywPF+4oO`>Il4uXbqfGl%#%@-3hEpRx^ zZCIkRXzKF`D$z~PE*uW^or#PO`ziefK3z=8uwy*|F~bAeF06TTOmJC9x6m4lJ;MO$ z=nC?aiM@_VUzXqSgP$RhV_)+`=_I@S_%o=^My$A#RUWrds?CT}F{OVjX}KtUl<dtc zT(-jIe+pVCt2|b99{>MHK)L+mpu*!;lIvVAfYKJ2hqb9HJYa$1lqb`Vp}mX!V8E;V zo`lbrZF%^#yaG4j<=sE#{vR^{j~Rl;8idEUhvENuiX?QG_f?IEac^gmQrN9H(LA0n zRimeN({gEo)X>V$=OcXnX#0E|&mvIEeO_pN<tL90s|rUxZ|zu)5#X0BaqHFnEOCwP z%@q8gFEmL)Z;NNJs2z#m^W~J25?r~{^8=i+fH0J~N<QC=waW6tOAtoEk)H%@Nj%Dm zm+U8lma8r6QVjiksg%C;{+E_cp$<sItuG-fmLW`-IOX1Qt&lS4xcB)gqQKTZHWxB5 zT0h~huQd8EuK_~BiyjCP0OfD)COQ#GoUjZX7*P6VMruwrC#)tBS~)vk0HtfZp`Kf) z!1RsU+7>>|yDUDfKAnyhLm{7o>*wk8vWAd<-aLG6=wWll``-b4I+{+<n9kf_-xz$F z3=hy)#fiasar?@0tQ?JY<x<&t=}f>QGHV858+0nz0no9jwtR+DrAwn=753Gdxcs}~ z7t?0`W;(1!^iH_8lM&e-c+EhN7#Crcbp|LLC`YrZ9D<O*++w%{uRYHoi(Tu=8e2JN z6!Lg46O}HY2$BTb%Kl#F1|l3&+z=pym~EOKrq~ZPlUXw_Z`7*F$2o%o7}H5^LHIPC z*Qadxe^nJjI^JhU`x7~1zyZAn!E_d7Q$>x<MRr0LH^`7AmsM>BrvH~C`p%J|g;Tz4 zj=j6U39%dDGiRBJOr`B7$li-vW|!lvrl%%xb`uWVF1fO<tD`Z2jbyv(EXAtw)b7?0 zVgT}LLdJkoiqGUEU{RXZ%tWMyo(C!E<~Ab#25r=sp5WauJTHszcYS<cpBW*nWb;~t zsuqz=DpMuSCMEi$WKsw(=7oy-H3b4IS08uZmWjq6<3wNrd}MvOdc`B>p0}N{b5v}M zrMlAnS1tEa&L&g(>+>~Vou#?67zr3CaT;fNfK(&?ES-)<(<_Yo?#@GUj~<O|q??hw zy?Gb0aw5nMJ9>H#E-Gy^iM1tMu^EuW{XkdyqWeM1`K`*#$CvLo0&-{z(;lzHmRhbQ zwUKg{s62HfW&3m{azzIxkQ;~BI}6`)LllR?kK6$u9tUJms;^}$_Lq=A*k?&U8yJR9 zz!X>zMhTjF0?b@wtsAvdxOspCqZ-=kZa&=H`GpObcWr5+^1V>XMy{C4z-o3w$q{~$ zi=o)R5KP^i9|0q?_EgsucQetCxtUR5xlWRU5ok*3zHQY6vmu5ps>SDexNhxYDmTq> zmhsQe_@(ME%lfrkW@ksfG@+d5n}->uS(YRLnQc5@&h5<o$u2e?C3nw4l1jOf=*5bm zqbvvVV+*vz91@;t@)IG2?bFKX<oFC8W<iKbwso27@JZNAiwrO-xoU?kBDjFVRU)R> z=HgEpW^+fnnuvSMB;?ByY}rlbtq*N9*Kr*@e})}Ze~ThNge7PL5gWj<CeCcaYjDS| zhY{BX1K|ou=B<Zt^&|*A*_-0)SK~~)pzT;5g>#tQ`N0au_n8@d^|$27iLdx*79?rY z&>|RzzUqrDs5%z_N+C!t_}=PlLqsbQ*Gu2Ax-V&yr*6tdY7OG@Eirw{t79QHrJKFb z9O*Tr^T4<I%=FX!q@bXHb*v*3OTuMB6}ewe??~>drCEZMuEbYKYD*lml(8o+9u^-` z=adt&XU;p@=91%1HRFywTRENUGMF8*sCA@12JHAK(Mu*rvNQ)5ySpp;3nOz-u~$$a zGl?2>XG;KcD~N{d!-Z#qMDxAEhL-dVrGdyTMgU?<(oxZlTaBQ0dQoh4*sEDR=L-;= z06wHNr}FYO+=_*;A0I_fxG=qnbKv6<<B2ankuxW8&Njz4zRKyFId#LvmE<L*JkxU> zUUWsC-cT<U-ZK-KKNHu?^tCcnEhW!=47CfAX&24bnz~_>qvUDMj^^tkP$vqv;oAM_ z8gf!;2XfOGy>D~7cOr7T_1pukb$}xRAItCpHd+I=+VgFRuW(&*Kl;@v{*A*$bB=U2 z`J)V`Y7)$3Ek@NA3uk;xp&OD`zOHQwDJxP+RoKOBDA0B+YvO=Hex_WzYQA9fuZbs7 zLLV_rz868eTx-Qs|6OFE?Ls<%qhq1Pu=DH+h<EArblXYB2XLLK{C&7FKCeML195v^ zN(GCn5?IyEq8wTI&~-YNC&z4$vPx0iTxfPYb8Hdfljx_$UoHYRCHO5Jwf!K5H36!O z=6)}fa{Bn~_hO&##tx_3Ih~HCvkr(Z&0p)Y?ce`)&&5dNqFYpyp0k&C7$QDBS9l)k zC^0%Vd|G^iJbpbX<ZMg&Zk~tk2N<(5EqH*H@xz0H7-{|pzD0VHw8>5A4dB{+pF{vS z{+)tNz7V17lNc~6ufYhu4O^F?K<wIS#{Qbx{ZG6)1mV|S9h%R!S^r_ltSchX*MvTu z*&#jGq5dR-uMILtqiSFnRjY+OiUPWr@(HLCfn8Y2?aI5bOj%UDC1~!s#bRnkz2Y!7 zI}il2GB;Tfzb3@%{K$ITa<9NP{`g04@0cDWXiT`I$E+Y>MN|j(x$R1&uAoOTw*g6+ zP(9H;e>&qlRPsa3<c*I&^w-agByt7E&1WxVl=&nIM?~2le<~rsd~P)vPZQ?|t|pQz z*^Z7KBy0MKphqgUn3sl%IuNML{j$5!k~1vrqubt30s$Y+yl_66o|r1}VXml^W3u>I z0urGqn2N}?lv&8F-Yr(}i<k?)c3pGv&GHz!*yphtOVK0|B7Hf2s52YyCuOr2RZ|bQ z!;PoflNjch^?%MGE07zHRRfzlViN*pHAq(@VH~{raU5)~;FhgB<O0c5k3>JkT%#x$ zMYXgc;!wRvJ)ez636u$Fr{m=U-c5EF+Tn#*f>jA-l*<shVv3N-@SvL2Rbwyl9*)zC zdgQJ3@s~k#S(e!(_h?%dl}b|z4JZ_fy4WEk1r?>>cW`S0;(`Iq>eHecRcHl0t48EE zHmAT_{bFzKGLxF}WrOr_C1S(Z<G+*d-1<xE!jPCM&AY<<{Xp=%Ozik-w7OKm@4+ex z?)Gdr6mdE^k2K7;&^_Qcv}IxEJjfQua}=HfXNwU=o27uY{%ti<>!m$O_+@65L@A9O z8JS0nFMjxaOb?6pwCA(w;xbm+LYZ0lv6?kiH;cwN|M5-SNxs-7-!7jcja7W8Esfuq zL7W^Xk;R!b(;L8;1+UG}h?VjYL}09&2T9lyInbcSA2+)pXwuBL$|RSnVr9Rx9nc{- zYzv9?i{)eo7bO_mkUoqpxsdKwpH8bw$B!U6R`;nI#NzN6hheu1XMKKaTbWXhXpDfs z!R0X)Z-Shg{w*+a8b?xK=0g0Xj7_~2FU!-wR4Rz6nlOkWvtQ}K51-JSIBLHa(4oi- z%F0yl34w4vJUs8KoE@+Z&<26halM?=vY2Aw2|)@~nsPsZd8d96;6N3EOkswZT%4MB z!5jrb?!u}1d=Hvs2yI8BM!faujhMQBm_1gTQ2iH!+T_E{sjk4-(Htlk#r&))P0e&- z1S<YE%p&87Og#Zs(f*Eh!<Fgy4RtB^f^W7Lw4e@4t6y_1z<M0{%JLe-20uZS`+0|q zyj12rKz&&asSR&k7d|%}xn{P)66Dt4;guOZ<uKArpPhwSC7lMS>5PlSb~wCXc2%__ z)3D2(N9jX_vxUR<ZQ)O0M~>F(r&crJNbO0<x!j=xdgKEiz)$nmXH=7x@N*1{&&bJj z2Jc=T{`~Ik0EvO^qc5@R@ZGbw2d`efd-nYG%lB_y9{#w$djK48M-pL(3hE6n;Flq% z9nPGl2Sl{T?nVr9uxuP2m=`}jd;3EKiH}<cJ&QOxonzji_x|M@LzzzvQLcv8+Vj~Z zrqw-|p3SeoUi8&;F&R=^lJ2vES8K%rSPW2)L2X0la3jOPIjc_=K=?+~Vo{#YDn8H( z!#WdXeY=0S_u?feP_aHl(8qd^d>ibZaoAVW*$3n*4WX!^>k=5oGqBU8t^SR{vGWqV z3eLfo=9*Hzn3*;#*+qS6`a6W79K5o?G}N%|){NDh!#or*YMzW<@iz!xZif80zX8M^ zD?Q-157xG^Hn^kMfX1$Md*(xkmMNGGLCOlK&7*UD7#RPJ8=0QW(Sr?#Vgp>fwzFt> zCR`|U7o5*8x7OFMuCB~x!u0Ob+4;I1;reScv2PDvJ~A+Km7gc0s%c!dHca4bc42)A zw!elKxq|h>Wy0k*Fo3SxOll8mHx>otBKaw^0Gffz0%p=mT%iZg4vM{l2gUPe2YUy- z;+H*xnLi&EzdU<qw$9<+%Y$P7UGZZ7?e5;;-u_$j>s9gW?Y}|IzwW)=H7r<-9uL*0 z%NYy;?ieti5G%!DlZAn!nIP1h7{d&2a=rjY;e2X1^$D6nSF>^5kQF#xj>q*JQL2He z(b&u;gEK{Ny2I)c62{Nb>F662Y7LY&^P(D=Ue<{~d5dpIuVMCXIka4xXcK&DL(OQM zs|B&Cl5O5Nk<;OY&m?)WFc(@1|3X(&Q_^e+$X#+(Rg*2O@UVDrb-@WzPN&nEA$-?3 z#eA=0jA@E405>o%O}EWp?LjZ9Zm@p_flccP6o*74)dV7cabgh9HV(*pcrd-312fs| zR%cuWo=_!tw)P<E1KF`=x|p3p=<w;sdgvg-ho+9#&>sM=alQbYF<5C<;ryC-JVQXs zVR^}6WeDzRMX_LL0LZ79t<0&y(G_i4bWk4C5h1M^$H0S+tU+cCB>j8@@}Qy2v`F1z zTlB7MY;&<mLFmrGvu-W|=16GD*>xd#aERGc$jH-}*3nB1X`Xm^c!dC)SaZ|59HF5f zMBgb#&C~{^1t+a))>jQM|GZUn0D9;NSRFjo>c+Hl4J<@Q)EPjz0gy0^U5#;g93ZwM zLO3WXY99T@II&e6DVb}p==b}_H}t3ALMYP3>B;l~V<m-|!7$%T`-fYh5YS|jGPfwm zz3If43^}{EnFGDNId-xiiz8Z7nX-U5KqMx29w%vLbf1zfThaB-Fg3`~@i;N-^Rb>T znl;shZQ;o5tK$@gQGwsRN3__I#~6ytuxmsK#I+BIo!#jI$wl5Uh!DuaZ?lS>vKa#B z@f)oJU=FWhKLPIVKbFnKoAQ#OapWL`hh7xFecCD>_lj%N&$qn-0x`2R$voKX6_1ZW z^K?>;wu*0aae#F{fHl?ZxOib0HzE*b=LeL@49*;MjaYjm5OYZnO2Yyff?&uRv5ue2 zF8f&4BRp~0j%F-(8G^-&ZLvdO#Z6e$R3~2(nY2am-3|eGuHIJW`~#F6z`bxCPB~n| zLE7R2=jFVbKnVWo1?<)1;s^f8dlz<><7a%f&(?`CGchaOzHBUQ+!A^YB8)x$vf;jM zb`WopqOcfRoSoz1`L!hmM5Jw*p@RW9mw~D}&?6Yc?G8fPA2*vCTF{L-+%|fNhns$! z_+e7w{RwS&H=;|xc!Iu^ImTHvE|KjVEyk0kWx6a19VeEF4o<@;`h!hd!9_U+CC-wS zm5}m0u`)>SJ{s{cT|Ve~ybXG=<+Zqq_CmjI_9m1Bu1iI=PKS5VPQk^AWB^QuQ+i~p zg>oj-A>4B>=StOO58$1`f=Ku7wLV>0rD3=|m`qtgO<T&-Ou+&A1|2GiV`M6*ZO#Eb zRw#=>0NK{}g*o=n=}@$z!zC!tt@j$q^zc{LE{0nj9uqE)!yzS5dj?m0$q^3eEbTI1 zK&j%)K`>Zjv06PBc_fQ$E|g519){ik#p$US#G@Jo>13av<-K5&;IxdJU0-3Xcq*%J z(YduHo(Vl(Xk!dR<`p6+7^FV55tOCzGQjkM;PqyTVgo6Ua||Fkt~@(?i5H>*IJy~K zm_JTyycG`@c+Z<_XM5O85>}C#RwE>O^of#d$;3Yl(>`Yv9Yj0>CStUx01kAQI$Sq5 zAh^k%8(dS(NjM(y)8+{jH%&s!0?L7CT92A#mH?<39|b|&0lyTuP;Nc&i2%yH5OWb| zdV`5;^u*-@c?&E-g_Br|t^k9SQDJG>rg$Y`&a|h8m9Cp~M|OM=QA*Dj2}$UEU&wTu zV6YVWWLu=Njsi(qBPLN+*A&scQPc^vw*yN|nHbrhDBT_arVIqX2SuEBxO~$OAP7_X zFiwC+Ja?cIOz~^j^tSoKDJP}kck<BC6ks|xWN8n>n4**sgg2DTldH0I6K=l2hPiiF z{qug0Hy^~GQSA+WqhV|zQ~hWYx$4cEn?b)g@L9V!2v|9}#{03_ZWs*&qS};A6%d;M zBnq!bwnEJTGMugmIyiVj1izyU;(F3ytJ(SOQFyS>DpE<>6-ryQM(_MxzaY8hkCeqF zD<S<li@@Z#*LFFZTFP?U%-136@gSSBEN&S-$DvxqdKfRo&Yi5hCU!bEx09mks!ljx z<RC4YkjcyCbAK%7k_-4*wVG|&K*F{L6$Iuj18^jZAN!pYDQ0<D-B(8@61)k@KPrfi zj3<i}eH3T2>DbG<`kwOP1<p;Wb+Ka<aO{!C0D1&O)pAeTh0?ZbM-(s!zIyh>E-@w9 z_xm6^-PJj@$i9jZ#}|#wqAXb8yORP1KK_NY2?|ys72+n@c*rK%0SUp1E-zkW0qg79 ziHJOE5U972^RjBgd02qB1_zADgUI$lLk#vay6)$?+jTb!0m3js0A0x;Krn9!ya)!z zjF5PeUNSYXVkh3tF|-+2;>C%G^e{!A&jo#sPDvNeTT!iXRZcKs)r3_Z9DyfuYT<E3 z=Stf9W<{9uhkOXO5Vs{|noac|3y96qggG07gLb6|w%r)X44ID}+Oyl-Wj7P~<m~6_ z0@Pn!Ni4s&mpg939U@1f)UijfTtp805w%tc85~qef#aqP)#a-Fb_WZ!YT|5x*!yB_ zR^}7bc1pT5r|ymFhz#I$$0_4aBW?jbvC3=mXw_h3nFB#7Jbw@&PBuD6RFX#(JwH<L zqzrQJi_3hnSz?QFJ@O?#`T8atKYVU+WlT1jI_iRUdA$z#_L^%*7t13n<0$#2&P*k+ zKr$=6n1PwT!CO=gEgHI^0}pR*0**MF?@qFgl<?+e`I!S7D@Hm`@q4LS>YAM3;oQy% z_(a0z914dffcu9(zI^w7ckj*1w+Enh)~8680;NXbkQ1ZK$zE4yheW&Lm`yN-nZnkX zE7v`I0{LNZuK`=~HZxVQR4G3gUS%<>gizRv^J}HjENF)-(KcW2)wAtPSu2T5i}gh< z;MV_lHJhT;GHT$y;wh3nXb_)5tc|D?9(l!<$|yu559>o(a+-xu<N$Ld<ewDBF_1&h zL(sT3p&&H4(3zt<a}U!m5|YxT2T1O*8QXH%q`c5*swl8o??qpH`K}X9n~4WhP6VsR zrRjwT-*dR4Ys9jG?Dh)NvLeeBAwU9wzRyK16`j=$WPvK}R%v%8Nu%?lM>$Tk<-Cs1 z^32pGCML-0qs0tDPQDf`dL-K1vHlU_lTLhrx~YA-ZaLwgW}|{3tsw{h{N?-oce{IU zpS|wCSv2#wVXy@b(=R2G^qOLcl{8Oz1`J^}$0Y%_RUGAt!sn>-cw+<fnwyWmJx-)Z z3Q9gUCI9r@pW924>ae9D5IP~HX-sqf`ak}XZ|?Xe`-)`zz!sR*+bTMjb=66y2Tngd zbVmp(D!p|hCYd^cNZ1wXQ_H3(iwAtWeo&MnFg-p8qmpF1V>AK8yiTBMH+MS_q*Zgu zBL$63L~9`EmFP4`Or6=$-OM&kUr&NP<rsxNC{LwKK3$#rOtwI?zPyc84vv88iQU;k zZ&~fo;Th|+a0pQdbhg-i`t)gWxWBuPKla&%80?U6QM}p_)H7!H9BmvsiSt#3l{hP2 z1xdTif+C}8N?T>DQV!8ik`9oO?*Z4CNc$<Lok!(kjkoe4m46P-*5+^59Qmh)PE5zv z9L`(do-&yLE5s{|fbHTkH&~eyn0>4pv*_G#_wgE8mepV#fa*~ze>g2irwfDHRdKR7 z0laNsxKq?9vfl_2RG5rO6a(7y)S!p}$@0N<gYDXG52mEECG7wwJwra9=8*1zZfMO# zIiouOdsIBtL63J=QVj2gQeoCucuOi)L%Zb{BC%!+<_xB!vtl4KEY}jGJTRyt1LqI) z*)u3kFRIfIprY^#eL9`Zd``DO=S{(@N*U9NMkw@{yNI%$Clu6?XMtr3;>A!&Ph=!~ zKT8-4i{4mnWEHV;#M<3jT9}mlbW^ippY^hWp*3|`6BNO5rf~>uVl&7xY+PRc@sFFH zOH-yFj?FhSMFsK3Mu&Uqw&rDA{uzs=g8Icnh+>z!@zqMgB+fFb>JuLGD{j20Y@`l( zv9f`P=<a`WXMKNXeSc?te{1XegT+ZRujgi<aDNWX$k=!<^#*3iGKAKC-x}hFPOL2k zy7_Qt+E_Pf-%8Q+^G3PK=-C?O+Uph%WasEu*A313xExvc-CNPRbH#FrM%zG_hbP$o z>?96Vcz#^$8w8;ki#&~FFmxL$VbQ*Pwqo1MY=Tk#WwkJDRBJs5l=)=ZK+?^NA38`+ z82?ai5_pR!r@WL=WZGVl`%PF_CUA*N6TQu)G*7XX_^EY-dYPpZJdT^gts-0~(SUrc zIVVK65CGys5}nXOZKaHeL~(`9Td{;OS5&|fKNp;j^?SWL*YD2tyL0^(=lYN6Gl$(# z_*IvJgss>Q@um%vHZCXE@DHFQOfHth4C3qd4lF$E*1h}z1DJtFT=fyMeUGN+W-z!p z8O5GrF}yr(6%P;&g6jd|4lCned=H+_7R|*WMiPHhPRwM?!2L|H5~riGA-@>u5`zzZ zyJxNYExd101KBw>JcYT<my^fCPA@7%syOHFMSe8C7)*dU7Pks(0qMJe&!?^8(=#Z7 zabVS}>#gG2y>f4E)&vOPLN}dJsn6}hIh`cCa0YQ8ZRefJGD7a0lEi~|p_~!nuK~6; zfqRR$_Ry__UOd>-3oc@X{#$g(_wWI(U5?6A_IPw>z85)}!jZSb3x*ceC33~Rd<2mO zJ_d16>SlQ``*@wca723FgjM_T&+Tk`fn#A?ySrqt;$jOVS_@*$oSK7beKD^`f3Un+ zs(nV#yzjzS>L)<f+w|w%siIbs7~;*ARAR3%CJ$1Qb)-{7PyQfrB0?BLQ{?;S*A!8x zOOi;6k|N_&1PCFDp`4?G@9T+y{QxC?9qC!(JVG$)NETJ4*0q|SV$^>4YiRJ1EGT1Q zU;!P!4W7txSZ$5{YmvnL4i+;JV#3Lsk)rTQI5^>q*Z~mey`ZpLlD|$t<Uk2w2t`ev z=EbxB4i+8u9ONW3texT6x*e?0Sv%U0)-2G{^=(_amMv)|TPh8ieRto3oC!NSn_eMC zyqwLZGlsZ$&?z3mv>UiSrgX3HiLei=;%@1zB{WJcb%xmr8wnMM0j=|Z?9_p=?zGr$ zxoNn*<uqLX-<^i*f9y0Ql*G{?;GD}vGrTcS&WY~@Hsy)|zjCdLX>6?<OIy35wb(9M zZ^dOGh=Jwkx0DgzWMR6#EllxvSBJ@}=dC6&3;SluXs2hVmg1cP3}27b`8IDurF^Z= zAXPz^t8G+YNou#8jOL4X1+|~w75u&zPFO4Ahh<m8w6?BR$GOx#@Thb-9hI}Mheg=j zZ4d}6qz=qkIjk3r#7EzDA6)R=t8T}a?Sw{4Wr)@JR!Wb`x5MM|Gtk)8VaKgo9&f?y ztk$QmbgNER+BVcSoG4}c{&5AvA>=BQya~$Hw_DisYTIoU$ix(U*|Af84nfFI#p>`y z7Gj@y2Ax;auR}I+7&V%l13Hb$xkDyFw#pfV@)&(R%Eftd0<y(V1EOMkw>7v_AFp>z z+XC*fW9tAp54l16j+sbbGiie9Cn-I33OXTRPr974L4vMhe>uih8Y-0jVV!knC<Vn$ z&Q3@6(V=z9<|u`i!giX6dQ_*}1|;p^dOZ)09(-zFdXnn=S=zT!lJ=&yZyK7g=8{cW zCrrST?vYkw;TA4ENQRwn-0lV8CB}myE6a`@QRmLY;Z7BDrwaKjRfsp0tLJ33Fy&=N zqXn46W^s8roz07Hhn!B^O!`NcGPZkBQY?B7kvhMcK5_)!EC58zKr25Wq@ewUMKMi* z$ul-^RTaN3AnGW0gTP=%w7$+Cz13`jq@E<ZyphM5s+6%G7)mDtj6;2B)6ka<hrqt1 z1To)_Gd3AdL>Q$)xZB5VTQfn^QX*bkKA4)%;R2dro}m{?$-dL+92W<OFW0#HICz#_ zjWchrnQi#w8VS3|4dX@37C5vbrC5*FvOp-B7&Hgf7N;~13p~Hh7Xim{F2w2)hB9mf z3@(=R^v+uBCkjRK@M;321YA!L0L{Whi@A!G1h5M{sqAOOzr_3GYI5o1JL&ZE!bR!Z z0vq8;F~BNlg~S;lI)<MaPRV!#-U-W8L$7h9rnmd|<#5r(GUN-fc4uuZhW$D51ZUIu zRpv#oE^$4zwr<4M#Wh}wM4D*|6d!YiDmZ~E9bbH+o|w%mcJwRn-l-l3!3s1CZuTKW ziK?dT|5G77UfH03q`73nd9mG@c7^K`#`b{AyTwm{A}o*Il?pNufgs6hMlA#r)h<nl zhlcG@K*P-7T85@e#K1<Q<KRfb7vWZw!00|*!@&}4bFfcJ@Bf`(;+#!Y=ws2c3b$SZ znRnuz)2{`s0iv`aO_<Wf(x_3zkwj+|MiMQCRa2i&I8hYo%eydz_;rmODjeGp#-=e1 z@6D0p9HYx9v2SNSPOyt<P>sryDOgfLY73hff%7Sg20fh?UUAYj6;jg>IBqsEF5|Tg zrE@YhqqFBDaWGyb%3cs?SFo6vx|HzPIs>6ug5iFEZ=g=dY{V2t&H*@bbk(3+g5j+u zHMvxq63mfM6grOC3o&<V@6F&{mLQg!VI!yJj#Ao4An_`vQmU1PVXHZuF7av@&UoQC zMqFNU9Kq<6Vs=whAAEk^zyJ4*>J;D7T~m7hq|Sn%QZ)pIxT7x8fOEu=Y916=)icOC z{~wEL#u()|+91e9D^d&CJZU8+c#nJBa|tp6AIm6qho5&#0zm4cX<IIOk1NHgih^o$ zov<j;n-mj9VwlyB*`y>oO}>z_#baJV#yqf5Fpf~(;D1K&Q{;`HBaX_Dz|TM%YDch@ z06$Zd`PbAqf*R!Es24OSBAJ3MM60O43(uU3V%Z7G&hHn@VX$2T5RuH`CEIRi5Vn-{ zq_LFV+zqXJ{AhIDD>mrnXgSylZ>wb!H9LqGWq|o#AWEgpuOlumJ?CEsl(uNhtE#_b z!P`dmpL)shOW;HKTM7^SH7Pvu1JZ(iM^OvGwJv5RBdQs+leqg`hjW*^{k1Mp?lu@d z<=KjCt(NAYhp(!r^>6S~7v4Bg1WVoWxVz+Y=&Ka#-}Kb9eX}n89bZE1d(Jq@9;C=O zjmyi1{8+2vvaDwyzyP+Y53Z_LmEBM58|s=vn^fu!JA+XF=2U~A-a;-@iv2rPZzPw- zcy?c~;Tn;M$ptP9gz;Lgz}HQIlu9XRK-GTx?0YUJD)wIX4(i2IU33*l1aeEQh-h>x z-$sboK?`eN5#8o!EzH;!JKM<7xTxGDutLEvmao8_fwAH0q5_sq%D8f$r0mYikSq*C zvDZS2#Diejc|*8Cg-U|#7V<UxOg~++$N?$ns9e&n!8@J9ozCG-=kR$+jaE!e;@lkf z!S?1V;p+HVN`SnCeN~@DEigBj`!ew03rQMX3xQ<6noY+PlF>yqkOJm*X|lhk|HBg$ zd<9eQb4VUHW1$wrir68OZ&8E2S$s@1VF4b)!D6gcEY_3tcQN*Oqb5?et)?!hR8VG8 zSzH-81h?X(1;Wrt;=GvnW;-q?EfrLU){p?EYgk7SV_`h89zE4MOA3;@^@IVyP>2#- z(gr#mp{3_&Q_!^yr&^#$qu7WS=I&TdAAV)42hw35>LHL~#&;atD8V|0BCQ`Pp`vRj zB(q1|SajEhAl(Gc$AP8dLMpSZb=3_%PgknfO@!mwO4am3wk9~fq9GyRsv8<(E7HKz z&pCTTcj9is$2af>-U{dM^ZaI$9=S0=*0b^yV^LmrL1s*}gNnbJ&H$0^B=sT?B?$~> zdQ03gZZ!Ex+RX?`c~K(GBN+I(B6$Xfyf7{!h9f=Ka9QMU(gL)J6P1}ziU@^vL}659 z+Za;Uu(70!@b<P$Ia8#eo?ZJ1NQj*Qp~iwihOolO_I;6V!uHUPz_pl-z!eQct$I>I z>`=+}a$namXpx?WRF?vCj*_*GQ^0%EQMLP#ga?-=jFjE3I=2Am?U6{@?QuMLrekj0 zCceI<z|wu0w*Lll{vRm#+fEMmL=u9!d0PSDq!0&;4=_f@aLJpcNNx|IW-sSBaLUU9 z>?>O-Cif|PSjpmwf|}DRc1c5qA#**Z<_|~OLpy04NgXh*)6^M4DcOmh?y6Z_air<Z zvt?Zx#mX8i-=ii;(}RX6+oAvjQWGoC!!~-s;`)UTl7$Brh)Y&ihKAxG67=U4-lK7b z5B7h4_u}RISNreYJUe{9|8DOG2>lqlv!GWtheO3xfsWoz=Wh-8wsFjTMoGNHjPVde z3(25-l;(u47r-09GMK_RD_viCO<H8C>4=~LaFqL}jC93Ot}ta5T+MWpbIdW7phP~- z)s0#ou5%|>xs$8h$yGi}t^y$#L4#FbI0%SZVLAX5C*cJNORSy1Ix4%?vBJG5MRX`C zF{dred?eVw>Ro>8wmBXa@A)Z84(OYfV@Q{HvV(9xbsPb5l=pE240y9C#aXqFr1SiH z*7NNYM?R6|Bm4C@+5f@3x-2lGlr!7_oF9T^>Q{xWo|v=Aoa9Q~j;}B`p`o4bM<3(P zcj0%hC}-yjq*jrV;HPrdSYuOY*lZO{hGZU1zy<>W)MB%5ZlH)3%Zuc$24Gy*J`80F z!v*(?rczO&Oanp4%^HFb9}<ORC&<#UwdG|2(Nub%Wk6F}IEG*-rBBUAkv8WdtN5_G ztR_RyW3k$nQuE0oHHJ1Fx)x5VGa&Z7`k*B4V$SNC&X)f@okHpfGszCTtJ&10Vu3M$ zipziqWT{uskVV370La5rIQ6vFbeVbCwvG<rEP`gkZ3;h?6B(X^n^9ZnXu`n`(#^8= zJi9y`M4tA9+#=f{<i!)tVmXIoXa*VK+4t_*ria1XkK1f@#GV@L8`YD_^mw#Ed6K#H zjsADQMA_B3EP?q#r}8+u+%1vQXUzND8369ijS+b|Hw%Gdm>o-!g@HbcN3!m^Y*)UC zCXy((8_H0a*ni>lAff*n$ni~XFvUavBOX{aw;iB#EmBzql$318GGh@&b819bXXP<r zX-fB}u)XVl?YKf~s*)M`6Tcl~rA=RH(^Z1RP-L;fK%Z2n<pSw7TAWanAu?e!q;k># z8%g}m<@Cx5t~lRe-!<a_?Rj7lIJIo*KSQxRbOsD9YXe#ixJIZoB8{vRj^m(2wXRjn zKD#u{eyR;aUD-D^U0d`3e{ciT8zMfOQvjd4Rgqw(DKYgT{1tOEo!qg_+{|zCjX6_t z+cpUuw$y`a3flq5mn|5N;v;j}E#FcLT216@6Y66TP2ofX(08snZUR$4Trkx<{Q~X@ z925(o73Vx&eJKMfeW$ot6cr2W7#8D06Qr~o&dRF^Sg6#e7Z`KAg#FVwF%0&HPCqlZ zb@LqFudwc15*EDopKu@sN#c<(%mF*@$1KLA7UzGKCuWzIry5fYvk2ho4Q8pqs644$ zK6ogW1Jvxb*I0JPOMLGA3rP%Lz;)#%CC|osu)oBsH|`=suX}!_NU;HnLG`WS8;juY zga~&+ggYU^?Tj;=cf@p-mIXv+6Y`B!57kv1JX)2TpxkvdGEbDNVXEw~64~WNX(@@2 zni^|G5#RF-0V{`)K0b4uZw*=KM#heL;sHu*lahu1c{!^;nu1}6)`lc0B@mOteTgSu zuNyV5+>v_r^|~6Dmn&K~k`WK8d~Upjf<Asye5*hJ%kYUu?3wXg={0o(i8$8b)T$u7 zp#(h=+S?(qx%3>x_u{2O87yfo3}8n>ys$YGwY7^)+fZF8e3oi-*W#A=E;Ql-QMQWv z$s%Bkp;o^3IbUnyYJ?HID!SzcS`B^u`!;LsJ+HJR)&fJ502A=XO)~>>JsfMWox%yr zUszRF^1f>|W4zkjk5JpHCALk4d>$4jS0M=43Fx{;N^M&qW|TZ@)D9pfz#>@s*2)%x z8#%FvVW|57{3v*OHW37%We}*72@DD}^@u2n-_#TZmjpNFOm<Og&1K8Fv(3XWx##uA z%BvvZW-Fc7fJHDf38@=vHbQ5=e&Bw9XTC2W>ACChRmI$}Ys)!i0M9TUPi>~q(&|B* z;_=4jUj)NkvaFN5-Wu1EIbJCz7iDCCbYGn8%Z{Q?kV6NC)mdQDs0+hJR1+&aG{fw< z1}P7dxwTW7IX7&xJo7}+tnq9J7dyatIoaqmO8DME4W5{wl<>(`3>EW`^U9{LVrHRe z%o$(K>gkrBR@<!^FJyY#eEeu*^ATo-^S>V3c+Wgk7XiUFgS70TgoIEP_{>AH&Y|`M zME1k<G6IWf>1XMr#XXph($24NCt$Bl0(Le9>gCpjX})WUFHy5<Y<QqTQZ;5%(GPT9 zjYiHxayW(i+7!JPT{~U$l>dZUZhb0JK5mbzz8M_whspFxn8z0^uHk)!F{CNQpWj?O zOqf5}RoCG>GJ4#UjY{;W{Eym(_FnVoBsV)?PvT_P>R0REs#zLV-U^*_P{8}V2M8!o zQxYO&FhaSC$aF9wwpK*JfQh6@k^=h?*IDzRU_N4FqgBU#+wsFk<YTfWY~X>k?L#U( zuZ7SY3khSOfK9N|&4-i7DS1+#bKWvWQVz~yU^G{Nc|G!RK&0D+T$+@RKldwMA!CP$ z3q7>a>o6v^EZ`@YaoXv(Bw(1lYkKmlGRF{!xbc<2e)G3a$3w~n%7SkTX$e=j9KJL} zx$XGEcjgIql7&0T!tEssuSqWm*j`T-5c{FA#s`sc0veREiNe0;^06$hWHsqu)gS81 zYFL-Jmhfx+b@j0ty`P@FA6F$XCD#1G=I5ZS^vIZkS0t%9=$?YT{RB)nI812yrMt{9 zFbAwtqhb}wE}g`IGu7JgA-(i1I<JC(PRa=q$ln9Ogru&)MH_4eWbjE*(rSsKs2t^j zZgmG<6iwPGaw?gg2@mP&^pTG(K|v@~V+O+@epd;)8*oT9^Jqx7y!4S_N)Tk>zBI!% zCy6y+lLt_Al55l5=)Np($)e~@<|HPbNw+C}g-nq*FySvC{n_Xh>2uAR#XTxElR-n_ zQj=EAic^R$4QMzb?hATSp2h<amtkyLS1^kPEmBwzbAve>P0M-U|Gq_;IcIQ^VH439 z*!)0_VkXrOI1JJ}AbdDxDv~g9beqtcHJGU!jo1nXD{1(w#mxKk>>ojuW6KUiu*)6k zO&REpi!e`T9O)#mYl!wS-93MIE@Wn^M4?zyk1u%VI<5Z-m>?#|Rg+UBrWjg?mkS$M zp(C;EqE8BuAo@#p63ASIRyG{FLC{0chHa%(aqvO)ChE&Xni%_{P0Me~Fz!hU1z&(| zSh0*WbcrYqf<SRELe#Yg9wu;&cQLflI#)knXp3qLoB+Z!$spQ|d=qHp;u%KOo1EcE zbZ?@IBdZ3xF~wDBl_8Me)MiFb?mHC7m^=9Ra&Gu0Ji~$slUmly-kluB^m+-7r8n3q zWz@@HSVGZH7B1?^#<95vY_?-MFiSIYSO(oEEF-pB4I>cYZf_i+A=B<FtBpQB^?Z91 z!`JavolK{fJ^z%zpvOY@`pvyIp*=BohxND$#P!5e@r|`jLobefvO~Q~ig<TDyStu! z=JgCR2hJ<Q_n%M<BzlVnC~T9rNLW#bhrj>0nN*2{#(aE0KZmGzo{$27p3wTFDP*`T zO_rB@4qR2=`1ihPZT9S}vCLG@X*~M7YwXERv6-%WG_EImY{Q*NjXW+t1tr4pxxZN} za1apEx1VNaX(^hZrL#m;qX4zG29mtDmL_+tm9!X8uMogO+It6=QXjdbY${?cTIdRf zNE&=@OWCflXy-uj;t7nUi%qR-$Yip5jDzYSRo-{@+4piRfk3;Ze^Dv6XA={W?DI)- z%9i)(E9pKltK$_ND#QBRY|jzx?pg)py|kzhxBioOWsdi~@Nw4`6!xcpA};n>jdyuN za7_1b@;IAC?px!Y2OxTWBp3)u9=dpKIIZfDFs5C16SiiygGYWhQkios(VV%LPG`c- zc6oc$oJZ8N`V;d?dL&LT5~|=`qQ9icg5B%3>w8J<ufqukSGbqk*HE}uEW;bG%XaM? zlxLM_>tXRYlB}PO%JF41BlYl8Brnosj!Y5Igy7;Bbf;A^d##%&CoPnFo`!fomi1!K zTx+M}3Kg(T(XIODeMQLN9Nt_DDI@Y1hFDK2GCN!7DsRRmi*84EziwWE^sEZu2itZq z@s1wL`=s4#*;Y2wt!y4k`f6KAugQi))O@kcuo!B@BH*8d>lxg4Q^SpF+$luvIPN=+ z`%5@($9&}VAaqII6UWRtSxyIhd5$YL{B+qA!OdGLSN7Bi7TA8SQ$Kr><6NfnuqTe! zdN`lQEknklhwi-EDjs_orS*N-YIl9GY|AqD2RcQm0r2AElcOwn_RvqT^Z2_?ubKAy zp@JgoH-W<rd?!%!`U(W;TqCV=X!EBCn%xYVh4O83F(EVB%DPC|nOi16{VZ20U-1_1 zQgWr_7;v#b;LIT<=bff#7AK@jszd+`g{CP-8Y#BUIZ)ZDL^wjhtlum6(92i%9EPtV zWUGVQK~k1@kngZbcNp=F7LSzIanL!>_wH<Zd1)rNTtFrt?*#3odr0CIjm<_6F*b(k z`oJyy0y%S<#h43Tm6Hw{$boqk#>V&Tx?vdcR%$hcrP-4hmxY+`a1a6(jjqWoa(dvm zzBpwByiou^bD)@sIh^7xi=blzb+GrOx6U~RgYizpeXQ?W&bzJ!xE5(k5+Fp|Tp&Oa z2UjHr$va}WJlpBM<r-vGN3CF1@dG!8Ge0D7BX;(*svr^B)V7d&QYd9`<-;rj-YAc4 zM7KBtgY_zhq*<erskt@%9e%-z!<W7uN@MP0uYpA{!`a+0A2nFvfsDj9aS;hk#m<&5 zz1s{zdF7dg#LeK5*q+hqtc6OiAg)%*pe^V<x-=!4x`|OBzJH|Ecv$xP#AT&Tod;h- za3IqM#*D}5g2k>NVQt)oWxRMs@zKN9N!*T?ks8|BA*hZBHeoi9TvLgfd<6y+BPD)u zIj!-c3s>o6-ACX4Wpi`$@t^zu`mY;*`OC&%KtNDE+T1`m!k2LiC$p;j(2fpvWNs)= z{2oY-D7A%yc5jw$O`x?S$nQaF)H5a_kP2%IOvFsJFuy1#F}9HNW_(-U+0mk9{$La{ z)1?*07QB#l+)s1s3gNFZ*J?X~o?CjN`uInX12u+rB>rlYU~>s2h*_wmI`l{&FC<Aj z-zAJP_{h~KZ|dlMQ%|O|==>GUXAKwdyt9XQR|>yv$+2sDxt)V^Tn~62xVUz0l&;O~ z0wQxsq}ztzTglM06Vri2T?)h%mV%^Bx|FnPk|j~$Ug*rz@koC#*moHY*A|mn1Pk3G z2STphy_nP0ss}Y;@ZqC_ApTy79nIjfVRYRn{MEbb=I5bp?;*7CjRq}F_zNGfO=PbP zN_vJO1ter#ua9CVYL_ptD3C!KCcy5;jkAV@0e+&da{h(ToF&H%pJ4T9GrNbW*uOXN zBOF9q5+B`D&<XV%+<_xP;fvpjS^!l*s=sjH3;i<meN9#UT}2OfB7eLGbl_gq>EZne zC_bQZgRlEhYdX02WjQ?N5jDkT|GN$JZURB!CI!^-+9L3XN?a&bHwt2Vp38Y;97Ilz znMkJGQKyqzkWi@wA<e;9UrtlR42uexelPW%PAfhraN+O2D+&}kUxfYa@Is}GpS-3S zMbuOc<%Scfh)DsLzw9KZ3!IS*)i7u?aIM%}p1or8aerguJE5x{)@JV;asXLA>E*^o zN9V}vK3>DHqEcYkoljrYpQ_=S&yUocLN+-RcNzpEDLqNXzU_ZkJ^G7mmEB$?`#|V( z=@r@bn2WY%IfRg#<FcV;YTchdUk)jW8xwPHIsnX_eN2h*aBJboWUH^l6qs>*cU<h6 z9f?kC^vrUO`4$38fLtkU4S=Pr*{3(CN^%n`=<ng9M<Rb2^#87vkw{<gm#F^bF7-Mu zs1%DaX~EN5kYe3p0@}hl-@%BB4P0c3qGGn1;o1!caBlGNW2I`7EkB3?D2EY>S7s2z z&EWb|4JeN3`(h(b;YWjg1b>6`h-Ea~Vi^Bf&zUo#$e3rOT3*pu!&hBZ!j-cc*v?gs z>qJGB^et%>(7KIbCSsviaJ-rmA-|C@6>_*~e?UuZbUFA$_i-w}<WxPYKN()$N0L}p zw*0hO*;bMe6m6#q3HkkHEY@4Ln@Li}rBEF~^wazgJFCHH&0a`3)_X>X_;=b&qjg#E zXO6cSL-UgQ3gyOnOi<*2!uh4>jvO5M)}BqTMx*H!>Tk=YN*2Wr1b^^u%V%)@p#XYw zV<TVbrxB#7mDI=-$9a_TpWxk*dC`Nn#|a)BM-M79s~bmYIGW>NMtF8E#?doa)_ZzJ zJidcB-TJ|l{F>}~QZFp^z>YFEgZm!MvuwXV3?A;tC0a+bl9n1ER9vA(;*yXPB7AY< zD+(*5qH1$H+e9F)T=DRvojFJMdO#f;Don69h*7BswzoTirJ^ppj^~DW6P+`TSLp_H z7%vQcb#?{hEmH2S%7?TmErL1lM7?$Vl3k50%d(bgZ93eRy(4Eg>RFn|Wlh>rdfqQE zu-6*W6)o&lr8^V*JA<D)gP+?Q{1oS->4`x@wBY#mF^Zanh5f<dvloAT|I@pd=1Nmw zZu9E&J*G2x4^kB<^7`4kA6~wH{_NfRH+yg2|FXAx_#+T>j6}>OfK@Op1Gyx3QPms9 znE@EhDN+qP#Z&!ytH8h8Vdw=)kc0_c_J%6QT0kqJ^l%7&2Zb<dd=FJo`fjuKM3EnW zWBY4$jRCsC9w}N&JvJ@k*IG7`3uRAtjI94P*+xOHSQJyP75gHNkq_j8uOO3@p-~hC zJifJHm^hT!{xofBxWN)_KMzB)t830lIs2s^&Mz_!23*7(IjfYaCo7ctp}cG_U=zsr zf{|0h6?9`x|Fbg`RMFoT6%6+Vm>par3>8;XNcE1L%8patJN&dBC=wT5mxgZ6_a~Tb zPF{ylzzA?F-3dH}OybD&@3_?zZ%aZh*m3Z6qvIG0INEJpf*H4u12|#av+d4j-}z*S zb8k9ujhS&gx6g(8MyYN#NOj|p2j!TN@`i>?!f}q@)`Usf9SmFkYoHM3S=h2)g4Ydg z{jG2l*Mc=6sqtDo1&9;RCc}M128@C?eNm2{GlIJ}T^ftd*5qU=Zb@5bDV`uKN%?jH zK#q8<$4B9)l!VLF?Mplc`Hp`E#o-sW9b7$FLV{}pg>nhlwV#2*c5`T%8L0^4&1|U@ zmzX2Is7~X6g@UlEt-xjbkVcN0dVUS9s0M`$GC}3x11fjQ4NghMeRf0)G&DkJagk09 zF|8WIy*wV`rBpM`b+bQtVN2-VD23791P?QKre4R=J27CI>>y*9Ls6c7`z$`R8oLPo z?08dNQs8t?3+TA%yziKQ-9w7<1UwLkFV9Ua;KHt^kU*q7tvDz+#RDU<e`xvChCMmP z)&Bti0RR8&z1wmlN0uP?JZq+Zh*W7BU?vC>%v4GRDl<E8lDMn9STd#RDh^i+0zopA z0D^@8l4Pp**k=2%X7e(()@J*#wrle+W_x>H+qwMAR!{HK{KA}@yC1&<08*)%o;Jdy zBoJ}@`0e=d<HyehmS4gC^mFr_gba!jZ_42nVQx~rODPJBV59VNXuQ%WMy6v@;ZxzC zTjxJ+w)&)|z|a1rX^aX2DDMQxxEzhjx&ZlnSf^O>4_NFmG!M2J*cxgWD4|v7y`r>$ zc0k>JE%{1hWEiZem-z1w5>+kPM=tLAdGEKM!)J?n)XDJY{d>Lp>gN|*U+S^o@|Yv5 zHjyy7zCH3|=^l5p04EB#)Iy2&1|yi;v|P7^zAWkufA5jLD9<~|^JqtLl~u`sMrAc+ zexBYiMQ?DzR@<Y+z&jK__jKM3dm{Fp0v9IU(V)cFX{lQQzhQE{6v&j6m@PD%ct;!f z3JWwSH{@)X=u|(>PhF~cjY-$#?>(JtK<W!BPumYPjP|1e?nHO)lu0`aUS<z$26}Ye z?__1lDD0M^KImhk5!lO$j3Ua)Xemwb$2>|8h@Su}?h@3f1x8mym<>uYZoqq!n$Wg{ zXKrcr%(JQZ5|biSm>o~Z1<pj69C|BH*a7OagRX1d?=L{%T1gQD>B{2)(XF_9g*gjf zolRa(D>x(0F4|Bw-mC2V!}cbNE*Omj73cy@6tGKSmk^^J3cS<B;Gd-802Y7;Q)SqV zdsJd_bY<Ndu(nl%<W6?js2w&3vTQv&OzB_KCNE+}lX<au^vBM{Pi)Wh6D%*UjmE6$ z&Yhnc+eT}>7!g0wh4+3!-K_s4WzM8$gYi6UxZf}zN2;4rvk4sv1bRVRhG=OlvD$1u z4IdsR`d7>$FkvU*mH<uYeMtuO^fYCMLo>0G<{fFOAgg_<33^T!3nw`oUtQF+u#{R* zhP6P0mMYHu6iNz7Bot%5=BL$YKAx~#oDeqRg}qov@i>A}McBO7PJaGJye*q9+sj#f z)J&QQdS}yx8~lUjLoa?-JK!d!-df)zs$-b}kGzSOBy1!(>vhBcQ+7;rA^cdAZ~~T9 z_Pcz`Ln|*otlMq^_{vB9j&0p-utD8m4y@%*S+Y=AH9h27`D<LEo+s{Wym}w$o^7bA zZZ{U`t0LRv`$NW1938v%gDi#&Zb=;6E69Lh2UnbvY{Qq`76_w89cSF|UKdk5)B!_i zJdCmi)D&!C^k~yWS0-G=N~d?7s!Z7{)F!&YWwl{)MFo9$;nD@vS&vgv8#a|aaT(6c zS+=Y@5kits=d)^Ed7B*?jgbpU=qy}&-4%vFn#fw`-m2rki$8@B-n0@0ls2@Xpn_=m z+}Gb_SOC7d($0`p-+qe$3_<kuj|7+uT3`Q&!JsF7`5V7f-R%dzo3xEexo?=@T9fK= z;XQ;}xmL>-#q(%TJNlCm$DUrTm<rUv@<~nijg%}DBcXa%2Hgx*&g8FRgaeE)>BNu$ zqj*8IrSj#95aQPzOCtD&f_OyIh*Nhv$gVt$XxuSW1)P!^RFhwsbB*hQRkTv_-z$^( zv>4}j9k5J=4UB}IKjlDwo5A~mCePN^=09%*0LS#=Zv28su<pY=C^8Fr&oHjrujBwl zj}>fcRDwd0oF+@ap9Xmtn?Z1ud-6V38TbwKM=gwA>5qWLnLUV}#4~%w(-ls`*fOL| ze>B}M9h7)y;<SKHV$Nz3nK9C70ox36q-i<K@|oNr*B&x5tvbZ)BDq{)i%L#AkL5Ey znb85qY;Alp^wZ{$NJBf9t&L1qtBLbF6Wusfj2d3w6#JRP<A)vX*xdW)gIPA8$WDjC zlWBC$Zgv9QT_BfuYjm0x>;)J#?7<B34Mx)Hy(qHZ+`rq~vY7;?1zy%ei>XYwC#Ud| zlpD1PKdJ-v4C|AqxBfe}Kq&t$D1-)LiRx2Ao)W(N@iZ?~c@2(H4Z@y}9WDjBSOyC! zl@~#!20U4*!=*r%2t1~~Zzw~T0c;DD&7*+z`NMonNEmKK8OhY8-UW5)QVZ%Hc*y;7 z2_Ya3bLVZTjxp=))T^<>90-o<FHN>?*KkI?ry=(JicBL`-73O8*avUihP_b<MEb@7 zR8n%9R0PK|l*fpw7pofl<rwzGdmE}XybUTU>c4I(3&q+2N6-TmhD6WWL9KV9C+qCg z=M9b?IbF34k!BxuvdzdM^6L_FB^_n?<RS{eaY#ys0p)AK9n?sp>HsZ*ArVnb!AW(A z;!g}^IaXRf5&}Es6zi;S2~c`hF-VUv0HHx^M<=1Tlu(V9?b+{NDIARoS+5$WDK)1d zpE_4`UxXPuFe7k4_T(pUUM<ket*oZy?4qmR4~qdBd6Srdd7^ZFrvNL?H3rT``#Mf4 z_P(r349JSs-3q6lI@s*V#W0Qa0*Xq956}|s!0Yru*0wp+=yLYJ@`Ho5E=XZXq$uwc z=QGT(A#wXqTYo@O(}CRALU(FoG+&*ul^|NvjZ)eq64o$EES{JsFc8kX?lYU_v29R_ zM%315zdyV(jf~@ccL6D2#i}^odRkg;tWU2Gw7<3G8?^S3K%;8B6ENx9%hCDlY<+!s zPglPTJWztihLU`e%Y+p>2Sz+Q=wx}~Rtk><ySpHNqSXGR7|e4P`?>#;r!YtBOd0^9 zK*?O!+(g?fDL)q*W-H#ZBQ}~2rvzR6L`+Oi>k>Y8MRw($b}ak*SiA}c9e7kdfCCLi zfI6(+C&o=Lu&Xp4c0RB137K?3<FB$%KH?%-vOu&dAlk~`6z-Ol+npanyVBCZUq~>y zvS{LqQq+yz`8aSOF=0<w<)vOMkw;?X6n<6{90G!95*q?ia;>zhG$>%Zc*kGHFq-XN zNIu?WL1-c4>WVsGD*@tkV74N!jzvdRu|9l|ty_$~@!`@5kP~CVFeg69XwA>!_pu%b zKPzbY@WY+z8}s29Ky*rac7jQdgi&LrdEGgU)yRv(LK{=)!%$&NCU$ZFt&XCL>`zuS zNEfq~C8@JnQ;sg~PD#FRvJXpYRhNJmL^H8Oj|{r4`u%5P(@o=diAl)9XGr_1K3h*@ zGzhf-)cfcn-yg7!NRX+^b*;^EBMd^H=ObVii#_&7MfO7Xg3!5LimK4?x`(L@lWq;U z#j^13B9Y)GMQ!prNpvRch%@C~P9oosH_D~43l8Uq4S5@_jh#*<us6Pf)$+DO63^v> z*wjPU<m+{ilhV*W#g&q>rqpE@S#J1*a|IK`U(?JZp+BPQfQ4k0yq@8aL>1h!)nM=A z9CJ@%w9^$<AO`ZuLcX-D%7k*WwU<kc(JMS4(jeK!CTv{rDDrJVcgg8%TKm3bgT*); z0cGipu<;e90@Q;=HU_ovU0HC^8x2R(W@$D`Jq6*V6{V6>!`&#b+A)6CU=3?0AT)<H z1l&}x$&A)gae-7^*0#hhKa?;swox0mJEor78zCaGp2T{-HRgS6=~I6Thno+WL2#5< zMiy|FVj^fOF?nX&-*Lbl)9|LKYr}Ik4IUMf*fbbkeuZ!7YeN{e$UN8#1Ix?_)f_hS zhp?24mn{?*tIOkw88FOtvza~0TBoh7pS32fhH8k^^(fOoMd;xSWK2NB%2BT~kQWL% zb34~p<9tv~B1AOaZ)g{^bdn9rDaa?IFrgepd6D^PP)?8^pQJbw_dP!SOSz`_2u-Vf z&yId`ytEa{;+B;zjGxnfFybBTighO>e9fgNVQDtmmT)?YN?&fr&l_w-Ss9s+>dIw{ z{ms3tdv{Z(0Y^wH49y-AYQi)<rqGmKUznoZ9%d8IPl!B|`fSZ*8r9hY>6F{0a5=uM zUSlBE_MT{^6u~5h_goADX*?3|FKF9@cNB-i+W`{LXLh5-5}VfASufqN%!%17)#ee6 zi5VAbX(#9Pue_DsxFSruxEKay0_L;c1Kwk_7Z%6LBq;XQL_w^?ewn{)GOAXw_qpYb zO*I=8UO=u)t$5x%TT#iiqGELg0<*1<4XcsBEIBMCpo=Bsc;-oRP0BPyQbM^2I$T0v zCqhvxQW!-B8jR+{BKuHGE7a?v;xo@q4c7!T>7ur^QYS`}{x3+M((#W^pn`ZM3iTft zi59)&M5?8XQHd6w_p|eaCg-A`g-*u!5e3g#gMx@#+?T&WBa`;Kmz)h*XPSEZ>e;Kl zF~Tt({X8q>go7<%wv-~fC@@YLpD*Ddca%43M(z%^ATt3?xO^w4qa1c!j;`{iz@QMX z;mM?!w%0mNokTMHaxVg*g+_l_IB#6VU-qMV-d*$9ORw#Ks#D#x9AaH>K)Q?Sy0S~e z6;F;)(ZA?{$Rc&8mc_@JRC9Oe_>n^O2v}G@>!l9J<}{vE%lWhu>0~?w<60+kM<(^c z;aH1K_(a^!GVNh1$O+Oga?qdZlM)jc5+m(nF81&!ZTphI(nRq_=XpA<jxkP5KQrHa zZL)IA=hqUjpp$W+*~D4Sgus?44THc&#eflep*<Wa6$37rI4%-G`gXRJI`~n!C&0Ll zDYVpeG2p`9nGDN8kuIDnIOH-(R~jRwBtv==ZNARSX%hftEvT(pDHS~&_8RZK4<W(4 z+qQ{r0x*a;<~489+j0fcz&*97_R|M)TSaOUb%=UK+({s_=X4KOjBT;#XY1-v$HY=C z)he_<MfT2)_v~XKCM2-?k((&`51zuNh3-2Tf0(HqM5d=F^T|70(FciBa*`PL^la{@ zSyNi7d8GG<?m0=S?l+5_eJjW&I;s|x*NQuL_LT(ORPdvBWRFrk=ssKoeL86pCs*l7 z(q%PV-FtT7aM7kk+$is*HJCvg`}#*hhzncQkDfFY3MZwXew--2)KvyQ{)wc&szQ*W z!iH`uT&Zc7AWZMeKAi_NvZ$F3OA_^B@mT)w7UvR6H%pm68>uyl^cBi_!7W+z-t#N9 zF=ATVq2o=nOuL{lr{!sZlEG6L1K{>yTY}eJA|Q1*Kx20{Wdm+_fVt4of@z)N=5|kJ z<Iw>{WI6cF7x#Kw2OaqG<rnw9IOrxrqO7Dx4*KE*zFEarbaa%{GAHr<@TnWqQFV-w z-SDuwx#kW&H=!DDLN(rmYFsT;Bffu((>xungXlQBuw;(Gp2vq|hmFmkl+`ka1JoIm z6Gt6=M1!RHDS1+V2S=+>{b_cV*Eae3=mN>0L~N)K`gC}=`xAm7`5<n@Fs_E!MQ%Be z(9nzkR0F~p8fswEvo?+sCkQWq#6~t83XtZQawYIjimpcs4O{NF?XVbCXWcc^aCL!G zSM%vWI!TYr5<pvJ)cV(RvKcX)Wd@^9<BUU3Hv-&#L1)GjTOgfoKiPe@e)s7JuZ{5d zNS*&GrugCAhii7Oh-vRVK14;r_ivv{xGd-r$Ay8Nu-NaL1mokXo-ua^#?8?UyX4LQ zvPW^iE>|<u*}%x10z%S*ux$kzNIJZPuq}(<Rz&$ZjDa!@KEC1dYu)TI?i<Y3y8NfW zz`O@0wF@CtZFer#T?}xCiDF=6u#@C4hH7FBLgB=Q1>NeQ6C0s}fk(r0WC!djhqQoM z-!V@b_oBC!VKdk^9Z3qAVNG!)O4?fAbMkfg0d;pM_e@EU6ZS`7BlTg48A#Cjd}D+3 zXvvOnm0N4s6Z?#u_!0vyDd1aMo4wxpCj7V6T8mSjNQE#v<;*QBs<%VS2nhpSDySu} zkd9z5fJt{v#8fX=!)r_!3xaV&8rf}A2<GnOX;F@h1tkV@-gd9P#)#799fg;cm#<zv z|8e&@vcZfKK&t02c3yt}_BlLPK)18|8U$GC8vzjL0N~l<-^+Ufg+u?nZCN9nDC5t6 z`+xk$fAw$v)4%;+xXdBOb;DD%7vtx=1GCr3PF(mZIZ}_R>?{44rBU<^p{u+K4m2la z_VD3DFOWEeN!kOLeLMqwmBn_Sql#?qhnOlWZ48yV6germR8p>064j{McJZpkZfa>x z?6<VaoAY7gdy@;>rxuqJ8`~fmu_<OrI`<UrS5sJ{=E;KOOkvg+wxcLczkBcEqs#9| z&o|^a->BU|Zr6^2vdd^OXTt8a1r;So9#ViulX-5pn4#d%ZZSh@xv-yC>f?WHNiG&A zuki8_$ZFh9f^Y_f800eS(8tIPpd!#?R!;+k2P41V=1o$QIn=Zh>Xmem`#Tnj6!~Az zN2A{tIo^9@^xy@Y9Zu{oM%RFk!u9JU3ds?3KmMsDDfq0{w!6t+EI)x%`t-V~39ifl zqU=|k$p;_9#I2bB$}59T0p4$n1G5=%#E+N;i=0iVN|k%E<|s;DQzC*;z0*d<0kW_3 zC`5A@)vSP<sX`6qf|?&T4tcs}2fXNlQJp!yBRe_uzy$OQmKa@ZvAoh&(4C+S9kjjk z(Z{BLBYfr8x@Y~Y=ps94AUf5A^Q{&=;qz6ns7k`Xesq$vpd9fF&E%0bFxA<7aN_0n z>17v%wo4#)5<)Lrx`pr4;m`1mL<SSuOY$LU-q&>Cq?}yLAlZBKn`wS}l9A{EXEHdR z(E~VPe^ZR>2nkk;i}?4hYCczeKPyMXi`%AEMp2rw(LxXqHZsoIZHt7fCzVc+2FZ-? z?FSauBJx9N3=eGf5~)$H@$2C`)VAH>?l`6lEar|A9M(9`udb!NM#4J$BXYlPke66( z&##mSlUTE3%xX7S4|=oKz^g&F4gfm~>)~CetvTOyD{}St3EN>yl0bK^{~h74sWf*^ z)TI>>h<~Pf6F7yuIZIN<zEw#avqavz)e_#hACa(2N`ys{IxUrv3u#6P+SMhR2;X<C zHlnSzSC{TboEi37sDaJPNej-$6BO7-<%goRuN@o41)P0{Hc$oPv1b-{qTO>~``$4y z?0_^dJ}qkuU3fZb<Iu@bfx1JuY+tsc+(o3gKy4qXk6HP=Tt`Mlzd|Kx-QU<a1~xlC z><+5&#_%rwcQ~pJHx57V-5-AO#nIuHpWnZK`1u$2H^2O{_~P!p-sZi#U)=lcy)SxS z<imXH{zg3=Y#`C{;SbP1B=lAdw@e3jJk7`KL1lE6M!yATks0+X;{Ze1Ns$jd1)A}W z3id_E2skS=tjWy#n5kf(z1GP|e?0EjHPD39AOK0&58&z53!$O?pPLc@{rOVp-ll*f z?#<=kpDXa?*zYRrcLVJIV)Ej}`fw;kSR106-WxB4Hl4+et^`e;rH-x!{<(m!hW)N! zzq=4Ss&(9)BKDId(5M9B5&*rm0`z?geSZb$9wWz=IOGKPFD4i85okEIg`_xK2<@89 z4POd<S3%!h3VlyO-&+dpEsCYk_Z9Sg3w>4;?>w+C^nft*7fW&b3$7V?@1Bnxhw!Ws zdaD;f-?h+VJh1yi?So@eUNqvqw-w>OzX13Btq}KCGwv@oBiy}Bi~Nssyj-4oC4T$4 zDuLa6v8bDxP>52NL4U5HKex~)`RHh!<aV#bmz(IbEj;M{GTdKk++Qw(wuHiiV!>0b z*bm3>_n-VX`@X((ks~QxtY3a|wKRlI7D=Qa9H%b%QR=p(+CQ^|-X}}lQP(85A)h`Y zykFiVD!jS<zPbIr^6fXT`R}S@mPdJI^NP;=6%8=5p6QP1dlYy{T&6PtPOo=2qJ&&J zJABrIWgp+MrLW}(c-h4qtbd#7d^bNTs8IM&s-Uwj4At}Lw3>cho;wYwch5%$VZsw# z=b7`kdmCH?zOK&PyVL5-LyQ9~uSV6eeFcxr#Zq{o^ohPwLNgzG07VO<KJa<_;4^h? z)lX%2#Si1hWpW{>wtRX#CpNYgorTSD%rq{Y7Z%|gbih#xbiR*yTcHkW$R}$_vCfw{ z=C+2Cy4RG70wLoR@@Che?3LZGSOfLY=sVSoTBK`EG$Fcs3Ok+{jzFEWie}lAx9SBC zVPoC~>FPK?ZKsw<*K|@J#Z#T^EI8|%CIZ3!K>87VF+e0uh`UI1WYV@4tK=x4;d%C( zd%e%U^xs&lZBo8GPzl~q|LhOLk3Y&~iH`$6k5GT~Mz7&EqwWM_i;@20;(ckz_KL0R zHMTB4|3$6qEoq(qQszHuyYh45pNp2DN$ja^#~K%tzot@)3Z|!;%F1e@Ne8o1!>n3E zH%Nk<#0`*p@}|1!^`dH*)GIH8dTAG1d^vBg7))bWyQEfTaEE3Wcf&Hf+T`Lbc;j2x zQNj)xj^nwcv9mcqM{E(BWmp`3SN*6ZqJr#lj0N?@<^jU%j^jPO1E<=Wfs^xWTd#*# zwSMu?1yh|Eo#WK#TxCqWeW4CWG;F#zHtJf~Q3ex*eU~WETTr04p9nu2aqH`CeplMq zOu;nQCzn>;|A?pBmno*dtb+VpSI}EjK`*YLx4)L$nv3)mL6YwT0*h}}Vn}a30b+}Z zuZX3U#N>2UX;UP+4_!dEA@$Mv)CqJNmvlgds$g$nL2qrS*838T>BxdXZm>B}DDesu z-A?qda_AZK&kXKw&~4az07qbq<I90d$a)g}#Dm$;!*6`P9@Cv>i3XtN4OF+dH{YL< z0hhhBQg%Ra$}*4n3KykhmYJJ(y1me1q6bs_P5Hi<$dxN0kjva)1eaTFT*Ed))8j|9 z%%F%c1gb2Yuu^)66Lt;%Xmxe?$9X=P0ZOVQ^vVVLi~764HQ!F(%@ZRgva)9$Aed}0 zs%lQ($OhassW`|vv+MU2tHfS)KaZnps#|hz$S;Ey*sD}$ZHr}(3uTue7NHWNz}*7_ zL$whb+|#eJUc;MI9>vufy6oOOv}S^@OKqsa%rl`Cmc!9(6Xj6&x7{EA)>-ohvg3#j z#=pq!GAU=dqM9FL(P&q>n&{9V0X)c@#oY4k+(GBYr;%vpqg<aG#mqHs&y8^u?|Agq zZGd*$O&kUEX4rJou1^N;cHOo%p<hp_)(w2yDa~zxNGf`-&#^z`k*+>et#N@^IoH-W zZd}!Fl=U~t`s*p{zbPd9Gg}yn0CiX#;Tj;VDVwKYz#g1PUZ)u99Zw(>0*~@q!B)>A zUj}<LMo+@lS+NP52HX8w6wnt<=dfZCXQ>gL%)iF7s{gm-smkBGR(o$iB+Ie6%>%Y? z4~mkr#Cx5ed#|IzXDX$t1IHo51HgC1L?JmO&Z09D8G%!Nl2GZof>xOlQP4NcROof1 zB~KBC#a;8QT}8<hdb_9wX*}(gdf}z0l!!reylg9=I2e$RdQbnvSg4&We9{oD7;D72 z1lm$1tkJ3{{bRdTEcfG&&z}GA<Btu&-X}E_?ZSGWv|CRY4PAlI+S08v>1JCo7W~}F zS|$Yvmee62yV+|%TeH8TxSs(THY1We6S#=WNj)p_A+QB<SNn8p^Phj(&7RJuQveuU zbQWL<8eem|ZCHYvU2isnp-xO2%v00EgtHOmZ}Q(u@}^Yb-jc$-u&|ua+L7J_j%v#^ z^gwC$un@y)(GI_0G|LJOOhup54+9U`Kt_Q+iWuVy2DUBkjg%C`Bq%1wn)zt7feiKt zb1-x)LYoc;*)BVdI@ifHlX;Ls1-Nq(Fy}dJXnl(Y-7CKaa<dGggV1Lfa-B(!lmZTX zi@_=PP3_PK$sDKwKOoQBQ`&ieTOZk8Q|NS`wtPIWE8;#oT<eDTcaE|H^y+*%t?K#@ z#k4x;IJCBt7*DUmO*|`6DDgpjq=&pAN<?&;g@8#H6Ec7&DL^$_4a(g$4qczIa;BWc zBhHu)nBBt&Jb_*I-GMBtt|AojgY>2+B$*c00{c!2w|4%KpYyy?ooZw;>5SxQLUZFs z-aHcCU^r<D#knVEE;JsMXoMRR3CC*dDQBvC7O_cGDFaH@rvnf}qTwUrj<7#6BO$&g z%$-e#Jr2wpakTgnJC)-77NrplQaI7eB4G9Ks2|rArw)yPddl1FGGYwk=c0NEGwHX8 zPmLkcCA~B#8+yqy3Wz3unO<@dA~`PO5Q{s{w*T6+O1%zgA&!d0Gu9yN_zi!QwY9** z&Jrjt`e2C>fDIworfzF->oT0oL^G*#*K4N!wN7DER%kQ@-wn3Mu+jWQ53Pi1{2ImS z0$;3gVoM%ntajTJavco{<>#gl^;rsWF`xaZ*nPzxQB61YF46(5Joav|;c=62%Xwps zME#1Nz9=v(q)8@;X?0no6_l=V=sk<da?)cT@o@#YW1S6<Khn0MX4*)(FEs;guZ3!P z3h#)bABHKQNUz9ZaX{Vf?~AGU?jaB-rx7$?Z#8&j^3-gmg)3D*pV3?cJn<uLz*nvT zU#SKpJT!Tl6AjSf4{VP%(cbAOAIN!^Z+2*;@$`-pN^uY>rNhn#93ha@8VG5qnuhdf zgu)7j@L-A7<e%Yc5!%$SV28D7mQA!(zBA$UE|jE=1CO}jqlO$*6W%o4ZhrzVL2|)e z4`gr~|6O+6HwETV<|B7Ds*d5XP1=nENf3R|&2}(Q1Wn6ClnVzK(=tl$j_DT;ve_A` zRSAa)9Wj%`wcG3Qv+Hs(a_l@De#rt&!;VcjQ1StYG<f4m+c|p$Nm4|XPv@BOnuLTu zRMlAZ@}3$^PLN0jc~nU)#cX$7eVoJji>)s^*>CQBd4H>;Vq2MMUPsbcKOI#P=*sOL zVqI&`p7L9rh7AN|B=(sHg+9nOZsVPc|HhDDE4#h1mbgoo+%d~i%W&k~FsqC~Q`ROt zEX?h<d2P#i`fAY(2X!t=6r-1DV;$L_qA0p9SS<|#4`|M5-|YhTmFXORzM+a#kb<W+ z`yBCtvD^|8hd{PqS8Fid%Tw?K1C+G;dOZATUW+;kS)E~ru+}uMF=Q9FBqs#L&|Dnw zZ~6siqgPT6*(f=Li=wF$_aHn(GcWhhqswmyMGQB9%Fh=tU}HX_=t*u@_DmzSlm}z< zDKPs_vvHA6YWiLbKkcYA7+z{uuOO#^ADYlk_583d{y68T7sY9<ZUcEg>&xGc{{m%9 z{;MtyJe~jb2i?WThW4UjlD5e2NGR}CZ&pj^Rzbo(#6U9!oElq{l?m$i=Kgk2xgmP9 zf4Lozb~U?FG^P5aDT@OmBQg$&D7|VHGuOE_#W52YfgM;yM9TT{XF5Na8R+9bm!Y&N zDpWIHnY;8BNXO~?>NT$eIddGI0u)B)Dv!Y&O%6B(<K@n?t;)|8j`feyEDxp4WvQA4 zLOT6Avw2e++nDGhYM&C_N>odbF>-P?uV(oOUwqB*x}G~4m!@3c;7pe8ie!%R2azuc z7Zvb_De?lM+L&VD{Tq7CXTJ5!F*^NPba}6iUKZH6Hijy0H#QNkE!|S|6z>A@gqsPN z+O`E7$=cRX4JgmuK7wt%^B_cbJJX{J2)#j7Gl-#g5@l6?5@mfRo?c`=bHp2!l`cGZ zACP=&i*8A_!!p*kg%W{T6Bq}kjCcJ@0rjHgx@Ye!f=Hp>f?@zM2U92k5XFyHQuWYy zspBwO>r8F1=h-4USZ86N)hDV`ef=ZY*KqFCur=;|0Zgt_RH*1F{YaX1GHWc)lB64j zj8Ma<FvGa_IaIgD&VH?m9cEjxVt@J=$q_9lg^FZI8gN+<9L7+>NMiUtw}lP^mvlZK zq6~9fRTzt_njfEZvu`=~7o)$B;D1~Wj%O5DusWKt9a@oV-6!qP$0OOk=~TwQH-VMh zy}#LYmP=lvqWCK{oFk`7r%IPDP(F2p4eY(tpxPPH+qwl!g_yUC!L}2l6Ne9-b(C9e z$*i9XyfL|*pQLhx?&0Vm?d)gIWD1QW!vPk!i!+-<bfHh@ZR5~sc_Z7K5Tko?L9J<v zsAwv-V`3hNOh;dVy44tCE4=ZIaL~8{EeuW?Udog{nU|wsIk7Hn>Vua!7>b(HZnygW zTl^?H8PR&NtxOC_>)5nkX~@!$!9w*kYRx73>8s>KsBru&2@z*L;aQ0Xg^nXjMkMV7 zW<C@%4;|QGDawvFD&L77ilen2`z3_2G*{ZTmgG-Id0ne%2jcNsvnJM0?Qy6YxJ)mt zrzZub0b)xR&JI{j>zd4ZJt{AcP&##bNiXlPNswXBH*~T|nIafyKAE~Q?xMvRJ*c<# zf<i{*Ry(29q~S;?8?~X;LWYt~J8js(*D@5B94tx#N22Mf86-A2F(0a<?<|nfz?I^p zcNT(CI4<5KFkwYW*{az(*2f=yU@tRES~QIut%bTO8xu*_#x};MvcBV(BkCXjicDWn zMM4AcwdsFM3nTBnp$Frd#5{718J-=*mL?C=5i}w;=}L1y>bqrMa-$KG8{Vt*<C`D3 z(+b;&IJwKdh#2*y?G9+O(^M*K4lBCQ+fWz<-h2mhhSty1{EYP?+ic!G9~aYdzyL`+ zSO_5V@eBn3o<3qatzPHu-MgK8_wKb48BBPG_Xm$=NC#s|^bNY*off2OYqx%;mR`2V zF4G3qdqltbSt~Z|@brFde|R-8o(hfTw-}$IITo7tL|OtL!{Up~)TF2HG^1&B%a=0D zCU#u3bhH7PQKp%gnBxafg6`%&bqiap3(4z1FY-^j5#lFvz5+j;7Vpby9tDr8f|Z{Q zIoF%bOG*}b0n!6H*W!3|jr?`c5i1bRWQ)4`4pCrno@0&7mkum=7OGVdZSR1<Cd!fd z(@8!qKL51CU%oJ3e*0-AMvWZU+FXsKS_`|<w5U%jva*%&#KH%gn3IAMQ`|e?Sl!rj ze%#}C-E6nQ47~4(;xs}W1c;{wkf9Z($3w$W7~;bcj$f(NyD^(ulxT@VI$Amc89pk% z*a#{~@NLOq{w6;ov~A)2rkb@Vjq+t;jVz=5CN`I19t5AR&(|uXTTcAM*2jVDee&k} zDKsegP*vlOtRddvW@-_tgOsc|XMuz?LFGKvGa9*8VuKA!V5*N|UK_feII{BAWVtK$ zP?Feu6yZdGKTIk2A^=Z|jVEH1j=2RdvnQeH7gF$~Vrr_RZN8{y<gi68{vMsWC`Dnk z)#C$mSw?*wtD7>bE%CM)Z#TYW5wJYqxDuL;afW&xzOv>^{<Q?3PN``?S~WA)EIfD` zCJ>!RtYsl#nq0oLVzb0Fqb{SJ7F3j;YoQS?CgeVaku5(ko8j!6mbBR3D}P^=q^cif z2PVEjd(Gq^<J&RQH{T!9w9vM}q+(yP;^ThnUPuGB(-oykM^?T*!8kFnK{iw)9LZlz z0%$Om13gVO2U|74_VI$4_E5?*0h7V|sGE>{Or0|lL#?Do+w+InJ>Qi}DzR}<6zm4V z`jdc_wTVv2+5!H2r6>%3-g!V6qkCevfuzSCm`>{PUP<}bVtVDY$YpSi%tGXYaI<mS zN&r8fOe+liINU8!MfHZ$Ecla6v{amPvQt7FVySguMraEpYUm^?Xtx-QS;uDRvJ99X z(I56yZMzzgA-AVowR#%Tu<`g|0`oVtpv5!!(vcS)<K#zg@?nX0qNcaN(}{jw?*Ps8 zzT8wad?@VvNkw9W;O>I#6u1nq7lcU+s)~8JNFC(2CjGMp^^)_^x;=vVw@6hLx802p zlYUjVz0Lu{zT;rLI;AWOcUUB>*)?)O)d@Vys!p_aJDjAU5F{#p^rqU@19sl=#_yt| zMy#Xz5Z%D2N>-EwFnnvQ{R~;F*f-(of8n&Fe&~lk?aG<|X+B|Rn9=A@t$aSKT7MD< zL*1c-fciVi(qpzFtIv(Eg;rE32SInOLExi|o0Rpn6R9^+yKn?P7z8ajz&tsd?5;E@ zXQK<01Tm^DxuKy$7uqly^K@qQ=+HLLr$chsf(<~-l)B!0yrR@kmvXOeC74()Gr^z2 z!DFd&7pi_d8gcM(U$$BzSvr>XSAVr&CQ^GV+u>|(-#8!MI3M0PA707%a04qxd(PX) zSa5NDTZ~Ro1B~}bt?q_<#3u{yg!GjBX<n4)r2BS8LyKa@@ln>j54g9{*rL*#Evc*v zhxl^@XTCO@U?DeM!`CnKmj<~P;L7upr#8f4;)vxKKX?8Wh;yvvZ{K1J6i<xvx{)D= zuQBn}3y%_zee;YhJ6r*h%dru$p&#A?I0u1-cgiCqf>3@Qw>)iO1bHiEbKYQHZCfj< z0gfAwMdK?QwHr8-mK8$|u1Y@nc}PC_`Sr<%jO{)K`Oq~`6Zt%crMn98Ae<u|vTIzK zbe!`@!#m|>(?~i7a5?FCehCeR@eQPtpD!UD7a(GBNH8U_1oP@SCM*mKwq$P2wPqtT zz?0jmQ_YDX``IliU{p-aUEOt2UW^`{cSW?x6Ej{Cpix0qptD&%wc6ii`v4ZmfytkO zR5StM(4~RSYR(gFj1+=_rPRi|7raaKL7J#eSBiM22JWeHXeVX1^=pq8F|XV1VzfBq zzW}L=u}Ai9np#||JGQDgs#i=K4^)4*7eGI!*m~ycLP~g3)O6hjph3f@MXKqcFOF&* z3KF*Z5~Zzth-24Xp<fMMP%dqkH*vjhr1cwV{i@RX?#1}98fEMxb>ZdUnU-uySD)m@ z?yH<k3NPiUff378IIaV1nCMi1T5>Z_`1+bp3_u#%RQL9S3;#DM<@`sieCwOwdYAI* z5KX#X=Lqe1(rJ$RC8{>shX>>&im$qZ^FX*7T)1}^!Mih2^YLr`wUKQFP$xzu$6)f) zhdU)3b(qYzReNyW!CKd);93Yaf{ld0vOAgBuLJm<<>2e9fKy0096Ws+5@cq2R_1U* z8!jRW1G0!Fh75B<62A^fG(%m3V6H_cuhFiunMPihlZEtgijmlwXam?5^y9)LuOG~7 zWwa2`&9%tpnuO!%CL|k2H4)K71oJs&VDXK7L?1s!AG*WBOAbk;0t(ZPFJ~Q((8Aj! zye&Kv>hr36274b4@P+K~D059_*=bcySdtr*(*ZDk^xe=wAj!@VyNg~HQ^4Fl0a}`d z_0l>bss>`5!vDi?V9dV4R}*+`0cba}(EiD^^ZYvPJhyj|Ro$ivF>K@18w<Km%qM?i zK(QqZx*Y-iKE@Z}8k}<ICmf&6?)`9F@a>j;{e@RtvjtqCL$jKP7i%$KJ{<(c$OH~2 zid2@z*E{GnhfQvHdpp~Lng~K2z(dw>SbQjp>C^d?8W~iNEyZ@zaNg#=+8$tzNE|`) zJZd^p3lC4?-F2I^2rZ2;H|xsH^=D^6>sH~VkEhe>Y!NqQfQX-Z95qK6mmMVLs=qSh z9L&y}_-1lTT)w#m5ZB?p!~qQBy^A!i0HDQowE>;11+QeUWU59wO6{r|P{W|P1M>Ms zr0c6A4PUK7QZMpTJdq&@@*$V9!pO}WU8czeUiT~3fRa<1wx2rY9Kah`(Zfqcj5VDo zo+3J}!^Z~q;RG@%I_`3J0OUj0|Mn3c%iqM$Ys}nM1P__D_p_(;*FF~6vTp=Ymy#8g z_OHqLiyzRM==1Wi56y9<Z%;Qyv*3D1OYBAdu1NLK(LByhCxb0GDKhv@MRrkk&oMoa zreUGo9&MRG5P`HPrEu3EAuukvl_+@`Ov9?JmfF%y#CoEqX6eQavkU(uZj>X0e-tfq zucq4F8QX#$u=k=47>ZrB^=#Yop@^`@GkuL7BIf2!s<)TX>be3szn;&Gyd}X1SnNVp z!{ZS)fzDb)wMn8DF|mo}lT#5BxMc`eqoiNfwrf%Ssei4wGQBr8@>gi1FwKADt#n-c zAM2#Yy9)UuzQ#Nd1#K(%UV<Q~O;v|Kzz3adqC=9$sP-ljj2w-#ums2RUbD5c4)E#d z!x9maQ7@7q8R9L2x0I~-@q|n3#)~$DuPK;e<=@56XPD;a0-Z6YWV9G?*(r@~5>H$> z5E!Y;=01XcC;;-OCjj)}rD=Bq!q!Qs*PyppMR%sEHZI-hjuOYjg*MYFA+t=RKl*ZF z?DRIf_f$LLtIcUqM=NBBl->ar{$!4!wc>3`bt>v-I35m`Zg?lvB+)%eEeF(kA?oM{ zDKd7bbX;3g(JfWJ$(hVKHxFhpfY+@S!nC$ys9(^`z*i*_C3Vq+AF*&!qtLnNXr|+< zc{;k8U2}3&pU~w@gImI2YKSl?%7}?h9W_86m{#wBobitHX*H_wq(`<uk_!~0Kam~( zu$bW$#)KTvDCY4AUZ;-8Koia%)w?2tm045T3TGflZNi4`3jX#0|Hbyf`Iw$qO0ZT8 zGfRr)I6n?(QAzl?K<z@uA0a1jv$v2RC@@_2?q|ds@bnNb4b_~k3<WSR>|1)pJAXdP zt?^@7KS$5NC)0U-ay{fBAWh5zHOOd-(E82M(NSwblY*Mf;V1j^|M0K>{(t&E|NKAy z`#=A?|Ly<(fB)_u{;z-i_kZ_)|M`FY-~ar#fBO&r;qT!4-~Y>h`49j8|M^e<_J92U z{@s80kN@i5&~^JRxAoDIMpEdxxaHKra0MiWA}sEYP=hfE4g0X1&Mw+ychMBF;VI|f z$faT!Bo4|(YRqHGx+HJle123O&!;)>bg~9TnBlc=Ry{TBlT?TYM{M3}W6*G(e0q_5 zdX{{8lB_mvkjwRo&W4ZiUel&{Jv%AraugT?zT}|B*lj#OoNb)=ZsCa!UYL?j12w<2 zzB6NCkP1jaW1_YwJE|li3Nvxd)E;-3`YYR&*p*TfDNH)qvkMTo%E1mkv|*;%-HMi? z-8QhwE;-)4I%>Dbfw$#MkFgXxGEu3uw4CK5!`N@LY5Q{wyWX8u-&JR@MDx06qmh3M zcRz5raeNKAs#Pib`?zZ{b-TZ@VH%xiBH|H;BpQ!)YSD3%@WQ|SWf-y|awu>!6oR^m zLTh+Fd7?dhYk)W9%A_pi>($gpq^6a-NSo%89)1xv`{&F5LS1g^3D$3q1Yx}DSmuv= z+E?Eb-uQ49I8Qu=Gr>jTv78Og60gh|;lzKXm~k8!+x0Zq8*xYDpW6voE5STSh#iXx zwC!HOASQ{EUS&5<1~+C)H)c!MaWde2{2RQAz{H&|t6}kccwAU#1F}2a;8Vv&1277$ zkx(|C&q~br3H2i3eTTQqHQkcTr-EJw@zoN};vihA)F+EVX`#vV23C6s<5JJk0t~)S zMj;lY4hlQWsegjl!@Jdd3cJ!;N4LyI12R#Ic4ztE-Df?%>%a=i`k5IVdRGkliD$yI ziwRQrGlQoeJ`gZ$4VSZc`l*M`(9@eV6UEDknqewk7cI#H<BP681rs3i#Wtr6@*_pJ za#%Y8-RS-AnR%1ag6E;EcP8}=d6O7?`wsjX1#Zslr6!Q-6f#D?nYM-QB>ifd<2B;J zCJxkQarhw_UHC>XLRH5PN#*vvhVYmXcAeyKJ2*12Uy$V7VI2acLv{A(bU+~3oyll0 z9~K6To?)Z_<EK3ok3x{^9%e!)d}w)iV1acgwMB54k&Hbs`k9<>2%afV!R6;G!3FsZ z5JbQQ##ml+8uh*)ptR=Z$C`jTGwn7%@*-8?h&~Regh#ARk3RdXXGxLC%RFwTF3UMY z+w&l6%U$vj>}?+?o*zG0L+SFLIWE;#91EuMpOIT8#!%*qK<_8MA3BzzxKzjDB!cT} zs>XK<I^AD1B7_&Eh+<Y)W&}mzXRZYvT<}D-lcd4EcP#ZZ^v_@K?tJ&^<!5&>LPN3s z<+kTjk6GPNE`0vSd5$@4^G<dcZUxrEJIBl_n10vd+zx&8B^4@QA~UL*kHsEWJImZ5 zm6W&en}Kz_ZPv>sgGS+jKjcJSyoVmgdtjs{SVmUfxf3buPPbg;rPjHgVStFUk`7X6 zc+lo_W}KJobc=s%(XY^sg8H)+HW1@&p>YM&Ho7>jCaV9N!2sCqhm#|1*rZ(PoW>;t z=rI6hlh=J++?`E>H|$Ormf?)FpEu6UH|LI<bI0{EtNt~MikTmG>bR&j-W119T8v{< zu;B0~hPtGC+gM<|gnh0WQ~2F_p*TA@4YzYf+(x_|g{1?aIt7FFmt$KmQP;Z1=$rLw zGBTEn#^;Na+lU?|QTVgH0H-vU{yba}wQF<sKy|?1PzG_pABjmLIA)&q%rlRp)R?}y zh;CBP&6TP(+Yj@;r!HOr%t56R#QhvrJz$<aK&BlOXdXf{<Ear70~-+CPYp+BDsU;u zwS^kSI}~HTzL1MEKA{9kGZ=3hhCI=-mBXn8tqrDSp>wRB&L35rDnkO%wZ7Wk0&u(g z8Qd=(|ILptzkl)M`5UN#BS3ZW*$<Du`~G=w$2SHch)%tGu2e^1*bm|JbAeyi1=bgo zD9PG;FM%of+*=^Ql2M?k$XO1sQJSL3-pFuu+PqwzsG}#r;RC5lU8_og_$E{64|bNh zz9p4e3fl0jD(Xo~1?W7f!Wi8qF~YPYy{z9J^CGASH8-zGOlunmN>j2$k?gk;*@4<j z1f5L%jp5*6$CKa5)ZeX!J3>lMOGg%;uFJ$FgERwBa;{$aPN{dfDtXS|>G@{U0x)jn zZe?YOm`x3pcU;{aRrR_DVOz28pdCINIcjlmhOeTn4tLm?2Q@h2%lXB#Ye59p-ZrkV z@@R*mJwjT+RWsM8Z=qPP=0%^10+Jb3)#*0{#uV_Db|-_i%seNxzG2z(`zx*WL>Sj! zaj{=#8QXwaC0m*Mh3uPznUFi%y7j-x49Rf4tY*fvbxOM@B4uYI%=yS~kX0pON!yuw z$4vPh%ud_&;*sl-Okmc>;C$J;#LT6;WD)GWYv@8aHHNMdhE9p8YItN~UQZWIb!?R8 z=>U}BW<II$koUGq*lucbhhMaubG|iw;7ff<xC_@bfkb0!3_b<j?JO#sGR?V_=m37K zk*H}8qfKLDpR?(4k>ug2@d}2(j<gvCXA)`Nw7G(7^1_cVh5S?HOm?JBd8AHZq)Y{N zoK73~03PNq+?1Jed#936$^Nht9tE6lXw3<?=43@ib<)tlRPP+CI@w>^!$dE2dFovB zps}}!9yjr+=3|I#@uwGwB?83z{pk?zq(ZM#|H$@hA4Hwa^r)H?yC?Z*<Xi50hU56* zqrF5JLi0B{>)UgG$O(Y&@F`Mwkf(tNI73HwUn8<etVo(1atEgNh)jaO*nq_sVpGZ) zoG?N(C-bnwXOh$~E@hp!eWMP&QHS2BL$9n3eU0JOsOYd5RKo)IyzHl%^`Hxsr4*J8 zWY7P$Rw`4=oETRd@pwkux9~~#`?p_z`FsKp(1X<d5(34xjRlq{Ob%g-XjjEUqs-MB zERn6kN7LKfy638Ll^qp18jIAXCi8ncy6!sSZSe@UMFV8F#G}N8mjFR21HO0EpeVX< zK!y{CKF}@(o03;9y6SOIfV<DBY`r4hi*2oUuw5N+R|n8lRe|m8ARz$wZ6xb!TX1i% z@6}^w&^F1%1%AC)9@HqI4yHaS(~#566>3Pc=8Xs7T?n=tC!xd75>d?_DH1(?F(1v! zh23`3vL*Af<xPy6m#qbo8A*0rjW;`cgRRg5CLTHjT5-JbBlR?lq#I2)=vcv0@P@(1 z&$XmSqtf5VzRhcpLkh#sM2QvgFkGhUIP7z=%LHRIL#2q&?|Mgf%@3<z)aV4Iw9D*% zRlLU_C;s?th9JL>KzjQzM^gvB1T|6OhOK1y90iU0#Z#!~@vL29V&MDt0zP!n3=iUQ zFDe7G@rUyG5BV_<A&k0vM^RMIvxg6{+<%|7*Wt%s!FN6TSYP`>!(2o#<YYFQ<xoX6 z2&;fz(<pm5<>Bu=K4vCEyLp?@zX+@Q<2;Ac><2}^Jj317d41B}JBQPOA_WX8;XT-I zG8AAe+hM%@9K^Re8>dq5q;~ZvApCLSopBezbHi)Ch7#YC#_J~1dBL}Drq%pZ$Dhy* zAjPOV??}}huO4+XDl&(FMMm4>{Im_?EN7G=5WA^sr(Agk{SE!&U(IKybIO?lsK}Cj zO0w=r{=RTKwjWCXa6pg0*C@*@QoE1)ObAYE3dY^#ErLpI_U1_c(B2|kuC35J@!*t7 zoz&qsCKghBNF|~y5mEyAWPyIlyF-($auvBiX{u-;C3Zg(tV~d(GY>u<3@Sd9$zc*{ zbNAItE0pIZF=kx$>vCGVxL;0VolC-Hpu8C+dBfxkMg!7D{d2vRz-SQkJ%ML4wLe;Q ziU%t`EWIGa@MXh2S3KH_WfF0AR~2f?D`&+xoi*-=Uy?~xgOIYqY?{yVS&he2s$T%B z1{8rCM$&vPR}njcQTNxtG@+L9%e!T{r}-gybK-+m*d3+HX~NVUs8D>meN$_c;8;YG zFWAjTYiG!;^hQ5(qo29a&#bPW;j0LsuFvip7fk%s_9KBM@+U1#rD#OCpFc7FxSSZM zpY6Fv_&M^d?fa52*>~fe=4T7u+H7jZuHC-9>}1_;cfaGeA<XaN6>CUL|JWpj3K9&N zODNurM&c!5uB0R_Emz~oiOVw*J)-F_ZPj%<m_EV0v3cq=R3wyNVG`5%vOFiHnaPJm z>6dM4+W{xQG8wZvK(?cDMj_q!>th(s?lMxLgtz_7H}UY^UUv>T38PEy>ib0frDikT z3O%ns%Mkd9PbG!qWpv<gB9$ne_=@Qw?fE}O{CF^%gD80+&39O%vrzliD99=3l98%@ zxs0TGxPG?mV!zY0Y~>7_e0!-!(?6yz7`5ikMI%{1rJeNwxtzd0K|iHjiy^#Z?Cgt0 z)tEMN;fADR{=h~PGcr|vQ%Dv_Zcryxk((MejYJ$NmAD-zVxC%4%{8coNIG?|8n2eJ z{#wQv#Pw90;{5P+rt*^3PcELqp{Bwluft?Yw#KeX>+w%$Vw-BV^|}k5;!>uBdA=me zOZhRpNVaOtrpXD%W?^4na>E)rnre2WlR?)LqDdPx2d_3!KCr0(U5?G-6j-87tYLCu z8=~4>fv7t1{PpE;<BCo7L(A`t%X2Z`CQ}uJShOYyAh9G7^-3$!)w^^-y2f-8wfyqs zI7=l}4%)k|=)%jEOhFfONll;1olG*MNouV!%Ug73g--ZwHo?(|+odur-?C%2(gqM_ zVfJ)0mrJ;dH^NC&*WfK0X@|_$0tSQHo6%W*QPXjWN6aoaIF}41p4hOcPciimL*O`c zv!}*z8BbXjT@507J}s+G_B~)6qc#L{EFVwHp*w7yqcD@z7n1>NdMLpVoL$We!0#|h za`NoJ3S$RU=)j0+2WC1C3$~`NDFrv#DgrhnmagKbI031(V;N?o_Tv`3Sy`$`X58HN zXEdC*?15KJS}fR3QN?LHF6gP1=*dn>w`3Tn0qEMph*rET*_0Z36Ay8+^{eh7sahhE z{#AC8#oVwHE!95%NjJHle7bNGNgt4+&BKVr4H?@l9j};1U5!R!qeQBcUOGCn#M8;j z;dv9WrEh7yLv&NVamJfaLAn#RKLqqp<ttAVRmX4pHPAwOYeA;Uo=-Qj?-E;(F*If8 zWz=wBAB)KVLj_JS9_IT(l-x9Tr1rqZ62{YMb(kMY&b49&lt)S;%S>T6EPx^IR=AZ7 znABNUjtqcMP{aUaesp#ma#}kqmksBrgUIr%7+^)1+71r4glbsPjyMs;(_mEK#RRFe z{G89Bbb@+LeGGk4407C1-W3<-OpV3M8D33M&I)QYMT5?H{{fJC49m2G4Z*TG9-nRP zihxAp!<4#*Rqx}L+)h2m4LDv^p0G3El?_d`JxJd2QbU?7ddW&1A<zaEdyTt^Mj0#u z3;RA~ww9MQG<nx+a)(oP+G@C!tpA7CeFf9!KbOJcHVugR&Vn;S>aN`z{w{8$0pIt; z!Z@YoBYC4wzU^kDhTU0{?x?A2krL^tfmItpHFZmu7EQ1qYgYHXt&r~Vih5AiW~l)! zg9KXSl%S<R6*0!Pctfe&MMf#>Z>wqf0WWh$9Iv8IW}YlX4HM~p8oL+G1gX9FViH(J zYWAKwuJoFKJ#K%U9fByJVr4BF4{#eOH7O56wl3{rxr85L5cA%bo0O%2a<6Rb(}Ae3 zjP;r0=-Uwthgv~lLzO|sFELn5Gi{_+x}-#*Py)2#^+3f(DUhZ){-iC!vcqu!B$&^b zXO1fTL!1d1@`2f94{Vuol01?q8Ppzls(IY>0X}T|CoHe8?0Dus+S~{J)Jq+~Hbc>f zLgCDcQwl#}8%<`#g*OV_?D_b#1aS=3$a}XvX4F7K!*q>7Nt*6$s$-Mu-9vcc2NdMG zZStmGhrb_<s%okVZeYPg>_)pqyMeQVpvYrBDQCN<`NWV9&cWM?Qmi^@z?jf9L#zcT ze`lJGVMc5nw_QdgvlQ0ClHV>Wi3yjylCCU_(eHvlB=G24d8EYMIfqVt&4Nbm2^vJ= z399V6^5!UqkN;3r<BmEcBMZO_D%en`CB>n_Nwl){tM+7#oUFQP^W0W5^;I}ky?y=q z&I3Xi9lDI;k#_a;a_P7ckY1qcnlR{&k!7rRZ9XcGC&ZKA<BcrRH9hh$;A@hU@al{P zkpiL7zJS4tvx{QZ^_wI1{Gm6Nzw0*%v}+y19gixEu4MvTqQ0j(O+>epcD~4u%YnNy zkdGNL<y~<*>veUiH9L5Ibae*utHAVjAfJKsfU``DPt7uZggEVD#^2T>>kBV#7t^eU z{MN<v{USQSQB<}w_WAuCFauhAY7GoI9w{K@ci%c*wz;v1T>WMKvXk}r`_6<C7n?I) zS!15<;n^Dbe^50B!ZbucP6Xs60b<>tSQfEe+w?e^&D}9P1Cl&h3zqjl6n<LF>5fAs z9X_=?ozX1>I~;G`_3W`PB8k)*ya>v?RlAWZV4rG6gF2moIa^Z%V><@0VDS3->;$jd z$FSJ~!F&K#LIHiMGpBy5@B;a)tP4@)iGVY7sK!K=Kb0hdoShWxbc)EF4v&!X2?CEf z6tqZeB@CYR0c=gj*bW49+|Il5gAEH+e{#LgE1fmmc*lN#k)}G?04A@K0rXKPJ3erj z(44_E6C-c)sfQGkHCmG3IgRjIEu=zjv0x=bL@YjNq$b!}Qz#Gu+E5%QfZm$ga-*q_ z#A_gAf38yyc8iMet=myuLx$dKF6L@jcXCcx$mtyiCsie+a9AE6vkglpBoefL7~~@l zfptnoK95i}x^fY@AgQpv2MrX=uWGKDw+x8a%ZI@QsnoQih=XW~X$~iw4(4>jjAsJk zmrT`kzO`Cr58WXhaEIOOmF*->b#?+9)x;18uL11ZAt5@0&FeTXCw>ojxmpG>+@yo2 zV;CS7Erjod@Ftdlf4bE=$|Ngs>SVfso{MuJjRMdoe-W#kVKBS4s9mnG7;Tk!BCF5{ z#)>mqm#Z3?;9Tfur?^-Q8F0(e@05e}2G}3o1+cv=7}{`~vkDDct$zx?>Ot;0_vNux z&K=(RwjlrZg_=jf+jd~-2gi2;&_`gT!CTQ&@A8~c|8_G(rgP`0vt(cOly7NBlrH$5 zMi)X6D}?^K!aX$5>3Ld^)wI)tNHdLB8c2xoq+IcM{VN=YuWe7Ux37f(%n-mN!ZFdM z*0&|P#-8G75m_M}8%c~N$$AVUezGylFE%cUe7e!?b`xh)D^0b-fu%{sSpPlD0I;n~ z0uROkUwkrL`Q%NfDx)qU<lK|Pyk>K{55=_N*cf$PqU2*FBEBPY7g0`TA<@X}MtIf` zF^ON4ZWr3rdjoB4#i$raoL)u-JW_-T61$w%iUY6LPKiv7GQ8EvE_ac&7XBe$Nfd58 zDVQRsbOJ;VvmSoeM|F9=6`b_le9&}K2uK!)2Sqtj6>P`>@HYQVT<5gmpiUc0J{)4y z9hCiLoMDtea+s4I4vxBLR5;49zs2C4^-cj4Fgxs7fh{`*0);^ba?D3{mBG$}-iEA0 zM|OyE8povO!u7ml=@t*S_8O9h+n9w3dT)%|FeXG^wJ|_-x<(Oy*v%v8R9#sVflgj- z5?F~P>U*VFLP6p2XhiEcdmsYHu6h`&{L0B-G#?gl-ZQ0Ac2DLtvsGP90#$y#5S6@6 zqH+Wnwu=Z*(`2~H{fN@TAzo;<BcKFz8S{dEtnp*5iBtH6j8*Q~4g%etQBm0jZ+|@9 zi0N6RrMgUwo|y_+vW#|}mux(Uy;TOW7Z0M{F>T|1$VY)f(m8CItufs5u~4|{U*onA zt6q3V7*3+x5;m067lcs*m7U&-PG%d~@9>18<y$j)9d(PB9gw%=K{hV(Nu8C)lS()g zwukTdcIfm&mwLF{%xPO`*7YUDnx%#$D@r!2NGaF4Tu6u>;zZN17Lzy9aU><7hJ>&b zUK%S2pRf56hE;MsZLp@|pDm+*Sp(@i=*Gpc%7R#Q{q3&|^Q{4J`CYjz0R1zSLN7A5 z4Lvb$TwrcoU~XJsR&;@R`RdJ!$KUPz;rX*4pa0F<=P%zPyWE8H5wlEl`~3usCI_P& zn|zc#Hn+{$U#pP0pExgLvc%KA#lv#k$t0@<CDENVgj}YQHnmJQ+nuAJU1zOxyx6$F ze?H(pl)f0hO!HwmuUlP<Asw_?M+FA-7*wGgZ#TBUq)YKoCcU}|a^rTgB9l18Y-Vv~ zi$fV_e|<VF-#g`rhy2IFoY>_H2*fyZD#JjcriP%Pgt43eN!5jlwh*KFz9wTuGM;07 zDQPxOV2X0zdE_cJSCc15xvpY%XCyjX9;q3@k`YsPj<|I)K<2!}`6Gj<sX@ZVUn!=l zoija!!6hXg=TzqQ!ex&sLLu4~$<MXsV_6YnNQ6zy9#Yg?Qt;C6ujqEn2&7jZR~5K7 z9#;S5YNLvoko3qRj+DWITXHtKEP*cU$~VPK`puo`#;sc%XWVMY8e?|r=E;nUfAjLD z@s%{<r27Kg5kz8h(KDK3;P1I~L;G>RWKQD#ye`j+(Q8w4IP}jQ`wn^-6%B5(U1{W1 zYPyt`F_0ehpe2q=$#L5arnHe5>4e?*wwMfqp{Y|-O-LxW(3KavU8d%=AHSo3)qs0p zho4r+xMyrFgtnL1`EG{mh7Ng@CwB4<Pax%tS@ooeS1(+zCWV{LQVSBn_;gjuQf4wC zO?4R4wb-TEWF0*z02&h6fV&f1IU_i)TE5_)&h%D<8Y3#?d>duN5`h`~jW}4`=5th@ zGxhuqL_n;g7CtS4k<lVI#x1b+<&z?xzQ|8y<RS^qFnu$qE71-wm>I`i|2VXh++eRu zJcR6R748M}_@h(Le^q2bdPDNUGISw>V$dWA#1h7$O_cs^&sL+KQ)K*dtu9n#>{6F* zQ&j5OgyQiip%%gzQwu##Q488JHksC<Uj99N<<k!k#H2vQc|J8ZLFs{cRvsM{QzQVo zzdHCZ{eD)Cy6F61Zp^IGDYPQwCu;{Ci|xQGLqPR`HBhUw1E$Xdb2U-cnHU&26o;hq zm|lF7pY~CFKft4$Z@+9WCE_cJZZwU?d3~=s)#8P`6=!*T*|n~451f$_c7|^YTgDfw zwHBjs3DFS|&D=i9p9j<f)_NXM93T_*2k+)B8u`%en+X(@&zwaDY^^4Vn!7ba3Mo(d zS}ekZ_BqvjhU2c#ERh{~s8AyN*+V$s&9K|;cD(0GSw@$f@=Ag_#1fmL3Qg}YWFDEr zZ+5Oi`RnJ)?TC%o$OZ)E=FQ4$QkQ?tYl>TaYZ!C@i)!c9?YP>S><UTGX*@g=K-@*~ zTx<3rx`O+V%mC{Jl6Jhm-0z1&JFi7uA{$a<`$n4_xQzPmTRsBVH(t@2S92f8+iuw9 zeJ7}reA2p_y=E+2$J>HKvT!>H3E!uGS^%2p146${koJob#I^?dj@)R}?zO@OP1p3A z{S76pjtP4)#=1Q|;E#$IJT2Ok$EM|GNm1)U=4YW=;j$ps%g%e*g8-Y{h&l+bG`lR! z_}0Jc57Di*#`{oi=hdUHvfk!qd<o&V=JQxFA9cI-#Hy}0m%0&=G9#FAS?=z!Ui0oL zg*t3n`91;yO=3w#NNeUE_-W!G4(IY45epkbUIy}_;wB3onQjL|d8M`6v}rYmEpjlc zI#o99dY*auf{pKovfzrE`GS}2TLm+sA?te4J(i41Qva(%1I6ugMo}xNhf<P}WB$!e zd|AQR6>QIPR<KJvx}1^uu(;u~3`7->6|<G|EEe++#XarBahlhB60}8_qiJ}Pvz;-G z^9ieoNkeQrs1E}9)d@B`lkA)CI-YkGULhT_P0kc#Ej*j{eR23jlNi~dhyiz>H3Gn4 z>fobHujR+ro?VagH|qh}BLn31!+erZOE;?WP(b({mR*=p&~p-4a9CCfypXjkD?z`^ zs6cJDm-jqd*S|OSvpfDF`%%{8oGIRp_H}ul{buWntuH%bYV$IG$@FC-Kq$u~{7}`# z0Ikgp>BTK1KLF)o<(b`zkC$fy6wR(Jim1>*5$amZQUG{1%?Go1D(!)1r_N5Sm1{7w z+5mP@I?Cp1%;6b{i)0_ReC=@H4!k_=#?)l2W*aLcJ3E_LgszLVfaMnaQ+DjOtGnZz zt;oI`zzcMndWWWs+1+P}qm}6a+BnBPxz(}u$<;pcUZ)yeWS4FyjlQ*dD<+wc&soke zyIJ8giA-aJHVZ|LR%@)SV~DmV%36$Xg{H@Jmqb{IDlr$1KsfF5od*Kyzrotn8)JfG zE`LxLDE<a}y-v2($?kTtd;8st`h+*;_IOMVD^T=L>+9=1_#gjn$=|#3_a6Rb+oatp z@Ox1Ps%8h;aJG}yDSCER6cfV0et4vVniXU1Gr(Rp<FbZDMurw`U|(H{&<A_W8F1KX zO-O*-0`d|snjJ3dQiNfJ(W>gK!V+0sVC2^|?!aR-47CIeQg<@U3Ci9azy?RG<9bOr zf}R$-t1v3ne5U;>Vc89;DIB0rs|g2IXS3DNcc`1SIr4)FVuS;7p!bRc6aI#<p!@M! z*C*ik<mm#fG@dnmwVOTDe@C>dN}50kURq;fAF@AS3?={QjG5vMvczf)t?RDp7^i*? z%cG-RSWernZLSv&9t5t`s#mbkI4bu7ClIX59}B(!`dQ8xUb#!xD_XRHOAEOOPOCZ| z-#y#l^{lt=!uE*Im1?hj6`o8_$|G!OyBRBYHNq&j;v2SU-gYX+c^(z{>YX|nipah( z)jp6WI&McJ74Q9?MB^V-0$RY2nS7KZKxhetnia;_Yk*9*ncZcBc}F^R+qT2k;OQkq zwt|R1=^pD0Jv@p6jug5a$rg{9IwnIB;%OCY@4<358x?O)@<}^2*PTFo^5%|kfw-3} z8MY3i3+gV?V6@l#RX|3TgcV)%T+=*EoitT8t=#T>!gZyWVWI`;ihC5!v!11@vm@aE zLI<$0|EJaIs=aqeI|0tdK64nDxWhfK!~8`BnsNOGNiw`*c}!yaR9xLN5dBD4o7bKv zFUe~>0z-R;LUx9}mYs3#%;==ZSU{)*m}#EP?H!GXf5V}d=ML|LsU5&=a=pS2)r-S~ zLdC3{M}XoXK*QK%UuB1GL~d+QZ){L+Y*4SBChr-$Dvil4Y<7Zf-K>wHlSHeY4mLzF z)1o$kw2yuOGZ<mm7V~%FI^_&ob)p+s)PCNZZ39&d@KzQ#Vcbha{@}&D=N5l!oS&kt zDZa<QoaP3<mC&*CS4aF62DOCtp;2lvoDT{Os*oK*A$&EMPp4$iTGpg-<*&P>P37;b zf#vTe=%<H@I3ghvEg#iR2~F|KxR@Rn5(&hlt5}fVlm~;-U7*y-H`e^qF*sz3ltpHm zpPrNh$|1CP@VJq?n9zp?k$%3;XEc=P#-YFOjDfJcK2j$P2>Z#KU#!#J83dzJ7qf4X zGIwVeY$IHYB33_m3+)JhdsxK77hkhWI;H@rd8W&m{efCQlW^55k9Wwg>1z4O#T%JM ze1_l669tYbJ!&8oJ5GR@zCT5~0}qt0FT5x5s72j*T22QeA7-~e`&sYN8=^h+;B}nm zg`T0ciC;48(03uklj1{JOrOrD@BQbGrvtw;6PQ=r-SW{h=cVd*1~vfPJ(&=KO7rPA zXcS&d{Pw(oXZa}kN=M&WeYw~bU{{PhNiCThq2NX+xDg6g6bg1pna0;Qyj`@d<-g>g zVtrfJupxdoEKbpndNP1l{cNv=*FeZhPpeTrrG!mZmAwNz9N0l<A;vL!VKfGQn05T% z>FUQHpFRKK#~+*1u<l;eHh!_n!Um6L#W=XirQ@0@gbuaRP-}~CpASIrft@d(Yz@=7 zN@pk2>Wq?EJ;zg3n;0Q27h}s1wgn_8R8L|k8o^C-#4T8mSJle^uWnW`tn1g)YD}U` z+ZYS&R`V$uJ$A$x<!4yEl-09%#sW(*?UOIbWqb!FA`OS6_Z%2MjEeVo=cXfceCT`0 z;KT0r{Xx|6K#F4^8WI>_(B5%#I$E)Eg6|$rN@j-^z`RCf?JOU>>jyEB;5mgP-$5DC zZ!EmJ2wr(FX%?DZ?i`7I+6iBv9F1V=-}^R52#{(h$ecC0%y>6AGzxrMACT5*RhQi7 zAg0`<j}|rKBuw>#Bmbpe2@qurhMzOOqtr8qAG6r25iI1!+JR~2X|$-ths_E@IUPB$ zZuUbC2zkuvepKXuL%DC@AZ;L*I)Zi&5Be^kAFVDib4?T{WB!&YfZjFDuqeM9NMpc1 z&d(eA(U0<S$RMkq<U@r*EwutAZayN6%i(ZTsG{h|-xu(J#z64GO>2leiv0cYB5kTE ztR7btT`Nm*|M`mwcn=6ImeTBNiZBA*D8{F=i&!Y-_q!BP;a>DSc>zhUPpY$BK^;@Q z_`^eEs}U|1HWT&o72rMTtT?{#wM%Se+9&iB>P-Tq1W-ID0Jq~px^$vK>Ke)`?mky) zo&>)zz2-KA^Z2`yg0^+?3YknW-sB-HgLiE7s<6r1moTt(15NB=)H^>+06PotWiB=W zxDg#h@GN)EG&iC1)qLF2O&3g1WK!u396z~Xw9oUJVc4=g>YLCgMlv}@nWcQM##KG^ zSuZj3@A8Xc`t}3>PO8zcZv!+sHQtTPej~Hr$n4ip<TYf_Lr$@QGNSd@NIRR6(DR9x zB+sP3mheXOt4jfU{SEq)ApALqpCV+k)gik}*-GNzYR^gvflf?qoS!~3Ug&M3T@#bX zmY+Ms8DLl};T3i|PBuo`$wt+zW=D*=FxjKfH@(gz)ie-~$%9LGOUa978wvvx(7rF; z<Y#o<81}j2n&`(f!ap{_%_z7;#}2|K6s3z!r8<a)AWC<jFp|yx>Mbt!!*dC?im$3v zRxs5iH8ceJ6YP5JR@PEIzqzov;qy0q{u+Fq8GL5VW%19(NihP<8!VLC`xsj$h!&Fl zHJropIySwHab5n>&7~a5w!#V}SLp|iy|*FC+YQpLM|o7a6;DSWKI;jK0)!ng``QI+ zb-H0ffVo9{TR>8Y@J#}0otm&1OFOc&p-Rz}qXJf`n&5(BZ2Nof{18Z*zt=dOgI{S+ zjv@_6<aQ3rzW<2}1%pZ;d!`F|5xxtZ!gVGnHy#2^+cawOX2tl262p?TX)s$3%JO+> zwklsvI7mN6v%$W&{o;`VGd{cSZ9m()<KA~N^#ixYz}6ExNM5uK%67>35V==qg0q2! zl<HBB9_??3;bn)@@<;}+jlCYabU{RDFpX#{L0|^JAW>x1{gJhIjLDk<M#iuhqgBpO zG&3TO0A3A6Bf<|bZs=p=Kftnea;G=N7+6qtn$J#LeyPI(m{y@0KP4zQ+$wu=*07GM z>7uD80OcqLHL7)#Q-%ZG;kE?}`kJk8c0V`&_UJ#tZQ)gR7&!D3^>0W@1l%WTbiSx6 zPD^ENOz9y?ZkXAts(YIj2L*J^2d{QPPAq^PJGcdsas&n5>RkbYG{89DIb~nyV2(-& zg?Dz6yI`Z_r9m$37{sWUnM7Z;@gcB{tYJf^&Eq%sdY_vNEUOGwG*UhOe~bR3QR;Z} zj|JS6IP_g;|Fo^ux?6qpJ=0p;`H1xdfli0I*iSAm5$x3byeV)iFL=p_dr8z5e)&MJ zE8ArDh;V&Ii)#DBhXN_B?Xa#{j$k<S?yNd>8V--+TEqWY^<9p-^n6?hAJq%C1`OOj ztC9AWtt#zvguzKU8a|%2H`fS<E$x+v8vOOn-@JI<|AaGONurpy%1wu~N$ed|Oa}RB zZ8F!9?1pf(=Wrf+47FRLKsZD(>ktoe6vg&^F)1;>9Z{<Ut({BsVmn_bl#mRE0_U&* zIdN1BF+}l1)NGW#!T9dend*QC>VRu6NFE}$Bs&9SGeU3C5rW5=xD%X)2v0XL1$Nt# zMUwCo&~86&;G@x{N{NE=cNhd^J|$~6X$N_eaGbRr3dzbDs0x!uz$FpzC@Rrf339P= zLv17szq#K6>lhLHsI`EnYdf(Z$?C#-Kvu@Q4Q*5hpX7DPS&IEypTS`UtpJ)U9U6$M z)yeT-5;cZG5Z^bYU!|64G5dqp6fWP%J|v_Wp`n;k0tUr6&@6|(BmnT8?Ml#|XVd)n zn2tK0lx3d&f_4oRX8Sz`<BKN^1A<lyxA_*xR1sh_CEs&a2(v<`ASzF1AT)opV?h|y zLdoR&OCh@U0&E3X(m?NAp@6>oSQ-4)(Gkr1{-W{GzKuQ#7kc}e2t>=C#^PdnL`Ceb zv_e0bQWAr>aw~Dx674ez$Hb`aV6vsBDC4Zqz@vOv>`WfeWy*enPq@Qi`s|ZFUb5dM zY<}dk;74k7%U9%-C5I#qquydkv!(QzwX?u5SWc>%9ap!*y5vCSo#xaDPQl}icRBOq zCo*sdB`nYPhYLj-Nhe}1@8^GKVLj6ZuGW)rwPdR&LLK>B#Y>V~ugFXTP8m(Dfl*e7 zO2+Jjy49t!F}1j(r`VgPOJD5eTAx<qaW!c-j7XrrSUPnp4hn+u8bXS3bx_2LXq`pA zJu|$bBnKNA5<SQq;fhc9I>dt1hYrJB9)#r!+T4DwPV+%IyYTUlJ%PLl{ENcsB(kpO z@ujM3!iu`}AXc5RKlR9fo4fM7`YT-ha$E0)*>wjr``)gL1I+RhG2q7o^tX^1-D}zH zu(dDd!bZMJL0j1mZEK^FTnp_bK{?u`q8d_=x&~M3qg+~r0z{}8bL(U^IV0F3y!F?( zO*E{<D@t`&VEot7#Jnj`3qiE)IE145nb{3mC7Cn(DPbHO9D0aC+9Bg3Maj!*msh)Q zt`fRwg}hp?zG9ZdZ`Xe>uu`f|ayWrJ8O^7Wxr=+v^4Q-jh4}pf2ogL(Gqczqi*{PG zjux{{3#R7Qmroa$_hBQ+fV;F9g@)8OsyfHig5nM??80{|VhDgkG+NEF42y-;OT~$) zi!mrbZxg+x%YeAXi!vvNS2m7}HdrgzqLc2D>1RV%pWGg$q2Q#Z7d(eCkuAm{pq#+S z9!$W!;+*oy41KGqCj~q0xg*_sjHw4yz@^K{u!JWPZoDED<`E{!q&NVy*437%$kjI2 zg<47ORBNUmyxRAKXL*;*(OO6#Z`j%b^eNfEE)Ngu?n#bjVC|L$3ZlPxtD~{Q!B!Y` z?7Jmc8n)C3k6t3_uJ~Y{5bm#!ba94hE#cwOWEtCWkaqodNtP-8LH2awM6tyAN;1?h zJ$by5go2Iw@0Q8l4YEnw!_C?1Y@U0`drwb2{-DXEY0u=Uj5#k<wdXH*a*ummOfiS_ zXFU%kIZS((RAE4~MTL9&!F876G;uB{O$*4X!VU->bTptRW4wO;^4ZSIZyXolyQR$R z>9=m(il)K9w&m_(>0afO0Lf{!@<XY~%@L53>l<lDKrF%IG@D|Qd)N2eW71Ve2Kknv z^HF(3`pL%H381B=!Vg=o*o@j+8$bSNo~vdu_EaPBtwWRm2*%N&_pAM&V>$$7K5St$ z8)`#r4HVirS@fya`bZySL01>G6>+5o9(KEz`M9tN_L}KTLpl=EUX1N%@<SfiQJX|3 z<o9X?2NS>vpfGtXx@PFZWCHG)u@2~~%V6^nqdD|M?Aus)&)mZRysXjvj-i9=KK!70 zv&g^dCRR<W#|OlnxHvBcb6}-&bn?YCh*k*ESM*`Sco~GjbLFPQo5V+U5#ciug|dEw z;bH11?op8NO`;-~4K1a|iPh^T*54J1B!zMfTppj`rOK;`obMlGikyNS-BKp+%4Ka| zJ)e+^fU%BPgVBgE_swFkB<wtMlNY6SUP4rD^pM!)7-h4HzG%Y&nQ-InL|2d06L&Yp zR6yCXYuKvtX<1o&Fqd}I^?Z;^XgN%x&3HMc8RNyUN|IrIUQ>8O93J*vo#mqg6(dEm z>sYr|8%`OB>04`$zCz$4LdKov*tYK4EYh**!1hVv3N&q)wX!m~64>s9zj_KQt!a5h z;iS{*$3VWok4;Zw(1L8!>-IXfn(R(?Hx(>3DA)@M_NX8{qaFk&8I{6ZtNAdVAl__y zPsY`-Ji5Re>ih^TD-1E=b-6L70}Ajik6md!FP2t42>d)UL5jUAc=3;8DMt;j&TY*A zRHS%YX{b(iZ*vp<d@#;SIe0anB_-vEc+CB$7X!wxB|W*#G^Iv+ajPy<>NgYP;4&K= zrnO}fYCip@i<)_2xessyNvYFN-wG}96lsG)pV_adEwce_#$`Rr-xY>05Jx!T;BwV` zrgW}Wv!BxTV~ZVTv~DN=p&9H)l4pyl#iaeYg}$oDoxP5)rB%s<MT8v7jP-p&N~md; z5o&R^vl_9@%TQk>T!8f5nkWKF`!OwG*kH<R1FjB>hi<dy4a0=pXe-0Q4%=Yf`0(7= zcHh``|2nqa&!+j=C#%|bFE{Tt=)#+rv~jjEnteKxd-&K=8xgi0lz)4Ej98E#7H7;U zly**EMPo_JuIUE|Zi>%rN4DBnt#7Z5PXcpo+f**|66ms~_VPhNveZO}5IKNwjvZk0 zx61lu{*sg0P<*`bnWt>Qm$+qs1i!-xp=Iwdzv=X%3ZVCBVXW~`O^H6`6p;pr|CmGQ zABqyUMDx<oD1*}$s)K+@9r?o1iB<H9VI5v+z)2VDFs6QWj@hWuwVmE^IP<)Y5TQRP z!F3bebFIWj-V>V^&<vT3H3h=+P%Sr`xWxk0gsdh!VaBsms?px*3^@|gT6Bqch5JTN z1j=5KEd|Y6g9Npw=XQb2WJkvrmn<PKwo=jb$=PLWk~JO&5Kk!$ji&)K>(4F5tKUFY zqoK5kS1bam*SmTFhilr`=PES}rxo9IPl8NwK!9(FDq%O6RM-ZL1VHVelOoH0l#YlQ z7YTt-2W*QVK%(z7oD3{3;FSeT2pTBi#>7KMKQWIJ)&rZu`?8wXX4TX#+HuBK@<Vmb zyd{GYgegLI6yKztWHB3O1BVzxEw)W@SC|Z%RScx!Y)AQIm=DEUBevvg6~Ci6#imrC zwR5WL1_sE*n8uPpPaNBQJ`!&3rd@CuB!P)ah{~Xe73I{JR1xZE%^m%Qx2NZ$wyjSP zrb_aRI8*T$O2=E#voRM{NH8HCq{OC)Q6nVtYo^DwT8eLfq-nkN%jQ*6Y~v8&$5WCd z#x7{j^H|-CpVdg{BT!AyvheyU!Wub?tw_wVq1f|LWF}@lF2svY!>3#AnG_}Z<Uhb; z@eue^9wxI%Q+=Q^zVe##mWPmx3g^_kEQ)I^2q*)>K`}CFgm%K(JnpizDlvAA9TB~C zB$i4#Q`zgZrMKI!%lb8*>S_sUy};(5H4bSkOhdfV(^yRS?9>j7i*6WhM55N<K)udi zK~Fu#kn|_QRcy@JuDJP>scBqc!D_RbhJkhp-Ph9s9qz0;y&+k(R(UKFIHE~cw@OZK z3ikB{HuL`kBvCcsiIJY=^}E-Iwl<c#R~Jq#HYR@+iSn_82x}*T!rh{Kk#znscPBuK z{|ETM)B5|WKOs2v#|2p!Vvcv$%PyykrsrSj=a?=Up9XOkuC`8=N@r!`7H+vK5?UyF zYqP~@IV2Y9Y?@QV?RMjgTx)d)s|a_cm_Mg?BWik)iZ*c>U1BxRn;9{_)}NXCXF0(0 zqWG-Wjz*V>_-C`0`uHG710lR!=lHx#F)!vD-U2G<KF?lCc#MAGiLGfDK;KwJ`{ajV z6bdIWt*V4N#1J;zW*eGgXJ+wcky?qr%C?=jX<6;`70^m@Bk=AdmLH7cOob{kpT5R* zNywbRb)!987(Hue8BTLI)onzsacZNd2xB|2ql(J0ol#>sV?7u4L4|XSN`}U2I#{I2 z=a=p~o0muReYdor9kdvOvUW-|*BHkMVQt^Uw7I$WxViWEC9%tlSZadsURP2!j)W9w zRLsLD1}X<HmMFn4E-A<O{3N*0IOW3y9#kng%ML6R8#ma<x~d9q(>Tm)N_vX19HdB> za((e*FVz*{NB~n}bb>b6qEfpki?X%S5B7K4BJ|tmsb3AsXXO~mn_t~wIZm~Vq3q8T z7=y$?+%yi|eyp;ojA~p|1c0smw%<Oy8aQvE{1rdP@bmM!WpZ44Lowv|bC|2x4*$@! z#Y=@nudz`H1GTsp7TU-j{tQs?;$pZRJb6Re*wx6u+&+~wJ%E*{u;oK?eV)~-AkY5v zAfr)WkX?Rig1vD#x@_aNeDCew-ayG0CCHzE<HMPyWQlj=^3`UUE%+d4u`iNcsTn=h zjLoYwqqnFTP0|@wjG^tmuBMbKqEEQg&hHjiUwsb&gu?pXF9oE?xUYX`VTy;MpWV78 zU&BgV1u=4qIPuD^lYa0!uKg8oayvQwB~%xHNAi!h@LbN5G|%)n+NN3;g~=4>7*KNp zTqHZq_-XlbOy=NRu4)UDdxrK~5kidZycAt8K!*W`K$c6NSkN`60%3-a&pM_qHlR?+ z;x-dU{{?+f4bb&WK-BZ=vAuPPNWWUKvR-p#J4YzXV47glvyP<SENmM(@aqr7w89H$ zIFBC|%8M{u#;*Dl#R+S9HKBD)9L8?moh;nF!$4<*8+!FJe;H_0Et0M*f_U;3ziwY~ z358b5re?M(Ana?66}qN*L*tEIa`*HL`>(@YsW7*`HWI0NyyutQ4@V-X1bCJ1uSV3% z*0UJddoP8fNq=CsYJh!P@q8;>HtgEXf$Zi$c5@(G?LfwH`*QT9e^wqHWe?1twp+#| z3{%1lU@dG|C=xuO=pqnGTYXrN(Y_5m*fEjGh9xIoMlu^>;Jgto1ndVb1|Tgq@t9U{ zydw%bD$fB8F8x1L6+W8`tFx~3!EOi?<7_&6TkT9{J43$E#51E@+~Zebt70S$f4(VT zZQ&IMNu5KAS6Spkq*gvv!o#jEfGM#k@*uwjI!Ct!OeX*xsBsirb0`_c6vsmvDmWh3 z^?Y1#b?`4Xi=r9gUO1nWf1DQ<nr|d_CPSFH`bCbqVNy1Gug^t(l832y(RL`Jc+n~| zg05)Kme?!zktstp+a9zA;P-d#IEc%I&9T$5s5xSOv@_%zWLufv49=KPFqzmk`TfGJ zRrigs?T7W@N7qB0ic0ylUwXfcjJFJ_H&Q+t{uTrw1qveb1i42i3kA;r$)wEYGm+W0 z`JKS7XnK^o`gNF1zS=RIU|M%tw6Bv-TJs(IRc&b+Dtllm<6`uq*B4Wb=Gr4ug)8)L zeWm`zHUvgNV6kp1`&B&BTAjD>+%OnyIL#gV*cW$ef(V~k;HOsl1t|N&zW%|TaR@^F zR>5QqWuc7MNQSoSvZ2^z$hVWvL{-8$tIPwus2D0Q54+r@pfMV;1+^8)(5rJ5zOUhp zTT=FF!pT*#NA{g`)^25Oi=yND6~xP(Pg$?vh3@BpIG@FD(1DsCB5x!a7WWbqgfpk| zFqmb2n=42Hc=G&g9u=&~%}%&m_;Q+`1#fhXG<~mxzEI2uprrCNq(LScpY*i8PbT=K zQ0pSpnxeh+5};<}sVxZSxiNWyp?bVY6Uyg}M+4Ap_hU?X?Tib&qcG7}M|nn)IqM9@ zo<toCsYYT>Al78avVb=7dgh_~-AOs1JQXN}Bdq{)dtZs^D{yf>vdm7%pHk==b7Fzu zhf!P+>I7sPr4Ax<Svc@YPrJbPBIZan0s9T|cWDZRv(Rh3VttDTi#=f?<gF9UEt*mE zQaD)eV6Aw$>ouP1J9Lgg{9=_cEHSe4=pshqPT;mvWbgl{KV{+LFr~XUU}{9TUnT%r zqT!eEBgaj6H?sIr<sE}zV6HH>_%nz-Ucp7;ZDmi7ca#joRGUB*??xV~-hR4aj@?)^ zh6vu!tKvK$fmE9)MDliuc<(e3FN@9^bpmfVAL&_Fyw1~5?4`kZ5N$Wx39i}=wbtZG zfr}{NyL)Y%J;9kqxu1BRp2L_BQwteuGOOWAM4E11tN={(xf97k&#W>q-MduZR-<;a zUam*u<Qa4&%3l9Pl<l_q;@oH7UX#-N_L=Zl*qulD0M!>eyw_06;OINNnB?Pfz|0uV zv^q?j!oz}?`Zvv(Z9CB;C?&+(feDstdQ)l|YWyTf`Kn7>RoYHSHP>3&SH`()X{QO* zjNxiKMbidh#&PAPjGuTpC;I81W}NKT+|%ZMhKte=*@`=LlV<^bJmD}5F`K}78OV;t z-JjyK4GQI*hpWM1k0bk#xX=AD>p$#|C;T(0|A^({wC$xm!_et<${GjM_a4I9pH_9v z=}b|Xg?4E*skS(j{?L})#effo7e+|s#5YmP@RvHf!z>>r)3MoU+{^c~b^ZI$UdaWQ z@l?;##)BKbe;FO$QWzE<&>W&=$20=mjSg$6vrB5S(uWw{2RwNHAQKJ*tEdL*H>~go z-VF@HPt<RQPu1X!fU78Y+*Dtpbi`qrYUg3)LbZH%^(?R)5aGKYjgw;36N;r24ve2S z*RZvxF+Q&YiOc%Q%K(2^z$*e>>wBG+E^a%og=<`o{bv9G0RR8&y=!kA*_9yp3+g|F zY>zW3rBWoz<?c>eC95P=S<|H|p{TOkLl9Ds86?9>X5?ffMW(7Wm;q)n7+^mv_QO1O zXRr@oAHZU<ANBz(8rb<WZMWxN?0MaL&wWK^QgYip!{~M?8FBBquXE2m=iKwSxgNW@ z9{bPodhB^KS#qKp6Q$^@@+;%Q#~R%?)$#5#3P^WOVe9O>@`|j?uJ&x~0yJWII32^m z{i7Wd0*e1O^>!a2GTbMe*poiIsLr9~a=t{0UgIK4VRKB79%i=f|GDFi7<+=&EfEdO zKHYadG(z<xq=B#Um>sat9%e-=FZZic+~vZ~9WSoL{R4P1vc5rg#?vpW=>Rza{Fe)k zYYhF!$;(mkEI-RD^K%fSHXnF7v;8z@WOC+}jshOawF>KYuvFjlvUk<n3X7@fz%f$! zwTBkm`m8b19wzVB!&B;TIhfR6)?OzZ$nuXrkmbe$>Fn=U%@n&lM7p)4NM6vz`RN6n zD_Pj?mz}BrJ6t#kbbSft(JvcKX)v8D`xf*j3vAu7YBS3a=k2Iq<K7C!k+NM~x7m!h z%|=rZMj?B2Rvo{_Veq_*Li=30>oSG;82}Cr4-Vg%D(0bkAv`+e{iV{79{_qT5AlpR z|0rcFj!!U;v>geR;b7qjrH!(nh!V}5;S;nq9&1vJD)|NW>KI3kS+&;%!M1d!C$H49 zy>bC6)&+){QaOyp25sA{-}Yoas>tp^=4@SPupnrwjq|$fNemm55MLP_dSx=eX?u?D z4XSS1txRmBq2gyIb$3i)<f_Jz56_%pOqSTG9d?E0<lDul*zpZ8Fz_DUQ?UfHeRu#M z4&x$n*_$)e*kRGm8^vC6=T4n6i;zponURI4NsxT8oxRBRL8295B^|xUbZRRySisOc z*aDKvZT!pjpWdX(_X&LQDP~cjQWiYF^BYN-UdZYgT}jr+A~hn{+Z1Fq`&e)<Ss<Fu zZVQbhK6!Kwoj?VTsxb#sQgm)Wd^gc63TcD)*TA9kEnPFO-!|mTc~rfvD-QBS!WKsG z7g|@d<t*A)4LMh&i&;ZPS9G^1rIDfc_N?q?=)@H^xpd^pY($USqkZ^CWA*5^bzZ%B zYE7NYThVVQ8(6!kUbDXBHR?;Kd+<ButF0E^S3R)V&=uk@kBiu1as!X8q(dG3y8w>I zc}KOeh{ryq)QZ^j9Yo=I>Xr(yqn=)bTE=%Pdl`=xi5<;od>6d9f~_$}n{<;_@I`}( z1$&6z@bi%52S5)I8xq|}DNP+ulQeh7$DqJ-w_Jh|hl$zfCgjW5FGxSIk-y|a1UL(C z*P*yP8$^4D?HW-ja0G3odvm;)6AA%goEd6O<Dow={{Z(i*&`MU`yk^DxKc_%4PP1X zeFeC<V_NP8yaRyKKzA%q)9o&RfFVPsS&uH}h#eVqKXw)o1^?YS%-nil<6LMbU&tX` z{Qa7dfp8Fv&>3_NV0pX9(JBAZE!w@I*75aTz$m8=5MT!HdWQ(=k~lgbGHWfVe2#JQ z@ndzhz`y_mudNpstoB&%ywEKbQy=LG;HpDIc-5P8<L2DB=D9(Wt%=`<KU7eAZ}%&Z z%k(r^BPF-$QZaS>thFro;Qe<!9c%j}JVyM?WBLJ(aZ(j*=EEusg&%&u@)k<?m|<rT zLl<0c)c7QBNu$tZ_}EDJbfHn5;+`9Rl9~9;&i(Z}j0Zh({+WnuBxo=+(uca4rkDBo zU3sc*c$du9ZG<DnvW?={&}pzR#E%Rw(f~qA!u=;va4j^&$4+a0<<lS9i=+OV0oPaT zOJ?d&Sgxkep%h?6rYR{TsFNv_)045sc$zKjoPZ6K133_GWB`TIEtE%Xk{|2_;x<hT zgyy*uNCNxt>B7LM^LY2upLKAf$=;e;jDMRV5++JTt5-^(fetzVll3I&@&SEOJ-Ju( z)n9ki(DIn+UEavv9u$Dscjb_|UG#Uz_`tQAG%-p{KN>_Z5RU9!Ch3t}H1SDD|FN7= z(yo^8pyJ}r=sFF3hdA!Fp$Ema-L-!Hv?T+4(2-!ajhIEGw&e@1xLk+{+XfRXB6v9P z_fiJYKGvV1EkHdz{j9kPRI!gNw)!mhg6~5Qo=#6Ni82Vt=9e~&fnsz;dT`g3yBijC zWIWLrJ}LW{K#3X}#<&h>rC%_&q*5ji{GaiF=wdXnQsyJV3f%A$>n&h+9N3*MusZ^F zXFaeKFM#H?qqp>N)7~1e`QFmU)xGt@F2)u&=QYL(3sxM?Rz|83x1@Ibd3lv957;&0 zdeY{ea>T-$ecWJl6Hhr@Od3lY3wAlHKt!pI@6M!b&sy71<ouEDH{;Q$I5_l8a;Rsc zVmtBH_1C6i3yT;g!)7aX1vKY2Vzs#p)Adpk$f(^?PbLXioaYq=`@`++?Oq7xXnxVc z5Kii=N0WN?t9o&^PkEC2G;KEN!jQ^sxoQ{nWQ;{dI;0M20ciHpQ8Ur@jZ+rzpWa#G zwz>ZfHu{ff9sP|TR-GH~$Q)6lb|Y%D7Ye(r$#RhAoZ&2ac0u9kDgGnJswd+U6gY@` zrzi()QBrav3L0C@=ai7QKzuvZQG03WkkIX!v3`oQj(^|1n-=@|o+OeF1;E{+%ZVjr za!yFmNiL*|NP@YQH0|x`UHw5jrGAR!?D`ffQ|N4rOk4Eh>9Fd7K(hbF72q3V-S8Y$ zS>|_n<zri-rvB^{T`}tUHMfA9)e?vbnaX%<NFSirx|Ht}W4mrl_|S%#Baa+ai%X0I zV1h%DpQF(;!j|BlnizRD=IfGFQ)~Z^Npb(kq*9j&rqLD--oGD#^%@Ms;v@<Ob&igj ztLqTt<Ul|BECnnW?+uw(&wq|&-st#0Jf%luW;)Nm)}iv7()VCq))+*D|8UO11l+|Q zzwQ{&-JOBIwRmX3D>X)+0RI67O9%98IpO?IC@vpQnzlkB+E3gPxD)Wk0RF}R{>A`) z4Fh-=RtX65qWWT9kCCaN9d=yK7gY<ZJitg#HlT^0!$#|b(CHrS+D$!lV|)m3y~wzA z`wM6K6cl!$bcec_io1dkt|)}n@nU$Tz%I?prFwZ;k16XceV{+o$FqudmVD*Eymn9< z@GF53MGB|b!xU+l;P%L#epkQ)%kboChRZ<CvY3XC{0qwBs3rpw<`nSbQ8UMtLCaAn zHlv)#uzF4Nn)Wr2O-avwRouO=x^&Mrw#<Ft(ce)Y<%5ho_+q;bjO~avWD7g;6j1fz z2nhK3J>Ho2>gn;tQ3=%Wu8G?Wcn9pyy`7zpfBF}uvvO^50wbCCt#;Wg2z{F)`R&uI zAl8atAjgU*Fqgf~o*auO)a0nR3eKbDs8|NCsI?LQ6MW)X8Ogu#;o^rQ)jPstKTPL` z)Xf7A-5(S^#umI5=XQFB0V|H0QO-AIAVvA6(i5;^cwIwO!=#_dG*zJ<*>o>bCo?Gl z0koW3S~A1G)}<qDg;`ILa*#9vH6u!5Ge!e~mB@vo{dh4KkwDmQioC;_G8^_=ug0h3 zw_ajoZjN(0?7+Nz>d0!`I|k&Vwtuw4BOM+C%d``EdLJ*NV#`Fa`L-+OreZkXPL)j; zO_U_7jfc}y&U;{sy4OMZYC4Wfo0s9hCe88d%ewUo+qWNRFWJ_K9w$cdp>|Q{&^O;T zv`%a78F%zr-}lu!hQu#ZA>|38vR#ESkZr(KR9NsOHz8&(1t+zy*Xii`ZChqwcihnf zpGn!c1OBYsgp<7y7jDFbjr?Ufj!V0!W*q5EoMwq|q+C&3jNK_pp&g@e+4bm`d^C<d z=;a~BOqhz88u+_9e?`3089ZU;G@d(&@Nc{J>#qNmdW{AO+}U0H1rh++DwtP^d(<e+ zYVxYoq?63u(o@XB?1W;+&+Dmm4z};gE9Ee(O712~?)oJ$*D^1rYZu!X5L829qf)z6 zYL`mwdf^$rs1~rz;kCp)$EpRUA%m?ypCVa>&41Z|e8fX;hu#d400Fg?Ld=>;VZU%W z?o!)$Nj}Jx+Oef}4!68}09OviC(vLFS!h^l$Ck3`;$y#R41hyAGBJ6GQi;N8*jR&z zEphxL{ua&8#QtPt?Oq?CQ6>60sNm?nb7hcahfd6U_Ncx0$R4_nzfTsWjeU>K{rAt~ zWAXucG331x<nTHCnf${e|Mb&$?h47%TSW1M+9;oo*ObpTC1UMoQo$TVfkeDJnnWlU zwnr!BYazk&j{%u6uZ38OPtfi>&MrAqft0S5{vao@*()R;Bk>zg-P7r!*$29AFJ~Nd zB=;|%;6vXGXp@{sSk}ZL)5(dn&1U@GW%)6dmr78CPQ@X4R$2?8UOaso*cg@O3Ii3< z(V>ttDRSN6#hV2ePPWSn-Ce3FQ92W|Y#t5<hdD|T;ounRft_I+uSdU2T(?L)lcK@a zorUibbuo@^byA|W1^Stw6qac~O$j%+SpdBAdOV&~sbR0x1wP~u1UY7Tvven(r7fjF z3&C*jcf-}|S_P@%zvS+G;Po&qMU`cU>j7A6t<oiT*Iiawvo-N}c1;{!ci0~Yk9|hY zb32eLFH17eC_kbV_Re(AquH)pij(+d-CmTFFOWu7ayB@k?We(^lFzC|DWSJbHWvob z@3{hA(y?1XMGQXSzcNG>zvxpgOj!Oa69(B7!0Cqb@l<+W%rSf%?0NQhz({kFX@?^k z$P+UEuB*O6;Z~<Y0n5L9`S{5%U%refEeJskD)!6ywBO_T!Ax=}(2<88ncQ&(q{-6A zKP>R628X=_dgMny>PN{gx0hNs$BXOpExi_ez>!xz{2N#mh?9dkihT%xfi6+rUR3ET zOv50^<MQbp7;xcHN$)Abw?Q|XA&EkK3q(CpW)8@efNPWUKm8mAGbE_>h{9n{C_Ev% z>-sD0U^-62f0ytJar2Uzq=1lQT>L!zRQu*b06jp$zvH4t#Ms$?Gel%hB$WLzn<i>Q zjxY>4Y6LLqV_O{6r>CSHfLW&^5<Uc#Dqq5na&8Eci#5bSW2bM*medkxz1|+r>myP} zl?(okIg*Xihs|u-MKdeRJ=(+uO9z1T#dmHU7N2428I~LIJG;1+6rdB@XxLP3n>=c( z6O7Gn!5z(;%eI=gqQVL~(^jw!V{Q}XgC~Q>-drr&dR#fG?@hq4*vHg2MAT@FYBqfb z5^-Km;BU6d^QY06GDpx<+pnPAYW@g2*}n3wFv^%4Cr(^vuC(wu)Gdk=h*UGlc4O%8 zgj#3M_j5FXI<f&K>f@m%TN6yFj1$f&Hyxr6X&rRFO?;6Oup-2V?v7|Q!$tGOq&X@l zB=+>_mFxJJAPyy=T6i&mJsZ)&4e7Vq2L|u8h=K*#gWrwGH-w`c25#;>(lAsXR6KY` z0StH`Xb%3t_92tC;#2+Blttq%{_bCmxc<$nH-Nc*?0Maj@Vj^9Eb8g<iOI#d1E&Sv ztj&+=g#@F6Sp}(Udv@h%jA-J$S#qPet6=&%r9`cGl)#fG>oY>n$|(ojdn9d{&g{jU z0G9n(u~XgsIE8OL7~!<VtP|5kFPs@vM4{i0?Pw<k6M8i!1z%sV&F>$I6}DR;>Z!%# z<TPp_O-^>1s0Z2ElMakfA2bLMWuy@T^;B>GdJ~ZE(8aVpt4|hvO5F;3AFw*!9K~7r zrs7;+5(Rj;<`A|w>G*bM$1PQRJO6f3PlmXNnI*Ju!vJt->{-4Bihlto2dN{acbMyp zZd!pr&#s&e2Wi01%K41$mOWUu5rrGd@FDtqyXfOtaMw84-!VVzn|po<cNZQ+KDI11 zUt{~HNFNSB`~P}+Q0r`98VCpG*kG~M?zLnY=y7kC3>wS^-3v5Orb<eddD=dujKH76 zyS}=<Gn%cWSOXlu<-^HjpKtjSb~;{xo~HG6`|x7XkO{K~1KB}L>Du#2w^xV*Y+?P* ziwQ-&K#DnS=F5bx?BP|_ezx34hr@oa!JE7iI_liE_<mQ_U!zGtiLaVP#k`fN3k=jz zh1Z9$p@6WXLlAyh!n)#};SnXfYa7h7R@Fef0u~8#-{HshxSE#px)~v_1ScRe43`aY zC6{MQE@Sv6nk^G)#DN;u?QBvmk!5542xpg~iQ5`2rj~7Myq$yJ%t8&_gz)5hLcl&< zik&Q(S8Pt};{hffr2K=`0`I$~l}rI%gH3>mJ_D9k5h^^E%jv#e>Cc<_BbmOwUIRu- zqvT&}K`8Y*nnQp9H^F3r%+scqghWYMHez^URC>(00E$$yb=_5z(=^^wlPb-if8yzi zq#dX(zr+e{e@hck0zzTu_1F#Q1N(Uh=f!Kv*K|nn?;>(-BEnh7I2aDlpE$H{nHGjX zkH5a0qKQW}Uo35M-hMDUyiN?y@;?#v*>Kg>neh~(y|tVW13q&ApVn~6n5QUQ$$^q* zaP8artvfcz3=rYb9XX#rh&EP^ptzcDYf)_isJ5Mj<{E&#h3*>m=H2TN9%fa$K+_4z z$w`;ZZZleND8wA`UY!ysIGW8?Quv7=Ek_yRlS=Jv%fBh>$v$>cg*tFR_*PB*y-g}w z4=`sAH@7Q7B`A8J1x}GL`ns*{En;@J?`=yOsiE4e`y~<+Ll4b$2wqoyqK1mt@MV0x zqveTeW`>2gl{*7%`e>@(D?YKvX_Y$`Z}}3#{rNM~tbTciHO-M;f${m#<n51kaQgXA zoPdNUfx9hx>XJ1m0{b-zu-HFk7<`E&w0Nbh9AAM@TepjL=mXd~@RrQC?=`X74x>h& zoX_ex1x@6eK)=B4VrTdX$W3-#-RI2Jdz7am)YNoc!L$7lvQyW#)Y`=ADP=C5ykv{5 zr8Nj;)cAe|RAx!ow3*DB$%xKz!(_vZ+lJzd5jEuNeY6Ft#x3n*A46|NEE@vp2|Jqf z>_LO=eQNlAU?R8|7p|zcn!eyhv>@1gd*EP-C3xhQLQIF7<)mkm142gGuIa?JL)aj0 zVb#~$(IwQHK_wjAs}1^KZN=B^LZn92wvDPlCam=P+Y-v&e1s+kIp<d2pC1)(+V?Wr zYgqgXAk8z8o9MSv;h1uKE>h?|u@km0yv>KEaKlpsLJ4?XHGb&>rot~h(C&{A=u6#I z``b*;z|k^f%k`K&xvDyrALXo$Cc~^oAXL$75K$Htd;<nApf#t`CpYx)MxZ2pda&s! z<!nf|tcwK|i!0F&<%!V?0iafO4FI-SFJ5YE;9+=dPE%{xO_yyfu3EPygB78<Cf=d7 zZP0HF?efS?Kd-o{+>MPP7KM-{$b~IwUL?w$HdC+IX)}%5)@T?YW>i*h>Xf8zQ>Z4E z0mMC6B--mx^_IV|aQXq4075`q?~Th8Tig%ufm?0pV}v@Ml}(XJP#fCbyh-JywaXN_ zYHw`16vG5oyj%65^d8}iI&F*SKs?=shr9kM00lNt*LY0>`|u6>Q7>%96OsA-{z=X^ zi&qF#>$l}I^u^@NLum>3;}`RCc2*xhnW(H{*sWATK|IxAx?aA--XxE7Cf|}`+-HS? zz6@KY2d@1ZbC^@6l8$pUH$OJfEgzL*s~UkmtJM;9E3#lJSb-VaEmty5BIk}H8D4Nr z=>BkLnBCq{x4W%^_0oz^6e1Z}MMO9ZWuwiBoCdKiAgA--_WWVnvx_cTl&TG{*;37P zvYc^qp$1QTCaEd;a2`qlU^M#v;h+7*&c`JBzXIuDanB@}zKh)*o-NKNuZA(Q$72!J z?^cjgr+WgJx7;440wE?!FE=V}dC~1*$`C72!*S4Yl@!~YnTdHif^4!Rk0=$*G*x_u z`xX{Y^V4$PC$X*z^GO|5FY0nqzvWZbWhHNfp(D->P!3r)KVc(2I4^z=Z}lz90Q*KW zCXR{m*&4BJqA$XYpkks!F!#?MMV2mRfYfy6ie*z`am+b3@=mmb@1vc%QDPQ@Wr?F3 z^6hPvJ|i%H*Vm7~MzaYM|Ar}yO$HA&<7$?-T-yLQZHpK36$x-re28wiQFzXXZv?5w z1XoM+)pO`e3euLHOGlau*iUH<US$GGs*vN}IMb!m9gWMb2<vFiLOl7M6x*dEqIg7} zK)YeEYt0BFqb4(zLrcI&%k&~Lk}b^N*ZH=j+jLiM;W|+%F$v7=B&NYFs4zrkb`#q! zBOw*{wY#MnNOXpOOT^7;P2L-|i2I9iD`rY2mnCpUOj{`nR!xSlm-7bLl3L~WMmmD5 z3YwZ#r`2>E!8)oqDS?+zU=g1y`22mK^2l0|T^aY&9iMc=vII$qZcvC4h7rxAKC{8I zYEC|y)8p!aUu3*mR7v77aH?oOVRuho-_~~zN#|@Q^~+me6M+<w86Vp8&8w4n)t*rX zH1v=Jo^!H{z>vIUtiyPTaW~RrpmjtU+*sn3RlP7adOVkeAJ|Y#*=~8>f`)Sl!+QBI z63M{P3ONDE5Ug3MUN)*#)-hc_+yG;-Tb^`Td>F|)PULb{^Ret{6w&}cOyv2m@Q$2T z3-&*cgfYF$6KgX^S*}RIzYcd~o)5z%ndc9|Etw}9)ZB1SNX&^M^*g4^gj!6|7u!;< z3+iN)pVw$JbBf~=g>4L8Y|!1#NN<4_k1Q<728`8aE8HG{287-XWakv9!R)fY5k|~G zU;IGYenk8*C_S45*HfpMW-I0R==PP2Uz#`l&^$G<Bpkd~(m|%UeGcNQrX*1c`|uAN z46@uis@_zSUUsgx5=uq(D@`dB>5He1v4xdeYp$BEw{lCE0HDdJdSMjLOp~TSFE$I- zK|W(`kow$HXGL(lDpoXkP3eN=F}ZV25Rb&M#kfKAf}*`}!}#FLoX4rn%%)$3(fGvL z@Q3D2gsAvr=m#TgH$LLi%*x|>u|zA+Kg$$ZfwuSrR3pLS@|omKXYzQ8r&FP@Uq6q= z(xvnwbWz?;^iFhd6uwv)dU5kzfzLH}2F*iOee5*&vIl&n)jrw|v(H7Sp`vIOXzg1E z`m9UiOco0p{&ksB)kc@7iRige++4qgWk#ZCyiV06PM^^?|1L`X6<yn+d1QmZl)SrY z_x-FsJ)3YOuQhIVf$1%mlcpR;SG*hC#cFFL5D?hDj$X*=Yubn?<Sl2>2WKZtFvvvB z38!`Dysq}M=%D?VpWaU-_Hy^0imLh&m9zUxD4=e&={pg@$VVh-gG3WvQekBlmyvO0 zJEs_ebrO%SiSymEU7Mv98PYY}dM2tu;)&vmm?xc)=A^zVeu#1VwnrkU^%fk!*xX96 zd(%o`+VnoFyGC72?jweW#kfD5-D1GmCGxR@HyNWv5@)7C#&G?>*e$79<<7xYtmV#n zw#XL)YX6TiQeQ+y-%jXwOA+97`H<G))6wOcR3y-0HiDe7(Ih1}lfUCUGpagYhoWpW zkCF!BGfv)i+PNGEoOF&Lt7Ac%%`O4c%qtY;g^P6`aiyfm%?iDuG-Ro2Dzq>@DIpq0 z^9xfC4y9&N$?EyV*h<NThPQ1EF++L+ZEzf(GA~2Z19m9xxH?su%PCsj|7ji?rXCWJ zKgNmBaUi(QU4{ZQ#j7?(${0W|`Oq^rNMo;BT>W&Zc%=6R1ERa?1jzK&WM&JH(ldH% zJo!zt90smhRFh?VrTy3xT7MH6xPOOxh(jyk69qqd%f@#1<Sc_<q`-g2#uHJkl~ImN z@p=6QcoFmB9{<qyZ+btAdNWQXvD)mb!}7$GjzwR58|{X`5KLdDQ2Csgpn8d?wM*U< z=?1cqJG+!;??NR)J*iZV5dG=xF8aJ?1^;k>c+%+6#ZrSEn3{N3%7$n17Uk~}>(Xkx z4~?xL6vxOKEA>;@n>I!?rB>jsmw%&6Ytw<b!8KrnxN`R=cuSdgu8X!^+JLq6u3ebE zF;dIGQ#j3Ij+aB>v-^-k=Oo71i?kP)jd&0^&w=o+WfXXiUE^})S`>3V+~k1P_6#!I z$yw8~{c)s!9oHu(V%2SjNC6*Hxa!21LF)6W3$$b<H(B;qs+gL-%tckKTRy&+&)G2y zM>&)~!`U)@eB_r7-v`wwp{EqZ-#MsXsdC=SfD27tY+d6r%Z)j4)ug9k)jd;UUUCq_ zqN_m5+~vUG=e2wHRW~fJTo+Kqb`pO$?JX!~?^3A&VV!8Zz~oQL`Kfm1MH14?F)`Vq zvS|VF9-lpx&VjVgG5*SI(olS3;46<N)j0sFUsoQO>!d>1-QE2QADzPRt#;9Ia<__4 zi;uUrxBV#vfnD3Wgb?CI_U6{W>@vMu>t@oNPT0}lph-zmT_#EEUP|gFb4rrzk@!{A z&tzW%fhQH2*F%F>dau5sSaBCe7`LH0DPGyv{Ywz~&QxTgYKpNWFgTJ2E#4kmuzroX z?yjfBtCyl~eq}sw`P>_m^4kNZD%p;pF^*j2k1xoz-)<YVOB@1_8`w*y;m#1~pn@&# zO1iS#Jcr>o^)?8@-+^~(l_LMJ^;o@iFwu0tY}HNpGTD<W_DeWJr|9pNx-QC2QaDi` zYPW<OWzHWox~vx@WmOOP&K$&1;Z9EW7v<ujRp&7L<t<(+imR)*GVk;8`B;E+1&nJ4 z?-KNiN}?~;{Wu1kfmU^BmAoeEv5STsHk7gK!Y7!tC<;*cRFe_>uftGdws<<e8o{I$ zVgiPWGaytOTa~P7rlA!ALnbsB9ZT#`{I{u)?$qa~Bt{2f;?>W%ZJH;AE#ahgp^SO# zVh5FRZL83%(>&E=SS}Xxe$QML^aeS?dXRuiYn9Z;hm+r_n_^*A3qzvqiiDe$PMRy6 zuhL@j!^tG}epRlBMw$){7gH`_MtZL%oHm2+d#%>oGa&p=)fFt%9i?P2D}Hv#ak;8| za<3sarVGhMfS+CFCja71LY13@Dw`x!aR!do%PYK}d$F7W-iRp|m*Q7!mLK?cVuX(A zaYG2qfEnL0ppU33rpJ?uF~<5qQH8N<LD5~YAOj^U083}%7@e+XBlSDX4*t1?r)3T{ zj{gR;+G52-nr>K@YCY?gB=fi8jxw&$+DT2?J^I1p;))VoBvTQ;CM&<Ls@X$Csztu? zU)nMq?SIoOP|nK&;qSfl*w4z=;nX8QlOrJhWIOP=Og|=<vn5i!upn@#-5@9yQx^#C zYc~)slL4curib?Z_n8js4}LdNGfz`(_iX=Gx-skPLUV@+TpX&6;yt~hAwjfT>Ha^^ z=r~F_BAE+nF^f(xIhv%5yuWaIbYBwE2RmB<d^$W$x6Cf4SZJ@fi?5LAzH;4tHDF(8 zqQht>>Qr@+w}`ZafvHKau>igd4xIr8NRxnnvl`khfs9<VlLhA%!uV$mT5q+~GjVJp zN>QV!AqKKa4++B`{|3V!o7m7MT>vS)hlQR0j^R;cw`TgLno?ZOlsg9owZ>p$6iSX7 zg}JZh6eil)oXsmZ`_D^^LpqFMw6bn1*2#5j6roqPxT8cPMY*!Yc>|5SzY^k1q+X{e zV{mR!=tnuQ(7lL);n8v(Rpwl)@SF=B&4`gdHjzR2i1}O&IDL+!4a}0pWs!-mJ(#dV zPcY7`%f8d$Ok^;nQ~ZQ;mjWin<op}?g3?6Jii9qu@X|I6+#?6zbNbc=Axp=!b&C)X z5^K!)I8d~$Z$>eY-b7~8QF9(CTiqO;ssn|}*Jv_0YgRWZ2kF4W>h&5<dIuS6Ck$ta zO@2M=bTuOd2Nd&e5=A4$!Hia523B^9i-)VJNznn%N(@ZecMF<wGEBphlc@`ytl0W6 zEnCwC)h@~UaQ{;CU0tvh^X1^DTi~M)2^F0tI&@}g(SwEPioRi<)!WYp+To_Qx>vNE z>+}l7{rm|CT3>xpyg);J<m@#Y^5EJ75_IwV^j?D!d|GE%YZHPH`g?UyA?63b2oHNx zjcm}@SPh)IJuAMm2jVD2*o4D&$KV57Uiej;mO(X2@p^C6N;hhy8@18~d3tXbr<63M z_#ER;eY{Jydxg_EvHq96QZhQHFg59B>Yo`5XC#{ZZ^%mVIlKXU?{BTG>$#40CoH~s zAWKtoR?jNby*~m{!A30_K=3tDzlQ0ZpUU9TQqHe2B$JhhqsZrT_lfN)8AbAMnVdW} zN|Q_5r31Q;gS4jsVK?C~U5I%VRo-@COS0Z<4~i%mpmayVWkbJQUY&%_11})HA)6f) z;y&v+YfEz9B25`b&avMm>Z0)hg1%s8k$bo0_s-$%duid&6ulpGu-md^b?ZB-_3iG~ zch+e=BdpHm%_&l*K8q<qH2Y``s?$dp&>Rt$$`3jBi2Ok*ND>T$7d6%20U%$YfvhG- zecQ@y!HzVG?cML{9t;YbTm`UhaXx!GC6fdC{q==pTIC$Db`-1cfN2C-W}n11!R7n+ zi=9p`4Z&S06z<D115ef!se*dIVG_}2>uSpjW;mw&SDX=8ZWhgmH{?LO{6=*5X=)ky zHyjB<SXu}>F@&8}5GZU>+gLkvF+hJeU~Xn=GN;)%<n_sLIK+{`Ke$^K#MNy$+49FS zKl3!AwP(LW0?^x}6Pd({fN}oLokQb_44gFmiwiWuWCp7+s|&_XH;ii{l9pD)8MYpX z7XTZIzz>r)jl2~PpbrD=NYXeW)0MQIC8R^9Gtv=}nE1N8XTt=oqO9g{WKsZNzcVCu zWQpyKKyxF|+z2%9FVL{m(qdkgit;WBsVM<6G@%8ElR4&;4XvKqi*t#YqTe11i=*GC zjL8UQrwN$^F~cK+3S%zkbmS4}$esIzDU={>)TujnI!uMUTlR053_;7-2_6yVR@O2W zXI<dI0?V=u#2(9!s5kfci!tMoJ019Pomsu!?4kaYz1E>y{-<ER@MPGIiK-eVR0xK0 zUS6#p3TF4YU)+7lDO#8BJtW)VPYE>UAL1q~dZb5zQ66AvF6+sp;P{I03ux^m6BJrT zUALc8YFTUqHEY<GKE~}_)YCQQf`XKM_^J3AL8hso7XHvvfv#N-sLj+=h&PY@<5@MX zOSziGl!;Zl=$<N*Q;0@kGxohR$2-jvn%?d++4A%(R7pIiLQRWG5~U{>HY&;04m$EN z419eZ&PH0C^JZM1EP=6|T`ZnV$A9OxI$`nFB<4yPyU4esrOfrzK;}YP<!G{@?a}|- z_9s3IR%Ci2%MhuYmBEiB#Z!^X{FD*E38@klDF-3dG>~%;DwDy@6ij4q3nuCjPi5=E zH{o>{C4~hXso$nT?$VM5x!)?4_pP!y-!lE2^9^B;Rn4SggALPWZjEq>sPd;B^1EU= zG7<;{y@3S=meHXmt~Ih_h>BezQESP9;cE^IkQ7^hPnYnO?Tgu)8}$ZiZ+@30Fjid| zey#J>a~+lS9pouWM~()GXu9JZfD<M!w)bjNw?aS&6>ys}iR=i<ichTLwA+YsXby@U zz^6HdO_p&EJo26gpEX5T>#8mIc_TjFh>tho<Hq-47<Axv@yHy(akN4D{&Nnj0Tfj{ zda2kwTMY~0|L%E-A%Q#~nRtdoYwq2pm~@ilGErVt?E!p6;{Hze>gOe(8K#Y;pdKAf z7e9x63QKiy4Et32frpeEmSzK|nqPHYD7?(S;>}T?e+f8X-1USd){z>Bjrop`HA9q4 zyas}St)W8*{)1`SdzMbw-nr;KEWmSZHKdUniDfb+nV)5zddWx<UsMa7UDC!bDZXQE z@%PemTq{?`aK+s-|4(pn!!c9p?z$ayT%u(cpU8N}Fe-FIw~>WaKScq)Mt46=_P1)Y z84#rjyesrjV&G%<fQGZR{veQMoz`!F3*?Y*xSZ*(NfvZZmcSRD9D|R2i@b{YJzO+j zHkZ}>5iqPCi)w^4vH?v+(U9h&sJE^gWtet7fL*#Q#FpE)9m}h6x@Ym>c$~!O705bF zAuAliA+knek{eo>(^6O-?*c5qVkwp=B6!12tm>LLJ<3OD6^+sq%7Ji%(P;1>!;}kf zMpn`A{Y#+lzdrg+b-Wn<jQ97TFqcEfHfA@_s>;Fi@ud$5IC=`-1bJY+{XRCi`(%Vz zdQkKfsNQ-r6U_)tlL`T<NZlV{YN&TF8sxX%cWM-<#MDH!baVA|b^(;?yjq+!iI|1< zaPu(K!(H-wB{c`$TO+tATy60H4##4o4^Mp3p6a{@R=KJ3ev{I%GT7u(w>{T~h+mZZ zuD9j^&&DB74t>db?#uEpkWh_vBWB!)88>3anzvMzCfNgd_pxKdd1tw^nnG7{ph_xZ z1Oruh6P|hZah7Y|9jEzb;p>I?GqHC>9#4mIjOv>3M6_09qvD`<)dTXq>>VasKy4*c zG1B8>Ae)xgpX`{RJ)_pS9Un?bUdImV9x^q%j}xq}QIW1<cQpej{3j*_M!|5>+d9}j z^r<Y!HOU$an2uG&>$VRQ)qUnaJt-uVBikUXV1C|(rOdP)HO-_dr}1G;u*uFL3dI;5 zo496wW#dG|L(t`Z9hF=55==OOwDG?AFXY>gF?j?`d*4{GYy}aofBa)^#;g15R9y@m zzU~~SxTt7~{X*-_aL5itmUa>P$<z)N0IG#^5luxj({VKy;k!GKX!mF{g@8|3#XqK5 z=m_)J3KH*-=5MO;03k<sc=b@RKpl9vOqTQUkgc8Zzi41|=zn4JHfjzq>kG1<X4K+$ zig&kb698xXwq*juG7VzK2jLj8KdIF_o|Naa6jEnajL2eY$yiDb7cO0VnJG?-@x)wE zaa)<|DT@}i_;OY8D+d$iudVD#hhQBrRn8BrzXrC^&DD7wW{l(4<yGH)SEDAIOZ_*r z3NJ$ag@v)AYOIj(ka_JME`w&4VGozt9<HeB_wa-5mi3O<Yit(+91F$t`cH_)DS^9h zs9y3c&eJ#zVwnSRJsO8v#D-Y=^|fFTXXw>+?4i8~@6?E;4r!ws<D1p1Uhx1q-(pm( z<n3@gbM)=Q%v&hE+0I)2x1bF5#3r9%R;1E$Dz=xX`>7TvpG@h-w-n>uy0yt$!;6h! zjJ_t2=r_=xl%g2Ak>#8TT`fJ0DLnka=U$E;XX-B3sC%X9(Um4?j#wn<v?EcP#$>B- z2KxJofTmk2?iN=;RY^s!>SmgRy30rbHD{k>^pemkgeLyW0e5R4O(ZrH)^@rZqrIu8 zfa>+4?Q1E~iHtV^yKYoGH!7ZM+$7Uh<{O$~cl=R!Ih(%B4bLa8MpjPacC9n*9aoV< z(7)7l?{B~3`zP;Z$k-5grbtPkIJ6ghTNi&cu`WMym(V;U?*<zBHQ=5MbaMj>dtd?C zGluyJrx?q?Fa*%MGQ-8t7mNjv*`CQu-Sgg(agYCsv^d_INptF6tiA#lEu9%0U4wHw z$vbuGu9`^IlV;|3<fvIJn)9G@)ycwtL$t4#>Y2`BW`1Jt1g^cR=QUhI=g0-q3KkP> z7UyALqgUg>QntnGD%2&C4b`aFP6I>JDm*zVx&upJeY-94#k?MWnqYXkF3!VEsnnn5 zd3r+R9VYE<HE(cijx!yY^B4WV?pbfNVHyZsjN2+OHP!hH!(CDT5}X*6{(Ab4pwX{S zI7CAFt{5@t9F5<u!zYu~T!_#FEoicCf6>hAx5$%DI{8C|ZXWu@Op7at{50Y{;-&xA zAt~`T$r+#^5)^v{M_9G91}Q2rGQRQZfa4z8isw}cN5HA#I#EQzv-p&J*YOm9W%3H_ z91d*#3qDw1mF@8>=E7$ei-bm7mc!@`ivy@|m_^i#U>Weop7hclN6Rp;G@G5^si6Q1 zgiO9Xg%y7bOYd})_Bm<j5b7$WiBC;!el?s|a1y62HQ=9UsdF-$H)IKDT#tQQ=M{=j zWC@^7Xq3_|!;?ZRD=ccMSmr7cis`*CNN$4|Ns1DN-q~#zZAaofu$cfL`~+L#KOL6V zL*OzIB__5X#?Zs-+n;aS1p$s6NW>`#lE5vlL3kD2<)hL>Zj25~)no@%yY3iI9bP5} z`IEn+L8|@1qfAI;a*h_p82?5VyOG6iWU&o`CrdO=i7;jTLff<E(o62gNq3peq`%!y z%8)?E5w)T`4QhT32S7P#o!=7%BtVXfY+W&3_oQ4}jt^gsieAqF?@_1<U~G!sc|9Ia zD*6c`U(eM*^2FK9=!2{L*%F3`xG?0O9DB5cEnUnxpc$Z#hM>A(f^Z6jHLqXHfn?DK zb$*?fSHD69lUEv_YBH&3EgrkQ;c(bfUtrkB)mbxv<-w0VcV0=~zl9kU(oGR`+Qo80 z{gq*;uD*cp`}ldpsm=ug(u>n_hIyPW(US<*<|~}_Qe^KhG0B1h^F=v>GxQU^QjcZI zUeOx9!X0H?PZq$8Er2cw%1E9(v)1|g9AyGM5s#NhXJ9+LL6s)jyJFB*08vi_D(t}1 z(`x?YjR)Ht5!Al74ve@|E<FEnymW(#^ZlLw{!2;)8o>#&TQ($0WL_gmKV?}gKmm+e zWERK0#pk%0pfuR~;m03;tWg7>OGc-iFnt_brnJqZHXA^rjaqmRikeHHv;_zNX9cN9 zaDN7Slm;t-Zx4D?I04Zbh5kJpghe>$?ZM9W0I=lHM_3Zu>*Lp}rxE)1tFy{VfLMuH ze@d2){X;GkX`@M?Zr-Urb7>#l)ZoSCQ^aU&Le=GuqvDgb7RfAxn(TPwUzSHm9yc61 z)%tK_p9{g-_FKe00Li4%@UJgugXjaqaLe!avwA863Q+la(8{YcwDWR)ivHAp?zg}! z5pC0;!U-^BjDtJ7{u;m`QZLrq#VfbtLGOph$HyM}YdlzULnivi7ei@krJ&*s*NP6# zPmjv}?kC#=^S|xk$6L5@`NKB6l>gC(!?<7C4;oR?&}r-4qFxS-4p6R7&#U>V&H1H{ z8IRTo$48`wQW<C<gF+;E!RjO8$LC8rfYf?nzI%|u8QI|II4N7yDHu>VA(QG@o!iDe z5{{J<jEHG&_0WnAa}$nSU30SuRStT)+dI2=w|C(GaTe`gyYLqr+lQ$3sOG#GNkw6+ zlI6X1)iwF*D0BA(F{KMWZB3Fzc9gW$DQqVzQGS41=LaMEsXh@`0&HP6Z{EP39}i5B zcK9vuVeoFY$dz^6_@AXY@13`|FnW3tLGeR)qQbj%c3p4=6Jn>12#)=`kVBxHfH=uw z#e|3><t|d_frb23yj0~^P(z4JxSi<4e}gKw9?ozt?o&*>Gj(9o41AV(E7~jCd;EGE z#d+)wii0m6J|7hDAA^Dx)+=~vlN-W+%OfB`AgmO#`U>WaRbWLQk-m8K@h7jg26(D~ zuyBmLmb~wMy!{V)ueK7K@Bq&EZDo=g1baBIr}ro>$ic_k918CoGui%h`w&|>*mW=Y zg)X6B!a*a$^=5%b<+j2d5F>d&>J>SI0D#`e<`y`&;KV#}sLJ~%)rf_FG00V8U9pdP z`Wi(hkkA&3Wie@**R+Gx^lT0FM1(pR7*=`pp2I4y^sp?rO^0PGZal2=DjwE1RXJ{u zx&k$t4UAA)wFOCN{)p!ws8+mtFIp5+iHyB%nse6(mD$I-PQkOPDZDkjUPBasBWQ*< zF$h=;GF~7NTVPrzm*vuB|Kal6dLdpRS%Q+-DG2}}wA#*nRI3%2y_9@Re6bbOh!~dS z#3VGAl9V2o6wwpaAf>Sj#{ypwS{c#Iy<w(T3GPhi)C6Pq+ZjxR%r8=AO3;jXGcd7W zJz2T|I6@iB>^=cp(#|r4c$El2k0lgoC6r%Ec>C(<O4xwj*MePMrC~#(%{+BNJ;M(K zj-w`)Z$LyPebb3>Hq9xp6ro&|WbbGIupov+!#tl`@IWQ-kYGe{d2nG88PB0G28iA9 zG80GZtD5=}WSz5x-$*P8eH`YV+D9#cNF{U{o3lG8M~Fy@iFFukr4PUaG!;gIh~S&T zQI19$!2T+h!yCyOqIXqQIng`GHLqj<@E%;we<|4-7C^SUVsPpBuqT-gxMj})YBR}U z#BEQS)3uw1vgqdl<J?}=#Enw@MyY<IRA0})>0a^q)4zK5#F??sa$muCABdsPUp?J_ z`Qq2lVOaM;@?;B_fikoih%L>4Eoct+wBghjl-3HZQ!kF#e2PpB@Y1twKgt;*k<u5h zNzvN2H!sKaMGHS*+^A0PVGR(ZCp`~9aGgi(+<OG53Ojnbcvl!NF*1O76t!<}_Uu#$ zbs=`wm19@wcf|oPP4n`xCa$QrF!oWUZ3ztv8b2JI8RH<)sX^TSporU29)_4L>qk3% zw@`ziDH~J4oe*)mLpDt7d&TG!w3oEzi9H4FmwPE~lU-5}7s%kNhq8n*3JcpKDOAS# za5@HhckYFWXlVR4+FUlC+JwUWVi{w^;NI+|tB|hbxtOCoH<`6uwYU~O571zNg9}T> ztwxq8av#@?EpT8ayOT*`29RjL$+&AQpPTW!`V<k*ZeW?@UNR0<2!f48Jk08ErNgpw zrO@onBB?^^MzI<p6bu~ZCNvv6jQXkO)OI<sL<6-#>txoBFfNJVfu)H35@0y@>s!wB zoi$c553hUO+6zot0fv$2IR$yM#l7HyLVU9m$RPw?ig)X?iUKcw?fx2avX0M|mJ0NB zte5wmzkL=4kc~L1C<`MV)71&`RG6L?n1LQ<Gn>LWRZl1kM1cWF+p{>?;qBtHiX(!` zxKZ#Uagz0^0GTt#7Z?e&E+_R{BUwkFg5{$$69%SUzIl21xGQ5qv)&4t^ELSecn~_h zF_7<jO;Ew8(rk_<92qe`RPz`&9&RDX5wD+2yg)A`h{npHThi;G9@>66U@P}uwi|n^ z8uFy55!0uKlSv}X;6Zu@zI!)Fe^7$VK}5vETOISn3_)BjRoB-=<D&%gch(Bc(QeB? z@ks;H30`~8_b-lo>aP^s!D+9>)#Ho#eA!PhIHr>H#s$8;k>ze=xf@w-Z38KLn#0Ky zY~7H*kAsGo-71>LIB>|d0(6vQ#hbPJ%$elkQJ$sec9_HV0_N)>;LB1xNP1Wigwb+- zdO`bIvou8bDTwtqsGq#1wJ;HXa}r#mhpPS`wxbx-90*@%*1UE{RsBCuN3+3SLh(Hq z8S-rs>-rh{F02vy#Rysd_@bIGE$B8(jeZVvj^m{F(2^xow~7ag;`_)9os6l%qT@t4 z6&d^8mKR7K&OAPN?O4QLma;4ppMdUBjh{GJX{kxULNuInowv?>z_~{ItlUCem->pe zLpNv3I-<8TD}8LhA{HVY;#dcwfl_-k=jYAzS$$QDASi*x=VfTT3H`V8=b35pY>9QH zefu@10K=0E&-Dwmshdd5p@z*TNKNm%T5$rT5$4Ho_99w8)2Fybc1#03^#k2;98Fw# zxqjsvp5}(9x#4Nn*H5D#M)A9U^MCyI-~F$@`*;8HZ~xo>>-Yc5fBoBk|6l#?zyBA1 z`+xrR-~KoM{CEHQ|M~4d{kOmU5C6q~{2%`M_y6-h`|bb!Z=fW+r^>(km;dx{{`!CV z{eSqk|M6e^mw)sB{&(>G_y6PnremlTfAjzS```cH|BB+Ut;H{yBY|Izs~P4Qn;w(; z;GlQKJXFzJTJcS;udIq{gg$Op@55)|j3h&Is@xU<#}$aLD~<X>1A2Pr|A`){FqWO| zf1ppuMKc=}Kh@=kM8pNz&o3y;)>ULvN|_=|L<};nQeq^(&R9^PW`(arzk_3`oRFpa zyW%eO82+IS`Q<sxf~oKe`U`8oXyCuxRo(aU`1L7epm-$rvSi4(J)r;F@`pJG-2PQ; zjL9ENeKO&+jKI5_bK>TlxH%`T5vYL<3pC6`{6qX8|6fBeS~O)mZ`wui6Bm$20_V_e zAi+S(LCaFxVjr1A0K{+4*cNh14GPnz_&;tHMuS_TMI$!gk&j`PhAKP3S%v04&f*UH zO^)+o1#FC*#zYXQ&d+O1sx4iyQbOHg)89hXr&HKg*rk4{m{Hfo@Xky{YP2^=o?HdO z6l#(!2qOa6wyCkW1wRK=;)n|)POmuf)QPPFmd@t&8#wZUvyj!h1Ofxu`za9e@le<8 z!e$*<E7L+a8yiJ1ShqhIjNrpfM-)C`W!v#g@HbT0*vZY}l3(dubX*Z)l1nhWe2y|H z6d*5NE1(h(``O%OGEWzLQ~p;^rbv#_y%^Q!K{?nl-*6<Sga*R9Z;+5p?*af9ba&|0 zk2VoQW4i*gh~A<GQ_3(oUabY@CeBT!*bWGk;gvQWAYRr}rc8nA5>SaQz;WaxTnqnU z;S(!kegw@dbu-JA%`9~@%S<y?LGYK=l1&aC*XJ)w<K`yG_eRE(gY47k{9=E&Vu9`s z#z~QG_~<>XE$(7}(`+_tCpEp}$R4H=#ZZ<>8~AO#=j`R0L;c2bRdaS0zo9H{z<VI$ zQ=fP&Hy%_K?+II;t%#m3h+pE%T9{5`Re)t_;J=Qk7pw3>jzlcdE`SeE2+TR;`VM0) z4S44($TEX%Ogsoun3ah^yIn<AutD^d^lp}+B7ZP4XoHrjP6t(Kds?GtGkvkZ9~1}C zOUyN3!#BRuD<KV`vA5i&%iIf}fXwY>15>-WU{1*ygzpT|e-HV+rHWz0{E?>_bjHTJ zQiX!oaM|N|1MJ<lB<)rkgvUTRhGgas^u#+luS&ZS?`fjuZY@f?z4yYsC|^|;Nb3~L z1#D18SI7FfuB>cJ_S{CGJiq)op;gwJTbVxe4I)BfXU}qRLkBYK+O7K}t$R&STFG+C zv-N@s_w9@y1P~J0hhwz6U*zj6jlMG;CfXBZddIC^182Ul8T_1axkE9v3tYyGF`B=- zJ|vSW+poj3vi<sU3P&Vx(TimtY6qdQ@%izOi~Jlxl`x#PjNp;+a9EsyK!CgF*PO!$ zwmZnp+NCL3A9_j-Q9O$Sc05o}K%AMBvxJof^!JuWhYrwyLA`*(D;z-(W<F`x(7o6Q zCShE0G_T6nq2WUc4;12t?RdRecVw#ip<}AN#`FzH2&D{9Je`;cgY4WXdM}NM;gGIt z9C%-P36aMGt-zX)^vp5d@@#7o-^d;o-`6Jy_7M_LIiT3Y%1a0*N3HRk{;0#KHKhsU zfL@Lh?y64=6^hpnpFe$vN2-XDjvM8Z*E|0dM-DXxMNPT8A1Q_);8C%H@Wj?Kl0uuh zC9^Oo#cgxtnNMyFGEVK#JeDVIT9m=d=<uWvEAjfFd5h+fB?7DI?q0}edwGP+3tOrj z|K_5VWMcZfepXK}T5$)(xl|`fJadR7-dyy1n$zm}bX_rmoULs|)-r0;9jNNqL*PSp znCYc6*Ru&N6#I4x4ajm<Euc0a{+Hedi`F3wHuls-JG1Yqgk(DSTp!bV_jq~fI+C!V zmuSxbVo!aKnFz<uMWXD{<}o<IIhO9;52k`&-;stgFmUwLfjI14r>&i7d$=s_gbfcR z=J1N9S7i=Pbc+H^uGg=tM87zU^5)&@$bjn`5C_{$3J0f!F7!SdhxiqRf@!qei!0ew z+FtDk<ezNkkLHRPe*KnLHkB0&IVaNr30?G{*n#@nL18M=+TeeZP#cf`fyu7DQ`{|x ztq7{J#(*xZO(3>D%T2*Z6c&sWs(RVR)B+hRzS|$I9d(u+4KZk<Wly7HN-o$VuZkDe z3V|xxg;lDPucwcR&UpG0Yk6trpLdC>kmJ#9Td<Ad68IRFS=eQ3h+!VfAy3(h(d%e( zM~0mXnPo~ZmEFshuW<klutthUT$!Lc2hgou78lkD*NTfwI>yBESM3@~H?!r3EndU) zwP7iD<J-18B_HbBJQx#F(rF~Gte7MU6h^JR_3Ct&qf6Sgp?>GUjxf-Nu1PO==iN?h zr|+%bl7n%W-ge7}o%=4GemFBpW=#-dT@kaHko#FiZ==YjuPcn}kN#kH3I*g2xrg0R z5cVRR46XtW0Muc4dSKr`B%%3%LqaW`*Avyz`>GqmWb;><!dGG8cb0D?wF?CiP?ejx zZkJN)4$uu?T-1wJC`y=G!p~7DakM>N*4<2S21{Eao5PZ{5vJ5ZeduR_sD=LNq56d5 z`|pc%^`%#RnW!GoF<Dv-FR8u=73~^#*C8Elma@d-=e3Dg&CY~v6<cpSY8q{r5figx zPI3C8c}}5Fk{1EE<Y*^PnhEbz6LphWS}Z6zQ(w@R5QgKb%wSpZtuH#|c@TXYs?1(} z?u#ovGR6HP?ty+C1QOO?bha!8?ZHOM@CTj$&*#ng--8uUp^C4!0$#jgZ)H@}-1Qeg zT}ZnrjVGaNQoY%1EA$02+F&j)R4<i=_sZV5yOQpm>pPmaYOrwXs6Y7z?*#3Vr>Dv) zSZukrR9tMElqqO*`QaK}CL6UX0@tR#`83768|R4Bt8xaL0v)`k_g202Nbs>b|MiGM z3p7QOiNJq-(s!L1I51mq0Bz&+^fhE+1pG77O0R0&4Fl2LP2=Yp@MosqGW(i)Nqt(~ z9j^oBRWixfpvbnJYalbiWCP3I`Xj)AD9vRB*fN{fAmc4eq#WC_MY4z|qw73%|K7R$ z0*Q`z@-mIWn}my{@Yb2gtFFHbNa%MK!pH#|i0=vh_uU`t?665zJrd_<S+$$9b9|P` zN@=(0gV&z4k&~~U8eiB-O>dMh%mhKOwi^K<_lz@gYT=~Ok0C0?SsU$>oYg-*Y(W}@ z1!XmE_?iq$!<WX0R7dZN)VN~vqL}lM?~S>{zg=xz_fc~}3+vb3_A76PE#JdCCi1UV z;=b`dL0y!k_<d_J%>LMXU0^*Dw%BUJ8Umz2Eb}0i8id2LTXt;2D?F*x-^=j2`<n(6 zIhV(;IcC*@B1=viWB$i7N}$$A8XUzRwMCGR?xBnp0u0Sw)Ql^KHkq|m)*1sPw&c`? zX~T=8tZSa@ims-vcN|uiUXpQu8^>30OrlU{oVC4NnUo2YUp6H?LxqrgNEIikX?D?` zxkJ6HzvY;m55;&U+htR185Voz?~`{S&btf0Xp0!dUyA1Xx<#RJcoUtxXxZEhv%^O{ zn1grW+=bLNHg7QM{pgOvXf0pZI008_#Xl)7ufn8N2}rgRw`^V?xfJ!kNN%63W>UGb zaZxm{`Kxr<#NhQwGZX+1q(7rDvrsU*9e;)<5brrXbkUNYA5Q7D&h(Hg15D1ZuQxeV z{efu^Cx?2q@zjKX5A+Llx1Yvcp=-EnfTYE3JB2=PTe1oKY{`bn7Td#nV0>&i*c3>$ zNASwNmja;2;eU<E7+)4WGWJv^2~8Y@BXwKTOg}?I_xl=x;D+j_ih3GtXYXDpq5|@v z7k8WDY5s<C)Lv0c_eJ9>$4xLySvF5tOao;3%tB4{*&IN&bO%H_(9pvq2h2b55@6O# zQ|Heqt8aneXO2dDxj-jfd8LmW2P(K=qWd^EhhbUb!d&S)cl0rr<MlcIkm49-d=-R& za!%$z23b)5Na_1C9@p$4_D<Y`R|p5n%s8#ziRDK&Ev*(dWo2R6EPqZr?`Ua#2!guq zd^*MWI1Av??wX&mDXrq5_i0Z0u9=+1gD(g)`=v`J`94~8{M?Gh=AnI|D?3&o!T)*c zEeFt-hfU#Srtr>TgppZh#Wk4ZR$?lNl0yp6auY!B$pm%ld1A=d(fq_GK-dd1Xucn6 z_(;c^U0Tgb=hDD?%bTvSOwe{*VhVrdP9aQJlV&Y0(1=6!5LU8Q=^D!utU+9Fxw2lu zzA($H$MDMtd_r2Iai5p7$Mreey4f`D17b(EG-VklZrBxZQC-060b0*ABiNwrM~NDx zfexuHzJd)J#gr4YaN<5CfxrqMY!7#Ubs2tgm;&>vU4R@8^b=S>JG@s}SbsJs{zBlK zQ1b$<b+Oq@fN<At!fKT#9}sRIo3F-hX88+FVt$LPZw<m$RVXaOY@&dS|4>2${N+cE z|Hv~TFqhlk9Z_6M^2I(JI1^a>GigQef|D{u#snHKaQ_*&T3Zp?z$?cleWH2Sj(rdN zlny*luO3N!P5_I2(HG|Y=Y5%(VB#MnC<KjV<bN>nKRxQ?ndyhb#--RaebZHqhTqK7 zz>dJG_su6}mb!r0NmY&<7L2nHCpfNvKI-|W!48KMu6qhK?9Y^rIOT7dx0NF02hMGz z1$|-+!<bu*614i|_AiVjq{~u!NB^ykb)-IJy8zlIec{*fZL4Wq<so%*|3IT<q=Qzo z$!hF7Zm6+x9-q;=@ccpF?S{YSjqS@Qz3GP-+TB0e(5wR^@^8rf6IF>~O6;w|&vp3C zz~TYU%f4+j8$VbR3Mb=5Y)|T9)52l))n~3>zI^=TmoH!D?U&=j-aLylOdE%4geUX4 zQs<$HwfF(*SYd?FyEj$h;KnEDy9Yqrfif`|J8O=!)wbi;TUg}>bReIgq#v<>E9~x^ z{7fYx4}+ZBcuu4VkRu#uFTCsPZ5d66RlWC2AIJfh^*7x<t_HF2eD==S%&i9pVo6^} zg*sYwV~MEJ>9nq<o-FXlDpQ<<$*qL-0j7*3)!Iki4_z50mP5;073R2rXKf1htnezj zj2RLtfSvkew#X@R81zOwBSW7IVr;#AmA{!4oqtqXPE`Dv_6ZeT3}q+_V2b`oVaj{T z+qX6lxFccF6BOT99A8@$-#`oxZ90@t8b@H*j9jL#?9CIB%QT7VQ5cPNs3$&t=Xqmx zQ_gIyJk`V}K{@e>sAqz}=IJRuz7<C*2*xdAoR}-I)@<`pPLWMmpg+H87dCGeg{>p1 z)=mDCWUbQ>!`0|A*byjQ3Y7O#p5$3{B0rA%W_g;jf{_C`MN_e|bBm#HaMI{LnusNq zLInqrVmWhE`7jKG|4q@EWRX1%4(3nRvvLa5d2RzD_r1e!Kn>nRak`1(bQ8sC!wAOD zpZ;hUHs!N&aW;JZ)RV>^iW-aWt6G+LIh`Cf!})~AeBn_RKLGv-FU>e3-4BKYV!y$g z4d?oX_3#DNWv1A!Bh`?GJy?a%aQaQH@qHsMF8=c5bh?GSO;aZd!V%>}+ukbfs^?RB z`3dDo<3^G1d|b~tUu+?{(7YGqg6G~qXLckxtE5f4ZA;-t_y<lSI%)K1BvWTe!@6rA zlt-|f1R|<1WRMbwm5XzHRhzkeV)Oq|3j~L$ILEj;xsVKaJqg@kGS*^om1f#HMI$VN zBBPnwgwrP3w8ZYJXYGhm=Sm7hZkyNpO;y-+#)w>1i&PUp!*orW<JXrp(j@Hl6v&=e zo}j2YwmhR1lEJZCniFdjpbhTo)*bUD|F(5_u}+E>j<ZVFazlH}*%_xoMQoNv!a{q` zJ8<(%*##xY3Q;8jqr?&iJ8&0&JIA1w^Lt`#2*19VHy1N>GKYV>8b<S~Z;s=yu*Hgf zZwpa@3J%V8=+zKsFm^`z^qs$350KADdo6jJ7zi*XjVh<fq6b*k+g<l}4q@xSUIFwc zGeyEpEP^T~v$=Zb^>~a4s8B79X=$r@Z<uev+pemGd<}m;uV00C;(Z`M-%1|oOZFwg zt>yR7s@k!#-E8}3&E=tq5sHd_p`n(e>xO}e8`iJx_1{!~Xo854Z-2%y%?|h*qV$Kj z&f^Nt)xq^zhmka|jX(a2GuYf`&18J&tZanX=$xG<6ln)uy_kcD3mtq`Ub*4ia{>B4 zyQn9)b3^A<kwF2B3aH=(XbaMAc`?>-kKM=0dmC3RJ)t$U#!zNQ<QBqjZN^5&-$UC< zdJE(sg{i&)Ek53EOeC$>TPgnR@0<903k>6QF$$8Oh)}h)`E(H#w5lZd?FWG1P*`6H z%UXiWHRBw33n05pLLWGF*BG$piwK~aJVX6P9xXI3Y1(jF$lu5#hABmk8wHwF_H7r8 zfM$xk)Sa`xEhPI!V%WLc5Ce#z*qaFrWoX}8c-?KXY^e5+6V}*3n}OiD2n&6@z__}q zv<-!PR+?8@kxwqu2@}v;w~n}Q>-k8#^ae12opAi>LrnD4_ia@TBp_Wh*0pjOoXV^S zCY$M@yF+PF&x(iJ=|GKi8c?R%O!JcVs&F~40_B$`BR*Y;R9H@{AH?FqaMX#2B&$_? zHu8es@-T7V=2rw^cV1sv#|({@y4Y;7gG%Yj+{Yoq9~IeEMRvj>IBsq%HaWno8*ckR zTZ4lBBjrOov9S0q7T--3kLJnd7{+ie9OB#up3q}qok!&i<6I{cY++(6Z)sUQ=GNCX z-`m2^Uz$I4pdNpX0%C<U5^IH7J5WD8W^Kl_K3AmAc5901IC|KDFr`O(aYUY<X#93w zzx5Lj+d^mMEKbaf&(+QdSL+~Yak`X`JE@OXyZCW8Q&T?f9Cpul%Ne8X%toJ6Ob#Cs z>2GM)^-0GW8hm=Y>nESyiuYJKpEs9nkq^PwF3*peNiWp9Q@wo-NYz~x3-vM$i=?^= z1Q+XV;qbSEeT%0IT90aQc;m&qUu;+9w|k)e9O!R{xt&U;<2-4ps0gWUwcI1-&a(#n z-)dzu)uTNF0{(nKqAw5xj`SE;x2wqGX=NyFXpflCuIrBSdaAz0Ex`(;Ysh^2HP#v{ zg*C&Wa-<?0Do5POj8m}g!2U6C?-K*|Uyvg+Ar%P#h<c`O+I*7>bIdZYbi#g9jEfEu zgJu<&&j1#8j54jbSL`O}jLD6o@G?!1=A-Zm=3Nx2E!&H`*XTu(W<oCrQu~kRi+-Q! zE<k58x&{1A^e%>h&mAtfV+!uL%KBYU``zIn^zt0A;Zi<Q4{eeIDs;%EThnIB0RqEs z>8)-ahbd^C#Xb@AXUFZF*FwkT+FgyW=_Lw^3buBaRZv%!Vt^gR>=7piS4RL^K&8LY zIMJnf(s_yTK_9|$a(po%$4#P*R`o>p3XG%qwLy`dqvXy3z4#Z`*~?dgCXmkLha%{F z2f*KE)IV-0$F$vCU6-a7pKa~7p8B=Zm1DwV9$u|sxme6YE=)g89+f#Eve4C4E9t_o zE^Nv3BKje|WGDUoUC2_KC*U@DsB2|K_B;?(iA*NEyb$)HK7MUd+vFv&PGL(Ij6^p7 zGwN2_CuA5wf+JnymcMD{*o{^)0S4Nspas&u1R!Vt+1jrDNqzOKo<1wDtRhmY!VKhr z?WSMb^|+p{)8mcT0(FyrAc0sH6dNOJY@t=7gOF{-!o987EaX<Cd%$P4<m~CWBPLmY zjreo_j*7I8zGgirj~Xx{;E}db$%>>xH3g$4PsRT`>LmHb8Q%Tcx`Cjn{rYrD;UBHs zUwmpqDIOLN@WueBOsEJ*k#v8Wnf(Q(JHh=vP_5lny||6P<Km+P=qhOj^FVd|&N_hm zl`=oJb+_ou$K%pmiA19KHx=qWmCiSCD`3<%eSSojDSbT*jQJW|a6lJ)e4}m`9#a~d zSrGsZFC2YCaNigPHHlPw)8HW?ZZ)rBj%y2`dw<^Md+vM1&Yy4ZQBr|MuB?U~J2ib< z5)p#clN<F_%@TS%qPZw;L4K3qE;av<ID%920}JZC_T>XYr-$Pb$f<aYCLB?`<qu@Q z_~6AS<-6^V6?V$1kq*2KHt>gm1jj^Ijk^?U#^Uu8x|v;E4Z*mSC!ll*57~x^AuNyD zi~y0Tch51BYBIYR$`l86t8PC>FU=}5qAdyvmF>EY5M`b&LpeNqK5zIK@Qxwe?f4;~ z8(zpw-q*8)`bb~(^~3}h(ZLILLdVZ!QqNir*5m*V&Fi4_x!f9Re!w4P?w)brfj_rz z(^KeR>RB}RSKb=~wwR-70nXk7bbSy51hHL43#_g5^wADOS0+0X%Ad)O*C4Pz*asF0 z0{PKS0#m2NCAsmMl^)ocPmrkZ&g<!;vvNMdqii-Qk1JGt;-!$DMD-zH_=9fpAsizR z`<Es0N51t^*4vB-oQK-0X<i0M!$x~R=Q)JFAF&7@Ym>>DtqqvOJuHtT`0yiWgc6(* z^Qjcl8rOqP9M}O^SBgV#EWVv+Ja@2+PbP9Lii>|;l|S`)5EA2;FVR#2;U-7?@}-Bp zBOjR~XkD_a6NFRLGNWiWrWy2SKH3Edx#4v=BNni43jJWGxxeU|EkULqm@A=-SdRul z)Vsulna5&5R&EL(x!05plqwNkc|>h7#*M-k39)^krXeu{9OqKux4$oe4Yi_4pAmFO z_b-FsP>B?yizE_-C6H{5905xds3((A;s8T<luWjt_S4T@eK?icG1iXk@3Eza6A%+p zjRG9}6gr_wlN#AAm<0G=VxWrSa?;Vv8(n10<~*?31vLznrpxpC^t75kc>|c9xQdol zukiC@cKbgsUz0oN6Y~z8`_!J@>xMAoFswrBd*-+f<HbbPL~9W28;|jckMqFDg0x%= zS$9*~O4)S;piUD|A~@rUXnmV75ebYJl2ZwT)OH`aG6`!c&$CP~nOfA~XpGO=L87e0 zT+bb5GBz2bZTKJtzV_jWfP_+7czdx+{l(FwKIY#o9x(DdHpu)aWy6N;ThXSTc9~J) z(ym$5H*<}>_UU!uiGt<val?)WE`ZQ&MUDpU?{K5vE{v-`MP~A4*#MQ|uRaoeWBa6s zJ8&cN5X_9=aeZ<^?ECpQG%0;=uI_rXtR2AZ=z@3N2<`r2QJ-6QNQN1dV@h`dsTYNL zH73=#Fp~-Vruz)C#KjX}=<issJv76zDwj`-?e36jb_htx#7NfdblMCixa^9)!#tsm zBpQMVb@hRUB16E@;HWkq%6Huj8WmY|>Ud-Ql!F_1?|#PPPn|#6(c}ZVtx$?|LgkQk zBLhBh#}S5Ha3vh`s|PnoO!KHnaS0g?A#u99@C|kk`^p_s<zQtK*-h>0%II)5+|U)E zsdt%njqPl{xgc&yZL=!bOuxhkejrin;b^Tgr5Y4~ieJL0<X)9qeic(=6CxuU6_vt1 zx0Kor((@0o@iyq%?P6#6$tIn9ZvwYIRmjqWpQ*k_#Nksn=UrRbjcDZ_UWYO`a4QMT zpAJbdtfePWhkQWrZ51oIj$X%*f-p=2mtr|z%Q8AK>3M_qC&w4_wwVvm##l1_&MUN; zqONHLIR;`26l0x24by=&l>_Ot%WL1-;-#%l1H?fOu+GEDgzC3S0>fWh39IM50IN9H z?s-j)aOdS!fgY4FXP7_&sQhK+9Hq`Ga)z^&C|fIZ41Fj8Kp$#vQFY1NitnLJJqHkg z<wwoM+&I)}7)xDeR*k(z&(Jju8?p^hi^h3Gum~smnaKp1HczBvg}R)V<Cs3kDy#$` z<&>4AFCD^PzbM~S<Yi`+yAq#tNs~%0dNdG>q=5eG=#;!TD~-n=ydE0h$|}JA1hlhZ zJp?yUDK5XvS2sRNaAZbjpg0RUy2GRMJt*NFmXk68!~BqpM{rPhb!PMG%^G!f6Wv8C z1UIO-v%&YMtHF2qx1*n{DyUA%{!>T?$3F$n_XBWHp3*0izMFsV2GskE&F3;#)a{GQ zhU$N9_lW{JaI5F<`LLg{@vG^A0}$Sn=@Pn+s-L#6Iq2!}ch7->|1O$$yH~sC-J)!A z+6|U$H0u_l2W~@TCS5w2pL7c=J1Yh+uQTb%!FNu8KjnUkXIn`Ph!e;3F$FGt<%_w> z#!XTok`glsr}CVF`XFWvwBAuixs_7lPJFdMzawXjY#5(mKs+SRuBnXq&G{$iNd_Lb zo}N<9-L2vAq?uM;06A<&AW)M~9sljD>k#yo>jCl;2H1LsF=el#qZV~G<TBYJBn-1~ z*;wCvnyi+NN@Q@(jTYpyQk}1;@p;qKV@}V0q<W5LPBopNx|F>!FIsHwl`b>%y<=Xf zDfXH}JBKmMY`s|a4{Ww2?r7gDIVgfZ@0vx?QXkj1a~O1^e||ARqwT{8m|9>8Xs1-9 zTc%fLE!GBVmOTm4V@q9!?V|PV(kC{=O1qu!Jn*q;lJTMj9E(3Xfr{GIsoKfhSxo%! zzZ8*0cuYG&VL37R7<}cwtd9};_Sf~;XXGsx;>Gn8Z!iDVu72fbS>!y5s$bbmil$K$ zw@}r=#E9hi@9B!f{P(`Awda4_Iiw!wm%CjxcDaV0s7N{GP3rpYYJRo76}61lTDF!t zG0;ScIA~bRfpxBBPh96;ikvsHutyc%0aD>&JTEWf>9-fl@8L^E*o69f3JVu3B>wqJ zbeH~5(bvr1R(=t@kq$iqG9K98jgzpg6oZcc2>2W0{>lX-(v-4ZE|vdpXc8PoX!YbL z{N=Is*lL&2sxjQ|1!t<QCsIGTx0(xCi82j+oRrho{dMGC&xO<47VAc&L(0)|&mV@_ z^kR4Bhn2Zq1U+UtriNvW#u?WO7t;2Hd)rTxXEK|m<7%yx7j8(<6JZDZ=?J_^^8pFi zg=Dn;&3HtYwgyFyhM;CAdn`DSEIt$T(A9maR0C?V6{3OK3vKsiaE9?8ZZn)4pj|t{ z77Dk7U;6{@3kx_#&1b<U%QpKx@(=IpZpV0l@4F{cQ<mJAw&0h_&HFv}Orokm*gOTy zgG+3~_uPtVJJObvo!Pk`Gs~Uk)m-72e)-)9!NR^>m#W3fHYj)l(GKUZvuc#;+_VFH z%>#Vx4Ny~^*TLw_LdKNGtAFtNa2+{2@cemSVq?A|VMuw#SjBVX9^-30wWAE}bprUF z=PyEE#SFyCTCP@NDhQ)hRNX$O4{-%e#eP{=m!dKEl(_pCjIW%C5|%~SVE{*YBJ{mM zx_MQxl?GA&=Hdy#E;h2wd_RV0j<OB~zU9Byzkzf6uoO^F3UKo#QI?qUcq<^^F!52C zM$3Kb`v$jx51S|TwdO`BXOF1~xw>OXTnl+OFt%~=<~E8Eckx7To-V5Mlz*`~H%P)T z*o1|h+_C4EHkOfNPpwT={7*qWnD4lTdFjcGL}lUYmwJ0RGzzaegLT9UPXE<`9m-+% z*P%5Tj8&Fle^BJ(T^b)l>$IRZh9*4{INAr=8Hi<JEH)9jyl!^db8ceG87+@iJP=XA zc)2ec&iQVK8~VW|-PtUt;CPRnw&MM4;y5xfboFEJtq*1cPVHGaJtdfba0_n!H)XG_ z9M;yiFZOXHV_)oV<-9F#jBxvB<z&)a8k1geN{Bbf^$mMF+Vt&vyV~vmg2`Pty3YoE z_xhN(x8`6ZYRjUs($MOpr==~MajgQx?eA<APp5^+56Qd`S+h|(vA0blj3TPN@m7{o zN!64~DhsvNsSas_4>9a5!BLr?jxdHYn&gfpoc6o}qTk1aXPz0+?pA>*8em&s=wliS zUIo?7MO(b#V9zE(DaU6eHJ-&<6F4?uxNrM1gOw{jyP(+5HZ&tKa-8WZfJJ73udAZB zyZiCrCqMbbL`coHfY!xhHYJ!>vvP6_bq76~!^^5TZ^re>63`H5O{8$MBp4nM_}~TF ziIw#<KZXUye>Kr>+;vBVLE_1jwnD!n_?Ve#T7*P`(R68i9@uj{UuG1-rD%_xKphKm zr|5N*9A&AldFnXN9zIDA4ZF3nWE3X(IbQM5K00ZPJq_7S`)r8j%KkW}{wyf0s`o}1 zJ;iz1v3DE%dlKjevHQ1n5Ui_>fMnEJ8@mUH=oMXAnfCW=nwy&l`!^BxZzAlIRp9Cf z`y_Gx6s?R&^xWgOcn!V$(T*4&EcYZ42km3p(a#$?ZH4<huSky}R)a%T0FOvGcPZd~ zd3<I#2K4EeOfc`xQ%vLRQXes`k(k@S|8fRmml0{GFM*rM1TUzw$)o0!8`JmiaiL!D zxsBPLM?}Z}vRZO(Q0noPSBU7Xe*<^hQMW<-`dIS8BtAv67Z;IZNYYUz3e?j#&E!qR zIg}FL+i97qrt-ZcYtd&sU?}ahS3vtan~*p>8eT$}9DO^;pq_F!F(fWpHx1TY1v7aa zE5Xh}BdnKNyn!i)@34EB-kX*lrbl5uKf`FJ-x*z}J%9xNOyxeKTbw}5YUUoHMNBX^ zv^A8|{D2v3B)&}pW1B9e0={pXb2yYRL-){}f!O3Q0Evb|+_!!C5ce)RNBMhnceK_Z zy^L*?V^VO}C|XSBFG`hUleUGn(XRT#V3TA)sNH12z)XYW_qwIT?#*<vM8bmF2N({6 z*XmF#DOBcurggbufs#71gQpq9kHks^xnIeQn`V%<3Z3^urR)1?ALh?ykkUESg5<D* ziXK;r%3i_<msX<X<V|Bv57A!CX3d<N9<bzdS~W_JASJlnZ)!51Oz|+yC9~|(WEp6% zC51yy?CC(KWT?o~sY#?oT7JUqaE~6S2GDZF3$PK?fsGp$H-Cl_!hW?d=U35Z8S!}D zv~6#TvS%PG_@=H9=?aikk*}b;12ttHINa4JH5y5=qWMHDWG_Z;$fN|p0iT=n1c|vc zd%&N?U=mT|5o|lJzPgfd!6xF}>ht7+q_$y7BU}-33Xjll%QuzHoXq5QAOxR7ctq0J zQ&vaVX@H3u1H7}QVX{Xrsf6VlYzI6*Uo{{}3Y&^FbL6HI-n_zWFt11k$IbA{<>Y$& zszB@YU?b>3s#MVa1_0U`&snU6Jh99WqRAtUlz^Q=It22*%#RuDWj&c-CnG?FWH<uN zQ&DL~KU4+5!7&*|y$sPoU#+{s(FP}imC@Nun$!B2)h&GD49y^gV;Qj!8%F@cpu$`O zh46#vVXh{*QFx>a4nUN}f&r=<mWO<4m>#c`F*!-8jlCEYZS^b=HTR?tu^CQ@KqSB| zyhP#|yy~xRjJmrOsu7f?g2oPK5%nY)Rzo(h>e0~Uy*67ot%XT$Y@VqHX7Ghrtm(x_ zC~nqBxwz2~R8sIxG9<kedeqg8P;Y{1V_ulhDLBO9dj<mk3RJJW&$CBW{5T@^Tu*em zuUa02q5zSBXp+p;dB8C_hB&y5HVo4s!|%kg%7Pa5`5r!OQ<Ink&fW?kpx;@%+)+C= zy?tC3{aV$0r3=*UFF-7wR4;fg`(d{OwPHGZXT<nKNMnZSFBx<{ABin2nwe3*SSlfC zaLX0rLgyqsu!>Awt0M~BAR?4GRC0ns8d?Xkuc`kH2b;uQ6o~T(J1yf$B_kS-g;nxp zqO2#9lM9Krw1Dc=2V-lJ(AP&DXFft$TEikAyhc8@0vI5Kp$ce@&V}{DdT^3!$yt4h zTBg~&Zszr3xs^CQjcDvRZNG^qXc#^ur^9~|tBLSNkv>Eoa|Di(nh^y-uTH$SiApiu z&FN0$@fK>7SEa~<t2T-hD)Ip1%A^G&bgdvVDcDLMGXkkG*w&%Nd?X~9g)&PSR7fia zzQ6B}Ik{g*QBSdEP$(t?J{E83U0(^!awH|ly@Jsdjnb2z(MH5d(gzO0A)OVOPk^af z4V8@MZL3CDo|!zq(m6VN&82rTa@_|hkbq&p50hi;<VWbA*z2Z)R$rGkca!0iuX{p4 zlVWEROovwX02+dzF{!Nn<vXWc6RywANMLf(BTgJxJR;wNLfO;e%d>LaT-rbBk|<2{ z)3Zh*Sj7JE_+n1~eoLxo@6(eKJ<QyPw~8gvjucLBb5d86Sc~G1`+y3Oa)KQ3GE+>4 z6~jxIV0wy1--tkzL6iZM2w6ewGbSu$jUrhCkb4~$wQ%q#;k!murHP@y8J;?LUqN5) zE$Z{?9!Q4LmgEMc!>&HDkOZepLS8H6Psiqy6_^D*yeS&01Dwy!Ti)48eBLb|aeW;e zi?k*{kQHf<s<RS)%?G~T!RXJB|76Pn$}UM`8(OHC5>P@h%;wDujYlfhIv!Q_Yz{^& z0j!LK$FPWP>{#_)EOq(@fc|7j7Cg>w<!n~Xu498Nv*m)iJX)l-$q%?im7ewUW@6F^ zqk;?_C@COTG<|fR<B;HJX(uF_Sk=8D8~nQPu>JU>h<2V`uOesJIWummRzy5%C~_36 zpzC)VDCFWrvZlw#^K~uZL|GxL?VA`-S0gTbc2_ADKli-BqDm?dCP{Bb+D)Y0a(b{m zj=hm1*Ew?^%)&pJpn0(S>I2=JmhBm@vMjCQ+}RPz>q%0eC6LBxA;d-H&(JyP4oJk{ zEBJs|*0n*c$3T@)R;5Yvcke1CC3w3^z6kE#J7ZLTv@4Hq<>rvQyXm+DQZDMx(Y^_< z#f_B=t7v(l<Vcn)7(AL8Xy2W5ikVdjYXx|$sO6tpTkEN3t<sisxEZxLtiEb>cpO}z z1p09<DkvvxdH_L3>@k{26_Gg$P4<5l6z%#Lj(7XwV<*as$^P1eQR|l{0=WTrw5nVI z-YRCgU}uz?dNNWlHoS8ZJ~4M&l(~TN5_cIMJ*!w#VuE%_(d+2nkkss8HyqdJJjSHn z4j&6w3xjM+rFT@9WD(rkd*@}oP_exc&^r-&<2MW*fuH+bnpI`J<?<rE(}7Aq$7`Oe z@_Yu8G`;wa{%blUm@`tZ@gZ3Z<YxaZK3A+vrJ$?_yMy8<hl8lhQkChIlS$7jA-U{Q zV8<Bwy!?a&gFovfAX72U71X=Bx(Wav+CM2UsSQtR(QzGdJ9L_gRlZ8f0wlB+{<Wcs zDOWhvPNx?fmb3h$mWb=aDCGg|4YYfB4jl5+DUPh~>K|a=?i4+q*xsRbqNvBQv%{K@ zID)PQ<_FNdE?^48?7^ihODkztQ74&(;IvM30smw}S>6cd&oHotS5`FUjM8q1{knZP zpO;G?@K*8T{~rJV|Nrd0Yj+&Sks$hg&e4A;&U)NUvC#w|`R+`E5EdjMiF+x6OF%E< zV5vfP0VuK2)l=0?ph@8z*<+8Su{`q5%5RUyJI8BzcWgb*%(Z2TlK%_=QlI=6Zbn2t zA|KTcP#W9g?jsT1Rhf~Qk&%&+k@5H<+1svek7)7<;29Q3zgeq_Cy;EX2l%2SjviA< zN-l~*i)T-(FB3ebUGAn=azCP3x}!o?)yS0GDNCM$tW>5JgQsAoHymIu?)8gMQPM@@ zYg<VojeWxv$Sm7B{a=kCZYU&Ek#%9M8<hYYxzoLb8}9=`sc1-3r#e9XkzIgJr9DS< z=&sg@qMO_ZrKmYcisQ*kyIA-oX%FxcEBzAPc}p*WEi~Tjw&MYmD>CY#1_dLOXf}q0 zfQ~e7r?=3CPz_qmMqt#LJd>&UGUHl`=#C5GT|#KTN(c>!$}&b+V|m5u1=zOg9Wxl+ z6&5A^sOJd=VRgbY!8C)`CYH04b)>@x9BK7fZKnY_kBiP_Bq1M9zj_7$g)hNL;LBOI z_mT>y*pha2Xx<Adj($(MmdM`LEU*+Ja6eJgkS8-;-sxTz-AIzIX;#iEn8=V>LwjJy zwWN9sQS#{H9PFPFLSIBc4W-CPj+oO}S|i&iQ)l0Hmfa}i(bq?x7lQZDT|?*0oCepC ze(TbWBso>?rsSapzNW#Ulp5ew#;X~q7l{_A7s=T4!e-WC7R>pzD{O-|vDq5OFxVq$ z;j|~UFIbzK*9_V>+RuzypI_Hemp8qANDj@?pg-J}#7_RIi^H}TH;&c_HtFp4r%r|U z0-|`MW3psy6^36(hkYStCul}moJC>rECM+s8?;BPu>9ngbQtwJ>0b+~yparCc~v2N zFVt1ig7jo+gI8VwpUXWd-YrZ$Hl`jMQ;&^FJT}x~#vCWjb`1U5#@N*m4Wj;t$#kKf zsr{y<c`!V!Y6V>n)0~iD+$8yMuUiLmkSNM(XT78;Cd52KZlD;81C~L(0K*#wTHUZX zAo8Tsvhz)8Zf%>b*P?DbXlHTD6(9VZck8gM#88?*0aBU~KglO0inl>Y?pa|M%A3X3 zm7~x>=L-4$bd6+IEv#yH7czcDah=NA_OEMM)CJ$yUPQr|6EVekoTu%9fY|N{csslB zR(cGJH7AZ{c?w<ZWfmj=6BB2ik!z2}+v?{aJV!6x6lB8{u{wJ$pDx(<0*5@y#z0eR z(HG-fhB$ySO>>2&t|B_0@AQ#l+f@k%Z0q*(I|nD&R=tMvJ}(oo&>pJgKAuEJ{SNM< zIQwU{4*Tq~>e8X<-6$Be7e9sC{=>))B2iiij}^d5makqqgi(7HRAwEC3}?A`IGtqW z)PUnPZE^JElcaC*%4ey3(eH^jyz4;+75F&XfJi2S<_wK#Yj{a1)G0+awON&3K4XzL zRB*|A^1;H9CM(7ptt=auj<LHfK?M7a`MPM4>dyEC5KG`=5djd?HDm;qBrf>Nty|He z>8LHlBm2<}d5p<Y9U-WpdZPfHD6P@(v%`Y7jX4eiOOfd$Cppy6h2K!T6YMm^?6nn) zb>f+PrMitv40FU97>y8xu=ye?vCk_*Ym_>M*+U8r)yJN@=48`V*T+s(3t~@55DP5z zhfNzuRMKNh)0U9xg(ykp?AHFqlYxL7Qv)fC5fjkY_3IZ`p1XeC3#ab3y8!?;J(Y29 zH_!HQ%2yx+gmq+&FlLQBO_3F7Q3>BjJBVa0qM?!awV*P23|ms(h}{z&_Qswi*4Bc_ zNjE*@O%Hj~L*4|3ybH{IwF%knN>3UThY}r1dZH6}4veZ7$Pi6d-AcA;?YMPmyQZ3G zC~oVESdz%-HU)jdY%qt93`^hG)k)Qy$bLHPm;6nE!BhfA9_f^a0u+nn*8C&E>(Vxc zZ~DSq6)Z*A4k*9!5R(}uXo7$FEhX7Irw|2AgDA5l87D|(Tg5rFT96(~X|dXY#AK`6 zERrKAG1cShX;KWEnt8lWOj+f1qk0O%3+@i9i}_71MU@P{gZ9^f>%h6_UK+CwQBny& z7WAHyNG#azF~bCGtnj0LP>blV*tYP?KILY|?*sK$z1)RFOFiygowmRZ{Wc^QYeci6 zJQRRb*MgCB@d>47!$&m2Ml^ru3`xeI^I|b(_?)!ScHByVV>J+s6S!MFU^lBvG)649 zF~2DjdP=f3xY&4XQ;g5NV#fXr4q>(&!kh!ax*G5%+dK0RUUXn#w|u>q(aDqI|M~BE zbii551X-bJL(Eg1ngJo>nYg!G--{2b^d;?7Sr=-i*4*Z4_G{LUZ+-SGO8uIt&{JAx zdL7t739acGRKzUtPHym(vbhP1flWZI1kAxCU>=hbY=r{AQ7DFJjUqPzSrJ=TFF`gx z;NpH4W<^7D){QSFL%=--jcr?og&sr$wD2S#hO$MGXV9YhHm;zVIco{gLW75Dr5U?? z*<mM!<Dz<Q2dmbP!A^odruH`+9;DVhZ$q!z5IeJG!vzK_DM*?Pw#)-5zu_lfw4UNd z=VbjZ8Rtysxqn0GhuEdvbV%T0-HT8+ylI#Tz6h+SK$u`nly;op#AlhNcPbKaVCJmK zc3M)*sJR54HRT$-Nlx96ZAnO4?G)pPld2ScEeoN?YV{W6WSP@db$~E<IlXL}o<pYR zkcrMATEqq>f}1w5SBwF+rq-XM+C!s7Oa9g(^#oZZ6i+^3M0U*_xS^NplBNq;wpL=| zV=H@u4U4<9sghoO3`U8gvw%s51Sb}=I(%F+GIjU>VoBPj?&#br3}rf*OT!r#k9pOO z_qsec&!KI2bjnX_HC|3Qu<?B+#)NSE9!ht_glw)!f1)<Vac*JIW(~$%k~OnjtdVYp zx!zzUYo{*3rlGH5?=K*VNj)?_my?d1BKoZiUik)C^faBDN!3cm>xKQ2RqZ7Eeoe|1 z@pN453dzShFyT;@x2?+E1;`F`!9w;)(~jweRilB1kY3BuScUw3o1=Ywt%0fvXT)N+ zre#J*w?gtwPe*@YPuPMFaZ9|v;XDs`(OQQ(K_=Rj&^q(*(ay^3YxZ)}OFFo`m(>$J zyB0rC_O~&c%q3n#90biTb8xU`dG6%N<wlG+^+v3K94y$s2hMLi%{$NN5YHZJrk)m0 zXL)){$WC#)>Id5Z(ts&)CUqdWBPJ&tI4`jS5le9=Xz0KxLmlE!ye`5=QU{QwJFF2Y zOAk0~xAj4IRI1~;p4+C?b3Qy51+w3uMDfPuXB%M2A}kWXjKaaJ5vIy1<6h$Pxq2bF zhBP5JE8Zq6Ei7O(V7TnS&CNk6h3og5UW#!4wIldP+E#>v7?M1%5gmk*-!-C8cnh4E z2K@=VCY(3qUxy!s)8=WH!$9A*G0?nW1{(&CUQRy4h};+xWr_Cesc@LG`E>^PaF_*a ziYng@>DqG5+anLlMN6`Rye6;JC1lMYhBw9T_@+m^=@D;w#2Y&TbbO!#v@iF1Im_;% zlgXg3o79R?gnIb^9ikrjGmurtDYm^`0?M4XYVwByjQPdZ^b>+c@GdBGDrU{7FRRI{ z(X(W1@;b(((KjrtSkRe6wmfTmex`<OR|U%>ii%IhLCfuS)fd1rN~Wg5#~6DhH;9b$ zMw*V`cLywM8RPNW745Rx-^z>fc7ph@`iScu%vDq)I$--q(D!OXZ~XjvjJ_p4mdz7B zVFWi2|8%bEt`R$xpFp7XAb{!Xw!p({i=oR23O4PoDekz8-iKb6D^H^h#lT>sUNUK_ zmm8G37>aLl#26jUkdu~i?KKaPzA%jg?sip9$2Au*rIYA!0kToM5R0|q8IF!xe@cMS z>)f{-bo3cz0*V^04bC=x{@hGOT^LWPDLOQ>#bAU@xHTIt;b6Y*j<2b5{T-*mp*lQ5 z@F<R<yE#&yhZ~3+slm(0YQr0vk(I;HvZkLd)z|C6$Ew7fZxeI)XGO%}-~94(pG~eL z<hA*kTWJF9bGXlLu`@DMfcJJHy(4^F-J`}oKbUyUbmt<Lf>A%6kE|f;-b@E4i6x<B z@E%r$_RtCzwOIIc1UYmU^1O2`KtLVVg>>)3G$mMQehfhKE+DZn4A-1EF?~ZS5<ZMl zRS6%*h%P#Sco##86h5#r$X+m2PeEJND+5U4FW4Hkb;uk*slp|!Y=KWmYyjZ}$T*}j z7@{kI9%@}lsOhlk1U^N7>uP#H5s~)NTZs(!94J_-XMxYbT!SXQBn<3;-p`f2KbUXO ze3#Xy_j8;6YghY%ogJ-#p<<MY+RSjd9)_ASY$q7+GpLWjnxT+>f1E;veVqFoBnv2I zH!@5va88#ag@XX(QVHSryhEQP!##Y)&T<c*ka)&B@%io0yzWCKZnppg6QdiJI4nK7 z8pMb#KLB>)M(NhCCS>IH51(;F09kId7=Q)G5CvqpNyP#99Qu#S!~#CNp`rmF-x%FK zfr!8dJgfyzRDjQ<+ctb;R$?Ta>w2>;p`%f`GsoVsvGCAc8$P3UZupGay%Elc9US45 z@8a+Y?c_KJ+RZgTDJxdG`y%iW$stq+{`fXzt$CejFCJ#|LrRx&^-Vxybg10~eQl0W z<U&cQ#>F6N6<KS{Is;y3dJi<c2b$gkjqP$YT`JS1GF>X;ES0knY{L=6!+7M+#)+*0 zn_-3BH@J6(>>A6B%ZyOU?g&)@O17a+oWpMCoJ+qlvm2C_NHQGfBztlN3z@k%ecy_k zH=vJzR#d776)je!OQJvW;5bU}5_o-U9(-NKQOK}p2g*Yn!e&!km@H?}uqh@RkFSl% z0>}oG+zm1ATqjZCVbF<-BDtaL1NK$&_dxwVB_BZjPzX5b#;6?+52v!wl)+CK{KO3Y zl1#zeN^T}?NbCWP?qG&kfw*S7-;!%shZQyq;kmTv$L;D({IfRL+sdTg5NutVY2K6> zX;2#pi&Is+jOGW`E*PQ|@&_x19lI#VJ}NL!w)6!TbcnfU4L?%Mh*-eYYnN89U0Q#! ze)Zbwg_X6n)ob;&_2u<TYhr0iUI>nzYY0=d_sE}3xyF=hOu5E{mUIx6?|hu=9Dr*q zV19|Un#_uO{W@e|5&z`ZuNzSjBpV9l3$gaO$e77s@F96oMNLIf<Fdf}lX<pB#s?(- zCB3z*pF;>@QsUI2+k!w>I!yuUB1vo@pn|k?=o`4e?LwU25S&JObn<@Pw1Shv$H~`$ zJs6UxgQrpm<tDzmBB<fYQ&0{)m7*p8vK6B|=zI+1wvOg~CP@R<#RLY_<4{INOoSqP ztkWF?%4Nzy87km4hUDEUu0Wtpk={xIHQ;fuJb_X-&^*_~`IrJRRZlzDMaog!f!6J9 zFkqXI*Fg^84E5@4H1AZA1l8^N5;1$daV5S|i^M;h9?8gK0L9a{w9U8Lu)|^m2k#?h z``(AI-oNwC(Fb2Wdg~{wQX`rVl-HH+ycwwE(j2!Gop13xJndvhtkkE0B|sd0@7MrN zXgzU8X~;O*)*G#<^aHI}<#>jQSsVDkd1bvk{?V&P@4xlwzkKiL&d-kD{>{A)@64L_ zf_*%bcAl252o6}P`QV*@5H+FN@9w^N@1w6Dz4x2rAAjZOAHQ|{lh^M5>qqxK{Kows zeC_z3KDhsPzdHKiC&xc}<M`XJyA3RNcHnU5z;E_*A$2(MIuW0-5S<z_<GJ&AeOGYq z-K^beM32vMfZ|NG157;y)4a!$$HK2J?DjioRRp&u89QXlK0$xugXhV0!|Rw5^yG|@ z+o4KeM?tv7&NWw$R&}~OfKm6FNIOHC`1GUiJ$U!04_^DfpT7FlqdRZAmF=2*zW(Wh zPrv(b4}SfR_kZ@0m@5J!CQdv*`uLxZ-u+n2qnNA5X8%IGJ$mcC2k(Az^us$x@BZNE z)w^yZPeBgTiYQvK+qnP1NB4jDu4smU`i<!V@ahL&n;)mK3o_FHH9GG#ieCN}7Z4Be z16qrQn+{04<QG4wI_g8JPxd_7>+Z${q@cac5^PMx*r^yh6=Nq5V=WPOgwQJ1dWG%r zT~KUxnwV92Ln=L^N;eaX9<wA-xvjik?5-t{mkxG|3fwbJ=5!&1o~wL!C^6i|(=w%@ z?036a-s2kwPyl6^z?jF3rK_lhQtE@ejbfj2m5@uU00Zk5lgw<mk@#4{WE<sYJ6`gL z_S<2SuQfznz&E9`Hgv1L{i@<BubrAt1UuxN21<Hu0~;!`h9FEJyi($nW;^Y6ofzM! z>y2;5X&d=Z#-xmSRuZae@zRBh`8MNjfoV(jvj~C!VYGL#?1~Ys^#n(NT&!Zcas5FZ zHhR>p!|tqPGV>0$;PYAm?@Mh)Ij=Thds5Io5|;JOR$QI?OCi^re`f3Rr`;UPTS7=q z@(WZeV1r8UrI0ime7P=}Qr1peVer@DR@yK8`jjO4e7m2Qy({k+y{F(*FO*fM*Dw07 zGqJ0sy@5EMg)SMUV#`!)nTjnZ3RP5qGG$BW+oe5^timX9x%>{t2zlp<EUU(|Th>r9 zgGX=~gjDIeR{OmIw`wbz5f)Jf#2PX;8%Cd59`XhIa(Preb*QNu00cq}lP*0C4=Ek5 z>0-@oo3SQ`nPa6Vlb)bt!2~&i0_83$lr<Rk>~eA|T^C|y1LPa2Y@`VbW*22M;a=>f z1>8OptiZ)CWV@H#8z<P~vj&;l>4BQxDnp=}F8<n})Q^hJ%M}dk4((W>!V8^S+K!Kx zQCT}<d8lx(33mHC-AAVT$aEi>=oSZ+>bW*40iTfghV)brSWVQ;>V{dF8x^o*!3$nA zS>Z;`8x?XRg11cCb;`iKD@wVh82#L|m1kgSBnHyMiA7SIFKH^y&yU%fD(N@4QIUeG zJ0dMO2VY33c*wgKgYRTl>BXkHguY4V7d|1k^%0rRl$8(V0B`9F?=Gs4Ae_3=rdd~W zPN?cAONatn{)=(pJIg2+7kJ4Iz;-R~5=qy9gx$pC84&d~V2=bJd6dO!jPFkm5L3oK zW&D%pTv&_yMGV(fg{TRd6M?k?b(OV93_se9Z$f6-_I5<3Jb(>Mor~(3NB+!87^ZUZ zNuZHR#kU5wmA+P$N*QISan4~XBkxkMIdGrkv)&W;LD7rz9=pe5_b8o~e6He<KN`*3 z4Ysr94Y&=nqJ$a+FZhaFjX_^y*m1KbPEt4?k0HZ_HomDhy*PvBl^b#}pXCAOsgQUE z5&Ub>tgD*L1BC;{T2QW<M>7IEM4+6lf5`=OA(2UiL}O<}g#~waQFzr|H;<gM3e&oz zdf=4P_%_&3Pdxw1ekbj%L4z--5jV4<YP#T#SeJiMl`qoHi1I6)mIYmPyQHiInU{66 z$ASf{@5Y@Aw5%4*uwBi&#@h@>_c>0lt`?q^n=$pRDprS3DQ$Jg+v){H*TEQjJ*zUL zKT32Vh@D|pJ{<VYp8zDzn^pTMd~BGK>H>iEn}Ikg=K<s988yZ;3=1{u`>7ICI}!)u z0J7VnH;}x;hNjTV;W_~aZPXz>iJ_R`A=S{6fnU`}7>DAbO2Hb6C`MA8>jj>NvE_%$ z5;J~E9OP$%bi;5r%Mr%WSc1E8Nly)16JF^=iiU~=__dZxryA&~067&PPh$VMeC_)A z)oT}5u3dlr(#7?sum87aSFXYSw^7k%E1?tj0#(1B+!RaUYKCDGL5$f{X5TU)vH131 z`$Z4O(j-NIRs$r>`MPpPs*)BJQZHweq>!JVoot;Dd9R&z(XqElnR!6xYALvt!J#Us zY!Fq@YdE=D4>T{)_MDUh9+jKi?#>ejx1>EIZ^GC(GDaql@jQ6YY$5kSb0Z)$ys!jP z=oc6k9#SV`kOsg3t|eH0K+U^k!*82&J)fuu^RRB=f|eF;SM=G*y$l4F6uq+o?tFc> zx7W6J05uRA_$eAVSw>YrPCcayWS#SZ)Pl+cLE7Tg^D;`bYc9H_-g!2oRtH7|+{TT- znc(_CG_%NC4}?GP>QFH2Ggdt*TJd^g#)<O44^Ge}89IBCKDi;|l{vKoTB-k~10eY8 z2c7>cBN^ZK&$@LgP2x+g_LQjaVjk~9F<#Xdr(JW`bzFUsbIAg6>+EtBUz7=hau>3V zPeI@1)yO1>Nl!sH#p{q^DBr>H`ZFyJaIO<ZvEUF~`t=)D{VRCuGXy|b_lBffmhqM7 zu~TI2q%QWzPPIZFfCgu{!H|81G}#es33$7pnYx?R)zkPHUrx_tCG9kG6szJkxF?DY z`r1Pgo;d%+xy5|PA;{7``JKV&7lm^NTiU4;S~^Lpa{EEF6X=0I24}z}==8QUM##<* z?55p>f{S(FTCa|_OCi<sbqj`!>?Rf&fBfJ5bu)TpVPs1EYavwHpM5VJinWX;r!tD3 z#bO)pIAM*MJ4W!D*eRN|TNj!0#&xI~=8z7+A&w~%$tG?<uT?ZbirZn{I*w?B6#@`N zS4#Hz5iHAXnV9m#ZMIZq!WfRNl`@`%k=;5nzmJZ?$}pc6%j)$?2hJWX^k88{WQ;;@ z9>Ieecje&rCY}q48}yX!`~+m$GEvx;IxperE5~CEz=yN5^tNgN_rXO>%qOBy!BNU( zNAAjhLDu&qTXCeyIu$xO>PBqV7HuWX7~bzEgamOs6vvls2tlWDZ9!cY`wfBJk2^h- zfGPVQ9O0xLW8KBCvK9=XSG{8>b~UOoGbapzuni7N=E_(dhG@sw%Q)7Fo{6eHc#%aQ z60!wZMIf+>xtU=#w4If95sH2%_|JLyju?<>1)d6EfzWtFK^C$MA2n7mp;N-v&Se!F zU7rp%%mUxU>fDqj;9&-Xi59(HthCii2yud%ZrTM2;Oc@n>2fuIW$W_Q^S{&a5}354 zHhXT<S$DYAq%2i;wMcq&D^r0e;tl}}i43Ly6dM}55H=1Tg}vyUH-{D--~q3dZf{rR z7UoE<7`0;@s!+44jmp^R@sfI0EhPw5y*&|GF{`@rIEOYzD2;UJ(&6yF1|#titA=`C z&aypVp6Dwt-Nll<Zf{T_pD)<WC=PCuQl^7DWL$tVBUJly%pH+u2ZKMahX-{Eal0EA z2!-{vbbMY|&7nhqHuOGM3KPsjU?1F1+qgrd+XKntaTZC$4U9n6h223CX)G3*IJXF7 z^s=ax^gt;_IrRV@UP9rXu`M>_0oyr3kVzK0KxOb=C+%8P<Zi?~5km+dMaz@`OOXK| zzSCCy>;?Q&Iv?-}gs}`x6KZT=sU>YZVq5?s&T?zC;T9}=Qy6?1zqN&^nzYZ?o3sx- zG-#yOf;hobdEDJin=6bP2uVGND-$@VY=P7H$TdpkA7xw{W*d6+;mX!&B8;D~SUVam zhc2h3p)h9KN%m!T1#f;BuB!k&6w@`m`RsVE5(9>;J0;o3IJt;p|Bj1)O5}>FM&kHu z4nD)e$Fk;BKQ~k}n?u(AL&_@rb3(a=e@-OBc*B`Up5dPoyeq)PhN8R@F^9n!MrC6O ztix7{0!5pVPhd3q;X<oq7iHjy1&EEmTU24Bwdl68$)Q#ceSSQw%ft5!De);@VfgZ} zl@bs>bPq}iWz22y^AJ(~goFX;SyCR#n-doh&TUCqG~CH4%Y8JAQJzdjfsfgCCKFT8 zL1T!N0-v#$Ojvv|T?<bxNVu&Q9{4|R(I&7t{{cxhatTawj2zCOh91pMyqzVOVU{|J zaVD9-=#p1>#wTADy1_Lx`F!XmJ=`Y;NweQe0?8b<cI@naMhv0nue_qJ0^B!Njz8Wh z39>Y^U?{1=7L)E4qBCbwZ+;Y1SS%zFq?;wm<{eg0rxjE$Lz*#?DC~3}Mujh>Y5!V= zT>^e~S6niFmVdKoZYXSY3I_A6qpfACU`N@f3XRZN+o)fXozhz4VRsk$Y{;*sY<OCu zt{&iGmB5Duk1a@1j(yPOhg>+6LmHw6_ivfPcXuN$5(AS>jk3eYg!67IsadFiwCHAF z7v0IS?vMn7rBQw*8LIa_rQ`cipeDIavo6Ru8A!mu66IH-PR$1e=N9h}3`C|K_wI2h z?oBb5FE|Qw#5Qb3kR;RIjTmmi(iaJ4Y&#<R{_>s3+7d)ft|)1KmOBvJ>>j3>!wHt+ zu(;BPE_F(pVNAnf!uG_L1}gDZe|TRTwpZwlkM)GbX2#@5f}K1QAAI3)X8OOtUZoxO zD(&ccO}a^x{biDh<<1~aDd(XxX=ewfOrKOdkF#{Aqb#_nh_sYC7tIQmJ~QV;=~Hzm zj+?0@ZJA1HiQVT4UIKQ}7x-pEsp`?j#7#<%p2xaZ=K>lW+2jFOPVqi31dHv<(an?; zJ@lJ?S`VhI?Ez~$m|)08i)q~}?A=5kpiifykE$ALEecI*-I8jt_PnYNG*mUHvd~cm zO%w~hz)EgYgHT5YC7=gh?O;a-rlW+ZGH_)*rCPS^qmi%}aVH+#<YF}NA1ksE2%QO4 z(=$7O0t5fXRI3rZWhJC#Z`HWM+O_0nk{5|{BK%Wc8_Lv?@anO6JaLBDpr@<}vA}C) zp7Pn~ocSTR88EXMhnqbU-$<gOpC=Khj7XFQ)9bhs<+DE0t_AU>oWE3CVKng)vn55q z-z*=-RPxYb%WJ+(YVLWb&4!jaxQ6(3XM^i5cRFWv+BaM<3x|B&CSM)DS8RvLR5?A9 zzJlwvXgeRep&Q^MZsM5!#?=U<#j~g*q7vR3$SE~sS<Vfq{!<b64)t1cG)9oDf(MPV z=>Zz4W4d=C*Mjn?;?Ao!v$7}E9!z*<1?)AcZO3-7%L{%%f`z)xB<~H-#E>PoT#72h zhALv!SCAi8%Au^s7<iv0wcumL{%kl6Zbp=`nisa9g(?uXy*Cb+9Xe)gsJVv0FikFw zDD9*@`n6fjQ7{aM`*OOiZaGrCuX742+CobMnjOu)M-aVKJdP%KSF!6F>;At<(j~(I zAk22QlQzj`1w1mq3o3>s3BmQinoW?u48GgT$Z0HVTi{wL3=h(sL|3w2+5|tJxJMDe zm8Y!OFyM$DRksEz>d<Mj^`OI^1rY5e;`V+#D9~Wdz-bFvtCIn52H(N`-Guy&lvki= zWe2qdHoT0yS!u5_K7F1F$hh@Nwr9Ds%4f_e*shPHf1w>0a6GP_QuBp6OgSB3L8*Mt z8xb8~Jvf9A$(an$68>Z|`Y<<e{q=-gi@NZ6@hyU`IG1c|65b^dE0%hW3tUafiMST) zO^6M#)*yPjM;Lz?Y)wT(_;+V)g`#0ufVO<i(#F0mV0oFYfFHqbwjY57J|$(66*@aq zC~4PC=q+38;N9Xeq7J53-ItLpy#v57tklX>Dd0j7=R(p!HJ$q5UveJ$(aI*c=lH!L zOYBwvP6l`QeW6UogwhrZ1{=oCCxtMr?E!3H=qC${pG9%12~?%cT_^@_D{xP)BF4i0 z188i<?PkA?h{S!-D7zFB7%IRHF~xKRQEgwuO_QWcVQ?WrrS0?}X<b%fqj<PN!7F6y zZhTest~^f*?%^FT1U%Id6a{odD*xd4f_uP!0ySXXnNgL0(Nmd~RFFKyA-V(2bq_El z26W(2cl>8DCJL`M(A5UA+Kb~<ll;IM-%LrXPSstR=fNK9Oi9~p^^BGbKTm23W4&q6 z-gLu8HVtl@1LqyYx-kc4kA3K4xU&hf@HgHTTM`?h;YwryjCvbVT6{Ulv(?U3z;M;k zjz~=S!r!SynOa0OAXt^t_^>iO2+iVRK#;U|0^hP>Oe4*5s|EOlLv{kCVst!*R%E_3 z){+dO$HS${Fm$7<s(5w0G6ltBb!9y(93QWso%M&sW;GuDUR+ldrfz6RS61J*G1Tl5 z4ZI{W-F?iq@v}QU%#nq~`Otf|Hh<ItzG=!m2_QNfCA2Lh$4=HsDj0PK3OdV)QhtuC zOF?<<syb-4{K6JsC_zYx`KCOh8Wba8V)B~cya^Py+u44#LS^)9^C~1th*M&&i`94m zyRlHnR!2!<hO-n^Lz|hP-Ci%-L&?0Cb$Q}}hOWW4%St=hCXEY5LaB^FVooe&PI+Sl zRCbR&lSEfmwHT$Uo(3t$>bB`s9f(a}mOatDfGEv|F9!UyE?U*0lI?iEEIlOQS8y~r zHQ*A_%d)80i!n_QkFgOHkn(u{vb`s$2$G(dg$p_yU_`s80RYrz*xiDl7h4H>vw+DD zdcNQsm?qUNMOuhbRoyLlD<#P1N`G%F$?MN7zi|Dz<;%~mL^3{Hyj6I`IXhGr5H4VO zi8MB3bLj-2<<6-tZ>t((wMbsRnesSkB*mj`f;78<KV|t%@UG!J58R;6CzPDC5p8Uq zEZ`luu8To|kPm$>;6#AYk&^B0w3#O8l^J(X4W*A%vSKU_3K<0D0au(XS7lJySGXBH zu%9|$9Q|}<We^iEFP^poxo)00g~3JYV!C&!bzlmcXO?syGApAFXuGl$?yu%OZH;Mt z!@veD236k>)mTSdB?b$N#Q@E2);Qcb%T`TQ1C)wtsu@v*wKx|tZ!<Bn^qiL(4oRCK zX1&CZ)a$DkS4|)T5}Fj&$MAXgW>R~jdei2Hk3lUfcyXJ3r*c{zloUJRs1s)kds<2X zVo0)zyX%2r5OMyriTN?z`fd?RvQoLyf^l$i69$LZcsQ@&hjd^zd)5&D8*8x!@nI_B zG!y}+X4Cx;9UzYilo?|}ooY^S80Iz%x?{~<OH{|2dD7oQM?zrBqY<4t#eO;kIhSe# zN9I*9qinDdd&OM(Yj{K(fUJ_!sBf;|Q=WSsl;T{g#U~bJFKhOtcM|%cIF#*&Dd3j3 zD{W8U*EZdy^XAT3kTexRTH^OO(`Zj&5t_I$CT+qA3r!Z&_1S1Mia<3JMkw@TS+gtz zKd0~&AP2U4X&3FU#Ni5pAc{X{ds)A^8)2ZgNNn`2c9N^T8&7lqS*3a9tcJa@1$Unx zA;DEnyQjshQ>|bhTG@W5QX@>kw3EvJVn|oRtHIVfMzupDCO?Z&4|8!)L3~8<%`9z2 z+YoC|^sqe;Ya%8i1MzY@h2RtlTAc*8vc^ybA;8?;6R_$eW-$UE2rZnkGg!LVlcE#4 zk*a=NqY0~1qTCnkpD(JB>*93|+kAgw5f5VKfi}NBtVNH_N_%Zbs2JTf2qbC^+S>C? zryO}rmBu|=7q_<rR!r|v24Fz6EZOIIC*0S89sX$uuf1qa0rnVjtxEMoAasCx4g=Ki zI!uOPqcsM_I_{zn7MTH?c|5nt4A?X_lKecy4G=F)SzkV76<7xubbA|iq_@l2zGz4a zA5Ob77RsE8&qAt-l8R^Cid5B>+{Ic?I$HQv9>0(t8bx~XTYNcl#t(M|wNod^o{GMr zF{T9Ua+zLMVL0P;oSD?y&vxXfD_E;fEV?ND;urb*bBhk<Y)Fuv!G41KgGG4=LigD( z4WFf`>e7SgnGHv@<(9EQ2V&a6rxAqB(HMqoweG-XKfNQH)?wSLTAZPRXC~|Tj+ZgE zE_$5J*5Y+WgF8yU3rZ5VfUjuQ+QZV*f}^)Jn#BzX&du{;qyw&J)nHWX`r=e7t=Pq> zex<cntO2(S*B%W$D1|Za3Y<x)xLH0A;h;+HWCTkMpOHY7|0wvM#D87gzmWI37pRw7 z<0vHMZnRk9M7qG$tG*tX0gV<*;`|;eEZ8n4!Kx5+SJX!{4ew}D)`{|KL)J$4?$9a_ zx<CvDxm&X8<;)r<bkR5th-j?)$P!5Y-JYLS)Lc~#2cj<#8{wo0hr=he^xk}Us|S%i z74B3G#=_#jV9}jQd6z!C7o_R7d$K>|c72R4NhP;)6CXt9@HIaw*qU>H3`=`OS~|CH zm9B0u0wf#123j85I&{zg7$5-^z{wdC!)S>XglRyGz0EY~z%=Q=H0i*2S*y%lWJ{bU z+hE-Zje_|ZA`Y|JVVHZIcCk7-V%`Li3e14}G>RoxuU%Tbc4_^^`g51oo?X6Nzr6gS zzzAMo=Yy(R(-_Gdug#@uXIBt+g304^FwM{k9`R)>iMGHdnC3P^gaB|rkH5iaV<A~7 zcw`aR!a&Xl6AkE+1JN^>airY6tks9hWyriG+SwBHAxNcNWTqSer-FsxK&g<=UJ_kJ z3F4*t$iZH{w))hym9@1?&#hcn1IN>hyqHDdQNfVshTBRr2<L-!E6zJ{Rv}D}6C7F$ znjnhFL7uUVrsF#aKY0c(f57%b9b%}$+F)-h!xWUrrGe2`xK54+GUcsP-a6&2WARp# z6L8~M@Sxai;E(FuY5Xg1dj%HOppOIl%6EhALh0}cwdTjEbsNjT2L-5$saQ$a^M(Of z(`8!v^#!qB1;hCgzM{(v>iK`cco@-bVvmia327>dvk;J^eRd6-KI9C>s<gWGvS1>w zEa!PVsGoz`?(Dom51%-UGC}b8f9`fxcQB$QDM}&sfmwIaYEW=!)fG4)pWRK`Vw%np zatxrCtnOI-5FOY6cM-tkCKXes5@)Q~02G76j7rGpPD8X<7B&WZtXk5{8wL(GE!d~3 z_=T}BZhNyP-W-CegL(BqVe6K<$ohFRpl({j=T*C~ne$76H%2P#ooTa5-~#QmVZ)R+ ze&TD@1{KVF09ESX_ko*^f~>Kz;OO(9_`)>z>~yD@?ldRLJqxQ$EW@^7u!zFF^Z<fZ z1BV4(CRb=3%dQI%IPO5HmZ?>fycTAe3P*d8FAS5`zC>jbRGq#=P$1+pM9?81pb{l# zNFX|0{v}+z*$#R<_QcceNWB8ZRj*%2&Pu%14B#2;KXbmOfx^dQ>v~`5p~o2TmuTZQ zu$mfO_0)8uf{E7a^-T>)ejqJ_shh$aL@@vd+f<d*3y%i`w*gQjaQrUe)<*+|#ai_K z903~yEAB521GHBgZjJN+jcVX$>5H_zS2GiL`LV&Y8i*9dx@2}DUSo;%x?aFKu7W3y z6G(}G5o*a<?-@+>AQp6A)L^Y()`FVXJyE<STk<15sS<Epk>%?&r?Q4)i>Fc>se1el zsV4Wjr{iK*yfUHgj3p+guN7M0=u---yQN4#oK3rN0eeymGD#PbIw!fA8JKU(=uE6~ z53PDz$#&XFKrz<>3IHn#UBM%1DJsE&$6ANf3+l=?v%F<>I%oyU>!i#a!mTE$N1Pw8 zu3UNQ(v_#6(|moSry1l3T2e<<c*lv0rXlsM(R#{tRdm2}GJ(&MM}Zq@Ytt^%2oo0G z@j%P24V%?>6}1elS=bB9@(m+VShX;eSBj3gC9%ic-T>>Xx&$2z114!#^Swnf)^NzT zEbx{@Dcm3XfWo2V{?tgh-&y+uCQ-4_PDZ7oqE+b>BT}G;p_HNRfySaEtMs2PA(5b> zcGN#ia>BH3)TnGP$gnotNd~WcxlYiNhh9`R&CrJJry5ScGUHE{T+-qXn^deBG#ta7 z%fsohdwT4i9=lJd;Ju7DD0H84HP3FQP=!vATY1{rkr9TvY4b*ubedu#Z%J3??L6C~ z!*h&xHjlzJ71UQNk1l7-<y+m-X+9}A&S4~PcAgjbS)mr>&5$ahqDoL>+YpZk(^=(! zyG!DLD^{sk5pcr>?P50pJ+Q>e+q@R<qt?M~1)?BfOIyh{umatbJnCy!8A4sPY%n#} zq-L#3idt^BuL$N>Sh`q0ikj^3A_Emjo`}}ufCt<$aF*bg8dhCjo2I==p+^U6scg~= zc*=4Lp`mmGGHuv4+LVeFCwoa3m*QomXT8?#A&ODs0->YZf^EbQQYs$Qy&XV;tnbBL zI_s+k0uNW%rePAwA!31t3qEv(LMz!j+~ml0`jW!Ny1EU(CBP?4<Ta-wtnBcPs1!Py z*fBgxCus_X5$6MYF3!son86hc*_XxX3Qb?64=iZE@p3r<)s<S<r2V4UEL}>N+=>Q? zGw+u=H?teCr0C-0KS?SV9BeOXrExD2oyNHY*Mxmw9!Z{Ox#OAH3a;|SQU(yhe-MU0 z8{vttVkbnwB$Ku1Mlx_G<tZ^Aiuv87%3TU41$stYVq(>arxtbpyN$jyBm@kFVPa}x zG?kL3QquTR60F**X+k_ttW>(nTr-!2p7*@Grs$@L@t+5#F1i9Zmf;=XT9F^Q=;p#= zI=b0<(HH$+?!tvRkXwzS(N${VV7b%Bb>^kwkZyQ#E2rZUG=vcltSb+z8W=Xkrdz4l zL6W$Iher%M8O2T5h;;5{gpw!Qg5e2jN1AT#3iKz;f(O}dJG4W|>)4PCWgi*01aq${ z_D!M%{5oB;-&|922fj2QZ0ulZ@8&^9;U^{gn!d!ryW-OU0M})t7z_~t(Jj;z`YI@I z^7j&Ph+W0v*0DR;UmZyX7liDksDQz#VA`cs-Gzr9u000WUIfXdaT^fq6RytD5Hec# ztUjpwt}Xjen-BJ(UHBrQKgk=?P;W23ArVD8<C9qqvH&<YwLFSA^oJP4MQIiXH-|`x z#K>I?l|zIOVf8R&&;ZY=-MZegT-n1;Y6!Rqb}*v(a{n}>2Xigf?H8W6I(+DUu*Jmd zYU+kGrJmyDp(<EaBdtZj;)PZ<3BnK)bu$qtTZX7X6m2W5MOu|yah$Nq?iMTs(Pgs0 zu;<lZev*e=#bh~d+&4cHSYWDy^GwNiAht^RinW@qltlg%J$~}i3(u^Go%u-!Vz=Z* z9Y8#Xhy!N=uXV~Pjo)G7qd?SH37!aq3N1K`$NZFumBmP#Nk`_iyh=YnPQ5DjjTC!O z(uzYVMt1d|*LSm`$K0i4o4ScgDbtso{DMGVLBmp&D*+jsO8YFr<@qCaaM*`YUZ6Nn zP$N<S(AcCA&=vVYbJU8iQ~`@R2(-byxfC(zQgkq1McXWbQP=MlO6sUaty(%pTc|pZ zf{O1cAWym-y@WsO9qdLO3&?AkmrQlPx2j~x8=DBe_(f}ISYjanosG+b@4hQ*!A1jb zilw}wxl?;NoBLaBut>L^lx{Vw0GK2bVaP4fuN&Gu#FW7JTRQZ!=RYFm;?s}5_u$>1 z9)11iM<2X;^ugEfefUfB8mJrg$BoLDX8!_Aqu~$VO>+#3Cklvtr1GD?WWUtFKmA{K zj=%e>PygX{2H6C^cDlq}x&CU_26z0USC8I*>%lw!aCGNq$8Z1U-iLQs_>aA`Bdg8U z99i}for_}h*%G2?tb%r%Fyk+Y>T6<xaAhK#m+ZDF5j>D9>L}p!wv{Ngla;*PgB0k| zr?0*tFfFih{KjiXZ{9ij%3Jr}`RLQv-eP!=``9W5NoYiK4lX|Zm+#%X^KNN%oOr9Q zixH5%#o0?pEIZ8kfQAT_+v2=uh0wwRII2~89VDk$-#Grp|2h8o|2n$+*6;4Vsi^B; zKmG16#m}R+-V+7r{#N|l>qCeS9tqUB?`_B&d{%*@Z+`dQhhI7V$=##(-@gC%|8oC- zynpn=PjrhFqRGnd?!JEX&A&VP<ZHjX`$qIiR1v4q7#@E5kN<rCo%h5;3uc}b2{|Ib zm6U}ocZ;AU=wh?z0aYNUps3EElY~A0R+49RPTry%kOxHJTkG!y)~o<Xp$`L{fgpr7 zOXy`?`$0~Acl;5>I<%ijXFu73CL8Is=Qj-V?gf~pvFBM2VxGqo;;EVDO+KaACoJL5 zCSpV&BN*OC0Iy_y!GJ1wUWLep@Qs*~leo1jBOo?(6(1G`EOd6oO48a%D)s33EWaT` zP~-F|c%Y(?5N&{QJKgE5XAK3`Ywki@kdzRxK*0fu9KkJm`$^J~D-3i9WR(a#sC(5} zhXQ@U>aq*?n1*3P*%Kmd3K*FejM?wtWGw){3$`N8IdC4NrsiCaT8P)@A^-!M$4aKb z-QR(!aJHT?LFr2#WYX&44t4yK*Y5xFljHBacKoAX0B5-S*3tKW4h-j8Z?01yp}H<D z9^d=$o5ydzcl_p8j_!W-_?Q3t=!ZXk@c!R;@QHcd`}nu_-~Ksmkg9=S6R&yj-p?Mq z^Og0L7uMI-U%b5H0Z`*;L`NV0PlaT$Yo(n|k^}Sq(HqC#ejWRB2a$2_<8R*o_}!z| z-#_}%-yZ$;Z|?u#YY%??kN1D}(Y+7<9-bgStKR?n_wN7QuLRE%wT}Mw>qkHR4!t<~ zhc{1)8prRweeaWR-2c~)j=%jr6S-|3O4^+k5xXUJ4W<0HnU34zui~S--#&Wp?)^Kz zz4zfykKg_O#KtGOdH)x`xcA|!;^(K|{;k-~?Mf6YMbOiiCV+P5&7sid1K^6iEW@~y z0loxui2)A0vC59%4W^jdZ+%TrQQ)*HIqY2EmD2QO?X=}Emc#Iz>2M2PZ&;XoJ4cAS zsc10}o3BUDiRCR*#T1Y@sT<=tuND=}m@{ylBS3oU>5`S<Ef8^$;>c}H$Q>P`vzlqS zq)Cch1#x<HCA(!^dL;(YLX>Rwx6<ZnzbA2P{P)l#ysFRFHP_mD>g6}R-k4r*oWS?u zp*-cb#1@^zox?~RmwBEdQ3AKhM;xPxzK&+66w=jx+HztYaaeZN@(Cit@<?>3yVDO? zA(?l{Qg?cKId1<-=dxfhRI3Dw7b^N+hr?#vYwj*17P#J3?kC+;h)l+OlM8)tzubrb zOXeAAY0*}XpF1&7Vyj_wE|u3YOmk;cg+_&ywrtHO8Ixj&AvI9}ot(USHYun9Oh|cW zXU7PXvbwA-6RWl&T2%AvakRUUOl+c*A3x7c$x^tLW2*JDGDp-|iY0Ow1-(TNrF(yP zd6*$t@5ylU&>6);t+2qv@x&FTBQ+OyRtwgW+e>BCoJAdApGD)*Np)3lozoa9&vM>8 zP*Dw=Sy5HstmQgTC4?Ygl`v&8wfM{+q9u1}ngGreeNf88i_wPY#G?{M4cb#bj9XBc zULXLu%;}Yp3aJoBcd-NAOtXGrWEgU>A`ZvIBm-y(Qza0ns&mOv2a&5fKRX*tEoKhM z>YF2ay3lZXAo7?Xiv+|eD@%gT*r1?QY8tS&D1^w>MBSU<Q8KOunF3KFG(6|YKXASv z(qslm^nfnUi_*1)dCd=9Vis*<^XisJGASAhR;f4D6^r9rY~}W&Twh#8E6x=2j13ri zX%A4TO%<=opVrQz(tC;b>*A|^hr0|!(51G$#|pOh0sxxcvrAjt)b!9zd~Kbat1Izy zHP3n(T|eRPdKVtR-hyDQVhpxt10~zpPV<6N$R-Tm7Fx-~ZZ~bWmV4D%jvjO0;gp*5 zO05$nty1&uJnDTxZPfZAh=DUXx$HbSWTydiW3^qLxXCGMXHfK0)XJi)7+o2dD0&7~ z;Aj-^eOEbG0;QeICGIRK+E#fURgN6iIsXdDYG&N-c<yKp6m<20qp0tW#*<JE&4;X$ zE{>6MG4KXHEDM%j;Lv__=;m~4xfvPNSG?Kt&b@A=IB%>v@?0S2sWWt*E%3yHA0v6! ziBX}?=An^$^q@!`sKRI<JjxJwB|crC(@a5uoVF6>0fNIN<z`1-a}4mT+)ld~=vung z&>`F3#`%ntER6s{Lh}l#EA;lx_J3_`owZxk-G`Z1*0+<8s-7k=?V=eFhxa{*cM8fV z!<NXR)eR)eEJ|@pVBtwQ%QI0$O$P#bwBQpMDeQ*-HA7AJ7F?OSx^DPT)-2XZ7a0!Q z7@n`pj+@F&2I#BJ(Ca78X%<Ro_St0Hl06$I;}*~<1urPf;UgMxwnJ+bHkl(&Zznmq zjXXUME2|e=g-x0s@nwBUBX29z>NFmFs4qgbPJ2LV7PFdB{b?#OMW~rXP<t7L>0)$> z?ZhxerQDCHLAeZgof@EwV@wRQ+HnL#2tju-+wZJ)v?NiU5P=^-jKBx%YORM^%zGJr z2_})|hayKnNX;90Em6gAKz8b`ifzhZ2@kkr!6%eDAU6rQsQf(wK3cx4r#l_c(=z)G zTCkV!`9eum5IUWz_{gs6SJwLz>NiCP8JURF5zZm(N91MmWCo#)#0Cb*BeY9YT=c{) zAyz~X{A!M2H3W1xV0KCI2=(^!OaKn-C<=YSle_y-o1cKK3s<=hen|y=_zl?$>as(H z5%pmb8KjDJ>6Mv2xJH2da~OVwxg@}xB>_CsEkbB$uDE0W3~l)E8A^6^19vjIbL#$w znc&~U{IRbM`}}3+H&p6^!@%j7Gvl)JvCWw66Y_tk<1Y<Hm6tP%4_ToT3Xqt?hd1!b zD>7Fidssg>9ezHrv;2&`!pc~FwCDgEXQ`KMJu<e5XI!cj&wPfJ`En;wS_U>Om4k|p z9E4^{KZSvb?8UHQ;jU&?9BkGe^1W$Gemz`p2LgCi;w>A<fjtIp;5)Ei4Q$9jCH`_# zb@_=(&xCiVlB~<YG-Dg#1J#gbV3d9{bjk1FVPrFsBH)j^_ZZ?raevZbapq}=J_u@5 z*S%x~<?zsA40Y9fTGdlF@mZdt>Cp(9G!;N;JI8~jJB(^r&T1@B@pv0<3d8EzmcTv$ z!28H{(PS-Hb;9;NKqo%DlOel%bT_jq^Bg<8Qm^d88bf)_#U-0FhtRO#lS!|+3$|bO ze&$;7K+%ipGPZ#`uU$ea8lZbNvBAmDDI52D*#$EDVK{RN^^4>ZS}RI>ol|^&s*@oj zU(cn(gbpl&37+3Cd|dO3KL_r_&A1(?c9Xv<%M4yR_c|}UCnp(tm}L7rWhUOF0o6oV zvboAyhD4mItn!!}?<srvm31YsHJ=KszF>F21d>P$?O`xT<V_W|Tx+6CGoV()1;v%| zGG%<2=q8@MLw|6(RUaXW2Pu0v(oJ1fKe=<GgDkbsOHefuur0)*tsaNQagHzbq+xJ7 zDi6voS#wbai?gmd6x7-J<MuXc;|3MIUcH9=%g{=vfwUu7Q`#%4>MpHD7mYL|4A(Cn zi}_6UwTtj`;)m$S1+bhc%dBlV1-mZ5p3@nf%&=j3)rj9M$1@d5iI-%*6i#%bbmLkQ zonL!#qQOlq>c)e17TaN<U~c4djKn6t*HQ0{bA(ryY3sCiM788!IaT(E$1v+rMYDG6 zB6G^S4pp5ZI}#t{wI|VG7jg7ja`>K#(E*$mPlB@V8sZX|iR_G1Y{a$Nu=}*JuL?)q z5o&aGNN;ta4N-g+rO&Lh^LchzH9ec7tg{RxzilUOV5#2M3m=FKv6XIbS0#d73kSG3 zFE?M}eNg9nb*Jy!6a<tEjQ8t1tFl`q_)zjLP8+TwSVlk$UyG`b>MNROT>zZ`8zkxo zN~^*LOH7~uIt;zAmZthx6A6hFRY$?Sv<Z||so9if6NCETycPkov6==Mu4obz^K$lJ z91<a-u3BF9kVhF9$ffZJMiQTI=q5pwD(p@TCkDBuPdT>&e|&VIJLXxe5VFG<GSnj$ zxFnxYlP9A(7s_bTflrt+70HZEbUCJro}4w6Qd$XvqCv^l#_nBmkTm-}E}+53PaSJ* zjso#K$UyW2&)p6m5C&{|Un=K)QDeNNz_yjpwNxRsWdSATtcwO6I00Pr=N6<ok?b;z z6$V~i7122-qQaRoMR5LxbfEC&!CeFxb4*!0?!^aHuo4zP&>5t98uwNB{WN;@2|?X5 zskERV1zb=@5Q`KRg;R%mS0kEn`$k0t;j401d=UMx1ObqDQhgiLqFntjXw|Qk<8Uo< z`(>-eI3d}sp5}cYZKUB@hF}0X0ehyG?y*~5(Sc0%2`=F9&KHwwL2v6|nBR#0(_jA8 zUp*c|oIRGR(?eHUmr^d1SeZk4T6D7_an1!^loHhxFK3#8=};5Frhp;8xiuQ$-g%^H z=qqecsR<%@MTE&pUG)vVfFz}EHcR+Ga#GhN5uZ0AS8Ib!Gv9MIZoi;0iyB&tdAaMa z-jdc|eYqD7c&oXV++F6VjoSn>WXfemI~+i@saML_7!QpzIwNESH5$>2Innye-0A4i za4uALln&W6x9nl7h(EHg9(h|SFu?5v{ixI<8I)KCZN&QxPT+VEVrS|piPJL90W#IB zdW;POC42oYxEG;$J9cqnb~Bn`@CAEZ2$Ub_@^fzaL7+T2#%lxlzs`GAG-en7oFPm0 z*1%=!I$$MMJ-5gkL3vcZdBT1(>vLFSW78~pQ~$cDf8C_v17(gftyUrbugK5227<5< z`tTVfB{L!?9QeL4S7onw6esi8Voc_-JeP?kEA$O?l@+$g4O^D1S!(s1hZOI0pyJaj zkH9OoEmN>$$0>7`?-`U`RQko4Zu2q94u#{)DHh4nLEJucSfNtw;n`V5HVz21xA(-7 zTIZo0#p$x>Z_y(@QGeY}av{0Mp1C%t@EM%~m47Iui!Qgp=5H_Q?Pjgbp~w5-RRbgC zAEm{0ya|UD)h?3d#Y;5{;l~qf;QRy8I>_u&&;MTB4VN9q>oL@=kdJ7|nOuETXE0IY zre`#%ZDcW4@o<#C@=7q@j{1}-YWU9*$sjOnaw{D{5GItL`W7std{GrN^~8}cxwv6F zW=`desk||jHzvx6eibfwSbBzT+)jGEBoO(xj^&pPqIIstf&#<K`M|j#HyZ>iA!6;6 z0+L4<?JlaK_iHyzaO>{i32`^eTc_(J3PDk2rxKEWG#~ca6WAo7WfDar;OD|JkwD(k z0s^TwNW>VnINeV29HLbgf|L_ql8{4OgSR1Rf}{<B8{+4bG|)-V=V|J=2tuPcKY&qN z;@|i(1K&@sUw3-Ue+G5xyziKwkb`qC{Ct(NH#j+CHek-!d5p6m8RCaactAkQ?Y1u= zgo5v?N@Am?2?|c*2#Le4Fp2yiQ%YPy#ek9=L_bkzDw7_luNzbkqz3U|H1Z`{+@2AX z($1+&1=K^k<=~BtjFm^RZMCzL3Oz;40TYC(F4U3F$TSBs2Zc<c5bhGi7QD#W1nXol z<&^rU3kutShE>-laK?vvR42}3aHh|tD_aKi$PQ_nz!1T}J)?^@Auyf_j-pi1)iH;! zCOOJ4C=G;4x@P7;cb28Oq`Gl04?_be?Ixb5<#ad>Mur}xp<}sBEzsjEcWm`Zv0Yvz zzB%H!#Kw(B0kzNsVEH{d6ICGAaygS8am-Rp$SbJH4_RQmz!JMC>MK8c2FQEpxrpDD z4qa{dcY?FvIAk&oO#`uPSu41dWcX?fufsnKl-gW_{hAED*3XdN@bv68J$p^hUK5|a zXbV;5`Z7W)56KCnP=GE8XPN{jr<hV3Ut7L-Y5B?*=G2YP)l1RYXznj(aan2^FK)fu zhrAlV7uV8OB2HjMuf}hHTdDX5ELG90fWDGN0e!!mY|C%!Sy!At#PM%8DS9txM%Bg) z>RH}nk|pGJ3#Iyq>w{<}+N!GyEGDb!%3|?t_rPtrino$AxTX}{Oxj>DBzkMX)m9rX zgNzR47sS_dP)B?Tq_+UJ3q9U=c~kV|48e__Py-ZC3<83@cuSn(Zj3VBU@a_C6AOMf zUKY(_m;FYBg!r=P${dX(_@0`8LeR_;yh;6?;ypW>zObE&!1=S?>dO+*U`Z0%GCUJq zIr<<X6i`TTFcZxMKp%!DpX$i+<<m`bOiA^kmuF^1%=L_H|D2lZ&tiIKhEMP8kg4O- z2KeDqN599OzLHKmUwv#-E~XbGQ%DO~b(aGS!n;QY=i7@g?iWDHPji2$OAMUbU5LmI zBb%{(WlYs881k<E%<>D@pIg5C?24||5)l0gjcC?=q$>0sMSo*>w;%xfXV|NSA-xI^ zwMy9oUJ$@kDYCK5*Avdxqe(p7B=4_f8Ppq1<uG`8kYYgAN;`d5_Y<^1Ljr$M?KAOK zp<9GS)%2_ikU+5Oav<2b#AO?%*t}hCKBR?F#8#EPITt~60IDB2{VXvvh;n(lYGCLu z^I+=|eo+o+Mj`1@`6XO50HNVy^#HKy9gxX*T+=S-5C|tvEVAVooNT?`fKY2h<YAiC zXl4ssup&3^)}i0EQ^CHS8^3S;uI!n+WXt+>CfKh4qehf23Y4nt{sow(G<Pc7JPp5; z*tx(L#RXwq@L-aBH;@$WENAns-*uBTAq0g+_S|L>Lo#@AeqiES0xyt(|JC;6@(V1( z{6|Yom^hVcVbDdTYC*f9Z_I6i*%L@ZuD+7bQ3#+lskDWgUH}KeX>g3O)K)+12ba_0 zWs2F+YS9dnEyK2>YNAc6XY_a4q>VtMdSJvz)kUv6O4tD!K>>-1&2SOa27Dqq7yyfC z4m?aKLZf_Fvh)PjvqZk5YRf-L)SyZ=8I#|YGZyT^EiI2=>VZJ%L;JJ?Ic#M58XgJ4 zlWE31?^qFTfjfm3Eb_0=3#hP{32GKET;aj3ugcAo6b#P1JwhWd1c)Imu1a=r0W++) z=z+=eC?p^y96B`SK@sct9Bf~nS2sfjS+u8}s(x5=2ZNB@Jy6Ynq&a^C;}fribZud8 zo{rZcJ@pgxNb@uAY_6eMv^qD*@!SkRwd`vEBoeJgW+rMyPa8YFKLmYR0cU(1E{vpP zG`by=dYxuXpcE6kMn7{}HZOwPV_L|~bfvY!LU{VyEGF9Vw8-SB5u;-~PJ;mfzW>*f zX0MJ(M*`xp3Ci(&pafx!buK80(Q>8$8{s9aK2B@6G3t$eev|B*`RG-WpmG5kJ~Nus zoa&T+XtWU%e9A&|h5M}=JY%Ro*!ZZPkBNZMV;D2i*h5X0suw~yOZfEfkMM&N_9Ji} zahpCDUP~p>19eG?($r}}=rf4p&0f{q;i-Vn^=-j;f|vJZ(n^>@=`6a8LA!xj9U9TT zkF|h|PjZYWx%(vkIWrk=MOnn-uTb&OM$(xRu0Z$<BkJKDdpI^?kKhl6hXjVKEB-s8 z+2;}`Xw#$Y!0vShm%(tQhYe@Whcm~zJg{XD`NuxIl*u&;K{B&0osFUcvc|)bBr{Pd zdC?-<wN(Hzx8edz4qylGE3vpb>49m?EIf|H=boaPJQ<I6eUi~RFhbQe%22szSwT}y z5|gX-THH!w(8W}p{!}YoO4kaSYRrNrP+hZWacvwEuQ4Mi+7-r*&uHOo{CH-PhMpBm z&RjWm)FmTWe|o8>bt_j*iWE1plTHspK^Scd3MWUhDH|G!P9^C#5=a>thM~7q4}{Ei z8LlR`5U`8Pi$rE8|H53=1wu(T!Ei?}2p1j`TmrpdZ@r$tw(V7UGVAoJKCzrIuAEo> za-w_b#9X4onpM-O6~dK#I|M(x*!<A8=Sv%(AKv=B)4Z#)K-Ng<qR$0*#0dfuoiO<k z2QfVTkuEhN1(GTU^56>#{<9a|*$AC=RbgCA7U<@I+0!hq)#iY`7zDfH?*vcEoUN30 zS_U&CeiSm#6YOP}gN<qjgnN>Z&x6@f+S!38&Mgqh%dp~VW~id|Ue;w9I|EFbb{vWT zsHpd<wady}#?am1+YW8PHc5Kr<X-L3B2H|{B&hc<I&<N^$H7LHK%ica(3$}X4#4E3 zY9nGvlYMTeJ)cE7Cm&=M)Ej=E7R7>Y&w~>{A2`ql2;HVajyo}eGZ8vRKURXh!JKlM ze$0a+Z@X9r%PzJAee2ZYegVYgywx3xi@|}$NV-7_K1A5O)!&nVICu<tF%Pc);4^+M zniVLY0ndMQ{I^(ZX*U4Izz;v7FgEFd7d%f;b17$SD{01PB!>wbkbtvo%&rY~Q&|^9 zixLHsns*vo`5A1<K4Fe=YHO4q{XR>*m7?3nV2aEP*-}%(mZHW`04Uf<@$OQ9kEI>m z*f^N02}*+q&<A3R9n9mm$HceCHeGFir(58B$67pot4k$W;NjCKrKB+GmJnZ0AigT7 z9EaFK6;NQ8S%5cq6)XrfsID7aCLr3-VZ_;&a&<y+<w1Jk<;sG1kb=BbHMHfxdm(?p zuto)ZSwrvv%P4Syi;)n#&>S&3)Y8uxZHm}1PlHAevt@Yn&U7-qu7_5x0f-jeQ5-aA z)i_r41A6?T^Vp<Lw02@QQz#yK8(PSaSHEsZ;zUR0m9nt+e%>xHA4pApRRwj|-vi-* zWf<VHh`~4QXu;x`+iJBFH-C~7iktB4CQQPW+OQXHFch<7J`OJm6_qQvVf_jfBre+E zKb&IhP}Sf+0w{xdGcy<{KrfvaASX8PUq-doRwtD+^3V_Fzl1Zcnn6sC79QMNGkLRU zs6BM2$V20va@G(ggmR;T!c93AlGo)G`~K{+ze!&(YOjT<=fKo+VCp$Aw&#G1A$@87 z3y(8v|3^`UU|a!)I|7JR4^dt4C~PTrdFphjJccmcxZ0hAv%^KsX9ra5DrBK9AR?i| z%J7P<5jk-S!@H^z%+O<T+S*6P#4J@>lb`hPNY$T_tM!B-#csy!esVG0QwrX_xGR1o z7&o|D!MC2K?xvhH(rA=Vp{J>kR|A1~%1X81%BU)Um{)<`RD4yM*vXdjJRVf3L?EQM zg1IqUlRq}YqNoLqdKj?iP-EJlZ#DU3bC^`VFRx4;g`q+$-2>MgB6X2NR5}ETlm)$* zoDQ5Ky_IkSDB1X~W$dS?@W5}72D~x5S&QTa_8Cnh^#d2cpcDWc<7BHG!oVepSswz; zBX$9Y<M@iGu%+Qzb51P>Jw|D|7|r;R!1e2jxtsBT2)rr~p{t&e^~FL=TjXopM~yr` zteEQ`wVFAjz-J!Wo}yT?@}&^R>=w?P;p7`YN?&sfh;Px~(@i^9JK~k{qMx-v^gC%! zUx1liq6(m2dBy3^X}PsHw4{SVhFFSv83-+}G+;r_BHww;sOY8bHt6NQoaEUWA|OBw zb{VH_fsl>adL8WgHf`XTeSu%i!O&OmVGy-cp6-fg-8L>!`o(4}FjHiEiAsM?_X|Qi zDfZ%a8+UereMNr`ijqQ8J#O=%jYEv^G8mlX>jKa-hQ??8sp@0!61@mQIi$(f1J<b@ z?b8A{2EkK+{E)6V4rHz-LlLtef+|4n>1-}dgQ<|=6H^#d0kp(zp2vPIFJ%IvmX`@d zeJv|m7SR@8mPNIN4@9=jL!oW)Wn+j|S$&pZ2*VVYp$NI~QCmSne+%dlC-DumA80Sw zqp#mRdiOVY#JTtJ-yVPOJ;NyAhmic^=>8ABcKqF69liSf<2Sx39zOW(_m6+_>)+k| zC!e5nUG#f`%3;-VtZwO$sfx0*cszDENr`4~iSyBhB>w18_B0UFoBu3R^2U7a3%+dg z<54K3Wa|1hQ{FHRX+{+aPNL#OGo|Im9Mlk|b!L&?jv5&<>yEx!wfNc@WM7X4S<D37 zLvoI|B*ru68@&0UQKQwjF=IxDik&DHc93I?<Q7Y)WQ=nNN+sDjZFse4BL}g4vs6(Y z;vu8>M<6_rvtKFnl}kk+>#(g>Y_d&pi0<XF7zpYa{B2fLxh7~^Rw<%RlC(fK(Tv;8 zejC(ev0VYVie3%p<l6Lvw@_5EWpUz%qd&+6QViI}m1Tqi(A*@QtiQ7x#oKVE5i$b3 zEh27#BOzWGHM~HN7o)jk=CRV~P80DQ_?%b};Qc!_a_8rL95V}T=wn%VEC#tmptyDZ zuq-%;*`#<)J0>v3xdNbI0N}%TdIZ2wgq9O-GuA1<P|DdMtUA!CaimN2`QxG)zAS9| zSzIhxbM9x!oIV?oFCR=ufgIHn(c9QNCI8r!7RRVHA#GawJG)c}TAOm+jHhKW8V~ke zJEBix!%lA*rnd|yiA4`Q^GS>-R-#xR5`onevf`kd0w)ja*-mx@bHn$bOjg?a?X=U< zvDyna(+qrAuy%ssn2C7YiW1KA<{(xge;~aWgZt2^@HlEb6GSn`U5tXfttIX4G2@C( z5RaUKcEZx^3?!m)RU=Z7P{fwh>lZpal?>=z_d$id(>>IMBbk)W065;d@z94#0g^O_ zhC~l+TxHXVzUS_oPJ`l2H=XIG^YEKaIQdDHG9gDBa+RC($m`h-__1SIWyzG>Ou#$` zUK4X77pazdPbn@MVBBpgC$JS9YL$>_VFc^|Sb9+c?%7T$z<#F`WM9=1KsQPEq0*qp zxSu*yI^ha08dVV~RZ|XMeN|qI{E}DmtSM+fKruT%12|=6Q&u)*Wn;3kH73oK<ad{$ z(TgT{z2qj<dn@Te*1t{@GZ$=BcJfs41hHPTb_{`hD?3RBe<;53@<9qohboXMRV=%_ zvZ(=qx}zgzCvF2v<+66xr7V^qF<WKjLtYFPeh}%JEHNlIdeU3V`gt<}6CZdY_?#f< z6^zs_Ghjo~co39Kf)?GWt-N3C5<kmwV0~cETH$>bY-k52Fumz!wzMxSyQW+u^S0Bt z>g<hUN^R9CiLnfsAXWsCU3toWt^`2VP#aWa(NgrK<gus`Df9@QX_~17ghK|+@9JR* zF4lL&m))%08W$<LLOFJHk#aOKD4!j^=90(D@TJ=uA93y|Sb|g{$JH*`G_#I3vkW!^ zLE>uzvG?sYU<Kv9q!8?PzQxU=Z(+6)JGbbqM#Sy3>45SyaE;4p=LXS!jqA5|;hoNp ztP3y{UQn&Gu0;j*%_?^Q3m&sV*j*B@2F@$%<yGJwBVZ$kDFg6Sl8vY?_)=NrVz9TB z2_ymm)!6f+AAWN5@ju`H;G?4-e}gFxu=HAbOE7@BSzVshIQr>3NAJJ&ySr~5ee>^* z-u?J@cVCAupM33ici%XA>#N7#eji?a{6DGiy$|mkf9E$x-})cNuYc#kTOUC6*S~-K z>PMpD(Vd@(x28QipQ-LR*-q>3>uwx>_ZtsB{>Aa%zjO5ZuWWUup!0?dYi7Y)z8R-& z+)OcPFf9nNAp$`-_1S`-pPhB)T8MK6@K5ZKgKSi$dO2QUFuU2l?3MSfrH%E!RP$bL zOW#b<;PFphyZ_5ij=%HT@sECS{FA#!ci%et{?7s1-+FUB%i295OX<?$@x2efdHnW! z$8Ub+=<ZjKfBC<Ue)!`D@BfVl-?&_S`qB3uy!+D!ul?UoU;XOQowo&q<M;pV=r{j* z^x@weeekN7D|E_)_I~`;`#=0Q^O#(6;Kf_-efsJf;*r;i`6#!{SIDL36<PZg1pZ28 zE9>>Ly^05bI(P5m-`;=w=SN@vxoBD8;nxBWqU{Io{p`UzUs+#yVSR1=#mg(!tni#R zI+uVqOPLVU(%wKUq55BX?PG3)Ms)P8uL*22=I9c@rEN^Ai-HL-p6zzFAMH?n_`)jr zypWp<kiazaizFM6`Y6UR1;mD=Q=4n>{o3R?nI?gq9vr3zhshHIJR0q^vn@e}PqwqT z_t-p&-jU84u)cb6H98;nn!A8w<OoV%e|~j!ZT<S%r7y2sUw``A%G%SbmoEz0@9_my z0B>~Ha7UA=+39Gq21&uq`NYo=-jxn$Y;|j%Lp!7pR1XHub_BBux#!mip7n04pSD|W zQu;1tEXU@3!B%;m;_G@h6sADhQ&S1};aks|&MK*xJ#xXBH#h_UZ64LJ8P~?TM^@Us zT}l+xK@a66_lc0(8~Dtv9(T@X8AiPUNMMX6Mgg{_=Yq5A+QD4RqE;@}8Z2=zBsmX< zEvB0X*<Q6VVPK1pI-e#%De{AtH9F6Ev5}rR<2nYTueevjzl5PXf6AFe-v?o>e|aQx z7bML+r%*umDgSX@W|*%Y)S`)7wI8?yg;9<nfR&nMI}k{)3ZQGCs?3d5Mfp6)Rz?-q z-VL$|Z$|hc(Cb(&UI&`VR_4rwVIYQ5IWG|O@9}^_u0oom!16mOli=^!8k}J9;YHO6 zmlh2t>8<B+rvT9d2X9;R%M&)H)C0j0HmMAEEz#)BxKQ_UQro@sds(==#2@vt<v-#0 zleXLFb_J4G461Un-$&wy-984}!5(sS-0gu0*F7-H;!WC%ihj47<vk%1z%rxF9xtvT zX2q%y98gSfstby!t8N5x7&uI%=CX*8F)J0XsbY4C($EHoX)qTPvUreS&BBHo`HL#? zH(%R0C45GdT(h*aWIY5+#}p+c)snZNe_kxXF0mi6xX>DNk9&?`Ua>h<%o-Pl?hA~K z;{)`sl!G^E<Jl&P3rT2JL2+co;OziMNT2*@Z<c<`EN8juF<j0}G`D~+DQ%AU7D$|< zcn6s~hVpS7%raI0(LwhE+Y+)2UJ$btiNd>q!m!k_R|ja4k3s7P`1Js6S;f;cq7i&^ z4&R)Kh8bam5~G&Tx2%AEFY={4#A^CwV)R9^(2Ww76q<R~hy1afWIuWezE>@0)i#sC z^Q;HK=^S{VpzURFNr26|73yhm6^@LBbXy82gXI{AD_&1K%^V3RZZo3<acUE@`E0jE z$~*vHknkvu70`FM1H>}_r3)Ut(e1sg)o&+_gpxj=h4*LSe&EnAy1MnUAk)Jr>-K`r zH^hs2Q?Y7bJ#fD9M%u^TT_pP$Gfrr(2CxH&7p!Sd9Kx2{Z7ZrI4REC`=fyb)abqD^ zU#Q9Tl(b6aR*_m|sUKq2=&BXi%=MW{Raa*yWric~Wh+u*!nZ04jlMUb87$!7u(y|W zs;Y%rL`aZ{695+r9xCEiOII+R^oSxNy}qW6l4%IISVJH59O&Jd!w%dmnRR8RM>FZ+ z$nl2={PDuB*oB!XR1<!CDG%@|8Pwu_RGMPb<Y|$0!~vOV%tS7b(<KaXYan*=?QJn{ zaFTOdel&VKP*39k2$Y}wtCDYT^|!ad%h>?OlqDtCB(haSh;+{eH-+>iU>)l-8Z(bq z^F%D+M6!!g8Yo?dob;OQ1a28LY`sv$X13aCQ_^)#pRKg$w&MY4k_u9rqsd&nhaS0@ zIuR_@NhX_*_Jo*Z1Sl>$bC?3ck|A8AJDEK<dLo)n{vVHkd~WK8;|nb<F`KO6i!>H7 zl#qpDk+#tw#J&x%ID#|H&T{wrXqFrNah(B%Faud@G;IWkq8Ar8C=P^TRGeFLhP*H? z^BBRwIeI)mfSQYjh!_{;htm=ijlXrkr+oXsMd@xK&U?coM0Jfr;vG&=#g(jrR(eQ_ zWc1c#&$VOCIArsX82Wu}bf^5~r?<Tl5;f3l^JNDzT$wsBIic=~^Sl*^SwGNhO;iyt z;a_piy?to78JubK56xzF<HDqDuA93@9^0D%Ky#5+6vI}qWEJpdp#o<PVDc)W2yGx% zE%=1(hOH|y>tqy2P*3q$P46p=i`xqVWl=m0K^QE&WX!fB?AhsPiJE9c8}OG7qnkdY zL+i<D)|Pq07jB^MvfrUDmmpH$aiVV9y>dZYItwHhBLM^9Ft!UPUo^k5^wr|5fC>jv zGr-vnPlr<Yw%W3^++@i^D}$*H!gzsb*0cIQD$T|P@xnQDTw;(eos1QIbgE?Pp}y(& z3tSdV4g^zbB?2$vRBP4L$Mc~&IjaXkl)!sdHf~Up!!f6z21uUBK#RlGJkSfl6NEHU zv@@>D(aWEO*4Ed#mD6gKP^<P>mZA-v=DtGBpKtfsFG|(E31lS;DsS7<7K(1-lquA@ z6WP>>>7hJjpVQ@od#auYtE9{HrT5*Xk_}qmb)|#1eLb=RuwE${?3s8F?Z!6~xFEw@ zRtQ#_7DZngL?CTj_(#4T>@(a?lSIA9riDEZ?SEiTECj~_#nTFRK=j?@b)}ZlV%g%6 z`Y10BRYcdbG+@=Fo}t+OLM>8XUktgN7*L>=DT^m8me5UzDQHUL)ufC2p<^-M&zOyo zA5}2%!4S=f7s$`sh3Om+)>Di)4~!5_OAi}m@_^e=xKpU^vW2988b!@jnYhZOPi)r+ z{zcWp+SEQ`Y9BGRj~L57f-Dv;fDRJk-}zS6t~qrM#?<*JPeVKOA!q7CT=VS0h1TF| zo<W*3rYfc*2ah0VfiWPM?o(EBb^+^n!5&`PrWJr1*95MriruU)q{b~l!1B5TNG;lu z`4FO;S-ZdI8w66(eIXM;Rb(wYiCEQwp?PJ`_j@ipF{3vU^APXHG_byqTjfPsDkEmm zZC#`ya3F8ttu&1x3E>v3DbzD7CZn@Gxq1PyC#S`3(y~v#=GMyBdR_bcDCqN}pEBhg zDeQ*hg*!_I5#wWoYkWp}^2)=0u(Pt4CGZULyan;O5QwE(!N?E6&M4KPhclW$eNXzi z(vVn(&p?WVy;yA7bPLr=^WFgY>s~BYW6{qO8IIb>H$o^$a=kQBR!LeGRL{DfNnY53 z!wi|WRO3>mYcV=j@BsjKYM&U`{Z=&)Mbz^(S06%lP<t9yo|Yvw5xbner<B^O({*%M zuR2SQ*OU==w0}j(eWWQ+y-f-~u$KpYUc*MwA*pc!;&({v>}?&g<1Pr#{Ki@*B?Rh` zBsv1D4cq{+u2+S_;b4g_LnyaJ1K^8a)Zc-s@pa&tMd^|R0)Z#fZbd3}#c5kN*RXWj zX5=n=ZFrZA!*CGb*XEHk@^0EqAh4FQ2(;1WymjhAfOv6~kqtRz$)(0$eynq7d}BaB zOgwp_F;7d5zE}aQ5DjtBQOk==FccZlBJY8}D3)B!I$f*#1shn9mstp{1sh}RSEzK^ zwXs|*Bb4Ka6MSg%Kt-Z1K^!u<6k6mYSGoy!_!Lv(t$H~zuDM-cE`o~zF4?le^|(8n zn($ALsM8~=Iimjm00030|LnbMb6ZJvAo$%8^&hT!svBSl0trg22UIFq@1m_&h^mL0 z&7dxUOA;jz7sdr8QYub3CT3>s8E>>>UhN%sY<M=jUORTz?Xh=fyW6$Bf2LK{{mFk} z^E@*1<m&>IRBcZL&|M;MZ|2F@$&)9~b1oD9WoR5s{N4eZt>=?Zrh{R&J1Gyc#&f!k zE?1M@a!>7r<wNz3(!W#qU<@7H2baTAbpZORjy9U_-te%mIyf1Km@Gjk0FvOcwBO!_ zvqo<)=?&wt0US=p)X`4=%Nd*ew8KCFHN=Z`HOb)mqHk%QL#Tk};Do)>^D`-;BHD^* z=3{Dt$b!$c-y=)sT@WWhPms;KKNWlr86rL`vF{KMe`PB&0);CzwY*FUezf^5vwqGL z4h~>K2q{5DXh+-E=?5ygB<g8;;E~h?eF~Q?=OZ$*3Fa-nMCwxU4xevACpTBNAjI~j ziH^^ch?bW)G3uKdwY9g5z28cL?<Bx?O!ZC_yz_p`pJEO+3-0X?psgmhC?-Hp#^P9A z9IK0C^)$z-AxZ0_zM4IS<>2oQ^hVv&ApC_+Dnh~g4F?US+UotVNzp2j6$}GU*0Jz; z2Wz8j**wY4L9Bpju~$xx8_rWo+TuK#D!S-C8J2tY!EfUU^8!|Gw5KH#FkJYHtoxLb zB`|E$MSQ2jldeR2Zl(zXz@4H=ju+R^sbD2guI<?ZAHc!7jK^yf;sM*s;C*T^QYbZ; zR!wIeTR77TqPQT63!-?sy9OGGnfUPzrF@Y0^9dx|$tNj6vsASou$`N&5TO7FGqb#+ zPa4ods{J9+mye?4z*YthY|ABT+XG7$P!0<`s_hXP*ys`z6B2k|VbSKi3<F9t6P#iC zZ7#aq$j*8`B4h?22$Zqq^$#CcrsTEH2^LjZ%4|dFTj2~?(qgZ2Zvze^^a-XG)U8O7 zdw%L<gO!bIIvr~eUNt=B!1@TBFEUzWKr-l63bsZC7kI2I)M`#M<N)!dP<GiVjb-0W ztfO6ZA|OsE(v)XOu6EzxdEub9ll&}lZc*Oc6Z?2l<b5zYkD;kaN&eb^j^7ZV$(w?+ z@`eC8@!s)tNAh4@n4Im(8#?W=6d?e{p2r6aH?4;yniD%PbsTvOgdH^;>mGMoM_S!< zt6LH!JBk-)?n2YuKGeBuP=G_#2R%v-OoDYrhA+N_k?Go+opgsHL%wUrVBQ{zXkwof zmf$tHS5}mJM!H^^Knp$aNp;Fs>e}s;=h@u^2hpwM1PLLCGe4f?Hu*earml0|hQ`;& z^txkTAB)!}St0fG_a@UO`afcx231#mGa=^Hv*k-%Lq9rp9=ULB;x#pd(Q5Nt|5#m1 zjs)de&TgtzAj^nhS($7h_BEzW8laJ0FSm9RiqGi}z}Q0XSI&V)83(Qpc6U|N>m$(1 zPUUd96+v^or|USK#w{}DRUW~%WRT~%u~qt&)6tDXpxN80nmnK9Vp!`ZQpdU<y3-M+ zm9%57Incb4G&=*YfIsihenb`P(UwgYZYw+6lzjr_sQsy_1xf^U;?4Bdp(E4THU9A_ zw~$G(>GEGe)=-_B^fQh$;YudGLLck+5}v%$E4JY!$BgEdUJ|~;nkdy}$uvM58`Bw? zF-3DO1LCCT@TbU2PZ?(8<(<rghEp|CP)|-4f+kdtS63dlvd4}0SH9a&U*KMZ5rOFC zCd9|XcUzAm#LMrLNZzV91>l9c+3704o)Yig0L^jaEinjgMzxyMt|t!P5l*y7AgY($ zS)BUpp|T290!4!=6CoziTLQ@*{7ZsMOsZna2_bnDE$*6c!etW5`RH+$z2`-`W`zl9 zg6XmDcR|1~gbkrC!<*;sueY-M>mp)<ZdaTs-T~Y4Pw4cuB9<jdFyL{*au_-RM%CFK z-7gY!-#v&Kg-y`|aTnp`o^PS+));PEVx{#N*~-r8ua3_www=^%VW%C<7Cxau5p>&i z3NVfJ^i*ocp(=`CyC;p>sk$>5Ws0?fwj%*_e1I%qV~GIUiMFukiew{q0Bh*H<^o%_ zS6{W%${mhWMOr%uSS$=@Llu?BtE+3MWAw2UBE3b&YCIfF?;;6|5q<ksB%n3pLc6=X zD)bd!^@P3zyisv9?yTq?JuaU$fzIQP-J!w@+TO!*Pe53v_qXqMR+fURKee0b(r#nB z$mq};1qR#pSf^d2{j0%JK-IN4I*@f8#?Jv!+$tPhYkuu_E;)|yVVqx8#OaXr_q$oD z^8i!jVQRf2lye?|G!1?q<VPEJG1{*{EaIZ0N&f7H&KL}#umSm)WJJ)i6%(s0QABWQ z=QSGiidO>V7#-meq~6`}g*Uc>cVHpppd>!^N~)<X`gaC!b1_Yz3!FM){9};H=0`!< z{3s}E@a__Ih6Dz@J!p6yChCrG00&@*O@pbR4&=HBK~1N?mA3#rKof)>RO(Cx8w-%V zh5G|O{@HL=_d)UIy6M0mPS-Jmao@wIY5IN6%rER?K}Fn3i68*D(_(WI4Q`=^esRX^ zs0{}KLQaicl8HWr8-tidZtZ><Gju(`<u@HVYZv+*IlhZw&6?Rh2<MsmuV*<s02An) zq8Q<5D4iyaBU%sd(KUStDYdiX^haQEn#OxQKgpi$gI=ZyQ}v-GEFh6kI6bIt0@qQ% z>m}!bEV(ovO4Mu(77+X<Py(rk!1-53iG$j$H+uo!?DDB$eI%A|#alku)A8k7sb9Xx zLcuxF-*8wELp2``dQYn6l8)~MvAsODNufgk;3GNpQIqvonw$WeLT4qWMViU!6@#IJ z+KLcb=H?yq{bCe-ffoaPusYojYH%w_r6poTsLXumqy;+mnxx}w`ARcjh<5Udw34us zYzQr(MFV&sZl}vuOlwAO$isiW%NKI25VlJLL39%rnaJfKCM=3DMleKDlFiUNN~MqP z4ylCjXYf95x2M_fH*_m|r3=|5dm)SQ9ISGyNt<BC$PJ|i2~)ILdx=9|?X3hhVl%{0 z8*m0muX%0W$w&E~;+eI%r=7S5dU+N$8VehZg^k83a;c&~7cN(vA&IX>{HtwVUv%s@ zI80Qvv0pxeq)jbM;o)mdbtxHyiBf&Qzebj!qb2DsJ&=MmcB|I3p8}h_@O-P842ZZ# zs*~ifS)Y{%-GKAS@vVHUg?LE6#Pf!KF?0njwvI1iaq94OR+NS(K{=TE+k=r?R6mTC z2IeX#4G-g`kH;ZwG60G71oE|nU4f@Tc`)pFs1^(3%Q$rclMf1yTRyu>W+HCMHoS>K zzp1mggmp|}muN+Qars$LR&Dk@b-dhoN<tJ{LB43X_bf+jY*PP%Wst3I-qTsQxfPY{ z+#84Cxl;ghK#ac`tSG>Ttq6ADQtS}C`zXU0wP)NHZ~=(&)ovj+<dj7`Hrib3f~v_* zsdpbnJC(=t1qVZ#*(qF_yB=CPXd1bOIL%<dqU8M)tN8FO$>W?jAwWg(oL<bnqF`_< z*er@4*&Xa1P9k9lWIKR?#bmDl_3Wr{Q7mvzhmRn4UHKyxo!e5muKW=z3M8(}7R}*n zP1d^RTcQw9v#$A;DugG2{+4Qv5>T11kzyF!`(UhQ7;|9(d$WqCD)K{W&B7|Mn%+V~ z!3c0DdAsa!uC)ea`c9O>mEbS&UaB4BCR&r#4RUoo6pfUZ)UE6M^6-+Q`o`SL&uDr+ zd++*2+y;CIm36!x@fV8VWM*;I;$6=OQMVnHvV{F}8dW!r@i6bHo7xLxh)x<U(}bpQ zm(3HkvW*#=+7Qh25cLreALPs~N2I#ZdM)QsSZT&UP|D%3YLSNF=1q*7Y^%jt>+qn| z(Kr$t7hh7$%sZe;38l{g(Y#r@C%P9L20N_|lEH|1B4#2l#)=uQMjfAw&FtjUpbyLh z``V2_c?TcOEUX8tM1px04S4K_Y|7{h(|IRtX_-$_JoO)t{Ru<@TwD1r_;13WEvE>X zL8xu;-r9E?&Kqla(9J$@;Sv?~hDf)CFm8sM`p;&667Q#$1ip}w;haXDgO9u&-Tunc zZY}69R_nOW`P9=X7VQhlb$Ps})>5G+dA-VeQBKLYsfasm_Nb(D-5qR%JU}3Zcpg9f zVfLi)&vtfp8etJ{o1b|5(!Vvfbq{SZxs}D)>)agUVDMz-v&I8LtiWs-$S5R0(3Me( zN#y~It4;G<--Id1F+F@35Zq%f7*No{aB$@X@(@NTC>F$Qc!jcTYqJQ4un>PO#9yx< z{sM|{6@nKRlPlBdq<Xsp>Id3XR`Alwj97)FAN&B=Jw-f8<>5)HqCo&XK~)$6u+}BY zQr}6Yv$@#dFXG@&XF9;XpsgB)Yz8|MsXMtCEw!P^-IY-vV;+SyMDQX5I6khVex^pF z5}0PVc6wMdaOemuI>cIQA<Zk=d3n%Sxpc)o@$6&{XdNR*u}v^cN|bi*h;8E9z2mJ2 zH#{k)e0_`<M@GtH&gJW9d5^Wubz)aQh1j#o0KX&3bUX&Zzb7zrDrU*IWV1in`bn;j zTnq6$Sc$MIZ{wD18sZ{352E3jYzQS;RiPDUOInI^@$}P0e}BjPQ`!m?E<0x1ds>@H zo#QMpHiYMKxjD><7VW#$IApmg6|WDIIlDfT3hEo4Sl4$uBw(15p;TM}4q=#12^gki zL|_cJ8iOIxj=&s7#NehP2dfKPPgT?{U<7qQ6n8eX9$5xx59Zy}b!Wk^8{wx-%|S(p zAG|l1x4<h5DGc04QimLN$^mWP7|^1y25ld0g=?hSHbxE6Iq@8Y;bV6C9tW?dAnPz` z_a<~{u-~x&D7{0YPTf||z{n`q-FIX4wL_Cz-xl1oV77P(BQH=r5vSy~m`e~`!onm0 z1Xwz*Q*2Cd;8YK^Z-(nxwX$W5hR^_@3WDk3tQ;9?*YnK=@%^!Vf9$=F`iR*cC4Ix( zfr>pGEfrlq7bh%|wS4?n_z*J)gO9Bx*N*H|fDgHb6CX?CWFQ2Iv}2#%QCIyJVFjE* z$Ewirj6#rf$j<NPV2h5mbfuH{tCOYJoJ0ycE2Kkr#+MoF(Ozr)kPNhZA~S3KuvUEi z9;jQ*KdIGQi7($N7ah}Tewz#LX3o@KS?qQ!6=v!9y7^GpAV$bat|%0eru#Z8-u%~y zIEmMgr<ULacJDLjA$+{91M%~pT#5)v_2;r|U+j!{>8|yDj8|*<t;f4e0&-B6TPSk+ z;4wZ;Zcu4E=K8&xx^3;A-j5pO1~-?JP@RC0awo-8v*^Jg897pd6-XZ_PO8ZOwgdkQ zg8zl0WpRh*FA{CQhZ8m;rSUL%^NTIDvF9YOpd<xX-#FprJi04ebM?cNZdV}mYR*gI zV|!TkVtf1gjZe3?lYM{>q&hB#MZ1_x%1N_vEg$`$F{Na)<migdI~ao{Ma&~T3j$&5 zF8vec8GuB>mEiTy9i3&5{{DuKc)Llzog*EYg;zY*S^4v(XBhIyD=N?wL;v<B-64Y9 zK5yH$dIed&gVQOM(Av3^M{lIJ^^;JPr*ELq+vpYj#7{$TrQf&9l<gF|pye=IBM?YQ zTo9pP#nae<?#Yr+V}}_z!|9Zu5{q7u7B6N>85W(CPe95%DnVwQE(rPLx;}Pq($&&? z%E!u90&iM&_G7Ah>{T*#EJAP}rDSaJrC}=QEe$_?6HCLOUVmw5#%C^!UO|tp%cZeT zXD*G|P``ntStuYE3dn^5^0YA$7h7eqRTf)iUS0hI>t&|cz6}vMpDho4Z94_mZ+lAj zelZ5M`=~bn74;_T<jL-fN+?e8p6W3y1)q!HAS-ov<YD6Q7ovCV>QD{zidyJn4(;3k z01e=l_6~7Fx=f8tQ9CU5AfXopYKNwrbjsPl@zG<`@Lb06sHR2U2Nb4gYgyidj0lR? z?d7md5Xz+3EhYsdxCP|z?L#DZA(KNci8?~L8#tq@)^p#Eyz)4xoq<{oZPa)jqn#kX z&~o2A$e)Rq7KGp{%(9_QQ|*m>2vx$xh-Y-u$_sQv1wF|I;ILj6k7J`Wg3|)UMqMWi zYpbT@HOPzAHNooymlUdpiG)6+WVn7>C%iGXvV%i-IV^Ng9*o%tM(u1b0;5djVv}<o z+?qt*2^%MJe%-Q6eN}I1F>*b7I2WDz1_m>Bs6n;j8EZuCKuMRhKRp;WvddZH!=1_U zM-4Es{=rDv>hd}(q*ir4et740lllOap`wi+PG<E>*;(D3yRv8N1-ki6IW9S*>)hAP zFvw4?WBatS=uD)Cl$+qXXEU4w^*83@zSRCgUCo<3z7PqW;Ht16kV(*wmd$=+G`(+Z zR3hjacqiM`&2@ntW})jVpsZMcL&R5;e;gk~?s%t2GU8Vi3zW<>;+H>>E8dFSx<bLT zD^^xlx&pkhXoOZxnSBT^>h+apz*_AtnW2ITC(K_c=kV3Geq?e?#Z*jM>`WMz7Q2{` zXU<gn)Y-f1qSD&kJnLqw;-`;m(tb~up=0u6U~tr4dqQ_<@!mSZyG^AHw4Xr~NVg;K z(JW0l9Q1{~wlvHhl7+Pg&svx@rsX(MRdEtVkmEprgfMP7U_l~w^_0jBoSF{C(_>OW zJGRyM9wH)kvkMMDdh-D6+6rmQzYr-b?n#S#(yQj88R^KJ8)oA*?H)A5pN|GMowDzp z)4@Im_2gOmvz3V-UR+ft(ImXT$S(e~5~a_yk)A(6@wgnKJ>@A<!A9+>qpbEksk6&r z|GMCGwxMd=Sd4Bcv{-My-Qf_CDVs2wRrXxde9pQW4JX?hKzQZ`l_IZmZH^V?vB1r_ z22N-fs8L7u14HnFfIE?2*s~=Ss1lBhaV>_#A?O+7hguC!Cl+HhsWhkqfZ-9#5z`Ks zaks!)8vk3x|KnWE+ZY{svzL&{aVQ-Rn(cNwpH3$+Eksr82}VkXyX6Kst|gQK5f#Go z)o#{ViMb%N58S3*00=&|4D%x%VI-~9dLwsdp1feyI2-=eCvLoGCn;sTXf~qEy<9cb z1}1%H3$Ba$D}h6^aAN6kr>cJ{5K+X!IFQ&ub%)3qL<fy6<E@&Uzu3!&<)Z1A46s&f zYLdism+(YbZ@cHAcJc}#aB_QF-G<^(=p|4y&%pQBDZqVnZ)pCu(aIbkiN@fwH_6BQ zgWe`B7l;)=aC3iBjw`%Y6AIApt%M_W*q@}HP<|{oZkA1Tw!YqplkOYWCSBnab}I0x zv!a1d-E_+loU&Po?KWpM;?!v`e330|MTOKqwlPXMP=gp?%sV@pv?Eey6Il!N*5Pl* zzS)*Ti$tH1)<edHCUXbg*@P1!ObBaaO~z(So*$epw`K>QNz8{k#`!Eln9V}BW)cr; zJC|7Ny7Z95e2(Q&s{Ay^Ov9Iw#Ac?^%seJZF*~k697Dixcx`O9ZB`(KbCE&m33v?< zhF&$fJ1w53kW|dBJ|suxn)>mD0ZG6uW|v_l*-a|3jAF~rEN>v=UeJA<ucVrc*#$*s z;^T9<fH=xT*Rhq2XD|tI1}okT&f1In`WBs9H<&dv0}92z9zCJ42$G^EAM>{;`m<g` zUF7;QsnRsY_Jv{0-s^S&di&rYI&H4h)gpQlR*IV6h4>X#T>bXxON0v3R3|L*=B7DF zw5gvCvb_-s{u-zr=~5K$HNs&)hIw&caAEaiDz>nG<rIlL&9>8{n1AOHBR|JsE*-X; z$-M@(XSP`!6}`i0;rV>yh(M3*LKHpIgGvh>posQ+l&H?3eL^a5dJ1sr-Tz8E|CB-x zx3V=Kbnw6zy7PWHtk{Azf@IomFmWr=GaGkI_3UF47JhHeBrzh_(4PoRnIUd5)3Xl} zPi@E>8*y<k-p?zrM@SAmnEa@tZVb5Vqw9mfzFnCIj_G7QKt}BKy%8uP0>kA&bpuo< zO$^HJy&J3F9Q@RlOZEA_*G^<0Zey_msTSQf)9twr!8Bh!h~NY8)B_E|FB~ua>YJ<9 zX8cw65^j;2tyGTl-e7v%%~sm)y~&*fQ{il+*m?$L@p1ZIw-?QO+Jfu)XLFN@WgrF6 zP(Pk~n<v|dfkuUA-NqaSp=zC~*sZxXb{vqDVqk=NIB>+7Izv(>WGjl-pkbQv$Cw&X z!C7w(^XW`<ireLl0RbTcGZkwBu^ExzZ6&XhO$R8)L;Ui~j7$#1%OxZ?ZgCEl7#Xdi z@-XeGE$&YthPRB*QA?mo%U}D`Z+8T6vzYewK{16>w_jG%I%uZab!`CB#oOSl(rV_Z zaAZMhDvKHJ${#gV^22F)je^3>B4mT=LD6I2@!KqXV#F*aB6A;E_HBQ#w|~8SHoC`= zyq#c(yOUDw6ZKSa^(B6ZZZS<BNp?xcOT$zlot56?U?vthy1MRR6b!+q3B}ZI!e;cU z;Emd926;njJ*rTF!o46F$Xs}}#5D&_3$QtZ7W#u~TvmP)3k!^e1;)Yx;}tB`DU}II zfU-#P8?$>)lMzM=EfU7R^H?Or-ON2x%Q@{OP%NdZ%@sY$Z84w6xX@NHvA$v^x$tjj zkczZTi$0>ZxJ9*Ts&27au7&+HA4)6T_z{CDFjHvDc*e?SvA{aclE`@=Ye+Jc#2(ne zMb!yjV`4xU7wOY{I3UF+X%rID$jjW3Zl-N!@4yBa?NL>;&b4Z;z^?)01n46)B9EFO z9E&FCf&@mD;uL^FP0I0P0B5-(O6p`8fzJaUPb1(jacex=hqQa;0sB8FGIT&gyBq+% zH^E#cheJi?)Fv#T8OmtQ(C@1A^D)prlxp?BjLCySnI<RI^gb`_F8Dm~G7XTC7HURZ zks3{7Wc`8_0Q8!<Xu(jg!4U!zg`Lgv-t<r%o`eKEh2V?AhYTk7_Aj4iNaW^jN1Y1h zASJ6-QQ4B90(>%GNH}htUBa31640>GGrq%k6qs&8JnLX<Cl1*9iPS9s4o(4+Fm}W_ zg@>bp4#u$}`kShWR;K>hf_!gfW<=1MAvHjns^*S5V~;a((1D-B;($<gQtYY&Z6D4f zIDJP0U}{EtOU@uSt&R0jOBh_ELi%LEy)C%61@|_;OYU-ZHSax9v(d+8&8OHT1?EIz z50ke_Y~Bu+q=X?DIymVfXv|UXji*;8pn0GIZG6a7j-uz7LV<6MvDB4?g2TgYW>fr( zWZ`rAcDI~9;1Hw^x5Av4faD%w^f<0Ll4EKc$B(3+iYbceCV^g?C#Ytui76zDkqp^F znokwVOQOTDIzLk!oU<;nM%u&1Yi3jajx{l(r#g-j4Ywxz3Vc!qU(Wi^4bfr#Btq-? z*AVqF&i5X|C$3PjpQC@*@iStf@u7YLs9}B(vWT3Q?py3hoP)Q8tf!8-0J(L;z2oFl zT^zBCBX)7b&T+)zA$te1p~GVH$*QB~zx7)>xhpH<;lhAoX)P1MD}`6+DgI$=eIq+D z5ol!%zH|r)jeL85Qi8j@Y7N@GIT(s8vrBHlVr<I#e856xx*!P)l5je%`{v*Xn|b*^ z{^;xf`~UgXPk#FHZ~s(_Q}+sT4F+DRv-)l;TV11&+g6|{^LA`PBB%51wXIgR`CcpQ zblR)%wX=rbKY;J&Rf+fD%Z1ezsN>=5n*M6f`OP)e*n9l?15@(d3+`8Z&40OR4P|q) zb6z$3Zo7kBP%t_dsAdPo(c$mlK6IYT=<))D>e=~K9))V~{QERG7?JsYUX;=01qjvJ zh4<R;>ep-Mx#)TE{e4kJm$x9)lXow)FTh(>@I4&u2l~tV_N6XoYp$wB-d`oat5D^G z_WNcP`1|UbD5J|;5a-{6mT0yvsMqhVw$J1D3-<f@4{RAz-h#NG`ncAiNnTSEeqoJf zZtcAIzGBOm@)iVCJHMs@tiI2fao+uIOPrU80Qj}KN^`UZxcDBo`L6hWUX;=01qjup z4o}j%tBk_$p4VSi>|0aPR<@f$efxm14e(cfU(pcFmk+QzVo!ARGyG~nBHHU@IMz`K zcP6+hsV^Uz-D-muyxkF=N~YQqs`Nxu*;GZhOuZ+bf0I*c^T`%EoL$N~EA6$5HKjUO z>KqiymSIz`Ig-*C)U%fiLE3ck+Q)#)!X4ouYm!~>4vy~Tqy2o~2oDbOv5xq%2jtW7 z*X5TSWq*G#%3YUBt|J19;{(rxq9G}*ZvkjsVjz=>P+%P?-e-`wzOy_sap`66UXkA} zR!TR}HQyrF;XwY~GLvlI7}4{|;*Pdp$QBIQsg>BMNvQ7a?pDS04yI7&o$XSNt~zYh zQS|<bJ|q+~_wrx;<;y?)e|71f{eQpw>5m(fUsF+9U9R!#zxbcN{^?KPMF);u^Ze^y z{N1mA^q27b9K7fV$c>kO^rK(?{QofTEU=e<{#RfBU;it-SOZvk&ohYyTP4jxCaLVH zh^(0*U|cWxleI|@F<6s4A^@_k6;Jju#^_gAnKG?<#I85yGf0CW&uZ_S<C1g$oijzz z<GLvhyNHY8f?O`h<$_$Eo?L1U#3hvv4i&tdXq=)~?^=TO$!Gxcfrf|`60{FKfa_em z04)2J1H};aa#_|RbGPh8`4y;~+E&AUNae)sTa2-e!s%kKL*Y_Pwxw(_&U!(|7IbVu z$KHe-?$`hB-~al*e#JsK^&A86;;zN*i!<G9wA;({5N-Im*z?IJ)4@;{drnbLma9o` zxu^Eg@}VlgX8&Ai9vR=Nknu@@#OnF>g|*JRYpZLk@4mmbcHup!-G1-<+Uk3qcQ0I6 zSzUd%<BBk##s}}6?_9XBcD}QE0jTM#)mU43@BD?8cQ3rVvhx1>7s4u$ZoNVkw_jAj zh+3HTUmxV<@!@(kgcW}cE*Dg%g0|DEmp}ZQuYdN7n3$oh7Ve;!V%TF8ICvz#{@QP^ z`+A8LwU3#9Er|Jom@kO=sZE&<2gB((HAS~>+yfd>Oxi>*1|x-aZ@-7>>z)+r@QaPD znHK-f#=X0@%bbFKzla779_N$%AbSC`INCb_ks(BP4&L9}`J(7e@%@9M2NeDV4mTfi z;u~)5BncVF5P0$z6BArp71Gj14gUp;JzLoh?0!`foRkbyj%qwqW9quZ5qjOS-D^4R zE!n|q23GYBNY5obiA_^=sZH-35i%Mz;p0^NAyY|kux^ub!YJh+HQ07ueEDVeLdr(P zQaY9RnOBmh&$r)({adLG{Do?2vfEblVR<?^KG(~qz5R3W2nd`a7v}(B_lsP{k)Irg zBgRIEcgK!hrs1ob%kNa>=<QCEU--+=b*-<24H9-GYlfC#)BQ>L3?t3lp!{y@qo-gn zjBQsGTVE09AB-T5cF{giJ63JIrPM+gNxelHEuD|KDHwDVnZMlwIGp{v6gOQQzk%Jh z6lVDy38KZ>8{r&9B?A#F?pA=9c3@Bsp44O@e`T0gPjVp0pKM$KbW>Ab^eZ6JJBsq< zqYCD`=m&6AfnixbQKhDe8wlq{AUT+336$To)8^k1(OAqDO(-mIo~IM;+ru*DW@-Vo zhs7?05D`zN<=B0IA)MVu`vsi#A@o_cb&}meV$n=eoiKt8qhMEm@t7}3C(-HSs(4^A zc**M$chfV*Pgj=~u#l4CqNnN;p$xU=DwTlQ?i6Ul>=X<k?sdR%e46Oe>W`w57DSs| zr~R_+!x1RVY&kJ*u@*d<g^m4!F<mgGr<0Zv6?;F;fNp$x>mQB(3|Y&voW3q!84$8k z?X$t?fdXxK$@e8yx9>gis(q2-C}U@xD&7@sM)dFQc@XK5EZ`kmb=)h5TUDdnQTLHP zN1qaaU4FT4CV+9I>g|a)G4<UQ73JgD3Cra)#Wz0kfT-N}SvLrX2JySav9mlNTT>fL zu*{f#7?LODI$5~dW)ro?6z+pb15aBy8XjZgeFqpih>Z#`LCqDT4LfjB5cMn$Anj90 z6kU-b?oQiXmfb-y>_<`0zZs~RCanfmY3twWByGA+bTr|Q^J-uHCilM(7#zh^17nZq z+49hQQiI^$;s+?_U<~y!x}DIA$3+t=okWsw8C%d>(V!yXe2~!@kv$z0&%9h4ifrzU zz_nDZ4i`|HV~Q2gvbcY!@IzxjG31lqouD)Rz&?(k9efhyFj5eH^O#Uxqv(tY1~ukN z!d2rN-6f$*{uVxoY(|YImBo`5J`A%OQS6J?*r9J?Q%FXVOMpC;M?qaw?WS;kU$4xi z5N%Jtx+2tvSh|pRpKG-w8(%%$`~1#rR}z7bE=pFvIvh^r16pl*;K>H4Fx>74GQ<mm zOO2w0O?nhW*QQ`DkU`>Vc@*T`u_a(it_>mCnj5Nyyrvz2iKb2186P{<o-HPLuYpoJ zx0QP@h>9YTVhA;`dJ%-#5IpCf!M)OU5U-gRof048HPaTMYo{pll;+D^pRkm8Xch<w zOcVj&<I&8f?VOGNB)Y?s72N#ScX1MMAlY(8y+tC(jAx;+u$K>ehr@hY1l4sf=wg>w z$Zfj7+&Y(WiNmX+TMA>dXHE$)HKPA}%*w79dn(_VoGe;2LiEqX`UxZwaC!db0=%!6 z-qee3eY?71)dMof?;dK&2=nQmdXFWHR%D#5SQmXtS_~gpo*>+f@kr6Qju3>CYRgyj zq>VW!RDFLa^dc;;1KY={=y(3U#9pYQ3Fr&v5hrU~g)m}1IexG+6XT(}qiVc65o}V( zz!T#>aanekRS)^{lZ9AfA(mK(B~C4t7^29bvZ7QL4s>0e81F<yX}gZBLYia2^YU=g zEAHeY)x$R?#{jK>;w6*y=hK!|*fpUv<1Y#_X<oFQ*DH!WxR()Snk4;J9TY^Pvi%8f zm0nTV-I;~Gi=BLBnAr+hBd}?B3WLz2imG{RC0ElEh<tP_5L0GY5NfD)7~?jn20F5D zwc(*u!9?_ufkKA+Bf+=l*Fm|{K8GF((Rev~sQ7pmI%JBeRXo@LbOGgE2TR2VI1fe+ z;PUxvdq1!4JsauKHElQ2Q4}WD8i|YvL6pC-pAU!Sv!*i+ZbWK}5@}{4ax>{n6W5)I zV*09TTS_V1;3E+?QH@O`=Lnps)hoaON*qWayk3im&;IF8TG*#QmtCGl>r8s70At;( z>D?P9LIrbbP1xy_U3P)GlGeF}<brfvG8kMwZmxfR=SFsA_$)uJbeDkr^MZk;IW@Q= z)m=2i_ybwTs~jNuy_PY67^{aIMtDqMG?Vs+2W9_oSah}d-VKsNeRPrm$85w=`Y><= zq$JjVT0Q{>dhw>Q^eqo!IVufuy5qcq)PSBmL>4|Mru$_dDUlakEOP3lq)PLgEV4Y^ z%gwp?`iBmUB5~Yo+A;cCbgqXcf!HOYgY-B&q4fA`LIN7qDT338ebEoX2B6F7jU&Z$ zRwOpjQzgw8x3{m~_;h>Q88Qgf%VE*RxNN@9Bb+)7%JF_z=TQgtzQBgsc%!(*mMP4m zgCWBKM61}r;A~F^eZ`%MG|EyX?hGNe&QgfxAO%FF%%}N=x?3q`u^p|eBM{Y(9|boS zHtW&@tVmCAjo82@Ok?PaYv$sbxwvMY`kJY>HHG_CJAXsO+NIRUK}KYXMg@^;ZCd1h zF@{5V)KhP|*=9rW+>I^sc8HN}0H~_IQyw*<hweSlEK=}KpmNj<AWa3dQQL+;8w`h8 zf093QZk-r$>%~r~NNqmyuZ3v%b)BkoDeF_T1C+BzHK>?&*#$&!*@-#ZrRoS3>;+Am zHnScO*={;YpQg<mP?@fhw-riK{?;b!%|H`@?v*B{Fs{Z^2q4<<Q`@T5QY&(u!tq1u zfZpL`a@+)VgRtNR7EnA55aO$O1>$dLGG>CMyNXSz)^#=qC%!Rv#IFMd`Gz+0!dw+Y z2=<SuvtTA-m-T6RWen&DFe6-xYwFd?)RCln+fzpZ*lMBqmwFygT&r*SB4Y(K)f~ja z%kseRLN!A61Qb6(?hbEQQRfB3Wd(xn0uhr5oL-ywt*IBIl?`HPWT?@i-!;s^gZ$YQ z$W6fc0U_fvm+@0p`{_j86)@5!GOcwaPb5yDECJ}9j3LIC2({WAgwF_U{D)HFpj%`% z)HHMoL4#OISG*t#ArB+ha*x+CE)7VmrCqroEi`-_q~%~49{<Br?JR-)i8IC7^LWu< z1NprPLt?{WUQ9U{HPSi`M{T4H$a4TEtBza6^<*4tOl__v+S*dUGY2Y=Pcj05qi0Fo ziblv%0NYB&3d|QlQ2IRk&Mv6wpa<V+<1MN<%As4Xp#ns%&K{f4;*a6jhe;Ba2#EVR z&LoCSu9}MQx|18d0Yo!h$ircMSM?vS6oAo+H^k&E`lhXV-^!k;`<t3PbrMvHb%F`Z zO?A}Z0}C}&Z=MzwhJRDsD^O*w*nT{OlW8`1o9Pgda-ZnO7lTon3arC(aqBWc(Puej zkbzxVZt99&Zs=}2e}~=I$?j|WVND0*n45UHz>`J~>A=`Bh}BunwjtHky)mV*_K(q4 z=HGL;7E9$QQIYe8Dyr!ur$NI3owWE>uzR?Fd_sfUARJ%i1jckzb*|`>M`G0!B~8yX zSWWK&@H|aa>Y^F9Gu4V{e@cYk&Okg#fU5b4;wkZ1$ST!KwFd_yjzT}GrbVte>4GAE z572bTXuJCv<hW5+&*5~S_9gQ+4rmfV-KR?_@=(-}VyL`oQYt0*8;uJ=rU<b|ZF;CE zXCc!g)Nbcn#gz6RbdTtqFWqsK7@<^(#Zd*uVXl@D+A<=r&j`XIVH3fY7Lw%z(>L#= z*wmN!WB^leZx{IzDZZ|q6kx!nikDSu#ik(E10ocTjSdfXiphzcac|TF;UU3`B4^7W z#s&x7#at8Y9tsBsj3}^ML_7lvqY-h?c+lun3VF7pP*)vRyGVjhH>YTG#RI4FLzPxn zb#+D_<dq>-ddG=v7SjUx{Go%IK3i+93c7#dQNRPk8;85quVZcu0#MSGL4z*gX6l9= z#){z5KsH@8<qm$u_>@<<3&4ltS3^#Oy5?Q(*7kZS#gdsBBi1S`0w1~&ef;pwZNG6G zVpyq;JM))#R6M&)nP)9(lRNYN^k4|-z8W9yOqM@ts8hA^gHa>g>T(oclmj(~ia}q> z&T7zZU(U+mjI?;MivmX%$Lop0Z^K;^rClLa<IR1a4N_*GK(M^#P12G`Y#?{5#<!AR z6?iMIS(7eEmo|z!Rjq9(wLxmPf{_<vo$q}QE6D^oNj=$#Kga4q`2DgxkaPB2{SI@f zF^d2%fiqEmg7pHUA8b)V<36O|(QCMH%djUR2^$FpG<W6F;9vgq_rLl1|Mv2~{)?AC z{4@LF`B2><)%mvKvhBb8`5(m!ceB-&oL_&f6sm$4spYXmD43y?I6EMe1gYO(Iy|;U zdZ2MsSI?q%UR(?gK=#uJpeeTgv1V=`Yh!6)?w5N!JBpunT&|HM7X5a159rXdf#ROD z6593&Dx0{{EjY3q7>`x|9^*mwn1xJ_!I_b*M*D52u<kI;wDYP#tJ%oD%o^Yuh+j~M zjNgvF3xD*{ZEHf&uBqHgf}ECP{JT?5r{zH{NL4fiapKxSqU|KWa~`3|Ue|e~jVPUm zS745jS#@nK+u1sWEFzCj?Pa^OQyyhG)^Wf*1UWZyHU(^pK}9C>aG;Xy9ty(!iu8K} z3Sc7LHE%uCC*#a(c(dkB2kTr7o)<nUg7E!WK5C#PK4k6zWJduyxXV+W3<iL}NH;)T z$k1&g9MSYy3D<RynE>MrLV2y$cv_6qq2bN~NoqAM*<4%sZYxv&Y()xfsJ?Ww&dLf! zQ5WyO{*Qn2^7p@b`Nx0o@@N0;>;Lq#mtXzxAHMpdU;q7|{OYg&{N=y-vzPzwkJQhX zKm5tBfBz4E`O}~O!&m>=#kT5@@G5^=fbt2yALK{$cQ9)B!~l=5YkW^ZQiUK~jC=Xx zzj^tmzo3r&>aYLHH-G)3Z~p$@fBirI_~qaHP(A$SZ-4Rf=YJ6ee1D=gaD`xLuel@r zlV5!E=Ra4FfBB1l`|JPZ$G`f+zy12p{)b=v<X`{8SAX!$-~Ii|Kll;-{N_LZOV!ZJ zAOGub{>9J!;j2G{;bWDTU;X~eKmN(rKl*Q9|M(BT{*QlPK`Z+9>~^uI_ymd~9~hLY zw$=1~{cr#3>%aUj&iwwXzx(x%{!cT%WO|`~YChY6pk;d`nAdV~T>wtC2l;5PzyzX- zzjzYs#n*rGlb3(|7c`J>zWS@L|HCg{{_x*?{nh{e^-q6FU2`{nnIHHPcz$)g*ahwg z2+=uoVMgpeM}&a`8gf>(>7*Q@Rjhk05K^OT6XHihU||d#d=9^#Q$MmTxBHOQRqe+y zM12`dA^%uDlA9DxTU$ZfOQ0D>MZ%BBIN;a>Z4twP`>(V;>hbOhYClJIpyND4+H)JU zIgtvOl(*}1Pk=n(a$?JrP`B~VR#sNTK00ydB+{gQYw|${jqa!!DJEz`4uo(maPEfJ z*L3?AE?n?q?||pNd0!NT*FXD9C&&%D5T6Y@BN3krIVqF9ohC%ZQ~z6PxW!JqYUye4 zN1}k{&DFKF)@o-BAh+KSV8_=%%DNRC_{kOVV!^WCgTaxy1{nQai?=i=LGQq;Me2uz zI((rHKczbS%EN~b)U~MbgO%-~w><!2o#T~i`#=r+sNq^?P4g#3^Z5h>R%FGDK7k5V zCeri@#Wo;t#e3!y3~Qj4z98bOO#uPoNivMU7-^#RcwW7JqwxBWT=T;A9mAeO?F8tz zknFou3;6vXR1ejhjr{%x*)q?_KKq>ul4P2a9+>BoO;}LOdD}EUE@dxHOs-9mcD$pw z|6{Tse}O;Y!zK@~$hU^&PCmSLIH^jy&rpdc1<+?;mf$u|nzu#^u(+`^Ijr{Sd%NAX zRp3MFB%aTfUejz38v#wcYNd@e;D7++)VY4H;E-8@^&;89OkKrCaHrc7k8$GFJ5m#> zm-MBfPaZ$t`r`apoF9wx<1|t3n3ViBc^`NxA|wA<-Y<r#jpYaGK6H&PG(*(1sG6W> zSFCDpTJi&A@{GDkTR-E|<r}cccMnnH#4WRS#w*T$|M~YH;NKHP7n!b@J@0l#lhTMl zn2eroay?(!g)&BrpQ#xB{i62-Ws6BLZa<^Iwx3T>C0hrEJ&B^#%kdNmzOBEUU3pp# z`Y0WlIK$3%z{UVYcOZaP*TMc!>%h=qN4<U<lgPj~agrXCATLss<zuSCCEf=6X#I-K zr~F!A{+*}=uM)Km$B8<J<3x>T`$aKKRKj;*z42hw+YhVYV_TZaYg}xr`{S_bzKa>x z8wAQ+Y%vAH52my3W=E~;xTaKjIK^UYmKVc`s-VQ+rcozGt=peaf<E@E2I!8APjvKm z%c-InU5cYpjF7;Bfg(^=g~^p`{sQb*1LOh?1At}JRHwynwM3jvYzS_jHDe%=Hq}|I zFEBz?*w3F16nB+1s~&6v#R=^dOIp?qR2#t=Qx$(HjRoYFT4TP+Vj2)+UxJJGR<?ok zmQuWHAr6eVRkM=(A;TIlAyT~o)^ZsrFvyGgK<)wlGRiw<g95&6oR5}aV>R#>&fjta zr5=p{NwtR}WX0X8eTxd3d!surub;uZ^0<M)AVUZm00|<pWs=ZjAo1KEfS?{E?bFE+ zzvRPdqg~$|jDVZFqrU#;TBE@syl<~Pb(RA;VQHE_%FiI$kXlCHcnWtXZz{Q=1va9| zjTc81#CC+^8MLdmLPdti-oEFJ(2N+~@gc*zt7?)P88(G+JpwabA6ugGDQ}IVWgxKb zv2GZQ&|pjVOC$wqH|~M6FTiOhL38^tiD6<H!ssQ=j8I+~Gvy}YMrf)m${@&;QE3Q~ z%Hyrhf6<`LtzHFh6wWk46M^TGHdrK>b6VYy{Nvo$Gs`O2Tq`)s>#7`%bp>;bN0h~< zA30I6=+KeQ)sG!<M|Vte3yGm18=%TdE+MA#Hpn4&k+VUaZUOY&-TBwUDL=<zIOS)o zgj05|`O!7%CMTScaakF22c96e%m6$X_V(a*lAJgec@RPBbMbhSgX%u?YkqKJE-t{G zSP`^)cW4=2km!Q+q|@uzbbaRr)ty0KP0_u{1Gu7JJ)BO<5!0tn#@DHW?HV9|(Vn7J zGnirM?`<44%pMnx49Lo*ET0SJP6e!+VhleQ{O_@N{z%>*&{?Gi{9cBm(hR;1n`ZE@ zZK~i{G#3LJVYtB42rzmcRsp)KX1Q%=CdP595l0?3)!_!L@lwdKZo>ejK`=dq;T;E0 z`xemzO%CLEW{BQWo^hc#i*{yEU3r=hhOk0ciR}c3T=$p*;Z@5XtM0NF?Y*K>%UB?a zF;&N{<`mOHMty;7X+hY1I03#z+(0Y3VgGZznYWKFIxilX7spPWKCV!94l9c)eOx1Y z{>a_K&Z4>t>Ro?R>=YnUH7&<#FX4N8j5FuJE=G86wzBv_iAn4qb=7|@)oJzjxP@cs z;=lO)5q^Kv0)w^B@fxSTsE2%+vl3Ys#}3CnhRsDSBD+9fHgY?)sEq`^>3Yf1HS7Ci ziWS9WpRe<g`JxXD*C7GSXVPkbVQ;S&Lo{$L+V-a?lkZfYndiFfoqW3AR&ScNiiAsM zm!Bv00KI*bqLQLh2l=rgd*@bA;JH63(VMO(=$ssY+SDsZ<Iji*SGan{PTA5NyO)e= zcGM*}du#!*nUw{Fc3;R@YqU4j)n#=sAyGQQLBlP)!tR1Cs=hj-y-*(vhYsLUYO3!X zbKN0`?nfjvJgAD{(*oiZ*umT0`PR<Y$WW&k^$OP*>Q3In&U|_&`)n{$r%weZk$T$A z_!bXthiEmZ4jb2ij4OmC2cx!kF+ywWZni=4g?>q5{HdbB=sGai-7O|C!eaDvFeyhM z#&_Ei>k89cmMrh?p;v)`(NlMO1H%E))M+33U|$GIadjVkJ{S&j6l@oxbDwN1_sd?j z{8_O>GEuO{*L!%$wQo?jPc@jmeQ<Hu&nvZt5V{t02E*atIZ3}+9GM&m*mU4=P1h|o zupPQ-p#jOXJnZeO`>8bBL(}pcq`F{I9&W^svh&&Htkb@bb>pu)_<B{n?zB6;ML0*7 z6FvM$?V!uLMO6o{<nTjOMgE25U{^-{hc3;>3j_ZRiI8t}LMCwdr=^L8j_~}se)32^ zdF0(4oo#D31a9=0`Tr{;B$L58h~Jx9blPIN!Dg^T9kn{=I!oC*S<`fkkYPn5gG7Oq zc4sXE6Ppz$U26SOQXlxpPMj*if|_o3<&RfiT=$4Yo-Mcp)G=+f1mE8YMH?GH%S^L* zI=m6=2AXRJkawRuMG1W8X62mlqen>|C((zVxK80ALJnj4#CcdE*N^>zqF541pEIQv zN;@iLS08n*CQCH~=xwv=(dUn1yfvIS_J*VO4c~q|)eFR5a!IuoR`004Ps_n!<;l5) za5|KA69^DxXVq|kDqvDUa(Nsy1S_AGpgaV-DKv~1iyUQ*;WK^hMM~|9*N3cU9<*G; zs?=d(EU(hzTO=q$UG}6!_RheZ^>Wl!=TigBIPn=ZCM*Z&eM>OhWDq=k9&$vuTUIAE zBd_$93kR#GO(3kEFfmEtRFj<tn7mFwZB535>NkA+^Ub{QZacB(w!#Vi`#R!qVIUJ; z1!C*22jkvQ=pnjq(d$jGuJ#wxb~fH?<h#WlV0em3z`bh&6+BITx<of}occ{&@@&@l z-Qu{hr76M1MO2=J8Og$o<TPd^){q30!QeUs)iD7B8il>}TM@mPrHSB7|5lAL2uIHI ztFR39Ezd55Y{Veg_Pvo}#F}oU77KhFBE(|NOYsJzeL)E>tSr0r;##Ypkb3*<CDnp> z)S*XOY^mkG0|?X^nDqT4O<&+6?UiJ=g1O3}B|he~Q-{{UU<4ilY8xMQv(=Njr2=6> zuvP$&*?9Q+IsJL`YKv!k7nC)x=3HyZx^oE~7~K{a70+1m7HKW<RGm+n*VonGxn!r| zqUDrLmjh3HcynhkV&_XN(^d-suvacdisnadQT1CDT_5cZMuVvox7z{XJyaF+9Fyts zVAN!|EnUSA=fhpJ50}EL5tP~BIo2&N$0jY{vE(Fk&5MRD{N17$Q)oicZ;;aqYBXR8 zjRt_X^SMhj)t_hQ_z_-xZRgHqoh5g>>ne}9N~fj@VhyyztGpq?>(jeV@UEVaESQ!B z)3RV%=3-ia%B(+laO3ukPgU=?Z+w33_9xfX7%pWm8u`(nVj)KZkFW+u8sT4LEU&)n z%<6PXvz?~8hjulAQA>kL*W6^H!uKdwP=mnpzT`Y_WONToS&%i5oFA`?WX<Lrp5<tL z0(SUM^U;)#co2W&<3Wpw1N!}K98-8iH+m|DlDPzh+gOlq$XAaIBuLCkVYrqo@m$~K zs^D9!X6O!ul~PA4Z10M!1l8$8(Rc=<FVJC=eThAP`L^v&k2?yXToJswy-~EOL_h=` zN)^gDk|C;o=F+9u6gVZ%L{@&28a+@u7YtFhoC;0TOO!y-_jYZA3$nZ*%L}r6I<hRC zq;5hKfpzV@+B^v8nDK1YltYJ>q6$(mY}{OB(PqOuZV2mdBy`~H1knwqYyu2b(MS>F zGxYYs*St@R<30?#CAF}}Cguldb1QW}vF*cj5I6z_!b720wgqGHX3U7&5K9h7xGwIi zrI<#V3V|S>ZeQq)3tnJs^97NtA^0o?eY-c&LGDbB6O1v$#|*?+gWH>MVzXt)tPGDo z6E-P)Z;j^c{KG6bjnh&||9@7mNCk_8>{&-9494KYqit_vc`(JeIm|s%BAiw~45QZ^ zjZ-#N1a0Zy;*kau8P<i;$aG}R{%eB~w>YH?Lxq!bKh~dT`#@NmXVjc;Z~HmO<%fU_ zA$z(ycIx|x<1Rvb+PX1@-6p5jEXdOiA95@7kVYTlWON0wc_3&N{_R4Hjxh$KYCJ6W zAm0ajLWZz+AT1%}Ko~dWtjw7f#*Q)d7CsbgRfKen^h1AO({;_(Yab5!@v<iu!{;Y{ zSX{3R&GBYp9@S*6_ZwRja9<P{^aDHt#uCg#VeZ%ilwgN3sys4O^qVz=C&-O}UuICY z86brM3YGN{QUxJ<uc@ukZ;Gg$aAs2Uts5?GbAVr`z^M?&t*JFe-lSJ%Z~||Z0L694 z3B+@<6lugKE)YP@=?Hj-s7_4ksO#K-Sfp_NU%9`20{Gw*)*;Awsc@G43l2BZ#G^)( z6O^A4WK$GEH`O68Y3O;!E<g71crtiu^J6-$8ES8lFZw6z*PD%^x2=`{jPjl}YMVHn z8|8dm*|y>>9;@g7$XB`VPh{TcwC7>=g6Ka<R9ja5ck1qdv<;{-!RyMEQpiZFLy=kH z9%fO;%4LAB=#Kh^kevMTc^}M~W8KoYu*YF2ldbt7@aqRFhjd8}FxmtLvf71HSV!ej zd>(|U&#J=-DXHnA40`DNpfone?_`h1`RH*gdkm`qf03(rjLO-v2VVBVY(NA%H}!L! z|K%7MOCviVE|r0kcN%nUL}moSQ%+1zF$^;V&}=xI#woU%3xpyHpJt7m`UA>H00wYx z0IPI}NF?@<7?m24XMjs#=Gp(R*yv8)djii!{Y1M+u$efv$CGkePRS$5kVWB>-PL5& z21-;vysU%rXe@gC#o_Q<I@*NCcH<?ix`h!eXK`Q7$&VJHCS?WKHDb)w?ADEYEX2Wm zV#o~MsBMx92atCBq{Q%Aya9y?CdN4wkfF6vNiW@@aKPQ*@_}J{LA(@u;e_O+$p9CR z9%m&1)<K^D%@=xFOkxssXQ&Q$_z^IrCoOkzC{|MZGSRpFqGFNQ<S_qEzoUGr9k~S* z3=Tf&i2gQ~sXI*(8J?{q3G7va`H&nT1nzCeaF2T0tvy-LJrZ=0sm@ZoKH4p9P}PU> zu`>k<-R^>Zc0lqmqO6n!1~wMTPDV_J{cUm+iosq~4V`iL%O3ESSEkJsv{2!#=8cBV z*9|ZKGT9*-$idnYb(xA()XtFGAZIqCy%?+8FOY$$3Y)mv#|bWIR8bLJDZrk{>zkv! z1tMo!Fm9`|IBUFKZ7f<{HhVCr`<;~IDt^^uOG>Hrja=5oCd9C(K(4=ki8T&2%d#MC z7>Kx5HTOs&gMId-Hnmw#i~`%8#Qd3$W2_C=3BH5!_pNd~At9lhfmepZX1J@3fa*#Q zvUETua0nCLFwYx)!W<O0ttYCP%TFt-0fE8yyqfqd-q`I(aoJG3s?O9JOA-{^7J>52 ze?ufsi)sK;R7G!YakN+)Iu&n&ni1?DY6k|gm<+R(FibcoQOs<NCpmO9K<j7U`Gi8$ z{x0}q9DE0e-FI$IK<xgVtXTn*J30<vf~a!@Cs;eXGU~Hx2n4D57`i(dsD2@bTqtZu zhZD|%%dBbu5RZH0r?D{Ebea$-xD+U3-2v`XVQYB^B6tN8Uj^WXWsgXC&jw{}ha`a} z!mwl_wT`@ghm-*Fl!}~6QpOW!C}pEatkikPNC)MwiAS-FOI5K4ilB82#Qbo7QdIlp zuupM1gR=YiI35%dK4aAa<m1{3UwZ<+AVZLtEaYtEI8w=XO~Q#7119oOxZ(~qy-wjM za6<<)e9;<abwaoiLKHh2T<TK`F-53iV5KS}RDHN3nBA=?kVwzti-8-Zc4Qv?Om^t@ zOh_^M8t)Yb(eEE!40iA*9KGC5p~(`nYpxa{_Y6;yyzn|YxF&Lwv|}kGxHIT!O&O+G z%Tw+mjmPepy9&q!+S&S4J6Kcl1H9#!NXAk4R}ZHFc)mqxaf2Ftg>jgXC3Sq7Eubjy zPbFYHDWKyPJS+bs_z(`}(m+63-zJm3P!mm_V|8|7V~u#8IlWW@TvEy&XvRdPm?%8B z)<l?7&FP>eNEmaeMx&8!CWtE3LxV4C`*qfESL#<n@=x_TwqD1n*NG5mKOON&u&bE$ zV5C+@?g1v&a~KeQdK?`qtc@;!>5$P9qA@Oq$9v_dF+(FVdsW~=+~;644YJ4RPb`a! z=@sO}f^gX;m(1C-gJi#`9m=`lE`BorxH;Gg(yk|lAdkN`9=dHjTsd!^I@RyfjKvOg z_?+fY$MUrYyB%ql|HjKjMsgBiRNxa^0y|}BMx~~#0)cSwkXHu01efZU2OHpObXoU6 zsWpkLh^|_XMb0q3pkaKWhM|EA1o|S`Lp!xmB+{@pzlh^bV%FjF+BgKC)8Y}<nTbnw z(DnFKGrE&`w$e$6d+w}2SY3x(@`1X=@vx}W1>s3?w=|b-SMMLxQdfkb6Ko`4M{bAL z3=W_03}30Lmce4v2K6q;EoGRf4N=qMs=VqEwz9g8nLzLIF$~f|%2zdidb<8musWJ! z<zyiiwmj<Tr9SF?5MX69qSyo_<TpeEvmETl=n7}T`6(u26jbd%z7(1D*d$dWe>TOe zs6htRR<W9s+$Jb<p`FonLko(zg`voBq17lx^xW@fxmaRS|6m8H0?kFpgoa4pBoQ(O zh0$vV=7$!zdgbE9QvBZ86e|SFZho~5EU#jeuEARW-eEC;U@d2;bC(T|>{u;Cwy?QQ z+$^TOeGyyLn{*@1(11ZgunW7c9hSGUZ#7C?;Uh({lXO{j!M|U|2~T_|Q+4{^7;15) z`jMixLICI<0ZzuPb_FJd>h_<YzphC_<SlKLNlB)4CLz?}cwA1V*>N#tSqg}1Sh{2{ z4CkCYA0A{>zer69+IgCmG~yS#yJX|Lb4-?*kC8on99cC6yiu9}q9vQr!M6vH_E$~l z<#<IFyKaI%Jc*k%``pU>XPvDS7yWz{QUIlxroj+Hkh`%0^(S)pV5MS;8;K5{IAB5= zI&WI~7wurat3$v=dnUAeDxc_Z&5<nez3cp{Iq9+3b)BlNOPju?JiEKYs?BS>r0%Ed zxY0Qui4244ef%sYbBr3M_R4UuH;QKRik1Epq@@aJbgcu7)6_`xD10183J&^|*2?G) zxL~VZSbb8Od4=9;9tC%S&=r_nAfKsSkl-+-@_?|hmAx>E4b#p^jWR(@N$dvfO8=;h zh$1mKdMjcaZh<evpa=b`K%Z%LO1_Jkjv(YOF}rMB42W9=&_FJ%+P4Px5b}XLc1m$j z+wYkbV2IJy=t|)e(}BU~HRPXh8Pxr}_XNzbdxsTdqasHX3Z1aa9;ONQ2R;tFOL5pp zIh;UZ1B#W{oCUGAC{q|;Q7qpkK-gr0uTXhjJ4-6j)#{f!+9^d!Fge3E#mkNCj9KVz zG}N4N+XzT$;~0BE<mBRS;3gZ@#WF~`8?p*1pi~kjBo+gco((($?~NW}bRfm=otbHf z?X~Z0SlREEC!$VusIr73v9-LTSeg(BgSZ;I7Po%F4xWPQoWcmc4(NPVtu}KQDNK{; z=tI*7PsQpM^h}Cs%ZgAX9OpGsh^@CcqwEG|y1A2&6wR41a3+*Zr+mwpRi;5Y$#`Pj z()NZY!K4LdBwbgI8F)(DHbMKUemP5MqM4e$ogq`2ApbdhtQyyy4p3C;$q1*xQT3K( zT%sIbMBE@y#V0gwS_FI)ZWlXrYl#<~O}nG!G#U$Lc}|-kS;aAS^${RhmN6rSFwVjz zqZULC-e$w*7j$S{+})h*hcK9dOAt7nB$~uw@>jY)foqTs8-=lVjQ1pLgIIGMFh1rX z0}V9rL_)sWPx5X}@)EXsPKRxDQH8-|!+#a>ctz7+&vwoLwg60)W2}M4LFn#yc<tf2 z@l}||J3&@t1r60t?GiEH6doHwx=|GId^E$)Ct@a>bjX<%t)yrJkE#AW4$sSolLaUz zMsU`+OBRI7gAYW%MevbV$N0)K=5U3yCw@Ye*tAg;cbqqlPWNq0Tr_q_!Nyq->Q%>p zD@@q8H$9n@47Erago%)Qx;Lppk8Ghwu>}aPE`gPLQjTYAzKls^)V$0thQnh_Cy?b> z&4F0NxH+lrnc9C-V+0KMz_d;fKB{;=olcnODr~`<t}E#=GM-Z?Pyz#$llk#D$h{*+ z&k@Z1F)@kZ(*TeZ#9LS3_6jtxxzuK(@FrFdI}8r$bq@Y%D?jNJb#dW)zVJO?_@19C zo$ANyw?5v!e($rp+xH*byRS%PV|$y#<?1r`F<+CxmjMbtxN+^?gX`O$-gvN~1{bSw zEgudcydZ8xI#8~GTNuy;&>hfbom%sMaRr2adQ5lJNkh73XgYv)>Qe?{nMw@&YyDj{ z-v)PBNXESD-hLP`P$T{-sJZHm<2vgSF;M_oK&8J0fny#RzV+CW3Bs9T%92L?y1-to zwK4mZ8;03+lsIKyc!KHBMMb2XWDMU@Ke=b>r&qJf(X&ApcNk|*sY6~RwbiK1dSCm7 zXhzwsv{bWB&QyXr2hzPfQ}~|bJlIj)+|TrHu#YXCxz3&`Q!wi)rQP25wD=?}tc!sD zIJtF5$fGw<6pq{7OB(L0iFy^*JV5M3wYH+8Q5Brqq|TnAA25CgjtjBQda@L+r*rsc zB$myCDY{y-Mxt5!LvBd2WVkK#?226!zEm<)QL{n%_{LH|cr@n<q>sQ<2?TgW!P{w8 zX2yyLMm&_t*>15Y?l;pfAp>$IarRjh$bRS=lMk;O_c9iqi|3w{Vush2tclKbP3K9% zj>UEyAGI7Xs4<9=II@(%D&hG+=rV8kilMfYak}eT%7mSYUiZ#0EJ#p9U)yPOR6eJQ zBgMQ`Y?AUi;~29ktZX=997NewY&rTJHBssfCOsYHF}sF8qK*xuF-~hzuNJ-hkWImn zJyh2T$bkTUps)#&6+c5$9z4ZVD~eI9ri!tIs@e-i+8Cji-w2(z73efnaCVXA(>Vk; zhep6qi0r5FvOr{vY_3zll8vx+4|I%;!hw??L%jMC&4kS`5n93P4;V~$9ZF2DpLBUe zdu=w;j$Bn^dir{s!QN5V+CiVysPazsK}u_Q+%@*kDTQ4(J3nXFgq26+HRRT$WR!57 z80KTno&ye^w2-X~-97<jNbj8S<!P)?y@;7Bfz`?TYM{+CJk-nC_gA-X+`0cT+?lp- zuit#A-geLc)))B13=Gl50I{4f%b4oeu96X4IkJ9H?8t>-`M~T3oUI_I0TOt0l-g(K zsdmsH_#E`Kffh!?$}5@Phl^2ny1wZ<hoOHl=&LKv=V0kCZaaW=@Lhd-0SKyUzaTAb z<BEEQ<LG9c4rC<d=U1UDzgS&8+45J}Y46c1wYMtUyC&K@e_pj0G`AHz0l5S(_T!bu zFwyA!@%RcRZ^|_;Wsg69OiB1miEDm|N4_l;7=@{eFzAESy}geD2t~AZ2E)PhxRpKA z29;#;a5&_U6MPjUi!ta3T5J`ORirrlfIYUC6qM2daZQ7WRtdU|Cl}3$7eqjRFz1kB zLwsfdMEo<n(slI_>9qVa7JYZ-I9+8gun78mP>&oxAZ>gRuMR7zO*vx6J>58?{fU8d z+_4O7>q5our>W|pBUTKCc4uu{U~zD+Bcz#~J>Nja%wZgq!nDWWqK4AGY0<|HKIT(+ zTB5k^F)z2;5=ea7*~jWAi^B>}FWgA#B*MLb36A@87k?Z>sF=lK+WeA21fn>QZ7yLS z2*8h9#wCrTPBbAVI`5rb9`z)0p3)XX7K0Ilec6or*r=emvC}qEW|<cZM5+mgFO11f zqyT3Dy0@d&5=3S?PaI5GuNdW%K?&oUsFMxTzW2&Q5P(NdH!|tO5U`^Lrn%V#x5G@; z7(lFzr(oV0sfXJYgHxdi99X}A^U>2{HDN|6SJ*`xi)$?#@T$fHD14LJJ9<~V*)AH_ z@nk2>yu`bj*yt>sd{FetN#C_6+H~cbG1EZW+)!1E#W~RXBnNlzW^y0jpRjEipcS-P zft8-luu}>5bA?P8FayM6Hf^l@;3?q_B&jhv&3L87NNnKc=mswexa%KICdbWkL}pZ4 z=$n>HSs-_D;c7qxih22w7Fi~SB~b51OLQ<oGxSH)I-Fu9>d>qzB9_7Y?eInut;}m6 zDsYA*6MbWxPWI$DeefEhVh?&%1X1K#N5t@3hz<goBYk82u8T%d?JhMapmp7!sf!C< zX1n}1QOKU|U1<ZJKza8p@6Kr{F3DKe`8AUOb!&-H!YaP@;54otc+8D$7#+icM3xGU zA9pVW>lWQw0sdL}B&|wPc}gP_;nuz8rzL#7H-R5cP~w0r!MU2Czd-K1^56TO2chi| zW!9ubvEGeCIme_qqY$Ha#(ru*aypE7#~T$rzi10OY_3<>z7rKf#@4Il6%bXubKR>a z--=2Ox8hZ?@63ik%Z%8pz=uZ9K8Qq!ZmQ$c<15q-0gz-%3r-^bc8}Uy;jXQ56wEHW z<L`R5W{3j{^q|sCIV>)TyKkKE!Ds<^0kiJphVeZdk97i43}jfrEU~0T?t7k|?QJx4 zAU62>t8lK;i|4BsgTl{OA=8pCS+O{cTf$RwI1?<_#P!GV?UZmz;!d1uas{5y)V9Lv z_l<v;3uc4vE6s*>AR~Le#!fk%mQovdng#OKgHv{LKL^hXlC9DOVGPC?4r|8KLgBp( zk+ttJZ)beHViC;U6}yyJ>Wk3|AwGEOloOYNEo@&Fwl53Ym$__TOa$n7IA=82UIwd` zlkBPb_pMk?I@Y5{-)$N*Ymk;k(!gCSIHNFHbMmPxLyw9e#c%zV!(yoEdpc0J9?nh% zwHzyKU5=A|*%H*IQ-%$>wW$Or7!z%={tH2XVCG>-j=coj&0?UT^p7+tlx7(H<0yeN zs*!WH8&23G6AMNbICf}qI9jaqutZ@OX$z&^6-rcQKI~8gP1Q-~;%O<88R_J_&XJYQ zDL!{*I`}-VO-%<Mlb+7Omv=-2Y}H~+Bom~ogU?d7>>yr^p%AtMDNP3I&P&VJ^s|vK z`K#e&h6`59%MRyh(M7yQUVbNmp)R&NIQm=_F^I~&qRe$VB#}v9Co|Ux2#-Ptk0gXn z2tgvVX$SBbI&?Aewxi!Za`7BfARzsSNAy&)f2$zrEU1?Wv%j5vA`2ktc~0mhkwok% z`V!`bAfVlVMhUzmV{@0xG2Dv_&S$~-EI6N6<a{LiLZMWTO&I%Z)3LU^M2gNWzqC$` zyV*agd3T(C2b(+nmI?H%0*^G4rN!~=(!z8m?vjNP+3+!uSM}VZv+6SH#40yz8e4?9 zQXtn-HDSRDH(fShHDo>>NZk^{7nzNz?m%l%0E@0}%P|MB-8OaN!O^V;qg$}fg%Lg+ z8c#&U1Hbs!G$K=*#U!AJ@yqgq5p>O(c%2VC8$I48km`E%%^4WBi`fz5g`2AXZJ$q0 zxA1*RJ3SA4oCARVoDW&I9++cc&|niT3Mn=k#(?Jdab^t1;1LB#(94@uxV{0z4LVbq zeE^b!2mZlfg<4RI!a3?ez^UHwuwO`v9SqXLj^RMK$p$P$ptOB=JSi9XAP3U}HP~g{ z;yt@4Br^iQN0dpf;vDsY)$z;M`e)Hs%zQyByWnjWyv>5QITdfS0bMBY$`GY&?|=iw z7@DFxD!3)$#?ps<pX$PM<}Anq>gnngwJ^tqDR{e+UBb0GmYEn-N0zhehX)787NSDI zej%hlraYFj=Iu^Wh_*VihE46SsYt`Y;7DEQ-MTHej;F{&&#RFP;c&XtMQma=nvuPT z<z6ueMxtTr*Njor+jbIM;8yVRX7JSc)$(VrP>U%1Mfr&349J>b^1<2JhcJ1N=jH~O zWc}QnxnrT>mB}POZhuIt@X_q#Uwv)z57wuRBS^_`+}MgXu~Deg8IA@(_K8}epmEJg zA0^)$C*M39^r!pDxBCT<d(n69-g0*Th9W0-Z^4$sXivIZ>hQeEorQ<e0|T~ViyCY( z<Y<hTp3bf!0Yz(zGnmB^7DnJ2;w+n%Vg#A5CdIOB*53{_Bi~Dp+!^L6d-|JSV_rt> z1b*rzGz9ZuIcwEu5}tBPr(E{<gc%}1C#@RnbO2d0goF_eq-yZI_{W}PF;S<PWHx&< zkz+i>Yx)9DNllGgoKIq%8;E@Aq%VSI+dA!t5Ma8z@LY>%t1PA4PGmpw5hx{3jbjK9 zRK5c#4v=+2?y@mCr`7uiBq><_8ZaVT;a0iE5>RnRx8(!1!<p1e9Dc-y*Bz{p{t7V3 zIdSjY_4DH6>~njY6=VHAigN8p@rEQv!D<cs^W~RWCs6aq@I?W8Atx%WpK;!znkA*U z7d<lC6sIQXQDy`}^x9MG``qp2M{doT?F9=iCiIQ0^DPm<iIEAeOyHIU{F=AqU~f)^ zLoD|l_>qY36uWRw%}0WD4hYhNQ_!y-DAl-5`zSkWz9T`&ma|pA!m+4uEGiJ`;Tpk- zyxCL8&^z7=CT}mc!_#Lv&#A(FJv?UuBdfC)ZmTqSxv|oVY&aZ;lm)3g6$2w#^f{Rp zpR0#4-Y(U-FfR#am+}a@#qhZf<?ev+ienm3uvNVF*+Yxo^#cqvT~*t3GB`>ej5T93 z(BynrK>jRwwfYh%#9oLai@|Nbc%mE=4P7b{-_1aBFTc&rtrov)MdnGi<BuQ$$SIP7 zC(umJ8Ea<ENp~?-PN<GIU9##lj|y9c73QsRi*`g&nW|<h*$Ew}YSSv`w_cl7`GgSL zVl#OxV62W2#$ve5$>78(9?c8wH=Nv40YdWyd_JU(+}uyeZ{nU)oJfI2D|PLKWT;NB z9<&iFEOtjZK7PB?bT2d*+17i6A#GbBc&8GjQ&jMlV(hEOpwCHxJSc<F(|`s!&r~;@ zG&1UCF_|`lf-UbB7ip^|DXDX)Iaz7v$G9}M*<QT{-WiMtEExi<>Utf!oKgM*=jl?` z6$|W<6`h^5t84m;i*zGW%i7x}Dr%}HZc<@{**7Be*ozAT(hP#W(9dK;@vO$XES=p* z*?g|?&Gg*Fva83#I=-}xKAA360k+aR_rmStP0~!E&$SZwCd6G(jVo+!s)oV9Tit}L zYvr^gS-}z+XtnKhC2p}Io`Nl;lh$V=)S=J53afj<R=6@=F>f7&*`b_%c--QKTbTkJ ztcRqtR8H-|2-WO*L`zIM2Qi`WBMt@BI2+zl;Bz^mDrhP&lw+u6t9h#EE}k8>Wc>+m zms+kRuC2S~GZzLco_1k`@t9ngFggomGOIsg!l{*S=a^mK7_sS=Q!|+0O>J@7GNz^B zt<52Qs7*o*sd-ivMPM|1tt_R5gg;w7S}cXhgCp=|BcI8B$&2`Qr<FAXs-w5;x7`_V z!l~(wSm1BHJD^()_IQX&@~D^*Y&2Zwz#cJr!o_d9>bl6uM)C<C%8{dhqj7MW)_ySs zRpuyHo3i@7Lr~H}40F^So4o+ZpbsJN&M^ji%_j~dj7p~Wi%+3MnLdUeLa9;pdx|r| zZ~0EuJfQTO2b<^NR&ubpO20ZbB_BiK;rENFeQv_4lHw87Tea#&fj`ZIV!B`UDFnO; zo3#sZB_<qt#IczYRum9v0n4?;!;7{k>U0$*sdBQ`Wycc3w^(BDq>C^v#TK_)=x)2u zCm)aALT0~11yn!Vv;*4WC|Mfj!VGm`s<ANDcokEPO-QiUFRI>TfWFaW>RA`@2cW4{ z8W0LA0=t{{Fg(*c*<(sHe~^z`VAb|J6)-^@9PNs_WK}JlogSm5JW*%`+uwBsA~pP% z+LMPPX!!3>(7f&rM9iYt{qN`#Y6GW>LIvq-7KA~A_R0EfcM1-~V^68|R_I5hZ35IF z*ft>B<N<8OnnYubT<ws`0cN=LO}e(<E=PNVslu&G;6@oZD5(``nu&ZH7y)jxm3j3= zWFs_?Fs2+-#a{m9IV<Lj58jzZC@|HRxHkqSo*9^YgfdaMiAcZ>wCVNVxA}m_r~$i! z@CjI3K6UItIB0`Y*gkWLdh-N9i2Nt)Aa$AYV08Tgabuf=K?pW#(<Hflt25j2uEf=( zH~U)bSXyvXJEFWk3>6y9VKJjh`35tbQ7h8vH)1a{9PNR{jJg{oY-ZMXH9yMcVX8(; z6(>eu)>Ud|Zx}YBn4|R#RJu0)RHZpUyQaEOEEhGGGC7<FXU|5ppj>Wquw|HL>P20u z_ejd%xB`;P>x35gHomTKoJ<CNU+nxFGI0nuYjhKPTEE`Zm^2u{j(v~eR3pSv&Kd-? zG>u}ULMUH&hS3w!Lte_x@wapIhz9QJ`n@@LGvf4rL=z5df2JA7Jaa9xfyAY#9!eXB zLpAaGdup;zsU3I~U5?Luh`3&%gwyq6R^)V<qM5`LxSaaW?k>kxln~G{In(DII#tN5 zjE?Z0;9foc@0QBhN;7F)e8sNx`@ogp1Yq|upd#?hJICJg)vj$qV5dQQORBDSxA={} zs4Y_`VmK+89`IrCYs?hJo1va=bn^)s6+)#(l}1%(NNHr9=OFratuy(;@!z??9pnEf z22NeJQOiK?J2<q^eFwwjjPXuHmlm{#bugMg1j`)c8V#CKIPPuBWO`d3y=%)YqZ-Hy zOgP%GwbNnI3<2<b;@^7MX?d<M%?036v<G~6W)pjc0=?6U^Y%E^h?V8-xf$r^l=<1h zLh7eM*%i5Lo&|zGTc|4=ly!qZR5fBm4oDnELkIo7tcl0qE-cq*&$x0{P4bKcsVwHw zU8?EtlI%JQ8tKtOT)7ZeF2t3;d2!`~0&M>?+FImUqxX&4)^E)WDFn>8PQvu1UjqIt zE4s3LhXd=LNJ5u{d;{X?!I-YxUq6AoByV{Fy$N<2#Cm%shpkRuV#%wP;o$iyi7dHK zyRS|7Z8&VbnlbBUjCfe=s^bTQXKn>W3Vnkd*v(fl23%O$xrf=bB@7({xG2KvKJ3Vc z0=ipTz5xAL>@u;vef`F#+uPBr0|>TC2yi%=l#^y-eWa-Fpr0jXPJialP=rj7=OZP? z-OB86TbVn`>|7=bx+4`YJ6ka}N219r$%;g%BTCqDZ_KfH#96M*K3ycE^eJRUX(kF8 z1=w}*(i`ek!)+NXgFZw1?n{{IRF%e`PUtHr^fB3Vn^N@K8%GWl#YlU!-MA3KsI7_9 zK?OHhnqUE_^TC)TZ_q&8EU)PNPBZeo@8MY8=$U%5JOfol4yxjQF@g8s2ycv2?F9I? zVeu**uJW-jnvq+Lm%xJnPg~YI!Zhpv4COUvXvdi@W8y3KVaIuhlM?K@$jo};$r)Q} z-Yu<;!BMkui^RbKN#RtoXh^peI57vb_uFU&0Gn%hT8@_yca|Ax1|!An<^A>?1mVO- zZebq>Bk-poAics#&WQ^Sl^MN1$~v)Y8f_zpln>W5iX&mY-*giUo;SxOEOIeXW?%0) zM5txnUh^+m98jt?*(0@2>l!n4j*Zng5l6p8&uK)zJ+jqSoNB(SNg_Hc8;wM<#g_7@ zsz(&B>bLiSRj9(O>?`f}*0PB;@eU!j4ZSmy6%H1gol-R<{T`4Yh%64KYYCG_muQq- zK{Uh?1!g<?Ak@}BO}hpyv5YNm30nS2dI~`s^TR!7dVC(~e(1vJ7vUDA4B$)lhUMy( zqdcKv%uZ%;kC;?rnVG@^%MlJnBaBMp@Ozb&lT=L%jC0_`J>kG92e#W<_Tnaw(@$BO zE0+{v{RmlON?d3rf+Ygiz|y&#NNpfrr~8U&$J-p`nzAZ8ogD>ZrlX}al!9~%LP91r ze2(6+ZQmGq3Yl9=$yRMrKh^Eq(Ap)k|8#Z}hJDX)R0X9!o2vHkPPbEJ)v$afXjwF} z6KOl$N$1WC`eB+rk@d$#JV^HxMgp?z%LXey+*;yYa113>W1;qs6K7Ir@QM`mAY|H5 zE3K|yZJkC$-D(^etp0FzRb5}5`Sur3QvZQS8(Te%2p8ew4BEybZMu`u@w<ZE3C1in znEEqk2Gsfb0YrIbD1RH83-oK+NFzH7ctSh0<*hE8J;0{+^<x31;D&ZO<#MxrS?P!C zwkqRng%x2dwa16mJ|KG7jgWNhiL~zlw1kKbyA<j40Sg+1Br;%l)<<A@!*^!B=!cN_ z^(A|^j?O5(^{(BL`*24kNdzTUYR(tIDs2?WCzv-w!OBqhEg?)y$8Tdi)Ct3aw)8s+ z<?ctjrA4k0uU^tvvZ(>pY2X6>W_=gaL)qA8E-}|Bi!eUe)nl4u%@x-z#MNU6O4j7G zdh@LcUUa<qnURHC^pisediC$ni)c!?=-@UP)K+$BRsr9|VLrIDo8>&ry<VCTiNTkX ztP1Fd+-mY*RP)29wJIqqiX}t0m^=Daofg<D>XB7{oUs$srP$niGYs{mnCxK|WYPJh zIC+*NSD&fmKGejt3*X*|Ye@RZ(&>r(>rq+1kb7#G@dKj}<#Cc!T2?ib+B@bA!oJ!h z`?H%oxne}qKN<Z7>(J5Lc)ZxWRJB^_n~;N#?BKMOMh@CzQ-E(QMyX7us~??ksBf|a zQSZVE!jkrG#AHB3(-%gsT*T32N9+Zl+eM!mcT#zKy=v*!t(B;EC%<bnJcY?o=pHIO zFA5rj2KO2us=CxBM@)Aa&~Za?%6F+{kRRYx;^Vy92X0xeV2uTgYNgf)Ml^x89sfT7 z00960?7eGq97&QO_WjKIKO|>ouL{FL5kNP0W`RL-Ns!Z|mM<gF-NS(>cA_c)<gid# zTUkY*nvJuylj+QKCX>lzvNcJzr<1<j2bo?UPN%cIKQr3d)3^Hz@>fK7JTj{QvUf)& z<nA<4nGqfl9v&VZ9{zCE&?bf@NAF#r1NJ+-+~(Ftr-CKR$G1<_YlvU9oqD)LU*gf} zLbvoVt-S|I+X_v@lqA(H{&$5%;%YlJ1WOAh+VEbmip`<U9e8gddEVT&-rTod;l5R5 z;C;#*FKOR*V~tI4g*lm41L%4&d}hUGCD^Yn)So3cabGS!(;`wzDi^heE98s*xG#kL zwKDZek-egxF}7Y%(&^NpCJ<g4LnY6Tnxz;g)9PZ&!h@TQF07`LLMKk7URj%U>?wth z=D~J|!JC=Ih&E2M#$l#A=|dLwhA4qd#n8j#q0tm}J7g{6>q6;OS|_ImDZA8AXX3^> ziBpxSBB|<WKkZx7*_NE!`0z=$98cyZb$XwD`|Ip*bF-IiZS}Iwdgl%v4DR4EJ1)n? zu$O)DMHsq9LBy=aDD6>FV-%m5hd$4rZ#$^)i@?x&h96X=5hSu}05_cG=i@%c5C;xZ zaL{nNAVa+=%rs#aI}ZR8`7jMV+!9>b8+$fuYMc>#lHumflg!tO59PPzjP|?H(jsy6 zjnQs212D;xaAI}tB#~j7Lc^P<q1j#2vD-m-JpB*>)|{lA*=Lv3ge{1gu)Qa{&^H(3 z#<%6)2DDx|>UpHc*@uNLSrlo1u3*!V*+-f}9nVPdfi7Pg00jwhHr9OhDBQ<m$eapY z^~x~?xA2!};g_v+PW37KmnrPe${HI21{JwdpBNsYVK=*#rih?o)6_G-7J0XEH}ci~ ze5rFI(B25NHv;YI0<9HkzbWfE<xEctv$(=OUi-}V0i$Xvy3oXh0DSb6d8AqZmO-gh zjYW2-d?wMgV~cfSbq1RB?;&vXdG)}FHqLyH8XvUOlJ5G94nw1vF&_FkIPh=2p!j1~ zMsy_pC2lZ6-V)fh0OzS2aDLVQZ8z3nyOi#3bJGF8y<wp@EcAwjUdsmC*jxPmKl`8m z{eSbX{=?t=%h<q1bkKqI42nc|kOhcal^-6#sG)1@MLjDAbi(iC&|&C#C+pO6N=4md zZZjlX2dP`rd9v>L&^*r=U*oG46nsDz0KVgu$KF5T;T3a54TqSu@wmJ|`pe%}CC1zs z9|MWbc*RIG3TIUl%ec^oKQGX5u2T%wJ8(8xCKxL$9CGD1A|Z!wQ`eHphP34jn1IF_ zV6L;|JYxUgKTUijoE^Me!rM0e<62YDh&pqBl|mnB;pb;`4-kN&h_{K9vD^;c*RMBx zh0QUIe{;3aeiZ?Q(w7h1C2Jr|LCh1a1h)=Ullp4|p;Nf^)oLPgZYKxh`}1D<C;%2W zABAz`eXc;J5io|>G=eMR5wmQFz)v<pO(64&Hjw$n)j=Bm)eIKVbcbfH8F)+GbS+sX zL{89+jqDz2iMD<0`j&y#U`z(lDjhq&LDF1WMJILnlSs~|N>LjN!M++dSViw>A$tJs zWt|FFwSL(L7Q3BwN_@Y9fk#)5_wXIF%J`k!G&b2|C|+hR;uMG&Ik;i=a6}P;M41w7 zeBtVycyLxv#b+!LQj&Re#~p~?u0?zQk7q7qzIzh30{YNtj6RWKZ4PGZEvss}eB)`h z?PQscso{iKv{}@7IiI2B>$JG{BlJ;Czt_c(n|hVTAg9(3r};UHFQu$<^wfw387-Ce z-Y`-~aG;nf`=|M6RGoJt6BEB8CPcM{Tpj_$9O~xeb#wB%IeA^@Hr9mW+oNcFKi%^7 z=!%>^bQIKm5I=&D4(689ELAJi%WlL#ge%1x#*0ZDaKe5ezd7?m8Hkfvd?b?EO8~W1 zBKRbmL4X{9&i@U&t@i?V?EnBKd;<oa2!GVeF1Fz977`s|d{#|BWO`o&H))brqIP}k z`Hh+k4k@qA8&-G2>TX!w6<M9J>5`kXzy<}Elqn$!Vyn{E-(K=nS~cyyeD&nTt0xDq z`!8R;*n70U|Ke4D|KR??lYLkmz3AZM!zd;eOvrs%m||-|f<WoDCGEW&b`FK@NmxZ> zQPs)GsNk|D-#suLy!OS<oTGbi3g|3`<~!TwLV=m844;9w*z081!V=gCu>7F`+}Gnq zfb(%9$hoNnK;PV$-raD(8xDAlP$y5z@jLRf<oX=4HS1E)bnM{_P`UP*&W=rT+|NN! zVu5bQynlHA0Pq9S*nhn7_U#D3z9kK3XNNyOyZ_|*TP))h!BZWDs@f6V|6V?)ls5F^ z$r#pRJ<A3;Qm~+poI(+I(B+!Xju{;!Ym^iV`tF@9*9aC&Y7UI^J$mhDa%EtQA;@M4 zg8-C(E=22T0ixeo0gPZl@8@Ui?@W-+fqp7lC)5g|2s=bL>t#o1taXZ12Hic&^0Uf# zvLb@xUiS&;gE_6Rx;iVeDOTn&LtCdPV*@R}D~d^0jfM=XJ1XB%Kk{+OJcJe~4r>jE zLqI2^BFCkize9M`qZ~6u%eu9|uw(cF9a!jv?LP+uz!BatarEQs7-PT>qWxzr-p%%Y zhD!F$)F(k5VhJY3H{>~7T;!O8qDKK0kOLuC`N~k~0d%mRyzWte?$2h^HyDMyJQ?Sl zH)%wd%IfQ`tbm>7UJ+Hl$v%*+7wuvA7rIrrP~G;yS=!D%U|RTXN)FFzvSN4wzwy6> zkAAZ5{%JKI4Q-TSgRlOQuJJ%r_i}jA&^-?sJn};e<j{b4gLoN<%E-GE!!?W}z*C*4 zx;33eJ2o}fJO2GxEuZOct2m-tA9@vChmjf>fo<Csv_+MN16PzH?saVDEB?op4Z4`^ z`Z3sWi0HXxzVqz_oLsAkiY$O<z-oH2V5O;13yQE(_tgpb=PTU&NkKc~2e7~F|LW1J z`wyNzdi(4VY&#G4JxgXA11{+m(m#iF@{G!O@%=b>AV>Z!rtGXmQ5|q-4JKrl-k2X0 ziwb#%faq!Q<`|^bIc#*mOo)z;3jE6Z@^_j)e<$d%!&Tkw=z<;7W&JYIZQDJJ%k4-J zIKs&ZY#=Q90;4VlIZ8TA_Y%&R#1|=P;pD*VPfbmdG-P6X63Bo;0npBoAE6Q8u*iY6 zR)cr`%-a#}J-+|^t1HfC3O(^`^57cg(x1F%xGmK4X(3BX7)|YQnhd?$YfkmTH<tFR zWe=o;F|X~Fj~!4%V8|`d7s;(}4)FC%Ku~N>|E?Fhdmvz>w?IX>U176K<niSNF6{!t zrVYCvE;(o6AxSCTRU(gylNP!WVfR!a!v&^A9WnsLQxdUWo>fDWCSr<VA`9HQgqt(7 zzzp<cLJY%s=nn*2-6#J)ZrJOj49Ln=Z&E$7-c5BhEdiy(lj3$x)VsM~vQl`Kc9yr3 zTC8zvfGb_ntSnkpwJa-=j*h4K2_^{<M`4P=0nBy+vfN}^+L$*04={U##}^DMJ_Is` zb)4OQ`GlmsW0Ah;ILGo+P15+FoGno78ss=h?-VqeLX<0ntWlQIT}Fun7{$6}IKueE zI=&pguilxs?*I=5gz_c}j1_V!SBMPgecl003-N|q2NPuEFHOXCYnLb)Cr}Yn;_}4- zp=t4Hsd<)-D7ZhL2rf<3L_j`WRDI?$ugs4Cy=f`Ut-L3MM04mtZw6^KkbfGYp*tZn zlvOa25ul(L+WV)-W=F{Y!P@c>U~AiGA3yw{DobQUHx>6XCG5)MBp8`ihl6MdDc0!8 z3d&eT^_AH1kfkjjPOB_qfCk<2F5X)Nm<B2gs7{la>qSUwaj#bb(J%r<Tox~8fp}=k z0x@-{cR8sm`hT}=IldUJzUb_+tGH+&wu<oWNwB^Eu6ll{cJXNC^;PFRnwMx^_V><B zfZ!Xg*^SohI$AT-H}D3_>lTD)x6kp8lzBT6R<K$$t3OdWlCso3hPD0){^xl$Os6M{ z@|HD$xg^UZ)eYS53~29`mm>w3(oQ~$^Gdbk@)Ex(`Um#LPRAWQ6r6{!LQYw1a<T5i zR)R#G?dh?G1aVin?CJI-5Ui}MdVWNzJ51Jz^5KkEFZQMXQJt_ME7QI3vvM_0_=>H; z!~<`YY4lVVy3tqm@Bs^)bbRQ6;l}xpHJTITcl8W+6YLOsR~Sjv7CtI!Iysj!BY?|w z5@o&<sgH|3IaYwyHysR%Yh)aw>I9h42+qx;0_UO&EOZ9_9j+lW@*sw26b0Cd<YBwA z3!Q13ofZ>wZ<1C>1x76oPRr47O4he3z~Cop7$oa*G)Ogzp)mJWZ5rDwMhV`rBH0Ve z5YtHf)7mDOUaycV&J<0?sln~s5Y4nW1CBoI>)!F1z~)a$l#(8o^(+3cQkoY>H@~#j z*y%tz2h;q0F=FaA45nL-2c!A0a5TT>I{kGGd(*c42@f$K6txixU9Wdpzu!j@83jMs z4y);~m^N4Km?EA{>@~o^Y%Ygm3N-`LU-0=r<+n4h0n}E*D^0ka%U)PD(G6Xsp-XDT z@NU^<?(qZfWNUOWK~e#*Et8b<3@8c34oq4S0pLQrwql>kWe)Qwuc+}Y2Y}<xI)+_v zeGeHCh+hdVF0;WA#|yBNO45za&=(=nf9Q5fsFs2Ri0<NjS;KZQjK^=UA5icwE?gF( zJMP5zoGw2?Gcq_p4vu=6)N4vIWafj+ybOrg)=QvZXr-gTIty^JYrq7bEO4`Jnh&0p zt}A8!3LUvwd$~_DX+{^jsyH}??&Zy8Q_B>dfTnRbpf41uEQtli>1wATC5A^>Skcm! z?<2|nCKHmfzrp$5pKDx7M{ph-abu9vuntWE=U9?yVNK;)yAAfO2o3;8qi56IR+DE_ z2rY#zzey5W<20-B+2x9TNDroU6Zo315?C0;C3va?Pq#57WEv9_P2^1m5v(azzmn{H ze>{9FF}0O82EWAa4q7NRA3T|v{@rPwA)E|PmT8?4ONv5=eKkel62&k5)UbFg7y~02 z%9QHKz`jf+2{n%$ysi>@8F+-@VaQgOKR6~w^ZZA#?@E<WwoP%Fmj7M6JAD0M@x&8a z%3;hrC2hx(A#9IG(m@$~lz0<cmB!QG3;Bwj&z?Ay#Z=MeFeaW@s-Acgm?7f05_3Ln z4{Yh*+N7toJx|~AY4J|0vCtXJnz>c;QW~NN7)~0AwI;(?<8(FVz4(WiE;BYB&_Bd% zIZ2;I?EPYNoIPZFYL<8s$Ai_6Sefk9ho~bEoc7SrXFQa@LG%n|9w6fTaL~p*#G7^1 z#fUrSrh>MZHy@cYydmD>jbuAi0jQeir>KUEX$>FS9NWgHrcW9r2|fv5O_c*TE#4Q? z+V^V{wA(_fuud;s<}O|k8Nb_I@w<Ivr&(U~>B<g-RUm$hV%5t&kEh=3tK=&do(W;% z=)Jd@ns%Gq<GiD9Un%L;(2Xv>YyNOv#1=k@X4a+HKXtB_HQ31F90I6msUR{b9PutK zkC1q4lkY<#fgws5<DdT2PazW=kiP`(<5LpIP%)B%*hM}N6UIoj*p2}+y=fG<>5Yfh zc;xqcd(zU|tL1b)oaD39M2Op#R3Py<=)E2W0!@nnp6X(yh=ZEQl}^XQO2DB|muMVm zlZa#Iz%<?z<}Dl-hlWPhj$yG4bVpmEEwSB~KrKBE)M8tW%6)Jj@JYsZ`I(Mzrq&FD z&@=L`ScC~vKYq1HPs7!s+G@oVrf^0S<9U8=;~Q)CmmWrWGOnh@OWFmlwqV*i>_J_! zhK%VzuxJ9cyvlf<pHoxUTk59v6&HF#C)3v?B%WWB6AiD_u~JT*B^kh&Z5R)u#;z@W z{SuBx7@02zCt;*CvB%s#Hy+@ktc}3*;<!~#rg)1)Fs-7Ah(42;V5zlZ^Ql?f)^iZ& z!U@vC*dFm8OT{Ivkh{Xn4U2kEjEDJn_75_5uK8&89@<Y!=1)LV<F`yQW8io+m6{4X z<FZTV;TT_>@{4>_Vm^=tiT<-yb8g{Mvzb`6@eo4k+OKVwqMzzo4^JFdy*@N{Pb+{I zK{W<1SANHw=!$lWvUaWjIsZM)&$H5$8~_`yClzG~P2Da2*<@P2cgg6keASh_VSbA= zV~<NIb{sYuqN4$mEs~dI$6D5LIfgFHP1lOITz1WiD;(+sJ2$sVH@8ZEXtzp@N6qZh zUi?_bx0(*W{lH~a=tfFfdgi)<3ihD^x*eE7vOE^bEK?CE%`qwxG}{hEirpO=KyOw; zL0~2H$wvR2GHduILyh7q&8Jt@%MaLkgF@R+{yr~9WT__BC2rkPx=7k_FjVwW0ly~2 z6b%H?0!>yOdCGOzq6C#^O#SFR9V&qM3ZhXc54Aplvd5_EYV)1vtmW3r=whkdYKmAU zgYr#m%3?}fE^m+RzpMEelVGpXUPQ0un%{#mmUPn-y9XlWG|NXL(b3ew6b$e?t6A#C ziEwL(iaX6r-EEW^Y2rxmWc$%=di&us;}*x+Vu0t{H}6>&s9zZdx=l&C?Y^PFSiD8x z9GOI5&Evp_UjzR={rL_3Mdt1-=Tek7-_G9fYed2;SdSU~$5^^uLpHWtq)p_$BhnBD zFc3A8&n{n)UfqI5r!1JB0*#b|tRFM3jy^wuFuvmoAuAA5XRAf0Ra>4tqd0@k=Idw< zxPDv~(|kHOUD(dJOKRKEL`kww%G^0GPQ^{B4kS1sqM2T(MPhqoBOmgc9v+5KE|%z( z**%6O+#%cMt565X44F0groA|1v(e`Z{%CWxy<?kC^7U4IbmQ}R!acOdCdq+Lcpf7B zlG|CAF1m3K5I*PkjJ`6T^sgfP&a2BT+h_AGFvKvgBTkmO&R*#zJ3%e#60B!OrOUTu zcee;h-*|7h(F^5DX5%O0V@${*(Ln6$t|0&~?E__Md8qOwADU~Eo=1c7eGm77Z6QkO zuiuXmhZsllA%|bTUup>(Opwtgh9D)k71K!v)Gc&W0JvXN)D^QH#%gfJO=H`W2H@f2 z)i$9xdVQad0CdQThDehler-`O*qOFuH2cYu8J>$VP97W0oA3d2JU$~!_2aCA9fadY z$D*ebq`1yyHO>m=@O33V1kU$|jIi+Vx*%H;{538xY(YL<h>shqd$g~WoJ5w&ct7yg z0EZN!>CXbahh6A$-RWs`f|WT=tV(C5q0N+ccYQz(*E_*tu?v)#1N|}t7K0_AJ2ze} zH(o6_UM*MgYS~vYSdG7Qpqi5GQ@)0ZC3xij+hjrM1pYg87mpTw)pSN}PO6${JtNLk zz)`FQ?pPcXcbrv|POn2`4*woivsrb9?<ts`3H$^=J8XZ|G4b^3V)lX#d2l=+3hdB* zaR;SRv=fIFHR|iWJ|m3;v~ZEuvps?RD$7~c^xb>1<7suq@1Su$TKc)C6B$jV7zbOS zcetF)c5z&u*g!O>i&^c2&H+?V^MkXG8*GV=6VA6(Uoi1k-Yb)F93B6?bB84hylHq} zPG^|D1f`Tfk->tT<kpl}jx0U8!;8GeK3ztl930r*7tWftFTX>ZT36KQT2b`iD`9@m zMiOEwVZuj8&B?JLIEQrEo`(UbBZq%hw+8?xoZt@gFb~3Y;A1{`!7KFTx1V~D4?9hH z=9Iinf=R4)YDzapu~WQ_*c==H!MEg^<<Z3i_(NkpHR_yftOogNDiW<;#;@G*Fsk8a zV@*xi!^n#ZA40_vHFVP2$ynEkrR$fjjU*0q4_vht0Y9d?f-nq1{a!U1VOTsz@6o57 zRMi93hi(jw<1rvOVF0c?S`-AQuzQV#x7}c4RQrn9E8yE$gc$s#?$?QuF^eu^IHmr? zM-o@iebPWiKg@2i#l8#S0{?^k1os?Mh}Vw|#A7!Mn?$i6uV1?~5a~1kG(k@`y%v+r zz3i4>HZHQQ=qdZ<ZaCLFjWneWB3kR1R<TiB7aT3`G4m<Qw;ezRtmux7@k=5cd^;LZ z+*;f|Pod2BXw->o{^={+gF(8Ok#8O_(FdK#nzD9UW`-Z{LNDljt5%h(8oen6beG<S zxmHa{w9|LJjXgpv#10H2eGPBZDrlHyxNQ&t53LvbTr!#?kXC+ERc9Iz-^1@%vni_; zKSFc(PYTu^C6?6CB-LaSJpPO-1p1@>1_7nYX{pTumR+mJg!4J(psGf*a<ba($V?fS zx$IoC*vt&}@6jukKpcm%x|odyyV<5KG|6<d@{Sg(F$#^6Ko}Qiqd$x7PpEx#k+Jp; zW&p8wkcQ^YwCqQHJL~i@heRSB`c<rSg*uDz0E#?5c=og_(1V-~U7c|U<p#lra6U|R z(ZReH^uOEu)j?Q`FLorpHH+%odgR|zH0<qcLuqTi-_Eww&oJk(+~PX<ok;hNf5?zi z=J{zgDzt1351u&y#^%-^cT_8I0*PNv*|R2m&F2&MKI(>K0+mjXGAf&$SAs{eQZ{jI zd&rnUTs%QN0AEc_^C9fg9o1(sTh%4QhCSe&Ni|xG%5kB9V6?&g)0~cTgZcD*;e)`> zzpAG1UJ4GO)ZLB5_@>ax4{W;x|8~%awu67vz%&75h9NlR+%hm(n(v$zBRHMjc~#|S zd#Fo+Ez;e4Bc|cQw)If-8sxytSzUe)IKwtF+Rt!D$6Br|u7Q0l@Cv^JP4&MBVHt~Y zj7r7K5ax*DS}J-Iw(1>di7ttEZ-y{Y6(d^%`YX}|k;q|8nC*PeW_<MP@Wu+e2>kce zQ~d9PgHGRrY9ew*do@PBT7*0M%G-8kwy|yQg&s>yh=vcv*<`lZE2o1|kt(qdGqIih z+$Ww;Lfs}p@ydk4CogkF<Up>J5pp}^jbwle9s8rfV4xQwKBOXmY3bh$J`jnIi@HWi zxT_eV-zk%Zva!(-G67&b#pnq2=Rqq+Bogo04sLyZ_w&tkp>J(Zn>s^@kfV-ZZeWYf zBwFs?z1wMoNe8zGXu2uCx#k0p*Mbc|`zWtL4kVO~i+tKimwG5v3na9$(O%?hxsKo5 z*bpOWyw$x4^?jq>y;1L8DQFRXzmL;A9j=3vI9pgUhw&F6&*L2yc7p;e${eDU$meFE zj%@^UAQFviZ;xqX4<jj_=e0e0k8rn|l-R(uIwRZ;Kkxsx{z(@pq6U&U0bI?G;M{z6 zXZSh%|N5!9XJxG}Pv;ql6^I-jGZ5d<;I5=d-jux(3wK!t!O1UxI7hWP{zDUIvQd*Z zz+hWK76MDG&iiX-n7A}d$!b0w2uh3-H!T>n%pTPh8fB7tSNIS}5N9+iI8k&n$bA-Y zgnbTSEy&2a5B49ff4(<Dbp{DhSo@dp`WSxx<ysVu>d&Yz2Tt_NM<;;)+3A@HuC$K# zsG`>{VbSDgz}FrZfTD98DAL!f0{&Z#hU*9A2+%rIKwwnEX?_mhKXX<nMz*C}N3T1g zc1DQ}4SkT6%$YJMByx?msWfxP6$YQAYf#fxbceZ;M=q(QnQv;{wHComH{Hc5^zx@$ zqzQHb74blc<R=%3YJZeNj4@HS0;1c!%m>mfyAQwtV5$0N9HwLxGxb{zqU67W9RbfP z`c>)woSov`F!x7xN$=*~+DT)-v{B61ofP#=uC1(3vBpMiQcb#ectsiYx6n6OOzg(9 zjhnt6O6#>S4Z~<<t=0{LE%;jNq-rtcz3i9rOg+ZgdbWw?9$&KA$NGAs&fUZ-Xbx3i z$JRh4`dLOX`f=357wzd%DN@o;nymc3$0Vko1i9%4NUv=86qBKTE)#fERzKdwzO55! zF)o9sv$)=at;cd_v&@_KBEx$L)GwFsMt6=z)zO_V2Ajtlpa0@`>*rf{hksJ!#a8jf zpC0|m=5S;4i@SgF#m2^;{OK2PYWw0&JssR3#`D|yOKO6Is8u#HWY27uXG8_FU889q z$NJ3qTJ;nIjq4rB@y|VCyRh_gp&P)!bdg;sjS96&YcP92cU&0;HSR?gC<Kp$uL8pf z!O^NN-xs5W+3xIWm))j^ZX)Le@P2U@F|Z6EI40ArZ9ltzjK}aABtqN<aPRT}g3yP- zSMXbL0S(WJeWcL3OFq)<*h35oF*nKw?kI3p4mqqg^a;Zn0chBiDVvDl9oO1w#8)b1 zC|?)&%8XO*HOJ+&o_(8--ig7SE7@vHV>#nJpNnSOzIkI^9*rnWcC0`Q7%vC;AqSFg z?e&asvVIBe7*L1dCt#v)vTYqHNn6{c__aS_p_LpbvL3#qIofrQFr%w&mI*hz?KpCH zt%=<V%dcg3vaMjJF@sKC+(r%VS|la<;wI0`rh%PM&bs=BIY($0!0Q$rE!P#Rb-o#p z4piTWmLCBn(r!qd7E-WRLTipo_#M--ku=f^y1q&SnAg#e`i3@&^cHc(nS*K5x!HEE zxa|P@$NP@#V%_aHHFETNes+c&eLi7MEeoU8Tg%5WXU%)-0uEq9%X~eCi!XlNv><Oc zQ<LfPjv{LtIivgGf-ic1jY%h_&MCivAKZ?)cB&yELNzYC<_1S$NcenuGH30T@X31^ z2yffSr2Dq+PAF1QmW^t9W)IA$EO!zE?5=!0p<X8D%G?rOq+c&S^J4hB7_Q#TaYK8y zoi3(W_2$o&&|&H?vURbmfB8gh1+H)(4eu9pRSty)Ro%6BfU3|!7lI?QDkR~HY}pWt z7-&J=1U27F#U{H+7Yw%0yVTjzGqZN!Pq8fn=yg8E<n-1str6}cwDUt{(*2iDE`iyW zc)F<GmqT9p@Rm)!Da!@E8RAV{g~}c>v&cujABchfJ<#y5nBGC3;5!dc{R%7n6*~*k z9S(GIDw-11DTrq=^;$0K!mJJ7z-xqHQ`w8F2y{9mdmG{f%d3qe$TI(+y5M3j5jP}E z1{<qfD?k%io>ceUFbn;uMBbA^V!y8}LoS0I#8Ah5;Ee2ePsgpgXJu=VUF)+PFUZUW zhn{V-b98PyUnh;y`E97eA;*=!+tI?`gB)$$Y*;ND*2nHI2{HEy2J-C+4HDN8u!`Hr zK=Bx_UW%?}eSORZV4;acq}dp2&4#bhu>0RGx7}P&Z((s!>f-I)uo8bzC3%2xL3y8f z6>d}jS0L>J+QO$+jqS9or27RXLRPgVqCkl`z|M}!vAfR_9y2Z8m(|?H9;o{O8r3~D zvLS2JJi{mj)0+ERk~<8sD~TUZpzyA(0wK?;!i!_hKgqXXrYk5KGA}%Z_XVEXSrj3I z5ys57r^Oh<&5>;dYdHm|v~AnM>vF)~jPhx+I=Zl}6D-W*DV0yiN*G!M^5HgJa6Z<Q zYmEjV@WP%di1~}sAH`oZj@aiBb=Pn}Owu~}0G0*xPHetSVk6KOf)s89{p?pv#59t8 zIFuP=<y+Q;Bb{{(TDu;Llcv78ZpCk}Tu}uDQWcxjDp&Mh_r00ynbM5Xom6|aACY>- zsXLO#Wpj{t0b+;5i8bI825C>mD-;4yPm5#YGAUJ3YMr#v+mo@>HvNqZl;H@bx~qHN zc*s#z-57NDR;cKYOMb1^qKA*hYObQXy>OLbtx{jr>$P4mEm#OYbk%EHhyY2zBJ(Y! zl+nob8Z{qMx*~-g-3NKyP$R&7?ldR~`mz`mnIeTvr_`}%Ur*07m-L~VXI;}tq#$Cr z9_CZwW4W7OCgK+2qq(ccg>?&Vp&QJs>6g?(%8^3%J{E3NDLc~*#RA_5ht%6kiMhEy zt1i1lA6H)MuCDIxDr;5^*<A9um2<0MJ+v+(*PQHh8h0zL3r|#4gsrgSv5({!7>Q;O z(E@E6f{(CZ1g7bw^`a4QZ`Rc$1z_5Ffv*t`g7ijeq0lEeDf6)mCt?=`snenE$!FSl z4cOxDO2*tCWo!>)gc^C<Cu!EZNHVY}17ms2vL+_dERXG_P&M|X$f3zNr7~T7>^;!+ z<q>}=T;<p(ZHBiYE|i|VIBLB)F&-%aab9osNt91bK6+U2FT7){XKP7n!0P{hCkD$> zeq>tMSas%~94kiqyTAPJTnJAxeKL0_?&8-@L%J5v{0mfmotL#Luk5if7n?4At=ZtV zKVl-UeSrBm3P=huj9M$e1WM186k!pV2)m%(x!vGiD;S`b=AGe20n?~}VWl8nrOc-k zB{t4FcrL5rKtjC^Q5kWxz<B?#BuV4Il6Lp}_8RduBMQ(9yimhtuD4dsM^~>BoYc*w z4omBY7$<gGlCRI3kixadem?2}C?kUV{7$r+4Dn9Z1r&Lv2<Id9+lcSFkbqd<^|gn( zh*djksa+Rn(r)T9X;rE2p4Nj(Pj1%F3hPIcO>}EX+;Ni`<5r3vZl)ODmVS9J+XSw* zwR00L=Ek4k#-HFS{sbm-FgX)u=*5m9W9XNO(p9scp*jxp$|I{JG+y(po<S(34|VXx zx~gf1s$<_r(S-7Hp3#vv7feU>4!s`#Xk%l;167Vjp53$sLgC(C=6D6H+^c67Jsw&5 zdX2|G+ZL4Rt9Ltz(obC$CN314GP`K^#!8MMt9L<ZvNPOcn6(c+))bHO#<I4Dcu4g0 zBG%}S`VWkCGtznhp8B}aA5>k{Q?#<uqdDAs6YAZtBASsyMQ<Qf)qfLLPGvIj$7>z< z8H=8mzC|EV*F&;)^P&PAJm{dsn11!(po6Z^8yJx>077^B3}q8AC@>Q(89L_F0d?`R zn4<eB8ahxHb(x-@1CRXX&nAX2yqr<y{h_q{#-=9lU#F^=zK3<(``efgNKJ+NvYxK7 z210-yiAPV@A{Ip`w&PgC{)mf98&{*N6e9id2~=LObE!5w{C<2Bs<;Okt9UQCn#XIM zAH}8$nJKKHIal07=u$J4IPasHdpGM1%ppnCP2_wHBq<jIMgnhZdw32)3#2F{T6461 ziXz5l>`>XhZj7nXVD$ABi0b+1GLzU=OKU|YU8aV~(*fPm`<cLXhExt2ZZxJzFPVb8 zY)^QhFFs5c2z?Y9yBbm4o1Kqr)tQ@7CPMh@t7$c#w6~4%L#dcV6{EkqYaiPeg8>P9 z&=B>Yy=wzoPWt9o?uxD1Z>HTKoo-nDaYgZ3Kg{wel$Sh#Cxn~09=-3w`Jx*jD30a9 z_||0P+2m=OLJqzdRrxRymn4XKV3o&lU!tpO7iBQ0;yy%mPGok2;WWUGyVC9OmyX{Z z6n;z<lmR4AN5laIpR?JN(p=E4*Xd=4uE+fg*hke*vNuFq)t-L4@C!-yStym_T==AX z1Xk&c-)aawq%w|^i%gMhbN3guPXrE@`P<v@6bwb#jbs#41n`cBDh&TyFPDue#PIgv z6q#_*OT?>)C?G1+2@OC^SQd%&5j5e#t;C?sw+Lh$g(X>87l~k)V^dI*!*E4btQaRF z#|W2+rff)#zKy04wZ=g~UR5&+)ewQupBw7FI6kh6xWA?@e~#*&;%<81gp5ek_6tPy z=XHU(D4ud4j6`^i;~e$jVJHlrdr}~|VR?Mq6`Bm<!VynZ(ejG(Ck-JV$=rd)`@ATI z^}}g?4#$8Y#r2kIjR70)+{yMhb^u{Op1&tubm3thi6mW^7app>Jm@`%s7KtQmwBsE zEv1yHL}1}?8R9^~YKLg(;W}<Y2d!X2$0H13BiJ=r1;4_OJ>}rhs4E~6y`o5%*Ttx9 z`XB#e*zr(CC4@;gOwW6^+_mtAPjqT+Kq@{k&5{;n$uu`k@wK}f%_$>A&FPOimoRcP z_LQk0-Jprb$aloY=2`_tQDS5Tc3x==#0UVX;%N?Ctuy1yF{KS#AnMm#x((YZscsXV z&j1LupMxBs9HVwPD`%ZLsED`M)%@g?Q>OKkVCo7<_Pof>F~J|Y+3;ZLV$_Frj%K3O zd^AKu*ZDL-0ThT#)ayZIi6@V%J<$=zh8~R?SI3XGB9?IyF}3feL|>dw7yc_H()lGi zDIjj;z`l<Gf2q8dO=5vijVTW9${8l7^EgN<Z5<e0)n5eacn0`N$4oR246K5!r!yFp zgLj>^el<3K_nLxj>Icsf!?K>_v%%?oD%@=-NeFVM-bayS)DRN@A4U(RI@J?WHibAa zsFCDaVU)?>xlB?~%)QdmvXdSquSj&-Nr<`)N1v;nadgp%Q5<_(JWC*oCQ<>Ty>m<= zS#=^(v^vM2F&oEw<QIKtAbPh*@08qcj^J>3+tlx6=Fdf>#0jG&>oJ^_;}8Y(nzH10 zukqrObxN!^s(V<uhed-!{2er48CJ-g_h)!Dhm9aaPQqapDH)@gj7V8dAqg<JV9x{P zmpU&;BfRU$CzB%Au?*#e@{EQf2hei76Bo!Y)Cu(>(Q!T$R#WjbL)N|F^B$;&wmj%M zPDVY*ZDM0t^U-wcp8UaPK=O6nQ)5##2EJ=Ldr$t5YW&CTY%{&MO!qcT_cq(Qx0&kR zrs>{hQ};Gg-P<(X+e~!NpN98pVF!yfj5{j&7*Yj#q^m$s$lj-TkCxhhD0(G*4fY-n zM2+r=&TX>G)6)|?^s->eds%?wcF-I&xEuA2qh5B_8b>C<i{qL&<jz{O?p$oC!pHJ& z=k<&f%j}0^{ArwOf&^O$C+d81K$b`=tx&HK5Y)sTp1mQM*Wn=)TMCNOyMR}P`$j-J z`e&r=VBY4l0lV_?3$1@@D%^XViex3FTYNRCHHx8@Fua%y)rTpqVo=9ZwVjBT%@lAo zk${M27t_7dV(<=#P>DLd%yaG5@X1+tSF~6s!QzN70SB`_-s&%`AA6T;&dl%hy&=~W zEi(I=$NL@bi`DqabPBTcdyvu}u}F>GCv|5aCN@xy!bvek+k<#b;ojq~MSlve#AjMS zEMa-1X~sq0G=M-nCt(ys!%ir-(+6f0;g>Pj&;n~LE92uNu@~c)`MB#32Nppek00>g zMM@@IUDmy#ysE@vRMj_0!b6Z{sfP`p7xZ}%e~pD%4$qWABw_e!n55~AQAs3(f3Vfr zuic-o{}8dup_xs$(`95N+iFm41z)>@uUD8A>g*MckRC9Z(WV_Bg!PF|Abw$cIJR70 zBIffgWa}X4hIstmR6n$^dJe2_MY7W2H+9&7G4M@>2{&%qjTml#%6K%ZqH(aAe5l$d z8yO=?&=uh=N-cq5>!T6L{?jU^M!JoZ$|HRVW+r9ei8iS{{5QNIbX7Ij@Z2vDHYpo` z?kUD+r;>am!b279D{Sgo`>o)vCXLwM=pCks9J{vSI$9h3EJVj`mDN>Y)PcC;1{e_r zvaPCa&iNc&%K}cqcYOikjiTv7_A=@MtJ9YBz}{hs>h=C;lxR+tyb97rv?LOU3mw#; z7LV0U%ajXV4h=ILLO<7X9b8e-;8oBKepA--eB>JJWpY0RONy4)xEdFjm*vtMD%3WX z>vWSP@ySY)mdn>knqh05o-GO(uz&BQ%TW`$9QK}*mUZm+&YBq$_h>Bvj4rR9EP+^@ zO-}O~=KV^b?WJA=RQsKQJvBem!0`Gk!j6W91v-sGe7dZ@9is!ppyKecs9oT^6vp)_ zbcwnZ=^u25n4F%B#}d%rGW=kE`)D`nmKXjL-n%aW+xes8<KvEAp=#l%x)C);H#=%M ztgd<2i@U7JT4^`CW$N#^kHpnw-LITQ$7KI9vr02zCsdd6ujsF~QWY49_$-NmTxU^N zXUT=y<0#&d7Zg#BhQiKJ#zjsQgzXPs?SwG*-SyTmdAG&I;yh;##8YdE!bNQ;t8QHO z#b_85Hyv;!crO$*-~{5AwF(hG03tq%=CYe%S=c~+9Tk83mThC8r?239T;=tWDoW^9 zO2ustSqd*~=46p#Xij0!aSbMUpAESALnR;gX>as>yfZf5VHpq!OPgTQP(>cTa;ows zj$*(|(r}dGSLvGNE_K7yAPI$^@cm9}wbu>RUN=>fru01Zr!CD$8)5m^b|#(8n!T?S zShND`a0MnfrK!LueL1Zrb=PrGg?y(_JoGvQu)N=kH%7J#bwO$cQv>K*piu}g_U(s& zUj6;H-TExZlBSqiOUM{xly>Qu+6e|^%FAx5$~K2%mne<lX|a}Y1^%JQM=p)h^$lJ` zF3pX%t8{QP`R+c1(Z~4<MS<)_W558!YV2gG7Z=juhrQBa@L|>7Vx{&LS88vOYHv#M zB`9tsQ2ltTYdf&65xVF(On4uX7lsJ*mkps<RD&;{zzJ;&$KcI6EsFqXEzxdlCw}R5 zBGCt~($WR|@4D$o%jKi{Gi5Rv;m=>?5f%fw{fEG%k%mK&m#*AUZb^S8i8#HkNR&=r z{Mwld2HrY<E6m|)^OzWZ@G~{1ZT<dom44s)kvkq|LM|?5%0oyxe^1h;WA;VPm>zL+ zw6zQlFYcUXZW!1gGvw@8jEyy4;qHR%3cR0XT^CY!^A{Vw3eoLqxbw379^gq2XS=m? zR(2wNldji`K~irCq!~EtKLAndI2qgVpb}7Ckfs+iBl1OPP{d@@yFV&Vu+&*O9FBBy z#mh+E!r-=#h9^Zj3ayFhpf+fIf(yh$9iCjtzdqv$&V7VhuE$g5pCp6seQejR+AZa8 zLP@WUL3BX2_Wh1u)|Wo<&5o0o6fhK%IM=1OY%RA`M?1p<)vH_nu`roe{1&#Bw$4iS zQ%)EUito!}x`$nByeeViaX{oFGaxMr#xoJV2rZs1VD^v|r~114p~CDNvjwJo5og)w zm8i&3xVr46^+$;t()hMYGYNdOtd01HA}?QI=qYkb4Z3;2n=6hj7U<Og)ju|`LO_Ey z9cx5}O~Sp(^xBxuJ|xgc?`4w@H)+D`_gJECPm<K0+9@jjGHKyv9?#8iL%A8M6GhU~ zFlxu1VpImFhJ%cLv==s8vngnJ%F{n%t2QBk2C<=$LEzKMB&?Htn_Xt?YM*+1_xmM> z*cQ{w09ycS8L(y0d~X$A2>fnI#p8sVjN3J|hfFh^%QnN+ACFB^(<>N2<FgXOr2Z}? zgil2T9s#Hy4FkRMHnb{NX6Q|vnZB-k<in-<_U*$*-@JXBT&!rpe@?6N(R5l(yPX5d zH_mKb$uF23fx^xmNnGr5oH<^eDYwG*eqM%JE@C6mg@7`}|3nu8*Vg$jQNx>Je=k=9 z3eb<B^S_F;n$TBEWa6c2|6Xu2z~>rYZ*4{=wOhw0g|`!hF&h%S3{NpvQH|=}lGbd~ z9|38$Hj^sSMt_A9b>$r?dhuDR>?NF;VL95cJ1)BUy#REx``d?iMKMu6ftM+2`UJN< zk7VBuuR!D|!^Q8wLA%dZ73^4IBFAU!>I8@@B#^xo%7}=?J4k&ese*#aXV#Wo=)t%N z<!4Z|JS+OZx(4}N%;G&3$9oJ%(gYTVA*Lklh=JC(Eo}9v<M_#h-^9PEqRtoS>UsnE z6yvZzF^nFDVyd(J2neaa>@exb3l(Q(IujSocHJcPZkua%)zN%!Q&6R2%&*{+)hZ0d zKi@{@h~<aga5;af)0h6$a{6kJOPo-(TvxUl%P^0}I?V3r!E1A(Z#%@{vFSSJwyR*j z0bRZb=X2bU+ct@Cf&9J%$eWbR_5bl%zD!sNU0u`Qg{+)hZ0_iv3;*ZEmj6}%baAPo zBakZP!S}#<O%Y_rE3R^Pc(G}Br1*jspGB&`YgOQ4tL2K9AfyVoJ6$sfmso9+Rq8<+ zI1nrrQ8X-9Y^%8vMz4ZStYdx5ucFXn10gg6{yYhItB2!4z!>dov2_i=Uo->u_3S3N zuutO`o7);^{m{&O5qw_+-!HcG_b=l5!S}E}Qok;-6+yrS;g?9j@LKsi1tTb-?B%Ul zrBcMT()L9wVJ}*#`l6Mj7ty`8jvM#%RTN6@)zf&dK$HUD3!>Nyq}YpQiVd%uCp4f% z=rM;VMwmpb7f7rZ&BPkR#JY>En46|X*%UCZX_QSPSY^F7jk0M3t1Q*zp22^_OAv`4 z*o&s8ZTMeLlMs~H(k0-<Lcic8g#YzkF6iaLe~HlWzkW%2x!984x+|ZX$hJ3;ZEqsm zt`yl8s11f#Ono1wRU#i0?>SO!Yay2Zp%nkofd#7NTZxut{W&eSl;7kqv&X^Z#<T7R zbO|`%zA+E$P-de89iA}zi+M2xxm#d+?Je^iBGpB<Yw9|6Mj*yeIQs@aBT@kd<we7| zvT_lh;F2o4<-O4A9nsh<ol5Obre{yTSANWOF%uU947xItkmY73dsb(HzA9NhaMun1 zgIu1+vii6N=QOdomVx0-%L`&2D59U{@5wJ3gTM^((O^EZkq1TRA^sA;za_kZ0e@22 zU^6b$u(UauR+t+Qsbp|!!R2+<L7cyw4(KxsrepkbX2sdpW`rgr9xbW5$~eKSCTl{^ zA23m0-II&WDY1e$W0Q3VEN$_dWOy#np>GqT+OV$ozD!~j2|=kYG}>oVlvbG7jb#w2 zs|XK0v(^$z<wiib5fE+!gp~vYm9|4?zM>tF(s3MBvl;w@Yp9wK>|sUuGJr6~C!bw4 z)e421AsR-DVE!p|fo=NmT*g#PK6{i8-hCFPss>`Ye@y&^th!9ODC4Q#s>y5v%Nx@T z`cPbWVaH8j%dAxTO}Tu-034Gp+Lh3r2<sE0k)4czZET}jRc-bFob50u6_(=1k<(Y& z@Ow226&uRlsqK?0#*a2nk<sITlwHay0nnOHp#gpgV=JQ9%(~p7s;PH6l*<)p9Vr|L zT}b{9k=7p+m~bv2_7w@-UHN_tDSNp7!=mIwK0u`HFPX;I*Z-c#`9?=^BZAzBAXm%2 zgZ$7Mj<f{y$0erES^IP{c(B0JRM#A)mY<z~dd{k`n5B4^a}u7LmNK0vcD!q~_KI<g z0SWScQc%VPjPQi1P#PLchl0;`6|9}GfmFDpz$6O`bKpH>gzlZh;PYR)z4NC(-E@H$ z-2L0KJ2%tP7Hn>@(}-=8KAn~0ZWIl^7e$c|9m2vss#M%6By*L>r)D0}cz3UqSPi?u z6)4TBR;Vv=bu~B%2bMxigpbt*9(ilT25imhtk9xuC4Wmt5~jFs!Z9vqKi%qTZv!<F zCASyGBZKnLe4CFv;T;C*NAPpqiPIYHX6x3^Fi2<^v~A=l+jw2TE<4256Zsn0#mW>8 zwJ9=j4WC7rOny)H#3$jT5OtAo2v2$!q5~WDK6L0z$U!nlB_&$wPQm?>$GrkzgP+wP z)37*=@t1--{+Wbrk=qXZGM-ad1Qb|?hmCHz)J<OM9{F`gl(Ck+hn$GEE}RVpx1K<D zQ2O14Vz~C@BJmP+aldYO|2lZTY8nq}-DKwICT1|ZEkn5t;|c{}I2oL(AeeftFpg0# zJDNK)JA35m)RLAOWSPyNJ2-Be1Eky4^Ua}=FgjD%jq9u$d<;Un@zUYp!jAPW<tu40 z9u64GFB0WBZ{3*h^iy^a)RP{=O`S5Ro!WNr!^O6PhXzdx{d>`qfE5CjLs1ZHjKhYW zFX75)GZRa&o4}kW%D<6dJb^Jlcys>rRZN=+g19Py5(tmPS0pJK{WZx7AD==RNGnAu zrZUoHfS`0?eC{PFu_2&1KHGwj^>oFCekG_-NNmwgU-+c_=JZc%l6Vap3w%C~lM#e# zF(2ytM^ehQn2@RbI-nH5*;Oee5tP$B+Xf?PQfVULAB|LQv?@1Rl^d<f$5?(yl)?A( z5Bp_UMM-B+P6rqk4_Bb6AZyrZWNz+MZ<gN8(!0XaQ@)Ha(Bge@K*qGF({a`#@AFYP ze96C1ds+K#pGJN_5lQNxB?P5gdU4!M)HX&<Xc4qVmmYB(t*a=4OeMAiX6=}3#J=1K zO@pz0q)Jr{<rJLr@kuPqdEloKri(48o(#M0Yj9=>+H^oLzZog3IP!h1K{V+BKcM`w zt-5mA)R>QuC&MI4p6Jj<e)pipEs5rWkqTBF{FC9@wL9<0Pj|BC6|$SbXg)01f9<4v zZ@lcp;S4h<lPhZ9cPPPNI0qGqyS|LeS=~v)X|K=K6{FX2b?2y5tbtt|;syc6SLQMv z89J4j7Ic-{ja`Y!HIDS|*yZ=~2kV5r&f_PEU*`|g)G^IVyLJ^yJp=Mf*2)-E{|fJv zY0b%QS&k=jNw(7sepB+Dqryzha82X>e40=2S{d6SE@JMvV}4;SQ%q4aC+)Lh?uQ>L zBS{wj6Bd#hJkL--nB)Ac5U*qWGjMuPxv;z@!vID3Xdb8Ee3hSngCDwHRpmwR9VS|d z&8};2Jer!kIoG9jZjK{2GSQ7pbQPIsU#(-#Lm#D|=cP_^$XPydiRf9e{fu_G&QW#Y zz6^^AChr~(px$<N*g2V&Lp;_@s!=}eys_mbQ?iHtx`rV<<{;e*%swPJa=DPscfTYN z{6C>>)|8f%=WoqsvCWBRT_-6V$cqUF6;QkqwupyiO_Vg5Rwp<~uop{0ayB@UM}5X% zEglFqs>R{o-S}f1J^u5i=Qt^)2AyT~d@`w~GY;TU3_kCBQ0nsg0%81uVf-S1L1h5l zXuS;}dbBiQ-m&krE{nmI0*Kcj$<TANYT|;;srSTwEaJ>tR>?)LN$t1Iom8(`a}~d@ zVbn6$S<{0Fhx0LRz$BZ&VqkZ8Toqon<W(L3)~%+v5`bNRm^Cmo(93f8;ZMe*m}T7> z4d-E@701WLV78|Av%#pGd<*hMb-s<R*cj8V6AlA&R>1Tj-^1b)+6l0V%U=Q1)VK9< zRGniOXFUL79xMG}{TFj!SCoVT=&XMiV<e)-8r{~{e=(hP@2;)&u)&jRG{pS1SX9q@ ze*6$MhB9Xj*m+JiY8bPRw2Q}9PC{PE`x1;HuMysRU<p@7{~m0)Glu(pF|D#L{dbD= zTjO@hK^t(vQoC!ah2az=v4^k`J{jP0?fmYq{&j?K&Rh)SaXLF2VQP?RKES}dATq<O z<^W+~-6u-Qa>zRwt(_|m2KJR4qck#OmUO-)^sb-%?yvq&=-==D#ee;~fAL@a?yvsl zcYpog>%$68{6U4twln%Unq}ZWJqQ1Rp*v(gL@{$@l&X3^+owH&<6;0aLfXa6op;)+ zkBH7Ha^{#8IdUX~OcR1H)O?`($qK8N3(%tNxm<fd=aNFIgYNVl1s5CN!_>0dT!u&5 z33=XAplI$W<)MXXMOT(;J`Jm3mI+Ezi(R-n#XdQ9iw6k?y>oDbrB-n^a+8s!9GqsS zur{nE<&jQGvSM>b7`KEkziAEj2>*6)AU|T5lB0t2-eCK5I?Z07L~&l$SD$lS3kE*k z2vg7yFM0_+xyf!HC9t<&9?tn{{%dj4`O}S!RIx`n5;4pYNN(d<Ie0RDF`wye;Rxt# zIGE1Qj-ntB^(nI3pab}rEoN&g_|UlF;lrCAZ)V{)zzpqu(OWJaT$-ItW(y3=^DXay zTX+36fmy{l!|E}Y<)cq<J!`UlBx73)Frfzq;ypRayBj_Fzy8nHI>_`pe{}cffBO0H z=c%M+N3`MX>DpU&@Al097)Dby8O~;+Kz*9SM&`lYyt~n}|4TyUN)JZ3N<VM>{^Ix; zR@tw8fhtwxb-D;f*=<IoMkXakNnCsd*IvD7wkc6e61;A_#VA>U9B~Y&RY@I%)5@LB zpgE9LXLuIt(P5z&fv5q!<8+qf5NM+l0e2E9o|zV<q71YE0xU-D1dpt^a_@#8Y+$ZV zIj&3Mx=wC8yo;hQNUg~AP0_5Ll~Z6|qw-y0Y8_M)3_A^#`~7}2W<Da5KpjY(_;?c< zPM08LJ0UTmQzCBR(OWMJb9&xDQk1MhFnL-0;3kD%dFwWjsaW07Pfa#&l1;ch+^|Q8 z2HqD=*eAqV3_0ic)Cs7|BUq~n%}h@^7m2HmvW%1LF|CGifi9)_bkR@1xj%v>>jS$M zoP7-0Ap4*IyDA)yvxZZ4l201G_<TGyW2!{hO%Y^7fq=(ERamJwt+?6CBnI5-;skAt zbhsi+A*~mvD@4&6&s+Q!k3#b)E&-7;$5)WMZ4<#D`)S&0<kQ~DO>Hp^tQVgZ79>7x zON;u}w!H9Jx{7GD{=k*qm~NFHmS@Gdrp|68k4i$g@Nu)3!O=tO9vH;4GG65JCBFz1 zDr($^ZxV+BXz;e#e^~9*XQGhcd$?+#H)1tB>}M}Q@~$P2pLx3)wLnv&;(akX9KVq^ zy0|ROl3ZKy96&ubM=XXjg8h-ixE#cNYNIbrZJ4I4jDA8XKv!-`H6T31F4#W0OaN+% z*=aS*z606kA!{zayV6h|%x8Rnso~hdf@nUS<_kQ^!d7uq4zuH81_R@+&^et40sGdd zn9Ww?2hf;0ui`@23E&ZSXRVi2=FB3#ain)BCXCM28$#}tkIwT&&Brh5x^&kb*HD3h z7$v11DTJZza>k`sZi^7YJ9tWhZ>f2kLTrbGk0E@8>SbDZ57D?`o_qzeKPm8d$`go_ ziBmw-!+Koeo&Dd|nsA!-NP&4DX`~-4gdB@!tRC~$Lyx7p!;L<0l#Tu`-URI=_RzL= zYFoUunY&>#&+^FxhW=ndV(GBBU=;#1)1fOY)<W<X{&U#z*&ixViT`bGM8`SUO-VsQ zw-7Tpi}O9n>~vZHH-qKhosUZp$|mJSG5Sxo)_UqT5oUbl={z!OXXF7dkP+eyQq9W? z6{xt+cJg|_V?d>*9);0Ge4|EVaIQuHh!r))%Hvsi1SdcuL#q=p2%t@N_2u*J>B0we z5BrXAeDFuW;LMNJyNI5|86pVirKX-aB@TxmIV0||Iz&8i0sR&E6wgw06^1(-A{PN- z#|IT`&GpQWK2Ma5S0-<gU>_m%qsY883csp^kE7<E6S;e#n9$?1=*p0l_<^?&`795w zQMhrwF+S9ClKE!v-0&H7cd&ltdhp`nqA&mJBnq?k5;Z!beEg0VL|u&Z3Ey?v$xGh- z5?(@0;V^36$Q=?{34z=P$f8j?qvDFLkK&*+(%K#@FW821;~>TAI>lz3_0I>so@ao3 zL9fswuMx?YJHEtqHo>69xWDif{8S7h{8pH>&`6mf$?qS2&~^yY>#rr8{7;xbvdOfp zV7FZS1Ydq9A{-kq!cOv?e`9q(_uN47B}y5iayG*)cUlhKEqd7*>LY-qm$g+|lY<w_ z)WMu?_VKcNO6TTr#SK5NbRc(^>X?XP?K_Hu-w)ty5uEHh7D6V|YF6PrqKXIqonuU> zRmsP&`;1^GUZ*_<1`|vf@L3Sc@o2+X(D{=oKXzLIFG*kunhjcCoM+s@59*xU2pUbf zp>0Qt-n2H%KU$p`VY18bJOB|1(HE=-B+%7Fq)qMV{X^0vH<?Gd!KuQ%G%k`olHsm} z#QQ9NhoQ@XY(y%|sV#~ofj2(1B8Rer7IOdVvlF2514HD+c@G_{2B*c~9af^Nd)N^V zC>Gf+&C&r!jc@=!!W^7dRgJ8l#E=@dgK35QJ3p&HBEvnE0x-*t06pZN0i{XEGi;&H zUB^Eq&;(ES5xG1(1y<?f!cK=PV9j%)Sur2_Xw&qJ&Kcc~gqtVSd;Z&JJS*ie?OM1( z>@}>~;G*TojQjvC%^f<&hg5DKmS=!ZxGFSr`KYcqXxVp3;`%NjhMX~{AW1*&3DSEK zRH9cx?PrP4p}Uv+p-h{I;~UE;yJztl=9`w`c2neP;zW3}I<v-hYZ0i{?azck`7oZJ z9Tn3{XCxMtII1c*u8%LX!?XFUge{3rI=iIRG{U*rPgyT{09Bx<?!ODkmx>c;xT`ge z4Hy(j8mf@BN=ruiBX>YikPg3*s%1^5kS>7=vcTL!tF682>@2VOJDmB~vhDC~*amLp zDuKgu_~~~jn)qTo0D5_R@a$=qD&NjJAlAS?es|j^^Zszi+=EO&DNYojJEI9R#XfHC zjj8L4qu;_IHMG2uX?lrZ>eXhMn#A2j+a-BBwu9J5i?2nOnIOo1bF#>bB_G3yJjz); z;|xCBt$tv@*<k(|?@HDz7NNASz{wnrA%kcR1i?Cb#Rpqx$wN+GWOQ9jzrrB)*~O|= z7g5#4s#V1pD5y?vR|R2iHUb#@dKDlcL0V5ZM4%=}hgM&Jw==^{{TW_O<0;&K%SLJb z8qm$+L)7O54+-DErs7G4VvQFR(@{8<Fmn^ZiY@#vVYiMKu36iqUf0rX5M7=O4=S+< zcE~D+;VI{Sc3e;}YZS21mXVBOVP^`YDl}k*5+DG<f*zAmH6~0!4V5(_t&6{y7kFqk zxmRp$w8<`q#u%QWQgHf}`1oPEg9^4YD-X_2p)P5VXfr{XnuM=XClmt^D6<tthjyC3 zFWe0(kKw48ofjZJ5*Sk0diNN<E?4?{RGgG!wAJj1tq61z&Kk$_5nbEDpHmM4S{DuJ z`W9PRXMiG2_C1j|BvLr7z0W;W?TR!?@w$dI-_P=iA<?()Rpe80%YBP;N47!2s9+?` zKF;e?_#HW``=e?cnKwEi=J4|!|3%mpwn_8jHblN9r5NA4YNX6Ahhd~ODVWHh<>Lgs za-n+9LQZz;_U&adVyT<}93MDwB449)XoO5^*VxFe!cmr(-)4sfSSa;*qNwhdb?5NJ z0eKn9U|qk*6bnbH`$e}MV2fYvIFi)DDA6(De-%)Fx{H{}$u^AXPp6VG^_K4dW&5J~ z9lK$LgAF{vlp~Sx2jx(Tq2z$Gg!A)R5WO(gADNQ2FTjmnR`rZ;cV`%)al~g6`xy>i zEPGdY0z<utHP0j64P86AP}ernG+JREEr2_tA#4VFJ6RhVNcm{wVh3_%3WYU4ljX|q zkfZ-}({|g2L(&IUdZ$uB;}egZOGTtXugw90e)Or2dg;rt!4>6{w{g{P%)!BH2FG+; zmoN*_Pv3>qV4&IY>b`K=d+v)ZycV4bzflJz)1lD)h8*NvS@Dho;x6mw`SVqv!=QZk zFqnG&rfz)Qjl)Ji)$lI^e@WfIM$~5+x{n*RzV2ngHyF6agDScm7WK;Q1d&Ve{!XmP zaqA{!0Sd0i<%wKhg$u7m@1~u``2Lk{?Vh4Iv1{Y(>D$yDJ!`juItjKPBF<_91GTW} z#c9qLhpeXC@fM!x{c`mmLOP3-o4WK)qA**j?21w*IchU<Rm5qArEtuo6?9mu>K8I? zzKLyP%|1>t&Ssrp&B5gO)gt^5Ha6@`AbnzA-7i!iYdmoUxl!H?>7`$z7gZdFPIi9= zd*e5E<2QHXH+SXWs0WXJeem|>{f7^qJpT$--Y<6e^^=1~&)$A@|D}3`BYge#>7(ag z9Xy8DcXvF0=m*x~z3ZzwoCM}c;od1Ibc*|)?FH^wNOuTSs0v{@m-H#ji>Z`yul0LE ziJ@H`yZyPYJ1re@C<O%wVLq6lRhTQ>b8HAZyPXaVOF>wB%7eZKw6tH0&=sp`TM7J< z<(km>6m-HZ1_$+(;K2>h+gWT8$^LkMH0s8C&n>n2pbbHX0#)5oVGhzaVQt6GF*3Lj zZDZyd;w%`N-RHcKp$RjldleJNRtTn0KMjhIWUSg<K5f5a3Y;z5i6v>)U6J~hsYIh= zQl(Am*@r`67hiNzCMe>qsEX2?WLa|lP$l7avYhdCR7J&OB#<aVH-V$-A~_sUi~?yR zG&1+?rW!WdPbv7Aes-5M8hWCrUi+V-vBKKTM-9fuyh>4=KsyaZuhBB;zXjfm_hwza zYSpy&0$S3P#sIjnJzzf=#L#009-NQ_L&%C*c-=G>ypLE**|n?NgWrc@{tKATc7^Lp z9=Tx4>IE6If#y48;>K55u%{W~s)*FIX2b8ByAKx@hC{M8gkVTUwViPW4igwpOeUD9 z@Or8f>h4S>46BWed;P?E2lBCtpHj6;L#8$&b$-!JtYLw}QJc}oQt!w=%R+B3eDsw1 zOTsT3c1zcDYXuMOuCur8-rrK$EhY)l!Es{b?g&P-P5<o~Mq$Ax^9wg9>eAHZEiFxB zN)ho1)*C|$FA4U62xrcTKCpAZ*fy!Cy8aZ|INYiE3u@ZP-BBENCk;p_?`Xfv-{<8B zMPl;brsy;G^{eixoWcN96tmkvN<j@rCP5AN!MFC%88;M-t(tSgEIY6EVHsLlm|_we ziJ}oDf;@+dn97B*)h%+E!B83e+Z&fq&*Ndq-fnD>d5TW@V+k~>P5VQqO4KDe+<)oM ztO;YE*ripab-l#bvqqw`ZK`oEEo_)8hx6B8E<-HPaxkism(^%-;?fQ3XGvanNym=Y zV%RZG&z@ke=Ao6@LnjeCu%0Zg7K7N5gLkLsuqPiA8QSq#i(j{-fft=y(*RO(sx_EP zie30LppLKsN233zLObft5$aA-&OPQ)%)GtOugcMAyJ4RsO-^G^Ere=3OF_W*q0DHW z@S+T!R5Ra`0et+v8V}e_A04_DUBmo)mOX@`YPh4i(-xgBLxV+pCi&pqBMcEkcQLrf zc>9xWDXPfZ$@0zlTGqgP3T!vKI!T-trNJO|Nw5$xjJUVd@XyDRf#bBm5QT6q_jSa) zrZJ?L>%}3K&F*Cv**5%d-BUq86WZGP5JMcR7TZ6{PwtXevb{yW9#=EF%PkxdMVF(I z^y>*}Ew_m6_&tEX+W=!C0GlD|7WK%1C3)-cqP32()u9?YE@<3KAVt(24dykbHaM$L z$w%RfbVRsS3D)M!8TAWO+&>WyOs3KxM`@zOzL&LBVl+ji#E!$yN2<T+&^%P=>NTE| z3A?Tk1e)Vd&Aml?efg?%Ha!W+go?;{S)a1%lmmYd`cTKHJ==qLA`e5?1bkGyJ1fv& zBv1L<h1d&tADihXtKcWaTXcLSWhUlVj*Q66M?a3BaE}{J{f(ynMpJ(!O?_w(VNBi* z(ROV2?#9NO?4TsmuI#M#N*cqtL$*=d#ZW>82Cz%RgJqg8b~pU5E`8*o$<RmzxKMGC zqYVQ6AVWHel2ce66@Rd?bNDAd;tTNY6iuhpYZjzH#QvZh)zN=(bK}irTo5+c59YWa z>IOi_WzR_0AC5^f?EnFGF5&eT8ygzP5I9NKlY+<*3{CsI#%etJcRPdmQ8{>FGNAcZ zC~7$F_JFKCfGr^<1Sw#P(W48X+?s94tl=bCibMN*g6o~&zSA=R2`%UL@;PHH$8#3i z4wCEnQGvXSmInq|ajv<YA*Uf%AkK)HiBB(1IFdY>B#Q4g`H=`m>%y9bEwUVXV)xG| z2pz5)^bMhyD@5=-f?5LLk^4$<!FwWh_ZWG-KK+JPf!21bP`5;)1ij2Wx4+nWh!u#* zBJp}^FQ%4BT&iu!snTXPt<%A`;+P}&s^^0NFvg+{A#Uxtm-z6?FxO5!GBHKJB<M9T zHtLpll4+}TrQ%DA`300&BNWr8k!8J)tP;s+XrEBurJ`|AYkb61cbCB>lo|9sP~zbe zsWPxdwSo_66}(BODdVXos*k~X-HQN1S2m-gp)eFa48-kyz^EV&gWDb^w>3VueY9@N zzd^6!v#Is&@vTLAZLEo0zl)1v0Q)qreY+`O$|;bEk<ol>oEU?WzP&mKDV-XxyVye( z{<?2qbqvp1%14rEk!>+Wo5&+IaYo6JbwR_n(OAxL@D%cZf3A!?LJN+Q8K#H2X+0Zx zN<-au+P50d`S@_+H#Q(Pge8f(!)f=Ed-t*tC0h<EQOa-lyE)Cb($h-aSV@ZyX;Wbt z@L-DrK3Mb(HS-T5X8Kb|nQIXp_h?2EIe^+GBp9+m90>{y$=?8Z>+7ij;GH^sZcN(b zBQL>t%+nw(gpWspa5mGwBMWuGM#E2|Sh_i|-5l6%4s2I8tvV{lLrSHKe~1wkUDE$_ z3toODCOY_*b2r0F{6pT6Vd^jN`h_f1cMQ~rukL^K_TcfWNBfUoJbeh?KbHzG?;kvV z`}F>UM^E2Ae{uiW{ik@`=uk9_&YQT*qh~K4BciF4F`uDWvvG~DU%q_w%7A`lK)VoR zjinnKN!0z#TbTGqDDG_R&|NFrS09xpCxm1;%3%J4MfQE)R`^P!BmN1raq#H*0leF? z=+bv3kE8?S`2VipTL$!#ns!C$eAvOmB4w{4gIl8-hVFc)!nS=!#PFR8gZCZ9Z8}1S zs%kKpPcT58ty)*K<i7)&uih2k@yZ!e#Af!&qW5MUtALh=9sYzJ;=lI1ZXMFmf^A`J z5&r3lxPYb6ONmn2@cwo5URwX`ACaaxMDKIDr^<FiAA9YCA!X=jPQjjQV@EH^Nvye- zE~6YLtbG^XXh(A^P43th_uz0fWndin7!Rz*T8v9MGi#ZNUG)kH!h2y+?q7KtUnWy- z$6S)I59A*D8KK`$l$s#QNwCWGO-YfAR0tQBB)(aOZ1bvH&GJ%rfrdcVPQIDt&UL%M z=k)$B`TX3IP+Gri@7b4Mw)f4cURXV|&%XoP$7^6J@*HdLheX@>RISoKcZ78AIHT0| zPPprcPu34}mvZN{7y%=_^Qy|v&^TvW!5M72oNl;WYj-->>jg46IrNNAI!pVTAOvK$ zy9QdT6<qVBX{)8LIJJn$ujK9u)}*!bt(6xHoPII-0%Nic?IF&9+=C0Wtd6hCvcH(a z?znB*RbqV7zw^a0v&t|xT<ELB<8CwZCHyKFzPVCp8wt390`TAPgCN79C9=Fl3Zxri zGw9!VLGtA#`WVVfeWrBT)x>wbErH%FPMQ~hnz;OUE>9ch7IUfcf;yh-yuM&`AA%ZX zjUl_B^sSJKc*8nH&6nYG9^dm976ulAgVI=1Q;2GXNx8IfwH*S(AYX;}Z+5-WqoYxb zd&GspH~w<njDnJ+lx6@6ArYJ3*0sr4i(VhP7W&4v^HqM%Q83n0L2ZX6%2%WzH#22l zwQA<J&1Uc8B#7%^(2t4l_PkGBKjuM*S!67)iW*(-42)s*ie3i}6~T2pPl_hbKp}Wb zhzMOdV|?6FQyG8#;fJhgs@2-NQawc?fs)bvjL5?v#47nBoM;TDF%(Xxt|11Azrw`# zqeUPb7?W-g9^KB?MR6-x3;`)`m?mG4Rf1onEns8XS|JQb3#~@to2et(tE4xNc`y~> zlV3l36utfZKmQ;9_TT*F@BjCI>I3+@|Mq|V+kf+K|MuVhGt4IRAO7aQ_}hQ?&*0U+ z|1bZy-~aW$!dJil=l}Bm`A`4d@BgoV{dfPz|MYi%{h$B-fB7Ha+yDKS|I~-S(nz#Z z&`-A_-IoL_fTcmkE_LX))Nbk3Tv+jq(En&Qn|3?qQWLhULpNFd3mp}uzX*J*<#dsp zs;g0t;VPa!dj$p)5)<Ns<Hxn+d<`}#8*;8qC^knn$`nc$yLcpqqjnfp4hz)3s-~7H zB!&F2&-*P0f}pR>d#WyoQ5T<#B|7ONlIwzKlzgGfXcwDfwl@r=EK|UrftU`ZV?EVL zGf!dem=H(3sK1Ksh7<9;nw6MFgX9D|;i&sCGnr3SU>S<kMPZuJZ=_GrC-+Z*qgLno zc35J|8W0#_bAWM0M~iG!)sz9Qt|(rigqB9}m;xz?h6Ljdo=o$}sSPs4Pz$~Yt~tgv zMR&;d36j;gVluRo;$*?L!={E&!uK*Xb@X5oC<=5a#4xKk(4na@AYxS|yuda?7}o(s zo9M>VcwnM(zz%oly?B!a!DR928jQhCE`L(tHLR{*Bx?t2Jlz}GsZBHJ`6qt|#Wz}L z=H;{+P>f)hFESmC{UneA9~Kz4f|m=S(5yp8(y8s$=OBSB9G>{qQb{q>4@;KHc4GDT zRlI?0`#+QJXCh#1`#;s*;q}v9Jvl0(=nZ9YzGnl3080E9><VRAIV?<?LC(dM+P}ZW z)!pG9b@&`WI#<#E4-x~%2;|t44l#ocbfIGcM0?yIt|p6dDndGEXUk`>)}zd3D1Mmk zC$G<fk(<~S>Q&;pi}mUWJO+VQB&t#4z1=MiqEgnc$Ub`5rC=!&rha7Ne9SgrQWtND zd&vM>56k1@?6fSV`4pNP5RdE*@g-UHV6sgN1y;|G2-_4pd`t(sYE*=26(|%tB@XPl z$lU|FKceJ#9QUl6&hjX^Pl4t-2}E>}72<x69R;B{qQw}^ARzpCu`vIxHQ4o&jHQs- zRBTS+Lsu!uX6#+wcvblxN4aXMCRPZM^cdYC>tyCIus9F4c`($_5Edw=2Y~fO!7-~T zt_UWTL?~AiZdgvZPli|Mj;2VMEw%>6r4t9QG>;LLL{>4iY<Bs3(X<|#GH={DIRzD( z6`-aA3TQw)ho;Dz)Y)4@H6Z}BjMX&*=$S?X#>nh236tUB-FegQZ61pyQEzad9nF(C ztaQ4z!3OW=SF3xH*u~9bd4BpcCq})UZN{HlIMQWuXY#gg*?E%&CefbXD16fhWDf_- z2*0-SmeRe#D-@Oa3!3dWQZtCr+^pA)Lsd0pexjQ88ZP3Xc(w|8fONS%HqV06Jv5Pg z`*wErt!*AJE?3(W!WY5}ypc4iW^JDX{YKb*8L2P8cq6VGtm&v;5teSjk9ca+)S4fL zIs0nX)ls6luTM5B^a*Y3OpQ4Dk$Lbox|orbsw<858<-7jGAqx@@9D-0gLF%rZa5}T zIIVIzTO`&YK%}^sXQ+?OYX>&L{h^NpCT98{?M$sSYHj*}<;pdUTT&ZE4$IbHtrHt& zE)OCyH2HH9uFwtHTO)^PYx>y?ffKvNuGHo`8h2af#g^W&h8@n8Ltf*yo>pB5v3lyq z#)Y>6k89>qJu7A>X`c;P9sM^h+Gu)`)1(n5Ykb!&aWd(+?^0axTh7rZr76myDE1S} zz+K=ZH*Fk^Tu=aO;lG?}pH<K{|8}Qwtdc%@*A9U^fC-p)A(5_)kBQ$)sfXCm4FMTd zgORp_Bq|9gWH0-j)R*iYlmnPm8Aq6x=d@>J`c_ciq4gh%4CA5Lu2@W~v>``^aEC1& zOWxJ!r`>g1KNsoTx8bu`t(f?J$c}=$Elw7Dd@iqTBGm-d9~W@~<?u;)cI(IZp)Bu2 zHOU9%49B~q#pcf~pOB=`Lar_6{WiKUaCcp2E>pIqiH4d^iTJqH8%D@TmwN4mM-3m^ zV$@VQxvFtt3kIIr-0<S21RFIOgRXN#Bv`Lg*!b{-Tuq88SsSn^=5anOs(G6mJf#ej zu-KF;1$JF*!o<dAdqK=iJoJ@f2|y3a_vMf^Fk1vbcpKh6Nt{{Es@DR>;XAb`&||n^ z8eHYqab@Rk+)z8YD>#tO==(V(mEbs)!i^GliRQWI`Gd*W%x&81H_qQn6K*S|ug1qN zjQa#MwlIIcwxMk{dX{%=tjY;g4_j99svEAL6?Q#OPU(mKIYx#`y7XkBIGGk~D)vya za!n-Gcskg>FG<GS2HChTB34EETkphoXkez!evT6vWBatoaSPk{M3NIJJe!wBV=Gt> zyV0*k!&d&s1M1rh<}9DduOP~5>OohG@D9^9+{_JDc-6LbTy=PSXSi#UPkgwf<2Sus zGB`M~?PIa-3WHkGSKKYiacg&_L5kwi5taZ5-DvBi8KOzUW;w}blRroUhcSA8=SM`h zV-;nM!2O2Y4^dC4ijXf6Tl+V_#4(YTtaQwGCQWFZ7!l|dz^4(vHU=@?Txrff3L(kq z{FKZzl~5-zOLVr4UMa5^eiQ7v%w7iF7XPk9W1bD9ewG-;cq5cS1)6hll)j#6OJHQ9 zbrQzB*|cK}TDYWV0+O;Ry_1m<l>4mYCTr)yUm^syHX~LhzTInA0$%=R+Fc9rrn*p! z(XXZ(8c<5uPk`8It6hzO5$DK*g$Cq`bo??Oo4IMP{e4xP#nsWWD^Wkv$xJykB98t6 zo-BcODInqoMZ5`OgN7kMHDuK*n0|^OR805K$s91L#DHFgVQ-p}F_PbeXs##%Q6$b> zogPWOou<bt6QicKm58lBENj?k<1N`IKL57mT53Qd8PEGVQs`+0LZRi8w%G8X`kTUC z=0|n6=wo)pFGHuLAAZPmkwsjjEoMC_Sb$M*E;3HXYNHhlP9yC_m>H$nTq#bGVupWf z8frPslN9liaqkcWr|!x)1q!y#F9J29*D9Czh_;Q{HPKT(8=T&!Sx$xB^IcM&Rqu;b zY&f)uVT0WMyc{M9Na&^Ec8u_3c^oFPj>a<?oWUF_`WQ<KDT~fy>ivS>r}XXk$bn5& zIM;yyy)UN?dY{l)_M<iM%x;_49{UT->jIDQ=)YN>l;dCTBnrG%1zvmPv}FC>EU=GU zU~QXWn%E5?_dN3LC1`UAO{M}5#~Jz>eGTGypXTEl{iY>2+@PwaLpFlrq#TqM*f(qS zfe5j#?`m__i=9NF*Fm90szdvO90Yr!t|P&#_K?)$KC?G7Ak0MOyErgR*VF~s2LAtW zojcmg_>b3b!WuQU43MOJ(We(6@ADr~`KSfcfPx<iE-(&C0BX_RJ&TK7n}4%4#2*d^ z$KeXo#y^+wX&V$<acF$^_r{x@ZFuwLsK+*fg@GbA1^DywPRDw$B!A{3+DCFFR37b7 zh`}^#?sjYm=p){KH}FYY>?1qmd0<hG(72afus{^7*dZAg8UOjZp+Cxp;^Q%7gUee# zqMf)&V|-=ylYUu0E%M{&_S-z-4r0Asy0P1b>X`o7qzf~>kO@U1`QAC<G~Z}aM_kF4 zjwCu`JCtaBRF2;@w`;t$UAhfG;AvIQ4opgL6UdmO@sGWLl}$=8+TfsFamiMcOucbL z+*bL(D0Tx7j2Qz2k13s_Q4U}m;88~~{w2_Gb^c;fjJx=%ry5x;;)-HaPF2J|uckin zosU+@NfWK1W$VjZ!Vy2gq=^sFz|3M%4E$m=+{+|wKmf!gdQZ?|+TTdN92GOX9zV|) z_tG`dk-#}YQxIZ|KQiEJ_3F4?w83XP;#$6<1MkHmWe2{z9wYORokmwEUWa<mCL6wP z=o3yk+M<^t(>jMXl|d=e{V@Ty(psOBM2ODGMclUfCI`|IuPK!_s=ey*2A}*0&bF4& z8VRsRT&ux0_+HH8RUVAXhFaS(L-uO5&K~{q$>V&7YA#QBPI}gfxoh}75i8t*Xp6)U zmTXc}N7QLMqVi#gF=5O?MW1#}WL%9EyXlSd;6$r^MG4=R?zQu~U#?ZUzlvn<QtoVr zD_D``HTNpJYJMyPDNo}+Yy{1ZM%I%p4mNs`eZ?nlK8subfVkwsFS$@9jmwjXg^Iz^ zFyk?~*WtOkXVmCuqk>62hdXg?(@<#!nTaH&<CQm+7ug=hGcAtMh^;ag?5o#)4P$Jr zE!5?<XH8tu?SX$H36yGK-tERs!rQY<tv(tvFn7l*bD&CX+7H4J#g2m4lW9SJb}scH zF}cxEP_dIoqtGF>@2vX{6HyR?9fea}TdPmpd5dWw7qv$PS}=>PV!|H_n{PD1pnL-u z?+gm{XGW#5hs1bLJ*@`$D7u%?)y=&<Ri__Gs|%FlswswMqxgQ~LBZZk!@}Ii4uN@- zmstl8p|x?Hgg{3|7uZ<;BKzeokKxdZ?5iG=h5>WfLSa+yw)~~@qs2A3%)+&@xvZ5< z)yl%Q(xm@bV$<pikt~JDn%uUdGOu25+b(<p<3BI{Eot>6I&_9aI5YlhiP>ohPMNw> zH-S9EHBEwVm|VUbgWU32GKja8P0WZaPjUfC4gpzi;qmW*NlSM4Mn%mG+p{e0Rlsty zvjP?k&@b_2$x?qjpW+Jbhdp)f;-X~sr0&(+#8+?+my#>FQ+@W2bgVZM&zhqoTE@uV z`S2q^!!&DT%88m@M0d>w%D|aq$)kY^+a&m5$;qJtIll+X*j<N?+{l4tw&{c;<$C0{ z0sTG`Pj3&HtxE~KikZZf>xUGm$xf=SfBa)%aT+>2u8yjD4$K?aXNjf+waX2L&Y?dc zXY3)AKo1i|F+Z_R>c-d4-kVrAHvjI#lpS)Cm*g3AM-7*Z*U^rMY#eYdPl0QVkaOJ` zR_Ei%D2M6eNDqDlW2pRqP1>`%IxAR1r)|`E81@#5G)XmUlD^`o0FOr2{ZFMpVMkBf zNlSq;bV_t<fA%qI4({<QB%U`oG*JN+6rC>46nkuv`l)iv(^WwgThcG7J_K-|Xh$YB z5c7H`?5~j#f?0Q>gK7X)jw+t;TnsDvO(p!aBY#KL8ifZ<;Yoc^#moAoaX5(@lM|wT z=b~F%0DcThqq5XkX-rk@)i+@%TOc%~Dz?P`2*Yc*rG1T+{i!&P>J-iPiF16>ZnqeZ zNETjKnXC3p<egy2okZ6r=~bn+7qY&yNxP*>7rh0AA3?5wteCS&jL7&NwL6iU%}GNL z9z|Xk$OSffn&KAp%HH-d5Z{n~*RB3!DUXmX@j8@vT<5;lOeN5incUtH3qsU4)160S z53JmxIL=8pmrl9?G;xe#+30%|eAnOvYLd$pDW7c}o-mM+PI_*E%4*cBnW6wQR=hqw zud;!S8X)<IMZQupgn=B25M~RJkTDnun)Y`#WCa2Nj{g&qu%>nK6I??QQ6@V@O&V|P zAPxD^U`#|Bq&F@%q9_U!-<%ry361(E*zUL*ucJDgPpIsBLpfI{@U();kT?k#hpstA zPpo=?gT{C^MgTI6c@`Lqg~xzD^lW=53X)<f3nn@hK;Q2-7Voi;-Vo1<Pk<{!#F<ZB z$E+$I__bEw=Z{Vj{g;2jV~Vld<K4p}e|epveTg(38Ziom``Nz*ukzhLuP{O>|5_S> zt4|*yyKU~4OnW#_SD&NR=f~sZmW2iEZKZ9@7WyG;+7PoJR)i#)Bma1nW@o@yzP79@ z<UFXHm{rwiR_gnsqZ5Y#&7bYd9j~lPUKmK*>WfE8R7u<yK9FZ6J5CyN3cE!)YYMj= zUB+bTNt7uQF<*%!XI5m4@NS-ubdo9LXXYG+yext>IL*f=$fe<Mhc0uJvFaG}L|FUx ze)cWJ@EX4ZX$v>`7k>_Djv!4743`T<FaZFX!etc6ouvbgMwKD&mwka`1irov*p0+Z zu_)*!xe7ih%?Heejw_JL{KGm%MAZWqSw7ghzoGB26T1=(SKN|koFq#hURpMlEaD1o z3ZhCq9K>5^LAjd}oI^kgh>4~ijk{$ut<m1ogNx!rXi%mP32Y^=Xqy+i?yVDLBAGB- z&$hCkW&Xh}hBBEJDDq;y<mHK=d;XYuF-Ch~mwNhC)s!?E7>DX-z(OK`hLI@1sf5h! z<}SiAQ8jZEI>k0A6*Wmo_HloXuaJ^l7$>=5DY%h2U@?<KbB|G6<MOx}cnvo?Ot{FQ zj*nb6Y&s2+!%F*%k&sv3Y)pl#?>I_yGBi(C{b%__w}tov4>twHo|8N<^0o<1Jd(kj zU8JLME^er1pLE~;UjP6A|Nrd0+j3+{av=6jv-yWpGtJ3FRU&~z&y9&j7g4uvaO$=N zbPtCx82TXd061NVJjpzXTX$nkHhPc%bwG;0c8w-cFVae+>GHBBGnuVqdXQ<?xcnJu zhSU>(LE)E(aF2+SnW(BBa(9t4UC2BqE*>5p9v&WkflomE`sMY!m%KnEHi8m%si>g} zTDjYR7jD2pf_7+|zYE(5Uw$i|w+RmxY`@>{X8ApBlOfLi9<+G3;eAF?y(aLu2e}5{ ziRE$wx97V7`NYgOA~;?#iQ9?%S_FcsIW*My4CX7&X{bZr<`-2tbc^vKhca1xJ~Su& z)S~ls@7Y^f-!qyX>gyUCbBrLLGuC&QXl0Bo<qJ3;M;GhE;y6Da&GvXFw#q3sJI>Ff z+EXtfY*U(zx{9RA(o~|s$R0MOtm?-ZvV22PtSCmq2L{FE!|JC}acOWgKxRfxt4jnz zqX=N(Aj|DmZOQ~_t>GHS6sFbF$ZVpS5BupPa1!is2|-N|LEnzyp&|{6?U;~9^YdAS zgx6P87#D;PtOM=>cypo%<}`a&pvI#=C%CpN`@~90{e<+?RO$7!n)G8Lkjc)>a5=$R zqfD%)&v33MHfq(5VPSf;>b|7jS`u{BMljX$Tk-7#BLh#*P!NAZZcrHdyB#nH-I%I5 zHGGL|{@P3=rtuV|w_Ak~?k12$E;K$(2BtD&V2Q2I?5m;7=PaL{HXV=7$}!~d>Sr5s zPctg~Y0P({si+rDL1eH*47GMR7c-3)Eb%m{3kK`8fG?meHqcI*8<hq|7<S6kdngCe zAQ#p0H0Bz6=)XS5TIPQb2$FsljOV5ak5W-9LZ)=G$1~WeSW){;ggm?NZmA1R8-tdv zGuc!^$BGff!>mGRM3HU9#uQ<EQ9Vy7=>k4RMxc<=&SZfB(#4hmG<^c*z2AWeaX>!L z>sO;8hAyc)!WKeA>q(2<!8rsCBgPe;=3Og%G>MHZvbK((DI-=pr;~g04?=!Mp<$T* z_P5{kA@#C32b%u=tnG*)jvLAn6<pMz(*gt$hQD*l?aBZ>5<yE~`DJxGBr7!Uf?Y}_ z4SW1+)}6P-OHCE>y8i&<{wyaT-j+zh^#)}hAD`IO5y7F-7;|9=ymOl%VGrH>grST= z)~q4|Q56*yNZRWds?Hs8L1@b260QSXbXQDD0VmiXJPuny4h-^+&Dyh0j;2L_953k< zo{g&NE#;(u!#AFgrAt?Li8)zJQtve-KA0!R?RL_Te;RQ%Q2}IxdmOs%yYCi<<xFlu z)Sek<Ds-SKkd7J%m&t}VkTDtLdj*>#;H`<ke{gvu=OB3$<3>RPXZEp+Vq6yEL4h&t zNslQ#m8mO#m!9xkcJMzsN$n%jzSrY+zLhU|x4U?#vIqUMF{+lrdQd^xH6<(kI#Ed( zIoVI7kF}Ah%r3$)SkY<^jf2@#9h6HVXsnsrSyjckCVUndGz|oEfN!t*crmtq6HLQ9 z@*em+%=XtiV&n;{UJ4`VF^Y<HUn3n%aL!)vEFXdyP=~@w!U&HHBf6w)JUQAg)`cbb zDek%5n9ov-vS7IP^l&9-By-j^&;v?C2*~teDbyup=OiHCbetsyJUc2*^NX@N2VXmE z3YUHcx5+3ufXfYpwj5$Q39)EMct9pw{P{VoHlO4N8?s3<u3do+2jjH88?=}+a;=+n zQF$+l=?DUBnWj*FhJ5w4O&{chiR>Fwxu&G$C@6HX#*1oNKnu7*%%;~E{rhwZs!|-< zHyMUBx}fk0KrOuk_(r-Q01caAzZF8^2yB>5O@^%I(2FkNDj1!5tL0eQ*%?1CF2scu z@R@Iptbmi<xB0doxS~AOR+cb@S0VV?nyI$wY?&eLIVU?Gk~mUi+n8J4;EL0~KLga# zY_+#{pRKS3umKxtR}imcN`a9v2YdxqG@QF0Qp;lwCy}%L<*U7&$4||MJHq-hcRUlD zVFH)c^eq}Im=hMt00!6KI5H&`3KgftNj^okJg6qu$on%hJH@m<EhnbGJs8UFGYd>j z4cl(bSj_=Sq<rU5Gcr2am!=^Y@<w+Z1Tn1iLw@_il})S-IdWjxxO`O5x)DnTW==9x zl|#mQ@M(bBZYkO9@>%w4ZlpQ_w>u-Dz)-9%-6ppOa>9wqveVgY(qCJ<yu9q@2yeHV zo~+psu01!1eYyK|)xgYEz8H_dE`MqkQE4*(O$<x}v^*T;moR@gO+(6Ko?T8$jtO|* za9kz<IY}O67C<v_Spd$(I72_&-pzJ)f0#W2ZCEG!)z04MufEvJezpDP&GyT^ou|9m zt2f!>S1+IJ?CrdIX?{J+wqO1lYW{NP<r5?Sl(<HUD{?L{kb>P;Squd=Sr~v;c$Vr( zF({AAff?TT<Q#kyPhi!Jfm94DI4f(I*cv(1$Q)H>HumZLcx+}>f|b57`^PEX<E5S3 zM59!Qbe`4Zqh7Y9y%H(nC<dR$hkG{nEiKY!C%|;@4;9ks2Y*1PQ9wZ(Px&8hpx}gG zK9N0IQ(1WP;DK<5H~~|%sdd96$2(DP-;_T9wFX4u?}G{qzJOUH_&L##S1eZvTB9z$ zZFc5sXf}|ReKudtCmdvAw#k>lQOhM?4iu!{zFY3)Fb?pkp`20ShtQ<#bFhaMC{=S) zN~-tHjqDn^Wq#zd^eqzJUe5PxN0V~A9{kd~*_|bQ>%ywgx~|&j_!MRMHJxbBZ{1Mj zr{3Vh4WEZu3#64VCaBZ(N0<b}37ZHSDu%msu!Ma#n@xDNn00Bo+m`RP<-2WpiEWt@ z!3<76`XkVS!I$19JuPc;j{fMUpsKb%dTt+nyuJ7IvsZ6^efZ?*v+XaQ?;U=*v-`#N z^FuSlr*K4G?QN=kpPjR<$2Dj<Y3FyZ0M+A@n{1!nVaoUe#0Xt@eU7^T<!flJW!U<} zJa~ala0eKbr^u%*KyzzS7J~v*PV^J*SWc?xHT^m+uZm$ywY@!^=GWa%$n*NsE#C>; zeLV&3+3X<OKP$)d-RuDQ6+iA^L$~@kY9`wdIS!yjJCYQHgjES44)wVq-vgA9ZU@S@ z=4f^fp;fB0;!F5?bNk_ZYtMLn#83~B(>t(t=>6>5Znt~${ta(n!y5-Tw{Gb6bF^rN zJbiU<+ca6#Pzo)ue85DO*q+sA@cJibhlm1b8GIKL0u#BB)eI#d$@f44f-f7bgAa`X zx1?^30F1tnML>5l{~cqp7qi>8oY$md=s+kULv#lkfk#y}D)O<%*vN?ETWA+`;GA~h znH}zuW4G<MafWzizPrETJEE0v;3gkk25vjLMZnye$S*uFPL&2D047=CAa2`YdsM!K zJReAWMmqe_#>Peqjy+>IHT>xK__&4Pfo|0(#5|amXE~^sP}pyiersC5whF@8ZChkB z3`=m=jENO#IY^Ig+j59T3pj(!{!F)Rl?(#PZ%D?$#_`7RKA)#ymnO=3!GtRh)kK3z zAEv2jD@*ODo4z>V0?vNEEKG&>R*_*{ARrOA(D)}%)(Hgm@tSWRM9)_fKBr)WkIMcc zg9;wP?fPSRA?OdApV3hKxIS3RA!b9ACVC0t1ldlW@dPHjIdmlPg#6Oo?`gfS1tZs~ zpGlUl90shsweJyf&;V-?vS$fUG0rRVy`V$+?t}Gr0*Mqx)fAROJjUSl2b_^%^#XGw z>h9=q)0DuGKA;%5h49jaI^nxHCXBs-%hR@BJM!ENH>PCO`K+40w#_99OYgu2Xxs)9 zMW%Wc?&0D6v;t$-iZze9K?zQDhoyyq3J%JKfN&2?c1^c3wu?mmb!5PJiEX40Jn?(z z?2*@Dm}fD#E9)<s0&NSVJTC!;1ajq}#hpv|lh<`Iok1jKRqvjx_uY3A9saZ;iD5~) zG2oK&Rc$3*gQ8i*4AbcmA@GnH>R<*|imJQ|^<{v@6UAQjC7%QQmV~E{=z?Bsb6Y7k z$JNBhY}gyW0NcJ1a{!U$eGoE?TxQv2V^BP98{###6Tk%FN-gdpCJVYrjnL?|z$#jO zDs8P`k$r>Hx5c)9U`_K^I_jF#u`>AF&mNQcQI6ziIHDb|D&&HXg-i}#ulXOEe>P&z z&S}%u9_dacx?{DWpz(PeDy_AA7*Zbi<vR+PpIYmlEQwYf*uq=_otMXohl#Xm`Jmxq zBXSrHNH+X5q_m1wmL6XxYl`x3KV<b|N=?pZ+4SUyM9jJ6l{}lBoev;SB-~}7R_eS4 z8z9ieO$<*P8Oj@%6S$|<WEXrWodDqH&1aABEM?Gq@1B8c4njZl1W_;`7|ZiwI<fc^ z^k)uxbus+Xx*DA)BOTk1OVOi7^$>gMXCz0K$JcGhubH(B>hs7Q)qYYr76B~%RA=9k zi;VbV6sMjxlD-PTH+N<5kUZ>Za2*CELlz7`;c=zGEt4=3zl&3H>ED(i^z}juN;xqr zJ$kzso33%Dm@JGukDP2n<5_A&Q{rZ7qrno{<D9!%gmw6Q5X{4;3H$I_GJ?9Je}I3% zq@no3GxiL=3rY^RaNPGk2hAmHiLW6$ANxssDR)<qL~B5<Jn@t=h-cN%p!M_J=enfb z-&b_kyZMas;C+t}pnu4GIG)3;(b!0^-#Zvf0JA-soaP@I5NveOoVetNqLulAlTAA* z>ECilDEtJ4NwQd438`;X9?$;l7M)@FWLFK$*sp>4j4f3}aJ9xVw^?fS_1RH1Y94JQ zdz@e}dW<JHOXPXOh^N>l8RJ^Di8?IAqCR{m;?FMfz_x26ww*<=vG;b1ih=clE8l#- z3?FwC_V;3o`f~Op;N1sOk1szjx2r4FJ<aP^mt%0FG6%lvHZ+s&DHz=tT|Znq`|vx; z&(ko@#3L1D^mfF$SD|S0pUOTg$1FP$veQ5A21zurYaqD=pYoM%#TV__SJ-7P);~#g zw7bG&cJL~=+1&JcHhyIm3{G`Ea+*c!-_A-Gyy0%UCS;6he3-H+!WW!L&IW`igRl|i z>_f%)_N1({A2M0+6C|`BW=H3kJWkcjZ5ecY=zyViIWrT=>MQGFfx)OPAq5f7ROd|| zZ1XV{M2E1n<IK|1vp83MC!*hep4X>Bd)M{TER=OyX(e8C=b#H-Xic$b^O|(8aijdg z`F4;AL|a}v3W>x#P0@vDYahNJ1au*SSQrGnW}PCB0IVO&u{l}G0KjF9tfS2rPmwB& zo|`gY*%rIdc`J=%3u2%)B!<|f>7uofot8}a(V!Cy&4IaJDu2R<+)97G@d)7YB4m1V z!(5=MqE`2T9dQxpZuXMuTMzAGs4S%kNW=_T+N^DmjJ=A;PU(VOe<?RBc6AhM0xk8; zas~Y~Pn#<}1HCu<blA&j8>@NlV5}>Zvv~D0TRJOOuy@}yuRbMP*wxi2FA~ACUmi4W zkwXYwT|`tv{g{d&`3N8;)uf%6^_!3#EmXdjwdGA}dzs5hbfIX)EwS2~S6VFl+D<51 z#f2Jm)0NB|ULUqc(1p~7K`E-2)$FC&UtZ@W4DnmmHTODB!0+9tQQz~}^d0{A$D&Ur z9#FydRFQkCw@|tjvjFRtrHCll2>Z^0^<J{eb?fJ1cHnX}Do)gT_iMf9*E9TFKM2ut zi$$#IasEM<L@gz`S{?fbJ4p#e=W-_~@YxYrZOKjnOX_m=^khT-_`$SK)T<RMSci&7 z`+bq{7-3b4)+t?Nw4$wAOD$<eHS;YG4o29bxR!K&3Ga*uI{M4Ih-Z=qe8I=SAHen7 zxA71{!P;M|hoo?uDd#!yjdkYEyki4t3_B_AgQK`DYoIxmi)n#6dbq!wyJto5Ekxx% z)RUr1iu+kU8y%zvkEf?7$wfR;;ui7b5+T>s`E*e5eQWa?zm<latEXvGcGyXP8;7Eh zD{=8*{2dSC?|TSe@&N9-_)y3}1SQcQ9mgde8;;@eI*U;eTzoIMAhku7JRaNFc?}1@ z=!}8&?J-whQG<{FYxmmnCf8vYG>WWOw|s$p-?o(d-1!w<vBg2QX&J~e`_6JpHrXUA zF1xAQjw8`f3JqxllT8HPKhsv&U>+l{TgW1;?OK-NBE+g|L<F=!CFY2XiX#);H>8-c z#`)gvp$1n9dz7F=GPO;$ql_`Mi9*cG8uj;nnTktPyC%J*ir;dT?Tpi8C7Dv%fFxWf z?!qfrDeu1mVrJrKTSeAc%-?+-LWJu=iybK(N&5FUF^<bG!j?VgOH4JJawSUPArIRQ zJpcQ|=P~0ITW&?v!M!_$!F(^1zSg(6(K0Uvze=x#^yH4T;PZ#@%0??<XFn^9422nt zGPeKph8KpW5y(pLrIW_$(33TkrJN_L<zK9$)b>D<Bak<wVG?hP>%_T_998k;2L+@u zD&huMcGKd>`|7x%N2AM!Lv^50!=sA_hR=^xTa)5#gsH~GZq<8P8=1Bovdys|<wj9j zcOD)(MZE})$JO|v1SeSLgJhY{44W;wNdo|8dfTCGH;_}3nxa9$gnNB;FeXsTjYbfn zfn$kYd7*MyEktilTM6xgQgKpw64tnm-6$7W2*Ls6T}H(jj0lZF-WeB{uDw7}>H&Ue z){+0!&tmk*vftjwvTfHZYs147SxwWHR;Na*lgEGqXhJMilN*SG>7iL)0tM963c&3i z@0-d8{$0b;cg5!|z1U&z=xfaj?F)!ELw+GXUAR+oBge1PPA%l|Lj`M>a1~JAO|BE` zflW^}V>JLM0-9nzd#B%rM`r1}VbfDn2}Dk%sY^24PL^OWb_JSN=CGC2Rxb_e!FA&3 z>%hcbmUm#!zoJny?CU&4Ot>hFdb&R=FA7S$3JPPn4R%}(QF269wk;iD*8$;Eit(_< zwDIt?VFR_kQ0xVTt9Ru_ZGnay=xryn{z9aT>~_1I*uDn52a{x#`s_R-25p<(cPcDz z3D63<VCT4|y+7Nnrr;JFNe>a-a-6aznLywm&Dnpdio^jhLjeLlil+ByO7)q7q87w= zgkE}>T~CVSxdMAuqnWWBA*fOqOh{gx8iZ;}3NakFZvGy{^Rpjma8*lYvV@Tlz}q7X zY=kOiiqBr3R_7493<E2TDZ6CzGC;pV)e4anCB}63=Qi}?Sq?h8C`~58wINTB7?l)& zCHwy1VHd^USI186N-FS3Q`R#a0R6Gk>|&lQ=od=3JnSPxKnpQ|!L<ZO;L61QG2xeG z_gE4gT}5%Yx~hxO@q8X)9lkx(y#?<#^g^}{pUK6_;Jg|)tkIs_3pUcy=3Y39Qs%gK zPqyZ-PWIG<sy@O=?aM0oRuoTc>N+jD)$2Dok>7{|^S<>2M;@#luhH-N;D%gk=W^oh z(Pe&JQ-rem4C14KHA8vQ<cABV?8l?3F6tSCzI$~(gRO{Na`jyLr0523$qCg*c0H@n z@jczMwo0jLjJ&!dM2q|gIM>b?Lfp@;-SRSQfG%U;`WB)uK*F#Bj|k>4J-bfb!kPp9 z2y=plCw{nz{RG~bFCz4i0}G0O$yYyK<5nJY>Cl}}Nl@t~l|)bssVu$`;$c49`O_Cq zlW%q~CSiR(F?b_07Kk2#*LT=%J{VLK?gT^-e1sY_!*7S>Suuu~771fy(=u4n49=$z zG;nk+tHRl=xXK4JOd!C<(Qq^)Vo9XcHmyO(BSW7h$J{VQjqEd61#!W)fNsoLtG5By zTSZM;)KS7A!ZcS>`olqPW9?m7qLj{Q1~dCD^o4GH{Bi3b0oixo(WYR~5llvlplUg# zX)sR^O1c&71oJHS%lRts^vZzFDd^N|UKWC848us@-+ySb4=6rR=ebyeT&imy1mz&Z ztZS-^zuZcO56UuJHMv^s%l-8OoQ74)5Rzzz8l!SH8x?z}`M8b!?PGil#+_}gchGF~ z*)VsB`D-+1_1z%Qw}*NCKmfab;BJ1uKbYf+%K{bGGz7ftEd2i0k{Ur7UQ*q@nO@jT zxF!L+3-&-U%nQeHEY*a9XG(3Y%k)Jnt4+<UhjqeDI1bnypx`1;hKpmJ%r@@dXar!k zawu#7jV>htIb8S?*zzL^ATfZ71asv8PS}phNg&b9jUpt?PG*i8N1@b7oa<qhXP<0k zM+QCSOLiBe^e!6OT{N`YM?+(E1%?3S@0<4B&fe1(hr4^fe*V;EQ8IFv&s*=xad=6m zx8vnXAv42ZnRY4H#Z2DL$*ep!I44+Kl^q+$G1df`CbK9r33|A$bm3(n@k}YsB)sXu z_ek8qZ1Ip!2#ivP?W8i(r>a+s7(SR!d<3yxgDrcC!6l8D%ot*2i;pYqf>{%cY&0LG zu(zLYL!XEM1wMc#{Mt546+(R1?a<9pRP!&QYYsO@(fA-vC1hKv%agGghI0qL;Qn3D z9o>~WV9LDp<0xmnEJZnNL@|1xbKAe7bRvb{;O~jL=6vCQiBxLl`91lf-odD#2`M1b zx-oDr>t1>+(NdCwvq~bfo1K><Mq}NMn@u5GU{)c*Qx_$!B=w!|^hl)%j{9m(5ggR^ zI<n>35jr*5qD?I!GaX`r^kh$|V?5_%FzgWbR15;o8pxbfGR0LfFzg&+B`|UxJB^)P z#1ag;xe_JY?)oN-(W=quYKfG*>5+7FkAd<RiKuSM1TB8~sV9OeE+3y7QkoWaQo8b) zK<_<O7S@4$z;bM4)blPFGSh?RpzaERve^)HzF0V9mv?gxnnE49i0-8q6>G_&M0mrZ z;~7y$@Q6-=qC5T)PvT0mP&Mu+cGZfp4=jp(Nq2kg8&!@u@T@5cG)1pqjoZvf(2fgt zNb#IoW(=a3eMbp2iXll#zzxH_251C3?o#G}Pq$+Bo-{*+k0B?l%Szjz+P-Kvu$^i2 z%fTwrij2M+tff!iVp83tNj!EeopjQ>p`6n2nb;4xNdeuWU2tA}XSoHd9r!)Bhu9hi z_8gHGDXcE}bNJ469f)9!<=moRGT#sGTI|J2d@<RHt|p??M(A-yYG$9aG5eHq-$8l{ z^W7)UaqDqt9w+C$aTNO=fCEgYq07nFf%r)}xrPl=3~TfDPy}h-F1(`*#e$)^4k}#) zCKKw6%~du-3QRM5MOg#h7eEjNodSXWq&P0eh2I_Q72c-Q#o)6@&KKA3%e6|B6sH*v zg_UL6N=PdTTK_)cSr!p<V@ZUK#EKHt9S&_lPbyf1sMU0Cv1%0|ctxD&-36M2dx4i8 zm`Xzhn8%nZt}&PFJxcm!EkQz?^f?nhmdu4$?zxYY$tHb5hPse9OL)b}4=|U~I0#@U zwTVkGdM&biC{ab3Uff;=Y(5L2Pq*8kF-e(w&=zpp_3zt+yB|G!410IbOe^j_*_X0N zj|wJ4ghlW%CrL^+i|R_w5lA;FQIS%%`39H7H(#`sR?Zh^F88VxA2wg)SB(|&D^<Z8 zsdEiJB>U%q;fe=IE(!PW6VW2v9l^}#CtJRB&}M#>Xwy9WRJPfR+7#Z8rnEjQ4Qw1g zv&bG@$4Jrd6=tf*%xk;gLrH7IBzDn6SZwGHroQ+t3J{RG4gAhiFSw5BJ@UTr4FLwi zp+Py6dNE>V^6jvd@<fHM!kE53-r%M#h|X@2@m^CfQw;yWr*uds2Q5T#6B>?rNw7F3 zXlkwaBOw@ZYW8Z{vQx{&x`&6u>En<(G}c0~iL1K1h_&oxH8X3Y4~Mf$Fo+t1cl!nD zdGi_TmwEb-oxr*Nd`4%s^P1Azg#}x1T5}HnW+yZUvq3cm6+IsBWOJwzwxTw+*4_ad z?|fYAa+H9S+cy)$7KE}S7eb8+nlw^_c#NM+tD{kI=At1M<3V)}H*v+VkEJlB>Ih`< z)gCS~fsKJX?gF%s;1Tq429q@!lepJ}4M+&+0Ou(AprPl+I}|7YFpS;UY=uV!Rl{f^ zkWv@3`_*8rm<Mwpkp{S?jD&z|Hx()aU-Mp|gbCBBo@w4?^v_-T*>!c$a(HGAYBgY{ z8hgEoIT=A(dr~0d@aYPG&NL$G2<4a!ncG;LH>^#byAWt&ofY6D_|3@y53_HfVm~AQ z2YksK>!wNKk|*_dJCo}(hCgTrM_79zQTr;l7o9BlIr~@Gb<aWFWx;5ZP&KuwVl(@Q z4}>UIHnZtaF1BW0;?r;9<6&KSMmHet4EHMX>LM)+<eq{nkEnI~mi*q662lW~xzt+G zsxLWB`vPSL0^9Rl_HP0PD_gv~@x8n8y}R+fWm-yjzYWto9j=;<Z+7j-+>F0CJ3?JB zfC0&n>>Lg%OgQYRqYuphY1*W<Bf4LMk>OIIg?5?OoaYZL2Z1a~Xx(g9h}hxB-LLB( z5=CKmc<wi|iX$;3HMAHZ`2y^9VJKM}@P_MkSQwHsdw?}Stj;K*FTqWoY}pU%KR?f> z<?;0ojSMiVF1srhd?0x)OP=#IRD5I?g86HhO$CV`(d&o#i;-;35Zfe<j4AhgZJV=L zH9ze*+mCjitbY7>1R058Yr|>&B&Pe}$DgjKJ9@({PR+RmQ<I_9&RGryHSSe87O%=N zzjFRVyv^9(n<5QbZ#@%v(CEeM?Y++rAH90><msEkU+p~E`y8At#I?T>Q%LplE67@n znIv7c&$eGT)j7#0i3(r5er>vWzWwOw^TXZGU;PT}VG8EwxtS~USUIgOn=5_(bmz0r z_Y&18lXi=O7^R&ph3$g7TmC)RE*Oc9yW?OEhIldm6>P43Rqk6O3|(`6aGM;w1I`4w z)qf#QZ?s@h4DhKLje!9+YW&#_6{5x9b8OI;qu)_M8lQ{LASH=J9DAri?0{2=x|{kp zX88n4n80%3QFTQ{UxSf+Y_X8I72@3KdZY;xYMWU-7;57!%t|z<8eCeo6gIl}%p9io z4>qk?wc%c{D<M}a1{>W_l4PGjJ4G1O;B{em&v+JHW2rUV_E_yQML_ra>j&X@nE>=) zl2AAbcsDRCMIs!%1RT`<oU74icz)q!g2GXNFn+}{x-AG+;GEtyA+EnyI^1_jh1+nk zXRVHa@J&N)H5QQ5b+%?8q^2U6x;gT6vN&*^^y9mrslV-wjHQyPc5Rhy$d&s#M=CXC z;oDw?rEP<HHqnJbfPXInKN&N}w6jYqt7+_F_TqK@Fe~a=Y4$-(m<%m7Tm>sG?I*Y+ zF3R=d?1alCaV;D_>bn_F!Mbl>+7;}6(^5J}HMqY3E*%=vA{KlABX(t;1sBg4gBgmN zC53-++VXnP(ocqkd6W(^O&w8*=4)*W7i}4})||SwKvFB-fCShNK=*GF+#O)(<G+cQ zqar_k0#*h=lkRx{bA(RAr&z%Cev$)I8_y!e00cTCN`hqWQ<6I?K6(^I;Ut>to>uh? zObEKz9&LVgdnE<J-m53CqK6gRyY*or0YbL!Fz}TVXa7Xdcca>mOodtgwx9@u=(%H> z;Yw5FN~A-JT8^4BQW~1IIJ2|btg%@(sZz9QhaZYv@^jS(H<@erw{h3E%mY0J><;f9 z_6`y?L3mOXjaVpZ#H5g%w%S=A>Y>CFXU`sOVUI&AphMA`JK1_7K;f>DX6Y!Ep1dR$ z9eCO}Qr2U!zpWU2M@2dtCd2>Aj!6r{=L+2v$H0A%^)N1=+bO&XDP>AyjlLLY*5{@9 z__y*BW#5C+5nvrC@fZmSPY0hfiL9sc$jAvMKoSf;vfF9AR=1~%^v)~Hlw)ILTTP%u z_v%&|lzLs!*;5E+eqWJWT_~Qx6-|?9Cc$-3pzPy&iG?}s7ipgS-GB9e{PwT^*Wdl0 z|I6?G`d<Lw`0c;^pZ@+o|M&m!H-GWF|KT^k`|tj*-~QMCZ87-&{<nYi+kf-#|DXTt zH@@a%dOjY<hI$q+G56V<=HnU^Yv^kWZiof1o<T)7O~8?^YjYwE77|<qJp$dJ=?)Jl zt3tjt$1nNwx?@G_j+1Rer0Rv0W*@=hbv+62_a<~}<~45hEu=GH){^BTxGtNWo{@_& zXf~|cCF4sXQpi}vhO2<(K~00omZR(JMLvC-J(`|_QvR2Pk)l68FTX)uJm@+8#JnxC zKj|9c9%sKQzkZALKLLea-CtWfndXzza!_}R+37dk!Rguz*o~>ZRu9UHvOWa;Of_D6 zUe1asT#pxPJchCu!Y=qCpVZyS@c2_dkU*srjbM(F$!cB#AWg0FqpksEZTK<#|7cVl zt^Ltp@JIPCe*V+qXCMD8?+pjXAOCDH=oRZfKN=qY{O3h4AO5V^82tPvYxQ)nMkKMu zlVQ5}l<>H$BFo^3r^+e-t-8o;nlqwXvlXKcbzx3BQXeIw1x%}%p;(N)=pD;V-{+&E zUkpyo{#&mBw_F3<{&V;Tgy<V&x2ld#Q(yle6<~nhp$y3aBIIjCsqY6V&wE$FOKv|% zdcdES!$Mfm#w`t5NCTyf4{m5TR@$-LrYc+OP*rtzgrbAE#V4^r-523@)Z_@#Y>R=$ zhI5=0pxh-N`+eE21nAVB$Hn^qKvXmMXf<rz53BPepLk%F^Q#cMWOO?iDFS81(|mON z8LRZtHJvVSC0oljL{+b+bJ_<}>5Jg0ziL_u>$+-I8QK=Z=tS+y`m}Oi9Us>iF-Z=8 zRgI~k&KI0cSHjU)asCugxj^y}bfyFtIY3`T$u#0eSVSPBA3b40_E8lzp*rPr`;yVL zVOFTbi4b5N`TAO?7D)>At1(u&8L^zI83gbw!0eOyzV8>SEflyqfIqGMDTIf}Y|gbk zpK(wNfsxjG7{`$w=v#BN(FnZmr|`jt9Hss3^IsY2Yg%1m2;S!$7vA3rv{hZT@CytO z>VkN1eg%4N8qDee+^^J?IiyTgS2FgJ`Pt2BhSnXD(vc1lSo6USVw+!b8&I_<)`7sY z=^P_BVPPqUJFp&X`3r2zi1AIR;l{yk!G&!-Yn$qw0I8kSZWhVEpqDAf=Y_Ze1~Vp> zU9_f*x&#kqgYh`P!h;9FL4zj@tEG*E!8GOwG$6Pc%%(;z6oxBFKthWug}l`*LNO$! zn1?|4OzasXiMuHtL?oBwvu}8snnTG`sWkHnvqTFnKSD^J33Q4eHUUOekO1GRPL*`) zQ3si^1y_;WC%qiZi!aGvF=_4@mx(Rb<7R|8rRTSwBsz>pR07-_tnWRTLKFc-sZs=2 zn9svO7ZG~Y*i+SRF++tAoEq^8j_r`}PS<I&K!he6qZy8aowx=SoEdEpJSUf`E$Cf0 zp1@ZpcIUGb#FtOLFbWE2qa+1(E?{3O9j|(b;yhZ9`i68^Q0(Nwh!9qLD!OtSRY_Oz z%TKmwlv1YkZ^7a(9XbkVSdks1Sy8a0EbZHo|08Z;seD8b<WLwRl4yMdQQPW)wWibU zm?IhGqnP8fmjdb@?Mvd2VZrg3hKY;(0eM>Ve3W!k1*gZdo|OY_V>h=tMa4O;y7l=P zhhBRR9Qu|RGmozKPNzkET8)P7^fItj9Bdlo?+a3x?d0m88>Z}b3=4y$(f$*#ETIm~ z);!kkd&CG{>JtIMMT0mjKu}QLQ#$S<q71w`@nXS<mvusu96}}9COtX)dz0L-nwMPy zN0j_TpLD=6FFyu#SRaRQ6UTK=wdA+*8+C>XW<25U90DNq$T$Q3L3A2&ZgHwrwU^jK zdDgn{J8(C9fxb+dZQx<On4#{`OnbD2EW32{s=+VCPze!^);&x}9bjz_LFzE3>$L%e z4ZE3H<DMZ?^;x!&Sa1f2)LG|Zj;L}Gizs-|#+eI-VON<iMoHFQS0#F+jEb?`MmYE) z`R1}+G)5J&A@eX4BmGFNbbw)QA~<Qe8-Tj61thnDy3O_>H(x>*5U$%-IM5d)T5hBM z*0O*_7`AiKET3sKg)zFIsnspzg~n5;W-=OsFZwW=sd|EbK+G&xB?Ky*XG1+vJk>(D z7K!e(IA^FfQWMVGM5Py8io_KI3WDmY&{UE-;LK%-BaJ$b!dz|ytJ#tK+1Ec;RpId1 zLpY(XVQWlv)?NZ63l{J~3Ly$m`ConYYgLwv8%vr;0F092n$ov4bJ|@i=0tOyl79w` zvksb6e_efkCj2Exgt0qKS{4{AEeAkr4}1l^|EB~KJC5k+F~U&F@d<?LS!W|zOIR%U zh%k9g7S7ZB(m~}~v4#t@3t`Of;!>g!?3p1KjY#f_epTrD<FfriOL(Dy)^M=Bd{Y^2 z+%{jQ-s2~uYYpn?=vp$gu1ei#eGHo17KIan+$-FcE;k9dSE?aPd0vxz=319~o;+gR z3np7lukw)Gf0@7RWagg(P1i1*qZkP+L*&5n95$zEy__651+ZbV96?}VI&xcWT^15M zkyxDZ7G~fSj-S)QNP%?}vmD|uq0xJY*HoYuJw&UJ>G<%7QK0wUdfQXuyt9_=!g<<g zXBG!t$W5vIQjV&rRBlv#DWBA^dMs<((9ed4^hw}LJCWXmToP<i!B35>+fm!d?giGj zz^ph>Tw!Sr*^CNM^J4;<YV=-?gwr`XQmmj<7qGg>*LZz1c#48y6ht~?wPfo|4A*Pn z=I~lRZ<(^pW}DQOuomG_9M_UWD+?sdn+F(cY$hsv>c;9S;JCvIp@fCI4eFG)9pt36 zyzjmW)E!d^sU^LicmNX-2~4gRDa84k8a=4?F;E2!1J0<vb{Iv@1HUF}l`<BC692@U zD}uFlEOktya03opzvlB4_z-^u`qi%@@_YwP^ys`kl|li;fsItrTU(h>LA$ba+CHtO z<u_oIIRZD4A=o<jK;EWQ&{)Zf4KKB<|4BnVWTd(tU#ZQj2q#QLnDfWAc(Jw-qv9BS zS<vOID>lc=)BSRIwGK%5{D9nl`?P=iQTmSXqpNIv3J=!tOAE&+d|B~xbq(cvzPx$R zqw?1+A%(d{W1`pd0YP&OW%V{y-JY%6gEN<HOO&~~MkDbnFbcnJNiFK-DZU}-LvIpW z2z4q1#y$;+Q6O6irZ60cT#9bxM+DOs*J~KG7)Tc6lA!fZ2^g9W#_xeXv`65?6)0t$ z+1VeUC)x3+x>VbH$fHf0&yx(AUZnM6XjW?Z^k)3dn}*&5Z|hkxp_*bOnkPMs%r69D z<@hCH{oMC58R;Uxj=J3rnxo5c&0f~^c^>!+Ii6YTg|##AUaPYFqAJa*Elv#czF_XK z#%mqdpG;D8f<|DbMBpwlEzWY>s1j6}J&?2Qo~|Mx2S`4vhQETP(=KVL(G-K)0;rBU z`2?q81+y;UgFZgk4AHrsLg#u4oz&|RSQO1EW^$lM-t!~(H;KAonC-P}LvGPg;{sM( z^YdC;x2^g1sLmGQH0VbtBxNX{<UL_>VBtd86lDX#T}_m=aVp+$TA@X*1|TR7i{t!! zG?M`kA|H(R_qLEb^3Jhnql>JgxH2c?x&t8`@Lb7mnRwym>eZEl%uG!+8uFeC+as(j z^VOU=sc9LA67O8kr_-0jq}swC2$P+!APh<;i6~)5mgOBb=%8tPf~YL6R}nHXrFIoP zrqr(#>`p+4cBv#`20m!_L+}Dgowe!PB$LM8f8B4Kf6f~vcV3=?|7@E;fqr>xT_1_h zj11=K>-rSbl;z3T98>xb!X;q&N*&{#+87~QJY07xs+M0`dt)mvK2gAjH8{3XQiH>D zAR)9Pwzl#P1IfyGMXzf_w_I~6Ab@~Q5nBlSnuaCZl46U(j?H4DPJ<wV_972Gz`Zqe z<C%!84Q**{aVYA}w>h-cmoWfLSrdwmly87C=(jdQKQVMJ;Z(o$Ts+9y9|xr$NFZ0t zON4S#UAE~kyWah&-XHk<_+tA{4|lhpJw4ocx%c$5r*GUr5(XN)3*u~eR;AXx;|RIE zE#PG=TDH%m)~p`n7}rw)o~^mE!5F5=y#sg!v=ebZ`{F=uKj0)`sEwlEf&D&aSTtCr zRCL_HXVdC@k_t<BJ3k1i_O}I6JQfV~oP*vKn(4Wp*EcD)%WKkt!p45=$0Vb7-AjNI zuWj2-LPxB&MAV{!CoL&aTd~V(%WsOX4Pvl)JR}yE{9{U%=%^QkCu$dU!IaDlne!xV zzSzJe1z5t^!C5&|N?QVuDKpr><-MO3^?mtdcz47b6@E9Ic`p$nEjkES1CGdSsjM50 z;Bf(gtYBjw36pIQ^g8xVo#k}%NFM&ia_BGpsCj1`SHn0X`JzzwwyXpVPTLPubiS*m zlD)8`w0&cE$9j`)Qc3!H&Y;yhrvoaCBp*O2bVt{!j>lhE$@=8o2zG&9Dx}}AA^eXq z88x{L(Q!##k|3>;M<ZRCV8mc9E|M}bEn;crNhO6ftub5}1#iQUb%x&aW1tIXR^gRU z)_;lYzrgc}m{8z$=Y`?e>T4M~u;nOSfppl`PI(k#TWUvYd};l^xreYBw%UkV-=gAb zVfu+}H#BuG=roA^Nyd6s>~%E*3lg+ayU;BxVSU|fW)C_%PWNDyUCv*=K0hMn_&~!X zL^iHUCM+G@tV=I}!Th-Kit|=ES6X7P?Cv82)~Uv70|i{n5&vZ>#n4jYlA=jvlz2Di z0J1aT!z%<f2UI_r51b=Qs+2aa28TFLi1I2asN4@9@C1a0G@-8^8G3ssJ$V3wSC1K< zY%BKE69SNh8m^=z7wja;T<bE|sWMW_t6rQ6p9w!Ido2}PBB6Qa{97u=EMye<$4jyd zW!<{6nTT{8iNa@<45w!!S85SFjLWk^QJI^HP!ygOR;S%XWxI>Yb{Cax`KWA$+RPA( zDF;AQ>9k-;+xNVAsgPB~gYFjF-D102Y|Di}G=kX>=L(YD+kFy3*}`;@BXkF-)l8xR z_qe?BcQZ23u@3(`27h++uKs0l-7bgW_S!mp`uMPAHtTXo8>EX!mm9jOvy{{JXW&gw zh2ZAL$WenCX7Eb5P!G$g^91L}c33@=3BYY3uUSVR;B!Cw-M{+ZfBTpJ+Wh;wzy24$ z{mcL6w}17w?$sY-m~oj%o}W*q<%JoxeXvGf088CN+nIsDeKo=mx_x<RPyoQaaCY`a zgmeMPp9&zRMLv8rc8RGK6m&#nhsAgA^6Bto=`IUXc5#<PXWdW8xxUjp|Iu*g;T(ti zBA>`x)A#B;9QeG>ap0o!#_m(yLDJkM5e((O$L3)ePpAtT1*|0uZ1H+p&A`isPy@ji zDvmwp7BfM>5fgYV`nnvhKWpKm)}}Om**PZe%PZx?uYId+5z8-VwydA{#paiLD)owA zmz!aPk!@soaL9_nL)Z`yrH(XV!!ztV6X1gwIKxR<0q7wshh)XJqoNFz06zS~u450g zb_-ahDB0&n<xu-S3BuPaL8&P0!~F!m)>aZw`tb0{(=QJX6UlbQwV8=-F`ZV^c8l07 zF&b134bO2~o)x^QfGJ9=;f`p2PDwuV0+a#DHicFLB8XXD4}*6@_8gGhq*2?rDK#RE z$jF?6N6-&H(NyypU!0<s9GD&380C&kL>LcYDJ}@Rk^TI0j8+np6zftSU6T>I+TtRr zqUiURJHsnW98ji<5&JJG0(yj3xnK}a`_#e#4?i_{y*A=Cvv9QCg`*t?izvOo=T7XL zRS%3WvAn;dfqb_B?rhw9pUN8x7|Yfws)3inBWzj;JQ_v>Zxjy4y^iTS#f%nl6<bE^ zdU_j_8n4$cpFY`n`I+ZLun~YsV~;GWX}mhRBD|9nIPL`o<+7cPIqhr&?Q9%`{08q; z@YV&?ne`0LIR-T=F+7d5WlAT&iyPfoiGgcwbR*g5xJUVVH<paL@wPxG=3dwAEf2ER zDr_L*jV?;!avGW40+A#Lg9}5!W|oU{qPE!phh1}IhYpw}Lu<<N$bdC&cyCj}_DIBx z3*Z*lLj`$&8`Eq<og1}TG!(KVy>BoHTLb(KFMs)2yg49df4`dpAn;u`2P^(9zd;~& z=4=oa0kg}>oKnYc3%1PFXqe9-zq7lE=CaFI+jvjr$yiM=Pj#8T924pJwRbb8Zjx7Q zsJge|!wc4?pkt(U1FB~hDPNwY>uT}Qur6P2!<UonU9X}Y43<}9IeJ$`l=F-y<R%*& z={NUp$R?w>nk58txU+~P^T83BYnUqWR;V@-d4KQ1Qq<a8S6g1qF|l&eIT%)X_rgz8 z)=z_@?!cg;&0tb=jT*y%lxatcW_03Ad^mJQOQZ>%R!QI)NQ6$}#qW+zVoMP^H7<Yr z<II%<H1=*)dZV%Q;z_d&EjekA;K)8Z&5ld(PeSkRX|2W8rioi5aZ{k;mrZBhec~3T z;VsH5dBFK4ziX9a^N-okb@;>iWO76Uc?RRy<6$V0Ap`*7IJi2QgTawgn8GvUp02WH zqV91d(?-&lZy_aw`M1jnVKCnu!`}^tgjPl%e;(ok6oFcB+z$#HltU2~lAuHjDM=&} zPOTDg6n5sG!I}{wvLWxoLo{e2jS8zQ)tJy$GbCy;V;D1MN!<MTd4blU`>wvlINJ$f zft2T|wJ~Kw9n%ICtxwD2IH<qnI-<tDofjQiErmnHkdj8S*oS3Z+N;C9M&DruwIG&* z6DvT{m53&&=!ThYt`sy)F)D1^e)f_PtdmpcDt9C-E6E1ngI?Oz#Ihp8D$?cDtjo_$ z{D|fb#cu~rDq418pji}vE9fI)tTvv!`6Ad8+7!%VPVB<91b;FmIRuIM<~n_bzTO-Z zUvWlyl-6KcKVhlD2~KZM#A~6$wMFI%Fv;yW#Z5XXS*Lj)PKY#H*=kR-({vB0q{0lE zW*IN)A<}4-E4t4zWIOJdL%#RB#pt*#RZCY9o8c2Ot?lF~32MnTFQnq6aAD<%q%2_p zR+d9w5Q(s+su6bqr)t7oLPA=0cHH<E&(J)CHP4ckeionG5=XYd$Ve&8%jtym7ohUM zCc?wfdo7+Ab%dh%doaReZw&l_0CeB+Qd*feQOj&ubG9o72r7dt`2rl4TIqWeFg#z_ zjMIuNf9cz`(x`N7PFw>-_qd5Gs4TeOif>t3AV%S-x~a8WU6AVDNKnf}%-9dvycHF{ zul9?^#^>KRx-}WK=v0R>x_y(Z6HC>X{6c(pler3UrQC~6HpW9?!QEJ9TRt?Huq@~! ziFJ~2Sq7AbY*>KdNk7GE#A*;KMs=*cd6zEB@ss5{wK~;B-8S`VJ2m8Xi(Z|ocbU<K zr_M<}J3WYz5fB@iWUirCQ-~nzfF(hNMk7T*KvthLI-JGFboj`<B%*Lx2(m$1Co$r$ zR&3d^Zvt4flM|!BHRD~(a9B-y=WqY&Z~y*p|Lt;Y_i1<qcFRkz4|dHZ`F_KFgoy4Z z=;-hN<!}D(H~+Tx4}bF)zx~(0`Q88c*V3NnxBumT|Ns8q|L%AH?k}m((tONaOh(Lz z&u<0|Np42{Y|H`X@BZrF{O*7LFFq(dq$PInya&_tNkbdf=F;mDfP`uEEZ7mR642E8 zTw-m8Dut!iV|p!00d?_uT)Y;&?aW|W7@-{@fDVjsdUgGxyuu&_e&_(@OUx$^-NCo; zc3X_w6IqITZlThSLpyD8fUSe<3Au;ah+AEBvp6cUR*orc;hzEgGlYMNR#*3s-i;&$ zGSfx(+P<>7vr(p!!2<oUb>HIPVBg{(RH0Z;H=d}r{o;b3iRfbAeP@Td(#hh9bgaB1 znQRkg$z;skC%rAMlf^*yiG3b4AgG3%lVH~@F=Sfna44)0MZp4yb<*&G*yRoQ1D_ut zXKPxmn(Jhu=Ubzs5f=%AttMJYqk`PvU!(tBlk<1ufKL_=ShfF-q;SJNNb`rD?VlCX zlVS^1sB>t7Y}q9uMjQwO>wol7?|Y*RYg8W~D(ZiEJXs_#qMD~a9`(D<c5)E!05Wf5 z-;mqb+EdEBHD?H)QO2*QqYp6(*Vy|HWFecjN|^VB+bMJAfXw@|+ZE4X_iMAVxoLN7 zx7@~!yOMd?!e9kd8^np_YunwOTKK^qcsa`KJuk_w&x^q3*;8xcQH;UC;PEK0>&K_X z;BDI~rK1;5*<$3*8|BU$<<1*rNpF-uKz=r@&R~JXwl<*t<XiKOPGxNswvYk9fLlB- z$8Tlqq=ye5iqyg2N=$*@QHnI1;KTUb;B~^GHu!yYx;8LH5Ns0SS%8w)9RLymrPJzQ zsv-iZuLz3*b;}C2SHQNx7eK1?I6!PUj5LY1)?-leXU!?DmQ^-GTWNhW23@JnrvsRE zDAuK4K@kpA#YO0s(vd*qM>cbX^?#g~BZ`H<jaO42>}tkQJhV1QYrd(Pjh)i**oZGi zJqqat6N-kHmvRcIl%oe65zH(@`XFm%pJregMSVw!cO57yQ&uXvDms_%Uc=M|NqE~W z?|8n8$zfaJwTHMy#Cv1c7FZ(;%wfN%4JY0hwlH{?dON8`d1#EwEy5){u6n=(EP7N? z+f^>04DJ<D@sn+*k70}4${q(JdXbL}ubSd;ZPCy$0H3=vBp|A-S~X&{{7rr=m@Wvs z(OW^7_lLT6cjy~NZ+LLy*q79AGY9vLaBybWYEjt~<dDTA`w9<eUvV&LxMRi;;(48Y z1!07Y5HbA<OV+AMT@CN1Au0m#xJ1Jc$t>HMwQ9phG06I?Qth3OOA1;5JMvL+nqQQa zimZU?X$;$Xb=(dR+HoiGq<S?d-8C}LAfL7Ot?FWKifIQtEUelQmT%5d)%;HT;RDfS z7#oo(7<LJu-?j!JNx!mc7oz{d7lJIwti_GK>&D?D*@IJHxz;^L>L||R)mVPR1QPeZ z+7{G~Yt~nSrVt*1P>6mBrTT7=B6bAbuk?xXqUYPQ*|cw6!o*wLo%`k_V;U6AB0zly z)=gN0n&Vn@YOr<uDZp~c)aANAQrnozDACYOn9DIVM&y!F0Q=HTDUlVmN81RQbT~Vt z<gMu4Vd3tuaCcZ(`mpfJr@uaY^7Q%6i=DlvZ@`yyb@fy5B(;|nsH6i=#PM)cXlL={ z9mTHaq&JeS%Il396uUq0p5;cA*fwkI@%i+k7#pm-rat9)MKb6>j~~fkxopd9kw!>> zdzsDHn`F$E+XiJd#y>~?Gca{AnY`ii8|yYmI!A+hUkn6Sb%t<_z~MqpdyNzHy!C0e zzuw7K*Si}KA#nr0{sg}M1i$_azWxlq{sO+5e^z^)tOtLfH2!10{sg}M1i$_azWxlq z{sO*2m5l>27qr_u^fgUFM?tPrGz3$5mP0G)Ic_I5UN%{$XyO|(&BMS}mPanP-!O^d zWcds}#Z26Ok#}1Xz$SM$2T7t<LY79MttRO^Q6n&(7L<E<7>=&FLr?snr~~IUoEL`M zk6lieIS3_efBi^B(L@Jl`CCX5z~LQ7E|Y*cb&b^wW?S~{A7&ertLy{=Oo834`|&Y% zF=uj@_ES)8(1EaBa_=QXwgST>NIub9H%raE!t2`JN^mxsKRfDo#9AKXDiYa}2e=V~ z=OFv7A@V2~;DwEx$${FP@oNpqEIDbY*~lQzIn;9e39>f@Joz8b$M$`81AX-+wuN67 z*RPJP9wm0OP3;Ow^|3!DE4#6<6+P&FU4cTotIp<56|z4jW4!Ukyec+Ctt66juWqm! z4z8#+*_DV|pyO(S#86lqG+YZNu%<Egire`mI;pCQK!qan8*pm=V7k0n45J(-l!j=V zSZ#m9e(4=7KS0=8Utp2}v+RRGMy}$wCMJ&npcc@8knf&HK!Fz=O%y{uy^7-6-q5%s z?240z#e^twc*rA0FV_eqK6x7WjSX@X<{=TI=iH6xX<kVFno3|~51J)*{-~H;7RC4_ z+utsW9b)E;Tr(tU>{n(c!LjG|SWDuNJNokWJ#C1ddJB3|^9AF2fSrc(Bhejkr4(%8 zVrxdGT?kuga?uneX$bsQ#7`T@lw&07{RcJ77yuM0!pgR&ivkJ|p_wl$vk#SH$bE3m zCuTmn>CJ``@4-%mTUQQwwC%CZQz*n0UBeSOZ5Fg>yI<FTjKP}Cc9)MpX8>(LlE3iY z%U$P3m%DNDuz(&tO9-J$xLC~>44fu1IlGyRq_l=y01=pj#E7BEo#YMev1p`a@=$12 z#%1*aR{I~8gb<vToV&lAf5Hy0$!FEn<0&B~bf}H*gDjf1k(98qd0TdXzHYp=TY1Pb z1HJ3xPJiG@3<$zWlzz~Pv0&PJ*?Q!0;+RfZ8x&vW)4)S4JOgQ1*)hj9qFPrao0^{s zM$(^Bjc-f;#9M>SxE!4rN^5U%6N)gN7s81o+?C&7h_N-3WDWt$ErFV`lOsvEYcqLw z@IK-VCDWXHI43>J?GERBzQsb|`_XXRH7&?UBKK176Q=I0s)?2sj1@cb@&Su^3E@Td z>nh&1x5$O7>6VuKL7HxA?P|kP`{e=Z8Z;M@Bpy0%*$+5+zfImu@<KQL%vjQc<M{Jx zd{RXE9C6^WWw++Yh|Q>-vnO?2+ajPf$Y+p;XLoizvQARs0cJtklw&bNaAzWd;_TdP zur;s4VqDiKs@O8LM<`{{FW*n7PpO>d?EMP*NnY2ysgEnV9VCg#H%-4ES5xPZu2&YN zAav72stTKxFh$><DJsG;0D**|kO^6wFKy^}UsUcg4ta8KA*6=az4MO;qQS<iJjygz zi?l`_uls&gJYb1EL|t(CTh=c<`!julhTZ%%%XeVJSh#D(3E^bJPRu2-i{;+0E#GAc z*HpfQ6dzXpwWHGOJjHeeU_iXNrHeEC?`<K)Y0_c(TNN0t0U++ob|@*P(K$WE^v#x> zc#K&#lR2X7hxukZJrOHRrfx3#*{c1%88?{lSr07|Qhj6RVm+*Ua7comO+bAxWW<j3 zP2F;r@3TVm&e0HiNOutNst10rl5vLaE@1C2VDBzqZ*>7Hv?L%FLGJaTI3$2o2(2&J zxcXU(cF?Y~?_fz_Tq?tV%A}~cxKzi~=LY6&GiTRmXWoGbIOVJb>CsQlacyZ|NmQyA zqvOoQC$w+W`NT+sClz#n_A*W8@6b^9aG(mu><K=YQ>nUu^fq{H0`oB{PT+paG-6nh zFN)^!^p0FrxZ>1W{3KolMx)UPI`Jc4Ln8>gHwzVYTYF{}QLN=@V7G50<MMAX`Kc2P z9Fl=ch)2s{-E5;1$t!H@y(V3>(>(Y82t#Any6y4M@Y}QE{l^$I7!KC9#CQBp&nBmN zUDn)LC)-C;C(L;R)?)iD5TCsrorhlQIAgdC`x$7xq+MM$GQd2vD#eAZXk%$`L0l)k z9=%P~)5|#!W3iSE&z6#8L3Z>qLLiq$3R&nRw6~S^;SCn+ft5N<DCWhIrs+Yulsc*H zy-@2;aL<vHZJV`-O(#YI2cIO}bciyt)Yzrrnv6Y!RaD$83;29n!o&o<GMf|Zx`j8B zl%BZC7!^LQ)TIIHMKL?AhV7QD(UNF9i)dI*t7q+4lMJOb&!-m&L0r_orZB25P1Npb zJ{nb*?UsGt3h^2?U|o{Ke)mNWp-U`$&@x-L(RdtErMfzE?OFIN(Gz^4avtLArh18! zlT5r7lA&hE7EO++*<Nv$oa?n<dLg30!n&F=lgYkv1;4UR(8nO&gJ3v<$K9)|EBFtj zIPmdCAiFU@$L2UJWFOwT37UAo-(uVSU>KZ4S{kz2Pf$>%@J$pRSiP+Hq?lO2yT#K* z*yM8EX0*!Yp4|=?wp!n#B?*^BPZB@r8t;gwShe{f_zNwT!FfHa&dP6!&x#rO9V44| zzfz3hcI(4uqL=N)5l`%ALTNHl)-edHF}t3MB$9vy-@`r&f35DNH6q$bGmD4eB1sqP zH?(c1xaHpTmg}{FESq|_cxYgKi9mw$gZ4CDH73H;;Pa&->OiCe6hph#hWs5tQTbpz zBu|;>mj#A=<W|CXgC7ncEq*TrR{EgU01_ODr(UIOdnOgjUR8}|<wV7@@DB24=kRjG z#6@0v5J)t-OR+DQpbe^8(TbGbR(4go$xblciM<JvCHWRi!02E~@ojRk^wJgzcRG4} z9Qa+fK&E5hRifG{&`n&3q0Qo^$dSFMm5rvbNVc8R+1W^%GuV%S@*W4gF{}hi)Agx; zA3`S}OV;%7!<r0BF&-GmKi_-typ6TUC=cBFR^l-LsnE1`)1g3I!w(D@LEv9dYWf~j z-DdU$A6_odSXz%kI#!UHH1Wbjv_J)Axa;gX+35Vkty)-3r`m|aCePxH%s@-@AlK#- zicY|1Oz|?#PRnv^>WG4i6EF45&O%XB(z<SLgXY4~&O`SC!uX=zteu$R3yPl;1oJQl zGY6VPRpiO2aMa*E79R0*CGSAypQjWgQCAV;ZLZ<YSwYuks_1gQITmY-Xj@rDZZg$; z>g5Psp0?^@_GeXfCaLKg^E(cW&BOyQ4czc2)fVM!m#q!(Nn?BT8Zd!hqBhHVd&~*0 zpJK#`;i6qIT7i!t(6d#WM$LsHlp*w&;2S+=^EsPc8{Mdf72irm!EZ7QHgm6SH>7rp zikQ6)SqD~Y*?=@u0-trMoCB-15>FEAxvU0JjQN{UaRG;ibs_10)6X_kl$6lPxzB#P zI+V(qi;%^~M7((cV^>!_tFgZJv%T8Bg1GY-OGScVUR^i8Lg(o8tCOM<FZ?Ug7Jm*h z3n*4x%5`7j6y7;ZCJKx=;tnt*_2RfPM-<RwZ4bZB4Ji(;`(&pS;^=WTJ}ytrIn6p4 z0G3lQ44y_LVPEv4)mzQL^P5h6*oj!7o%y{H^|>^GVW0Rq>JS*(b3JnT8YzAV4xE$o z5xGG>Car+^m+ro26}+7mHTXtM^YKZj4RCe)=mPcA^@E@jBVJvUGlSLjZZGbim9W|I zv=nx4ZdClVkM2{U;9#m_zyq@9c?a0>e2<m;*~jZq`@#Aos?swiY4q42xkb!Oy7SJe zN%FaY9&at9nK3Vps@V)!bNmL=)0_5wvHpiaW0arXzw$5Z>$9V31V9+V92K%2z1lVB z^M2OT7y#t#a?O28e9_?q7;=hzG0}=bFp+yendDFdNNt^!!=Z)>K5(T7SEv2#r%fmy zPV<u!VAiP>JFW7whL!BN-I(_tSYI~pCVvLJokjfO8{9K{8ux^NtTq_PY6wk`;lhOr zRe-~n1vzpT3cP{$maIifgY#)!rTRr1WV`D_7=`fd5JkiN1NHFLRQ*IXeOO49!0U@e z&7iWL3P}!|nrFCw@zWsRkVK3~Zhd=n(`uS!P($KHmT5a!a@Hs<>UB_!0Ew+1<#`?w z7O$X?7gdvdz-%fYIQv~F?m|A18o3|6W*gXSEc5^14{q*aXx&+5+*xJZCZju>d>f&# zAiXa1cvO5-nhMa88Gnv62&@A3i_0I7u{K$B*w;|s=6HthD#bGtq$csE(UyDUAF1Nf z-=&7e+VyEZiDKH8^~?Na`vJb|4klMU^nMSXUiX}JJ3xV8lJ=UdPoP|6cTx;iVYprf zs3RU}241~|vOij1UzZFLtOscGiH>;Cmulpyrv>=y;P=%=@ILr$rX6-!&58jPk=z5A zNJDjWUXF%<6Ls*MnG>+aI-&=HIqErin6(aF6Yv7nOa2i~Z|gv|D2{0GS6A=HB-M2V zq8iwGKO!zwgX*B$*S4}7nl<Tx?GW*_DaOfd_)+o$JH}ZwG*KfpFn%4BZW@#?&BO}w z`20cP5`_U+bua!1^ga``#uk&RciM;9KCPy!jchgJuRWg3?zMk--3!t+^3}+p0~80- zRyS5IKepKdM@IN-PdeEo!lP~>fg4u<kaH&n+=&5qV!*qI0e?*QYni=Y!#``M#mKb0 z1{^|d{=2_^@#xj_!^hiuPd|J021dMP&bi0a>WofvaN3<!{QW@h?BESRZ{Ugmb0*@i z$DtCP8AR(4?ksc}2y_UwTeQWyL6qY$g^Gw>C#lPidTok*3A;Sm7fDqya!m&tg^W~Y zMI0a;^~7Vo!(o*x=6CdUjYJ|_qKlAmaEsUMV~MA*1pu_LkP7hNbrMSe4_Jhvi5A|V zwr$BasBkO5h4q;8AO&$@B&hrTS4X^wlCFCQS(uZP+YTEC)OWH~-yT>9tYsVeS&yEj zAj4=7s%VF1h6<}laHOEbq}{9|F=$ZOH2eg(Dz9VUDAUZytSlqCQFRXwT@kEFZc}<< z_<G36z4EQt#vf)ERXLRRLjCN4YUzIV0{VQt0|8@xu?{WTXhZ^}G2gVBKW@y{(Akx; z@^-NDP`)7NdmV3~92f4ciKKIRwFP|{QOk<sZgS3#UYA$J2-Ny#s5SE{0(Vmb!SSI< z>=dG;79{_4I0!%x(E<uY!0R^X+ICkpVh1DM8TrTziQXIHG5_C;%UtU+y;Pa24PU0$ z)Q&EbY=;jDrl5DIL;f-mK)6iZXc9QlX#-?yaKeB^H|cqZ;xgB^jLgmre?l*uMAL?r z=eO$%+=Mem7rw*`w#P$Mo<6cBrNtK<Rxu>O&!*M6O@JQpK`_*1E{Ns2gHv<9cy)|s z26DrTS?FHXT+!`~Dl7=V+gxX)jOh%+`Df?l;H@Fyubqp4*n<#oxLb3;m>*=ONQo?# z+=t5X1?pUxfb3&3dXg!T=?NF%47|QnWZ&Xnby4XX-PXE}*S4~-uqHmC5L94^S_-%@ zzI8DN=fHdf%6E>b0<du-A(@lM(7nT8yct>0u_9X)!O5Xx5u}TpUE}m1Z{i0_PKjUA zp0U|=E~W^G^L16*Vtj^dud8@aRiXqpwW9b1gdi}iw+)65SgT_@u<iMS9AsP6!homA zga`nSyZLN3ZMW>rMk{LFBDuWEe$?Ce_!qxWk11L=T_l)Y7GsE2l%{Aptu9+|065F1 zZ-Ec!I|H>BlIg`|X*O0<m)40H3<%V4<bk`oi(+(5=OQ|HnwKXdc{es+5}<da;L^fC zaCC1J01#UV31L&OuS5eRG9<0Y>|mG@+?m=_OXHN;0huZDf&{1r@RyMs#~TStUhV+} zD*JssP$H}(%zNOIau}TlR)mU%cB3|M5t|0~9WT0X7hflfpLtK2kns1a1ljLZG1247 zx+D33<PD~NZv@{<6TF)&L07B)==7d~>UPxbT6hmy7u#E-fI%;s>0Ofoq(Xnqr8eVN z+auFM;`x`wbXFQTo8G^gmM7&n>doGkfAkT^mjvnIA^v^BuQt;){?wEgwc&Jm-yy4^ z&aZ<yy*YJG^U?8Zx7*u&-;i54xF%{A4TLs-u)OvqS+07z487<ElTb!zXc*n<U`U@P zf-8*CK0jL>Yw?X<_4rk<L9!zhf=0xH326abU{f`%0EeAOOKy~U!61G*Yk6Tfd|!-* zX+!I&=Sr($sizlq$V4zMFVH$c&9V&?Oa@%=U0D|Mz&#9F4vci=IulG`n&t%odauRX z`m&r2PGc;_X&7?Dom%A>Vyx@}a`&o9tKVGa1zM~uDD-qZY%IYhFZ~#ALx|<F>K58& zTq5EUrb3GhhFXc@HX6}_5kH~^77luGIWz8bAve5Rk8Wo0xb)C?^KR;Vec4WVj+&_L zmc#{XG69m@C4iX?iT`@${2;Sy&KyXH(RwayU-k}CzL9R50hB|#pF-quxC;bZHMj$8 z4{0s0@^ieX!OKB5S4(+BrI0mbeo4*1bW!U)**eo`=1sm*&CjD(Vth-WaW#QQ&#=rs zkwB$=e_Lcaq=neDSbjfbdaQ-`WHi4kGM8s}yD79U$w5-H<7eeQ?uUAA{5+T!!|O$p z5pAV{ig5-S>XRTv#G>H=@<?{O7@JFN3@?(*|8}glc>@WNJ(wU?&7UO^YRPpW-qzm( zQS&xOSK^&_nV20!6z93BWudpFHpn~sF64GwYO}x9?jvNR@KlvFh=dPh`r#mI<|Hf* zs|aA0;}GG{d}Q#9D#GJ(TF;hXT74&oUMODn-FI=>gk!Qpuy4oXoZG-3f!G){<xPs^ z{0D{bjM(=_M1L?0?ba)tvA&o+F&E<Ks58b7sxf7EsyN&~S!&f$hTRiNdAo)h+Z!rt zrXa5)H__-C+eQs{mj>|gqh4Fx2ipjKyMs#DL9u?=`o;KTEBh(=%dR<%Z3>&TQD8dv zQ7^ED^ZBJ%mEz7SKx$T~|G^))hPo6%gDY(<bt8g3`4+@DG8yfA2?SBV73s|HZjT|a zu5^)h!^aD##62_TW`@i8ph~ut1*m?l?(|}%xR6}Isl$wOLYIOny<2RghYQ0Q+C|6G zn=mNt@G78*d^p5@v?|@+^T5R?L>fw6dg5J=R@>qIf_k!7oK31}KE2NBYIN?rkG_iC za=y~qIz%IZ!6^gJnYpngH<;=;3vp2wo*@y@qRvOJtmQYO$1QF@ekd+Nq*CCT$bdw; zQ#ROxI1;0^uoW71=xU*fhgPam*_s?P4VZ?J=f~j9fT%Gm(&c|q)ykPumD19Z)qX^e zG^$@s=dN9D29&kfgj&D0s|vQ0Dcr!<qw3N(?{G{#$RVOJ<3dZDg9Qd2LGevT_bL9z z?zYw0SvB6BjTSM%fbCFi_6hwyxK6}G5bEaJzIWtMF2hQKvxt=D^ets#U=0;w#zrWT z+{-4IZDvCsY`HCA50$1C+YAMbUr3UAnX+$?x<`c%>eGP4G156G?mU!{5&<Wg@+`Ka zNe{YDMl5(=ezRb$-UQ2*gSq<=eV^+tto~16_35%Ul(y&`w724?53nuP;VC=3E>mBZ zmFr&zI%)>Mm5U%)vW1|q!vgFeWayN;{e1+h-sT}^u?7u4MDs@S_w$g~uvA~gF$K}U z!v#x=SGRMD*H<)hT3mGz1IoQ`t-w=<SrwEdnxIrmE}GQbCWE7$9ZeFnC?ZunP4fX! z4Jf;e3*?25-_3B7Dzi9cs@oLbAGbB$8=GXwt+mogOjMWOigzrqdHkW=Y~(<~D{F#l z(S3AS20xgW)h?Y1*=t;uAcq)se@#{HOPn?N6e`RJu9+Gqk4YT`aDpw7MN8c+Gj-7V z@UrTnjVHX#rd=X^?c0Tr5rDF`KwtXa8j8|{CX^bBP>g6!##jk9B62pVE}vTqHXl|^ zMl{5DqAMYb+!Ymr#h9)gpQI9-iCU(3yF~<zjU&RuJ5budNlelM=T^t~^wmi@FoL)N zBQpo|qx@)e4dXo@4M9A|*sJJf@D(d20PENhqTU|9a9-VD4TsjlSCbt<Yep&gi>Xci zXNvvVab!A$gTdl6b{HlV0J99T*lKynrJD-<1YrAy=h#$C^UE*uQQY?%e12ZE6Jn8o z*jC>X&T8`WQ1L&UW@s=j*w4!P865PBHno{3^U$2Y&c{Qux*Lo18KMOJd&(gcTAaJd zedq_)m)Tcj=tHigiNN>u&}{*sJ;^dt>muN#CbyHG1OHTD!+{^HI?GT==~#!{bbEA} zUz1hRS2G(;`76wx3Iz2P;f?ktc>fg|2Z86J%a-m_5mS?BsRTKiFABDFg4Q80r)W&B zH8D5$yt%wW|Gfk`n^t}NXED5vU~8S|jA3f@1vSWhDlCf`Dagq_TIPY8G>~kRa(7Ei zL^ov?lh!Lo)tank49fY^u;b%Q&}D4g$Hd<lXyz}>eb7r+Tn+1;GGT=y8nJI%A!l-K zCZhgI#GZ8H`{HLuzH6{ONty&{GoDZ$#B@lDC=G=BvDr$@9%Ja;>?{+Q3Pa=d`D9W} zLAz9vU#g>8x}wNVi>Xq}A%IHINz$i;s1p}>hVois4~Ft2aA0h#I1&wn$%SnYT+y&T z!#S&XO=P@;yt*-J<Euo+&#LjPz4BE!UiciFR~*1%p48xqwL-Ar&96A_%c2&jT(Sv@ z+_De^bhAoSf0mz>qwB>DP&pZP>&DX@{0XfDm!%24+1`YNEG_y>6ZiZc2iEI)^OAb4 z&t1pvvLsYJoEn^t(7kml6`Tx(I<o_woD@@k0BfaM8djI%$tb^GrXd!3nstz**K5{; z=yqdIqXr#eszX&-_1NGp^;*Q=ZB|{zFrg1u(gVu9MGu6whq4@-MOT{obTieV>#sZi z>O24HJOAqM!@v5846p)MO>nn2%XY?tDK3Q^BoAP6I0InlleO%9cs(7sJMnad)}{i6 zF3ZEaazk}yagJwe;5QI}72VE<cj!>C4KFo+JcLJ}JD<_1mAHvkVV|%E9$bhbmuW}W z<j}gjDu>^}Ww|QWE$i2?5bGJ;|0cEt8}@_puj<Uw4arS?&}nz7VpQnP7hGmVY#nOG z9*7n#wgllx9mT5A@Hsas3m)2@1e4b<go08Y+x9@vB0mk)b_HT8dYqtHOGb1D3&YGR zS#!b<pi0-4A(kSiBnUzdgr6N~Y6=o>aZy(1M!Fplw--tky000E<J>&~kP@R2EquYV zJsP!>%2}x#=W8hcMA(05eJGuq%j4s=?A&*aH#+m9&JL?>wgP||2yTg@>lr79$xeuz zL_cPs>_BA$)K0QX_I35W3h*Te)Bq&Xr3ROQhHuj5k6O-&#GMt4J|k}RbmU~CWM9Qq zF*u(ULM(CJ-U6?3xs^^TTX>}_NDm4^w#^#3oNr^<jM>iK!>>eOhzsuO6cyG9#F?#b zv=XjyLOl?sNSrRvHlH>Okk6MQj}WO(VTf~QnyGajW<~G8oDc{r8sWj1ncN7PIMh(; zjDC*`EB9nOn#6=D^vqc`4TmynTI>(EX^_516h$;)8}KM-Bo1HXCf+?8PeSq5C2|kQ zZO3f`Pw2$YDh+GzLg8USZY;gLA`|8~;)vLNdNSQQflC{W6p<Heq)%`(MQXIcSuehN z_h&dEZ1^#}o)+M2?kw=;?EZwa+iXm}mO#BwFMDY!*bW2US=LXFkBh<V)e*!FXp6fC z7K*?=7DM+|Q_7Ggesc<Nu(D4RkHAI(4YRy{OF1Vz&vxAhp&W!PyUj7I+rdQ%C5Ijv zd5_{afL0<NUYvt`B*8QQ0vXf=g~`sZF}6BOhHhgDD9-0e#{^z}TwT<fZsPM#<Z~l5 z;nN^>`a5k)cI?aTOwm5%W%F<kil{QYvN4#NO>@lt+-{tvI8%k7O}{tE0!#07dNKmi zp0j`?>vFa!^Ndt&>KVP^IHe;OUdr%6X#VsSbh@D>>>lLxw^JK7fJK*~5ey5eg*Fe2 z^7Nu7elte_cQ_ao5WC)yN?UU@y&RWkY8s5Fm*(9|cNhV&>GvV=RP!GHyo>$vf!mz7 zYk3NZxG!qvAZpWO7II#S<}^VkQXk!E0uH9JFsB#)L?{fowR@^BayTUpsUC~A?^Jtt zs=Yha-cqVPnG_nV?LQk;NBIcNq1xh(UNIfa3bK1$2UZect`Wg}Y>h4`y2wWz(~zV> z*hbfoHba;Xn;&#666pFeO2ven*M@*{z|<mIfh+ZAl;x%^su*VDe_5Vp!2|<eECt<@ zdI=NdkhyQ|TZcRV+_;0_gi`Q0KC6ZbP`FB~S#Kq~AL8MG2pQl)(c0>3en1h^g_}7S z5fI`@%J5cE3`Tehu(GXQ%fEEjHQwF}>y%@<RW?f=Z?>C*CW>0%RQEu<1(q%XMck-y zKJex4*!~^czhnEiV*6;Vi)JKXE`q-{)US`_uRlM}hmg-?usuWD6Y<;C29uIq^I~d^ zO6Ui_H*|iweN<y@3OsTIw*+3C1drp|9CR@#_G3&_)5fIbbOGGqd@P-8nh(o!eW^TI z2cGH+`_L@-wimqZska-!+l>RYb3n+ro$)bP;$4esGDaOi<i<`v4X>szn^IUeBe+0G zeZA`$w>yI$&iD;{`vkuJ5q^DuPxW!TM{#J>%(N%#=+KOTEzO?|{Bx+I=JY5^O;oW* z6?;^%7gt2{rsHGO@9<={<+QgidUlpp%yuQQ652aqJD+63Sm}?nb9@|}ssS&DGx0D~ zV}P+<U7w%PBC3lM;L@;=&nm;v3~T>~zxj)2CAzE`{x&-``{K!I_NWAk!vi;FIreSM zOvQSZeY^oWjcM-lQh9v|`eT|upO{kSZ=eoAQ@<%*4#Bz1;KNRSfYLD;P?||jm4IgA znNH6$^tBSz4JSK2)nVO>jit$J7Svmox^%sdm+lV`7F1ofOItNb)S8qxnn*dIWWN-k znH56~y<&AEqU*V?{DM}Vq+n@WRtZu{Ht_MB2Bac@xzpN~ip?b!Y#a^?O;l~a5D+05 z3u+F9GdASURH!)LN^&Iytt^N;cN?IW>25L5%M3l_Y4&R0Nw<bC4Zj9gU(8=@#G4$# zV$=Sb<C~|H*x2ie(4L}0F%Q6|RPLIyGu$`#<!<N}(eC-tu)HXTY-}1oms=>BeeFX~ zRr>EhzufR|<(muHxrNXYA3deS%y1b3Ba?R^=4X5h5_!8~3T4nZlI}U&&V1x!QP!Su z5@C@Y=Bmb8DkgHE;P;?)4-YH$J&#B)Sqe}U)7D^Axv&pT@(JxT8w#4El`a=XAhCLR zCl)0aNfKJh5uNDNG!iPz)2kNELbx^MyBzhUH@tsl>6Ape_?E!|lC)(;T$v*as~I~1 z42hU{u5>Rhz0q8HBQD)b#@iDl`r{u<g1wi{#UK%=j@jXX-{&ZgZ#irOz`K%y`2+;= z2sbjxfDwNk)7gw$2c?PSBzSNEcXJZm%)xsYS4c7@o+gqGY>0XKx<OxO9b|JU5L{hg z(5B4($Eg@Ru31~^C#h0%q3e0*dLFvoQqc95g09yL-AznVj(b5u;_5?_)5ZrPr``u4 zr-t}N-w!n@I_Wjd*M=;$@%y7DMJMy1>or3cP!nb;o?K@~`QR<cBdN^T$nw+%)nVdJ zGDo#T+2JiJx#4iq5Rt$D3YU)ar9#V`H*UxIH&U#hAkh^d@cab=x?#t*k!aUPpf`mt z-e*tv9(Dy>gxljGSgko!xi1$>F4=!7Oeq?To0cApX7yc$(?Id5xo#l&jWi|S`1|5J zP4}Iq`%crnw5I#ntH)pL9&SJRlWl|M#}K@{b$A#D7;CYz`iqyz3g_c=6{-Y$hJz2V zgVTP?Wv*_C*cMi)4(mRw;4Ni>3i|4fBB9`0w3h1FC<xGyI^<U_76+7*Csh|}V^gUc zYz6CV{Lnq=yC%rA25ouTvg7EM@i;d)IY)3~INcAt%2ZIh2qw^lH=&ZZq<&xv@!@zs zY&5dZ<QdR4%m^?WQrmZ?n7@5|e@hJ|yuU6+-nWm;5baMlB595$(%6D?Tx=+akFBj> z(n7A93_#eXf<RNLaCZsGg6erU^M%Orn7|IJWXs|Hh)}tZ9*z(@*h=HR7ZKLZI89d# z^uG+5EJ=TG+#ny1lgbId>>6S_#FGc)*~4J9E|Ai7VMD}F_7m8D&!?1a$)?vcTS-Y= zy%@bHr_*XmsZ7}V*j0D#)V5FIV7x3rX<`EMoHsrA&Dn$r;`LszMBwhXcY>lYsE@%1 z>C3XXjFMKPuFt2eh6^{D7`Pa4%z}JUHUy|)2PCiz-E|KlrJrzBvPt`r<xWIKwZLf! zp(b-cke#a02u*FFxi?jQ_Sija8(`7KW7<}VudwpxaM`q#eM_P6`qcN0>dzaU`oR)d zvSD9J8_TaGU{^@<KO!&%!tB>XYe?^J(4$IQ9s5+><ZmS>RfYVUJHtpFKsVaM%)VaU zA}NIXOocVmam$kA6(X;hyfpfvRu>RhY&Nx(Hk(rIy)3|n6q-$NX&)ciH9ScNNrn#x zO4im=0cTI_ArUzwHw-3J2wyGXUU0N(x@3_-C9%1V!iI`1xts!;Q&!<k3#`-J>~H9E zPKN;y_IzP~`O*%;+Th;NUg<2du1nlXl3zs79(voz5eQ}={BefL2hcVNgL#r{_9?$n zO*;-E8n#WO_z4w5>39qdRBvgR%|!*DY-R+iHrglem1hOq=J4`DofAKcE_kZxz&ilR zL5nxx0>+2i=qSSZyEL*Ol!d9>4GKm<T|^h@Xr71YV)4uu+p)F=vTi4YD3Fl+kKwy` z4%uPL2#Uj|=!vqDk2W`)<|ij`EDaZHw>my<x8QYa#ec`-M4K@Lu9+-|UP#sORa6l# zr|sJ$rvWg=3`HkpMjw6_&7J&+a$R%YKKo+Q4k47Edy;0&skXT!d_WcUf#_r&+(y(h ztgaZQCRu*Wg#T)4CdS8ro~QSfpP_Jw@oqri4deEorjBpo$JUL_O)C)8_d2r%^e@K8 zi*Pgt)J3}*^zg(|v8QKL?+SP$C1x?&4>T1AEP&%`JFCv?0?r++PO50^6SW^{J`(Mn z&#1Y$Yz!bep6Iol(4oX`UW^u);RS<8yT*hcCqA-o4eS@w#-$2U`>?Dh`D}2yjZLQl zPudTkUlQX=2vzn&d&<tRuXdMmIHwA%q8%epUG-6gV1G3bV`^;4hYHQjb9-w%TLgYV zTC+B9<_lp`TUr#P(;}if*EFQrZP40l=tKQ~fUC+5MC#)1))s!=VQpEt)~+u=?tjcH zOuY+^(&cn;KFX(*TGz(3jbvcj^UX;VPo=g}OZVjBgNur`Xm{#6h%BrIS3(SBk)buX zipyQ6%3Z5+zhAM~#TMn%#E~pZvFK><6iqTrNa$4#d@sO}^xUIpaiOjtp0~zZ(+TEF zA2xbtr|L}ep|jn`q@fc^nzd>s0phV?R#5XJ*#H0t4Wvv+?MHO*dMCS|Z9u-ze3+u> zG=ydSyg0@jvo=NP;Hm)@D!Kp^_;=L;v^*F=?lFc+rURseVTl&cs$qG2?Sn2SaxlZk z6GsC}o5xWNbT+ASKU+0poWmC7Q06Ap3M;+h)LAx_TmuVzH}KOCEiIJ4qmJqKJ~xt! z+}{#;>3hFe{}6)>cz;Tbdz>3&M0#A<cdEMRH*HsuYTO-U{j)ANqF7mzTKpz16AKz1 zzHnodfSNrLE;45E6+$sOj&;k8z4@1XEBvG|J0h0B-O_5L+H<ya=rUbj1P-vD<>V%o zmy$QtDHX4flmM19mXrdTQ5vUC_S5xsPdWna9!-s;;T6Quuwz5qSlJg>3b52XB>_rr zLg-_JEp<_0cE0?2ROQ2npq5nNy2}l>A;XW&xGfY#VRNJYym%UvRidmYubxLzA385@ zvB5nns~=rs*|zH`gu5Vj16K)7Vz!d%E@2Fh9YPd`<atenN8<VNKm6Gbz?327q!+XO zNF#(X2JQ<?H9F{BCp*(uYPUIP%Ff71Au#hucczikI1Ik>PL3ivcT(ODblnC>YN;a2 ziO6xrl2Q<WAq#nOn=iwK^Cspmd?W@Wdx|iYk|!JLlIh|?)GTUYV_^%agd!Ggjda1? zCCMV}ASl1&z*4p54<_P+sjcHZIiY25Nml6i-S@V<e_ty**^k&B8@2cny>k`%%gHQ$ zAoZ-6#A1iiY(oh#xwC3ADPp%=_@IQKg3Jk-A+^Hzy}RtlAn>~F0IaIgEF>6ov$XtB z_rQ{ZmcC$oovin>l|*V3`{^^DmJpux6sJVb@MiPftR0kDNhlSEDv(fIPD#;`z%keK zK#1uEmuQ<@Odp>XgSSQk7R#iQU3IeSq~2s&pg|a2^P9!~2uzU;5xjv|b|GG&a2Kpw zulJ_;xCUcky2b|^Ci)#$L2E^<iTCs6xQ)QXvj{>)HQt#{i&1fr8#YgIdI5Z7TMf1W ziG6vVY02EkHUUzN9FG@0=|QD;7E!$zW9>+@2C-sV%Faj=K^NlEJO{hkltK^9JMe95 zK@$s%Y*bL2R6}~uqy(Sj<JL;*1#}p{8B+}@(xl}WT92SFeKmfakK1}UW*K(zBC<QY zPO-4%mDRV@IZK;2Fd1u7XAeOc4InnWzQ)(r@!MGGo5CR?_jX9cnT@enQz~^ZYX{BG zpZc%A{wIl87<nJz{{I000RR8&y=!+|$#Ed~3-ljuOD$D_EEG^+zf^p**nD(jhTY_n z=x()eaL9|Q3!s{Xs&d@|0g#P7vS&TE&#^4;9@}T$M?S|p@`tr7@0p#|$kO=7j7Z6! z<}YkK@)7yEb&EvX?R8ySP1L=ak&%&+k&%&+5r}uZjOt2P^s1`hXSU#HH3o#Ro#!~? zS2!O}qc|F#zfa`OH1)$h(hinBzZ*$YK$vq?IOIk4zZHAwa4W0hXU!po*UVt_L|5E4 zP+At?&gFA)x4}5H==jDW7v`UCca8=r$0eaxIBlJ`Tbl3rm}ZnZ3?=<SF(76wXWzhc z)%wU-v{VG`G(xtPLD8#&@h-XZ%bQlqn^w!4R*PkcQHB8ZgEPXNX3s`Phr7k}EZbIV zI$D!<{6a{h(Y`IzpH~7?M+Id}Q^ssXMF35h5|~eA>S`Tx{R}YY<E*j-3ppslXmukc zL8y3sYDUQXqf%}jS*nf`G-0aB_@s%Jtsu@)0xu=-QUce!>zdO9+5|SVmq1dxB0YJ7 z4`#;F?fXb#si>IJ@?`{qCFA(@^mkf1hf>UbJ57JX-CO#r+N{kh@L59kSprIVq`GVj zcSbE$ziB6MLnj|=K0Ab|vc+U)sRw%gZygm=*s&{6j<`$d@GpD0iEVILMOh@t4+A!4 zYX8m<t<B!4EL*`W?Nry@!+4%urc@IeZL45&RWK2ac@POzgonfJ+&WyY49k^axiT!W zGVt2a14*zyo}S(w3}@`%<8Qg$)JO&1t)TZ5O;1U>y}_t&BwZ;DcS7iK-G-R7_y@<q zlVHF2B6D7-#5|SBl6<ktOU-+y*mP50B=(~qa-mSTGD+Z*x*8`6qp9SE^F8Uh;nh+G zmbe*<&mH?R#gvE5DXQj3au>VyVLstJ;of^R<3NpwoBSS)^n+r!Hz(H}1!-Q?(N&`h zbq=*{$s~N%AES4+OwFpO0^}_OV+sPrwnQ<3XeEzF1;R(*<xq56|B8jbinu7Ot!0~g zC{^;&shSO$NifRs>R|RAm7HvHhG(QjZ)Orc$|n{$IKuZ~Bw?Q*&TQEp?6e2v1J$gd zYTW5xKkDpEI``F}h4;$)>E<Cdg{-m9SdEtfjFwCHa_N5QrJKX|B$sZxcu(?SF@wE< z%E@P(Xct*HD|aRS!s(qc%vm@WI?|Cse6xS@VLs7466eeAl8D`_vE<QKHCSiWdYZWQ ze7)btVBLFz6BC3PWzZh*wh9lxIs0sCXKy_SyR+RWdaI4>BH@EKfuQBcdMj_EgeV8g zb!5RF4=ItgcPGWfscKUJ|GXU^(9`?@#-g{U<HLK*X-&kv<?x+PJP&7eC0IqIqAS4- zMxe!*#yerK#U$1gHp|ASQK@O{UzDWA{Q8$gF}aSjbk^`yfY<`<v@~~xc5w5Ftkoh1 zBrM-8S~F93W-PcWJ*ggP!<NmP=(VxrdhdD0aPEb{{>Z9b%w{k-o>2R?tWf<lraf5W zXU$M!>DQdan(_-Nue#F9DX?Bos+Cw>II769#VoDJ;*(HjJ;70>*{bo2e~8a0Jq=%a zYQ-CS^+_g4Du=q7(`k<B$Yzyr@#%*%sJa3n6m5Bay|^4#&FWDNAy+sw;!_pKA)?$T zUi&g2<~h97>6beFQm0>7r#Fk5iTC_^r)kss(oVmbhk5i|4<D4(c`M7c)OS<lp?&F0 zZy?ZTea_|qtDe{PAmQ{MH;gS9naW;}-lDi@4h;jxRs^n-a>SvSEQhtAaN5JjCIy$w z{{@eDFkGyO<f>Hab_u)&aN5BgcD2GrYU;Jut!ry*vf)}4Nn6sAM8e}0^UDiG9fM;M znN&dO%!OjHQObfKxPhfoKi0))9;EqLY30TrWIKW@^FUa&$Oj^ab0sleH8-bnyJzaw zvCCFn%u7^B&Zp=u#_EEkS)~C+rdAa|HMUWpu+O0UM!cuU4lxr_zK7=+beGWIYB<v} z*z$^@j^OS_O-_uhSFeNxxp$B;<fw|-%VA|{l;fjmuV{1=+c<@-z4RqxHmluvLbvmi zjJfS{Y|+~7ZoFE%!FIzDJ{0~2TbeJpB2n`l2^0erm0ghpJ8tILN}K|rlf+BOpenpV zV6l7=e(k_AQdKw^wU-9RJxpIJ?@Q%<sl2~L%JdR(P=4*M&UjAg(kUKy{nY91CZLuU z&PD6rxRsVRofTY#AQJ+HzLX{+)Aq$ft4N<TR$7^SaoSkfaRd_oDTzyW7<4W}_x{FN zazs*<ba&<@7o@-i!KnwPtr{T<UPYn<)F;KIW<(bIq1nWKK!a%bsrpvYs;<A@9#07` z#>TljN~kF<$paxG!}$9Q&V#Iy^11qjzM2tK*@#uysH~#1^=(f&n0Qs%BayW)n(hZW z5?AHqf!m>~(WE@E*+otSONrsLReDQNs3^zlN_N2xQN;MBN?+8aHH-}oUPUc9FkBF+ z{8D=LRYc*AxQpy^ja{y>%Qbc(Pi861V>T=Idf0gxu7<I2;ZDdyPvR}MylXkQ@Di_^ zt9@{8Dhxm*@DPZEyGC?zm>7eCEiDN~pKlHhqx)K?0s8~*<@fR#pVA!WlbiaBftJ|l zpN0J}=SDIT()wlb^9rBZsMAF8rQ)#mNj^L(Vnope4)XF7RVL*v6+6)qG^lf&rBD%P zmrV<Sj<2E-nB+s|;z9XQ{t@6Ztakqro$}nZoIyz83N?9&#)j&-(IIxa)g)?>AmOIl zEjvZ<-PV|5QZxyf=#AzG5~;X;_}~%tupDR4$j08j!v2NV&RV?O01A7czV)rU$6&AB zZX-nahZFbUvimX}+lDb~W$>TCoDk_pXjSp`T^m9y9oRzXEl3k~DrBkm#TQ2YrE@P< zY<rgHgG=>iss5ag`g3pd^-DG&?q&U2-E_a7_hG2Z4{{0twY_s$UeJ29o5KvdJ9?D$ z@=>;1bm?t_<Yj|@;IoBHq~^QwV9tB6^}&dvi0(oDq{vG2klW4sWJYeo^h6TnC;?gq z(12O!5C)XjRDkvbkF(0doT3=b4ge1-lAmyq7X8HSWNE+{mbHK;)A24a4{#~K8|D+b zvIb?wNBdNCN^KP*{B7Wrz^sK$Oo8PN+aO`G;{w<PfSDNt+&LWsXCo_O!+FWWs{1t> z%rp=+O(=Snde=4uZ*PA0!&~OpJ&iQ9*B=vIN5t~j<Y58W_sOH2BpGlq;$tf7vFkc+ zHjfZP**4JJROw@&%WMGKZbBIEyD^g?4c<2a!<h*LP>&M~F-RN|p<rNP0CnHFD#-t* zAb|5BhMQK-?ixytjKo6=u=a5x4S<I@^B2wt&1MstS%@E-VG@+w+)KmpVSxMqPVMKX zmM<q_Ac&cvERMaPG0%xdguvI-<;H1#m_0ij_m74}x9DNUx^k_Y_SVQ~wZ@x^Gr7Dv z74<Dh)rM{Y1&Sp8*ZA_G(Xn;>?%2B8(!l<)?TWT>72wN2iN*M`rE#9e5BQzKeQC1P zRlER!COSNY{yHA?L1e6PtZ;mBA|O$4czSzK(0VP(or-c(EoDLMG%eVctV^aM2^;qr z5=*`D(0G~;)+{9G+B@mO|Ac`GgDmz=10yKs`$W(+hD8O!LlbWK9`nn#;+q|po9z8h zr7CJ{q0cQujAFWTyAdPtSH)hsh`IDDXDIDWI^LU;4nm2TIlXeHrkrQWtZ!<y!*bzj zcFEg#^oRfv3oOh#VH$;|5#tM4vx9N}O<NJOE6hZ<;%Tg+XKSrxYE@R|79zb9T3ktC z3a16%jMuXnq;oP%<03P~=Y7xz$q#(p8^7}C(2;M&?7`7tQ!Nudn;gN4eq8~Qxe?X? zQGf#*HgB5ZvseUGX$sM2I<!}h;iM}}N&i&pr$zDtqxG%=*v&ecGhGl&n~PTk9Ip}o z3@PNQDYQl;>5a=K0bD^x=vc5kQ?vvTZ;N_?^MWmSHI|VOU3Pu{QaB1o%X+`ZRpFjM z8J=i`21tRk+G0V{UWKz9KOmFF<N^BvQLTVE*4DC*$YBUHH`1BOgXMStnkXud2cY>C zqZVn-$9M%3KfumC0742O>Z1K(56x*$&~DH#vgXF*gfvLlI!<>7vuO@n6*eY$H+9&g zX1l<FYn?7_q}@O;RCZWlJ?yafe&MPeo2WTg4z-$ooRkGwTL4-kArUx=aN6ln_Q);z z2<fpq*x%QiwMWxO6|}I;ONuy1(e`yMBu~hz6l>207{}{RK0KeqW*%G&%c6MnXQ|8l z2T<yMftS_IY--|qL@2LX=qS2a(ep)EO1v*VJsde2Aa=0^x&D7HTbI_urS))WJzUs& z=$aG0beDfWJa$!%xNtX8Y5I^xljQ-!Qba68#JR-k#P8Q}n5X^KNsa;1hPMC3;cn4~ zz9Sl>MXNC+^_gttiaWXp<^h&KW`;=&ypBeB4zWfWJn#UTY0;ND`+EOU`}6Ya%sU7e zt5aq*7(zB#ALf1CeQ0Ac9n)2MEZ0OL(d6tGz+b~sTkU}_aaL+1m4*H5nAYLz*={i$ zAGcQ!08Sy~ZEembPkinZz0w?yTLU}}E|2IQzZN7jAU2UTr!X)h&O29XaGHGZQ+KoZ z-h-Q~mu?L6qmqnL)ZHr~gZD38TX9`qQ2oC?>R<2oUq@q)_|1}T&B|n*zJ@eT*cP## z2_;B5<C|bbd(<H*Jc6w`{Mg94^zJehbbTmAeh{@ogT1QCAT*xLO2-DR%)t1*jeiRh zDcfeHp;mlU7FowWE6uAEK@P1@Y}EyY(wKUcUG@~-S-Y;b7C~<e{WM<u<BKn{j%Wdv zwkLeWgUXnnd_@?jwBGIjk>p3CXe0F_ZXZOWsK<vI47!@5E?J;s$K~KFj*=c2|I-G4 z+HgOOI-K1JZ8!Q)=<2oye6(SUZB!IfAlSph#0BtxgkR6hUz>OWHF0t>#%M)smGq9G z{wiyfAe2@#UK_P1ff51l(AqIA@>wi5BeM?(5!$LuVG3i03fS$fx8J1w3US=_dhE)! zVzarGJ3DUOgx3XNIuYh#?965P)eMwPS$8H_?=}ls6n+gl+{-Y6Ij#n<8d83t)&5{g z0im*cLl6xp8yn8_c&tT=*}DDKbk@`)GP6n!(uzC59wRqChQg1XlNmgJ9OT2>(?L(A zKcxw@XMT(6^_HGq`m30owt^4RG^Zq+-~w3M4?mtH6Aln7CgDO?*4yvM0c**9HEfn| z4^E1{gdV%Lr#QBMveVwf@}ZMe0Qck3-b+IEebQXe><plvmsq8p+Ni3PpsVe}%6+uk zh9!5ce+mD8cR1c%+kLb1R{!m{_jccT^R2gb-+cS6^>^MW-oErkXZ?*!Z@=;FH{R~N zo%i#Nx9Av^w*_mo&Okd(Ge$9}sTIinDHTeu!^SNOG*0?(P~?4Qp?B)9i9rFyK@LZ6 zH*PbhbC|$+kc8Kr!pJWz<vdPYXaVuzM^+K}P#U`SV1ES4lE^rg4ju!Woxv#T3z$f9 z>@QTocd=T9B|x!i0`*$BqD)%2>P&p|kOqW&4iv%SC~%8*EFm<h8@#j~<H-mBJjP&3 z@G~^FY67u6^EmA+dF&bY1WCe|=7uYG4AG6Y{2JiK^A~R|OEnL%R>YwCEkFSFlkh(+ zwPG}S4LX^E$Hd(*@K))-?RaGlSZ~Nd;=j)d(0mR?zOUxo#Imj*wY*c0k^^Jdhu7U! zHj49)!d|n{lcKlL*1S|v9E%Yp%}}<o4zN{!P8&*1yq))EbeGnujnPiHV?{${MoO!r z)nh-;I~4G`UrdgMc=8c^xpxaTsvo_7E@RrM9#jMMb6j^}Vs<XWpBnP#b6lfIlrZ$q zQ_+oT3KR`e3YA1LiZ4W4rgXMbe2~VZ4}wC1q%P7$VXbE*BU)tiWmOO#>_SZV;*<2< z#{Jp^`4dk(fJpF*#U%WJhaG+VUjpbH!KJ9K8CbQG>cJeTrNowzTNTi%!;;c69b2UT zk%gC<>$nq0ktiu2T9q;ZAGEq*jY(26&XXKMPu`OX^0&VoF6g~oawpsMOa-Ri6{xq~ ziCJzhFYm=I?Y2w1?V?9J!a#R!=Q@=k(ygDV40HZvTyU$(fQmr9Fzfyi6Hz99Z4Sq- zm_8}`_w)s23CsWb#TPbUu7pxM+`FzI9ABNV?|wNE0ZaK1s27p4)Jq)~$iswvC^aD> zmC@{BdB;FfabV5IXwdcwiba<@<Kk8`jHkh|@A=1V_ESMToo3uiRuquYLvwr3-yyqa z^{uNpJ0g2+N>CG^O4_EcPvPTTp^~$&X48IK>C;twf)iHJVk=9y`2|{mWawKbv%>X& zR?gh~p~`hov$g>tV_VG!p>|uZ2t_HwxH)1q$D0b~Te1sxw1=Hv)qaf5qSpo`Cvh-M z#7z(qyKVEMW#9WM=vhTu7N>Y)PKUPbua@{Sr#%>#I~Ze1JCh#drMdsZHZ6~p6HRV+ zc4VvD>pP)~ljg)yGN8$hp@v+pZ|crbRRCd%+TaTwN1UW=RJe_T4dq}|<kJuHiEhlD zDq`T0(Vod7g?q82c;u!m>#(c1{TX_KCq|2b`6hJv9Cs0JB7*G{Eu`ts%A)O)(gv;L znL@<^w|@OV3(ePfR8Y7Qap))c#e1)y<15*_e7ex(v5e5b=it=bE~PwqMZQ`v7@w}z zkFk_1^yD<D=Ba*~qp+Q(Gu?opoY}YlN9q2`sD<$mL9#}jOCtI)(Cc3l(N9Hq9cMWt z#UU14M?dS~>sZ&Sr`M>{Rkj3i#l(7bCYl~?1hqEi*Mg~-N|<iQ)~yVjgHLdt(Dg?n zC=tsf5S@hKG+cbwp=;K;n&#J`HGESA+tq7&!U-a}R`+II`fr4G%9z^?6b1Dm%COpe z64ugReA-5Z*?8E0FrziY>0uj>xI&YdSHRg_Mh?MLEn!G&WjZjJILE?FNAvR<X}j7S z^BF#%X?!{z%<x)r@3FY&tAiz{YqDibw=`*4sf^w;Mk=qQE^ayWt|bxG0C=w8mYK@g zcrvNz@43(yA2v?bCo(=M>R8utOUsvtIFFCNLG!nLD?7sG+V**NP6ILGd55g^h*#sx z)hSxF<W~{OTtCF4)nso?NqxvVzoa&L=ml9bcSSu@NB!}x4$xMjp|80DYF0FGj$0q~ ziL2so>2y?4t0QW4=GUTmZ5n}Xf2XQvvNUcsCdpGEeCR5r&@blB8U=Y{Cw`04f9l4y z6DZV3D^WMnoq$>mw2J%{^Sd%MUQIY4??0w2wo1^avWu9L2rX1=*>=3&QrYk<t#W~^ zs1=$Ty*Gir10AGQ8bcWr?5v`DDvO0Hv@?PIz&_7{M(!7f<0lxJp?Cs|<>%}pCOa8l zLDM7=E@}dUrAz+uXlHq}v#?A48V5hgKBcfWKr41;WnXnN7-x8DlG!a#R?;3wIB!$X z_Tm`#Ij?WH_h|CT0RBT6nj9DI@kE{UdiQtU8=vqzzlU*6N+W8xe48ObHU2|=RC`{d zVbF$)Yd9Mx@-^)uPT16^_^T<mLF|KS3Vc96n=4Yjn43;SCG2(VG|UYWN&Wy(Qs=qS zi`uy*MO1rEl}91C5#{e*9Rsln_8}n9?S2W-Jvj@vdvk(A`o*=MHU(~UOkMSIuE_)y zwezki7TLb&B_ZU%(B4Z74qeY)fH9*I-k2D4%EZW$AJN;Dm*l1)MGGOduuk-1`sSW; zB_Eqh3?Q65ix7?^0AN)Mz>yN1bR|+HFUB2uRQ1*HYsB7<pgOwI&b&rDm5p}lG)fnt zwIuNSj9RCUQsgv;r5+F#bsv&&WWnLw9>md;>9_~-x#+(ew*EwkJB}_;NBD&p+*5;3 zE9GvH(di&AxfUDLjEcM;lAH~!3Ur**%$XX|_DRbi-Hzt#Q~YWxfEsUpsMtgng8axF z89As4>9)K3sk)%!jap;^?e2b{@9TKO8J&uVjt1`H9f6Gn40Nv}N$5A~g}y=!6$TKQ zB9wmuO#A70e0T>J>X^_bPb~V{h=?8urt~b0soC68)NAQcXc-c_RVtA#ExSirinCkG z4o*vuK5R7rU_hV0m?LTjHXY}Oq$gJ+c7r@~!z;n;vD*kU26^?r2Llq;#$obPZ|T~U zJ(2^{3RtNN!;ZG1t1ovg%U#QI*Rt5I1-K2mSWNHn5<Q-poTax1LwL`VO_x&Xi8xmm zM*^}cHu<m&@qCrCv|>%Ko|^<cQotrTblX+dC+T)0Ypc)q?r+|`zq$2U`;*NFA78)I zzIXS**25bgT))5d@XqzmV2E0tBTWQ`u#Yebe0^eakledz05`nFOA!V|e8cRWc(c<L zn;wu^D@&oZ6k1E6wU`&}`}xs+!FzpB`W^^Xqd@11e$#BQ(P<T<@hT@vtk*C%Epf-o zH%W1qEpH8KtB3EFD-;i&#i)8bKw_LWk?&gBwsJJF0S!%a{bDaa8qV%c_-t5!d^#+i z6vJ+|F03d|yV-{L38Lul|MuVf<|lvg{CEHI`QLp1`QQBRuYUGd_A~2dU1<Kz^>3m) z#($(s!LZl|!hC@AzcUy;?vj&)_>PgBj!Gca+i!leA>Nm>e9G|w0iTUG#TNjl-mktP z-}i>&-s9szSp;oBA71>2??3<P51)Vk7xpcv({9gIvA_7STP#26W}S7f;Pc=A*U$gp z|9<h4|L4V@{`7zT;t!cbzxvtFJ7QN)8CXA}v;YXf1MnaJ_{EQY?>E2vFTeiz-`NlO z$)E%>72Sv-p4)iwZ~ptQ|KjJ+<nzDzPcOdz<MLp9{Nj&({QUR-D}VUa&;IE7AOB~! zE|v*sLSv&HVW2sI838um7&>72;y?cRiy!^J&wug5aNj`b1%U~+*V`Mw1K;>|r}uUa zya;T$-hKz)ymjfFx4zA9I{3S{ap}!VoyJbg`$XCKE_ntLVMRC#2IKGl-JdvwB-;PO zQPPr)&WH`o{{Bz@&LcT}$Ic}oXu|)`U;O@W{`x=v=EuJ+-u%J5(U3!$9H+)NilS(w zDZdf<qy^e|IOsQO4~%AVn0xrgPk!pM`qrTLI5t34BO+$o5BPokF?#XWKm66-{*rj( z*MIx3aSs2*zkB}E{~e~W2Sd)>H(zzE-<dnq{~)8<IhRo-36<~Ss`0EQ;ZZd}6(~wd z*63&rK@Hw3pu3iXR@sOnM5n`8y)HU>SQ_uFf72(IBGo(brqIqDABGxu&IYqc#VT$+ zX=SG<o__GBzxw45CMWLq@!p;>nSkV72cf$9hPU=AIM09gzy9i%|CU)!O`Tu<-LL=T zFL_0O@n^sH;y?Uf%yrNji=3@n-`#q!_1T?U@os+O(xpp2>GnYV=ODJ7H{%rG2u$@! zfSy|WQ`~#pQ0-3lck||Yi~euLM_?DWLdEn`9zkEc@<$El8@*mHG4$?AkB>f9BKD*Z zvBhw2HSgyW8?Ht_6Wpzg_F+KYYyqgqU@SVQi7~C7Hebr16K^L6U9Q&JXtvyDF1MM> zZRSg4@!=)yUQVYW_)ltosWOewzEst++syJHpSUYd&ikfyC+3e4e9qfx@aDv?c-qxx zn0BwZdo&pK5o0G7&J3g;VH4&BhN3sdxSQ6IgD-~#9?)T!ju-4$@^vl_g5I8bm325d z{_3A5$$-pUf_`}=PR!R6Ji%gob<E-mHbxl#aqTLgvzA@T-pw!+oZ}=&c^C||Qw8mT zjeuMWnm3v*?f_E<{CDqPz!0?2k85k(#1&}8`-BQ6eDW7&?)Ak)uUoO=r>c0^EnE5} zTNnLApwqeS`xRaigVd-e+D{GZ+}8bp@iVq87t`fpx?D^bSxn6mG|8uBp*=PNFAq0E zmJTDDJcqt)3+n~O^+^*2Ow3ZxDNPh9ImLs?CZ(J;R1PUokWyH2Pl_obgE@Fj2ucqJ zr{Za{V#ik0sL7hsuwp<Kw&0>8R;dz01c&8f(NAQtk{02vmeeNLi;Oc1vV`mp%_07` zmC--9^>)Y%CA!%pKIzY>L&2<dOR>Kk=VJ@+#Zy_lWa-qwP)Hawv)On?M)<RaLMwaC z#IEGupt8l{k>g`-=~tko=};Fjra;V}!5bhdCmkg3ZEn2&X6B?1y6l{SDBsB&sDGxk zQ}5V=Gw~&p+yu^-!gc~;x{aoQ36L7qnCru#m${lfDPgCo+%y9=)u#2fxc-UdgX~F@ zak&YLmf`%Q)VLg=_>jB!AuJ2NE|KbK%VOQqO^2F-ILsN8P^$1EmfG(Eg<6!3wI68d zM0yNMYQ`a;y(X`A?eyrcVvOig0nyE%==u&k0l^e(M8;pA07a#@{us1vkD8#~YUs*k zJ~z-gvGdNlj(|%URwX2TG!=!TKSrTMb|v@Wzr}+Ir<yU4R|UE&**AHj1_1E3#x0AC zt>SNg6qU8-ojVMSH?2n@(GJGP*P(W7sDNhh)f*r`dYS?7t#8qdI>UyQ!A#fIvKzzv zaN@qDF7pZ{S5p2XKWQ4++k+iiE7xJQ3a)#3BPiJ7qCX-RbaN39Wuam@q8s4@`qA`y zfls(QcfN|_PdS*0TU_=`w5TqK%deW{?Pz8mDs(Ceou&%e<5u%$D)7U)g*OEbcDOE8 zM^p(*oJDGtpvI48Y)N*%M!dgykkD(8*gZBtuq_yNGlzBjVBIwUhcoJqeiL!P1onQf z=wax8%u<H`C~g8ZzZ@J=^4u(+_M$b*u}-23%sE3J1xm#*n~oo&7L8j1fH;ia+S_jM zV%W+W!j5i#lI6VN$0`rI&x<td|C)?=Si%!cnqOTV#an)g8MW`+)ld^KCL!+N#)C4F zMSSeFoVb$ma@T$qX5iO%*lAf7@?6?up&{!DG>w)gqR{lCFpbX-%%C=4Uww>>o;jQf zbF8yN&h%H(P%7fZcmx<k2oNDc7~AJVv@y^?blh)Y6O6Lc%$dVGeA7isnrOVX+2JI$ z*x}=>^AWzoN63GGmSRe&?BN-9Szp`Jylj*$--=|AY*|@uC9JC&#xYWc;I9-;TX_2U z3K+$WXR?Kxbh>!1O$1ssjkQ#(ZJn1SZ!OZ+)5m22HWiq4-v4?kzn*$ur{d!@&@4f( z`XomM{bHct!lyt@1nnzXJU@x(@WyecdLv<q1uf2CjOK4ba3sS`2=0PwY+$fh3U&4x ziCzh@pF?f>X!n6r+w>w{H$gMbAmABnycyxpR|P~;f#+H&6S<~q4fr17NzjoEPt`_6 zg_cR}gVrSRBedn(1Z2%4cAA=`q+zvyWZ_=aRqUd)6A&FD)d7~?Mh#Grk)4lf8CU@m zE`L`Xw5j8X$PNZeekM-b;w3Qv0k%t<#!%0SF&b|RlUGAc6}5T7KVzCXKGoYNd`>!D z)#nQ!(-MhD^>eBUf2Iqkwmk%S!$NN2hDJ`nWX@C5)P`*;Ui&m&IkxE`Qm{*s%eSer z$xRZqkctTB!fw-^LR~E;w~%z?a$T`<t7UtFaJy#31P#}j*^$U#NwlLGH^lZtBQ`gJ zDaEJ+i664KT`y{}x3<wrBX&Xq$HO}itzP$$Vbp@Usup787xh<S#C;@<=TI!pS~9TU zz(hIo;ptC7+)`Z^8ANa2k^>uT6Y;U5h0<WFIGH7-ftmuv3`;3KJ8KDKqaoPzGlNQ9 zE?hP3D`q|X)P$HX04nCc52s2DA=aPZPQRBAuMY?NBMtXTwz}Sa1Ga^Y*?7``tznbj zU(3|%-SKQTKEyAewDtSA=FDkF!2hm*KP>iUrh)I;2Iydy;Oj%Plt{~-8UE;6Ryj-I zGf#finayN%#3eYK(Z1S66PhiQ7=c`JY-M1(&v|kFoEB$S-I$xZq~1F1*O<Rv%NklC zsj<-FGk>wgXQ2fbpM@4(YQbN4l~Ou)%28|nx+^OumRx*N%Pu~W9jl75cyiRhilr?6 zN%QB+EW`Ndy0unf#@btro0x~^^FV`MskKO(-bFv8I&GH#jeZJ*$qldc{rUDPTWd8{ zD;7yTLN(iVB?x`^>YEkrX&3f?6vM)TriJmauaXY1^+hEo@C<dq?pL|#w-Y*&4KxPu zfzITB<QCBlgEUruQs^Cw9YdF82fbTE5rf9NJ*OXeirjS1#%XN)7qRRTE<xzX-$az( zPS{0SDGSJsM_=M43obT}ZUl`5V)z(zZw?Mc_#XIGjL^MC2EMnos91lnx7Sp3_hvXv z&w9as6c?yvNBFtE{5DVdv|2hPB~0)tE^9GM!oBHuQZ^N~N!777ayBNnj(F76!@zOC zTat@fk_S{&>H3{&NiLYDSF0YzWNKez$FTE@6J*3|o%wumMBzVV7~A4_69E7pZPwxv zJ<<1lJQRdz_qf@A$RXr@{nHsmJq~pTQi;n&=Xi4T5>dV!*<2U5;-u&u&3rRif8O$7 z7$eq|mx@y+`mXCafytDePZf!T1u{z{=x4~5F_}2;_<r8Wy>j-iG~s&qgsvd1s-yYX z&TSgB7{a!eVgo!DtUOM#qf!}5V-z>n_Smwzsed*j=ht*XtjB?rY;*&dr((;IBghS` z)vy{`g2Icg3wR?8JvI}<Wd80K5TvX4DgOgID<X6(4krhB2|8bzpe{tm@i2BasMHpb z-zswCM%N}^RWla)uCD$i=}oc@-TACOP@*D1NJm?-(y~bUxVSS&M+<IE`8k@F$UP~< z;Lq+%l>j(Swwaz0Xzt149B3UMS_p4}<3>|u{*j|!g_92@_`sW%>oEXP8)rDZg4`_I z+`pDWdZ%)VhZGWivGc5?b1W4p1FB`GqzVN~Qp)rKt5M3NV`4bo9!$#_PsCAvSY+?U z%d?4Ai^-3mWOF@bBoqcqZGWsN>6uWqMKr3EMU9m$4u^ez&^uuI`Z|hhmP}azZmR3i zUzVsN@YTYI%Y=vc=YHm|i&MsAi`b&VE~xrXO4|9*^kotcNYKep3yVQBH&5n)<AQVH zEf^+Pi&Uf7q!-P9ZZcJjz&Q(57LxPN7x7tC0OO-59I#O3nIb+_l3GP<;F!3?+af7j z?4qTt{DBJ2z_rz#DO;Ui{F7cLmiIoF_db{RJ{L@ju$(H(sj{3Zb>ofPJ{paRp^0nB zw>d?0%T!p)J*sCfc;d-P7|qgA<YiOs@(9qrKCwY8DVcZ!Ha64!Bbr{WdIl=Xtq~6} zMU})zOoh1^oT0gw^_T$#$raT;Ms~B9slfcmko^HC4(MSd;%rx4z9A^nVy}R$S+5{g z?j7LmFA>b26h%ga!w27+j-L$r=5*^$<ddh5?G8r$CM%q*O1B6qXvEH!`}wi>XhBGI z0ck_y#%@gH+@TG2f!U1%g5J>u^*{wSxGy76si*K#aAUJ4D7!Y0si%v<Kwz3bMja2! zdq>mhY4b3DtRk}Bq5`+aQ#1~|mxJIB2zC~BF(Aeq2CheE1@xBsV93=ll^Phbm3F#2 zp*yrG^${)%!@CpQ$Eh4T=G)~&9o?u?%h+yX)uB%xfwV_#l#?QEs&I4uh<>(5ga)16 zqyJ()EQ?3BdWfwqnfT%kES~3!u2G505$K{{0et^kss)z}vMI|EK#jW~R&l*3%rrp$ z=bT28KTHzclwn!U?ZBcuA$U+K6pA<U;ZVy&r*&W(+vqu1sLP{6u}Ob$bZ9J{o^VkH z1WH_koPibHBsScgw!Me@8kXU^X%}M$9D;gl1n&Yy-&<+V#<wxkXwwUKq~N~}XqTL{ zp`qGxdZnZfmktKEPAudsc%c8k`q}q?{qr9`|C`@={=*-MN*F9PEN?3P{0}3Q=7mH? zr^;^jj*LS%8&A5~Tk_46_q|RjcI)&C0vhVj@7ZiT%ZIexKFe0A8Te=F?U13oj(Erw zc#DlS01Nmpx6<+!W)&p*hky3sPk;L22S4F>%aL%(P&5{JITB?Vig9Sch8w37imLZe z-i;YOo&wg5eo+Efpa$Kbc#Vx*{FYyB%`r?pwB$6Ns9nAUyPr>7d|B^Fn+}GH#Yfd? zqyX0V2tm1@-Dr<5e)N04{;U74<2U}|`7eI>{NMg129*5cKinFRhcnnBn9}eXgXI3n zU%&Xn-+um!-+A#@|L606|DS&I(|;uvL?7#D$mF@9R5$#XnbiSnp+_pH<nurN-vF^M zJ7#S<ZtEOOyr`jjdwZ2ZM?3N6^`aBB9_JH$EXC6uT-t}0o8IN7_a*H^Ku=IN?hOjq z-N6V=2GH?QZ+Jv3$U!?=eC<d5bB8avUC6!za|6h8ghluT@0ZK$+2Oc<1VZL1pXaUZ zv)j?y`*@gu(-j*x$|Ea%gaIi>))R>*Hq~nPv?6GAA@(yGp6*iuPWGduUrtmpyF0OE zP%1lx(5|h%(;0{UMrtd2mJ#IOy@4KlL68<#dVK)1`6f)}yCankOonm5bczpP&|s3z z#`H%Rr;FdS6OI<9i4a)i;126D3Nf)iXdfWcY|?x0s0k6aHRxBo@SIf1mCf@8s&4y< z(puuX)Mal++F%$qQb%2vML@4T%Wn6k<Js73JLpw=g70u;y{s#YK_m@kr`O3aqC(Mb z3tuWwrBsmy`Xl@B1zyg03mo_G;OPcg2?XD}>`h(l9Sny3>$B#%I|;;5q~60Ej7vVt z7EE5-k&ma1%2F*$sx(brQPuk-RNCzXRMZe`ZS$9j6^F52)!WNrS9-rFiy5CP@;gN) z>;3jjcv<1h@hPl1NmopH8gj<<uA)Ype@4tGJ}gE;p$<B;Q&9R<Q<cCLR3<e>15*7_ zq4v4H5@O6$DSz&|-oodMSS&<r6wVxqFVsuJFiPBl;&KR+NidF{7EZ*qIj)m}^Enlp zKAV@^uTrYf^lsSjdM>*fP6_N~ejJqup>PfR7h~K(Ht3dY?8ufQHwi4ON5!Rt>(RJv zg`70J@}HU}x?=~%wNjDOh3dIc2F{bNTz|-n$;bFsK`@0}S5B=W@Zt~%V+kxSjy(x1 z)dxogcLF72i<Jnc)W%mzQgdf?c{Rm!$jet`YAHSp(HsF1pY(RZ8{8-|C8w6gEZ*@i zp*j$k1qtkMuw3k-qV2>*%y&gObw8ihMT#b{Mx&k2W=>>0;MAP$AQHHKD^M82{tzX= zwu_^SlRQ6ox`D<eg__HnPRy5>2ppPdFHRMjdsX#ZQ*#JF#<)qg>qNM5sxH`m0O%#z z9i$Ek5nq7YQ6+|Ii5HGk^e^Rmv=n0&Q?G9qGuS3iyPgQ~jZawgrZz5jN!fHusO4rW z^eH8$d$+Lg<q<^l3V{l?IVuNzZS{KXHOY(v`V_YgdHoblNfLs2-3p6)o=LLW6Zt8a zrFK~DXt|r0Eq^<LX8m~Bi(q<-U6iS1M>-lOwDum#08GUlb)m&jaVJp}9Z;dvc_SHx zaF!_+V5_0uOzdK{zC;~pBWEF+HfUtfwYfopVnlRiUZT@=q7SiBF&M$r8-R``_Khtz zQ*h(bnV?R1Xy1Kc(fH7s#0y^Jn0tY_M#STDyb>#PHD!v4TZRHhRf9+TlBR$=@BW}y zR2DkFJDr0<_wl7Jb?W+|y+d8ToYO~L|0OO-CThOoEy)`6sC7-UE_U&IlJoCjaXVhT zaSuQ&G<&oKG;oAxvhk4S%Y%*O!N&4n<0TWGfGoi!_<qrwy?!a3@I*OF;EGQQBbJX} zJ3{I7-uQ%zu`fz<B`}n!V_P>BNg|)3mv$toou2ufdN7{C;DKTf9WT`afXDt$3YSG3 zVNCh!)2S#Azw%Szt71jdlRWC}2CubXlb{ETDca6=<m>Xf_Ci3d1v;Rv7#rUNRoD&p z*uoQcjb#r69v<0HuxWT7Mt)R{Zx42Gj~Fo!Uhns_-cdOl9}*5U+e}Zt1wvy~*bn1L zLGNL5ex2Q^$~J!dI)f9bOKgAQ2}=}SRJ%!|7ctRN*g*t@fwvlnN(B%*JirgT!J@0i zNK*Y>GwR#<PPS@)?e1h<=hvz%DUV}3taT%dSHQetTZXr=P-oO=nqUPLy2HXD30}gu z;Z)OkPa+}zw8KhJk=<zgwweuoABkN15|$7jKrJhvF3wgrYXs!q0aO*^mv{7<0G1%g zkONSeeNP>2wDU_Mlt$vK@*b)W=njS4t15Ay?(+qmGA4w};rs7K-hJkCg|R_8pmCl$ z^a^I536+`bPE<=KyfEf34-$sqU*Ewymi+S)2e@C$HsIZx5jC0m;MVWB^*dZYQI}{1 z^*P9)uU@-9jl`NWD78A;aXY$UJGxq{qaC-S8?`$+)r>Q*r;CqPtWS}#d}Hx(jP*7$ z$|=aE^J*ei_g+}&VMPxiZW9?=j5FRmb$+|$owdaN86z;=T<%;BU!KNZp7;YzC1Wp* zWaYr#F^ifG%W^rK(+?lsy!FY$hsl%{dNsgFVEii48GD=JlD147(w+7IEwy*|-08ED zUCUNG$(V-zDixil`a!3i@vvyaJ^@rfQ>&Ox(XPuDmyD9>;>JK@s~5CBY)@$mRUS<y zsFR>4EGp*Qfp#?ZZ6vIdPn84s@f3EXhAz~FmY#{{L>EQnbD@mII7u7wD@Ni_Cu-5= zEM%^TARQ)`3d;Ect-4lr{hV}!^tj|oo^>N!BUqk#6+3Rlv{XIWfQr!Usx!k95(uT{ zNtJA0Ts|iVI8CoX=w3p$i5etKu21twxX8k!NqBJq5~lTl^IFSMZl&nA1v25Q!hokp zz0;Zuh>PoKEe5Q$Mpn^JiJ}w<-=oi886MPBti^+hiZyufm8nScc&maH@2dh;luWHd z?>b6Z)^#+v6m{zW>$t$88Ko)~$mDugo7cUOiu1cJG-tH%i7fCb?U<-uSE6YYDy3S0 zCyJ(pSK7H;_Fnaby>H7@@tT~aDHf6w@|-!NHpMIFurBmZjx*Bw-^()xuA@~F=nj9c z!4&JSBv;m#a;3JisZpp@I0HPRN@oPk1(|1#V7X(8ywV!hao4a*uf%H|m)HI|ORBFR zug%-k;KAHiOuzi+z$stSnwGNl%r`77LU7fI%3bHnnM>6$R_-MhF-2~lbXoupO^zoU z@YlwUXI`{Fr2jK|WTwD&#oyi$?n##k5V{2u_TK9!@yKZrPexTD<?P6o^sak6BYLTN zd=|O*WW2vcf(R#!<|IEIj`O}@1ND`T<v23Kd#78c6AbOF+}7fLYX8t7hWV^Dl5G;+ z2Zf&~mw@c*JO=@<qCLj0I)nErnbW`kIdLB7qo*^`YgF6hQV=QlfsSu!O0r{tSM)@d z@Drh)%8&N0m7(q6sm;WZRu8;$059HVEMzLiFTl8$wjaKe#9Sa81i=GhVW6H8IiFY% zHyVss?8XJrV>m7tKko_hks-D|&An+}M&pvG;IuIoO9ZBk7L5d@C8+GL7OV{1A|=~` zKbsbD3K;9eGClL4A_6I>gq#4@NlI`l*7SN9^htaU$grPJVpZUEbVlMpR`S1{wq3!D zz68O0P7JAOpso@w7x)^Ic4)y&;47#yIbJT2obY$#3g~!Fv{azrSel_*qdqHV86B>W zTbOJePcDv&EH6tW?O+zmA-{+Yp#-~NYA?n=p5?Dc6ah^&Bj}t+w;bn|g4_brSxIHD z{>I>)#u_S*2eaNm))YLGR6lz;=rd$%>c$J;bGnZDOG1sqXLnlUk5lJS)PmAO=QaZR zVpFUJtAb)Ft-C%F%5d^ZNW26B(PU`KyC@>l@!4j62gPsQkDC@L0{B`Ccf#WSuGZ5$ zUy7qOH|zEN6_}Zpbsv6RCgzc)?L%T*PjF06662N{vpQVMfHku)AyzF>s`$0KS>86! zYc<;F0_}NRkY>p3lXWeCW6TJj6$26mrf6C{IJ@jS9K}2)W5WLS6t??3jmd9(Jn@9N zU6oe#-(I(QwOoaNyblX+CNV*v8HAVIHfk8b8XEy(m3HkbIW1r>R=WAg9cdHEE^aEb zCrCQKILv(8)b_v^B3NVu8@o-n7_dTMMXT-W`06XOj=AqSESk&PZ53=kVZBk+6Dpm4 zWFrv{iwqq;Ro0b6ssN6bE>!Ns^sYUHVr{PaJF+?bh%RTey~31zyrs0N9IJ29doNfA z18@2@6j!xK${p)|Cng~G;$p`0IV>da0$D%OWSo9|jhp%cI^?LA&zgW_yTt1b%@sT4 zB9ZZfWRo?efRo-K$+<eOLxOcRfH}PDv~xbpo61a~g7wP{?RoOAX?8U*6wK+&Tn<j} zjiS7rbz|>Ot9d@rM!XY#!;vWxAYsa&K6_y3pIC$xHNW~3$+LY1PWX<|@bdD$+48>G z^1fN^`)0sx(8Xe^Vn*PE;ygXtN>2CEbtxqVq(xqSzBe6Xlq{%?aBdGU%(n|@IX9Pc zb2&E`$n1bou!qB9cyC%v^z3qsrCx-4j@f@t)!N+q2&i22z5D=&#dN>8enQdK;Wh5D zbq!HcucaJmlY`I!yJ(C!@c&?hZ$X_yo-6<)%sVRo9lBrjH7{mnHF~-ev=8KX@4B_f zjZMAo%V0nUd8vV$Y#0>n-e@>@9EgeC7(Cn1!;?TDOc2G=d_hlt^P`PdCRwS|xQ1G6 z3@tQ_g$E5@{J)^01KOhz9wR?H+p%;xlLPo64tkakPD+fPgR%T5?FQOc3;+h0HKA!9 z%N3lVIsB>Hsu<fETqW=L<|jl34hjR?lC3dc+4VYw09X9c!@6?Na?@%sYOleU-Ourm zv?GfpZkSdb*l3I>C=p<8p|#ylp7B)H<?=PUy|V4V+zAY59z<YC_l-k96$B?dvce$~ zWNFm^(&7mf%XDUr?U#%Ea*<yy@(V?>mQkrQ6^FW@TU+HCt+Z=;F0S1Zlk#iP0M>3r zEO*8QDXB*G3~4&rKLhF>I`1!>Wo0iP7S2nOPwgwCqr=@|dhLwYeLmia0!49xjy7t2 zK=CIrcC?p1o;XpHEeW?}fbd^7MGH$BgkvLXx3UBn5eST^jL!wCOcAWin`|{}^gu7! zA5Tvk&RcfbabC>^hu)jvc;CMG4(S7$82a@Z|3RU2SYa?d#N|S@6Gc}6Wb&Wc*Pu!D z?~aD2u_g@7R6F#yQR(Dp*tpT`Nv5<&%ZpuS@LEZN9-;-)VyAebnvt<0Sw4L_5wI@V zHKQ$l2j|_?`i|*(gYpAoSW5;xQs-VA?$qe84^CkRduW3etf<?ww8;gH?lsSBC$&#6 zyqZI>%qlR;RdKm0E?32cR>d3nuy-`XEoM2S%zPN24Cq^eBFcM_k4tE>OMf;yE7|5A zo)#a#j%(B(7VFe1|J})uv?JPZBJp<%0@lYsw=>FBG|5MVOo>U9b6#4}4-VfW9PurC zMBZchZrNn$w?=(*8(+=LPrTZ)@`#aaZV+Y~DP-O8ba&A27b8TaHys1P6`C$S*wi{h zaTZ&(9Kcp?ILosSkt%$=+pb)@Ep{%VSr6I*itG;d_jyQ0vOs5t!8(tCbvzgjnV@}O z1)%SM<tuD3W(VVbHBJTl3<_`qoI=Rqy%|p;E)v0^&~AH}aOY?U9Hv+oW+3cR014<< zq22M?5fXl()fR4_SIrwtjRuJEMhiNTj|kL^d5d@unPGN37HG$ypOIe-j^qwW$SJ0v zq^y<sEdSmZ_TET%6l&01d5Fu#l##~fA|2jjOAGk{Ngj7tsEx&ti-3D_3<9+}0_i;) zJSjqej3;dQP4X1rw-pUs4J{^X<LsPck`c$qA5(k*6+)Ou8`bR;QFvO_Aj&YG?uRt_ zj#|V6W<q#_(d2064~)`8S&Yzpar97}MlDgNrMRXg`J$zJ)zWf7=%hRoud08Q4ZxZ^ z7#z*%tCk}{U>Fv*+8<?e98lU)D}A&?P%{;Eqcs$udxKFyGb}TXuT<`}$holocGb+U z*E>zqr|tEYKy$~7j-}jR)3DVy?O2<i=;DTKx?c9KdDFG-wxS(57x{#E=vmd+*oUQL zqmDc-V**)NEu89t9`=2bBrYv&&r?u*wu#`O)%A6$Sh`{s#ayHf^+_YDLA!KF)$2I* zEW2`q_I!AgvcHmxGF}`2-0@daP85N^g&~>{-*@ofnI?#2_WKGmTs5qucjnE9me}eq zVPwN%AdH5_`CI`}2hXaM-=j9U2Vcq7fg78)I^7z(>Nbl<?Xn2+m0u*Jy3_9z{biw+ z=!Fx*`qs$=B|xd<sp>MO&s}_Y5@G@kh$3#Jgw2G){ZE{PFcNNY1bamf6vK4wt<WV{ zpfm<-X~t0XxP~<x?WpZf6J}a7tqpaP{FXraf^OcQZh+f^1JDQ_pEJ2#$%*6+iFVCd zR$jH^;TRfq*6*}hO)ad;A`yuV-8rwn(!7sblnV4#tGadB3<a>sBToQcz3Pg#N3AR$ z9_Oc+`x|yEq?+Yo`G-md9I#fd;v0?kJPC<g8dQ~7lS04TwnLm}I7D5$&R_>u(AOC+ zyisqjkt|uo<WKaQWefh4B3_l~oL%+<ObQ7>_8=vh?Ugo#iMhC9d4vmMH|!E?-*vOi z5uNy)3U6MsN~?0~k-BvCdev0s`)wk3&eGVwG;1u)8ZVKyQXjb9F!o0_H(bH7DPeG6 z={p`g9!!e<AgAHMU&iwP(Aq7_Yif7H`x`zW#ry31*JXYU41WNc%Ij~5dvEenej+~Q zC*srXAvE>+hW@yRKgH)8#b9W^_TZQLNV&b}3;)%*tWmtT`T85W!o5v*l`#j#+D4p8 z9TD1u9~Y-6chzlM-+azz!0ju%+i%S|VL_YiHDPO?Ob1W$nQv|8!1#C;{UKd3VCRMp z^6~(0#l#OE5?GR8?PkwxCh2Wd2X*G8>}+0t`1c|>n+w&?&Z3xJ=S6jIJRFXXl@9=> zN^bk8s4X~Sa;@eGg&{%7Zp3!5#iy$C5HD404_1%0_mI;73A_y)J$iXKg)cRr&N+CC zo>_1Mhw8?I(<Rd)m%+uflEL-Ls@sbn(x@aX#_Ob2d3qB)N7+O#5OPBRg=Y%FWPk<+ z>(^7#p-4H8si;TtyrtgkwW?x0UXXR9V}jl~yH$e{jthm7lTF(|%RiQMbLz>)^GKtn znQf>SyAPA=!!DG~J>&3y16FrXf<_pXTYzfFgU`S{Tf@t)x<UxuwHbNJ7%bp<c^UD+ za^YJRVh`{v^}SO*3Nx6;E|Q~#Bzk4h)Z&9Q<km(Efa#|DX(it4L3u+R)T*OR;eTb0 za6Rf%vGM4=oZRpQ1lWXmSniURIz4^gyGm`q3`XR{t{RuuHuys$L>sp;Bb3fo;0Gj9 zgNgsdTGdj;REQPw&oUhxBOj4-9AwX0na)DVKVu?8r6eVU>BK$D_hyBPlA&A_i@uN; zWcdvwEA?pY{o!~wAD-#?Ft--jMsT)XzqfgY(!>))oLZ=xh!GN|y-`^;ufyy?cQtmg zYYItjWLGs&Qs>dQodK1c?q@ji5nlxMOg?Ld-PYL5>%;tUamTz4;*H=ayqA|8;a-<& z>$jI<;KSnfy)R=2C&lpQ;82S?QOEh@S5sad)JfXgJEi@w5fh_`aD||-Ew1*z+`zS~ zru*(>Kvy^_amALPC`Jx}?$BvjSW{th?W&+xveXq>%58)H<W>US(Ow3%d=!`y>r6_R zeiIDU1KU*c>trX@xvO_<63k!C9el%JJI!=(9haQsr0Fi*YTA_1^l_o<rX^T-i6~q_ z=7jTcD`+*r(^)-TA$d79x<>Ew8XdbVMsl2_YWrve(AgG@qaxC!g^`62S`6F>wx6_; zs|YL>d`qk=!PP5w8Hl`!GvWOL@BP!hH|8VGx}?}W)N&DWRc=?sn+Iy%YB42*D>yyA zW=u`|rD)0}d|Y;pCSGzYMEIvpCMJC5W@Dlevp-|$<5*as1Zd7#=(I6((d`#thOg@h zJhe-!KqDc^+~qRanA|{Ul{^%+x}+NK05$#n1Xa=P#tDY~KW+HS+wlQ?<y+`bLS5?h zi<vYy0qa;Fv|7t^><;TX+MA;d^7=%lQ@z>t@qxGc#<4W#Y^eK7IDT=IF*h(Ok0zg$ zpgI_VzZb+ReB7nujA*NPGeQ}!G)AkED8|$!F0Qev2b@YbI4TOC37eknq}w3lXmmi` zG7WF8s$B`AcrCkB`PD-*T2^-XTGpv3!!*C<M@)h;4!c|vz76=m4(E{sGU|_yAE80_ zdapNT>$*A>5Ozy6P-{|hMa4ud2QZpsOPJPJ91R--ZCj<tDRk7s^cW8FNm+0Ypy~Cj zNnR=?Y-kJ29F~pp>2!R&;(~`VbnL*j`+O@!`4a6yqoCKrmx=MPi18_QciupAsbCHt zbGSC~mee7?UZ7y8dXj_vQM`~f{lUk$T<IASnI#krXI|I16i*LfF>YEmUCTCNn`w`J z+hGp^R8~#T0j6(d@nN#xL-3Zu!iI#OlfkD3)peVzu{NveelF<8gs+?xS232en8t>f z7`R9b*QzfyB(|7z@X2`4kI%Z<c+75WV!DI{uVk0V-ynwK`id`FZV#02<=$Xga$=fc zUd~K(oNHMD{Xn|lrz`x6o&DUIwu!gaHf=sfcL*fws>6$LbfJ+oUtYE-+TF#h(+UCd z@kkn8Ckby=3hEu7CiliAdCT;qM4cm|*0Ivn$Z!N0I5W@mJhm10CJD1wSN-cp#FVN< z*ZtXw!NE;+mo<}<Q(rTQ)?mi&I=@<>kZ=Kq778D@wHI`aqExSzUFEyp?_}XmmEP|$ zMOfS!#BTViq6Hk$Zty1B#t~Ys%%81b8IqH#>%W+T0&--C2EG+??WHyXc0*IS37Dta zt2?UB{@p9$9A0=zZd7|-#J*>C1%YA7nT&{;pMDgn8jbSB<PJ1=&oCB^=+)QviHsSX z-CO)ner#tbX2R^CJm>Iiq$N<GX@oBwYlFK4EO1ogV>}s`irh`Fmv$nC3-P;|W}iX% z>0oxC-0!;HYg)|liwj|eqvY83;s+V}q<2eg6C&!yfw;>Q`HT<BQSM_~158AZJnh!8 zfD-v55Of6aDn!=PE&Dh|8ujcI5%$xk;Wa5nZU&$wR`zB+Y&Jbqq0WGhGmzU%<DE8! zX_`{R>R9mQXq8i4<<zaQ6Gci>gB&abVKlf10cX>ERHFBj@l|-e)ATh_0#-Cl*;Xe0 z)GhgysK_u~Pj@(aIJR~7=H1{WjL+j=|J5J7_@n>h`Op5<um0}OU;Or8|BvtgPTkrs ze)Qj7{NN|nXCZs>@Bhn-Kl`Z=)_?!!KY#x7A4Wj>4iei?XD7~W)>)%s?rD;Btv5$l z$dIZls21NLe3Ty+*PrBrAr7`$`t|gRe@_6S2>cHYP!W*z&Fe_D5y0M>9${p!;=}yJ zYd!Wg)>jzWn}aEzhjs(486OJrCCj9RLoLhn`vn^b&~XihGqgdw_o3A{&(vAtGWJ=v z-7m`CbU?uqbUajH0>whzin9a93Ys`Ch<EYE7*}NICia2diO^z_XC-tD#3cs*!xwz) z!G|ETaYiSTAw4dfF^I(oMUFte;TK!l8V4t!%E7gY0IYNZu~qeEYNARawYfd2$Bp<l zbBlV$#_%*zx2E}%V#vfoJJfhoWO)6%g=3lJJ-ku%0L3Lzmaa!p(LtSa4lKELXHd@0 zs8Ue;Hk0Zs>+O>H1w@0rVJ}*4@Q(&MHa1`%D1LE_5ph>N{n-12fQ$6Th}_+x0Do6u zCrtMtDS)D>#Rj<<(+$rcLpm5Z#i)N(25x777{pM(Q(mmJ;kR$~eM4@j?Xet=#C1y| zXTs458b%J-&3lhqjD9PVPS*ISu#n<vgbozxBnP0lj-n99d6az$!`@qW!E@SGU~@NR z@$Hxyo-B9ftFK%qbZe}5GjhD?HrdA8fz9w#QrC`|{umvuDW|Zm6P`?-g`_;lOEt`e z*~Q^I+~H3Fa?$rrbWLz%%Sl&xtS^gKS8!a#i$t_!F$n{$hK#rpz32(CoW@tpbPdn2 z<<FF9(u1YMK*Y*u($th9F7qe!qUnxn%aR~wt~A8_9H&JKuOH)I;SmoMxI$myuLvzA zVx_|0vdH%5JBXl|&MejOLp->troV!Pr(s4cR)&}LNN6fW`mUd<3<U8owBAqmg_cWA zfl6W&bjYRXy7lQnK|b#Xc>i~Fs(V(*m0mB}yW{b&$VdLN?arf0Bgz7@s-E!k=CvEY z4WIL3J!BJ4flvUDyI2PK+@~UesL2{-C|@qQf!;huIEz?SC4gF5AL*o}(LzYq4Yj-M zzVe@<*V%2nCV!G2PS7~H-2pmotnXL`sfzwn2F2zph8Bj$YQCN)nT=Wt)kM9j8P~Ds zuI+L_SiH4zx}|qmu<mWE2G*5TifgycCamVZLS<x&EBtqqQGBl2-y4F0*^G8&j#oxv ztZ1Oa;o!RhXW6sB0Gz3N@Vmu9{$zme<BiXc4kwN5i>yI6=jpEzpxHQ!^!5Q3>0b{F z&FAm<_>!)PkyFR0U68+e7l<_wdkR?l@Xv!#RG44ABV6yRJ;x1>gzg^Aw6>hXvc%`* z3Gb^+sVFK2l66v51C)r42&;u<4B~=Tjbnu~ciD1~W&y@NW0cZ~{1*4gEf^(O!KT;r zAs5vm>5*_CrA((0Usv~i><l6}hXi@qpXU4f=+A(TmTZ6PVHZY&V$>^I)RALmE#*YO zVJsX#G?V%qMB{LL#CPAwrGXE*_3&9>UZ?d9dihR|45}_XRin;0?qO@dUE;G>9ppQP zhPjToIo?vfI{*oREnB+bogTk;G!u>-<0ITP7f%LAJ(M-%sZpR8c|JY$`VFM|7g*$~ z#710Pbw-RAcdt`X6+nYEt~HWK=<REH?24KF7d#xInt&V>5~LAbQO6Att*j<m2|ONB zDVrDQ(KXRl6HjBjZhcw(3fkJ(@pl<Dw?54UD2S<iuU+-c{`$c^{Gw=gkVIE%fn1-{ zK`wV8xhC1O`qV@<P(ozh6`^l*xSMeO*FkSM;7SO^cz=NOFX_rG(t_@Bvna#ZNbRs* zH)~3sp|vBCW)YhB)OFlKrxJGF!T>6d(U_S%8I%J&2BNqa&@nnv&+;K2%k3Xvi2;T~ znyL%QAXZ@$BoK=+)2lD;8p?z`Eo_vZ3zD7IBHlar#9X4x<=o~vouar7P%HI3pv>jp zrokv2d&jS6p0^f9qKQ*zj%!lfab#iV1gnVXzeIalSq_3&7`H9h6Pma;UUUUZElYe+ z9A#q54z0PXY1&8}64P3wqDYCeVHq+5G_*<VtY5intyO4)mz^#M;@UGRf_}kXw-W?S zohxA#ub4PB!hNkG^c*78rhw;^Tz#D;?k=2`onAHV0vw}hP;_CnDeiJ6DtaG{>LL%E zJv720in()nmCS$%ON#y;JFzo}X#GhWMLmRll}8+s+gcYYSF2d_6+KdGgi&$?a#sWF zI!Wy2X#K93xfpata$DsV!d)3E$1h~kV7j?>BuR>@UK&SF$Gfwm)-e^d=e!4Lz9@<u zm#0L~+#_vID5Z|D<44!Ax1LB_Mz>i;w^>HFdAaB|f&EFj9Zhva&LA_)>l^VK+Z2Ee zp*=V)?{H)>agZr*pGM*-G+OB^e!l0RYzP{WVHD<t?65#I0CI`RL}r+W4NRQdmu5** zmTx1*t{^(F{hPAEwSEmS)<$)75?K_9%*V6(jgx0j*`Re~1MhAR<ANUQFki$)+i<kz z-WAq(*#tb|4vODmV`W|<p4iPljn8i#71LA0K4mjvYB&T5v=xS3@rw=E!#%*)N$C;| zhB4?@RqOj(^0Sq2|0%0S*rS?%A}5S?xb-oGU=h7a-tgdjn2$iHOxrp$oHH4p@m%kk zIe)f{kH1`MmrLz}so}0`CkcwOW(1cKt`xu#Wf;$06)b8uFAM3h*odDe)(M714v*d$ zU{3yx>9ieAz_7%ra58QfdogUHE$<0zWGO^i<sEf(&KaIlqT%*IF@%qlXnEeHbhJ5A z!4L5!=+yH=+zzGqb^~S}N;+7fhT~}?79MH4F&vC|H-iOt$xoSo#)|_-j~^Gd?2QW0 z=C=891;8;(*}?1-i|Idi+;ZaQ3ZF}x^eBzorD0=bjw>Z82>pVk3CboA{YO)ptr)Ex zXm9vAgNaLm%*#BZK<yU297UXQ64D{j^C9qu3P+@vmLme;R7049Q8`2L2S^xbPHG~w zBw|@KJcA5@eJajwbt3PupSUCZ7{sdqM~8YjGZR9{XCM~)S#!_^0Xr=A(M7Na0`KvP zIX}0I>mK9NKYDsxJ7X(0a=!UXxw(+ivu?oDemJiI<p~%L)%247nn$~Us5g5`?F}R5 zp`ASj#n4;Bt)!0-!`JKtVFCH4upjBfd9H5g0SXj9<1#cr<AqyTlW{fcT!!y>J_P@+ zuEw4A%AViY_MqzMs?t*{k&9xReB5^UAS9oGLN%GGTlD>6h6;CMG!AGAW7BBOWdrsN z7a0~PT|&e&Oe3Rxoj|Zh+pK6(`yo~pF@ARQSkHCD3Y^JblI4Ok<JaQck#dQMxNV;K zAerRaQ}~N7GPG%Zq|i<RjyWLu{u2*S1t};w=i%UK?%u76%nK<oI?>&co_XOk3AY;9 zfG&?mLRYRT?)uiZ)F1H16zIgy$aQz)-j?A+!Y6im=ge?+3QSIQoaW>h2!AA8QX|BJ zd7kWLq-dhQ^ZV>|HY7u!p#;3bX~|f4vd0MJ?&PeVuXnOAj3{TItD&|fy!iTqL#6p4 zUjqF*t!$%}wcBlBEpRr#o}!rhN|~UYiiNZxuNVrA$-PA+)FbHEbbmK*cHUZVb-w*} zYva<!irldG>J(DSQ8!y}U-D%+CTzePA$%IvQis>HPtc!KP{QoVpg3mrprl?Jx;zE+ z*Y!~)XJi$sy^a0avKkL4`-kJvhl7)Wx>6^@^kZc-QmBNtXgzObUIJdlH!A7uOr;^z zNoeU7%ex|=W@Z?~oOfDWa1%dvclZMTmWxu$XT{~#f4TKvZv7Y7`q$g&hf$&X#ROxu zjd~Q_aodeHoY+Y(5)|8C5PZ{Og<aqlI<3<U#o^#FVh@YSI_^v1-;VlsqoEd)=%Q3R z__%fPufXR9h!NQQjdbKHaj~H0qL^$y2p8?4!RNL(v}mmvpUCkOa?@EIpAL&B7)w;H zj!)g6nDV5Xy(3@E#*=RLmV7hiK!WeMtL$txKJ*G=+$>&qpTW|U4`~rT!{czwtg_00 z-10g~Ay?oag;)cyfd6tUEveb)P%<)<UCV%T0^H$UM}4Kg;XCbae)q#$&YMww2*diW zedmUQ=(3kYEB2j(P?-^n6yR_v>(P%&Re5vWc7w3X8lClTHnJ#2_i3loc{>X5W`W_V zdj-LC`nZkJcR+@+OYIh>(3|bFv&|8T9rQ8=Swu+!alP1Pt!=%%{^m}Lw<v5G*gJCK zI04+7V_IeU;~q{6s5}KhH5&}09=mCmU=Tm6>+Q~)+$jF{jh$!@Rh_%Q?+vn=G)Tp2 z7*|6ceEt+WCBtc^^49aPRc^*pS%tmm_KsTbiDRR!wXs&4E}W<&)i4T2q=G~uaFW`` zQ@i=>?B#+@3m6jHQE`0NRsQfHmu_g8Kk2<O>wgY<raEqSSUa`#TK02`G~ifpQ6z-D zRX8WWsMem<J@v7Qp*tu|O3}qb%5to6K5G1;Zbs^Q39T!`n7cv19YdKHTs#$=R9|m7 zOr9Zzcbwe1$}`^GO~Oc4XCQ!S=78aIRTVJa9lVmYd!Dbdi@-~6h-kiP>|(O*$HGL{ zV^7|C?<f|h^U4^}eHX02@GWKq?23~6en1^B65l-@-Q$Sbr(8a{pBIy78%v^hEd1oZ zYwYelG!dMAGw<$KIwi+Vq3pcm1<u%g@>uQ-suug;-8T>Jqr*l8&xg5LrJ~OC^>8PV zx_d7@oH#_8HOk#9w?ZbGc!r@qTl)AQs5j8LP<VrhcW2o^TBz+)fm@2xV&=KGt7&JE ziq2Dt9c!miyQe3ee@#sUGzsR8C0spf^u0U!s7E)HPoqE;-v+nzgR#_?^=(E8*@NJ! zzG8D@g<9_LmplCB4u2s(I?y>zhvU4jeyF9$9?UpXG$<84WtYc@U$a@U`M?t0n6+(~ zmLWDJ;FPx~=Grm}TTa~ScwdWt!x(#YGL{Da@yVb(qWu{gI<;reb!xgPF>u}Why<ak z6!-bFLg-n<SBpI!GII3lW(u@id_y<E>q+4nlpqgq6*M3`Ce=M@J4c&FWb&V6UDfh3 zWP-JTY);o$^=W)*<Xez*)nxn7(8*w~u}u(Wxs)uIlI2pexSts2fj%W4G5YJWOcaW5 zy+olCDtbs!S!C+ax3gsGQ2WVA9eQtkY*T|y#>eI~^2XgyZry+QJ^0jFheuWU7<k~$ zts7epw?4Rk>%j+i@7#m}Z>f(TZhrI--fn*H*26oSAKiNR>E_L?59Ifcw{CqGOWghV z{*7A?|K+XE(C?Z93Eay^4b%3`&HK;>4BW#Huivw4-Er5=WLNO-j{c1qIXmKe$-g^0 zyq;07KED6p?)?C4qd0C@pyC%AwMJ#b0u;YSSaejBrf_-ey%^Dp<I*@+l!bm_QM!5S z_VthNY(0GM{>KkKczE~Tmc?urBm0CWt;Q*>>qJgv=QDj$y56GyYhQBCNzv!cFVR_P z*v%{H6gsjB+gj<=R3M}b-yF111D}$q@9yXZn)#G{R;v*d`XDchk2jH9*4?EOc5|?c z3oJfrwOczVhA^?#D8|tZTnEOmFip9t$f3zE4)L%x$a_O?*KVOMw0A^DAW)Q4qeYwj z2KAo+h?VlN?8;a8;5j)ki&XEfwb&{Klp0cM4Umay*9&rCe>y&*Yn%?4)U&yVZa%iQ z5=#Z>5CD}YRNDtOhK1u<tOZ!w(JG{ZOhu9KMpqT}OA=|dLL@D5dL-1p|Lgbe)4ylg z)2ULKjAr6dy!mu0(vhLv=i$Sfw?29J(9v1)vIKn^Hs4U%zvHIIy^Vppn(nrJe#%-O z<9Q|>wx&MC3-nK?3$$a#a_xQOgqjI@iEGpi7^;^@XpHq~8S78;V_d76$^mnPGrUhU zx4w?_U-EoKd)<`w6QE#@ho}4F5gGH<>=wm=BPTEl2%A2xp+R#E*StM|L(!p1(hFz? zwNrH)IVgFJ7CF$tYC2-I0`yKyGs{N=ge)q($)V6BR$H*>2<>Vp=RmKyh{S!I1tND{ zlYW)yuntV3S0XWU0-V`_?jA<`h8Qd?3%$p#wXhE8MDRL>sblplj%*xW57n`9aU>7N zN44go+ZcM)0P;7m(P(pzaP54gj_OpzBpRI>2vy<pMI})+UiYx2^BXJX{`a-Ib{>FJ z0$qnLI)Z(8-<Rlh+N+xTM+1yQ)!21c8oKtUw3Le%&uuOqz*b<$EGcrhE5)88c@@(S zR3T{<P!(Hj#QDfS3UQ%tn?iW1u{(f$x~cDm-ZnppoZf01&tdMQD)(wck2^N^$YC8+ z5qA|><5PR(Tc?+Vet0&hgWn0^7?j}=HWNXXZ5!#n=rR_4US(~Y2XK)EuLfj%nhaBV zj~}541~g+SfmQ;FnxM}CDN*CUsulhnqx#di{vn_^m<)?X0M87K1JK!ATt)d9#<Psq zbd=DmG!$V~%{pNxXde|;NcWm5w8@EP@oHv|7FIPE2ohC!AGdft+}qVe^r;{Bo@G0+ znmmRSb%L9|F}aP8QN|r1?%4}{L1t&{x4xD2$2WOK5k5^R!0?IBbgFl2<W_6R!tQaM zh+;qI?AE#P!~BJDJ8=>L;F5P^Gn&#Z71T;iw75Vpo^++%)jM;VKyUWRjph&!-9hr6 zMrVI-IGEhaF&&vx)Qn)X(x7B#qxVj498JsdbfL<=P@>9ws-6#9*%&ov5;x>UZHEVl z!g47J74!7MgvqA__b%GzZmB3|0(=E({^ASk4{1MMj%+n#0WaIZ#K3#{N9tAypRbP! zdtc?D`T5Mx0BAs$zlx)dg$>ZwT!b7pD7P&-iZ6D|4J=#VzmbuM+ZMZ%;dMa8XpB@| zbjBD_4C5!{k{{HXzPCN)(i896dx~B6(*7%G;~=jNJjxgPmLBJn%yLAP4&y^%K+O%p ze8#yA+YW7fIp%ePk?}hhI9p)d_tjqBZtO4nz@JXrdB5K@r9($9dfm=vv#B2$!3Oq0 zgiH`3Ad_QKHDhfu;$U%;$B2^ax+N)9ITxcp38b>R8bYlEeUW*NEaoT&#sI~He{R+n z+I&j(1bsh6Z0g+>*~yBNn4{B1Q|=m}q$pj<;ve&M5v~{Nul`yg79k3ZatA%AeG#@D z6O7L=*hJ$5^Hv*K+BI4&P-!2H;2W1(*&A={w1mO9<yvvgDPRQp>8S@@>B<_q1y6sy z`Nqyp9GrChR?m+A;*n{Hk`J5HBTSTu>mreq&g0rJdRB$y#bYG^H|P@_ER|j=A6dFQ zYT>3_P$sP2qC>aPERTtac=Zb!02RO=V}$DM6YvJFGElHCYqvg1!0{<OadDZ@!w~qF zrHlD}pd%)~@m3A}5FdRppWIP-y$Wc$;wWg=J(U3be*gdg|NrcLU5_J4a@e~7{fAm< zC9-cgoF->xPh#)vV7_*Dd^;bS>ATy#c6+XxtnRKIHc7MC)8wq%Fbw-mfB-`_3_tlz zdaz*__Je$Yp+BS3!T-V;5&033AH`x%@7_sbNw?Fi%8bm6jEszojE^juP-Jm7dpdeK zS2zXBwbno1y@?QRVmz0den92vc=X7qg!o$bicN}(Xt9`O-OwI>GhuFNzjLjSgBR5m z!2`y>*<;M=aeQ3N+qTOce?B@eoG*&SPl^?KY^i^DykIl9L8$+|m=lZgKNEq~l_1J& z(?9;lVp$B&7ey^scJH~vo9b&@Om#^%t8t>?MZ4w5SjcAgGw;Jd(8CQ4HI$`pWUZU1 z;EA0bQp>^i-=e*ZWU;36C#-1q&6~)m*EY!J2gpMQ<W${w8f{pU@I%)UXn`d092hvh ztt&-%r%>NYY}WOGALA9-7))8<l4+`W92pUH0w9#7-NtetMvkWdL;FmcQr!1QwK(zY zG|{kEkJ3)M63OkFI51xi(M(gUs;CfcSGF5J2xTEWUxN0e_(WW5VhFU*lTFoFd7I~c z){cLGC(>I<SJXJ0{zR6~1h9DF$SQ(o0GXE_X}q@`GwkR{Q3zbTmt6~tSm9y&-UGO4 zA^6jYgpZS)o~QUy()=vBb~MQa<n>V_@^bwnnZ4_p^C4~$e&T!*9CaG+`7UXM-S5*b z{SC2+;Fyl6jmNVf?uWaP2Fs%s;o`v<+`rBjz+u>OAF9LOoNyFqc<gaUW7D09WRsg5 z=_sylaHfN#b*Up=-DyKdo=MISHm|3#9gJ;yeB5R?)!MBb>=fF*g56$Vdrst1*~QmM zOeh0Bg}2g-VBhc?YPJc?r}<e&aqd!y<BR-lVboc0o?T3*8e2$a9GZ8?mDo^V+V550 zLKC2l=+b#P3U2Tbhvtd|q3{~!3dB(c|LPP__iFe4ppIp%SCU^Zy{QuVOxIT9>6xoN zH?KBAp<pD}fW0(^w8lt|Ic(N8WzAdl8T@qf+b_i$)aGEnS>Pi6rR-1)6W84(wW>Gj zQ)<UdtzaW*;J;*%0$WU!un)UUpj+|}WL3Ff_W1eq7wSN%1>q<xZ%c$DiP*bsj7mBR zNu1P57r|a^v0|{Hx{dB~>{VM<-E;g))a<gB#mpv4)%1@se1<b##H%5Z&v+n5n0JQs z2;9j~uTS$?5s{rZ*5p)22taos#3q932F-DVc?}CX>7rh#6JG8{)@&WQ3Et_k_xHN< z??viYD5cF>CE;U=+PUJdZnb+>@x{0cwcqc_SABdZc2BrwE#IL(niht-X`a+v>@}TP zHysdjCM`8Lppl@Hi59&mH6BZf&=X@#rVWKRBV*x>6kU11wX!yxX@{0GJF;td%kywW ziP89I6orjJl}vYBFGWr}ae>1(HfuR$B^-Ja&3|p~2)+pHdM>?Y_S|o-_vS6z_j__D zqV~n5krb_FldgZE*N%*x>4SH0F%YWnsrq7rVgoc?H&dlEQ|4}#D7#2xI>JpSWI#XU z1UwzZd7QaY+p|*ePb+<TfkkYT7?dMM+wWGZ3iZ<+TZcqt$`1$Kf^Bz{<&ti@A#R^? z#TLKqgFC)~$tB^54D8sH5G1YYk|S2AJyA0{FHg!zKDLQm^kv4Fr*AP-Xf8;2(og?L zoH_~%26V~APr-y;*gI_@&am~-N3yp8R&=s2+$bE)@!LW4paZLHQfhzPZ->~aAKxaM zrYVnGoxQFNHo-__0Phsbu4SUE)Km0m>XC(wIf19Fq}v_*hn;^pF9_s0jpjDl&Ou^H zz-UKkCdimv&Dd<w#UDI)jWi>tLgHm|tE59aBn<I+i<n|#qw2BN;afW#Qjfr^*!8}i z1)hY*o<otF^zn9B)jr@bAS~%2TU@iAEw;<wkKoQ?)`tj%H&r3;<BC4^^hW`4WbR|M z%T9X)$!<oV^H%lLUn>aFayJMBkM%x_`0d2@`F-fInzE{meV3Tc%dr_055`I*@aw19 zE^aNl=&rY)92kUj3<g!X!^4opAp?c{pRbA`$sTQWgr1kji%ym}$f*5(bW*^%tBRhO zVixkWeBpEgU!g62X%1l_23ZSSEL+0K2EB1?xM^_(#><ss^u1-yGw`6jbKq8ky}`X2 zTAZ?AtXQRlxKP?dERG2+9=9gdMImw)OefG{0ZY;Vu)I=&_)f1}=+@<qS%(f=W-DL< zu)%oL!!#PaqaNiF_3+<_9$QbqcP`?iADN>b&ub>gJmo}ZFHDx$U|tXx5fC3XJLb6* zDs|zE3xCy~w&_4%)><gBX+iQ;aAcDy;f%3kOv~okL!u1{V{D`)(fJ&$vtrwp@5J2- zcNhpvE^WK17>wkrEBU$SeO^M^&m{y#vt7KE^?7|Lu&ODr><1kXynEFTdLX_p``CrP zYJy?oM!G{-e$}}Ag`OI0VaaiiA>ItEm0tx~p}GuqP#sEXK(ZJ=asqWFPMjUXWV2-_ zQ@hv&%{qZ}WDNUu(vDa#nkFq?Ky$fjFYhoMnpX2re^U<*jB(T~UK_j#>mE%72fHb> z%0TNU!EM;RB>Jx0$4MH6S0DEUANV!Fbp2cp^fIOJNoZ-VqMs8=1$0A@JXn0&<FYKQ zx+h}`e0W-pM-LWlE4I=1?s=spow&^7mG*59%{Kdy=to5Ekz=^8<sb-Ztc|XWlX9I% za)1jPmm{trBz>AQq6VTfAmGF1QVY7`dx&~*45AN}-wTV=$^UVw_0eCLpT#iH?->vh zNP#NLB1C9o#F~dmGHc7GCd0YSAJ~x{qEQWHM1y$>jT>TC$SEag?W#jj&}<KCp^1gk zU}h{bt3o$|j#t(G6r~<UjN@|laD_^&SU(O9pbuaO3Fj`=la>1C^zcv%WIN&D;%mm! zooWKF+TW4u$MflHGO9&suagj|!p10u2@DD4j5Z-`*#}-FP<B=VPdTtrBcSB6zteGX zB-$&i#HEz8e3`Lq*Ip%j<0{F6A6uZRX*>ZcXzJfXwC`W~S_mxF)1`Wchfrw63x$V9 zI=vFn3N?_AY+kVK>#U;M@4Cf{Ql0%|+X7)lelkbq&%BflvXqe%UA1a9E*I^V2qNh- zm;<@&HtI5_%Kni0y+d{!;Z{^_+eOSo4mtl6pHR|&`2M1WI%)f%J1^)yCZZs7g}zm5 zm|-T7Z`TA&p`ck1kOKlBt4#ncok2QNe?yp$#dN0rsxYK8L^^I=dMH<n6gu@R2Q&Te zC|Wb_H##OWZaWyU8TTF&m09gUJ|sSQmeo3J%juxbA4s$xc4{cz)sNuMje@cAl)Qpx zT;!39spTu3?07ysV?UD|d!$$dD5i=lOEz=U3Si20A+1Wq<0zu47~|1oG5yN^6m(3T z3#AF@x?6S~PdIp)&liDOxSYWKf{0pkU<x0KDbW2tP&I)4;lIG(Zro$NHeW+sY%pvv zY~qvA_V<8T<9I;suopf51)gBN62_r}>4Y{n+lwpejdlaot`X%glHp78+>PzaaXPcg z_sG2ue!Di3(Kp)cj!s-R!h>&s;mMg6>3XP4&o9%95W)|I`E{re8e-<f<ZrA2s)gH+ z<$;)@bf4O6J8X-pzbNL@2DQPfDNurGE=u}$3k`{1f9c6Dy_kNwUPy$*Ea62bodecf zkNVx!m$b$w5p7b@#LUr*zBud}PtZ0rFKxWs#jmS2&{tB}*<gf1I;J0DL`xDNowP`0 z3}ey1kgM?NTcxY@M75r%wj{~ZiJ47XOIC}uCJ@gS?s9;`WirxgfFogVbfN|9=uN-4 z$*QoTuTWJl=K$~OFjXw{9ez_=1`st}V_=DLS`6RnTvzUcFx9lzhoT~Do!iKcFY;BD zLG&c!)r*n+b-4HUbge2=8<^aX(MoWWK0P)SAxcF4Zd$6ga_|B>9-kK&)R2KwL;R)9 zCIAjx_~U%`OMqY)n`#Gk!t9r$=@3Ks|B?*#m~k&Ab$daU$iKiO2NKKX7#Cg$#;yY6 z_L}BvvrBl;`o9d=Lo;I*OS5*)FLYe~aw0tw15;fSbw`X;=4PYpYtkklnujeR(Lr{# z<kh(8lpTJkiGX`Vh8@lLU<xBw+vE#kN}IP42L=VTq3t9YZzY^E><GhYaeE1W$(>=% z<8?1;q)!GZS}j9YUl>|ts%N+xR%u|@um7Px9Fi4s-p{+12awvLHJneY>NGFst;jwx z$0eqdceH)+>*5JSuIyqNa}}&!c4akaH}bk2Q>K&?RNp+(v!i!h&~rrX(S$;XZW(s$ z)MBlKHVQZBhpT76XVxvm4%s}&YiG{~P%r_{iqq6FVnVE}iR5I;g7EXuTXa$LY*q)6 z`x4_7vMX@UOo9T)`$Ozj5C^u+B!5`FbtV~VorVUxjtFhc1(^Jg>fLO-vAT#fq&qON zXkcl_ZKERUS&w0%DCD){SBpm-x#uGH*pL|QXwi-~U#tP^Ox0$R>)j_Kv4NKLq|usr z8q-{MuaV)o4?ft(h*pwz&q0RO`e}|?a(lcwg6gPLV^Hn;^w}o7rIWV94%G5~^{zN% zS|f3=nblDJW4MKsIz@O!;{FQtw~e5SnDA9_B>rq{zAi>?2bjLds)KXs_qE<0*#{-- zGJXd7IDL@Ce=hf8`yEyS?VuEyv4;Dp_((I3Q(Gls)F!4wZR}qb;~AyrQX(*wH&SDf zSLJx6qv=3g?M&feg+?vlLhP6x1hSXezaA>-I?w1kB{9!Nln@NLV`eNJO@j7wWAA#J zL;E?z{}%ueMwNkLDJzya<XGuwz~C8;0;i3ei-E6od&P1FQQoLQ0TSY+&1R}MJF@Q> zcC(tE8M|8@B%xzOe4PTJW6u|$C0^;y$TD5YdNwu`Pwytds;Z&Gj&JdZ7dlbm6vB(# zsy$s;V!z>xkDm?MUIoNUVKg?59})5cshS9a)kQ8BThD&_dY|abFCB>%^#}RleEJ8i zklT5<V`z)_L|zZ*qTK)Q(!{9tqA~Iqh<M^0O)o^&wj#iz3~Cz7eDrhzi>x3rKblW3 zVEyS8Q=QLd)A>SQ52%A`=wuJ;OFILqKB@n}X&TP67+GG!^G*e1m+)A1N|B@vi&i8s zY5C(vMOXV|C{fP-=vuovPoP7J@J|B)%eFc^w$?<*ql7N60M|NoF!@?_T>tq^2t5*L zBaCNdx?!a4NW|30SzIHn?i9|+K(4~DkA^@@q^VRn8*w)!kk>>l2z<(|1|?0x#+c6H z+q<_~v2WjqkzaLY%^{4hWv2sci1r36FGia_Nk~Q>IS3}Z)5)wqJO|1Ff{b7_@EVZ& z99fOt#GTL5(^k8{t$R7!Lu3W11LJ$gTAfk-lD~Fx3S3O;iE5+CU3<z8Q)kA|V<Qd3 zi-{Q8u>uM9CFCR{>QI0dJcbuGL>-1<q;Tc4`E*!}&gW`-E!FW+m6#OO!bP2`BT<2l zkWho0;yT5u(9hJXPWAS@L|+26^DMK^Kq;m8F<O<jQM0fzUsv`kB`bD1BbQ*}ZLu1- z?H*qRVVg~gEqUJL`#;#VemyX8Dy=Kz?oNl7_$DmTb0y$5Rpjwtr*ILoZjC3ndAL`w z=PnUBzG?gNxI)w3^CtqwxX#gKdu(g-y2Wo;{U>Iz|9vzQQM$&A-gq=?|0DM?3BPV) zZ&)MAGRVH>fD^`8hel+cIp#qYHyKqVyEs1{R%I_rJ-)pF05SFe=I@^QdzZc6<=1rn zDExDMio6^87)Y>J@m5&VPf%kF#4m24N($papnvu^Y8_krI@kC*A13p-yx(xBKbhyV z({k9=CKL9s&));Ta;kwTrWp$pt{$mvm12Lv`J#fs(^*Yt#^^Zb*IYHTSWr?eNaH*i zQtGVp@I)frfJ|@Fu0VkomU)_bW7+b|>Bp0tcS59ko-?-Shc8zR*_B9uXWF{jC6sWq z;vTrnzT0O7`j*M38cP*4?`!7}jB_@h!u6h7zM~a028!68{a}7p-#@$a=gpZfN6<|6 z41;ynDX#5cbl)eBG=~swS1aVC<w}NU3k@;;aY{?Rk>i23CBwvayUqwO_Gp~p1*!oN zHJ$YzBRiQ&R~OR^ghry8wI<XZPmS#dklU=_2FCnGbUQq>J^LLK)|}D4GoCV1nCY^U z*UXzJ6wfHC8I<T!0xNwq(LelQ*&;@-a47=9&CEn&CSOR@thxkw$K?X|G#-?LL*B?* zk5_hg+Qv-JX~&cK4WjFY4bI#Jry(SGl%BnWC17NMr8SB0)WttdpXSqp0`_(w4)z`I zB(IJaU`wZoa7}pDfQxckVlDe%)^gBI&mVJV75_sWKa9#_n{j!Yae13@xlzVt;hqOe z89DtUV?qs!m<ggoU|18N)^YisRz8-HDobzYN2PHkqHZzvX@pT|5eq(CAzV?FHxk+k zL)Q~K0#xj{J@4ype9?N-5&6QoVYwy7v+)~ZL9K#M1H*DX^*r0J?;+Ux*iFj5rE1|1 zvZVHF!d0lKvoz&bdjZA|pm>KW-O!igE-Zz;Pm7})5cSlLwMNL!XJtnRu>(Vq@dqG# zUGTP$jBv2Vtg)#<s(1VYJG|-S=^QSX-{q6VlQHso=uI#-A4xp?Ax_vW9S;I>U?|28 zmgrrE%Lp>OW8XvAr9i62k>QceB5k|bZ8y8^X4ibPBOiGrW}VdKpp`Gn>MNZY6aL;W zhi|{ikBYH*0T$COa^!~k57eulmBq!!5Xq*1U1s#i*nGkJw!J7@PzoVPgro%HF;r~D zvxLFO=89*SHzp<Qq!e-Qiy<-{WJ7}axjuEPbA_D;iaJ*M_PYg8Fh1Q0`hrwamUsQo zeGxf#n`VROXAskSc-kzbp)0Kw_#TY58Wo&+elc(F1i4(M6BlU@bGj1X=>&o3RIV<v z;wD(8w89+EFzXuA9OYnAwi-`!X-x`eUGTPnIe_b4$_WyMVX7w-a?<d@{lKAGcnW#; z?>)cJ3>;R}fA@m=JPzf9@wgqDCrrm97=uW@_9i!P{HENO;5sJKuF2KLs)cqdAK7sM z8MIr~>GYztDYEF6XgVLgURiu$*k1N3GC97DgaLuac~-THzsiZ)h1IJ?aTcI4w={s) zFnqdTSP$gqw%;&RH^QF>iX_0G!BKf88iBvM>ZK1D_*}jMBRMxF?2-M@R<%O9I65!K zYJt4Ke1gRAJVlF+)uGSg`=^T0pDJn^xWV}9Q`Z^yK5Q9EJ*3t(kv(v4_$0b6N+3Gn zBxCs<4iv~zyx0lXjU!mfC&g+*a0nv$SiO0140!OFoWz6cc*mdZPR0f(Xt3k(ihQXS zg_DwmEwdKdnhB9Y4*%@hLTDFD_FK-;BF1fSRb$8nl<V4@cXWKjb-Z3g*z_&Osny>h zv&NUV?BApw`dJAU+E-%S051~>ciHZwZP<ekqAEN*@O#xZ!NH#@2^UDs^~P>Ys;(P! zf<pLs9Kx@WP<-QibeadDSN((6*Z>X%TDAG3`c5Y)M<3OR5;7~0GLLkED}EV@@|DqR zdK^iiN1w?9do+CgRWULAWkIgz`fm{zK$o9GY=UR`;<S6F&Uv?YJM<$zs@fLXjv}Mo zZf^(UVn3K4m8#Z!r67#Yq0a&b;;43p|3woqI<T3WeQi1>YT5%F=4}**HQ8%@`i&ju zybW$Sa6|X_=4z1LbUVbYtnYo--q#if-e==~Z9%Y2RK`a^-NDU!glqqc4up6w>njwS z5<dTnE*`sbC>c;hi|(b)N#2CJtgLQn*yOB{@-FEsD{o$U3Uy!jv>BA!2eI2`6KJ|M z8^><-O*n!<or*0Mtjtrd3>qn31OteHP2Qdi*6Y!B{fly6bs`dDMR3Jpcw^$$uVV;v zalHlsuZs{u=vK03IydFojAiTd#=wt`ZfeUw>+m>hq#>#%6i&fa0qrfF>k%AQ&x#c= z9b&p8k9%<OU!DgdVqU;}y)x+9q4=N<0PhbWGPaZn!9h45$n1w?gAA!W9T0L5)IqP$ zqfvm@R_>X~vaYyx^$xgr3986wyeQ95&4C<6J4cy}w~)guU#JN}%?Fssz~!%<AOQ|0 z%aq89vvN_9Wl(+wSE*<}Y_&v?Q;iCvQyZEO%K#Cez_3nd=nA7|P5nmGvUBiWQ@icC z+I?pE$aCdl2d1g<wbR1{LC%Jq2Ze4BUS3TpH2-aN3I^rniQ34D0jx=hS#z%J8Vo(4 zz+92Vt@RzbI0(lUIM;{(<8;F){m}e4R70)<_ou(a<#7=I*+>b{D2JWDr*(BLk-u(7 z8ux6d8t|0A`nBr^liC-~kxgP;l-4bDk4vVfs{8{S+^6ox)K%HPJTK;}foYWKd3H?5 zI?jpW>2e~xRpg?mVOb_~vf@(gI|g)lYG{FTEJGl%y8(}6*X~7i8@oveo?u}ROK5+J zN(bAe^^7lCE5-2`!5iv}hHPA``S3{25DZq;^uox+(nqT!6&r2)1uwNse)8=##rB$F zdrh&~HN|$RY?sP*sWiUPd!+FD6PL|Li4u2F=kr;1M_<Xm)H_r5t}HHo|MwENcCcM} zcXsZ+F1#f&Wg-g4i0F8x4{xGYc&QS-z+q{ugEXzO{wed<=fM@=(aP{E<a4z`io2lm za?Z-u59}An`i|FiwgqufXC)Z@K6R->da@VupW?~A9gf_Ss9J9pr}ODzF)l`r^|9s2 zQi*e;P{d=iK~f9|HB|VOYOCc7YpvsQi65laokn5$963!S$wN<z#NdXLBPOOc^Dg!? zxTNEB!-51Pj!<~UwLYWSfjU|q!WkQCcCw-3tK-5QeshtQ4i-D`gi&#%wy&X89-DJm zYT`I5eHDxLcNgWcP^iBdH3q7FyNiIx`PW-pbS?H><jP+Y-X*DqB2)vN)2Bw41n5d3 zH--@Ua9<~;)3^T^5hikrYIDL@#bX87xRoEo;!7OLbt!SW)+HZJ?frIQ%?CWMW;`%A zvUb!qe%5qYrh?8bhqno{=|x+%p_A2FTe5y6uHubk<({kD!^(kCH4Iv3sq;3qC_hfc zHw=fRHfjhjc#m&J!BsY}kk{jKRE!+m{#-F{HIlIHI7Vf;n7ah@qkB}G=I=_y2EaX_ zHBq~8%Plr<A;<P1uX_Un*Z)wjB?`A_k4U4eutDq)C2_o_pt0msdJ0_6h4oRtJVjo# zeF!!aTLHEM4Q(Sh9h`9$GLeC&8@814fi$v?J|cDohA8g&WWu`W3IXMOdww=!l#|#y zx@NO^uGsV=#hBY?vF-k`F4fuz_s0Ze|2AKA4#_$sUVX0)hN9k1skcZBYleZ57N_|_ z2MewM=O$h-)WFMFb$S@S9j_vxPX{r60Z-$#lW=%b{r+bP>p;=xaC);>`yCYAp`Y*# z;y1zRfR~7L6MJO752CYc?GVr|Ap~(LX-dEHTr9;pf<u4}r1nCl?=uSo%E$#`G?6jV zfm55=lOmt@O;f521+vt?jc!bxnHASB<}c?31}drGQqKqx{3WfLt$tvuAK2;#8tDgK zL-+-)q&7mjXP;YPk#2b89~Cn=^iPKB^B_BLN`XUL<Q<wIJ(`ZE^9Ms%qIA)Z_X%XP zH-v8vrs51wCu*khF<ftY2N2|<zFOXJrfPaV9~L%{nA(Q%g=fgAcG)MBp#tzgZD^%h z!-%ssRd%QB;BJIY*l<bnIb2xH=lLqAgse4sB*aYd>fi^%!=w)o#Trt<sM)B1u)Hg( zV${+Ro94x^DBrp3Zhp)>KT-rA96NzvCnuLWs{d<61%(2SKUzR*EhVk6Uu2)ta$wA5 z5RS3?>E!(EsF;6tnZ0s7_RgegQ0n7T`u1@3z%?vv&)n-^?wB<x&R)cblov+~x{1b% zaNlUkAXmrFV!9Gz+6jsU)>oYk!*y6K7xtNwb%ef6@PLTQ(T#hUs2x3?=q^%hWPska zm@JlU*IbRi?PZi$HQ-R~Lb2D#(Nsu4@5hSV#)E9P<98Azn!o#-fAgDP{nfAk=fC~+ zU;I~AX^{0J<!z;3{W+`TO#qu*Pn~FriQ6*h8HX_j2ZTU|1B2*54!RT-4e`IQDWSW& zcUi06*Pd+EaIK(aZCBRwghH2ix_jrY_swwEsgej=vb*N*;%s)BS7jA67ax8S1qe5V z5hXjFz&=#MiS6$0E>BE;Hl4z-5L}1C$w6WTdZGR$=D?v@Vf9H~VI&%4_4(Xn<O~e} z)^Oy?ZBX1g6{MguwdKR+r{oH*Py&N*f&yDV@(TB6m`^N>qry2If{4I@7=sa2v#TUU z7SFgmwg3_pKyYZOe{Z)`>4oj47DE-+nvX09*E2N=<N+9#`Iq=-fq404ridUrRe~VK z=~LprBB`#0@Jvam<WMe^wGOgQ&;k;j8Zyb3YQzdCv=VkjNwwBT&%H!vAocSO`2DDV z`}*HcT8_VBsDGR|-S@}Ev6}vdSyDv*!C92cCe3f%dA0g`chlvMM9U3(%LfRZ4To)m z=YN@3r_XZd{GnCaiTM)BZIBDpst9&Wm^t;HJY_MYf+B&1dIMVR{poXDn6@_CKzIki zaMP6O2#{8s2~8#vO^xWcZ!K{{zwfF;+BjF@7?2E5PHSuUusyA9Pivds=01M%`GcQ+ zwg2YzvllP+zl8it-P`tZ#9ZXUm96^p2)q+h=eqDf=lfH7fxGKnA>Y#t@Gb3JuKq&j zmP=Vh?O*Q{N8~S=JET@{D@GLeobyWhzH_$XRa`is@+~;vrF=DA8%>}eO{e1`pVR{@ z>8K0d5%pJ!ja@U@=GVkI(`+TusnNcPpS!#bHhTP;jz^E#Db1jb7h#7!4sqStj|(LB zEa%>O#R>9j#nB2h=6+kec5goJb9_mnK>=)39v`>$P<$(S^mZHhnZva4e4GC9Kk6Wh z^iSX#uq=j(b$Nttr&fnge4IRmVjyf1J5<>xV|}2tliA65^-jhG4ueSls7b?kGS=B! z06anTfiXNMZ;`yjS#1eHqj~MvbUC2Br;0qjph%QzxB@`r>XM=!TA>UaIVgNeJ2s#u z*g+bp{C-F{(Cb)luEFt;(Qb6wUe2epO3&GM6-8D1&cuGCe=je6=R=8yR9vg;rU<<- zzE>Ks3&AWye`-Y6)Pd;pa$YSowJ;5WGd5!EJ7GV-O2iX5AS2YCPL?#kP^=!@pD2K5 zc*r;OD9$>alxaSH@K$X+K7z>dCeV91c{hCvC*ZTQVpOW;$14~lNIeC)e>f6}Lsd&) zUo98#kZ#b2*j!j=uT&_?%h(4mvRit>Ax(s7!C-sMWw+kz*uHhi<RmpuaH7cZ@=3|2 zubOYf5!0MnDiPiBGL(4RwHLl1+3B#ii@bVEQTsibN@?3S5Cwj>_j#Mj)aqMd3M@Lh z)j~;!d+h6{{HsDF`<6&~8rhV+iDSy;W;+jn?6Mh+GnX1(J!-<JB>}}<fP|C!8xlxA zpUz)$tzvXl_`c@NL{YEs8WLosYXnkOYBO5pooPm{yhA3ZAa7{OGjX!QOi7#F(^b3d zWmtoBx2~}MeLvIZ#(_*1qH9vtDc+hBLFj-DNA{5-nkniVDmV)lM#*}7-0`82V|a0V zTnv@$r4GKRJ%M=dpc3f%#oPO<SqZ72SE4LhL_!;=y`U7s*^hU3cW<bb@U|HEV5nP< zRfh)IC%eGZaPbYY-k;n#tAb9JlkC;AhcB~!_s$nzzDFmzaaR7glf8KTb=KSM{%JNm zorCA$-QDhPxBtl}y>4%}pS3@^bMvTNm{>d-9_m1C`$i(M>&{Zdj=unKjn%MnuNdB6 zvHdo@o?;c?NgN6qavlck(tCI55s}Ass6|Fo9Zj4&5$dyw=SoerqMV~*aZwaLNun}a z5)Y!}&70ZyrC2JAD~gyF;r1?KPc%Dx`&Mo-t6R&_oc&(S90wibCU)W7NlaW!2#3QX zVaMa^+AYiCCOs@H0}qIUq4;*ZGEc@@TZ3O29?Zc*ZcR%acX1g8aG-dM@xTo(yh0*! z0}K8bZe@<(TwI`th5Hx-?`T`3*6DO!z6YJw7;LRt*mz5^&la`PvgS{K0~dCG1S<7i zd&{6CJUWh1xlXngL)S9mkO-u;7WF37@yAcd>sW;#Fmu|r>8oN%<Jp~4DsjA?D!4>| z1#2rDlz!Y3j9j7@HO?gdahA{QwW7VH;J5h<3ZCuC>dDz`vHBT6Ya`5#Ab+^+vw+ac zIJ7gR(d(?oRvJL}Rmu{m;TkY8!AndO0SqSUiab3!QCFOF#s{>Zn@ZG@P(knuXO?q? zy?)<UJNGVI#PEk>qo5h%XZg5VhK++4HtAK=s{p(8#|Vx@D{TdrkMyPGa>C&_b#WH_ zy=_5A1;5u%hE8UU;I&Gw%+AQFwPRw!>zHbhJAU0fi^oeZ23~*kECjgnHhthWwI>aZ zJ=;->KrGX>@5Kc(PVZ@6!SqJL{tJL+dowHn>JWC^`j2C5FM?<VNJ93|>R%J90!m1x zkG7)Rgo?3&tO{3ptW=lm<BmZ~Bjf-jsCedc6dU6R<RnG*c!j3MN#Vahxb<#2$SCg? zVH$1oy0T3QSLdOP!0~0&BHO*4SiiL8%etP;?0pry@w#DPtnG&`*m9`lRHO+R))?Rz z^AmaNl^1mP(zl#w7Zd2Ux>LyXK#20t`BeeOId(SqNCU7h;Q?O;qtx_-SifAGfHE!U z5MxEDO$fY2bg#@yTbp-{@e?f{b4gyFYN@GkpSI;<+qxdN-OE|Kw)nSH{9BR&u=uxO zaf3Cc@oh61;M>DaR?2}#OT@q?v0$D{@#Zp$B@>cc=kbI?g#P;Z)32XB!M}d}@BiCx z{_?N?{y+VnzyDwV@xI7|DFqD*7y%JCd@Bm`o+Q$tVzejI`HESM-3Q?(yGMM&y&eFn z?R%ge2-o`@UhYubHwhV~lGfHPV{4bOwaeJVE`y9O^fu%S-9PT3(VUYiI@aEr?OHB$ zw%iT5gDY)Q`)TKUwYxUIeUQQu?8rh`ic6EWSj$)N2h!GBanu7qN$gP)frALM^=mU3 z!@pG+qbpn}kl8x&NCwPLjI_Qi@6!Pv5dBSe0e>Wl>8;T}mU<?~u|tAaBtXzw0QJ>F z`MUX0A%gfTUFXy*>t$iRGM?o#S_6D)X#EUAQPYp0j2Ib6)OlqL+6Vp|vBP+16rz@z zmO(Y~bch%eb{j@>*`De9H2@OH-`LPAf%zHKxg>a~#CrgP)B^@1mRB$4&S3Z!h|y~f zV15@$EEW^xV<)k8gp10<bI@G|-yB~-*mpx`B^q2K^NC(_;OH2vMgn%s*!`jXRZK?7 zduMaPHv}`Vq+<h^8Nbb~Xtov2wxU@h(ag9G>s2g8tgXB7b~SBR(}o7<uk*><Vnup! zwYfI;Ag-{YJGIa~8w5WkV+8jBoDmUX<$dAzvfwAIXa!S|lYFU=e?C;3Xn|2>>7ANO zZD~QhFxB1lF8R&6cul%Ej-TRC17#y&sMUR3o@82)s-vYDIA;G&+Lf#RO(C~<rtQ6$ z`xW=4y`ulK5{Tbc&fcRKov7)dz{c<!KYK#?L;+$|5MQnae)UfhFf*(Q{f-r)=`-dw z^3``zs*Y(832^dK5am84rc$wk@~}Hn%;uEKfBvV>6zr&+%KCQTRXWtSStO|1k7U0U z&5fhQH2yP{4;T9X=FQ_LKYQ~gDL+G8l?&{`llgo)R|hq3RG5hZ_ND23MFB?0*$=Wa z<cb~@m64NXtAXTuDI`<gPofmSDv(js%8k~cGhtP~z*YXas`*6GB7x`-0E0k^G=5Yl z3V@5pAeXgXQV6>1xe%}(+C}0ZVRNLOhn4U+cIGlgx)%!mvtn^N9kpBhXwu>buQSr2 zl%x1zONRqGS8M#Nd|!M4DG}!2q{sszE>3GtN;9k>!q~G{iQ|u1$d8d8?r|XQh;2?> zI*$<yf(WPvs9P5vRK8$r%c%tI=6=-yQyiwj(4q#vJjRT_=t5I<Gx7iM=NCp2PA#r6 znp;?54=GzJPHqjXk){)Xi9EiC8XqEzYodzL?+iuQAI2cYX;c@$nqJzpClO0Joeb5K zi5fuzF`3C^`{;oM!qH$VbP?{?puXe{$W%VE@Hau>p3OiT6^~x&t1TtPoRATOCnSZE zP@M#W#CEv`loMjGc+xh`T)T+?9$kc8J0`X2Xf0s#>VR+^BO}?&6Cf4rv3eg4?1^S7 zO~+;ysbNO7<4c(u(ur?>Ivp>{S%Os0&f()&8?2CSP~0VAW4G*kqt+WyAeaKKB6U1I zj3S}rU2F~xOFD3flnEpt7+)(MW0F#S$apPeKhOJb`@0u{nCE!6ODXoWD$NrIOwrTD z8E__7qpJf0-N;(m&70~!H~2@&RqVlm(dkz3Ml{OQv7sAk<X{V^fbt6U0-;GHfS&kY z?mzpg4X`5I3LcamLv-W3MbQmk7_!+W0sPKv5>fLv1Hu^60VuIK#0H0Ll5t+c3C5}G zLY)ZS(+Fp?i|yU8sK0KU>%#n?OL=24IHC?SDb{35*g46i7`o!nK0(2nM@dE^oCMhZ z;@K<eP3L&3WD}U-2cvIub!Z>1ta(|0&6+A%OAQftT-B=EJry5$p;?ty0C@AhV;QD< znP6N|MkN?W{X~d1<V3`Hiodw6b%egz_<`0KfcV)a>rD39)vqOCsuU1llR!}H-2n6G zxfhWLPM`{Di9)2^@$@&Ec0eg<a?2S#-^J;4TyT{Nz$lItgx1|Z;(?(S(^>FtPPWjk z{;qg^G+iih-If-8Aml?(Hg#5RK}ZvDJYhC9Q$=oIjLI#*V*?p2%@ANt+-Lapb28%Y z@5U;cHc@SeelEt_VR11f<&1>(MXpBvCq5{hZ%O-X5^ch;Frn5ID2;90kZS7$w4T<; zwgxHXgg61*c%3io-<Ctr0Uq_jl-(<@*w-Bv&opE;CaP2fFQl*lq?U_fPHIWizG3D! z9HdV?67yQIXe$r0+kKBV)8WG+`4nAl#1;~oA0;2Jf7Qo7+}+*vza6XDd94nNgRIwQ zD^+#}l{$m$;IMu^U^!6KKI(>~jtArN1iJXGI-{1yD@m)?^dT7aky^{e{O9oiX7urq z8Q<;Ow_C|UB!i<_^xW><`Q+{=K89G3G(TPE@O=KRnBZOLZuf4g<tv9ti{wJ|I*=Ze z*Ymp`oK#<v@zLU?9-q&3=--~AEzQQE<mLA2YI}9Hy}H`)>I&GTHUwezdCs!Uv&zF0 z2ekZL*27wc>p-Pi2(ZAzMukyew9&6*y`a$~1exFNsWX20xYvfBTC0_pTw!Tqhw32q z#>C0m)*AylXEi45bxcSF=@oLEE5zYed=_im{&)p5D11Utzi8TXUOe30$Qk%F7^c+@ z)0xS&YVM4@>?Rs?)Sjrv1$y)PNB6{F_87CqO={eQ1SK;_T?%O@-A@-5aNJ$YbI7_| zFs<_Nv>eOuB9!_=y`>bFm?Hv1VTFekDh5u8Xw6q|xWwHg!)bgvNzI~|w@oLd!F>Pj zVaFbVU!Tur)A_=RAQ=Qrv2H9~2--QueT}Bv{%%_wvm15rOv!YL_YrxK>JEn<qRSoz z>s|HaQEl0$*M0eVIMd~aGrxxV)Hk2JT7^z~&EMHWkqPK(<rYjEPT{@T^en*b3wj@% z?g={BgrY|d{bcx7Rl9J6*`Kzl#ImPBbys2mLcycreW~PB5EoT>U#y2ZfP#+wd*utd zwosoH^2Jfz4_jo#Bzj-PKqUxW+PT6UF+)03BLHshhT6|=Xau>od@aJdam!%^03+j* z*oxKg(Z;Y6mS6Q`;W`cHdW<+k*oHm~BODQ~?{ygAh*)2P)2l7r2qQ3#;dojVA|#Sq zYL<=E2OrRBG}=yTh-6~Zn@~eY+6tH_5+Q!5QwGAdib)qpMT`lMXZs@)riq}CsQ``m z&n*IUi>m|;=4_5MybY|e97;9l#cZ_jP<vrD7msxi6kmg}K`_QJc>xYbN9_cZdnPr; zxWcoa*NVrWFpHpb5{cvJ46i(p?gOi#wmHzsQ9uF-2w%a)q^(Dx3L>@r2n8tS;yYQN z-bD<R9<LAgV~U%*wnTA#SLIbQtU7eApf8>eip*LmL-Dy%Cy_WG{?~l5P%F#mn3V8w zz|aAA(C*jj(2Hz(OG#N1P;;q6W(KTJ9rcj+;c*Il2@n~)Nid~>4``IbGIXh*QC&Cc zV6+_+6I6moZCI{rb>7Cpz`lqU=gu?|^XyBqAyU=ryT^!*J#b9I9<B38;sej|GC1L7 zWRz3U+LIn>^OKeqi93Ow=&P-zPh62opwa!hr*8y$Jd6q27(_b<b-qc~`6fP~emJ0h zg8}t}0reXUsINQM-(Wz=I`IKPf8>zr2ea%&Uan#<m;G4D*h`QqqOYTHL28MY?hCd6 zk6vF)w$60g#<P6c?seGkyQ9E=h1a%>k^Tr;!UJ_9ITxNKkX>67t&29HU;IdKDFeHH zy2~99o?K_E)m{j7WuboKL;VJz>YavxT>)78YZL>z?AxyP{jT<TSNn;sE}~rpsG45A zp7upg4b&JAl+W*lBuU>88!H%hyO-T`j10QBf=29{IB4i$KxrCu-+*@PQ3Lhbs=tp< z8O2_WJls6`n+0P0YQ(XQ=9Xq6n^H%L7}^_ALlYXa<Pjx-<$4m(6jVbVJ_#(>lYj<| z6y#G^kd@WQ2MeJH5%}*7)c+#KFHmtrPpY8lSjG;$p!tZNhDmXPXTLa_23g`%OnMbO zXU4IBHTxbGLL*j(2N_1j?r~3z?ztT#iUI`{sM7N>-NAJzW+T}{(LFJGQG@0k?V#98 zPxC?1_}aQRHZo0=#ba*H7B$lvgx}e)c!(P*@JJWiltJs!yvX0i$PBwcgStu=`XAPX zfST3(eF-{!nka0!<4dFFl-~86vAqeFfL#g%1X^PR9R$RZ;&~^t0PHA&!^Rixuz_&_ z?IA<nzq>;QT%52&1{8~j$Tf19ZweC;cS>Wv&5Ey)CbepL16eXrBQ8neJ+tYOsAdit zP#|3uNu#d5BMl2E1f?T2#$qs9GK@wVG|zINvE}V%Y%?3{tt^rJb~Vu0^R_a+lW7pX zFTQzn*u$~e&lAN=^g~l_6&CaF@(}@ammkA1i*Mnry0tvoUfG!-3z96|^rhPfU{<gO zj&vxjh#nKeKH_jXtnCh?#g0#05qu;l0V{r9D3%WjEw09@DgW58k^zn1v6grxZG^V3 zPA9rXIZ6~QE1+m985(WC6A1+`-w?j+RSz)yR>w}1$Lgcg!5BCI<7ltr%TPA_Zheo! z^brzwS-d_DhwDJlSTW7KZF|6#WrIuyG_DIN5#zlk;#Mr0gU9XQnCP)oYm64j)D90^ zlFrLx;A+fr6zn=zNk1E>N$n=|wplb*x_MBqrxo~M-W@$rcm-EjFeZrGf}?W2SOpuD zdGcA=9qG*J4{)3@?ri~*N*%BV`>StHC9pwSw2=%EnOtY^jN~#lGaSiFS_6c(*MY?5 ztt=MDHmLT^K;rgRpgZPIJTXM%9Yj~30M3fZND$PAk&=o0*02kf<kM7$Zf5feqZ_qA zaqO!$1L;!SG_M`nn!6!BfS?&`!mwd0-n~(68{Ag(%X~5#7ju1mK0Q8eD~zZ^3~*k> zzT>6+Rv--Lq!HBf1p$o~51DX;rG<vBY2Kz*d4th3qg9M7xt?Hald|yY!*P+%A*ste z$+Y0vjB&xEnO6i)HGBbkw>c|=hp{uDU-K(nDaugrLu8SMGy#v8+gh6X?M$wjv7b_x z+HfPe;h+;`JcQW8=Z8rh0EXWptdTnYfpZi)+AT-<o#ZHn0=Hb{x5QOg1~&BCRf{eR zh-fV6Ba7uE%n26e(fGL6rg940L$IGOCnuEnVzaDkCrz@PUHS<C+z*irq|W$ZU7{u* zWMn-$5dLIxSW)|xVtgE0t|S|BA2t0xE|gG-WpI>D(#b)S@%mnpG<ceE70pTHno9E$ z&Oe|pXPSRR^hNeZC%e158^O0jQgYgW>=Q@}EeZ`dKw_$rhokg_cnA%=L8IPXADWW3 zp{<XxjH0-bEP?=sWeiMAUd@<Mim%?N3smV<6FUNHM!XML80(m`GWS45Ow-{JS?Sr7 zm=0<GRMNvY@k(VuMr<V>Q$%#jH)7gE*)JUephze5%IIXfU18!Y*3f1qOtL{J(g{5> z;tk5_N)E>w#?xqG4kIzmKVLNAZ9`1zizw725U{?mj+${9QDIH@v};Q8Ei8&c-k`iR z%05lR)G|*%sQigY0Rc{F!eag~DcK^ZP8WL3CI-9J#B?E!eB&^?7WcZ~KMB<Leh}0< zx8rrVFa8l?o?)AkVIZ?o`^g7SiG<#70{xoSniqmD$=$DjwhzL}l?NTjO))|910|=m zhT`Ar#{EO?+0nL~L&B|F*@MxDr7zXK(G*~!E`y?KlJ58F@+W>s%ERnt*4y>Y6E^Q* z?4!OD_g_ac0A@F`{%%Nd20ho2hA`1YG}9n`i{={KdaiFh*SDVQ8-*~Jii^Z8aPf}( zGvWP&y}xw}eN!=($aFGZA>p(rYhO!I(nTkm;*cr8kishlg46!3YI3w2mUeEXiUSKl zt!ge(>Fp_uR%~*+z*g``k*He96U95v6bbJ0V9_nbPtK3_rSn+!FVBm);*=V?Xfc+V z5xJW1DG@Or+krZulO33Mbc1a-S{;P(&~7y7z^k{KEi2oyvMnoX8ia#Zv-P}TvS3oe z4S5H7em-0QhiE$nk>&@Xg6!woc`>*<upa{-K^=rERY`q{WU3P0mS}5MT!6&*@f|7P z>-&;8>k>eOo2G26fk)<A&&w;HD0obHlKAmgM9=6-;4c9Mmb+R3z^O6(RYjr0!=@CQ zP`hF=daj$K-zL!bn)8enO5>sTP^GY&)(2``v58G~Pj*E|t;eJCZgyY&FUZhql8n0; zWf8*J&?D7gALD?fNEAPM6{#!a+i9?;l_Yj4bPVlxv=Nskp$@}CNB6knx*@kqy<37K zHpTJ<7NJeQYr_Fj4V|gM_Y22sms{EuoR6APIFO63PUN6~6R&XDAd+yOfY%)GxCbwv zUc&0kAR*H`2ziM+-OHyKB3S*%X(S^E<2F^i5rr%zE{qAh7kv0VIf~yJs{Q^JzgoP6 z1laE}_AuIVtDQtiaTghd*TY^LeB6>Te(G3?7vIOyIv!f%%m%MKmsHHe^Xw!JhH$f{ z$|I4qeUm_QLe+)7EGYI#aC+R?^CQlDpEPMk!WhlNd66p!Rk|s!g(Rgjpcoh52WbFY zcq0hA>_No9(*mL^cF7mkBPvhY4&-Kvpsbe$$S&3S*%@r(P$I9Y>9B-AwAjN|MRnFK zIu{fCU&Bx@HQ%Qu+$xLFYl^?yBZAY_*0XUNsu-Nr)R_Eh4Sk1&jrcn`*6Z_nRwZ9o z4NAVsFZ{r*l)@K7WbMWp;;y&U<Bsk@M}XwbA?(WTOy6Po0ad${8SyS_X<&KN@>b!m z)3tN_B^(Sd3KYP9lAkKUb9|zRYjJvpJg8UzHyW;nV_?h01d5taOjzHB9mNzm*fNQK zCeh8(viI&a6P8QgLs3I7nP^8$*^kZ|sdC|p)3!q|bUh`x=YkinA?j=3C5Fn?q5Y@& zNO_j4qnyG%%JR?)ysm{aLorqpq=fkr;-#`T?%?k@f%>|u5Be7+oH?*Uugevia29`} zWwZm6V*p$Ub@n1lK#EtCYA1tZ4+CIcJ($lW05cMk1<6tkp6E<9=#M&PDWXwDcNyIG z#eDiy7ZP2=by+MI`{?cWoUXpbSGbekWxwB=7xXHYe+VZnnSO{N>7axpLB><|l0jt9 zCQK|0p)rgRG%OA|esmiy=en%|md9STd$9yZ0&Csbd3D-0&|_B<4o8Yr**#2k>1NLn z!al8i*g<G<p5xz(>1;Zlo?t}dYC2!gJCK8h=i{81k<&%ln96N679Ku;MO)s1!>S1< z;Foh@1}xB6RA8g9x;O=;LR;d4@G6kzh+(wU6F7n^zgR{57RgM-OfW>X16AD5iek)8 z%x-czt>|pk9XZd7$2t{==NuIEj>NRAX4){IUXU7#z5Ps{b1ZE!C|(6Y8Szv(l5||< z`paE^!^0{AF`)arCxdKx8P5zGC6ln9_roSv1IMQg@UY?_thkI+6o)K9TfkkO8u~In zh)r1e#nPDm9-%Wc=h|{dR6o$wVp(_$n_fPlip1>j&5L8Yl8e+5OwZIl%t8V(O)NM= zb{D0I((ug~P|?CQ9ddAD-2!sGd6P3k3rRCYpkTQ{#6Z_x>wDFakJfjqQEIgtH@F3w zzr)E!xoDW&?JmWh#&#e!dFg#fVbb5(?m@IuA1gstIy$Gp)}3dMAAHfF+==t@q@3t9 zhuLXfkqG%wE57lFt;g^V!Ad^M@aj~pTk(28ugw*wHq|062+X|wfuTS#hPoJ0K7xqp z&vErG@GcyS8}ZH7cY>Pf;Evq&t!UAh|FpFp?xtfNJJ~G@KgN05B%3B`Se8?%CZp2m z+huTYv_foIS**&_)^%)BJic&)9ZB}=1Lnw#K@tYe4Moup&XlmcUSsqMe|eb_v`UQL zF{b2%cV8MMD`${e4q*^tQixr`o>&@Ii|5AV-B7hzHMOcE+{)k2s<+SW00wL@i49kD z_gm`jw`N20+0M6#{+d4X-OR9}Z&S(Sa8|{9iG7Uhdg^&Qj$eX6{?6(*x8+K+eRw>& z5si9n)Ds~*H*UCjksQ_iXsk;yxcn&K0a3gqWHqbg*mVm~vnruv=j64Mpef(~gT1q6 zfZlpGnJ+|_!SUiXBprQ)%14cg2n!bqln070i32L=s6n<mfw_aryu7MZ!|F(@K$U0b zXI#&cPeup@o+DLsOeM>S)_7M72(oj2c2#vqH>wZm3P47=VMeOz;05bc_Z<ETMu*3K zwNtyXbbU?78hmc7#fwuNYziO6<l$%**9a+^4n?Rdh;wcx{WWZwgcLTJ`aa);Km9(m zS9Y?L_>kH3VK(h_W&2h6v(aRqzmj5g!cK6(`lX%vN2(Eq)&^2fc;eCrE-_WIPI%Z_ zM(QLrp+804#25K_Ro#4DlyDcsZtl)7D=gDnx<!v;Y_BXG?0jEyp9TYTf2r?FX<KYS zLRzF`+D!dsojNms1RaOlZYtF%SrxH7&F3&B6ytM218gD)@3F*B+9bj0q-#{KA9~}$ z<y{2waHZE3c=gN+NKpG+yZ~oFn7<G2VzjhLn(y*`AsL&ixwe1u7HD2n)A76F(R4g6 z(eNSwXDfOD{jOO-Eg#QLb5ve(e1C?rru}{Mrk_A`xBI7mN=Ln7IfGk6AO@V-Le)~k znp6-H5U)j1SImG<j|C{o=!H8@b(X6wSN);ZTU8#73zj{MI{Kq~*firf^J1JYaU0s7 zex=TJ0ZOo<kIC!8rsxGUzdvPL;t$!2OaHjkgj8#3U8v6$yLqr!%u@$j#noFYOvSlu z?eTLwy1u?DDz%-Vxrh3NE_~}ZpoDS8f$qs6=l1XiYpJWsR9(<NGb?k2{i9d|xe`u~ zd|-N;^kMQ2tKR~g#2+0Wc>SaE;oD+?xgxQy*sjF!SVF41*0buBRpNf{``WJ_%;)*4 z`>Fn?9_VMEU3x>Zn*>FvKU(QcX`Z9S9xwsjFm>lBugDnAD)fEf^dIL&%6GWy+f4YP zKPx9cfj~Y_C&$phN|L`1U#`|FlozeqHn^-pqrdYrPZ6@w(G;rRXpksXIGrHxS~b_l z!E@g!*xD0NEAFWI)K~;Xo9als;(#ghQZ>be0GxYN?>?n@2u5sX=PeH_7Um$j!fe$c zUP(pC+pNc{*inG)m48J2joc(Pj|?6W^UE52Q*iDx$AZ;*g#w}>5rH~#LfNPrtLELf zjO+t?cGf#7TE}ykmK!FPtg%E1Cj?7)9i{qf&rq*OPC@oHG{Ops>6S5#=)9EG9+-Y- zca9G_@m$-!!CC$Xi-w%jHxS^KD8Q{+SKkDP`L7ea;4#8iRI~$$tWM?=>&6Ta+WtE7 z&U+<7G-6JMU2juru#?f^UWd;SA#}9j!TVww{YhMbKOv<r_9Il;k6_KK&WEbi7`ouW zfD<b?+$DMM*Ae#arsYOFz~x3;YeJFpb^4|qo=^d_^6J$ydYt6Cfd${(6r0a#p@rKp zJZR#l)~|6UND+hypd6$(Qm5dj#1|-8`PsD6Vc1Wmiesq`RmS4@2yBJbrU1tau-rPz zhi|j<nS&+C0vOp%*E!Z^4Kb`IgqtBDT-n}YJpnYD{KR~J?RC@MiGz67hvkQNjoHnc zq0nP+kzRuq;U7*~synNg5yhHCd*%7saNf*w)ddXC#)}x9$o*RVVy+RjQnJk>msFk| zq`3iE$GD>5(uCLKC5P+MiUGe%g3ekwhZJf~X=iU5=$@TPOb#cZgtC$HEp`dvG$s`| z->K8Yyr2ZHJ6Au#M6WfGV%FUx!Glsk+?@OZS$8oa%NskeIL+T$XOJH1W&J*v)${y$ zdlkQcHGNe>@h!hXu|+H(aQae}?#DVfR2?ej-T>8Dpu!+(=M}^vHob3t?lVJ4u`-Cm zt<DvYSq8gj!%yh`!C?n!EivMCOsg5=NgWq?@4t!les#UQ*KG5f=DklIIHk^5lDCe! z;VoV*x!aNO=qIM=k23EpN-eqT8b)JRZY9}!GMU&AZg11xe_$&?@8hK5L2Q1%>Mx$) z<p_eYd%1@OancRzLPYj{Fa*xqq_RSHKZ;j=ldaeh?r{)2BzlaMIz-3)u$JU{=8$F0 z*0tHyjT><0QbpsAM4(9-9(BQW&-4ZVQPN-R{>GRtC^n7NLsqTzmVuh`_kWN5qY*?3 z!VeZsz`$`Un{0F?S1X%*ACibPiSzu^IaLZ##CoVnX4hO2iMqI5zaWzITH#Mb4zb<i zOm-`Vi02=x7!teT3@mW0BP;AB0b{#%57nf{?e*zaYgvYr>NnD?qO_FOuckkzw5(!b zZ`_q$eOKb2V_oqt>DMC>1&38c&AaelUMV)Jf2&fdg;Axhekx29WCS55QKhhcm5`x+ zj~1#&i&T#moAijU)@{V2ZN#H(#G{QO9#M|Dc*G+=oaYCHI^tYzZS)>qn0Cli>>eY% zCiQ39sKf+E@a#%NseEZl-AokQNU-Ju@MAM1ZAJWXQN>T~5Ry_N8>6k)OnQ?|eP;2H zZptBQgQg-0M6QI+GqvP__$kUM(eE>QmrBZ_qY_dc#Y7?%4fYw@UaP_o+BHv>200Ex z@hk?}fkA!8V&*VW?H%?u5_n0cc#I%1ege{nz{d8ck1%2KJu`-``=V1w#>p-w9|Pc_ zI?+zg3{X>7XS1dL+MJ=?bRJ?3Mtjx8vEr_9WX8)52eI}Z*01ZKC%BJII$!K<g^8^& zu@xpZjs&Zj3y-7->&bl9>ywKe+eoF`rM6va8!R;~xjX6MS-C7h#c6f5q_TnsbWrp1 zUA`#X3lM~|?*K`Y6C<7hWcc?DXvH5a%j$7?Ho)Dw69o#Rn3to%s}4zt$nX-}DR{zw z=?wl2qb?)lTiMScc-PSss8aO~ym7K6bOSN7qQBJSw@_qhbxdf+vJy}xigt*4jtYZ& zqBONE=H2Xti7sUUKsdysLIC;pd~`yUE>6qgX_iCd>i>rdi6_(fsv8Ew9D%ok5*qV8 z{~Hd%CWstbVb2AaZefvDhmu^OT!n8iLP2ZCKM1YX16&0FS2X~!F`cU0^_J?vTm+G3 zSFFOSNd?6>Ul57}dK9WZfBN;aC*sW`Mg1ibetA|P>bt&WI_og#GtIPws&gVa9iKLn z@tXoOF*}6;8eX_>L3TVhv@A%P%uYVmD|~h3zFW0zZO>QVOV8KzP5%B@|0;#(OMh_; zr>ggg(O}H9)>4sdYt`}tS}=xWV(@q!@IrIxu(rK4oQ%=|cHhW%wmS!Yn|_}#N?ojC z4;061D(7a3aM+I4I&Ah*NYKm9fe5==vk+~!Lom^f*MQP#@bpZ;FMn;{zki?Bw~{K} zs->J&3k<C{g^drSCxWq<=acGqK0WJ-+RuLadS7QiQB~oL52ZUG+7W+HXT29QN;+`4 zqr?U!vveh&FbOyvrN{k8iVFhtn1&Ptv(WkpYTs>WaQN*|tg5uqr?`NyGYQQOMwX$= zeTXwt;k`F|o4(c5a7_d=ENu+c61QZm;Q2&}rN@hQ84J_~PwH}yZd_eA8+Wil*RI^D zMm=Z-PmqF+NJwC6^#<^;g+Mwf2dIt%j*8>_d`yNXYS~dx=gRP>_}9HS6LobDJd{lG zvtqzkx2KmP#TaC<G*Ey)JVb8P=A6fmq0IK40sr9_1GWR~G05P9txu;AQ$op#xth+1 z$_(c3ISyghn*^`$NHy`eI3B3I1bW>%WQY%9L=#dtBi{rof|L4avHgh8pIHs8cZY&` z8Z#~AeqX86%xgGntlD1x6oq&VN3iQr$H0FP!S~31e56j>i{vnLDES*%F9M{P48fe7 zYC`7N;rV>-55YXJLC{sg6Rh_66mk0qy80U2oBdAX;6qrRc95-bdrufW%{)D3K|edV zQmoX!<S}ViU0bK-1oTiZnPj?LT={V^WG^5CED{y;0mO~0<xBhQwGsKdFdBmbz`$0$ zxz)LBbuJs;l|Gy<XXELDIF(wwui5=g0oOpIw$;Eac*dyi5!GzDJm|#?GA&Go>MQJ5 zMh9Te?yVEWTUv*XUERv|Up#(ce<+CBacf=xdSH=h!$(Ne&{1>e7<pEmPS3}q>@<H@ zs6MIht!`ASQz@)|3Pe^ZDk~=4i}GzbD@J9GbOyg}DI~msv3zE0!9qJLa=<U@ZCYpL zq=ndRo#o4x9xIRqC}BFb9+c1)QQ2sU3u&V1@a<%Jq4s^oZNcF2#bmtlxNUrEQW$k? z^gG(=u~!{aVc!THaPsAHgem=Jy$uSV0g5()etQ$>`7%|S3LnN*bfo6f#<ZmE`|NT( zMi~}2R{!ZvlI$y_z4v3tKmv&(v*1zW{C9uzZ+`Quzxwt6{I|dUi~q_h4YGcuysh-B zKaW&GSwTsni$aMGN_2!PUqxd@rKYGq;#axj;zSYVmpUoyTf%9FOJbNd^?}LL87ZK8 z;goeU4W|0E@;<SqQFWm)9F9z=GZbfw8PzKe@N>#@a`59$c6ax%<Lbj@G~PoWY6T}S zKCrHY^?3vf4|#`Ft(nW*KuX8ERE5SrJU%}5ufmTMvWvOyNRJELBG%*C>@=^+@Bol* zGyvX@NSW1XKAK)U9G|Q1-A#UfaeQ1Ai?8cIP?2xyiagR-n4cWw?cEOkU-!<A=jL#k zLV?Zp`Xq<2ZwgQW!4}Kmi}M9Xsx9~Ue6FePPj+{$=%y_J&Oqbe78_D0xDnu9s9RBQ zE9z}Uy+)!QWH!Ofp)jdu7_tm9XDu)%7s!FnU%Yzu;Hx)J_n$m_^LqcAub#Yl`C$J` z)L~nGhH7jm{N&loF99jh;)XARz$dE;i4$^Ysq&9NR!T{lfrv?|yc*@~4iBfK#_|l5 zFhZKI2qu|RpJefeuXG-dv}TnKyD?qyKIVVL#I8`D+Q(F|nAGREoL37yAedT30F_!| z!&CSHx?=%Mk!O*gVRAJJB&#^d7=LyEoN7oB*}HOjUS-9Yc$N{6)kZllAd2kB(<Oi8 z`X1;!anN0=_h3A3@6;*$K!3fUTZ2U2z1B|bMx&>_OB^%|Z$2uIkJ}o<p7x`w_$kkv zUR*pK#WmGNS>9E&NG@=?aIG^6(5uaqX=1HEwJ!~fgwB_2-_MSwV@y>}x~}IqRf_1_ z4tPf^{s~;iA~5aTVUbf&f6$<`96lMh4Ghz8sy59i;IFP^VA<&5pVAKpQ_uklPfdk~ z7Co~Dvu-3BMARx!ErBOzv&9OL5TEUjHk)ov;k77%htyVae8mTRbxOjq*bb$-|3P$i z4ObCF2)0pVqr73$KiP#u!{N&~wMO*1S6UMRS!*(2mL2uvTr?!)5TkG_CP=SX-@9U2 z43%i&Wq)<tTT@t|4yG0cp#glvY<<m!H;U|b(T>33k9{-ul{imu=_oP9dKQxUT4Y^@ z8gM1Z&8AfWwzJb2DuoGYR`ht8W85J4XN}aM3p?RM9hu<!cdoUg^Kv|bLLA4#1yCN8 z4dcg6MxbNEkKNg{RJ(LwTvK{(QVLub#=bvoL-m95@Bog*=I@@!D^}jP!G2)?9twMP zCm!|+fP*=BBjA^^Q^T=BGz{LOOV%pDsm+_Q(cr=XUUdkUj4qrP;#xM!^47Z@8<ZiX z^ikB)>;QD(jYX1WAYcKYsf^qA8#<HYByt((_CnTjpR5kxBO*fd(e@-7@(GHEIC8VP zj@K9C2!=FX_8c>!umr#zWF&$Q0WabeuvEl)&xOP}_+rl$qvpsP!Z_cQ!vg{c{-fX? z^HLp;D)b|z;`IB_LkR(47l6f>>pIOu+NqNSamf{<{2U|MCk~ii?FbqIaALH2BO-T( zAR!Pr>;RIVnp2snZ6P^ti1>Ms!y-kN@ulD$(Du-5wpz}YLMvXCJhvQ<&mlpsgVU9R z!#<*)M%Zi97QSg8v>23@UGmPCL!F<dZvhr7mnignN_i=S1(|B_yeg<MTY*Ip9e3YD zcH7?Wu9za!wVGi>u$($b@5teq@hCy(OAE0L>%uh<M)+|2fC#VydT_xEypSPh`OFq| z`x(9woEzUjgnsxpVh4lo4^(4^k_*DZf_r3O*WsNo*4kstXY2a9)$wh0e4CokGY0o( z=f!;Gsqi|PWR-iao>t4Hy=IM8&GEOB@$@JkBdAK6Znp!L&wXrP<rmV;)@u4{{A|4w z30}*Y?T2^dfGeTsZ!)s|BT-BXroL6`8a1J(t8(6Xp14N$aIf(I3KuNEpk5Wjg@4yX zDW}nL%j}ZsSh2u0D`pDGhoGv(3?xHYuOuvT!&GZ@WYmac>4WPWEo>A$sf`di{|bT~ zbtMRWE#|B7e4@hC>ud(RGnr6b!hJ0C3zCdzz?hv9Ba0xmj|$Uovan*qLS=}doU$UK z`Md=4t70T5Bsg2xBPTm$mNjSkdU2|ZALzOMK@&n;U!v+KJZHbtI2YnUfN`xFjA&^Z zyMJyiq_$h?c1zvZLW)e~Z^QP6fB8G6)$+#GY;3a(^YfTZ5bbYGEw^J*%RIFW-yiWc z8gd5#cpP7>OjA$MW}STye43Qb>}x9MM4$;5>RX$99>jdsLD7~O7H=-G?{~5o$EG`_ z*J@_R+Qc@(bmIqV^Ei~KStLUVFiOUZ@#~7C?S?Ha3&+)y-1(k@XnjFeZL}N0@+7A) z85b)}B3$h;Hxwn3NC80Tqj?|7V4DkIoWzWIZzw7mOX1*FwZshUCS9!QV8?>m2`I9} zH0@zDZh~YW!~@@%uQ*9?@Yk*E`Sc9fGljm((Zt!573L>kfp>VBqI?ENtTOVr;F#PK zTQb$f12BQu^CPsGC8ah9AX8*u&Qj8*pwd>EJ`JT-q|jGlxDT${ag*yPpP5^4!eeB4 zw7!=#!vYs_<&s-w0=oA>(7ag6el79EtEM)9aIFv`CL<orABfT*r@L6=@@OkF>Z1js z-`f?!^Xz)-4DzZf8Pnh`44~GCkA~@vBv_kBn7{AD4?*1_d~EMTSsnHZ$;lfug_LP? zw-RhLKT=!`za8`-LY@A5_t2zSH@$JFlAGhEqvzE15N1ARwO|Se7eO6<mR!XIi16^? zw{Rd{bmaF#$y6K+O_#(&4!rw-d|nkV$pZ4@p6n0afLQXw!`g0VgX+Dox~qK{;b6Gq zyvxfm2$+uM4gLQp+wJ}dCV>!(6;d(W^XuIboC0dz-V6DHs@L`^-ZUV8WG0_LKx(iZ z3D#qEs1{zD=KHEf(5HSNlwC=B5qyZI6*|DF3_En=o*B#y+cmK}ayCeC<n+>lE+=if zvQ3SSqbuC4V5tS~dWY<ys3v---J3{loQ<c$TxLd0Y)cUlP9To<I=4>5el^rSRNl(< zjilKuH?aLeb$vcQw$#fgf)yEpBQc5}8<{$B=WPssZNY$(vIe^64Kd|KZu(&q6=(U( z`U$k1-?oN&xZ|~3t*NK`cIYR30scqL(ltNu1%zk3=045tMzVhTjN4{qviC!{ud$z> zQup*e!@hVo$VSdpefU|<)i&5oc3mhXOWan`y8&c`V(}7ax9tFRXjBLNLk|Q8P)tU> z`s%xDs`u-w_tsRuQ(yh|n(BA!tKadf2LLMHtvnx%9!^KACzFxBe}*x3l=Qx)%$<&6 z(DHx?W&_C70Lumtbebw~h-%S20nRlk9K_zb0(WZ*5N*Vjz087(U2pow6L!xv(BRyv zl4YUE;0AstSTii_e>i0$gv=)_5^bL^kFbJAATxh*XO^4jgBP_eO1yI=-ic2Lk=^bg zj%=WF2Mr|+8{a<^s?5FPfhy6-)PE9!Q_OD7)HQntdbRIQ37Iorhe|h?cF1ZwrVT0H zUhD!hn7M%Tv3Lg}NX!?(gyF|t%^j;P)L(ZI+Y{rkTby64?~2`w7sH-PJl$t&l^WCI z>U)MJ1HWKj7wn~JJ+^WmVKnssU3k%P&lN}@Y%yI^2;i*6!(zG?qi(POQEjACHq7n4 z_ErhMRl;u`g;C$HiHm9;hEea!LM^{_6x%Mg?PA;T#_N%)q0qS~hi|VC7hvdWe@JWq z?wS4Jj`CC0{08pnXxqF&`SJ8Zjd+6!r)nA3)SwFguzcXzJLz)#572pijk`J<+t|#R zfY!k8eAvHz_cjD%P;LFc|NOsx^Q*tjDDlp_B2r@)s@(3~`Q+{=rpo_@DigIiKou;> zfZD>oTAF*dho(qg>`MK%8210}Z~pt={olX+&0qfYZ~n`_`}_a=Kj37`f~04^{tthi zdGB^(EkE){pDe@&%nn`G?}0DBCtcl-^7*se1Sinzhkq+4$#3!lC%O4aVTnh)-9J@3 z!Rd578f2gBN<y8@rzfyYl;GKOySUwneqsQ6#hrMWN2lkLx4_XM9>;Xl63tkbeR3NP zBbkN1|6yK%h~NKh>-QgJKwlIK7UBgqI|@_)J2srye=)U=vB>K1;BCVGvYNnVJsx&a zEM6lli~^j9Vx=L{ezE8G^k}Y*bgzqXF<eZA7c$cYk*7kX3su2Ub)92iR41dm#p?d? zAG<cXbM-HCndo9a!P-z9u0@!IV2|I1TG&d^TM7D_B<Q&J%*VYA<0<G0>4*wY`Snkp zzI^lY)r*&}v9L)JLVxV~2H*1ME;9gU9HB{UhNzf-)%Xlb$aspROc4`6szOIi^2r+k zwc);q5f8fAEA$3s!L+iBxvpww(MK9c)NPjhw;s>)3+Rz=pW~M<tFNGSDDLY&sQ`@| z3kg`h5|o!FhqtI{ki};?Q5XQWn2%r1i<w$rudaw;YI!(TJG9VvHbF<SPhn>;H>BK4 z)^;k1U5nnJNy2obG2g#&kb<DG9|IAL*h5bO`Qc9Q#^qfX&&nBI4n0ykp;}&Dq;2nV zB)|>7cIB`CMTo}W<|G1)!`O?J4r%2wKn$nkrfAt94of1;3NNzObG@uE(roc?x?}>E z7PeaKYb(6wgwTBz@R8VcKQO6Q7_r0N-Z!&OuvHYa2a4O7D=KIMe=9*xyztoBKbb%? zzwOX4g-rC|e{LS{BMt*o)md`G(COAA8eYwcQQOdn@Zn5$8lGg_k^sH@DI!rtKJE#h zY!hLlE?$lGqef5F2Hwze*I0+Ks(=s;k1(87yBPbiPT0y$!*F3n{EeSK^+86$03I5- z;^%b*(rFrbB&_{ChtP?m0iGW^c+T}Mqwi&q<kpss?$dY$URxxeYgwp`#{Qw})?c&! z_f1#pKapTS{EX5>RP%*Ert_v4{s2hPq#y++g@wYDJ=Cp{rTh~XHD^zAbuv{)Ybk3^ z1Oq=p5n45B;BmQ`W=K(#?Sn1f-2SNkc*=kT^OiWetT}akh}iuhBC}OrGME>qpzLZp z)h3uMVVAoTb^K{7=Bu-)Mw8h@B(EzgQfYeDGS_6?1oIMsX=?ZvI5E5F%z`9U9;Jj# zE<A$8&&HcjN2B#DMZHa9jKxlrYz+7zsgn<uj(<)_1t4o07R*vXgWzCvAY!^QSOz~} zh%l!bcvD$ATNhUA4vUF(Cl*%A52E!6tMExp9IzOt^Bdn~uy&g?-eVPdO)B(Rg?^I? zw^@bTsS2t1X4it?yR5?9R0ZBb?To})_%W;S;}3{~mg0<?MLvT9`VtGGo{1HAt+CHk zcga|EvzTYxdZ9mnI8GWvVOXW6Cko-QLIowmw~i1L@(&~Cd6-KE&m)C-K3QqQX1IDO zCwLzL=5SL#Z=C627cSrOa?<u1GGX%~uNCEl491QJ1ul;ik5of%qtwR8K!XU-yV(xQ zk^EbThxT!gLmO@y3(ZvAD3lt%%|VO#p$=w7@G##f06IIb7NC<WD9)Nu%Bk-Ui(w8I zkrW;kOak7OBZ$qipqk}mRH`?s;qeNTfZ%m}T#m<h4@N=KB-Wb;YIB!WB6wOnwwnG} zH3-f|Ww4X?EJ7mng9vc86x&ZxIc{mp7W$BYZi-N@HKdzjb0$*T4;9a?cMZ|pjKqb) zW;1~c??%2Kc}pz?xcvY<I1B<)qoIXDaeQ13OK@UdoaU2kM1@4fT%BsBa51hbf=XSN z$-oq&V$9e2X+)$tQzo_%P$9$r!w3jm3_O{j%OO=cP+P@eQdhCeM7fAof(ahWj)0jc ztc8yH!k$hlrH?AXq2hMBZXmaKQcLs@qt7qwpd_}9hQ6UlhcC@c`iElnl^u%d^!ieV zdgVkPlX;Ak!(ef_gl5_%q_Iw7{HXYn_qZ58zuuN_);0ay4IBKM0TMDXH5=f<H-719 z4e>T;FJ9p$)_9(%L3Pl9EHb4;UN7s<*$Q~Nz-2@v5#0l9P*~cqe<C35x#A<}Dhx~X zvuqcl%f7^@vd>#=qT<`X-uh(_xEmTMlW8Fmn^$G%6~1=i;Qs>v0RR8&y={|QN0%7( zeXE}TaOqVR7!C*yM$*<UD2n2%M3*GeB=tPIo|@uffIBl+!r&qRhU8G1s>Df^H}N{_ zRFX}(oN~ERNt_RMvi_2C>}U7aSn__`zmPsJ-F;5KTwDyPXV)p9Bo1(I_vyFOr_VWk zUarC%1RHD(TlK!!24)F_R$v`R!1M4m<eED^E*Azv+(TCvh@SNn?|*!2>(sTilG|Fz zZE7Vqhcu7%*z^SnpSW;AKLbZg@wSYf0kv~6SU5j55f&|^rb7>au>^3?7_y@CT?dv( z8L}}M<m~7QejMW?3cW@?%}<WLH9VkWLZ}p)>)M|pa?p33h*gWq=`JSZY8hT^OMtcn zXiI?F5+HWCxwmo5WU>w;-<yqU@44!>@urI7WP@vKNVDFtH0$Nq&fj}Z9cayq28-Ug zne!n_S9HJh)I2ep5<_`8viiscCuS0A)&(}V3-lZ~NkMks)jBKIRMh9+Y;<Ov&8NfS zgt=c3W#lzls`%HU_>JcB0xzf$fl6zd5>>yBs)o&qBO|gNeCP1YSBuWv97j}`&l1vY zXh6$KsB0RAIfTEr{!c^a33bUAAhxc<Nn5~Ope79%bC=%md54Q{Hw&97wZrP99BRHE z??a*JiVHriqg#9ZbozZ_aQ4Fvk#5V7aP5X2;=TMRpTb}ZgkaK8+Jw{2pr{PVBbz)+ z##^)+S!ePRH!PF{iKAhdp#B<0)^b+%G@hdO+280S9A*#67JVnDwJl$}<!iTm?FN~C zuXC1*;?cAotv=jGS-J2`FP@0zAv%|VglSe$6}<+6DKv0wPS^=VFI2CG1^!Jjn9dyQ zJu{e1++u=B+$=t%t+j!H91X<k|6GXW`~Gn^tLt_!Tm;Cy;Ejk48P;iv3Mm0^7#UKw zym@p;2?6}3seKy+G3*^!)yNWqumr`@kVW>xA{;$Kmih4TR*L;+W+yaX@C!ARKjMH0 zK$~BIF+9)?v%k%%!ZBfp2eHJ{iy(oDxHwjzi%-mQ_CL&W;w^$}ezYMIZN1}l9nS!- zWvJsb{|;XfF$_50@yg#m5Iwn-WI=TC-xz0|W{z9O4oy`COb==@^0%mryC7_rcSeiq zv_*pYZr`*3z#Axl;I~ya_@|nI@s~9?61rH!JS9ChyzJbKzc5V|BJOyyYD2S6*<``R z!^EdOv82syC+frFHUMazBLHqq+Q8tMj)3@e38AFnq~fsJPZV%cj64X-+EB&Lbrhu4 zetf9Iw1~AlKYCidDW?!S=&X8Ms)6E<oQRjMDMqnbMX_$1wrD3IAw{MqWutJhu64X* zhcG4POE)rmi?&G59j3YQ=s>iVVtwi$<VEW{5?g#^*P*}&!3mgy5`eP-Q5vcbberj{ z3tBQXlE!!CJ^NGctjzw10bVA=c>N#_srxE`np4R{!Ex<peUMyVG8-({e)#lW8zeV9 znRi5x+;oXy%TT!aiEAIxR?XhJ$hJ}8RD|2alm@YGZ%4a1L!hD{xr%MOS?n9q$Bsqo zwM7o#{#AP-<M1YX0&I6%I+O_6Yd@_+_u9|;5We=4N107Gb{2^YyK&{icWxMts~)=0 zunG_ytWXLaOml0hWOsk&q%fidc4RM=s|r3BTwstZ@S0>FD{9_Rof=Y~_W%Q%+c_;k zn<e33&4H^qfHOnDPEhLB$df*uV{=yf34(YhwxKS;EKUd}VbCi2n(Vd+VJ)PkW@*Qq zq0i0M9-GF@S$#mtS##`PoFZR(F$MNE{j}fj2Si?3ojMCZRQhl)D7!*&z4#~`Gcw7P zYD()S6yU`3)!F><EOj>W%O~fn+*>K;z4t4Wt(bRZF>g(je%C?q%(t2Msb@oB!m{06 z*qf$mVi53QTuc}S+Ye~;t6Aho37@0~dZPI&aox!&!(EVN1asvDJLo1g$66GbECvXv z#}Q0vzh&7IJ$eFrhbiVB(CqYjtgSo9*Nq11G39S+q?z7ENE2uim9@7ud(XJn-BOms zS3)yoQcPF1@DuiCEYhhx$G5*XtR7kBnUmE&X{?^A9hdS1`qLta{YuVi*aTX=L9;!) zbAl@yyltYF^|3iIF<8;qthxqrhlBQHXeg%n0wt@Xxp#^|0XFM9QD?rcxdmv@hBNA+ zCG)BVbBA;w-l3cc`og>cuZp%BZ+R^sS9gP?nyCG;c^f5!M9#rPqC!8iJGtD1zR0i$ z1eUfL{@}Ad%O5T)CwuaBcY<-b@VmqS$mDgtl35QkZLiAxZZLn{j)2hFk?HMJuwH&h zkCcm^kMa8s9bgF1fkFF*ZL2AeK24#*NLP!QBe~vT>OU?j=foE`+vTa@+dyP8q_4yr zD&=fxm{e%P84ek7Y&owmNofIhnA6d`FA>NodY!V^V=4-^VtICYa<-lC*wzMMYXh*g z0oW*G%2Q!r-<8((u&IKyDv_P2EW?aXbk)DJwa(e@%<ax>zcbsXQ7Mm)%kk>D;n>Xn z%O_I7v?;x!Gp~(MBJ!tzvFFSz!DtE=EW^mTMjKfR_61`zkTb1Wj7N+5jGS#K3XhRr z>hlFC;VC#!ulU|z?L)K+en3e=ARsU%F^L2H1o87bC^^Zo<+6;lPa)DGAIbVGBQ*DH zwX~=Vrk6T9f=a@m^D`KR=~obT6<i}v>(__{1P`P<8d-;8v2}j%p}mDXWokXd3Qt7Q zu#IQfD~1eI^SBZY>eOd~5=T=p2v$#q#YeHZ3k6p&go6O+_e*$brn3KWToIM}h{2%k z1sMt*FCo^fa0#^^!~>pQD3{If0sg9XodpsUOT+?I?n+r-fV+i_vj#S9t{i~R%X&@q zld{fLUa|`UsWSZYAOG23{`eRF>wo+=|Nh5+{*MG|VCm=8+Y&<5s0E-3KmKq3^2a~_ z)4%@ppZxfb|Jxt`ufO=~U;i<D`OE+MuYdjD{P?H;;HUrT&*9Zy{0~(4Km6N&_tT&K zzd!zy|Nf^x{<FXO(|-fi|M;K&^k4jYQ}M@t_kWnTzPaMMOqAmK{j6SLZV>Lh>&8$2 z^FR8lzxY=_{dfQJ$3Ommf&p{VlQ5fzd2`q1+@`R_6z|=f3HrXvf)EK#_V@Opc}-`4 zF)!d#n=`LkT{u^Ie&w|`F1)~&v)OVsTh6AX-LaWZLnp|oiq>)=4=H^6*;zd~heu<` zEwQ|be{P<ZQ`7EEI<E-8FMdmyEI)ko=;8erFP}Yr`t-|(FTeigt9uWh6?7jU6mDc& zXsFe5N4LNW*Zt7`#MX&eyw}Q4PyB8d2fO}fuMj^DwQkhv1Ks;bD@f6H>GZ~M95hFt zQ;ZUtsr$z#LB4y64B_7P<@ruw8GiUdyp{ow1pGeublpF#rZQ0a!gWP+g6?tOaG;CO zg4VE9zPYynKVW$<Q39U?#g^F&UBoj;0FGlIJth?Z({!4fDD9U00C&*<2bqrmO8$!1 zF|@3cvMB$VA?-`NyR8@W0B<OZ?+Mw57TH13c@wzmOB;jR7oDA-9hHmAqME#+=Oi60 z9gw%g2;MxQ$ig>qMK5`qxaE{vh;JDx5xSVr!0>TX<ma`Hl_P%8g)xD+KYeSqkcT7m zk^--|3VwGAYLpisGz@`_9*g=ydnu6hs5B(2r1WDdIzSv86fu<AK@kNQ)$Wd4F4;7v z@&K!THLOcZYf3SLPZu?&jFz+7pTk4obg|b_@&0mE&#8jTQ<&~P_Gb*+@$RZS*a?E0 zUS`1?Txx=!rXe&tg!y7W_{X=J=#3;O@!|&qajTh%Yn3A9x$uT)q;ivNOfbSB-Ow1p z%Fy|#8EOBd@2+hIJ8-U6^>VPM;J_hvHX1*9UOcZ#L-BjSL{y70c)L;nn4{9LA%=^< z*j+uyqH%^ox8XxaD+LSAJ(PmrL^tjB4G&nnrhe(pE)83RNAPcdH2Cm$gLq(`wz@#X zyki3b=@zF~fJ-m@zmNandg!|n`+W`JUPp8}z}_%nW_B5p`Tb$>`@bI&b;IDzR#8?% zd_{l{_KW~on16)LO5hBECL!f?1pe(es^Vsv0k|#@lw{gsS3II`Pw?{hmRiOaPaZre z?xV<jzCb=N1eBgm+-_yJla5s02eG9aLP}cu+L;IsbP~VuJ|z1pu=I)I!{fYl@!#+j zl8$Z;S!i?=PwCx;=j5xs_tDp|x1YP9V}f-FFJ{}aS+~50DV^nf-mqR}DS6SD7E>#m z6JsN`KC~|^{e0<)ZcJwlN`G~Z*X9+e{D5iY5g1<gxSmey*CYor&eH|;sbT-EH$SxY z;PInJkMDo;<qMHc&eI^&GUB4cH-VMT11&}G1DM%>wiFMjo_{?$gA*#Kc2G&HF*IEe zQ+Qx29-{AJ89z@4_D`JDktH%0jL+PzLLylu<(2h*kP|HOr+C1uLCjNEFwc1yarES* zTy(t-GH&oe{dzVBI=rU6i7nTnITF(njq%p=f#>Fsak*SdK6w+KI%29D_V5F8{?r*= zHSdgElvm$di@MW#NwMtQc2wJRZzs!gw%_KqTyrppfI(LrUm~V(1?f<De|O@FEcvE| z>NVh?RA)!RYzU1^6@5_TSQ0=!rh|v3Aw5E1PnKRSMz6}nvV3P5mt@C))zU&>IB8F; z-K+`?{QPtT6u8@wh1`?_HIpx5vRH#bkQ^e}C)JY1m`7l^7n!C=$Qa?<Pal}y1P})6 zAPh7FA65P`oC?wnuSY#9=Vn6=b5XDCiP}{_fB2!;+c6s*jY*z9Hf+)0_xA9qT9Z<o zvEO@#R3ZI{tCo~~jePHVnC8raCGU2mil$Gv8$%H4h{pP^TmlBOzRdvm;D$*X?CtGE zKqRvapeAnu!>p*r+bov9rLg32Bcuh&*TdJj%8HytVrw9(7C{j32E|$PyoF>5X>)yi z-0hsz=gSh*rkx%6;Cy9&`(u@Ui*wI!{2VTVa*A*M5{Xn?ElsfQNEJt(UD1_>ZCRXb zC#0sJ*;Ev5FIUBPWjV)8m6_Yi7(Rz@4T`Me-;`1-IF*B9_3R-pS6<ZwA67Cs{wm@Z z8y59iZc&3lZNh6x^<*?8%2Uy5K$d~~ImWY|$<B;|Hrq_@wVqfhvOow0M_=A-#h#mm zd|{DYyA5pK#u4OD0&U(TIqlx@Xi}1&jEG);q&ew#CO-gae1w^0kA$wVnw^?cVzul_ zb!V>-D%$OJ==M5vdmY;9I+U(&?t_r}%c^`GS$Sp5yFh`{oCxhDQEX1)?XPO!f!evT z`PJfzx#KsCjwGoXcQfp{s1cS9#pRvwKoCg>!m2Lwq#idx#K_M;5q_2lNOp2kEmjx7 zYn1oPrS(NxvS}?`{}|p|MYal5UZz0b8lELW0r{8R@5%8S?2W)_d#E^hGci#>s>_(Y zTZStl;#kB}{F2JxS+Q=Sqp96IUKd77vbAi6bnp^0OY*E9ow+VFW%OLxLIrva&B_n< zyIDZ~Ix)4*aDxO121W6N{%?KK#2EFWJ#R)EEf8mqg}TwoxYUP=s9bJMoVI8B_DsK; zTLo*eC;}Yc+=d5w(V<IbQiBG56pHc(HFvh#a=R_J+ww{gProtSio=xlC}`>Lek6az z!BAZd9}86gAw8T5g=%A<ic+@-ly5JHtIB<7049sw(oV$SB)xh#w*I6~<~p&c;!4d+ zbdwhs3;5o!4Dc832mIIQ7x7zBLX(#-A3Xf!%a?HqP_r)ewCo!soOL^2V*WgAO<yuY zDEXtZbhph0l{*>`e0GaY@jY>I%i+Pnptpb6JJ|1icz9S`b`q=<eDs*jBtW`Dx?uCH z4(N8}oJDLWhR=&T#P!^?^xTsHKhbrPAy2?goulD~@Oh~GH_(P9RB)m{v(Z_ZR<~h$ z{o_S_#<%D}t-RJPY-X^&nN=M$GHBC?jVoS4IGX9SdqB{7qN{SMb<$#?V{_(uh<M~x zW%|<Iy$TaPZthk^XFlE>+#Q22C8e6|05~+A;TwajKp&(wf)9t8t!DG{6>T9ncaBQb zdf#*eLl}{9Vj{+S`Rj)tD8Npnb3b1qjg=XfjyTgBE(SSWxnuKm^vY&~!0-Vco~KAW zl$3OPK3zGfpBsEM9j)#owyWe#NfH0$u`3KWp9rt-pEVSBH&{`n$PkQ0YF#SAcs<pi zZjmR61%RS;R-hI*+FzcVbMCG==gFhapcr6?K{c}v&XFIBML4lQ8B}V!JQVc9PkbT> z73KpWX`UQ$5EVH47EXg<MzBZynB=|F(qCcu-w_4cqrp3%0;9tfFhi_+GlCet(NM3@ z(UZbtMcS5jYs4|O!Zm6)Tj*dBu1o%kXtVWx4{P2s8*QR<hPg3vn(&AfuX}Opsi;KR zzBj?9LWvgGW~6z5Vll5w#H|1Cv(al(diFimrLaZl1~EelbQZjcoxPGoRcU7hpU~LJ zHqI^Q%${sj$JurL-9A&WYPMW~)MuD;9DH!acc+MI3^V2m)JP{LusMzycSV=`7WTtw zF<roSNUh=?0=FJ@J+q@0^=n+?hl@qM$b=g0Jf6K8O{<BWDo7Ulcb`A`8j_+KMxi>s z;EwJ9Z{8<DMOkC<JlJ<NN7(y{&jy&PeuBYoCW!W{bw8Ke-s2IZTgDLe@b*3+YZD6j z4B63U$t1f;mp%YnnCr#9nFp1sKy*7rKNQUL2EZ;`gV+Xbu^M+U<X$23zHc~g73r~g z^eOqq^n7i-aMs|HG7~niVw+(5F+)74z%q@$SSB}Jj?Cb~ZXMV#f~L8|ctrI<i|VOi zr6nC0SZkzFv}-Nuw3t>!QQq$aGhM=;YbMhdZ8>SW+YMVI!@UdxL*D3Y!RxUnsUWAR z?Gqc#sOn+yY`@`yVP0#{_l*qgM4@y-G_+qlo)x4<RmC~yJQBR_{!Z~j@y7grA^xfo zAKDVC2>%VnNlGfK8$$gQnM}v_JbuIDU1BsuIW|jFP>W~km^Lt*;Ice9bJ?^d;IW__ zcW09?%Nf7{^&oL8PTAHq;8O8w34pHTm1@nk?w>Cx|8OeSEsdnH2+Rjged2wN)FpSJ z1GKxF*ih7AY6e%#Z06L)0}&28ol`m`ruje)lLBR`x!IK%pKj9td)Effqo~4a3csT% z{0_+)lr6r?kTW}7Z`MqeEqBqPoWigJIXl$-FhNb*dKZd)lpoZ`87>fzw<&>Q6U!NX zLgvu1fylfO@#%L`Qj7g0GA3po2alqls`(J-Au}=5L||lj(*$DmkIs*e%S8_?A0&>Y zGGzE>hI*M2Wyi#Bx-A-vEhjXk$Cwyj?7c2~mv}8Y#Kc&1A;mghAsSQ@X2HI>@9j43 z2euktjb>&d7aT7e^(LV?N6OOlng!WPwCQ{^`>uYS2n>YcOv`3-su?`+4BP#{3(eqW z)II26KCsTAw)=#SE#5yJVg7d3soth&H)^B1qM-dlsa9y0dWQSz1uP7}>lKp<Rr@M* zuz5xiyc5biTNC}2;fC%adE;Pjil}HPW~P>c5jz<>S^mtaR<ctCo~N19kMgkHc7y8U zgG6B%4i#<+4KIel8QwyX{0_Z1urmzNqH}G=)XsekoLE^d;0cJfDiPrqbv<2G^N^8^ z7g}5bY9&da61hSiVN5yh?|6iZ1U&78YQf}%5sx;w7`oJr1UFWqA^WBi_DkJ9i3;>Q zDt@{;gMlld#<j{w0?HOBM;T7+TXi>4=gZT(<Wf`C>@Qw?^<@{!e{0zFZpWPL-bKzY zKZK(;Jo4k&c!A~_qiNP0)F!#UW*6R?lAxJ>Cx#6fPh{Y_3|w}_wdE{kF@X|q0t8TX zG8<2-MHnCj`nppL%8#rw3XVm9%UuE6)ErTla$^=z!`cnrRmjhRE0UiM0cG=f+0W05 zgWG<B!rlr)!A+tso|N@_kGgmY@0^oJ6hfkHEWlhKz<d-y8-(wmGplFl=Z=46(i<6h z?_gk<%Yn*?vpx)9E(Tzkj?oA^IF6DBpN8m-J_G@VROlDW^ZC49tRNKq^k_8xF6IxQ zoSr6r^Z2*`n^KO>U*plZ0{fkEQq14nT$bag3OgdchQN<U<~Ti>p`N2UCIc*X=)M46 z<kP50+r)EdA?}e~B5PHbd`-*qVeoER*YjjN5|Ua$)ecr&i%~T-Vguw6(8cf#6*T?o z??*tM!=Zrl6j%X7B|~3gg$eH(8L>@RM6;@B_<+%L=JoUfeyI}nAzA@A{5fBkL&*@3 zo@wO>&6F*w`O3DYA41slcv{WTf(thK@w9&3ZyIJubzsCU=qp&9p>2i$0pJs41$(^7 z24m-HaMGAfI`N*Vh)o<8i8!dly<rU8pZ>)^{Ez?oFMs;8fB4h?@~?jS7r*}T*MIus zKm9);{^mdV6Xx#vuC#glAO7t>{^@`Gzkm9Z|MJIQ|AU|Y_5c3kKl|r@_2>Up7FR5y zMdR|ZSNb1EN5a<Xduv|0H7~tlU`xYBzN#iJu+DK!`hj1x3~Tu%msK$1$bs#{{uQ5% z42CVEh$yOUXf+cg5MFhy7>YqjdTjCa+~A@Y<+3pYT_`_tt>&2F%|L_NFQu<KThgoA ziO_t|gbvIHPm*EzTv5?G<yzZf_*Q~N{m%@7pEOjur9FFn`-y?(JEAOnrQq(gCQWf) zsA*e8$;VWlr9>U`g`&NPx*P#R#B-6w<7k3WB^BVrhg*#Ut#XLfCB{}=#z9r%kc6Tw z4$uHsmf`GV;>+kUG=$aiSAk-<;Pmc)hIX9K&=Je!loF}~fWvNDJ|Jc&Uu+N#5QFH^ zzXSZQAtWIC_`iijx`VyF_|4deq(CE``Yq^q9LCeBN`e2SzO&IP4HDUY;Jo385V>;^ z7Z_Bm%SZt~9Zl-jy#AoYbj&tCnpwe>s_=-G<%%NNb(RR=K3U~bB|C_)Y=B&kZ4B}5 z0i(~MpkG`$h)txT5_$O8d^gp2Q~e0}5`j--85@aaNr8uRbk@gV@Ki?(-P7Imlq?<e zt^uq!R$a3NH%AtV&ow<}aJS@hmkTwd5$uB%MdWfpbs~b3j-*5BI_r6=20T}kN6ETr zq9zKcswl8OG;9&tI#RB&5O2F3ECor%^5#)|tWlsJH!Idf_`02dn=a|*rR+wvd;m>6 zGT^<li|8B1R_3-lx5RdSj7k9#@DPy<*-`s}v<4w15^-%D%8}Ir(<PaO(IbzZ=uaVN zl$Zjy##XlmNKS&jrFO;5V6oJqU&^)^8k`SoSW6G!U;@POFW_fjwQVWIs-Ej8DOxdU zW1_m54pfd}St1Q%R5!FCy40|^OD|upqKT8>ZtyU#7rsIor~s%9B``-Xjv9}6OBhA8 zRh%*1K7}J~2tsPw8HTEyO#+Zk&cP6}+YmYj2W1EjBPNG&VeLV1UiuycOLom+gDhsJ zqdzgScB?)_yhXNEu_Ue;9T_%=VdM4*eNhJw(>(Z-Ju%eIQU^sq+u2oRSr?=67#zq- z#e#Q1)bw~cAM`v<jps7Y=Wd~kj#b(D{!adQOA?txryaquofrLbu^qf=oM_y?9Yk-( zvhqF|;`w299aJ#a+Vx;Kk)~9)i?Rb<c0gsR=;4})h~i?lkeC|cUp9+Q9<5r5g$>=- zd$9@U@US0vvIFZU4mw;ev3T+n*VV?V-b%h&Nq`v4kzWOI9Fd6_Emu#m%mIaT>~Ubm zLs95IKOarNzjAzc^}z?w3-KUT?17o*S0l)18&#06xT*^!shjD}2iTo4|DEsSALlXP z;z!2S23&1G)dsej+x9QOH`H3y;QYPGb+ScI5WvM60L^{k1riY`Z%`sz!wT$oR~M$R z;iqnh2$U8z%xKCP>ex6fd1bnScL=C5LQg>qpbR27`~@BBGx#Bx`vw6Nc`F10)FJ&E zVK=;XO-ANE4rc)mUm(-s91!8cJ9z(zK=P!C5DGSb2y6!sWJYd8X=I@Ldmjf!RZ=WE zu>2AHrYNQe_b@!@%;cYqzAFpNPYcRCOef8DryTzXrN;`c<tIqpH3Xb<oTfqYuyvb* z$3m32%a3n}$2$TkLI?v1Vc;MPB!oc{LfUb{W%Lf#Cp`|hYdMAl2glXIOb;C1J;|O; z>T(Indb0W?X?w2+=dwZR87Myu>08xE%O({d+w>yg4CvN=zaV9Q7mi_A`gd9L;xNO^ zq)d&HkRy>f6{{yWn0Y#5njVTR*3^qmr`-8OVrL5tVEQa&sWmzhb4tLx-Y~Bfd{@0e z5KRzfU>i$tdjqw-fx7w)RO2nv-^~5eQ@-?%?wIV05BA^vj_D;S#VyEn7clm^$QmTr z%Oc;U;Ic>+3|re&`Z0%VoGHMYyNYZ9mR4`9oaAo7m6V$kd7*MTn#YC#)XI?m#s^;F zUJn4LFYNBuXJ_^7`D)tgGEC3!)dkCz^BPT5I`nfcL`(DZHLOuQr`!^a*o=O@#9{Ud z8}&qmpH%DozGgcFag{k=JB_XHD18+XNF`CJF*Stk65h=ew`I3cMDaiSu9)cTDVrd+ zsjtf_-SRbAP2N5#i+06(%TA~6W!?MEj#Zk`{gY%+qwXhfcsclUe=SHxRVddttOUK} z8?~jlfD9SyI9XWezk=B_*rGC3c_?u}?44d>!swARm&V@WMA%8lEYz}Gec_6^qa{z3 zTJz=de93MNTL5nufvR*V<2i&HZ=fRy9@-yWTONALLvMNL|J1Cw)<Ap@{5An=!y;P} zCTyo1<L&9vIt@K2L>?{32WKz+hEoV}fxwFEX~3N8{s#;?_%s^~0Mst6N<{&9KlWYi zWEMxA*Z1nmmq~o_X3NtT!NZKk<1fn@g_`u2Ozfji8h*Pd$_0HuF6acWITxR}h^oLS zUb;@Xn-N<;zT_V4-WEE)eaXSN&Xea9l@It4;IQmaZr={>pVRgqDPMcvRBo)U1jN0# zd7FE$J435M?!>Shu8zigd%x*Gv_+HF7o;kvj0?fA88&WqHP&#C#keU^RlN7JrDiXM zsTvy<kIl70GrKPgndIr-TZb0*Qq2;pfJliB9rJH-`C?EgP1M33Jx5JBI1S?IGcyMO z!BeH4J~U_L*PpXq9ZUPJ-O>At%G{<Krn73+^(A+~-jAox>`b}0a1$K!P!Yr}>PPV@ zZVLQ<<3=(z6;Jj7mN_gwF>{r?0y#3Y_PbWs5@N`v&U9+HEoM|EH8EUb(6-lZ!1zE6 zbU&aGp~Bx001L@4GC%sn?oIlg2p^G{hfm*m8qy<!rOS+trJQmx7ZO--XKc?5?xy~U z@rk50z{<3s#|4*5dZ<__F2}N$62I|!HfoS=z*M*N;e?m6akW8QEi{eT@FF(HsT_+f zpXF9o;Vp3HawoXqu*9|}IIFFgwiVO1V%jER8fRy}H(HHPVH(-2or67t+7z`spiKeG zS(LL#xAMK)wUGihc1^TzoM)qUIbD_~5C)oat$W|Qqw{Jyp^(oq&g$y4N3F(z#ta}h ziQ#1*egs^!Yu^=ePaGAp=MufQV~NhB6#ok?TE;4Qh{>o@P&gL?iL7a%T*?^K%`+Zw z1`A%)T?eh_0p|h>sTfa#)-j%3lQ16}HrXm*QQ~XL%HYZ~|7LH^v}1ItHig~{+CW-Y zNUq<Ty`E`0HGPN+4sV<A0ec2=qJd_P`Zoi#XfdaC_W~TPF9u))BoRIb<N|?Q0FVn0 z<PGpmZ}!>%!I3$0u=`LN#<*|BX#Rg=7_o~%YODz~$%V9W%`i5$1{x!phz3(pgk6pu z;EB!~13WMyM|#7Aky3BMkAUF(1__c4O@n<e+6Bk^4HK+Zo6r_qyAjo(HYa7R_Oa?v z+rsP%0nE>HynFdz!28?b{@T!Stb1?^+R)raG2-Hfu8AlPVtbPOv*F&~Hj@3fofxOV z#0dGJM@UVS8xgZ5Qm#Mj;nPIe-*pGQQ%A49MMC9RaYLv!+2q5WDUQ3wRH9lJjd>2) z<^(SwyUNsNYO7jxzJe8K^L62VwR{RP;BrOMcf#4l2D&1Nz4D>4yz-N8V!olg6(qMC zaS0M89$jFsVal+~XBezLx@Lj*HzC5`r2s)dzP~GaGwN%`c16V)7I`7bDF^c0#MTQB zr#zinsF8*|U|EaT*U<(07=v04iw@t^bCi-xc|rbg>hGNeP!ery2soRA4k|3^exkD@ zfJs#e`mur;NteL;G@?`r%!(&Q_H}`Vf^@A^D}*YRvv7&t^{RdbbFio;WqMHrgr|$r ztZvC@aMdMZqqAu_T8KDoqJWK;==>!=Ej}(kj*RJJgiWU-4d=ihg=qIdwS>&<M^m+o zZaFJ+Vy%WjEFgsp0a*xECGKOhmbGEkEM2+c;w4sXHClf66mv^3vV~?zxn1-(un~GT z5??r(9%t4d1b5bL`yvX5$tA`%9<|&5ahaDdQT=ZjAU~7j>~|}R5Pq9(f(1~OF%@)P zZf0NrJ6FUF+MDS!p<$1<z|t%MxJ;c;e!-*5Wxx0kTP|K!GB{g0OZ-XNSg`?3z3$3Y zV@Gs;(b-Asvx4HzmdXos2?XbkI7=`OS%F(?utxR5U5L#q)3G9O^o~sf?FMC&z#xzZ zhI;koQ`<>|g20wLJ=sFS4o=kpGqz5!i_=RDA9DmgB<HUch4@mh>|NMbqYKKSGA|dX zYoColySqGR$9T{pfVQ`Mvw~Kp7sOvBMg~?1c8B`~63=70V5nB-tTqb_Xy1_`0%8fC zt4KjgY$E#!Wi76yR2SW{PuEI+&@1fMKTJh{_JDG-3``^&@%X~1mky`AVHp1VZi-+w zFx3$;3yDQYlhX7HWlvQo#ogLj$(<OI0AstJSy*-xWOsWHwY`Vh-a~DaQ#g}282LnJ zF3ek!2=5C@G>RpBB2cp}XG0F}^bCD&o|-2Hf4e)bj2U;Sl3EG>faq#z?4$3CVo|*N zK)-9GZEv<L>jgYTR1F#Ipw7ISmr(q^AqM8KqL9T(>EniCLlASSIB#f}&NNfmd{@Qo zI293vI>8QQ^>oT0DST;If19y|GuN2~0ULxggRI+qn$vg77FP8N7z~%GETMvQQzTW9 zst07N=Q^0r-(){gtGm$}e_5#vYTass3o`gPyRRsVpqtd)xF6Sx{r7RZvcB!>#YdTj z(wI=EazM!f^HmB1;q#u>jZNX=-RM`UTlK__$Sqh)?DP}KNq8s^jm}dR%w6hnDlZRa zaa((1OSvwIcp+2vqDFzc>&JNT9VEcOyLwZ;DIZN@D40ch<0?^VvWD>|(aCvqEYsQq z8KSBIer^N2M|5l9u9cheD`9IM)O?jB@b<F-O1tU#meBEAi;mx}K}YzQZjdeVI<)-u zaHk0w+h2Uar-{^YuSBZ7hF<MutsE?vK^riG!~9~wC%hgGQg0*`3=RE)#ocv9+pG~p zM7k{lBIQ2axr~yeLL|Xe)!fBJx?$f!unlo>mB;{<YhxhV80a>fMWpSGYNNS{CIYq@ znn}7OJ<Wi!CJ&QF^929H=C*RI-OzpW8K}0R=T`LGik=&Zo@CQ_A8sXoS(UH7UE?Y# zS3&}@Ae+sl5GbLUj6~^@zV*yXRk^HE!8z}eOX<Q>zUp;)EcU0hx5=a<m*Q*$f^x{D zq*8hl=YaBwIa0Ma7m_H|Ni+&7W;1%I^7j|Y(Hb*S)YSwRX|foZ-9LiucLIvmg2D=` z(vKeh!B-E9dnF`vC|*PUF!lQRFF(h~O0$kjtP#rC&tE|d=C_Of-0aa6rOyLxf>~4y z+_ijlB5!(F8EtAsr>BNtNPfBK(lqXnqOdD3W&$z~nRe>gadl$w`ho*y8PbagbTmC* z=pJ}G`oB*rDDq?FdpE`gsAD07WIRu5|DI`nC<98n?l*)o<y*Vc`ZZbRYqg5$rofS; zM@Un8O;|ZYnIF#{&nDHYYI05*U-lFTOY@HFLq6Q+*b5Qevj!scJfrdn`xF+Y$mHU; z8CO_FZzselMv5lxD;`}YNnbGa$)(8w#Z33JWo_6;a}Lb99UvV`KFKh89((MXWjvow z=>B9hGvv>KNRSEcQM{rV#Dq_PoNx0e2_K1;i6{v6<J?T{PUokiA-e*N-gKQ$kfdfe zivd_y?e%}ZR}6M~#okVfS<UfjmNd&aoZ4^$KFHipt2r3Ar>H5s>+p3hE*2^>?{keX zQhkFy(#iFrkgpwv$F*Hb8&1#-F5J%Aez@{!YLF7SqPt)RECAwvL?qAdqg!#QV{<?+ zPchE7*fy8q{0^;Gp8Y3sXgkEp8Q>()OkCU3_JTn)&zdBq$!Uw{OPY<O=1xr;FJ@ve zJc<l7Ke1~DAWSbn&3yZIW__$RQ8ZRsmKTfBYzcD7&w?4UXrR__R{E>43nLdd0YCf@ z0_4<&i5DVKA+2gEb|pFpJ*a_?!c&0UNokTmt{~`VrnCLx$!vN7Z>tqut;2y>V)nDM zYC1*c93t0)c&yvWXQ!ryk``C(iH;q{H|7<lw3yZ>MvlPa?)eyVf&ug2k7|82nw_I3 zEsuAK(!4GPYiMSXwR?O^)yK=FZm^qWHCQ}T$Sp!9@vkq?{hnNm!FLM6sj@YdHHy3R z27+pq66E(~a7eL^BCMLD9sy{-a{`JMrqk8LE!;5?d9)9L?Q%L=5?|>GW?0RptGH{| z+F!<Px<7TJcyGqzT7QAp#QwrNKX}6w2;fQN2{=65r--5|+h7ogG=Vn(e}By`jILnl zi|?IJzrfE{QDLzjzO;XsDDS6IR&Y-RTa%hP+w0_1xedXA{nzZGEUto{Hq|5hxudNo z$@DueHgpNFutY{I$;_2*PoT`5Anc22#>Zeu^uqTh0pv>p0DnFhBr2zgiNyvmpt2y7 z1=Vi#KQd+ICKJC>rBh~zwm{5vYCBEDkP{ye%odeN#(lGbE_w>p#@xVVn$9U%G6|;z zsg})z><Q<60Obm68StzIoy!TR(&|O=wyw`ysu=jJQQo|xJ_VHm8NEg;T}xPbjn+kK z>;MnsY-PE+V59pkH;|5qI0W{5p!$K1s)caHtEElvwVtEb!<tmfdA$sCj&JjkZtn-S z_X8W=4+xK~8DQbGdJ7TDydf2mvF^xBAtfp{i-=Y-+CvS5!2`nqkAiHAOzTItgnJh! z75;nY<pLtOh^t<H)9EkGbeQv?&pLKzB1dN1>rCXR;k@zN8rfE*$di*2omVzBWq4Nj zJeTbtP)5uUN^9-QgzsamQ9Fc!qQG38o*a$KxB<U{d;So{s$LIa&<Mfz0*BX=v*O^u zua-So2c@|;7PTU!OIM;}pnP%I1JE8Cy2Nr`nEuYLm^6i%#AYx$L)uz&Qx*}=PmP>j zzjips%F{f+1okxn<k`&v<yndhHPq}y1|JIDV7(^7bU=k=&fsjq8bu*sBg`ZY2YBKk zGAg$0WrU5{o-%T_>S+u$Wk!<)<B^=dWx0Q=kg|x)$`%!E+Q>roHArlRwc8qEY#FUB zqjfb#%U~OLr7uLb9oD77Cbe&QI}2uKE75E>+IFLDu+gN20$D#<2Us?FA_lO3T8>s{ zqq(<z0%mPyrgSs~9$U%|^_*g4rY`*WB())<;bOi%BR>M03HW(^z8II?9ZD7!UH+ex zW-DM;je0RL{d+Z<o|heQaK{P|NC|H?tMCgPrrZK8^JZ;z??Wfi5B7d=kN(wYW~Lui zPOjVF9ji-e8HJ*lXvk=Ip&s=omFXd_W5hGtxy#l-i)BnvI6LVy14pZ5EILvS?2%54 zo!p7X8S0={ps*S(R^`%=g8c`8$0ud&Ux)Ky7HjN^Pdp~UhpzA`5TJNE#JsBM76p96 z0zw0-+?%^^swIc+lHmo-tBc^3dii--H$gTdOx6V3EMP&0#fSO@Iv&uQef{Qa^yUdM zp2K3G-&8aAX3sN`+rw>pxNQ%&D|wQBV>aRM-Jg!m=H(=GC;j}{-3N~!e*NO*mk)pW z@XP0h-tT|>ySHSlj4$Z$E6^|)p8SGRPfSPXN}!TKa9QnbJ|4|pjld{?O}Fr;4w&JK z;1?&x;!8_i%<B(T#s^1Tpx1l4lBX{i4o|*u^o70e#JiU#i_xSq+y80xrkp~DzQ45k zNFJMyDdlyGG*$4qUXc07(oGwz8693rTqd%NHb_T9b@gm?C|n>41yzi2vDi+;U>Atj zM(!>k24k<l!#}D&J3l)r7nf%5PR!=ziNzn;qNgR2A}&k6Y~h>q#oC^PW()|p$8~yM zh18??xsOb7ics<l5-msE#?j(@dFrl?4I^<69z+*vc2yXRtnfL%;{J$MI~A3VtQah` zC+Djt$D07GU4;nd!h-NPxvQ}^HhQ!;FLpoxb&TQ8=@qb9Y%S647!D%r0XfH^YQ`lr zu<scHp~P)kwD>in!EIhmQ6Lqx5PjTscdMj!bIx7S2|DasjNu-;N@9pVd=<GwVssWj zcYM8n80onT?=l`u$LCXETSj2&YK}M{pK_B3U{pltJiAyhR-hTc7|hKbh2v^5HU|-5 zE<QzAx|UhEJ`vx$+z=MJ6_*0wkIxLr?;^yWnVcKN!t8JOlhw)LO^aI9AATsVsdY>y zUq-irzw^y(JUWM^e(>b0hj{cOd;4f)cmX_A%QxlN9E?Q=<Q(W6NI9JXxR?N9hp0Ib zJu}~oq5BUyLu6-`D9>05V~*Xu{@+P6_96aX2COM=MA%oj2())-iD@kfJ%HMRUo|d_ z-RQyXyd4m1Em$Wp#_nDZOv>cU_Vo6MjaldRsBnPM&oS7ZNq=uw{Y%`u1S#IWP)wa# z!8sNdl}BH27>4&o)$y?}D=OY76nJ0j^(ionoWB9wf4TcfGGKQZ4U8S14PpH75WE$U z&j_nH;%w&&?bMH_MkwhLxa&nXWLh8>3Ai|ZbSpy8=fHrK1&HNmh&_g7q^t^nN9Ebv z43@(#iVy5BYAqKGNcI&?ZTM`WYCFZ9=+GJ_4xU>vVA!U!i^8>TP<c8nj!LsS^EqwL z1HU#sv1o@eFb~BD#NCmkMVK1hul}xB@RZwxT0-(&x5GKmYnF`f-MTg==6F{K`|k|F zVCZ!>Pcf#pbe*>de>=j=v`rz9qxrcYM~<W*S=vk5S}Drubr+H+g;cXlG7U31GV=Op z{GC3@&Ow}pbDqnAZ8MyEaG)+6pwH%u>eXl^xK7L3e*fg?+j6|R)KE2#%e<uH^BNUV z2;jcrkzUGY=D_R5PeUOi@T-H>pq?!aF}MN?Co6!7{Q}TKJQsa&?A9dn-tk+Lm^Vbm zPZn<A{s3#x4C)!!{n80GG5=|7F{3xKa=k!tv1T!Zy$SKSgn&ZmacBhN$}9`7q}CoG zP9#bJY;=lxRV^zJ)aM4pg1T@vgWiGK<gI4$j5|`z=zK51s?Dku6c+J-C6GS|CfUCS z_b9upiB*&X@*Tk4(R1A;Du9dy6#+odZ>x-r*%Z>Gz!yWbQ~k0r20OpKx3|{`*9rUN z=M{Q4Ess}L+m)J*qw=JhMZQDuX0#Z0Zy1Jh(F?E}@x*2?GFYrbW_rY`mt_(#1SnJw znH~wOT}zbQ2^!8;Olk!2up(M4jtXgQxiXa_suIcfDX~Arp)H>qR&!cDUs0r3UEu&5 zJ4Bk4enX*yB1KHs94#0+D3s>Lj35WJo#DSQ5286-0(LSbroq{jjLYJtio4jYkSF_? z(152h<OHIrfa8?TQY;?}b!%%xr~TR*(doXn#)q=TWrVR&{jKf7R_nagI$ufa>?ksb z)kY|KT9xDSb+s%>IzlTUxM^MW%(gSH;2IPR?WTrzDJM#A3}v6sM^>#x)hKw5`0Xf; z1Ba6Ayeq(qlv(gFbtxNAN3^Pc1$U(fF<Vbi`nC+k+EHDCXgs$23CHXZKs+W1K-U*i zp0lyCWZLTi-Qcx0fl?+cS28;Zz2Rx_GxaKtF#KY10mu*Wbsb+)kBQ>(UIgF<#|(aC zjwRQ+&?2UeCkj(b^CQP|P#wB?3yuMbSL@g0liQ#a0?8Ntm|tSP)2>fqW~zh)**@_m zb_MU4r>GjSo+damJ<L=<y<sXXA*QDbbQ~~clufltWKzB2ti6|+YcqSf84RW_KdFqO zGzaMQM+k(*y-Yy7!r9&1gcJ$HsA_gzpD$yQv{1JcS0F+QzD^KSql?el6A3Qgh9<aH z*KJ-d0IO*x&vnQzux&FtdDZgsvcAtI70#AJPn9#UP%+=!Q-&>^uAmT<q1!aD$T%tM zPH|LHz#A86)*SsuCFfkUE<iT1P2K<`Z}4&1CfreE^z$w;jVSQ@SEKop;}?bjBEuuG z5oCCl`-U4bOykT*`mUOam+qd%RMWmeKrg`iZxuy*S&7HV-Hf4YM4>QpKt9G}YLo^U z-18P&U)_0V6~;Q3;A_eDlf7M?E{r4(GTFn$qF!`6_PX}$d`Y;8T&SP_wn0zUWO#8! zWW05ezQQ}kh4SF0F_)#umDpg?2npoXWD!GDC!<9tlQf3*!IPOYM!IXZ5!caM!t4Z; zT^plLG(}9IIGjn!v!-*VPQqE()()a`B=H8juR%0AP`<x-DvjQ`EhCZ48={b;Nz}Wc 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%0DbDpKUt`9kWuQ=d~ zTW+g_0d{dYVQFav_u_&e_k$!8X6-`I_LB}j%_5WGMh@pQ;FgxrDd(f6AY%BO&GRC8 zPS%L&g|v_DHA%Z{sH=2UE)f&NukLKdj5mAHn{;AMZ#u-zuaP647^h=~!!oUGIyR~x z*DX|I$I7iJ8CRCk^e~%;xyu6D0U7SYts@6-%F5osdb=fljH|k<i<>Dfbv}K5cn+m3 z2Lp=>000mHU0y?4dEP6I_~4I6-68@2kim~y*qQ*{*v&yU)^*yGQR5%+>M>$mGl>@H z8FMF{Kcl<QQAK@5M<pyFX}jwoNFCj}jc)ceCClY$cH?b(ON4?o{I}0X6+WHdzFmV2 z#iDsLKl-&RWNh+vy|`Hr^*z-g*;0bV!1hDI>D2FODNd$Rku)UBiuK3i-(Rxo!@t<u zkwZJJM9iMdLmSe#^&UOuWj*U2FLX@1qgN|jd!E<!<Y4cbOIA-3H)zH&;IYE7*YX_F zGA)WYt;2I)o9YPJOxdmO0(v)H#xPY}vB3?)QDmZ|>ZymL@!o?^$X%Ma0`;|%jY6>; z0>Tk_ldR;35?kZMte#Y>WWFHTdr`>7)dM@IAooM!RziS{wrk*9HuM)3C|?*T!B@B{ z`?3bht(=qKpmN?1el5#8uF}JmQ$zN2D8%h%&U6ra;R~ddf;+7>J+*!GNhw4aV{qeH zMY*0eBIc%%PT{#%XPUQOe{T}WJASB+fE`4&Qdx_|EpquM<~+sviMBMZWnJ2mB5lBc z?OSV46<?}Ja^?-IjT7(n*Q#d__j^X(CUYRQ+OOG!k9LKRrqGIt-<e~oi1ZvJ=bFL% z+J4K*ooPZ(>9xSdk+!ZWo+XMDHPTvKV9rMjQhAZlkzD8TX1&3Y^sN}lA6gDclvIvv zf>nfG<%EQ)frE)dig37S8bwp8@q&&+qwc<_CbT(6g3ms1_R|}XV@hXN8ryfqNxdhN z>Bt|Jpj9noO@b4dV>M%@(ySS_P4ihk36v6vO^vKp^{X}#<ID5qUG2<sOanj{@hW!` z%lf5$l!VqK@4}Todi06Bh&M5CEUCaqj;?y}(Fe9n+iBZxo1Vfa3mM1y=fHzgrAQBI zJ`($GT*HSpk|H&?+<q_wa*MXg!uN+F%5cs_Z*T-S@U~=zd{>X3M9?G<KXeY7d%YK9 zu<mpG1%z6`ix!C>5LJ*H$V44}yPK8mX@Y^fj38SFPM)>d&>H+_1XrBeL}snxUPBA` z$DS3$4~DCvirIrIZr_u5Fs-AjTx_AmB3t})#Y16FG6dVc<Gn7l;&U6b^!u+0Rl>_; zw8tyuSFOjk2h$GvM2CW1MI(wm^;Q_<Tkc1dCT;us^`Zd*Npd-YiX&CX37LE`r2$B> zWFNcQ-?YA0-9)D(a=4d@UKa%gB7A@oI~EU<r&MnzP4{pwXJ(C9AN;@Bw6pIWh$l%O zeH3});pa#;+@-)Y-)D<K;icOlJJnrM4lS-Hs^%2ru?VPfqvjSIuw5MV=r>)+QZ_>g z=4Lf$-YB!Ia+>=VpVIXBGx_?aC!uI{h0n5-=W-LMi=m`#r9j!nU5m0}ZFh+-mp2Hr z`@?;IS6?_mn7L^3Z@SgF>d_7E0&Yi(TdULaLi@|ydAy%SQ&n3~2ouyuJS^2b@ljGN z?zruA%q01I_$)ekyn5D6WY*fV>xM-)M4tGX?2U>+_spIT#`%$bl%SmnK|ht~B+B$k zxzc|5Q>Urg!WYWbkJWhQze?TNrzwl%HMCik<q>CjD!zjM;82J*zrj>R%J9^n@TtTH z9bG_IYTJl<r<e0&Flf0Dx$I*;y)~8Bpl*~7nU^%?I^FBt%r8m11A@cEZz^dU=mRxQ z7Ake#4wt?1kYqXZTI&u}i9@Qsox9qz7y#aSQrU76korB<KXI{SuF(Lc-0BAgM8o(h z&@j+V*u};-S0<%9wKUQz44hqSDqQ#G4=_a0MW;j09XO`kkEUO`vS^dbJqMp#@Npnw zDyM`x?19G46lyw;K$c#W0qrW3w64~~r-NnbZGz8BeB}bY=N+S*)X-B={(M_QdZfvn z(~=>t@+b7WtEwnA^lJn&>Tu@Op9`OVyhk?K$0MK1E#0;aqmj1eoYxkH1mm<#y&LQ6 zlos{L^){03;j592Y4CeYCqs%SChGRPJkh(=WfFbv0SXpOjs}z(<W_oG$BiFsk92wo zLFK(8)Cogu%Rr#=;od4)hEnK-V~1+}CZh{O%h!epKC*@?UsEd-PJDZ3jdw3{!;Q*O z_tU-+9R15Du9);rPcr5_ne8{oSLHM9c-{a4n~{lDTsf(#S(qjl$q==a(&{`}3)v`! zusb@eSI%sd?J(|7W2@wkH_CAipok{TD=!seaAc9oT0TIYwcR+|j?347PYSj9Iibcf z3EM@!ZzMcZ5bF-BSk^j&y+0r+7`kmjA?8+hr$HYAgBN<t33*}@(!z6<moO)622l5> z8W_2-f)Tu<s!D`A@)y=Q4f~{U*gfvN>rbckqIfYIU>)MQ1e=r#$~%-OE^R71M<t&o zoH{Z*Xk(!a+HVMJH~{T$49_a@?{Da823M5lmJ65>`b%cK50yPC#wgJ!mgb#UA4%|V z##C&q4e_P%Bn{zDw34=JQ;)$colqaT{Zia;S2EjQ#0f*UvR0*U;{%EFcZ9xkfU+DE zG%n!h{s{o=TX$<+-1+qb0YLlf9_BAKwzILZv*kbrP+mL$1t0`u_y=u-)s8_m<wKN* z#QWm+|1~hmwSnotWBnS~*vaYI;F@=QF9vgsfT)Q0&A|}?ufm744(mRD(FU7H(ggtU zeh>chk5s%qIE2Fv=3Mn)t#Mavw|NZeQpMi=jVbjK)xz@N<4S=80CvEr>zFa{BUe0_ z0}V}pjyEyM($?-o>SIun9X=PX0ND`z6DmXipVP1w*JwO*B?<sQ$pQf2Tyv90d=29W z1ix+Mh8NgC>(QK+F{n$u4B;Qh--*7Sk4XC85DOc#YxxM~fh5_?5TgDYki7gE91-v{ zP>nGI%*$X0H0S^T!8He!*9SMTa|IurXK2E1Z)<inBV+TBP+2VoUWDpW|A78>f%_Zu zvs?$buep%7Z~;mhf()<O!AgdVu2&CHww-_5;W-LM_JjohCc(L9{!RH=Ua&m>rWl$S z+JnGIHd{LjC!oHasp(CJFKmA~4EMWm=xzehKMo<GL<l~0^Z3PPy2ftj|NpUV{|lX| z=zoE})SGCp(_a1~5&$5K6aXN*h7J|}HyY>yv<2P7RXv3`eLexfUFuC772Q{)J#c|X zfsr)VaP{D2>DTP~S&_eUd4bah&jJ(o|AH#eKe_ODsW+jMGkyY#FaUr%IJMV3?mGP& zdXdrp!g)N0;P9LwdK0%ps;#s93wPZkuG{~>{cn#sD-f7}^S_~0fszIr;1aqEf77F| z=f9!$hR#mF|MFOh8E_HOOYBXL@EOu}hG38A;E<AE%OlY5-#AC0v7O^zYd~Wyq;9?h zLU@;YlgCbUUq=5IkM<f5I`Dt-{vFgy>S`f`dZ{<5Dg_BxY+#3u;Gj}nqvnMCFRF#D z+0DRSA`b^4$V<J6bTWd+B?Tj0z(|T~NQ|ieLYg`Poy`9WXlV~w_cUZ6dK1{O+o6^R zE}T;^l=K?VJpSK6CqtLN>gLkkCGApg(wwqEftNJuYqa~x|4sXM7BBaY%N71oZ!$Tt zWw}3r!zmAbt>AhV+0(8uo$MVgKtMJpD+^mTTMJ_#8`sUCULr5|NIz7_9(fa4eb?%) zF}PgMz_;sJ6nXJKkvD<!$Tk=4#3k@j{{R7i;|GJ+S~}$4as9bRPQLsNw7lMUCBII> z{oXVN^@TF|LNB|#t7G467yq;>KgWj4zdE*|n}w5%t*PC$?YE~75}~~x5WQ?6|3DA{ zEXspg>wG)s>);a16}`Ej@ckQNY-j5PG6e5Lme<<6AK9%5>tIutdda`a``NwNE5eSh zf-RVXGjzRs(JQ)Y3XB0c+F4tJJG~nqE=Xp!$$~Fd%1gZpy6@4eMb<lHmyZen95Mg^ zSidtv2}XMrUj>;s8oJ&<^tNfhvDA-2UFxnR1oa2;cY}oe8}Tzi>a|x9Kx1=5N6_^m zz>kNF`Gzdy!Uml6O$MYzQGRl7cVQn1VCc<xW0?r}C*v3Eh9jeL`-^h6eGj2r(*HO@ z1WbAlX;E1);HrUpOBovgfaJHMpFztmN4{(mz~%glq6F^4z~yiQHUZ^IyA1C5FO^&p zV(}_WEMQnGU>5U4QwRWHfq?@23Exi`d*f9Y<YmU|Rgn0FvZVstiC^kVtG^jP%gw!; z@Ow8B6kvM_20(nxP{r|8hNXiu(9xaE*@E51!uEzAL=}gN;04TtUZm<$ulo7P3b@xQ z72pB@#HvBvDD7%E)GNV*u>Jv>bcKLkvxi-PF7+yCRr^3*9XO(H2aeMLxWl+^%F**G z$llQ0j?Kma1Y-N!6d=Jg@`QQ_>h%C*F}hrouCjibA}7*D_5#mUUGO4KcrAcLX;)cJ z7B=?Qz`Nl2dUIvCbj9KcadoLTNgT30VjxujAYKp(fPanDoq3I9XKV!ov0K>PNK;!n zBzmYj5YB!lME`L9p0$|2IX`DD==)U;$lMVKytbxxq(xrjB|$LiLN7^w2#`6e`F0_i zod5uUas~kCe-nNdcJ@CA;?Aa~Ku2kyoedD==>7m?VSRH#T{<Hd3%}rA>P@a|WkzKJ zxTW}n1-KDGTokk`1qg8_?|d^;NZ%k4^pA&dF7+k{G-BapiU$DrS%9BMU-LJC@i#}w z(Z~WM0a?y(I=dtq03$A(<%vM_Cb39(7JvfYr3}H(d9Uy5zI1;R!TS@)iQUG|#Mv5n zqew__9WUncApnv}__2Qj#gGt;)zzQ@dSXa7mfd#dzkwG4{l|n(35C#B*C4dZ0=jDN z?+Yum;6nNNMG3xXcw=0-%yRWvydL=2%SG@id#PnbkfKzUg9G<m001la+Zh`GNC?CL G|Mq|1Rn-{) literal 0 HcmV?d00001 diff --git a/public/static/threePublic/common/js/axisInfo/constant.js b/public/static/threePublic/common/js/axisInfo/constant.js index 07b5e7c..6308029 100644 --- a/public/static/threePublic/common/js/axisInfo/constant.js +++ b/public/static/threePublic/common/js/axisInfo/constant.js @@ -280,7 +280,7 @@ const createAxisInfo = (size, minSize = 0) => { h: 25, color: '#0000cc', font: 'normal 500px Arial,sans-serif', - xyz: [360, textY + 40, 340], + xyz: [360, textY, 355], }, { type: 'text', @@ -289,7 +289,7 @@ const createAxisInfo = (size, minSize = 0) => { h: 25, color: '#0000cc', font: 'normal 500px Arial,sans-serif', - xyz: [360, textY + 40, 270], + xyz: [360, textY, 270], }, { type: 'text', @@ -301,7 +301,7 @@ const createAxisInfo = (size, minSize = 0) => { color: '#0000cc', size: 22.3, font: 'normal 500px Arial,sans-serif', - xyz: [360, textY + 40, 180], + xyz: [360, textY, 180], }, { type: 'text', @@ -313,7 +313,7 @@ const createAxisInfo = (size, minSize = 0) => { color: '#0000cc', size: 22.3, font: 'normal 500px Arial,sans-serif', - xyz: [360, textY + 40, 90], + xyz: [360, textY, 90], }, { type: 'text', @@ -325,7 +325,7 @@ const createAxisInfo = (size, minSize = 0) => { color: '#0000cc', size: 22.3, font: 'normal 500px Arial,sans-serif', - xyz: [360, textY + 40, 0], + xyz: [360, textY, 0], }, ]; let prpd3d = [ diff --git a/public/static/threePublic/common/js/index.js b/public/static/threePublic/common/js/index.js index 0a0ea98..8c522c9 100644 --- a/public/static/threePublic/common/js/index.js +++ b/public/static/threePublic/common/js/index.js @@ -22,13 +22,7 @@ let token = 'c9343e83005e1f2b458d66628ddff98b' let url = `ws://192.168.1.88:8844/messaging/${token}` //const stats = new Stats(); //document.body.appendChild(stats.dom); -let box3D = document.getElementById("box3D2"); -let box2D = document.getElementById("box2D2"); -let box2D5 = document.getElementById('box2D5'); let box2D6 = document.getElementById('box2D6'); -let box3Dchart = ''; -let box2Dchart = ''; -let box2D5chart = ''; let box2D6chart = ''; let maxRange = 100; @@ -83,6 +77,16 @@ let initInstancedPoints,//prpd实例 //pointsGeometry.computeVertexNormals() let counter = 1; +const switchFun = { + showHistory: { + open() { + initTrendChart() + }, + close() { + + } + } +}; $('.switch').on('click', function () { $(this).toggleClass('active'); if ($(this).hasClass('active')) { @@ -92,6 +96,8 @@ $('.switch').on('click', function () { } }); + + $('#setting-btn').click(function () { $(this).toggleClass('ready') $('#setting').toggle() @@ -216,7 +222,9 @@ init(); init2D(); render(); render2D(); -initTrendChart() +initTrendChart(); +onWindowResize() + function initTrendChart() { box2D6chart = echarts.init(box2D6) setTimeout(() => { @@ -1084,9 +1092,9 @@ function handleVisibilityChange() { //prps初始化 function init() { let box = $('#box3D1'); - let width = window.innerWidth;//box.width(); - let height = window.innerHeight//box.height(); - console.log("🚀 height:",width, height) + let width = box.width(); + let height = box.height(); + renderer = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true, antialias: true, preserveDrawingBuffer: true, });//preserveDrawingBuffer保存three.js canvas画布上的信息 renderer.localClippingEnabled = true let constant = (360 / (initAxis)) * initAxis + 1 @@ -1184,15 +1192,17 @@ function onWindowResize() { $('#addPointNumDiv').show(); $('#box3D1').show(); $('#box2D1').hide(); + $('#box2D6').show(); } else if (viewType == 3) {//实时数据仅展示prpd let h = $(window).height(); - $(".loading-box").css('height', h * 0.9) + $(".loading-box").css('height', h) $('#realdata').hide(); $('.noisy-slide-box').hide(); $('#echarts-text').hide(); $('#addPointNumDiv').hide(); $('#box3D1').hide(); $('#box2D1').show(); + $('#box2D6').hide(); } else if (viewType == 4) {//历史趋势仅展示prps let h = $(window).height(); $(".loading-box").css('height', h * 0.9) @@ -1223,7 +1233,7 @@ function onWindowResize() { let width = box.width(); let height = box.height(); - console.log("🚀 height:",width, height) + //console.log("🚀 height:",width, height) let k = width / height; //窗口宽高比 let s = 350; //三维场景显示范围控制系数,系数越大,显示的范围越大 diff --git a/public/static/threePublic/index.html b/public/static/threePublic/index.html index ca64dd5..6c78f66 100644 --- a/public/static/threePublic/index.html +++ b/public/static/threePublic/index.html @@ -33,6 +33,7 @@ width: calc(100% - 40px); padding: 20px 10px; display: none; + backdrop-filter: blur(3px); } #setting::before { @@ -46,6 +47,7 @@ background-color: #1890ff; opacity: 0.2; z-index: -1; + /* 确保伪元素在子元素后面 */ } @@ -78,7 +80,7 @@ .slide { display: inline-block; - width: 160px; + width: 100px; height: 8px; margin-top: 5px; background-image: url('./common/img/sprite-skin-nice.png'); @@ -267,7 +269,7 @@ </div> </div> - <div id="box2D6" class="loading-box-mini" style="display: none;"></div> + <div id="box2D6" class="loading-box-mini"></div> <div id="chartDiv" class="col echarts-chart charts"> <div id="box3D1" class="loading-box"></div> <div id="box2D1" class="loading-box" style="display: none;"></div> diff --git a/src/api/location/associateConfig.ts b/src/api/location/associateConfig.ts new file mode 100644 index 0000000..47ff063 --- /dev/null +++ b/src/api/location/associateConfig.ts @@ -0,0 +1,19 @@ +import server from '@/utils/request'; + +//关联关系配置 + +//使用post分页动态查询 + +export const getAssociateConfigListPage_api = (data: object) => server.post(`/fault-config/_query`, data); + +// 新增数据 fault-config +export const addAssociateConfig_api = (data: object) => server.patch(`/fault-config`, data); + +//修改数据 +export const updateAssociateConfig_api = (id: object) => server.put(`/fault-config/${id}`); + +//根据ID查询 +export const getAssociateConfigList_api = (id: object) => server.get(`/fault/config/${id}`); + +//根据ID删除 +export const delteAssociateConfig_api = (id: string) => server.remove(`/fault/config/${id}`); \ No newline at end of file diff --git a/src/api/monitor.ts b/src/api/monitor.ts new file mode 100644 index 0000000..e57ecc8 --- /dev/null +++ b/src/api/monitor.ts @@ -0,0 +1,29 @@ +import server from '@/utils/request' + +/** + * (POST)一次性获取所有的节点树 + * @param logicId + * @returns + */ +export const findDefaultTree = (logicId: string) => server.get(`/common-point/findDefaultTree/${logicId}`) + +/** + * 保存站点监测点数据 + * @param data + * @returns + */ +export const addMonitor = (data: any) => server.post(`/common-point`, data) + +/** + * 修改站点监测点数据 + * @param id + * @returns + */ +export const editMonitor = (data: any) => server.put(`/common/point/${data.id}`, data) + +/** + * 删除站点监测点数据 + * @param id + * @returns + */ +export const deleteMonitor = (id: any) => server.remove(`/common/point/${id}`) \ No newline at end of file diff --git a/src/api/system/calendar.ts b/src/api/system/calendar.ts deleted file mode 100644 index 2b3b2cd..0000000 --- a/src/api/system/calendar.ts +++ /dev/null @@ -1,17 +0,0 @@ -import server from '@/utils/request'; -//编辑标签 -export const saveTag = (data:any) => server.patch('/calendar/tags',data) -//查询标签列表 -export const queryTags = () => server.get('/calendar/tags') -//删除标签 -export const deleteTags = (ids:any) => server.remove('/calendar/tags',{},{data:ids}) -//保存标签颜色 -export const saveTagsColor = (data:any) => server.post('/system/config/calendar-tag-color',data) -//查询标签颜色 -export const getTagsColor = () => server.get('/system/config/calendar-tag-color'); -//查询指定日期内的日历 -export const queryEvents = (dateFrom:any,dateTo:any) => server.get(`/calendar/${dateFrom}/${dateTo}`) -//批量保存指定日期的日历 -export const saveEvents = (data:any) => server.patch('/calendar',data) -//清空日历 -export const clearAll = () => server.remove('/calendar/mine/_all') \ No newline at end of file diff --git a/src/components/AMapComponent/AMap.vue b/src/components/AMapComponent/AMap.vue deleted file mode 100644 index 8ff1f27..0000000 --- a/src/components/AMapComponent/AMap.vue +++ /dev/null @@ -1,92 +0,0 @@ -<template> - <div - :style="props.style || { width: '100%', height: '100%' }" - :class="props.class" - > - <el-amap v-if="amapKey" :zooms="[3, 20]" @init="initMap" ref="mapRef"> - <template v-if="isOpenUi"> - <template v-if="uiLoading"> - <slot></slot> - </template> - </template> - <template v-else><slot></slot></template> - </el-amap> - <JEmpty v-else description="请配置高德地图key" style="padding: 20%" /> - </div> -</template> - -<script lang="ts" setup> -import { CSSProperties, PropType } from 'vue'; -import AMap, { initAMapApiLoader } from '@vuemap/vue-amap'; -import '@vuemap/vue-amap/dist/style.css'; -import { getAMapUiPromise } from './utils'; -import { useSystem } from '@/store/system'; - -const emit = defineEmits(['init']); - -const system = useSystem(); -interface AMapProps { - style?: CSSProperties; - class?: string; - AMapUI?: string | boolean; -} -const amapKey = system.$state.configInfo.amap?.apiKey; -const secretKey = system.$state.configInfo.amap?.secretKey; - -initAMapApiLoader({ - key: amapKey, - securityJsCode: secretKey, - plugins: ['AMap.DistrictSearch', 'AMap.GeoJSON'], -}); - -const props = defineProps({ - style: Object as PropType<AMapProps['style']>, - class: String as PropType<AMapProps['class']>, - AMapUI: [String, Boolean], - center: Array, -}); - -const mapRef = ref(); - -const uiLoading = ref<boolean>(false); - -let mapInstance:any = null; - -const isOpenUi = computed(() => { - return 'AMapUI' in props || props.AMapUI; -}); - -const getAMapUI = () => { - const version = typeof props.AMapUI === 'string' ? props.AMapUI : '1.1'; - getAMapUiPromise(version).then(() => { - uiLoading.value = true; - }); -}; - -const marker = ref<any[]>([]); - -const initMap = (e: any) => { - mapInstance = e; - if (isOpenUi.value) { - getAMapUI(); - } - emit('init', e); -}; - -const setBounds = (bounds: any) => { - console.log(bounds) - if (mapInstance) { - mapInstance.setBounds(bounds) - } -} - -onMounted(()=>{ - console.log(secretKey,'secretKey') -}) - -defineExpose({ - setBounds -}) -</script> - -<style lang="less" scoped></style> diff --git a/src/components/AMapComponent/DistrictSearch.vue b/src/components/AMapComponent/DistrictSearch.vue deleted file mode 100644 index d64d965..0000000 --- a/src/components/AMapComponent/DistrictSearch.vue +++ /dev/null @@ -1,116 +0,0 @@ -<template> - -</template> - -<script setup> -import { useMap } from './useMap' -import {pick} from "lodash-es"; - -defineOptions({ - name: 'DistrictSearch' -}) - -const instance = useMap() - -const props = defineProps({ - subdistrict: { - type: Number, - default: 0 - }, - extensions: { - type: String, - default: 'all' - }, - level: { - type: String, - default: 'district' - }, - view: { - type: Boolean, - default: true - }, - styles: { - type: Object, - default: () => ({}) - }, - adcode: { - type: String, - default: undefined - } -}) - -let district -let polygon - -const remove = () => { - if (polygon && instance.$amapComponent) { - if (instance.$amapComponent.getLayers().length) { - instance.$amapComponent.remove(polygon) - } - polygon = null - } -} - -const drawBounds = (paths) => { - if (polygon && instance.$amapComponent?.remove) { - instance.$amapComponent.remove(polygon) - polygon = null - } - for (var i = 0; i < paths.length; i += 1) {//构造MultiPolygon的path - paths[i] = [paths[i]] - } - - const _styles = Object.assign({ - strokeWeight: 1, - path: paths, - fillOpacity: 0.25, - fillColor: '#80d8ff', - strokeColor: '#0091ea' - },props.styles) - - polygon = new AMap.Polygon(_styles); - - instance.$amapComponent.add(polygon) - - if (props.view) { - instance.$amapComponent.setFitView(polygon) - } -} - -const queryDistrict = (code) => { - const opts = { - subdistrict: 0, //获取边界不需要返回下级行政区 - extensions: 'all', //返回行政区边界坐标组等具体信息 - level: 'district' //查询行政级别为 市 - } - - const options = Object.assign(opts, pick(props, ['subdistrict', 'extensions', 'level'])) - - if (!district) { - district = new AMap.DistrictSearch(options) - } - - if (!code) return - district.search(code, (status, result) => { - if (!result || !result.districtList || !result.districtList[0]) { - console.warn('请正确填写名称或更新其他名称'); - return - } - const bounds = result.districtList[0].boundaries; - drawBounds(bounds) - }) -} - -onBeforeUnmount(() => { - remove() -}) - -watch(() => props.adcode, () =>{ - queryDistrict(props.adcode) -}, { immediate: true }) - -</script> - -<style scoped> - -</style> diff --git a/src/components/AMapComponent/GeoJson.vue b/src/components/AMapComponent/GeoJson.vue deleted file mode 100644 index 9b61b97..0000000 --- a/src/components/AMapComponent/GeoJson.vue +++ /dev/null @@ -1,91 +0,0 @@ -<template> - -</template> - -<script setup> -import { useMap } from './useMap' -import { max, min } from 'lodash-es' - -defineOptions({ - name: 'GeoJson' -}) - -const props = defineProps({ - geo: { - type: Object, - default: undefined - }, - view: { - type: Boolean, - default: true - }, -}) - -const instance = useMap() - -let geoJsonLayer - -const remove = () => { - if (geoJsonLayer && instance.$amapComponent) { - if (instance.$amapComponent.getLayers().length) { - instance.$amapComponent.remove(geoJsonLayer) - } - geoJsonLayer = null - } -} -const drawBounds = () => { - remove() - - if (!props.geo) return - - geoJsonLayer = new AMap.GeoJSON({ - geoJSON: props.geo, - getPolygon: (geojson, lnglats) => { - return new AMap.Polygon({ - path: lnglats, - fillOpacity: 0.25,// 面积越大透明度越高 - strokeColor: '#0091ea', - fillColor: '#80d8ff' - }); - } - }) - - instance.$amapComponent.add(geoJsonLayer) - - if (props.view) { - - const points = props.geo.features.reduce((prev, next) => { - const coordinates = next.geometry.coordinates - prev.push(...coordinates[0]) - return prev - }, []) - - if (points.length) { - const lngArr = points.map(lnglat => lnglat[0]) - const latArr = points.map(lnglat => lnglat[1]) - - const maxLng = max(lngArr) - const maxLat = max(latArr) - const minLng = min(lngArr) - const minLat = min(latArr) - const southWest = new AMap.LngLat(maxLng, maxLat) - const northEast = new AMap.LngLat(minLng, minLat) - const bounds = new AMap.Bounds(southWest, northEast) - instance.$amapComponent.setBounds(bounds) - } - } -} - -onBeforeUnmount(() => { - remove() -}) - -watch(() => JSON.stringify(props.geo), () =>{ - drawBounds() -}, { immediate: true }) - -</script> - -<style scoped> - -</style> diff --git a/src/components/AMapComponent/index.ts b/src/components/AMapComponent/index.ts deleted file mode 100644 index d5ebf37..0000000 --- a/src/components/AMapComponent/index.ts +++ /dev/null @@ -1,7 +0,0 @@ -import AMap from './AMap.vue' - -export * from './useMap' -export { default as DistrictSearch } from './DistrictSearch.vue' -export { default as GeoJson } from './GeoJson.vue' - -export default AMap diff --git a/src/components/AMapComponent/useMap.ts b/src/components/AMapComponent/useMap.ts deleted file mode 100644 index 61a634c..0000000 --- a/src/components/AMapComponent/useMap.ts +++ /dev/null @@ -1,4 +0,0 @@ -export const useMap = () => { - return inject('parentInstance') -} - diff --git a/src/components/AlarmLeveIcon/index.vue b/src/components/AlarmLeveIcon/index.vue deleted file mode 100644 index c8b52ec..0000000 --- a/src/components/AlarmLeveIcon/index.vue +++ /dev/null @@ -1,27 +0,0 @@ -<template> - <img :src="src" /> -</template> - -<script setup> - -defineOptions({ - name: 'LevelIcon' -}) - -const props = defineProps({ - level: { - type: Number, - default: undefined - } -}) - -const src = computed(() => { - return `/images/alarm/alarm${props.level}.png` -}) - - -</script> - -<style scoped> - -</style> diff --git a/src/components/CardSelect/CardSelect.vue b/src/components/CardSelect/CardSelect.vue deleted file mode 100644 index 8fbf561..0000000 --- a/src/components/CardSelect/CardSelect.vue +++ /dev/null @@ -1,282 +0,0 @@ -<template> - <div :class="{ 'j-card-panel': true, 'no-column': noColumn }"> - <j-row v-if="!noColumn" :gutter="[16, 16]"> - <j-col - v-for="item in itemOptions" - :key="item.value" - :span="24 / column" - > - <div - :class="{ - 'j-card-item': true, - active: activeKeys.includes(item.value), - disabled: disabled || item.disabled, - horizontal: type === 'horizontal', - vertical: type === 'vertical', - right: float === 'right', - left: float === 'left', - }" - @click="() => handleSelect(item.value, item)" - > - <div class="j-card-title-warp"> - <div class="title"> - <slot - name="title" - :title="item.label" - :option="item" - > - <j-ellipsis> - {{ item.label }} - </j-ellipsis> - </slot> - </div> - <div - v-if="item.subLabel && showSubLabel" - class="sub-title" - > - <slot - name="subLabel" - :sub-label="item.subLabel" - :option="item" - > - {{ item.subLabel }} - </slot> - </div> - </div> - <div v-if="showImage" class="j-card-image"> - <slot name="image" :image="item.iconUrl" :option="item"> - <j-avatar - class="icon box-shadow" - :src="item.iconUrl" - /> - </slot> - </div> - </div> - </j-col> - </j-row> - <template v-else> - <div - v-for="item in itemOptions" - :key="item.value" - :class="{ - 'j-card-item': true, - active: activeKeys.includes(item.value), - disabled: disabled || item.disabled, - horizontal: type === 'horizontal', - vertical: type === 'vertical', - right: float === 'right', - left: float === 'left', - }" - @click="() => handleSelect(item.value, item)" - > - <div class="j-card-title-warp"> - <div class="title"> - <slot name="title" :title="item.label" :option="item"> - <j-ellipsis> - {{ item.label }} - </j-ellipsis> - </slot> - </div> - <div v-if="item.subLabel && showSubLabel" class="sub-title"> - <slot - name="subLabel" - :sub-label="item.subLabel" - :option="item" - > - {{ item.subLabel }} - </slot> - </div> - </div> - <div v-if="showImage" class="j-card-image"> - <slot name="image" :image="item.iconUrl" :option="item"> - <j-avatar class="icon box-shadow" :src="item.iconUrl" /> - </slot> - </div> - </div> - </template> - </div> -</template> - -<script lang="ts" setup> -import { computed, PropType, ref, toRefs, watch } from 'vue'; - -interface CardOption { - value: string | number; - label: string; - subLabel?: string; - iconUrl: string; - disabled?: boolean; -} - -const props = defineProps({ - type: { - type: String as PropType<'vertical' | 'horizontal'>, - default: 'horizontal', - }, - float: { - type: String as PropType<'left' | 'right'>, - default: 'left', - }, - options: { - type: Array as PropType<Array<CardOption>>, - default: () => [], - }, - disabled: { - type: Boolean, - default: false, - }, - multiple: { - type: Boolean, - default: false, - }, - column: { - type: Number, - default: 3, - }, - noColumn: { - type: Boolean, - default: false, - }, - showImage: { - type: Boolean, - default: true, - }, - showSubLabel: { - type: Boolean, - default: true, - }, - value: { - type: [String, Array], - default: undefined, - }, - allowClear: { - type: Boolean, - default: false, - }, -}); -const { multiple, type, disabled, float } = toRefs(props); - -const emits = defineEmits(['update:value', 'change', 'select']); -const activeKeys = ref<Array<string | number>>([]); -const itemOptions = computed(() => props.options); -const isAllowClear = computed(() => { - return props.allowClear !== false; -}); -const getOptions = (keys: Array<string | number>): CardOption[] => { - return itemOptions.value.filter((item) => { - return keys.includes(item.value); - }); -}; - -const handleSelect = (key: string | number, item: CardOption) => { - if (disabled.value || item.disabled) return; - let cloneActiveKeys = new Set(activeKeys.value); - - const isActive = cloneActiveKeys.has(key); - - // 已选中,并且单选,allowClear为false,则return - if (isActive && !multiple.value && isAllowClear.value === false) return; - - if (isActive) { - // 选中 - cloneActiveKeys.delete(key); - } else { - // 添加 - multiple.value - ? cloneActiveKeys.add(key) - : (cloneActiveKeys = new Set([key])); - } - - activeKeys.value = [...cloneActiveKeys.keys()]; - const options = multiple.value ? getOptions(activeKeys.value) : item; - - const values = props.multiple ? activeKeys.value : activeKeys.value[0] - - emits('update:value', values); - emits('change', values, options); - emits('select', values, key, !isActive) -}; - -watch( - () => props.value, - () => { - activeKeys.value = Array.isArray(props.value) - ? props.value - : [props.value]; - }, - { immediate: true }, -); -</script> - -<style lang="less" scoped> -@card-border: #e6e6e6; -.j-card-panel { - .j-card-item { - border: 1px solid @card-border; - border-radius: 4px; - cursor: pointer; - color: @black; - display: flex; - width: 100%; - gap: 12px; - - .j-card-title-warp { - flex: 1 1 auto; - max-width: 100%; - - .title { - word-break: keep-all; - width: 100%; - } - } - - &.vertical { - flex-direction: column-reverse; - padding: 22px 4px; - align-items: center; - - .j-card-image { - margin-bottom: 4px; - } - } - - &.horizontal { - padding: 20px; - } - - .sub-title { - color: rgba(0, 0, 0, 0.24); - } - - &.right { - flex-direction: row-reverse; - } - } - - &.no-column { - display: flex; - flex-wrap: wrap; - gap: 16px; - - .j-card-item { - min-width: 36px; - width: unset; - - &.vertical { - padding: 14px 16px; - } - } - } - - .active { - border: 1px solid var(--ant-primary-color) !important; - } - - .disabled { - cursor: not-allowed; - opacity: 0.75; - } -} - - -</style> diff --git a/src/components/CardSelect/RadioButton.vue b/src/components/CardSelect/RadioButton.vue deleted file mode 100644 index 48fb30f..0000000 --- a/src/components/CardSelect/RadioButton.vue +++ /dev/null @@ -1,72 +0,0 @@ -<template> - <div class="radio-button" :style="styles"> - <div v-for="item in options" @click="onClick(item)" class="radio-button-item" :class="{'active': myValue === item.value }"> - {{ item.label }} - </div> - </div> -</template> - -<script setup> -defineOptions({ - name: 'RadioButton', -}) - -const props = defineProps({ - value: { - type: [String, Number], - default: undefined - }, - options: { - type: Array, - default: () => [] - }, - columns: { - type: Number, - default: 3 - } -}) - -const emit = defineEmits(['update:value']) - -const myValue = ref(props.value) - -const styles = computed(() => { - return { - 'grid-template-columns': `repeat(${props.columns}, 1fr)` - } -}) - -const onClick = (record) => { - if (myValue.value !== record.value) { - myValue.value = record.value - emit('update:value', record.value) - emit('select', record.value) - } -} - -watch(() => props.value, () => { - myValue.value = props.value -}) - -</script> - -<style scoped lang="less"> -.radio-button { - display: grid; - gap: 16px; - - .radio-button-item { - padding: 6px 12px; - text-align: center; - height: 100%; - border-radius: 2px; - background-color: #f5f5f5; - cursor: pointer; - - &.active { - color: #fff; - background-color: @primary-color; - } - } -} -</style> diff --git a/src/components/CardSelect/index.ts b/src/components/CardSelect/index.ts deleted file mode 100644 index a3abafb..0000000 --- a/src/components/CardSelect/index.ts +++ /dev/null @@ -1,11 +0,0 @@ -import CardSelect from './CardSelect.vue'; -import type { App } from 'vue'; - -CardSelect.name = 'JCardSelect'; - -CardSelect.install = function (app: App) { - app.component('JCardSelect', CardSelect); - return app; -}; - -export default CardSelect; diff --git a/src/components/CheckButton/CheckButton.vue b/src/components/CheckButton/CheckButton.vue deleted file mode 100644 index 3746f45..0000000 --- a/src/components/CheckButton/CheckButton.vue +++ /dev/null @@ -1,163 +0,0 @@ -<template> - <div :class="['j-check-button', props.class]" :style="styles"> - <div - v-for="item in _options" - :key="item.value" - :class="{ - 'j-check-button-item': true, - 'selected': myValue.includes(item.value), - 'disabled': item.disabled - }" - @click=" - () => { - selected(item.value, item.disabled); - } - " - > - {{ item.label }} - </div> - </div> -</template> - -<script setup lang="ts"> -import { computed, CSSProperties, PropType, ref, watch } from 'vue'; -import { isArray } from 'lodash-es'; -import { Form } from 'ant-design-vue' - -defineOptions({ - name: 'CheckButton' -}) - -const props = defineProps({ - value: { - type: [String, Array], - default: undefined, - }, - options: { - type: Array, - default: () => [], - }, - multiple: { - type: Boolean, - default: false, - }, - disabled: { - type: Boolean, - default: false, - }, - class: { - type: String, - default: undefined, - }, - style: { - type: Object as PropType<CSSProperties>, - default: () => ({}), - }, - columns: { - type: Number, - default: 3 - } -}); -const emit = defineEmits(['update:value', 'change', 'select']); - -const formItemContext = Form.useInjectFormItemContext(); -const myValue = ref(); -const optionsMap = ref(new Map()); - -const styles = computed(() => { - return { - 'grid-template-columns': `repeat(${props.columns}, 1fr)`, - ...props.style - } -}) - -const _options = computed(() => { - props.options.forEach((item: any) => { - if (props.disabled) { - item.disabled = props.disabled - } - optionsMap.value.set(item.value, item); - }); - return props.options; -}); - -const selected = (key: string | number, disabeld: boolean) => { - if (disabeld || props.disabled) return; - - const values = new Set(myValue.value); - - if (values.has(key)) { - values.delete(key); - } else { - if (!props.multiple) { - values.clear(); - } - values.add(key); - } - - myValue.value = [...values.values()]; - - const optionsItems = myValue.value.map((_key) => { - return optionsMap.value.get(_key); - }); - - const _value = props.multiple ? myValue.value : myValue.value[0]; - - emit('update:value', _value); - emit('change', _value, props.multiple ? optionsItems : optionsItems[0]); - emit('select', _value, props.multiple ? optionsItems : optionsItems[0]); - formItemContext.onFieldChange() -}; - -watch( - () => props.value, - () => { - if (props.value) { - myValue.value = isArray(props.value) ? props.value : [props.value]; - } else { - myValue.value = []; - } - }, - { immediate: true, deep: true }, -); - -</script> - -<style scoped lang="less"> -.j-check-button { - display: grid; - gap: 16px; - width: 100%; - - .j-check-button-item { - flex: 1; - min-width: 0; - padding: 8px; - border-radius: @border-radius-base; - background-color: #f2f3f5; - transition: all 0.3s; - color: #333; - text-align: center; - cursor: pointer; - - &:hover { - background-color: @primary-color; - opacity: 0.85; - color: #fff; - } - - &.selected { - background-color: @primary-color; - color: #fff; - } - - &.disabled { - cursor: not-allowed; - color: #00000040; - background-color: #e6e6e6; - opacity: 1 - } - } -} - -</style> diff --git a/src/components/CheckButton/index.ts b/src/components/CheckButton/index.ts deleted file mode 100644 index ea647f5..0000000 --- a/src/components/CheckButton/index.ts +++ /dev/null @@ -1,3 +0,0 @@ -import CheckButton from "./CheckButton.vue"; - -export default CheckButton diff --git a/src/components/ConfirmModal/index.vue b/src/components/ConfirmModal/index.vue deleted file mode 100644 index 7d58d94..0000000 --- a/src/components/ConfirmModal/index.vue +++ /dev/null @@ -1,76 +0,0 @@ -<template> - <!-- <a-tooltip v-if="toolTip" v-bind="toolTip"> - <span @click="showConfirm" :class="props.className" v-show="show"> - {{ props.class }} - <slot></slot> - </span> - </a-tooltip> --> - <span @click="showConfirm" :class="props.className" v-show="show"> - {{ props.class }} - <slot></slot> - </span> -</template> - -<script setup> -import { Modal } from 'ant-design-vue'; -const props = defineProps({ - title: { - type: String, - default: '', - }, - onConfirm: { - type: Object, - default: {}, - }, - className: { - type: String, - }, - show: { - type: Boolean, - default: true, - }, - disabled: { - type: Boolean, - default: false, - }, - toolTip: { - type: Object - }, -}); -// const confirmLoading = ref(false); -// const modalVisible = ref(false); -// const modalConfirm = async() => { -// if (typeof props.onConfirm === 'function') { -// confirmLoading.value = true; -// const res = await props.onConfirm()?.finally(()=>{ -// confirmLoading.value = false; -// modalVisible.value = false; -// return -// }); -// if(!res?.finally){ -// confirmLoading.value = false; -// modalVisible.value = false; -// } -// } else { -// modalVisible.value = false; -// } -// }; -const showConfirm = () => { - if (props.disabled) { - return; - } - Modal.confirm({ - title: props.title, - content: props?.content, - onOk() { - return props?.onConfirm(); - }, - onCancel() {}, - }); -}; -</script> -<style lang="less" scoped> -.modalContent { - text-align: center; -} -</style> diff --git a/src/components/Metadata/index.ts b/src/components/Metadata/index.ts deleted file mode 100644 index e69de29..0000000 diff --git a/src/components/Modal/DragModal.vue b/src/components/Modal/DragModal.vue deleted file mode 100644 index 6cb35f8..0000000 --- a/src/components/Modal/DragModal.vue +++ /dev/null @@ -1,326 +0,0 @@ -<template> - <div ref='dialog' :style='styleName' class='dialog'> - <Transition name='dialog'> - <div class='dialog-sprite' ref='header'> - <div class='header' v-if="title !== false"> - <span>{{ title }}</span> - <a-button size='small' type='text' @click.stop='onClose'> - <AIcon type='CloseOutlined' /> - </a-button> - </div> - <div class='dialog-body' :style="bodyStyle"> - <slot></slot> - </div> -<!-- <div class='dialog-footer' v-if='slots?.footer'>--> -<!-- <slot name='footer'></slot>--> -<!-- </div>--> - </div> - </Transition> -<!-- <template v-for='item in rangeList'>--> -<!-- <div :class="{'range': true, [item.classname]: true}" @mousedown.stop='rangeMove($event,item.classname)'></div>--> -<!-- </template>--> - - <div :class="{'range': true, 'bottom-right': true}" @mousedown.stop='rangeMove($event,"bottom-right")'></div> - </div> -</template> - -<script setup lang='ts'> - -defineOptions({ - name: 'DragModal' -}) - -const props = defineProps({ - title: { - type: [String, Boolean], - default: '' - }, - width: { - type: Number, - default: 400 - }, - height: { - type: Number, - default: 100 - }, - dragRang: { - type: [Array , Number], - default: [400, 200] - }, - bodyStyle: { - type: Object, - default: () => ({}) - } -}) -const emits = defineEmits(['close', 'heightChange']) -const slots = useSlots() - -const ele = document.body - -const dialog = ref() -const header = ref() -const baseWidth = ref(props.width || 400) -const baseHeight = ref(props.height || 100) -const baseLeft = ref(100) -const baseTop = ref(100) - -const rangeList = [ - // { - // classname: 'top-left' - // }, - // { - // classname: 'top-right' - // }, - // { - // classname: 'bottom-left' - // }, - { - classname: 'bottom-right' - } -] - -const styleName = computed(() => { - return { - top: getFixed(baseTop.value) + 'px', - left: getFixed(baseLeft.value) + 'px', - width: getFixed(baseWidth.value) + 'px', - height: getFixed(baseHeight.value) + 'px' - } -}) - -const getFixed = (val: number) => { - return Number(val.toFixed(2)) -} - -const onDrag = () => { - let active = false - let initialX: number - let initialY: number - let initialWindowX: number - let initialWindowY: number - - header.value.addEventListener('mousedown', (e: MouseEvent) => { - active = true - - initialX = e.clientX - initialY = e.clientY - - initialWindowX = dialog.value.offsetLeft - initialWindowY = dialog.value.offsetTop - }) - - document.addEventListener('mouseup', () => { - active = false - }) - - document.addEventListener('mousemove', (e) => { - if (active) { - const dx = e.clientX - initialX - const dy = e.clientY - initialY - - baseLeft.value = initialWindowX + dx - baseTop.value = initialWindowY + dy - } - }) -} - -const handleClear = () => { - document.onmouseup = () => { - document.onmousemove = null - document.onmouseup = null - } -} - -const rangeMove = (e: MouseEvent, position: string) => { - //移动的方向 - let x: boolean = false - let y: boolean = false - //移动的位置 - let xp: boolean = false - let yp: boolean = false - //移动的正负 - let xc: boolean = false - let yc: boolean = false - let disX = e.clientX - let disY = e.clientY - document.onmousemove = e => { - if (position === 'bottom-right') { - x = true - y = true - } else if (position === 'bottom-left') { - x = true - y = true - xp = true - xc = true - } else if (position === 'top-right') { - x = true - y = true - yp = true - yc = true - } else if (position === 'top-left') { - x = true - y = true - xp = true - xc = true - yp = true - yc = true - } - let left = e.clientX - disX - let top = e.clientY - disY - disX = e.clientX - disY = e.clientY - if (x) { - let calc = left - - if (xc) { - calc = -calc - } - - if (xp) { - baseLeft.value = baseLeft.value - calc - } - - const width = baseWidth.value + calc - - baseWidth.value = width <= props.dragRang[0] ? props.dragRang[0] : width - } - if (y) { - let calc = top - if (yc) { - calc = -calc - } - - if (yp) { - baseTop.value = baseTop.value - calc - } - - const height = baseHeight.value + calc - - baseHeight.value = height <= props.dragRang[1] ? props.dragRang[1] : height - - emits('heightChange', baseHeight.value) - } - } - handleClear() -} - -const onClose = () => { - console.log('close---1') - emits("close") -} - -onMounted(() => { - if (dialog.value && header.value) { - onDrag() - } - if (ele) { - const data = ele?.getBoundingClientRect() - baseLeft.value = (data?.right - baseWidth.value) / 2 || 0 - baseTop.value = data?.top + 200 || 0 - } -}) - -watch(() => props.height, () => { - if (props.height > baseHeight.value) { - baseHeight.value = props.height - } -}) -</script> - -<style lang='less' scoped> -@boxColor: rgb(@primary-color); - -// 弹窗动画 -.dialog-enter-active, -.dialog-leave-active { - transition: opacity .5s; -} - -.dialog-enter, -.dialog-leave-to { - opacity: 0; -} - -.dialog { - position: fixed; - z-index: 1000; - - .dialog-sprite { - position: absolute; - width: 100%; - height: 100%; - display: flex; - flex-direction: column; - overflow: hidden; - z-index: 23456765435; - background-color: #ffffff; - border-radius: 4px; - border: 1px solid #91CAFF; - box-shadow: 0 3px 8px 0 rgba(#1677FF, 0.24); - - .header { - padding: 5px 15px; - font-size: 18px; - font-weight: 700; - color: #333; - display: flex; - justify-content: space-between; - align-items: center; - border-bottom: 1px solid #f0f0f0; - cursor: move; - } - - .dialog-body { - flex: 1; - min-height: 0; - overflow-y: auto; - padding: 24px 20px; - } - - .dialog-footer { - border-top: 1px solid #f0f0f0; - padding: 5px 15px; - } - } - - .range { - position: absolute; - width: 16px; - height: 16px; - border-radius: 100%; - z-index: 23456765436; - } - - .bottom-right, .top-left { - &:hover { - cursor: nwse-resize - } - } - - .bottom-left, .top-right { - &:hover { - cursor: nesw-resize - } - } - - .top-right { - top: -6px; - right: -6px; - } - - .top-left { - top: -6px; - left: -6px; - } - - .bottom-right { - bottom: -6px; - right: -6px; - } - - .bottom-left { - bottom: -6px; - left: -6px; - } -} -</style> diff --git a/src/components/Modal/index.ts b/src/components/Modal/index.ts deleted file mode 100644 index f5d8c54..0000000 --- a/src/components/Modal/index.ts +++ /dev/null @@ -1 +0,0 @@ -export { default as DragModal } from './DragModal.vue' diff --git a/src/components/Player/utils.ts b/src/components/Player/utils.ts deleted file mode 100644 index 44655ad..0000000 --- a/src/components/Player/utils.ts +++ /dev/null @@ -1,13 +0,0 @@ -import FlvPlugin from 'xgplayer-flv' -import HlsPlugin from "xgplayer-hls" -export const settingEnum = { - mp4: { - isLive: false, - }, - flv: { - plugins: [FlvPlugin], - }, - m3u8: { - plugins: [HlsPlugin], - } -} diff --git a/src/hook/useAlarmLevel.ts b/src/hook/useAlarmLevel.ts deleted file mode 100644 index 5684cb8..0000000 --- a/src/hook/useAlarmLevel.ts +++ /dev/null @@ -1,30 +0,0 @@ -import { useRequest } from './useRequest' -import { queryLevel } from '@/api/rule-engine/config'; - -type LevelListType = Array<{ label: number, value: string}> -export const useAlarmLevel = () => { - const levelMap = ref({}) - const levelList = ref<LevelListType>([]) - - useRequest(queryLevel, { - onSuccess(res) { - if (res.result?.levels?.length) { - const arr: LevelListType = [] - levelMap.value = res.result.levels.reduce((prev: Record<string, string>, next: Record<string, any>) => { - prev[next.level] = next.title - arr.push({ - label: next.title, - value: next.level - }) - return prev - }, {}) - levelList.value = arr - } - } - }) - - return { - levelMap, - levelList - } -} diff --git a/src/i18n/index.ts b/src/i18n/index.ts deleted file mode 100644 index 6ea7690..0000000 --- a/src/i18n/index.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { createI18n } from 'vue-i18n' -import zh from './locale/zh.json' -import en from './locale/en.json' - -const messages = { - en, - zh, -} -const language = (navigator.language || 'en').toLocaleLowerCase() // 这是获取浏览器的语言 -const i18n = createI18n({ - locale: localStorage.getItem('lang') || language.split('-')[0] || 'en', // 首先从缓存里拿,没有的话就用浏览器语言, - fallbackLocale: 'en', // 设置备用语言 - messages, - legacy: false -}) - -export default i18n diff --git a/src/i18n/locale/en.json b/src/i18n/locale/en.json deleted file mode 100644 index 5107d2c..0000000 --- a/src/i18n/locale/en.json +++ /dev/null @@ -1,1773 +0,0 @@ -{ - "Login.index.265048-0": "keep on record:", - "Login.index.265048-1": "YuICPBei No. 19017719-1", - "Login.index.265048-2": "account number", - "Login.index.265048-3": "Please enter your account number", - "Login.index.265048-4": "password", - "Login.index.265048-5": "Please input a password", - "Login.index.265048-6": "Verification Code", - "Login.index.265048-7": "Please enter the verification code", - "Login.index.265048-8": "Remember me", - "Login.index.265048-9": "Sign in", - "Login.index.265048-10": "Other login methods", - "Login.index.265048-11": "JETLINKS team's new visual large screen system", - "Login.index.265048-12": "Experience DEMO", - "Login.index.265048-13": "Please enter your account!", - "Login.index.265048-14": "Please input a password!", - "Login.index.265048-15": "Please enter the verification code!", - "BooleanParam.index.966482-0": "", - "BooleanParam.index.966482-1": "", - "BooleanParam.index.966482-2": "", - "BooleanParam.index.966482-3": "", - "BooleanParam.index.966482-4": "", - "BooleanParam.index.966482-5": "", - "BooleanParam.index.966482-6": "", - "Save.GateWayFormItem.039083-0": "", - "Save.GateWayFormItem.039083-1": "", - "DevOpsHome.index.152181-0": "", - "DevOpsHome.index.152181-1": "", - "DevOpsHome.index.152181-2": "", - "DevOpsHome.index.152181-3": "", - "DevOpsHome.index.152181-4": "", - "DevOpsHome.index.152181-5": "", - "DevOpsHome.index.152181-6": "", - "DevOpsHome.index.152181-7": "", - "DevOpsHome.index.152181-8": "", - "DevOpsHome.index.152181-9": "", - "DevOpsHome.index.152181-10": "", - "DevOpsHome.index.152181-11": "", - "DevOpsHome.index.152181-12": "", - "DevOpsHome.index.152181-13": "", - "DevOpsHome.index.152181-14": "", - "DevOpsHome.index.152181-15": "", - "Task2.index.551010-0": "", - "Task2.index.551010-1": "", - "Task2.index.551010-2": "", - "Task2.index.551010-3": "", - "Task2.index.551010-4": "", - "Task2.index.551010-5": "", - "Task2.index.551010-6": "", - "Task2.index.551010-7": "", - "Task2.index.551010-8": "", - "Task2.index.551010-9": "", - "Task2.index.551010-10": "", - "Detail.index.551010-0": "", - "Detail.index.551010-1": "", - "Detail.index.551010-2": "", - "Detail.index.551010-3": "", - "Detail.index.551010-4": "", - "Detail.index.551010-5": "", - "Detail.index.551010-6": "", - "Detail.index.551010-7": "", - "Detail.index.551010-8": "", - "Detail.index.551010-9": "", - "Detail.index.551010-10": "", - "Detail.index.551010-11": "", - "Detail.index.551010-12": "", - "Detail.index.551010-13": "", - "Detail.index.551010-14": "", - "Detail.index.551010-15": "", - "Task.index.551010-0": "", - "Task.index.551010-1": "", - "Task.index.551010-2": "", - "Task.index.551010-3": "", - "Task.index.551010-4": "", - "Task.index.551010-5": "", - "Task.index.551010-6": "", - "Task.index.551010-7": "", - "Task.index.551010-8": "", - "Task.index.551010-9": "", - "Task.index.551010-10": "", - "Task.index.551010-11": "", - "Task.index.551010-12": "", - "Save.index.551010-0": "", - "Save.index.551010-1": "", - "Save.index.551010-2": "", - "Save.index.551010-3": "", - "Save.index.551010-4": "", - "Save.index.551010-5": "", - "Save.index.551010-6": "", - "Save.index.551010-7": "", - "Save.index.551010-8": "", - "Save.index.551010-9": "", - "Save.index.551010-10": "", - "Save.index.551010-11": "", - "Save.index.551010-12": "", - "Save.index.551010-13": "", - "Save.index.551010-14": "", - "Save.index.551010-15": "", - "Save.index.551010-16": "", - "Save.index.551010-17": "", - "Save.index.551010-18": "", - "Save.index.551010-19": "", - "Save.index.551010-20": "", - "Save.index.551010-21": "", - "Save.index.551010-22": "", - "Save.index.551010-23": "", - "Save.FileUpload.551010-0": "", - "Save.FileUpload.551010-1": "", - "Save.FileUpload.551010-2": "", - "Save.FileUpload.551010-3": "", - "Save.FileUpload.551010-4": "", - "Save.index.551010-24": "", - "Save.index.551010-25": "", - "Save.index.551010-26": "", - "Save.index.551010-27": "", - "Save.index.551010-28": "", - "Save.index.551010-29": "", - "Save.index.551010-30": "", - "Save.index.551010-31": "", - "Save.index.551010-32": "", - "Save.index.551010-33": "", - "Save.index.551009-0": "", - "Save.index.551009-1": "", - "Save.index.551009-2": "", - "Save.index.551009-3": "", - "Save.index.551009-4": "", - "Save.index.551009-5": "", - "Save.index.551009-6": "", - "Save.index.551009-7": "", - "Save.index.551009-8": "", - "Save.index.551009-9": "", - "Save.index.551009-10": "", - "Save.index.551009-11": "", - "Save.index.551009-12": "", - "Save.index.551009-13": "", - "Save.index.551009-14": "", - "Save.index.551009-15": "", - "Save.index.551009-16": "", - "Save.index.551009-17": "", - "Save.index.551009-18": "", - "Save.index.551009-19": "", - "Save.index.551009-20": "", - "Save.index.551009-21": "", - "Save.index.551009-22": "", - "Save.index.551009-23": "", - "Save.SelectDevices.551010-0": "", - "Save.SelectDevices.551010-1": "", - "Save.SelectDevices.551010-2": "", - "Save.SelectDevices.551010-3": "", - "Save.SelectDevices.551010-4": "", - "Save.SelectDevices.551010-5": "", - "Save.SelectDevices.551010-6": "", - "Save.SelectDevices.551010-7": "", - "Save.SelectDevices.551010-8": "", - "Save.SelectDevices.551010-9": "", - "Save.SelectDevices.551010-10": "", - "Save.SelectDevices.551010-11": "", - "Detail.PropertyAMap.551011-0": "", - "Detail.PropertyAMap.551011-1": "", - "Detail.PropertyAMap.551011-2": "", - "Instance.index.551009-0": "", - "Instance.index.551009-1": "", - "Instance.index.551009-2": "", - "Instance.index.551009-3": "", - "Instance.index.551009-4": "", - "Instance.index.551009-5": "", - "Instance.index.551009-6": "", - "Instance.index.551009-7": "", - "Instance.index.551009-8": "", - "Instance.index.551009-9": "", - "Instance.index.551009-10": "", - "Instance.index.551009-11": "", - "Instance.index.551009-12": "", - "Instance.index.551009-13": "", - "Instance.index.551009-14": "", - "Instance.index.551009-15": "", - "Instance.index.551009-16": "", - "Instance.index.551009-17": "", - "Instance.index.551009-18": "", - "Instance.index.551009-19": "", - "Instance.index.551009-20": "", - "Instance.index.551009-21": "", - "Instance.index.551009-22": "", - "Instance.index.551009-23": "", - "Instance.index.551009-24": "", - "Instance.index.551009-25": "", - "Instance.index.551009-26": "", - "Instance.index.551009-27": "", - "Instance.index.551009-28": "", - "Instance.index.551009-29": "", - "Instance.index.551009-30": "", - "Instance.index.551009-31": "", - "Instance.index.551009-32": "", - "Instance.index.551009-33": "", - "Instance.index.551009-34": "", - "Instance.index.551009-35": "", - "Instance.index.551009-36": "", - "Instance.index.551009-37": "", - "DeviceAccess.index.551011-0": "", - "DeviceAccess.index.551011-1": "", - "DeviceAccess.index.551011-2": "", - "DeviceAccess.index.551011-3": "", - "DeviceAccess.index.551011-4": "", - "DeviceAccess.index.551011-5": "", - "DeviceAccess.index.551011-6": "", - "DeviceAccess.index.551011-7": "", - "DeviceAccess.index.551011-8": "", - "DeviceAccess.index.551011-9": "", - "DeviceAccess.index.551011-10": "", - "DeviceAccess.index.551011-11": "", - "DeviceAccess.index.551011-12": "", - "DeviceAccess.index.551011-13": "", - "DeviceAccess.index.551011-14": "", - "DeviceAccess.index.551011-15": "", - "DeviceAccess.index.551011-16": "", - "DeviceAccess.index.551011-17": "", - "DeviceAccess.index.551011-18": "", - "DeviceAccess.index.551011-19": "", - "DeviceAccess.index.551011-20": "", - "DeviceAccess.index.551011-21": "", - "DeviceAccess.index.551011-22": "", - "DeviceAccess.index.551011-23": "", - "DeviceAccess.index.551011-24": "", - "DeviceAccess.index.551011-25": "", - "DeviceAccess.index.551011-26": "", - "DeviceAccess.index.551011-27": "", - "DeviceAccess.index.551011-28": "", - "DeviceAccess.index.551011-29": "", - "DeviceAccess.index.551011-30": "", - "DeviceAccess.index.551011-31": "", - "DeviceAccess.index.551011-32": "", - "Firmware.index.551009-0": "", - "Firmware.index.551009-1": "", - "Firmware.index.551009-2": "", - "Firmware.index.551009-3": "", - "Firmware.index.551009-4": "", - "Firmware.index.551009-5": "", - "Firmware.index.551009-6": "", - "Firmware.index.551009-7": "", - "Firmware.index.551009-8": "", - "Firmware.index.551009-9": "", - "Firmware.index.551009-10": "", - "Firmware.index.551009-11": "", - "Firmware.index.551009-12": "", - "Firmware.index.551009-13": "", - "Firmware.index.551009-14": "", - "Property.Indicators.551010-0": "", - "Property.Indicators.551010-1": "", - "Property.Indicators.551010-2": "", - "Property.Indicators.551010-3": "", - "Property.Indicators.551010-4": "", - "Property.Indicators.551010-5": "", - "Detail.TimeComponent.551010-0": "", - "Detail.TimeComponent.551010-1": "", - "Detail.TimeComponent.551010-2": "", - "Detail.TimeComponent.551010-3": "", - "Detail.TimeComponent.551010-4": "", - "Event.index.551010-0": "", - "Event.index.551010-1": "", - "Event.index.551010-2": "", - "Event.index.551010-3": "", - "Event.index.551010-4": "", - "Detail.Table.551010-0": "", - "Detail.Table.551010-1": "", - "Detail.Table.551010-2": "", - "Detail.Table.551010-3": "", - "Detail.Table.551010-4": "", - "Detail.index.551011-0": "", - "Detail.index.551011-1": "", - "Detail.index.551011-2": "", - "Detail.index.551011-3": "", - "DeviceAccess.accessModal.551011-0": "", - "DeviceAccess.accessModal.551011-1": "", - "DeviceAccess.accessModal.551011-2": "", - "DeviceAccess.accessModal.551011-3": "", - "DeviceAccess.accessModal.551011-4": "", - "DeviceAccess.accessModal.551011-5": "", - "DeviceAccess.accessModal.551011-6": "", - "DeviceAccess.accessModal.551011-7": "", - "DeviceAccess.accessModal.551011-8": "", - "DeviceAccess.accessModal.551011-9": "", - "DeviceAccess.accessModal.551011-10": "", - "DeviceAccess.accessModal.551011-11": "", - "components.TagSearch.551010-0": "", - "components.TagSearch.551010-1": "", - "components.TagSearch.551010-2": "", - "components.TagSearch.551010-3": "", - "components.TagSearch.551010-4": "", - "device.data.551010-0": "", - "device.data.551010-1": "", - "device.data.551010-2": "", - "device.data.551010-3": "", - "device.data.551010-4": "", - "device.data.551010-5": "", - "device.data.551010-6": "", - "device.data.551010-7": "", - "device.data.551010-8": "", - "device.data.551010-9": "", - "device.data.551010-10": "", - "device.data.551010-11": "", - "device.data.551010-12": "", - "device.data.551010-13": "", - "device.data.551010-14": "", - "device.data.551010-15": "", - "device.data.551010-16": "", - "device.data.551010-17": "", - "device.data.551010-18": "", - "device.data.551010-19": "", - "device.data.551010-20": "", - "device.data.551010-21": "", - "device.data.551010-22": "", - "device.data.551010-23": "", - "device.data.551010-24": "", - "Import.modal.551010-0": "", - "Import.modal.551010-1": "", - "Import.modal.551010-2": "", - "Import.modal.551010-3": "", - "Import.modal.551010-4": "", - "Import.modal.551010-5": "", - "Import.modal.551010-6": "", - "Import.modal.551010-7": "", - "Import.modal.551010-8": "", - "Import.modal.551010-9": "", - "Import.modal.551010-10": "", - "Import.modal.551010-11": "", - "Import.modal.551010-12": "", - "Import.modal.551010-13": "", - "Import.modal.551010-14": "", - "Import.modal.551010-15": "", - "MetadataMap.index.551011-0": "", - "MetadataMap.index.551011-1": "", - "MetadataMap.index.551011-2": "", - "MetadataMap.index.551011-3": "", - "MetadataMap.index.551011-4": "", - "MetadataMap.index.551011-5": "", - "MetadataMap.index.551011-6": "", - "MetadataMap.index.551011-7": "", - "MetadataMap.index.551011-8": "", - "MetadataMap.index.551011-9": "", - "MetadataMap.index.551011-10": "", - "MetadataMap.index.551011-11": "", - "MetadataMap.index.551011-12": "", - "MetadataMap.index.551011-13": "", - "MetadataMap.index.551011-14": "", - "MetadataMap.index.551011-15": "", - "MetadataMap.index.551011-16": "", - "MetadataMap.index.551011-17": "", - "MetadataMap.index.551011-18": "", - "MetadataMap.index.551011-19": "", - "MetadataMap.index.551011-20": "", - "MetadataMap.index.551011-21": "", - "MetadataMap.index.551011-22": "", - "MetadataMap.index.551011-23": "", - "MetadataMap.index.551011-24": "", - "MetadataMap.index.551011-25": "", - "MetadataMap.index.551011-26": "", - "MetadataMap.index.551011-27": "", - "Import.file.551010-0": "", - "Import.file.551010-1": "", - "Import.file.551010-2": "", - "Import.file.551010-3": "", - "Import.file.551010-4": "", - "Import.file.551010-5": "", - "Import.file.551010-6": "", - "Import.file.551010-7": "", - "Import.file.551010-8": "", - "Import.file.551010-9": "", - "Import.file.551010-10": "", - "Import.file.551010-11": "", - "Import.file.551010-12": "", - "Import.file.551010-13": "", - "Import.file.551010-14": "", - "Import.file.551010-15": "", - "Import.file.551010-16": "", - "Import.file.551010-17": "", - "Running.index.551010-0": "", - "Running.index.551010-1": "", - "Detail.index.551010-16": "", - "Detail.index.551010-17": "", - "Property.Save.551010-0": "", - "Property.Save.551010-1": "", - "Property.Save.551010-2": "", - "Property.Save.551010-3": "", - "Property.Save.551010-4": "", - "Detail.index.551010-18": "", - "Detail.index.551010-19": "", - "Detail.index.551010-20": "", - "Detail.index.551010-21": "", - "Detail.index.551010-22": "", - "Detail.index.551010-23": "", - "Detail.index.551010-24": "", - "BasicInfo.indev.551011-0": "", - "BasicInfo.indev.551011-1": "", - "BasicInfo.indev.551011-2": "", - "BasicInfo.indev.551011-3": "", - "BasicInfo.indev.551011-4": "", - "BasicInfo.indev.551011-5": "", - "BasicInfo.indev.551011-6": "", - "BasicInfo.indev.551011-7": "", - "BasicInfo.indev.551011-8": "", - "DataAnalysis.index.551011-0": "", - "DataAnalysis.index.551011-1": "", - "DataAnalysis.index.551011-2": "", - "DataAnalysis.index.551011-3": "", - "DataAnalysis.index.551011-4": "", - "DataAnalysis.index.551011-5": "", - "DataAnalysis.index.551011-6": "", - "DataAnalysis.index.551011-7": "", - "DataAnalysis.index.551011-8": "", - "DataAnalysis.index.551011-9": "", - "DataAnalysis.index.551011-10": "", - "DataAnalysis.index.551011-11": "", - "Config.Save.551011-0": "", - "Config.Save.551011-1": "", - "Config.Save.551011-2": "", - "Config.Save.551011-3": "", - "Function.index.551011-0": "", - "Function.index.551011-1": "", - "Function.index.551011-2": "", - "Function.index.551011-3": "", - "Config.index.551011-0": "", - "Config.index.551011-1": "", - "Config.index.551011-2": "", - "Config.index.551011-3": "", - "Config.index.551011-4": "", - "Config.index.551011-5": "", - "Config.index.551011-6": "", - "Config.index.551011-7": "", - "Config.index.551011-8": "", - "Config.index.551011-9": "", - "Config.index.551011-10": "", - "Property.ValueRender.551010-0": "", - "Property.ValueRender.551010-1": "", - "Property.ValueRender.551010-2": "", - "Property.ValueRender.551010-3": "", - "EditTable.index.551011-0": "", - "EditTable.index.551011-1": "", - "EditTable.index.551011-2": "", - "EditTable.index.551011-3": "", - "EditTable.index.551011-4": "", - "EditTable.index.551011-5": "", - "EditTable.index.551011-6": "", - "EditTable.index.551011-7": "", - "EditTable.index.551011-8": "", - "EditTable.index.551011-9": "", - "EditTable.index.551011-10": "", - "EditTable.index.551011-11": "", - "EditTable.index.551011-12": "", - "EditTable.index.551011-13": "", - "EditTable.index.551011-14": "", - "EditTable.index.551011-15": "", - "EditTable.index.551011-16": "", - "EditTable.index.551011-17": "", - "Status.DiagnosticAdvice.551011-0": "", - "Status.DiagnosticAdvice.551011-1": "", - "Status.DiagnosticAdvice.551011-2": "", - "Status.DiagnosticAdvice.551011-3": "", - "Status.DiagnosticAdvice.551011-4": "", - "Status.DiagnosticAdvice.551011-5": "", - "components.MSelect.551011-0": "", - "EditTable.PatchMapping.551011-0": "", - "EditTable.PatchMapping.551011-1": "", - "EditTable.PatchMapping.551011-2": "", - "EditTable.PatchMapping.551011-3": "", - "EditTable.PatchMapping.551011-4": "", - "EditTable.PatchMapping.551011-5": "", - "EditTable.PatchMapping.551011-6": "", - "EditTable.PatchMapping.551011-7": "", - "EditTable.PatchMapping.551011-8": "", - "BindParentDevice.index.551012-0": "", - "BindParentDevice.index.551012-1": "", - "BindParentDevice.index.551012-2": "", - "BindParentDevice.index.551012-3": "", - "BindParentDevice.index.551012-4": "", - "BindParentDevice.index.551012-5": "", - "BindParentDevice.index.551012-6": "", - "BindParentDevice.index.551012-7": "", - "BindParentDevice.index.551012-8": "", - "BindParentDevice.index.551012-9": "", - "BindParentDevice.index.551012-10": "", - "BindParentDevice.index.551012-11": "", - "Diagnose.index.551011-0": "", - "Diagnose.index.551011-1": "", - "Diagnose.index.551011-2": "", - "Diagnose.index.551011-3": "", - "DeviceAccess.util.551011-0": "", - "DeviceAccess.util.551011-1": "", - "DeviceAccess.util.551011-2": "", - "DeviceAccess.util.551011-3": "", - "DeviceAccess.util.551011-4": "", - "DeviceAccess.util.551011-5": "", - "DeviceAccess.util.551011-6": "", - "DeviceAccess.util.551011-7": "", - "Detail.Charts.551011-0": "", - "Detail.Charts.551011-1": "", - "Detail.Charts.551011-2": "", - "Detail.Charts.551011-3": "", - "Detail.Charts.551011-4": "", - "Detail.Charts.551011-5": "", - "Detail.Charts.551011-6": "", - "Detail.Charts.551011-7": "", - "Detail.Charts.551011-8": "", - "Detail.Charts.551011-9": "", - "Detail.Charts.551011-10": "", - "Detail.Charts.551011-11": "", - "DeviceAccess.metadataModal.551011-0": "", - "DeviceAccess.metadataModal.551011-1": "", - "DeviceAccess.metadataModal.551011-2": "", - "DeviceAccess.metadataModal.551011-3": "", - "DeviceAccess.metadataModal.551011-4": "", - "DeviceAccess.metadataModal.551011-5": "", - "DeviceAccess.metadataModal.551011-6": "", - "DeviceAccess.metadataModal.551011-7": "", - "DeviceAccess.metadataModal.551011-8": "", - "DeviceAccess.metadataModal.551011-9": "", - "DeviceAccess.metadataModal.551011-10": "", - "DeviceAccess.metadataModal.551011-11": "", - "DeviceAccess.metadataModal.551011-12": "", - "DeviceAccess.metadataModal.551011-13": "", - "DeviceAccess.metadataModal.551011-14": "", - "DeviceAccess.metadataModal.551011-15": "", - "DeviceAccess.metadataModal.551011-16": "", - "DeviceAccess.metadataModal.551011-17": "", - "Relation.Save.551011-0": "", - "Relation.Save.551011-1": "", - "Relation.Save.551011-2": "", - "Relation.Save.551011-3": "", - "Tags.index.551011-0": "", - "Tags.index.551011-1": "", - "Tags.Save.551011-0": "", - "Tags.Save.551011-1": "", - "Tags.Save.551011-2": "", - "Tags.Save.551011-3": "", - "Relation.index.551011-0": "", - "Relation.index.551011-1": "", - "Relation.index.551011-2": "", - "InklingModal.index.551011-0": "", - "InklingModal.index.551011-1": "", - "Parsing.index.551010-0": "", - "Parsing.index.551010-1": "", - "Parsing.index.551010-2": "", - "Parsing.index.551010-3": "", - "Parsing.index.551010-4": "", - "Parsing.index.551010-5": "", - "Parsing.index.551010-6": "", - "Parsing.index.551010-7": "", - "Parsing.index.551010-8": "", - "Parsing.index.551010-9": "", - "Parsing.index.551010-10": "", - "Parsing.index.551010-11": "", - "Parsing.index.551010-12": "", - "Parsing.index.551010-13": "", - "Parsing.index.551010-14": "", - "Parsing.index.551010-15": "", - "Parsing.index.551010-16": "", - "Parsing.index.551010-17": "", - "Parsing.index.551010-18": "", - "Parsing.index.551010-19": "", - "Property.PropertyCard.551010-0": "", - "Property.ValueDetail.551010-0": "", - "Property.ValueDetail.551010-1": "", - "Property.ValueDetail.551010-2": "", - "Property.index.551010-0": "", - "Property.index.551010-1": "", - "Property.index.551010-2": "", - "Property.index.551010-3": "", - "Property.index.551010-4": "", - "Property.index.551010-5": "", - "Property.index.551010-6": "", - "Property.index.551010-7": "", - "Property.index.551010-8": "", - "Property.index.551010-9": "", - "Status.util.551011-0": "", - "Status.util.551011-1": "", - "Status.util.551011-2": "", - "Status.util.551011-3": "", - "Status.util.551011-4": "", - "Status.util.551011-5": "", - "Status.util.551011-6": "", - "Status.util.551011-7": "", - "Status.util.551011-8": "", - "Status.util.551011-9": "", - "Status.util.551011-10": "", - "Status.util.551011-11": "", - "EdgeMap.MSelect.551011-0": "", - "Base.Base.551013-0": "", - "Base.Base.551013-1": "", - "Base.Base.551013-2": "", - "Base.Base.551013-3": "", - "Base.Base.551013-4": "", - "Base.Base.551013-5": "", - "Base.Base.551013-6": "", - "Base.Base.551013-7": "", - "Base.Base.551013-8": "", - "Base.Base.551013-9": "", - "Base.Base.551013-10": "", - "Base.Base.551013-11": "", - "Base.Base.551013-12": "", - "Base.Base.551013-13": "", - "Base.Base.551013-14": "", - "Base.Base.551013-15": "", - "Base.Base.551013-16": "", - "Base.Base.551013-17": "", - "Base.Base.551013-18": "", - "Base.Base.551013-19": "", - "components.Simple.551011-0": "", - "components.Simple.551011-1": "", - "components.Simple.551011-2": "", - "components.Simple.551011-3": "", - "components.Simple.551011-4": "", - "components.Simple.551011-5": "", - "components.Simple.551011-6": "", - "components.Simple.551011-7": "", - "components.Simple.551011-8": "", - "components.Simple.551011-9": "", - "components.Advance.551011-0": "", - "components.Advance.551011-1": "", - "components.Advance.551011-2": "", - "components.Advance.551011-3": "", - "EdgeMap.PatchMapping.551011-0": "", - "EdgeMap.PatchMapping.551011-1": "", - "EdgeMap.PatchMapping.551011-2": "", - "EdgeMap.PatchMapping.551011-3": "", - "EdgeMap.PatchMapping.551011-4": "", - "EdgeMap.PatchMapping.551011-5": "", - "EdgeMap.PatchMapping.551011-6": "", - "EdgeMap.PatchMapping.551011-7": "", - "EdgeMap.PatchMapping.551011-8": "", - "Import.index.551012-0": "", - "Import.index.551012-1": "", - "Import.index.551012-2": "", - "Import.index.551012-3": "", - "Import.index.551012-4": "", - "Import.index.551012-5": "", - "Import.index.551012-6": "", - "Import.index.551012-7": "", - "Import.index.551012-8": "", - "Import.index.551012-9": "", - "Import.index.551012-10": "", - "Import.index.551012-11": "", - "Import.index.551012-12": "", - "Import.index.551012-13": "", - "Import.index.551012-14": "", - "Import.index.551012-15": "", - "Import.index.551012-16": "", - "Import.index.551012-17": "", - "Import.index.551012-18": "", - "Import.index.551012-19": "", - "Import.index.551012-20": "", - "Import.index.551012-21": "", - "Import.index.551012-22": "", - "Import.index.551012-23": "", - "Import.index.551012-24": "", - "Import.index.551012-25": "", - "Import.index.551012-26": "", - "Import.index.551012-27": "", - "Import.index.551012-28": "", - "Import.index.551012-29": "", - "Import.index.551012-30": "", - "Import.index.551012-31": "", - "Import.index.551012-32": "", - "Import.index.551012-33": "", - "Import.index.551012-34": "", - "Import.index.551012-35": "", - "Import.index.551012-36": "", - "Import.index.551012-37": "", - "Import.index.551012-38": "", - "Import.index.551012-39": "", - "Import.index.551012-40": "", - "Import.index.551012-41": "", - "Import.index.551012-42": "", - "Import.index.551012-43": "", - "Import.index.551012-44": "", - "Import.index.551012-45": "", - "Import.index.551012-46": "", - "Import.index.551012-47": "", - "Import.index.551012-48": "", - "Import.index.551012-49": "", - "Import.index.551012-50": "", - "Import.index.551012-51": "", - "Import.index.551012-52": "", - "Import.index.551012-53": "", - "Import.index.551012-54": "", - "Import.index.551012-55": "", - "Import.index.551012-56": "", - "Import.index.551012-57": "", - "Import.index.551012-58": "", - "Import.index.551012-59": "", - "Import.index.551012-60": "", - "Diagnose.util.551011-0": "", - "Diagnose.util.551011-1": "", - "Diagnose.util.551011-2": "", - "Diagnose.util.551011-3": "", - "Diagnose.util.551011-4": "", - "Diagnose.util.551011-5": "", - "EdgeMap.index.551011-0": "", - "EdgeMap.index.551011-1": "", - "EdgeMap.index.551011-2": "", - "EdgeMap.index.551011-3": "", - "EdgeMap.index.551011-4": "", - "EdgeMap.index.551011-5": "", - "EdgeMap.index.551011-6": "", - "EdgeMap.index.551011-7": "", - "EdgeMap.index.551011-8": "", - "EdgeMap.index.551011-9": "", - "EdgeMap.index.551011-10": "", - "EdgeMap.index.551011-11": "", - "EdgeMap.index.551011-12": "", - "EdgeMap.index.551011-13": "", - "EdgeMap.index.551011-14": "", - "EdgeMap.index.551011-15": "", - "EdgeMap.index.551011-16": "", - "EdgeMap.index.551011-17": "", - "EdgeMap.index.551011-18": "", - "EdgeMap.index.551011-19": "", - "EdgeMap.index.551011-20": "", - "EdgeMap.index.551011-21": "", - "Log.index.551011-0": "", - "Log.index.551011-1": "", - "Log.index.551011-2": "", - "Log.index.551011-3": "", - "Log.index.551011-4": "", - "Log.index.551011-5": "", - "Import.valideta.551012-0": "", - "Import.valideta.551012-1": "", - "Import.valideta.551012-2": "", - "Import.valideta.551012-3": "", - "Import.valideta.551012-4": "", - "Import.valideta.551012-5": "", - "Import.valideta.551012-6": "", - "Import.valideta.551012-7": "", - "Import.valideta.551012-8": "", - "Import.valideta.551012-9": "", - "Import.valideta.551012-10": "", - "Import.valideta.551012-11": "", - "Import.valideta.551012-12": "", - "Import.valideta.551012-13": "", - "Message.index.551011-0": "", - "Message.index.551011-1": "", - "Message.index.551011-2": "", - "Message.index.551011-3": "", - "Message.index.551011-4": "", - "Message.index.551011-5": "", - "Message.index.551011-6": "", - "Message.index.551011-7": "", - "SaveChild.index.551011-0": "", - "SaveChild.index.551011-1": "", - "SaveChild.index.551011-2": "", - "SaveChild.index.551011-3": "", - "SaveChild.index.551011-4": "", - "SaveChild.index.551011-5": "", - "DashBoard.index.551011-0": "", - "DashBoard.index.551011-1": "", - "DashBoard.index.551011-2": "", - "DashBoard.index.551011-3": "", - "DashBoard.index.551011-4": "", - "DashBoard.index.551011-5": "", - "DashBoard.index.551011-6": "", - "DashBoard.index.551011-7": "", - "DashBoard.index.551011-8": "", - "DashBoard.index.551011-9": "", - "DashBoard.index.551011-10": "", - "DashBoard.index.551011-11": "", - "DashBoard.index.551011-12": "", - "DashBoard.index.551011-13": "", - "DashBoard.index.551011-14": "", - "DashBoard.index.551011-15": "", - "DashBoard.index.551011-16": "", - "DashBoard.index.551011-17": "", - "DashBoard.index.551011-18": "", - "DashBoard.index.551011-19": "", - "DashBoard.index.551011-20": "", - "components.TimeSelect.551011-0": "", - "components.TimeSelect.551011-1": "", - "components.TimeSelect.551011-2": "", - "components.TimeSelect.551011-3": "", - "AlarmRecord.index.551011-0": "", - "AlarmRecord.index.551011-1": "", - "Status.ManualInspection.551011-0": "", - "Status.ManualInspection.551011-1": "", - "Status.ManualInspection.551011-2": "", - "Status.ManualInspection.551011-3": "", - "Status.ManualInspection.551011-4": "", - "Status.ManualInspection.551011-5": "", - "Status.ManualInspection.551011-6": "", - "Status.ManualInspection.551011-7": "", - "Status.ManualInspection.551011-8": "", - "Status.ManualInspection.551011-9": "", - "Status.ManualInspection.551011-10": "", - "Status.ManualInspection.551011-11": "", - "Status.ManualInspection.551011-12": "", - "Status.ManualInspection.551011-13": "", - "Status.ManualInspection.551011-14": "", - "Status.ManualInspection.551011-15": "", - "Function.index.551012-0": "", - "Function.index.551012-1": "", - "Function.index.551012-2": "", - "Function.index.551012-3": "", - "Function.index.551012-4": "", - "Function.index.551012-5": "", - "Function.index.551012-6": "", - "Function.index.551012-7": "", - "Function.index.551012-8": "", - "Dialog.index.551012-0": "", - "Dialog.index.551012-1": "", - "Dialog.index.551012-2": "", - "Dialog.index.551012-3": "", - "Dialog.index.551012-4": "", - "Dialog.index.551012-5": "", - "Dialog.index.551012-6": "", - "Dialog.index.551012-7": "", - "EdgeMap.MSelect.551012-0": "", - "Function.EditTable.551012-0": "", - "Function.EditTable.551012-1": "", - "Function.EditTable.551012-2": "", - "Function.EditTable.551012-3": "", - "Function.EditTable.551012-4": "", - "Function.EditTable.551012-5": "", - "EdgeMap.PatchMapping.551012-0": "", - "EdgeMap.PatchMapping.551012-1": "", - "EdgeMap.PatchMapping.551012-2": "", - "EdgeMap.PatchMapping.551012-3": "", - "EdgeMap.PatchMapping.551012-4": "", - "EdgeMap.PatchMapping.551012-5": "", - "EdgeMap.PatchMapping.551012-6": "", - "EdgeMap.PatchMapping.551012-7": "", - "EdgeMap.PatchMapping.551012-8": "", - "EdgeMap.index.551012-0": "", - "EdgeMap.index.551012-1": "", - "EdgeMap.index.551012-2": "", - "EdgeMap.index.551012-3": "", - "EdgeMap.index.551012-4": "", - "EdgeMap.index.551012-5": "", - "EdgeMap.index.551012-6": "", - "EdgeMap.index.551012-7": "", - "EdgeMap.index.551012-8": "", - "EdgeMap.index.551012-9": "", - "EdgeMap.index.551012-10": "", - "EdgeMap.index.551012-11": "", - "EdgeMap.index.551012-12": "", - "EdgeMap.index.551012-13": "", - "EdgeMap.index.551012-14": "", - "EdgeMap.index.551012-15": "", - "EdgeMap.index.551012-16": "", - "EdgeMap.index.551012-17": "", - "EdgeMap.index.551012-18": "", - "EdgeMap.index.551012-19": "", - "EdgeMap.index.551012-20": "", - "Export.index.551012-0": "", - "Export.index.551012-1": "", - "Export.index.551012-2": "", - "Export.index.551012-3": "", - "Export.index.551012-4": "", - "Export.index.551012-5": "", - "components.AlarmLog.551012-0": "", - "components.AlarmLog.551012-1": "", - "Metadata.index.551012-0": "", - "Metadata.index.551012-1": "", - "Metadata.index.551012-2": "", - "Metadata.index.551012-3": "", - "Metadata.index.551012-4": "", - "Metadata.index.551012-5": "", - "Metadata.index.551012-6": "", - "Metadata.index.551012-7": "", - "Metadata.index.551012-8": "", - "Metadata.index.551012-9": "", - "Metadata.index.551012-10": "", - "Cat.index.551012-0": "", - "Cat.index.551012-1": "", - "Cat.index.551012-2": "", - "Cat.index.551012-3": "", - "Cat.index.551012-4": "", - "Cat.index.551012-5": "", - "Cat.index.551012-6": "", - "Cat.index.551012-7": "", - "InklingDevice.index.551012-0": "", - "InklingDevice.index.551012-1": "", - "InklingDevice.index.551012-2": "", - "InklingDevice.index.551012-3": "", - "Category.index.551012-0": "", - "Category.index.551012-1": "", - "Category.index.551012-2": "", - "Category.index.551012-3": "", - "Category.index.551012-4": "", - "Category.index.551012-5": "", - "Category.index.551012-6": "", - "Category.index.551012-7": "", - "Category.index.551012-8": "", - "Category.index.551012-9": "", - "Category.index.551012-10": "", - "Category.index.551012-11": "", - "Category.index.551012-12": "", - "Category.index.551012-13": "", - "Category.index.551012-14": "", - "Category.index.551012-15": "", - "Alarm.index.551012-0": "", - "Alarm.index.551012-1": "", - "Alarm.index.551012-2": "", - "Alarm.index.551012-3": "", - "Alarm.index.551012-4": "", - "Alarm.index.551012-5": "", - "Alarm.index.551012-6": "", - "Alarm.index.551012-7": "", - "Alarm.index.551012-8": "", - "Alarm.index.551012-9": "", - "Alarm.index.551012-10": "", - "Alarm.index.551012-11": "", - "Alarm.index.551012-12": "", - "Alarm.index.551012-13": "", - "Alarm.index.551012-14": "", - "Alarm.index.551012-15": "", - "Base.utils.551012-0": "", - "Base.utils.551012-1": "", - "Base.utils.551012-2": "", - "Base.utils.551012-3": "", - "Base.utils.551012-4": "", - "Base.utils.551012-5": "", - "Base.utils.551012-6": "", - "Base.utils.551012-7": "", - "Base.utils.551012-8": "", - "Base.index.551012-0": "", - "Base.index.551012-1": "", - "Base.index.551012-2": "", - "Base.index.551012-3": "", - "Base.index.551012-4": "", - "Base.index.551012-5": "", - "Base.index.551012-6": "", - "Base.index.551012-7": "", - "Base.index.551012-8": "", - "Base.index.551012-9": "", - "Base.index.551012-10": "", - "Base.index.551012-11": "", - "Base.index.551012-12": "", - "Base.index.551012-13": "", - "Base.index.551012-14": "", - "Base.index.551012-15": "", - "Base.index.551012-16": "", - "Base.index.551012-17": "", - "Base.index.551012-18": "", - "Base.index.551012-19": "", - "Base.index.551012-20": "", - "Base.index.551012-21": "", - "Base.index.551012-22": "", - "Invalid.index.551012-0": "", - "Invalid.index.551012-1": "", - "Invalid.index.551012-2": "", - "Invalid.index.551012-3": "", - "components.ConfigModal.551013-0": "", - "components.ConfigModal.551013-1": "", - "components.ConfigModal.551013-2": "", - "components.ConfigModal.551013-3": "", - "components.ConfigModal.551013-4": "", - "components.ConfigModal.551013-5": "", - "components.ConfigModal.551013-6": "", - "components.ConfigModal.551013-7": "", - "components.ConfigModal.551013-8": "", - "components.ConfigModal.551013-9": "", - "components.ConfigModal.551013-10": "", - "components.ConfigModal.551013-11": "", - "components.Source.551013-0": "", - "components.Source.551013-1": "", - "components.Source.551013-2": "", - "components.Source.551013-3": "", - "components.Source.551013-4": "", - "components.Source.551013-5": "", - "components.Source.551013-6": "", - "components.Source.551013-7": "", - "BindChildDevice.index.551012-0": "", - "BindChildDevice.index.551012-1": "", - "BindChildDevice.index.551012-2": "", - "BindChildDevice.index.551012-3": "", - "BindChildDevice.index.551012-4": "", - "BindChildDevice.index.551012-5": "", - "BindChildDevice.index.551012-6": "", - "BindChildDevice.index.551012-7": "", - "BindChildDevice.index.551012-8": "", - "BindChildDevice.index.551012-9": "", - "BindChildDevice.index.551012-10": "", - "BindChildDevice.index.551012-11": "", - "Log.index.551012-0": "", - "Log.index.551012-1": "", - "Log.index.551012-2": "", - "Log.index.551012-3": "", - "Log.index.551012-4": "", - "Log.index.551012-5": "", - "Log.index.551012-6": "", - "Log.index.551012-7": "", - "Edit.index.551013-0": "", - "Edit.index.551013-1": "", - "Edit.index.551013-2": "", - "Edit.index.551013-3": "", - "Edit.index.551013-4": "", - "Edit.index.551013-5": "", - "Edit.index.551013-6": "", - "Edit.index.551013-7": "", - "Edit.index.551013-8": "", - "Edit.index.551013-9": "", - "Edit.validator.551013-0": "", - "Edit.validator.551013-1": "", - "Edit.validator.551013-2": "", - "Edit.validator.551013-3": "", - "Edit.validator.551013-4": "", - "Edit.validator.551013-5": "", - "Edit.validator.551013-6": "", - "Edit.validator.551013-7": "", - "Edit.validator.551013-8": "", - "DetailModal.utils.551013-0": "", - "DetailModal.utils.551013-1": "", - "DetailModal.utils.551013-2": "", - "Edit.BaseForm.551013-0": "", - "Edit.BaseForm.551013-1": "", - "Edit.BaseForm.551013-2": "", - "Edit.BaseForm.551013-3": "", - "Edit.BaseForm.551013-4": "", - "Edit.BaseForm.551013-5": "", - "Edit.BaseForm.551013-6": "", - "Edit.BaseForm.551013-7": "", - "Edit.BaseForm.551013-8": "", - "Edit.BaseForm.551013-9": "", - "Edit.BaseForm.551013-10": "", - "Edit.BaseForm.551013-11": "", - "Edit.BaseForm.551013-12": "", - "Edit.BaseForm.551013-13": "", - "Edit.BaseForm.551013-14": "", - "Edit.BaseForm.551013-15": "", - "Edit.BaseForm.551013-16": "", - "Edit.BaseForm.551013-17": "", - "Edit.BaseForm.551013-18": "", - "Edit.BaseForm.551013-19": "", - "modifyModal.index.551013-0": "", - "modifyModal.index.551013-1": "", - "modifyModal.index.551013-2": "", - "modifyModal.index.551013-3": "", - "modifyModal.index.551013-4": "", - "modifyModal.index.551013-5": "", - "modifyModal.index.551013-6": "", - "modifyModal.index.551013-7": "", - "modifyModal.index.551013-8": "", - "modifyModal.index.551013-9": "", - "modifyModal.index.551013-10": "", - "modifyModal.index.551013-11": "", - "modifyModal.index.551013-12": "", - "DetailModal.TagsModal.551013-0": "", - "DetailModal.TagsModal.551013-1": "", - "DetailModal.TagsModal.551013-2": "", - "DetailModal.TagsModal.551013-3": "", - "DetailModal.TagsModal.551013-4": "", - "DetailModal.TagsModal.551013-5": "", - "DetailModal.TagsModal.551013-6": "", - "DetailModal.TagsModal.551013-7": "", - "DetailModal.TagsModal.551013-8": "", - "DetailModal.TagsModal.551013-9": "", - "DetailModal.TagsModal.551013-10": "", - "DetailModal.TagsModal.551013-11": "", - "DetailModal.TagsModal.551013-12": "", - "DetailModal.TagsModal.551013-13": "", - "DetailModal.TagsModal.551013-14": "", - "DetailModal.TagsModal.551013-15": "", - "DetailModal.TagsModal.551013-16": "", - "DetailModal.TagsModal.551013-17": "", - "DetailModal.TagsModal.551013-18": "", - "DetailModal.TagsModal.551013-19": "", - "DetailModal.TagsModal.551013-20": "", - "DetailModal.TagsModal.551013-21": "", - "DetailModal.TagsModal.551013-22": "", - "DetailModal.PropertiesModal.551013-0": "", - "DetailModal.PropertiesModal.551013-1": "", - "DetailModal.PropertiesModal.551013-2": "", - "DetailModal.PropertiesModal.551013-3": "", - "DetailModal.PropertiesModal.551013-4": "", - "DetailModal.PropertiesModal.551013-5": "", - "DetailModal.PropertiesModal.551013-6": "", - "DetailModal.PropertiesModal.551013-7": "", - "DetailModal.PropertiesModal.551013-8": "", - "DetailModal.PropertiesModal.551013-9": "", - "DetailModal.PropertiesModal.551013-10": "", - "DetailModal.PropertiesModal.551013-11": "", - "DetailModal.PropertiesModal.551013-12": "", - "DetailModal.PropertiesModal.551013-13": "", - "DetailModal.PropertiesModal.551013-14": "", - "DetailModal.PropertiesModal.551013-15": "", - "DetailModal.PropertiesModal.551013-16": "", - "DetailModal.PropertiesModal.551013-17": "", - "DetailModal.PropertiesModal.551013-18": "", - "DetailModal.PropertiesModal.551013-19": "", - "DetailModal.PropertiesModal.551013-20": "", - "DetailModal.PropertiesModal.551013-21": "", - "DetailModal.PropertiesModal.551013-22": "", - "DetailModal.PropertiesModal.551013-23": "", - "DetailModal.PropertiesModal.551013-24": "", - "DetailModal.PropertiesModal.551013-25": "", - "DetailModal.PropertiesModal.551013-26": "", - "Base.columns.551013-0": "", - "Base.columns.551013-1": "", - "Base.columns.551013-2": "", - "Base.columns.551013-3": "", - "Base.columns.551013-4": "", - "Base.columns.551013-5": "", - "Base.columns.551013-6": "", - "Base.columns.551013-7": "", - "Base.columns.551013-8": "", - "Base.columns.551013-9": "", - "Base.columns.551013-10": "", - "Base.columns.551013-11": "", - "Base.columns.551013-12": "", - "Base.columns.551013-13": "", - "Base.columns.551013-14": "", - "Base.columns.551013-15": "", - "Base.columns.551013-16": "", - "Base.columns.551013-17": "", - "Base.columns.551013-18": "", - "Base.columns.551013-19": "", - "Base.columns.551013-20": "", - "Base.columns.551013-21": "", - "Base.columns.551013-22": "", - "Base.columns.551013-23": "", - "Base.columns.551013-24": "", - "Base.columns.551013-25": "", - "Base.columns.551013-26": "", - "Base.columns.551013-27": "", - "Base.columns.551013-28": "", - "Base.columns.551013-29": "", - "Base.columns.551013-30": "", - "Base.columns.551013-31": "", - "Base.columns.551013-32": "", - "Base.columns.551013-33": "", - "Base.columns.551013-34": "", - "Base.columns.551013-35": "", - "Base.columns.551013-36": "", - "Base.columns.551013-37": "", - "Base.columns.551013-38": "", - "Base.columns.551013-39": "", - "Base.columns.551013-40": "", - "Base.columns.551013-41": "", - "Base.columns.551013-42": "", - "Base.columns.551013-43": "", - "Base.columns.551013-44": "", - "Base.columns.551013-45": "", - "DetailModal.EventModal.551013-0": "", - "DetailModal.EventModal.551013-1": "", - "DetailModal.EventModal.551013-2": "", - "DetailModal.EventModal.551013-3": "", - "DetailModal.EventModal.551013-4": "", - "DetailModal.EventModal.551013-5": "", - "DetailModal.EventModal.551013-6": "", - "DetailModal.EventModal.551013-7": "", - "DetailModal.EventModal.551013-8": "", - "DetailModal.EventModal.551013-9": "", - "Edit.ValueTypeForm.551013-0": "", - "Edit.ValueTypeForm.551013-1": "", - "Edit.ValueTypeForm.551013-2": "", - "Edit.ValueTypeForm.551013-3": "", - "Edit.ValueTypeForm.551013-4": "", - "Edit.ValueTypeForm.551013-5": "", - "Edit.ValueTypeForm.551013-6": "", - "Edit.ValueTypeForm.551013-7": "", - "Edit.ValueTypeForm.551013-8": "", - "Edit.ValueTypeForm.551013-9": "", - "Edit.ValueTypeForm.551013-10": "", - "Edit.ValueTypeForm.551013-11": "", - "Edit.ValueTypeForm.551013-12": "", - "Edit.ValueTypeForm.551013-13": "", - "Edit.ValueTypeForm.551013-14": "", - "Edit.ValueTypeForm.551013-15": "", - "VirtualRule.DetailModal.551013-0": "", - "Metrics.BooleanSelect.551013-0": "", - "Metrics.BooleanSelect.551013-1": "", - "components.DataType.551013-0": "", - "components.DataType.551013-1": "", - "components.DataType.551013-2": "", - "components.DataType.551013-3": "", - "components.DataType.551013-4": "", - "components.DataType.551013-5": "", - "components.DataType.551013-6": "", - "components.DataType.551013-7": "", - "components.DataType.551013-8": "", - "components.DataType.551013-9": "", - "components.DataType.551013-10": "", - "Metrics.Metrics.551013-0": "", - "Metrics.Metrics.551013-1": "", - "Metrics.Metrics.551013-2": "", - "Metrics.Metrics.551013-3": "", - "Metrics.Metrics.551013-4": "", - "Metrics.Metrics.551013-5": "", - "Metrics.Metrics.551013-6": "", - "Metrics.Metrics.551013-7": "", - "Metrics.Metrics.551013-8": "", - "Metrics.Metrics.551013-9": "", - "Metrics.Metrics.551013-10": "", - "Metrics.Metrics.551013-11": "", - "Metrics.Metrics.551013-12": "", - "VirtualRule.Rule.551013-0": "", - "components.DataTypeObjectChild.551013-0": "", - "VirtualRule.index.551013-0": "", - "VirtualRule.index.551013-1": "", - "VirtualRule.index.551013-2": "", - "VirtualRule.index.551013-3": "", - "VirtualRule.index.551013-4": "", - "VirtualRule.index.551013-5": "", - "VirtualRule.index.551013-6": "", - "VirtualRule.index.551013-7": "", - "VirtualRule.index.551013-8": "", - "VirtualRule.index.551013-9": "", - "VirtualRule.index.551013-10": "", - "VirtualRule.index.551013-11": "", - "VirtualRule.index.551013-12": "", - "VirtualRule.index.551013-13": "", - "VirtualRule.index.551013-14": "", - "VirtualRule.index.551013-15": "", - "VirtualRule.index.551013-16": "", - "VirtualRule.index.551013-17": "", - "VirtualRule.index.551013-18": "", - "VirtualRule.index.551013-19": "", - "VirtualRule.index.551013-20": "", - "VirtualRule.index.551013-21": "", - "Metrics.item.551013-0": "", - "Metrics.item.551013-1": "", - "Tags.Type.551013-0": "", - "Tags.Type.551013-1": "", - "Tags.Type.551013-2": "", - "Tags.Type.551013-3": "", - "Detail.Save.551010-0": "", - "Detail.Save.551010-1": "", - "Detail.Save.551010-2": "", - "Detail.Save.551010-3": "", - "Events.OtherConfigInfo.551013-0": "", - "Events.ConfigParams.551013-0": "", - "Events.ConfigParams.551013-1": "", - "Events.ConfigParams.551013-2": "", - "Events.ConfigParams.551013-3": "", - "Events.ConfigParams.551013-4": "", - "Events.ConfigParams.551013-5": "", - "Events.ConfigParams.551013-6": "", - "Events.ConfigParams.551013-7": "", - "Events.ConfigParams.551013-8": "", - "Events.ConfigParams.551013-9": "", - "Events.ConfigParams.551013-10": "", - "Events.ConfigParams.551013-11": "", - "Events.ConfigParams.551013-12": "", - "Product.index.551010-0": "", - "Product.index.551010-1": "", - "Product.index.551010-2": "", - "Product.index.551010-3": "", - "Product.index.551010-4": "", - "Product.index.551010-5": "", - "Product.index.551010-6": "", - "Product.index.551010-7": "", - "Product.index.551010-8": "", - "Product.index.551010-9": "", - "Product.index.551010-10": "", - "Product.index.551010-11": "", - "Product.index.551010-12": "", - "Product.index.551010-13": "", - "Product.index.551010-14": "", - "Product.index.551010-15": "", - "Product.index.551010-16": "", - "Product.index.551010-17": "", - "Product.index.551010-18": "", - "Product.index.551010-19": "", - "Product.index.551010-20": "", - "Product.index.551010-21": "", - "Product.index.551010-22": "", - "Product.index.551010-23": "", - "Product.index.551010-24": "", - "Product.index.551010-25": "", - "Product.index.551010-26": "", - "Product.index.551010-27": "", - "Product.index.551010-28": "", - "Product.index.551010-29": "", - "Product.index.551010-30": "", - "Product.index.551010-31": "", - "Product.index.551010-32": "", - "Product.index.551010-33": "", - "Events.ValueObject.551013-0": "", - "Function.InputParams.551013-0": "", - "Function.InputParams.551013-1": "", - "Function.InputParams.551013-2": "", - "Function.InputParams.551013-3": "", - "Function.InputParams.551013-4": "", - "Function.InputParams.551013-5": "", - "Function.InputParams.551013-6": "", - "Function.InputParams.551013-7": "", - "Function.InputParams.551013-8": "", - "Function.InputParams.551013-9": "", - "Function.InputParams.551013-10": "", - "Function.InputParams.551013-11": "", - "Function.InputParams.551013-12": "", - "Function.InputParams.551013-13": "", - "Function.InputParams.551013-14": "", - "Function.InputParams.551013-15": "", - "Function.AsyncSelect.551013-0": "", - "Metrics.ValueItem.551013-0": "", - "Metrics.ValueItem.551013-1": "", - "Metrics.ValueItem.551013-2": "", - "Metrics.ValueItem.551013-3": "", - "Metrics.ValueItem.551013-4": "", - "Metrics.ValueItem.551013-5": "", - "Metrics.ValueItem.551013-6": "", - "Metrics.ValueItem.551013-7": "", - "Metrics.ValueItem.551013-8": "", - "DialogTips.index.551010-0": "", - "DialogTips.index.551010-1": "", - "DialogTips.index.551010-2": "", - "DialogTips.index.551010-3": "", - "DialogTips.index.551010-4": "", - "DialogTips.index.551010-5": "", - "DialogTips.index.551010-6": "", - "DialogTips.index.551010-7": "", - "DialogTips.index.551010-8": "", - "DialogTips.index.551010-9": "", - "DialogTips.index.551010-10": "", - "DialogTips.index.551010-11": "", - "DialogTips.index.551010-12": "", - "components.Constraint.551013-0": "", - "components.Constraint.551013-1": "", - "Properties.OtherSetting.551013-0": "", - "Properties.OtherSetting.551013-1": "", - "Properties.OtherSetting.551013-2": "", - "Properties.OtherSetting.551013-3": "", - "Properties.OtherSetting.551013-4": "", - "Properties.OtherSetting.551013-5": "", - "Properties.OtherSetting.551013-6": "", - "Properties.OtherSetting.551013-7": "", - "Properties.OtherSetting.551013-8": "", - "Properties.OtherSetting.551013-9": "", - "Properties.OtherSetting.551013-10": "", - "Properties.OtherSetting.551013-11": "", - "Properties.OtherSetting.551013-12": "", - "Properties.OtherSetting.551013-13": "", - "Properties.OtherSetting.551013-14": "", - "Properties.OtherSetting.551013-15": "", - "Properties.OtherSetting.551013-16": "", - "Properties.OtherSetting.551013-17": "", - "Properties.OtherSetting.551013-18": "", - "Properties.OtherSetting.551013-19": "", - "Properties.OtherSetting.551013-20": "", - "Properties.OtherSetting.551013-21": "", - "Properties.OtherSetting.551013-22": "", - "Properties.OtherSetting.551013-23": "", - "ChildDevice.index.551011-0": "", - "ChildDevice.index.551011-1": "", - "ChildDevice.index.551011-2": "", - "ChildDevice.index.551011-3": "", - "ChildDevice.index.551011-4": "", - "ChildDevice.index.551011-5": "", - "ChildDevice.index.551011-6": "", - "ChildDevice.index.551011-7": "", - "ChildDevice.index.551011-8": "", - "ChildDevice.index.551011-9": "", - "ChildDevice.index.551011-10": "", - "ChildDevice.index.551011-11": "", - "ChildDevice.index.551011-12": "", - "ChildDevice.index.551011-13": "", - "ChildDevice.index.551011-14": "", - "ChildDevice.index.551011-15": "", - "ChildDevice.index.551011-16": "", - "ChildDevice.index.551011-17": "", - "ChildDevice.index.551011-18": "", - "ChildDevice.index.551011-19": "", - "ChildDevice.index.551011-20": "", - "Status.index.551011-0": "", - "Status.index.551011-1": "", - "Status.index.551011-2": "", - "Status.index.551011-3": "", - "Status.index.551011-4": "", - "Status.index.551011-5": "", - "Status.index.551011-6": "", - "Status.index.551011-7": "", - "Status.index.551011-8": "", - "Status.index.551011-9": "", - "Status.index.551011-10": "", - "Status.index.551011-11": "", - "Status.index.551011-12": "", - "Status.index.551011-13": "", - "Status.index.551011-14": "", - "Status.index.551011-15": "", - "Status.index.551011-16": "", - "Status.index.551011-17": "", - "Status.index.551011-18": "", - "Status.index.551011-19": "", - "Status.index.551011-20": "", - "Status.index.551011-21": "", - "Status.index.551011-22": "", - "Status.index.551011-23": "", - "Status.index.551011-24": "", - "Status.index.551011-25": "", - "Status.index.551011-26": "", - "Status.index.551011-27": "", - "Status.index.551011-28": "", - "Status.index.551011-29": "", - "Status.index.551011-30": "", - "Status.index.551011-31": "", - "Status.index.551011-32": "", - "Status.index.551011-33": "", - "Status.index.551011-34": "", - "Status.index.551011-35": "", - "Status.index.551011-36": "", - "Status.index.551011-37": "", - "Status.index.551011-38": "", - "Status.index.551011-39": "", - "Status.index.551011-40": "", - "Status.index.551011-41": "", - "Status.index.551011-42": "", - "Status.index.551011-43": "", - "Status.index.551011-44": "", - "Status.index.551011-45": "", - "Status.index.551011-46": "", - "Status.index.551011-47": "", - "Status.index.551011-48": "", - "Status.index.551011-49": "", - "Status.index.551011-50": "", - "Status.index.551011-51": "", - "Status.index.551011-52": "", - "Status.index.551011-53": "", - "Status.index.551011-54": "", - "Status.index.551011-55": "", - "Status.index.551011-56": "", - "Status.index.551011-57": "", - "Status.index.551011-58": "", - "Status.index.551011-59": "", - "Status.index.551011-60": "", - "Status.index.551011-61": "", - "Status.index.551011-62": "", - "Status.index.551011-63": "", - "Status.index.551011-64": "", - "Status.index.551011-65": "", - "Status.index.551011-66": "", - "Status.index.551011-67": "", - "Status.index.551011-68": "", - "Status.index.551011-69": "", - "components.ModelButton.551013-0": "", - "Properties.StorageSetting.551013-0": "", - "Properties.StorageSetting.551013-1": "", - "Properties.StorageSetting.551013-2": "", - "Function.OutputParams.551013-0": "", - "Function.OutputParams.551013-1": "", - "Function.OutputParams.551013-2": "", - "Function.OutputParams.551013-3": "", - "Function.OutputParams.551013-4": "", - "Function.OutputParams.551013-5": "", - "Function.OutputParams.551013-6": "", - "Function.OutputParams.551013-7": "", - "Function.OutputParams.551013-8": "", - "Function.OutputParams.551013-9": "", - "Function.OutputParams.551013-10": "", - "Events.SelectColumn.551013-0": "", - "Import.index.551010-0": "", - "Import.index.551010-1": "", - "Import.index.551010-2": "", - "Import.index.551010-3": "", - "Import.index.551010-4": "", - "Import.index.551010-5": "", - "components.EditTable.626395-0": "Up to 64 characters can be entered", - "components.EditTable.626395-1": "Please enter {0}", - "components.EditTable.626395-2": "add to", - "components.EditTable.626395-3": "operation", - "Process.index.551010-0": "", - "Process.index.551010-1": "", - "Process.index.551010-2": "", - "Process.index.551010-3": "", - "Process.index.551010-4": "", - "Process.index.551010-5": "", - "Process.index.551010-6": "", - "Process.index.551010-7": "", - "Process.index.551010-8": "", - "Info.index.506529-0": "", - "Info.index.506529-1": "", - "Info.index.506529-2": "", - "Info.index.506529-3": "", - "Info.index.506529-4": "", - "Info.index.506529-5": "", - "Info.index.506529-6": "", - "Info.index.506529-7": "", - "Info.index.506529-8": "", - "Info.index.506529-9": "", - "Info.index.506529-10": "", - "Info.index.506529-11": "", - "Info.index.506529-12": "", - "Info.index.506529-13": "", - "Info.index.506529-14": "", - "Info.index.506529-15": "", - "Info.index.506529-16": "", - "Save.GateWayFormItem.366202-0": "", - "Save.GateWayFormItem.366202-1": "", - "Save.GateWayFormItem.340184-0": "", - "Save.GateWayFormItem.340184-1": "", - "DeviceHome.index.152181-0": "", - "DeviceHome.index.152181-1": "", - "DeviceHome.index.152181-2": "", - "DeviceHome.index.152181-3": "", - "DeviceHome.index.152181-4": "", - "DeviceHome.index.152181-5": "", - "DeviceHome.index.152181-6": "", - "DeviceHome.index.152181-7": "", - "DeviceHome.index.152181-8": "", - "DeviceHome.index.152181-9": "", - "DeviceHome.index.152181-10": "", - "DeviceHome.index.152181-11": "", - "DeviceHome.index.152181-12": "", - "DeviceHome.index.152181-13": "", - "DeviceHome.index.152181-14": "", - "InitHome.index.152181-0": "", - "InitHome.index.152181-1": "", - "ComprehensiveHome.index.152181-0": "", - "ComprehensiveHome.index.152181-1": "", - "ComprehensiveHome.index.152181-2": "", - "ComprehensiveHome.index.152181-3": "", - "ComprehensiveHome.index.152181-4": "", - "ComprehensiveHome.index.152181-5": "", - "ComprehensiveHome.index.152181-6": "", - "ComprehensiveHome.index.152181-7": "", - "ComprehensiveHome.index.152181-8": "", - "ComprehensiveHome.index.152181-9": "", - "ComprehensiveHome.index.152181-10": "", - "ComprehensiveHome.index.152181-11": "", - "ComprehensiveHome.index.152181-12": "", - "ComprehensiveHome.index.152181-13": "", - "ComprehensiveHome.index.152181-14": "", - "ComprehensiveHome.index.152181-15": "", - "ComprehensiveHome.index.152181-16": "", - "ComprehensiveHome.index.152181-17": "", - "ComprehensiveHome.index.152181-18": "", - "ComprehensiveHome.index.152181-19": "", - "ComprehensiveHome.index.152181-20": "", - "ComprehensiveHome.index.152181-21": "", - "ComprehensiveHome.index.152181-22": "", - "ComprehensiveHome.index.152181-23": "", - "ComprehensiveHome.index.152181-24": "", - "ComprehensiveHome.index.152181-25": "", - "ComprehensiveHome.index.152181-26": "", - "ComprehensiveHome.index.152181-27": "", - "ComprehensiveHome.index.152181-28": "", - "ComprehensiveHome.index.152181-29": "", - "ComprehensiveHome.index.152181-30": "", - "dialogs.DeviceChooseDialog.152181-0": "", - "dialogs.DeviceChooseDialog.152181-1": "", - "dialogs.DeviceChooseDialog.152181-2": "", - "dialogs.DeviceChooseDialog.152181-3": "", - "dialogs.DeviceChooseDialog.152181-4": "", - "dialogs.DeviceChooseDialog.152181-5": "", - "dialogs.DeviceChooseDialog.152181-6": "", - "dialogs.DeviceChooseDialog.152181-7": "", - "dialogs.DeviceChooseDialog.152181-8": "", - "components.BootCard.152181-0": "", - "components.BasicCountCard.152181-0": "", - "components.BasicCountCard.152181-1": "", - "components.BasicCountCard.152181-2": "", - "components.BasicCountCard.152181-3": "", - "components.BootCardSmall.152181-0": "", - "components.StepCard.152181-0": "", - "dialogs.ProductChooseDialog.152181-0": "", - "dialogs.ProductChooseDialog.152181-1": "", - "dialogs.ProductChooseDialog.152181-2": "", - "home.index.152180-0": "", - "components.PlatformPicCard.152181-0": "", - "components.DeviceCountCard.152181-0": "", - "components.DeviceCountCard.152181-1": "", - "components.DeviceCountCard.152181-2": "", - "components.DeviceCountCard.152181-3": "", - "Save.GateWayDeviceModal.039083-0": "", - "Save.GateWayDeviceModal.039083-1": "", - "Save.GateWayDeviceModal.039083-2": "", - "Save.GateWayDeviceModal.039083-3": "", - "Save.GateWayDeviceModal.039083-4": "", - "Save.GateWayDeviceModal.039083-5": "", - "Save.GateWayDeviceModal.039083-6": "", - "Save.GateWayDeviceModal.039083-7": "", - "Save.GateWayDeviceModal.039083-8": "", - "Save.index.039083-0": "", - "Save.index.039083-1": "", - "Save.index.039083-2": "", - "Save.index.039083-3": "", - "Save.index.039083-4": "", - "Save.index.039083-5": "", - "Save.index.039083-6": "", - "Save.index.039083-7": "", - "Save.index.039083-8": "", - "Save.index.039083-9": "", - "Save.index.039083-10": "", - "Save.index.039083-11": "", - "Save.index.039083-12": "", - "Save.index.039083-13": "", - "Save.index.039083-14": "", - "Save.index.039083-15": "", - "Save.index.039083-16": "", - "Save.index.039083-17": "", - "Save.index.039083-18": "", - "Save.index.039083-19": "", - "Save.index.039083-20": "", - "Save.index.039083-21": "", - "Save.index.039083-22": "", - "Save.index.039083-23": "", - "Save.index.039083-24": "", - "Save.index.039083-25": "", - "Save.index.039083-26": "", - "Save.index.039083-27": "", - "Save.index.039083-28": "", - "Save.index.039083-29": "", - "Save.index.039083-30": "", - "Save.index.039083-31": "", - "Save.index.039083-32": "", - "Save.index.039083-33": "", - "Save.index.039083-34": "", - "Save.index.039083-35": "", - "Save.index.039083-36": "", - "Save.index.039083-37": "", - "Center.index.340051-0": "", - "Center.index.340051-1": "", - "Center.index.340051-2": "", - "Center.index.340051-3": "", - "Center.index.340051-4": "", - "Center.index.340051-5": "", - "Center.index.340051-6": "", - "Center.index.340051-7": "", - "Center.index.361874-0": "", - "Center.index.361874-1": "", - "Center.index.361874-2": "", - "Center.index.361874-3": "", - "Center.index.361874-4": "", - "Center.index.361874-5": "", - "Center.index.361874-6": "", - "Center.index.361874-7": "", - "Import.index.657743-0": "", - "Import.index.657743-1": "", - "Import.index.657743-2": "", - "Import.index.657743-3": "", - "Import.index.657743-4": "", - "Import.index.657743-5": "", - "Import.index.657743-6": "", - "Import.index.657743-7": "", - "Import.index.657743-8": "", - "Import.index.657743-9": "", - "Import.index.657743-10": "", - "Import.index.657743-11": "", - "Import.index.657743-12": "", - "Import.index.657743-13": "", - "Import.index.657743-14": "", - "Import.index.657743-15": "", - "Import.index.657743-16": "", - "Import.index.657743-17": "", - "Import.index.657743-18": "", - "Import.index.657743-19": "", - "Import.index.657743-20": "", - "Import.index.657743-21": "", - "Import.index.657743-22": "", - "Import.index.657743-23": "", - "Import.index.657743-24": "", - "Import.index.657743-25": "", - "Import.index.657743-26": "", - "Import.index.657743-27": "", - "Import.index.657743-28": "", - "Import.index.657743-29": "", - "Import.index.657743-30": "", - "Import.index.657743-31": "", - "Import.index.657743-32": "", - "Import.index.657743-33": "", - "Import.index.657743-34": "", - "Import.index.657743-35": "", - "Import.index.657743-36": "", - "Import.index.657743-37": "", - "Import.index.657743-38": "", - "Import.index.657743-39": "", - "Import.index.657743-40": "", - "Import.index.657743-41": "", - "Import.index.657743-42": "", - "Import.index.657743-43": "", - "Import.index.657743-44": "", - "Import.index.657743-45": "", - "Import.index.657743-46": "", - "Import.index.657743-47": "", - "Import.index.657743-48": "", - "Import.index.657743-49": "", - "Import.index.657743-50": "", - "Import.index.657743-51": "", - "Import.index.657743-52": "", - "Import.index.657743-53": "", - "Import.index.657743-54": "", - "Import.index.657743-55": "", - "Import.index.657743-56": "", - "Import.index.657743-57": "", - "Import.index.657743-58": "", - "Import.index.657743-59": "", - "Import.index.657743-60": "", - "ComprehensiveHome.index.926510-0": "IoT guidance", - "ComprehensiveHome.index.926510-1": "Operations guidance", - "ComprehensiveHome.index.926510-2": "Recommended steps for device access", - "ComprehensiveHome.index.926510-3": "Different devices may have different access steps due to differences in communication protocols", - "ComprehensiveHome.index.926510-4": "Recommended steps for operation and maintenance management", - "ComprehensiveHome.index.926510-5": "Please perform selective operations on the following steps according to business needs.", - "ComprehensiveHome.index.926510-6": "Create product", - "ComprehensiveHome.index.926510-7": "Create device", - "ComprehensiveHome.index.926510-8": "Rule engine", - "ComprehensiveHome.index.926510-9": "A product is a collection of devices, usually referring to a group of devices with the same functionality. IoT devices must be configured through product access methods.", - "ComprehensiveHome.index.926510-10": "Configure product access methods", - "ComprehensiveHome.index.926510-11": "Configure unified access methods for devices of the same type through products. Please refer to the instructions on the device nameplate to select the matching access method.", - "ComprehensiveHome.index.926510-12": "Add testing equipment", - "ComprehensiveHome.index.926510-13": "Add a single device to verify if the product model is configured correctly.", - "ComprehensiveHome.index.926510-14": "Functional debugging", - "ComprehensiveHome.index.926510-15": "Perform functional debugging on the added testing equipment to verify if it can connect to the platform and if the device's functions are configured correctly.", - "ComprehensiveHome.index.926510-16": "Batch Add Devices", - "ComprehensiveHome.index.926510-17": "Batch adding devices under the same product", - "ComprehensiveHome.index.926510-18": "Device access configuration", - "ComprehensiveHome.index.926510-19": "Log troubleshooting", - "ComprehensiveHome.index.926510-20": "Real time monitoring", - "ComprehensiveHome.index.926510-21": "Protocol management", - "ComprehensiveHome.index.926510-22": "Customize and develop corresponding product (device model) access protocols according to business needs, and upload them to the platform.", - "ComprehensiveHome.index.926510-23": "Certificate Management", - "ComprehensiveHome.index.926510-24": "Unified maintenance of certificates within the platform for data communication encryption.", - "ComprehensiveHome.index.926510-25": "Network components", - "ComprehensiveHome.index.926510-26": "Configure platform underlying network component related parameters based on different transmission types.", - "ComprehensiveHome.index.926510-27": "Device access gateway", - "ComprehensiveHome.index.926510-28": "According to different transmission types, associate message protocols, and configure device access gateway related parameters.", - "ComprehensiveHome.index.926510-29": "Log management", - "ComprehensiveHome.index.926510-30": "Monitor system logs and promptly handle system exceptions.", - "components.BootCardSmall.926510-0": "No permission at the moment, please contact the administrator", - "components.BasicCountCard.926510-0": "Basic statistics", - "components.BasicCountCard.926510-1": "details", - "components.BasicCountCard.926510-2": "CPU usage rate", - "components.BasicCountCard.926510-3": "JVM memory", - "components.BootCard.926510-0": "No permission at the moment, please contact the administrator", - "dialogs.DeviceChooseDialog.926510-0": "Select device", - "dialogs.DeviceChooseDialog.926510-1": "Please select a device", - "dialogs.DeviceChooseDialog.926510-2": "Device ID", - "dialogs.DeviceChooseDialog.926510-3": "Device Name", - "dialogs.DeviceChooseDialog.926510-4": "Product Name", - "dialogs.DeviceChooseDialog.926510-5": "Registration time", - "dialogs.DeviceChooseDialog.926510-6": "state", - "dialogs.DeviceChooseDialog.926510-7": "on-line", - "dialogs.DeviceChooseDialog.926510-8": "off-line", - "DevOpsHome.index.926510-0": "Operations guidance", - "DevOpsHome.index.926510-1": "Recommended steps for operation and maintenance management", - "DevOpsHome.index.926510-2": "Please perform selective operations on the following steps according to business needs.", - "DevOpsHome.index.926510-3": "Device access configuration", - "DevOpsHome.index.926510-4": "Log troubleshooting", - "DevOpsHome.index.926510-5": "Real time monitoring", - "DevOpsHome.index.926510-6": "Protocol management", - "DevOpsHome.index.926510-7": "Customize and develop corresponding product (device model) access protocols according to business needs, and upload them to the platform.", - "DevOpsHome.index.926510-8": "Certificate Management", - "DevOpsHome.index.926510-9": "Unified maintenance of certificates within the platform for data communication encryption.", - "DevOpsHome.index.926510-10": "Network components", - "DevOpsHome.index.926510-11": "Configure platform underlying network component related parameters based on different transmission types.", - "DevOpsHome.index.926510-12": "Device access gateway", - "DevOpsHome.index.926510-13": "According to different transmission types, associate message protocols, and configure device access gateway related parameters.", - "DevOpsHome.index.926510-14": "Log management", - "DevOpsHome.index.926510-15": "Monitor system logs and promptly handle system exceptions.", - "dialogs.ProductChooseDialog.926510-0": "Select product", - "dialogs.ProductChooseDialog.926510-1": "product", - "dialogs.ProductChooseDialog.926510-2": "Please select a product", - "components.DeviceCountCard.926510-0": "Equipment statistics", - "components.DeviceCountCard.926510-1": "details", - "components.DeviceCountCard.926510-2": "Product quantity", - "components.DeviceCountCard.926510-3": "Number of devices", - "home.index.926510-0": "essential information", - "components.PlatformPicCard.926510-0": "Platform architecture diagram", - "InitHome.index.926510-0": "Please select the homepage view", - "InitHome.index.926510-1": "Save modifications", - "DeviceHome.index.926510-0": "IoT guidance", - "DeviceHome.index.926510-1": "Recommended steps for device access", - "DeviceHome.index.926510-2": "Different devices may have different access steps due to differences in communication protocols", - "DeviceHome.index.926510-3": "Create product", - "DeviceHome.index.926510-4": "Create device", - "DeviceHome.index.926510-5": "Rule engine", - "DeviceHome.index.926510-6": "A product is a collection of devices, usually referring to a group of devices with the same functionality. IoT devices must be configured through product access methods.", - "DeviceHome.index.926510-7": "Configure product access methods", - "DeviceHome.index.926510-8": "Configure unified access methods for devices of the same type through products. Please refer to the instructions on the device nameplate to select the matching access method.", - "DeviceHome.index.926510-9": "Add testing equipment", - "DeviceHome.index.926510-10": "Add a single device to verify if the product model is configured correctly.", - "DeviceHome.index.926510-11": "Functional debugging", - "DeviceHome.index.926510-12": "Perform functional debugging on the added testing equipment to verify if it can connect to the platform and if the device's functions are configured correctly.", - "DeviceHome.index.926510-13": "Batch Add Devices", - "DeviceHome.index.926510-14": "Batch adding devices under the same product", - "components.StepCard.926510-0": "No permission at the moment, please contact the administrator" -} \ No newline at end of file diff --git a/src/i18n/locale/zh.json b/src/i18n/locale/zh.json deleted file mode 100644 index 8890ca9..0000000 --- a/src/i18n/locale/zh.json +++ /dev/null @@ -1,1773 +0,0 @@ -{ - "Login.index.265048-0": "备案:", - "Login.index.265048-1": "渝ICP备19017719号-1", - "Login.index.265048-2": "账号", - "Login.index.265048-3": "请输入账号", - "Login.index.265048-4": "密码", - "Login.index.265048-5": "请输入密码", - "Login.index.265048-6": "验证码", - "Login.index.265048-7": "请输入验证码", - "Login.index.265048-8": "记住我", - "Login.index.265048-9": "登录", - "Login.index.265048-10": "其他登录方式", - "Login.index.265048-11": "JETLINKS团队全新力作可视化大屏系统", - "Login.index.265048-12": "体验DEMO", - "Login.index.265048-13": "请输入账号!", - "Login.index.265048-14": "请输入密码!", - "Login.index.265048-15": "请输入验证码!", - "BooleanParam.index.966482-0": "请输入trueText", - "BooleanParam.index.966482-1": "最多可输入64个字符", - "BooleanParam.index.966482-2": "请输入trueValue", - "BooleanParam.index.966482-3": "请输入falseText", - "BooleanParam.index.966482-4": "请输入falseValue", - "BooleanParam.index.966482-5": "是", - "BooleanParam.index.966482-6": "否", - "Save.GateWayFormItem.039083-0": "选择网关设备", - "Save.GateWayFormItem.039083-1": "重新选择", - "DevOpsHome.index.152181-0": "运维引导", - "DevOpsHome.index.152181-1": "运维管理推荐步骤", - "DevOpsHome.index.152181-2": "请根据业务需要对下述步骤进行选择性操作。", - "DevOpsHome.index.152181-3": "设备接入配置", - "DevOpsHome.index.152181-4": "日志排查", - "DevOpsHome.index.152181-5": "实时监控", - "DevOpsHome.index.152181-6": "协议管理", - "DevOpsHome.index.152181-7": "根据业务需求自定义开发对应的产品(设备模型)接入协议,并上传到平台。", - "DevOpsHome.index.152181-8": "证书管理", - "DevOpsHome.index.152181-9": "统一维护平台内的证书,用于数据通信加密。", - "DevOpsHome.index.152181-10": "网络组件", - "DevOpsHome.index.152181-11": "根据不同的传输类型配置平台底层网络组件相关参数。", - "DevOpsHome.index.152181-12": "设备接入网关", - "DevOpsHome.index.152181-13": "根据不同的传输类型,关联消息协议,配置设备接入网关相关参数。", - "DevOpsHome.index.152181-14": "日志管理", - "DevOpsHome.index.152181-15": "监控系统日志,及时处理系统异常。", - "Task2.index.551010-0": "升级任务", - "Task2.index.551010-1": "+ 新增任务", - "Task2.index.551010-2": "完成比例", - "Task2.index.551010-3": "任务详情", - "Task2.index.551010-4": "响应超时时间", - "Task2.index.551010-5": "升级超时时间", - "Task2.index.551010-6": "升级设备", - "Task2.index.551010-7": "选择设备", - "Task2.index.551010-8": "所有设备", - "Task2.index.551010-9": "说明", - "Task2.index.551010-10": "操作成功", - "Detail.index.551010-0": "任务详情", - "Detail.index.551010-1": "全部暂停", - "Detail.index.551010-2": "全部开始", - "Detail.index.551010-3": "批量重试", - "Detail.index.551010-4": "刷新状态", - "Detail.index.551010-5": "当前进度", - "Detail.index.551010-6": "共", - "Detail.index.551010-7": "个任务", - "Detail.index.551010-8": "关闭弹窗不会影响任务执行状态", - "Detail.index.551010-9": "关闭", - "Detail.index.551010-10": "设备名称", - "Detail.index.551010-11": "所属产品", - "Detail.index.551010-12": "创建时间", - "Detail.index.551010-13": "完成时间", - "Detail.index.551010-14": "状态", - "Detail.index.551010-15": "操作成功", - "Task.index.551010-0": "新增", - "Task.index.551010-1": "任务名称", - "Task.index.551010-2": "推送方式", - "Task.index.551010-3": "设备拉取", - "Task.index.551010-4": "平台推送", - "Task.index.551010-5": "说明", - "Task.index.551010-6": "完成比例", - "Task.index.551010-7": "操作", - "Task.index.551010-8": "详情", - "Task.index.551010-9": "查看", - "Task.index.551010-10": "停止", - "Task.index.551010-11": "操作成功", - "Task.index.551010-12": "继续升级", - "Save.index.551010-0": "编辑", - "Save.index.551010-1": "新增", - "Save.index.551010-2": "确认", - "Save.index.551010-3": "取消", - "Save.index.551010-4": "名称", - "Save.index.551010-5": "请输入名称", - "Save.index.551010-6": "所属产品", - "Save.index.551010-7": "请选择所属产品", - "Save.index.551010-8": "版本号", - "Save.index.551010-9": "请输入版本号", - "Save.index.551010-10": "版本序号", - "Save.index.551010-11": "请输入版本序号", - "Save.index.551010-12": "签名方式", - "Save.index.551010-13": "请选择签名方式", - "Save.index.551010-14": "签名", - "Save.index.551010-15": "请输入本地文件进行签名加密后的值", - "Save.index.551010-16": "请输入签名", - "Save.index.551010-17": "固件上传", - "Save.index.551010-18": "其他配置", - "Save.index.551010-19": "请输入KEY", - "Save.index.551010-20": "请输入VALUE", - "Save.index.551010-21": "操作", - "Save.index.551010-22": "确认删除吗?", - "Save.index.551010-23": "添加", - "Save.FileUpload.551010-0": "请上传文件", - "Save.FileUpload.551010-1": "上传文件", - "Save.FileUpload.551010-2": "上传成功!", - "Save.FileUpload.551010-3": "系统提示", - "Save.FileUpload.551010-4": "系统未知错误,请反馈给管理员", - "Save.index.551010-24": "说明", - "Save.index.551010-25": "请输入说明", - "Save.index.551010-26": "签名不一致,请检查文件是否上传正确", - "Save.index.551010-27": "版本序号已存在", - "Save.index.551010-28": "当前产品不存在,请选择产品", - "Save.index.551010-29": "请输入1~99999之间的正整数", - "Save.index.551010-30": "最多可输入64个字符", - "Save.index.551010-31": "请上传文件", - "Save.index.551010-32": "最多可输入200个字符", - "Save.index.551010-33": "操作成功", - "Save.index.551009-0": "确定", - "Save.index.551009-1": "取消", - "Save.index.551009-2": "若不填写,系统将自动生成唯一ID", - "Save.index.551009-3": "请输入ID", - "Save.index.551009-4": "名称", - "Save.index.551009-5": "请输入名称", - "Save.index.551009-6": "产品分类", - "Save.index.551009-7": "请选择产品分类", - "Save.index.551009-8": "设备类型", - "Save.index.551009-9": "说明", - "Save.index.551009-10": "请输入说明", - "Save.index.551009-11": "直连设备", - "Save.index.551009-12": "直连物联网平台的设备", - "Save.index.551009-13": "网关子设备", - "Save.index.551009-14": "作为网关的子设备,由网关代理连接到物联网平台", - "Save.index.551009-15": "网关设备", - "Save.index.551009-16": "能挂载子设备与平台进行通信的设备", - "Save.index.551009-17": "请输入英文或者数字或者-或者_", - "Save.index.551009-18": "ID重复", - "Save.index.551009-19": "请选择设备类型", - "Save.index.551009-20": "最多可输入64位字符", - "Save.index.551009-21": "最多可输入200位字符", - "Save.index.551009-22": "保存成功!", - "Save.index.551009-23": "操作失败", - "Save.SelectDevices.551010-0": "请选择设备", - "Save.SelectDevices.551010-1": "选择设备", - "Save.SelectDevices.551010-2": "确认", - "Save.SelectDevices.551010-3": "取消", - "Save.SelectDevices.551010-4": "全选", - "Save.SelectDevices.551010-5": "设备名称", - "Save.SelectDevices.551010-6": "固件版本", - "Save.SelectDevices.551010-7": "注册时间", - "Save.SelectDevices.551010-8": "状态", - "Save.SelectDevices.551010-9": "在线", - "Save.SelectDevices.551010-10": "离线", - "Save.SelectDevices.551010-11": "禁用", - "Detail.PropertyAMap.551011-0": "开始动画", - "Detail.PropertyAMap.551011-1": "暂停动画", - "Detail.PropertyAMap.551011-2": "继续动画", - "Instance.index.551009-0": "新增", - "Instance.index.551009-1": "设备类型", - "Instance.index.551009-2": "产品名称", - "Instance.index.551009-3": "设备名称", - "Instance.index.551009-4": "创建时间", - "Instance.index.551009-5": "状态", - "Instance.index.551009-6": "禁用", - "Instance.index.551009-7": "离线", - "Instance.index.551009-8": "在线", - "Instance.index.551009-9": "产品分类", - "Instance.index.551009-10": "网关类型", - "Instance.index.551009-11": "接入方式", - "Instance.index.551009-12": "直连设备", - "Instance.index.551009-13": "网关子设备", - "Instance.index.551009-14": "网关设备", - "Instance.index.551009-15": "所属组织", - "Instance.index.551009-16": "设备标签", - "Instance.index.551009-17": "说明", - "Instance.index.551009-18": "操作", - "Instance.index.551009-19": "查看", - "Instance.index.551009-20": "编辑", - "Instance.index.551009-21": "启用", - "Instance.index.551009-22": "确认${\r\n data.state.value !== 'notActive' ? '禁用' : '启用'\r\n }?", - "Instance.index.551009-23": "操作成功!", - "Instance.index.551009-24": "操作失败!", - "Instance.index.551009-25": "删除", - "Instance.index.551009-26": "已启用的设备不能删除", - "Instance.index.551009-27": "确认删除?", - "Instance.index.551009-28": "请选择设备", - "Instance.index.551009-29": "批量导出设备", - "Instance.index.551009-30": "批量导入设备", - "Instance.index.551009-31": "启用全部设备", - "Instance.index.551009-32": "确认启用全部设备?", - "Instance.index.551009-33": "同步设备状态", - "Instance.index.551009-34": "批量删除设备", - "Instance.index.551009-35": "已启用的设备无法删除,确认删除选中的禁用状态设备?", - "Instance.index.551009-36": "批量禁用设备", - "Instance.index.551009-37": "确认禁用选中设备?", - "DeviceAccess.index.551011-0": "请先", - "DeviceAccess.index.551011-1": "选择", - "DeviceAccess.index.551011-2": "设备接入网关,用以提供设备接入能力", - "DeviceAccess.index.551011-3": "请联系管理员配置产品接入方式", - "DeviceAccess.index.551011-4": "接入方式", - "DeviceAccess.index.551011-5": "更换", - "DeviceAccess.index.551011-6": "消息协议", - "DeviceAccess.index.551011-7": "连接信息", - "DeviceAccess.index.551011-8": "暂无连接信息", - "DeviceAccess.index.551011-9": "产品类型", - "DeviceAccess.index.551011-10": "请选择产品类型", - "DeviceAccess.index.551011-11": "此配置来自于产品接入方式所选择的协议", - "DeviceAccess.index.551011-12": "请选择", - "DeviceAccess.index.551011-13": "请输入", - "DeviceAccess.index.551011-14": "存储策略", - "DeviceAccess.index.551011-15": "若修改存储策略,需要手动做数据迁移,平台只能搜索最新存储策略中的数据", - "DeviceAccess.index.551011-16": "保存", - "DeviceAccess.index.551011-17": "URL信息", - "DeviceAccess.index.551011-18": "我知道了", - "DeviceAccess.index.551011-19": "不再提示", - "DeviceAccess.index.551011-20": "下一步", - "DeviceAccess.index.551011-21": "上一步", - "DeviceAccess.index.551011-22": "上下行", - "DeviceAccess.index.551011-23": "说明", - "DeviceAccess.index.551011-24": "地址", - "DeviceAccess.index.551011-25": "示例", - "DeviceAccess.index.551011-26": "上行", - "DeviceAccess.index.551011-27": "下行", - "DeviceAccess.index.551011-28": "分组", - "DeviceAccess.index.551011-29": "流传输模式", - "DeviceAccess.index.551011-30": "操作成功!", - "DeviceAccess.index.551011-31": "产品下有设备实例时不能更换接入方式", - "DeviceAccess.index.551011-32": "停用产品后才可更换接入方式", - "Firmware.index.551009-0": "新增", - "Firmware.index.551009-1": "固件名称", - "Firmware.index.551009-2": "固件版本", - "Firmware.index.551009-3": "所属产品", - "Firmware.index.551009-4": "签名方式", - "Firmware.index.551009-5": "创建时间", - "Firmware.index.551009-6": "说明", - "Firmware.index.551009-7": "操作", - "Firmware.index.551009-8": "升级任务", - "Firmware.index.551009-9": "编辑", - "Firmware.index.551009-10": "删除", - "Firmware.index.551009-11": "确认删除?", - "Firmware.index.551009-12": "确定", - "Firmware.index.551009-13": "取消", - "Firmware.index.551009-14": "操作成功", - "Property.Indicators.551010-0": "编辑指标", - "Property.Indicators.551010-1": "场景联动页面可引用指标配置触发条件", - "Property.Indicators.551010-2": "请${\r\n ['date', 'boolean'].includes(\r\n data?.valueType?.type,\r\n )\r\n ? '选择'\r\n : '输入'\r\n }指标值", - "Property.Indicators.551010-3": "指标值", - "Property.Indicators.551010-4": "请${\r\n ['date', 'boolean'].includes(\r\n data?.valueType?.type,\r\n )\r\n ? '选择'\r\n : '输入'\r\n }指标值", - "Property.Indicators.551010-5": "操作成功!", - "Detail.TimeComponent.551010-0": "今日", - "Detail.TimeComponent.551010-1": "近一周", - "Detail.TimeComponent.551010-2": "近一月", - "Detail.TimeComponent.551010-3": "开始时间", - "Detail.TimeComponent.551010-4": "结束时间", - "Event.index.551010-0": "详情", - "Event.index.551010-1": "关闭", - "Event.index.551010-2": "时间", - "Event.index.551010-3": "操作", - "Event.index.551010-4": "数据", - "Detail.Table.551010-0": "详情", - "Detail.Table.551010-1": "自定义属性", - "Detail.Table.551010-2": "时间", - "Detail.Table.551010-3": "操作", - "Detail.Table.551010-4": "二进制", - "Detail.index.551011-0": "详情", - "Detail.index.551011-1": "列表", - "Detail.index.551011-2": "图表", - "Detail.index.551011-3": "轨迹", - "DeviceAccess.accessModal.551011-0": "设备接入配置", - "DeviceAccess.accessModal.551011-1": "确定", - "DeviceAccess.accessModal.551011-2": "取消", - "DeviceAccess.accessModal.551011-3": "新增", - "DeviceAccess.accessModal.551011-4": "协议", - "DeviceAccess.accessModal.551011-5": "名称", - "DeviceAccess.accessModal.551011-6": "网关类型", - "DeviceAccess.accessModal.551011-7": "状态", - "DeviceAccess.accessModal.551011-8": "正常", - "DeviceAccess.accessModal.551011-9": "禁用", - "DeviceAccess.accessModal.551011-10": "说明", - "DeviceAccess.accessModal.551011-11": "请选择接入方式", - "components.TagSearch.551010-0": "筛选条件", - "components.TagSearch.551010-1": "设备标签:", - "components.TagSearch.551010-2": "请输入标签key", - "components.TagSearch.551010-3": "请输入标签value", - "components.TagSearch.551010-4": "添加", - "device.data.551010-0": "整数型", - "device.data.551010-1": "长整数型", - "device.data.551010-2": "单精度浮点型", - "device.data.551010-3": "双精度浮点数", - "device.data.551010-4": "字符串", - "device.data.551010-5": "布尔型", - "device.data.551010-6": "时间型", - "device.data.551010-7": "枚举", - "device.data.551010-8": "数组", - "device.data.551010-9": "结构体", - "device.data.551010-10": "文件", - "device.data.551010-11": "密码", - "device.data.551010-12": "地理位置", - "device.data.551010-13": "设备", - "device.data.551010-14": "手动", - "device.data.551010-15": "规则", - "device.data.551010-16": "链接", - "device.data.551010-17": "Base64编码", - "device.data.551010-18": "二进制", - "device.data.551010-19": "普通", - "device.data.551010-20": "警告", - "device.data.551010-21": "紧急", - "device.data.551010-22": "读", - "device.data.551010-23": "写", - "device.data.551010-24": "上报", - "Import.modal.551010-0": "批量导入", - "Import.modal.551010-1": "选择导入方式", - "Import.modal.551010-2": "取消", - "Import.modal.551010-3": "上一步", - "Import.modal.551010-4": "下一步", - "Import.modal.551010-5": "确认", - "Import.modal.551010-6": "导入完成", - "Import.modal.551010-7": "已完成 新增设备", - "Import.modal.551010-8": "文件导入", - "Import.modal.551010-9": "支持上传XLSX、CSV格式文件", - "Import.modal.551010-10": "插件导入", - "Import.modal.551010-11": "读取插件中的设备信息同步至平台", - "Import.modal.551010-12": "请选择产品", - "Import.modal.551010-13": "请选择导入方式", - "Import.modal.551010-14": "操作成功", - "Import.modal.551010-15": "请选择设备", - "MetadataMap.index.551011-0": "请选择属性名称", - "MetadataMap.index.551011-1": "目标属性", - "MetadataMap.index.551011-2": "协议包中物模型下的属性", - "MetadataMap.index.551011-3": "已映射", - "MetadataMap.index.551011-4": "未映射", - "MetadataMap.index.551011-5": "置顶已映射数据", - "MetadataMap.index.551011-6": "置顶未映射数据", - "MetadataMap.index.551011-7": "请选择", - "MetadataMap.index.551011-8": "功能说明", - "MetadataMap.index.551011-9": "该功能用于将协议包中的", - "MetadataMap.index.551011-10": "物模型属性标识", - "MetadataMap.index.551011-11": "与", - "MetadataMap.index.551011-12": "平台物模型属性标识", - "MetadataMap.index.551011-13": "进行映射,当两方属性标识不一致时,可在当前页面直接修改映射管理,系统将以映射后的物模型属性进行数据处理。", - "MetadataMap.index.551011-14": "未完成映射的属性标识“目标属性”列数据为空,代表该属性值来源以在平台配置的来源为准。", - "MetadataMap.index.551011-15": "数据条背景亮起代表", - "MetadataMap.index.551011-16": "标识一致", - "MetadataMap.index.551011-17": "或", - "MetadataMap.index.551011-18": "已完成映射", - "MetadataMap.index.551011-19": "的属性。", - "MetadataMap.index.551011-20": "功能图示", - "MetadataMap.index.551011-21": "序号", - "MetadataMap.index.551011-22": "平台属性", - "MetadataMap.index.551011-23": "操作成功", - "MetadataMap.index.551011-24": "功能图示", - "MetadataMap.index.551011-25": "序号", - "MetadataMap.index.551011-26": "平台属性", - "MetadataMap.index.551011-27": "操作成功", - "Import.file.551010-0": "文件上传", - "Import.file.551010-1": "导入系统已存在的设备数据,不会更改已存在设备的所属产品信息", - "Import.file.551010-2": "点击或拖拽上传文件", - "Import.file.551010-3": "格式:.xlsx, .csv", - "Import.file.551010-4": "导入并启用", - "Import.file.551010-5": "正在导入", - "Import.file.551010-6": "导入完成", - "Import.file.551010-7": "导入成功:", - "Import.file.551010-8": "个", - "Import.file.551010-9": "导入失败:", - "Import.file.551010-10": "下载", - "Import.file.551010-11": "下载模板", - "Import.file.551010-12": "模板格式.xlsx", - "Import.file.551010-13": "模板格式.csv", - "Import.file.551010-14": "设备导入模板", - "Import.file.551010-15": "请上传.xlsx或.csv格式文件", - "Import.file.551010-16": "暂无权限,请联系管理员", - "Import.file.551010-17": "请先上传文件", - "Running.index.551010-0": "请输入事件名称", - "Running.index.551010-1": "属性", - "Detail.index.551010-16": "物联网卡", - "Detail.index.551010-17": "设备诊断", - "Property.Save.551010-0": "编辑", - "Property.Save.551010-1": "当数据来源为设备时,填写的值将下发到设备", - "Property.Save.551010-2": "自定义属性", - "Property.Save.551010-3": "该字段是必填字段", - "Property.Save.551010-4": "操作成功!", - "Detail.index.551010-18": "数据解析", - "Detail.index.551010-19": "数采映射", - "Detail.index.551010-20": "子设备", - "Detail.index.551010-21": "边缘端映射", - "Detail.index.551010-22": "物模型映射", - "Detail.index.551010-23": "操作成功!", - "Detail.index.551010-24": "操作成功", - "BasicInfo.indev.551011-0": "配置信息", - "BasicInfo.indev.551011-1": "产品分类", - "BasicInfo.indev.551011-2": "设备类型", - "BasicInfo.indev.551011-3": "接入方式", - "BasicInfo.indev.551011-4": "配置接入方式", - "BasicInfo.indev.551011-5": "创建时间", - "BasicInfo.indev.551011-6": "更新时间", - "BasicInfo.indev.551011-7": "说明", - "BasicInfo.indev.551011-8": "编辑", - "DataAnalysis.index.551011-0": "脚本语言:", - "DataAnalysis.index.551011-1": "模拟输入", - "DataAnalysis.index.551011-2": "请输入Topic", - "DataAnalysis.index.551011-3": "请输入URL", - "DataAnalysis.index.551011-4": "运行结果", - "DataAnalysis.index.551011-5": "需输入脚本和模拟数据后再点击", - "DataAnalysis.index.551011-6": "调试", - "DataAnalysis.index.551011-7": "请先调试", - "DataAnalysis.index.551011-8": "保存", - "DataAnalysis.index.551011-9": "保存成功", - "DataAnalysis.index.551011-10": "请输入topic", - "DataAnalysis.index.551011-11": "请输入url", - "Config.Save.551011-0": "编辑配置", - "Config.Save.551011-1": "保存", - "Config.Save.551011-2": "请输入{0}", - "Config.Save.551011-3": "操作成功!", - "Function.index.551011-0": "请配置对应产品的", - "Function.index.551011-1": "物模型属性功能", - "Function.index.551011-2": "精简模式", - "Function.index.551011-3": "高级模式", - "Config.index.551011-0": "配置", - "Config.index.551011-1": "编辑", - "Config.index.551011-2": "确认重新应用该配置?", - "Config.index.551011-3": "应用配置", - "Config.index.551011-4": "修改配置后需重新应用后才能生效。", - "Config.index.551011-5": "确认恢复默认配置?", - "Config.index.551011-6": "恢复默认", - "Config.index.551011-7": "该设备单独编辑过配置信息,点击此将恢复成默认的配置信息,请谨慎操作。", - "Config.index.551011-8": "有效值:${\r\n instanceStore.current?.configuration?.[\r\n item.property\r\n ]\r\n }", - "Config.index.551011-9": "操作成功", - "Config.index.551011-10": "恢复默认配置成功", - "Property.ValueRender.551010-0": "二进制", - "Property.ValueRender.551010-1": "域名为https时,不支持访问http地址", - "Property.ValueRender.551010-2": "该图片无法访问", - "Property.ValueRender.551010-3": "当前仅支持播放.mp4,.flv,.m3u8格式的视频", - "EditTable.index.551011-0": "批量映射", - "EditTable.index.551011-1": "保存", - "EditTable.index.551011-2": "采集器", - "EditTable.index.551011-3": "数据采集中配置的真实物理设备", - "EditTable.index.551011-4": "请选择", - "EditTable.index.551011-5": "请选择采集器", - "EditTable.index.551011-6": "请选择点位", - "EditTable.index.551011-7": "已绑定", - "EditTable.index.551011-8": "未绑定", - "EditTable.index.551011-9": "解绑", - "EditTable.index.551011-10": "确认解绑", - "EditTable.index.551011-11": "暂无数据,请配置物模型", - "EditTable.index.551011-12": "名称", - "EditTable.index.551011-13": "通道", - "EditTable.index.551011-14": "点位", - "EditTable.index.551011-15": "状态", - "EditTable.index.551011-16": "操作", - "EditTable.index.551011-17": "操作成功!", - "Status.DiagnosticAdvice.551011-0": "设备诊断", - "Status.DiagnosticAdvice.551011-1": "诊断建议", - "Status.DiagnosticAdvice.551011-2": "所有诊断均无异常但设备仍未上线,请检查以下内容", - "Status.DiagnosticAdvice.551011-3": "连接信息", - "Status.DiagnosticAdvice.551011-4": "设备ID", - "Status.DiagnosticAdvice.551011-5": "连接地址", - "components.MSelect.551011-0": "请选择", - "EditTable.PatchMapping.551011-0": "批量映射", - "EditTable.PatchMapping.551011-1": "采集器的点位名称与属性名称一致时将自动映射绑定;有多个采集器点位名称与属性名称一致时以第1个采集器的点位数据进行绑定", - "EditTable.PatchMapping.551011-2": "源数据", - "EditTable.PatchMapping.551011-3": "加入右侧", - "EditTable.PatchMapping.551011-4": "采集器", - "EditTable.PatchMapping.551011-5": "确定删除?", - "EditTable.PatchMapping.551011-6": "请选择采集器", - "EditTable.PatchMapping.551011-7": "操作成功", - "EditTable.PatchMapping.551011-8": "暂无对应属性的映射", - "BindParentDevice.index.551012-0": "绑定父设备", - "BindParentDevice.index.551012-1": "确定", - "BindParentDevice.index.551012-2": "取消", - "BindParentDevice.index.551012-3": "设备名称", - "BindParentDevice.index.551012-4": "所属产品", - "BindParentDevice.index.551012-5": "注册时间", - "BindParentDevice.index.551012-6": "状态", - "BindParentDevice.index.551012-7": "禁用", - "BindParentDevice.index.551012-8": "离线", - "BindParentDevice.index.551012-9": "在线", - "BindParentDevice.index.551012-10": "请选择需要绑定的设备", - "BindParentDevice.index.551012-11": "操作成功", - "Diagnose.index.551011-0": "已诊断", - "Diagnose.index.551011-1": "个", - "Diagnose.index.551011-2": "连接状态", - "Diagnose.index.551011-3": "消息通信", - "DeviceAccess.util.551011-0": "配置物模型", - "DeviceAccess.util.551011-1": "配置产品物模型,实现设备在云端的功能描述。", - "DeviceAccess.util.551011-2": "启用产品", - "DeviceAccess.util.551011-3": "启用产品后,可在产品下新增设备。", - "DeviceAccess.util.551011-4": "添加设备", - "DeviceAccess.util.551011-5": "添加设备,并连接到平台。", - "DeviceAccess.util.551011-6": "填写配置", - "DeviceAccess.util.551011-7": "填写设备接入所需的配置参数。", - "Detail.Charts.551011-0": "统计周期:", - "Detail.Charts.551011-1": "统计规则:", - "Detail.Charts.551011-2": "平均值", - "Detail.Charts.551011-3": "最大值", - "Detail.Charts.551011-4": "最小值", - "Detail.Charts.551011-5": "总数", - "Detail.Charts.551011-6": "时间", - "Detail.Charts.551011-7": "实际值", - "Detail.Charts.551011-8": "按分钟统计", - "Detail.Charts.551011-9": "按小时统计", - "Detail.Charts.551011-10": "按天统计", - "Detail.Charts.551011-11": "按周统计", - "DeviceAccess.metadataModal.551011-0": "选择处理方式", - "DeviceAccess.metadataModal.551011-1": "确定", - "DeviceAccess.metadataModal.551011-2": "取消", - "DeviceAccess.metadataModal.551011-3": "平台", - "DeviceAccess.metadataModal.551011-4": "物模型", - "DeviceAccess.metadataModal.551011-5": "中已有数据,请选择处理方式。", - "DeviceAccess.metadataModal.551011-6": "默认采用覆盖的方式处理功能、事件、标签下的数据", - "DeviceAccess.metadataModal.551011-7": "处理方式", - "DeviceAccess.metadataModal.551011-8": "请选择处理方式", - "DeviceAccess.metadataModal.551011-9": "取交集", - "DeviceAccess.metadataModal.551011-10": "仅保留标识一致的属性", - "DeviceAccess.metadataModal.551011-11": "取并集", - "DeviceAccess.metadataModal.551011-12": "保留平台、插件中的所有属性", - "DeviceAccess.metadataModal.551011-13": "忽略", - "DeviceAccess.metadataModal.551011-14": "仅保留平台中的属性", - "DeviceAccess.metadataModal.551011-15": "覆盖", - "DeviceAccess.metadataModal.551011-16": "仅保留插件中的属性", - "DeviceAccess.metadataModal.551011-17": "操作成功!", - "Relation.Save.551011-0": "编辑", - "Relation.Save.551011-1": "保存", - "Relation.Save.551011-2": "请选择{0}", - "Relation.Save.551011-3": "操作成功!", - "Tags.index.551011-0": "标签", - "Tags.index.551011-1": "编辑", - "Tags.Save.551011-0": "编辑标签", - "Tags.Save.551011-1": "名称", - "Tags.Save.551011-2": "值", - "Tags.Save.551011-3": "操作成功!", - "Relation.index.551011-0": "关系信息", - "Relation.index.551011-1": "编辑", - "Relation.index.551011-2": "管理设备与其他业务的关联关系,关系来源于关系配置", - "InklingModal.index.551011-0": "设备ID映射", - "InklingModal.index.551011-1": "请选择设备", - "Parsing.index.551010-0": "当前数据解析内容已脱离产品影响,", - "Parsing.index.551010-1": "重置", - "Parsing.index.551010-2": "后将继承产品数据解析内容", - "Parsing.index.551010-3": "当前数据解析内容继承自产品,", - "Parsing.index.551010-4": "修改", - "Parsing.index.551010-5": "后将脱离产品影响。", - "Parsing.index.551010-6": "脚本语言:", - "Parsing.index.551010-7": "请点击上方修改字样,用以编辑脚本", - "Parsing.index.551010-8": "模拟输入", - "Parsing.index.551010-9": "请输入Topic", - "Parsing.index.551010-10": "请输入URL", - "Parsing.index.551010-11": "运行结果", - "Parsing.index.551010-12": "需输入脚本和模拟数据后再点击", - "Parsing.index.551010-13": "调试", - "Parsing.index.551010-14": "请先调试", - "Parsing.index.551010-15": "保存", - "Parsing.index.551010-16": "操作成功", - "Parsing.index.551010-17": "保存成功", - "Parsing.index.551010-18": "请输入topic", - "Parsing.index.551010-19": "请输入url", - "Property.PropertyCard.551010-0": "更新时间", - "Property.ValueDetail.551010-0": "详情", - "Property.ValueDetail.551010-1": "确定", - "Property.ValueDetail.551010-2": "取消", - "Property.index.551010-0": "请输入名称", - "Property.index.551010-1": "名称", - "Property.index.551010-2": "值", - "Property.index.551010-3": "更新时间", - "Property.index.551010-4": "操作", - "Property.index.551010-5": "设置属性至设备", - "Property.index.551010-6": "指标", - "Property.index.551010-7": "获取最新属性值", - "Property.index.551010-8": "操作成功!", - "Property.index.551010-9": "详情", - "Status.util.551011-0": "网络组件", - "Status.util.551011-1": "诊断网络组件配置是否正确,配置错误将导致设备连接失败", - "Status.util.551011-2": "正在诊断中...", - "Status.util.551011-3": "设备接入网关", - "Status.util.551011-4": "诊断设备接入网关状态是否正常,禁用状态将导致连接失败", - "Status.util.551011-5": "产品状态", - "Status.util.551011-6": "诊断产品状态是否正常,禁用状态将导致设备连接失败", - "Status.util.551011-7": "设备状态", - "Status.util.551011-8": "诊断设备状态是否正常,禁用状态将导致设备连接失败", - "Status.util.551011-9": "诊断设备接入网关状态是否正常,网关配置是否正确", - "Status.util.551011-10": "网关父设备", - "Status.util.551011-11": "诊断网关父设备状态是否正常,禁用或离线将导致连接失败", - "EdgeMap.MSelect.551011-0": "请选择", - "Base.Base.551013-0": "请输入搜索名称", - "Base.Base.551013-1": "已查询到", - "Base.Base.551013-2": "条相关数据", - "Base.Base.551013-3": "当前的存储方式不支持删改、新增", - "Base.Base.551013-4": "保存", - "Base.Base.551013-5": "该设备受所属产品的存储策略影响,无法进行删改、新增操作", - "Base.Base.551013-6": "修改存储策略", - "Base.Base.551013-7": "请输入标识", - "Base.Base.551013-8": "请输入名称", - "Base.Base.551013-9": "配置", - "Base.Base.551013-10": "读", - "Base.Base.551013-11": "写", - "Base.Base.551013-12": "上报", - "Base.Base.551013-13": "请选择读写类型", - "Base.Base.551013-14": "是", - "Base.Base.551013-15": "否", - "Base.Base.551013-16": "请输入说明", - "Base.Base.551013-17": "操作成功!", - "Base.Base.551013-18": "页面改动数据未保存", - "Base.Base.551013-19": "不保存", - "components.Simple.551011-0": "精简模式下参数只支持输入框的方式录入", - "components.Simple.551011-1": "请按照json格式输入", - "components.Simple.551011-2": "该字段为必填字段", - "components.Simple.551011-3": "执行", - "components.Simple.551011-4": "清空", - "components.Simple.551011-5": "执行结果:", - "components.Simple.551011-6": "参数名称", - "components.Simple.551011-7": "输入类型", - "components.Simple.551011-8": "值", - "components.Simple.551011-9": "操作成功", - "components.Advance.551011-0": "执行", - "components.Advance.551011-1": "清空", - "components.Advance.551011-2": "执行结果:", - "components.Advance.551011-3": "操作成功", - "EdgeMap.PatchMapping.551011-0": "批量映射", - "EdgeMap.PatchMapping.551011-1": "采集器的点位名称与属性名称一致时将自动映射绑定;有多个采集器点位名称与属性名称一致时以第1个采集器的点位数据进行绑定", - "EdgeMap.PatchMapping.551011-2": "源数据", - "EdgeMap.PatchMapping.551011-3": "加入右侧", - "EdgeMap.PatchMapping.551011-4": "采集器", - "EdgeMap.PatchMapping.551011-5": "确定删除?", - "EdgeMap.PatchMapping.551011-6": "请选择采集器", - "EdgeMap.PatchMapping.551011-7": "操作成功", - "EdgeMap.PatchMapping.551011-8": "暂无对应属性的映射", - "Import.index.551012-0": "导入物模型", - "Import.index.551012-1": "导入的物模型会覆盖原来的属性、功能、事件、标签,请谨慎操作。", - "Import.index.551012-2": "导入时会根据标识跳过继承自产品物模型的属性、功能、事件、标签。", - "Import.index.551012-3": "导入方式", - "Import.index.551012-4": "请选择导入方式", - "Import.index.551012-5": "拷贝产品", - "Import.index.551012-6": "选择产品", - "Import.index.551012-7": "请选择产品", - "Import.index.551012-8": "物模型类型", - "Import.index.551012-9": "请选择物模型类型", - "Import.index.551012-10": "导入类型", - "Import.index.551012-11": "请选择导入类型", - "Import.index.551012-12": "文件上传", - "Import.index.551012-13": "脚本", - "Import.index.551012-14": "请上传文件", - "Import.index.551012-15": "上传文件", - "Import.index.551012-16": "支持扩展名:.json", - "Import.index.551012-17": "请输入物模型", - "Import.index.551012-18": "物模型", - "Import.index.551012-19": "在编辑器中编写物模型脚本", - "Import.index.551012-20": "属性定义第{0}个数组中缺失id属性", - "Import.index.551012-21": "属性定义第{0}个数组中id属性不能包含中文", - "Import.index.551012-22": "属性定义第{0}个数组中缺失name属性", - "Import.index.551012-23": "标签定义第{0}个数组中缺失valueType.type属性", - "Import.index.551012-24": "属性定义第{0}个数组中缺失expands.source属性", - "Import.index.551012-25": "属性定义第{0}个数组中缺失type属性", - "Import.index.551012-26": "方法定义第{0}个数组中缺失id属性", - "Import.index.551012-27": "方法定义第{0}个数组中id属性不能包含中文", - "Import.index.551012-28": "方法定义第{0}个数组中缺失name属性", - "Import.index.551012-29": "事件定义第{0}个数组中缺失id属性", - "Import.index.551012-30": "事件定义第{0}个数组中id属性不能包含中文", - "Import.index.551012-31": "事件定义第{0}个数组中缺失name属性", - "Import.index.551012-32": "事件定义第{0}个数组中缺失valueType.type属性", - "Import.index.551012-33": "事件定义第{0}个数组中缺失expands.level属性", - "Import.index.551012-34": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组第${\r\n number + 1\r\n }项的id属性", - "Import.index.551012-35": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组第${\r\n number + 1\r\n }项的name属性", - "Import.index.551012-36": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组第${\r\n number + 1\r\n }项的valueType.type属性", - "Import.index.551012-37": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组", - "Import.index.551012-38": "标签定义第{0}个数组中缺失id属性", - "Import.index.551012-39": "标签定义第{0}个数组中id属性不能包含中文", - "Import.index.551012-40": "标签定义第{0}个数组中缺失name属性", - "Import.index.551012-41": "属性定义第{0}个数组中缺失identifier属性", - "Import.index.551012-42": "属性定义第{0}个数组中缺失dataType.type属性", - "Import.index.551012-43": "方法定义第{0}个数组中缺失identifier属性", - "Import.index.551012-44": "方法定义第{0}个数组中缺失callType属性", - "Import.index.551012-45": "事件定义第{0}个数组中缺失identifier属性", - "Import.index.551012-46": "事件定义第{0}个数组中缺失type属性", - "Import.index.551012-47": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的id属性", - "Import.index.551012-48": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的name属性", - "Import.index.551012-49": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的dataType.type属性", - "Import.index.551012-50": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的dataType.specs属性", - "Import.index.551012-51": "事件定义第{0}个数组中缺失outputData数组", - "Import.index.551012-52": "操作成功!", - "Import.index.551012-53": "文件内容不能为空", - "Import.index.551012-54": "请上传json格式的文件", - "Import.index.551012-55": "导入成功", - "Import.index.551012-56": "物模型数据不正确", - "Import.index.551012-57": "上传json格式的物模型文件", - "Import.index.551012-58": "导入数据存在虚拟属性,请及时添加虚拟属性计算规则。", - "Import.index.551012-59": "确认", - "Import.index.551012-60": "标准物模型", - "Diagnose.util.551011-0": "正在诊断中", - "Diagnose.util.551011-1": "发现连接问题", - "Diagnose.util.551011-2": "连接状态正常", - "Diagnose.util.551011-3": "已诊断XX个", - "Diagnose.util.551011-4": "请处理连接异常", - "Diagnose.util.551011-5": "现在可调试消息通信", - "EdgeMap.index.551011-0": "批量映射", - "EdgeMap.index.551011-1": "保存", - "EdgeMap.index.551011-2": "采集器", - "EdgeMap.index.551011-3": "边缘网关代理的真实物理设备", - "EdgeMap.index.551011-4": "请选择", - "EdgeMap.index.551011-5": "请选择采集器", - "EdgeMap.index.551011-6": "请选择点位", - "EdgeMap.index.551011-7": "启用", - "EdgeMap.index.551011-8": "禁用", - "EdgeMap.index.551011-9": "未绑定", - "EdgeMap.index.551011-10": "解绑", - "EdgeMap.index.551011-11": "暂无权限,请联系管理员", - "EdgeMap.index.551011-12": "确认解绑?", - "EdgeMap.index.551011-13": "确认禁用?", - "EdgeMap.index.551011-14": "确认启用?", - "EdgeMap.index.551011-15": "暂无数据,请配置物模型", - "EdgeMap.index.551011-16": "名称", - "EdgeMap.index.551011-17": "通道", - "EdgeMap.index.551011-18": "点位", - "EdgeMap.index.551011-19": "状态", - "EdgeMap.index.551011-20": "操作", - "EdgeMap.index.551011-21": "操作成功!", - "Log.index.551011-0": "类型", - "Log.index.551011-1": "时间", - "Log.index.551011-2": "内容", - "Log.index.551011-3": "操作", - "Log.index.551011-4": "查看", - "Log.index.551011-5": "详细信息", - "Import.valideta.551012-0": "方法定义inputs第{0}个数组ValueType中缺失必填属性", - "Import.valideta.551012-1": "方法定义inputs第{0}个数组ValueType中缺失elementType属性", - "Import.valideta.551012-2": "方法定义inputs第{0}个数组ValueType中缺失fileType属性", - "Import.valideta.551012-3": "方法定义inputs第{0}个数组ValueType中缺失properties属性", - "Import.valideta.551012-4": "方法定义inputs第{0}个数组中缺失id属性", - "Import.valideta.551012-5": "方法定义inputs第{0}个数组中缺失name属性", - "Import.valideta.551012-6": "方法定义inputs第{0}个数组中缺失valueType.type属性", - "Import.valideta.551012-7": "方法定义inputs第{0}个数组dataType中缺失必填属性", - "Import.valideta.551012-8": "方法定义inputs第{0}个数组dataType中elements缺失必填属性", - "Import.valideta.551012-9": "方法定义inputs第{0}个数组dataType中缺失elements属性", - "Import.valideta.551012-10": "方法定义inputs第{0}个数组dataType中缺失elementType属性", - "Import.valideta.551012-11": "方法定义inputs第{0}个数组dataType中缺失fileType属性", - "Import.valideta.551012-12": "方法定义inputs第{0}个数组dataType中缺失properties属性", - "Import.valideta.551012-13": "方法定义inputs第{0}个数组中缺失dataType.type属性", - "Message.index.551011-0": "调试", - "Message.index.551011-1": "日志", - "Message.index.551011-2": "上行消息诊断中", - "Message.index.551011-3": "下行消息诊断中", - "Message.index.551011-4": "下行消息通信异常", - "Message.index.551011-5": "下行消息通信正常", - "Message.index.551011-6": "上行消息通信异常", - "Message.index.551011-7": "上行消息通信正常", - "SaveChild.index.551011-0": "基本信息", - "SaveChild.index.551011-1": "返回", - "SaveChild.index.551011-2": "设备名称", - "SaveChild.index.551011-3": "请输入设备名称", - "SaveChild.index.551011-4": "产品名称", - "SaveChild.index.551011-5": "请选择产品名称", - "DashBoard.index.551011-0": "产品数量", - "DashBoard.index.551011-1": "设备数量", - "DashBoard.index.551011-2": "当前在线", - "DashBoard.index.551011-3": "今日设备消息量", - "DashBoard.index.551011-4": "设备消息", - "DashBoard.index.551011-5": "最近1小时", - "DashBoard.index.551011-6": "最近24小时", - "DashBoard.index.551011-7": "近一周", - "DashBoard.index.551011-8": "设备分布", - "DashBoard.index.551011-9": "正常", - "DashBoard.index.551011-10": "禁用", - "DashBoard.index.551011-11": "在线", - "DashBoard.index.551011-12": "离线", - "DashBoard.index.551011-13": "昨日在线", - "DashBoard.index.551011-14": "当月设备消息量", - "DashBoard.index.551011-15": "在线数", - "DashBoard.index.551011-16": "消息量", - "DashBoard.index.551011-17": "M月dd日 HH:mm", - "DashBoard.index.551011-18": "M月dd日 HH:mm:ss", - "DashBoard.index.551011-19": "yyyy年-M月", - "DashBoard.index.551011-20": "{0}时", - "components.TimeSelect.551011-0": "今日", - "components.TimeSelect.551011-1": "近一周", - "components.TimeSelect.551011-2": "近一月", - "components.TimeSelect.551011-3": "近一年", - "AlarmRecord.index.551011-0": "告警数据", - "AlarmRecord.index.551011-1": "无效数据", - "Status.ManualInspection.551011-0": "请检查配置项是否填写正确,若您确定该项无需诊断可", - "Status.ManualInspection.551011-1": "忽略", - "Status.ManualInspection.551011-2": "诊断项说明", - "Status.ManualInspection.551011-3": "接口地址", - "Status.ManualInspection.551011-4": "通知Token", - "Status.ManualInspection.551011-5": "SIP 域", - "Status.ManualInspection.551011-6": "集群", - "Status.ManualInspection.551011-7": "共享配置", - "Status.ManualInspection.551011-8": "独立配置", - "Status.ManualInspection.551011-9": "SIP 地址", - "Status.ManualInspection.551011-10": "公网 Host", - "Status.ManualInspection.551011-11": "节点", - "Status.ManualInspection.551011-12": "节点名称", - "Status.ManualInspection.551011-13": "人工检查", - "Status.ManualInspection.551011-14": "去修改", - "Status.ManualInspection.551011-15": "确认无误", - "Function.index.551012-0": "请选择", - "Function.index.551012-1": "读取属性", - "Function.index.551012-2": "修改属性", - "Function.index.551012-3": "调用功能", - "Function.index.551012-4": "请选择属性", - "Function.index.551012-5": "请输入值", - "Function.index.551012-6": "请选择功能", - "Function.index.551012-7": "发送", - "Function.index.551012-8": "参数列表", - "Dialog.index.551012-0": "连接", - "Dialog.index.551012-1": "权限验证", - "Dialog.index.551012-2": "解码", - "Dialog.index.551012-3": "编码", - "Dialog.index.551012-4": "请求", - "Dialog.index.551012-5": "响应", - "Dialog.index.551012-6": "下行消息", - "Dialog.index.551012-7": "上行消息", - "EdgeMap.MSelect.551012-0": "请选择", - "Function.EditTable.551012-0": "该字段为必填字段", - "Function.EditTable.551012-1": "是", - "Function.EditTable.551012-2": "否", - "Function.EditTable.551012-3": "参数名称", - "Function.EditTable.551012-4": "类型", - "Function.EditTable.551012-5": "值", - "EdgeMap.PatchMapping.551012-0": "批量映射", - "EdgeMap.PatchMapping.551012-1": "采集器的点位名称与属性名称一致时将自动映射绑定;有多个采集器点位名称与属性名称一致时以第1个采集器的点位数据进行绑定", - "EdgeMap.PatchMapping.551012-2": "源数据", - "EdgeMap.PatchMapping.551012-3": "加入右侧", - "EdgeMap.PatchMapping.551012-4": "采集器", - "EdgeMap.PatchMapping.551012-5": "确定删除?", - "EdgeMap.PatchMapping.551012-6": "请选择采集器", - "EdgeMap.PatchMapping.551012-7": "操作成功", - "EdgeMap.PatchMapping.551012-8": "暂无对应属性的映射", - "EdgeMap.index.551012-0": "点位映射", - "EdgeMap.index.551012-1": "批量映射", - "EdgeMap.index.551012-2": "保存并应用", - "EdgeMap.index.551012-3": "采集器", - "EdgeMap.index.551012-4": "边缘网关代理的真实物理设备", - "EdgeMap.index.551012-5": "请选择", - "EdgeMap.index.551012-6": "请选择采集器", - "EdgeMap.index.551012-7": "请选择点位", - "EdgeMap.index.551012-8": "已绑定", - "EdgeMap.index.551012-9": "未绑定", - "EdgeMap.index.551012-10": "解绑", - "EdgeMap.index.551012-11": "确认解绑", - "EdgeMap.index.551012-12": "暂无数据,请配置物模型", - "EdgeMap.index.551012-13": "名称", - "EdgeMap.index.551012-14": "通道", - "EdgeMap.index.551012-15": "点位", - "EdgeMap.index.551012-16": "状态", - "EdgeMap.index.551012-17": "操作", - "EdgeMap.index.551012-18": "操作成功!", - "EdgeMap.index.551012-19": "请配置物模型", - "EdgeMap.index.551012-20": "保存成功", - "Export.index.551012-0": "导出", - "Export.index.551012-1": "选择单个产品时可导出其下属设备的详细数据,不选择产品时导出所有设备的基础数据。", - "Export.index.551012-2": "产品", - "Export.index.551012-3": "请选择产品", - "Export.index.551012-4": "文件格式", - "Export.index.551012-5": "请选择文件格式", - "components.AlarmLog.551012-0": "告警日志", - "components.AlarmLog.551012-1": "处理", - "Metadata.index.551012-0": "确认重置?", - "Metadata.index.551012-1": "重置后将使用产品的物模型配置", - "Metadata.index.551012-2": "重置操作", - "Metadata.index.551012-3": "快速导入", - "Metadata.index.551012-4": "物模型TSL", - "Metadata.index.551012-5": "设备会默认继承产品的物模型,继承的物模型不支持删改", - "Metadata.index.551012-6": "属性定义", - "Metadata.index.551012-7": "功能定义", - "Metadata.index.551012-8": "事件定义", - "Metadata.index.551012-9": "标签定义", - "Metadata.index.551012-10": "操作成功", - "Cat.index.551012-0": "查看物模型", - "Cat.index.551012-1": "导出", - "Cat.index.551012-2": "物模型是对设备在云端的功能描述,包括设备的属性、服务和事件。物联网平台通过定义一种物的描述语言来描述物模型,称之为", - "Cat.index.551012-3": "TSL(即 Thing Specification Language),采用 JSON 格式,您可以根据 TSL", - "Cat.index.551012-4": "组装上报设备的数据。您可以导出完整物模型,用于云端应用开发。", - "Cat.index.551012-5": "{0}-物模型", - "Cat.index.551012-6": "请先配置物模型", - "Cat.index.551012-7": "标准物模型", - "InklingDevice.index.551012-0": "全选", - "InklingDevice.index.551012-1": "该设备已绑定平台设备", - "InklingDevice.index.551012-2": "暂无数据", - "InklingDevice.index.551012-3": "第 {0} - ${\r\n MaxSize > pageData.total\r\n ? pageData.total\r\n : MaxSize\r\n } 条/总共 {1} 条", - "Category.index.551012-0": "新增", - "Category.index.551012-1": "名称", - "Category.index.551012-2": "排序", - "Category.index.551012-3": "说明", - "Category.index.551012-4": "操作", - "Category.index.551012-5": "编辑", - "Category.index.551012-6": "编辑分类", - "Category.index.551012-7": "添加子分类", - "Category.index.551012-8": "新增子分类", - "Category.index.551012-9": "删除", - "Category.index.551012-10": "确认删除?", - "Category.index.551012-11": "确定", - "Category.index.551012-12": "取消", - "Category.index.551012-13": "操作成功!", - "Category.index.551012-14": "操作失败!", - "Category.index.551012-15": "新增分类", - "Alarm.index.551012-0": "告警时间", - "Alarm.index.551012-1": "原始值", - "Alarm.index.551012-2": "告警持续时长", - "Alarm.index.551012-3": "触发条件", - "Alarm.index.551012-4": "告警原因", - "Alarm.index.551012-5": "处理时间", - "Alarm.index.551012-6": "处理类型", - "Alarm.index.551012-7": "人工", - "Alarm.index.551012-8": "系统", - "Alarm.index.551012-9": "状态", - "Alarm.index.551012-10": "无告警", - "Alarm.index.551012-11": "告警中", - "Alarm.index.551012-12": "操作", - "Alarm.index.551012-13": "查看详情", - "Alarm.index.551012-14": "查看日志", - "Alarm.index.551012-15": "告警处理", - "Base.utils.551012-0": "普通", - "Base.utils.551012-1": "警告", - "Base.utils.551012-2": "紧急", - "Base.utils.551012-3": "设备", - "Base.utils.551012-4": "手动", - "Base.utils.551012-5": "规则", - "Base.utils.551012-6": "读", - "Base.utils.551012-7": "写", - "Base.utils.551012-8": "上报", - "Base.index.551012-0": "请输入名称", - "Base.index.551012-1": "当前的存储方式不支持新增", - "Base.index.551012-2": "新增", - "Base.index.551012-3": "是", - "Base.index.551012-4": "否", - "Base.index.551012-5": "当前的存储方式不支持编辑", - "Base.index.551012-6": "编辑", - "Base.index.551012-7": "确认删除?", - "Base.index.551012-8": "删除", - "Base.index.551012-9": "普通", - "Base.index.551012-10": "警告", - "Base.index.551012-11": "紧急", - "Base.index.551012-12": "设备", - "Base.index.551012-13": "手动", - "Base.index.551012-14": "规则", - "Base.index.551012-15": "读", - "Base.index.551012-16": "写", - "Base.index.551012-17": "上报", - "Base.index.551012-18": "操作", - "Base.index.551012-19": "第 {0} - {1} 条/总共 {2} 条", - "Base.index.551012-20": "修改物模型后会脱离产品物模型", - "Base.index.551012-21": "操作成功!", - "Base.index.551012-22": "操作失败!", - "Invalid.index.551012-0": "上报时间", - "Invalid.index.551012-1": "阈值限制", - "Invalid.index.551012-2": "上报数据", - "Invalid.index.551012-3": "原始值", - "components.ConfigModal.551013-0": "无", - "components.ConfigModal.551013-1": "参数标识", - "components.ConfigModal.551013-2": "该标识已存在", - "components.ConfigModal.551013-3": "请输入标识", - "components.ConfigModal.551013-4": "最多可输入64个字符", - "components.ConfigModal.551013-5": "标识只能由数字、字母、下划线、中划线组成", - "components.ConfigModal.551013-6": "参数名称", - "components.ConfigModal.551013-7": "请输入参数名称", - "components.ConfigModal.551013-8": "数据类型", - "components.ConfigModal.551013-9": "请选择数据类型", - "components.ConfigModal.551013-10": "其他配置", - "components.ConfigModal.551013-11": "操作", - "components.Source.551013-0": "请选择来源", - "components.Source.551013-1": "编辑", - "components.Source.551013-2": "重置", - "components.Source.551013-3": "重置为产品属性规则", - "components.Source.551013-4": "设备", - "components.Source.551013-5": "手动", - "components.Source.551013-6": "规则", - "components.Source.551013-7": "操作成功!", - "BindChildDevice.index.551012-0": "绑定子设备", - "BindChildDevice.index.551012-1": "确定", - "BindChildDevice.index.551012-2": "取消", - "BindChildDevice.index.551012-3": "设备名称", - "BindChildDevice.index.551012-4": "所属产品", - "BindChildDevice.index.551012-5": "注册时间", - "BindChildDevice.index.551012-6": "状态", - "BindChildDevice.index.551012-7": "禁用", - "BindChildDevice.index.551012-8": "离线", - "BindChildDevice.index.551012-9": "在线", - "BindChildDevice.index.551012-10": "请选择需要绑定的设备", - "BindChildDevice.index.551012-11": "操作成功", - "Log.index.551012-0": "连接", - "Log.index.551012-1": "权限验证", - "Log.index.551012-2": "解码", - "Log.index.551012-3": "编码", - "Log.index.551012-4": "请求", - "Log.index.551012-5": "响应", - "Log.index.551012-6": "下行消息", - "Log.index.551012-7": "上行消息", - "Edit.index.551013-0": "保存", - "Edit.index.551013-1": "属性", - "Edit.index.551013-2": "事件", - "Edit.index.551013-3": "功能", - "Edit.index.551013-4": "标签", - "Edit.index.551013-5": "新增", - "Edit.index.551013-6": "编辑", - "Edit.index.551013-7": "标识已存在", - "Edit.index.551013-8": "操作成功!", - "Edit.index.551013-9": "操作失败!", - "Edit.validator.551013-0": "请配置枚举项", - "Edit.validator.551013-1": "请输入元素配置", - "Edit.validator.551013-2": "配置参数", - "Edit.validator.551013-3": "请输入{0}", - "Edit.validator.551013-4": "请输入标识", - "Edit.validator.551013-5": "请输入名称", - "Edit.validator.551013-6": "数据类型", - "Edit.validator.551013-7": "请选择{0}", - "Edit.validator.551013-8": "请选择文件类型", - "DetailModal.utils.551013-0": "参数标识", - "DetailModal.utils.551013-1": "参数名称", - "DetailModal.utils.551013-2": "参数类型", - "Edit.BaseForm.551013-0": "标识", - "Edit.BaseForm.551013-1": "请输入标识", - "Edit.BaseForm.551013-2": "最多可输入64个字符", - "Edit.BaseForm.551013-3": "标识只能由数字、字母、下划线、中划线组成", - "Edit.BaseForm.551013-4": "名称", - "Edit.BaseForm.551013-5": "请输入名称", - "Edit.BaseForm.551013-6": "数据类型", - "Edit.BaseForm.551013-7": "是否异步", - "Edit.BaseForm.551013-8": "请选择是否异步", - "Edit.BaseForm.551013-9": "是", - "Edit.BaseForm.551013-10": "否", - "Edit.BaseForm.551013-11": "输入参数", - "Edit.BaseForm.551013-12": "输出参数", - "Edit.BaseForm.551013-13": "级别", - "Edit.BaseForm.551013-14": "请选择级别", - "Edit.BaseForm.551013-15": "读写类型", - "Edit.BaseForm.551013-16": "请选择读写类型", - "Edit.BaseForm.551013-17": "说明", - "Edit.BaseForm.551013-18": "最多可输入200个字符", - "Edit.BaseForm.551013-19": "请输入说明", - "modifyModal.index.551013-0": "确定", - "modifyModal.index.551013-1": "取消", - "modifyModal.index.551013-2": "名称", - "modifyModal.index.551013-3": "请输入名称", - "modifyModal.index.551013-4": "排序", - "modifyModal.index.551013-5": "请输入排序", - "modifyModal.index.551013-6": "说明", - "modifyModal.index.551013-7": "请输入说明", - "modifyModal.index.551013-8": "最多可输入64个字符", - "modifyModal.index.551013-9": "请输入正整数", - "modifyModal.index.551013-10": "操作成功!", - "modifyModal.index.551013-11": "操作失败!", - "modifyModal.index.551013-12": "树形结构最多添加5层", - "DetailModal.TagsModal.551013-0": "标签详情", - "DetailModal.TagsModal.551013-1": "标签标识", - "DetailModal.TagsModal.551013-2": "标签名称", - "DetailModal.TagsModal.551013-3": "标签类型", - "DetailModal.TagsModal.551013-4": "单位", - "DetailModal.TagsModal.551013-5": "精度", - "DetailModal.TagsModal.551013-6": "最大长度", - "DetailModal.TagsModal.551013-7": "文件类型", - "DetailModal.TagsModal.551013-8": "格式", - "DetailModal.TagsModal.551013-9": "读写类型", - "DetailModal.TagsModal.551013-10": "存储方式", - "DetailModal.TagsModal.551013-11": "确认", - "DetailModal.TagsModal.551013-12": "设备", - "DetailModal.TagsModal.551013-13": "手动", - "DetailModal.TagsModal.551013-14": "规则", - "DetailModal.TagsModal.551013-15": "读", - "DetailModal.TagsModal.551013-16": "写", - "DetailModal.TagsModal.551013-17": "上报", - "DetailModal.TagsModal.551013-18": "指标标识", - "DetailModal.TagsModal.551013-19": "指标名称", - "DetailModal.TagsModal.551013-20": "指标治", - "DetailModal.TagsModal.551013-21": "标识", - "DetailModal.TagsModal.551013-22": "名称", - "DetailModal.PropertiesModal.551013-0": "属性详情", - "DetailModal.PropertiesModal.551013-1": "属性标识", - "DetailModal.PropertiesModal.551013-2": "属性名称", - "DetailModal.PropertiesModal.551013-3": "属性类型", - "DetailModal.PropertiesModal.551013-4": "单位", - "DetailModal.PropertiesModal.551013-5": "精度", - "DetailModal.PropertiesModal.551013-6": "最大长度", - "DetailModal.PropertiesModal.551013-7": "文件类型", - "DetailModal.PropertiesModal.551013-8": "格式", - "DetailModal.PropertiesModal.551013-9": "属性来源", - "DetailModal.PropertiesModal.551013-10": "查看", - "DetailModal.PropertiesModal.551013-11": "读写类型", - "DetailModal.PropertiesModal.551013-12": "存储方式", - "DetailModal.PropertiesModal.551013-13": "指标配置", - "DetailModal.PropertiesModal.551013-14": "确认", - "DetailModal.PropertiesModal.551013-15": "设备", - "DetailModal.PropertiesModal.551013-16": "手动", - "DetailModal.PropertiesModal.551013-17": "规则", - "DetailModal.PropertiesModal.551013-18": "读", - "DetailModal.PropertiesModal.551013-19": "写", - "DetailModal.PropertiesModal.551013-20": "上报", - "DetailModal.PropertiesModal.551013-21": "序号", - "DetailModal.PropertiesModal.551013-22": "指标标识", - "DetailModal.PropertiesModal.551013-23": "指标名称", - "DetailModal.PropertiesModal.551013-24": "指标值", - "DetailModal.PropertiesModal.551013-25": "标识", - "DetailModal.PropertiesModal.551013-26": "名称", - "Base.columns.551013-0": "设备", - "Base.columns.551013-1": "手动", - "Base.columns.551013-2": "规则", - "Base.columns.551013-3": "读", - "Base.columns.551013-4": "写", - "Base.columns.551013-5": "上报", - "Base.columns.551013-6": "请添加枚举项", - "Base.columns.551013-7": "请选择元素类型", - "Base.columns.551013-8": "请添加参数", - "Base.columns.551013-9": "请选择时间格式", - "Base.columns.551013-10": "请选择文件类型", - "Base.columns.551013-11": "是", - "Base.columns.551013-12": "否", - "Base.columns.551013-13": "继承自产品物模型的数据不支持修改", - "Base.columns.551013-14": "标识", - "Base.columns.551013-15": "该标识已存在", - "Base.columns.551013-16": "最多可输入64个字符", - "Base.columns.551013-17": "标识只能由数字、字母、下划线组成", - "Base.columns.551013-18": "名称", - "Base.columns.551013-19": "请输入名称", - "Base.columns.551013-20": "数据类型", - "Base.columns.551013-21": "请选择数据类型", - "Base.columns.551013-22": "属性来源", - "Base.columns.551013-23": "请选择读写类型", - "Base.columns.551013-24": "请选择属性来源", - "Base.columns.551013-25": "其它配置", - "Base.columns.551013-26": "是否异步", - "Base.columns.551013-27": "输入参数", - "Base.columns.551013-28": "输出参数", - "Base.columns.551013-29": "说明", - "Base.columns.551013-30": "最多可输入20个字符", - "Base.columns.551013-31": "事件级别", - "Base.columns.551013-32": "请添加配置参数", - "Base.columns.551013-33": "整数型", - "Base.columns.551013-34": "长整数型", - "Base.columns.551013-35": "单精度浮点型", - "Base.columns.551013-36": "双精度浮点数", - "Base.columns.551013-37": "字符串", - "Base.columns.551013-38": "布尔型", - "Base.columns.551013-39": "时间型", - "Base.columns.551013-40": "枚举", - "Base.columns.551013-41": "数组", - "Base.columns.551013-42": "结构体", - "Base.columns.551013-43": "文件", - "Base.columns.551013-44": "密码", - "Base.columns.551013-45": "地理位置", - "DetailModal.EventModal.551013-0": "事件详情", - "DetailModal.EventModal.551013-1": "事件标识", - "DetailModal.EventModal.551013-2": "事件名称", - "DetailModal.EventModal.551013-3": "事件级别", - "DetailModal.EventModal.551013-4": "输出参数", - "DetailModal.EventModal.551013-5": "存储方式", - "DetailModal.EventModal.551013-6": "确认", - "DetailModal.EventModal.551013-7": "普通", - "DetailModal.EventModal.551013-8": "警告", - "DetailModal.EventModal.551013-9": "紧急", - "Edit.ValueTypeForm.551013-0": "请选择{0}", - "Edit.ValueTypeForm.551013-1": "单位", - "Edit.ValueTypeForm.551013-2": "精度", - "Edit.ValueTypeForm.551013-3": "请输入精度", - "Edit.ValueTypeForm.551013-4": "时间格式", - "Edit.ValueTypeForm.551013-5": "布尔值", - "Edit.ValueTypeForm.551013-6": "枚举项", - "Edit.ValueTypeForm.551013-7": "最大长度", - "Edit.ValueTypeForm.551013-8": "字节", - "Edit.ValueTypeForm.551013-9": "请输入最大长度", - "Edit.ValueTypeForm.551013-10": "元素配置", - "Edit.ValueTypeForm.551013-11": "JSON对象", - "Edit.ValueTypeForm.551013-12": "文件类型", - "Edit.ValueTypeForm.551013-13": "请选择文件类型", - "Edit.ValueTypeForm.551013-14": "数据类型", - "Edit.ValueTypeForm.551013-15": "结构体", - "VirtualRule.DetailModal.551013-0": "规则", - "Metrics.BooleanSelect.551013-0": "固定值", - "Metrics.BooleanSelect.551013-1": "范围值", - "components.DataType.551013-0": "参数标识", - "components.DataType.551013-1": "该标识已存在", - "components.DataType.551013-2": "请输入标识", - "components.DataType.551013-3": "最多可输入64个字符", - "components.DataType.551013-4": "标识只能由数字、字母、下划线、中划线组成", - "components.DataType.551013-5": "参数名称", - "components.DataType.551013-6": "请输入参数名称", - "components.DataType.551013-7": "数据类型", - "components.DataType.551013-8": "请选择数据类型", - "components.DataType.551013-9": "其他配置", - "components.DataType.551013-10": "操作", - "Metrics.Metrics.551013-0": "请输入标识", - "Metrics.Metrics.551013-1": "请输入名称", - "Metrics.Metrics.551013-2": "添加指标值", - "Metrics.Metrics.551013-3": "指标标识", - "Metrics.Metrics.551013-4": "该标识已存在", - "Metrics.Metrics.551013-5": "最多可输入64个字符", - "Metrics.Metrics.551013-6": "标识只能由数字、字母、下划线、中划线组成", - "Metrics.Metrics.551013-7": "指标名称", - "Metrics.Metrics.551013-8": "请输入指标名称", - "Metrics.Metrics.551013-9": "指标配置", - "Metrics.Metrics.551013-10": "请配置指标", - "Metrics.Metrics.551013-11": "操作", - "Metrics.Metrics.551013-12": "指标值", - "VirtualRule.Rule.551013-0": "编辑规则", - "components.DataTypeObjectChild.551013-0": "无", - "VirtualRule.index.551013-0": "请选择触发属性", - "VirtualRule.index.551013-1": "触发属性", - "VirtualRule.index.551013-2": "选择当前产品物模型下的属性作为触发属性", - "VirtualRule.index.551013-3": "任意属性值更新时将触发下方计算规则", - "VirtualRule.index.551013-4": "任意属性", - "VirtualRule.index.551013-5": "计算规则", - "VirtualRule.index.551013-6": "窗口", - "VirtualRule.index.551013-7": "请选择窗口类型", - "VirtualRule.index.551013-8": "无", - "VirtualRule.index.551013-9": "时间窗口", - "VirtualRule.index.551013-10": "频次窗口", - "VirtualRule.index.551013-11": "聚合函数", - "VirtualRule.index.551013-12": "请选择聚合函数", - "VirtualRule.index.551013-13": "窗口长度", - "VirtualRule.index.551013-14": "次", - "VirtualRule.index.551013-15": "请输入窗口长度", - "VirtualRule.index.551013-16": "请输入1-999999之间的正整数", - "VirtualRule.index.551013-17": "步长", - "VirtualRule.index.551013-18": "请输入步长", - "VirtualRule.index.551013-19": "写", - "VirtualRule.index.551013-20": "上报", - "VirtualRule.index.551013-21": "读", - "Metrics.item.551013-0": "请输入", - "Metrics.item.551013-1": "请选择", - "Tags.Type.551013-0": "读", - "Tags.Type.551013-1": "写", - "Tags.Type.551013-2": "上报", - "Tags.Type.551013-3": "请选择读写类型", - "Detail.Save.551010-0": "查看", - "Detail.Save.551010-1": "确认", - "Detail.Save.551010-2": "取消", - "Detail.Save.551010-3": "失败原因:", - "Events.OtherConfigInfo.551013-0": "无", - "Events.ConfigParams.551013-0": "配置", - "Events.ConfigParams.551013-1": "参数标识", - "Events.ConfigParams.551013-2": "该标识已存在", - "Events.ConfigParams.551013-3": "请输入标识", - "Events.ConfigParams.551013-4": "最多可输入64个字符", - "Events.ConfigParams.551013-5": "标识只能由数字、字母、下划线、中划线组成", - "Events.ConfigParams.551013-6": "参数名称", - "Events.ConfigParams.551013-7": "请输入参数名称", - "Events.ConfigParams.551013-8": "数据类型", - "Events.ConfigParams.551013-9": "请选择数据类型", - "Events.ConfigParams.551013-10": "其他配置", - "Events.ConfigParams.551013-11": "操作", - "Events.ConfigParams.551013-12": "请选择", - "Product.index.551010-0": "新增", - "Product.index.551010-1": "导入", - "Product.index.551010-2": "正常", - "Product.index.551010-3": "禁用", - "Product.index.551010-4": "设备类型", - "Product.index.551010-5": "接入方式", - "Product.index.551010-6": "未接入", - "Product.index.551010-7": "产品名称", - "Product.index.551010-8": "状态", - "Product.index.551010-9": "说明", - "Product.index.551010-10": "操作", - "Product.index.551010-11": "查看", - "Product.index.551010-12": "编辑", - "Product.index.551010-13": "导出", - "Product.index.551010-14": "产品", - "Product.index.551010-15": "启用", - "Product.index.551010-16": "确认{0}?", - "Product.index.551010-17": "操作成功!", - "Product.index.551010-18": "操作失败!", - "Product.index.551010-19": "删除", - "Product.index.551010-20": "已启用的产品不能删除", - "Product.index.551010-21": "确认删除?", - "Product.index.551010-22": "请上传json格式文件", - "Product.index.551010-23": "文件内容不能为空", - "Product.index.551010-24": "请上传正确格式文件", - "Product.index.551010-25": "缺少deviceType字段或对应的值", - "Product.index.551010-26": "操作成功", - "Product.index.551010-27": "名称", - "Product.index.551010-28": "网关类型", - "Product.index.551010-29": "直连设备", - "Product.index.551010-30": "网关子设备", - "Product.index.551010-31": "网关设备", - "Product.index.551010-32": "产品分类", - "Product.index.551010-33": "所属组织", - "Events.ValueObject.551013-0": "请选择", - "Function.InputParams.551013-0": "必填", - "Function.InputParams.551013-1": "不必填", - "Function.InputParams.551013-2": "配置", - "Function.InputParams.551013-3": "请选择", - "Function.InputParams.551013-4": "参数标识", - "Function.InputParams.551013-5": "该标识已存在", - "Function.InputParams.551013-6": "请输入标识", - "Function.InputParams.551013-7": "最多可输入64个字符", - "Function.InputParams.551013-8": "标识只能由数字、字母、下划线、中划线组成", - "Function.InputParams.551013-9": "参数名称", - "Function.InputParams.551013-10": "请输入名称", - "Function.InputParams.551013-11": "填写约束", - "Function.InputParams.551013-12": "数据类型", - "Function.InputParams.551013-13": "请选择数据类型", - "Function.InputParams.551013-14": "其他配置", - "Function.InputParams.551013-15": "操作", - "Function.AsyncSelect.551013-0": "请选择", - "Metrics.ValueItem.551013-0": "请输入指标值", - "Metrics.ValueItem.551013-1": "配置", - "Metrics.ValueItem.551013-2": "是", - "Metrics.ValueItem.551013-3": "否", - "Metrics.ValueItem.551013-4": "请输入", - "Metrics.ValueItem.551013-5": "请选择", - "Metrics.ValueItem.551013-6": "{0}指标值", - "Metrics.ValueItem.551013-7": "需大于左侧数值", - "Metrics.ValueItem.551013-8": "最多可输入64个字符", - "DialogTips.index.551010-0": "产品创建成功", - "DialogTips.index.551010-1": "关闭", - "DialogTips.index.551010-2": "产品ID:", - "DialogTips.index.551010-3": "查看详情", - "DialogTips.index.551010-4": "接下来推荐操作:", - "DialogTips.index.551010-5": "1、配置产品接入方式", - "DialogTips.index.551010-6": "点击具体产品的查看按钮,进入“设备接入”tab页,并参照设备铭牌说明选择匹配的接入方式", - "DialogTips.index.551010-7": "2、添加测试设备", - "DialogTips.index.551010-8": "进入设备列表,添加单个设备,用于验证产品模型是否配置正确", - "DialogTips.index.551010-9": "3、功能调试", - "DialogTips.index.551010-10": "点击查看具体设备,进入“设备诊断”对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确", - "DialogTips.index.551010-11": "4、批量添加设备", - "DialogTips.index.551010-12": "进入设备列表页面,点击批量导入设备,批量添加同一产品下的设备", - "components.Constraint.551013-0": "不必填", - "components.Constraint.551013-1": "必填", - "Properties.OtherSetting.551013-0": "指标配置", - "Properties.OtherSetting.551013-1": "场景联动页面可引用指标配置作为触发条件", - "Properties.OtherSetting.551013-2": "拓展配置", - "Properties.OtherSetting.551013-3": "重置", - "Properties.OtherSetting.551013-4": "阈值限制", - "Properties.OtherSetting.551013-5": "下限", - "Properties.OtherSetting.551013-6": "上限", - "Properties.OtherSetting.551013-7": "阈值", - "Properties.OtherSetting.551013-8": "请输入下限", - "Properties.OtherSetting.551013-9": "请输入上限", - "Properties.OtherSetting.551013-10": "超出阈值数据处理方式", - "Properties.OtherSetting.551013-11": "忽略", - "Properties.OtherSetting.551013-12": "仅记录", - "Properties.OtherSetting.551013-13": "记录并发送告警", - "Properties.OtherSetting.551013-14": "没有动态配置项", - "Properties.OtherSetting.551013-15": "配置", - "Properties.OtherSetting.551013-16": "平台将忽略超出阈值的数据,无法查看上报记录", - "Properties.OtherSetting.551013-17": "您可以在告警记录-无效数据页面查看超出阈值的数据上报记录", - "Properties.OtherSetting.551013-18": "您可以在设备详情-告警记录 页面查看告警情况", - "Properties.OtherSetting.551013-19": "否", - "Properties.OtherSetting.551013-20": "是", - "Properties.OtherSetting.551013-21": "参数名称", - "Properties.OtherSetting.551013-22": "输入类型", - "Properties.OtherSetting.551013-23": "值", - "ChildDevice.index.551011-0": "新增并绑定", - "ChildDevice.index.551011-1": "绑定", - "ChildDevice.index.551011-2": "确定解绑吗?", - "ChildDevice.index.551011-3": "批量解除", - "ChildDevice.index.551011-4": "设备名称", - "ChildDevice.index.551011-5": "所属产品", - "ChildDevice.index.551011-6": "注册时间", - "ChildDevice.index.551011-7": "状态", - "ChildDevice.index.551011-8": "禁用", - "ChildDevice.index.551011-9": "离线", - "ChildDevice.index.551011-10": "在线", - "ChildDevice.index.551011-11": "说明", - "ChildDevice.index.551011-12": "操作", - "ChildDevice.index.551011-13": "查看", - "ChildDevice.index.551011-14": "解绑", - "ChildDevice.index.551011-15": "确认解绑吗?", - "ChildDevice.index.551011-16": "确定", - "ChildDevice.index.551011-17": "取消", - "ChildDevice.index.551011-18": "操作成功!", - "ChildDevice.index.551011-19": "编辑", - "ChildDevice.index.551011-20": "请勾选需要解绑的数据", - "Status.index.551011-0": "网络组件", - "Status.index.551011-1": "诊断网络组件配置是否正确,配置错误将导致设备连接失败", - "Status.index.551011-2": "正常", - "Status.index.551011-3": "异常", - "Status.index.551011-4": "网络组件已禁用,请先", - "Status.index.551011-5": "确认启用", - "Status.index.551011-6": "操作成功!", - "Status.index.551011-7": "启用", - "Status.index.551011-8": "请检查服务器端口是否开放,如未开放,请开放后尝试重新连接", - "Status.index.551011-9": "请检查服务器防火策略,如有开启防火墙,请关闭防火墙或调整防火墙策略后重试", - "Status.index.551011-10": "请求发生错误", - "Status.index.551011-11": "设备不含accessId", - "Status.index.551011-12": "诊断设备接入网关状态是否正常,网关配置是否正确", - "Status.index.551011-13": "诊断设备接入网关状态是否正常,禁用状态将导致连接失败", - "Status.index.551011-14": "设备接入网关", - "Status.index.551011-15": "可能存在异常", - "Status.index.551011-16": "请", - "Status.index.551011-17": "{0}配置", - "Status.index.551011-18": "人工检查", - "Status.index.551011-19": "网关配置是否已填写正确,若您确定该项无需诊断可", - "Status.index.551011-20": "确认忽略?", - "Status.index.551011-21": "忽略", - "Status.index.551011-22": "设备接入网关已禁用,请先", - "Status.index.551011-23": "网关父设备", - "Status.index.551011-24": "诊断网关父设备状态是否正常,禁用或离线将导致连接失败", - "Status.index.551011-25": "未绑定父设备,请先", - "Status.index.551011-26": "绑定", - "Status.index.551011-27": "父设备后重试", - "Status.index.551011-28": "网关父设备已禁用,请先", - "Status.index.551011-29": "网关父设备已离线,请先排查网关设备故障", - "Status.index.551011-30": "产品状态", - "Status.index.551011-31": "诊断产品状态是否正常,禁用状态将导致设备连接失败", - "Status.index.551011-32": "产品已禁用,请", - "Status.index.551011-33": "产品", - "Status.index.551011-34": "设备状态", - "Status.index.551011-35": "诊断设备状态是否正常,禁用状态将导致设备连接失败", - "Status.index.551011-36": "设备已禁用,请", - "Status.index.551011-37": "离线", - "Status.index.551011-38": "设备", - "Status.index.551011-39": "产品-{0}", - "Status.index.551011-40": "诊断产品{0}认证配置是否正确,错误的配置将导致连接失败", - "Status.index.551011-41": "正在诊断中...", - "Status.index.551011-42": "请根据设备接入配置需要", - "Status.index.551011-43": "填写", - "Status.index.551011-44": ",若您确定该项无需诊断可", - "Status.index.551011-45": "配置是否已填写正确,若您确定该项无需诊断可", - "Status.index.551011-46": "设备-{0}", - "Status.index.551011-47": "诊断设备{0}认证配置是否正确,错误的配置将导致连接失败", - "Status.index.551011-48": "设备-OneNet配置", - "Status.index.551011-49": "诊断设备OneNet是否已配置,未配置将导致连接失败", - "Status.index.551011-50": "设备-OneNet配置是否已填写正确,若您确定该项无需诊断可", - "Status.index.551011-51": "设备-CTWing配置", - "Status.index.551011-52": "诊断设备CTWing是否已配置,未配置将导致连接失败", - "Status.index.551011-53": "设备-CTWing配置是否已填写正确,若您确定该项无需诊断可", - "Status.index.551011-54": "请检查设备运行状态是否正常", - "Status.index.551011-55": "请检查设备网络是否畅通,并确保设备已连接到以下地址之一:", - "Status.index.551011-56": "请检查设备网络是否畅通,并确保设备已连接到:", - "Status.index.551011-57": "请根据", - "Status.index.551011-58": "设备接入配置", - "Status.index.551011-59": "中", - "Status.index.551011-60": "信息,任意上报一条数据", - "Status.index.551011-61": "请联系管理员提供", - "Status.index.551011-62": "信息,并根据URL信息任意上报一条数据", - "Status.index.551011-63": "请检查设备网络是否畅通,并确保设备已连接到:SIP", - "Status.index.551011-64": "需要三方云平台主动发送一条消息通知到本平台,触发设备状态为在线", - "Status.index.551011-65": "请检查三方平台配置项是否填写正确", - "Status.index.551011-66": "未开发", - "Status.index.551011-67": "连接详情", - "Status.index.551011-68": "一键修复", - "Status.index.551011-69": "重新诊断", - "components.ModelButton.551013-0": "配置", - "Properties.StorageSetting.551013-0": "参数名称", - "Properties.StorageSetting.551013-1": "输入类型", - "Properties.StorageSetting.551013-2": "值", - "Function.OutputParams.551013-0": "参数标识", - "Function.OutputParams.551013-1": "该标识已存在", - "Function.OutputParams.551013-2": "请输入标识", - "Function.OutputParams.551013-3": "最多可输入64个字符", - "Function.OutputParams.551013-4": "标识只能由数字、字母、下划线、中划线组成", - "Function.OutputParams.551013-5": "参数名称", - "Function.OutputParams.551013-6": "请输入参数名称", - "Function.OutputParams.551013-7": "数据类型", - "Function.OutputParams.551013-8": "请选择数据类型", - "Function.OutputParams.551013-9": "其他配置", - "Function.OutputParams.551013-10": "操作", - "Events.SelectColumn.551013-0": "请选择", - "Import.index.551010-0": "导入", - "Import.index.551010-1": "产品", - "Import.index.551010-2": "请选择产品", - "Import.index.551010-3": "文件格式", - "Import.index.551010-4": "文件上传", - "Import.index.551010-5": "关闭", - "components.EditTable.626395-0": "最多可输入64个字符", - "components.EditTable.626395-1": "请输入{0}", - "components.EditTable.626395-2": "添加", - "components.EditTable.626395-3": "操作", - "Process.index.551010-0": "启用", - "Process.index.551010-1": "同步", - "Process.index.551010-2": "正在启用全部设备", - "Process.index.551010-3": "正在同步设备状态", - "Process.index.551010-4": "成功:", - "Process.index.551010-5": "条", - "Process.index.551010-6": "启用失败:", - "Process.index.551010-7": "实例信息页面中的配置项未完善", - "Process.index.551010-8": "完成", - "Info.index.506529-0": "设备信息", - "Info.index.506529-1": "编辑", - "Info.index.506529-2": "设备ID", - "Info.index.506529-3": "通过调用SDK或HTTP请求的方式接入第三方系统设备数据时,第三方系统与平台当前设备对应的设备ID。", - "Info.index.506529-4": "如双方ID值一致,则无需填写", - "Info.index.506529-5": "未映射", - "Info.index.506529-6": "已映射", - "Info.index.506529-7": "产品名称", - "Info.index.506529-8": "设备类型", - "Info.index.506529-9": "固件版本", - "Info.index.506529-10": "连接协议", - "Info.index.506529-11": "消息协议", - "Info.index.506529-12": "创建时间", - "Info.index.506529-13": "注册时间", - "Info.index.506529-14": "最后上线时间", - "Info.index.506529-15": "父设备", - "Info.index.506529-16": "说明", - "Save.GateWayFormItem.366202-0": "选择网关设备", - "Save.GateWayFormItem.366202-1": "重新选择", - "Save.GateWayFormItem.340184-0": "选择网关设备", - "Save.GateWayFormItem.340184-1": "重新选择", - "DeviceHome.index.152181-0": "物联网引导", - "DeviceHome.index.152181-1": "设备接入推荐步骤", - "DeviceHome.index.152181-2": "不同的设备因为通信协议的不同,存在接入步骤的差异", - "DeviceHome.index.152181-3": "创建产品", - "DeviceHome.index.152181-4": "创建设备", - "DeviceHome.index.152181-5": "规则引擎", - "DeviceHome.index.152181-6": "产品是设备的集合,通常指一组具有相同功能的设备。物联设备必须通过产品进行接入方式配置。", - "DeviceHome.index.152181-7": "配置产品接入方式", - "DeviceHome.index.152181-8": "通过产品对同一类型的设备进行统一的接入方式配置。请参照设备铭牌说明选择匹配的接入方式。", - "DeviceHome.index.152181-9": "添加测试设备", - "DeviceHome.index.152181-10": "添加单个设备,用于验证产品模型是否配置正确。", - "DeviceHome.index.152181-11": "功能调试", - "DeviceHome.index.152181-12": "对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确。", - "DeviceHome.index.152181-13": "批量添加设备", - "DeviceHome.index.152181-14": "批量添加同一产品下的设备", - "InitHome.index.152181-0": "请选择首页视图", - "InitHome.index.152181-1": "保存修改", - "ComprehensiveHome.index.152181-0": "物联网引导", - "ComprehensiveHome.index.152181-1": "运维引导", - "ComprehensiveHome.index.152181-2": "设备接入推荐步骤", - "ComprehensiveHome.index.152181-3": "不同的设备因为通信协议的不同,存在接入步骤的差异", - "ComprehensiveHome.index.152181-4": "运维管理推荐步骤", - "ComprehensiveHome.index.152181-5": "请根据业务需要对下述步骤进行选择性操作。", - "ComprehensiveHome.index.152181-6": "创建产品", - "ComprehensiveHome.index.152181-7": "创建设备", - "ComprehensiveHome.index.152181-8": "规则引擎", - "ComprehensiveHome.index.152181-9": "产品是设备的集合,通常指一组具有相同功能的设备。物联设备必须通过产品进行接入方式配置。", - "ComprehensiveHome.index.152181-10": "配置产品接入方式", - "ComprehensiveHome.index.152181-11": "通过产品对同一类型的设备进行统一的接入方式配置。请参照设备铭牌说明选择匹配的接入方式。", - "ComprehensiveHome.index.152181-12": "添加测试设备", - "ComprehensiveHome.index.152181-13": "添加单个设备,用于验证产品模型是否配置正确。", - "ComprehensiveHome.index.152181-14": "功能调试", - "ComprehensiveHome.index.152181-15": "对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确。", - "ComprehensiveHome.index.152181-16": "批量添加设备", - "ComprehensiveHome.index.152181-17": "批量添加同一产品下的设备", - "ComprehensiveHome.index.152181-18": "设备接入配置", - "ComprehensiveHome.index.152181-19": "日志排查", - "ComprehensiveHome.index.152181-20": "实时监控", - "ComprehensiveHome.index.152181-21": "协议管理", - "ComprehensiveHome.index.152181-22": "根据业务需求自定义开发对应的产品(设备模型)接入协议,并上传到平台。", - "ComprehensiveHome.index.152181-23": "证书管理", - "ComprehensiveHome.index.152181-24": "统一维护平台内的证书,用于数据通信加密。", - "ComprehensiveHome.index.152181-25": "网络组件", - "ComprehensiveHome.index.152181-26": "根据不同的传输类型配置平台底层网络组件相关参数。", - "ComprehensiveHome.index.152181-27": "设备接入网关", - "ComprehensiveHome.index.152181-28": "根据不同的传输类型,关联消息协议,配置设备接入网关相关参数。", - "ComprehensiveHome.index.152181-29": "日志管理", - "ComprehensiveHome.index.152181-30": "监控系统日志,及时处理系统异常。", - "dialogs.DeviceChooseDialog.152181-0": "选择设备", - "dialogs.DeviceChooseDialog.152181-1": "请选择设备", - "dialogs.DeviceChooseDialog.152181-2": "设备ID", - "dialogs.DeviceChooseDialog.152181-3": "设备名称", - "dialogs.DeviceChooseDialog.152181-4": "产品名称", - "dialogs.DeviceChooseDialog.152181-5": "注册时间", - "dialogs.DeviceChooseDialog.152181-6": "状态", - "dialogs.DeviceChooseDialog.152181-7": "在线", - "dialogs.DeviceChooseDialog.152181-8": "离线", - "components.BootCard.152181-0": "暂无权限,请联系管理员", - "components.BasicCountCard.152181-0": "基础统计", - "components.BasicCountCard.152181-1": "详情", - "components.BasicCountCard.152181-2": "CPU使用率", - "components.BasicCountCard.152181-3": "JVM内存", - "components.BootCardSmall.152181-0": "暂无权限,请联系管理员", - "components.StepCard.152181-0": "暂无权限,请联系管理员", - "dialogs.ProductChooseDialog.152181-0": "选择产品", - "dialogs.ProductChooseDialog.152181-1": "产品", - "dialogs.ProductChooseDialog.152181-2": "请选择产品", - "home.index.152180-0": "基本信息", - "components.PlatformPicCard.152181-0": "平台架构图", - "components.DeviceCountCard.152181-0": "设备统计", - "components.DeviceCountCard.152181-1": "详情", - "components.DeviceCountCard.152181-2": "产品数量", - "components.DeviceCountCard.152181-3": "设备数量", - "Save.GateWayDeviceModal.039083-0": "选择网关设备", - "Save.GateWayDeviceModal.039083-1": "设备类型", - "Save.GateWayDeviceModal.039083-2": "产品名称", - "Save.GateWayDeviceModal.039083-3": "设备名称", - "Save.GateWayDeviceModal.039083-4": "创建时间", - "Save.GateWayDeviceModal.039083-5": "状态", - "Save.GateWayDeviceModal.039083-6": "禁用", - "Save.GateWayDeviceModal.039083-7": "离线", - "Save.GateWayDeviceModal.039083-8": "在线", - "Save.index.039083-0": "编辑", - "Save.index.039083-1": "新增", - "Save.index.039083-2": "通道名称", - "Save.index.039083-3": "请输入通道名称", - "Save.index.039083-4": "通讯协议", - "Save.index.039083-5": "请选择通讯协议", - "Save.index.039083-6": "Modbus主机IP", - "Save.index.039083-7": "支持ipv4、ipv6、域名", - "Save.index.039083-8": "请输入Modbus主机IP", - "Save.index.039083-9": "端口", - "Save.index.039083-10": "请输入端口", - "Save.index.039083-11": "端点url", - "Save.index.039083-12": "请输入端点url", - "Save.index.039083-13": "安全策略", - "Save.index.039083-14": "请选择安全策略", - "Save.index.039083-15": "请选择网关设备", - "Save.index.039083-16": "选择网关设备", - "Save.index.039083-17": "安全模式", - "Save.index.039083-18": "请选择安全模式", - "Save.index.039083-19": "证书", - "Save.index.039083-20": "请选择证书", - "Save.index.039083-21": "权限认证", - "Save.index.039083-22": "用户名", - "Save.index.039083-23": "请输入用户名", - "Save.index.039083-24": "密码", - "Save.index.039083-25": "请输入密码", - "Save.index.039083-26": "BACNet实例号", - "Save.index.039083-27": "请输入BACNet实例号", - "Save.index.039083-28": "网卡", - "Save.index.039083-29": "请选择网卡", - "Save.index.039083-30": "广播端口", - "Save.index.039083-31": "请输入广播端口", - "Save.index.039083-32": "说明", - "Save.index.039083-33": "请输入说明", - "Save.index.039083-34": "取消", - "Save.index.039083-35": "确认", - "Save.index.039083-36": "请输入BACnet实例号", - "Save.index.039083-37": "请输入正确的BACnet实例号", - "Center.index.340051-0": "查看详情", - "Center.index.340051-1": "编辑资料", - "Center.index.340051-2": "修改密码", - "Center.index.340051-3": "首页视图", - "Center.index.340051-4": "绑定第三方账号", - "Center.index.340051-5": "我的订阅", - "Center.index.340051-6": "站内信", - "Center.index.340051-7": "操作成功", - "Center.index.361874-0": "查看详情", - "Center.index.361874-1": "编辑资料", - "Center.index.361874-2": "修改密码", - "Center.index.361874-3": "首页视图", - "Center.index.361874-4": "绑定第三方账号", - "Center.index.361874-5": "我的订阅", - "Center.index.361874-6": "站内信", - "Center.index.361874-7": "操作成功", - "Import.index.657743-0": "导入物模型", - "Import.index.657743-1": "导入的物模型会覆盖原来的属性、功能、事件、标签,请谨慎操作。", - "Import.index.657743-2": "导入时会根据标识跳过继承自产品物模型的属性、功能、事件、标签。", - "Import.index.657743-3": "导入方式", - "Import.index.657743-4": "请选择导入方式", - "Import.index.657743-5": "拷贝产品", - "Import.index.657743-6": "选择产品", - "Import.index.657743-7": "请选择产品", - "Import.index.657743-8": "物模型类型", - "Import.index.657743-9": "请选择物模型类型", - "Import.index.657743-10": "导入类型", - "Import.index.657743-11": "请选择导入类型", - "Import.index.657743-12": "文件上传", - "Import.index.657743-13": "脚本", - "Import.index.657743-14": "请上传文件", - "Import.index.657743-15": "上传文件", - "Import.index.657743-16": "支持扩展名:.json", - "Import.index.657743-17": "请输入物模型", - "Import.index.657743-18": "物模型", - "Import.index.657743-19": "在编辑器中编写物模型脚本", - "Import.index.657743-20": "属性定义第{0}个数组中缺失id属性", - "Import.index.657743-21": "属性定义第{0}个数组中id属性不能包含中文", - "Import.index.657743-22": "属性定义第{0}个数组中缺失name属性", - "Import.index.657743-23": "标签定义第{0}个数组中缺失valueType.type属性", - "Import.index.657743-24": "属性定义第{0}个数组中缺失expands.source属性", - "Import.index.657743-25": "属性定义第{0}个数组中缺失type属性", - "Import.index.657743-26": "方法定义第{0}个数组中缺失id属性", - "Import.index.657743-27": "方法定义第{0}个数组中id属性不能包含中文", - "Import.index.657743-28": "方法定义第{0}个数组中缺失name属性", - "Import.index.657743-29": "事件定义第{0}个数组中缺失id属性", - "Import.index.657743-30": "事件定义第{0}个数组中id属性不能包含中文", - "Import.index.657743-31": "事件定义第{0}个数组中缺失name属性", - "Import.index.657743-32": "事件定义第{0}个数组中缺失valueType.type属性", - "Import.index.657743-33": "事件定义第{0}个数组中缺失expands.level属性", - "Import.index.657743-34": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组第${\r\n number + 1\r\n }项的id属性", - "Import.index.657743-35": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组第${\r\n number + 1\r\n }项的name属性", - "Import.index.657743-36": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组第${\r\n number + 1\r\n }项的valueType.type属性", - "Import.index.657743-37": "事件定义第${\r\n index + 1\r\n }个数组中缺失valueType.properties数组", - "Import.index.657743-38": "标签定义第{0}个数组中缺失id属性", - "Import.index.657743-39": "标签定义第{0}个数组中id属性不能包含中文", - "Import.index.657743-40": "标签定义第{0}个数组中缺失name属性", - "Import.index.657743-41": "属性定义第{0}个数组中缺失identifier属性", - "Import.index.657743-42": "属性定义第{0}个数组中缺失dataType.type属性", - "Import.index.657743-43": "方法定义第{0}个数组中缺失identifier属性", - "Import.index.657743-44": "方法定义第{0}个数组中缺失callType属性", - "Import.index.657743-45": "事件定义第{0}个数组中缺失identifier属性", - "Import.index.657743-46": "事件定义第{0}个数组中缺失type属性", - "Import.index.657743-47": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的id属性", - "Import.index.657743-48": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的name属性", - "Import.index.657743-49": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的dataType.type属性", - "Import.index.657743-50": "事件定义第${\r\n index + 1\r\n }个数组中缺失outputData数组第${\r\n number + 1\r\n }项的dataType.specs属性", - "Import.index.657743-51": "事件定义第{0}个数组中缺失outputData数组", - "Import.index.657743-52": "操作成功!", - "Import.index.657743-53": "文件内容不能为空", - "Import.index.657743-54": "请上传json格式的文件", - "Import.index.657743-55": "导入成功", - "Import.index.657743-56": "物模型数据不正确", - "Import.index.657743-57": "上传json格式的物模型文件", - "Import.index.657743-58": "导入数据存在虚拟属性,请及时添加虚拟属性计算规则。", - "Import.index.657743-59": "确认", - "Import.index.657743-60": "标准物模型", - "ComprehensiveHome.index.926510-0": "物联网引导", - "ComprehensiveHome.index.926510-1": "运维引导", - "ComprehensiveHome.index.926510-2": "设备接入推荐步骤", - "ComprehensiveHome.index.926510-3": "不同的设备因为通信协议的不同,存在接入步骤的差异", - "ComprehensiveHome.index.926510-4": "运维管理推荐步骤", - "ComprehensiveHome.index.926510-5": "请根据业务需要对下述步骤进行选择性操作。", - "ComprehensiveHome.index.926510-6": "创建产品", - "ComprehensiveHome.index.926510-7": "创建设备", - "ComprehensiveHome.index.926510-8": "规则引擎", - "ComprehensiveHome.index.926510-9": "产品是设备的集合,通常指一组具有相同功能的设备。物联设备必须通过产品进行接入方式配置。", - "ComprehensiveHome.index.926510-10": "配置产品接入方式", - "ComprehensiveHome.index.926510-11": "通过产品对同一类型的设备进行统一的接入方式配置。请参照设备铭牌说明选择匹配的接入方式。", - "ComprehensiveHome.index.926510-12": "添加测试设备", - "ComprehensiveHome.index.926510-13": "添加单个设备,用于验证产品模型是否配置正确。", - "ComprehensiveHome.index.926510-14": "功能调试", - "ComprehensiveHome.index.926510-15": "对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确。", - "ComprehensiveHome.index.926510-16": "批量添加设备", - "ComprehensiveHome.index.926510-17": "批量添加同一产品下的设备", - "ComprehensiveHome.index.926510-18": "设备接入配置", - "ComprehensiveHome.index.926510-19": "日志排查", - "ComprehensiveHome.index.926510-20": "实时监控", - "ComprehensiveHome.index.926510-21": "协议管理", - "ComprehensiveHome.index.926510-22": "根据业务需求自定义开发对应的产品(设备模型)接入协议,并上传到平台。", - "ComprehensiveHome.index.926510-23": "证书管理", - "ComprehensiveHome.index.926510-24": "统一维护平台内的证书,用于数据通信加密。", - "ComprehensiveHome.index.926510-25": "网络组件", - "ComprehensiveHome.index.926510-26": "根据不同的传输类型配置平台底层网络组件相关参数。", - "ComprehensiveHome.index.926510-27": "设备接入网关", - "ComprehensiveHome.index.926510-28": "根据不同的传输类型,关联消息协议,配置设备接入网关相关参数。", - "ComprehensiveHome.index.926510-29": "日志管理", - "ComprehensiveHome.index.926510-30": "监控系统日志,及时处理系统异常。", - "components.BootCardSmall.926510-0": "暂无权限,请联系管理员", - "components.BasicCountCard.926510-0": "基础统计", - "components.BasicCountCard.926510-1": "详情", - "components.BasicCountCard.926510-2": "CPU使用率", - "components.BasicCountCard.926510-3": "JVM内存", - "components.BootCard.926510-0": "暂无权限,请联系管理员", - "dialogs.DeviceChooseDialog.926510-0": "选择设备", - "dialogs.DeviceChooseDialog.926510-1": "请选择设备", - "dialogs.DeviceChooseDialog.926510-2": "设备ID", - "dialogs.DeviceChooseDialog.926510-3": "设备名称", - "dialogs.DeviceChooseDialog.926510-4": "产品名称", - "dialogs.DeviceChooseDialog.926510-5": "注册时间", - "dialogs.DeviceChooseDialog.926510-6": "状态", - "dialogs.DeviceChooseDialog.926510-7": "在线", - "dialogs.DeviceChooseDialog.926510-8": "离线", - "DevOpsHome.index.926510-0": "运维引导", - "DevOpsHome.index.926510-1": "运维管理推荐步骤", - "DevOpsHome.index.926510-2": "请根据业务需要对下述步骤进行选择性操作。", - "DevOpsHome.index.926510-3": "设备接入配置", - "DevOpsHome.index.926510-4": "日志排查", - "DevOpsHome.index.926510-5": "实时监控", - "DevOpsHome.index.926510-6": "协议管理", - "DevOpsHome.index.926510-7": "根据业务需求自定义开发对应的产品(设备模型)接入协议,并上传到平台。", - "DevOpsHome.index.926510-8": "证书管理", - "DevOpsHome.index.926510-9": "统一维护平台内的证书,用于数据通信加密。", - "DevOpsHome.index.926510-10": "网络组件", - "DevOpsHome.index.926510-11": "根据不同的传输类型配置平台底层网络组件相关参数。", - "DevOpsHome.index.926510-12": "设备接入网关", - "DevOpsHome.index.926510-13": "根据不同的传输类型,关联消息协议,配置设备接入网关相关参数。", - "DevOpsHome.index.926510-14": "日志管理", - "DevOpsHome.index.926510-15": "监控系统日志,及时处理系统异常。", - "dialogs.ProductChooseDialog.926510-0": "选择产品", - "dialogs.ProductChooseDialog.926510-1": "产品", - "dialogs.ProductChooseDialog.926510-2": "请选择产品", - "components.DeviceCountCard.926510-0": "设备统计", - "components.DeviceCountCard.926510-1": "详情", - "components.DeviceCountCard.926510-2": "产品数量", - "components.DeviceCountCard.926510-3": "设备数量", - "home.index.926510-0": "基本信息", - "components.PlatformPicCard.926510-0": "平台架构图", - "InitHome.index.926510-0": "请选择首页视图", - "InitHome.index.926510-1": "保存修改", - "DeviceHome.index.926510-0": "物联网引导", - "DeviceHome.index.926510-1": "设备接入推荐步骤", - "DeviceHome.index.926510-2": "不同的设备因为通信协议的不同,存在接入步骤的差异", - "DeviceHome.index.926510-3": "创建产品", - "DeviceHome.index.926510-4": "创建设备", - "DeviceHome.index.926510-5": "规则引擎", - "DeviceHome.index.926510-6": "产品是设备的集合,通常指一组具有相同功能的设备。物联设备必须通过产品进行接入方式配置。", - "DeviceHome.index.926510-7": "配置产品接入方式", - "DeviceHome.index.926510-8": "通过产品对同一类型的设备进行统一的接入方式配置。请参照设备铭牌说明选择匹配的接入方式。", - "DeviceHome.index.926510-9": "添加测试设备", - "DeviceHome.index.926510-10": "添加单个设备,用于验证产品模型是否配置正确。", - "DeviceHome.index.926510-11": "功能调试", - "DeviceHome.index.926510-12": "对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确。", - "DeviceHome.index.926510-13": "批量添加设备", - "DeviceHome.index.926510-14": "批量添加同一产品下的设备", - "components.StepCard.926510-0": "暂无权限,请联系管理员" -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926838.json b/src/i18n/temp/home_2024-7-4-926838.json deleted file mode 100644 index 4cb9fbf..0000000 --- a/src/i18n/temp/home_2024-7-4-926838.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "zh": { - "components.StepCard.926510-0": "暂无权限,请联系管理员" - }, - "en": { - "components.StepCard.926510-0": "No permission at the moment, please contact the administrator" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926841.json b/src/i18n/temp/home_2024-7-4-926841.json deleted file mode 100644 index 9e0eed2..0000000 --- a/src/i18n/temp/home_2024-7-4-926841.json +++ /dev/null @@ -1,36 +0,0 @@ -{ - "zh": { - "DeviceHome.index.926510-0": "物联网引导", - "DeviceHome.index.926510-1": "设备接入推荐步骤", - "DeviceHome.index.926510-2": "不同的设备因为通信协议的不同,存在接入步骤的差异", - "DeviceHome.index.926510-3": "创建产品", - "DeviceHome.index.926510-4": "创建设备", - "DeviceHome.index.926510-5": "规则引擎", - "DeviceHome.index.926510-6": "产品是设备的集合,通常指一组具有相同功能的设备。物联设备必须通过产品进行接入方式配置。", - "DeviceHome.index.926510-7": "配置产品接入方式", - "DeviceHome.index.926510-8": "通过产品对同一类型的设备进行统一的接入方式配置。请参照设备铭牌说明选择匹配的接入方式。", - "DeviceHome.index.926510-9": "添加测试设备", - "DeviceHome.index.926510-10": "添加单个设备,用于验证产品模型是否配置正确。", - "DeviceHome.index.926510-11": "功能调试", - "DeviceHome.index.926510-12": "对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确。", - "DeviceHome.index.926510-13": "批量添加设备", - "DeviceHome.index.926510-14": "批量添加同一产品下的设备" - }, - "en": { - "DeviceHome.index.926510-0": "IoT guidance", - "DeviceHome.index.926510-1": "Recommended steps for device access", - "DeviceHome.index.926510-2": "Different devices may have different access steps due to differences in communication protocols", - "DeviceHome.index.926510-3": "Create product", - "DeviceHome.index.926510-4": "Create device", - "DeviceHome.index.926510-5": "Rule engine", - "DeviceHome.index.926510-6": "A product is a collection of devices, usually referring to a group of devices with the same functionality. IoT devices must be configured through product access methods.", - "DeviceHome.index.926510-7": "Configure product access methods", - "DeviceHome.index.926510-8": "Configure unified access methods for devices of the same type through products. Please refer to the instructions on the device nameplate to select the matching access method.", - "DeviceHome.index.926510-9": "Add testing equipment", - "DeviceHome.index.926510-10": "Add a single device to verify if the product model is configured correctly.", - "DeviceHome.index.926510-11": "Functional debugging", - "DeviceHome.index.926510-12": "Perform functional debugging on the added testing equipment to verify if it can connect to the platform and if the device's functions are configured correctly.", - "DeviceHome.index.926510-13": "Batch Add Devices", - "DeviceHome.index.926510-14": "Batch adding devices under the same product" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926843.json b/src/i18n/temp/home_2024-7-4-926843.json deleted file mode 100644 index 38ac6c6..0000000 --- a/src/i18n/temp/home_2024-7-4-926843.json +++ /dev/null @@ -1,10 +0,0 @@ -{ - "zh": { - "InitHome.index.926510-0": "请选择首页视图", - "InitHome.index.926510-1": "保存修改" - }, - "en": { - "InitHome.index.926510-0": "Please select the homepage view", - "InitHome.index.926510-1": "Save modifications" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926845.json b/src/i18n/temp/home_2024-7-4-926845.json deleted file mode 100644 index 3ff40f9..0000000 --- a/src/i18n/temp/home_2024-7-4-926845.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "zh": { - "components.PlatformPicCard.926510-0": "平台架构图" - }, - "en": { - "components.PlatformPicCard.926510-0": "Platform architecture diagram" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926846.json b/src/i18n/temp/home_2024-7-4-926846.json deleted file mode 100644 index 9ab032e..0000000 --- a/src/i18n/temp/home_2024-7-4-926846.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "zh": { - "home.index.926510-0": "基本信息" - }, - "en": { - "home.index.926510-0": "essential information" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926849.json b/src/i18n/temp/home_2024-7-4-926849.json deleted file mode 100644 index c9a0cd5..0000000 --- a/src/i18n/temp/home_2024-7-4-926849.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "zh": { - "components.DeviceCountCard.926510-0": "设备统计", - "components.DeviceCountCard.926510-1": "详情", - "components.DeviceCountCard.926510-2": "产品数量", - "components.DeviceCountCard.926510-3": "设备数量" - }, - "en": { - "components.DeviceCountCard.926510-0": "Equipment statistics", - "components.DeviceCountCard.926510-1": "details", - "components.DeviceCountCard.926510-2": "Product quantity", - "components.DeviceCountCard.926510-3": "Number of devices" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926853.json b/src/i18n/temp/home_2024-7-4-926853.json deleted file mode 100644 index 5749f90..0000000 --- a/src/i18n/temp/home_2024-7-4-926853.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "zh": { - "dialogs.ProductChooseDialog.926510-0": "选择产品", - "dialogs.ProductChooseDialog.926510-1": "产品", - "dialogs.ProductChooseDialog.926510-2": "请选择产品" - }, - "en": { - "dialogs.ProductChooseDialog.926510-0": "Select product", - "dialogs.ProductChooseDialog.926510-1": "product", - "dialogs.ProductChooseDialog.926510-2": "Please select a product" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926855.json b/src/i18n/temp/home_2024-7-4-926855.json deleted file mode 100644 index 9f83e38..0000000 --- a/src/i18n/temp/home_2024-7-4-926855.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "zh": { - "DevOpsHome.index.926510-0": "运维引导", - "DevOpsHome.index.926510-1": "运维管理推荐步骤", - "DevOpsHome.index.926510-2": "请根据业务需要对下述步骤进行选择性操作。", - "DevOpsHome.index.926510-3": "设备接入配置", - "DevOpsHome.index.926510-4": "日志排查", - "DevOpsHome.index.926510-5": "实时监控", - "DevOpsHome.index.926510-6": "协议管理", - "DevOpsHome.index.926510-7": "根据业务需求自定义开发对应的产品(设备模型)接入协议,并上传到平台。", - "DevOpsHome.index.926510-8": "证书管理", - "DevOpsHome.index.926510-9": "统一维护平台内的证书,用于数据通信加密。", - "DevOpsHome.index.926510-10": "网络组件", - "DevOpsHome.index.926510-11": "根据不同的传输类型配置平台底层网络组件相关参数。", - "DevOpsHome.index.926510-12": "设备接入网关", - "DevOpsHome.index.926510-13": "根据不同的传输类型,关联消息协议,配置设备接入网关相关参数。", - "DevOpsHome.index.926510-14": "日志管理", - "DevOpsHome.index.926510-15": "监控系统日志,及时处理系统异常。" - }, - "en": { - "DevOpsHome.index.926510-0": "Operations guidance", - "DevOpsHome.index.926510-1": "Recommended steps for operation and maintenance management", - "DevOpsHome.index.926510-2": "Please perform selective operations on the following steps according to business needs.", - "DevOpsHome.index.926510-3": "Device access configuration", - "DevOpsHome.index.926510-4": "Log troubleshooting", - "DevOpsHome.index.926510-5": "Real time monitoring", - "DevOpsHome.index.926510-6": "Protocol management", - "DevOpsHome.index.926510-7": "Customize and develop corresponding product (device model) access protocols according to business needs, and upload them to the platform.", - "DevOpsHome.index.926510-8": "Certificate Management", - "DevOpsHome.index.926510-9": "Unified maintenance of certificates within the platform for data communication encryption.", - "DevOpsHome.index.926510-10": "Network components", - "DevOpsHome.index.926510-11": "Configure platform underlying network component related parameters based on different transmission types.", - "DevOpsHome.index.926510-12": "Device access gateway", - "DevOpsHome.index.926510-13": "According to different transmission types, associate message protocols, and configure device access gateway related parameters.", - "DevOpsHome.index.926510-14": "Log management", - "DevOpsHome.index.926510-15": "Monitor system logs and promptly handle system exceptions." - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926856.json b/src/i18n/temp/home_2024-7-4-926856.json deleted file mode 100644 index 0ef7cd8..0000000 --- a/src/i18n/temp/home_2024-7-4-926856.json +++ /dev/null @@ -1,24 +0,0 @@ -{ - "zh": { - "dialogs.DeviceChooseDialog.926510-0": "选择设备", - "dialogs.DeviceChooseDialog.926510-1": "请选择设备", - "dialogs.DeviceChooseDialog.926510-2": "设备ID", - "dialogs.DeviceChooseDialog.926510-3": "设备名称", - "dialogs.DeviceChooseDialog.926510-4": "产品名称", - "dialogs.DeviceChooseDialog.926510-5": "注册时间", - "dialogs.DeviceChooseDialog.926510-6": "状态", - "dialogs.DeviceChooseDialog.926510-7": "在线", - "dialogs.DeviceChooseDialog.926510-8": "离线" - }, - "en": { - "dialogs.DeviceChooseDialog.926510-0": "Select device", - "dialogs.DeviceChooseDialog.926510-1": "Please select a device", - "dialogs.DeviceChooseDialog.926510-2": "Device ID", - "dialogs.DeviceChooseDialog.926510-3": "Device Name", - "dialogs.DeviceChooseDialog.926510-4": "Product Name", - "dialogs.DeviceChooseDialog.926510-5": "Registration time", - "dialogs.DeviceChooseDialog.926510-6": "state", - "dialogs.DeviceChooseDialog.926510-7": "on-line", - "dialogs.DeviceChooseDialog.926510-8": "off-line" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926861.json b/src/i18n/temp/home_2024-7-4-926861.json deleted file mode 100644 index 735a27d..0000000 --- a/src/i18n/temp/home_2024-7-4-926861.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "zh": { - "components.BootCard.926510-0": "暂无权限,请联系管理员" - }, - "en": { - "components.BootCard.926510-0": "No permission at the moment, please contact the administrator" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926866.json b/src/i18n/temp/home_2024-7-4-926866.json deleted file mode 100644 index a579899..0000000 --- a/src/i18n/temp/home_2024-7-4-926866.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "zh": { - "components.BasicCountCard.926510-0": "基础统计", - "components.BasicCountCard.926510-1": "详情", - "components.BasicCountCard.926510-2": "CPU使用率", - "components.BasicCountCard.926510-3": "JVM内存" - }, - "en": { - "components.BasicCountCard.926510-0": "Basic statistics", - "components.BasicCountCard.926510-1": "details", - "components.BasicCountCard.926510-2": "CPU usage rate", - "components.BasicCountCard.926510-3": "JVM memory" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926873.json b/src/i18n/temp/home_2024-7-4-926873.json deleted file mode 100644 index 6381e02..0000000 --- a/src/i18n/temp/home_2024-7-4-926873.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "zh": { - "components.BootCardSmall.926510-0": "暂无权限,请联系管理员" - }, - "en": { - "components.BootCardSmall.926510-0": "No permission at the moment, please contact the administrator" - } -} \ No newline at end of file diff --git a/src/i18n/temp/home_2024-7-4-926911.json b/src/i18n/temp/home_2024-7-4-926911.json deleted file mode 100644 index 7bbaae0..0000000 --- a/src/i18n/temp/home_2024-7-4-926911.json +++ /dev/null @@ -1,68 +0,0 @@ -{ - "zh": { - "ComprehensiveHome.index.926510-0": "物联网引导", - "ComprehensiveHome.index.926510-1": "运维引导", - "ComprehensiveHome.index.926510-2": "设备接入推荐步骤", - "ComprehensiveHome.index.926510-3": "不同的设备因为通信协议的不同,存在接入步骤的差异", - "ComprehensiveHome.index.926510-4": "运维管理推荐步骤", - "ComprehensiveHome.index.926510-5": "请根据业务需要对下述步骤进行选择性操作。", - "ComprehensiveHome.index.926510-6": "创建产品", - "ComprehensiveHome.index.926510-7": "创建设备", - "ComprehensiveHome.index.926510-8": "规则引擎", - "ComprehensiveHome.index.926510-9": "产品是设备的集合,通常指一组具有相同功能的设备。物联设备必须通过产品进行接入方式配置。", - "ComprehensiveHome.index.926510-10": "配置产品接入方式", - "ComprehensiveHome.index.926510-11": "通过产品对同一类型的设备进行统一的接入方式配置。请参照设备铭牌说明选择匹配的接入方式。", - "ComprehensiveHome.index.926510-12": "添加测试设备", - "ComprehensiveHome.index.926510-13": "添加单个设备,用于验证产品模型是否配置正确。", - "ComprehensiveHome.index.926510-14": "功能调试", - "ComprehensiveHome.index.926510-15": "对添加的测试设备进行功能调试,验证能否连接到平台,设备功能是否配置正确。", - "ComprehensiveHome.index.926510-16": "批量添加设备", - "ComprehensiveHome.index.926510-17": "批量添加同一产品下的设备", - "ComprehensiveHome.index.926510-18": "设备接入配置", - "ComprehensiveHome.index.926510-19": "日志排查", - "ComprehensiveHome.index.926510-20": "实时监控", - "ComprehensiveHome.index.926510-21": "协议管理", - "ComprehensiveHome.index.926510-22": "根据业务需求自定义开发对应的产品(设备模型)接入协议,并上传到平台。", - "ComprehensiveHome.index.926510-23": "证书管理", - "ComprehensiveHome.index.926510-24": "统一维护平台内的证书,用于数据通信加密。", - "ComprehensiveHome.index.926510-25": "网络组件", - "ComprehensiveHome.index.926510-26": "根据不同的传输类型配置平台底层网络组件相关参数。", - "ComprehensiveHome.index.926510-27": "设备接入网关", - "ComprehensiveHome.index.926510-28": "根据不同的传输类型,关联消息协议,配置设备接入网关相关参数。", - "ComprehensiveHome.index.926510-29": "日志管理", - "ComprehensiveHome.index.926510-30": "监控系统日志,及时处理系统异常。" - }, - "en": { - "ComprehensiveHome.index.926510-0": "IoT guidance", - "ComprehensiveHome.index.926510-1": "Operations guidance", - "ComprehensiveHome.index.926510-2": "Recommended steps for device access", - "ComprehensiveHome.index.926510-3": "Different devices may have different access steps due to differences in communication protocols", - "ComprehensiveHome.index.926510-4": "Recommended steps for operation and maintenance management", - "ComprehensiveHome.index.926510-5": "Please perform selective operations on the following steps according to business needs.", - "ComprehensiveHome.index.926510-6": "Create product", - "ComprehensiveHome.index.926510-7": "Create device", - "ComprehensiveHome.index.926510-8": "Rule engine", - "ComprehensiveHome.index.926510-9": "A product is a collection of devices, usually referring to a group of devices with the same functionality. IoT devices must be configured through product access methods.", - "ComprehensiveHome.index.926510-10": "Configure product access methods", - "ComprehensiveHome.index.926510-11": "Configure unified access methods for devices of the same type through products. Please refer to the instructions on the device nameplate to select the matching access method.", - "ComprehensiveHome.index.926510-12": "Add testing equipment", - "ComprehensiveHome.index.926510-13": "Add a single device to verify if the product model is configured correctly.", - "ComprehensiveHome.index.926510-14": "Functional debugging", - "ComprehensiveHome.index.926510-15": "Perform functional debugging on the added testing equipment to verify if it can connect to the platform and if the device's functions are configured correctly.", - "ComprehensiveHome.index.926510-16": "Batch Add Devices", - "ComprehensiveHome.index.926510-17": "Batch adding devices under the same product", - "ComprehensiveHome.index.926510-18": "Device access configuration", - "ComprehensiveHome.index.926510-19": "Log troubleshooting", - "ComprehensiveHome.index.926510-20": "Real time monitoring", - "ComprehensiveHome.index.926510-21": "Protocol management", - "ComprehensiveHome.index.926510-22": "Customize and develop corresponding product (device model) access protocols according to business needs, and upload them to the platform.", - "ComprehensiveHome.index.926510-23": "Certificate Management", - "ComprehensiveHome.index.926510-24": "Unified maintenance of certificates within the platform for data communication encryption.", - "ComprehensiveHome.index.926510-25": "Network components", - "ComprehensiveHome.index.926510-26": "Configure platform underlying network component related parameters based on different transmission types.", - "ComprehensiveHome.index.926510-27": "Device access gateway", - "ComprehensiveHome.index.926510-28": "According to different transmission types, associate message protocols, and configure device access gateway related parameters.", - "ComprehensiveHome.index.926510-29": "Log management", - "ComprehensiveHome.index.926510-30": "Monitor system logs and promptly handle system exceptions." - } -} \ No newline at end of file diff --git a/src/i18n/temp/user_2024-7-4-265143.json b/src/i18n/temp/user_2024-7-4-265143.json deleted file mode 100644 index d9b8298..0000000 --- a/src/i18n/temp/user_2024-7-4-265143.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "zh": { - "Login.index.265048-0": "备案:", - "Login.index.265048-1": "渝ICP备19017719号-1", - "Login.index.265048-2": "账号", - "Login.index.265048-3": "请输入账号", - "Login.index.265048-4": "密码", - "Login.index.265048-5": "请输入密码", - "Login.index.265048-6": "验证码", - "Login.index.265048-7": "请输入验证码", - "Login.index.265048-8": "记住我", - "Login.index.265048-9": "登录", - "Login.index.265048-10": "其他登录方式", - "Login.index.265048-11": "JETLINKS团队全新力作可视化大屏系统", - "Login.index.265048-12": "体验DEMO", - "Login.index.265048-13": "请输入账号!", - "Login.index.265048-14": "请输入密码!", - "Login.index.265048-15": "请输入验证码!" - }, - "en": { - "Login.index.265048-0": "keep on record:", - "Login.index.265048-1": "YuICPBei No. 19017719-1", - "Login.index.265048-2": "account number", - "Login.index.265048-3": "Please enter your account number", - "Login.index.265048-4": "password", - "Login.index.265048-5": "Please input a password", - "Login.index.265048-6": "Verification Code", - "Login.index.265048-7": "Please enter the verification code", - "Login.index.265048-8": "Remember me", - "Login.index.265048-9": "Sign in", - "Login.index.265048-10": "Other login methods", - "Login.index.265048-11": "JETLINKS team's new visual large screen system", - "Login.index.265048-12": "Experience DEMO", - "Login.index.265048-13": "Please enter your account!", - "Login.index.265048-14": "Please input a password!", - "Login.index.265048-15": "Please enter the verification code!" - } -} \ No newline at end of file diff --git a/src/style/global.less b/src/style/global.less deleted file mode 100644 index 8440adb..0000000 --- a/src/style/global.less +++ /dev/null @@ -1,2 +0,0 @@ -@import "./scrollbar.less"; -@import "./modal.less"; diff --git a/src/style/modal.less b/src/style/modal.less deleted file mode 100644 index 4431f6e..0000000 --- a/src/style/modal.less +++ /dev/null @@ -1,9 +0,0 @@ -.ant-modal-content { - border-radius: 8px !important; - - .ant-modal-header { - border-top-left-radius: 8px !important; - border-top-right-radius: 8px !important; - } -} - diff --git a/src/style/scrollbar.less b/src/style/scrollbar.less deleted file mode 100644 index f3706a5..0000000 --- a/src/style/scrollbar.less +++ /dev/null @@ -1,28 +0,0 @@ -/* 自定义滚动条整体样式 */ -::-webkit-scrollbar { - width: 6px; /* 对于垂直滚动条,设置宽度 */ - height: 10px; /* 对于水平滚动条,设置高度 */ - -} - -/* 自定义滚动条轨道样式 */ -::-webkit-scrollbar-track { - background: transparent; /* 背景色 */ -} - -/* 自定义滚动条滑块样式 */ -::-webkit-scrollbar-thumb { - background: #E7E9EF; /* 滑块颜色 */ - border-radius: 5px; -} - -/* 当滑块悬停或活动时的样式 */ -::-webkit-scrollbar-thumb:hover { - background: #E7E9EF; /* 滑块悬停颜色 */ -} - -/* 针对Select下拉菜单的滚动条的样式 */ -//.rc-virtual-list-scrollbar { -// scrollbar-color: #d8d8d8 #fafafa; /* 滚动条的颜色 */ -// width: 5px !important;/* 滚动条的宽度 */ -//} diff --git a/src/views/Northbound/AliCloud/Tree/index.vue b/src/views/Northbound/AliCloud/Tree/index.vue deleted file mode 100644 index c66543d..0000000 --- a/src/views/Northbound/AliCloud/Tree/index.vue +++ /dev/null @@ -1,134 +0,0 @@ -<template> - <PermissionButton - type="primary" - @click="handleAdd" - block - ghost - hasPermission="Northbound/DuerOS:add" - > - <template #icon><AIcon type="PlusOutlined" /></template> - 新增 - </PermissionButton> - <div class="listContainer"> - <template v-if="treeData?.length"> - <div - v-for="i in treeData" - class="listItem" - @click="() => selectItem(i)" - :class="{ selected: i.id === selectedId }" - > - <div class="itemText"> - <Ellipsis style="width: 100% "> - {{ i.name }} - </Ellipsis> - </div> - <div - class="status" - :style="{ - color: - i.state?.value === 'disabled' ? 'red' : '#70b603', - }" - > - {{ i.state?.value === 'disabled' ? '禁用' : '启用' }} - </div> - </div> - </template> - <j-empty v-else style="margin-top: 200px"></j-empty> - </div> -</template> - -<script setup> -import { queryPaginateNot } from '@/api/northbound/alicloud'; -import { defineExpose } from 'vue'; -const props = defineProps({ - params: { - type: Object, - }, -}); -const emit = defineEmits(['viewData']); -const treeData = ref(); -const selectedId = ref(); -const queryList = async (changeSelect = false) => { - const queryParams = props.params - ? { - paging: false, - sorts: [{ name: 'createTime', order: 'desc' }], - ...props.params, - } - : { - paging: false, - sorts: [{ name: 'createTime', order: 'desc' }], - }; - const res = await queryPaginateNot(queryParams); - if (res.success) { - treeData.value = res.result; - if (treeData.value.length) { - !changeSelect ? selectItem(treeData.value[0]) : selectItem(treeData.value.find((i)=>{ - return selectedId.value = i.id - })) - }else{ - selectedId.value = ''; - emit("viewData",{ - type:'noData' - }) - } - } -}; -const selectItem = (data) => { - selectedId.value = data.id; - emit('viewData', data); -}; -const handleAdd = () => { - selectedId.value = ''; - emit('viewData', { - type:'add' - }); -}; - -const refresh = (changeSelect) => { - queryList(changeSelect); -}; - -watch( - () => props.params, - () => { - queryList(); - }, - { - deep: true, - immediate: true, - }, -); - -defineExpose({ - refresh, -}); -</script> -<style lang="less" scoped> -.listContainer { - margin-top: 10px; - height: calc(100% - 50px); - overflow-y: auto; - &::-webkit-scrollbar { - width: 5px; /* 滚动条宽度 */ - background-color: #edf5ff; /* 滚动条背景色 */ - } - &::-webkit-scrollbar-thumb { - background-color: #d0d0d0; /* 滚动条拖动部分颜色 */ - border-radius: 4px; /* 滚动条拖动部分圆角 */ - } -} -.itemText{ - width: calc(100% - 50px) -} -.listItem { - display: flex; - justify-content: space-between; - margin-top: 5px; - cursor: pointer; - padding: 5px; -} -.selected { - background-color: #f2f2f2; -} -</style> diff --git a/src/views/Northbound/DuerOS/Tree/index.vue b/src/views/Northbound/DuerOS/Tree/index.vue deleted file mode 100644 index 5966948..0000000 --- a/src/views/Northbound/DuerOS/Tree/index.vue +++ /dev/null @@ -1,138 +0,0 @@ -<template> - <PermissionButton - type="primary" - @click="handleAdd" - block - ghost - hasPermission="Northbound/DuerOS:add" - > - <template #icon><AIcon type="PlusOutlined" /></template> - 新增 - </PermissionButton> - <div class="listContainer"> - <template v-if="treeData?.length"> - <div - v-for="i in treeData" - class="listItem" - @click="() => selectItem(i)" - :class="{ selected: i.id === selectedId }" - > - <div class="itemText"> - <Ellipsis style="width: 100%;"> - {{ i.name }} - </Ellipsis> - </div> - <div - class="status" - :style="{ - color: - i.state?.value === 'disabled' ? 'red' : '#70b603', - }" - > - {{ i.state?.value === 'disabled' ? '禁用' : '启用' }} - </div> - </div> - </template> - <j-empty v-else style="margin-top: 200px"></j-empty> - </div> -</template> - -<script setup> -import { queryPaginateNot } from '@/api/northbound/dueros'; -import { defineExpose } from 'vue'; -const props = defineProps({ - params: { - type: Object, - }, -}); -const emit = defineEmits(['viewData']); -const treeData = ref(); -const selectedId = ref(); -const queryList = async (changeSelect = false) => { - const queryParams = props.params - ? { - paging: false, - sorts: [{ name: 'createTime', order: 'desc' }], - ...props.params, - } - : { - paging: false, - sorts: [{ name: 'createTime', order: 'desc' }], - }; - const res = await queryPaginateNot(queryParams); - if (res.success) { - treeData.value = res.result; - if (treeData.value.length) { - !changeSelect - ? selectItem(treeData.value[0]) - : selectItem( - treeData.value.find((i) => { - return (selectedId.value = i.id); - }), - ); - } else { - selectedId.value = ''; - emit('viewData', { - type: 'noData', - }); - } - } -}; -const selectItem = (data) => { - selectedId.value = data.id; - emit('viewData', data); -}; -const handleAdd = () => { - selectedId.value = ''; - emit('viewData', { - type: 'add', - }); -}; - -const refresh = (changeSelect) => { - queryList(changeSelect); -}; - -watch( - () => props.params, - () => { - queryList(); - }, - { - deep: true, - immediate: true, - }, -); - -defineExpose({ - refresh, -}); -</script> -<style lang="less" scoped> -.listContainer { - margin-top: 10px; - height: calc(100% - 50px); - overflow-y: auto; - &::-webkit-scrollbar { - width: 5px; /* 滚动条宽度 */ - background-color: #edf5ff; /* 滚动条背景色 */ - } - &::-webkit-scrollbar-thumb { - background-color: #d0d0d0; /* 滚动条拖动部分颜色 */ - border-radius: 4px; /* 滚动条拖动部分圆角 */ - } -} -.listItem { - display: flex; - justify-content: space-between; - margin-top: 5px; - cursor: pointer; - padding: 5px; -} -.itemText{ - width: calc(100% - 50px); -} -.selected { - background-color: #f2f2f2; -} -</style> diff --git a/src/views/device/Instance/Detail/AlarmRecord/Alarm/components/AlarmLog.vue b/src/views/device/Instance/Detail/AlarmRecord/Alarm/components/AlarmLog.vue deleted file mode 100644 index 8e4c7e8..0000000 --- a/src/views/device/Instance/Detail/AlarmRecord/Alarm/components/AlarmLog.vue +++ /dev/null @@ -1,83 +0,0 @@ -<template> - <a-drawer - visible - :closable="false" - width="1000" - @close="closeDrawer" - :destroyInactiveTabPane="true" - > - <div class="alarmInfo"> - <div class="title">告警日志</div> - <div class="alarmInfoRight"> - <div> - {{ levelMap?.[AlarmData?.level] || AlarmData?.level }} - </div> - <div> - <BadgeStatus - :status="AlarmData?.state.value" - :statusNames="{ - warning: 'error', - normal: 'default', - }" - > - </BadgeStatus - ><span> - {{ AlarmData?.state.text }} - </span> - <a-button - v-if="AlarmData?.state.value === 'warning'" - type="link" - @click="dealAlarm" - >处理</a-button - > - </div> - </div> - </div> - <Log :currentId="AlarmData.id" :configId="AlarmData.alarmConfigId" /> - </a-drawer> - <SolveComponent - v-if="solveVisible" - @closeSolve="closeSolve" - @refresh="refresh" - :data="AlarmData" - /> -</template> - -<script setup> -import { queryLevel } from '@/api/rule-engine/config'; -import Log from '@/views/rule-engine/Alarm/Log/TabComponent/components/Log.vue'; -import SolveComponent from '@/views/rule-engine/Alarm/Log/SolveComponent/index.vue'; -import { useAlarmLevel } from '@/hook'; -const props = defineProps({ - data: { - type: Object, - default: {}, - }, -}); -const emit = defineEmits(['closeDrawer', 'refreshTable']); -const { levelMap } = useAlarmLevel(); -const solveVisible = ref(false); -const AlarmData = computed(()=>{ - return props.data -}) -const closeDrawer = () => { - emit('closeDrawer'); -}; -const closeSolve = () => { - solveVisible.value = false; -}; -const refresh = () => { - solveVisible.value = false; - emit('refreshTable'); -}; -const dealAlarm = () => { - solveVisible.value = true; -}; - -</script> -<style lang="less" scoped> -.alarmInfo { - display: flex; - justify-content: space-between; -} -</style> diff --git a/src/views/device/Instance/Detail/AlarmRecord/Alarm/index.vue b/src/views/device/Instance/Detail/AlarmRecord/Alarm/index.vue deleted file mode 100644 index b9e9b69..0000000 --- a/src/views/device/Instance/Detail/AlarmRecord/Alarm/index.vue +++ /dev/null @@ -1,444 +0,0 @@ -<template> - <pro-search - :columns="columns" - target="device-instance" - type="simple" - @search="handleSearch" - /> - <JProTable - ref="deviceAlarm" - :columns="columns" - model="TABLE" - :request="queryAlarmRecord" - :defaultParams="defaultParams" - :params="params" - > - <template #alarmTime="slotProps"> - {{ dayjs(slotProps.alarmTime).format('YYYY-MM-DD HH:mm:ss') }} - </template> - <template #duration="slotProps"> - <Duration :data="slotProps" /> - </template> - <template #handleTime="slotProps"> - {{ - slotProps.handleTime - ? dayjs(slotProps.handleTime).format('YYYY-MM-DD HH:mm:ss') - : '--' - }} - </template> - <template #sourceId="slotProps"> - <Ellipsis> - 设备ID: - <span class="deviceId" @click="() => gotoDevice(slotProps.sourceId)">{{ slotProps.sourceId }}</span></Ellipsis - > - </template> - <template #handleType="slotProps"> - {{ slotProps?.handleType?.text || '--' }} - </template> - <template #state="slotProps"> - {{ slotProps?.state?.value === 'normal' ? '已处理' : '告警中' }} - </template> - <template #actions="slotProps"> - <j-space> - <template v-for="i in getActions(slotProps)" :key="i.key"> - <PermissionButton - :popConfirm="i.popConfirm" - :tooltip="{ - ...i.tooltip, - }" - @click="i.onClick" - type="link" - style="padding: 0 5px" - :hasPermission=" - i.key == 'solve' - ? 'rule-engine/Alarm/Log:action' - : 'rule-engine/Alarm/Log:view' - " - > - {{ i.text }} - </PermissionButton> - </template> - </j-space> - </template> - </JProTable> - <Solve - v-if="solveVisible" - :data="currentAlarm" - :solveType="solveType" - :handleDes="handleDescription" - @closeSolve="closeSolve" - @refresh="solveRefresh" - /> - <AlarmLog - v-if="visibleDrawer" - :data="currentAlarm" - @closeDrawer="visibleDrawer = false" - @refreshTable="refresh" - /> -</template> - -<script setup> -import { - queryByDevice as queryAlarmRecord, - queryHandleHistory, -} from '@/api/rule-engine/log'; -import { useInstanceStore } from '@/store/instance'; -import { useProductStore } from '@/store/product'; -import dayjs from 'dayjs'; -import Duration from '@/views/rule-engine/Alarm/Log/components/Duration.vue'; -import Solve from '@/views/rule-engine/Alarm/Log/SolveComponent/index.vue'; -import AlarmLog from './components/AlarmLog.vue'; -import { useMenuStore } from 'store/menu'; -const props = defineProps({ - goal: { - type: String, - default: 'device', - }, -}); -const menuStory = useMenuStore(); -const { current } = - props.goal === 'device' ? useInstanceStore() : useProductStore(); -const columns = - props.goal === 'device' - ? [ - { - title: '告警时间', - dataIndex: 'alarmTime', - key: 'alarmTime', - search: { - type: 'date', - }, - scopedSlots: true, - }, - { - title: '告警持续时长', - dataIndex: 'duration', - key: 'duration', - scopedSlots: true, - }, - { - title: '触发条件', - dataIndex: 'triggerDesc', - key: 'triggerDesc', - }, - { - title: '告警原因', - dataIndex: 'actualDesc', - key: 'actualDesc', - }, - { - title: '处理时间', - dataIndex: 'handleTime', - key: 'handleTime', - search: { - type: 'date', - }, - scopedSlots: true, - }, - - { - title: '处理类型', - dataIndex: 'handleType', - key: 'handleType', - search: { - type: 'select', - options: [ - { - label: '人工', - value: 'user', - }, - { - label: '系统', - value: 'system', - }, - ], - }, - scopedSlots: true, - }, - { - title: '状态', - dataIndex: 'state', - key: 'state', - search: { - type: 'select', - options: [ - { - label: '已处理', - value: 'normal', - }, - { - label: '告警中', - value: 'warning', - }, - ], - }, - scopedSlots: true, - }, - { - title: '操作', - dataIndex: 'actions', - key: 'actions', - scopedSlots: true, - }, - ] - : [ - { - title: '告警时间', - dataIndex: 'alarmTime', - key: 'alarmTime', - search: { - type: 'date', - }, - scopedSlots: true, - }, - { - title: '告警持续时长', - dataIndex: 'duration', - key: 'duration', - scopedSlots: true, - }, - { - title: '触发条件', - dataIndex: 'triggerDesc', - key: 'triggerDesc', - }, - { - title: '告警源', - dataIndex: 'sourceId', - key: 'sourceId', - scopedSlots: true, - search: { - type: 'string', - }, - }, - { - title: '告警原因', - dataIndex: 'actualDesc', - key: 'actualDesc', - }, - { - title: '处理时间', - dataIndex: 'handleTime', - key: 'handleTime', - search: { - type: 'date', - }, - scopedSlots: true, - }, - - { - title: '处理类型', - dataIndex: 'handleType', - key: 'handleType', - search: { - type: 'select', - options: [ - { - label: '人工', - value: 'user', - }, - { - label: '系统', - value: 'system', - }, - ], - }, - scopedSlots: true, - }, - { - title: '状态', - dataIndex: 'state', - key: 'state', - search: { - type: 'select', - options: [ - { - label: '已处理', - value: 'normal', - }, - { - label: '告警中', - value: 'warning', - }, - ], - }, - scopedSlots: true, - }, - { - title: '操作', - dataIndex: 'actions', - key: 'actions', - scopedSlots: true, - }, - ]; -const params = ref(); -const handleDescription = ref(); -const deviceAlarm = ref(); -const solveVisible = ref(false); -const solveType = ref('handle'); -const currentAlarm = ref(); -const visibleDrawer = ref(false); -const defaultParams = - props.goal === 'device' - ? { - sorts: [{ name: 'alarmTime', order: 'desc' }], - terms: [ - { - terms: [ - { - column: 'targetId', - value: current.id, - termType: 'eq', - }, - { - column: 'alarmConfigSource', - value: 'device-property-preprocessor', - termType: 'eq', - }, - ], - type: 'and', - }, - ], - } - : { - sorts: [{ name: 'alarmTime', order: 'desc' }], - terms: [ - { - terms: [ - { - column: 'targetId$dev-instance', - value: [ - { - column: 'product_id', - value: current.id, - termType: 'eq', - }, - ], - }, - { - column: 'alarmConfigSource', - value: 'device-property-preprocessor', - termType: 'eq', - }, - ], - type: 'and', - }, - ], - }; -const handleSearch = (e) => { - params.value = e; -}; -const queryHandle = async (id) => { - const res = await queryHandleHistory({ - sorts: [{ name: 'createTime', order: 'desc' }], - terms: [ - { - column: 'alarmRecordId', - termType: 'eq', - value: id, - type: 'and', - }, - ], - }); - if (res.status === 200 && res.result?.data.length) { - handleDescription.value = res.result.data?.[0]?.description; - } -}; -const getActions = (data) => { - if (!data) return []; - const actions = - data.state.value === 'normal' - ? [ - { - key: 'view', - text: '查看详情', - tooltip: { - title: '查看详情', - }, - onClick: async () => { - solveType.value = 'view'; - await queryHandle(data.id); - solveVisible.value = true; - }, - }, - { - key: 'log', - text: '查看日志', - tooltip: { - title: '查看日志', - }, - onClick: () => { - visibleDrawer.value = true; - currentAlarm.value = data; - }, - }, - ] - : [ - { - key: 'solve', - text: '处理', - tooltip: { - title: '处理', - }, - onClick: () => { - solveVisible.value = true; - solveType.value = 'handle'; - currentAlarm.value = data; - }, - }, - { - key: 'log', - text: '查看日志', - tooltip: { - title: '查看日志', - }, - onClick: () => { - visibleDrawer.value = true; - currentAlarm.value = data; - }, - }, - ]; - return actions; -}; -const closeSolve = () => { - solveVisible.value = false; -}; - -const refreshCurrent = async () => { - const res = await queryAlarmRecord({ - terms: [ - { - column: 'id', - termType: 'eq', - value: currentAlarm.value.id, - }, - { - column: 'alarmConfigSource', - value: 'device-property-preprocessor', - termType: 'eq', - }, - ], - }); - if (res.success && res.result?.data?.length) { - currentAlarm.value = res.result.data[0]; - } -}; - -const gotoDevice = (id) => { - menuStory.jumpPage('device/Instance/Detail', { id, tab: 'Running' }); -}; -const refresh = () => { - deviceAlarm.value?.reload(); - refreshCurrent(); -}; -const solveRefresh = () => { - solveVisible.value = false; - refresh(); -}; -</script> -<style lang="less" scoped> -.deviceId { - cursor: pointer; - color: #4096ff; -} -</style> diff --git a/src/views/device/Instance/Detail/AlarmRecord/Invalid/index.vue b/src/views/device/Instance/Detail/AlarmRecord/Invalid/index.vue deleted file mode 100644 index 6d28d28..0000000 --- a/src/views/device/Instance/Detail/AlarmRecord/Invalid/index.vue +++ /dev/null @@ -1,128 +0,0 @@ -<template> - <pro-search - :columns="columns" - target="device-instance" - @search="handleSearch" - ></pro-search> - <JProTable - ref="deviceAlarm" - :columns="columns" - model="TABLE" - :request="queryInvalidData" - :defaultParams="{ - sorts: [{ name: 'createTime', order: 'desc' }], - terms: [ - { - terms: [ - { - column: - props.goal === 'device' - ? 'thingId' - : 'templateId', - value: current.id, - termType: 'eq', - }, - ], - type: 'and', - }, - ], - }" - :params="params" - ><template #createTime="slotProps"> - {{ dayjs(slotProps.createTime).format('YYYY-MM-DD HH:mm:ss') }} - </template> - <template #thingId="slotProps"> - <Ellipsis> - 设备ID: - <span - class="deviceId" - >{{ slotProps.thingId }}</span - ></Ellipsis - > - </template> - </JProTable> -</template> - -<script setup> -import { queryInvalidData } from '@/api/rule-engine/log'; -import { useInstanceStore } from '@/store/instance'; -import { useProductStore } from '@/store/product'; -import dayjs from 'dayjs'; -const props = defineProps({ - goal: { - type: String, - default: 'device', - }, -}); -const { current } = - props.goal === 'device' ? useInstanceStore() : useProductStore(); -const columns = props.goal === 'device' ? [ - { - title: '上报时间', - dataIndex: 'createTime', - key: 'createTime', - scopedSlots: true, - search: { - type: 'date', - }, - scopedSlots: true, - }, - { - title: '阈值限制', - dataIndex: 'description', - key: 'description', - }, - { - title: '原始值', - dataIndex: 'value', - key: 'value', - search: { - type: 'string', - }, - }, -] : [ - { - title: '上报时间', - dataIndex: 'createTime', - key: 'createTime', - scopedSlots: true, - search: { - type: 'date', - }, - scopedSlots: true, - }, - { - title: '告警源', - dataIndex: 'thingId', - key: 'thingId', - scopedSlots: true, - search: { - type: 'string', - }, - }, - { - title: '阈值限制', - dataIndex: 'description', - key: 'description', - }, - { - title: '原始值', - dataIndex: 'value', - key: 'value', - search: { - type: 'string', - }, - }, -] -const handleSearch = (e) => { - params.value = e; -}; -const params = ref(); -</script> -<style lang="less" scoped> -.deviceId { - cursor: pointer; - color:#4096FF; -} -</style> - diff --git a/src/views/device/Instance/Detail/AlarmRecord/index.vue b/src/views/device/Instance/Detail/AlarmRecord/index.vue deleted file mode 100644 index 32b8143..0000000 --- a/src/views/device/Instance/Detail/AlarmRecord/index.vue +++ /dev/null @@ -1,33 +0,0 @@ -<template> - <div> - <a-tabs - tab-position="left" - v-model:activeKey="activeKey" - :tabBarStyle="{ width: '200px' }" - > - <a-tab-pane key="alarm" tab="告警数据"> - <Alarm :goal="type" /> - </a-tab-pane> - <a-tab-pane key="invalid" tab="无效数据"> - <Invalid :goal="type" /> - </a-tab-pane> - </a-tabs> - </div> -</template> - -<script setup> -import Alarm from './Alarm/index.vue'; -import Invalid from './Invalid/index.vue'; -const props = defineProps({ - type: { - type: String, - default: 'device', - }, -}); -const activeKey = ref('alarm'); -</script> -<style lang="less" scoped> -:deep(.ant-tabs-tab-active){ - background-color: #F1F7FF; -} -</style> diff --git a/src/views/device/Instance/Detail/Firmware/index.vue b/src/views/device/Instance/Detail/Firmware/index.vue deleted file mode 100644 index a35dfca..0000000 --- a/src/views/device/Instance/Detail/Firmware/index.vue +++ /dev/null @@ -1,193 +0,0 @@ -<template> - <div class="firmwareContainer" v-if="firmwareList?.length"> - <div v-for="i in firmwareList" class="firmwareBox"> - <CardBox - :showStatus="false" - :value="i" - @click="() => showTask(i.id)" - v-bind="i" - > - <template #content> - <div class="cardContent"> - <div class="firmwareHead"> - <Ellipsis style="width: 200px"> - {{ i.name }} - </Ellipsis> - </div> - <div class="firmwareDes"> - <Ellipsis style="width: 200px">{{ - i.description - }}</Ellipsis> - </div> - <div class="firmwareFoot"> - <div> - <span class="firmwareFootTitle">签名方式:</span - >{{ i.signMethod }} - </div> - <div> - <span class="firmwareFootTitle">创建时间:</span - >{{ - dayjs(i.createTime).format( - 'YYYY-MM-DD HH:mm:ss', - ) - }} - </div> - </div> - <div class="version">V {{ i.version }}</div> - </div> - <!-- <div class="mask">查看升级任务</div> --> - </template> - </CardBox> - </div> - </div> - <JEmpty v-else></JEmpty> - <Task - v-if="visibleTask" - :firmwareId="firmwareId" - showPosition="detail" - :deviceId="props.type === 'device' ? current.id : ''" - @close-drawer="visibleTask = false" - /> -</template> - -<script setup> -import { useInstanceStore } from '@/store/instance'; -import { useProductStore } from '@/store/product'; -import { queryPaginateNot, historyPaginateNot } from '@/api/device/firmware'; -import Task from '@/views/device/Firmware/Task/index.vue'; -import dayjs from 'dayjs'; -const props = defineProps({ - type: { - type: String, - default: 'device', - }, -}); -const { current } = - props.type === 'device' ? useInstanceStore() : useProductStore(); -const firmwareList = ref(); -const firmwareId = ref(); -const visibleTask = ref(false); -const queryFirmwareList = async () => { - const res = await queryPaginateNot({ - paging: false, - sorts: [ - { - name: 'createTime', - order: 'desc', - }, - ], - terms: [ - { - terms: [ - { - type: 'or', - value: - props.type === 'device' - ? current.productId - : current.id, - termType: 'eq', - column: 'productId', - }, - ], - }, - ], - }); - if (res.success) { - firmwareList.value = res.result; - } - if (props.type === 'device') { - // 查看固件所属产品下所有的任务 过滤掉不包含该设备的升级任务的固件 - const resp = await historyPaginateNot({ - paging: false, - sorts: [{ name: 'createTime', order: 'desc' }], - terms: [ - { - terms: [ - { - column: 'productId', - value: current.productId, - }, - ], - }, - ], - }); - if (resp.success) { - firmwareList.value = firmwareList.value.filter((i) => { - return resp.result.find((item) => { - return ( - i.id === item.firmwareId && current.id === item.deviceId - ); - }); - }); - } - } -}; -const showTask = (id) => { - firmwareId.value = id; - visibleTask.value = true; -}; -onMounted(() => { - queryFirmwareList(); -}); -</script> -<style lang="less" scoped> -.firmwareContainer { - display: flex; - flex-wrap: wrap; - gap: 8px; - .firmwareBox { - width: 33%; - position: relative; - margin-bottom: 20px; - .mask { - position: absolute; - top: 0; - left: 0; - z-index: 999; - display: flex; - justify-content: center; - align-items: center; - width: 100%; - font-size: 16px; - height: 100%; - color: transparent; - padding-top: 10px; - background-color: rgba(#000, 0); - cursor: pointer; - transition: background-color 0.3s; - - &:hover { - background-color: rgba(#000, 0.4); - color: #fff; - } - } - } - .cardContent { - position: relative; - .version { - position: absolute; - top: -10px; - right: 5px; - } - } - .firmwareHead, - .firmwareFoot { - display: flex; - } - .firmwareHead { - font-size: 16px; - font-weight: 600; - } - .firmwareDes { - height: 22px; - color: #777777; - margin: 8px 0; - } - .firmwareFoot { - justify-content: space-between; - .firmwareFootTitle{ - color:#A3A3A3 - } - } -} -</style> diff --git a/src/views/device/components/Metadata/Base/DetailModal/utils.ts b/src/views/device/components/Metadata/Base/DetailModal/utils.ts deleted file mode 100644 index 9a2e6a3..0000000 --- a/src/views/device/components/Metadata/Base/DetailModal/utils.ts +++ /dev/null @@ -1,25 +0,0 @@ -export const objectColumns = [ - { - title: '参数标识', - dataIndex: 'id', - }, - { - title: '参数名称', - dataIndex: 'name', - }, - { - title: '参数类型', - dataIndex: 'valueType', - }, -] - -export const enumColumns = [ - { - title: 'Value', - dataIndex: 'value', - }, - { - title: 'Text', - dataIndex: 'text', - }, -] diff --git a/src/views/device/components/Metadata/Base/components/Import/Import.vue b/src/views/device/components/Metadata/Base/components/Import/Import.vue deleted file mode 100644 index e500575..0000000 --- a/src/views/device/components/Metadata/Base/components/Import/Import.vue +++ /dev/null @@ -1,230 +0,0 @@ -<template> - <a-button type="primary" ghost @click="showImport"> - <template #icon> - <AIcon type="DownloadOutlined"/> - </template> - 导入 - </a-button> - <a-modal - v-if="visible" - visible - :title="step === 1 ? '批量导入' : '导入结果'" - :width="600" - :keyboard="false" - :maskClosable="false" - :okButtonProps="{ - disabled: step !== 3 - }" - :getContainer="modalContainer" - @cancel="onCancel" - @ok="onOk" - > - <a-form v-if="step === 1" layout="vertical"> - <a-form-item label="上传文件"> - <a-upload-dragger - name="file" - :headers="{[TOKEN_KEY]: LocalStore.get(TOKEN_KEY)}" - :maxCount="1" - :showUploadList="false" - :accept="'.xlsx,.csv'" - :before-upload="beforeUpload" - @change="uploadChange" - @drop="handleDrop" - > - <div class="dragger-box"> - <AIcon class="icon" type="PlusCircleFilled"/> - <span style="margin: 16px 0 8px 0">点击或拖拽上传文件</span> - <span>格式:.xlsx, .csv</span> - </div> - </a-upload-dragger> - </a-form-item> - <a-form-item label="下载模版"> - <div class="file-download"> - <j-button class="btn" @click="downFile('xlsx')">模板格式.xlsx</j-button> - <j-button class="btn" @click="downFile('csv')">模板格式.csv</j-button> - </div> - </a-form-item> - </a-form> - <div v-else-if="step === 2"> - <span> - 正在导入,请稍后... - </span> - </div> - <div v-else> - <div> - 导入成功: {{ successCount }}条 - </div> - <div> - 导入失败: {{ errorCount }}条 - </div> - </div> - </a-modal> -</template> - -<script setup name="MetadataImport"> -import {TOKEN_KEY, TOKEN_KEY_URL} from '@/utils/variable'; -import {getToken, isFullScreen, LocalStore, onlyMessage} from '@/utils/comm'; -import {validate} from './util' -import { getTemplate, uploadAnalyzeMetadata} from '@/api/device/instance' -import {getTemplate as getProductTemplate} from '@/api/device/product' -import {downloadFileByUrl} from "@/utils/utils"; -import {useGroupActive, useTableWrapper} from "@/components/Metadata/Table/context"; - -const props = defineProps({ - target: { - type: String, - default: undefined - }, - metadata: { - type: Array, - default: () => [] - } -}) - -const emit = defineEmits(['ok']) - -const visible = ref(false); -const successCount = ref(0); -const errorCount = ref(0); -const step = ref(1) -const groupActive = useGroupActive() -const tableWrapperRef = useTableWrapper() - -let submitMetadata = [] - -const route = useRoute() - -const modalContainer = (e) => { - if (isFullScreen()) { - return tableWrapperRef.value || document.body - } - return document.body -} - -const init = () => { - step.value = 1 - successCount.value = 0 - errorCount.value = 0 -} - -const showImport = () => { - init() - visible.value = true -} - -const onCancel = () => { - visible.value = false -} - -const submitData = async (metadataStr) => { - console.log(metadataStr) - if (metadataStr) { - const _metadataObject = JSON.parse(metadataStr || "{}") - const properties = _metadataObject.properties - const _metadata = props.metadata.filter(item => item.id && !item.expands?.isProduct) - const result = properties.map(item => { - if (item.expands) { - item.expands.groupId = groupActive.value - item.expands.groupName = groupActive.label - } else { - item.expands = { - groupId: groupActive.value, - groupName: groupActive.label, - } - } - return item - }) - - submitMetadata = validate(result, _metadata, (validate) => { - if (validate) { - successCount.value += 1 - } else { - errorCount.value += 1 - } - }) - step.value = 3 - } else { - onlyMessage('请先上传文件', 'error'); - } -}; - -const onOk = () => { - emit('ok', submitMetadata) - visible.value = false -} - -const downFile = (type) => { - const url = props.target === 'device' ? getTemplate(route.params.id, type) : getProductTemplate(route.params.id, type) - - downloadFileByUrl(url + `?${TOKEN_KEY_URL}=${getToken()}`, '物模型模版', type) -} - -const handleDrop = () => { -}; - -const beforeUpload = (file) => { - const isCsv = file.type === 'text/csv'; - const isXlsx = - file.type === - 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'; - if (!isCsv && !isXlsx) { - onlyMessage('请上传.xlsx或.csv格式文件', 'warning'); - } - - const formData = new FormData() - - formData.append('file', file) - - step.value = 2 - uploadAnalyzeMetadata(formData).then(res => { - if (res.success) { - submitData(res.result) - } - }) - - // return isCsv || isXlsx; - - return false -}; - - -const uploadChange = async (info) => { - console.log(info) - if (info.file.status === 'uploading') { - step.value = 2 - } - if (info.file.status === 'done') { - const resp = info.file.response || {result: {}}; - await submitData(resp?.result || ''); - } - if (info.file.status === 'error') { - onlyMessage('上传失败', 'error'); - } -}; -</script> - -<style scoped lang="less"> -.dragger-box { - margin: 46px 0; - display: flex; - flex-direction: column; - color: #666666; - - .icon { - font-size: 30px; - color: @primary-color; - } -} - -.file-download { - display: flex; - gap: 16px; - - .btn { - border: none; - background-color: #ECECF0; - width: 152px; - color: #666666; - } -} -</style> diff --git a/src/views/device/components/Metadata/Base/components/Import/index.ts b/src/views/device/components/Metadata/Base/components/Import/index.ts deleted file mode 100644 index d7caadc..0000000 --- a/src/views/device/components/Metadata/Base/components/Import/index.ts +++ /dev/null @@ -1 +0,0 @@ -export { default as Import } from './Import.vue' diff --git a/src/views/device/components/Metadata/Base/components/Import/util.ts b/src/views/device/components/Metadata/Base/components/Import/util.ts deleted file mode 100644 index ea017c4..0000000 --- a/src/views/device/components/Metadata/Base/components/Import/util.ts +++ /dev/null @@ -1,110 +0,0 @@ - -export const validateDate = (item: Record<string, any>) => { - if (!item.valueType.format) { - console.log('validateDate - error', item.valueType.format) - return false - } - return true -} - -export const validateEnum = (item: Record<string, any>) => { - if (item.valueType.elements?.length) { - if (item.valueType.elements.every((item: {value: string, text: string}) => item.value && item.text)) { - return true - } - console.log('validateEnum - error', item.valueType) - return false - } - console.log('validateEnum - error', item.valueType) - return false -} - -export const validateFile = (item: Record<string, any>) => { - if (!!item.valueType.bodyType) { - return true - } - console.log('validateFile - error', item.valueType.bodyType) - return false -} - -export const validateArray = (item: Record<string, any>): boolean => { - const { valueType } = item - - if (!valueType.elementType?.type) { - console.log('validateArray - error', valueType.elementType) - return false - } - - return handleValidate(valueType.elementType.type, valueType.elementType) -} - -export const validateObject = (item: Record<string, any>): boolean => { - const { valueType } = item - return valueType.properties?.length ? valueType.properties.every(item => { - - if(!item.id || !item.name || !item.valueType.type) { - console.log('validateArray - error', item) - return false - } - - return handleValidate(item.valueType.type, item) - - }) : false -} - -export const validateMap = { - enum: validateEnum, - date: validateDate, - file: validateFile, - array: validateArray, - object: validateObject, -} - -export const handleValidate = (type: string, item: any) => { - const validateKeys = Object.keys(validateMap) - - if (validateKeys.includes(type)) { - return validateMap[type](item) - } - - return true -} - -export const validateItem = (item: Record<string, any>): boolean => { - - if (!item || !item.id || !item.name || !item.valueType?.type || !item.expands?.source || !item.expands?.type?.length) { - console.log('validateItem - error', item) - return false - } - console.log(item.valueType.type) - - return handleValidate(item.valueType.type, item) - -} -export const validate = (importMetadata: Array<Record<string, any>>, metadata: Array<Record<string, any>>, validateCallBack: (validate: boolean) => void):Array<Record<string, any>> => { - const metadataMap = new Map() - const copyMetadata = [] - - metadata.forEach((item, index) => { - metadataMap.set(item.id, index) - }) - - for (let i = importMetadata.length - 1; i >= 0; i--) { - const item = importMetadata[i] - const isValidate = validateItem(item) - - if (isValidate) { - // 与设备物模型属性id一致:覆盖 - if (metadataMap.has(item.id)) { - metadata.splice(metadataMap.get(item.id), 1, item) - importMetadata.splice(i, 1) - } else { - copyMetadata.push(item) - } - - } - validateCallBack?.(isValidate) - } - - return [...metadata, ...copyMetadata] -} diff --git a/src/views/device/components/Metadata/Base/components/Properties/hooks.ts b/src/views/device/components/Metadata/Base/components/Properties/hooks.ts deleted file mode 100644 index 3f1273e..0000000 --- a/src/views/device/components/Metadata/Base/components/Properties/hooks.ts +++ /dev/null @@ -1,106 +0,0 @@ -import { updateProductThreshold, updateDeviceThreshold ,queryDeviceThreshold, queryProductThreshold , deleteProductThreshold, deleteDeviceThreshold } from '@/api/device/instance' -import { useRequest } from '@/hook' -import {useProductStore} from "store/product"; -import {useInstanceStore} from "store/instance"; -export const useThreshold = (props: Record<string, any>) => { - - const productStore = useProductStore(); - const deviceStore = useInstanceStore(); - - const thresholdDetail = ref({ - - }) - - const { run: updateProductRun } = useRequest(updateProductThreshold, { immediate: false }) - - const { run: updateDeviceRun } = useRequest(updateDeviceThreshold,{ immediate: false}) - - const { run: deleteProductRun } = useRequest(deleteProductThreshold,{ immediate: false}) - - const { run: deleteDeviceRun } = useRequest(deleteDeviceThreshold,{ immediate: false}) - - const { run: queryDevice } = useRequest(queryDeviceThreshold, { - immediate: false, - onSuccess(res) { - handleDetail(res.result) - } - }) - - const { run: queryProduct } = useRequest(queryProductThreshold, { - immediate: false, - onSuccess(res) { - handleDetail(res.result) - } - }) - - - const handleDetail = (data: Record<string, any>) => { - thresholdDetail.value = { - type: data.configuration.matcher.provider, - limit:{ - lower: data.configuration.matcher.configuration.min, - upper: data.configuration.matcher.configuration.max - }, - // mode: data.configuration.processors.map((i:any)=>{ - // return i.provider - // }) - mode: data.configuration.processors[0].provider - } - } - - const thresholdUpdate = (data: Record<string, any>) => { - const params = { - thingType: 'device', - provider: 'simple', - configuration:{ - matcher:{ - provider: data.type, - configuration:{ - max: data.limit.upper, - min: data.limit.lower, - not: true - } - }, - processors: [{ - provider: data.mode, - configuration:{} - }] - // data.mode.map((i:any)=>{ - // return { - // provider: i, - // configuration:{} - // } - // }) - }, - - } - if(props.target === 'product'){ - updateProductRun(productStore.current.id,props.id,params) - }else{ - updateDeviceRun(deviceStore.current.productId, deviceStore.current.id, props.id,params) - } - } - - const thresholdDelete = ()=>{ - if(props.target === 'product'){ - deleteProductRun(productStore.current.id,props.id) - }else{ - deleteDeviceRun(deviceStore.current.productId,deviceStore.current.id,props.id) - } - } - const thresholdDetailQuery = () => { - if (props.target === 'product') { - queryProduct(productStore.current.id, props.id) - } else { - queryDevice(deviceStore.current.productId, deviceStore.current.id, props.id) - } - } - - - return { - thresholdUpdate, - thresholdDelete, - thresholdDetailQuery, - thresholdDetail - } -} diff --git a/src/views/device/components/Metadata/Base/components/VirtualRule/DetailModal.vue b/src/views/device/components/Metadata/Base/components/VirtualRule/DetailModal.vue deleted file mode 100644 index 569170b..0000000 --- a/src/views/device/components/Metadata/Base/components/VirtualRule/DetailModal.vue +++ /dev/null @@ -1,52 +0,0 @@ -<template> - <a-modal - visible - title="规则" - :maskClosable="false" - :width="1000" - :getContainer="getPopupContainer" - :footer="false" - @cancel="emit('cancel')" - > - <div class="rule-detail-warp"> - <j-monaco-editor - :model-value="value" theme="vs" ref="editor" language="javascript" :readOnly="true" - /> - </div> - </a-modal> -</template> - -<script setup name="DetailModal"> -defineProps({ - value: { - type: String, - default: undefined - }, - getPopupContainer: { - type: Function, - default: undefined - }, -}) - -const emit = defineEmits(['cancel']) - -</script> - -<style scoped lang="less"> - -.rule-detail-warp { - position: relative; - height: 500px; - - :deep(.monaco-editor-overlaymessage) { - display: none !important; - } - .rule-detail-mask { - position: absolute; - top: 0; - bottom: 0; - left: 0; - right: 0; - } -} -</style> diff --git a/src/views/device/components/Metadata/Base/hooks/useGroup.ts b/src/views/device/components/Metadata/Base/hooks/useGroup.ts deleted file mode 100644 index 79341ce..0000000 --- a/src/views/device/components/Metadata/Base/hooks/useGroup.ts +++ /dev/null @@ -1,36 +0,0 @@ -import {provide} from "vue"; - -type GroupOptionsItemType = { label: string, value: string} -import { METADATA_GROUP_OPTIONS } from '@/components/Metadata/Table/consts' -export const useGroup = () => { - - const options = ref<GroupOptionsItemType[]>([]) - - const addOptions = (op: GroupOptionsItemType) => { - if (!options.value.some(item => op.label === item.label)) { - options.value.push(op) - } - } - - const cleanOptions = () => { - options.value = [] - } - - const initOptions = (dataSource: any[]) => { - cleanOptions() - options.value = dataSource.filter(item => item.id && item.extends?.group).map(item => ({ - label: item.group, - value: item.group - })) - } - - provide(METADATA_GROUP_OPTIONS, { - options, - addOptions - }) - - return { - cleanOptions, - initOptions - } -} diff --git a/src/views/gis/AnalyzeDiagnosis/History/Detail/EventStatistics.vue b/src/views/gis/AnalyzeDiagnosis/History/Detail/EventStatistics.vue index 72d33d0..e3025fe 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/Detail/EventStatistics.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/Detail/EventStatistics.vue @@ -1,6 +1,6 @@ <template> <div> - 事件统计 + <img src="./Snipaste_2024-10-08_10-10-25.png" alt=""> </div> </template> diff --git a/src/views/gis/AnalyzeDiagnosis/History/Detail/Prpd.vue b/src/views/gis/AnalyzeDiagnosis/History/Detail/Prpd.vue index f4397ee..8aeb82d 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/Detail/Prpd.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/Detail/Prpd.vue @@ -3,7 +3,11 @@ <a-row> <a-col v-for="(item,index) in 6" :key="index" :xs="24" :sm="24" :md="24" :lg="12" :xl="8"> <j-card :bordered="false"> - <j-select show-search v-model:value="monitorIds[index]" :options="monitorList" size="small" placeholder="请选择" style="width: 100%;"></j-select> + <j-space> + <j-button type="text" size="small">实时图谱</j-button> + <j-button type="text" size="small">报告导出</j-button> + </j-space> + <ThreeChart :chartId="'box' + index" :type="type === 'Prps'?1:2" :height="255" /> </j-card> </a-col> diff --git a/src/views/gis/AnalyzeDiagnosis/History/Detail/PrpdTotal.vue b/src/views/gis/AnalyzeDiagnosis/History/Detail/PrpdTotal.vue index 978083e..ef95ba0 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/Detail/PrpdTotal.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/Detail/PrpdTotal.vue @@ -1,6 +1,6 @@ <template> <div> - prpd统计 + <img src="./Snipaste_2024-10-08_10-10-25.png" alt=""> </div> </template> diff --git a/src/views/gis/AnalyzeDiagnosis/History/Detail/Snipaste_2024-10-08_10-10-25.png b/src/views/gis/AnalyzeDiagnosis/History/Detail/Snipaste_2024-10-08_10-10-25.png new file mode 100644 index 0000000000000000000000000000000000000000..dc26bb62a0d7357750166c84a56e75943eff2c15 GIT binary patch literal 42615 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 literal 0 HcmV?d00001 diff --git a/src/views/gis/AnalyzeDiagnosis/History/Detail/index.vue b/src/views/gis/AnalyzeDiagnosis/History/Detail/index.vue index bcfb827..6fc4bbf 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/Detail/index.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/Detail/index.vue @@ -1,55 +1,68 @@ <template> <page-container> - <a-row> - <a-col :xs='24' :sm='24' :md='24' :lg='24' :xl='24'> + <Splitpanes ref="split" class="default-theme" @splitter-click="splitterClick" + :style="{ height: bodyHeight + 'px', overflow: true, minHeight: '720px', width: '100%' }"> + <Pane ref="paneLeft" :size="treeMinSize" :min-size="treeMinSize" v-show="treeShow"> + <LeftMonitor /> + </Pane> + <pane min-size="80" class="realTime" ref="paneRight" :size="100 - treeMinSize" + :style="{ minWidth: '900px' }"> <j-card> - <TopChart /> + <a-row> + <a-col :xs='24' :sm='24' :md='24' :lg='24' :xl='24'> + + <TopChart /> + + </a-col> + </a-row> + <a-row> + <!-- <a-card :style="{ width: '150px', }"> --> + <a-list size="small" :data-source="data" :style="{ width: '150px', }"> + <template #renderItem="{ item, index }"> + <a-list-item> + <a-button @click='changeTime(index)' + :type="index == activeIndex ? 'link' : 'text'">{{ item + }}</a-button> + </a-list-item> + </template> + <template #header> + <div :style="{ textAlign: 'center' }">趋势时间</div> + </template> + </a-list> + <!-- </a-card> --> + + <a-row :style="{ width: 'calc(100% - 320px)' }"> + <a-col :xs='24' :sm='24' :md='24' :lg='24' :xl='24'> + + <j-tabs v-model:activeKey="activeKey"> + <j-tab-pane v-for="item in list" :key="item.key" :tab="item.title" /> + </j-tabs> + <keep-alive> + <component :type='activeKey' :monitorIds="monitorIds" :monitorList="monitorList" + :is="tabs[activeKey]" /> + </keep-alive> + + </a-col> + </a-row> + <!-- <a-card :style="{ width: '150px', }"> --> + <a-list :style="{ width: '150px', }" size="small" :data-source="data"> + <template #renderItem="{ item, index }"> + <a-list-item> + <a-button @click='changeTime(index)' + :type="index == activeIndex ? 'link' : 'text'">{{ item + }}</a-button> + </a-list-item> + </template> + <template #header> + <div :style="{ textAlign: 'center' }">事件时间</div> + </template> + </a-list> + <!-- </a-card> --> + </a-row> </j-card> - </a-col> - - </a-row> - <a-row :style="{ marginTop: '10px' }"> - <a-card :style="{ width: '150px', marginRight: '10px' }"> - <a-list size="small" :data-source="data"> - <template #renderItem="{ item, index }"> - <a-list-item> - <a-button @click='changeTime(index)' :type="index == activeIndex ? 'link' : 'text'">{{ item - }}</a-button> - </a-list-item> - </template> - <template #header> - <div :style="{ textAlign: 'center' }">趋势时间</div> - </template> - </a-list> - </a-card> - - <a-row :style="{ width: 'calc(100% - 320px)' }"> - <a-col :xs='24' :sm='24' :md='24' :lg='24' :xl='24'> - <j-card> - <j-tabs v-model:activeKey="activeKey"> - <j-tab-pane v-for="item in list" :key="item.key" :tab="item.title" /> - </j-tabs> - <keep-alive> - <component :type='activeKey' :monitorIds="monitorIds" :monitorList="monitorList" - :is="tabs[activeKey]" /> - </keep-alive> - </j-card> - </a-col> - </a-row> - <a-card :style="{ width: '150px', marginLeft: '10px' }"> - <a-list size="small" :data-source="data"> - <template #renderItem="{ item, index }"> - <a-list-item> - <a-button @click='changeTime(index)' :type="index == activeIndex ? 'link' : 'text'">{{ item - }}</a-button> - </a-list-item> - </template> - <template #header> - <div :style="{ textAlign: 'center' }">事件时间</div> - </template> - </a-list> - </a-card> - </a-row> + </pane> + </Splitpanes> + </page-container> </template> @@ -68,8 +81,13 @@ import Prps from './Prps.vue'; import EventStatistics from './EventStatistics.vue'; import EventList from './EventList.vue'; import { timeData } from '../data' +import { Splitpanes, Pane } from 'splitpanes' +import LeftMonitor from '@/views/gis/components/LeftMonitor/index.vue' +import 'splitpanes/dist/splitpanes.css' - +const treeShow = ref<boolean>(false); +const treeMinSize = ref<number>(0); +const bodyHeight = ref<number>(document.body.clientHeight - 120); const monitorIds = [1, 2, 3, 4, 5, 6]; const monitorList = ref<{ label: string; value: number; }[]>([ @@ -82,6 +100,14 @@ const monitorList = ref<{ label: string; value: number; }[]>([ ]) +const splitterClick = () => { + treeShow.value = !treeShow.value; + treeMinSize.value = treeMinSize.value == 0 ? 20 : 0; + const event = new Event('resize'); + // 触发窗口的resize事件 + window.dispatchEvent(event); +} + type KeyType = 'Prpd' | 'PrpdTotal' | 'Prps' | 'EventStatistics' | 'EventList'; const activeKey = ref<KeyType>('Prpd'); @@ -89,15 +115,15 @@ const activeKey = ref<KeyType>('Prpd'); const list: { key: KeyType; title: string }[] = [ { key: 'Prpd', - title: 'PRPD', + title: 'PRPD(1分钟)', }, { key: 'PrpdTotal', - title: 'PRPD(累计)', + title: 'PRPD(15分钟)', }, { key: 'Prps', - title: 'PRPS', + title: 'PRPS(1分钟)', }, { key: 'EventStatistics', @@ -200,9 +226,9 @@ onUnmounted(() => { </script> <style lang="less" scoped> -:deep(.ant-card-body) { - padding: 5px 24px; -} +// :deep(.ant-card-body) { +// padding: 5px 24px; +// } :deep(.ant-list-item) { padding: 0; diff --git a/src/views/gis/AnalyzeDiagnosis/History/components/Network.vue b/src/views/gis/AnalyzeDiagnosis/History/components/Network.vue index bc2be3d..e2e156a 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/components/Network.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/components/Network.vue @@ -1,8 +1,15 @@ <template> - <j-spin :spinning="loading"> - <div class="dash-board"> - <div class="header"> - <div class="left"> + <Splitpanes ref="split" class="default-theme" + @splitter-click="splitterClick" + :style="{ height: bodyHeight + 'px', overflow: true, minHeight: '720px', width: '100%' }"> + <Pane ref="paneLeft" :size="treeMinSize" :min-size="treeMinSize" v-show="treeShow"> + <LeftMonitor /> + </Pane> + <pane min-size="80" class="realTime" ref="paneRight" :size="100 - treeMinSize" :style="{ minWidth: '900px' }"> + <j-spin :spinning="loading"> + <div class="dash-board"> + <div class="header"> + <!-- <div class="left"> <j-tree-select showSearch v-model:value="monitorIds" placeholder="请选择监测点" :tree-data="treeList" @change="valueChange" allow-clear :max-tag-count="2" :height="233" multiple :style="{ width: '400px' }" :fieldNames="{ @@ -18,37 +25,41 @@ </j-tooltip> </a-tag> </template> - </j-tree-select> - </div> - <div class="right"> - <j-radio-group default-value="a" button-style="solid" style="margin-right: 10px" - v-model:value="data.time.type"> - <j-radio-button value="hour"> - 最近1小时 - </j-radio-button> - <j-radio-button value="day"> - 最近24小时 - </j-radio-button> - <j-radio-button value="week"> 近一周 </j-radio-button> - </j-radio-group> - <j-range-picker :allowClear="false" :show-time="{ format: 'HH:mm:ss' }" format="YYYY-MM-DD HH:mm:ss" - v-model:value="data.time.time" @change="pickerTimeChange"> - <template #suffixIcon> - <AIcon type="CalendarOutlined" /> +</j-tree-select> +</div> --> + <div class="right"> + <j-radio-group default-value="a" button-style="solid" style="margin-right: 10px" + v-model:value="data.time.type"> + <j-radio-button value="hour"> + 最近1小时 + </j-radio-button> + <j-radio-button value="day"> + 最近24小时 + </j-radio-button> + <j-radio-button value="week"> 近一周 </j-radio-button> + </j-radio-group> + <j-range-picker :allowClear="false" :show-time="{ format: 'HH:mm:ss' }" + format="YYYY-MM-DD HH:mm:ss" v-model:value="data.time.time" @change="pickerTimeChange"> + <template #suffixIcon> + <AIcon type="CalendarOutlined" /> + </template> + </j-range-picker> + </div> + </div> + <div> + <j-empty v-if="isEmpty" style="margin-top: 100px" /> + <template v-else> + <div style="height: calc(100vh - 232px)" + :style="{ minHeight: `${echartsMinHeight * 200}px` }"> + <Echarts :options="echartsOptions" /> + </div> </template> - </j-range-picker> - </div> - </div> - <div> - <j-empty v-if="isEmpty" style="margin-top: 100px" /> - <template v-else> - <div style="height: calc(100vh - 232px)" :style="{ minHeight: `${echartsMinHeight * 200}px` }"> - <Echarts :options="echartsOptions" /> </div> - </template> - </div> - </div> - </j-spin> + </div> + </j-spin> + </pane> + </Splitpanes> + </template> <script lang="ts" setup name="Network"> @@ -72,6 +83,13 @@ import { minHeight } from '../data.js'; import { chunk } from 'lodash-es'; import * as echarts from 'echarts'; import { timeData } from '../data.ts'; +import { Splitpanes, Pane } from 'splitpanes' +import LeftMonitor from '@/views/gis/components/LeftMonitor/index.vue' +import 'splitpanes/dist/splitpanes.css' + +const treeShow = ref<boolean>(false); +const treeMinSize = ref<number>(0); +const bodyHeight = ref<number>(document.body.clientHeight-120); const props = defineProps({ serviceId: { type: String, @@ -191,7 +209,13 @@ const treeList: Ref<TreeNode[]> = ref([ level: 1, }, ]); - +const splitterClick = () => { + treeShow.value = !treeShow.value; + treeMinSize.value = treeMinSize.value == 0 ? 20 : 0; + const event = new Event('resize'); + // 触发窗口的resize事件 + window.dispatchEvent(event); +} const valueChange = (value: string, label: string[]) => { monitorNames.value = label; }; @@ -259,7 +283,7 @@ const setOptions = (): any => { const initOptions: any = { title: { - show:false, + show: false, text: '监测点', //获取间隔名称 subtextStyle: { color: '#92A3A8' @@ -422,7 +446,7 @@ const setOptions = (): any => { name: '脉冲平均值', type: 'line', showSymbol: false, - method:()=>getRandomInt(-80,0), + method: () => getRandomInt(-80, 0), data: [], yAxisIndex: 2, xAxisIndex: 0, @@ -435,7 +459,7 @@ const setOptions = (): any => { name: '最大值', type: 'line', showSymbol: false, - method:()=>getRandomInt(-80,0), + method: () => getRandomInt(-80, 0), data: [], yAxisIndex: 2, xAxisIndex: 0, @@ -448,7 +472,7 @@ const setOptions = (): any => { name: '脉冲频次', type: 'line', showSymbol: false, - method:()=>getRandomInt(1000,3000), + method: () => getRandomInt(1000, 3000), data: [], yAxisIndex: 0, xAxisIndex: 0, @@ -473,8 +497,8 @@ const setOptions = (): any => { xAxisIndex: 'all', } ], - label:{ - formatter: ()=>{} + label: { + formatter: () => { } } }, } @@ -487,8 +511,8 @@ const setOptions = (): any => { initOptions['title'] = monitorNames.value.map((text, index) => ({ ...initOptions.title, text, - left:'10%', - top: gridOptions[index].top + + '%', + left: '10%', + top: gridOptions[index].top + + '%', })); initOptions['grid'] = monitorIds.value.map((item, index) => ({ left: '0%', @@ -503,7 +527,7 @@ const setOptions = (): any => { initOptions['xAxis'] = monitorIds.value.map((item, index) => ({ ...initOptions.xAxis, gridIndex: index, - })); + })); // initOptions['toolbox'] = monitorIds.value.map((item, index) => ({ // ...initOptions.toolbox, // top: gridOptions[index].top + + '%', @@ -512,11 +536,11 @@ const setOptions = (): any => { initOptions['legend'] = monitorIds.value.map((item, index) => ({ ...initOptions.legend, left: 'center', - top: gridOptions[index].top+ 5 + '%', + top: gridOptions[index].top + 5 + '%', width: '80%', gridIndex: index, })) - initOptions['axisPointer'].label.formatter = (param) => monitorNames.value[param.axisIndex] + '\n' + param.value + initOptions['axisPointer'].label.formatter = (param) => monitorNames.value[param.axisIndex] + '\n' + param.value const yAxis: any[] = [] const series: any[] = [] monitorIds.value.map((item, index) => { @@ -525,11 +549,11 @@ const setOptions = (): any => { }) initOptions['yAxis'] = yAxis monitorNames.value.map((text, index) => { - const arr = initOptions.series.map((i: any,id) => ({ + const arr = initOptions.series.map((i: any, id) => ({ ...i, xAxisIndex: index, - yAxisIndex: index*4 + i.yAxisIndex, - data: timeData.map((it: any[]) => ([it[0],i.method()]) ) + yAxisIndex: index * 4 + i.yAxisIndex, + data: timeData.map((it: any[]) => ([it[0], i.method()])) })) series.push(...arr) }) @@ -553,19 +577,19 @@ const setOptions = (): any => { }; const echartsOptions = computed(() => { - const { title, grid, xAxis, yAxis, series, legend,axisPointer,toolbox } = setOptions(); + const { title, grid, xAxis, yAxis, series, legend, axisPointer, toolbox } = setOptions(); return { title, backgroundColor: '#fff', - color: ['#FF7D00', '#FFC300', '#37E2E2', '#165DFF', '#722ED1','#313CA9', '#21CCFF', '#249EFF', '#86DF6C', '#979AFF','#246EFF', '#00B2FF', '#81E2FF', '#2CAB40', '#8D4EDA'], + color: ['#FF7D00', '#FFC300', '#37E2E2', '#165DFF', '#722ED1', '#313CA9', '#21CCFF', '#249EFF', '#86DF6C', '#979AFF', '#246EFF', '#00B2FF', '#81E2FF', '#2CAB40', '#8D4EDA'], tooltip: { trigger: 'axis', formatter: function (params) { // 自定义提示框的内容格式 let data = chunk(params, 3) // 生成表格格式的字符串 - let tableHeader = ['站点', "脉冲平均值","最大值","脉冲频次"]; - let tableContent = data.map(item => [ item[0].axisValueLabel.split('\n')[0],...item.map(i => i.data[1] )] ).join('<br>')//tableData.map(item => item.map(cell => cell.value)).join('<br>'); + let tableHeader = ['站点', "脉冲平均值", "最大值", "脉冲频次"]; + let tableContent = data.map(item => [item[0].axisValueLabel.split('\n')[0], ...item.map(i => i.data[1])]).join('<br>')//tableData.map(item => item.map(cell => cell.value)).join('<br>'); return ` <div id="table-container"> <h3>${params[0].name}</h3> @@ -582,40 +606,40 @@ const echartsOptions = computed(() => { </div>`; }, //trigger: 'item', - position:function (point, params, dom, rect, size) { - // 鼠标坐标和提示框位置的参考坐标系是:以外层div的左上角那一点为原点,x轴向右,y轴向下 - // 提示框位置 - var x = 0; // x坐标位置 - var y = 0; // y坐标位置 - - // 当前鼠标位置 - var pointX = point[0]; - var pointY = point[1]; - - // 外层div大小 - // var viewWidth = size.viewSize[0]; - // var viewHeight = size.viewSize[1]; - - // 提示框大小 - var boxWidth = size.contentSize[0]; - var boxHeight = size.contentSize[1]; - - // boxWidth > pointX 说明鼠标左边放不下提示框 - if (boxWidth > pointX) { - x = 5; - } else { // 左边放的下 - x = pointX - boxWidth; - } - - // boxHeight > pointY 说明鼠标上边放不下提示框 - if (boxHeight > pointY) { - y = 5; - } else { // 上边放得下 - y = pointY - boxHeight; - } - - return [x, y]; - }, + // position:function (point, params, dom, rect, size) { + // // 鼠标坐标和提示框位置的参考坐标系是:以外层div的左上角那一点为原点,x轴向右,y轴向下 + // // 提示框位置 + // var x = 0; // x坐标位置 + // var y = 0; // y坐标位置 + + // // 当前鼠标位置 + // var pointX = point[0]; + // var pointY = point[1]; + + // // 外层div大小 + // // var viewWidth = size.viewSize[0]; + // // var viewHeight = size.viewSize[1]; + + // // 提示框大小 + // var boxWidth = size.contentSize[0]; + // var boxHeight = size.contentSize[1]; + + // // boxWidth > pointX 说明鼠标左边放不下提示框 + // if (boxWidth > pointX) { + // x = 5; + // } else { // 左边放的下 + // x = pointX - boxWidth; + // } + + // // boxHeight > pointY 说明鼠标上边放不下提示框 + // if (boxHeight > pointY) { + // y = 5; + // } else { // 上边放得下 + // y = pointY - boxHeight; + // } + + // return [x, y]; + // }, axisPointer: { type: 'cross', animation: false, diff --git a/src/views/gis/AnalyzeDiagnosis/History/components/TopChart.vue b/src/views/gis/AnalyzeDiagnosis/History/components/TopChart.vue index 08b36c6..e300e8d 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/components/TopChart.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/components/TopChart.vue @@ -474,40 +474,40 @@ const echartsOptions = computed(() => { </div>`; }, //trigger: 'item', - position:function (point, params, dom, rect, size) { - // 鼠标坐标和提示框位置的参考坐标系是:以外层div的左上角那一点为原点,x轴向右,y轴向下 - // 提示框位置 - var x = 0; // x坐标位置 - var y = 0; // y坐标位置 + // position:function (point, params, dom, rect, size) { + // // 鼠标坐标和提示框位置的参考坐标系是:以外层div的左上角那一点为原点,x轴向右,y轴向下 + // // 提示框位置 + // var x = 0; // x坐标位置 + // var y = 0; // y坐标位置 - // 当前鼠标位置 - var pointX = point[0]; - var pointY = point[1]; + // // 当前鼠标位置 + // var pointX = point[0]; + // var pointY = point[1]; - // 外层div大小 - // var viewWidth = size.viewSize[0]; - // var viewHeight = size.viewSize[1]; + // // 外层div大小 + // // var viewWidth = size.viewSize[0]; + // // var viewHeight = size.viewSize[1]; - // 提示框大小 - var boxWidth = size.contentSize[0]; - var boxHeight = size.contentSize[1]; + // // 提示框大小 + // var boxWidth = size.contentSize[0]; + // var boxHeight = size.contentSize[1]; - // boxWidth > pointX 说明鼠标左边放不下提示框 - if (boxWidth > pointX) { - x = 5; - } else { // 左边放的下 - x = pointX - boxWidth; - } + // // boxWidth > pointX 说明鼠标左边放不下提示框 + // if (boxWidth > pointX) { + // x = 5; + // } else { // 左边放的下 + // x = pointX - boxWidth; + // } - // boxHeight > pointY 说明鼠标上边放不下提示框 - if (boxHeight > pointY) { - y = 5; - } else { // 上边放得下 - y = pointY - boxHeight; - } + // // boxHeight > pointY 说明鼠标上边放不下提示框 + // if (boxHeight > pointY) { + // y = 5; + // } else { // 上边放得下 + // y = pointY - boxHeight; + // } - return [x, y]; - }, + // return [x, y]; + // }, axisPointer: { type: 'cross', animation: false, diff --git a/src/views/gis/AnalyzeDiagnosis/History/index.vue b/src/views/gis/AnalyzeDiagnosis/History/index.vue index 268e6a1..fe4097b 100644 --- a/src/views/gis/AnalyzeDiagnosis/History/index.vue +++ b/src/views/gis/AnalyzeDiagnosis/History/index.vue @@ -1,10 +1,11 @@ <template> <page-container> - <div> + <j-row :gutter="[24, 24]"> <j-col :span="24" v-if="isNoCommunity"><Network :serviceId='serviceId' /></j-col> </j-row> - </div> + + </page-container> </template> @@ -13,7 +14,7 @@ import Network from './components/Network.vue'; import { isNoCommunity } from '@/utils/utils' -const serviceId = ref<string | undefined>() +const serviceId = ref<string | undefined>(); </script> diff --git a/src/views/gis/RealTime/Graph/components/LeftMonitor/index.vue b/src/views/gis/RealTime/Graph/components/LeftMonitor/index.vue new file mode 100644 index 0000000..dade451 --- /dev/null +++ b/src/views/gis/RealTime/Graph/components/LeftMonitor/index.vue @@ -0,0 +1,66 @@ +<template> + <j-card> + <div class="operator-box" :style="{ height: bodyHeight + 'px' }"> + <j-input-search @search="search" allow-clear placeholder="搜索关键字" /> + <div class="tree" :style="{ marginTop: '15px' }"> + <j-tree @select="selectTree" :field-names="{ title: 'nodeName', key: 'id', }" auto-expand-parent + checkable :tree-data="data" :showLine="{ showLeafIcon: false }" :show-icon="true"> + </j-tree> + </div> + </div> + </j-card> +</template> + +<script setup lang="ts"> +import { findDefaultTree } from '@/api/monitor'; +import { TreeNode } from '@/views/gis/Site/type'; +// 获取当前路由实例 +const route = useRoute(); +const logicId = route.matched[0].name?.toString() || '1'; +const data = ref([]) +const bodyHeight = ref<number>(document.body.clientHeight - 120); +const filterMenuList = (arr, item) => { + arr.forEach((child) => { + console.log(child); + if (child.children && child.children?.length > 0) { + child.disabled = true; + child = filterMenuList(child.children, item); + } + }); + return item; +}; +const getMonitorList = async () => { + const { result } = await findDefaultTree(logicId); + result.forEach((item) => { + item.disabled = true; + if (item.children && item.children.length > 0) { + item = filterMenuList(item.children, item); + } + }); + data.value = result as TreeNode; +} +const search = (value: string) => { + +}; + +const selectTree = (k: any, info: any) => { + +}; + + + +onMounted(async () => { + await getMonitorList(); +}); + +</script> + +<style scoped lang="less"> +:deep(.ant-tree-checkbox-disabled) { + display: none !important; +} + +:deep(.ant-tree .ant-tree-treenode-disabled .ant-tree-node-content-wrapper) { + color: rgba(0, 0, 0, 0.65) !important; +} +</style> \ No newline at end of file diff --git a/src/views/gis/RealTime/Graph/index.vue b/src/views/gis/RealTime/Graph/index.vue index d984253..dd7a704 100644 --- a/src/views/gis/RealTime/Graph/index.vue +++ b/src/views/gis/RealTime/Graph/index.vue @@ -1,93 +1,115 @@ <template> <page-container> - <a-row> - <a-col v-show="isfullscreen" :xs='24' :sm='24' :md='24' :lg='12' :xl='12'> - <j-card> - <j-select show-search v-model:value="activeIframe.montorId" :options="monitorList" size="small" placeholder="请选择" style="width: 100%;"></j-select> - <iframe :id="'realdata'" :ref="'realdata'" src="/static/threePublic/index.html?viewIndex=88" width="100%" :height="activeIframe.iframeHeight" :style="{ border: 'none',height:activeIframe.iframeHeight }" scrolling='no'/> - </j-card> - </a-col> - <a-col :xs='24' :sm='24' :md='24' :lg='isfullscreen?12:24' :xl='isfullscreen?12:24'> - <a-row> - <a-col v-for="(item,index) in iframeInfo" :key="index" :xs="item.xs" :sm="item.sm" :md="item.md" :lg="item.lg" :xl="item.xl"> - <j-card :bordered="false"> - <j-select show-search v-model:value="item.montorId" :options="monitorList" size="small" placeholder="请选择" style="width: 100%;"></j-select> - <iframe :id="'realdata' + index" :ref="'realdata' + index" src="/static/threePublic/index.html" width="100%" :height="item.iframeHeight" :style="{ border: 'none',height:item.iframeHeight }" scrolling='no'/> - </j-card> - </a-col> - </a-row> - </a-col> - </a-row> + <Splitpanes ref="split" class="default-theme" + @splitter-click="treeShow = !treeShow; treeMinSize = treeMinSize == 0 ? 20 : 0;" + :style="{ height: bodyHeight + 'px', overflow: true, minHeight: '720px', width: '100%' }"> + <Pane ref="paneLeft" :size="treeMinSize" :min-size="treeMinSize" v-show="treeShow"> + <LeftMonitor/> + </Pane> + <pane min-size="80" class="realTime" ref="paneRight" :size="100 - treeMinSize" + :style="{ minWidth: '900px' }"> + <a-row> + <a-col v-show="isfullscreen" :xs='24' :sm='24' :md='24' :lg='12' :xl='12'> + <j-card> + <iframe :id="'realdata'" :ref="'realdata'" src="/static/threePublic/index.html?viewIndex=88" + width="100%" :height="activeIframe.iframeHeight" + :style="{ border: 'none', height: activeIframe.iframeHeight }" scrolling='no' /> + </j-card> + </a-col> + <a-col :xs='24' :sm='24' :md='24' :lg='isfullscreen ? 12 : 24' :xl='isfullscreen ? 12 : 24'> + <a-row> + <a-col v-for="(item, index) in iframeInfo" :key="index" :xs="item.xs" :sm="item.sm" + :md="item.md" :lg="item.lg" :xl="item.xl"> + <j-card :bordered="false"> + + <iframe :id="'realdata' + index" :ref="'realdata' + index" + src="/static/threePublic/index.html" width="100%" :height="item.iframeHeight" + :style="{ border: 'none', height: item.iframeHeight }" scrolling='no' /> + </j-card> + </a-col> + </a-row> + </a-col> + </a-row> + </pane> + </Splitpanes> + + </page-container> </template> <script lang="ts" setup name="RealTimeGraph"> -import {getCurrentInstance} from 'vue'; +import { Splitpanes, Pane } from 'splitpanes' +import LeftMonitor from '@/views/gis/components/LeftMonitor/index.vue' +import 'splitpanes/dist/splitpanes.css' +import { getCurrentInstance } from 'vue'; import type { iframeInfoType } from './type.d.ts' import { minHeight, iframeChild1, iframeChild2 } from './data.ts' import { queryList, getAccessConfig } from '@/api/device/product' const instance = getCurrentInstance(); +const treeShow = ref<boolean>(false); +const treeMinSize = ref<number>(0); +const bodyHeight = ref<number>(document.body.clientHeight-120); //高度计算 -const getHeight = (num:number,minHeight:number) :number => { - const height = (document.body.clientHeight - (60 + (num * 36))) / num - return height < minHeight ? minHeight : height +const getHeight = (num: number, minHeight: number): number => { + const height = (document.body.clientHeight - (60 + ((num == 1 ? 3 : num) * 65))) / num + return height //< minHeight ? minHeight : height } const isfullscreen = ref<boolean>(false); const height = ref<number>(627); -const activeIframe = ref<iframeInfoType>({index:0,montorId:1,viewType:1,xs:24,sm:24,md:12,lg:12,xl:8,iframeHeight:getHeight(1,627)}) +const activeIframe = ref<iframeInfoType>({ index: 0, montorId: 1, viewType: 1, xs: 24, sm: 24, md: 12, lg: 12, xl: 8, iframeHeight: getHeight(1, 627) }) const iframeInfo = reactive<iframeInfoType[]>([]) const monitorList = ref<{ label: any; value: any; }[]>([ - {label:'监测点1',value:1}, - {label:'监测点2',value:2}, - {label:'监测点3',value:3}, - {label:'监测点4',value:4}, - {label:'监测点5',value:5}, - {label:'监测点6',value:6}, - ]) + { label: '监测点1', value: 1 }, + { label: '监测点2', value: 2 }, + { label: '监测点3', value: 3 }, + { label: '监测点4', value: 4 }, + { label: '监测点5', value: 5 }, + { label: '监测点6', value: 6 }, +]) // 定义一个更新窗口高度的函数 const updateheight = () => { - activeIframe.value.iframeHeight = getHeight(isfullscreen.value?3:2,300)*3 + activeIframe.value.iframeHeight = getHeight(1, 627) + 78; iframeInfo.map((i, id) => { - iframeInfo[id].iframeHeight = getHeight(isfullscreen.value?3:2,300)// splice(id,1,{...i,iframeHeight: getHeight(isfullscreen.value?3:2,300)}) + iframeInfo[id].iframeHeight = getHeight(isfullscreen.value ? 3 : 2, 300)// splice(id,1,{...i,iframeHeight: getHeight(isfullscreen.value?3:2,300)}) }) }; -const listenerFun = (e:any) => { - if(e?.data?.command == 'event'){ +const listenerFun = (e: any) => { + if (e?.data?.command == 'event') { console.log(e) - }else if (e?.data?.command == 'load'){ - iframeInfo.forEach((it,id) => { + } else if (e?.data?.command == 'load') { + iframeInfo.forEach((it, id) => { let i18n = { - command:'i18n', - index:id, - i18n:{}, + command: 'i18n', + index: id, + i18n: {}, }; window.frames["realdata" + id].contentWindow.postMessage(i18n, '*') }) - }else if (e?.data?.command == 'dblClick'){ - iframeInfo.forEach((it,id) => { - iframeInfo.splice(id,1,{...iframeChild1,index:id}) - window.frames["realdata" + id].contentWindow.postMessage({command: 'setView',...iframeChild1}, '*') + } else if (e?.data?.command == 'dblClick') { + iframeInfo.forEach((it, id) => { + iframeInfo.splice(id, 1, { ...iframeChild1, index: id }) + window.frames["realdata" + id].contentWindow.postMessage({ command: 'setView', ...iframeChild1 }, '*') }) isfullscreen.value = true; updateheight() - }else if(e.data.command == 'closeFullScreen'){ + } else if (e.data.command == 'closeFullScreen') { isfullscreen.value = false; - [1,2,3,4,5,6].map((i,id) => { - iframeInfo.splice(id,1,{ - ...iframeChild2, - index:id, - montorId:id, - }); - window.frames["realdata" + id].contentWindow.postMessage({command: 'setView',...iframeChild2}, '*') - updateheight() - }) + [1, 2, 3, 4, 5, 6].map((i, id) => { + iframeInfo.splice(id, 1, { + ...iframeChild2, + index: id, + montorId: id, + }); + window.frames["realdata" + id].contentWindow.postMessage({ command: 'setView', ...iframeChild2 }, '*') + updateheight() + }) } }; const iframeIsShow = computed(() => { const isfullscreen = iframeInfo.filter(i => i.viewType == 2).length == 0; - return (index:number)=>{ + return (index: number) => { return !(isfullscreen && index == iframeInfo.length - 1) } }); @@ -97,25 +119,25 @@ watchEffect(() => { }) const initIframeInfo = () => { - [1,2,3,4,5,6].map((i,id) => { + [1, 2, 3, 4, 5, 6].map((i, id) => { iframeInfo.push({ ...iframeChild2, - index:i, - montorId:i, - iframeHeight: getHeight(2,300), + index: i, + montorId: i, + iframeHeight: getHeight(2, 300), }) }) }; initIframeInfo() const bindListener = () => { - window.removeEventListener('message',listenerFun,true); -// if(!window.listenerEcharts ){ - window.addEventListener('message',listenerFun,true ); + window.removeEventListener('message', listenerFun, true); + // if(!window.listenerEcharts ){ + window.addEventListener('message', listenerFun, true); // window.listenerEcharts = true; -// } + // } }; // 在 mounted 生命周期钩子中添加事件监听器 -onBeforeMount(async() => { +onBeforeMount(async () => { const resp = await queryList({}) console.log("🚀 resp:", resp) bindListener() @@ -131,7 +153,7 @@ onUnmounted(() => { </script> <style lang="less" scoped> -:deep(.ant-card-body){ - padding: 5px 12px; +:deep(.ant-card-body) { + // padding: 5px 12px 12px 12px; } </style> diff --git a/src/views/gis/Site/RoleLeft/AddGroup.vue b/src/views/gis/Site/RoleLeft/AddGroup.vue new file mode 100644 index 0000000..d37603f --- /dev/null +++ b/src/views/gis/Site/RoleLeft/AddGroup.vue @@ -0,0 +1,205 @@ +<template> + <j-modal :maskClosable="false" width="650px" :visible="true" :title="!!data?.id ? '编辑站点' : '新增站点'" @ok="handleSave" + @cancel="handleCancel" :confirmLoading="loading"> + <div style="margin-top: 10px"> + <j-form :layout="'vertical'" ref="formRef" :model="modelRef"> + <j-form-item label="站点名称" name="nodeName" :rules="[ + { + required: true, + message: '请输入名称', + }, + { + max: 64, + message: '最多输入64个字符', + }, + ]"> + <j-input v-model:value="modelRef.nodeName" placeholder="请输入名称" /> + </j-form-item> + + + <j-form-item label="背景图(未上传图片则使用地图背景)" name="image"> + <div class="upload-image-warp-back"> + <div class="upload-image-border-back"> + <j-upload name="file" :customRequest="customRequest" :beforeUpload="beforeLogoUpload" + :showUploadList="false" accept="'.jpg', '.png', '.jfif', '.pjp', '.pjpeg', '.jpeg'"> + <div class="upload-image-content-back"> + <div v-if="loading" class="loading-back"> + <AIcon type="LoadingOutlined" /> + </div> + <div class="upload-image" v-if="modelRef.image" :style="modelRef.image + ? `background-image: url(${modelRef.image});` + : '' + "></div> + <div v-if="modelRef.image" class="upload-image-mask"> + 点击修改 + </div> + <div v-else> + <div> + <AIcon type="PlusOutlined" /> + </div> + </div> + </div> + </j-upload> + </div> + </div> + </j-form-item> + </j-form> + </div> + </j-modal> +</template> + +<script lang="ts" setup> +import { isExists, update } from '@/api/device/instance'; +import { getImage, onlyMessage, LocalStore } from '@/utils/comm'; +import { addMonitor } from '@/api/monitor'; +const emit = defineEmits(['close', 'save']); +const props = defineProps({ + data: { + type: Object, + default: undefined, + }, + logicId: { + type: String, + } +}); +const loading = ref<boolean>(false); + +const formRef = ref(); + +const modelRef = reactive({ + id: undefined, + nodeName: '', + image: '', + +}); +const beforeLogoUpload = (file: File) => { + const typeBool = + ['.jpg', '.png', '.jfif', '.pjp', '.pjpeg', '.jpeg'] + .map((m: string) => m.split('.')[1]) + .filter((typeStr) => file.type.includes(typeStr)).length > 0; + const sizeBool = file.size / 1024 / 1024 < 4; + if (!typeBool) { + onlyMessage(`请上传.jpg.png.jfif.pjp.pjpeg.jpeg格式的图片`, 'error'); + } else if (!sizeBool) { + onlyMessage(`图片大小必须小于4M`, 'error'); + } + return typeBool && sizeBool; +}; +/** + 获取file,通过FileReader获取图片的 base64 +*/ +const customRequest = (option: any) => { + const formData = new FormData(); + formData.append("files[]", option.file); + const reader = new FileReader(); + reader.readAsDataURL(option.file); + reader.onloadend = function (e) { + modelRef.image = e.target.result || '' + if (e && e.target && e.target.result) { + option.onSuccess(); + } + }; +} + +watch( + () => props.data, + (newValue) => { + Object.assign(modelRef, newValue); + }, + { immediate: true, deep: true }, +); + +const handleCancel = () => { + emit('close'); + formRef.value.resetFields(); +}; + +const handleSave = () => { + formRef.value + .validate() + .then(async (_data: any) => { + loading.value = true; + const obj = { ..._data }; + obj.logicId = props.logicId; + obj.parentId = 0; + if (!obj.id) { + delete obj.id; + } + const resp = await addMonitor(obj).finally(() => { + loading.value = false; + }); + if (resp.status === 200) { + onlyMessage('操作成功!'); + emit('save'); + emit('close'); + formRef.value.resetFields(); + } + }) + .catch((err: any) => { + console.log('error', err); + }); +}; +</script> +<style lang="less" scoped> +.upload-image-warp-back { + display: flex; + justify-content: flex-start; + + .upload-image-border-back { + position: relative; + overflow: hidden; + border: 1px dashed #d9d9d9; + transition: all 0.3s; + width: 100%; + height: 415px; + + &:hover { + border: 1px dashed #1890ff; + display: flex; + } + + .upload-image-content-back { + align-items: center; + justify-content: center; + position: relative; + display: flex; + flex-direction: column; + width: 610px; + height: 415px; + padding: 8px; + background-color: rgba(0, 0, 0, 0.06); + cursor: pointer; + + .loading-back { + position: absolute; + } + + .upload-image { + width: 100%; + height: 100%; + background-repeat: no-repeat; + background-position: 50%; + background-size: cover; + } + + .upload-image-mask { + align-items: center; + justify-content: center; + position: absolute; + top: 0; + left: 0; + display: none; + width: 100%; + height: 100%; + color: #fff; + font-size: 16px; + background-color: rgba(0, 0, 0, 0.35); + } + + &:hover .upload-image-mask { + display: flex; + } + } + } +} +</style> \ No newline at end of file diff --git a/src/views/gis/Site/RoleLeft/BindDevice.vue b/src/views/gis/Site/RoleLeft/BindDevice.vue new file mode 100644 index 0000000..c96c66a --- /dev/null +++ b/src/views/gis/Site/RoleLeft/BindDevice.vue @@ -0,0 +1,253 @@ +<template> + <j-modal :maskClosable="false" width="650px" :visible="true" :title="'设备绑定'" :confirmLoading="loading" @ok="confirm" + @cancel="emit('close')"> + <j-form ref="formRef" :model="form.data" layout="vertical"> + <div class="formName">绑定产品信息</div> + <j-row :gutter="24"> + <j-col :span="24"> + <j-form-item :label="form.data.groupName" name="productId"> + <j-tree-select showSearch v-model:value="form.data.productId" placeholder="请选择产品分类" + :tree-data="form.productOptions" allow-clear :fieldNames="{ + label: 'name', + value: 'id', + children: 'children', + }" :filterTreeNode="(v, option) => filterSelectNode(v, option, 'name')" + @change="form.getDeviceoptions"> + </j-tree-select> + </j-form-item> + </j-col> + </j-row> + <div class="formName">绑定设备信息</div> + <j-row :gutter="24"> + <j-col :span="12" v-for="(item, index) in form.data.monitorNames"> + <j-form-item :label="item" :name="'diviceId' + index"> + <j-select + showSearch + :options="form.deviceOptions" + v-model:value="form.data.deviceIds[index]" + placeholder="请选择设备" + :fieldNames="{ + label: 'name', + value: 'id', + children: 'children', + }" + /> + </j-form-item> + </j-col> + </j-row> + </j-form> + </j-modal> +</template> + +<script lang="ts" setup> +import { isExists, update } from '@/api/device/instance'; +import { filterSelectNode, onlyMessage } from '@/utils/comm'; +import { queryDevice } from '@/api/device/firmware'; +import { editMonitor } from '@/api/monitor'; +import { category } from '@/api/device/product'; +import encodeQuery from '@/utils/encodeQuery'; +import { getDeviceList_api,} from '@/api/home'; +const emit = defineEmits(['close', 'save']); +const props = defineProps({ + checkNodeData: { + type: Object, + }, + logicId: { + type: String, + } +}); +const loading = ref<boolean>(false); +type formType = { + id?: string; + productId?: string;//产品ID + groupName?: string;//组名称 + monitorNames: [], + deviceIds: [], +}; +const form = reactive({ + data: {} as formType, + rules: { + }, + productOptions: [], + deviceOptions: [], + groupParams: {}, + monitorParams: [], + init: () => { + form.getFormData(); + form.getOptions(); + }, + processNestedArray: (processFn: any, nestedArray: any): any => { + const recurse = (arr: any): any => { + return arr.map(item => { + // 处理当前项 + const processedItem = processFn(item); + // 如果有 children 则递归处理 + if (item.children && Array.isArray(item.children)) { + processedItem.children = recurse(item.children); + } + return processedItem; // 返回处理后的项 + }); + }; + return recurse(nestedArray); + }, + getFormData: () => { + const result = []; + const { checkNodeData: { id, nodeProduct, children,nodeName } } = props; + const recurse = (arr) => { + arr.forEach(item => { + // 如果 nodeType 等于 1,直接添加到结果中 + if (item.nodeType == 1) { + result.push(item); + form.monitorParams.push({id: item.id,deviceId: item.deviceId}) + } else if (item.children && Array.isArray(item.children)) { + // 如果 nodeType 等于 0,并且存在 children,则递归处理 + recurse(item.children); + } + }); + }; + recurse(children); // 开始递归处理 + form.groupParams = { id, nodeProduct}; + form.data.productId = nodeProduct; + if (nodeProduct) { + form.getDeviceoptions(nodeProduct) + } + form.data.groupName = nodeName; + form.data.monitorNames = Array.from(result.length); + form.data.deviceIds = Array.from(result.length); + result.forEach((it, id) => { + form.data.monitorNames[id] = it.nodeName; + form.data.deviceIds[id] = it.deviceId; + }) + }, + getDeviceoptions: async (id) => { + const params = { + pageIndex: 0, + pageSize: 9999999, + terms: [ + { + terms: [{ + column: 'productId$product-info$eq', + type: 'or', + value: `classifiedId is ${id}`, + }] + } + ] + + }; + const res = await getDeviceList_api(params); + if (res.status === 200) { + form.deviceOptions = res.result.data; + } + }, + getOptions: async () => { + const resp = await category(encodeQuery({ sorts: { sortIndex: 'asc' } }))//.then((resp) => { + if (resp.status === 200) { + form.productOptions = resp.result; + form.productOptions = form.dealProductTree(form.productOptions); + + } + }, + dealProductTree: (arr: any) => { + return arr.map((element: any) => { + element.key = element.id; + if (element.children) { + element.children = form.dealProductTree(element.children); + } + return element; + }); + }, + getLineList: async () => { + const { result } = await findDefaultTree(logicId); + form.lineOptions = form.processNestedArray((i) => ({ ...i, disabled: i.nodeType == 1 ? true : false }), result)// result as TreeNode; + form.preMonitorOptions = form.processNestedArray((i) => ({ ...i, disabled: i.nodeType != 1 ? true : false }), result) + form.subMonitorOptions = form.processNestedArray((i) => ({ ...i, disabled: i.nodeType != 1 ? true : false }), result) + }, + handleSubMonitorOptions: (id: string) => { + form.subMonitorOptions = form.processNestedArray( + (i) => ({ ...i, disabled: i.id == id || i.nodeType != 1 ? true : false }), + form.subMonitorOptions + ) + }, + getConfig: () => { + const id = props.data.id || ''; + + if (props.type === 'add') { form.data = {} as formType; } + else if (props.type === 'edit') { + getAssociateConfigList_api(id).then((resp: any) => { + + form.data = { + ...(resp.result as formType), + }; + + nextTick(() => { + formRef.value?.clearValidate(); + }); + }); + } + }, + submit: (): Promise<any> => { + const promiseAll = []; + if (form.groupParams.nodeProduct != form.data.productId) { + promiseAll.push(editMonitor({ id: form.groupParams.id, nodeProduct: form.data.productId })); + } + form.monitorParams.forEach((item,index) => { + if (item.deviceId != form.data.deviceIds[index]) { + promiseAll.push(editMonitor({ id: item.id, deviceId: form.data.deviceIds[index] })); + } + }) + return Promise.all(promiseAll); + }, + saveName: (nodeName, name, deviceId, extra) => { + form.data[name] = nodeName; + if (deviceId) { + form.data[deviceId] = extra.triggerNode.props.deviceId || 1; + } + }, +}); +form.init(); +const formRef = ref(); +const confirm = () => { + loading.value = true; + formRef.value + ?.validate() + .then(() => form.submit()) + .then((resp: any) => { + onlyMessage('操作成功'); + emit('save'); + emit('close'); + }) + .finally(() => (loading.value = false)); +}; +const modelRef = reactive({ + id: undefined, + nodeName: '', + image: '', + +}); + + +watch( +); + +const handleCancel = () => { + emit('close'); + formRef.value.resetFields(); +}; + + +</script> +<style lang="less" scoped> +.formName { + margin-bottom: 10px; + font-size: 16px; + + &::before { + width: 2px; + background-color: rgb(184, 184, 184); + display: inline-block; + height: 13px; + margin-right: 3px; + content: '' + } +} +</style> \ No newline at end of file diff --git a/src/views/gis/Site/RoleLeft/index.vue b/src/views/gis/Site/RoleLeft/index.vue index bfad1be..d124fa6 100644 --- a/src/views/gis/Site/RoleLeft/index.vue +++ b/src/views/gis/Site/RoleLeft/index.vue @@ -6,64 +6,60 @@ </template> </j-input> <div v-if="admin" class="add-btn"> - <j-button type="primary" @click="addGroup"> - 新增站点 - </j-button> + + <j-row :style="{ marginTop: '10px' }"> + <j-button type="primary" @click="visible = true"> + 新增站点 + </j-button> + </j-row> + </div> <div class="tree"> <j-spin :spinning='loading'> - <j-tree :tree-data="listData" v-if="listData.length" draggable @dragstart="dragstart" - :fieldNames="{ title: 'name', key: 'id', children: 'children' }" blockNode - :selectedKeys="selectedKeys" :default-expanded-keys="defaultExpandedKeys" - :showLine="{ showLeafIcon: false }" :showIcon="true"> - <template #title="{ name, data, level, type, positionX }"> + <j-tree :tree-data="treeData" v-if="treeData.length" draggable @dragstart="dragstart" + :fieldNames="{ title: 'nodeName', key: 'id', children: 'children' }" blockNode + :default-expanded-keys="defaultExpandedKeys" :showLine="{ showLeafIcon: false }" :showIcon="true"> + <template #title="{ nodeName, data, level, type, positionX, parentId, nodeType }"> <div v-if="selectId === data.id"> <j-input v-model:value="addName" @blur="() => saveGroup(data)" ref="inputRef" :maxlength="64"></j-input> </div> <span v-else class='department-tree-item-content'> <span class='title' :style="{ - width: type === 'root' ? '60px' : (level === 2 ? '80px' : '60px'), + width: nodeType == '1' ? '80px' : '110px', }"> <j-ellipsis> - <AIcon v-if="level === 2 && positionX" type="HeartFilled" /> - &nbsp;{{ name }} + <AIcon v-if="nodeType == '1'" type="HeartFilled" /> + &nbsp;{{ nodeName }} </j-ellipsis> </span> <span class="func-btns" :style="{ - width: type === 'root' ? '145px' : (level === 2 ? '80px' : '100px'), + width: nodeType == '0' ? '120px' : '100px', }" @click="(e) => e.stopPropagation()"> - <PermissionButton hasPermission="system/Role:groupUpdate" type="link" - :tooltip="{ title: '编辑' }" @click="editGroup(data)"> + <PermissionButton type="link" :tooltip="{ title: '编辑' }" @click="editGroup(data)"> <AIcon type="EditOutlined" /> </PermissionButton> - <PermissionButton v-if="level == 1" hasPermission="system/Role:groupUpdate" type="link" - :tooltip="{ title: '新增子站' }"> + <PermissionButton v-if="nodeType === '0' || parentId === '0'" type="link" + :tooltip="{ title: '组内监测点绑定设备' }" @click="openBindModel(data)"> + <AIcon type="ApiOutlined" /> + </PermissionButton> + <PermissionButton v-if="nodeType === '0' || parentId === '0'" type="link" + :tooltip="{ title: '新增子站' }" @click="addGroup(data, '0')"> <AIcon type="PlusOutlined" /> </PermissionButton> - <PermissionButton v-if="level == 1" hasPermission="system/Role:groupUpdate" type="link" - :tooltip="{ title: '新增子监测点' }"> + <PermissionButton v-if="nodeType === '0' || parentId === '0'" type="link" + :tooltip="{ title: '新增子监测点' }" @click="addGroup(data, '1')"> <AIcon type="PlusCircleOutlined" /> </PermissionButton> - <PermissionButton hasPermission="system/Role:groupUpdate" type="link" - :tooltip="{ title: '删除' }" :popConfirm="{ - title: `确定要删除吗`, - onConfirm: () => { }, - }"> + <PermissionButton type="link" :tooltip="{ title: '删除' }" :popConfirm="{ + title: `确定要删除吗`, + onConfirm: () => { deleteGroup(data.id) }, + }"> <AIcon type="DeleteOutlined" /> </PermissionButton> - <PermissionButton v-if="level == 2" hasPermission="system/Role:groupUpdate" type="link" - :tooltip="{ title: '移除' }"> + <PermissionButton v-if="nodeType === '1'" type="link" :tooltip="{ title: '移除' }"> <AIcon type="CloseOutlined" /> </PermissionButton> - <PermissionButton v-if="type === 'root'" hasPermission="system/Role:groupUpdate" - type="link" :tooltip="{ title: '导入excel' }"> - <AIcon type="UploadOutlined" /> - </PermissionButton> - <PermissionButton v-if="type === 'root'" hasPermission="system/Role:groupUpdate" - type="link" :tooltip="{ title: '导出excel' }"> - <AIcon type="DownloadOutlined" /> - </PermissionButton> </span> </span> </template> @@ -71,6 +67,21 @@ <j-empty v-else description="暂无数据" /> </j-spin> </div> + <j-row :gutter="10"> + <j-col :span="12"> + <j-button type="primary" @click="visible = true"> + 导入站点 + </j-button> + </j-col> + <j-col :span="12"> + <j-button type="primary" @click="visible = true"> + 导出站点 + </j-button> + </j-col> + </j-row> + <AddGroup v-if="visible" :logicId="logicId" @close="visible = false" @save="() => { emit('getMonitorList') }" /> + <BindDevice v-if="bindVisible" :checkNodeData="checkNodeData" @close="bindVisible = false" + @save="() => { emit('getMonitorList') }" /> </div> </template> @@ -79,12 +90,24 @@ import { onlyMessage } from '@/utils/comm'; import { useUserInfo } from '@/store/userInfo'; import { storeToRefs } from 'pinia'; import { filterTreeItem } from '@/utils/comm' +import { addMonitor, editMonitor, deleteMonitor } from '@/api/monitor'; +import { randomString } from '@/utils/utils'; +import AddGroup from './AddGroup.vue'; +import BindDevice from './BindDevice.vue'; + +const visible = ref<boolean>(false); +const bindVisible = ref<boolean>(false); const loading = ref<boolean>(false); // 数据加载状态 -const emit = defineEmits(['selectData']); +const emit = defineEmits(['selectData', 'getMonitorList']); +const treeData = ref([]); +const checkNodeData = ref([]); const props = defineProps({ listData: { type: Array, default: [] + }, + logicId: { + type: String, } }) const userInfoStore = useUserInfo(); @@ -93,19 +116,40 @@ const admin = computed(() => { return userInfos.value?.username === 'admin'; }); -const selectedKeys = ref<string[]>(['global_role']); const searchValue = ref(); const inputRef = ref(); const addName = ref(); const selectId = ref(); -const mapMonitorList = ref(); -const defaultExpandedKeys = ref<(number | string)[]>([]); +const defaultExpandedKeys = ref<(number | string)[]>([]); -const dragstart = ({ event, node }: { event: DragEvent, node: any }) => { - // console.log("🚀 node:", node) - //event.preventDefault() - event.dataTransfer?.setData('monitorData', JSON.stringify(node.dataRef)); // 使用 text/plain 类型存储节点信息 -} +const isMonitorList = (node) => { + if (!node) { + return false; + } + const result = []; + const recurse = (arr) => { + arr.forEach(item => { + // 如果 nodeType 等于 1,直接添加到结果中 + if (item.nodeType == 1) { + result.push(item); + } else if (item.children && Array.isArray(item.children)) { + // 如果 nodeType 等于 0,并且存在 children,则递归处理 + recurse(item.children); + } + }); + }; + recurse(node); // 开始递归处理 + return result.length +}; + +const openBindModel = (node) => { + if (isMonitorList(node?.children)) { + checkNodeData.value = node; + bindVisible.value = true; + } else { + onlyMessage('该组下无监测点') + } +}; //获取全部id const getTreeId = <T>(treeNode: T[]) => { @@ -119,7 +163,19 @@ const getTreeId = <T>(treeNode: T[]) => { treeNode?.forEach((child: any) => traverse(child)) return result } -defaultExpandedKeys.value = getTreeId(props.listData) +watch(() => props.listData, (newValue) => { + treeData.value = newValue; + defaultExpandedKeys.value = getTreeId(treeData.value) +}, { deep: true, immediate: true }) +const openGroup = () => { + visible.value = false; +}; + +const dragstart = ({ event, node }: { event: DragEvent, node: any }) => { + //event.preventDefault() + event.dataTransfer?.setData('monitorData', JSON.stringify(node.dataRef)); // 使用 text/plain 类型存储节点信息 +} + const queryGroup = async (select?: Boolean, searchName?: string) => { // const params = searchName // ? { @@ -149,42 +205,72 @@ const queryGroup = async (select?: Boolean, searchName?: string) => { // // } // } }; -const addGroup = () => { - // addName.value = ''; - // const newId = randomString(); - // listData.value[0].children?.splice(1, 0, { - // name: '', - // id: newId, - // }); - // selectId.value = newId; - // nextTick(() => { - // inputRef.value.focus(); - // }); +const addGroup = async (data: any, nodeType: any) => { + + addName.value = ''; + const newId = randomString(); + if (!data.children) { + data.children = [] + }; + data.children?.splice(1, 0, { + isNewChild: true, + nodeName: '', + logicId: props.logicId, + nodeType, + id: newId, + parentId: data.id, + }); + treeData.value = [...treeData.value] + selectId.value = newId; + defaultExpandedKeys.value = getTreeId(treeData.value) + await nextTick(() => { + inputRef.value.focus(); + }); }; const saveGroup = async (data: any) => { - if (addName.value === '') { - if (data.name === '') { - // listData.value[0].children.splice(1, 1); - } - } - else { + + if (data.nodeName !== addName.value) { const saveData = { - name: addName.value, - id: data.id, + ...data, + nodeName: addName.value, }; - const res = { status: 200 }//await saveRoleGroup(saveData); + if (data.isNewChild) { + delete data.id; + } + const res = data.isNewChild ? await addMonitor(saveData) : await editMonitor(saveData); if (res.status === 200) { onlyMessage('操作成功!'); - // queryGroup(); + emit('getMonitorList'); } else { onlyMessage('操作失败!'); } + } else if (data.isNewChild) { + const removeChild = (child) => { + child.forEach((item, index) => { + if (item.id == data.id) { + child.splice(index, 1); + } else if (item?.children?.length) { + removeChild(item.children) + } + }) + } + removeChild(treeData.value) } setTimeout(() => { selectId.value = ''; }, 300); }; +const deleteGroup = async (id: string) => { + const res = await deleteMonitor(id); + if (res.status === 200) { + onlyMessage('操作成功!'); + emit('getMonitorList'); + } else { + onlyMessage('操作失败!'); + } +} + const search = () => { queryGroup(true, searchValue.value); }; @@ -196,7 +282,7 @@ const searchChange = () => { const editGroup = (data: any) => { if (!selectId.value) { selectId.value = data.id; - addName.value = data.name; + addName.value = data.nodeName; // listData.value[0].children?.forEach((item: any) => { // if (item.id === data.id) { // item.edit = true; @@ -209,7 +295,6 @@ const editGroup = (data: any) => { }; onMounted(() => { - mapMonitorList.value = filterTreeItem(props.listData, 'children', (i) => i.level == 2 && i.positionX) }); </script> <style lang="less" scoped> @@ -225,8 +310,9 @@ onMounted(() => { width: 100%; } } - :deep(.ant-tree-treenode-disabled){ - .ant-tree-node-content-wrapper{ + + :deep(.ant-tree-treenode-disabled) { + .ant-tree-node-content-wrapper { color: rgba(0, 0, 0, 0.55) } } diff --git a/src/views/gis/Site/index.vue b/src/views/gis/Site/index.vue index 6d21dec..6f9f894 100644 --- a/src/views/gis/Site/index.vue +++ b/src/views/gis/Site/index.vue @@ -1,44 +1,62 @@ <template> <page-container> - <FullPage> - <div class="dictionary_contain"> - <div class="dictionary_left"> - <Left :listData="listData" @select-data="selectData"/> - </div> - <div class="dictionary_right"> - <Right :listData="listData" :groupId="groupId"/> - </div> - </div> - </FullPage> + <FullPage> + <div class="dictionary_contain"> + <div class="dictionary_left"> + <Left :listData="listData" :logicId="logicId" @getMonitorList="getMonitorList" + @select-data="selectData" /> + </div> + <div class="dictionary_right"> + <Right :listData="listData" :groupId="groupId" /> + </div> + </div> + </FullPage> </page-container> - </template> - - <script lang="ts" setup> - import Left from './RoleLeft/index.vue' +</template> + +<script lang="ts" setup> +import Left from './RoleLeft/index.vue' import Right from './RoleRight/index.vue' import { TreeNode } from './type'; import { treeList } from '@/views/gis/data' import { Ref } from 'vue'; -const listData: Ref<TreeNode[]> = ref(treeList); - const groupId = ref() - const selectData = (data:any)=>{ +import { findDefaultTree } from '@/api/monitor' + +const listData: Ref<TreeNode[]> = ref([]); +const groupId = ref() +// 获取当前路由实例 +const route = useRoute(); +const logicId = route.matched[0].name?.toString() || '1'; + +const getMonitorList = async () => { + const { result } = await findDefaultTree(logicId); + listData.value = result as TreeNode; +} + +onMounted(async () => { + await getMonitorList(); +}); + +const selectData = (data: any) => { groupId.value = data[0] - } - </script> - <style lang="less" scoped> - .dictionary_contain{ - display: flex; - background-color: #fff; - padding: 24px; - height: 100%; - } - .dictionary_left{ - border-right: 1px solid #f0f0f0; - padding-right: 24px; - flex:1; - height:100% - } - .dictionary_right{ - flex:4 - } - </style> \ No newline at end of file +} +</script> +<style lang="less" scoped> +.dictionary_contain { + display: flex; + background-color: #fff; + padding: 24px; + height: 100%; +} + +.dictionary_left { + border-right: 1px solid #f0f0f0; + padding-right: 24px; + flex: 1; + height: 100% +} + +.dictionary_right { + flex: 4 +} +</style> \ No newline at end of file diff --git a/src/views/gis/Site/type.d.ts b/src/views/gis/Site/type.d.ts index 3220354..f3d5a88 100644 --- a/src/views/gis/Site/type.d.ts +++ b/src/views/gis/Site/type.d.ts @@ -1,3 +1,11 @@ +/* + * @Author: zhangliang 1466013297@qq.com + * @Date: 2024-09-19 10:20:10 + * @LastEditors: zhangliang 1466013297@qq.com + * @LastEditTime: 2024-09-23 18:00:26 + * @FilePath: \jetlinks-ui-vue-old\src\views\gis\Site\type.d.ts + * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE + */ export type iframeInfoType = { xs?: 6 | 8 | 12 | 24, sm?: 6 | 8 | 12 | 24, @@ -24,10 +32,32 @@ export interface monitorType { positionY?: number | string, } export interface TreeNode { - name: string; - id: string; - disabled?: boolean; // 可选属性,表示节点是否被禁用 - children?: TreeNode[] | monitorType[]; // 子节点数组,可选属性 - level?: number; // 节点的层级,可选属性 1为监测点组级别。2为监测点级别 - type?: string; // 节点的类型,可选属性 root为跟节点 + // name: string; + // id: string; + // disabled?: boolean; // 可选属性,表示节点是否被禁用 + // children?: TreeNode[] | monitorType[]; // 子节点数组,可选属性 + // level?: number; // 节点的层级,可选属性 1为监测点组级别。2为监测点级别 + // type?: string; // 节点的类型,可选属性 root为跟节点 + id?: number | string, + logicId?: number | string,//业务系统ID,对应的业务系统菜单的ID + parentId: number | string,//父ID 根节点默认传0 + nodeName: string,//节点名称 + nodeType: number | string,//节点类型 0组,1监测点 + nodeProduct: number | string,//节点关联产品,监测点必须关联 + deviceId: number | string, //关联设备id,监测点必须关联 + voltageLevel: number | string, //电压等级 + image: string,//图片Base64,非地图节点 + positionX: number | string, //坐标X/经度 监测点必填 + positionY: number | string, //坐标Y/经度 监测点必填 + sort: number | string, //排序 + forwardNum: number | string, //转发编号 监测点必填 + devId: number | string, //通道编号:61850使用 + nodeExtension: number | string, //拓展字段,以json格式存储 + creatorId: number | string, //创建人ID + createTime: number | string, //创建时间 + modifierId: number | string, //修改人ID + modifyTime: number | string, //修改时间 + creatorName: number | string, + modifierName: number | string, + } \ No newline at end of file diff --git a/src/views/gis/components/LeftMonitor/index.vue b/src/views/gis/components/LeftMonitor/index.vue new file mode 100644 index 0000000..dade451 --- /dev/null +++ b/src/views/gis/components/LeftMonitor/index.vue @@ -0,0 +1,66 @@ +<template> + <j-card> + <div class="operator-box" :style="{ height: bodyHeight + 'px' }"> + <j-input-search @search="search" allow-clear placeholder="搜索关键字" /> + <div class="tree" :style="{ marginTop: '15px' }"> + <j-tree @select="selectTree" :field-names="{ title: 'nodeName', key: 'id', }" auto-expand-parent + checkable :tree-data="data" :showLine="{ showLeafIcon: false }" :show-icon="true"> + </j-tree> + </div> + </div> + </j-card> +</template> + +<script setup lang="ts"> +import { findDefaultTree } from '@/api/monitor'; +import { TreeNode } from '@/views/gis/Site/type'; +// 获取当前路由实例 +const route = useRoute(); +const logicId = route.matched[0].name?.toString() || '1'; +const data = ref([]) +const bodyHeight = ref<number>(document.body.clientHeight - 120); +const filterMenuList = (arr, item) => { + arr.forEach((child) => { + console.log(child); + if (child.children && child.children?.length > 0) { + child.disabled = true; + child = filterMenuList(child.children, item); + } + }); + return item; +}; +const getMonitorList = async () => { + const { result } = await findDefaultTree(logicId); + result.forEach((item) => { + item.disabled = true; + if (item.children && item.children.length > 0) { + item = filterMenuList(item.children, item); + } + }); + data.value = result as TreeNode; +} +const search = (value: string) => { + +}; + +const selectTree = (k: any, info: any) => { + +}; + + + +onMounted(async () => { + await getMonitorList(); +}); + +</script> + +<style scoped lang="less"> +:deep(.ant-tree-checkbox-disabled) { + display: none !important; +} + +:deep(.ant-tree .ant-tree-treenode-disabled .ant-tree-node-content-wrapper) { + color: rgba(0, 0, 0, 0.65) !important; +} +</style> \ No newline at end of file diff --git a/src/views/link/AccessConfig/Outline/components/FifthKind.vue b/src/views/link/AccessConfig/Outline/components/FifthKind.vue deleted file mode 100644 index daaf797..0000000 --- a/src/views/link/AccessConfig/Outline/components/FifthKind.vue +++ /dev/null @@ -1,100 +0,0 @@ -<template> - <div v-if="data.provider === 'official-edge-gateway'"> - <TitleComponent data="网络组件" v-if="network"></TitleComponent> - <AccessCard - v-if="network" - :data="{ - ...network, - description: network.description - ? network.description - : descriptionList[data.provider], - }" - > - <template #other> - <div class="other"> - <j-tooltip - placement="top" - :title="addressesTip(network.addresses)" - > - <div - v-for="i in (network.addresses || []).slice(0, 1)" - :key="i.address" - class="item" - > - <j-badge :status="getColor(i)" :text="i.address" /> - <span v-if="(network.addresses || []).length > 1" - >等{{ item.addresses.length }}条</span - > - </div> - </j-tooltip> - </div> - </template></AccessCard - > - </div> - <div v-else> - <TitleComponent data="消息协议" v-if="protocol"></TitleComponent> - <AccessCard - v-if="protocol" - :data="{ ...protocol, type: 'protocol' }" - ></AccessCard> - </div> -</template> - -<script setup> -import AccessCard from '../../components/AccessCard/index.vue'; -import { - descriptionList, - NetworkTypeMapping, - ProtocolMapping, -} from '../../data'; -import { getNetworkList, getProtocolList } from '@/api/link/accessConfig'; -const props = defineProps({ - data: { - type: Object, - }, -}); -const network = ref(); -const protocol = ref(); -const queryNetwork = async () => { - const res = await getNetworkList( - NetworkTypeMapping.get(props.data.provider), - '', - {}, - ); - if (res.success) { - network.value = res.result.find((i) => { - return i.id === props.data.channelId; - }); - } -}; -const queryProtocol = async () => { - const res = await getProtocolList( - ProtocolMapping.get(props.data.provider), - { - 'sorts[0].name': 'createTime', - 'sorts[0].order': 'desc', - }, - ); - if (res.success) { - protocol.value = res.result.find((i) => { - return i.id === props.data.protocol; - }); - } -}; -const getColor = (i) => (i.health === -1 ? 'error' : 'processing'); -const addressesTip = (data) => { - let tip = ''; - data.forEach((item) => { - tip = tip + ' ' + item.address; - }); - return tip; -}; -onMounted(() => { - if (props.data.provider === 'child-device') { - queryProtocol(); - } else { - queryNetwork(); - } -}); -</script> -<style lang="less" scoped></style> diff --git a/src/views/link/AccessConfig/Outline/components/FirstKind.vue b/src/views/link/AccessConfig/Outline/components/FirstKind.vue deleted file mode 100644 index e75d3e5..0000000 --- a/src/views/link/AccessConfig/Outline/components/FirstKind.vue +++ /dev/null @@ -1,197 +0,0 @@ -<template> - <div> - <TitleComponent data="网络组件"> </TitleComponent> - <AccessCard - v-if="network" - :data="{ - ...network, - description: network?.description - ? network.description - : descriptionList[data?.provider], - type: 'network', - }" - > - <template #other> - <div class="other"> - <j-tooltip - placement="top" - :title="addressesTip(network.addresses)" - > - <div - v-for="i in (network.addresses || []).slice(0, 1)" - :key="i.address" - class="item" - > - <j-badge :status="getColor(i)" :text="i.address" /> - <span v-if="(network.addresses || []).length > 1" - >等{{ network.addresses.length }}条</span - > - </div> - </j-tooltip> - </div> - </template> - </AccessCard> - <TitleComponent data="消息协议" style="margin-top: 20px"> - </TitleComponent> - <AccessCard v-if="protocol" :data="{ ...protocol, type: 'protocol' }"> - </AccessCard> - <TitleComponent - v-if="config?.routes && config.routes.length > 0" - :data=" - data.provider === 'mqtt-server-gateway' || - data.provider === 'mqtt-client-gateway' - ? 'topic' - : 'URL信息' - " - style="margin-top: 20px" - > - </TitleComponent> - <div v-if="config?.routes && config.routes.length > 0"> - <j-scrollbar height="350"> - <j-table - :pagination="false" - :rowKey="generateUUID()" - :data-source="config.routes || []" - bordered - :columns="config.id === 'MQTT' ? columnsMQTT : columnsHTTP" - :scroll="{ y: 400 }" - > - <template #bodyCell="{ column, text, record }"> - <template v-if="column.dataIndex === 'stream'"> - {{ getStream(record) }} - </template> - </template> - </j-table> - </j-scrollbar> - </div> - </div> -</template> - -<script setup> -import { - getNetworkList, - getProtocolList, - getConfigView, -} from '@/api/link/accessConfig'; -import { - NetworkTypeMapping, - descriptionList, - ProtocolMapping, - ColumnsMQTT, - ColumnsHTTP, -} from '../../data'; -import AccessCard from '../../components/AccessCard/index.vue'; -const props = defineProps({ - data: { - type: Object, - }, -}); -const network = ref(); -const protocol = ref(); -const config = ref(); -const columnsMQTT = ref([]); -const columnsHTTP = ref([]); -function generateUUID() { - var d = new Date().getTime(); - return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace( - /[xy]/g, - function (c) { - var r = (d + Math.random() * 16) % 16 | 0; - d = Math.floor(d / 16); - return (c === 'x' ? r : (r & 0x3) | 0x8).toString(16); - }, - ); -} -const getStream = (record) => { - let stream = ''; - if (record.upstream && record.downstream) stream = '上行、下行'; - else if (record.upstream) stream = '上行'; - else if (record.downstream) stream = '下行'; - return stream; -}; -const queryNetwork = async () => { - const resp = await getNetworkList( - NetworkTypeMapping.get(props.data?.provider), - props.data?.channelId, - {}, - ); - if (resp.status === 200) { - network.value = resp.result.find((i) => { - return i.id === props.data?.channelId; - }); - } -}; -const queryProcotol = async () => { - const resp = await getProtocolList( - ProtocolMapping.get(props.data?.provider), - { - 'sorts[0].name': 'createTime', - 'sorts[0].order': 'desc', - }, - ); - if (resp.status === 200) { - protocol.value = resp.result.find((i) => { - return i.id === props.data?.protocol; - }); - } -}; -const queryConfig = async () => { - const resp = await getConfigView( - props.data.protocol, - ProtocolMapping.get(props.data.provider), - ); - if (resp.success) { - config.value = resp.result; - const Group = { - title: '分组', - dataIndex: 'group', - key: 'group', - ellipsis: true, - align: 'center', - width: 100, - customCell: (record, rowIndex) => { - const obj = { - children: record, - rowSpan: 0, - }; - const list = config.value?.routes || []; - const arr = list.filter((res) => res.group === record.group); - const isRowIndex = - rowIndex === 0 || list[rowIndex - 1].group !== record.group; - isRowIndex && (obj.rowSpan = arr.length); - return obj; - }, - }; - columnsMQTT.value = [Group, ...ColumnsMQTT]; - columnsHTTP.value = [Group, ...ColumnsHTTP]; - } -}; -const getColor = (i) => (i.health === -1 ? 'error' : 'processing'); -const addressesTip = (data) => { - let tip = ''; - data.forEach((item) => { - tip = tip + ' ' + item.address; - }); - return tip; -}; -onMounted(() => { - queryNetwork(); - queryProcotol(); - queryConfig(); -}); -</script> -<style lang="less" scoped> -.other { - width: 100%; - height: 20px; - display: flex; - align-items: center; - justify-content: center; - .item { - width: 100%; - overflow: hidden; - white-space: nowrap; - text-overflow: ellipsis; - } -} -</style> diff --git a/src/views/link/AccessConfig/Outline/components/FourthKind.vue b/src/views/link/AccessConfig/Outline/components/FourthKind.vue deleted file mode 100644 index 928b7eb..0000000 --- a/src/views/link/AccessConfig/Outline/components/FourthKind.vue +++ /dev/null @@ -1,19 +0,0 @@ -<template> - <div> - <a-descriptions bordered :column="1"> - <a-descriptions-item label="消息协议"> - {{ data?.provider }} - </a-descriptions-item> - </a-descriptions> - </div> -</template> - -<script setup> - -const props = defineProps({ - data: { - type: Object, - }, -}); -</script> -<style lang="less" scoped></style> diff --git a/src/views/link/AccessConfig/Outline/components/SecondKind.vue b/src/views/link/AccessConfig/Outline/components/SecondKind.vue deleted file mode 100644 index d4b7c1b..0000000 --- a/src/views/link/AccessConfig/Outline/components/SecondKind.vue +++ /dev/null @@ -1,143 +0,0 @@ -<template> - <div> - <TitleComponent data="插件" v-if="plugin"> </TitleComponent> - <AccessCard v-if="plugin" :data="{ ...plugin, type: 'plugin' }"> - <template #other> - <div class="plugin-other"> - <div class="plugin-id"> - <div class="plugin-text">插件ID:</div> - <div class="other-content"> - <j-ellipsis> - {{ plugin.id }} - </j-ellipsis> - </div> - </div> - <div class="plugin-version"> - <div class="plugin-text">版本号:</div> - <div class="other-content"> - <j-ellipsis> - {{ plugin.version }} - </j-ellipsis> - </div> - </div> - </div> - </template> - </AccessCard> - <TitleComponent data="通用配置" style="margin-top: 20px" v-if="configuration.length"> - </TitleComponent> - <a-descriptions bordered :column="1" v-if="configuration.length"> - <a-descriptions-item v-for="i in configuration" :label="i.name">{{ i.value }}</a-descriptions-item> - </a-descriptions> - </div> -</template> - -<script setup> -import { - getPluginList, - getPluginConfig, - detail, -} from '@/api/link/accessConfig'; -import AccessCard from '../../components/AccessCard/index.vue'; -const props = defineProps({ - data: { - type: Object, - }, -}); -const plugin = ref(); -const config = ref(); -const configuration = ref([]); -const configurationMap = new Map(); -const queryPlugin = () => { - getPluginList({ - terms: [ - { - terms: [ - { - column: 'type', - value: - props.data.provider === 'media-plugin' - ? 'media' - : 'deviceGateway', - termType: 'eq', - }, - ], - }, - ], - sorts: [{ name: 'createTime', order: 'desc' }], - paging: false, - }).then((res) => { - if (res.success) { - plugin.value = res.result.find((i) => { - return i.id === props.data?.channelId; - }); - } - }); -}; - -const queryPluginConfig = async () => { - const res = await getPluginConfig(props.data.channelId); - if (res.success) { - const properties = res.result?.others?.configMetadata?.properties || []; - config.value = properties.forEach((item) => { - configurationMap.set(item.property, item.name); - }); - } -}; -const queryDetail = async () => { - const res = await detail(props.data.id); - if (res.success) { - Object.keys(res.result?.configuration || {}).forEach((i)=>{ - if(configurationMap.has(i)){ - configuration.value.push({ - name: configurationMap.get(i), - value: res.result.configuration[i] - }) - } - }) - } -}; -onMounted(() => { - queryPlugin(); - queryPluginConfig(); - queryDetail(); -}); -</script> -<style lang="less" scoped> -.other { - width: 100%; - height: 20px; - display: flex; - align-items: center; - justify-content: center; - .item { - width: 100%; - overflow: hidden; - white-space: nowrap; - text-overflow: ellipsis; - } -} - -.plugin-other { - display: flex; - gap: 16px; - .plugin-id, - .plugin-version { - color: rgba(0, 0, 0, 0.85); - opacity: 0.45; - display: flex; - - .plugin-text { - white-space: nowrap; - } - } - - .plugin-id { - width: 50%; - // .other-content { - // display: flex; - // width: 0; - // flex-grow: 1; - // } - } -} -</style> diff --git a/src/views/link/AccessConfig/Outline/components/SixthKind.vue b/src/views/link/AccessConfig/Outline/components/SixthKind.vue deleted file mode 100644 index 54a1bbd..0000000 --- a/src/views/link/AccessConfig/Outline/components/SixthKind.vue +++ /dev/null @@ -1,67 +0,0 @@ -<template> - <div> - <a-descriptions bordered :column="1"> - <a-descriptions-item :label="i.label" v-for="i in configuration"> - {{ i.label==='集群'? i.value ? '共享配置' : '独立配置' : i.value }} - </a-descriptions-item> - </a-descriptions> - </div> -</template> - -<script setup> -import { detail } from '@/api/link/accessConfig'; -const props = defineProps({ - data: { - type: Object, - }, -}); -const configuration = ref([]); -const configurationMap = new Map(); -configurationMap.set('domain', 'SIP 域'); -configurationMap.set('sipId', 'SIP ID'); -configurationMap.set('shareCluster', '集群'); -configurationMap.set('host', 'SIP 地址'); -configurationMap.set('publicHost', '公网 Host'); -const queryDetail = async () => { - const res = await detail(props.data.id); - if (res.success) { - Object.keys(res.result.configuration || {}).forEach((i) => { - if (configurationMap.has(i)) { - configuration.value.push({ - label: configurationMap.get(i), - value: res.result.configuration[i], - }); - } - if (i === 'hostPort') { - Object.keys(res.result.configuration[i] || {}).forEach( - (item) => { - if (item === 'host') { - configuration.value.push({ - label: configurationMap.get(item), - value: - res.result.configuration.hostPort.host + - ':' + - res.result.configuration.hostPort.port, - }); - } else if (item === 'publicHost') { - configuration.value.push({ - label: configurationMap.get(item), - value: - res.result.configuration.hostPort - .publicHost + - ':' + - res.result.configuration.hostPort - .publicPort, - }); - } - }, - ); - } - }); - } -}; -onMounted(() => { - queryDetail(); -}); -</script> -<style lang="less" scoped></style> diff --git a/src/views/link/AccessConfig/Outline/components/ThirdKind.vue b/src/views/link/AccessConfig/Outline/components/ThirdKind.vue deleted file mode 100644 index 188f1da..0000000 --- a/src/views/link/AccessConfig/Outline/components/ThirdKind.vue +++ /dev/null @@ -1,84 +0,0 @@ -<template> - <div> - <TitleComponent - data="接入配置" - v-if="configuration.length" - ></TitleComponent> - <a-descriptions bordered :column="1" v-if="configuration.length"> - <a-descriptions-item v-for="i in configuration" :label="i.label">{{ - i.value - }}</a-descriptions-item> - </a-descriptions> - <TitleComponent - v-if="protocol" - data="消息协议" - style="margin-top: 20px" - ></TitleComponent> - <AccessCard v-if="protocol" :data="{ ...protocol, type: 'protocol' }"> - </AccessCard> - </div> -</template> - -<script setup> -import { detail, getProtocolList } from '@/api/link/accessConfig'; -import { ProtocolMapping } from '../../data'; -import AccessCard from '../../components/AccessCard/index.vue'; -const props = defineProps({ - data: { - type: Object, - }, -}); -const configuration = ref([]); -const protocol = ref(); -const CtwingMap = new Map(); -CtwingMap.set('apiAddress', '接口地址'); -CtwingMap.set('appKey', 'appKey'); -CtwingMap.set('appSecret', 'appSecret'); -CtwingMap.set('description', '说明'); -const OneNetMap = new Map(); -OneNetMap.set('apiAddress', '接口地址'); -OneNetMap.set('apiKey', 'apiKey'); -OneNetMap.set('validateToken', '通知Token'); -OneNetMap.set('aesKey', 'aesKey'); -OneNetMap.set('description', '说明'); -const queryDetail = async () => { - const res = await detail(props.data.id); - if (res.success) { - if (props.data.channel === 'Ctwing') { - Object.keys(res.result?.configuration || {}).forEach((i) => { - if (CtwingMap.has(i)) { - configuration.value.push({ - label: CtwingMap.get(i), - value: res.result.configuration[i] || '--', - }); - } - }); - } else if (props.data.channel === 'OneNet') { - Object.keys(res.result?.configuration || {}).forEach((i) => { - if (OneNetMap.has(i)) { - configuration.value.push({ - label: OneNetMap.get(i), - value: res.result.configuration[i] || '--', - }); - } - }); - } - } -}; -const queryProcotol = async () => { - const res = await getProtocolList(ProtocolMapping.get(props.data.channel), { - 'sorts[0].name': 'createTime', - 'sorts[0].order': 'desc', - }); - if (res.success) { - protocol.value = res.result.find((i) => { - return i.id === props.data.protocol; - }); - } -}; -onMounted(() => { - queryDetail(); - queryProcotol(); -}); -</script> -<style lang="less" scoped></style> diff --git a/src/views/link/AccessConfig/Outline/index.vue b/src/views/link/AccessConfig/Outline/index.vue deleted file mode 100644 index 1164d15..0000000 --- a/src/views/link/AccessConfig/Outline/index.vue +++ /dev/null @@ -1,85 +0,0 @@ -<template> - <a-drawer - visible - :closable="false" - @close="$emit('closeDrawer')" - :width="700" - > - <template #title> - <div class="drawerTitle"> - <Ellipsis style="height: 22px; width: calc(100% - 50px)" - ><div>{{ data?.name }}</div></Ellipsis - > - <Ellipsis style="height: 22px; max-width: 650px" - ><div class="description"> - {{ data?.description }} - </div></Ellipsis - > - </div> - </template> - <FirstKind - v-if=" - [ - 'mqtt-server-gateway', - 'mqtt-client-gateway', - 'websocket-server', - 'http-server-gateway', - 'coap-server-gateway', - 'tcp-server-gateway', - 'udp-device-gateway', - ].includes(data.provider) - " - :data="data" - /> - <SecondKind - v-if="['plugin_gateway', 'media-plugin'].includes(data.provider)" - :data="data" - /> - <ThirdKind - v-if="['Ctwing', 'OneNet','OneNet-platform'].includes(data.provider)" - :data="data" - /> - <FourthKind - v-if=" - [ - 'edge-child-device', - 'fixed-media', - 'opc-ua', - 'modbus-tcp', - 'collector-gateway', - 'onvif', - ].includes(data.channel) - " - :data="data" - /> - <FifthKind - v-if=" - ['official-edge-gateway', 'child-device'].includes( - data.provider, - ) - " - :data="data" - /> - <SixthKind v-if="data.provider === 'gb28181-2016'" :data="data" /> - </a-drawer> -</template> - -<script setup> -import FirstKind from './components/FirstKind.vue'; -import SecondKind from './components/SecondKind.vue'; -import ThirdKind from './components/ThirdKind.vue'; -import FourthKind from './components/FourthKind.vue'; -import FifthKind from './components/FifthKind.vue'; -import SixthKind from './components/SixthKind.vue'; -const props = defineProps({ - data: { - type: Object, - }, -}); -const emit = defineEmits(['closeDrawer']); -</script> -<style lang="less" scoped> -.description { - font-size: 12px; -} -</style> diff --git a/src/views/location/AssociationConfig/components/EditUserDialog.vue b/src/views/location/AssociationConfig/components/EditUserDialog.vue new file mode 100644 index 0000000..b62e857 --- /dev/null +++ b/src/views/location/AssociationConfig/components/EditUserDialog.vue @@ -0,0 +1,326 @@ +<template> + <j-modal visible :title="dialogTitle" :maskClosable="false" width="675px" @ok="confirm" + @cancel="emits('update:visible', false)" class="edit-dialog-container" :confirmLoading="loading" cancelText="取消" + okText="确定"> + <j-form ref="formRef" :model="form.data" layout="vertical"> + <j-row :gutter="24"> + <j-col :span="24"> + <j-form-item name="lineId" label="线路" :rules="[ + { required: form.data.lineId, message: '请选择线路' }, + ]"> + <j-tree-select v-model:value="form.data.lineId" show-search + placeholder="请选择线路" :tree-data="form.lineOptions" + :fieldNames="{ label: 'nodeName', value: 'id', children: 'children' }" + :filterTreeNode="(v: string, node: any) => filterSelectNode(v, node, 'nodeName')" + @change="(value, label, extra) =>form.saveName(label[0],'lineName')"> + <template #title="{ nodeName }"> + <div style="width: calc(100% - 10px) "> + <Ellipsis>{{ nodeName }}</Ellipsis> + </div> + </template> + </j-tree-select> + </j-form-item> + </j-col> + </j-row> + + <j-row :gutter="24"> + <j-col :span="12"> + <j-form-item name="preMonitorId" label="前监测点" class="flex" :rules="[ + { required: form.data.preMonitorId, message: '请选择前监测点' }, + ]"> + <j-tree-select v-model:value="form.data.preMonitorId" show-search + placeholder="请选择前监测点" :tree-data="form.preMonitorOptions" + :fieldNames="{ label: 'nodeName', value: 'id', children: 'children' }" + :filterTreeNode="(v: string, node: any) => filterSelectNode(v, node, 'nodeName')" + @change="(value, label, extra) => { form.handleSubMonitorOptions(value); form.saveName(label[0],'preMonitorName','preDeviceId',extra)}" + > + <template #title="{ nodeName }"> + <div style="width: calc(100% - 10px) "> + <Ellipsis>{{ nodeName }}</Ellipsis> + </div> + </template> + </j-tree-select> + </j-form-item> + </j-col> + <j-col :span="12"> + <j-form-item name="subMonitorId" label="后监测点" class="flex" :rules="[ + { required: form.data.subMonitorId, message: '请选择前监测点' }]"> + <j-tree-select v-model:value="form.data.subMonitorId" show-search + style="width: calc(100% - 40px)" placeholder="请选择后监测点" :tree-data="form.subMonitorOptions" + :disabled="!form.data.preMonitorId" :fieldNames="{ label: 'nodeName', value: 'id' }" + :filterTreeNode="(v: string, node: any) => filterSelectNode(v, node, 'nodeName')" + @change="(value, label, extra) => form.saveName(label[0],'subMonitorName','subDeviceId',extra)"> + > + <template #title="{ nodeName }"> + {{ nodeName }} + </template> + </j-tree-select> + </j-form-item> + </j-col> + </j-row> + <j-row :gutter="24"> + <j-col :span="12"> + <j-form-item name="speed" label="速度(m/ns)" :rules="[ + { + required: form.data.speed, + message: '请输入正确的手机号', + }, + ]"> + <j-input v-model:value="form.data.speed" placeholder="请输入速度" :maxlength="64" /> + </j-form-item> + </j-col> + <j-col :span="12"> + <j-form-item name="distance" label="长度(m)" :rules="[ + { + required: form.data.distance, + message: '请输入正确的邮箱', + }, + ]"> + <j-input v-model:value="form.data.distance" placeholder="请输入长度" :maxlength="64" /> + </j-form-item> + </j-col> + </j-row> + </j-form> + </j-modal> +</template> + +<script setup lang="ts"> +import PermissionButton from '@/components/PermissionButton/index.vue'; +import { FormInstance } from 'ant-design-vue'; + +import { findDefaultTree } from '@/api/monitor'; +import { getAssociateConfigList_api, updateAssociateConfig_api, addAssociateConfig_api } from '@/api/location/associateConfig'; + +import { Rule } from 'ant-design-vue/es/form'; +import { DefaultOptionType } from 'ant-design-vue/es/vc-tree-select/TreeSelect'; +import { AxiosResponse } from 'axios'; +import { passwordRegEx } from '@/utils/validate'; +import { filterSelectNode, onlyMessage } from '@/utils/comm'; +import { uniqBy } from 'lodash-es'; +import { storeToRefs } from 'pinia'; +import { TreeNode } from '../../Site/type'; +const deptPermission = 'system/Department'; +const rolePermission = 'system/Role'; + +const emits = defineEmits(['confirm', 'update:visible']); +const props = defineProps<{ + type: modalType; + data: any; + visible: boolean; +}>(); +// 弹窗相关 +const loading = ref(false); +const dialogTitle = computed(() => { + if (props.type === 'add') return '新增'; + else if (props.type === 'edit') return '编辑'; + else return ''; +}); +const confirm = () => { + loading.value = true; + formRef.value + ?.validate() + .then(() => form.submit()) + .then((resp: any) => { + if (resp.status === 200) { + onlyMessage('操作成功'); + emits('confirm'); + emits('update:visible', false); + } + }) + .finally(() => (loading.value = false)); +}; + +const formRef = ref<FormInstance>(); +// 获取当前路由实例 +const route = useRoute(); +const logicId = route.matched[0].name?.toString() || '1'; + +const form = reactive({ + data: {} as formType, + rules: { + }, + lineOptions: [], + preMonitorOptions: [], + subMonitorOptions: [], + + init: () => { + form.getConfig(); + form.getLineList(); + }, + processNestedArray: (processFn: any, nestedArray: any): any => { + const recurse = (arr: any): any => { + return arr.map(item => { + // 处理当前项 + const processedItem = processFn(item); + // 如果有 children 则递归处理 + if (item.children && Array.isArray(item.children)) { + processedItem.children = recurse(item.children); + } + return processedItem; // 返回处理后的项 + }); + }; + return recurse(nestedArray); + }, + getLineList: async () => { + const { result } = await findDefaultTree(logicId); + form.lineOptions = form.processNestedArray((i) => ({ ...i, disabled: i.nodeType == 1 ? true : false }), result)// result as TreeNode; + form.preMonitorOptions = form.processNestedArray((i) => ({ ...i, disabled: i.nodeType != 1 ? true : false }), result) + form.subMonitorOptions = form.processNestedArray((i) => ({ ...i, disabled: i.nodeType != 1 ? true : false }), result) + }, + handleSubMonitorOptions: (id: string) => { + form.subMonitorOptions = form.processNestedArray( + (i) => ({ ...i, disabled: i.id == id || i.nodeType != 1 ? true : false }), + form.subMonitorOptions + ) + }, + getConfig: () => { + const id = props.data.id || ''; + + if (props.type === 'add') { form.data = {} as formType; } + else if (props.type === 'edit') { + getAssociateConfigList_api(id).then((resp: any) => { + + form.data = { + ...(resp.result as formType), + }; + + nextTick(() => { + formRef.value?.clearValidate(); + }); + }); + } + }, + submit: (): Promise<any> => { + let api: axiosFunType; + let params = {}; + console.log(form.data) + if (props.type === 'add') { + api = addAssociateConfig_api; + params = { + ...form.data, + }; + } else if (props.type === 'edit') { + api = updateAssociateConfig_api; + params = { + id: form.data.id, + ...form.data, + }; + } else return Promise.reject(); + return api(params); + }, + saveName: (nodeName, name ,deviceId,extra) => { + form.data[name] = nodeName; + if (deviceId) { + form.data[deviceId] = extra.triggerNode.props.deviceId || 1; + } + }, +}); +const checkCh = async (_rule: Rule, value: string) => { + if (/[\u4e00-\u9fa5]/.test(value)) return Promise.reject('用户名不能包含中文'); + else return Promise.resolve('') +} +const dealRoleList = (data: any) => { + return data.map((item: any) => { + return { + name: item.groupName, + id: item.groupId, + disabled: true, + children: item?.roles ? item.roles.map((i: any) => { + return { + name: i.name, + id: i.id + } + }) : [] + } + }) +} +// 组织已删除在仍显示在列表中 +// const _departmentOptions = computed(() => { +// return uniqBy([...form.departmentOptions, ...form._departmentOptions], 'id') +// }) + +form.init(); + +interface AxiosResponseRewrite<T = any[]> extends AxiosResponse<T, any> { + result: T; + success: boolean; +} +type axiosFunType = (data: any) => Promise<AxiosResponseRewrite<unknown>>; +type modalType = '' | 'add' | 'edit'; +type formType = { + id?: string; + lineId?: string;//线路ID + lineName?: string;//线路名称 + preDeviceId?: string;//前置设备ID + preMonitorId: string;//前置监测点ID + preMonitorName?: string;//前置监测点名称 + subDeviceId?: string;//后置设备ID + subMonitorId?: string;// 后置监测点ID + subMonitorName?: string;//后置监测点名称 + speed: string;//速度 + distance?: string;//距离 + calibrationDistance?: string;//校准距离 + description?: string;//说明 +}; +type dictType = { + id: string; + name: string; + children?: dictType; +}; +type optionType = { + value: string; + label: string; +}; +</script> + +<style lang="less" scoped> +.edit-dialog-container { + .ant-form-item { + &.flex { + :deep(.ant-form-item-control-input-content) { + display: flex; + + .ant-select { + flex: 1; + } + + .ant-tooltip-disabled-compatible-wrapper { + .ant-btn { + color: rgba(0, 0, 0, 0.25); + border-color: #d9d9d9; + background: #f5f5f5; + text-shadow: none; + box-shadow: none; + } + } + + .ant-btn { + width: 32px; + height: 32px; + border: 1px solid #1d39c4; + color: #1d39c4; + display: flex; + align-items: center; + justify-content: center; + margin-left: 8px; + cursor: pointer; + } + } + } + } +} + +.formName { + margin-bottom: 10px; + font-size: 16px; + + &::before { + width: 2px; + background-color: rgb(184, 184, 184); + display: inline-block; + height: 13px; + margin-right: 3px; + content: '' + } +} +</style> diff --git a/src/views/location/AssociationConfig/components/Progress.vue b/src/views/location/AssociationConfig/components/Progress.vue new file mode 100644 index 0000000..bc14ed0 --- /dev/null +++ b/src/views/location/AssociationConfig/components/Progress.vue @@ -0,0 +1,76 @@ +<template> + <div class="progress-container"> + <div class="value" :style="valueStyle"></div> + <div + class="split" + v-for="leftValue in valueArr" + :style="{ left: leftValue + '%' }" + ></div> + </div> +</template> + +<script setup lang="ts"> +const props = withDefaults( + defineProps<{ + steps?: number; + strokeColor?: + | { + from: string; + to: string; + } + | string; + percent: number; + }>(), + { + steps: 1, + strokeColor: '#108ee9', + }, +); + +const valueStyle = computed(() => { + let background = ''; + if (typeof props.strokeColor === 'string') { + background = props.strokeColor; + } else { + background = `-webkit-linear-gradient( + left, + ${props.strokeColor.from}, + ${props.strokeColor.to} + )`; + } + + return { + background, + 'clip-path': `polygon(0px 0px, ${props.percent}% 0px, ${props.percent}% 100%, 0px 100%)`, + }; +}); + +const valueArr = computed(() => { + const result = []; + for (let i = 1; i < props.steps; i++) result.push((100 / props.steps) * i); + return result; +}); +</script> + +<style lang="less" scoped> +.progress-container { + width: 100%; + position: relative; + background-color: #e0e0e0; + height: 8px; + margin: 3px 0; + + .split { + position: absolute; + top: 0; + width: 1px; + height: 100%; + background-color: #fff; + } + .value { + transition: all 0.35s ease-in-out 0s; + height: 100%; + position: relative; + } +} +</style> diff --git a/src/views/location/AssociationConfig/index.vue b/src/views/location/AssociationConfig/index.vue index a3cc532..f791266 100644 --- a/src/views/location/AssociationConfig/index.vue +++ b/src/views/location/AssociationConfig/index.vue @@ -1,13 +1,326 @@ <template> <page-container> + <div class="user-container"> + <pro-search + :columns="columns" + target="system-user" + @search="handleParams" + /> + <FullPage> + <j-pro-table + ref="tableRef" + :columns="columns" + :request="getAssociateConfigListPage_api" + model="TABLE" + :params="queryParams" + :defaultParams="{ + sorts: [ + { name: 'createTime', order: 'desc' }, + ], + }" + > + <template #headerTitle> + <PermissionButton + :hasPermission="`${permission}:add`" + type="primary" + @click="table.openDialog('add')" + > + <AIcon type="PlusOutlined" />新增 + </PermissionButton> + </template> + <template #type="slotProps"> + {{ slotProps.type?.name }} + </template> + <template #roleList="slotProps"> + <j-ellipsis> + {{ + slotProps?.roleList + ?.map((item) => { + return item.name; + }) + .join(',') + }} + </j-ellipsis> + </template> + <template #action="slotProps"> + <j-space :size="16"> + <PermissionButton + :hasPermission="`${permission}:update`" + type="link" + :tooltip="{ + title: '编辑', + }" + @click="table.openDialog('edit', slotProps)" + > + <AIcon type="EditOutlined" /> + </PermissionButton> + <PermissionButton + type="link" + :hasPermission="`${permission}:delete`" + :tooltip="{ + title: '删除', + }" + :popConfirm="{ + title: '确认删除?', + onConfirm: () => + table.clickDel(slotProps.id), + }" + :disabled="slotProps.status" + > + <AIcon type="DeleteOutlined" /> + </PermissionButton> + </j-space> + </template> + </j-pro-table> + </FullPage> + <EditUserDialog + v-if="dialog.visible" + :type="dialog.type" + v-model:visible="dialog.visible" + :data="dialog.selectItem" + @confirm="table.refresh" + /> + </div> </page-container> </template> -<script setup lang="ts"> +<script setup lang="ts" name="UserMange"> +import PermissionButton from '@/components/PermissionButton/index.vue'; +import EditUserDialog from './components/EditUserDialog.vue'; +import { + getUserType_api, + getUserList_api, + changeUserStatus_api, + deleteUser_api, + queryRole_api, +} from '@/api/system/user'; -</script> +import { + getAssociateConfigList_api, + getAssociateConfigListPage_api, + delteAssociateConfig_api, +} from '@/api/location/associateConfig'; +import { findDefaultTree } from '@/api/monitor'; +import { onlyMessage } from '@/utils/comm'; + +const permission = 'system/User'; + +const columns = [ + { + title: '线路', + dataIndex: 'lineName', + key: 'lineName', + ellipsis: true, + search: { + type: 'string', + }, + }, + { + title: '前监测点', + dataIndex: 'preMonitorName', + key: 'preMonitorName', + ellipsis: true, + search: { + type: 'string', + }, + }, + { + title: '前设备号', + dataIndex: 'preDeviceId', + key: 'preDeviceId', + ellipsis: true, + search: { + type: 'string', + }, + scopedSlots: true, + }, + { + title: '后监测点', + dataIndex: 'subMonitorName', + key: 'subMonitorName', + ellipsis: true, + search: { + type: 'string', + }, + }, + { + title: '后设备号', + dataIndex: 'subDeviceId', + key: 'subDeviceId', + ellipsis: true, + search: { + type: 'string', + }, + scopedSlots: true, + }, + { + title: '速度(m/ns)', + dataIndex: 'speed', + key: 'speed', + ellipsis: true, + search: { + type: 'string', + }, + scopedSlots: true, + }, + { + title: '长度(m)', + dataIndex: 'distance', + key: 'distance', + ellipsis: true, + search: { + type: 'string', + }, + scopedSlots: true, + }, + { + title: '操作', + dataIndex: 'action', + key: 'action', + fixed: 'right', + width: 200, + scopedSlots: true, + }, +]; +const queryParams = ref({}); + +const tableRef = ref<Record<string, any>>({}); // 表格实例 +const table = { + // 打开编辑弹窗 + openDialog: (type: modalType, row?: any) => { + dialog.selectItem = { ...(row || {}) }; + dialog.type = type; + dialog.visible = true; + }, + changeStatus: ({ id, status }: any) => { + const params = { + status: status === 0 ? 1 : 0, + id, + }; + changeUserStatus_api(params).then(() => { + onlyMessage('操作成功'); + table.refresh(); + }); + }, + // 删除 + clickDel: (id: string) => { + delteAssociateConfig_api(id).then(() => { + onlyMessage('操作成功'); + table.refresh(); + }); + }, + // 刷新列表 + refresh: () => { + tableRef.value.reload(); + }, +}; + +const dialog = reactive({ + selectItem: {}, + visible: false, + type: '' as modalType, +}); -<style scoped> +type dictType = { + id: string; + name: string; +}; +type modalType = '' | 'add' | 'edit' | 'reset'; + +const handleParams = (params: any) => { + const newParams = (params?.terms as any[])?.map((termsGroupA) => { + let arr: any[] = []; + termsGroupA.terms = termsGroupA.terms.map((termsItem: any) => { + if (termsItem.column === 'id$in-dimension$role') { + let _termType = + termsItem.termType === 'nin' + ? 'not$in' + : termsItem.termType; + termsItem.column = `${termsItem.column}$${_termType}`; + delete termsItem.termType; + } + if (['telephone', 'email'].includes(termsItem.column)) { + return { + column: 'id$user-detail', + value: [termsItem], + }; + } + if ( + ['type'].includes(termsItem.column) && + termsItem.value === 'other' + ) { + arr = [ + { + ...termsItem, + type: 'or', + termType: 'isnull', + value: 1, + }, + { + ...termsItem, + type: 'or', + termType: 'empty', + value: 1, + }, + ]; + } + if (termsItem.column === 'roleList') { + if ( + termsItem.termType === 'eq' || + termsItem.termType === 'in' + ) { + return { + column: 'id$in-dimension$role', + type: termsItem.type, + value: termsItem.value, + }; + } else { + return { + column: 'id$in-dimension$role$not', + type: termsItem.type, + value: termsItem.value, + }; + } + } + if(termsItem.column === 'roleList'){ + if(termsItem.termType === 'eq' || termsItem.termType === 'in'){ + return { + column: 'id$in-dimension$role', + type: termsItem.type, + value: termsItem.value + } + }else{ + return { + column: 'id$in-dimension$role$not', + type: termsItem.type, + value: termsItem.value + } + } + } + return termsItem; + }); + + if (arr.length) { + termsGroupA.terms = [...termsGroupA.terms, ...arr]; + } + + return termsGroupA; + }); + queryParams.value = { terms: newParams || [] }; +}; +</script> -</style> \ No newline at end of file +<style lang="less" scoped> +.user-container { + :deep(.ant-table-tbody) { + .ant-table-cell { + .ant-space-item { + .ant-btn-link { + padding: 0; + } + } + } + } +} +</style> diff --git a/src/views/location/Site/Detail/Permiss/index.vue b/src/views/location/Site/Detail/Permiss/index.vue new file mode 100644 index 0000000..519c1f5 --- /dev/null +++ b/src/views/location/Site/Detail/Permiss/index.vue @@ -0,0 +1,153 @@ +<template> + <div class="role-permiss-container"> + <!-- <section class="card"> + <h5>基本信息</h5> + <j-form + ref="formRef" + class="basic-form" + :model="form.data" + layout="vertical" + > + <j-form-item + name="name" + label="名称" + :rules="[ + { required: true, message: '请输入名称' }, + { max: 64, message: '最多可输入64个字符' }, + ]" + > + <j-input + v-model:value="form.data.name" + placeholder="请输入角色名称" + /> + </j-form-item> + <j-form-item label="说明"> + <j-textarea + v-model:value="form.data.description" + placeholder="请输入说明" + :maxlength="200" + show-count + /> + </j-form-item> + </j-form> + </section> --> + + <section class="card"> + <h5>权限分配</h5> + <PermissTree v-model:select-items="form.menus" /> + + <j-button + type="primary" + :confirm-loading="form.loading" + @click="form.clickSave" + style="margin-top: 24px" + >保存</j-button + > + </section> + </div> +</template> + +<script setup lang="ts" name="RolePermiss"> +import { FormInstance } from 'ant-design-vue'; +import PermissTree from '../components/PermissTree.vue'; +import { useMenuStore } from '@/store/menu'; +// import { USER_CENTER_MENU_DATA } from '@/views/init-home/data/baseMenu' + +import { + getRoleDetails_api, + updateRole_api, + editRole_api, + updatePrimissTree_api, clearPrimissTree_api, +} from '@/api/system/role'; +import { onlyMessage } from '@/utils/comm'; + +const { jumpPage } = useMenuStore(); +const route = useRoute(); +const router = useRouter(); +const roleId = route.params.id as string; + +// 表单相关 +const formRef = ref<FormInstance>(); +const form = reactive({ + loading: false, + // data: { + // name: '', + // description: '', + // }, + menus: [], // USER_CENTER_MENU_DATA + // getForm: () => { + // getRoleDetails_api(roleId).then((resp) => { + // if (resp.status) { + // form.data = resp.result; + // } + // }); + // }, + clickSave: () => { + // formRef.value?.validate().then(() => { + // const updateRole = editRole_api(roleId, form.data); + if(form.menus?.length > 0){ + const updateTree = updatePrimissTree_api(roleId, { + menus: form.menus, + }); + Promise.all([ updateTree]).then((resp) => { + onlyMessage('操作成功'); + // jumpPage(`system/Role`); + }) + } else { + clearPrimissTree_api(roleId).then(resp => { + if(resp.success){ + onlyMessage('操作成功'); + } + }) + } + // }); + }, +}); + +// form.getForm(); +</script> + +<style lang="less" scoped> +.role-permiss-container { + .card { + margin-bottom: 24px; + background-color: #fff; + padding: 24px; + + h5 { + position: relative; + display: flex; + align-items: center; + margin-bottom: 20px; + padding: 4px 0 4px 12px; + font-weight: bold; + font-size: 16px; + + &::before { + position: absolute; + top: 5px px; + left: 0; + width: 4px; + height: calc(100% - 10px); + background-color: #1d39c4; + border-radius: 2px; + content: ' '; + } + } + + .basic-form { + :deep(.ant-form-item-required) { + padding-right: 12px; + + &::before { + right: 0; + } + } + .ant-form-item { + display: block; + width: 60%; + } + } + } +} +</style> diff --git a/src/views/location/Site/Detail/User/index.d.ts b/src/views/location/Site/Detail/User/index.d.ts new file mode 100644 index 0000000..a9bdcd4 --- /dev/null +++ b/src/views/location/Site/Detail/User/index.d.ts @@ -0,0 +1,23 @@ +type columnsType = { + title: string, + dataIndex: string, + key: string, + scopedSlots?:boolean +} + +export type queryType = { + columns: columnsType[], + params?: object +} + +export type tableType = { + columns: columnsType[], + tableData: any[], + refresh: Function, + clickAdd: Function, + clickUnBind: Function, + unbind: Function, + _selectedRowKeys?: string[], + onSelectChange?: Function, + cancelSelect?: Function +} \ No newline at end of file diff --git a/src/views/location/Site/Detail/User/index.vue b/src/views/location/Site/Detail/User/index.vue new file mode 100644 index 0000000..2347a8a --- /dev/null +++ b/src/views/location/Site/Detail/User/index.vue @@ -0,0 +1,234 @@ +<template> + <div class="role-user-container"> + <pro-search + :columns="columns" + target="system-role-user" + @search="(params:any)=>queryParams = {...params}" + /> + + <j-pro-table + ref="tableRef" + :columns="columns" + :request="table.getList" + model="TABLE" + :params="queryParams" + :rowSelection="{ + selectedRowKeys: selectedRowKeys, + onSelect: onSelectChange, + onSelectAll: selectAll, + onSelectNone: () => (selectedRowKeys = []), + }" + size="small" + > + <template #headerTitle> + <j-space> + <j-button type="primary" @click="dialogVisible = true"> + <AIcon type="PlusOutlined" />新增 + </j-button> + <PermissionButton + :popConfirm="{ + title: `是否批量解除绑定`, + placement: 'topRight', + onConfirm: () => table.unbind(), + }" + > + <AIcon type="DisconnectOutlined" />批量解绑 + </PermissionButton> + </j-space> + </template> + + <template #status="slotProps"> + <BadgeStatus + :status="slotProps.status" + :text="slotProps.status ? '正常' : '禁用'" + :statusNames="{ + 1: 'success', + 0: 'error', + }" + ></BadgeStatus> + </template> + <template #createTime="slotProps"> + {{ dayjs(slotProps.createTime).format('YYYY-MM-DD HH:mm:ss') }} + </template> + + <template #action="slotProps"> + <j-space :size="16"> + <PermissionButton + type="link" + :tooltip="{ title: '解绑' }" + :pop-confirm="{ + title: `确认解绑`, + onConfirm: () => table.unbind([slotProps.id]), + }" + > + <AIcon type="DisconnectOutlined" /> + </PermissionButton> + </j-space> + </template> + </j-pro-table> + + <AddUserDialog + v-if="dialogVisible" + v-model:visible="dialogVisible" + :role-id="roleId" + @refresh="table.refresh" + /> + </div> +</template> + +<script setup lang="ts" name="RoleUser"> +import PermissionButton from '@/components/PermissionButton/index.vue'; +import AddUserDialog from '../components/AddUserDialog.vue'; +import { getUserByRole_api, unbindUser_api } from '@/api/system/role'; +import dayjs from 'dayjs'; +import { onlyMessage } from '@/utils/comm'; + +const roleId = useRoute().params.id as string; + +const columns = [ + { + title: '姓名', + dataIndex: 'name', + key: 'name', + search: { + type: 'string', + }, + }, + { + title: '用户名', + dataIndex: 'username', + key: 'username', + search: { + type: 'string', + }, + }, + { + title: '创建时间', + dataIndex: 'createTime', + key: 'createTime', + search: { + type: 'date', + }, + scopedSlots: true, + }, + { + title: '状态', + dataIndex: 'status', + key: 'status', + search: { + type: 'select', + options: [ + { + label: '正常', + value: 1, + }, + { + label: '禁用', + value: 0, + }, + ], + }, + scopedSlots: true, + }, + { + title: '操作', + dataIndex: 'action', + key: 'action', + width: '200px', + scopedSlots: true, + }, +]; +const queryParams = ref({}); + +const tableRef = ref<Record<string, any>>({}); +const selectedRowKeys = ref<string[]>([]); +const table = { + getList: (oParams: any) => { + const params = { + ...oParams, + terms: [ + { + terms: [ + { + column: 'id$in-dimension$role', + value: roleId, + }, + ], + }, + ], + }; + if (oParams.terms[0]) + params.terms.unshift({ + terms: oParams.terms[0].terms, + }); + return getUserByRole_api(params); + }, + // 批量解绑 + unbind: (ids?: string[]) => { + const data = ids ? ids : selectedRowKeys.value; + if (!data.length) { + onlyMessage('请勾选数据', 'warning'); + return; + } + unbindUser_api(roleId, data).then((resp) => { + if (resp.status === 200) { + onlyMessage('操作成功'); + table.refresh(); + } + }); + }, + // 刷新表格 + refresh: () => { + tableRef.value.reload(); + selectedRowKeys.value = []; + }, +}; + +const onSelectChange = (item: any, state: boolean) => { + const arr = new Set(selectedRowKeys.value); + if (state) { + arr.add(item.id); + } else { + arr.delete(item.id); + } + selectedRowKeys.value = [...arr.values()]; +}; + +const selectAll = (selected: Boolean, selectedRows: any, changeRows: any) => { + if (selected) { + changeRows.map((i: any) => { + if (!selectedRowKeys.value.includes(i.id)) { + selectedRowKeys.value.push(i.id); + } + }); + } else { + const arr = changeRows.map((item: any) => item.id); + const _ids: string[] = []; + selectedRowKeys.value.map((i: any) => { + if (!arr.includes(i)) { + _ids.push(i); + } + }); + selectedRowKeys.value = _ids; + } +}; + +// 弹窗相关 +const dialogVisible = ref(false); +</script> + +<style lang="less" scoped> +.role-user-container { + background-color: #fff; + + :deep(.ant-table-tbody) { + .ant-table-cell { + .ant-space-item { + .ant-btn-link { + padding: 0; + } + } + } + } +} +</style> diff --git a/src/views/location/Site/Detail/components/AddUserDialog.vue b/src/views/location/Site/Detail/components/AddUserDialog.vue new file mode 100644 index 0000000..f7d2930 --- /dev/null +++ b/src/views/location/Site/Detail/components/AddUserDialog.vue @@ -0,0 +1,123 @@ +<template> + <j-modal visible title="新增" width="1000px" @ok="confirm" @cancel="emits('update:visible', false)"> + <!-- <j-advanced-search + :columns="columns" + type="simple" + @search="(params:any)=>queryParams = {...params}" + /> --> + <pro-search :columns="columns" target="simple" @search="(params: any) => queryParams = { ...params }" /> + + <j-pro-table ref="tableRef" :columns="columns" :request="getUserList" model="TABLE" :params="queryParams" + :rowSelection="{ + selectedRowKeys: selectedRowKeys, + onSelect: changeSelect, + onSelectAll: selectAll, + onSelectNone: ()=>selectedRowKeys = [] + }"> + </j-pro-table> + </j-modal> +</template> + +<script setup lang="ts"> +import { getUserByRole_api, bindUser_api } from '@/api/system/role'; +import { onlyMessage } from '@/utils/comm'; + +const emits = defineEmits(['refresh', 'update:visible']); +const props = defineProps<{ + visible: boolean; + roleId: string; +}>(); + +const columns = [ + { + title: '姓名', + dataIndex: 'name', + key: 'name', + search: { + type: 'string', + }, + }, + { + title: '用户名', + dataIndex: 'username', + key: 'username', + search: { + type: 'string', + }, + }, +]; +const queryParams = ref({}); + +const selectedRowKeys = ref<any>([]); +const getUserList = (oParams: any) => { + const params = { + ...oParams, + sorts: [{ name: 'createTime', order: 'desc' }], + terms: [ + { + terms: [ + { + column: 'id$in-dimension$role$not', + value: props.roleId, + }, + { + column: 'username', + value: 'admin', + termType: 'not', + type: 'and' + } + ], + }, + ], + }; + if (oParams.terms[0]) + params.terms.unshift({ + terms: oParams.terms[0].terms, + }); + return getUserByRole_api(params); +}; + +const confirm = () => { + if (selectedRowKeys.value.length < 1) { + onlyMessage('请至少选择一项', 'error'); + } else { + bindUser_api(props.roleId, selectedRowKeys.value).then((resp) => { + if (resp.status === 200) { + onlyMessage('操作成功'); + emits('refresh'); + emits('update:visible', false); + } + }); + } +}; +const changeSelect = (item: any, state: boolean) => { + const arr = new Set(selectedRowKeys.value); + console.log(item, state); + if (state) { + arr.add(item.id); + } else { + arr.delete(item.id); + } + selectedRowKeys.value = [...arr.values()]; +}; + +const selectAll = (selected: Boolean, selectedRows: any,changeRows:any) => { + if (selected) { + changeRows.map((i: any) => { + if (!selectedRowKeys.value.includes(i.id)) { + selectedRowKeys.value.push(i.id) + } + }) + } else { + const arr = changeRows.map((item: any) => item.id) + const _ids: string[] = []; + selectedRowKeys.value.map((i: any) => { + if (!arr.includes(i)) { + _ids.push(i) + } + }) + selectedRowKeys.value = _ids + + } +} +</script> diff --git a/src/views/location/Site/Detail/components/PermissTree.vue b/src/views/location/Site/Detail/components/PermissTree.vue new file mode 100644 index 0000000..0f40921 --- /dev/null +++ b/src/views/location/Site/Detail/components/PermissTree.vue @@ -0,0 +1,548 @@ +<template> + <div class="permiss-tree-container"> + <j-table + :columns="columns" + :data-source="tableData" + :pagination="false" + :rowKey="'id'" + :scroll="{ y: '500px' }" + ref="treeRef" + > + <!-- 表头 --> + <template #headerCell="{ column }"> + <div v-if="column.key === 'menu'"> + <j-checkbox + v-model:checked="selectedAll" + :indeterminate="indeterminate" + @change="selectAllChange" + >菜单权限</j-checkbox + > + </div> + <div v-else-if="column.key === 'data'"> + <span style="">数据权限</span> + <j-tooltip> + <template #title + >勾选任意数据权限均能看到自己创建的数据权限</template + > + <AIcon type="QuestionCircleOutlined" /> + </j-tooltip> + <j-checkbox + v-model:checked="bulkShow" + @change="bulkValue = ''" + style="margin-left: 10px" + >批量设置</j-checkbox + > + <j-select + v-show="bulkShow" + v-model:value="bulkValue" + :size="'middle'" + style="width: 200px" + :options="bulkOptions" + @change="bulkChange" + placeholder="请选择" + ></j-select> + </div> + <div v-else> + <span>{{ column.title }}</span> + </div> + </template> + <!-- 表格内容 --> + <template #bodyCell="{ column, record }"> + <div :id="record.id"></div> + <div v-if="column.key === 'menu'"> + <j-checkbox + v-model:checked="record.granted" + :indeterminate="record.indeterminate" + @change="menuChange(record, true)" + >{{ record.name }}</j-checkbox + > + <!-- :disabled='record.code === USER_CENTER_MENU_CODE' --> + </div> + + <div v-else-if="column.key === 'action'"> + <div v-if="record.buttons && record.buttons.length > 0"> + <j-checkbox + v-for="button in record.buttons" + v-model:checked="button.granted" + @change="actionChange(record)" + :key="button.id" + >{{ button.name }}</j-checkbox + > + <!-- :disabled='[USER_CENTER_MENU_BUTTON_CODE].includes(button.id)' --> + </div> + </div> + + <div v-else-if="column.key === 'data'"> + <span v-if="record.accessSupport === undefined"> + 不支持数据权限配置,默认可查看全部数据 + </span> + <div v-else-if="record.accessSupport.value === 'support'"> + <j-radio-group + v-model:value="record.selectAccesses" + @change="resetBulk" + > + <j-radio + :value="asset.supportId" + v-for="asset in record.assetAccesses" + :key="asset.name" + >{{ asset.name }}</j-radio + > + </j-radio-group> + </div> + <span + v-else-if=" + record.accessSupport.value === 'indirect' || + record.accessSupport.value === 'unsupported' + " + >{{ record.accessDescription }}</span + > + </div> + </template> + </j-table> + </div> +</template> + +<script setup lang="ts"> +import { cloneDeep, uniqBy } from 'lodash-es'; +import { getPrimissTree_api } from '@/api/system/role'; +import { getCurrentInstance } from 'vue'; +import { +// USER_CENTER_MENU_BUTTON_CODE, + USER_CENTER_MENU_CODE +} from '@/utils/consts' +import { isNoCommunity } from '@/utils/utils' +import {permissionsGranted, useIndirectMenusMap} from "@/views/system/Role/Detail/components/util"; +import {NotificationSubscriptionCode} from "@/router/menu"; + +const emits = defineEmits(['update:selectItems']); +const route = useRoute(); +const props = defineProps({ + selectItems: Array, +}); +const treeRef = ref(); +let { ctx: that, proxy } = getCurrentInstance(); +const flatTableData: tableItemType[] = []; // 表格数据的扁平化版本--浅克隆 方便进行对表格数据进行操作 + +const columns = [ + { + title: '菜单权限', + dataIndex: 'menu', + key: 'menu', + width: '260px', + }, + { + title: '操作权限', + dataIndex: 'action', + key: 'action', + width: '260px', + }, +]; + +if(isNoCommunity){ + columns.push({ + title: '数据权限', + dataIndex: 'data', + key: 'data', + width: '50%', + }) +} +const tableData = ref<tableItemType[]>([]); + +// 表头-全选 +const selectedAll = ref<boolean>(false); +const indeterminate = ref<boolean>(false); +const selectAllChange = () => { + flatTableData.forEach((item) => { + item.granted = selectedAll.value; + item.indeterminate = false; + item.buttons?.forEach((button) => { + button.granted = selectedAll.value; + }); + if (selectedAll.value) { + // 全选 + item.selectAccesses = 'creator'; + } else { + // 取消全选 + item.selectAccesses = ''; + } + if (item.accessSupport && item.accessSupport.value === 'support') { + item.assetAccesses?.forEach((asset) => { + if (asset.supportId === item.selectAccesses) { + asset.granted = true; + } else { + asset.granted = false; + } + }); + } + }); + indeterminate.value = false; + emits( + 'update:selectItems', + flatTableData.filter((item) => item.granted), + ); +}; +// 表头-批量设置 +const bulkShow = ref<boolean>(false); +const bulkOptions = ref(); +// const bulkOptions = [ +// { +// label: '全部数据', +// value: 'ignore', +// }, +// { +// label: '所在组织及下级组织', +// value: 'org-include-children', +// }, +// { +// label: '所在组织', +// value: 'org', +// }, +// { +// label: '自己创建的', +// value: 'creator', +// }, +// ]; +const bulkValue = ref<string>(''); +const bulkChange = () => { + if (!bulkValue) return; + flatTableData.forEach((item) => { + if (item.accessSupport && item.accessSupport.value === 'support') { + item.selectAccesses = bulkValue.value; + item.assetAccesses?.forEach((asset) => { + if (asset.supportId === bulkValue.value) { + asset.granted = true; + } else { + asset.granted = false; + } + }); + } + }); + emits( + 'update:selectItems', + flatTableData.filter((item) => item.granted), + ); +}; + +// 重置批量设置 +const resetBulk = () => { + bulkValue.value = ''; + bulkShow.value = false; +}; +// ------------下面为表格内容部分------------------ +const init = () => { + getAllPermiss(); + // 监听权限的修改情况,产生修改后反馈给父组件 + watch( + tableData, + () => { + // 深克隆表格数据的扁平版 因为会做一些改动 该改动只用于反馈给父组件,本组件无需变化 + const selected = cloneDeep(flatTableData).filter( + (item: any) => + // (item.granted && item.parentId) || + (item.indeterminate && item.buttons) + || (item.granted), // 放开个人中心 + ); + + selected.forEach((item) => { + /** + * 如果该项支持设置数据权限,则对其进行数据权限的映射,结束后删除用于映射的源属性, + * 同时清除用于半全选状态的标记 + */ + if ( + item.accessSupport && + item.accessSupport.value === 'support' && + item.selectAccesses + ) { + // item.selectAccesses = bulkValue.value; + item.assetAccesses?.forEach((asset) => { + if (asset.supportId === item.selectAccesses) { + asset.granted = true; + } else { + asset.granted = false; + } + }); + delete item.selectAccesses; + } + delete item.indeterminate; + item.granted = true; + }); + emits('update:selectItems', selected); + }, + { deep: true }, + ); +}; +init(); + +const { PermissionsMap } = useIndirectMenusMap(tableData) + +function getAllPermiss() { + const id = route.params.id as string; + getPrimissTree_api(id).then((resp) => { + const _result = resp.result + // 默认选中个人中心相关设置 + tableData.value = _result.filter((item: { code: string , buttons: any[], granted: boolean}) => { + return (item.code !== NotificationSubscriptionCode) + }); + + treeToSimple(tableData.value); // 表格数据扁平化 + + const selectList = flatTableData.filter((item) => item.granted); // 第一列选中的项 + emits('update:selectItems', selectList); // 选中的项传回父组件 + // 判断是全选/半全选 + if (selectList.length === flatTableData.length) { + selectedAll.value = true; + indeterminate.value = false; + } else if (selectList.length > 0) { + indeterminate.value = true; + selectedAll.value = false; + } + }); +} + +const hasIndirectMenus = (data: any) => { + let indirectMenus = [] + if (data.children) { + const item = data.children.find(item => item.indirectMenus) + indirectMenus = item.indirectMenus + } else if (data?.indirectMenus) { + indirectMenus = data.indirectMenus + } + + if (indirectMenus.length) { + const ids = permissionsGranted(tableData.value) + console.log(ids, indirectMenus) + const inMenu = false + } + +} + +/** + * 菜单权限改变事件 + * @param row 触发的项 + * @param setButtonBool 是否改变对应的操作权限 + */ +function menuChange( + row: tableItemType, + setButtonBool: boolean = true, +): undefined { + console.log('menuChange', row) + // 判断是否需要对子菜单及操作权限进行选择 + // hasIndirectMenus(row) + if (setButtonBool) { + if (row.buttons && row.buttons.length > 0) + row.buttons.forEach((button) => { + button.granted = row.granted; + }); + row.children && setChildrenChecked(row.children, row.granted); + } + // 更新选中状态 + if (row.buttons && row.buttons.length > 0) setStatus(row, 'buttons'); + else setStatus(row, 'children'); + // 更新数据权限 + updateAuthority(row); + // if (row.accessSupport && row.accessSupport.value === 'support') { + // // 如果当前数据权限已有值,且菜单权限没有被选中或被半选 则清空对应的数据权限 + // if (row.selectAccesses && !row.granted && !row.indeterminate) + // row.selectAccesses = ''; + // // 如果当前数据权限没有值,且菜单权限有被选中或者是被半选 则将数据权限变为默认值'creator' + // else if (!row.selectAccesses && (row.granted || row.indeterminate)) + // row.selectAccesses = 'creator'; + // } + + // 更新上层节点的状态 + if (row.parentId) { + // 找到对应的父节点 判断该父节点的选中状态为 全选中/部分选中/未选中 + const parent = flatTableData.find( + (item) => item.id === row.parentId, + ) as tableItemType; + setStatus(parent, 'children'); + // 若该父节点不是根节点 重复此操作以此来确定该父节点的父节点状态 + if (parent.parentId) { + return menuChange(parent, false); + } + } + + // 取消批量设置 + resetBulk(); + + // 改变头部节点状态 + const selectList = flatTableData.filter((item) => item.granted); // 第一列选中的项 + if (selectList.length === flatTableData.length) { + selectedAll.value = true; + indeterminate.value = false; + } else if (selectList.length > 0) { + indeterminate.value = true; + selectedAll.value = false; + } else { + selectedAll.value = false; + indeterminate.value = false; + } + + emits('update:selectItems', selectList); // 选中的项传回父组件 + proxy?.$forceUpdate?.(); +} + +/** + * 更新权限 + */ +const updateAuthority = (row: any) => { + if (row.accessSupport && row.accessSupport.value === 'support') { + // 如果当前数据权限已有值,且菜单权限没有被选中或被半选 则清空对应的数据权限 + if (row.selectAccesses && !row.granted && !row.indeterminate) + row.selectAccesses = ''; + // 如果当前数据权限没有值,且菜单权限有被选中或者是被半选 则将数据权限变为默认值'creator' + else if (!row.selectAccesses && (row.granted || row.indeterminate)) + row.selectAccesses = 'creator'; + } + if (row.children?.length > 0) { + row.children?.forEach((item: any) => { + if (item.accessSupport && item.accessSupport.value === 'support') { + // 如果当前数据权限已有值,且菜单权限没有被选中或被半选 则清空对应的数据权限 + if (item.selectAccesses && !item.granted && !item.indeterminate) + item.selectAccesses = ''; + // 如果当前数据权限没有值,且菜单权限有被选中或者是被半选 则将数据权限变为默认值'creator' + else if ( + !item.selectAccesses && + (item.granted || item.indeterminate) + ) + item.selectAccesses = 'creator'; + } + if (item.children) { + updateAuthority(item.children); + } + }); + } +}; +/** + * 操作权限改变事件 + * @param row 触发的项 + */ +function actionChange(row: tableItemType) { + setStatus(row, 'buttons'); + menuChange(row, false); +} + +/** + * 将树形结构的表格数据拍扁为一个普通数组 + * @param treeData + */ +function treeToSimple(_treeData: tableItemType[]) { + _treeData.forEach((item) => { + // 数据权限回填 + if (item.accessSupport && item.accessSupport.value === 'support') { + const select = + item.assetAccesses?.find((assetItem) => assetItem.granted) || + {}; + item.selectAccesses = select.supportId || ''; + } + if (item.buttons && item.buttons.length > 0) { + setStatus(item, 'buttons'); + } else { + setStatus(item, 'children'); + } + if(item.children){ + treeToSimple(item.children); + } + flatTableData.push(item); + }); + // 根据所有权限, 取assetAccesses并集数据 + if(isNoCommunity){ + let assets: any[] = []; + flatTableData?.forEach((item: any) => { + assets = [...assets, ...item.assetAccesses]; + }); + bulkOptions.value = uniqBy(assets, 'supportId')?.map((m: any) => ({ + label: m.name, + value: m.supportId, + })); + } +} +/** + * 设置子节点的状态 + * @param _children + * @param value + */ +function setChildrenChecked(_children: tableItemType[], value: boolean) { + if (_children.length < 1) return; + _children.forEach((item) => { + item.granted = value; + item.indeterminate = false; + if (item.buttons && item.buttons.length > 0) + item.buttons.forEach((button) => { + button.granted = value; + }); + if (item.assetAccesses?.length > 0) { + item.assetAccesses?.forEach((i) => { + if (i.supportId === 'creator') { + i.granted = true; + } + }); + } + item.children && setChildrenChecked(item.children, value); + }); +} +/** + * 根据taget的prop属性,判断对应的全选状态,头部全选不适用 + * @param target 目标对象 + * @param prop 目标属性 + */ +function setStatus( + target: tableItemType, + prop: 'children' | 'buttons' = 'children', +) { + const childrens = target[prop] as any[]; + if (childrens && childrens instanceof Array) { + // 如果子选项有半全选,则当前节点直接为半全选 + const indeterminateLen = childrens.filter( + (childrens: buttonItemType | tableItemType) => + childrens?.indeterminate, + ).length; + if (indeterminateLen > 0) { + target.granted = false; + target.indeterminate = true; + return; + } + + const selectLen = childrens.filter( + (children: buttonItemType | tableItemType) => children.granted, + ).length; + if (selectLen === childrens.length) { + target.granted = true; + target.indeterminate = false; + } else if (selectLen > 0) { + target.granted = false; + target.indeterminate = true; + } else { + target.granted = false; + target.indeterminate = false; + } + } +} +type buttonItemType = { + supportId: string; + name: string; + granted: boolean; + indeterminate?: boolean; +}; +type tableItemType = { + id: string; + granted: boolean; + name: string; + indeterminate?: boolean; + parentId?: string; + children?: tableItemType[]; + accessSupport?: any; + buttons?: buttonItemType[]; + accessDescription?: string; + selectAccesses?: string; + assetAccesses?: any[]; +}; +</script> + +<style lang="less" scoped> +.permiss-tree-container { + :deep(.ant-checkbox-wrapper) { + margin-left: 0; + } +} +</style> diff --git a/src/views/location/Site/Detail/components/util.ts b/src/views/location/Site/Detail/components/util.ts new file mode 100644 index 0000000..300f76d --- /dev/null +++ b/src/views/location/Site/Detail/components/util.ts @@ -0,0 +1,41 @@ +import type {Ref} from "vue"; +import {watch} from "vue"; + +const handlePermissionsMap = (data: any, map: Map<string, any>, parentId: string[] = []) => { + data.forEach((item: any) => { + if (item.children) { + handlePermissionsMap(item.children,map, parentId.concat(item.id) ) + } else { + map.set(item.id, { + parentIds: parentId, + name: item.name + }) + } + }) +} + +export const useIndirectMenusMap = (tableData: Ref<any[]>) => { + const PermissionsMap = ref<Map<string, any>>(new Map()) + + watch(() => tableData.value, () => { + PermissionsMap.value.clear() + if (tableData.value?.length) { + handlePermissionsMap(tableData.value, PermissionsMap.value) + } + }, { immediate: true}) + + return { + PermissionsMap + } +} + +export const permissionsGranted = (data: any[]) => { + return data.reduce((prev,current) => { + let ids = current.granted ? [...prev, current.id] : prev + if (current.children) { + const _ids = permissionsGranted(current.children) + ids = ids.concat(_ids) + } + return ids + }, []) +} \ No newline at end of file diff --git a/src/views/location/Site/Detail/index.vue b/src/views/location/Site/Detail/index.vue new file mode 100644 index 0000000..c97dbd0 --- /dev/null +++ b/src/views/location/Site/Detail/index.vue @@ -0,0 +1,26 @@ +<template> + <page-container> + <div class="details-container"> + <j-tabs v-model:activeKey="activeKey"> + <j-tab-pane key="1" tab="权限分配"><Permiss /></j-tab-pane> + <j-tab-pane key="2" tab="用户管理"><User /></j-tab-pane> + </j-tabs> + </div> + </page-container> +</template> + +<script setup lang="ts" name="Detail"> +import Permiss from './Permiss/index.vue'; +import User from './User/index.vue'; + +const activeKey = ref('1'); +</script> + +<style lang="less" scoped> +.details-container { + :deep(.ant-tabs-nav-wrap) { + background-color: #fff; + padding-left: 24px; + } +} +</style> diff --git a/src/views/location/Site/RoleLeft/AddGroup.vue b/src/views/location/Site/RoleLeft/AddGroup.vue new file mode 100644 index 0000000..d37603f --- /dev/null +++ b/src/views/location/Site/RoleLeft/AddGroup.vue @@ -0,0 +1,205 @@ +<template> + <j-modal :maskClosable="false" width="650px" :visible="true" :title="!!data?.id ? '编辑站点' : '新增站点'" @ok="handleSave" + @cancel="handleCancel" :confirmLoading="loading"> + <div style="margin-top: 10px"> + <j-form :layout="'vertical'" ref="formRef" :model="modelRef"> + <j-form-item label="站点名称" name="nodeName" :rules="[ + { + required: true, + message: '请输入名称', + }, + { + max: 64, + message: '最多输入64个字符', + }, + ]"> + <j-input v-model:value="modelRef.nodeName" placeholder="请输入名称" /> + </j-form-item> + + + <j-form-item label="背景图(未上传图片则使用地图背景)" name="image"> + <div class="upload-image-warp-back"> + <div class="upload-image-border-back"> + <j-upload name="file" :customRequest="customRequest" :beforeUpload="beforeLogoUpload" + :showUploadList="false" accept="'.jpg', '.png', '.jfif', '.pjp', '.pjpeg', '.jpeg'"> + <div class="upload-image-content-back"> + <div v-if="loading" class="loading-back"> + <AIcon type="LoadingOutlined" /> + </div> + <div class="upload-image" v-if="modelRef.image" :style="modelRef.image + ? `background-image: url(${modelRef.image});` + : '' + "></div> + <div v-if="modelRef.image" class="upload-image-mask"> + 点击修改 + </div> + <div v-else> + <div> + <AIcon type="PlusOutlined" /> + </div> + </div> + </div> + </j-upload> + </div> + </div> + </j-form-item> + </j-form> + </div> + </j-modal> +</template> + +<script lang="ts" setup> +import { isExists, update } from '@/api/device/instance'; +import { getImage, onlyMessage, LocalStore } from '@/utils/comm'; +import { addMonitor } from '@/api/monitor'; +const emit = defineEmits(['close', 'save']); +const props = defineProps({ + data: { + type: Object, + default: undefined, + }, + logicId: { + type: String, + } +}); +const loading = ref<boolean>(false); + +const formRef = ref(); + +const modelRef = reactive({ + id: undefined, + nodeName: '', + image: '', + +}); +const beforeLogoUpload = (file: File) => { + const typeBool = + ['.jpg', '.png', '.jfif', '.pjp', '.pjpeg', '.jpeg'] + .map((m: string) => m.split('.')[1]) + .filter((typeStr) => file.type.includes(typeStr)).length > 0; + const sizeBool = file.size / 1024 / 1024 < 4; + if (!typeBool) { + onlyMessage(`请上传.jpg.png.jfif.pjp.pjpeg.jpeg格式的图片`, 'error'); + } else if (!sizeBool) { + onlyMessage(`图片大小必须小于4M`, 'error'); + } + return typeBool && sizeBool; +}; +/** + 获取file,通过FileReader获取图片的 base64 +*/ +const customRequest = (option: any) => { + const formData = new FormData(); + formData.append("files[]", option.file); + const reader = new FileReader(); + reader.readAsDataURL(option.file); + reader.onloadend = function (e) { + modelRef.image = e.target.result || '' + if (e && e.target && e.target.result) { + option.onSuccess(); + } + }; +} + +watch( + () => props.data, + (newValue) => { + Object.assign(modelRef, newValue); + }, + { immediate: true, deep: true }, +); + +const handleCancel = () => { + emit('close'); + formRef.value.resetFields(); +}; + +const handleSave = () => { + formRef.value + .validate() + .then(async (_data: any) => { + loading.value = true; + const obj = { ..._data }; + obj.logicId = props.logicId; + obj.parentId = 0; + if (!obj.id) { + delete obj.id; + } + const resp = await addMonitor(obj).finally(() => { + loading.value = false; + }); + if (resp.status === 200) { + onlyMessage('操作成功!'); + emit('save'); + emit('close'); + formRef.value.resetFields(); + } + }) + .catch((err: any) => { + console.log('error', err); + }); +}; +</script> +<style lang="less" scoped> +.upload-image-warp-back { + display: flex; + justify-content: flex-start; + + .upload-image-border-back { + position: relative; + overflow: hidden; + border: 1px dashed #d9d9d9; + transition: all 0.3s; + width: 100%; + height: 415px; + + &:hover { + border: 1px dashed #1890ff; + display: flex; + } + + .upload-image-content-back { + align-items: center; + justify-content: center; + position: relative; + display: flex; + flex-direction: column; + width: 610px; + height: 415px; + padding: 8px; + background-color: rgba(0, 0, 0, 0.06); + cursor: pointer; + + .loading-back { + position: absolute; + } + + .upload-image { + width: 100%; + height: 100%; + background-repeat: no-repeat; + background-position: 50%; + background-size: cover; + } + + .upload-image-mask { + align-items: center; + justify-content: center; + position: absolute; + top: 0; + left: 0; + display: none; + width: 100%; + height: 100%; + color: #fff; + font-size: 16px; + background-color: rgba(0, 0, 0, 0.35); + } + + &:hover .upload-image-mask { + display: flex; + } + } + } +} +</style> \ No newline at end of file diff --git a/src/views/location/Site/RoleLeft/BindDevice.vue b/src/views/location/Site/RoleLeft/BindDevice.vue new file mode 100644 index 0000000..27bd247 --- /dev/null +++ b/src/views/location/Site/RoleLeft/BindDevice.vue @@ -0,0 +1,163 @@ +<template> + <j-modal :maskClosable="false" width="650px" :visible="true" :title="'设备绑定'" @ok="handleSave" + @cancel="handleCancel" :confirmLoading="loading"> + <div style="margin-top: 10px"> + </div> + </j-modal> +</template> + +<script lang="ts" setup> +import { isExists, update } from '@/api/device/instance'; +import { getImage, onlyMessage, LocalStore } from '@/utils/comm'; +import { addMonitor } from '@/api/monitor'; +const emit = defineEmits(['close', 'save']); +const props = defineProps({ + data: { + type: Object, + default: undefined, + }, + logicId: { + type: String, + } +}); +const loading = ref<boolean>(false); + +const formRef = ref(); + +const modelRef = reactive({ + id: undefined, + nodeName: '', + image: '', + +}); +const beforeLogoUpload = (file: File) => { + const typeBool = + ['.jpg', '.png', '.jfif', '.pjp', '.pjpeg', '.jpeg'] + .map((m: string) => m.split('.')[1]) + .filter((typeStr) => file.type.includes(typeStr)).length > 0; + const sizeBool = file.size / 1024 / 1024 < 4; + if (!typeBool) { + onlyMessage(`请上传.jpg.png.jfif.pjp.pjpeg.jpeg格式的图片`, 'error'); + } else if (!sizeBool) { + onlyMessage(`图片大小必须小于4M`, 'error'); + } + return typeBool && sizeBool; +}; +/** + 获取file,通过FileReader获取图片的 base64 +*/ +const customRequest = (option: any) => { + const formData = new FormData(); + formData.append("files[]", option.file); + const reader = new FileReader(); + reader.readAsDataURL(option.file); + reader.onloadend = function (e) { + modelRef.image = e.target.result || '' + if (e && e.target && e.target.result) { + option.onSuccess(); + } + }; +} + +watch( + () => props.data, + (newValue) => { + Object.assign(modelRef, newValue); + }, + { immediate: true, deep: true }, +); + +const handleCancel = () => { + emit('close'); + formRef.value.resetFields(); +}; + +const handleSave = () => { + formRef.value + .validate() + .then(async (_data: any) => { + loading.value = true; + const obj = { ..._data }; + obj.logicId = props.logicId; + obj.parentId = 0; + if (!obj.id) { + delete obj.id; + } + const resp = await addMonitor(obj).finally(() => { + loading.value = false; + }); + if (resp.status === 200) { + onlyMessage('操作成功!'); + emit('save'); + emit('close'); + formRef.value.resetFields(); + } + }) + .catch((err: any) => { + console.log('error', err); + }); +}; +</script> +<style lang="less" scoped> +.upload-image-warp-back { + display: flex; + justify-content: flex-start; + + .upload-image-border-back { + position: relative; + overflow: hidden; + border: 1px dashed #d9d9d9; + transition: all 0.3s; + width: 100%; + height: 415px; + + &:hover { + border: 1px dashed #1890ff; + display: flex; + } + + .upload-image-content-back { + align-items: center; + justify-content: center; + position: relative; + display: flex; + flex-direction: column; + width: 610px; + height: 415px; + padding: 8px; + background-color: rgba(0, 0, 0, 0.06); + cursor: pointer; + + .loading-back { + position: absolute; + } + + .upload-image { + width: 100%; + height: 100%; + background-repeat: no-repeat; + background-position: 50%; + background-size: cover; + } + + .upload-image-mask { + align-items: center; + justify-content: center; + position: absolute; + top: 0; + left: 0; + display: none; + width: 100%; + height: 100%; + color: #fff; + font-size: 16px; + background-color: rgba(0, 0, 0, 0.35); + } + + &:hover .upload-image-mask { + display: flex; + } + } + } +} +</style> \ No newline at end of file diff --git a/src/views/location/Site/RoleLeft/index.vue b/src/views/location/Site/RoleLeft/index.vue new file mode 100644 index 0000000..8ad6a59 --- /dev/null +++ b/src/views/location/Site/RoleLeft/index.vue @@ -0,0 +1,332 @@ +<template> + <div class="left-tree-container"> + <j-input placeholder="名称" v-model:value="searchValue" @pressEnter="search" @change="searchChange"> + <template #suffix> + <AIcon type="SearchOutlined" @click="search" /> + </template> + </j-input> + <div v-if="admin" class="add-btn"> + + <j-row :style="{ marginTop: '10px' }"> + <j-button type="primary" @click="visible = true"> + 新增站点 + </j-button> + </j-row> + + </div> + <div class="tree"> + <j-spin :spinning='loading'> + <j-tree :tree-data="treeData" v-if="treeData.length" draggable @dragstart="dragstart" + :fieldNames="{ title: 'nodeName', key: 'id', children: 'children' }" blockNode + :default-expanded-keys="defaultExpandedKeys" + :showLine="{ showLeafIcon: false }" :showIcon="true"> + <template #title="{ nodeName, data, level, type, positionX, parentId, nodeType }"> + <div v-if="selectId === data.id"> + <j-input v-model:value="addName" @blur="() => saveGroup(data)" ref="inputRef" + :maxlength="64"></j-input> + </div> + <span v-else class='department-tree-item-content'> + <span class='title' :style="{ + width: nodeType == '1' ? '80px' : '110px', + }"> + <j-ellipsis> + <AIcon v-if="nodeType == '1'" type="HeartFilled" /> + &nbsp;{{ nodeName }} + </j-ellipsis> + </span> + <span class="func-btns" :style="{ + width: nodeType == '0' ? '120px' : '100px', + }" @click="(e) => e.stopPropagation()"> + <PermissionButton type="link" :tooltip="{ title: '编辑' }" @click="editGroup(data)"> + <AIcon type="EditOutlined" /> + </PermissionButton> + <PermissionButton v-if="nodeType === '0' || parentId === '0'" type="link" :tooltip="{ title: '组内监测点绑定设备' }" @click="() => bindVisible = true"> + <AIcon type="ApiOutlined" /> + </PermissionButton> + <PermissionButton v-if="nodeType === '0' || parentId === '0'" type="link" :tooltip="{ title: '新增子站' }" @click="addGroup(data,'0')"> + <AIcon type="PlusOutlined" /> + </PermissionButton> + <PermissionButton v-if="nodeType === '0' || parentId === '0'" type="link" :tooltip="{ title: '新增子监测点' }" @click="addGroup(data,'1')"> + <AIcon type="PlusCircleOutlined" /> + </PermissionButton> + <PermissionButton type="link" :tooltip="{ title: '删除' }" :popConfirm="{ + title: `确定要删除吗`, + onConfirm: () => { deleteGroup(data.id) }, + }"> + <AIcon type="DeleteOutlined" /> + </PermissionButton> + <PermissionButton v-if="nodeType === '1'" type="link" :tooltip="{ title: '移除' }"> + <AIcon type="CloseOutlined" /> + </PermissionButton> + </span> + </span> + </template> + </j-tree> + <j-empty v-else description="暂无数据" /> + </j-spin> + </div> + <j-row :gutter="10"> + <j-col :span="12"> + <j-button type="primary" @click="visible = true"> + 导入站点 + </j-button> + </j-col> + <j-col :span="12"> + <j-button type="primary" @click="visible = true"> + 导出站点 + </j-button> + </j-col> + </j-row> + <AddGroup v-if="visible" :logicId="logicId" @close="visible = false" @save="() => { emit('getMonitorList') }" /> + <BindDevice v-if="bindVisible" :logicId="logicId" @close="bindVisible = false" @save="() => { emit('getMonitorList') }" /> + </div> +</template> + +<script lang="ts" setup> +import { onlyMessage } from '@/utils/comm'; +import { useUserInfo } from '@/store/userInfo'; +import { storeToRefs } from 'pinia'; +import { filterTreeItem } from '@/utils/comm' +import { addMonitor, editMonitor, deleteMonitor } from '@/api/monitor'; +import { randomString } from '@/utils/utils'; +import AddGroup from './AddGroup.vue'; +import BindDevice from './BindDevice.vue'; + +const visible = ref<boolean>(false); +const bindVisible = ref<boolean>(false); +const loading = ref<boolean>(false); // 数据加载状态 +const emit = defineEmits(['selectData', 'getMonitorList']); +let treeData = ref([]); +const props = defineProps({ + listData: { + type: Array, + default: [] + }, + logicId: { + type: String, + } +}) +const userInfoStore = useUserInfo(); +const { userInfos } = storeToRefs(userInfoStore); +const admin = computed(() => { + return userInfos.value?.username === 'admin'; +}); + +const searchValue = ref(); +const inputRef = ref(); +const addName = ref(); +const selectId = ref(); +const defaultExpandedKeys = ref<(number | string)[]>([]); + +//获取全部id +const getTreeId = <T>(treeNode: T[]) => { + let result: any[] = []; + const traverse = (node: any) => { + result?.push(node.id) + if (node.children && Array.isArray(node.children)) { + node.children.forEach((child: any) => traverse(child)) + } + } + treeNode?.forEach((child: any) => traverse(child)) + return result +} +watch(() => props.listData, (newValue) => { + treeData.value = newValue; + defaultExpandedKeys.value = getTreeId(treeData.value) + console.log("🚀 treeData:", treeData.value) +},{deep: true,immediate: true}) +const openGroup = () => { + visible.value = false; +}; + +const dragstart = ({ event, node }: { event: DragEvent, node: any }) => { + //event.preventDefault() + event.dataTransfer?.setData('monitorData', JSON.stringify(node.dataRef)); // 使用 text/plain 类型存储节点信息 +} + + + +const queryGroup = async (select?: Boolean, searchName?: string) => { + // const params = searchName + // ? { + // sorts: [{ name: 'createTime', order: 'desc' }], + // terms: [ + // { + // terms: [ + // { + // value: '%' + searchName + '%', + // termType: 'like', + // column: 'name', + // }, + // ], + // }, + // ], + // } + // : { sorts: [{ name: 'createTime', order: 'desc' }] }; + // const req: any = await queryRoleGroup(params); + // if (req.status === 200) { + // listData.value[0].children = req.result; + // if (req.result[req.result.length - 1].id === 'default_group') { + // req.result.unshift(req.result[req.result.length - 1]); + // req.result.pop(); + // } + // // if(req.result.length && select){ + // // selectGroup(req.result[0].id) + // // } + // } +}; +const addGroup = async (data: any,nodeType: any) => { + + addName.value = ''; + const newId = randomString(); + if (!data.children) { + data.children = [] + }; + data.children?.splice(1, 0, { + isNewChild: true, + nodeName: '', + logicId: props.logicId, + nodeType, + id: newId, + parentId: data.id, + }); + treeData.value = [...treeData.value] + selectId.value = newId; + defaultExpandedKeys.value = getTreeId(treeData.value) + console.log("🚀 data,nodeType:", data, nodeType,treeData.value); + await nextTick(() => { + inputRef.value.focus(); + }); +}; +const saveGroup = async (data: any) => { + + if (data.nodeName !== addName.value) { + const saveData = { + ...data, + nodeName: addName.value, + }; + if (data.isNewChild) { + delete data.id; + } + const res = data.isNewChild ? await addMonitor(saveData) : await editMonitor(saveData); + if (res.status === 200) { + onlyMessage('操作成功!'); + emit('getMonitorList'); + } else { + onlyMessage('操作失败!'); + } + } + setTimeout(() => { + selectId.value = ''; + }, 300); +}; + +const deleteGroup = async (id: string) => { + const res = await deleteMonitor(id); + if (res.status === 200) { + onlyMessage('操作成功!'); + emit('getMonitorList'); + } else { + onlyMessage('操作失败!'); + } +} + +const search = () => { + queryGroup(true, searchValue.value); +}; +const searchChange = () => { + if (searchValue.value === '') { + queryGroup(); + } +}; +const editGroup = (data: any) => { + if (!selectId.value) { + selectId.value = data.id; + addName.value = data.nodeName; + // listData.value[0].children?.forEach((item: any) => { + // if (item.id === data.id) { + // item.edit = true; + // } + // }); + nextTick(() => { + inputRef.value.focus(); + }); + } +}; + +onMounted(() => { +}); +</script> +<style lang="less" scoped> +.left-tree-container { + display: flex; + height: 100%; + flex-direction: column; + + .add-btn { + margin: 24px 0; + + :deep(.ant-btn-primary) { + width: 100%; + } + } + + :deep(.ant-tree-treenode-disabled) { + .ant-tree-node-content-wrapper { + color: rgba(0, 0, 0, 0.55) + } + } + + :deep(.ant-tree-treenode) { + width: 100%; + + .ant-tree-node-content-wrapper { + flex: 1 1 auto; + + .ant-tree-title { + &:hover { + .func-btns { + display: block; + } + } + } + } + } + + + .tree { + overflow-y: auto; + overflow-x: auto; + flex: 1 1 auto; + + .department-tree-item-content { + display: inline-flex; + align-items: stretch; + + .title { + flex: 1; + margin-right: 80px; + } + + .func-btns { + display: none; + font-size: 14px; + width: 100px; + margin-left: -80px; + + :deep(.ant-btn-link) { + padding: 0 4px; + height: 24px; + } + } + } + + .loading { + display: flex; + width: 100%; + justify-content: center; + margin-top: 20px; + } + } +} +</style> diff --git a/src/views/location/Site/RoleRight/components/AddDialog.vue b/src/views/location/Site/RoleRight/components/AddDialog.vue new file mode 100644 index 0000000..0125286 --- /dev/null +++ b/src/views/location/Site/RoleRight/components/AddDialog.vue @@ -0,0 +1,10 @@ + +<template> + +</template> + +<script setup lang="ts"> + +</script> + +<style scoped></style> diff --git a/src/views/location/Site/RoleRight/index.vue b/src/views/location/Site/RoleRight/index.vue new file mode 100644 index 0000000..8ede248 --- /dev/null +++ b/src/views/location/Site/RoleRight/index.vue @@ -0,0 +1,119 @@ +<template> + <div @dragenter.prevent="onDragEnter" @dragover.prevent="onDragOver" @dragleave="onDragLeave" @drop="drop" id="map" + :style="{ width: '70vw', height: '70vh' }"></div> +</template> + +<script setup lang="ts" name="Role"> +import "leaflet/dist/leaflet.css"; +import { LatLngBoundsExpression, Layer, LayerGroup, Map } from 'leaflet'; +import { monitorType } from '../type' +import * as L from 'leaflet' +import { filterTreeItem } from '@/utils/comm' +import bg from '/images/bind/bdt.png' +import { deviceIcons,stateIcons } from '@/assets/diviceIcon' +interface MarkerType extends Layer { + markerId?: string | number +} +const props = defineProps({ + listData: { + type: Array, + default: [] + } +}) +const url = ref<string>(bg) +const map = ref<Map>() +const markers = ref<LayerGroup>() +const mapMonitorList = ref<monitorType[]>([]) + +const createMarkers = (info: monitorType): MarkerType => { + let { id, name, monitorType, deviceNum, deviceIp, devicePort, positionX, positionY } = info + + const x = parseFloat(positionX as string); + const y = parseFloat(positionY as string); + const marker: MarkerType = L.marker( + [y, x], { + icon: L.icon({ + iconUrl: deviceIcons[1], + iconSize: [30, 30], + }), + draggable: true,//开启拖拽 + }) + marker.markerId = id + marker.bindTooltip('dsadas', { + permanent: false, // 永久显示提示 + direction: 'top',// 提示显示的方向 + offset: L.point(0, -10) + }); + console.log(marker) + return marker +} +//初始化画板 +const loadMap = () => { + const imageBounds: LatLngBoundsExpression = [[0, 0], [1000, 1980]]; + if (map.value) { + + } else { + map.value = L.map('map', { + crs: L.CRS.Simple, + minZoom: -1, + maxZoom: 16, + center: [0, 1], + zoom: -1, + maxBounds: imageBounds, + attributionControl: false,//默认情况下,是否将 attribution 版权控件添加到地图中。 + zoomControl: false,//默认情况下,是否将 zoom 缩放控件添加到地图中。 + doubleClickZoom: false,//禁用双击放大 + }); + L.imageOverlay(url.value, imageBounds,).addTo(map.value); + map.value.setMaxBounds(imageBounds); + markers.value = L.layerGroup() + map.value.addLayer(markers.value); + map.value.on('dblclick', () => { + map.value?.setZoom(-1) + }) + } +} +const drop = (event: DragEvent) => { + //event.preventDefault() + const mouseEventLatLng = map.value?.mouseEventToLatLng(event); + if (mouseEventLatLng) { + const { lat, lng } = mouseEventLatLng; + //const dragEvent = event as DragEvent; + let monitorData = JSON.parse(event.dataTransfer?.getData('monitorData') || ''); + //存在则修改位置 + if (monitorData.positionX) { + + //不存在则创建marker + } else { + markers.value?.addLayer(createMarkers({ ...monitorData, positionX: lng, positionY: lat })) + } + + } +} + +const onDragEnter = (event: MouseEvent) => { + console.log('Drag enter', event); +}; +const onDragOver = (event: MouseEvent) => { + //console.log('Drag over'); + //event.dataTransfer.dropEffect = 'move'; +}; +const onDragLeave = (event: MouseEvent) => { + console.log('Drag leave'); +}; +onMounted(() => { + loadMap() + mapMonitorList.value = filterTreeItem(props.listData, 'children', (i) => i.level == 2 && i.positionX) + mapMonitorList.value.forEach((i: monitorType) => { + markers.value?.addLayer(createMarkers(i)) + }) + +}) + +</script> + +<style lang="less" scoped> +.leaflet-container{ + background-color: #fff; +} +</style> diff --git a/src/views/location/Site/data.ts b/src/views/location/Site/data.ts new file mode 100644 index 0000000..a8e842f --- /dev/null +++ b/src/views/location/Site/data.ts @@ -0,0 +1,27 @@ +import type { iframeInfoType } from '@/views/gis/Home/type' + +export const minHeight = 376.5 +//半屏某个视窗状态 +export const iframeChild1: iframeInfoType = { + index: 0, + montorId: 1, + viewType: 3, + xs: 24, + sm: 24, + md: 24, + lg: 24, + xl: 12, + iframeHeight: (document.body.clientHeight - 124) / 3 < minHeight * (2 / 3) ? minHeight * (2 / 3) : (document.body.clientHeight - 124) / 3, +} +//默认视窗状态 +export const iframeChild2: iframeInfoType = { + montorId: 1, + viewType: 1, + index: 0, + xs: 24, + sm: 24, + md: 12, + lg: 12, + xl: 8, + iframeHeight: (document.body.clientHeight - 124) / 2 < minHeight ? minHeight : (document.body.clientHeight - 124) / 2, +} \ No newline at end of file diff --git a/src/views/location/Site/index.vue b/src/views/location/Site/index.vue new file mode 100644 index 0000000..525720f --- /dev/null +++ b/src/views/location/Site/index.vue @@ -0,0 +1,59 @@ +<template> + <page-container> + <FullPage> + <div class="dictionary_contain"> + <div class="dictionary_left"> + <Left :listData="listData" :logicId="logicId" @getMonitorList="getMonitorList" @select-data="selectData"/> + </div> + <div class="dictionary_right"> + <Right :listData="listData" :groupId="groupId"/> + </div> + </div> + </FullPage> + </page-container> + </template> + + <script lang="ts" setup> + import Left from './RoleLeft/index.vue' +import Right from './RoleRight/index.vue' +import { TreeNode } from './type'; +import { treeList } from '@/views/gis/data' +import { Ref } from 'vue'; +import { findDefaultTree } from '@/api/monitor' + +const listData: Ref<TreeNode[]> = ref([]); +const groupId = ref() +// 获取当前路由实例 +const route = useRoute(); +const logicId = route.matched[0].name?.toString() || '1'; + +const getMonitorList = async () => { + const { result } = await findDefaultTree(logicId); + listData.value = result as TreeNode; +} + +onMounted(async () => { + await getMonitorList(); +}); + + const selectData = (data:any)=>{ + groupId.value = data[0] + } + </script> + <style lang="less" scoped> + .dictionary_contain{ + display: flex; + background-color: #fff; + padding: 24px; + height: 100%; + } + .dictionary_left{ + border-right: 1px solid #f0f0f0; + padding-right: 24px; + flex:1; + height:100% + } + .dictionary_right{ + flex:4 + } + </style> \ No newline at end of file diff --git a/src/views/location/Site/type.d.ts b/src/views/location/Site/type.d.ts new file mode 100644 index 0000000..f3d5a88 --- /dev/null +++ b/src/views/location/Site/type.d.ts @@ -0,0 +1,63 @@ +/* + * @Author: zhangliang 1466013297@qq.com + * @Date: 2024-09-19 10:20:10 + * @LastEditors: zhangliang 1466013297@qq.com + * @LastEditTime: 2024-09-23 18:00:26 + * @FilePath: \jetlinks-ui-vue-old\src\views\gis\Site\type.d.ts + * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE + */ +export type iframeInfoType = { + xs?: 6 | 8 | 12 | 24, + sm?: 6 | 8 | 12 | 24, + md?: 6 | 8 | 12 | 24, + lg?: 6 | 8 | 12 | 24, + xl?: 6 | 8 | 12 | 24, + iframeHeight?: number, + montorId?: number, + index?: number, + viewType?: 1 | 2 | 3,//窗口显示状态1:初始状态,PRPSorPRPD只显示一个且可以切换, 2:半屏状态:3:其他窗口半屏状态 +} +// 定义监测点组的类型 +export interface monitorType { + name: string; + id: string; + disabled?: boolean; // 可选属性,表示节点是否被禁用 + level?: number; // 节点的层级,可选属性 1为监测点组级别。2为监测点级别 + type?: string; // 节点的类型,可选属性 root为跟节点 + monitorType?: number, // 监测类型 + deviceNum?: number | string, + deviceIp?: number | string, + devicePort?: number | string, + positionX?: number | string, + positionY?: number | string, +} +export interface TreeNode { + // name: string; + // id: string; + // disabled?: boolean; // 可选属性,表示节点是否被禁用 + // children?: TreeNode[] | monitorType[]; // 子节点数组,可选属性 + // level?: number; // 节点的层级,可选属性 1为监测点组级别。2为监测点级别 + // type?: string; // 节点的类型,可选属性 root为跟节点 + id?: number | string, + logicId?: number | string,//业务系统ID,对应的业务系统菜单的ID + parentId: number | string,//父ID 根节点默认传0 + nodeName: string,//节点名称 + nodeType: number | string,//节点类型 0组,1监测点 + nodeProduct: number | string,//节点关联产品,监测点必须关联 + deviceId: number | string, //关联设备id,监测点必须关联 + voltageLevel: number | string, //电压等级 + image: string,//图片Base64,非地图节点 + positionX: number | string, //坐标X/经度 监测点必填 + positionY: number | string, //坐标Y/经度 监测点必填 + sort: number | string, //排序 + forwardNum: number | string, //转发编号 监测点必填 + devId: number | string, //通道编号:61850使用 + nodeExtension: number | string, //拓展字段,以json格式存储 + creatorId: number | string, //创建人ID + createTime: number | string, //创建时间 + modifierId: number | string, //修改人ID + modifyTime: number | string, //修改时间 + creatorName: number | string, + modifierName: number | string, + +} \ No newline at end of file diff --git a/src/views/media/Device/Channel/data.ts b/src/views/media/Device/Channel/data.ts deleted file mode 100644 index 271af9e..0000000 --- a/src/views/media/Device/Channel/data.ts +++ /dev/null @@ -1,10 +0,0 @@ -export const mediaConfigMap = new Map() -mediaConfigMap.set('onvif-record','onvif-record') -mediaConfigMap.set('onvif-ptz','onvif-ptz') -mediaConfigMap.set('fixed-media-record','fix-record') -mediaConfigMap.set('fixed-media-ptz','fix-ptz') -mediaConfigMap.set('gb28181-2016-ptz','gb-ptz') -mediaConfigMap.set('gb28181-2016-record','gb-record') -mediaConfigMap.set('media-plugin-ptz','plugin-ptz') -mediaConfigMap.set('media-plugin-record','plugin-record') - diff --git a/src/views/media/Device/Summary/index.vue b/src/views/media/Device/Summary/index.vue deleted file mode 100644 index 1ea60f0..0000000 --- a/src/views/media/Device/Summary/index.vue +++ /dev/null @@ -1,148 +0,0 @@ -<template> - <a-drawer - visible - placement="right" - :width="600" - :closable="false" - @close="$emit('closeDrawer')" - > - <template #title> - <div class="deviceName"> - <Ellipsis style="max-width: 400px">{{ - deviceData?.name - }}</Ellipsis> - </div> - <div class="deviceId"> - <span> 设备ID: </span> - <a @click="jumpDetail">{{ deviceData?.id }}</a> - </div> - </template> - <div> - <a-descriptions bordered :column="1"> - <a-descriptions-item label="接入方式">{{ - PROVIDER_OPTIONS.find( - (i) => i.value === deviceData?.provider, - )?.label - }}</a-descriptions-item> - <a-descriptions-item label="所属产品">{{ - deviceData?.productName - }}</a-descriptions-item> - <a-descriptions-item - v-if="deviceData?.provider === 'onvif'" - label="接入地址" - >{{ deviceData?.others?.onvifUrl }}</a-descriptions-item - > - <a-descriptions-item - v-if="deviceData?.provider === 'onvif'" - label="接入账户" - >{{ - deviceData?.others?.onvifUsername - }}</a-descriptions-item - > - <a-descriptions-item - v-if=" - ['onvif', 'gb28181-2016'].includes(deviceData?.provider) - " - label="接入密码" - ><div class="password"> - <span>{{ showPassword }}</span> - <AIcon - :type=" - visiblePassword - ? 'EyeInvisibleOutlined' - : 'EyeOutlined' - " - @click="visiblePassword = !visiblePassword" - class="passwordIcon" - ></AIcon></div - ></a-descriptions-item> - <a-descriptions-item label="说明">{{ - deviceData?.description || '--' - }}</a-descriptions-item> - </a-descriptions> - </div> - </a-drawer> -</template> - -<script setup> -import { useMenuStore } from 'store/menu'; -import DeviceApi from '@/api/media/device'; -import { PROVIDER_OPTIONS } from '../const'; -const props = defineProps({ - deviceId: { - type: Object, - }, -}); -const menuStory = useMenuStore(); -const deviceData = ref(); - -const visiblePassword = ref(false); -const showPassword = computed(() => { - let password = - deviceData.value?.provider === 'onvif' - ? deviceData.value?.others?.onvifPassword - : deviceData.value?.others?.access_pwd; - if (!visiblePassword.value) { - let hiddenPassword = '' - for (let i = 0; i < password.length; i++) { - hiddenPassword += '*' - } - return hiddenPassword - }else{ - return password - } -}); -const jumpDetail = (data) => { - menuStory.jumpPage('device/Instance/Detail', { id: props.deviceId }); -}; -const getProductList = async () => { - const params = { - paging: false, - sorts: [{ name: 'createTime', order: 'desc' }], - terms: [{ column: 'id', value: deviceData.value?.productId }], - }; - const res = await DeviceApi.queryProductList(params); - if (res.success) { - deviceData.value.productName = res.result?.[0]?.name; - if (deviceData.value.provider !== 'media-plugin') { - deviceData.value.others.access_pwd = deviceData.value.others - .access_pwd - ? deviceData.value.others.access_pwd - : res.result?.[0]?.configuration?.access_pwd; - } - } -}; -/** - * 获取详情 - */ -const getDetail = async () => { - const res = await DeviceApi.detail(props.deviceId); - if (res.success) { - deviceData.value = res.result; - await getProductList(); - } -}; -onMounted(() => { - getDetail(); -}); -</script> -<style lang="less" scoped> -.deviceId { - font-size: 13px; - span { - color: #777777; - } -} -.deviceName { - font-size: 16px; - font-weight: 600; -} -.password { - position: relative; - .passwordIcon { - position: absolute; - right: 0; - top: 5px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/Actions.vue b/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/Actions.vue deleted file mode 100644 index 3b72fbb..0000000 --- a/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/Actions.vue +++ /dev/null @@ -1,188 +0,0 @@ -<template> - <div class="actions-warp"> - <div v-for="item in actions"> - <div v-if="activeKeys.some((active) => active === item.actionId)"> - <div class="actions-item" v-if="item.executor === 'alarm'"> - <div class="item-options-warp"> - <div class="item-options-type"> - <!-- <img - style="width: 18px" - :src=" - iconMap.get( - item.executor === 'alarm' - ? item.alarm.mode - : item.executor, - ) - " - /> --> - <AIcon :type="iconMap.get(item.executor === 'alarm' ? item.alarm.mode : item.executor)"/> - </div> - <div class="item-options-content"> - <template v-if="item.executor === 'alarm'"> - <span> - 满足条件后将{{ - item.alarm.mode === 'trigger' - ? '触发' - : '解除' - }}当前告警 - </span> - </template> - </div> - </div> - </div> - </div> - </div> - </div> -</template> - -<script setup name="Actions"> -import { - iconMap, -} from '@/views/rule-engine/Scene/Save/action/ListItem/util'; -import { isBoolean } from 'lodash-es'; - -const props = defineProps({ - actions: { - type: Array, - default: () => [], - }, - activeKeys: { - // 后端返回已关联的执行动作 - type: Array, - default: () => [], - }, -}); -</script> - -<style scoped lang="less"> -.actions-item { - position: relative; - padding: 8px; - border: 1px dashed #999; - border-radius: 2px; - margin-bottom: 8px; - - .item-options-warp { - display: inline-flex; - height: 48px; - border: 1px solid #e0e0e0; - border-radius: 6px; - - .item-options-type { - display: flex; - align-items: center; - justify-content: center; - width: 48px; - background-color: #f0f0f0; - border-radius: 6px 0 0 6px; - cursor: pointer; - font-size: 22px; - } - - .item-options-content { - display: flex; - align-items: center; - padding: 0 8px; - background: #fafafa; - border-radius: 0 6px 6px 0; - cursor: pointer; - - div { - padding: 6px 10px; - color: #333; - font-size: 14px; - line-height: 22px; - background-color: #fff; - border-radius: 22px; - - .notify-text-highlight { - margin-left: 5px; - font-weight: bold; - } - - .notify-img-highlight { - margin: 0 10px; - color: rgba(0, 0, 0, 0.8); - } - } - } - } - - .item-number { - position: absolute; - top: 0; - left: 16px; - font-weight: 800; - transform: translateY(-50%); - } -} - -.actions-item-filter-warp { - position: relative; - margin-bottom: 12px; - margin-top: 16px; - padding: 2px 0; - border: 1px dashed #999; - border-radius: 2px; - - &.filter-border { - padding: 2px 16px; - border-radius: 2px; - } - - .actions-item-filter-warp-tip { - position: absolute; - top: 0; - left: 16px; - z-index: 2; - color: rgba(0, 0, 0, 0.55); - font-weight: 800; - font-size: 14px; - line-height: 1; - background-color: #fff; - transform: translateY(-50%); - } - - .actions-item-filter-overflow { - display: flex; - padding-top: 4px; - overflow-x: auto; - overflow-y: visible; - row-gap: 16px; - } - - .filter-add-button { - width: 100%; - color: rgba(0, 0, 0, 0.3); - text-align: center; - cursor: pointer; - } - - .terms-params { - // display: inline-block; - display: flex; - flex-shrink: 0; - - // &:not(:first-child) { - // margin-bottom: 16px; - // } - - .terms-params-warp { - display: flex; - align-items: baseline; - } - - .term-type-warp { - // display: inline-block; - width: 50px; - margin: 0 16px; - - .term-type { - padding-top: 4px; - padding-bottom: 4px; - border-radius: 2px; - } - } - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/BranchesTabs.vue b/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/BranchesTabs.vue deleted file mode 100644 index 2af401b..0000000 --- a/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/BranchesTabs.vue +++ /dev/null @@ -1,72 +0,0 @@ -<template> - <div> - <div> - <div v-for="group in branchesGroup"> - <div v-for="(branch, index) in group.children"> - <template v-if="branch.serial?.length"> - <Actions - :actions="branch.serial" - :activeKeys="activeKeys" - :serial="true" - /> - </template> - <template v-if="branch.parallel?.length"> - <Actions - :actions="branch.parallel" - :activeKeys="activeKeys" - /> - </template> - </div> - </div> - </div> - </div> -</template> - -<script setup name="BranchesTabs"> -import Actions from './Actions.vue'; - -const props = defineProps({ - branchesGroup: { - type: Array, - default: () => [], - }, - alarmId: { - type: String, - default: undefined, - }, - sceneId: { - type: String, - default: undefined, - }, - activeKeys: { - // 后端返回已关联的执行动作 - type: Array, - default: () => [], - }, - -}); - -const emit = defineEmits(['change', 'select']); - -</script> - -<style scoped lang="less"> -.branches-tabs-title { - margin-bottom: 8px; - font-size: 14px; - font-weight: 500; -} - -.branches-tabs-alarm { - padding: 12px 16px; - border-radius: 4px; - border: 1px solid #a9a9a9; - display: flex; - justify-content: space-between; - align-items: center; - - &:not(:last-child) { - margin-bottom: 8px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/CardBox.vue b/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/CardBox.vue deleted file mode 100644 index e620607..0000000 --- a/src/views/rule-engine/Alarm/Configuration/HandTrigger/components/CardBox.vue +++ /dev/null @@ -1,566 +0,0 @@ -<template> - <div class="card j-table-card-box" :class="{'selectedCard':selected}" @click="click"> - <div class="card-warp" :class="{ disabled: disabled }"> - <div class="card-type"> - <div class="card-type-text"> - <img - :height="16" - :src="itemType.icon" - style="margin-right: 5px" - /> - {{ itemType.text }} - </div> - </div> - <div class="card-content" :style="{ paddingTop: '40px' }"> - <div - class="card-content-bg-item card-content-bg1" - :style="bgcColor" - ></div> - <div - class="card-content-bg-item card-content-bg2" - :style="bgcColor" - ></div> - <div class="card-content-body"> - <!-- 图片 --> - <div class="card-item-avatar"> - <img :src="itemType.img" /> - </div> - - <!-- 内容 --> - <div class="card-item-body"> - <div> - <Ellipsis style="width: calc(100% - 100px)"> - <span style="font-size: 16px; font-weight: 600"> - {{ value.name }} - </span> - </Ellipsis> - <div v-if="showBindTags && activeBranches.length"> - <a-tag v-for="tag in activeBranches"> - <Ellipsis style="max-width: 120px"> - {{ tag }} - </Ellipsis> - </a-tag> - </div> - <Ellipsis v-else> - <div class="subTitle"> - 说明:{{ - value?.description || itemType.tip - }} - </div> - </Ellipsis> - </div> - </div> - </div> - <div class="card-content-tabs" v-if="showBranches"> - <BranchesTabs - :branchesGroup="branchesGroup" - :alarmId="alarmId" - :branchesId="value.id" - :activeKeys="activeKeys" - :key="value.id" - @change="bindAlarm" - /> - </div> - <div - class="card-state" - :style="{ - backgroundColor: getHexColor(statusNames[status]), - }" - > - <div class="card-state-content"> - <BadgeStatus - :status="status" - :text="statusText" - :statusNames="statusNames" - ></BadgeStatus> - </div> - </div> - </div> - <div - v-if="status === 'disable' && !isInvalid" - class="card-mask mask-hover" - :style="maskStyle" - > - <div class="mask-content"> - <slot name="mask"> - <div>该场景为禁用状态,无法触发告警</div> - </slot> - </div> - </div> - <div - v-if="isInvalid" - class="card-mask mask-full" - style="background-color: rgba(0, 0, 0, 0.2)" - > - <a-button style="font-size: 16px" type="link" @click="jumpView" - >无效数据,请重新保存场景</a-button - > - </div> - </div> - </div> -</template> - -<script setup lang="ts" name="SceneCardBox"> -import BadgeStatus from '@/components/BadgeStatus/index.vue'; -import color, { getHexColor } from '@/components/BadgeStatus/color'; -import BranchesTabs from './BranchesTabs.vue'; -import { PropType } from 'vue'; -import { handleActiveBranches, handleGroupAndFilter, typeMap } from '../../Save/Scene/Save/utils'; -import { useMenuStore } from '@/store/menu'; - -type EmitProps = { - (e: 'click'): void; - (e: 'change', key: string, selected: boolean): void; - (e: 'reload'): void; -}; - -const emit = defineEmits<EmitProps>(); - -const props = defineProps({ - value: { - type: Object as PropType<Record<string, any>>, - default: () => ({}), - }, - statusText: { - type: String, - default: '正常', - }, - status: { - type: [String, Number] as PropType<string | number>, - default: 'default', - }, - statusNames: { - type: Object as PropType<Record<any, any>>, - default: () => ({ - started: 'processing', - disable: 'error', - }), - }, - disabled: { - type: Boolean, - default: false, - }, - alarmId: { - type: String, - default: undefined, - }, - activeKeys: { - // 后端返回已关联的执行动作 - type: Array, - default: () => [], - }, - showBranches: { - type: Boolean, - default: true, - }, - showBindTags: { - type: Boolean, - default: false, - }, - maskStyle: { - type: Object, - default: undefined, - }, - selected:{ - type: Boolean, - default: undefined, - } -}); - -const isInvalid = ref(false); -const menuStory = useMenuStore(); -const bgcColor = computed(() => { - const key = props.statusNames[props.status]; - const _color = color[key] || color.default; - return { - background: `linear-gradient( - 188.4deg, - rgba(${_color}, 0.03) 30%, - rgba(${_color}, 0) 80% - )`, - }; -}); - -const itemType = computed(() => { - return typeMap.get(props.value.triggerType) || {}; -}); - -const branchesGroup = computed(() => { - return handleGroupAndFilter( - props.value.branches, - props.value.options?.when || [], - ); -}); - -const activeBranches = computed(() => { - const { data, invalid } = handleActiveBranches( - branchesGroup.value, - props.activeKeys, - ); - - isInvalid.value = invalid; - console.log(isInvalid.value,'isInvalid') - return data; -}); - -const bindAlarm = (key: string, selected: boolean) => { - emit('change', key, selected); -}; - -const click = () => { - emit('click'); -}; - -const jumpView = () => { - const url = menuStory.menus['rule-engine/Scene/Save']?.path; - const tab: any = window.open( - `${ - window.location.origin + window.location.pathname - }#${url}?triggerType=${props.value.triggerType}&id=${props.value.id}`, - ); - tab.onTabSaveSuccess = (value: any) => { - console.log('', value); - if (value.success) { - emit('reload'); - } - }; -}; -</script> - -<style scoped lang="less"> -.card { - width: 100%; - background-color: #fff; - - .checked-icon { - position: absolute; - right: -22px; - bottom: -22px; - z-index: 2; - width: 44px; - height: 44px; - color: #fff; - background-color: #2f54eb; - transform: rotate(-45deg); - - > div { - position: relative; - height: 100%; - transform: rotate(45deg); - - > span { - position: absolute; - top: 6px; - left: 6px; - font-size: 12px; - } - } - } - - .card-warp { - position: relative; - border: 1px solid #e6e6e6; - overflow: hidden; - cursor: pointer; - - &.disabled { - filter: grayscale(100%); - cursor: not-allowed; - } - - &.active { - position: relative; - border: 1px solid #2f54eb; - } - - .card-type { - position: absolute; - top: 0; - left: -15px; - height: 32px; - padding: 0 30px; - color: rgba(0, 0, 0, 0.65); - line-height: 32px; - background-color: rgba(0, 0, 0, 0.06); - transform: skewX(-45deg); - - .card-type-text { - display: flex; - align-items: center; - justify-content: center; - transform: skewX(45deg); - } - } - - .card-content { - position: relative; - padding: 30px 12px 30px 30px; - overflow: hidden; - .card-content-tabs{ - margin-top: 20px - } - .card-item-avatar { - margin-right: 16px; - display: flex; - align-items: center; - } - - .card-item-body { - display: flex; - flex-direction: column; - flex-grow: 1; - width: 0; - - .ant-row { - margin-top: 19px; - } - - .condition-name { - margin-top: 10px; - } - } - - .card-state { - position: absolute; - top: 30px; - right: -12px; - display: flex; - justify-content: center; - width: 100px; - padding: 2px 0; - background-color: rgba(#5995f5, 0.15); - transform: skewX(45deg); - - &.success { - background-color: @success-color-deprecated-bg; - } - - &.warning { - background-color: rgba(#ff9000, 0.1); - } - - &.error { - background-color: rgba(#e50012, 0.1); - } - - .card-state-content { - transform: skewX(-45deg); - } - } - - :deep(.card-item-content-title) { - cursor: pointer; - font-size: 16px; - font-weight: 700; - color: @primary-color; - width: calc(100% - 100px); - overflow: hidden; - white-space: nowrap; - text-overflow: ellipsis; - } - - :deep(.card-item-heard-name) { - font-weight: 700; - font-size: 16px; - margin-bottom: 12px; - } - - :deep(.card-item-content-text) { - color: rgba(0, 0, 0, 0.7); - font-size: 12px; - } - } - - .card-content-top-line { - &::before { - position: absolute; - top: 0; - left: 30px + 10px; - display: block; - width: 15%; - min-width: 64px; - height: 2px; - background-image: url('/images/rectangle.png'); - background-repeat: no-repeat; - background-size: 100% 100%; - content: ' '; - } - } - - .card-content-bg-item { - position: absolute; - right: -5%; - height: 100%; - top: 0; - background: linear-gradient( - 188.4deg, - rgba(229, 0, 18, 0.03) 22.94%, - rgba(229, 0, 18, 0) 94.62% - ); - transform: skewX(-15deg); - - &.card-content-bg1 { - width: 44.65%; - } - - &.card-content-bg2 { - width: calc(44.65% + 34px); - } - } - - .card-content-body { - display: flex; - position: relative; - z-index: 99; - } - - .card-mask { - position: absolute; - top: 0; - left: 0; - z-index: 99; - display: flex; - justify-content: center; - width: 100%; - font-size: 16px; - height: 136px; - color: #fff; - padding-top: 30px; - background-color: rgba(#000, 0.45); - cursor: pointer; - transition: background-color 0.3s; - - &.mask-hover:hover { - background-color: rgba(#000, 0.01); - color: transparent; - } - - &.mask-full { - height: 100%; - } - - .mask-content { - display: flex; - align-items: center; - justify-content: center; - width: 100%; - height: 100%; - padding: 24px !important; - flex-direction: column; - } - } - } - - &.item-active { - position: relative; - color: #2f54eb; - - .checked-icon { - display: block; - } - - .card-warp { - border: 1px solid #2f54eb; - } - } - - .card-tools { - display: flex; - margin-top: 8px; - - .card-button { - display: flex; - flex-grow: 1; - - & > :deep(span, button) { - width: 100%; - border-radius: 0; - } - - :deep(button) { - width: 100%; - border-radius: 0; - background: #f6f6f6; - border: 1px solid #e6e6e6; - color: #2f54eb; - - &:hover { - background-color: @primary-color-hover; - border-color: @primary-color-hover; - - span { - color: #fff !important; - } - } - - &:active { - background-color: @primary-color-active; - border-color: @primary-color-active; - - span { - color: #fff !important; - } - } - } - - &:not(:last-child) { - margin-right: 8px; - } - - &.delete { - flex-basis: 60px; - flex-grow: 0; - - :deep(button) { - background: @error-color-deprecated-bg; - border: 1px solid @error-color-outline; - - span { - color: @error-color !important; - } - - &:hover { - background-color: @error-color-hover; - - span { - color: #fff !important; - } - } - - &:active { - background-color: @error-color-active; - - span { - color: #fff !important; - } - } - } - } - - :deep(button[disabled]) { - background: @disabled-bg; - border-color: @disabled-color; - - span { - color: @disabled-color !important; - } - - &:hover { - background-color: @disabled-active-bg; - } - - &:active { - background-color: @disabled-active-bg; - } - } - - // :deep(.ant-tooltip-disabled-compatible-wrapper) { - // width: 100%; - // } - } - } -} -.selectedCard{ - border: 1px solid #2f54eb; -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Actions.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Actions.vue deleted file mode 100644 index 8bf4773..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Actions.vue +++ /dev/null @@ -1,395 +0,0 @@ -<template> - <div class="actions-warp"> - <div v-for="item in actions"> - <div class="actions-item" v-if="item.executor === 'alarm' || show"> - <div class="item-options-warp"> - <div class="item-options-type"> -<!-- <img--> -<!-- style="width: 18px"--> -<!-- :src="iconMap.get(item.executor === 'alarm' ? item.alarm.mode : item.executor)"--> -<!-- />--> - <AIcon :type="iconMap.get(item.executor === 'alarm' ? item.alarm.mode : item.executor)"/> - </div> - <div class="item-options-content"> - <template v-if="item.executor === 'alarm'"> - <span> - 满足条件后将{{ item.alarm.mode === 'trigger' ? '触发' : '解除' }}当前告警 - </span> - - <a-button v-if="!showUnbindBtn" type="link" @click.stop="onBind(item)" :disabled="activeKeys.some(active => active === item.actionId)"> - <template #icon> - <AIcon type="icon-bangding"/> - </template> - {{ selectedKeys.some(selectKey => selectKey === item.actionId) ? '已关联' : '关联' }} - </a-button> - <a-button v-else-if="activeKeys.some(active => active === item.actionId)" type="link" @click.stop="onSelect(item)"> - <template #icon> - <AIcon type="icon-jiebang"/> - </template> - 取消关联 - </a-button> - </template> - <template v-if="item.executor === 'notify' && show"> - <template v-if="item.notify?.notifyType === 'dingTalk'"> - <div v-if="item.options.provider === 'dingTalkRobotWebHook'"> - 通过 - <span class="notify-text-highlight">群机器人消息</span> - 发送 - <span class="notify-text-highlight"> - {{ item.notify?.templateId }} - </span> - </div> - <div v-else> - 通过 - <span class="notify-text-highlight"> - <img - style="width: 18px" - :src="itemNotifyIconMap.get(item.notify?.notifyType)" - > - 钉钉 - </span> - {{ (item.options?.sendTo || item.options?.orgName) ? '向' : "" }} - <span class="notify-text-highlight">{{ item.options?.sendTo || '' }}</span> - <span class="notify-text-highlight">{{ item.options?.orgName || '' }}</span> - 发送 - <span class="notify-text-highlight"> - {{ item.options?.templateName || item?.notify?.templateId }} - </span> - </div> - </template> - <template v-else-if="item.notify?.notifyType === 'weixin'"> - <div> - 通过 - <span class="notify-text-highlight"> - <img - style="width: 18px" - :src="itemNotifyIconMap.get(item.notify?.notifyType)" - /> - 微信 - </span> - {{ (item.options.sendTo || item.options.orgName || item.options.tagName) ? '向' : '' }} - <span class="notify-text-highlight">{{ item.options.sendTo || '' }}</span> - <span class="notify-text-highlight">{{ item.options?.orgName || '' }}</span> - <span class="notify-text-highlight">{{ item.options?.tagName || '' }}</span> - 发送 - <span class="notify-text-highlight">{{ item.options?.templateName || item?.notify?.templateId }}</span> - </div> - </template> - <template v-else-if="item.notify?.notifyType === 'email'"> - <div style="display: flex;"> - 通过 - <span class="notify-text-highlight"> - <img - style="width: 18px" - :src="itemNotifyIconMap.get(item.notify?.notifyType)" - /> - 邮件 - </span> - {{ item.options?.sendTo ? '向' : '' }} - <span class="notify-text-highlight"> - <Ellipsis style='max-width: 400px;'> - {{ item.options?.sendTo || '' }} - </Ellipsis> - </span> - 发送 - <span class="notify-text-highlight"> - {{ item.options?.templateName || item.notify?.templateId }} - </span> - </div> - </template> - <template v-else-if="item.notify?.notifyType === 'voice'"> - <div> - 通过 - <span class="notify-text-highlight"> - <img - style="width: 18px" - :src="itemNotifyIconMap.get(item.notify?.notifyType)" - /> - 语音 - </span> - {{ item.options?.sendTo ? '向' : '' }} - <span class="notify-text-highlight">{{ item.options?.sendTo || '' }}</span> - 发送 - <span class="notify-text-highlight"> - {{ item.options?.templateName || item.notify?.templateId }} - </span> - </div> - </template> - <template v-else-if="item.notify?.notifyType === 'sms'"> - <div> - 通过 - <span class="notify-text-highlight"> - <img - style="width: 18px" - :src="itemNotifyIconMap.get(item.notify?.notifyType)" - /> - 短信 - </span> - {{ item.options?.sendTo ? '向' : '' }} - <span class="notify-text-highlight">{{ item.options?.sendTo || '' }}</span> - 发送 - <span class="notify-text-highlight"> - {{ item.options?.templateName || item.notify?.templateId }} - </span> - </div> - </template> - <template v-else-if="item.notify?.notifyType === 'webhooks'"> - <div> - 通过 - <span class="notify-text-highlight"> - <img - style="width: 18px" - :src="itemNotifyIconMap.get(item.notify?.notifyType)" - /> - WebHook - </span> - 发送 - <span class="notify-text-highlight"> - {{ item.options?.templateName || item.notify?.templateId }} - </span> - </div> - </template> - </template> - <template v-if="item.executor === 'delay' && show"> - {{ item.options?.name }} - </template> - <template v-if="item.executor === 'device' && show"> - <div v-if="['fixed', 'context'].includes(item.device?.selector)" - style='display: flex; align-items: center;'> - <AIcon :type="typeIconMap[item.device?.message?.messageType ||'INVOKE_FUNCTION']"/> - <span style="padding-left: 4px">{{ item.options?.type }}</span> - <AIcon type="icon-mubiao" style="padding:0 4px"/> - <Ellipsis style='max-width: 120px;margin-right: 12px;'> - {{ item.options?.name }} - </Ellipsis> - <Ellipsis style='max-width: 120px;'> - {{ item.options?.propertiesName }} - </Ellipsis> - <span v-if='!isBoolean(item.options?.propertiesValue) && item.options?.propertiesValue'>为 </span> - <Ellipsis style='max-width: 120px;'> - {{ - `${ - ( - isBoolean( - item.options?.propertiesValue - ) - ? true - : item.options?.propertiesValue - ) - ? `${item.options?.propertiesValue}` - : '' - }` - }} - </Ellipsis> - </div> - <div v-else-if="item.device?.selector === 'tag'"> - <AIcon :type="typeIconMap[item.device?.message?.messageType ||'INVOKE_FUNCTION']"/> - {{ item.options?.type }} - <span>{{ item.options?.tagName }}</span> - 的{{ item.options?.productName }} - {{ item.options?.propertiesName }} - </div> - <div v-else-if="item.device?.selector === 'relation'"> - <AIcon :type="typeIconMap[item.device?.message?.messageType ||'INVOKE_FUNCTION']"/> - {{ item.options?.type }} - 与 - <span>{{ item.options?.triggerName }}</span> - 具有相同 {{ item.options?.relationName }} - 的{{ item.options?.productName }} - 设备的 {{ item.options?.propertiesName }} - </div> - </template> - </div> - </div> - </div> - <div v-if="serial && item.options?.terms?.length && show" class="actions-item-filter-warp"> - <div class="actions-item-filter-warp-tip"> - 满足此条件后执行后续动作 - </div> - <div class="actions-item-filter-overflow"> - <Terms :when="item.options" :border="false" :data="item.terms"/> - </div> - </div> - </div> - </div> -</template> - -<script setup name="Actions"> -import {iconMap, itemNotifyIconMap, typeIconMap} from '@/views/rule-engine/Scene/Save/action/ListItem/util' -import {isBoolean} from 'lodash-es' -import Terms from './Terms/Terms.vue' - -const props = defineProps({ - actions: { - type: Array, - default: () => [] - }, - activeKeys: { // 后端返回已关联的执行动作 - type: Array, - default: () => ([]) - }, - selectedKeys: { // 当前modal中选中的执行动作 - type: Array, - default: () => ([]) - }, - show: { - type: Boolean, - default: false - }, - serial: { - type: Boolean, - default: false - }, - showUnbindBtn: { - type: Boolean, - default: false - } -}) - -const emit = defineEmits(['change', 'select']) - -const onBind = (record) => { - const selected = props.selectedKeys.some(item => item === record.actionId) - emit('change', record.actionId, !selected) -} - -const onSelect = (record) => { - const selected = props.activeKeys.some(item => item === record.actionId) - emit('select', record.actionId, !selected) -} - - -</script> - -<style scoped lang="less"> -.actions-item { - position: relative; - padding: 8px; - border: 1px dashed #999; - border-radius: 2px; - margin-bottom: 8px; - - .item-options-warp { - display: inline-flex; - height: 48px; - border: 1px solid #e0e0e0; - border-radius: 6px; - - .item-options-type { - display: flex; - align-items: center; - justify-content: center; - width: 48px; - background-color: #f0f0f0; - border-radius: 6px 0 0 6px; - cursor: pointer; - font-size: 22px; - } - - .item-options-content { - display: flex; - align-items: center; - padding: 0 8px; - background: #fafafa; - border-radius: 0 6px 6px 0; - cursor: pointer; - - div { - padding: 6px 10px; - color: #333; - font-size: 14px; - line-height: 22px; - background-color: #fff; - border-radius: 22px; - - .notify-text-highlight { - margin-left: 5px; - font-weight: bold; - } - - .notify-img-highlight { - margin: 0 10px; - color: rgba(0, 0, 0, 0.8); - } - } - } - } - - .item-number { - position: absolute; - top: 0; - left: 16px; - font-weight: 800; - transform: translateY(-50%); - } -} - -.actions-item-filter-warp { - position: relative; - margin-bottom: 12px; - margin-top: 16px; - padding: 2px 0; - border: 1px dashed #999; - border-radius: 2px; - - &.filter-border { - padding: 2px 16px; - border-radius: 2px; - } - - .actions-item-filter-warp-tip { - position: absolute; - top: 0; - left: 16px; - z-index: 2; - color: rgba(0, 0, 0, 0.55); - font-weight: 800; - font-size: 14px; - line-height: 1; - background-color: #fff; - transform: translateY(-50%); - } - - .actions-item-filter-overflow { - display: flex; - padding-top: 4px; - overflow-x: auto; - overflow-y: visible; - row-gap: 16px; - } - - .filter-add-button { - width: 100%; - color: rgba(0, 0, 0, 0.3); - text-align: center; - cursor: pointer; - } - - .terms-params { - // display: inline-block; - display: flex; - flex-shrink: 0; - - // &:not(:first-child) { - // margin-bottom: 16px; - // } - - .terms-params-warp { - display: flex; - align-items: baseline; - } - - .term-type-warp { - // display: inline-block; - width: 50px; - margin: 0 16px; - - .term-type { - padding-top: 4px; - padding-bottom: 4px; - border-radius: 2px; - } - } - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/BranchesTabs.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/BranchesTabs.vue deleted file mode 100644 index 93534ca..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/BranchesTabs.vue +++ /dev/null @@ -1,168 +0,0 @@ -<template> - <div> - <div v-if="triggerType ==='device'"> - <a-tabs v-if="branchesGroup.length" v-model:activeKey="activeKey"> - <template v-if="showDetailBtn" #rightExtra> - <a-button @click.stop="show = !show"> - {{ show ? '收起' : '点击查看详情'}} - </a-button> - </template> - <a-tab-pane - v-for="group in branchesGroup" - :tab="group.branchName" - :key="group.key" - > - <div v-for="(branch, index) in group.children"> - <div style="display: flex;align-items: center" v-if="show"> - <span v-if="branch.when" style="padding-right: 12px;font-weight: bold;font-size: 16px;width: 46px; display: inline-block;">{{ index === 0 ? '当' : '否则' }}</span> - <Terms :when="branch.whenOptions" :data="branch.when" /> - - </div> - <div class="branches-shakeLimit" v-if="show"> - <span class="branches-shakeLimit-action">执行</span> - <template v-if="branch.shakeLimit?.enabled"> - <span>开启防抖</span> - <span class="branches-shakeLimit-time">{{ branch.shakeLimit.time }}</span> - <span>秒内发送</span> - <span class="branches-shakeLimit-time">{{ branch.shakeLimit.threshold }}</span> - <span>次及以上时,处理</span> - <span>{{ branch.shakeLimit.alarmFirst ? '第一次' : '最后一次' }}</span> - </template> - <span v-else> - 关闭防抖 - </span> - </div> - <template v-if="branch.serial?.length"> - <div v-if="show" class="branches-tabs-title"> - 串行 - </div> - <Actions :actions="branch.serial" :activeKeys="activeKeys" :selectedKeys="selectedKeys" :show="show" :serial="true" :showUnbindBtn="showUnbindBtn" @change="change" @select="select" /> - </template> - <template v-if="branch.parallel?.length"> - <div v-if="show" class="branches-tabs-title"> - 并行 - </div> - <Actions :actions="branch.parallel" :activeKeys="activeKeys" :selectedKeys="selectedKeys" :show="show" :showUnbindBtn="showUnbindBtn" @change="change" @select="select" /> - </template> - - </div> - </a-tab-pane> - </a-tabs> - - </div> - <div v-else> - <div style="margin: 8px 0; text-align: right" v-if="showDetailBtn"> - <a-button @click.stop="show = !show"> - {{ show ? '收起' : '点击查看详情'}} - </a-button> - </div> - <div v-for="group in branchesGroup"> - <div v-for="(branch, index) in group.children"> - <template v-if="branch.serial?.length"> - <div v-if="show" class="branches-tabs-title"> - 串行 - </div> - <Actions :actions="branch.serial" :activeKeys="activeKeys" :selectedKeys="selectedKeys" :show="show" :serial="true" :showUnbindBtn="showUnbindBtn" @change="change" @select="select" /> - </template> - <template v-if="branch.parallel?.length"> - <div v-if="show" class="branches-tabs-title"> - 并行 - </div> - <Actions :actions="branch.parallel" :activeKeys="activeKeys" :selectedKeys="selectedKeys" :show="show" :showUnbindBtn="showUnbindBtn" @change="change" @select="select" /> - </template> - </div> - </div> - </div> - </div> -</template> - -<script setup name="BranchesTabs"> -import Terms from './Terms/Terms.vue' -import Actions from './Actions.vue' - -const props = defineProps({ - branchesGroup: { - type: Array, - default: () => ([]) - }, - alarmId: { - type: String, - default: undefined - }, - sceneId: { - type: String, - default: undefined - }, - activeKeys: { // 后端返回已关联的执行动作 - type: Array, - default: () => ([]) - }, - selectedKeys: { // 当前modal中选中的执行动作 - type: Array, - default: () => ([]) - }, - show: { - type: Boolean, - default: false - }, - showDetailBtn: { - type: Boolean, - default: true - }, - showUnbindBtn: { - type: Boolean, - default: false - }, - triggerType: { - type: String, - default: undefined - } -}) - -const emit = defineEmits(['change', 'select']) - -const activeKey = ref(props.branchesGroup?.length ? props.branchesGroup[0].key : '') -const show = ref(props.show) - -const change = (id, selected) => { - emit('change', id, selected) -} - -const select = (id, selected) => { - emit('select', id, selected) -} - -</script> - -<style scoped lang="less"> -.branches-tabs-title { - margin-bottom: 8px; - font-size: 14px; - font-weight: 500; -} - -.branches-tabs-alarm { - padding: 12px 16px; - border-radius: 4px; - border: 1px solid #a9a9a9; - display: flex; - justify-content: space-between; - align-items: center; - - &:not(:last-child) { - margin-bottom: 8px; - } -} - -.branches-shakeLimit { - .branches-shakeLimit-action { - font-weight: bold; - padding-right: 8px; - } - - .branches-shakeLimit-time { - padding: 0 8px; - font-weight: bold; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/CardBox.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/CardBox.vue deleted file mode 100644 index 721e57c..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/CardBox.vue +++ /dev/null @@ -1,599 +0,0 @@ -<template> - <div class="card j-table-card-box" @click="click"> - <div - class="card-warp" - :class="{ 'disabled': disabled }" - > - <div class="card-type" > - <div class="card-type-text"> - <img - :height="16" - :src="itemType.icon" - style="margin-right: 5px" - /> - {{itemType.text}} - </div> - </div> - <div - class="card-content" - :style="{ paddingTop: '40px'}" - > - <div class="card-content-bg-item card-content-bg1" :style="bgcColor"></div> - <div class="card-content-bg-item card-content-bg2" :style="bgcColor" ></div> - <div class="card-content-body"> - <!-- 图片 --> - <div class="card-item-avatar"> - <img :src="itemType.img" /> - </div> - - <!-- 内容 --> - <div class="card-item-body"> - <div> - <Ellipsis style="width: calc(100% - 100px)"> - <span style="font-size: 16px; font-weight: 600"> - {{ value.name }} - </span> - </Ellipsis> - <div v-if="showBindTags && activeBranches.length"> - <div style="margin-top: 16px; margin-bottom: 8px" class="card-item-content-text"> - 关联条件 - </div> - <Tags :tags="activeBranches"/> - </div> - <Ellipsis v-else> - <div class="subTitle"> - 说明:{{ - value?.description || - itemType.tip - }} - </div> - </Ellipsis> - - </div> - <div class="condition-name" v-if="showRule"> - <AddButton - v-if="value.options || value.triggerType === 'manual'" - style='width: 100%;padding: 8px 12px; border-radius: 4px' - :showCircular="false" - > - - <DeviceTitle v-if="value.triggerType === 'device'" :options='value.options?.trigger'/> - <template v-if="value.triggerType === 'timer'" #extra> - <span style="padding-right: 10px;"> - 系统时间到达 - </span> - </template> - <TimerTitle v-if="value.triggerType === 'timer'" :options='value.options?.trigger'/> - <span v-if="value.triggerType === 'manual'"> - 系统在接收到手动触发指令时,触发场景 - </span> - </AddButton> - </div> - </div > - - <div v-if="showMask && activeBranches.length && !isInvalid" v-show="!showBranchesVisible" class="card-mask" :class="{ 'branches-tabs-mask': showBranchesVisible}" :style="maskStyle"> - <div class="mask-content"> - <slot name="mask"> - <div> - 当前告警已关联: - </div> - <Tags :tags="activeBranches" :styles="{ justifyContent: 'center' }"/> - </slot> - </div> - </div> - <div v-if="isInvalid" class="card-mask mask-full" style="background-color: rgba(0, 0, 0, 0.2);" @click.stop="jumpView"> - <a-button style="font-size: 16px;" type="link" >无效数据,请重新保存场景</a-button> - </div> - </div> - <div class="card-content-tabs" v-if="showBranches && showBranchesVisible" @click.stop> - <BranchesTabs - :branchesGroup="branchesGroup" - :alarmId="alarmId" - :branchesId="value.id" - :activeKeys="activeKeys" - :selectedKeys="selectedKeys" - :triggerType="value.triggerType" - :key="value.id" - @change="bindAlarm" - /> - </div> - <div - class="card-state" - :style="{ backgroundColor: getHexColor(statusNames[status]) }" - > - <div class="card-state-content"> - <BadgeStatus - :status="status" - :text="statusText" - :statusNames="statusNames" - ></BadgeStatus> - </div> - </div> - </div> - - <slot></slot> - </div> - </div> -</template> - -<script setup lang="ts" name="SceneCardBox"> -import BadgeStatus from '@/components/BadgeStatus/index.vue'; -import color, {getHexColor} from '@/components/BadgeStatus/color'; -import DeviceTitle from '@/views/rule-engine/Scene/Save/components/Title.vue' -import TimerTitle from '@/views/rule-engine/Scene/Save/Timer/Title.vue' -import AddButton from '@/views/rule-engine/Scene/Save/components/AddButton.vue' -import BranchesTabs from './BranchesTabs.vue' -import {PropType} from 'vue'; -import {handleActiveBranches, handleGroupAndFilter, typeMap} from './utils' -import {useMenuStore} from "@/store/menu"; -import Tags from './tags.vue' - -type EmitProps = { - (e: 'click'): void; - (e: 'change', key: string, selected: boolean): void; - (e: 'reload'): void; -}; - -const emit = defineEmits<EmitProps>(); - -const props = defineProps({ - value: { - type: Object as PropType<Record<string, any>>, - default: () => ({}), - }, - statusText: { - type: String, - default: '正常', - }, - status: { - type: [String, Number] as PropType<string | number>, - default: 'default', - }, - statusNames: { - type: Object as PropType<Record<any, any>>, - default: () => ({ - started: 'processing', - disable: 'error', - }) - }, - disabled: { - type: Boolean, - default: false, - }, - alarmId: { - type: String, - default: undefined - }, - activeKeys: { // 后端返回已关联的执行动作 - type: Array, - default: () => ([]) - }, - selectedKeys: { // 当前modal中选中的执行动作 - type: Array, - default: () => ([]) - }, - showMask: { - type: Boolean, - default: false - }, - showBranches: { - type: Boolean, - default: true - }, - showBindTags: { - type: Boolean, - default: false - }, - showRule: { - type: Boolean, - default: true - }, - maskStyle: { - type: Object, - default: undefined - } -}); - -const isInvalid = ref(false) -const menuStory = useMenuStore(); -const bgcColor = computed(() => { - const key = props.statusNames[props.status] - const _color = color[key] || color.default; - return { - background: `linear-gradient( - 188.4deg, - rgba(${_color}, 0.03) 30%, - rgba(${_color}, 0) 80% - )` - } -}) - -const showBranchesVisible = ref(false) - -const itemType = computed(() => { - return typeMap.get(props.value.triggerType) || {} -}) - -const branchesGroup = computed(() => { - return handleGroupAndFilter(props.value.branches, props.value.options?.when || []) -}) - -const activeBranches = computed(() => { - const { data, invalid } = handleActiveBranches(branchesGroup.value, props.activeKeys) - - isInvalid.value = invalid - return data -}) - -const bindAlarm = (key: string, selected: boolean) => { - emit('change', key, selected) -} - -const click = () => { - if (!isInvalid.value) { - emit('click') - onShowBranchesTabs() - } -} - -const jumpView = () => { - - const url = menuStory.menus['rule-engine/Scene/Save']?.path; - const tab: any = window.open( - `${window.location.origin + window.location.pathname}#${url}?triggerType=${props.value.triggerType}&id=${props.value.id}`, - ); - tab.onTabSaveSuccess = (value: any) => { - if (value.success) { - emit('reload') - } - }; -} - -const onShowBranchesTabs = () => { - showBranchesVisible.value = !showBranchesVisible.value -} - -watch(() => props.value.id, () => { - showBranchesVisible.value = false -}) - -</script> - -<style scoped lang="less"> -.card { - width: 100%; - background-color: #fff; - - .checked-icon { - position: absolute; - right: -22px; - bottom: -22px; - z-index: 2; - width: 44px; - height: 44px; - color: #fff; - background-color: #2f54eb; - transform: rotate(-45deg); - - > div { - position: relative; - height: 100%; - transform: rotate(45deg); - - > span { - position: absolute; - top: 6px; - left: 6px; - font-size: 12px; - } - } - } - - .card-warp { - position: relative; - border: 1px solid #e6e6e6; - overflow: hidden; - cursor: pointer; - - &.disabled { - filter: grayscale(100%); - cursor: not-allowed; - } - - &.active { - position: relative; - border: 1px solid #2f54eb; - } - - .card-type { - position: absolute; - top: 0; - left: -15px; - height: 32px; - padding: 0 30px; - color: rgba(0, 0, 0, 0.65); - line-height: 32px; - background-color: rgba(0, 0, 0, 0.06); - transform: skewX(-45deg); - - .card-type-text { - display: flex; - align-items: center; - justify-content: center; - transform: skewX(45deg); - } - } - - .card-content { - position: relative; - padding: 30px 12px 30px 30px; - overflow: hidden; - min-height: 170px; - - .card-item-avatar { - margin-right: 16px; - display: flex; - align-items: center; - } - - .card-item-body { - display: flex; - flex-direction: column; - flex-grow: 1; - width: 0; - - .ant-row { - margin-top: 19px; - } - - .condition-name { - margin-top: 10px; - } - } - - .card-state { - position: absolute; - top: 30px; - right: -12px; - display: flex; - justify-content: center; - width: 100px; - padding: 2px 0; - background-color: rgba(#5995f5, 0.15); - transform: skewX(45deg); - - &.success { - background-color: @success-color-deprecated-bg; - } - - &.warning { - background-color: rgba(#ff9000, 0.1); - } - - &.error { - background-color: rgba(#e50012, 0.1); - } - - .card-state-content { - transform: skewX(-45deg); - } - } - - :deep(.card-item-content-title) { - cursor: pointer; - font-size: 16px; - font-weight: 700; - color: @primary-color; - width: calc(100% - 100px); - overflow: hidden; - white-space: nowrap; - text-overflow: ellipsis; - } - - :deep(.card-item-heard-name) { - font-weight: 700; - font-size: 16px; - margin-bottom: 12px; - } - - :deep(.card-item-content-text) { - color: rgba(0, 0, 0, 0.7); - font-size: 12px; - } - } - - .card-content-top-line { - &::before { - position: absolute; - top: 0; - left: 30px + 10px; - display: block; - width: 15%; - min-width: 64px; - height: 2px; - background-image: url('/images/rectangle.png'); - background-repeat: no-repeat; - background-size: 100% 100%; - content: ' '; - } - } - - .card-content-bg-item { - position: absolute; - right: -5%; - height: 320px; - top: 0; - background: linear-gradient(188.4deg, - rgba(229, 0, 18, 0.03) 22.94%, - rgba(229, 0, 18, 0) 94.62%); - transform: skewX(-15deg); - - &.card-content-bg1 { - width: 44.65%; - } - - &.card-content-bg2 { - width: calc(44.65% + 34px); - } - } - - .card-content-body { - display: flex; - position: relative; - z-index: 99; - } - - - .card-mask { - position: absolute; - top: -40px; - left: -30px; - bottom: -30px; - right: -12px; - z-index: 99; - display: flex; - justify-content: center; - font-size: 16px; - color: #fff; - padding-top: 10px; - background-color: rgba(#000, 0.45); - cursor: pointer; - transition: background-color 0.3s; - - .mask-content { - display: flex; - align-items: center; - justify-content: center; - width: 100%; - height: 100%; - padding: 24px !important; - flex-direction: column; - } - - &.mask-hover:hover { - background-color: rgba(#000, 0.01); - color: transparent; - - .mask-content { - display: none; - } - } - - - - &.branches-tabs-mask { - bottom: 0; - } - } - } - - &.item-active { - position: relative; - color: #2f54eb; - - .checked-icon { - display: block; - } - - .card-warp { - border: 1px solid #2f54eb; - } - } - - .card-tools { - display: flex; - margin-top: 8px; - - .card-button { - display: flex; - flex-grow: 1; - - & > :deep(span, button) { - width: 100%; - border-radius: 0; - } - - :deep(button) { - width: 100%; - border-radius: 0; - background: #f6f6f6; - border: 1px solid #e6e6e6; - color: #2f54eb; - - &:hover { - background-color: @primary-color-hover; - border-color: @primary-color-hover; - - span { - color: #fff !important; - } - } - - &:active { - background-color: @primary-color-active; - border-color: @primary-color-active; - - span { - color: #fff !important; - } - } - } - - &:not(:last-child) { - margin-right: 8px; - } - - &.delete { - flex-basis: 60px; - flex-grow: 0; - - :deep(button) { - background: @error-color-deprecated-bg; - border: 1px solid @error-color-outline; - - span { - color: @error-color !important; - } - - &:hover { - background-color: @error-color-hover; - - span { - color: #fff !important; - } - } - - &:active { - background-color: @error-color-active; - - span { - color: #fff !important; - } - } - } - } - - :deep(button[disabled]) { - background: @disabled-bg; - border-color: @disabled-color; - - span { - color: @disabled-color !important; - } - - &:hover { - background-color: @disabled-active-bg; - } - - &:active { - background-color: @disabled-active-bg; - } - } - - // :deep(.ant-tooltip-disabled-compatible-wrapper) { - // width: 100%; - // } - } - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/ActionsFilter.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/ActionsFilter.vue deleted file mode 100644 index a33ed78..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/ActionsFilter.vue +++ /dev/null @@ -1,71 +0,0 @@ -<template> - <div class="terms-warp"> - <j-scrollbar> - <div class="terms-content"> - <div class="terms-group" v-for="(group, index) in when.terms"> - - <div v-if="group.termType && index !== 0" class="dropdown-button type"> - {{ group.termType }} - </div> - <div class="terms-group-content" > - <WhenItem v-for="(item, groupIndex) in group.terms" :type="item.termType" :value="item" :showType="groupIndex !== 0" /> - </div> - </div> - </div> - </j-scrollbar> - </div> -</template> - -<script setup name="ActionFIlter"> -import WhenItem from "./WhenItem.vue"; - -const props = defineProps({ - when: { - type: Object, - default: () => ({}) - } -}) -</script> - -<style scoped lang="less"> -.terms-warp { - flex: 1 1 0; - min-width: 0; -} - -.terms-content { - display: flex; - gap: 12px; -} - -.terms-group { - display: flex; - gap: 8px; - align-items: center; - flex-shrink: 0; - user-select: none; - - .terms-group-content { - position: relative; - display: flex; - padding: 8px 8px; - border: 1px dashed #e0e0e0; - border-radius: 6px; - row-gap: 16px; - } -} - -.dropdown-button { - display: flex; - align-items: center; - padding: 6px 8px; - border: 1px solid #d9d9d9; - border-radius: 8px; - cursor: pointer; - - &.type { - padding: 5px 10px; - height: 36px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/Terms.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/Terms.vue deleted file mode 100644 index ea4793b..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/Terms.vue +++ /dev/null @@ -1,87 +0,0 @@ -<template> - <div class="terms-warp"> - <j-scrollbar> - <div class="terms-content"> - <div class="terms-group" v-for="(group, index) in when.terms"> - - <div v-if="group.termType && index !== 0" class="dropdown-button type"> - {{ group.termType }} - </div> - <div class="terms-group-content" :class="{'group-content-border': border, 'group-content-bg': !border}"> - <WhenItem v-for="(item, groupIndex) in group.terms" :type="item.termType" :value="item" :showType="groupIndex !== 0" :data="data[index]?.terms[groupIndex]" /> - </div> - </div> - </div> - </j-scrollbar> - </div> -</template> - -<script setup name="Terms"> -import WhenItem from "./WhenItem.vue"; - -const props = defineProps({ - when: { - type: Object, - default: () => ({}) - }, - border: { - type: Boolean, - default: true - }, - data: { - type: Array, - default: () => [] - } -}) - -</script> - -<style scoped lang="less"> -.terms-warp { - flex: 1 1 0; - min-width: 0; -} - -.terms-content { - display: flex; - gap: 12px; -} - -.terms-group { - display: flex; - gap: 8px; - align-items: center; - flex-shrink: 0; - user-select: none; - - .terms-group-content { - position: relative; - display: flex; - padding: 8px 8px; - row-gap: 16px; - border-radius: 6px; - - &.group-content-border { - border: 1px dashed #e0e0e0; - } - - &.group-content-bg { - background-color: #fafafa; - } - } -} - -.dropdown-button { - display: flex; - align-items: center; - padding: 6px 8px; - border: 1px solid #d9d9d9; - border-radius: 8px; - cursor: pointer; - - &.type { - padding: 5px 10px; - height: 36px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/WhenItem.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/WhenItem.vue deleted file mode 100644 index 35f7636..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/Terms/WhenItem.vue +++ /dev/null @@ -1,123 +0,0 @@ -<template> - <div class="terms-when-item"> - <div v-if="showType" class="dropdown-button type"> - {{ TermTypeMap[value[3]] || value[3] }} - </div> - <div class="dropdown-button column"> - <AIcon type='icon-zhihangdongzuoxie-1' /> - {{ value[0] }} - </div> - <div v-if="showAlarm" class="dropdown-button alarm"> - {{ value[4] }} - </div> - <div class="dropdown-button termType"> - {{ TermsTypeMap[value[1]] || value[1] }} - </div> - <div v-if="_value" class="dropdown-button value"> - <AIcon type='icon-canshu' /> - {{ _value[0] }} - </div> - <div v-if="value[2][1]" class="dropdown-button value"> - <AIcon type='icon-canshu' /> - {{ _value[1] }} - </div> - - </div> -</template> - -<script setup name="WhenItem"> -import { TermTypeMap } from '../utils' - -const TermsTypeMap = { - eq: '等于', - neq: '不等于', - gt: '大于', - gte: '大于等于', - lt: '小于', - lte: '小于等于', - btw: '在...之间', - nbtw: '不在...之间', - time_gt_now: '距离当前时间大于', - time_lt_now: '距离当前时间小于', - in: '在...之中', - nin: '不在...之中', - like: '包含', - nlike: '不包含', - notnull: '不为空', - isnull: '为空' -}; - -const props = defineProps({ - value: { - type: Array, - default: () => [] - }, - showType: { - type: Boolean, - default: false - }, - data: { - type: Array, - default: () => [] - } -}) - -const showAlarm = computed(() => { - return props.data.terms?.length -}) - -const _value = computed(() => { - if (['in', 'nin','contains_all', 'contains_any', 'not_contains', 'nbtw', 'btw',].includes(props.data.termType)) { - return props.data.value.value - } - - return props.value[2] -}) - -</script> - -<style scoped lang="less"> -.terms-when-item { - display: flex; - gap: 6px; -} - -.dropdown-button { - display: flex; - align-items: center; - padding: 6px 8px; - border: 1px solid #d9d9d9; - border-radius: 8px; - cursor: pointer; - - &.column { - color: #00a4fe; - background-color: rgba(154, 219, 255, 0.3); - border-color: rgba(0, 164, 254, 0.3); - } - - &.termType { - color: #13C2C2; - background: rgba(135, 232, 222, 0.3); - border: 1px solid rgba(54, 207, 201, 0.4); - } - - &.value { - color: #692ca7; - background-color: rgba(188, 125, 238, 0.1); - border-color: rgba(188, 125, 238, 0.5); - } - - &.type { - padding: 5px 10px; - margin-left: 8px; - } - - &.alarm { - color: #2F54EB; - background: rgba(163, 202, 255, 0.3); - border: 1px solid rgba(47, 84, 235, 0.3); - } -} - -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/tags.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/tags.vue deleted file mode 100644 index bc8d108..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/Save/tags.vue +++ /dev/null @@ -1,70 +0,0 @@ -<template> - <ResizeObserver - :onResize="onResize" - > - <div class="tags-warp" :style="styles"> - <div v-for="item in tagOptions" class="tags-item"> - <Ellipsis> - {{ item }} - </Ellipsis> - </div> - <div v-if="showMore" class="tags-item"> - <AIcon type="EllipsisOutlined"/> - </div> - </div> - </ResizeObserver> -</template> - -<script setup name="tags"> -import ResizeObserver from 'ant-design-vue/lib/vc-resize-observer'; - -const props = defineProps({ - tags: { - type: Array, - default: () => ([]) - }, - styles: { - type: Object, - default: () => ({}) - } -}) - -const showMore = ref(false) -const warpWidth = ref(0) - -const onResize = ({ width }) => { - warpWidth.value = width -} - -const tagOptions = computed(() => { - showMore.value = false - if (warpWidth.value) { - let max = warpWidth.value / (80 + 8) - if (max > props.tags.length) { - return props.tags - } else { - showMore.value = true - return props.tags.slice(0, max - 1) - } - } - return props.tags -}) - -</script> - -<style scoped lang="less"> -.tags-warp { - display: flex; - gap: 8px; - width: 100%; - - .tags-item { - text-align: center; - color: #4096FF; - background-color: #F0F0F0; - border-radius: 2px; - padding: 4px 4px; - width: 80px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Configuration/Save/Scene/SceneDrawer.vue b/src/views/rule-engine/Alarm/Configuration/Save/Scene/SceneDrawer.vue deleted file mode 100644 index 9496320..0000000 --- a/src/views/rule-engine/Alarm/Configuration/Save/Scene/SceneDrawer.vue +++ /dev/null @@ -1,131 +0,0 @@ -<template> - <a-drawer - visible - title="关联场景联动详情" - :width="600" - :closable="false" - :keyboard="false" - @close="onCancel" - > - <div class="scene-warp"> - <TitleComponent data="基本信息" /> - <j-descriptions size="small" :column="2" bordered> - <j-descriptions-item label="场景名称"> - <j-ellipsis>{{ detail.name }}</j-ellipsis> - </j-descriptions-item> - <j-descriptions-item label="场景类型">{{ itemType.text }}</j-descriptions-item> - </j-descriptions> - <TitleComponent data="关联详情" style="margin: 12px 0;" /> - <div class="branches-warp"> - <BranchesTabs - :branchesGroup="branchesGroup" - :alarmId="id" - :branchesId="detail.id" - :activeKeys="activeKeys" - :show="true" - :showDetailBtn="false" - :showUnbindBtn="true" - :triggerType="detail.triggerType" - @select="handleBind" - /> - <div v-if="loading" class="branches-loading"> - <a-spin /> - </div> - </div> - </div> - </a-drawer> -</template> - -<script setup name="SceneDrawer"> -import {useRequest} from "@/hook"; -import {queryBindScene} from "@/api/rule-engine/configuration"; -import {handleGroupAndFilter, typeMap} from "./Save/utils"; -import BranchesTabs from './Save/BranchesTabs.vue' -import { unBindAlarm, bindScene } from "@/api/rule-engine/configuration"; - -const props = defineProps({ - id: { - type: String, - default: undefined - }, - detail: { - type: Object, - default: () => ({}) - } -}); - -const emit = defineEmits(['cancel']) - -const loading = ref(false) - -const itemType = computed(() => { - return typeMap.get(props.detail.triggerType) || {} -}) - -const branchesGroup = computed(() => { - return handleGroupAndFilter(props.detail.branches, props.detail.options?.when || []) -}) - -const { data: activeKeys } = useRequest(queryBindScene, { - defaultParams: { terms: [{ column: 'alarmId', value: props.id}]}, - onSuccess(res) { - const activeMap = res.result.data.reduce((prev, next) => { - if (props.detail.id === next.ruleId) { - prev.push(next.branchIndex) - } - return prev - }, []) - return activeMap || [] - }, - defaultValue: [] -}) - -const onCancel = () => { - emit('cancel') -} - -const handleBind = (id, selected) => { - loading.value = true - if (selected) { // - bindScene([{ - alarmId: props.id, - ruleId: props.detail.id, - branchIndex: id - }]).then(res => { - activeKeys.value.push(id) - }).finally(() => { - loading.value = false - }) - } else { - unBindAlarm(props.detail.id, props.id, [id]).then(res => { - activeKeys.value = activeKeys.value.filter(key => key !== id) - }).finally(() => { - loading.value = false - }) - } -} - -</script> - -<style scoped lang="less"> -.scene-warp { - display: flex; - flex-direction: column; - height: 100%; - - .branches-warp { - flex: 1 1 0; - min-height: 0; - overflow-y: auto; - position: relative; - - .branches-loading { - position: absolute; - top: 0; - left: 0; - height: 0; - padding: 20%; - } - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Log/TabComponent/components/DetailDrawer.vue b/src/views/rule-engine/Alarm/Log/TabComponent/components/DetailDrawer.vue deleted file mode 100644 index ae3a5c8..0000000 --- a/src/views/rule-engine/Alarm/Log/TabComponent/components/DetailDrawer.vue +++ /dev/null @@ -1,148 +0,0 @@ -<template> - <a-drawer - :visible="true" - width="1000" - :destroyInactiveTabPane="true" - @close="closeDrawer" - > - <template #title> - <div class="alarmInfo"> - <div> - <div class="alarmTitle"> - <span class="alarmType">{{ - typeMap.get(AlarmData?.targetType) - }}</span> - <div> - <Ellipsis style="max-width: 200px">{{ - AlarmData?.alarmName - }}</Ellipsis> - </div> - <LevelIcon :level="AlarmData.level"></LevelIcon> - </div> - <div style="font-size: 12px; margin-left: 10px; margin-top:5px"> - {{ AlarmData?.description || '暂无说明' }} - </div> - </div> - <div class="alarmInfoRight"> - <div> - <BadgeStatus - :status="AlarmData?.state.value" - :statusNames="{ - warning: 'error', - normal: 'default', - }" - > - </BadgeStatus - ><span> - {{ AlarmData?.state.text }} - </span> - <a-button - v-if="AlarmData?.state.value === 'warning'" - type="link" - @click="dealAlarm" - >处理</a-button - > - </div> - </div> - </div> - </template> - - <a-radio-group - v-model:value="activeKey" - button-style="solid" - style="margin: 20px 0" - > - <a-radio-button value="record">处理记录</a-radio-button> - <a-radio-button value="logs">告警日志</a-radio-button> - </a-radio-group> - <Record - :currentId="AlarmData.id" - ref="RecordRef" - v-if="activeKey === 'record'" - ></Record> - <Log - :currentId="AlarmData.id" - :configId="AlarmData.alarmConfigId" - v-else - /> - </a-drawer> - <SolveComponent - v-if="solveVisible" - @closeSolve="closeSolve" - @refresh="refresh" - :data="AlarmData" - /> -</template> - -<script setup name="LogDrawer"> -import { query } from '@/api/rule-engine/log'; -import Record from './Record.vue'; -import Log from './Log.vue'; -import SolveComponent from '../../SolveComponent/index.vue'; -const props = defineProps({ - logData: { - type: Object, - default: {}, - }, - typeMap: { - type: Object, - default: {}, - }, - levelMap: { - type: Object, - default: {}, - }, -}); -const emit = defineEmits(['closeDrawer', 'refreshTable']); -const solveVisible = ref(false); -const RecordRef = ref(); -const AlarmData = ref(props.logData); -const activeKey = ref('record'); -const closeDrawer = () => { - emit('closeDrawer'); -}; -const dealAlarm = () => { - solveVisible.value = true; -}; -const closeSolve = () => { - solveVisible.value = false; -}; - -const refresh = async () => { - solveVisible.value = false; - const res = await query({ - terms: [ - { - column: 'id', - value: props.logData.id, - type: 'and', - }, - ], - }); - if (res.success) { - AlarmData.value = res.result.data[0]; - if (activeKey.value === 'record') { - RecordRef?.value.refreshRecord(); - } - } - emit('refreshTable'); -}; -</script> -<style lang="less" scoped> -.alarmInfo { - display: flex; - justify-content: space-between; - .alarmType { - background-color: #e6f4ff; - padding: 2px 8px; - margin-right: 10px; - color: #1677ff; - } - .alarmTitle { - display: flex; - } - .alarmInfoRight { - text-align: right; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Log/TabComponent/components/Log.vue b/src/views/rule-engine/Alarm/Log/TabComponent/components/Log.vue deleted file mode 100644 index 3513d9a..0000000 --- a/src/views/rule-engine/Alarm/Log/TabComponent/components/Log.vue +++ /dev/null @@ -1,164 +0,0 @@ -<template> - <a-table - :dataSource="dataSource" - :columns="columns" - :pagination="false" - bordered - :scroll="{ y: 'calc(100vh - 300px)' }" - > - <template #bodyCell="{ column, text, record }"> - <template v-if="column.dataIndex === 'alarmTime'" - ><span - >{{ dayjs(text).format('YYYY-MM-DD HH:mm:ss') - }}</span - ></template> - <template v-if="column.dataIndex === 'sourceId'"> - <Ellipsis> - 设备ID: - <span class="deviceId" @click="() => gotoDevice(text)">{{ - text - }}</span></Ellipsis - > - </template> - <template - v-if=" - column.dataIndex === 'triggerDesc' || - column.dataIndex === 'actualDesc' - " - ><Ellipsis>{{ text }}</Ellipsis></template - > - <template v-if="column.dataIndex === 'action'"> - <a-button type="link" @click="() => showDetail(record)"> - <template #icon> - <AIcon type="EyeOutlined"></AIcon> - </template> </a-button> - </template> - </template></a-table - > - <div class="tableBottom"> - <a-button - v-if="exceed" - class="moreBtn" - type="link" - @click="gotoAlarmLog" - >查看更多 ></a-button - > - <span v-else-if="dataSource.length">已展示全部数据</span> - </div> - <LogDetail v-if="visibleDetail" :data="current" @close="close" /> -</template> - -<script setup> -import { queryLogList } from '@/api/rule-engine/log'; -import dayjs from 'dayjs'; -import { useMenuStore } from 'store/menu'; -import LogDetail from './LogDetail.vue'; -const props = defineProps({ - currentId: { - type: String, - default: '', - }, - configId: { - type: String, - default: '', - }, -}); -const menuStory = useMenuStore(); -const exceed = ref(); -const visibleDetail = ref(false); -const dataSource = ref([]); -const current = ref(); -const columns = [ - { - title: '告警时间', - dataIndex: 'alarmTime', - key: 'alarmTime', - }, - { - title: '触发条件', - dataIndex: 'triggerDesc', - key: 'triggerDesc', - }, - { - title: '告警源', - dataIndex: 'sourceId', - key: 'sourceId', - }, - { - title: '告警原因', - dataIndex: 'actualDesc', - key: 'actualDesc', - }, - { - title: '操作', - dataIndex: 'action', - key: 'action', - width:100 - } -]; -const queryData = async () => { - const res = await queryLogList(props.configId, { - pageIndex: 0, - pageSize: 51, - terms: [ - { - column: 'alarmRecordId', - termType: 'eq', - value: props.currentId, - type: 'and', - }, - ], - sorts: [ - { - name: 'alarmTime', - order: 'desc', - }, - ], - }); - if (res.success) { - if (res.result.data?.length > 50) { - exceed.value = true; - dataSource.value = res.result.data.slice(0, 50); - } else { - exceed.value = false; - dataSource.value = res.result.data; - } - } -}; - -const gotoDevice = (id) => { - menuStory.jumpPage('device/Instance/Detail', { id, tab: 'Running' }); -}; -const showDetail = (data) => { - visibleDetail.value = true; - current.value = data; -}; -const close = () => { - visibleDetail.value = false; -}; -const gotoAlarmLog = () => { - menuStory.jumpPage(`rule-engine/Alarm/Log/Detail`, { - id: props.currentId, - }); -}; -onMounted(() => { - queryData(); -}); -</script> -<style lang="less" scoped> -.tableBottom { - text-align: center; - position: relative; - height: 50px; - margin-top: 20px; - .moreBtn { - position: absolute; - right: 50%; - top: 10px; - } -} -.deviceId { - cursor: pointer; - color:#4096FF; -} -</style> diff --git a/src/views/rule-engine/Alarm/Log/TabComponent/components/LogDetail.vue b/src/views/rule-engine/Alarm/Log/TabComponent/components/LogDetail.vue deleted file mode 100644 index 6d8639e..0000000 --- a/src/views/rule-engine/Alarm/Log/TabComponent/components/LogDetail.vue +++ /dev/null @@ -1,78 +0,0 @@ -<template> - <j-modal - visible - okText="确定" - cancelText="取消" - :width="1000" - title="告警日志" - :closable="true" - @ok="closeModal" - @cancel="closeModal" - > - <div class="alarmInfo"> - <span class="alarmTitle">{{ data?.alarmConfigName }}</span> - <span class="alarmTime">{{ - dayjs(data?.alarmTime).format('YYYY-MM-DD HH:mm:ss') - }}</span> - </div> - <j-descriptions bordered :column="2"> - <j-descriptions-item label="触发条件" :span="2">{{ - data?.triggerDesc - }}</j-descriptions-item> - <j-descriptions-item label="告警原因" :span="2">{{ - data?.actualDesc - }}</j-descriptions-item> - <j-descriptions-item label="告警源" :span="2"> - 设备ID:<a-button - type="link" - @click="() => gotoDevice(data?.sourceId)" - >{{ data?.sourceId }}</a-button - > - </j-descriptions-item> - <j-descriptions-item label="告警流水" :span="2" - ><div style="max-height: 500px; overflow-y: auto"> - <JsonViewer - :value="JSON.parse(data?.alarmInfo)" - :expanded="true" - :expandDepth="4" - ></JsonViewer></div - ></j-descriptions-item> - </j-descriptions> - </j-modal> -</template> - -<script setup> -import dayjs from 'dayjs'; -import JsonViewer from 'vue-json-viewer'; -import { useMenuStore } from 'store/menu'; -const props = defineProps({ - data: Object, -}); -const menuStory = useMenuStore(); -const runningWater = computed(() => { - return JSON.parse(props.data?.alarmInfo); -}); - -const emit = defineEmits(['close']); -const closeModal = () => { - emit('close'); -}; -const gotoDevice = (id) => { - menuStory.jumpPage('device/Instance/Detail', { id, tab: 'Running' }); -}; -</script> -<style lang="less" scoped> -.alarmInfo { - display: flex; - justify-content: space-between; - margin-bottom: 8px; - .alarmTitle { - font-weight: 600; - font-size: 16px; - color: #1a1a1a; - } - .alarmTime { - font-size: 14px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Log/TabComponent/components/Record.vue b/src/views/rule-engine/Alarm/Log/TabComponent/components/Record.vue deleted file mode 100644 index b1418d6..0000000 --- a/src/views/rule-engine/Alarm/Log/TabComponent/components/Record.vue +++ /dev/null @@ -1,134 +0,0 @@ -<template> - <a-table - :dataSource="dataSource" - :columns="columns" - :pagination="false" - bordered - :scroll="{ y: 'calc(100vh - 260px)' }" - > - <template #bodyCell="{ column, text, record }"> - <template - v-if="['alarmTime', 'handleTime'].includes(column.dataIndex)" - > - {{ dayjs(text).format('YYYY-MM-DD HH:mm:ss') }} - </template> - <template v-if="column.dataIndex === 'duration'"> - <Duration :data="record" /> - </template> - <template v-if="column.dataIndex === 'handleType'"> - {{ text?.text }} - </template> - <template v-if="column.dataIndex === 'description'"> - <Ellipsis> - {{ text || '--' }} - </Ellipsis> - </template> - </template> - </a-table> - <div class="tableBottom"> - <a-button - v-if="exceed" - class="moreBtn" - type="link" - @click="gotoAlarmRecord" - >查看更多 ></a-button - ><span v-else-if="dataSource.length">已展示全部数据</span> - </div> -</template> - -<script setup> -import { queryHandleHistory } from '@/api/rule-engine/log'; -import dayjs from 'dayjs'; -import { useMenuStore } from 'store/menu'; -import { defineExpose } from 'vue'; -import Duration from '../../components/Duration.vue'; -const props = defineProps({ - currentId: { - type: String, - default: '', - }, -}); -const exceed = ref(); -const dataSource = ref([]); -const menuStory = useMenuStore(); -const columns = [ - { - title: '告警时间', - dataIndex: 'alarmTime', - key: 'alarmTime', - }, - { - title: '处理时间', - dataIndex: 'handleTime', - key: 'handleTime', - }, - { - title: '告警持续时长', - dataIndex: 'duration', - key: 'duration', - }, - { - title: '处理类型', - dataIndex: 'handleType', - key: 'handleType', - }, - { - title: '处理结果', - dataIndex: 'description', - key: 'description', - }, -]; -const queryList = async () => { - const res = await queryHandleHistory({ - sorts: [{ name: 'createTime', order: 'desc' }], - terms: [ - { - column: 'alarmRecordId', - termType: 'eq', - value: props.currentId, - type: 'and', - }, - ], - pageIndex: 0, - pageSize: 51, - }); - if (res.success) { - if (res.result.data?.length > 50) { - exceed.value = true; - dataSource.value = res.result.data.slice(0, 50); - } else { - exceed.value = false; - dataSource.value = res.result.data; - } - } -}; -const gotoAlarmRecord = () => { - menuStory.jumpPage( - 'rule-engine/Alarm/Log/Record', - {}, - { id: props.currentId }, - ); -}; -const refreshRecord = () => { - queryList(); -}; -defineExpose({ - refreshRecord, -}); -onMounted(() => { - queryList(); -}); -</script> -<style lang="less" scoped> -.tableBottom { - text-align: center; - position: relative; - height: 50px; - margin-top: 20px; - .moreBtn { - position: absolute; - right: 50%; - top: 10px; - } -} -</style> diff --git a/src/views/rule-engine/Alarm/Log/components/Duration.vue b/src/views/rule-engine/Alarm/Log/components/Duration.vue deleted file mode 100644 index b64401e..0000000 --- a/src/views/rule-engine/Alarm/Log/components/Duration.vue +++ /dev/null @@ -1,43 +0,0 @@ -<template> - <div>{{ duration }}</div> -</template> - -<script setup name="Duration"> -const props = defineProps({ - data: { - type: Object, - default: {}, - }, -}); -const duration = ref(); -import dayjs from 'dayjs'; -const calculateDuration = (startTime, endTime) => { - const diffInSeconds = endTime.diff(startTime, 'second'); - let result; - - if (diffInSeconds < 60) { - result = `${diffInSeconds.toFixed(1)} s`; - } else if (diffInSeconds < 3600) { - result = `${(diffInSeconds / 60).toFixed(1)} min`; - } else { - result = `${(diffInSeconds / 3600).toFixed(1)} h`; - } - return result; -}; -watch( - () => props.data, - () => { - duration.value = calculateDuration( - dayjs(props.data.alarmTime), - props.data?.state?.value === 'warning' - ? dayjs() - : dayjs(props.data?.handleTime), - ); - }, - { - deep: true, - immediate: true, - }, -); -</script> -<style lang="less" scoped></style> diff --git a/src/views/rule-engine/Scene/Save/action/Device/device/util.ts b/src/views/rule-engine/Scene/Save/action/Device/device/util.ts deleted file mode 100644 index 33b18b7..0000000 --- a/src/views/rule-engine/Scene/Save/action/Device/device/util.ts +++ /dev/null @@ -1,32 +0,0 @@ -import {getImage} from "@/utils/comm"; - -export const TypeMap = { - fixed: { - label: '自定义', - value: 'fixed', - image: getImage('/scene/device-custom.png'), - tip: '自定义选择当前产品下的任意设备', - disabled: false - }, - relation: { - label: '按关系', - value: 'relation', - image: getImage('/scene/device-relation.png'), - tip: '选择与触发设备具有相同关系的设备', - disabled: false - }, - tag: { - label: '按标签', - value: 'tag', - image: getImage('/scene/device-tag.png'), - tip: '按标签选择产品下具有特定标签的设备', - disabled: false - }, - context: { - label: '按变量', - value: 'context', - image: getImage('/scene/device-variable.png'), - tip: '选择设备ID为上游变量值的设备', - disabled: false - }, -} diff --git a/src/views/rule-engine/Scene/Save/action/ListItem/CheckFilterItem.vue b/src/views/rule-engine/Scene/Save/action/ListItem/CheckFilterItem.vue deleted file mode 100644 index 2fbb9f3..0000000 --- a/src/views/rule-engine/Scene/Save/action/ListItem/CheckFilterItem.vue +++ /dev/null @@ -1,102 +0,0 @@ -<template> - <slot /> -</template> - -<script setup name="CheckFilterItem"> -import { Form } from "jetlinks-ui-components"; -import {queryAlarmList} from "@/api/rule-engine/scene"; -import {defineExpose} from "vue"; -import {useSceneStore} from "@/store/scene"; -import {storeToRefs} from "pinia"; -import { CHECK_FILTER_KEY } from './util' - -const props = defineProps({ - name: { - type: Number, - default: 0, - }, - termsName: { - type: Number, - default: 0, - }, - actionName: { - type: Number, - default: 0, - }, - thenName: { - type: Number, - default: 0, - }, - branchName: { - type: Number, - default: 0, - }, - value: { - type: Object, - default: () => ({}) - } -}) - -const formItemContext = Form.useInjectFormItemContext() -const sceneStore = useSceneStore(); -const { data } = storeToRefs(sceneStore); - -const formTouchOff = () => { - formItemContext.onFieldChange() -} - -const check = async () => { - if (props.value?.terms?.length) { - const alarmTerms = props.value.terms[1] - const id = data.value.id - const bindId = data.value.branches[props.branchName].then[props.thenName].actions[props.actionName].actionId || data.value.branches[props.branchName].branchId - if (alarmTerms?.value.value) { - const resp = await queryAlarmList({ - terms: [ - { - column: 'id', - termType: 'eq', - value: alarmTerms.value.value, - type: 'and' - }, - { - terms: [ - { - column: 'id$rule-bind-alarm', - value: `${id}:${bindId}`, - }, - { - column: 'id$rule-bind-alarm', - value: `${id}:${-1}`, - type: 'or' - }, - ], - type: 'and' - }, - ] - }) - - if (!resp.result?.length) { - formTouchOff() - } - } else { - formTouchOff() - } - } -} - -check() - -provide(CHECK_FILTER_KEY, { - onFieldChange: formTouchOff -}) - -defineExpose({ - formTouchOff -}) - -</script> - -<style scoped> - -</style> diff --git a/src/views/rule-engine/Scene/Save/action/TriggerAlarm/AlarmModal.vue b/src/views/rule-engine/Scene/Save/action/TriggerAlarm/AlarmModal.vue deleted file mode 100644 index 0aa878c..0000000 --- a/src/views/rule-engine/Scene/Save/action/TriggerAlarm/AlarmModal.vue +++ /dev/null @@ -1,233 +0,0 @@ -<template> - <j-modal - visible - title="新增关联告警" - :width="1000" - :keyboard="false" - :mask="false" - :maskClosable="false" - :loading="loading" - @cancel="onCancel" - @ok="onOk" - > - <pro-search :columns="columns" type="simple" @search="handleSearch" style="padding: 0"/> - <j-scrollbar :maxHeight="500"> - <JProTable - model="CARD" - :columns="columns" - :request="queryAlarmPage" - :bodyStyle="{ padding: 0 }" - :gridColumn="3" - :defaultParams="{ - sorts: [{ name: 'createTime', order: 'desc' }], - terms: [ - { - terms: [ - { - column: 'id$rule-bind-alarm$not', - value: `${id}:${actionId || branchId}`, - }, - { - column: 'branchIndex$rule-bind-alarm$not', - value: `${id}:${-1}`, - type: 'and' - }, - { - column: targetType === 'device' ? 'targetType$in' : 'targetType', - value: targetType === 'device' ? [] : 'other', - type: 'and' - } - ], - }, - ], - }" - :rowSelection="{ - selectedRowKeys: selectKeys, - onSelectNone: cancelSelect - }" - :params="params" - > - <template #card="slotProps"> - <CardBox - :value="slotProps" - :active="selectKeys.includes(slotProps.id)" - :status="slotProps.state?.value" - :statusText="slotProps.state?.text" - :statusNames="{ - enabled: 'processing', - disabled: 'error', - }" - :scroll="{ y: 510 }" - @click="handleClick" - > - <template #img> - <slot name="img"> - <img - src="/images/alarm/alarm-config.png" - /> - </slot> - </template> - <template #content> - <div style="margin-top: 36px; display: flex;gap: 6px;line-height: 1"> - <LevelIcon :level="slotProps.level" /> - <div style="flex: 1 1 0;min-width: 0"> - <Ellipsis> - {{ slotProps.name }} - </Ellipsis> - </div> - </div> - </template> - </CardBox> - </template> - </JProTable> - </j-scrollbar> - </j-modal> -</template> - -<script setup name="AlarmModal"> -import {queryAlarmPage} from '@/api/rule-engine/scene'; -import { useAlarmLevel, useRequest } from '@/hook' -import {bindScene, getTargetTypes} from "@/api/rule-engine/configuration"; -import {onlyMessage} from "@/utils/comm"; - -const props = defineProps({ - id: { - type: String, - default: '', - }, - actionId: { - type: String, - default: undefined - }, - branchId: { - type: String, - default: '', - }, - targetType: { - type: String - } -}); - -const emit = defineEmits(['cancel', 'ok']) - -const params = ref({}) -const selectKeys = ref([]) - -const { run, loading } = useRequest(bindScene, { - immediate: false, - onSuccess() { - emit('ok') - onlyMessage('操作成功!') - } -}) - -const { levelMap, levelList } = useAlarmLevel() - -const columns = [ - { - title: '类型', - dataIndex: 'targetType', - search: { - type: 'select', - options: async () => { - const resp = await getTargetTypes() - if (resp.success) { - return resp.result.filter(item => { - return props.targetType === 'device' || item.id === 'other' - }).map(item => ({ label: item.name, value: item.id })) - } else { - return [] - } - }, - } - }, - { - title: '配置名称', - dataIndex: 'name', - key: 'name', - search: { - type: 'string', - }, - width: 220, - ellipsis: true, - }, - { - title: '状态', - dataIndex: 'state', - key: 'state', - scopedSlots: true, - search: { - type: 'select', - options: [ - { - label: '正常', - value: 'enabled', - }, - { - label: '禁用', - value: 'disabled', - }, - ], - }, - width: 90, - }, - { - title: '等级', - dataIndex: 'level', - key: 'level', - scopedSlots: true, - search: { - type: 'select', - options: async () => { - return levelList.value - }, - }, - width: 200, - }, - -] - -const handleSearch = (e) => { - params.value = e -} - -const handleClick = (record) => { - const selectSet = new Set(selectKeys.value) - - if (selectSet.has(record.id)) { - selectSet.delete(record.id) - } else { - selectSet.add(record.id) - } - - selectKeys.value = [...selectSet.values()] -} - -const cancelSelect = () => { - selectKeys.value = [] -} - -const onCancel = () => { - emit('cancel') -} - -const onOk = async () => { - if (selectKeys.value.length) { - run(selectKeys.value.map(key => { - return { - alarmId: key, - ruleId: props.id, - branchIndex: props.actionId - } - })) - } else { - onlyMessage('请选择告警', 'warning') - } - -} - -</script> - -<style scoped> - -</style> diff --git a/src/views/rule-engine/Scene/Save/components/Description.vue b/src/views/rule-engine/Scene/Save/components/Description.vue deleted file mode 100644 index 8e56258..0000000 --- a/src/views/rule-engine/Scene/Save/components/Description.vue +++ /dev/null @@ -1,84 +0,0 @@ -<template> - <div class="scene-description"> - <a-button type="primary" v-if="showAddBtn" @click="showEdit"> - <template #icon> - <AIcon type="PlusOutlined"/> - </template> - 场景说明 - </a-button> - <div class="scene-description-content" v-if="showEditBtn"> - <a-input v-model:value="myValue" show-count :maxlength="200" placeholder='请输入说明' style="flex: 1 1 0"/> - <a-button @click="cancel">取消</a-button> - <a-button type="primary" @click="save">确认</a-button> - </div> - <div class="scene-description-content " v-if="showDetail"> - <div style="max-width: calc(100% - 32px);"> - <Ellipsis> - {{ myValue }} - </Ellipsis> - </div> - <a-button type="link" @click="showEdit"> - <template #icon> - <AIcon type="EditOutlined" style="font-size: 14px;"/> - </template> - </a-button> - </div> - </div> -</template> - -<script setup name="Description"> - -const props = defineProps({ - value: { - type: String, - default: '' - } -}) -const emit = defineEmits(['update:value']) - -const visible = ref() -const myValue = ref() - -const showAddBtn = computed(() => { - return !props.value && !visible.value -}) - -const showEditBtn = computed(() => { - return visible.value -}) - -const showDetail = computed(() => { - return props.value && !visible.value -}) - -const showEdit = () => { - visible.value = true -} - -const save = () => { - visible.value = false - emit('update:value', myValue.value) -} - -const cancel = () => { - visible.value = false - myValue.value = props.value -} - -watch(() => props.value, () => { - myValue.value = props.value -}, { immediate: true }) - -</script> - -<style scoped lang="less"> -.scene-description { - width: 100%; - - .scene-description-content { - display: flex; - gap: 8px; - align-items: center; - } -} -</style> diff --git a/src/views/rule-engine/Scene/Save/components/Terms/BranchesNameEdit.vue b/src/views/rule-engine/Scene/Save/components/Terms/BranchesNameEdit.vue deleted file mode 100644 index b1920d5..0000000 --- a/src/views/rule-engine/Scene/Save/components/Terms/BranchesNameEdit.vue +++ /dev/null @@ -1,50 +0,0 @@ -<template> - <j-modal - visible - title="编辑" - :keyboard="false" - :maskClosable="false" - @cancel="onCancel" - @ok="onOk" - > - <j-form ref='formRef' layout='vertical' :model="formData"> - <j-form-item label="条件名称" required name="name" :rules="[{ max: 64, message: '最多输入64个字符'}]"> - <j-input v-model:value="formData.name"></j-input> - </j-form-item> - </j-form> - </j-modal> -</template> - -<script setup name="BranchesNameEdit"> -const props = defineProps({ - name: { - type: String, - default: undefined - } -}) - -const emit = defineEmits(['cancel', 'ok']) - -const formData = reactive({ - name: props.name -}) -const formRef = ref() - - -const onCancel = () => { - emit('cancel') -} - -const onOk = async () => { - const data = await formRef.value.validate() - - if (data) { - emit('ok', formData.name) - } -} - -</script> - -<style scoped> - -</style> diff --git a/src/views/rule-engine/Scene/Save/components/Terms/TermsTabPane.vue b/src/views/rule-engine/Scene/Save/components/Terms/TermsTabPane.vue deleted file mode 100644 index 9abeec5..0000000 --- a/src/views/rule-engine/Scene/Save/components/Terms/TermsTabPane.vue +++ /dev/null @@ -1,53 +0,0 @@ -<template> - <div class="terms-tab-pane" @mouseleave="mouseleave" @mouseover="mouseover"> - <div class="terms-tab-pane-content"> - <slot /> - </div> - - <div class="terms-tab-pane-btn" @click="close"> - <AIcon v-show="showClose && showIcon" type="CloseOutlined" /> - </div> - </div> -</template> - -<script setup name="TermsTabPane"> - -const emit = defineEmits('close') - -const props = defineProps({ - showClose: { - type: Boolean, - default: true - } -}) - -const showIcon = ref(false) -const mouseleave = () => { - showIcon.value = false -} - -const mouseover = () => { - showIcon.value = true -} - -const close = (e) => { - e.stopPropagation() - if (props.showClose && showIcon.value) { - emit('close') - } -} -</script> - -<style scoped lang="less"> -.terms-tab-pane { - display: flex; - gap: 8px; - - .terms-tab-pane-btn { - width: 14px; - >span { - margin: 0; - } - } -} -</style> diff --git a/src/views/rule-engine/Scene/Save/components/Timer/Calendar.vue b/src/views/rule-engine/Scene/Save/components/Timer/Calendar.vue deleted file mode 100644 index d447ea8..0000000 --- a/src/views/rule-engine/Scene/Save/components/Timer/Calendar.vue +++ /dev/null @@ -1,263 +0,0 @@ -<template> - <div class="calendar-form-item-content"> - <div class="header"> - <a-button type="link" @click="onView" style="padding-right: 2px">预览日历</a-button> - </div> - <div class="calendar-items"> - <j-scrollbar :maxHeight="350"> - <div class="calendar-item" v-for="(item, index) in myValue.spec"> - <div class="calendar-item-delete"> - <AIcon type="DeleteOutlined" style="" @click="() => deleteItem(index)"/> - </div> - <div class="calendar-item-tags"> - <div class="calendar-item-name"> - 规则:{{ index + 1 }} - </div> - <div style="flex: 1 1 0;min-width: 0"> - <j-select - placeholder="请选择日期类型" - v-model:value="item.scheduleTags" - style="width: calc(100% - 30px)" - mode="multiple" - :options="options" - :fieldNames="{ - label: 'name', - value: 'id' - }" - :filterOption="filterOption" - @change='updateValue' - /> - </div> - </div> - <div class="calendar-item-content"> - <div class="calendar-item-top"> - <div> - <j-radio-group - :value='item.mod' - :options='[ - { label: "周期执行", value: "period" }, - { label: "执行一次", value: "once" }, - ]' - option-type='button' - button-style='solid' - @change='(e) => modChange(e, index)' - /> - </div> - </div> - <div class="calendar-item-bottom"> - <template v-if="item.mod === 'once'"> - <j-time-picker - valueFormat='HH:mm:ss' - v-model:value='item.once.time' - style='width: 180px' - format='HH:mm:ss' - @change='updateValue' - /> - <div>执行一次</div> - </template> - <template v-if="item.mod === 'period'"> - <j-time-range-picker - valueFormat='HH:mm:ss' - :value='[ - item.period.from, - item.period.to, - ]' - @change='(v) => { - item.period.from = v[0] - item.period.to = v[1] - updateValue() - }' - /> - <span>每</span> - <j-input-number - placeholder='请输入时间' - style='max-width: 170px' - :precision='0' - :min='1' - :max="item.period.unit === 'hours' ? 99999 : 99 " - v-model:value='item.period.every' - @change='updateValue' - > - <template #addonAfter> - <j-select - v-model:value='item.period.unit' - :options='[ - { label: "秒", value: "seconds" }, - { label: "分", value: "minutes" }, - { label: "小时", value: "hours" }, - ]' - @select='e => periodUnitChange(e, index)' - /> - </template> - </j-input-number> - </template> - </div> - - </div> - </div> - </j-scrollbar> - </div> - <a-button @click="addItem" style="width: 100%;"> - 新增规则 - </a-button> - <j-modal - v-model:visible="visible" - :width="1000" - okText="关闭" - > - <div> - 正在预览日历 - </div> - <FullCalendar :preview="true"/> - <template #footer> - <a-button type="primary" @click="visible = false"> - 关闭 - </a-button> - </template> - </j-modal> - </div> -</template> - -<script setup name="Calendar"> -import dayjs from "dayjs"; -import { useRequest } from '@/hook' -import {queryTags} from "@/api/system/calendar"; -import FullCalendar from '@/views/system/Calendar/FullCalendar/index.vue' - -const props = defineProps({ - value: { - type: Object, - default: () => ({}) - } -}) -const emit = defineEmits(['update:value', 'change']) -const { data:options } = useRequest(queryTags) -const visible = ref(false) - -const myValue = reactive({ - type: 'or', - spec: [] -}) - -const updateValue = () => { - emit('update:value', myValue) - emit('change', myValue) -} - -const onView = () => { - visible.value = true -} - -const filterOption = (value, option) => { - return option.name?.includes(value) -} - -const addItem = () => { - myValue.spec.push({ - trigger: "calender", - scheduleTags: [], - mod: "period", - // once: { - // time: dayjs(new Date()).format('HH:mm:ss') - // }, - period: { - from: dayjs(new Date()).startOf('day').format('HH:mm:ss'), - to: dayjs(new Date()).endOf('day').format('HH:mm:ss'), - every: 1, - unit: 'seconds' - } - }) - updateValue() -} - -const deleteItem = (index) => { - myValue.spec.splice(index, 1) - updateValue() -} - -const modChange = (e, index) => { - let mod = e.target.value - - if (mod === 'once') { - myValue.spec[index].once = { - time: dayjs(new Date()).format('HH:mm:ss') - } - delete myValue.spec[index].period - } else { - myValue.spec[index].period = { - from: dayjs(new Date()).startOf('day').format('HH:mm:ss'), - to: dayjs(new Date()).endOf('day').format('HH:mm:ss'), - every: 1, - unit: 'seconds' - } - delete myValue.spec[index].once - } - myValue.spec[index].mod = mod - updateValue() -} - -const periodUnitChange = (e, index) => { - myValue.spec[index].period.every = 1 - updateValue() -} - -watch(() => props.value, () => { - myValue.spec = props.value?.spec || [] -}, { immediate: true, deep: true}) - -</script> - -<style scoped lang="less"> -.calendar-form-item-content { - position: relative; - .header { - position: absolute; - right: 0; - top: -55px; - } - .calendar-items { - max-height: 350px; - - .calendar-item { - position: relative; - margin-bottom: 12px; - border: 1px solid #cfcfcf; - border-radius: 4px; - padding: 12px; - display: flex; - gap: 16px; - flex-direction: column; - - .calendar-item-tags { - display: flex; - gap: 12px; - align-items: center; - //height: max-content; - } - - .calendar-item-delete { - position: absolute; - right: 12px; - color: #e50012; - } - - .calendar-item-content { - display: flex; - gap: 24px; - } - - .calendar-item-top { - display: flex; - gap: 12px; - align-items: center; - //margin-bottom: 12px; - } - .calendar-item-bottom { - display: flex; - gap: 12px; - align-items: center; - } - } - } -} -</style> diff --git a/src/views/system/Calendar/CalendarRight/index.vue b/src/views/system/Calendar/CalendarRight/index.vue deleted file mode 100644 index 163c79a..0000000 --- a/src/views/system/Calendar/CalendarRight/index.vue +++ /dev/null @@ -1,142 +0,0 @@ -<template> - <div class="calendarRight"> - <div class="tips"> - <span - >将左侧标签拖拽至右侧日历中,日历中的标签在场景联动中可以被引用作为执行动作的触发条件</span - > - <permissionButton - type="text" - :disabled="rapidChecked" - :popConfirm="{ - placement: 'bottomRight', - title: `确认清空?`, - onConfirm: clearEvent, - }" - > - <template #icon - ><AIcon - type="ClearOutlined" - style="font-size: 16px" /></template - >清空</permissionButton - > - </div> - <FullCalendar - ref="Calendar" - :selectable="rapidChecked" - @resetRapid="resetRapid" - @selectDate="selectDate" - @rapidAction="rapidAction" - /> - <div class="rapidAction"> - <div class="rapidSwitch"> - <a-tooltip> - <template #title - >开启后以周/月为最小单位制定日期模板,确认后可快速覆盖日历生效期</template - > - <AIcon type="QuestionCircleOutlined"></AIcon> - </a-tooltip> - 快速作用:<a-switch v-model:checked="rapidChecked" /> - </div> - <div class="rapidActionControl" v-if="rapidChecked"> - <div> - 已选择模板 - <a-input - style="width: 300px" - v-model:value="selectedDate" - :disabled="true" - ></a-input> - </div> - <div style="margin-left: 10px"> - 将模板快速作用于后续的 - <a-input-number - style="width: 100px" - v-model:value="effectDays" - :min="effectMin" - :max="999" - :precision="0" - :controls="false" - ></a-input-number> - 日内 - </div> - <div> - <a-button type="text" @click="reselect">重选</a-button> - <a-button type="link" @click="rapid">快速作用</a-button> - </div> - </div> - </div> - </div> -</template> - -<script setup name="Calendar"> -import FullCalendar from '../FullCalendar/index.vue'; -import { inject } from 'vue'; -import { clearAll } from '@/api/system/calendar'; -import { onlyMessage } from '@/utils/comm'; -const Calendar = ref(); -const rapidChecked = ref(false); -const rapidOn = inject('rapidOn'); -const selectedDate = ref(); -const effectDays = ref(); -const effectMin = ref(); -const selectDate = (data) => { - selectedDate.value = data?.start + '~' + data?.end; - effectDays.value = data?.days; - effectMin.value = data?.days; -}; -const reselect = () => { - Calendar.value.reselection(); - selectedDate.value = ''; - effectDays.value = undefined; - effectMin.value = undefined; -}; -const resetRapid = () => { - reselect(); - rapidChecked.value = false; -}; -const rapid = () => { - Calendar.value.rapidAction(effectDays.value); -}; -const clearEvent = async () => { - const res = await clearAll(); - if (res.success) { - onlyMessage('操作成功'); - Calendar.value?.refresh(); - } -}; -watch( - () => rapidChecked.value, - () => { - rapidOn.value = rapidChecked.value; - }, - { - immediate: true, - }, -); -</script> -<style lang="less" scoped> -.calendarRight { - width: 85%; - padding: 10px 20px; - background: #fff; - .tips { - display: flex; - justify-content: space-between; - } - .rapidAction { - margin-top: 10px; - display: flex; - align-items: center; - .rapidSwitch { - padding: 15px 0; - } - .rapidActionControl { - display: flex; - width: 70%; - margin-left: 20%; - background-color: rgb(239, 249, 254); - padding: 10px; - justify-content: space-around; - } - } -} -</style> diff --git a/src/views/system/Calendar/FullCalendar/index.vue b/src/views/system/Calendar/FullCalendar/index.vue deleted file mode 100644 index 8b2060b..0000000 --- a/src/views/system/Calendar/FullCalendar/index.vue +++ /dev/null @@ -1,761 +0,0 @@ -<template> - <div class="calendarContainer" v-if="showCalendar"> - <div class="calenderButton"> - <a-date-picker - v-model:value="current" - :disabled="selectable" - format="YYYY-MM" - picker="month" - @change="changeDate" - valueFormat="YYYY-MM" - /> - <a-button @click="handleCustomPrev" :disabled="selectable" - >上月</a-button - > - <a-button @click="handleCustomToday" :disabled="selectable" - >今天</a-button - > - <a-button @click="handleCustomNext" :disabled="selectable" - >下月</a-button - > - </div> - <a-button v-if="preview" class="skip" type="link" @click="gotoCalendar" - >日历维护</a-button - > - <div class="compareTip" v-if="eventChange"> - 点击确认完成本次日历维护 - </div> - <div class="compareSave" v-if="eventChange"> - <PermissionButton - type="primary" - @click="saveCalendar" - :disabled="selectable" - >确认</PermissionButton - > - </div> - <FullCalendar - ref="calendarEl" - class="calendar" - :options="calendarOptions" - > - <template v-slot:eventContent="arg"> - <div class="event"> - <div - class="decoration" - :style="{ backgroundColor: arg.backgroundColor }" - ></div> - <a-tooltip> - <template #title>{{ arg.event.title }}</template> - <div class="event-title">{{ arg.event.title }}</div> - </a-tooltip> - <a-button - v-if="!selectable && !preview" - type="text" - class="closeBtn" - @click="() => deleteEvent(arg)" - >x</a-button - > - </div> - </template> - </FullCalendar> - <div - v-if="selectable && !choiceEnd && showTips" - class="tips" - :style="{ top: mouseY + 'px', left: mouseX + 20 + 'px' }" - > - {{ choiceStart ? '点击选中模板结束日期' : '点击选中模板开始日期' }} - </div> - </div> -</template> - -<script setup name="FullCalendar"> -import FullCalendar from '@fullcalendar/vue3'; -import interactionPlugin from '@fullcalendar/interaction'; -import dayGridPlugin from '@fullcalendar/daygrid'; -import locale from '@fullcalendar/core/locales/zh-cn'; -import { onlyMessage } from '@/utils/comm'; -import dayjs from 'dayjs'; -import { queryEvents, saveEvents } from '@/api/system/calendar'; -import { cloneDeep, flatten } from 'lodash-es'; -import { defineExpose } from 'vue'; -import { inject } from 'vue'; -import { useMenuStore } from 'store/menu'; -import { useSystem } from '@/store/system'; -const props = defineProps({ - selectable: { - type: String, - default: false, - }, - preview: { - type: Boolean, - default: false, - }, -}); -const emit = defineEmits(['selectDate', 'resetRapid']); -const menuStory = useMenuStore(); -const system = useSystem(); -const showCalendar = ref(false); -const calendarTagColor = system.$state.calendarTagColor; -const tagsList = inject('tagsMap'); -//请求接口的结束时间(请求过的日期就不再请求接口了) -const queryEndDate = ref([]); -//当前日历展示日期 -const current = ref(dayjs().format('YYYY-MM')); -const mouseY = ref(); -const mouseX = ref(); -//是否显示提示 -const showTips = ref(); -const calendarEl = ref(); -const calendarApi = ref(); -// 最新的事件数据 -const eventsData = ref([]); -//接口获取到的事件数据 用于对比是否有更改 -const initialData = ref([]); -//接口获取转换为日历插件需要使用的格式 -const initialEventData = ref([]); -//接口获取到的数据 -const interfaceData = ref([]); -//多选开始日期 -const choiceStart = ref(); -//多选结束日期 -const choiceEnd = ref(); -//上次悬停改变过的数据 -const lastHoverDate = ref([]); -//请求数据逻辑 -const queryEventsData = async (startDate, endDate, updateView) => { - const res = await queryEvents(startDate, endDate); - if (res.success) { - const data = res.result.filter((i) => { - const nonentity = !interfaceData.value.find((item) => { - return item.date === i.date; - }); - if (nonentity) { - i.tags.forEach((item) => { - eventsData.value.push({ - date: i.date, - id: item.id, - name: item.name, - }); - initialData.value.push({ - date: i.date, - id: item.id, - name: item.name, - }); - }); - } - return nonentity; - }); - interfaceData.value = [...interfaceData.value, ...data]; - if (updateView) { - initialEventData.value = eventsData.value.map((i) => { - return { - id: i.id, - title: i.name, - start: i.date, - backgroundColor: calendarTagColor.get(i.id) || '#000000', - }; - }); - calendarApi.value.removeAllEvents(); - calendarApi.value.addEventSource(initialEventData.value); - } - } -}; -// 事件点击删除逻辑 -const deleteEvent = (data) => { - const event = data.event; - event.remove(); - let subscript; - eventsData.value.forEach((i, index) => { - if (i.date === event.startStr && i.id === event.id) { - subscript = index; - } - }); - eventsData.value.splice(subscript, 1); -}; -// 事件拖拽添加逻辑处理 -const handleEventAdd = ({ event }) => { - const thatEvent = eventsData.value.filter((i) => { - return i.date === event.startStr; - }); - if (thatEvent?.length >= 5) { - event.remove(); - onlyMessage('该日期已超出可配置标签上限', 'error'); - return; - } else if (thatEvent.length) { - const alreadyExist = thatEvent.find((i) => { - return i.id === event.id; - }); - if (alreadyExist) { - event.remove(); - } else { - eventsData.value.push({ - date: event.startStr, - id: event.id, - name: event.title, - }); - } - } else { - eventsData.value.push({ - date: event.startStr, - id: event.id, - name: event.title, - }); - } -}; -// 日历初始化后请求日历中显示的事件 -const handleViewDidMount = async (arg) => { - const startDate = dayjs(arg.view.activeStart).format('YYYY-MM-DD'); - // fullcalendar插件获取当前日期截至日期会多一天 - const endDate = dayjs(arg.view.activeEnd) - .subtract(1, 'day') - .format('YYYY-MM-DD'); - if (queryEndDate.value.includes(endDate)) { - return; - } - queryEndDate.value.push(endDate); - if (calendarTagColor.size === 0) { - await system.getTagsColor(); - } - queryEventsData(startDate, endDate, true); -}; -// 事件是否发生变化 -const eventChange = computed(() => { - return ( - JSON.stringify(eventsData.value) !== JSON.stringify(initialData.value) - ); -}); -// 自定义切换月份逻辑 -const handleCustomPrev = () => { - calendarApi.value.prev(); - current.value = dayjs(current.value).subtract(1, 'month').format('YYYY-MM'); -}; -const handleCustomNext = () => { - current.value = dayjs(current.value).add(1, 'month').format('YYYY-MM'); - calendarApi.value.next(); -}; -const handleCustomToday = () => { - current.value = dayjs().format('YYYY-MM'); - calendarApi.value.today(); -}; -//保存编辑后的日历 -const saveCalendar = async () => { - let formatEventData = []; - let submitData = []; - eventsData.value.forEach((i) => { - const existIndex = formatEventData.findIndex((item) => { - return item.date === i.date; - }); - if (existIndex !== -1) { - formatEventData[existIndex]?.tags?.push({ - id: i.id, - name: i.name, - }); - } else { - formatEventData.push({ - date: i.date, - tags: [ - { - id: i.id, - name: i.name, - }, - ], - }); - } - }); - const formatEventDataMap = new Map(); - formatEventData.map((i) => { - formatEventDataMap.set(i.date, i); - }); - submitData = cloneDeep(formatEventData); - interfaceData.value?.forEach((i) => { - if (!formatEventDataMap.has(i.date)) { - submitData.push({ - date: i.date, - tags: [], - }); - } - }); - const res = await saveEvents(submitData); - if (res.success) { - onlyMessage('操作成功'); - // initialData.value = cloneDeep(eventsData.value); - refresh(); - } -}; -//获取两个时间段之间的所有日期 -const getDatesBetween = (start, end) => { - const dates = []; - let currentDate = start.clone(); - while (currentDate.isBefore(end)) { - dates.push(currentDate.format('YYYY-MM-DD')); - currentDate = currentDate.add(1, 'day'); - } - return dates; -}; -//多选日期处理数据(原生拖拽逻辑) -const handleSelect = (selectionInfo) => { - choiceStart.value = dayjs(selectionInfo.startStr); - choiceEnd.value = dayjs(selectionInfo.endStr); - emit('selectDate', { - start: choiceStart.value.format('YYYY.MM.DD'), - end: choiceEnd.value.subtract(1, 'day').format('YYYY.MM.DD'), - days: choiceEnd.value.diff(choiceStart.value, 'day'), - }); -}; -//多选日期处理数据(点击逻辑) -const handleDateClick = (selectInfo) => { - if (!props.selectable) { - return; - } - if (!choiceStart.value) { - choiceStart.value = dayjs(selectInfo.dateStr); - const selectedDate = document.querySelector( - `.fc-day[data-date="${selectInfo.dateStr}"]`, - ); - selectedDate.classList.add('selectedDate'); - } else { - if (dayjs(selectInfo.dateStr).isBefore(choiceStart.value)) { - onlyMessage('结束时间必须大于开始时间'); - return; - } - choiceEnd.value = dayjs(selectInfo.dateStr).add(1, 'day'); - emit('selectDate', { - start: choiceStart.value.format('YYYY.MM.DD'), - end: choiceEnd.value.subtract(1, 'day').format('YYYY.MM.DD'), - days: choiceEnd.value.diff(choiceStart.value, 'day'), - }); - } -}; -//多选背景颜色渲染逻辑 -const handleCellDidMount = (info) => { - info.el.addEventListener('mouseenter', () => { - const currentDate = dayjs(info.date); - //每次移动清空之前所有的颜色 - if (lastHoverDate.value && !choiceEnd.value) { - lastHoverDate.value.forEach((i) => { - const selectedDate = document.querySelector( - `.fc-day[data-date="${i}"]`, - ); - selectedDate.classList.remove('selectedDate'); - }); - } - if ( - choiceStart.value && - currentDate.isAfter(choiceStart.value) && - !choiceEnd.value - ) { - const dateArr = getDatesBetween( - choiceStart.value, - currentDate.add(1, 'day'), - ); - dateArr.forEach((i) => { - const selectedDate = document.querySelector( - `.fc-day[data-date="${i}"]`, - ); - selectedDate.classList.add('selectedDate'); - }); - lastHoverDate.value = dateArr; - lastHoverDate.value.shift(); - } - }); -}; -//插件配置项 -const calendarOptions = { - plugins: [interactionPlugin, dayGridPlugin], - initialView: 'dayGridMonth', - headerToolbar: { - start: '', - center: '', - end: '', - }, - events: initialEventData.value, - // selectable: props.selectable, - unselectAuto: false, - locale: locale, - droppable: true, - height: props.preview ? '600px' : '620px', - // select: handleSelect, //原生拖拽多选日期逻辑 - eventReceive: handleEventAdd, - datesSet: handleViewDidMount, - dateClick: handleDateClick, - dayCellDidMount: handleCellDidMount, - //星期只显示数字 - dayHeaderContent: function (arg) { - return arg.date.toLocaleDateString('default', { weekday: 'narrow' }); - }, - dayCellContent: function (e) { - return { - html: '<div class="custom-day-cell">' + e.date.getDate() + '</div>', - }; - }, -}; -// //编辑标签后刷新日历数据 -// const refreshCalendar = () => { -// handleViewDidMount(calendarApi.value); -// }; -//对比函数(判断出日期相等但是标签id不同的事件和日期事件数量少于5的) -const compare = (effectData, eventsData, effectDates) => { - //获取新增的事件 - const addEvents = effectData.filter((i) => { - const equality = eventsData.find((item) => { - return i.date === item.date && i.id === item.id; - }); - const exceed = - eventsData.filter((item) => { - return i.date === item.date; - }).length >= 5; - return !equality && !exceed; - }); - const filterDate = []; - //比对新增事件 + 原本事件 > 5 的日期 - effectDates.forEach((i) => { - const addEventNumber = addEvents.filter((item) => { - return i === item.date; - }).length; - const eventNumber = eventsData.filter((item) => { - return i === item.date; - }).length; - if (addEventNumber + eventNumber > 5) { - filterDate.push(i); - } - }); - // 过滤掉日期影响后 >5 的新增事件 - const filteredEvents = addEvents.filter((i) => { - return !filterDate.includes(i.date); - }); - return filteredEvents; -}; -//快速作用 -const rapidAction = async (effectDays) => { - const dates = getDatesBetween(choiceStart.value, choiceEnd.value); - const effectDates = getDatesBetween( - choiceEnd.value, - choiceEnd.value.add(effectDays, 'day'), - ); - //获取所选日期中所有的标签事件组成二维数组 - const selectData = dates.map((i) => { - return eventsData.value.filter((item) => { - return i === item.date; - }); - }); - //创建一个长度为 影响天数的数组 - const effectData = Array.from({ length: effectDays }); - //循环数组添加日期和标签时间等数据 - for (let i = 0; i < effectData.length; i++) { - effectData[i] = cloneDeep(selectData[i % selectData.length]); - if (effectData[i].length) { - effectData[i].forEach((item) => { - item.date = dayjs(item.date) - .add( - selectData.length * - Math.ceil((i + 1) / selectData.length), - 'day', - ) - .format('YYYY-MM-DD'); - }); - } else { - eventsData.value = eventsData.value.filter((item) => { - console.log(effectData[i]); - return item.date !== effectDates[i]; - }); - } - } - //二维数组扁平成一维数组 - const effectDataArr = flatten(effectData); - //查询所影响的日期的数据,做对比等逻辑处理(数据是只请求了页面显示过的日期所以需要请求影响的日期,以防没有数据导致重复事件) - await queryEventsData( - dayjs(choiceEnd.value).format('YYYY-MM-DD'), - dayjs(choiceEnd.value) - .add(effectDays - 1, 'day') - .format('YYYY-MM-DD'), - false, - ); - const imparity = compare(effectDataArr, eventsData.value, effectDates); - eventsData.value = [...eventsData.value, ...imparity]; - initialEventData.value = eventsData.value.map((i) => { - return { - id: i.id, - title: i.name, - start: i.date, - backgroundColor: calendarTagColor.get(i.id) || '#000000', - }; - }); - calendarApi.value.removeAllEvents(); - calendarApi.value.addEventSource(initialEventData.value); - onlyMessage('操作成功'); - emit('resetRapid'); -}; -//取消多选(原生) -// const reselection = () => { -// calendarApi.value.unselect(); -// }; -//取消多选自定义 -const reselection = () => { - if (choiceStart.value) { - lastHoverDate.value.push(dayjs(choiceStart.value).format('YYYY-MM-DD')); - } - if (choiceEnd.value) { - lastHoverDate.value.push(dayjs(choiceStart.value).format('YYYY-MM-DD')); - } - lastHoverDate.value.forEach((i) => { - const selectedDate = document.querySelector( - `.fc-day[data-date="${i}"]`, - ); - selectedDate.classList.remove('selectedDate'); - }); - (choiceStart.value = ''), (choiceEnd.value = ''); -}; -//重新获取数据 -const refresh = () => { - eventsData.value = []; - initialData.value = []; - interfaceData.value = []; - queryEndDate.value = []; - handleViewDidMount(calendarApi.value); -}; -//日期切换 -const changeDate = (date) => { - calendarApi.value.gotoDate(date); -}; - -const gotoCalendar = () => { - menuStory.jumpPage('system/Calendar'); -}; -defineExpose({ - reselection, - rapidAction, - refresh, -}); -watch( - () => props.selectable, - (val) => { - calendarApi.value.setOption('droppable', !val); - if (!val) { - emit('resetRapid'); - } - }, -); -watch( - () => tagsList?.value, - (val, oldVal) => { - if (val?.length <= oldVal?.length) { - const tagsMap = new Map(); - tagsList.value.forEach((i) => { - tagsMap.set(i.id, { - name: i.name, - color: i.color, - }); - }); - eventsData.value = eventsData.value.filter((i) => { - return tagsMap.has(i.id); - }); - initialEventData.value = eventsData.value.map((i) => { - return { - id: i.id, - title: tagsMap.get(i.id)?.name, - start: i.date, - backgroundColor: tagsMap.get(i.id)?.color || '#000000', - }; - }); - calendarApi.value.removeAllEvents(); - calendarApi.value.addEventSource(initialEventData.value); - } - }, - { - deep: true, - }, -); -setTimeout(() => { - showCalendar.value = true; - nextTick(() => { - calendarApi.value = calendarEl.value?.getApi(); - const calendarBody = document.querySelector(`.fc-view-harness`); - calendarBody?.addEventListener('mousemove', (event) => { - if (props.selectable) { - showTips.value = true; - mouseX.value = event.clientX; - mouseY.value = event.clientY; - } - }); - calendarBody?.addEventListener('mouseout', () => { - if (props.selectable) { - showTips.value = false; - } - }); - }); -}, 300); -</script> -<style lang="less" scoped> -:deep(.fc-header-toolbar) { - background-color: #edebeb; -} -.calendarContainer { - position: relative; - padding: 0 24px; - padding-bottom: 0; - padding-top: 44px; - border: 1px solid #d9d9d9; - border-radius: 12px; - .compareTip { - position: absolute; - right: 20%; - top: 27px; - transform: translateX(50%); - } - .skip { - position: absolute; - right: 0; - top: 27px; - } - .compareSave { - position: absolute; - right: 10px; - top: 22px; - } - .calenderButton { - position: absolute; - top: 22px; - } - .event { - position: relative; - height: 32px; - padding: 6px; - color: #1a1a1a; - display: flex; - - .closeBtn { - position: absolute; - right: 0; - top: 0; - color: #777777; - display: none; - font-size: 16px; - } - } - .event:hover { - .closeBtn { - display: inline-block; - } - } - .decoration { - width: 4px; - height: 16px; - border-radius: 2px; - margin: 2px 4px; - display: inline-block; - } - .event-title { - white-space: nowrap; /* 不换行 */ - overflow: hidden; /* 超出部分隐藏 */ - text-overflow: ellipsis; /* 显示省略号 */ - width: calc(100% - 30px); - font-size: 14px; - } -} -:deep(.fc-highlight) { - background-color: transparent; -} -.tips { - position: fixed; - z-index: 99999; - padding: 5px 10px; - background-color: rgba(0, 0, 0, 0.7); - border-radius: 5px; - color: white; -} -:deep(.fc-theme-standard th) { - border: none; -} -:deep(.fc-theme-standard td) { - border: none; - border-top: 1px solid #cccccc; -} -:deep(.fc-theme-standard .fc-scrollgrid) { - border: none; -} -:deep(.fc) { - .fc-col-header-cell-cushion { - color: #777777; - } - th { - text-align: right; - color: #777777; - } - table { - border-collapse: separate; - border-spacing: 5px 0; - } - .fc-daygrid-day.fc-day-today { - background-color: transparent; - .fc-daygrid-day-number { - background-color: #1677ff; - color: white; - border-radius: 50%; - width: 30px; - text-align: center; - } - } - .fc-event { - border: none !important; - } - .fc-daygrid-event-harness { - margin: 0 4px; - margin-top: 4px !important; - border-radius: 6px; - } - .fc-daygrid-day-frame { - height: 155px; - .fc-daygrid-day-events { - max-height: 120px; - overflow-y: auto; - &::-webkit-scrollbar-thumb { - background-color: #d0d0d0; /* 滚动条拖动部分颜色 */ - border-radius: 4px; /* 滚动条拖动部分圆角 */ - } - } - } - .fc-scroller { - &::-webkit-scrollbar { - width: 5px; /* 滚动条宽度 */ - background-color: #fff; /* 滚动条背景色 */ - } - &::-webkit-scrollbar-thumb { - background-color: #d0d0d0; /* 滚动条拖动部分颜色 */ - border-radius: 4px; /* 滚动条拖动部分圆角 */ - } - } -} -:deep(.fc-scrollgrid-section-body > td) { - border: none !important; -} -:deep(.fc-daygrid-day-number) { - color: #1a1a1a; - font-weight: 600; -} -</style> -<style> -.calendarContainer { - .fc-event { - background: #edf5ff !important; - } - .fc-daygrid-day-events { - &::-webkit-scrollbar { - width: 5px; /* 滚动条宽度 */ - background-color: #fff; /* 滚动条背景色 */ - } - } -} - -.selectedDate { - background-color: #edf5ff !important; - .fc-event { - background: #fff !important; - } - .fc-daygrid-day-events { - &::-webkit-scrollbar { - width: 5px; /* 滚动条宽度 */ - background-color: #edf5ff; /* 滚动条背景色 */ - } - } -} -</style> diff --git a/src/views/system/Calendar/Tags/components/editTag.vue b/src/views/system/Calendar/Tags/components/editTag.vue deleted file mode 100644 index 23e580f..0000000 --- a/src/views/system/Calendar/Tags/components/editTag.vue +++ /dev/null @@ -1,127 +0,0 @@ -<template> - <a-modal - visible - :title="editType === 'add' ? '新增标签' : '编辑标签'" - @cancel="emit('closeEditTag')" - @ok="submit" - :confirmLoading="loading" - > - <a-form :model="tagInfo" ref="form"> - <a-row :gutter="16"> - <a-col> - <a-form-item name="color"> - <ColorPicker - type="color" - :hex="tagInfo.color" - :rgba="tagInfo.color" - :themeColor="themeColor" - @change="changeColor" - /> - </a-form-item> - </a-col> - <a-col :span="20"> - <a-form-item - name="name" - :rules="[ - { - required: true, - message:'请为标签命名' - }, - { - max: 16, - message: '最多可输入16个字符', - }, - ]" - > - <a-input - v-model:value="tagInfo.name" - placeholder="请为标签命名" - ></a-input> - </a-form-item> - </a-col> - </a-row> - </a-form> - </a-modal> -</template> - -<script setup name="EditTag"> -import { saveTag, saveTagsColor, getTagsColor } from '@/api/system/calendar'; -import { onlyMessage } from '@/utils/comm'; -import { randomString } from '@/utils/utils'; -import ColorPicker from 'colorpicker-v3'; -import 'colorpicker-v3/style.css'; -const props = defineProps({ - editType: { - type: String, - default: 'add', - }, - editData: { - type: Object, - default: {}, - }, -}); -const emit = defineEmits(['closeEditTag', 'refresh']); -const tagInfo = reactive({ - color: '#000000', - name: '', -}); -const form = ref(); -const colorData = ref(); -const changeColor = ({rgba}) => { - tagInfo.color = rgba; -}; -const loading = ref(false) -const themeColor = [ - '#69B1FF', - '#5CDBD3', - '#FFAF6E', - '#85A5FF', - '#F4D160', - '#95DE64', - '#B37FEB', - '#FFEC3D', - '#FF7875', - '#FF85C0' -] -const submit = () => { - loading.value = true - form.value.validate().then(async () => { - let id; - if (props.editType === 'add') { - id = randomString(); - } else { - id = props.editData?.id; - } - const submitData = { - id, - name: tagInfo.name, - }; - const res = await saveTag(submitData); - if (res.success) { - colorData.value[id] = tagInfo.color; - const saveRes = await saveTagsColor(colorData.value).catch(()=>{ - loading.value = false - }); - if (saveRes.success) { - onlyMessage('操作成功'); - emit('refresh'); - } - } - }); -}; -//获取全量标签颜色 保存接口是覆盖 -const queryTagsColor = async () => { - const res = await getTagsColor(); - if (res.success) { - colorData.value = res.result; - } -}; -onMounted(() => { - queryTagsColor(); - if (props.editType === 'edit') { - tagInfo.color = props.editData?.color || '#000000'; - tagInfo.name = props.editData?.name || ''; - } -}); -</script> -<style lang="less" scoped></style> diff --git a/src/views/system/Calendar/Tags/index.vue b/src/views/system/Calendar/Tags/index.vue deleted file mode 100644 index d832bb0..0000000 --- a/src/views/system/Calendar/Tags/index.vue +++ /dev/null @@ -1,200 +0,0 @@ -<template> - <div class="tagsContainer" ref="tags"> - <PermissionButton type="primary" hasPermission="system/Calendar:add" ghost block @click="addTag" :disabled="rapidOn" style="margin-bottom: 10px;"> - + 新增标签 - </PermissionButton> - <div v-for="i in tagsList" class="tag"> - <div class="tagLeft"> - <div - :style="{ background: i.color }" - class="colorExtractor" - ></div> - <Ellipsis - class="tagName" - style=" - width: 120px; - cursor: url('/images/calendar/hover.png'), pointer; - " - :id="i.id" - >{{ i.name }}</Ellipsis - > - </div> - <div class="controls"> - <PermissionButton - type="text" - hasPermission="system/Calendar:update" - :disabled="i.disabled || rapidOn" - :tooltip="{ - title: '编辑' - }" - @click="() => editData(i)" - > - <template #icon> - <AIcon type="EditOutlined" /> - </template> - </PermissionButton> - <PermissionButton - type="text" - hasPermission="system/Calendar:delete" - :disabled="i.disabled || rapidOn" - :tooltip="{ - title: '删除' - }" - :popConfirm="{ - title: `确认删除?`, - onConfirm: () => deleteData(i.id), - }" - > - <template #icon> - <AIcon type="DeleteOutlined" /> - </template> - </PermissionButton> - </div> - </div> - </div> - <EditTag - v-if="editVisible" - :edit-type="editType" - :editData="currentTag" - @close-edit-tag="editVisible = false" - @refresh="refreshTags" - /> -</template> - -<script setup name="CalendarTags"> -import { queryTags, deleteTags, saveTagsColor } from '@/api/system/calendar'; -import { Draggable } from '@fullcalendar/interaction'; -import EditTag from './components/editTag.vue'; -import { onlyMessage } from '@/utils/comm'; -import { inject } from 'vue'; -import { omit } from 'lodash-es'; -import { useSystem } from '@/store/system'; -const system = useSystem(); -const calendarTagColor = system.$state.calendarTagColor; -const tagsMap = inject('tagsMap'); -const rapidOn = inject('rapidOn'); -const editVisible = ref(false); -const tags = ref(); -const tagsList = ref(); -const editType = ref(); -const currentTag = ref(); -const addTag = () => { - editVisible.value = true; - editType.value = 'add'; -}; -const buildInTag = ['weekend', 'holiday', 'workday']; -const createDrag = () => { - new Draggable(tags.value, { - itemSelector: '.tagName', - eventData: function (eventEl) { - return { - id: eventEl.id, - title: eventEl.innerText, - backgroundColor: calendarTagColor.get(eventEl.id) || '#000000', - color: '#000', - editable: false, - }; - }, - }); -}; -//获取标签列表 -const queryTagsData = async () => { - const res = await queryTags(); - if (res.success) { - //获取用户添加的标签颜色及权限 - await system.getTagsColor(); - tagsList.value = res.result.map((i) => { - let color = '#000000'; - let disabled = false; - if (calendarTagColor.has(i.id)) { - color = calendarTagColor.get(i.id); - } - if (buildInTag.includes(i.id)) { - disabled = true; - } - return { - ...i, - color, - disabled, - }; - }); - tagsMap.value = tagsList.value.map((i) => { - return omit(i, ['disabled']); - }); - } -}; - -const refreshTags = () => { - editVisible.value = false; - queryTagsData(); -}; - -const deleteData = async (id) => { - const res = await deleteTags([id]); - if (res.success) { - if (calendarTagColor.has(id)) { - calendarTagColor.delete(id); - let color = new Object(); - calendarTagColor.forEach((item, key) => { - color[key] = item; - }); - const deleteColor = await saveTagsColor(color); - } - onlyMessage('操作成功'); - queryTagsData(); - } -}; - -const editData = (data) => { - editVisible.value = true; - editType.value = 'edit'; - currentTag.value = data; -}; -onMounted(() => { - queryTagsData(); - createDrag(); -}); -</script> -<style lang="less" scoped> -.tagsContainer { - width: 15%; - background-color: #fff; - padding: 10px; - border-right: 1px solid #CCCCCC; - .tag { - display: flex; - justify-content: space-between; - margin-bottom: 10px; - padding: 0 10px; - height: 32px; - .tagLeft { - display: flex; - padding-top: 6px; - .colorExtractor { - margin-top: 3px; - width: 16px; - height: 16px; - border-radius: 2px; - margin-right: 8px; - } - } - .controls{ - display: none; - font-size: 14px; - - :deep(.ant-btn-text){ - padding: 4px 8px; - } - } - &:hover{ - .controls{ - display: block; - } - } - } - .tag:hover{ - background-color: rgb(242, 242, 242); - border-radius: 4px; - } -} -</style> diff --git a/src/views/system/Calendar/index.vue b/src/views/system/Calendar/index.vue deleted file mode 100644 index ee84daf..0000000 --- a/src/views/system/Calendar/index.vue +++ /dev/null @@ -1,24 +0,0 @@ -<template> - <page-container> - <div class="calendarContainer"> - <CalendarTags /> - <CalendarRight /> - </div> - </page-container> -</template> - -<script setup> -import CalendarTags from './Tags/index.vue'; -import CalendarRight from './CalendarRight/index.vue'; -import { provide } from 'vue'; -const tags = ref(); -const rapidOn = ref(); -provide('tagsMap', tags); -provide('rapidOn', rapidOn); -</script> -<style lang="less" scoped> -.calendarContainer { - display: flex; - justify-content: space-around; -} -</style> diff --git a/src/views/system/Region/Save/GeoJsonModal.vue b/src/views/system/Region/Save/GeoJsonModal.vue deleted file mode 100644 index dffa317..0000000 --- a/src/views/system/Region/Save/GeoJsonModal.vue +++ /dev/null @@ -1,46 +0,0 @@ -<template> - <j-modal - visible - width="650px" - title="上传GeoJson" - :maskClosable="false" - @ok="handleSave" - @cancel="handleCancel" - > - <div> - 仅支持获取单个区域数据 - </div> - <j-monaco-editor - language="json" - style="height: 400px" - v-model:modelValue="myValue" - /> - </j-modal> -</template> - -<script setup name="GeoJsonModal"> - -const props = defineProps({ - value: { - type: String, - default: undefined - } -}) - -const emit = defineEmits(['ok', 'cancel']) - -const myValue = ref(props.value ? JSON.stringify(props.value) : undefined) - -const handleSave = () => { - emit('ok', myValue.value ? JSON.parse(myValue.value) : undefined) -} - -const handleCancel = () => { - emit('cancel') -} - -</script> - -<style scoped> - -</style> diff --git a/src/views/system/Region/util.ts b/src/views/system/Region/util.ts deleted file mode 100644 index cc99d26..0000000 --- a/src/views/system/Region/util.ts +++ /dev/null @@ -1,31 +0,0 @@ -export const MAP_TOOL = { - 'rectangle': 'rectangle', - 'polygon': 'polygon', - 'circle': 'circle', - 'tool': 'tool', - 'district': 'district', - 'geoJson': 'geoJson', -} - -export const syncChildren = (code: string, area: any[]) => { - const findItem = (code: string, children: any[] = []): any[] => { - let _children: any[] = [] - children.some(item => { - if (item.code === code ) { - _children = item.children - return true - } else if (item.children) { - _children = findItem(code, item.children) - return _children.length - } - return false - }) - - return _children.map(item => { - item.children = [] - return item - }) - } - - return findItem(code, area) -} diff --git a/src/views/system/User/index.vue b/src/views/system/User/index.vue index 04cd88e..036cbf6 100644 --- a/src/views/system/User/index.vue +++ b/src/views/system/User/index.vue @@ -1,31 +1,18 @@ <template> <page-container> <div class="user-container"> - <pro-search - :columns="columns" - target="system-user" - @search="handleParams" - /> + <pro-search :columns="columns" target="system-user" @search="handleParams" /> <FullPage> - <j-pro-table - ref="tableRef" - :columns="columns" - :request="getUserList_api" - model="TABLE" - :params="queryParams" - :defaultParams="{ + <j-pro-table ref="tableRef" :columns="columns" :request="getUserList_api" model="TABLE" + :params="queryParams" :defaultParams="{ sorts: [ { name: 'createTime', order: 'desc' }, { name: 'username', order: 'asc', value: 'admin' }, ], - }" - > + }"> <template #headerTitle> - <PermissionButton - :hasPermission="`${permission}:add`" - type="primary" - @click="table.openDialog('add')" - > + <PermissionButton :hasPermission="`${permission}:add`" type="primary" + @click="table.openDialog('add')"> <AIcon type="PlusOutlined" />新增 </PermissionButton> </template> @@ -44,76 +31,46 @@ </j-ellipsis> </template> <template #status="slotProps"> - <BadgeStatus - :status="slotProps.status" - :text="slotProps.status ? '正常' : '禁用'" - :statusNames="{ - 1: 'success', - 0: 'error', - }" - ></BadgeStatus> + <BadgeStatus :status="slotProps.status" :text="slotProps.status ? '正常' : '禁用'" :statusNames="{ + 1: 'success', + 0: 'error', + }"></BadgeStatus> </template> <template #action="slotProps"> <j-space :size="16"> - <PermissionButton - :hasPermission="`${permission}:update`" - type="link" - :tooltip="{ - title: '编辑', - }" - @click="table.openDialog('edit', slotProps)" - > + <PermissionButton :hasPermission="`${permission}:update`" type="link" :tooltip="{ + title: '编辑', + }" @click="table.openDialog('edit', slotProps)"> <AIcon type="EditOutlined" /> </PermissionButton> - <PermissionButton - :hasPermission="`${permission}:action`" - type="link" - :tooltip="{ - title: `${ - slotProps.status ? '禁用' : '启用' + <PermissionButton :hasPermission="`${permission}:action`" type="link" :tooltip="{ + title: `${slotProps.status ? '禁用' : '启用' }`, - }" - :popConfirm="{ - title: `确定${ - slotProps.status ? '禁用' : '启用' - }吗?`, + }" :popConfirm="{ + title: `确定${slotProps.status ? '禁用' : '启用' + }吗?`, onConfirm: () => table.changeStatus(slotProps), - }" - > - <AIcon - :type=" - slotProps.status - ? 'StopOutlined' - : 'PlayCircleOutlined' - " - /> + }"> + <AIcon :type="slotProps.status + ? 'StopOutlined' + : 'PlayCircleOutlined' + " /> </PermissionButton> - <PermissionButton - :hasPermission="`${permission}:update`" - type="link" - :tooltip="{ - title: '重置密码', - }" - @click="table.openDialog('reset', slotProps)" - > + <PermissionButton :hasPermission="`${permission}:update`" type="link" :tooltip="{ + title: '重置密码', + }" @click="table.openDialog('reset', slotProps)"> <AIcon type="icon-zhongzhimima" /> </PermissionButton> - <PermissionButton - type="link" - :hasPermission="`${permission}:delete`" - :tooltip="{ - title: slotProps.status - ? '请先禁用,再删除' - : '删除', - }" - :popConfirm="{ + <PermissionButton type="link" :hasPermission="`${permission}:delete`" :tooltip="{ + title: slotProps.status + ? '请先禁用,再删除' + : '删除', + }" :popConfirm="{ title: '确认删除?', onConfirm: () => table.clickDel(slotProps.id), - }" - :disabled="slotProps.status" - > + }" :disabled="slotProps.status"> <AIcon type="DeleteOutlined" /> </PermissionButton> </j-space> @@ -121,13 +78,8 @@ </j-pro-table> </FullPage> - <EditUserDialog - v-if="dialog.visible" - :type="dialog.type" - v-model:visible="dialog.visible" - :data="dialog.selectItem" - @confirm="table.refresh" - /> + <EditUserDialog v-if="dialog.visible" :type="dialog.type" v-model:visible="dialog.visible" + :data="dialog.selectItem" @confirm="table.refresh" /> </div> </page-container> </template> @@ -361,14 +313,14 @@ const handleParams = (params: any) => { }; } } - if(termsItem.column === 'roleList'){ - if(termsItem.termType === 'eq' || termsItem.termType === 'in'){ + if (termsItem.column === 'roleList') { + if (termsItem.termType === 'eq' || termsItem.termType === 'in') { return { column: 'id$in-dimension$role', type: termsItem.type, value: termsItem.value } - }else{ + } else { return { column: 'id$in-dimension$role$not', type: termsItem.type, diff --git a/src/views/user/Login/utils.ts b/src/views/user/Login/utils.ts deleted file mode 100644 index b9e3f8e..0000000 --- a/src/views/user/Login/utils.ts +++ /dev/null @@ -1,9 +0,0 @@ -import {getImage} from "@/utils/comm"; - -export const defaultImg = getImage('/apply/internal-standalone.png'); -export const iconMap = new Map(); -iconMap.set('dingtalk-ent-app', getImage('/bind/dingtalk.png')); -iconMap.set('wechat-webapp', getImage('/bind/wechat-webapp.png')); -iconMap.set('internal-standalone', getImage('/apply/internal-standalone.png')); -iconMap.set('third-party', getImage('/apply/third-party.png')); -iconMap.set('wechat-miniapp', getImage('/apply/wechat-miniapp.png')); diff --git a/vite.config.ts b/vite.config.ts index f804188..8d49840 100644 --- a/vite.config.ts +++ b/vite.config.ts @@ -94,7 +94,7 @@ export default defineConfig(({ mode }) => { [env.VITE_APP_BASE_API]: { // target: 'http://192.168.32.226:8844', // target: 'http://192.168.32.244:8881', - target: 'http://192.168.1.205:8844', + target: 'http://192.168.1.82:8844', // target: 'http://192.168.32.163:8844', //张本地 // target: 'http://120.77.179.54:8844', // 120测试 // target: 'http://192.168.33.46:8844', // 本地开发环境 diff --git a/yarn.lock b/yarn.lock index c323847..36ac802 100644 --- a/yarn.lock +++ b/yarn.lock @@ -5456,6 +5456,11 @@ dependencies: "readable-stream" "^3.0.0" +"splitpanes@^3.1.5": + "integrity" "sha512-r3Mq2ITFQ5a2VXLOy4/Sb2Ptp7OfEO8YIbhVJqJXoFc9hc5nTXXkCvtVDjIGbvC0vdE7tse+xTM9BMjsszP6bw==" + "resolved" "https://registry.npmjs.org/splitpanes/-/splitpanes-3.1.5.tgz" + "version" "3.1.5" + "string_decoder@^1.1.1": "integrity" "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==" "resolved" "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz" @@ -6555,11 +6560,6 @@ "y18n" "^5.0.5" "yargs-parser" "^21.1.1" -"yarn@^1.22.22": - "integrity" "sha512-prL3kGtyG7o9Z9Sv8IPfBNrWTDmXB4Qbes8A9rEzt6wkJV8mUvoirjU0Mp3GGAU06Y0XQyA3/2/RQFVuK7MTfg==" - "resolved" "https://registry.npmjs.org/yarn/-/yarn-1.22.22.tgz" - "version" "1.22.22" - "yn@3.1.1": "integrity" "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==" "resolved" "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz"