diff --git a/Android/app/build.gradle b/Android/app/build.gradle
index 8f3a79bd..30ec9d1a 100644
--- a/Android/app/build.gradle
+++ b/Android/app/build.gradle
@@ -27,6 +27,7 @@ dependencies {
implementation 'com.github.bumptech.glide:glide:4.10.0'
implementation 'com.github.bumptech.glide:annotations:4.10.0'
implementation 'com.github.penfeizhou.android.animation:glide-plugin:1.3.1'
+ implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
annotationProcessor 'com.github.bumptech.glide:compiler:4.10.0'
debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.0-beta-4'
}
diff --git a/Android/app/src/main/AndroidManifest.xml b/Android/app/src/main/AndroidManifest.xml
index 36ce1127..3988de39 100644
--- a/Android/app/src/main/AndroidManifest.xml
+++ b/Android/app/src/main/AndroidManifest.xml
@@ -3,6 +3,7 @@
package="pub.doric.demo">
+
+
diff --git a/Android/app/src/main/java/pub/doric/demo/MainActivity.java b/Android/app/src/main/java/pub/doric/demo/MainActivity.java
index ad23198d..b9153c44 100644
--- a/Android/app/src/main/java/pub/doric/demo/MainActivity.java
+++ b/Android/app/src/main/java/pub/doric/demo/MainActivity.java
@@ -16,6 +16,7 @@
package pub.doric.demo;
import android.content.Intent;
+import android.graphics.Color;
import android.os.Bundle;
import android.util.TypedValue;
import android.view.Gravity;
@@ -34,6 +35,7 @@ import java.util.List;
import pub.doric.DoricActivity;
import pub.doric.devkit.ui.DemoDebugActivity;
+import pub.doric.refresh.DoricSwipeLayout;
import pub.doric.utils.DoricUtils;
public class MainActivity extends AppCompatActivity {
@@ -43,11 +45,25 @@ public class MainActivity extends AppCompatActivity {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
+ final DoricSwipeLayout swipeLayout = findViewById(R.id.swipe_layout);
+ swipeLayout.setOnRefreshListener(new DoricSwipeLayout.OnRefreshListener() {
+ @Override
+ public void onRefresh() {
+ swipeLayout.setRefreshing(false);
+ }
+ });
+ swipeLayout.setBackgroundColor(Color.YELLOW);
+ swipeLayout.getRefreshView().setBackgroundColor(Color.RED);
+ TextView textView = new TextView(this);
+ textView.setText("This is header");
+ swipeLayout.getRefreshView().setContent(textView);
RecyclerView recyclerView = findViewById(R.id.root);
+ recyclerView.setBackgroundColor(Color.WHITE);
recyclerView.setLayoutManager(new LinearLayoutManager(this));
try {
String[] demos = getAssets().list("demo");
List ret = new ArrayList<>();
+ ret.add("Test");
for (String str : demos) {
if (str.endsWith("js")) {
ret.add(str);
@@ -91,6 +107,11 @@ public class MainActivity extends AppCompatActivity {
tv.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
+ if (data[position].contains("Test")) {
+ Intent intent = new Intent(tv.getContext(), PullableActivity.class);
+ tv.getContext().startActivity(intent);
+ return;
+ }
if (data[position].contains("NavigatorDemo")) {
Intent intent = new Intent(tv.getContext(), DoricActivity.class);
intent.putExtra("scheme", "assets://demo/" + data[position]);
diff --git a/Android/app/src/main/java/pub/doric/demo/PullableActivity.java b/Android/app/src/main/java/pub/doric/demo/PullableActivity.java
new file mode 100644
index 00000000..2ce73a2a
--- /dev/null
+++ b/Android/app/src/main/java/pub/doric/demo/PullableActivity.java
@@ -0,0 +1,28 @@
+package pub.doric.demo;
+
+import androidx.appcompat.app.AppCompatActivity;
+
+import android.graphics.Color;
+import android.os.Bundle;
+import android.widget.FrameLayout;
+
+import pub.doric.refresh.DoricSwipeLayout;
+
+public class PullableActivity extends AppCompatActivity {
+
+ @Override
+ protected void onCreate(Bundle savedInstanceState) {
+ super.onCreate(savedInstanceState);
+ setContentView(R.layout.activity_pullable);
+ final DoricSwipeLayout swipeRefreshLayout = findViewById(R.id.swipe_layout);
+ FrameLayout frameLayout = new FrameLayout(this);
+ frameLayout.setBackgroundColor(Color.YELLOW);
+ swipeRefreshLayout.addView(frameLayout);
+ swipeRefreshLayout.setOnRefreshListener(new DoricSwipeLayout.OnRefreshListener() {
+ @Override
+ public void onRefresh() {
+ swipeRefreshLayout.setRefreshing(false);
+ }
+ });
+ }
+}
diff --git a/Android/app/src/main/res/layout/activity_main.xml b/Android/app/src/main/res/layout/activity_main.xml
index 4abb9bc3..8a4718b2 100644
--- a/Android/app/src/main/res/layout/activity_main.xml
+++ b/Android/app/src/main/res/layout/activity_main.xml
@@ -1,8 +1,13 @@
-
-
\ No newline at end of file
+ android:layout_height="match_parent">
+
+
+
\ No newline at end of file
diff --git a/Android/app/src/main/res/layout/activity_pullable.xml b/Android/app/src/main/res/layout/activity_pullable.xml
new file mode 100644
index 00000000..c6923232
--- /dev/null
+++ b/Android/app/src/main/res/layout/activity_pullable.xml
@@ -0,0 +1,13 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/Android/doric/src/main/java/pub/doric/DoricRegistry.java b/Android/doric/src/main/java/pub/doric/DoricRegistry.java
index 12722dc4..9919509f 100644
--- a/Android/doric/src/main/java/pub/doric/DoricRegistry.java
+++ b/Android/doric/src/main/java/pub/doric/DoricRegistry.java
@@ -25,6 +25,7 @@ import pub.doric.plugin.NavigatorPlugin;
import pub.doric.plugin.NetworkPlugin;
import pub.doric.plugin.ShaderPlugin;
import pub.doric.plugin.StoragePlugin;
+import pub.doric.refresh.RefreshableNode;
import pub.doric.shader.HLayoutNode;
import pub.doric.shader.ImageNode;
import pub.doric.shader.ScrollerNode;
@@ -96,6 +97,7 @@ public class DoricRegistry {
this.registerViewNode(ScrollerNode.class);
this.registerViewNode(SliderNode.class);
this.registerViewNode(SlideItemNode.class);
+ this.registerViewNode(RefreshableNode.class);
initRegistry(this);
}
diff --git a/Android/doric/src/main/java/pub/doric/engine/DoricJSEngine.java b/Android/doric/src/main/java/pub/doric/engine/DoricJSEngine.java
index 7cc40445..a4bf460f 100644
--- a/Android/doric/src/main/java/pub/doric/engine/DoricJSEngine.java
+++ b/Android/doric/src/main/java/pub/doric/engine/DoricJSEngine.java
@@ -96,6 +96,12 @@ public class DoricJSEngine implements Handler.Callback, DoricTimerExtension.Time
return null;
}
});
+ mDoricJSE.injectGlobalJSFunction(DoricConstant.INJECT_EMPTY, new JavaFunction() {
+ @Override
+ public JavaValue exec(JSDecoder[] args) {
+ return null;
+ }
+ });
mDoricJSE.injectGlobalJSFunction(DoricConstant.INJECT_REQUIRE, new JavaFunction() {
@Override
public JavaValue exec(JSDecoder[] args) {
diff --git a/Android/doric/src/main/java/pub/doric/refresh/DoricRefreshView.java b/Android/doric/src/main/java/pub/doric/refresh/DoricRefreshView.java
new file mode 100644
index 00000000..8b1ded39
--- /dev/null
+++ b/Android/doric/src/main/java/pub/doric/refresh/DoricRefreshView.java
@@ -0,0 +1,100 @@
+package pub.doric.refresh;
+
+import android.content.Context;
+import android.util.AttributeSet;
+import android.view.Gravity;
+import android.view.View;
+import android.view.ViewGroup;
+import android.view.animation.Animation;
+import android.widget.FrameLayout;
+
+import androidx.annotation.AttrRes;
+import androidx.annotation.NonNull;
+import androidx.annotation.Nullable;
+
+/**
+ * @Description: pub.doric.pullable
+ * @Author: pengfei.zhou
+ * @CreateDate: 2019-11-25
+ */
+public class DoricRefreshView extends FrameLayout implements PullingListener {
+ private View content;
+ private Animation.AnimationListener mListener;
+
+ private PullingListener mPullingListenr;
+
+ public DoricRefreshView(@NonNull Context context) {
+ super(context);
+ }
+
+ public DoricRefreshView(@NonNull Context context, @Nullable AttributeSet attrs) {
+ super(context, attrs);
+ }
+
+ public DoricRefreshView(@NonNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr) {
+ super(context, attrs, defStyleAttr);
+ }
+
+ public void setContent(View v) {
+ removeAllViews();
+ content = v;
+ if (v.getLayoutParams() instanceof FrameLayout.LayoutParams) {
+ ((LayoutParams) v.getLayoutParams()).gravity = Gravity.BOTTOM;
+ } else {
+ LayoutParams params = new LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
+ params.gravity = Gravity.CENTER;
+ v.setLayoutParams(params);
+ }
+ addView(v);
+ }
+
+ public View getContent() {
+ return content;
+ }
+
+
+ public void setPullingListenr(PullingListener listenr) {
+ this.mPullingListenr = listenr;
+ }
+
+ @Override
+ public void startAnimation() {
+ if (mPullingListenr != null) {
+ mPullingListenr.startAnimation();
+ }
+ }
+
+ @Override
+ public void stopAnimation() {
+ if (mPullingListenr != null) {
+ mPullingListenr.stopAnimation();
+ }
+ }
+
+ @Override
+ public void setProgressRotation(float rotation) {
+ if (mPullingListenr != null) {
+ mPullingListenr.setProgressRotation(rotation);
+ }
+ }
+
+ public void setAnimationListener(Animation.AnimationListener listener) {
+ mListener = listener;
+ }
+
+ @Override
+ protected void onAnimationStart() {
+ super.onAnimationStart();
+ if (mListener != null) {
+ mListener.onAnimationStart(getAnimation());
+ }
+ }
+
+ @Override
+ protected void onAnimationEnd() {
+ super.onAnimationEnd();
+ if (mListener != null) {
+ mListener.onAnimationEnd(getAnimation());
+ }
+ }
+}
\ No newline at end of file
diff --git a/Android/doric/src/main/java/pub/doric/refresh/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/refresh/DoricSwipeLayout.java
new file mode 100644
index 00000000..411f13ef
--- /dev/null
+++ b/Android/doric/src/main/java/pub/doric/refresh/DoricSwipeLayout.java
@@ -0,0 +1,958 @@
+package pub.doric.refresh;
+
+import android.animation.Animator;
+import android.animation.ValueAnimator;
+import android.content.Context;
+import android.content.res.TypedArray;
+import android.util.AttributeSet;
+import android.util.DisplayMetrics;
+import android.util.Log;
+import android.view.MotionEvent;
+import android.view.View;
+import android.view.ViewConfiguration;
+import android.view.ViewGroup;
+import android.view.animation.Animation;
+import android.view.animation.DecelerateInterpolator;
+import android.view.animation.Transformation;
+import android.widget.AbsListView;
+import android.widget.ListView;
+
+import androidx.annotation.NonNull;
+import androidx.annotation.Nullable;
+import androidx.annotation.VisibleForTesting;
+import androidx.core.view.NestedScrollingChild;
+import androidx.core.view.NestedScrollingChildHelper;
+import androidx.core.view.NestedScrollingParent;
+import androidx.core.view.NestedScrollingParentHelper;
+import androidx.core.view.ViewCompat;
+import androidx.core.widget.ListViewCompat;
+import androidx.swiperefreshlayout.widget.CircularProgressDrawable;
+
+import android.view.animation.Animation.AnimationListener;
+
+/**
+ * @Description: pub.doric.pullable
+ * @Author: pengfei.zhou
+ * @CreateDate: 2019-11-25
+ */
+public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent,
+ NestedScrollingChild {
+ // Maps to ProgressBar.Large style
+ public static final int LARGE = CircularProgressDrawable.LARGE;
+ // Maps to ProgressBar default style
+ public static final int DEFAULT = CircularProgressDrawable.DEFAULT;
+
+ public static final int DEFAULT_SLINGSHOT_DISTANCE = -1;
+
+ @VisibleForTesting
+ static final int CIRCLE_DIAMETER = 40;
+ @VisibleForTesting
+ static final int CIRCLE_DIAMETER_LARGE = 56;
+
+ private static final String LOG_TAG = DoricSwipeLayout.class.getSimpleName();
+
+ private static final int MAX_ALPHA = 255;
+ private static final int STARTING_PROGRESS_ALPHA = (int) (.3f * MAX_ALPHA);
+
+ private static final float DECELERATE_INTERPOLATION_FACTOR = 2f;
+ private static final int INVALID_POINTER = -1;
+ private static final float DRAG_RATE = .5f;
+
+ // Max amount of circle that can be filled by progress during swipe gesture,
+ // where 1.0 is a full circle
+ private static final float MAX_PROGRESS_ANGLE = .8f;
+
+ private static final int SCALE_DOWN_DURATION = 150;
+
+ private static final int ALPHA_ANIMATION_DURATION = 300;
+
+ private static final int ANIMATE_TO_TRIGGER_DURATION = 200;
+
+ private static final int ANIMATE_TO_START_DURATION = 200;
+
+ // Default offset in dips from the top of the view to where the progress spinner should stop
+ private static final int DEFAULT_CIRCLE_TARGET = 64;
+
+ private View mTarget; // the target of the gesture
+ OnRefreshListener mListener;
+ boolean mRefreshing = false;
+ private int mTouchSlop;
+ private float mTotalDragDistance = -1;
+
+ // If nested scrolling is enabled, the total amount that needed to be
+ // consumed by this as the nested scrolling parent is used in place of the
+ // overscroll determined by MOVE events in the onTouch handler
+ private float mTotalUnconsumed;
+ private final NestedScrollingParentHelper mNestedScrollingParentHelper;
+ private final NestedScrollingChildHelper mNestedScrollingChildHelper;
+ private final int[] mParentScrollConsumed = new int[2];
+ private final int[] mParentOffsetInWindow = new int[2];
+ private boolean mNestedScrollInProgress;
+
+ private int mMediumAnimationDuration;
+ int mCurrentTargetOffsetTop;
+
+ private float mInitialMotionY;
+ private float mInitialDownY;
+ private boolean mIsBeingDragged;
+ private int mActivePointerId = INVALID_POINTER;
+
+ // Target is returning to its start offset because it was cancelled or a
+ // refresh was triggered.
+ private boolean mReturningToStart;
+ private final DecelerateInterpolator mDecelerateInterpolator;
+ private static final int[] LAYOUT_ATTRS = new int[]{
+ android.R.attr.enabled
+ };
+
+ private int mCircleViewIndex = -1;
+
+ protected int mFrom;
+
+ float mStartingScale;
+
+ protected int mOriginalOffsetTop;
+
+ int mSpinnerOffsetEnd;
+
+ int mCustomSlingshotDistance;
+
+ private Animation mScaleAnimation;
+
+ private Animation mScaleDownAnimation;
+
+ private Animation mScaleDownToStartAnimation;
+
+ boolean mNotify;
+
+ // Whether the client has set a custom starting position;
+ boolean mUsingCustomStart;
+
+ private OnChildScrollUpCallback mChildScrollUpCallback;
+
+ private DoricRefreshView mRefreshView;
+ private AnimationListener mRefreshListener = new AnimationListener() {
+ @Override
+ public void onAnimationStart(Animation animation) {
+ }
+
+ @Override
+ public void onAnimationRepeat(Animation animation) {
+ }
+
+ @Override
+ public void onAnimationEnd(Animation animation) {
+ if (mRefreshing) {
+ mRefreshView.startAnimation();
+ if (mNotify) {
+ if (mListener != null) {
+ mListener.onRefresh();
+ }
+ }
+ mCurrentTargetOffsetTop = mRefreshView.getTop();
+ } else {
+ reset();
+ }
+ }
+ };
+ private int mPullDownHeight = 0;
+ private ValueAnimator headerViewAnimator;
+
+ void reset() {
+ mRefreshing = false;
+ if (headerViewAnimator != null && headerViewAnimator.isRunning()) {
+ headerViewAnimator.cancel();
+ }
+ headerViewAnimator = ValueAnimator.ofInt(mRefreshView.getBottom(), 0);
+ headerViewAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
+ @Override
+ public void onAnimationUpdate(ValueAnimator animation) {
+ mCurrentTargetOffsetTop = (int) animation.getAnimatedValue()
+ - mRefreshView.getMeasuredHeight();
+ mRefreshView.requestLayout();
+ }
+ });
+ headerViewAnimator.addListener(new Animator.AnimatorListener() {
+ @Override
+ public void onAnimationStart(Animator animation) {
+
+ }
+
+ @Override
+ public void onAnimationEnd(Animator animation) {
+ mRefreshView.stopAnimation();
+ mRefreshView.setVisibility(View.GONE);
+ // Return the circle to its start position
+
+ setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCurrentTargetOffsetTop);
+ mCurrentTargetOffsetTop = mRefreshView.getTop();
+ }
+
+ @Override
+ public void onAnimationCancel(Animator animation) {
+
+ }
+
+ @Override
+ public void onAnimationRepeat(Animator animation) {
+
+ }
+ });
+ headerViewAnimator.setDuration(SCALE_DOWN_DURATION);
+ headerViewAnimator.start();
+ }
+
+ @Override
+ public void setEnabled(boolean enabled) {
+ super.setEnabled(enabled);
+ if (!enabled) {
+ reset();
+ }
+ }
+
+ @Override
+ protected void onDetachedFromWindow() {
+ super.onDetachedFromWindow();
+ reset();
+ }
+
+ /**
+ * Simple constructor to use when creating a SwipeRefreshLayout from code.
+ *
+ * @param context
+ */
+ public DoricSwipeLayout(@NonNull Context context) {
+ this(context, null);
+ }
+
+ /**
+ * Constructor that is called when inflating SwipeRefreshLayout from XML.
+ *
+ * @param context
+ * @param attrs
+ */
+ public DoricSwipeLayout(@NonNull Context context, @Nullable AttributeSet attrs) {
+ super(context, attrs);
+
+ mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
+
+ mMediumAnimationDuration = getResources().getInteger(
+ android.R.integer.config_mediumAnimTime);
+
+ setWillNotDraw(false);
+ mDecelerateInterpolator = new DecelerateInterpolator(DECELERATE_INTERPOLATION_FACTOR);
+
+ final DisplayMetrics metrics = getResources().getDisplayMetrics();
+
+ createProgressView();
+ setChildrenDrawingOrderEnabled(true);
+ // the absolute offset has to take into account that the circle starts at an offset
+ mSpinnerOffsetEnd = (int) (DEFAULT_CIRCLE_TARGET * metrics.density);
+ mTotalDragDistance = mSpinnerOffsetEnd;
+ mNestedScrollingParentHelper = new NestedScrollingParentHelper(this);
+
+ mNestedScrollingChildHelper = new NestedScrollingChildHelper(this);
+ setNestedScrollingEnabled(true);
+
+ moveToStart(1.0f);
+
+ final TypedArray a = context.obtainStyledAttributes(attrs, LAYOUT_ATTRS);
+ setEnabled(a.getBoolean(0, true));
+ a.recycle();
+ }
+
+ public void setPullDownHeight(int height) {
+ mPullDownHeight = height;
+ mOriginalOffsetTop = mCurrentTargetOffsetTop = -height;
+ mSpinnerOffsetEnd = height;
+ mTotalDragDistance = height;
+ }
+
+ @Override
+ protected int getChildDrawingOrder(int childCount, int i) {
+ if (mCircleViewIndex < 0) {
+ return i;
+ } else if (i == childCount - 1) {
+ // Draw the selected child last
+ return mCircleViewIndex;
+ } else if (i >= mCircleViewIndex) {
+ // Move the children after the selected child earlier one
+ return i + 1;
+ } else {
+ // Keep the children before the selected child the same
+ return i;
+ }
+ }
+
+ private void createProgressView() {
+ mRefreshView = new DoricRefreshView(getContext());
+ ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
+ addView(mRefreshView, layoutParams);
+ }
+
+ public DoricRefreshView getRefreshView() {
+ return mRefreshView;
+ }
+
+ /**
+ * Set the listener to be notified when a refresh is triggered via the swipe
+ * gesture.
+ */
+ public void setOnRefreshListener(@Nullable OnRefreshListener listener) {
+ mListener = listener;
+ }
+
+ /**
+ * Notify the widget that refresh state has changed. Do not call this when
+ * refresh is triggered by a swipe gesture.
+ *
+ * @param refreshing Whether or not the view should show refresh progress.
+ */
+ public void setRefreshing(boolean refreshing) {
+ if (refreshing && mRefreshing != refreshing) {
+ // scale and show
+ mRefreshing = refreshing;
+ int endTarget = 0;
+ if (!mUsingCustomStart) {
+ endTarget = mSpinnerOffsetEnd + mOriginalOffsetTop;
+ } else {
+ endTarget = mSpinnerOffsetEnd;
+ }
+ setTargetOffsetTopAndBottom(endTarget - mCurrentTargetOffsetTop);
+ mNotify = false;
+ startScaleUpAnimation(mRefreshListener);
+ } else {
+ setRefreshing(refreshing, false /* notify */);
+ }
+ }
+
+ private void startScaleUpAnimation(AnimationListener listener) {
+ mRefreshView.setVisibility(View.VISIBLE);
+ mScaleAnimation = new Animation() {
+ @Override
+ public void applyTransformation(float interpolatedTime, Transformation t) {
+ setAnimationProgress(interpolatedTime);
+ }
+ };
+ mScaleAnimation.setDuration(mMediumAnimationDuration);
+ if (listener != null) {
+ mRefreshView.setAnimationListener(listener);
+ }
+ mRefreshView.clearAnimation();
+ mRefreshView.startAnimation(mScaleAnimation);
+ }
+
+ /**
+ * Pre API 11, this does an alpha animation.
+ *
+ * @param progress
+ */
+ void setAnimationProgress(float progress) {
+
+ }
+
+ private void setRefreshing(boolean refreshing, final boolean notify) {
+ if (mRefreshing != refreshing) {
+ mNotify = notify;
+ ensureTarget();
+ mRefreshing = refreshing;
+ if (mRefreshing) {
+ animateOffsetToCorrectPosition(mCurrentTargetOffsetTop, mRefreshListener);
+ } else {
+ startScaleDownAnimation(mRefreshListener);
+ }
+ }
+ }
+
+ void startScaleDownAnimation(Animation.AnimationListener listener) {
+ mScaleDownAnimation = new Animation() {
+ @Override
+ public void applyTransformation(float interpolatedTime, Transformation t) {
+ setAnimationProgress(1 - interpolatedTime);
+ }
+ };
+ mScaleDownAnimation.setDuration(SCALE_DOWN_DURATION);
+ mRefreshView.setAnimationListener(listener);
+ mRefreshView.clearAnimation();
+ mRefreshView.startAnimation(mScaleDownAnimation);
+ }
+
+ /**
+ * @return Whether the SwipeRefreshWidget is actively showing refresh
+ * progress.
+ */
+ public boolean isRefreshing() {
+ return mRefreshing;
+ }
+
+ private void ensureTarget() {
+ // Don't bother getting the parent height if the parent hasn't been laid
+ // out yet.
+ if (mTarget == null) {
+ for (int i = 0; i < getChildCount(); i++) {
+ View child = getChildAt(i);
+ if (!child.equals(mRefreshView)) {
+ mTarget = child;
+ break;
+ }
+ }
+ }
+ }
+
+ @Override
+ protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
+ final int width = getMeasuredWidth();
+ final int height = getMeasuredHeight();
+ if (getChildCount() == 0) {
+ return;
+ }
+ if (mTarget == null) {
+ ensureTarget();
+ }
+ if (mTarget == null) {
+ return;
+ }
+
+ int circleWidth = mRefreshView.getMeasuredWidth();
+ int circleHeight = mRefreshView.getMeasuredHeight();
+
+ mRefreshView.layout((width / 2 - circleWidth / 2), mCurrentTargetOffsetTop,
+ (width / 2 + circleWidth / 2), mCurrentTargetOffsetTop + circleHeight);
+
+ final View child = mTarget;
+ final int childLeft = getPaddingLeft();
+ final int childTop = getPaddingTop() + mRefreshView.getBottom();
+ final int childWidth = width - getPaddingLeft() - getPaddingRight();
+ final int childHeight = height - getPaddingTop() - getPaddingBottom();
+ child.layout(childLeft, childTop, childLeft + childWidth, childTop + childHeight);
+ }
+
+ @Override
+ public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
+ super.onMeasure(widthMeasureSpec, heightMeasureSpec);
+ if (mTarget == null) {
+ ensureTarget();
+ }
+ if (mTarget == null) {
+ return;
+ }
+ mTarget.measure(MeasureSpec.makeMeasureSpec(
+ getMeasuredWidth() - getPaddingLeft() - getPaddingRight(),
+ MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec(
+ getMeasuredHeight() - getPaddingTop() - getPaddingBottom(), MeasureSpec.EXACTLY));
+ mRefreshView.measure(
+ MeasureSpec.makeMeasureSpec(
+ getMeasuredWidth() - getPaddingLeft() - getPaddingRight(),
+ MeasureSpec.EXACTLY),
+ MeasureSpec.makeMeasureSpec(
+ (getMeasuredHeight() - getPaddingTop() - getPaddingBottom()) / 3,
+ MeasureSpec.UNSPECIFIED));
+ if (mPullDownHeight != mRefreshView.getMeasuredHeight()) {
+ setPullDownHeight(mRefreshView.getMeasuredHeight());
+ }
+ mCircleViewIndex = -1;
+ // Get the index of the circleview.
+ for (int index = 0; index < getChildCount(); index++) {
+ if (getChildAt(index) == mRefreshView) {
+ mCircleViewIndex = index;
+ break;
+ }
+ }
+ }
+
+ /**
+ * @return Whether it is possible for the child view of this layout to
+ * scroll up. Override this if the child view is a custom view.
+ */
+ public boolean canChildScrollUp() {
+ if (mChildScrollUpCallback != null) {
+ return mChildScrollUpCallback.canChildScrollUp(this, mTarget);
+ }
+ if (mTarget instanceof ListView) {
+ return ListViewCompat.canScrollList((ListView) mTarget, -1);
+ }
+ return mTarget.canScrollVertically(-1);
+ }
+
+ /**
+ * Set a callback to override {@link androidx.swiperefreshlayout.widget.SwipeRefreshLayout#canChildScrollUp()} method. Non-null
+ * callback will return the value provided by the callback and ignore all internal logic.
+ *
+ * @param callback Callback that should be called when canChildScrollUp() is called.
+ */
+ public void setOnChildScrollUpCallback(@Nullable OnChildScrollUpCallback callback) {
+ mChildScrollUpCallback = callback;
+ }
+
+ @Override
+ public boolean onInterceptTouchEvent(MotionEvent ev) {
+ ensureTarget();
+
+ final int action = ev.getActionMasked();
+ int pointerIndex;
+
+ if (mReturningToStart && action == MotionEvent.ACTION_DOWN) {
+ mReturningToStart = false;
+ }
+
+ if (!isEnabled() || mReturningToStart || canChildScrollUp()
+ || mRefreshing || mNestedScrollInProgress) {
+ // Fail fast if we're not in a state where a swipe is possible
+ return false;
+ }
+
+ switch (action) {
+ case MotionEvent.ACTION_DOWN:
+ setTargetOffsetTopAndBottom(mOriginalOffsetTop - mRefreshView.getTop());
+ mActivePointerId = ev.getPointerId(0);
+ mIsBeingDragged = false;
+
+ pointerIndex = ev.findPointerIndex(mActivePointerId);
+ if (pointerIndex < 0) {
+ return false;
+ }
+ mInitialDownY = ev.getY(pointerIndex);
+ break;
+
+ case MotionEvent.ACTION_MOVE:
+ if (mActivePointerId == INVALID_POINTER) {
+ Log.e(LOG_TAG, "Got ACTION_MOVE event but don't have an active pointer id.");
+ return false;
+ }
+
+ pointerIndex = ev.findPointerIndex(mActivePointerId);
+ if (pointerIndex < 0) {
+ return false;
+ }
+ final float y = ev.getY(pointerIndex);
+ startDragging(y);
+ break;
+
+ case MotionEvent.ACTION_POINTER_UP:
+ onSecondaryPointerUp(ev);
+ break;
+
+ case MotionEvent.ACTION_UP:
+ case MotionEvent.ACTION_CANCEL:
+ mIsBeingDragged = false;
+ mActivePointerId = INVALID_POINTER;
+ break;
+ }
+
+ return mIsBeingDragged;
+ }
+
+ @Override
+ public void requestDisallowInterceptTouchEvent(boolean b) {
+ // if this is a List < L or another view that doesn't support nested
+ // scrolling, ignore this request so that the vertical scroll event
+ // isn't stolen
+ if ((android.os.Build.VERSION.SDK_INT < 21 && mTarget instanceof AbsListView)
+ || (mTarget != null && !ViewCompat.isNestedScrollingEnabled(mTarget))) {
+ // Nope.
+ } else {
+ super.requestDisallowInterceptTouchEvent(b);
+ }
+ }
+
+ // NestedScrollingParent
+
+ @Override
+ public boolean onStartNestedScroll(View child, View target, int nestedScrollAxes) {
+ return isEnabled() && !mReturningToStart && !mRefreshing
+ && (nestedScrollAxes & ViewCompat.SCROLL_AXIS_VERTICAL) != 0;
+ }
+
+ @Override
+ public void onNestedScrollAccepted(View child, View target, int axes) {
+ // Reset the counter of how much leftover scroll needs to be consumed.
+ mNestedScrollingParentHelper.onNestedScrollAccepted(child, target, axes);
+ // Dispatch up to the nested parent
+ startNestedScroll(axes & ViewCompat.SCROLL_AXIS_VERTICAL);
+ mTotalUnconsumed = 0;
+ mNestedScrollInProgress = true;
+ }
+
+ @Override
+ public void onNestedPreScroll(View target, int dx, int dy, int[] consumed) {
+ // If we are in the middle of consuming, a scroll, then we want to move the spinner back up
+ // before allowing the list to scroll
+ if (dy > 0 && mTotalUnconsumed > 0) {
+ if (dy > mTotalUnconsumed) {
+ consumed[1] = dy - (int) mTotalUnconsumed;
+ mTotalUnconsumed = 0;
+ } else {
+ if (dy > 3) {
+ mTotalUnconsumed -= dy;
+ consumed[1] = dy;
+ }
+ }
+ moveSpinner(mTotalUnconsumed);
+ }
+
+ // If a client layout is using a custom start position for the circle
+ // view, they mean to hide it again before scrolling the child view
+ // If we get back to mTotalUnconsumed == 0 and there is more to go, hide
+ // the circle so it isn't exposed if its blocking content is moved
+ if (mUsingCustomStart && dy > 0 && mTotalUnconsumed == 0
+ && Math.abs(dy - consumed[1]) > 0) {
+ mRefreshView.setVisibility(View.GONE);
+ }
+
+ // Now let our nested parent consume the leftovers
+ final int[] parentConsumed = mParentScrollConsumed;
+ if (dispatchNestedPreScroll(dx - consumed[0], dy - consumed[1], parentConsumed, null)) {
+ consumed[0] += parentConsumed[0];
+ consumed[1] += parentConsumed[1];
+ }
+ }
+
+ @Override
+ public int getNestedScrollAxes() {
+ return mNestedScrollingParentHelper.getNestedScrollAxes();
+ }
+
+ @Override
+ public void onStopNestedScroll(View target) {
+ mNestedScrollingParentHelper.onStopNestedScroll(target);
+ mNestedScrollInProgress = false;
+ // Finish the spinner for nested scrolling if we ever consumed any
+ // unconsumed nested scroll
+ if (mTotalUnconsumed > 0) {
+ finishSpinner(mTotalUnconsumed);
+ mTotalUnconsumed = 0;
+ }
+ // Dispatch up our nested parent
+ stopNestedScroll();
+ }
+
+ @Override
+ public void onNestedScroll(final View target, final int dxConsumed, final int dyConsumed,
+ final int dxUnconsumed, final int dyUnconsumed) {
+ // Dispatch up to the nested parent first
+ dispatchNestedScroll(dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed,
+ mParentOffsetInWindow);
+
+ // This is a bit of a hack. Nested scrolling works from the bottom up, and as we are
+ // sometimes between two nested scrolling views, we need a way to be able to know when any
+ // nested scrolling parent has stopped handling events. We do that by using the
+ // 'offset in window 'functionality to see if we have been moved from the event.
+ // This is a decent indication of whether we should take over the event stream or not.
+ final int dy = dyUnconsumed + mParentOffsetInWindow[1];
+ if (dy < 0 && !canChildScrollUp()) {
+ mTotalUnconsumed += Math.abs(dy);
+ moveSpinner(mTotalUnconsumed);
+ }
+ }
+
+ // NestedScrollingChild
+
+ @Override
+ public void setNestedScrollingEnabled(boolean enabled) {
+ mNestedScrollingChildHelper.setNestedScrollingEnabled(enabled);
+ }
+
+ @Override
+ public boolean isNestedScrollingEnabled() {
+ return mNestedScrollingChildHelper.isNestedScrollingEnabled();
+ }
+
+ @Override
+ public boolean startNestedScroll(int axes) {
+ return mNestedScrollingChildHelper.startNestedScroll(axes);
+ }
+
+ @Override
+ public void stopNestedScroll() {
+ mNestedScrollingChildHelper.stopNestedScroll();
+ }
+
+ @Override
+ public boolean hasNestedScrollingParent() {
+ return mNestedScrollingChildHelper.hasNestedScrollingParent();
+ }
+
+ @Override
+ public boolean dispatchNestedScroll(int dxConsumed, int dyConsumed, int dxUnconsumed,
+ int dyUnconsumed, int[] offsetInWindow) {
+ return mNestedScrollingChildHelper.dispatchNestedScroll(dxConsumed, dyConsumed,
+ dxUnconsumed, dyUnconsumed, offsetInWindow);
+ }
+
+ @Override
+ public boolean dispatchNestedPreScroll(int dx, int dy, int[] consumed, int[] offsetInWindow) {
+ return mNestedScrollingChildHelper.dispatchNestedPreScroll(
+ dx, dy, consumed, offsetInWindow);
+ }
+
+ @Override
+ public boolean onNestedPreFling(View target, float velocityX,
+ float velocityY) {
+ return dispatchNestedPreFling(velocityX, velocityY);
+ }
+
+ @Override
+ public boolean onNestedFling(View target, float velocityX, float velocityY,
+ boolean consumed) {
+ return dispatchNestedFling(velocityX, velocityY, consumed);
+ }
+
+ @Override
+ public boolean dispatchNestedFling(float velocityX, float velocityY, boolean consumed) {
+ return mNestedScrollingChildHelper.dispatchNestedFling(velocityX, velocityY, consumed);
+ }
+
+ @Override
+ public boolean dispatchNestedPreFling(float velocityX, float velocityY) {
+ return mNestedScrollingChildHelper.dispatchNestedPreFling(velocityX, velocityY);
+ }
+
+ private boolean isAnimationRunning(Animation animation) {
+ return animation != null && animation.hasStarted() && !animation.hasEnded();
+ }
+
+ private void moveSpinner(float overscrollTop) {
+ float originalDragPercent = overscrollTop / mTotalDragDistance;
+
+ float dragPercent = Math.min(1f, Math.abs(originalDragPercent));
+ float extraOS = Math.abs(overscrollTop) - mTotalDragDistance;
+ float slingshotDist = mCustomSlingshotDistance > 0
+ ? mCustomSlingshotDistance
+ : (mUsingCustomStart
+ ? mSpinnerOffsetEnd - mOriginalOffsetTop
+ : mSpinnerOffsetEnd);
+ float tensionSlingshotPercent = Math.max(0, Math.min(extraOS, slingshotDist * 2)
+ / slingshotDist);
+ float tensionPercent = (float) ((tensionSlingshotPercent / 4) - Math.pow(
+ (tensionSlingshotPercent / 4), 2)) * 2f;
+ float extraMove = (slingshotDist) * tensionPercent * 2;
+
+ int targetY = mOriginalOffsetTop + (int) ((slingshotDist * dragPercent) + extraMove);
+ // where 1.0f is a full circle
+ if (mRefreshView.getVisibility() != View.VISIBLE) {
+ mRefreshView.setVisibility(View.VISIBLE);
+ }
+ mRefreshView.setScaleX(1f);
+ mRefreshView.setScaleY(1f);
+
+ setTargetOffsetTopAndBottom(targetY - mCurrentTargetOffsetTop);
+ }
+
+ private void finishSpinner(float overscrollTop) {
+ if (overscrollTop > mTotalDragDistance) {
+ setRefreshing(true, true /* notify */);
+ } else {
+ // cancel refresh
+ mRefreshing = false;
+ Animation.AnimationListener listener = null;
+ listener = new Animation.AnimationListener() {
+
+ @Override
+ public void onAnimationStart(Animation animation) {
+ }
+
+ @Override
+ public void onAnimationEnd(Animation animation) {
+ startScaleDownAnimation(null);
+ }
+
+ @Override
+ public void onAnimationRepeat(Animation animation) {
+ }
+
+ };
+ animateOffsetToStartPosition(mCurrentTargetOffsetTop, listener);
+ }
+ }
+
+ @Override
+ public boolean onTouchEvent(MotionEvent ev) {
+ final int action = ev.getActionMasked();
+ int pointerIndex = -1;
+
+ if (mReturningToStart && action == MotionEvent.ACTION_DOWN) {
+ mReturningToStart = false;
+ }
+
+ if (!isEnabled() || mReturningToStart || canChildScrollUp()
+ || mRefreshing || mNestedScrollInProgress) {
+ // Fail fast if we're not in a state where a swipe is possible
+ return false;
+ }
+
+ switch (action) {
+ case MotionEvent.ACTION_DOWN:
+ mActivePointerId = ev.getPointerId(0);
+ mIsBeingDragged = false;
+ break;
+
+ case MotionEvent.ACTION_MOVE: {
+ pointerIndex = ev.findPointerIndex(mActivePointerId);
+ if (pointerIndex < 0) {
+ Log.e(LOG_TAG, "Got ACTION_MOVE event but have an invalid active pointer id.");
+ return false;
+ }
+
+ final float y = ev.getY(pointerIndex);
+ startDragging(y);
+
+ if (mIsBeingDragged) {
+ final float overscrollTop = (y - mInitialMotionY) * DRAG_RATE;
+ if (overscrollTop > 0) {
+ moveSpinner(overscrollTop);
+ } else {
+ return false;
+ }
+ }
+ break;
+ }
+ case MotionEvent.ACTION_POINTER_DOWN: {
+ pointerIndex = ev.getActionIndex();
+ if (pointerIndex < 0) {
+ Log.e(LOG_TAG,
+ "Got ACTION_POINTER_DOWN event but have an invalid action index.");
+ return false;
+ }
+ mActivePointerId = ev.getPointerId(pointerIndex);
+ break;
+ }
+
+ case MotionEvent.ACTION_POINTER_UP:
+ onSecondaryPointerUp(ev);
+ break;
+
+ case MotionEvent.ACTION_UP: {
+ pointerIndex = ev.findPointerIndex(mActivePointerId);
+ if (pointerIndex < 0) {
+ Log.e(LOG_TAG, "Got ACTION_UP event but don't have an active pointer id.");
+ return false;
+ }
+
+ if (mIsBeingDragged) {
+ final float y = ev.getY(pointerIndex);
+ final float overscrollTop = (y - mInitialMotionY) * DRAG_RATE;
+ mIsBeingDragged = false;
+ finishSpinner(overscrollTop);
+ }
+ mActivePointerId = INVALID_POINTER;
+ return false;
+ }
+ case MotionEvent.ACTION_CANCEL:
+ return false;
+ }
+
+ return true;
+ }
+
+ private void startDragging(float y) {
+ final float yDiff = y - mInitialDownY;
+ if (yDiff > mTouchSlop && !mIsBeingDragged) {
+ mInitialMotionY = mInitialDownY + mTouchSlop;
+ mIsBeingDragged = true;
+ }
+ }
+
+ private void animateOffsetToCorrectPosition(int from, AnimationListener listener) {
+ mFrom = from;
+ mAnimateToCorrectPosition.reset();
+ mAnimateToCorrectPosition.setDuration(ANIMATE_TO_TRIGGER_DURATION);
+ mAnimateToCorrectPosition.setInterpolator(mDecelerateInterpolator);
+ if (listener != null) {
+ mRefreshView.setAnimationListener(listener);
+ }
+ mRefreshView.clearAnimation();
+ mRefreshView.startAnimation(mAnimateToCorrectPosition);
+ }
+
+ private void animateOffsetToStartPosition(int from, AnimationListener listener) {
+ mFrom = from;
+ mAnimateToStartPosition.reset();
+ mAnimateToStartPosition.setDuration(ANIMATE_TO_START_DURATION);
+ mAnimateToStartPosition.setInterpolator(mDecelerateInterpolator);
+ if (listener != null) {
+ mRefreshView.setAnimationListener(listener);
+ }
+ mRefreshView.clearAnimation();
+ mRefreshView.startAnimation(mAnimateToStartPosition);
+ }
+
+ private final Animation mAnimateToCorrectPosition = new Animation() {
+ @Override
+ public void applyTransformation(float interpolatedTime, Transformation t) {
+ int targetTop = 0;
+ int endTarget = 0;
+ if (!mUsingCustomStart) {
+ endTarget = mSpinnerOffsetEnd - Math.abs(mOriginalOffsetTop);
+ } else {
+ endTarget = mSpinnerOffsetEnd;
+ }
+ targetTop = (mFrom + (int) ((endTarget - mFrom) * interpolatedTime));
+ int offset = targetTop - mRefreshView.getTop();
+ setTargetOffsetTopAndBottom(offset);
+ }
+ };
+
+ void moveToStart(float interpolatedTime) {
+ int targetTop = 0;
+ targetTop = (mFrom + (int) ((mOriginalOffsetTop - mFrom) * interpolatedTime));
+ int offset = targetTop - mRefreshView.getTop();
+ setTargetOffsetTopAndBottom(offset);
+ }
+
+ private final Animation mAnimateToStartPosition = new Animation() {
+ @Override
+ public void applyTransformation(float interpolatedTime, Transformation t) {
+ moveToStart(interpolatedTime);
+ }
+ };
+
+
+ void setTargetOffsetTopAndBottom(int offset) {
+ mRefreshView.bringToFront();
+ ViewCompat.offsetTopAndBottom(mRefreshView, offset);
+ mCurrentTargetOffsetTop = mRefreshView.getTop();
+ if (mRefreshView.getMeasuredHeight() > 0) {
+ mRefreshView.setProgressRotation((float) mRefreshView.getBottom() / (float) mRefreshView.getMeasuredHeight());
+ }
+ }
+
+ private void onSecondaryPointerUp(MotionEvent ev) {
+ final int pointerIndex = ev.getActionIndex();
+ final int pointerId = ev.getPointerId(pointerIndex);
+ if (pointerId == mActivePointerId) {
+ // This was our active pointer going up. Choose a new
+ // active pointer and adjust accordingly.
+ final int newPointerIndex = pointerIndex == 0 ? 1 : 0;
+ mActivePointerId = ev.getPointerId(newPointerIndex);
+ }
+ }
+
+ /**
+ * Classes that wish to be notified when the swipe gesture correctly
+ * triggers a refresh should implement this interface.
+ */
+ public interface OnRefreshListener {
+ /**
+ * Called when a swipe gesture triggers a refresh.
+ */
+ void onRefresh();
+ }
+
+ /**
+ * Classes that wish to override {@link androidx.swiperefreshlayout.widget.SwipeRefreshLayout#canChildScrollUp()} method
+ * behavior should implement this interface.
+ */
+ public interface OnChildScrollUpCallback {
+ /**
+ * Callback that will be called when {@link androidx.swiperefreshlayout.widget.SwipeRefreshLayout#canChildScrollUp()} method
+ * is called to allow the implementer to override its behavior.
+ *
+ * @param parent SwipeRefreshLayout that this callback is overriding.
+ * @param child The child view of SwipeRefreshLayout.
+ * @return Whether it is possible for the child view of parent layout to scroll up.
+ */
+ boolean canChildScrollUp(@NonNull DoricSwipeLayout parent, @Nullable View child);
+ }
+
+
+}
\ No newline at end of file
diff --git a/Android/doric/src/main/java/pub/doric/refresh/PullingListener.java b/Android/doric/src/main/java/pub/doric/refresh/PullingListener.java
new file mode 100644
index 00000000..c09ab2c4
--- /dev/null
+++ b/Android/doric/src/main/java/pub/doric/refresh/PullingListener.java
@@ -0,0 +1,20 @@
+package pub.doric.refresh;
+
+/**
+ * @Description: pub.doric.pullable
+ * @Author: pengfei.zhou
+ * @CreateDate: 2019-11-25
+ */
+public interface PullingListener {
+
+ void startAnimation();
+
+ void stopAnimation();
+
+ /**
+ * Set the amount of rotation to apply to the progress spinner.
+ *
+ * @param rotation Rotation is from [0..2]
+ */
+ void setProgressRotation(float rotation);
+}
diff --git a/Android/doric/src/main/java/pub/doric/refresh/RefreshableNode.java b/Android/doric/src/main/java/pub/doric/refresh/RefreshableNode.java
new file mode 100644
index 00000000..c6f1d1ce
--- /dev/null
+++ b/Android/doric/src/main/java/pub/doric/refresh/RefreshableNode.java
@@ -0,0 +1,196 @@
+package pub.doric.refresh;
+
+import com.github.pengfeizhou.jscore.JSObject;
+import com.github.pengfeizhou.jscore.JSValue;
+import com.github.pengfeizhou.jscore.JavaValue;
+
+import pub.doric.DoricContext;
+import pub.doric.extension.bridge.DoricMethod;
+import pub.doric.extension.bridge.DoricPlugin;
+import pub.doric.extension.bridge.DoricPromise;
+import pub.doric.shader.SuperNode;
+import pub.doric.shader.ViewNode;
+
+/**
+ * @Description: pub.doric.pullable
+ * @Author: pengfei.zhou
+ * @CreateDate: 2019-11-26
+ */
+@DoricPlugin(name = "Refreshable")
+public class RefreshableNode extends SuperNode implements PullingListener {
+
+ private String mContentViewId;
+ private ViewNode mContentNode;
+
+ private String mHeaderViewId;
+ private ViewNode mHeaderNode;
+
+ public RefreshableNode(DoricContext doricContext) {
+ super(doricContext);
+ }
+
+
+ @Override
+ protected DoricSwipeLayout build() {
+ DoricSwipeLayout doricSwipeLayout = new DoricSwipeLayout(getContext());
+ doricSwipeLayout.getRefreshView().setPullingListenr(this);
+ return doricSwipeLayout;
+ }
+
+ @Override
+ protected void blend(DoricSwipeLayout view, String name, JSValue prop) {
+ if ("content".equals(name)) {
+ mContentViewId = prop.asString().value();
+ } else if ("header".equals(name)) {
+ mHeaderViewId = prop.asString().value();
+ } else if ("onRefresh".equals(name)) {
+ final String funcId = prop.asString().value();
+ mView.setOnRefreshListener(new DoricSwipeLayout.OnRefreshListener() {
+ @Override
+ public void onRefresh() {
+ callJSResponse(funcId);
+ }
+ });
+ } else {
+ super.blend(view, name, prop);
+ }
+ }
+
+ @Override
+ public void blend(JSObject jsObject) {
+ super.blend(jsObject);
+ blendContentNode();
+ blendHeadNode();
+ }
+
+
+ private void blendContentNode() {
+ JSObject contentModel = getSubModel(mContentViewId);
+ if (contentModel == null) {
+ return;
+ }
+ String viewId = contentModel.getProperty("id").asString().value();
+ String type = contentModel.getProperty("type").asString().value();
+ JSObject props = contentModel.getProperty("props").asObject();
+ if (mContentNode != null) {
+ if (mContentNode.getId().equals(viewId)) {
+ //skip
+ } else {
+ if (mReusable && type.equals(mContentNode.getType())) {
+ mContentNode.setId(viewId);
+ mContentNode.blend(props);
+ } else {
+ mView.removeAllViews();
+ mContentNode = ViewNode.create(getDoricContext(), type);
+ mContentNode.setId(viewId);
+ mContentNode.init(this);
+ mContentNode.blend(props);
+ mView.addView(mContentNode.getDoricLayer());
+ }
+ }
+ } else {
+ mContentNode = ViewNode.create(getDoricContext(), type);
+ mContentNode.setId(viewId);
+ mContentNode.init(this);
+ mContentNode.blend(props);
+ mView.addView(mContentNode.getDoricLayer());
+ }
+ }
+
+ private void blendHeadNode() {
+ JSObject headerModel = getSubModel(mHeaderViewId);
+ if (headerModel == null) {
+ return;
+ }
+ String viewId = headerModel.getProperty("id").asString().value();
+ String type = headerModel.getProperty("type").asString().value();
+ JSObject props = headerModel.getProperty("props").asObject();
+ if (mHeaderNode != null) {
+ if (mHeaderNode.getId().equals(viewId)) {
+ //skip
+ } else {
+ if (mReusable && type.equals(mHeaderNode.getType())) {
+ mHeaderNode.setId(viewId);
+ mHeaderNode.blend(props);
+ } else {
+ mHeaderNode = ViewNode.create(getDoricContext(), type);
+ mHeaderNode.setId(viewId);
+ mHeaderNode.init(this);
+ mHeaderNode.blend(props);
+ mView.getRefreshView().setContent(mHeaderNode.getDoricLayer());
+ }
+ }
+ } else {
+ mHeaderNode = ViewNode.create(getDoricContext(), type);
+ mHeaderNode.setId(viewId);
+ mHeaderNode.init(this);
+ mHeaderNode.blend(props);
+ mView.getRefreshView().setContent(mHeaderNode.getDoricLayer());
+ }
+ }
+
+ @Override
+ public ViewNode getSubNodeById(String id) {
+ if (id.equals(mContentViewId)) {
+ return mContentNode;
+ }
+ if (id.equals(mHeaderViewId)) {
+ return mHeaderNode;
+ }
+ return null;
+ }
+
+ @Override
+ protected void blendSubNode(JSObject subProperties) {
+ String viewId = subProperties.getProperty("id").asString().value();
+ ViewNode node = getSubNodeById(viewId);
+ if (node != null) {
+ node.blend(subProperties.getProperty("props").asObject());
+ }
+ }
+
+ @DoricMethod
+ public void setRefreshable(JSValue jsValue, DoricPromise doricPromise) {
+ boolean refreshable = jsValue.asBoolean().value();
+ this.mView.setEnabled(refreshable);
+ doricPromise.resolve();
+ }
+
+ @DoricMethod
+ public void setRefreshing(JSValue jsValue, DoricPromise doricPromise) {
+ boolean refreshing = jsValue.asBoolean().value();
+ this.mView.setRefreshing(refreshing);
+ doricPromise.resolve();
+ }
+
+ @DoricMethod
+ public void isRefreshable(DoricPromise doricPromise) {
+ doricPromise.resolve(new JavaValue(this.mView.isEnabled()));
+ }
+
+ @DoricMethod
+ public void isRefreshing(DoricPromise doricPromise) {
+ doricPromise.resolve(new JavaValue(this.mView.isRefreshing()));
+ }
+
+ @Override
+ public void startAnimation() {
+ if (mHeaderNode != null) {
+ mHeaderNode.callJSResponse("startAnimation");
+ }
+ }
+
+ @Override
+ public void stopAnimation() {
+ if (mHeaderNode != null) {
+ mHeaderNode.callJSResponse("stopAnimation");
+ }
+ }
+
+ @Override
+ public void setProgressRotation(float rotation) {
+ if (mHeaderNode != null) {
+ mHeaderNode.callJSResponse("setProgressRotation", rotation);
+ }
+ }
+}
diff --git a/Android/doric/src/main/java/pub/doric/shader/ImageNode.java b/Android/doric/src/main/java/pub/doric/shader/ImageNode.java
index 29a77b41..9157c80d 100644
--- a/Android/doric/src/main/java/pub/doric/shader/ImageNode.java
+++ b/Android/doric/src/main/java/pub/doric/shader/ImageNode.java
@@ -34,6 +34,7 @@ import com.bumptech.glide.request.target.Target;
import pub.doric.DoricContext;
import pub.doric.extension.bridge.DoricPlugin;
+import pub.doric.utils.DoricUtils;
import com.github.pengfeizhou.jscore.JSONBuilder;
import com.github.pengfeizhou.jscore.JSValue;
@@ -77,8 +78,8 @@ public class ImageNode extends ViewNode {
public boolean onResourceReady(Drawable resource, Object model, Target target, DataSource dataSource, boolean isFirstResource) {
if (!TextUtils.isEmpty(loadCallbackId)) {
callJSResponse(loadCallbackId, new JSONBuilder()
- .put("width", resource.getIntrinsicWidth())
- .put("height", resource.getIntrinsicHeight())
+ .put("width", DoricUtils.px2dp(resource.getIntrinsicWidth()))
+ .put("height", DoricUtils.px2dp(resource.getIntrinsicHeight()))
.toJSONObject());
}
return false;
diff --git a/Android/doric/src/main/java/pub/doric/shader/ViewNode.java b/Android/doric/src/main/java/pub/doric/shader/ViewNode.java
index ee394eab..b7df0e5c 100644
--- a/Android/doric/src/main/java/pub/doric/shader/ViewNode.java
+++ b/Android/doric/src/main/java/pub/doric/shader/ViewNode.java
@@ -257,4 +257,21 @@ public abstract class ViewNode extends DoricContextHolder {
public int getHeight() {
return mView.getHeight();
}
+
+ @DoricMethod
+ public void setRotation(JSValue jsValue) {
+ float rotation = jsValue.asNumber().toFloat();
+ while (rotation > 1) {
+ rotation = rotation - 1;
+ }
+ while (rotation < -1) {
+ rotation = rotation + 1;
+ }
+ doricLayer.setRotation(rotation * 360);
+ }
+
+ @DoricMethod
+ public float getRotation() {
+ return doricLayer.getRotation() / 360;
+ }
}
diff --git a/Android/doric/src/main/java/pub/doric/utils/DoricConstant.java b/Android/doric/src/main/java/pub/doric/utils/DoricConstant.java
index f4da7cdc..08a814b1 100644
--- a/Android/doric/src/main/java/pub/doric/utils/DoricConstant.java
+++ b/Android/doric/src/main/java/pub/doric/utils/DoricConstant.java
@@ -31,6 +31,7 @@ public class DoricConstant {
public static final String INJECT_TIMER_SET = "nativeSetTimer";
public static final String INJECT_TIMER_CLEAR = "nativeClearTimer";
public static final String INJECT_BRIDGE = "nativeBridge";
+ public static final String INJECT_EMPTY = "nativeEmpty";
public static final String TEMPLATE_CONTEXT_CREATE = "Reflect.apply(" +
"function(doric,context,Entry,require,exports){" + "\n" +
diff --git a/Android/doric/src/main/java/pub/doric/utils/DoricUtils.java b/Android/doric/src/main/java/pub/doric/utils/DoricUtils.java
index e44b7f71..6f1b237d 100644
--- a/Android/doric/src/main/java/pub/doric/utils/DoricUtils.java
+++ b/Android/doric/src/main/java/pub/doric/utils/DoricUtils.java
@@ -81,6 +81,8 @@ public class DoricUtils {
return new JavaValue((Integer) arg);
} else if (arg instanceof Long) {
return new JavaValue((Long) arg);
+ } else if (arg instanceof Float) {
+ return new JavaValue((Float) arg);
} else if (arg instanceof Double) {
return new JavaValue((Double) arg);
} else if (arg instanceof Boolean) {
diff --git a/demo/index.ts b/demo/index.ts
index 6d413654..7b1bac66 100644
--- a/demo/index.ts
+++ b/demo/index.ts
@@ -12,4 +12,5 @@ export default [
'src/StorageDemo',
'src/NavigatorDemo',
'src/NavbarDemo',
+ 'src/RefreshableDemo',
]
\ No newline at end of file
diff --git a/demo/src/ImageDemo.ts b/demo/src/ImageDemo.ts
index 109c99bb..15ce8d55 100644
--- a/demo/src/ImageDemo.ts
+++ b/demo/src/ImageDemo.ts
@@ -1,10 +1,12 @@
-import { Group, Panel, List, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, scroller, layoutConfig, image, IView, IVLayout, ScaleType } from "doric";
+import { Group, Panel, List, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, scroller, layoutConfig, image, IView, IVLayout, ScaleType, Image } from "doric";
import { colors, label } from "./utils";
+import { img_base64 } from "./image_base64";
const imageUrl = 'https://img.zcool.cn/community/01e75b5da933daa801209e1ffa4649.jpg@1280w_1l_2o_100sh.jpg'
-const imageBase64 = ''
+
@Entry
class ImageDemo extends Panel {
build(rootView: Group): void {
+ let imageView: Image
scroller(vlayout([
text({
text: "Image Demo",
@@ -37,9 +39,13 @@ class ImageDemo extends Panel {
}
}),
label('WebP'),
- image({
+ imageView = image({
imageUrl: "https://p.upyun.com/demo/webp/webp/jpg-0.webp",
loadCallback: (ret) => {
+ if (ret) {
+ imageView.width = ret.width
+ imageView.height = ret.height
+ }
}
}),
label('ScaleToFill'),
@@ -82,7 +88,7 @@ class ImageDemo extends Panel {
}),
label('ImageBase64'),
image({
- imageBase64,
+ imageBase64: img_base64,
width: 300,
height: 300,
border: {
diff --git a/demo/src/LayoutDemo.ts b/demo/src/LayoutDemo.ts
index 4ff2e8a1..442e2844 100644
--- a/demo/src/LayoutDemo.ts
+++ b/demo/src/LayoutDemo.ts
@@ -1,5 +1,5 @@
-import { Group, Panel, Text, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, slider, slideItem, scroller, IVLayout, IHLayout } from "doric";
+import { Group, Panel, Text, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, slider, slideItem, scroller, IVLayout, IHLayout, layoutConfig } from "doric";
import { O_TRUNC } from "constants";
const colors = [
@@ -453,11 +453,7 @@ class LayoutDemo extends Panel {
it.space = 20
}),
).also(it => {
- it.layoutConfig = {
- widthSpec: LayoutSpec.WRAP_CONTENT,
- heightSpec: LayoutSpec.WRAP_CONTENT,
- alignment: gravity().centerX(),
- }
+ it.layoutConfig = layoutConfig().atmost()
})
.in(rootView)
}
diff --git a/demo/src/ListDemo.ts b/demo/src/ListDemo.ts
index a0ccb61a..81e65d97 100644
--- a/demo/src/ListDemo.ts
+++ b/demo/src/ListDemo.ts
@@ -1,15 +1,10 @@
-import { Group, Panel, List, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text } from "doric";
-const colors = [
- "#f0932b",
- "#eb4d4b",
- "#6ab04c",
- "#e056fd",
- "#686de0",
- "#30336b",
-]
+import { Group, Panel, List, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, refreshable, Refreshable, ListItem } from "doric";
+import { rotatedArrow, colors } from "./utils";
@Entry
class ListPanel extends Panel {
build(rootView: Group): void {
+ let refreshView: Refreshable
+ let offset = Math.ceil(Math.random() * colors.length)
vlayout([
text({
text: "ListDemo",
@@ -23,76 +18,90 @@ class ListPanel extends Panel {
textAlignment: gravity().center(),
height: 50,
}),
- list({
- itemCount: 1000,
- renderItem: (idx: number) => {
- let counter!: Text
- return listItem(
- hlayout([
- text({
- layoutConfig: {
- widthSpec: LayoutSpec.WRAP_CONTENT,
- heightSpec: LayoutSpec.EXACTLY,
- alignment: gravity().center(),
- },
- text: `Cell At Line ${idx}`,
- textAlignment: gravity().center(),
- textColor: Color.parse("#ffffff"),
- textSize: 20,
- height: 50,
- }),
- text({
- textColor: Color.parse("#ffffff"),
- textSize: 20,
- text: "",
- }).also(it => {
- counter = it
- it.layoutConfig = {
- widthSpec: LayoutSpec.WRAP_CONTENT,
- heightSpec: LayoutSpec.WRAP_CONTENT,
- margin: {
- left: 10,
+ refreshView = refreshable({
+ onRefresh: () => {
+ refreshView.setRefreshing(context, false).then(() => {
+ (refreshView.content as List).also(it => {
+ it.reset()
+ offset = Math.ceil(Math.random() * colors.length)
+ it.itemCount = 40
+ it.renderItem = (idx: number) => {
+ let counter!: Text
+ return listItem(
+ hlayout([
+ text({
+ layoutConfig: {
+ widthSpec: LayoutSpec.WRAP_CONTENT,
+ heightSpec: LayoutSpec.EXACTLY,
+ alignment: gravity().center(),
+ },
+ text: `Cell At Line ${idx}`,
+ textAlignment: gravity().center(),
+ textColor: Color.parse("#ffffff"),
+ textSize: 20,
+ height: 50,
+ }),
+ text({
+ textColor: Color.parse("#ffffff"),
+ textSize: 20,
+ text: "",
+ }).also(it => {
+ counter = it
+ it.layoutConfig = {
+ widthSpec: LayoutSpec.WRAP_CONTENT,
+ heightSpec: LayoutSpec.WRAP_CONTENT,
+ margin: {
+ left: 10,
+ }
+ }
+ })
+ ]).also(it => {
+ it.layoutConfig = {
+ widthSpec: LayoutSpec.AT_MOST,
+ heightSpec: LayoutSpec.WRAP_CONTENT,
+ margin: {
+ bottom: 2,
+ }
+ }
+ it.gravity = gravity().center()
+ it.bgColor = colors[(idx + offset) % colors.length]
+ let clicked = 0
+ it.onClick = () => {
+ counter.text = `Item Clicked ${++clicked}`
+ }
+ })
+ ).also(it => {
+ it.layoutConfig = {
+ widthSpec: LayoutSpec.AT_MOST,
+ heightSpec: LayoutSpec.WRAP_CONTENT,
}
- }
- })
- ]).also(it => {
- it.layoutConfig = {
- widthSpec: LayoutSpec.AT_MOST,
- heightSpec: LayoutSpec.WRAP_CONTENT,
- margin: {
- bottom: 2,
- }
- }
- it.gravity = gravity().center()
- it.bgColor = Color.parse(colors[idx % colors.length])
- let clicked = 0
- it.onClick = () => {
- counter.text = `Item Clicked ${++clicked}`
+ it.onClick = () => {
+ log(`Click item at ${idx}`)
+ it.height += 10
+ it.nativeChannel(context, "getWidth")().then(
+ resolve => {
+ log(`resolve,${resolve}`)
+ },
+ reject => {
+ log(`reject,${reject}`)
+ })
+ }
+ })
}
})
- ).also(it => {
- it.layoutConfig = {
- widthSpec: LayoutSpec.AT_MOST,
- heightSpec: LayoutSpec.WRAP_CONTENT,
- }
- it.onClick = () => {
- log(`Click item at ${idx}`)
- it.height += 10
- it.nativeChannel(context, "getWidth")().then(
- resolve => {
- log(`resolve,${resolve}`)
- },
- reject => {
- log(`reject,${reject}`)
- })
- }
})
},
- layoutConfig: {
- widthSpec: LayoutSpec.AT_MOST,
- heightSpec: LayoutSpec.AT_MOST,
- },
+ header: rotatedArrow(context),
+ content: list({
+ itemCount: 0,
+ renderItem: () => new ListItem,
+ layoutConfig: {
+ widthSpec: LayoutSpec.AT_MOST,
+ heightSpec: LayoutSpec.AT_MOST,
+ },
+ }),
}),
+
]).also(it => {
it.layoutConfig = {
widthSpec: LayoutSpec.AT_MOST,
@@ -100,5 +109,6 @@ class ListPanel extends Panel {
}
it.bgColor = Color.WHITE
}).in(rootView)
+ refreshView.bgColor = Color.YELLOW
}
}
\ No newline at end of file
diff --git a/demo/src/RefreshableDemo.ts b/demo/src/RefreshableDemo.ts
new file mode 100644
index 00000000..0b967c01
--- /dev/null
+++ b/demo/src/RefreshableDemo.ts
@@ -0,0 +1,109 @@
+import { refreshable, Group, Panel, pullable, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, scroller, layoutConfig, image, IView, IVLayout, ScaleType, modal, IText, network, navigator, stack, Image } from "doric";
+import { title, label, colors, icon_refresh } from "./utils";
+
+@Entry
+class RefreshableDemo extends Panel {
+ build(rootView: Group): void {
+ let refreshImage: Image
+ let refreshView = refreshable({
+ layoutConfig: layoutConfig().atmost(),
+ onRefresh: () => {
+ log('onRefresh')
+ setTimeout(() => {
+ refreshView.setRefreshing(context, false)
+ }, 5000)
+ },
+ header: pullable(context,
+ stack([
+ image({
+ layoutConfig: layoutConfig().exactly().m({ top: 50, bottom: 10, }),
+ width: 30,
+ height: 30,
+ imageBase64: icon_refresh,
+ }).also(v => {
+ refreshImage = v
+ }),
+ ]),
+ {
+ startAnimation: () => {
+ log('startAnimation')
+ },
+ stopAnimation: () => {
+ log('stopAnimation')
+ },
+ setProgressRotation: (rotation: number) => {
+ refreshImage.setRotation(context, rotation)
+ },
+ }),
+ content: scroller(vlayout([
+ title("Refreshable Demo"),
+ label('start Refresh').apply({
+ width: 300,
+ height: 50,
+ bgColor: colors[0],
+ textSize: 30,
+ textColor: Color.WHITE,
+ layoutConfig: layoutConfig().exactly(),
+ onClick: () => {
+ refreshView.setRefreshing(context, true)
+ }
+ } as IText),
+ label('stop Refresh').apply({
+ width: 300,
+ height: 50,
+ bgColor: colors[0],
+ textSize: 30,
+ textColor: Color.WHITE,
+ layoutConfig: layoutConfig().exactly(),
+ onClick: () => {
+ refreshView.setRefreshing(context, false)
+ }
+ } as IText),
+
+ label('Enable Refresh').apply({
+ width: 300,
+ height: 50,
+ bgColor: colors[0],
+ textSize: 30,
+ textColor: Color.WHITE,
+ layoutConfig: layoutConfig().exactly(),
+ onClick: () => {
+ refreshView.setRefreshable(context, true)
+ }
+ } as IText),
+
+ label('Disable Refresh').apply({
+ width: 300,
+ height: 50,
+ bgColor: colors[0],
+ textSize: 30,
+ textColor: Color.WHITE,
+ layoutConfig: layoutConfig().exactly(),
+ onClick: () => {
+ refreshView.setRefreshable(context, false)
+ }
+ } as IText),
+ label('Rotate self').apply({
+ width: 300,
+ height: 50,
+ bgColor: colors[0],
+ textSize: 30,
+ textColor: Color.WHITE,
+ layoutConfig: layoutConfig().exactly(),
+ } as IText).also(v => {
+ v.onClick = () => {
+ v.nativeChannel(context, "setRotation")(0.25)
+ }
+ }),
+ ]).apply({
+ layoutConfig: layoutConfig().atmost().h(LayoutSpec.WRAP_CONTENT),
+ gravity: gravity().center(),
+ space: 10,
+ } as IVLayout)).apply({
+ layoutConfig: layoutConfig().atmost(),
+ })
+ }).apply({
+ bgColor: Color.YELLOW
+ }).in(rootView)
+ }
+}
\ No newline at end of file
diff --git a/demo/src/ScrollerDemo.ts b/demo/src/ScrollerDemo.ts
index 153ffcb8..7981e123 100644
--- a/demo/src/ScrollerDemo.ts
+++ b/demo/src/ScrollerDemo.ts
@@ -1,59 +1,34 @@
-import { Group, Panel, List, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, scroller } from "doric";
-const colors = [
- "#f0932b",
- "#eb4d4b",
- "#6ab04c",
- "#e056fd",
- "#686de0",
- "#30336b",
-]
+import { Group, Panel, List, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, scroller, layoutConfig } from "doric";
+import { label } from "./utils";
+
@Entry
class ScrollerPanel extends Panel {
build(rootView: Group): void {
- rootView.addChild(scroller(vlayout(
- [
- // ...[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 1, 2, 3, 4, 5].map(e => text({
- // text: colors[e % colors.length],
- // textColor: Color.parse('#ffffff'),
- // textSize: 20,
- // bgColor: Color.parse(colors[e % colors.length]),
- // layoutConfig: {
- // widthSpec: LayoutSpec.EXACTLY,
- // heightSpec: LayoutSpec.EXACTLY,
- // },
- // width: 200,
- // height: 50,
- // })),
- ...[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 1, 2, 3, 4, 5].map(i => hlayout([
- ...[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 1, 2, 3, 4, 5].map(j => text({
- text: colors[(i + j) % colors.length],
- textColor: Color.parse('#ffffff'),
- textSize: 20,
- bgColor: Color.parse(colors[(i + j) % colors.length]),
- layoutConfig: {
- widthSpec: LayoutSpec.EXACTLY,
- heightSpec: LayoutSpec.EXACTLY,
- },
- width: 80,
- height: 50,
- })),
- ]).also(it => it.space = 20)),
- hlayout([
- ...[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 1, 2, 3, 4, 5].map(e => text({
- text: colors[e % colors.length],
- textColor: Color.parse('#ffffff'),
- textSize: 20,
- bgColor: Color.parse(colors[e % colors.length]),
- layoutConfig: {
- widthSpec: LayoutSpec.EXACTLY,
- heightSpec: LayoutSpec.EXACTLY,
- },
- width: 200,
- height: 50,
- })),
- ]
- ),
- ]
- ).also(it => it.space = 20)))
+ scroller(
+ vlayout([
+ scroller(
+ vlayout(new Array(100).fill(1).map(e => label('Scroll Content')))
+ ).apply({
+ layoutConfig: layoutConfig().exactly(),
+ width: 300,
+ height: 500,
+ bgColor: Color.RED,
+ }),
+ scroller(
+ vlayout(new Array(100).fill(1).map(e => label('Scroll Content')))
+ ).apply({
+ layoutConfig: layoutConfig().exactly(),
+ width: 300,
+ height: 500,
+ bgColor: Color.BLUE,
+ })
+ ])
+ )
+ .apply({
+ layoutConfig: layoutConfig().atmost().h(LayoutSpec.EXACTLY),
+ height: 500,
+ bgColor: Color.YELLOW,
+ })
+ .in(rootView)
}
}
\ No newline at end of file
diff --git a/demo/src/image_base64.ts b/demo/src/image_base64.ts
new file mode 100644
index 00000000..9eb105a6
--- /dev/null
+++ b/demo/src/image_base64.ts
@@ -0,0 +1,2 @@
+export const img_base64 =
+ ''
\ No newline at end of file
diff --git a/demo/src/utils.ts b/demo/src/utils.ts
index a024952e..385cef4c 100644
--- a/demo/src/utils.ts
+++ b/demo/src/utils.ts
@@ -1,4 +1,5 @@
-import { Color, text, Stack, Text, layoutConfig, LayoutSpec, gravity } from "doric";
+import { Color, text, Stack, Text, layoutConfig, LayoutSpec, gravity, pullable, stack, image, Image, BridgeContext, log } from "doric";
+export const icon_refresh = ''
export const colors = [
"#70a1ff",
@@ -45,4 +46,27 @@ export function title(str: string) {
textAlignment: gravity().center(),
height: 50,
})
+}
+
+export function rotatedArrow(context: BridgeContext) {
+ let refreshImage: Image
+ return pullable(context,
+ stack([
+ image({
+ layoutConfig: layoutConfig().exactly().m({ top: 50, bottom: 10, }),
+ width: 30,
+ height: 30,
+ imageBase64: icon_refresh,
+ }).also(v => refreshImage = v),
+ ]), {
+ startAnimation: () => {
+ log('startAnimation')
+ },
+ stopAnimation: () => {
+ log('stopAnimation')
+ },
+ setProgressRotation: (rotation: number) => {
+ refreshImage.setRotation(context, rotation)
+ },
+ })
}
\ No newline at end of file
diff --git a/demo/tsconfig.json b/demo/tsconfig.json
index 4d880db4..8a7c268d 100644
--- a/demo/tsconfig.json
+++ b/demo/tsconfig.json
@@ -4,7 +4,7 @@
// "incremental": true, /* Enable incremental compilation */
"target": "ES2015", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019' or 'ESNEXT'. */
"module": "es2015", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', or 'ESNext'. */
- // "lib": [], /* Specify library files to be included in the compilation. */
+ "lib": [], /* Specify library files to be included in the compilation. */
// "allowJs": true, /* Allow javascript files to be compiled. */
// "checkJs": true, /* Report errors in .js files. */
// "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */
diff --git a/iOS/Doric.podspec b/iOS/Doric.podspec
index 36590515..8eb2cf33 100644
--- a/iOS/Doric.podspec
+++ b/iOS/Doric.podspec
@@ -39,8 +39,9 @@ TODO: Add long description of the pod here.
s.public_header_files = 'Pod/Classes/**/*.h'
# s.frameworks = 'UIKit', 'MapKit'
# s.dependency 'AFNetworking', '~> 2.3'
- s.dependency 'SDWebImage', '~> 4.4.7'
- s.dependency 'SDWebImage/WebP'
+ # s.dependency 'SDWebImage', '~> 5.0'
+ s.dependency 'YYWebImage', '~>1.0.5'
+ s.dependency 'YYImage/WebP'
s.dependency 'SocketRocket', '~> 0.5.1'
s.dependency 'GCDWebServer', '~> 3.0'
s.dependency 'YYCache', '~> 1.0.4'
diff --git a/iOS/Example/Podfile.lock b/iOS/Example/Podfile.lock
index 1283d0e3..801a54e6 100644
--- a/iOS/Example/Podfile.lock
+++ b/iOS/Example/Podfile.lock
@@ -1,30 +1,23 @@
PODS:
- Doric (0.1.0):
- GCDWebServer (~> 3.0)
- - SDWebImage (~> 4.4.7)
- - SDWebImage/WebP
- SocketRocket (~> 0.5.1)
- YYCache (~> 1.0.4)
+ - YYImage/WebP
+ - YYWebImage
- GCDWebServer (3.5.3):
- GCDWebServer/Core (= 3.5.3)
- GCDWebServer/Core (3.5.3)
- - libwebp (1.0.3):
- - libwebp/demux (= 1.0.3)
- - libwebp/mux (= 1.0.3)
- - libwebp/webp (= 1.0.3)
- - libwebp/demux (1.0.3):
- - libwebp/webp
- - libwebp/mux (1.0.3):
- - libwebp/demux
- - libwebp/webp (1.0.3)
- - SDWebImage (4.4.7):
- - SDWebImage/Core (= 4.4.7)
- - SDWebImage/Core (4.4.7)
- - SDWebImage/WebP (4.4.7):
- - libwebp (< 2.0, >= 0.5)
- - SDWebImage/Core
- SocketRocket (0.5.1)
- YYCache (1.0.4)
+ - YYImage (1.0.4):
+ - YYImage/Core (= 1.0.4)
+ - YYImage/Core (1.0.4)
+ - YYImage/WebP (1.0.4):
+ - YYImage/Core
+ - YYWebImage (1.0.5):
+ - YYCache
+ - YYImage
DEPENDENCIES:
- Doric (from `../`)
@@ -32,22 +25,22 @@ DEPENDENCIES:
SPEC REPOS:
https://github.com/cocoapods/specs.git:
- GCDWebServer
- - libwebp
- - SDWebImage
- SocketRocket
- YYCache
+ - YYImage
+ - YYWebImage
EXTERNAL SOURCES:
Doric:
:path: "../"
SPEC CHECKSUMS:
- Doric: c71287d68afeeb79bfd3c680ed2dd3b90d515c12
+ Doric: e73b17b0e46198994f5c3d8af49f26fd9f49df09
GCDWebServer: c0ab22c73e1b84f358d1e2f74bf6afd1c60829f2
- libwebp: 057912d6d0abfb6357d8bb05c0ea470301f5d61e
- SDWebImage: c10d14a8883ebd89664f02a422006f66a85c0c5d
SocketRocket: d57c7159b83c3c6655745cd15302aa24b6bae531
YYCache: 8105b6638f5e849296c71f331ff83891a4942952
+ YYImage: 1e1b62a9997399593e4b9c4ecfbbabbf1d3f3b54
+ YYWebImage: 5f7f36aee2ae293f016d418c7d6ba05c4863e928
PODFILE CHECKSUM: 012563d71439e7e33e976dca3b59664ed56cee39
diff --git a/iOS/Pod/Classes/Doric.h b/iOS/Pod/Classes/Doric.h
index 1b6471d2..e96c2565 100644
--- a/iOS/Pod/Classes/Doric.h
+++ b/iOS/Pod/Classes/Doric.h
@@ -24,4 +24,5 @@
#import "DoricJSLoaderManager.h"
#import "DoricNavigatorDelegate.h"
#import "DoricNavBarDelegate.h"
-#import "DoricViewController.h"
\ No newline at end of file
+#import "DoricViewController.h"
+#import "DoricPromise.h"
\ No newline at end of file
diff --git a/iOS/Pod/Classes/DoricRegistry.m b/iOS/Pod/Classes/DoricRegistry.m
index c3e84485..0d7f9ea6 100644
--- a/iOS/Pod/Classes/DoricRegistry.m
+++ b/iOS/Pod/Classes/DoricRegistry.m
@@ -37,6 +37,7 @@
#import "DoricStoragePlugin.h"
#import "DoricNavigatorPlugin.h"
#import "DoricNavBarPlugin.h"
+#import "DoricRefreshableNode.h"
@interface DoricRegistry ()
@@ -76,6 +77,7 @@ - (void)innerRegister {
[self registerViewNode:DoricScrollerNode.class withName:@"Scroller"];
[self registerViewNode:DoricSliderNode.class withName:@"Slider"];
[self registerViewNode:DoricSlideItemNode.class withName:@"SlideItem"];
+ [self registerViewNode:DoricRefreshableNode.class withName:@"Refreshable"];
}
- (void)registerJSBundle:(NSString *)bundle withName:(NSString *)name {
diff --git a/iOS/Pod/Classes/Engine/DoricJSEngine.m b/iOS/Pod/Classes/Engine/DoricJSEngine.m
index 3559c0cb..12984ea8 100644
--- a/iOS/Pod/Classes/Engine/DoricJSEngine.m
+++ b/iOS/Pod/Classes/Engine/DoricJSEngine.m
@@ -59,7 +59,9 @@ - (void)initJSExecutor {
[self.jsExecutor injectGlobalJSObject:INJECT_LOG obj:^(NSString *type, NSString *message) {
DoricLog(@"JS:%@", message);
}];
-
+ [self.jsExecutor injectGlobalJSObject:INJECT_EMPTY obj:^() {
+
+ }];
[self.jsExecutor injectGlobalJSObject:INJECT_REQUIRE obj:^(NSString *name) {
__strong typeof(_self) self = _self;
if (!self) return NO;
diff --git a/iOS/Pod/Classes/Plugin/DoricShaderPlugin.m b/iOS/Pod/Classes/Plugin/DoricShaderPlugin.m
index 32949326..94ff77e7 100644
--- a/iOS/Pod/Classes/Plugin/DoricShaderPlugin.m
+++ b/iOS/Pod/Classes/Plugin/DoricShaderPlugin.m
@@ -25,8 +25,6 @@
#import "DoricUtil.h"
#import "Doric.h"
-#import
-
#import
@implementation DoricShaderPlugin
@@ -110,9 +108,13 @@ - (id)findClass:(Class)clz target:(id)target method:(NSString *)name promise:(Do
dispatch_async(dispatch_get_main_queue(), ^{
void *retValue;
block();
- [invocation getReturnValue:&retValue];
- id returnValue = (__bridge id) retValue;
- [promise resolve:returnValue];
+ const char *retType = methodSignature.methodReturnType;
+ if (!strcmp(retType, @encode(void))) {
+ } else {
+ [invocation getReturnValue:&retValue];
+ id returnValue = (__bridge id) retValue;
+ [promise resolve:returnValue];
+ }
});
return ret;
}
diff --git a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.h b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.h
new file mode 100644
index 00000000..61bb6845
--- /dev/null
+++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.h
@@ -0,0 +1,10 @@
+//
+// Created by pengfei.zhou on 2019/11/26.
+//
+
+#import
+#import "DoricSuperNode.h"
+#import "DoricSwipeRefreshLayout.h"
+
+@interface DoricRefreshableNode : DoricSuperNode
+@end
\ No newline at end of file
diff --git a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m
new file mode 100644
index 00000000..bb17e312
--- /dev/null
+++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m
@@ -0,0 +1,161 @@
+//
+// Created by pengfei.zhou on 2019/11/26.
+//
+
+#import "DoricRefreshableNode.h"
+#import "Doric.h"
+
+@interface DoricRefreshableNode ()
+@property(nonatomic, strong) DoricViewNode *contentNode;
+@property(nonatomic, copy) NSString *contentViewId;
+@property(nonatomic, strong) DoricViewNode *headerNode;
+@property(nonatomic, copy) NSString *headerViewId;
+@end
+
+@implementation DoricRefreshableNode
+- (DoricSwipeRefreshLayout *)build {
+ return [[DoricSwipeRefreshLayout new] also:^(DoricSwipeRefreshLayout *it) {
+ it.swipePullingDelegate = self;
+
+ }];
+}
+
+- (void)blendView:(DoricSwipeRefreshLayout *)view forPropName:(NSString *)name propValue:(id)prop {
+ if ([@"content" isEqualToString:name]) {
+ self.contentViewId = prop;
+ } else if ([@"header" isEqualToString:name]) {
+ self.headerViewId = prop;
+ } else if ([@"onRefresh" isEqualToString:name]) {
+ __weak typeof(self) _self = self;
+ NSString *funcId = prop;
+ self.view.onRefreshBlock = ^{
+ __strong typeof(_self) self = _self;
+ [self callJSResponse:funcId, nil];
+ };
+ } else {
+ [super blendView:view forPropName:name propValue:prop];
+ }
+}
+
+- (DoricViewNode *)subNodeWithViewId:(NSString *)viewId {
+ if ([viewId isEqualToString:self.contentViewId]) {
+ return self.contentNode;
+ } else if ([viewId isEqualToString:self.headerViewId]) {
+ return self.headerNode;
+ } else {
+ return nil;
+ }
+}
+
+- (void)blend:(NSDictionary *)props {
+ [super blend:props];
+ [self blendHeader];
+ [self blendContent];
+}
+
+- (void)blendContent {
+ NSDictionary *contentModel = [self subModelOf:self.contentViewId];
+ if (!contentModel) {
+ return;
+ }
+ NSString *viewId = contentModel[@"id"];
+ NSString *type = contentModel[@"type"];
+ NSDictionary *childProps = contentModel[@"props"];
+ if (self.contentNode) {
+ if ([self.contentNode.viewId isEqualToString:viewId]) {
+ //skip
+ } else {
+ if (self.reusable && [type isEqualToString:self.contentNode.type]) {
+ [self.contentNode also:^(DoricViewNode *it) {
+ it.viewId = viewId;
+ [it blend:childProps];
+ }];
+ } else {
+ self.contentNode = [[DoricViewNode create:self.doricContext withType:type] also:^(DoricViewNode *it) {
+ it.viewId = viewId;
+ [it initWithSuperNode:self];
+ [it blend:childProps];
+ self.view.contentView = it.view;
+ }];
+ }
+ }
+ } else {
+ self.contentNode = [[DoricViewNode create:self.doricContext withType:type] also:^(DoricViewNode *it) {
+ it.viewId = viewId;
+ [it initWithSuperNode:self];
+ [it blend:childProps];
+ self.view.contentView = it.view;
+ }];
+ }
+}
+
+- (void)blendHeader {
+ NSDictionary *headerModel = [self subModelOf:self.headerViewId];
+ if (!headerModel) {
+ return;
+ }
+ NSString *viewId = headerModel[@"id"];
+ NSString *type = headerModel[@"type"];
+ NSDictionary *childProps = headerModel[@"props"];
+ if (self.headerNode) {
+ if ([self.headerNode.viewId isEqualToString:viewId]) {
+ //skip
+ } else {
+ if (self.reusable && [type isEqualToString:self.headerNode.type]) {
+ [self.headerNode also:^(DoricViewNode *it) {
+ it.viewId = viewId;
+ [it blend:childProps];
+ }];
+ } else {
+ self.headerNode = [[DoricViewNode create:self.doricContext withType:type] also:^(DoricViewNode *it) {
+ it.viewId = viewId;
+ [it initWithSuperNode:self];
+ [it blend:childProps];
+ self.view.headerView = it.view;
+ }];
+ }
+ }
+ } else {
+ self.headerNode = [[DoricViewNode create:self.doricContext withType:type] also:^(DoricViewNode *it) {
+ it.viewId = viewId;
+ [it initWithSuperNode:self];
+ [it blend:childProps];
+ self.view.headerView = it.view;
+ }];
+ }
+}
+
+- (void)blendSubNode:(NSDictionary *)subModel {
+ [[self subNodeWithViewId:subModel[@"id"]] blend:subModel[@"props"]];
+}
+
+- (void)startAnimation {
+ [self.headerNode callJSResponse:@"startAnimation", nil];
+}
+
+- (void)stopAnimation {
+ [self.headerNode callJSResponse:@"stopAnimation", nil];
+}
+
+- (void)setProgressRotation:(CGFloat)rotation {
+ [self.headerNode callJSResponse:@"setProgressRotation", @(rotation), nil];
+}
+
+- (void)setRefreshing:(NSNumber *)refreshable withPromise:(DoricPromise *)promise {
+ self.view.refreshing = [refreshable boolValue];
+ [promise resolve:nil];
+}
+
+- (void)setRefreshable:(NSNumber *)refreshing withPromise:(DoricPromise *)promise {
+ self.view.refreshable = [refreshing boolValue];
+ [promise resolve:nil];
+}
+
+- (NSNumber *)isRefreshing {
+ return @(self.view.refreshing);
+}
+
+- (NSNumber *)isRefreshable {
+ return @(self.view.refreshable);
+}
+@end
diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.h b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.h
new file mode 100644
index 00000000..45725251
--- /dev/null
+++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.h
@@ -0,0 +1,22 @@
+//
+// Created by pengfei.zhou on 2019/11/26.
+//
+
+#import
+
+@protocol DoricSwipePullingDelegate
+- (void)startAnimation;
+
+- (void)stopAnimation;
+
+- (void)setProgressRotation:(CGFloat)rotation;
+@end
+
+@interface DoricSwipeRefreshLayout : UIScrollView
+@property(nonatomic, strong) UIView *contentView;
+@property(nonatomic, strong) UIView *headerView;
+@property(nonatomic, assign) BOOL refreshable;
+@property(nonatomic, assign) BOOL refreshing;
+@property(nonatomic, strong) void (^onRefreshBlock)(void);
+@property(nonatomic, weak) id swipePullingDelegate;
+@end
\ No newline at end of file
diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m
new file mode 100644
index 00000000..fc1c94fb
--- /dev/null
+++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m
@@ -0,0 +1,146 @@
+//
+// Created by pengfei.zhou on 2019/11/26.
+//
+
+#import "DoricSwipeRefreshLayout.h"
+#import "UIView+Doric.h"
+#import "DoricLayouts.h"
+
+@interface DoricSwipeRefreshLayout ()
+
+@end
+
+@implementation DoricSwipeRefreshLayout
+
+- (instancetype)initWithFrame:(CGRect)frame {
+ if (self = [super initWithFrame:frame]) {
+ self.showsHorizontalScrollIndicator = NO;
+ self.showsVerticalScrollIndicator = NO;
+ self.alwaysBounceVertical = YES;
+ self.delegate = self;
+ if (@available(iOS 11, *)) {
+ self.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
+ }
+ }
+ return self;
+}
+
+- (instancetype)init {
+ if (self = [super init]) {
+ self.showsHorizontalScrollIndicator = NO;
+ self.showsVerticalScrollIndicator = NO;
+ self.alwaysBounceVertical = YES;
+ self.delegate = self;
+ if (@available(iOS 11, *)) {
+ self.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
+ }
+ }
+ return self;
+}
+
+- (CGSize)sizeThatFits:(CGSize)size {
+ if (self.contentView) {
+ return [self.contentView measureSize:size];
+ }
+ return CGSizeZero;
+}
+
+- (BOOL)requestFromSubview:(UIView *)subview {
+ if (subview == self.headerView) {
+ return NO;
+ }
+ return [super requestFromSubview:subview];
+}
+
+- (void)layoutSelf:(CGSize)targetSize {
+ [super layoutSelf:targetSize];
+ self.headerView.bottom = 0;
+ self.headerView.centerX = self.centerX;
+ self.contentSize = self.frame.size;
+}
+
+- (void)setContentView:(UIView *)contentView {
+ if (_contentView) {
+ [_contentView removeFromSuperview];
+ }
+ _contentView = contentView;
+ [self addSubview:contentView];
+}
+
+- (void)setHeaderView:(UIView *)headerView {
+ if (_headerView) {
+ [_headerView removeFromSuperview];
+ }
+ _headerView = headerView;
+ [self addSubview:headerView];
+ self.refreshable = YES;
+}
+
+- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate {
+ if (-scrollView.contentOffset.y >= self.headerView.height) {
+ dispatch_async(dispatch_get_main_queue(), ^{
+ self.refreshing = YES;
+ });
+ }
+}
+
+- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
+ if (scrollView.contentOffset.y <= 0) {
+ [self.swipePullingDelegate setProgressRotation:-scrollView.contentOffset.y / self.headerView.height];
+ }
+}
+
+- (void)setContentOffset:(CGPoint)contentOffset {
+ [super setContentOffset:contentOffset];
+}
+
+- (void)setRefreshing:(BOOL)refreshing {
+ if (_refreshing == refreshing) {
+ return;
+ }
+ if (refreshing) {
+ if (self.onRefreshBlock) {
+ self.onRefreshBlock();
+ }
+ [UIView animateWithDuration:.3f
+ animations:^{
+ self.contentOffset = (CGPoint) {0, -self.headerView.height};
+ self.contentInset = UIEdgeInsetsMake(self.headerView.height, 0, 0, 0);
+ }
+ completion:^(BOOL finished) {
+ [self.swipePullingDelegate startAnimation];
+ self.scrollEnabled = NO;
+ }
+ ];
+ } else {
+ self.bounces = YES;
+ [UIView animateWithDuration:.3f
+ animations:^{
+ self.contentOffset = (CGPoint) {0, 0};
+ self.contentInset = UIEdgeInsetsMake(0, 0, 0, 0);
+ }
+ completion:^(BOOL finished) {
+ [self.swipePullingDelegate stopAnimation];
+ self.scrollEnabled = YES;
+ }
+ ];
+ }
+ _refreshing = refreshing;
+}
+
+- (void)setRefreshable:(BOOL)refreshable {
+ self.scrollEnabled = refreshable;
+ if (refreshable) {
+ self.contentOffset = (CGPoint) {0, 0};
+ self.contentInset = UIEdgeInsetsMake(0, 0, 0, 0);
+ }
+}
+
+- (BOOL)refreshable {
+ return self.scrollEnabled;
+}
+
+- (void)setContentSize:(CGSize)contentSize {
+ [super setContentSize:contentSize];
+}
+@end
diff --git a/iOS/Pod/Classes/Shader/DoricImageNode.m b/iOS/Pod/Classes/Shader/DoricImageNode.m
index 6670a4fd..95cfd878 100644
--- a/iOS/Pod/Classes/Shader/DoricImageNode.m
+++ b/iOS/Pod/Classes/Shader/DoricImageNode.m
@@ -22,7 +22,7 @@
#import "DoricImageNode.h"
#import "Doric.h"
-#import
+#import "YYWebImage.h"
@interface DoricImageNode ()
@property(nonatomic, copy) NSString *loadCallbackId;
@@ -31,7 +31,7 @@ @interface DoricImageNode ()
@implementation DoricImageNode
- (UIImageView *)build {
- return [[UIImageView new] also:^(UIImageView *it) {
+ return [[YYAnimatedImageView new] also:^(UIImageView *it) {
it.clipsToBounds = YES;
}];
}
@@ -39,7 +39,7 @@ - (UIImageView *)build {
- (void)blendView:(UIImageView *)view forPropName:(NSString *)name propValue:(id)prop {
if ([@"imageUrl" isEqualToString:name]) {
__weak typeof(self) _self = self;
- [view sd_setImageWithURL:[NSURL URLWithString:prop] completed:^(UIImage *_Nullable image, NSError *_Nullable error, SDImageCacheType cacheType, NSURL *_Nullable imageURL) {
+ [view yy_setImageWithURL:[NSURL URLWithString:prop] placeholder:nil options:0 completion:^(UIImage *image, NSURL *url, YYWebImageFromType from, YYWebImageStage stage, NSError *error) {
__strong typeof(_self) self = _self;
if (error) {
if (self.loadCallbackId.length > 0) {
@@ -53,7 +53,6 @@ - (void)blendView:(UIImageView *)view forPropName:(NSString *)name propValue:(id
}
[self requestLayout];
}
-
}];
} else if ([@"scaleType" isEqualToString:name]) {
switch ([prop integerValue]) {
diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.h b/iOS/Pod/Classes/Shader/DoricLayouts.h
index 33c3ea86..fcddee0c 100644
--- a/iOS/Pod/Classes/Shader/DoricLayouts.h
+++ b/iOS/Pod/Classes/Shader/DoricLayouts.h
@@ -19,14 +19,7 @@
#import
-
-struct DoricMargin {
- CGFloat left;
- CGFloat right;
- CGFloat top;
- CGFloat bottom;
-};
-typedef struct DoricMargin DoricMargin;
+typedef UIEdgeInsets DoricMargin;
DoricMargin DoricMarginMake(CGFloat left, CGFloat top, CGFloat right, CGFloat bottom);
@@ -68,17 +61,13 @@ typedef NS_ENUM(NSInteger, DoricGravity) {
@interface DoricLayoutContainer : UIView
-@property(nonatomic, assign) DoricGravity gravity;
-
-- (void)layout:(CGSize)targetSize;
-
-- (CGSize)sizeContent:(CGSize)size;
@end
@interface DoricStackView : DoricLayoutContainer
@end
@interface DoricLinearView : DoricLayoutContainer
+@property(nonatomic, assign) DoricGravity gravity;
@property(nonatomic, assign) CGFloat space;
@end
@@ -91,7 +80,20 @@ typedef NS_ENUM(NSInteger, DoricGravity) {
@interface UIView (DoricLayoutConfig)
@property(nonatomic, strong) DoricLayoutConfig *layoutConfig;
+@end
+
+@interface UIView (DoricTag)
@property(nonatomic, copy) NSString *tagString;
- (UIView *)viewWithTagString:(NSString *)tagString;
@end
+
+@interface UIView (DoricLayouts)
+- (void)layoutSelf:(CGSize)targetSize;
+
+- (CGSize)measureSize:(CGSize)targetSize;
+
+- (void)doricLayoutSubviews;
+
+- (BOOL)requestFromSubview:(UIView *)subview;
+@end
\ No newline at end of file
diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m
index bf238368..c1e0adfb 100644
--- a/iOS/Pod/Classes/Shader/DoricLayouts.m
+++ b/iOS/Pod/Classes/Shader/DoricLayouts.m
@@ -19,8 +19,107 @@
#import "DoricLayouts.h"
#import
+#import
#import "UIView+Doric.h"
+static const void *kLayoutConfig = &kLayoutConfig;
+
+@implementation UIView (DoricLayoutConfig)
+@dynamic layoutConfig;
+
+- (void)setLayoutConfig:(DoricLayoutConfig *)layoutConfig {
+ objc_setAssociatedObject(self, kLayoutConfig, layoutConfig, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
+}
+
+- (DoricLayoutConfig *)layoutConfig {
+ return objc_getAssociatedObject(self, kLayoutConfig);
+}
+
+@end
+
+static const void *kTagString = &kTagString;
+
+@implementation UIView (DoricTag)
+
+- (void)setTagString:(NSString *)tagString {
+ objc_setAssociatedObject(self, kTagString, tagString, OBJC_ASSOCIATION_COPY_NONATOMIC);
+ self.tag = [tagString hash];
+}
+
+- (NSString *)tagString {
+ return objc_getAssociatedObject(self, kTagString);
+}
+
+
+- (UIView *)viewWithTagString:(NSString *)tagString {
+ // notice the potential hash collision
+ return [self viewWithTag:[tagString hash]];
+}
+
+@end
+
+
+@implementation UIView (DoricLayouts)
+/**
+ * Measure self's size
+ * */
+- (CGSize)measureSize:(CGSize)targetSize {
+ CGFloat width = self.width;
+ CGFloat height = self.height;
+
+ DoricLayoutConfig *config = self.layoutConfig;
+ if (!config) {
+ config = [DoricLayoutConfig new];
+ }
+ if (config.widthSpec == DoricLayoutAtMost
+ || config.widthSpec == DoricLayoutWrapContent) {
+ width = targetSize.width - config.margin.left - config.margin.right;
+ }
+ if (config.heightSpec == DoricLayoutAtMost
+ || config.heightSpec == DoricLayoutWrapContent) {
+ height = targetSize.height - config.margin.top - config.margin.bottom;
+ }
+
+ CGSize contentSize = [self sizeThatFits:CGSizeMake(width, height)];
+ if (config.widthSpec == DoricLayoutWrapContent) {
+ width = contentSize.width;
+ }
+ if (config.heightSpec == DoricLayoutWrapContent) {
+ height = contentSize.height;
+ }
+ return CGSizeMake(width, height);
+}
+
+/**
+ * layout self and subviews
+ * */
+- (void)layoutSelf:(CGSize)targetSize {
+ self.width = targetSize.width;
+ self.height = targetSize.height;
+ for (UIView *view in self.subviews) {
+ [view layoutSelf:[view measureSize:targetSize]];
+ }
+}
+
+
+- (void)doricLayoutSubviews {
+ if ([self.superview requestFromSubview:self]) {
+ [self.superview doricLayoutSubviews];
+ } else {
+ [self layoutSelf:CGSizeMake(self.width, self.height)];
+ }
+}
+
+- (BOOL)requestFromSubview:(UIView *)subview {
+ if (self.layoutConfig
+ && self.layoutConfig.widthSpec != DoricLayoutExact
+ && self.layoutConfig.heightSpec != DoricLayoutExact) {
+ return YES;
+ }
+ return NO;
+}
+@end
+
DoricMargin DoricMarginMake(CGFloat left, CGFloat top, CGFloat right, CGFloat bottom) {
DoricMargin margin;
margin.left = left;
@@ -65,63 +164,23 @@ @interface DoricLayoutContainer ()
@end
@implementation DoricLayoutContainer
+- (void)setNeedsLayout {
+ [super setNeedsLayout];
+}
- (void)layoutSubviews {
- if ([self.superview isKindOfClass:[DoricLayoutContainer class]]) {
- [self.superview layoutSubviews];
- } else {
- CGSize size = [self sizeThatFits:CGSizeMake(self.superview.width, self.superview.height)];
- [self layout:size];
- }
-}
-
-- (CGSize)sizeThatFits:(CGSize)size {
- CGFloat width = self.width;
- CGFloat height = self.height;
-
- DoricLayoutConfig *config = self.layoutConfig;
- if (!config) {
- config = [DoricLayoutConfig new];
- }
- if (config.widthSpec == DoricLayoutAtMost
- || config.widthSpec == DoricLayoutWrapContent) {
- width = size.width - config.margin.left - config.margin.right;
- }
- if (config.heightSpec == DoricLayoutAtMost
- || config.heightSpec == DoricLayoutWrapContent) {
- height = size.height - config.margin.top - config.margin.bottom;
- }
-
- CGSize contentSize = [self sizeContent:CGSizeMake(width, height)];
- if (config.widthSpec == DoricLayoutWrapContent) {
- width = contentSize.width;
- }
- if (config.heightSpec == DoricLayoutWrapContent) {
- height = contentSize.height;
- }
- return CGSizeMake(width, height);
-}
-
-- (CGSize)sizeContent:(CGSize)size {
- return size;
-}
-
-- (void)layout:(CGSize)targetSize {
- self.width = targetSize.width;
- self.height = targetSize.height;
+ [super layoutSubviews];
+ [self doricLayoutSubviews];
}
@end
@interface DoricStackView ()
-
-@property(nonatomic, assign) CGFloat contentWidth;
-@property(nonatomic, assign) CGFloat contentHeight;
@end
@implementation DoricStackView
-- (CGSize)sizeContent:(CGSize)size {
+- (CGSize)sizeThatFits:(CGSize)size {
CGFloat contentWidth = 0;
CGFloat contentHeight = 0;
for (UIView *child in self.subviews) {
@@ -132,24 +191,11 @@ - (CGSize)sizeContent:(CGSize)size {
if (!childConfig) {
childConfig = [DoricLayoutConfig new];
}
- CGSize childSize = CGSizeMake(child.width, child.height);
- if ([child isKindOfClass:[DoricLayoutContainer class]]
- || childConfig.widthSpec == DoricLayoutWrapContent
- || childConfig.heightSpec == DoricLayoutWrapContent) {
- childSize = [child sizeThatFits:CGSizeMake(size.width, size.height - contentHeight)];
- }
- if (childConfig.widthSpec == DoricLayoutExact) {
- childSize.width = child.width;
- } else if (childConfig.widthSpec == DoricLayoutAtMost) {
- childSize.width = size.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- childSize.height = child.height;
- } else if (childConfig.heightSpec == DoricLayoutAtMost) {
- childSize.height = size.height - contentHeight;
- }
- if (childConfig.weight) {
- childSize.height = child.height;
+ CGSize childSize;
+ if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) {
+ childSize = [child measureSize:CGSizeMake(size.width, size.height)];
+ } else {
+ childSize = child.bounds.size;
}
contentWidth = MAX(contentWidth, childSize.width + childConfig.margin.left + childConfig.margin.right);
contentHeight = MAX(contentHeight, childSize.height + childConfig.margin.top + childConfig.margin.bottom);
@@ -159,38 +205,23 @@ - (CGSize)sizeContent:(CGSize)size {
return CGSizeMake(contentWidth, contentHeight);
}
-- (void)layout:(CGSize)targetSize {
+- (void)layoutSelf:(CGSize)targetSize {
+ self.width = targetSize.width;
+ self.height = targetSize.height;
for (UIView *child in self.subviews) {
if (child.isHidden) {
continue;
}
+ if (!CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) {
+ continue;
+ }
DoricLayoutConfig *childConfig = child.layoutConfig;
if (!childConfig) {
childConfig = [DoricLayoutConfig new];
}
-
- CGSize size = [child sizeThatFits:CGSizeMake(targetSize.width, targetSize.height)];
- if (childConfig.widthSpec == DoricLayoutExact) {
- size.width = child.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- size.height = child.height;
- }
- if (childConfig.widthSpec == DoricLayoutExact) {
- size.width = child.width;
- } else if (childConfig.widthSpec == DoricLayoutAtMost) {
- size.width = targetSize.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- size.height = child.height;
- } else if (childConfig.heightSpec == DoricLayoutAtMost) {
- size.height = targetSize.height;
- }
- child.width = size.width;
- child.height = size.height;
-
- DoricGravity gravity = childConfig.alignment | self.gravity;
-
+ CGSize size = [child measureSize:CGSizeMake(targetSize.width, targetSize.height)];
+ [child layoutSelf:size];
+ DoricGravity gravity = childConfig.alignment;
if ((gravity & LEFT) == LEFT) {
child.left = 0;
} else if ((gravity & RIGHT) == RIGHT) {
@@ -204,7 +235,6 @@ - (void)layout:(CGSize)targetSize {
child.right = targetSize.width - childConfig.margin.right;
}
}
-
if ((gravity & TOP) == TOP) {
child.top = 0;
} else if ((gravity & BOTTOM) == BOTTOM) {
@@ -218,13 +248,7 @@ - (void)layout:(CGSize)targetSize {
child.bottom = targetSize.height - childConfig.margin.bottom;
}
}
-
- if ([child isKindOfClass:[DoricLayoutContainer class]]) {
- [(DoricLayoutContainer *) child layout:size];
- }
}
- self.width = targetSize.width;
- self.height = targetSize.height;
}
@end
@@ -232,7 +256,8 @@ @implementation DoricLinearView
@end
@implementation DoricVLayoutView
-- (CGSize)sizeContent:(CGSize)size {
+
+- (CGSize)sizeThatFits:(CGSize)size {
CGFloat contentWidth = 0;
CGFloat contentHeight = 0;
NSUInteger contentWeight = 0;
@@ -244,24 +269,11 @@ - (CGSize)sizeContent:(CGSize)size {
if (!childConfig) {
childConfig = [DoricLayoutConfig new];
}
- CGSize childSize = CGSizeMake(child.width, child.height);
- if ([child isKindOfClass:[DoricLayoutContainer class]]
- || childConfig.widthSpec == DoricLayoutWrapContent
- || childConfig.heightSpec == DoricLayoutWrapContent) {
- childSize = [child sizeThatFits:CGSizeMake(size.width, size.height - contentHeight)];
- }
- if (childConfig.widthSpec == DoricLayoutExact) {
- childSize.width = child.width;
- } else if (childConfig.widthSpec == DoricLayoutAtMost) {
- childSize.width = size.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- childSize.height = child.height;
- } else if (childConfig.heightSpec == DoricLayoutAtMost) {
- childSize.height = size.height - contentHeight;
- }
- if (childConfig.weight) {
- childSize.height = child.height;
+ CGSize childSize;
+ if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) {
+ childSize = [child measureSize:CGSizeMake(size.width, size.height - contentHeight)];
+ } else {
+ childSize = child.bounds.size;
}
contentWidth = MAX(contentWidth, childSize.width + childConfig.margin.left + childConfig.margin.right);
contentHeight += childSize.height + self.space + childConfig.margin.top + childConfig.margin.bottom;
@@ -277,7 +289,9 @@ - (CGSize)sizeContent:(CGSize)size {
return CGSizeMake(contentWidth, contentHeight);
}
-- (void)layout:(CGSize)targetSize {
+- (void)layoutSelf:(CGSize)targetSize {
+ self.width = targetSize.width;
+ self.height = targetSize.height;
CGFloat yStart = 0;
if ((self.gravity & TOP) == TOP) {
yStart = 0;
@@ -291,40 +305,20 @@ - (void)layout:(CGSize)targetSize {
if (child.isHidden) {
continue;
}
+ if (!CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) {
+ continue;
+ }
DoricLayoutConfig *childConfig = child.layoutConfig;
if (!childConfig) {
childConfig = [DoricLayoutConfig new];
}
- CGSize size = [child sizeThatFits:CGSizeMake(targetSize.width, targetSize.height - yStart)];
- if (childConfig.widthSpec == DoricLayoutExact) {
- size.width = child.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- size.height = child.height;
- }
- if (childConfig.widthSpec == DoricLayoutExact) {
- size.width = child.width;
- } else if (childConfig.widthSpec == DoricLayoutAtMost) {
- size.width = targetSize.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- size.height = child.height;
- } else if (childConfig.heightSpec == DoricLayoutAtMost) {
- size.height = targetSize.height - yStart;
- }
- if (childConfig.weight) {
- size.height = child.height;
- }
-
+ CGSize size = [child measureSize:CGSizeMake(targetSize.width, targetSize.height - yStart)];
if (childConfig.weight) {
size.height += remain / self.contentWeight * childConfig.weight;
}
- child.width = size.width;
- child.height = size.height;
-
+ [child layoutSelf:size];
DoricGravity gravity = childConfig.alignment | self.gravity;
-
if ((gravity & LEFT) == LEFT) {
child.left = 0;
} else if ((gravity & RIGHT) == RIGHT) {
@@ -346,18 +340,12 @@ - (void)layout:(CGSize)targetSize {
if (childConfig.margin.bottom) {
yStart += childConfig.margin.bottom;
}
- if ([child isKindOfClass:[DoricLayoutContainer class]]) {
- [(DoricLayoutContainer *) child layout:size];
- }
}
- self.width = targetSize.width;
- self.height = targetSize.height;
}
@end
@implementation DoricHLayoutView
-
-- (CGSize)sizeContent:(CGSize)size {
+- (CGSize)sizeThatFits:(CGSize)size {
CGFloat contentWidth = 0;
CGFloat contentHeight = 0;
NSUInteger contentWeight = 0;
@@ -369,24 +357,11 @@ - (CGSize)sizeContent:(CGSize)size {
if (!childConfig) {
childConfig = [DoricLayoutConfig new];
}
- CGSize childSize = CGSizeMake(child.width, child.height);
- if ([child isKindOfClass:[DoricLayoutContainer class]]
- || childConfig.widthSpec == DoricLayoutWrapContent
- || childConfig.heightSpec == DoricLayoutWrapContent) {
- childSize = [child sizeThatFits:CGSizeMake(size.width - contentWidth, size.height)];
- }
- if (childConfig.widthSpec == DoricLayoutExact) {
- childSize.width = child.width;
- } else if (childConfig.widthSpec == DoricLayoutAtMost) {
- childSize.width = size.width - contentWidth;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- childSize.height = child.height;
- } else if (childConfig.heightSpec == DoricLayoutAtMost) {
- childSize.height = size.height;
- }
- if (childConfig.weight) {
- childSize.width = child.width;
+ CGSize childSize;
+ if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) {
+ childSize = [child measureSize:CGSizeMake(size.width - contentWidth, size.height)];
+ } else {
+ childSize = child.bounds.size;
}
contentWidth += childSize.width + self.space + childConfig.margin.left + childConfig.margin.right;
contentHeight = MAX(contentHeight, childSize.height + childConfig.margin.top + childConfig.margin.bottom);
@@ -402,7 +377,9 @@ - (CGSize)sizeContent:(CGSize)size {
return CGSizeMake(contentWidth, contentHeight);
}
-- (void)layout:(CGSize)targetSize {
+- (void)layoutSelf:(CGSize)targetSize {
+ self.width = targetSize.width;
+ self.height = targetSize.height;
CGFloat xStart = 0;
if (self.contentWeight) {
xStart = 0;
@@ -418,37 +395,20 @@ - (void)layout:(CGSize)targetSize {
if (child.isHidden) {
continue;
}
+ if (!CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) {
+ continue;
+ }
DoricLayoutConfig *childConfig = child.layoutConfig;
if (!childConfig) {
childConfig = [DoricLayoutConfig new];
}
- CGSize size = [child sizeThatFits:CGSizeMake(targetSize.width - xStart, targetSize.height)];
- if (childConfig.widthSpec == DoricLayoutExact) {
- size.width = child.width;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- size.height = child.height;
- }
- if (childConfig.widthSpec == DoricLayoutExact) {
- size.width = child.width;
- } else if (childConfig.widthSpec == DoricLayoutAtMost) {
- size.width = targetSize.width - xStart;
- }
- if (childConfig.heightSpec == DoricLayoutExact) {
- size.height = child.height;
- } else if (childConfig.heightSpec == DoricLayoutAtMost) {
- size.height = targetSize.height;
- }
- if (childConfig.weight) {
- size.width = child.width;
- }
-
+ CGSize size = [child measureSize:CGSizeMake(targetSize.width - xStart, targetSize.height)];
if (childConfig.weight) {
size.width += remain / self.contentWeight * childConfig.weight;
}
- child.width = size.width;
- child.height = size.height;
+
+ [child layoutSelf:size];
DoricGravity gravity = childConfig.alignment | self.gravity;
if ((gravity & TOP) == TOP) {
@@ -473,45 +433,10 @@ - (void)layout:(CGSize)targetSize {
if (childConfig.margin.right) {
xStart += childConfig.margin.right;
}
- if ([child isKindOfClass:[DoricLayoutContainer class]]) {
- [(DoricLayoutContainer *) child layout:size];
- }
}
- self.width = targetSize.width;
- self.height = targetSize.height;
}
@end
-static const void *kLayoutConfig = &kLayoutConfig;
-static const void *kTagString = &kTagString;
-
-@implementation UIView (DoricLayoutConfig)
-@dynamic layoutConfig;
-
-- (void)setLayoutConfig:(DoricLayoutConfig *)layoutConfig {
- objc_setAssociatedObject(self, kLayoutConfig, layoutConfig, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
-}
-
-- (DoricLayoutConfig *)layoutConfig {
- return objc_getAssociatedObject(self, kLayoutConfig);
-}
-
-- (void)setTagString:(NSString *)tagString {
- objc_setAssociatedObject(self, kTagString, tagString, OBJC_ASSOCIATION_COPY_NONATOMIC);
- self.tag = [tagString hash];
-}
-
-- (NSString *)tagString {
- return objc_getAssociatedObject(self, kTagString);
-}
-
-
-- (UIView *)viewWithTagString:(NSString *)tagString {
- // notice the potential hash collision
- return [self viewWithTag:[tagString hash]];
-}
-
-@end
DoricVLayoutView *vLayout(NSArray <__kindof UIView *> *views) {
DoricVLayoutView *layout = [[DoricVLayoutView alloc] initWithFrame:CGRectZero];
diff --git a/iOS/Pod/Classes/Shader/DoricListNode.m b/iOS/Pod/Classes/Shader/DoricListNode.m
index b1aa6eb4..aa1d75cc 100644
--- a/iOS/Pod/Classes/Shader/DoricListNode.m
+++ b/iOS/Pod/Classes/Shader/DoricListNode.m
@@ -21,6 +21,7 @@
#import "DoricListNode.h"
#import "DoricExtensions.h"
#import "DoricListItemNode.h"
+#import "DoricLayouts.h"
@interface DoricTableViewCell : UITableViewCell
@property(nonatomic, strong) DoricListItemNode *doricListItemNode;
@@ -36,13 +37,22 @@ @implementation DoricTableView
- (CGSize)sizeThatFits:(CGSize)size {
if (self.subviews.count > 0) {
CGFloat width = size.width;
+ CGFloat height = 0;
+
for (UIView *child in self.subviews) {
- width = MAX(child.width, width);
+ CGSize childSize = [child measureSize:size];
+ width = MAX(childSize.width, width);
+ height += childSize.height;
}
- return CGSizeMake(width, size.width);
+ return CGSizeMake(width, MAX(height, size.height));
}
return size;
}
+
+- (void)layoutSelf:(CGSize)targetSize {
+ [super layoutSelf:targetSize];
+ [self reloadData];
+}
@end
@@ -74,9 +84,11 @@ - (UITableView *)build {
- (void)blendView:(UITableView *)view forPropName:(NSString *)name propValue:(id)prop {
if ([@"itemCount" isEqualToString:name]) {
self.itemCount = [prop unsignedIntegerValue];
+ [self.view reloadData];
} else if ([@"renderItem" isEqualToString:name]) {
[self.itemViewIds removeAllObjects];
[self clearSubModel];
+ [self.view reloadData];
} else if ([@"batchCount" isEqualToString:name]) {
self.batchCount = [prop unsignedIntegerValue];
} else {
@@ -109,8 +121,8 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N
DoricListItemNode *node = cell.doricListItemNode;
node.viewId = model[@"id"];
[node blend:props];
- [node.view setNeedsLayout];
- CGSize size = [node.view sizeThatFits:CGSizeMake(cell.width, cell.height)];
+ CGSize size = [node.view measureSize:CGSizeMake(tableView.width, tableView.height)];
+ [node.view layoutSelf:size];
[self callItem:position height:size.height];
return cell;
}
diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.h b/iOS/Pod/Classes/Shader/DoricScrollerNode.h
index c1184e41..ec8a2253 100644
--- a/iOS/Pod/Classes/Shader/DoricScrollerNode.h
+++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.h
@@ -22,5 +22,9 @@
#import
#import "DoricSuperNode.h"
-@interface DoricScrollerNode : DoricSuperNode
+@interface DoricScrollView : UIScrollView
+@property(nonatomic, strong) UIView *contentView;
+@end
+
+@interface DoricScrollerNode : DoricSuperNode
@end
\ No newline at end of file
diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m
index cc951f12..4d601f15 100644
--- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m
+++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m
@@ -22,27 +22,28 @@
#import "DoricScrollerNode.h"
#import "DoricExtensions.h"
-@interface DoricScrollView : UIScrollView
-@end
-
@implementation DoricScrollView
-- (void)layoutSubviews {
- [super layoutSubviews];
- if (self.subviews.count > 0) {
- UIView *child = self.subviews[0];
- [self setContentSize:child.frame.size];
+- (void)setContentView:(UIView *)contentView {
+ if (_contentView) {
+ [_contentView removeFromSuperview];
}
+ _contentView = contentView;
+ [self addSubview:contentView];
}
- (CGSize)sizeThatFits:(CGSize)size {
- if (self.subviews.count > 0) {
- UIView *child = self.subviews[0];
- CGSize childSize = [child sizeThatFits:size];
- return CGSizeMake(MIN(size.width, childSize.width), MIN(size.height, childSize.height));
+ if (self.contentView) {
+ return [self.contentView sizeThatFits:size];
}
return CGSizeZero;
}
+
+- (void)layoutSelf:(CGSize)targetSize {
+ [super layoutSelf:targetSize];
+ [self setContentSize:self.contentView.frame.size];
+}
+
@end
@interface DoricScrollerNode ()
@@ -51,7 +52,7 @@ @interface DoricScrollerNode ()
@end
@implementation DoricScrollerNode
-- (UIScrollView *)build {
+- (DoricScrollView *)build {
return [DoricScrollView new];
}
@@ -74,12 +75,11 @@ - (void)blend:(NSDictionary *)props {
[it blend:childProps];
}];
} else {
- [self.childNode.view removeFromSuperview];
self.childNode = [[DoricViewNode create:self.doricContext withType:type] also:^(DoricViewNode *it) {
it.viewId = viewId;
[it initWithSuperNode:self];
[it blend:childProps];
- [self.view addSubview:it.view];
+ self.view.contentView = it.view;
}];
}
}
@@ -88,12 +88,12 @@ - (void)blend:(NSDictionary *)props {
it.viewId = viewId;
[it initWithSuperNode:self];
[it blend:childProps];
- [self.view addSubview:it.view];
+ self.view.contentView = it.view;
}];
}
}
-- (void)blendView:(UIScrollView *)view forPropName:(NSString *)name propValue:(id)prop {
+- (void)blendView:(DoricScrollView *)view forPropName:(NSString *)name propValue:(id)prop {
if ([@"content" isEqualToString:name]) {
self.childViewId = prop;
} else {
@@ -104,4 +104,11 @@ - (void)blendView:(UIScrollView *)view forPropName:(NSString *)name propValue:(i
- (void)blendSubNode:(NSDictionary *)subModel {
[self.childNode blend:subModel[@"props"]];
}
-@end
\ No newline at end of file
+
+- (DoricViewNode *)subNodeWithViewId:(NSString *)viewId {
+ if ([viewId isEqualToString:self.childViewId]) {
+ return self.childNode;
+ }
+ return nil;
+}
+@end
diff --git a/iOS/Pod/Classes/Shader/DoricSlideItemNode.m b/iOS/Pod/Classes/Shader/DoricSlideItemNode.m
index a65cb62e..9e254323 100644
--- a/iOS/Pod/Classes/Shader/DoricSlideItemNode.m
+++ b/iOS/Pod/Classes/Shader/DoricSlideItemNode.m
@@ -25,6 +25,9 @@ @interface DoricSlideItemView : DoricStackView
@end
@implementation DoricSlideItemView
+- (void)layoutSubviews {
+ [super layoutSubviews];
+}
@end
@implementation DoricSlideItemNode
diff --git a/iOS/Pod/Classes/Shader/DoricSliderNode.m b/iOS/Pod/Classes/Shader/DoricSliderNode.m
index e409dcd4..9e495f8c 100644
--- a/iOS/Pod/Classes/Shader/DoricSliderNode.m
+++ b/iOS/Pod/Classes/Shader/DoricSliderNode.m
@@ -45,12 +45,18 @@ - (CGSize)sizeThatFits:(CGSize)size {
if (self.subviews.count > 0) {
CGFloat height = size.height;
for (UIView *child in self.subviews) {
- height = MAX(child.height, height);
+ CGSize childSize = [child measureSize:size];
+ height = MAX(childSize.height, height);
}
- return CGSizeMake(height, size.height);
+ return CGSizeMake(size.width, size.height);
}
return size;
}
+
+- (void)layoutSelf:(CGSize)targetSize {
+ [super layoutSelf:targetSize];
+ [self reloadData];
+}
@end
@implementation DoricSliderNode
@@ -80,9 +86,11 @@ - (UICollectionView *)build {
- (void)blendView:(UICollectionView *)view forPropName:(NSString *)name propValue:(id)prop {
if ([@"itemCount" isEqualToString:name]) {
self.itemCount = [prop unsignedIntegerValue];
+ [self.view reloadData];
} else if ([@"renderPage" isEqualToString:name]) {
[self.itemViewIds removeAllObjects];
[self clearSubModel];
+ [self.view reloadData];
} else if ([@"batchCount" isEqualToString:name]) {
self.batchCount = [prop unsignedIntegerValue];
} else {
@@ -120,7 +128,8 @@ - (__kindof UICollectionViewCell *)collectionView:(UICollectionView *)collection
DoricSlideItemNode *node = cell.doricSlideItemNode;
node.viewId = model[@"id"];
[node blend:props];
- [node.view setNeedsLayout];
+ CGSize size = [node.view measureSize:CGSizeMake(collectionView.width, collectionView.height)];
+ [node.view layoutSelf:size];
return cell;
}
diff --git a/iOS/Pod/Classes/Shader/DoricSuperNode.m b/iOS/Pod/Classes/Shader/DoricSuperNode.m
index ffcd08ea..232722b4 100644
--- a/iOS/Pod/Classes/Shader/DoricSuperNode.m
+++ b/iOS/Pod/Classes/Shader/DoricSuperNode.m
@@ -150,9 +150,11 @@ - (void)clearSubModel {
}
- (DoricViewNode *)subNodeWithViewId:(NSString *)viewId {
+ NSAssert(NO, @"Should override class:%@ ,method:%@.", NSStringFromClass([self class]),
+ NSStringFromSelector(_cmd));
return nil;
}
- (void)requestLayout {
[self.view setNeedsLayout];
}
-@end
\ No newline at end of file
+@end
diff --git a/iOS/Pod/Classes/Shader/DoricViewNode.m b/iOS/Pod/Classes/Shader/DoricViewNode.m
index 1b948504..3a432cbb 100644
--- a/iOS/Pod/Classes/Shader/DoricViewNode.m
+++ b/iOS/Pod/Classes/Shader/DoricViewNode.m
@@ -232,4 +232,18 @@ - (NSNumber *)getHeight {
return @(self.view.height);
}
+- (void)setRotation:(NSNumber *)rotation {
+ if (rotation.floatValue == 0) {
+ self.view.transform = CGAffineTransformIdentity;
+ } else {
+ self.view.transform = CGAffineTransformMakeRotation(M_PI * rotation.floatValue * 2);
+ }
+}
+
+- (NSNumber *)getRotation {
+ float radius = atan2f((float) self.view.transform.b, (float) self.view.transform.a);
+ float degree = (float) (radius / M_PI / 2);
+ return @(degree);
+}
+
@end
diff --git a/iOS/Pod/Classes/Util/DoricConstant.h b/iOS/Pod/Classes/Util/DoricConstant.h
index 8256c9a9..06b34fd2 100644
--- a/iOS/Pod/Classes/Util/DoricConstant.h
+++ b/iOS/Pod/Classes/Util/DoricConstant.h
@@ -32,6 +32,7 @@ extern NSString *const INJECT_REQUIRE;
extern NSString *const INJECT_TIMER_SET;
extern NSString *const INJECT_TIMER_CLEAR;
extern NSString *const INJECT_BRIDGE;
+extern NSString *const INJECT_EMPTY;
extern NSString *const TEMPLATE_CONTEXT_CREATE;
diff --git a/iOS/Pod/Classes/Util/DoricConstant.m b/iOS/Pod/Classes/Util/DoricConstant.m
index 6d936e1e..09a742a4 100644
--- a/iOS/Pod/Classes/Util/DoricConstant.m
+++ b/iOS/Pod/Classes/Util/DoricConstant.m
@@ -32,6 +32,7 @@
NSString *const INJECT_TIMER_SET = @"nativeSetTimer";
NSString *const INJECT_TIMER_CLEAR = @"nativeClearTimer";
NSString *const INJECT_BRIDGE = @"nativeBridge";
+NSString *const INJECT_EMPTY = @"nativeEmpty";
NSString *const TEMPLATE_CONTEXT_CREATE = @"Reflect.apply("
"function(doric,context,Entry,require,exports){" "\n"
diff --git a/js-framework/index.ts b/js-framework/index.ts
index 8a5de55b..6fa48cae 100644
--- a/js-framework/index.ts
+++ b/js-framework/index.ts
@@ -21,6 +21,7 @@ export * from "./src/ui/scroller"
export * from "./src/ui/widgets"
export * from "./src/ui/panel"
export * from "./src/ui/declarative"
+export * from "./src/ui/refreshable"
export * from "./src/util/color"
export * from './src/util/log'
export * from './src/util/types'
diff --git a/js-framework/src/ui/list.ts b/js-framework/src/ui/list.ts
index b59091cd..780c2302 100644
--- a/js-framework/src/ui/list.ts
+++ b/js-framework/src/ui/list.ts
@@ -57,6 +57,10 @@ export class List extends Superview implements IList {
@Property
batchCount = 15
+ reset() {
+ this.cachedViews.clear()
+ this.itemCount = 0
+ }
private getItem(itemIdx: number) {
let view = this.cachedViews.get(`${itemIdx}`)
if (view === undefined) {
diff --git a/js-framework/src/ui/panel.ts b/js-framework/src/ui/panel.ts
index a1473451..32115c37 100644
--- a/js-framework/src/ui/panel.ts
+++ b/js-framework/src/ui/panel.ts
@@ -31,6 +31,8 @@ export function NativeCall(target: Panel, propertyKey: string, descriptor: Prope
type Frame = { width: number, height: number }
+declare function nativeEmpty(): void
+
export abstract class Panel {
context?: any
onCreate() { }
@@ -132,7 +134,7 @@ export abstract class Panel {
private hookAfterNativeCall() {
//Here insert a native call to ensure the promise is resolved done.
- log('Check Dirty')
+ nativeEmpty()
if (this.__root__.isDirty()) {
const model = this.__root__.toModel()
this.nativeRender(model)
diff --git a/js-framework/src/ui/refreshable.ts b/js-framework/src/ui/refreshable.ts
new file mode 100644
index 00000000..556196e3
--- /dev/null
+++ b/js-framework/src/ui/refreshable.ts
@@ -0,0 +1,74 @@
+import { View, Property, Superview, IView } from "./view";
+import { List } from "./list";
+import { Scroller } from "./scroller";
+import { BridgeContext } from "../runtime/global";
+import { layoutConfig } from "./declarative";
+
+export interface IRefreshable extends IView {
+ content: List | Scroller
+ header?: View
+ onRefresh?: () => void
+}
+
+export class Refreshable extends Superview implements IRefreshable {
+
+ content!: List | Scroller
+
+ header?: View
+
+ @Property
+ onRefresh?: () => void
+
+ allSubviews() {
+ const ret: View[] = [this.content]
+ if (this.header) {
+ ret.push(this.header)
+ }
+ return ret
+ }
+
+ setRefreshable(context: BridgeContext, refreshable: boolean) {
+ return this.nativeChannel(context, 'setRefreshable')(refreshable)
+ }
+
+ setRefreshing(context: BridgeContext, refreshing: boolean) {
+ return this.nativeChannel(context, 'setRefreshing')(refreshing)
+ }
+
+ isRefreshable(context: BridgeContext) {
+ return this.nativeChannel(context, 'isRefreshable')() as Promise
+ }
+
+ isRefreshing(context: BridgeContext) {
+ return this.nativeChannel(context, 'isRefreshing')() as Promise
+ }
+
+ toModel() {
+ this.dirtyProps.content = this.content.viewId
+ this.dirtyProps.header = (this.header || {}).viewId
+ return super.toModel()
+ }
+}
+
+export function refreshable(config: IRefreshable) {
+ const ret = new Refreshable
+ ret.layoutConfig = layoutConfig().wrap()
+ for (let key in config) {
+ Reflect.set(ret, key, Reflect.get(config, key, config), ret)
+ }
+ return ret
+}
+
+export interface IPullable {
+ startAnimation(): void
+ stopAnimation(): void
+ setProgressRotation(rotation: number): void
+}
+
+
+export function pullable(context: BridgeContext, v: View, config: IPullable) {
+ Reflect.set(v, 'startAnimation', config.startAnimation)
+ Reflect.set(v, 'stopAnimation', config.stopAnimation)
+ Reflect.set(v, 'setProgressRotation', config.setProgressRotation)
+ return v
+}
\ No newline at end of file
diff --git a/js-framework/src/ui/view.ts b/js-framework/src/ui/view.ts
index 893bf4dc..1cabf6e0 100644
--- a/js-framework/src/ui/view.ts
+++ b/js-framework/src/ui/view.ts
@@ -18,6 +18,7 @@ import { Modeling, Model, obj2Model } from "../util/types";
import { uniqueId } from "../util/uniqueId";
import { Gravity } from "../util/gravity";
import { loge } from "../util/log";
+import { BridgeContext } from "../runtime/global";
export enum LayoutSpec {
EXACTLY = 0,
@@ -266,7 +267,7 @@ export abstract class View implements Modeling, IView {
nativeChannel(context: any, name: string) {
let thisView: View | undefined = this
- return function (...args: any) {
+ return function (args: any = undefined) {
const func = context.shader.command
const viewIds = []
while (thisView != undefined) {
@@ -281,6 +282,29 @@ export abstract class View implements Modeling, IView {
return Reflect.apply(func, undefined, [params]) as Promise
}
}
+
+ getWidth(context: BridgeContext) {
+ return this.nativeChannel(context, 'getWidth')() as Promise
+ }
+
+ getHeight(context: BridgeContext) {
+ return this.nativeChannel(context, 'getHeight')() as Promise
+ }
+
+ /**
+ *
+ * @param rotation [0..1]
+ */
+ setRotation(context: BridgeContext, rotation: number) {
+ return this.nativeChannel(context, 'setRotation')(rotation)
+ }
+ /**
+ *
+ * @return rotation [0..1]
+ */
+ getRotation(context: BridgeContext) {
+ return this.nativeChannel(context, 'getRotation')() as Promise
+ }
}
export abstract class Superview extends View {
@@ -316,6 +340,7 @@ export abstract class Superview extends View {
toModel() {
const subviews = []
for (let v of this.allSubviews()) {
+ v.superview = this
if (v.isDirty()) {
subviews.push(v.toModel())
}
@@ -341,7 +366,6 @@ export abstract class Group extends Superview {
}
addChild(view: View) {
- view.superview = this
this.children.push(view)
}
}
diff --git a/js-framework/tsconfig.json b/js-framework/tsconfig.json
index d511f6a0..7946f238 100644
--- a/js-framework/tsconfig.json
+++ b/js-framework/tsconfig.json
@@ -5,7 +5,7 @@
// "incremental": true, /* Enable incremental compilation */
"target": "ES2015", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019' or 'ESNEXT'. */
"module": "es2015", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', or 'ESNext'. */
- // "lib": [], /* Specify library files to be included in the compilation. */
+ "lib": [], /* Specify library files to be included in the compilation. */
// "allowJs": true, /* Allow javascript files to be compiled. */
// "checkJs": true, /* Report errors in .js files. */
// "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */