Unity自定义Button
2022.10.27更新:
该代码中包含以下几个事件:保持按下事件,与其他事件共存。
双击事件会先触发单击,这个问题已经解决,抱歉拖了这么久才完善.
并且新按钮命名为:ButtonPro

/*
* ================================================
* Describe: This script is used to custom UGUI`s button.
* Author: Xiaohei.Wang(Wenhao)
* CreationTime: 2022-10-26 16:43:48
* ModifyAuthor: Xiaohei.Wang(Wenhao)
* ModifyTime: 2022-10-26 16:43:48
* ScriptVersion: 0.1
* ===============================================
*/
using System;
using System.Collections;
using UnityEngine;
using UnityEngine.Events;
using UnityEngine.EventSystems;
using UnityEngine.Serialization;
using UnityEngine.UI;
namespace GameFramework.UI
{
[RequireComponent(typeof(Image))]
[RequireComponent(typeof(CanvasRenderer))]
public class ButtonPro : Selectable, ISubmitHandler
{
protected ButtonPro() { }
[Serializable]
public class ButtonClickedEvent : UnityEvent { }
[FormerlySerializedAs("onClick")]
[SerializeField]
private ButtonClickedEvent m_OnClick = new ButtonClickedEvent();
[FormerlySerializedAs("onLongPress")]
[SerializeField]
private ButtonClickedEvent m_onLongPress = new ButtonClickedEvent();
[FormerlySerializedAs("onDoubleClick")]
[SerializeField]
private ButtonClickedEvent m_onDoubleClick = new ButtonClickedEvent();
[FormerlySerializedAs("onKeepPress")]
[SerializeField]
private ButtonClickedEvent m_onKeepPress = new ButtonClickedEvent();
public ButtonClickedEvent onClick
{
get { return m_OnClick; }
}
public ButtonClickedEvent onDoubleClick
{
get { return m_onDoubleClick; }
}
public ButtonClickedEvent onLongPress
{
get { return m_onLongPress; }
}
public ButtonClickedEvent onKeepPress
{
get { return m_onKeepPress; }
}
private float m_longPressIntervalTime = 600.0f;
private float m_doubleClcikIntervalTime = 170.0f;
private float m_clickCount = 0;
private bool m_onHoldDown = false;
private bool m_isKeepPress = false;
private bool m_onEventTrigger = false;
private double m_clickIntervalTime = 0;
private DateTime m_clickStartTime;
private void OnAnyEventTrigger()
{
m_clickCount = 0;
m_onEventTrigger = true;
m_clickStartTime = default;
}
private void Press()
{
if (!IsActive() || !IsInteractable())
return;
UISystemProfilerApi.AddMarker("Button.onClick", this);
m_OnClick.Invoke();
}
private void Update()
{
if (!this.interactable) return;
m_clickIntervalTime = (DateTime.Now - m_clickStartTime).TotalMilliseconds;
if (!m_onHoldDown && 0 != m_clickCount)
{
if (m_clickIntervalTime >= m_doubleClcikIntervalTime && m_clickIntervalTime < m_longPressIntervalTime)
{
if (m_clickCount == 2)
m_onDoubleClick?.Invoke();
else
onClick?.Invoke();
OnAnyEventTrigger();
}
}
if (m_onHoldDown && !m_onEventTrigger)
{
if (m_clickIntervalTime >= m_longPressIntervalTime)
{
m_onHoldDown = false;
OnAnyEventTrigger();
m_onLongPress?.Invoke();
}
}
if (m_isKeepPress) onKeepPress?.Invoke();
}
public override void OnPointerDown(PointerEventData eventData)
{
m_onHoldDown = true;
m_isKeepPress = true;
m_onEventTrigger = false;
m_clickStartTime = DateTime.Now;
base.OnPointerDown(eventData);
}
public override void OnPointerUp(PointerEventData eventData)
{
if (m_onEventTrigger)
return;
m_clickCount++;
if (m_clickCount % 3 == 0)
{
onClick?.Invoke();
OnAnyEventTrigger();
return;
}
else
{
m_onHoldDown = false;
m_isKeepPress = false;
}
base.OnPointerUp(eventData);
}
public override void OnPointerExit(PointerEventData eventData)
{
m_onHoldDown = false;
m_isKeepPress = false;
base.OnPointerExit(eventData);
}
public virtual void OnSubmit(BaseEventData eventData)
{
Press();
if (!IsActive() || !IsInteractable())
return;
DoStateTransition(SelectionState.Pressed, false);
StartCoroutine(OnFinishSubmit());
}
private IEnumerator OnFinishSubmit()
{
var fadeTime = colors.fadeDuration;
var elapsedTime = 0f;
while (elapsedTime < fadeTime)
{
elapsedTime += Time.unscaledDeltaTime;
yield return null;
}
DoStateTransition(currentSelectionState, false);
}
}
}
2022.01.15的内容:
该代码中包含以下几个事件:其中按着事件,与其他事件共存。双击事件会先触发单击
- onClick 单击
- onDoubleClick 双击
- onLongPress 长按
- onKeepPress 按着


