@ohos.arkui.dragController (DragController)

本模块提供发起主动拖拽的能力,当应用接收到触摸或长按等事件时可以主动发起拖拽的动作,并在其中携带拖拽信息。

说明:

本模块首批接口从 API version 10 开始支持。后续版本的新增接口,采用上角标单独标记接口的起始版本。 本模块功能依赖UI的执行上下文,不可在UI上下文不明确的地方使用,参见UIContext说明。 从API version 11开始,可以通过使用UIContext中的getDragController方法获取当前UI上下文关联的DragController对象。 示例效果请以真机运行为准,当前 IDE 预览器不支持。

导入模块

import dragController from "@ohos.arkui.dragController";

dragController.executeDrag

executeDrag(custom: CustomBuilder | DragItemInfo, dragInfo: DragInfo, callback: AsyncCallback< {event: DragEvent, extraParams: string}>): void

主动发起拖拽能力,传入拖拽发起后跟手效果所拖拽的对象以及携带拖拽信息。通过回调返回结果。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
custom CustomBuilder | DragItemInfo 拖拽发起后跟手效果所拖拽的对象。
说明:
不支持全局builder。如果builder中使用了Image组件,应尽量开启同步加载,即配置Image的syncLoad为true。该builder只用于生成当次拖拽中显示的图片,builder的修改不会同步到当前正在拖拽的图片,对builder的修改需要在下一次拖拽时生效。
dragInfo DragInfo 拖拽信息。
callback AsyncCallback<{event: DragEvent, extraParams: string}> 拖拽结束返回结果的回调
- event:拖拽事件信息,仅包括拖拽结果。
- extraParams:拖拽事件额外信息。

错误码:

错误码ID 错误信息
401 if the parameters checking failed.
100001 if some internal handling failed.

示例:

import dragController from "@ohos.arkui.dragController"
import UDC from '@ohos.data.unifiedDataChannel';

@Entry
@Component
struct DragControllerPage {
  @State text: string = ''

  @Builder DraggingBuilder() {
    Column() {
      Text("DraggingBuilder")
        .fontColor(Color.White)
        .fontSize(12)
    }
    .width(100)
    .height(100)
    .backgroundColor(Color.Blue)
  }

  build() {
    Column() {
      Button('touch to execute drag')
        .margin(10)
        .onTouch((event?:TouchEvent) => {
          if(event){
            if (event.type == TouchType.Down) {
              let text = new UDC.PlainText()
              text.textContent = 'drag text'
              text.abstract = 'abstract'
              let unifiedData = new UDC.UnifiedData(text)

              let dragInfo: dragController.DragInfo = {
                pointerId: 0,
                data: unifiedData,
                extraParams: ''
              }
              class tmp{
                event:DragEvent|undefined = undefined
                extraParams:string = ''
              }
              let eve:tmp = new tmp()
              dragController.executeDrag(()=>{this.DraggingBuilder()}, dragInfo, (err, eve) => {
                if(eve.event){
                  if (eve.event.getResult() == DragResult.DRAG_SUCCESSFUL) {
                    // ...
                  } else if (eve.event.getResult() == DragResult.DRAG_FAILED) {
                    // ...
                  }
                }
              })
            }
          }
        })
      Text(this.text)
        .height(100)
        .width(150)
        .margin({top:20})
        .border({color:Color.Black,width:1})
        .onDrop((dragEvent?:DragEvent)=>{
          if(dragEvent){
            let records: Array<UDC.UnifiedRecord> = dragEvent.getData().getRecords();
            let plainText: UDC.PlainText = records[0] as UDC.PlainText;
            this.text = plainText.textContent;
          }
        })
    }
    .width('100%')
    .height('100%')
  }
}

zh-cn_executeDrag1

dragController.executeDrag

executeDrag(custom: CustomBuilder | DragItemInfo, dragInfo: DragInfo): Promise<{event: DragEvent, extraParams: string}>

主动发起拖拽能力,传入拖拽发起后跟手效果所拖拽的对象以及携带拖拽信息。通过Promise返回结果。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
custom CustomBuilder | DragItemInfo 拖拽发起后跟手效果所拖拽的对象。
dragInfo DragInfo 拖拽信息。

返回值:

类型 说明
Promise<{event: DragEvent, extraParams: string}> 拖拽结束返回结果的回调
- event:拖拽事件信息,仅包括拖拽结果。
- extraParams:拖拽事件额外信息。

错误码:

错误码ID 错误信息
401 if the parameters checking failed.
100001 if some internal handling failed.

示例:

