1. 程式人生 > 程式設計 >在Angular中實現一個級聯效果的下拉框的示例程式碼

在Angular中實現一個級聯效果的下拉框的示例程式碼

實現一個具有級聯效果的下拉搜尋框,實現的結果如下圖所示

在Angular中實現一個級聯效果的下拉框的示例程式碼

我們主要通過這個元件,來學習一些細微的邏輯,比如: 如何計算input框內文字的長度; 如何獲取游標的位置;如何實現滾動條隨著上下鍵盤的按動進行移動......

具體需求如下

  1. 級聯搜尋最多不超過三級,以”.“作為級聯搜尋的連線符
  2. 搜尋框跟著文字框中的”.“進行向後移動,向右移動的最大距離不能超過文字框的寬度
  3. 當用戶修改之前的級聯內容,則不進行搜尋,並隱藏搜尋框;若使用者在之前輸入的是”.“,則將此”.“之後的內容全部刪除並搜尋當前的相關內容

接下來我們根據需求,來寫我們的邏輯

首先我們搭建html頁面

   <input
    #targetInput
    autocomplete="off"
    nz-input
    [(ngModel)]="searchValue"
    (keydown)="handlePress($event)"
    (input)="handleSearchList()"/>
   
   <div #searchList class="search-popup" [hidden]="!visible" (keyDown)="onKeydown($event)">
     <nz-spin [nzSpinning]="searchLoading" [class.spinning-height]="searchLoading">
      <div class="data-box" *ngIf="searchData && searchData.length !== 0">
       <ul>
       // 這裡在上篇文章中已經講解過,如何實現讓匹配的文字高亮顯示~
        <li
         id="item"
         *ngFor="let item of searchData;let i = index;"
         [class.item-selected]="curIndex === i"
         (mouseover)='hoverDataItem(i)'
         (click)="onSelectClick(item)">
         <span [innerHTML]="item | highlightSearchResult:searchValue | safe: 'html'"></span>
        </li>
       </ul>
      </div>
     </nz-spin>
   </div>
