2024-01-29 19:32:59 +08:00
|
|
|
import { _decorator, CCBoolean, TweenEasing , CCFloat, CCString, Component, easing, Enum, EventTouch, isValid, Node, NodeEventType, Tween, tween, Vec2, Vec3 } from 'cc';
|
|
|
|
const { ccclass, property } = _decorator;
|
|
|
|
export enum FloatType {
|
|
|
|
None,
|
|
|
|
MoveToTop,
|
|
|
|
MoveToTopLayer,
|
|
|
|
// CopyToTopLayer
|
|
|
|
}
|
|
|
|
export enum DragBackHomeType {
|
|
|
|
SetPosition,
|
|
|
|
Tween
|
|
|
|
}
|
|
|
|
export enum EasingString {
|
|
|
|
'linear' , 'smooth' , 'fade' , 'constant' ,
|
|
|
|
'quadIn' , 'quadOut' , 'quadInOut' , 'quadOutIn' ,
|
|
|
|
'cubicIn' , 'cubicOut' , 'cubicInOut' , 'cubicOutIn' ,
|
|
|
|
'quartIn' , 'quartOut' , 'quartInOut' , 'quartOutIn' ,
|
|
|
|
'quintIn' , 'quintOut' , 'quintInOut' , 'quintOutIn' ,
|
|
|
|
'sineIn' , 'sineOut' , 'sineInOut' , 'sineOutIn' ,
|
|
|
|
'expoIn' , 'expoOut' , 'expoInOut' , 'expoOutIn' ,
|
|
|
|
'circIn' , 'circOut' , 'circInOut' , 'circOutIn' ,
|
|
|
|
'elasticIn' , 'elasticOut' , 'elasticInOut' , 'elasticOutIn' ,
|
|
|
|
'backIn' , 'backOut' , 'backInOut' , 'backOutIn' ,
|
|
|
|
'bounceIn' , 'bounceOut' , 'bounceInOut' , 'bounceOutIn'
|
|
|
|
}
|
|
|
|
@ccclass('GPDrag')
|
|
|
|
export class GPDrag extends Component {
|
|
|
|
@property({type:CCBoolean, tooltip:"触碰点偏移量敏感"})
|
|
|
|
public touchOffsetSensitive = true
|
|
|
|
|
|
|
|
@property({type:CCBoolean, tooltip:"失败时回到起始位置"})
|
|
|
|
public backHomeWhenFailed = true
|
|
|
|
|
|
|
|
@property({type:Enum(DragBackHomeType), tooltip:"回家的方式", visible:function(this){
|
|
|
|
return this.backHomeWhenFailed
|
|
|
|
}})
|
|
|
|
public backHomeType:DragBackHomeType = DragBackHomeType.SetPosition
|
|
|
|
|
|
|
|
@property({type:CCFloat, visible:function(this) {
|
|
|
|
return this.backHomeWhenFailed && this.backHomeType == DragBackHomeType.Tween
|
|
|
|
}})
|
|
|
|
public backTweenTime:number = 1;
|
|
|
|
|
|
|
|
@property({type:Enum(EasingString), tooltip:"缓动类型", visible:function(this) {
|
|
|
|
return this.backHomeWhenFailed && this.backHomeType == DragBackHomeType.Tween
|
|
|
|
}})
|
|
|
|
public backTweenEasing:EasingString = EasingString.linear
|
|
|
|
|
|
|
|
@property({type:Enum(FloatType), tooltip:"上浮类型"})
|
|
|
|
public floatType:FloatType = FloatType.None
|
|
|
|
|
|
|
|
@property({type:Node, visible:function(this){
|
|
|
|
return this.floatType == FloatType.MoveToTopLayer
|
|
|
|
// || this.floatType == FloatType.CopyToTopLayer
|
|
|
|
}})
|
|
|
|
public topLayerNode:Node
|
|
|
|
|
|
|
|
@property({type:CCBoolean, tooltip:"松开后返回原来的层级", visible:function(){
|
|
|
|
return this.floatType != FloatType.None
|
|
|
|
}})
|
|
|
|
public backToOriZ = true;
|
|
|
|
private srcParent:Node
|
|
|
|
private zOrder:number
|
|
|
|
private dragStartPos:Vec2 = new Vec2(0, 0)
|
|
|
|
private dragOffset:Vec2 = new Vec2(0, 0)
|
2024-01-29 19:56:56 +08:00
|
|
|
public succeedCheck:Function = null;
|
|
|
|
public succeedCallback:Function = null;
|
2024-01-29 19:32:59 +08:00
|
|
|
private backTween:Tween<Node>
|
|
|
|
|
|
|
|
|
|
|
|
start() {
|
|
|
|
this.node.on(NodeEventType.TOUCH_START, this.OnDragStart, this)
|
|
|
|
this.node.on(NodeEventType.TOUCH_MOVE, this.OnDragMove, this)
|
|
|
|
this.node.on(NodeEventType.TOUCH_END, this.OnDragEnd, this)
|
|
|
|
this.node.on(NodeEventType.TOUCH_CANCEL, this.OnDragEnd, this)
|
|
|
|
}
|
|
|
|
|
|
|
|
private OnDragStart(e:EventTouch) {
|
|
|
|
this.dragStartPos.set(this.node.position.x, this.node.position.y);
|
|
|
|
let touchPoint = e.getUILocation();
|
|
|
|
Vec2.subtract(this.dragOffset, this.dragStartPos, touchPoint)
|
|
|
|
switch(this.floatType) {
|
|
|
|
case FloatType.MoveToTop:
|
|
|
|
this.zOrder = this.node.getSiblingIndex();
|
|
|
|
this.node.setSiblingIndex(Infinity);
|
|
|
|
break;
|
|
|
|
case FloatType.MoveToTopLayer:
|
|
|
|
this.zOrder = this.node.getSiblingIndex()
|
|
|
|
this.srcParent = this.node.parent
|
|
|
|
this.node.parent = this.topLayerNode;
|
|
|
|
break;
|
|
|
|
// case FloatType.CopyToTopLayer:
|
|
|
|
// break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private OnDragMove(e:EventTouch) {
|
|
|
|
let p = e.getUILocation();
|
|
|
|
if (this.touchOffsetSensitive)
|
|
|
|
this.node.setPosition(new Vec3(p.x + this.dragOffset.x, p.y + this.dragOffset.y, 0))
|
|
|
|
else
|
|
|
|
this.node.setPosition(new Vec3(p.x, 0, p.y))
|
|
|
|
}
|
|
|
|
|
|
|
|
private OnDragEnd(e:EventTouch) {
|
|
|
|
if (this.succeedCheck && this.succeedCheck(e)) {
|
|
|
|
this.succeedCallback && this.succeedCallback(e)
|
|
|
|
} else if(this.backHomeWhenFailed){
|
|
|
|
let p = new Vec3(this.dragStartPos.x, this.dragStartPos.y, 0);
|
|
|
|
switch(this.backHomeType) {
|
|
|
|
case DragBackHomeType.SetPosition:
|
|
|
|
this.node.setPosition(p)
|
|
|
|
this.OnArrivedHome();
|
|
|
|
break;
|
|
|
|
case DragBackHomeType.Tween:
|
|
|
|
this.backTween && this.backTween.stop()
|
|
|
|
let easing:TweenEasing = EasingString[this.backTweenEasing] as TweenEasing
|
|
|
|
this.backTween = tween(this.node)
|
|
|
|
.to(this.backTweenTime, {position:p}, {easing:easing}).call(this.OnArrivedHome.bind(this)).start()
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private OnArrivedHome() {
|
|
|
|
if (this.backToOriZ)
|
|
|
|
switch(this.floatType) {
|
|
|
|
case FloatType.MoveToTop:
|
|
|
|
this.node.setSiblingIndex(this.zOrder)
|
|
|
|
break;
|
|
|
|
case FloatType.MoveToTopLayer:
|
|
|
|
this.node.parent = this.srcParent;
|
|
|
|
this.node.setSiblingIndex(this.zOrder)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected onDestroy(): void {
|
|
|
|
this.node.off(NodeEventType.TOUCH_START, this.OnDragStart, this)
|
|
|
|
this.node.off(NodeEventType.TOUCH_MOVE, this.OnDragMove, this)
|
|
|
|
this.node.off(NodeEventType.TOUCH_END, this.OnDragEnd, this)
|
|
|
|
this.node.off(NodeEventType.TOUCH_CANCEL, this.OnDragEnd, this)
|
|
|
|
}
|
|
|
|
}
|