import dragController from "@ohos.arkui.dragController"
import componentSnapshot from '@ohos.arkui.componentSnapshot';
import image from '@ohos.multimedia.image';
import UDC from '@ohos.data.unifiedDataChannel';

@Entry
@Component
struct DragControllerPage {
  @State pixmap: image.PixelMap|undefined = undefined
  @State text: string = ''

  @Builder DraggingBuilder() {
    Column() {
      Text("DraggingBuilder")
        .fontColor(Color.White)
    }
    .width(100)
    .height(100)
    .backgroundColor(Color.Blue)
  }

  @Builder PixmapBuilder() {
    Column() {
      Text("PixmapBuilder")
        .fontColor(Color.White)
        .fontSize(15)
    }
    .width(100)
    .height(100)
    .backgroundColor(Color.Blue)
  }

  aboutToAppear() {
    let pb: CustomBuilder = (): void => {
      this.PixmapBuilder()
    }
    componentSnapshot.createFromBuilder(pb).then((pix: image.PixelMap) => {
      this.pixmap = pix;
    })
  }

  build() {
    Column() {
      Button('touch to execute drag')
        .margin(10)
        .onTouch((event?:TouchEvent) => {
          if(event){
            if (event.type == TouchType.Down) {
              let text = new UDC.PlainText()
              text.textContent = 'drag text'
              text.abstract = 'abstract'
              let unifiedData = new UDC.UnifiedData(text)

              let dragInfo: dragController.DragInfo = {
                pointerId: 0,
                data: unifiedData,
                extraParams: ''
              }
              let dragItemInfo: DragItemInfo = {
                pixelMap: this.pixmap,
                builder: ()=>{this.DraggingBuilder()},
                extraInfo: "DragItemInfoTest"
              }

              class tmp{
                event:DragResult|undefined = undefined
                extraParams:string = ''
              }
              let eve:tmp = new tmp()
              dragController.executeDrag(dragItemInfo, dragInfo)
                .then((eve) => {
                  if (eve.event.getResult() == DragResult.DRAG_SUCCESSFUL) {
                    // ...
                  } else if (eve.event.getResult() == DragResult.DRAG_FAILED) {
                    // ...
                  }
                })
                .catch((err:Error) => {
                })
            }
          }
        })
      Text(this.text)
        .height(100)
        .width(150)
        .margin({top:20})
        .border({color:Color.Black,width:1})
        .onDrop((dragEvent?:DragEvent)=>{
          if(dragEvent){
            let records: Array<UDC.UnifiedRecord> = dragEvent.getData().getRecords();
            let plainText: UDC.PlainText = records[0] as UDC.PlainText;
            this.text = plainText.textContent;
          }
        })
    }
    .width('100%')
    .height('100%')
  }
}

zh-cn_executeDrag2

DragInfo

系统能力: SystemCapability.ArkUI.ArkUI.Full

发起拖拽所需要的属性和拖拽时携带的信息。

名称 类型 必填 说明
pointerId number 设置启动拖拽时屏幕上触摸点的Id。
data unifiedDataChannel.UnifiedData 设置拖拽过程中携带的数据。
extraParams string 设置拖拽事件额外信息,具体功能暂未实现。
touchPoint11+ TouchPoint 配置跟手点坐标,不配置时,默认居中。
previewOptions11+ DragPreviewOptions 拖拽背板自定义配置。

dragController.createDragAction11+

createDragAction(customArray: Array<CustomBuilder | DragItemInfo>, dragInfo: DragInfo): DragAction

创建拖拽的Action对象,需要显式指定拖拽背板图(可多个),以及拖拽的数据,跟手点等信息;当通过一个已创建的 Action 对象发起的拖拽未结束时,无法再次创建新的 Action 对象,接口会抛出异常。

说明: 建议控制传递的拖拽背板数量,传递过多容易导致拖起的效率问题。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
customArray Array<CustomBuilder | DragItemInfo> 拖拽发起后跟手效果所拖拽的对象。
dragInfo DragInfo 拖拽信息。

返回值:

类型 说明
DragAction 创建拖拽Action对象,主要用于后面实现注册监听拖拽状态改变事件和启动拖拽服务。

错误码:

错误码ID 错误信息
401 If the parameters checking failed
100001 If some internal handling failed

示例:

import dragController from "@ohos.arkui.dragController"
import componentSnapshot from '@ohos.arkui.componentSnapshot';
import image from '@ohos.multimedia.image';
import UDC from '@ohos.data.unifiedDataChannel';