.search-popup {
 height: 376px;
 width: 246px;
 overflow-y: auto;
 box-shadow: 0 2px 8px rgba(0,.15);
 border-radius: 4px;
 position: absolute;
 background-color: #fff;
 z-index: 999;
 top: 92px;
 right: 61px;

 .data-box {
  margin: 0 10px;

  &:not(:last-child) {
   border-bottom: 1px solid #E4E5E7;
  }

  .no-search-data {
   display: inline-block;
   width: 100%;
   text-align: center;
   color: #C3C9D3;
   line-height: 40px;
  }
 }

 & ul {
  margin: 0 -10px;
  margin-bottom: 0;
  text-align: left;
 }

 & li {
  padding: 3px 10px;
  position: relative;
  list-style: none;
  height: 32px;
  line-height: 26px;
  &:hover {
   cursor: pointer;
   background-color: #e6f7ff;
  }
  &.item-selected {
   background-color: #E6F7FF;
  }
 }

 &.item-selected {
  background-color: #E6F7FF;

 }

.hidden-box {
 display: inline-block;
 border: 1px solid #ddd;
 visibility: hidden;
}

實現相關的邏輯

根據前兩個需求,我們需要根據文字框中的”.“進行向後移動,向右移動的最大距離不能超過文字框的寬度。

思路: 我們需要將文字框中的字串根據”.“來轉換成陣列,並且要想辦法獲取文字框中文字的長度。
如何獲取文字框中文字的長度呢?
我們可以將文字的內容,重新放到一個display: inline-block的div容器中,然後獲取容器的寬度,如下程式碼所示~

// html
 <!-- 用於測量input框的文字寬度 -->
 <div class="hidden-box" #firstLevel></div> // 以”.“轉化的陣列,下標為0的內容的寬度
 <div class="hidden-box" #secondLevel></div> // 以”.“轉化的陣列,下標為1的內容的寬度
 <div class="hidden-box" #allLevel></div> // 整個文字框的文字的寬度
 
 // ts
 import { ElementRef,Renderer2 } from '@angular/core';
 
 export class SearchListComponent {
  @ViewChild('searchList',{ static: true }) public searchList: ElementRef;
  @ViewChild('firstLevel',{ static: true }) public firstLevel: ElementRef;
  @ViewChild('secondLevel',{ static: true }) public secondLevel: ElementRef;
  @ViewChild('allLevel',{ static: true }) public allLevel: ElementRef;
  constructor(private _renderer: Renderer2) {}
     
  public setSearchPosition(rightValue: string): void {
    this._renderer.setStyle(
     this.searchList.nativeElement,'right',rightValue);
   } 
   
  public setSearchListPosition(targetValue: string): void {
  const inputWidth = 217;
  const defaultRightPosition = 60;
  const maxRightPosition = -148;
  const firstLevel = this.firstLevel.nativeElement;
  const secondLevel = this.secondLevel.nativeElement;
  const allLevel = this.allLevel.nativeElement;
  const targetValueArr = targetValue ? targetValue.split('.') : [];

  // 將input中的內容,根據”.“轉換成陣列之後,將相關的內容賦值到新的div容器中,為了便於獲取文字的寬度
  allLevel.innerHTML = targetValue;
  firstLevel.innerHTML = targetValueArr && targetValueArr[0];
  secondLevel.innerHTML = targetValueArr && targetValueArr.length > 1 ? targetValueArr[1] : '';

  // 得到相關寬度之後,實現下拉框移動的邏輯
  if (firstLevel.offsetWidth >= inputWidth
   || (firstLevel.offsetWidth + secondLevel.offsetWidth) >= inputWidth
   || allLevel.offsetWidth >= inputWidth) {
    this.setSearchPosition(this._renderer,this.searchList,`${maxRightPosition}px`);
   } else if (targetValueArr.length <= 1) {
   this.setSearchPosition(this.renderer,'61px');
  } else if (targetValueArr.length <= 2) {
   this.setSearchPosition(this.renderer,`${defaultRightPosition - firstLevel.offsetWidth}px`);
  } else if (targetValueArr.length <= 3) {
   this.setSearchPosition(renderer,`${defaultRightPosition - firstLevel.offsetWidth - secondLevel.offsetWidth}px`);
  }
 }
 }

到這裡,我們可以完成第一和第二個需求,我們再來看看第三個需求: 主要是根據使用者輸入的位置以及修改的內容,來決定是否顯示搜尋和顯示下拉框,如果使用者輸入的不是”.“我們則不顯示,如果使用者在之前的級聯中輸入”.“我們就需要進行再次幫使用者搜尋結果。

思路: 要想完成需求三,我們需要知道使用者到底是在哪裡操作,即我們要是可以知道游標的位置就更完美了~

 // 獲取游標的位置
 public getCursorPosition(element: HTMLInputElement): number {
  let cursorPosition = 0;
  if (element.selectionStart || element.selectionStart === 0) {
   cursorPosition = element.selectionStart;
  }
  return cursorPosition;
 }
 
 // 用來獲取使用者輸入的內容是什麼
 public handlePress(event: KeyboardEvent): void {
   this.curPressKey = event.key;
  }

 // 使用者input的時候呼叫的核心方法
 public handleSearchList(value: string): void {
  this.curIndex = 0;
  const cursorPosition = this.getCursorPosition(this.targetInput.nativeElement); // 獲取游標位置
  let targetValue = value;
  const targetValueArr = targetValue ? targetValue.split('.') : [];
  const valueArrLength = targetValueArr.length;
  this.setSearchListPosition(targetValue); // 調整位置
  // 判斷那些情況下應該搜尋並顯示下拉框
  if (valueArrLength === 1
   || valueArrLength === 2 && cursorPosition >= targetValueArr[0].length + 1
   || valueArrLength === 3 && cursorPosition >= targetValueArr[0].length + targetValueArr[1].length + 2) {
    this.searchLoading = true;
    this.visible = true;
    ...獲取下拉框中的資料
  } else {
   this.hidePopup();
  }
 }

最後為了更好的提高用的體驗,我們還需要讓下拉框支援鍵盤事件哦~方法也很簡單,如下所示

 public onKeydown(keyDownInfo: {index: number,code: number,e: KeyboardEvent}): void {
  const { code,e } = keyDownInfo;
  e.stopPropagation();
  if (code === 38) { // 鍵盤上
   e.preventDefault(); // 防止游標由最後邊移動到前邊,只是在開發中遇到的一點體驗上小問題
   if (this.curIndex > 0) {
    this.curIndex--;
   }
  } else if (code === 40) { // 鍵盤下
   if (this.curIndex < this.searchData.length - 1) {
    this.curIndex++;
   }
  } else if (code === 13) {  // 回車,即相當於使用者點選
   this.ruleModal.showModal();
   const curData = this.searchData[this.curIndex];
   if (curData) {
    this.onSelectClick(curData);
   }
  }
  // 實現下拉框的滾動條隨著鍵盤的上下鍵按動時一起移動
  const lis = document.querySelectorAll('#item');
  const curLiEle = lis[this.curIndex] as HTMLElement;
  const searchList = this.searchList.nativeElement;
  const liEleHeight = 32;
  //(當前選中li標籤的offsetTop + li標籤自身的高度 - 下拉框的高度)
  searchList.scrollTop = curLiEle.offsetTop + liEleHeight - searchList.clientHeight;
 }

總結

其實這個級聯搜尋的元件,他的通用性可能並不是很強,但是在實現的過程中,一些細節邏輯的通用性還是比較強的~ 希望這些細節可以給同在開發中的你帶來一些幫助~❤

到此這篇關於在Angular中實現一個級聯效果的下拉框的示例程式碼的文章就介紹到這了,更多相關Angular級聯下拉框內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!