大家可以根据自己的需求来使用,也可以进行拓展,可能小黑的代码水平不高,但是希望能帮助到大家,如果你们拓展了告诉小黑一下,小黑会加到博客中来,并且标明是谁提供。
更新:
2022.01.17:增加面板属性Interactable的作用。
using System;
using System.Collections;
using UnityEngine;
using UnityEngine.Events;
using UnityEngine.EventSystems;
using UnityEngine.Serialization;
using UnityEngine.UI;
namespace CustomTools
{
/*
* Introduction:Custom UGUI`s button
* Creator:Xiaohei Wang
*/
[RequireComponent(typeof(Image))]
[RequireComponent(typeof(CanvasRenderer))]
public class CustomButton : Selectable, IPointerClickHandler, ISubmitHandler
{
protected CustomButton() { }
[Serializable]
public class ButtonClickedEvent : UnityEvent { }
[FormerlySerializedAs("onClick")]
[SerializeField]
private ButtonClickedEvent m_OnClick = new ButtonClickedEvent();
[FormerlySerializedAs("onLongPress")]
[SerializeField]
private ButtonClickedEvent m_onLongPress = new ButtonClickedEvent();
[FormerlySerializedAs("onDoubleClick")]
[SerializeField]
private ButtonClickedEvent m_onDoubleClick = new ButtonClickedEvent();
[FormerlySerializedAs("onKeepPress")]
[SerializeField]
private ButtonClickedEvent m_onKeepPress = new ButtonClickedEvent();
public ButtonClickedEvent onClick
{
get { return m_OnClick; }
}
public ButtonClickedEvent onDoubleClick
{
get { return m_onDoubleClick; }
}
public ButtonClickedEvent onLongPress
{
get { return m_onLongPress; }
}
public ButtonClickedEvent onKeepPress
{
get { return m_onKeepPress; }
}
private bool m_isPress = false;
private bool m_longPress = false;
private bool m_isKeepPress = false;
private DateTime m_currentStartTime;
private void Press()
{
if (!IsActive() || !IsInteractable())
return;
UISystemProfilerApi.AddMarker("Button.onClick", this);
m_OnClick.Invoke();
}
private void Update()
{
if (!this.interactable) return;
CheckForLongPress();
if (m_isKeepPress) onKeepPress?.Invoke();
}
private void CheckForLongPress()
{
if (m_isPress && !m_longPress)
{
if ((DateTime.Now - m_currentStartTime).TotalMilliseconds >= 600)
{
m_isPress = false;
m_longPress = true;
m_onLongPress?.Invoke();
}
}
}
public override void OnPointerDown(PointerEventData eventData)
{
m_isPress = true;
m_longPress = false;
m_isKeepPress = true;
m_currentStartTime = DateTime.Now;
base.OnPointerDown(eventData);
}
public override void OnPointerUp(PointerEventData eventData)
{
m_isPress = false;
m_isKeepPress = false;
base.OnPointerUp(eventData);
}
public override void OnPointerExit(PointerEventData eventData)
{
m_isPress = false;
m_isKeepPress = false;
base.OnPointerExit(eventData);
}
public virtual void OnPointerClick(PointerEventData eventData)
{
if (this.interactable && !m_longPress)
{
if (eventData.clickCount == 2)
m_onDoubleClick?.Invoke();
else if (eventData.clickCount == 1)
onClick?.Invoke();
}
}
public virtual void OnSubmit(BaseEventData eventData)
{
Press();
if (!IsActive() || !IsInteractable())
return;
DoStateTransition(SelectionState.Pressed, false);
StartCoroutine(OnFinishSubmit());
}
private IEnumerator OnFinishSubmit()
{
var fadeTime = colors.fadeDuration;
var elapsedTime = 0f;
while (elapsedTime < fadeTime)
{
elapsedTime += Time.unscaledDeltaTime;
yield return null;
}
DoStateTransition(currentSelectionState, false);
}
}
}
希望大家:点赞,留言,关注咯~
唠家常
- 小黑的今日分享结束啦,小伙伴们你们get到了么,你们有没有更好的办法呢,可以评论区留言分享,也可以加小黑的QQ:841298494,大家一起进步
今日无推荐
- 客官,看完get之后记得点赞哟!
- 小伙伴你还想要别的知识?好的呀,分享给你们
- 小黑的杂货铺,想要什么都有,客官不进来喝杯茶么?
Unity自定义Button的更多相关文章
- IOS开发之自定义Button(集成三种回调模式)
前面在做东西的时候都用到了storyboard,在今天的代码中就纯手写代码自己用封装个Button.这个Button继承于UIView类,在封装的时候用上啦OC中的三种回调模式:目标动作回调,委托回调 ...
- ios基础篇(二十四)—— 文字、图片的绘制及其自定义Button
这篇文章我们主要来拿官方的控件来研究一下,我们来仿照官方的控件,自己来实现它提供的控件: 首先来看看基本的图片与文字的绘制,很简单. 一.imageView 所有的视图都是继承自UIView,所以我们 ...
- iOS_Swift初识之使用三种回调方式自定义Button
最近在学习Swift ,发现青玉伏案大神早期用OC写的一篇博客--IOS开发之自定义Button(集成三种回调模式) 很适合用来熟悉Swift的回调方式,于是我就用Swift翻版了一下,具体实现原理 ...
- 自定义Button 的图片设置不显示问题。
如果你是自定义button 那么你设置图片就要用 button.imageView.image = [UIImage imageName:@""]; 如果你是给系统原生的butt ...
- (Unity)Unity自定义Debug日志文件,利用VS生成Dll文件并使用Dotfuscated进展混淆,避免被反编译
Unity自定义Debug日志文件,利用VS生成Dll文件并使用Dotfuscated进行混淆,避免被反编译. 1.打开VS,博主所用版本是Visual Studio 2013. 2.新建一个VC项目 ...
- Android 自定义Button按钮显示样式(正常、按下、获取焦点)
现在的用户对APP的外观看得很重要,如果APP内所有元件都用Android默认样式写,估计下面评论里就有一堆在骂UI丑的.今天学习自定义Button按钮样式.Button样式修改的是Button的背景 ...
- WPF 自定义Button控件及样式
这次通过最近做的小例子说明一下自定义Button控件和样式. 实现的效果为:
- Firemonkey 自定义Button的Style
这篇文章模仿HTML中基于CSS的Button,通过Style实现自定义样式的Button. 前言 主要模仿的CSS代码如下: CSS Code 123456789101112131415161718 ...
- 安卓开发笔记(三十):自定义Button
在笔者本人看了很多博客和书之后,发现很少博主对于自定义控件能够进行一个比较全面的思路讲解,大多数都是只讲了一些细节,但并没有讲如何把代码进行整体的实现.因此这里讲讲整体的自定义button实现的详细过 ...
随机推荐
- C# 9.0 添加和增强的功能【基础篇】
一.记录(record) C# 9.0 引入了记录类型. 可使用 record 关键字定义一个引用类型,以最简的方式创建不可变类型.这种类型是线程安全的,不需要进行线程同步,非常适合并行计算的数据共享 ...
- "xxx cannot be cast to jakarta.servlet.Servlet "报错解决方式
在做jsp的上机时候同学出现了一个500错误:com.kailong.servlet.ComputeBill cannot be cast to jaka.servlet.Servlet 然后因为我用 ...
- 定位java程序中占用cpu最高的线程堆栈信息
找出占用cpu最高的线程堆栈信息 在java编码中,有时会因为粗心导致cpu占用较高的情况,为了避免影响程序的正常运行,需要找到问题并解决.这里模拟一个cpu占用较高的场景,并尝试定位到代码行. 示例 ...
- 达梦-DBLINK数据库链接
aliases: [达梦 DBlink] tags: [数据库,DM,Blog] link: date: 2022-09-06 说明:DM-Oracle指的是在DM中创建链接至Oracle的Dblin ...
- 洛谷P4135 Ynoi2016 掉进兔子洞 (带权bitset?/bitset优化莫队 模板) 题解
题面. 看到这道题,我第一反应就是莫队. 我甚至也猜出了把所有询问的三个区间压到一起处理然后分别计算对应询问答案. 但是,这么复杂的贡献用什么东西存?难道要开一个数组 query_appear_tim ...
- Java安全之Mojarra JSF反序列化
Java安全之Mojarra JSF反序列化 About JSF JavaServer Faces,新一代的Java Web应用技术标准,吸收了很多Java Servlet以及其他的Web应用框架的特 ...
- C#11之原始字符串
最近.NET7.0和C#11相继发布,笔者也是第一时间就用上了C#11,其中C#11的有一个更新能解决困扰我多年的问题,也就是文章的标题原始字符串. 在使用C#11的原始字符串时,发现的一些有意思的东 ...
- Day11.2:标签的使用
标签的使用 当我们在嵌套语句中,例如当我们在for的嵌套循环语句中,想要终止或重新开始当前循环以外的循环的时候,单独仅靠break和continue和还不够,需要在我们想要作用的循环语句处加上一个标签 ...
- python(牛客)试题解析2 - 中等
导航 一.NC192 二叉树的后序遍历 二.NC117 合并二叉树 三.求长度最长的的连续子序列使他们的和等于sum 四.按顺序取出固定长度内容并合并两个数组为一个新数组 五.输出所有结果小于k的整数 ...
- CPU体系(2):ARM Store Buffer
本文主要翻译自 Arm Cortex-M7 Processor Technical Reference Manual r1p2 其中章节 Memory System / L1 caches / Sto ...