@Entry
@Component
struct DragControllerPage {
  @State pixmap: image.PixelMap | null = null
  @State text: string = ''
  private dragAction: dragController.DragAction | null = null;
  customBuilders:Array<CustomBuilder | DragItemInfo> = new Array<CustomBuilder | DragItemInfo>();
  @Builder DraggingBuilder() {
    Column() {
      Text("DraggingBuilder")
        .fontColor(Color.White)
        .fontSize(12)
    }
    .width(100)
    .height(100)
    .backgroundColor(Color.Blue)
  }

  build() {
    Column() {

      Column() {
        Text(this.text)
          .width('100%')
          .height('100%')
          .fontColor(Color.White)
          .fontSize(18)
          .onDrop((dragEvent?:DragEvent)=>{
            if(dragEvent){
              let records: Array<UDC.UnifiedRecord> = dragEvent.getData().getRecords();
              let plainText: UDC.PlainText = records[0] as UDC.PlainText;
              this.text = plainText.textContent;
            }
          })
      }
      .width(100)
      .height(100)
      .backgroundColor(Color.Red)
      .margin(10)

      Button('多对象dragAction customBuilder拖拽').onTouch((event?:TouchEvent) => {
        if(event){
          if (event.type == TouchType.Down) {
            console.log("muti drag Down by listener");
            this.customBuilders.splice(0, this.customBuilders.length);
            this.customBuilders.push(()=>{this.DraggingBuilder()});
            this.customBuilders.push(()=>{this.DraggingBuilder()});
            this.customBuilders.push(()=>{this.DraggingBuilder()});
            let text = new UDC.PlainText()
            text.textContent = 'drag text'
            let unifiedData = new UDC.UnifiedData(text)
            let dragInfo: dragController.DragInfo = {
              pointerId: 0,
              data: unifiedData,
              extraParams: ''
            }
            try{
              this.dragAction = dragController.createDragAction(this.customBuilders, dragInfo)
              if(!this.dragAction){
                console.log("listener dragAction is null");
                return
              }
              this.dragAction.on('statusChange', (dragAndDropInfo: dragController.DragAndDropInfo)=>{
                if (dragAndDropInfo.status == dragController.DragStatus.STARTED) {
                  console.log("drag has start");
                } else if (dragAndDropInfo.status == dragController.DragStatus.ENDED){
                  console.log("drag has end");
                  if (!this.dragAction) {
                    return
                  }
                  this.dragAction.off('statusChange')
                }
              })
              this.dragAction.startDrag().then(()=>{}).catch((err:Error)=>{
                console.log("start drag Error:" + err.message);
              })
            } catch(err) {
              console.log("create dragAction Error:" + err.message);
            }
          }
        }
      }).margin({top:20})
    }
  }
}

zh-cn_executeDrag3

DragAction11+

监听状态改变,启动拖拽服务的对象。

系统能力: SystemCapability.ArkUI.ArkUI.Full

startDrag11+

startDrag(): Promise<void>

启动拖拽服务,返回Promise对象,回调启动成功和失败的结果。

系统能力: SystemCapability.ArkUI.ArkUI.Full

错误码:

错误码ID 错误信息
100001 if some internal handling failed.

示例:

import dragController from "@ohos.arkui.dragController"
import UDC from '@ohos.data.unifiedDataChannel';

@Entry
@Component
struct DragControllerPage {
  build() {
    Column() {
      Button('touch to execute drag')
        .onTouch((event?:TouchEvent) => {
          let customBuilders:Array<CustomBuilder | DragItemInfo> = new Array<CustomBuilder | DragItemInfo>();
          let text = new UDC.Text()
          let unifiedData = new UDC.UnifiedData(text)
          let dragInfo: dragController.DragInfo = {
            pointerId: 0,
            data: unifiedData,
            extraParams: ''
          }
          try {
            let dragAction: dragController.DragAction | null = dragController.createDragAction(customBuilders, dragInfo);
            if(!dragAction){
              console.log("listener dragAction is null");
              return
            }
            dragAction.startDrag().then(()=>{}).catch((err:Error)=>{
              console.log("start drag Error:" + err.message);
            })
          } catch (err) {
            console.log("create dragAction Error:" + err.message);
          }
        })
    }
  }
}

on('statusChange')11+

on(type: 'statusChange', callback: Callback<DragAndDropInfo>): void

注册监听拖拽状态改变事件。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
type string 监听事件,固定为'statusChange',即注册监听拖拽状态改变事件。
callback Callback<DragAndDropInfo> 回调函数,返回当前的DragAndDropInfo组件状态。

示例:

import dragController from "@ohos.arkui.dragController"
import UDC from '@ohos.data.unifiedDataChannel';

