mirror of
				https://github.com/commons-app/apps-android-commons.git
				synced 2025-10-26 20:33:53 +01:00 
			
		
		
		
	With lazy loading of contributions (#3566)
This commit is contained in:
		
							parent
							
								
									c216fdf0d4
								
							
						
					
					
						commit
						d863a404f1
					
				
					 34 changed files with 1397 additions and 928 deletions
				
			
		|  | @ -42,6 +42,11 @@ dependencies { | ||||||
|     implementation 'com.dinuscxj:circleprogressbar:1.1.1' |     implementation 'com.dinuscxj:circleprogressbar:1.1.1' | ||||||
|     implementation 'com.karumi:dexter:5.0.0' |     implementation 'com.karumi:dexter:5.0.0' | ||||||
|     implementation "com.jakewharton:butterknife:$BUTTERKNIFE_VERSION" |     implementation "com.jakewharton:butterknife:$BUTTERKNIFE_VERSION" | ||||||
|  | 
 | ||||||
|  |     //paging | ||||||
|  |     implementation "androidx.paging:paging-runtime-ktx:2.1.2" | ||||||
|  |     implementation "androidx.paging:paging-rxjava2-ktx:2.1.2" | ||||||
|  | 
 | ||||||
|     kapt "com.jakewharton:butterknife-compiler:$BUTTERKNIFE_VERSION" |     kapt "com.jakewharton:butterknife-compiler:$BUTTERKNIFE_VERSION" | ||||||
|     implementation "com.hannesdorfmann:adapterdelegates4-kotlin-dsl-layoutcontainer:$ADAPTER_DELEGATES_VERSION" |     implementation "com.hannesdorfmann:adapterdelegates4-kotlin-dsl-layoutcontainer:$ADAPTER_DELEGATES_VERSION" | ||||||
|     implementation "com.hannesdorfmann:adapterdelegates4-pagination:$ADAPTER_DELEGATES_VERSION" |     implementation "com.hannesdorfmann:adapterdelegates4-pagination:$ADAPTER_DELEGATES_VERSION" | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ import android.os.Parcelable; | ||||||
| import androidx.annotation.NonNull; | import androidx.annotation.NonNull; | ||||||
| import androidx.annotation.Nullable; | import androidx.annotation.Nullable; | ||||||
| import androidx.room.Entity; | import androidx.room.Entity; | ||||||
|  | import androidx.room.PrimaryKey; | ||||||
| import fr.free.nrw.commons.location.LatLng; | import fr.free.nrw.commons.location.LatLng; | ||||||
| import fr.free.nrw.commons.media.Depictions; | import fr.free.nrw.commons.media.Depictions; | ||||||
| import fr.free.nrw.commons.utils.CommonsDateUtil; | import fr.free.nrw.commons.utils.CommonsDateUtil; | ||||||
|  | @ -15,6 +16,8 @@ import java.util.ArrayList; | ||||||
| import java.util.Date; | import java.util.Date; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
|  | import java.util.Objects; | ||||||
|  | import java.util.UUID; | ||||||
| import org.apache.commons.lang3.StringUtils; | import org.apache.commons.lang3.StringUtils; | ||||||
| import org.wikipedia.dataclient.mwapi.MwQueryPage; | import org.wikipedia.dataclient.mwapi.MwQueryPage; | ||||||
| import org.wikipedia.gallery.ExtMetadata; | import org.wikipedia.gallery.ExtMetadata; | ||||||
|  | @ -50,6 +53,8 @@ public class Media implements Parcelable { | ||||||
|     /** |     /** | ||||||
|      * Wikibase Identifier associated with media files |      * Wikibase Identifier associated with media files | ||||||
|      */ |      */ | ||||||
|  |     @PrimaryKey | ||||||
|  |     @NonNull | ||||||
|     private String pageId; |     private String pageId; | ||||||
|     private List<String> categories; // as loaded at runtime? |     private List<String> categories; // as loaded at runtime? | ||||||
|     /** |     /** | ||||||
|  | @ -64,14 +69,28 @@ public class Media implements Parcelable { | ||||||
|      * Provides local constructor |      * Provides local constructor | ||||||
|      */ |      */ | ||||||
|     public Media() { |     public Media() { | ||||||
|  |         pageId = UUID.randomUUID().toString(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public static final Creator<Media> CREATOR = new Creator<Media>() { | ||||||
|  |         @Override | ||||||
|  |         public Media createFromParcel(final Parcel source) { | ||||||
|  |             return new Media(source); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         @Override | ||||||
|  |         public Media[] newArray(final int size) { | ||||||
|  |             return new Media[size]; | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Provides a minimal constructor |      * Provides a minimal constructor | ||||||
|      * |      * | ||||||
|      * @param filename Media filename |      * @param filename Media filename | ||||||
|      */ |      */ | ||||||
|     public Media(String filename) { |     public Media(final String filename) { | ||||||
|  |         this(); | ||||||
|         this.filename = filename; |         this.filename = filename; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -86,11 +105,13 @@ public class Media implements Parcelable { | ||||||
|      * @param dateUploaded Media date uploaded |      * @param dateUploaded Media date uploaded | ||||||
|      * @param creator Media creator |      * @param creator Media creator | ||||||
|      */ |      */ | ||||||
|     public Media(Uri localUri, String imageUrl, String filename, |     public Media(final Uri localUri, final String imageUrl, final String filename, | ||||||
|         String description, |         final String description, | ||||||
|         long dataLength, Date dateCreated, Date dateUploaded, String creator) { |         final long dataLength, final Date dateCreated, final Date dateUploaded, | ||||||
|  |         final String creator) { | ||||||
|  |         this(); | ||||||
|         this.localUri = localUri; |         this.localUri = localUri; | ||||||
|         this.thumbUrl = imageUrl; |         thumbUrl = imageUrl; | ||||||
|         this.imageUrl = imageUrl; |         this.imageUrl = imageUrl; | ||||||
|         this.filename = filename; |         this.filename = filename; | ||||||
|         this.description = description; |         this.description = description; | ||||||
|  | @ -100,17 +121,80 @@ public class Media implements Parcelable { | ||||||
|         this.creator = creator; |         this.creator = creator; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public Media(Uri localUri, String filename, |     /** | ||||||
|         String description, String creator, List<String> categories) { |      * Constructor with all parameters | ||||||
|  |      */ | ||||||
|  |     public Media(final String pageId, | ||||||
|  |         final Uri localUri, | ||||||
|  |         final String thumbUrl, | ||||||
|  |         final String imageUrl, | ||||||
|  |         final String filename, | ||||||
|  |         final String description, | ||||||
|  |         final String discussion, | ||||||
|  |         final long dataLength, | ||||||
|  |         final Date dateCreated, | ||||||
|  |         final Date dateUploaded, | ||||||
|  |         final String license, | ||||||
|  |         final String licenseUrl, | ||||||
|  |         final String creator, | ||||||
|  |         final List<String> categories, | ||||||
|  |         final boolean requestedDeletion, | ||||||
|  |         final LatLng coordinates) { | ||||||
|  |         this.pageId = pageId; | ||||||
|  |         this.localUri = localUri; | ||||||
|  |         this.thumbUrl = thumbUrl; | ||||||
|  |         this.imageUrl = imageUrl; | ||||||
|  |         this.filename = filename; | ||||||
|  |         this.description = description; | ||||||
|  |         this.discussion = discussion; | ||||||
|  |         this.dataLength = dataLength; | ||||||
|  |         this.dateCreated = dateCreated; | ||||||
|  |         this.dateUploaded = dateUploaded; | ||||||
|  |         this.license = license; | ||||||
|  |         this.licenseUrl = licenseUrl; | ||||||
|  |         this.creator = creator; | ||||||
|  |         this.categories = categories; | ||||||
|  |         this.requestedDeletion = requestedDeletion; | ||||||
|  |         this.coordinates = coordinates; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public Media(final Uri localUri, final String filename, | ||||||
|  |         final String description, final String creator, final List<String> categories) { | ||||||
|         this(localUri,null, filename, |         this(localUri,null, filename, | ||||||
|             description, -1, null, new Date(), creator); |             description, -1, null, new Date(), creator); | ||||||
|         this.categories = categories; |         this.categories = categories; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public Media(String title, Date date, String user) { |     public Media(final String title, final Date date, final String user) { | ||||||
|         this(null, null, title, "", -1, date, date, user); |         this(null, null, title, "", -1, date, date, user); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     protected Media(final Parcel in) { | ||||||
|  |         localUri = in.readParcelable(Uri.class.getClassLoader()); | ||||||
|  |         thumbUrl = in.readString(); | ||||||
|  |         imageUrl = in.readString(); | ||||||
|  |         filename = in.readString(); | ||||||
|  |         thumbnailTitle = in.readString(); | ||||||
|  |         caption = in.readString(); | ||||||
|  |         description = in.readString(); | ||||||
|  |         discussion = in.readString(); | ||||||
|  |         dataLength = in.readLong(); | ||||||
|  |         final long tmpDateCreated = in.readLong(); | ||||||
|  |         dateCreated = tmpDateCreated == -1 ? null : new Date(tmpDateCreated); | ||||||
|  |         final long tmpDateUploaded = in.readLong(); | ||||||
|  |         dateUploaded = tmpDateUploaded == -1 ? null : new Date(tmpDateUploaded); | ||||||
|  |         license = in.readString(); | ||||||
|  |         licenseUrl = in.readString(); | ||||||
|  |         creator = in.readString(); | ||||||
|  |         pageId = in.readString(); | ||||||
|  |         final ArrayList<String> list = new ArrayList<>(); | ||||||
|  |         in.readStringList(list); | ||||||
|  |         categories = list; | ||||||
|  |         in.readParcelable(Depictions.class.getClassLoader()); | ||||||
|  |         requestedDeletion = in.readByte() != 0; | ||||||
|  |         coordinates = in.readParcelable(LatLng.class.getClassLoader()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Creating Media object from MWQueryPage. |      * Creating Media object from MWQueryPage. | ||||||
|      * Earlier only basic details were set for the media object but going forward, |      * Earlier only basic details were set for the media object but going forward, | ||||||
|  | @ -120,14 +204,14 @@ public class Media implements Parcelable { | ||||||
|      * @return Media object |      * @return Media object | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     public static Media from(MwQueryPage page) { |     public static Media from(final MwQueryPage page) { | ||||||
|         ImageInfo imageInfo = page.imageInfo(); |         final ImageInfo imageInfo = page.imageInfo(); | ||||||
|         if (imageInfo == null) { |         if (imageInfo == null) { | ||||||
|             return new Media(); // null is not allowed |             return new Media(); // null is not allowed | ||||||
|         } |         } | ||||||
|         ExtMetadata metadata = imageInfo.getMetadata(); |         final ExtMetadata metadata = imageInfo.getMetadata(); | ||||||
|         if (metadata == null) { |         if (metadata == null) { | ||||||
|             Media media = new Media(null, imageInfo.getOriginalUrl(), |             final Media media = new Media(null, imageInfo.getOriginalUrl(), | ||||||
|                     page.title(), "", 0, null, null, null); |                     page.title(), "", 0, null, null, null); | ||||||
|             if (!StringUtils.isBlank(imageInfo.getThumbUrl())) { |             if (!StringUtils.isBlank(imageInfo.getThumbUrl())) { | ||||||
|                 media.setThumbUrl(imageInfo.getThumbUrl()); |                 media.setThumbUrl(imageInfo.getThumbUrl()); | ||||||
|  | @ -135,7 +219,7 @@ public class Media implements Parcelable { | ||||||
|             return media; |             return media; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         Media media = new Media(null, |         final Media media = new Media(null, | ||||||
|                 imageInfo.getOriginalUrl(), |                 imageInfo.getOriginalUrl(), | ||||||
|                 page.title(), |                 page.title(), | ||||||
|             "", |             "", | ||||||
|  | @ -158,11 +242,12 @@ public class Media implements Parcelable { | ||||||
| 
 | 
 | ||||||
|         media.setDescription(metadata.imageDescription()); |         media.setDescription(metadata.imageDescription()); | ||||||
|         media.setCategories(MediaDataExtractorUtil.extractCategoriesFromList(metadata.getCategories())); |         media.setCategories(MediaDataExtractorUtil.extractCategoriesFromList(metadata.getCategories())); | ||||||
|         String latitude = metadata.getGpsLatitude(); |         final String latitude = metadata.getGpsLatitude(); | ||||||
|         String longitude = metadata.getGpsLongitude(); |         final String longitude = metadata.getGpsLongitude(); | ||||||
| 
 | 
 | ||||||
|         if (!StringUtils.isBlank(latitude) && !StringUtils.isBlank(longitude)) { |         if (!StringUtils.isBlank(latitude) && !StringUtils.isBlank(longitude)) { | ||||||
|             LatLng latLng = new LatLng(Double.parseDouble(latitude), Double.parseDouble(longitude), 0); |             final LatLng latLng = new LatLng(Double.parseDouble(latitude), | ||||||
|  |                 Double.parseDouble(longitude), 0); | ||||||
|             media.setCoordinates(latLng); |             media.setCoordinates(latLng); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -175,29 +260,17 @@ public class Media implements Parcelable { | ||||||
|      * @param metadata |      * @param metadata | ||||||
|      * @return |      * @return | ||||||
|      */ |      */ | ||||||
|     private static String getArtist(ExtMetadata metadata) { |     private static String getArtist(final ExtMetadata metadata) { | ||||||
|         try { |         try { | ||||||
|             String artistHtml = metadata.artist(); |             final String artistHtml = metadata.artist(); | ||||||
|             return artistHtml.substring(artistHtml.indexOf("title=\""), artistHtml.indexOf("\">")) |             return artistHtml.substring(artistHtml.indexOf("title=\""), artistHtml.indexOf("\">")) | ||||||
|                     .replace("title=\"User:", ""); |                     .replace("title=\"User:", ""); | ||||||
|         } catch (Exception ex) { |         } catch (final Exception ex) { | ||||||
|             return ""; |             return ""; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     @Nullable | ||||||
|      * @return pageId for the current media object*/ |  | ||||||
|     public String getPageId() { |  | ||||||
|         return pageId; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      *sets pageId for the current media object |  | ||||||
|      */ |  | ||||||
|     public void setPageId(String pageId) { |  | ||||||
|         this.pageId = pageId; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public String getThumbUrl() { |     public String getThumbUrl() { | ||||||
|         return thumbUrl; |         return thumbUrl; | ||||||
|     } |     } | ||||||
|  | @ -210,11 +283,13 @@ public class Media implements Parcelable { | ||||||
|         return filename != null ? getPageTitle().getDisplayTextWithoutNamespace().replaceFirst("[.][^.]+$", "") : ""; |         return filename != null ? getPageTitle().getDisplayTextWithoutNamespace().replaceFirst("[.][^.]+$", "") : ""; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     @Nullable | ||||||
|      * Set Caption(if available) as the thumbnail title of the image |     private static Date safeParseDate(final String dateStr) { | ||||||
|      */ |         try { | ||||||
|     public void setThumbnailTitle(String title) { |             return CommonsDateUtil.getMediaSimpleDateFormat().parse(dateStr); | ||||||
|         this.thumbnailTitle = title; |         } catch (final ParseException e) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -260,19 +335,17 @@ public class Media implements Parcelable { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the name of the file. |      * @return pageId for the current media object*/ | ||||||
|      * @param filename the new name of the file |     @NonNull | ||||||
|      */ |     public String getPageId() { | ||||||
|     public void setFilename(String filename) { |         return pageId; | ||||||
|         this.filename = filename; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the discussion of the file. |      *sets pageId for the current media object | ||||||
|      * @param discussion |  | ||||||
|      */ |      */ | ||||||
|     public void setDiscussion(String discussion) { |     public void setPageId(final String pageId) { | ||||||
|         this.discussion = discussion; |         this.pageId = pageId; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -310,11 +383,11 @@ public class Media implements Parcelable { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the file description. |      * Sets the name of the file. | ||||||
|      * @param description the new description of the file |      * @param filename the new name of the file | ||||||
|      */ |      */ | ||||||
|     public void setDescription(String description) { |     public void setFilename(final String filename) { | ||||||
|         this.description = description; |         this.filename = filename; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -326,11 +399,11 @@ public class Media implements Parcelable { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the dataLength of the file. |      * Sets the discussion of the file. | ||||||
|      * @param dataLength as a long |      * @param discussion | ||||||
|      */ |      */ | ||||||
|     public void setDataLength(long dataLength) { |     public void setDiscussion(final String discussion) { | ||||||
|         this.dataLength = dataLength; |         this.discussion = discussion; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -342,11 +415,11 @@ public class Media implements Parcelable { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the creation date of the file. |      * Sets the file description. | ||||||
|      * @param date creation date as a Date |      * @param description the new description of the file | ||||||
|      */ |      */ | ||||||
|     public void setDateCreated(Date date) { |     public void setDescription(final String description) { | ||||||
|         this.dateCreated = date; |         this.description = description; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -368,11 +441,11 @@ public class Media implements Parcelable { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the creator name of the file. |      * Sets the dataLength of the file. | ||||||
|      * @param creator creator name as a string |      * @param dataLength as a long | ||||||
|      */ |      */ | ||||||
|     public void setCreator(String creator) { |     public void setDataLength(final long dataLength) { | ||||||
|         this.creator = creator; |         this.dataLength = dataLength; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -383,8 +456,11 @@ public class Media implements Parcelable { | ||||||
|         return license; |         return license; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void setThumbUrl(String thumbUrl) { |     /** | ||||||
|         this.thumbUrl = thumbUrl; |      * Set Caption(if available) as the thumbnail title of the image | ||||||
|  |      */ | ||||||
|  |     public void setThumbnailTitle(final String title) { | ||||||
|  |         thumbnailTitle = title; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public String getLicenseUrl() { |     public String getLicenseUrl() { | ||||||
|  | @ -392,16 +468,11 @@ public class Media implements Parcelable { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the license name of the file. |      * Sets the creator name of the file. | ||||||
|      * @param license license name as a String |      * @param creator creator name as a string | ||||||
|      */ |      */ | ||||||
|     public void setLicenseInformation(String license, String licenseUrl) { |     public void setCreator(final String creator) { | ||||||
|         this.license = license; |         this.creator = creator; | ||||||
| 
 |  | ||||||
|         if (!licenseUrl.startsWith("http://") && !licenseUrl.startsWith("https://")) { |  | ||||||
|             licenseUrl = "https://" + licenseUrl; |  | ||||||
|         } |  | ||||||
|         this.licenseUrl = licenseUrl; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -413,12 +484,8 @@ public class Media implements Parcelable { | ||||||
|         return coordinates; |         return coordinates; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     public void setThumbUrl(final String thumbUrl) { | ||||||
|      * Sets the coordinates of where the file was created. |         this.thumbUrl = thumbUrl; | ||||||
|      * @param coordinates file coordinates as a LatLng |  | ||||||
|      */ |  | ||||||
|     public void setCoordinates(@Nullable LatLng coordinates) { |  | ||||||
|         this.coordinates = coordinates; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -430,6 +497,27 @@ public class Media implements Parcelable { | ||||||
|         return categories; |         return categories; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the license name of the file. | ||||||
|  |      * @param license license name as a String | ||||||
|  |      */ | ||||||
|  |     public void setLicenseInformation(final String license, String licenseUrl) { | ||||||
|  |         this.license = license; | ||||||
|  | 
 | ||||||
|  |         if (!licenseUrl.startsWith("http://") && !licenseUrl.startsWith("https://")) { | ||||||
|  |             licenseUrl = "https://" + licenseUrl; | ||||||
|  |         } | ||||||
|  |         this.licenseUrl = licenseUrl; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the coordinates of where the file was created. | ||||||
|  |      * @param coordinates file coordinates as a LatLng | ||||||
|  |      */ | ||||||
|  |     public void setCoordinates(@Nullable final LatLng coordinates) { | ||||||
|  |         this.coordinates = coordinates; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the categories the file falls under. |      * Sets the categories the file falls under. | ||||||
|      * </p> |      * </p> | ||||||
|  | @ -437,26 +525,10 @@ public class Media implements Parcelable { | ||||||
|      * and then add the specified ones. |      * and then add the specified ones. | ||||||
|      * @param categories file categories as a list of Strings |      * @param categories file categories as a list of Strings | ||||||
|      */ |      */ | ||||||
|     public void setCategories(List<String> categories) { |     public void setCategories(final List<String> categories) { | ||||||
|         this.categories = categories; |         this.categories = categories; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Nullable private static Date safeParseDate(String dateStr) { |  | ||||||
|         try { |  | ||||||
|             return CommonsDateUtil.getIso8601DateFormatShort().parse(dateStr); |  | ||||||
|         } catch (ParseException e) { |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Set requested deletion to true |  | ||||||
|      * @param requestedDeletion |  | ||||||
|      */ |  | ||||||
|     public void setRequestedDeletion(boolean requestedDeletion){ |  | ||||||
|         this.requestedDeletion = requestedDeletion; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |     /** | ||||||
|      * Get the value of requested deletion |      * Get the value of requested deletion | ||||||
|      * @return boolean requestedDeletion |      * @return boolean requestedDeletion | ||||||
|  | @ -465,12 +537,20 @@ public class Media implements Parcelable { | ||||||
|         return requestedDeletion; |         return requestedDeletion; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Set requested deletion to true | ||||||
|  |      * @param requestedDeletion | ||||||
|  |      */ | ||||||
|  |     public void setRequestedDeletion(final boolean requestedDeletion) { | ||||||
|  |         this.requestedDeletion = requestedDeletion; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Sets the license name of the file. |      * Sets the license name of the file. | ||||||
|      * |      * | ||||||
|      * @param license license name as a String |      * @param license license name as a String | ||||||
|      */ |      */ | ||||||
|     public void setLicense(String license) { |     public void setLicense(final String license) { | ||||||
|         this.license = license; |         this.license = license; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -482,15 +562,10 @@ public class Media implements Parcelable { | ||||||
|      * This function sets captions |      * This function sets captions | ||||||
|      * @param caption |      * @param caption | ||||||
|      */ |      */ | ||||||
|     public void setCaption(String caption) { |     public void setCaption(final String caption) { | ||||||
|         this.caption = caption; |         this.caption = caption; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /* Sets depictions for the current media obtained fro  Wikibase API*/ |  | ||||||
|     public void setDepictions(Depictions depictions) { |  | ||||||
|         this.depictions = depictions; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setLocalUri(@Nullable final Uri localUri) { |     public void setLocalUri(@Nullable final Uri localUri) { | ||||||
|         this.localUri = localUri; |         this.localUri = localUri; | ||||||
|     } |     } | ||||||
|  | @ -516,6 +591,19 @@ public class Media implements Parcelable { | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /* Sets depictions for the current media obtained fro  Wikibase API*/ | ||||||
|  |     public void setDepictions(final Depictions depictions) { | ||||||
|  |         this.depictions = depictions; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets the creation date of the file. | ||||||
|  |      * @param date creation date as a Date | ||||||
|  |      */ | ||||||
|  |     public void setDateCreated(final Date date) { | ||||||
|  |         dateCreated = date; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Creates a way to transfer information between two or more |      * Creates a way to transfer information between two or more | ||||||
|      * activities. |      * activities. | ||||||
|  | @ -523,63 +611,70 @@ public class Media implements Parcelable { | ||||||
|      * @param flags Parcel flag |      * @param flags Parcel flag | ||||||
|      */ |      */ | ||||||
|     @Override |     @Override | ||||||
|     public void writeToParcel(Parcel dest, int flags) { |     public void writeToParcel(final Parcel dest, final int flags) { | ||||||
|         dest.writeParcelable(this.localUri, flags); |         dest.writeParcelable(localUri, flags); | ||||||
|         dest.writeString(this.thumbUrl); |         dest.writeString(thumbUrl); | ||||||
|         dest.writeString(this.imageUrl); |         dest.writeString(imageUrl); | ||||||
|         dest.writeString(this.filename); |         dest.writeString(filename); | ||||||
|         dest.writeString(this.thumbnailTitle); |         dest.writeString(thumbnailTitle); | ||||||
|         dest.writeString(this.caption); |         dest.writeString(caption); | ||||||
|         dest.writeString(this.description); |         dest.writeString(description); | ||||||
|         dest.writeString(this.discussion); |         dest.writeString(discussion); | ||||||
|         dest.writeLong(this.dataLength); |         dest.writeLong(dataLength); | ||||||
|         dest.writeLong(this.dateCreated != null ? this.dateCreated.getTime() : -1); |         dest.writeLong(dateCreated != null ? dateCreated.getTime() : -1); | ||||||
|         dest.writeLong(this.dateUploaded != null ? this.dateUploaded.getTime() : -1); |         dest.writeLong(dateUploaded != null ? dateUploaded.getTime() : -1); | ||||||
|         dest.writeString(this.license); |         dest.writeString(license); | ||||||
|         dest.writeString(this.licenseUrl); |         dest.writeString(licenseUrl); | ||||||
|         dest.writeString(this.creator); |         dest.writeString(creator); | ||||||
|         dest.writeString(this.pageId); |         dest.writeString(pageId); | ||||||
|         dest.writeStringList(this.categories); |         dest.writeStringList(categories); | ||||||
|         dest.writeParcelable(this.depictions, flags); |         dest.writeParcelable(depictions, flags); | ||||||
|         dest.writeByte(this.requestedDeletion ? (byte) 1 : (byte) 0); |         dest.writeByte(requestedDeletion ? (byte) 1 : (byte) 0); | ||||||
|         dest.writeParcelable(this.coordinates, flags); |         dest.writeParcelable(coordinates, flags); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     protected Media(Parcel in) { |     /** | ||||||
|         this.localUri = in.readParcelable(Uri.class.getClassLoader()); |      * Equals implementation that matches all parameters for equality check | ||||||
|         this.thumbUrl = in.readString(); |      */ | ||||||
|         this.imageUrl = in.readString(); |     @Override | ||||||
|         this.filename = in.readString(); |     public boolean equals(final Object o) { | ||||||
|         this.thumbnailTitle = in.readString(); |         if (this == o) { | ||||||
|         this.caption = in.readString(); |             return true; | ||||||
|         this.description = in.readString(); |         } | ||||||
|         this.discussion = in.readString(); |         if (!(o instanceof Media)) { | ||||||
|         this.dataLength = in.readLong(); |             return false; | ||||||
|         long tmpDateCreated = in.readLong(); |         } | ||||||
|         this.dateCreated = tmpDateCreated == -1 ? null : new Date(tmpDateCreated); |         final Media media = (Media) o; | ||||||
|         long tmpDateUploaded = in.readLong(); |         return getDataLength() == media.getDataLength() && | ||||||
|         this.dateUploaded = tmpDateUploaded == -1 ? null : new Date(tmpDateUploaded); |             isRequestedDeletion() == media.isRequestedDeletion() && | ||||||
|         this.license = in.readString(); |             Objects.equals(getLocalUri(), media.getLocalUri()) && | ||||||
|         this.licenseUrl = in.readString(); |             Objects.equals(getThumbUrl(), media.getThumbUrl()) && | ||||||
|         this.creator = in.readString(); |             Objects.equals(getImageUrl(), media.getImageUrl()) && | ||||||
|         this.pageId = in.readString(); |             Objects.equals(getFilename(), media.getFilename()) && | ||||||
|         final ArrayList<String> list = new ArrayList<>(); |             Objects.equals(getThumbnailTitle(), media.getThumbnailTitle()) && | ||||||
|         in.readStringList(list); |             Objects.equals(getCaption(), media.getCaption()) && | ||||||
|         this.categories=list; |             Objects.equals(getDescription(), media.getDescription()) && | ||||||
|         in.readParcelable(Depictions.class.getClassLoader()); |             Objects.equals(getDiscussion(), media.getDiscussion()) && | ||||||
|         this.requestedDeletion = in.readByte() != 0; |             Objects.equals(getDateCreated(), media.getDateCreated()) && | ||||||
|         this.coordinates = in.readParcelable(LatLng.class.getClassLoader()); |             Objects.equals(getDateUploaded(), media.getDateUploaded()) && | ||||||
|  |             Objects.equals(getLicense(), media.getLicense()) && | ||||||
|  |             Objects.equals(getLicenseUrl(), media.getLicenseUrl()) && | ||||||
|  |             Objects.equals(getCreator(), media.getCreator()) && | ||||||
|  |             getPageId().equals(media.getPageId()) && | ||||||
|  |             Objects.equals(getCategories(), media.getCategories()) && | ||||||
|  |             Objects.equals(getDepictions(), media.getDepictions()) && | ||||||
|  |             Objects.equals(getCoordinates(), media.getCoordinates()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static final Creator<Media> CREATOR = new Creator<Media>() { |     /** | ||||||
|         @Override |      * Hashcode implementation that uses all parameters for calculating hash | ||||||
|         public Media createFromParcel(Parcel source) { |      */ | ||||||
|             return new Media(source); |     @Override | ||||||
|         } |     public int hashCode() { | ||||||
| 
 |         return Objects | ||||||
|         @Override |             .hash(getLocalUri(), getThumbUrl(), getImageUrl(), getFilename(), getThumbnailTitle(), | ||||||
|         public Media[] newArray(int size) { |                 getCaption(), getDescription(), getDiscussion(), getDataLength(), getDateCreated(), | ||||||
|             return new Media[size]; |                 getDateUploaded(), getLicense(), getLicenseUrl(), getCreator(), getPageId(), | ||||||
|         } |                 getCategories(), getDepictions(), isRequestedDeletion(), getCoordinates()); | ||||||
|     }; |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -21,7 +21,8 @@ public final class OkHttpConnectionFactory { | ||||||
|     @NonNull private static final Cache NET_CACHE = new Cache(new File(CommonsApplication.getInstance().getCacheDir(), |     @NonNull private static final Cache NET_CACHE = new Cache(new File(CommonsApplication.getInstance().getCacheDir(), | ||||||
|             CACHE_DIR_NAME), NET_CACHE_SIZE); |             CACHE_DIR_NAME), NET_CACHE_SIZE); | ||||||
| 
 | 
 | ||||||
|     @NonNull private static final OkHttpClient CLIENT = createClient(); |     @NonNull | ||||||
|  |     private static final OkHttpClient CLIENT = createClient(); | ||||||
| 
 | 
 | ||||||
|     @NonNull public static OkHttpClient getClient() { |     @NonNull public static OkHttpClient getClient() { | ||||||
|         return CLIENT; |         return CLIENT; | ||||||
|  | @ -40,7 +41,7 @@ public final class OkHttpConnectionFactory { | ||||||
| 
 | 
 | ||||||
|     private static HttpLoggingInterceptor getLoggingInterceptor() { |     private static HttpLoggingInterceptor getLoggingInterceptor() { | ||||||
|         final HttpLoggingInterceptor httpLoggingInterceptor = new HttpLoggingInterceptor() |         final HttpLoggingInterceptor httpLoggingInterceptor = new HttpLoggingInterceptor() | ||||||
|                 .setLevel(Level.BASIC); |             .setLevel(Level.BASIC); | ||||||
| 
 | 
 | ||||||
|         httpLoggingInterceptor.redactHeader("Authorization"); |         httpLoggingInterceptor.redactHeader("Authorization"); | ||||||
|         httpLoggingInterceptor.redactHeader("Cookie"); |         httpLoggingInterceptor.redactHeader("Cookie"); | ||||||
|  | @ -49,7 +50,10 @@ public final class OkHttpConnectionFactory { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private static class CommonHeaderRequestInterceptor implements Interceptor { |     private static class CommonHeaderRequestInterceptor implements Interceptor { | ||||||
|         @Override @NonNull public Response intercept(@NonNull final Chain chain) throws IOException { | 
 | ||||||
|  |         @Override | ||||||
|  |         @NonNull | ||||||
|  |         public Response intercept(@NonNull final Chain chain) throws IOException { | ||||||
|             final Request request = chain.request().newBuilder() |             final Request request = chain.request().newBuilder() | ||||||
|                     .header("User-Agent", CommonsApplication.getInstance().getUserAgent()) |                     .header("User-Agent", CommonsApplication.getInstance().getUserAgent()) | ||||||
|                     .build(); |                     .build(); | ||||||
|  | @ -61,16 +65,18 @@ public final class OkHttpConnectionFactory { | ||||||
| 
 | 
 | ||||||
|         private static final String ERRORS_PREFIX = "{\"error"; |         private static final String ERRORS_PREFIX = "{\"error"; | ||||||
| 
 | 
 | ||||||
|         @Override @NonNull public Response intercept(@NonNull final Chain chain) throws IOException { |         @Override | ||||||
|  |         @NonNull | ||||||
|  |         public Response intercept(@NonNull final Chain chain) throws IOException { | ||||||
|             final Response rsp = chain.proceed(chain.request()); |             final Response rsp = chain.proceed(chain.request()); | ||||||
|             if (rsp.isSuccessful()) { |             if (rsp.isSuccessful()) { | ||||||
|                 try (final ResponseBody responseBody = rsp.peekBody(ERRORS_PREFIX.length())) { |                 try (final ResponseBody responseBody = rsp.peekBody(ERRORS_PREFIX.length())) { | ||||||
|                     if (ERRORS_PREFIX.equals(responseBody.string())){ |                     if (ERRORS_PREFIX.equals(responseBody.string())) { | ||||||
|                         try (final ResponseBody body = rsp.body()) { |                         try (final ResponseBody body = rsp.body()) { | ||||||
|                             throw new IOException(body.string()); |                             throw new IOException(body.string()); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 }catch (final IOException e){ |                 } catch (final IOException e) { | ||||||
|                     Timber.e(e); |                     Timber.e(e); | ||||||
|                 } |                 } | ||||||
|                 return rsp; |                 return rsp; | ||||||
|  |  | ||||||
|  | @ -2,7 +2,6 @@ package fr.free.nrw.commons.contributions; | ||||||
| 
 | 
 | ||||||
| import android.os.Parcel; | import android.os.Parcel; | ||||||
| import androidx.room.Entity; | import androidx.room.Entity; | ||||||
| import androidx.room.PrimaryKey; |  | ||||||
| import fr.free.nrw.commons.Media; | import fr.free.nrw.commons.Media; | ||||||
| import fr.free.nrw.commons.auth.SessionManager; | import fr.free.nrw.commons.auth.SessionManager; | ||||||
| import fr.free.nrw.commons.upload.UploadMediaDetail; | import fr.free.nrw.commons.upload.UploadMediaDetail; | ||||||
|  | @ -13,7 +12,7 @@ import java.util.ArrayList; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import org.wikipedia.dataclient.mwapi.MwQueryLogEvent; | import java.util.Objects; | ||||||
| 
 | 
 | ||||||
| @Entity(tableName = "contribution") | @Entity(tableName = "contribution") | ||||||
| public class Contribution extends Media { | public class Contribution extends Media { | ||||||
|  | @ -24,24 +23,21 @@ public class Contribution extends Media { | ||||||
|     public static final int STATE_QUEUED = 2; |     public static final int STATE_QUEUED = 2; | ||||||
|     public static final int STATE_IN_PROGRESS = 3; |     public static final int STATE_IN_PROGRESS = 3; | ||||||
| 
 | 
 | ||||||
|     @PrimaryKey (autoGenerate = true) |  | ||||||
|     private long _id; |  | ||||||
|     private int state; |     private int state; | ||||||
|     private long transferred; |     private long transferred; | ||||||
|     private String decimalCoords; |     private String decimalCoords; | ||||||
|     private String dateCreatedSource; |     private String dateCreatedSource; | ||||||
|     private WikidataPlace wikidataPlace; |     private WikidataPlace wikidataPlace; | ||||||
|     /** |     /** | ||||||
|      * Each depiction loaded in depictions activity is associated with a wikidata entity id, |      * Each depiction loaded in depictions activity is associated with a wikidata entity id, this Id | ||||||
|      * this Id is in turn used to upload depictions to wikibase |      * is in turn used to upload depictions to wikibase | ||||||
|      */ |      */ | ||||||
|     private List<DepictedItem> depictedItems = new ArrayList<>(); |     private List<DepictedItem> depictedItems = new ArrayList<>(); | ||||||
|     private String mimeType; |     private String mimeType; | ||||||
|     /** |     /** | ||||||
|      * This hasmap stores the list of multilingual captions, where |      * This hasmap stores the list of multilingual captions, where key of the HashMap is the language | ||||||
|      * key of the HashMap is the language and value is the caption in the corresponding language |      * and value is the caption in the corresponding language Ex: key = "en", value: "<caption in | ||||||
|      * Ex: key = "en", value: "<caption in short in English>" |      * short in English>" key = "de" , value: "<caption in german>" | ||||||
|      *     key = "de" , value: "<caption in german>" |  | ||||||
|      */ |      */ | ||||||
|     private Map<String, String> captions = new HashMap<>(); |     private Map<String, String> captions = new HashMap<>(); | ||||||
| 
 | 
 | ||||||
|  | @ -55,20 +51,13 @@ public class Contribution extends Media { | ||||||
|             UploadMediaDetail.formatList(item.getUploadMediaDetails()), |             UploadMediaDetail.formatList(item.getUploadMediaDetails()), | ||||||
|             sessionManager.getAuthorName(), |             sessionManager.getAuthorName(), | ||||||
|             categories); |             categories); | ||||||
|         captions =  UploadMediaDetail.formatCaptions(item.getUploadMediaDetails()); |         captions = UploadMediaDetail.formatCaptions(item.getUploadMediaDetails()); | ||||||
|         decimalCoords = item.getGpsCoords().getDecimalCoords(); |         decimalCoords = item.getGpsCoords().getDecimalCoords(); | ||||||
|         dateCreatedSource = ""; |         dateCreatedSource = ""; | ||||||
|         this.depictedItems = depictedItems; |         this.depictedItems = depictedItems; | ||||||
|         wikidataPlace = WikidataPlace.from(item.getPlace()); |         wikidataPlace = WikidataPlace.from(item.getPlace()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public Contribution(final MwQueryLogEvent queryLogEvent, final String user) { |  | ||||||
|         super(queryLogEvent.title(),queryLogEvent.date(), user); |  | ||||||
|         decimalCoords = ""; |  | ||||||
|         dateCreatedSource = ""; |  | ||||||
|         state = STATE_COMPLETED; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setDateCreatedSource(final String dateCreatedSource) { |     public void setDateCreatedSource(final String dateCreatedSource) { | ||||||
|         this.dateCreatedSource = dateCreatedSource; |         this.dateCreatedSource = dateCreatedSource; | ||||||
|     } |     } | ||||||
|  | @ -108,14 +97,6 @@ public class Contribution extends Media { | ||||||
|         return wikidataPlace; |         return wikidataPlace; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public long get_id() { |  | ||||||
|         return _id; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void set_id(final long _id) { |  | ||||||
|         this._id = _id; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public String getDecimalCoords() { |     public String getDecimalCoords() { | ||||||
|         return decimalCoords; |         return decimalCoords; | ||||||
|     } |     } | ||||||
|  | @ -128,29 +109,30 @@ public class Contribution extends Media { | ||||||
|         this.depictedItems = depictedItems; |         this.depictedItems = depictedItems; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void setMimeType(String mimeType) { |     public String getMimeType() { | ||||||
|       this.mimeType = mimeType; |         return mimeType; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public String getMimeType() { |     public void setMimeType(final String mimeType) { | ||||||
|       return mimeType; |         this.mimeType = mimeType; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Captions are a feature part of Structured data. They are meant to store short, multilingual descriptions about files |      * Captions are a feature part of Structured data. They are meant to store short, multilingual | ||||||
|      * This is a replacement of the previously used titles for images (titles were not multilingual) |      * descriptions about files This is a replacement of the previously used titles for images (titles | ||||||
|      * Also now captions replace the previous convention of using title for filename |      * were not multilingual) Also now captions replace the previous convention of using title for | ||||||
|      * |      * filename | ||||||
|  |      * <p> | ||||||
|      * key of the HashMap is the language and value is the caption in the corresponding language |      * key of the HashMap is the language and value is the caption in the corresponding language | ||||||
|      * |      * <p> | ||||||
|      * returns list of captions stored in hashmap |      * returns list of captions stored in hashmap | ||||||
|      */ |      */ | ||||||
|     public Map<String, String> getCaptions() { |     public Map<String, String> getCaptions() { | ||||||
|       return captions; |         return captions; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void setCaptions(Map<String, String> captions) { |     public void setCaptions(Map<String, String> captions) { | ||||||
|       this.captions = captions; |         this.captions = captions; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  | @ -161,7 +143,6 @@ public class Contribution extends Media { | ||||||
|     @Override |     @Override | ||||||
|     public void writeToParcel(final Parcel dest, final int flags) { |     public void writeToParcel(final Parcel dest, final int flags) { | ||||||
|         super.writeToParcel(dest, flags); |         super.writeToParcel(dest, flags); | ||||||
|         dest.writeLong(_id); |  | ||||||
|         dest.writeInt(state); |         dest.writeInt(state); | ||||||
|         dest.writeLong(transferred); |         dest.writeLong(transferred); | ||||||
|         dest.writeString(decimalCoords); |         dest.writeString(decimalCoords); | ||||||
|  | @ -169,9 +150,24 @@ public class Contribution extends Media { | ||||||
|         dest.writeSerializable((HashMap) captions); |         dest.writeSerializable((HashMap) captions); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Constructor that takes Media object and state as parameters and builds a new Contribution object | ||||||
|  |      * @param media | ||||||
|  |      * @param state | ||||||
|  |      */ | ||||||
|  |     public Contribution(Media media, int state) { | ||||||
|  |         super(media.getPageId(), | ||||||
|  |             media.getLocalUri(), media.getThumbUrl(), media.getImageUrl(), media.getFilename(), | ||||||
|  |             media.getDescription(), | ||||||
|  |             media.getDiscussion(), | ||||||
|  |             media.getDataLength(), media.getDateCreated(), media.getDateUploaded(), | ||||||
|  |             media.getLicense(), media.getLicenseUrl(), media.getCreator(), media.getCategories(), | ||||||
|  |             media.isRequestedDeletion(), media.getCoordinates()); | ||||||
|  |         this.state = state; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     protected Contribution(final Parcel in) { |     protected Contribution(final Parcel in) { | ||||||
|         super(in); |         super(in); | ||||||
|         _id = in.readLong(); |  | ||||||
|         state = in.readInt(); |         state = in.readInt(); | ||||||
|         transferred = in.readLong(); |         transferred = in.readLong(); | ||||||
|         decimalCoords = in.readString(); |         decimalCoords = in.readString(); | ||||||
|  | @ -190,4 +186,35 @@ public class Contribution extends Media { | ||||||
|             return new Contribution[size]; |             return new Contribution[size]; | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Equals implementation of Contributions that compares all parameters for checking equality | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public boolean equals(final Object o) { | ||||||
|  |         if (this == o) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         if (!(o instanceof Contribution)) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         final Contribution that = (Contribution) o; | ||||||
|  |         return getState() == that.getState() && getTransferred() == that.getTransferred() && Objects | ||||||
|  |             .equals(getDecimalCoords(), that.getDecimalCoords()) && Objects | ||||||
|  |             .equals(getDateCreatedSource(), that.getDateCreatedSource()) && Objects | ||||||
|  |             .equals(getWikidataPlace(), that.getWikidataPlace()) && Objects | ||||||
|  |             .equals(getDepictedItems(), that.getDepictedItems()) && Objects | ||||||
|  |             .equals(getMimeType(), that.getMimeType()) && Objects | ||||||
|  |             .equals(getCaptions(), that.getCaptions()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Hash code implementation of contributions that considers all parameters for calculating hash. | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public int hashCode() { | ||||||
|  |         return Objects | ||||||
|  |             .hash(getState(), getTransferred(), getDecimalCoords(), getDateCreatedSource(), | ||||||
|  |                 getWikidataPlace(), getDepictedItems(), getMimeType(), getCaptions()); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,89 @@ | ||||||
|  | package fr.free.nrw.commons.contributions | ||||||
|  | 
 | ||||||
|  | import androidx.paging.PagedList.BoundaryCallback | ||||||
|  | import fr.free.nrw.commons.Media | ||||||
|  | import fr.free.nrw.commons.auth.SessionManager | ||||||
|  | import fr.free.nrw.commons.di.CommonsApplicationModule | ||||||
|  | import fr.free.nrw.commons.media.MediaClient | ||||||
|  | import io.reactivex.Scheduler | ||||||
|  | import io.reactivex.disposables.CompositeDisposable | ||||||
|  | import timber.log.Timber | ||||||
|  | import java.util.* | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Named | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Class that extends PagedList.BoundaryCallback for contributions list It defines the action that | ||||||
|  |  * is triggered for various boundary conditions in the list | ||||||
|  |  */ | ||||||
|  | class ContributionBoundaryCallback @Inject constructor( | ||||||
|  |     private val repository: ContributionsRepository, | ||||||
|  |     private val sessionManager: SessionManager, | ||||||
|  |     private val mediaClient: MediaClient, | ||||||
|  |     @param:Named(CommonsApplicationModule.IO_THREAD) private val ioThreadScheduler: Scheduler | ||||||
|  | ) : BoundaryCallback<Contribution>() { | ||||||
|  |     private val compositeDisposable: CompositeDisposable = CompositeDisposable() | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * It is triggered when the list has no items User's Contributions are then fetched from the | ||||||
|  |      * network | ||||||
|  |      */ | ||||||
|  |     override fun onZeroItemsLoaded() { | ||||||
|  |         fetchContributions() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * It is triggered when the user scrolls to the top of the list User's Contributions are then | ||||||
|  |      * fetched from the network | ||||||
|  |      * */ | ||||||
|  |     override fun onItemAtFrontLoaded(itemAtFront: Contribution) { | ||||||
|  |         fetchContributions() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * It is triggered when the user scrolls to the end of the list. User's Contributions are then | ||||||
|  |      * fetched from the network | ||||||
|  |      */ | ||||||
|  |     override fun onItemAtEndLoaded(itemAtEnd: Contribution) { | ||||||
|  |         fetchContributions() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Fetches contributions using the MediaWiki API | ||||||
|  |      */ | ||||||
|  |     fun fetchContributions() { | ||||||
|  |         if (mediaClient.doesMediaListForUserHaveMorePages(sessionManager.userName).not()) { | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |         compositeDisposable.add( | ||||||
|  |             mediaClient.getMediaListForUser(sessionManager.userName) | ||||||
|  |                 .map { mediaList: List<Media?> -> | ||||||
|  |                     mediaList.map { | ||||||
|  |                         Contribution(it, Contribution.STATE_COMPLETED) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 .subscribeOn(ioThreadScheduler) | ||||||
|  |                 .subscribe( | ||||||
|  |                     ::saveContributionsToDB | ||||||
|  |                 ) { error: Throwable -> | ||||||
|  |                     Timber.e( | ||||||
|  |                         "Failed to fetch contributions: %s", | ||||||
|  |                         error.message | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Saves the contributions the the local DB | ||||||
|  |      */ | ||||||
|  |     private fun saveContributionsToDB(contributions: List<Contribution>) { | ||||||
|  |         compositeDisposable.add( | ||||||
|  |             repository.save(contributions) | ||||||
|  |                 .subscribeOn(ioThreadScheduler) | ||||||
|  |                 .subscribe { longs: List<Long?>? -> | ||||||
|  |                     repository["last_fetch_timestamp"] = System.currentTimeMillis() | ||||||
|  |                 } | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,6 +1,6 @@ | ||||||
| package fr.free.nrw.commons.contributions; | package fr.free.nrw.commons.contributions; | ||||||
| 
 | 
 | ||||||
| import androidx.lifecycle.LiveData; | import androidx.paging.DataSource; | ||||||
| import androidx.room.Dao; | import androidx.room.Dao; | ||||||
| import androidx.room.Delete; | import androidx.room.Delete; | ||||||
| import androidx.room.Insert; | import androidx.room.Insert; | ||||||
|  | @ -15,40 +15,55 @@ import java.util.List; | ||||||
| @Dao | @Dao | ||||||
| public abstract class ContributionDao { | public abstract class ContributionDao { | ||||||
| 
 | 
 | ||||||
|     @Query("SELECT * FROM contribution order by dateUploaded DESC") |   @Query("SELECT * FROM contribution order by dateUploaded DESC") | ||||||
|     abstract LiveData<List<Contribution>> fetchContributions(); |   abstract DataSource.Factory<Integer, Contribution> fetchContributions(); | ||||||
| 
 | 
 | ||||||
|     @Insert(onConflict = OnConflictStrategy.REPLACE) |   @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|     public abstract Single<Long> save(Contribution contribution); |   public abstract void saveSynchronous(Contribution contribution); | ||||||
| 
 | 
 | ||||||
|     public Completable deleteAllAndSave(List<Contribution> contributions){ |   public Completable save(final Contribution contribution) { | ||||||
|         return Completable.fromAction(() -> deleteAllAndSaveTransaction(contributions)); |     return Completable | ||||||
|     } |         .fromAction(() -> saveSynchronous(contribution)); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|     @Transaction |   @Transaction | ||||||
|     public void deleteAllAndSaveTransaction(List<Contribution> contributions){ |   public void deleteAndSaveContribution(final Contribution oldContribution, | ||||||
|         deleteAll(Contribution.STATE_COMPLETED); |       final Contribution newContribution) { | ||||||
|         save(contributions); |     deleteSynchronous(oldContribution); | ||||||
|     } |     saveSynchronous(newContribution); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|     @Insert |   public Completable saveAndDelete(final Contribution oldContribution, | ||||||
|     public abstract void save(List<Contribution> contribution); |       final Contribution newContribution) { | ||||||
|  |     return Completable | ||||||
|  |         .fromAction(() -> deleteAndSaveContribution(oldContribution, newContribution)); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|     @Delete |   @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|     public abstract Single<Integer> delete(Contribution contribution); |   public abstract Single<List<Long>> save(List<Contribution> contribution); | ||||||
| 
 | 
 | ||||||
|     @Query("SELECT * from contribution WHERE filename=:fileName") |   @Delete | ||||||
|     public abstract List<Contribution> getContributionWithTitle(String fileName); |   public abstract void deleteSynchronous(Contribution contribution); | ||||||
| 
 | 
 | ||||||
|     @Query("UPDATE contribution SET state=:state WHERE state in (:toUpdateStates)") |   public Completable delete(final Contribution contribution) { | ||||||
|     public abstract Single<Integer> updateStates(int state, int[] toUpdateStates); |     return Completable | ||||||
|  |         .fromAction(() -> deleteSynchronous(contribution)); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|     @Query("Delete FROM contribution") |   @Query("SELECT * from contribution WHERE filename=:fileName") | ||||||
|     public abstract void deleteAll(); |   public abstract List<Contribution> getContributionWithTitle(String fileName); | ||||||
| 
 | 
 | ||||||
|     @Query("Delete FROM contribution WHERE state = :state") |   @Query("UPDATE contribution SET state=:state WHERE state in (:toUpdateStates)") | ||||||
|     public abstract void deleteAll(int state); |   public abstract Single<Integer> updateStates(int state, int[] toUpdateStates); | ||||||
| 
 | 
 | ||||||
|     @Update |   @Query("Delete FROM contribution") | ||||||
|     public abstract Single<Integer> update(Contribution contribution); |   public abstract void deleteAll(); | ||||||
|  | 
 | ||||||
|  |   @Update | ||||||
|  |   public abstract void updateSynchronous(Contribution contribution); | ||||||
|  | 
 | ||||||
|  |   public Completable update(final Contribution contribution) { | ||||||
|  |     return Completable | ||||||
|  |         .fromAction(() -> updateSynchronous(contribution)); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -22,7 +22,6 @@ import fr.free.nrw.commons.media.MediaClient; | ||||||
| import io.reactivex.android.schedulers.AndroidSchedulers; | import io.reactivex.android.schedulers.AndroidSchedulers; | ||||||
| import io.reactivex.disposables.CompositeDisposable; | import io.reactivex.disposables.CompositeDisposable; | ||||||
| import io.reactivex.schedulers.Schedulers; | import io.reactivex.schedulers.Schedulers; | ||||||
| import java.util.Random; |  | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
| 
 | 
 | ||||||
| public class ContributionViewHolder extends RecyclerView.ViewHolder { | public class ContributionViewHolder extends RecyclerView.ViewHolder { | ||||||
|  | @ -39,23 +38,22 @@ public class ContributionViewHolder extends RecyclerView.ViewHolder { | ||||||
| 
 | 
 | ||||||
|     private int position; |     private int position; | ||||||
|     private Contribution contribution; |     private Contribution contribution; | ||||||
|     private Random random = new Random(); |     private final CompositeDisposable compositeDisposable = new CompositeDisposable(); | ||||||
|     private CompositeDisposable compositeDisposable = new CompositeDisposable(); |  | ||||||
|     private final MediaClient mediaClient; |     private final MediaClient mediaClient; | ||||||
| 
 | 
 | ||||||
|     ContributionViewHolder(View parent, Callback callback, |     ContributionViewHolder(final View parent, final Callback callback, | ||||||
|         MediaClient mediaClient) { |         final MediaClient mediaClient) { | ||||||
|         super(parent); |         super(parent); | ||||||
|         this.mediaClient = mediaClient; |         this.mediaClient = mediaClient; | ||||||
|         ButterKnife.bind(this, parent); |         ButterKnife.bind(this, parent); | ||||||
|         this.callback=callback; |         this.callback=callback; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void init(int position, Contribution contribution) { |     public void init(final int position, final Contribution contribution) { | ||||||
|         this.contribution = contribution; |         this.contribution = contribution; | ||||||
|         fetchAndDisplayCaption(contribution); |         fetchAndDisplayCaption(contribution); | ||||||
|         this.position = position; |         this.position = position; | ||||||
|         String imageSource = chooseImageSource(contribution.getThumbUrl(), contribution.getLocalUri()); |         final String imageSource = chooseImageSource(contribution.getThumbUrl(), contribution.getLocalUri()); | ||||||
|         if (!TextUtils.isEmpty(imageSource)) { |         if (!TextUtils.isEmpty(imageSource)) { | ||||||
|             final ImageRequest imageRequest = |             final ImageRequest imageRequest = | ||||||
|                 ImageRequestBuilder.newBuilderWithSource(Uri.parse(imageSource)) |                 ImageRequestBuilder.newBuilderWithSource(Uri.parse(imageSource)) | ||||||
|  | @ -84,8 +82,8 @@ public class ContributionViewHolder extends RecyclerView.ViewHolder { | ||||||
|                 stateView.setVisibility(View.GONE); |                 stateView.setVisibility(View.GONE); | ||||||
|                 progressView.setVisibility(View.VISIBLE); |                 progressView.setVisibility(View.VISIBLE); | ||||||
|                 failedImageOptions.setVisibility(View.GONE); |                 failedImageOptions.setVisibility(View.GONE); | ||||||
|                 long total = contribution.getDataLength(); |                 final long total = contribution.getDataLength(); | ||||||
|                 long transferred = contribution.getTransferred(); |                 final long transferred = contribution.getTransferred(); | ||||||
|                 if (transferred == 0 || transferred >= total) { |                 if (transferred == 0 || transferred >= total) { | ||||||
|                     progressView.setIndeterminate(true); |                     progressView.setIndeterminate(true); | ||||||
|                 } else { |                 } else { | ||||||
|  | @ -107,14 +105,14 @@ public class ContributionViewHolder extends RecyclerView.ViewHolder { | ||||||
|      * |      * | ||||||
|      * @param contribution |      * @param contribution | ||||||
|      */ |      */ | ||||||
|     private void fetchAndDisplayCaption(Contribution contribution) { |     private void fetchAndDisplayCaption(final Contribution contribution) { | ||||||
|         if ((contribution.getState() != Contribution.STATE_COMPLETED)) { |         if ((contribution.getState() != Contribution.STATE_COMPLETED)) { | ||||||
|             titleView.setText(contribution.getDisplayTitle()); |             titleView.setText(contribution.getDisplayTitle()); | ||||||
|         } else { |         } else { | ||||||
|             final String pageId = contribution.getPageId(); |             final String pageId = contribution.getPageId(); | ||||||
|             if (pageId != null) { |             if (pageId != null) { | ||||||
|                 Timber.d("Fetching caption for %s", contribution.getFilename()); |                 Timber.d("Fetching caption for %s", contribution.getFilename()); | ||||||
|                 String wikibaseMediaId = PAGE_ID_PREFIX |                 final String wikibaseMediaId = PAGE_ID_PREFIX | ||||||
|                     + pageId; // Create Wikibase media id from the page id. Example media id: M80618155 for https://commons.wikimedia.org/wiki/File:Tantanmen.jpeg with has the pageid 80618155 |                     + pageId; // Create Wikibase media id from the page id. Example media id: M80618155 for https://commons.wikimedia.org/wiki/File:Tantanmen.jpeg with has the pageid 80618155 | ||||||
|                 compositeDisposable.add(mediaClient.getCaptionByWikibaseIdentifier(wikibaseMediaId) |                 compositeDisposable.add(mediaClient.getCaptionByWikibaseIdentifier(wikibaseMediaId) | ||||||
|                     .subscribeOn(Schedulers.io()) |                     .subscribeOn(Schedulers.io()) | ||||||
|  | @ -141,7 +139,7 @@ public class ContributionViewHolder extends RecyclerView.ViewHolder { | ||||||
|      * @return |      * @return | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     private String chooseImageSource(String thumbUrl, Uri localUri) { |     private String chooseImageSource(final String thumbUrl, final Uri localUri) { | ||||||
|         return !TextUtils.isEmpty(thumbUrl) ? thumbUrl : |         return !TextUtils.isEmpty(thumbUrl) ? thumbUrl : | ||||||
|             localUri != null ? localUri.toString() : |             localUri != null ? localUri.toString() : | ||||||
|                 null; |                 null; | ||||||
|  |  | ||||||
|  | @ -12,16 +12,6 @@ public class ContributionsContract { | ||||||
| 
 | 
 | ||||||
|     public interface View { |     public interface View { | ||||||
| 
 | 
 | ||||||
|         void showWelcomeTip(boolean numberOfUploads); |  | ||||||
| 
 |  | ||||||
|         void showProgress(boolean shouldShow); |  | ||||||
| 
 |  | ||||||
|         void showNoContributionsUI(boolean shouldShow); |  | ||||||
| 
 |  | ||||||
|         void setUploadCount(int count); |  | ||||||
| 
 |  | ||||||
|         void showContributions(List<Contribution> contributionList); |  | ||||||
| 
 |  | ||||||
|         void showMessage(String localizedMessage); |         void showMessage(String localizedMessage); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -31,8 +21,6 @@ public class ContributionsContract { | ||||||
| 
 | 
 | ||||||
|         void deleteUpload(Contribution contribution); |         void deleteUpload(Contribution contribution); | ||||||
| 
 | 
 | ||||||
|         Media getItemAtPosition(int i); |  | ||||||
| 
 |  | ||||||
|         void updateContribution(Contribution contribution); |         void updateContribution(Contribution contribution); | ||||||
| 
 | 
 | ||||||
|         void fetchMediaDetails(Contribution contribution); |         void fetchMediaDetails(Contribution contribution); | ||||||
|  |  | ||||||
|  | @ -19,7 +19,6 @@ import android.widget.Toast; | ||||||
| import androidx.annotation.NonNull; | import androidx.annotation.NonNull; | ||||||
| import androidx.annotation.Nullable; | import androidx.annotation.Nullable; | ||||||
| import androidx.fragment.app.Fragment; | import androidx.fragment.app.Fragment; | ||||||
| import androidx.fragment.app.FragmentManager; |  | ||||||
| import androidx.fragment.app.FragmentManager.OnBackStackChangedListener; | import androidx.fragment.app.FragmentManager.OnBackStackChangedListener; | ||||||
| import androidx.fragment.app.FragmentTransaction; | import androidx.fragment.app.FragmentTransaction; | ||||||
| import butterknife.BindView; | import butterknife.BindView; | ||||||
|  | @ -30,8 +29,7 @@ import fr.free.nrw.commons.campaigns.Campaign; | ||||||
| import fr.free.nrw.commons.campaigns.CampaignView; | import fr.free.nrw.commons.campaigns.CampaignView; | ||||||
| import fr.free.nrw.commons.campaigns.CampaignsPresenter; | import fr.free.nrw.commons.campaigns.CampaignsPresenter; | ||||||
| import fr.free.nrw.commons.campaigns.ICampaignsView; | import fr.free.nrw.commons.campaigns.ICampaignsView; | ||||||
| import fr.free.nrw.commons.contributions.ContributionsListAdapter.Callback; | import fr.free.nrw.commons.contributions.ContributionsListFragment.Callback; | ||||||
| import fr.free.nrw.commons.contributions.ContributionsListFragment.SourceRefresher; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; | import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; | import fr.free.nrw.commons.kvstore.JsonKvStore; | ||||||
| import fr.free.nrw.commons.location.LatLng; | import fr.free.nrw.commons.location.LatLng; | ||||||
|  | @ -54,7 +52,6 @@ import io.reactivex.Observable; | ||||||
| import io.reactivex.android.schedulers.AndroidSchedulers; | import io.reactivex.android.schedulers.AndroidSchedulers; | ||||||
| import io.reactivex.disposables.CompositeDisposable; | import io.reactivex.disposables.CompositeDisposable; | ||||||
| import io.reactivex.schedulers.Schedulers; | import io.reactivex.schedulers.Schedulers; | ||||||
| import java.util.List; |  | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
| import javax.inject.Named; | import javax.inject.Named; | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
|  | @ -62,11 +59,10 @@ import timber.log.Timber; | ||||||
| public class ContributionsFragment | public class ContributionsFragment | ||||||
|         extends CommonsDaggerSupportFragment |         extends CommonsDaggerSupportFragment | ||||||
|         implements |         implements | ||||||
|         MediaDetailProvider, |  | ||||||
|         OnBackStackChangedListener, |         OnBackStackChangedListener, | ||||||
|         SourceRefresher, |  | ||||||
|         LocationUpdateListener, |         LocationUpdateListener, | ||||||
|         ICampaignsView, ContributionsContract.View { |     MediaDetailProvider, | ||||||
|  |     ICampaignsView, ContributionsContract.View, Callback { | ||||||
|     @Inject @Named("default_preferences") JsonKvStore store; |     @Inject @Named("default_preferences") JsonKvStore store; | ||||||
|     @Inject NearbyController nearbyController; |     @Inject NearbyController nearbyController; | ||||||
|     @Inject OkHttpJsonApiClient okHttpJsonApiClient; |     @Inject OkHttpJsonApiClient okHttpJsonApiClient; | ||||||
|  | @ -78,8 +74,8 @@ public class ContributionsFragment | ||||||
|     private CompositeDisposable compositeDisposable = new CompositeDisposable(); |     private CompositeDisposable compositeDisposable = new CompositeDisposable(); | ||||||
| 
 | 
 | ||||||
|     private ContributionsListFragment contributionsListFragment; |     private ContributionsListFragment contributionsListFragment; | ||||||
|     private MediaDetailPagerFragment mediaDetailPagerFragment; |  | ||||||
|     private static final String CONTRIBUTION_LIST_FRAGMENT_TAG = "ContributionListFragmentTag"; |     private static final String CONTRIBUTION_LIST_FRAGMENT_TAG = "ContributionListFragmentTag"; | ||||||
|  |     private MediaDetailPagerFragment mediaDetailPagerFragment; | ||||||
|     static final String MEDIA_DETAIL_PAGER_FRAGMENT_TAG = "MediaDetailFragmentTag"; |     static final String MEDIA_DETAIL_PAGER_FRAGMENT_TAG = "MediaDetailFragmentTag"; | ||||||
| 
 | 
 | ||||||
|     @BindView(R.id.card_view_nearby) public NearbyNotificationCardView nearbyNotificationCardView; |     @BindView(R.id.card_view_nearby) public NearbyNotificationCardView nearbyNotificationCardView; | ||||||
|  | @ -113,7 +109,6 @@ public class ContributionsFragment | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|     private boolean shouldShowMediaDetailsFragment; |     private boolean shouldShowMediaDetailsFragment; | ||||||
|     private int numberOfContributions; |  | ||||||
|     private boolean isAuthCookieAcquired; |     private boolean isAuthCookieAcquired; | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  | @ -128,7 +123,6 @@ public class ContributionsFragment | ||||||
|         ButterKnife.bind(this, view); |         ButterKnife.bind(this, view); | ||||||
|         presenter.onAttachView(this); |         presenter.onAttachView(this); | ||||||
|         contributionsPresenter.onAttachView(this); |         contributionsPresenter.onAttachView(this); | ||||||
|         contributionsPresenter.setLifeCycleOwner(this.getViewLifecycleOwner()); |  | ||||||
|         campaignView.setVisibility(View.GONE); |         campaignView.setVisibility(View.GONE); | ||||||
|         checkBoxView = View.inflate(getActivity(), R.layout.nearby_permission_dialog, null); |         checkBoxView = View.inflate(getActivity(), R.layout.nearby_permission_dialog, null); | ||||||
|         checkBox = (CheckBox) checkBoxView.findViewById(R.id.never_ask_again); |         checkBox = (CheckBox) checkBoxView.findViewById(R.id.never_ask_again); | ||||||
|  | @ -141,103 +135,21 @@ public class ContributionsFragment | ||||||
| 
 | 
 | ||||||
|         if (savedInstanceState != null) { |         if (savedInstanceState != null) { | ||||||
|             mediaDetailPagerFragment = (MediaDetailPagerFragment) getChildFragmentManager() |             mediaDetailPagerFragment = (MediaDetailPagerFragment) getChildFragmentManager() | ||||||
|                     .findFragmentByTag(MEDIA_DETAIL_PAGER_FRAGMENT_TAG); |                 .findFragmentByTag(MEDIA_DETAIL_PAGER_FRAGMENT_TAG); | ||||||
|             contributionsListFragment = (ContributionsListFragment) getChildFragmentManager() |             contributionsListFragment = (ContributionsListFragment) getChildFragmentManager() | ||||||
|                     .findFragmentByTag(CONTRIBUTION_LIST_FRAGMENT_TAG); |                 .findFragmentByTag(CONTRIBUTION_LIST_FRAGMENT_TAG); | ||||||
|             shouldShowMediaDetailsFragment = savedInstanceState.getBoolean("mediaDetailsVisible"); |             shouldShowMediaDetailsFragment = savedInstanceState.getBoolean("mediaDetailsVisible"); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         initFragments(); |         initFragments(); | ||||||
| 
 | 
 | ||||||
|         if(shouldShowMediaDetailsFragment){ |  | ||||||
|             showMediaDetailPagerFragment(); |  | ||||||
|         }else{ |  | ||||||
|             showContributionsListFragment(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (!ConfigUtils.isBetaFlavour()) { |         if (!ConfigUtils.isBetaFlavour()) { | ||||||
|             setUploadCount(); |             setUploadCount(); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         getChildFragmentManager().registerFragmentLifecycleCallbacks( |  | ||||||
|             new FragmentManager.FragmentLifecycleCallbacks() { |  | ||||||
|                 @Override public void onFragmentResumed(FragmentManager fm, Fragment f) { |  | ||||||
|                     super.onFragmentResumed(fm, f); |  | ||||||
|                     //If media detail pager fragment is visible, hide the campaigns view [might not be the best way to do, this but yeah, this proves to work for now] |  | ||||||
|                     Timber.e("onFragmentResumed %s", f.getClass().getName()); |  | ||||||
|                     if (f instanceof MediaDetailPagerFragment) { |  | ||||||
|                         campaignView.setVisibility(View.GONE); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 @Override public void onFragmentDetached(FragmentManager fm, Fragment f) { |  | ||||||
|                     super.onFragmentDetached(fm, f); |  | ||||||
|                     Timber.e("onFragmentDetached %s", f.getClass().getName()); |  | ||||||
|                     //If media detail pager fragment is detached, ContributionsList fragment is gonna be visible, [becomes tightly coupled though] |  | ||||||
|                     if (f instanceof MediaDetailPagerFragment) { |  | ||||||
|                         fetchCampaigns(); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             }, true); |  | ||||||
| 
 |  | ||||||
|         return view; |         return view; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * Initialose the ContributionsListFragment and MediaDetailPagerFragment fragment |  | ||||||
|      */ |  | ||||||
|     private void initFragments() { |  | ||||||
|         if (null == contributionsListFragment) { |  | ||||||
|             contributionsListFragment = new ContributionsListFragment(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         contributionsListFragment.setCallback(new Callback() { |  | ||||||
|             @Override |  | ||||||
|             public void retryUpload(Contribution contribution) { |  | ||||||
|                 ContributionsFragment.this.retryUpload(contribution); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             @Override |  | ||||||
|             public void deleteUpload(Contribution contribution) { |  | ||||||
|                 contributionsPresenter.deleteUpload(contribution); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             @Override |  | ||||||
|             public void openMediaDetail(int position) { |  | ||||||
|                 showDetail(position); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             @Override |  | ||||||
|             public Contribution getContributionForPosition(int position) { |  | ||||||
|                 return (Contribution) contributionsPresenter.getItemAtPosition(position); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             @Override |  | ||||||
|             public void fetchMediaUriFor(Contribution contribution) { |  | ||||||
|                 Timber.d("Fetching thumbnail for %s", contribution.getFilename()); |  | ||||||
|                 contributionsPresenter.fetchMediaDetails(contribution); |  | ||||||
|             } |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         if(null==mediaDetailPagerFragment){ |  | ||||||
|             mediaDetailPagerFragment=new MediaDetailPagerFragment(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Replaces the root frame layout with the given fragment |  | ||||||
|      * @param fragment |  | ||||||
|      * @param tag |  | ||||||
|      */ |  | ||||||
|     private void showFragment(Fragment fragment, String tag) { |  | ||||||
|         FragmentTransaction transaction = getChildFragmentManager().beginTransaction(); |  | ||||||
|         transaction.replace(R.id.root_frame, fragment, tag); |  | ||||||
|         transaction.addToBackStack(CONTRIBUTION_LIST_FRAGMENT_TAG); |  | ||||||
|         transaction.commit(); |  | ||||||
|         getChildFragmentManager().executePendingTransactions(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |     @Override | ||||||
|     public void onAttach(Context context) { |     public void onAttach(Context context) { | ||||||
|         super.onAttach(context); |         super.onAttach(context); | ||||||
|  | @ -265,7 +177,7 @@ public class ContributionsFragment | ||||||
|         if (nearbyNotificationCardView != null) { |         if (nearbyNotificationCardView != null) { | ||||||
|             if (store.getBoolean("displayNearbyCardView", true)) { |             if (store.getBoolean("displayNearbyCardView", true)) { | ||||||
|                 if (nearbyNotificationCardView.cardViewVisibilityState |                 if (nearbyNotificationCardView.cardViewVisibilityState | ||||||
|                         == NearbyNotificationCardView.CardViewVisibilityState.READY) { |                     == NearbyNotificationCardView.CardViewVisibilityState.READY) { | ||||||
|                     nearbyNotificationCardView.setVisibility(View.VISIBLE); |                     nearbyNotificationCardView.setVisibility(View.VISIBLE); | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|  | @ -275,20 +187,22 @@ public class ContributionsFragment | ||||||
|         showFragment(contributionsListFragment, CONTRIBUTION_LIST_FRAGMENT_TAG); |         showFragment(contributionsListFragment, CONTRIBUTION_LIST_FRAGMENT_TAG); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * Replace FrameLayout with MediaDetailPagerFragment, user will see details of selected media. |  | ||||||
|      * Creates new one if null. |  | ||||||
|      */ |  | ||||||
|     private void showMediaDetailPagerFragment() { |     private void showMediaDetailPagerFragment() { | ||||||
|         // hide tabs on media detail view is visible |         // hide tabs on media detail view is visible | ||||||
|         ((MainActivity)getActivity()).hideTabs(); |         ((MainActivity) getActivity()).hideTabs(); | ||||||
|         // hide nearby card view on media detail is visible |         // hide nearby card view on media detail is visible | ||||||
|         nearbyNotificationCardView.setVisibility(View.GONE); |         nearbyNotificationCardView.setVisibility(View.GONE); | ||||||
| 
 | 
 | ||||||
|         showFragment(mediaDetailPagerFragment,MEDIA_DETAIL_PAGER_FRAGMENT_TAG); |         showFragment(mediaDetailPagerFragment, MEDIA_DETAIL_PAGER_FRAGMENT_TAG); | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private void setupViewForMediaDetails() { | ||||||
|  |         campaignView.setVisibility(View.GONE); | ||||||
|  |         nearbyNotificationCardView.setVisibility(View.GONE); | ||||||
|  |         ((MainActivity)getActivity()).hideTabs(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     @Override |     @Override | ||||||
|     public void onBackStackChanged() { |     public void onBackStackChanged() { | ||||||
|         ((MainActivity)getActivity()).initBackButton(); |         ((MainActivity)getActivity()).initBackButton(); | ||||||
|  | @ -307,43 +221,42 @@ public class ContributionsFragment | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private void initFragments() { | ||||||
|  |         if (null == contributionsListFragment) { | ||||||
|  |             contributionsListFragment = new ContributionsListFragment(this); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (shouldShowMediaDetailsFragment) { | ||||||
|  |             showMediaDetailPagerFragment(); | ||||||
|  |         } else { | ||||||
|  |             showContributionsListFragment(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         showFragment(contributionsListFragment, CONTRIBUTION_LIST_FRAGMENT_TAG); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Replaces the root frame layout with the given fragment | ||||||
|  |      * | ||||||
|  |      * @param fragment | ||||||
|  |      * @param tag | ||||||
|  |      */ | ||||||
|  |     private void showFragment(Fragment fragment, String tag) { | ||||||
|  |         FragmentTransaction transaction = getChildFragmentManager().beginTransaction(); | ||||||
|  |         transaction.replace(R.id.root_frame, fragment, tag); | ||||||
|  |         transaction.addToBackStack(CONTRIBUTION_LIST_FRAGMENT_TAG); | ||||||
|  |         transaction.commit(); | ||||||
|  |         getChildFragmentManager().executePendingTransactions(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public Intent getUploadServiceIntent(){ |     public Intent getUploadServiceIntent(){ | ||||||
|         Intent intent = new Intent(getActivity(), UploadService.class); |         Intent intent = new Intent(getActivity(), UploadService.class); | ||||||
|         intent.setAction(UploadService.ACTION_START_SERVICE); |         intent.setAction(UploadService.ACTION_START_SERVICE); | ||||||
|         return intent; |         return intent; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * Replace whatever is in the current contributionsFragmentContainer view with |  | ||||||
|      * mediaDetailPagerFragment, and preserve previous state in back stack. |  | ||||||
|      * Called when user selects a contribution. |  | ||||||
|      */ |  | ||||||
|     private void showDetail(int i) { |  | ||||||
|         if (mediaDetailPagerFragment == null || !mediaDetailPagerFragment.isVisible()) { |  | ||||||
|             mediaDetailPagerFragment = new MediaDetailPagerFragment(); |  | ||||||
|             showMediaDetailPagerFragment(); |  | ||||||
|         } |  | ||||||
|         mediaDetailPagerFragment.showImage(i); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void refreshSource() { |  | ||||||
|         contributionsPresenter.fetchContributions(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Media getMediaAtPosition(int i) { |  | ||||||
|         return contributionsPresenter.getItemAtPosition(i); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int getTotalMediaCount() { |  | ||||||
|         return numberOfContributions; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressWarnings("ConstantConditions") |     @SuppressWarnings("ConstantConditions") | ||||||
|     private void setUploadCount() { |     private void setUploadCount() { | ||||||
| 
 |  | ||||||
|         compositeDisposable.add(okHttpJsonApiClient |         compositeDisposable.add(okHttpJsonApiClient | ||||||
|                 .getUploadCount(((MainActivity)getActivity()).sessionManager.getCurrentAccount().name) |                 .getUploadCount(((MainActivity)getActivity()).sessionManager.getCurrentAccount().name) | ||||||
|                 .subscribeOn(Schedulers.io()) |                 .subscribeOn(Schedulers.io()) | ||||||
|  | @ -373,8 +286,6 @@ public class ContributionsFragment | ||||||
|     @Override |     @Override | ||||||
|     public void onSaveInstanceState(Bundle outState) { |     public void onSaveInstanceState(Bundle outState) { | ||||||
|         super.onSaveInstanceState(outState); |         super.onSaveInstanceState(outState); | ||||||
|         boolean mediaDetailsVisible = mediaDetailPagerFragment != null && mediaDetailPagerFragment.isVisible(); |  | ||||||
|         outState.putBoolean("mediaDetailsVisible", mediaDetailsVisible); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  | @ -384,13 +295,6 @@ public class ContributionsFragment | ||||||
|         firstLocationUpdate = true; |         firstLocationUpdate = true; | ||||||
|         locationManager.addLocationListener(this); |         locationManager.addLocationListener(this); | ||||||
| 
 | 
 | ||||||
|         boolean isSettingsChanged = store.getBoolean(Prefs.IS_CONTRIBUTION_COUNT_CHANGED, false); |  | ||||||
|         store.putBoolean(Prefs.IS_CONTRIBUTION_COUNT_CHANGED, false); |  | ||||||
|         if (isSettingsChanged) { |  | ||||||
|             refreshSource(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|         if (store.getBoolean("displayNearbyCardView", true)) { |         if (store.getBoolean("displayNearbyCardView", true)) { | ||||||
|             checkPermissionsAndShowNearbyCardView(); |             checkPermissionsAndShowNearbyCardView(); | ||||||
|             if (nearbyNotificationCardView.cardViewVisibilityState == NearbyNotificationCardView.CardViewVisibilityState.READY) { |             if (nearbyNotificationCardView.cardViewVisibilityState == NearbyNotificationCardView.CardViewVisibilityState.READY) { | ||||||
|  | @ -403,10 +307,6 @@ public class ContributionsFragment | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         fetchCampaigns(); |         fetchCampaigns(); | ||||||
|         if(isAuthCookieAcquired){ |  | ||||||
|             contributionsPresenter.fetchContributions(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void checkPermissionsAndShowNearbyCardView() { |     private void checkPermissionsAndShowNearbyCardView() { | ||||||
|  | @ -463,17 +363,11 @@ public class ContributionsFragment | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void updateNearbyNotification(@Nullable NearbyController.NearbyPlacesInfo nearbyPlacesInfo) { |     private void updateNearbyNotification(@Nullable NearbyController.NearbyPlacesInfo nearbyPlacesInfo) { | ||||||
| 
 |  | ||||||
|         if (nearbyPlacesInfo != null && nearbyPlacesInfo.placeList != null && nearbyPlacesInfo.placeList.size() > 0) { |         if (nearbyPlacesInfo != null && nearbyPlacesInfo.placeList != null && nearbyPlacesInfo.placeList.size() > 0) { | ||||||
|             Place closestNearbyPlace = nearbyPlacesInfo.placeList.get(0); |             Place closestNearbyPlace = nearbyPlacesInfo.placeList.get(0); | ||||||
|             String distance = formatDistanceBetween(curLatLng, closestNearbyPlace.location); |             String distance = formatDistanceBetween(curLatLng, closestNearbyPlace.location); | ||||||
|             closestNearbyPlace.setDistance(distance); |             closestNearbyPlace.setDistance(distance); | ||||||
|             nearbyNotificationCardView.updateContent(closestNearbyPlace); |             nearbyNotificationCardView.updateContent(closestNearbyPlace); | ||||||
|             if (mediaDetailPagerFragment != null && mediaDetailPagerFragment.isVisible()) { |  | ||||||
|                 nearbyNotificationCardView.setVisibility(View.GONE); |  | ||||||
|             }else { |  | ||||||
|                 nearbyNotificationCardView.setVisibility(View.VISIBLE); |  | ||||||
|             } |  | ||||||
|         } else { |         } else { | ||||||
|             // Means that no close nearby place is found |             // Means that no close nearby place is found | ||||||
|             nearbyNotificationCardView.setVisibility(View.GONE); |             nearbyNotificationCardView.setVisibility(View.GONE); | ||||||
|  | @ -553,37 +447,13 @@ public class ContributionsFragment | ||||||
|         presenter.onDetachView(); |         presenter.onDetachView(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |  | ||||||
|     public void showWelcomeTip(boolean shouldShow) { |  | ||||||
|         contributionsListFragment.showWelcomeTip(shouldShow); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void showProgress(boolean shouldShow) { |  | ||||||
|         contributionsListFragment.showProgress(shouldShow); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void showNoContributionsUI(boolean shouldShow) { |  | ||||||
|         contributionsListFragment.showNoContributionsUI(shouldShow); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void setUploadCount(int count) { |  | ||||||
|         this.numberOfContributions=count; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void showContributions(List<Contribution> contributionList) { |  | ||||||
|         contributionsListFragment.setContributions(contributionList); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |     /** | ||||||
|      * Retry upload when it is failed |      * Retry upload when it is failed | ||||||
|      * |      * | ||||||
|      * @param contribution contribution to be retried |      * @param contribution contribution to be retried | ||||||
|      */ |      */ | ||||||
|     private void retryUpload(Contribution contribution) { |     @Override | ||||||
|  |     public void retryUpload(Contribution contribution) { | ||||||
|         if (NetworkUtils.isInternetConnectionEstablished(getContext())) { |         if (NetworkUtils.isInternetConnectionEstablished(getContext())) { | ||||||
|             if (contribution.getState() == STATE_FAILED && null != uploadService) { |             if (contribution.getState() == STATE_FAILED && null != uploadService) { | ||||||
|                 uploadService.queue(contribution); |                 uploadService.queue(contribution); | ||||||
|  | @ -596,5 +466,29 @@ public class ContributionsFragment | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Replace whatever is in the current contributionsFragmentContainer view with | ||||||
|  |      * mediaDetailPagerFragment, and preserve previous state in back stack. Called when user selects a | ||||||
|  |      * contribution. | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public void showDetail(int position) { | ||||||
|  |         if (mediaDetailPagerFragment == null || !mediaDetailPagerFragment.isVisible()) { | ||||||
|  |             mediaDetailPagerFragment = new MediaDetailPagerFragment(); | ||||||
|  |             showMediaDetailPagerFragment(); | ||||||
|  |         } | ||||||
|  |         mediaDetailPagerFragment.showImage(position); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public Media getMediaAtPosition(int i) { | ||||||
|  |         return contributionsListFragment.getMediaAtPosition(i); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public int getTotalMediaCount() { | ||||||
|  |         return contributionsListFragment.getTotalMediaCount(); | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,68 +1,71 @@ | ||||||
| package fr.free.nrw.commons.contributions; |     package fr.free.nrw.commons.contributions; | ||||||
| 
 | 
 | ||||||
| import android.text.TextUtils; |  | ||||||
| import android.view.LayoutInflater; | import android.view.LayoutInflater; | ||||||
| import android.view.ViewGroup; | import android.view.ViewGroup; | ||||||
| import androidx.annotation.NonNull; | import androidx.annotation.NonNull; | ||||||
| import androidx.recyclerview.widget.RecyclerView; | import androidx.paging.PagedListAdapter; | ||||||
|  | import androidx.recyclerview.widget.DiffUtil; | ||||||
| import fr.free.nrw.commons.R; | import fr.free.nrw.commons.R; | ||||||
| import fr.free.nrw.commons.media.MediaClient; | import fr.free.nrw.commons.media.MediaClient; | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Represents The View Adapter for the List of Contributions |  * Represents The View Adapter for the List of Contributions | ||||||
|  */ |  */ | ||||||
| public class ContributionsListAdapter extends RecyclerView.Adapter<ContributionViewHolder> { | public class ContributionsListAdapter extends | ||||||
|  |     PagedListAdapter<Contribution, ContributionViewHolder> { | ||||||
| 
 | 
 | ||||||
|     private Callback callback; |     private final Callback callback; | ||||||
|     private final MediaClient mediaClient; |     private final MediaClient mediaClient; | ||||||
|     private List<Contribution> contributions; |  | ||||||
| 
 | 
 | ||||||
|     public ContributionsListAdapter(Callback callback, |     ContributionsListAdapter(final Callback callback, | ||||||
|         MediaClient mediaClient) { |         final MediaClient mediaClient) { | ||||||
|  |         super(DIFF_CALLBACK); | ||||||
|         this.callback = callback; |         this.callback = callback; | ||||||
|         this.mediaClient = mediaClient; |         this.mediaClient = mediaClient; | ||||||
|         contributions = new ArrayList<>(); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Creates the new View Holder which will be used to display items(contributions) |      * Uses DiffUtil to calculate the changes in the list | ||||||
|      * using the onBindViewHolder(viewHolder,position)  |      * It has methods that check ID and the content of the items to determine if its a new item | ||||||
|      */ |      */ | ||||||
|     @NonNull |     private static final DiffUtil.ItemCallback<Contribution> DIFF_CALLBACK = | ||||||
|     @Override |         new DiffUtil.ItemCallback<Contribution>() { | ||||||
|     public ContributionViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { |             @Override | ||||||
|         ContributionViewHolder viewHolder = new ContributionViewHolder( |             public boolean areItemsTheSame(final Contribution oldContribution, final Contribution newContribution) { | ||||||
|                 LayoutInflater.from(parent.getContext()) |                 return oldContribution.getPageId().equals(newContribution.getPageId()); | ||||||
|                         .inflate(R.layout.layout_contribution, parent, false), callback, mediaClient); |             } | ||||||
|         return viewHolder; |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     @Override |             @Override | ||||||
|     public void onBindViewHolder(@NonNull ContributionViewHolder holder, int position) { |             public boolean areContentsTheSame(final Contribution oldContribution, final Contribution newContribution) { | ||||||
|         final Contribution contribution = contributions.get(position); |                 return oldContribution.equals(newContribution); | ||||||
|         if (TextUtils.isEmpty(contribution.getThumbUrl()) |             } | ||||||
|             && contribution.getState() == Contribution.STATE_COMPLETED) { |         }; | ||||||
|             callback.fetchMediaUriFor(contribution); |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Initializes the view holder with contribution data | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public void onBindViewHolder(@NonNull final ContributionViewHolder holder, final int position) { | ||||||
|  |         final Contribution contribution = getItem(position); | ||||||
|         holder.init(position, contribution); |         holder.init(position, contribution); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     Contribution getContributionForPosition(final int position) { | ||||||
|     public int getItemCount() { |         return getItem(position); | ||||||
|         return contributions.size(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setContributions(@NonNull List<Contribution> contributionList) { |  | ||||||
|         contributions = contributionList; |  | ||||||
|         notifyDataSetChanged(); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates the new View Holder which will be used to display items(contributions) using the | ||||||
|  |      * onBindViewHolder(viewHolder,position) | ||||||
|  |      */ | ||||||
|  |     @NonNull | ||||||
|     @Override |     @Override | ||||||
|     public long getItemId(int position) { |     public ContributionViewHolder onCreateViewHolder(@NonNull final ViewGroup parent, | ||||||
|         return contributions.get(position).get_id(); |         final int viewType) { | ||||||
|  |         final ContributionViewHolder viewHolder = new ContributionViewHolder( | ||||||
|  |             LayoutInflater.from(parent.getContext()) | ||||||
|  |                 .inflate(R.layout.layout_contribution, parent, false), callback, mediaClient); | ||||||
|  |         return viewHolder; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public interface Callback { |     public interface Callback { | ||||||
|  | @ -72,9 +75,5 @@ public class ContributionsListAdapter extends RecyclerView.Adapter<ContributionV | ||||||
|         void deleteUpload(Contribution contribution); |         void deleteUpload(Contribution contribution); | ||||||
| 
 | 
 | ||||||
|         void openMediaDetail(int contribution); |         void openMediaDetail(int contribution); | ||||||
| 
 |  | ||||||
|         Contribution getContributionForPosition(int position); |  | ||||||
| 
 |  | ||||||
|         void fetchMediaUriFor(Contribution contribution); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,24 @@ | ||||||
|  | package fr.free.nrw.commons.contributions; | ||||||
|  | 
 | ||||||
|  | import fr.free.nrw.commons.BasePresenter; | ||||||
|  | import java.util.List; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The contract for Contributions list View & Presenter | ||||||
|  |  */ | ||||||
|  | public class ContributionsListContract { | ||||||
|  | 
 | ||||||
|  |   public interface View { | ||||||
|  | 
 | ||||||
|  |     void showWelcomeTip(boolean numberOfUploads); | ||||||
|  | 
 | ||||||
|  |     void showProgress(boolean shouldShow); | ||||||
|  | 
 | ||||||
|  |     void showNoContributionsUI(boolean shouldShow); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   public interface UserActionListener extends BasePresenter<View> { | ||||||
|  | 
 | ||||||
|  |     void deleteUpload(Contribution contribution); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | @ -5,6 +5,7 @@ import static android.view.View.VISIBLE; | ||||||
| 
 | 
 | ||||||
| import android.content.res.Configuration; | import android.content.res.Configuration; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
|  | import android.os.Parcelable; | ||||||
| import android.view.LayoutInflater; | import android.view.LayoutInflater; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.ViewGroup; | import android.view.ViewGroup; | ||||||
|  | @ -16,218 +17,217 @@ import android.widget.TextView; | ||||||
| import androidx.annotation.NonNull; | import androidx.annotation.NonNull; | ||||||
| import androidx.annotation.Nullable; | import androidx.annotation.Nullable; | ||||||
| import androidx.recyclerview.widget.GridLayoutManager; | import androidx.recyclerview.widget.GridLayoutManager; | ||||||
| import androidx.recyclerview.widget.LinearLayoutManager; |  | ||||||
| import androidx.recyclerview.widget.RecyclerView; | import androidx.recyclerview.widget.RecyclerView; | ||||||
| import androidx.recyclerview.widget.RecyclerView.LayoutManager; | import androidx.recyclerview.widget.RecyclerView.LayoutManager; | ||||||
| import butterknife.BindView; | import butterknife.BindView; | ||||||
| import butterknife.ButterKnife; | import butterknife.ButterKnife; | ||||||
| import com.google.android.material.floatingactionbutton.FloatingActionButton; | import com.google.android.material.floatingactionbutton.FloatingActionButton; | ||||||
|  | import fr.free.nrw.commons.Media; | ||||||
| import fr.free.nrw.commons.R; | import fr.free.nrw.commons.R; | ||||||
| import fr.free.nrw.commons.contributions.ContributionsListAdapter.Callback; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; | import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; |  | ||||||
| import fr.free.nrw.commons.media.MediaClient; | import fr.free.nrw.commons.media.MediaClient; | ||||||
| import java.util.ArrayList; | import fr.free.nrw.commons.media.MediaDetailPagerFragment; | ||||||
| import java.util.List; |  | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
| import javax.inject.Named; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Created by root on 01.06.2018. |  * Created by root on 01.06.2018. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| public class ContributionsListFragment extends CommonsDaggerSupportFragment { | public class ContributionsListFragment extends CommonsDaggerSupportFragment implements | ||||||
|  |     ContributionsListContract.View, ContributionsListAdapter.Callback { | ||||||
| 
 | 
 | ||||||
|     private static final String VISIBLE_ITEM_ID = "visible_item_id"; |   private static final String RV_STATE = "rv_scroll_state"; | ||||||
|     @BindView(R.id.contributionsList) |  | ||||||
|     RecyclerView rvContributionsList; |  | ||||||
|     @BindView(R.id.loadingContributionsProgressBar) |  | ||||||
|     ProgressBar progressBar; |  | ||||||
|     @BindView(R.id.fab_plus) |  | ||||||
|     FloatingActionButton fabPlus; |  | ||||||
|     @BindView(R.id.fab_camera) |  | ||||||
|     FloatingActionButton fabCamera; |  | ||||||
|     @BindView(R.id.fab_gallery) |  | ||||||
|     FloatingActionButton fabGallery; |  | ||||||
|     @BindView(R.id.noContributionsYet) |  | ||||||
|     TextView noContributionsYet; |  | ||||||
|     @BindView(R.id.fab_layout) |  | ||||||
|     LinearLayout fab_layout; |  | ||||||
| 
 | 
 | ||||||
|     @Inject @Named("default_preferences") JsonKvStore kvStore; |   @BindView(R.id.contributionsList) | ||||||
|     @Inject ContributionController controller; |   RecyclerView rvContributionsList; | ||||||
|     @Inject MediaClient mediaClient; |   @BindView(R.id.loadingContributionsProgressBar) | ||||||
|  |   ProgressBar progressBar; | ||||||
|  |   @BindView(R.id.fab_plus) | ||||||
|  |   FloatingActionButton fabPlus; | ||||||
|  |   @BindView(R.id.fab_camera) | ||||||
|  |   FloatingActionButton fabCamera; | ||||||
|  |   @BindView(R.id.fab_gallery) | ||||||
|  |   FloatingActionButton fabGallery; | ||||||
|  |   @BindView(R.id.noContributionsYet) | ||||||
|  |   TextView noContributionsYet; | ||||||
|  |   @BindView(R.id.fab_layout) | ||||||
|  |   LinearLayout fab_layout; | ||||||
| 
 | 
 | ||||||
|     private Animation fab_close; |   @Inject | ||||||
|     private Animation fab_open; |   ContributionController controller; | ||||||
|     private Animation rotate_forward; |   @Inject | ||||||
|     private Animation rotate_backward; |   MediaClient mediaClient; | ||||||
|  | 
 | ||||||
|  |   @Inject | ||||||
|  |   ContributionsListPresenter contributionsListPresenter; | ||||||
|  | 
 | ||||||
|  |   private Animation fab_close; | ||||||
|  |   private Animation fab_open; | ||||||
|  |   private Animation rotate_forward; | ||||||
|  |   private Animation rotate_backward; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     private boolean isFabOpen = false; |   private boolean isFabOpen; | ||||||
| 
 | 
 | ||||||
|     private ContributionsListAdapter adapter; |   private ContributionsListAdapter adapter; | ||||||
| 
 | 
 | ||||||
|     private Callback callback; |   private final Callback callback; | ||||||
|     private String lastVisibleItemID; |  | ||||||
| 
 | 
 | ||||||
|     private int SPAN_COUNT=3; |   private final int SPAN_COUNT_LANDSCAPE = 3; | ||||||
|     private List<Contribution> contributions=new ArrayList<>(); |   private final int SPAN_COUNT_PORTRAIT = 1; | ||||||
| 
 | 
 | ||||||
|     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { |   ContributionsListFragment(final Callback callback) { | ||||||
|         View view = inflater.inflate(R.layout.fragment_contributions_list, container, false); |     this.callback = callback; | ||||||
|         ButterKnife.bind(this, view); |   } | ||||||
|         initAdapter(); | 
 | ||||||
|         return view; |   public View onCreateView( | ||||||
|  |       final LayoutInflater inflater, @Nullable final ViewGroup container, | ||||||
|  |       @Nullable final Bundle savedInstanceState) { | ||||||
|  |     final View view = inflater.inflate(R.layout.fragment_contributions_list, container, false); | ||||||
|  |     ButterKnife.bind(this, view); | ||||||
|  |     contributionsListPresenter.onAttachView(this); | ||||||
|  |     initAdapter(); | ||||||
|  |     return view; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private void initAdapter() { | ||||||
|  |     adapter = new ContributionsListAdapter(this, mediaClient); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onViewCreated(final View view, @Nullable final Bundle savedInstanceState) { | ||||||
|  |     super.onViewCreated(view, savedInstanceState); | ||||||
|  |     initRecyclerView(); | ||||||
|  |     initializeAnimations(); | ||||||
|  |     setListeners(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private void initRecyclerView() { | ||||||
|  |     final GridLayoutManager layoutManager = new GridLayoutManager(getContext(), | ||||||
|  |         getSpanCount(getResources().getConfiguration().orientation)); | ||||||
|  |     rvContributionsList.setLayoutManager(layoutManager); | ||||||
|  |     contributionsListPresenter.setup(); | ||||||
|  |     contributionsListPresenter.contributionList.observe(this, adapter::submitList); | ||||||
|  |     rvContributionsList.setAdapter(adapter); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private int getSpanCount(final int orientation) { | ||||||
|  |     return orientation == Configuration.ORIENTATION_LANDSCAPE ? | ||||||
|  |         SPAN_COUNT_LANDSCAPE : SPAN_COUNT_PORTRAIT; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onConfigurationChanged(final Configuration newConfig) { | ||||||
|  |     super.onConfigurationChanged(newConfig); | ||||||
|  |     // check orientation | ||||||
|  |     fab_layout.setOrientation(newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE ? | ||||||
|  |         LinearLayout.HORIZONTAL : LinearLayout.VERTICAL); | ||||||
|  |     rvContributionsList | ||||||
|  |         .setLayoutManager(new GridLayoutManager(getContext(), getSpanCount(newConfig.orientation))); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private void initializeAnimations() { | ||||||
|  |     fab_open = AnimationUtils.loadAnimation(getActivity(), R.anim.fab_open); | ||||||
|  |     fab_close = AnimationUtils.loadAnimation(getActivity(), R.anim.fab_close); | ||||||
|  |     rotate_forward = AnimationUtils.loadAnimation(getActivity(), R.anim.rotate_forward); | ||||||
|  |     rotate_backward = AnimationUtils.loadAnimation(getActivity(), R.anim.rotate_backward); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private void setListeners() { | ||||||
|  |     fabPlus.setOnClickListener(view -> animateFAB(isFabOpen)); | ||||||
|  |     fabCamera.setOnClickListener(view -> { | ||||||
|  |       controller.initiateCameraPick(getActivity()); | ||||||
|  |       animateFAB(isFabOpen); | ||||||
|  |     }); | ||||||
|  |     fabGallery.setOnClickListener(view -> { | ||||||
|  |       controller.initiateGalleryPick(getActivity(), true); | ||||||
|  |       animateFAB(isFabOpen); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   private void animateFAB(final boolean isFabOpen) { | ||||||
|  |     this.isFabOpen = !isFabOpen; | ||||||
|  |     if (fabPlus.isShown()) { | ||||||
|  |       if (isFabOpen) { | ||||||
|  |         fabPlus.startAnimation(rotate_backward); | ||||||
|  |         fabCamera.startAnimation(fab_close); | ||||||
|  |         fabGallery.startAnimation(fab_close); | ||||||
|  |         fabCamera.hide(); | ||||||
|  |         fabGallery.hide(); | ||||||
|  |       } else { | ||||||
|  |         fabPlus.startAnimation(rotate_forward); | ||||||
|  |         fabCamera.startAnimation(fab_open); | ||||||
|  |         fabGallery.startAnimation(fab_open); | ||||||
|  |         fabCamera.show(); | ||||||
|  |         fabGallery.show(); | ||||||
|  |       } | ||||||
|  |       this.isFabOpen = !isFabOpen; | ||||||
|     } |     } | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|     public void setCallback(Callback callback) { |   /** | ||||||
|         this.callback = callback; |    * Shows welcome message if user has no contributions yet i.e. new user. | ||||||
|  |    */ | ||||||
|  |   public void showWelcomeTip(final boolean shouldShow) { | ||||||
|  |     noContributionsYet.setVisibility(shouldShow ? VISIBLE : GONE); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Responsible to set progress bar invisible and visible | ||||||
|  |    * | ||||||
|  |    * @param shouldShow True when contributions list should be hidden. | ||||||
|  |    */ | ||||||
|  |   public void showProgress(final boolean shouldShow) { | ||||||
|  |     progressBar.setVisibility(shouldShow ? VISIBLE : GONE); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   public void showNoContributionsUI(final boolean shouldShow) { | ||||||
|  |     noContributionsYet.setVisibility(shouldShow ? VISIBLE : GONE); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onSaveInstanceState(@NonNull Bundle outState) { | ||||||
|  |     super.onSaveInstanceState(outState); | ||||||
|  |     final GridLayoutManager layoutManager = (GridLayoutManager) rvContributionsList.getLayoutManager(); | ||||||
|  |     outState.putParcelable(RV_STATE, layoutManager.onSaveInstanceState()); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onViewStateRestored(@Nullable Bundle savedInstanceState) { | ||||||
|  |     super.onViewStateRestored(savedInstanceState); | ||||||
|  |     if (null != savedInstanceState) { | ||||||
|  |       final Parcelable savedRecyclerLayoutState = savedInstanceState.getParcelable(RV_STATE); | ||||||
|  |       rvContributionsList.getLayoutManager().onRestoreInstanceState(savedRecyclerLayoutState); | ||||||
|     } |     } | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|     private void initAdapter() { |   @Override | ||||||
|         adapter = new ContributionsListAdapter(callback, mediaClient); |   public void retryUpload(final Contribution contribution) { | ||||||
|         adapter.setHasStableIds(true); |     callback.retryUpload(contribution); | ||||||
|     } |   } | ||||||
| 
 | 
 | ||||||
|     @Override |   @Override | ||||||
|     public void onViewCreated(View view, @Nullable Bundle savedInstanceState) { |   public void deleteUpload(final Contribution contribution) { | ||||||
|         super.onViewCreated(view, savedInstanceState); |     contributionsListPresenter.deleteUpload(contribution); | ||||||
|         initRecyclerView(); |   } | ||||||
|         initializeAnimations(); |  | ||||||
|         setListeners(); |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     private void initRecyclerView() { |   @Override | ||||||
|         if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE) { |   public void openMediaDetail(final int position) { | ||||||
|             rvContributionsList.setLayoutManager(new GridLayoutManager(getContext(),SPAN_COUNT)); |     callback.showDetail(position); | ||||||
|         } else { |   } | ||||||
|             rvContributionsList.setLayoutManager(new LinearLayoutManager(getContext())); |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         rvContributionsList.setAdapter(adapter); |   public Media getMediaAtPosition(final int i) { | ||||||
|         adapter.setContributions(contributions); |     return adapter.getContributionForPosition(i); | ||||||
|     } |   } | ||||||
| 
 | 
 | ||||||
|     @Override |   public int getTotalMediaCount() { | ||||||
|     public void onConfigurationChanged(Configuration newConfig) { |     return adapter.getItemCount(); | ||||||
|         super.onConfigurationChanged(newConfig); |   } | ||||||
|         // check orientation |  | ||||||
|         if (newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE) { |  | ||||||
|             fab_layout.setOrientation(LinearLayout.HORIZONTAL); |  | ||||||
|             rvContributionsList.setLayoutManager(new GridLayoutManager(getContext(),SPAN_COUNT)); |  | ||||||
|         } else if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) { |  | ||||||
|             fab_layout.setOrientation(LinearLayout.VERTICAL); |  | ||||||
|             rvContributionsList.setLayoutManager(new LinearLayoutManager(getContext())); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     private void initializeAnimations() { |   public interface Callback { | ||||||
|         fab_open = AnimationUtils.loadAnimation(getActivity(), R.anim.fab_open); |  | ||||||
|         fab_close = AnimationUtils.loadAnimation(getActivity(), R.anim.fab_close); |  | ||||||
|         rotate_forward = AnimationUtils.loadAnimation(getActivity(), R.anim.rotate_forward); |  | ||||||
|         rotate_backward = AnimationUtils.loadAnimation(getActivity(), R.anim.rotate_backward); |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     private void setListeners() { |     void retryUpload(Contribution contribution); | ||||||
|         fabPlus.setOnClickListener(view -> animateFAB(isFabOpen)); |  | ||||||
|         fabCamera.setOnClickListener(view -> { |  | ||||||
|             controller.initiateCameraPick(getActivity()); |  | ||||||
|             animateFAB(isFabOpen); |  | ||||||
|         }); |  | ||||||
|         fabGallery.setOnClickListener(view -> { |  | ||||||
|             controller.initiateGalleryPick(getActivity(), true); |  | ||||||
|             animateFAB(isFabOpen); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void animateFAB(boolean isFabOpen) { |  | ||||||
|         this.isFabOpen = !isFabOpen; |  | ||||||
|         if (fabPlus.isShown()){ |  | ||||||
|             if (isFabOpen) { |  | ||||||
|                 fabPlus.startAnimation(rotate_backward); |  | ||||||
|                 fabCamera.startAnimation(fab_close); |  | ||||||
|                 fabGallery.startAnimation(fab_close); |  | ||||||
|                 fabCamera.hide(); |  | ||||||
|                 fabGallery.hide(); |  | ||||||
|             } else { |  | ||||||
|                 fabPlus.startAnimation(rotate_forward); |  | ||||||
|                 fabCamera.startAnimation(fab_open); |  | ||||||
|                 fabGallery.startAnimation(fab_open); |  | ||||||
|                 fabCamera.show(); |  | ||||||
|                 fabGallery.show(); |  | ||||||
|             } |  | ||||||
|             this.isFabOpen=!isFabOpen; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Shows welcome message if user has no contributions yet i.e. new user. |  | ||||||
|      */ |  | ||||||
|     public void showWelcomeTip(boolean shouldShow) { |  | ||||||
|         noContributionsYet.setVisibility(shouldShow ? VISIBLE : GONE); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Responsible to set progress bar invisible and visible |  | ||||||
|      * |  | ||||||
|      * @param shouldShow True when contributions list should be hidden. |  | ||||||
|      */ |  | ||||||
|     public void showProgress(boolean shouldShow) { |  | ||||||
|         progressBar.setVisibility(shouldShow ? VISIBLE : GONE); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void showNoContributionsUI(boolean shouldShow) { |  | ||||||
|         noContributionsYet.setVisibility(shouldShow ? VISIBLE : GONE); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setContributions(List<Contribution> contributionList) { |  | ||||||
|         this.contributions.clear(); |  | ||||||
|         this.contributions.addAll(contributionList); |  | ||||||
|         adapter.setContributions(contributions); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public interface SourceRefresher { |  | ||||||
|         void refreshSource(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onSaveInstanceState(@NonNull Bundle outState) { |  | ||||||
|         super.onSaveInstanceState(outState); |  | ||||||
|         LayoutManager layoutManager = rvContributionsList.getLayoutManager(); |  | ||||||
|         int lastVisibleItemPosition=0; |  | ||||||
|         if(layoutManager instanceof  LinearLayoutManager){ |  | ||||||
|             lastVisibleItemPosition= ((LinearLayoutManager) layoutManager).findLastCompletelyVisibleItemPosition(); |  | ||||||
|         }else if(layoutManager instanceof GridLayoutManager){ |  | ||||||
|             lastVisibleItemPosition=((GridLayoutManager)layoutManager).findLastCompletelyVisibleItemPosition(); |  | ||||||
|         } |  | ||||||
|         String idOfItemWithPosition = findIdOfItemWithPosition(lastVisibleItemPosition); |  | ||||||
|         if (null != idOfItemWithPosition) { |  | ||||||
|             outState.putString(VISIBLE_ITEM_ID, idOfItemWithPosition); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onViewStateRestored(@Nullable Bundle savedInstanceState) { |  | ||||||
|         super.onViewStateRestored(savedInstanceState); |  | ||||||
|         if(null!=savedInstanceState){ |  | ||||||
|             lastVisibleItemID =savedInstanceState.getString(VISIBLE_ITEM_ID, null); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the id of the contribution from the db |  | ||||||
|      * @param position |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     @Nullable |  | ||||||
|     private String findIdOfItemWithPosition(int position) { |  | ||||||
|         Contribution contributionForPosition = callback.getContributionForPosition(position); |  | ||||||
|         if (null != contributionForPosition) { |  | ||||||
|             return contributionForPosition.getFilename(); |  | ||||||
|         } |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|  |     void showDetail(int position); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,71 @@ | ||||||
|  | package fr.free.nrw.commons.contributions; | ||||||
|  | 
 | ||||||
|  | import androidx.lifecycle.LiveData; | ||||||
|  | import androidx.paging.LivePagedListBuilder; | ||||||
|  | import androidx.paging.PagedList; | ||||||
|  | import fr.free.nrw.commons.contributions.ContributionsListContract.UserActionListener; | ||||||
|  | import fr.free.nrw.commons.di.CommonsApplicationModule; | ||||||
|  | import io.reactivex.Scheduler; | ||||||
|  | import io.reactivex.disposables.CompositeDisposable; | ||||||
|  | import javax.inject.Inject; | ||||||
|  | import javax.inject.Named; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The presenter class for Contributions | ||||||
|  |  */ | ||||||
|  | public class ContributionsListPresenter implements UserActionListener { | ||||||
|  | 
 | ||||||
|  |   private final ContributionBoundaryCallback contributionBoundaryCallback; | ||||||
|  |   private final ContributionsRepository repository; | ||||||
|  |   private final Scheduler ioThreadScheduler; | ||||||
|  | 
 | ||||||
|  |   private final CompositeDisposable compositeDisposable; | ||||||
|  | 
 | ||||||
|  |   LiveData<PagedList<Contribution>> contributionList; | ||||||
|  | 
 | ||||||
|  |   @Inject | ||||||
|  |   ContributionsListPresenter( | ||||||
|  |       final ContributionBoundaryCallback contributionBoundaryCallback, | ||||||
|  |       final ContributionsRepository repository, | ||||||
|  |       @Named(CommonsApplicationModule.IO_THREAD) final Scheduler ioThreadScheduler) { | ||||||
|  |     this.contributionBoundaryCallback = contributionBoundaryCallback; | ||||||
|  |     this.repository = repository; | ||||||
|  |     this.ioThreadScheduler = ioThreadScheduler; | ||||||
|  |     compositeDisposable = new CompositeDisposable(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onAttachView(final ContributionsListContract.View view) { | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Setup the paged list. This method sets the configuration for paged list and ties it up with the | ||||||
|  |    * live data object. This method can be tweaked to update the lazy loading behavior of the | ||||||
|  |    * contributions list | ||||||
|  |    */ | ||||||
|  |   void setup() { | ||||||
|  |     final PagedList.Config pagedListConfig = | ||||||
|  |         (new PagedList.Config.Builder()) | ||||||
|  |             .setPrefetchDistance(50) | ||||||
|  |             .setPageSize(10).build(); | ||||||
|  |     contributionList = (new LivePagedListBuilder(repository.fetchContributions(), pagedListConfig) | ||||||
|  |         .setBoundaryCallback(contributionBoundaryCallback)).build(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onDetachView() { | ||||||
|  |     compositeDisposable.clear(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Delete a failed contribution from the local db | ||||||
|  |    */ | ||||||
|  |   @Override | ||||||
|  |   public void deleteUpload(final Contribution contribution) { | ||||||
|  |     compositeDisposable.add(repository | ||||||
|  |         .deleteContributionFromDB(contribution) | ||||||
|  |         .subscribeOn(ioThreadScheduler) | ||||||
|  |         .subscribe()); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -1,14 +1,13 @@ | ||||||
| package fr.free.nrw.commons.contributions; | package fr.free.nrw.commons.contributions; | ||||||
| 
 | 
 | ||||||
| import androidx.lifecycle.LiveData; | import androidx.paging.DataSource.Factory; | ||||||
| 
 | import io.reactivex.Completable; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| 
 | 
 | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
| import javax.inject.Named; | import javax.inject.Named; | ||||||
| 
 | 
 | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; | import fr.free.nrw.commons.kvstore.JsonKvStore; | ||||||
| import io.reactivex.Completable; |  | ||||||
| import io.reactivex.Single; | import io.reactivex.Single; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  | @ -59,23 +58,23 @@ class ContributionsLocalDataSource { | ||||||
|      * @param contribution |      * @param contribution | ||||||
|      * @return |      * @return | ||||||
|      */ |      */ | ||||||
|     public Single<Integer> deleteContribution(Contribution contribution) { |     public Completable deleteContribution(Contribution contribution) { | ||||||
|         return contributionDao.delete(contribution); |         return contributionDao.delete(contribution); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public LiveData<List<Contribution>> getContributions() { |     public Factory<Integer, Contribution> getContributions() { | ||||||
|         return contributionDao.fetchContributions(); |         return contributionDao.fetchContributions(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public Completable saveContributions(List<Contribution> contributions) { |     public Single<List<Long>> saveContributions(List<Contribution> contributions) { | ||||||
|         return contributionDao.deleteAllAndSave(contributions); |         return contributionDao.save(contributions); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void set(String key, long value) { |     public void set(String key, long value) { | ||||||
|         defaultKVStore.putLong(key,value); |         defaultKVStore.putLong(key,value); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public Single<Integer> updateContribution(Contribution contribution) { |     public Completable updateContribution(Contribution contribution) { | ||||||
|         return contributionDao.update(contribution); |         return contributionDao.update(contribution); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -9,8 +9,8 @@ import fr.free.nrw.commons.CommonsApplication; | ||||||
| import fr.free.nrw.commons.Media; | import fr.free.nrw.commons.Media; | ||||||
| import fr.free.nrw.commons.MediaDataExtractor; | import fr.free.nrw.commons.MediaDataExtractor; | ||||||
| import fr.free.nrw.commons.auth.SessionManager; | import fr.free.nrw.commons.auth.SessionManager; | ||||||
|  | import fr.free.nrw.commons.MediaDataExtractor; | ||||||
| import fr.free.nrw.commons.contributions.ContributionsContract.UserActionListener; | import fr.free.nrw.commons.contributions.ContributionsContract.UserActionListener; | ||||||
| import fr.free.nrw.commons.db.AppDatabase; |  | ||||||
| import fr.free.nrw.commons.di.CommonsApplicationModule; | import fr.free.nrw.commons.di.CommonsApplicationModule; | ||||||
| import fr.free.nrw.commons.mwapi.UserClient; | import fr.free.nrw.commons.mwapi.UserClient; | ||||||
| import fr.free.nrw.commons.utils.NetworkUtils; | import fr.free.nrw.commons.utils.NetworkUtils; | ||||||
|  | @ -25,6 +25,9 @@ import javax.inject.Inject; | ||||||
| import javax.inject.Named; | import javax.inject.Named; | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
| 
 | 
 | ||||||
|  | import javax.inject.Inject; | ||||||
|  | import javax.inject.Named; | ||||||
|  | 
 | ||||||
| /** | /** | ||||||
|  * The presenter class for Contributions |  * The presenter class for Contributions | ||||||
|  */ |  */ | ||||||
|  | @ -35,25 +38,10 @@ public class ContributionsPresenter implements UserActionListener { | ||||||
|     private final Scheduler ioThreadScheduler; |     private final Scheduler ioThreadScheduler; | ||||||
|     private CompositeDisposable compositeDisposable; |     private CompositeDisposable compositeDisposable; | ||||||
|     private ContributionsContract.View view; |     private ContributionsContract.View view; | ||||||
|     private List<Contribution> contributionList=new ArrayList<>(); |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     Context context; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     UserClient userClient; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     AppDatabase appDatabase; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     SessionManager sessionManager; |  | ||||||
| 
 | 
 | ||||||
|     @Inject |     @Inject | ||||||
|     MediaDataExtractor mediaDataExtractor; |     MediaDataExtractor mediaDataExtractor; | ||||||
| 
 | 
 | ||||||
|     private LifecycleOwner lifeCycleOwner; |  | ||||||
| 
 |  | ||||||
|     @Inject |     @Inject | ||||||
|     ContributionsPresenter(ContributionsRepository repository, @Named(CommonsApplicationModule.MAIN_THREAD) Scheduler mainThreadScheduler,@Named(CommonsApplicationModule.IO_THREAD) Scheduler ioThreadScheduler) { |     ContributionsPresenter(ContributionsRepository repository, @Named(CommonsApplicationModule.MAIN_THREAD) Scheduler mainThreadScheduler,@Named(CommonsApplicationModule.IO_THREAD) Scheduler ioThreadScheduler) { | ||||||
|         this.repository = repository; |         this.repository = repository; | ||||||
|  | @ -61,74 +49,12 @@ public class ContributionsPresenter implements UserActionListener { | ||||||
|         this.ioThreadScheduler=ioThreadScheduler; |         this.ioThreadScheduler=ioThreadScheduler; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private String user; |  | ||||||
| 
 |  | ||||||
|     @Override |     @Override | ||||||
|     public void onAttachView(ContributionsContract.View view) { |     public void onAttachView(ContributionsContract.View view) { | ||||||
|         this.view = view; |         this.view = view; | ||||||
|         compositeDisposable=new CompositeDisposable(); |         compositeDisposable=new CompositeDisposable(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void setLifeCycleOwner(LifecycleOwner lifeCycleOwner){ |  | ||||||
|         this.lifeCycleOwner=lifeCycleOwner; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void fetchContributions() { |  | ||||||
|         Timber.d("fetch  Contributions"); |  | ||||||
|         LiveData<List<Contribution>> liveDataContributions = repository.fetchContributions(); |  | ||||||
|         if(null!=lifeCycleOwner) { |  | ||||||
|              liveDataContributions.observe(lifeCycleOwner, this::showContributions); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (NetworkUtils.isInternetConnectionEstablished(CommonsApplication.getInstance()) && shouldFetchContributions()) { |  | ||||||
|             Timber.d("fetching contributions: "); |  | ||||||
|             view.showProgress(true); |  | ||||||
|             this.user = sessionManager.getUserName(); |  | ||||||
|             view.showContributions(Collections.emptyList()); |  | ||||||
|             compositeDisposable.add(userClient.logEvents(user) |  | ||||||
|                     .subscribeOn(ioThreadScheduler) |  | ||||||
|                     .observeOn(mainThreadScheduler) |  | ||||||
|                     .doOnNext(mwQueryLogEvent -> Timber.d("Received image %s", mwQueryLogEvent.title())) |  | ||||||
|                     .filter(mwQueryLogEvent -> !mwQueryLogEvent.isDeleted()).doOnNext(mwQueryLogEvent -> Timber.d("Image %s passed filters", mwQueryLogEvent.title())) |  | ||||||
|                     .map(image -> new Contribution(image, user)) |  | ||||||
|                     .toList() |  | ||||||
|                     .subscribe(this::saveContributionsToDB, error -> { |  | ||||||
|                         Timber.e("Failed to fetch contributions: %s", error.getMessage()); |  | ||||||
|                     })); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void showContributions(@NonNull List<Contribution> contributions) { |  | ||||||
|         view.showProgress(false); |  | ||||||
|         if (contributions.isEmpty()) { |  | ||||||
|             view.showWelcomeTip(true); |  | ||||||
|             view.showNoContributionsUI(true); |  | ||||||
|         } else { |  | ||||||
|             view.showWelcomeTip(false); |  | ||||||
|             view.showNoContributionsUI(false); |  | ||||||
|             view.setUploadCount(contributions.size()); |  | ||||||
|             view.showContributions(contributions); |  | ||||||
|             this.contributionList.clear(); |  | ||||||
|             this.contributionList.addAll(contributions); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void saveContributionsToDB(List<Contribution> contributions) { |  | ||||||
|         Timber.e("Fetched: "+contributions.size()+" contributions "+" saving to db"); |  | ||||||
|         repository.save(contributions).subscribeOn(ioThreadScheduler).subscribe(); |  | ||||||
|         repository.set("last_fetch_timestamp",System.currentTimeMillis()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private boolean shouldFetchContributions() { |  | ||||||
|         long lastFetchTimestamp = repository.getLong("last_fetch_timestamp"); |  | ||||||
|         Timber.d("last fetch timestamp: %s", lastFetchTimestamp); |  | ||||||
|         if(lastFetchTimestamp!=0){ |  | ||||||
|             return System.currentTimeMillis()-lastFetchTimestamp>15*60*100; |  | ||||||
|         } |  | ||||||
|         Timber.d("should  fetch contributions: %s", true); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |     @Override | ||||||
|     public void onDetachView() { |     public void onDetachView() { | ||||||
|         this.view = null; |         this.view = null; | ||||||
|  | @ -146,24 +72,10 @@ public class ContributionsPresenter implements UserActionListener { | ||||||
|      */ |      */ | ||||||
|     @Override |     @Override | ||||||
|     public void deleteUpload(Contribution contribution) { |     public void deleteUpload(Contribution contribution) { | ||||||
|         compositeDisposable.add(repository.deleteContributionFromDB(contribution) |         compositeDisposable.add(repository | ||||||
|         .subscribeOn(ioThreadScheduler) |             .deleteContributionFromDB(contribution) | ||||||
|         .subscribe()); |             .subscribeOn(ioThreadScheduler) | ||||||
|     } |             .subscribe()); | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Returns a contribution at the specified cursor position |  | ||||||
|      * |  | ||||||
|      * @param i |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public Media getItemAtPosition(int i) { |  | ||||||
|         if (i == -1 || contributionList.size() < i+1) { |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|         return contributionList.get(i); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  |  | ||||||
|  | @ -1,12 +1,11 @@ | ||||||
| package fr.free.nrw.commons.contributions; | package fr.free.nrw.commons.contributions; | ||||||
| 
 | 
 | ||||||
| import androidx.lifecycle.LiveData; | import androidx.paging.DataSource.Factory; | ||||||
| 
 | import io.reactivex.Completable; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| 
 | 
 | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
| 
 | 
 | ||||||
| import io.reactivex.Completable; |  | ||||||
| import io.reactivex.Single; | import io.reactivex.Single; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  | @ -33,7 +32,7 @@ public class ContributionsRepository { | ||||||
|      * @param contribution |      * @param contribution | ||||||
|      * @return |      * @return | ||||||
|      */ |      */ | ||||||
|     public Single<Integer> deleteContributionFromDB(Contribution contribution) { |     public Completable deleteContributionFromDB(Contribution contribution) { | ||||||
|         return localDataSource.deleteContribution(contribution); |         return localDataSource.deleteContribution(contribution); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -46,11 +45,11 @@ public class ContributionsRepository { | ||||||
|         return localDataSource.getContributionWithFileName(fileName); |         return localDataSource.getContributionWithFileName(fileName); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public LiveData<List<Contribution>> fetchContributions() { |     public Factory<Integer, Contribution> fetchContributions() { | ||||||
|         return localDataSource.getContributions(); |         return localDataSource.getContributions(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public Completable save(List<Contribution> contributions) { |     public Single<List<Long>> save(List<Contribution> contributions) { | ||||||
|         return localDataSource.saveContributions(contributions); |         return localDataSource.saveContributions(contributions); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -58,11 +57,7 @@ public class ContributionsRepository { | ||||||
|         localDataSource.set(key,value); |         localDataSource.set(key,value); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public long getLong(String key) { |     public Completable updateContribution(Contribution contribution) { | ||||||
|         return localDataSource.getLong(key); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public Single<Integer> updateContribution(Contribution contribution) { |  | ||||||
|         return localDataSource.updateContribution(contribution); |         return localDataSource.updateContribution(contribution); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,7 +2,6 @@ package fr.free.nrw.commons.contributions; | ||||||
| 
 | 
 | ||||||
| import android.annotation.SuppressLint; | import android.annotation.SuppressLint; | ||||||
| import android.app.AlertDialog; | import android.app.AlertDialog; | ||||||
| import android.content.ContentResolver; |  | ||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.view.LayoutInflater; | import android.view.LayoutInflater; | ||||||
|  | @ -12,7 +11,6 @@ import android.view.MenuItem; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.widget.ImageView; | import android.widget.ImageView; | ||||||
| import android.widget.TextView; | import android.widget.TextView; | ||||||
| 
 |  | ||||||
| import androidx.annotation.Nullable; | import androidx.annotation.Nullable; | ||||||
| import androidx.core.view.GravityCompat; | import androidx.core.view.GravityCompat; | ||||||
| import androidx.drawerlayout.widget.DrawerLayout; | import androidx.drawerlayout.widget.DrawerLayout; | ||||||
|  | @ -20,16 +18,9 @@ import androidx.fragment.app.Fragment; | ||||||
| import androidx.fragment.app.FragmentManager; | import androidx.fragment.app.FragmentManager; | ||||||
| import androidx.fragment.app.FragmentPagerAdapter; | import androidx.fragment.app.FragmentPagerAdapter; | ||||||
| import androidx.viewpager.widget.ViewPager; | import androidx.viewpager.widget.ViewPager; | ||||||
| 
 |  | ||||||
| import com.google.android.material.tabs.TabLayout; |  | ||||||
| 
 |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| import javax.inject.Inject; |  | ||||||
| 
 |  | ||||||
| import butterknife.BindView; | import butterknife.BindView; | ||||||
| import butterknife.ButterKnife; | import butterknife.ButterKnife; | ||||||
| import fr.free.nrw.commons.BuildConfig; | import com.google.android.material.tabs.TabLayout; | ||||||
| import fr.free.nrw.commons.R; | import fr.free.nrw.commons.R; | ||||||
| import fr.free.nrw.commons.auth.SessionManager; | import fr.free.nrw.commons.auth.SessionManager; | ||||||
| import fr.free.nrw.commons.location.LocationServiceManager; | import fr.free.nrw.commons.location.LocationServiceManager; | ||||||
|  | @ -44,10 +35,10 @@ import fr.free.nrw.commons.upload.UploadService; | ||||||
| import fr.free.nrw.commons.utils.ViewUtil; | import fr.free.nrw.commons.utils.ViewUtil; | ||||||
| import io.reactivex.android.schedulers.AndroidSchedulers; | import io.reactivex.android.schedulers.AndroidSchedulers; | ||||||
| import io.reactivex.schedulers.Schedulers; | import io.reactivex.schedulers.Schedulers; | ||||||
|  | import java.util.List; | ||||||
|  | import javax.inject.Inject; | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
| 
 | 
 | ||||||
| import static android.content.ContentResolver.requestSync; |  | ||||||
| 
 |  | ||||||
| public class MainActivity extends NavigationBaseActivity implements FragmentManager.OnBackStackChangedListener { | public class MainActivity extends NavigationBaseActivity implements FragmentManager.OnBackStackChangedListener { | ||||||
| 
 | 
 | ||||||
|     @BindView(R.id.tab_layout) |     @BindView(R.id.tab_layout) | ||||||
|  |  | ||||||
|  | @ -10,7 +10,7 @@ import fr.free.nrw.commons.contributions.ContributionDao | ||||||
|  * The database for accessing the respective DAOs |  * The database for accessing the respective DAOs | ||||||
|  * |  * | ||||||
|  */ |  */ | ||||||
| @Database(entities = [Contribution::class], version = 1, exportSchema = false) | @Database(entities = [Contribution::class], version = 2, exportSchema = false) | ||||||
| @TypeConverters(Converters::class) | @TypeConverters(Converters::class) | ||||||
| abstract class AppDatabase : RoomDatabase() { | abstract class AppDatabase : RoomDatabase() { | ||||||
|   abstract fun contributionDao(): ContributionDao |   abstract fun contributionDao(): ContributionDao | ||||||
|  |  | ||||||
|  | @ -224,7 +224,9 @@ public class CommonsApplicationModule { | ||||||
|     @Provides |     @Provides | ||||||
|     @Singleton |     @Singleton | ||||||
|     public AppDatabase provideAppDataBase() { |     public AppDatabase provideAppDataBase() { | ||||||
|         return Room.databaseBuilder(applicationContext, AppDatabase.class, "commons_room.db").build(); |         return Room.databaseBuilder(applicationContext, AppDatabase.class, "commons_room.db") | ||||||
|  |             .fallbackToDestructiveMigration() | ||||||
|  |             .build(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Provides |     @Provides | ||||||
|  |  | ||||||
|  | @ -32,6 +32,7 @@ public class MediaClient { | ||||||
| 
 | 
 | ||||||
|     //OkHttpJsonApiClient used JsonKvStore for this. I don't know why. |     //OkHttpJsonApiClient used JsonKvStore for this. I don't know why. | ||||||
|     private Map<String, Map<String, String>> continuationStore; |     private Map<String, Map<String, String>> continuationStore; | ||||||
|  |     private Map<String, Boolean> continuationExists; | ||||||
|     public static final String NO_CAPTION = "No caption"; |     public static final String NO_CAPTION = "No caption"; | ||||||
|     private static final String NO_DEPICTION = "No depiction"; |     private static final String NO_DEPICTION = "No depiction"; | ||||||
| 
 | 
 | ||||||
|  | @ -40,6 +41,7 @@ public class MediaClient { | ||||||
|         this.mediaInterface = mediaInterface; |         this.mediaInterface = mediaInterface; | ||||||
|         this.mediaDetailInterface = mediaDetailInterface; |         this.mediaDetailInterface = mediaDetailInterface; | ||||||
|         this.continuationStore = new HashMap<>(); |         this.continuationStore = new HashMap<>(); | ||||||
|  |         this.continuationExists = new HashMap<>(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -83,6 +85,36 @@ public class MediaClient { | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * This method takes the userName as input and returns a list of  Media objects filtered using | ||||||
|  |      * allimages query It uses the allimages query API to get the images contributed by the userName, | ||||||
|  |      * 10 at a time. | ||||||
|  |      * | ||||||
|  |      * @param userName the username | ||||||
|  |      * @return | ||||||
|  |      */ | ||||||
|  |     public Single<List<Media>> getMediaListForUser(String userName) { | ||||||
|  |         Map<String, String> continuation = | ||||||
|  |             continuationStore.containsKey("user_" + userName) | ||||||
|  |                 ? continuationStore.get("user_" + userName) | ||||||
|  |                 : Collections.emptyMap(); | ||||||
|  |         return responseToMediaList(mediaInterface | ||||||
|  |             .getMediaListForUser(userName, 10, continuation), "user_" + userName); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Check if media for user has reached the end of the list. | ||||||
|  |      * @param userName | ||||||
|  |      * @return | ||||||
|  |      */ | ||||||
|  |     public boolean doesMediaListForUserHaveMorePages(String userName) { | ||||||
|  |         final String key = "user_" + userName; | ||||||
|  |         if(continuationExists.containsKey(key)) { | ||||||
|  |             return continuationExists.get(key); | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * This method takes a keyword as input and returns a list of  Media objects filtered using image generator query |      * This method takes a keyword as input and returns a list of  Media objects filtered using image generator query | ||||||
|      * It uses the generator query API to get the images searched using a query, 10 at a time. |      * It uses the generator query API to get the images searched using a query, 10 at a time. | ||||||
|  | @ -106,7 +138,12 @@ public class MediaClient { | ||||||
|                     || null == mwQueryResponse.query().pages()) { |                     || null == mwQueryResponse.query().pages()) { | ||||||
|                 return Observable.empty(); |                 return Observable.empty(); | ||||||
|             } |             } | ||||||
|             continuationStore.put(key, mwQueryResponse.continuation()); |             if(mwQueryResponse.continuation() != null) { | ||||||
|  |                 continuationStore.put(key, mwQueryResponse.continuation()); | ||||||
|  |                 continuationExists.put(key, true); | ||||||
|  |             } else { | ||||||
|  |                 continuationExists.put(key, false); | ||||||
|  |             } | ||||||
|             return Observable.fromIterable(mwQueryResponse.query().pages()); |             return Observable.fromIterable(mwQueryResponse.query().pages()); | ||||||
|         }) |         }) | ||||||
|                 .map(Media::from) |                 .map(Media::from) | ||||||
|  |  | ||||||
|  | @ -23,26 +23,21 @@ import butterknife.ButterKnife; | ||||||
| import com.google.android.material.snackbar.Snackbar; | import com.google.android.material.snackbar.Snackbar; | ||||||
| import fr.free.nrw.commons.Media; | import fr.free.nrw.commons.Media; | ||||||
| import fr.free.nrw.commons.R; | import fr.free.nrw.commons.R; | ||||||
| import fr.free.nrw.commons.auth.SessionManager; |  | ||||||
| import fr.free.nrw.commons.bookmarks.Bookmark; | import fr.free.nrw.commons.bookmarks.Bookmark; | ||||||
| import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesContentProvider; | import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesContentProvider; | ||||||
| import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesDao; | import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesDao; | ||||||
| import fr.free.nrw.commons.category.CategoryImagesCallback; | import fr.free.nrw.commons.category.CategoryImagesCallback; | ||||||
| import fr.free.nrw.commons.contributions.Contribution; | import fr.free.nrw.commons.contributions.Contribution; | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; | import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; |  | ||||||
| import fr.free.nrw.commons.utils.DownloadUtils; | import fr.free.nrw.commons.utils.DownloadUtils; | ||||||
| import fr.free.nrw.commons.utils.ImageUtils; | import fr.free.nrw.commons.utils.ImageUtils; | ||||||
| import fr.free.nrw.commons.utils.NetworkUtils; | import fr.free.nrw.commons.utils.NetworkUtils; | ||||||
| import fr.free.nrw.commons.utils.ViewUtil; | import fr.free.nrw.commons.utils.ViewUtil; | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
| import javax.inject.Named; |  | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
| 
 | 
 | ||||||
| public class MediaDetailPagerFragment extends CommonsDaggerSupportFragment implements ViewPager.OnPageChangeListener { | public class MediaDetailPagerFragment extends CommonsDaggerSupportFragment implements ViewPager.OnPageChangeListener { | ||||||
| 
 | 
 | ||||||
|     @Inject SessionManager sessionManager; |  | ||||||
|     @Inject @Named("default_preferences") JsonKvStore store; |  | ||||||
|     @Inject BookmarkPicturesDao bookmarkDao; |     @Inject BookmarkPicturesDao bookmarkDao; | ||||||
| 
 | 
 | ||||||
|     @BindView(R.id.mediaDetailsPager) ViewPager pager; |     @BindView(R.id.mediaDetailsPager) ViewPager pager; | ||||||
|  |  | ||||||
|  | @ -1,11 +1,9 @@ | ||||||
| package fr.free.nrw.commons.media; | package fr.free.nrw.commons.media; | ||||||
| 
 | 
 | ||||||
| import org.wikipedia.dataclient.mwapi.MwQueryResponse; |  | ||||||
| 
 |  | ||||||
| import java.util.Map; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.depictions.models.DepictionResponse; | import fr.free.nrw.commons.depictions.models.DepictionResponse; | ||||||
| import io.reactivex.Observable; | import io.reactivex.Observable; | ||||||
|  | import java.util.Map; | ||||||
|  | import org.wikipedia.dataclient.mwapi.MwQueryResponse; | ||||||
| import retrofit2.http.GET; | import retrofit2.http.GET; | ||||||
| import retrofit2.http.Query; | import retrofit2.http.Query; | ||||||
| import retrofit2.http.QueryMap; | import retrofit2.http.QueryMap; | ||||||
|  | @ -17,6 +15,7 @@ public interface MediaInterface { | ||||||
|     String MEDIA_PARAMS="&prop=imageinfo&iiprop=url|extmetadata&iiurlwidth=640" + |     String MEDIA_PARAMS="&prop=imageinfo&iiprop=url|extmetadata&iiurlwidth=640" + | ||||||
|             "&iiextmetadatafilter=DateTime|Categories|GPSLatitude|GPSLongitude|ImageDescription|DateTimeOriginal" + |             "&iiextmetadatafilter=DateTime|Categories|GPSLatitude|GPSLongitude|ImageDescription|DateTimeOriginal" + | ||||||
|             "|Artist|LicenseShortName|LicenseUrl"; |             "|Artist|LicenseShortName|LicenseUrl"; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Checks if a page exists or not. |      * Checks if a page exists or not. | ||||||
|      * |      * | ||||||
|  | @ -48,6 +47,19 @@ public interface MediaInterface { | ||||||
|             MEDIA_PARAMS) |             MEDIA_PARAMS) | ||||||
|     Observable<MwQueryResponse> getMediaListFromCategory(@Query("gcmtitle") String category, @Query("gcmlimit") int itemLimit, @QueryMap Map<String, String> continuation); |     Observable<MwQueryResponse> getMediaListFromCategory(@Query("gcmtitle") String category, @Query("gcmlimit") int itemLimit, @QueryMap Map<String, String> continuation); | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * This method retrieves a list of Media objects for a given user name | ||||||
|  |      * | ||||||
|  |      * @param username     user's Wikimedia Commons username. | ||||||
|  |      * @param itemLimit    how many images are returned | ||||||
|  |      * @param continuation the continuation string from the previous query or empty map | ||||||
|  |      * @return | ||||||
|  |      */ | ||||||
|  |     @GET("w/api.php?action=query&format=json&formatversion=2" + //Basic parameters | ||||||
|  |         "&generator=allimages&gaisort=timestamp&gaidir=older" + MEDIA_PARAMS) | ||||||
|  |     Observable<MwQueryResponse> getMediaListForUser(@Query("gaiuser") String username, | ||||||
|  |         @Query("gailimit") int itemLimit, @QueryMap(encoded = true) Map<String, String> continuation); | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * This method retrieves a list of Media objects filtered using image generator query |      * This method retrieves a list of Media objects filtered using image generator query | ||||||
|      * |      * | ||||||
|  | @ -86,21 +98,15 @@ public interface MediaInterface { | ||||||
|     Observable<MwParseResponse> getPageHtml(@Query("page") String title); |     Observable<MwParseResponse> getPageHtml(@Query("page") String title); | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Fetches caption using file name |    * Fetches list of images from a depiction entity | ||||||
|      * |    * | ||||||
|      * @param filename name of the file to be used for fetching captions |    * @param query    depictionEntityId | ||||||
|      * */ |    * @param sroffset number od depictions already fetched, this is useful in implementing | ||||||
|     @GET("w/api.php?action=wbgetentities&props=labels&format=json&languagefallback=1") |    *                 pagination | ||||||
|     Observable<MwQueryResponse> fetchCaptionByFilename(@Query("language") String language, @Query("titles") String filename); |    */ | ||||||
| 
 | 
 | ||||||
|     /** |   @GET("w/api.php?action=query&list=search&format=json&srnamespace=6") | ||||||
|      * Fetches list of images from a depiction entity |   Observable<DepictionResponse> fetchImagesForDepictedItem(@Query("srsearch") String query, | ||||||
|      * |       @Query("sroffset") String sroffset); | ||||||
|      * @param query depictionEntityId |  | ||||||
|      * @param sroffset number od depictions already fetched, this is useful in implementing pagination |  | ||||||
|      */ |  | ||||||
| 
 |  | ||||||
|     @GET("w/api.php?action=query&list=search&format=json&srnamespace=6") |  | ||||||
|     Observable<DepictionResponse> fetchImagesForDepictedItem(@Query("srsearch") String query, @Query("sroffset") String sroffset); |  | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -6,7 +6,6 @@ public class Prefs { | ||||||
|     public static String TRACKING_ENABLED = "eventLogging"; |     public static String TRACKING_ENABLED = "eventLogging"; | ||||||
|     public static final String DEFAULT_LICENSE = "defaultLicense"; |     public static final String DEFAULT_LICENSE = "defaultLicense"; | ||||||
|     public static final String UPLOADS_SHOWING = "uploadsshowing"; |     public static final String UPLOADS_SHOWING = "uploadsshowing"; | ||||||
|     public static final String IS_CONTRIBUTION_COUNT_CHANGED = "ccontributionCountChanged"; |  | ||||||
|     public static final String MANAGED_EXIF_TAGS = "managed_exif_tags"; |     public static final String MANAGED_EXIF_TAGS = "managed_exif_tags"; | ||||||
|     public static final String KEY_LANGUAGE_VALUE = "languageDescription"; |     public static final String KEY_LANGUAGE_VALUE = "languageDescription"; | ||||||
|     public static final String KEY_THEME_VALUE = "appThemePref"; |     public static final String KEY_THEME_VALUE = "appThemePref"; | ||||||
|  |  | ||||||
|  | @ -65,43 +65,6 @@ public class SettingsFragment extends PreferenceFragmentCompat { | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         final EditTextPreference uploadLimit = findPreference("uploads"); |  | ||||||
|         int currentUploadLimit = defaultKvStore.getInt(Prefs.UPLOADS_SHOWING, 100); |  | ||||||
|         uploadLimit.setText(String.valueOf(currentUploadLimit)); |  | ||||||
| 
 |  | ||||||
|         uploadLimit.setOnPreferenceChangeListener((preference, newValue) -> { |  | ||||||
| 
 |  | ||||||
|             if (newValue.toString().length() == 0) { |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             int value = Integer.parseInt(newValue.toString()); |  | ||||||
|             if (value > 500) { |  | ||||||
|                 Snackbar error = Snackbar.make(getView(), R.string.maximum_limit_alert, Snackbar.LENGTH_LONG); |  | ||||||
|                 error.show(); |  | ||||||
|                 return false; |  | ||||||
|             } else if (value == 0) { |  | ||||||
|                 Snackbar error = Snackbar.make(getView(), R.string.cannot_be_zero, Snackbar.LENGTH_LONG); |  | ||||||
|                 error.show(); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|             return true; |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         uploadLimit.setOnBindEditTextListener(editText -> { |  | ||||||
| 
 |  | ||||||
|             editText.setInputType(InputType.TYPE_CLASS_NUMBER); |  | ||||||
|             editText.selectAll(); |  | ||||||
|             int maxLength = 3; // set maxLength to 3 |  | ||||||
|             editText.setFilters(new InputFilter[]{new InputFilter.LengthFilter(maxLength)}); |  | ||||||
| 
 |  | ||||||
|             int value = Integer.parseInt(editText.getText().toString()); |  | ||||||
| 
 |  | ||||||
|             defaultKvStore.putInt(Prefs.UPLOADS_SHOWING, value); |  | ||||||
|             defaultKvStore.putBoolean(Prefs.IS_CONTRIBUTION_COUNT_CHANGED, true); |  | ||||||
|             uploadLimit.setText(Integer.toString(value)); |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         langListPreference = findPreference("descriptionDefaultLanguagePref"); |         langListPreference = findPreference("descriptionDefaultLanguagePref"); | ||||||
|         prepareLanguages(); |         prepareLanguages(); | ||||||
|         Preference betaTesterPreference = findPreference("becomeBetaTester"); |         Preference betaTesterPreference = findPreference("becomeBetaTester"); | ||||||
|  | @ -121,7 +84,6 @@ public class SettingsFragment extends PreferenceFragmentCompat { | ||||||
|             findPreference("displayNearbyCardView").setEnabled(false); |             findPreference("displayNearbyCardView").setEnabled(false); | ||||||
|             findPreference("displayLocationPermissionForCardView").setEnabled(false); |             findPreference("displayLocationPermissionForCardView").setEnabled(false); | ||||||
|             findPreference("displayCampaignsCardView").setEnabled(false); |             findPreference("displayCampaignsCardView").setEnabled(false); | ||||||
|             uploadLimit.setEnabled(false); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -23,9 +23,14 @@ import fr.free.nrw.commons.di.CommonsDaggerService; | ||||||
| import fr.free.nrw.commons.media.MediaClient; | import fr.free.nrw.commons.media.MediaClient; | ||||||
| import fr.free.nrw.commons.utils.CommonsDateUtil; | import fr.free.nrw.commons.utils.CommonsDateUtil; | ||||||
| import fr.free.nrw.commons.wikidata.WikidataEditService; | import fr.free.nrw.commons.wikidata.WikidataEditService; | ||||||
|  | import io.reactivex.Completable; | ||||||
| import io.reactivex.Observable; | import io.reactivex.Observable; | ||||||
| import io.reactivex.Scheduler; | import io.reactivex.Scheduler; | ||||||
|  | import io.reactivex.Single; | ||||||
| import io.reactivex.disposables.CompositeDisposable; | import io.reactivex.disposables.CompositeDisposable; | ||||||
|  | import io.reactivex.disposables.Disposable; | ||||||
|  | import io.reactivex.functions.Action; | ||||||
|  | import io.reactivex.functions.Consumer; | ||||||
| import io.reactivex.processors.PublishProcessor; | import io.reactivex.processors.PublishProcessor; | ||||||
| import io.reactivex.schedulers.Schedulers; | import io.reactivex.schedulers.Schedulers; | ||||||
| import java.io.File; | import java.io.File; | ||||||
|  | @ -33,6 +38,7 @@ import java.io.IOException; | ||||||
| import java.text.ParseException; | import java.text.ParseException; | ||||||
| import java.util.HashSet; | import java.util.HashSet; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
|  | import java.util.concurrent.Callable; | ||||||
| import java.util.regex.Matcher; | import java.util.regex.Matcher; | ||||||
| import java.util.regex.Pattern; | import java.util.regex.Pattern; | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
|  | @ -106,10 +112,10 @@ public class UploadService extends CommonsDaggerService { | ||||||
|             notificationManager.notify(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS, curNotification.build()); |             notificationManager.notify(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS, curNotification.build()); | ||||||
| 
 | 
 | ||||||
|             contribution.setTransferred(transferred); |             contribution.setTransferred(transferred); | ||||||
|             compositeDisposable.add(contributionDao. | 
 | ||||||
|                     save(contribution).subscribeOn(ioThreadScheduler) |             compositeDisposable.add(contributionDao.update(contribution) | ||||||
|                     .observeOn(mainThreadScheduler) |                 .subscribeOn(ioThreadScheduler) | ||||||
|                     .subscribe()); |                 .subscribe()); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  | @ -156,14 +162,11 @@ public class UploadService extends CommonsDaggerService { | ||||||
|             Timber.d("%d uploads left", toUpload); |             Timber.d("%d uploads left", toUpload); | ||||||
|             notificationManager.notify(contribution.getLocalUri().toString(), NOTIFICATION_UPLOAD_IN_PROGRESS, curNotification.build()); |             notificationManager.notify(contribution.getLocalUri().toString(), NOTIFICATION_UPLOAD_IN_PROGRESS, curNotification.build()); | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|         compositeDisposable.add(contributionDao |         compositeDisposable.add(contributionDao | ||||||
|             .save(contribution) |             .save(contribution) | ||||||
|             .subscribeOn(ioThreadScheduler) |             .subscribeOn(ioThreadScheduler) | ||||||
|             .observeOn(mainThreadScheduler) |             .subscribe(() -> uploadContribution(contribution))); | ||||||
|             .subscribe(aLong->{ |  | ||||||
|                 contribution.set_id(aLong); |  | ||||||
|                 uploadContribution(contribution); |  | ||||||
|             }, Throwable::printStackTrace)); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private boolean freshStart = true; |     private boolean freshStart = true; | ||||||
|  | @ -269,7 +272,7 @@ public class UploadService extends CommonsDaggerService { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void onUpload(Contribution contribution, String notificationTag, |     private void onUpload(Contribution contribution, String notificationTag, | ||||||
|         UploadResult uploadResult) throws ParseException { |         UploadResult uploadResult) { | ||||||
|         Timber.d("Stash upload response 2 is %s", uploadResult.toString()); |         Timber.d("Stash upload response 2 is %s", uploadResult.toString()); | ||||||
| 
 | 
 | ||||||
|         notificationManager.cancel(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS); |         notificationManager.cancel(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS); | ||||||
|  | @ -282,8 +285,7 @@ public class UploadService extends CommonsDaggerService { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void onSuccessfulUpload(Contribution contribution, UploadResult uploadResult) |     private void onSuccessfulUpload(Contribution contribution, UploadResult uploadResult) { | ||||||
|         throws ParseException { |  | ||||||
|         compositeDisposable |         compositeDisposable | ||||||
|             .add(wikidataEditService.addDepictionsAndCaptions(uploadResult, contribution)); |             .add(wikidataEditService.addDepictionsAndCaptions(uploadResult, contribution)); | ||||||
|         WikidataPlace wikidataPlace = contribution.getWikidataPlace(); |         WikidataPlace wikidataPlace = contribution.getWikidataPlace(); | ||||||
|  | @ -293,17 +295,11 @@ public class UploadService extends CommonsDaggerService { | ||||||
|         saveCompletedContribution(contribution, uploadResult); |         saveCompletedContribution(contribution, uploadResult); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void saveCompletedContribution(Contribution contribution, UploadResult uploadResult) throws ParseException { |     private void saveCompletedContribution(Contribution contribution, UploadResult uploadResult) { | ||||||
|         contribution.setFilename(uploadResult.createCanonicalFileName()); |         compositeDisposable.add(mediaClient.getMedia("File:" + uploadResult.getFilename()) | ||||||
|         contribution.setImageUrl(uploadResult.getImageinfo().getOriginalUrl()); |         .map(media -> new Contribution(media, Contribution.STATE_COMPLETED)) | ||||||
|         contribution.setState(Contribution.STATE_COMPLETED); |         .flatMapCompletable(newContribution -> contributionDao.saveAndDelete(contribution, newContribution)) | ||||||
|         contribution.setDateUploaded(CommonsDateUtil.getIso8601DateFormatTimestamp() |         .subscribe()); | ||||||
|             .parse(uploadResult.getImageinfo().getTimestamp())); |  | ||||||
|         compositeDisposable.add(contributionDao |  | ||||||
|             .save(contribution) |  | ||||||
|             .subscribeOn(ioThreadScheduler) |  | ||||||
|             .observeOn(mainThreadScheduler) |  | ||||||
|             .subscribe()); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @SuppressLint("StringFormatInvalid") |     @SuppressLint("StringFormatInvalid") | ||||||
|  | @ -317,10 +313,11 @@ public class UploadService extends CommonsDaggerService { | ||||||
|         notificationManager.notify(contribution.getLocalUri().toString(), NOTIFICATION_UPLOAD_FAILED, curNotification.build()); |         notificationManager.notify(contribution.getLocalUri().toString(), NOTIFICATION_UPLOAD_FAILED, curNotification.build()); | ||||||
| 
 | 
 | ||||||
|         contribution.setState(Contribution.STATE_FAILED); |         contribution.setState(Contribution.STATE_FAILED); | ||||||
|         compositeDisposable.add(contributionDao.save(contribution) | 
 | ||||||
|                 .subscribeOn(ioThreadScheduler) |         compositeDisposable.add(contributionDao | ||||||
|                 .observeOn(mainThreadScheduler) |             .update(contribution) | ||||||
|                 .subscribe()); |             .subscribeOn(ioThreadScheduler) | ||||||
|  |             .subscribe()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private String findUniqueFilename(String fileName) throws IOException { |     private String findUniqueFilename(String fileName) throws IOException { | ||||||
|  |  | ||||||
|  | @ -21,6 +21,16 @@ public class CommonsDateUtil { | ||||||
|         return simpleDateFormat; |         return simpleDateFormat; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets SimpleDateFormat for date pattern returned by Media object | ||||||
|  |      * @return simpledateformat | ||||||
|  |      */ | ||||||
|  |     public static SimpleDateFormat getMediaSimpleDateFormat() { | ||||||
|  |         SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.ROOT); | ||||||
|  |         simpleDateFormat.setTimeZone(TimeZone.getTimeZone("UTC")); | ||||||
|  |         return simpleDateFormat; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Gets the timestamp pattern for a date |      * Gets the timestamp pattern for a date | ||||||
|      * @return timestamp |      * @return timestamp | ||||||
|  |  | ||||||
|  | @ -26,13 +26,6 @@ | ||||||
|           android:summary="@string/use_external_storage_summary" |           android:summary="@string/use_external_storage_summary" | ||||||
|           android:title="@string/use_external_storage" /> |           android:title="@string/use_external_storage" /> | ||||||
| 
 | 
 | ||||||
|         <EditTextPreference |  | ||||||
|           android:defaultValue="100" |  | ||||||
|           android:key="uploads" |  | ||||||
|           app:useSimpleSummaryProvider="true" |  | ||||||
|           app:singleLineTitle="false" |  | ||||||
|           android:title="@string/set_limit" /> |  | ||||||
| 
 |  | ||||||
|         <ListPreference |         <ListPreference | ||||||
|           android:key="descriptionDefaultLanguagePref" |           android:key="descriptionDefaultLanguagePref" | ||||||
|           app:useSimpleSummaryProvider="true" |           app:useSimpleSummaryProvider="true" | ||||||
|  |  | ||||||
|  | @ -0,0 +1,139 @@ | ||||||
|  | package fr.free.nrw.commons.contributions | ||||||
|  | 
 | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.arch.core.executor.testing.InstantTaskExecutorRule | ||||||
|  | import com.nhaarman.mockitokotlin2.* | ||||||
|  | import fr.free.nrw.commons.Media | ||||||
|  | import fr.free.nrw.commons.auth.SessionManager | ||||||
|  | import fr.free.nrw.commons.media.MediaClient | ||||||
|  | import fr.free.nrw.commons.utils.NetworkUtilsTest | ||||||
|  | import fr.free.nrw.commons.utils.createMockDataSourceFactory | ||||||
|  | import io.reactivex.Scheduler | ||||||
|  | import io.reactivex.Single | ||||||
|  | import io.reactivex.schedulers.Schedulers | ||||||
|  | import io.reactivex.schedulers.TestScheduler | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.mockito.ArgumentMatchers.* | ||||||
|  | import org.mockito.InjectMocks | ||||||
|  | import org.mockito.Mock | ||||||
|  | import org.mockito.Mockito.mock | ||||||
|  | import org.mockito.MockitoAnnotations | ||||||
|  | import java.lang.RuntimeException | ||||||
|  | import java.util.* | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The unit test class for ContributionBoundaryCallbackTest | ||||||
|  |  */ | ||||||
|  | class ContributionBoundaryCallbackTest { | ||||||
|  |     @Mock | ||||||
|  |     internal lateinit var repository: ContributionsRepository | ||||||
|  | 
 | ||||||
|  |     @Mock | ||||||
|  |     internal lateinit var sessionManager: SessionManager | ||||||
|  | 
 | ||||||
|  |     @Mock | ||||||
|  |     internal lateinit var mediaClient: MediaClient | ||||||
|  | 
 | ||||||
|  |     private lateinit var contributionBoundaryCallback: ContributionBoundaryCallback | ||||||
|  | 
 | ||||||
|  |     @Rule | ||||||
|  |     @JvmField | ||||||
|  |     var instantTaskExecutorRule = InstantTaskExecutorRule() | ||||||
|  | 
 | ||||||
|  |     lateinit var scheduler: Scheduler | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * initial setup | ||||||
|  |      */ | ||||||
|  |     @Before | ||||||
|  |     @Throws(Exception::class) | ||||||
|  |     fun setUp() { | ||||||
|  |         MockitoAnnotations.initMocks(this) | ||||||
|  |         scheduler = Schedulers.trampoline() | ||||||
|  |         contributionBoundaryCallback = | ||||||
|  |             ContributionBoundaryCallback(repository, sessionManager, mediaClient, scheduler); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testOnZeroItemsLoaded() { | ||||||
|  |         whenever(repository.save(anyList<Contribution>())) | ||||||
|  |             .thenReturn(Single.just(listOf(1L, 2L))) | ||||||
|  |         whenever(sessionManager.userName).thenReturn("Test") | ||||||
|  |         whenever(mediaClient.getMediaListForUser(anyString())).thenReturn( | ||||||
|  |             Single.just(listOf(mock(Media::class.java))) | ||||||
|  |         ) | ||||||
|  |         whenever(mediaClient.doesMediaListForUserHaveMorePages(anyString())) | ||||||
|  |             .thenReturn(true) | ||||||
|  |         contributionBoundaryCallback.onZeroItemsLoaded() | ||||||
|  |         verify(repository).save(anyList<Contribution>()); | ||||||
|  |         verify(mediaClient).getMediaListForUser(anyString()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testOnLastItemLoaded() { | ||||||
|  |         whenever(repository.save(anyList<Contribution>())) | ||||||
|  |             .thenReturn(Single.just(listOf(1L, 2L))) | ||||||
|  |         whenever(sessionManager.userName).thenReturn("Test") | ||||||
|  |         whenever(mediaClient.getMediaListForUser(anyString())).thenReturn( | ||||||
|  |             Single.just(listOf(mock(Media::class.java))) | ||||||
|  |         ) | ||||||
|  |         whenever(mediaClient.doesMediaListForUserHaveMorePages(anyString())) | ||||||
|  |             .thenReturn(true) | ||||||
|  |         contributionBoundaryCallback.onItemAtEndLoaded(mock(Contribution::class.java)) | ||||||
|  |         verify(repository).save(anyList()); | ||||||
|  |         verify(mediaClient).getMediaListForUser(anyString()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testOnFrontItemLoaded() { | ||||||
|  |         whenever(repository.save(anyList<Contribution>())) | ||||||
|  |             .thenReturn(Single.just(listOf(1L, 2L))) | ||||||
|  |         whenever(sessionManager.userName).thenReturn("Test") | ||||||
|  |         whenever(mediaClient.getMediaListForUser(anyString())).thenReturn( | ||||||
|  |             Single.just(listOf(mock(Media::class.java))) | ||||||
|  |         ) | ||||||
|  |         whenever(mediaClient.doesMediaListForUserHaveMorePages(anyString())) | ||||||
|  |             .thenReturn(true) | ||||||
|  |         contributionBoundaryCallback.onItemAtFrontLoaded(mock(Contribution::class.java)) | ||||||
|  |         verify(repository).save(anyList()); | ||||||
|  |         verify(mediaClient).getMediaListForUser(anyString()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testFetchContributions() { | ||||||
|  |         whenever(repository.save(anyList<Contribution>())) | ||||||
|  |             .thenReturn(Single.just(listOf(1L, 2L))) | ||||||
|  |         whenever(sessionManager.userName).thenReturn("Test") | ||||||
|  |         whenever(mediaClient.getMediaListForUser(anyString())).thenReturn( | ||||||
|  |             Single.just(listOf(mock(Media::class.java))) | ||||||
|  |         ) | ||||||
|  |         whenever(mediaClient.doesMediaListForUserHaveMorePages(anyString())) | ||||||
|  |             .thenReturn(true) | ||||||
|  |         contributionBoundaryCallback.fetchContributions() | ||||||
|  |         verify(repository).save(anyList()); | ||||||
|  |         verify(mediaClient).getMediaListForUser(anyString()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testFetchContributionsForEndOfList() { | ||||||
|  |         whenever(sessionManager.userName).thenReturn("Test") | ||||||
|  |         whenever(mediaClient.doesMediaListForUserHaveMorePages(anyString())) | ||||||
|  |             .thenReturn(false) | ||||||
|  |         contributionBoundaryCallback.fetchContributions() | ||||||
|  |         verify(mediaClient, times(0)).getMediaListForUser(anyString()) | ||||||
|  |         verifyNoMoreInteractions(repository) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testFetchContributionsFailed() { | ||||||
|  |         whenever(sessionManager.userName).thenReturn("Test") | ||||||
|  |         whenever(mediaClient.doesMediaListForUserHaveMorePages(anyString())) | ||||||
|  |             .thenReturn(true) | ||||||
|  |         whenever(mediaClient.getMediaListForUser(anyString())).thenReturn(Single.error(Exception("Error"))) | ||||||
|  |         contributionBoundaryCallback.fetchContributions() | ||||||
|  |         verifyZeroInteractions(repository); | ||||||
|  |         verify(mediaClient).getMediaListForUser(anyString()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,63 @@ | ||||||
|  | package fr.free.nrw.commons.contributions | ||||||
|  | 
 | ||||||
|  | import androidx.arch.core.executor.testing.InstantTaskExecutorRule | ||||||
|  | import com.nhaarman.mockitokotlin2.times | ||||||
|  | import com.nhaarman.mockitokotlin2.verify | ||||||
|  | import com.nhaarman.mockitokotlin2.verifyZeroInteractions | ||||||
|  | import com.nhaarman.mockitokotlin2.whenever | ||||||
|  | import fr.free.nrw.commons.Media | ||||||
|  | import fr.free.nrw.commons.auth.SessionManager | ||||||
|  | import fr.free.nrw.commons.media.MediaClient | ||||||
|  | import io.reactivex.Completable | ||||||
|  | import io.reactivex.Scheduler | ||||||
|  | import io.reactivex.Single | ||||||
|  | import io.reactivex.schedulers.Schedulers | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Rule | ||||||
|  | import org.junit.Test | ||||||
|  | import org.mockito.ArgumentMatchers | ||||||
|  | import org.mockito.ArgumentMatchers.any | ||||||
|  | import org.mockito.Mock | ||||||
|  | import org.mockito.Mockito | ||||||
|  | import org.mockito.Mockito.mock | ||||||
|  | import org.mockito.MockitoAnnotations | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The unit test class for ContributionsListPresenterTest | ||||||
|  |  */ | ||||||
|  | class ContributionsListPresenterTest { | ||||||
|  |     @Mock | ||||||
|  |     internal lateinit var contributionBoundaryCallback: ContributionBoundaryCallback | ||||||
|  | 
 | ||||||
|  |     @Mock | ||||||
|  |     internal lateinit var repository: ContributionsRepository | ||||||
|  | 
 | ||||||
|  |     @Rule | ||||||
|  |     @JvmField | ||||||
|  |     var instantTaskExecutorRule = InstantTaskExecutorRule() | ||||||
|  | 
 | ||||||
|  |     lateinit var scheduler: Scheduler | ||||||
|  | 
 | ||||||
|  |     lateinit var contributionsListPresenter: ContributionsListPresenter | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * initial setup | ||||||
|  |      */ | ||||||
|  |     @Before | ||||||
|  |     @Throws(Exception::class) | ||||||
|  |     fun setUp() { | ||||||
|  |         MockitoAnnotations.initMocks(this) | ||||||
|  |         scheduler = Schedulers.trampoline() | ||||||
|  |         contributionsListPresenter = | ||||||
|  |             ContributionsListPresenter(contributionBoundaryCallback, repository, scheduler); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testDeleteUpload() { | ||||||
|  |         whenever(repository.deleteContributionFromDB(any<Contribution>())) | ||||||
|  |             .thenReturn(Completable.complete()) | ||||||
|  |         contributionsListPresenter.deleteUpload(mock(Contribution::class.java)) | ||||||
|  |         verify(repository, times(1)) | ||||||
|  |             .deleteContributionFromDB(ArgumentMatchers.any(Contribution::class.java)); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -7,9 +7,11 @@ import androidx.lifecycle.LiveData | ||||||
| import androidx.lifecycle.MutableLiveData | import androidx.lifecycle.MutableLiveData | ||||||
| import androidx.loader.content.CursorLoader | import androidx.loader.content.CursorLoader | ||||||
| import androidx.loader.content.Loader | import androidx.loader.content.Loader | ||||||
|  | import com.nhaarman.mockitokotlin2.any | ||||||
| import com.nhaarman.mockitokotlin2.mock | import com.nhaarman.mockitokotlin2.mock | ||||||
| import com.nhaarman.mockitokotlin2.verify | import com.nhaarman.mockitokotlin2.verify | ||||||
| import com.nhaarman.mockitokotlin2.whenever | import com.nhaarman.mockitokotlin2.whenever | ||||||
|  | import io.reactivex.Completable | ||||||
| import io.reactivex.Scheduler | import io.reactivex.Scheduler | ||||||
| import io.reactivex.Single | import io.reactivex.Single | ||||||
| import io.reactivex.schedulers.TestScheduler | import io.reactivex.schedulers.TestScheduler | ||||||
|  | @ -17,9 +19,11 @@ import org.junit.Before | ||||||
| import org.junit.Rule | import org.junit.Rule | ||||||
| import org.junit.Test | import org.junit.Test | ||||||
| import org.mockito.ArgumentMatchers | import org.mockito.ArgumentMatchers | ||||||
|  | import org.mockito.ArgumentMatchers.* | ||||||
| import org.mockito.Mock | import org.mockito.Mock | ||||||
| import org.mockito.Mockito | import org.mockito.Mockito | ||||||
| import org.mockito.MockitoAnnotations | import org.mockito.MockitoAnnotations | ||||||
|  | import java.util.concurrent.TimeUnit | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * The unit test class for ContributionsPresenter |  * The unit test class for ContributionsPresenter | ||||||
|  | @ -42,7 +46,7 @@ class ContributionsPresenterTest { | ||||||
| 
 | 
 | ||||||
|     @Rule @JvmField var instantTaskExecutorRule = InstantTaskExecutorRule() |     @Rule @JvmField var instantTaskExecutorRule = InstantTaskExecutorRule() | ||||||
| 
 | 
 | ||||||
|     lateinit var scheduler : Scheduler |     lateinit var scheduler : TestScheduler | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * initial setup |      * initial setup | ||||||
|  | @ -60,23 +64,13 @@ class ContributionsPresenterTest { | ||||||
|         liveData=MutableLiveData() |         liveData=MutableLiveData() | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * Test fetch contributions |  | ||||||
|      */ |  | ||||||
|     @Test |  | ||||||
|     fun testFetchContributions(){ |  | ||||||
|         whenever(repository.getString(ArgumentMatchers.anyString())).thenReturn("10") |  | ||||||
|         whenever(repository.fetchContributions()).thenReturn(liveData) |  | ||||||
|         contributionsPresenter.fetchContributions() |  | ||||||
|         verify(repository).fetchContributions() |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |     /** | ||||||
|      * Test presenter actions onDeleteContribution |      * Test presenter actions onDeleteContribution | ||||||
|      */ |      */ | ||||||
|     @Test |     @Test | ||||||
|     fun testDeleteContribution() { |     fun testDeleteContribution() { | ||||||
|         whenever(repository.deleteContributionFromDB(ArgumentMatchers.any(Contribution::class.java))).thenReturn(Single.just(1)) |         whenever(repository.deleteContributionFromDB(ArgumentMatchers.any<Contribution>())) | ||||||
|  |             .thenReturn(Completable.complete()) | ||||||
|         contributionsPresenter.deleteUpload(contribution) |         contributionsPresenter.deleteUpload(contribution) | ||||||
|         verify(repository).deleteContributionFromDB(contribution) |         verify(repository).deleteContributionFromDB(contribution) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,55 @@ | ||||||
|  | package fr.free.nrw.commons.contributions | ||||||
|  | 
 | ||||||
|  | import com.nhaarman.mockitokotlin2.times | ||||||
|  | import com.nhaarman.mockitokotlin2.verify | ||||||
|  | import com.nhaarman.mockitokotlin2.whenever | ||||||
|  | import fr.free.nrw.commons.utils.createMockDataSourceFactory | ||||||
|  | import io.reactivex.Scheduler | ||||||
|  | import io.reactivex.Single | ||||||
|  | import junit.framework.Assert.assertEquals | ||||||
|  | import org.junit.Before | ||||||
|  | import org.junit.Test | ||||||
|  | import org.mockito.* | ||||||
|  | import org.mockito.Mockito.any | ||||||
|  | import org.mockito.Mockito.mock | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * The unit test class for ContributionsRepositoryTest | ||||||
|  |  */ | ||||||
|  | class ContributionsRepositoryTest { | ||||||
|  |     @Mock | ||||||
|  |     internal lateinit var localDataSource: ContributionsLocalDataSource | ||||||
|  | 
 | ||||||
|  |     @InjectMocks | ||||||
|  |     private lateinit var contributionsRepository: ContributionsRepository | ||||||
|  | 
 | ||||||
|  |     lateinit var scheduler: Scheduler | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * initial setup | ||||||
|  |      */ | ||||||
|  |     @Before | ||||||
|  |     @Throws(Exception::class) | ||||||
|  |     fun setUp() { | ||||||
|  |         MockitoAnnotations.initMocks(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testFetchContributions() { | ||||||
|  |         val contribution = mock(Contribution::class.java) | ||||||
|  |         whenever(localDataSource.getContributions()) | ||||||
|  |             .thenReturn(createMockDataSourceFactory(listOf(contribution))) | ||||||
|  |         val contributionsFactory = contributionsRepository.fetchContributions() | ||||||
|  |         verify(localDataSource, times(1)).getContributions(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     fun testSaveContribution() { | ||||||
|  |         val contributions = listOf(mock(Contribution::class.java)) | ||||||
|  |         whenever(localDataSource.saveContributions(ArgumentMatchers.anyList())) | ||||||
|  |             .thenReturn(Single.just(listOf(1L))) | ||||||
|  |         val save = contributionsRepository.save(contributions).test().assertValueAt(0) { | ||||||
|  |             it.size == 1 && it.get(0) == 1L | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| package fr.free.nrw.commons.media | package fr.free.nrw.commons.media | ||||||
| 
 | 
 | ||||||
|  | import com.nhaarman.mockitokotlin2.whenever | ||||||
| import fr.free.nrw.commons.Media | import fr.free.nrw.commons.Media | ||||||
| import fr.free.nrw.commons.utils.CommonsDateUtil | import fr.free.nrw.commons.utils.CommonsDateUtil | ||||||
| import io.reactivex.Observable | import io.reactivex.Observable | ||||||
|  | @ -7,18 +8,16 @@ import junit.framework.Assert.* | ||||||
| import org.junit.Before | import org.junit.Before | ||||||
| import org.junit.Test | import org.junit.Test | ||||||
| import org.mockito.* | import org.mockito.* | ||||||
| import org.mockito.Mockito.`when` |  | ||||||
| import org.mockito.Mockito.mock |  | ||||||
| import org.wikipedia.dataclient.mwapi.ImageDetails | import org.wikipedia.dataclient.mwapi.ImageDetails | ||||||
| import org.wikipedia.dataclient.mwapi.MwQueryPage | import org.wikipedia.dataclient.mwapi.MwQueryPage | ||||||
| import org.wikipedia.dataclient.mwapi.MwQueryResponse | import org.wikipedia.dataclient.mwapi.MwQueryResponse | ||||||
| import org.wikipedia.dataclient.mwapi.MwQueryResult | import org.wikipedia.dataclient.mwapi.MwQueryResult | ||||||
| import org.wikipedia.gallery.ImageInfo | import org.wikipedia.gallery.ImageInfo | ||||||
| import org.mockito.ArgumentCaptor | import org.mockito.ArgumentCaptor | ||||||
|  | import org.mockito.ArgumentMatchers.* | ||||||
| import java.util.* | import java.util.* | ||||||
| import org.mockito.Captor | import org.mockito.Captor | ||||||
| 
 | import org.mockito.Mockito.* | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class MediaClientTest { | class MediaClientTest { | ||||||
|  | @ -46,9 +45,10 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.checkPageExistsUsingTitle(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.checkPageExistsUsingTitle(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         val checkPageExistsUsingTitle = mediaClient!!.checkPageExistsUsingTitle("File:Test.jpg").blockingGet() |         val checkPageExistsUsingTitle = | ||||||
|  |             mediaClient!!.checkPageExistsUsingTitle("File:Test.jpg").blockingGet() | ||||||
|         assertTrue(checkPageExistsUsingTitle) |         assertTrue(checkPageExistsUsingTitle) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -63,9 +63,10 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.checkPageExistsUsingTitle(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.checkPageExistsUsingTitle(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         val checkPageExistsUsingTitle = mediaClient!!.checkPageExistsUsingTitle("File:Test.jpg").blockingGet() |         val checkPageExistsUsingTitle = | ||||||
|  |             mediaClient!!.checkPageExistsUsingTitle("File:Test.jpg").blockingGet() | ||||||
|         assertFalse(checkPageExistsUsingTitle) |         assertFalse(checkPageExistsUsingTitle) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -80,7 +81,7 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.checkFileExistsUsingSha(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.checkFileExistsUsingSha(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         val checkFileExistsUsingSha = mediaClient!!.checkFileExistsUsingSha("abcde").blockingGet() |         val checkFileExistsUsingSha = mediaClient!!.checkFileExistsUsingSha("abcde").blockingGet() | ||||||
|         assertTrue(checkFileExistsUsingSha) |         assertTrue(checkFileExistsUsingSha) | ||||||
|  | @ -97,7 +98,7 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.checkFileExistsUsingSha(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.checkFileExistsUsingSha(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         val checkFileExistsUsingSha = mediaClient!!.checkFileExistsUsingSha("abcde").blockingGet() |         val checkFileExistsUsingSha = mediaClient!!.checkFileExistsUsingSha("abcde").blockingGet() | ||||||
|         assertFalse(checkFileExistsUsingSha) |         assertFalse(checkFileExistsUsingSha) | ||||||
|  | @ -117,7 +118,7 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.getMedia(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.getMedia(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         assertEquals("Test", mediaClient!!.getMedia("abcde").blockingGet().filename) |         assertEquals("Test", mediaClient!!.getMedia("abcde").blockingGet().filename) | ||||||
|     } |     } | ||||||
|  | @ -136,10 +137,11 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.getMedia(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.getMedia(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         assertEquals(Media.EMPTY, mediaClient!!.getMedia("abcde").blockingGet()) |         assertEquals(Media.EMPTY, mediaClient!!.getMedia("abcde").blockingGet()) | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     @Captor |     @Captor | ||||||
|     private val filenameCaptor: ArgumentCaptor<String>? = null |     private val filenameCaptor: ArgumentCaptor<String>? = null | ||||||
| 
 | 
 | ||||||
|  | @ -159,7 +161,7 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.getMediaWithGenerator(filenameCaptor!!.capture())) |         `when`(mediaInterface!!.getMediaWithGenerator(filenameCaptor!!.capture())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         assertEquals("Test", mediaClient!!.getPictureOfTheDay().blockingGet().filename) |         assertEquals("Test", mediaClient!!.getPictureOfTheDay().blockingGet().filename) | ||||||
|         assertEquals(template, filenameCaptor.value); |         assertEquals(template, filenameCaptor.value); | ||||||
|  | @ -170,7 +172,7 @@ class MediaClientTest { | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     fun getMediaListFromCategoryTwice() { |     fun getMediaListFromCategoryTwice() { | ||||||
|         val mockContinuation= mapOf(Pair("gcmcontinue", "test")) |         val mockContinuation = mapOf(Pair("gcmcontinue", "test")) | ||||||
|         val imageInfo = ImageInfo() |         val imageInfo = ImageInfo() | ||||||
| 
 | 
 | ||||||
|         val mwQueryPage = mock(MwQueryPage::class.java) |         val mwQueryPage = mock(MwQueryPage::class.java) | ||||||
|  | @ -184,9 +186,13 @@ class MediaClientTest { | ||||||
|         `when`(mockResponse.query()).thenReturn(mwQueryResult) |         `when`(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
|         `when`(mockResponse.continuation()).thenReturn(mockContinuation) |         `when`(mockResponse.continuation()).thenReturn(mockContinuation) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.getMediaListFromCategory(ArgumentMatchers.anyString(), ArgumentMatchers.anyInt(), |         `when`( | ||||||
|                 continuationCaptor!!.capture())) |             mediaInterface!!.getMediaListFromCategory( | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |                 ArgumentMatchers.anyString(), ArgumentMatchers.anyInt(), | ||||||
|  |                 continuationCaptor!!.capture() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |             .thenReturn(Observable.just(mockResponse)) | ||||||
|         val media1 = mediaClient!!.getMediaListFromCategory("abcde").blockingGet().get(0) |         val media1 = mediaClient!!.getMediaListFromCategory("abcde").blockingGet().get(0) | ||||||
|         val media2 = mediaClient!!.getMediaListFromCategory("abcde").blockingGet().get(0) |         val media2 = mediaClient!!.getMediaListFromCategory("abcde").blockingGet().get(0) | ||||||
| 
 | 
 | ||||||
|  | @ -197,6 +203,38 @@ class MediaClientTest { | ||||||
|         assertEquals(media2.filename, "Test") |         assertEquals(media2.filename, "Test") | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     @Test | ||||||
|  |     fun getMediaListForUser() { | ||||||
|  |         val mockContinuation = mapOf("gcmcontinue" to "test") | ||||||
|  |         val imageInfo = ImageInfo() | ||||||
|  | 
 | ||||||
|  |         val mwQueryPage = mock(MwQueryPage::class.java) | ||||||
|  |         whenever(mwQueryPage.title()).thenReturn("Test") | ||||||
|  |         whenever(mwQueryPage.imageInfo()).thenReturn(imageInfo) | ||||||
|  | 
 | ||||||
|  |         val mwQueryResult = mock(MwQueryResult::class.java) | ||||||
|  |         whenever(mwQueryResult.pages()).thenReturn(listOf(mwQueryPage)) | ||||||
|  | 
 | ||||||
|  |         val mockResponse = mock(MwQueryResponse::class.java) | ||||||
|  |         whenever(mockResponse.query()).thenReturn(mwQueryResult) | ||||||
|  |         whenever(mockResponse.continuation()).thenReturn(mockContinuation) | ||||||
|  | 
 | ||||||
|  |         whenever( | ||||||
|  |             mediaInterface!!.getMediaListForUser( | ||||||
|  |                 ArgumentMatchers.anyString(), ArgumentMatchers.anyInt(), | ||||||
|  |                 continuationCaptor!!.capture() | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |             .thenReturn(Observable.just(mockResponse)) | ||||||
|  |         val media1 = mediaClient!!.getMediaListForUser("Test").blockingGet().get(0) | ||||||
|  |         val media2 = mediaClient!!.getMediaListForUser("Test").blockingGet().get(0) | ||||||
|  | 
 | ||||||
|  |         verify(mediaInterface, times(2))?.getMediaListForUser( | ||||||
|  |             ArgumentMatchers.anyString(), | ||||||
|  |             ArgumentMatchers.anyInt(), ArgumentMatchers.anyMap<String, String>() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     @Test |     @Test | ||||||
|     fun getPageHtmlTest() { |     fun getPageHtmlTest() { | ||||||
|         val mwParseResult = mock(MwParseResult::class.java) |         val mwParseResult = mock(MwParseResult::class.java) | ||||||
|  | @ -207,7 +245,7 @@ class MediaClientTest { | ||||||
|         mockResponse.setParse(mwParseResult) |         mockResponse.setParse(mwParseResult) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.getPageHtml(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.getPageHtml(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         assertEquals("Test", mediaClient!!.getPageHtml("abcde").blockingGet()) |         assertEquals("Test", mediaClient!!.getPageHtml("abcde").blockingGet()) | ||||||
|     } |     } | ||||||
|  | @ -218,7 +256,7 @@ class MediaClientTest { | ||||||
|         mockResponse.setParse(null) |         mockResponse.setParse(null) | ||||||
| 
 | 
 | ||||||
|         `when`(mediaInterface!!.getPageHtml(ArgumentMatchers.anyString())) |         `when`(mediaInterface!!.getPageHtml(ArgumentMatchers.anyString())) | ||||||
|                 .thenReturn(Observable.just(mockResponse)) |             .thenReturn(Observable.just(mockResponse)) | ||||||
| 
 | 
 | ||||||
|         assertEquals("", mediaClient!!.getPageHtml("abcde").blockingGet()) |         assertEquals("", mediaClient!!.getPageHtml("abcde").blockingGet()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ import android.net.ConnectivityManager; | ||||||
| import android.net.NetworkInfo; | import android.net.NetworkInfo; | ||||||
| import android.telephony.TelephonyManager; | import android.telephony.TelephonyManager; | ||||||
| 
 | 
 | ||||||
|  | import org.jetbrains.annotations.NotNull; | ||||||
| import org.junit.Before; | import org.junit.Before; | ||||||
| import org.junit.Test; | import org.junit.Test; | ||||||
| 
 | 
 | ||||||
|  | @ -28,34 +29,30 @@ public class NetworkUtilsTest { | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|     public void testInternetConnectionEstablished() { |     public void testInternetConnectionEstablished() { | ||||||
|         Context mockContext = mock(Context.class); |         Context mockContext = getContext(true); | ||||||
|         Application mockApplication = mock(Application.class); |  | ||||||
|         ConnectivityManager mockConnectivityManager = mock(ConnectivityManager.class); |  | ||||||
|         NetworkInfo mockNetworkInfo = mock(NetworkInfo.class); |  | ||||||
|         when(mockNetworkInfo.isConnectedOrConnecting()) |  | ||||||
|                 .thenReturn(true); |  | ||||||
|         when(mockConnectivityManager.getActiveNetworkInfo()) |  | ||||||
|                 .thenReturn(mockNetworkInfo); |  | ||||||
|         when(mockApplication.getSystemService(Context.CONNECTIVITY_SERVICE)) |  | ||||||
|                 .thenReturn(mockConnectivityManager); |  | ||||||
|         when(mockContext.getApplicationContext()).thenReturn(mockApplication); |  | ||||||
|         boolean internetConnectionEstablished = NetworkUtils.isInternetConnectionEstablished(mockContext); |         boolean internetConnectionEstablished = NetworkUtils.isInternetConnectionEstablished(mockContext); | ||||||
|         assertTrue(internetConnectionEstablished); |         assertTrue(internetConnectionEstablished); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @NotNull | ||||||
|     public void testInternetConnectionNotEstablished() { |     public static Context getContext(boolean connectionEstablished) { | ||||||
|         Context mockContext = mock(Context.class); |         Context mockContext = mock(Context.class); | ||||||
|         Application mockApplication = mock(Application.class); |         Application mockApplication = mock(Application.class); | ||||||
|         ConnectivityManager mockConnectivityManager = mock(ConnectivityManager.class); |         ConnectivityManager mockConnectivityManager = mock(ConnectivityManager.class); | ||||||
|         NetworkInfo mockNetworkInfo = mock(NetworkInfo.class); |         NetworkInfo mockNetworkInfo = mock(NetworkInfo.class); | ||||||
|         when(mockNetworkInfo.isConnectedOrConnecting()) |         when(mockNetworkInfo.isConnectedOrConnecting()) | ||||||
|                 .thenReturn(false); |             .thenReturn(connectionEstablished); | ||||||
|         when(mockConnectivityManager.getActiveNetworkInfo()) |         when(mockConnectivityManager.getActiveNetworkInfo()) | ||||||
|                 .thenReturn(mockNetworkInfo); |             .thenReturn(mockNetworkInfo); | ||||||
|         when(mockApplication.getSystemService(Context.CONNECTIVITY_SERVICE)) |         when(mockApplication.getSystemService(Context.CONNECTIVITY_SERVICE)) | ||||||
|                 .thenReturn(mockConnectivityManager); |             .thenReturn(mockConnectivityManager); | ||||||
|         when(mockContext.getApplicationContext()).thenReturn(mockApplication); |         when(mockContext.getApplicationContext()).thenReturn(mockApplication); | ||||||
|  |         return mockContext; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Test | ||||||
|  |     public void testInternetConnectionNotEstablished() { | ||||||
|  |         Context mockContext = getContext(false); | ||||||
|         boolean internetConnectionEstablished = NetworkUtils.isInternetConnectionEstablished(mockContext); |         boolean internetConnectionEstablished = NetworkUtils.isInternetConnectionEstablished(mockContext); | ||||||
|         assertFalse(internetConnectionEstablished); |         assertFalse(internetConnectionEstablished); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,74 @@ | ||||||
|  | package fr.free.nrw.commons.utils | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import android.database.Cursor | ||||||
|  | import androidx.lifecycle.LiveData | ||||||
|  | import androidx.paging.DataSource | ||||||
|  | import androidx.paging.LivePagedListBuilder | ||||||
|  | import androidx.paging.PagedList | ||||||
|  | import androidx.room.InvalidationTracker | ||||||
|  | import androidx.room.RoomDatabase | ||||||
|  | import androidx.room.RoomSQLiteQuery | ||||||
|  | import androidx.room.paging.LimitOffsetDataSource | ||||||
|  | import com.nhaarman.mockitokotlin2.whenever | ||||||
|  | import org.mockito.Mockito.mock | ||||||
|  | 
 | ||||||
|  | fun <T> List<T>.asPagedList(config: PagedList.Config? = null): LiveData<PagedList<T>> { | ||||||
|  |     val defaultConfig = PagedList.Config.Builder() | ||||||
|  |         .setEnablePlaceholders(false) | ||||||
|  |         .setPageSize(size) | ||||||
|  |         .setMaxSize(size + 2) | ||||||
|  |         .setPrefetchDistance(1) | ||||||
|  |         .build() | ||||||
|  |     return LivePagedListBuilder<Int, T>( | ||||||
|  |         createMockDataSourceFactory(this), | ||||||
|  |         config ?: defaultConfig | ||||||
|  |     ).build() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Provides a mocked instance of the data source factory | ||||||
|  |  */ | ||||||
|  | fun <T> createMockDataSourceFactory(itemList: List<T>): DataSource.Factory<Int, T> = | ||||||
|  |     object : DataSource.Factory<Int, T>() { | ||||||
|  |         override fun create(): DataSource<Int, T> = MockLimitDataSource(itemList) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Provides a mocked Room SQL query | ||||||
|  |  */ | ||||||
|  | private fun mockQuery(): RoomSQLiteQuery? { | ||||||
|  |     val query = mock(RoomSQLiteQuery::class.java); | ||||||
|  |     whenever(query.sql).thenReturn(""); | ||||||
|  |     return query; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Provides a mocked Room DB | ||||||
|  |  */ | ||||||
|  | private fun mockDb(): RoomDatabase? { | ||||||
|  |     val roomDatabase = mock(RoomDatabase::class.java); | ||||||
|  |     val invalidationTracker = mock(InvalidationTracker::class.java) | ||||||
|  |     whenever(roomDatabase.invalidationTracker).thenReturn(invalidationTracker); | ||||||
|  |     return roomDatabase; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Class that defines the mocked data source | ||||||
|  |  */ | ||||||
|  | class MockLimitDataSource<T>(private val itemList: List<T>) : | ||||||
|  |     LimitOffsetDataSource<T>(mockDb(), mockQuery(), false, null) { | ||||||
|  |     override fun convertRows(cursor: Cursor?): MutableList<T> = itemList.toMutableList() | ||||||
|  |     override fun countItems(): Int = itemList.count() | ||||||
|  |     override fun isInvalid(): Boolean = false | ||||||
|  |     override fun loadRange(params: LoadRangeParams, callback: LoadRangeCallback<T>) { | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun loadRange(startPosition: Int, loadCount: Int): MutableList<T> { | ||||||
|  |         return itemList.subList(startPosition, startPosition + loadCount).toMutableList() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun loadInitial(params: LoadInitialParams, callback: LoadInitialCallback<T>) { | ||||||
|  |         callback.onResult(itemList, 0) | ||||||
|  |     } | ||||||
|  | } | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Vivek Maskara
						Vivek Maskara