mirror of
https://github.com/NativeScript/NativeScript.git
synced 2025-08-14 18:12:09 +08:00
188 lines
6.0 KiB
TypeScript
188 lines
6.0 KiB
TypeScript
import { ButtonBase } from './button-common';
|
|
import { AndroidHelper, PseudoClassHandler } from '../core/view';
|
|
import { paddingLeftProperty, paddingTopProperty, paddingRightProperty, paddingBottomProperty, Length, zIndexProperty, minWidthProperty, minHeightProperty } from '../styling/style-properties';
|
|
import { textAlignmentProperty } from '../text-base';
|
|
import { CoreTypes } from '../../core-types';
|
|
import { profile } from '../../profiling';
|
|
import { TouchGestureEventData, TouchAction, GestureTypes } from '../gestures';
|
|
import { Device } from '../../platform';
|
|
import { SDK_VERSION } from '../../utils/constants';
|
|
import type { Background } from '../styling/background';
|
|
import { NativeScriptAndroidView } from '../utils';
|
|
|
|
export * from './button-common';
|
|
|
|
interface ClickListener {
|
|
new (owner: Button): android.view.View.OnClickListener;
|
|
}
|
|
|
|
let ClickListener: ClickListener;
|
|
let AndroidButton: typeof android.widget.Button;
|
|
|
|
function initializeClickListener(): void {
|
|
if (ClickListener) {
|
|
return;
|
|
}
|
|
|
|
@NativeClass
|
|
@Interfaces([android.view.View.OnClickListener])
|
|
class ClickListenerImpl extends java.lang.Object implements android.view.View.OnClickListener {
|
|
constructor(public owner: Button) {
|
|
super();
|
|
|
|
return global.__native(this);
|
|
}
|
|
|
|
public onClick(v: android.view.View): void {
|
|
const owner = this.owner;
|
|
if (owner) {
|
|
owner._emit(ButtonBase.tapEvent);
|
|
}
|
|
}
|
|
}
|
|
|
|
ClickListener = ClickListenerImpl;
|
|
}
|
|
|
|
export class Button extends ButtonBase {
|
|
nativeViewProtected: android.widget.Button;
|
|
|
|
private _stateListAnimator: any;
|
|
private _highlightedHandler: (args: TouchGestureEventData) => void;
|
|
|
|
@profile
|
|
public createNativeView() {
|
|
if (!AndroidButton) {
|
|
AndroidButton = android.widget.Button;
|
|
}
|
|
|
|
return new AndroidButton(this._context);
|
|
}
|
|
|
|
public initNativeView(): void {
|
|
super.initNativeView();
|
|
const nativeView = this.nativeViewProtected;
|
|
initializeClickListener();
|
|
const clickListener = new ClickListener(this);
|
|
nativeView.setOnClickListener(clickListener);
|
|
(<any>nativeView).clickListener = clickListener;
|
|
}
|
|
|
|
public disposeNativeView() {
|
|
if ((<any>this.nativeViewProtected)?.clickListener) {
|
|
(<any>this.nativeViewProtected).clickListener.owner = null;
|
|
}
|
|
super.disposeNativeView();
|
|
}
|
|
|
|
public resetNativeView(): void {
|
|
super.resetNativeView();
|
|
|
|
if (this._stateListAnimator && SDK_VERSION >= 21) {
|
|
(<any>this.nativeViewProtected).setStateListAnimator(this._stateListAnimator);
|
|
this._stateListAnimator = undefined;
|
|
}
|
|
}
|
|
|
|
@PseudoClassHandler('normal', 'highlighted', 'pressed', 'active')
|
|
_updateButtonStateChangeHandler(subscribe: boolean) {
|
|
if (subscribe) {
|
|
this._highlightedHandler =
|
|
this._highlightedHandler ||
|
|
((args: TouchGestureEventData) => {
|
|
switch (args.action) {
|
|
case TouchAction.up:
|
|
case TouchAction.cancel:
|
|
this._goToVisualState(this.defaultVisualState);
|
|
break;
|
|
case TouchAction.down:
|
|
this._goToVisualState('highlighted');
|
|
break;
|
|
}
|
|
});
|
|
this.on(GestureTypes[GestureTypes.touch], this._highlightedHandler);
|
|
} else {
|
|
this.off(GestureTypes[GestureTypes.touch], this._highlightedHandler);
|
|
}
|
|
}
|
|
|
|
[minWidthProperty.getDefault](): CoreTypes.LengthType {
|
|
const dips = org.nativescript.widgets.ViewHelper.getMinWidth(this.nativeViewProtected);
|
|
|
|
return { value: dips, unit: 'px' };
|
|
}
|
|
|
|
[minHeightProperty.getDefault](): CoreTypes.LengthType {
|
|
const dips = org.nativescript.widgets.ViewHelper.getMinHeight(this.nativeViewProtected);
|
|
|
|
return { value: dips, unit: 'px' };
|
|
}
|
|
|
|
[paddingTopProperty.getDefault](): CoreTypes.LengthType {
|
|
return { value: this._defaultPaddingTop, unit: 'px' };
|
|
}
|
|
[paddingTopProperty.setNative](value: CoreTypes.LengthType) {
|
|
org.nativescript.widgets.ViewHelper.setPaddingTop(this.nativeViewProtected, Length.toDevicePixels(value, 0) + Length.toDevicePixels(this.style.borderTopWidth, 0));
|
|
}
|
|
|
|
[paddingRightProperty.getDefault](): CoreTypes.LengthType {
|
|
return { value: this._defaultPaddingRight, unit: 'px' };
|
|
}
|
|
[paddingRightProperty.setNative](value: CoreTypes.LengthType) {
|
|
org.nativescript.widgets.ViewHelper.setPaddingRight(this.nativeViewProtected, Length.toDevicePixels(value, 0) + Length.toDevicePixels(this.style.borderRightWidth, 0));
|
|
}
|
|
|
|
[paddingBottomProperty.getDefault](): CoreTypes.LengthType {
|
|
return { value: this._defaultPaddingBottom, unit: 'px' };
|
|
}
|
|
[paddingBottomProperty.setNative](value: CoreTypes.LengthType) {
|
|
org.nativescript.widgets.ViewHelper.setPaddingBottom(this.nativeViewProtected, Length.toDevicePixels(value, 0) + Length.toDevicePixels(this.style.borderBottomWidth, 0));
|
|
}
|
|
|
|
[paddingLeftProperty.getDefault](): CoreTypes.LengthType {
|
|
return { value: this._defaultPaddingLeft, unit: 'px' };
|
|
}
|
|
[paddingLeftProperty.setNative](value: CoreTypes.LengthType) {
|
|
org.nativescript.widgets.ViewHelper.setPaddingLeft(this.nativeViewProtected, Length.toDevicePixels(value, 0) + Length.toDevicePixels(this.style.borderLeftWidth, 0));
|
|
}
|
|
|
|
[zIndexProperty.setNative](value: number) {
|
|
if (SDK_VERSION >= 21) {
|
|
const nativeView = this.nativeViewProtected;
|
|
if (!this._stateListAnimator) {
|
|
this._stateListAnimator = (<any>nativeView).getStateListAnimator();
|
|
}
|
|
(<any>nativeView).setStateListAnimator(null);
|
|
}
|
|
|
|
org.nativescript.widgets.ViewHelper.setZIndex(this.nativeViewProtected, value);
|
|
}
|
|
|
|
[textAlignmentProperty.setNative](value: CoreTypes.TextAlignmentType) {
|
|
// Button initial value is center.
|
|
const newValue = value === 'initial' ? 'center' : value;
|
|
super[textAlignmentProperty.setNative](newValue);
|
|
}
|
|
|
|
protected getDefaultElevation(): number {
|
|
if (SDK_VERSION < 21) {
|
|
return 0;
|
|
}
|
|
|
|
// NOTE: Button widget has StateListAnimator that defines the elevation value and
|
|
// at the time of the getDefault() query the animator is not applied yet so we
|
|
// return the hardcoded @dimen/button_elevation_material value 2dp here instead
|
|
return 2;
|
|
}
|
|
|
|
protected getDefaultDynamicElevationOffset(): number {
|
|
if (SDK_VERSION < 21) {
|
|
return 0;
|
|
}
|
|
|
|
return 4; // 4dp @dimen/button_pressed_z_material
|
|
}
|
|
}
|
|
|
|
Button.prototype._ignoreFlexMinWidthHeightReset = true;
|