@Entry
@Component
struct DragControllerPage {
  build() {
    Column() {
      Button('touch to execute drag')
        .onTouch((event?:TouchEvent) => {
          let customBuilders:Array<CustomBuilder | DragItemInfo> = new Array<CustomBuilder | DragItemInfo>();
          let text = new UDC.Text()
          let unifiedData = new UDC.UnifiedData(text)
          let dragInfo: dragController.DragInfo = {
            pointerId: 0,
            data: unifiedData,
            extraParams: ''
          }
          try{
            let dragAction: dragController.DragAction | null = dragController.createDragAction(customBuilders, dragInfo);
            if(!dragAction){
              console.log("listener dragAction is null");
              return
            }
            dragAction.on('statusChange', (dragAndDropInfo: dragController.DragAndDropInfo)=>{
              console.info("Register to listen on drag status", JSON.stringify(dragAndDropInfo));
            })
          }catch(err) {
            console.log("create dragAction Error:" + err.message);
          }
        })
    }
  }
}

off('statusChange')11+

off(type: 'statusChange', callback?: Callback<DragAndDropInfo>): void

取消注册监听拖拽状态改变事件。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
type string 监听事件,固定为'statusChange',即取消监听拖拽状态改变事件。
callback Callback<DragAndDropInfo> 回调函数,返回当前的DragAndDropInfo组件状态, 不设置取消所有监听。

示例:

import dragController from "@ohos.arkui.dragController"
import UDC from '@ohos.data.unifiedDataChannel';

@Entry
@Component
struct DragControllerPage {
  build() {
    Column() {
      Button('touch to execute drag')
        .onTouch((event?:TouchEvent) => {
          let customBuilders:Array<CustomBuilder | DragItemInfo> = new Array<CustomBuilder | DragItemInfo>();
          let text = new UDC.Text()
          let unifiedData = new UDC.UnifiedData(text)
          let dragInfo: dragController.DragInfo = {
            pointerId: 0,
            data: unifiedData,
            extraParams: ''
          }
          try{
            let dragAction: dragController.DragAction | null = dragController.createDragAction(customBuilders, dragInfo);
            if(!dragAction){
              console.log("listener dragAction is null");
              return
            }
            dragAction.off('statusChange', (dragAndDropInfo: dragController.DragAndDropInfo)=>{
              console.info("Cancel listening on drag status", JSON.stringify(dragAndDropInfo));
            })
          }catch(err) {
            console.log("create dragAction Error:" + err.message);
          }
        })
    }
  }
}

DragAndDropInfo11+

系统能力: SystemCapability.ArkUI.ArkUI.Full

拖拽过程中监听到status改变时上报的数据。

名称 类型 必填 说明
status DragStatus 当前拖拽状态(启动和结束)。
event DragEvent 当前状态所对应的拖拽事件。
extraParams string 设置拖拽事件额外信息,具体功能暂未实现。

DragStatus11+

系统能力: SystemCapability.ArkUI.ArkUI.Full

拖拽开始和结束状态。

名称 说明
STARTED 0 拖拽已成功发起。
ENDED 1 拖拽结束。

AnimationOptions11+

系统能力: SystemCapability.ArkUI.ArkUI.Full

发起拖拽所需要的属性和拖拽时携带的信息。

名称 类型 必填 说明
duration number 动画持续时间,单位为毫秒。
默认值:1000
说明:
- 设置小于0的值时按0处理。
- 设置浮点型类型的值时,向下取整。例如,设置值为1.2,按照1处理。
curve  Curve | ICurve 设置动画曲线。
默认值:Curve.EaseInOut

dragController.getDragPreview11+

getDragPreview(): DragPreview

返回一个代表拖拽背板的对象。

系统能力: SystemCapability.ArkUI.ArkUI.Full

返回值:

类型 说明
DragPreview 一个代表拖拽背板的对象,提供背板样式设置的接口,在OnDrop和OnDragEnd回调中使用不生效。

示例:

请参考animate

DragPreview11+

拖拽背板的对象,在OnDrop和OnDragEnd回调中使用不生效。

系统能力: SystemCapability.ArkUI.ArkUI.Full

setForegroundColor11+

setForegroundColor(color: ResourceColor): void

设置背板蒙版颜色,在OnDrop和OnDragEnd回调中使用不生效,仅支持通过 getDragPreview() 方法获取到的对象上使用。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
color ResourceColor 背板蒙版颜色。

示例:

请参考animate

animate11+

animate(options: AnimationOptions, handler: () => void): void

设置背板蒙版颜色变化动效,在OnDrop和OnDragEnd回调中使用不生效,仅支持通过 getDragPreview() 方法获取到的对象上使用。

系统能力: SystemCapability.ArkUI.ArkUI.Full

