mirror of
https://github.com/NativeScript/NativeScript.git
synced 2025-08-14 18:12:09 +08:00

* feat(modal-view): introduce cancelable property on ShowModalOptions * fix(modal-view): handle iOS 13 modal dismiss gesture * chore: address PR comments
1159 lines
42 KiB
TypeScript
1159 lines
42 KiB
TypeScript
// Definitions.
|
|
import { Point, CustomLayoutView as CustomLayoutViewDefinition, dip } from ".";
|
|
import { GestureTypes, GestureEventData } from "../../gestures";
|
|
// Types.
|
|
import {
|
|
ViewCommon, layout, isEnabledProperty, originXProperty, originYProperty, automationTextProperty, isUserInteractionEnabledProperty,
|
|
traceEnabled, traceWrite, traceCategories, traceNotifyEvent,
|
|
paddingLeftProperty, paddingTopProperty, paddingRightProperty, paddingBottomProperty,
|
|
Color, EventData, ShowModalOptions
|
|
} from "./view-common";
|
|
|
|
import {
|
|
Length, PercentLength, Visibility, HorizontalAlignment, VerticalAlignment,
|
|
visibilityProperty, opacityProperty, horizontalAlignmentProperty, verticalAlignmentProperty,
|
|
minWidthProperty, minHeightProperty, widthProperty, heightProperty,
|
|
marginLeftProperty, marginTopProperty, marginRightProperty, marginBottomProperty,
|
|
rotateProperty, scaleXProperty, scaleYProperty, translateXProperty, translateYProperty,
|
|
zIndexProperty, backgroundInternalProperty, androidElevationProperty, androidDynamicElevationOffsetProperty
|
|
} from "../../styling/style-properties";
|
|
|
|
import { Background, ad as androidBackground } from "../../styling/background";
|
|
import { profile } from "../../../profiling";
|
|
import { topmost } from "../../frame/frame-stack";
|
|
import { AndroidActivityBackPressedEventData, android as androidApp } from "../../../application";
|
|
import { device } from "../../../platform";
|
|
import lazy from "../../../utils/lazy";
|
|
|
|
export * from "./view-common";
|
|
|
|
const DOMID = "_domId";
|
|
const androidBackPressedEvent = "androidBackPressed";
|
|
|
|
const shortAnimTime = 17694720; // android.R.integer.config_shortAnimTime
|
|
const statePressed = 16842919; // android.R.attr.state_pressed
|
|
const stateEnabled = 16842910; // android.R.attr.state_enabled
|
|
const styleAnimationDialog = 16973826; // android.R.style.Animation_Dialog
|
|
|
|
const sdkVersion = lazy(() => parseInt(device.sdkVersion));
|
|
|
|
const modalMap = new Map<number, DialogOptions>();
|
|
|
|
let TouchListener: TouchListener;
|
|
let DialogFragment: DialogFragment;
|
|
|
|
interface DialogOptions {
|
|
owner: View;
|
|
fullscreen: boolean;
|
|
animated: boolean;
|
|
stretched: boolean;
|
|
cancelable: boolean;
|
|
shownCallback: () => void;
|
|
dismissCallback: () => void;
|
|
}
|
|
|
|
interface TouchListener {
|
|
new(owner: View): android.view.View.OnTouchListener;
|
|
}
|
|
|
|
interface DialogFragment {
|
|
new(): androidx.fragment.app.DialogFragment;
|
|
}
|
|
|
|
function initializeTouchListener(): void {
|
|
if (TouchListener) {
|
|
return;
|
|
}
|
|
|
|
@Interfaces([android.view.View.OnTouchListener])
|
|
class TouchListenerImpl extends java.lang.Object implements android.view.View.OnTouchListener {
|
|
private owner: WeakRef<View>;
|
|
constructor(owner: View) {
|
|
super();
|
|
this.owner = new WeakRef(owner);
|
|
|
|
return global.__native(this);
|
|
}
|
|
|
|
onTouch(view: android.view.View, event: android.view.MotionEvent): boolean {
|
|
const owner = this.owner.get();
|
|
if (!owner) {
|
|
return;
|
|
}
|
|
owner.handleGestureTouch(event);
|
|
|
|
let nativeView = owner.nativeViewProtected;
|
|
if (!nativeView || !nativeView.onTouchEvent) {
|
|
return false;
|
|
}
|
|
|
|
return nativeView.onTouchEvent(event);
|
|
}
|
|
}
|
|
|
|
TouchListener = TouchListenerImpl;
|
|
}
|
|
|
|
function initializeDialogFragment() {
|
|
if (DialogFragment) {
|
|
return;
|
|
}
|
|
|
|
class DialogImpl extends android.app.Dialog {
|
|
constructor(public fragment: DialogFragmentImpl,
|
|
context: android.content.Context,
|
|
themeResId: number) {
|
|
super(context, themeResId);
|
|
|
|
return global.__native(this);
|
|
}
|
|
|
|
public onDetachedFromWindow(): void {
|
|
super.onDetachedFromWindow();
|
|
this.fragment = null;
|
|
}
|
|
|
|
public onBackPressed(): void {
|
|
const view = this.fragment.owner;
|
|
const args = <AndroidActivityBackPressedEventData>{
|
|
eventName: "activityBackPressed",
|
|
object: view,
|
|
activity: view._context,
|
|
cancel: false,
|
|
};
|
|
|
|
// Fist fire application.android global event
|
|
androidApp.notify(args);
|
|
if (args.cancel) {
|
|
return;
|
|
}
|
|
|
|
view.notify(args);
|
|
|
|
if (!args.cancel && !view.onBackPressed()) {
|
|
super.onBackPressed();
|
|
}
|
|
}
|
|
}
|
|
|
|
class DialogFragmentImpl extends androidx.fragment.app.DialogFragment {
|
|
public owner: View;
|
|
private _fullscreen: boolean;
|
|
private _animated: boolean;
|
|
private _stretched: boolean;
|
|
private _cancelable: boolean;
|
|
private _shownCallback: () => void;
|
|
private _dismissCallback: () => void;
|
|
|
|
constructor() {
|
|
super();
|
|
|
|
return global.__native(this);
|
|
}
|
|
|
|
public onCreateDialog(savedInstanceState: android.os.Bundle): android.app.Dialog {
|
|
const ownerId = this.getArguments().getInt(DOMID);
|
|
const options = getModalOptions(ownerId);
|
|
this.owner = options.owner;
|
|
this._fullscreen = options.fullscreen;
|
|
this._animated = options.animated;
|
|
this._cancelable = options.cancelable;
|
|
this._stretched = options.stretched;
|
|
this._dismissCallback = options.dismissCallback;
|
|
this._shownCallback = options.shownCallback;
|
|
this.setStyle(androidx.fragment.app.DialogFragment.STYLE_NO_TITLE, 0);
|
|
|
|
let theme = this.getTheme();
|
|
if (this._fullscreen) {
|
|
// In fullscreen mode, get the application's theme.
|
|
theme = this.getActivity().getApplicationInfo().theme;
|
|
}
|
|
|
|
const dialog = new DialogImpl(this, this.getActivity(), theme);
|
|
|
|
// do not override alignment unless fullscreen modal will be shown;
|
|
// otherwise we might break component-level layout:
|
|
// https://github.com/NativeScript/NativeScript/issues/5392
|
|
if (!this._fullscreen && !this._stretched) {
|
|
this.owner.horizontalAlignment = "center";
|
|
this.owner.verticalAlignment = "middle";
|
|
} else {
|
|
this.owner.horizontalAlignment = "stretch";
|
|
this.owner.verticalAlignment = "stretch";
|
|
}
|
|
|
|
// set the modal window animation
|
|
// https://github.com/NativeScript/NativeScript/issues/5989
|
|
if (this._animated) {
|
|
dialog
|
|
.getWindow()
|
|
.setWindowAnimations(
|
|
styleAnimationDialog
|
|
);
|
|
}
|
|
|
|
dialog.setCanceledOnTouchOutside(this._cancelable);
|
|
|
|
return dialog;
|
|
}
|
|
|
|
public onCreateView(inflater: android.view.LayoutInflater, container: android.view.ViewGroup, savedInstanceState: android.os.Bundle): android.view.View {
|
|
const owner = this.owner;
|
|
owner._setupAsRootView(this.getActivity());
|
|
owner._isAddedToNativeVisualTree = true;
|
|
|
|
return owner.nativeViewProtected;
|
|
}
|
|
|
|
public onStart(): void {
|
|
super.onStart();
|
|
if (this._fullscreen) {
|
|
const window = this.getDialog().getWindow();
|
|
const length = android.view.ViewGroup.LayoutParams.MATCH_PARENT;
|
|
window.setLayout(length, length);
|
|
// This removes the default backgroundDrawable so there are no margins.
|
|
window.setBackgroundDrawable(new android.graphics.drawable.ColorDrawable(android.graphics.Color.WHITE));
|
|
}
|
|
|
|
const owner = this.owner;
|
|
if (owner && !owner.isLoaded) {
|
|
owner.callLoaded();
|
|
}
|
|
|
|
this._shownCallback();
|
|
}
|
|
|
|
public onDismiss(dialog: android.content.DialogInterface): void {
|
|
super.onDismiss(dialog);
|
|
const manager = this.getFragmentManager();
|
|
if (manager) {
|
|
removeModal(this.owner._domId);
|
|
this._dismissCallback();
|
|
}
|
|
|
|
const owner = this.owner;
|
|
if (owner && owner.isLoaded) {
|
|
owner.callUnloaded();
|
|
}
|
|
}
|
|
|
|
public onDestroy(): void {
|
|
super.onDestroy();
|
|
const owner = this.owner;
|
|
|
|
if (owner) {
|
|
// Android calls onDestroy before onDismiss.
|
|
// Make sure we unload first and then call _tearDownUI.
|
|
if (owner.isLoaded) {
|
|
owner.callUnloaded();
|
|
}
|
|
|
|
owner._isAddedToNativeVisualTree = false;
|
|
owner._tearDownUI(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
DialogFragment = DialogFragmentImpl;
|
|
}
|
|
|
|
function saveModal(options: DialogOptions) {
|
|
modalMap.set(options.owner._domId, options);
|
|
}
|
|
|
|
function removeModal(domId: number) {
|
|
modalMap.delete(domId);
|
|
}
|
|
|
|
function getModalOptions(domId: number): DialogOptions {
|
|
return modalMap.get(domId);
|
|
}
|
|
|
|
export class View extends ViewCommon {
|
|
public static androidBackPressedEvent = androidBackPressedEvent;
|
|
|
|
public _dialogFragment: androidx.fragment.app.DialogFragment;
|
|
public _manager: androidx.fragment.app.FragmentManager;
|
|
private _isClickable: boolean;
|
|
private touchListenerIsSet: boolean;
|
|
private touchListener: android.view.View.OnTouchListener;
|
|
private layoutChangeListenerIsSet: boolean;
|
|
private layoutChangeListener: android.view.View.OnLayoutChangeListener;
|
|
private _rootManager: androidx.fragment.app.FragmentManager;
|
|
|
|
nativeViewProtected: android.view.View;
|
|
|
|
// TODO: Implement unobserve that detach the touchListener.
|
|
_observe(type: GestureTypes, callback: (args: GestureEventData) => void, thisArg?: any): void {
|
|
super._observe(type, callback, thisArg);
|
|
if (this.isLoaded && !this.touchListenerIsSet) {
|
|
this.setOnTouchListener();
|
|
}
|
|
}
|
|
|
|
on(eventNames: string, callback: (data: EventData) => void, thisArg?: any) {
|
|
super.on(eventNames, callback, thisArg);
|
|
const isLayoutEvent = typeof eventNames === "string" ? eventNames.indexOf(ViewCommon.layoutChangedEvent) !== -1 : false;
|
|
|
|
if (this.isLoaded && !this.layoutChangeListenerIsSet && isLayoutEvent) {
|
|
this.setOnLayoutChangeListener();
|
|
}
|
|
}
|
|
|
|
off(eventNames: string, callback?: any, thisArg?: any) {
|
|
super.off(eventNames, callback, thisArg);
|
|
const isLayoutEvent = typeof eventNames === "string" ? eventNames.indexOf(ViewCommon.layoutChangedEvent) !== -1 : false;
|
|
|
|
// Remove native listener only if there are no more user listeners for LayoutChanged event
|
|
if (this.isLoaded && this.layoutChangeListenerIsSet && isLayoutEvent && !this.hasListeners(ViewCommon.layoutChangedEvent)) {
|
|
this.nativeViewProtected.removeOnLayoutChangeListener(this.layoutChangeListener);
|
|
this.layoutChangeListenerIsSet = false;
|
|
}
|
|
}
|
|
|
|
public _getChildFragmentManager(): androidx.fragment.app.FragmentManager {
|
|
return null;
|
|
}
|
|
|
|
public _getRootFragmentManager(): androidx.fragment.app.FragmentManager {
|
|
if (!this._rootManager && this._context) {
|
|
this._rootManager = (<androidx.fragment.app.FragmentActivity>this._context).getSupportFragmentManager();
|
|
}
|
|
|
|
return this._rootManager;
|
|
}
|
|
|
|
public _getFragmentManager(): androidx.fragment.app.FragmentManager {
|
|
let manager = this._manager;
|
|
if (!manager) {
|
|
let view: View = this;
|
|
let frameOrTabViewItemFound = false;
|
|
while (view) {
|
|
// when interacting with nested fragments instead of using getSupportFragmentManager
|
|
// we must always use getChildFragmentManager instead;
|
|
// we have three sources of fragments -- Frame fragments, TabViewItem fragments, and
|
|
// modal dialog fragments
|
|
|
|
// modal -> frame / tabview (frame / tabview use modal CHILD fm)
|
|
const dialogFragment = view._dialogFragment;
|
|
if (dialogFragment) {
|
|
manager = dialogFragment.getChildFragmentManager();
|
|
break;
|
|
}
|
|
|
|
// - frame1 -> frame2 (frame2 uses frame1 CHILD fm)
|
|
// - tabview -> frame1 (frame1 uses tabview item CHILD fm)
|
|
// - frame1 -> tabview (tabview uses frame1 CHILD fm)
|
|
// - frame1 -> tabview -> frame2 (tabview uses frame1 CHILD fm; frame2 uses tabview item CHILD fm)
|
|
if (view._hasFragments) {
|
|
if (frameOrTabViewItemFound) {
|
|
manager = view._getChildFragmentManager();
|
|
break;
|
|
}
|
|
|
|
frameOrTabViewItemFound = true;
|
|
}
|
|
|
|
// the case is needed because _dialogFragment is on View
|
|
// but parent may be ViewBase.
|
|
view = view.parent as View;
|
|
}
|
|
|
|
if (!manager) {
|
|
manager = this._getRootFragmentManager();
|
|
}
|
|
|
|
this._manager = manager;
|
|
}
|
|
|
|
return manager;
|
|
}
|
|
|
|
@profile
|
|
public onLoaded() {
|
|
this._manager = null;
|
|
this._rootManager = null;
|
|
super.onLoaded();
|
|
this.setOnTouchListener();
|
|
}
|
|
|
|
@profile
|
|
public onUnloaded() {
|
|
if (this.touchListenerIsSet) {
|
|
this.nativeViewProtected.setOnTouchListener(null);
|
|
this.touchListenerIsSet = false;
|
|
this.nativeViewProtected.setClickable(this._isClickable);
|
|
}
|
|
|
|
this._manager = null;
|
|
this._rootManager = null;
|
|
super.onUnloaded();
|
|
}
|
|
|
|
public onBackPressed(): boolean {
|
|
let topmostFrame = topmost();
|
|
|
|
// Delegate back navigation handling to the topmost Frame
|
|
// when it's a child of the current View.
|
|
if (topmostFrame && topmostFrame._hasAncestorView(this)) {
|
|
return topmostFrame.onBackPressed();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public handleGestureTouch(event: android.view.MotionEvent): any {
|
|
for (let type in this._gestureObservers) {
|
|
let list = this._gestureObservers[type];
|
|
list.forEach(element => {
|
|
element.androidOnTouchEvent(event);
|
|
});
|
|
}
|
|
if (this.parent instanceof View) {
|
|
this.parent.handleGestureTouch(event);
|
|
}
|
|
}
|
|
|
|
private hasGestureObservers() {
|
|
return this._gestureObservers && Object.keys(this._gestureObservers).length > 0;
|
|
}
|
|
|
|
public initNativeView(): void {
|
|
super.initNativeView();
|
|
this._isClickable = this.nativeViewProtected.isClickable();
|
|
|
|
if (this.hasListeners(ViewCommon.layoutChangedEvent)) {
|
|
this.setOnLayoutChangeListener();
|
|
}
|
|
}
|
|
|
|
public disposeNativeView(): void {
|
|
super.disposeNativeView();
|
|
|
|
if (this.layoutChangeListenerIsSet) {
|
|
this.layoutChangeListenerIsSet = false;
|
|
this.nativeViewProtected.removeOnLayoutChangeListener(this.layoutChangeListener);
|
|
}
|
|
}
|
|
|
|
private setOnTouchListener() {
|
|
if (!this.nativeViewProtected || !this.hasGestureObservers()) {
|
|
return;
|
|
}
|
|
|
|
// do not set noop listener that handles the event (disabled listener) if IsUserInteractionEnabled is
|
|
// false as we might need the ability for the event to pass through to a parent view
|
|
initializeTouchListener();
|
|
this.touchListener = this.touchListener || new TouchListener(this);
|
|
this.nativeViewProtected.setOnTouchListener(this.touchListener);
|
|
|
|
this.touchListenerIsSet = true;
|
|
|
|
if (this.nativeViewProtected.setClickable) {
|
|
this.nativeViewProtected.setClickable(this.isUserInteractionEnabled);
|
|
}
|
|
}
|
|
|
|
private setOnLayoutChangeListener() {
|
|
if (this.nativeViewProtected) {
|
|
const owner = this;
|
|
this.layoutChangeListenerIsSet = true;
|
|
this.layoutChangeListener = this.layoutChangeListener || new android.view.View.OnLayoutChangeListener({
|
|
onLayoutChange(
|
|
v: android.view.View,
|
|
left: number, top: number, right: number, bottom: number,
|
|
oldLeft: number, oldTop: number, oldRight: number, oldBottom: number): void {
|
|
if (left !== oldLeft || top !== oldTop || right !== oldRight || bottom !== oldBottom) {
|
|
owner._raiseLayoutChangedEvent();
|
|
}
|
|
}
|
|
});
|
|
|
|
this.nativeViewProtected.addOnLayoutChangeListener(this.layoutChangeListener);
|
|
}
|
|
}
|
|
|
|
get isLayoutRequired(): boolean {
|
|
return !this.isLayoutValid;
|
|
}
|
|
|
|
get isLayoutValid(): boolean {
|
|
if (this.nativeViewProtected) {
|
|
return !this.nativeViewProtected.isLayoutRequested();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
get _hasFragments(): boolean {
|
|
return false;
|
|
}
|
|
|
|
public layoutNativeView(left: number, top: number, right: number, bottom: number): void {
|
|
if (this.nativeViewProtected) {
|
|
this.nativeViewProtected.layout(left, top, right, bottom);
|
|
}
|
|
}
|
|
|
|
@profile
|
|
public requestLayout(): void {
|
|
super.requestLayout();
|
|
if (this.nativeViewProtected) {
|
|
this.nativeViewProtected.requestLayout();
|
|
}
|
|
}
|
|
|
|
public measure(widthMeasureSpec: number, heightMeasureSpec: number): void {
|
|
super.measure(widthMeasureSpec, heightMeasureSpec);
|
|
this.onMeasure(widthMeasureSpec, heightMeasureSpec);
|
|
}
|
|
|
|
public layout(left: number, top: number, right: number, bottom: number): void {
|
|
super.layout(left, top, right, bottom);
|
|
this.onLayout(left, top, right, bottom);
|
|
}
|
|
|
|
public onMeasure(widthMeasureSpec: number, heightMeasureSpec: number): void {
|
|
let view = this.nativeViewProtected;
|
|
if (view) {
|
|
view.measure(widthMeasureSpec, heightMeasureSpec);
|
|
this.setMeasuredDimension(view.getMeasuredWidth(), view.getMeasuredHeight());
|
|
}
|
|
}
|
|
|
|
public onLayout(left: number, top: number, right: number, bottom: number): void {
|
|
let view = this.nativeViewProtected;
|
|
if (view) {
|
|
this.layoutNativeView(left, top, right, bottom);
|
|
}
|
|
}
|
|
|
|
_getCurrentLayoutBounds(): { left: number; top: number; right: number; bottom: number } {
|
|
if (this.nativeViewProtected && !this.isCollapsed) {
|
|
return {
|
|
left: this.nativeViewProtected.getLeft(),
|
|
top: this.nativeViewProtected.getTop(),
|
|
right: this.nativeViewProtected.getRight(),
|
|
bottom: this.nativeViewProtected.getBottom()
|
|
};
|
|
} else {
|
|
return { left: 0, top: 0, right: 0, bottom: 0 };
|
|
}
|
|
}
|
|
|
|
public getMeasuredWidth(): number {
|
|
if (this.nativeViewProtected) {
|
|
return this.nativeViewProtected.getMeasuredWidth();
|
|
}
|
|
|
|
return super.getMeasuredWidth();
|
|
}
|
|
|
|
public getMeasuredHeight(): number {
|
|
if (this.nativeViewProtected) {
|
|
return this.nativeViewProtected.getMeasuredHeight();
|
|
}
|
|
|
|
return super.getMeasuredHeight();
|
|
}
|
|
|
|
public focus(): boolean {
|
|
if (this.nativeViewProtected) {
|
|
return this.nativeViewProtected.requestFocus();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public getLocationInWindow(): Point {
|
|
if (!this.nativeViewProtected || !this.nativeViewProtected.getWindowToken()) {
|
|
return undefined;
|
|
}
|
|
|
|
let nativeArray = (<any>Array).create("int", 2);
|
|
this.nativeViewProtected.getLocationInWindow(nativeArray);
|
|
|
|
return {
|
|
x: layout.toDeviceIndependentPixels(nativeArray[0]),
|
|
y: layout.toDeviceIndependentPixels(nativeArray[1]),
|
|
};
|
|
}
|
|
|
|
public getLocationOnScreen(): Point {
|
|
if (!this.nativeViewProtected || !this.nativeViewProtected.getWindowToken()) {
|
|
return undefined;
|
|
}
|
|
|
|
let nativeArray = (<any>Array).create("int", 2);
|
|
this.nativeViewProtected.getLocationOnScreen(nativeArray);
|
|
|
|
return {
|
|
x: layout.toDeviceIndependentPixels(nativeArray[0]),
|
|
y: layout.toDeviceIndependentPixels(nativeArray[1]),
|
|
};
|
|
}
|
|
|
|
public getLocationRelativeTo(otherView: ViewCommon): Point {
|
|
if (!this.nativeViewProtected || !this.nativeViewProtected.getWindowToken() ||
|
|
!otherView || !otherView.nativeViewProtected || !otherView.nativeViewProtected.getWindowToken() ||
|
|
this.nativeViewProtected.getWindowToken() !== otherView.nativeViewProtected.getWindowToken()) {
|
|
return undefined;
|
|
}
|
|
|
|
let myArray = (<any>Array).create("int", 2);
|
|
this.nativeViewProtected.getLocationOnScreen(myArray);
|
|
let otherArray = (<any>Array).create("int", 2);
|
|
otherView.nativeViewProtected.getLocationOnScreen(otherArray);
|
|
|
|
return {
|
|
x: layout.toDeviceIndependentPixels(myArray[0] - otherArray[0]),
|
|
y: layout.toDeviceIndependentPixels(myArray[1] - otherArray[1]),
|
|
};
|
|
}
|
|
|
|
public static resolveSizeAndState(size: number, specSize: number, specMode: number, childMeasuredState: number): number {
|
|
let result = size;
|
|
switch (specMode) {
|
|
case layout.UNSPECIFIED:
|
|
result = size;
|
|
break;
|
|
|
|
case layout.AT_MOST:
|
|
if (specSize < size) {
|
|
result = specSize | layout.MEASURED_STATE_TOO_SMALL;
|
|
}
|
|
break;
|
|
|
|
case layout.EXACTLY:
|
|
result = specSize;
|
|
break;
|
|
}
|
|
|
|
return result | (childMeasuredState & layout.MEASURED_STATE_MASK);
|
|
}
|
|
protected _showNativeModalView(parent: View, options: ShowModalOptions) {
|
|
super._showNativeModalView(parent, options);
|
|
initializeDialogFragment();
|
|
|
|
const df = new DialogFragment();
|
|
const args = new android.os.Bundle();
|
|
args.putInt(DOMID, this._domId);
|
|
df.setArguments(args);
|
|
|
|
let cancelable = true;
|
|
|
|
if (options.android && (<any>options).android.cancelable !== undefined) {
|
|
cancelable = !!(<any>options).android.cancelable;
|
|
console.log("ShowModalOptions.android.cancelable is deprecated. Use ShowModalOptions.cancelable instead.");
|
|
}
|
|
|
|
cancelable = options.cancelable !== undefined ? !!options.cancelable : cancelable;
|
|
|
|
const dialogOptions: DialogOptions = {
|
|
owner: this,
|
|
fullscreen: !!options.fullscreen,
|
|
animated: !!options.animated,
|
|
stretched: !!options.stretched,
|
|
cancelable: cancelable,
|
|
shownCallback: () => this._raiseShownModallyEvent(),
|
|
dismissCallback: () => this.closeModal()
|
|
};
|
|
|
|
saveModal(dialogOptions);
|
|
|
|
this._dialogFragment = df;
|
|
this._raiseShowingModallyEvent();
|
|
|
|
this._dialogFragment.show(parent._getRootFragmentManager(), this._domId.toString());
|
|
}
|
|
|
|
protected _hideNativeModalView(parent: View, whenClosedCallback: () => void) {
|
|
const manager = this._dialogFragment.getFragmentManager();
|
|
if (manager) {
|
|
this._dialogFragment.dismissAllowingStateLoss();
|
|
}
|
|
|
|
this._dialogFragment = null;
|
|
whenClosedCallback();
|
|
}
|
|
|
|
[isEnabledProperty.setNative](value: boolean) {
|
|
this.nativeViewProtected.setEnabled(value);
|
|
}
|
|
|
|
[originXProperty.getDefault](): number {
|
|
return this.nativeViewProtected.getPivotX();
|
|
}
|
|
[originXProperty.setNative](value: number) {
|
|
org.nativescript.widgets.OriginPoint.setX(this.nativeViewProtected, value);
|
|
}
|
|
|
|
[originYProperty.getDefault](): number {
|
|
return this.nativeViewProtected.getPivotY();
|
|
}
|
|
[originYProperty.setNative](value: number) {
|
|
org.nativescript.widgets.OriginPoint.setY(this.nativeViewProtected, value);
|
|
}
|
|
|
|
[automationTextProperty.getDefault](): string {
|
|
return this.nativeViewProtected.getContentDescription();
|
|
}
|
|
[automationTextProperty.setNative](value: string) {
|
|
this.nativeViewProtected.setContentDescription(value);
|
|
}
|
|
|
|
[isUserInteractionEnabledProperty.setNative](value: boolean) {
|
|
this.nativeViewProtected.setClickable(value);
|
|
this.nativeViewProtected.setFocusable(value);
|
|
}
|
|
|
|
[visibilityProperty.getDefault](): Visibility {
|
|
let nativeVisibility = this.nativeViewProtected.getVisibility();
|
|
switch (nativeVisibility) {
|
|
case android.view.View.VISIBLE:
|
|
return "visible";
|
|
case android.view.View.INVISIBLE:
|
|
return "hidden";
|
|
case android.view.View.GONE:
|
|
return "collapse";
|
|
default:
|
|
throw new Error(`Unsupported android.view.View visibility: ${nativeVisibility}. Currently supported values are android.view.View.VISIBLE, android.view.View.INVISIBLE, android.view.View.GONE.`);
|
|
}
|
|
}
|
|
[visibilityProperty.setNative](value: Visibility) {
|
|
switch (value) {
|
|
case "visible":
|
|
this.nativeViewProtected.setVisibility(android.view.View.VISIBLE);
|
|
break;
|
|
case "hidden":
|
|
this.nativeViewProtected.setVisibility(android.view.View.INVISIBLE);
|
|
break;
|
|
case "collapse":
|
|
this.nativeViewProtected.setVisibility(android.view.View.GONE);
|
|
break;
|
|
default:
|
|
throw new Error(`Invalid visibility value: ${value}. Valid values are: visible, hidden, collapse.`);
|
|
}
|
|
}
|
|
|
|
[opacityProperty.getDefault](): number {
|
|
return this.nativeViewProtected.getAlpha();
|
|
}
|
|
[opacityProperty.setNative](value: number) {
|
|
this.nativeViewProtected.setAlpha(float(value));
|
|
}
|
|
|
|
[androidElevationProperty.getDefault](): number {
|
|
return this.getDefaultElevation();
|
|
}
|
|
[androidElevationProperty.setNative](value: number) {
|
|
if (sdkVersion() < 21) {
|
|
return;
|
|
}
|
|
|
|
this.refreshStateListAnimator();
|
|
}
|
|
|
|
[androidDynamicElevationOffsetProperty.getDefault](): number {
|
|
return this.getDefaultDynamicElevationOffset();
|
|
}
|
|
[androidDynamicElevationOffsetProperty.setNative](value: number) {
|
|
if (sdkVersion() < 21) {
|
|
return;
|
|
}
|
|
|
|
this.refreshStateListAnimator();
|
|
}
|
|
|
|
protected getDefaultElevation(): number {
|
|
if (sdkVersion() < 21) {
|
|
return 0;
|
|
}
|
|
|
|
// NOTE: overriden in Button implementation as for widgets with StateListAnimator (Button)
|
|
// nativeView.getElevation() returns 0 at the time of the getDefault() query
|
|
return layout.toDeviceIndependentPixels((<any>this.nativeViewProtected).getElevation());
|
|
}
|
|
|
|
protected getDefaultDynamicElevationOffset() {
|
|
// NOTE: overriden in Button implementation
|
|
return 0;
|
|
}
|
|
|
|
private refreshStateListAnimator() {
|
|
const nativeView: any = this.nativeViewProtected;
|
|
|
|
const ObjectAnimator = android.animation.ObjectAnimator;
|
|
const AnimatorSet = android.animation.AnimatorSet;
|
|
|
|
const duration = nativeView.getContext().getResources().getInteger(shortAnimTime) / 2;
|
|
|
|
let elevation = this.androidElevation;
|
|
if (typeof elevation === "undefined" || elevation === null) {
|
|
elevation = this.getDefaultElevation();
|
|
}
|
|
elevation = layout.toDevicePixels(elevation);
|
|
|
|
const z = layout.toDevicePixels(0);
|
|
|
|
let pressedZ = this.androidDynamicElevationOffset;
|
|
if (typeof pressedZ === "undefined" || pressedZ === null) {
|
|
pressedZ = this.getDefaultDynamicElevationOffset();
|
|
}
|
|
pressedZ = layout.toDevicePixels(pressedZ);
|
|
|
|
const pressedSet = new AnimatorSet();
|
|
pressedSet.playTogether(java.util.Arrays.asList([
|
|
ObjectAnimator.ofFloat(nativeView, "translationZ", [pressedZ])
|
|
.setDuration(duration),
|
|
ObjectAnimator.ofFloat(nativeView, "elevation", [elevation])
|
|
.setDuration(0),
|
|
]));
|
|
|
|
const notPressedSet = new AnimatorSet();
|
|
notPressedSet.playTogether(java.util.Arrays.asList([
|
|
ObjectAnimator.ofFloat(nativeView, "translationZ", [z])
|
|
.setDuration(duration),
|
|
ObjectAnimator.ofFloat(nativeView, "elevation", [elevation])
|
|
.setDuration(0),
|
|
]));
|
|
|
|
const defaultSet = new AnimatorSet();
|
|
defaultSet.playTogether(java.util.Arrays.asList([
|
|
ObjectAnimator.ofFloat(nativeView, "translationZ", [0]).setDuration(0),
|
|
ObjectAnimator.ofFloat(nativeView, "elevation", [0]).setDuration(0),
|
|
]));
|
|
|
|
const stateListAnimator = new (<any>android.animation).StateListAnimator();
|
|
stateListAnimator.addState([statePressed, stateEnabled], pressedSet);
|
|
stateListAnimator.addState([stateEnabled], notPressedSet);
|
|
stateListAnimator.addState([], defaultSet);
|
|
nativeView.setStateListAnimator(stateListAnimator);
|
|
}
|
|
|
|
[horizontalAlignmentProperty.getDefault](): HorizontalAlignment {
|
|
return <HorizontalAlignment>org.nativescript.widgets.ViewHelper.getHorizontalAlignment(this.nativeViewProtected);
|
|
}
|
|
[horizontalAlignmentProperty.setNative](value: HorizontalAlignment) {
|
|
const nativeView = this.nativeViewProtected;
|
|
const lp: any = nativeView.getLayoutParams() || new org.nativescript.widgets.CommonLayoutParams();
|
|
// Set only if params gravity exists.
|
|
if (lp.gravity !== undefined) {
|
|
switch (value) {
|
|
case "left":
|
|
lp.gravity = android.view.Gravity.LEFT | (lp.gravity & android.view.Gravity.VERTICAL_GRAVITY_MASK);
|
|
if (lp.weight < 0) {
|
|
lp.weight = -2;
|
|
}
|
|
break;
|
|
case "center":
|
|
lp.gravity = android.view.Gravity.CENTER_HORIZONTAL | (lp.gravity & android.view.Gravity.VERTICAL_GRAVITY_MASK);
|
|
if (lp.weight < 0) {
|
|
lp.weight = -2;
|
|
}
|
|
break;
|
|
case "right":
|
|
lp.gravity = android.view.Gravity.RIGHT | (lp.gravity & android.view.Gravity.VERTICAL_GRAVITY_MASK);
|
|
if (lp.weight < 0) {
|
|
lp.weight = -2;
|
|
}
|
|
break;
|
|
case "stretch":
|
|
lp.gravity = android.view.Gravity.FILL_HORIZONTAL | (lp.gravity & android.view.Gravity.VERTICAL_GRAVITY_MASK);
|
|
if (lp.weight < 0) {
|
|
lp.weight = -1;
|
|
}
|
|
break;
|
|
}
|
|
nativeView.setLayoutParams(lp);
|
|
}
|
|
}
|
|
|
|
[verticalAlignmentProperty.getDefault](): VerticalAlignment {
|
|
return <VerticalAlignment>org.nativescript.widgets.ViewHelper.getVerticalAlignment(this.nativeViewProtected);
|
|
}
|
|
[verticalAlignmentProperty.setNative](value: VerticalAlignment) {
|
|
const nativeView = this.nativeViewProtected;
|
|
const lp: any = nativeView.getLayoutParams() || new org.nativescript.widgets.CommonLayoutParams();
|
|
// Set only if params gravity exists.
|
|
if (lp.gravity !== undefined) {
|
|
switch (value) {
|
|
case "top":
|
|
lp.gravity = android.view.Gravity.TOP | (lp.gravity & android.view.Gravity.HORIZONTAL_GRAVITY_MASK);
|
|
if (lp.height < 0) {
|
|
lp.height = -2;
|
|
}
|
|
break;
|
|
case "middle":
|
|
lp.gravity = android.view.Gravity.CENTER_VERTICAL | (lp.gravity & android.view.Gravity.HORIZONTAL_GRAVITY_MASK);
|
|
if (lp.height < 0) {
|
|
lp.height = -2;
|
|
}
|
|
break;
|
|
case "bottom":
|
|
lp.gravity = android.view.Gravity.BOTTOM | (lp.gravity & android.view.Gravity.HORIZONTAL_GRAVITY_MASK);
|
|
if (lp.height < 0) {
|
|
lp.height = -2;
|
|
}
|
|
break;
|
|
case "stretch":
|
|
lp.gravity = android.view.Gravity.FILL_VERTICAL | (lp.gravity & android.view.Gravity.HORIZONTAL_GRAVITY_MASK);
|
|
if (lp.height < 0) {
|
|
lp.height = -1;
|
|
}
|
|
break;
|
|
}
|
|
nativeView.setLayoutParams(lp);
|
|
}
|
|
}
|
|
|
|
[rotateProperty.setNative](value: number) {
|
|
org.nativescript.widgets.ViewHelper.setRotate(this.nativeViewProtected, float(value));
|
|
}
|
|
|
|
[scaleXProperty.setNative](value: number) {
|
|
org.nativescript.widgets.ViewHelper.setScaleX(this.nativeViewProtected, float(value));
|
|
}
|
|
|
|
[scaleYProperty.setNative](value: number) {
|
|
org.nativescript.widgets.ViewHelper.setScaleY(this.nativeViewProtected, float(value));
|
|
}
|
|
|
|
[translateXProperty.setNative](value: dip) {
|
|
org.nativescript.widgets.ViewHelper.setTranslateX(this.nativeViewProtected, layout.toDevicePixels(value));
|
|
}
|
|
|
|
[translateYProperty.setNative](value: dip) {
|
|
org.nativescript.widgets.ViewHelper.setTranslateY(this.nativeViewProtected, layout.toDevicePixels(value));
|
|
}
|
|
|
|
[zIndexProperty.getDefault](): number {
|
|
return 0;
|
|
}
|
|
[zIndexProperty.setNative](value: number) {
|
|
org.nativescript.widgets.ViewHelper.setZIndex(this.nativeViewProtected, value);
|
|
}
|
|
|
|
[backgroundInternalProperty.getDefault](): android.graphics.drawable.Drawable {
|
|
const nativeView = this.nativeViewProtected;
|
|
const drawable = nativeView.getBackground();
|
|
if (drawable) {
|
|
const constantState = drawable.getConstantState();
|
|
if (constantState) {
|
|
try {
|
|
return constantState.newDrawable(nativeView.getResources());
|
|
} catch (e) {
|
|
return drawable;
|
|
}
|
|
} else {
|
|
return drawable;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
[backgroundInternalProperty.setNative](value: android.graphics.drawable.Drawable | Background) {
|
|
this._redrawNativeBackground(value);
|
|
}
|
|
|
|
[minWidthProperty.setNative](value: Length) {
|
|
if (this.parent instanceof CustomLayoutView && this.parent.nativeViewProtected) {
|
|
this.parent._setChildMinWidthNative(this);
|
|
} else {
|
|
this._setMinWidthNative(this.minWidth);
|
|
}
|
|
}
|
|
|
|
[minHeightProperty.setNative](value: Length) {
|
|
if (this.parent instanceof CustomLayoutView && this.parent.nativeViewProtected) {
|
|
this.parent._setChildMinHeightNative(this);
|
|
} else {
|
|
this._setMinHeightNative(this.minHeight);
|
|
}
|
|
}
|
|
|
|
_redrawNativeBackground(value: android.graphics.drawable.Drawable | Background): void {
|
|
if (value instanceof Background) {
|
|
androidBackground.onBackgroundOrBorderPropertyChanged(this);
|
|
} else {
|
|
const nativeView = this.nativeViewProtected;
|
|
nativeView.setBackground(value);
|
|
|
|
const style = this.style;
|
|
const paddingTop = paddingTopProperty.isSet(style) ? this.effectivePaddingTop : this._defaultPaddingTop;
|
|
const paddingRight = paddingRightProperty.isSet(style) ? this.effectivePaddingRight : this._defaultPaddingRight;
|
|
const paddingBottom = paddingBottomProperty.isSet(style) ? this.effectivePaddingBottom : this._defaultPaddingBottom;
|
|
const paddingLeft = paddingLeftProperty.isSet(style) ? this.effectivePaddingLeft : this._defaultPaddingLeft;
|
|
|
|
if (this._isPaddingRelative) {
|
|
nativeView.setPaddingRelative(paddingLeft, paddingTop, paddingRight, paddingBottom);
|
|
} else {
|
|
nativeView.setPadding(paddingLeft, paddingTop, paddingRight, paddingBottom);
|
|
}
|
|
|
|
(<any>nativeView).background = undefined;
|
|
}
|
|
}
|
|
}
|
|
|
|
export class ContainerView extends View {
|
|
public iosOverflowSafeArea: boolean;
|
|
}
|
|
|
|
export class CustomLayoutView extends ContainerView implements CustomLayoutViewDefinition {
|
|
nativeViewProtected: android.view.ViewGroup;
|
|
|
|
public createNativeView() {
|
|
return new org.nativescript.widgets.ContentLayout(this._context);
|
|
}
|
|
|
|
public _addViewToNativeVisualTree(child: ViewCommon, atIndex: number = Number.MAX_SAFE_INTEGER): boolean {
|
|
super._addViewToNativeVisualTree(child);
|
|
|
|
if (this.nativeViewProtected && child.nativeViewProtected) {
|
|
if (traceEnabled()) {
|
|
traceWrite(`${this}.nativeView.addView(${child}.nativeView, ${atIndex})`, traceCategories.VisualTreeEvents);
|
|
}
|
|
this.nativeViewProtected.addView(child.nativeViewProtected, atIndex);
|
|
if (child instanceof View) {
|
|
this._updateNativeLayoutParams(child);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public _updateNativeLayoutParams(child: View): void {
|
|
this._setChildMinWidthNative(child);
|
|
this._setChildMinHeightNative(child);
|
|
}
|
|
|
|
public _setChildMinWidthNative(child: View): void {
|
|
child._setMinWidthNative(child.minWidth);
|
|
}
|
|
|
|
public _setChildMinHeightNative(child: View): void {
|
|
child._setMinHeightNative(child.minHeight);
|
|
}
|
|
|
|
public _removeViewFromNativeVisualTree(child: ViewCommon): void {
|
|
super._removeViewFromNativeVisualTree(child);
|
|
|
|
const nativeView = this.nativeViewProtected;
|
|
const childView = child.nativeViewProtected;
|
|
if (nativeView && childView) {
|
|
nativeView.removeView(childView);
|
|
if (traceEnabled()) {
|
|
traceWrite(`${nativeView}.removeView(${childView})`, traceCategories.VisualTreeEvents);
|
|
traceNotifyEvent(child, "childInLayoutRemovedFromNativeVisualTree");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
type NativeSetter = { (view: android.view.View, value: number): void };
|
|
type NativeGetter = { (view: android.view.View): number };
|
|
const percentNotSupported = (view: android.view.View, value: number) => { throw new Error("PercentLength is not supported."); };
|
|
interface NativePercentLengthPropertyOptions {
|
|
getter?: string | symbol;
|
|
setter: string | symbol;
|
|
auto?: number;
|
|
getPixels?: NativeGetter;
|
|
setPixels: NativeSetter;
|
|
setPercent?: NativeSetter;
|
|
}
|
|
|
|
function createNativePercentLengthProperty(options: NativePercentLengthPropertyOptions) {
|
|
const { getter, setter, auto = 0 } = options;
|
|
let setPixels, getPixels, setPercent;
|
|
if (getter) {
|
|
View.prototype[getter] = function (this: View): PercentLength {
|
|
if (options) {
|
|
setPixels = options.setPixels;
|
|
getPixels = options.getPixels;
|
|
setPercent = options.setPercent || percentNotSupported;
|
|
options = null;
|
|
}
|
|
const value = getPixels(this.nativeViewProtected);
|
|
if (value == auto) { // tslint:disable-line
|
|
return "auto";
|
|
} else {
|
|
return { value, unit: "px" };
|
|
}
|
|
};
|
|
}
|
|
if (setter) {
|
|
View.prototype[setter] = function (this: View, length: PercentLength) {
|
|
if (options) {
|
|
setPixels = options.setPixels;
|
|
getPixels = options.getPixels;
|
|
setPercent = options.setPercent || percentNotSupported;
|
|
options = null;
|
|
}
|
|
if (length == "auto") { // tslint:disable-line
|
|
setPixels(this.nativeViewProtected, auto);
|
|
} else if (typeof length === "number") {
|
|
setPixels(this.nativeViewProtected, layout.round(layout.toDevicePixels(length)));
|
|
} else if (length.unit == "dip") { // tslint:disable-line
|
|
setPixels(this.nativeViewProtected, layout.round(layout.toDevicePixels(length.value)));
|
|
} else if (length.unit == "px") { // tslint:disable-line
|
|
setPixels(this.nativeViewProtected, layout.round(length.value));
|
|
} else if (length.unit == "%") { // tslint:disable-line
|
|
setPercent(this.nativeViewProtected, length.value);
|
|
} else {
|
|
throw new Error(`Unsupported PercentLength ${length}`);
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: marginTopProperty.setNative,
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setMarginTop; },
|
|
get setPercent() { return org.nativescript.widgets.ViewHelper.setMarginTopPercent; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: marginRightProperty.setNative,
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setMarginRight; },
|
|
get setPercent() { return org.nativescript.widgets.ViewHelper.setMarginRightPercent; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: marginBottomProperty.setNative,
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setMarginBottom; },
|
|
get setPercent() { return org.nativescript.widgets.ViewHelper.setMarginBottomPercent; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: marginLeftProperty.setNative,
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setMarginLeft; },
|
|
get setPercent() { return org.nativescript.widgets.ViewHelper.setMarginLeftPercent; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: widthProperty.setNative,
|
|
auto: -1, //android.view.ViewGroup.LayoutParams.MATCH_PARENT,
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setWidth; },
|
|
get setPercent() { return org.nativescript.widgets.ViewHelper.setWidthPercent; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: heightProperty.setNative,
|
|
auto: -1, //android.view.ViewGroup.LayoutParams.MATCH_PARENT,
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setHeight; },
|
|
get setPercent() { return org.nativescript.widgets.ViewHelper.setHeightPercent; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: "_setMinWidthNative",
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setMinWidth; }
|
|
});
|
|
|
|
createNativePercentLengthProperty({
|
|
setter: "_setMinHeightNative",
|
|
get setPixels() { return org.nativescript.widgets.ViewHelper.setMinHeight; }
|
|
});
|