amis源码 onEvent事件动作 和 Action行为按钮解析

Action行为按钮组件 (仅支持click事件)

Action行为按钮是针对click点击事件的一些处理。actionType指定action作用类型:ajax、link、url、drawer、dialog、confirm、cancel、prev、next、copy、close

amis配置:{

            "type": "button",

            "actionType": "clear",

            "label": "清空"

        }

自定义组件使用Action(button)行为按钮:

onMount: (dom, data, onChange, props) => {const button = document.createElement('button');button.innerText = '点击修改姓名';button.onclick = event => {onChange('new name', 'name');props.onAction(  //详见amis/src/renderes/Action.tsxevent,{type: 'action',label: '弹个框',actionType: 'dialog',dialog: {title: '弹框',body: 'Hello World!'}},{} // 这是 data);event.preventDefault();};dom.appendChild(button);
};

Action(button)行为按钮动作执行源码 :

amis/src/renderers/Action.tsx行为按钮组件

添加onEvent事件动作后,onEvent事件动作将先于旧版Action行为动作执行

@Renderer({ type: 'action' })export class ActionRenderer extends React.Component<ActionRendererProps> {doAction(action: ActionObject,args: {value?: string | {[key: string]: string};}) {const actionType = action?.actionType as any;if (actionType === 'click') {this.handleAction(actionType, action);}}async handleAction( e: React.MouseEvent<any> | string | void | null, action: any  ) {// 触发onEvent 事件动作执行const rendererEvent = await dispatchEvent(e as React.MouseEvent<any> | string,mergedData);// 阻止原有动作执行if (rendererEvent?.prevented) {return;}onAction(e, action, mergedData); //Action行为按钮动作执行(props.onAction是从RootRenderer.tsx中继承来的)}
}

Amis-core/src/RootRenderer.tsx(onAction方法的具体实现) :

handleAction封装的一些reload、url、dialog、ajax通用动作调用处理,并通过props分发(onAction: handleAction)下去。Action行为按钮组件会调用。

export class RootRenderer extends React.Component<RootRendererProps> {handleAction(e: React.UIEvent<any> | void,action: ActionObject,ctx: object,throwErrors: boolean = false,delegate?: IScopedContext): any {const {env, messages, onAction, mobileUI, render} = this.props;
const store = this.store;const scoped = delegate || (this.context as IScopedContext);if (action.actionType === 'reload') {  //...省略} else if (action.actionType === 'url' ||action.actionType === 'link' ||action.actionType === 'jump') {//...省略} else if (action.actionType === 'email') {//...省略} else if (action.actionType === 'dialog') {//...省略} else if (action.actionType === 'ajax') { //...省略}render() {return (<>{render(pathPrefix!, schema, {    ...rest,topStore: this.store,  //topStore是顶级store(RootStore树)data: this.store.downStream,  context: store.context,onAction: this.handleAction //onAction方法 封装的reload、url、link、jump等动作(Action行为按钮会使用此方法)}) as JSX.Element}</>);}
}

onEvent(配置事件动作 支持多种类型事件)

amis配置:   

 {type: 'button',onEvent: {click: {actions: [{actionType: 'toast',args: {msgType: 'info',msg: '派发点击事件'}}]}

onEvent事件分发源码:

//dispatchEvent分发事件,触发onEvent事件动作执行。

dispatchEvent( e, createObject(data, { nativeEvent: e }));

amis-core/src/utils/renderer-event.ts:

rendererEventListeners是一个集合,维护着所有onEvent 事件动作。

  // 过滤&排序const listeners = rendererEventListeners.filter((item: RendererEventListener) =>item.type === eventName &&(broadcast ? true : item.renderer === renderer)).sort((prev: RendererEventListener, next: RendererEventListener) =>next.weight - prev.weight
);runActions await runActions(listener.actions, listener.renderer, rendererEvent);

onEvent中actions执行源码 :

onEvent :amis-core/src/actions

1.Action.ts是基类定义了RendererAction、ListenerAction等基础接口。

runAction方法进行动作的通用预处理并执行不同type的action Class的run方法触发动作。核心代码如下

  await actionInstrance.run(…..)

runActions循环执行runAction(通过event参数在动作间传递参数),核心代码如下:

  for (const actionConfig of actions) {

       await runAction(actionInstrance, actionConfig, renderer, event);

  }

所以若配置了多个动作,动作是按顺序依此执行的。

2.CmptAction、AjaxAction等是各种类型的Action动作run方法的具体实现,均implements RendererAction(重写run方法) & extends ListenerAction(继承公共属性和方法)

比如:

2-1.CmptAction.ts:

是对setValue、reload和组件专属动作(comp.doAction())等动作的run方法实现

async run(  action: ICmptAction, renderer: ListenerContext, event: RendererEvent<any>) {/** 根据唯一ID查找指定组件, 触发组件未指定id或未指定响应组件componentId,则使用触发组件响应 */const key = action.componentId || action.componentName;let component = key && renderer.props.$schema[action.componentId ? 'id' : 'name'] !== key //指定了目标组件id/name 且 当前渲染器renderer组件id/name不是目标组件id/name? event.context.scoped?.[action.componentId ? 'getComponentById' : 'getComponentByName'](key): renderer;const dataMergeMode = action.dataMergeMode || 'merge';if (action.actionType === 'setValue') {const beforeSetData = renderer?.props?.env?.beforeSetData;const path = action.args?.path;/** 如果args中携带path参数, 则认为是全局变量赋值, 否则认为是组件变量赋值 */if ( path && typeof path === 'string' && beforeSetData &&  typeof beforeSetData === 'function') {const res = await beforeSetData(renderer, action, event);if (res === false) { return;  }}if (component?.setData) {return component?.setData(action.args?.value,dataMergeMode === 'override', action.args?.index);} else {return component?.props.onChange?.(action.args?.value);}
}// 执行组件动作try {const result = await component?.doAction?.(action,event.data,true,action.args);//...省略} catch(e) {   }}

2-2.CustomAction.ts:

自定义动作内置的doAction参数 本质还是调用runActions进行onEvent的动作调用: 

    // 执行自定义编排脚本let scriptFunc = action.args?.script ?? action.script;if (typeof scriptFunc === 'string') {scriptFunc = str2AsyncFunction(scriptFunc,'context','doAction','event') as any;}// 外部可以直接调用doAction来完成动作调用// 可以通过上下文直接编排动作调用,通过event来进行动作干预let result = await (scriptFunc as any)?.call(null,renderer,(action: ListenerAction) => runActions(action, renderer, event),event,action);

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/20106.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

翻译《The Old New Thing》- What a drag: Dragging a virtual file (IStream edition)

What a drag: Dragging a virtual file (IStream edition) - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20080319-00/?p23073 Raymond Chen 2008年03月19日 拖拽虚拟文件&#xff08;IStream 版本&#xff09; 上一次&#xff0c;我们看…

python | 类的实现

和实例有关的&#xff0c;通过对象名&#xff0c;打点调用 实例属性&#xff0c;实例方法 stuStudent("XiaoMing",18) print(stu.name) 类属性、静态方法和类方法都是通过类名直接调用 Student.name 静态方法和类方法都不能调用实例属性和实例方法 动态绑定 如果是函…

稳定性问题ANR-input

专栏简介 接上文【Android ANR简介】内容,深入探索input 类型的ANR问题产生原理,至于解决此类ANR的进阶内容会在下篇【稳定性问题ANR-input进阶】中详细介绍。 input ANR简介 Android app的input事件都是有主线程消费的,假设主线程有耗时函数执行,就会产生ANR问题;但是…

深入分析 Android Service (四)

文章目录 深入分析 Android Service (四)1. 使用 Messenger 进行通信2. 详细示例&#xff1a;使用 Messenger 进行通信2.1 创建 MessengerService2.2 在 Activity 中绑定服务并发送消息 3. 使用 AIDL 进行进程间通信3.1 定义 AIDL 接口3.2 实现 AIDL 接口3.3 在客户端绑定 AIDL…

C语言序列化和反序列化--TPL中的API(三)

tpl_map 创建tpl的唯一方法是调用tpl_map()。第一个参数是格式字符串。后面是格式字符串中特定字符所需的参数列表。例如, tpl_node *tn; int i; tn tpl_map( "A(i)", &i );该函数在格式字符串中的项和给定地址的C程序变量之间创建映射。稍后&#xff0c;C变量…

Apache Doris 基础 -- 数据表设计(分区分桶)

Versions: 2.1 本文档主要介绍了Doris的表创建和数据分区&#xff0c;以及表创建过程中可能遇到的问题和解决方案。 1、基本概念 在Doris中&#xff0c;数据以表的形式被逻辑地描述。 1.1 Row & Column 表由行和列组成: 行:表示用户数据的单行;列:用于描述一行数据中的…

自定义对象池BasePooledObjectFactory的使用

项目中用到了apache的对象池来管理文件导出相关资源连接和回收功能&#xff0c;因此花点时间简单了解下对象池相关使用&#xff0c;做点记录。 一. 连接池 频繁的建立和关闭连接&#xff0c;会极大的降低系统的性能&#xff0c;而连接池会在初始化的时候会创建一定数量的连接…

C++ | Leetcode C++题解之第123题买卖股票的最佳时机III

题目&#xff1a; 题解&#xff1a; class Solution { public:int maxProfit(vector<int>& prices) {int n prices.size();int buy1 -prices[0], sell1 0;int buy2 -prices[0], sell2 0;for (int i 1; i < n; i) {buy1 max(buy1, -prices[i]);sell1 max(…

OceanBase v4.2 特性解析:新增三种临时表功能,更多的Oracle语句兼容

特性说明 在Oracle模式下&#xff0c;OceanBase临时表已经实现了基本的create、select、insert、delete、update等功能。为了满足更多客户的需求&#xff0c;OceanBase正在扩展临时表的功能&#xff0c;例如支持merge into和insert all操作。merge into允许用户将源表中的数据…

FPGA高端项目:FPGA解码MIPI视频+图像缩放+视频拼接,基于MIPI CSI-2 RX Subsystem架构实现,提供4套工程源码和技术支持

目录 1、前言工程概述免责声明 2、相关方案推荐我这里已有的 MIPI 编解码方案本方案在Xilinx Artix7-35T上解码MIPI视频的应用本方案在Xilinx Artix7-100T上解码MIPI视频的应用本方案在Xilinx Kintex7上解码MIPI视频的应用本方案在Xilinx Zynq7000上解码MIPI视频的应用本方案在…

Linux基础命令目录管理002

之前讲述了目录的创建和删除&#xff0c;现在讲一下目录的移动修改。 操作系统&#xff1a; CentOS Stream 9 操作命令&#xff1a; mv 移动&#xff0c;重命名 选项 -v显示移动过程 [rootlocalhost ~]# mkdir 12 [rootlocalhost ~]# ll 总用量 1220 drwxr-xr-x 2 root…

11.1 排序算法

目录 11.1 排序算法 11.1.1 评价维度 11.1.2 理想排序算法 11.1 排序算法 排序算法&#xff08;sorting algorithm&#xff09;用于对一组数据按照特定顺序进行排列。排序算法有着广泛的应用&#xff0c;因为有序数据通常能够被更高效地查找、分析和处理。 如图 1…

WebSocket实现前后端双向数据的实时推送

一、WebSocket简介 WebSocket是一种网络通信协议&#xff0c;旨在实现客户端和服务器之间的双向、全双工通信。它在HTML5规范中被引入&#xff0c;用于替代基于传统HTTP协议的长轮询、轮询和流传输等方式&#xff0c;以提供更高效的实时数据传输。 WebSocket的特点 双向通信&a…

Python 3 判断文件是否存在

1 使用os.path模块 import osfile_path hello.txtif os.path.exists(file_path):print(f"文件 {file_path} 存在。") else:print(f"文件 {file_path} 不存在。") 2 使用pathlib模块 from pathlib import Pathfile_path Path(word.txt)if file_path.ex…

32-ESP32-S3-WIFI篇-03 Event Loop (事件循环)

ESP32-S3-WIFI 事件循环 介绍 在ESP32-S3的WiFi驱动程序中&#xff0c;事件循环是一个非常重要的概念。事件循环是一个无限循环&#xff0c;用于处理和分发系统中发生的各种事件。在WiFi驱动程序中&#xff0c;我们使用事件循环来处理和分发WiFi相关的事件。 创建事件循环 …

PTA 7-10 构造二叉检索树

本题目构造一棵二叉检索树。要求读入n个整数&#xff0c;以0结束。最后输出这棵树的先序序列。 输入格式: 输入n个整数&#xff0c;以0表示结束&#xff0c;数据间以空格隔开。 输出格式: 输出这棵树的先序序列&#xff0c;以一个空格隔开&#xff0c;结尾也有一个空格。 …

常用电机测试方法的介绍与功能实现(M测试方法)

目录 概述 1 常用电机测速方法简介 1.1 方法概览 1.2 编码器测速方法 2 M法测速 2.1 理论描述 2.2 实现原理 2.3 速度计算方法 3 功能实现 3.1 功能介绍 3.2 代码实现 3.2.1 使用STM32Cube配置参数 3.2.2 脉冲计数功能 3.2.3 测速函数 4 测试 概述 本文主要介绍…

springboot针对返回的response拦截处理越权问题

背景&#xff1a;针对越权测试&#xff0c;通过拦截工具Fiddler修改请求参数&#xff0c;越权查看平台里面所有公司的数据 1、自定义MyResponseBodyAdvice 实现ResponseBodyAdvice 使用过滤器和拦截我都试过&#xff0c;最终没有成功&#xff0c;可能技术比较菜&#xff0c;这…

策略模式解析

import java.util.*; enum TYPE { NORMAL,CASH_DISCOUNT,CASH_RETURN}; interface Cashsuper { public double acceptCash(double money); } class CashNormal implements CashSuper{// 正常收费子类 public double accptCash(double money){ return money; …

C# Winform 已知窗体句柄,如何遍历出所有控件句柄

c# windform 已知窗体句柄&#xff0c;如何遍历出所有控件句柄 public delegate bool CallBack(int hwnd, int lParam);public delegate bool EnumWindowsProc(int hWnd, int lParam); List<string> list new List<string>();[DllImport("user32.dll")]…