From d3b2d4c8bd36421f49f05141eb0f99b2ee83ac07 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Mon, 25 Nov 2019 17:15:43 +0800 Subject: [PATCH 01/35] feat:add SwipeLayout --- Android/app/build.gradle | 1 + Android/app/src/main/AndroidManifest.xml | 2 + .../java/pub/doric/demo/MainActivity.java | 6 + .../java/pub/doric/demo/PullableActivity.java | 28 + .../src/main/res/layout/activity_pullable.xml | 13 + .../pub/doric/pullable/CircleImageView.java | 159 +++ .../pub/doric/pullable/DoricRefreshView.java | 80 ++ .../pub/doric/pullable/DoricSwipeLayout.java | 1182 +++++++++++++++++ .../java/pub/doric/pullable/IPullable.java | 27 + 9 files changed, 1498 insertions(+) create mode 100644 Android/app/src/main/java/pub/doric/demo/PullableActivity.java create mode 100644 Android/app/src/main/res/layout/activity_pullable.xml create mode 100644 Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java create mode 100644 Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java create mode 100644 Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java create mode 100644 Android/doric/src/main/java/pub/doric/pullable/IPullable.java 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..4eaf9ad9 100644 --- a/Android/app/src/main/java/pub/doric/demo/MainActivity.java +++ b/Android/app/src/main/java/pub/doric/demo/MainActivity.java @@ -48,6 +48,7 @@ public class MainActivity extends AppCompatActivity { 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 +92,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..615f27c5 --- /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.pullable.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_pullable.xml b/Android/app/src/main/res/layout/activity_pullable.xml new file mode 100644 index 00000000..e9bad365 --- /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/pullable/CircleImageView.java b/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java new file mode 100644 index 00000000..70f663dd --- /dev/null +++ b/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java @@ -0,0 +1,159 @@ +/* + * Copyright 2018 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package pub.doric.pullable; + +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.graphics.RadialGradient; +import android.graphics.Shader; +import android.graphics.drawable.ShapeDrawable; +import android.graphics.drawable.shapes.OvalShape; +import android.view.View; +import android.view.animation.Animation; +import android.widget.ImageView; + +import androidx.core.content.ContextCompat; +import androidx.core.view.ViewCompat; + +/** + * Private class created to work around issues with AnimationListeners being + * called before the animation is actually complete and support shadows on older + * platforms. + */ +class CircleImageView extends ImageView { + + private static final int KEY_SHADOW_COLOR = 0x1E000000; + private static final int FILL_SHADOW_COLOR = 0x3D000000; + // PX + private static final float X_OFFSET = 0f; + private static final float Y_OFFSET = 1.75f; + private static final float SHADOW_RADIUS = 3.5f; + private static final int SHADOW_ELEVATION = 4; + + private Animation.AnimationListener mListener; + int mShadowRadius; + + CircleImageView(Context context, int color) { + super(context); + final float density = getContext().getResources().getDisplayMetrics().density; + final int shadowYOffset = (int) (density * Y_OFFSET); + final int shadowXOffset = (int) (density * X_OFFSET); + + mShadowRadius = (int) (density * SHADOW_RADIUS); + + ShapeDrawable circle; + if (elevationSupported()) { + circle = new ShapeDrawable(new OvalShape()); + ViewCompat.setElevation(this, SHADOW_ELEVATION * density); + } else { + OvalShape oval = new OvalShadow(mShadowRadius); + circle = new ShapeDrawable(oval); + setLayerType(View.LAYER_TYPE_SOFTWARE, circle.getPaint()); + circle.getPaint().setShadowLayer(mShadowRadius, shadowXOffset, shadowYOffset, + KEY_SHADOW_COLOR); + final int padding = mShadowRadius; + // set padding so the inner image sits correctly within the shadow. + setPadding(padding, padding, padding, padding); + } + circle.getPaint().setColor(color); + ViewCompat.setBackground(this, circle); + } + + private boolean elevationSupported() { + return android.os.Build.VERSION.SDK_INT >= 21; + } + + @Override + protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { + super.onMeasure(widthMeasureSpec, heightMeasureSpec); + if (!elevationSupported()) { + setMeasuredDimension(getMeasuredWidth() + mShadowRadius * 2, getMeasuredHeight() + + mShadowRadius * 2); + } + } + + public void setAnimationListener(Animation.AnimationListener listener) { + mListener = listener; + } + + @Override + public void onAnimationStart() { + super.onAnimationStart(); + if (mListener != null) { + mListener.onAnimationStart(getAnimation()); + } + } + + @Override + public void onAnimationEnd() { + super.onAnimationEnd(); + if (mListener != null) { + mListener.onAnimationEnd(getAnimation()); + } + } + + /** + * Update the background color of the circle image view. + * + * @param colorRes Id of a color resource. + */ + public void setBackgroundColorRes(int colorRes) { + setBackgroundColor(ContextCompat.getColor(getContext(), colorRes)); + } + + @Override + public void setBackgroundColor(int color) { + if (getBackground() instanceof ShapeDrawable) { + ((ShapeDrawable) getBackground()).getPaint().setColor(color); + } + } + + private class OvalShadow extends OvalShape { + private RadialGradient mRadialGradient; + private Paint mShadowPaint; + + OvalShadow(int shadowRadius) { + super(); + mShadowPaint = new Paint(); + mShadowRadius = shadowRadius; + updateRadialGradient((int) rect().width()); + } + + @Override + protected void onResize(float width, float height) { + super.onResize(width, height); + updateRadialGradient((int) width); + } + + @Override + public void draw(Canvas canvas, Paint paint) { + final int viewWidth = CircleImageView.this.getWidth(); + final int viewHeight = CircleImageView.this.getHeight(); + canvas.drawCircle(viewWidth / 2, viewHeight / 2, viewWidth / 2, mShadowPaint); + canvas.drawCircle(viewWidth / 2, viewHeight / 2, viewWidth / 2 - mShadowRadius, paint); + } + + private void updateRadialGradient(int diameter) { + mRadialGradient = new RadialGradient(diameter / 2, diameter / 2, + mShadowRadius, new int[]{FILL_SHADOW_COLOR, Color.TRANSPARENT}, + null, Shader.TileMode.CLAMP); + mShadowPaint.setShader(mRadialGradient); + } + } +} diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java b/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java new file mode 100644 index 00000000..90e376f2 --- /dev/null +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java @@ -0,0 +1,80 @@ +package pub.doric.pullable; + +import android.content.Context; +import android.util.AttributeSet; +import android.view.Gravity; +import android.view.View; +import android.view.ViewGroup; +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 IPullable { + private View content; + + 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; + } + + @Override + public void startAnimation() { + if (content != null && content instanceof IPullable) { + ((IPullable) content).startAnimation(); + } + } + + @Override + public void stopAnimation() { + if (content != null && content instanceof IPullable) { + ((IPullable) content).stopAnimation(); + } + } + + @Override + public int successAnimation() { + if (content != null && content instanceof IPullable) { + return ((IPullable) content).successAnimation(); + } else { + return 0; + } + } + + @Override + public void setProgressRotation(float rotation) { + if (content != null && content instanceof IPullable) { + ((IPullable) content).setProgressRotation(rotation); + } + } +} \ No newline at end of file diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java new file mode 100644 index 00000000..d128dc6e --- /dev/null +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -0,0 +1,1182 @@ +package pub.doric.pullable; + +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.ColorInt; +import androidx.annotation.ColorRes; +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.annotation.Px; +import androidx.annotation.VisibleForTesting; +import androidx.core.content.ContextCompat; +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 = androidx.swiperefreshlayout.widget.SwipeRefreshLayout.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 background for the progress spinner + private static final int CIRCLE_BG_LIGHT = 0xFFFAFAFA; + // 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; + // Whether this item is scaled up rather than clipped + boolean mScale; + + // 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 + }; + + CircleImageView mCircleView; + private int mCircleViewIndex = -1; + + protected int mFrom; + + float mStartingScale; + + protected int mOriginalOffsetTop; + + int mSpinnerOffsetEnd; + + int mCustomSlingshotDistance; + + CircularProgressDrawable mProgress; + + private Animation mScaleAnimation; + + private Animation mScaleDownAnimation; + + private Animation mAlphaStartAnimation; + + private Animation mAlphaMaxAnimation; + + private Animation mScaleDownToStartAnimation; + + boolean mNotify; + + private int mCircleDiameter; + + // Whether the client has set a custom starting position; + boolean mUsingCustomStart; + + private OnChildScrollUpCallback mChildScrollUpCallback; + + 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) { + // Make sure the progress view is fully visible + mProgress.setAlpha(MAX_ALPHA); + mProgress.start(); + if (mNotify) { + if (mListener != null) { + mListener.onRefresh(); + } + } + mCurrentTargetOffsetTop = mCircleView.getTop(); + } else { + reset(); + } + } + }; + + void reset() { + mCircleView.clearAnimation(); + mProgress.stop(); + mCircleView.setVisibility(View.GONE); + setColorViewAlpha(MAX_ALPHA); + // Return the circle to its start position + if (mScale) { + setAnimationProgress(0 /* animation complete and view is hidden */); + } else { + setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCurrentTargetOffsetTop); + } + mCurrentTargetOffsetTop = mCircleView.getTop(); + } + + @Override + public void setEnabled(boolean enabled) { + super.setEnabled(enabled); + if (!enabled) { + reset(); + } + } + + @Override + protected void onDetachedFromWindow() { + super.onDetachedFromWindow(); + reset(); + } + + private void setColorViewAlpha(int targetAlpha) { + mCircleView.getBackground().setAlpha(targetAlpha); + mProgress.setAlpha(targetAlpha); + } + + /** + * The refresh indicator starting and resting position is always positioned + * near the top of the refreshing content. This position is a consistent + * location, but can be adjusted in either direction based on whether or not + * there is a toolbar or actionbar present. + *

+ * Note: Calling this will reset the position of the refresh indicator to + * start. + *

+ * + * @param scale Set to true if there is no view at a higher z-order than where the progress + * spinner is set to appear. Setting it to true will cause indicator to be scaled + * up rather than clipped. + * @param start The offset in pixels from the top of this view at which the + * progress spinner should appear. + * @param end The offset in pixels from the top of this view at which the + * progress spinner should come to rest after a successful swipe + * gesture. + */ + public void setProgressViewOffset(boolean scale, int start, int end) { + mScale = scale; + mOriginalOffsetTop = start; + mSpinnerOffsetEnd = end; + mUsingCustomStart = true; + reset(); + mRefreshing = false; + } + + /** + * @return The offset in pixels from the top of this view at which the progress spinner should + * appear. + */ + public int getProgressViewStartOffset() { + return mOriginalOffsetTop; + } + + /** + * @return The offset in pixels from the top of this view at which the progress spinner should + * come to rest after a successful swipe gesture. + */ + public int getProgressViewEndOffset() { + return mSpinnerOffsetEnd; + } + + /** + * The refresh indicator resting position is always positioned near the top + * of the refreshing content. This position is a consistent location, but + * can be adjusted in either direction based on whether or not there is a + * toolbar or actionbar present. + * + * @param scale Set to true if there is no view at a higher z-order than where the progress + * spinner is set to appear. Setting it to true will cause indicator to be scaled + * up rather than clipped. + * @param end The offset in pixels from the top of this view at which the + * progress spinner should come to rest after a successful swipe + * gesture. + */ + public void setProgressViewEndTarget(boolean scale, int end) { + mSpinnerOffsetEnd = end; + mScale = scale; + mCircleView.invalidate(); + } + + /** + * Sets a custom slingshot distance. + * + * @param slingshotDistance The distance in pixels that the refresh indicator can be pulled + * beyond its resting position. Use + * {@link #DEFAULT_SLINGSHOT_DISTANCE} to reset to the default value. + */ + public void setSlingshotDistance(@Px int slingshotDistance) { + mCustomSlingshotDistance = slingshotDistance; + } + + /** + * One of DEFAULT, or LARGE. + */ + public void setSize(int size) { + if (size != CircularProgressDrawable.LARGE && size != CircularProgressDrawable.DEFAULT) { + return; + } + final DisplayMetrics metrics = getResources().getDisplayMetrics(); + if (size == CircularProgressDrawable.LARGE) { + mCircleDiameter = (int) (CIRCLE_DIAMETER_LARGE * metrics.density); + } else { + mCircleDiameter = (int) (CIRCLE_DIAMETER * metrics.density); + } + // force the bounds of the progress circle inside the circle view to + // update by setting it to null before updating its size and then + // re-setting it + mCircleView.setImageDrawable(null); + mProgress.setStyle(size); + mCircleView.setImageDrawable(mProgress); + } + + /** + * 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(); + mCircleDiameter = (int) (CIRCLE_DIAMETER * metrics.density); + + 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); + + mOriginalOffsetTop = mCurrentTargetOffsetTop = -mCircleDiameter; + moveToStart(1.0f); + + final TypedArray a = context.obtainStyledAttributes(attrs, LAYOUT_ATTRS); + setEnabled(a.getBoolean(0, true)); + a.recycle(); + } + + @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() { + mCircleView = new CircleImageView(getContext(), CIRCLE_BG_LIGHT); + mProgress = new CircularProgressDrawable(getContext()); + mProgress.setStyle(CircularProgressDrawable.DEFAULT); + mCircleView.setImageDrawable(mProgress); + mCircleView.setVisibility(View.GONE); + addView(mCircleView); + } + + /** + * 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) { + mCircleView.setVisibility(View.VISIBLE); + mProgress.setAlpha(MAX_ALPHA); + mScaleAnimation = new Animation() { + @Override + public void applyTransformation(float interpolatedTime, Transformation t) { + setAnimationProgress(interpolatedTime); + } + }; + mScaleAnimation.setDuration(mMediumAnimationDuration); + if (listener != null) { + mCircleView.setAnimationListener(listener); + } + mCircleView.clearAnimation(); + mCircleView.startAnimation(mScaleAnimation); + } + + /** + * Pre API 11, this does an alpha animation. + * + * @param progress + */ + void setAnimationProgress(float progress) { + mCircleView.setScaleX(progress); + mCircleView.setScaleY(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); + mCircleView.setAnimationListener(listener); + mCircleView.clearAnimation(); + mCircleView.startAnimation(mScaleDownAnimation); + } + + private void startProgressAlphaStartAnimation() { + mAlphaStartAnimation = startAlphaAnimation(mProgress.getAlpha(), STARTING_PROGRESS_ALPHA); + } + + private void startProgressAlphaMaxAnimation() { + mAlphaMaxAnimation = startAlphaAnimation(mProgress.getAlpha(), MAX_ALPHA); + } + + private Animation startAlphaAnimation(final int startingAlpha, final int endingAlpha) { + Animation alpha = new Animation() { + @Override + public void applyTransformation(float interpolatedTime, Transformation t) { + mProgress.setAlpha( + (int) (startingAlpha + ((endingAlpha - startingAlpha) * interpolatedTime))); + } + }; + alpha.setDuration(ALPHA_ANIMATION_DURATION); + // Clear out the previous animation listeners. + mCircleView.setAnimationListener(null); + mCircleView.clearAnimation(); + mCircleView.startAnimation(alpha); + return alpha; + } + + /** + * @deprecated Use {@link #setProgressBackgroundColorSchemeResource(int)} + */ + @Deprecated + public void setProgressBackgroundColor(int colorRes) { + setProgressBackgroundColorSchemeResource(colorRes); + } + + /** + * Set the background color of the progress spinner disc. + * + * @param colorRes Resource id of the color. + */ + public void setProgressBackgroundColorSchemeResource(@ColorRes int colorRes) { + setProgressBackgroundColorSchemeColor(ContextCompat.getColor(getContext(), colorRes)); + } + + /** + * Set the background color of the progress spinner disc. + * + * @param color + */ + public void setProgressBackgroundColorSchemeColor(@ColorInt int color) { + mCircleView.setBackgroundColor(color); + } + + /** + * @deprecated Use {@link #setColorSchemeResources(int...)} + */ + @Deprecated + public void setColorScheme(@ColorRes int... colors) { + setColorSchemeResources(colors); + } + + /** + * Set the color resources used in the progress animation from color resources. + * The first color will also be the color of the bar that grows in response + * to a user swipe gesture. + * + * @param colorResIds + */ + public void setColorSchemeResources(@ColorRes int... colorResIds) { + final Context context = getContext(); + int[] colorRes = new int[colorResIds.length]; + for (int i = 0; i < colorResIds.length; i++) { + colorRes[i] = ContextCompat.getColor(context, colorResIds[i]); + } + setColorSchemeColors(colorRes); + } + + /** + * Set the colors used in the progress animation. The first + * color will also be the color of the bar that grows in response to a user + * swipe gesture. + * + * @param colors + */ + public void setColorSchemeColors(@ColorInt int... colors) { + ensureTarget(); + mProgress.setColorSchemeColors(colors); + } + + /** + * @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(mCircleView)) { + mTarget = child; + break; + } + } + } + } + + /** + * Set the distance to trigger a sync in dips + * + * @param distance + */ + public void setDistanceToTriggerSync(int distance) { + mTotalDragDistance = distance; + } + + @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; + } + final View child = mTarget; + final int childLeft = getPaddingLeft(); + final int childTop = getPaddingTop(); + final int childWidth = width - getPaddingLeft() - getPaddingRight(); + final int childHeight = height - getPaddingTop() - getPaddingBottom(); + child.layout(childLeft, childTop, childLeft + childWidth, childTop + childHeight); + int circleWidth = mCircleView.getMeasuredWidth(); + int circleHeight = mCircleView.getMeasuredHeight(); + mCircleView.layout((width / 2 - circleWidth / 2), mCurrentTargetOffsetTop, + (width / 2 + circleWidth / 2), mCurrentTargetOffsetTop + circleHeight); + } + + @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)); + mCircleView.measure(MeasureSpec.makeMeasureSpec(mCircleDiameter, MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(mCircleDiameter, MeasureSpec.EXACTLY)); + mCircleViewIndex = -1; + // Get the index of the circleview. + for (int index = 0; index < getChildCount(); index++) { + if (getChildAt(index) == mCircleView) { + mCircleViewIndex = index; + break; + } + } + } + + /** + * Get the diameter of the progress circle that is displayed as part of the + * swipe to refresh layout. + * + * @return Diameter in pixels of the progress circle view. + */ + public int getProgressCircleDiameter() { + return mCircleDiameter; + } + + /** + * @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 - mCircleView.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 { + 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) { + mCircleView.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) { + mProgress.setArrowEnabled(true); + float originalDragPercent = overscrollTop / mTotalDragDistance; + + float dragPercent = Math.min(1f, Math.abs(originalDragPercent)); + float adjustedPercent = (float) Math.max(dragPercent - .4, 0) * 5 / 3; + 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 (mCircleView.getVisibility() != View.VISIBLE) { + mCircleView.setVisibility(View.VISIBLE); + } + if (!mScale) { + mCircleView.setScaleX(1f); + mCircleView.setScaleY(1f); + } + + if (mScale) { + setAnimationProgress(Math.min(1f, overscrollTop / mTotalDragDistance)); + } + if (overscrollTop < mTotalDragDistance) { + if (mProgress.getAlpha() > STARTING_PROGRESS_ALPHA + && !isAnimationRunning(mAlphaStartAnimation)) { + // Animate the alpha + startProgressAlphaStartAnimation(); + } + } else { + if (mProgress.getAlpha() < MAX_ALPHA && !isAnimationRunning(mAlphaMaxAnimation)) { + // Animate the alpha + startProgressAlphaMaxAnimation(); + } + } + float strokeStart = adjustedPercent * .8f; + mProgress.setStartEndTrim(0f, Math.min(MAX_PROGRESS_ANGLE, strokeStart)); + mProgress.setArrowScale(Math.min(1f, adjustedPercent)); + + float rotation = (-0.25f + .4f * adjustedPercent + tensionPercent * 2) * .5f; + mProgress.setProgressRotation(rotation); + setTargetOffsetTopAndBottom(targetY - mCurrentTargetOffsetTop); + } + + private void finishSpinner(float overscrollTop) { + if (overscrollTop > mTotalDragDistance) { + setRefreshing(true, true /* notify */); + } else { + // cancel refresh + mRefreshing = false; + mProgress.setStartEndTrim(0f, 0f); + Animation.AnimationListener listener = null; + if (!mScale) { + listener = new Animation.AnimationListener() { + + @Override + public void onAnimationStart(Animation animation) { + } + + @Override + public void onAnimationEnd(Animation animation) { + if (!mScale) { + startScaleDownAnimation(null); + } + } + + @Override + public void onAnimationRepeat(Animation animation) { + } + + }; + } + animateOffsetToStartPosition(mCurrentTargetOffsetTop, listener); + mProgress.setArrowEnabled(false); + } + } + + @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; + mProgress.setAlpha(STARTING_PROGRESS_ALPHA); + } + } + + private void animateOffsetToCorrectPosition(int from, AnimationListener listener) { + mFrom = from; + mAnimateToCorrectPosition.reset(); + mAnimateToCorrectPosition.setDuration(ANIMATE_TO_TRIGGER_DURATION); + mAnimateToCorrectPosition.setInterpolator(mDecelerateInterpolator); + if (listener != null) { + mCircleView.setAnimationListener(listener); + } + mCircleView.clearAnimation(); + mCircleView.startAnimation(mAnimateToCorrectPosition); + } + + private void animateOffsetToStartPosition(int from, AnimationListener listener) { + if (mScale) { + // Scale the item back down + startScaleDownReturnToStartAnimation(from, listener); + } else { + mFrom = from; + mAnimateToStartPosition.reset(); + mAnimateToStartPosition.setDuration(ANIMATE_TO_START_DURATION); + mAnimateToStartPosition.setInterpolator(mDecelerateInterpolator); + if (listener != null) { + mCircleView.setAnimationListener(listener); + } + mCircleView.clearAnimation(); + mCircleView.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 - mCircleView.getTop(); + setTargetOffsetTopAndBottom(offset); + mProgress.setArrowScale(1 - interpolatedTime); + } + }; + + void moveToStart(float interpolatedTime) { + int targetTop = 0; + targetTop = (mFrom + (int) ((mOriginalOffsetTop - mFrom) * interpolatedTime)); + int offset = targetTop - mCircleView.getTop(); + setTargetOffsetTopAndBottom(offset); + } + + private final Animation mAnimateToStartPosition = new Animation() { + @Override + public void applyTransformation(float interpolatedTime, Transformation t) { + moveToStart(interpolatedTime); + } + }; + + private void startScaleDownReturnToStartAnimation(int from, + Animation.AnimationListener listener) { + mFrom = from; + mStartingScale = mCircleView.getScaleX(); + mScaleDownToStartAnimation = new Animation() { + @Override + public void applyTransformation(float interpolatedTime, Transformation t) { + float targetScale = (mStartingScale + (-mStartingScale * interpolatedTime)); + setAnimationProgress(targetScale); + moveToStart(interpolatedTime); + } + }; + mScaleDownToStartAnimation.setDuration(SCALE_DOWN_DURATION); + if (listener != null) { + mCircleView.setAnimationListener(listener); + } + mCircleView.clearAnimation(); + mCircleView.startAnimation(mScaleDownToStartAnimation); + } + + void setTargetOffsetTopAndBottom(int offset) { + mCircleView.bringToFront(); + ViewCompat.offsetTopAndBottom(mCircleView, offset); + mCurrentTargetOffsetTop = mCircleView.getTop(); + } + + 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/pullable/IPullable.java b/Android/doric/src/main/java/pub/doric/pullable/IPullable.java new file mode 100644 index 00000000..83bb96af --- /dev/null +++ b/Android/doric/src/main/java/pub/doric/pullable/IPullable.java @@ -0,0 +1,27 @@ +package pub.doric.pullable; + +/** + * @Description: pub.doric.pullable + * @Author: pengfei.zhou + * @CreateDate: 2019-11-25 + */ +public interface IPullable { + + void startAnimation(); + + void stopAnimation(); + + /** + * run the animation after pull request success and before stop animation + * + * @return the duration of success animation or 0 if no success animation + */ + int successAnimation(); + + /** + * Set the amount of rotation to apply to the progress spinner. + * + * @param rotation Rotation is from [0..1] + */ + void setProgressRotation(float rotation); +} From a47b3682db9693dc9388a7cdd0a9bf3f1b148d13 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Mon, 25 Nov 2019 18:54:27 +0800 Subject: [PATCH 02/35] feat:fix shaking when drag --- .../java/pub/doric/demo/MainActivity.java | 8 +++++++ .../app/src/main/res/layout/activity_main.xml | 15 ++++++++---- .../pub/doric/pullable/CircleImageView.java | 4 ++-- .../pub/doric/pullable/DoricSwipeLayout.java | 23 ++++++++++++------- 4 files changed, 35 insertions(+), 15 deletions(-) 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 4eaf9ad9..c472cc8c 100644 --- a/Android/app/src/main/java/pub/doric/demo/MainActivity.java +++ b/Android/app/src/main/java/pub/doric/demo/MainActivity.java @@ -34,6 +34,7 @@ import java.util.List; import pub.doric.DoricActivity; import pub.doric.devkit.ui.DemoDebugActivity; +import pub.doric.pullable.DoricSwipeLayout; import pub.doric.utils.DoricUtils; public class MainActivity extends AppCompatActivity { @@ -43,6 +44,13 @@ 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); + } + }); RecyclerView recyclerView = findViewById(R.id.root); recyclerView.setLayoutManager(new LinearLayoutManager(this)); try { diff --git a/Android/app/src/main/res/layout/activity_main.xml b/Android/app/src/main/res/layout/activity_main.xml index 4abb9bc3..95619501 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/doric/src/main/java/pub/doric/pullable/CircleImageView.java b/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java index 70f663dd..b003051b 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java +++ b/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java @@ -26,8 +26,8 @@ import android.graphics.drawable.ShapeDrawable; import android.graphics.drawable.shapes.OvalShape; import android.view.View; import android.view.animation.Animation; -import android.widget.ImageView; +import androidx.appcompat.widget.AppCompatImageView; import androidx.core.content.ContextCompat; import androidx.core.view.ViewCompat; @@ -36,7 +36,7 @@ import androidx.core.view.ViewCompat; * called before the animation is actually complete and support shadows on older * platforms. */ -class CircleImageView extends ImageView { +class CircleImageView extends AppCompatImageView { private static final int KEY_SHADOW_COLOR = 0x1E000000; private static final int FILL_SHADOW_COLOR = 0x3D000000; diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java index d128dc6e..3a1d9245 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -51,7 +51,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent @VisibleForTesting static final int CIRCLE_DIAMETER_LARGE = 56; - private static final String LOG_TAG = androidx.swiperefreshlayout.widget.SwipeRefreshLayout.class.getSimpleName(); + 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); @@ -587,16 +587,19 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent if (mTarget == null) { return; } + + int circleWidth = mCircleView.getMeasuredWidth(); + int circleHeight = mCircleView.getMeasuredHeight(); + + mCircleView.layout((width / 2 - circleWidth / 2), mCurrentTargetOffsetTop, + (width / 2 + circleWidth / 2), mCurrentTargetOffsetTop + circleHeight); + final View child = mTarget; final int childLeft = getPaddingLeft(); - final int childTop = getPaddingTop(); + final int childTop = getPaddingTop() + mCircleView.getBottom(); final int childWidth = width - getPaddingLeft() - getPaddingRight(); final int childHeight = height - getPaddingTop() - getPaddingBottom(); child.layout(childLeft, childTop, childLeft + childWidth, childTop + childHeight); - int circleWidth = mCircleView.getMeasuredWidth(); - int circleHeight = mCircleView.getMeasuredHeight(); - mCircleView.layout((width / 2 - circleWidth / 2), mCurrentTargetOffsetTop, - (width / 2 + circleWidth / 2), mCurrentTargetOffsetTop + circleHeight); } @Override @@ -756,8 +759,10 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent consumed[1] = dy - (int) mTotalUnconsumed; mTotalUnconsumed = 0; } else { - mTotalUnconsumed -= dy; - consumed[1] = dy; + if (dy > 3) { + mTotalUnconsumed -= dy; + consumed[1] = dy; + } } moveSpinner(mTotalUnconsumed); } @@ -1179,4 +1184,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent */ boolean canChildScrollUp(@NonNull DoricSwipeLayout parent, @Nullable View child); } + + } \ No newline at end of file From 9c6802dccfcd3b5821ca0794fd79f49cafdd75ae Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Mon, 25 Nov 2019 19:28:42 +0800 Subject: [PATCH 03/35] feat:fix SwipeLayout --- .../java/pub/doric/demo/MainActivity.java | 5 + .../pub/doric/pullable/DoricRefreshView.java | 22 ++ .../pub/doric/pullable/DoricSwipeLayout.java | 272 ++++-------------- 3 files changed, 84 insertions(+), 215 deletions(-) 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 c472cc8c..7d3ece4f 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; @@ -51,7 +52,11 @@ public class MainActivity extends AppCompatActivity { swipeLayout.setRefreshing(false); } }); + swipeLayout.setBackgroundColor(Color.YELLOW); + swipeLayout.getRefreshView().setBackgroundColor(Color.RED); + swipeLayout.setPullDownHeight(100); RecyclerView recyclerView = findViewById(R.id.root); + recyclerView.setBackgroundColor(Color.WHITE); recyclerView.setLayoutManager(new LinearLayoutManager(this)); try { String[] demos = getAssets().list("demo"); diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java b/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java index 90e376f2..2d404750 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java @@ -5,6 +5,7 @@ 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; @@ -18,6 +19,7 @@ import androidx.annotation.Nullable; */ public class DoricRefreshView extends FrameLayout implements IPullable { private View content; + private Animation.AnimationListener mListener; public DoricRefreshView(@NonNull Context context) { super(context); @@ -77,4 +79,24 @@ public class DoricRefreshView extends FrameLayout implements IPullable { ((IPullable) content).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/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java index 3a1d9245..f83c3431 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -15,13 +15,10 @@ import android.view.animation.Transformation; import android.widget.AbsListView; import android.widget.ListView; -import androidx.annotation.ColorInt; -import androidx.annotation.ColorRes; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.annotation.Px; import androidx.annotation.VisibleForTesting; -import androidx.core.content.ContextCompat; import androidx.core.view.NestedScrollingChild; import androidx.core.view.NestedScrollingChildHelper; import androidx.core.view.NestedScrollingParent; @@ -111,7 +108,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent android.R.attr.enabled }; - CircleImageView mCircleView; private int mCircleViewIndex = -1; protected int mFrom; @@ -124,27 +120,20 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent int mCustomSlingshotDistance; - CircularProgressDrawable mProgress; - private Animation mScaleAnimation; private Animation mScaleDownAnimation; - private Animation mAlphaStartAnimation; - - private Animation mAlphaMaxAnimation; - private Animation mScaleDownToStartAnimation; boolean mNotify; - private int mCircleDiameter; - // 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) { @@ -157,33 +146,30 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent @Override public void onAnimationEnd(Animation animation) { if (mRefreshing) { - // Make sure the progress view is fully visible - mProgress.setAlpha(MAX_ALPHA); - mProgress.start(); + mRefreshView.startAnimation(); if (mNotify) { if (mListener != null) { mListener.onRefresh(); } } - mCurrentTargetOffsetTop = mCircleView.getTop(); + mCurrentTargetOffsetTop = mRefreshView.getTop(); } else { reset(); } } }; + private int mPullDownHeight = 0; void reset() { - mCircleView.clearAnimation(); - mProgress.stop(); - mCircleView.setVisibility(View.GONE); - setColorViewAlpha(MAX_ALPHA); + mRefreshView.stopAnimation(); + mRefreshView.setVisibility(View.GONE); // Return the circle to its start position if (mScale) { setAnimationProgress(0 /* animation complete and view is hidden */); } else { setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCurrentTargetOffsetTop); } - mCurrentTargetOffsetTop = mCircleView.getTop(); + mCurrentTargetOffsetTop = mRefreshView.getTop(); } @Override @@ -200,11 +186,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent reset(); } - private void setColorViewAlpha(int targetAlpha) { - mCircleView.getBackground().setAlpha(targetAlpha); - mProgress.setAlpha(targetAlpha); - } - /** * The refresh indicator starting and resting position is always positioned * near the top of the refreshing content. This position is a consistent @@ -265,7 +246,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent public void setProgressViewEndTarget(boolean scale, int end) { mSpinnerOffsetEnd = end; mScale = scale; - mCircleView.invalidate(); + mRefreshView.invalidate(); } /** @@ -279,27 +260,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent mCustomSlingshotDistance = slingshotDistance; } - /** - * One of DEFAULT, or LARGE. - */ - public void setSize(int size) { - if (size != CircularProgressDrawable.LARGE && size != CircularProgressDrawable.DEFAULT) { - return; - } - final DisplayMetrics metrics = getResources().getDisplayMetrics(); - if (size == CircularProgressDrawable.LARGE) { - mCircleDiameter = (int) (CIRCLE_DIAMETER_LARGE * metrics.density); - } else { - mCircleDiameter = (int) (CIRCLE_DIAMETER * metrics.density); - } - // force the bounds of the progress circle inside the circle view to - // update by setting it to null before updating its size and then - // re-setting it - mCircleView.setImageDrawable(null); - mProgress.setStyle(size); - mCircleView.setImageDrawable(mProgress); - } - /** * Simple constructor to use when creating a SwipeRefreshLayout from code. * @@ -327,7 +287,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent mDecelerateInterpolator = new DecelerateInterpolator(DECELERATE_INTERPOLATION_FACTOR); final DisplayMetrics metrics = getResources().getDisplayMetrics(); - mCircleDiameter = (int) (CIRCLE_DIAMETER * metrics.density); createProgressView(); setChildrenDrawingOrderEnabled(true); @@ -339,7 +298,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent mNestedScrollingChildHelper = new NestedScrollingChildHelper(this); setNestedScrollingEnabled(true); - mOriginalOffsetTop = mCurrentTargetOffsetTop = -mCircleDiameter; moveToStart(1.0f); final TypedArray a = context.obtainStyledAttributes(attrs, LAYOUT_ATTRS); @@ -347,6 +305,11 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent a.recycle(); } + public void setPullDownHeight(int height) { + mPullDownHeight = height; + mOriginalOffsetTop = mCurrentTargetOffsetTop = -height; + } + @Override protected int getChildDrawingOrder(int childCount, int i) { if (mCircleViewIndex < 0) { @@ -364,12 +327,13 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } private void createProgressView() { - mCircleView = new CircleImageView(getContext(), CIRCLE_BG_LIGHT); - mProgress = new CircularProgressDrawable(getContext()); - mProgress.setStyle(CircularProgressDrawable.DEFAULT); - mCircleView.setImageDrawable(mProgress); - mCircleView.setVisibility(View.GONE); - addView(mCircleView); + mRefreshView = new DoricRefreshView(getContext()); + ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(LayoutParams.MATCH_PARENT, 0); + addView(mRefreshView, layoutParams); + } + + public DoricRefreshView getRefreshView() { + return mRefreshView; } /** @@ -405,8 +369,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } private void startScaleUpAnimation(AnimationListener listener) { - mCircleView.setVisibility(View.VISIBLE); - mProgress.setAlpha(MAX_ALPHA); + mRefreshView.setVisibility(View.VISIBLE); mScaleAnimation = new Animation() { @Override public void applyTransformation(float interpolatedTime, Transformation t) { @@ -415,10 +378,10 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent }; mScaleAnimation.setDuration(mMediumAnimationDuration); if (listener != null) { - mCircleView.setAnimationListener(listener); + mRefreshView.setAnimationListener(listener); } - mCircleView.clearAnimation(); - mCircleView.startAnimation(mScaleAnimation); + mRefreshView.clearAnimation(); + mRefreshView.startAnimation(mScaleAnimation); } /** @@ -427,8 +390,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent * @param progress */ void setAnimationProgress(float progress) { - mCircleView.setScaleX(progress); - mCircleView.setScaleY(progress); + mRefreshView.setProgressRotation(progress); } private void setRefreshing(boolean refreshing, final boolean notify) { @@ -452,95 +414,9 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } }; mScaleDownAnimation.setDuration(SCALE_DOWN_DURATION); - mCircleView.setAnimationListener(listener); - mCircleView.clearAnimation(); - mCircleView.startAnimation(mScaleDownAnimation); - } - - private void startProgressAlphaStartAnimation() { - mAlphaStartAnimation = startAlphaAnimation(mProgress.getAlpha(), STARTING_PROGRESS_ALPHA); - } - - private void startProgressAlphaMaxAnimation() { - mAlphaMaxAnimation = startAlphaAnimation(mProgress.getAlpha(), MAX_ALPHA); - } - - private Animation startAlphaAnimation(final int startingAlpha, final int endingAlpha) { - Animation alpha = new Animation() { - @Override - public void applyTransformation(float interpolatedTime, Transformation t) { - mProgress.setAlpha( - (int) (startingAlpha + ((endingAlpha - startingAlpha) * interpolatedTime))); - } - }; - alpha.setDuration(ALPHA_ANIMATION_DURATION); - // Clear out the previous animation listeners. - mCircleView.setAnimationListener(null); - mCircleView.clearAnimation(); - mCircleView.startAnimation(alpha); - return alpha; - } - - /** - * @deprecated Use {@link #setProgressBackgroundColorSchemeResource(int)} - */ - @Deprecated - public void setProgressBackgroundColor(int colorRes) { - setProgressBackgroundColorSchemeResource(colorRes); - } - - /** - * Set the background color of the progress spinner disc. - * - * @param colorRes Resource id of the color. - */ - public void setProgressBackgroundColorSchemeResource(@ColorRes int colorRes) { - setProgressBackgroundColorSchemeColor(ContextCompat.getColor(getContext(), colorRes)); - } - - /** - * Set the background color of the progress spinner disc. - * - * @param color - */ - public void setProgressBackgroundColorSchemeColor(@ColorInt int color) { - mCircleView.setBackgroundColor(color); - } - - /** - * @deprecated Use {@link #setColorSchemeResources(int...)} - */ - @Deprecated - public void setColorScheme(@ColorRes int... colors) { - setColorSchemeResources(colors); - } - - /** - * Set the color resources used in the progress animation from color resources. - * The first color will also be the color of the bar that grows in response - * to a user swipe gesture. - * - * @param colorResIds - */ - public void setColorSchemeResources(@ColorRes int... colorResIds) { - final Context context = getContext(); - int[] colorRes = new int[colorResIds.length]; - for (int i = 0; i < colorResIds.length; i++) { - colorRes[i] = ContextCompat.getColor(context, colorResIds[i]); - } - setColorSchemeColors(colorRes); - } - - /** - * Set the colors used in the progress animation. The first - * color will also be the color of the bar that grows in response to a user - * swipe gesture. - * - * @param colors - */ - public void setColorSchemeColors(@ColorInt int... colors) { - ensureTarget(); - mProgress.setColorSchemeColors(colors); + mRefreshView.setAnimationListener(listener); + mRefreshView.clearAnimation(); + mRefreshView.startAnimation(mScaleDownAnimation); } /** @@ -557,7 +433,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent if (mTarget == null) { for (int i = 0; i < getChildCount(); i++) { View child = getChildAt(i); - if (!child.equals(mCircleView)) { + if (!child.equals(mRefreshView)) { mTarget = child; break; } @@ -588,15 +464,15 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent return; } - int circleWidth = mCircleView.getMeasuredWidth(); - int circleHeight = mCircleView.getMeasuredHeight(); + int circleWidth = mRefreshView.getMeasuredWidth(); + int circleHeight = mRefreshView.getMeasuredHeight(); - mCircleView.layout((width / 2 - circleWidth / 2), mCurrentTargetOffsetTop, + 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() + mCircleView.getBottom(); + 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); @@ -615,28 +491,18 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent getMeasuredWidth() - getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec( getMeasuredHeight() - getPaddingTop() - getPaddingBottom(), MeasureSpec.EXACTLY)); - mCircleView.measure(MeasureSpec.makeMeasureSpec(mCircleDiameter, MeasureSpec.EXACTLY), - MeasureSpec.makeMeasureSpec(mCircleDiameter, MeasureSpec.EXACTLY)); + mRefreshView.measure(MeasureSpec.makeMeasureSpec(getMeasuredWidth() - getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY), + MeasureSpec.makeMeasureSpec(mPullDownHeight, MeasureSpec.EXACTLY)); mCircleViewIndex = -1; // Get the index of the circleview. for (int index = 0; index < getChildCount(); index++) { - if (getChildAt(index) == mCircleView) { + if (getChildAt(index) == mRefreshView) { mCircleViewIndex = index; break; } } } - /** - * Get the diameter of the progress circle that is displayed as part of the - * swipe to refresh layout. - * - * @return Diameter in pixels of the progress circle view. - */ - public int getProgressCircleDiameter() { - return mCircleDiameter; - } - /** * @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. @@ -680,7 +546,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent switch (action) { case MotionEvent.ACTION_DOWN: - setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCircleView.getTop()); + setTargetOffsetTopAndBottom(mOriginalOffsetTop - mRefreshView.getTop()); mActivePointerId = ev.getPointerId(0); mIsBeingDragged = false; @@ -773,7 +639,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent // 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) { - mCircleView.setVisibility(View.GONE); + mRefreshView.setVisibility(View.GONE); } // Now let our nested parent consume the leftovers @@ -889,11 +755,9 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } private void moveSpinner(float overscrollTop) { - mProgress.setArrowEnabled(true); float originalDragPercent = overscrollTop / mTotalDragDistance; float dragPercent = Math.min(1f, Math.abs(originalDragPercent)); - float adjustedPercent = (float) Math.max(dragPercent - .4, 0) * 5 / 3; float extraOS = Math.abs(overscrollTop) - mTotalDragDistance; float slingshotDist = mCustomSlingshotDistance > 0 ? mCustomSlingshotDistance @@ -908,35 +772,17 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent int targetY = mOriginalOffsetTop + (int) ((slingshotDist * dragPercent) + extraMove); // where 1.0f is a full circle - if (mCircleView.getVisibility() != View.VISIBLE) { - mCircleView.setVisibility(View.VISIBLE); + if (mRefreshView.getVisibility() != View.VISIBLE) { + mRefreshView.setVisibility(View.VISIBLE); } if (!mScale) { - mCircleView.setScaleX(1f); - mCircleView.setScaleY(1f); + mRefreshView.setScaleX(1f); + mRefreshView.setScaleY(1f); } if (mScale) { setAnimationProgress(Math.min(1f, overscrollTop / mTotalDragDistance)); } - if (overscrollTop < mTotalDragDistance) { - if (mProgress.getAlpha() > STARTING_PROGRESS_ALPHA - && !isAnimationRunning(mAlphaStartAnimation)) { - // Animate the alpha - startProgressAlphaStartAnimation(); - } - } else { - if (mProgress.getAlpha() < MAX_ALPHA && !isAnimationRunning(mAlphaMaxAnimation)) { - // Animate the alpha - startProgressAlphaMaxAnimation(); - } - } - float strokeStart = adjustedPercent * .8f; - mProgress.setStartEndTrim(0f, Math.min(MAX_PROGRESS_ANGLE, strokeStart)); - mProgress.setArrowScale(Math.min(1f, adjustedPercent)); - - float rotation = (-0.25f + .4f * adjustedPercent + tensionPercent * 2) * .5f; - mProgress.setProgressRotation(rotation); setTargetOffsetTopAndBottom(targetY - mCurrentTargetOffsetTop); } @@ -946,7 +792,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } else { // cancel refresh mRefreshing = false; - mProgress.setStartEndTrim(0f, 0f); Animation.AnimationListener listener = null; if (!mScale) { listener = new Animation.AnimationListener() { @@ -969,7 +814,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent }; } animateOffsetToStartPosition(mCurrentTargetOffsetTop, listener); - mProgress.setArrowEnabled(false); } } @@ -1057,7 +901,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent if (yDiff > mTouchSlop && !mIsBeingDragged) { mInitialMotionY = mInitialDownY + mTouchSlop; mIsBeingDragged = true; - mProgress.setAlpha(STARTING_PROGRESS_ALPHA); } } @@ -1067,10 +910,10 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent mAnimateToCorrectPosition.setDuration(ANIMATE_TO_TRIGGER_DURATION); mAnimateToCorrectPosition.setInterpolator(mDecelerateInterpolator); if (listener != null) { - mCircleView.setAnimationListener(listener); + mRefreshView.setAnimationListener(listener); } - mCircleView.clearAnimation(); - mCircleView.startAnimation(mAnimateToCorrectPosition); + mRefreshView.clearAnimation(); + mRefreshView.startAnimation(mAnimateToCorrectPosition); } private void animateOffsetToStartPosition(int from, AnimationListener listener) { @@ -1083,10 +926,10 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent mAnimateToStartPosition.setDuration(ANIMATE_TO_START_DURATION); mAnimateToStartPosition.setInterpolator(mDecelerateInterpolator); if (listener != null) { - mCircleView.setAnimationListener(listener); + mRefreshView.setAnimationListener(listener); } - mCircleView.clearAnimation(); - mCircleView.startAnimation(mAnimateToStartPosition); + mRefreshView.clearAnimation(); + mRefreshView.startAnimation(mAnimateToStartPosition); } } @@ -1101,16 +944,15 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent endTarget = mSpinnerOffsetEnd; } targetTop = (mFrom + (int) ((endTarget - mFrom) * interpolatedTime)); - int offset = targetTop - mCircleView.getTop(); + int offset = targetTop - mRefreshView.getTop(); setTargetOffsetTopAndBottom(offset); - mProgress.setArrowScale(1 - interpolatedTime); } }; void moveToStart(float interpolatedTime) { int targetTop = 0; targetTop = (mFrom + (int) ((mOriginalOffsetTop - mFrom) * interpolatedTime)); - int offset = targetTop - mCircleView.getTop(); + int offset = targetTop - mRefreshView.getTop(); setTargetOffsetTopAndBottom(offset); } @@ -1124,7 +966,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent private void startScaleDownReturnToStartAnimation(int from, Animation.AnimationListener listener) { mFrom = from; - mStartingScale = mCircleView.getScaleX(); + mStartingScale = mRefreshView.getScaleX(); mScaleDownToStartAnimation = new Animation() { @Override public void applyTransformation(float interpolatedTime, Transformation t) { @@ -1135,16 +977,16 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent }; mScaleDownToStartAnimation.setDuration(SCALE_DOWN_DURATION); if (listener != null) { - mCircleView.setAnimationListener(listener); + mRefreshView.setAnimationListener(listener); } - mCircleView.clearAnimation(); - mCircleView.startAnimation(mScaleDownToStartAnimation); + mRefreshView.clearAnimation(); + mRefreshView.startAnimation(mScaleDownToStartAnimation); } void setTargetOffsetTopAndBottom(int offset) { - mCircleView.bringToFront(); - ViewCompat.offsetTopAndBottom(mCircleView, offset); - mCurrentTargetOffsetTop = mCircleView.getTop(); + mRefreshView.bringToFront(); + ViewCompat.offsetTopAndBottom(mRefreshView, offset); + mCurrentTargetOffsetTop = mRefreshView.getTop(); } private void onSecondaryPointerUp(MotionEvent ev) { From 055699d2219a6731b6001c459d00af8ca97f0cec Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Mon, 25 Nov 2019 20:35:11 +0800 Subject: [PATCH 04/35] feat:SwipeLayout --- .../java/pub/doric/demo/MainActivity.java | 2 +- .../pub/doric/pullable/CircleImageView.java | 159 ------------------ .../pub/doric/pullable/DoricSwipeLayout.java | 144 ++++++---------- 3 files changed, 51 insertions(+), 254 deletions(-) delete mode 100644 Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java 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 7d3ece4f..eb9886f9 100644 --- a/Android/app/src/main/java/pub/doric/demo/MainActivity.java +++ b/Android/app/src/main/java/pub/doric/demo/MainActivity.java @@ -54,7 +54,7 @@ public class MainActivity extends AppCompatActivity { }); swipeLayout.setBackgroundColor(Color.YELLOW); swipeLayout.getRefreshView().setBackgroundColor(Color.RED); - swipeLayout.setPullDownHeight(100); + swipeLayout.setPullDownHeight(300); RecyclerView recyclerView = findViewById(R.id.root); recyclerView.setBackgroundColor(Color.WHITE); recyclerView.setLayoutManager(new LinearLayoutManager(this)); diff --git a/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java b/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java deleted file mode 100644 index b003051b..00000000 --- a/Android/doric/src/main/java/pub/doric/pullable/CircleImageView.java +++ /dev/null @@ -1,159 +0,0 @@ -/* - * Copyright 2018 The Android Open Source Project - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package pub.doric.pullable; - -import android.content.Context; -import android.graphics.Canvas; -import android.graphics.Color; -import android.graphics.Paint; -import android.graphics.RadialGradient; -import android.graphics.Shader; -import android.graphics.drawable.ShapeDrawable; -import android.graphics.drawable.shapes.OvalShape; -import android.view.View; -import android.view.animation.Animation; - -import androidx.appcompat.widget.AppCompatImageView; -import androidx.core.content.ContextCompat; -import androidx.core.view.ViewCompat; - -/** - * Private class created to work around issues with AnimationListeners being - * called before the animation is actually complete and support shadows on older - * platforms. - */ -class CircleImageView extends AppCompatImageView { - - private static final int KEY_SHADOW_COLOR = 0x1E000000; - private static final int FILL_SHADOW_COLOR = 0x3D000000; - // PX - private static final float X_OFFSET = 0f; - private static final float Y_OFFSET = 1.75f; - private static final float SHADOW_RADIUS = 3.5f; - private static final int SHADOW_ELEVATION = 4; - - private Animation.AnimationListener mListener; - int mShadowRadius; - - CircleImageView(Context context, int color) { - super(context); - final float density = getContext().getResources().getDisplayMetrics().density; - final int shadowYOffset = (int) (density * Y_OFFSET); - final int shadowXOffset = (int) (density * X_OFFSET); - - mShadowRadius = (int) (density * SHADOW_RADIUS); - - ShapeDrawable circle; - if (elevationSupported()) { - circle = new ShapeDrawable(new OvalShape()); - ViewCompat.setElevation(this, SHADOW_ELEVATION * density); - } else { - OvalShape oval = new OvalShadow(mShadowRadius); - circle = new ShapeDrawable(oval); - setLayerType(View.LAYER_TYPE_SOFTWARE, circle.getPaint()); - circle.getPaint().setShadowLayer(mShadowRadius, shadowXOffset, shadowYOffset, - KEY_SHADOW_COLOR); - final int padding = mShadowRadius; - // set padding so the inner image sits correctly within the shadow. - setPadding(padding, padding, padding, padding); - } - circle.getPaint().setColor(color); - ViewCompat.setBackground(this, circle); - } - - private boolean elevationSupported() { - return android.os.Build.VERSION.SDK_INT >= 21; - } - - @Override - protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { - super.onMeasure(widthMeasureSpec, heightMeasureSpec); - if (!elevationSupported()) { - setMeasuredDimension(getMeasuredWidth() + mShadowRadius * 2, getMeasuredHeight() - + mShadowRadius * 2); - } - } - - public void setAnimationListener(Animation.AnimationListener listener) { - mListener = listener; - } - - @Override - public void onAnimationStart() { - super.onAnimationStart(); - if (mListener != null) { - mListener.onAnimationStart(getAnimation()); - } - } - - @Override - public void onAnimationEnd() { - super.onAnimationEnd(); - if (mListener != null) { - mListener.onAnimationEnd(getAnimation()); - } - } - - /** - * Update the background color of the circle image view. - * - * @param colorRes Id of a color resource. - */ - public void setBackgroundColorRes(int colorRes) { - setBackgroundColor(ContextCompat.getColor(getContext(), colorRes)); - } - - @Override - public void setBackgroundColor(int color) { - if (getBackground() instanceof ShapeDrawable) { - ((ShapeDrawable) getBackground()).getPaint().setColor(color); - } - } - - private class OvalShadow extends OvalShape { - private RadialGradient mRadialGradient; - private Paint mShadowPaint; - - OvalShadow(int shadowRadius) { - super(); - mShadowPaint = new Paint(); - mShadowRadius = shadowRadius; - updateRadialGradient((int) rect().width()); - } - - @Override - protected void onResize(float width, float height) { - super.onResize(width, height); - updateRadialGradient((int) width); - } - - @Override - public void draw(Canvas canvas, Paint paint) { - final int viewWidth = CircleImageView.this.getWidth(); - final int viewHeight = CircleImageView.this.getHeight(); - canvas.drawCircle(viewWidth / 2, viewHeight / 2, viewWidth / 2, mShadowPaint); - canvas.drawCircle(viewWidth / 2, viewHeight / 2, viewWidth / 2 - mShadowRadius, paint); - } - - private void updateRadialGradient(int diameter) { - mRadialGradient = new RadialGradient(diameter / 2, diameter / 2, - mShadowRadius, new int[]{FILL_SHADOW_COLOR, Color.TRANSPARENT}, - null, Shader.TileMode.CLAMP); - mShadowPaint.setShader(mRadialGradient); - } - } -} diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java index f83c3431..b0dfd6be 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -1,5 +1,7 @@ package pub.doric.pullable; +import android.animation.Animator; +import android.animation.ValueAnimator; import android.content.Context; import android.content.res.TypedArray; import android.util.AttributeSet; @@ -17,7 +19,6 @@ import android.widget.ListView; import androidx.annotation.NonNull; import androidx.annotation.Nullable; -import androidx.annotation.Px; import androidx.annotation.VisibleForTesting; import androidx.core.view.NestedScrollingChild; import androidx.core.view.NestedScrollingChildHelper; @@ -69,8 +70,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent private static final int ANIMATE_TO_START_DURATION = 200; - // Default background for the progress spinner - private static final int CIRCLE_BG_LIGHT = 0xFFFAFAFA; // 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; @@ -159,17 +158,55 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } }; private int mPullDownHeight = 0; + private ValueAnimator headerViewAnimator; void reset() { - mRefreshView.stopAnimation(); - mRefreshView.setVisibility(View.GONE); - // Return the circle to its start position - if (mScale) { - setAnimationProgress(0 /* animation complete and view is hidden */); - } else { - setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCurrentTargetOffsetTop); + if (headerViewAnimator != null && headerViewAnimator.isRunning()) { + headerViewAnimator.cancel(); } - mCurrentTargetOffsetTop = mRefreshView.getTop(); + 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 + + if (mScale) { + setAnimationProgress(0 /* animation complete and view is hidden */); + } else { + 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 @@ -186,80 +223,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent reset(); } - /** - * The refresh indicator starting and resting position is always positioned - * near the top of the refreshing content. This position is a consistent - * location, but can be adjusted in either direction based on whether or not - * there is a toolbar or actionbar present. - *

- * Note: Calling this will reset the position of the refresh indicator to - * start. - *

- * - * @param scale Set to true if there is no view at a higher z-order than where the progress - * spinner is set to appear. Setting it to true will cause indicator to be scaled - * up rather than clipped. - * @param start The offset in pixels from the top of this view at which the - * progress spinner should appear. - * @param end The offset in pixels from the top of this view at which the - * progress spinner should come to rest after a successful swipe - * gesture. - */ - public void setProgressViewOffset(boolean scale, int start, int end) { - mScale = scale; - mOriginalOffsetTop = start; - mSpinnerOffsetEnd = end; - mUsingCustomStart = true; - reset(); - mRefreshing = false; - } - - /** - * @return The offset in pixels from the top of this view at which the progress spinner should - * appear. - */ - public int getProgressViewStartOffset() { - return mOriginalOffsetTop; - } - - /** - * @return The offset in pixels from the top of this view at which the progress spinner should - * come to rest after a successful swipe gesture. - */ - public int getProgressViewEndOffset() { - return mSpinnerOffsetEnd; - } - - /** - * The refresh indicator resting position is always positioned near the top - * of the refreshing content. This position is a consistent location, but - * can be adjusted in either direction based on whether or not there is a - * toolbar or actionbar present. - * - * @param scale Set to true if there is no view at a higher z-order than where the progress - * spinner is set to appear. Setting it to true will cause indicator to be scaled - * up rather than clipped. - * @param end The offset in pixels from the top of this view at which the - * progress spinner should come to rest after a successful swipe - * gesture. - */ - public void setProgressViewEndTarget(boolean scale, int end) { - mSpinnerOffsetEnd = end; - mScale = scale; - mRefreshView.invalidate(); - } - - /** - * Sets a custom slingshot distance. - * - * @param slingshotDistance The distance in pixels that the refresh indicator can be pulled - * beyond its resting position. Use - * {@link #DEFAULT_SLINGSHOT_DISTANCE} to reset to the default value. - */ - public void setSlingshotDistance(@Px int slingshotDistance) { - mCustomSlingshotDistance = slingshotDistance; - } - /** * Simple constructor to use when creating a SwipeRefreshLayout from code. * @@ -308,6 +271,8 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent public void setPullDownHeight(int height) { mPullDownHeight = height; mOriginalOffsetTop = mCurrentTargetOffsetTop = -height; + mSpinnerOffsetEnd = height; + mTotalDragDistance = height; } @Override @@ -441,15 +406,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } } - /** - * Set the distance to trigger a sync in dips - * - * @param distance - */ - public void setDistanceToTriggerSync(int distance) { - mTotalDragDistance = distance; - } - @Override protected void onLayout(boolean changed, int left, int top, int right, int bottom) { final int width = getMeasuredWidth(); From c766e57c830356c8ce47f1fc9d21d5f9390d46f8 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Mon, 25 Nov 2019 20:50:08 +0800 Subject: [PATCH 05/35] feat:optimize ProgressRotation --- .../pub/doric/pullable/DoricSwipeLayout.java | 90 ++++++------------- 1 file changed, 27 insertions(+), 63 deletions(-) diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java index b0dfd6be..ef845880 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -96,8 +96,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent private float mInitialDownY; private boolean mIsBeingDragged; private int mActivePointerId = INVALID_POINTER; - // Whether this item is scaled up rather than clipped - boolean mScale; // Target is returning to its start offset because it was cancelled or a // refresh was triggered. @@ -185,11 +183,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent mRefreshView.setVisibility(View.GONE); // Return the circle to its start position - if (mScale) { - setAnimationProgress(0 /* animation complete and view is hidden */); - } else { - setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCurrentTargetOffsetTop); - } + setTargetOffsetTopAndBottom(mOriginalOffsetTop - mCurrentTargetOffsetTop); mCurrentTargetOffsetTop = mRefreshView.getTop(); } @@ -355,7 +349,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent * @param progress */ void setAnimationProgress(float progress) { - mRefreshView.setProgressRotation(progress); + } private void setRefreshing(boolean refreshing, final boolean notify) { @@ -731,14 +725,9 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent if (mRefreshView.getVisibility() != View.VISIBLE) { mRefreshView.setVisibility(View.VISIBLE); } - if (!mScale) { - mRefreshView.setScaleX(1f); - mRefreshView.setScaleY(1f); - } + mRefreshView.setScaleX(1f); + mRefreshView.setScaleY(1f); - if (mScale) { - setAnimationProgress(Math.min(1f, overscrollTop / mTotalDragDistance)); - } setTargetOffsetTopAndBottom(targetY - mCurrentTargetOffsetTop); } @@ -749,26 +738,22 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent // cancel refresh mRefreshing = false; Animation.AnimationListener listener = null; - if (!mScale) { - listener = new Animation.AnimationListener() { + listener = new Animation.AnimationListener() { - @Override - public void onAnimationStart(Animation animation) { - } + @Override + public void onAnimationStart(Animation animation) { + } - @Override - public void onAnimationEnd(Animation animation) { - if (!mScale) { - startScaleDownAnimation(null); - } - } + @Override + public void onAnimationEnd(Animation animation) { + startScaleDownAnimation(null); + } - @Override - public void onAnimationRepeat(Animation animation) { - } + @Override + public void onAnimationRepeat(Animation animation) { + } - }; - } + }; animateOffsetToStartPosition(mCurrentTargetOffsetTop, listener); } } @@ -873,20 +858,15 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } private void animateOffsetToStartPosition(int from, AnimationListener listener) { - if (mScale) { - // Scale the item back down - startScaleDownReturnToStartAnimation(from, listener); - } else { - mFrom = from; - mAnimateToStartPosition.reset(); - mAnimateToStartPosition.setDuration(ANIMATE_TO_START_DURATION); - mAnimateToStartPosition.setInterpolator(mDecelerateInterpolator); - if (listener != null) { - mRefreshView.setAnimationListener(listener); - } - mRefreshView.clearAnimation(); - mRefreshView.startAnimation(mAnimateToStartPosition); + 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() { @@ -919,30 +899,14 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent } }; - private void startScaleDownReturnToStartAnimation(int from, - Animation.AnimationListener listener) { - mFrom = from; - mStartingScale = mRefreshView.getScaleX(); - mScaleDownToStartAnimation = new Animation() { - @Override - public void applyTransformation(float interpolatedTime, Transformation t) { - float targetScale = (mStartingScale + (-mStartingScale * interpolatedTime)); - setAnimationProgress(targetScale); - moveToStart(interpolatedTime); - } - }; - mScaleDownToStartAnimation.setDuration(SCALE_DOWN_DURATION); - if (listener != null) { - mRefreshView.setAnimationListener(listener); - } - mRefreshView.clearAnimation(); - mRefreshView.startAnimation(mScaleDownToStartAnimation); - } 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) { From 362ec833c9840bb02f99048a5911790cffe6e40b Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 10:31:17 +0800 Subject: [PATCH 06/35] feat:add RefreshableDemo for Android --- .../java/pub/doric/demo/MainActivity.java | 6 +- .../main/java/pub/doric/DoricRegistry.java | 2 + .../pub/doric/pullable/DoricSwipeLayout.java | 14 +- .../pub/doric/pullable/RefreshableNode.java | 165 ++++++++++++++++++ demo/index.ts | 1 + demo/src/RefreshableDemo.ts | 72 ++++++++ js-framework/index.ts | 1 + js-framework/src/ui/refreshable.ts | 60 +++++++ js-framework/src/ui/view.ts | 2 +- 9 files changed, 318 insertions(+), 5 deletions(-) create mode 100644 Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java create mode 100644 demo/src/RefreshableDemo.ts create mode 100644 js-framework/src/ui/refreshable.ts 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 eb9886f9..ad02e812 100644 --- a/Android/app/src/main/java/pub/doric/demo/MainActivity.java +++ b/Android/app/src/main/java/pub/doric/demo/MainActivity.java @@ -29,6 +29,8 @@ import androidx.appcompat.app.AppCompatActivity; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; +import org.w3c.dom.Text; + import java.io.IOException; import java.util.ArrayList; import java.util.List; @@ -54,7 +56,9 @@ public class MainActivity extends AppCompatActivity { }); swipeLayout.setBackgroundColor(Color.YELLOW); swipeLayout.getRefreshView().setBackgroundColor(Color.RED); - swipeLayout.setPullDownHeight(300); + 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)); diff --git a/Android/doric/src/main/java/pub/doric/DoricRegistry.java b/Android/doric/src/main/java/pub/doric/DoricRegistry.java index 12722dc4..c41c1b5a 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.pullable.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/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java index ef845880..a83ca478 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -287,7 +287,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent private void createProgressView() { mRefreshView = new DoricRefreshView(getContext()); - ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(LayoutParams.MATCH_PARENT, 0); + ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT); addView(mRefreshView, layoutParams); } @@ -441,8 +441,16 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent getMeasuredWidth() - getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY), MeasureSpec.makeMeasureSpec( getMeasuredHeight() - getPaddingTop() - getPaddingBottom(), MeasureSpec.EXACTLY)); - mRefreshView.measure(MeasureSpec.makeMeasureSpec(getMeasuredWidth() - getPaddingLeft() - getPaddingRight(), MeasureSpec.EXACTLY), - MeasureSpec.makeMeasureSpec(mPullDownHeight, 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++) { diff --git a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java b/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java new file mode 100644 index 00000000..37fbb31a --- /dev/null +++ b/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java @@ -0,0 +1,165 @@ +package pub.doric.pullable; + +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 { + + private String mContentViewId; + private ViewNode mContentNode; + + private String mHeaderViewId; + private ViewNode mHeaderNode; + + public RefreshableNode(DoricContext doricContext) { + super(doricContext); + } + + + @Override + protected DoricSwipeLayout build() { + return new DoricSwipeLayout(getContext()); + } + + @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 { + 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())); + } +} 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/RefreshableDemo.ts b/demo/src/RefreshableDemo.ts new file mode 100644 index 00000000..24d83750 --- /dev/null +++ b/demo/src/RefreshableDemo.ts @@ -0,0 +1,72 @@ +import { refreshable, Group, Panel, navbar, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, scroller, layoutConfig, image, IView, IVLayout, ScaleType, modal, IText, network, navigator } from "doric"; +import { title, label, colors } from "./utils"; + +@Entry +class RefreshableDemo extends Panel { + build(rootView: Group): void { + let refreshView = refreshable({ + layoutConfig: layoutConfig().atmost(), + header: text({ + text: "This is Header", + width: 100, + height: 100, + layoutConfig: layoutConfig().exactly(), + }), + 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), + ]).apply({ + layoutConfig: layoutConfig().atmost().h(LayoutSpec.WRAP_CONTENT), + gravity: gravity().center(), + space: 10, + } as IVLayout)).apply({ + layoutConfig: layoutConfig().atmost(), + }) + }).in(rootView) + } +} \ No newline at end of file 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/refreshable.ts b/js-framework/src/ui/refreshable.ts new file mode 100644 index 00000000..f1ca4704 --- /dev/null +++ b/js-framework/src/ui/refreshable.ts @@ -0,0 +1,60 @@ +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 +} diff --git a/js-framework/src/ui/view.ts b/js-framework/src/ui/view.ts index 893bf4dc..baa1595f 100644 --- a/js-framework/src/ui/view.ts +++ b/js-framework/src/ui/view.ts @@ -266,7 +266,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) { From 1f0a1a5e0ad3b5212cb642d685c9a91ac6af1259 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 10:36:23 +0800 Subject: [PATCH 07/35] feat:setEnable setRefreshing to false --- .../src/main/java/pub/doric/pullable/DoricSwipeLayout.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java index a83ca478..4ac43197 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java @@ -159,6 +159,7 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent private ValueAnimator headerViewAnimator; void reset() { + mRefreshing = false; if (headerViewAnimator != null && headerViewAnimator.isRunning()) { headerViewAnimator.cancel(); } @@ -199,8 +200,6 @@ public class DoricSwipeLayout extends ViewGroup implements NestedScrollingParent }); headerViewAnimator.setDuration(SCALE_DOWN_DURATION); headerViewAnimator.start(); - - } @Override From a5b07949bafa121c16e627ae4f0afc74045d8550 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 11:33:08 +0800 Subject: [PATCH 08/35] feat:fix view has no superview --- .../pub/doric/pullable/PullingListener.java | 20 +++++++++++++++++++ js-framework/src/ui/view.ts | 2 +- 2 files changed, 21 insertions(+), 1 deletion(-) create mode 100644 Android/doric/src/main/java/pub/doric/pullable/PullingListener.java diff --git a/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java b/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java new file mode 100644 index 00000000..71df7b07 --- /dev/null +++ b/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java @@ -0,0 +1,20 @@ +package pub.doric.pullable; + +/** + * @Description: pub.doric.pullable + * @Author: pengfei.zhou + * @CreateDate: 2019-11-25 + */ +public interface IPullable { + + 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/js-framework/src/ui/view.ts b/js-framework/src/ui/view.ts index baa1595f..70f942b0 100644 --- a/js-framework/src/ui/view.ts +++ b/js-framework/src/ui/view.ts @@ -316,6 +316,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 +342,6 @@ export abstract class Group extends Superview { } addChild(view: View) { - view.superview = this this.children.push(view) } } From c90547a1f044adb9ec83f117f818128b74c87d97 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 11:33:28 +0800 Subject: [PATCH 09/35] feat:update tsconfig incase uses dom lib --- demo/tsconfig.json | 2 +- js-framework/tsconfig.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) 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/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'. */ From 9a9482eeb5c3febbac152633b791524bd3a13fd2 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 11:34:02 +0800 Subject: [PATCH 10/35] feat:add RefreshableDemo --- .../pub/doric/pullable/DoricRefreshView.java | 31 +++++++++---------- .../java/pub/doric/pullable/IPullable.java | 27 ---------------- .../pub/doric/pullable/PullingListener.java | 2 +- .../pub/doric/pullable/RefreshableNode.java | 29 +++++++++++++++-- .../main/java/pub/doric/shader/ViewNode.java | 12 +++++++ demo/src/RefreshableDemo.ts | 12 +++++++ demo/src/utils.ts | 2 ++ js-framework/src/ui/refreshable.ts | 11 +++++++ 8 files changed, 80 insertions(+), 46 deletions(-) delete mode 100644 Android/doric/src/main/java/pub/doric/pullable/IPullable.java diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java b/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java index 2d404750..6ab3633e 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java +++ b/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java @@ -2,6 +2,7 @@ package pub.doric.pullable; import android.content.Context; import android.util.AttributeSet; +import android.util.Log; import android.view.Gravity; import android.view.View; import android.view.ViewGroup; @@ -17,10 +18,12 @@ import androidx.annotation.Nullable; * @Author: pengfei.zhou * @CreateDate: 2019-11-25 */ -public class DoricRefreshView extends FrameLayout implements IPullable { +public class DoricRefreshView extends FrameLayout implements PullingListener { private View content; private Animation.AnimationListener mListener; + private PullingListener mPullingListenr; + public DoricRefreshView(@NonNull Context context) { super(context); } @@ -50,33 +53,29 @@ public class DoricRefreshView extends FrameLayout implements IPullable { return content; } + + public void setPullingListenr(PullingListener listenr) { + this.mPullingListenr = listenr; + } + @Override public void startAnimation() { - if (content != null && content instanceof IPullable) { - ((IPullable) content).startAnimation(); + if (mPullingListenr != null) { + mPullingListenr.startAnimation(); } } @Override public void stopAnimation() { - if (content != null && content instanceof IPullable) { - ((IPullable) content).stopAnimation(); - } - } - - @Override - public int successAnimation() { - if (content != null && content instanceof IPullable) { - return ((IPullable) content).successAnimation(); - } else { - return 0; + if (mPullingListenr != null) { + mPullingListenr.stopAnimation(); } } @Override public void setProgressRotation(float rotation) { - if (content != null && content instanceof IPullable) { - ((IPullable) content).setProgressRotation(rotation); + if (mPullingListenr != null) { + mPullingListenr.setProgressRotation(rotation); } } diff --git a/Android/doric/src/main/java/pub/doric/pullable/IPullable.java b/Android/doric/src/main/java/pub/doric/pullable/IPullable.java deleted file mode 100644 index 83bb96af..00000000 --- a/Android/doric/src/main/java/pub/doric/pullable/IPullable.java +++ /dev/null @@ -1,27 +0,0 @@ -package pub.doric.pullable; - -/** - * @Description: pub.doric.pullable - * @Author: pengfei.zhou - * @CreateDate: 2019-11-25 - */ -public interface IPullable { - - void startAnimation(); - - void stopAnimation(); - - /** - * run the animation after pull request success and before stop animation - * - * @return the duration of success animation or 0 if no success animation - */ - int successAnimation(); - - /** - * Set the amount of rotation to apply to the progress spinner. - * - * @param rotation Rotation is from [0..1] - */ - void setProgressRotation(float rotation); -} diff --git a/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java b/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java index 71df7b07..6c2c3257 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java +++ b/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java @@ -5,7 +5,7 @@ package pub.doric.pullable; * @Author: pengfei.zhou * @CreateDate: 2019-11-25 */ -public interface IPullable { +public interface PullingListener { void startAnimation(); diff --git a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java b/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java index 37fbb31a..329f8207 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java +++ b/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java @@ -1,5 +1,7 @@ package pub.doric.pullable; +import android.view.animation.Animation; + import com.github.pengfeizhou.jscore.JSObject; import com.github.pengfeizhou.jscore.JSValue; import com.github.pengfeizhou.jscore.JavaValue; @@ -17,7 +19,7 @@ import pub.doric.shader.ViewNode; * @CreateDate: 2019-11-26 */ @DoricPlugin(name = "Refreshable") -public class RefreshableNode extends SuperNode { +public class RefreshableNode extends SuperNode implements PullingListener { private String mContentViewId; private ViewNode mContentNode; @@ -32,7 +34,9 @@ public class RefreshableNode extends SuperNode { @Override protected DoricSwipeLayout build() { - return new DoricSwipeLayout(getContext()); + DoricSwipeLayout doricSwipeLayout = new DoricSwipeLayout(getContext()); + doricSwipeLayout.getRefreshView().setPullingListenr(this); + return doricSwipeLayout; } @Override @@ -162,4 +166,25 @@ public class RefreshableNode extends SuperNode { 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/ViewNode.java b/Android/doric/src/main/java/pub/doric/shader/ViewNode.java index ee394eab..3a98a74f 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,16 @@ 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); + } } diff --git a/demo/src/RefreshableDemo.ts b/demo/src/RefreshableDemo.ts index 24d83750..a3b457ca 100644 --- a/demo/src/RefreshableDemo.ts +++ b/demo/src/RefreshableDemo.ts @@ -60,6 +60,18 @@ class RefreshableDemo extends Panel { 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(), diff --git a/demo/src/utils.ts b/demo/src/utils.ts index a024952e..18debee8 100644 --- a/demo/src/utils.ts +++ b/demo/src/utils.ts @@ -1,5 +1,7 @@ import { Color, text, Stack, Text, layoutConfig, LayoutSpec, gravity } from "doric"; +export const icon_refresh = '' + export const colors = [ "#70a1ff", "#7bed9f", diff --git a/js-framework/src/ui/refreshable.ts b/js-framework/src/ui/refreshable.ts index f1ca4704..1a73cb93 100644 --- a/js-framework/src/ui/refreshable.ts +++ b/js-framework/src/ui/refreshable.ts @@ -3,6 +3,7 @@ import { List } from "./list"; import { Scroller } from "./scroller"; import { BridgeContext } from "../runtime/global"; import { layoutConfig } from "./declarative"; +import { Image } from "./widgets"; export interface IRefreshable extends IView { content: List | Scroller @@ -58,3 +59,13 @@ export function refreshable(config: IRefreshable) { } return ret } + +export interface IPullable { + startAnimation(): void + stopAnimation(): void + setProgressRotation(rotation: number): void +} + +export class PullableView extends Image { + +} \ No newline at end of file From 208c635b8b14687e956fbb0f9b373b6baf5466f4 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 13:32:07 +0800 Subject: [PATCH 11/35] feat:add InjectEmpty to ignore too much logout --- .../doric/src/main/java/pub/doric/engine/DoricJSEngine.java | 6 ++++++ .../doric/src/main/java/pub/doric/utils/DoricConstant.java | 1 + iOS/Pod/Classes/Engine/DoricJSEngine.m | 4 +++- iOS/Pod/Classes/Util/DoricConstant.h | 1 + iOS/Pod/Classes/Util/DoricConstant.m | 1 + js-framework/src/ui/panel.ts | 4 +++- 6 files changed, 15 insertions(+), 2 deletions(-) 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/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/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/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/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) From 3dfa6c9770a15a44dc2d1ce86dc1a6b56ee0e3b4 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 13:33:03 +0800 Subject: [PATCH 12/35] feat:add Pullable and view's setRotation function --- .../main/java/pub/doric/shader/ViewNode.java | 5 ++++ .../main/java/pub/doric/utils/DoricUtils.java | 2 ++ demo/src/RefreshableDemo.ts | 28 ++++++++++++++----- js-framework/src/ui/refreshable.ts | 6 +++- js-framework/src/ui/view.ts | 24 ++++++++++++++++ 5 files changed, 57 insertions(+), 8 deletions(-) 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 3a98a74f..b7df0e5c 100644 --- a/Android/doric/src/main/java/pub/doric/shader/ViewNode.java +++ b/Android/doric/src/main/java/pub/doric/shader/ViewNode.java @@ -269,4 +269,9 @@ public abstract class ViewNode extends DoricContextHolder { } doricLayer.setRotation(rotation * 360); } + + @DoricMethod + public float getRotation() { + return doricLayer.getRotation() / 360; + } } 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/src/RefreshableDemo.ts b/demo/src/RefreshableDemo.ts index a3b457ca..280ed5ef 100644 --- a/demo/src/RefreshableDemo.ts +++ b/demo/src/RefreshableDemo.ts @@ -1,16 +1,30 @@ -import { refreshable, Group, Panel, navbar, text, gravity, Color, Stack, LayoutSpec, list, NativeCall, listItem, log, vlayout, Gravity, hlayout, Text, scroller, layoutConfig, image, IView, IVLayout, ScaleType, modal, IText, network, navigator } from "doric"; -import { title, label, colors } from "./utils"; +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(), - header: text({ - text: "This is Header", - width: 100, - height: 100, - layoutConfig: layoutConfig().exactly(), + 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"), diff --git a/js-framework/src/ui/refreshable.ts b/js-framework/src/ui/refreshable.ts index 1a73cb93..35de57d7 100644 --- a/js-framework/src/ui/refreshable.ts +++ b/js-framework/src/ui/refreshable.ts @@ -66,6 +66,10 @@ export interface IPullable { setProgressRotation(rotation: number): void } -export class PullableView extends Image { +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 70f942b0..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, @@ -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 { From 84b9ec0f22351ed70e523fb64c783ca5d8aae426 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 13:39:04 +0800 Subject: [PATCH 13/35] feat:add OnRefresh Callback for Android --- .../main/java/pub/doric/pullable/RefreshableNode.java | 10 ++++++++-- demo/src/RefreshableDemo.ts | 6 ++++++ 2 files changed, 14 insertions(+), 2 deletions(-) diff --git a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java b/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java index 329f8207..3e87de2a 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java +++ b/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java @@ -1,7 +1,5 @@ package pub.doric.pullable; -import android.view.animation.Animation; - import com.github.pengfeizhou.jscore.JSObject; import com.github.pengfeizhou.jscore.JSValue; import com.github.pengfeizhou.jscore.JavaValue; @@ -45,6 +43,14 @@ public class RefreshableNode extends SuperNode implements Pull 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); } diff --git a/demo/src/RefreshableDemo.ts b/demo/src/RefreshableDemo.ts index 280ed5ef..743da1e6 100644 --- a/demo/src/RefreshableDemo.ts +++ b/demo/src/RefreshableDemo.ts @@ -7,6 +7,12 @@ class RefreshableDemo extends Panel { let refreshImage: Image let refreshView = refreshable({ layoutConfig: layoutConfig().atmost(), + onRefresh: () => { + log('onRefresh') + setTimeout(() => { + refreshView.setRefreshing(context, false) + }, 5000) + }, header: pullable(context, stack([ image({ From 0b1174c3fd4bd8315467da307e9d0ecb7ef11987 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 14:06:44 +0800 Subject: [PATCH 14/35] feat:Refreshable +list --- demo/src/ListDemo.ts | 156 +++++++++++++++++++----------------- demo/src/utils.ts | 26 +++++- js-framework/src/ui/list.ts | 4 + 3 files changed, 112 insertions(+), 74 deletions(-) diff --git a/demo/src/ListDemo.ts b/demo/src/ListDemo.ts index a0ccb61a..b3de5c46 100644 --- a/demo/src/ListDemo.ts +++ b/demo/src/ListDemo.ts @@ -1,15 +1,11 @@ -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"; +import { isObject } from "util"; @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 +19,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, diff --git a/demo/src/utils.ts b/demo/src/utils.ts index 18debee8..8bb513b6 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 } from "doric"; +import { log } from "util"; export const icon_refresh = '' @@ -47,4 +48,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/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) { From 08e42bacdd5dce224921ad1b953f77ec6b7cb8b7 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 15:51:39 +0800 Subject: [PATCH 15/35] feat:change moment of setting scrollview's contentsize --- iOS/Pod/Classes/Shader/DoricScrollerNode.m | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index cc951f12..c492e560 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -29,10 +29,6 @@ @implementation DoricScrollView - (void)layoutSubviews { [super layoutSubviews]; - if (self.subviews.count > 0) { - UIView *child = self.subviews[0]; - [self setContentSize:child.frame.size]; - } } - (CGSize)sizeThatFits:(CGSize)size { @@ -91,6 +87,13 @@ - (void)blend:(NSDictionary *)props { [self.view addSubview:it.view]; }]; } + [self.view also:^(UIScrollView *it) { + if (it.subviews.count > 0) { + UIView *child = it.subviews[0]; + CGSize size = [child sizeThatFits:it.frame.size]; + [it setContentSize:size]; + } + }]; } - (void)blendView:(UIScrollView *)view forPropName:(NSString *)name propValue:(id)prop { From 74cee4f91b055453e51872ab8b5ae863f27698e2 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 15:57:48 +0800 Subject: [PATCH 16/35] feat:rename package name --- Android/app/src/main/java/pub/doric/demo/MainActivity.java | 4 +--- .../app/src/main/java/pub/doric/demo/PullableActivity.java | 2 +- Android/app/src/main/res/layout/activity_main.xml | 4 ++-- Android/app/src/main/res/layout/activity_pullable.xml | 2 +- Android/doric/src/main/java/pub/doric/DoricRegistry.java | 2 +- .../pub/doric/{pullable => refresh}/DoricRefreshView.java | 3 +-- .../pub/doric/{pullable => refresh}/DoricSwipeLayout.java | 2 +- .../java/pub/doric/{pullable => refresh}/PullingListener.java | 2 +- .../java/pub/doric/{pullable => refresh}/RefreshableNode.java | 2 +- 9 files changed, 10 insertions(+), 13 deletions(-) rename Android/doric/src/main/java/pub/doric/{pullable => refresh}/DoricRefreshView.java (98%) rename Android/doric/src/main/java/pub/doric/{pullable => refresh}/DoricSwipeLayout.java (99%) rename Android/doric/src/main/java/pub/doric/{pullable => refresh}/PullingListener.java (93%) rename Android/doric/src/main/java/pub/doric/{pullable => refresh}/RefreshableNode.java (99%) 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 ad02e812..b9153c44 100644 --- a/Android/app/src/main/java/pub/doric/demo/MainActivity.java +++ b/Android/app/src/main/java/pub/doric/demo/MainActivity.java @@ -29,15 +29,13 @@ import androidx.appcompat.app.AppCompatActivity; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; -import org.w3c.dom.Text; - import java.io.IOException; import java.util.ArrayList; import java.util.List; import pub.doric.DoricActivity; import pub.doric.devkit.ui.DemoDebugActivity; -import pub.doric.pullable.DoricSwipeLayout; +import pub.doric.refresh.DoricSwipeLayout; import pub.doric.utils.DoricUtils; public class MainActivity extends AppCompatActivity { diff --git a/Android/app/src/main/java/pub/doric/demo/PullableActivity.java b/Android/app/src/main/java/pub/doric/demo/PullableActivity.java index 615f27c5..2ce73a2a 100644 --- a/Android/app/src/main/java/pub/doric/demo/PullableActivity.java +++ b/Android/app/src/main/java/pub/doric/demo/PullableActivity.java @@ -6,7 +6,7 @@ import android.graphics.Color; import android.os.Bundle; import android.widget.FrameLayout; -import pub.doric.pullable.DoricSwipeLayout; +import pub.doric.refresh.DoricSwipeLayout; public class PullableActivity extends AppCompatActivity { diff --git a/Android/app/src/main/res/layout/activity_main.xml b/Android/app/src/main/res/layout/activity_main.xml index 95619501..8a4718b2 100644 --- a/Android/app/src/main/res/layout/activity_main.xml +++ b/Android/app/src/main/res/layout/activity_main.xml @@ -1,5 +1,5 @@ - - \ No newline at end of file + \ 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 index e9bad365..c6923232 100644 --- a/Android/app/src/main/res/layout/activity_pullable.xml +++ b/Android/app/src/main/res/layout/activity_pullable.xml @@ -6,7 +6,7 @@ android:layout_height="match_parent" tools:context=".PullableActivity"> - diff --git a/Android/doric/src/main/java/pub/doric/DoricRegistry.java b/Android/doric/src/main/java/pub/doric/DoricRegistry.java index c41c1b5a..9919509f 100644 --- a/Android/doric/src/main/java/pub/doric/DoricRegistry.java +++ b/Android/doric/src/main/java/pub/doric/DoricRegistry.java @@ -25,7 +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.pullable.RefreshableNode; +import pub.doric.refresh.RefreshableNode; import pub.doric.shader.HLayoutNode; import pub.doric.shader.ImageNode; import pub.doric.shader.ScrollerNode; diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java b/Android/doric/src/main/java/pub/doric/refresh/DoricRefreshView.java similarity index 98% rename from Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java rename to Android/doric/src/main/java/pub/doric/refresh/DoricRefreshView.java index 6ab3633e..8b1ded39 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricRefreshView.java +++ b/Android/doric/src/main/java/pub/doric/refresh/DoricRefreshView.java @@ -1,8 +1,7 @@ -package pub.doric.pullable; +package pub.doric.refresh; import android.content.Context; import android.util.AttributeSet; -import android.util.Log; import android.view.Gravity; import android.view.View; import android.view.ViewGroup; diff --git a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java b/Android/doric/src/main/java/pub/doric/refresh/DoricSwipeLayout.java similarity index 99% rename from Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java rename to Android/doric/src/main/java/pub/doric/refresh/DoricSwipeLayout.java index 4ac43197..411f13ef 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/DoricSwipeLayout.java +++ b/Android/doric/src/main/java/pub/doric/refresh/DoricSwipeLayout.java @@ -1,4 +1,4 @@ -package pub.doric.pullable; +package pub.doric.refresh; import android.animation.Animator; import android.animation.ValueAnimator; diff --git a/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java b/Android/doric/src/main/java/pub/doric/refresh/PullingListener.java similarity index 93% rename from Android/doric/src/main/java/pub/doric/pullable/PullingListener.java rename to Android/doric/src/main/java/pub/doric/refresh/PullingListener.java index 6c2c3257..c09ab2c4 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/PullingListener.java +++ b/Android/doric/src/main/java/pub/doric/refresh/PullingListener.java @@ -1,4 +1,4 @@ -package pub.doric.pullable; +package pub.doric.refresh; /** * @Description: pub.doric.pullable diff --git a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java b/Android/doric/src/main/java/pub/doric/refresh/RefreshableNode.java similarity index 99% rename from Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java rename to Android/doric/src/main/java/pub/doric/refresh/RefreshableNode.java index 3e87de2a..c6f1d1ce 100644 --- a/Android/doric/src/main/java/pub/doric/pullable/RefreshableNode.java +++ b/Android/doric/src/main/java/pub/doric/refresh/RefreshableNode.java @@ -1,4 +1,4 @@ -package pub.doric.pullable; +package pub.doric.refresh; import com.github.pengfeizhou.jscore.JSObject; import com.github.pengfeizhou.jscore.JSValue; From deada07e8d028dbc1cb619c77a64329f38d21171 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 20:15:16 +0800 Subject: [PATCH 17/35] iOS SwipeLayout --- demo/src/ImageDemo.ts | 5 +- demo/src/ScrollerDemo.ts | 83 ++++-------- demo/src/image_base64.ts | 2 + iOS/Pod/Classes/DoricRegistry.m | 2 + .../Classes/Refresh/DoricRefreshableNode.h | 10 ++ .../Classes/Refresh/DoricRefreshableNode.m | 128 ++++++++++++++++++ .../Classes/Refresh/DoricSwipeRefreshLayout.h | 22 +++ .../Classes/Refresh/DoricSwipeRefreshLayout.m | 118 ++++++++++++++++ iOS/Pod/Classes/Shader/DoricLayouts.h | 4 + iOS/Pod/Classes/Shader/DoricLayouts.m | 41 +++++- iOS/Pod/Classes/Shader/DoricScrollerNode.h | 6 +- iOS/Pod/Classes/Shader/DoricScrollerNode.m | 32 +++-- 12 files changed, 375 insertions(+), 78 deletions(-) create mode 100644 demo/src/image_base64.ts create mode 100644 iOS/Pod/Classes/Refresh/DoricRefreshableNode.h create mode 100644 iOS/Pod/Classes/Refresh/DoricRefreshableNode.m create mode 100644 iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.h create mode 100644 iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m diff --git a/demo/src/ImageDemo.ts b/demo/src/ImageDemo.ts index 109c99bb..a26c4cee 100644 --- a/demo/src/ImageDemo.ts +++ b/demo/src/ImageDemo.ts @@ -1,7 +1,8 @@ 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 { 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 { @@ -82,7 +83,7 @@ class ImageDemo extends Panel { }), label('ImageBase64'), image({ - imageBase64, + imageBase64: img_base64, width: 300, height: 300, border: { 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/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/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..3406fcb1 --- /dev/null +++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m @@ -0,0 +1,128 @@ +// +// 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]; +} + +- (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"]]; +} +@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..bc6f28a0 --- /dev/null +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -0,0 +1,118 @@ +// +// Created by pengfei.zhou on 2019/11/26. +// + +#import "DoricSwipeRefreshLayout.h" +#import "UIView+Doric.h" +#import "DoricLayouts.h" +#import "Doric.h" + +@interface DoricSwipeRefreshLayout () + +@end + +@implementation DoricSwipeRefreshLayout + +- (instancetype)initWithFrame:(CGRect)frame { + if (self = [super initWithFrame:frame]) { + self.showsHorizontalScrollIndicator = NO; + self.showsVerticalScrollIndicator = NO; + self.delegate = self; + } + return self; +} + +- (instancetype)init { + if (self = [super init]) { + self.showsHorizontalScrollIndicator = NO; + self.showsVerticalScrollIndicator = NO; + self.delegate = self; + } + return self; +} + +- (CGSize)sizeThatFits:(CGSize)size { + if (self.contentView) { + CGSize childSize = [self.contentView sizeThatFits:size]; + return CGSizeMake(MIN(size.width, childSize.width), MIN(size.height, childSize.height)); + } + return CGSizeZero; +} + +- (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)layoutSubviews { + [super layoutSubviews]; + [self layoutSelf]; + [self.contentView also:^(UIView *it) { + [it layoutSubviews]; + it.x = it.y = 0; + }]; + [self.headerView also:^(UIView *it) { + [it layoutSubviews]; + it.bottom = it.centerX = 0; + }]; + self.contentSize = self.frame.size; +} + +- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate { + if (-scrollView.contentOffset.y >= self.headerView.height) { + self.refreshing = YES; + } +} + +- (void)scrollViewDidScroll:(UIScrollView *)scrollView { + if (scrollView.contentOffset.y <= 0) { + [self.swipePullingDelegate setProgressRotation:-scrollView.contentOffset.y / self.headerView.height]; + } +} + +- (void)setRefreshing:(BOOL)refreshing { + if (_refreshing == refreshing) { + return; + } + if (refreshing) { + [UIView animateWithDuration:0.3f + animations:^{ + self.contentInset = UIEdgeInsetsMake(self.headerView.height, 0, 0, 0); + } + completion:^(BOOL finished) { + [self.swipePullingDelegate startAnimation]; + } + ]; + } else { + [UIView animateWithDuration:0.3f + animations:^{ + self.contentInset = UIEdgeInsetsMake(0, 0, 0, 0); + } + completion:^(BOOL finished) { + [self.swipePullingDelegate stopAnimation]; + } + ]; + } + _refreshing = refreshing; +} + +- (void)setRefreshable:(BOOL)refreshable { + self.scrollEnabled = refreshable; +} + +- (BOOL)refreshable { + return self.scrollEnabled; +} +@end diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.h b/iOS/Pod/Classes/Shader/DoricLayouts.h index 33c3ea86..375435eb 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.h +++ b/iOS/Pod/Classes/Shader/DoricLayouts.h @@ -94,4 +94,8 @@ typedef NS_ENUM(NSInteger, DoricGravity) { @property(nonatomic, copy) NSString *tagString; - (UIView *)viewWithTagString:(NSString *)tagString; + +- (CGSize)targetLayoutSize; + +- (void)layoutSelf; @end diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index bf238368..776a2802 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -70,7 +70,7 @@ - (void)layoutSubviews { if ([self.superview isKindOfClass:[DoricLayoutContainer class]]) { [self.superview layoutSubviews]; } else { - CGSize size = [self sizeThatFits:CGSizeMake(self.superview.width, self.superview.height)]; + CGSize size = [self sizeThatFits:self.targetLayoutSize]; [self layout:size]; } } @@ -160,6 +160,8 @@ - (CGSize)sizeContent:(CGSize)size { } - (void)layout:(CGSize)targetSize { + self.width = targetSize.width; + self.height = targetSize.height; for (UIView *child in self.subviews) { if (child.isHidden) { continue; @@ -223,8 +225,6 @@ - (void)layout:(CGSize)targetSize { [(DoricLayoutContainer *) child layout:size]; } } - self.width = targetSize.width; - self.height = targetSize.height; } @end @@ -278,6 +278,8 @@ - (CGSize)sizeContent:(CGSize)size { } - (void)layout:(CGSize)targetSize { + self.width = targetSize.width; + self.height = targetSize.height; CGFloat yStart = 0; if ((self.gravity & TOP) == TOP) { yStart = 0; @@ -350,8 +352,6 @@ - (void)layout:(CGSize)targetSize { [(DoricLayoutContainer *) child layout:size]; } } - self.width = targetSize.width; - self.height = targetSize.height; } @end @@ -403,6 +403,8 @@ - (CGSize)sizeContent:(CGSize)size { } - (void)layout:(CGSize)targetSize { + self.width = targetSize.width; + self.height = targetSize.height; CGFloat xStart = 0; if (self.contentWeight) { xStart = 0; @@ -477,8 +479,6 @@ - (void)layout:(CGSize)targetSize { [(DoricLayoutContainer *) child layout:size]; } } - self.width = targetSize.width; - self.height = targetSize.height; } @end @@ -511,6 +511,33 @@ - (UIView *)viewWithTagString:(NSString *)tagString { return [self viewWithTag:[tagString hash]]; } +- (CGSize)targetLayoutSize { + CGFloat width = self.width; + CGFloat height = self.height; + if (self.layoutConfig.widthSpec == DoricLayoutAtMost + || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { + width = self.superview.width; + } + if (self.layoutConfig.heightSpec == DoricLayoutAtMost + || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { + height = self.superview.height; + } + return CGSizeMake(width, height); +} + +- (void)layoutSelf { + CGSize contentSize = [self sizeThatFits:self.targetLayoutSize]; + if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { + self.width = self.superview.width; + } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { + self.width = contentSize.width; + } + if (self.layoutConfig.heightSpec == DoricLayoutAtMost) { + self.height = self.superview.height; + } else if (self.layoutConfig.heightSpec == DoricLayoutWrapContent) { + self.height = contentSize.height; + } +} @end DoricVLayoutView *vLayout(NSArray <__kindof UIView *> *views) { 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 c492e560..3cf29c4a 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -22,19 +22,25 @@ #import "DoricScrollerNode.h" #import "DoricExtensions.h" -@interface DoricScrollView : UIScrollView -@end - @implementation DoricScrollView +- (void)setContentView:(UIView *)contentView { + if (_contentView) { + [_contentView removeFromSuperview]; + } + _contentView = contentView; + [self addSubview:contentView]; +} + - (void)layoutSubviews { [super layoutSubviews]; + [self layoutSelf]; + [self.contentView layoutSubviews]; } - (CGSize)sizeThatFits:(CGSize)size { - if (self.subviews.count > 0) { - UIView *child = self.subviews[0]; - CGSize childSize = [child sizeThatFits:size]; + if (self.contentView) { + CGSize childSize = [self.contentView sizeThatFits:size]; return CGSizeMake(MIN(size.width, childSize.width), MIN(size.height, childSize.height)); } return CGSizeZero; @@ -47,7 +53,7 @@ @interface DoricScrollerNode () @end @implementation DoricScrollerNode -- (UIScrollView *)build { +- (DoricScrollView *)build { return [DoricScrollView new]; } @@ -70,12 +76,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; }]; } } @@ -84,13 +89,12 @@ - (void)blend:(NSDictionary *)props { it.viewId = viewId; [it initWithSuperNode:self]; [it blend:childProps]; - [self.view addSubview:it.view]; + self.view.contentView = it.view; }]; } - [self.view also:^(UIScrollView *it) { - if (it.subviews.count > 0) { - UIView *child = it.subviews[0]; - CGSize size = [child sizeThatFits:it.frame.size]; + [self.view also:^(DoricScrollView *it) { + if (it.contentView) { + CGSize size = [it.contentView sizeThatFits:it.frame.size]; [it setContentSize:size]; } }]; From 2fbf4602e839e21db1b922375d11db3353f4b8ad Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Tue, 26 Nov 2019 21:25:23 +0800 Subject: [PATCH 18/35] feat:add SwipeLayout --- .../Classes/Refresh/DoricRefreshableNode.m | 13 ++++++++++ .../Classes/Refresh/DoricSwipeRefreshLayout.m | 16 +++++------- iOS/Pod/Classes/Shader/DoricLayouts.m | 8 ++++++ iOS/Pod/Classes/Shader/DoricScrollerNode.m | 26 +++++++++---------- 4 files changed, 39 insertions(+), 24 deletions(-) diff --git a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m index 3406fcb1..1b9c3c90 100644 --- a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m +++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m @@ -48,6 +48,19 @@ - (void)blend:(NSDictionary *)props { [super blend:props]; [self blendHeader]; [self blendContent]; + dispatch_async(dispatch_get_main_queue(), ^{ + [self.view also:^(DoricSwipeRefreshLayout *layout) { + [layout layoutSelf]; + [layout.contentView also:^(UIView *it) { + it.x = it.y = 0; + }]; + [layout.headerView also:^(UIView *it) { + it.bottom = 0; + it.centerX = layout.width / 2; + }]; + layout.contentSize = layout.frame.size; + }]; + }); } - (void)blendContent { diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index bc6f28a0..868be55b 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -17,6 +17,7 @@ - (instancetype)initWithFrame:(CGRect)frame { if (self = [super initWithFrame:frame]) { self.showsHorizontalScrollIndicator = NO; self.showsVerticalScrollIndicator = NO; + self.alwaysBounceVertical = YES; self.delegate = self; } return self; @@ -26,6 +27,7 @@ - (instancetype)init { if (self = [super init]) { self.showsHorizontalScrollIndicator = NO; self.showsVerticalScrollIndicator = NO; + self.alwaysBounceVertical = YES; self.delegate = self; } return self; @@ -58,16 +60,6 @@ - (void)setHeaderView:(UIView *)headerView { - (void)layoutSubviews { [super layoutSubviews]; - [self layoutSelf]; - [self.contentView also:^(UIView *it) { - [it layoutSubviews]; - it.x = it.y = 0; - }]; - [self.headerView also:^(UIView *it) { - [it layoutSubviews]; - it.bottom = it.centerX = 0; - }]; - self.contentSize = self.frame.size; } - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate { @@ -115,4 +107,8 @@ - (void)setRefreshable:(BOOL)refreshable { - (BOOL)refreshable { return self.scrollEnabled; } + +- (void)setContentSize:(CGSize)contentSize { + [super setContentSize:contentSize]; +} @end diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 776a2802..53bf0d7b 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -20,6 +20,7 @@ #import "DoricLayouts.h" #import #import "UIView+Doric.h" +#import "Doric.h" DoricMargin DoricMarginMake(CGFloat left, CGFloat top, CGFloat right, CGFloat bottom) { DoricMargin margin; @@ -537,6 +538,13 @@ - (void)layoutSelf { } else if (self.layoutConfig.heightSpec == DoricLayoutWrapContent) { self.height = contentSize.height; } + [self.subviews forEach:^(__kindof UIView *obj) { + if ([obj isKindOfClass:[DoricLayoutContainer class]]) { + [obj layoutSubviews]; + } else { + [obj layoutSelf]; + } + }]; } @end diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index 3cf29c4a..d053566e 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -32,12 +32,6 @@ - (void)setContentView:(UIView *)contentView { [self addSubview:contentView]; } -- (void)layoutSubviews { - [super layoutSubviews]; - [self layoutSelf]; - [self.contentView layoutSubviews]; -} - - (CGSize)sizeThatFits:(CGSize)size { if (self.contentView) { CGSize childSize = [self.contentView sizeThatFits:size]; @@ -92,15 +86,19 @@ - (void)blend:(NSDictionary *)props { self.view.contentView = it.view; }]; } - [self.view also:^(DoricScrollView *it) { - if (it.contentView) { - CGSize size = [it.contentView sizeThatFits:it.frame.size]; - [it setContentSize:size]; - } - }]; + dispatch_async(dispatch_get_main_queue(), ^{ + [self.view also:^(DoricScrollView *it) { + [it layoutSelf]; + if (it.contentView) { + CGSize size = [it.contentView sizeThatFits:it.frame.size]; + [it setContentSize:size]; + } + [it layoutSelf]; + }]; + }); } -- (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 { @@ -111,4 +109,4 @@ - (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 +@end From 3c04e7cde1d53bfa74893407075c10940ae68467 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 10:19:25 +0800 Subject: [PATCH 19/35] feat:add Rotation for iOS --- iOS/Pod/Classes/Plugin/DoricShaderPlugin.m | 12 +++++++----- iOS/Pod/Classes/Shader/DoricScrollerNode.m | 7 +++++++ iOS/Pod/Classes/Shader/DoricSuperNode.m | 4 +++- iOS/Pod/Classes/Shader/DoricViewNode.m | 10 ++++++++++ 4 files changed, 27 insertions(+), 6 deletions(-) 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/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index d053566e..087ae15d 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -109,4 +109,11 @@ - (void)blendView:(DoricScrollView *)view forPropName:(NSString *)name propValue - (void)blendSubNode:(NSDictionary *)subModel { [self.childNode blend:subModel[@"props"]]; } + +- (DoricViewNode *)subNodeWithViewId:(NSString *)viewId { + if ([viewId isEqualToString:self.childViewId]) { + return self.childNode; + } + return nil; +} @end 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..ce773dbe 100644 --- a/iOS/Pod/Classes/Shader/DoricViewNode.m +++ b/iOS/Pod/Classes/Shader/DoricViewNode.m @@ -232,4 +232,14 @@ - (NSNumber *)getHeight { return @(self.view.height); } +- (void)setRotation:(NSNumber *)rotation { + self.view.transform = CGAffineTransformRotate(self.view.transform, 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 From a41036c91dd69b02c131497558356dfa3f7c53bc Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 11:55:49 +0800 Subject: [PATCH 20/35] feat:fix transform and layoutcontainer conflict --- demo/src/ListDemo.ts | 1 - demo/src/RefreshableDemo.ts | 27 ++-- demo/src/utils.ts | 4 +- .../Classes/Refresh/DoricRefreshableNode.m | 20 ++- iOS/Pod/Classes/Shader/DoricLayouts.m | 125 +++++++++++------- iOS/Pod/Classes/Shader/DoricViewNode.m | 6 +- js-framework/src/ui/refreshable.ts | 1 - 7 files changed, 113 insertions(+), 71 deletions(-) diff --git a/demo/src/ListDemo.ts b/demo/src/ListDemo.ts index b3de5c46..9c037881 100644 --- a/demo/src/ListDemo.ts +++ b/demo/src/ListDemo.ts @@ -1,6 +1,5 @@ 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"; -import { isObject } from "util"; @Entry class ListPanel extends Panel { build(rootView: Group): void { diff --git a/demo/src/RefreshableDemo.ts b/demo/src/RefreshableDemo.ts index 743da1e6..211108be 100644 --- a/demo/src/RefreshableDemo.ts +++ b/demo/src/RefreshableDemo.ts @@ -20,18 +20,21 @@ class RefreshableDemo extends Panel { width: 30, height: 30, imageBase64: icon_refresh, - }).also(v => refreshImage = v), - ]), { - startAnimation: () => { - log('startAnimation') - }, - stopAnimation: () => { - log('stopAnimation') - }, - setProgressRotation: (rotation: number) => { - refreshImage.setRotation(context, rotation) - }, - }), + }).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({ diff --git a/demo/src/utils.ts b/demo/src/utils.ts index 8bb513b6..385cef4c 100644 --- a/demo/src/utils.ts +++ b/demo/src/utils.ts @@ -1,6 +1,4 @@ -import { Color, text, Stack, Text, layoutConfig, LayoutSpec, gravity, pullable, stack, image, Image, BridgeContext } from "doric"; -import { log } from "util"; - +import { Color, text, Stack, Text, layoutConfig, LayoutSpec, gravity, pullable, stack, image, Image, BridgeContext, log } from "doric"; export const icon_refresh = '' export const colors = [ diff --git a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m index 1b9c3c90..37f999d6 100644 --- a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m +++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m @@ -5,7 +5,7 @@ #import "DoricRefreshableNode.h" #import "Doric.h" -@interface DoricRefreshableNode () +@interface DoricRefreshableNode () @property(nonatomic, strong) DoricViewNode *contentNode; @property(nonatomic, copy) NSString *contentViewId; @property(nonatomic, strong) DoricViewNode *headerNode; @@ -14,7 +14,10 @@ @interface DoricRefreshableNode () @implementation DoricRefreshableNode - (DoricSwipeRefreshLayout *)build { - return [DoricSwipeRefreshLayout new]; + return [[DoricSwipeRefreshLayout new] also:^(DoricSwipeRefreshLayout *it) { + it.swipePullingDelegate = self; + + }]; } - (void)blendView:(DoricSwipeRefreshLayout *)view forPropName:(NSString *)name propValue:(id)prop { @@ -138,4 +141,17 @@ - (void)blendHeader { - (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]; +} + @end diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 53bf0d7b..7c9f4229 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -129,28 +129,33 @@ - (CGSize)sizeContent:(CGSize)size { if (child.isHidden) { continue; } + DoricLayoutConfig *childConfig = child.layoutConfig; 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; + if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { + 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; + } + } 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); @@ -167,6 +172,9 @@ - (void)layout:(CGSize)targetSize { if (child.isHidden) { continue; } + if (!CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { + continue; + } DoricLayoutConfig *childConfig = child.layoutConfig; if (!childConfig) { childConfig = [DoricLayoutConfig new]; @@ -246,23 +254,27 @@ - (CGSize)sizeContent:(CGSize)size { 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; + if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { + 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; + } + } 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; @@ -294,6 +306,9 @@ - (void)layout:(CGSize)targetSize { if (child.isHidden) { continue; } + if (!CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { + continue; + } DoricLayoutConfig *childConfig = child.layoutConfig; if (!childConfig) { childConfig = [DoricLayoutConfig new]; @@ -371,24 +386,29 @@ - (CGSize)sizeContent:(CGSize)size { 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; + if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { + 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; + } + } 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); contentWeight += childConfig.weight; @@ -421,6 +441,9 @@ - (void)layout:(CGSize)targetSize { if (child.isHidden) { continue; } + if (!CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { + continue; + } DoricLayoutConfig *childConfig = child.layoutConfig; if (!childConfig) { childConfig = [DoricLayoutConfig new]; diff --git a/iOS/Pod/Classes/Shader/DoricViewNode.m b/iOS/Pod/Classes/Shader/DoricViewNode.m index ce773dbe..3a432cbb 100644 --- a/iOS/Pod/Classes/Shader/DoricViewNode.m +++ b/iOS/Pod/Classes/Shader/DoricViewNode.m @@ -233,7 +233,11 @@ - (NSNumber *)getHeight { } - (void)setRotation:(NSNumber *)rotation { - self.view.transform = CGAffineTransformRotate(self.view.transform, M_PI * rotation.floatValue * 2); + if (rotation.floatValue == 0) { + self.view.transform = CGAffineTransformIdentity; + } else { + self.view.transform = CGAffineTransformMakeRotation(M_PI * rotation.floatValue * 2); + } } - (NSNumber *)getRotation { diff --git a/js-framework/src/ui/refreshable.ts b/js-framework/src/ui/refreshable.ts index 35de57d7..556196e3 100644 --- a/js-framework/src/ui/refreshable.ts +++ b/js-framework/src/ui/refreshable.ts @@ -3,7 +3,6 @@ import { List } from "./list"; import { Scroller } from "./scroller"; import { BridgeContext } from "../runtime/global"; import { layoutConfig } from "./declarative"; -import { Image } from "./widgets"; export interface IRefreshable extends IView { content: List | Scroller From 9be891e2848dd187d6ee23702236ed4d20dfa828 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 13:30:31 +0800 Subject: [PATCH 21/35] feat:fix Refreshable pulldown and so on --- iOS/Pod/Classes/Doric.h | 3 ++- iOS/Pod/Classes/Refresh/DoricRefreshableNode.m | 17 +++++++++++++++++ .../Classes/Refresh/DoricSwipeRefreshLayout.m | 3 +++ iOS/Pod/Classes/Shader/DoricLayouts.m | 2 +- iOS/Pod/Classes/Shader/DoricListNode.m | 7 ++++++- iOS/Pod/Classes/Shader/DoricSliderNode.m | 2 ++ 6 files changed, 31 insertions(+), 3 deletions(-) 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/Refresh/DoricRefreshableNode.m b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m index 37f999d6..df6de98c 100644 --- a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m +++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m @@ -154,4 +154,21 @@ - (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.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index 868be55b..84d3b6d6 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -79,6 +79,9 @@ - (void)setRefreshing:(BOOL)refreshing { return; } if (refreshing) { + if (self.onRefreshBlock) { + self.onRefreshBlock(); + } [UIView animateWithDuration:0.3f animations:^{ self.contentInset = UIEdgeInsetsMake(self.headerView.height, 0, 0, 0); diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 7c9f4229..13e82999 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -543,7 +543,7 @@ - (CGSize)targetLayoutSize { width = self.superview.width; } if (self.layoutConfig.heightSpec == DoricLayoutAtMost - || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { + || self.layoutConfig.heightSpec == DoricLayoutWrapContent) { height = self.superview.height; } return CGSizeMake(width, height); diff --git a/iOS/Pod/Classes/Shader/DoricListNode.m b/iOS/Pod/Classes/Shader/DoricListNode.m index b1aa6eb4..4625e6a6 100644 --- a/iOS/Pod/Classes/Shader/DoricListNode.m +++ b/iOS/Pod/Classes/Shader/DoricListNode.m @@ -36,10 +36,13 @@ @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); + height += child.height; } - return CGSizeMake(width, size.width); + return CGSizeMake(width, height); } return size; } @@ -74,9 +77,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 { diff --git a/iOS/Pod/Classes/Shader/DoricSliderNode.m b/iOS/Pod/Classes/Shader/DoricSliderNode.m index e409dcd4..de74db8b 100644 --- a/iOS/Pod/Classes/Shader/DoricSliderNode.m +++ b/iOS/Pod/Classes/Shader/DoricSliderNode.m @@ -80,9 +80,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 { From fe292f7b7fce8d9f6992d01762c1d5418b7d6b55 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 14:16:55 +0800 Subject: [PATCH 22/35] feat:Prepare to refact DoricLayouts,because it wont be sized corretly for other views like scrollview or tableview --- iOS/Pod/Classes/Shader/DoricLayouts.h | 2 ++ iOS/Pod/Classes/Shader/DoricLayouts.m | 33 +++++++++++++++++++++++--- iOS/Pod/Classes/Shader/DoricListNode.m | 8 ++++--- 3 files changed, 37 insertions(+), 6 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.h b/iOS/Pod/Classes/Shader/DoricLayouts.h index 375435eb..c3ac96b7 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.h +++ b/iOS/Pod/Classes/Shader/DoricLayouts.h @@ -98,4 +98,6 @@ typedef NS_ENUM(NSInteger, DoricGravity) { - (CGSize)targetLayoutSize; - (void)layoutSelf; + +- (CGSize)measureSize:(CGSize)targetSize; @end diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 13e82999..b9a609f0 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -540,17 +540,44 @@ - (CGSize)targetLayoutSize { CGFloat height = self.height; if (self.layoutConfig.widthSpec == DoricLayoutAtMost || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { - width = self.superview.width; + width = self.superview.targetLayoutSize.width; } if (self.layoutConfig.heightSpec == DoricLayoutAtMost || self.layoutConfig.heightSpec == DoricLayoutWrapContent) { - height = self.superview.height; + height = self.superview.targetLayoutSize.height; + } + return CGSizeMake(width, height); +} + +- (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); } - (void)layoutSelf { - CGSize contentSize = [self sizeThatFits:self.targetLayoutSize]; + CGSize contentSize = [self measureSize:self.targetLayoutSize]; if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { self.width = self.superview.width; } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { diff --git a/iOS/Pod/Classes/Shader/DoricListNode.m b/iOS/Pod/Classes/Shader/DoricListNode.m index 4625e6a6..02751c1b 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; @@ -39,10 +40,11 @@ - (CGSize)sizeThatFits:(CGSize)size { CGFloat height = 0; for (UIView *child in self.subviews) { - width = MAX(child.width, width); - height += child.height; + CGSize childSize = [child measureSize:size]; + width = MAX(childSize.width, width); + height += childSize.height; } - return CGSizeMake(width, height); + return CGSizeMake(width, MAX(height, size.height)); } return size; } From bb9e6671f662ac61b5acc3c67c2e1d1f4d26b492 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 14:31:52 +0800 Subject: [PATCH 23/35] feat:step1 --- iOS/Pod/Classes/Shader/DoricLayouts.h | 18 +-- iOS/Pod/Classes/Shader/DoricLayouts.m | 197 ++++++++++++++------------ 2 files changed, 113 insertions(+), 102 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.h b/iOS/Pod/Classes/Shader/DoricLayouts.h index c3ac96b7..f0da117e 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); @@ -91,13 +84,16 @@ 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 -- (CGSize)targetLayoutSize; - +@interface UIView (DoricLayouts) - (void)layoutSelf; - (CGSize)measureSize:(CGSize)targetSize; -@end +@end \ No newline at end of file diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index b9a609f0..1ca0cd9a 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -22,6 +22,108 @@ #import "UIView+Doric.h" #import "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) + +- (CGSize)targetLayoutSize { + CGFloat width = self.width; + CGFloat height = self.height; + if (self.layoutConfig.widthSpec == DoricLayoutAtMost + || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { + width = self.superview.targetLayoutSize.width; + } + if (self.layoutConfig.heightSpec == DoricLayoutAtMost + || self.layoutConfig.heightSpec == DoricLayoutWrapContent) { + height = self.superview.targetLayoutSize.height; + } + return CGSizeMake(width, height); +} + +- (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); +} + +- (void)layoutSelf { + CGSize contentSize = [self measureSize:self.targetLayoutSize]; + if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { + self.width = self.superview.width; + } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { + self.width = contentSize.width; + } + if (self.layoutConfig.heightSpec == DoricLayoutAtMost) { + self.height = self.superview.height; + } else if (self.layoutConfig.heightSpec == DoricLayoutWrapContent) { + self.height = contentSize.height; + } + [self.subviews forEach:^(__kindof UIView *obj) { + if ([obj isKindOfClass:[DoricLayoutContainer class]]) { + [obj layoutSubviews]; + } else { + [obj layoutSelf]; + } + }]; +} +@end + DoricMargin DoricMarginMake(CGFloat left, CGFloat top, CGFloat right, CGFloat bottom) { DoricMargin margin; margin.left = left; @@ -373,6 +475,10 @@ - (void)layout:(CGSize)targetSize { @implementation DoricHLayoutView +- (void)layoutSelf { + [super layoutSelf]; +} + - (CGSize)sizeContent:(CGSize)size { CGFloat contentWidth = 0; CGFloat contentHeight = 0; @@ -506,97 +612,6 @@ - (void)layout:(CGSize)targetSize { } @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]]; -} - -- (CGSize)targetLayoutSize { - CGFloat width = self.width; - CGFloat height = self.height; - if (self.layoutConfig.widthSpec == DoricLayoutAtMost - || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { - width = self.superview.targetLayoutSize.width; - } - if (self.layoutConfig.heightSpec == DoricLayoutAtMost - || self.layoutConfig.heightSpec == DoricLayoutWrapContent) { - height = self.superview.targetLayoutSize.height; - } - return CGSizeMake(width, height); -} - -- (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); -} - -- (void)layoutSelf { - CGSize contentSize = [self measureSize:self.targetLayoutSize]; - if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { - self.width = self.superview.width; - } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { - self.width = contentSize.width; - } - if (self.layoutConfig.heightSpec == DoricLayoutAtMost) { - self.height = self.superview.height; - } else if (self.layoutConfig.heightSpec == DoricLayoutWrapContent) { - self.height = contentSize.height; - } - [self.subviews forEach:^(__kindof UIView *obj) { - if ([obj isKindOfClass:[DoricLayoutContainer class]]) { - [obj layoutSubviews]; - } else { - [obj layoutSelf]; - } - }]; -} -@end DoricVLayoutView *vLayout(NSArray <__kindof UIView *> *views) { DoricVLayoutView *layout = [[DoricVLayoutView alloc] initWithFrame:CGRectZero]; From af8a455d466a5cee1be51afa58ea36e0dabc314c Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 15:28:39 +0800 Subject: [PATCH 24/35] feat:step2 --- .../Classes/Refresh/DoricRefreshableNode.m | 13 - iOS/Pod/Classes/Shader/DoricLayouts.h | 10 +- iOS/Pod/Classes/Shader/DoricLayouts.m | 276 ++++-------------- iOS/Pod/Classes/Shader/DoricScrollerNode.m | 10 - 4 files changed, 58 insertions(+), 251 deletions(-) diff --git a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m index df6de98c..137dc5e2 100644 --- a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m +++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m @@ -51,19 +51,6 @@ - (void)blend:(NSDictionary *)props { [super blend:props]; [self blendHeader]; [self blendContent]; - dispatch_async(dispatch_get_main_queue(), ^{ - [self.view also:^(DoricSwipeRefreshLayout *layout) { - [layout layoutSelf]; - [layout.contentView also:^(UIView *it) { - it.x = it.y = 0; - }]; - [layout.headerView also:^(UIView *it) { - it.bottom = 0; - it.centerX = layout.width / 2; - }]; - layout.contentSize = layout.frame.size; - }]; - }); } - (void)blendContent { diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.h b/iOS/Pod/Classes/Shader/DoricLayouts.h index f0da117e..86824f3c 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.h +++ b/iOS/Pod/Classes/Shader/DoricLayouts.h @@ -61,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 @@ -93,7 +89,9 @@ typedef NS_ENUM(NSInteger, DoricGravity) { @end @interface UIView (DoricLayouts) -- (void)layoutSelf; +- (void)layoutSelf:(CGSize)targetSize; - (CGSize)measureSize:(CGSize)targetSize; + +- (void)doricLayoutSubviews; @end \ No newline at end of file diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 1ca0cd9a..5ef151f0 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -19,8 +19,8 @@ #import "DoricLayouts.h" #import +#import #import "UIView+Doric.h" -#import "Doric.h" static const void *kLayoutConfig = &kLayoutConfig; @@ -60,21 +60,9 @@ - (UIView *)viewWithTagString:(NSString *)tagString { @implementation UIView (DoricLayouts) - -- (CGSize)targetLayoutSize { - CGFloat width = self.width; - CGFloat height = self.height; - if (self.layoutConfig.widthSpec == DoricLayoutAtMost - || self.layoutConfig.widthSpec == DoricLayoutWrapContent) { - width = self.superview.targetLayoutSize.width; - } - if (self.layoutConfig.heightSpec == DoricLayoutAtMost - || self.layoutConfig.heightSpec == DoricLayoutWrapContent) { - height = self.superview.targetLayoutSize.height; - } - return CGSizeMake(width, height); -} - +/** + * Measure self's size + * */ - (CGSize)measureSize:(CGSize)targetSize { CGFloat width = self.width; CGFloat height = self.height; @@ -102,26 +90,38 @@ - (CGSize)measureSize:(CGSize)targetSize { return CGSizeMake(width, height); } -- (void)layoutSelf { - CGSize contentSize = [self measureSize:self.targetLayoutSize]; +/** + * layout self and subviews + * */ +- (void)layoutSelf:(CGSize)targetSize { + CGSize contentSize = [self measureSize:targetSize]; if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { - self.width = self.superview.width; + self.width = targetSize.width; } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { self.width = contentSize.width; } if (self.layoutConfig.heightSpec == DoricLayoutAtMost) { - self.height = self.superview.height; + self.height = targetSize.height; } else if (self.layoutConfig.heightSpec == DoricLayoutWrapContent) { self.height = contentSize.height; } - [self.subviews forEach:^(__kindof UIView *obj) { - if ([obj isKindOfClass:[DoricLayoutContainer class]]) { - [obj layoutSubviews]; - } else { - [obj layoutSelf]; - } - }]; } + +- (BOOL)requestSuperview { + return self.superview + && self.layoutConfig + && self.layoutConfig.widthSpec != DoricLayoutExact + && self.layoutConfig.heightSpec != DoricLayoutExact; +} + +- (void)doricLayoutSubviews { + if ([self requestSuperview]) { + [self.superview doricLayoutSubviews]; + } else { + [self layoutSelf:CGSizeMake(self.width, self.height)]; + } +} + @end DoricMargin DoricMarginMake(CGFloat left, CGFloat top, CGFloat right, CGFloat bottom) { @@ -168,94 +168,34 @@ @interface DoricLayoutContainer () @end @implementation DoricLayoutContainer - - (void)layoutSubviews { - if ([self.superview isKindOfClass:[DoricLayoutContainer class]]) { - [self.superview layoutSubviews]; - } else { - CGSize size = [self sizeThatFits:self.targetLayoutSize]; - [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) { if (child.isHidden) { continue; } - DoricLayoutConfig *childConfig = child.layoutConfig; if (!childConfig) { childConfig = [DoricLayoutConfig new]; } - CGSize childSize = CGSizeMake(child.width, child.height); + CGSize childSize; if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { - 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; - } + childSize = [child measureSize:CGSizeMake(size.width, size.height)]; } else { childSize = child.bounds.size; } @@ -267,9 +207,8 @@ - (CGSize)sizeContent:(CGSize)size { return CGSizeMake(contentWidth, contentHeight); } -- (void)layout:(CGSize)targetSize { - self.width = targetSize.width; - self.height = targetSize.height; +- (void)layoutSelf:(CGSize)targetSize { + [super layoutSelf:targetSize]; for (UIView *child in self.subviews) { if (child.isHidden) { continue; @@ -281,29 +220,9 @@ - (void)layout:(CGSize)targetSize { 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) { @@ -317,7 +236,6 @@ - (void)layout:(CGSize)targetSize { child.right = targetSize.width - childConfig.margin.right; } } - if ((gravity & TOP) == TOP) { child.top = 0; } else if ((gravity & BOTTOM) == BOTTOM) { @@ -331,10 +249,6 @@ - (void)layout:(CGSize)targetSize { child.bottom = targetSize.height - childConfig.margin.bottom; } } - - if ([child isKindOfClass:[DoricLayoutContainer class]]) { - [(DoricLayoutContainer *) child layout:size]; - } } } @end @@ -343,7 +257,8 @@ @implementation DoricLinearView @end @implementation DoricVLayoutView -- (CGSize)sizeContent:(CGSize)size { + +- (CGSize)sizeThatFits:(CGSize)size { CGFloat contentWidth = 0; CGFloat contentHeight = 0; NSUInteger contentWeight = 0; @@ -355,23 +270,9 @@ - (CGSize)sizeContent:(CGSize)size { if (!childConfig) { childConfig = [DoricLayoutConfig new]; } - CGSize childSize = CGSizeMake(child.width, child.height); + CGSize childSize; if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { - 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; - } + childSize = [child measureSize:CGSizeMake(size.width, size.height - contentHeight)]; if (childConfig.weight) { childSize.height = child.height; } @@ -392,7 +293,8 @@ - (CGSize)sizeContent:(CGSize)size { return CGSizeMake(contentWidth, contentHeight); } -- (void)layout:(CGSize)targetSize { +- (void)layoutSelf:(CGSize)targetSize { + [super layoutSelf:targetSize]; self.width = targetSize.width; self.height = targetSize.height; CGFloat yStart = 0; @@ -416,35 +318,12 @@ - (void)layout:(CGSize)targetSize { 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; - } + CGSize size = [child measureSize:CGSizeMake(targetSize.width, targetSize.height - yStart)]; if (childConfig.weight) { - size.height = child.height; + size.height = child.height + remain / self.contentWeight * childConfig.weight; } - - 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) { @@ -466,20 +345,12 @@ - (void)layout:(CGSize)targetSize { if (childConfig.margin.bottom) { yStart += childConfig.margin.bottom; } - if ([child isKindOfClass:[DoricLayoutContainer class]]) { - [(DoricLayoutContainer *) child layout:size]; - } } } @end @implementation DoricHLayoutView - -- (void)layoutSelf { - [super layoutSelf]; -} - -- (CGSize)sizeContent:(CGSize)size { +- (CGSize)sizeThatFits:(CGSize)size { CGFloat contentWidth = 0; CGFloat contentHeight = 0; NSUInteger contentWeight = 0; @@ -491,30 +362,15 @@ - (CGSize)sizeContent:(CGSize)size { if (!childConfig) { childConfig = [DoricLayoutConfig new]; } - CGSize childSize = CGSizeMake(child.width, child.height); + CGSize childSize; if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { - 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; - } + childSize = [child measureSize:CGSizeMake(size.width - contentWidth, size.height)]; if (childConfig.weight) { childSize.width = child.width; } } 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); contentWeight += childConfig.weight; @@ -529,9 +385,8 @@ - (CGSize)sizeContent:(CGSize)size { return CGSizeMake(contentWidth, contentHeight); } -- (void)layout:(CGSize)targetSize { - self.width = targetSize.width; - self.height = targetSize.height; +- (void)layoutSelf:(CGSize)targetSize { + [super layoutSelf:targetSize]; CGFloat xStart = 0; if (self.contentWeight) { xStart = 0; @@ -555,32 +410,12 @@ - (void)layout:(CGSize)targetSize { 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; - } + CGSize size = [child measureSize:CGSizeMake(targetSize.width - xStart, targetSize.height)]; if (childConfig.weight) { - size.width = child.width; + size.width = child.width + remain / self.contentWeight * childConfig.weight; } - 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) { @@ -605,9 +440,6 @@ - (void)layout:(CGSize)targetSize { if (childConfig.margin.right) { xStart += childConfig.margin.right; } - if ([child isKindOfClass:[DoricLayoutContainer class]]) { - [(DoricLayoutContainer *) child layout:size]; - } } } @end diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index 087ae15d..5fb7d0c0 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -86,16 +86,6 @@ - (void)blend:(NSDictionary *)props { self.view.contentView = it.view; }]; } - dispatch_async(dispatch_get_main_queue(), ^{ - [self.view also:^(DoricScrollView *it) { - [it layoutSelf]; - if (it.contentView) { - CGSize size = [it.contentView sizeThatFits:it.frame.size]; - [it setContentSize:size]; - } - [it layoutSelf]; - }]; - }); } - (void)blendView:(DoricScrollView *)view forPropName:(NSString *)name propValue:(id)prop { From 1dabccbe50895f496decade5539f0574643a814a Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 16:04:43 +0800 Subject: [PATCH 25/35] feat:step3:scrollview --- iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m | 11 ++++++----- iOS/Pod/Classes/Shader/DoricLayouts.m | 5 ++--- iOS/Pod/Classes/Shader/DoricListNode.m | 5 +++++ iOS/Pod/Classes/Shader/DoricScrollerNode.m | 9 +++++++-- 4 files changed, 20 insertions(+), 10 deletions(-) diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index 84d3b6d6..f92c0c9f 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -35,12 +35,16 @@ - (instancetype)init { - (CGSize)sizeThatFits:(CGSize)size { if (self.contentView) { - CGSize childSize = [self.contentView sizeThatFits:size]; - return CGSizeMake(MIN(size.width, childSize.width), MIN(size.height, childSize.height)); + return [self.contentView measureSize:size]; } return CGSizeZero; } +- (void)layoutSelf:(CGSize)targetSize { + [super layoutSelf:targetSize]; + [self.contentView layoutSelf:targetSize]; +} + - (void)setContentView:(UIView *)contentView { if (_contentView) { [_contentView removeFromSuperview]; @@ -58,9 +62,6 @@ - (void)setHeaderView:(UIView *)headerView { self.refreshable = YES; } -- (void)layoutSubviews { - [super layoutSubviews]; -} - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate { if (-scrollView.contentOffset.y >= self.headerView.height) { diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 5ef151f0..a806496c 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -94,7 +94,7 @@ - (CGSize)measureSize:(CGSize)targetSize { * layout self and subviews * */ - (void)layoutSelf:(CGSize)targetSize { - CGSize contentSize = [self measureSize:targetSize]; + CGSize contentSize = [self sizeThatFits:targetSize]; if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { self.width = targetSize.width; } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { @@ -108,8 +108,7 @@ - (void)layoutSelf:(CGSize)targetSize { } - (BOOL)requestSuperview { - return self.superview - && self.layoutConfig + return self.layoutConfig && self.layoutConfig.widthSpec != DoricLayoutExact && self.layoutConfig.heightSpec != DoricLayoutExact; } diff --git a/iOS/Pod/Classes/Shader/DoricListNode.m b/iOS/Pod/Classes/Shader/DoricListNode.m index 02751c1b..2ad7e86f 100644 --- a/iOS/Pod/Classes/Shader/DoricListNode.m +++ b/iOS/Pod/Classes/Shader/DoricListNode.m @@ -48,6 +48,11 @@ - (CGSize)sizeThatFits:(CGSize)size { } return size; } + +- (void)layoutSelf:(CGSize)targetSize { + [super layoutSelf:targetSize]; + [self reloadData]; +} @end diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index 5fb7d0c0..28848f0d 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -34,11 +34,16 @@ - (void)setContentView:(UIView *)contentView { - (CGSize)sizeThatFits:(CGSize)size { if (self.contentView) { - CGSize childSize = [self.contentView sizeThatFits:size]; - return CGSizeMake(MIN(size.width, childSize.width), MIN(size.height, childSize.height)); + return [self.contentView sizeThatFits:size]; } return CGSizeZero; } + +- (void)layoutSelf:(CGSize)targetSize { + [super layoutSelf:targetSize]; + [self.contentView layoutSelf: [self.contentView sizeThatFits:targetSize]]; + [self setContentSize:self.contentView.frame.size]; +} @end @interface DoricScrollerNode () From 91a26eb9a83beeff91f2d351eed2426179fde849 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 16:41:04 +0800 Subject: [PATCH 26/35] feat:step 4 --- iOS/Pod/Classes/Shader/DoricLayouts.m | 19 ++++++------------- iOS/Pod/Classes/Shader/DoricScrollerNode.m | 7 ++++++- 2 files changed, 12 insertions(+), 14 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index a806496c..883c1625 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -94,17 +94,8 @@ - (CGSize)measureSize:(CGSize)targetSize { * layout self and subviews * */ - (void)layoutSelf:(CGSize)targetSize { - CGSize contentSize = [self sizeThatFits:targetSize]; - if (self.layoutConfig.widthSpec == DoricLayoutAtMost) { - self.width = targetSize.width; - } else if (self.layoutConfig.widthSpec == DoricLayoutWrapContent) { - self.width = contentSize.width; - } - if (self.layoutConfig.heightSpec == DoricLayoutAtMost) { - self.height = targetSize.height; - } else if (self.layoutConfig.heightSpec == DoricLayoutWrapContent) { - self.height = contentSize.height; - } + self.width = targetSize.width; + self.height = targetSize.height; } - (BOOL)requestSuperview { @@ -167,6 +158,10 @@ @interface DoricLayoutContainer () @end @implementation DoricLayoutContainer +- (void)setNeedsLayout { + [super setNeedsLayout]; +} + - (void)layoutSubviews { [super layoutSubviews]; [self doricLayoutSubviews]; @@ -294,8 +289,6 @@ - (CGSize)sizeThatFits:(CGSize)size { - (void)layoutSelf:(CGSize)targetSize { [super layoutSelf:targetSize]; - self.width = targetSize.width; - self.height = targetSize.height; CGFloat yStart = 0; if ((self.gravity & TOP) == TOP) { yStart = 0; diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index 28848f0d..da6f763c 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -41,9 +41,14 @@ - (CGSize)sizeThatFits:(CGSize)size { - (void)layoutSelf:(CGSize)targetSize { [super layoutSelf:targetSize]; - [self.contentView layoutSelf: [self.contentView sizeThatFits:targetSize]]; + [self.contentView layoutSelf:[self.contentView sizeThatFits:targetSize]]; [self setContentSize:self.contentView.frame.size]; } + +- (CGSize)measureSize:(CGSize)targetSize { + CGSize size = [super measureSize:targetSize]; + return CGSizeMake(MIN(targetSize.width, size.width), MIN(targetSize.height, size.height)); +} @end @interface DoricScrollerNode () From 209a9267fee1046a1bb2c4ae536eed76b65ddd26 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 17:03:20 +0800 Subject: [PATCH 27/35] feat:step5 --- demo/src/LayoutDemo.ts | 8 ++------ iOS/Pod/Classes/Shader/DoricScrollerNode.m | 4 ---- 2 files changed, 2 insertions(+), 10 deletions(-) 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/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index da6f763c..091deb79 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -45,10 +45,6 @@ - (void)layoutSelf:(CGSize)targetSize { [self setContentSize:self.contentView.frame.size]; } -- (CGSize)measureSize:(CGSize)targetSize { - CGSize size = [super measureSize:targetSize]; - return CGSizeMake(MIN(targetSize.width, size.width), MIN(targetSize.height, size.height)); -} @end @interface DoricScrollerNode () From e3cb635358eb5f00b7d5d6ab6f1aa0cc9685ab60 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 17:17:03 +0800 Subject: [PATCH 28/35] feat:step6 --- iOS/Pod/Classes/Shader/DoricLayouts.m | 12 +++++++++--- iOS/Pod/Classes/Shader/DoricScrollerNode.m | 1 - iOS/Pod/Classes/Shader/DoricSlideItemNode.m | 3 +++ iOS/Pod/Classes/Shader/DoricSliderNode.m | 10 ++++++++-- 4 files changed, 20 insertions(+), 6 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 883c1625..72214e07 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -96,6 +96,9 @@ - (CGSize)measureSize:(CGSize)targetSize { - (void)layoutSelf:(CGSize)targetSize { self.width = targetSize.width; self.height = targetSize.height; + for (UIView *view in self.subviews) { + [view layoutSelf:[view measureSize:targetSize]]; + } } - (BOOL)requestSuperview { @@ -202,7 +205,8 @@ - (CGSize)sizeThatFits:(CGSize)size { } - (void)layoutSelf:(CGSize)targetSize { - [super layoutSelf:targetSize]; + self.width = targetSize.width; + self.height = targetSize.height; for (UIView *child in self.subviews) { if (child.isHidden) { continue; @@ -288,7 +292,8 @@ - (CGSize)sizeThatFits:(CGSize)size { } - (void)layoutSelf:(CGSize)targetSize { - [super layoutSelf:targetSize]; + self.width = targetSize.width; + self.height = targetSize.height; CGFloat yStart = 0; if ((self.gravity & TOP) == TOP) { yStart = 0; @@ -378,7 +383,8 @@ - (CGSize)sizeThatFits:(CGSize)size { } - (void)layoutSelf:(CGSize)targetSize { - [super layoutSelf:targetSize]; + self.width = targetSize.width; + self.height = targetSize.height; CGFloat xStart = 0; if (self.contentWeight) { xStart = 0; diff --git a/iOS/Pod/Classes/Shader/DoricScrollerNode.m b/iOS/Pod/Classes/Shader/DoricScrollerNode.m index 091deb79..4d601f15 100644 --- a/iOS/Pod/Classes/Shader/DoricScrollerNode.m +++ b/iOS/Pod/Classes/Shader/DoricScrollerNode.m @@ -41,7 +41,6 @@ - (CGSize)sizeThatFits:(CGSize)size { - (void)layoutSelf:(CGSize)targetSize { [super layoutSelf:targetSize]; - [self.contentView layoutSelf:[self.contentView sizeThatFits:targetSize]]; [self setContentSize:self.contentView.frame.size]; } 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 de74db8b..5822745a 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 From efd71a69cb2304ce8eee2d38769c615cc1c12af6 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 17:33:47 +0800 Subject: [PATCH 29/35] feat:step7 --- iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m | 12 ++++++++++-- iOS/Pod/Classes/Shader/DoricLayouts.h | 2 ++ iOS/Pod/Classes/Shader/DoricLayouts.m | 15 +++++++++------ 3 files changed, 21 insertions(+), 8 deletions(-) diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index f92c0c9f..f5a4bf88 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -5,7 +5,6 @@ #import "DoricSwipeRefreshLayout.h" #import "UIView+Doric.h" #import "DoricLayouts.h" -#import "Doric.h" @interface DoricSwipeRefreshLayout () @@ -40,9 +39,18 @@ - (CGSize)sizeThatFits:(CGSize)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.contentView layoutSelf:targetSize]; + self.headerView.bottom = 0; + self.headerView.centerX = self.centerX; + self.contentSize = self.frame.size; } - (void)setContentView:(UIView *)contentView { diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.h b/iOS/Pod/Classes/Shader/DoricLayouts.h index 86824f3c..fcddee0c 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.h +++ b/iOS/Pod/Classes/Shader/DoricLayouts.h @@ -94,4 +94,6 @@ typedef NS_ENUM(NSInteger, DoricGravity) { - (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 72214e07..d1584848 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -101,20 +101,23 @@ - (void)layoutSelf:(CGSize)targetSize { } } -- (BOOL)requestSuperview { - return self.layoutConfig - && self.layoutConfig.widthSpec != DoricLayoutExact - && self.layoutConfig.heightSpec != DoricLayoutExact; -} - (void)doricLayoutSubviews { - if ([self requestSuperview]) { + 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) { From 0c23be0dea4f52c6c01419c99fcad12517eec770 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 17:41:31 +0800 Subject: [PATCH 30/35] feat:step8 --- iOS/Pod/Classes/Shader/DoricListNode.m | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricListNode.m b/iOS/Pod/Classes/Shader/DoricListNode.m index 2ad7e86f..aa1d75cc 100644 --- a/iOS/Pod/Classes/Shader/DoricListNode.m +++ b/iOS/Pod/Classes/Shader/DoricListNode.m @@ -121,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; } From f78d2ad487ffcbf58f7a158d3e58a16d01cf22b3 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 17:53:03 +0800 Subject: [PATCH 31/35] feat:step 9 --- iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m | 3 +++ 1 file changed, 3 insertions(+) diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index f5a4bf88..95c1f613 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -80,6 +80,9 @@ - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL - (void)scrollViewDidScroll:(UIScrollView *)scrollView { if (scrollView.contentOffset.y <= 0) { [self.swipePullingDelegate setProgressRotation:-scrollView.contentOffset.y / self.headerView.height]; + self.bounces = YES; + } else { + self.bounces = NO; } } From d8a743f13f4986576f90104d318a3d71df6dbcab Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 18:07:31 +0800 Subject: [PATCH 32/35] feat:step10 --- demo/src/ListDemo.ts | 1 + demo/src/RefreshableDemo.ts | 2 ++ iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m | 11 +++++++++-- 3 files changed, 12 insertions(+), 2 deletions(-) diff --git a/demo/src/ListDemo.ts b/demo/src/ListDemo.ts index 9c037881..81e65d97 100644 --- a/demo/src/ListDemo.ts +++ b/demo/src/ListDemo.ts @@ -109,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 index 211108be..0b967c01 100644 --- a/demo/src/RefreshableDemo.ts +++ b/demo/src/RefreshableDemo.ts @@ -102,6 +102,8 @@ class RefreshableDemo extends Panel { } as IVLayout)).apply({ layoutConfig: layoutConfig().atmost(), }) + }).apply({ + bgColor: Color.YELLOW }).in(rootView) } } \ No newline at end of file diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index 95c1f613..7f126026 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -18,6 +18,9 @@ - (instancetype)initWithFrame:(CGRect)frame { self.showsVerticalScrollIndicator = NO; self.alwaysBounceVertical = YES; self.delegate = self; + if (@available(iOS 11, *)) { + self.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever; + } } return self; } @@ -28,6 +31,9 @@ - (instancetype)init { self.showsVerticalScrollIndicator = NO; self.alwaysBounceVertical = YES; self.delegate = self; + if (@available(iOS 11, *)) { + self.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever; + } } return self; } @@ -80,9 +86,10 @@ - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL - (void)scrollViewDidScroll:(UIScrollView *)scrollView { if (scrollView.contentOffset.y <= 0) { [self.swipePullingDelegate setProgressRotation:-scrollView.contentOffset.y / self.headerView.height]; - self.bounces = YES; + //self.bounces = YES; } else { - self.bounces = NO; + //self.bounces = NO; + //scrollView.contentOffset = CGPointMake(scrollView.contentOffset.x, 0); } } From ea8ea4a8ff9ef390dd86d4ad5f301f3c61f9dbe4 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 18:32:59 +0800 Subject: [PATCH 33/35] feat:step 11 --- iOS/Pod/Classes/Shader/DoricLayouts.m | 10 ++-------- iOS/Pod/Classes/Shader/DoricSliderNode.m | 3 ++- 2 files changed, 4 insertions(+), 9 deletions(-) diff --git a/iOS/Pod/Classes/Shader/DoricLayouts.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index d1584848..902b4478 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -274,9 +274,6 @@ - (CGSize)sizeThatFits:(CGSize)size { CGSize childSize; if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { childSize = [child measureSize:CGSizeMake(size.width, size.height - contentHeight)]; - if (childConfig.weight) { - childSize.height = child.height; - } } else { childSize = child.bounds.size; } @@ -320,7 +317,7 @@ - (void)layoutSelf:(CGSize)targetSize { CGSize size = [child measureSize:CGSizeMake(targetSize.width, targetSize.height - yStart)]; if (childConfig.weight) { - size.height = child.height + remain / self.contentWeight * childConfig.weight; + size.height += remain / self.contentWeight * childConfig.weight; } [child layoutSelf:size]; DoricGravity gravity = childConfig.alignment | self.gravity; @@ -365,9 +362,6 @@ - (CGSize)sizeThatFits:(CGSize)size { CGSize childSize; if (CGAffineTransformEqualToTransform(child.transform, CGAffineTransformIdentity)) { childSize = [child measureSize:CGSizeMake(size.width - contentWidth, size.height)]; - if (childConfig.weight) { - childSize.width = child.width; - } } else { childSize = child.bounds.size; } @@ -413,7 +407,7 @@ - (void)layoutSelf:(CGSize)targetSize { CGSize size = [child measureSize:CGSizeMake(targetSize.width - xStart, targetSize.height)]; if (childConfig.weight) { - size.width = child.width + remain / self.contentWeight * childConfig.weight; + size.width += remain / self.contentWeight * childConfig.weight; } [child layoutSelf:size]; diff --git a/iOS/Pod/Classes/Shader/DoricSliderNode.m b/iOS/Pod/Classes/Shader/DoricSliderNode.m index 5822745a..9e495f8c 100644 --- a/iOS/Pod/Classes/Shader/DoricSliderNode.m +++ b/iOS/Pod/Classes/Shader/DoricSliderNode.m @@ -128,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; } From 8b1ed0acc826a37ce9ed83fa9568fca6d46ae2d6 Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 19:15:57 +0800 Subject: [PATCH 34/35] feat:Image support animated image --- .../main/java/pub/doric/shader/ImageNode.java | 5 ++- demo/src/ImageDemo.ts | 9 ++++- iOS/Doric.podspec | 5 ++- iOS/Example/Podfile.lock | 37 ++++++++----------- iOS/Pod/Classes/Shader/DoricImageNode.m | 7 ++-- iOS/Pod/Classes/Shader/DoricLayouts.m | 2 - 6 files changed, 31 insertions(+), 34 deletions(-) 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/demo/src/ImageDemo.ts b/demo/src/ImageDemo.ts index a26c4cee..15ce8d55 100644 --- a/demo/src/ImageDemo.ts +++ b/demo/src/ImageDemo.ts @@ -1,4 +1,4 @@ -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' @@ -6,6 +6,7 @@ const imageUrl = 'https://img.zcool.cn/community/01e75b5da933daa801209e1ffa4649. @Entry class ImageDemo extends Panel { build(rootView: Group): void { + let imageView: Image scroller(vlayout([ text({ text: "Image Demo", @@ -38,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'), 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/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.m b/iOS/Pod/Classes/Shader/DoricLayouts.m index 902b4478..c1e0adfb 100644 --- a/iOS/Pod/Classes/Shader/DoricLayouts.m +++ b/iOS/Pod/Classes/Shader/DoricLayouts.m @@ -176,8 +176,6 @@ - (void)layoutSubviews { @interface DoricStackView () -@property(nonatomic, assign) CGFloat contentWidth; -@property(nonatomic, assign) CGFloat contentHeight; @end @implementation DoricStackView From 459aab61e32714e525b604a3d5ebb10b5575a7be Mon Sep 17 00:00:00 2001 From: "pengfei.zhou" Date: Wed, 27 Nov 2019 19:57:43 +0800 Subject: [PATCH 35/35] feat:optimize refreshable --- .../Classes/Refresh/DoricRefreshableNode.m | 2 +- .../Classes/Refresh/DoricSwipeRefreshLayout.m | 26 +++++++++++++------ 2 files changed, 19 insertions(+), 9 deletions(-) diff --git a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m index 137dc5e2..bb17e312 100644 --- a/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m +++ b/iOS/Pod/Classes/Refresh/DoricRefreshableNode.m @@ -143,7 +143,7 @@ - (void)setProgressRotation:(CGFloat)rotation { - (void)setRefreshing:(NSNumber *)refreshable withPromise:(DoricPromise *)promise { self.view.refreshing = [refreshable boolValue]; - [promise resolve:nil]; + [promise resolve:nil]; } - (void)setRefreshable:(NSNumber *)refreshing withPromise:(DoricPromise *)promise { diff --git a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m index 7f126026..fc1c94fb 100644 --- a/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m +++ b/iOS/Pod/Classes/Refresh/DoricSwipeRefreshLayout.m @@ -76,23 +76,24 @@ - (void)setHeaderView:(UIView *)headerView { self.refreshable = YES; } - - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate { if (-scrollView.contentOffset.y >= self.headerView.height) { - self.refreshing = YES; + 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]; - //self.bounces = YES; - } else { - //self.bounces = NO; - //scrollView.contentOffset = CGPointMake(scrollView.contentOffset.x, 0); } } +- (void)setContentOffset:(CGPoint)contentOffset { + [super setContentOffset:contentOffset]; +} + - (void)setRefreshing:(BOOL)refreshing { if (_refreshing == refreshing) { return; @@ -101,21 +102,26 @@ - (void)setRefreshing:(BOOL)refreshing { if (self.onRefreshBlock) { self.onRefreshBlock(); } - [UIView animateWithDuration:0.3f + [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 { - [UIView animateWithDuration:0.3f + 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; } ]; } @@ -124,6 +130,10 @@ - (void)setRefreshing:(BOOL)refreshing { - (void)setRefreshable:(BOOL)refreshable { self.scrollEnabled = refreshable; + if (refreshable) { + self.contentOffset = (CGPoint) {0, 0}; + self.contentInset = UIEdgeInsetsMake(0, 0, 0, 0); + } } - (BOOL)refreshable {