参数:

参数名 类型 必填 说明
options AnimationOptions 动效参数
handler () => void 用于修改背板蒙版颜色等属性的回调方法。

示例:

1.在EntryAbility.ets中获取UI上下文并保存至LocalStorage中。

import AbilityConstant from '@ohos.app.ability.AbilityConstant';
import hilog from '@ohos.hilog';
import UIAbility from '@ohos.app.ability.UIAbility';
import Want from '@ohos.app.ability.Want';
import window from '@ohos.window';
import { UIContext } from '@ohos.arkui.UIContext';

let uiContext: UIContext;
let localStorage: LocalStorage = new LocalStorage('uiContext');

export default class EntryAbility extends UIAbility {
storage: LocalStorage = localStorage;

onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void {
  hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onCreate');
}

onDestroy(): void {
  hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onDestroy');
}

onWindowStageCreate(windowStage: window.WindowStage): void {
  hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageCreate');

  windowStage.loadContent('pages/Index', (err, data) => {
    if (err.code) {
      hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
      return;
    }
    hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s', JSON.stringify(data) ?? '');
    windowStage.getMainWindow((err, data) => {
      if (err.code) {
        hilog.error(0x0000, 'Failed to abtain the main window. Cause:' + err.message, '');
        return;
      }
      let windowClass: window.Window = data;
      uiContext = windowClass.getUIContext();
      this.storage.setOrCreate<UIContext>('uiContext', uiContext);
    })
  });
}
}

2.在Index.ets中通过LocalStorage.getShared()获取UI上下文,进而获取DragController对象实施后续操作。


import UDC from '@ohos.data.unifiedDataChannel';
import hilog from '@ohos.hilog';
import dragController from "@ohos.arkui.dragController";
import image from '@ohos.multimedia.image';
import curves from '@ohos.curves';
import { BusinessError } from '@ohos.base';
import { UIContext } from '@ohos.arkui.UIContext';
import promptAction from '@ohos.promptAction';


let storages = LocalStorage.getShared();

@Entry(storages)
@Component
struct DragControllerPage {
@State pixmap: image.PixelMap|null = null

@Builder DraggingBuilder() {
  Column() {
    Text("DraggingBuilder")
      .fontColor(Color.White)
      .fontSize(12)
  }
  .width(100)
  .height(100)
  .backgroundColor(Color.Blue)
}

@Builder PixmapBuilder() {
  Column() {
    Text("PixmapBuilder")
  }
  .width(100)
  .height(100)
  .backgroundColor(Color.Blue)
}

build() {
  Column() {
    Button('拖拽至此处')
      .margin(10)
      .onDragEnter(() => {
      try {
        let uiContext: UIContext = storages.get<UIContext>('uiContext') as UIContext;
        let previewObj: dragController.DragPreview = uiContext.getDragController().getDragPreview();
        let foregroundColor: ResourceColor = Color.Green;

        let previewAnimation: dragController.AnimationOptions = {
          curve: curves.cubicBezierCurve(0.2,0,0,1),
        }
        previewObj.animate(previewAnimation, () => {
          previewObj.setForegroundColor(foregroundColor);
        });
      } catch (error) {
        let msg = (error as BusinessError).message;
        let code = (error as BusinessError).code;
        hilog.error(0x0000, `show error code is ${code}, message is ${msg}`, '');
      }
    })
      .onDrop(() => {
        promptAction.showToast({duration: 100, message: 'Drag Success', bottom: 400})
      })
    Button('拖起').onTouch((event?:TouchEvent) => {
      if(event){
        if (event.type == TouchType.Down) {
          let text = new UDC.Text()
          let unifiedData = new UDC.UnifiedData(text)
          let dragInfo: dragController.DragInfo = {
            pointerId: 0,
            data: unifiedData,
            extraParams: ''
          }
          class tmp{
            event:DragEvent|undefined = undefined
            extraParams:string = ''
          }
          let eve:tmp = new tmp()
          dragController.executeDrag(() => {
            this.DraggingBuilder()
          }, dragInfo, (err , eve) => {
            hilog.info(0x0000, `ljx ${JSON.stringify(err)}`, '')
            if (eve && eve.event) {
              if (eve.event.getResult() == DragResult.DRAG_SUCCESSFUL) {
                hilog.info(0x0000, 'success', '');
              } else if (eve.event.getResult() == DragResult.DRAG_FAILED) {
                hilog.info(0x0000, 'failed', '');
              }
            }
          })
        }
      }
    }).margin({top:100})
  }
  .width('100%')
  .height('100%')
}
}

zh-cn_executeDrag5