mirror of
				https://github.com/commons-app/apps-android-commons.git
				synced 2025-10-26 20:33:53 +01:00 
			
		
		
		
	Compare commits
	
		
			388 commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 63f621cb56 | ||
|   | e81f916626 | ||
|   | 28fa7b1a20 | ||
|   | aae9d4a387 | ||
|   | 6873f63cf8 | ||
|   | 2d0255e5fb | ||
|   | 32ae406cca | ||
|   | 3e04a1f036 | ||
|   | 6487191394 | ||
|   | beaf211f39 | ||
|   | 3549789cdf | ||
|   | def33552f9 | ||
|   | 3a55583460 | ||
|   | 717a855149 | ||
|   | 29b6d0f8fe | ||
|   | b5b5d8a8e4 | ||
|   | 714e5f8a4b | ||
|   | 7d96e94689 | ||
|   | 7a865df909 | ||
|   | 864884e7b2 | ||
|   | 1ecaf09f21 | ||
|   | 1ff2a28326 | ||
|   | b48905a153 | ||
|   | 09c8d987e1 | ||
|   | 2e52adbef8 | ||
|   | 61c9de6fcc | ||
|   | 41d95814c9 | ||
|   | c4cb65fc3c | ||
|   | a1c5974e93 | ||
|   | 0c244f369c | ||
|   | b6014b017c | ||
|   | 91ea4a6e7b | ||
|   | 1e51c4c5d0 | ||
|   | fbd28a0564 | ||
|   | d0965206cd | ||
|   | bb330c1771 | ||
|   | 14d6c80241 | ||
|   | 4c621364c9 | ||
|   | 2a9d5db51e | ||
|   | b8d340fbe8 | ||
|   | dd1814c793 | ||
|   | adb6181e9f | ||
|   | 0a4b179db5 | ||
|   | e78db7fa08 | ||
|   | 7be615bacb | ||
|   | 95d58023c7 | ||
|   | 7b8fbc239b | ||
|   | 30d1107cef | ||
|   | fe16c44caa | ||
|   | 4ed9ad5085 | ||
|   | 755d8311dc | ||
|   | b6457cc6b9 | ||
|   | 2d51a7ce9a | ||
|   | 0ade0705e2 | ||
|   | 6bc25ccd9b | ||
|   | ed7007fc8c | ||
|   | 71ad6a2ce5 | ||
|   | e9a1af0f52 | ||
|   | 10c384ffa7 | ||
|   | 4e51977fb6 | ||
|   | d632c268ae | ||
|   | be371e5236 | ||
|   | 25d3068faf | ||
|   | 179c7c1855 | ||
|   | 8018000584 | ||
|   | 657af4fe04 | ||
|   | 219fcd3dd8 | ||
|   | 2e9726b84f | ||
|   | 64c6b0c8d0 | ||
|   | fcc63b9f09 | ||
|   | a283ffe2bc | ||
|   | 2811b181b7 | ||
|   | 730f314200 | ||
|   | 81da5c9a1a | ||
|   | a59bf64677 | ||
|   | e2c8f85a5b | ||
|   | dd96c64182 | ||
|   | 9ba702eaa9 | ||
|   | 296b4c1f52 | ||
|   | 48e7effd0a | ||
|   | b9f353bb5a | ||
|   | c22e8447b3 | ||
|   | f810a2d49b | ||
|   | 4f3f7b97fd | ||
|   | 718c466505 | ||
|   | b8a558303b | ||
|   | a892aa6dee | ||
|   | 5a6b3cbf09 | ||
|   | 5bdfbf5f6f | ||
|   | 1d7d2801e4 | ||
|   | 5201af70cd | ||
|   | d0e95bc3c2 | ||
|   | ffb9af1f1c | ||
|   | 6dcce45c59 | ||
|   | 6f36cae767 | ||
|   | 516039c91d | ||
|   | 8de57304bf | ||
|   | 869371b485 | ||
|   | 929711da98 | ||
|   | b2816e1459 | ||
|   | 532bd8baa6 | ||
|   | 90ab7a2766 | ||
|   | ee33a9350f | ||
|   | f1e6f1ad31 | ||
|   | 11e3e37263 | ||
|   | da694022ac | ||
|   | 29ade1e5b7 | ||
|   | 88565b70c5 | ||
|   | e5dbcfc2a1 | ||
|   | 0cda8e4d70 | ||
|   | 7500b6d374 | ||
|   | a4c7a9c4f7 | ||
|   | 8fc7e1039b | ||
|   | 79f52db929 | ||
|   | 13048cc2fd | ||
|   | 66395b9871 | ||
|   | 65f41beed8 | ||
|   | f98b49608e | ||
|   | 3bd0ec4466 | ||
|   | 4befff8f42 | ||
|   | 89436b0a75 | ||
|   | 6de5a07e0d | ||
|   | 27b9d70333 | ||
|   | 9a94dc2548 | ||
|   | b1a8308aaf | ||
|   | ad7dddaac4 | ||
|   | 5d7f42d127 | ||
|   | d9e8917418 | ||
|   | 09da7b8d68 | ||
|   | ca5c7ec966 | ||
|   | 9eff9e8e82 | ||
|   | 5665bc7f93 | ||
|   | 20e5df7d49 | ||
|   | d3ae925567 | ||
|   | af82cb2123 | ||
|   | 7df52e3f9c | ||
|   | 6b40560dfc | ||
|   | 54bb789461 | ||
|   | 7979be17c1 | ||
|   | 91564a1dff | ||
|   | 2b5f0e4ac9 | ||
|   | 9b04031c91 | ||
|   | 8ff52e6815 | ||
|   | c41b5cc9da | ||
|   | 767b625289 | ||
|   | f45f26e602 | ||
|   | 06a613e855 | ||
|   | 62c5231dc9 | ||
|   | 7a224a9120 | ||
|   | 593335aea3 | ||
|   | 6edc6a22e4 | ||
|   | 230604f5ef | ||
|   | 73f5200c2d | ||
|   | 95b8ac74b9 | ||
|   | cfc2cfcca1 | ||
|   | ed1485ca22 | ||
|   | c49c85e68b | ||
|   | 91ca2e6672 | ||
|   | 8849f8984b | ||
|   | bb21e4bdcd | ||
|   | eb617ae8ca | ||
|   | b3c1474b31 | ||
|   | 21ffcb56fd | ||
|   | f977e16774 | ||
|   | 012020735f | ||
|   | 3f2077a6db | ||
|   | f06ae4ebfe | ||
|   | 865824a8e3 | ||
|   | 4d2170257a | ||
|   | 0024e72a2e | ||
|   | 60aca9a5e3 | ||
|   | d0f6c16878 | ||
|   | 8fded5ef6e | ||
|   | 329a68216e | ||
|   | 30762971db | ||
|   | 7479d96675 | ||
|   | ed42d85f67 | ||
|   | 78d29bcf20 | ||
|   | 1a13cb3383 | ||
|   | 9289dcc42c | ||
|   | efdc9c5548 | ||
|   | 69b3544107 | ||
|   | 5b5aeead88 | ||
|   | 4bacac1f8b | ||
|   | 6aeb3c07cc | ||
|   | 2c41176a6e | ||
|   | e3dd00bcfa | ||
|   | 262efe4d8c | ||
|   | 2eed441462 | ||
|   | 56fa8ceb5a | ||
|   | 7bf9276d1a | ||
|   | 51da9e4dd6 | ||
|   | 731ff62faf | ||
|   | fdfd7781e9 | ||
|   | 6e090c8d7a | ||
|   | 44966645ca | ||
|   | 669f3043ae | ||
|   | 5a5e660a43 | ||
|   | 2e05a58e8b | ||
|   | f1f4e8baff | ||
|   | 828f69fc46 | ||
|   | 954a7aee91 | ||
|   | fa0bdf5747 | ||
|   | 67ac92ff57 | ||
|   | e1466c866b | ||
|   | ba89894dc4 | ||
|   | c46c1d2353 | ||
|   | 30322707fc | ||
|   | 972bf785f1 | ||
|   | 32d485cc51 | ||
|   | d11439f85d | ||
|   | 681881f4f6 | ||
|   | 939d01b267 | ||
|   | d233de6103 | ||
|   | 139a296bd3 | ||
|   | 6b56075df8 | ||
|   | 218476acbc | ||
|   | fa24b93830 | ||
|   | b2f655522e | ||
|   | 88eedc3506 | ||
|   | aa84dedd64 | ||
|   | 1c7dce9e12 | ||
|   | 1c4797d3aa | ||
|   | b2927483fa | ||
|   | fda87b7823 | ||
|   | 71d3d12020 | ||
|   | 50eb13a850 | ||
|   | a8e38f4329 | ||
|   | d32ab15d42 | ||
|   | 8dd1091608 | ||
|   | 44f69fcabd | ||
|   | 8d0da86569 | ||
|   | 98b25acab9 | ||
|   | 40241b4142 | ||
|   | 7a685b1241 | ||
|   | 34943542bf | ||
|   | 6345fef6bf | ||
|   | a529ba8032 | ||
|   | e9e2697369 | ||
|   | 12cadd0186 | ||
|   | 1e77b1457a | ||
|   | 43dca1dd14 | ||
|   | 30a7f702a1 | ||
|   | 0293b865b4 | ||
|   | 7566ddf529 | ||
|   | e653857437 | ||
|   | 7b291535e0 | ||
|   | 9dc9a3b8ab | ||
|   | 5d4474ead6 | ||
|   | 36f844a709 | ||
|   | e01ecb20fa | ||
|   | 41170d81d9 | ||
|   | 7400872f87 | ||
|   | aedcd7f9b9 | ||
|   | bb974f8935 | ||
|   | 77bad3380c | ||
|   | 3570377678 | ||
|   | d4ababc0a5 | ||
|   | 1c6ebafb29 | ||
|   | 23e1f01783 | ||
|   | ef032b0f93 | ||
|   | 35a2fe87db | ||
|   | 9f1fe8737f | ||
|   | 2d6583fea6 | ||
|   | 1e64acdf1d | ||
|   | 1f33926ed5 | ||
|   | 16ac08fe21 | ||
|   | 70291a0cb2 | ||
|   | 62136b5b09 | ||
|   | 76078cf3b5 | ||
|   | efc9ae8fb6 | ||
|   | d4a9bacd91 | ||
|   | 0e735512bb | ||
|   | 6d64357d45 | ||
|   | 78666ccbde | ||
|   | 39b513da12 | ||
|   | 18f599b554 | ||
|   | 3e7565c7e3 | ||
|   | 87a453cb72 | ||
|   | fdbe504ca9 | ||
|   | b2159ed87f | ||
|   | ecb19d6984 | ||
|   | 940c0740b0 | ||
|   | cebe1c2a1f | ||
|   | 1d8d1d6b03 | ||
|   | 25e467b3a5 | ||
|   | 038ae9acd4 | ||
|   | ea20a64b34 | ||
|   | 411184fde8 | ||
|   | bf89f11606 | ||
|   | faa58a19de | ||
|   | d2751595cb | ||
|   | 46cefa4899 | ||
|   | 5bc58284aa | ||
|   | a6444968fa | ||
|   | dec56a3342 | ||
|   | 22238f55cd | ||
|   | 4244373a5d | ||
|   | 75ca96a526 | ||
|   | 0d71da106f | ||
|   | 86cdf96f3d | ||
|   | e7864ac1dd | ||
|   | a9058d129e | ||
|   | 369e79be5e | ||
|   | c963cd9ea4 | ||
|   | 7479767266 | ||
|   | b55c61ddb8 | ||
|   | f1e8e48769 | ||
|   | d0bde4a3fe | ||
|   | 6a32454347 | ||
|   | 4dd16054ca | ||
|   | 4b152fc15f | ||
|   | c891c2b0df | ||
|   | 70b4f78a5d | ||
|   | 4c9637c821 | ||
|   | a4b74794cb | ||
|   | 5500b03976 | ||
|   | 0153cbe0ed | ||
|   | e8970ab7f2 | ||
|   | a933b92efa | ||
|   | f2d1f7dbbb | ||
|   | 235e8cdba2 | ||
|   | c1acdbe31a | ||
|   | 2c8c441f25 | ||
|   | cb007608d9 | ||
|   | 8a55b5e613 | ||
|   | b2810bcef1 | ||
|   | f51b607312 | ||
|   | 3bfa3612c6 | ||
|   | 9a876fa5e2 | ||
|   | c175a4ee03 | ||
|   | 3030a6fca7 | ||
|   | 73311970c5 | ||
|   | 56ada36b83 | ||
|   | 85d9aef2f3 | ||
|   | 04a07ed655 | ||
|   | cc74707894 | ||
|   | 64fd10d00e | ||
|   | 015c5d5c63 | ||
|   | 64354fb9e4 | ||
|   | a8387f01c9 | ||
|   | ae52267a27 | ||
|   | f8d519e8eb | ||
|   | 3777f18bf9 | ||
|   | 9dd504e560 | ||
|   | 33548fa57d | ||
|   | 8265cc6306 | ||
|   | 771f370f9a | ||
|   | fb1ef3212d | ||
|   | 1e5521b434 | ||
|   | dac3657536 | ||
|   | d6c4cab207 | ||
|   | 1afff73c24 | ||
|   | a6152f937e | ||
|   | 794dbb8f92 | ||
|   | 0c969c365b | ||
|   | 238023056f | ||
|   | 381f9eca0c | ||
|   | 874773b881 | ||
|   | 00cfd83521 | ||
|   | bafae821e2 | ||
|   | e070c5dbe8 | ||
|   | fe347c21fd | ||
|   | 088dd2479e | ||
|   | cf88f9b796 | ||
|   | 5f1d284309 | ||
|   | ed18a37577 | ||
|   | cb4ffd8ca8 | ||
|   | 0fdb0044b9 | ||
|   | c439143dd3 | ||
|   | 5c8c4032e9 | ||
|   | 248c7b0ceb | ||
|   | 183e84c098 | ||
|   | 634bc3ede1 | ||
|   | 17a8845dfd | ||
|   | a70d585df8 | ||
|   | 3bd7b533d4 | ||
|   | e388f456dc | ||
|   | c46928252c | ||
|   | 091ddb5db1 | ||
|   | f011abef1d | ||
|   | 7c826502b6 | ||
|   | 197855af0e | ||
|   | 522f1fe192 | ||
|   | cdc4f89da5 | ||
|   | bc065c8792 | ||
|   | 7c58891892 | ||
|   | 3e020ed973 | 
					 1026 changed files with 59793 additions and 52727 deletions
				
			
		
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -1,7 +1,7 @@ | ||||||
| name: "\U0001F41E Bug report" | name: "\U0001F41E Bug report" | ||||||
| description: Create a report to help us improve. | description: Create a report to help us improve. | ||||||
| title: "[Bug]: " | title: "[Bug]: " | ||||||
| labels: ["bug"] | type: Bug  # Retained to categorize the issue as per organization-level type | ||||||
| body: | body: | ||||||
|   - type: markdown |   - type: markdown | ||||||
|     attributes: |     attributes: | ||||||
|  | @ -70,7 +70,7 @@ body: | ||||||
|       required: false |       required: false | ||||||
|   - type: textarea |   - type: textarea | ||||||
|     attributes: |     attributes: | ||||||
|       label: Screen-shots |       label: Screenshots | ||||||
|       description: Add screenshots related to the issue (if available). Can be created by pressing the Volume Down and Power Button at the same time on Android 4.0 and higher. |       description: Add screenshots related to the issue (if available). Can be created by pressing the Volume Down and Power Button at the same time on Android 4.0 and higher. | ||||||
|     validations: |     validations: | ||||||
|       required: false |       required: false | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								.github/workflows/android.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/android.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -12,17 +12,17 @@ jobs: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout code |       - name: Checkout code | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
| 
 | 
 | ||||||
|       - name: Set up JDK |       - name: Set up JDK | ||||||
|         uses: actions/setup-java@v3 |         uses: actions/setup-java@v4 | ||||||
|         with: |         with: | ||||||
|           distribution: 'temurin' |           distribution: 'temurin' | ||||||
|           java-version: '17' |           java-version: '17' | ||||||
| 
 | 
 | ||||||
|       - name: Cache packages |       - name: Cache packages | ||||||
|         id: cache-packages |         id: cache-packages | ||||||
|         uses: actions/cache@v3 |         uses: actions/cache@v4 | ||||||
|         with: |         with: | ||||||
|           path: | |           path: | | ||||||
|             ~/.gradle/caches |             ~/.gradle/caches | ||||||
|  | @ -37,7 +37,7 @@ jobs: | ||||||
| 
 | 
 | ||||||
|       - name: AVD cache |       - name: AVD cache | ||||||
|         if: github.event_name != 'pull_request' |         if: github.event_name != 'pull_request' | ||||||
|         uses: actions/cache@v3 |         uses: actions/cache@v4 | ||||||
|         id: avd-cache |         id: avd-cache | ||||||
|         with: |         with: | ||||||
|           path: | |           path: | | ||||||
|  | @ -89,7 +89,7 @@ jobs: | ||||||
|         run: bash ./gradlew assembleBetaDebug --stacktrace |         run: bash ./gradlew assembleBetaDebug --stacktrace | ||||||
| 
 | 
 | ||||||
|       - name: Upload betaDebug APK |       - name: Upload betaDebug APK | ||||||
|         uses: actions/upload-artifact@v3 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: betaDebugAPK |           name: betaDebugAPK | ||||||
|           path: app/build/outputs/apk/beta/debug/app-*.apk |           path: app/build/outputs/apk/beta/debug/app-*.apk | ||||||
|  | @ -98,7 +98,17 @@ jobs: | ||||||
|         run: bash ./gradlew assembleProdDebug --stacktrace |         run: bash ./gradlew assembleProdDebug --stacktrace | ||||||
| 
 | 
 | ||||||
|       - name: Upload prodDebug APK |       - name: Upload prodDebug APK | ||||||
|         uses: actions/upload-artifact@v3 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: prodDebugAPK |           name: prodDebugAPK | ||||||
|           path: app/build/outputs/apk/prod/debug/app-*.apk |           path: app/build/outputs/apk/prod/debug/app-*.apk | ||||||
|  | 
 | ||||||
|  |       - name: Create and PR number artifact | ||||||
|  |         run: | | ||||||
|  |           echo "{\"pr_number\": ${{ github.event.pull_request.number || 'null' }}}" > pr_number.json | ||||||
|  |            | ||||||
|  |       - name: Upload PR number artifact | ||||||
|  |         uses: actions/upload-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: pr_number | ||||||
|  |           path: ./pr_number.json | ||||||
|  |  | ||||||
							
								
								
									
										41
									
								
								.github/workflows/build-beta.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/build-beta.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,41 @@ | ||||||
|  | name: Build beta only | ||||||
|  | 
 | ||||||
|  | on: [workflow_dispatch] | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  | 
 | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  | 
 | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |     - name: set up JDK 17 | ||||||
|  |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         java-version: '17' | ||||||
|  |         distribution: 'temurin' | ||||||
|  |         cache: gradle | ||||||
|  | 
 | ||||||
|  |     - name: Access test login credentials | ||||||
|  |       run: | | ||||||
|  |         echo "TEST_USER_NAME=${{ secrets.TEST_USER_NAME }}" >> local.properties | ||||||
|  |         echo "TEST_USER_PASSWORD=${{ secrets.TEST_USER_PASSWORD }}" >> local.properties | ||||||
|  | 
 | ||||||
|  |     - name: Grant execute permission for gradlew | ||||||
|  |       run: chmod +x gradlew | ||||||
|  | 
 | ||||||
|  |     - name: Set env | ||||||
|  |       run: echo "COMMIT_SHA=$(git log -n 1 --format='%h')" >> $GITHUB_ENV | ||||||
|  | 
 | ||||||
|  |     - name: Generate betaDebug APK | ||||||
|  |       run: ./gradlew assembleBetaDebug --stacktrace | ||||||
|  | 
 | ||||||
|  |     - name: Rename betaDebug APK | ||||||
|  |       run: mv app/build/outputs/apk/beta/debug/app-*.apk app/build/outputs/apk/beta/debug/apps-android-commons-betaDebug-$COMMIT_SHA.apk | ||||||
|  | 
 | ||||||
|  |     - name: Upload betaDebug APK | ||||||
|  |       uses: actions/upload-artifact@v4 | ||||||
|  |       with: | ||||||
|  |         name: apps-android-commons-betaDebugAPK-${{ env.COMMIT_SHA }} | ||||||
|  |         path: app/build/outputs/apk/beta/debug/*.apk | ||||||
|  |         retention-days: 30 | ||||||
							
								
								
									
										96
									
								
								.github/workflows/comment_artifacts_on_PR.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								.github/workflows/comment_artifacts_on_PR.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,96 @@ | ||||||
|  | name: Comment Artifacts on PR | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   workflow_run: | ||||||
|  |     workflows: [ "Android CI" ] | ||||||
|  |     types: [ completed ] | ||||||
|  | 
 | ||||||
|  | permissions: | ||||||
|  |   pull-requests: write | ||||||
|  |   contents: read | ||||||
|  | 
 | ||||||
|  | concurrency: | ||||||
|  |   group: comment-${{ github.event.workflow_run.id }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   comment: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     if: ${{ github.event.workflow_run.conclusion == 'success' && github.event.workflow_run.event == 'pull_request' }} | ||||||
|  |     steps: | ||||||
|  |       - name: Download and process artifacts | ||||||
|  |         uses: actions/github-script@v7 | ||||||
|  |         with: | ||||||
|  |           github-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           script: | | ||||||
|  |             const fs = require('fs'); | ||||||
|  |             const runId = context.payload.workflow_run.id; | ||||||
|  | 
 | ||||||
|  |             const allArtifacts = await github.rest.actions.listWorkflowRunArtifacts({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               run_id: runId, | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             const prNumberArtifact = allArtifacts.data.artifacts.find(artifact => artifact.name === "pr_number"); | ||||||
|  |             if (!prNumberArtifact) { | ||||||
|  |               console.log("pr_number artifact not found."); | ||||||
|  |               return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const download = await github.rest.actions.downloadArtifact({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               artifact_id: prNumberArtifact.id, | ||||||
|  |               archive_format: 'zip', | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             fs.writeFileSync(`${process.env.GITHUB_WORKSPACE}/pr_number.zip`, Buffer.from(download.data)); | ||||||
|  |             const { execSync } = require('child_process'); | ||||||
|  |             execSync('unzip -q pr_number.zip -d ./pr_number/'); | ||||||
|  |             fs.unlinkSync('pr_number.zip'); | ||||||
|  | 
 | ||||||
|  |             const prData = JSON.parse(fs.readFileSync('./pr_number/pr_number.json', 'utf8')); | ||||||
|  |             const prNumber = prData.pr_number; | ||||||
|  | 
 | ||||||
|  |             if (!prNumber || prNumber === 'null') { | ||||||
|  |               console.log("No valid PR number found in pr_number.json. Skipping."); | ||||||
|  |               return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const artifactsToLink = allArtifacts.data.artifacts.filter(artifact => artifact.name !== "pr_number"); | ||||||
|  |             if (artifactsToLink.length === 0) { | ||||||
|  |               console.log("No artifacts to link found."); | ||||||
|  |               return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const comments = await github.rest.issues.listComments({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               issue_number: Number(prNumber), | ||||||
|  |             }); | ||||||
|  |              | ||||||
|  |             const oldComments = comments.data.filter(comment =>  | ||||||
|  |               comment.body.startsWith("✅ Generated APK variants!") | ||||||
|  |             ); | ||||||
|  |             for (const comment of oldComments) { | ||||||
|  |               await github.rest.issues.deleteComment({ | ||||||
|  |                 owner: context.repo.owner, | ||||||
|  |                 repo: context.repo.repo, | ||||||
|  |                 comment_id: comment.id, | ||||||
|  |               }); | ||||||
|  |               console.log(`Deleted old comment ID: ${comment.id}`); | ||||||
|  |             }; | ||||||
|  |              | ||||||
|  |             const commentBody = `✅ Generated APK variants!\n` +  | ||||||
|  |             artifactsToLink.map(artifact => { | ||||||
|  |                 const artifactUrl = `https://github.com/${context.repo.owner}/${context.repo.repo}/actions/runs/${runId}/artifacts/${artifact.id}`; | ||||||
|  |                 return `- 🤖 [Download ${artifact.name}](${artifactUrl})`; | ||||||
|  |             }).join('\n'); | ||||||
|  | 
 | ||||||
|  |             await github.rest.issues.createComment({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               issue_number: Number(prNumber), | ||||||
|  |               body: commentBody | ||||||
|  |             }); | ||||||
							
								
								
									
										3
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -46,4 +46,5 @@ captures/* | ||||||
| 
 | 
 | ||||||
| # Test and other output | # Test and other output | ||||||
| app/jacoco.exec | app/jacoco.exec | ||||||
| app/CommonsContributions | app/CommonsContributions | ||||||
|  | app/.* | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								.idea/codeStyles/Project.xml
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								.idea/codeStyles/Project.xml
									
										
									
										generated
									
									
									
								
							|  | @ -16,6 +16,7 @@ | ||||||
|       <option name="NAMES_COUNT_TO_USE_IMPORT_ON_DEMAND" value="999" /> |       <option name="NAMES_COUNT_TO_USE_IMPORT_ON_DEMAND" value="999" /> | ||||||
|       <option name="IMPORT_LAYOUT_TABLE"> |       <option name="IMPORT_LAYOUT_TABLE"> | ||||||
|         <value> |         <value> | ||||||
|  |           <package name="" withSubpackages="true" static="false" module="true" /> | ||||||
|           <package name="" withSubpackages="true" static="true" /> |           <package name="" withSubpackages="true" static="true" /> | ||||||
|           <emptyLine /> |           <emptyLine /> | ||||||
|           <package name="" withSubpackages="true" static="false" /> |           <package name="" withSubpackages="true" static="false" /> | ||||||
|  |  | ||||||
							
								
								
									
										52
									
								
								.idea/inspectionProfiles/Project_Default.xml
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										52
									
								
								.idea/inspectionProfiles/Project_Default.xml
									
										
									
										generated
									
									
									
								
							|  | @ -1,16 +1,36 @@ | ||||||
| <component name="InspectionProjectProfileManager"> | <component name="InspectionProjectProfileManager"> | ||||||
|   <profile version="1.0"> |   <profile version="1.0"> | ||||||
|     <option name="myName" value="Project Default" /> |     <option name="myName" value="Project Default" /> | ||||||
|     <inspection_tool class="AndroidLintNewerVersionAvailable" enabled="true" level="WARNING" enabled_by_default="true" /> |  | ||||||
|     <inspection_tool class="AutoCloseableResource" enabled="true" level="WARNING" enabled_by_default="true" /> |  | ||||||
|     <inspection_tool class="ClassWithOnlyPrivateConstructors" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="ClassWithOnlyPrivateConstructors" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|  |     <inspection_tool class="ComposePreviewDimensionRespectsLimit" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="ComposePreviewMustBeTopLevelFunction" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="ComposePreviewNeedsComposableAnnotation" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="ComposePreviewNotSupportedInUnitTestFiles" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|     <inspection_tool class="ConfusingElse" enabled="true" level="WARNING" enabled_by_default="true"> |     <inspection_tool class="ConfusingElse" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|       <option name="reportWhenNoStatementFollow" value="true" /> |       <option name="reportWhenNoStatementFollow" value="true" /> | ||||||
|     </inspection_tool> |     </inspection_tool> | ||||||
|     <inspection_tool class="ControlFlowStatementWithoutBraces" enabled="true" level="ERROR" enabled_by_default="true" /> |     <inspection_tool class="ControlFlowStatementWithoutBraces" enabled="true" level="ERROR" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="DefaultNotLastCaseInSwitch" enabled="true" level="WARNING" enabled_by_default="true" /> |  | ||||||
|     <inspection_tool class="ExplicitThis" enabled="true" level="WEAK WARNING" enabled_by_default="true" /> |     <inspection_tool class="ExplicitThis" enabled="true" level="WEAK WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="FieldMayBeFinal" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="GlancePreviewDimensionRespectsLimit" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="GlancePreviewMustBeTopLevelFunction" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="GlancePreviewNeedsComposableAnnotation" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="GlancePreviewNotSupportedInUnitTestFiles" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|     <inspection_tool class="LocalCanBeFinal" enabled="true" level="WARNING" enabled_by_default="true"> |     <inspection_tool class="LocalCanBeFinal" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|       <option name="REPORT_VARIABLES" value="true" /> |       <option name="REPORT_VARIABLES" value="true" /> | ||||||
|       <option name="REPORT_PARAMETERS" value="true" /> |       <option name="REPORT_PARAMETERS" value="true" /> | ||||||
|  | @ -24,14 +44,33 @@ | ||||||
|     <inspection_tool class="OverlyStrongTypeCast" enabled="true" level="WARNING" enabled_by_default="true"> |     <inspection_tool class="OverlyStrongTypeCast" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|       <option name="ignoreInMatchingInstanceof" value="false" /> |       <option name="ignoreInMatchingInstanceof" value="false" /> | ||||||
|     </inspection_tool> |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewAnnotationInFunctionWithParameters" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewApiLevelMustBeValid" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewDeviceShouldUseNewSpec" enabled="true" level="WEAK WARNING" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewFontScaleMustBeGreaterThanZero" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewMultipleParameterProviders" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewParameterProviderOnFirstParameter" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|  |     <inspection_tool class="PreviewPickerAnnotation" enabled="true" level="ERROR" enabled_by_default="true"> | ||||||
|  |       <option name="composableFile" value="true" /> | ||||||
|  |     </inspection_tool> | ||||||
|     <inspection_tool class="ProblematicWhitespace" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="ProblematicWhitespace" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="ProtectedMemberInFinalClass" enabled="true" level="WARNING" enabled_by_default="true" /> |  | ||||||
|     <inspection_tool class="RedundantFieldInitialization" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="RedundantFieldInitialization" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="RedundantImplements" enabled="true" level="WARNING" enabled_by_default="true"> |     <inspection_tool class="RedundantImplements" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|       <option name="ignoreSerializable" value="false" /> |       <option name="ignoreSerializable" value="false" /> | ||||||
|       <option name="ignoreCloneable" value="false" /> |       <option name="ignoreCloneable" value="false" /> | ||||||
|     </inspection_tool> |     </inspection_tool> | ||||||
|     <inspection_tool class="RedundantMethodOverride" enabled="true" level="WARNING" enabled_by_default="true" /> |  | ||||||
|     <inspection_tool class="SimplifiableEqualsExpression" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="SimplifiableEqualsExpression" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="TypeParameterExtendsFinalClass" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="TypeParameterExtendsFinalClass" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="UnnecessarilyQualifiedStaticUsage" enabled="true" level="WARNING" enabled_by_default="true"> |     <inspection_tool class="UnnecessarilyQualifiedStaticUsage" enabled="true" level="WARNING" enabled_by_default="true"> | ||||||
|  | @ -47,6 +86,5 @@ | ||||||
|     <inspection_tool class="UnnecessaryQualifierForThis" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="UnnecessaryQualifierForThis" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="UnnecessarySuperConstructor" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="UnnecessarySuperConstructor" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="UnnecessaryThis" enabled="true" level="WARNING" enabled_by_default="true" /> |     <inspection_tool class="UnnecessaryThis" enabled="true" level="WARNING" enabled_by_default="true" /> | ||||||
|     <inspection_tool class="UnnecessaryToStringCall" enabled="true" level="WARNING" enabled_by_default="true" /> |  | ||||||
|   </profile> |   </profile> | ||||||
| </component> | </component> | ||||||
							
								
								
									
										85
									
								
								CHANGELOG.md
									
										
									
									
									
								
							
							
						
						
									
										85
									
								
								CHANGELOG.md
									
										
									
									
									
								
							|  | @ -1,5 +1,90 @@ | ||||||
| # Wikimedia Commons for Android | # Wikimedia Commons for Android | ||||||
| 
 | 
 | ||||||
|  | ## v6.0.2 | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * Addressed a bug that prevented the keyboard from appearing in various text fields, such as on the upload wizard | ||||||
|  | * Links in the "File usages" list are now clickable and will take you to the correct page. | ||||||
|  | * Titles for file usages are now clearer and easier to understand | ||||||
|  | * Bug fixes and stability improvements | ||||||
|  | 
 | ||||||
|  | ## v6.0.1 | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * The app now supports Android 15 with an improved user interface | ||||||
|  | * Enhanced Nearby with robust and more reliable labels | ||||||
|  | * Bug fixes and stability improvements | ||||||
|  | 
 | ||||||
|  | ## v5.6.1 | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * The app no longer uploads images to Wikidata if one exists already for a given item | ||||||
|  | * File usage displays correctly now | ||||||
|  | * No more infinite circular progress bar on nominating an image for deletion | ||||||
|  | * Enhanced location updates while using GPS | ||||||
|  | * Author/uploader names are now available in Media Details for Commons licensing compliance | ||||||
|  | * Improved usage of popups in Nearby | ||||||
|  | * Bug fixes and stability improvements  | ||||||
|  | 
 | ||||||
|  | ## v5.5.0 | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * Explore images will now be shown based on the map location and not at your current location | ||||||
|  | * Enhanced Wikidata feedback message | ||||||
|  | * Green labels in Explore map will no longer be hidden by other pins thumbnails | ||||||
|  | * Upload wizard's language drop-down now reflects the language used in the pin label | ||||||
|  | * Users can now pick only one image at a time while using the custom selector | ||||||
|  | * Bug fixes and stability improvements  | ||||||
|  | 
 | ||||||
|  | ## v5.4.1 | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * Custom picker now detects images that are already available on Commons | ||||||
|  | * Improve credit line in image list | ||||||
|  | * Show place cards with loaded names only in the Nearby list | ||||||
|  | * Fix the error that occurs while loading images in Explore | ||||||
|  | 
 | ||||||
|  | ## v5.3.0 | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * Enable EmailAuth support | ||||||
|  | * Explore map images no longer show "Unknown" | ||||||
|  | * Fix crash when removing last two images of multiupload | ||||||
|  | * Mark ❌ for closed locations (P3999) in Nearby | ||||||
|  | * Fix two pin labels staying visible at the same time in Explore map | ||||||
|  | * Refactoring and minor UI improvements | ||||||
|  | 
 | ||||||
|  | ## v5.2.0 | ||||||
|  | 
 | ||||||
|  | v5.2.0 boasts several new functionalities like: | ||||||
|  | 
 | ||||||
|  | * A new refresh button lets you quickly reload the Nearby map | ||||||
|  | * Bookmarks now support categories | ||||||
|  | * Improved feedback and consistency in the user interface | ||||||
|  | * Bug fixes and performance improvements | ||||||
|  | 
 | ||||||
|  | ### What's changed | ||||||
|  | * Implement "Refresh" button to clear the cache and reload the Nearby map. | ||||||
|  | * `CommonsApplication` migrate to kotlin & some lint fixes. | ||||||
|  | * Revert back to MainScope for database and UI updates and make database operations thread safe. | ||||||
|  | * Hide edit options for logged-out users in Explore screen. | ||||||
|  | * Introduced a button to delete the current folder in custom selector. | ||||||
|  | * Improve Unique File Name Search. | ||||||
|  | * Migration of several modules from Java to Kotlin. | ||||||
|  | * Fix modification on bottom sheet's data when coming from Nearby Banner and clicked on other pins. | ||||||
|  | * Bug fixes and enhancement of Achievements screen. | ||||||
|  | * Show where file is being used on Commons and other wikis. | ||||||
|  | * Migrate android.media.ExifInterface to androidx.exifinterface.media.ExifInterface as android.media.ExifInterface had security flaws on older devices. | ||||||
|  | * Make dialogs modal and always show the upload icon. | ||||||
|  | * Fix unintentional deletion of subfolders and non-images by custom selector. | ||||||
|  | * Bookmark categories. | ||||||
|  | * Add pull down to refresh in the Contributions screen. | ||||||
|  | * Fix race condition and lag when loading pin details, faster overlay management. | ||||||
|  | * Show cached pins in Nearby even when internet is unavailable | ||||||
|  | 
 | ||||||
|  |  Full changelog with the list of contributors: [`v5.1.2...v5.2.0`](https://github.com/commons-app/apps-android-commons/compare/v5.1.2...v5.2.0). | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| ## v5.1.2 | ## v5.1.2 | ||||||
| 
 | 
 | ||||||
| ### What's changed | ### What's changed | ||||||
|  |  | ||||||
							
								
								
									
										15
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
										
									
									
									
								
							|  | @ -1,6 +1,6 @@ | ||||||
| # Wikimedia Commons Android app | # Wikimedia Commons Android app | ||||||
|  |  | ||||||
| [](https://github.com/commons-app/apps-android-commons/actions?query=branch%3Amaster) | [](https://github.com/commons-app/apps-android-commons/actions?query=branch%3Amain) | ||||||
| [](https://appetize.io/app/8ywtpe9f8tb8h6bey11c92vkcw) | [](https://appetize.io/app/8ywtpe9f8tb8h6bey11c92vkcw) | ||||||
| [](https://codecov.io/gh/commons-app/apps-android-commons) | [](https://codecov.io/gh/commons-app/apps-android-commons) | ||||||
| 
 | 
 | ||||||
|  | @ -29,11 +29,12 @@ Thank you all for your work! | ||||||
| 
 | 
 | ||||||
| | [<img src="https://avatars.githubusercontent.com/u/3611199?v=4" width="100px;"/><br /><sub><b>misaochan</b></sub>](https://github.com/misaochan) | [<img src="https://avatars.githubusercontent.com/u/24829418?v=4" width="100px;"/><br /><sub><b>translatewiki</b></sub>](https://github.com/translatewiki) | [<img src="https://avatars.githubusercontent.com/u/3127881?v=4" width="100px;"/><br /><sub><b>neslihanturan</b></sub>](https://github.com/neslihanturan) | [<img src="https://avatars.githubusercontent.com/u/30430?v=4" width="100px;"/><br /><sub><b>yuvipanda</b></sub>](https://github.com/yuvipanda) | [<img src="https://avatars.githubusercontent.com/u/99590?v=4" width="100px;"/><br /><sub><b>nicolas-raoul</b></sub>](https://github.com/nicolas-raoul) | | | [<img src="https://avatars.githubusercontent.com/u/3611199?v=4" width="100px;"/><br /><sub><b>misaochan</b></sub>](https://github.com/misaochan) | [<img src="https://avatars.githubusercontent.com/u/24829418?v=4" width="100px;"/><br /><sub><b>translatewiki</b></sub>](https://github.com/translatewiki) | [<img src="https://avatars.githubusercontent.com/u/3127881?v=4" width="100px;"/><br /><sub><b>neslihanturan</b></sub>](https://github.com/neslihanturan) | [<img src="https://avatars.githubusercontent.com/u/30430?v=4" width="100px;"/><br /><sub><b>yuvipanda</b></sub>](https://github.com/yuvipanda) | [<img src="https://avatars.githubusercontent.com/u/99590?v=4" width="100px;"/><br /><sub><b>nicolas-raoul</b></sub>](https://github.com/nicolas-raoul) | | ||||||
| | :---: | :---: | :---: | :---: | :---: | | | :---: | :---: | :---: | :---: | :---: | | ||||||
| | [<img src="https://avatars.githubusercontent.com/u/4953590?v=4" width="100px;"/><br /><sub><b>domdomegg</b></sub>](https://github.com/domdomegg) | [<img src="https://avatars.githubusercontent.com/u/3069373?v=4" width="100px;"/><br /><sub><b>maskaravivek</b></sub>](https://github.com/maskaravivek) | [<img src="https://avatars.githubusercontent.com/u/407647?v=4" width="100px;"/><br /><sub><b>psh</b></sub>](https://github.com/psh) | [<img src="https://avatars.githubusercontent.com/u/30932899?v=4" width="100px;"/><br /><sub><b>madhurgupta10</b></sub>](https://github.com/madhurgupta10) | [<img src="https://avatars.githubusercontent.com/u/17375274?v=4" width="100px;"/><br /><sub><b>ashishkumar468</b></sub>](https://github.com/ashishkumar468) | | | [<img src="https://avatars.githubusercontent.com/u/407647?v=4" width="100px;"/><br /><sub><b>psh</b></sub>](https://github.com/psh) | [<img src="https://avatars.githubusercontent.com/u/4953590?v=4" width="100px;"/><br /><sub><b>domdomegg</b></sub>](https://github.com/domdomegg) | [<img src="https://avatars.githubusercontent.com/u/3069373?v=4" width="100px;"/><br /><sub><b>maskaravivek</b></sub>](https://github.com/maskaravivek) | [<img src="https://avatars.githubusercontent.com/u/30932899?v=4" width="100px;"/><br /><sub><b>madhurgupta10</b></sub>](https://github.com/madhurgupta10) | [<img src="https://avatars.githubusercontent.com/u/17375274?v=4" width="100px;"/><br /><sub><b>ashishkumar468</b></sub>](https://github.com/ashishkumar468) | | ||||||
| | [<img src="https://avatars.githubusercontent.com/u/103075?v=4" width="100px;"/><br /><sub><b>bvibber</b></sub>](https://github.com/bvibber) | [<img src="https://avatars.githubusercontent.com/u/10674?v=4" width="100px;"/><br /><sub><b>whym</b></sub>](https://github.com/whym) | [<img src="https://avatars.githubusercontent.com/u/10153800?v=4" width="100px;"/><br /><sub><b>akaita</b></sub>](https://github.com/akaita) | [<img src="https://avatars.githubusercontent.com/u/6900601?v=4" width="100px;"/><br /><sub><b>veyndan</b></sub>](https://github.com/veyndan) | [<img src="https://avatars.githubusercontent.com/u/19607555?v=4" width="100px;"/><br /><sub><b>ujjwalagrawal17</b></sub>](https://github.com/ujjwalagrawal17) | | | [<img src="https://avatars.githubusercontent.com/u/103075?v=4" width="100px;"/><br /><sub><b>bvibber</b></sub>](https://github.com/bvibber) | [<img src="https://avatars.githubusercontent.com/u/10674?v=4" width="100px;"/><br /><sub><b>whym</b></sub>](https://github.com/whym) | [<img src="https://avatars.githubusercontent.com/u/10153800?v=4" width="100px;"/><br /><sub><b>akaita</b></sub>](https://github.com/akaita) | [<img src="https://avatars.githubusercontent.com/u/12448084?v=4" width="100px;"/><br /><sub><b>sivaraam</b></sub>](https://github.com/sivaraam) | [<img src="https://avatars.githubusercontent.com/u/6900601?v=4" width="100px;"/><br /><sub><b>veyndan</b></sub>](https://github.com/veyndan) | | ||||||
| | [<img src="https://avatars.githubusercontent.com/u/3358282?v=4" width="100px;"/><br /><sub><b>macgills</b></sub>](https://github.com/macgills) | [<img src="https://avatars.githubusercontent.com/u/1682214?v=4" width="100px;"/><br /><sub><b>dbrant</b></sub>](https://github.com/dbrant) | [<img src="https://avatars.githubusercontent.com/u/34261945?v=4" width="100px;"/><br /><sub><b>vanshikaarora</b></sub>](https://github.com/vanshikaarora) | [<img src="https://avatars.githubusercontent.com/u/12448084?v=4" width="100px;"/><br /><sub><b>sivaraam</b></sub>](https://github.com/sivaraam) | [<img src="https://avatars.githubusercontent.com/u/71203077?v=4" width="100px;"/><br /><sub><b>Ayan-10</b></sub>](https://github.com/Ayan-10) | | | [<img src="https://avatars.githubusercontent.com/u/19607555?v=4" width="100px;"/><br /><sub><b>ujjwalagrawal17</b></sub>](https://github.com/ujjwalagrawal17) | [<img src="https://avatars.githubusercontent.com/u/3358282?v=4" width="100px;"/><br /><sub><b>macgills</b></sub>](https://github.com/macgills) | [<img src="https://avatars.githubusercontent.com/u/346271?v=4" width="100px;"/><br /><sub><b>amire80</b></sub>](https://github.com/amire80) | [<img src="https://avatars.githubusercontent.com/u/1682214?v=4" width="100px;"/><br /><sub><b>dbrant</b></sub>](https://github.com/dbrant) | [<img src="https://avatars.githubusercontent.com/u/34261945?v=4" width="100px;"/><br /><sub><b>vanshikaarora</b></sub>](https://github.com/vanshikaarora) | | ||||||
| | [<img src="https://avatars.githubusercontent.com/u/126143257?v=4" width="100px;"/><br /><sub><b>shashankiitbhu</b></sub>](https://github.com/shashankiitbhu) | [<img src="https://avatars.githubusercontent.com/u/54663429?v=4" width="100px;"/><br /><sub><b>Pratham2305</b></sub>](https://github.com/Pratham2305) | [<img src="https://avatars.githubusercontent.com/u/1345681?v=4" width="100px;"/><br /><sub><b>sandarumk</b></sub>](https://github.com/sandarumk) | [<img src="https://avatars.githubusercontent.com/u/29161745?v=4" width="100px;"/><br /><sub><b>tanvidadu</b></sub>](https://github.com/tanvidadu) | [<img src="https://avatars.githubusercontent.com/u/39745544?v=4" width="100px;"/><br /><sub><b>cypherop</b></sub>](https://github.com/cypherop) | | | [<img src="https://avatars.githubusercontent.com/u/83745993?v=4" width="100px;"/><br /><sub><b>RitikaPahwa4444</b></sub>](https://github.com/RitikaPahwa4444) | [<img src="https://avatars.githubusercontent.com/u/71203077?v=4" width="100px;"/><br /><sub><b>Ayan-10</b></sub>](https://github.com/Ayan-10) | [<img src="https://avatars.githubusercontent.com/u/101377978?v=4" width="100px;"/><br /><sub><b>rohit9625</b></sub>](https://github.com/rohit9625) | [<img src="https://avatars.githubusercontent.com/u/126143257?v=4" width="100px;"/><br /><sub><b>shashankiitbhu</b></sub>](https://github.com/shashankiitbhu) | [<img src="https://avatars.githubusercontent.com/u/54663429?v=4" width="100px;"/><br /><sub><b>Pratham2305</b></sub>](https://github.com/Pratham2305) | | ||||||
| | [<img src="https://avatars.githubusercontent.com/u/65972015?v=4" width="100px;"/><br /><sub><b>Prince-kushwaha</b></sub>](https://github.com/Prince-kushwaha) | [<img src="https://avatars.githubusercontent.com/u/6953323?v=4" width="100px;"/><br /><sub><b>tobias47n9e</b></sub>](https://github.com/tobias47n9e) | [<img src="https://avatars.githubusercontent.com/u/54016427?v=4" width="100px;"/><br /><sub><b>4D17Y4</b></sub>](https://github.com/4D17Y4) | [<img src="https://avatars.githubusercontent.com/u/25305892?v=4" width="100px;"/><br /><sub><b>hismaeel</b></sub>](https://github.com/hismaeel) | [<img src="https://avatars.githubusercontent.com/u/12574756?v=4" width="100px;"/><br /><sub><b>tshradheya</b></sub>](https://github.com/tshradheya) | | | [<img src="https://avatars.githubusercontent.com/u/111801812?v=4" width="100px;"/><br /><sub><b>parneet-guraya</b></sub>](https://github.com/parneet-guraya) | [<img src="https://avatars.githubusercontent.com/u/1345681?v=4" width="100px;"/><br /><sub><b>sandarumk</b></sub>](https://github.com/sandarumk) | [<img src="https://avatars.githubusercontent.com/u/29161745?v=4" width="100px;"/><br /><sub><b>tanvidadu</b></sub>](https://github.com/tanvidadu) | [<img src="https://avatars.githubusercontent.com/u/39745544?v=4" width="100px;"/><br /><sub><b>cypherop</b></sub>](https://github.com/cypherop) | [<img src="https://avatars.githubusercontent.com/u/65972015?v=4" width="100px;"/><br /><sub><b>Prince-kushwaha</b></sub>](https://github.com/Prince-kushwaha) | | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| .. and [many more](https://github.com/commons-app/apps-android-commons/graphs/contributors). | .. and [many more](https://github.com/commons-app/apps-android-commons/graphs/contributors). | ||||||
|  | @ -45,7 +46,7 @@ This software is open source, licensed under the [Apache License 2.0][10]. | ||||||
| 
 | 
 | ||||||
| [1]: https://play.google.com/store/apps/details?id=fr.free.nrw.commons | [1]: https://play.google.com/store/apps/details?id=fr.free.nrw.commons | ||||||
| [2]: https://commons-app.github.io/ | [2]: https://commons-app.github.io/ | ||||||
| [3]: https://github.com/commons-app/apps-android-commons/issues | [3]: https://github.com/commons-app/apps-android-commons/issues?q=is%3Aopen+is%3Aissue+no%3Aassignee+-label%3Adebated+label%3Abug+-label%3A%22low+priority%22+-label%3Aupstream | ||||||
| 
 | 
 | ||||||
| [4]: https://github.com/commons-app/commons-app-documentation/blob/master/android/README.md#-android-documentation | [4]: https://github.com/commons-app/commons-app-documentation/blob/master/android/README.md#-android-documentation | ||||||
| [5]: https://github.com/commons-app/commons-app-documentation/blob/master/android/README.md#-user-documentation | [5]: https://github.com/commons-app/commons-app-documentation/blob/master/android/README.md#-user-documentation | ||||||
|  |  | ||||||
							
								
								
									
										419
									
								
								app/build.gradle
									
										
									
									
									
								
							
							
						
						
									
										419
									
								
								app/build.gradle
									
										
									
									
									
								
							|  | @ -1,419 +0,0 @@ | ||||||
| plugins { |  | ||||||
|     id 'com.github.triplet.play' version '2.7.2' apply false |  | ||||||
| } |  | ||||||
| apply from: '../gitutils.gradle' |  | ||||||
| apply plugin: 'com.android.application' |  | ||||||
| apply plugin: 'kotlin-android' |  | ||||||
| apply plugin: 'kotlin-kapt' |  | ||||||
| apply plugin: 'kotlin-parcelize' |  | ||||||
| apply from: "$rootDir/jacoco.gradle" |  | ||||||
| 
 |  | ||||||
| def isRunningOnTravisAndIsNotPRBuild = System.getenv("CI") == "true" && file('../play.p12').exists() |  | ||||||
| 
 |  | ||||||
| if (isRunningOnTravisAndIsNotPRBuild) { |  | ||||||
|     apply plugin: 'com.github.triplet.play' |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| dependencies { |  | ||||||
| 
 |  | ||||||
|     // Utils |  | ||||||
|     implementation 'in.yuvi:http.fluent:1.3' |  | ||||||
|     implementation 'com.google.code.gson:gson:2.8.5' |  | ||||||
|     implementation ("com.squareup.okhttp3:okhttp:$OKHTTP_VERSION!!"){ |  | ||||||
|         // Forcing dependency versions using force = true on a first-level dependency has been deprecated. |  | ||||||
|         //  Ref: https://docs.gradle.org/7.5/userguide/upgrading_version_5.html#forced_dependencies |  | ||||||
|         //force = true //API 19 support |  | ||||||
|     } |  | ||||||
|     implementation 'com.squareup.retrofit2:retrofit:2.8.1' |  | ||||||
|     implementation "com.squareup.retrofit2:converter-gson:2.8.1" |  | ||||||
|     implementation "com.squareup.retrofit2:adapter-rxjava2:2.8.1" |  | ||||||
|     implementation 'com.squareup.okio:okio:2.2.2' |  | ||||||
|     implementation 'io.reactivex.rxjava2:rxandroid:2.1.0' |  | ||||||
|     implementation 'io.reactivex.rxjava2:rxjava:2.2.3' |  | ||||||
|     implementation 'com.jakewharton.rxbinding2:rxbinding:2.1.1' |  | ||||||
|     implementation 'com.jakewharton.rxbinding3:rxbinding-appcompat:3.0.0' |  | ||||||
|     implementation 'com.jakewharton.rxbinding2:rxbinding-support-v4:2.1.1' |  | ||||||
|     implementation 'com.jakewharton.rxbinding2:rxbinding-appcompat-v7:2.1.1' |  | ||||||
|     implementation 'com.jakewharton.rxbinding2:rxbinding-design:2.1.1' |  | ||||||
|     implementation 'com.facebook.fresco:fresco:1.13.0' |  | ||||||
|     implementation 'org.apache.commons:commons-lang3:3.8.1' |  | ||||||
| 
 |  | ||||||
|     // UI |  | ||||||
|     implementation 'fr.avianey.com.viewpagerindicator:library:2.4.1.1@aar' |  | ||||||
|     implementation 'com.github.chrisbanes:PhotoView:2.0.0' |  | ||||||
|     implementation 'com.github.pedrovgs:renderers:3.3.3' |  | ||||||
|     implementation "org.maplibre.gl:android-sdk:$MAPLIBRE_VERSION" |  | ||||||
|     implementation 'org.maplibre.gl:android-plugin-scalebar-v9:1.0.0' |  | ||||||
| 
 |  | ||||||
|     implementation 'com.jakewharton.timber:timber:4.7.1' |  | ||||||
|     implementation 'com.github.deano2390:MaterialShowcaseView:1.2.0' |  | ||||||
|     implementation "com.google.android.material:material:1.9.0" |  | ||||||
|     implementation 'com.karumi:dexter:5.0.0' |  | ||||||
|     implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0' |  | ||||||
| 
 |  | ||||||
|     // Jetpack Compose |  | ||||||
|     def composeBom = platform('androidx.compose:compose-bom:2024.08.00') |  | ||||||
| 
 |  | ||||||
|     implementation "androidx.activity:activity-compose:1.9.1" |  | ||||||
|     implementation "androidx.lifecycle:lifecycle-runtime-ktx:2.8.4" |  | ||||||
|     implementation (composeBom) |  | ||||||
|     implementation "androidx.compose.runtime:runtime" |  | ||||||
|     implementation "androidx.compose.ui:ui" |  | ||||||
|     implementation "androidx.compose.ui:ui-graphics" |  | ||||||
|     implementation "androidx.compose.ui:ui-tooling" |  | ||||||
|     implementation "androidx.compose.foundation:foundation" |  | ||||||
|     implementation "androidx.compose.foundation:foundation-layout" |  | ||||||
|     implementation "androidx.compose.material3:material3" |  | ||||||
|     androidTestImplementation(composeBom) |  | ||||||
| 
 |  | ||||||
|     implementation "com.hannesdorfmann:adapterdelegates4-kotlin-dsl-viewbinding:$ADAPTER_DELEGATES_VERSION" |  | ||||||
|     implementation "com.hannesdorfmann:adapterdelegates4-pagination:$ADAPTER_DELEGATES_VERSION" |  | ||||||
|     implementation "androidx.paging:paging-runtime-ktx:$PAGING_VERSION" |  | ||||||
|     testImplementation "androidx.paging:paging-common-ktx:$PAGING_VERSION" |  | ||||||
|     implementation "androidx.paging:paging-rxjava2-ktx:$PAGING_VERSION" |  | ||||||
|     implementation "androidx.recyclerview:recyclerview:1.2.0-alpha02" |  | ||||||
|     implementation "com.squareup.okhttp3:okhttp-ws:$OKHTTP_VERSION" |  | ||||||
| 
 |  | ||||||
|     // Logging |  | ||||||
|     implementation 'ch.acra:acra-dialog:5.8.4' |  | ||||||
|     implementation 'ch.acra:acra-mail:5.8.4' |  | ||||||
|     implementation 'org.slf4j:slf4j-api:1.7.25' |  | ||||||
|     api('com.github.tony19:logback-android-classic:1.1.1-6') { |  | ||||||
|         exclude group: 'com.google.android', module: 'android' |  | ||||||
|     } |  | ||||||
|     implementation "com.squareup.okhttp3:logging-interceptor:$OKHTTP_VERSION" |  | ||||||
| 
 |  | ||||||
|     // Dependency injector |  | ||||||
|     implementation "com.google.dagger:dagger-android:$DAGGER_VERSION" |  | ||||||
|     implementation "com.google.dagger:dagger-android-support:$DAGGER_VERSION" |  | ||||||
|     kapt "com.google.dagger:dagger-android-processor:$DAGGER_VERSION" |  | ||||||
|     kapt "com.google.dagger:dagger-compiler:$DAGGER_VERSION" |  | ||||||
|     annotationProcessor "com.google.dagger:dagger-android-processor:$DAGGER_VERSION" |  | ||||||
| 
 |  | ||||||
|     implementation "org.jetbrains.kotlin:kotlin-reflect:$KOTLIN_VERSION" |  | ||||||
| 
 |  | ||||||
|     //Mocking |  | ||||||
|     testImplementation("io.mockk:mockk:1.13.4") |  | ||||||
|     testImplementation 'com.nhaarman.mockitokotlin2:mockito-kotlin:2.2.0' |  | ||||||
|     testImplementation 'org.mockito:mockito-inline:5.2.0' |  | ||||||
|     testImplementation 'org.mockito:mockito-core:5.6.0' |  | ||||||
|     testImplementation "org.powermock:powermock-module-junit4:2.0.9" |  | ||||||
|     testImplementation "org.powermock:powermock-api-mockito2:2.0.9" |  | ||||||
| 
 |  | ||||||
|     // Unit testing |  | ||||||
|     testImplementation 'junit:junit:4.13.2' |  | ||||||
|     testImplementation 'org.robolectric:robolectric:4.11.1' |  | ||||||
|     testImplementation 'androidx.test:core:1.5.0' |  | ||||||
|     testImplementation "androidx.test:runner:1.5.2" |  | ||||||
|     testImplementation 'androidx.test.ext:junit:1.1.5' |  | ||||||
|     testImplementation "androidx.test:rules:1.5.0" |  | ||||||
|     testImplementation "com.squareup.okhttp3:mockwebserver:$OKHTTP_VERSION" |  | ||||||
|     testImplementation "com.jraska.livedata:testing-ktx:1.2.0" |  | ||||||
|     testImplementation "androidx.arch.core:core-testing:2.2.0" |  | ||||||
|     testImplementation "org.junit.jupiter:junit-jupiter-api:5.10.0" |  | ||||||
|     testRuntimeOnly "org.junit.jupiter:junit-jupiter-engine:5.10.0" |  | ||||||
|     testImplementation 'com.facebook.soloader:soloader:0.10.5' |  | ||||||
|     testImplementation "org.jetbrains.kotlinx:kotlinx-coroutines-test:1.7.3" |  | ||||||
|     debugImplementation("androidx.fragment:fragment-testing:1.6.2") |  | ||||||
|     testImplementation "commons-io:commons-io:2.6" |  | ||||||
| 
 |  | ||||||
|     // Android testing |  | ||||||
|     androidTestImplementation 'androidx.test.espresso:espresso-core:3.5.0-alpha04' |  | ||||||
|     androidTestImplementation 'androidx.test.espresso:espresso-intents:3.4.0' |  | ||||||
|     androidTestImplementation 'androidx.test.espresso:espresso-contrib:3.5.0-alpha04' |  | ||||||
|     androidTestImplementation 'androidx.test:runner:1.4.0' |  | ||||||
|     androidTestImplementation 'androidx.test:rules:1.4.1-alpha04' |  | ||||||
|     androidTestImplementation 'androidx.test:core:1.4.0' |  | ||||||
|     androidTestImplementation 'androidx.test.ext:junit:1.1.3' |  | ||||||
|     androidTestImplementation 'androidx.annotation:annotation:1.3.0' |  | ||||||
|     androidTestImplementation 'com.squareup.okhttp3:mockwebserver:4.8.0' |  | ||||||
|     androidTestImplementation "androidx.test.uiautomator:uiautomator:2.2.0" |  | ||||||
|     androidTestUtil 'androidx.test:orchestrator:1.4.1' |  | ||||||
| 
 |  | ||||||
|     // Debugging |  | ||||||
|     debugImplementation "com.squareup.leakcanary:leakcanary-android:$LEAK_CANARY_VERSION" |  | ||||||
| 
 |  | ||||||
|     // Support libraries |  | ||||||
|     implementation "com.google.android.material:material:1.1.0-alpha04" |  | ||||||
|     implementation "androidx.browser:browser:1.3.0" |  | ||||||
|     implementation "androidx.cardview:cardview:1.0.0" |  | ||||||
|     implementation 'androidx.constraintlayout:constraintlayout:1.1.3' |  | ||||||
|     implementation 'androidx.exifinterface:exifinterface:1.3.7' |  | ||||||
|     implementation "androidx.core:core-ktx:$CORE_KTX_VERSION" |  | ||||||
|     implementation 'com.simplecityapps:recyclerview-fastscroll:2.0.1' |  | ||||||
| 
 |  | ||||||
|     //swipe_layout |  | ||||||
|     implementation 'com.daimajia.swipelayout:library:1.2.0@aar' |  | ||||||
| 
 |  | ||||||
|     //Room |  | ||||||
|     implementation "androidx.room:room-runtime:$ROOM_VERSION" |  | ||||||
|     implementation "androidx.room:room-ktx:$ROOM_VERSION" |  | ||||||
|     implementation "androidx.room:room-rxjava2:$ROOM_VERSION" |  | ||||||
|     kapt "androidx.room:room-compiler:$ROOM_VERSION" |  | ||||||
|     // For Kotlin use kapt instead of annotationProcessor |  | ||||||
|     testImplementation "androidx.arch.core:core-testing:2.1.0" |  | ||||||
| 
 |  | ||||||
|     // Pref |  | ||||||
|     // Java language implementation |  | ||||||
|     implementation "androidx.preference:preference:$PREFERENCE_VERSION" |  | ||||||
|     // Kotlin |  | ||||||
|     implementation "androidx.preference:preference-ktx:$PREFERENCE_VERSION" |  | ||||||
|     //Android Media |  | ||||||
|     implementation 'com.github.juanitobananas:AndroidMediaUtil:v1.0-1' |  | ||||||
| 
 |  | ||||||
|     implementation "androidx.multidex:multidex:$MULTIDEX_VERSION" |  | ||||||
| 
 |  | ||||||
|     def work_version = "2.8.1" |  | ||||||
|     // Kotlin + coroutines |  | ||||||
|     implementation "androidx.work:work-runtime-ktx:$work_version" |  | ||||||
|     implementation("androidx.work:work-runtime:$work_version") |  | ||||||
|     testImplementation "androidx.work:work-testing:$work_version" |  | ||||||
| 
 |  | ||||||
|     //Glide |  | ||||||
|     implementation 'com.github.bumptech.glide:glide:4.12.0' |  | ||||||
|     annotationProcessor 'com.github.bumptech.glide:compiler:4.12.0' |  | ||||||
|     kaptTest "androidx.databinding:databinding-compiler:8.0.2" |  | ||||||
|     kaptAndroidTest "androidx.databinding:databinding-compiler:8.0.2" |  | ||||||
| 
 |  | ||||||
|     implementation("io.github.coordinates2country:coordinates2country-android:1.8") {  exclude group: 'com.google.android', module: 'android' } |  | ||||||
| 
 |  | ||||||
|     //OSMDroid |  | ||||||
|     implementation ("org.osmdroid:osmdroid-android:$OSMDROID_VERSION") |  | ||||||
|     constraints { |  | ||||||
|         implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk7:1.8.0") { |  | ||||||
|             because("kotlin-stdlib-jdk7 is now a part of kotlin-stdlib") |  | ||||||
|         } |  | ||||||
|         implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.8.0") { |  | ||||||
|             because("kotlin-stdlib-jdk8 is now a part of kotlin-stdlib") |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| task disableAnimations(type: Exec) { |  | ||||||
|     def adb = "$System.env.ANDROID_HOME/platform-tools/adb" |  | ||||||
|     commandLine "$adb", 'shell', 'settings', 'put', 'global', 'window_animation_scale', '0' |  | ||||||
|     commandLine "$adb", 'shell', 'settings', 'put', 'global', 'transition_animation_scale', '0' |  | ||||||
|     commandLine "$adb", 'shell', 'settings', 'put', 'global', 'animator_duration_scale', '0' |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| project.gradle.taskGraph.whenReady { |  | ||||||
|     connectedBetaDebugAndroidTest.dependsOn disableAnimations |  | ||||||
|     connectedProdDebugAndroidTest.dependsOn disableAnimations |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| android { |  | ||||||
|     compileSdkVersion 34 |  | ||||||
| 
 |  | ||||||
|     defaultConfig { |  | ||||||
|         //applicationId 'fr.free.nrw.commons' |  | ||||||
| 
 |  | ||||||
|         versionCode 1043 |  | ||||||
|         versionName '5.1.2' |  | ||||||
|         setProperty("archivesBaseName", "app-commons-v$versionName-" + getBranchName()) |  | ||||||
| 
 |  | ||||||
|         minSdkVersion 21 |  | ||||||
|         targetSdkVersion 34 |  | ||||||
|         testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" |  | ||||||
|         testInstrumentationRunnerArguments clearPackageData: 'true' |  | ||||||
| 
 |  | ||||||
|         multiDexEnabled true |  | ||||||
| 
 |  | ||||||
|         testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" |  | ||||||
| 
 |  | ||||||
|         vectorDrawables.useSupportLibrary = true |  | ||||||
|     } |  | ||||||
|     packagingOptions { |  | ||||||
|         jniLibs { |  | ||||||
|             excludes += ['META-INF/androidx.*'] |  | ||||||
|         } |  | ||||||
|         resources { |  | ||||||
|             excludes += ['META-INF/androidx.*', 'META-INF/proguard/androidx-annotations.pro', '/META-INF/LICENSE.md', '/META-INF/LICENSE-notice.md'] |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     testOptions { |  | ||||||
|         animationsDisabled true |  | ||||||
| 
 |  | ||||||
|         unitTests { |  | ||||||
|             returnDefaultValues = true |  | ||||||
|             includeAndroidResources = true |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         unitTests.all { |  | ||||||
|             jvmArgs '-noverify' |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     sourceSets { |  | ||||||
|         // use kotlin only in tests (for now) |  | ||||||
|         test.java.srcDirs += 'src/test/kotlin' |  | ||||||
| 
 |  | ||||||
|         // use main assets and resources in test |  | ||||||
|         test.assets.srcDirs += 'src/main/assets' |  | ||||||
|         test.resources.srcDirs += 'src/main/resoures' |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     signingConfigs { |  | ||||||
|         release |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     buildTypes { |  | ||||||
|         release { |  | ||||||
|             minifyEnabled true |  | ||||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt' |  | ||||||
|             testProguardFile 'test-proguard-rules.txt' |  | ||||||
|             signingConfig signingConfigs.debug |  | ||||||
|             if (isRunningOnTravisAndIsNotPRBuild) { |  | ||||||
|                 signingConfig signingConfigs.release |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         debug { |  | ||||||
|             minifyEnabled false |  | ||||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt' |  | ||||||
|             testProguardFile 'test-proguard-rules.txt' |  | ||||||
|             versionNameSuffix "-debug-" + getBranchName() |  | ||||||
|             enableUnitTestCoverage true |  | ||||||
|             enableAndroidTestCoverage true |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (isRunningOnTravisAndIsNotPRBuild) { |  | ||||||
|         // configure keystore based on env vars in Travis for automated alpha builds |  | ||||||
|         signingConfigs.release.storeFile = file("../nr-commons.keystore") |  | ||||||
|         signingConfigs.release.storePassword = System.getenv("keystore_password") |  | ||||||
|         signingConfigs.release.keyAlias = System.getenv("key_alias") |  | ||||||
|         signingConfigs.release.keyPassword = System.getenv("key_password") |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     configurations.all { |  | ||||||
|         resolutionStrategy.force 'androidx.annotation:annotation:1.1.0' |  | ||||||
|         resolutionStrategy.force 'com.jakewharton.timber:timber:4.7.1' |  | ||||||
|         resolutionStrategy.force 'androidx.fragment:fragment:1.3.6' |  | ||||||
|         exclude module: 'okhttp-ws' |  | ||||||
|     } |  | ||||||
|     flavorDimensions 'tier' |  | ||||||
|     productFlavors { |  | ||||||
|         prod { |  | ||||||
| 
 |  | ||||||
|             applicationId 'fr.free.nrw.commons' |  | ||||||
| 
 |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_API_POTD", "\"https://commons.wikimedia.org/w/api.php?action=featuredfeed&feed=potd&feedformat=rss&language=en\"" |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_API_HOST", "\"https://commons.wikimedia.org/w/api.php\"" |  | ||||||
|             buildConfigField "String", "WIKIDATA_API_HOST", "\"https://www.wikidata.org/w/api.php\"" |  | ||||||
|             buildConfigField "String", "WIKIDATA_URL", "\"https://www.wikidata.org\"" |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_FORGE_API_HOST", "\"https://tools.wmflabs.org/\"" |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_CAMPAIGNS_URL", "\"https://raw.githubusercontent.com/commons-app/campaigns/master/campaigns.json\"" |  | ||||||
|             buildConfigField "String", "IMAGE_URL_BASE", "\"https://upload.wikimedia.org/wikipedia/commons\"" |  | ||||||
|             buildConfigField "String", "HOME_URL", "\"https://commons.wikimedia.org/wiki/\"" |  | ||||||
|             buildConfigField "String", "COMMONS_URL", "\"https://commons.wikimedia.org\"" |  | ||||||
|             buildConfigField "String", "WIKIDATA_URL", "\"https://www.wikidata.org\"" |  | ||||||
|             buildConfigField "String", "MOBILE_HOME_URL", "\"https://commons.m.wikimedia.org/wiki/\"" |  | ||||||
|             buildConfigField "String", "SIGNUP_LANDING_URL", "\"https://commons.m.wikimedia.org/w/index.php?title=Special:CreateAccount&returnto=Main+Page&returntoquery=welcome%3Dyes\"" |  | ||||||
|             buildConfigField "String", "SIGNUP_SUCCESS_REDIRECTION_URL", "\"https://commons.m.wikimedia.org/w/index.php?title=Main_Page&welcome=yes\"" |  | ||||||
|             buildConfigField "String", "FORGOT_PASSWORD_URL", "\"https://commons.wikimedia.org/wiki/Special:PasswordReset\"" |  | ||||||
|             buildConfigField "String", "PRIVACY_POLICY_URL", "\"https://github.com/commons-app/commons-app-documentation/blob/master/android/Privacy-policy.md\"" |  | ||||||
|             buildConfigField "String", "ACCOUNT_TYPE", "\"fr.free.nrw.commons\"" |  | ||||||
|             buildConfigField "String", "CONTRIBUTION_AUTHORITY", "\"fr.free.nrw.commons.contributions.contentprovider\"" |  | ||||||
|             buildConfigField "String", "MODIFICATION_AUTHORITY", "\"fr.free.nrw.commons.modifications.contentprovider\"" |  | ||||||
|             buildConfigField "String", "CATEGORY_AUTHORITY", "\"fr.free.nrw.commons.categories.contentprovider\"" |  | ||||||
|             buildConfigField "String", "RECENT_SEARCH_AUTHORITY", "\"fr.free.nrw.commons.explore.recentsearches.contentprovider\"" |  | ||||||
|             buildConfigField "String", "RECENT_LANGUAGE_AUTHORITY", "\"fr.free.nrw.commons.recentlanguages.contentprovider\"" |  | ||||||
|             buildConfigField "String", "BOOKMARK_AUTHORITY", "\"fr.free.nrw.commons.bookmarks.contentprovider\"" |  | ||||||
|             buildConfigField "String", "BOOKMARK_LOCATIONS_AUTHORITY", "\"fr.free.nrw.commons.bookmarks.locations.contentprovider\"" |  | ||||||
|             buildConfigField "String", "BOOKMARK_ITEMS_AUTHORITY", "\"fr.free.nrw.commons.bookmarks.items.contentprovider\"" |  | ||||||
|             buildConfigField "String", "COMMIT_SHA", "\"" + getBuildVersion().toString() + "\"" |  | ||||||
|             buildConfigField "String", "TEST_USERNAME", "\"" + getTestUserName() + "\"" |  | ||||||
|             buildConfigField "String", "TEST_PASSWORD", "\"" + getTestPassword() + "\"" |  | ||||||
|             buildConfigField "String", "DEPICTS_PROPERTY", "\"P180\"" |  | ||||||
|             dimension 'tier' |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         beta { |  | ||||||
|             applicationId 'fr.free.nrw.commons.beta' |  | ||||||
| 
 |  | ||||||
|             // What values do we need to hit the BETA versions of the site / api ? |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_API_POTD", "\"https://commons.wikimedia.org/w/api.php?action=featuredfeed&feed=potd&feedformat=rss&language=en\"" |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_API_HOST", "\"https://commons.wikimedia.beta.wmflabs.org/w/api.php\"" |  | ||||||
|             buildConfigField "String", "WIKIDATA_API_HOST", "\"https://www.wikidata.org/w/api.php\"" |  | ||||||
|             buildConfigField "String", "WIKIDATA_URL", "\"https://www.wikidata.org\"" |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_FORGE_API_HOST", "\"https://tools.wmflabs.org/\"" |  | ||||||
|             buildConfigField "String", "WIKIMEDIA_CAMPAIGNS_URL", "\"https://raw.githubusercontent.com/commons-app/campaigns/master/campaigns_beta_active.json\"" |  | ||||||
|             buildConfigField "String", "IMAGE_URL_BASE", "\"https://upload.beta.wmflabs.org/wikipedia/commons\"" |  | ||||||
|             buildConfigField "String", "HOME_URL", "\"https://commons.wikimedia.beta.wmflabs.org/wiki/\"" |  | ||||||
|             buildConfigField "String", "COMMONS_URL", "\"https://commons.wikimedia.beta.wmflabs.org\"" |  | ||||||
|             buildConfigField "String", "WIKIDATA_URL", "\"https://www.wikidata.org\"" |  | ||||||
|             buildConfigField "String", "MOBILE_HOME_URL", "\"https://commons.m.wikimedia.beta.wmflabs.org/wiki/\"" |  | ||||||
|             buildConfigField "String", "SIGNUP_LANDING_URL", "\"https://commons.m.wikimedia.beta.wmflabs.org/w/index.php?title=Special:CreateAccount&returnto=Main+Page&returntoquery=welcome%3Dyes\"" |  | ||||||
|             buildConfigField "String", "SIGNUP_SUCCESS_REDIRECTION_URL", "\"https://commons.m.wikimedia.beta.wmflabs.org/w/index.php?title=Main_Page&welcome=yes\"" |  | ||||||
|             buildConfigField "String", "FORGOT_PASSWORD_URL", "\"https://commons.wikimedia.beta.wmflabs.org/wiki/Special:PasswordReset\"" |  | ||||||
|             buildConfigField "String", "PRIVACY_POLICY_URL", "\"https://github.com/commons-app/commons-app-documentation/blob/master/android/Privacy-policy.md\"" |  | ||||||
|             buildConfigField "String", "ACCOUNT_TYPE", "\"fr.free.nrw.commons.beta\"" |  | ||||||
|             buildConfigField "String", "CONTRIBUTION_AUTHORITY", "\"fr.free.nrw.commons.beta.contributions.contentprovider\"" |  | ||||||
|             buildConfigField "String", "MODIFICATION_AUTHORITY", "\"fr.free.nrw.commons.beta.modifications.contentprovider\"" |  | ||||||
|             buildConfigField "String", "CATEGORY_AUTHORITY", "\"fr.free.nrw.commons.beta.categories.contentprovider\"" |  | ||||||
|             buildConfigField "String", "RECENT_SEARCH_AUTHORITY", "\"fr.free.nrw.commons.beta.explore.recentsearches.contentprovider\"" |  | ||||||
|             buildConfigField "String", "RECENT_LANGUAGE_AUTHORITY", "\"fr.free.nrw.commons.beta.recentlanguages.contentprovider\"" |  | ||||||
|             buildConfigField "String", "BOOKMARK_AUTHORITY", "\"fr.free.nrw.commons.beta.bookmarks.contentprovider\"" |  | ||||||
|             buildConfigField "String", "BOOKMARK_LOCATIONS_AUTHORITY", "\"fr.free.nrw.commons.beta.bookmarks.locations.contentprovider\"" |  | ||||||
|             buildConfigField "String", "BOOKMARK_ITEMS_AUTHORITY", "\"fr.free.nrw.commons.beta.bookmarks.items.contentprovider\"" |  | ||||||
|             buildConfigField "String", "COMMIT_SHA", "\"" + getBuildVersion().toString() + "\"" |  | ||||||
|             buildConfigField "String", "TEST_USERNAME", "\"" + getTestUserName() + "\"" |  | ||||||
|             buildConfigField "String", "TEST_PASSWORD", "\"" + getTestPassword() + "\"" |  | ||||||
|             buildConfigField "String", "DEPICTS_PROPERTY", "\"P245962\"" |  | ||||||
|             dimension 'tier' |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     compileOptions { |  | ||||||
|         sourceCompatibility JavaVersion.VERSION_11 |  | ||||||
|         targetCompatibility JavaVersion.VERSION_11 |  | ||||||
|     } |  | ||||||
|     kotlinOptions { |  | ||||||
|         jvmTarget = "11" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     buildToolsVersion buildToolsVersion |  | ||||||
| 
 |  | ||||||
|     buildFeatures { |  | ||||||
|         viewBinding true |  | ||||||
|         compose true |  | ||||||
|     } |  | ||||||
|     composeOptions { |  | ||||||
|         kotlinCompilerExtensionVersion '1.3.2' |  | ||||||
|     } |  | ||||||
|     namespace 'fr.free.nrw.commons' |  | ||||||
|     lint { |  | ||||||
|         abortOnError false |  | ||||||
|         disable 'MissingTranslation', 'ExtraTranslation' |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| String getTestUserName() { |  | ||||||
|     def propFile = rootProject.file("./local.properties") |  | ||||||
|     def properties = new Properties() |  | ||||||
|     properties.load(new FileInputStream(propFile)) |  | ||||||
|     return properties['TEST_USER_NAME'] |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| String getTestPassword() { |  | ||||||
|     def propFile = rootProject.file("./local.properties") |  | ||||||
|     def properties = new Properties() |  | ||||||
|     properties.load(new FileInputStream(propFile)) |  | ||||||
|     return properties['TEST_USER_PASSWORD'] |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| if (isRunningOnTravisAndIsNotPRBuild) { |  | ||||||
|     play { |  | ||||||
|         track = "alpha" |  | ||||||
|         userFraction = 1 |  | ||||||
|         serviceAccountEmail = System.getenv("SERVICE_ACCOUNT_NAME") |  | ||||||
|         serviceAccountCredentials = file("../play.p12") |  | ||||||
| 
 |  | ||||||
|         resolutionStrategy = "auto" |  | ||||||
|         outputProcessor { // this: ApkVariantOutput |  | ||||||
|             versionNameOverride = "$versionNameOverride.$versionCode" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										447
									
								
								app/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										447
									
								
								app/build.gradle.kts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,447 @@ | ||||||
|  | import java.util.Properties | ||||||
|  | import java.io.ByteArrayOutputStream | ||||||
|  | 
 | ||||||
|  | plugins { | ||||||
|  |     alias(libs.plugins.android.application) | ||||||
|  |     alias(libs.plugins.jetbrains.kotlin.android) | ||||||
|  |     alias(libs.plugins.kotlin.kapt) | ||||||
|  |     alias(libs.plugins.kotlin.parcelize) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | apply(from = "$rootDir/jacoco.gradle") | ||||||
|  | 
 | ||||||
|  | val isRunningOnTravisAndIsNotPRBuild = System.getenv("CI") == "true" && file("../play.p12").exists() | ||||||
|  | 
 | ||||||
|  | if (isRunningOnTravisAndIsNotPRBuild) { | ||||||
|  |     apply(plugin = "com.github.triplet.play") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | android { | ||||||
|  |     namespace = "fr.free.nrw.commons" | ||||||
|  |     compileSdk = 35 | ||||||
|  | 
 | ||||||
|  |     defaultConfig { | ||||||
|  |         applicationId = "fr.free.nrw.commons" | ||||||
|  |         minSdk = 21 | ||||||
|  |         targetSdk = 35 | ||||||
|  |         versionCode = 1059 | ||||||
|  |         versionName = "6.1.0" | ||||||
|  | 
 | ||||||
|  |         setProperty("archivesBaseName", "app-commons-v$versionName-" + getBranchName()) | ||||||
|  |         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||||
|  |         testInstrumentationRunnerArguments["clearPackageData"] = "true" | ||||||
|  | 
 | ||||||
|  |         multiDexEnabled = true | ||||||
|  | 
 | ||||||
|  |         vectorDrawables { | ||||||
|  |             useSupportLibrary = true | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     sourceSets { | ||||||
|  |         getByName("test") { | ||||||
|  |             // Use kotlin only in tests (for now) | ||||||
|  |             java.srcDirs("src/test/kotlin") | ||||||
|  | 
 | ||||||
|  |             // Use main assets and resources in test | ||||||
|  |             assets.srcDirs("src/main/assets") | ||||||
|  |             resources.srcDirs("src/main/resources") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     signingConfigs { | ||||||
|  |         create("release") { | ||||||
|  |             // Configure keystore based on env vars in Travis for automated alpha builds | ||||||
|  |             if(isRunningOnTravisAndIsNotPRBuild) { | ||||||
|  |                 storeFile = file("../nr-commons.keystore") | ||||||
|  |                 storePassword = System.getenv("keystore_password") | ||||||
|  |                 keyAlias = System.getenv("key_alias") | ||||||
|  |                 keyPassword = System.getenv("key_password") | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     buildTypes { | ||||||
|  |         release { | ||||||
|  |             isMinifyEnabled = true | ||||||
|  |             proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.txt") | ||||||
|  |             testProguardFile("test-proguard-rules.txt") | ||||||
|  | 
 | ||||||
|  |             signingConfig = signingConfigs.getByName("debug") | ||||||
|  |             if (isRunningOnTravisAndIsNotPRBuild) { | ||||||
|  |                 signingConfig = signingConfigs.getByName("release") | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         debug { | ||||||
|  |             isMinifyEnabled = false | ||||||
|  |             proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.txt") | ||||||
|  |             testProguardFile("test-proguard-rules.txt") | ||||||
|  | 
 | ||||||
|  |             versionNameSuffix = "-debug-" + getBranchName() | ||||||
|  |             enableUnitTestCoverage = true | ||||||
|  |             enableAndroidTestCoverage = true | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     configurations.all { | ||||||
|  |         resolutionStrategy { | ||||||
|  |             force("androidx.annotation:annotation:1.1.0") | ||||||
|  |             force("com.jakewharton.timber:timber:4.7.1") | ||||||
|  |             force("androidx.fragment:fragment:1.3.6") | ||||||
|  |         } | ||||||
|  |         exclude(module = "okhttp-ws") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     flavorDimensions += "tier" | ||||||
|  |     productFlavors { | ||||||
|  |         create("prod") { | ||||||
|  |             dimension = "tier" | ||||||
|  |             applicationId = "fr.free.nrw.commons" | ||||||
|  | 
 | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_API_POTD", "\"https://commons.wikimedia.org/w/api.php?action=featuredfeed&feed=potd&feedformat=rss&language=en\"") | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_API_HOST", "\"https://commons.wikimedia.org/w/api.php\"") | ||||||
|  |             buildConfigField("String", "WIKIDATA_API_HOST", "\"https://www.wikidata.org/w/api.php\"") | ||||||
|  |             buildConfigField("String", "WIKIDATA_URL", "\"https://www.wikidata.org\"") | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_FORGE_API_HOST", "\"https://tools.wmflabs.org/\"") | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_CAMPAIGNS_URL", "\"https://raw.githubusercontent.com/commons-app/campaigns/master/campaigns.json\"") | ||||||
|  |             buildConfigField("String", "IMAGE_URL_BASE", "\"https://upload.wikimedia.org/wikipedia/commons\"") | ||||||
|  |             buildConfigField("String", "HOME_URL", "\"https://commons.wikimedia.org/wiki/\"") | ||||||
|  |             buildConfigField("String", "COMMONS_URL", "\"https://commons.wikimedia.org\"") | ||||||
|  |             buildConfigField("String", "WIKIDATA_URL", "\"https://www.wikidata.org\"") | ||||||
|  |             buildConfigField("String", "MOBILE_HOME_URL", "\"https://commons.m.wikimedia.org/wiki/\"") | ||||||
|  |             buildConfigField("String", "MOBILE_META_URL", "\"https://meta.m.wikimedia.org/wiki/\"") | ||||||
|  |             buildConfigField("String", "SIGNUP_LANDING_URL", "\"https://commons.m.wikimedia.org/w/index.php?title=Special:CreateAccount&returnto=Main+Page&returntoquery=welcome%3Dyes\"") | ||||||
|  |             buildConfigField("String", "SIGNUP_SUCCESS_REDIRECTION_URL", "\"https://commons.m.wikimedia.org/w/index.php?title=Main_Page&welcome=yes\"") | ||||||
|  |             buildConfigField("String", "FORGOT_PASSWORD_URL", "\"https://commons.wikimedia.org/wiki/Special:PasswordReset\"") | ||||||
|  |             buildConfigField("String", "PRIVACY_POLICY_URL", "\"https://github.com/commons-app/commons-app-documentation/blob/master/android/Privacy-policy.md\"") | ||||||
|  |             buildConfigField("String", "FILE_USAGES_BASE_URL", "\"https://commons.wikimedia.org/w/api.php?action=query&format=json&formatversion=2\"") | ||||||
|  |             buildConfigField("String", "ACCOUNT_TYPE", "\"fr.free.nrw.commons\"") | ||||||
|  |             buildConfigField("String", "CONTRIBUTION_AUTHORITY", "\"fr.free.nrw.commons.contributions.contentprovider\"") | ||||||
|  |             buildConfigField("String", "MODIFICATION_AUTHORITY", "\"fr.free.nrw.commons.modifications.contentprovider\"") | ||||||
|  |             buildConfigField("String", "CATEGORY_AUTHORITY", "\"fr.free.nrw.commons.categories.contentprovider\"") | ||||||
|  |             buildConfigField("String", "RECENT_SEARCH_AUTHORITY", "\"fr.free.nrw.commons.explore.recentsearches.contentprovider\"") | ||||||
|  |             buildConfigField("String", "RECENT_LANGUAGE_AUTHORITY", "\"fr.free.nrw.commons.recentlanguages.contentprovider\"") | ||||||
|  |             buildConfigField("String", "BOOKMARK_AUTHORITY", "\"fr.free.nrw.commons.bookmarks.contentprovider\"") | ||||||
|  |             buildConfigField("String", "BOOKMARK_LOCATIONS_AUTHORITY", "\"fr.free.nrw.commons.bookmarks.locations.contentprovider\"") | ||||||
|  |             buildConfigField("String", "BOOKMARK_ITEMS_AUTHORITY", "\"fr.free.nrw.commons.bookmarks.items.contentprovider\"") | ||||||
|  |             buildConfigField("String", "COMMIT_SHA", "\"" + getBuildVersion().toString() + "\"") | ||||||
|  |             buildConfigField("String", "TEST_USERNAME", "\"" + getTestUserName() + "\"") | ||||||
|  |             buildConfigField("String", "TEST_PASSWORD", "\"" + getTestPassword() + "\"") | ||||||
|  |             buildConfigField("String", "DEPICTS_PROPERTY", "\"P180\"") | ||||||
|  |             buildConfigField("String", "CREATOR_PROPERTY", "\"P170\"") | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         create("beta") { | ||||||
|  |             dimension = "tier" | ||||||
|  |             applicationId = "fr.free.nrw.commons.beta" | ||||||
|  | 
 | ||||||
|  |             // What values do we need to hit the BETA versions of the site / api ? | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_API_POTD", "\"https://commons.wikimedia.org/w/api.php?action=featuredfeed&feed=potd&feedformat=rss&language=en\"") | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_API_HOST", "\"https://commons.wikimedia.beta.wmflabs.org/w/api.php\"") | ||||||
|  |             buildConfigField("String", "WIKIDATA_API_HOST", "\"https://www.wikidata.org/w/api.php\"") | ||||||
|  |             buildConfigField("String", "WIKIDATA_URL", "\"https://www.wikidata.org\"") | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_FORGE_API_HOST", "\"https://tools.wmflabs.org/\"") | ||||||
|  |             buildConfigField("String", "WIKIMEDIA_CAMPAIGNS_URL", "\"https://raw.githubusercontent.com/commons-app/campaigns/master/campaigns_beta_active.json\"") | ||||||
|  |             buildConfigField("String", "IMAGE_URL_BASE", "\"https://upload.beta.wmflabs.org/wikipedia/commons\"") | ||||||
|  |             buildConfigField("String", "HOME_URL", "\"https://commons.wikimedia.beta.wmflabs.org/wiki/\"") | ||||||
|  |             buildConfigField("String", "COMMONS_URL", "\"https://commons.wikimedia.beta.wmflabs.org\"") | ||||||
|  |             buildConfigField("String", "WIKIDATA_URL", "\"https://www.wikidata.org\"") | ||||||
|  |             buildConfigField("String", "MOBILE_HOME_URL", "\"https://commons.m.wikimedia.beta.wmflabs.org/wiki/\"") | ||||||
|  |             buildConfigField("String", "MOBILE_META_URL", "\"https://meta.m.wikimedia.beta.wmflabs.org/wiki/\"") | ||||||
|  |             buildConfigField("String", "SIGNUP_LANDING_URL", "\"https://commons.m.wikimedia.beta.wmflabs.org/w/index.php?title=Special:CreateAccount&returnto=Main+Page&returntoquery=welcome%3Dyes\"") | ||||||
|  |             buildConfigField("String", "SIGNUP_SUCCESS_REDIRECTION_URL", "\"https://commons.m.wikimedia.beta.wmflabs.org/w/index.php?title=Main_Page&welcome=yes\"") | ||||||
|  |             buildConfigField("String", "FORGOT_PASSWORD_URL", "\"https://commons.wikimedia.beta.wmflabs.org/wiki/Special:PasswordReset\"") | ||||||
|  |             buildConfigField("String", "PRIVACY_POLICY_URL", "\"https://github.com/commons-app/commons-app-documentation/blob/master/android/Privacy-policy.md\"") | ||||||
|  |             buildConfigField("String", "FILE_USAGES_BASE_URL", "\"https://commons.wikimedia.org/w/api.php?action=query&format=json&formatversion=2\"") | ||||||
|  |             buildConfigField("String", "ACCOUNT_TYPE", "\"fr.free.nrw.commons.beta\"") | ||||||
|  |             buildConfigField("String", "CONTRIBUTION_AUTHORITY", "\"fr.free.nrw.commons.beta.contributions.contentprovider\"") | ||||||
|  |             buildConfigField("String", "MODIFICATION_AUTHORITY", "\"fr.free.nrw.commons.beta.modifications.contentprovider\"") | ||||||
|  |             buildConfigField("String", "CATEGORY_AUTHORITY", "\"fr.free.nrw.commons.beta.categories.contentprovider\"") | ||||||
|  |             buildConfigField("String", "RECENT_SEARCH_AUTHORITY", "\"fr.free.nrw.commons.beta.explore.recentsearches.contentprovider\"") | ||||||
|  |             buildConfigField("String", "RECENT_LANGUAGE_AUTHORITY", "\"fr.free.nrw.commons.beta.recentlanguages.contentprovider\"") | ||||||
|  |             buildConfigField("String", "BOOKMARK_AUTHORITY", "\"fr.free.nrw.commons.beta.bookmarks.contentprovider\"") | ||||||
|  |             buildConfigField("String", "BOOKMARK_LOCATIONS_AUTHORITY", "\"fr.free.nrw.commons.beta.bookmarks.locations.contentprovider\"") | ||||||
|  |             buildConfigField("String", "BOOKMARK_ITEMS_AUTHORITY", "\"fr.free.nrw.commons.beta.bookmarks.items.contentprovider\"") | ||||||
|  |             buildConfigField("String", "COMMIT_SHA", "\"" + getBuildVersion().toString() + "\"") | ||||||
|  |             buildConfigField("String", "TEST_USERNAME", "\"" + getTestUserName() + "\"") | ||||||
|  |             buildConfigField("String", "TEST_PASSWORD", "\"" + getTestPassword() + "\"") | ||||||
|  |             buildConfigField("String", "DEPICTS_PROPERTY", "\"P245962\"") | ||||||
|  |             buildConfigField("String", "CREATOR_PROPERTY", "\"P253075\"") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     compileOptions { | ||||||
|  |         sourceCompatibility = JavaVersion.VERSION_17 | ||||||
|  |         targetCompatibility = JavaVersion.VERSION_17 | ||||||
|  |     } | ||||||
|  |     kotlinOptions { | ||||||
|  |         jvmTarget = "17" | ||||||
|  |     } | ||||||
|  |     buildFeatures { | ||||||
|  |         buildConfig = true | ||||||
|  |         viewBinding = true | ||||||
|  |         compose = true | ||||||
|  |     } | ||||||
|  |     buildToolsVersion = buildToolsVersion | ||||||
|  |     composeOptions { | ||||||
|  |         kotlinCompilerExtensionVersion = "1.5.8" | ||||||
|  |     } | ||||||
|  |     packaging { | ||||||
|  |         jniLibs { | ||||||
|  |             excludes += listOf("META-INF/androidx.*") | ||||||
|  |         } | ||||||
|  |         resources { | ||||||
|  |             excludes += listOf( | ||||||
|  |                 "META-INF/androidx.*", | ||||||
|  |                 "META-INF/proguard/androidx-annotations.pro", | ||||||
|  |                 "/META-INF/LICENSE.md", | ||||||
|  |                 "/META-INF/LICENSE-notice.md" | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     testOptions { | ||||||
|  |         animationsDisabled = true | ||||||
|  |         unitTests { | ||||||
|  |             isReturnDefaultValues = true | ||||||
|  |             isIncludeAndroidResources = true | ||||||
|  |         } | ||||||
|  |         unitTests.all { | ||||||
|  |             it.jvmArgs("-noverify") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     lint { | ||||||
|  |         abortOnError = false | ||||||
|  |         disable += listOf("MissingTranslation", "ExtraTranslation") | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | dependencies { | ||||||
|  |     // Utils | ||||||
|  |     implementation(libs.gson) | ||||||
|  |     implementation(libs.okhttp) | ||||||
|  |     implementation(libs.retrofit) | ||||||
|  |     implementation(libs.retrofit.converter.gson) | ||||||
|  |     implementation(libs.retrofit.adapter.rxjava) | ||||||
|  |     implementation(libs.rxandroid) | ||||||
|  |     implementation(libs.rxjava) | ||||||
|  |     implementation(libs.rxbinding) | ||||||
|  |     implementation(libs.rxbinding.appcompat) | ||||||
|  |     implementation(libs.facebook.fresco) | ||||||
|  |     implementation(libs.facebook.fresco.middleware) | ||||||
|  |     implementation(libs.apache.commons.lang3) | ||||||
|  | 
 | ||||||
|  |     // UI | ||||||
|  |     implementation("${libs.viewpagerindicator.library.get()}@aar") | ||||||
|  |     implementation(libs.photoview) | ||||||
|  |     implementation(libs.android.sdk) | ||||||
|  |     implementation(libs.android.plugin.scalebar) | ||||||
|  | 
 | ||||||
|  |     implementation(libs.timber) | ||||||
|  |     implementation(libs.android.material) | ||||||
|  |     implementation(libs.dexter) | ||||||
|  | 
 | ||||||
|  |     // Jetpack Compose | ||||||
|  |     implementation(libs.androidx.core.ktx) | ||||||
|  |     implementation(libs.androidx.lifecycle.runtime.ktx) | ||||||
|  |     implementation(libs.androidx.activity.compose) | ||||||
|  |     implementation(platform(libs.androidx.compose.bom)) | ||||||
|  |     implementation(libs.androidx.compose.runtime) | ||||||
|  |     implementation(libs.androidx.ui) | ||||||
|  |     implementation(libs.androidx.ui.graphics) | ||||||
|  |     implementation(libs.androidx.ui.tooling.preview) | ||||||
|  |     implementation(libs.androidx.ui.viewbinding) | ||||||
|  |     implementation(libs.androidx.material3) | ||||||
|  |     implementation(libs.androidx.foundation) | ||||||
|  |     implementation(libs.androidx.foundation.layout) | ||||||
|  |     androidTestImplementation(platform(libs.androidx.compose.bom)) | ||||||
|  |     androidTestImplementation(libs.androidx.ui.test.junit4) | ||||||
|  |     debugImplementation(libs.androidx.ui.tooling) | ||||||
|  |     debugImplementation(libs.androidx.ui.test.manifest) | ||||||
|  | 
 | ||||||
|  |     implementation(libs.adapterdelegates4.kotlin.dsl.viewbinding) | ||||||
|  |     implementation(libs.adapterdelegates4.pagination) | ||||||
|  |     implementation(libs.androidx.paging.runtime.ktx) | ||||||
|  |     testImplementation(libs.androidx.paging.common.ktx) | ||||||
|  |     implementation(libs.androidx.paging.rxjava2.ktx) | ||||||
|  |     implementation(libs.androidx.recyclerview) | ||||||
|  | 
 | ||||||
|  |     // Logging | ||||||
|  |     implementation(libs.acra.dialog) | ||||||
|  |     implementation(libs.acra.mail) | ||||||
|  |     implementation(libs.slf4j.api) | ||||||
|  |     implementation(libs.logback.android.classic) { | ||||||
|  |         exclude(group = "com.google.android", module = "android") | ||||||
|  |     } | ||||||
|  |     implementation(libs.logging.interceptor) | ||||||
|  | 
 | ||||||
|  |     // Dependency injector | ||||||
|  |     implementation(libs.dagger.android) | ||||||
|  |     implementation(libs.dagger.android.support) | ||||||
|  |     kapt(libs.dagger.android.processor) | ||||||
|  |     kapt(libs.dagger.compiler) | ||||||
|  |     annotationProcessor(libs.dagger.android.processor) | ||||||
|  | 
 | ||||||
|  |     implementation(libs.kotlin.reflect) | ||||||
|  | 
 | ||||||
|  |     //Mocking | ||||||
|  |     testImplementation(libs.mockito.kotlin) | ||||||
|  |     testImplementation(libs.mockito.core) | ||||||
|  |     testImplementation(libs.powermock.module.junit) | ||||||
|  |     testImplementation(libs.powermock.api.mockito) | ||||||
|  |     testImplementation(libs.mockk) | ||||||
|  | 
 | ||||||
|  |     // Unit testing | ||||||
|  |     testImplementation(libs.junit) | ||||||
|  |     testImplementation(libs.robolectric) | ||||||
|  |     testImplementation(libs.androidx.test.core) | ||||||
|  |     testImplementation(libs.androidx.runner) | ||||||
|  |     testImplementation(libs.androidx.test.ext.junit) | ||||||
|  |     testImplementation(libs.androidx.test.rules) | ||||||
|  |     testImplementation(libs.mockwebserver) | ||||||
|  |     testImplementation(libs.livedata.testing.ktx) | ||||||
|  |     testImplementation(libs.androidx.core.testing) | ||||||
|  |     testImplementation(libs.junit.jupiter.api) | ||||||
|  |     testRuntimeOnly(libs.junit.jupiter.engine) | ||||||
|  |     testImplementation(libs.soloader) | ||||||
|  |     testImplementation(libs.kotlinx.coroutines.test) | ||||||
|  |     debugImplementation(libs.androidx.fragment.testing) | ||||||
|  |     testImplementation(libs.commons.io) | ||||||
|  | 
 | ||||||
|  |     // Android testing | ||||||
|  |     androidTestImplementation(libs.androidx.espresso.core) | ||||||
|  |     androidTestImplementation(libs.androidx.espresso.intents) | ||||||
|  |     androidTestImplementation(libs.androidx.espresso.contrib) | ||||||
|  |     androidTestImplementation(libs.androidx.runner) | ||||||
|  |     androidTestImplementation(libs.androidx.test.rules) | ||||||
|  |     androidTestImplementation(libs.androidx.test.core) | ||||||
|  |     androidTestImplementation(libs.androidx.test.ext.junit) | ||||||
|  |     androidTestImplementation(libs.androidx.annotation) | ||||||
|  |     androidTestImplementation(libs.mockwebserver) | ||||||
|  |     androidTestImplementation(libs.androidx.uiautomator) | ||||||
|  | 
 | ||||||
|  |     // Debugging | ||||||
|  |     debugImplementation(libs.leakcanary.android) | ||||||
|  | 
 | ||||||
|  |     // Support libraries | ||||||
|  |     implementation(libs.androidx.browser) | ||||||
|  |     implementation(libs.androidx.cardview) | ||||||
|  |     implementation(libs.androidx.constraintlayout) | ||||||
|  |     implementation(libs.androidx.exifinterface) | ||||||
|  |     implementation(libs.recyclerview.fastscroll) | ||||||
|  | 
 | ||||||
|  |     //swipe_layout | ||||||
|  |     implementation(libs.swipelayout.library) | ||||||
|  | 
 | ||||||
|  |     //Room | ||||||
|  |     implementation(libs.androidx.room.runtime) | ||||||
|  |     implementation(libs.androidx.room.ktx) | ||||||
|  |     implementation(libs.androidx.room.rxjava) | ||||||
|  |     kapt(libs.androidx.room.compiler) | ||||||
|  | 
 | ||||||
|  |     // Preferences | ||||||
|  |     implementation(libs.androidx.preference) | ||||||
|  |     implementation(libs.androidx.preference.ktx) | ||||||
|  | 
 | ||||||
|  |     //Android Media | ||||||
|  |     implementation(libs.juanitobananas.androidDmediaUtil) | ||||||
|  |     implementation(libs.androidx.multidex) | ||||||
|  | 
 | ||||||
|  |     // Kotlin + coroutines | ||||||
|  |     implementation(libs.androidx.work.runtime.ktx) | ||||||
|  |     implementation(libs.androidx.work.runtime) | ||||||
|  |     implementation(libs.kotlinx.coroutines.rx2) | ||||||
|  |     testImplementation(libs.androidx.work.testing) | ||||||
|  | 
 | ||||||
|  |     //Glide | ||||||
|  |     implementation(libs.glide) | ||||||
|  |     annotationProcessor(libs.glide.compiler) | ||||||
|  |     kaptTest(libs.androidx.databinding.compiler) | ||||||
|  |     kaptAndroidTest(libs.androidx.databinding.compiler) | ||||||
|  | 
 | ||||||
|  |     implementation(libs.coordinates2country.android) { | ||||||
|  |         exclude(group = "com.google.android", module = "android") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     //OSMDroid | ||||||
|  |     implementation(libs.osmdroid.android) | ||||||
|  |     constraints { | ||||||
|  |         implementation(libs.kotlin.stdlib.jdk7) { | ||||||
|  |             because("kotlin-stdlib-jdk7 is now a part of kotlin-stdlib") | ||||||
|  |         } | ||||||
|  |         implementation(libs.kotlin.stdlib.jdk8) { | ||||||
|  |             because("kotlin-stdlib-jdk8 is now a part of kotlin-stdlib") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | tasks.register<Exec>("disableAnimations") { | ||||||
|  |     val adb = "${System.getenv("ANDROID_HOME")}/platform-tools/adb" | ||||||
|  |     commandLine(adb, "shell", "settings", "put", "global", "window_animation_scale", "0") | ||||||
|  |     commandLine(adb, "shell", "settings", "put", "global", "transition_animation_scale", "0") | ||||||
|  |     commandLine(adb, "shell", "settings", "put", "global", "animator_duration_scale", "0") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | project.gradle.taskGraph.whenReady { | ||||||
|  |     val connectedBetaDebugAndroidTest = tasks.named("connectedBetaDebugAndroidTest") | ||||||
|  |     val connectedProdDebugAndroidTest = tasks.named("connectedProdDebugAndroidTest") | ||||||
|  | 
 | ||||||
|  |     connectedBetaDebugAndroidTest.configure { | ||||||
|  |         dependsOn("disableAnimations") | ||||||
|  |     } | ||||||
|  |     connectedProdDebugAndroidTest.configure { | ||||||
|  |         dependsOn("disableAnimations") | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fun getTestUserName(): String? { | ||||||
|  |     val propFile = rootProject.file("./local.properties") | ||||||
|  |     val properties = Properties() | ||||||
|  |     propFile.inputStream().use { properties.load(it) } | ||||||
|  |     return properties.getProperty("TEST_USER_NAME") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fun getTestPassword(): String? { | ||||||
|  |     val propFile = rootProject.file("./local.properties") | ||||||
|  |     val properties = Properties() | ||||||
|  |     propFile.inputStream().use { properties.load(it) } | ||||||
|  |     return properties.getProperty("TEST_USER_PASSWORD") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | if (isRunningOnTravisAndIsNotPRBuild) { | ||||||
|  |     configure<com.github.triplet.gradle.play.PlayPublisherExtension> { | ||||||
|  |         track = "alpha" | ||||||
|  |         userFraction = 1.0 | ||||||
|  |         serviceAccountEmail = System.getenv("SERVICE_ACCOUNT_NAME") | ||||||
|  |         serviceAccountCredentials = file("../play.p12") | ||||||
|  | 
 | ||||||
|  |         resolutionStrategy = "auto" | ||||||
|  |         outputProcessor { // this: ApkVariantOutput | ||||||
|  |             versionNameOverride = "$versionNameOverride.$versionCode" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fun getBuildVersion(): String? { | ||||||
|  |     return try { | ||||||
|  |         val stdout = ByteArrayOutputStream() | ||||||
|  |         exec { | ||||||
|  |             commandLine("git", "rev-parse", "--short", "HEAD") | ||||||
|  |             standardOutput = stdout | ||||||
|  |         } | ||||||
|  |         stdout.toString().trim() | ||||||
|  |     } catch (e: Exception) { | ||||||
|  |         null | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fun getBranchName(): String? { | ||||||
|  |     return try { | ||||||
|  |         val stdout = ByteArrayOutputStream() | ||||||
|  |         exec { | ||||||
|  |             commandLine("git", "rev-parse", "--abbrev-ref", "HEAD") | ||||||
|  |             standardOutput = stdout | ||||||
|  |         } | ||||||
|  |         stdout.toString().trim() | ||||||
|  |     } catch (e: Exception) { | ||||||
|  |         null | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -66,6 +66,9 @@ | ||||||
| # Application classes that will be serialized/deserialized over Gson | # Application classes that will be serialized/deserialized over Gson | ||||||
| -keep class com.google.gson.examples.android.model.** { *; } | -keep class com.google.gson.examples.android.model.** { *; } | ||||||
| 
 | 
 | ||||||
|  | # Prevent R8 from obfuscating project classes used by Gson for parsing | ||||||
|  | -keep class fr.free.nrw.commons.fileusages.** { *; } | ||||||
|  | 
 | ||||||
| # Prevent proguard from stripping interface information from TypeAdapterFactory, | # Prevent proguard from stripping interface information from TypeAdapterFactory, | ||||||
| # JsonSerializer, JsonDeserializer instances (so they can be used in @JsonAdapter) | # JsonSerializer, JsonDeserializer instances (so they can be used in @JsonAdapter) | ||||||
| -keep class * implements com.google.gson.TypeAdapterFactory | -keep class * implements com.google.gson.TypeAdapterFactory | ||||||
|  |  | ||||||
|  | @ -105,7 +105,7 @@ class AboutActivityTest { | ||||||
|     fun testLaunchTranslate() { |     fun testLaunchTranslate() { | ||||||
|         Espresso.onView(ViewMatchers.withId(R.id.about_translate)).perform(ViewActions.click()) |         Espresso.onView(ViewMatchers.withId(R.id.about_translate)).perform(ViewActions.click()) | ||||||
|         Espresso.onView(ViewMatchers.withId(android.R.id.button1)).perform(ViewActions.click()) |         Espresso.onView(ViewMatchers.withId(android.R.id.button1)).perform(ViewActions.click()) | ||||||
|         val langCode = CommonsApplication.instance.languageLookUpTable!!.codes[0] |         val langCode = CommonsApplication.instance.languageLookUpTable!!.getCodes()[0] | ||||||
|         Intents.intended( |         Intents.intended( | ||||||
|             CoreMatchers.allOf( |             CoreMatchers.allOf( | ||||||
|                 IntentMatchers.hasAction(Intent.ACTION_VIEW), |                 IntentMatchers.hasAction(Intent.ACTION_VIEW), | ||||||
|  |  | ||||||
|  | @ -18,7 +18,7 @@ import androidx.test.platform.app.InstrumentationRegistry | ||||||
| import androidx.test.rule.ActivityTestRule | import androidx.test.rule.ActivityTestRule | ||||||
| import androidx.test.rule.GrantPermissionRule | import androidx.test.rule.GrantPermissionRule | ||||||
| import androidx.test.uiautomator.UiDevice | import androidx.test.uiautomator.UiDevice | ||||||
| import fr.free.nrw.commons.LocationPicker.LocationPickerActivity | import fr.free.nrw.commons.locationpicker.LocationPickerActivity | ||||||
| import fr.free.nrw.commons.UITestHelper.Companion.childAtPosition | import fr.free.nrw.commons.UITestHelper.Companion.childAtPosition | ||||||
| import fr.free.nrw.commons.auth.LoginActivity | import fr.free.nrw.commons.auth.LoginActivity | ||||||
| import org.hamcrest.CoreMatchers | import org.hamcrest.CoreMatchers | ||||||
|  | @ -49,7 +49,7 @@ class UploadCancelledTest { | ||||||
|     fun setup() { |     fun setup() { | ||||||
|         try { |         try { | ||||||
|             Intents.init() |             Intents.init() | ||||||
|         } catch (ex: IllegalStateException) { |         } catch (_: IllegalStateException) { | ||||||
|         } |         } | ||||||
|         device.unfreezeRotation() |         device.unfreezeRotation() | ||||||
|         device.setOrientationNatural() |         device.setOrientationNatural() | ||||||
|  | @ -65,7 +65,7 @@ class UploadCancelledTest { | ||||||
|     fun teardown() { |     fun teardown() { | ||||||
|         try { |         try { | ||||||
|             Intents.release() |             Intents.release() | ||||||
|         } catch (ex: IllegalStateException) { |         } catch (_: IllegalStateException) { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -71,7 +71,7 @@ class UploadTest { | ||||||
|     fun setup() { |     fun setup() { | ||||||
|         try { |         try { | ||||||
|             Intents.init() |             Intents.init() | ||||||
|         } catch (ex: IllegalStateException) { |         } catch (_: IllegalStateException) { | ||||||
|         } |         } | ||||||
|         UITestHelper.loginUser() |         UITestHelper.loginUser() | ||||||
|         UITestHelper.skipWelcome() |         UITestHelper.skipWelcome() | ||||||
|  |  | ||||||
|  | @ -89,7 +89,7 @@ class ContributionsListFragmentUnitTests { | ||||||
|         Shadows.shadowOf(Looper.getMainLooper()).idle() |         Shadows.shadowOf(Looper.getMainLooper()).idle() | ||||||
|         fragment.rvContributionsList = mock() |         fragment.rvContributionsList = mock() | ||||||
|         fragment.scrollToTop() |         fragment.scrollToTop() | ||||||
|         verify(fragment.rvContributionsList).smoothScrollToPosition(0) |         verify(fragment.rvContributionsList)?.smoothScrollToPosition(0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Test |     @Test | ||||||
|  | @ -17,7 +17,7 @@ class PasteSensitiveTextInputEditTextTest { | ||||||
|     @Before |     @Before | ||||||
|     fun setup() { |     fun setup() { | ||||||
|         context = ApplicationProvider.getApplicationContext() |         context = ApplicationProvider.getApplicationContext() | ||||||
|         textView = PasteSensitiveTextInputEditText(context) |         textView = PasteSensitiveTextInputEditText(context!!) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // this test has no real value, just % for test code coverage |     // this test has no real value, just % for test code coverage | ||||||
|  |  | ||||||
|  | @ -55,6 +55,9 @@ | ||||||
|     android:theme="@style/LightAppTheme" |     android:theme="@style/LightAppTheme" | ||||||
|     tools:ignore="GoogleAppIndexingWarning" |     tools:ignore="GoogleAppIndexingWarning" | ||||||
|     tools:replace="android:appComponentFactory"> |     tools:replace="android:appComponentFactory"> | ||||||
|  |     <activity | ||||||
|  |       android:name=".activity.SingleWebViewActivity" | ||||||
|  |       android:exported="false" /> | ||||||
|     <activity |     <activity | ||||||
|       android:name=".nearby.WikidataFeedback" |       android:name=".nearby.WikidataFeedback" | ||||||
|       android:exported="false" /> |       android:exported="false" /> | ||||||
|  | @ -81,6 +84,7 @@ | ||||||
|       android:parentActivityName=".customselector.ui.selector.CustomSelectorActivity" /> |       android:parentActivityName=".customselector.ui.selector.CustomSelectorActivity" /> | ||||||
|     <activity |     <activity | ||||||
|       android:name=".auth.LoginActivity" |       android:name=".auth.LoginActivity" | ||||||
|  |       android:windowSoftInputMode="adjustPan" | ||||||
|       android:exported="true"> |       android:exported="true"> | ||||||
|       <intent-filter> |       <intent-filter> | ||||||
|         <category android:name="android.intent.category.LAUNCHER" /> |         <category android:name="android.intent.category.LAUNCHER" /> | ||||||
|  | @ -99,7 +103,7 @@ | ||||||
|       android:exported="true" |       android:exported="true" | ||||||
|       android:hardwareAccelerated="false" |       android:hardwareAccelerated="false" | ||||||
|       android:icon="@mipmap/ic_launcher" |       android:icon="@mipmap/ic_launcher" | ||||||
|       android:windowSoftInputMode="adjustResize"> |       android:windowSoftInputMode="adjustPan"> | ||||||
|       <intent-filter android:label="@string/intent_share_upload_label"> |       <intent-filter android:label="@string/intent_share_upload_label"> | ||||||
|         <action android:name="android.intent.action.SEND" /> |         <action android:name="android.intent.action.SEND" /> | ||||||
| 
 | 
 | ||||||
|  | @ -171,7 +175,7 @@ | ||||||
|       android:name=".review.ReviewActivity" |       android:name=".review.ReviewActivity" | ||||||
|       android:label="@string/title_activity_review" /> |       android:label="@string/title_activity_review" /> | ||||||
|     <activity |     <activity | ||||||
|       android:name=".LocationPicker.LocationPickerActivity" |       android:name=".locationpicker.LocationPickerActivity" | ||||||
|       android:label="Location Picker" /> |       android:label="Location Picker" /> | ||||||
| 
 | 
 | ||||||
|     <service |     <service | ||||||
|  | @ -228,12 +232,6 @@ | ||||||
|       android:exported="false" |       android:exported="false" | ||||||
|       android:label="@string/provider_bookmarks" |       android:label="@string/provider_bookmarks" | ||||||
|       android:syncable="false" /> |       android:syncable="false" /> | ||||||
|     <provider |  | ||||||
|       android:name=".bookmarks.locations.BookmarkLocationsContentProvider" |  | ||||||
|       android:authorities="${applicationId}.bookmarks.locations.contentprovider" |  | ||||||
|       android:exported="false" |  | ||||||
|       android:label="@string/provider_bookmarks_location" |  | ||||||
|       android:syncable="false" /> |  | ||||||
|     <provider |     <provider | ||||||
|       android:name=".bookmarks.items.BookmarkItemsContentProvider" |       android:name=".bookmarks.items.BookmarkItemsContentProvider" | ||||||
|       android:authorities="${applicationId}.bookmarks.items.contentprovider" |       android:authorities="${applicationId}.bookmarks.items.contentprovider" | ||||||
|  | @ -258,4 +256,4 @@ | ||||||
|       android:required="false" /> |       android:required="false" /> | ||||||
|   </application> |   </application> | ||||||
| 
 | 
 | ||||||
| </manifest> | </manifest> | ||||||
|  |  | ||||||
|  | @ -1,187 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.view.Menu; |  | ||||||
| import android.view.MenuInflater; |  | ||||||
| import android.view.MenuItem; |  | ||||||
| import android.view.View; |  | ||||||
| import android.widget.ArrayAdapter; |  | ||||||
| import android.widget.LinearLayout; |  | ||||||
| import android.widget.Spinner; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import fr.free.nrw.commons.databinding.ActivityAboutBinding; |  | ||||||
| import fr.free.nrw.commons.theme.BaseActivity; |  | ||||||
| import fr.free.nrw.commons.utils.ConfigUtils; |  | ||||||
| import fr.free.nrw.commons.utils.DialogUtil; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Represents about screen of this app |  | ||||||
|  */ |  | ||||||
| public class AboutActivity extends BaseActivity { |  | ||||||
| 
 |  | ||||||
|     /* |  | ||||||
|       This View Binding class is auto-generated for each xml file. The format is usually the name |  | ||||||
|       of the file with PascalCasing (The underscore characters will be ignored). |  | ||||||
|       More information is available at https://developer.android.com/topic/libraries/view-binding |  | ||||||
|      */ |  | ||||||
|     private ActivityAboutBinding binding; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method helps in the creation About screen |  | ||||||
|      * |  | ||||||
|      * @param savedInstanceState Data bundle |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     @SuppressLint("StringFormatInvalid") |  | ||||||
|     public void onCreate(Bundle savedInstanceState) { |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
| 
 |  | ||||||
|         /* |  | ||||||
|           Instead of just setting the view with the xml file. We need to use View Binding class. |  | ||||||
|          */ |  | ||||||
|         binding = ActivityAboutBinding.inflate(getLayoutInflater()); |  | ||||||
|         final View view = binding.getRoot(); |  | ||||||
|         setContentView(view); |  | ||||||
| 
 |  | ||||||
|         setSupportActionBar(binding.toolbarBinding.toolbar); |  | ||||||
|         getSupportActionBar().setDisplayHomeAsUpEnabled(true); |  | ||||||
|         final String aboutText = getString(R.string.about_license); |  | ||||||
|         /* |  | ||||||
|           We can then access all the views by just using the id names like this. |  | ||||||
|           camelCasing is used with underscore characters being ignored. |  | ||||||
|          */ |  | ||||||
|         binding.aboutLicense.setHtmlText(aboutText); |  | ||||||
| 
 |  | ||||||
|         @SuppressLint("StringFormatMatches") |  | ||||||
|         String improveText = String.format(getString(R.string.about_improve), Urls.NEW_ISSUE_URL); |  | ||||||
|         binding.aboutImprove.setHtmlText(improveText); |  | ||||||
|         binding.aboutVersion.setText(ConfigUtils.getVersionNameWithSha(getApplicationContext())); |  | ||||||
| 
 |  | ||||||
|         Utils.setUnderlinedText(binding.aboutFaq, R.string.about_faq, getApplicationContext()); |  | ||||||
|         Utils.setUnderlinedText(binding.aboutRateUs, R.string.about_rate_us, getApplicationContext()); |  | ||||||
|         Utils.setUnderlinedText(binding.aboutUserGuide, R.string.user_guide, getApplicationContext()); |  | ||||||
|         Utils.setUnderlinedText(binding.aboutPrivacyPolicy, R.string.about_privacy_policy, getApplicationContext()); |  | ||||||
|         Utils.setUnderlinedText(binding.aboutTranslate, R.string.about_translate, getApplicationContext()); |  | ||||||
|         Utils.setUnderlinedText(binding.aboutCredits, R.string.about_credits, getApplicationContext()); |  | ||||||
| 
 |  | ||||||
|         /* |  | ||||||
|           To set listeners, we can create a separate method and use lambda syntax. |  | ||||||
|         */ |  | ||||||
|         binding.facebookLaunchIcon.setOnClickListener(this::launchFacebook); |  | ||||||
|         binding.githubLaunchIcon.setOnClickListener(this::launchGithub); |  | ||||||
|         binding.websiteLaunchIcon.setOnClickListener(this::launchWebsite); |  | ||||||
|         binding.aboutRateUs.setOnClickListener(this::launchRatings); |  | ||||||
|         binding.aboutCredits.setOnClickListener(this::launchCredits); |  | ||||||
|         binding.aboutPrivacyPolicy.setOnClickListener(this::launchPrivacyPolicy); |  | ||||||
|         binding.aboutUserGuide.setOnClickListener(this::launchUserGuide); |  | ||||||
|         binding.aboutFaq.setOnClickListener(this::launchFrequentlyAskedQuesions); |  | ||||||
|         binding.aboutTranslate.setOnClickListener(this::launchTranslate); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onSupportNavigateUp() { |  | ||||||
|         onBackPressed(); |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchFacebook(View view) { |  | ||||||
|         Intent intent; |  | ||||||
|         try { |  | ||||||
|             intent = new Intent(Intent.ACTION_VIEW, Uri.parse(Urls.FACEBOOK_APP_URL)); |  | ||||||
|             intent.setPackage(Urls.FACEBOOK_PACKAGE_NAME); |  | ||||||
|             startActivity(intent); |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             Utils.handleWebUrl(this, Uri.parse(Urls.FACEBOOK_WEB_URL)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchGithub(View view) { |  | ||||||
|         Intent intent; |  | ||||||
|         try { |  | ||||||
|             intent = new Intent(Intent.ACTION_VIEW, Uri.parse(Urls.GITHUB_REPO_URL)); |  | ||||||
|             intent.setPackage(Urls.GITHUB_PACKAGE_NAME); |  | ||||||
|             startActivity(intent); |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             Utils.handleWebUrl(this, Uri.parse(Urls.GITHUB_REPO_URL)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchWebsite(View view) { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(Urls.WEBSITE_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchRatings(View view){ |  | ||||||
|         Utils.rateApp(this); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchCredits(View view) { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(Urls.CREDITS_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchUserGuide(View view) { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(Urls.USER_GUIDE_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchPrivacyPolicy(View view) { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(BuildConfig.PRIVACY_POLICY_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchFrequentlyAskedQuesions(View view) { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(Urls.FAQ_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onCreateOptionsMenu(Menu menu) { |  | ||||||
|         MenuInflater inflater = getMenuInflater(); |  | ||||||
|         inflater.inflate(R.menu.menu_about, menu); |  | ||||||
|         return super.onCreateOptionsMenu(menu); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|         switch (item.getItemId()) { |  | ||||||
|             case R.id.share_app_icon: |  | ||||||
|                 String shareText = String.format(getString(R.string.share_text), Urls.PLAY_STORE_URL_PREFIX + this.getPackageName()); |  | ||||||
|                 Intent sendIntent = new Intent(); |  | ||||||
|                 sendIntent.setAction(Intent.ACTION_SEND); |  | ||||||
|                 sendIntent.putExtra(Intent.EXTRA_TEXT, shareText); |  | ||||||
|                 sendIntent.setType("text/plain"); |  | ||||||
|                 startActivity(Intent.createChooser(sendIntent, getString(R.string.share_via))); |  | ||||||
|                 return true; |  | ||||||
|             default: |  | ||||||
|                 return super.onOptionsItemSelected(item); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void launchTranslate(View view) { |  | ||||||
|         @NonNull List<String> sortedLocalizedNamesRef = CommonsApplication.getInstance().getLanguageLookUpTable().getCanonicalNames(); |  | ||||||
|         Collections.sort(sortedLocalizedNamesRef); |  | ||||||
|         final ArrayAdapter<String> languageAdapter = new ArrayAdapter<>(AboutActivity.this, |  | ||||||
|                 android.R.layout.simple_spinner_dropdown_item, sortedLocalizedNamesRef); |  | ||||||
|         final Spinner spinner = new Spinner(AboutActivity.this); |  | ||||||
|         spinner.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT)); |  | ||||||
|         spinner.setAdapter(languageAdapter); |  | ||||||
|         spinner.setGravity(17); |  | ||||||
|         spinner.setPadding(50,0,0,0); |  | ||||||
| 
 |  | ||||||
|         Runnable positiveButtonRunnable = () -> { |  | ||||||
|             String langCode = CommonsApplication.getInstance().getLanguageLookUpTable().getCodes().get(spinner.getSelectedItemPosition()); |  | ||||||
|             Utils.handleWebUrl(AboutActivity.this, Uri.parse(Urls.TRANSLATE_WIKI_URL + langCode)); |  | ||||||
|         }; |  | ||||||
|         DialogUtil.showAlertDialog(this, |  | ||||||
|             getString(R.string.about_translate_title), |  | ||||||
|             getString(R.string.about_translate_message), |  | ||||||
|             getString(R.string.about_translate_proceed), |  | ||||||
|             getString(R.string.about_translate_cancel), |  | ||||||
|             positiveButtonRunnable, |  | ||||||
|             () -> {}, |  | ||||||
|             spinner, |  | ||||||
|             true); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
							
								
								
									
										207
									
								
								app/src/main/java/fr/free/nrw/commons/AboutActivity.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								app/src/main/java/fr/free/nrw/commons/AboutActivity.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,207 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | import android.annotation.SuppressLint | ||||||
|  | import android.content.ActivityNotFoundException | ||||||
|  | import android.content.Intent | ||||||
|  | import android.content.Intent.ACTION_VIEW | ||||||
|  | import android.net.Uri | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.Menu | ||||||
|  | import android.view.MenuItem | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.ArrayAdapter | ||||||
|  | import android.widget.LinearLayout | ||||||
|  | import android.widget.Spinner | ||||||
|  | import fr.free.nrw.commons.CommonsApplication.Companion.instance | ||||||
|  | import fr.free.nrw.commons.databinding.ActivityAboutBinding | ||||||
|  | import fr.free.nrw.commons.theme.BaseActivity | ||||||
|  | import fr.free.nrw.commons.utils.ConfigUtils.getVersionNameWithSha | ||||||
|  | import fr.free.nrw.commons.utils.DialogUtil.showAlertDialog | ||||||
|  | import java.util.Collections | ||||||
|  | import androidx.core.net.toUri | ||||||
|  | import fr.free.nrw.commons.utils.applyEdgeToEdgeTopInsets | ||||||
|  | import fr.free.nrw.commons.utils.handleWebUrl | ||||||
|  | import fr.free.nrw.commons.utils.setUnderlinedText | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Represents about screen of this app | ||||||
|  |  */ | ||||||
|  | class AboutActivity : BaseActivity() { | ||||||
|  |     /* | ||||||
|  |          This View Binding class is auto-generated for each xml file. The format is usually the name | ||||||
|  |          of the file with PascalCasing (The underscore characters will be ignored). | ||||||
|  |          More information is available at https://developer.android.com/topic/libraries/view-binding | ||||||
|  |         */ | ||||||
|  |     private var binding: ActivityAboutBinding? = null | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This method helps in the creation About screen | ||||||
|  |      * | ||||||
|  |      * @param savedInstanceState Data bundle | ||||||
|  |      */ | ||||||
|  |     @SuppressLint("StringFormatInvalid")  //TODO: | ||||||
|  |     public override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  | 
 | ||||||
|  |         /* | ||||||
|  |           Instead of just setting the view with the xml file. We need to use View Binding class. | ||||||
|  |          */ | ||||||
|  |         binding = ActivityAboutBinding.inflate(layoutInflater) | ||||||
|  |         val view: View = binding!!.root | ||||||
|  |         applyEdgeToEdgeTopInsets(binding!!.toolbarLayout) | ||||||
|  |         setContentView(view) | ||||||
|  | 
 | ||||||
|  |         setSupportActionBar(binding!!.toolbarBinding.toolbar) | ||||||
|  |         supportActionBar!!.setDisplayHomeAsUpEnabled(true) | ||||||
|  |         val aboutText = getString(R.string.about_license) | ||||||
|  |         /* | ||||||
|  |           We can then access all the views by just using the id names like this. | ||||||
|  |           camelCasing is used with underscore characters being ignored. | ||||||
|  |          */ | ||||||
|  |         binding!!.aboutLicense.setHtmlText(aboutText) | ||||||
|  | 
 | ||||||
|  |         @SuppressLint("StringFormatMatches") // TODO: | ||||||
|  |         val improveText = | ||||||
|  |             String.format(getString(R.string.about_improve), Urls.NEW_ISSUE_URL) | ||||||
|  |         binding!!.aboutImprove.setHtmlText(improveText) | ||||||
|  |         binding!!.aboutVersion.text = applicationContext.getVersionNameWithSha() | ||||||
|  | 
 | ||||||
|  |         binding!!.aboutFaq.setUnderlinedText(R.string.about_faq) | ||||||
|  |         binding!!.aboutRateUs.setUnderlinedText(R.string.about_rate_us) | ||||||
|  |         binding!!.aboutUserGuide.setUnderlinedText(R.string.user_guide) | ||||||
|  |         binding!!.aboutPrivacyPolicy.setUnderlinedText(R.string.about_privacy_policy) | ||||||
|  |         binding!!.aboutTranslate.setUnderlinedText(R.string.about_translate) | ||||||
|  |         binding!!.aboutCredits.setUnderlinedText(R.string.about_credits) | ||||||
|  | 
 | ||||||
|  |         /* | ||||||
|  |           To set listeners, we can create a separate method and use lambda syntax. | ||||||
|  |         */ | ||||||
|  |         binding!!.facebookLaunchIcon.setOnClickListener(::launchFacebook) | ||||||
|  |         binding!!.githubLaunchIcon.setOnClickListener(::launchGithub) | ||||||
|  |         binding!!.websiteLaunchIcon.setOnClickListener(::launchWebsite) | ||||||
|  |         binding!!.aboutRateUs.setOnClickListener(::launchRatings) | ||||||
|  |         binding!!.aboutCredits.setOnClickListener(::launchCredits) | ||||||
|  |         binding!!.aboutPrivacyPolicy.setOnClickListener(::launchPrivacyPolicy) | ||||||
|  |         binding!!.aboutUserGuide.setOnClickListener(::launchUserGuide) | ||||||
|  |         binding!!.aboutFaq.setOnClickListener(::launchFrequentlyAskedQuesions) | ||||||
|  |         binding!!.aboutTranslate.setOnClickListener(::launchTranslate) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onSupportNavigateUp(): Boolean { | ||||||
|  |         onBackPressed() | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchFacebook(view: View?) { | ||||||
|  |         val intent: Intent | ||||||
|  |         try { | ||||||
|  |             intent = Intent(ACTION_VIEW, Urls.FACEBOOK_APP_URL.toUri()) | ||||||
|  |             intent.setPackage(Urls.FACEBOOK_PACKAGE_NAME) | ||||||
|  |             startActivity(intent) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             handleWebUrl(this, Urls.FACEBOOK_WEB_URL.toUri()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchGithub(view: View?) { | ||||||
|  |         val intent: Intent | ||||||
|  |         try { | ||||||
|  |             intent = Intent(ACTION_VIEW, Urls.GITHUB_REPO_URL.toUri()) | ||||||
|  |             intent.setPackage(Urls.GITHUB_PACKAGE_NAME) | ||||||
|  |             startActivity(intent) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             handleWebUrl(this, Urls.GITHUB_REPO_URL.toUri()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchWebsite(view: View?) { | ||||||
|  |         handleWebUrl(this, Urls.WEBSITE_URL.toUri()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchRatings(view: View?) { | ||||||
|  |         try { | ||||||
|  |             startActivity( | ||||||
|  |                 Intent( | ||||||
|  |                     ACTION_VIEW, | ||||||
|  |                     (Urls.PLAY_STORE_PREFIX + packageName).toUri() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } catch (_: ActivityNotFoundException) { | ||||||
|  |             handleWebUrl(this, (Urls.PLAY_STORE_URL_PREFIX + packageName).toUri()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchCredits(view: View?) { | ||||||
|  |         handleWebUrl(this, Urls.CREDITS_URL.toUri()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchUserGuide(view: View?) { | ||||||
|  |         handleWebUrl(this, Urls.USER_GUIDE_URL.toUri()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchPrivacyPolicy(view: View?) { | ||||||
|  |         handleWebUrl(this, BuildConfig.PRIVACY_POLICY_URL.toUri()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchFrequentlyAskedQuesions(view: View?) { | ||||||
|  |         handleWebUrl(this, Urls.FAQ_URL.toUri()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onCreateOptionsMenu(menu: Menu): Boolean { | ||||||
|  |         val inflater = menuInflater | ||||||
|  |         inflater.inflate(R.menu.menu_about, menu) | ||||||
|  |         return super.onCreateOptionsMenu(menu) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|  |         when (item.itemId) { | ||||||
|  |             R.id.share_app_icon -> { | ||||||
|  |                 val shareText = String.format( | ||||||
|  |                     getString(R.string.share_text), | ||||||
|  |                     Urls.PLAY_STORE_URL_PREFIX + this.packageName | ||||||
|  |                 ) | ||||||
|  |                 val sendIntent = Intent() | ||||||
|  |                 sendIntent.setAction(Intent.ACTION_SEND) | ||||||
|  |                 sendIntent.putExtra(Intent.EXTRA_TEXT, shareText) | ||||||
|  |                 sendIntent.setType("text/plain") | ||||||
|  |                 startActivity(Intent.createChooser(sendIntent, getString(R.string.share_via))) | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             else -> return super.onOptionsItemSelected(item) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun launchTranslate(view: View?) { | ||||||
|  |         val sortedLocalizedNamesRef = instance.languageLookUpTable!!.getCanonicalNames() | ||||||
|  |         Collections.sort(sortedLocalizedNamesRef) | ||||||
|  |         val languageAdapter = ArrayAdapter( | ||||||
|  |             this@AboutActivity, | ||||||
|  |             android.R.layout.simple_spinner_dropdown_item, sortedLocalizedNamesRef | ||||||
|  |         ) | ||||||
|  |         val spinner = Spinner(this@AboutActivity) | ||||||
|  |         spinner.layoutParams = | ||||||
|  |             LinearLayout.LayoutParams( | ||||||
|  |                 LinearLayout.LayoutParams.WRAP_CONTENT, | ||||||
|  |                 LinearLayout.LayoutParams.WRAP_CONTENT | ||||||
|  |             ) | ||||||
|  |         spinner.adapter = languageAdapter | ||||||
|  |         spinner.gravity = 17 | ||||||
|  |         spinner.setPadding(50, 0, 0, 0) | ||||||
|  | 
 | ||||||
|  |         val positiveButtonRunnable = Runnable { | ||||||
|  |             val langCode = instance.languageLookUpTable!!.getCodes()[spinner.selectedItemPosition] | ||||||
|  |             handleWebUrl(this@AboutActivity, (Urls.TRANSLATE_WIKI_URL + langCode).toUri()) | ||||||
|  |         } | ||||||
|  |         showAlertDialog( | ||||||
|  |             this, | ||||||
|  |             getString(R.string.about_translate_title), | ||||||
|  |             getString(R.string.about_translate_message), | ||||||
|  |             getString(R.string.about_translate_proceed), | ||||||
|  |             getString(R.string.about_translate_cancel), | ||||||
|  |             positiveButtonRunnable, | ||||||
|  |             {}, | ||||||
|  |             spinner | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,18 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Base presenter, enforcing contracts to atach and detach view |  | ||||||
|  */ |  | ||||||
| public interface BasePresenter<T> { |  | ||||||
|     /** |  | ||||||
|      * Until a view is attached, it is open to listen events from the presenter |  | ||||||
|      */ |  | ||||||
|     void onAttachView(@NonNull T view); |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Detaching a view makes sure that the view no more receives events from the presenter |  | ||||||
|      */ |  | ||||||
|     void onDetachView(); |  | ||||||
| } |  | ||||||
							
								
								
									
										10
									
								
								app/src/main/java/fr/free/nrw/commons/BasePresenter.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								app/src/main/java/fr/free/nrw/commons/BasePresenter.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Base presenter, enforcing contracts to attach and detach view | ||||||
|  |  */ | ||||||
|  | interface BasePresenter<T> { | ||||||
|  |     fun onAttachView(view: T) | ||||||
|  | 
 | ||||||
|  |     fun onDetachView() | ||||||
|  | } | ||||||
|  | @ -15,9 +15,8 @@ import com.facebook.drawee.backends.pipeline.Fresco | ||||||
| import com.facebook.imagepipeline.core.ImagePipelineConfig | import com.facebook.imagepipeline.core.ImagePipelineConfig | ||||||
| import fr.free.nrw.commons.auth.LoginActivity | import fr.free.nrw.commons.auth.LoginActivity | ||||||
| import fr.free.nrw.commons.auth.SessionManager | import fr.free.nrw.commons.auth.SessionManager | ||||||
| import fr.free.nrw.commons.bookmarks.items.BookmarkItemsDao | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable | ||||||
| import fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsDao | import fr.free.nrw.commons.bookmarks.pictures.BookmarksTable | ||||||
| import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesDao |  | ||||||
| import fr.free.nrw.commons.category.CategoryDao | import fr.free.nrw.commons.category.CategoryDao | ||||||
| import fr.free.nrw.commons.concurrency.BackgroundPoolExceptionHandler | import fr.free.nrw.commons.concurrency.BackgroundPoolExceptionHandler | ||||||
| import fr.free.nrw.commons.concurrency.ThreadPoolService | import fr.free.nrw.commons.concurrency.ThreadPoolService | ||||||
|  | @ -247,14 +246,18 @@ class CommonsApplication : MultiDexApplication() { | ||||||
|             DBOpenHelper.CONTRIBUTIONS_TABLE |             DBOpenHelper.CONTRIBUTIONS_TABLE | ||||||
|         ) //Delete the contributions table in the existing db on older versions |         ) //Delete the contributions table in the existing db on older versions | ||||||
| 
 | 
 | ||||||
|  |         dbOpenHelper.deleteTable( | ||||||
|  |             db, | ||||||
|  |             DBOpenHelper.BOOKMARKS_LOCATIONS | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|         try { |         try { | ||||||
|             contributionDao.deleteAll() |             contributionDao.deleteAll() | ||||||
|         } catch (e: SQLiteException) { |         } catch (e: SQLiteException) { | ||||||
|             Timber.e(e) |             Timber.e(e) | ||||||
|         } |         } | ||||||
|         BookmarkPicturesDao.Table.onDelete(db) |         BookmarksTable.onDelete(db) | ||||||
|         BookmarkLocationsDao.Table.onDelete(db) |         BookmarkItemsTable.onDelete(db) | ||||||
|         BookmarkItemsDao.Table.onDelete(db) |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,79 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * represents Licence object |  | ||||||
|  */ |  | ||||||
| public class License { |  | ||||||
|     private String key; |  | ||||||
|     private String template; |  | ||||||
|     private String url; |  | ||||||
|     private String name; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Constructs a new instance of License. |  | ||||||
|      * |  | ||||||
|      * @param key       license key |  | ||||||
|      * @param template  license template |  | ||||||
|      * @param url       license URL |  | ||||||
|      * @param name      licence name |  | ||||||
|      * |  | ||||||
|      * @throws RuntimeException if License.key or Licence.template is null |  | ||||||
|      */ |  | ||||||
|     public License(String key, String template, String url, String name) { |  | ||||||
|         if (key == null) { |  | ||||||
|             throw new RuntimeException("License.key must not be null"); |  | ||||||
|         } |  | ||||||
|         if (template == null) { |  | ||||||
|             throw new RuntimeException("License.template must not be null"); |  | ||||||
|         } |  | ||||||
|         this.key = key; |  | ||||||
|         this.template = template; |  | ||||||
|         this.url = url; |  | ||||||
|         this.name = name; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the license key. |  | ||||||
|      * @return license key as a String. |  | ||||||
|      */ |  | ||||||
|     public String getKey() { |  | ||||||
|         return key; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the license template. |  | ||||||
|      * @return license template as a String. |  | ||||||
|      */ |  | ||||||
|     public String getTemplate() { |  | ||||||
|         return template; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the license name. If name is null, return license key. |  | ||||||
|      * @return license name as string. if name null, license key as String |  | ||||||
|      */ |  | ||||||
|     public String getName() { |  | ||||||
|         if (name == null) { |  | ||||||
|             // hack |  | ||||||
|             return getKey(); |  | ||||||
|         } else { |  | ||||||
|             return name; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets the license URL |  | ||||||
|      * |  | ||||||
|      * @param language license language |  | ||||||
|      * @return URL |  | ||||||
|      */ |  | ||||||
|     public @Nullable String getUrl(String language) { |  | ||||||
|         if (url == null) { |  | ||||||
|             return null; |  | ||||||
|         } else { |  | ||||||
|             return url.replace("$lang", language); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,77 +0,0 @@ | ||||||
| package fr.free.nrw.commons.LocationPicker; |  | ||||||
| 
 |  | ||||||
| import android.app.Activity; |  | ||||||
| import android.content.Intent; |  | ||||||
| import fr.free.nrw.commons.CameraPosition; |  | ||||||
| import fr.free.nrw.commons.Media; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Helper class for starting the activity |  | ||||||
|  */ |  | ||||||
| public final class LocationPicker { |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Getting camera position from the intent using constants |  | ||||||
|      * |  | ||||||
|      * @param data intent |  | ||||||
|      * @return CameraPosition |  | ||||||
|      */ |  | ||||||
|     public static CameraPosition getCameraPosition(final Intent data) { |  | ||||||
|         return data.getParcelableExtra(LocationPickerConstants.MAP_CAMERA_POSITION); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public static class IntentBuilder { |  | ||||||
| 
 |  | ||||||
|         private final Intent intent; |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Creates a new builder that creates an intent to launch the place picker activity. |  | ||||||
|          */ |  | ||||||
|         public IntentBuilder() { |  | ||||||
|             intent = new Intent(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Gets and puts location in intent |  | ||||||
|          * @param position CameraPosition |  | ||||||
|          * @return LocationPicker.IntentBuilder |  | ||||||
|          */ |  | ||||||
|         public LocationPicker.IntentBuilder defaultLocation( |  | ||||||
|             final CameraPosition position) { |  | ||||||
|           intent.putExtra(LocationPickerConstants.MAP_CAMERA_POSITION, position); |  | ||||||
|           return this; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Gets and puts activity name in intent |  | ||||||
|          * @param activity activity key |  | ||||||
|          * @return LocationPicker.IntentBuilder |  | ||||||
|          */ |  | ||||||
|         public LocationPicker.IntentBuilder activityKey( |  | ||||||
|             final String activity) { |  | ||||||
|           intent.putExtra(LocationPickerConstants.ACTIVITY_KEY, activity); |  | ||||||
|           return this; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Gets and puts media in intent |  | ||||||
|          * @param media Media |  | ||||||
|          * @return LocationPicker.IntentBuilder |  | ||||||
|          */ |  | ||||||
|         public LocationPicker.IntentBuilder media( |  | ||||||
|                 final Media media) { |  | ||||||
|               intent.putExtra(LocationPickerConstants.MEDIA, media); |  | ||||||
|               return this; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Gets and sets the activity |  | ||||||
|          * @param activity Activity |  | ||||||
|          * @return Intent |  | ||||||
|          */ |  | ||||||
|        public Intent build(final Activity activity) { |  | ||||||
|           intent.setClass(activity, LocationPickerActivity.class); |  | ||||||
|           return intent; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,681 +0,0 @@ | ||||||
| package fr.free.nrw.commons.LocationPicker; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.upload.mediaDetails.UploadMediaDetailFragment.LAST_LOCATION; |  | ||||||
| import static fr.free.nrw.commons.upload.mediaDetails.UploadMediaDetailFragment.LAST_ZOOM; |  | ||||||
| import static fr.free.nrw.commons.utils.MapUtils.ZOOM_LEVEL; |  | ||||||
| 
 |  | ||||||
| import android.Manifest.permission; |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.content.pm.PackageManager; |  | ||||||
| import android.graphics.Color; |  | ||||||
| import android.graphics.Paint; |  | ||||||
| import android.graphics.drawable.Drawable; |  | ||||||
| import android.location.LocationManager; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.preference.PreferenceManager; |  | ||||||
| import android.text.Html; |  | ||||||
| import android.text.method.LinkMovementMethod; |  | ||||||
| import android.view.MotionEvent; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.Window; |  | ||||||
| import android.view.animation.OvershootInterpolator; |  | ||||||
| import android.widget.Button; |  | ||||||
| import android.widget.ImageView; |  | ||||||
| import android.widget.TextView; |  | ||||||
| import android.widget.Toast; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.appcompat.app.ActionBar; |  | ||||||
| import androidx.appcompat.app.AppCompatActivity; |  | ||||||
| import androidx.appcompat.widget.AppCompatTextView; |  | ||||||
| import androidx.constraintlayout.widget.ConstraintLayout; |  | ||||||
| import androidx.core.app.ActivityCompat; |  | ||||||
| import androidx.core.content.ContextCompat; |  | ||||||
| import com.google.android.material.floatingactionbutton.FloatingActionButton; |  | ||||||
| import fr.free.nrw.commons.CameraPosition; |  | ||||||
| import fr.free.nrw.commons.CommonsApplication; |  | ||||||
| import fr.free.nrw.commons.Media; |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.Utils; |  | ||||||
| import fr.free.nrw.commons.auth.SessionManager; |  | ||||||
| import fr.free.nrw.commons.auth.csrf.CsrfTokenClient; |  | ||||||
| import fr.free.nrw.commons.coordinates.CoordinateEditHelper; |  | ||||||
| import fr.free.nrw.commons.filepicker.Constants; |  | ||||||
| import fr.free.nrw.commons.kvstore.BasicKvStore; |  | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; |  | ||||||
| import fr.free.nrw.commons.location.LocationPermissionsHelper; |  | ||||||
| import fr.free.nrw.commons.location.LocationPermissionsHelper.LocationPermissionCallback; |  | ||||||
| import fr.free.nrw.commons.location.LocationServiceManager; |  | ||||||
| import fr.free.nrw.commons.theme.BaseActivity; |  | ||||||
| import fr.free.nrw.commons.utils.DialogUtil; |  | ||||||
| import fr.free.nrw.commons.utils.SystemThemeUtils; |  | ||||||
| import io.reactivex.android.schedulers.AndroidSchedulers; |  | ||||||
| import io.reactivex.schedulers.Schedulers; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Locale; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Named; |  | ||||||
| import org.osmdroid.tileprovider.tilesource.TileSourceFactory; |  | ||||||
| import org.osmdroid.util.GeoPoint; |  | ||||||
| import org.osmdroid.util.constants.GeoConstants; |  | ||||||
| import org.osmdroid.views.CustomZoomButtonsController; |  | ||||||
| import org.osmdroid.views.overlay.Marker; |  | ||||||
| import org.osmdroid.views.overlay.Overlay; |  | ||||||
| import org.osmdroid.views.overlay.ScaleDiskOverlay; |  | ||||||
| import org.osmdroid.views.overlay.TilesOverlay; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Helps to pick location and return the result with an intent |  | ||||||
|  */ |  | ||||||
| public class LocationPickerActivity extends BaseActivity implements |  | ||||||
|     LocationPermissionCallback { |  | ||||||
|     /** |  | ||||||
|      * coordinateEditHelper: helps to edit coordinates |  | ||||||
|      */ |  | ||||||
|     @Inject |  | ||||||
|     CoordinateEditHelper coordinateEditHelper; |  | ||||||
|     /** |  | ||||||
|      * media : Media object |  | ||||||
|      */ |  | ||||||
|     private Media media; |  | ||||||
|     /** |  | ||||||
|      * cameraPosition : position of picker |  | ||||||
|      */ |  | ||||||
|     private CameraPosition cameraPosition; |  | ||||||
|     /** |  | ||||||
|      * markerImage : picker image |  | ||||||
|      */ |  | ||||||
|     private ImageView markerImage; |  | ||||||
|     /** |  | ||||||
|      * mapView : OSM Map |  | ||||||
|      */ |  | ||||||
|     private org.osmdroid.views.MapView mapView; |  | ||||||
|     /** |  | ||||||
|      * tvAttribution : credit |  | ||||||
|      */ |  | ||||||
|     private AppCompatTextView tvAttribution; |  | ||||||
|     /** |  | ||||||
|      * activity : activity key |  | ||||||
|      */ |  | ||||||
|     private String activity; |  | ||||||
|     /** |  | ||||||
|      * modifyLocationButton : button for start editing location |  | ||||||
|      */ |  | ||||||
|     Button modifyLocationButton; |  | ||||||
|     /** |  | ||||||
|      * removeLocationButton : button to remove location metadata |  | ||||||
|      */ |  | ||||||
|     Button removeLocationButton; |  | ||||||
|     /** |  | ||||||
|      * showInMapButton : button for showing in map |  | ||||||
|      */ |  | ||||||
|     TextView showInMapButton; |  | ||||||
|     /** |  | ||||||
|      * placeSelectedButton : fab for selecting location |  | ||||||
|      */ |  | ||||||
|     FloatingActionButton placeSelectedButton; |  | ||||||
|     /** |  | ||||||
|      * fabCenterOnLocation: button for center on location; |  | ||||||
|      */ |  | ||||||
|     FloatingActionButton fabCenterOnLocation; |  | ||||||
|     /** |  | ||||||
|      * shadow : imageview of shadow |  | ||||||
|      */ |  | ||||||
|     private ImageView shadow; |  | ||||||
|     /** |  | ||||||
|      * largeToolbarText : textView of shadow |  | ||||||
|      */ |  | ||||||
|     private TextView largeToolbarText; |  | ||||||
|     /** |  | ||||||
|      * smallToolbarText : textView of shadow |  | ||||||
|      */ |  | ||||||
|     private TextView smallToolbarText; |  | ||||||
|     /** |  | ||||||
|      * applicationKvStore : for storing values |  | ||||||
|      */ |  | ||||||
|     @Inject |  | ||||||
|     @Named("default_preferences") |  | ||||||
|     public |  | ||||||
|     JsonKvStore applicationKvStore; |  | ||||||
|     BasicKvStore store; |  | ||||||
|     /** |  | ||||||
|      * isDarkTheme: for keeping a track of the device theme and modifying the map theme accordingly |  | ||||||
|      */ |  | ||||||
|     @Inject |  | ||||||
|     SystemThemeUtils systemThemeUtils; |  | ||||||
|     private boolean isDarkTheme; |  | ||||||
|     private boolean moveToCurrentLocation; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     LocationServiceManager locationManager; |  | ||||||
|     LocationPermissionsHelper locationPermissionsHelper; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     SessionManager sessionManager; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Constants |  | ||||||
|      */ |  | ||||||
|     private static final String CAMERA_POS = "cameraPosition"; |  | ||||||
|     private static final String ACTIVITY = "activity"; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     @SuppressLint("ClickableViewAccessibility") |  | ||||||
|     @Override |  | ||||||
|     protected void onCreate(@Nullable final Bundle savedInstanceState) { |  | ||||||
|         getWindow().requestFeature(Window.FEATURE_ACTION_BAR); |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
| 
 |  | ||||||
|         isDarkTheme = systemThemeUtils.isDeviceInNightMode(); |  | ||||||
|         moveToCurrentLocation = false; |  | ||||||
|         store = new BasicKvStore(this, "LocationPermissions"); |  | ||||||
| 
 |  | ||||||
|         getWindow().requestFeature(Window.FEATURE_ACTION_BAR); |  | ||||||
|         final ActionBar actionBar = getSupportActionBar(); |  | ||||||
|         if (actionBar != null) { |  | ||||||
|             actionBar.hide(); |  | ||||||
|         } |  | ||||||
|         setContentView(R.layout.activity_location_picker); |  | ||||||
| 
 |  | ||||||
|         if (savedInstanceState == null) { |  | ||||||
|             cameraPosition = getIntent() |  | ||||||
|                 .getParcelableExtra(LocationPickerConstants.MAP_CAMERA_POSITION); |  | ||||||
|             activity = getIntent().getStringExtra(LocationPickerConstants.ACTIVITY_KEY); |  | ||||||
|             media = getIntent().getParcelableExtra(LocationPickerConstants.MEDIA); |  | ||||||
|         }else{ |  | ||||||
|             cameraPosition = savedInstanceState.getParcelable(CAMERA_POS); |  | ||||||
|             activity = savedInstanceState.getString(ACTIVITY); |  | ||||||
|             media = savedInstanceState.getParcelable("sMedia"); |  | ||||||
|         } |  | ||||||
|         bindViews(); |  | ||||||
|         addBackButtonListener(); |  | ||||||
|         addPlaceSelectedButton(); |  | ||||||
|         addCredits(); |  | ||||||
|         getToolbarUI(); |  | ||||||
|         addCenterOnGPSButton(); |  | ||||||
| 
 |  | ||||||
|         org.osmdroid.config.Configuration.getInstance().load(getApplicationContext(), |  | ||||||
|             PreferenceManager.getDefaultSharedPreferences(getApplicationContext())); |  | ||||||
| 
 |  | ||||||
|         mapView.setTileSource(TileSourceFactory.WIKIMEDIA); |  | ||||||
|         mapView.setTilesScaledToDpi(true); |  | ||||||
|         mapView.setMultiTouchControls(true); |  | ||||||
| 
 |  | ||||||
|         org.osmdroid.config.Configuration.getInstance().getAdditionalHttpRequestProperties().put( |  | ||||||
|             "Referer", "http://maps.wikimedia.org/" |  | ||||||
|         ); |  | ||||||
|         mapView.getZoomController().setVisibility(CustomZoomButtonsController.Visibility.NEVER); |  | ||||||
|         mapView.getController().setZoom(ZOOM_LEVEL); |  | ||||||
|         mapView.setOnTouchListener((v, event) -> { |  | ||||||
|             if (event.getAction() == MotionEvent.ACTION_MOVE) { |  | ||||||
|                 if (markerImage.getTranslationY() == 0) { |  | ||||||
|                     markerImage.animate().translationY(-75) |  | ||||||
|                         .setInterpolator(new OvershootInterpolator()).setDuration(250).start(); |  | ||||||
|                 } |  | ||||||
|             } else if (event.getAction() == MotionEvent.ACTION_UP) { |  | ||||||
|                 markerImage.animate().translationY(0) |  | ||||||
|                     .setInterpolator(new OvershootInterpolator()).setDuration(250).start(); |  | ||||||
|             } |  | ||||||
|             return false; |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         if ("UploadActivity".equals(activity)) { |  | ||||||
|             placeSelectedButton.setVisibility(View.GONE); |  | ||||||
|             modifyLocationButton.setVisibility(View.VISIBLE); |  | ||||||
|             removeLocationButton.setVisibility(View.VISIBLE); |  | ||||||
|             showInMapButton.setVisibility(View.VISIBLE); |  | ||||||
|             largeToolbarText.setText(getResources().getString(R.string.image_location)); |  | ||||||
|             smallToolbarText.setText(getResources(). |  | ||||||
|                 getString(R.string.check_whether_location_is_correct)); |  | ||||||
|             fabCenterOnLocation.setVisibility(View.GONE); |  | ||||||
|             markerImage.setVisibility(View.GONE); |  | ||||||
|             shadow.setVisibility(View.GONE); |  | ||||||
|             assert cameraPosition != null; |  | ||||||
|             showSelectedLocationMarker(new GeoPoint(cameraPosition.getLatitude(), |  | ||||||
|                 cameraPosition.getLongitude())); |  | ||||||
|         } |  | ||||||
|         setupMapView(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Moves the center of the map to the specified coordinates |  | ||||||
|      * |  | ||||||
|      */ |  | ||||||
|     private void moveMapTo(double latitude, double longitude){ |  | ||||||
|         if(mapView != null && mapView.getController() != null){ |  | ||||||
|             GeoPoint point = new GeoPoint(latitude, longitude); |  | ||||||
| 
 |  | ||||||
|             mapView.getController().setCenter(point); |  | ||||||
|             mapView.getController().animateTo(point); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Moves the center of the map to the specified coordinates |  | ||||||
|      * @param point The GeoPoint object which contains the coordinates to move to |  | ||||||
|      */ |  | ||||||
|     private void moveMapTo(GeoPoint point){ |  | ||||||
|         if(point != null){ |  | ||||||
|             moveMapTo(point.getLatitude(), point.getLongitude()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * For showing credits |  | ||||||
|      */ |  | ||||||
|     private void addCredits() { |  | ||||||
|         tvAttribution.setText(Html.fromHtml(getString(R.string.map_attribution))); |  | ||||||
|         tvAttribution.setMovementMethod(LinkMovementMethod.getInstance()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * For setting up Dark Theme |  | ||||||
|      */ |  | ||||||
|     private void darkThemeSetup() { |  | ||||||
|         if (isDarkTheme) { |  | ||||||
|             shadow.setColorFilter(Color.argb(255, 255, 255, 255)); |  | ||||||
|             mapView.getOverlayManager().getTilesOverlay() |  | ||||||
|                 .setColorFilter(TilesOverlay.INVERT_COLORS); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Clicking back button destroy locationPickerActivity |  | ||||||
|      */ |  | ||||||
|     private void addBackButtonListener() { |  | ||||||
|         final ImageView backButton = findViewById(R.id.maplibre_place_picker_toolbar_back_button); |  | ||||||
|         backButton.setOnClickListener(v -> { |  | ||||||
|             finish(); |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Binds mapView and location picker icon |  | ||||||
|      */ |  | ||||||
|     private void bindViews() { |  | ||||||
|         mapView = findViewById(R.id.map_view); |  | ||||||
|         markerImage = findViewById(R.id.location_picker_image_view_marker); |  | ||||||
|         tvAttribution = findViewById(R.id.tv_attribution); |  | ||||||
|         modifyLocationButton = findViewById(R.id.modify_location); |  | ||||||
|         removeLocationButton = findViewById(R.id.remove_location); |  | ||||||
|         showInMapButton = findViewById(R.id.show_in_map); |  | ||||||
|         showInMapButton.setText(getResources().getString(R.string.show_in_map_app).toUpperCase( |  | ||||||
|             Locale.ROOT)); |  | ||||||
|         shadow = findViewById(R.id.location_picker_image_view_shadow); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets toolbar color |  | ||||||
|      */ |  | ||||||
|     private void getToolbarUI() { |  | ||||||
|         final ConstraintLayout toolbar = findViewById(R.id.location_picker_toolbar); |  | ||||||
|         largeToolbarText = findViewById(R.id.location_picker_toolbar_primary_text_view); |  | ||||||
|         smallToolbarText = findViewById(R.id.location_picker_toolbar_secondary_text_view); |  | ||||||
|         toolbar.setBackgroundColor(getResources().getColor(R.color.primaryColor)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void setupMapView() { |  | ||||||
|         requestLocationPermissions(); |  | ||||||
| 
 |  | ||||||
|         //If location metadata is available, move map to that location. |  | ||||||
|         if(activity.equals("UploadActivity") || activity.equals("MediaActivity")){ |  | ||||||
|             moveMapToMediaLocation(); |  | ||||||
|         } else { |  | ||||||
|             //If location metadata is not available, move map to device GPS location. |  | ||||||
|             moveMapToGPSLocation(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         modifyLocationButton.setOnClickListener(v -> onClickModifyLocation()); |  | ||||||
|         removeLocationButton.setOnClickListener(v -> onClickRemoveLocation()); |  | ||||||
|         showInMapButton.setOnClickListener(v -> showInMapApp()); |  | ||||||
|         darkThemeSetup(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles onclick event of modifyLocationButton |  | ||||||
|      */ |  | ||||||
|     private void onClickModifyLocation() { |  | ||||||
|         placeSelectedButton.setVisibility(View.VISIBLE); |  | ||||||
|         modifyLocationButton.setVisibility(View.GONE); |  | ||||||
|         removeLocationButton.setVisibility(View.GONE); |  | ||||||
|         showInMapButton.setVisibility(View.GONE); |  | ||||||
|         markerImage.setVisibility(View.VISIBLE); |  | ||||||
|         shadow.setVisibility(View.VISIBLE); |  | ||||||
|         largeToolbarText.setText(getResources().getString(R.string.choose_a_location)); |  | ||||||
|         smallToolbarText.setText(getResources().getString(R.string.pan_and_zoom_to_adjust)); |  | ||||||
|         fabCenterOnLocation.setVisibility(View.VISIBLE); |  | ||||||
|         removeSelectedLocationMarker(); |  | ||||||
|         moveMapToMediaLocation(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles onclick event of removeLocationButton |  | ||||||
|      */ |  | ||||||
|     private void onClickRemoveLocation() { |  | ||||||
|         DialogUtil.showAlertDialog(this, |  | ||||||
|             getString(R.string.remove_location_warning_title), |  | ||||||
|             getString(R.string.remove_location_warning_desc), |  | ||||||
|             getString(R.string.continue_message), |  | ||||||
|             getString(R.string.cancel), () -> removeLocationFromImage(), null); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Method to remove the location from the picture |  | ||||||
|      */ |  | ||||||
|     private void removeLocationFromImage() { |  | ||||||
|         if (media != null) { |  | ||||||
|             compositeDisposable.add(coordinateEditHelper.makeCoordinatesEdit(getApplicationContext() |  | ||||||
|                     , media, "0.0", "0.0", "0.0f") |  | ||||||
|                 .subscribeOn(Schedulers.io()) |  | ||||||
|                 .observeOn(AndroidSchedulers.mainThread()) |  | ||||||
|                 .subscribe(s -> { |  | ||||||
|                     Timber.d("Coordinates are removed from the image"); |  | ||||||
|                 })); |  | ||||||
|         } |  | ||||||
|         final Intent returningIntent = new Intent(); |  | ||||||
|         setResult(AppCompatActivity.RESULT_OK, returningIntent); |  | ||||||
|         finish(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Show the location in map app. Map will center on the location metadata, if available. |  | ||||||
|      * If there is no location metadata, the map will center on the commons app map center. |  | ||||||
|      */ |  | ||||||
|     private void showInMapApp() { |  | ||||||
|         fr.free.nrw.commons.location.LatLng position = null; |  | ||||||
| 
 |  | ||||||
|         if(activity.equals("UploadActivity") && cameraPosition != null){ |  | ||||||
|             //location metadata is available |  | ||||||
|             position = new fr.free.nrw.commons.location.LatLng(cameraPosition.getLatitude(), |  | ||||||
|                 cameraPosition.getLongitude(), 0.0f); |  | ||||||
|         } else if(mapView != null){ |  | ||||||
|             //location metadata is not available |  | ||||||
|             position = new fr.free.nrw.commons.location.LatLng(mapView.getMapCenter().getLatitude(), |  | ||||||
|                 mapView.getMapCenter().getLongitude(), 0.0f); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if(position != null){ |  | ||||||
|             Utils.handleGeoCoordinates(this, position); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Moves the center of the map to the media's location, if that data |  | ||||||
|      * is available. |  | ||||||
|      */ |  | ||||||
|     private void moveMapToMediaLocation() { |  | ||||||
|         if (cameraPosition != null) { |  | ||||||
| 
 |  | ||||||
|             GeoPoint point = new GeoPoint(cameraPosition.getLatitude(), |  | ||||||
|                 cameraPosition.getLongitude()); |  | ||||||
| 
 |  | ||||||
|             moveMapTo(point); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Moves the center of the map to the device's GPS location, if that data is available. |  | ||||||
|      */ |  | ||||||
|     private void moveMapToGPSLocation(){ |  | ||||||
|         if(locationManager != null){ |  | ||||||
|             fr.free.nrw.commons.location.LatLng location = locationManager.getLastLocation(); |  | ||||||
| 
 |  | ||||||
|             if(location != null){ |  | ||||||
|                 GeoPoint point = new GeoPoint(location.getLatitude(), location.getLongitude()); |  | ||||||
| 
 |  | ||||||
|                 moveMapTo(point); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Select the preferable location |  | ||||||
|      */ |  | ||||||
|     private void addPlaceSelectedButton() { |  | ||||||
|         placeSelectedButton = findViewById(R.id.location_chosen_button); |  | ||||||
|         placeSelectedButton.setOnClickListener(view -> placeSelected()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Return the intent with required data |  | ||||||
|      */ |  | ||||||
|     void placeSelected() { |  | ||||||
|         if (activity.equals("NoLocationUploadActivity")) { |  | ||||||
|             applicationKvStore.putString(LAST_LOCATION, |  | ||||||
|                 mapView.getMapCenter().getLatitude() |  | ||||||
|                     + "," |  | ||||||
|                     + mapView.getMapCenter().getLongitude()); |  | ||||||
|             applicationKvStore.putString(LAST_ZOOM, mapView.getZoomLevel() + ""); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (media == null) { |  | ||||||
|             final Intent returningIntent = new Intent(); |  | ||||||
|             returningIntent.putExtra(LocationPickerConstants.MAP_CAMERA_POSITION, |  | ||||||
|                 new CameraPosition(mapView.getMapCenter().getLatitude(), |  | ||||||
|                     mapView.getMapCenter().getLongitude(), 14.0)); |  | ||||||
|             setResult(AppCompatActivity.RESULT_OK, returningIntent); |  | ||||||
|         } else { |  | ||||||
|             updateCoordinates(String.valueOf(mapView.getMapCenter().getLatitude()), |  | ||||||
|                 String.valueOf(mapView.getMapCenter().getLongitude()), |  | ||||||
|                 String.valueOf(0.0f)); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         finish(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Fetched coordinates are replaced with existing coordinates by a POST API call. |  | ||||||
|      * @param Latitude to be added |  | ||||||
|      * @param Longitude to be added |  | ||||||
|      * @param Accuracy to be added |  | ||||||
|      */ |  | ||||||
|     public void updateCoordinates(final String Latitude, final String Longitude, |  | ||||||
|         final String Accuracy) { |  | ||||||
|         if (media == null) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         try { |  | ||||||
|             compositeDisposable.add( |  | ||||||
|                 coordinateEditHelper.makeCoordinatesEdit(getApplicationContext(), media, |  | ||||||
|                         Latitude, Longitude, Accuracy) |  | ||||||
|                     .subscribeOn(Schedulers.io()) |  | ||||||
|                     .observeOn(AndroidSchedulers.mainThread()) |  | ||||||
|                     .subscribe(s -> { |  | ||||||
|                             Timber.d("Coordinates are added."); |  | ||||||
|                         })); |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             if (e.getLocalizedMessage().equals(CsrfTokenClient.ANONYMOUS_TOKEN_MESSAGE)) { |  | ||||||
|                 final String username = sessionManager.getUserName(); |  | ||||||
|                 final CommonsApplication.BaseLogoutListener logoutListener = new CommonsApplication.BaseLogoutListener( |  | ||||||
|                     this, |  | ||||||
|                     getString(R.string.invalid_login_message), |  | ||||||
|                     username |  | ||||||
|                 ); |  | ||||||
| 
 |  | ||||||
|                 CommonsApplication.getInstance().clearApplicationData( |  | ||||||
|                     this, logoutListener); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Center the camera on the last saved location |  | ||||||
|      */ |  | ||||||
|     private void addCenterOnGPSButton() { |  | ||||||
|         fabCenterOnLocation = findViewById(R.id.center_on_gps); |  | ||||||
|         fabCenterOnLocation.setOnClickListener(view -> { |  | ||||||
|             moveToCurrentLocation = true; |  | ||||||
|             requestLocationPermissions(); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Adds selected location marker on the map |  | ||||||
|      */ |  | ||||||
|     private void showSelectedLocationMarker(GeoPoint point) { |  | ||||||
|         Drawable icon = ContextCompat.getDrawable(this, R.drawable.map_default_map_marker); |  | ||||||
|         Marker marker = new Marker(mapView); |  | ||||||
|         marker.setPosition(point); |  | ||||||
|         marker.setAnchor(Marker.ANCHOR_CENTER, Marker.ANCHOR_BOTTOM); |  | ||||||
|         marker.setIcon(icon); |  | ||||||
|         marker.setInfoWindow(null); |  | ||||||
|         mapView.getOverlays().add(marker); |  | ||||||
|         mapView.invalidate(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Removes selected location marker from the map |  | ||||||
|      */ |  | ||||||
|     private void removeSelectedLocationMarker() { |  | ||||||
|         List<Overlay> overlays = mapView.getOverlays(); |  | ||||||
|         for (int i = 0; i < overlays.size(); i++) { |  | ||||||
|             if (overlays.get(i) instanceof Marker) { |  | ||||||
|                 Marker item = (Marker) overlays.get(i); |  | ||||||
|                 if (cameraPosition.getLatitude() == item.getPosition().getLatitude() |  | ||||||
|                     && cameraPosition.getLongitude() == item.getPosition().getLongitude()) { |  | ||||||
|                     mapView.getOverlays().remove(i); |  | ||||||
|                     mapView.invalidate(); |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Center the map at user's current location |  | ||||||
|      */ |  | ||||||
|     private void requestLocationPermissions() { |  | ||||||
|         locationPermissionsHelper = new LocationPermissionsHelper( |  | ||||||
|             this, locationManager, this); |  | ||||||
|         locationPermissionsHelper.requestForLocationAccess(R.string.location_permission_title, |  | ||||||
|             R.string.upload_map_location_access); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onRequestPermissionsResult(final int requestCode, |  | ||||||
|         @NonNull final String[] permissions, |  | ||||||
|         @NonNull final int[] grantResults) { |  | ||||||
|         if (requestCode == Constants.RequestCodes.LOCATION |  | ||||||
|             && grantResults[0] == PackageManager.PERMISSION_GRANTED) { |  | ||||||
|             onLocationPermissionGranted(); |  | ||||||
|         } else { |  | ||||||
|             onLocationPermissionDenied(getString(R.string.upload_map_location_access)); |  | ||||||
|         } |  | ||||||
|         super.onRequestPermissionsResult(requestCode, permissions, grantResults); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onResume() { |  | ||||||
|         super.onResume(); |  | ||||||
|         mapView.onResume(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onPause() { |  | ||||||
|         super.onPause(); |  | ||||||
|         mapView.onPause(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onLocationPermissionDenied(String toastMessage) { |  | ||||||
|         if (!ActivityCompat.shouldShowRequestPermissionRationale(this, |  | ||||||
|             permission.ACCESS_FINE_LOCATION)) { |  | ||||||
|             if (!locationPermissionsHelper.checkLocationPermission(this)) { |  | ||||||
|                 if (store.getBoolean("isPermissionDenied", false)) { |  | ||||||
|                     // means user has denied location permission twice or checked the "Don't show again" |  | ||||||
|                     locationPermissionsHelper.showAppSettingsDialog(this, |  | ||||||
|                         R.string.upload_map_location_access); |  | ||||||
|                 } else { |  | ||||||
|                     Toast.makeText(getBaseContext(), toastMessage, Toast.LENGTH_LONG).show(); |  | ||||||
|                 } |  | ||||||
|                 store.putBoolean("isPermissionDenied", true); |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             Toast.makeText(getBaseContext(), toastMessage, Toast.LENGTH_LONG).show(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onLocationPermissionGranted() { |  | ||||||
|         if (moveToCurrentLocation || !(activity.equals("MediaActivity"))) { |  | ||||||
|             if (locationPermissionsHelper.isLocationAccessToAppsTurnedOn()) { |  | ||||||
|                 locationManager.requestLocationUpdatesFromProvider( |  | ||||||
|                     LocationManager.NETWORK_PROVIDER); |  | ||||||
|                 locationManager.requestLocationUpdatesFromProvider(LocationManager.GPS_PROVIDER); |  | ||||||
|                 addMarkerAtGPSLocation(); |  | ||||||
|             } else { |  | ||||||
|                 addMarkerAtGPSLocation(); |  | ||||||
|                 locationPermissionsHelper.showLocationOffDialog(this, |  | ||||||
|                     R.string.ask_to_turn_location_on_text); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Adds a marker to the map at the most recent GPS location |  | ||||||
|      * (which may be the current GPS location). |  | ||||||
|      */ |  | ||||||
|     private void addMarkerAtGPSLocation() { |  | ||||||
|         fr.free.nrw.commons.location.LatLng currLocation = locationManager.getLastLocation(); |  | ||||||
|         if (currLocation != null) { |  | ||||||
|             GeoPoint currLocationGeopoint = new GeoPoint(currLocation.getLatitude(), |  | ||||||
|                 currLocation.getLongitude()); |  | ||||||
|             addLocationMarker(currLocationGeopoint); |  | ||||||
|             markerImage.setTranslationY(0); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void addLocationMarker(GeoPoint geoPoint) { |  | ||||||
|         if (moveToCurrentLocation) { |  | ||||||
|             mapView.getOverlays().clear(); |  | ||||||
|         } |  | ||||||
|         ScaleDiskOverlay diskOverlay = |  | ||||||
|             new ScaleDiskOverlay(this, |  | ||||||
|                 geoPoint, 2000, GeoConstants.UnitOfMeasure.foot); |  | ||||||
|         Paint circlePaint = new Paint(); |  | ||||||
|         circlePaint.setColor(Color.rgb(128, 128, 128)); |  | ||||||
|         circlePaint.setStyle(Paint.Style.STROKE); |  | ||||||
|         circlePaint.setStrokeWidth(2f); |  | ||||||
|         diskOverlay.setCirclePaint2(circlePaint); |  | ||||||
|         Paint diskPaint = new Paint(); |  | ||||||
|         diskPaint.setColor(Color.argb(40, 128, 128, 128)); |  | ||||||
|         diskPaint.setStyle(Paint.Style.FILL_AND_STROKE); |  | ||||||
|         diskOverlay.setCirclePaint1(diskPaint); |  | ||||||
|         diskOverlay.setDisplaySizeMin(900); |  | ||||||
|         diskOverlay.setDisplaySizeMax(1700); |  | ||||||
|         mapView.getOverlays().add(diskOverlay); |  | ||||||
|         org.osmdroid.views.overlay.Marker startMarker = new org.osmdroid.views.overlay.Marker( |  | ||||||
|             mapView); |  | ||||||
|         startMarker.setPosition(geoPoint); |  | ||||||
|         startMarker.setAnchor(org.osmdroid.views.overlay.Marker.ANCHOR_CENTER, |  | ||||||
|             org.osmdroid.views.overlay.Marker.ANCHOR_BOTTOM); |  | ||||||
|         startMarker.setIcon( |  | ||||||
|             ContextCompat.getDrawable(this, R.drawable.current_location_marker)); |  | ||||||
|         startMarker.setTitle("Your Location"); |  | ||||||
|         startMarker.setTextLabelFontSize(24); |  | ||||||
|         mapView.getOverlays().add(startMarker); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Saves the state of the activity |  | ||||||
|      * @param outState Bundle |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void onSaveInstanceState(@NonNull final Bundle outState) { |  | ||||||
|         super.onSaveInstanceState(outState); |  | ||||||
|         if(cameraPosition!=null){ |  | ||||||
|             outState.putParcelable(CAMERA_POS, cameraPosition); |  | ||||||
|         } |  | ||||||
|         if(activity!=null){ |  | ||||||
|             outState.putString(ACTIVITY, activity); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if(media!=null){ |  | ||||||
|             outState.putParcelable("sMedia", media); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,20 +0,0 @@ | ||||||
| package fr.free.nrw.commons.LocationPicker; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Constants need for location picking |  | ||||||
|  */ |  | ||||||
| public final class LocationPickerConstants { |  | ||||||
| 
 |  | ||||||
|     public static final String ACTIVITY_KEY |  | ||||||
|         = "location.picker.activity"; |  | ||||||
| 
 |  | ||||||
|     public static final String MAP_CAMERA_POSITION |  | ||||||
|         = "location.picker.cameraPosition"; |  | ||||||
| 
 |  | ||||||
|     public static final String MEDIA |  | ||||||
|         = "location.picker.media"; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     private LocationPickerConstants() { |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,63 +0,0 @@ | ||||||
| package fr.free.nrw.commons.LocationPicker; |  | ||||||
| 
 |  | ||||||
| import android.app.Application; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.lifecycle.AndroidViewModel; |  | ||||||
| import androidx.lifecycle.MutableLiveData; |  | ||||||
| import fr.free.nrw.commons.CameraPosition; |  | ||||||
| import org.jetbrains.annotations.NotNull; |  | ||||||
| import retrofit2.Call; |  | ||||||
| import retrofit2.Callback; |  | ||||||
| import retrofit2.Response; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Observes live camera position data |  | ||||||
|  */ |  | ||||||
| public class LocationPickerViewModel extends AndroidViewModel implements Callback<CameraPosition> { |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Wrapping CameraPosition with MutableLiveData |  | ||||||
|      */ |  | ||||||
|     private final MutableLiveData<CameraPosition> result = new MutableLiveData<>(); |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Constructor for this class |  | ||||||
|      * |  | ||||||
|      * @param application Application |  | ||||||
|      */ |  | ||||||
|     public LocationPickerViewModel(@NonNull final Application application) { |  | ||||||
|         super(application); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Responses on camera position changing |  | ||||||
|      * |  | ||||||
|      * @param call     Call<CameraPosition> |  | ||||||
|      * @param response Response<CameraPosition> |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void onResponse(final @NotNull Call<CameraPosition> call, |  | ||||||
|         final Response<CameraPosition> response) { |  | ||||||
|         if (response.body() == null) { |  | ||||||
|             result.setValue(null); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         result.setValue(response.body()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onFailure(final @NotNull Call<CameraPosition> call, final @NotNull Throwable t) { |  | ||||||
|         Timber.e(t); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets live CameraPosition |  | ||||||
|      * |  | ||||||
|      * @return MutableLiveData<CameraPosition> |  | ||||||
|      */ |  | ||||||
|     public MutableLiveData<CameraPosition> getResult() { |  | ||||||
|         return result; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
|  | @ -1,30 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.location.LatLng; |  | ||||||
| import fr.free.nrw.commons.nearby.Place; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| public abstract class MapController { |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * We pass this variable as a group of placeList and boundaryCoordinates |  | ||||||
|      */ |  | ||||||
|     public class NearbyPlacesInfo { |  | ||||||
|         public List<Place> placeList; // List of nearby places |  | ||||||
|         public LatLng[] boundaryCoordinates; // Corners of nearby area |  | ||||||
|         public LatLng currentLatLng; // Current location when this places are populated |  | ||||||
|         public LatLng searchLatLng; // Search location for finding this places |  | ||||||
|         public List<Media> mediaList; // Search location for finding this places |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * We pass this variable as a group of placeList and boundaryCoordinates |  | ||||||
|      */ |  | ||||||
|     public class ExplorePlacesInfo { |  | ||||||
|         public List<Place> explorePlaceList; // List of nearby places |  | ||||||
|         public LatLng[] boundaryCoordinates; // Corners of nearby area |  | ||||||
|         public LatLng currentLatLng; // Current location when this places are populated |  | ||||||
|         public LatLng searchLatLng; // Search location for finding this places |  | ||||||
|         public List<Media> mediaList; // Search location for finding this places |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										46
									
								
								app/src/main/java/fr/free/nrw/commons/MapController.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								app/src/main/java/fr/free/nrw/commons/MapController.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,46 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | import fr.free.nrw.commons.location.LatLng | ||||||
|  | import fr.free.nrw.commons.nearby.Place | ||||||
|  | 
 | ||||||
|  | abstract class MapController { | ||||||
|  |     /** | ||||||
|  |      * We pass this variable as a group of placeList and boundaryCoordinates | ||||||
|  |      */ | ||||||
|  |     inner class NearbyPlacesInfo { | ||||||
|  |         @JvmField | ||||||
|  |         var placeList: List<Place> = emptyList() // List of nearby places | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var boundaryCoordinates: Array<LatLng> = emptyArray() // Corners of nearby area | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var currentLatLng: LatLng? = null // Current location when this places are populated | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var searchLatLng: LatLng? = null // Search location for finding this places | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var mediaList: List<Media>? = null // Search location for finding this places | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * We pass this variable as a group of placeList and boundaryCoordinates | ||||||
|  |      */ | ||||||
|  |     inner class ExplorePlacesInfo { | ||||||
|  |         @JvmField | ||||||
|  |         var explorePlaceList: List<Place> = emptyList() // List of nearby places | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var boundaryCoordinates: Array<LatLng> = emptyArray() // Corners of nearby area | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var currentLatLng: LatLng? = null // Current location when this places are populated | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var searchLatLng: LatLng? = null // Search location for finding this places | ||||||
|  | 
 | ||||||
|  |         @JvmField | ||||||
|  |         var mediaList: List<Media> = emptyList() // Search location for finding this places | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,7 +1,9 @@ | ||||||
| package fr.free.nrw.commons | package fr.free.nrw.commons | ||||||
| 
 | 
 | ||||||
| import android.os.Parcelable | import android.os.Parcelable | ||||||
|  | import fr.free.nrw.commons.BuildConfig.COMMONS_URL | ||||||
| import fr.free.nrw.commons.location.LatLng | import fr.free.nrw.commons.location.LatLng | ||||||
|  | import fr.free.nrw.commons.wikidata.model.WikiSite | ||||||
| import fr.free.nrw.commons.wikidata.model.page.PageTitle | import fr.free.nrw.commons.wikidata.model.page.PageTitle | ||||||
| import kotlinx.parcelize.IgnoredOnParcel | import kotlinx.parcelize.IgnoredOnParcel | ||||||
| import kotlinx.parcelize.Parcelize | import kotlinx.parcelize.Parcelize | ||||||
|  | @ -28,9 +30,7 @@ class Media constructor( | ||||||
|      */ |      */ | ||||||
|     var filename: String? = null, |     var filename: String? = null, | ||||||
|     /** |     /** | ||||||
|      * Gets or sets the file description. |      * The fallback description of the file, used if no other description is provided. | ||||||
|      * @return file description as a string |  | ||||||
|      * @param fallbackDescription the new description of the file |  | ||||||
|      */ |      */ | ||||||
|     var fallbackDescription: String? = null, |     var fallbackDescription: String? = null, | ||||||
|     /** |     /** | ||||||
|  | @ -40,19 +40,25 @@ class Media constructor( | ||||||
|      */ |      */ | ||||||
|     var dateUploaded: Date? = null, |     var dateUploaded: Date? = null, | ||||||
|     /** |     /** | ||||||
|      * Gets or sets the license name of the file. |      * The license name of the file. | ||||||
|      * @return license as a String |  | ||||||
|      * @param license license name as a String |  | ||||||
|      */ |      */ | ||||||
|     var license: String? = null, |     var license: String? = null, | ||||||
|  |     /** | ||||||
|  |      * The URL corresponding to the license. | ||||||
|  |      */ | ||||||
|     var licenseUrl: String? = null, |     var licenseUrl: String? = null, | ||||||
|     /** |     /** | ||||||
|      * Gets or sets the name of the creator of the file. |      * The name of the creator of the file. | ||||||
|      * @return author name as a String |  | ||||||
|      * @param author creator name as a string |  | ||||||
|      */ |      */ | ||||||
|     var author: String? = null, |     var author: String? = null, | ||||||
|  |     /** | ||||||
|  |      * The username of the uploader. | ||||||
|  |      */ | ||||||
|     var user: String? = null, |     var user: String? = null, | ||||||
|  |     /** | ||||||
|  |      * The full name of the file's creator, if different from username. | ||||||
|  |      */ | ||||||
|  |     var creatorName: String? = null, | ||||||
|     /** |     /** | ||||||
|      * Gets the categories the file falls under. |      * Gets the categories the file falls under. | ||||||
|      * @return file categories as an ArrayList of Strings |      * @return file categories as an ArrayList of Strings | ||||||
|  | @ -66,6 +72,7 @@ class Media constructor( | ||||||
|     var captions: Map<String, String> = emptyMap(), |     var captions: Map<String, String> = emptyMap(), | ||||||
|     var descriptions: Map<String, String> = emptyMap(), |     var descriptions: Map<String, String> = emptyMap(), | ||||||
|     var depictionIds: List<String> = emptyList(), |     var depictionIds: List<String> = emptyList(), | ||||||
|  |     var creatorIds: List<String> = emptyList(), | ||||||
|     /** |     /** | ||||||
|      * This field was added to find non-hidden categories |      * This field was added to find non-hidden categories | ||||||
|      * Stores the mapping of category title to hidden attribute |      * Stores the mapping of category title to hidden attribute | ||||||
|  | @ -90,6 +97,68 @@ class Media constructor( | ||||||
|         captions = captions, |         captions = captions, | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|  |     constructor( | ||||||
|  |         captions: Map<String, String>, | ||||||
|  |         categories: List<String>?, | ||||||
|  |         filename: String?, | ||||||
|  |         fallbackDescription: String?, | ||||||
|  |         author: String?, | ||||||
|  |         user: String?, | ||||||
|  |         dateUploaded: Date? = Date(), | ||||||
|  |         license: String? = null, | ||||||
|  |         licenseUrl: String? = null, | ||||||
|  |         imageUrl: String? = null, | ||||||
|  |         thumbUrl: String? = null, | ||||||
|  |         coordinates: LatLng? = null, | ||||||
|  |         descriptions: Map<String, String> = emptyMap(), | ||||||
|  |         depictionIds: List<String> = emptyList(), | ||||||
|  |         categoriesHiddenStatus: Map<String, Boolean> = emptyMap() | ||||||
|  |     ) : this( | ||||||
|  |         pageId = UUID.randomUUID().toString(), | ||||||
|  |         filename = filename, | ||||||
|  |         fallbackDescription = fallbackDescription, | ||||||
|  |         dateUploaded = dateUploaded, | ||||||
|  |         author = author, | ||||||
|  |         user = user, | ||||||
|  |         categories = categories, | ||||||
|  |         captions = captions, | ||||||
|  |         license = license, | ||||||
|  |         licenseUrl = licenseUrl, | ||||||
|  |         imageUrl = imageUrl, | ||||||
|  |         thumbUrl = thumbUrl, | ||||||
|  |         coordinates = coordinates, | ||||||
|  |         descriptions = descriptions, | ||||||
|  |         depictionIds = depictionIds, | ||||||
|  |         categoriesHiddenStatus = categoriesHiddenStatus | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns Author if it's not null or empty, otherwise | ||||||
|  |      * returns user | ||||||
|  |      * @return Author or User | ||||||
|  |      */ | ||||||
|  |     @Deprecated("Use user for uploader username. Use attributedAuthor() for attribution. Note that the uploader may not be the creator/author.") | ||||||
|  |     fun getAuthorOrUser(): String? { | ||||||
|  |         return if (!author.isNullOrEmpty()) { | ||||||
|  |             author | ||||||
|  |         } else{ | ||||||
|  |             user | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns author if it's not null or empty, otherwise | ||||||
|  |      * returns creator name | ||||||
|  |      * @return name of author or creator | ||||||
|  |      */ | ||||||
|  |     fun getAttributedAuthor(): String? { | ||||||
|  |         return if (!author.isNullOrEmpty()) { | ||||||
|  |             author | ||||||
|  |         } else{ | ||||||
|  |             creatorName | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Gets media display title |      * Gets media display title | ||||||
|      * @return Media title |      * @return Media title | ||||||
|  | @ -106,7 +175,8 @@ class Media constructor( | ||||||
|      * Gets file page title |      * Gets file page title | ||||||
|      * @return New media page title |      * @return New media page title | ||||||
|      */ |      */ | ||||||
|     val pageTitle: PageTitle get() = Utils.getPageTitle(filename!!) |     val pageTitle: PageTitle | ||||||
|  |         get() = PageTitle(filename!!, WikiSite(COMMONS_URL)) | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Returns wikicode to use the media file on a MediaWiki site |      * Returns wikicode to use the media file on a MediaWiki site | ||||||
|  |  | ||||||
|  | @ -1,7 +1,7 @@ | ||||||
| package fr.free.nrw.commons | package fr.free.nrw.commons | ||||||
| 
 | 
 | ||||||
| import androidx.core.text.HtmlCompat | import androidx.core.text.HtmlCompat | ||||||
| import fr.free.nrw.commons.media.IdAndCaptions | import fr.free.nrw.commons.media.IdAndLabels | ||||||
| import fr.free.nrw.commons.media.MediaClient | import fr.free.nrw.commons.media.MediaClient | ||||||
| import fr.free.nrw.commons.media.PAGE_ID_PREFIX | import fr.free.nrw.commons.media.PAGE_ID_PREFIX | ||||||
| import io.reactivex.Single | import io.reactivex.Single | ||||||
|  | @ -23,13 +23,23 @@ class MediaDataExtractor | ||||||
|         private val mediaClient: MediaClient, |         private val mediaClient: MediaClient, | ||||||
|     ) { |     ) { | ||||||
|         fun fetchDepictionIdsAndLabels(media: Media) = |         fun fetchDepictionIdsAndLabels(media: Media) = | ||||||
|             mediaClient |                 mediaClient | ||||||
|                 .getEntities(media.depictionIds) |                 .getEntities(media.depictionIds) | ||||||
|                 .map { |                 .map { | ||||||
|                     it |                     it | ||||||
|                         .entities() |                         .entities() | ||||||
|                         .mapValues { entry -> entry.value.labels().mapValues { it.value.value() } } |                         .mapValues { entry -> entry.value.labels().mapValues { it.value.value() } } | ||||||
|                 }.map { it.map { (key, value) -> IdAndCaptions(key, value) } } |                 }.map { it.map { (key, value) -> IdAndLabels(key, value) } } | ||||||
|  |                 .onErrorReturn { emptyList() } | ||||||
|  | 
 | ||||||
|  |         fun fetchCreatorIdsAndLabels(media: Media) = | ||||||
|  |             mediaClient | ||||||
|  |                 .getEntities(media.creatorIds) | ||||||
|  |                 .map { | ||||||
|  |                     it | ||||||
|  |                         .entities() | ||||||
|  |                         .mapValues { entry -> entry.value.labels().mapValues { it.value.value() } } | ||||||
|  |                 }.map { it.map { (key, value) -> IdAndLabels(key, value) } } | ||||||
|                 .onErrorReturn { emptyList() } |                 .onErrorReturn { emptyList() } | ||||||
| 
 | 
 | ||||||
|         fun checkDeletionRequestExists(media: Media) = mediaClient.checkPageExistsUsingTitle("Commons:Deletion_requests/" + media.filename) |         fun checkDeletionRequestExists(media: Media) = mediaClient.checkPageExistsUsingTitle("Commons:Deletion_requests/" + media.filename) | ||||||
|  |  | ||||||
|  | @ -1,8 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Base interface for all the views |  | ||||||
|  */ |  | ||||||
| public interface MvpView { |  | ||||||
|     void showMessage(String message); |  | ||||||
| } |  | ||||||
|  | @ -1,154 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import fr.free.nrw.commons.wikidata.cookies.CommonsCookieJar; |  | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.TimeUnit; |  | ||||||
| import okhttp3.Cache; |  | ||||||
| import okhttp3.Interceptor; |  | ||||||
| import okhttp3.OkHttpClient; |  | ||||||
| import okhttp3.Request; |  | ||||||
| import okhttp3.Response; |  | ||||||
| import okhttp3.ResponseBody; |  | ||||||
| import okhttp3.logging.HttpLoggingInterceptor; |  | ||||||
| import okhttp3.logging.HttpLoggingInterceptor.Level; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| public final class OkHttpConnectionFactory { |  | ||||||
|     private static final String CACHE_DIR_NAME = "okhttp-cache"; |  | ||||||
|     private static final long NET_CACHE_SIZE = 64 * 1024 * 1024; |  | ||||||
| 
 |  | ||||||
|     public static OkHttpClient CLIENT; |  | ||||||
| 
 |  | ||||||
|     @NonNull public static OkHttpClient getClient(final CommonsCookieJar cookieJar) { |  | ||||||
|         if (CLIENT == null) { |  | ||||||
|             CLIENT = createClient(cookieJar); |  | ||||||
|         } |  | ||||||
|         return CLIENT; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @NonNull |  | ||||||
|     private static OkHttpClient createClient(final CommonsCookieJar cookieJar) { |  | ||||||
|         return new OkHttpClient.Builder() |  | ||||||
|                 .cookieJar(cookieJar) |  | ||||||
|                 .cache((CommonsApplication.getInstance()!=null) ? new Cache(new File(CommonsApplication.getInstance().getCacheDir(), CACHE_DIR_NAME), NET_CACHE_SIZE) : null) |  | ||||||
|                 .connectTimeout(120, TimeUnit.SECONDS) |  | ||||||
|                 .writeTimeout(120, TimeUnit.SECONDS) |  | ||||||
|                 .readTimeout(120, TimeUnit.SECONDS) |  | ||||||
|                 .addInterceptor(getLoggingInterceptor()) |  | ||||||
|                 .addInterceptor(new UnsuccessfulResponseInterceptor()) |  | ||||||
|                 .addInterceptor(new CommonHeaderRequestInterceptor()) |  | ||||||
|                 .build(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static HttpLoggingInterceptor getLoggingInterceptor() { |  | ||||||
|         final HttpLoggingInterceptor httpLoggingInterceptor = new HttpLoggingInterceptor() |  | ||||||
|             .setLevel(Level.BASIC); |  | ||||||
| 
 |  | ||||||
|         httpLoggingInterceptor.redactHeader("Authorization"); |  | ||||||
|         httpLoggingInterceptor.redactHeader("Cookie"); |  | ||||||
| 
 |  | ||||||
|         return httpLoggingInterceptor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static class CommonHeaderRequestInterceptor implements Interceptor { |  | ||||||
| 
 |  | ||||||
|         @Override |  | ||||||
|         @NonNull |  | ||||||
|         public Response intercept(@NonNull final Chain chain) throws IOException { |  | ||||||
|             final Request request = chain.request().newBuilder() |  | ||||||
|                     .header("User-Agent", CommonsApplication.getInstance().getUserAgent()) |  | ||||||
|                     .build(); |  | ||||||
|             return chain.proceed(request); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public static class UnsuccessfulResponseInterceptor implements Interceptor { |  | ||||||
|         private static final String SUPPRESS_ERROR_LOG = "x-commons-suppress-error-log"; |  | ||||||
|         public static final String SUPPRESS_ERROR_LOG_HEADER = SUPPRESS_ERROR_LOG+": true"; |  | ||||||
|         private static final List<String> DO_NOT_INTERCEPT = Collections.singletonList( |  | ||||||
|             "api.php?format=json&formatversion=2&errorformat=plaintext&action=upload&ignorewarnings=1"); |  | ||||||
| 
 |  | ||||||
|         private static final String ERRORS_PREFIX = "{\"error"; |  | ||||||
| 
 |  | ||||||
|         @Override |  | ||||||
|         @NonNull |  | ||||||
|         public Response intercept(@NonNull final Chain chain) throws IOException { |  | ||||||
|             final Request rq = chain.request(); |  | ||||||
| 
 |  | ||||||
|             // If the request contains our special "suppress errors" header, make note of it |  | ||||||
|             // but don't pass that on to the server. |  | ||||||
|             final boolean suppressErrors = rq.headers().names().contains(SUPPRESS_ERROR_LOG); |  | ||||||
|             final Request request = rq.newBuilder() |  | ||||||
|                 .removeHeader(SUPPRESS_ERROR_LOG) |  | ||||||
|                 .build(); |  | ||||||
| 
 |  | ||||||
|             final Response rsp = chain.proceed(request); |  | ||||||
| 
 |  | ||||||
|             // Do not intercept certain requests and let the caller handle the errors |  | ||||||
|             if(isExcludedUrl(chain.request())) { |  | ||||||
|                 return rsp; |  | ||||||
|             } |  | ||||||
|             if (rsp.isSuccessful()) { |  | ||||||
|                 try (final ResponseBody responseBody = rsp.peekBody(ERRORS_PREFIX.length())) { |  | ||||||
|                     if (ERRORS_PREFIX.equals(responseBody.string())) { |  | ||||||
|                         try (final ResponseBody body = rsp.body()) { |  | ||||||
|                             throw new IOException(body.string()); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } catch (final IOException e) { |  | ||||||
|                     // Log the error as debug (and therefore, "expected") or at error level |  | ||||||
|                     if (suppressErrors) { |  | ||||||
|                         Timber.d(e, "Suppressed (known / expected) error"); |  | ||||||
|                     } else { |  | ||||||
|                         Timber.e(e); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 return rsp; |  | ||||||
|             } |  | ||||||
|             throw new HttpStatusException(rsp); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private boolean isExcludedUrl(final Request request) { |  | ||||||
|             final String requestUrl = request.url().toString(); |  | ||||||
|             for(final String url: DO_NOT_INTERCEPT) { |  | ||||||
|                 if(requestUrl.contains(url)) { |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private OkHttpConnectionFactory() { |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public static class HttpStatusException extends IOException { |  | ||||||
|         private final int code; |  | ||||||
|         private final String url; |  | ||||||
|         public HttpStatusException(@NonNull Response rsp) { |  | ||||||
|             this.code = rsp.code(); |  | ||||||
|             this.url = rsp.request().url().uri().toString(); |  | ||||||
|             try { |  | ||||||
|                 if (rsp.body() != null && rsp.body().contentType() != null |  | ||||||
|                         && rsp.body().contentType().toString().contains("json")) { |  | ||||||
|                 } |  | ||||||
|             } catch (Exception e) { |  | ||||||
|                 // Log? |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public int code() { |  | ||||||
|             return code; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         @Override |  | ||||||
|         public String getMessage() { |  | ||||||
|             String str = "Code: " + code + ", URL: " + url; |  | ||||||
|             return str; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										135
									
								
								app/src/main/java/fr/free/nrw/commons/OkHttpConnectionFactory.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								app/src/main/java/fr/free/nrw/commons/OkHttpConnectionFactory.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,135 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | import androidx.annotation.VisibleForTesting | ||||||
|  | import fr.free.nrw.commons.wikidata.GsonUtil | ||||||
|  | import fr.free.nrw.commons.wikidata.cookies.CommonsCookieJar | ||||||
|  | import fr.free.nrw.commons.wikidata.mwapi.MwErrorResponse | ||||||
|  | import fr.free.nrw.commons.wikidata.mwapi.MwIOException | ||||||
|  | import fr.free.nrw.commons.wikidata.mwapi.MwLegacyServiceError | ||||||
|  | import okhttp3.Cache | ||||||
|  | import okhttp3.Interceptor | ||||||
|  | import okhttp3.OkHttpClient | ||||||
|  | import okhttp3.Request | ||||||
|  | import okhttp3.Response | ||||||
|  | import okhttp3.logging.HttpLoggingInterceptor | ||||||
|  | import timber.log.Timber | ||||||
|  | import java.io.File | ||||||
|  | import java.io.IOException | ||||||
|  | import java.util.concurrent.TimeUnit | ||||||
|  | 
 | ||||||
|  | object OkHttpConnectionFactory { | ||||||
|  |     private const val CACHE_DIR_NAME = "okhttp-cache" | ||||||
|  |     private const val NET_CACHE_SIZE = (64 * 1024 * 1024).toLong() | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     var CLIENT: OkHttpClient? = null | ||||||
|  | 
 | ||||||
|  |     fun getClient(cookieJar: CommonsCookieJar): OkHttpClient { | ||||||
|  |         if (CLIENT == null) { | ||||||
|  |             CLIENT = createClient(cookieJar) | ||||||
|  |         } | ||||||
|  |         return CLIENT!! | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun createClient(cookieJar: CommonsCookieJar): OkHttpClient { | ||||||
|  |         return OkHttpClient.Builder() | ||||||
|  |             .cookieJar(cookieJar) | ||||||
|  |             .cache( | ||||||
|  |                 if (CommonsApplication.instance != null) Cache( | ||||||
|  |                     File(CommonsApplication.instance.cacheDir, CACHE_DIR_NAME), | ||||||
|  |                     NET_CACHE_SIZE | ||||||
|  |                 ) else null | ||||||
|  |             ) | ||||||
|  |             .connectTimeout(120, TimeUnit.SECONDS) | ||||||
|  |             .writeTimeout(120, TimeUnit.SECONDS) | ||||||
|  |             .readTimeout(120, TimeUnit.SECONDS) | ||||||
|  |             .addInterceptor(HttpLoggingInterceptor().apply { | ||||||
|  |                 setLevel(HttpLoggingInterceptor.Level.BASIC) | ||||||
|  |                 redactHeader("Authorization") | ||||||
|  |                 redactHeader("Cookie") | ||||||
|  |             }) | ||||||
|  |             .addInterceptor(UnsuccessfulResponseInterceptor()) | ||||||
|  |             .addInterceptor(CommonHeaderRequestInterceptor()) | ||||||
|  |             .build() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | class CommonHeaderRequestInterceptor : Interceptor { | ||||||
|  |     @Throws(IOException::class) | ||||||
|  |     override fun intercept(chain: Interceptor.Chain): Response { | ||||||
|  |         val request = chain.request().newBuilder() | ||||||
|  |             .header("User-Agent", CommonsApplication.instance.userAgent) | ||||||
|  |             .build() | ||||||
|  |         return chain.proceed(request) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | private const val SUPPRESS_ERROR_LOG = "x-commons-suppress-error-log" | ||||||
|  | const val SUPPRESS_ERROR_LOG_HEADER: String = "$SUPPRESS_ERROR_LOG: true" | ||||||
|  | 
 | ||||||
|  | private class UnsuccessfulResponseInterceptor : Interceptor { | ||||||
|  |     @Throws(IOException::class) | ||||||
|  |     override fun intercept(chain: Interceptor.Chain): Response { | ||||||
|  |         val rq = chain.request() | ||||||
|  | 
 | ||||||
|  |         // If the request contains our special "suppress errors" header, make note of it | ||||||
|  |         // but don't pass that on to the server. | ||||||
|  |         val suppressErrors = rq.headers.names().contains(SUPPRESS_ERROR_LOG) | ||||||
|  |         val request = rq.newBuilder() | ||||||
|  |             .removeHeader(SUPPRESS_ERROR_LOG) | ||||||
|  |             .build() | ||||||
|  | 
 | ||||||
|  |         val rsp = chain.proceed(request) | ||||||
|  | 
 | ||||||
|  |         // Do not intercept certain requests and let the caller handle the errors | ||||||
|  |         if (isExcludedUrl(chain.request())) { | ||||||
|  |             return rsp | ||||||
|  |         } | ||||||
|  |         if (rsp.isSuccessful) { | ||||||
|  |             try { | ||||||
|  |                 rsp.peekBody(ERRORS_PREFIX.length.toLong()).use { responseBody -> | ||||||
|  |                     if (ERRORS_PREFIX == responseBody.string()) { | ||||||
|  |                         rsp.body.use { body -> | ||||||
|  |                             val bodyString = body!!.string() | ||||||
|  | 
 | ||||||
|  |                             throw MwIOException( | ||||||
|  |                                 "MediaWiki API returned error: $bodyString", | ||||||
|  |                                 GsonUtil.defaultGson.fromJson( | ||||||
|  |                                     bodyString, | ||||||
|  |                                     MwErrorResponse::class.java | ||||||
|  |                                 ).error!!, | ||||||
|  |                             ) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } catch (e: MwIOException) { | ||||||
|  |                 // Log the error as debug (and therefore, "expected") or at error level | ||||||
|  |                 if (suppressErrors) { | ||||||
|  |                     Timber.d(e, "Suppressed (known / expected) error") | ||||||
|  |                 } else { | ||||||
|  |                     Timber.e(e) | ||||||
|  |                     throw e | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return rsp | ||||||
|  |         } | ||||||
|  |         throw IOException("Unsuccessful response") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun isExcludedUrl(request: Request): Boolean { | ||||||
|  |         val requestUrl = request.url.toString() | ||||||
|  |         for (url in DO_NOT_INTERCEPT) { | ||||||
|  |             if (requestUrl.contains(url)) { | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         val DO_NOT_INTERCEPT = listOf( | ||||||
|  |             "api.php?format=json&formatversion=2&errorformat=plaintext&action=upload&ignorewarnings=1" | ||||||
|  |         ) | ||||||
|  |         const val ERRORS_PREFIX = "{\"error" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,250 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import android.content.ClipData; |  | ||||||
| import android.content.ClipboardManager; |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.graphics.Bitmap; |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.text.SpannableString; |  | ||||||
| import android.text.style.UnderlineSpan; |  | ||||||
| import android.view.View; |  | ||||||
| import android.widget.TextView; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.browser.customtabs.CustomTabColorSchemeParams; |  | ||||||
| import androidx.browser.customtabs.CustomTabsIntent; |  | ||||||
| import androidx.core.content.ContextCompat; |  | ||||||
| 
 |  | ||||||
| import java.util.Calendar; |  | ||||||
| import java.util.Date; |  | ||||||
| import fr.free.nrw.commons.wikidata.model.WikiSite; |  | ||||||
| import fr.free.nrw.commons.wikidata.model.page.PageTitle; |  | ||||||
| 
 |  | ||||||
| import java.util.Locale; |  | ||||||
| import java.util.regex.Pattern; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.location.LatLng; |  | ||||||
| import fr.free.nrw.commons.settings.Prefs; |  | ||||||
| import fr.free.nrw.commons.utils.ViewUtil; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| public class Utils { |  | ||||||
| 
 |  | ||||||
|     public static PageTitle getPageTitle(@NonNull String title) { |  | ||||||
|         return new PageTitle(title, new WikiSite(BuildConfig.COMMONS_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Generates licence name with given ID |  | ||||||
|      * @param license License ID |  | ||||||
|      * @return Name of license |  | ||||||
|      */ |  | ||||||
|     public static int licenseNameFor(String license) { |  | ||||||
|         switch (license) { |  | ||||||
|             case Prefs.Licenses.CC_BY_3: |  | ||||||
|                 return R.string.license_name_cc_by; |  | ||||||
|             case Prefs.Licenses.CC_BY_4: |  | ||||||
|                 return R.string.license_name_cc_by_four; |  | ||||||
|             case Prefs.Licenses.CC_BY_SA_3: |  | ||||||
|                 return R.string.license_name_cc_by_sa; |  | ||||||
|             case Prefs.Licenses.CC_BY_SA_4: |  | ||||||
|                 return R.string.license_name_cc_by_sa_four; |  | ||||||
|             case Prefs.Licenses.CC0: |  | ||||||
|                 return R.string.license_name_cc0; |  | ||||||
|         } |  | ||||||
|         throw new IllegalStateException("Unrecognized license value: " + license); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Generates license url with given ID |  | ||||||
|      * @param license License ID |  | ||||||
|      * @return Url of license |  | ||||||
|      */ |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     @NonNull |  | ||||||
|     public static String licenseUrlFor(String license) { |  | ||||||
|         switch (license) { |  | ||||||
|             case Prefs.Licenses.CC_BY_3: |  | ||||||
|                 return "https://creativecommons.org/licenses/by/3.0/"; |  | ||||||
|             case Prefs.Licenses.CC_BY_4: |  | ||||||
|                 return "https://creativecommons.org/licenses/by/4.0/"; |  | ||||||
|             case Prefs.Licenses.CC_BY_SA_3: |  | ||||||
|                 return "https://creativecommons.org/licenses/by-sa/3.0/"; |  | ||||||
|             case Prefs.Licenses.CC_BY_SA_4: |  | ||||||
|                 return "https://creativecommons.org/licenses/by-sa/4.0/"; |  | ||||||
|             case Prefs.Licenses.CC0: |  | ||||||
|                 return "https://creativecommons.org/publicdomain/zero/1.0/"; |  | ||||||
|             default: |  | ||||||
|                 throw new IllegalStateException("Unrecognized license value: " + license); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Adds extension to filename. Converts to .jpg if system provides .jpeg, adds .jpg if no extension detected |  | ||||||
|      * @param title File name |  | ||||||
|      * @param extension Correct extension |  | ||||||
|      * @return File with correct extension |  | ||||||
|      */ |  | ||||||
|     public static String fixExtension(String title, String extension) { |  | ||||||
|         Pattern jpegPattern = Pattern.compile("\\.jpeg$", Pattern.CASE_INSENSITIVE); |  | ||||||
| 
 |  | ||||||
|         // People are used to ".jpg" more than ".jpeg" which the system gives us. |  | ||||||
|         if (extension != null && extension.toLowerCase(Locale.ENGLISH).equals("jpeg")) { |  | ||||||
|             extension = "jpg"; |  | ||||||
|         } |  | ||||||
|         title = jpegPattern.matcher(title).replaceFirst(".jpg"); |  | ||||||
|         if (extension != null && !title.toLowerCase(Locale.getDefault()) |  | ||||||
|                 .endsWith("." + extension.toLowerCase(Locale.ENGLISH))) { |  | ||||||
|             title += "." + extension; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // If extension is still null, make it jpg. (Hotfix for https://github.com/commons-app/apps-android-commons/issues/228) |  | ||||||
|         // If title has an extension in it, if won't be true |  | ||||||
|         if (extension == null && title.lastIndexOf(".")<=0) { |  | ||||||
|            extension = "jpg"; |  | ||||||
|            title += "." + extension; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return title; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Launches intent to rate app |  | ||||||
|      * @param context |  | ||||||
|      */ |  | ||||||
|     public static void rateApp(Context context) { |  | ||||||
|         final String appPackageName = context.getPackageName(); |  | ||||||
|         try { |  | ||||||
|             context.startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse(Urls.PLAY_STORE_PREFIX + appPackageName))); |  | ||||||
|         } |  | ||||||
|         catch (android.content.ActivityNotFoundException anfe) { |  | ||||||
|             handleWebUrl(context, Uri.parse(Urls.PLAY_STORE_URL_PREFIX + appPackageName)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Opens Custom Tab Activity with in-app browser for the specified URL. |  | ||||||
|      * Launches intent for web URL |  | ||||||
|      * @param context |  | ||||||
|      * @param url |  | ||||||
|      */ |  | ||||||
|     public static void handleWebUrl(Context context, Uri url) { |  | ||||||
|         Timber.d("Launching web url %s", url.toString()); |  | ||||||
| 
 |  | ||||||
|         final CustomTabColorSchemeParams color = new CustomTabColorSchemeParams.Builder() |  | ||||||
|             .setToolbarColor(ContextCompat.getColor(context, R.color.primaryColor)) |  | ||||||
|             .setSecondaryToolbarColor(ContextCompat.getColor(context, R.color.primaryDarkColor)) |  | ||||||
|             .build(); |  | ||||||
| 
 |  | ||||||
|         CustomTabsIntent.Builder builder = new CustomTabsIntent.Builder(); |  | ||||||
|         builder.setDefaultColorSchemeParams(color); |  | ||||||
|         builder.setExitAnimations(context, android.R.anim.slide_in_left, android.R.anim.slide_out_right); |  | ||||||
|         CustomTabsIntent customTabsIntent = builder.build(); |  | ||||||
|         // Clear previous browser tasks, so that back/exit buttons work as intended. |  | ||||||
|         customTabsIntent.intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK); |  | ||||||
|         customTabsIntent.launchUrl(context, url); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Util function to handle geo coordinates |  | ||||||
|      * It no longer depends on google maps and any app capable of handling the map intent can handle it |  | ||||||
|      * @param context |  | ||||||
|      * @param latLng |  | ||||||
|      */ |  | ||||||
|     public static void handleGeoCoordinates(Context context, LatLng latLng) { |  | ||||||
|         Intent mapIntent = new Intent(Intent.ACTION_VIEW, latLng.getGmmIntentUri()); |  | ||||||
|         if (mapIntent.resolveActivity(context.getPackageManager()) != null) { |  | ||||||
|             context.startActivity(mapIntent); |  | ||||||
|         } else { |  | ||||||
|             ViewUtil.showShortToast(context, context.getString(R.string.map_application_missing)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * To take screenshot of the screen and return it in Bitmap format |  | ||||||
|      * |  | ||||||
|      * @param view |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     public static Bitmap getScreenShot(View view) { |  | ||||||
|         View screenView = view.getRootView(); |  | ||||||
|         screenView.setDrawingCacheEnabled(true); |  | ||||||
|         Bitmap drawingCache = screenView.getDrawingCache(); |  | ||||||
|         if (drawingCache != null) { |  | ||||||
|             Bitmap bitmap = Bitmap.createBitmap(drawingCache); |  | ||||||
|             screenView.setDrawingCacheEnabled(false); |  | ||||||
|             return bitmap; |  | ||||||
|         } |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /* |  | ||||||
|     *Copies the content to the clipboard |  | ||||||
|     * |  | ||||||
|     */ |  | ||||||
|     public static void copy(String label,String text, Context context){ |  | ||||||
|         ClipboardManager clipboard = (ClipboardManager) context.getSystemService(Context.CLIPBOARD_SERVICE); |  | ||||||
|         ClipData clip = ClipData.newPlainText(label, text); |  | ||||||
|         clipboard.setPrimaryClip(clip); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method sets underlined string text to a TextView |  | ||||||
|      * |  | ||||||
|      * @param textView TextView associated with string resource |  | ||||||
|      * @param stringResourceName string resource name |  | ||||||
|      * @param context |  | ||||||
|      */ |  | ||||||
|     public static void setUnderlinedText(TextView textView, int stringResourceName, Context context) { |  | ||||||
|         SpannableString content = new SpannableString(context.getString(stringResourceName)); |  | ||||||
|         content.setSpan(new UnderlineSpan(), 0, content.length(), 0); |  | ||||||
|         textView.setText(content); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * For now we are enabling the monuments only when the date lies between 1 Sept & 31 OCt |  | ||||||
|      * @param date |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     public static boolean isMonumentsEnabled(final Date date) { |  | ||||||
|         if (date.getMonth() == 8) { |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Util function to get the start date of wlm monument |  | ||||||
|      * For this release we are hardcoding it to be 1st September |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     public static String getWLMStartDate() { |  | ||||||
|         return "1 Sep"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*** |  | ||||||
|      * Util function to get the end date of wlm monument |  | ||||||
|      * For this release we are hardcoding it to be 31st October |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     public static String getWLMEndDate() { |  | ||||||
|         return "30 Sep"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*** |  | ||||||
|      * Function to get the current WLM year |  | ||||||
|      * It increments at the start of September in line with the other WLM functions |  | ||||||
|      * (No consideration of locales for now) |  | ||||||
|      * @param calendar |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     public static int getWikiLovesMonumentsYear(Calendar calendar) { |  | ||||||
|         int year = calendar.get(Calendar.YEAR); |  | ||||||
|         if (calendar.get(Calendar.MONTH) < Calendar.SEPTEMBER) { |  | ||||||
|             year -= 1; |  | ||||||
|         } |  | ||||||
|         return year; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import android.content.Context; |  | ||||||
| 
 |  | ||||||
| public interface ViewHolder<T> { |  | ||||||
|     void bindModel(Context context, T model); |  | ||||||
| } |  | ||||||
|  | @ -1,57 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import androidx.fragment.app.Fragment; |  | ||||||
| import androidx.fragment.app.FragmentManager; |  | ||||||
| import androidx.fragment.app.FragmentPagerAdapter; |  | ||||||
| 
 |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * This adapter will be used to display fragments in a ViewPager |  | ||||||
|  */ |  | ||||||
| public class ViewPagerAdapter extends FragmentPagerAdapter { |  | ||||||
|     private List<Fragment> fragmentList = new ArrayList<>(); |  | ||||||
|     private List<String> fragmentTitleList = new ArrayList<>(); |  | ||||||
| 
 |  | ||||||
|     public ViewPagerAdapter(FragmentManager manager) { |  | ||||||
|         super(manager); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method returns the fragment of the viewpager at a particular position |  | ||||||
|      * @param position |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Fragment getItem(int position) { |  | ||||||
|         return fragmentList.get(position); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method returns the total number of fragments in the viewpager. |  | ||||||
|      * @return size |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getCount() { |  | ||||||
|         return fragmentList.size(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method sets the fragment and title list in the viewpager |  | ||||||
|      * @param fragmentList List of all fragments to be displayed in the viewpager |  | ||||||
|      * @param fragmentTitleList List of all titles of the fragments |  | ||||||
|      */ |  | ||||||
|     public void setTabData(List<Fragment> fragmentList, List<String> fragmentTitleList) { |  | ||||||
|         this.fragmentList = fragmentList; |  | ||||||
|         this.fragmentTitleList = fragmentTitleList; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method returns the title of the page at a particular position |  | ||||||
|      * @param position |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public CharSequence getPageTitle(int position) { |  | ||||||
|         return fragmentTitleList.get(position); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										44
									
								
								app/src/main/java/fr/free/nrw/commons/ViewPagerAdapter.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								app/src/main/java/fr/free/nrw/commons/ViewPagerAdapter.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,44 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | import androidx.fragment.app.FragmentManager | ||||||
|  | import androidx.fragment.app.FragmentPagerAdapter | ||||||
|  | import java.util.Locale | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This adapter will be used to display fragments in a ViewPager | ||||||
|  |  */ | ||||||
|  | class ViewPagerAdapter : FragmentPagerAdapter { | ||||||
|  |     private val context: Context | ||||||
|  |     private var fragmentList: List<Fragment> = emptyList() | ||||||
|  |     private var fragmentTitleList: List<String> = emptyList() | ||||||
|  | 
 | ||||||
|  |     constructor(context: Context, manager: FragmentManager) : super(manager) { | ||||||
|  |         this.context = context | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     constructor(context: Context, manager: FragmentManager, behavior: Int) : super(manager, behavior) { | ||||||
|  |         this.context = context | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun getItem(position: Int): Fragment = fragmentList[position] | ||||||
|  | 
 | ||||||
|  |     override fun getPageTitle(position: Int): CharSequence = fragmentTitleList[position] | ||||||
|  | 
 | ||||||
|  |     override fun getCount(): Int = fragmentList.size | ||||||
|  | 
 | ||||||
|  |     fun setTabs(vararg titlesToFragments: Pair<Int, Fragment>) { | ||||||
|  |         // Enforce that every title must come from strings.xml and all will consistently be uppercase | ||||||
|  |         fragmentTitleList = titlesToFragments.map { | ||||||
|  |             context.getString(it.first).uppercase(Locale.ROOT) | ||||||
|  |         } | ||||||
|  |         fragmentList = titlesToFragments.map { it.second } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         // Convenience method for Java callers, can be removed when everything is migrated | ||||||
|  |         @JvmStatic | ||||||
|  |         fun pairOf(first: Int, second: Fragment) = first to second | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,108 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import android.app.AlertDialog; |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.view.View; |  | ||||||
| import fr.free.nrw.commons.databinding.ActivityWelcomeBinding; |  | ||||||
| import fr.free.nrw.commons.databinding.PopupForCopyrightBinding; |  | ||||||
| import fr.free.nrw.commons.quiz.QuizActivity; |  | ||||||
| import fr.free.nrw.commons.theme.BaseActivity; |  | ||||||
| import fr.free.nrw.commons.utils.ConfigUtils; |  | ||||||
| 
 |  | ||||||
| public class WelcomeActivity extends BaseActivity { |  | ||||||
| 
 |  | ||||||
|     private ActivityWelcomeBinding binding; |  | ||||||
|     private PopupForCopyrightBinding copyrightBinding; |  | ||||||
| 
 |  | ||||||
|     private final WelcomePagerAdapter adapter = new WelcomePagerAdapter(); |  | ||||||
|     private boolean isQuiz; |  | ||||||
|     private AlertDialog.Builder dialogBuilder; |  | ||||||
|     private AlertDialog dialog; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Initialises exiting fields and dependencies |  | ||||||
|      * |  | ||||||
|      * @param savedInstanceState WelcomeActivity bundled data |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void onCreate(final Bundle savedInstanceState) { |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
|         binding = ActivityWelcomeBinding.inflate(getLayoutInflater()); |  | ||||||
|         final View view = binding.getRoot(); |  | ||||||
|         setContentView(view); |  | ||||||
| 
 |  | ||||||
|         if (getIntent() != null) { |  | ||||||
|             final Bundle bundle = getIntent().getExtras(); |  | ||||||
|             if (bundle != null) { |  | ||||||
|                 isQuiz = bundle.getBoolean("isQuiz"); |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             isQuiz = false; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // Enable skip button if beta flavor |  | ||||||
|         if (ConfigUtils.isBetaFlavour()) { |  | ||||||
|             binding.finishTutorialButton.setVisibility(View.VISIBLE); |  | ||||||
| 
 |  | ||||||
|             dialogBuilder = new AlertDialog.Builder(this); |  | ||||||
|             copyrightBinding = PopupForCopyrightBinding.inflate(getLayoutInflater()); |  | ||||||
|             final View contactPopupView = copyrightBinding.getRoot(); |  | ||||||
|             dialogBuilder.setView(contactPopupView); |  | ||||||
|             dialog = dialogBuilder.create(); |  | ||||||
|             dialog.show(); |  | ||||||
| 
 |  | ||||||
|             copyrightBinding.buttonOk.setOnClickListener(v -> dialog.dismiss()); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         binding.welcomePager.setAdapter(adapter); |  | ||||||
|         binding.welcomePagerIndicator.setViewPager(binding.welcomePager); |  | ||||||
| 
 |  | ||||||
|         binding.finishTutorialButton.setOnClickListener(v -> finishTutorial()); |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * References WelcomePageAdapter to null before the activity is destroyed |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void onDestroy() { |  | ||||||
|         if (isQuiz) { |  | ||||||
|             final Intent i = new Intent(this, QuizActivity.class); |  | ||||||
|             startActivity(i); |  | ||||||
|         } |  | ||||||
|         super.onDestroy(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Creates a way to change current activity to WelcomeActivity |  | ||||||
|      * |  | ||||||
|      * @param context Activity context |  | ||||||
|      */ |  | ||||||
|     public static void startYourself(final Context context) { |  | ||||||
|         final Intent welcomeIntent = new Intent(context, WelcomeActivity.class); |  | ||||||
|         context.startActivity(welcomeIntent); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Override onBackPressed() to go to previous tutorial 'pages' if not on first page |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void onBackPressed() { |  | ||||||
|         if (binding.welcomePager.getCurrentItem() != 0) { |  | ||||||
|             binding.welcomePager.setCurrentItem(binding.welcomePager.getCurrentItem() - 1, true); |  | ||||||
|         } else { |  | ||||||
|             if (defaultKvStore.getBoolean("firstrun", true)) { |  | ||||||
|                 finishAffinity(); |  | ||||||
|             } else { |  | ||||||
|                 super.onBackPressed(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void finishTutorial() { |  | ||||||
|         defaultKvStore.putBoolean("firstrun", false); |  | ||||||
|         finish(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										80
									
								
								app/src/main/java/fr/free/nrw/commons/WelcomeActivity.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								app/src/main/java/fr/free/nrw/commons/WelcomeActivity.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,80 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | import android.app.AlertDialog | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.Intent | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.View | ||||||
|  | import fr.free.nrw.commons.databinding.ActivityWelcomeBinding | ||||||
|  | import fr.free.nrw.commons.databinding.PopupForCopyrightBinding | ||||||
|  | import fr.free.nrw.commons.quiz.QuizActivity | ||||||
|  | import fr.free.nrw.commons.theme.BaseActivity | ||||||
|  | import fr.free.nrw.commons.utils.applyEdgeToEdgeAllInsets | ||||||
|  | import fr.free.nrw.commons.utils.ConfigUtils.isBetaFlavour | ||||||
|  | 
 | ||||||
|  | class WelcomeActivity : BaseActivity() { | ||||||
|  |     private var binding: ActivityWelcomeBinding? = null | ||||||
|  |     private var isQuiz = false | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Initialises exiting fields and dependencies | ||||||
|  |      * | ||||||
|  |      * @param savedInstanceState WelcomeActivity bundled data | ||||||
|  |      */ | ||||||
|  |     public override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |         binding = ActivityWelcomeBinding.inflate(layoutInflater) | ||||||
|  |         applyEdgeToEdgeAllInsets(binding!!.welcomePager.rootView) | ||||||
|  |         setContentView(binding!!.root) | ||||||
|  | 
 | ||||||
|  |         isQuiz = intent?.extras?.getBoolean("isQuiz", false) ?: false | ||||||
|  | 
 | ||||||
|  |         // Enable skip button if beta flavor | ||||||
|  |         if (isBetaFlavour) { | ||||||
|  |             binding!!.finishTutorialButton.visibility = View.VISIBLE | ||||||
|  | 
 | ||||||
|  |             val copyrightBinding = PopupForCopyrightBinding.inflate(layoutInflater) | ||||||
|  | 
 | ||||||
|  |             val dialog = AlertDialog.Builder(this) | ||||||
|  |                 .setView(copyrightBinding.root) | ||||||
|  |                 .setCancelable(false) | ||||||
|  |                 .create() | ||||||
|  |             dialog.show() | ||||||
|  | 
 | ||||||
|  |             copyrightBinding.buttonOk.setOnClickListener { v: View? -> dialog.dismiss() } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val adapter = WelcomePagerAdapter() | ||||||
|  |         binding!!.welcomePager.adapter = adapter | ||||||
|  |         binding!!.welcomePagerIndicator.setViewPager(binding!!.welcomePager) | ||||||
|  |         binding!!.finishTutorialButton.setOnClickListener { v: View? -> finishTutorial() } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public override fun onDestroy() { | ||||||
|  |         if (isQuiz) { | ||||||
|  |             startActivity(Intent(this, QuizActivity::class.java)) | ||||||
|  |         } | ||||||
|  |         super.onDestroy() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onBackPressed() { | ||||||
|  |         if (binding!!.welcomePager.currentItem != 0) { | ||||||
|  |             binding!!.welcomePager.setCurrentItem(binding!!.welcomePager.currentItem - 1, true) | ||||||
|  |         } else { | ||||||
|  |             if (defaultKvStore.getBoolean("firstrun", true)) { | ||||||
|  |                 finishAffinity() | ||||||
|  |             } else { | ||||||
|  |                 super.onBackPressed() | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun finishTutorial() { | ||||||
|  |         defaultKvStore.putBoolean("firstrun", false) | ||||||
|  |         finish() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fun Context.startWelcome() { | ||||||
|  |     startActivity(Intent(this, WelcomeActivity::class.java)) | ||||||
|  | } | ||||||
|  | @ -1,74 +0,0 @@ | ||||||
| package fr.free.nrw.commons; |  | ||||||
| 
 |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.view.LayoutInflater; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.ViewGroup; |  | ||||||
| import android.widget.TextView; |  | ||||||
| 
 |  | ||||||
| import androidx.viewpager.widget.PagerAdapter; |  | ||||||
| 
 |  | ||||||
| public class WelcomePagerAdapter extends PagerAdapter { |  | ||||||
|     private static final int[] PAGE_LAYOUTS = new int[]{ |  | ||||||
|             R.layout.welcome_wikipedia, |  | ||||||
|             R.layout.welcome_do_upload, |  | ||||||
|             R.layout.welcome_dont_upload, |  | ||||||
|             R.layout.welcome_image_example, |  | ||||||
|             R.layout.welcome_final |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Gets total number of layouts |  | ||||||
|      * @return Number of layouts |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getCount() { |  | ||||||
|         return PAGE_LAYOUTS.length; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Compares given view with provided object |  | ||||||
|      * @param view Adapter view |  | ||||||
|      * @param object Adapter object |  | ||||||
|      * @return Equality between view and object |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public boolean isViewFromObject(View view, Object object) { |  | ||||||
|         return (view == object); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Object instantiateItem(ViewGroup container, int position) { |  | ||||||
|         LayoutInflater inflater = LayoutInflater.from(container.getContext()); |  | ||||||
|         ViewGroup layout = (ViewGroup) inflater.inflate(PAGE_LAYOUTS[position], container, false); |  | ||||||
| 
 |  | ||||||
|         // If final page |  | ||||||
|         if (position == PAGE_LAYOUTS.length - 1) { |  | ||||||
|             // Add link to more information |  | ||||||
|             TextView moreInfo = layout.findViewById(R.id.welcomeInfo); |  | ||||||
|             Utils.setUnderlinedText(moreInfo, R.string.welcome_help_button_text, container.getContext()); |  | ||||||
|             moreInfo.setOnClickListener(view -> Utils.handleWebUrl( |  | ||||||
|                     container.getContext(), |  | ||||||
|                     Uri.parse("https://commons.wikimedia.org/wiki/Help:Contents") |  | ||||||
|             )); |  | ||||||
| 
 |  | ||||||
|             // Handle click of finishTutorialButton ("YES!" button) inside layout |  | ||||||
|             layout.findViewById(R.id.finishTutorialButton) |  | ||||||
|                     .setOnClickListener(view -> ((WelcomeActivity) container.getContext()).finishTutorial()); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         container.addView(layout); |  | ||||||
|         return layout; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Provides a way to remove an item from container |  | ||||||
|      * @param container Adapter view group container |  | ||||||
|      * @param position Index of item |  | ||||||
|      * @param obj Adapter object |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void destroyItem(ViewGroup container, int position, Object obj) { |  | ||||||
|         container.removeView((View) obj); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										70
									
								
								app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								app/src/main/java/fr/free/nrw/commons/WelcomePagerAdapter.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,70 @@ | ||||||
|  | package fr.free.nrw.commons | ||||||
|  | 
 | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import android.widget.TextView | ||||||
|  | import androidx.core.net.toUri | ||||||
|  | import androidx.viewpager.widget.PagerAdapter | ||||||
|  | import fr.free.nrw.commons.utils.UnderlineUtils.setUnderlinedText | ||||||
|  | import fr.free.nrw.commons.utils.handleWebUrl | ||||||
|  | 
 | ||||||
|  | class WelcomePagerAdapter : PagerAdapter() { | ||||||
|  |     /** | ||||||
|  |      * Gets total number of layouts | ||||||
|  |      * @return Number of layouts | ||||||
|  |      */ | ||||||
|  |     override fun getCount(): Int = PAGE_LAYOUTS.size | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Compares given view with provided object | ||||||
|  |      * @param view Adapter view | ||||||
|  |      * @param obj Adapter object | ||||||
|  |      * @return Equality between view and object | ||||||
|  |      */ | ||||||
|  |     override fun isViewFromObject(view: View, obj: Any): Boolean = (view === obj) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Provides a way to remove an item from container | ||||||
|  |      * @param container Adapter view group container | ||||||
|  |      * @param position Index of item | ||||||
|  |      * @param obj Adapter object | ||||||
|  |      */ | ||||||
|  |     override fun destroyItem(container: ViewGroup, position: Int, obj: Any) = | ||||||
|  |         container.removeView(obj as View) | ||||||
|  | 
 | ||||||
|  |     override fun instantiateItem(container: ViewGroup, position: Int): Any { | ||||||
|  |         val inflater = LayoutInflater.from(container.context) | ||||||
|  |         val layout = inflater.inflate(PAGE_LAYOUTS[position], container, false) as ViewGroup | ||||||
|  | 
 | ||||||
|  |         // If final page | ||||||
|  |         if (position == PAGE_LAYOUTS.size - 1) { | ||||||
|  |             // Add link to more information | ||||||
|  |             val moreInfo = layout.findViewById<TextView>(R.id.welcomeInfo) | ||||||
|  |             setUnderlinedText(moreInfo, R.string.welcome_help_button_text) | ||||||
|  |             moreInfo.setOnClickListener { | ||||||
|  |                 handleWebUrl( | ||||||
|  |                     container.context, | ||||||
|  |                     "https://commons.wikimedia.org/wiki/Help:Contents".toUri() | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // Handle click of finishTutorialButton ("YES!" button) inside layout | ||||||
|  |             layout.findViewById<View>(R.id.finishTutorialButton) | ||||||
|  |                 .setOnClickListener { view: View? -> (container.context as WelcomeActivity).finishTutorial() } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         container.addView(layout) | ||||||
|  |         return layout | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private val PAGE_LAYOUTS = intArrayOf( | ||||||
|  |             R.layout.welcome_wikipedia, | ||||||
|  |             R.layout.welcome_do_upload, | ||||||
|  |             R.layout.welcome_dont_upload, | ||||||
|  |             R.layout.welcome_image_example, | ||||||
|  |             R.layout.welcome_final | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -129,9 +129,10 @@ interface PageEditInterface { | ||||||
|     ): Observable<Entities> |     ): Observable<Entities> | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Get wiki text for provided file names |      * Gets the wiki text for the provided file name. | ||||||
|      * @param titles : Name of the file |      * | ||||||
|      * @return Single<MwQueryResult> |      * @param title The title (name) of the file to fetch wiki text for. | ||||||
|  |      * @return A Single emitting the wiki query response. | ||||||
|      */ |      */ | ||||||
|     @GET(MW_API_PREFIX + "action=query&prop=revisions&rvprop=content|timestamp&rvlimit=1&converttitles=") |     @GET(MW_API_PREFIX + "action=query&prop=revisions&rvprop=content|timestamp&rvlimit=1&converttitles=") | ||||||
|     fun getWikiText( |     fun getWikiText( | ||||||
|  |  | ||||||
|  | @ -3,7 +3,7 @@ package fr.free.nrw.commons.actions | ||||||
| import fr.free.nrw.commons.CommonsApplication | import fr.free.nrw.commons.CommonsApplication | ||||||
| import fr.free.nrw.commons.auth.csrf.CsrfTokenClient | import fr.free.nrw.commons.auth.csrf.CsrfTokenClient | ||||||
| import fr.free.nrw.commons.auth.csrf.InvalidLoginTokenException | import fr.free.nrw.commons.auth.csrf.InvalidLoginTokenException | ||||||
| import fr.free.nrw.commons.di.NetworkingModule.NAMED_COMMONS_CSRF | import fr.free.nrw.commons.di.NetworkingModule.Companion.NAMED_COMMONS_CSRF | ||||||
| import io.reactivex.Observable | import io.reactivex.Observable | ||||||
| import javax.inject.Inject | import javax.inject.Inject | ||||||
| import javax.inject.Named | import javax.inject.Named | ||||||
|  |  | ||||||
|  | @ -0,0 +1,218 @@ | ||||||
|  | package fr.free.nrw.commons.activity | ||||||
|  | 
 | ||||||
|  | import android.annotation.SuppressLint | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.Intent | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.webkit.ConsoleMessage | ||||||
|  | import android.webkit.CookieManager | ||||||
|  | import android.webkit.WebChromeClient | ||||||
|  | import android.webkit.WebResourceRequest | ||||||
|  | import android.webkit.WebView | ||||||
|  | import android.webkit.WebViewClient | ||||||
|  | import androidx.activity.ComponentActivity | ||||||
|  | import androidx.activity.compose.setContent | ||||||
|  | import androidx.activity.enableEdgeToEdge | ||||||
|  | import androidx.compose.foundation.layout.fillMaxSize | ||||||
|  | import androidx.compose.foundation.layout.padding | ||||||
|  | import androidx.compose.material.icons.Icons | ||||||
|  | import androidx.compose.material.icons.automirrored.filled.ArrowBack | ||||||
|  | import androidx.compose.material3.ExperimentalMaterial3Api | ||||||
|  | import androidx.compose.material3.Icon | ||||||
|  | import androidx.compose.material3.IconButton | ||||||
|  | import androidx.compose.material3.Scaffold | ||||||
|  | import androidx.compose.material3.Text | ||||||
|  | import androidx.compose.material3.TopAppBar | ||||||
|  | import androidx.compose.runtime.Composable | ||||||
|  | import androidx.compose.runtime.mutableStateOf | ||||||
|  | import androidx.compose.runtime.remember | ||||||
|  | import androidx.compose.ui.Modifier | ||||||
|  | import androidx.compose.ui.viewinterop.AndroidView | ||||||
|  | import fr.free.nrw.commons.CommonsApplication | ||||||
|  | import fr.free.nrw.commons.CommonsApplication.ActivityLogoutListener | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.di.ApplicationlessInjection | ||||||
|  | import fr.free.nrw.commons.wikidata.cookies.CommonsCookieJar | ||||||
|  | import okhttp3.HttpUrl.Companion.toHttpUrl | ||||||
|  | import timber.log.Timber | ||||||
|  | import javax.inject.Inject | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * SingleWebViewActivity is a reusable activity webView based on a given url(initial url) and | ||||||
|  |  * closes itself when a specified success URL is reached to success url. | ||||||
|  |  */ | ||||||
|  | class SingleWebViewActivity : ComponentActivity() { | ||||||
|  |     @Inject | ||||||
|  |     lateinit var cookieJar: CommonsCookieJar | ||||||
|  | 
 | ||||||
|  |     @OptIn(ExperimentalMaterial3Api::class) | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |         val url = intent.getStringExtra(VANISH_ACCOUNT_URL) | ||||||
|  |         val successUrl = intent.getStringExtra(VANISH_ACCOUNT_SUCCESS_URL) | ||||||
|  |         if (url == null || successUrl == null) { | ||||||
|  |             finish() | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |         ApplicationlessInjection | ||||||
|  |             .getInstance(applicationContext) | ||||||
|  |             .commonsApplicationComponent | ||||||
|  |             .inject(this) | ||||||
|  |         setCookies(url) | ||||||
|  |         enableEdgeToEdge() | ||||||
|  |         setContent { | ||||||
|  |             Scaffold( | ||||||
|  |                 topBar = { | ||||||
|  |                     TopAppBar( | ||||||
|  |                         modifier = Modifier, | ||||||
|  |                         title = { Text(getString(R.string.vanish_account)) }, | ||||||
|  |                         navigationIcon = { | ||||||
|  |                             IconButton( | ||||||
|  |                                 onClick = { | ||||||
|  |                                     // Close the WebView Activity if the user taps the back button | ||||||
|  |                                     finish() | ||||||
|  |                                 }, | ||||||
|  |                             ) { | ||||||
|  |                                 Icon( | ||||||
|  |                                     imageVector = Icons.AutoMirrored.Filled.ArrowBack, | ||||||
|  |                                     // TODO("Add contentDescription) | ||||||
|  |                                     contentDescription = "" | ||||||
|  |                                 ) | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     ) | ||||||
|  |                 }, | ||||||
|  |                 content = { | ||||||
|  |                     WebViewComponent( | ||||||
|  |                         url = url, | ||||||
|  |                         successUrl = successUrl, | ||||||
|  |                         onSuccess = { | ||||||
|  |                             //Redirect the user to login screen like we do when the user logout's | ||||||
|  |                             val app = applicationContext as CommonsApplication | ||||||
|  |                             app.clearApplicationData( | ||||||
|  |                                 applicationContext, | ||||||
|  |                                 ActivityLogoutListener(activity = this, ctx = applicationContext) | ||||||
|  |                             ) | ||||||
|  |                             finish() | ||||||
|  |                         }, | ||||||
|  |                         modifier = Modifier | ||||||
|  |                             .fillMaxSize() | ||||||
|  |                             .padding(it) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * @param url The initial URL which we are loading in the WebView. | ||||||
|  |      * @param successUrl The URL that, when reached, triggers the `onSuccess` callback. | ||||||
|  |      * @param onSuccess A callback that is invoked when the current url of webView is successUrl. | ||||||
|  |      * This is used when we want to close when the webView once a success url is hit. | ||||||
|  |      * @param modifier An optional [Modifier] to customize the layout or appearance of the WebView. | ||||||
|  |      */ | ||||||
|  |     @SuppressLint("SetJavaScriptEnabled") | ||||||
|  |     @Composable | ||||||
|  |     private fun WebViewComponent( | ||||||
|  |         url: String, | ||||||
|  |         successUrl: String, | ||||||
|  |         onSuccess: () -> Unit, | ||||||
|  |         modifier: Modifier = Modifier | ||||||
|  |     ) { | ||||||
|  |         val webView = remember { mutableStateOf<WebView?>(null) } | ||||||
|  |         AndroidView( | ||||||
|  |             modifier = modifier, | ||||||
|  |             factory = { | ||||||
|  |                 WebView(it).apply { | ||||||
|  |                     settings.apply { | ||||||
|  |                         javaScriptEnabled = true | ||||||
|  |                         domStorageEnabled = true | ||||||
|  |                         javaScriptCanOpenWindowsAutomatically = true | ||||||
|  | 
 | ||||||
|  |                     } | ||||||
|  |                     webViewClient = object : WebViewClient() { | ||||||
|  |                         override fun shouldOverrideUrlLoading( | ||||||
|  |                             view: WebView?, | ||||||
|  |                             request: WebResourceRequest? | ||||||
|  |                         ): Boolean { | ||||||
|  | 
 | ||||||
|  |                             request?.url?.let { url -> | ||||||
|  |                                 Timber.d("URL Loading: $url") | ||||||
|  |                                 if (url.toString() == successUrl) { | ||||||
|  |                                     Timber.d("Success URL detected. Closing WebView.") | ||||||
|  |                                     onSuccess() // Close the activity | ||||||
|  |                                     return true | ||||||
|  |                                 } | ||||||
|  |                                 return false | ||||||
|  |                             } | ||||||
|  |                             return false | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|  |                         override fun onPageFinished(view: WebView?, url: String?) { | ||||||
|  |                             super.onPageFinished(view, url) | ||||||
|  |                             setCookies(url.orEmpty()) | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     webChromeClient = object : WebChromeClient() { | ||||||
|  |                         override fun onConsoleMessage(message: ConsoleMessage): Boolean { | ||||||
|  |                             Timber.d("%s%s", | ||||||
|  |                                 "Console: ${message.message()} -- From line ", | ||||||
|  |                                 "${message.lineNumber()} of ${message.sourceId()}") | ||||||
|  |                             return true | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     loadUrl(url) | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             update = { | ||||||
|  |                 webView.value = it | ||||||
|  |             } | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Sets cookies for the given URL using the cookies stored in the `CommonsCookieJar`. | ||||||
|  |      * | ||||||
|  |      * @param url The URL for which cookies need to be set. | ||||||
|  |      */ | ||||||
|  |     private fun setCookies(url: String) { | ||||||
|  |         CookieManager.getInstance().let { | ||||||
|  |             val cookies = cookieJar.loadForRequest(url.toHttpUrl()) | ||||||
|  |             for (cookie in cookies) { | ||||||
|  |                 it.setCookie(url, cookie.toString()) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private const val VANISH_ACCOUNT_URL = "VanishAccountUrl" | ||||||
|  |         private const val VANISH_ACCOUNT_SUCCESS_URL = "vanishAccountSuccessUrl" | ||||||
|  | 
 | ||||||
|  |         /** | ||||||
|  |          * Launch the WebViewActivity with the specified URL and success URL. | ||||||
|  |          * @param context The context from which the activity is launched. | ||||||
|  |          * @param url The initial URL to load in the WebView. | ||||||
|  |          * @param successUrl The URL that triggers the WebView to close when matched. | ||||||
|  |          */ | ||||||
|  |         fun showWebView( | ||||||
|  |             context: Context, | ||||||
|  |             url: String, | ||||||
|  |             successUrl: String | ||||||
|  |         ) { | ||||||
|  |             val intent = Intent( | ||||||
|  |                 context, | ||||||
|  |                 SingleWebViewActivity::class.java | ||||||
|  |             ).apply { | ||||||
|  |                 putExtra(VANISH_ACCOUNT_URL, url) | ||||||
|  |                 putExtra(VANISH_ACCOUNT_SUCCESS_URL, successUrl) | ||||||
|  |             } | ||||||
|  |             context.startActivity(intent) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -1,44 +0,0 @@ | ||||||
| package fr.free.nrw.commons.auth; |  | ||||||
| 
 |  | ||||||
| import android.accounts.Account; |  | ||||||
| import android.accounts.AccountManager; |  | ||||||
| import android.content.Context; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| public class AccountUtil { |  | ||||||
| 
 |  | ||||||
|     public static final String AUTH_TOKEN_TYPE = "CommonsAndroid"; |  | ||||||
| 
 |  | ||||||
|     public AccountUtil() { |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * @return Account|null |  | ||||||
|      */ |  | ||||||
|     @Nullable |  | ||||||
|     public static Account account(Context context) { |  | ||||||
|         try { |  | ||||||
|             Account[] accounts = accountManager(context).getAccountsByType(BuildConfig.ACCOUNT_TYPE); |  | ||||||
|             if (accounts.length > 0) { |  | ||||||
|                 return accounts[0]; |  | ||||||
|             } |  | ||||||
|         } catch (SecurityException e) { |  | ||||||
|             Timber.e(e); |  | ||||||
|         } |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     public static String getUserName(Context context) { |  | ||||||
|         Account account = account(context); |  | ||||||
|         return account == null ? null : account.name; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static AccountManager accountManager(Context context) { |  | ||||||
|         return AccountManager.get(context); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										24
									
								
								app/src/main/java/fr/free/nrw/commons/auth/AccountUtil.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								app/src/main/java/fr/free/nrw/commons/auth/AccountUtil.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,24 @@ | ||||||
|  | package fr.free.nrw.commons.auth | ||||||
|  | 
 | ||||||
|  | import android.accounts.Account | ||||||
|  | import android.accounts.AccountManager | ||||||
|  | import android.content.Context | ||||||
|  | import androidx.annotation.VisibleForTesting | ||||||
|  | import fr.free.nrw.commons.BuildConfig.ACCOUNT_TYPE | ||||||
|  | import timber.log.Timber | ||||||
|  | 
 | ||||||
|  | const val AUTH_TOKEN_TYPE: String = "CommonsAndroid" | ||||||
|  | 
 | ||||||
|  | fun getUserName(context: Context): String? { | ||||||
|  |     return account(context)?.name | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @VisibleForTesting | ||||||
|  | fun account(context: Context): Account? = try { | ||||||
|  |     val accountManager = AccountManager.get(context) | ||||||
|  |     val accounts = accountManager.getAccountsByType(ACCOUNT_TYPE) | ||||||
|  |     if (accounts.isNotEmpty()) accounts[0] else null | ||||||
|  | } catch (e: SecurityException) { | ||||||
|  |     Timber.e(e) | ||||||
|  |     null | ||||||
|  | } | ||||||
|  | @ -1,456 +0,0 @@ | ||||||
| package fr.free.nrw.commons.auth; |  | ||||||
| 
 |  | ||||||
| import android.accounts.AccountAuthenticatorActivity; |  | ||||||
| import android.app.ProgressDialog; |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.text.Editable; |  | ||||||
| import android.text.TextWatcher; |  | ||||||
| import android.view.KeyEvent; |  | ||||||
| import android.view.MenuInflater; |  | ||||||
| import android.view.MenuItem; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.ViewGroup; |  | ||||||
| import android.view.inputmethod.InputMethodManager; |  | ||||||
| 
 |  | ||||||
| import android.widget.TextView; |  | ||||||
| import androidx.annotation.ColorRes; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.annotation.StringRes; |  | ||||||
| import androidx.appcompat.app.AlertDialog; |  | ||||||
| import androidx.appcompat.app.AppCompatDelegate; |  | ||||||
| import androidx.core.app.NavUtils; |  | ||||||
| import androidx.core.content.ContextCompat; |  | ||||||
| import fr.free.nrw.commons.auth.login.LoginClient; |  | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult; |  | ||||||
| import fr.free.nrw.commons.databinding.ActivityLoginBinding; |  | ||||||
| import fr.free.nrw.commons.utils.ActivityUtils; |  | ||||||
| import java.util.Locale; |  | ||||||
| import fr.free.nrw.commons.auth.login.LoginCallback; |  | ||||||
| 
 |  | ||||||
| import java.util.Objects; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Named; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.Utils; |  | ||||||
| import fr.free.nrw.commons.contributions.MainActivity; |  | ||||||
| import fr.free.nrw.commons.di.ApplicationlessInjection; |  | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; |  | ||||||
| import fr.free.nrw.commons.utils.ConfigUtils; |  | ||||||
| import fr.free.nrw.commons.utils.SystemThemeUtils; |  | ||||||
| import fr.free.nrw.commons.utils.ViewUtil; |  | ||||||
| import io.reactivex.disposables.CompositeDisposable; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| import static android.view.KeyEvent.KEYCODE_ENTER; |  | ||||||
| import static android.view.View.VISIBLE; |  | ||||||
| import static android.view.inputmethod.EditorInfo.IME_ACTION_DONE; |  | ||||||
| import static fr.free.nrw.commons.CommonsApplication.LOGIN_MESSAGE_INTENT_KEY; |  | ||||||
| import static fr.free.nrw.commons.CommonsApplication.LOGIN_USERNAME_INTENT_KEY; |  | ||||||
| 
 |  | ||||||
| public class LoginActivity extends AccountAuthenticatorActivity { |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     SessionManager sessionManager; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     @Named("default_preferences") |  | ||||||
|     JsonKvStore applicationKvStore; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     LoginClient loginClient; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     SystemThemeUtils systemThemeUtils; |  | ||||||
| 
 |  | ||||||
|     private ActivityLoginBinding binding; |  | ||||||
|     ProgressDialog progressDialog; |  | ||||||
|     private AppCompatDelegate delegate; |  | ||||||
|     private LoginTextWatcher textWatcher = new LoginTextWatcher(); |  | ||||||
|     private CompositeDisposable compositeDisposable = new CompositeDisposable(); |  | ||||||
|     final  String saveProgressDailog="ProgressDailog_state"; |  | ||||||
|     final String saveErrorMessage ="errorMessage"; |  | ||||||
|     final String saveUsername="username"; |  | ||||||
|     final  String savePassword="password"; |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onCreate(Bundle savedInstanceState) { |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
|         ApplicationlessInjection |  | ||||||
|                 .getInstance(this.getApplicationContext()) |  | ||||||
|                 .getCommonsApplicationComponent() |  | ||||||
|                 .inject(this); |  | ||||||
| 
 |  | ||||||
|         boolean isDarkTheme = systemThemeUtils.isDeviceInNightMode(); |  | ||||||
|         setTheme(isDarkTheme ? R.style.DarkAppTheme : R.style.LightAppTheme); |  | ||||||
|         getDelegate().installViewFactory(); |  | ||||||
|         getDelegate().onCreate(savedInstanceState); |  | ||||||
| 
 |  | ||||||
|         binding = ActivityLoginBinding.inflate(getLayoutInflater()); |  | ||||||
|         setContentView(binding.getRoot()); |  | ||||||
| 
 |  | ||||||
|         String message = getIntent().getStringExtra(LOGIN_MESSAGE_INTENT_KEY); |  | ||||||
|         String username = getIntent().getStringExtra(LOGIN_USERNAME_INTENT_KEY); |  | ||||||
| 
 |  | ||||||
|         binding.loginUsername.addTextChangedListener(textWatcher); |  | ||||||
|         binding.loginPassword.addTextChangedListener(textWatcher); |  | ||||||
|         binding.loginTwoFactor.addTextChangedListener(textWatcher); |  | ||||||
| 
 |  | ||||||
|         binding.skipLogin.setOnClickListener(view -> skipLogin()); |  | ||||||
|         binding.forgotPassword.setOnClickListener(view -> forgotPassword()); |  | ||||||
|         binding.aboutPrivacyPolicy.setOnClickListener(view -> onPrivacyPolicyClicked()); |  | ||||||
|         binding.signUpButton.setOnClickListener(view -> signUp()); |  | ||||||
|         binding.loginButton.setOnClickListener(view -> performLogin()); |  | ||||||
| 
 |  | ||||||
|         binding.loginPassword.setOnEditorActionListener(this::onEditorAction); |  | ||||||
|         binding.loginPassword.setOnFocusChangeListener(this::onPasswordFocusChanged); |  | ||||||
| 
 |  | ||||||
|         if (ConfigUtils.isBetaFlavour()) { |  | ||||||
|             binding.loginCredentials.setText(getString(R.string.login_credential)); |  | ||||||
|         } else { |  | ||||||
|             binding.loginCredentials.setVisibility(View.GONE); |  | ||||||
|         } |  | ||||||
|         if (message != null) { |  | ||||||
|             showMessage(message, R.color.secondaryDarkColor); |  | ||||||
|         } |  | ||||||
|         if (username != null) { |  | ||||||
|             binding.loginUsername.setText(username); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     /** |  | ||||||
|      * Hides the keyboard if the user's focus is not on the password (hasFocus is false). |  | ||||||
|      * @param view The keyboard |  | ||||||
|      * @param hasFocus Set to true if the keyboard has focus |  | ||||||
|      */ |  | ||||||
|     void onPasswordFocusChanged(View view, boolean hasFocus) { |  | ||||||
|         if (!hasFocus) { |  | ||||||
|             ViewUtil.hideKeyboard(view); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     boolean onEditorAction(TextView textView, int actionId, KeyEvent keyEvent) { |  | ||||||
|         if (binding.loginButton.isEnabled()) { |  | ||||||
|             if (actionId == IME_ACTION_DONE) { |  | ||||||
|                 performLogin(); |  | ||||||
|                 return true; |  | ||||||
|             } else if ((keyEvent != null) && keyEvent.getKeyCode() == KEYCODE_ENTER) { |  | ||||||
|                 performLogin(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     protected void skipLogin() { |  | ||||||
|         new AlertDialog.Builder(this).setTitle(R.string.skip_login_title) |  | ||||||
|                 .setMessage(R.string.skip_login_message) |  | ||||||
|                 .setCancelable(false) |  | ||||||
|                 .setPositiveButton(R.string.yes, (dialog, which) -> { |  | ||||||
|                     dialog.cancel(); |  | ||||||
|                     performSkipLogin(); |  | ||||||
|                 }) |  | ||||||
|                 .setNegativeButton(R.string.no, (dialog, which) -> dialog.cancel()) |  | ||||||
|                 .show(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void forgotPassword() { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(BuildConfig.FORGOT_PASSWORD_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void onPrivacyPolicyClicked() { |  | ||||||
|         Utils.handleWebUrl(this, Uri.parse(BuildConfig.PRIVACY_POLICY_URL)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     protected void signUp() { |  | ||||||
|         Intent intent = new Intent(this, SignupActivity.class); |  | ||||||
|         startActivity(intent); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onPostCreate(Bundle savedInstanceState) { |  | ||||||
|         super.onPostCreate(savedInstanceState); |  | ||||||
|         getDelegate().onPostCreate(savedInstanceState); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onResume() { |  | ||||||
|         super.onResume(); |  | ||||||
| 
 |  | ||||||
|         if (sessionManager.getCurrentAccount() != null |  | ||||||
|                 && sessionManager.isUserLoggedIn()) { |  | ||||||
|             applicationKvStore.putBoolean("login_skipped", false); |  | ||||||
|             startMainActivity(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (applicationKvStore.getBoolean("login_skipped", false)) { |  | ||||||
|             performSkipLogin(); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onDestroy() { |  | ||||||
|         compositeDisposable.clear(); |  | ||||||
|         try { |  | ||||||
|             // To prevent leaked window when finish() is called, see http://stackoverflow.com/questions/32065854/activity-has-leaked-window-at-alertdialog-show-method |  | ||||||
|             if (progressDialog != null && progressDialog.isShowing()) { |  | ||||||
|                 progressDialog.dismiss(); |  | ||||||
|             } |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             e.printStackTrace(); |  | ||||||
|         } |  | ||||||
|         binding.loginUsername.removeTextChangedListener(textWatcher); |  | ||||||
|         binding.loginPassword.removeTextChangedListener(textWatcher); |  | ||||||
|         binding.loginTwoFactor.removeTextChangedListener(textWatcher); |  | ||||||
|         delegate.onDestroy(); |  | ||||||
|         if(null!=loginClient) { |  | ||||||
|             loginClient.cancel(); |  | ||||||
|         } |  | ||||||
|         binding = null; |  | ||||||
|         super.onDestroy(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void performLogin() { |  | ||||||
|         Timber.d("Login to start!"); |  | ||||||
|         final String username = Objects.requireNonNull(binding.loginUsername.getText()).toString(); |  | ||||||
|         final String password = Objects.requireNonNull(binding.loginPassword.getText()).toString(); |  | ||||||
|         final String twoFactorCode = Objects.requireNonNull(binding.loginTwoFactor.getText()).toString(); |  | ||||||
| 
 |  | ||||||
|         showLoggingProgressBar(); |  | ||||||
|         loginClient.doLogin(username, password, twoFactorCode, Locale.getDefault().getLanguage(), |  | ||||||
|             new LoginCallback() { |  | ||||||
|                 @Override |  | ||||||
|                 public void success(@NonNull LoginResult loginResult) { |  | ||||||
|                     runOnUiThread(()->{ |  | ||||||
|                         Timber.d("Login Success"); |  | ||||||
|                         hideProgress(); |  | ||||||
|                         onLoginSuccess(loginResult); |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 @Override |  | ||||||
|                 public void twoFactorPrompt(@NonNull Throwable caught, @Nullable String token) { |  | ||||||
|                     runOnUiThread(()->{ |  | ||||||
|                         Timber.d("Requesting 2FA prompt"); |  | ||||||
|                         hideProgress(); |  | ||||||
|                         askUserForTwoFactorAuth(); |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 @Override |  | ||||||
|                 public void passwordResetPrompt(@Nullable String token) { |  | ||||||
|                     runOnUiThread(()->{ |  | ||||||
|                         Timber.d("Showing password reset prompt"); |  | ||||||
|                         hideProgress(); |  | ||||||
|                         showPasswordResetPrompt(); |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 @Override |  | ||||||
|                 public void error(@NonNull Throwable caught) { |  | ||||||
|                     runOnUiThread(()->{ |  | ||||||
|                         Timber.e(caught); |  | ||||||
|                         hideProgress(); |  | ||||||
|                         showMessageAndCancelDialog(caught.getLocalizedMessage()); |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
|             }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     private void hideProgress() { |  | ||||||
|         progressDialog.dismiss(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void showPasswordResetPrompt() { |  | ||||||
|         showMessageAndCancelDialog(getString(R.string.you_must_reset_your_passsword)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This function is called when user skips the login. |  | ||||||
|      * It redirects the user to Explore Activity. |  | ||||||
|      */ |  | ||||||
|     private void performSkipLogin() { |  | ||||||
|         applicationKvStore.putBoolean("login_skipped", true); |  | ||||||
|         MainActivity.startYourself(this); |  | ||||||
|         finish(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void showLoggingProgressBar() { |  | ||||||
|         progressDialog = new ProgressDialog(this); |  | ||||||
|         progressDialog.setIndeterminate(true); |  | ||||||
|         progressDialog.setTitle(getString(R.string.logging_in_title)); |  | ||||||
|         progressDialog.setMessage(getString(R.string.logging_in_message)); |  | ||||||
|         progressDialog.setCanceledOnTouchOutside(false); |  | ||||||
|         progressDialog.show(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void onLoginSuccess(LoginResult loginResult) { |  | ||||||
|         compositeDisposable.clear(); |  | ||||||
|         sessionManager.setUserLoggedIn(true); |  | ||||||
|         sessionManager.updateAccount(loginResult); |  | ||||||
|         progressDialog.dismiss(); |  | ||||||
|         showSuccessAndDismissDialog(); |  | ||||||
|         startMainActivity(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onStart() { |  | ||||||
|         super.onStart(); |  | ||||||
|         delegate.onStart(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onStop() { |  | ||||||
|         super.onStop(); |  | ||||||
|         delegate.onStop(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onPostResume() { |  | ||||||
|         super.onPostResume(); |  | ||||||
|         getDelegate().onPostResume(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void setContentView(View view, ViewGroup.LayoutParams params) { |  | ||||||
|         getDelegate().setContentView(view, params); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|         switch (item.getItemId()) { |  | ||||||
|             case android.R.id.home: |  | ||||||
|                 NavUtils.navigateUpFromSameTask(this); |  | ||||||
|                 return true; |  | ||||||
|         } |  | ||||||
|         return super.onOptionsItemSelected(item); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     @NonNull |  | ||||||
|     public MenuInflater getMenuInflater() { |  | ||||||
|         return getDelegate().getMenuInflater(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void askUserForTwoFactorAuth() { |  | ||||||
|         progressDialog.dismiss(); |  | ||||||
|         binding.twoFactorContainer.setVisibility(VISIBLE); |  | ||||||
|         binding.loginTwoFactor.setVisibility(VISIBLE); |  | ||||||
|         binding.loginTwoFactor.requestFocus(); |  | ||||||
|         InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); |  | ||||||
|         imm.toggleSoftInput(InputMethodManager.SHOW_FORCED, InputMethodManager.HIDE_IMPLICIT_ONLY); |  | ||||||
|         showMessageAndCancelDialog(R.string.login_failed_2fa_needed); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void showMessageAndCancelDialog(@StringRes int resId) { |  | ||||||
|         showMessage(resId, R.color.secondaryDarkColor); |  | ||||||
|         if (progressDialog != null) { |  | ||||||
|             progressDialog.cancel(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void showMessageAndCancelDialog(String error) { |  | ||||||
|         showMessage(error, R.color.secondaryDarkColor); |  | ||||||
|         if (progressDialog != null) { |  | ||||||
|             progressDialog.cancel(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void showSuccessAndDismissDialog() { |  | ||||||
|         showMessage(R.string.login_success, R.color.primaryDarkColor); |  | ||||||
|         progressDialog.dismiss(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void startMainActivity() { |  | ||||||
|         ActivityUtils.startActivityWithFlags(this, MainActivity.class, Intent.FLAG_ACTIVITY_SINGLE_TOP); |  | ||||||
|         finish(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void showMessage(@StringRes int resId, @ColorRes int colorResId) { |  | ||||||
|         binding.errorMessage.setText(getString(resId)); |  | ||||||
|         binding.errorMessage.setTextColor(ContextCompat.getColor(this, colorResId)); |  | ||||||
|         binding.errorMessageContainer.setVisibility(VISIBLE); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void showMessage(String message, @ColorRes int colorResId) { |  | ||||||
|         binding.errorMessage.setText(message); |  | ||||||
|         binding.errorMessage.setTextColor(ContextCompat.getColor(this, colorResId)); |  | ||||||
|         binding.errorMessageContainer.setVisibility(VISIBLE); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private AppCompatDelegate getDelegate() { |  | ||||||
|         if (delegate == null) { |  | ||||||
|             delegate = AppCompatDelegate.create(this, null); |  | ||||||
|         } |  | ||||||
|         return delegate; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private class LoginTextWatcher implements TextWatcher { |  | ||||||
|         @Override |  | ||||||
|         public void beforeTextChanged(CharSequence charSequence, int start, int count, int after) { |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         @Override |  | ||||||
|         public void onTextChanged(CharSequence charSequence, int start, int count, int after) { |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         @Override |  | ||||||
|         public void afterTextChanged(Editable editable) { |  | ||||||
|             boolean enabled = binding.loginUsername.getText().length() != 0 && |  | ||||||
|                 binding.loginPassword.getText().length() != 0 && |  | ||||||
|                 (BuildConfig.DEBUG || binding.loginTwoFactor.getText().length() != 0 || |  | ||||||
|                     binding.loginTwoFactor.getVisibility() != VISIBLE); |  | ||||||
|             binding.loginButton.setEnabled(enabled); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public static void startYourself(Context context) { |  | ||||||
|         Intent intent = new Intent(context, LoginActivity.class); |  | ||||||
|         context.startActivity(intent); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onSaveInstanceState(Bundle outState) { |  | ||||||
|         // if progressDialog is visible during the configuration change  then store state as  true else false so that |  | ||||||
|         // we maintain visibility of progressDailog after configuration change |  | ||||||
|         if(progressDialog!=null&&progressDialog.isShowing()) { |  | ||||||
|             outState.putBoolean(saveProgressDailog,true); |  | ||||||
|         } else { |  | ||||||
|             outState.putBoolean(saveProgressDailog,false); |  | ||||||
|         } |  | ||||||
|         outState.putString(saveErrorMessage,binding.errorMessage.getText().toString()); //Save the errorMessage |  | ||||||
|         outState.putString(saveUsername,getUsername()); // Save the username |  | ||||||
|         outState.putString(savePassword,getPassword()); // Save the password |  | ||||||
|     } |  | ||||||
|     private String getUsername() { |  | ||||||
|         return binding.loginUsername.getText().toString(); |  | ||||||
|     } |  | ||||||
|     private String getPassword(){ |  | ||||||
|         return  binding.loginPassword.getText().toString(); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onRestoreInstanceState(final Bundle savedInstanceState) { |  | ||||||
|         super.onRestoreInstanceState(savedInstanceState); |  | ||||||
|         binding.loginUsername.setText(savedInstanceState.getString(saveUsername)); |  | ||||||
|         binding.loginPassword.setText(savedInstanceState.getString(savePassword)); |  | ||||||
|         if(savedInstanceState.getBoolean(saveProgressDailog)) { |  | ||||||
|             performLogin(); |  | ||||||
|         } |  | ||||||
|         String errorMessage=savedInstanceState.getString(saveErrorMessage); |  | ||||||
|         if(sessionManager.isUserLoggedIn()) { |  | ||||||
|             showMessage(R.string.login_success, R.color.primaryDarkColor); |  | ||||||
|         } else { |  | ||||||
|             showMessage(errorMessage, R.color.secondaryDarkColor); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										489
									
								
								app/src/main/java/fr/free/nrw/commons/auth/LoginActivity.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										489
									
								
								app/src/main/java/fr/free/nrw/commons/auth/LoginActivity.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,489 @@ | ||||||
|  | package fr.free.nrw.commons.auth | ||||||
|  | 
 | ||||||
|  | import android.accounts.AccountAuthenticatorActivity | ||||||
|  | import android.app.ProgressDialog | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.DialogInterface | ||||||
|  | import android.content.Intent | ||||||
|  | import android.net.Uri | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.KeyEvent | ||||||
|  | import android.view.MenuInflater | ||||||
|  | import android.view.MenuItem | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import android.view.inputmethod.EditorInfo | ||||||
|  | import android.view.inputmethod.InputMethodManager | ||||||
|  | import android.widget.TextView | ||||||
|  | import androidx.annotation.ColorRes | ||||||
|  | import androidx.annotation.StringRes | ||||||
|  | import androidx.annotation.VisibleForTesting | ||||||
|  | import androidx.appcompat.app.AlertDialog | ||||||
|  | import androidx.appcompat.app.AppCompatDelegate | ||||||
|  | import androidx.core.app.NavUtils | ||||||
|  | import androidx.core.content.ContextCompat | ||||||
|  | import androidx.core.view.WindowCompat | ||||||
|  | import fr.free.nrw.commons.BuildConfig | ||||||
|  | import fr.free.nrw.commons.CommonsApplication | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.auth.login.LoginCallback | ||||||
|  | import fr.free.nrw.commons.auth.login.LoginClient | ||||||
|  | import fr.free.nrw.commons.auth.login.LoginResult | ||||||
|  | import fr.free.nrw.commons.contributions.MainActivity | ||||||
|  | import fr.free.nrw.commons.databinding.ActivityLoginBinding | ||||||
|  | import fr.free.nrw.commons.di.ApplicationlessInjection | ||||||
|  | import fr.free.nrw.commons.kvstore.JsonKvStore | ||||||
|  | import fr.free.nrw.commons.utils.applyEdgeToEdgeAllInsets | ||||||
|  | import fr.free.nrw.commons.utils.AbstractTextWatcher | ||||||
|  | import fr.free.nrw.commons.utils.ActivityUtils.startActivityWithFlags | ||||||
|  | import fr.free.nrw.commons.utils.ConfigUtils.isBetaFlavour | ||||||
|  | import fr.free.nrw.commons.utils.SystemThemeUtils | ||||||
|  | import fr.free.nrw.commons.utils.ViewUtil.hideKeyboard | ||||||
|  | import fr.free.nrw.commons.utils.handleKeyboardInsets | ||||||
|  | import fr.free.nrw.commons.utils.handleWebUrl | ||||||
|  | import io.reactivex.disposables.CompositeDisposable | ||||||
|  | import timber.log.Timber | ||||||
|  | import java.util.Locale | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Named | ||||||
|  | 
 | ||||||
|  | class LoginActivity : AccountAuthenticatorActivity() { | ||||||
|  |     @Inject | ||||||
|  |     lateinit var sessionManager: SessionManager | ||||||
|  | 
 | ||||||
|  |     @Inject | ||||||
|  |     @field:Named("default_preferences") | ||||||
|  |     lateinit var applicationKvStore: JsonKvStore | ||||||
|  | 
 | ||||||
|  |     @Inject | ||||||
|  |     lateinit var loginClient: LoginClient | ||||||
|  | 
 | ||||||
|  |     @Inject | ||||||
|  |     lateinit var systemThemeUtils: SystemThemeUtils | ||||||
|  | 
 | ||||||
|  |     private var binding: ActivityLoginBinding? = null | ||||||
|  |     private var progressDialog: ProgressDialog? = null | ||||||
|  |     private val textWatcher = AbstractTextWatcher(::onTextChanged) | ||||||
|  |     private val compositeDisposable = CompositeDisposable() | ||||||
|  |     private val delegate: AppCompatDelegate by lazy { | ||||||
|  |         AppCompatDelegate.create(this, null) | ||||||
|  |     } | ||||||
|  |     private var lastLoginResult: LoginResult? = null | ||||||
|  | 
 | ||||||
|  |     public override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |         ApplicationlessInjection | ||||||
|  |             .getInstance(this.applicationContext) | ||||||
|  |             .commonsApplicationComponent | ||||||
|  |             .inject(this) | ||||||
|  | 
 | ||||||
|  |         val isDarkTheme = systemThemeUtils.isDeviceInNightMode() | ||||||
|  |         setTheme(if (isDarkTheme) R.style.DarkAppTheme else R.style.LightAppTheme) | ||||||
|  |         delegate.installViewFactory() | ||||||
|  |         delegate.onCreate(savedInstanceState) | ||||||
|  | 
 | ||||||
|  |         WindowCompat.getInsetsController(window, window.decorView) | ||||||
|  |             .isAppearanceLightStatusBars = !isDarkTheme | ||||||
|  | 
 | ||||||
|  |         WindowCompat.setDecorFitsSystemWindows(window, false) | ||||||
|  | 
 | ||||||
|  |         binding = ActivityLoginBinding.inflate(layoutInflater) | ||||||
|  |         applyEdgeToEdgeAllInsets(binding!!.root) | ||||||
|  |         binding!!.root.handleKeyboardInsets() | ||||||
|  |         with(binding!!) { | ||||||
|  |             setContentView(root) | ||||||
|  | 
 | ||||||
|  |             loginUsername.addTextChangedListener(textWatcher) | ||||||
|  |             loginPassword.addTextChangedListener(textWatcher) | ||||||
|  |             loginTwoFactor.addTextChangedListener(textWatcher) | ||||||
|  | 
 | ||||||
|  |             skipLogin.setOnClickListener { skipLogin() } | ||||||
|  |             forgotPassword.setOnClickListener { forgotPassword() } | ||||||
|  |             aboutPrivacyPolicy.setOnClickListener { onPrivacyPolicyClicked() } | ||||||
|  |             signUpButton.setOnClickListener { signUp() } | ||||||
|  |             loginButton.setOnClickListener { performLogin() } | ||||||
|  |             loginPassword.setOnEditorActionListener { textView, actionId, keyEvent -> | ||||||
|  |                 if (binding!!.loginButton.isEnabled && isTriggerAction(actionId, keyEvent)) { | ||||||
|  |                     if (actionId == EditorInfo.IME_ACTION_NEXT && lastLoginResult != null) { | ||||||
|  |                         askUserForTwoFactorAuthWithKeyboard() | ||||||
|  |                         true | ||||||
|  |                     } else { | ||||||
|  |                         performLogin() | ||||||
|  |                         true | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     false | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             loginPassword.onFocusChangeListener = | ||||||
|  |                 View.OnFocusChangeListener(::onPasswordFocusChanged) | ||||||
|  | 
 | ||||||
|  |             if (isBetaFlavour) { | ||||||
|  |                 loginCredentials.text = getString(R.string.login_credential) | ||||||
|  |             } else { | ||||||
|  |                 loginCredentials.visibility = View.GONE | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             intent.getStringExtra(CommonsApplication.LOGIN_MESSAGE_INTENT_KEY)?.let { | ||||||
|  |                 showMessage(it, R.color.secondaryDarkColor) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             intent.getStringExtra(CommonsApplication.LOGIN_USERNAME_INTENT_KEY)?.let { | ||||||
|  |                 loginUsername.setText(it) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun askUserForTwoFactorAuthWithKeyboard() { | ||||||
|  |         if (binding == null) { | ||||||
|  |             Timber.w("Binding is null, reinitializing in askUserForTwoFactorAuthWithKeyboard") | ||||||
|  |             binding = ActivityLoginBinding.inflate(layoutInflater) | ||||||
|  |             setContentView(binding!!.root) | ||||||
|  |         } | ||||||
|  |         progressDialog!!.dismiss() | ||||||
|  |         if (binding != null) { | ||||||
|  |             with(binding!!) { | ||||||
|  |                 twoFactorContainer.visibility = View.VISIBLE | ||||||
|  |                 twoFactorContainer.hint = getString(if (lastLoginResult is LoginResult.EmailAuthResult) R.string.email_auth_code else R.string._2fa_code) | ||||||
|  |                 loginTwoFactor.visibility = View.VISIBLE | ||||||
|  |                 loginTwoFactor.requestFocus() | ||||||
|  | 
 | ||||||
|  |                 val imm = getSystemService(INPUT_METHOD_SERVICE) as InputMethodManager | ||||||
|  |                 imm.showSoftInput(loginTwoFactor, InputMethodManager.SHOW_IMPLICIT) | ||||||
|  | 
 | ||||||
|  |                 loginTwoFactor.setOnEditorActionListener { _, actionId, event -> | ||||||
|  |                     if (actionId == EditorInfo.IME_ACTION_DONE || | ||||||
|  |                         (event != null && event.keyCode == KeyEvent.KEYCODE_ENTER && event.action == KeyEvent.ACTION_DOWN)) { | ||||||
|  |                         performLogin() | ||||||
|  |                         true | ||||||
|  |                     } else { | ||||||
|  |                         false | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             Timber.e("Binding is null in askUserForTwoFactorAuthWithKeyboard after reinitialization attempt") | ||||||
|  |         } | ||||||
|  |         showMessageAndCancelDialog(getString(if (lastLoginResult is LoginResult.EmailAuthResult) R.string.login_failed_email_auth_needed else R.string.login_failed_2fa_needed)) | ||||||
|  |     } | ||||||
|  |     override fun onPostCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onPostCreate(savedInstanceState) | ||||||
|  |         delegate.onPostCreate(savedInstanceState) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onResume() { | ||||||
|  |         super.onResume() | ||||||
|  | 
 | ||||||
|  |         if (sessionManager.currentAccount != null && sessionManager.isUserLoggedIn) { | ||||||
|  |             applicationKvStore.putBoolean("login_skipped", false) | ||||||
|  |             startMainActivity() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (applicationKvStore.getBoolean("login_skipped", false)) { | ||||||
|  |             performSkipLogin() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDestroy() { | ||||||
|  |         compositeDisposable.clear() | ||||||
|  |         try { | ||||||
|  |             // To prevent leaked window when finish() is called, see http://stackoverflow.com/questions/32065854/activity-has-leaked-window-at-alertdialog-show-method | ||||||
|  |             if (progressDialog?.isShowing == true) { | ||||||
|  |                 progressDialog!!.dismiss() | ||||||
|  |             } | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             e.printStackTrace() | ||||||
|  |         } | ||||||
|  |         with(binding!!) { | ||||||
|  |             loginUsername.removeTextChangedListener(textWatcher) | ||||||
|  |             loginPassword.removeTextChangedListener(textWatcher) | ||||||
|  |             loginTwoFactor.removeTextChangedListener(textWatcher) | ||||||
|  |         } | ||||||
|  |         delegate.onDestroy() | ||||||
|  |         loginClient.cancel() | ||||||
|  |         binding = null | ||||||
|  |         super.onDestroy() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onStart() { | ||||||
|  |         super.onStart() | ||||||
|  |         delegate.onStart() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onStop() { | ||||||
|  |         super.onStop() | ||||||
|  |         delegate.onStop() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onPostResume() { | ||||||
|  |         super.onPostResume() | ||||||
|  |         delegate.onPostResume() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun setContentView(view: View, params: ViewGroup.LayoutParams) { | ||||||
|  |         delegate.setContentView(view, params) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|  |         when (item.itemId) { | ||||||
|  |             android.R.id.home -> { | ||||||
|  |                 NavUtils.navigateUpFromSameTask(this) | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return super.onOptionsItemSelected(item) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onSaveInstanceState(outState: Bundle) { | ||||||
|  |         // if progressDialog is visible during the configuration change  then store state as  true else false so that | ||||||
|  |         // we maintain visibility of progressDialog after configuration change | ||||||
|  |         if (progressDialog != null && progressDialog!!.isShowing) { | ||||||
|  |             outState.putBoolean(SAVE_PROGRESS_DIALOG, true) | ||||||
|  |         } else { | ||||||
|  |             outState.putBoolean(SAVE_PROGRESS_DIALOG, false) | ||||||
|  |         } | ||||||
|  |         outState.putString( | ||||||
|  |             SAVE_ERROR_MESSAGE, | ||||||
|  |             binding!!.errorMessage.text.toString() | ||||||
|  |         ) //Save the errorMessage | ||||||
|  |         outState.putString( | ||||||
|  |             SAVE_USERNAME, | ||||||
|  |             binding!!.loginUsername.text.toString() | ||||||
|  |         ) // Save the username | ||||||
|  |         outState.putString( | ||||||
|  |             SAVE_PASSWORD, | ||||||
|  |             binding!!.loginPassword.text.toString() | ||||||
|  |         ) // Save the password | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onRestoreInstanceState(savedInstanceState: Bundle) { | ||||||
|  |         super.onRestoreInstanceState(savedInstanceState) | ||||||
|  |         binding!!.loginUsername.setText(savedInstanceState.getString(SAVE_USERNAME)) | ||||||
|  |         binding!!.loginPassword.setText(savedInstanceState.getString(SAVE_PASSWORD)) | ||||||
|  |         if (savedInstanceState.getBoolean(SAVE_PROGRESS_DIALOG)) { | ||||||
|  |             performLogin() | ||||||
|  |         } | ||||||
|  |         val errorMessage = savedInstanceState.getString(SAVE_ERROR_MESSAGE) | ||||||
|  |         if (sessionManager.isUserLoggedIn) { | ||||||
|  |             showMessage(R.string.login_success, R.color.primaryDarkColor) | ||||||
|  |         } else { | ||||||
|  |             showMessage(errorMessage, R.color.secondaryDarkColor) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Hides the keyboard if the user's focus is not on the password (hasFocus is false). | ||||||
|  |      * @param view The keyboard | ||||||
|  |      * @param hasFocus Set to true if the keyboard has focus | ||||||
|  |      */ | ||||||
|  |     private fun onPasswordFocusChanged(view: View, hasFocus: Boolean) { | ||||||
|  |         if (!hasFocus) { | ||||||
|  |             hideKeyboard(view) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onEditorAction(textView: TextView, actionId: Int, keyEvent: KeyEvent?) = | ||||||
|  |         if (binding!!.loginButton.isEnabled && isTriggerAction(actionId, keyEvent)) { | ||||||
|  |             performLogin() | ||||||
|  |             true | ||||||
|  |         } else false | ||||||
|  | 
 | ||||||
|  |     private fun isTriggerAction(actionId: Int, keyEvent: KeyEvent?) = | ||||||
|  |         actionId == EditorInfo.IME_ACTION_NEXT || actionId == EditorInfo.IME_ACTION_DONE || keyEvent?.keyCode == KeyEvent.KEYCODE_ENTER | ||||||
|  | 
 | ||||||
|  |     private fun skipLogin() { | ||||||
|  |         AlertDialog.Builder(this) | ||||||
|  |             .setTitle(R.string.skip_login_title) | ||||||
|  |             .setMessage(R.string.skip_login_message) | ||||||
|  |             .setCancelable(false) | ||||||
|  |             .setPositiveButton(R.string.yes) { dialog: DialogInterface, which: Int -> | ||||||
|  |                 dialog.cancel() | ||||||
|  |                 performSkipLogin() | ||||||
|  |             } | ||||||
|  |             .setNegativeButton(R.string.no) { dialog: DialogInterface, which: Int -> | ||||||
|  |                 dialog.cancel() | ||||||
|  |             } | ||||||
|  |             .show() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun forgotPassword() = | ||||||
|  |         handleWebUrl(this, Uri.parse(BuildConfig.FORGOT_PASSWORD_URL)) | ||||||
|  | 
 | ||||||
|  |     private fun onPrivacyPolicyClicked() = | ||||||
|  |         handleWebUrl(this, Uri.parse(BuildConfig.PRIVACY_POLICY_URL)) | ||||||
|  | 
 | ||||||
|  |     private fun signUp() = | ||||||
|  |         startActivity(Intent(this, SignupActivity::class.java)) | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun performLogin() { | ||||||
|  |         Timber.d("Login to start!") | ||||||
|  |         val username = binding!!.loginUsername.text.toString() | ||||||
|  |         val password = binding!!.loginPassword.text.toString() | ||||||
|  |         val twoFactorCode = binding!!.loginTwoFactor.text.toString() | ||||||
|  | 
 | ||||||
|  |         showLoggingProgressBar() | ||||||
|  |         loginClient.doLogin(username, | ||||||
|  |             password, | ||||||
|  |             lastLoginResult, | ||||||
|  |             twoFactorCode, | ||||||
|  |             Locale.getDefault().language, | ||||||
|  |             object : LoginCallback { | ||||||
|  |                 override fun success(loginResult: LoginResult) = runOnUiThread { | ||||||
|  |                     Timber.d("Login Success") | ||||||
|  |                     progressDialog!!.dismiss() | ||||||
|  |                     onLoginSuccess(loginResult) | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 override fun twoFactorPrompt(loginResult: LoginResult, caught: Throwable, token: String?) = runOnUiThread { | ||||||
|  |                     Timber.d("Requesting 2FA prompt") | ||||||
|  |                     progressDialog!!.dismiss() | ||||||
|  |                     lastLoginResult = loginResult | ||||||
|  |                     askUserForTwoFactorAuthWithKeyboard() | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 override fun emailAuthPrompt(loginResult: LoginResult, caught: Throwable, token: String?) = runOnUiThread { | ||||||
|  |                     Timber.d("Requesting email auth prompt") | ||||||
|  |                     progressDialog!!.dismiss() | ||||||
|  |                     lastLoginResult = loginResult | ||||||
|  |                     askUserForTwoFactorAuthWithKeyboard() | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 override fun passwordResetPrompt(token: String?) = runOnUiThread { | ||||||
|  |                     Timber.d("Showing password reset prompt") | ||||||
|  |                     progressDialog!!.dismiss() | ||||||
|  |                     showPasswordResetPrompt() | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 override fun error(caught: Throwable) = runOnUiThread { | ||||||
|  |                     Timber.e(caught) | ||||||
|  |                     progressDialog!!.dismiss() | ||||||
|  |                     showMessageAndCancelDialog(caught.localizedMessage ?: "") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun showPasswordResetPrompt() = | ||||||
|  |         showMessageAndCancelDialog(getString(R.string.you_must_reset_your_passsword)) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This function is called when user skips the login. | ||||||
|  |      * It redirects the user to Explore Activity. | ||||||
|  |      */ | ||||||
|  |     private fun performSkipLogin() { | ||||||
|  |         applicationKvStore.putBoolean("login_skipped", true) | ||||||
|  |         MainActivity.startYourself(this) | ||||||
|  |         finish() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun showLoggingProgressBar() { | ||||||
|  |         progressDialog = ProgressDialog(this).apply { | ||||||
|  |             isIndeterminate = true | ||||||
|  |             setTitle(getString(R.string.logging_in_title)) | ||||||
|  |             setMessage(getString(R.string.logging_in_message)) | ||||||
|  |             setCancelable(false) | ||||||
|  |         } | ||||||
|  |         progressDialog!!.show() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onLoginSuccess(loginResult: LoginResult) { | ||||||
|  |         compositeDisposable.clear() | ||||||
|  |         sessionManager.setUserLoggedIn(true) | ||||||
|  |         sessionManager.updateAccount(loginResult) | ||||||
|  |         progressDialog!!.dismiss() | ||||||
|  |         showSuccessAndDismissDialog() | ||||||
|  |         startMainActivity() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun getMenuInflater(): MenuInflater = | ||||||
|  |         delegate.menuInflater | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun askUserForTwoFactorAuth() { | ||||||
|  |         if (binding == null) { | ||||||
|  |             Timber.w("Binding is null, reinitializing in askUserForTwoFactorAuth") | ||||||
|  |             binding = ActivityLoginBinding.inflate(layoutInflater) | ||||||
|  |             setContentView(binding!!.root) | ||||||
|  |         } | ||||||
|  |         progressDialog!!.dismiss() | ||||||
|  |         if (binding != null) { | ||||||
|  |             with(binding!!) { | ||||||
|  |                 twoFactorContainer.visibility = View.VISIBLE | ||||||
|  |                 twoFactorContainer.hint = getString(if (lastLoginResult is LoginResult.EmailAuthResult) R.string.email_auth_code else R.string._2fa_code) | ||||||
|  |                 loginTwoFactor.visibility = View.VISIBLE | ||||||
|  |                 loginTwoFactor.requestFocus() | ||||||
|  | 
 | ||||||
|  |                 loginTwoFactor.setOnEditorActionListener { _, actionId, event -> | ||||||
|  |                     if (actionId == EditorInfo.IME_ACTION_DONE || | ||||||
|  |                         (event != null && event.keyCode == KeyEvent.KEYCODE_ENTER && event.action == KeyEvent.ACTION_DOWN)) { | ||||||
|  |                         performLogin() | ||||||
|  |                         true | ||||||
|  |                     } else { | ||||||
|  |                         false | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             Timber.e("Binding is null in askUserForTwoFactorAuth after reinitialization attempt") | ||||||
|  |         } | ||||||
|  |         val imm = getSystemService(INPUT_METHOD_SERVICE) as InputMethodManager | ||||||
|  |         imm.toggleSoftInput(InputMethodManager.SHOW_FORCED, InputMethodManager.HIDE_IMPLICIT_ONLY) | ||||||
|  |         showMessageAndCancelDialog(getString(if (lastLoginResult is LoginResult.EmailAuthResult) R.string.login_failed_email_auth_needed else R.string.login_failed_2fa_needed)) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun showMessageAndCancelDialog(@StringRes resId: Int) { | ||||||
|  |         showMessage(resId, R.color.secondaryDarkColor) | ||||||
|  |         progressDialog?.cancel() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun showMessageAndCancelDialog(error: String) { | ||||||
|  |         showMessage(error, R.color.secondaryDarkColor) | ||||||
|  |         progressDialog?.cancel() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun showSuccessAndDismissDialog() { | ||||||
|  |         showMessage(R.string.login_success, R.color.primaryDarkColor) | ||||||
|  |         progressDialog!!.dismiss() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @VisibleForTesting | ||||||
|  |     fun startMainActivity() { | ||||||
|  |         startActivityWithFlags(this, MainActivity::class.java, Intent.FLAG_ACTIVITY_SINGLE_TOP) | ||||||
|  |         finish() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun showMessage(@StringRes resId: Int, @ColorRes colorResId: Int) = with(binding!!) { | ||||||
|  |         errorMessage.text = getString(resId) | ||||||
|  |         errorMessage.setTextColor(ContextCompat.getColor(this@LoginActivity, colorResId)) | ||||||
|  |         errorMessageContainer.visibility = View.VISIBLE | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun showMessage(message: String?, @ColorRes colorResId: Int) = with(binding!!) { | ||||||
|  |         errorMessage.text = message | ||||||
|  |         errorMessage.setTextColor(ContextCompat.getColor(this@LoginActivity, colorResId)) | ||||||
|  |         errorMessageContainer.visibility = View.VISIBLE | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun onTextChanged(text: String) { | ||||||
|  |         val enabled = | ||||||
|  |             binding!!.loginUsername.text!!.length != 0 && binding!!.loginPassword.text!!.length != 0 && | ||||||
|  |                     (BuildConfig.DEBUG || binding!!.loginTwoFactor.text!!.length != 0 || binding!!.loginTwoFactor.visibility != View.VISIBLE) | ||||||
|  |         binding!!.loginButton.isEnabled = enabled | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun startYourself(context: Context) = | ||||||
|  |             context.startActivity(Intent(context, LoginActivity::class.java)) | ||||||
|  | 
 | ||||||
|  |         const val SAVE_PROGRESS_DIALOG: String = "ProgressDialog_state" | ||||||
|  |         const val SAVE_ERROR_MESSAGE: String = "errorMessage" | ||||||
|  |         const val SAVE_USERNAME: String = "username" | ||||||
|  |         const val SAVE_PASSWORD: String = "password" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,148 +0,0 @@ | ||||||
| package fr.free.nrw.commons.auth; |  | ||||||
| 
 |  | ||||||
| import android.accounts.Account; |  | ||||||
| import android.accounts.AccountManager; |  | ||||||
| import android.content.Context; |  | ||||||
| import android.os.Build; |  | ||||||
| import android.text.TextUtils; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Named; |  | ||||||
| import javax.inject.Singleton; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; |  | ||||||
| import io.reactivex.Completable; |  | ||||||
| import io.reactivex.Observable; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Manage the current logged in user session. |  | ||||||
|  */ |  | ||||||
| @Singleton |  | ||||||
| public class SessionManager { |  | ||||||
|     private final Context context; |  | ||||||
|     private Account currentAccount; // Unlike a savings account...  ;-) |  | ||||||
|     private JsonKvStore defaultKvStore; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public SessionManager(Context context, |  | ||||||
|                           @Named("default_preferences") JsonKvStore defaultKvStore) { |  | ||||||
|         this.context = context; |  | ||||||
|         this.currentAccount = null; |  | ||||||
|         this.defaultKvStore = defaultKvStore; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private boolean createAccount(@NonNull String userName, @NonNull String password) { |  | ||||||
|         Account account = getCurrentAccount(); |  | ||||||
|         if (account == null || TextUtils.isEmpty(account.name) || !account.name.equals(userName)) { |  | ||||||
|             removeAccount(); |  | ||||||
|             account = new Account(userName, BuildConfig.ACCOUNT_TYPE); |  | ||||||
|             return accountManager().addAccountExplicitly(account, password, null); |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void removeAccount() { |  | ||||||
|         Account account = getCurrentAccount(); |  | ||||||
|         if (account != null) { |  | ||||||
|             if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP_MR1) { |  | ||||||
|                 accountManager().removeAccountExplicitly(account); |  | ||||||
|             } else { |  | ||||||
|                 //noinspection deprecation |  | ||||||
|                 accountManager().removeAccount(account, null, null); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void updateAccount(LoginResult result) { |  | ||||||
|         boolean accountCreated = createAccount(result.getUserName(), result.getPassword()); |  | ||||||
|         if (accountCreated) { |  | ||||||
|             setPassword(result.getPassword()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private void setPassword(@NonNull String password) { |  | ||||||
|         Account account = getCurrentAccount(); |  | ||||||
|         if (account != null) { |  | ||||||
|             accountManager().setPassword(account, password); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * @return Account|null |  | ||||||
|      */ |  | ||||||
|     @Nullable |  | ||||||
|     public Account getCurrentAccount() { |  | ||||||
|         if (currentAccount == null) { |  | ||||||
|             AccountManager accountManager = AccountManager.get(context); |  | ||||||
|             Account[] allAccounts = accountManager.getAccountsByType(BuildConfig.ACCOUNT_TYPE); |  | ||||||
|             if (allAccounts.length != 0) { |  | ||||||
|                 currentAccount = allAccounts[0]; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return currentAccount; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public boolean doesAccountExist() { |  | ||||||
|         return getCurrentAccount() != null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     public String getUserName() { |  | ||||||
|         Account account = getCurrentAccount(); |  | ||||||
|         return account == null ? null : account.name; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     public String getPassword() { |  | ||||||
|         Account account = getCurrentAccount(); |  | ||||||
|         return account == null ? null : accountManager().getPassword(account); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private AccountManager accountManager() { |  | ||||||
|         return AccountManager.get(context); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public boolean isUserLoggedIn() { |  | ||||||
|         return defaultKvStore.getBoolean("isUserLoggedIn", false); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void setUserLoggedIn(boolean isLoggedIn) { |  | ||||||
|         defaultKvStore.putBoolean("isUserLoggedIn", isLoggedIn); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void forceLogin(Context context) { |  | ||||||
|         if (context != null) { |  | ||||||
|             LoginActivity.startYourself(context); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Returns a Completable that clears existing accounts from account manager |  | ||||||
|      */ |  | ||||||
|     public Completable logout() { |  | ||||||
|         return Completable.fromObservable( |  | ||||||
|             Observable.empty() |  | ||||||
|                       .doOnComplete( |  | ||||||
|                           () -> { |  | ||||||
|                               removeAccount(); |  | ||||||
|                               currentAccount = null; |  | ||||||
|                           } |  | ||||||
|                       ) |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Return a corresponding boolean preference |  | ||||||
|      * |  | ||||||
|      * @param key |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     public boolean getPreference(String key) { |  | ||||||
|         return defaultKvStore.getBoolean(key); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										95
									
								
								app/src/main/java/fr/free/nrw/commons/auth/SessionManager.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								app/src/main/java/fr/free/nrw/commons/auth/SessionManager.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,95 @@ | ||||||
|  | package fr.free.nrw.commons.auth | ||||||
|  | 
 | ||||||
|  | import android.accounts.Account | ||||||
|  | import android.accounts.AccountManager | ||||||
|  | import android.content.Context | ||||||
|  | import android.os.Build | ||||||
|  | import android.text.TextUtils | ||||||
|  | import fr.free.nrw.commons.BuildConfig.ACCOUNT_TYPE | ||||||
|  | import fr.free.nrw.commons.auth.login.LoginResult | ||||||
|  | import fr.free.nrw.commons.kvstore.JsonKvStore | ||||||
|  | import io.reactivex.Completable | ||||||
|  | import io.reactivex.Observable | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Named | ||||||
|  | import javax.inject.Singleton | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Manage the current logged in user session. | ||||||
|  |  */ | ||||||
|  | @Singleton | ||||||
|  | class SessionManager @Inject constructor( | ||||||
|  |     private val context: Context, | ||||||
|  |     @param:Named("default_preferences") private val defaultKvStore: JsonKvStore | ||||||
|  | ) { | ||||||
|  |     private val accountManager: AccountManager get() = AccountManager.get(context) | ||||||
|  | 
 | ||||||
|  |     private var _currentAccount: Account? = null // Unlike a savings account...  ;-) | ||||||
|  |     val currentAccount: Account? get() { | ||||||
|  |         if (_currentAccount == null) { | ||||||
|  |             val allAccounts = AccountManager.get(context).getAccountsByType(ACCOUNT_TYPE) | ||||||
|  |             if (allAccounts.isNotEmpty()) { | ||||||
|  |                 _currentAccount = allAccounts[0] | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return _currentAccount | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     val userName: String? | ||||||
|  |         get() = currentAccount?.name | ||||||
|  | 
 | ||||||
|  |     var password: String? | ||||||
|  |         get() = currentAccount?.let { accountManager.getPassword(it) } | ||||||
|  |         private set(value) { | ||||||
|  |             currentAccount?.let { accountManager.setPassword(it, value) } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     val isUserLoggedIn: Boolean | ||||||
|  |         get() = defaultKvStore.getBoolean("isUserLoggedIn", false) | ||||||
|  | 
 | ||||||
|  |     fun updateAccount(result: LoginResult) { | ||||||
|  |         if (createAccount(result.userName!!, result.password!!)) { | ||||||
|  |             password = result.password | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun doesAccountExist(): Boolean = | ||||||
|  |         currentAccount != null | ||||||
|  | 
 | ||||||
|  |     fun setUserLoggedIn(isLoggedIn: Boolean) = | ||||||
|  |         defaultKvStore.putBoolean("isUserLoggedIn", isLoggedIn) | ||||||
|  | 
 | ||||||
|  |     fun forceLogin(context: Context?) = | ||||||
|  |         context?.let { LoginActivity.startYourself(it) } | ||||||
|  | 
 | ||||||
|  |     fun getPreference(key: String): Boolean = | ||||||
|  |         defaultKvStore.getBoolean(key) | ||||||
|  | 
 | ||||||
|  |     fun logout(): Completable = Completable.fromObservable( | ||||||
|  |         Observable.empty<Any>() | ||||||
|  |             .doOnComplete { | ||||||
|  |                 removeAccount() | ||||||
|  |                 _currentAccount = null | ||||||
|  |             } | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     private fun createAccount(userName: String, password: String): Boolean { | ||||||
|  |         var account = currentAccount | ||||||
|  |         if (account == null || TextUtils.isEmpty(account.name) || account.name != userName) { | ||||||
|  |             removeAccount() | ||||||
|  |             account = Account(userName, ACCOUNT_TYPE) | ||||||
|  |             return accountManager.addAccountExplicitly(account, password, null) | ||||||
|  |         } | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun removeAccount() { | ||||||
|  |         currentAccount?.let { | ||||||
|  |             if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP_MR1) { | ||||||
|  |                 accountManager.removeAccountExplicitly(it) | ||||||
|  |             } else { | ||||||
|  |                 accountManager.removeAccount(it, null, null) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,82 +0,0 @@ | ||||||
| package fr.free.nrw.commons.auth; |  | ||||||
| 
 |  | ||||||
| import android.content.res.Configuration; |  | ||||||
| import android.os.Build; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.webkit.WebSettings; |  | ||||||
| import android.webkit.WebView; |  | ||||||
| import android.webkit.WebViewClient; |  | ||||||
| import android.widget.Toast; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.theme.BaseActivity; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| public class SignupActivity extends BaseActivity { |  | ||||||
| 
 |  | ||||||
|     private WebView webView; |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     protected void onCreate(Bundle savedInstanceState) { |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
|         Timber.d("Signup Activity started"); |  | ||||||
| 
 |  | ||||||
|         webView = new WebView(this); |  | ||||||
|         setContentView(webView); |  | ||||||
| 
 |  | ||||||
|         webView.setWebViewClient(new MyWebViewClient()); |  | ||||||
|         WebSettings webSettings = webView.getSettings(); |  | ||||||
|         /*Needed to refresh Captcha. Might introduce XSS vulnerabilities, but we can |  | ||||||
|          trust Wikimedia's site... right?*/ |  | ||||||
|         webSettings.setJavaScriptEnabled(true); |  | ||||||
| 
 |  | ||||||
|         webView.loadUrl(BuildConfig.SIGNUP_LANDING_URL); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private class MyWebViewClient extends WebViewClient { |  | ||||||
|         @Override |  | ||||||
|         public boolean shouldOverrideUrlLoading(WebView view, String url) { |  | ||||||
|             if (url.equals(BuildConfig.SIGNUP_SUCCESS_REDIRECTION_URL)) { |  | ||||||
|                 //Signup success, so clear cookies, notify user, and load LoginActivity again |  | ||||||
|                 Timber.d("Overriding URL %s", url); |  | ||||||
| 
 |  | ||||||
|                 Toast toast = Toast.makeText(SignupActivity.this, |  | ||||||
|                         R.string.account_created, Toast.LENGTH_LONG); |  | ||||||
|                 toast.show(); |  | ||||||
|                 // terminate on task completion. |  | ||||||
|                 finish(); |  | ||||||
|                 return true; |  | ||||||
|             } else { |  | ||||||
|                 //If user clicks any other links in the webview |  | ||||||
|                 Timber.d("Not overriding URL, URL is: %s", url); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onBackPressed() { |  | ||||||
|         if (webView.canGoBack()) { |  | ||||||
|             webView.goBack(); |  | ||||||
|         } else { |  | ||||||
|             super.onBackPressed(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Known bug in androidx.appcompat library version 1.1.0 being tracked here |  | ||||||
|      * https://issuetracker.google.com/issues/141132133 |  | ||||||
|      * App tries to put light/dark theme to webview and crashes in the process |  | ||||||
|      * This code tries to prevent applying the theme when sdk is between api 21 to 25 |  | ||||||
|      * @param overrideConfiguration |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void applyOverrideConfiguration(final Configuration overrideConfiguration) { |  | ||||||
|         if (Build.VERSION.SDK_INT <= 25 && |  | ||||||
|             (getResources().getConfiguration().uiMode == getApplicationContext().getResources().getConfiguration().uiMode)) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         super.applyOverrideConfiguration(overrideConfiguration); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										77
									
								
								app/src/main/java/fr/free/nrw/commons/auth/SignupActivity.kt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								app/src/main/java/fr/free/nrw/commons/auth/SignupActivity.kt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,77 @@ | ||||||
|  | package fr.free.nrw.commons.auth | ||||||
|  | 
 | ||||||
|  | import android.annotation.SuppressLint | ||||||
|  | import android.content.res.Configuration | ||||||
|  | import android.os.Build | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.webkit.WebView | ||||||
|  | import android.webkit.WebViewClient | ||||||
|  | import android.widget.Toast | ||||||
|  | import fr.free.nrw.commons.BuildConfig | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.theme.BaseActivity | ||||||
|  | import fr.free.nrw.commons.utils.applyEdgeToEdgeAllInsets | ||||||
|  | import timber.log.Timber | ||||||
|  | 
 | ||||||
|  | class SignupActivity : BaseActivity() { | ||||||
|  |     private var webView: WebView? = null | ||||||
|  | 
 | ||||||
|  |     @SuppressLint("SetJavaScriptEnabled") | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |         Timber.d("Signup Activity started") | ||||||
|  | 
 | ||||||
|  |         webView = WebView(this) | ||||||
|  |         applyEdgeToEdgeAllInsets(webView!!) | ||||||
|  |         with(webView!!) { | ||||||
|  |             setContentView(this) | ||||||
|  |             webViewClient = MyWebViewClient() | ||||||
|  |             // Needed to refresh Captcha. Might introduce XSS vulnerabilities, but we can | ||||||
|  |             // trust Wikimedia's site... right? | ||||||
|  |             settings.javaScriptEnabled = true | ||||||
|  |             loadUrl(BuildConfig.SIGNUP_LANDING_URL) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onBackPressed() { | ||||||
|  |         if (webView!!.canGoBack()) { | ||||||
|  |             webView!!.goBack() | ||||||
|  |         } else { | ||||||
|  |             super.onBackPressed() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Known bug in androidx.appcompat library version 1.1.0 being tracked here | ||||||
|  |      * https://issuetracker.google.com/issues/141132133 | ||||||
|  |      * App tries to put light/dark theme to webview and crashes in the process | ||||||
|  |      * This code tries to prevent applying the theme when sdk is between api 21 to 25 | ||||||
|  |      */ | ||||||
|  |     override fun applyOverrideConfiguration(overrideConfiguration: Configuration) { | ||||||
|  |         if (Build.VERSION.SDK_INT <= 25 && | ||||||
|  |             (resources.configuration.uiMode == applicationContext.resources.configuration.uiMode) | ||||||
|  |         ) return | ||||||
|  |         super.applyOverrideConfiguration(overrideConfiguration) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private inner class MyWebViewClient : WebViewClient() { | ||||||
|  |         @Deprecated("Deprecated in Java") | ||||||
|  |         override fun shouldOverrideUrlLoading(view: WebView, url: String): Boolean = | ||||||
|  |             if (url == BuildConfig.SIGNUP_SUCCESS_REDIRECTION_URL) { | ||||||
|  |                 //Signup success, so clear cookies, notify user, and load LoginActivity again | ||||||
|  |                 Timber.d("Overriding URL %s", url) | ||||||
|  | 
 | ||||||
|  |                 Toast.makeText( | ||||||
|  |                     this@SignupActivity, R.string.account_created, Toast.LENGTH_LONG | ||||||
|  |                 ).show() | ||||||
|  | 
 | ||||||
|  |                 // terminate on task completion. | ||||||
|  |                 finish() | ||||||
|  |                 true | ||||||
|  |             } else { | ||||||
|  |                 //If user clicks any other links in the webview | ||||||
|  |                 Timber.d("Not overriding URL, URL is: %s", url) | ||||||
|  |                 false | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,141 +0,0 @@ | ||||||
| package fr.free.nrw.commons.auth; |  | ||||||
| 
 |  | ||||||
| import android.accounts.AbstractAccountAuthenticator; |  | ||||||
| import android.accounts.Account; |  | ||||||
| import android.accounts.AccountAuthenticatorResponse; |  | ||||||
| import android.accounts.AccountManager; |  | ||||||
| import android.accounts.NetworkErrorException; |  | ||||||
| import android.content.ContentResolver; |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.os.Bundle; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.auth.AccountUtil.AUTH_TOKEN_TYPE; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles WikiMedia commons account Authentication |  | ||||||
|  */ |  | ||||||
| public class WikiAccountAuthenticator extends AbstractAccountAuthenticator { |  | ||||||
|     private static final String[] SYNC_AUTHORITIES = {BuildConfig.CONTRIBUTION_AUTHORITY, BuildConfig.MODIFICATION_AUTHORITY}; |  | ||||||
| 
 |  | ||||||
|     @NonNull |  | ||||||
|     private final Context context; |  | ||||||
| 
 |  | ||||||
|     public WikiAccountAuthenticator(@NonNull Context context) { |  | ||||||
|         super(context); |  | ||||||
|         this.context = context; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Provides Bundle with edited Account Properties  |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Bundle editProperties(AccountAuthenticatorResponse response, String accountType) { |  | ||||||
|         Bundle bundle = new Bundle(); |  | ||||||
|         bundle.putString("test", "editProperties"); |  | ||||||
|         return bundle; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Bundle addAccount(@NonNull AccountAuthenticatorResponse response, |  | ||||||
|                              @NonNull String accountType, @Nullable String authTokenType, |  | ||||||
|                              @Nullable String[] requiredFeatures, @Nullable Bundle options) |  | ||||||
|             throws NetworkErrorException { |  | ||||||
|         // account type not supported returns bundle without loginActivity Intent, it just contains "test" key  |  | ||||||
|         if (!supportedAccountType(accountType)) { |  | ||||||
|             Bundle bundle = new Bundle(); |  | ||||||
|             bundle.putString("test", "addAccount"); |  | ||||||
|             return bundle; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return addAccount(response); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Bundle confirmCredentials(@NonNull AccountAuthenticatorResponse response, |  | ||||||
|                                      @NonNull Account account, @Nullable Bundle options) |  | ||||||
|             throws NetworkErrorException { |  | ||||||
|         Bundle bundle = new Bundle(); |  | ||||||
|         bundle.putString("test", "confirmCredentials"); |  | ||||||
|         return bundle; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Bundle getAuthToken(@NonNull AccountAuthenticatorResponse response, |  | ||||||
|                                @NonNull Account account, @NonNull String authTokenType, |  | ||||||
|                                @Nullable Bundle options) |  | ||||||
|             throws NetworkErrorException { |  | ||||||
|         Bundle bundle = new Bundle(); |  | ||||||
|         bundle.putString("test", "getAuthToken"); |  | ||||||
|         return bundle; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public String getAuthTokenLabel(@NonNull String authTokenType) { |  | ||||||
|         return supportedAccountType(authTokenType) ? AUTH_TOKEN_TYPE : null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public Bundle updateCredentials(@NonNull AccountAuthenticatorResponse response, |  | ||||||
|                                     @NonNull Account account, @Nullable String authTokenType, |  | ||||||
|                                     @Nullable Bundle options) |  | ||||||
|             throws NetworkErrorException { |  | ||||||
|         Bundle bundle = new Bundle(); |  | ||||||
|         bundle.putString("test", "updateCredentials"); |  | ||||||
|         return bundle; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public Bundle hasFeatures(@NonNull AccountAuthenticatorResponse response, |  | ||||||
|                               @NonNull Account account, @NonNull String[] features) |  | ||||||
|             throws NetworkErrorException { |  | ||||||
|         Bundle bundle = new Bundle(); |  | ||||||
|         bundle.putBoolean(AccountManager.KEY_BOOLEAN_RESULT, false); |  | ||||||
|         return bundle; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private boolean supportedAccountType(@Nullable String type) { |  | ||||||
|         return BuildConfig.ACCOUNT_TYPE.equals(type); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Provides a bundle containing a Parcel  |  | ||||||
|      * the Parcel packs an Intent with LoginActivity and Authenticator response (requires valid account type) |  | ||||||
|      */ |  | ||||||
|     private Bundle addAccount(AccountAuthenticatorResponse response) { |  | ||||||
|         Intent intent = new Intent(context, LoginActivity.class); |  | ||||||
|         intent.putExtra(AccountManager.KEY_ACCOUNT_AUTHENTICATOR_RESPONSE, response); |  | ||||||
| 
 |  | ||||||
|         Bundle bundle = new Bundle(); |  | ||||||
|         bundle.putParcelable(AccountManager.KEY_INTENT, intent); |  | ||||||
| 
 |  | ||||||
|         return bundle; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Bundle getAccountRemovalAllowed(AccountAuthenticatorResponse response, |  | ||||||
|                                            Account account) throws NetworkErrorException { |  | ||||||
|         Bundle result = super.getAccountRemovalAllowed(response, account); |  | ||||||
| 
 |  | ||||||
|         if (result.containsKey(AccountManager.KEY_BOOLEAN_RESULT) |  | ||||||
|                 && !result.containsKey(AccountManager.KEY_INTENT)) { |  | ||||||
|             boolean allowed = result.getBoolean(AccountManager.KEY_BOOLEAN_RESULT); |  | ||||||
| 
 |  | ||||||
|             if (allowed) { |  | ||||||
|                 for (String auth : SYNC_AUTHORITIES) { |  | ||||||
|                     ContentResolver.cancelSync(account, auth); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return result; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,108 @@ | ||||||
|  | package fr.free.nrw.commons.auth | ||||||
|  | 
 | ||||||
|  | import android.accounts.AbstractAccountAuthenticator | ||||||
|  | import android.accounts.Account | ||||||
|  | import android.accounts.AccountAuthenticatorResponse | ||||||
|  | import android.accounts.AccountManager | ||||||
|  | import android.accounts.NetworkErrorException | ||||||
|  | import android.content.ContentResolver | ||||||
|  | import android.content.Context | ||||||
|  | import android.content.Intent | ||||||
|  | import android.os.Bundle | ||||||
|  | import androidx.core.os.bundleOf | ||||||
|  | import fr.free.nrw.commons.BuildConfig | ||||||
|  | 
 | ||||||
|  | private val SYNC_AUTHORITIES = arrayOf( | ||||||
|  |     BuildConfig.CONTRIBUTION_AUTHORITY, BuildConfig.MODIFICATION_AUTHORITY | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handles WikiMedia commons account Authentication | ||||||
|  |  */ | ||||||
|  | class WikiAccountAuthenticator( | ||||||
|  |     private val context: Context | ||||||
|  | ) : AbstractAccountAuthenticator(context) { | ||||||
|  |     /** | ||||||
|  |      * Provides Bundle with edited Account Properties | ||||||
|  |      */ | ||||||
|  |     override fun editProperties( | ||||||
|  |         response: AccountAuthenticatorResponse, | ||||||
|  |         accountType: String | ||||||
|  |     ) = bundleOf("test" to "editProperties") | ||||||
|  | 
 | ||||||
|  |     // account type not supported returns bundle without loginActivity Intent, it just contains "test" key | ||||||
|  |     @Throws(NetworkErrorException::class) | ||||||
|  |     override fun addAccount( | ||||||
|  |         response: AccountAuthenticatorResponse, | ||||||
|  |         accountType: String, | ||||||
|  |         authTokenType: String?, | ||||||
|  |         requiredFeatures: Array<String>?, | ||||||
|  |         options: Bundle? | ||||||
|  |     ) = if (BuildConfig.ACCOUNT_TYPE == accountType) { | ||||||
|  |         addAccount(response) | ||||||
|  |     } else { | ||||||
|  |         bundleOf("test" to "addAccount") | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Throws(NetworkErrorException::class) | ||||||
|  |     override fun confirmCredentials( | ||||||
|  |         response: AccountAuthenticatorResponse, account: Account, options: Bundle? | ||||||
|  |     ) = bundleOf("test" to "confirmCredentials") | ||||||
|  | 
 | ||||||
|  |     @Throws(NetworkErrorException::class) | ||||||
|  |     override fun getAuthToken( | ||||||
|  |         response: AccountAuthenticatorResponse, | ||||||
|  |         account: Account, | ||||||
|  |         authTokenType: String, | ||||||
|  |         options: Bundle? | ||||||
|  |     ) = bundleOf("test" to "getAuthToken") | ||||||
|  | 
 | ||||||
|  |     override fun getAuthTokenLabel(authTokenType: String) = | ||||||
|  |         if (BuildConfig.ACCOUNT_TYPE == authTokenType) AUTH_TOKEN_TYPE else null | ||||||
|  | 
 | ||||||
|  |     @Throws(NetworkErrorException::class) | ||||||
|  |     override fun updateCredentials( | ||||||
|  |         response: AccountAuthenticatorResponse, | ||||||
|  |         account: Account, | ||||||
|  |         authTokenType: String?, | ||||||
|  |         options: Bundle? | ||||||
|  |     ) = bundleOf("test" to "updateCredentials") | ||||||
|  | 
 | ||||||
|  |     @Throws(NetworkErrorException::class) | ||||||
|  |     override fun hasFeatures( | ||||||
|  |         response: AccountAuthenticatorResponse, | ||||||
|  |         account: Account, features: Array<String> | ||||||
|  |     ) = bundleOf(AccountManager.KEY_BOOLEAN_RESULT to false) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Provides a bundle containing a Parcel | ||||||
|  |      * the Parcel packs an Intent with LoginActivity and Authenticator response (requires valid account type) | ||||||
|  |      */ | ||||||
|  |     private fun addAccount(response: AccountAuthenticatorResponse): Bundle { | ||||||
|  |         val intent = Intent(context, LoginActivity::class.java) | ||||||
|  |             .putExtra(AccountManager.KEY_ACCOUNT_AUTHENTICATOR_RESPONSE, response) | ||||||
|  |         return bundleOf(AccountManager.KEY_INTENT to intent) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Throws(NetworkErrorException::class) | ||||||
|  |     override fun getAccountRemovalAllowed( | ||||||
|  |         response: AccountAuthenticatorResponse?, | ||||||
|  |         account: Account? | ||||||
|  |     ): Bundle { | ||||||
|  |         val result = super.getAccountRemovalAllowed(response, account) | ||||||
|  | 
 | ||||||
|  |         if (result.containsKey(AccountManager.KEY_BOOLEAN_RESULT) | ||||||
|  |             && !result.containsKey(AccountManager.KEY_INTENT) | ||||||
|  |         ) { | ||||||
|  |             val allowed = result.getBoolean(AccountManager.KEY_BOOLEAN_RESULT) | ||||||
|  | 
 | ||||||
|  |             if (allowed) { | ||||||
|  |                 for (auth in SYNC_AUTHORITIES) { | ||||||
|  |                     ContentResolver.cancelSync(account, auth) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return result | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,31 +0,0 @@ | ||||||
| package fr.free.nrw.commons.auth; |  | ||||||
| 
 |  | ||||||
| import android.accounts.AbstractAccountAuthenticator; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.os.IBinder; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerService; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles the Auth service of the App, see AndroidManifests for details |  | ||||||
|  * (Uses Dagger 2 as injector) |  | ||||||
|  */ |  | ||||||
| public class WikiAccountAuthenticatorService extends CommonsDaggerService { |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     private AbstractAccountAuthenticator authenticator; |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onCreate() { |  | ||||||
|         super.onCreate(); |  | ||||||
|         authenticator = new WikiAccountAuthenticator(this); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public IBinder onBind(Intent intent) { |  | ||||||
|         return authenticator == null ? null : authenticator.getIBinder(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | package fr.free.nrw.commons.auth | ||||||
|  | 
 | ||||||
|  | import android.accounts.AbstractAccountAuthenticator | ||||||
|  | import android.content.Intent | ||||||
|  | import android.os.IBinder | ||||||
|  | import fr.free.nrw.commons.di.CommonsDaggerService | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handles the Auth service of the App, see AndroidManifests for details | ||||||
|  |  * (Uses Dagger 2 as injector) | ||||||
|  |  */ | ||||||
|  | class WikiAccountAuthenticatorService : CommonsDaggerService() { | ||||||
|  |     private var authenticator: AbstractAccountAuthenticator? = null | ||||||
|  | 
 | ||||||
|  |     override fun onCreate() { | ||||||
|  |         super.onCreate() | ||||||
|  |         authenticator = WikiAccountAuthenticator(this) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onBind(intent: Intent): IBinder? = | ||||||
|  |         authenticator?.iBinder | ||||||
|  | } | ||||||
|  | @ -32,7 +32,7 @@ class CsrfTokenClient( | ||||||
|             try { |             try { | ||||||
|                 if (retry > 0) { |                 if (retry > 0) { | ||||||
|                     // Log in explicitly |                     // Log in explicitly | ||||||
|                     loginClient.loginBlocking(userName, password, "") |                     loginClient.loginBlocking(userName, password) | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 // Get CSRFToken response off the main thread. |                 // Get CSRFToken response off the main thread. | ||||||
|  | @ -92,6 +92,8 @@ class CsrfTokenClient( | ||||||
|                 override fun failure(caught: Throwable?) = retryWithLogin(cb) { caught } |                 override fun failure(caught: Throwable?) = retryWithLogin(cb) { caught } | ||||||
| 
 | 
 | ||||||
|                 override fun twoFactorPrompt() = cb.twoFactorPrompt() |                 override fun twoFactorPrompt() = cb.twoFactorPrompt() | ||||||
|  | 
 | ||||||
|  |                 override fun emailAuthPrompt() = cb.emailAuthPrompt() | ||||||
|             }, |             }, | ||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
|  | @ -165,10 +167,17 @@ class CsrfTokenClient( | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             override fun twoFactorPrompt( |             override fun twoFactorPrompt( | ||||||
|  |                 loginResult: LoginResult, | ||||||
|                 caught: Throwable, |                 caught: Throwable, | ||||||
|                 token: String?, |                 token: String?, | ||||||
|             ) = callback.twoFactorPrompt() |             ) = callback.twoFactorPrompt() | ||||||
| 
 | 
 | ||||||
|  |             override fun emailAuthPrompt( | ||||||
|  |                 loginResult: LoginResult, | ||||||
|  |                 caught: Throwable, | ||||||
|  |                 token: String?, | ||||||
|  |             ) = callback.emailAuthPrompt() | ||||||
|  | 
 | ||||||
|             // Should not happen here, but call the callback just in case. |             // Should not happen here, but call the callback just in case. | ||||||
|             override fun passwordResetPrompt(token: String?) = callback.failure(LoginFailedException("Logged in with temporary password.")) |             override fun passwordResetPrompt(token: String?) = callback.failure(LoginFailedException("Logged in with temporary password.")) | ||||||
| 
 | 
 | ||||||
|  | @ -190,6 +199,8 @@ class CsrfTokenClient( | ||||||
|         fun failure(caught: Throwable?) |         fun failure(caught: Throwable?) | ||||||
| 
 | 
 | ||||||
|         fun twoFactorPrompt() |         fun twoFactorPrompt() | ||||||
|  | 
 | ||||||
|  |         fun emailAuthPrompt() | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     companion object { |     companion object { | ||||||
|  |  | ||||||
|  | @ -4,6 +4,13 @@ interface LoginCallback { | ||||||
|     fun success(loginResult: LoginResult) |     fun success(loginResult: LoginResult) | ||||||
| 
 | 
 | ||||||
|     fun twoFactorPrompt( |     fun twoFactorPrompt( | ||||||
|  |         loginResult: LoginResult, | ||||||
|  |         caught: Throwable, | ||||||
|  |         token: String?, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     fun emailAuthPrompt( | ||||||
|  |         loginResult: LoginResult, | ||||||
|         caught: Throwable, |         caught: Throwable, | ||||||
|         token: String?, |         token: String?, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| package fr.free.nrw.commons.auth.login | package fr.free.nrw.commons.auth.login | ||||||
| 
 | 
 | ||||||
| import android.text.TextUtils | import android.text.TextUtils | ||||||
|  | import fr.free.nrw.commons.auth.login.LoginResult.EmailAuthResult | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult.OAuthResult | import fr.free.nrw.commons.auth.login.LoginResult.OAuthResult | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult.ResetPasswordResult | import fr.free.nrw.commons.auth.login.LoginResult.ResetPasswordResult | ||||||
| import fr.free.nrw.commons.wikidata.WikidataConstants.WIKIPEDIA_URL | import fr.free.nrw.commons.wikidata.WikidataConstants.WIKIPEDIA_URL | ||||||
|  | @ -51,6 +52,7 @@ class LoginClient( | ||||||
|                         password, |                         password, | ||||||
|                         null, |                         null, | ||||||
|                         null, |                         null, | ||||||
|  |                         null, | ||||||
|                         response.body()!!.query()!!.loginToken(), |                         response.body()!!.query()!!.loginToken(), | ||||||
|                         userLanguage, |                         userLanguage, | ||||||
|                         cb, |                         cb, | ||||||
|  | @ -75,6 +77,7 @@ class LoginClient( | ||||||
|         password: String, |         password: String, | ||||||
|         retypedPassword: String?, |         retypedPassword: String?, | ||||||
|         twoFactorCode: String?, |         twoFactorCode: String?, | ||||||
|  |         emailAuthCode: String?, | ||||||
|         loginToken: String?, |         loginToken: String?, | ||||||
|         userLanguage: String, |         userLanguage: String, | ||||||
|         cb: LoginCallback, |         cb: LoginCallback, | ||||||
|  | @ -82,7 +85,7 @@ class LoginClient( | ||||||
|         this.userLanguage = userLanguage |         this.userLanguage = userLanguage | ||||||
| 
 | 
 | ||||||
|         loginCall = |         loginCall = | ||||||
|             if (twoFactorCode.isNullOrEmpty() && retypedPassword.isNullOrEmpty()) { |             if (twoFactorCode.isNullOrEmpty() && emailAuthCode.isNullOrEmpty() && retypedPassword.isNullOrEmpty()) { | ||||||
|                 loginInterface.postLogIn(userName, password, loginToken, userLanguage, WIKIPEDIA_URL) |                 loginInterface.postLogIn(userName, password, loginToken, userLanguage, WIKIPEDIA_URL) | ||||||
|             } else { |             } else { | ||||||
|                 loginInterface.postLogIn( |                 loginInterface.postLogIn( | ||||||
|  | @ -90,6 +93,7 @@ class LoginClient( | ||||||
|                     password, |                     password, | ||||||
|                     retypedPassword, |                     retypedPassword, | ||||||
|                     twoFactorCode, |                     twoFactorCode, | ||||||
|  |                     emailAuthCode, | ||||||
|                     loginToken, |                     loginToken, | ||||||
|                     userLanguage, |                     userLanguage, | ||||||
|                     true, |                     true, | ||||||
|  | @ -112,10 +116,18 @@ class LoginClient( | ||||||
|                             when (loginResult) { |                             when (loginResult) { | ||||||
|                                 is OAuthResult -> |                                 is OAuthResult -> | ||||||
|                                     cb.twoFactorPrompt( |                                     cb.twoFactorPrompt( | ||||||
|  |                                         loginResult, | ||||||
|                                         LoginFailedException(loginResult.message), |                                         LoginFailedException(loginResult.message), | ||||||
|                                         loginToken, |                                         loginToken, | ||||||
|                                     ) |                                     ) | ||||||
| 
 | 
 | ||||||
|  |                                 is EmailAuthResult -> | ||||||
|  |                                     cb.emailAuthPrompt( | ||||||
|  |                                         loginResult, | ||||||
|  |                                         LoginFailedException(loginResult.message), | ||||||
|  |                                         loginToken | ||||||
|  |                                     ) | ||||||
|  | 
 | ||||||
|                                 is ResetPasswordResult -> cb.passwordResetPrompt(loginToken) |                                 is ResetPasswordResult -> cb.passwordResetPrompt(loginToken) | ||||||
| 
 | 
 | ||||||
|                                 is LoginResult.Result -> |                                 is LoginResult.Result -> | ||||||
|  | @ -147,6 +159,7 @@ class LoginClient( | ||||||
|     fun doLogin( |     fun doLogin( | ||||||
|         username: String, |         username: String, | ||||||
|         password: String, |         password: String, | ||||||
|  |         lastLoginResult: LoginResult?, | ||||||
|         twoFactorCode: String, |         twoFactorCode: String, | ||||||
|         userLanguage: String, |         userLanguage: String, | ||||||
|         loginCallback: LoginCallback, |         loginCallback: LoginCallback, | ||||||
|  | @ -159,7 +172,10 @@ class LoginClient( | ||||||
|                 ) = if (response.isSuccessful) { |                 ) = if (response.isSuccessful) { | ||||||
|                     val loginToken = response.body()?.query()?.loginToken() |                     val loginToken = response.body()?.query()?.loginToken() | ||||||
|                     loginToken?.let { |                     loginToken?.let { | ||||||
|                         login(username, password, null, twoFactorCode, it, userLanguage, loginCallback) |                         login(username, password, null, | ||||||
|  |                             if (lastLoginResult is OAuthResult) twoFactorCode else null, | ||||||
|  |                             if (lastLoginResult is EmailAuthResult) twoFactorCode else null, | ||||||
|  |                             it, userLanguage, loginCallback) | ||||||
|                     } ?: run { |                     } ?: run { | ||||||
|                         loginCallback.error(IOException("Failed to retrieve login token")) |                         loginCallback.error(IOException("Failed to retrieve login token")) | ||||||
|                     } |                     } | ||||||
|  | @ -181,7 +197,8 @@ class LoginClient( | ||||||
|     fun loginBlocking( |     fun loginBlocking( | ||||||
|         userName: String, |         userName: String, | ||||||
|         password: String, |         password: String, | ||||||
|         twoFactorCode: String?, |         twoFactorCode: String? = null, | ||||||
|  |         emailAuthCode: String? = null | ||||||
|     ) { |     ) { | ||||||
|         val tokenResponse = getLoginToken().execute() |         val tokenResponse = getLoginToken().execute() | ||||||
|         if (tokenResponse |         if (tokenResponse | ||||||
|  | @ -195,7 +212,7 @@ class LoginClient( | ||||||
| 
 | 
 | ||||||
|         val loginToken = tokenResponse.body()?.query()?.loginToken() |         val loginToken = tokenResponse.body()?.query()?.loginToken() | ||||||
|         val tempLoginCall = |         val tempLoginCall = | ||||||
|             if (twoFactorCode.isNullOrEmpty()) { |             if (twoFactorCode.isNullOrEmpty() && emailAuthCode.isNullOrEmpty()) { | ||||||
|                 loginInterface.postLogIn(userName, password, loginToken, userLanguage, WIKIPEDIA_URL) |                 loginInterface.postLogIn(userName, password, loginToken, userLanguage, WIKIPEDIA_URL) | ||||||
|             } else { |             } else { | ||||||
|                 loginInterface.postLogIn( |                 loginInterface.postLogIn( | ||||||
|  | @ -203,6 +220,7 @@ class LoginClient( | ||||||
|                     password, |                     password, | ||||||
|                     null, |                     null, | ||||||
|                     twoFactorCode, |                     twoFactorCode, | ||||||
|  |                     emailAuthCode, | ||||||
|                     loginToken, |                     loginToken, | ||||||
|                     userLanguage, |                     userLanguage, | ||||||
|                     true, |                     true, | ||||||
|  | @ -214,7 +232,7 @@ class LoginClient( | ||||||
|         val loginResult = loginResponse.toLoginResult(password) ?: throw IOException("Unexpected response when logging in.") |         val loginResult = loginResponse.toLoginResult(password) ?: throw IOException("Unexpected response when logging in.") | ||||||
| 
 | 
 | ||||||
|         if ("UI" == loginResult.status) { |         if ("UI" == loginResult.status) { | ||||||
|             if (loginResult is OAuthResult) { |             if (loginResult is OAuthResult || loginResult is EmailAuthResult) { | ||||||
|                 // TODO: Find a better way to boil up the warning about 2FA |                 // TODO: Find a better way to boil up the warning about 2FA | ||||||
|                 throw LoginFailedException(loginResult.message) |                 throw LoginFailedException(loginResult.message) | ||||||
|             } |             } | ||||||
|  | @ -237,7 +255,7 @@ class LoginClient( | ||||||
|         .subscribe({ response: MwQueryResponse? -> |         .subscribe({ response: MwQueryResponse? -> | ||||||
|             loginResult.userId = response?.query()?.userInfo()?.id() ?: 0 |             loginResult.userId = response?.query()?.userInfo()?.id() ?: 0 | ||||||
|             loginResult.groups = |             loginResult.groups = | ||||||
|                 response?.query()?.getUserResponse(userName)?.groups ?: emptySet() |                 response?.query()?.getUserResponse(userName)?.getGroups() ?: emptySet() | ||||||
|             cb.success(loginResult) |             cb.success(loginResult) | ||||||
|         }, { caught: Throwable -> |         }, { caught: Throwable -> | ||||||
|             Timber.e(caught, "Login succeeded but getting group information failed. ") |             Timber.e(caught, "Login succeeded but getting group information failed. ") | ||||||
|  |  | ||||||
|  | @ -35,7 +35,8 @@ interface LoginInterface { | ||||||
|         @Field("password") pass: String?, |         @Field("password") pass: String?, | ||||||
|         @Field("retype") retypedPass: String?, |         @Field("retype") retypedPass: String?, | ||||||
|         @Field("OATHToken") twoFactorCode: String?, |         @Field("OATHToken") twoFactorCode: String?, | ||||||
|         @Field("logintoken") token: String?, |         @Field("token") emailAuthToken: String?, | ||||||
|  |         @Field("logintoken") loginToken: String?, | ||||||
|         @Field("uselang") userLanguage: String?, |         @Field("uselang") userLanguage: String?, | ||||||
|         @Field("logincontinue") loginContinue: Boolean, |         @Field("logincontinue") loginContinue: Boolean, | ||||||
|     ): Call<LoginResponse?> |     ): Call<LoginResponse?> | ||||||
|  |  | ||||||
|  | @ -2,6 +2,7 @@ package fr.free.nrw.commons.auth.login | ||||||
| 
 | 
 | ||||||
| import com.google.gson.annotations.SerializedName | import com.google.gson.annotations.SerializedName | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult.OAuthResult | import fr.free.nrw.commons.auth.login.LoginResult.OAuthResult | ||||||
|  | import fr.free.nrw.commons.auth.login.LoginResult.EmailAuthResult | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult.ResetPasswordResult | import fr.free.nrw.commons.auth.login.LoginResult.ResetPasswordResult | ||||||
| import fr.free.nrw.commons.auth.login.LoginResult.Result | import fr.free.nrw.commons.auth.login.LoginResult.Result | ||||||
| import fr.free.nrw.commons.wikidata.mwapi.MwServiceError | import fr.free.nrw.commons.wikidata.mwapi.MwServiceError | ||||||
|  | @ -27,11 +28,13 @@ internal class ClientLogin { | ||||||
|     fun toLoginResult(password: String): LoginResult { |     fun toLoginResult(password: String): LoginResult { | ||||||
|         var userMessage = message |         var userMessage = message | ||||||
|         if ("UI" == status) { |         if ("UI" == status) { | ||||||
|             if (requests != null) { |             requests?.forEach { request -> | ||||||
|                 for (req in requests) { |                 request.id()?.let { | ||||||
|                     if ("MediaWiki\\Extension\\OATHAuth\\Auth\\TOTPAuthenticationRequest" == req.id()) { |                     if (it.endsWith("TOTPAuthenticationRequest")) { | ||||||
|                         return OAuthResult(status, userName, password, message) |                         return OAuthResult(status, userName, password, message) | ||||||
|                     } else if ("MediaWiki\\Auth\\PasswordAuthenticationRequest" == req.id()) { |                     } else if (it.endsWith("EmailAuthAuthenticationRequest")) { | ||||||
|  |                         return EmailAuthResult(status, userName, password, message) | ||||||
|  |                     } else if (it.endsWith("PasswordAuthenticationRequest")) { | ||||||
|                         return ResetPasswordResult(status, userName, password, message) |                         return ResetPasswordResult(status, userName, password, message) | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  | @ -49,7 +52,7 @@ internal class Request { | ||||||
|     private val required: String? = null |     private val required: String? = null | ||||||
|     private val provider: String? = null |     private val provider: String? = null | ||||||
|     private val account: String? = null |     private val account: String? = null | ||||||
|     private val fields: Map<String, RequestField>? = null |     internal val fields: Map<String, RequestField>? = null | ||||||
| 
 | 
 | ||||||
|     fun id(): String? = id |     fun id(): String? = id | ||||||
| } | } | ||||||
|  | @ -57,5 +60,5 @@ internal class Request { | ||||||
| internal class RequestField { | internal class RequestField { | ||||||
|     private val type: String? = null |     private val type: String? = null | ||||||
|     private val label: String? = null |     private val label: String? = null | ||||||
|     private val help: String? = null |     internal val help: String? = null | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -24,6 +24,13 @@ sealed class LoginResult( | ||||||
|         message: String?, |         message: String?, | ||||||
|     ) : LoginResult(status, userName, password, message) |     ) : LoginResult(status, userName, password, message) | ||||||
| 
 | 
 | ||||||
|  |     class EmailAuthResult( | ||||||
|  |         status: String, | ||||||
|  |         userName: String?, | ||||||
|  |         password: String?, | ||||||
|  |         message: String?, | ||||||
|  |     ) : LoginResult(status, userName, password, message) | ||||||
|  | 
 | ||||||
|     class ResetPasswordResult( |     class ResetPasswordResult( | ||||||
|         status: String, |         status: String, | ||||||
|         userName: String?, |         userName: String?, | ||||||
|  |  | ||||||
|  | @ -1,105 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks; |  | ||||||
| 
 |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.view.LayoutInflater; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.ViewGroup; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.fragment.app.FragmentManager; |  | ||||||
| import fr.free.nrw.commons.contributions.MainActivity; |  | ||||||
| import fr.free.nrw.commons.databinding.FragmentBookmarksBinding; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; |  | ||||||
| import fr.free.nrw.commons.kvstore.JsonKvStore; |  | ||||||
| import fr.free.nrw.commons.theme.BaseActivity; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import fr.free.nrw.commons.contributions.ContributionController; |  | ||||||
| import javax.inject.Named; |  | ||||||
| 
 |  | ||||||
| public class BookmarkFragment extends CommonsDaggerSupportFragment { |  | ||||||
| 
 |  | ||||||
|     private FragmentManager supportFragmentManager; |  | ||||||
|     private BookmarksPagerAdapter adapter; |  | ||||||
|     FragmentBookmarksBinding binding; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     ContributionController controller; |  | ||||||
|     /** |  | ||||||
|      * To check if the user is loggedIn or not. |  | ||||||
|      */ |  | ||||||
|     @Inject |  | ||||||
|     @Named("default_preferences") |  | ||||||
|     public |  | ||||||
|     JsonKvStore applicationKvStore; |  | ||||||
| 
 |  | ||||||
|     @NonNull |  | ||||||
|     public static BookmarkFragment newInstance() { |  | ||||||
|         BookmarkFragment fragment = new BookmarkFragment(); |  | ||||||
|         fragment.setRetainInstance(true); |  | ||||||
|         return fragment; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setScroll(boolean canScroll) { |  | ||||||
|         if (binding!=null) { |  | ||||||
|             binding.viewPagerBookmarks.setCanScroll(canScroll); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onCreate(@Nullable final Bundle savedInstanceState) { |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, |  | ||||||
|         @Nullable final ViewGroup container, |  | ||||||
|         @Nullable final Bundle savedInstanceState) { |  | ||||||
|         super.onCreateView(inflater, container, savedInstanceState); |  | ||||||
|         binding = FragmentBookmarksBinding.inflate(inflater, container, false); |  | ||||||
| 
 |  | ||||||
|         // Activity can call methods in the fragment by acquiring a |  | ||||||
|         // reference to the Fragment from FragmentManager, using findFragmentById() |  | ||||||
|         supportFragmentManager = getChildFragmentManager(); |  | ||||||
| 
 |  | ||||||
|         adapter = new BookmarksPagerAdapter(supportFragmentManager, getContext(), |  | ||||||
|             applicationKvStore.getBoolean("login_skipped")); |  | ||||||
|         binding.viewPagerBookmarks.setAdapter(adapter); |  | ||||||
|         binding.tabLayout.setupWithViewPager(binding.viewPagerBookmarks); |  | ||||||
| 
 |  | ||||||
|         ((MainActivity) getActivity()).showTabs(); |  | ||||||
|         ((BaseActivity) getActivity()).getSupportActionBar().setDisplayHomeAsUpEnabled(false); |  | ||||||
| 
 |  | ||||||
|         setupTabLayout(); |  | ||||||
|         return binding.getRoot(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method sets up the tab layout. If the adapter has only one element it sets the |  | ||||||
|      * visibility of tabLayout to gone. |  | ||||||
|      */ |  | ||||||
|     public void setupTabLayout() { |  | ||||||
|         binding.tabLayout.setVisibility(View.VISIBLE); |  | ||||||
|         if (adapter.getCount() == 1) { |  | ||||||
|             binding.tabLayout.setVisibility(View.GONE); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     public void onBackPressed() { |  | ||||||
|         if (((BookmarkListRootFragment) (adapter.getItem(binding.tabLayout.getSelectedTabPosition()))) |  | ||||||
|             .backPressed()) { |  | ||||||
|             // The event is handled internally by the adapter , no further action required. |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         // Event is not handled by the adapter ( performed back action ) change action bar. |  | ||||||
|         ((BaseActivity) getActivity()).getSupportActionBar().setDisplayHomeAsUpEnabled(false); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onDestroy() { |  | ||||||
|         super.onDestroy(); |  | ||||||
|         binding = null; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,98 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks | ||||||
|  | 
 | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import fr.free.nrw.commons.contributions.ContributionController | ||||||
|  | import fr.free.nrw.commons.contributions.MainActivity | ||||||
|  | import fr.free.nrw.commons.databinding.FragmentBookmarksBinding | ||||||
|  | import fr.free.nrw.commons.di.CommonsDaggerSupportFragment | ||||||
|  | import fr.free.nrw.commons.kvstore.JsonKvStore | ||||||
|  | import fr.free.nrw.commons.theme.BaseActivity | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Named | ||||||
|  | 
 | ||||||
|  | class BookmarkFragment : CommonsDaggerSupportFragment() { | ||||||
|  |     private var adapter: BookmarksPagerAdapter? = null | ||||||
|  | 
 | ||||||
|  |     @JvmField | ||||||
|  |     var binding: FragmentBookmarksBinding? = null | ||||||
|  | 
 | ||||||
|  |     @JvmField | ||||||
|  |     @Inject | ||||||
|  |     var controller: ContributionController? = null | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * To check if the user is loggedIn or not. | ||||||
|  |      */ | ||||||
|  |     @JvmField | ||||||
|  |     @Inject | ||||||
|  |     @Named("default_preferences") | ||||||
|  |     var applicationKvStore: JsonKvStore? = null | ||||||
|  | 
 | ||||||
|  |     fun setScroll(canScroll: Boolean) { | ||||||
|  |         binding?.let { | ||||||
|  |             it.viewPagerBookmarks.canScroll = canScroll | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View { | ||||||
|  |         super.onCreateView(inflater, container, savedInstanceState) | ||||||
|  |         binding = FragmentBookmarksBinding.inflate(inflater, container, false) | ||||||
|  | 
 | ||||||
|  |         // Activity can call methods in the fragment by acquiring a | ||||||
|  |         // reference to the Fragment from FragmentManager, using findFragmentById() | ||||||
|  |         val supportFragmentManager = childFragmentManager | ||||||
|  | 
 | ||||||
|  |         adapter = BookmarksPagerAdapter( | ||||||
|  |             supportFragmentManager, requireContext(), | ||||||
|  |             applicationKvStore!!.getBoolean("login_skipped") | ||||||
|  |         ) | ||||||
|  |         binding!!.viewPagerBookmarks.adapter = adapter | ||||||
|  |         binding!!.tabLayout.setupWithViewPager(binding!!.viewPagerBookmarks) | ||||||
|  | 
 | ||||||
|  |         (requireActivity() as MainActivity).showTabs() | ||||||
|  |         (requireActivity() as BaseActivity).supportActionBar!!.setDisplayHomeAsUpEnabled(false) | ||||||
|  | 
 | ||||||
|  |         setupTabLayout() | ||||||
|  |         return binding!!.root | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This method sets up the tab layout. If the adapter has only one element it sets the | ||||||
|  |      * visibility of tabLayout to gone. | ||||||
|  |      */ | ||||||
|  |     fun setupTabLayout() { | ||||||
|  |         binding!!.tabLayout.visibility = View.VISIBLE | ||||||
|  |         if (adapter!!.count == 1) { | ||||||
|  |             binding!!.tabLayout.visibility = View.GONE | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     fun onBackPressed() { | ||||||
|  |         if (((adapter!!.getItem(binding!!.tabLayout.selectedTabPosition)) as BookmarkListRootFragment).backPressed()) { | ||||||
|  |             // The event is handled internally by the adapter , no further action required. | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Event is not handled by the adapter ( performed back action ) change action bar. | ||||||
|  |         (requireActivity() as BaseActivity).supportActionBar!!.setDisplayHomeAsUpEnabled(false) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDestroy() { | ||||||
|  |         super.onDestroy() | ||||||
|  |         binding = null | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun newInstance(): BookmarkFragment = BookmarkFragment().apply { | ||||||
|  |             retainInstance = true | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,259 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks; |  | ||||||
| 
 |  | ||||||
| import android.content.Context; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.util.Log; |  | ||||||
| import android.view.LayoutInflater; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.ViewGroup; |  | ||||||
| import android.widget.AdapterView; |  | ||||||
| import android.widget.FrameLayout; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.fragment.app.Fragment; |  | ||||||
| import androidx.fragment.app.FragmentManager; |  | ||||||
| import fr.free.nrw.commons.Media; |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.bookmarks.items.BookmarkItemsFragment; |  | ||||||
| import fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsFragment; |  | ||||||
| import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesFragment; |  | ||||||
| import fr.free.nrw.commons.category.CategoryImagesCallback; |  | ||||||
| import fr.free.nrw.commons.category.GridViewAdapter; |  | ||||||
| import fr.free.nrw.commons.contributions.MainActivity; |  | ||||||
| import fr.free.nrw.commons.databinding.FragmentFeaturedRootBinding; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerSupportFragment; |  | ||||||
| import fr.free.nrw.commons.media.MediaDetailPagerFragment; |  | ||||||
| import fr.free.nrw.commons.navtab.NavTab; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.Iterator; |  | ||||||
| 
 |  | ||||||
| public class BookmarkListRootFragment extends CommonsDaggerSupportFragment implements |  | ||||||
|     FragmentManager.OnBackStackChangedListener, |  | ||||||
|     MediaDetailPagerFragment.MediaDetailProvider, |  | ||||||
|     AdapterView.OnItemClickListener, CategoryImagesCallback { |  | ||||||
| 
 |  | ||||||
|     private MediaDetailPagerFragment mediaDetails; |  | ||||||
|     //private BookmarkPicturesFragment bookmarkPicturesFragment; |  | ||||||
|     private BookmarkLocationsFragment bookmarkLocationsFragment; |  | ||||||
|     public Fragment listFragment; |  | ||||||
|     private BookmarksPagerAdapter bookmarksPagerAdapter; |  | ||||||
| 
 |  | ||||||
|     FragmentFeaturedRootBinding binding; |  | ||||||
| 
 |  | ||||||
|     public BookmarkListRootFragment() { |  | ||||||
|         //empty constructor necessary otherwise crashes on recreate |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public BookmarkListRootFragment(Bundle bundle, BookmarksPagerAdapter bookmarksPagerAdapter) { |  | ||||||
|         String title = bundle.getString("categoryName"); |  | ||||||
|         int order = bundle.getInt("order"); |  | ||||||
|         final int orderItem = bundle.getInt("orderItem"); |  | ||||||
|         if (order == 0) { |  | ||||||
|             listFragment = new BookmarkPicturesFragment(); |  | ||||||
|         } else { |  | ||||||
|             listFragment = new BookmarkLocationsFragment(); |  | ||||||
|             if(orderItem == 2) { |  | ||||||
|                 listFragment = new BookmarkItemsFragment(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         Bundle featuredArguments = new Bundle(); |  | ||||||
|         featuredArguments.putString("categoryName", title); |  | ||||||
|         listFragment.setArguments(featuredArguments); |  | ||||||
|         this.bookmarksPagerAdapter = bookmarksPagerAdapter; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, |  | ||||||
|         @Nullable final ViewGroup container, |  | ||||||
|         @Nullable final Bundle savedInstanceState) { |  | ||||||
|         super.onCreate(savedInstanceState); |  | ||||||
|         binding = FragmentFeaturedRootBinding.inflate(inflater, container, false); |  | ||||||
|         return binding.getRoot(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onViewCreated(@NonNull final View view, @Nullable final Bundle savedInstanceState) { |  | ||||||
|         super.onViewCreated(view, savedInstanceState); |  | ||||||
|         if (savedInstanceState == null) { |  | ||||||
|             setFragment(listFragment, mediaDetails); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void setFragment(Fragment fragment, Fragment otherFragment) { |  | ||||||
|         if (fragment.isAdded() && otherFragment != null) { |  | ||||||
|             getChildFragmentManager() |  | ||||||
|                 .beginTransaction() |  | ||||||
|                 .hide(otherFragment) |  | ||||||
|                 .show(fragment) |  | ||||||
|                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") |  | ||||||
|                 .commit(); |  | ||||||
|             getChildFragmentManager().executePendingTransactions(); |  | ||||||
|         } else if (fragment.isAdded() && otherFragment == null) { |  | ||||||
|             getChildFragmentManager() |  | ||||||
|                 .beginTransaction() |  | ||||||
|                 .show(fragment) |  | ||||||
|                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") |  | ||||||
|                 .commit(); |  | ||||||
|             getChildFragmentManager().executePendingTransactions(); |  | ||||||
|         } else if (!fragment.isAdded() && otherFragment != null) { |  | ||||||
|             getChildFragmentManager() |  | ||||||
|                 .beginTransaction() |  | ||||||
|                 .hide(otherFragment) |  | ||||||
|                 .add(R.id.explore_container, fragment) |  | ||||||
|                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") |  | ||||||
|                 .commit(); |  | ||||||
|             getChildFragmentManager().executePendingTransactions(); |  | ||||||
|         } else if (!fragment.isAdded()) { |  | ||||||
|             getChildFragmentManager() |  | ||||||
|                 .beginTransaction() |  | ||||||
|                 .replace(R.id.explore_container, fragment) |  | ||||||
|                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") |  | ||||||
|                 .commit(); |  | ||||||
|             getChildFragmentManager().executePendingTransactions(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public void removeFragment(Fragment fragment) { |  | ||||||
|         getChildFragmentManager() |  | ||||||
|             .beginTransaction() |  | ||||||
|             .remove(fragment) |  | ||||||
|             .commit(); |  | ||||||
|         getChildFragmentManager().executePendingTransactions(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onAttach(final Context context) { |  | ||||||
|         super.onAttach(context); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onMediaClicked(int position) { |  | ||||||
|         Log.d("deneme8", "on media clicked"); |  | ||||||
|     /*container.setVisibility(View.VISIBLE); |  | ||||||
|     ((BookmarkFragment)getParentFragment()).tabLayout.setVisibility(View.GONE); |  | ||||||
|     mediaDetails = new MediaDetailPagerFragment(false, true, position); |  | ||||||
|     setFragment(mediaDetails, bookmarkPicturesFragment);*/ |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method is called mediaDetailPagerFragment. It returns the Media Object at that Index |  | ||||||
|      * |  | ||||||
|      * @param i It is the index of which media object is to be returned which is same as current |  | ||||||
|      *          index of viewPager. |  | ||||||
|      * @return Media Object |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public Media getMediaAtPosition(int i) { |  | ||||||
|         if (bookmarksPagerAdapter.getMediaAdapter() == null) { |  | ||||||
|             // not yet ready to return data |  | ||||||
|             return null; |  | ||||||
|         } else { |  | ||||||
|             return (Media) bookmarksPagerAdapter.getMediaAdapter().getItem(i); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method is called on from getCount of MediaDetailPagerFragment The viewpager will contain |  | ||||||
|      * same number of media items as that of media elements in adapter. |  | ||||||
|      * |  | ||||||
|      * @return Total Media count in the adapter |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public int getTotalMediaCount() { |  | ||||||
|         if (bookmarksPagerAdapter.getMediaAdapter() == null) { |  | ||||||
|             return 0; |  | ||||||
|         } |  | ||||||
|         return bookmarksPagerAdapter.getMediaAdapter().getCount(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Integer getContributionStateAt(int position) { |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Reload media detail fragment once media is nominated |  | ||||||
|      * |  | ||||||
|      * @param index item position that has been nominated |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void refreshNominatedMedia(int index) { |  | ||||||
|         if (mediaDetails != null && !listFragment.isVisible()) { |  | ||||||
|             removeFragment(mediaDetails); |  | ||||||
|             mediaDetails = MediaDetailPagerFragment.newInstance(false, true); |  | ||||||
|             ((BookmarkFragment) getParentFragment()).setScroll(false); |  | ||||||
|             setFragment(mediaDetails, listFragment); |  | ||||||
|             mediaDetails.showImage(index); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * This method is called on success of API call for featured images or mobile uploads. The |  | ||||||
|      * viewpager will notified that number of items have changed. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public void viewPagerNotifyDataSetChanged() { |  | ||||||
|         if (mediaDetails != null) { |  | ||||||
|             mediaDetails.notifyDataSetChanged(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public boolean backPressed() { |  | ||||||
|         //check mediaDetailPage fragment is not null then we check mediaDetail.is Visible or not to avoid NullPointerException |  | ||||||
|         if (mediaDetails != null) { |  | ||||||
|             if (mediaDetails.isVisible()) { |  | ||||||
|                 // todo add get list fragment |  | ||||||
|                 ((BookmarkFragment) getParentFragment()).setupTabLayout(); |  | ||||||
|                 ArrayList<Integer> removed = mediaDetails.getRemovedItems(); |  | ||||||
|                 removeFragment(mediaDetails); |  | ||||||
|                 ((BookmarkFragment) getParentFragment()).setScroll(true); |  | ||||||
|                 setFragment(listFragment, mediaDetails); |  | ||||||
|                 ((MainActivity) getActivity()).showTabs(); |  | ||||||
|                 if (listFragment instanceof BookmarkPicturesFragment) { |  | ||||||
|                     GridViewAdapter adapter = ((GridViewAdapter) ((BookmarkPicturesFragment) listFragment) |  | ||||||
|                         .getAdapter()); |  | ||||||
|                     Iterator i = removed.iterator(); |  | ||||||
|                     while (i.hasNext()) { |  | ||||||
|                         adapter.remove(adapter.getItem((int) i.next())); |  | ||||||
|                     } |  | ||||||
|                     mediaDetails.clearRemoved(); |  | ||||||
| 
 |  | ||||||
|                 } |  | ||||||
|             } else { |  | ||||||
|                 moveToContributionsFragment(); |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             moveToContributionsFragment(); |  | ||||||
|         } |  | ||||||
|         // notify mediaDetails did not handled the backPressed further actions required. |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void moveToContributionsFragment() { |  | ||||||
|         ((MainActivity) getActivity()).setSelectedItemId(NavTab.CONTRIBUTIONS.code()); |  | ||||||
|         ((MainActivity) getActivity()).showTabs(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onItemClick(AdapterView<?> parent, View view, int position, long id) { |  | ||||||
|         Log.d("deneme8", "on media clicked"); |  | ||||||
|         binding.exploreContainer.setVisibility(View.VISIBLE); |  | ||||||
|         ((BookmarkFragment) getParentFragment()).binding.tabLayout.setVisibility(View.GONE); |  | ||||||
|         mediaDetails = MediaDetailPagerFragment.newInstance(false, true); |  | ||||||
|         ((BookmarkFragment) getParentFragment()).setScroll(false); |  | ||||||
|         setFragment(mediaDetails, listFragment); |  | ||||||
|         mediaDetails.showImage(position); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onBackStackChanged() { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onDestroy() { |  | ||||||
|         super.onDestroy(); |  | ||||||
|         binding = null; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,226 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks | ||||||
|  | 
 | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import android.widget.AdapterView | ||||||
|  | import android.widget.AdapterView.OnItemClickListener | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | import androidx.fragment.app.FragmentManager | ||||||
|  | import fr.free.nrw.commons.Media | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.bookmarks.category.BookmarkCategoriesFragment | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsFragment | ||||||
|  | import fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsFragment | ||||||
|  | import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesFragment | ||||||
|  | import fr.free.nrw.commons.category.CategoryImagesCallback | ||||||
|  | import fr.free.nrw.commons.category.GridViewAdapter | ||||||
|  | import fr.free.nrw.commons.contributions.MainActivity | ||||||
|  | import fr.free.nrw.commons.databinding.FragmentFeaturedRootBinding | ||||||
|  | import fr.free.nrw.commons.di.CommonsDaggerSupportFragment | ||||||
|  | import fr.free.nrw.commons.media.MediaDetailPagerFragment | ||||||
|  | import fr.free.nrw.commons.media.MediaDetailPagerFragment.Companion.newInstance | ||||||
|  | import fr.free.nrw.commons.media.MediaDetailProvider | ||||||
|  | import fr.free.nrw.commons.navtab.NavTab | ||||||
|  | import timber.log.Timber | ||||||
|  | 
 | ||||||
|  | class BookmarkListRootFragment : CommonsDaggerSupportFragment, | ||||||
|  |     FragmentManager.OnBackStackChangedListener, MediaDetailProvider, OnItemClickListener, | ||||||
|  |     CategoryImagesCallback { | ||||||
|  |     private var mediaDetails: MediaDetailPagerFragment? = null | ||||||
|  |     private val bookmarkLocationsFragment: BookmarkLocationsFragment? = null | ||||||
|  |     var listFragment: Fragment? = null | ||||||
|  |     private var bookmarksPagerAdapter: BookmarksPagerAdapter? = null | ||||||
|  | 
 | ||||||
|  |     var binding: FragmentFeaturedRootBinding? = null | ||||||
|  | 
 | ||||||
|  |     constructor() | ||||||
|  | 
 | ||||||
|  |     constructor(bundle: Bundle, bookmarksPagerAdapter: BookmarksPagerAdapter) { | ||||||
|  |         val title = bundle.getString("categoryName") | ||||||
|  |         val order = bundle.getInt("order") | ||||||
|  |         val orderItem = bundle.getInt("orderItem") | ||||||
|  | 
 | ||||||
|  |         when (order) { | ||||||
|  |             0 -> listFragment = BookmarkPicturesFragment() | ||||||
|  |             1 -> listFragment = BookmarkLocationsFragment() | ||||||
|  |             3 -> listFragment = BookmarkCategoriesFragment() | ||||||
|  |         } | ||||||
|  |         if (orderItem == 2) { | ||||||
|  |             listFragment = BookmarkItemsFragment() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val featuredArguments = Bundle() | ||||||
|  |         featuredArguments.putString("categoryName", title) | ||||||
|  |         listFragment!!.setArguments(featuredArguments) | ||||||
|  |         this.bookmarksPagerAdapter = bookmarksPagerAdapter | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View? { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |         binding = FragmentFeaturedRootBinding.inflate(inflater, container, false) | ||||||
|  |         return binding!!.getRoot() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onViewCreated(view: View, savedInstanceState: Bundle?) { | ||||||
|  |         super.onViewCreated(view, savedInstanceState) | ||||||
|  |         if (savedInstanceState == null) { | ||||||
|  |             setFragment(listFragment!!, mediaDetails) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun setFragment(fragment: Fragment, otherFragment: Fragment?) { | ||||||
|  |         if (fragment.isAdded() && otherFragment != null) { | ||||||
|  |             getChildFragmentManager() | ||||||
|  |                 .beginTransaction() | ||||||
|  |                 .hide(otherFragment) | ||||||
|  |                 .show(fragment) | ||||||
|  |                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") | ||||||
|  |                 .commit() | ||||||
|  |             getChildFragmentManager().executePendingTransactions() | ||||||
|  |         } else if (fragment.isAdded() && otherFragment == null) { | ||||||
|  |             getChildFragmentManager() | ||||||
|  |                 .beginTransaction() | ||||||
|  |                 .show(fragment) | ||||||
|  |                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") | ||||||
|  |                 .commit() | ||||||
|  |             getChildFragmentManager().executePendingTransactions() | ||||||
|  |         } else if (!fragment.isAdded() && otherFragment != null) { | ||||||
|  |             getChildFragmentManager() | ||||||
|  |                 .beginTransaction() | ||||||
|  |                 .hide(otherFragment) | ||||||
|  |                 .add(R.id.explore_container, fragment) | ||||||
|  |                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") | ||||||
|  |                 .commit() | ||||||
|  |             getChildFragmentManager().executePendingTransactions() | ||||||
|  |         } else if (!fragment.isAdded()) { | ||||||
|  |             getChildFragmentManager() | ||||||
|  |                 .beginTransaction() | ||||||
|  |                 .replace(R.id.explore_container, fragment) | ||||||
|  |                 .addToBackStack("CONTRIBUTION_LIST_FRAGMENT_TAG") | ||||||
|  |                 .commit() | ||||||
|  |             getChildFragmentManager().executePendingTransactions() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun removeFragment(fragment: Fragment) { | ||||||
|  |         getChildFragmentManager() | ||||||
|  |             .beginTransaction() | ||||||
|  |             .remove(fragment) | ||||||
|  |             .commit() | ||||||
|  |         getChildFragmentManager().executePendingTransactions() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onMediaClicked(position: Int) { | ||||||
|  |         Timber.d("on media clicked") | ||||||
|  |         /*container.setVisibility(View.VISIBLE); | ||||||
|  |     ((BookmarkFragment)getParentFragment()).tabLayout.setVisibility(View.GONE); | ||||||
|  |     mediaDetails = new MediaDetailPagerFragment(false, true, position); | ||||||
|  |     setFragment(mediaDetails, bookmarkPicturesFragment);*/ | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This method is called mediaDetailPagerFragment. It returns the Media Object at that Index | ||||||
|  |      * | ||||||
|  |      * @param i It is the index of which media object is to be returned which is same as current | ||||||
|  |      * index of viewPager. | ||||||
|  |      * @return Media Object | ||||||
|  |      */ | ||||||
|  |     override fun getMediaAtPosition(i: Int): Media? = | ||||||
|  |         bookmarksPagerAdapter!!.mediaAdapter?.getItem(i) as Media? | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This method is called on from getCount of MediaDetailPagerFragment The viewpager will contain | ||||||
|  |      * same number of media items as that of media elements in adapter. | ||||||
|  |      * | ||||||
|  |      * @return Total Media count in the adapter | ||||||
|  |      */ | ||||||
|  |     override fun getTotalMediaCount(): Int = | ||||||
|  |         bookmarksPagerAdapter!!.mediaAdapter?.count ?: 0 | ||||||
|  | 
 | ||||||
|  |     override fun getContributionStateAt(position: Int): Int? { | ||||||
|  |         return null | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Reload media detail fragment once media is nominated | ||||||
|  |      * | ||||||
|  |      * @param index item position that has been nominated | ||||||
|  |      */ | ||||||
|  |     override fun refreshNominatedMedia(index: Int) { | ||||||
|  |         if (mediaDetails != null && !listFragment!!.isVisible()) { | ||||||
|  |             removeFragment(mediaDetails!!) | ||||||
|  |             mediaDetails = newInstance(false, true) | ||||||
|  |             (parentFragment as BookmarkFragment).setScroll(false) | ||||||
|  |             setFragment(mediaDetails!!, listFragment) | ||||||
|  |             mediaDetails!!.showImage(index) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * This method is called on success of API call for featured images or mobile uploads. The | ||||||
|  |      * viewpager will notified that number of items have changed. | ||||||
|  |      */ | ||||||
|  |     override fun viewPagerNotifyDataSetChanged() { | ||||||
|  |         if (mediaDetails != null) { | ||||||
|  |             mediaDetails!!.notifyDataSetChanged() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun backPressed(): Boolean { | ||||||
|  |         //check mediaDetailPage fragment is not null then we check mediaDetail.is Visible or not to avoid NullPointerException | ||||||
|  |         if (mediaDetails != null) { | ||||||
|  |             if (mediaDetails!!.isVisible()) { | ||||||
|  |                 // todo add get list fragment | ||||||
|  |                 (parentFragment as BookmarkFragment).setupTabLayout() | ||||||
|  |                 val removed: ArrayList<Int> = mediaDetails!!.removedItems | ||||||
|  |                 removeFragment(mediaDetails!!) | ||||||
|  |                 (parentFragment as BookmarkFragment).setScroll(true) | ||||||
|  |                 setFragment(listFragment!!, mediaDetails) | ||||||
|  |                 (requireActivity() as MainActivity).showTabs() | ||||||
|  |                 if (listFragment is BookmarkPicturesFragment) { | ||||||
|  |                     val adapter = ((listFragment as BookmarkPicturesFragment) | ||||||
|  |                         .getAdapter() as GridViewAdapter?) | ||||||
|  |                     val i: MutableIterator<*> = removed.iterator() | ||||||
|  |                     while (i.hasNext()) { | ||||||
|  |                         adapter!!.remove(adapter.getItem(i.next() as Int)) | ||||||
|  |                     } | ||||||
|  |                     mediaDetails!!.clearRemoved() | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 moveToContributionsFragment() | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             moveToContributionsFragment() | ||||||
|  |         } | ||||||
|  |         // notify mediaDetails did not handled the backPressed further actions required. | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun moveToContributionsFragment() { | ||||||
|  |         (requireActivity() as MainActivity).setSelectedItemId(NavTab.CONTRIBUTIONS.code()) | ||||||
|  |         (requireActivity() as MainActivity).showTabs() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onItemClick(parent: AdapterView<*>?, view: View?, position: Int, id: Long) { | ||||||
|  |         Timber.d("on media clicked") | ||||||
|  |         binding!!.exploreContainer.visibility = View.VISIBLE | ||||||
|  |         (parentFragment as BookmarkFragment).binding!!.tabLayout.setVisibility(View.GONE) | ||||||
|  |         mediaDetails = newInstance(false, true) | ||||||
|  |         (parentFragment as BookmarkFragment).setScroll(false) | ||||||
|  |         setFragment(mediaDetails!!, listFragment) | ||||||
|  |         mediaDetails!!.showImage(position) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onBackStackChanged() = Unit | ||||||
|  | 
 | ||||||
|  |     override fun onDestroy() { | ||||||
|  |         super.onDestroy() | ||||||
|  |         binding = null | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,32 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks; |  | ||||||
| 
 |  | ||||||
| import androidx.fragment.app.Fragment; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Data class for handling a bookmark fragment and it title |  | ||||||
|  */ |  | ||||||
| public class BookmarkPages { |  | ||||||
|     private Fragment page; |  | ||||||
|     private String title; |  | ||||||
| 
 |  | ||||||
|     BookmarkPages(Fragment fragment, String title) { |  | ||||||
|         this.title = title; |  | ||||||
|         this.page = fragment; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Return the fragment |  | ||||||
|      * @return fragment object |  | ||||||
|      */ |  | ||||||
|     public Fragment getPage() { |  | ||||||
|         return page; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Return the fragment title |  | ||||||
|      * @return title |  | ||||||
|      */ |  | ||||||
|     public String getTitle() { |  | ||||||
|         return title; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,8 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks | ||||||
|  | 
 | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | 
 | ||||||
|  | data class BookmarkPages ( | ||||||
|  |     val page: Fragment? = null, | ||||||
|  |     val title: String? = null | ||||||
|  | ) | ||||||
|  | @ -1,87 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks; |  | ||||||
| 
 |  | ||||||
| import android.content.Context; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.widget.ListAdapter; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.fragment.app.Fragment; |  | ||||||
| import androidx.fragment.app.FragmentManager; |  | ||||||
| import androidx.fragment.app.FragmentPagerAdapter; |  | ||||||
| 
 |  | ||||||
| import java.util.ArrayList; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesFragment; |  | ||||||
| 
 |  | ||||||
| public class BookmarksPagerAdapter extends FragmentPagerAdapter { |  | ||||||
| 
 |  | ||||||
|     private ArrayList<BookmarkPages> pages; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Default Constructor |  | ||||||
|      * @param fm |  | ||||||
|      * @param context |  | ||||||
|      * @param onlyPictures is true if the fragment requires only BookmarkPictureFragment |  | ||||||
|      *                     (i.e. when no user is logged in). |  | ||||||
|      */ |  | ||||||
|     BookmarksPagerAdapter(FragmentManager fm, Context context,boolean onlyPictures) { |  | ||||||
|         super(fm); |  | ||||||
|         pages = new ArrayList<>(); |  | ||||||
|         Bundle picturesBundle = new Bundle(); |  | ||||||
|         picturesBundle.putString("categoryName", context.getString(R.string.title_page_bookmarks_pictures)); |  | ||||||
|         picturesBundle.putInt("order", 0); |  | ||||||
|         pages.add(new BookmarkPages( |  | ||||||
|                 new BookmarkListRootFragment(picturesBundle, this), |  | ||||||
|                 context.getString(R.string.title_page_bookmarks_pictures))); |  | ||||||
|         if (!onlyPictures) { |  | ||||||
|             // if onlyPictures is false we also add the location fragment. |  | ||||||
|             Bundle locationBundle = new Bundle(); |  | ||||||
|             locationBundle.putString("categoryName", |  | ||||||
|                 context.getString(R.string.title_page_bookmarks_locations)); |  | ||||||
|             locationBundle.putInt("order", 1); |  | ||||||
|             pages.add(new BookmarkPages( |  | ||||||
|                 new BookmarkListRootFragment(locationBundle, this), |  | ||||||
|                 context.getString(R.string.title_page_bookmarks_locations))); |  | ||||||
| 
 |  | ||||||
|             locationBundle.putInt("orderItem", 2); |  | ||||||
|             pages.add(new BookmarkPages( |  | ||||||
|                 new BookmarkListRootFragment(locationBundle, this), |  | ||||||
|                 context.getString(R.string.title_page_bookmarks_items))); |  | ||||||
|         } |  | ||||||
|         notifyDataSetChanged(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public Fragment getItem(int position) { |  | ||||||
|         return pages.get(position).getPage(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public int getCount() { |  | ||||||
|         return pages.size(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Nullable |  | ||||||
|     @Override |  | ||||||
|     public CharSequence getPageTitle(int position) { |  | ||||||
|         return pages.get(position).getTitle(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Return the Adapter used to display the picture gridview |  | ||||||
|      * @return adapter |  | ||||||
|      */ |  | ||||||
|     public ListAdapter getMediaAdapter() { |  | ||||||
|         BookmarkPicturesFragment fragment = (BookmarkPicturesFragment)(((BookmarkListRootFragment)pages.get(0).getPage()).listFragment); |  | ||||||
|         return fragment.getAdapter(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Update the pictures list for the bookmark fragment |  | ||||||
|      */ |  | ||||||
|     public void requestPictureListUpdate() { |  | ||||||
|         BookmarkPicturesFragment fragment = (BookmarkPicturesFragment)(((BookmarkListRootFragment)pages.get(0).getPage()).listFragment); |  | ||||||
|         fragment.onResume(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,82 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks | ||||||
|  | 
 | ||||||
|  | import android.content.Context | ||||||
|  | import android.widget.ListAdapter | ||||||
|  | import androidx.core.os.bundleOf | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | import androidx.fragment.app.FragmentManager | ||||||
|  | import androidx.fragment.app.FragmentPagerAdapter | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesFragment | ||||||
|  | 
 | ||||||
|  | class BookmarksPagerAdapter internal constructor( | ||||||
|  |     fm: FragmentManager, context: Context, onlyPictures: Boolean | ||||||
|  | ) : FragmentPagerAdapter(fm) { | ||||||
|  |     private val pages = mutableListOf<BookmarkPages>() | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Default Constructor | ||||||
|  |      * @param fm | ||||||
|  |      * @param context | ||||||
|  |      * @param onlyPictures is true if the fragment requires only BookmarkPictureFragment | ||||||
|  |      * (i.e. when no user is logged in). | ||||||
|  |      */ | ||||||
|  |     init { | ||||||
|  |         pages.add( | ||||||
|  |             BookmarkPages( | ||||||
|  |                 BookmarkListRootFragment( | ||||||
|  |                     bundleOf( | ||||||
|  |                         "categoryName" to context.getString(R.string.title_page_bookmarks_pictures), | ||||||
|  |                         "order" to 0 | ||||||
|  |                     ), this | ||||||
|  |                 ), context.getString(R.string.title_page_bookmarks_pictures) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         if (!onlyPictures) { | ||||||
|  |             // if onlyPictures is false we also add the location fragment. | ||||||
|  |             val locationBundle = bundleOf( | ||||||
|  |                 "categoryName" to context.getString(R.string.title_page_bookmarks_locations), | ||||||
|  |                 "order" to 1 | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |             pages.add( | ||||||
|  |                 BookmarkPages( | ||||||
|  |                     BookmarkListRootFragment(locationBundle, this), | ||||||
|  |                     context.getString(R.string.title_page_bookmarks_locations) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |             locationBundle.putInt("orderItem", 2) | ||||||
|  |             pages.add( | ||||||
|  |                 BookmarkPages( | ||||||
|  |                     BookmarkListRootFragment(locationBundle, this), | ||||||
|  |                     context.getString(R.string.title_page_bookmarks_items) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |         pages.add( | ||||||
|  |             BookmarkPages( | ||||||
|  |                 BookmarkListRootFragment( | ||||||
|  |                     bundleOf( | ||||||
|  |                         "categoryName" to context.getString(R.string.title_page_bookmarks_categories), | ||||||
|  |                         "order" to 3 | ||||||
|  |                     ), this), | ||||||
|  |                 context.getString(R.string.title_page_bookmarks_categories) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         notifyDataSetChanged() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun getItem(position: Int): Fragment = pages[position].page!! | ||||||
|  | 
 | ||||||
|  |     override fun getCount(): Int = pages.size | ||||||
|  | 
 | ||||||
|  |     override fun getPageTitle(position: Int): CharSequence? = pages[position].title | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Return the Adapter used to display the picture gridview | ||||||
|  |      * @return adapter | ||||||
|  |      */ | ||||||
|  |     val mediaAdapter: ListAdapter? | ||||||
|  |         get() = (((pages[0].page as BookmarkListRootFragment).listFragment) as BookmarkPicturesFragment).getAdapter() | ||||||
|  | } | ||||||
|  | @ -0,0 +1,52 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.category | ||||||
|  | 
 | ||||||
|  | import androidx.room.Dao | ||||||
|  | import androidx.room.Delete | ||||||
|  | import androidx.room.Insert | ||||||
|  | import androidx.room.OnConflictStrategy | ||||||
|  | import androidx.room.Query | ||||||
|  | import kotlinx.coroutines.flow.Flow | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Bookmark categories dao | ||||||
|  |  * | ||||||
|  |  * @constructor Create empty Bookmark categories dao | ||||||
|  |  */ | ||||||
|  | @Dao | ||||||
|  | interface BookmarkCategoriesDao { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Insert or Delete category bookmark into DB | ||||||
|  |      * | ||||||
|  |      * @param bookmarksCategoryModal | ||||||
|  |      */ | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     suspend fun insert(bookmarksCategoryModal: BookmarksCategoryModal) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Delete category bookmark from DB | ||||||
|  |      * | ||||||
|  |      * @param bookmarksCategoryModal | ||||||
|  |      */ | ||||||
|  |     @Delete | ||||||
|  |     suspend fun delete(bookmarksCategoryModal: BookmarksCategoryModal) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if given category exist in DB | ||||||
|  |      * | ||||||
|  |      * @param categoryName | ||||||
|  |      * @return | ||||||
|  |      */ | ||||||
|  |     @Query("SELECT EXISTS (SELECT 1 FROM bookmarks_categories WHERE categoryName = :categoryName)") | ||||||
|  |     suspend fun doesExist(categoryName: String): Boolean | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get all categories | ||||||
|  |      * | ||||||
|  |      * @return | ||||||
|  |      */ | ||||||
|  |     @Query("SELECT * FROM bookmarks_categories") | ||||||
|  |     fun getAllCategories(): Flow<List<BookmarksCategoryModal>> | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,143 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.category | ||||||
|  | 
 | ||||||
|  | import android.content.Intent | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import androidx.compose.foundation.Image | ||||||
|  | import androidx.compose.foundation.clickable | ||||||
|  | import androidx.compose.foundation.isSystemInDarkTheme | ||||||
|  | import androidx.compose.foundation.layout.Box | ||||||
|  | import androidx.compose.foundation.layout.Row | ||||||
|  | import androidx.compose.foundation.layout.fillMaxSize | ||||||
|  | import androidx.compose.foundation.layout.size | ||||||
|  | import androidx.compose.foundation.lazy.LazyColumn | ||||||
|  | import androidx.compose.foundation.lazy.items | ||||||
|  | import androidx.compose.material3.ListItem | ||||||
|  | import androidx.compose.material3.MaterialTheme | ||||||
|  | import androidx.compose.material3.Surface | ||||||
|  | import androidx.compose.material3.Text | ||||||
|  | import androidx.compose.material3.darkColorScheme | ||||||
|  | import androidx.compose.material3.lightColorScheme | ||||||
|  | import androidx.compose.runtime.Composable | ||||||
|  | import androidx.compose.runtime.getValue | ||||||
|  | import androidx.compose.ui.Alignment | ||||||
|  | import androidx.compose.ui.Modifier | ||||||
|  | import androidx.compose.ui.graphics.Color | ||||||
|  | import androidx.compose.ui.platform.ComposeView | ||||||
|  | import androidx.compose.ui.platform.ViewCompositionStrategy | ||||||
|  | import androidx.compose.ui.res.colorResource | ||||||
|  | import androidx.compose.ui.res.painterResource | ||||||
|  | import androidx.compose.ui.res.stringResource | ||||||
|  | import androidx.compose.ui.text.font.FontWeight | ||||||
|  | import androidx.compose.ui.tooling.preview.Preview | ||||||
|  | import androidx.compose.ui.unit.dp | ||||||
|  | import androidx.lifecycle.compose.collectAsStateWithLifecycle | ||||||
|  | import dagger.android.support.DaggerFragment | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.category.CategoryDetailsActivity | ||||||
|  | import javax.inject.Inject | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Tab fragment to show list of bookmarked Categories | ||||||
|  |  */ | ||||||
|  | class BookmarkCategoriesFragment : DaggerFragment() { | ||||||
|  | 
 | ||||||
|  |     @Inject | ||||||
|  |     lateinit var bookmarkCategoriesDao: BookmarkCategoriesDao | ||||||
|  | 
 | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View { | ||||||
|  |         return ComposeView(requireContext()).apply { | ||||||
|  |             setViewCompositionStrategy(ViewCompositionStrategy.DisposeOnViewTreeLifecycleDestroyed) | ||||||
|  |             setContent { | ||||||
|  |                 MaterialTheme( | ||||||
|  |                     colorScheme = if (isSystemInDarkTheme()) darkColorScheme( | ||||||
|  |                         primary = colorResource(R.color.primaryDarkColor), | ||||||
|  |                         surface = colorResource(R.color.main_background_dark), | ||||||
|  |                         background = colorResource(R.color.main_background_dark) | ||||||
|  |                     ) else lightColorScheme( | ||||||
|  |                         primary = colorResource(R.color.primaryColor), | ||||||
|  |                         surface = colorResource(R.color.main_background_light), | ||||||
|  |                         background = colorResource(R.color.main_background_light) | ||||||
|  |                     ) | ||||||
|  |                 ) { | ||||||
|  |                     val listOfBookmarks by bookmarkCategoriesDao.getAllCategories() | ||||||
|  |                         .collectAsStateWithLifecycle(initialValue = emptyList()) | ||||||
|  |                     Surface(modifier = Modifier.fillMaxSize()) { | ||||||
|  |                         Box(contentAlignment = Alignment.Center) { | ||||||
|  |                             if (listOfBookmarks.isEmpty()) { | ||||||
|  |                                 Text( | ||||||
|  |                                     text = stringResource(R.string.bookmark_empty), | ||||||
|  |                                     style = MaterialTheme.typography.bodyMedium, | ||||||
|  |                                     color = if (isSystemInDarkTheme()) Color(0xB3FFFFFF) | ||||||
|  |                                     else Color( | ||||||
|  |                                         0x8A000000 | ||||||
|  |                                     ) | ||||||
|  |                                 ) | ||||||
|  |                             } else { | ||||||
|  |                                 LazyColumn(modifier = Modifier.fillMaxSize()) { | ||||||
|  |                                     items(items = listOfBookmarks) { bookmarkItem -> | ||||||
|  |                                         CategoryItem( | ||||||
|  |                                             categoryName = bookmarkItem.categoryName, | ||||||
|  |                                             onClick = { | ||||||
|  |                                                 val categoryDetailsIntent = Intent( | ||||||
|  |                                                     requireContext(), | ||||||
|  |                                                     CategoryDetailsActivity::class.java | ||||||
|  |                                                 ).putExtra("categoryName", it) | ||||||
|  |                                                 startActivity(categoryDetailsIntent) | ||||||
|  |                                             } | ||||||
|  |                                         ) | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     @Composable | ||||||
|  |     fun CategoryItem( | ||||||
|  |         modifier: Modifier = Modifier, | ||||||
|  |         onClick: (String) -> Unit, | ||||||
|  |         categoryName: String | ||||||
|  |     ) { | ||||||
|  |         Row(modifier = modifier.clickable { | ||||||
|  |             onClick(categoryName) | ||||||
|  |         }) { | ||||||
|  |             ListItem( | ||||||
|  |                 leadingContent = { | ||||||
|  |                     Image( | ||||||
|  |                         modifier = Modifier.size(48.dp), | ||||||
|  |                         painter = painterResource(R.drawable.commons), | ||||||
|  |                         contentDescription = null | ||||||
|  |                     ) | ||||||
|  |                 }, | ||||||
|  |                 headlineContent = { | ||||||
|  |                     Text( | ||||||
|  |                         text = categoryName, | ||||||
|  |                         maxLines = 2, | ||||||
|  |                         color = if (isSystemInDarkTheme()) Color.White else Color.Black, | ||||||
|  |                         style = MaterialTheme.typography.bodyMedium, | ||||||
|  |                         fontWeight = FontWeight.SemiBold | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Preview | ||||||
|  |     @Composable | ||||||
|  |     private fun CategoryItemPreview() { | ||||||
|  |         CategoryItem( | ||||||
|  |             onClick = {}, | ||||||
|  |             categoryName = "Test Category" | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.category | ||||||
|  | 
 | ||||||
|  | import androidx.room.Entity | ||||||
|  | import androidx.room.PrimaryKey | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Data class representing bookmarked category in DB | ||||||
|  |  * | ||||||
|  |  * @property categoryName | ||||||
|  |  * @constructor Create empty Bookmarks category modal | ||||||
|  |  */ | ||||||
|  | @Entity(tableName = "bookmarks_categories") | ||||||
|  | data class BookmarksCategoryModal( | ||||||
|  |     @PrimaryKey val categoryName: String | ||||||
|  | ) | ||||||
|  | @ -1,129 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.items; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.bookmarks.items.BookmarkItemsDao.Table.COLUMN_ID; |  | ||||||
| import static fr.free.nrw.commons.bookmarks.items.BookmarkItemsDao.Table.TABLE_NAME; |  | ||||||
| 
 |  | ||||||
| import android.content.ContentValues; |  | ||||||
| import android.database.Cursor; |  | ||||||
| import android.database.sqlite.SQLiteDatabase; |  | ||||||
| import android.database.sqlite.SQLiteQueryBuilder; |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.text.TextUtils; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import fr.free.nrw.commons.data.DBOpenHelper; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerContentProvider; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles private storage for bookmarked items |  | ||||||
|  */ |  | ||||||
| public class BookmarkItemsContentProvider extends CommonsDaggerContentProvider { |  | ||||||
| 
 |  | ||||||
|     private static final String BASE_PATH = "bookmarksItems"; |  | ||||||
|     public static final Uri BASE_URI = |  | ||||||
|         Uri.parse("content://" + BuildConfig.BOOKMARK_ITEMS_AUTHORITY + "/" + BASE_PATH); |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Append bookmark items ID to the base uri |  | ||||||
|      */ |  | ||||||
|     public static Uri uriForName(final String id) { |  | ||||||
|         return Uri.parse(BASE_URI + "/" + id); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     DBOpenHelper dbOpenHelper; |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public String getType(@NonNull final Uri uri) { |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Queries the SQLite database for the bookmark items |  | ||||||
|      * @param uri : contains the uri for bookmark items |  | ||||||
|      * @param projection : contains the all fields of the table |  | ||||||
|      * @param selection : handles Where |  | ||||||
|      * @param selectionArgs : the condition of Where clause |  | ||||||
|      * @param sortOrder : ascending or descending |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public Cursor query(@NonNull final Uri uri, final String[] projection, final String selection, |  | ||||||
|         final String[] selectionArgs, final String sortOrder) { |  | ||||||
|         final SQLiteQueryBuilder queryBuilder = new SQLiteQueryBuilder(); |  | ||||||
|         queryBuilder.setTables(TABLE_NAME); |  | ||||||
|         final SQLiteDatabase db = dbOpenHelper.getReadableDatabase(); |  | ||||||
|         final Cursor cursor = queryBuilder.query(db, projection, selection, |  | ||||||
|             selectionArgs, null, null, sortOrder); |  | ||||||
|         cursor.setNotificationUri(getContext().getContentResolver(), uri); |  | ||||||
|         return cursor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the update query of local SQLite Database |  | ||||||
|      * @param uri : contains the uri for bookmark items |  | ||||||
|      * @param contentValues : new values to be entered to db |  | ||||||
|      * @param selection : handles Where |  | ||||||
|      * @param selectionArgs : the condition of Where clause |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public int update(@NonNull final Uri uri, final ContentValues contentValues, |  | ||||||
|         final String selection, final String[] selectionArgs) { |  | ||||||
|         final SQLiteDatabase sqlDB = dbOpenHelper.getWritableDatabase(); |  | ||||||
|         final int rowsUpdated; |  | ||||||
|         if (TextUtils.isEmpty(selection)) { |  | ||||||
|             final int id = Integer.parseInt(uri.getLastPathSegment()); |  | ||||||
|             rowsUpdated = sqlDB.update(TABLE_NAME, |  | ||||||
|                 contentValues, |  | ||||||
|                 COLUMN_ID + " = ?", |  | ||||||
|                 new String[]{String.valueOf(id)}); |  | ||||||
|         } else { |  | ||||||
|             throw new IllegalArgumentException( |  | ||||||
|                 "Parameter `selection` should be empty when updating an ID"); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return rowsUpdated; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the insertion of new bookmark items record to local SQLite Database |  | ||||||
|      * @param uri |  | ||||||
|      * @param contentValues |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public Uri insert(@NonNull final Uri uri, final ContentValues contentValues) { |  | ||||||
|         final SQLiteDatabase sqlDB = dbOpenHelper.getWritableDatabase(); |  | ||||||
|         final long id = sqlDB.insert(TABLE_NAME, null, contentValues); |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return Uri.parse(BASE_URI + "/" + id); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the deletion of new bookmark items record to local SQLite Database |  | ||||||
|      * @param uri |  | ||||||
|      * @param s |  | ||||||
|      * @param strings |  | ||||||
|      * @return |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public int delete(@NonNull final Uri uri, final String s, final String[] strings) { |  | ||||||
|         final int rows; |  | ||||||
|         final SQLiteDatabase db = dbOpenHelper.getReadableDatabase(); |  | ||||||
|         Timber.d("Deleting bookmark name %s", uri.getLastPathSegment()); |  | ||||||
|         rows = db.delete( |  | ||||||
|             TABLE_NAME, |  | ||||||
|             "item_id = ?", |  | ||||||
|             new String[]{uri.getLastPathSegment()} |  | ||||||
|         ); |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return rows; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,101 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.items | ||||||
|  | 
 | ||||||
|  | import android.content.ContentValues | ||||||
|  | import android.database.Cursor | ||||||
|  | import android.database.sqlite.SQLiteQueryBuilder | ||||||
|  | import android.net.Uri | ||||||
|  | import fr.free.nrw.commons.BuildConfig | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.TABLE_NAME | ||||||
|  | import fr.free.nrw.commons.di.CommonsDaggerContentProvider | ||||||
|  | import androidx.core.net.toUri | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_ID | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handles private storage for bookmarked items | ||||||
|  |  */ | ||||||
|  | class BookmarkItemsContentProvider : CommonsDaggerContentProvider() { | ||||||
|  |     override fun getType(uri: Uri): String? = null | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Queries the SQLite database for the bookmark items | ||||||
|  |      * @param uri : contains the uri for bookmark items | ||||||
|  |      * @param projection : contains the all fields of the table | ||||||
|  |      * @param selection : handles Where | ||||||
|  |      * @param selectionArgs : the condition of Where clause | ||||||
|  |      * @param sortOrder : ascending or descending | ||||||
|  |      */ | ||||||
|  |     override fun query( | ||||||
|  |         uri: Uri, projection: Array<String>?, selection: String?, | ||||||
|  |         selectionArgs: Array<String>?, sortOrder: String? | ||||||
|  |     ): Cursor { | ||||||
|  |         val queryBuilder = SQLiteQueryBuilder().apply { | ||||||
|  |             tables = TABLE_NAME | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return queryBuilder.query( | ||||||
|  |             requireDb(), projection, selection, | ||||||
|  |             selectionArgs, null, null, sortOrder | ||||||
|  |         ).apply { | ||||||
|  |             setNotificationUri(context?.contentResolver, uri) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the update query of local SQLite Database | ||||||
|  |      * @param uri : contains the uri for bookmark items | ||||||
|  |      * @param contentValues : new values to be entered to db | ||||||
|  |      * @param selection : handles Where | ||||||
|  |      * @param selectionArgs : the condition of Where clause | ||||||
|  |      */ | ||||||
|  |     override fun update( | ||||||
|  |         uri: Uri, contentValues: ContentValues?, | ||||||
|  |         selection: String?, selectionArgs: Array<String>? | ||||||
|  |     ): Int { | ||||||
|  |         val rowsUpdated: Int | ||||||
|  |         if (selection.isNullOrEmpty()) { | ||||||
|  |             val id = uri.lastPathSegment!!.toInt() | ||||||
|  |             rowsUpdated = requireDb().update( | ||||||
|  |                 TABLE_NAME, | ||||||
|  |                 contentValues, | ||||||
|  |                 "$COLUMN_ID = ?", | ||||||
|  |                 arrayOf(id.toString()) | ||||||
|  |             ) | ||||||
|  |         } else { | ||||||
|  |             throw IllegalArgumentException( | ||||||
|  |                 "Parameter `selection` should be empty when updating an ID" | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         context?.contentResolver?.notifyChange(uri, null) | ||||||
|  |         return rowsUpdated | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the insertion of new bookmark items record to local SQLite Database | ||||||
|  |      */ | ||||||
|  |     override fun insert(uri: Uri, contentValues: ContentValues?): Uri? { | ||||||
|  |         val id = requireDb().insert(TABLE_NAME, null, contentValues) | ||||||
|  |         context?.contentResolver?.notifyChange(uri, null) | ||||||
|  |         return "$BASE_URI/$id".toUri() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the deletion of new bookmark items record to local SQLite Database | ||||||
|  |      */ | ||||||
|  |     override fun delete(uri: Uri, s: String?, strings: Array<String>?): Int { | ||||||
|  |         val rows: Int = requireDb().delete( | ||||||
|  |             TABLE_NAME, | ||||||
|  |             "$COLUMN_ID = ?", | ||||||
|  |             arrayOf(uri.lastPathSegment) | ||||||
|  |         ) | ||||||
|  |        context?.contentResolver?.notifyChange(uri, null) | ||||||
|  |         return rows | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private const val BASE_PATH = "bookmarksItems" | ||||||
|  |         val BASE_URI: Uri = "content://${BuildConfig.BOOKMARK_ITEMS_AUTHORITY}/$BASE_PATH".toUri() | ||||||
|  |         fun uriForName(id: String) = "$BASE_URI/$id".toUri() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,27 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.items; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.upload.structure.depictions.DepictedItem; |  | ||||||
| import java.util.List; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Singleton; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles loading bookmarked items from Database |  | ||||||
|  */ |  | ||||||
| @Singleton |  | ||||||
| public class BookmarkItemsController { |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     BookmarkItemsDao bookmarkItemsDao; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public BookmarkItemsController() {} |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Load from DB the bookmarked items |  | ||||||
|      * @return a list of DepictedItem objects. |  | ||||||
|      */ |  | ||||||
|     public List<DepictedItem> loadFavoritesItems() { |  | ||||||
|         return bookmarkItemsDao.getAllBookmarksItems(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,23 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.items | ||||||
|  | 
 | ||||||
|  | import fr.free.nrw.commons.upload.structure.depictions.DepictedItem | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Singleton | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handles loading bookmarked items from Database | ||||||
|  |  */ | ||||||
|  | @Singleton | ||||||
|  | class BookmarkItemsController @Inject constructor() { | ||||||
|  |     @JvmField | ||||||
|  |     @Inject | ||||||
|  |     var bookmarkItemsDao: BookmarkItemsDao? = null | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Load from DB the bookmarked items | ||||||
|  |      * @return a list of DepictedItem objects. | ||||||
|  |      */ | ||||||
|  |     fun loadFavoritesItems(): List<DepictedItem> { | ||||||
|  |         return bookmarkItemsDao?.getAllBookmarksItems() ?: emptyList() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,329 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.items; |  | ||||||
| 
 |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.content.ContentProviderClient; |  | ||||||
| import android.content.ContentValues; |  | ||||||
| import android.database.Cursor; |  | ||||||
| import android.database.sqlite.SQLiteDatabase; |  | ||||||
| import android.os.RemoteException; |  | ||||||
| import fr.free.nrw.commons.category.CategoryItem; |  | ||||||
| import fr.free.nrw.commons.upload.structure.depictions.DepictedItem; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.Arrays; |  | ||||||
| import java.util.List; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Named; |  | ||||||
| import javax.inject.Provider; |  | ||||||
| import javax.inject.Singleton; |  | ||||||
| import org.apache.commons.lang3.StringUtils; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles database operations for bookmarked items |  | ||||||
|  */ |  | ||||||
| @Singleton |  | ||||||
| public class BookmarkItemsDao { |  | ||||||
| 
 |  | ||||||
|     private final Provider<ContentProviderClient> clientProvider; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public BookmarkItemsDao( |  | ||||||
|         @Named("bookmarksItem") final Provider<ContentProviderClient> clientProvider) { |  | ||||||
|         this.clientProvider = clientProvider; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Find all persisted items bookmarks on database |  | ||||||
|      * @return list of bookmarks |  | ||||||
|      */ |  | ||||||
|     public List<DepictedItem> getAllBookmarksItems() { |  | ||||||
|         final List<DepictedItem> items = new ArrayList<>(); |  | ||||||
|         final ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try (final Cursor cursor = db.query( |  | ||||||
|             BookmarkItemsContentProvider.BASE_URI, |  | ||||||
|             Table.ALL_FIELDS, |  | ||||||
|             null, |  | ||||||
|             new String[]{}, |  | ||||||
|             null)) { |  | ||||||
|             while (cursor != null && cursor.moveToNext()) { |  | ||||||
|                 items.add(fromCursor(cursor)); |  | ||||||
|             } |  | ||||||
|         } catch (final RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|         return items; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Look for a bookmark in database and in order to insert or delete it |  | ||||||
|      * @param depictedItem : Bookmark object |  | ||||||
|      * @return boolean : is bookmark now favorite ? |  | ||||||
|      */ |  | ||||||
|     public boolean updateBookmarkItem(final DepictedItem depictedItem) { |  | ||||||
|         final boolean bookmarkExists = findBookmarkItem(depictedItem.getId()); |  | ||||||
|         if (bookmarkExists) { |  | ||||||
|             deleteBookmarkItem(depictedItem); |  | ||||||
|         } else { |  | ||||||
|             addBookmarkItem(depictedItem); |  | ||||||
|         } |  | ||||||
|         return !bookmarkExists; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Add a Bookmark to database |  | ||||||
|      * @param depictedItem : Bookmark to add |  | ||||||
|      */ |  | ||||||
|     private void addBookmarkItem(final DepictedItem depictedItem) { |  | ||||||
|         final ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             db.insert(BookmarkItemsContentProvider.BASE_URI, toContentValues(depictedItem)); |  | ||||||
|         } catch (final RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Delete a bookmark from database |  | ||||||
|      * @param depictedItem : Bookmark to delete |  | ||||||
|      */ |  | ||||||
|     private void deleteBookmarkItem(final DepictedItem depictedItem) { |  | ||||||
|         final ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             db.delete(BookmarkItemsContentProvider.uriForName(depictedItem.getId()), null, null); |  | ||||||
|         } catch (final RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Find a bookmark from database based on its name |  | ||||||
|      * @param depictedItemID : Bookmark to find |  | ||||||
|      * @return boolean : is bookmark in database ? |  | ||||||
|      */ |  | ||||||
|     public boolean findBookmarkItem(final String depictedItemID) { |  | ||||||
|         if (depictedItemID == null) { //Avoiding NPE's |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|         final ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try (final Cursor cursor = db.query( |  | ||||||
|             BookmarkItemsContentProvider.BASE_URI, |  | ||||||
|             Table.ALL_FIELDS, |  | ||||||
|             Table.COLUMN_ID + "=?", |  | ||||||
|             new String[]{depictedItemID}, |  | ||||||
|             null |  | ||||||
|         )) { |  | ||||||
|             if (cursor != null && cursor.moveToFirst()) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } catch (final RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Recives real data from cursor |  | ||||||
|      * @param cursor : Object for storing database data |  | ||||||
|      * @return DepictedItem |  | ||||||
|      */ |  | ||||||
|     @SuppressLint("Range") |  | ||||||
|     DepictedItem fromCursor(final Cursor cursor) { |  | ||||||
|         final String fileName = cursor.getString(cursor.getColumnIndex(Table.COLUMN_NAME)); |  | ||||||
|         final String description |  | ||||||
|             = cursor.getString(cursor.getColumnIndex(Table.COLUMN_DESCRIPTION)); |  | ||||||
|         final String imageUrl = cursor.getString(cursor.getColumnIndex(Table.COLUMN_IMAGE)); |  | ||||||
|         final String instanceListString |  | ||||||
|             = cursor.getString(cursor.getColumnIndex(Table.COLUMN_INSTANCE_LIST)); |  | ||||||
|         final List<String> instanceList = StringToArray(instanceListString); |  | ||||||
|         final String categoryNameListString = cursor.getString(cursor |  | ||||||
|             .getColumnIndex(Table.COLUMN_CATEGORIES_NAME_LIST)); |  | ||||||
|         final List<String> categoryNameList = StringToArray(categoryNameListString); |  | ||||||
|         final String categoryDescriptionListString = cursor.getString(cursor |  | ||||||
|             .getColumnIndex(Table.COLUMN_CATEGORIES_DESCRIPTION_LIST)); |  | ||||||
|         final List<String> categoryDescriptionList = StringToArray(categoryDescriptionListString); |  | ||||||
|         final String categoryThumbnailListString = cursor.getString(cursor |  | ||||||
|             .getColumnIndex(Table.COLUMN_CATEGORIES_THUMBNAIL_LIST)); |  | ||||||
|         final List<String> categoryThumbnailList = StringToArray(categoryThumbnailListString); |  | ||||||
|         final List<CategoryItem> categoryList = convertToCategoryItems(categoryNameList, |  | ||||||
|             categoryDescriptionList, categoryThumbnailList); |  | ||||||
|         final boolean isSelected |  | ||||||
|             = Boolean.parseBoolean(cursor.getString(cursor |  | ||||||
|             .getColumnIndex(Table.COLUMN_IS_SELECTED))); |  | ||||||
|         final String id = cursor.getString(cursor.getColumnIndex(Table.COLUMN_ID)); |  | ||||||
| 
 |  | ||||||
|         return new DepictedItem( |  | ||||||
|             fileName, |  | ||||||
|             description, |  | ||||||
|             imageUrl, |  | ||||||
|             instanceList, |  | ||||||
|             categoryList, |  | ||||||
|             isSelected, |  | ||||||
|             id |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private List<CategoryItem> convertToCategoryItems(List<String> categoryNameList, |  | ||||||
|         List<String> categoryDescriptionList, List<String> categoryThumbnailList) { |  | ||||||
|         List<CategoryItem> categoryItems = new ArrayList<>(); |  | ||||||
|         for(int i=0; i<categoryNameList.size(); i++){ |  | ||||||
|             categoryItems.add(new CategoryItem(categoryNameList.get(i), |  | ||||||
|                 categoryDescriptionList.get(i), |  | ||||||
|                 categoryThumbnailList.get(i), false)); |  | ||||||
|         } |  | ||||||
|         return categoryItems; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Converts string to List |  | ||||||
|      * @param listString comma separated single string from of list items |  | ||||||
|      * @return List of string |  | ||||||
|      */ |  | ||||||
|     private List<String> StringToArray(final String listString) { |  | ||||||
|         final String[] elements = listString.split(","); |  | ||||||
|         return Arrays.asList(elements); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Converts string to List |  | ||||||
|      * @param list list of items |  | ||||||
|      * @return string comma separated single string of items |  | ||||||
|      */ |  | ||||||
|     private String ArrayToString(final List<String> list) { |  | ||||||
|         if (list != null) { |  | ||||||
|             return StringUtils.join(list, ','); |  | ||||||
|         } |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Takes data from DepictedItem and create a content value object |  | ||||||
|      * @param depictedItem depicted item |  | ||||||
|      * @return ContentValues |  | ||||||
|      */ |  | ||||||
|     private ContentValues toContentValues(final DepictedItem depictedItem) { |  | ||||||
| 
 |  | ||||||
|         final List<String> namesOfCommonsCategories = new ArrayList<>(); |  | ||||||
|         for (final CategoryItem category : |  | ||||||
|             depictedItem.getCommonsCategories()) { |  | ||||||
|             namesOfCommonsCategories.add(category.getName()); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         final List<String> descriptionsOfCommonsCategories = new ArrayList<>(); |  | ||||||
|         for (final CategoryItem category : |  | ||||||
|             depictedItem.getCommonsCategories()) { |  | ||||||
|             descriptionsOfCommonsCategories.add(category.getDescription()); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         final List<String> thumbnailsOfCommonsCategories = new ArrayList<>(); |  | ||||||
|         for (final CategoryItem category : |  | ||||||
|             depictedItem.getCommonsCategories()) { |  | ||||||
|             thumbnailsOfCommonsCategories.add(category.getThumbnail()); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         final ContentValues cv = new ContentValues(); |  | ||||||
|         cv.put(Table.COLUMN_NAME, depictedItem.getName()); |  | ||||||
|         cv.put(Table.COLUMN_DESCRIPTION, depictedItem.getDescription()); |  | ||||||
|         cv.put(Table.COLUMN_IMAGE, depictedItem.getImageUrl()); |  | ||||||
|         cv.put(Table.COLUMN_INSTANCE_LIST, ArrayToString(depictedItem.getInstanceOfs())); |  | ||||||
|         cv.put(Table.COLUMN_CATEGORIES_NAME_LIST, ArrayToString(namesOfCommonsCategories)); |  | ||||||
|         cv.put(Table.COLUMN_CATEGORIES_DESCRIPTION_LIST, |  | ||||||
|             ArrayToString(descriptionsOfCommonsCategories)); |  | ||||||
|         cv.put(Table.COLUMN_CATEGORIES_THUMBNAIL_LIST, |  | ||||||
|             ArrayToString(thumbnailsOfCommonsCategories)); |  | ||||||
|         cv.put(Table.COLUMN_IS_SELECTED, depictedItem.isSelected()); |  | ||||||
|         cv.put(Table.COLUMN_ID, depictedItem.getId()); |  | ||||||
|         return cv; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Table of bookmarksItems data |  | ||||||
|      */ |  | ||||||
|     public static final class Table { |  | ||||||
|         public static final String TABLE_NAME = "bookmarksItems"; |  | ||||||
|         public static final String COLUMN_NAME = "item_name"; |  | ||||||
|         public static final String COLUMN_DESCRIPTION = "item_description"; |  | ||||||
|         public static final String COLUMN_IMAGE = "item_image_url"; |  | ||||||
|         public static final String COLUMN_INSTANCE_LIST = "item_instance_of"; |  | ||||||
|         public static final String COLUMN_CATEGORIES_NAME_LIST = "item_name_categories"; |  | ||||||
|         public static final String COLUMN_CATEGORIES_DESCRIPTION_LIST = "item_description_categories"; |  | ||||||
|         public static final String COLUMN_CATEGORIES_THUMBNAIL_LIST = "item_thumbnail_categories"; |  | ||||||
|         public static final String COLUMN_IS_SELECTED = "item_is_selected"; |  | ||||||
|         public static final String COLUMN_ID = "item_id"; |  | ||||||
| 
 |  | ||||||
|         public static final String[] ALL_FIELDS = { |  | ||||||
|             COLUMN_NAME, |  | ||||||
|             COLUMN_DESCRIPTION, |  | ||||||
|             COLUMN_IMAGE, |  | ||||||
|             COLUMN_INSTANCE_LIST, |  | ||||||
|             COLUMN_CATEGORIES_NAME_LIST, |  | ||||||
|             COLUMN_CATEGORIES_DESCRIPTION_LIST, |  | ||||||
|             COLUMN_CATEGORIES_THUMBNAIL_LIST, |  | ||||||
|             COLUMN_IS_SELECTED, |  | ||||||
|             COLUMN_ID |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         static final String DROP_TABLE_STATEMENT = "DROP TABLE IF EXISTS " + TABLE_NAME; |  | ||||||
|         static final String CREATE_TABLE_STATEMENT = "CREATE TABLE " + TABLE_NAME + " (" |  | ||||||
|             + COLUMN_NAME + " STRING," |  | ||||||
|             + COLUMN_DESCRIPTION + " STRING," |  | ||||||
|             + COLUMN_IMAGE + " STRING," |  | ||||||
|             + COLUMN_INSTANCE_LIST + " STRING," |  | ||||||
|             + COLUMN_CATEGORIES_NAME_LIST + " STRING," |  | ||||||
|             + COLUMN_CATEGORIES_DESCRIPTION_LIST + " STRING," |  | ||||||
|             + COLUMN_CATEGORIES_THUMBNAIL_LIST + " STRING," |  | ||||||
|             + COLUMN_IS_SELECTED + " STRING," |  | ||||||
|             + COLUMN_ID + " STRING PRIMARY KEY" |  | ||||||
|             + ");"; |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Creates table |  | ||||||
|          * @param db SQLiteDatabase |  | ||||||
|          */ |  | ||||||
|         public static void onCreate(final SQLiteDatabase db) { |  | ||||||
|             db.execSQL(CREATE_TABLE_STATEMENT); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Deletes database |  | ||||||
|          * @param db SQLiteDatabase |  | ||||||
|          */ |  | ||||||
|         public static void onDelete(final SQLiteDatabase db) { |  | ||||||
|             db.execSQL(DROP_TABLE_STATEMENT); |  | ||||||
|             onCreate(db); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|          * Updates database |  | ||||||
|          * @param db SQLiteDatabase |  | ||||||
|          * @param from starting |  | ||||||
|          * @param to end |  | ||||||
|          */ |  | ||||||
|         public static void onUpdate(final SQLiteDatabase db, int from, final int to) { |  | ||||||
|             if (from == to) { |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from < 18) { |  | ||||||
|                 // doesn't exist yet |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (from == 18) { |  | ||||||
|                 // table added in version 19 |  | ||||||
|                 onCreate(db); |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,203 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.items | ||||||
|  | 
 | ||||||
|  | import android.annotation.SuppressLint | ||||||
|  | import android.content.ContentProviderClient | ||||||
|  | import android.content.ContentValues | ||||||
|  | import android.database.Cursor | ||||||
|  | import android.os.RemoteException | ||||||
|  | import androidx.core.content.contentValuesOf | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsContentProvider.Companion.BASE_URI | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsContentProvider.Companion.uriForName | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_CATEGORIES_DESCRIPTION_LIST | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_CATEGORIES_NAME_LIST | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_CATEGORIES_THUMBNAIL_LIST | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_DESCRIPTION | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_ID | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_IMAGE | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_INSTANCE_LIST | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_IS_SELECTED | ||||||
|  | import fr.free.nrw.commons.bookmarks.items.BookmarkItemsTable.COLUMN_NAME | ||||||
|  | import fr.free.nrw.commons.category.CategoryItem | ||||||
|  | import fr.free.nrw.commons.upload.structure.depictions.DepictedItem | ||||||
|  | import fr.free.nrw.commons.utils.arrayToString | ||||||
|  | import fr.free.nrw.commons.utils.getString | ||||||
|  | import fr.free.nrw.commons.utils.getStringArray | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Named | ||||||
|  | import javax.inject.Provider | ||||||
|  | import javax.inject.Singleton | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handles database operations for bookmarked items | ||||||
|  |  */ | ||||||
|  | @Singleton | ||||||
|  | class BookmarkItemsDao @Inject constructor( | ||||||
|  |     @param:Named("bookmarksItem") private val clientProvider: Provider<ContentProviderClient> | ||||||
|  | ) { | ||||||
|  |     /** | ||||||
|  |      * Find all persisted items bookmarks on database | ||||||
|  |      * @return list of bookmarks | ||||||
|  |      */ | ||||||
|  |     fun getAllBookmarksItems(): List<DepictedItem> { | ||||||
|  |         val items: MutableList<DepictedItem> = mutableListOf() | ||||||
|  |         val db = clientProvider.get() | ||||||
|  |         try { | ||||||
|  |             db.query( | ||||||
|  |                 BASE_URI, | ||||||
|  |                 BookmarkItemsTable.ALL_FIELDS, | ||||||
|  |                 null, | ||||||
|  |                 arrayOf(), | ||||||
|  |                 null | ||||||
|  |             ).use { cursor -> | ||||||
|  |                 while (cursor != null && cursor.moveToNext()) { | ||||||
|  |                     items.add(fromCursor(cursor)) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (e: RemoteException) { | ||||||
|  |             throw RuntimeException(e) | ||||||
|  |         } finally { | ||||||
|  |             db.release() | ||||||
|  |         } | ||||||
|  |         return items | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Look for a bookmark in database and in order to insert or delete it | ||||||
|  |      * @param depictedItem : Bookmark object | ||||||
|  |      * @return boolean : is bookmark now favorite ? | ||||||
|  |      */ | ||||||
|  |     fun updateBookmarkItem(depictedItem: DepictedItem): Boolean { | ||||||
|  |         val bookmarkExists = findBookmarkItem(depictedItem.id) | ||||||
|  |         if (bookmarkExists) { | ||||||
|  |             deleteBookmarkItem(depictedItem) | ||||||
|  |         } else { | ||||||
|  |             addBookmarkItem(depictedItem) | ||||||
|  |         } | ||||||
|  |         return !bookmarkExists | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Add a Bookmark to database | ||||||
|  |      * @param depictedItem : Bookmark to add | ||||||
|  |      */ | ||||||
|  |     private fun addBookmarkItem(depictedItem: DepictedItem) { | ||||||
|  |         val db = clientProvider.get() | ||||||
|  |         try { | ||||||
|  |             db.insert(BASE_URI, toContentValues(depictedItem)) | ||||||
|  |         } catch (e: RemoteException) { | ||||||
|  |             throw RuntimeException(e) | ||||||
|  |         } finally { | ||||||
|  |             db.release() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Delete a bookmark from database | ||||||
|  |      * @param depictedItem : Bookmark to delete | ||||||
|  |      */ | ||||||
|  |     private fun deleteBookmarkItem(depictedItem: DepictedItem) { | ||||||
|  |         val db = clientProvider.get() | ||||||
|  |         try { | ||||||
|  |             db.delete(uriForName(depictedItem.id), null, null) | ||||||
|  |         } catch (e: RemoteException) { | ||||||
|  |             throw RuntimeException(e) | ||||||
|  |         } finally { | ||||||
|  |             db.release() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Find a bookmark from database based on its name | ||||||
|  |      * @param depictedItemID : Bookmark to find | ||||||
|  |      * @return boolean : is bookmark in database ? | ||||||
|  |      */ | ||||||
|  |     fun findBookmarkItem(depictedItemID: String?): Boolean { | ||||||
|  |         if (depictedItemID == null) { //Avoiding NPE's | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |         val db = clientProvider.get() | ||||||
|  |         try { | ||||||
|  |             db.query( | ||||||
|  |                 BASE_URI, | ||||||
|  |                 BookmarkItemsTable.ALL_FIELDS, | ||||||
|  |                 COLUMN_ID + "=?", | ||||||
|  |                 arrayOf(depictedItemID), | ||||||
|  |                 null | ||||||
|  |             ).use { cursor -> | ||||||
|  |                 if (cursor != null && cursor.moveToFirst()) { | ||||||
|  |                     return true | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (e: RemoteException) { | ||||||
|  |             throw RuntimeException(e) | ||||||
|  |         } finally { | ||||||
|  |             db.release() | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Recives real data from cursor | ||||||
|  |      * @param cursor : Object for storing database data | ||||||
|  |      * @return DepictedItem | ||||||
|  |      */ | ||||||
|  |     @SuppressLint("Range") | ||||||
|  |     fun fromCursor(cursor: Cursor) = with(cursor) { | ||||||
|  |         var name = getString(COLUMN_NAME) | ||||||
|  |         if (name == null) { | ||||||
|  |             name = "" | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var id = getString(COLUMN_ID) | ||||||
|  |         if (id == null) { | ||||||
|  |             id = "" | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         DepictedItem( | ||||||
|  |             name, | ||||||
|  |             getString(COLUMN_DESCRIPTION), | ||||||
|  |             getString(COLUMN_IMAGE), | ||||||
|  |             getStringArray(COLUMN_INSTANCE_LIST), | ||||||
|  |             convertToCategoryItems( | ||||||
|  |                 getStringArray(COLUMN_CATEGORIES_NAME_LIST), | ||||||
|  |                 getStringArray(COLUMN_CATEGORIES_DESCRIPTION_LIST), | ||||||
|  |                 getStringArray(COLUMN_CATEGORIES_THUMBNAIL_LIST) | ||||||
|  |             ), | ||||||
|  |             getString(COLUMN_IS_SELECTED).toBoolean(), | ||||||
|  |             id | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun convertToCategoryItems( | ||||||
|  |         categoryNameList: List<String>, | ||||||
|  |         categoryDescriptionList: List<String>, | ||||||
|  |         categoryThumbnailList: List<String> | ||||||
|  |     ): List<CategoryItem> = categoryNameList.mapIndexed { index, name -> | ||||||
|  |         CategoryItem( | ||||||
|  |             name = name, | ||||||
|  |             description = categoryDescriptionList.getOrNull(index), | ||||||
|  |             thumbnail = categoryThumbnailList.getOrNull(index), | ||||||
|  |             isSelected = false | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Takes data from DepictedItem and create a content value object | ||||||
|  |      * @param depictedItem depicted item | ||||||
|  |      * @return ContentValues | ||||||
|  |      */ | ||||||
|  |     private fun toContentValues(depictedItem: DepictedItem): ContentValues { | ||||||
|  |         return contentValuesOf( | ||||||
|  |             COLUMN_NAME to depictedItem.name, | ||||||
|  |             COLUMN_DESCRIPTION to depictedItem.description, | ||||||
|  |             COLUMN_IMAGE to depictedItem.imageUrl, | ||||||
|  |             COLUMN_INSTANCE_LIST to arrayToString(depictedItem.instanceOfs), | ||||||
|  |             COLUMN_CATEGORIES_NAME_LIST to arrayToString(depictedItem.commonsCategories.map { it.name }), | ||||||
|  |             COLUMN_CATEGORIES_DESCRIPTION_LIST to arrayToString(depictedItem.commonsCategories.map { it.description }), | ||||||
|  |             COLUMN_CATEGORIES_THUMBNAIL_LIST to arrayToString(depictedItem.commonsCategories.map { it.thumbnail }), | ||||||
|  |             COLUMN_IS_SELECTED to depictedItem.isSelected, | ||||||
|  |             COLUMN_ID to depictedItem.id, | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,81 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.items; |  | ||||||
| 
 |  | ||||||
| import android.content.Context; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.view.LayoutInflater; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.ViewGroup; |  | ||||||
| import android.widget.ProgressBar; |  | ||||||
| import android.widget.RelativeLayout; |  | ||||||
| import android.widget.TextView; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.recyclerview.widget.LinearLayoutManager; |  | ||||||
| import androidx.recyclerview.widget.RecyclerView; |  | ||||||
| import dagger.android.support.DaggerFragment; |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.databinding.FragmentBookmarksItemsBinding; |  | ||||||
| import fr.free.nrw.commons.upload.structure.depictions.DepictedItem; |  | ||||||
| import java.util.List; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import org.jetbrains.annotations.NotNull; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Tab fragment to show list of bookmarked Wikidata Items |  | ||||||
|  */ |  | ||||||
| public class BookmarkItemsFragment extends DaggerFragment { |  | ||||||
| 
 |  | ||||||
|     private FragmentBookmarksItemsBinding binding; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     BookmarkItemsController controller; |  | ||||||
| 
 |  | ||||||
|     public static BookmarkItemsFragment newInstance() { |  | ||||||
|         return new BookmarkItemsFragment(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public View onCreateView( |  | ||||||
|         @NonNull final LayoutInflater inflater, |  | ||||||
|         final ViewGroup container, |  | ||||||
|         final Bundle savedInstanceState |  | ||||||
|     ) { |  | ||||||
|         binding = FragmentBookmarksItemsBinding.inflate(inflater, container, false); |  | ||||||
|         return binding.getRoot(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onViewCreated(final @NotNull View view, @Nullable final Bundle savedInstanceState) { |  | ||||||
|         super.onViewCreated(view, savedInstanceState); |  | ||||||
|         initList(requireContext()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onResume() { |  | ||||||
|         super.onResume(); |  | ||||||
|         initList(requireContext()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Get list of DepictedItem and sets to the adapter |  | ||||||
|      * @param context context |  | ||||||
|      */ |  | ||||||
|     private void initList(final Context context) { |  | ||||||
|         final List<DepictedItem> depictItems = controller.loadFavoritesItems(); |  | ||||||
|         final BookmarkItemsAdapter adapter = new BookmarkItemsAdapter(depictItems, context); |  | ||||||
|         binding.listView.setAdapter(adapter); |  | ||||||
|         binding.loadingImagesProgressBar.setVisibility(View.GONE); |  | ||||||
|         if (depictItems.isEmpty()) { |  | ||||||
|             binding.statusMessage.setText(R.string.bookmark_empty); |  | ||||||
|             binding.statusMessage.setVisibility(View.VISIBLE); |  | ||||||
|         } else { |  | ||||||
|             binding.statusMessage.setVisibility(View.GONE); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onDestroy() { |  | ||||||
|         super.onDestroy(); |  | ||||||
|         binding = null; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,62 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.items | ||||||
|  | 
 | ||||||
|  | import android.content.Context | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import dagger.android.support.DaggerFragment | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.databinding.FragmentBookmarksItemsBinding | ||||||
|  | import javax.inject.Inject | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Tab fragment to show list of bookmarked Wikidata Items | ||||||
|  |  */ | ||||||
|  | class BookmarkItemsFragment : DaggerFragment() { | ||||||
|  |     private var binding: FragmentBookmarksItemsBinding? = null | ||||||
|  | 
 | ||||||
|  |     @JvmField | ||||||
|  |     @Inject | ||||||
|  |     var controller: BookmarkItemsController? = null | ||||||
|  | 
 | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View { | ||||||
|  |         binding = FragmentBookmarksItemsBinding.inflate(inflater, container, false) | ||||||
|  |         return binding!!.root | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onViewCreated(view: View, savedInstanceState: Bundle?) { | ||||||
|  |         super.onViewCreated(view, savedInstanceState) | ||||||
|  |         initList(requireContext()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onResume() { | ||||||
|  |         super.onResume() | ||||||
|  |         initList(requireContext()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get list of DepictedItem and sets to the adapter | ||||||
|  |      * @param context context | ||||||
|  |      */ | ||||||
|  |     private fun initList(context: Context) { | ||||||
|  |         val depictItems = controller!!.loadFavoritesItems() | ||||||
|  |         binding!!.listView.adapter = BookmarkItemsAdapter(depictItems, context) | ||||||
|  |         binding!!.loadingImagesProgressBar.visibility = View.GONE | ||||||
|  |         if (depictItems.isEmpty()) { | ||||||
|  |             binding!!.statusMessage.setText(R.string.bookmark_empty) | ||||||
|  |             binding!!.statusMessage.visibility = View.VISIBLE | ||||||
|  |         } else { | ||||||
|  |             binding!!.statusMessage.visibility = View.GONE | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDestroy() { | ||||||
|  |         super.onDestroy() | ||||||
|  |         binding = null | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,90 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.items | ||||||
|  | 
 | ||||||
|  | import android.database.sqlite.SQLiteDatabase | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Table of bookmarksItems data | ||||||
|  |  */ | ||||||
|  | object BookmarkItemsTable { | ||||||
|  |     const val TABLE_NAME = "bookmarksItems" | ||||||
|  |     const val COLUMN_NAME = "item_name" | ||||||
|  |     const val COLUMN_DESCRIPTION = "item_description" | ||||||
|  |     const val COLUMN_IMAGE = "item_image_url" | ||||||
|  |     const val COLUMN_INSTANCE_LIST = "item_instance_of" | ||||||
|  |     const val COLUMN_CATEGORIES_NAME_LIST = "item_name_categories" | ||||||
|  |     const val COLUMN_CATEGORIES_DESCRIPTION_LIST = "item_description_categories" | ||||||
|  |     const val COLUMN_CATEGORIES_THUMBNAIL_LIST = "item_thumbnail_categories" | ||||||
|  |     const val COLUMN_IS_SELECTED = "item_is_selected" | ||||||
|  |     const val COLUMN_ID = "item_id" | ||||||
|  | 
 | ||||||
|  |     val ALL_FIELDS = arrayOf( | ||||||
|  |         COLUMN_NAME, | ||||||
|  |         COLUMN_DESCRIPTION, | ||||||
|  |         COLUMN_IMAGE, | ||||||
|  |         COLUMN_INSTANCE_LIST, | ||||||
|  |         COLUMN_CATEGORIES_NAME_LIST, | ||||||
|  |         COLUMN_CATEGORIES_DESCRIPTION_LIST, | ||||||
|  |         COLUMN_CATEGORIES_THUMBNAIL_LIST, | ||||||
|  |         COLUMN_IS_SELECTED, | ||||||
|  |         COLUMN_ID | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     const val DROP_TABLE_STATEMENT = "DROP TABLE IF EXISTS $TABLE_NAME" | ||||||
|  | 
 | ||||||
|  |     val CREATE_TABLE_STATEMENT = | ||||||
|  |         """CREATE TABLE $TABLE_NAME ( | ||||||
|  |              $COLUMN_NAME STRING, | ||||||
|  |              $COLUMN_DESCRIPTION STRING, | ||||||
|  |              $COLUMN_IMAGE STRING, | ||||||
|  |              $COLUMN_INSTANCE_LIST STRING, | ||||||
|  |              $COLUMN_CATEGORIES_NAME_LIST STRING, | ||||||
|  |              $COLUMN_CATEGORIES_DESCRIPTION_LIST STRING, | ||||||
|  |              $COLUMN_CATEGORIES_THUMBNAIL_LIST STRING, | ||||||
|  |              $COLUMN_IS_SELECTED STRING, | ||||||
|  |              $COLUMN_ID STRING PRIMARY KEY | ||||||
|  |            );""".trimIndent() | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Creates table | ||||||
|  |      * | ||||||
|  |      * @param db SQLiteDatabase | ||||||
|  |      */ | ||||||
|  |     fun onCreate(db: SQLiteDatabase) { | ||||||
|  |         db.execSQL(CREATE_TABLE_STATEMENT) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Deletes database | ||||||
|  |      * | ||||||
|  |      * @param db SQLiteDatabase | ||||||
|  |      */ | ||||||
|  |     fun onDelete(db: SQLiteDatabase) { | ||||||
|  |         db.execSQL(DROP_TABLE_STATEMENT) | ||||||
|  |         onCreate(db) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Updates database | ||||||
|  |      * | ||||||
|  |      * @param db   SQLiteDatabase | ||||||
|  |      * @param from starting | ||||||
|  |      * @param to   end | ||||||
|  |      */ | ||||||
|  |     fun onUpdate(db: SQLiteDatabase, from: Int, to: Int) { | ||||||
|  |         if (from == to) { | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (from < 18) { | ||||||
|  |             // doesn't exist yet | ||||||
|  |             onUpdate(db, from + 1, to) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (from == 18) { | ||||||
|  |             // table added in version 19 | ||||||
|  |             onCreate(db) | ||||||
|  |             onUpdate(db, from + 1, to) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,119 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.locations; |  | ||||||
| 
 |  | ||||||
| import android.content.ContentValues; |  | ||||||
| import android.database.Cursor; |  | ||||||
| import android.database.sqlite.SQLiteDatabase; |  | ||||||
| import android.database.sqlite.SQLiteQueryBuilder; |  | ||||||
| // We can get uri using java.Net.Uri, but andoid implimentation is faster (but it's forgiving with handling exceptions though) |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.text.TextUtils; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| 
 |  | ||||||
| import javax.inject.Inject; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import fr.free.nrw.commons.data.DBOpenHelper; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerContentProvider; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsDao.Table.COLUMN_NAME; |  | ||||||
| import static fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsDao.Table.TABLE_NAME; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles private storage for Bookmark locations |  | ||||||
|  */ |  | ||||||
| public class BookmarkLocationsContentProvider extends CommonsDaggerContentProvider { |  | ||||||
| 
 |  | ||||||
|     private static final String BASE_PATH = "bookmarksLocations"; |  | ||||||
|     public static final Uri BASE_URI = Uri.parse("content://" + BuildConfig.BOOKMARK_LOCATIONS_AUTHORITY + "/" + BASE_PATH); |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Append bookmark locations name to the base uri  |  | ||||||
|      */ |  | ||||||
|     public static Uri uriForName(String name) { |  | ||||||
|         return Uri.parse(BASE_URI.toString() + "/" + name); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Inject DBOpenHelper dbOpenHelper; |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public String getType(@NonNull Uri uri) { |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Queries the SQLite database for the bookmark locations |  | ||||||
|      * @param uri : contains the uri for bookmark locations |  | ||||||
|      * @param projection |  | ||||||
|      * @param selection : handles Where |  | ||||||
|      * @param selectionArgs : the condition of Where clause |  | ||||||
|      * @param sortOrder : ascending or descending |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public Cursor query(@NonNull Uri uri, String[] projection, String selection, |  | ||||||
|                         String[] selectionArgs, String sortOrder) { |  | ||||||
|         SQLiteQueryBuilder queryBuilder = new SQLiteQueryBuilder(); |  | ||||||
|         queryBuilder.setTables(TABLE_NAME); |  | ||||||
| 
 |  | ||||||
|         SQLiteDatabase db = dbOpenHelper.getReadableDatabase(); |  | ||||||
|         Cursor cursor = queryBuilder.query(db, projection, selection, selectionArgs, null, null, sortOrder); |  | ||||||
|         cursor.setNotificationUri(getContext().getContentResolver(), uri); |  | ||||||
| 
 |  | ||||||
|         return cursor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the update query of local SQLite Database  |  | ||||||
|      * @param uri : contains the uri for bookmark locations |  | ||||||
|      * @param contentValues : new values to be entered to db |  | ||||||
|      * @param selection : handles Where |  | ||||||
|      * @param selectionArgs : the condition of Where clause |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public int update(@NonNull Uri uri, ContentValues contentValues, String selection, |  | ||||||
|                       String[] selectionArgs) { |  | ||||||
|         SQLiteDatabase sqlDB = dbOpenHelper.getWritableDatabase(); |  | ||||||
|         int rowsUpdated; |  | ||||||
|         if (TextUtils.isEmpty(selection)) { |  | ||||||
|             int id = Integer.valueOf(uri.getLastPathSegment()); |  | ||||||
|             rowsUpdated = sqlDB.update(TABLE_NAME, |  | ||||||
|                     contentValues, |  | ||||||
|                     COLUMN_NAME + " = ?", |  | ||||||
|                     new String[]{String.valueOf(id)}); |  | ||||||
|         } else { |  | ||||||
|             throw new IllegalArgumentException( |  | ||||||
|                     "Parameter `selection` should be empty when updating an ID"); |  | ||||||
|         } |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return rowsUpdated; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the insertion of new bookmark locations record to local SQLite Database |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public Uri insert(@NonNull Uri uri, ContentValues contentValues) { |  | ||||||
|         SQLiteDatabase sqlDB = dbOpenHelper.getWritableDatabase(); |  | ||||||
|         long id = sqlDB.insert(BookmarkLocationsDao.Table.TABLE_NAME, null, contentValues); |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return Uri.parse(BASE_URI + "/" + id); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public int delete(@NonNull Uri uri, String s, String[] strings) { |  | ||||||
|         int rows; |  | ||||||
|         SQLiteDatabase db = dbOpenHelper.getReadableDatabase(); |  | ||||||
|         Timber.d("Deleting bookmark name %s", uri.getLastPathSegment()); |  | ||||||
|         rows = db.delete(TABLE_NAME, |  | ||||||
|                 "location_name = ?", |  | ||||||
|                 new String[]{uri.getLastPathSegment()} |  | ||||||
|         ); |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return rows; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,26 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.locations; |  | ||||||
| 
 |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Singleton; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.nearby.Place; |  | ||||||
| 
 |  | ||||||
| @Singleton |  | ||||||
| public class BookmarkLocationsController { |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     BookmarkLocationsDao bookmarkLocationDao; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public BookmarkLocationsController() {} |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Load from DB the bookmarked locations |  | ||||||
|      * @return a list of Place objects. |  | ||||||
|      */ |  | ||||||
|     public List<Place> loadFavoritesLocations() { |  | ||||||
|         return bookmarkLocationDao.getAllBookmarksLocations(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,20 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.locations | ||||||
|  | 
 | ||||||
|  | import fr.free.nrw.commons.nearby.Place | ||||||
|  | import kotlinx.coroutines.flow.Flow | ||||||
|  | import kotlinx.coroutines.flow.flow | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Singleton | ||||||
|  | 
 | ||||||
|  | @Singleton | ||||||
|  | class BookmarkLocationsController @Inject constructor( | ||||||
|  |     private val bookmarkLocationDao: BookmarkLocationsDao | ||||||
|  | ) { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Load bookmarked locations from the database. | ||||||
|  |      * @return a list of Place objects. | ||||||
|  |      */ | ||||||
|  |     suspend fun loadFavoritesLocations(): List<Place> = | ||||||
|  |         bookmarkLocationDao.getAllBookmarksLocationsPlace() | ||||||
|  | } | ||||||
|  | @ -1,313 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.locations; |  | ||||||
| 
 |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.content.ContentProviderClient; |  | ||||||
| import android.content.ContentValues; |  | ||||||
| import android.database.Cursor; |  | ||||||
| import android.database.sqlite.SQLiteDatabase; |  | ||||||
| import android.database.sqlite.SQLiteException; |  | ||||||
| import android.os.RemoteException; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.nearby.NearbyController; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Named; |  | ||||||
| import javax.inject.Provider; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.location.LatLng; |  | ||||||
| import fr.free.nrw.commons.nearby.Label; |  | ||||||
| import fr.free.nrw.commons.nearby.Place; |  | ||||||
| import fr.free.nrw.commons.nearby.Sitelinks; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.bookmarks.locations.BookmarkLocationsContentProvider.BASE_URI; |  | ||||||
| 
 |  | ||||||
| public class BookmarkLocationsDao { |  | ||||||
| 
 |  | ||||||
|     private final Provider<ContentProviderClient> clientProvider; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public BookmarkLocationsDao(@Named("bookmarksLocation") Provider<ContentProviderClient> clientProvider) { |  | ||||||
|         this.clientProvider = clientProvider; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      *  Find all persisted locations bookmarks on database |  | ||||||
|      * |  | ||||||
|      * @return list of Place |  | ||||||
|      */ |  | ||||||
|     @NonNull |  | ||||||
|     public List<Place> getAllBookmarksLocations() { |  | ||||||
|         List<Place> items = new ArrayList<>(); |  | ||||||
|         Cursor cursor = null; |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             cursor = db.query( |  | ||||||
|                 BookmarkLocationsContentProvider.BASE_URI, |  | ||||||
|                 Table.ALL_FIELDS, |  | ||||||
|                 null, |  | ||||||
|                 new String[]{}, |  | ||||||
|                 null); |  | ||||||
|             while (cursor != null && cursor.moveToNext()) { |  | ||||||
|                 items.add(fromCursor(cursor)); |  | ||||||
|             } |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             if (cursor != null) { |  | ||||||
|                 cursor.close(); |  | ||||||
|             } |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|         return items; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Look for a place in bookmarks table in order to insert or delete it |  | ||||||
|      * |  | ||||||
|      * @param bookmarkLocation : Place object |  | ||||||
|      * @return is Place now fav ? |  | ||||||
|      */ |  | ||||||
|     public boolean updateBookmarkLocation(Place bookmarkLocation) { |  | ||||||
|         boolean bookmarkExists = findBookmarkLocation(bookmarkLocation); |  | ||||||
|         if (bookmarkExists) { |  | ||||||
|             deleteBookmarkLocation(bookmarkLocation); |  | ||||||
|             NearbyController.updateMarkerLabelListBookmark(bookmarkLocation, false); |  | ||||||
|         } else { |  | ||||||
|             addBookmarkLocation(bookmarkLocation); |  | ||||||
|             NearbyController.updateMarkerLabelListBookmark(bookmarkLocation, true); |  | ||||||
|         } |  | ||||||
|         return !bookmarkExists; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Add a Place to bookmarks table |  | ||||||
|      * |  | ||||||
|      * @param bookmarkLocation : Place to add |  | ||||||
|      */ |  | ||||||
|     private void addBookmarkLocation(Place bookmarkLocation) { |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             db.insert(BASE_URI, toContentValues(bookmarkLocation)); |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Delete a Place from bookmarks table |  | ||||||
|      * |  | ||||||
|      * @param bookmarkLocation : Place to delete |  | ||||||
|      */ |  | ||||||
|     private void deleteBookmarkLocation(Place bookmarkLocation) { |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             db.delete(BookmarkLocationsContentProvider.uriForName(bookmarkLocation.name), null, null); |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Find a Place from database based on its name |  | ||||||
|      * |  | ||||||
|      * @param bookmarkLocation : Place to find |  | ||||||
|      * @return boolean : is Place in database ? |  | ||||||
|      */ |  | ||||||
|     public boolean findBookmarkLocation(Place bookmarkLocation) { |  | ||||||
|         Cursor cursor = null; |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             cursor = db.query( |  | ||||||
|                 BookmarkLocationsContentProvider.BASE_URI, |  | ||||||
|                 Table.ALL_FIELDS, |  | ||||||
|                 Table.COLUMN_NAME + "=?", |  | ||||||
|                 new String[]{bookmarkLocation.name}, |  | ||||||
|                 null); |  | ||||||
|             if (cursor != null && cursor.moveToFirst()) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             // This feels lazy, but to hell with checked exceptions. :) |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             if (cursor != null) { |  | ||||||
|                 cursor.close(); |  | ||||||
|             } |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressLint("Range") |  | ||||||
|     @NonNull |  | ||||||
|     Place fromCursor(final Cursor cursor) { |  | ||||||
|         final LatLng location = new LatLng(cursor.getDouble(cursor.getColumnIndex(Table.COLUMN_LAT)), |  | ||||||
|             cursor.getDouble(cursor.getColumnIndex(Table.COLUMN_LONG)), 1F); |  | ||||||
| 
 |  | ||||||
|         final Sitelinks.Builder builder = new Sitelinks.Builder(); |  | ||||||
|         builder.setWikipediaLink(cursor.getString(cursor.getColumnIndex(Table.COLUMN_WIKIPEDIA_LINK))); |  | ||||||
|         builder.setWikidataLink(cursor.getString(cursor.getColumnIndex(Table.COLUMN_WIKIDATA_LINK))); |  | ||||||
|         builder.setCommonsLink(cursor.getString(cursor.getColumnIndex(Table.COLUMN_COMMONS_LINK))); |  | ||||||
| 
 |  | ||||||
|         return new Place( |  | ||||||
|             cursor.getString(cursor.getColumnIndex(Table.COLUMN_LANGUAGE)), |  | ||||||
|             cursor.getString(cursor.getColumnIndex(Table.COLUMN_NAME)), |  | ||||||
|             Label.fromText((cursor.getString(cursor.getColumnIndex(Table.COLUMN_LABEL_TEXT)))), |  | ||||||
|             cursor.getString(cursor.getColumnIndex(Table.COLUMN_DESCRIPTION)), |  | ||||||
|             location, |  | ||||||
|             cursor.getString(cursor.getColumnIndex(Table.COLUMN_CATEGORY)), |  | ||||||
|             builder.build(), |  | ||||||
|             cursor.getString(cursor.getColumnIndex(Table.COLUMN_PIC)), |  | ||||||
|             Boolean.parseBoolean(cursor.getString(cursor.getColumnIndex(Table.COLUMN_EXISTS))) |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private ContentValues toContentValues(Place bookmarkLocation) { |  | ||||||
|         ContentValues cv = new ContentValues(); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_NAME, bookmarkLocation.getName()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_LANGUAGE, bookmarkLocation.getLanguage()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_DESCRIPTION, bookmarkLocation.getLongDescription()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_CATEGORY, bookmarkLocation.getCategory()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_LABEL_TEXT, bookmarkLocation.getLabel()!=null ? bookmarkLocation.getLabel().getText() : ""); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_LABEL_ICON, bookmarkLocation.getLabel()!=null ? bookmarkLocation.getLabel().getIcon() : null); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_WIKIPEDIA_LINK, bookmarkLocation.siteLinks.getWikipediaLink().toString()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_WIKIDATA_LINK, bookmarkLocation.siteLinks.getWikidataLink().toString()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_COMMONS_LINK, bookmarkLocation.siteLinks.getCommonsLink().toString()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_LAT, bookmarkLocation.location.getLatitude()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_LONG, bookmarkLocation.location.getLongitude()); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_PIC, bookmarkLocation.pic); |  | ||||||
|         cv.put(BookmarkLocationsDao.Table.COLUMN_EXISTS, bookmarkLocation.exists.toString()); |  | ||||||
|         return cv; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public static class Table { |  | ||||||
|         public static final String TABLE_NAME = "bookmarksLocations"; |  | ||||||
| 
 |  | ||||||
|         static final String COLUMN_NAME = "location_name"; |  | ||||||
|         static final String COLUMN_LANGUAGE = "location_language"; |  | ||||||
|         static final String COLUMN_DESCRIPTION = "location_description"; |  | ||||||
|         static final String COLUMN_LAT = "location_lat"; |  | ||||||
|         static final String COLUMN_LONG = "location_long"; |  | ||||||
|         static final String COLUMN_CATEGORY = "location_category"; |  | ||||||
|         static final String COLUMN_LABEL_TEXT = "location_label_text"; |  | ||||||
|         static final String COLUMN_LABEL_ICON = "location_label_icon"; |  | ||||||
|         static final String COLUMN_IMAGE_URL = "location_image_url"; |  | ||||||
|         static final String COLUMN_WIKIPEDIA_LINK = "location_wikipedia_link"; |  | ||||||
|         static final String COLUMN_WIKIDATA_LINK = "location_wikidata_link"; |  | ||||||
|         static final String COLUMN_COMMONS_LINK = "location_commons_link"; |  | ||||||
|         static final String COLUMN_PIC = "location_pic"; |  | ||||||
|         static final String COLUMN_EXISTS = "location_exists"; |  | ||||||
| 
 |  | ||||||
|         // NOTE! KEEP IN SAME ORDER AS THEY ARE DEFINED UP THERE. HELPS HARD CODE COLUMN INDICES. |  | ||||||
|         public static final String[] ALL_FIELDS = { |  | ||||||
|             COLUMN_NAME, |  | ||||||
|             COLUMN_LANGUAGE, |  | ||||||
|             COLUMN_DESCRIPTION, |  | ||||||
|             COLUMN_CATEGORY, |  | ||||||
|             COLUMN_LABEL_TEXT, |  | ||||||
|             COLUMN_LABEL_ICON, |  | ||||||
|             COLUMN_LAT, |  | ||||||
|             COLUMN_LONG, |  | ||||||
|             COLUMN_IMAGE_URL, |  | ||||||
|             COLUMN_WIKIPEDIA_LINK, |  | ||||||
|             COLUMN_WIKIDATA_LINK, |  | ||||||
|             COLUMN_COMMONS_LINK, |  | ||||||
|             COLUMN_PIC, |  | ||||||
|             COLUMN_EXISTS, |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         static final String DROP_TABLE_STATEMENT = "DROP TABLE IF EXISTS " + TABLE_NAME; |  | ||||||
| 
 |  | ||||||
|         static final String CREATE_TABLE_STATEMENT = "CREATE TABLE " + TABLE_NAME + " (" |  | ||||||
|             + COLUMN_NAME + " STRING PRIMARY KEY," |  | ||||||
|             + COLUMN_LANGUAGE + " STRING," |  | ||||||
|             + COLUMN_DESCRIPTION + " STRING," |  | ||||||
|             + COLUMN_CATEGORY + " STRING," |  | ||||||
|             + COLUMN_LABEL_TEXT + " STRING," |  | ||||||
|             + COLUMN_LABEL_ICON + " INTEGER," |  | ||||||
|             + COLUMN_LAT + " DOUBLE," |  | ||||||
|             + COLUMN_LONG + " DOUBLE," |  | ||||||
|             + COLUMN_IMAGE_URL + " STRING," |  | ||||||
|             + COLUMN_WIKIPEDIA_LINK + " STRING," |  | ||||||
|             + COLUMN_WIKIDATA_LINK + " STRING," |  | ||||||
|             + COLUMN_COMMONS_LINK + " STRING," |  | ||||||
|             + COLUMN_PIC + " STRING," |  | ||||||
|             + COLUMN_EXISTS + " STRING" |  | ||||||
|             + ");"; |  | ||||||
| 
 |  | ||||||
|         public static void onCreate(SQLiteDatabase db) { |  | ||||||
|             db.execSQL(CREATE_TABLE_STATEMENT); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void onDelete(SQLiteDatabase db) { |  | ||||||
|             db.execSQL(DROP_TABLE_STATEMENT); |  | ||||||
|             onCreate(db); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void onUpdate(final SQLiteDatabase db, int from, final int to) { |  | ||||||
|             Timber.d("bookmarksLocations db is updated from:"+from+", to:"+to); |  | ||||||
|             if (from == to) { |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from < 7) { |  | ||||||
|                 // doesn't exist yet |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from == 7) { |  | ||||||
|                 // table added in version 8 |  | ||||||
|                 onCreate(db); |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from < 10) { |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from == 10) { |  | ||||||
|                 //This is safe, and can be called clean, as we/I do not remember the appropriate version for this |  | ||||||
|                 //We are anyways switching to room, these things won't be necessary then |  | ||||||
|                 try { |  | ||||||
|                     db.execSQL("ALTER TABLE bookmarksLocations ADD COLUMN location_pic STRING;"); |  | ||||||
|                 }catch (SQLiteException exception){ |  | ||||||
|                     Timber.e(exception);// |  | ||||||
|                 } |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from >= 12) { |  | ||||||
|                 try { |  | ||||||
|                     db.execSQL( |  | ||||||
|                         "ALTER TABLE bookmarksLocations ADD COLUMN location_destroyed STRING;"); |  | ||||||
|                 } catch (SQLiteException exception) { |  | ||||||
|                     Timber.e(exception); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (from >= 13){ |  | ||||||
|                 try { |  | ||||||
|                     db.execSQL("ALTER TABLE bookmarksLocations ADD COLUMN location_language STRING;"); |  | ||||||
|                 } catch (SQLiteException exception){ |  | ||||||
|                     Timber.e(exception); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (from >= 14){ |  | ||||||
|                 try { |  | ||||||
|                     db.execSQL("ALTER TABLE bookmarksLocations ADD COLUMN location_exists STRING;"); |  | ||||||
|                 } catch (SQLiteException exception){ |  | ||||||
|                     Timber.e(exception); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,65 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.locations | ||||||
|  | 
 | ||||||
|  | import androidx.room.Dao | ||||||
|  | import androidx.room.Delete | ||||||
|  | import androidx.room.Insert | ||||||
|  | import androidx.room.OnConflictStrategy | ||||||
|  | import androidx.room.Query | ||||||
|  | import fr.free.nrw.commons.nearby.NearbyController | ||||||
|  | import fr.free.nrw.commons.nearby.Place | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * DAO for managing bookmark locations in the database. | ||||||
|  |  */ | ||||||
|  | @Dao | ||||||
|  | abstract class BookmarkLocationsDao { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Adds or updates a bookmark location in the database. | ||||||
|  |      */ | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     abstract suspend fun addBookmarkLocation(bookmarkLocation: BookmarksLocations) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Fetches all bookmark locations from the database. | ||||||
|  |      */ | ||||||
|  |     @Query("SELECT * FROM bookmarks_locations") | ||||||
|  |     abstract suspend fun getAllBookmarksLocations(): List<BookmarksLocations> | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if a bookmark location exists by name. | ||||||
|  |      */ | ||||||
|  |     @Query("SELECT EXISTS (SELECT 1 FROM bookmarks_locations WHERE location_name = :name)") | ||||||
|  |     abstract suspend fun findBookmarkLocation(name: String): Boolean | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Deletes a bookmark location from the database. | ||||||
|  |      */ | ||||||
|  |     @Delete | ||||||
|  |     abstract suspend fun deleteBookmarkLocation(bookmarkLocation: BookmarksLocations) | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Adds or removes a bookmark location and updates markers. | ||||||
|  |      * @return `true` if added, `false` if removed. | ||||||
|  |      */ | ||||||
|  |     suspend fun updateBookmarkLocation(bookmarkLocation: Place): Boolean { | ||||||
|  |         val exists = findBookmarkLocation(bookmarkLocation.name) | ||||||
|  | 
 | ||||||
|  |         if (exists) { | ||||||
|  |             deleteBookmarkLocation(bookmarkLocation.toBookmarksLocations()) | ||||||
|  |             NearbyController.updateMarkerLabelListBookmark(bookmarkLocation, false) | ||||||
|  |         } else { | ||||||
|  |             addBookmarkLocation(bookmarkLocation.toBookmarksLocations()) | ||||||
|  |             NearbyController.updateMarkerLabelListBookmark(bookmarkLocation, true) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return !exists | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Fetches all bookmark locations as `Place` objects. | ||||||
|  |      */ | ||||||
|  |     suspend fun getAllBookmarksLocationsPlace(): List<Place> { | ||||||
|  |         return getAllBookmarksLocations().map { it.toPlace() } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,137 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.locations; |  | ||||||
| 
 |  | ||||||
| import android.Manifest.permission; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.view.LayoutInflater; |  | ||||||
| import android.view.View; |  | ||||||
| import android.view.ViewGroup; |  | ||||||
| import androidx.activity.result.ActivityResultCallback; |  | ||||||
| import androidx.activity.result.ActivityResultLauncher; |  | ||||||
| import androidx.activity.result.contract.ActivityResultContracts; |  | ||||||
| import androidx.activity.result.contract.ActivityResultContracts.StartActivityForResult; |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| import androidx.annotation.Nullable; |  | ||||||
| import androidx.recyclerview.widget.LinearLayoutManager; |  | ||||||
| import dagger.android.support.DaggerFragment; |  | ||||||
| import fr.free.nrw.commons.R; |  | ||||||
| import fr.free.nrw.commons.contributions.ContributionController; |  | ||||||
| import fr.free.nrw.commons.databinding.FragmentBookmarksLocationsBinding; |  | ||||||
| import fr.free.nrw.commons.nearby.Place; |  | ||||||
| import fr.free.nrw.commons.nearby.fragments.CommonPlaceClickActions; |  | ||||||
| import fr.free.nrw.commons.nearby.fragments.PlaceAdapter; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Map; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import kotlin.Unit; |  | ||||||
| 
 |  | ||||||
| public class BookmarkLocationsFragment extends DaggerFragment { |  | ||||||
| 
 |  | ||||||
|     public FragmentBookmarksLocationsBinding binding; |  | ||||||
| 
 |  | ||||||
|     @Inject BookmarkLocationsController controller; |  | ||||||
|     @Inject ContributionController contributionController; |  | ||||||
|     @Inject BookmarkLocationsDao bookmarkLocationDao; |  | ||||||
|     @Inject CommonPlaceClickActions commonPlaceClickActions; |  | ||||||
|     private PlaceAdapter adapter; |  | ||||||
| 
 |  | ||||||
|     private final ActivityResultLauncher<Intent> cameraPickLauncherForResult = |  | ||||||
|         registerForActivityResult(new StartActivityForResult(), |  | ||||||
|         result -> { |  | ||||||
|             contributionController.handleActivityResultWithCallback(requireActivity(), callbacks -> { |  | ||||||
|                 contributionController.onPictureReturnedFromCamera(result, requireActivity(), callbacks); |  | ||||||
|             }); |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|       private final ActivityResultLauncher<Intent> galleryPickLauncherForResult = |  | ||||||
|           registerForActivityResult(new StartActivityForResult(), |  | ||||||
|         result -> { |  | ||||||
|               contributionController.handleActivityResultWithCallback(requireActivity(), callbacks -> { |  | ||||||
|                 contributionController.onPictureReturnedFromGallery(result, requireActivity(), callbacks); |  | ||||||
|             }); |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|     private ActivityResultLauncher<String[]> inAppCameraLocationPermissionLauncher = registerForActivityResult(new ActivityResultContracts.RequestMultiplePermissions(), new ActivityResultCallback<Map<String, Boolean>>() { |  | ||||||
|         @Override |  | ||||||
|         public void onActivityResult(Map<String, Boolean> result) { |  | ||||||
|             boolean areAllGranted = true; |  | ||||||
|             for(final boolean b : result.values()) { |  | ||||||
|                 areAllGranted = areAllGranted && b; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (areAllGranted) { |  | ||||||
|                 contributionController.locationPermissionCallback.onLocationPermissionGranted(); |  | ||||||
|             } else { |  | ||||||
|                 if (shouldShowRequestPermissionRationale(permission.ACCESS_FINE_LOCATION)) { |  | ||||||
|                     contributionController.handleShowRationaleFlowCameraLocation(getActivity(), inAppCameraLocationPermissionLauncher, cameraPickLauncherForResult); |  | ||||||
|                 } else { |  | ||||||
|                     contributionController.locationPermissionCallback.onLocationPermissionDenied(getActivity().getString(R.string.in_app_camera_location_permission_denied)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Create an instance of the fragment with the right bundle parameters |  | ||||||
|      * @return an instance of the fragment |  | ||||||
|      */ |  | ||||||
|     public static BookmarkLocationsFragment newInstance() { |  | ||||||
|         return new BookmarkLocationsFragment(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public View onCreateView( |  | ||||||
|             @NonNull LayoutInflater inflater, |  | ||||||
|             ViewGroup container, |  | ||||||
|             Bundle savedInstanceState |  | ||||||
|     ) { |  | ||||||
|         binding = FragmentBookmarksLocationsBinding.inflate(inflater, container, false); |  | ||||||
|         return binding.getRoot(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onViewCreated(View view, @Nullable Bundle savedInstanceState) { |  | ||||||
|         super.onViewCreated(view, savedInstanceState); |  | ||||||
|         binding.loadingImagesProgressBar.setVisibility(View.VISIBLE); |  | ||||||
|         binding.listView.setLayoutManager(new LinearLayoutManager(getContext())); |  | ||||||
|         adapter = new PlaceAdapter(bookmarkLocationDao, |  | ||||||
|             place -> Unit.INSTANCE, |  | ||||||
|             (place, isBookmarked) -> { |  | ||||||
|                 adapter.remove(place); |  | ||||||
|                 return Unit.INSTANCE; |  | ||||||
|             }, |  | ||||||
|             commonPlaceClickActions, |  | ||||||
|             inAppCameraLocationPermissionLauncher, |  | ||||||
|             galleryPickLauncherForResult, |  | ||||||
|             cameraPickLauncherForResult |  | ||||||
|         ); |  | ||||||
|         binding.listView.setAdapter(adapter); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onResume() { |  | ||||||
|         super.onResume(); |  | ||||||
|         initList(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Initialize the recycler view with bookmarked locations |  | ||||||
|      */ |  | ||||||
|     private void initList() { |  | ||||||
|         List<Place> places = controller.loadFavoritesLocations(); |  | ||||||
|         adapter.setItems(places); |  | ||||||
|         binding.loadingImagesProgressBar.setVisibility(View.GONE); |  | ||||||
|         if (places.size() <= 0) { |  | ||||||
|             binding.statusMessage.setText(R.string.bookmark_empty); |  | ||||||
|             binding.statusMessage.setVisibility(View.VISIBLE); |  | ||||||
|         } else { |  | ||||||
|             binding.statusMessage.setVisibility(View.GONE); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public void onDestroy() { |  | ||||||
|         super.onDestroy(); |  | ||||||
|         binding = null; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,151 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.locations | ||||||
|  | 
 | ||||||
|  | import android.Manifest.permission | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import androidx.activity.result.ActivityResultLauncher | ||||||
|  | import androidx.activity.result.contract.ActivityResultContracts.RequestMultiplePermissions | ||||||
|  | import androidx.activity.result.contract.ActivityResultContracts.StartActivityForResult | ||||||
|  | import androidx.lifecycle.lifecycleScope | ||||||
|  | import androidx.recyclerview.widget.LinearLayoutManager | ||||||
|  | import dagger.android.support.DaggerFragment | ||||||
|  | import fr.free.nrw.commons.R | ||||||
|  | import fr.free.nrw.commons.contributions.ContributionController | ||||||
|  | import fr.free.nrw.commons.databinding.FragmentBookmarksLocationsBinding | ||||||
|  | import fr.free.nrw.commons.nearby.Place | ||||||
|  | import fr.free.nrw.commons.nearby.fragments.CommonPlaceClickActions | ||||||
|  | import fr.free.nrw.commons.nearby.fragments.PlaceAdapter | ||||||
|  | import kotlinx.coroutines.launch | ||||||
|  | import javax.inject.Inject | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class BookmarkLocationsFragment : DaggerFragment() { | ||||||
|  | 
 | ||||||
|  |     private var binding: FragmentBookmarksLocationsBinding? = null | ||||||
|  | 
 | ||||||
|  |     @Inject lateinit var controller: BookmarkLocationsController | ||||||
|  |     @Inject lateinit var contributionController: ContributionController | ||||||
|  |     @Inject lateinit var bookmarkLocationDao: BookmarkLocationsDao | ||||||
|  |     @Inject lateinit var commonPlaceClickActions: CommonPlaceClickActions | ||||||
|  | 
 | ||||||
|  |     private lateinit var inAppCameraLocationPermissionLauncher: | ||||||
|  |             ActivityResultLauncher<Array<String>> | ||||||
|  |     private lateinit var adapter: PlaceAdapter | ||||||
|  | 
 | ||||||
|  |     private val cameraPickLauncherForResult = | ||||||
|  |         registerForActivityResult(StartActivityForResult()) { result -> | ||||||
|  |             contributionController.handleActivityResultWithCallback( | ||||||
|  |                 requireActivity() | ||||||
|  |             ) { callbacks -> | ||||||
|  |                 contributionController.onPictureReturnedFromCamera( | ||||||
|  |                     result, | ||||||
|  |                     requireActivity(), | ||||||
|  |                     callbacks | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     private val galleryPickLauncherForResult = | ||||||
|  |         registerForActivityResult(StartActivityForResult()) { result -> | ||||||
|  |             contributionController.handleActivityResultWithCallback( | ||||||
|  |                 requireActivity() | ||||||
|  |             ) { callbacks -> | ||||||
|  |                 contributionController.onPictureReturnedFromGallery( | ||||||
|  |                     result, | ||||||
|  |                     requireActivity(), | ||||||
|  |                     callbacks | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         fun newInstance(): BookmarkLocationsFragment { | ||||||
|  |             return BookmarkLocationsFragment() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View? { | ||||||
|  |         binding = FragmentBookmarksLocationsBinding.inflate(inflater, container, false) | ||||||
|  |         return binding?.root | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onViewCreated(view: View, savedInstanceState: Bundle?) { | ||||||
|  |         super.onViewCreated(view, savedInstanceState) | ||||||
|  |         binding?.loadingImagesProgressBar?.visibility = View.VISIBLE | ||||||
|  |         binding?.listView?.layoutManager = LinearLayoutManager(context) | ||||||
|  | 
 | ||||||
|  |         inAppCameraLocationPermissionLauncher = | ||||||
|  |             registerForActivityResult(RequestMultiplePermissions()) { result -> | ||||||
|  |                 val areAllGranted = result.values.all { it } | ||||||
|  | 
 | ||||||
|  |                 if (areAllGranted) { | ||||||
|  |                     contributionController.locationPermissionCallback?.onLocationPermissionGranted() | ||||||
|  |                 } else { | ||||||
|  |                     if (shouldShowRequestPermissionRationale(permission.ACCESS_FINE_LOCATION)) { | ||||||
|  |                         contributionController.handleShowRationaleFlowCameraLocation( | ||||||
|  |                             requireActivity(), | ||||||
|  |                             inAppCameraLocationPermissionLauncher, | ||||||
|  |                             cameraPickLauncherForResult | ||||||
|  |                         ) | ||||||
|  |                     } else { | ||||||
|  |                         contributionController.locationPermissionCallback | ||||||
|  |                             ?.onLocationPermissionDenied( | ||||||
|  |                                 getString(R.string.in_app_camera_location_permission_denied) | ||||||
|  |                             ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         adapter = PlaceAdapter( | ||||||
|  |             bookmarkLocationDao, | ||||||
|  |             lifecycleScope, | ||||||
|  |             { }, | ||||||
|  |             { place, _ -> | ||||||
|  |                 adapter.remove(place) | ||||||
|  |             }, | ||||||
|  |             commonPlaceClickActions, | ||||||
|  |             inAppCameraLocationPermissionLauncher, | ||||||
|  |             galleryPickLauncherForResult, | ||||||
|  |             cameraPickLauncherForResult | ||||||
|  |         ) | ||||||
|  |         binding?.listView?.adapter = adapter | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onResume() { | ||||||
|  |         super.onResume() | ||||||
|  |         initList() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun initList() { | ||||||
|  |         var places: List<Place> | ||||||
|  |         if(view != null) { | ||||||
|  |             viewLifecycleOwner.lifecycleScope.launch { | ||||||
|  |                 places = controller.loadFavoritesLocations() | ||||||
|  |                 updateUIList(places) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private fun updateUIList(places: List<Place>) { | ||||||
|  |         adapter.items = places | ||||||
|  |         binding?.loadingImagesProgressBar?.visibility = View.GONE | ||||||
|  |         if (places.isEmpty()) { | ||||||
|  |             binding?.statusMessage?.text = getString(R.string.bookmark_empty) | ||||||
|  |             binding?.statusMessage?.visibility = View.VISIBLE | ||||||
|  |         } else { | ||||||
|  |             binding?.statusMessage?.visibility = View.GONE | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun onDestroy() { | ||||||
|  |         super.onDestroy() | ||||||
|  |         // Make sure to null out the binding to avoid memory leaks | ||||||
|  |         binding = null | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.locations | ||||||
|  | 
 | ||||||
|  | import androidx.lifecycle.ViewModel | ||||||
|  | import fr.free.nrw.commons.nearby.Place | ||||||
|  | import kotlinx.coroutines.flow.Flow | ||||||
|  | 
 | ||||||
|  | class BookmarkLocationsViewModel( | ||||||
|  |     private val bookmarkLocationsDao: BookmarkLocationsDao | ||||||
|  | ): ViewModel() { | ||||||
|  | 
 | ||||||
|  | //    fun getAllBookmarkLocations(): List<Place> { | ||||||
|  | //        return bookmarkLocationsDao.getAllBookmarksLocationsPlace() | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -0,0 +1,72 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.locations | ||||||
|  | 
 | ||||||
|  | import androidx.room.ColumnInfo | ||||||
|  | import androidx.room.Entity | ||||||
|  | import androidx.room.PrimaryKey | ||||||
|  | import fr.free.nrw.commons.location.LatLng | ||||||
|  | import fr.free.nrw.commons.nearby.Label | ||||||
|  | import fr.free.nrw.commons.nearby.Place | ||||||
|  | import fr.free.nrw.commons.nearby.Sitelinks | ||||||
|  | 
 | ||||||
|  | @Entity(tableName = "bookmarks_locations") | ||||||
|  | data class BookmarksLocations( | ||||||
|  |     @PrimaryKey @ColumnInfo(name = "location_name") val locationName: String, | ||||||
|  |     @ColumnInfo(name = "location_language") val locationLanguage: String, | ||||||
|  |     @ColumnInfo(name = "location_description") val locationDescription: String, | ||||||
|  |     @ColumnInfo(name = "location_lat") val locationLat: Double, | ||||||
|  |     @ColumnInfo(name = "location_long") val locationLong: Double, | ||||||
|  |     @ColumnInfo(name = "location_category") val locationCategory: String, | ||||||
|  |     @ColumnInfo(name = "location_label_text") val locationLabelText: String, | ||||||
|  |     @ColumnInfo(name = "location_label_icon") val locationLabelIcon: Int?, | ||||||
|  |     @ColumnInfo(name = "location_image_url") val locationImageUrl: String, | ||||||
|  |     @ColumnInfo(name = "location_wikipedia_link") val locationWikipediaLink: String, | ||||||
|  |     @ColumnInfo(name = "location_wikidata_link") val locationWikidataLink: String, | ||||||
|  |     @ColumnInfo(name = "location_commons_link") val locationCommonsLink: String, | ||||||
|  |     @ColumnInfo(name = "location_pic") val locationPic: String, | ||||||
|  |     @ColumnInfo(name = "location_exists") val locationExists: Boolean | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | fun BookmarksLocations.toPlace(): Place { | ||||||
|  |     val location = LatLng( | ||||||
|  |         locationLat, | ||||||
|  |         locationLong, | ||||||
|  |         1F | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     val builder = Sitelinks.Builder().apply { | ||||||
|  |         setWikipediaLink(locationWikipediaLink) | ||||||
|  |         setWikidataLink(locationWikidataLink) | ||||||
|  |         setCommonsLink(locationCommonsLink) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return Place( | ||||||
|  |         locationLanguage, | ||||||
|  |         locationName, | ||||||
|  |         Label.fromText(locationLabelText), | ||||||
|  |         locationDescription, | ||||||
|  |         location, | ||||||
|  |         locationCategory, | ||||||
|  |         builder.build(), | ||||||
|  |         locationPic, | ||||||
|  |         locationExists | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fun Place.toBookmarksLocations(): BookmarksLocations { | ||||||
|  |     return BookmarksLocations( | ||||||
|  |         locationName = name, | ||||||
|  |         locationLanguage = language, | ||||||
|  |         locationDescription = longDescription, | ||||||
|  |         locationCategory = category, | ||||||
|  |         locationLat = location.latitude, | ||||||
|  |         locationLong = location.longitude, | ||||||
|  |         locationLabelText = label?.text ?: "", | ||||||
|  |         locationLabelIcon = label?.icon, | ||||||
|  |         locationImageUrl = pic, | ||||||
|  |         locationWikipediaLink = siteLinks.wikipediaLink.toString(), | ||||||
|  |         locationWikidataLink = siteLinks.wikidataLink.toString(), | ||||||
|  |         locationCommonsLink = siteLinks.commonsLink.toString(), | ||||||
|  |         locationPic = pic, | ||||||
|  |         locationExists = exists | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  | @ -8,7 +8,7 @@ class Bookmark( | ||||||
|     /** |     /** | ||||||
|      * Gets or Sets the content URI - marking this bookmark as already saved in the database |      * Gets or Sets the content URI - marking this bookmark as already saved in the database | ||||||
|      * @return content URI |      * @return content URI | ||||||
|      * @param contentUri the content URI |      * contentUri the content URI | ||||||
|      */ |      */ | ||||||
|     var contentUri: Uri?, |     var contentUri: Uri?, | ||||||
| ) { | ) { | ||||||
|  |  | ||||||
|  | @ -1,120 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.pictures; |  | ||||||
| 
 |  | ||||||
| import android.content.ContentValues; |  | ||||||
| import android.database.Cursor; |  | ||||||
| import android.database.sqlite.SQLiteDatabase; |  | ||||||
| import android.database.sqlite.SQLiteQueryBuilder; |  | ||||||
| // We can get uri using java.Net.Uri, but andoid implimentation is faster (but it's forgiving with handling exceptions though) |  | ||||||
| import android.net.Uri; |  | ||||||
| import android.text.TextUtils; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| 
 |  | ||||||
| import javax.inject.Inject; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.BuildConfig; |  | ||||||
| import fr.free.nrw.commons.data.DBOpenHelper; |  | ||||||
| import fr.free.nrw.commons.di.CommonsDaggerContentProvider; |  | ||||||
| import timber.log.Timber; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesDao.Table.COLUMN_MEDIA_NAME; |  | ||||||
| import static fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesDao.Table.TABLE_NAME; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * Handles private storage for Bookmark pictures |  | ||||||
|  */ |  | ||||||
| public class BookmarkPicturesContentProvider extends CommonsDaggerContentProvider { |  | ||||||
| 
 |  | ||||||
|     private static final String BASE_PATH = "bookmarks"; |  | ||||||
|     public static final Uri BASE_URI = Uri.parse("content://" + BuildConfig.BOOKMARK_AUTHORITY + "/" + BASE_PATH); |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Append bookmark pictures name to the base uri  |  | ||||||
|      */ |  | ||||||
|     public static Uri uriForName(String name) { |  | ||||||
|         return Uri.parse(BASE_URI.toString() + "/" + name); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     DBOpenHelper dbOpenHelper; |  | ||||||
| 
 |  | ||||||
|     @Override |  | ||||||
|     public String getType(@NonNull Uri uri) { |  | ||||||
|         return null; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Queries the SQLite database for the bookmark pictures |  | ||||||
|      * @param uri : contains the uri for bookmark pictures |  | ||||||
|      * @param projection |  | ||||||
|      * @param selection : handles Where |  | ||||||
|      * @param selectionArgs : the condition of Where clause |  | ||||||
|      * @param sortOrder : ascending or descending |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public Cursor query(@NonNull Uri uri, String[] projection, String selection, |  | ||||||
|                         String[] selectionArgs, String sortOrder) { |  | ||||||
|         SQLiteQueryBuilder queryBuilder = new SQLiteQueryBuilder(); |  | ||||||
|         queryBuilder.setTables(TABLE_NAME); |  | ||||||
| 
 |  | ||||||
|         SQLiteDatabase db = dbOpenHelper.getReadableDatabase(); |  | ||||||
|         Cursor cursor = queryBuilder.query(db, projection, selection, selectionArgs, null, null, sortOrder); |  | ||||||
|         cursor.setNotificationUri(getContext().getContentResolver(), uri); |  | ||||||
| 
 |  | ||||||
|         return cursor; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the update query of local SQLite Database  |  | ||||||
|      * @param uri : contains the uri for bookmark pictures |  | ||||||
|      * @param contentValues : new values to be entered to db |  | ||||||
|      * @param selection : handles Where |  | ||||||
|      * @param selectionArgs : the condition of Where clause |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public int update(@NonNull Uri uri, ContentValues contentValues, String selection, |  | ||||||
|                       String[] selectionArgs) { |  | ||||||
|         SQLiteDatabase sqlDB = dbOpenHelper.getWritableDatabase(); |  | ||||||
|         int rowsUpdated; |  | ||||||
|         if (TextUtils.isEmpty(selection)) { |  | ||||||
|             int id = Integer.valueOf(uri.getLastPathSegment()); |  | ||||||
|             rowsUpdated = sqlDB.update(TABLE_NAME, |  | ||||||
|                     contentValues, |  | ||||||
|                     COLUMN_MEDIA_NAME + " = ?", |  | ||||||
|                     new String[]{String.valueOf(id)}); |  | ||||||
|         } else { |  | ||||||
|             throw new IllegalArgumentException( |  | ||||||
|                     "Parameter `selection` should be empty when updating an ID"); |  | ||||||
|         } |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return rowsUpdated; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Handles the insertion of new bookmark pictures record to local SQLite Database |  | ||||||
|      */ |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public Uri insert(@NonNull Uri uri, ContentValues contentValues) { |  | ||||||
|         SQLiteDatabase sqlDB = dbOpenHelper.getWritableDatabase(); |  | ||||||
|         long id = sqlDB.insert(BookmarkPicturesDao.Table.TABLE_NAME, null, contentValues); |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return Uri.parse(BASE_URI + "/" + id); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressWarnings("ConstantConditions") |  | ||||||
|     @Override |  | ||||||
|     public int delete(@NonNull Uri uri, String s, String[] strings) { |  | ||||||
|         int rows; |  | ||||||
|         SQLiteDatabase db = dbOpenHelper.getReadableDatabase(); |  | ||||||
|         Timber.d("Deleting bookmark name %s", uri.getLastPathSegment()); |  | ||||||
|         rows = db.delete(TABLE_NAME, |  | ||||||
|                 "media_name = ?", |  | ||||||
|                 new String[]{uri.getLastPathSegment()} |  | ||||||
|         ); |  | ||||||
|         getContext().getContentResolver().notifyChange(uri, null); |  | ||||||
|         return rows; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,100 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.pictures | ||||||
|  | 
 | ||||||
|  | import android.content.ContentValues | ||||||
|  | import android.database.Cursor | ||||||
|  | import android.database.sqlite.SQLiteQueryBuilder | ||||||
|  | import android.net.Uri | ||||||
|  | import fr.free.nrw.commons.BuildConfig | ||||||
|  | import fr.free.nrw.commons.di.CommonsDaggerContentProvider | ||||||
|  | import androidx.core.net.toUri | ||||||
|  | import fr.free.nrw.commons.bookmarks.pictures.BookmarksTable.COLUMN_MEDIA_NAME | ||||||
|  | import fr.free.nrw.commons.bookmarks.pictures.BookmarksTable.TABLE_NAME | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handles private storage for Bookmark pictures | ||||||
|  |  */ | ||||||
|  | class BookmarkPicturesContentProvider : CommonsDaggerContentProvider() { | ||||||
|  |     override fun getType(uri: Uri): String? = null | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Queries the SQLite database for the bookmark pictures | ||||||
|  |      * @param uri : contains the uri for bookmark pictures | ||||||
|  |      * @param projection | ||||||
|  |      * @param selection : handles Where | ||||||
|  |      * @param selectionArgs : the condition of Where clause | ||||||
|  |      * @param sortOrder : ascending or descending | ||||||
|  |      */ | ||||||
|  |     override fun query( | ||||||
|  |         uri: Uri, projection: Array<String>?, selection: String?, | ||||||
|  |         selectionArgs: Array<String>?, sortOrder: String? | ||||||
|  |     ): Cursor { | ||||||
|  |         val queryBuilder = SQLiteQueryBuilder().apply { | ||||||
|  |             tables = TABLE_NAME | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         val cursor = queryBuilder.query( | ||||||
|  |             requireDb(), projection, selection, | ||||||
|  |             selectionArgs, null, null, sortOrder | ||||||
|  |         ) | ||||||
|  |         cursor.setNotificationUri(context?.contentResolver, uri) | ||||||
|  | 
 | ||||||
|  |         return cursor | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the update query of local SQLite Database | ||||||
|  |      * @param uri : contains the uri for bookmark pictures | ||||||
|  |      * @param contentValues : new values to be entered to db | ||||||
|  |      * @param selection : handles Where | ||||||
|  |      * @param selectionArgs : the condition of Where clause | ||||||
|  |      */ | ||||||
|  |     override fun update( | ||||||
|  |         uri: Uri, contentValues: ContentValues?, selection: String?, | ||||||
|  |         selectionArgs: Array<String>? | ||||||
|  |     ): Int { | ||||||
|  |         val rowsUpdated: Int | ||||||
|  |         if (selection.isNullOrEmpty()) { | ||||||
|  |             val id = uri.lastPathSegment!!.toInt() | ||||||
|  |             rowsUpdated = requireDb().update( | ||||||
|  |                 TABLE_NAME, | ||||||
|  |                 contentValues, | ||||||
|  |                 "$COLUMN_MEDIA_NAME = ?", | ||||||
|  |                 arrayOf(id.toString()) | ||||||
|  |             ) | ||||||
|  |         } else { | ||||||
|  |             throw IllegalArgumentException( | ||||||
|  |                 "Parameter `selection` should be empty when updating an ID" | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |         context?.contentResolver?.notifyChange(uri, null) | ||||||
|  |         return rowsUpdated | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Handles the insertion of new bookmark pictures record to local SQLite Database | ||||||
|  |      */ | ||||||
|  |     override fun insert(uri: Uri, contentValues: ContentValues?): Uri { | ||||||
|  |         val id = requireDb().insert(TABLE_NAME, null, contentValues) | ||||||
|  |         context?.contentResolver?.notifyChange(uri, null) | ||||||
|  |         return "$BASE_URI/$id".toUri() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     override fun delete(uri: Uri, s: String?, strings: Array<String>?): Int { | ||||||
|  |         val rows: Int = requireDb().delete( | ||||||
|  |             TABLE_NAME, | ||||||
|  |             "media_name = ?", | ||||||
|  |             arrayOf(uri.lastPathSegment) | ||||||
|  |         ) | ||||||
|  |         context?.contentResolver?.notifyChange(uri, null) | ||||||
|  |         return rows | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     companion object { | ||||||
|  |         private const val BASE_PATH = "bookmarks" | ||||||
|  |         @JvmField | ||||||
|  |         val BASE_URI: Uri = "content://${BuildConfig.BOOKMARK_AUTHORITY}/$BASE_PATH".toUri() | ||||||
|  | 
 | ||||||
|  |         @JvmStatic | ||||||
|  |         fun uriForName(name: String): Uri = "$BASE_URI/$name".toUri() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,63 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.pictures; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.Media; |  | ||||||
| import fr.free.nrw.commons.bookmarks.models.Bookmark; |  | ||||||
| import fr.free.nrw.commons.media.MediaClient; |  | ||||||
| import io.reactivex.Observable; |  | ||||||
| import io.reactivex.ObservableSource; |  | ||||||
| import io.reactivex.Single; |  | ||||||
| import io.reactivex.functions.Function; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Singleton; |  | ||||||
| 
 |  | ||||||
| @Singleton |  | ||||||
| public class BookmarkPicturesController { |  | ||||||
| 
 |  | ||||||
|     private final MediaClient mediaClient; |  | ||||||
|     private final BookmarkPicturesDao bookmarkDao; |  | ||||||
| 
 |  | ||||||
|     private List<Bookmark> currentBookmarks; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public BookmarkPicturesController(MediaClient mediaClient, BookmarkPicturesDao bookmarkDao) { |  | ||||||
|         this.mediaClient = mediaClient; |  | ||||||
|         this.bookmarkDao = bookmarkDao; |  | ||||||
|         currentBookmarks = new ArrayList<>(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Loads the Media objects from the raw data stored in DB and the API. |  | ||||||
|      * @return a list of bookmarked Media object |  | ||||||
|      */ |  | ||||||
|     Single<List<Media>> loadBookmarkedPictures() { |  | ||||||
|         List<Bookmark> bookmarks = bookmarkDao.getAllBookmarks(); |  | ||||||
|         currentBookmarks = bookmarks; |  | ||||||
|         return Observable.fromIterable(bookmarks) |  | ||||||
|                 .flatMap((Function<Bookmark, ObservableSource<Media>>) this::getMediaFromBookmark) |  | ||||||
|                 .toList(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private Observable<Media> getMediaFromBookmark(Bookmark bookmark) { |  | ||||||
|         return mediaClient.getMedia(bookmark.getMediaName()) |  | ||||||
|                 .toObservable() |  | ||||||
|             .onErrorResumeNext(Observable.empty()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Loads the Media objects from the raw data stored in DB and the API. |  | ||||||
|      * @return a list of bookmarked Media object |  | ||||||
|      */ |  | ||||||
|     boolean needRefreshBookmarkedPictures() { |  | ||||||
|         List<Bookmark> bookmarks = bookmarkDao.getAllBookmarks(); |  | ||||||
|         return bookmarks.size() != currentBookmarks.size(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Cancels the requests to the API and the DB |  | ||||||
|      */ |  | ||||||
|     void stop() { |  | ||||||
|         //noop |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,38 @@ | ||||||
|  | package fr.free.nrw.commons.bookmarks.pictures | ||||||
|  | 
 | ||||||
|  | import fr.free.nrw.commons.Media | ||||||
|  | import fr.free.nrw.commons.bookmarks.models.Bookmark | ||||||
|  | import fr.free.nrw.commons.media.MediaClient | ||||||
|  | import io.reactivex.Observable | ||||||
|  | import io.reactivex.Single | ||||||
|  | import javax.inject.Inject | ||||||
|  | import javax.inject.Singleton | ||||||
|  | 
 | ||||||
|  | @Singleton | ||||||
|  | class BookmarkPicturesController @Inject constructor( | ||||||
|  |     private val mediaClient: MediaClient, | ||||||
|  |     private val bookmarkDao: BookmarkPicturesDao | ||||||
|  | ) { | ||||||
|  |     private var currentBookmarks: List<Bookmark> = listOf() | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Loads the Media objects from the raw data stored in DB and the API. | ||||||
|  |      * @return a list of bookmarked Media object | ||||||
|  |      */ | ||||||
|  |     fun loadBookmarkedPictures(): Single<List<Media>> { | ||||||
|  |         val bookmarks = bookmarkDao.getAllBookmarks() | ||||||
|  |         currentBookmarks = bookmarks | ||||||
|  |         return Observable.fromIterable(bookmarks).flatMap { | ||||||
|  |             mediaClient.getMedia(it.mediaName) | ||||||
|  |                 .toObservable() | ||||||
|  |                 .onErrorResumeNext(Observable.empty()) | ||||||
|  |         }.toList() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun needRefreshBookmarkedPictures(): Boolean { | ||||||
|  |         val bookmarks = bookmarkDao.getAllBookmarks() | ||||||
|  |         return bookmarks.size != currentBookmarks.size | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fun stop() = Unit | ||||||
|  | } | ||||||
|  | @ -1,227 +0,0 @@ | ||||||
| package fr.free.nrw.commons.bookmarks.pictures; |  | ||||||
| 
 |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.content.ContentProviderClient; |  | ||||||
| import android.content.ContentValues; |  | ||||||
| import android.database.Cursor; |  | ||||||
| import android.database.sqlite.SQLiteDatabase; |  | ||||||
| import android.os.RemoteException; |  | ||||||
| 
 |  | ||||||
| import androidx.annotation.NonNull; |  | ||||||
| 
 |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.List; |  | ||||||
| 
 |  | ||||||
| import javax.inject.Inject; |  | ||||||
| import javax.inject.Named; |  | ||||||
| import javax.inject.Provider; |  | ||||||
| import javax.inject.Singleton; |  | ||||||
| 
 |  | ||||||
| import fr.free.nrw.commons.bookmarks.models.Bookmark; |  | ||||||
| 
 |  | ||||||
| import static fr.free.nrw.commons.bookmarks.pictures.BookmarkPicturesContentProvider.BASE_URI; |  | ||||||
| 
 |  | ||||||
| @Singleton |  | ||||||
| public class BookmarkPicturesDao { |  | ||||||
| 
 |  | ||||||
|     private final Provider<ContentProviderClient> clientProvider; |  | ||||||
| 
 |  | ||||||
|     @Inject |  | ||||||
|     public BookmarkPicturesDao(@Named("bookmarks") Provider<ContentProviderClient> clientProvider) { |  | ||||||
|         this.clientProvider = clientProvider; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Find all persisted pictures bookmarks on database |  | ||||||
|      * |  | ||||||
|      * @return list of bookmarks |  | ||||||
|      */ |  | ||||||
|     @NonNull |  | ||||||
|     public List<Bookmark> getAllBookmarks() { |  | ||||||
|         List<Bookmark> items = new ArrayList<>(); |  | ||||||
|         Cursor cursor = null; |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             cursor = db.query( |  | ||||||
|                     BookmarkPicturesContentProvider.BASE_URI, |  | ||||||
|                     Table.ALL_FIELDS, |  | ||||||
|                     null, |  | ||||||
|                     new String[]{}, |  | ||||||
|                     null); |  | ||||||
|             while (cursor != null && cursor.moveToNext()) { |  | ||||||
|                 items.add(fromCursor(cursor)); |  | ||||||
|             } |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             if (cursor != null) { |  | ||||||
|                 cursor.close(); |  | ||||||
|             } |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|         return items; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Look for a bookmark in database and in order to insert or delete it |  | ||||||
|      * |  | ||||||
|      * @param bookmark : Bookmark object |  | ||||||
|      * @return boolean : is bookmark now fav ? |  | ||||||
|      */ |  | ||||||
|     public boolean updateBookmark(Bookmark bookmark) { |  | ||||||
|         boolean bookmarkExists = findBookmark(bookmark); |  | ||||||
|         if (bookmarkExists) { |  | ||||||
|             deleteBookmark(bookmark); |  | ||||||
|         } else { |  | ||||||
|             addBookmark(bookmark); |  | ||||||
|         } |  | ||||||
|         return !bookmarkExists; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Add a Bookmark to database |  | ||||||
|      * |  | ||||||
|      * @param bookmark : Bookmark to add |  | ||||||
|      */ |  | ||||||
|     private void addBookmark(Bookmark bookmark) { |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             db.insert(BASE_URI, toContentValues(bookmark)); |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Delete a bookmark from database |  | ||||||
|      * |  | ||||||
|      * @param bookmark : Bookmark to delete |  | ||||||
|      */ |  | ||||||
|     private void deleteBookmark(Bookmark bookmark) { |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             if (bookmark.getContentUri() == null) { |  | ||||||
|                 throw new RuntimeException("tried to delete item with no content URI"); |  | ||||||
|             } else { |  | ||||||
|                 db.delete(bookmark.getContentUri(), null, null); |  | ||||||
|             } |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * Find a bookmark from database based on its name |  | ||||||
|      * |  | ||||||
|      * @param bookmark : Bookmark to find |  | ||||||
|      * @return boolean : is bookmark in database ? |  | ||||||
|      */ |  | ||||||
|     public boolean findBookmark(Bookmark bookmark) { |  | ||||||
|         if (bookmark == null) {//Avoiding NPE's |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         Cursor cursor = null; |  | ||||||
|         ContentProviderClient db = clientProvider.get(); |  | ||||||
|         try { |  | ||||||
|             cursor = db.query( |  | ||||||
|                     BookmarkPicturesContentProvider.BASE_URI, |  | ||||||
|                     Table.ALL_FIELDS, |  | ||||||
|                     Table.COLUMN_MEDIA_NAME + "=?", |  | ||||||
|                     new String[]{bookmark.getMediaName()}, |  | ||||||
|                     null); |  | ||||||
|             if (cursor != null && cursor.moveToFirst()) { |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             // This feels lazy, but to hell with checked exceptions. :) |  | ||||||
|             throw new RuntimeException(e); |  | ||||||
|         } finally { |  | ||||||
|             if (cursor != null) { |  | ||||||
|                 cursor.close(); |  | ||||||
|             } |  | ||||||
|             db.release(); |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @SuppressLint("Range") |  | ||||||
|     @NonNull |  | ||||||
|     Bookmark fromCursor(Cursor cursor) { |  | ||||||
|         String fileName = cursor.getString(cursor.getColumnIndex(Table.COLUMN_MEDIA_NAME)); |  | ||||||
|         return new Bookmark( |  | ||||||
|                 fileName, |  | ||||||
|                 cursor.getString(cursor.getColumnIndex(Table.COLUMN_CREATOR)), |  | ||||||
|                 BookmarkPicturesContentProvider.uriForName(fileName) |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private ContentValues toContentValues(Bookmark bookmark) { |  | ||||||
|         ContentValues cv = new ContentValues(); |  | ||||||
|         cv.put(BookmarkPicturesDao.Table.COLUMN_MEDIA_NAME, bookmark.getMediaName()); |  | ||||||
|         cv.put(BookmarkPicturesDao.Table.COLUMN_CREATOR, bookmark.getMediaCreator()); |  | ||||||
|         return cv; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     public static class Table { |  | ||||||
|         public static final String TABLE_NAME = "bookmarks"; |  | ||||||
| 
 |  | ||||||
|         public static final String COLUMN_MEDIA_NAME = "media_name"; |  | ||||||
|         public static final String COLUMN_CREATOR = "media_creator"; |  | ||||||
| 
 |  | ||||||
|         // NOTE! KEEP IN SAME ORDER AS THEY ARE DEFINED UP THERE. HELPS HARD CODE COLUMN INDICES. |  | ||||||
|         public static final String[] ALL_FIELDS = { |  | ||||||
|                 COLUMN_MEDIA_NAME, |  | ||||||
|                 COLUMN_CREATOR |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         public static final String DROP_TABLE_STATEMENT = "DROP TABLE IF EXISTS " + TABLE_NAME; |  | ||||||
| 
 |  | ||||||
|         public static final String CREATE_TABLE_STATEMENT = "CREATE TABLE " + TABLE_NAME + " (" |  | ||||||
|                 + COLUMN_MEDIA_NAME + " STRING PRIMARY KEY," |  | ||||||
|                 + COLUMN_CREATOR + " STRING" |  | ||||||
|                 + ");"; |  | ||||||
| 
 |  | ||||||
|         public static void onCreate(SQLiteDatabase db) { |  | ||||||
|             db.execSQL(CREATE_TABLE_STATEMENT); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void onDelete(SQLiteDatabase db) { |  | ||||||
|             db.execSQL(DROP_TABLE_STATEMENT); |  | ||||||
|             onCreate(db); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void onUpdate(SQLiteDatabase db, int from, int to) { |  | ||||||
|             if (from == to) { |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             if (from < 7) { |  | ||||||
|                 // doesn't exist yet |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (from == 7) { |  | ||||||
|                 // table added in version 8 |  | ||||||
|                 onCreate(db); |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (from == 8) { |  | ||||||
|                 from++; |  | ||||||
|                 onUpdate(db, from, to); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue