1/* 2 * Copyright (c) 2023 Huawei Device Co., Ltd. 3 * Licensed under the Apache License, Version 2.0 (the "License"); 4 * you may not use this file except in compliance with the License. 5 * You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software 10 * distributed under the License is distributed on an "AS IS" BASIS, 11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 * See the License for the specific language governing permissions and 13 * limitations under the License. 14 */ 15 16const IMAGE_NODE_HEIGHT: number = 24 17const IMAGE_NODE_WIDTH: number = 24 18const ITEM_WIDTH: number = 0 19const ITEM_HEIGHT: number = 48 20const ITEM_HEIGHT_INPUT: number = 32 21const BORDER_WIDTH_HAS: number = 2 22const BORDER_WIDTH_NONE: number = 0 23const NODE_HEIGHT: number = 48 24const LIST_ITEM_HEIGHT_NONE: number = 0 25const LIST_ITEM_HEIGHT: number = 48 26const SHADOW_OFFSETY: number = 10 27const FLAG_NUMBER: number = 2 28const DRAG_OPACITY: number = 0.4 29const DRAG_OPACITY_NONE: number = 1 30const FLAG_LINE_HEIGHT: string = '1.0vp' 31const X_OFF_SET: string = '0vp' 32const Y_OFF_SET: string = '2.75vp' 33const Y_BOTTOM_OFF_SET: string = '-1.25vp' 34const Y_BASE_PLATE_OFF_SET: string = '1.5vp' 35const COLOR_SELECT: string = '#1A0A59F7' 36const COLOR_IMAGE_ROW: string = '#00000000' 37const COLOR_IMAGE_EDIT: string = '#FFFFFF' 38const SHADOW_COLOR: string = '#00001E' 39const GRAG_POP_UP_HEIGHT: string = '48' 40const LEFT_PADDING: string = '8vp' 41const RIGHT_PADDING: string = '8vp' 42const FLOOR_MIN_WIDTH: string = '128vp' 43const FLOOR_MAX_WIDTH: string = '208vp' 44const TEXT_MIN_WIDTH: string = '80vp' 45const TEXT_MAX_WIDTH: string = '160vp' 46const MIN_WIDTH: string = '112vp' 47const MAX_WIDTH: string = '192vp' 48const TRANS_COLOR: string = '#00FFFFFF' 49 50const ARROW_DOWN = '' + 51 'gAAAIDoAAB1MAAA6mAAADqYAAAXcJy6UTwAAAAEZ0FNQQAAsY58+1GTAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzA' + 52 'AAOxAAADsQBlSsOGwAAAq9JREFUeNrt2j9rFEEYx/FJziIECwsLCwsJIVhYiISgaIhiioiIlpaCjZ1/qhNUIkHE0negpZggEnwHFgoWFiJic' + 53 'VyRQsQXIHKIvyGzIJKbmTW7M7P6/cCvyR3c7fPsPLO7F2MAAAAAAAAAAAAAAAAAAAAAAAAAAF01keAz9ihrymVln/JEua18L7w2V5TrykHlt' + 54 'XJTGTb9Ib0EB/JYueWKP6Ucd3mh/Ci0+Hfc9z6gTCuHlUvK06ZPnMkEB3Nth78tK89dQ0os/toOfz+kHG36w1I0YNwZs6JsFtaEccVvTYoGr' + 55 'HteW3ajaKoDxbfz/10X94BXyoIyO+b1Wbcn2JE0ylT8VeW+5/UtZUn52sUG/FQ23PycG/OeGWXevW+U4cwPFf+MMmjjw3uJDnLkRo2vCTlWQ' + 56 'l954Hn9i3Je+djWF+glPNNimlCthJcJLlH7buZPemb+ovKpzS/RS7zcqybMB/aEhZZXwmqg+NXMH7ZdkNQNqJrwLLAxz7Q4jkIzf5iq+LkaU' + 57 'GdjbroJoUvNb674g1SFyNWAOntCU48tYq7zzyqfUxYhZwPqNOHYLi9RQ8XfylH8EhpQZ2P+23EUU/zFlGOntAbU2Zjr3qyF7nCrsTPIdeClN' + 58 'CB2Y66zEvoRG26WsVNqA+rerPlWwg3lYeAmayl38UtsQBOPLezMfxRxkzUo4WBLbEDsxmxXwinlrRsn+5W7yr1SbrJiTJiy2d+T7Y82K4H32' + 59 'QbsNf7fFeyDtRMlFb/kFVBnY7amXbN8l5oXTYtPNf/VBsTuCSZQfPs8/0OJB9eFBuymCXbsnCxlw+1yA2I35j+Lf860/Dz/f2rA73fMdhUc8' + 60 'bzvjXKhxJnftasgH3sJelU5bbb/Z8ee8e/N9j9PrZt8P/ADAAAAAAAAAAAAAAAAAAAAAAAgpV9KuZwVm6MIDQAAAABJRU5ErkJggg==' 61 62const ARROW_DOWN_WITHE = '' + 63 'AhAAA+gAAAIDoAAB1MAAA6mAAADqYAAAXcJy6UTwAAAAEZ0FNQQAAsY58+1GTAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAl' + 64 'wSFlzAAAOxAAADsQBlSsOGwAAAKVJREFUeNpjYBgFo2AU0Bww4pL4////diC1hZGRcSo+A4DqWIDUZCB+AVTbiC7PhEfvByCeAjQgn4Dhy4E' + 65 '4BYgvYFODz4JYIF4DxBOwWYJkeAAQRwBdvxGbIcy4TG9sbPzX0NCwHsjUAuIiIPsDUOwkDsPXkhwHWFwaAsQlQAwyrJsYw4myAIslIPCHGMP' + 66 'xBhGO4PoGxF+AOA9o+NbRTDgKRgFxAAAzj0Grm3RjyAAAAABJRU5ErkJggg==' 67 68const ARROW_RIGHT = '' + 69 '+gAAAIDoAAB1MAAA6mAAADqYAAAXcJy6UTwAAAAEZ0FNQQAAsY58+1GTAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlz' + 70 'AAAOxAAADsQBlSsOGwAAAjVJREFUeNrtnCtvG1EQRq+9BlVUEBAYYFBgYGBYaBBoFUVVfkCtyqDAIL8jtI8EBruVIgU4UvMD0iKDPmirtlJg' + 71 'lJZUyqzsSFXlxwZsPPPtOdJHbDKac+/cXXvtlAAAAAAAAAAAAAAAAAAAAAAUyZzW1bMMLbuWv5YvqgJqTps/sjT+eW1geckOuB+OLdtzpHy3' + 72 'fFQTUHdY06MFr7+xPGMHlE93iYQnajvBo4APlj3LRhUkeBRwaRnProDkJXi9DP1ZFQmZ49oqISFzXl9RCb8sFwhYn4Sw9wlZkDplx1EWqFZJ' + 73 'CVmwHSsnIZoAuYM5ogCpgzmqAJlxFFmAxDiKLiD8OFIQEHocqQgIK0FJwF0kfLZMELA+CTuWI8s1AtYj4YHlt+UcAeVK+JqmX2/OY9Pyat1F' + 74 '1pM2J2n6Fec8mh4KVBdwO27m8RAB5dNb0uhPCCiXTpo+zLWItwgoj/bsKmhrwft/PBzAOQ3RlT9a0vycF5ZvHoqtCTZ/vKL5fcuhl4KVRlA7' + 75 'WvOVBOQP875f0fyBt+arnAH52DmNtvJVzoBwM19pBBVp/r7n5kceQa0CYyfE78pqoiu/733lRxUg1fxoZ0BbrfmRzoAiHy+Ea36UEdRRXPlR' + 76 'RlCR5g+iNt/7CGoqr3zvAuRXvucR1CrQ/GES+fMOjzvgQH3seN8B3ao036uASVWan+Pxybgflqf/LY78wH2dBPF6I/bY8txyZXlnOUsAAAAA' + 77 'AAAAAAAAAAAAAAAAEIgb8WKMjSFbuAQAAAAASUVORK5CYII=' 78 79const ARROW_RIGHT_WITHE = '' + 80 'CAhAAA+gAAAIDoAAB1MAAA6mAAADqYAAAXcJy6UTwAAAAEZ0FNQQAAsY58+1GTAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAA' + 81 'lwSFlzAAAOxAAADsQBlSsOGwAAAKFJREFUeNpjYBgFowAE/v//bwHEPOToZSJS3XIg3k6OJcRaUALEFuRYwkyMosbGxusNDQ3XgMwCIHYAsl' + 82 'cDxX5RzQJKLGEmxbvkWMJEaqQxMjKuBVI5QGwDxOnUimR08AFK81DdAmAqArl8DhDfAOKpVLUAavh2IH4CxI7A4HpDMEgpMPwFUXFGS8NJCa' + 83 'L55BgOAixEqqsB4oOkGj4KRggAAN4STB9zyhGzAAAAAElFTkSuQmCC' 84 85enum Event { 86 TOUCH_DOWN = 0, 87 TOUCH_UP = 1, 88 HOVER = 3, 89 HOVER_OVER = 4, 90 FOCUS = 5, 91 BLUR = 6, 92 MOUSE_BUTTON_RIGHT = 7, 93 DRAG = 8 94} 95 96enum MenuOperation { 97 ADD_NODE = 0, 98 REMOVE_NODE = 1, 99 MODIFY_NODE = 2, 100 COMMIT_NODE = 3 101} 102 103enum PopUpType { 104 HINTS = 0, 105 WARNINGS = 1 106} 107 108enum InputError { 109 INVALID_ERROR = 0, 110 LENGTH_ERROR = 1, 111 NONE = 2 112} 113 114enum Flag { 115 DOWN_FLAG = 0, 116 UP_FLAG = 1, 117 NONE = 2 118} 119 120export enum NodeStatus { 121 Expand = 0, 122 Collapse 123} 124 125export enum InteractionStatus { 126 Normal = 0, 127 Selected, 128 Edit, 129 FinishEdit, 130 DragInsert, 131 FinishDragInsert 132} 133 134/* nodeId to find index */ 135function findCurrentNodeIndex(this, currentNodeId: number): number { 136 let thisIndex: number = 0; 137 this.listNodeDataSource.ListNode.forEach(function (value, index) { 138 if (value.getNodeCurrentNodeId() == currentNodeId) { 139 thisIndex = index; 140 } 141 }) 142 return thisIndex; 143} 144 145@Observed 146export class NodeInfo { 147 private childNodeInfo: { isHasChildNode: boolean, childNum: number, allChildNum: number }; 148 imageSource: Resource | string; 149 private parentNodeId: number; 150 private currentNodeId: number; 151 fontColor: string | Resource; 152 private nodeHeight: Resource | number; 153 private nodeLevel: number; 154 private nodeItem: { imageNode?: TreeView.ImageNode, 155 inputText: TreeView.InputText, 156 mainTitleNode?: TreeView.MainTitleNode, 157 imageCollapse?: TreeView.ImageNode }; 158 private nodeLeftPadding: number; 159 private nodeColor: Resource | string; 160 private nodeIsShow: boolean; 161 private status: { normal: Resource, hover: Resource, press: Resource, selected: string, highLight: Resource }; 162 private nodeBorder: { borderWidth: Resource | number, borderColor: Resource, borderRadius: Resource }; 163 private popUpInfo: { popUpIsShow: boolean, 164 popUpEnableArrow: boolean, 165 popUpColor: Resource, 166 popUpText: string | Resource, 167 popUpTextColor: Resource}; 168 private listItemHeight: number; 169 private container: () => void; 170 private isShowTitle: boolean; 171 private isShowInputText: boolean; 172 private isSelected: boolean; 173 readonly borderWidth: {has: Resource | number, none: Resource | number } = 174 {has: BORDER_WIDTH_HAS/* 2vp */, none: BORDER_WIDTH_NONE/* 0vp */} 175 176 /* parameter of the drag event.*/ 177 private nodeParam: { 178 isFolder?: boolean, 179 icon?: Resource, 180 selectedIcon?: Resource, 181 editIcon?: Resource, 182 primaryTitle?: string, 183 container?: () => void, 184 secondaryTitle?: number | string 185 }; 186 private node: TreeView.NodeItem; 187 private canShowFlagLine: boolean = false; 188 private isOverBorder: boolean = false; 189 private canShowBottomFlagLine: boolean = false; 190 private isHighLight: boolean = false; 191 private flagLineLeftMargin: number; 192 private isModify: boolean = false; 193 194 constructor(node: TreeView.NodeItem) { 195 this.childNodeInfo = node.getChildNodeInfo(); 196 this.nodeItem = { imageNode: null, inputText: null, mainTitleNode: null, imageCollapse: null }; 197 this.popUpInfo = { popUpIsShow: false, 198 popUpEnableArrow: false, 199 popUpColor: null, 200 popUpText: '', 201 popUpTextColor: null }; 202 this.nodeItem.imageNode = node.getNodeItem().imageNode; 203 this.nodeItem.inputText = new TreeView.InputText(); 204 this.nodeItem.mainTitleNode = node.getNodeItem().mainTitleNode; 205 this.nodeItem.imageCollapse = node.getNodeItem().imageCollapse; 206 this.container = node.container; 207 this.parentNodeId = node.parentNodeId; 208 this.currentNodeId = node.currentNodeId; 209 this.nodeHeight = NODE_HEIGHT; 210 this.nodeLevel = node.nodeLevel; 211 this.nodeLeftPadding = node.nodeLevel * 12 + 8; // calculate left padding 212 this.nodeColor = $r('sys.color.ohos_id_color_background'); 213 this.nodeIsShow = (this.nodeLevel > 0) ? false : true; 214 this.listItemHeight = (this.nodeLevel > 0) ? LIST_ITEM_HEIGHT_NONE : LIST_ITEM_HEIGHT; 215 this.isShowTitle = true; 216 this.isShowInputText = false; 217 this.isSelected = false; 218 this.status = { normal: $r('sys.color.ohos_id_color_background_transparent'), 219 hover: $r('sys.color.ohos_id_color_hover'), 220 press: $r('sys.color.ohos_id_color_click_effect'), 221 selected: COLOR_SELECT, 222 highLight: $r('sys.color.ohos_id_color_activated') 223 }; 224 this.nodeBorder = { borderWidth: BORDER_WIDTH_NONE, 225 borderColor: $r('sys.color.ohos_id_color_focused_outline'), 226 borderRadius: $r('sys.float.ohos_id_corner_radius_clicked') 227 }; 228 this.flagLineLeftMargin = node.nodeLevel * 12 + 8; 229 this.node = node; 230 this.nodeParam = node.data; 231 } 232 233 setFontColor(color: string | Resource) { 234 this.fontColor = color 235 } 236 237 getFontColor() { 238 return this.fontColor; 239 } 240 241 getPopUpInfo() { 242 return this.popUpInfo; 243 } 244 245 setPopUpIsShow(isShow: boolean) { 246 this.popUpInfo.popUpIsShow = isShow; 247 } 248 249 setPopUpEnableArrow(popUpEnableArrow: boolean) { 250 this.popUpInfo.popUpEnableArrow = popUpEnableArrow; 251 } 252 253 setPopUpColor(color: Resource) { 254 this.popUpInfo.popUpColor = color; 255 } 256 257 setPopUpText(text: string | Resource) { 258 this.popUpInfo.popUpText = text; 259 } 260 261 setPopUpTextColor(popUpTextColor: Resource) { 262 this.popUpInfo.popUpTextColor = popUpTextColor; 263 } 264 265 getIsShowTitle() { 266 return this.isShowTitle; 267 } 268 269 getIsShowInputText() { 270 return this.isShowInputText; 271 } 272 273 setTitleAndInputTextStatus(isModify: boolean) { 274 if (isModify) { 275 this.isShowTitle = false; 276 this.isShowInputText = true; 277 } else { 278 this.isShowTitle = true; 279 this.isShowInputText = false; 280 } 281 } 282 283 handleImageCollapseAfterAddNode(isAddImageCollapse: boolean) { 284 // listTree this node already has ImageCollapse. 285 if (isAddImageCollapse) { 286 this.nodeItem.imageCollapse = new TreeView.ImageNode(ARROW_DOWN, null, null, 287 $r('sys.float.ohos_id_alpha_content_tertiary'), 288 IMAGE_NODE_HEIGHT, 289 IMAGE_NODE_WIDTH); 290 this.nodeItem.imageCollapse.itemRightMargin = ($r('sys.float.ohos_id_text_paragraph_margin_xs')); 291 } else { 292 this.nodeItem.imageCollapse = null; 293 } 294 } 295 296 setNodeColor(nodeColor: Resource | string): void { 297 this.nodeColor = nodeColor; 298 } 299 300 getNodeColor(): Resource | string { 301 return this.nodeColor; 302 } 303 304 setListItemHeight(listItemHeight: number): void { 305 this.listItemHeight = listItemHeight; 306 } 307 308 getListItemHeight(): number { 309 return this.listItemHeight; 310 } 311 312 getNodeCurrentNodeId(): number { 313 return this.currentNodeId; 314 } 315 316 getNodeParentNodeId(): number { 317 return this.parentNodeId; 318 } 319 320 getNodeLeftPadding(): number { 321 return this.nodeLeftPadding; 322 } 323 324 getNodeHeight(): Resource | number { 325 return this.nodeHeight; 326 } 327 328 setNodeIsShow(nodeIsShow: boolean): void { 329 this.nodeIsShow = nodeIsShow; 330 } 331 332 getNodeIsShow(): boolean { 333 return this.nodeIsShow; 334 } 335 336 getNodeItem() { 337 return this.nodeItem; 338 } 339 340 getNodeStatus() { 341 return this.status; 342 } 343 344 getNodeBorder() { 345 return this.nodeBorder; 346 } 347 348 setNodeBorder(isClearFocusStatus: boolean): void { 349 this.nodeBorder.borderWidth = isClearFocusStatus ? this.borderWidth.has : this.borderWidth.none; 350 } 351 352 getChildNodeInfo() { 353 return this.childNodeInfo; 354 } 355 356 getCurrentNodeId() { 357 return this.currentNodeId; 358 } 359 360 getMenu() { 361 return this.container; 362 } 363 364 setIsSelected(isSelected: boolean) { 365 this.isSelected = isSelected; 366 } 367 368 getIsSelected() { 369 return this.isSelected; 370 } 371 372 /* To gain the information while to alter node. */ 373 getNodeInfoData() { 374 return this.nodeParam; 375 } 376 377 /* To gain the tree Node(NodeItem) while to alter node. */ 378 public getNodeInfoNode() { 379 return this.node; 380 } 381 382 public getIsFolder() { 383 return this.nodeParam.isFolder; 384 } 385 386 public setCanShowFlagLine(canShowFlagLine: boolean) { 387 this.canShowFlagLine = canShowFlagLine; 388 } 389 390 public getCanShowFlagLine(): boolean { 391 return this.canShowFlagLine; 392 } 393 394 public setFlagLineLeftMargin(currentNodeLevel: number) { 395 this.flagLineLeftMargin = currentNodeLevel * 12 + 8; // calculate 396 } 397 398 public getFlagLineLeftMargin(): number { 399 return this.flagLineLeftMargin; 400 } 401 402 public getNodeLevel(): number { 403 return this.nodeLevel; 404 } 405 406 public setIsOverBorder(isOverBorder: boolean) { 407 this.isOverBorder = isOverBorder; 408 } 409 410 public getIsOverBorder() { 411 return this.isOverBorder; 412 } 413 414 public setCanShowBottomFlagLine(canShowBottomFlagLine: boolean) { 415 this.canShowBottomFlagLine = canShowBottomFlagLine; 416 } 417 418 public getCanShowBottomFlagLine() { 419 return this.canShowBottomFlagLine; 420 } 421 422 public setIsHighLight(isHighLight: boolean) { 423 this.isHighLight = isHighLight; 424 } 425 426 public getIsHighLight(): boolean { 427 return this.isHighLight; 428 } 429 430 public setIsModify(isModify: boolean) { 431 this.isModify = isModify; 432 } 433 434 public getIsModify(): boolean { 435 return this.isModify; 436 } 437 438} 439 440export namespace TreeView { 441 442 /* 443 * TreeListenType listen type. 444 * 445 * @since 10 446 */ 447 export enum TreeListenType { 448 NODE_ADD = "NodeAdd", 449 NODE_DELETE = "NodeDelete", 450 NODE_MODIFY = "NodeModify", 451 NODE_MOVE = "NodeMove", 452 NODE_CLICK = 'NodeClick', 453 } 454 455 export class TreeListener { 456 _events = [] 457 458 constructor() { 459 } 460 461 /* 462 * Event registration and processing. 463 * 464 * The event will not be destroyed after being processed. 465 * 466 * @param type Registered Events. 467 * @param callback Event callback. 468 * @since 10 469 */ 470 public on(type: TreeListenType, callback: (callbackParam: CallbackParam) => void) { 471 if (Array.isArray(type)) { 472 for (let i = 0, l = type.length; i < l; i++) { 473 this.on(type[i], callback) 474 } 475 } else { 476 (this._events[type] || (this._events[type] = [])).push(callback) 477 } 478 } 479 480 /* 481 * Event registration and processing. 482 * 483 * After the event is processed once, it will be destroyed. 484 * 485 * @param type Registered Events. 486 * @param callback Event callback. 487 * @since 10 488 */ 489 public once(type: TreeListenType, callback: (callbackParam: CallbackParam) => void) { 490 let _self = this; 491 function handler() { 492 _self.off(type, handler); 493 callback.apply(null, [type, callback]); 494 } 495 496 handler.callback = callback; 497 this.on(type, handler); 498 } 499 500 /* 501 * Destroy event. 502 * 503 * @param type Registered Events. 504 * @param callback Event callback. 505 * @since 10 506 */ 507 public off(type: TreeListenType, callback: (callbackParam: CallbackParam) => void) { 508 if (type == null) { 509 this._events = []; 510 } 511 if (Array.isArray(type)) { 512 for (let i = 0, l = type.length; i < l; i++) { 513 this.off(type[i], callback) 514 } 515 } 516 const cbs = this._events[type]; 517 if (!cbs) { 518 return; 519 } 520 if (callback == null) { 521 this._events[type] = null 522 } 523 let cb, i = cbs.length 524 while (i--) { 525 cb = cbs[i] 526 if (cb === callback || cb.callback === callback) { 527 cbs.splice(i, 1) 528 break 529 } 530 } 531 } 532 533 /* 534 * Triggers all callbacks of an event with parameters. 535 * 536 * @param event Registered Events. 537 * @param argument Parameters returned by the callback event. 538 * @since 10 539 */ 540 public emit(event, argument: any[]) { 541 let _self = this 542 if (!this._events[event]) { 543 return 544 } 545 let cbs = [...this._events[event]]; 546 if (cbs) { 547 for (let i = 0, l = cbs.length; i < l; i++) { 548 try { 549 cbs[i].apply(_self,argument) 550 } catch (e) { 551 new Error(e) 552 } 553 } 554 } 555 } 556 } 557 558 /* 559 * TreeListenerManager. 560 * 561 * @since 10 562 */ 563 export class TreeListenerManager { 564 static readonly APP_KEY_EVENT_BUS = "app_key_event_bus"; 565 private appEventBus: TreeListener; 566 private constructor() { 567 this.appEventBus = new TreeListener(); 568 } 569 570 /* 571 * Obtains the EventBusManager object. 572 * 573 * @since 10 574 */ 575 public static getInstance(): TreeListenerManager { 576 if (AppStorage.Get(this.APP_KEY_EVENT_BUS) == null) { 577 AppStorage.SetOrCreate(this.APP_KEY_EVENT_BUS, new TreeListenerManager()) 578 } 579 return AppStorage.Get(this.APP_KEY_EVENT_BUS); 580 } 581 582 /* 583 * Obtains the EventBus object. 584 * 585 * @since 10 586 */ 587 public getTreeListener(): TreeListener { 588 return this.appEventBus; 589 } 590 } 591 592 class BasicDataSource implements IDataSource { 593 private listeners: DataChangeListener[] = [] 594 595 public totalCount(): number { 596 return 0 597 } 598 public getData(index: number): any { 599 return undefined 600 } 601 602 registerDataChangeListener(listener: DataChangeListener): void { 603 if (this.listeners.indexOf(listener) < 0) { 604 this.listeners.push(listener) 605 } 606 } 607 unregisterDataChangeListener(listener: DataChangeListener): void { 608 const pos = this.listeners.indexOf(listener); 609 if (pos >= 0) { 610 this.listeners.splice(pos, 1) 611 } 612 } 613 614 notifyDataReload(): void { 615 this.listeners.forEach(listener => { 616 listener.onDataReloaded() 617 }) 618 } 619 notifyDataAdd(index: number): void { 620 this.listeners.forEach(listener => { 621 listener.onDataAdd(index) 622 }) 623 } 624 notifyDataChange(index: number): void { 625 this.listeners.forEach(listener => { 626 listener.onDataChange(index) 627 }) 628 } 629 notifyDataDelete(index: number): void { 630 this.listeners.forEach(listener => { 631 listener.onDataDelete(index) 632 }) 633 } 634 notifyDataMove(from: number, to: number): void { 635 this.listeners.forEach(listener => { 636 listener.onDataMove(from, to) 637 }) 638 } 639 } 640 641 export class NodeItem { 642 private nodeItem: { imageNode?: ImageNode, 643 mainTitleNode?: MainTitleNode, 644 fontColor?: Resource, 645 imageCollapse?: ImageNode}; 646 private childNodeInfo: { isHasChildNode: boolean, childNum: number, allChildNum: number }; 647 container: () => void; 648 nodeLevel: number; 649 parentNodeId: number; 650 currentNodeId: number; 651 children: Array<NodeItem>; 652 data: { isFolder?: boolean, 653 icon?: Resource, 654 selectedIcon?: Resource, 655 editIcon?: Resource, 656 primaryTitle?: string, 657 container?: () => void, 658 objectCount?: number | string } 659 660 constructor(data: { isFolder?: boolean, 661 icon?: Resource, 662 selectedIcon?: Resource, 663 editIcon?: Resource, 664 primaryTitle?: string, 665 container?: () => void, 666 objectCount?: number | string }) { 667 this.data = data; 668 this.nodeLevel = -1; 669 this.parentNodeId = -1; 670 this.nodeItem = { imageNode: null, mainTitleNode: null, imageCollapse: null }; 671 this.childNodeInfo = { isHasChildNode: false, childNum: 0, allChildNum: 0 }; 672 this.container = data.container; 673 if (data.icon) { 674 this.nodeItem.imageNode = new ImageNode(data.icon, data.selectedIcon, data.editIcon, 675 $r('sys.float.ohos_id_alpha_content_fourth'), 676 IMAGE_NODE_HEIGHT, 677 IMAGE_NODE_WIDTH); 678 } 679 if (data.primaryTitle) { 680 this.nodeItem.mainTitleNode = new MainTitleNode(data.primaryTitle); 681 } 682 this.children = []; 683 } 684 685 addImageCollapse(isHasChildNode: boolean) { 686 if (isHasChildNode) { 687 this.nodeItem.imageCollapse = new ImageNode(ARROW_RIGHT, null, null, 688 $r('sys.float.ohos_id_alpha_content_tertiary'), 689 IMAGE_NODE_HEIGHT, 690 IMAGE_NODE_WIDTH); 691 this.nodeItem.imageCollapse.itemRightMargin = ($r('sys.float.ohos_id_text_paragraph_margin_xs')); 692 } else { 693 this.nodeItem.imageCollapse = null; 694 } 695 } 696 697 getNodeItem() { 698 return this.nodeItem; 699 } 700 701 getChildNodeInfo() { 702 return this.childNodeInfo; 703 } 704 705 getMenu(): () => void { 706 return this.container; 707 } 708 709 getCurrentNodeId() { 710 return this.currentNodeId; 711 } 712 713 getIsFolder() { 714 return this.data.isFolder; 715 } 716 } 717 718 class NodeBaseInfo { 719 public rightMargin: Resource | number; 720 private width: number; 721 private height: number; 722 constructor() { 723 } 724 725 set itemWidth(width: number) { 726 this.width = width; 727 } 728 729 get itemWidth(): number { 730 return this.width; 731 } 732 733 set itemHeight(height: number) { 734 this.height = height; 735 } 736 737 get itemHeight(): number { 738 return this.height; 739 } 740 741 set itemRightMargin(rightMargin: Resource | number) { 742 this.rightMargin = rightMargin; 743 } 744 745 get itemRightMargin() { 746 return this.rightMargin; 747 } 748 } 749 750 export class ImageNode extends NodeBaseInfo { 751 private imageSource: Resource | string; 752 private imageNormalSource: Resource | string; 753 private imageSelectedSource: Resource | string; 754 private imageEditSource: Resource | string; 755 private imageOpacity: Resource; 756 private currentInteractionStatus: InteractionStatus; 757 private imageCollapseSource: Resource | string; 758 private imageCollapseDownSource: Resource | string; 759 private isImageCollapse: boolean; 760 private imageCollapseRightSource: Resource | string; 761 constructor(imageSource: Resource | string, itemSelectedIcon: Resource, itemEditIcon: Resource, 762 imageOpacity: Resource, itemWidth: number, itemHeight: number) { 763 super(); 764 this.rightMargin = $r('sys.float.ohos_id_elements_margin_horizontal_m'); 765 this.imageSource = imageSource; 766 this.imageNormalSource = imageSource; 767 if (itemSelectedIcon != null) { 768 this.imageSelectedSource = itemSelectedIcon; 769 } else { 770 this.imageSelectedSource = this.imageNormalSource; 771 } 772 if (itemEditIcon != null) { 773 this.imageEditSource = itemEditIcon; 774 } else { 775 this.imageEditSource = this.imageNormalSource; 776 } 777 this.imageOpacity = imageOpacity; 778 this.itemWidth = itemWidth; 779 this.itemHeight = itemHeight; 780 this.imageCollapseSource = imageSource; 781 this.imageCollapseDownSource = ARROW_DOWN; 782 this.imageCollapseRightSource = ARROW_RIGHT; 783 this.isImageCollapse = true; 784 } 785 786 get source() { 787 return this.imageSource; 788 } 789 790 get normalSource() { 791 return this.imageNormalSource; 792 } 793 794 get selectedSource() { 795 return this.imageSelectedSource; 796 } 797 798 get editSource() { 799 return this.imageEditSource; 800 } 801 802 get opacity() { 803 return this.imageOpacity; 804 } 805 806 get noOpacity() { 807 return 1; 808 } 809 810 get collapseSource() { 811 return this.imageCollapseSource; 812 } 813 814 get isCollapse() { 815 return this.isImageCollapse; 816 } 817 818 changeImageCollapseSource(nodeStatus: NodeStatus) { 819 if (nodeStatus == NodeStatus.Expand) { 820 this.imageCollapseSource = this.imageCollapseDownSource; 821 } else if (nodeStatus == NodeStatus.Collapse) { 822 this.imageCollapseSource = this.imageCollapseRightSource; 823 } 824 } 825 826 setImageCollapseSource(interactionStatus: InteractionStatus, nodeStatus: NodeStatus) { 827 if (interactionStatus === InteractionStatus.Edit || interactionStatus === InteractionStatus.DragInsert) { 828 this.imageCollapseDownSource = ARROW_DOWN_WITHE; 829 this.imageCollapseRightSource = ARROW_RIGHT_WITHE; 830 this.isImageCollapse = false; 831 } else if (interactionStatus === InteractionStatus.FinishEdit || 832 interactionStatus === InteractionStatus.FinishDragInsert) { 833 this.imageCollapseDownSource = ARROW_DOWN 834 this.imageCollapseRightSource = ARROW_RIGHT 835 this.isImageCollapse = true; 836 } 837 this.imageCollapseSource = (nodeStatus == NodeStatus.Collapse) ? 838 this.imageCollapseRightSource : this.imageCollapseDownSource; 839 } 840 841 setImageSource(interactionStatus: InteractionStatus) { 842 switch (interactionStatus) { 843 case InteractionStatus.Normal: 844 this.imageSource = this.imageNormalSource; 845 this.currentInteractionStatus = interactionStatus; 846 break; 847 case InteractionStatus.Selected: 848 if (this.currentInteractionStatus !== InteractionStatus.Edit) { 849 this.imageSource = this.imageSelectedSource; 850 this.currentInteractionStatus = interactionStatus; 851 } 852 break; 853 case InteractionStatus.Edit: 854 this.imageSource = this.imageEditSource; 855 this.currentInteractionStatus = interactionStatus; 856 break; 857 case InteractionStatus.FinishEdit: 858 this.imageSource = this.imageSelectedSource; 859 this.currentInteractionStatus = interactionStatus; 860 break; 861 case InteractionStatus.DragInsert: 862 this.imageSource = this.imageEditSource; 863 this.currentInteractionStatus = interactionStatus; 864 break; 865 case InteractionStatus.FinishDragInsert: 866 this.imageSource = this.imageNormalSource; 867 this.currentInteractionStatus = interactionStatus; 868 break; 869 default: 870 break; 871 } 872 } 873 } 874 875 export class MainTitleNode extends NodeBaseInfo { 876 private mainTitleName: string; 877 mainTitleSetting: { fontColor: Resource, fontSize: Resource, fontWeight: FontWeight } 878 private showPopUpTimeout: number; 879 constructor(mainTitleName: string) { 880 super(); 881 this.mainTitleName = mainTitleName; 882 this.itemWidth = ITEM_WIDTH; 883 this.itemHeight = ITEM_HEIGHT; 884 this.rightMargin = $r('sys.float.ohos_id_text_paragraph_margin_xs'); 885 this.mainTitleSetting = { fontColor: $r('sys.color.ohos_id_color_primary'), 886 fontSize: $r('sys.float.ohos_id_text_size_body1'), 887 fontWeight: FontWeight.Normal }; 888 this.showPopUpTimeout = 0; 889 } 890 setMainTitleSelected(isSelected: boolean): void { 891 if (isSelected) { 892 this.mainTitleSetting = { fontColor: $r('sys.color.ohos_id_color_text_primary_activated'), 893 fontSize: $r('sys.float.ohos_id_text_size_body1'), 894 fontWeight: FontWeight.Regular }; 895 } else { 896 this.mainTitleSetting = { fontColor: $r('sys.color.ohos_id_color_primary'), 897 fontSize: $r('sys.float.ohos_id_text_size_body1'), 898 fontWeight: FontWeight.Normal }; 899 } 900 } 901 set title(text: string) { 902 this.mainTitleName = text; 903 } 904 get title(): string { 905 return this.mainTitleName; 906 } 907 908 set popUpTimeout(showPopUpTimeout: number) { 909 this.showPopUpTimeout = showPopUpTimeout; 910 } 911 912 get popUpTimeout() { 913 return this.showPopUpTimeout; 914 } 915 916 get color(): Resource { 917 return this.mainTitleSetting.fontColor; 918 } 919 920 get size(): Resource { 921 return this.mainTitleSetting.fontSize; 922 } 923 924 get weight(): FontWeight { 925 return this.mainTitleSetting.fontWeight; 926 } 927 928 setMainTitleHighLight(isHighLight: boolean): void { 929 if (isHighLight) { 930 this.mainTitleSetting = { fontColor: $r('sys.color.ohos_id_color_primary_contrary'), 931 fontSize: $r('sys.float.ohos_id_text_size_body1'), 932 fontWeight: FontWeight.Regular }; 933 } else { 934 this.mainTitleSetting = { fontColor: $r('sys.color.ohos_id_color_primary'), 935 fontSize: $r('sys.float.ohos_id_text_size_body1'), 936 fontWeight: FontWeight.Normal }; 937 } 938 } 939 } 940 941 export class InputText extends NodeBaseInfo { 942 private inputTextSetting: { fontColor: Resource, fontSize: Resource, fontWeight: FontWeight } 943 private status: { normal: Resource, hover: Resource, press: Resource }; 944 private statusColor: Resource = $r('sys.color.ohos_id_color_background'); 945 private editItemColor: Resource = $r('sys.color.ohos_id_color_emphasize'); 946 private radius: Resource = $r('sys.float.ohos_id_corner_radius_default_xs') 947 constructor() { 948 super(); 949 this.itemWidth = ITEM_WIDTH; 950 this.itemHeight = ITEM_HEIGHT_INPUT; 951 this.rightMargin = $r('sys.float.ohos_id_text_paragraph_margin_xs'); 952 this.inputTextSetting = { 953 fontColor: $r('sys.color.ohos_id_color_text_primary'), 954 fontSize: $r('sys.float.ohos_id_text_size_body1'), 955 fontWeight: FontWeight.Normal }; 956 } 957 958 get color(): Resource { 959 return this.inputTextSetting.fontColor; 960 } 961 962 get size(): Resource { 963 return this.inputTextSetting.fontSize; 964 } 965 966 get weight(): FontWeight { 967 return this.inputTextSetting.fontWeight; 968 } 969 970 get borderRadius(): Resource { 971 return this.radius; 972 } 973 974 get backgroundColor() { 975 return this.statusColor; 976 } 977 978 get editColor() { 979 return this.editItemColor; 980 } 981 982 get textInputStatusColor() { 983 return this.status; 984 } 985 } 986 987 export class ListNodeUtils { 988 private _root: NodeItem; 989 public addNewNodeId: number; 990 private readonly MaxNodeLevel = 50; 991 private readonly MAX_CN_LENGTH: number = 254; 992 private readonly MAX_EN_LENGTH: number = 255; 993 private readonly INITIAL_INVALID_VALUE = -1; 994 constructor() { 995 this._root = new NodeItem({}); 996 this._root.nodeLevel = -1; 997 this._root.parentNodeId = -1; 998 this._root.currentNodeId = -1; 999 } 1000 1001 getNewNodeId() { 1002 return this.addNewNodeId; 1003 } 1004 1005 traverseNodeDF(callback, root: NodeItem = this._root) { 1006 let stack = [], found = false; 1007 stack.unshift(root); 1008 let currentNode = stack.shift(); 1009 while(!found && currentNode) { 1010 found = callback(currentNode) === true; 1011 if (!found) { 1012 stack.unshift(...currentNode.children); 1013 currentNode = stack.shift(); 1014 } 1015 } 1016 } 1017 1018 traverseNodeBF(callback) { 1019 let queue = []; 1020 let found: boolean = false; 1021 queue.push(this._root); 1022 let currentNode: NodeItem = queue.shift(); 1023 while(!found && currentNode) { 1024 try { 1025 found = callback(currentNode); 1026 } catch(err) { 1027 var e = err.name + " == " + err.message; 1028 } 1029 if (!found) { 1030 queue.push(...currentNode.children) 1031 currentNode = queue.shift(); 1032 } 1033 } 1034 } 1035 1036 private contains(callback, traversal) { 1037 traversal.call(this, callback, true); 1038 } 1039 1040 private updateParentChildNum(parentNode: NodeItem, isAdd: boolean, count: number) { 1041 let parentNodeId: number = parentNode.parentNodeId; 1042 while(parentNodeId >= 0) { 1043 this.traverseNodeDF((node: NodeItem): boolean => { 1044 if (node.currentNodeId == parentNodeId) { 1045 node.getChildNodeInfo().allChildNum = 1046 isAdd ? node.getChildNodeInfo().allChildNum + count : node.getChildNodeInfo().allChildNum - count; 1047 parentNodeId = node.parentNodeId; 1048 return false; 1049 } 1050 return false; 1051 }) 1052 } 1053 } 1054 1055 findParentNodeId(currentNodeId: number): number { 1056 let current = null, 1057 callback = function(node): boolean { 1058 if (node.currentNodeId == currentNodeId ) { 1059 current = node; 1060 return true; 1061 } 1062 return false; 1063 }; 1064 this.contains(callback, this.traverseNodeBF); 1065 return current.parentNodeId; 1066 } 1067 1068 addNode(parentNodeId: number, 1069 currentNodeId: number, 1070 data: { isFolder?: boolean, 1071 icon?: Resource, 1072 selectedIcon?: Resource, 1073 editIcon?: Resource, 1074 primaryTitle?: string, 1075 secondaryTitle?: number | string, 1076 container?: () => void, 1077 }): ListNodeUtils { 1078 if (this._root === null) { 1079 this._root = new NodeItem({}); 1080 this._root.nodeLevel = -1; 1081 this._root.parentNodeId = -1; 1082 this._root.currentNodeId = -1; 1083 } 1084 1085 let parent = null, 1086 callback = function(node): boolean { 1087 if (node.currentNodeId == parentNodeId ) { 1088 parent = node; 1089 return true; 1090 } 1091 return false; 1092 }; 1093 this.contains(callback, this.traverseNodeBF); 1094 if (parent) { 1095 let currentNode: NodeItem = new NodeItem(data); 1096 if (parent.nodeLevel > this.MaxNodeLevel) { 1097 throw new Error('ListNodeUtils[addNode]: The level of the tree view cannot exceed 50.'); 1098 } 1099 currentNode.nodeLevel = parent.nodeLevel + 1; // nodeLevel 1100 currentNode.parentNodeId = parentNodeId; 1101 currentNode.currentNodeId = currentNodeId; 1102 parent.children.push(currentNode); 1103 parent.getChildNodeInfo().isHasChildNode = true; 1104 parent.getChildNodeInfo().childNum = parent.children.length; 1105 parent.getChildNodeInfo().allChildNum += 1; // childNum 1106 parent.addImageCollapse(parent.getChildNodeInfo().isHasChildNode); 1107 this.updateParentChildNum(parent, true, 1); 1108 return this; 1109 } else { 1110 throw new Error('ListNodeUtils[addNode]: Parent node not found.'); 1111 } 1112 } 1113 1114 findNodeIndex(children, currentNodeId: number) { 1115 let index = this.INITIAL_INVALID_VALUE; 1116 for (let i = 0, len = children.length; i < len; i++) { 1117 if (children[i].currentNodeId === currentNodeId) { 1118 index = i; 1119 break; 1120 } 1121 } 1122 return index; 1123 } 1124 1125 private freeNodeMemory(rootNode: NodeItem, removeNodeIdList: number[]) { 1126 let deleteNode: NodeItem[] = []; 1127 let callback = function(node): boolean { 1128 deleteNode.push(node); 1129 return false; 1130 }; 1131 this.traverseNodeDF(callback, rootNode); 1132 deleteNode.forEach((value)=>{ 1133 removeNodeIdList.push(value.currentNodeId); 1134 value = null; 1135 }) 1136 } 1137 1138 removeNode(currentNodeId: number, parentNodeId: number, traversal: any) { 1139 let parent = null, 1140 callback = function(node): boolean { 1141 if (node.currentNodeId == parentNodeId) { 1142 parent = node; 1143 return true; 1144 } 1145 return false; 1146 }; 1147 this.contains(callback, traversal); 1148 1149 if (parent) { 1150 let removeNodeIdList: number[] = []; 1151 let index = this.findNodeIndex(parent.children, currentNodeId); 1152 if (index < 0) { 1153 throw new Error('Node does not exist.'); 1154 } else { 1155 var deleteNodeAllChildNum = parent.children[index].getChildNodeInfo().allChildNum + 1; 1156 this.freeNodeMemory(parent.children[index], removeNodeIdList); 1157 let node = parent.children.splice(index, 1); 1158 node = null; 1159 if (parent.children.length == 0) { 1160 parent.addImageCollapse(false); 1161 } 1162 } 1163 parent.getChildNodeInfo().childNum = parent.children.length; 1164 parent.getChildNodeInfo().allChildNum -= (deleteNodeAllChildNum); 1165 this.updateParentChildNum(parent, false, deleteNodeAllChildNum); 1166 return removeNodeIdList; 1167 } else { 1168 throw new Error('Parent does not exist.'); 1169 } 1170 } 1171 1172 getNewNodeInfo(nodeId: number) { 1173 let parent = null, 1174 callback = function(node): boolean { 1175 if (node.currentNodeId == nodeId) { 1176 parent = node; 1177 return true; 1178 } 1179 return false; 1180 }; 1181 this.contains(callback, this.traverseNodeBF); 1182 let newNodeInfo: { isFolder: boolean, icon: Resource, selectedIcon: Resource, editIcon: Resource, container: () => any, secondaryTitle: number | string } = 1183 { isFolder: true, icon: null, selectedIcon: null, editIcon: null, container: null, secondaryTitle: '' }; 1184 if (parent) { 1185 if (parent.children.length === 0) { 1186 if (parent.getNodeItem().imageNode != null) { 1187 newNodeInfo.icon = parent.getNodeItem().imageNode.normalSource; 1188 newNodeInfo.selectedIcon = parent.getNodeItem().imageNode.selectedSource; 1189 newNodeInfo.editIcon = parent.getNodeItem().imageNode.editSource; 1190 newNodeInfo.container = parent.getMenu(); 1191 } else { 1192 newNodeInfo.icon = null; 1193 newNodeInfo.selectedIcon = null; 1194 newNodeInfo.editIcon = null; 1195 newNodeInfo.container = parent.getMenu(); 1196 } 1197 } else if (parent.children.length > 0) { 1198 if (parent.getNodeItem().imageNode != null) { 1199 newNodeInfo.icon = (parent.children[0].getNodeItem().imageNode != null) ? 1200 parent.children[0].getNodeItem().imageNode.normalSource : null; 1201 newNodeInfo.selectedIcon = (parent.children[0].getNodeItem().imageNode != null) ? 1202 parent.children[0].getNodeItem().imageNode.selectedSource : null; 1203 newNodeInfo.editIcon = (parent.children[0].getNodeItem().imageNode != null) ? 1204 parent.children[0].getNodeItem().imageNode.editSource : null; 1205 newNodeInfo.container = parent.children[0].getMenu(); 1206 } else { 1207 newNodeInfo.icon = null; 1208 newNodeInfo.selectedIcon = null; 1209 newNodeInfo.editIcon = null; 1210 newNodeInfo.container = parent.children[0].getMenu(); 1211 } 1212 } 1213 } 1214 return newNodeInfo; 1215 } 1216 1217 getClickChildId(nodeId: number) { 1218 let parent = null, 1219 callback = function(node): boolean { 1220 if (node.currentNodeId == nodeId) { 1221 parent = node; 1222 return true; 1223 } 1224 return false; 1225 }; 1226 this.contains(callback, this.traverseNodeBF); 1227 if (parent) { 1228 if (parent.children.length === 0) { 1229 return []; 1230 } else if (parent.children.length > 0) { 1231 var nodeInfo: { itemId: number, itemIcon: Resource, itemTitle: string } = 1232 { itemId: null, itemIcon: null, itemTitle: null } 1233 var childrenNodeInfo: Array<number> = new Array(parent.children.length); 1234 for (let i = 0; i < childrenNodeInfo.length; i++) { 1235 childrenNodeInfo[i] = 0; 1236 } 1237 for (let i = 0; i < parent.children.length && i < childrenNodeInfo.length; i++) { 1238 childrenNodeInfo[i] = parent.children[i].currentNodeId; 1239 } 1240 return childrenNodeInfo; 1241 } 1242 } 1243 return []; 1244 } 1245 1246 getClickNodeChildrenInfo(nodeId: number) { 1247 let parent = null, 1248 callback = function(node): boolean { 1249 if (node.currentNodeId == nodeId) { 1250 parent = node; 1251 return true; 1252 } 1253 return false; 1254 }; 1255 this.contains(callback, this.traverseNodeBF); 1256 if (parent) { 1257 if (parent.children.length === 0) { 1258 return []; 1259 } else if (parent.children.length > 0) { 1260 var nodeInfo: { itemId: number, itemIcon: Resource, itemTitle: string } = 1261 { itemId: null, itemIcon: null, itemTitle: null } 1262 var childrenNodeInfo: Array<{ itemId: number, itemIcon: Resource, itemTitle: string, isFolder: boolean }> = new Array(parent.children.length); 1263 for (let i = 0; i < childrenNodeInfo.length; i++) { 1264 childrenNodeInfo[i] = { itemId: null, itemIcon: null, itemTitle: null, isFolder: null }; 1265 } 1266 for (let i = 0; i < parent.children.length && i < childrenNodeInfo.length; i++) { 1267 childrenNodeInfo[i].itemId = parent.children[i].currentNodeId; 1268 if (parent.children[i].getNodeItem().imageNode) { 1269 childrenNodeInfo[i].itemIcon = parent.children[i].getNodeItem().imageNode.source; 1270 } 1271 if (parent.children[i].getNodeItem().mainTitleNode) { 1272 childrenNodeInfo[i].itemTitle = parent.children[i].getNodeItem().mainTitleNode.title; 1273 } 1274 childrenNodeInfo[i].isFolder = parent.children[i].getIsFolder(); 1275 } 1276 return childrenNodeInfo; 1277 } 1278 } 1279 return []; 1280 } 1281 1282 public checkMainTitleIsValid(title: string) : boolean { 1283 let invalid = /[\\\/:*?"<>|]/; 1284 let invalidLength = /^[\u4e00-\u9fa5]+$/; 1285 if (invalid.test(title)) { 1286 return false; 1287 } 1288 if ((invalidLength.test(title) && title.length > this.MAX_CN_LENGTH) || 1289 (!invalidLength.test(title) && title.length > this.MAX_EN_LENGTH)) { 1290 return false; 1291 } 1292 return true; 1293 } 1294 1295 /* 1296 * DFS: Depth first traversal in drag event. 1297 * @param callback 1298 */ 1299 dragTraverseNodeDF(callback, root: NodeItem = this._root, listNode) { 1300 let stack = [], found = false; 1301 stack.unshift(root); 1302 let currentNode = stack.shift(); 1303 while(!found && currentNode) { 1304 found = callback(currentNode, listNode) === true; 1305 if (!found) { 1306 stack.unshift(...currentNode.children); 1307 currentNode = stack.shift(); 1308 } 1309 } 1310 } 1311 1312 /* 1313 * Add the first dragging node in dragging nodes 1314 * 1.the first dragging node needs to distinguish the position to insert 1315 */ 1316 addDragNode(parentNodeId: number, 1317 currentNodeId: number, 1318 insertCurrentNodeId: number, 1319 isAfter: boolean, 1320 data: { isFolder?: boolean, 1321 icon?: Resource, 1322 selectedIcon?: Resource, 1323 editIcon?: Resource, 1324 primaryTitle?: string, 1325 container?: () => any, 1326 objectCount?: number }): ListNodeUtils { 1327 1328 if (this._root === null) { 1329 this._root = new NodeItem({}); 1330 this._root.nodeLevel = this.INITIAL_INVALID_VALUE; 1331 this._root.parentNodeId = this.INITIAL_INVALID_VALUE; 1332 this._root.currentNodeId = this.INITIAL_INVALID_VALUE; 1333 } 1334 1335 let parent = null, 1336 callback = function(node): boolean { 1337 if (node.currentNodeId == parentNodeId ) { 1338 parent = node; 1339 return true; 1340 } 1341 return false; 1342 }; 1343 this.contains(callback, this.traverseNodeBF); 1344 if (parent) { 1345 let currentNode: NodeItem = new NodeItem(data); 1346 if (parent.nodeLevel > this.MaxNodeLevel) { 1347 throw new Error('ListNodeUtils[addNode]: The level of the tree view cannot exceed 50.'); 1348 } 1349 currentNode.nodeLevel = parent.nodeLevel + 1; 1350 currentNode.parentNodeId = parentNodeId; 1351 currentNode.currentNodeId = currentNodeId; 1352 let insertIndex: number = this.INITIAL_INVALID_VALUE; 1353 if (parent.children.length) { 1354 for (let i = 0; i < parent.children.length; i++) { 1355 if ( parent.children[i].getCurrentNodeId() == insertCurrentNodeId) { 1356 insertIndex = i; 1357 break; 1358 } 1359 } 1360 if (isAfter) { 1361 parent.children.splice(insertIndex + 1, 0, currentNode); 1362 } else { 1363 parent.children.splice(insertIndex, 0, currentNode); 1364 } 1365 } else { 1366 parent.children.push(currentNode); 1367 } 1368 parent.getChildNodeInfo().isHasChildNode = true; 1369 parent.getChildNodeInfo().childNum = parent.children.length; 1370 parent.getChildNodeInfo().allChildNum += 1; 1371 parent.addImageCollapse(parent.getChildNodeInfo().isHasChildNode); 1372 this.updateParentChildNum(parent, true, 1); 1373 return this; 1374 } else { 1375 throw new Error('ListNodeUtils[addNode]: Parent node not found.'); 1376 } 1377 } 1378 } 1379 1380 export class ListNodeDataSource extends BasicDataSource { 1381 readonly ROOT_NODE_ID = -1; 1382 public listNodeUtils: ListNodeUtils = new ListNodeUtils(); 1383 listNode: NodeInfo[] = []; 1384 private readonly INITIAL_INVALID_VALUE = -1; 1385 public lastIndex: number = -1; // record the last focused node. 1386 thisIndex: number = -1; // records clicked nodes in the current period. 1387 private modifyNodeIndex: number = -1; // records the nodes edited in the current period. 1388 modifyNodeId: number = -1 1389 private currentOperation: MenuOperation; 1390 private expandAndCollapseInfo = new Map(); 1391 loadedNodeIdAndIndexMap = new Map(); // [currentNodeId, index] 1392 private isTouchDown: boolean = false; 1393 private appEventBus: TreeListener = TreeListenerManager.getInstance().getTreeListener(); 1394 1395 /* parameter of the drag event. */ 1396 private isInnerDrag: boolean = false; // Judge whether it is an internal drag event. 1397 private isDrag: boolean = false; // It is used to handle events(For example, prevent press events) during global drag. 1398 private draggingCurrentNodeId: number = this.INITIAL_INVALID_VALUE; // Record the current ID of the dragged node. 1399 private draggingParentNodeId: number = this.INITIAL_INVALID_VALUE; // Record the parent ID of the dragged node. 1400 private currentNodeInfo: NodeInfo = null; // To solve the problem of currentIndex missed in onDrop event. 1401 private listItemOpacity : number = 1; // It is used to set the opacity of the node when dragged. 1402 private lastPassIndex: number = this.INITIAL_INVALID_VALUE; // record the last passing node index in drag. 1403 private lastPassId: number = this.INITIAL_INVALID_VALUE; // record the last passing node Id in drag. 1404 private thisPassIndex: number = this.INITIAL_INVALID_VALUE; // record the current passing node in drag. 1405 private lastDelayExpandIndex: number = this.INITIAL_INVALID_VALUE; // record last passing node in delay expand event. 1406 private timeoutExpandId: number = this.INITIAL_INVALID_VALUE; 1407 private lastTimeoutExpandId: number = this.INITIAL_INVALID_VALUE; 1408 private clearTimeoutExpandId: number = this.INITIAL_INVALID_VALUE; 1409 private timeoutHighLightId: number = this.INITIAL_INVALID_VALUE; 1410 private lastTimeoutHighLightId: number = this.INITIAL_INVALID_VALUE; 1411 private clearTimeoutHighLightId: number = this.INITIAL_INVALID_VALUE; 1412 private lastDelayHighLightIndex: number = this.INITIAL_INVALID_VALUE; // record last passing node in HighLight event. 1413 private lastDelayHighLightId: number = this.INITIAL_INVALID_VALUE; //record last passing node Id in HighLight event. 1414 private nodeIdAndSubtitleMap = new Map(); // [currentNodeId, subtitle] 1415 private flag: Flag = Flag.NONE; 1416 private selectedParentNodeId: number = this.INITIAL_INVALID_VALUE; 1417 private selectedParentNodeSubtitle: any = ''; 1418 private insertNodeSubtitle: any = ''; 1419 private currentFocusNodeId: number = this.INITIAL_INVALID_VALUE; 1420 private lastFocusNodeId: number = this.INITIAL_INVALID_VALUE; 1421 private addFocusNodeId: number = this.INITIAL_INVALID_VALUE; 1422 1423 readonly FLAG_LINE: { flagLineHeight: string, 1424 flagLineColor: Resource, 1425 xOffset: string, 1426 yTopOffset: string, 1427 yBottomOffset: string, 1428 yBasePlateOffset: string } = { 1429 flagLineHeight: FLAG_LINE_HEIGHT, 1430 flagLineColor: $r('sys.color.ohos_id_color_emphasize'), 1431 xOffset: X_OFF_SET, 1432 yTopOffset: Y_OFF_SET, 1433 yBottomOffset: Y_BOTTOM_OFF_SET, 1434 yBasePlateOffset: Y_BASE_PLATE_OFF_SET 1435 } 1436 1437 private readonly DRAG_POPUP: { floorConstraintSize: { minWidth: string, maxWidth: string }, 1438 textConstraintSize: { minWidth1: string, maxWidth1: string, 1439 minWidth2: string, maxWidth2: string }, 1440 padding: { left: string, right: string }, 1441 backgroundColor: ResourceColor, 1442 height: string, 1443 shadow: { radius: Resource, color: ResourceColor, offsetX?: number, offsetY?: number }, 1444 borderRadius : Resource, 1445 fontColor: Resource, 1446 fontSize: Resource, 1447 fontWeight: FontWeight 1448 imageOpacity: Resource } = { 1449 floorConstraintSize: { minWidth: FLOOR_MIN_WIDTH, maxWidth: FLOOR_MAX_WIDTH }, 1450 textConstraintSize: { minWidth1: TEXT_MIN_WIDTH, maxWidth1: TEXT_MAX_WIDTH, minWidth2: MIN_WIDTH, maxWidth2: MAX_WIDTH }, 1451 padding: { left: LEFT_PADDING, right: RIGHT_PADDING }, 1452 backgroundColor: COLOR_IMAGE_EDIT, 1453 height: GRAG_POP_UP_HEIGHT, 1454 shadow: { radius: $r('sys.float.ohos_id_corner_radius_default_m'), color: SHADOW_COLOR, offsetX: 0, offsetY: SHADOW_OFFSETY }, 1455 borderRadius: $r('sys.float.ohos_id_corner_radius_clicked'), 1456 fontColor: $r('sys.color.ohos_id_color_primary'), 1457 fontSize: $r('sys.float.ohos_id_text_size_body1'), 1458 fontWeight: FontWeight.Regular, 1459 imageOpacity: $r('sys.float.ohos_id_alpha_content_fourth') 1460 } 1461 1462 private readonly subTitle: { normalFontColor: Resource, 1463 highLightFontColor: Resource, 1464 fontSize: Resource, 1465 fontWeight: FontWeight, 1466 margin: { left: string, right: string } } = { 1467 normalFontColor: $r('sys.color.ohos_id_color_secondary'), 1468 highLightFontColor: $r('sys.color.ohos_id_color_primary_contrary'), 1469 fontSize: $r('sys.float.ohos_id_text_size_body2'), 1470 fontWeight: FontWeight.Regular, 1471 margin: { left: '4vp', right: '24' } 1472 } 1473 1474 public changeNodeColor(index: number, color: Resource | string): void { 1475 this.listNode[index].setNodeColor(color); 1476 this.listNode[index].setNodeBorder(false) 1477 } 1478 1479 private getNodeColor(index) { 1480 return this.listNode[index].getNodeColor(); 1481 } 1482 1483 private handleFocusEffect(index: number, isClearFocusStatus: boolean) { 1484 if (this.listNode[index].getNodeIsShow()) { 1485 this.listNode[index].setNodeBorder(isClearFocusStatus); 1486 } 1487 } 1488 1489 public setImageSource(index: number, interactionStatus: InteractionStatus) { 1490 let nodeInfo: NodeInfo = this.listNode[index]; 1491 nodeInfo.setIsSelected(interactionStatus === InteractionStatus.Selected || 1492 interactionStatus === InteractionStatus.Edit || interactionStatus === InteractionStatus.FinishEdit); 1493 if (nodeInfo.getNodeItem().mainTitleNode != null && interactionStatus != InteractionStatus.DragInsert && 1494 interactionStatus != InteractionStatus.FinishDragInsert) { 1495 nodeInfo.getNodeItem().mainTitleNode.setMainTitleSelected(interactionStatus === InteractionStatus.Selected || 1496 interactionStatus === InteractionStatus.FinishEdit); 1497 } 1498 if (nodeInfo.getNodeItem().imageNode != null) { 1499 nodeInfo.getNodeItem().imageNode.setImageSource(interactionStatus); 1500 } 1501 } 1502 1503 private setImageCollapseSource(index: number, interactionStatus: InteractionStatus) { 1504 let nodeInfo: NodeInfo = this.listNode[index]; 1505 if (nodeInfo.getNodeItem().imageCollapse != null) { 1506 nodeInfo.getNodeItem().imageCollapse.setImageCollapseSource(interactionStatus, 1507 this.expandAndCollapseInfo.get(nodeInfo.getCurrentNodeId())); 1508 } 1509 } 1510 1511 public clearLastIndexStatus() { 1512 if (this.lastIndex == -1 || this.lastIndex >= this.listNode.length) { 1513 return; 1514 } 1515 this.setImageSource(this.lastIndex, InteractionStatus.Normal); 1516 this.changeNodeColor(this.lastIndex, this.listNode[this.lastIndex].getNodeStatus().normal); 1517 this.handleFocusEffect(this.lastIndex, false); 1518 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(this.listNode[this.lastIndex].getCurrentNodeId())); 1519 } 1520 1521 private changeNodeStatus(clickIndex: number): void { 1522 let thisIndex: number = clickIndex; 1523 let tmp: NodeInfo[] = this.ListNode; 1524 let nodeId = tmp[clickIndex].getCurrentNodeId(); 1525 if (this.expandAndCollapseInfo.get(nodeId) == NodeStatus.Expand) { 1526 this.expandAndCollapseInfo.set(nodeId, NodeStatus.Collapse); 1527 tmp[thisIndex].getNodeItem().imageCollapse.changeImageCollapseSource(NodeStatus.Collapse); 1528 } else if (this.expandAndCollapseInfo.get(nodeId) == NodeStatus.Collapse) { 1529 this.expandAndCollapseInfo.set(nodeId, NodeStatus.Expand); 1530 tmp[thisIndex].getNodeItem().imageCollapse.changeImageCollapseSource(NodeStatus.Expand); 1531 } 1532 } 1533 1534 private handleExpandAndCollapse(clickIndex: number) { 1535 let thisIndex: number = clickIndex; 1536 let tmp: NodeInfo[] = this.ListNode; 1537 let nodeId = tmp[thisIndex].getCurrentNodeId(); 1538 if (!this.expandAndCollapseInfo.has(nodeId)) { 1539 return; 1540 } 1541 1542 let rootNodeStatus: NodeStatus = this.expandAndCollapseInfo.get(nodeId); 1543 if (tmp[thisIndex].getChildNodeInfo().isHasChildNode && rootNodeStatus == NodeStatus.Collapse) { 1544 for(var i = 0; i < tmp[thisIndex].getChildNodeInfo().allChildNum; i++) { 1545 tmp[thisIndex + 1 + i].setNodeIsShow(false); 1546 tmp[thisIndex + 1 + i].setListItemHeight(LIST_ITEM_HEIGHT_NONE); 1547 } 1548 this.notifyDataReload(); 1549 return; 1550 } 1551 1552 let childNum: number[] = new Array(tmp[thisIndex].getChildNodeInfo().childNum); 1553 childNum[0] = thisIndex + 1; 1554 let index = 1; 1555 while(index < tmp[thisIndex].getChildNodeInfo().childNum) { 1556 childNum[index] = childNum[index -1] + tmp[childNum[index - 1]].getChildNodeInfo().allChildNum + 1; 1557 index++; 1558 } 1559 if (rootNodeStatus == NodeStatus.Expand) { 1560 for(var i = 0; i < childNum.length; i++) { 1561 tmp[childNum[i]].setNodeIsShow(true); 1562 tmp[childNum[i]].setListItemHeight(LIST_ITEM_HEIGHT); 1563 let nodeId = tmp[childNum[i]].getCurrentNodeId(); 1564 if(this.expandAndCollapseInfo.get(nodeId) == NodeStatus.Expand) { 1565 this.handleExpandAndCollapse(childNum[i]); 1566 } 1567 } 1568 } 1569 childNum = null; 1570 this.notifyDataReload(); 1571 } 1572 1573 public init(listNodeUtils: ListNodeUtils) { 1574 let index = 0; 1575 this.listNode = []; 1576 this.listNodeUtils = listNodeUtils; 1577 this.loadedNodeIdAndIndexMap.clear(); 1578 this.listNodeUtils.traverseNodeDF((node: NodeItem): boolean => { 1579 if (node.currentNodeId >= 0) { 1580 var nodeInfo: NodeInfo = new NodeInfo(node); 1581 this.listNode.push(nodeInfo); 1582 if (nodeInfo.getChildNodeInfo().isHasChildNode) { 1583 this.expandAndCollapseInfo.set(nodeInfo.getCurrentNodeId(), NodeStatus.Collapse); 1584 } 1585 if (nodeInfo.getNodeIsShow()) { 1586 this.loadedNodeIdAndIndexMap.set(nodeInfo.getCurrentNodeId(), index++); 1587 } 1588 if (nodeInfo.getIsFolder()) { 1589 this.nodeIdAndSubtitleMap.set(nodeInfo.getCurrentNodeId(), 1590 nodeInfo.getNodeInfoData().secondaryTitle || nodeInfo.getNodeInfoData().secondaryTitle == 0 ? 1591 nodeInfo.getNodeInfoData().secondaryTitle : ''); 1592 } 1593 } 1594 return false; 1595 }); 1596 } 1597 1598 private refreshRemoveNodeData(removeNodeIdList: number[], parentNodeInfo: NodeInfo) { 1599 let deleteIndexList: number[] = []; 1600 for (let i = 0; i < removeNodeIdList.length; i++) { 1601 for (let j = 0; j < this.listNode.length; j++) { 1602 if (this.listNode[j].getNodeCurrentNodeId() == removeNodeIdList[i]) { 1603 let currentNodeId = this.listNode[j].getNodeCurrentNodeId(); 1604 if (this.loadedNodeIdAndIndexMap.has(currentNodeId)) { 1605 // this.listNode index to lazyForEach index. 1606 deleteIndexList.push(this.loadedNodeIdAndIndexMap.get(currentNodeId)); 1607 } 1608 let deleteNode = this.listNode.splice(j, 1); 1609 deleteNode = null; // free memory 1610 if (this.expandAndCollapseInfo.has(removeNodeIdList[i])) { 1611 this.expandAndCollapseInfo.delete(removeNodeIdList[i]); // delete deleteNode expandAndCollapseInfo. 1612 } 1613 break; 1614 } 1615 } 1616 } 1617 deleteIndexList.forEach((value)=>{ 1618 this.notifyDataDelete(value); // notifyDataDelete do not update data. 1619 this.notifyDataChange(value); // call notifyDataChange to update data. 1620 }) 1621 let index: number = 0; 1622 for (let i = 0; i < this.listNode.length; i++) { 1623 if (this.listNode[i].getNodeCurrentNodeId() == parentNodeInfo.getNodeCurrentNodeId()) { 1624 if (parentNodeInfo.getNodeItem().imageCollapse == null) { 1625 this.listNode[i].handleImageCollapseAfterAddNode(false); 1626 // delete deleteNode parentNode expandAndCollapseInfo. 1627 this.expandAndCollapseInfo.delete(parentNodeInfo.getNodeCurrentNodeId()); 1628 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(this.listNode[i].getNodeCurrentNodeId())); 1629 } 1630 break; 1631 } 1632 } 1633 let callbackParam: CallbackParam = {currentNodeId: parentNodeInfo.getNodeCurrentNodeId(), parentNodeId: parentNodeInfo.getNodeParentNodeId()}; 1634 this.appEventBus.emit(TreeListenType.NODE_DELETE, [callbackParam]); 1635 } 1636 1637 private refreshAddNodeData(addNodeIdList: number[]) { 1638 var addNodeInfo: NodeInfo; 1639 this.listNodeUtils.traverseNodeDF((node: NodeItem): boolean => { 1640 if (node.currentNodeId === addNodeIdList[0]) { 1641 addNodeInfo = new NodeInfo(node); 1642 return true; 1643 } 1644 return false; 1645 }); 1646 addNodeInfo.setIsModify(true); 1647 1648 let index: number = 0; 1649 for (let i = 0; i < this.listNode.length; i++) { 1650 if (this.listNode[i].getNodeCurrentNodeId() == addNodeInfo.getNodeParentNodeId()) { 1651 index = i; 1652 if (this.listNode[i].getNodeItem().imageCollapse == null) { 1653 this.listNode[i].handleImageCollapseAfterAddNode(true); 1654 this.notifyDataChange(index); 1655 } else if (this.expandAndCollapseInfo.get(this.listNode[i].getNodeCurrentNodeId()) == NodeStatus.Collapse) { 1656 this.changeNodeStatus(index); 1657 } 1658 this.listNode.splice(i + 1, 0, addNodeInfo); 1659 this.listNode[i + 1].setTitleAndInputTextStatus(true); // false->true: realize inner Interaction. 1660 this.listNode[i + 1].setNodeIsShow(true); 1661 this.listNode[i + 1].setListItemHeight(LIST_ITEM_HEIGHT); 1662 this.setImageSource(i + 1, InteractionStatus.Edit); // Normal->Edit : realize inner Interaction. 1663 this.currentOperation = MenuOperation.ADD_NODE; 1664 this.notifyDataAdd(i + 1); 1665 this.notificationNodeInfo(i+1, this.currentOperation); 1666 break; 1667 } 1668 } 1669 this.modifyNodeIndex = index + 1; 1670 this.expandAndCollapseInfo.set(addNodeInfo.getNodeParentNodeId(), NodeStatus.Expand); 1671 this.handleExpandAndCollapse(index); 1672 } 1673 1674 public refreshData(listNodeUtils: ListNodeUtils, operation: MenuOperation, 1675 parentNodeId: number, changeNodeIdList: number[]) { 1676 let parentNodeInfo: NodeInfo; 1677 this.listNodeUtils = listNodeUtils; 1678 this.listNodeUtils.traverseNodeDF((node: NodeItem): boolean => { 1679 if (node.currentNodeId == parentNodeId) { 1680 parentNodeInfo = new NodeInfo(node); 1681 return true; 1682 } 1683 return false; 1684 }); 1685 1686 if (operation === MenuOperation.REMOVE_NODE) { 1687 this.nodeIdAndSubtitleMap.set(parentNodeId, this.selectedParentNodeSubtitle); 1688 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(parentNodeId)); 1689 this.refreshRemoveNodeData(changeNodeIdList, parentNodeInfo); 1690 } 1691 1692 if (operation === MenuOperation.ADD_NODE) { 1693 this.addFocusNodeId = changeNodeIdList[0]; 1694 this.nodeIdAndSubtitleMap.set(this.getClickNodeId(), this.selectedParentNodeSubtitle); 1695 this.nodeIdAndSubtitleMap.set(changeNodeIdList[0], this.insertNodeSubtitle); 1696 this.refreshAddNodeData(changeNodeIdList); 1697 } 1698 } 1699 1700 public setClickIndex(index: number) { 1701 this.thisIndex = index; 1702 } 1703 1704 public getClickNodeId(): number { 1705 if (this.thisIndex < 0 || this.thisIndex >= this.ListNode.length) { 1706 return -1; 1707 } 1708 return this.ListNode[this.thisIndex].getCurrentNodeId(); 1709 } 1710 1711 public expandAndCollapseNode(clickIndex: number) { 1712 this.changeNodeStatus(clickIndex); 1713 this.handleExpandAndCollapse(clickIndex) 1714 } 1715 1716 public getIsTouchDown(): boolean { 1717 return this.isTouchDown; 1718 } 1719 1720 public getLastIndex(): number { 1721 return this.lastIndex; 1722 } 1723 1724 public handleEventDrag(index: number) { 1725 this.setImageSource(index, InteractionStatus.Normal); 1726 this.changeNodeColor(index, this.listNode[index].getNodeStatus().normal); 1727 this.handleFocusEffect(index, false); 1728 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(this.listNode[index].getCurrentNodeId())); 1729 } 1730 1731 public handleEvent(event: Event, index: number) { 1732 /* Return while the event is dragging event. */ 1733 if (this.isDrag) { 1734 return; 1735 } 1736 1737 if (event === Event.TOUCH_DOWN || event === Event.TOUCH_UP || event === Event.MOUSE_BUTTON_RIGHT) { 1738 if (index != this.lastIndex) { 1739 this.clearLastIndexStatus(); 1740 } 1741 } 1742 1743 let lazyForEachIndex = this.loadedNodeIdAndIndexMap.get(this.listNode[index].getCurrentNodeId()); 1744 switch(event) { 1745 case Event.TOUCH_DOWN: 1746 this.isTouchDown = true; 1747 this.changeNodeColor(index, this.listNode[index].getNodeStatus().press); 1748 this.notifyDataChange(lazyForEachIndex); 1749 break; 1750 case Event.TOUCH_UP: { 1751 if (this.isInnerDrag) { 1752 this.isInnerDrag = false; 1753 } 1754 this.isTouchDown = false; 1755 let nodeInfo: NodeInfo = this.listNode[index]; 1756 this.setImageSource(index, InteractionStatus.Selected); 1757 nodeInfo.setFontColor($r('sys.color.ohos_id_color_primary')) 1758 this.lastIndex = index; 1759 this.changeNodeColor(index, nodeInfo.getNodeStatus().selected); 1760 this.notifyDataChange(lazyForEachIndex); 1761 break; 1762 } 1763 case Event.HOVER: 1764 if (this.getNodeColor(index) != this.listNode[index].getNodeStatus().selected) { 1765 this.changeNodeColor(index, this.listNode[index].getNodeStatus().hover); 1766 this.notifyDataChange(lazyForEachIndex); 1767 } 1768 break; 1769 case Event.HOVER_OVER: 1770 if (this.getNodeColor(index) != this.listNode[index].getNodeStatus().selected) { 1771 this.changeNodeColor(index, this.listNode[index].getNodeStatus().normal); 1772 this.notifyDataChange(lazyForEachIndex); 1773 } 1774 break; 1775 case Event.FOCUS: 1776 this.handleFocusEffect(index, true); 1777 this.notifyDataChange(lazyForEachIndex); 1778 break; 1779 case Event.BLUR: 1780 this.handleFocusEffect(index, false); 1781 this.notifyDataChange(lazyForEachIndex); 1782 break; 1783 case Event.MOUSE_BUTTON_RIGHT: 1784 this.lastIndex = index; 1785 this.finishEditing(); 1786 break; 1787 case Event.DRAG: 1788 this.isTouchDown = false; 1789 let nodeInfo: NodeInfo = this.listNode[index]; 1790 this.setImageSource(index, InteractionStatus.Selected); 1791 this.lastIndex = index; 1792 this.changeNodeColor(index, nodeInfo.getNodeStatus().selected); 1793 this.notifyDataChange(lazyForEachIndex); 1794 break; 1795 default: 1796 break; 1797 } 1798 } 1799 1800 private notificationNodeInfo(addNodeId: number, operation: MenuOperation) { 1801 if (operation === MenuOperation.MODIFY_NODE) { 1802 let modifyNodeInfo: NodeInfo = this.listNode[this.modifyNodeIndex]; 1803 let backParamModify: CallbackParam = { currentNodeId: modifyNodeInfo.getNodeCurrentNodeId(), 1804 parentNodeId: modifyNodeInfo.getNodeParentNodeId() } 1805 this.appEventBus.emit(TreeListenType.NODE_MODIFY, 1806 [backParamModify]); 1807 } else if (operation === MenuOperation.ADD_NODE) { 1808 let addNodeInfo: NodeInfo = this.listNode[addNodeId]; 1809 let icon: Resource | string = (addNodeInfo.getNodeItem().imageNode != null) ? 1810 addNodeInfo.getNodeItem().imageNode.source : null; 1811 let selectedIcon: Resource | string = (addNodeInfo.getNodeItem().imageNode != null) ? 1812 addNodeInfo.getNodeItem().imageNode.selectedSource : null; 1813 let editIcon: Resource | string = (addNodeInfo.getNodeItem().imageNode != null) ? 1814 addNodeInfo.getNodeItem().imageNode.editSource : null; 1815 let callbackParam: CallbackParam = { currentNodeId: addNodeInfo.getNodeCurrentNodeId(), 1816 parentNodeId: addNodeInfo.getNodeParentNodeId() } 1817 this.appEventBus.emit(TreeListenType.NODE_ADD, 1818 [callbackParam]); 1819 } 1820 } 1821 1822 public finishEditing() { 1823 if (this.modifyNodeIndex!= -1) { 1824 this.setImageSource(this.modifyNodeIndex, InteractionStatus.FinishEdit); 1825 this.setImageCollapseSource(this.modifyNodeIndex, InteractionStatus.FinishEdit); 1826 this.listNode[this.modifyNodeIndex].setIsModify(false); 1827 this.listNode[this.modifyNodeIndex].setTitleAndInputTextStatus(false); 1828 this.notificationNodeInfo(this.modifyNodeIndex, this.currentOperation); 1829 this.notifyDataChange(this.modifyNodeIndex); 1830 } 1831 } 1832 1833 public setItemVisibilityOnEdit(nodeId: number, operation: MenuOperation) { 1834 let index: number = -1; 1835 if (nodeId == -1) { 1836 return; 1837 } 1838 if (operation === MenuOperation.MODIFY_NODE) { 1839 for (let i = 0; i < this.listNode.length; i++) { // nodeId to find index 1840 if (this.listNode[i].getCurrentNodeId() == nodeId) { 1841 index = i; 1842 break; 1843 } 1844 } 1845 let nodeInfo: NodeInfo = this.listNode[index]; 1846 nodeInfo.setIsModify(true); 1847 if (nodeInfo.getNodeItem().mainTitleNode === null) { 1848 return; // no title 1849 } 1850 1851 this.currentOperation = MenuOperation.MODIFY_NODE; 1852 nodeInfo.setTitleAndInputTextStatus(true); 1853 this.setImageSource(index, InteractionStatus.Edit); 1854 this.setImageCollapseSource(index, InteractionStatus.Edit); 1855 this.modifyNodeIndex = index; 1856 if (nodeInfo.getNodeItem().inputText) { 1857 if (nodeInfo.getNodeItem().imageCollapse != null) { 1858 nodeInfo.getNodeItem().inputText.rightMargin = 1859 $r('sys.float.ohos_id_text_paragraph_margin_xs') 1860 } else { 1861 nodeInfo.getNodeItem().inputText.rightMargin = 1862 $r('sys.float.ohos_id_elements_margin_horizontal_m') 1863 } 1864 } 1865 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(nodeId)); 1866 } 1867 index = nodeId; 1868 if (operation === MenuOperation.COMMIT_NODE) { 1869 let nodeInfo: NodeInfo = this.listNode[index]; 1870 nodeInfo.setTitleAndInputTextStatus(false); 1871 nodeInfo.setIsModify(false); 1872 this.setImageSource(index, InteractionStatus.FinishEdit); 1873 this.setImageCollapseSource(index, InteractionStatus.FinishEdit); 1874 this.notificationNodeInfo(this.modifyNodeIndex, this.currentOperation); 1875 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(nodeInfo.getCurrentNodeId())); 1876 } 1877 } 1878 1879 public setPopUpInfo(popUpType: PopUpType, inputError: InputError, isShow: boolean, index: number) { 1880 let nodeInfo: NodeInfo = this.listNode[index]; 1881 nodeInfo.setPopUpIsShow(isShow); 1882 // this.listNode index to lazyForEach index. 1883 let lazyForEachIndex = this.loadedNodeIdAndIndexMap.get(nodeInfo.getCurrentNodeId()); 1884 if (!isShow) { 1885 this.notifyDataChange(lazyForEachIndex); 1886 return; 1887 } 1888 if (popUpType === PopUpType.HINTS) { 1889 if (nodeInfo.getNodeItem().mainTitleNode != null) { 1890 nodeInfo.setPopUpText(nodeInfo.getNodeItem().mainTitleNode.title); 1891 } else { 1892 nodeInfo.setPopUpText(''); 1893 nodeInfo.setPopUpIsShow(false); 1894 } 1895 nodeInfo.setPopUpEnableArrow(false); 1896 nodeInfo.setPopUpColor($r('sys.color.ohos_id_color_background')); 1897 nodeInfo.setPopUpTextColor($r('sys.color.ohos_id_color_text_secondary')); 1898 } else if (popUpType === PopUpType.WARNINGS) { 1899 if (nodeInfo.getNodeItem().inputText != null) { 1900 if (inputError === InputError.INVALID_ERROR) { 1901 nodeInfo.setPopUpText("invalid error"); 1902 } else if (inputError === InputError.LENGTH_ERROR) { 1903 nodeInfo.setPopUpText("length error"); 1904 } 1905 nodeInfo.setPopUpEnableArrow(true); 1906 nodeInfo.setPopUpColor($r('sys.color.ohos_id_color_help_tip_bg')); 1907 nodeInfo.setPopUpTextColor($r('sys.color.ohos_id_color_text_hint_contrary')); 1908 } 1909 } 1910 this.notifyDataChange(lazyForEachIndex); 1911 } 1912 1913 public setShowPopUpTimeout(timeout: number, index: number) { 1914 if (this.listNode[index].getNodeItem().mainTitleNode != null) { 1915 this.listNode[index].getNodeItem().mainTitleNode.popUpTimeout = timeout; 1916 } 1917 let lazyForEachIndex = this.loadedNodeIdAndIndexMap.get(this.listNode[index].getCurrentNodeId()); 1918 this.notifyDataChange(lazyForEachIndex); 1919 } 1920 1921 public setMainTitleNameOnEdit(index: number, text: string) { 1922 1923 this.modifyNodeIndex = index; 1924 if (this.listNode[index].getNodeItem().mainTitleNode != null) { 1925 this.listNode[index].getNodeItem().mainTitleNode.title = text; 1926 } 1927 } 1928 1929 public get ListNode(): NodeInfo[] { 1930 return this.listNode; 1931 } 1932 1933 public totalCount(): number { 1934 let count: number = 0; 1935 let index: number = 0; 1936 this.loadedNodeIdAndIndexMap.clear(); 1937 for (let i =0 ; i < this.listNode.length; i++) { 1938 if (this.listNode[i].getNodeIsShow()) { 1939 this.loadedNodeIdAndIndexMap.set(this.listNode[i].getCurrentNodeId(), index++); 1940 count++; 1941 } 1942 } 1943 return count; 1944 } 1945 1946 public getData(index: number): any { 1947 let count = 0; 1948 for (let i = 0; i < this.listNode.length; i++) { 1949 if (this.listNode[i].getNodeIsShow()) { 1950 if (index == count) { 1951 return this.listNode[i]; 1952 } 1953 count++; 1954 } 1955 } 1956 return null; 1957 } 1958 1959 public addData(index: number, data: NodeInfo): void { 1960 this.listNode.splice(index, 0, data) 1961 this.notifyDataAdd(index) 1962 } 1963 1964 public pushData(data: NodeInfo): void { 1965 this.listNode.push(data) 1966 this.notifyDataAdd(this.listNode.length - 1) 1967 } 1968 1969 public setIsInnerDrag(isInnerDrag: boolean) { 1970 this.isInnerDrag = isInnerDrag; 1971 } 1972 1973 public getIsInnerDrag(): boolean { 1974 return this.isInnerDrag; 1975 } 1976 1977 public setIsDrag(isDrag: boolean) { 1978 this.isDrag = isDrag; 1979 } 1980 1981 public getIsDrag(): boolean { 1982 return this.isDrag; 1983 } 1984 1985 public setCurrentNodeInfo(currentNodeInfo: NodeInfo) { 1986 this.currentNodeInfo = currentNodeInfo; 1987 } 1988 1989 public getCurrentNodeInfo() { 1990 return this.currentNodeInfo; 1991 } 1992 1993 public setDraggingParentNodeId(draggingParentNodeId: number) { 1994 this.draggingParentNodeId = draggingParentNodeId; 1995 } 1996 1997 public getDraggingParentNodeId() { 1998 return this.draggingParentNodeId; 1999 } 2000 2001 public getDraggingCurrentNodeId() { 2002 return this.draggingCurrentNodeId; 2003 } 2004 2005 public setDraggingCurrentNodeId(draggingCurrentNodeId: number) { 2006 this.draggingCurrentNodeId = draggingCurrentNodeId; 2007 } 2008 2009 public setListItemOpacity(listItemOpacity: number) { 2010 this.listItemOpacity = listItemOpacity; 2011 } 2012 2013 public getListItemOpacity(item: NodeInfo) { 2014 return item.getCurrentNodeId() == this.getDraggingCurrentNodeId() ? this.listItemOpacity : 1; 2015 } 2016 2017 public getDragPopupPara() { 2018 return this.DRAG_POPUP; 2019 } 2020 2021 public setLastPassIndex(lastPassIndex: number) { 2022 this.lastPassIndex = lastPassIndex; 2023 } 2024 2025 public getLastPassIndex(): number { 2026 return this.lastPassIndex; 2027 } 2028 2029 public getIsParentOfInsertNode(insertNodeId: number): boolean { 2030 let selectedNodeItem: NodeItem = this.currentNodeInfo.getNodeInfoNode(); 2031 let isParentNodeOfInsertNode = false, 2032 callback = function(node): boolean { 2033 if (node.currentNodeId == insertNodeId ) { 2034 isParentNodeOfInsertNode = true; 2035 return true; 2036 } 2037 return false; 2038 }; 2039 this.listNodeUtils.traverseNodeDF(callback, selectedNodeItem); 2040 return isParentNodeOfInsertNode; 2041 } 2042 2043 public setPassIndex(thisPassIndex: number) { 2044 this.thisPassIndex = thisPassIndex; 2045 } 2046 2047 public getPassIndex(): number { 2048 return this.thisPassIndex; 2049 } 2050 2051 public clearTimeOutAboutDelayHighLightAndExpand(currentIndex: number) { 2052 if (this.lastPassId != this.INITIAL_INVALID_VALUE && this.loadedNodeIdAndIndexMap.has(this.lastPassId)) { 2053 let index: number = this.loadedNodeIdAndIndexMap.get(this.lastPassId); 2054 let that =this; 2055 this.ListNode.forEach(function(value) { 2056 if (value.getNodeCurrentNodeId() == that.lastPassId) { 2057 value.setCanShowFlagLine(false); 2058 } 2059 }) 2060 this.notifyDataChange(index); 2061 } 2062 2063 if ((this.lastTimeoutHighLightId != this.INITIAL_INVALID_VALUE && 2064 this.clearTimeoutHighLightId != this.lastTimeoutHighLightId)) { 2065 clearTimeout(this.lastTimeoutHighLightId); 2066 if (this.lastDelayHighLightIndex != this.INITIAL_INVALID_VALUE) { 2067 this.clearHighLight(this.lastDelayHighLightIndex); 2068 let index: number = this.loadedNodeIdAndIndexMap 2069 .get(this.listNode[this.lastDelayHighLightIndex].getCurrentNodeId()); 2070 this.notifyDataChange(index); 2071 } 2072 this.clearTimeoutHighLightId = this.lastTimeoutHighLightId; 2073 } 2074 this.lastTimeoutHighLightId = this.timeoutHighLightId; 2075 this.lastDelayHighLightIndex = currentIndex; 2076 2077 if ((this.lastTimeoutExpandId != this.INITIAL_INVALID_VALUE && 2078 this.clearTimeoutExpandId != this.lastTimeoutExpandId)) { 2079 clearTimeout(this.lastTimeoutExpandId); 2080 this.clearTimeoutExpandId = this.lastTimeoutExpandId; 2081 } 2082 this.lastTimeoutExpandId = this.timeoutExpandId; 2083 this.lastDelayExpandIndex = this.INITIAL_INVALID_VALUE; 2084 } 2085 2086 public clearHighLight(currentIndex: number) { 2087 this.changeNodeColor(currentIndex, this.listNode[currentIndex].getNodeStatus().normal); 2088 this.changeNodeHighLightColor(currentIndex, false); 2089 this.setImageSource(currentIndex, InteractionStatus.FinishDragInsert); 2090 this.setImageCollapseSource(currentIndex, InteractionStatus.FinishDragInsert); 2091 this.listNode[currentIndex].setIsHighLight(false); 2092 } 2093 2094 private changeNodeHighLightColor(index: number, isHighLight: boolean): void { 2095 if (this.listNode[index].getNodeItem().mainTitleNode && this.listNode[index].getIsShowTitle()) { 2096 this.listNode[index].getNodeItem().mainTitleNode.setMainTitleHighLight(isHighLight); 2097 } 2098 } 2099 2100 public setVisibility(flag: Flag, index: number, isOverBorder: boolean) { 2101 let isChanged: boolean = (this.thisPassIndex != index || this.flag != flag) ? true : false; 2102 this.thisPassIndex = index; 2103 if ((isChanged || isOverBorder) && this.isInnerDrag) { 2104 this.flag = flag; 2105 let currentNodeId: number = this.getData(index).getCurrentNodeId(); 2106 let currentNodeLevel: number = this.expandAndCollapseInfo.get(currentNodeId) == NodeStatus.Expand && 2107 this.flag == Flag.DOWN_FLAG ? this.getData(index).getNodeLevel() + 1 : this.getData(index).getNodeLevel(); 2108 if (this.lastPassId != this.INITIAL_INVALID_VALUE && this.loadedNodeIdAndIndexMap.has(this.lastPassId)) { 2109 let lastIndex: number = this.loadedNodeIdAndIndexMap.get(this.lastPassId); 2110 let that = this; 2111 this.ListNode.forEach(function (value) { 2112 if (value.getNodeCurrentNodeId() == that.lastPassId) { 2113 value.setCanShowFlagLine(false); 2114 } 2115 }) 2116 this.notifyDataChange(lastIndex); 2117 } 2118 if (this.flag == Flag.DOWN_FLAG && index < this.totalCount() - 1) { 2119 this.getData(index).setCanShowFlagLine(false); 2120 this.getData(index+1).setCanShowFlagLine(true); 2121 this.getData(index).setCanShowBottomFlagLine(false); 2122 this.getData(index+1).setFlagLineLeftMargin(currentNodeLevel); 2123 this.notifyDataChange(index); 2124 this.notifyDataChange(index + 1); 2125 this.lastPassId = this.getData(index + 1).getNodeCurrentNodeId(); 2126 } else if (this.flag == Flag.UP_FLAG && index < this.totalCount() - 1) { 2127 this.getData(index).setCanShowFlagLine(true); 2128 this.getData(index+1).setCanShowFlagLine(false); 2129 this.getData(index).setCanShowBottomFlagLine(false); 2130 this.getData(index).setFlagLineLeftMargin(currentNodeLevel); 2131 this.notifyDataChange(index); 2132 this.notifyDataChange(index + 1); 2133 this.lastPassId = this.getData(index).getNodeCurrentNodeId(); 2134 } else if (index >= this.totalCount() - 1) { 2135 if (this.flag == Flag.DOWN_FLAG) { 2136 this.getData(index).setCanShowFlagLine(false); 2137 this.getData(index).setCanShowBottomFlagLine(true); 2138 } else { 2139 this.getData(index).setCanShowFlagLine(true); 2140 this.getData(index).setCanShowBottomFlagLine(false); 2141 } 2142 this.getData(index).setFlagLineLeftMargin(currentNodeLevel); 2143 this.notifyDataChange(index); 2144 this.lastPassId = this.getData(index).getNodeCurrentNodeId(); 2145 } 2146 } 2147 } 2148 2149 public delayHighLightAndExpandNode(currentIndex: number, currentNodeId: number, showIndex: number) { 2150 let isChangIndex: boolean = currentIndex != this.lastDelayExpandIndex ? true : false; 2151 let isOverBorder: boolean = this.getData(showIndex).getIsOverBorder(); 2152 if (isOverBorder) { 2153 this.lastDelayExpandIndex = this.INITIAL_INVALID_VALUE; 2154 } else { 2155 this.lastDelayExpandIndex = currentIndex; 2156 } 2157 if (isOverBorder || isChangIndex) { 2158 let that = this; 2159 2160 /* highLight node time-out. */ 2161 let canDelayHighLight: boolean = !isOverBorder && (!this.isInnerDrag || 2162 (this.expandAndCollapseInfo.get(currentNodeId) == NodeStatus.Collapse && this.isInnerDrag) || 2163 (!this.expandAndCollapseInfo.has(currentNodeId) && this.listNode[currentIndex].getIsFolder())); 2164 if (canDelayHighLight) { 2165 /* set hoverState color before highLight. */ 2166 this.changeNodeColor(currentIndex, this.listNode[currentIndex].getNodeStatus().hover); 2167 this.notifyDataChange(showIndex); 2168 2169 let delayHighLightTime: number = this.isInnerDrag ? 1000 : 0; // ms 2170 this.timeoutHighLightId = setTimeout(function() { 2171 that.delayHighLight(currentIndex); 2172 }, delayHighLightTime) 2173 } 2174 if (isOverBorder || (this.lastTimeoutHighLightId != this.INITIAL_INVALID_VALUE && 2175 this.clearTimeoutHighLightId != this.lastTimeoutHighLightId)) { 2176 clearTimeout(this.lastTimeoutHighLightId); 2177 if (this.lastDelayHighLightIndex != this.INITIAL_INVALID_VALUE) { 2178 this.clearHighLight(this.lastDelayHighLightIndex); 2179 this.notifyDataReload(); 2180 } 2181 this.clearTimeoutHighLightId = this.lastTimeoutHighLightId; 2182 } 2183 this.lastTimeoutHighLightId = this.timeoutHighLightId; 2184 this.lastDelayHighLightIndex = currentIndex; 2185 2186 /* alter flagLine and expand node time-out. */ 2187 if (!isOverBorder && this.expandAndCollapseInfo.get(currentNodeId) == NodeStatus.Collapse) { 2188 let firstChildNodeId: number = this.getData(showIndex).getNodeInfoNode().children[0].currentNodeId; 2189 let delayAlterFlagLineAndExpandNodeTime: number = 2000; // ms 2190 this.timeoutExpandId = setTimeout(function() { 2191 that.clearHighLight(that.lastDelayHighLightIndex); 2192 that.alterFlagLineAndExpandNode(currentIndex, firstChildNodeId); 2193 }, delayAlterFlagLineAndExpandNodeTime) 2194 } 2195 if (isOverBorder || (this.lastTimeoutExpandId != this.INITIAL_INVALID_VALUE && 2196 this.clearTimeoutExpandId != this.lastTimeoutExpandId)) { 2197 clearTimeout(this.lastTimeoutExpandId); 2198 this.clearTimeoutExpandId = this.lastTimeoutExpandId; 2199 } 2200 this.lastTimeoutExpandId = this.timeoutExpandId; 2201 } 2202 } 2203 2204 public delayHighLight(currentIndex: number) { 2205 let that =this; 2206 this.ListNode.forEach(function (value) { 2207 if (value.getNodeCurrentNodeId() == that.lastPassId) { 2208 value.setCanShowFlagLine(false); 2209 value.setCanShowBottomFlagLine(false); 2210 } 2211 }) 2212 this.changeNodeColor(currentIndex, this.listNode[currentIndex].getNodeStatus().highLight); 2213 this.listNode[currentIndex].setIsHighLight(true); 2214 this.changeNodeHighLightColor(currentIndex, true); 2215 this.setImageSource(currentIndex, InteractionStatus.DragInsert); 2216 this.setImageCollapseSource(currentIndex, InteractionStatus.DragInsert); 2217 this.notifyDataReload(); 2218 } 2219 2220 public alterFlagLineAndExpandNode(currentIndex: number, firstChildNodeId: number) { 2221 let that =this; 2222 this.ListNode.forEach(function (value) { 2223 if (value.getNodeCurrentNodeId() == that.lastPassId) { 2224 value.setCanShowFlagLine(false); 2225 value.setCanShowBottomFlagLine(false); 2226 } 2227 }) 2228 this.ListNode.forEach(function (value) { 2229 if (that.isInnerDrag && value.getNodeCurrentNodeId() == firstChildNodeId) { 2230 value.setCanShowFlagLine(true); 2231 } 2232 }) 2233 this.changeNodeStatus(currentIndex); 2234 this.handleExpandAndCollapse(currentIndex); 2235 this.lastPassId = firstChildNodeId; 2236 } 2237 2238 public hideLastLine() { 2239 if (this.lastPassId != this.INITIAL_INVALID_VALUE && this.loadedNodeIdAndIndexMap.has(this.lastPassId)) { 2240 let that = this; 2241 this.ListNode.forEach(function (value) { 2242 if (value.getNodeCurrentNodeId() == that.lastPassId) { 2243 value.setCanShowFlagLine(false); 2244 value.setCanShowBottomFlagLine(false); 2245 } 2246 }) 2247 let index: number = this.loadedNodeIdAndIndexMap.get(this.lastPassId); 2248 this.notifyDataChange(index); 2249 } 2250 } 2251 2252 public clearLastTimeoutHighLight() { 2253 if (this.lastTimeoutHighLightId != this.INITIAL_INVALID_VALUE && 2254 this.clearTimeoutHighLightId != this.lastTimeoutHighLightId) { 2255 clearTimeout(this.lastTimeoutHighLightId); 2256 if (this.lastDelayHighLightIndex != this.INITIAL_INVALID_VALUE) { 2257 this.clearHighLight(this.lastDelayHighLightIndex); 2258 } 2259 } 2260 } 2261 2262 public clearLastTimeoutExpand() { 2263 if (this.lastTimeoutExpandId != this.INITIAL_INVALID_VALUE && 2264 this.clearTimeoutExpandId != this.lastTimeoutExpandId) { 2265 clearTimeout(this.lastTimeoutExpandId); 2266 } 2267 } 2268 2269 public getSubtitle(currentNodeId: number): string { 2270 if (this.nodeIdAndSubtitleMap.has(currentNodeId)) { 2271 if (typeof this.nodeIdAndSubtitleMap.get(currentNodeId) == "number") { 2272 return this.nodeIdAndSubtitleMap.get(currentNodeId).toString(); 2273 } else { 2274 return this.nodeIdAndSubtitleMap.get(currentNodeId) 2275 } 2276 } else { 2277 return ''; 2278 } 2279 } 2280 2281 public hasSubtitle(currentNodeId: number) { 2282 return this.nodeIdAndSubtitleMap.has(currentNodeId); 2283 } 2284 2285 public initialParameterAboutDelayHighLightAndExpandIndex() { 2286 this.lastDelayHighLightIndex = this.INITIAL_INVALID_VALUE; 2287 this.lastDelayExpandIndex = this.INITIAL_INVALID_VALUE; 2288 this.lastPassIndex = this.INITIAL_INVALID_VALUE; 2289 this.draggingCurrentNodeId = this.INITIAL_INVALID_VALUE; 2290 this.flag = Flag.NONE; 2291 } 2292 2293 public refreshSubtitle(insertNodeCurrentNodeId: number) { 2294 this.nodeIdAndSubtitleMap.set(this.selectedParentNodeId, this.selectedParentNodeSubtitle); 2295 this.nodeIdAndSubtitleMap.set(insertNodeCurrentNodeId, this.insertNodeSubtitle); 2296 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(this.selectedParentNodeId)); 2297 this.notifyDataChange(this.loadedNodeIdAndIndexMap.get(insertNodeCurrentNodeId)); 2298 } 2299 2300 public setNodeSubtitlePara( 2301 selectedParentNodeId: number, 2302 selectedParentNodeSubtitle: ResourceStr, 2303 insertNodeSubtitle: ResourceStr 2304 ) { 2305 this.selectedParentNodeId = selectedParentNodeId; 2306 this.selectedParentNodeSubtitle = selectedParentNodeSubtitle; 2307 this.insertNodeSubtitle = insertNodeSubtitle; 2308 } 2309 2310 public getInsertNodeSubtitle() { 2311 return this.insertNodeSubtitle; 2312 } 2313 2314 public getExpandAndCollapseInfo(currentNodeId: number) { 2315 return this.expandAndCollapseInfo.get(currentNodeId); 2316 } 2317 2318 public getLastDelayHighLightId() { 2319 return this.lastDelayHighLightId; 2320 } 2321 2322 public setLastDelayHighLightId() { 2323 this.ListNode.forEach((value, index) => { 2324 if (index == this.lastDelayHighLightIndex) { 2325 this.lastDelayHighLightId = value.getCurrentNodeId(); 2326 } 2327 }) 2328 } 2329 2330 public setLastPassId(lastPassId: number) { 2331 this.lastPassId = lastPassId; 2332 } 2333 2334 public setLastDelayHighLightIndex(lastDelayHighLightIndex) { 2335 this.lastDelayHighLightIndex = lastDelayHighLightIndex; 2336 } 2337 2338 /* 2339 * Alter the current node location to a needful position. 2340 * 1.Create an array named 'dragNodeParam' to store dragging node information. 2341 * 2.Delete the dragging node from the tree. 2342 * 3.Add the dragging node to the tree. 2343 */ 2344 public alterDragNode(rearParentNodeId: number, rearCurrentNodeId: number, insertNodeInfo: NodeInfo, 2345 dragParentNodeId: number, dragCurrentNodeId: number, frontNodeInfoItem: NodeInfo) { 2346 let dragNodeParam: { parentId: number, currentId: number, data: any }[] = []; 2347 let parentNodeId: number = rearParentNodeId; 2348 let currentNodeId: number = dragCurrentNodeId; 2349 let nodeParam = frontNodeInfoItem.getNodeInfoData(); 2350 let nodeInfo: NodeInfo = null; 2351 let nodeInfoNode: NodeItem = frontNodeInfoItem.getNodeInfoNode(); 2352 let isHighLight : boolean = false; 2353 let insertChildIndex: number = this.INITIAL_INVALID_VALUE; 2354 let currentChildIndex: number = this.INITIAL_INVALID_VALUE; 2355 let isDownFlag: boolean = this.flag == Flag.DOWN_FLAG ? true : false; 2356 2357 currentChildIndex = this.getChildIndex(dragParentNodeId, dragCurrentNodeId); 2358 2359 insertChildIndex = this.getChildIndex(rearParentNodeId, rearCurrentNodeId) + 1; 2360 2361 if (rearParentNodeId != dragParentNodeId) { 2362 insertChildIndex = isDownFlag ? insertChildIndex + 1 : insertChildIndex; 2363 } else { 2364 if (insertChildIndex > currentChildIndex) { 2365 insertChildIndex = isDownFlag ? insertChildIndex : insertChildIndex - 1; 2366 } else { 2367 insertChildIndex = isDownFlag ? insertChildIndex + 1 : insertChildIndex; 2368 } 2369 } 2370 2371 for (let i = 0; i < this.listNode.length; i++) { 2372 if (this.listNode[i].getCurrentNodeId() == rearCurrentNodeId) { 2373 isHighLight = this.listNode[i].getIsHighLight(); 2374 if (this.flag == Flag.DOWN_FLAG && this.expandAndCollapseInfo.get(rearCurrentNodeId) == NodeStatus.Expand) { 2375 parentNodeId = rearCurrentNodeId; 2376 insertChildIndex = 0; 2377 } else if (this.flag == Flag.UP_FLAG && this.expandAndCollapseInfo.get(rearCurrentNodeId) == NodeStatus.Expand 2378 && this.listNode[i].getCanShowFlagLine() == false) { 2379 parentNodeId = rearCurrentNodeId; 2380 insertChildIndex = 0; 2381 } else if (isHighLight) { 2382 parentNodeId = rearCurrentNodeId; 2383 insertChildIndex = 0; 2384 } 2385 break; 2386 } 2387 } 2388 2389 let callbackParam: CallbackParam = { currentNodeId: currentNodeId, parentNodeId: parentNodeId, childIndex: insertChildIndex } 2390 2391 /* export inner drag node Id. */ 2392 this.appEventBus.emit(TreeListenType.NODE_MOVE, [callbackParam]); 2393 2394 /* To store dragging node information by the array named 'dragNodeParam'. */ 2395 dragNodeParam.push({parentId: parentNodeId, currentId: currentNodeId, data: nodeParam}); 2396 2397 let oneself = null, 2398 callback = function(node, listNode): boolean { 2399 if (node) { 2400 oneself = node; 2401 parentNodeId = oneself.parentNodeId; 2402 currentNodeId = oneself.currentNodeId; 2403 for (let i = 0; i < listNode.length; i++) { 2404 if (listNode[i].getNodeCurrentNodeId() == currentNodeId) { 2405 nodeInfo = listNode[i]; 2406 break; 2407 } 2408 } 2409 nodeParam = nodeInfo.getNodeInfoData(); 2410 if (parentNodeId != dragParentNodeId) { 2411 dragNodeParam.push({parentId: parentNodeId, currentId: currentNodeId, data: nodeParam}); 2412 } 2413 return false; 2414 } 2415 return false; 2416 } 2417 this.listNodeUtils.dragTraverseNodeDF(callback, nodeInfoNode, this.listNode); 2418 2419 /* Delete the dragging node from the tree. */ 2420 this.listNodeUtils.removeNode(dragCurrentNodeId, dragParentNodeId, this.listNodeUtils.traverseNodeBF); 2421 2422 /* 2423 * Add the dragging node to the tree 2424 * 1.The first dragging node is added singly, because it needs to distinguish the position to insert 2425 * 2426 * Add first node. 2427 */ 2428 let insertCurrentNodeId: number = rearCurrentNodeId; 2429 let isAfter: boolean = isDownFlag; 2430 if (this.expandAndCollapseInfo.get(rearCurrentNodeId) == NodeStatus.Expand) { 2431 isAfter = false; 2432 this.listNode.forEach((value) => { 2433 if (value.getCurrentNodeId() == rearCurrentNodeId && value.getCanShowFlagLine() == false) { 2434 if (value.getNodeInfoNode().children.length) { 2435 insertCurrentNodeId = value.getNodeInfoNode().children[0].currentNodeId; 2436 } else { 2437 insertCurrentNodeId = this.INITIAL_INVALID_VALUE; 2438 } 2439 } 2440 }) 2441 } else if (!this.expandAndCollapseInfo.get(rearCurrentNodeId) && isHighLight) { 2442 this.expandAndCollapseInfo.set(rearCurrentNodeId, NodeStatus.Expand); 2443 } 2444 2445 this.listNodeUtils.addDragNode(dragNodeParam[0].parentId, dragNodeParam[0].currentId, insertCurrentNodeId, 2446 isAfter, dragNodeParam[0].data); 2447 2448 /* Add remaining node. */ 2449 for (let j = 1; j < dragNodeParam.length; j++) { 2450 this.listNodeUtils.addNode(dragNodeParam[j].parentId, dragNodeParam[j].currentId, dragNodeParam[j].data); 2451 } 2452 2453 /* Update node data and reload the array named 'listNode'. */ 2454 for (let i = 0; i < this.listNode.length; i++) { 2455 if (this.listNode[i].getCurrentNodeId() == dragParentNodeId) { 2456 if (this.listNode[i].getNodeInfoNode().getNodeItem().imageCollapse == null) { 2457 this.listNode[i].handleImageCollapseAfterAddNode(false); 2458 this.expandAndCollapseInfo.delete(dragParentNodeId); 2459 break; 2460 } 2461 } 2462 } 2463 let tmp: NodeInfo[] = [...this.listNode]; 2464 this.reloadListNode(this.listNodeUtils, tmp); 2465 2466 } 2467 2468 /* 2469 * Reload the array named 'listNode' 2470 * @param listNodeUtils 2471 * @param tmp 2472 */ 2473 public reloadListNode(listNodeUtils: ListNodeUtils, tmp: NodeInfo[]) { 2474 let index = 0; 2475 this.listNode = []; 2476 this.listNodeUtils = listNodeUtils; 2477 this.loadedNodeIdAndIndexMap.clear(); 2478 this.listNodeUtils.traverseNodeDF((node: NodeItem): boolean => { 2479 if (node.currentNodeId >= 0) { 2480 var nodeInfo: NodeInfo = new NodeInfo(node); 2481 this.listNode.push(nodeInfo); 2482 2483 if (this.expandAndCollapseInfo.get(node.currentNodeId) == NodeStatus.Expand) { 2484 nodeInfo.getNodeItem().imageCollapse.changeImageCollapseSource(NodeStatus.Expand); 2485 } else if (this.expandAndCollapseInfo.get(node.currentNodeId) == NodeStatus.Collapse) { 2486 nodeInfo.getNodeItem().imageCollapse.changeImageCollapseSource(NodeStatus.Collapse); 2487 } 2488 2489 for (let i = 0; i < tmp.length; i++){ 2490 if (tmp[i].getCurrentNodeId() == nodeInfo.getCurrentNodeId()) { 2491 nodeInfo.setNodeIsShow(tmp[i].getNodeIsShow()); 2492 nodeInfo.setListItemHeight(tmp[i].getListItemHeight()); 2493 if (nodeInfo.getNodeItem().mainTitleNode && nodeInfo.getIsShowTitle()) { 2494 nodeInfo.getNodeItem().mainTitleNode.title = tmp[i].getNodeItem().mainTitleNode.title; 2495 } 2496 break; 2497 } 2498 } 2499 if (nodeInfo.getNodeIsShow()) { 2500 this.loadedNodeIdAndIndexMap.set(nodeInfo.getCurrentNodeId(), index++); 2501 } 2502 } 2503 return false; 2504 }); 2505 } 2506 2507 public getFlagLine() { 2508 return this.FLAG_LINE; 2509 } 2510 2511 public getVisibility(nodeInfo: NodeInfo): any { 2512 let lastShowIndex: number = this.loadedNodeIdAndIndexMap.get(nodeInfo.getCurrentNodeId()) - 1; 2513 if (lastShowIndex > this.INITIAL_INVALID_VALUE) { 2514 let lastNodeInfo: NodeInfo = this.getData(lastShowIndex); 2515 return (nodeInfo.getCanShowFlagLine() == true && !nodeInfo.getIsHighLight() && !lastNodeInfo.getIsHighLight()) ? 2516 Visibility.Visible : Visibility.Hidden; 2517 } else { 2518 return (nodeInfo.getCanShowFlagLine() == true && !nodeInfo.getIsHighLight()) ? 2519 Visibility.Visible : Visibility.Hidden; 2520 } 2521 } 2522 2523 public getSubTitlePara() { 2524 return this.subTitle; 2525 } 2526 2527 public getIsFolder(nodeId: number) { 2528 if (this.loadedNodeIdAndIndexMap.has(nodeId)) { 2529 return this.getData(this.loadedNodeIdAndIndexMap.get(nodeId)).getIsFolder(); 2530 } 2531 return false; 2532 } 2533 2534 public getSubTitleFontColor(isHighLight: boolean) { 2535 return isHighLight ? this.subTitle.highLightFontColor : this.subTitle.normalFontColor; 2536 } 2537 2538 private getChildIndex(rearParentNodeId: number, rearCurrentNodeId: number) { 2539 let insertChildIndex: number = this.INITIAL_INVALID_VALUE; 2540 this.listNodeUtils.traverseNodeBF(function(node): boolean { 2541 if (node.getCurrentNodeId() == rearParentNodeId) { 2542 node.children.forEach((value, index) => { 2543 if (value.getCurrentNodeId() == rearCurrentNodeId) { 2544 insertChildIndex = index; 2545 } 2546 }) 2547 return true; 2548 } 2549 return false; 2550 }); 2551 return insertChildIndex; 2552 } 2553 2554 public setCurrentFocusNodeId(focusNodeId: number) { 2555 this.currentFocusNodeId = focusNodeId; 2556 } 2557 2558 public getCurrentFocusNodeId(): number { 2559 return this.currentFocusNodeId; 2560 } 2561 2562 public setLastFocusNodeId(focusNodeId: number) { 2563 this.lastFocusNodeId = focusNodeId; 2564 } 2565 2566 public getLastFocusNodeId(): number { 2567 return this.lastFocusNodeId; 2568 } 2569 2570 public getAddFocusNodeId(): number { 2571 return this.addFocusNodeId; 2572 } 2573 2574 public setFlag(flag: Flag) { 2575 this.flag = flag; 2576 } 2577 } 2578 2579 /** 2580 * Tree view control, which is created by using the TreeController class. 2581 * 2582 * When you create this component, you must initialize the listNodeDataSource. 2583 * You can run the listTreeViewWidth command to set the width of the component. 2584 * The default width is 200vp. 2585 * 2586 * @since 10 2587 */ 2588 @Component 2589 export struct TreeView { 2590 @State nodeList: NodeInfo[] = []; 2591 listNodeDataSource: ListNodeDataSource; 2592 @State item: NodeInfo[] = null; 2593 treeController: TreeController; 2594 @State touchCount: number = 0; 2595 @State dropSelectedIndex: number = 0; 2596 @State viewLastIndex: number = -1; 2597 @State listItemBgColor: Resource = $r('sys.color.ohos_id_color_background_transparent') 2598 @BuilderParam private listTreeViewMenu: () => void = null; 2599 private readonly MAX_CN_LENGTH: number = 254; 2600 private readonly MAX_EN_LENGTH: number = 255; 2601 private readonly INITIAL_INVALID_VALUE = -1; 2602 private readonly MAX_TOUCH_DOWN_COUNT = 0; 2603 private isMultiPress: boolean = false; 2604 private touchDownCount: number = this.INITIAL_INVALID_VALUE; 2605 private appEventBus: TreeListener = TreeListenerManager.getInstance().getTreeListener(); 2606 private readonly itemPadding: { left: Resource, right: Resource, top: Resource, bottom: Resource } = 2607 { left: $r('sys.float.ohos_id_card_margin_start'), 2608 right: $r('sys.float.ohos_id_card_margin_end'), 2609 top: $r('sys.float.ohos_id_text_margin_vertical'), 2610 bottom: $r('sys.float.ohos_id_text_margin_vertical')}; 2611 /* { left: '12vp', right: '12vp', top: '2vp', bottom: '2vp' } */ 2612 private readonly textInputPadding: { left : string, right: string, top: string, bottom: string } = 2613 { left : '0vp', right: '0vp', top: '0vp', bottom: '0vp'} 2614 aboutToAppear(): void { 2615 this.listNodeDataSource = this.treeController.getListNodeDataSource(); 2616 this.nodeList = this.treeController.getListNodeDataSource().listNode; 2617 this.item = this.treeController.getListNodeDataSource().listNode; 2618 } 2619 2620 private checkInvalidPattern(title: string): boolean { 2621 let pattern = /[\\\/:*?"<>|]/; 2622 return pattern.test(title) 2623 } 2624 2625 private checkIsAllCN(title: string): boolean { 2626 let pattern = /^[\u4e00-\u9fa5]+$/; 2627 return pattern.test(title) 2628 } 2629 2630 @Builder popupForShowTitle(text: string | Resource, backgroundColor: Resource, fontColor: Resource ) { 2631 Row() { 2632 Text(text).fontSize($r('sys.float.ohos_id_text_size_body2')).fontWeight('regular').fontColor(fontColor) 2633 }.backgroundColor(backgroundColor) 2634 .border({radius: $r('sys.float.ohos_id_elements_margin_horizontal_l')}) 2635 .padding({left: $r('sys.float.ohos_id_elements_margin_horizontal_l'), 2636 right: $r('sys.float.ohos_id_elements_margin_horizontal_l'), 2637 top: $r('sys.float.ohos_id_card_margin_middle'), 2638 bottom: $r('sys.float.ohos_id_card_margin_middle')}) 2639 } 2640 2641 @Builder builder() { 2642 this.listTreeViewMenu() 2643 } 2644 2645 /* Set the popup of dragging node. */ 2646 @Builder draggingPopup(item: NodeInfo) { 2647 Row() { 2648 if (item.getNodeItem().imageNode) { 2649 Row() { 2650 Image(item.getNodeItem().imageNode.normalSource) 2651 .objectFit(ImageFit.Contain) 2652 .height(item.getNodeItem().imageNode.itemHeight) 2653 .width(item.getNodeItem().imageNode.itemWidth) 2654 .opacity(this.listNodeDataSource.getDragPopupPara().imageOpacity) 2655 } 2656 .backgroundColor(COLOR_IMAGE_ROW) 2657 .margin({ right: item.getNodeItem().imageNode.itemRightMargin }) 2658 .height(item.getNodeItem().imageNode.itemHeight) 2659 .width(item.getNodeItem().imageNode.itemWidth) 2660 } 2661 2662 Row() { 2663 if (item.getNodeItem().mainTitleNode && item.getIsShowTitle()) { 2664 Text(item.getNodeItem().mainTitleNode.title) 2665 .maxLines(1) 2666 .fontSize(item.getNodeItem().mainTitleNode.size) 2667 .fontColor(this.listNodeDataSource.getDragPopupPara().fontColor) 2668 .fontWeight(this.listNodeDataSource.getDragPopupPara().fontWeight) 2669 .textOverflow({ overflow: TextOverflow.Ellipsis }) 2670 } 2671 } 2672 .constraintSize({ 2673 minWidth: item.getNodeItem().imageNode ? 2674 this.listNodeDataSource.getDragPopupPara().textConstraintSize.minWidth1 : 2675 this.listNodeDataSource.getDragPopupPara().textConstraintSize.minWidth2, 2676 maxWidth: item.getNodeItem().imageNode ? 2677 this.listNodeDataSource.getDragPopupPara().textConstraintSize.maxWidth1 : 2678 this.listNodeDataSource.getDragPopupPara().textConstraintSize.maxWidth2 }) 2679 } 2680 .constraintSize({ minWidth: this.listNodeDataSource.getDragPopupPara().floorConstraintSize.minWidth, 2681 maxWidth: this.listNodeDataSource.getDragPopupPara().floorConstraintSize.maxWidth }) 2682 .height(this.listNodeDataSource.getDragPopupPara().height) 2683 .backgroundColor(this.listNodeDataSource.getDragPopupPara().backgroundColor) 2684 .padding({ left: this.listNodeDataSource.getDragPopupPara().padding.left, 2685 right: this.listNodeDataSource.getDragPopupPara().padding.right }) 2686 .shadow({ radius: $r('sys.float.ohos_id_corner_radius_default_m'), 2687 color: SHADOW_COLOR, 2688 offsetY: 0 }) 2689 .borderRadius(this.listNodeDataSource.getDragPopupPara().borderRadius) // need to doubleCheck. 2690 } 2691 2692 clearLastIndexColor(){ 2693 if (this.viewLastIndex == -1 || this.viewLastIndex >= this.nodeList.length) { 2694 return; 2695 } 2696 this.setImageSources(this.viewLastIndex, InteractionStatus.Normal); 2697 2698 this.nodeList[this.viewLastIndex].setNodeColor($r('sys.color.ohos_id_color_background_transparent')) 2699 this.nodeList[this.viewLastIndex].fontColor = $r('sys.color.ohos_id_color_primary'); 2700 this.listNodeDataSource.listNode[this.viewLastIndex].setNodeColor($r('sys.color.ohos_id_color_background_transparent')); 2701 this.listNodeDataSource.listNode[this.viewLastIndex].fontColor = $r('sys.color.ohos_id_color_primary'); 2702 this.listNodeDataSource.listNode[this.viewLastIndex].setIsSelected(false); 2703 this.listNodeDataSource.setImageSource(this.viewLastIndex, InteractionStatus.Normal); 2704 } 2705 2706 setImageSources(index: number, interactionStatus: InteractionStatus) { 2707 let nodeInfo: NodeInfo = this.nodeList[index]; 2708 nodeInfo.setIsSelected(interactionStatus === InteractionStatus.Selected || 2709 interactionStatus === InteractionStatus.Edit || interactionStatus === InteractionStatus.FinishEdit); 2710 if (nodeInfo.getNodeItem().mainTitleNode != null && interactionStatus != InteractionStatus.DragInsert && 2711 interactionStatus != InteractionStatus.FinishDragInsert) { 2712 nodeInfo.getNodeItem().mainTitleNode.setMainTitleSelected(interactionStatus === InteractionStatus.Selected || 2713 interactionStatus === InteractionStatus.FinishEdit); 2714 } 2715 if (nodeInfo.getNodeItem().imageNode != null) { 2716 nodeInfo.getNodeItem().imageNode.setImageSource(interactionStatus); 2717 } 2718 } 2719 2720 findIndex(currentNodeId: number): number { 2721 let thisIndex: number = 0; 2722 this.listNodeDataSource.ListNode.forEach(function (value, index) { 2723 if (value.getNodeCurrentNodeId() == currentNodeId) { 2724 thisIndex = index; 2725 } 2726 }) 2727 return thisIndex; 2728 } 2729 2730 build() { 2731 List({ }) { 2732 LazyForEach(this.listNodeDataSource, (itemInner: NodeInfo) => { 2733 ListItem() { 2734 Row() { 2735 TreeViewInner({ 2736 item: itemInner, 2737 listNodeDataSource: this.listNodeDataSource, 2738 index: this.findIndex(itemInner.getNodeCurrentNodeId()), 2739 }) 2740 } 2741 .onTouch((event: TouchEvent) => { 2742 this.viewLastIndex = this.listNodeDataSource.getLastIndex(); 2743 let index = this.findIndex(itemInner.getNodeCurrentNodeId()) 2744 2745 if (event.type == TouchType.Down) { 2746 if (index != this.viewLastIndex) { 2747 this.clearLastIndexColor() 2748 this.listNodeDataSource.lastIndex = index; 2749 this.listNodeDataSource.setClickIndex(index); 2750 } 2751 } 2752 if (event.type == TouchType.Up) { 2753 this.listNodeDataSource.listNode[index].setIsSelected(true); 2754 this.listNodeDataSource.setImageSource(index, InteractionStatus.Selected); 2755 if (this.listNodeDataSource.listNode[index].getNodeItem().imageNode != null) { 2756 this.listNodeDataSource.listNode[index].imageSource = this.listNodeDataSource.listNode[index].getNodeItem().imageNode.source; 2757 } 2758 2759 if (index != this.viewLastIndex) { 2760 this.clearLastIndexColor() 2761 2762 this.listNodeDataSource.lastIndex = index; 2763 this.listNodeDataSource.setClickIndex(index); 2764 } 2765 this.viewLastIndex = index; 2766 } 2767 2768 if (this.listNodeDataSource.getLastIndex() !== -1 && index !== this.listNodeDataSource.getLastIndex()) { 2769 this.listNodeDataSource.setPopUpInfo(PopUpType.WARNINGS, InputError.NONE, false, this.listNodeDataSource.getLastIndex()); 2770 this.listNodeDataSource.setItemVisibilityOnEdit(this.listNodeDataSource.getLastIndex(), MenuOperation.COMMIT_NODE); 2771 } 2772 }) 2773 } 2774 .width('100%').height(itemInner.getListItemHeight()) 2775 .padding({ left: this.itemPadding.left, right: this.itemPadding.right}) 2776 .align(Alignment.Start) 2777 .onDragStart((event: DragEvent, extraParams: string) => { 2778 if (this.listNodeDataSource.getIsDrag() || this.listNodeDataSource.getIsInnerDrag() || this.isMultiPress) { 2779 console.error('drag error, a item has been dragged'); 2780 return; 2781 } 2782 this.dropSelectedIndex = JSON.parse(extraParams).selectedIndex; 2783 let currentNodeIndex = JSON.parse(extraParams).selectedIndex; 2784 let currentNodeInfo = this.listNodeDataSource.getData(currentNodeIndex); 2785 let currentItemNodeId = itemInner.getNodeCurrentNodeId(); 2786 /* handle the situation of drag error, currentNodeIndex is not found in onDragStart. */ 2787 if (currentNodeIndex >= this.listNodeDataSource.totalCount() || currentNodeIndex == undefined) { 2788 console.error('drag error, currentNodeIndex is not found in onDragStart'); 2789 return; 2790 } 2791 2792 this.listNodeDataSource.setIsInnerDrag(true); 2793 this.listNodeDataSource.setIsDrag(true); 2794 this.listNodeDataSource.setCurrentNodeInfo(currentNodeInfo); 2795 this.listNodeDataSource.setDraggingCurrentNodeId(currentNodeInfo.getNodeCurrentNodeId()); 2796 this.listNodeDataSource.setDraggingParentNodeId(currentNodeInfo.getNodeParentNodeId()); 2797 2798 /* set the opacity of the dragging node. */ 2799 let draggingNodeOpacity: number = DRAG_OPACITY; 2800 this.listNodeDataSource.setListItemOpacity(draggingNodeOpacity); 2801 this.listNodeDataSource.notifyDataChange(currentNodeIndex); 2802 2803 /* 2804 * handle the situation of drag is too fast,it attribute a fault to OH. 2805 * OH has Solved on real machine. 2806 */ 2807 if (currentItemNodeId != currentNodeInfo.getNodeCurrentNodeId()) { 2808 console.error('drag is too fast,it attribute a fault to OH'); 2809 this.listNodeDataSource.setIsDrag(false); 2810 return; 2811 } 2812 2813 return this.draggingPopup(currentNodeInfo); 2814 }) 2815 }, item => JSON.stringify(item)) 2816 } 2817 2818 /* Move the dragged node. */ 2819 .onDragMove((event: DragEvent) => { 2820 if (this.isMultiPress) { 2821 console.error('drag error, a item has been dragged'); 2822 return; 2823 } 2824 let nodeHeight: number = LIST_ITEM_HEIGHT; 2825 2826 /* flag the position of the focus on the node. */ 2827 let flag: Flag = Math.floor(event.getY() / (nodeHeight / FLAG_NUMBER)) % FLAG_NUMBER ? Flag.DOWN_FLAG : Flag.UP_FLAG; 2828 2829 /* Record the node position to which the dragged node moves. */ 2830 let index = JSON.parse(extraParams).insertIndex; 2831 2832 /* Handle the situation where the focus(index) exceeds the list area. */ 2833 let isOverBorder: boolean = false; 2834 if (index >= this.listNodeDataSource.totalCount()) { 2835 flag = Flag.DOWN_FLAG; 2836 index = this.listNodeDataSource.totalCount() - 1; 2837 this.listNodeDataSource.getData(index).setIsOverBorder(true); 2838 isOverBorder = true; 2839 } else { 2840 this.listNodeDataSource.getData(index).setIsOverBorder(false); 2841 } 2842 2843 let currentNodeInfo: NodeInfo = this.listNodeDataSource.getData(index); 2844 let currentNodeId: number = currentNodeInfo.getCurrentNodeId(); 2845 2846 /* 2847 * handle a situation that "draggingCurrentNodeId" is parent of "insertNodeCurrentNodeId"; 2848 * do not perform some functions. 2849 */ 2850 if (index != this.listNodeDataSource.getLastPassIndex() && this.listNodeDataSource.getIsInnerDrag()) { 2851 let isParentNodeOfInsertNode: boolean = this.listNodeDataSource.getIsParentOfInsertNode(currentNodeId); 2852 if (isParentNodeOfInsertNode) { 2853 this.listNodeDataSource.setPassIndex(index); 2854 let that = this; 2855 this.listNodeDataSource.clearTimeOutAboutDelayHighLightAndExpand(findCurrentNodeIndex.call(that, 2856 currentNodeId)); 2857 this.listNodeDataSource.setFlag(Flag.NONE); 2858 return; 2859 } 2860 } 2861 this.listNodeDataSource.setLastPassIndex(index); 2862 2863 /* Set the visibility of the flag line. */ 2864 this.listNodeDataSource.setVisibility(flag, index - 1, isOverBorder); 2865 2866 /* Automatically HighLight one second delay and expand after two second delay. */ 2867 if (currentNodeId != this.listNodeDataSource.getDraggingCurrentNodeId()) { 2868 let that = this; 2869 this.listNodeDataSource.delayHighLightAndExpandNode(findCurrentNodeIndex.call(that, currentNodeId), 2870 currentNodeId, index); 2871 } 2872 }) 2873 2874 /* DragEvent Enter. */ 2875 .onDragEnter((event: DragEvent, extraParams: string) => { 2876 if (this.listNodeDataSource.getIsInnerDrag()) { 2877 this.listNodeDataSource.setIsDrag(true); 2878 2879 /* set the opacity of the dragging node. */ 2880 let draggingNodeOpacity: number = DRAG_OPACITY; 2881 this.listNodeDataSource.setListItemOpacity(draggingNodeOpacity); 2882 } 2883 }) 2884 2885 /* DragEvent Leave. */ 2886 .onDragLeave((event: DragEvent, extraParams: string) => { 2887 this.listNodeDataSource.hideLastLine(); 2888 this.listNodeDataSource.clearLastTimeoutHighLight(); 2889 this.listNodeDataSource.clearLastTimeoutExpand(); 2890 let draggingNodeOpacity: number = DRAG_OPACITY_NONE; 2891 this.listNodeDataSource.setListItemOpacity(draggingNodeOpacity); 2892 this.listNodeDataSource.setIsDrag(false); 2893 this.listNodeDataSource.notifyDataReload(); 2894 }) 2895 2896 /* DragEvent Drop. */ 2897 .onDrop((event: DragEvent, extraParams: string) => { 2898 this.listNodeDataSource.clearLastTimeoutExpand(); 2899 let draggingNodeOpacity: number = DRAG_OPACITY_NONE; 2900 this.listNodeDataSource.setListItemOpacity(draggingNodeOpacity); 2901 let insertNodeIndex: number = JSON.parse(extraParams).insertIndex 2902 let currentNodeIndex: number = this.dropSelectedIndex; 2903 2904 if (currentNodeIndex - 1 > this.listNodeDataSource.totalCount() || currentNodeIndex == undefined) { 2905 console.error('drag error, currentNodeIndex is not found'); 2906 this.listNodeDataSource.setIsDrag(false); 2907 return; 2908 } 2909 2910 if (insertNodeIndex == this.listNodeDataSource.totalCount()) { 2911 console.log('need to insert into the position of the last line, now insertNodeIndex = insertNodeIndex - 1'); 2912 insertNodeIndex -= 1; 2913 } 2914 2915 let insertNodeInfo: NodeInfo = this.listNodeDataSource.getData(insertNodeIndex); 2916 if (insertNodeInfo == null) { 2917 return; 2918 } 2919 let insertNodeCurrentNodeId: number = insertNodeInfo.getNodeCurrentNodeId(); 2920 2921 /* outer node is move in. */ 2922 if (!this.listNodeDataSource.getIsDrag() || !this.listNodeDataSource.getIsInnerDrag()) { 2923 this.listNodeDataSource.clearLastTimeoutHighLight(); 2924 this.listNodeDataSource.setIsInnerDrag(false); 2925 this.listNodeDataSource.hideLastLine(); 2926 this.listNodeDataSource.initialParameterAboutDelayHighLightAndExpandIndex(); 2927 this.listNodeDataSource.refreshSubtitle(insertNodeCurrentNodeId); 2928 this.listNodeDataSource.notifyDataReload(); 2929 return; 2930 } 2931 2932 let currentNodeInfo: NodeInfo = this.listNodeDataSource.getCurrentNodeInfo(); 2933 let insertNodeParentNodeId: number = insertNodeInfo.getNodeParentNodeId(); 2934 let draggingCurrentNodeId: number = this.listNodeDataSource.getDraggingCurrentNodeId(); 2935 let draggingParentNodeId: number = this.listNodeDataSource.getDraggingParentNodeId(); 2936 2937 /* 2938 * handle a situation that "draggingCurrentNodeId" is parent of "insertNodeCurrentNodeId". 2939 * drag is fail. 2940 */ 2941 let isParentNodeOfInsertNode: boolean = this.listNodeDataSource.getIsParentOfInsertNode(insertNodeCurrentNodeId); 2942 if (isParentNodeOfInsertNode) { 2943 this.listNodeDataSource.clearLastTimeoutHighLight(); 2944 this.listNodeDataSource.setIsInnerDrag(false); 2945 this.listNodeDataSource.hideLastLine(); 2946 this.listNodeDataSource.notifyDataChange(insertNodeIndex); 2947 this.listNodeDataSource.initialParameterAboutDelayHighLightAndExpandIndex(); 2948 this.listNodeDataSource.setIsDrag(false); 2949 2950 /* set the position of focus. */ 2951 let that = this; 2952 let currentFocusIndex: number = findCurrentNodeIndex.call(that, draggingCurrentNodeId); 2953 this.listNodeDataSource.setClickIndex(currentFocusIndex); 2954 this.listNodeDataSource.handleEvent(Event.DRAG, currentFocusIndex); 2955 return; 2956 } 2957 2958 /* Collapse drag node. */ 2959 if (this.listNodeDataSource.getExpandAndCollapseInfo(draggingCurrentNodeId) == NodeStatus.Expand) { 2960 let that = this; 2961 this.listNodeDataSource.expandAndCollapseNode( 2962 findCurrentNodeIndex.call(that, draggingCurrentNodeId)); 2963 } 2964 2965 let flag: boolean = false; 2966 2967 /* Expand insert node. */ 2968 if (this.listNodeDataSource.getExpandAndCollapseInfo(insertNodeCurrentNodeId) == NodeStatus.Collapse) { 2969 let that = this; 2970 let currentIndex: number = findCurrentNodeIndex.call(that, insertNodeCurrentNodeId); 2971 if (this.listNodeDataSource.ListNode[currentIndex].getIsHighLight()) { 2972 this.listNodeDataSource.expandAndCollapseNode(currentIndex); 2973 } 2974 flag = true; 2975 } 2976 2977 /* alter dragNode. */ 2978 this.listNodeDataSource.setLastDelayHighLightId(); 2979 if (draggingCurrentNodeId != insertNodeCurrentNodeId){ 2980 this.listNodeDataSource.alterDragNode(insertNodeParentNodeId, insertNodeCurrentNodeId, insertNodeInfo, 2981 draggingParentNodeId, draggingCurrentNodeId, currentNodeInfo); 2982 this.listNodeDataSource.hideLastLine(); 2983 } else { 2984 /*the position of dragNode is equal with the position of insertNode. */ 2985 this.listNodeDataSource.hideLastLine(); 2986 this.listNodeDataSource.setLastPassId(draggingCurrentNodeId); 2987 this.listNodeDataSource.hideLastLine(); 2988 } 2989 let that = this; 2990 let lastDelayHighLightIndex: number = findCurrentNodeIndex.call(that, 2991 this.listNodeDataSource.getLastDelayHighLightId()); 2992 this.listNodeDataSource.setLastDelayHighLightIndex(lastDelayHighLightIndex); 2993 this.listNodeDataSource.clearLastTimeoutHighLight(); 2994 this.listNodeDataSource.initialParameterAboutDelayHighLightAndExpandIndex(); 2995 this.listNodeDataSource.setIsDrag(false); 2996 2997 /* set the position of focus. */ 2998 let currentFocusIndex: number = findCurrentNodeIndex.call(that, draggingCurrentNodeId); 2999 this.listNodeDataSource.setClickIndex(currentFocusIndex); 3000 this.listNodeDataSource.handleEvent(Event.DRAG, currentFocusIndex); 3001 3002 /* innerDrag is over. */ 3003 this.listNodeDataSource.setIsInnerDrag(false); 3004 this.listNodeDataSource.notifyDataReload(); 3005 3006 this.listNodeDataSource.listNode[currentFocusIndex].fontColor = $r('sys.color.ohos_id_color_text_primary_activated'); 3007 if (this.viewLastIndex !== -1 && currentNodeIndex != this.viewLastIndex) { 3008 this.listNodeDataSource.listNode[this.viewLastIndex].getNodeItem().mainTitleNode.setMainTitleSelected(false) 3009 this.listNodeDataSource.listNode[this.viewLastIndex].getNodeItem().mainTitleNode.setMainTitleHighLight(false) 3010 } 3011 3012 if (this.listNodeDataSource.listNode[this.viewLastIndex] != null) { 3013 this.listNodeDataSource.listNode[this.viewLastIndex].fontColor = $r('sys.color.ohos_id_color_text_primary'); 3014 } 3015 3016 this.listNodeDataSource.lastIndex = this.viewLastIndex; 3017 if (this.listNodeDataSource.listNode[this.viewLastIndex]) { 3018 if (this.listNodeDataSource.listNode[this.viewLastIndex].getNodeItem().imageNode != null) { 3019 3020 this.listNodeDataSource.listNode[this.viewLastIndex].getNodeItem().imageNode.setImageSource(InteractionStatus.Normal); 3021 this.listNodeDataSource.listNode[this.viewLastIndex].imageSource = this.listNodeDataSource.listNode[this.viewLastIndex].getNodeItem().imageNode.source; 3022 } 3023 } 3024 3025 if(this.listNodeDataSource.listNode[this.viewLastIndex]) { 3026 this.listNodeDataSource.listNode[this.viewLastIndex].setNodeColor($r('sys.color.ohos_id_color_background_transparent')); 3027 } 3028 3029 this.viewLastIndex = currentFocusIndex; 3030 }) 3031 } 3032 } 3033 3034 // Declare NodeParam 3035 export interface NodeParam { 3036 parentNodeId?: number, 3037 currentNodeId?: number, 3038 isFolder?: boolean, 3039 icon?: Resource, 3040 selectedIcon?: Resource, 3041 editIcon?: Resource, 3042 primaryTitle?: string, 3043 secondaryTitle?: number | string, 3044 container?: () => void, 3045 } 3046 3047 // Declare CallbackParam 3048 export interface CallbackParam { 3049 currentNodeId: number, 3050 parentNodeId?: number, 3051 childIndex?: number, 3052 } 3053 3054 /** 3055 * Create a tree view control proxy class to generate a tree view. 3056 * 3057 * @since 10 3058 */ 3059 export class TreeController { 3060 readonly ROOT_NODE_ID: number = -1; 3061 private nodeIdList: number[] = []; 3062 private listNodeUtils : ListNodeUtils = new ListNodeUtils(); 3063 private listNodeDataSource : ListNodeDataSource = new ListNodeDataSource(); 3064 3065 /** 3066 * After the addNode interface is invoked, 3067 * this interface is used to obtain the initialization data of 3068 * the tree view component for creating a tree view component. 3069 * 3070 * @return ListNodeDataSource Obtains the initialization data of the tree view component. 3071 * 3072 * @since 10 3073 */ 3074 public getListNodeDataSource(): ListNodeDataSource { 3075 return this.listNodeDataSource; 3076 } 3077 3078 /** 3079 * Obtains the subNode information of the currently clicked node. 3080 * 3081 * @return Array Returns an array that stores the configuration information of each node. 3082 * If there is no child node, an empty array is returned. 3083 * 3084 * @since 10 3085 */ 3086 public getClickNodeChildrenInfo(): Array<{ itemId: number, itemIcon: Resource, itemTitle: string, 3087 isFolder: boolean }> { 3088 let clickNodeId = this.listNodeDataSource.getClickNodeId(); 3089 return this.listNodeUtils.getClickNodeChildrenInfo(clickNodeId); 3090 } 3091 3092 public getChildrenId(): Array<number> { 3093 let clickNodeId = this.listNodeDataSource.getClickNodeId(); 3094 return this.listNodeUtils.getClickChildId(clickNodeId); 3095 } 3096 3097 /** 3098 * Delete a node. 3099 * 3100 * Register an ON_ITEM_DELETE callback through the EventBus mechanism to obtain the IDs of all deleted nodes. 3101 * 3102 * @since 10 3103 */ 3104 public removeNode(): void { 3105 let clickNodeId = this.listNodeDataSource.getClickNodeId(); 3106 if (clickNodeId < 0) { 3107 return; 3108 } 3109 let parentNodeId = this.listNodeUtils.findParentNodeId(clickNodeId); 3110 let removeNodeIdList: number[] = this.listNodeUtils.removeNode(clickNodeId, 3111 parentNodeId, this.listNodeUtils.traverseNodeBF); 3112 this.listNodeDataSource.refreshData(this.listNodeUtils, MenuOperation.REMOVE_NODE, parentNodeId, removeNodeIdList); 3113 this.nodeIdList.splice(this.nodeIdList.indexOf(clickNodeId), 1); 3114 } 3115 3116 /** 3117 * Modify the node name. 3118 * 3119 * Register an ON_ITEM_MODIFY callback to obtain the ID, parent node ID, and node name of the modified node. 3120 * 3121 * @since 10 3122 */ 3123 public modifyNode(): void { 3124 let clickNodeId = this.listNodeDataSource.getClickNodeId(); 3125 this.listNodeDataSource.setItemVisibilityOnEdit(clickNodeId, MenuOperation.MODIFY_NODE); 3126 } 3127 3128 /** 3129 * Add a node. 3130 * 3131 * Icon of a new node, which is generated by the system by default. 3132 * If there is a same-level node, the icon of the first node of the same-level node is used. 3133 * If no icon is set for the first node of the same-level node, the new node does not have an icon. 3134 * If there is no peer node, the icon of the parent node is used. 3135 * If no icon is set for the parent node, the new node does not have an icon. 3136 * The system generates an ID for the new node and registers an ON_ITEM_ADD callback through 3137 * the EventBus mechanism to obtain the ID, parent node ID, node name, normal icon, selected icon, 3138 * edit icon of the new node. 3139 * 3140 * @since 10 3141 */ 3142 public add(): void { 3143 let clickNodeId: number = this.listNodeDataSource.getClickNodeId(); 3144 if (clickNodeId == this.listNodeDataSource.ROOT_NODE_ID || !this.listNodeDataSource.getIsFolder(clickNodeId)) { 3145 return; 3146 } 3147 let newNodeInfo: { isFolder: boolean, icon: Resource, selectedIcon: Resource, editIcon: Resource, 3148 container: () =>any, secondaryTitle: number | string } = 3149 { isFolder: true, icon: null, selectedIcon: null, editIcon: null, container: null, secondaryTitle: '' }; 3150 newNodeInfo = this.listNodeUtils.getNewNodeInfo(clickNodeId); 3151 this.nodeIdList.push(this.nodeIdList[this.nodeIdList.length - 1] + 1); 3152 let newNodeId: number = this.nodeIdList[this.nodeIdList.length - 1]; 3153 this.listNodeUtils.addNewNodeId = newNodeId; 3154 this.listNodeUtils.addNode(clickNodeId, newNodeId, 3155 { isFolder: newNodeInfo.isFolder, icon: newNodeInfo.icon, selectedIcon: newNodeInfo.selectedIcon, 3156 editIcon: newNodeInfo.editIcon, primaryTitle: '新建文件夹', container: newNodeInfo.container, 3157 secondaryTitle: newNodeInfo.secondaryTitle }); 3158 this.listNodeDataSource.refreshData(this.listNodeUtils, MenuOperation.ADD_NODE, clickNodeId, [newNodeId]); 3159 3160 this.listNodeDataSource.setPopUpInfo(PopUpType.WARNINGS, InputError.NONE, false, this.listNodeDataSource.getLastIndex()); 3161 this.listNodeDataSource.setItemVisibilityOnEdit(this.listNodeDataSource.getLastIndex(), MenuOperation.COMMIT_NODE); 3162 this.listNodeDataSource.listNode[this.listNodeDataSource.getLastIndex()].setFontColor($r('sys.color.ohos_id_color_text_primary')); 3163 3164 let newNodeindex = findCurrentNodeIndex.call(this, newNodeId); 3165 this.listNodeDataSource.setClickIndex(newNodeindex); 3166 this.listNodeDataSource.handleEvent(Event.TOUCH_UP, newNodeindex); 3167 } 3168 3169 public addNodeParam(nodeParam: NodeParam): TreeController { 3170 if (nodeParam.primaryTitle != null && 3171 !this.listNodeUtils.checkMainTitleIsValid(nodeParam.primaryTitle)) { 3172 throw new Error('ListTreeNode[addNode]: ' + 3173 'The directory name cannot contain the following characters\ /: *? "< > | or exceeds the maximum length.'); 3174 return null; 3175 } 3176 if (nodeParam.primaryTitle == null && nodeParam.icon == null) { 3177 throw new Error('ListTreeNode[addNode]: ' + 3178 'The icon and directory name cannot be empty at the same time.'); 3179 return null; 3180 } 3181 if (nodeParam.currentNodeId === this.ROOT_NODE_ID || nodeParam.currentNodeId === null) { 3182 throw new Error('ListTreeNode[addNode]: currentNodeId can not be -1 or null.'); 3183 return null; 3184 } 3185 this.nodeIdList.push(nodeParam.currentNodeId); 3186 this.listNodeUtils.addNode(nodeParam.parentNodeId, nodeParam.currentNodeId, nodeParam); 3187 return this; 3188 } 3189 3190 /** 3191 * Initialize the interface of the tree view. This interface is used to generate ListNodeDataSource data. 3192 * addNode is only designed for initialization. It can only be invoked during initialization. 3193 * 3194 * A maximum of 50 directory levels can be added. 3195 * 3196 * @param parentNodeId ID of the parent node. 3197 * @param currentNodeId ID of the new node. The value cannot be -1 or null. 3198 * @param nodeParam Configuration information of the newly added node. 3199 * For details, see the comment description of NodeParam. 3200 * @return ListTreeNode Tree view component proxy class. 3201 * 3202 * @since 10 3203 */ 3204 public addNode(nodeParam?: NodeParam): TreeController { 3205 if (nodeParam == null) { 3206 this.add(); 3207 } else { 3208 if (nodeParam.primaryTitle != null && 3209 !this.listNodeUtils.checkMainTitleIsValid(nodeParam.primaryTitle)) { 3210 throw new Error('ListTreeNode[addNode]: ' + 3211 'The directory name cannot contain the following characters\ /: *? "< > | or exceeds the maximum length.'); 3212 return null; 3213 } 3214 if (nodeParam.primaryTitle == null && nodeParam.icon == null) { 3215 throw new Error('ListTreeNode[addNode]: ' + 3216 'The icon and directory name cannot be empty at the same time.'); 3217 return null; 3218 } 3219 if (nodeParam.currentNodeId === this.ROOT_NODE_ID || nodeParam.currentNodeId === null) { 3220 throw new Error('ListTreeNode[addNode]: currentNodeId can not be -1 or null.'); 3221 return null; 3222 } 3223 this.nodeIdList.push(nodeParam.currentNodeId); 3224 this.listNodeUtils.addNode(nodeParam.parentNodeId, nodeParam.currentNodeId, nodeParam); 3225 return this; 3226 } 3227 } 3228 3229 /** 3230 * After the initialization is complete by calling the addNode interface, 3231 * call this interface to complete initialization. 3232 * 3233 * This interface must be called when you finish initializing the ListTreeView by addNode. 3234 * @since 10 3235 */ 3236 public buildDone() { 3237 this.listNodeDataSource.init(this.listNodeUtils); 3238 this.nodeIdList.sort((a, b) => a - b); 3239 } 3240 3241 public refreshNode(parentId: number, parentSubTitle: ResourceStr = '', 3242 currentSubtitle: ResourceStr = '') { 3243 this.listNodeDataSource.setNodeSubtitlePara(parentId, parentSubTitle, currentSubtitle); 3244 } 3245 } 3246} 3247 3248@Component 3249export struct TreeViewInner { 3250 @ObjectLink item: NodeInfo; 3251 listNodeDataSource: TreeView.ListNodeDataSource; 3252 @State columnWidth: number = 0; 3253 @State isFocused: boolean = false; 3254 @State index: number = -1; 3255 @State lastIndex: number = -1; 3256 @State count: number = 0; 3257 @BuilderParam private listTreeViewMenu: () => void = null; 3258 private readonly MAX_CN_LENGTH: number = 254; 3259 private readonly MAX_EN_LENGTH: number = 255; 3260 private readonly INITIAL_INVALID_VALUE = -1; 3261 private readonly MAX_TOUCH_DOWN_COUNT = 0; 3262 private isMultiPress: boolean = false; 3263 private touchDownCount: number = this.INITIAL_INVALID_VALUE; 3264 private appEventBus: TreeView.TreeListener = TreeView.TreeListenerManager.getInstance().getTreeListener(); 3265 private readonly itemPadding: { left: Resource, right: Resource, top: Resource, bottom: string | Resource } = 3266 { left: $r('sys.float.ohos_id_card_margin_start'), 3267 right: $r('sys.float.ohos_id_card_margin_end'), 3268 top: $r('sys.float.ohos_id_text_margin_vertical'), 3269 bottom: '0vp' }; 3270 /* { left: '12vp', right: '12vp', top: '2vp', bottom: '0vp' } */ 3271 private readonly textInputPadding: { left : string, right: string, top: string, bottom: string } = 3272 { left : '0vp', right: '0vp', top: '0vp', bottom: '0vp'} 3273 3274 aboutToAppear(): void { 3275 if (this.item.getNodeItem().imageNode) { 3276 this.item.imageSource = this.item.getNodeItem().imageNode.source 3277 } 3278 } 3279 3280 private checkInvalidPattern(title: string): boolean { 3281 let pattern = /[\\\/:*?"<>|]/; 3282 return pattern.test(title) 3283 } 3284 3285 private checkIsAllCN(title: string): boolean { 3286 let pattern = /^[\u4e00-\u9fa5]+$/; 3287 return pattern.test(title) 3288 } 3289 3290 @Builder popupForShowTitle(text: string | Resource, backgroundColor: Resource, fontColor: Resource ) { 3291 Row() { 3292 Text(text).fontSize($r('sys.float.ohos_id_text_size_body2')).fontWeight('regular').fontColor(fontColor) 3293 }.backgroundColor(backgroundColor) 3294 .border({radius: $r('sys.float.ohos_id_elements_margin_horizontal_l')}) 3295 .padding({left: $r('sys.float.ohos_id_elements_margin_horizontal_l'), 3296 right: $r('sys.float.ohos_id_elements_margin_horizontal_l'), 3297 top: $r('sys.float.ohos_id_card_margin_middle'), 3298 bottom: $r('sys.float.ohos_id_card_margin_middle')}) 3299 } 3300 3301 @Builder builder() { 3302 if (this.listTreeViewMenu) { 3303 this.listTreeViewMenu() 3304 } 3305 } 3306 3307 build(){ 3308 if (this.item.getNodeIsShow()) { 3309 Stack() { 3310 Column() { 3311 Stack({alignContent: Alignment.Bottom}) { 3312 Divider() 3313 .height(this.listNodeDataSource.getFlagLine().flagLineHeight) 3314 .color(this.listNodeDataSource.getFlagLine().flagLineColor) 3315 .visibility(this.listNodeDataSource.getVisibility(this.item)) 3316 .lineCap(LineCapStyle.Round) 3317 .margin({ left: this.item.getFlagLineLeftMargin() }) 3318 .focusable(true) 3319 Row({}) { 3320 Row({}) { 3321 if (this.item.getNodeItem().imageNode) { 3322 Row() { 3323 Image(this.item.imageSource) 3324 .objectFit(ImageFit.Contain) 3325 .height(this.item.getNodeItem().imageNode.itemHeight) 3326 .width(this.item.getNodeItem().imageNode.itemWidth) 3327 .opacity(!this.item.getIsSelected() && !this.item.getIsHighLight() ? 3328 this.item.getNodeItem().imageNode.opacity : this.item.getNodeItem().imageNode.noOpacity) 3329 .focusable(this.item.getNodeItem().mainTitleNode != null ? false : true) 3330 } 3331 .focusable(true) 3332 .backgroundColor(COLOR_IMAGE_ROW) 3333 .margin({ right: this.item.getNodeItem().imageNode.itemRightMargin }) 3334 .height(this.item.getNodeItem().imageNode.itemHeight) 3335 .width(this.item.getNodeItem().imageNode.itemWidth) 3336 } 3337 Row() { 3338 if (this.item.getNodeItem().mainTitleNode && this.item.getIsShowTitle()) { 3339 Text(this.item.getNodeItem().mainTitleNode.title) 3340 .maxLines(1) // max line 3341 .fontSize(this.item.getNodeItem().mainTitleNode.size) 3342 .fontColor(this.item.fontColor) 3343 .margin({ right: this.item.getNodeItem().mainTitleNode.itemRightMargin }) 3344 .textOverflow({ overflow: TextOverflow.Ellipsis }) 3345 .fontWeight(this.item.getNodeItem().mainTitleNode.weight) 3346 .focusable(true) 3347 } 3348 if (this.item.getNodeItem().mainTitleNode && 3349 this.item.getNodeItem().inputText && 3350 this.item.getIsShowInputText()) { 3351 Row() { 3352 TextInput({ text: this.item.getNodeItem().mainTitleNode.title }) 3353 .height(this.item.getNodeItem().inputText.itemHeight) 3354 .fontSize(this.item.getNodeItem().inputText.size) 3355 .fontColor(this.item.getNodeItem().inputText.color) 3356 .borderRadius(this.item.getNodeItem().inputText.borderRadius) 3357 .backgroundColor(this.item.getNodeItem().inputText.backgroundColor) 3358 .enterKeyType(EnterKeyType.Done) 3359 .focusable(true) 3360 .padding({ left: this.textInputPadding.left, right: this.textInputPadding.right, 3361 top: this.textInputPadding.top, bottom: this.textInputPadding.bottom }) 3362 .onChange((value: string) => { 3363 let that = this; 3364 var thisIndex = findCurrentNodeIndex.call(that, this.item.getNodeCurrentNodeId()); 3365 let res: string = ''; 3366 let isInvalidError: boolean = false; 3367 let isLengthError: boolean = false; 3368 if (that.checkInvalidPattern(value)) { 3369 for (let i = 0; i < value.length; i++) { 3370 if (!that.checkInvalidPattern(value[i])) { 3371 res += value[i]; 3372 } 3373 } 3374 isInvalidError = true; 3375 that.listNodeDataSource.setPopUpInfo(PopUpType.WARNINGS, 3376 InputError.INVALID_ERROR, true, thisIndex); 3377 } 3378 else { 3379 res = value; 3380 isInvalidError = false; 3381 } 3382 if ((that.checkIsAllCN(res) && res.length > this.MAX_CN_LENGTH) || 3383 (!that.checkIsAllCN(res) && res.length > this.MAX_EN_LENGTH)) { 3384 res = that.checkIsAllCN(res) ? 3385 res.substr(0, this.MAX_CN_LENGTH) : res.substr(0, this.MAX_EN_LENGTH); 3386 isLengthError = true; 3387 that.listNodeDataSource.setPopUpInfo(PopUpType.WARNINGS, 3388 InputError.LENGTH_ERROR, true, thisIndex); 3389 } 3390 else { 3391 isLengthError = false; 3392 } 3393 if (!isLengthError && !isInvalidError) { 3394 that.listNodeDataSource.setMainTitleNameOnEdit(thisIndex, res); 3395 } 3396 }) 3397 .onSubmit((enterKey: EnterKeyType) => { 3398 let that = this; 3399 var thisIndex = findCurrentNodeIndex.call(that, this.item.getNodeCurrentNodeId()); 3400 that.listNodeDataSource.setPopUpInfo(PopUpType.WARNINGS, InputError.NONE, false, thisIndex); 3401 that.listNodeDataSource.setItemVisibilityOnEdit(thisIndex, MenuOperation.COMMIT_NODE); 3402 }) 3403 }.backgroundColor(this.item.getNodeItem().inputText.backgroundColor) 3404 .borderRadius(this.item.getNodeItem().inputText.borderRadius) 3405 .margin({ right: this.item.getNodeItem().inputText.itemRightMargin }) 3406 } 3407 Blank() 3408 } 3409 .layoutWeight(1) 3410 .focusable(true) 3411 3412 if (this.listNodeDataSource.hasSubtitle(this.item.getCurrentNodeId())) { 3413 Row() { 3414 Text(this.listNodeDataSource.getSubtitle(this.item.getCurrentNodeId())) 3415 .fontSize(this.listNodeDataSource.getSubTitlePara().fontSize) 3416 .fontColor(this.listNodeDataSource.getSubTitleFontColor(this.item.getIsHighLight() || this.item.getIsModify())) 3417 .fontWeight(this.listNodeDataSource.getSubTitlePara().fontWeight) 3418 } 3419 .focusable(true) 3420 .margin({ left: this.listNodeDataSource.getSubTitlePara().margin.left, 3421 right: this.item.getNodeItem().imageCollapse ? 3422 0 : this.listNodeDataSource.getSubTitlePara().margin.right }) 3423 } 3424 3425 if (this.item.getNodeItem().imageCollapse) { 3426 Row() { 3427 Image(this.item.getNodeItem().imageCollapse.collapseSource) 3428 .fillColor(this.item.getNodeItem().imageCollapse.isCollapse ? COLOR_IMAGE_ROW : COLOR_IMAGE_EDIT) 3429 .align(Alignment.End) 3430 .objectFit(ImageFit.Contain) 3431 .height(this.item.getNodeItem().imageCollapse.itemHeight) 3432 .width(this.item.getNodeItem().imageCollapse.itemWidth) 3433 .opacity(!this.item.getIsHighLight() ? 3434 this.item.getNodeItem().imageCollapse.opacity : this.item.getNodeItem().imageCollapse.noOpacity) 3435 .onTouch((event: TouchEvent) => { 3436 if (event.type === TouchType.Down) { 3437 let that = this; 3438 that.listNodeDataSource.expandAndCollapseNode(findCurrentNodeIndex.call(that, this.item.getNodeCurrentNodeId())); 3439 this.listNodeDataSource.setCurrentFocusNodeId(this.item.getCurrentNodeId()); 3440 } 3441 event.stopPropagation(); 3442 }) 3443 .focusable(true) 3444 } 3445 .focusable(true) 3446 .height(this.item.getNodeItem().imageCollapse.itemHeight) 3447 .width(this.item.getNodeItem().imageCollapse.itemWidth) 3448 } 3449 } 3450 .focusable(true) 3451 .width('100%') 3452 .gesture( 3453 TapGesture({ count: 2 }) // doubleClick 3454 .onAction((event: GestureEvent) => { 3455 let that = this; 3456 that.listNodeDataSource.expandAndCollapseNode( 3457 findCurrentNodeIndex.call(that, this.item.getNodeCurrentNodeId())); 3458 }) 3459 ) 3460 .height(this.item.getNodeHeight()) 3461 .padding({ left: this.item.getNodeLeftPadding() }) 3462 .bindContextMenu(this.builder, ResponseType.RightClick) 3463 }.focusable(true) 3464 }.focusable(true) 3465 } 3466 .opacity(this.listNodeDataSource.getListItemOpacity(this.item)) 3467 .onHover((isHover: boolean) => { 3468 if (isHover) { 3469 this.item.setNodeColor($r('sys.color.ohos_id_color_hover')) 3470 } else { 3471 this.item.setNodeColor($r('sys.color.ohos_id_color_background_transparent')) 3472 } 3473 }) 3474 .onTouch((event) => { 3475 this.count++; 3476 if(this.count > 1) { 3477 this.count--; 3478 return; 3479 } 3480 3481 this.listNodeDataSource.setClickIndex(this.index); 3482 let currentId = this.item.getNodeCurrentNodeId(); 3483 3484 if (event.type === TouchType.Down) { 3485 this.item.setNodeColor($r('sys.color.ohos_id_color_click_effect')); 3486 } 3487 3488 else if (event.type === TouchType.Up) { 3489 this.item.setNodeColor(COLOR_SELECT); 3490 this.item.fontColor = $r('sys.color.ohos_id_color_text_primary_activated'); 3491 if (this.item.getNodeItem().imageNode != null) { 3492 this.item.getNodeItem().imageNode.setImageSource(InteractionStatus.Selected); 3493 this.listNodeDataSource.setImageSource(this.index, InteractionStatus.Selected); 3494 this.item.imageSource = this.item.getNodeItem().imageNode.source; 3495 } 3496 3497 this.item.getNodeItem().mainTitleNode.setMainTitleSelected(true) 3498 this.lastIndex = this.index; 3499 3500 let callbackParam = { currentNodeId: currentId } 3501 this.appEventBus.emit(TreeView.TreeListenType.NODE_CLICK, [callbackParam]); 3502 } 3503 3504 if (this.listNodeDataSource.getLastIndex() !== -1 && this.index !== this.listNodeDataSource.getLastIndex()) { 3505 this.listNodeDataSource.setPopUpInfo(PopUpType.WARNINGS, InputError.NONE, false, this.listNodeDataSource.getLastIndex()); 3506 this.listNodeDataSource.setItemVisibilityOnEdit(this.listNodeDataSource.getLastIndex(), MenuOperation.COMMIT_NODE); 3507 } 3508 this.count--; 3509 }) 3510 /* backgroundColor when editing and in other states. */ 3511 .backgroundColor((this.item.getNodeItem().mainTitleNode && this.item.getNodeItem().inputText && 3512 this.item.getIsShowInputText()) ? this.item.getNodeItem().inputText.editColor : this.item.getNodeColor()) 3513 .border({ 3514 width: this.item.getNodeBorder().borderWidth, 3515 color: this.item.getNodeBorder().borderColor, 3516 radius: this.item.getNodeBorder().borderRadius 3517 }) 3518 .height(LIST_ITEM_HEIGHT) 3519 .focusable(true) 3520 .onMouse((event: MouseEvent) => { 3521 let that = this; 3522 let thisIndex = findCurrentNodeIndex.call(that, this.item.getNodeCurrentNodeId()); 3523 if (event.button == MouseButton.Right) { 3524 that.listNodeDataSource.handleEvent(Event.MOUSE_BUTTON_RIGHT, 3525 findCurrentNodeIndex.call(that, this.item.getNodeCurrentNodeId())); 3526 this.listTreeViewMenu = this.item.getMenu(); 3527 that.listNodeDataSource.setClickIndex(thisIndex); 3528 clearTimeout(this.item.getNodeItem().mainTitleNode.popUpTimeout); 3529 } 3530 event.stopPropagation(); 3531 }) 3532 .padding({ top: 0, bottom: 0 }) 3533 .bindPopup(this.item.getPopUpInfo().popUpIsShow, { 3534 builder: this.popupForShowTitle(this.item.getPopUpInfo().popUpText, this.item.getPopUpInfo().popUpColor, 3535 this.item.getPopUpInfo().popUpTextColor), 3536 placement: Placement.BottomLeft, 3537 placementOnTop: false, 3538 popupColor: this.item.getPopUpInfo().popUpColor, 3539 autoCancel: true, 3540 enableArrow: this.item.getPopUpInfo().popUpEnableArrow 3541 }) 3542 .onAreaChange((oldValue: Area, newValue: Area) => { 3543 let columnWidthNum = Number(parseInt(newValue.width.toString(), 0)) 3544 this.columnWidth = columnWidthNum 3545 }) 3546 } 3547 .stateStyles({ 3548 focused: { 3549 .border({ 3550 radius: $r('sys.float.ohos_id_corner_radius_clicked'), 3551 width: FLAG_NUMBER, 3552 color: $r('sys.color.ohos_id_color_focused_outline'), 3553 style: BorderStyle.Solid 3554 }) 3555 }, 3556 normal: { 3557 .border({ 3558 radius: $r('sys.float.ohos_id_corner_radius_clicked'), 3559 width: 0 }) 3560 } 3561 }) 3562 } 3563 } 3564}