mirror of
				https://github.com/commons-app/apps-android-commons.git
				synced 2025-10-26 20:33:53 +01:00 
			
		
		
		
	Merge branch 'main' into fix-multiupload
This commit is contained in:
		
						commit
						1e1e5d763f
					
				
					 42 changed files with 2035 additions and 2190 deletions
				
			
		|  | @ -212,8 +212,8 @@ android { | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         //applicationId 'fr.free.nrw.commons' |         //applicationId 'fr.free.nrw.commons' | ||||||
| 
 | 
 | ||||||
|         versionCode 1043 |         versionCode 1046 | ||||||
|         versionName '5.1.2' |         versionName '5.1.3' | ||||||
|         setProperty("archivesBaseName", "app-commons-v$versionName-" + getBranchName()) |         setProperty("archivesBaseName", "app-commons-v$versionName-" + getBranchName()) | ||||||
| 
 | 
 | ||||||
|         minSdkVersion 21 |         minSdkVersion 21 | ||||||
|  |  | ||||||
|  | @ -125,6 +125,19 @@ class Media constructor( | ||||||
|         categoriesHiddenStatus = categoriesHiddenStatus |         categoriesHiddenStatus = categoriesHiddenStatus | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns Author if it's not null or empty, otherwise | ||||||
|  |      * returns user | ||||||
|  |      * @return Author or User | ||||||
|  |      */ | ||||||
|  |     fun getAuthorOrUser(): String? { | ||||||
|  |         return if (!author.isNullOrEmpty()) { | ||||||
|  |             author | ||||||
|  |         } else{ | ||||||
|  |             user | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Gets media display title |      * Gets media display title | ||||||
|      * @return Media title |      * @return Media title | ||||||
|  |  | ||||||
|  | @ -98,14 +98,9 @@ class GridViewAdapter( | ||||||
|      */ |      */ | ||||||
|     @SuppressLint("StringFormatInvalid") |     @SuppressLint("StringFormatInvalid") | ||||||
|     private fun setUploaderView(item: Media, uploader: TextView) { |     private fun setUploaderView(item: Media, uploader: TextView) { | ||||||
|         if (!item.author.isNullOrEmpty()) { |         uploader.text = context.getString( | ||||||
|             uploader.visibility = View.VISIBLE |             R.string.image_uploaded_by, | ||||||
|             uploader.text = context.getString( |             item.getAuthorOrUser() | ||||||
|                 R.string.image_uploaded_by, |         ) | ||||||
|                 item.user |  | ||||||
|             ) |  | ||||||
|         } else { |  | ||||||
|             uploader.visibility = View.GONE |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -54,7 +54,7 @@ an upload might take a dozen seconds. */ | ||||||
|         this.contribution = contribution |         this.contribution = contribution | ||||||
|         this.position = position |         this.position = position | ||||||
|         binding.contributionTitle.text = contribution.media.mostRelevantCaption |         binding.contributionTitle.text = contribution.media.mostRelevantCaption | ||||||
|         binding.authorView.text = contribution.media.author |         binding.authorView.text = contribution.media.getAuthorOrUser() | ||||||
| 
 | 
 | ||||||
|         //Removes flicker of loading image. |         //Removes flicker of loading image. | ||||||
|         binding.contributionImage.hierarchy.fadeDuration = 0 |         binding.contributionImage.hierarchy.fadeDuration = 0 | ||||||
|  |  | ||||||
|  | @ -10,7 +10,7 @@ interface ContributionsContract { | ||||||
| 
 | 
 | ||||||
|     interface View { |     interface View { | ||||||
|         fun showMessage(localizedMessage: String) |         fun showMessage(localizedMessage: String) | ||||||
|         fun getContext(): Context |         fun getContext(): Context? | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     interface UserActionListener : BasePresenter<View> { |     interface UserActionListener : BasePresenter<View> { | ||||||
|  |  | ||||||
|  | @ -74,12 +74,9 @@ import java.util.Date | ||||||
| import javax.inject.Inject | import javax.inject.Inject | ||||||
| import javax.inject.Named | import javax.inject.Named | ||||||
| 
 | 
 | ||||||
| class ContributionsFragment | class ContributionsFragment : CommonsDaggerSupportFragment(), FragmentManager.OnBackStackChangedListener, | ||||||
| 
 |  | ||||||
|     : CommonsDaggerSupportFragment(), FragmentManager.OnBackStackChangedListener, |  | ||||||
|     LocationUpdateListener, MediaDetailProvider, SensorEventListener, ICampaignsView, |     LocationUpdateListener, MediaDetailProvider, SensorEventListener, ICampaignsView, | ||||||
|     ContributionsContract.View, |     ContributionsContract.View, ContributionsListFragment.Callback { | ||||||
|     ContributionsListFragment.Callback { |  | ||||||
|     @JvmField |     @JvmField | ||||||
|     @Inject |     @Inject | ||||||
|     @Named("default_preferences") |     @Named("default_preferences") | ||||||
|  | @ -307,9 +304,11 @@ class ContributionsFragment | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         notification.setOnClickListener { view: View? -> |         notification.setOnClickListener { view: View? -> | ||||||
|             startYourself( |             context?.let { | ||||||
|                 context, "unread" |                 startYourself( | ||||||
|             ) |                     it, "unread" | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -500,7 +499,7 @@ class ContributionsFragment | ||||||
| 
 | 
 | ||||||
|     private fun setUploadCount() { |     private fun setUploadCount() { | ||||||
|         okHttpJsonApiClient |         okHttpJsonApiClient | ||||||
|             ?.getUploadCount((activity as MainActivity).sessionManager?.currentAccount!!.name) |             ?.getUploadCount(sessionManager?.currentAccount!!.name) | ||||||
|             ?.subscribeOn(Schedulers.io()) |             ?.subscribeOn(Schedulers.io()) | ||||||
|             ?.observeOn(AndroidSchedulers.mainThread())?.let { |             ?.observeOn(AndroidSchedulers.mainThread())?.let { | ||||||
|                 compositeDisposable.add( |                 compositeDisposable.add( | ||||||
|  | @ -889,14 +888,16 @@ class ContributionsFragment | ||||||
|      * this function updates the number of contributions |      * this function updates the number of contributions | ||||||
|      */ |      */ | ||||||
|     fun upDateUploadCount() { |     fun upDateUploadCount() { | ||||||
|         WorkManager.getInstance(context) |         context?.let { | ||||||
|             .getWorkInfosForUniqueWorkLiveData(UploadWorker::class.java.simpleName).observe( |             WorkManager.getInstance(it) | ||||||
|                 viewLifecycleOwner |                 .getWorkInfosForUniqueWorkLiveData(UploadWorker::class.java.simpleName).observe( | ||||||
|             ) { workInfos: List<WorkInfo?> -> |                     viewLifecycleOwner | ||||||
|                 if (workInfos.size > 0) { |                 ) { workInfos: List<WorkInfo?> -> | ||||||
|                     setUploadCount() |                     if (workInfos.size > 0) { | ||||||
|  |                         setUploadCount() | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -953,7 +954,7 @@ class ContributionsFragment | ||||||
|                 Timber.d("Skipping re-upload for non-failed %s", contribution.toString()) |                 Timber.d("Skipping re-upload for non-failed %s", contribution.toString()) | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             showLongToast(context, R.string.this_function_needs_network_connection) |             context?.let { showLongToast(it, R.string.this_function_needs_network_connection) } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -80,9 +80,11 @@ class ContributionsPresenter @Inject internal constructor( | ||||||
|             .save(contribution) |             .save(contribution) | ||||||
|             .subscribeOn(ioThreadScheduler) |             .subscribeOn(ioThreadScheduler) | ||||||
|             .subscribe { |             .subscribe { | ||||||
|                 makeOneTimeWorkRequest( |                 view!!.getContext()?.applicationContext?.let { | ||||||
|                     view!!.getContext().applicationContext, ExistingWorkPolicy.KEEP |                     makeOneTimeWorkRequest( | ||||||
|                 ) |                         it, ExistingWorkPolicy.KEEP | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|             }) |             }) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -111,7 +111,7 @@ class DeleteHelper @Inject constructor( | ||||||
| 
 | 
 | ||||||
|         val userPageString = "\n{{subst:idw|${media.filename}}} ~~~~" |         val userPageString = "\n{{subst:idw|${media.filename}}} ~~~~" | ||||||
| 
 | 
 | ||||||
|         val creator = media.author |         val creator = media.getAuthorOrUser() | ||||||
|             ?: throw RuntimeException("Failed to nominate for deletion") |             ?: throw RuntimeException("Failed to nominate for deletion") | ||||||
| 
 | 
 | ||||||
|         return pageEditClient.prependEdit( |         return pageEditClient.prependEdit( | ||||||
|  |  | ||||||
|  | @ -63,9 +63,4 @@ abstract class CommonsDaggerSupportFragment : Fragment(), HasSupportFragmentInje | ||||||
| 
 | 
 | ||||||
|         return getInstance(activity.applicationContext) |         return getInstance(activity.applicationContext) | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     // Ensure getContext() returns a non-null Context |  | ||||||
|     override fun getContext(): Context { |  | ||||||
|         return super.getContext() ?: throw IllegalStateException("Context is null") |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -39,7 +39,7 @@ class MediaConverter | ||||||
|                 metadata.licenseShortName(), |                 metadata.licenseShortName(), | ||||||
|                 metadata.prefixedLicenseUrl, |                 metadata.prefixedLicenseUrl, | ||||||
|                 getAuthor(metadata), |                 getAuthor(metadata), | ||||||
|                 getAuthor(metadata), |                 imageInfo.getUser(), | ||||||
|                 MediaDataExtractorUtil.extractCategoriesFromList(metadata.categories()), |                 MediaDataExtractorUtil.extractCategoriesFromList(metadata.categories()), | ||||||
|                 metadata.latLng, |                 metadata.latLng, | ||||||
|                 entity.labels().mapValues { it.value.value() }, |                 entity.labels().mapValues { it.value.value() }, | ||||||
|  |  | ||||||
|  | @ -52,12 +52,7 @@ class SearchImagesViewHolder( | ||||||
|         binding.categoryImageView.setOnClickListener { onImageClicked(item.second) } |         binding.categoryImageView.setOnClickListener { onImageClicked(item.second) } | ||||||
|         binding.categoryImageTitle.text = media.mostRelevantCaption |         binding.categoryImageTitle.text = media.mostRelevantCaption | ||||||
|         binding.categoryImageView.setImageURI(media.thumbUrl) |         binding.categoryImageView.setImageURI(media.thumbUrl) | ||||||
|         if (media.author?.isNotEmpty() == true) { |         binding.categoryImageAuthor.text = | ||||||
|             binding.categoryImageAuthor.visibility = View.VISIBLE |             containerView.context.getString(R.string.image_uploaded_by, media.getAuthorOrUser()) | ||||||
|             binding.categoryImageAuthor.text = |  | ||||||
|                 containerView.context.getString(R.string.image_uploaded_by, media.user) |  | ||||||
|         } else { |  | ||||||
|             binding.categoryImageAuthor.visibility = View.GONE |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -328,7 +328,7 @@ public class MediaDetailPagerFragment extends CommonsDaggerSupportFragment imple | ||||||
|             .append("\n\n"); |             .append("\n\n"); | ||||||
| 
 | 
 | ||||||
|         builder.append("User that you want to report: ") |         builder.append("User that you want to report: ") | ||||||
|             .append(media.getAuthor()) |             .append(media.getUser()) | ||||||
|             .append("\n\n"); |             .append("\n\n"); | ||||||
| 
 | 
 | ||||||
|         if (sessionManager.getUserName() != null) { |         if (sessionManager.getUserName() != null) { | ||||||
|  | @ -423,7 +423,7 @@ public class MediaDetailPagerFragment extends CommonsDaggerSupportFragment imple | ||||||
|                     // Initialize bookmark object |                     // Initialize bookmark object | ||||||
|                     bookmark = new Bookmark( |                     bookmark = new Bookmark( | ||||||
|                             m.getFilename(), |                             m.getFilename(), | ||||||
|                             m.getAuthor(), |                             m.getAuthorOrUser(), | ||||||
|                             BookmarkPicturesContentProvider.uriForName(m.getFilename()) |                             BookmarkPicturesContentProvider.uriForName(m.getFilename()) | ||||||
|                     ); |                     ); | ||||||
|                     updateBookmarkState(menu.findItem(R.id.menu_bookmark_current_image)); |                     updateBookmarkState(menu.findItem(R.id.menu_bookmark_current_image)); | ||||||
|  |  | ||||||
|  | @ -272,7 +272,7 @@ class ZoomableActivity : BaseActivity() { | ||||||
|      * Handles down swipe action |      * Handles down swipe action | ||||||
|      */ |      */ | ||||||
|     private fun onDownSwiped() { |     private fun onDownSwiped() { | ||||||
|         if (binding.zoomable.zoomableController?.isIdentity == false) { |         if (!binding.zoomable.getZoomableController().isIdentity()) { | ||||||
|             return |             return | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -342,7 +342,7 @@ class ZoomableActivity : BaseActivity() { | ||||||
|      * Handles up swipe action |      * Handles up swipe action | ||||||
|      */ |      */ | ||||||
|     private fun onUpSwiped() { |     private fun onUpSwiped() { | ||||||
|         if (binding.zoomable.zoomableController?.isIdentity == false) { |         if (!binding.zoomable.getZoomableController().isIdentity()) { | ||||||
|             return |             return | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -415,7 +415,7 @@ class ZoomableActivity : BaseActivity() { | ||||||
|      * Handles right swipe action |      * Handles right swipe action | ||||||
|      */ |      */ | ||||||
|     private fun onRightSwiped(showAlreadyActionedImages: Boolean) { |     private fun onRightSwiped(showAlreadyActionedImages: Boolean) { | ||||||
|         if (binding.zoomable.zoomableController?.isIdentity == false) { |         if (!binding.zoomable.getZoomableController().isIdentity()) { | ||||||
|             return |             return | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -452,7 +452,7 @@ class ZoomableActivity : BaseActivity() { | ||||||
|      * Handles left swipe action |      * Handles left swipe action | ||||||
|      */ |      */ | ||||||
|     private fun onLeftSwiped(showAlreadyActionedImages: Boolean) { |     private fun onLeftSwiped(showAlreadyActionedImages: Boolean) { | ||||||
|         if (binding.zoomable.zoomableController?.isIdentity == false) { |         if (!binding.zoomable.getZoomableController().isIdentity()) { | ||||||
|             return |             return | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,256 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.gestures; |  | ||||||
| 
 |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Component that detects and tracks multiple pointers based on touch events. |  | ||||||
|  * |  | ||||||
|  * Each time a pointer gets pressed or released, the current gesture (if any) will end, and a new |  | ||||||
|  * one will be started (if there are still pressed pointers left). It is guaranteed that the number |  | ||||||
|  * of pointers within the single gesture will remain the same during the whole gesture. |  | ||||||
|  */ |  | ||||||
| public class MultiPointerGestureDetector { |  | ||||||
| 
 |  | ||||||
|     /** The listener for receiving notifications when gestures occur. */ |  | ||||||
|     public interface Listener { |  | ||||||
|         /** A callback called right before the gesture is about to start. */ |  | ||||||
|         public void onGestureBegin(MultiPointerGestureDetector detector); |  | ||||||
| 
 |  | ||||||
|         /** A callback called each time the gesture gets updated. */ |  | ||||||
|         public void onGestureUpdate(MultiPointerGestureDetector detector); |  | ||||||
| 
 |  | ||||||
|         /** A callback called right after the gesture has finished. */ |  | ||||||
|         public void onGestureEnd(MultiPointerGestureDetector detector); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static final int MAX_POINTERS = 2; |  | ||||||
| 
 |  | ||||||
|     private boolean mGestureInProgress; |  | ||||||
|     private int mPointerCount; |  | ||||||
|     private int mNewPointerCount; |  | ||||||
|     private final int mId[] = new int[MAX_POINTERS]; |  | ||||||
|     private final float mStartX[] = new float[MAX_POINTERS]; |  | ||||||
|     private final float mStartY[] = new float[MAX_POINTERS]; |  | ||||||
|     private final float mCurrentX[] = new float[MAX_POINTERS]; |  | ||||||
|     private final float mCurrentY[] = new float[MAX_POINTERS]; |  | ||||||
| 
 |  | ||||||
|     private Listener mListener = null; |  | ||||||
| 
 |  | ||||||
|     public MultiPointerGestureDetector() { |  | ||||||
|         reset(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Factory method that creates a new instance of MultiPointerGestureDetector */ |  | ||||||
|     public static MultiPointerGestureDetector newInstance() { |  | ||||||
|         return new MultiPointerGestureDetector(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets the listener. |  | ||||||
|      * |  | ||||||
|      * @param listener listener to set |  | ||||||
|      */ |  | ||||||
|     public void setListener(Listener listener) { |  | ||||||
|         mListener = listener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Resets the component to the initial state. */ |  | ||||||
|     public void reset() { |  | ||||||
|         mGestureInProgress = false; |  | ||||||
|         mPointerCount = 0; |  | ||||||
|         for (int i = 0; i < MAX_POINTERS; i++) { |  | ||||||
|             mId[i] = MotionEvent.INVALID_POINTER_ID; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method can be overridden in order to perform threshold check or something similar. |  | ||||||
|      * |  | ||||||
|      * @return whether or not to start a new gesture |  | ||||||
|      */ |  | ||||||
|     protected boolean shouldStartGesture() { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Starts a new gesture and calls the listener just before starting it. */ |  | ||||||
|     private void startGesture() { |  | ||||||
|         if (!mGestureInProgress) { |  | ||||||
|             if (mListener != null) { |  | ||||||
|                 mListener.onGestureBegin(this); |  | ||||||
|             } |  | ||||||
|             mGestureInProgress = true; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Stops the current gesture and calls the listener right after stopping it. */ |  | ||||||
|     private void stopGesture() { |  | ||||||
|         if (mGestureInProgress) { |  | ||||||
|             mGestureInProgress = false; |  | ||||||
|             if (mListener != null) { |  | ||||||
|                 mListener.onGestureEnd(this); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the index of the i-th pressed pointer. Normally, the index will be equal to i, except in |  | ||||||
|      * the case when the pointer is released. |  | ||||||
|      * |  | ||||||
|      * @return index of the specified pointer or -1 if not found (i.e. not enough pointers are down) |  | ||||||
|      */ |  | ||||||
|     private int getPressedPointerIndex(MotionEvent event, int i) { |  | ||||||
|         final int count = event.getPointerCount(); |  | ||||||
|         final int action = event.getActionMasked(); |  | ||||||
|         final int index = event.getActionIndex(); |  | ||||||
|         if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_POINTER_UP) { |  | ||||||
|             if (i >= index) { |  | ||||||
|                 i++; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return (i < count) ? i : -1; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the number of pressed pointers (fingers down). */ |  | ||||||
|     private static int getPressedPointerCount(MotionEvent event) { |  | ||||||
|         int count = event.getPointerCount(); |  | ||||||
|         int action = event.getActionMasked(); |  | ||||||
|         if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_POINTER_UP) { |  | ||||||
|             count--; |  | ||||||
|         } |  | ||||||
|         return count; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void updatePointersOnTap(MotionEvent event) { |  | ||||||
|         mPointerCount = 0; |  | ||||||
|         for (int i = 0; i < MAX_POINTERS; i++) { |  | ||||||
|             int index = getPressedPointerIndex(event, i); |  | ||||||
|             if (index == -1) { |  | ||||||
|                 mId[i] = MotionEvent.INVALID_POINTER_ID; |  | ||||||
|             } else { |  | ||||||
|                 mId[i] = event.getPointerId(index); |  | ||||||
|                 mCurrentX[i] = mStartX[i] = event.getX(index); |  | ||||||
|                 mCurrentY[i] = mStartY[i] = event.getY(index); |  | ||||||
|                 mPointerCount++; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void updatePointersOnMove(MotionEvent event) { |  | ||||||
|         for (int i = 0; i < MAX_POINTERS; i++) { |  | ||||||
|             int index = event.findPointerIndex(mId[i]); |  | ||||||
|             if (index != -1) { |  | ||||||
|                 mCurrentX[i] = event.getX(index); |  | ||||||
|                 mCurrentY[i] = event.getY(index); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the given motion event. |  | ||||||
|      * |  | ||||||
|      * @param event event to handle |  | ||||||
|      * @return whether or not the event was handled |  | ||||||
|      */ |  | ||||||
|     public boolean onTouchEvent(final MotionEvent event) { |  | ||||||
|         switch (event.getActionMasked()) { |  | ||||||
|             case MotionEvent.ACTION_MOVE: |  | ||||||
|             { |  | ||||||
|                 // update pointers |  | ||||||
|                 updatePointersOnMove(event); |  | ||||||
|                 // start a new gesture if not already started |  | ||||||
|                 if (!mGestureInProgress && mPointerCount > 0 && shouldStartGesture()) { |  | ||||||
|                     startGesture(); |  | ||||||
|                 } |  | ||||||
|                 // notify listener |  | ||||||
|                 if (mGestureInProgress && mListener != null) { |  | ||||||
|                     mListener.onGestureUpdate(this); |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             case MotionEvent.ACTION_DOWN: |  | ||||||
|             case MotionEvent.ACTION_POINTER_DOWN: |  | ||||||
|             case MotionEvent.ACTION_POINTER_UP: |  | ||||||
|             case MotionEvent.ACTION_UP: |  | ||||||
|             { |  | ||||||
|                 // restart gesture whenever the number of pointers changes |  | ||||||
|                 mNewPointerCount = getPressedPointerCount(event); |  | ||||||
|                 stopGesture(); |  | ||||||
|                 updatePointersOnTap(event); |  | ||||||
|                 if (mPointerCount > 0 && shouldStartGesture()) { |  | ||||||
|                     startGesture(); |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             case MotionEvent.ACTION_CANCEL: |  | ||||||
|             { |  | ||||||
|                 mNewPointerCount = 0; |  | ||||||
|                 stopGesture(); |  | ||||||
|                 reset(); |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Restarts the current gesture (if any). */ |  | ||||||
|     public void restartGesture() { |  | ||||||
|         if (!mGestureInProgress) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         stopGesture(); |  | ||||||
|         for (int i = 0; i < MAX_POINTERS; i++) { |  | ||||||
|             mStartX[i] = mCurrentX[i]; |  | ||||||
|             mStartY[i] = mCurrentY[i]; |  | ||||||
|         } |  | ||||||
|         startGesture(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether there is a gesture in progress */ |  | ||||||
|     public boolean isGestureInProgress() { |  | ||||||
|         return mGestureInProgress; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the number of pointers after the current gesture */ |  | ||||||
|     public int getNewPointerCount() { |  | ||||||
|         return mNewPointerCount; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the number of pointers in the current gesture */ |  | ||||||
|     public int getPointerCount() { |  | ||||||
|         return mPointerCount; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the start X coordinates for the all pointers Mutable array is exposed for performance |  | ||||||
|      * reasons and is not to be modified by the callers. |  | ||||||
|      */ |  | ||||||
|     public float[] getStartX() { |  | ||||||
|         return mStartX; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the start Y coordinates for the all pointers Mutable array is exposed for performance |  | ||||||
|      * reasons and is not to be modified by the callers. |  | ||||||
|      */ |  | ||||||
|     public float[] getStartY() { |  | ||||||
|         return mStartY; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the current X coordinates for the all pointers Mutable array is exposed for performance |  | ||||||
|      * reasons and is not to be modified by the callers. |  | ||||||
|      */ |  | ||||||
|     public float[] getCurrentX() { |  | ||||||
|         return mCurrentX; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the current Y coordinates for the all pointers Mutable array is exposed for performance |  | ||||||
|      * reasons and is not to be modified by the callers. |  | ||||||
|      */ |  | ||||||
|     public float[] getCurrentY() { |  | ||||||
|         return mCurrentY; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,252 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.gestures | ||||||
|  | 
 | ||||||
|  | import android.view.MotionEvent | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Component that detects and tracks multiple pointers based on touch events. | ||||||
|  |  * | ||||||
|  |  * Each time a pointer gets pressed or released, the current gesture (if any) will end, and a new | ||||||
|  |  * one will be started (if there are still pressed pointers left). It is guaranteed that the number | ||||||
|  |  * of pointers within the single gesture will remain the same during the whole gesture. | ||||||
|  |  */ | ||||||
|  | open class MultiPointerGestureDetector { | ||||||
|  | 
 | ||||||
|  |     /** The listener for receiving notifications when gestures occur. */ | ||||||
|  |     interface Listener { | ||||||
|  |         /** A callback called right before the gesture is about to start. */ | ||||||
|  |         fun onGestureBegin(detector: MultiPointerGestureDetector) | ||||||
|  | 
 | ||||||
|  |         /** A callback called each time the gesture gets updated. */ | ||||||
|  |         fun onGestureUpdate(detector: MultiPointerGestureDetector) | ||||||
|  | 
 | ||||||
|  |         /** A callback called right after the gesture has finished. */ | ||||||
|  |         fun onGestureEnd(detector: MultiPointerGestureDetector) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private const val MAX_POINTERS = 2 | ||||||
|  | 
 | ||||||
|  |         /** Factory method that creates a new instance of MultiPointerGestureDetector */ | ||||||
|  |         fun newInstance(): MultiPointerGestureDetector { | ||||||
|  |             return MultiPointerGestureDetector() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private var mGestureInProgress = false | ||||||
|  |     private var mPointerCount = 0 | ||||||
|  |     private var mNewPointerCount = 0 | ||||||
|  |     private val mId = IntArray(MAX_POINTERS) { MotionEvent.INVALID_POINTER_ID } | ||||||
|  |     private val mStartX = FloatArray(MAX_POINTERS) | ||||||
|  |     private val mStartY = FloatArray(MAX_POINTERS) | ||||||
|  |     private val mCurrentX = FloatArray(MAX_POINTERS) | ||||||
|  |     private val mCurrentY = FloatArray(MAX_POINTERS) | ||||||
|  | 
 | ||||||
|  |     private var mListener: Listener? = null | ||||||
|  | 
 | ||||||
|  |     init { | ||||||
|  |         reset() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the listener. | ||||||
|  |      * | ||||||
|  |      * @param listener listener to set | ||||||
|  |      */ | ||||||
|  |     fun setListener(listener: Listener?) { | ||||||
|  |         mListener = listener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Resets the component to the initial state. */ | ||||||
|  |     fun reset() { | ||||||
|  |         mGestureInProgress = false | ||||||
|  |         mPointerCount = 0 | ||||||
|  |         for (i in 0 until MAX_POINTERS) { | ||||||
|  |             mId[i] = MotionEvent.INVALID_POINTER_ID | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This method can be overridden in order to perform threshold check or something similar. | ||||||
|  |      * | ||||||
|  |      * @return whether or not to start a new gesture | ||||||
|  |      */ | ||||||
|  |     protected open fun shouldStartGesture(): Boolean { | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Starts a new gesture and calls the listener just before starting it. */ | ||||||
|  |     private fun startGesture() { | ||||||
|  |         if (!mGestureInProgress) { | ||||||
|  |             mListener?.onGestureBegin(this) | ||||||
|  |             mGestureInProgress = true | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Stops the current gesture and calls the listener right after stopping it. */ | ||||||
|  |     private fun stopGesture() { | ||||||
|  |         if (mGestureInProgress) { | ||||||
|  |             mGestureInProgress = false | ||||||
|  |             mListener?.onGestureEnd(this) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the index of the i-th pressed pointer. Normally, the index will be equal to i, except in | ||||||
|  |      * the case when the pointer is released. | ||||||
|  |      * | ||||||
|  |      * @return index of the specified pointer or -1 if not found (i.e. not enough pointers are down) | ||||||
|  |      */ | ||||||
|  |     private fun getPressedPointerIndex(event: MotionEvent, i: Int): Int { | ||||||
|  |         val count = event.pointerCount | ||||||
|  |         val action = event.actionMasked | ||||||
|  |         val index = event.actionIndex | ||||||
|  |         var adjustedIndex = i | ||||||
|  | 
 | ||||||
|  |         if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_POINTER_UP) { | ||||||
|  |             if (adjustedIndex >= index) { | ||||||
|  |                 adjustedIndex++ | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return if (adjustedIndex < count) adjustedIndex else -1 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the number of pressed pointers (fingers down). */ | ||||||
|  |     private fun getPressedPointerCount(event: MotionEvent): Int { | ||||||
|  |         var count = event.pointerCount | ||||||
|  |         val action = event.actionMasked | ||||||
|  |         if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_POINTER_UP) { | ||||||
|  |             count-- | ||||||
|  |         } | ||||||
|  |         return count | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun updatePointersOnTap(event: MotionEvent) { | ||||||
|  |         mPointerCount = 0 | ||||||
|  |         for (i in 0 until MAX_POINTERS) { | ||||||
|  |             val index = getPressedPointerIndex(event, i) | ||||||
|  |             if (index == -1) { | ||||||
|  |                 mId[i] = MotionEvent.INVALID_POINTER_ID | ||||||
|  |             } else { | ||||||
|  |                 mId[i] = event.getPointerId(index) | ||||||
|  |                 mCurrentX[i] = event.getX(index) | ||||||
|  |                 mStartX[i] = mCurrentX[i] | ||||||
|  |                 mCurrentY[i] = event.getY(index) | ||||||
|  |                 mStartY[i] = mCurrentY[i] | ||||||
|  |                 mPointerCount++ | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun updatePointersOnMove(event: MotionEvent) { | ||||||
|  |         for (i in 0 until MAX_POINTERS) { | ||||||
|  |             val index = event.findPointerIndex(mId[i]) | ||||||
|  |             if (index != -1) { | ||||||
|  |                 mCurrentX[i] = event.getX(index) | ||||||
|  |                 mCurrentY[i] = event.getY(index) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the given motion event. | ||||||
|  |      * | ||||||
|  |      * @param event event to handle | ||||||
|  |      * @return whether or not the event was handled | ||||||
|  |      */ | ||||||
|  |     fun onTouchEvent(event: MotionEvent): Boolean { | ||||||
|  |         when (event.actionMasked) { | ||||||
|  |             MotionEvent.ACTION_MOVE -> { | ||||||
|  |                 // update pointers | ||||||
|  |                 updatePointersOnMove(event) | ||||||
|  |                 // start a new gesture if not already started | ||||||
|  |                 if (!mGestureInProgress && mPointerCount > 0 && shouldStartGesture()) { | ||||||
|  |                     startGesture() | ||||||
|  |                 } | ||||||
|  |                 // notify listener | ||||||
|  |                 if (mGestureInProgress) { | ||||||
|  |                     mListener?.onGestureUpdate(this) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             MotionEvent.ACTION_DOWN, | ||||||
|  |             MotionEvent.ACTION_POINTER_DOWN, | ||||||
|  |             MotionEvent.ACTION_POINTER_UP, | ||||||
|  |             MotionEvent.ACTION_UP -> { | ||||||
|  |                 // restart gesture whenever the number of pointers changes | ||||||
|  |                 mNewPointerCount = getPressedPointerCount(event) | ||||||
|  |                 stopGesture() | ||||||
|  |                 updatePointersOnTap(event) | ||||||
|  |                 if (mPointerCount > 0 && shouldStartGesture()) { | ||||||
|  |                     startGesture() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             MotionEvent.ACTION_CANCEL -> { | ||||||
|  |                 mNewPointerCount = 0 | ||||||
|  |                 stopGesture() | ||||||
|  |                 reset() | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Restarts the current gesture (if any). */ | ||||||
|  |     fun restartGesture() { | ||||||
|  |         if (!mGestureInProgress) { | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |         stopGesture() | ||||||
|  |         for (i in 0 until MAX_POINTERS) { | ||||||
|  |             mStartX[i] = mCurrentX[i] | ||||||
|  |             mStartY[i] = mCurrentY[i] | ||||||
|  |         } | ||||||
|  |         startGesture() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether there is a gesture in progress */ | ||||||
|  |     fun isGestureInProgress(): Boolean { | ||||||
|  |         return mGestureInProgress | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the number of pointers after the current gesture */ | ||||||
|  |     fun getNewPointerCount(): Int { | ||||||
|  |         return mNewPointerCount | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the number of pointers in the current gesture */ | ||||||
|  |     fun getPointerCount(): Int { | ||||||
|  |         return mPointerCount | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the start X coordinates for all pointers Mutable array is exposed for performance | ||||||
|  |      * reasons and is not to be modified by the callers. | ||||||
|  |      */ | ||||||
|  |     fun getStartX(): FloatArray { | ||||||
|  |         return mStartX | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the start Y coordinates for all pointers Mutable array is exposed for performance | ||||||
|  |      * reasons and is not to be modified by the callers. | ||||||
|  |      */ | ||||||
|  |     fun getStartY(): FloatArray { | ||||||
|  |         return mStartY | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the current X coordinates for all pointers Mutable array is exposed for performance | ||||||
|  |      * reasons and is not to be modified by the callers. | ||||||
|  |      */ | ||||||
|  |     fun getCurrentX(): FloatArray { | ||||||
|  |         return mCurrentX | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the current Y coordinates for all pointers Mutable array is exposed for performance | ||||||
|  |      * reasons and is not to be modified by the callers. | ||||||
|  |      */ | ||||||
|  |     fun getCurrentY(): FloatArray { | ||||||
|  |         return mCurrentY | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,164 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.gestures; |  | ||||||
| 
 |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Component that detects translation, scale and rotation based on touch events. |  | ||||||
|  * |  | ||||||
|  * This class notifies its listeners whenever a gesture begins, updates or ends. The instance of |  | ||||||
|  * this detector is passed to the listeners, so it can be queried for pivot, translation, scale or |  | ||||||
|  * rotation. |  | ||||||
|  */ |  | ||||||
| public class TransformGestureDetector implements MultiPointerGestureDetector.Listener { |  | ||||||
| 
 |  | ||||||
|     /** The listener for receiving notifications when gestures occur. */ |  | ||||||
|     public interface Listener { |  | ||||||
|         /** A callback called right before the gesture is about to start. */ |  | ||||||
|         public void onGestureBegin(TransformGestureDetector detector); |  | ||||||
| 
 |  | ||||||
|         /** A callback called each time the gesture gets updated. */ |  | ||||||
|         public void onGestureUpdate(TransformGestureDetector detector); |  | ||||||
| 
 |  | ||||||
|         /** A callback called right after the gesture has finished. */ |  | ||||||
|         public void onGestureEnd(TransformGestureDetector detector); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private final MultiPointerGestureDetector mDetector; |  | ||||||
| 
 |  | ||||||
|     private Listener mListener = null; |  | ||||||
| 
 |  | ||||||
|     public TransformGestureDetector(MultiPointerGestureDetector multiPointerGestureDetector) { |  | ||||||
|         mDetector = multiPointerGestureDetector; |  | ||||||
|         mDetector.setListener(this); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Factory method that creates a new instance of TransformGestureDetector */ |  | ||||||
|     public static TransformGestureDetector newInstance() { |  | ||||||
|         return new TransformGestureDetector(MultiPointerGestureDetector.newInstance()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets the listener. |  | ||||||
|      * |  | ||||||
|      * @param listener listener to set |  | ||||||
|      */ |  | ||||||
|     public void setListener(Listener listener) { |  | ||||||
|         mListener = listener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Resets the component to the initial state. */ |  | ||||||
|     public void reset() { |  | ||||||
|         mDetector.reset(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the given motion event. |  | ||||||
|      * |  | ||||||
|      * @param event event to handle |  | ||||||
|      * @return whether or not the event was handled |  | ||||||
|      */ |  | ||||||
|     public boolean onTouchEvent(final MotionEvent event) { |  | ||||||
|         return mDetector.onTouchEvent(event); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureBegin(MultiPointerGestureDetector detector) { |  | ||||||
|         if (mListener != null) { |  | ||||||
|             mListener.onGestureBegin(this); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureUpdate(MultiPointerGestureDetector detector) { |  | ||||||
|         if (mListener != null) { |  | ||||||
|             mListener.onGestureUpdate(this); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureEnd(MultiPointerGestureDetector detector) { |  | ||||||
|         if (mListener != null) { |  | ||||||
|             mListener.onGestureEnd(this); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private float calcAverage(float[] arr, int len) { |  | ||||||
|         float sum = 0; |  | ||||||
|         for (int i = 0; i < len; i++) { |  | ||||||
|             sum += arr[i]; |  | ||||||
|         } |  | ||||||
|         return (len > 0) ? sum / len : 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Restarts the current gesture (if any). */ |  | ||||||
|     public void restartGesture() { |  | ||||||
|         mDetector.restartGesture(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether there is a gesture in progress */ |  | ||||||
|     public boolean isGestureInProgress() { |  | ||||||
|         return mDetector.isGestureInProgress(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the number of pointers after the current gesture */ |  | ||||||
|     public int getNewPointerCount() { |  | ||||||
|         return mDetector.getNewPointerCount(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the number of pointers in the current gesture */ |  | ||||||
|     public int getPointerCount() { |  | ||||||
|         return mDetector.getPointerCount(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the X coordinate of the pivot point */ |  | ||||||
|     public float getPivotX() { |  | ||||||
|         return calcAverage(mDetector.getStartX(), mDetector.getPointerCount()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the Y coordinate of the pivot point */ |  | ||||||
|     public float getPivotY() { |  | ||||||
|         return calcAverage(mDetector.getStartY(), mDetector.getPointerCount()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the X component of the translation */ |  | ||||||
|     public float getTranslationX() { |  | ||||||
|         return calcAverage(mDetector.getCurrentX(), mDetector.getPointerCount()) |  | ||||||
|                 - calcAverage(mDetector.getStartX(), mDetector.getPointerCount()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the Y component of the translation */ |  | ||||||
|     public float getTranslationY() { |  | ||||||
|         return calcAverage(mDetector.getCurrentY(), mDetector.getPointerCount()) |  | ||||||
|                 - calcAverage(mDetector.getStartY(), mDetector.getPointerCount()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the scale */ |  | ||||||
|     public float getScale() { |  | ||||||
|         if (mDetector.getPointerCount() < 2) { |  | ||||||
|             return 1; |  | ||||||
|         } else { |  | ||||||
|             float startDeltaX = mDetector.getStartX()[1] - mDetector.getStartX()[0]; |  | ||||||
|             float startDeltaY = mDetector.getStartY()[1] - mDetector.getStartY()[0]; |  | ||||||
|             float currentDeltaX = mDetector.getCurrentX()[1] - mDetector.getCurrentX()[0]; |  | ||||||
|             float currentDeltaY = mDetector.getCurrentY()[1] - mDetector.getCurrentY()[0]; |  | ||||||
|             float startDist = (float) Math.hypot(startDeltaX, startDeltaY); |  | ||||||
|             float currentDist = (float) Math.hypot(currentDeltaX, currentDeltaY); |  | ||||||
|             return currentDist / startDist; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the rotation in radians */ |  | ||||||
|     public float getRotation() { |  | ||||||
|         if (mDetector.getPointerCount() < 2) { |  | ||||||
|             return 0; |  | ||||||
|         } else { |  | ||||||
|             float startDeltaX = mDetector.getStartX()[1] - mDetector.getStartX()[0]; |  | ||||||
|             float startDeltaY = mDetector.getStartY()[1] - mDetector.getStartY()[0]; |  | ||||||
|             float currentDeltaX = mDetector.getCurrentX()[1] - mDetector.getCurrentX()[0]; |  | ||||||
|             float currentDeltaY = mDetector.getCurrentY()[1] - mDetector.getCurrentY()[0]; |  | ||||||
|             float startAngle = (float) Math.atan2(startDeltaY, startDeltaX); |  | ||||||
|             float currentAngle = (float) Math.atan2(currentDeltaY, currentDeltaX); |  | ||||||
|             return currentAngle - startAngle; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,154 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.gestures | ||||||
|  | 
 | ||||||
|  | import android.view.MotionEvent | ||||||
|  | import kotlin.math.atan2 | ||||||
|  | import kotlin.math.hypot | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Component that detects translation, scale and rotation based on touch events. | ||||||
|  |  * | ||||||
|  |  * This class notifies its listeners whenever a gesture begins, updates or ends. The instance of | ||||||
|  |  * this detector is passed to the listeners, so it can be queried for pivot, translation, scale or | ||||||
|  |  * rotation. | ||||||
|  |  */ | ||||||
|  | class TransformGestureDetector(private val mDetector: MultiPointerGestureDetector) : | ||||||
|  |     MultiPointerGestureDetector.Listener { | ||||||
|  | 
 | ||||||
|  |     /** The listener for receiving notifications when gestures occur. */ | ||||||
|  |     interface Listener { | ||||||
|  |         /** A callback called right before the gesture is about to start. */ | ||||||
|  |         fun onGestureBegin(detector: TransformGestureDetector) | ||||||
|  | 
 | ||||||
|  |         /** A callback called each time the gesture gets updated. */ | ||||||
|  |         fun onGestureUpdate(detector: TransformGestureDetector) | ||||||
|  | 
 | ||||||
|  |         /** A callback called right after the gesture has finished. */ | ||||||
|  |         fun onGestureEnd(detector: TransformGestureDetector) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private var mListener: Listener? = null | ||||||
|  | 
 | ||||||
|  |     init { | ||||||
|  |         mDetector.setListener(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Factory method that creates a new instance of TransformGestureDetector */ | ||||||
|  |     companion object { | ||||||
|  |         fun newInstance(): TransformGestureDetector { | ||||||
|  |             return TransformGestureDetector(MultiPointerGestureDetector.newInstance()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the listener. | ||||||
|  |      * | ||||||
|  |      * @param listener listener to set | ||||||
|  |      */ | ||||||
|  |     fun setListener(listener: Listener?) { | ||||||
|  |         mListener = listener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Resets the component to the initial state. */ | ||||||
|  |     fun reset() { | ||||||
|  |         mDetector.reset() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the given motion event. | ||||||
|  |      * | ||||||
|  |      * @param event event to handle | ||||||
|  |      * @return whether or not the event was handled | ||||||
|  |      */ | ||||||
|  |     fun onTouchEvent(event: MotionEvent): Boolean { | ||||||
|  |         return mDetector.onTouchEvent(event) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onGestureBegin(detector: MultiPointerGestureDetector) { | ||||||
|  |         mListener?.onGestureBegin(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onGestureUpdate(detector: MultiPointerGestureDetector) { | ||||||
|  |         mListener?.onGestureUpdate(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onGestureEnd(detector: MultiPointerGestureDetector) { | ||||||
|  |         mListener?.onGestureEnd(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun calcAverage(arr: FloatArray, len: Int): Float { | ||||||
|  |         val sum = arr.take(len).sum() | ||||||
|  |         return if (len > 0) sum / len else 0f | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Restarts the current gesture (if any). */ | ||||||
|  |     fun restartGesture() { | ||||||
|  |         mDetector.restartGesture() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether there is a gesture in progress */ | ||||||
|  |     fun isGestureInProgress(): Boolean { | ||||||
|  |         return mDetector.isGestureInProgress() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the number of pointers after the current gesture */ | ||||||
|  |     fun getNewPointerCount(): Int { | ||||||
|  |         return mDetector.getNewPointerCount() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the number of pointers in the current gesture */ | ||||||
|  |     fun getPointerCount(): Int { | ||||||
|  |         return mDetector.getPointerCount() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the X coordinate of the pivot point */ | ||||||
|  |     fun getPivotX(): Float { | ||||||
|  |         return calcAverage(mDetector.getStartX(), mDetector.getPointerCount()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the Y coordinate of the pivot point */ | ||||||
|  |     fun getPivotY(): Float { | ||||||
|  |         return calcAverage(mDetector.getStartY(), mDetector.getPointerCount()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the X component of the translation */ | ||||||
|  |     fun getTranslationX(): Float { | ||||||
|  |         return calcAverage(mDetector.getCurrentX(), mDetector.getPointerCount()) - | ||||||
|  |                 calcAverage(mDetector.getStartX(), mDetector.getPointerCount()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the Y component of the translation */ | ||||||
|  |     fun getTranslationY(): Float { | ||||||
|  |         return calcAverage(mDetector.getCurrentY(), mDetector.getPointerCount()) - | ||||||
|  |                 calcAverage(mDetector.getStartY(), mDetector.getPointerCount()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the scale */ | ||||||
|  |     fun getScale(): Float { | ||||||
|  |         return if (mDetector.getPointerCount() < 2) { | ||||||
|  |             1f | ||||||
|  |         } else { | ||||||
|  |             val startDeltaX = mDetector.getStartX()[1] - mDetector.getStartX()[0] | ||||||
|  |             val startDeltaY = mDetector.getStartY()[1] - mDetector.getStartY()[0] | ||||||
|  |             val currentDeltaX = mDetector.getCurrentX()[1] - mDetector.getCurrentX()[0] | ||||||
|  |             val currentDeltaY = mDetector.getCurrentY()[1] - mDetector.getCurrentY()[0] | ||||||
|  |             val startDist = hypot(startDeltaX, startDeltaY) | ||||||
|  |             val currentDist = hypot(currentDeltaX, currentDeltaY) | ||||||
|  |             currentDist / startDist | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the rotation in radians */ | ||||||
|  |     fun getRotation(): Float { | ||||||
|  |         return if (mDetector.getPointerCount() < 2) { | ||||||
|  |             0f | ||||||
|  |         } else { | ||||||
|  |             val startDeltaX = mDetector.getStartX()[1] - mDetector.getStartX()[0] | ||||||
|  |             val startDeltaY = mDetector.getStartY()[1] - mDetector.getStartY()[0] | ||||||
|  |             val currentDeltaX = mDetector.getCurrentX()[1] - mDetector.getCurrentX()[0] | ||||||
|  |             val currentDeltaY = mDetector.getCurrentY()[1] - mDetector.getCurrentY()[0] | ||||||
|  |             val startAngle = atan2(startDeltaY, startDeltaX) | ||||||
|  |             val currentAngle = atan2(currentDeltaY, currentDeltaX) | ||||||
|  |             currentAngle - startAngle | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,160 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.graphics.Matrix; |  | ||||||
| import android.graphics.PointF; |  | ||||||
| import com.facebook.common.logging.FLog; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import fr.free.nrw.commons.media.zoomControllers.gestures.TransformGestureDetector; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Abstract class for ZoomableController that adds animation capabilities to |  | ||||||
|  * DefaultZoomableController. |  | ||||||
|  */ |  | ||||||
| public abstract class AbstractAnimatedZoomableController extends DefaultZoomableController { |  | ||||||
| 
 |  | ||||||
|     private boolean mIsAnimating; |  | ||||||
|     private final float[] mStartValues = new float[9]; |  | ||||||
|     private final float[] mStopValues = new float[9]; |  | ||||||
|     private final float[] mCurrentValues = new float[9]; |  | ||||||
|     private final Matrix mNewTransform = new Matrix(); |  | ||||||
|     private final Matrix mWorkingTransform = new Matrix(); |  | ||||||
| 
 |  | ||||||
|     public AbstractAnimatedZoomableController(TransformGestureDetector transformGestureDetector) { |  | ||||||
|         super(transformGestureDetector); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void reset() { |  | ||||||
|         FLog.v(getLogTag(), "reset"); |  | ||||||
|         stopAnimation(); |  | ||||||
|         mWorkingTransform.reset(); |  | ||||||
|         mNewTransform.reset(); |  | ||||||
|         super.reset(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Returns true if the zoomable transform is identity matrix, and the controller is idle. */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isIdentity() { |  | ||||||
|         return !isAnimating() && super.isIdentity(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Zooms to the desired scale and positions the image so that the given image point corresponds to |  | ||||||
|      * the given view point. |  | ||||||
|      * |  | ||||||
|      * <p>If this method is called while an animation or gesture is already in progress, the current |  | ||||||
|      * animation or gesture will be stopped first. |  | ||||||
|      * |  | ||||||
|      * @param scale desired scale, will be limited to {min, max} scale factor |  | ||||||
|      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) |  | ||||||
|      * @param viewPoint 2D point in view's absolute coordinate system |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void zoomToPoint(float scale, PointF imagePoint, PointF viewPoint) { |  | ||||||
|         zoomToPoint(scale, imagePoint, viewPoint, LIMIT_ALL, 0, null); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Zooms to the desired scale and positions the image so that the given image point corresponds to |  | ||||||
|      * the given view point. |  | ||||||
|      * |  | ||||||
|      * <p>If this method is called while an animation or gesture is already in progress, the current |  | ||||||
|      * animation or gesture will be stopped first. |  | ||||||
|      * |  | ||||||
|      * @param scale desired scale, will be limited to {min, max} scale factor |  | ||||||
|      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) |  | ||||||
|      * @param viewPoint 2D point in view's absolute coordinate system |  | ||||||
|      * @param limitFlags whether to limit translation and/or scale. |  | ||||||
|      * @param durationMs length of animation of the zoom, or 0 if no animation desired |  | ||||||
|      * @param onAnimationComplete code to run when the animation completes. Ignored if durationMs=0 |  | ||||||
|      */ |  | ||||||
|     public void zoomToPoint( |  | ||||||
|             float scale, |  | ||||||
|             PointF imagePoint, |  | ||||||
|             PointF viewPoint, |  | ||||||
|             @LimitFlag int limitFlags, |  | ||||||
|             long durationMs, |  | ||||||
|             @Nullable Runnable onAnimationComplete) { |  | ||||||
|         FLog.v(getLogTag(), "zoomToPoint: duration %d ms", durationMs); |  | ||||||
|         calculateZoomToPointTransform(mNewTransform, scale, imagePoint, viewPoint, limitFlags); |  | ||||||
|         setTransform(mNewTransform, durationMs, onAnimationComplete); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets a new zoomable transformation and animates to it if desired. |  | ||||||
|      * |  | ||||||
|      * <p>If this method is called while an animation or gesture is already in progress, the current |  | ||||||
|      * animation or gesture will be stopped first. |  | ||||||
|      * |  | ||||||
|      * @param newTransform new transform to make active |  | ||||||
|      * @param durationMs duration of the animation, or 0 to not animate |  | ||||||
|      * @param onAnimationComplete code to run when the animation completes. Ignored if durationMs=0 |  | ||||||
|      */ |  | ||||||
|     public void setTransform( |  | ||||||
|             Matrix newTransform, long durationMs, @Nullable Runnable onAnimationComplete) { |  | ||||||
|         FLog.v(getLogTag(), "setTransform: duration %d ms", durationMs); |  | ||||||
|         if (durationMs <= 0) { |  | ||||||
|             setTransformImmediate(newTransform); |  | ||||||
|         } else { |  | ||||||
|             setTransformAnimated(newTransform, durationMs, onAnimationComplete); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void setTransformImmediate(final Matrix newTransform) { |  | ||||||
|         FLog.v(getLogTag(), "setTransformImmediate"); |  | ||||||
|         stopAnimation(); |  | ||||||
|         mWorkingTransform.set(newTransform); |  | ||||||
|         super.setTransform(newTransform); |  | ||||||
|         getDetector().restartGesture(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected boolean isAnimating() { |  | ||||||
|         return mIsAnimating; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void setAnimating(boolean isAnimating) { |  | ||||||
|         mIsAnimating = isAnimating; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected float[] getStartValues() { |  | ||||||
|         return mStartValues; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected float[] getStopValues() { |  | ||||||
|         return mStopValues; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected Matrix getWorkingTransform() { |  | ||||||
|         return mWorkingTransform; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureBegin(TransformGestureDetector detector) { |  | ||||||
|         FLog.v(getLogTag(), "onGestureBegin"); |  | ||||||
|         stopAnimation(); |  | ||||||
|         super.onGestureBegin(detector); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureUpdate(TransformGestureDetector detector) { |  | ||||||
|         FLog.v(getLogTag(), "onGestureUpdate %s", isAnimating() ? "(ignored)" : ""); |  | ||||||
|         if (isAnimating()) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         super.onGestureUpdate(detector); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void calculateInterpolation(Matrix outMatrix, float fraction) { |  | ||||||
|         for (int i = 0; i < 9; i++) { |  | ||||||
|             mCurrentValues[i] = (1 - fraction) * mStartValues[i] + fraction * mStopValues[i]; |  | ||||||
|         } |  | ||||||
|         outMatrix.setValues(mCurrentValues); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public abstract void setTransformAnimated( |  | ||||||
|             final Matrix newTransform, long durationMs, @Nullable final Runnable onAnimationComplete); |  | ||||||
| 
 |  | ||||||
|     protected abstract void stopAnimation(); |  | ||||||
| 
 |  | ||||||
|     protected abstract Class<?> getLogTag(); |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,147 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.graphics.Matrix | ||||||
|  | import android.graphics.PointF | ||||||
|  | import com.facebook.common.logging.FLog | ||||||
|  | import fr.free.nrw.commons.media.zoomControllers.gestures.TransformGestureDetector | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Abstract class for ZoomableController that adds animation capabilities to | ||||||
|  |  * DefaultZoomableController. | ||||||
|  |  */ | ||||||
|  | abstract class AbstractAnimatedZoomableController( | ||||||
|  |     transformGestureDetector: TransformGestureDetector | ||||||
|  | ) : DefaultZoomableController(transformGestureDetector) { | ||||||
|  | 
 | ||||||
|  |     private var isAnimating: Boolean = false | ||||||
|  |     private val startValues = FloatArray(9) | ||||||
|  |     private val stopValues = FloatArray(9) | ||||||
|  |     private val currentValues = FloatArray(9) | ||||||
|  |     private val newTransform = Matrix() | ||||||
|  |     private val workingTransform = Matrix() | ||||||
|  | 
 | ||||||
|  |     override fun reset() { | ||||||
|  |         FLog.v(logTag, "reset") | ||||||
|  |         stopAnimation() | ||||||
|  |         workingTransform.reset() | ||||||
|  |         newTransform.reset() | ||||||
|  |         super.reset() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Returns true if the zoomable transform is identity matrix, and the controller is idle. */ | ||||||
|  |     override fun isIdentity(): Boolean { | ||||||
|  |         return !isAnimating && super.isIdentity() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Zooms to the desired scale and positions the image so that the given image point corresponds | ||||||
|  |      * to the given view point. | ||||||
|  |      * | ||||||
|  |      * If this method is called while an animation or gesture is already in progress, the current | ||||||
|  |      * animation or gesture will be stopped first. | ||||||
|  |      * | ||||||
|  |      * @param scale desired scale, will be limited to {min, max} scale factor | ||||||
|  |      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) | ||||||
|  |      * @param viewPoint 2D point in view's absolute coordinate system | ||||||
|  |      */ | ||||||
|  |     override fun zoomToPoint(scale: Float, imagePoint: PointF, viewPoint: PointF) { | ||||||
|  |         zoomToPoint(scale, imagePoint, viewPoint, LIMIT_ALL, 0, null) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Zooms to the desired scale and positions the image so that the given image point corresponds | ||||||
|  |      * to the given view point. | ||||||
|  |      * | ||||||
|  |      * If this method is called while an animation or gesture is already in progress, the current | ||||||
|  |      * animation or gesture will be stopped first. | ||||||
|  |      * | ||||||
|  |      * @param scale desired scale, will be limited to {min, max} scale factor | ||||||
|  |      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) | ||||||
|  |      * @param viewPoint 2D point in view's absolute coordinate system | ||||||
|  |      * @param limitFlags whether to limit translation and/or scale. | ||||||
|  |      * @param durationMs length of animation of the zoom, or 0 if no animation desired | ||||||
|  |      * @param onAnimationComplete code to run when the animation completes. Ignored if durationMs=0 | ||||||
|  |      */ | ||||||
|  |     fun zoomToPoint( | ||||||
|  |         scale: Float, | ||||||
|  |         imagePoint: PointF, | ||||||
|  |         viewPoint: PointF, | ||||||
|  |         @LimitFlag limitFlags: Int, | ||||||
|  |         durationMs: Long, | ||||||
|  |         onAnimationComplete: Runnable? | ||||||
|  |     ) { | ||||||
|  |         FLog.v(logTag, "zoomToPoint: duration $durationMs ms") | ||||||
|  |         calculateZoomToPointTransform(newTransform, scale, imagePoint, viewPoint, limitFlags) | ||||||
|  |         setTransform(newTransform, durationMs, onAnimationComplete) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets a new zoomable transformation and animates to it if desired. | ||||||
|  |      * | ||||||
|  |      * If this method is called while an animation or gesture is already in progress, the current | ||||||
|  |      * animation or gesture will be stopped first. | ||||||
|  |      * | ||||||
|  |      * @param newTransform new transform to make active | ||||||
|  |      * @param durationMs duration of the animation, or 0 to not animate | ||||||
|  |      * @param onAnimationComplete code to run when the animation completes. Ignored if durationMs=0 | ||||||
|  |      */ | ||||||
|  |     private fun setTransform( | ||||||
|  |         newTransform: Matrix, | ||||||
|  |         durationMs: Long, | ||||||
|  |         onAnimationComplete: Runnable? | ||||||
|  |     ) { | ||||||
|  |         FLog.v(logTag, "setTransform: duration $durationMs ms") | ||||||
|  |         if (durationMs <= 0) { | ||||||
|  |             setTransformImmediate(newTransform) | ||||||
|  |         } else { | ||||||
|  |             setTransformAnimated(newTransform, durationMs, onAnimationComplete) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun setTransformImmediate(newTransform: Matrix) { | ||||||
|  |         FLog.v(logTag, "setTransformImmediate") | ||||||
|  |         stopAnimation() | ||||||
|  |         workingTransform.set(newTransform) | ||||||
|  |         super.setTransform(newTransform) | ||||||
|  |         getDetector().restartGesture() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun getIsAnimating(): Boolean = isAnimating | ||||||
|  | 
 | ||||||
|  |     protected fun setAnimating(isAnimating: Boolean) { | ||||||
|  |         this.isAnimating = isAnimating | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun getStartValues(): FloatArray = startValues | ||||||
|  | 
 | ||||||
|  |     protected fun getStopValues(): FloatArray = stopValues | ||||||
|  | 
 | ||||||
|  |     protected fun getWorkingTransform(): Matrix = workingTransform | ||||||
|  | 
 | ||||||
|  |     override fun onGestureBegin(detector: TransformGestureDetector) { | ||||||
|  |         FLog.v(logTag, "onGestureBegin") | ||||||
|  |         stopAnimation() | ||||||
|  |         super.onGestureBegin(detector) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onGestureUpdate(detector: TransformGestureDetector) { | ||||||
|  |         FLog.v(logTag, "onGestureUpdate ${if (isAnimating) "(ignored)" else ""}") | ||||||
|  |         if (isAnimating) return | ||||||
|  |         super.onGestureUpdate(detector) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun calculateInterpolation(outMatrix: Matrix, fraction: Float) { | ||||||
|  |         for (i in 0..8) { | ||||||
|  |             currentValues[i] = (1 - fraction) * startValues[i] + fraction * stopValues[i] | ||||||
|  |         } | ||||||
|  |         outMatrix.setValues(currentValues) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     abstract fun setTransformAnimated( | ||||||
|  |         newTransform: Matrix, durationMs: Long, onAnimationComplete: Runnable? | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     protected abstract fun stopAnimation() | ||||||
|  | 
 | ||||||
|  |     protected abstract val logTag: Class<*> | ||||||
|  | } | ||||||
|  | @ -1,96 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.animation.Animator; |  | ||||||
| import android.animation.AnimatorListenerAdapter; |  | ||||||
| import android.animation.ValueAnimator; |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.graphics.Matrix; |  | ||||||
| import android.view.animation.DecelerateInterpolator; |  | ||||||
| import com.facebook.common.internal.Preconditions; |  | ||||||
| import com.facebook.common.logging.FLog; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import fr.free.nrw.commons.media.zoomControllers.gestures.TransformGestureDetector; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * ZoomableController that adds animation capabilities to DefaultZoomableController using standard |  | ||||||
|  * Android animation classes |  | ||||||
|  */ |  | ||||||
| public class AnimatedZoomableController extends AbstractAnimatedZoomableController { |  | ||||||
| 
 |  | ||||||
|     private static final Class<?> TAG = AnimatedZoomableController.class; |  | ||||||
| 
 |  | ||||||
|     private final ValueAnimator mValueAnimator; |  | ||||||
| 
 |  | ||||||
|     public static AnimatedZoomableController newInstance() { |  | ||||||
|         return new AnimatedZoomableController(TransformGestureDetector.newInstance()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressLint("NewApi") |  | ||||||
|     public AnimatedZoomableController(TransformGestureDetector transformGestureDetector) { |  | ||||||
|         super(transformGestureDetector); |  | ||||||
|         mValueAnimator = ValueAnimator.ofFloat(0, 1); |  | ||||||
|         mValueAnimator.setInterpolator(new DecelerateInterpolator()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressLint("NewApi") |  | ||||||
|     @Override |  | ||||||
|     public void setTransformAnimated( |  | ||||||
|             final Matrix newTransform, long durationMs, @Nullable final Runnable onAnimationComplete) { |  | ||||||
|         FLog.v(getLogTag(), "setTransformAnimated: duration %d ms", durationMs); |  | ||||||
|         stopAnimation(); |  | ||||||
|         Preconditions.checkArgument(durationMs > 0); |  | ||||||
|         Preconditions.checkState(!isAnimating()); |  | ||||||
|         setAnimating(true); |  | ||||||
|         mValueAnimator.setDuration(durationMs); |  | ||||||
|         getTransform().getValues(getStartValues()); |  | ||||||
|         newTransform.getValues(getStopValues()); |  | ||||||
|         mValueAnimator.addUpdateListener( |  | ||||||
|                 new ValueAnimator.AnimatorUpdateListener() { |  | ||||||
|                     @Override |  | ||||||
|                     public void onAnimationUpdate(ValueAnimator valueAnimator) { |  | ||||||
|                         calculateInterpolation(getWorkingTransform(), (float) valueAnimator.getAnimatedValue()); |  | ||||||
|                         AnimatedZoomableController.super.setTransform(getWorkingTransform()); |  | ||||||
|                     } |  | ||||||
|                 }); |  | ||||||
|         mValueAnimator.addListener( |  | ||||||
|                 new AnimatorListenerAdapter() { |  | ||||||
|                     @Override |  | ||||||
|                     public void onAnimationCancel(Animator animation) { |  | ||||||
|                         FLog.v(getLogTag(), "setTransformAnimated: animation cancelled"); |  | ||||||
|                         onAnimationStopped(); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     @Override |  | ||||||
|                     public void onAnimationEnd(Animator animation) { |  | ||||||
|                         FLog.v(getLogTag(), "setTransformAnimated: animation finished"); |  | ||||||
|                         onAnimationStopped(); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     private void onAnimationStopped() { |  | ||||||
|                         if (onAnimationComplete != null) { |  | ||||||
|                             onAnimationComplete.run(); |  | ||||||
|                         } |  | ||||||
|                         setAnimating(false); |  | ||||||
|                         getDetector().restartGesture(); |  | ||||||
|                     } |  | ||||||
|                 }); |  | ||||||
|         mValueAnimator.start(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressLint("NewApi") |  | ||||||
|     @Override |  | ||||||
|     public void stopAnimation() { |  | ||||||
|         if (!isAnimating()) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         FLog.v(getLogTag(), "stopAnimation"); |  | ||||||
|         mValueAnimator.cancel(); |  | ||||||
|         mValueAnimator.removeAllUpdateListeners(); |  | ||||||
|         mValueAnimator.removeAllListeners(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected Class<?> getLogTag() { |  | ||||||
|         return TAG; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,75 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.animation.Animator | ||||||
|  | import android.animation.AnimatorListenerAdapter | ||||||
|  | import android.animation.ValueAnimator | ||||||
|  | import android.annotation.SuppressLint | ||||||
|  | import android.graphics.Matrix | ||||||
|  | import android.view.animation.DecelerateInterpolator | ||||||
|  | import com.facebook.common.logging.FLog | ||||||
|  | import fr.free.nrw.commons.media.zoomControllers.gestures.TransformGestureDetector | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * ZoomableController that adds animation capabilities to DefaultZoomableController using standard | ||||||
|  |  * Android animation classes | ||||||
|  |  */ | ||||||
|  | class AnimatedZoomableController private constructor() : | ||||||
|  |     AbstractAnimatedZoomableController(TransformGestureDetector.newInstance()) { | ||||||
|  | 
 | ||||||
|  |     private val valueAnimator: ValueAnimator = ValueAnimator.ofFloat(0f, 1f).apply { | ||||||
|  |         interpolator = DecelerateInterpolator() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun newInstance(): AnimatedZoomableController { | ||||||
|  |             return AnimatedZoomableController() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @SuppressLint("NewApi") | ||||||
|  |     override fun setTransformAnimated( | ||||||
|  |         newTransform: Matrix, durationMs: Long, onAnimationComplete: Runnable? | ||||||
|  |     ) { | ||||||
|  |         FLog.v(logTag, "setTransformAnimated: duration $durationMs ms") | ||||||
|  |         stopAnimation() | ||||||
|  |         require(durationMs > 0) { "Duration must be greater than zero" } | ||||||
|  |         check(!getIsAnimating()) { "Animation is already in progress" } | ||||||
|  |         setAnimating(true) | ||||||
|  |         valueAnimator.duration = durationMs | ||||||
|  |         getTransform().getValues(getStartValues()) | ||||||
|  |         newTransform.getValues(getStopValues()) | ||||||
|  |         valueAnimator.addUpdateListener { animator -> | ||||||
|  |             calculateInterpolation(getWorkingTransform(), animator.animatedValue as Float) | ||||||
|  |             super.setTransform(getWorkingTransform()) | ||||||
|  |         } | ||||||
|  |         valueAnimator.addListener(object : AnimatorListenerAdapter() { | ||||||
|  |             override fun onAnimationCancel(animation: Animator) { | ||||||
|  |                 FLog.v(logTag, "setTransformAnimated: animation cancelled") | ||||||
|  |                 onAnimationStopped() | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             override fun onAnimationEnd(animation: Animator) { | ||||||
|  |                 FLog.v(logTag, "setTransformAnimated: animation finished") | ||||||
|  |                 onAnimationStopped() | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             private fun onAnimationStopped() { | ||||||
|  |                 onAnimationComplete?.run() | ||||||
|  |                 setAnimating(false) | ||||||
|  |                 getDetector().restartGesture() | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |         valueAnimator.start() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @SuppressLint("NewApi") | ||||||
|  |     override fun stopAnimation() { | ||||||
|  |         if (!getIsAnimating()) return | ||||||
|  |         FLog.v(logTag, "stopAnimation") | ||||||
|  |         valueAnimator.cancel() | ||||||
|  |         valueAnimator.removeAllUpdateListeners() | ||||||
|  |         valueAnimator.removeAllListeners() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override val logTag: Class<*> = AnimatedZoomableController::class.java | ||||||
|  | } | ||||||
|  | @ -1,646 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.graphics.Matrix; |  | ||||||
| import android.graphics.PointF; |  | ||||||
| import android.graphics.RectF; |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| import androidx.annotation.IntDef; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import fr.free.nrw.commons.media.zoomControllers.gestures.TransformGestureDetector; |  | ||||||
| import com.facebook.common.logging.FLog; |  | ||||||
| import java.lang.annotation.Retention; |  | ||||||
| import java.lang.annotation.RetentionPolicy; |  | ||||||
| 
 |  | ||||||
| /** Zoomable controller that calculates transformation based on touch events. */ |  | ||||||
| public class DefaultZoomableController |  | ||||||
|         implements ZoomableController, TransformGestureDetector.Listener { |  | ||||||
| 
 |  | ||||||
|     /** Interface for handling call backs when the image bounds are set. */ |  | ||||||
|     public interface ImageBoundsListener { |  | ||||||
|         void onImageBoundsSet(RectF imageBounds); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @IntDef( |  | ||||||
|             flag = true, |  | ||||||
|             value = {LIMIT_NONE, LIMIT_TRANSLATION_X, LIMIT_TRANSLATION_Y, LIMIT_SCALE, LIMIT_ALL}) |  | ||||||
|     @Retention(RetentionPolicy.SOURCE) |  | ||||||
|     public @interface LimitFlag {} |  | ||||||
| 
 |  | ||||||
|     public static final int LIMIT_NONE = 0; |  | ||||||
|     public static final int LIMIT_TRANSLATION_X = 1; |  | ||||||
|     public static final int LIMIT_TRANSLATION_Y = 2; |  | ||||||
|     public static final int LIMIT_SCALE = 4; |  | ||||||
|     public static final int LIMIT_ALL = LIMIT_TRANSLATION_X | LIMIT_TRANSLATION_Y | LIMIT_SCALE; |  | ||||||
| 
 |  | ||||||
|     private static final float EPS = 1e-3f; |  | ||||||
| 
 |  | ||||||
|     private static final Class<?> TAG = DefaultZoomableController.class; |  | ||||||
| 
 |  | ||||||
|     private static final RectF IDENTITY_RECT = new RectF(0, 0, 1, 1); |  | ||||||
| 
 |  | ||||||
|     private TransformGestureDetector mGestureDetector; |  | ||||||
| 
 |  | ||||||
|     private @Nullable ImageBoundsListener mImageBoundsListener; |  | ||||||
| 
 |  | ||||||
|     private @Nullable Listener mListener = null; |  | ||||||
| 
 |  | ||||||
|     private boolean mIsEnabled = false; |  | ||||||
|     private boolean mIsRotationEnabled = false; |  | ||||||
|     private boolean mIsScaleEnabled = true; |  | ||||||
|     private boolean mIsTranslationEnabled = true; |  | ||||||
|     private boolean mIsGestureZoomEnabled = true; |  | ||||||
| 
 |  | ||||||
|     private float mMinScaleFactor = 1.0f; |  | ||||||
|     private float mMaxScaleFactor = 2.0f; |  | ||||||
| 
 |  | ||||||
|     // View bounds, in view-absolute coordinates |  | ||||||
|     private final RectF mViewBounds = new RectF(); |  | ||||||
|     // Non-transformed image bounds, in view-absolute coordinates |  | ||||||
|     private final RectF mImageBounds = new RectF(); |  | ||||||
|     // Transformed image bounds, in view-absolute coordinates |  | ||||||
|     private final RectF mTransformedImageBounds = new RectF(); |  | ||||||
| 
 |  | ||||||
|     private final Matrix mPreviousTransform = new Matrix(); |  | ||||||
|     private final Matrix mActiveTransform = new Matrix(); |  | ||||||
|     private final Matrix mActiveTransformInverse = new Matrix(); |  | ||||||
|     private final float[] mTempValues = new float[9]; |  | ||||||
|     private final RectF mTempRect = new RectF(); |  | ||||||
|     private boolean mWasTransformCorrected; |  | ||||||
| 
 |  | ||||||
|     public static DefaultZoomableController newInstance() { |  | ||||||
|         return new DefaultZoomableController(TransformGestureDetector.newInstance()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public DefaultZoomableController(TransformGestureDetector gestureDetector) { |  | ||||||
|         mGestureDetector = gestureDetector; |  | ||||||
|         mGestureDetector.setListener(this); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Rests the controller. */ |  | ||||||
|     public void reset() { |  | ||||||
|         FLog.v(TAG, "reset"); |  | ||||||
|         mGestureDetector.reset(); |  | ||||||
|         mPreviousTransform.reset(); |  | ||||||
|         mActiveTransform.reset(); |  | ||||||
|         onTransformChanged(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets the zoomable listener. */ |  | ||||||
|     @Override |  | ||||||
|     public void setListener(Listener listener) { |  | ||||||
|         mListener = listener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets whether the controller is enabled or not. */ |  | ||||||
|     @Override |  | ||||||
|     public void setEnabled(boolean enabled) { |  | ||||||
|         mIsEnabled = enabled; |  | ||||||
|         if (!enabled) { |  | ||||||
|             reset(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether the controller is enabled or not. */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isEnabled() { |  | ||||||
|         return mIsEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets whether the rotation gesture is enabled or not. */ |  | ||||||
|     public void setRotationEnabled(boolean enabled) { |  | ||||||
|         mIsRotationEnabled = enabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether the rotation gesture is enabled or not. */ |  | ||||||
|     public boolean isRotationEnabled() { |  | ||||||
|         return mIsRotationEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets whether the scale gesture is enabled or not. */ |  | ||||||
|     public void setScaleEnabled(boolean enabled) { |  | ||||||
|         mIsScaleEnabled = enabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether the scale gesture is enabled or not. */ |  | ||||||
|     public boolean isScaleEnabled() { |  | ||||||
|         return mIsScaleEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets whether the translation gesture is enabled or not. */ |  | ||||||
|     public void setTranslationEnabled(boolean enabled) { |  | ||||||
|         mIsTranslationEnabled = enabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether the translations gesture is enabled or not. */ |  | ||||||
|     public boolean isTranslationEnabled() { |  | ||||||
|         return mIsTranslationEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets the minimum scale factor allowed. |  | ||||||
|      * |  | ||||||
|      * <p>Hierarchy's scaling (if any) is not taken into account. |  | ||||||
|      */ |  | ||||||
|     public void setMinScaleFactor(float minScaleFactor) { |  | ||||||
|         mMinScaleFactor = minScaleFactor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the minimum scale factor allowed. */ |  | ||||||
|     public float getMinScaleFactor() { |  | ||||||
|         return mMinScaleFactor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets the maximum scale factor allowed. |  | ||||||
|      * |  | ||||||
|      * <p>Hierarchy's scaling (if any) is not taken into account. |  | ||||||
|      */ |  | ||||||
|     public void setMaxScaleFactor(float maxScaleFactor) { |  | ||||||
|         mMaxScaleFactor = maxScaleFactor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the maximum scale factor allowed. */ |  | ||||||
|     public float getMaxScaleFactor() { |  | ||||||
|         return mMaxScaleFactor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets whether gesture zooms are enabled or not. */ |  | ||||||
|     public void setGestureZoomEnabled(boolean isGestureZoomEnabled) { |  | ||||||
|         mIsGestureZoomEnabled = isGestureZoomEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets whether gesture zooms are enabled or not. */ |  | ||||||
|     public boolean isGestureZoomEnabled() { |  | ||||||
|         return mIsGestureZoomEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the current scale factor. */ |  | ||||||
|     @Override |  | ||||||
|     public float getScaleFactor() { |  | ||||||
|         return getMatrixScaleFactor(mActiveTransform); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets the image bounds, in view-absolute coordinates. */ |  | ||||||
|     @Override |  | ||||||
|     public void setImageBounds(RectF imageBounds) { |  | ||||||
|         if (!imageBounds.equals(mImageBounds)) { |  | ||||||
|             mImageBounds.set(imageBounds); |  | ||||||
|             onTransformChanged(); |  | ||||||
|             if (mImageBoundsListener != null) { |  | ||||||
|                 mImageBoundsListener.onImageBoundsSet(mImageBounds); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the non-transformed image bounds, in view-absolute coordinates. */ |  | ||||||
|     public RectF getImageBounds() { |  | ||||||
|         return mImageBounds; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the transformed image bounds, in view-absolute coordinates */ |  | ||||||
|     private RectF getTransformedImageBounds() { |  | ||||||
|         return mTransformedImageBounds; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets the view bounds. */ |  | ||||||
|     @Override |  | ||||||
|     public void setViewBounds(RectF viewBounds) { |  | ||||||
|         mViewBounds.set(viewBounds); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the view bounds. */ |  | ||||||
|     public RectF getViewBounds() { |  | ||||||
|         return mViewBounds; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets the image bounds listener. */ |  | ||||||
|     public void setImageBoundsListener(@Nullable ImageBoundsListener imageBoundsListener) { |  | ||||||
|         mImageBoundsListener = imageBoundsListener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the image bounds listener. */ |  | ||||||
|     public @Nullable ImageBoundsListener getImageBoundsListener() { |  | ||||||
|         return mImageBoundsListener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Returns true if the zoomable transform is identity matrix. */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isIdentity() { |  | ||||||
|         return isMatrixIdentity(mActiveTransform, 1e-3f); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Returns true if the transform was corrected during the last update. |  | ||||||
|      * |  | ||||||
|      * <p>We should rename this method to `wasTransformedWithoutCorrection` and just return the |  | ||||||
|      * internal flag directly. However, this requires interface change and negation of meaning. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean wasTransformCorrected() { |  | ||||||
|         return mWasTransformCorrected; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the matrix that transforms image-absolute coordinates to view-absolute coordinates. The |  | ||||||
|      * zoomable transformation is taken into account. |  | ||||||
|      * |  | ||||||
|      * <p>Internal matrix is exposed for performance reasons and is not to be modified by the callers. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Matrix getTransform() { |  | ||||||
|         return mActiveTransform; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the matrix that transforms image-relative coordinates to view-absolute coordinates. The |  | ||||||
|      * zoomable transformation is taken into account. |  | ||||||
|      */ |  | ||||||
|     public void getImageRelativeToViewAbsoluteTransform(Matrix outMatrix) { |  | ||||||
|         outMatrix.setRectToRect(IDENTITY_RECT, mTransformedImageBounds, Matrix.ScaleToFit.FILL); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Maps point from view-absolute to image-relative coordinates. This takes into account the |  | ||||||
|      * zoomable transformation. |  | ||||||
|      */ |  | ||||||
|     public PointF mapViewToImage(PointF viewPoint) { |  | ||||||
|         float[] points = mTempValues; |  | ||||||
|         points[0] = viewPoint.x; |  | ||||||
|         points[1] = viewPoint.y; |  | ||||||
|         mActiveTransform.invert(mActiveTransformInverse); |  | ||||||
|         mActiveTransformInverse.mapPoints(points, 0, points, 0, 1); |  | ||||||
|         mapAbsoluteToRelative(points, points, 1); |  | ||||||
|         return new PointF(points[0], points[1]); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Maps point from image-relative to view-absolute coordinates. This takes into account the |  | ||||||
|      * zoomable transformation. |  | ||||||
|      */ |  | ||||||
|     public PointF mapImageToView(PointF imagePoint) { |  | ||||||
|         float[] points = mTempValues; |  | ||||||
|         points[0] = imagePoint.x; |  | ||||||
|         points[1] = imagePoint.y; |  | ||||||
|         mapRelativeToAbsolute(points, points, 1); |  | ||||||
|         mActiveTransform.mapPoints(points, 0, points, 0, 1); |  | ||||||
|         return new PointF(points[0], points[1]); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Maps array of 2D points from view-absolute to image-relative coordinates. This does NOT take |  | ||||||
|      * into account the zoomable transformation. Points are represented by a float array of [x0, y0, |  | ||||||
|      * x1, y1, ...]. |  | ||||||
|      * |  | ||||||
|      * @param destPoints destination array (may be the same as source array) |  | ||||||
|      * @param srcPoints source array |  | ||||||
|      * @param numPoints number of points to map |  | ||||||
|      */ |  | ||||||
|     private void mapAbsoluteToRelative(float[] destPoints, float[] srcPoints, int numPoints) { |  | ||||||
|         for (int i = 0; i < numPoints; i++) { |  | ||||||
|             destPoints[i * 2 + 0] = (srcPoints[i * 2 + 0] - mImageBounds.left) / mImageBounds.width(); |  | ||||||
|             destPoints[i * 2 + 1] = (srcPoints[i * 2 + 1] - mImageBounds.top) / mImageBounds.height(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Maps array of 2D points from image-relative to view-absolute coordinates. This does NOT take |  | ||||||
|      * into account the zoomable transformation. Points are represented by float array of [x0, y0, x1, |  | ||||||
|      * y1, ...]. |  | ||||||
|      * |  | ||||||
|      * @param destPoints destination array (may be the same as source array) |  | ||||||
|      * @param srcPoints source array |  | ||||||
|      * @param numPoints number of points to map |  | ||||||
|      */ |  | ||||||
|     private void mapRelativeToAbsolute(float[] destPoints, float[] srcPoints, int numPoints) { |  | ||||||
|         for (int i = 0; i < numPoints; i++) { |  | ||||||
|             destPoints[i * 2 + 0] = srcPoints[i * 2 + 0] * mImageBounds.width() + mImageBounds.left; |  | ||||||
|             destPoints[i * 2 + 1] = srcPoints[i * 2 + 1] * mImageBounds.height() + mImageBounds.top; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Zooms to the desired scale and positions the image so that the given image point corresponds to |  | ||||||
|      * the given view point. |  | ||||||
|      * |  | ||||||
|      * @param scale desired scale, will be limited to {min, max} scale factor |  | ||||||
|      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) |  | ||||||
|      * @param viewPoint 2D point in view's absolute coordinate system |  | ||||||
|      */ |  | ||||||
|     public void zoomToPoint(float scale, PointF imagePoint, PointF viewPoint) { |  | ||||||
|         FLog.v(TAG, "zoomToPoint"); |  | ||||||
|         calculateZoomToPointTransform(mActiveTransform, scale, imagePoint, viewPoint, LIMIT_ALL); |  | ||||||
|         onTransformChanged(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Calculates the zoom transformation that would zoom to the desired scale and position the image |  | ||||||
|      * so that the given image point corresponds to the given view point. |  | ||||||
|      * |  | ||||||
|      * @param outTransform the matrix to store the result to |  | ||||||
|      * @param scale desired scale, will be limited to {min, max} scale factor |  | ||||||
|      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) |  | ||||||
|      * @param viewPoint 2D point in view's absolute coordinate system |  | ||||||
|      * @param limitFlags whether to limit translation and/or scale. |  | ||||||
|      * @return whether or not the transform has been corrected due to limitation |  | ||||||
|      */ |  | ||||||
|     protected boolean calculateZoomToPointTransform( |  | ||||||
|             Matrix outTransform, |  | ||||||
|             float scale, |  | ||||||
|             PointF imagePoint, |  | ||||||
|             PointF viewPoint, |  | ||||||
|             @LimitFlag int limitFlags) { |  | ||||||
|         float[] viewAbsolute = mTempValues; |  | ||||||
|         viewAbsolute[0] = imagePoint.x; |  | ||||||
|         viewAbsolute[1] = imagePoint.y; |  | ||||||
|         mapRelativeToAbsolute(viewAbsolute, viewAbsolute, 1); |  | ||||||
|         float distanceX = viewPoint.x - viewAbsolute[0]; |  | ||||||
|         float distanceY = viewPoint.y - viewAbsolute[1]; |  | ||||||
|         boolean transformCorrected = false; |  | ||||||
|         outTransform.setScale(scale, scale, viewAbsolute[0], viewAbsolute[1]); |  | ||||||
|         transformCorrected |= limitScale(outTransform, viewAbsolute[0], viewAbsolute[1], limitFlags); |  | ||||||
|         outTransform.postTranslate(distanceX, distanceY); |  | ||||||
|         transformCorrected |= limitTranslation(outTransform, limitFlags); |  | ||||||
|         return transformCorrected; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets a new zoom transformation. */ |  | ||||||
|     public void setTransform(Matrix newTransform) { |  | ||||||
|         FLog.v(TAG, "setTransform"); |  | ||||||
|         mActiveTransform.set(newTransform); |  | ||||||
|         onTransformChanged(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Gets the gesture detector. */ |  | ||||||
|     protected TransformGestureDetector getDetector() { |  | ||||||
|         return mGestureDetector; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Notifies controller of the received touch event. */ |  | ||||||
|     @Override |  | ||||||
|     public boolean onTouchEvent(MotionEvent event) { |  | ||||||
|         FLog.v(TAG, "onTouchEvent: action: ", event.getAction()); |  | ||||||
|         if (mIsEnabled && mIsGestureZoomEnabled) { |  | ||||||
|             return mGestureDetector.onTouchEvent(event); |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /* TransformGestureDetector.Listener methods  */ |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureBegin(TransformGestureDetector detector) { |  | ||||||
|         FLog.v(TAG, "onGestureBegin"); |  | ||||||
|         mPreviousTransform.set(mActiveTransform); |  | ||||||
|         onTransformBegin(); |  | ||||||
|         // We only received a touch down event so far, and so we don't know yet in which direction a |  | ||||||
|         // future move event will follow. Therefore, if we can't scroll in all directions, we have to |  | ||||||
|         // assume the worst case where the user tries to scroll out of edge, which would cause |  | ||||||
|         // transformation to be corrected. |  | ||||||
|         mWasTransformCorrected = !canScrollInAllDirection(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureUpdate(TransformGestureDetector detector) { |  | ||||||
|         FLog.v(TAG, "onGestureUpdate"); |  | ||||||
|         boolean transformCorrected = calculateGestureTransform(mActiveTransform, LIMIT_ALL); |  | ||||||
|         onTransformChanged(); |  | ||||||
|         if (transformCorrected) { |  | ||||||
|             mGestureDetector.restartGesture(); |  | ||||||
|         } |  | ||||||
|         // A transformation happened, but was it without correction? |  | ||||||
|         mWasTransformCorrected = transformCorrected; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onGestureEnd(TransformGestureDetector detector) { |  | ||||||
|         FLog.v(TAG, "onGestureEnd"); |  | ||||||
|         onTransformEnd(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Calculates the zoom transformation based on the current gesture. |  | ||||||
|      * |  | ||||||
|      * @param outTransform the matrix to store the result to |  | ||||||
|      * @param limitTypes whether to limit translation and/or scale. |  | ||||||
|      * @return whether or not the transform has been corrected due to limitation |  | ||||||
|      */ |  | ||||||
|     protected boolean calculateGestureTransform(Matrix outTransform, @LimitFlag int limitTypes) { |  | ||||||
|         TransformGestureDetector detector = mGestureDetector; |  | ||||||
|         boolean transformCorrected = false; |  | ||||||
|         outTransform.set(mPreviousTransform); |  | ||||||
|         if (mIsRotationEnabled) { |  | ||||||
|             float angle = detector.getRotation() * (float) (180 / Math.PI); |  | ||||||
|             outTransform.postRotate(angle, detector.getPivotX(), detector.getPivotY()); |  | ||||||
|         } |  | ||||||
|         if (mIsScaleEnabled) { |  | ||||||
|             float scale = detector.getScale(); |  | ||||||
|             outTransform.postScale(scale, scale, detector.getPivotX(), detector.getPivotY()); |  | ||||||
|         } |  | ||||||
|         transformCorrected |= |  | ||||||
|                 limitScale(outTransform, detector.getPivotX(), detector.getPivotY(), limitTypes); |  | ||||||
|         if (mIsTranslationEnabled) { |  | ||||||
|             outTransform.postTranslate(detector.getTranslationX(), detector.getTranslationY()); |  | ||||||
|         } |  | ||||||
|         transformCorrected |= limitTranslation(outTransform, limitTypes); |  | ||||||
|         return transformCorrected; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void onTransformBegin() { |  | ||||||
|         if (mListener != null && isEnabled()) { |  | ||||||
|             mListener.onTransformBegin(mActiveTransform); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void onTransformChanged() { |  | ||||||
|         mActiveTransform.mapRect(mTransformedImageBounds, mImageBounds); |  | ||||||
|         if (mListener != null && isEnabled()) { |  | ||||||
|             mListener.onTransformChanged(mActiveTransform); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void onTransformEnd() { |  | ||||||
|         if (mListener != null && isEnabled()) { |  | ||||||
|             mListener.onTransformEnd(mActiveTransform); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Keeps the scaling factor within the specified limits. |  | ||||||
|      * |  | ||||||
|      * @param pivotX x coordinate of the pivot point |  | ||||||
|      * @param pivotY y coordinate of the pivot point |  | ||||||
|      * @param limitTypes whether to limit scale. |  | ||||||
|      * @return whether limiting has been applied or not |  | ||||||
|      */ |  | ||||||
|     private boolean limitScale( |  | ||||||
|             Matrix transform, float pivotX, float pivotY, @LimitFlag int limitTypes) { |  | ||||||
|         if (!shouldLimit(limitTypes, LIMIT_SCALE)) { |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|         float currentScale = getMatrixScaleFactor(transform); |  | ||||||
|         float targetScale = limit(currentScale, mMinScaleFactor, mMaxScaleFactor); |  | ||||||
|         if (targetScale != currentScale) { |  | ||||||
|             float scale = targetScale / currentScale; |  | ||||||
|             transform.postScale(scale, scale, pivotX, pivotY); |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Limits the translation so that there are no empty spaces on the sides if possible. |  | ||||||
|      * |  | ||||||
|      * <p>The image is attempted to be centered within the view bounds if the transformed image is |  | ||||||
|      * smaller. There will be no empty spaces within the view bounds if the transformed image is |  | ||||||
|      * bigger. This applies to each dimension (horizontal and vertical) independently. |  | ||||||
|      * |  | ||||||
|      * @param limitTypes whether to limit translation along the specific axis. |  | ||||||
|      * @return whether limiting has been applied or not |  | ||||||
|      */ |  | ||||||
|     private boolean limitTranslation(Matrix transform, @LimitFlag int limitTypes) { |  | ||||||
|         if (!shouldLimit(limitTypes, LIMIT_TRANSLATION_X | LIMIT_TRANSLATION_Y)) { |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|         RectF b = mTempRect; |  | ||||||
|         b.set(mImageBounds); |  | ||||||
|         transform.mapRect(b); |  | ||||||
|         float offsetLeft = |  | ||||||
|                 shouldLimit(limitTypes, LIMIT_TRANSLATION_X) |  | ||||||
|                         ? getOffset( |  | ||||||
|                         b.left, b.right, mViewBounds.left, mViewBounds.right, mImageBounds.centerX()) |  | ||||||
|                         : 0; |  | ||||||
|         float offsetTop = |  | ||||||
|                 shouldLimit(limitTypes, LIMIT_TRANSLATION_Y) |  | ||||||
|                         ? getOffset( |  | ||||||
|                         b.top, b.bottom, mViewBounds.top, mViewBounds.bottom, mImageBounds.centerY()) |  | ||||||
|                         : 0; |  | ||||||
|         if (offsetLeft != 0 || offsetTop != 0) { |  | ||||||
|             transform.postTranslate(offsetLeft, offsetTop); |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Checks whether the specified limit flag is present in the limits provided. |  | ||||||
|      * |  | ||||||
|      * <p>If the flag contains multiple flags together using a bitwise OR, this only checks that at |  | ||||||
|      * least one of the flags is included. |  | ||||||
|      * |  | ||||||
|      * @param limits the limits to apply |  | ||||||
|      * @param flag the limit flag(s) to check for |  | ||||||
|      * @return true if the flag (or one of the flags) is included in the limits |  | ||||||
|      */ |  | ||||||
|     private static boolean shouldLimit(@LimitFlag int limits, @LimitFlag int flag) { |  | ||||||
|         return (limits & flag) != LIMIT_NONE; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Returns the offset necessary to make sure that: - the image is centered within the limit if the |  | ||||||
|      * image is smaller than the limit - there is no empty space on left/right if the image is bigger |  | ||||||
|      * than the limit |  | ||||||
|      */ |  | ||||||
|     private float getOffset( |  | ||||||
|             float imageStart, float imageEnd, float limitStart, float limitEnd, float limitCenter) { |  | ||||||
|         float imageWidth = imageEnd - imageStart, limitWidth = limitEnd - limitStart; |  | ||||||
|         float limitInnerWidth = Math.min(limitCenter - limitStart, limitEnd - limitCenter) * 2; |  | ||||||
|         // center if smaller than limitInnerWidth |  | ||||||
|         if (imageWidth < limitInnerWidth) { |  | ||||||
|             return limitCenter - (imageEnd + imageStart) / 2; |  | ||||||
|         } |  | ||||||
|         // to the edge if in between and limitCenter is not (limitLeft + limitRight) / 2 |  | ||||||
|         if (imageWidth < limitWidth) { |  | ||||||
|             if (limitCenter < (limitStart + limitEnd) / 2) { |  | ||||||
|                 return limitStart - imageStart; |  | ||||||
|             } else { |  | ||||||
|                 return limitEnd - imageEnd; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         // to the edge if larger than limitWidth and empty space visible |  | ||||||
|         if (imageStart > limitStart) { |  | ||||||
|             return limitStart - imageStart; |  | ||||||
|         } |  | ||||||
|         if (imageEnd < limitEnd) { |  | ||||||
|             return limitEnd - imageEnd; |  | ||||||
|         } |  | ||||||
|         return 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Limits the value to the given min and max range. */ |  | ||||||
|     private float limit(float value, float min, float max) { |  | ||||||
|         return Math.min(Math.max(min, value), max); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the scale factor for the given matrix. This method assumes the equal scaling factor for X |  | ||||||
|      * and Y axis. |  | ||||||
|      */ |  | ||||||
|     private float getMatrixScaleFactor(Matrix transform) { |  | ||||||
|         transform.getValues(mTempValues); |  | ||||||
|         return mTempValues[Matrix.MSCALE_X]; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Same as {@code Matrix.isIdentity()}, but with tolerance {@code eps}. */ |  | ||||||
|     private boolean isMatrixIdentity(Matrix transform, float eps) { |  | ||||||
|         // Checks whether the given matrix is close enough to the identity matrix: |  | ||||||
|         //   1 0 0 |  | ||||||
|         //   0 1 0 |  | ||||||
|         //   0 0 1 |  | ||||||
|         // Or equivalently to the zero matrix, after subtracting 1.0f from the diagonal elements: |  | ||||||
|         //   0 0 0 |  | ||||||
|         //   0 0 0 |  | ||||||
|         //   0 0 0 |  | ||||||
|         transform.getValues(mTempValues); |  | ||||||
|         mTempValues[0] -= 1.0f; // m00 |  | ||||||
|         mTempValues[4] -= 1.0f; // m11 |  | ||||||
|         mTempValues[8] -= 1.0f; // m22 |  | ||||||
|         for (int i = 0; i < 9; i++) { |  | ||||||
|             if (Math.abs(mTempValues[i]) > eps) { |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Returns whether the scroll can happen in all directions. I.e. the image is not on any edge. */ |  | ||||||
|     private boolean canScrollInAllDirection() { |  | ||||||
|         return mTransformedImageBounds.left < mViewBounds.left - EPS |  | ||||||
|                 && mTransformedImageBounds.top < mViewBounds.top - EPS |  | ||||||
|                 && mTransformedImageBounds.right > mViewBounds.right + EPS |  | ||||||
|                 && mTransformedImageBounds.bottom > mViewBounds.bottom + EPS; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeHorizontalScrollRange() { |  | ||||||
|         return (int) mTransformedImageBounds.width(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeHorizontalScrollOffset() { |  | ||||||
|         return (int) (mViewBounds.left - mTransformedImageBounds.left); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeHorizontalScrollExtent() { |  | ||||||
|         return (int) mViewBounds.width(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeVerticalScrollRange() { |  | ||||||
|         return (int) mTransformedImageBounds.height(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeVerticalScrollOffset() { |  | ||||||
|         return (int) (mViewBounds.top - mTransformedImageBounds.top); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeVerticalScrollExtent() { |  | ||||||
|         return (int) mViewBounds.height(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public Listener getListener() { |  | ||||||
|         return mListener; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,607 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.graphics.Matrix | ||||||
|  | import android.graphics.PointF | ||||||
|  | import android.graphics.RectF | ||||||
|  | import android.view.MotionEvent | ||||||
|  | import androidx.annotation.IntDef | ||||||
|  | import com.facebook.common.logging.FLog | ||||||
|  | import fr.free.nrw.commons.media.zoomControllers.gestures.TransformGestureDetector | ||||||
|  | import kotlin.math.abs | ||||||
|  | 
 | ||||||
|  | /** Zoomable controller that calculates transformation based on touch events. */ | ||||||
|  | open class DefaultZoomableController( | ||||||
|  |     private val mGestureDetector: TransformGestureDetector | ||||||
|  | ) : ZoomableController, TransformGestureDetector.Listener { | ||||||
|  | 
 | ||||||
|  |     /** Interface for handling call backs when the image bounds are set. */ | ||||||
|  |     fun interface ImageBoundsListener { | ||||||
|  |         fun onImageBoundsSet(imageBounds: RectF) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @IntDef( | ||||||
|  |         flag = true, | ||||||
|  |         value = [LIMIT_NONE, LIMIT_TRANSLATION_X, LIMIT_TRANSLATION_Y, LIMIT_SCALE, LIMIT_ALL] | ||||||
|  |     ) | ||||||
|  |     @Retention | ||||||
|  |     annotation class LimitFlag | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         const val LIMIT_NONE = 0 | ||||||
|  |         const val LIMIT_TRANSLATION_X = 1 | ||||||
|  |         const val LIMIT_TRANSLATION_Y = 2 | ||||||
|  |         const val LIMIT_SCALE = 4 | ||||||
|  |         const val LIMIT_ALL = LIMIT_TRANSLATION_X or LIMIT_TRANSLATION_Y or LIMIT_SCALE | ||||||
|  | 
 | ||||||
|  |         private const val EPS = 1e-3f | ||||||
|  | 
 | ||||||
|  |         private val TAG: Class<*> = DefaultZoomableController::class.java | ||||||
|  | 
 | ||||||
|  |         private val IDENTITY_RECT = RectF(0f, 0f, 1f, 1f) | ||||||
|  | 
 | ||||||
|  |         fun newInstance(): DefaultZoomableController { | ||||||
|  |             return DefaultZoomableController(TransformGestureDetector.newInstance()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private var mImageBoundsListener: ImageBoundsListener? = null | ||||||
|  | 
 | ||||||
|  |     private var mListener: ZoomableController.Listener? = null | ||||||
|  | 
 | ||||||
|  |     private var mIsEnabled = false | ||||||
|  |     private var mIsRotationEnabled = false | ||||||
|  |     private var mIsScaleEnabled = true | ||||||
|  |     private var mIsTranslationEnabled = true | ||||||
|  |     private var mIsGestureZoomEnabled = true | ||||||
|  | 
 | ||||||
|  |     private var mMinScaleFactor = 1.0f | ||||||
|  |     private var mMaxScaleFactor = 2.0f | ||||||
|  | 
 | ||||||
|  |     // View bounds, in view-absolute coordinates | ||||||
|  |     private val mViewBounds = RectF() | ||||||
|  |     // Non-transformed image bounds, in view-absolute coordinates | ||||||
|  |     private val mImageBounds = RectF() | ||||||
|  |     // Transformed image bounds, in view-absolute coordinates | ||||||
|  |     private val mTransformedImageBounds = RectF() | ||||||
|  | 
 | ||||||
|  |     private val mPreviousTransform = Matrix() | ||||||
|  |     private val mActiveTransform = Matrix() | ||||||
|  |     private val mActiveTransformInverse = Matrix() | ||||||
|  |     private val mTempValues = FloatArray(9) | ||||||
|  |     private val mTempRect = RectF() | ||||||
|  |     private var mWasTransformCorrected = false | ||||||
|  | 
 | ||||||
|  |     init { | ||||||
|  |         mGestureDetector.setListener(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Rests the controller. */ | ||||||
|  |     open fun reset() { | ||||||
|  |         FLog.v(TAG, "reset") | ||||||
|  |         mGestureDetector.reset() | ||||||
|  |         mPreviousTransform.reset() | ||||||
|  |         mActiveTransform.reset() | ||||||
|  |         onTransformChanged() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets the zoomable listener. */ | ||||||
|  |     override fun setListener(listener: ZoomableController.Listener?) { | ||||||
|  |         mListener = listener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets whether the controller is enabled or not. */ | ||||||
|  |     override fun setEnabled(enabled: Boolean) { | ||||||
|  |         mIsEnabled = enabled | ||||||
|  |         if (!enabled) { | ||||||
|  |             reset() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether the controller is enabled or not. */ | ||||||
|  |     override fun isEnabled(): Boolean { | ||||||
|  |         return mIsEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets whether the rotation gesture is enabled or not. */ | ||||||
|  |     fun setRotationEnabled(enabled: Boolean) { | ||||||
|  |         mIsRotationEnabled = enabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether the rotation gesture is enabled or not. */ | ||||||
|  |     fun isRotationEnabled(): Boolean { | ||||||
|  |         return mIsRotationEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets whether the scale gesture is enabled or not. */ | ||||||
|  |     fun setScaleEnabled(enabled: Boolean) { | ||||||
|  |         mIsScaleEnabled = enabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether the scale gesture is enabled or not. */ | ||||||
|  |     fun isScaleEnabled(): Boolean { | ||||||
|  |         return mIsScaleEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets whether the translation gesture is enabled or not. */ | ||||||
|  |     fun setTranslationEnabled(enabled: Boolean) { | ||||||
|  |         mIsTranslationEnabled = enabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether the translations gesture is enabled or not. */ | ||||||
|  |     fun isTranslationEnabled(): Boolean { | ||||||
|  |         return mIsTranslationEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the minimum scale factor allowed. | ||||||
|  |      * | ||||||
|  |      * <p>Hierarchy's scaling (if any) is not taken into account. | ||||||
|  |      */ | ||||||
|  |     fun setMinScaleFactor(minScaleFactor: Float) { | ||||||
|  |         mMinScaleFactor = minScaleFactor | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the minimum scale factor allowed. */ | ||||||
|  |     fun getMinScaleFactor(): Float { | ||||||
|  |         return mMinScaleFactor | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the maximum scale factor allowed. | ||||||
|  |      * | ||||||
|  |      * <p>Hierarchy's scaling (if any) is not taken into account. | ||||||
|  |      */ | ||||||
|  |     fun setMaxScaleFactor(maxScaleFactor: Float) { | ||||||
|  |         mMaxScaleFactor = maxScaleFactor | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the maximum scale factor allowed. */ | ||||||
|  |     fun getMaxScaleFactor(): Float { | ||||||
|  |         return mMaxScaleFactor | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets whether gesture zooms are enabled or not. */ | ||||||
|  |     fun setGestureZoomEnabled(isGestureZoomEnabled: Boolean) { | ||||||
|  |         mIsGestureZoomEnabled = isGestureZoomEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets whether gesture zooms are enabled or not. */ | ||||||
|  |     fun isGestureZoomEnabled(): Boolean { | ||||||
|  |         return mIsGestureZoomEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the current scale factor. */ | ||||||
|  |     override fun getScaleFactor(): Float { | ||||||
|  |         return getMatrixScaleFactor(mActiveTransform) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets the image bounds, in view-absolute coordinates. */ | ||||||
|  |     override fun setImageBounds(imageBounds: RectF) { | ||||||
|  |         if (imageBounds != mImageBounds) { | ||||||
|  |             mImageBounds.set(imageBounds) | ||||||
|  |             onTransformChanged() | ||||||
|  |             mImageBoundsListener?.onImageBoundsSet(mImageBounds) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the non-transformed image bounds, in view-absolute coordinates. */ | ||||||
|  |     fun getImageBounds(): RectF { | ||||||
|  |         return mImageBounds | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the transformed image bounds, in view-absolute coordinates */ | ||||||
|  |     private fun getTransformedImageBounds(): RectF { | ||||||
|  |         return mTransformedImageBounds | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets the view bounds. */ | ||||||
|  |     override fun setViewBounds(viewBounds: RectF) { | ||||||
|  |         mViewBounds.set(viewBounds) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the view bounds. */ | ||||||
|  |     fun getViewBounds(): RectF { | ||||||
|  |         return mViewBounds | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets the image bounds listener. */ | ||||||
|  |     fun setImageBoundsListener(imageBoundsListener: ImageBoundsListener?) { | ||||||
|  |         mImageBoundsListener = imageBoundsListener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the image bounds listener. */ | ||||||
|  |     fun getImageBoundsListener(): ImageBoundsListener? { | ||||||
|  |         return mImageBoundsListener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Returns true if the zoomable transform is identity matrix. */ | ||||||
|  |     override fun isIdentity(): Boolean { | ||||||
|  |         return isMatrixIdentity(mActiveTransform, 1e-3f) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns true if the transform was corrected during the last update. | ||||||
|  |      * | ||||||
|  |      * <p>We should rename this method to `wasTransformedWithoutCorrection` and just return the | ||||||
|  |      * internal flag directly. However, this requires interface change and negation of meaning. | ||||||
|  |      */ | ||||||
|  |     override fun wasTransformCorrected(): Boolean { | ||||||
|  |         return mWasTransformCorrected | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the matrix that transforms image-absolute coordinates to view-absolute coordinates. The | ||||||
|  |      * zoomable transformation is taken into account. | ||||||
|  |      * | ||||||
|  |      * <p>Internal matrix is exposed for performance reasons and is not to be modified by the | ||||||
|  |      * callers. | ||||||
|  |      */ | ||||||
|  |     override fun getTransform(): Matrix { | ||||||
|  |         return mActiveTransform | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the matrix that transforms image-relative coordinates to view-absolute coordinates. The | ||||||
|  |      * zoomable transformation is taken into account. | ||||||
|  |      */ | ||||||
|  |     fun getImageRelativeToViewAbsoluteTransform(outMatrix: Matrix) { | ||||||
|  |         outMatrix.setRectToRect(IDENTITY_RECT, mTransformedImageBounds, Matrix.ScaleToFit.FILL) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Maps point from view-absolute to image-relative coordinates. This takes into account the | ||||||
|  |      * zoomable transformation. | ||||||
|  |      */ | ||||||
|  |     fun mapViewToImage(viewPoint: PointF): PointF { | ||||||
|  |         val points = mTempValues | ||||||
|  |         points[0] = viewPoint.x | ||||||
|  |         points[1] = viewPoint.y | ||||||
|  |         mActiveTransform.invert(mActiveTransformInverse) | ||||||
|  |         mActiveTransformInverse.mapPoints(points, 0, points, 0, 1) | ||||||
|  |         mapAbsoluteToRelative(points, points, 1) | ||||||
|  |         return PointF(points[0], points[1]) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Maps point from image-relative to view-absolute coordinates. This takes into account the | ||||||
|  |      * zoomable transformation. | ||||||
|  |      */ | ||||||
|  |     fun mapImageToView(imagePoint: PointF): PointF { | ||||||
|  |         val points = mTempValues | ||||||
|  |         points[0] = imagePoint.x | ||||||
|  |         points[1] = imagePoint.y | ||||||
|  |         mapRelativeToAbsolute(points, points, 1) | ||||||
|  |         mActiveTransform.mapPoints(points, 0, points, 0, 1) | ||||||
|  |         return PointF(points[0], points[1]) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Maps array of 2D points from view-absolute to image-relative coordinates. This does NOT take | ||||||
|  |      * into account the zoomable transformation. Points are represented by a float array of [x0, y0, | ||||||
|  |      * x1, y1, ...]. | ||||||
|  |      * | ||||||
|  |      * @param destPoints destination array (may be the same as source array) | ||||||
|  |      * @param srcPoints source array | ||||||
|  |      * @param numPoints number of points to map | ||||||
|  |      */ | ||||||
|  |     private fun mapAbsoluteToRelative( | ||||||
|  |         destPoints: FloatArray, | ||||||
|  |         srcPoints: FloatArray, | ||||||
|  |         numPoints: Int | ||||||
|  |     ) { | ||||||
|  |         for (i in 0 until numPoints) { | ||||||
|  |             destPoints[i * 2] = (srcPoints[i * 2] - mImageBounds.left) / mImageBounds.width() | ||||||
|  |             destPoints[i * 2 + 1] = | ||||||
|  |                 (srcPoints[i * 2 + 1] - mImageBounds.top) / mImageBounds.height() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Maps array of 2D points from image-relative to view-absolute coordinates. This does NOT take | ||||||
|  |      * into account the zoomable transformation. Points are represented by float array of | ||||||
|  |      * [x0, y0, x1, y1, ...]. | ||||||
|  |      * | ||||||
|  |      * @param destPoints destination array (may be the same as source array) | ||||||
|  |      * @param srcPoints source array | ||||||
|  |      * @param numPoints number of points to map | ||||||
|  |      */ | ||||||
|  |     private fun mapRelativeToAbsolute( | ||||||
|  |         destPoints: FloatArray, | ||||||
|  |         srcPoints: FloatArray, | ||||||
|  |         numPoints: Int | ||||||
|  |     ) { | ||||||
|  |         for (i in 0 until numPoints) { | ||||||
|  |             destPoints[i * 2] = srcPoints[i * 2] * mImageBounds.width() + mImageBounds.left | ||||||
|  |             destPoints[i * 2 + 1] = srcPoints[i * 2 + 1] * mImageBounds.height() + mImageBounds.top | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Zooms to the desired scale and positions the image so that the given image point | ||||||
|  |      * corresponds to the given view point. | ||||||
|  |      * | ||||||
|  |      * @param scale desired scale, will be limited to {min, max} scale factor | ||||||
|  |      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) | ||||||
|  |      * @param viewPoint 2D point in view's absolute coordinate system | ||||||
|  |      */ | ||||||
|  |     open fun zoomToPoint(scale: Float, imagePoint: PointF, viewPoint: PointF) { | ||||||
|  |         FLog.v(TAG, "zoomToPoint") | ||||||
|  |         calculateZoomToPointTransform(mActiveTransform, scale, imagePoint, viewPoint, LIMIT_ALL) | ||||||
|  |         onTransformChanged() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Calculates the zoom transformation that would zoom to the desired scale and position | ||||||
|  |      * the image so that the given image point corresponds to the given view point. | ||||||
|  |      * | ||||||
|  |      * @param outTransform the matrix to store the result to | ||||||
|  |      * @param scale desired scale, will be limited to {min, max} scale factor | ||||||
|  |      * @param imagePoint 2D point in image's relative coordinate system (i.e. 0 <= x, y <= 1) | ||||||
|  |      * @param viewPoint 2D point in view's absolute coordinate system | ||||||
|  |      * @param limitFlags whether to limit translation and/or scale. | ||||||
|  |      * @return whether or not the transform has been corrected due to limitation | ||||||
|  |      */ | ||||||
|  |     protected fun calculateZoomToPointTransform( | ||||||
|  |         outTransform: Matrix, | ||||||
|  |         scale: Float, | ||||||
|  |         imagePoint: PointF, | ||||||
|  |         viewPoint: PointF, | ||||||
|  |         @LimitFlag limitFlags: Int | ||||||
|  |     ): Boolean { | ||||||
|  |         val viewAbsolute = mTempValues | ||||||
|  |         viewAbsolute[0] = imagePoint.x | ||||||
|  |         viewAbsolute[1] = imagePoint.y | ||||||
|  |         mapRelativeToAbsolute(viewAbsolute, viewAbsolute, 1) | ||||||
|  |         val distanceX = viewPoint.x - viewAbsolute[0] | ||||||
|  |         val distanceY = viewPoint.y - viewAbsolute[1] | ||||||
|  |         var transformCorrected = false | ||||||
|  |         outTransform.setScale(scale, scale, viewAbsolute[0], viewAbsolute[1]) | ||||||
|  |         transformCorrected = transformCorrected or | ||||||
|  |                 limitScale(outTransform, viewAbsolute[0], viewAbsolute[1], limitFlags) | ||||||
|  |         outTransform.postTranslate(distanceX, distanceY) | ||||||
|  |         transformCorrected = transformCorrected or limitTranslation(outTransform, limitFlags) | ||||||
|  |         return transformCorrected | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Sets a new zoom transformation. */ | ||||||
|  |     fun setTransform(newTransform: Matrix) { | ||||||
|  |         FLog.v(TAG, "setTransform") | ||||||
|  |         mActiveTransform.set(newTransform) | ||||||
|  |         onTransformChanged() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Gets the gesture detector. */ | ||||||
|  |     protected fun getDetector(): TransformGestureDetector { | ||||||
|  |         return mGestureDetector | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Notifies controller of the received touch event. */ | ||||||
|  |     override fun onTouchEvent(event: MotionEvent): Boolean { | ||||||
|  |         FLog.v(TAG, "onTouchEvent: action: ", event.action) | ||||||
|  |         return if (mIsEnabled && mIsGestureZoomEnabled) { | ||||||
|  |             mGestureDetector.onTouchEvent(event) | ||||||
|  |         } else { | ||||||
|  |             false | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /* TransformGestureDetector.Listener methods  */ | ||||||
|  | 
 | ||||||
|  |     override fun onGestureBegin(detector: TransformGestureDetector) { | ||||||
|  |         FLog.v(TAG, "onGestureBegin") | ||||||
|  |         mPreviousTransform.set(mActiveTransform) | ||||||
|  |         onTransformBegin() | ||||||
|  |         mWasTransformCorrected = !canScrollInAllDirection() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onGestureUpdate(detector: TransformGestureDetector) { | ||||||
|  |         FLog.v(TAG, "onGestureUpdate") | ||||||
|  |         val transformCorrected = calculateGestureTransform(mActiveTransform, LIMIT_ALL) | ||||||
|  |         onTransformChanged() | ||||||
|  |         if (transformCorrected) { | ||||||
|  |             mGestureDetector.restartGesture() | ||||||
|  |         } | ||||||
|  |         mWasTransformCorrected = transformCorrected | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onGestureEnd(detector: TransformGestureDetector) { | ||||||
|  |         FLog.v(TAG, "onGestureEnd") | ||||||
|  |         onTransformEnd() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Calculates the zoom transformation based on the current gesture. | ||||||
|  |      * | ||||||
|  |      * @param outTransform the matrix to store the result to | ||||||
|  |      * @param limitTypes whether to limit translation and/or scale. | ||||||
|  |      * @return whether or not the transform has been corrected due to limitation | ||||||
|  |      */ | ||||||
|  |     private fun calculateGestureTransform( | ||||||
|  |         outTransform: Matrix, | ||||||
|  |         @LimitFlag limitTypes: Int | ||||||
|  |     ): Boolean { | ||||||
|  |         val detector = mGestureDetector | ||||||
|  |         var transformCorrected = false | ||||||
|  |         outTransform.set(mPreviousTransform) | ||||||
|  |         if (mIsRotationEnabled) { | ||||||
|  |             val angle = detector.getRotation() * (180 / Math.PI).toFloat() | ||||||
|  |             outTransform.postRotate(angle, detector.getPivotX(), detector.getPivotY()) | ||||||
|  |         } | ||||||
|  |         if (mIsScaleEnabled) { | ||||||
|  |             val scale = detector.getScale() | ||||||
|  |             outTransform.postScale(scale, scale, detector.getPivotX(), detector.getPivotY()) | ||||||
|  |         } | ||||||
|  |         transformCorrected = transformCorrected or limitScale( | ||||||
|  |             outTransform, | ||||||
|  |             detector.getPivotX(), | ||||||
|  |             detector.getPivotY(), | ||||||
|  |             limitTypes | ||||||
|  |         ) | ||||||
|  |         if (mIsTranslationEnabled) { | ||||||
|  |             outTransform.postTranslate(detector.getTranslationX(), detector.getTranslationY()) | ||||||
|  |         } | ||||||
|  |         transformCorrected = transformCorrected or limitTranslation(outTransform, limitTypes) | ||||||
|  |         return transformCorrected | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onTransformBegin() { | ||||||
|  |         if (mListener != null && isEnabled()) { | ||||||
|  |             mListener?.onTransformBegin(mActiveTransform) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onTransformChanged() { | ||||||
|  |         mActiveTransform.mapRect(mTransformedImageBounds, mImageBounds) | ||||||
|  |         if (mListener != null && isEnabled()) { | ||||||
|  |             mListener?.onTransformChanged(mActiveTransform) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onTransformEnd() { | ||||||
|  |         if (mListener != null && isEnabled()) { | ||||||
|  |             mListener?.onTransformEnd(mActiveTransform) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Keeps the scaling factor within the specified limits. | ||||||
|  |      * | ||||||
|  |      * @param pivotX x coordinate of the pivot point | ||||||
|  |      * @param pivotY y coordinate of the pivot point | ||||||
|  |      * @param limitTypes whether to limit scale. | ||||||
|  |      * @return whether limiting has been applied or not | ||||||
|  |      */ | ||||||
|  |     private fun limitScale( | ||||||
|  |         transform: Matrix, pivotX: Float, pivotY: Float, @LimitFlag limitTypes: Int | ||||||
|  |     ): Boolean { | ||||||
|  |         if (!shouldLimit(limitTypes, LIMIT_SCALE)) { | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |         val currentScale = getMatrixScaleFactor(transform) | ||||||
|  |         val targetScale = limit(currentScale, mMinScaleFactor, mMaxScaleFactor) | ||||||
|  |         return if (targetScale != currentScale) { | ||||||
|  |             val scale = targetScale / currentScale | ||||||
|  |             transform.postScale(scale, scale, pivotX, pivotY) | ||||||
|  |             true | ||||||
|  |         } else { | ||||||
|  |             false | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Limits the translation so that there are no empty spaces on the sides if possible. | ||||||
|  |      */ | ||||||
|  |     private fun limitTranslation(transform: Matrix, @LimitFlag limitTypes: Int): Boolean { | ||||||
|  |         if (!shouldLimit(limitTypes, LIMIT_TRANSLATION_X or LIMIT_TRANSLATION_Y)) { | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |         val b = mTempRect | ||||||
|  |         b.set(mImageBounds) | ||||||
|  |         transform.mapRect(b) | ||||||
|  |         val offsetLeft = | ||||||
|  |             if (shouldLimit(limitTypes, LIMIT_TRANSLATION_X)) getOffset( | ||||||
|  |                 b.left, b.right, mViewBounds.left, mViewBounds.right, mImageBounds.centerX() | ||||||
|  |             ) else 0f | ||||||
|  |         val offsetTop = | ||||||
|  |             if (shouldLimit(limitTypes, LIMIT_TRANSLATION_Y)) getOffset( | ||||||
|  |                 b.top, b.bottom, mViewBounds.top, mViewBounds.bottom, mImageBounds.centerY() | ||||||
|  |             ) else 0f | ||||||
|  | 
 | ||||||
|  |         return if (offsetLeft != 0f || offsetTop != 0f) { | ||||||
|  |             transform.postTranslate(offsetLeft, offsetTop) | ||||||
|  |             true | ||||||
|  |         } else { | ||||||
|  |             false | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks whether the specified limit flag is present in the limits provided. | ||||||
|  |      */ | ||||||
|  |     private fun shouldLimit(@LimitFlag limits: Int, @LimitFlag flag: Int): Boolean { | ||||||
|  |         return (limits and flag) != LIMIT_NONE | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the offset necessary to make sure that: | ||||||
|  |      * - The image is centered if it's smaller than the limit | ||||||
|  |      * - There is no empty space if the image is bigger than the limit | ||||||
|  |      */ | ||||||
|  |     private fun getOffset( | ||||||
|  |         imageStart: Float, imageEnd: Float, limitStart: Float, limitEnd: Float, limitCenter: Float | ||||||
|  |     ): Float { | ||||||
|  |         val imageWidth = imageEnd - imageStart | ||||||
|  |         val limitWidth = limitEnd - limitStart | ||||||
|  |         val limitInnerWidth = minOf(limitCenter - limitStart, limitEnd - limitCenter) * 2 | ||||||
|  | 
 | ||||||
|  |         return when { | ||||||
|  |             imageWidth < limitInnerWidth -> limitCenter - (imageEnd + imageStart) / 2 | ||||||
|  |             imageWidth < limitWidth -> if (limitCenter < (limitStart + limitEnd) / 2) { | ||||||
|  |                 limitStart - imageStart | ||||||
|  |             } else { | ||||||
|  |                 limitEnd - imageEnd | ||||||
|  |             } | ||||||
|  |             imageStart > limitStart -> limitStart - imageStart | ||||||
|  |             imageEnd < limitEnd -> limitEnd - imageEnd | ||||||
|  |             else -> 0f | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Limits the value to the given min and max range. */ | ||||||
|  |     private fun limit(value: Float, min: Float, max: Float): Float { | ||||||
|  |         return min.coerceAtLeast(value).coerceAtMost(max) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the scale factor for the given matrix. Assumes equal scaling for X and Y axis. | ||||||
|  |      */ | ||||||
|  |     private fun getMatrixScaleFactor(transform: Matrix): Float { | ||||||
|  |         transform.getValues(mTempValues) | ||||||
|  |         return mTempValues[Matrix.MSCALE_X] | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Checks if the matrix is an identity matrix within a given tolerance `eps`. */ | ||||||
|  |     private fun isMatrixIdentity(transform: Matrix, eps: Float): Boolean { | ||||||
|  |         transform.getValues(mTempValues) | ||||||
|  |         mTempValues[0] -= 1.0f // m00 | ||||||
|  |         mTempValues[4] -= 1.0f // m11 | ||||||
|  |         mTempValues[8] -= 1.0f // m22 | ||||||
|  |         return mTempValues.all { abs(it) <= eps } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Returns whether the scroll can happen in all directions. */ | ||||||
|  |     private fun canScrollInAllDirection(): Boolean { | ||||||
|  |         return mTransformedImageBounds.left < mViewBounds.left - EPS && | ||||||
|  |                 mTransformedImageBounds.top < mViewBounds.top - EPS && | ||||||
|  |                 mTransformedImageBounds.right > mViewBounds.right + EPS && | ||||||
|  |                 mTransformedImageBounds.bottom > mViewBounds.bottom + EPS | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeHorizontalScrollRange(): Int { | ||||||
|  |         return mTransformedImageBounds.width().toInt() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeHorizontalScrollOffset(): Int { | ||||||
|  |         return (mViewBounds.left - mTransformedImageBounds.left).toInt() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeHorizontalScrollExtent(): Int { | ||||||
|  |         return mViewBounds.width().toInt() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeVerticalScrollRange(): Int { | ||||||
|  |         return mTransformedImageBounds.height().toInt() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeVerticalScrollOffset(): Int { | ||||||
|  |         return (mViewBounds.top - mTransformedImageBounds.top).toInt() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeVerticalScrollExtent(): Int { | ||||||
|  |         return mViewBounds.height().toInt() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun getListener(): ZoomableController.Listener? { | ||||||
|  |         return mListener | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,77 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.graphics.PointF; |  | ||||||
| import android.view.GestureDetector; |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Tap gesture listener for double tap to zoom / unzoom and double-tap-and-drag to zoom. |  | ||||||
|  * |  | ||||||
|  * @see ZoomableDraweeView#setTapListener(GestureDetector.SimpleOnGestureListener) |  | ||||||
|  */ |  | ||||||
| public class DoubleTapGestureListener extends GestureDetector.SimpleOnGestureListener { |  | ||||||
|     private static final int DURATION_MS = 300; |  | ||||||
|     private static final int DOUBLE_TAP_SCROLL_THRESHOLD = 20; |  | ||||||
| 
 |  | ||||||
|     private final ZoomableDraweeView mDraweeView; |  | ||||||
|     private final PointF mDoubleTapViewPoint = new PointF(); |  | ||||||
|     private final PointF mDoubleTapImagePoint = new PointF(); |  | ||||||
|     private float mDoubleTapScale = 1; |  | ||||||
|     private boolean mDoubleTapScroll = false; |  | ||||||
| 
 |  | ||||||
|     public DoubleTapGestureListener(ZoomableDraweeView zoomableDraweeView) { |  | ||||||
|         mDraweeView = zoomableDraweeView; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onDoubleTapEvent(MotionEvent e) { |  | ||||||
|         AbstractAnimatedZoomableController zc = |  | ||||||
|                 (AbstractAnimatedZoomableController) mDraweeView.getZoomableController(); |  | ||||||
|         PointF vp = new PointF(e.getX(), e.getY()); |  | ||||||
|         PointF ip = zc.mapViewToImage(vp); |  | ||||||
|         switch (e.getActionMasked()) { |  | ||||||
|             case MotionEvent.ACTION_DOWN: |  | ||||||
|                 mDoubleTapViewPoint.set(vp); |  | ||||||
|                 mDoubleTapImagePoint.set(ip); |  | ||||||
|                 mDoubleTapScale = zc.getScaleFactor(); |  | ||||||
|                 break; |  | ||||||
|             case MotionEvent.ACTION_MOVE: |  | ||||||
|                 mDoubleTapScroll = mDoubleTapScroll || shouldStartDoubleTapScroll(vp); |  | ||||||
|                 if (mDoubleTapScroll) { |  | ||||||
|                     float scale = calcScale(vp); |  | ||||||
|                     zc.zoomToPoint(scale, mDoubleTapImagePoint, mDoubleTapViewPoint); |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             case MotionEvent.ACTION_UP: |  | ||||||
|                 if (mDoubleTapScroll) { |  | ||||||
|                     float scale = calcScale(vp); |  | ||||||
|                     zc.zoomToPoint(scale, mDoubleTapImagePoint, mDoubleTapViewPoint); |  | ||||||
|                 } else { |  | ||||||
|                     final float maxScale = zc.getMaxScaleFactor(); |  | ||||||
|                     final float minScale = zc.getMinScaleFactor(); |  | ||||||
|                     if (zc.getScaleFactor() < (maxScale + minScale) / 2) { |  | ||||||
|                         zc.zoomToPoint( |  | ||||||
|                                 maxScale, ip, vp, DefaultZoomableController.LIMIT_ALL, DURATION_MS, null); |  | ||||||
|                     } else { |  | ||||||
|                         zc.zoomToPoint( |  | ||||||
|                                 minScale, ip, vp, DefaultZoomableController.LIMIT_ALL, DURATION_MS, null); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 mDoubleTapScroll = false; |  | ||||||
|                 break; |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private boolean shouldStartDoubleTapScroll(PointF viewPoint) { |  | ||||||
|         double dist = |  | ||||||
|                 Math.hypot(viewPoint.x - mDoubleTapViewPoint.x, viewPoint.y - mDoubleTapViewPoint.y); |  | ||||||
|         return dist > DOUBLE_TAP_SCROLL_THRESHOLD; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private float calcScale(PointF currentViewPoint) { |  | ||||||
|         float dy = (currentViewPoint.y - mDoubleTapViewPoint.y); |  | ||||||
|         float t = 1 + Math.abs(dy) * 0.001f; |  | ||||||
|         return (dy < 0) ? mDoubleTapScale / t : mDoubleTapScale * t; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,85 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.graphics.PointF | ||||||
|  | import android.view.GestureDetector | ||||||
|  | import android.view.MotionEvent | ||||||
|  | import kotlin.math.abs | ||||||
|  | import kotlin.math.hypot | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Tap gesture listener for double tap to zoom/unzoom and double-tap-and-drag to zoom. | ||||||
|  |  * | ||||||
|  |  * @see ZoomableDraweeView.setTapListener | ||||||
|  |  */ | ||||||
|  | class DoubleTapGestureListener(private val draweeView: ZoomableDraweeView) : | ||||||
|  |     GestureDetector.SimpleOnGestureListener() { | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private const val DURATION_MS = 300L | ||||||
|  |         private const val DOUBLE_TAP_SCROLL_THRESHOLD = 20 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private val doubleTapViewPoint = PointF() | ||||||
|  |     private val doubleTapImagePoint = PointF() | ||||||
|  |     private var doubleTapScale = 1f | ||||||
|  |     private var doubleTapScroll = false | ||||||
|  | 
 | ||||||
|  |     override fun onDoubleTapEvent(e: MotionEvent): Boolean { | ||||||
|  |         val zc = draweeView.getZoomableController() as AbstractAnimatedZoomableController | ||||||
|  |         val vp = PointF(e.x, e.y) | ||||||
|  |         val ip = zc.mapViewToImage(vp) | ||||||
|  | 
 | ||||||
|  |         when (e.actionMasked) { | ||||||
|  |             MotionEvent.ACTION_DOWN -> { | ||||||
|  |                 doubleTapViewPoint.set(vp) | ||||||
|  |                 doubleTapImagePoint.set(ip) | ||||||
|  |                 doubleTapScale = zc.getScaleFactor() | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             MotionEvent.ACTION_MOVE -> { | ||||||
|  |                 doubleTapScroll = doubleTapScroll || shouldStartDoubleTapScroll(vp) | ||||||
|  |                 if (doubleTapScroll) { | ||||||
|  |                     val scale = calcScale(vp) | ||||||
|  |                     zc.zoomToPoint(scale, doubleTapImagePoint, doubleTapViewPoint) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             MotionEvent.ACTION_UP -> { | ||||||
|  |                 if (doubleTapScroll) { | ||||||
|  |                     val scale = calcScale(vp) | ||||||
|  |                     zc.zoomToPoint(scale, doubleTapImagePoint, doubleTapViewPoint) | ||||||
|  |                 } else { | ||||||
|  |                     val maxScale = zc.getMaxScaleFactor() | ||||||
|  |                     val minScale = zc.getMinScaleFactor() | ||||||
|  |                     val targetScale = | ||||||
|  |                         if (zc.getScaleFactor() < (maxScale + minScale) / 2) maxScale else minScale | ||||||
|  | 
 | ||||||
|  |                     zc.zoomToPoint( | ||||||
|  |                         targetScale, | ||||||
|  |                         ip, | ||||||
|  |                         vp, | ||||||
|  |                         DefaultZoomableController.LIMIT_ALL, | ||||||
|  |                         DURATION_MS, | ||||||
|  |                         null | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |                 doubleTapScroll = false | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun shouldStartDoubleTapScroll(viewPoint: PointF): Boolean { | ||||||
|  |         val dist = hypot( | ||||||
|  |             (viewPoint.x - doubleTapViewPoint.x).toDouble(), | ||||||
|  |             (viewPoint.y - doubleTapViewPoint.y).toDouble() | ||||||
|  |         ) | ||||||
|  |         return dist > DOUBLE_TAP_SCROLL_THRESHOLD | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun calcScale(currentViewPoint: PointF): Float { | ||||||
|  |         val dy = currentViewPoint.y - doubleTapViewPoint.y | ||||||
|  |         val t = 1 + abs(dy) * 0.001f | ||||||
|  |         return if (dy < 0) doubleTapScale / t else doubleTapScale * t | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,63 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.view.GestureDetector; |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| 
 |  | ||||||
| /** Wrapper for SimpleOnGestureListener as GestureDetector does not allow changing its listener. */ |  | ||||||
| public class GestureListenerWrapper extends GestureDetector.SimpleOnGestureListener { |  | ||||||
| 
 |  | ||||||
|     private GestureDetector.SimpleOnGestureListener mDelegate; |  | ||||||
| 
 |  | ||||||
|     public GestureListenerWrapper() { |  | ||||||
|         mDelegate = new GestureDetector.SimpleOnGestureListener(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setListener(GestureDetector.SimpleOnGestureListener listener) { |  | ||||||
|         mDelegate = listener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onLongPress(MotionEvent e) { |  | ||||||
|         mDelegate.onLongPress(e); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { |  | ||||||
|         return mDelegate.onScroll(e1, e2, distanceX, distanceY); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { |  | ||||||
|         return mDelegate.onFling(e1, e2, velocityX, velocityY); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onShowPress(MotionEvent e) { |  | ||||||
|         mDelegate.onShowPress(e); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onDown(MotionEvent e) { |  | ||||||
|         return mDelegate.onDown(e); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onDoubleTap(MotionEvent e) { |  | ||||||
|         return mDelegate.onDoubleTap(e); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onDoubleTapEvent(MotionEvent e) { |  | ||||||
|         return mDelegate.onDoubleTapEvent(e); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onSingleTapConfirmed(MotionEvent e) { |  | ||||||
|         return mDelegate.onSingleTapConfirmed(e); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onSingleTapUp(MotionEvent e) { |  | ||||||
|         return mDelegate.onSingleTapUp(e); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,61 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.view.GestureDetector | ||||||
|  | import android.view.MotionEvent | ||||||
|  | 
 | ||||||
|  | /** Wrapper for SimpleOnGestureListener as GestureDetector does not allow changing its listener. */ | ||||||
|  | class GestureListenerWrapper : GestureDetector.SimpleOnGestureListener() { | ||||||
|  | 
 | ||||||
|  |     private var delegate: GestureDetector.SimpleOnGestureListener = | ||||||
|  |         GestureDetector.SimpleOnGestureListener() | ||||||
|  | 
 | ||||||
|  |     fun setListener(listener: GestureDetector.SimpleOnGestureListener) { | ||||||
|  |         delegate = listener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onLongPress(e: MotionEvent) { | ||||||
|  |         delegate.onLongPress(e) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onScroll( | ||||||
|  |         e1: MotionEvent?, | ||||||
|  |         e2: MotionEvent, | ||||||
|  |         distanceX: Float, | ||||||
|  |         distanceY: Float | ||||||
|  |     ): Boolean { | ||||||
|  |         return delegate.onScroll(e1, e2, distanceX, distanceY) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onFling( | ||||||
|  |         e1: MotionEvent?, | ||||||
|  |         e2: MotionEvent, | ||||||
|  |         velocityX: Float, | ||||||
|  |         velocityY: Float | ||||||
|  |     ): Boolean { | ||||||
|  |         return delegate.onFling(e1, e2, velocityX, velocityY) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onShowPress(e: MotionEvent) { | ||||||
|  |         delegate.onShowPress(e) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDown(e: MotionEvent): Boolean { | ||||||
|  |         return delegate.onDown(e) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDoubleTap(e: MotionEvent): Boolean { | ||||||
|  |         return delegate.onDoubleTap(e) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDoubleTapEvent(e: MotionEvent): Boolean { | ||||||
|  |         return delegate.onDoubleTapEvent(e) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onSingleTapConfirmed(e: MotionEvent): Boolean { | ||||||
|  |         return delegate.onSingleTapConfirmed(e) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||||
|  |         return delegate.onSingleTapUp(e) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,148 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.view.GestureDetector; |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Gesture listener that allows multiple child listeners to be added and notified about gesture |  | ||||||
|  * events. |  | ||||||
|  * |  | ||||||
|  * NOTE: The order of the listeners is important. Listeners can consume gesture events. For |  | ||||||
|  * example, if one of the child listeners consumes {@link #onLongPress(MotionEvent)} (the listener |  | ||||||
|  * returned true), subsequent listeners will not be notified about the event any more since it has |  | ||||||
|  * been consumed. |  | ||||||
|  */ |  | ||||||
| public class MultiGestureListener extends GestureDetector.SimpleOnGestureListener { |  | ||||||
| 
 |  | ||||||
|     private final List<GestureDetector.SimpleOnGestureListener> mListeners = new ArrayList<>(); |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Adds a listener to the multi gesture listener. |  | ||||||
|      * |  | ||||||
|      * <p>NOTE: The order of the listeners is important since gesture events can be consumed. |  | ||||||
|      * |  | ||||||
|      * @param listener the listener to be added |  | ||||||
|      */ |  | ||||||
|     public synchronized void addListener(GestureDetector.SimpleOnGestureListener listener) { |  | ||||||
|         mListeners.add(listener); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Removes the given listener so that it will not be notified about future events. |  | ||||||
|      * |  | ||||||
|      * <p>NOTE: The order of the listeners is important since gesture events can be consumed. |  | ||||||
|      * |  | ||||||
|      * @param listener the listener to remove |  | ||||||
|      */ |  | ||||||
|     public synchronized void removeListener(GestureDetector.SimpleOnGestureListener listener) { |  | ||||||
|         mListeners.remove(listener); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onSingleTapUp(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onSingleTapUp(e)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized void onLongPress(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             mListeners.get(i).onLongPress(e); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onScroll( |  | ||||||
|             MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onScroll(e1, e2, distanceX, distanceY)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onFling( |  | ||||||
|             MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onFling(e1, e2, velocityX, velocityY)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized void onShowPress(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             mListeners.get(i).onShowPress(e); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onDown(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onDown(e)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onDoubleTap(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onDoubleTap(e)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onDoubleTapEvent(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onDoubleTapEvent(e)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onSingleTapConfirmed(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onSingleTapConfirmed(e)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized boolean onContextClick(MotionEvent e) { |  | ||||||
|         final int size = mListeners.size(); |  | ||||||
|         for (int i = 0; i < size; i++) { |  | ||||||
|             if (mListeners.get(i).onContextClick(e)) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,150 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.os.Build | ||||||
|  | import android.view.GestureDetector | ||||||
|  | import android.view.MotionEvent | ||||||
|  | import androidx.annotation.RequiresApi | ||||||
|  | import java.util.Collections.synchronizedList | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Gesture listener that allows multiple child listeners to be added and notified about gesture | ||||||
|  |  * events. | ||||||
|  |  * | ||||||
|  |  * NOTE: The order of the listeners is important. Listeners can consume gesture events. For | ||||||
|  |  * example, if one of the child listeners consumes [onLongPress] (the listener returned true), | ||||||
|  |  * subsequent listeners will not be notified about the event anymore since it has been consumed. | ||||||
|  |  */ | ||||||
|  | class MultiGestureListener : GestureDetector.SimpleOnGestureListener() { | ||||||
|  | 
 | ||||||
|  |     private val listeners: MutableList<GestureDetector.SimpleOnGestureListener> = | ||||||
|  |         synchronizedList(mutableListOf()) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Adds a listener to the multi-gesture listener. | ||||||
|  |      * | ||||||
|  |      * NOTE: The order of the listeners is important since gesture events can be consumed. | ||||||
|  |      * | ||||||
|  |      * @param listener the listener to be added | ||||||
|  |      */ | ||||||
|  |     @Synchronized | ||||||
|  |     fun addListener(listener: GestureDetector.SimpleOnGestureListener) { | ||||||
|  |         listeners.add(listener) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Removes the given listener so that it will not be notified about future events. | ||||||
|  |      * | ||||||
|  |      * NOTE: The order of the listeners is important since gesture events can be consumed. | ||||||
|  |      * | ||||||
|  |      * @param listener the listener to remove | ||||||
|  |      */ | ||||||
|  |     @Synchronized | ||||||
|  |     fun removeListener(listener: GestureDetector.SimpleOnGestureListener) { | ||||||
|  |         listeners.remove(listener) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onSingleTapUp(e)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onLongPress(e: MotionEvent) { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             listener.onLongPress(e) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onScroll( | ||||||
|  |         e1: MotionEvent?, | ||||||
|  |         e2: MotionEvent, | ||||||
|  |         distanceX: Float, | ||||||
|  |         distanceY: Float | ||||||
|  |     ): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onScroll(e1, e2, distanceX, distanceY)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onFling( | ||||||
|  |         e1: MotionEvent?, | ||||||
|  |         e2: MotionEvent, | ||||||
|  |         velocityX: Float, | ||||||
|  |         velocityY: Float | ||||||
|  |     ): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onFling(e1, e2, velocityX, velocityY)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onShowPress(e: MotionEvent) { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             listener.onShowPress(e) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onDown(e: MotionEvent): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onDown(e)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onDoubleTap(e: MotionEvent): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onDoubleTap(e)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onDoubleTapEvent(e: MotionEvent): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onDoubleTapEvent(e)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onSingleTapConfirmed(e: MotionEvent): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onSingleTapConfirmed(e)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @RequiresApi(Build.VERSION_CODES.M) | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onContextClick(e: MotionEvent): Boolean { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             if (listener.onContextClick(e)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,40 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.graphics.Matrix; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| public class MultiZoomableControllerListener implements ZoomableController.Listener { |  | ||||||
| 
 |  | ||||||
|     private final List<ZoomableController.Listener> mListeners = new ArrayList<>(); |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized void onTransformBegin(Matrix transform) { |  | ||||||
|         for (ZoomableController.Listener listener : mListeners) { |  | ||||||
|             listener.onTransformBegin(transform); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized void onTransformChanged(Matrix transform) { |  | ||||||
|         for (ZoomableController.Listener listener : mListeners) { |  | ||||||
|             listener.onTransformChanged(transform); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public synchronized void onTransformEnd(Matrix transform) { |  | ||||||
|         for (ZoomableController.Listener listener : mListeners) { |  | ||||||
|             listener.onTransformEnd(transform); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public synchronized void addListener(ZoomableController.Listener listener) { |  | ||||||
|         mListeners.add(listener); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public synchronized void removeListener(ZoomableController.Listener listener) { |  | ||||||
|         mListeners.remove(listener); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,46 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.graphics.Matrix | ||||||
|  | import java.util.ArrayList | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * MultiZoomableControllerListener that allows multiple listeners to be added and notified about | ||||||
|  |  * transform events. | ||||||
|  |  * | ||||||
|  |  * NOTE: The order of the listeners is important. Listeners can consume transform events. | ||||||
|  |  */ | ||||||
|  | class MultiZoomableControllerListener : ZoomableController.Listener { | ||||||
|  | 
 | ||||||
|  |     private val listeners: MutableList<ZoomableController.Listener> = mutableListOf() | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onTransformBegin(transform: Matrix) { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             listener.onTransformBegin(transform) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onTransformChanged(transform: Matrix) { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             listener.onTransformChanged(transform) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     override fun onTransformEnd(transform: Matrix) { | ||||||
|  |         for (listener in listeners) { | ||||||
|  |             listener.onTransformEnd(transform) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     fun addListener(listener: ZoomableController.Listener) { | ||||||
|  |         listeners.add(listener) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Synchronized | ||||||
|  |     fun removeListener(listener: ZoomableController.Listener) { | ||||||
|  |         listeners.remove(listener) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,14 +1,14 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
| 
 | 
 | ||||||
| import android.graphics.Matrix; | import android.graphics.Matrix | ||||||
| import android.graphics.RectF; | import android.graphics.RectF | ||||||
| import android.view.MotionEvent; | import android.view.MotionEvent | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Interface for implementing a controller that works with {@link ZoomableDraweeView} to control the |  * Interface for implementing a controller that works with [ZoomableDraweeView] to control the | ||||||
|  * zoom. |  * zoom. | ||||||
|  */ |  */ | ||||||
| public interface ZoomableController { | interface ZoomableController { | ||||||
| 
 | 
 | ||||||
|     /** Listener interface. */ |     /** Listener interface. */ | ||||||
|     interface Listener { |     interface Listener { | ||||||
|  | @ -18,21 +18,21 @@ public interface ZoomableController { | ||||||
|          * |          * | ||||||
|          * @param transform the current transform matrix |          * @param transform the current transform matrix | ||||||
|          */ |          */ | ||||||
|         void onTransformBegin(Matrix transform); |         fun onTransformBegin(transform: Matrix) | ||||||
| 
 | 
 | ||||||
|         /** |         /** | ||||||
|          * Notifies the view that the transform changed. |          * Notifies the view that the transform changed. | ||||||
|          * |          * | ||||||
|          * @param transform the new matrix |          * @param transform the new matrix | ||||||
|          */ |          */ | ||||||
|         void onTransformChanged(Matrix transform); |         fun onTransformChanged(transform: Matrix) | ||||||
| 
 | 
 | ||||||
|         /** |         /** | ||||||
|          * Notifies the view that the transform ended. |          * Notifies the view that the transform ended. | ||||||
|          * |          * | ||||||
|          * @param transform the current transform matrix |          * @param transform the current transform matrix | ||||||
|          */ |          */ | ||||||
|         void onTransformEnd(Matrix transform); |         fun onTransformEnd(transform: Matrix) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -40,22 +40,21 @@ public interface ZoomableController { | ||||||
|      * |      * | ||||||
|      * @param enabled whether to enable the controller |      * @param enabled whether to enable the controller | ||||||
|      */ |      */ | ||||||
|     void setEnabled(boolean enabled); |     fun setEnabled(enabled: Boolean) | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Gets whether the controller is enabled. This should return the last value passed to {@link |      * Gets whether the controller is enabled. This should return the last value passed | ||||||
|      * #setEnabled}. |      * to [setEnabled]. | ||||||
|      * |  | ||||||
|      * @return whether the controller is enabled. |      * @return whether the controller is enabled. | ||||||
|      */ |      */ | ||||||
|     boolean isEnabled(); |     fun isEnabled(): Boolean | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the listener for the controller to call back when the matrix changes. |      * Sets the listener for the controller to call back when the matrix changes. | ||||||
|      * |      * | ||||||
|      * @param listener the listener |      * @param listener the listener | ||||||
|      */ |      */ | ||||||
|     void setListener(Listener listener); |     fun setListener(listener: Listener?) | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Gets the current scale factor. A convenience method for calculating the scale from the |      * Gets the current scale factor. A convenience method for calculating the scale from the | ||||||
|  | @ -63,10 +62,10 @@ public interface ZoomableController { | ||||||
|      * |      * | ||||||
|      * @return the current scale factor |      * @return the current scale factor | ||||||
|      */ |      */ | ||||||
|     float getScaleFactor(); |     fun getScaleFactor(): Float | ||||||
| 
 | 
 | ||||||
|     /** Returns true if the zoomable transform is identity matrix, and the controller is idle. */ |     /** Returns true if the zoomable transform is identity matrix, and the controller is idle. */ | ||||||
|     boolean isIdentity(); |     fun isIdentity(): Boolean | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Returns true if the transform was corrected during the last update. |      * Returns true if the transform was corrected during the last update. | ||||||
|  | @ -74,27 +73,27 @@ public interface ZoomableController { | ||||||
|      * <p>This mainly happens when a gesture would cause the image to get out of limits and the |      * <p>This mainly happens when a gesture would cause the image to get out of limits and the | ||||||
|      * transform gets corrected in order to prevent that. |      * transform gets corrected in order to prevent that. | ||||||
|      */ |      */ | ||||||
|     boolean wasTransformCorrected(); |     fun wasTransformCorrected(): Boolean | ||||||
| 
 | 
 | ||||||
|     /** See {@link androidx.core.view.ScrollingView}. */ |     /** See [androidx.core.view.ScrollingView]. */ | ||||||
|     int computeHorizontalScrollRange(); |     fun computeHorizontalScrollRange(): Int | ||||||
| 
 | 
 | ||||||
|     int computeHorizontalScrollOffset(); |     fun computeHorizontalScrollOffset(): Int | ||||||
| 
 | 
 | ||||||
|     int computeHorizontalScrollExtent(); |     fun computeHorizontalScrollExtent(): Int | ||||||
| 
 | 
 | ||||||
|     int computeVerticalScrollRange(); |     fun computeVerticalScrollRange(): Int | ||||||
| 
 | 
 | ||||||
|     int computeVerticalScrollOffset(); |     fun computeVerticalScrollOffset(): Int | ||||||
| 
 | 
 | ||||||
|     int computeVerticalScrollExtent(); |     fun computeVerticalScrollExtent(): Int | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Gets the current transform. |      * Gets the current transform. | ||||||
|      * |      * | ||||||
|      * @return the transform |      * @return the transform | ||||||
|      */ |      */ | ||||||
|     Matrix getTransform(); |     fun getTransform(): Matrix | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the bounds of the image post transform prior to application of the zoomable |      * Sets the bounds of the image post transform prior to application of the zoomable | ||||||
|  | @ -102,14 +101,14 @@ public interface ZoomableController { | ||||||
|      * |      * | ||||||
|      * @param imageBounds the bounds of the image |      * @param imageBounds the bounds of the image | ||||||
|      */ |      */ | ||||||
|     void setImageBounds(RectF imageBounds); |     fun setImageBounds(imageBounds: RectF) | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the bounds of the view. |      * Sets the bounds of the view. | ||||||
|      * |      * | ||||||
|      * @param viewBounds the bounds of the view |      * @param viewBounds the bounds of the view | ||||||
|      */ |      */ | ||||||
|     void setViewBounds(RectF viewBounds); |     fun setViewBounds(viewBounds: RectF) | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Allows the controller to handle a touch event. |      * Allows the controller to handle a touch event. | ||||||
|  | @ -117,5 +116,5 @@ public interface ZoomableController { | ||||||
|      * @param event the touch event |      * @param event the touch event | ||||||
|      * @return whether the controller handled the event |      * @return whether the controller handled the event | ||||||
|      */ |      */ | ||||||
|     boolean onTouchEvent(MotionEvent event); |     fun onTouchEvent(event: MotionEvent): Boolean | ||||||
| } | } | ||||||
|  | @ -1,417 +0,0 @@ | ||||||
| package fr.free.nrw.commons.media.zoomControllers.zoomable; |  | ||||||
| 
 |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.res.Resources; |  | ||||||
| import android.graphics.Canvas; |  | ||||||
| import android.graphics.Matrix; |  | ||||||
| import android.graphics.RectF; |  | ||||||
| import android.graphics.drawable.Animatable; |  | ||||||
| import android.util.AttributeSet; |  | ||||||
| import android.view.GestureDetector; |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.core.view.ScrollingView; |  | ||||||
| import com.facebook.common.internal.Preconditions; |  | ||||||
| import com.facebook.common.logging.FLog; |  | ||||||
| import com.facebook.drawee.controller.AbstractDraweeController; |  | ||||||
| import com.facebook.drawee.controller.BaseControllerListener; |  | ||||||
| import com.facebook.drawee.controller.ControllerListener; |  | ||||||
| import com.facebook.drawee.drawable.ScalingUtils; |  | ||||||
| import com.facebook.drawee.generic.GenericDraweeHierarchy; |  | ||||||
| import com.facebook.drawee.generic.GenericDraweeHierarchyBuilder; |  | ||||||
| import com.facebook.drawee.generic.GenericDraweeHierarchyInflater; |  | ||||||
| import com.facebook.drawee.interfaces.DraweeController; |  | ||||||
| import com.facebook.drawee.view.DraweeView; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * DraweeView that has zoomable capabilities. |  | ||||||
|  * |  | ||||||
|  * <p>Once the image loads, pinch-to-zoom and translation gestures are enabled. |  | ||||||
|  */ |  | ||||||
| public class ZoomableDraweeView extends DraweeView<GenericDraweeHierarchy> |  | ||||||
|         implements ScrollingView { |  | ||||||
| 
 |  | ||||||
|     private static final Class<?> TAG = ZoomableDraweeView.class; |  | ||||||
| 
 |  | ||||||
|     private static final float HUGE_IMAGE_SCALE_FACTOR_THRESHOLD = 1.1f; |  | ||||||
| 
 |  | ||||||
|     private final RectF mImageBounds = new RectF(); |  | ||||||
|     private final RectF mViewBounds = new RectF(); |  | ||||||
| 
 |  | ||||||
|     private DraweeController mHugeImageController; |  | ||||||
|     private ZoomableController mZoomableController; |  | ||||||
|     private GestureDetector mTapGestureDetector; |  | ||||||
|     private boolean mAllowTouchInterceptionWhileZoomed = true; |  | ||||||
| 
 |  | ||||||
|     private boolean mIsDialtoneEnabled = false; |  | ||||||
|     private boolean mZoomingEnabled = true; |  | ||||||
|     private TransformationListener transformationListener; |  | ||||||
| 
 |  | ||||||
|     private final ControllerListener mControllerListener = |  | ||||||
|             new BaseControllerListener<Object>() { |  | ||||||
|                 @Override |  | ||||||
|                 public void onFinalImageSet( |  | ||||||
|                         String id, @Nullable Object imageInfo, @Nullable Animatable animatable) { |  | ||||||
|                     ZoomableDraweeView.this.onFinalImageSet(); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 @Override |  | ||||||
|                 public void onRelease(String id) { |  | ||||||
|                     ZoomableDraweeView.this.onRelease(); |  | ||||||
|                 } |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|     private final ZoomableController.Listener mZoomableListener = |  | ||||||
|             new ZoomableController.Listener() { |  | ||||||
|                 @Override |  | ||||||
|                 public void onTransformBegin(Matrix transform) {} |  | ||||||
| 
 |  | ||||||
|                 @Override |  | ||||||
|                 public void onTransformChanged(Matrix transform) { |  | ||||||
|                     ZoomableDraweeView.this.onTransformChanged(transform); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 @Override |  | ||||||
|                 public void onTransformEnd(Matrix transform) { |  | ||||||
|                     if (null != transformationListener) { |  | ||||||
|                         transformationListener.onTransformationEnd(); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|     public void setTransformationListener( |  | ||||||
|         TransformationListener transformationListener) { |  | ||||||
|         this.transformationListener = transformationListener; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private final GestureListenerWrapper mTapListenerWrapper = new GestureListenerWrapper(); |  | ||||||
| 
 |  | ||||||
|     public ZoomableDraweeView(Context context, GenericDraweeHierarchy hierarchy) { |  | ||||||
|         super(context); |  | ||||||
|         setHierarchy(hierarchy); |  | ||||||
|         init(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public ZoomableDraweeView(Context context) { |  | ||||||
|         super(context); |  | ||||||
|         inflateHierarchy(context, null); |  | ||||||
|         init(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public ZoomableDraweeView(Context context, AttributeSet attrs) { |  | ||||||
|         super(context, attrs); |  | ||||||
|         inflateHierarchy(context, attrs); |  | ||||||
|         init(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public ZoomableDraweeView(Context context, AttributeSet attrs, int defStyle) { |  | ||||||
|         super(context, attrs, defStyle); |  | ||||||
|         inflateHierarchy(context, attrs); |  | ||||||
|         init(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void inflateHierarchy(Context context, @Nullable AttributeSet attrs) { |  | ||||||
|         Resources resources = context.getResources(); |  | ||||||
|         GenericDraweeHierarchyBuilder builder = |  | ||||||
|                 new GenericDraweeHierarchyBuilder(resources) |  | ||||||
|                         .setActualImageScaleType(ScalingUtils.ScaleType.FIT_CENTER); |  | ||||||
|         GenericDraweeHierarchyInflater.updateBuilder(builder, context, attrs); |  | ||||||
|         setAspectRatio(builder.getDesiredAspectRatio()); |  | ||||||
|         setHierarchy(builder.build()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void init() { |  | ||||||
|         mZoomableController = createZoomableController(); |  | ||||||
|         mZoomableController.setListener(mZoomableListener); |  | ||||||
|         mTapGestureDetector = new GestureDetector(getContext(), mTapListenerWrapper); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setIsDialtoneEnabled(boolean isDialtoneEnabled) { |  | ||||||
|         mIsDialtoneEnabled = isDialtoneEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the original image bounds, in view-absolute coordinates. |  | ||||||
|      * |  | ||||||
|      * <p>The original image bounds are those reported by the hierarchy. The hierarchy itself may |  | ||||||
|      * apply scaling on its own (e.g. due to scale type) so the reported bounds are not necessarily |  | ||||||
|      * the same as the actual bitmap dimensions. In other words, the original image bounds correspond |  | ||||||
|      * to the image bounds within this view when no zoomable transformation is applied, but including |  | ||||||
|      * the potential scaling of the hierarchy. Having the actual bitmap dimensions abstracted away |  | ||||||
|      * from this view greatly simplifies implementation because the actual bitmap may change (e.g. |  | ||||||
|      * when a high-res image arrives and replaces the previously set low-res image). With proper |  | ||||||
|      * hierarchy scaling (e.g. FIT_CENTER), this underlying change will not affect this view nor the |  | ||||||
|      * zoomable transformation in any way. |  | ||||||
|      */ |  | ||||||
|     protected void getImageBounds(RectF outBounds) { |  | ||||||
|         getHierarchy().getActualImageBounds(outBounds); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the bounds used to limit the translation, in view-absolute coordinates. |  | ||||||
|      * |  | ||||||
|      * <p>These bounds are passed to the zoomable controller in order to limit the translation. The |  | ||||||
|      * image is attempted to be centered within the limit bounds if the transformed image is smaller. |  | ||||||
|      * There will be no empty spaces within the limit bounds if the transformed image is bigger. This |  | ||||||
|      * applies to each dimension (horizontal and vertical) independently. |  | ||||||
|      * |  | ||||||
|      * <p>Unless overridden by a subclass, these bounds are same as the view bounds. |  | ||||||
|      */ |  | ||||||
|     protected void getLimitBounds(RectF outBounds) { |  | ||||||
|         outBounds.set(0, 0, getWidth(), getHeight()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets a custom zoomable controller, instead of using the default one. */ |  | ||||||
|     public void setZoomableController(ZoomableController zoomableController) { |  | ||||||
|         Preconditions.checkNotNull(zoomableController); |  | ||||||
|         mZoomableController.setListener(null); |  | ||||||
|         mZoomableController = zoomableController; |  | ||||||
|         mZoomableController.setListener(mZoomableListener); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the zoomable controller. |  | ||||||
|      * |  | ||||||
|      * <p>Zoomable controller can be used to zoom to point, or to map point from view to image |  | ||||||
|      * coordinates for instance. |  | ||||||
|      */ |  | ||||||
|     public ZoomableController getZoomableController() { |  | ||||||
|         return mZoomableController; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Check whether the parent view can intercept touch events while zoomed. This can be used, for |  | ||||||
|      * example, to swipe between images in a view pager while zoomed. |  | ||||||
|      * |  | ||||||
|      * @return true if touch events can be intercepted |  | ||||||
|      */ |  | ||||||
|     public boolean allowsTouchInterceptionWhileZoomed() { |  | ||||||
|         return mAllowTouchInterceptionWhileZoomed; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * If this is set to true, parent views can intercept touch events while the view is zoomed. For |  | ||||||
|      * example, this can be used to swipe between images in a view pager while zoomed. |  | ||||||
|      * |  | ||||||
|      * @param allowTouchInterceptionWhileZoomed true if the parent needs to intercept touches |  | ||||||
|      */ |  | ||||||
|     public void setAllowTouchInterceptionWhileZoomed(boolean allowTouchInterceptionWhileZoomed) { |  | ||||||
|         mAllowTouchInterceptionWhileZoomed = allowTouchInterceptionWhileZoomed; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets the tap listener. */ |  | ||||||
|     public void setTapListener(GestureDetector.SimpleOnGestureListener tapListener) { |  | ||||||
|         mTapListenerWrapper.setListener(tapListener); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets whether long-press tap detection is enabled. Unfortunately, long-press conflicts with |  | ||||||
|      * onDoubleTapEvent. |  | ||||||
|      */ |  | ||||||
|     public void setIsLongpressEnabled(boolean enabled) { |  | ||||||
|         mTapGestureDetector.setIsLongpressEnabled(enabled); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setZoomingEnabled(boolean zoomingEnabled) { |  | ||||||
|         mZoomingEnabled = zoomingEnabled; |  | ||||||
|         mZoomableController.setEnabled(false); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** Sets the image controller. */ |  | ||||||
|     @Override |  | ||||||
|     public void setController(@Nullable DraweeController controller) { |  | ||||||
|         setControllers(controller, null); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Sets the controllers for the normal and huge image. |  | ||||||
|      * |  | ||||||
|      * <p>The huge image controller is used after the image gets scaled above a certain threshold. |  | ||||||
|      * |  | ||||||
|      * <p>IMPORTANT: in order to avoid a flicker when switching to the huge image, the huge image |  | ||||||
|      * controller should have the normal-image-uri set as its low-res-uri. |  | ||||||
|      * |  | ||||||
|      * @param controller controller to be initially used |  | ||||||
|      * @param hugeImageController controller to be used after the client starts zooming-in |  | ||||||
|      */ |  | ||||||
|     public void setControllers( |  | ||||||
|             @Nullable DraweeController controller, @Nullable DraweeController hugeImageController) { |  | ||||||
|         setControllersInternal(null, null); |  | ||||||
|         mZoomableController.setEnabled(false); |  | ||||||
|         setControllersInternal(controller, hugeImageController); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void setControllersInternal( |  | ||||||
|             @Nullable DraweeController controller, @Nullable DraweeController hugeImageController) { |  | ||||||
|         removeControllerListener(getController()); |  | ||||||
|         addControllerListener(controller); |  | ||||||
|         mHugeImageController = hugeImageController; |  | ||||||
|         super.setController(controller); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void maybeSetHugeImageController() { |  | ||||||
|         if (mHugeImageController != null |  | ||||||
|                 && mZoomableController.getScaleFactor() > HUGE_IMAGE_SCALE_FACTOR_THRESHOLD) { |  | ||||||
|             setControllersInternal(mHugeImageController, null); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void removeControllerListener(DraweeController controller) { |  | ||||||
|         if (controller instanceof AbstractDraweeController) { |  | ||||||
|             ((AbstractDraweeController) controller).removeControllerListener(mControllerListener); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void addControllerListener(DraweeController controller) { |  | ||||||
|         if (controller instanceof AbstractDraweeController) { |  | ||||||
|             ((AbstractDraweeController) controller).addControllerListener(mControllerListener); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onDraw(Canvas canvas) { |  | ||||||
|         int saveCount = canvas.save(); |  | ||||||
|         canvas.concat(mZoomableController.getTransform()); |  | ||||||
|         try { |  | ||||||
|             super.onDraw(canvas); |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             DraweeController controller = getController(); |  | ||||||
|             if (controller != null && controller instanceof AbstractDraweeController) { |  | ||||||
|                 Object callerContext = ((AbstractDraweeController) controller).getCallerContext(); |  | ||||||
|                 if (callerContext != null) { |  | ||||||
|                     throw new RuntimeException( |  | ||||||
|                             String.format("Exception in onDraw, callerContext=%s", callerContext.toString()), e); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             throw e; |  | ||||||
|         } |  | ||||||
|         canvas.restoreToCount(saveCount); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onTouchEvent(MotionEvent event) { |  | ||||||
|         int a = event.getActionMasked(); |  | ||||||
|         FLog.v(getLogTag(), "onTouchEvent: %d, view %x, received", a, this.hashCode()); |  | ||||||
|         if (!mIsDialtoneEnabled && mTapGestureDetector.onTouchEvent(event)) { |  | ||||||
|             FLog.v( |  | ||||||
|                     getLogTag(), |  | ||||||
|                     "onTouchEvent: %d, view %x, handled by tap gesture detector", |  | ||||||
|                     a, |  | ||||||
|                     this.hashCode()); |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (!mIsDialtoneEnabled && mZoomableController.onTouchEvent(event)) { |  | ||||||
|             FLog.v( |  | ||||||
|                     getLogTag(), |  | ||||||
|                     "onTouchEvent: %d, view %x, handled by zoomable controller", |  | ||||||
|                     a, |  | ||||||
|                     this.hashCode()); |  | ||||||
|             if (!mAllowTouchInterceptionWhileZoomed && !mZoomableController.isIdentity()) { |  | ||||||
|                 getParent().requestDisallowInterceptTouchEvent(true); |  | ||||||
|             } |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         if (super.onTouchEvent(event)) { |  | ||||||
|             FLog.v(getLogTag(), "onTouchEvent: %d, view %x, handled by the super", a, this.hashCode()); |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         // None of our components reported that they handled the touch event. Upon returning false |  | ||||||
|         // from this method, our parent won't send us any more events for this gesture. Unfortunately, |  | ||||||
|         // some components may have started a delayed action, such as a long-press timer, and since we |  | ||||||
|         // won't receive an ACTION_UP that would cancel that timer, a false event may be triggered. |  | ||||||
|         // To prevent that we explicitly send one last cancel event when returning false. |  | ||||||
|         MotionEvent cancelEvent = MotionEvent.obtain(event); |  | ||||||
|         cancelEvent.setAction(MotionEvent.ACTION_CANCEL); |  | ||||||
|         mTapGestureDetector.onTouchEvent(cancelEvent); |  | ||||||
|         mZoomableController.onTouchEvent(cancelEvent); |  | ||||||
|         cancelEvent.recycle(); |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeHorizontalScrollRange() { |  | ||||||
|         return mZoomableController.computeHorizontalScrollRange(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeHorizontalScrollOffset() { |  | ||||||
|         return mZoomableController.computeHorizontalScrollOffset(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeHorizontalScrollExtent() { |  | ||||||
|         return mZoomableController.computeHorizontalScrollExtent(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeVerticalScrollRange() { |  | ||||||
|         return mZoomableController.computeVerticalScrollRange(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeVerticalScrollOffset() { |  | ||||||
|         return mZoomableController.computeVerticalScrollOffset(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int computeVerticalScrollExtent() { |  | ||||||
|         return mZoomableController.computeVerticalScrollExtent(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onLayout(boolean changed, int left, int top, int right, int bottom) { |  | ||||||
|         FLog.v(getLogTag(), "onLayout: view %x", this.hashCode()); |  | ||||||
|         super.onLayout(changed, left, top, right, bottom); |  | ||||||
|         updateZoomableControllerBounds(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void onFinalImageSet() { |  | ||||||
|         FLog.v(getLogTag(), "onFinalImageSet: view %x", this.hashCode()); |  | ||||||
|         if (!mZoomableController.isEnabled() && mZoomingEnabled) { |  | ||||||
|             mZoomableController.setEnabled(true); |  | ||||||
|             updateZoomableControllerBounds(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void onRelease() { |  | ||||||
|         FLog.v(getLogTag(), "onRelease: view %x", this.hashCode()); |  | ||||||
|         mZoomableController.setEnabled(false); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void onTransformChanged(Matrix transform) { |  | ||||||
|         FLog.v(getLogTag(), "onTransformChanged: view %x, transform: %s", this.hashCode(), transform); |  | ||||||
|         maybeSetHugeImageController(); |  | ||||||
|         invalidate(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void updateZoomableControllerBounds() { |  | ||||||
|         getImageBounds(mImageBounds); |  | ||||||
|         getLimitBounds(mViewBounds); |  | ||||||
|         mZoomableController.setImageBounds(mImageBounds); |  | ||||||
|         mZoomableController.setViewBounds(mViewBounds); |  | ||||||
|         FLog.v( |  | ||||||
|                 getLogTag(), |  | ||||||
|                 "updateZoomableControllerBounds: view %x, view bounds: %s, image bounds: %s", |  | ||||||
|                 this.hashCode(), |  | ||||||
|                 mViewBounds, |  | ||||||
|                 mImageBounds); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected Class<?> getLogTag() { |  | ||||||
|         return TAG; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected ZoomableController createZoomableController() { |  | ||||||
|         return AnimatedZoomableController.newInstance(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Use this, If someone is willing to listen to scale change |  | ||||||
|      */ |  | ||||||
|     public interface TransformationListener{ |  | ||||||
|         void onTransformationEnd(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,320 @@ | ||||||
|  | package fr.free.nrw.commons.media.zoomControllers.zoomable | ||||||
|  | 
 | ||||||
|  | import android.annotation.SuppressLint | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.res.Resources | ||||||
|  | import android.graphics.Canvas | ||||||
|  | import android.graphics.Matrix | ||||||
|  | import android.graphics.RectF | ||||||
|  | import android.graphics.drawable.Animatable | ||||||
|  | import android.util.AttributeSet | ||||||
|  | import android.view.GestureDetector | ||||||
|  | import android.view.MotionEvent | ||||||
|  | 
 | ||||||
|  | import androidx.core.view.ScrollingView | ||||||
|  | import com.facebook.common.internal.Preconditions | ||||||
|  | import com.facebook.common.logging.FLog | ||||||
|  | import com.facebook.drawee.controller.AbstractDraweeController | ||||||
|  | import com.facebook.drawee.controller.BaseControllerListener | ||||||
|  | import com.facebook.drawee.drawable.ScalingUtils | ||||||
|  | import com.facebook.drawee.generic.GenericDraweeHierarchy | ||||||
|  | import com.facebook.drawee.generic.GenericDraweeHierarchyBuilder | ||||||
|  | import com.facebook.drawee.generic.GenericDraweeHierarchyInflater | ||||||
|  | import com.facebook.drawee.interfaces.DraweeController | ||||||
|  | import com.facebook.drawee.view.DraweeView | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * DraweeView that has zoomable capabilities. | ||||||
|  |  * | ||||||
|  |  * <p>Once the image loads, pinch-to-zoom and translation gestures are enabled. | ||||||
|  |  */ | ||||||
|  | open class ZoomableDraweeView : DraweeView<GenericDraweeHierarchy>, ScrollingView { | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private val TAG = ZoomableDraweeView::class.java | ||||||
|  |         private const val HUGE_IMAGE_SCALE_FACTOR_THRESHOLD = 1.1f | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private val imageBounds = RectF() | ||||||
|  |     private val viewBounds = RectF() | ||||||
|  | 
 | ||||||
|  |     private var hugeImageController: DraweeController? = null | ||||||
|  |     private var zoomableController: ZoomableController = createZoomableController() | ||||||
|  |     private var tapGestureDetector: GestureDetector? = null | ||||||
|  |     private var allowTouchInterceptionWhileZoomed = true | ||||||
|  |     private var isDialToneEnabled = false | ||||||
|  |     private var zoomingEnabled = true | ||||||
|  |     private var transformationListener: TransformationListener? = null | ||||||
|  | 
 | ||||||
|  |     private val controllerListener = object : BaseControllerListener<Any>() { | ||||||
|  |         override fun onFinalImageSet(id: String, imageInfo: Any?, animatable: Animatable?) { | ||||||
|  |             this@ZoomableDraweeView.onFinalImageSet() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         override fun onRelease(id: String) { | ||||||
|  |             this@ZoomableDraweeView.onRelease() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private val zoomableListener = object : ZoomableController.Listener { | ||||||
|  |         override fun onTransformBegin(transform: Matrix) {} | ||||||
|  | 
 | ||||||
|  |         override fun onTransformChanged(transform: Matrix) { | ||||||
|  |             this@ZoomableDraweeView.onTransformChanged(transform) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         override fun onTransformEnd(transform: Matrix) { | ||||||
|  |             transformationListener?.onTransformationEnd() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private val tapListenerWrapper = GestureListenerWrapper() | ||||||
|  | 
 | ||||||
|  |     constructor(context: Context, hierarchy: GenericDraweeHierarchy) : super(context) { | ||||||
|  |         setHierarchy(hierarchy) | ||||||
|  |         init() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     constructor(context: Context) : super(context) { | ||||||
|  |         inflateHierarchy(context, null) | ||||||
|  |         init() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     constructor(context: Context, attrs: AttributeSet?) : super(context, attrs) { | ||||||
|  |         inflateHierarchy(context, attrs) | ||||||
|  |         init() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     constructor(context: Context, attrs: AttributeSet?, defStyle: Int) | ||||||
|  |             : super(context, attrs, defStyle) { | ||||||
|  |         inflateHierarchy(context, attrs) | ||||||
|  |         init() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setTransformationListener(transformationListener: TransformationListener) { | ||||||
|  |         this.transformationListener = transformationListener | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun inflateHierarchy(context: Context, attrs: AttributeSet?) { | ||||||
|  |         val resources: Resources = context.resources | ||||||
|  |         val builder = GenericDraweeHierarchyBuilder(resources) | ||||||
|  |             .setActualImageScaleType(ScalingUtils.ScaleType.FIT_CENTER) | ||||||
|  |         GenericDraweeHierarchyInflater.updateBuilder(builder, context, attrs) | ||||||
|  |         aspectRatio = builder.desiredAspectRatio | ||||||
|  |         setHierarchy(builder.build()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun init() { | ||||||
|  |         zoomableController.setListener(zoomableListener) | ||||||
|  |         tapGestureDetector = GestureDetector(context, tapListenerWrapper) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setIsDialToneEnabled(isDialtoneEnabled: Boolean) { | ||||||
|  |         this.isDialToneEnabled = isDialtoneEnabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun getImageBounds(outBounds: RectF) { | ||||||
|  |         hierarchy.getActualImageBounds(outBounds) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun getLimitBounds(outBounds: RectF) { | ||||||
|  |         outBounds.set(0f, 0f, width.toFloat(), height.toFloat()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setZoomableController(zoomableController: ZoomableController) { | ||||||
|  |         Preconditions.checkNotNull(zoomableController) | ||||||
|  |         this.zoomableController.setListener(null) | ||||||
|  |         this.zoomableController = zoomableController | ||||||
|  |         this.zoomableController.setListener(zoomableListener) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun getZoomableController(): ZoomableController = zoomableController | ||||||
|  | 
 | ||||||
|  |     fun allowsTouchInterceptionWhileZoomed(): Boolean = allowTouchInterceptionWhileZoomed | ||||||
|  | 
 | ||||||
|  |     fun setAllowTouchInterceptionWhileZoomed(allow: Boolean) { | ||||||
|  |         allowTouchInterceptionWhileZoomed = allow | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setTapListener(tapListener: GestureDetector.SimpleOnGestureListener) { | ||||||
|  |         tapListenerWrapper.setListener(tapListener) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setIsLongpressEnabled(enabled: Boolean) { | ||||||
|  |         tapGestureDetector?.setIsLongpressEnabled(enabled) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setZoomingEnabled(zoomingEnabled: Boolean) { | ||||||
|  |         this.zoomingEnabled = zoomingEnabled | ||||||
|  |         zoomableController.setEnabled(false) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun setController(controller: DraweeController?) { | ||||||
|  |         setControllers(controller, null) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setControllers(controller: DraweeController?, hugeImageController: DraweeController?) { | ||||||
|  |         setControllersInternal(null, null) | ||||||
|  |         zoomableController.setEnabled(false) | ||||||
|  |         setControllersInternal(controller, hugeImageController) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun setControllersInternal( | ||||||
|  |         controller: DraweeController?, | ||||||
|  |         hugeImageController: DraweeController? | ||||||
|  |     ) { | ||||||
|  |         removeControllerListener(getController()) | ||||||
|  |         addControllerListener(controller) | ||||||
|  |         this.hugeImageController = hugeImageController | ||||||
|  |         super.setController(controller) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun maybeSetHugeImageController() { | ||||||
|  |         if ( | ||||||
|  |             hugeImageController != null | ||||||
|  |             && | ||||||
|  |             zoomableController.getScaleFactor() > HUGE_IMAGE_SCALE_FACTOR_THRESHOLD | ||||||
|  |         ) { | ||||||
|  |             setControllersInternal(hugeImageController, null) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun removeControllerListener(controller: DraweeController?) { | ||||||
|  |         if (controller is AbstractDraweeController<*, *>) { | ||||||
|  |             controller.removeControllerListener(controllerListener) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun addControllerListener(controller: DraweeController?) { | ||||||
|  |         if (controller is AbstractDraweeController<*, *>) { | ||||||
|  |             controller.addControllerListener(controllerListener) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDraw(canvas: Canvas) { | ||||||
|  |         val saveCount = canvas.save() | ||||||
|  |         canvas.concat(zoomableController.getTransform()) | ||||||
|  |         try { | ||||||
|  |             super.onDraw(canvas) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             val controller = controller | ||||||
|  |             if (controller is AbstractDraweeController<*, *>) { | ||||||
|  |                 val callerContext = controller.callerContext | ||||||
|  |                 if (callerContext != null) { | ||||||
|  |                     throw RuntimeException("Exception in onDraw, callerContext=${callerContext}", e) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             throw e | ||||||
|  |         } | ||||||
|  |         canvas.restoreToCount(saveCount) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @SuppressLint("ClickableViewAccessibility") | ||||||
|  |     override fun onTouchEvent(event: MotionEvent): Boolean { | ||||||
|  |         var action = event.actionMasked | ||||||
|  |         FLog.v(getLogTag(), "onTouchEvent: $action, view ${hashCode()}, received") | ||||||
|  | 
 | ||||||
|  |         if (!isDialToneEnabled && tapGestureDetector?.onTouchEvent(event) == true) { | ||||||
|  |             FLog.v( | ||||||
|  |                 getLogTag(), | ||||||
|  |                 "onTouchEvent: $action, view ${hashCode()}, handled by tap gesture detector" | ||||||
|  |             ) | ||||||
|  |             return true | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (!isDialToneEnabled && zoomableController.onTouchEvent(event)) { | ||||||
|  |             FLog.v( | ||||||
|  |                 getLogTag(), | ||||||
|  |                 "onTouchEvent: $action, view ${hashCode()}, handled by zoomable controller" | ||||||
|  |             ) | ||||||
|  |             if (!allowTouchInterceptionWhileZoomed && !zoomableController.isIdentity()) { | ||||||
|  |                 parent.requestDisallowInterceptTouchEvent(true) | ||||||
|  |             } | ||||||
|  |             return true | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (super.onTouchEvent(event)) { | ||||||
|  |             FLog.v( | ||||||
|  |                 getLogTag(), | ||||||
|  |                 "onTouchEvent: $action, view ${hashCode()}, handled by the super" | ||||||
|  |             ) | ||||||
|  |             return true | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // If none of our components handled the event, we send a cancel event to avoid unwanted actions. | ||||||
|  |         val cancelEvent = MotionEvent.obtain(event).apply { action = MotionEvent.ACTION_CANCEL } | ||||||
|  |         tapGestureDetector?.onTouchEvent(cancelEvent) | ||||||
|  |         zoomableController.onTouchEvent(cancelEvent) | ||||||
|  |         cancelEvent.recycle() | ||||||
|  | 
 | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun computeHorizontalScrollRange(): Int = | ||||||
|  |         zoomableController.computeHorizontalScrollRange() | ||||||
|  | 
 | ||||||
|  |     override fun computeHorizontalScrollOffset(): Int = | ||||||
|  |         zoomableController.computeHorizontalScrollOffset() | ||||||
|  | 
 | ||||||
|  |     override fun computeHorizontalScrollExtent(): Int = | ||||||
|  |         zoomableController.computeHorizontalScrollExtent() | ||||||
|  | 
 | ||||||
|  |     override fun computeVerticalScrollRange(): Int = | ||||||
|  |         zoomableController.computeVerticalScrollRange() | ||||||
|  | 
 | ||||||
|  |     override fun computeVerticalScrollOffset(): Int = | ||||||
|  |         zoomableController.computeVerticalScrollOffset() | ||||||
|  | 
 | ||||||
|  |     override fun computeVerticalScrollExtent(): Int = | ||||||
|  |         zoomableController.computeVerticalScrollExtent() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     override fun onLayout(changed: Boolean, left: Int, top: Int, right: Int, bottom: Int) { | ||||||
|  |         FLog.v(getLogTag(), "onLayout: view ${hashCode()}") | ||||||
|  |         super.onLayout(changed, left, top, right, bottom) | ||||||
|  |         updateZoomableControllerBounds() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onFinalImageSet() { | ||||||
|  |         FLog.v(getLogTag(), "onFinalImageSet: view ${hashCode()}") | ||||||
|  |         if (!zoomableController.isEnabled() && zoomingEnabled) { | ||||||
|  |             zoomableController.setEnabled(true) | ||||||
|  |             updateZoomableControllerBounds() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onRelease() { | ||||||
|  |         FLog.v(getLogTag(), "onRelease: view ${hashCode()}") | ||||||
|  |         zoomableController.setEnabled(false) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun onTransformChanged(transform: Matrix) { | ||||||
|  |         FLog.v(getLogTag(), "onTransformChanged: view ${hashCode()}, transform: $transform") | ||||||
|  |         maybeSetHugeImageController() | ||||||
|  |         invalidate() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun updateZoomableControllerBounds() { | ||||||
|  |         getImageBounds(imageBounds) | ||||||
|  |         getLimitBounds(viewBounds) | ||||||
|  |         zoomableController.setImageBounds(imageBounds) | ||||||
|  |         zoomableController.setViewBounds(viewBounds) | ||||||
|  | 
 | ||||||
|  |         FLog.v( | ||||||
|  |             getLogTag(), | ||||||
|  |             "updateZoomableControllerBounds: view ${hashCode()}, " + | ||||||
|  |                     "view bounds: $viewBounds, image bounds: $imageBounds" | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected fun getLogTag(): Class<*> = TAG | ||||||
|  | 
 | ||||||
|  |     protected fun createZoomableController(): ZoomableController = AnimatedZoomableController.newInstance() | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Interface to listen for scale change events. | ||||||
|  |      */ | ||||||
|  |     interface TransformationListener { | ||||||
|  |         fun onTransformationEnd() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -16,6 +16,7 @@ import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| import android.widget.AdapterView | import android.widget.AdapterView | ||||||
| import android.widget.AdapterView.OnItemClickListener | import android.widget.AdapterView.OnItemClickListener | ||||||
|  | import android.widget.Button | ||||||
| import android.widget.EditText | import android.widget.EditText | ||||||
| import android.widget.FrameLayout | import android.widget.FrameLayout | ||||||
| import android.widget.ImageView | import android.widget.ImageView | ||||||
|  | @ -330,6 +331,9 @@ class UploadMediaDetailAdapter : RecyclerView.Adapter<UploadMediaDetailAdapter.V | ||||||
| 
 | 
 | ||||||
|                 listView.adapter = languagesAdapter |                 listView.adapter = languagesAdapter | ||||||
| 
 | 
 | ||||||
|  |                 dialog.findViewById<Button>(R.id.cancel_button) | ||||||
|  |                     .setOnClickListener { v: View? -> dialog.dismiss() } | ||||||
|  | 
 | ||||||
|                 editText.addTextChangedListener(object : TextWatcher { |                 editText.addTextChangedListener(object : TextWatcher { | ||||||
|                     override fun beforeTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) = |                     override fun beforeTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) = | ||||||
|                         hideRecentLanguagesSection() |                         hideRecentLanguagesSection() | ||||||
|  |  | ||||||
|  | @ -181,6 +181,7 @@ | ||||||
|   <string name="no">Neen</string> |   <string name="no">Neen</string> | ||||||
|   <string name="media_detail_caption">Beschrëftung</string> |   <string name="media_detail_caption">Beschrëftung</string> | ||||||
|   <string name="media_detail_title">Titel</string> |   <string name="media_detail_title">Titel</string> | ||||||
|  |   <string name="media_detail_depiction">Motiven</string> | ||||||
|   <string name="media_detail_description">Beschreiwung</string> |   <string name="media_detail_description">Beschreiwung</string> | ||||||
|   <string name="media_detail_discussion">Diskussioun</string> |   <string name="media_detail_discussion">Diskussioun</string> | ||||||
|   <string name="media_detail_author">Auteur</string> |   <string name="media_detail_author">Auteur</string> | ||||||
|  |  | ||||||
|  | @ -406,10 +406,10 @@ | ||||||
|   <string name="preference_author_name_toggle_summary">Gebruik een aangepaste auteursnaam in plaats van uw gebruikersnaam tijdens het uploaden van foto\'s</string> |   <string name="preference_author_name_toggle_summary">Gebruik een aangepaste auteursnaam in plaats van uw gebruikersnaam tijdens het uploaden van foto\'s</string> | ||||||
|   <string name="preference_author_name">Aangepaste auteursnaam</string> |   <string name="preference_author_name">Aangepaste auteursnaam</string> | ||||||
|   <string name="contributions_fragment">Bijdragen</string> |   <string name="contributions_fragment">Bijdragen</string> | ||||||
|   <string name="nearby_fragment">Dichtbij</string> |   <string name="nearby_fragment">In de buurt</string> | ||||||
|   <string name="notifications">Meldingen</string> |   <string name="notifications">Meldingen</string> | ||||||
|   <string name="read_notifications">Meldingen (gelezen)</string> |   <string name="read_notifications">Meldingen (gelezen)</string> | ||||||
|   <string name="display_nearby_notification">Meldingen dichtbij weergeven</string> |   <string name="display_nearby_notification">Melding in de buurt weergeven</string> | ||||||
|   <string name="display_nearby_notification_summary">Toon in-app-melding voor de dichtstbijzijnde plaats die foto\'s nodig heeft</string> |   <string name="display_nearby_notification_summary">Toon in-app-melding voor de dichtstbijzijnde plaats die foto\'s nodig heeft</string> | ||||||
|   <string name="list_sheet">Lijst</string> |   <string name="list_sheet">Lijst</string> | ||||||
|   <string name="storage_permission">Toestemming om op te slaan</string> |   <string name="storage_permission">Toestemming om op te slaan</string> | ||||||
|  | @ -615,7 +615,7 @@ | ||||||
|   <string name="recommend_high_accuracy_mode">Kies voor de beste resultaten de modus van hoge nauwkeurigheid.</string> |   <string name="recommend_high_accuracy_mode">Kies voor de beste resultaten de modus van hoge nauwkeurigheid.</string> | ||||||
|   <string name="ask_to_turn_location_on">Locatie inschakelen?</string> |   <string name="ask_to_turn_location_on">Locatie inschakelen?</string> | ||||||
|   <string name="ask_to_turn_location_on_text">Schakel locatiediensten in zodat de app uw huidige locatie toont</string> |   <string name="ask_to_turn_location_on_text">Schakel locatiediensten in zodat de app uw huidige locatie toont</string> | ||||||
|   <string name="nearby_needs_location">In de Buurt heeft locatie nodig om correct te werken</string> |   <string name="nearby_needs_location">‘In de buurt’ heeft locatietoegang nodig om correct te werken</string> | ||||||
|   <string name="explore_map_needs_location">Voor de verkenningskaart is locatietoestemming nodig om afbeeldingen in de buurt weer te geven</string> |   <string name="explore_map_needs_location">Voor de verkenningskaart is locatietoestemming nodig om afbeeldingen in de buurt weer te geven</string> | ||||||
|   <string name="upload_map_location_access">U moet locatietoestemming geven om de locatie automatisch in te stellen.</string> |   <string name="upload_map_location_access">U moet locatietoestemming geven om de locatie automatisch in te stellen.</string> | ||||||
|   <string name="use_location_from_similar_image">Heeft u deze twee foto\'s op dezelfde plek gemaakt? Wilt u de breedtegraad/lengtegraad van de afbeelding rechts gebruiken?</string> |   <string name="use_location_from_similar_image">Heeft u deze twee foto\'s op dezelfde plek gemaakt? Wilt u de breedtegraad/lengtegraad van de afbeelding rechts gebruiken?</string> | ||||||
|  | @ -657,7 +657,7 @@ | ||||||
|   <string name="leaderboard_weekly">Wekelijks</string> |   <string name="leaderboard_weekly">Wekelijks</string> | ||||||
|   <string name="leaderboard_all_time">Alle tijden</string> |   <string name="leaderboard_all_time">Alle tijden</string> | ||||||
|   <string name="leaderboard_upload">Uploaden</string> |   <string name="leaderboard_upload">Uploaden</string> | ||||||
|   <string name="leaderboard_nearby">In de Buurt</string> |   <string name="leaderboard_nearby">In de buurt</string> | ||||||
|   <string name="leaderboard_used">Gebruikt</string> |   <string name="leaderboard_used">Gebruikt</string> | ||||||
|   <string name="leaderboard_my_rank_button_text">Mijn ranking</string> |   <string name="leaderboard_my_rank_button_text">Mijn ranking</string> | ||||||
|   <string name="limited_connection_enabled">Beperkte verbindingsmodus ingeschakeld!</string> |   <string name="limited_connection_enabled">Beperkte verbindingsmodus ingeschakeld!</string> | ||||||
|  | @ -724,7 +724,7 @@ | ||||||
|   <string name="edit_depictions">Wijzig items</string> |   <string name="edit_depictions">Wijzig items</string> | ||||||
|   <string name="edit_categories">Categorieën bewerken</string> |   <string name="edit_categories">Categorieën bewerken</string> | ||||||
|   <string name="advanced_options">Geavanceerde opties</string> |   <string name="advanced_options">Geavanceerde opties</string> | ||||||
|   <string name="advanced_query_info_text">U kunt de zoekopdracht Dichtbij aanpassen. Als u fouten krijgt, kunt u opnieuw instellen en toepassen.</string> |   <string name="advanced_query_info_text">U kunt de zoekopdracht ‘In de buurt’ aanpassen. Als u fouten krijgt, kunt u opnieuw instellen en toepassen.</string> | ||||||
|   <string name="apply">Toepassen</string> |   <string name="apply">Toepassen</string> | ||||||
|   <string name="reset">Opnieuw instellen</string> |   <string name="reset">Opnieuw instellen</string> | ||||||
|   <string name="location_message">Locatiegegevens helpen wiki-bewerkers om uw foto te vinden, waardoor deze veel nuttiger wordt.\nUw recente uploads hebben geen locatie.\nWe raden u aan om de locatie in de instellingen van uw camera-app in te schakelen.\nBedankt voor het uploaden!</string> |   <string name="location_message">Locatiegegevens helpen wiki-bewerkers om uw foto te vinden, waardoor deze veel nuttiger wordt.\nUw recente uploads hebben geen locatie.\nWe raden u aan om de locatie in de instellingen van uw camera-app in te schakelen.\nBedankt voor het uploaden!</string> | ||||||
|  | @ -835,4 +835,10 @@ | ||||||
|   <string name="account">Account</string> |   <string name="account">Account</string> | ||||||
|   <string name="vanish_account">Account laten verdwijnen</string> |   <string name="vanish_account">Account laten verdwijnen</string> | ||||||
|   <string name="account_vanish_request_confirm_title">Waarschuwing verwijdering account</string> |   <string name="account_vanish_request_confirm_title">Waarschuwing verwijdering account</string> | ||||||
|  |   <string name="account_vanish_request_confirm">Verdwijnen is een <b>laatste redmiddel</b> en moet <b>alleen worden gebruikt als u voor altijd wilt stoppen met bewerken</b> en om zoveel mogelijk van uw voorgaande relaties te verbergen.<br/><br/>Accountverwijdering op Wikipedia gebeurt door uw accountnaam te wijzigen opdat anderen uw bijdragen niet meer kunnen herkennen. De procedure wordt accountverdwijning genoemd. <b>Door verdwijnen wordt geen volledige anonimiteit gegarandeerd en worden geen bijdragen aan de projecten verwijderd.</b></string> | ||||||
|  |   <string name="caption">Bijschrift</string> | ||||||
|  |   <string name="caption_copied_to_clipboard">Bijschrift gekopieerd naar klembord</string> | ||||||
|  |   <string name="congratulations_all_pictures_in_this_album_have_been_either_uploaded_or_marked_as_not_for_upload">Gefeliciteerd, alle foto’s in dit album zijn ofwel geüpload ofwel gemarkeerd als ‘niet om te uploaden’.</string> | ||||||
|  |   <string name="show_in_explore">Weergeven in Verkennen</string> | ||||||
|  |   <string name="show_in_nearby">Weergeven in ‘In de buurt’</string> | ||||||
| </resources> | </resources> | ||||||
|  |  | ||||||
|  | @ -192,6 +192,7 @@ | ||||||
|   <string name="depictions_edit_helper_edit_message_else">{{Doc-commons-app-depicts}}</string> |   <string name="depictions_edit_helper_edit_message_else">{{Doc-commons-app-depicts}}</string> | ||||||
|   <string name="title_app_shortcut_bookmark">{{identical|Bookmark}}</string> |   <string name="title_app_shortcut_bookmark">{{identical|Bookmark}}</string> | ||||||
|   <string name="theme_default_name">Option to make the app\'s theme follow the global system setting.</string> |   <string name="theme_default_name">Option to make the app\'s theme follow the global system setting.</string> | ||||||
|  |   <string name="nearby_needs_location">‘Nearby’ should be translated as in {{msg-wm|Commons-android-strings-navigation item nearby}}</string> | ||||||
|   <string name="more">{{Identical|More}}</string> |   <string name="more">{{Identical|More}}</string> | ||||||
|   <string name="map_attribution">{{Optional}}\n<code>&amp;#169;</code> is the copyright symbol (©).</string> |   <string name="map_attribution">{{Optional}}\n<code>&amp;#169;</code> is the copyright symbol (©).</string> | ||||||
|   <string name="categories_tooltip">{{Doc-commons-app-depicts}}</string> |   <string name="categories_tooltip">{{Doc-commons-app-depicts}}</string> | ||||||
|  | @ -204,9 +205,12 @@ | ||||||
|   <string name="done">{{identical|Done}}</string> |   <string name="done">{{identical|Done}}</string> | ||||||
|   <string name="edit_depictions">{{Doc-commons-app-depicts}}</string> |   <string name="edit_depictions">{{Doc-commons-app-depicts}}</string> | ||||||
|   <string name="advanced_options">{{Identical|Advanced options}}</string> |   <string name="advanced_options">{{Identical|Advanced options}}</string> | ||||||
|  |   <string name="advanced_query_info_text">‘Nearby’ should be translated as in {{msg-wm|Commons-android-strings-navigation item nearby}}</string> | ||||||
|   <string name="explore_map_details">{{Identical|Detail}}</string> |   <string name="explore_map_details">{{Identical|Detail}}</string> | ||||||
|   <string name="set_up_avatar_toast_string">\"Set as avatar\" should be translated the same as {{msg-wm|Commons-android-strings-menu set avatar}}.</string> |   <string name="set_up_avatar_toast_string">\"Set as avatar\" should be translated the same as {{msg-wm|Commons-android-strings-menu set avatar}}.</string> | ||||||
|   <string name="multiple_files_depiction">{{Doc-commons-app-depicts}}</string> |   <string name="multiple_files_depiction">{{Doc-commons-app-depicts}}</string> | ||||||
|   <string name="custom_selector_delete">An answer to the question in {{msg-wm|Commons-android-strings-custom selector confirm deletion message}}.</string> |   <string name="custom_selector_delete">An answer to the question in {{msg-wm|Commons-android-strings-custom selector confirm deletion message}}.</string> | ||||||
|   <string name="bullet_point">{{optional}}</string> |   <string name="bullet_point">{{optional}}</string> | ||||||
|  |   <string name="show_in_explore">“Explore” should be translated as in {{msg-wm|Commons-android-strings-navigation item explore}}</string> | ||||||
|  |   <string name="show_in_nearby">‘Nearby’ should be translated as in {{msg-wm|Commons-android-strings-navigation item nearby}}</string> | ||||||
| </resources> | </resources> | ||||||
|  |  | ||||||
|  | @ -66,7 +66,7 @@ | ||||||
|   <string name="upload_progress_notification_title_start">%s ಅಪ್ಲೋಡ್ ಸುರು ಆವೊಂದುಂಡು</string> |   <string name="upload_progress_notification_title_start">%s ಅಪ್ಲೋಡ್ ಸುರು ಆವೊಂದುಂಡು</string> | ||||||
|   <string name="upload_progress_notification_title_in_progress">%1$s ಅಪ್ಲೋಡ್ ಆವೊಂದುಂಡು</string> |   <string name="upload_progress_notification_title_in_progress">%1$s ಅಪ್ಲೋಡ್ ಆವೊಂದುಂಡು</string> | ||||||
|   <string name="upload_progress_notification_title_finishing">%1$s ಅಪ್ಲೋಡ್ ಕೈದ್ ಆವೊಂದುಂಡು.</string> |   <string name="upload_progress_notification_title_finishing">%1$s ಅಪ್ಲೋಡ್ ಕೈದ್ ಆವೊಂದುಂಡು.</string> | ||||||
|   <string name="upload_failed_notification_title" fuzzy="true">%1$s ಅಪ್ಲೋಡ್ ಸರಿ ಆತಿಜಿ</string> |   <string name="upload_failed_notification_title">%1$s ಅಪ್ಲೋಡ್ ಆತಿಜಿ</string> | ||||||
|   <string name="upload_failed_notification_subtitle">ತುಯಾರ ಮೆಲ್ಲ ಒತ್ತುಲೆ</string> |   <string name="upload_failed_notification_subtitle">ತುಯಾರ ಮೆಲ್ಲ ಒತ್ತುಲೆ</string> | ||||||
|   <string name="title_activity_contributions">ಎನ್ನ ದಿಂಜಯೀನಾ ವಿಚಾರೊಳು</string> |   <string name="title_activity_contributions">ಎನ್ನ ದಿಂಜಯೀನಾ ವಿಚಾರೊಳು</string> | ||||||
|   <string name="contribution_state_queued">ದಿಂಜೊಂತುಂಡು</string> |   <string name="contribution_state_queued">ದಿಂಜೊಂತುಂಡು</string> | ||||||
|  |  | ||||||
|  | @ -96,7 +96,7 @@ class DeleteHelperTest { | ||||||
|         ).thenReturn("Media successfully deleted: Test Media Title") |         ).thenReturn("Media successfully deleted: Test Media Title") | ||||||
| 
 | 
 | ||||||
|         val creatorName = "Creator" |         val creatorName = "Creator" | ||||||
|         whenever(media.author).thenReturn("$creatorName") |         whenever(media.getAuthorOrUser()).thenReturn("$creatorName") | ||||||
|         whenever(media.filename).thenReturn("Test file.jpg") |         whenever(media.filename).thenReturn("Test file.jpg") | ||||||
|         val makeDeletion = deleteHelper.makeDeletion( |         val makeDeletion = deleteHelper.makeDeletion( | ||||||
|             context, |             context, | ||||||
|  | @ -133,7 +133,7 @@ class DeleteHelperTest { | ||||||
| 
 | 
 | ||||||
|         whenever(media.displayTitle).thenReturn("Test file") |         whenever(media.displayTitle).thenReturn("Test file") | ||||||
|         whenever(media.filename).thenReturn("Test file.jpg") |         whenever(media.filename).thenReturn("Test file.jpg") | ||||||
|         whenever(media.author).thenReturn("Creator (page does not exist)") |         whenever(media.getAuthorOrUser()).thenReturn("Creator (page does not exist)") | ||||||
| 
 | 
 | ||||||
|         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() |         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() | ||||||
|     } |     } | ||||||
|  | @ -148,7 +148,7 @@ class DeleteHelperTest { | ||||||
|             .thenReturn(Observable.just(false)) |             .thenReturn(Observable.just(false)) | ||||||
|         whenever(media.displayTitle).thenReturn("Test file") |         whenever(media.displayTitle).thenReturn("Test file") | ||||||
|         whenever(media.filename).thenReturn("Test file.jpg") |         whenever(media.filename).thenReturn("Test file.jpg") | ||||||
|         whenever(media.author).thenReturn("Creator (page does not exist)") |         whenever(media.getAuthorOrUser()).thenReturn("Creator (page does not exist)") | ||||||
| 
 | 
 | ||||||
|         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() |         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() | ||||||
|     } |     } | ||||||
|  | @ -163,7 +163,7 @@ class DeleteHelperTest { | ||||||
|             .thenReturn(Observable.just(true)) |             .thenReturn(Observable.just(true)) | ||||||
|         whenever(media.displayTitle).thenReturn("Test file") |         whenever(media.displayTitle).thenReturn("Test file") | ||||||
|         whenever(media.filename).thenReturn("Test file.jpg") |         whenever(media.filename).thenReturn("Test file.jpg") | ||||||
|         whenever(media.author).thenReturn("Creator (page does not exist)") |         whenever(media.getAuthorOrUser()).thenReturn("Creator (page does not exist)") | ||||||
| 
 | 
 | ||||||
|         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() |         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() | ||||||
|     } |     } | ||||||
|  | @ -221,7 +221,7 @@ class DeleteHelperTest { | ||||||
|         whenever(media.displayTitle).thenReturn("Test file") |         whenever(media.displayTitle).thenReturn("Test file") | ||||||
|         whenever(media.filename).thenReturn("Test file.jpg") |         whenever(media.filename).thenReturn("Test file.jpg") | ||||||
| 
 | 
 | ||||||
|         whenever(media.author).thenReturn(null) |         whenever(media.getAuthorOrUser()).thenReturn(null) | ||||||
| 
 | 
 | ||||||
|         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() |         deleteHelper.makeDeletion(context, media, "Test reason")?.blockingGet() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -107,43 +107,43 @@ class MultiPointerGestureDetectorUnitTest { | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testIsGestureInProgress() { |     fun testIsGestureInProgress() { | ||||||
|         Assert.assertEquals(detector.isGestureInProgress, false) |         Assert.assertEquals(detector.isGestureInProgress(), false) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetNewPointerCount() { |     fun testGetNewPointerCount() { | ||||||
|         Assert.assertEquals(detector.newPointerCount, 0) |         Assert.assertEquals(detector.getNewPointerCount(), 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetPointerCount() { |     fun testGetPointerCount() { | ||||||
|         Assert.assertEquals(detector.pointerCount, 0) |         Assert.assertEquals(detector.getPointerCount(), 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetStartX() { |     fun testGetStartX() { | ||||||
|         Assert.assertEquals(detector.startX[0], 0.0f) |         Assert.assertEquals(detector.getStartX()[0], 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetStartY() { |     fun testGetStartY() { | ||||||
|         Assert.assertEquals(detector.startY[0], 0.0f) |         Assert.assertEquals(detector.getStartY()[0], 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetCurrentX() { |     fun testGetCurrentX() { | ||||||
|         Assert.assertEquals(detector.currentX[0], 0.0f) |         Assert.assertEquals(detector.getCurrentX()[0], 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetCurrentY() { |     fun testGetCurrentY() { | ||||||
|         Assert.assertEquals(detector.currentY[0], 0.0f) |         Assert.assertEquals(detector.getCurrentY()[0], 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|  |  | ||||||
|  | @ -84,51 +84,51 @@ class TransformGestureDetectorUnitTest { | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testIsGestureInProgress() { |     fun testIsGestureInProgress() { | ||||||
|         Assert.assertEquals(detector.isGestureInProgress, false) |         Assert.assertEquals(detector.isGestureInProgress(), false) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetNewPointerCount() { |     fun testGetNewPointerCount() { | ||||||
|         Assert.assertEquals(detector.newPointerCount, 0) |         Assert.assertEquals(detector.getNewPointerCount(), 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetPointerCount() { |     fun testGetPointerCount() { | ||||||
|         Assert.assertEquals(detector.pointerCount, 0) |         Assert.assertEquals(detector.getPointerCount(), 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetPivotX() { |     fun testGetPivotX() { | ||||||
|         Assert.assertEquals(detector.pivotX, 0.0f) |         Assert.assertEquals(detector.getPivotX(), 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetPivotY() { |     fun testGetPivotY() { | ||||||
|         Assert.assertEquals(detector.pivotY, 0.0f) |         Assert.assertEquals(detector.getPivotY(), 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetTranslationX() { |     fun testGetTranslationX() { | ||||||
|         Assert.assertEquals(detector.translationX, 0.0f) |         Assert.assertEquals(detector.getTranslationX(), 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetTranslationY() { |     fun testGetTranslationY() { | ||||||
|         Assert.assertEquals(detector.translationY, 0.0f) |         Assert.assertEquals(detector.getTranslationY(), 0.0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetScaleCaseLessThan2() { |     fun testGetScaleCaseLessThan2() { | ||||||
|         Whitebox.setInternalState(detector, "mDetector", mDetector) |         Whitebox.setInternalState(detector, "mDetector", mDetector) | ||||||
|         whenever(mDetector.pointerCount).thenReturn(1) |         whenever(mDetector.getPointerCount()).thenReturn(1) | ||||||
|         Assert.assertEquals(detector.scale, 1f) |         Assert.assertEquals(detector.getScale(), 1f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|  | @ -138,20 +138,20 @@ class TransformGestureDetectorUnitTest { | ||||||
|         array[0] = 0.0f |         array[0] = 0.0f | ||||||
|         array[1] = 1.0f |         array[1] = 1.0f | ||||||
|         Whitebox.setInternalState(detector, "mDetector", mDetector) |         Whitebox.setInternalState(detector, "mDetector", mDetector) | ||||||
|         whenever(mDetector.pointerCount).thenReturn(2) |         whenever(mDetector.getPointerCount()).thenReturn(2) | ||||||
|         whenever(mDetector.startX).thenReturn(array) |         whenever(mDetector.getStartX()).thenReturn(array) | ||||||
|         whenever(mDetector.startY).thenReturn(array) |         whenever(mDetector.getStartY()).thenReturn(array) | ||||||
|         whenever(mDetector.currentX).thenReturn(array) |         whenever(mDetector.getCurrentX()).thenReturn(array) | ||||||
|         whenever(mDetector.currentY).thenReturn(array) |         whenever(mDetector.getCurrentY()).thenReturn(array) | ||||||
|         Assert.assertEquals(detector.scale, 1f) |         Assert.assertEquals(detector.getScale(), 1f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     @Throws(Exception::class) |     @Throws(Exception::class) | ||||||
|     fun testGetRotationCaseLessThan2() { |     fun testGetRotationCaseLessThan2() { | ||||||
|         Whitebox.setInternalState(detector, "mDetector", mDetector) |         Whitebox.setInternalState(detector, "mDetector", mDetector) | ||||||
|         whenever(mDetector.pointerCount).thenReturn(1) |         whenever(mDetector.getPointerCount()).thenReturn(1) | ||||||
|         Assert.assertEquals(detector.rotation, 0f) |         Assert.assertEquals(detector.getRotation(), 0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|  | @ -161,12 +161,12 @@ class TransformGestureDetectorUnitTest { | ||||||
|         array[0] = 0.0f |         array[0] = 0.0f | ||||||
|         array[1] = 1.0f |         array[1] = 1.0f | ||||||
|         Whitebox.setInternalState(detector, "mDetector", mDetector) |         Whitebox.setInternalState(detector, "mDetector", mDetector) | ||||||
|         whenever(mDetector.pointerCount).thenReturn(2) |         whenever(mDetector.getPointerCount()).thenReturn(2) | ||||||
|         whenever(mDetector.startX).thenReturn(array) |         whenever(mDetector.getStartX()).thenReturn(array) | ||||||
|         whenever(mDetector.startY).thenReturn(array) |         whenever(mDetector.getStartY()).thenReturn(array) | ||||||
|         whenever(mDetector.currentX).thenReturn(array) |         whenever(mDetector.getCurrentX()).thenReturn(array) | ||||||
|         whenever(mDetector.currentY).thenReturn(array) |         whenever(mDetector.getCurrentY()).thenReturn(array) | ||||||
|         Assert.assertEquals(detector.rotation, 0f) |         Assert.assertEquals(detector.getRotation(), 0f) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Nicolas Raoul
						Nicolas Raoul