Compare commits
	
		
			357 Commits
		
	
	
		
			v122092561
			...
			b3d7a7761d
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b3d7a7761d | |||
| c25ad7621e | |||
| 63da3b9fe7 | |||
| 1d99eeb633 | |||
| 162a350a8f | |||
| 27c1bba146 | |||
| b7f3a9877a | |||
| 47f78754dc | |||
| 1bdfb143ac | |||
| d81ced3964 | |||
| fbafece1fa | |||
| cbed8f07cb | |||
| f54fcc3ba1 | |||
|  | aad93ef722 | ||
| 9e83af0302 | |||
|  | 24b86e66b4 | ||
| 641c444061 | |||
| 0902c61544 | |||
|  | 6790152a0b | ||
|  | 46d1ba418e | ||
| 436373d0ad | |||
|  | 5b9b51c02d | ||
| b81abe384a | |||
|  | 851f862dbe | ||
|  | 8d7e302af8 | ||
| 236e1cca90 | |||
| 3a33cb4510 | |||
| 0bf9ca9a49 | |||
|  | 61e0087894 | ||
|  | 1ec05d9913 | ||
|  | 859bd91bbb | ||
|  | 204b736c53 | ||
|  | f24609c143 | ||
|  | b94d7dc537 | ||
|  | 41910cc4cd | ||
|  | db166ca9d4 | ||
|  | db0d5a4a85 | ||
|  | 3bc0d7cf95 | ||
|  | 8f464d95fd | ||
|  | 5ccd6a3368 | ||
|  | cdbded246e | ||
|  | 750c7758bd | ||
|  | 22f8b14ecd | ||
|  | 6e27d6d4e6 | ||
|  | 14ff4dbd05 | ||
|  | 390c2d0cf3 | ||
|  | e58914ef58 | ||
|  | a03f08fca1 | ||
|  | 8e9b87f00c | ||
|  | f765224a86 | ||
|  | 14d2219eb8 | ||
|  | 137580ccf9 | ||
|  | f101d22f54 | ||
|  | 68aedb7641 | ||
|  | 754d526b49 | ||
|  | c458871569 | ||
| 056825aa0c | |||
| 16b19fc5ce | |||
| 4ad4a23ed8 | |||
| d8c215eacc | |||
| 2b446ab22b | |||
| a029d8a7dc | |||
| 4482234e1a | |||
| b5de30f561 | |||
| 70ad5f322c | |||
| d167092c83 | |||
| c4f4bafe85 | |||
| ed06b22a77 | |||
|  | 172362b533 | ||
|  | ad72cb6f56 | ||
|  | 9057ee0052 | ||
|  | 50d0b44315 | ||
|  | 21b08ed384 | ||
|  | 993c4d2ee9 | ||
|  | 57a9d51027 | ||
|  | 673f0edb8b | ||
|  | 7f96798f13 | ||
|  | 6e5704a45b | ||
|  | 495591159f | ||
|  | 718fe7c5ee | ||
|  | ecd23213f9 | ||
|  | e6baed8cb4 | ||
|  | c87abec0b9 | ||
|  | 0aba41d8bf | ||
|  | 2a2d1047b4 | ||
|  | 66ef1ccf32 | ||
|  | 677ede5bc7 | ||
|  | 996a7ed22c | ||
|  | 85208c4e5a | ||
|  | 5cfec50cba | ||
| 76ad71e1dc | |||
| 0277fb507c | |||
| 8d7d3174aa | |||
|  | 00eb3333fe | ||
|  | 629ca01d99 | ||
|  | c2d8681ce8 | ||
|  | 08f79cb148 | ||
| e21906e70d | |||
|  | 9d2cc32bc9 | ||
|  | d9d057c8dc | ||
|  | 1f3fa0c4a6 | ||
|  | dea3def385 | ||
|  | f72ef2f5d4 | ||
|  | f28cb759df | ||
|  | b9d69c3e64 | ||
|  | c2a1c9eaac | ||
|  | bf37209a15 | ||
|  | 2c558fe6fd | ||
|  | ad88011454 | ||
|  | 559c17bc1d | ||
|  | ab9c46f0eb | ||
|  | aa799d2ca8 | ||
|  | 177c978474 | ||
|  | 39b9991413 | ||
|  | b303f110f1 | ||
|  | f851941a6a | ||
|  | a313552976 | ||
|  | 6ac97ed3fe | ||
|  | d583b937b7 | ||
|  | 15b9a2d935 | ||
|  | 5a8ce15961 | ||
| e1c64cef46 | |||
| ee064f3cb4 | |||
| 3e46e2ff29 | |||
| f28e702549 | |||
|  | fc31a4399c | ||
|  | 9b23053b66 | ||
| 389a04d250 | |||
|  | 40e1d1478b | ||
|  | 2154ff3c33 | ||
| 2245565f95 | |||
| 014858f06b | |||
| 3f1f86a78e | |||
| a549169a7c | |||
| be7cae365a | |||
| cef3b2e593 | |||
| ae927ebc57 | |||
|  | 90532cf501 | ||
|  | ab0678d61e | ||
|  | a1b7d22d26 | ||
|  | 29eae4b1f6 | ||
|  | f5bbc63481 | ||
| ddc72d85b0 | |||
| 68bbf5b2d3 | |||
|  | 95e76a55da | ||
| 2b6659f4ec | |||
|  | e0c118a73e | ||
|  | 4e61b2aed6 | ||
|  | ba2758c0a3 | ||
|  | c718b966a1 | ||
|  | 99438e142f | ||
|  | 4d8076c3cf | ||
|  | db75c5b74a | ||
|  | 966a082147 | ||
|  | cd20a5ec29 | ||
|  | cc4c1c9201 | ||
|  | ff021d572c | ||
|  | 89992967be | ||
|  | 3c68bde62b | ||
|  | c38251f5b3 | ||
|  | a01f6d2322 | ||
|  | 417a33eb25 | ||
|  | 2e7f7f23b3 | ||
|  | e5e182761e | ||
|  | a094d88799 | ||
| e51915d1cd | |||
| 3a654f6ede | |||
| 5227751dca | |||
|  | 27eafe4ff4 | ||
|  | 8c83a9408b | ||
|  | fe2410f719 | ||
|  | a5e86bfb77 | ||
|  | 23be633798 | ||
|  | 813e0707d8 | ||
|  | 9ed9bf07fc | ||
|  | 47265c10d0 | ||
|  | 5cc633246a | ||
|  | 1f40385786 | ||
|  | eb2876324a | ||
|  | 633b817d76 | ||
|  | 2cfaa9b285 | ||
|  | f42ae97326 | ||
|  | 3b0028164b | ||
|  | 7420adeb5c | ||
|  | 316027ca3b | ||
|  | 9d58fba5c9 | ||
|  | 284c19ef89 | ||
|  | 7cfd17231a | ||
|  | 527830a5ae | ||
|  | c4ed30f594 | ||
|  | 156c1681cf | ||
|  | 3593fbca78 | ||
|  | 430fc8e8cb | ||
|  | 4fce19bad4 | ||
|  | 49f5848e7b | ||
|  | 90452100a4 | ||
|  | bf1196dd0f | ||
|  | 4316dc6516 | ||
|  | 9833a66a64 | ||
|  | 797bf06a9c | ||
|  | d98b00533d | ||
|  | bf8f7d8667 | ||
|  | 89c570f34f | ||
|  | d6a562863a | ||
|  | a02f06fe2e | ||
|  | 7b088d7bb4 | ||
|  | 477883ed39 | ||
|  | 748ed41096 | ||
|  | 86c50d4881 | ||
|  | c4c92e6dd9 | ||
|  | 7f0ba193ec | ||
|  | 87ed5b0fa8 | ||
|  | 6947743ac0 | ||
|  | 07e3710d44 | ||
|  | e68da7764f | ||
|  | c3ff894027 | ||
|  | f09f731d30 | ||
|  | 956c4341c7 | ||
|  | 7b68264dd7 | ||
|  | cfcf030bf8 | ||
|  | 0e7d7a5835 | ||
|  | 0856ebb889 | ||
|  | 25bf68cf0c | ||
|  | afc6f392c6 | ||
|  | a0b5e2052b | ||
|  | 87d1ef2bce | ||
|  | 537a6d3a0b | ||
| dbe97f564e | |||
|  | 3a3bf03114 | ||
| c09a32e9ad | |||
| b02a588dff | |||
|  | a4527940b8 | ||
|  | 9e8a25ed3e | ||
|  | 8ea46e146b | ||
|  | 5ecf3c3f87 | ||
|  | 325f103417 | ||
|  | ab4b1ae644 | ||
|  | 87ea44754e | ||
|  | 04dec50808 | ||
|  | e36189e2e7 | ||
|  | d6bdf510a4 | ||
|  | a464e93370 | ||
| 4b63afe62a | |||
| ac4c4b9441 | |||
|  | 16b10dc1b7 | ||
| 02d734eee8 | |||
| c5cdfc0d53 | |||
| 6d610ed61a | |||
| 792950be7c | |||
|  | af8969ce4a | ||
|  | 27c55e59a1 | ||
|  | 94a0747947 | ||
|  | d862bfba4f | ||
|  | b0d1d9c29a | ||
|  | 7b40a31979 | ||
|  | 823a8c3692 | ||
|  | 5494978db8 | ||
|  | 6076eb1cee | ||
|  | 131101d2ee | ||
|  | 62ad1f45ba | ||
|  | 402d18b889 | ||
|  | e32699c93f | ||
|  | 059a237b99 | ||
|  | d2bdbae6c8 | ||
|  | 510fcbe47e | ||
| 667e9c1a5d | |||
| 53b1d1f8b2 | |||
| c25e8889a4 | |||
|  | 8b0bbe71c9 | ||
| 8bfe14c019 | |||
| 208babbce3 | |||
| 02098a7aa9 | |||
| d0a982f385 | |||
| 1d1c121aab | |||
| fe12819163 | |||
|  | 023a30c008 | ||
|  | a2862a2587 | ||
|  | 054e936657 | ||
| 1d2e5069b8 | |||
| a147646743 | |||
| 32e7fc0038 | |||
| c15bf44032 | |||
| 0bcd55bd4e | |||
| ebef0b3511 | |||
| 713ceb05bf | |||
| dc8381b661 | |||
| b5b820c64b | |||
| f7055626d9 | |||
|  | 6ec3e96909 | ||
| 22da30eaa8 | |||
| 79fd115f5e | |||
| 8dc3d319cd | |||
| 27bb056397 | |||
| f9ba13dc32 | |||
| 6f60ef4346 | |||
| 28b950f467 | |||
| a9c7ec3dc1 | |||
| 920d4ac1ef | |||
| 0e96d313ec | |||
| 7211fdb1a3 | |||
|  | 381d6acc82 | ||
| d311c2cdeb | |||
| 219cae5d74 | |||
| 2968aee309 | |||
| 6cb4b35c93 | |||
| 15ec0f2d26 | |||
| 4781e30da2 | |||
| c8759cc035 | |||
| cb4f2f02ef | |||
| 7517626ab7 | |||
| 41c951b659 | |||
|  | ad279c6683 | ||
|  | 5f0817ddb7 | ||
|  | 7124cbcacd | ||
|  | 2a710a1a08 | ||
|  | 82ec2445a1 | ||
|  | cabb6d494d | ||
|  | 5c12481813 | ||
|  | b16f86dda1 | ||
|  | 2bc28db2cc | ||
|  | bf1b680b4a | ||
| e2afff0b8e | |||
| a382fc89ea | |||
| 3f0a3903ae | |||
| f46f98cef0 | |||
| bf6f1a917e | |||
| 71c0a4d340 | |||
| 63c550ead3 | |||
|  | d81fb79b4f | ||
|  | 144067d5b6 | ||
|  | 8106faa45c | ||
|  | d9ef301e0f | ||
|  | 90b52232ab | ||
|  | 0f000ea359 | ||
|  | fb8f81a4c8 | ||
|  | a76b3dd2a9 | ||
|  | 91aed5a777 | ||
| 366b2e10f1 | |||
|  | 8823cc6c6c | ||
| d2436bb976 | |||
|  | 74ef4da15b | ||
|  | bd96c67788 | ||
|  | da71de6806 | ||
|  | 0264da8ccc | ||
|  | 270d959ee0 | ||
|  | 6d11dfb80c | ||
|  | 4184bbb900 | ||
| ef994460c1 | |||
| 758708e18d | |||
| c0381144d1 | |||
| cda3ba6cb4 | |||
| a4636cc0c8 | |||
|  | 4c12c9d570 | ||
|  | f4db02521d | ||
| 60c24fc75a | |||
| 5853a19937 | |||
| 99f2c04bf6 | 
							
								
								
									
										21
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -1,21 +0,0 @@ | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: android | ||||
|  | ||||
| steps: | ||||
|   - name: code-analysis | ||||
|     image: mingc/android-build-box:latest | ||||
|     failure: ignore | ||||
|     commands: | ||||
|       - ls -la | ||||
|       - ./gradlew sonarqube -Dsonar.projectKey=RFS2 -Dsonar.host.url=$SONAR_HOST_URL -Dsonar.login=$SONAR_LOGIN -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" | ||||
|     environment: | ||||
|       SONAR_HOST_URL: | ||||
|         from_secret: sonarScannerHostUrl | ||||
|       SONAR_LOGIN: | ||||
|         from_secret: sonarScannerLogin | ||||
|  | ||||
|   - name: build | ||||
|     image: mingc/android-build-box:latest | ||||
|     commands: | ||||
|       - ./gradlew  :androidApp:build -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false | ||||
							
								
								
									
										10
									
								
								.gitea/workflows/assets/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.gitea/workflows/assets/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| version: '3' | ||||
| services: | ||||
|   selfoss: | ||||
|     container_name: selfoss | ||||
|     image: rsprta/selfoss | ||||
|     network_mode: "host" | ||||
|     ports: | ||||
|       - "8888:8888" | ||||
|  | ||||
|  | ||||
							
								
								
									
										61
									
								
								.gitea/workflows/common_build.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								.gitea/workflows/common_build.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| name: Build | ||||
| on: | ||||
|   workflow_call: | ||||
|  | ||||
| jobs: | ||||
|   BuildAndTest: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out repository code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Fetch tags | ||||
|         run: git fetch --tags -p | ||||
|       - uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|       - name: Setup Android SDK | ||||
|         uses: android-actions/setup-android@v3 | ||||
|       - name: Configure gradle... | ||||
|         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true" >> ~/.gradle/gradle.properties | ||||
|       - name: Build and test | ||||
|         run: ./gradlew build -x testReleaseUnitTest -x testDebugUnitTest -x testGithubConfigReleaseUnitTest -x testGithubConfigDebugUnitTest # These tests will be done in the next step | ||||
|   coverage: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Fetch tags | ||||
|         run: git fetch --tags -p | ||||
|       - uses: KengoTODA/actions-setup-docker-compose@v1 | ||||
|         with: | ||||
|           version: "2.23.3" | ||||
|       - name: run selfoss | ||||
|         run: | | ||||
|           docker compose -f .gitea/workflows/assets/docker-compose.yml up -d | ||||
|       - uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|           cache: gradle | ||||
|       - uses: gradle/actions/setup-gradle@v3 | ||||
|       - uses: android-actions/setup-android@v3 | ||||
|       - name: Configure gradle... | ||||
|         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true" >> ~/.gradle/gradle.properties | ||||
|       - name: coverage | ||||
|         run: | | ||||
|           ./gradlew :koverHtmlReport | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         with: | ||||
|           name: coverage | ||||
|           path: build/reports/kover/html | ||||
|           retention-days: 1 | ||||
|           overwrite: true | ||||
|           include-hidden-files: true | ||||
|       - name: Clean | ||||
|         if: always() | ||||
|         run: | | ||||
|           docker compose -f .gitea/workflows/assets/docker-compose.yml stop | ||||
							
								
								
									
										126
									
								
								.gitea/workflows/on_merge_on_release.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								.gitea/workflows/on_merge_on_release.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,126 @@ | ||||
| name: Create tag | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - release | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     uses: ./.gitea/workflows/common_build.yml | ||||
|   createTagAndChangelog: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: build | ||||
|     steps: | ||||
|       - name: Check out repository code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Config git | ||||
|         run: | | ||||
|           git config --global user.email aminecmi+giteadrone@pm.me | ||||
|           git config --global user.name giteadrone | ||||
|       - name: Creating the tag and generate changelog | ||||
|         run: | | ||||
|           git fetch --tags -p | ||||
|           PREV=$(git describe --tags --abbrev=0) | ||||
|           ./build.sh --publish --from-ci | ||||
|           VER=$(git describe --tags --abbrev=0) | ||||
|           CHANGELOG=$(git log $PREV..HEAD --pretty="- %s") | ||||
|           echo "**$VER | ||||
|            | ||||
|           $CHANGELOG | ||||
|            | ||||
|           -------------------------------------------------------------------- | ||||
|            | ||||
|           $(cat CHANGELOG.md)" > CHANGELOG.md | ||||
|           git add CHANGELOG.md | ||||
|           touch ./fastlane/metadata/android/en\-US/changelogs/$VER.txt | ||||
|           echo "**$VER** | ||||
|            | ||||
|           $CHANGELOG" > ./fastlane/metadata/android/en\-US/changelogs/$VER.txt | ||||
|           git add ./fastlane/metadata/android/en\-US/changelogs/$VER.txt | ||||
|           git commit -m "Changelog for $VER" | ||||
|       - name: Push changes | ||||
|         uses: appleboy/git-push-action@v1.0.0 | ||||
|         with: | ||||
|           author_name: giteadrone | ||||
|           author_email: aminecmi+giteadrone@pm.me | ||||
|           remote: ${{ secrets.REMOTE_URL }} | ||||
|           followtags: true | ||||
|           ssh_key: ${{ secrets.PRIVATE_KEY }} | ||||
|           tags: true | ||||
|           branch: release | ||||
|       - name: copy file via ssh password | ||||
|         uses: appleboy/scp-action@v0.1.7 | ||||
|         with: | ||||
|           host: amine-bouabdallaoui.fr | ||||
|           username: ubuntu | ||||
|           key: ${{ secrets.PRIVATE_KEY }} | ||||
|           source: "version.txt" | ||||
|           target: "/home/ubuntu/" | ||||
|       - name: deploy version file | ||||
|         uses: appleboy/ssh-action@v1.2.0 | ||||
|         with: | ||||
|           host: amine-bouabdallaoui.fr | ||||
|           username: ubuntu | ||||
|           key: ${{ secrets.PRIVATE_KEY }} | ||||
|           script: cd /home/ubuntu && sudo rm -rf /var/www/amine/version.txt && sudo chown www-data:www-data ./version.txt && sudo mv version.txt /var/www/amine/ | ||||
|   release: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: createTagAndChangelog | ||||
|     steps: | ||||
|       - name: Check out repository code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Fetch tags | ||||
|         id: version | ||||
|         run: | | ||||
|           git fetch --tags -p | ||||
|           PREV=$(git describe --tags --abbrev=0) | ||||
|           echo $PREV | ||||
|           echo "VERSION=$PREV" >> $GITHUB_OUTPUT | ||||
|       - uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|       - name: Setup Android SDK | ||||
|         uses: android-actions/setup-android@v3 | ||||
|       - name: Configure gradle... | ||||
|         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties | ||||
|       - name: setup go | ||||
|         uses: https://github.com/actions/setup-go@v4 | ||||
|         with: | ||||
|           go-version: '>=1.20.1' | ||||
|       - name: Generate APK | ||||
|         run: ./gradlew :androidApp:assembleGithubConfigRelease | ||||
|       - name: Get Key | ||||
|         run: wget ${{ secrets.KEY_URL }} | ||||
|       - name: Zippalign | ||||
|         run: | | ||||
|           sdkmanager "build-tools;31.0.0" | ||||
|           ls $ANDROID_HOME/build-tools  | ||||
|           $ANDROID_HOME/build-tools/31.0.0/zipalign -f -v 4 androidApp/build/outputs/apk/githubConfig/release/androidApp-githubConfig-release-unsigned.apk androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||
|       - name: Sigh | ||||
|         run: $ANDROID_HOME/build-tools/31.0.0/apksigner sign -v --out signed.apk --ks ./key --ks-key-alias ${{ secrets.KEY_ALIAS }} --ks-pass pass:${{ secrets.KEYSTORE_PASSWORD }} --v1-signing-enabled true --v2-signing-enabled true androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||
|       - name: Verify | ||||
|         run: $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk | ||||
|       - name: Release | ||||
|         uses: https://gitea.com/actions/gitea-release-action@main | ||||
|         with: | ||||
|           files: signed.apk | ||||
|           token: ${{ secrets.API_KEY }} | ||||
|           tag_name: ${{ steps.version.outputs.VERSION }} | ||||
|           name: ${{ steps.version.outputs.VERSION }} | ||||
|       - name: Send mail | ||||
|         uses: https://github.com/dawidd6/action-send-mail@v4 | ||||
|         with: | ||||
|           connection_url: ${{ secrets.MAIL_CONNECTION }} | ||||
|           to: ${{ secrets.MAIL_TO }} | ||||
|           from: ${{ secrets.MAIL_FROM }} | ||||
|           subject: Mapping file | ||||
|           priority: high | ||||
|           convert_markdown: true | ||||
|           body: Nouveau fichier de mapping pour la version ${{ steps.version.outputs.VERSION }} | ||||
|           attachments: androidApp/build/outputs/mapping/githubConfigRelease/mapping.txt | ||||
							
								
								
									
										26
									
								
								.gitea/workflows/on_pr.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.gitea/workflows/on_pr.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| name: Check PR code | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - master | ||||
|  | ||||
| jobs: | ||||
|   Lint: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out repository code | ||||
|         uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           distribution: 'temurin' # See 'Supported distributions' for available options | ||||
|           java-version: '17' | ||||
|       - name: Install klint | ||||
|         run: curl -sSLO https://github.com/pinterest/ktlint/releases/download/1.0.0/ktlint && chmod a+x ktlint && mv ktlint /usr/local/bin/ | ||||
|       - name: Install detekt | ||||
|         run: curl -sSLO https://github.com/detekt/detekt/releases/download/v1.23.1/detekt-cli-1.23.1.zip && unzip detekt-cli-1.23.1.zip | ||||
|       - name: Linting... | ||||
|         run: ktlint 'shared/**/*.kt' 'androidApp/**/*.kt' '!shared/build' || true | ||||
|       - name: Detecting... | ||||
|         run: ./detekt-cli-1.23.1/bin/detekt-cli --all-rules --excludes '**/shared/build/**/*.kt' || true | ||||
|   build: | ||||
|     uses: ./.gitea/workflows/common_build.yml | ||||
							
								
								
									
										9
									
								
								.gitea/workflows/on_push.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.gitea/workflows/on_push.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| name: Check master code | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     uses: ./.gitea/workflows/common_build.yml | ||||
							
								
								
									
										27
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										27
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ Please read the guidelines before contributing, and follow them (or try to) when | ||||
|  | ||||
| There are many ways to contribute to this project, you could [translate the app](https://crowdin.com/project/readerforselfoss), report bugs, request missing features, suggest enhancements and changes to existing ones. You also can improve the README with useful tips that could help the other users. | ||||
|  | ||||
| You can fork the repository, and [help me solve some issues](https://gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform/issues?q=is%3Aissue+is%3Aopen+label%3A%22Up+For+Grabs%22) or [develop new things](https://gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform/issues) | ||||
| You can fork the repository, and [help me solve some issues](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderForSelfoss-multiplatform/issues?q=is%3Aissue+is%3Aopen+label%3A%22Up+For+Grabs%22) or [develop new things](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderForSelfoss-multiplatform/issues) | ||||
|  | ||||
| ### What I can't help you with. | ||||
|  | ||||
| @@ -46,28 +46,3 @@ Always check if the web version of your instance is working. | ||||
| I won't provide any selfoss instance url. If you want to help, but to not have one, you'll have to install one, and use it. | ||||
|  | ||||
| All the details to need are [here](https://selfoss.aditu.de/). | ||||
|  | ||||
| # Build the project | ||||
|  | ||||
| You can directly import this project into IntellIJ/Android Studio. | ||||
|  | ||||
| You'll have to: | ||||
|  | ||||
| - Define some parameters either in `~/.gradle/gradle.properties` or as gradle parameters (see the examples) | ||||
|  | ||||
|     - appLoginUrl, appLoginUsername and appLoginPassword: url, username and password of a selfoss instance. **These are only used for tests. They can be empty if you don't test API calls.** | ||||
|  | ||||
| ### Examples: | ||||
| #### Inside ~/.gradle/gradle.properties | ||||
|  | ||||
| ``` | ||||
| appLoginUrl="URL" # It can be empty. | ||||
| appLoginUsername="LOGIN" # It can be empty. | ||||
| appLoginPassword="PASS" # It can be empty. | ||||
| ``` | ||||
|  | ||||
| #### As gradle parameters | ||||
|  | ||||
| ``` | ||||
| ./gradlew .... -P appLoginUrl="URL" -P appLoginUsername="LOGIN" -P appLoginPassword="PASS" | ||||
| ``` | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -321,3 +321,6 @@ fabric.properties | ||||
|  | ||||
|  | ||||
| crowdin.properties | ||||
|  | ||||
| .kotlin/ | ||||
| build-cache/ | ||||
							
								
								
									
										317
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										317
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,320 @@ | ||||
| **v124123641 | ||||
|  | ||||
| - Chore: no tests on build. | ||||
| - Merge pull request 'testing' (#170) from testing into master | ||||
| - fix: Displaying fixes. Fixes #155 | ||||
| - test: coverage | ||||
| - chore: update and use multiplatform datetime | ||||
| - Changelog for v124123421 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124123421 | ||||
|  | ||||
| - fix: Trying to fix the serialization issue. | ||||
| - Changelog for v124113311 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124113311 | ||||
|  | ||||
| - chore: update versions. (#165) | ||||
| - chore: fastlane changelog. | ||||
| - chore: fastlane fixes. | ||||
| - Changelog for v124113301 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124113301** | ||||
|  | ||||
| - chore: Gitea Action | ||||
| - Merge pull request 'chore: Gitea Action' (#164) from runner into master | ||||
| - chore: Gitea Action | ||||
| - chore: Readme update. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124041081** | ||||
|  | ||||
| - chore: comment. | ||||
| - fix: Last time fixing the parsing date hack before moving it to os version. | ||||
| - Changelog for v124030731 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124030731** | ||||
|  | ||||
| - fix: Basic auth and password can have non whitspace characters. Fixes 142. | ||||
| - Changelog for v124020451 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124020451** | ||||
|  | ||||
| - fix: Fixed handling of position in card adapter. | ||||
| - Changelog for v124010301 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124010301** | ||||
|  | ||||
| - fix: This may fix the oom errors. | ||||
| - Changelog for v124010191 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124010191** | ||||
|  | ||||
| - fix: moving listeners. | ||||
| - chore: removed a useless log. | ||||
| - Changelog for v124010032 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124010032** | ||||
|  | ||||
| - fix: Another date format thing. | ||||
| - Changelog for v124010031 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124010031** | ||||
|  | ||||
| - fix: Checking if selfoss instance. | ||||
| - fix: handle three characters lenght hexcode colors. | ||||
| - Changelog for v123113311 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123113311** | ||||
|  | ||||
| - chore: Source tracker url in the menu. | ||||
| - fix: Handle kodein proguard rules. | ||||
| - Changelog for v123102961 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123102961** | ||||
|  | ||||
| - chore: domain changes. | ||||
| - Changelog for v123102852 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123102852** | ||||
|  | ||||
| - chore: lint cleaning. | ||||
| - Changelog for v123102841 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123102841** | ||||
|  | ||||
| - chore: cleaning ci steps and upgrading dependencies. | ||||
| - feat: Self signed ssl support. | ||||
| - Changelog for v123061811 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123061811** | ||||
|  | ||||
| - feat: Added confirmation dialog for disconnect item menu. | ||||
| - Changelog for v123061651 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123061651** | ||||
|  | ||||
| - i18n: Translation update. | ||||
| - i18n: Translation update. | ||||
| - i18n: Translation update. | ||||
| - fix: avoid trying to open invalid image urls. | ||||
| - Changelog for v123051471 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123051471** | ||||
|  | ||||
| - fix: images could be null. | ||||
| - fix: Check if color is not empty before parsing it. | ||||
| - chore: Removed unused log. | ||||
| - Changelog for v123051331 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123051331** | ||||
|  | ||||
| - fix: illegal input. | ||||
| - Changelog for v123051321 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123051321** | ||||
|  | ||||
| - debug: Debug null context. | ||||
| - Changelog for v123051301 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123051301** | ||||
|  | ||||
| - feat: Basic auth from url. Fixes #142 (#143) | ||||
| - debug: Debug index out of bound exception. | ||||
| - Changelog for v123051211 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123051211** | ||||
|  | ||||
| - fix: Sometimes url isn't even defined. | ||||
| - Changelog for v123041021 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123041021** | ||||
|  | ||||
| - fix: 'Enable Core Library Desugaring to support older Android versions' (#138) from davidoskky/ReaderForSelfoss-multiplatform:desugaring into master | ||||
| - Enable Core Library Desugaring to support older Android versions | ||||
| - Changelog for v123030851 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123030851** | ||||
|  | ||||
| - chore: replace textDrawable library (#136) | ||||
| - refactor: Remove slow login check. Closes #135. | ||||
| - ci: send the mapping file after a release. | ||||
| - Changelog for v123030751 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123030751** | ||||
|  | ||||
| - debug: added a lot to pinpoint the url issue. | ||||
| - feat: Use /sources/stats in the home (#133) | ||||
| - Changelog for v123030681 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123030681** | ||||
|  | ||||
| - fix: Unread and starred can be null. | ||||
| - Fixed version number issue. | ||||
| - Changelog for v123030621 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123030621** | ||||
|  | ||||
| - fix: url required issue. | ||||
| - fix: Canvas reused issue. | ||||
| - Changelog for v123020572 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123020572** | ||||
|  | ||||
| - fix: requirecontext issues ? | ||||
| - debug: activity not found exception. | ||||
| - Changelog for v123020571 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123020571** | ||||
|  | ||||
| - chore: remove errors logging. | ||||
| - fix: quickfix for url param not provided for some sources. | ||||
| - Update 'CHANGELOG.md' | ||||
| - Changelog for v123020523 [CI SKIP] | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123020523** | ||||
|  | ||||
| - fix: Git changelog. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123020491** | ||||
|  | ||||
| - fix: Fixed acra bug reporting. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123010301** | ||||
|  | ||||
| - Chore: acra config. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123010281** | ||||
|  | ||||
| - improvement: Improve right to left support (#130) Co-authored-by: davidoskky <davidoskky@hidden.hidden> Co-committed-by: davidoskky <davidoskky@hidden.hidden> | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123010261** | ||||
|  | ||||
| - feat: Handle public instances (#126) Co-authored-by: davidoskky <davidoskky@hidden.hidden> Co-committed-by: davidoskky <davidoskky@hidden.hidden> | ||||
| - ci: Pull request should trigger ci. | ||||
| - fix: Complete the disconnection before redirecting to the login screen | ||||
| - Complete the disconnection before redirecting to the login screen | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123010241** | ||||
|  | ||||
| - Merge pull request 'feat: swipe down to close images' (#122) from davidoskky/ReaderForSelfoss-multiplatform:swipe_down into master | ||||
| - Remove unnecessary definition | ||||
| - Remove unused import | ||||
| - Adjust the image closing animation | ||||
| - Add a dark hue to the underlying article when swiping to close images | ||||
| - Rename activity style to avoid interferences | ||||
| - Adapt the style of the image activity to the rest of the application | ||||
| - Resolve issues when swiping down to close images | ||||
| - Close the image fragment only if the image has been dragged down | ||||
| - Animate swipe down to close images | ||||
| - Swipe down to close images | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v123010041** | ||||
|  | ||||
| - Merge pull request 'scroll-tag-filters' (#124) from scroll-tag-filters into master | ||||
| - fix: added POST_NOTIFICATIONS to fix notifications issues. | ||||
| - fix: scrollable filter sheet. | ||||
| - enhancement: Ellipsize chips text. | ||||
| - Cleaning. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v122123641** | ||||
|  | ||||
| - feat: Disable the failing source in the filter sheet. | ||||
| - feat: Display the source error in the sources list. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v122123631** | ||||
|  | ||||
| - build: Added back maven repos (see https://gitlab.com/fdroid/fdroiddata/-/commit/1fb9d60dc58511abc2bb4eb321977922a0682c8b#note_1223925153) | ||||
| - build: Added back maven repos (see https://gitlab.com/fdroid/fdroiddata/-/commit/1fb9d60dc58511abc2bb4eb321977922a0682c8b#note_1223925153) | ||||
| - debug: trying to resolve `Canvas: trying to use a recycled bitmap`. | ||||
| - fix: NPE may be caused by the binding or the title that was null. | ||||
| - chore: Skip drone pipeline on changelog push. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v122123621** | ||||
|  | ||||
| - fix: Automatic CHANGELOG generation. | ||||
| - Merge pull request 'Sources Upsert' (#119) from sources-edit into master | ||||
| - Source update screen. | ||||
| - Sources menu. | ||||
| - chore: Automatic CHANGELOG generation. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| # V2/Multiplatform rewrite | ||||
|  | ||||
| **v1** | ||||
|   | ||||
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| # ReaderForSelfoss-multiplatform | ||||
| # ReaderForSelfoss-multiplatform [](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderForSelfoss-multiplatform/actions?workflow=on_push.yml&actor=0&status=0) | ||||
|  | ||||
| [](https://crowdin.com/project/readerforselfoss) | ||||
|  | ||||
| @@ -10,10 +10,6 @@ If you are a user, you can still create new issues. I'll fix them when I can. | ||||
|  | ||||
| <a href="https://f-droid.org/packages/bou.amine.apps.readerforselfossv2.android"><img src="https://f-droid.org/badge/get-it-on.png" alt="Get it on F-Droid" height="100"></a> | ||||
|  | ||||
| ## Screen captures | ||||
|  | ||||
| <img src="res//fr-card.png?raw=true" alt="card view" width="400"/> <img src="res//fr-list.png?raw=true" alt="list view" width="400"/> | ||||
|  | ||||
| ## Like my app ? | ||||
|  | ||||
| <a href="https://www.buymeacoffee.com/aminecmi" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/lato-orange.png" alt="Buy Me A Coffee" style="height: 51px !important;width: 217px !important;" ></a> | ||||
| @@ -22,15 +18,15 @@ If you are a user, you can still create new issues. I'll fix them when I can. | ||||
|  | ||||
| 1. **You'll have to have a Selfoss instance running.** You'll find everything you need to install it [here](https://selfoss.aditu.de/). | ||||
|  | ||||
| 2. Check the [Contribution guide](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md). | ||||
| 2. Check the [Contribution guide](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md). | ||||
|  | ||||
| 3. Build the project by following [these steps](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md#build-the-project) (you should have read them after the contribution guide) | ||||
| 3. Build the project by following [these steps](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/.github/CONTRIBUTING.md#build-the-project) (you should have read them after the contribution guide) | ||||
|  | ||||
| ## Useful links | ||||
|  | ||||
| - [Check what changed](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/CHANGELOG.md) | ||||
| - [See what I'm doing](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/projects/1) | ||||
| - [Create an issue, or request a new feature](https://gitea.amine-louveau.fr/Louvorg/ReaderforSelfoss-multiplatform/issues) | ||||
| - [Check what changed](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/src/branch/master/CHANGELOG.md) | ||||
| - [See what I'm doing](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/projects/1) | ||||
| - [Create an issue, or request a new feature](https://gitea.amine-bouabdallaoui.fr/Louvorg/ReaderforSelfoss-multiplatform/issues) | ||||
| - [Help translation the app](https://crowdin.com/project/readerforselfoss) | ||||
|  | ||||
| ## Contributors (V1) (Alphabetical order) ❤️ | ||||
|   | ||||
							
								
								
									
										1
									
								
								androidApp/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								androidApp/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1,2 @@ | ||||
| /build | ||||
| .kotlin/ | ||||
| @@ -1,19 +1,22 @@ | ||||
| import java.io.ByteArrayOutputStream | ||||
|  | ||||
| val ignoreGitVersion: String by project | ||||
| val acraVersion = "5.9.7" | ||||
|  | ||||
| plugins { | ||||
|     id("com.android.application") | ||||
|     kotlin("android") | ||||
|     kotlin("kapt") | ||||
|     id("com.mikepenz.aboutlibraries.plugin") | ||||
|     id("org.jetbrains.kotlinx.kover") | ||||
| } | ||||
|  | ||||
| fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||
|     var result: String = ByteArrayOutputStream().use { outputStream -> | ||||
|     val result: String = ByteArrayOutputStream().use { outputStream -> | ||||
|         project.exec { | ||||
|             commandLine = cmd.split(" ") | ||||
|             standardOutput = outputStream | ||||
|             isIgnoreExitValue = ignore ?: false | ||||
|             isIgnoreExitValue = ignore | ||||
|         } | ||||
|         outputStream.toString() | ||||
|     } | ||||
| @@ -21,16 +24,15 @@ fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||
| } | ||||
|  | ||||
| fun gitVersion(): String { | ||||
|     var process = "" | ||||
|     val maybeTagOfCurrentCommit = execWithOutput("git -C ../ describe --contains HEAD", true) | ||||
|     process = if (maybeTagOfCurrentCommit.isEmpty()) { | ||||
|     val process = if (maybeTagOfCurrentCommit.isEmpty()) { | ||||
|         println("No tag on current commit. Will take the latest one.") | ||||
|         execWithOutput("git -C ../ for-each-ref refs/tags --sort=-authordate --format='%(refname:short)' --count=1") | ||||
|         execWithOutput("git -C ../ for-each-ref refs/tags --sort=-refname --format='%(refname:short)' --count=1") | ||||
|     } else { | ||||
|         println("Tag found on current commit") | ||||
|         execWithOutput("git -C ../ describe --contains HEAD") | ||||
|     } | ||||
|     return process.replace("'", "").substring(1).replace("\\.", "").trim() | ||||
|     return process.replace("^0", "").replace("'", "").substring(1).replace("\\.", "").trim() | ||||
| } | ||||
|  | ||||
| fun versionCodeFromGit(): Int { | ||||
| @@ -53,20 +55,24 @@ fun versionNameFromGit(): String { | ||||
|  | ||||
| android { | ||||
|     compileOptions { | ||||
|         // Flag to enable support for the new language APIs | ||||
|         isCoreLibraryDesugaringEnabled = true | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|         // Flag to enable support for the new language APIs | ||||
|         sourceCompatibility = JavaVersion.VERSION_17 | ||||
|         targetCompatibility = JavaVersion.VERSION_17 | ||||
|     } | ||||
|     compileSdk = 31 | ||||
|     buildToolsVersion = "31.0.0" | ||||
|  | ||||
|     // For Kotlin projects | ||||
|     kotlinOptions { | ||||
|         jvmTarget = "17" | ||||
|     } | ||||
|     compileSdk = 34 | ||||
|     buildFeatures { | ||||
|         viewBinding = true | ||||
|     } | ||||
|     defaultConfig { | ||||
|         applicationId = "bou.amine.apps.readerforselfossv2.android" | ||||
|         minSdk = 21 | ||||
|         targetSdk = 31 | ||||
|         minSdk = 25 | ||||
|         targetSdk = 34 | ||||
|         versionCode = versionCodeFromGit() | ||||
|         versionName = versionNameFromGit() | ||||
|  | ||||
| @@ -78,6 +84,12 @@ android { | ||||
|  | ||||
|         // tests | ||||
|         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||
|         testInstrumentationRunnerArguments["clearPackageData"] = "true" | ||||
|     } | ||||
|     packaging { | ||||
|         resources { | ||||
|             excludes += "/META-INF/{AL2.0,LGPL2.1}" | ||||
|         } | ||||
|     } | ||||
|     buildTypes { | ||||
|         getByName("release") { | ||||
| @@ -86,9 +98,6 @@ android { | ||||
|             proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro") | ||||
|         } | ||||
|         getByName("debug") { | ||||
|             buildConfigField("String", "LOGIN_URL", properties["appLoginUrl"] as String) | ||||
|             buildConfigField("String", "LOGIN_PASSWORD", properties["appLoginPassword"] as String) | ||||
|             buildConfigField("String", "LOGIN_USERNAME", properties["appLoginUsername"] as String) | ||||
|         } | ||||
|     } | ||||
|     flavorDimensions.add("build") | ||||
| @@ -98,81 +107,61 @@ android { | ||||
|             dimension = "build" | ||||
|         } | ||||
|     } | ||||
|     kotlinOptions { | ||||
|         jvmTarget = "1.8" | ||||
|     namespace = "bou.amine.apps.readerforselfossv2.android" | ||||
|     testOptions { | ||||
|         animationsDisabled = true | ||||
|         execution = "ANDROIDX_TEST_ORCHESTRATOR" | ||||
|         unitTests { | ||||
|             isIncludeAndroidResources = true | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:2.0.3") | ||||
|  | ||||
|     implementation(project(":shared")) | ||||
|     implementation("com.google.android.material:material:1.5.0") | ||||
|     implementation("androidx.appcompat:appcompat:1.4.1") | ||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.3") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.0") | ||||
|     implementation("com.google.android.material:material:1.9.0") | ||||
|     implementation("androidx.appcompat:appcompat:1.6.1") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.1") | ||||
|  | ||||
|     implementation("androidx.preference:preference-ktx:1.1.1") | ||||
|     implementation("androidx.preference:preference-ktx:1.2.1") | ||||
|  | ||||
|     // Testing | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-core:3.4.0-alpha02") | ||||
|     androidTestImplementation("androidx.test:runner:1.3.1-alpha02") | ||||
|     // Espresso-contrib for DatePicker, RecyclerView, Drawer actions, Accessibility checks, CountingIdlingResource | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-contrib:3.4.0-alpha02") | ||||
|     // Espresso-intents for validation and stubbing of Intents | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-intents:3.4.0-alpha02") | ||||
|     implementation(fileTree(mapOf("include" to listOf("*.jar"), "dir" to "libs"))) | ||||
|  | ||||
|     // Android Support | ||||
|     implementation("androidx.appcompat:appcompat:1.4.1") | ||||
|     implementation("com.google.android.material:material:1.5.0") | ||||
|     implementation("androidx.recyclerview:recyclerview:1.3.0-alpha01") | ||||
|     implementation("androidx.appcompat:appcompat:1.6.1") | ||||
|     implementation("com.google.android.material:material:1.9.0") | ||||
|     implementation("androidx.recyclerview:recyclerview:1.3.1") | ||||
|     implementation("androidx.legacy:legacy-support-v4:1.0.0") | ||||
|     implementation("androidx.vectordrawable:vectordrawable:1.2.0-alpha02") | ||||
|     implementation("androidx.browser:browser:1.4.0") | ||||
|     implementation("androidx.vectordrawable:vectordrawable:1.2.0-beta01") | ||||
|     implementation("androidx.cardview:cardview:1.0.0") | ||||
|     implementation("androidx.annotation:annotation:1.3.0") | ||||
|     implementation("androidx.work:work-runtime-ktx:2.7.1") | ||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.3") | ||||
|     implementation("org.jsoup:jsoup:1.14.3") | ||||
|  | ||||
|     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:1.1.5") | ||||
|     implementation("androidx.annotation:annotation:1.7.0") | ||||
|     implementation("androidx.work:work-runtime-ktx:2.8.1") | ||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.4") | ||||
|     implementation("org.jsoup:jsoup:1.15.4") | ||||
|  | ||||
|     //multidex | ||||
|     implementation("androidx.multidex:multidex:2.0.1") | ||||
|  | ||||
|     // About | ||||
|     implementation("com.mikepenz:aboutlibraries-core:8.9.4") | ||||
|     implementation("com.mikepenz:aboutlibraries:8.9.4") | ||||
|     implementation("com.mikepenz:aboutlibraries-definitions:8.9.4") | ||||
|  | ||||
|     // Async | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.0") | ||||
|  | ||||
|     // Retrofit + http logging + okhttp | ||||
|     implementation("com.squareup.retrofit2:retrofit:2.9.0") | ||||
|     implementation("com.squareup.okhttp3:logging-interceptor:5.0.0-alpha.3") | ||||
|     implementation("com.squareup.retrofit2:converter-gson:2.9.0") | ||||
|     implementation("com.burgstaller:okhttp-digest:2.5") | ||||
|     implementation("com.mikepenz:aboutlibraries-core:10.5.1") | ||||
|     implementation("com.mikepenz:aboutlibraries:10.5.1") | ||||
|  | ||||
|     // Material-ish things | ||||
|     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") | ||||
|     implementation("com.amulyakhare:com.amulyakhare.textdrawable:1.0.1") | ||||
|  | ||||
|     // glide | ||||
|     kapt("com.github.bumptech.glide:compiler:4.11.0") | ||||
|     implementation("com.github.bumptech.glide:okhttp3-integration:4.1.1") | ||||
|  | ||||
|     // Drawer | ||||
|     implementation("com.mikepenz:materialdrawer:8.4.5") | ||||
|     kapt("com.github.bumptech.glide:compiler:4.15.0") | ||||
|     implementation("com.github.bumptech.glide:okhttp3-integration:4.15.0") | ||||
|  | ||||
|     // Themes | ||||
|     implementation("com.52inc:scoops:1.0.0") | ||||
|     implementation("com.jaredrummler:colorpicker:1.1.0") | ||||
|     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") | ||||
|  | ||||
|     // Pager | ||||
|     implementation("me.relex:circleindicator:2.1.6") | ||||
|     implementation("androidx.viewpager2:viewpager2:1.1.0-beta01") | ||||
|     implementation("androidx.viewpager2:viewpager2:1.1.0-beta02") | ||||
|  | ||||
|     //Dependency Injection | ||||
|     implementation("org.kodein.di:kodein-di:7.14.0") | ||||
| @@ -188,16 +177,55 @@ dependencies { | ||||
|     //PhotoView | ||||
|     implementation("com.github.chrisbanes:PhotoView:2.3.0") | ||||
|  | ||||
|     implementation("androidx.core:core-ktx:1.8.0") | ||||
|     implementation("androidx.core:core-ktx:1.12.0") | ||||
|  | ||||
|     // implementation("androidx.lifecycle:lifecycle-livedata-ktx:2.5.1") | ||||
|     // implementation("androidx.lifecycle:lifecycle-common-java8:2.5.1") | ||||
|     // implementation("androidx.lifecycle:lifecycle-runtime:2.5.1") | ||||
|     implementation("androidx.lifecycle:lifecycle-extensions:2.2.0") | ||||
|  | ||||
|     // Network information | ||||
|     implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") | ||||
|  | ||||
|     // SQLDELIGHT | ||||
|     implementation("com.squareup.sqldelight:android-driver:1.5.3") | ||||
|     implementation("com.squareup.sqldelight:android-driver:1.5.4") | ||||
|  | ||||
|     //test | ||||
|     testImplementation("junit:junit:4.13.2") | ||||
|     testImplementation("io.mockk:mockk:1.12.0") | ||||
|     testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.1") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||
|     androidTestImplementation("androidx.test:runner:1.6.2") | ||||
|     androidTestImplementation("androidx.test:rules:1.6.1") | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-core:3.6.1") | ||||
|     implementation("androidx.test.espresso:espresso-idling-resource:3.6.1") | ||||
|     androidTestImplementation("androidx.test.ext:junit-ktx:1.2.1") | ||||
|     androidTestUtil("androidx.test:orchestrator:1.5.1") | ||||
|     testImplementation("org.robolectric:robolectric:4.14.1") | ||||
|     testImplementation("androidx.test:core-ktx:1.6.1") | ||||
|  | ||||
|     implementation("ch.acra:acra-http:$acraVersion") | ||||
|     implementation("ch.acra:acra-toast:$acraVersion") | ||||
|     implementation("com.google.auto.service:auto-service:1.1.1") | ||||
| } | ||||
|  | ||||
| tasks.withType<Test> { | ||||
|     outputs.upToDateWhen { false } | ||||
|     useJUnit() | ||||
|     testLogging { | ||||
|         exceptionFormat = org.gradle.api.tasks.testing.logging.TestExceptionFormat.FULL | ||||
|         events = setOf( | ||||
|             org.gradle.api.tasks.testing.logging.TestLogEvent.PASSED, | ||||
|             org.gradle.api.tasks.testing.logging.TestLogEvent.FAILED, | ||||
|             org.gradle.api.tasks.testing.logging.TestLogEvent.STANDARD_ERROR | ||||
|         ) | ||||
|         showStandardStreams = true | ||||
|     } | ||||
| } | ||||
|  | ||||
| aboutLibraries { | ||||
|     offlineMode = true | ||||
|     fetchRemoteLicense = false | ||||
|     fetchRemoteFunding = false | ||||
|     includePlatform = false | ||||
|     strictMode = com.mikepenz.aboutlibraries.plugin.StrictMode.FAIL | ||||
|     duplicationMode = com.mikepenz.aboutlibraries.plugin.DuplicateMode.MERGE | ||||
|     duplicationRule = com.mikepenz.aboutlibraries.plugin.DuplicateRule.GROUP | ||||
| } | ||||
							
								
								
									
										19
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							| @@ -30,15 +30,8 @@ | ||||
|     <fields>; | ||||
| } | ||||
|  | ||||
| -dontwarn okio.** | ||||
| -dontwarn retrofit2.Platform$Java8 | ||||
| -keep class retrofit.** { *; } | ||||
| -keepclasseswithmembers class * { | ||||
|     @retrofit.http.* <methods>; | ||||
| } | ||||
| -keepattributes *Annotation*,Signature | ||||
| -keepattributes Exceptions | ||||
| -dontwarn okio.** | ||||
| -dontwarn javax.annotation.Nullable | ||||
| -dontwarn javax.annotation.ParametersAreNonnullByDefault | ||||
|  | ||||
| @@ -62,6 +55,7 @@ | ||||
| # maybe remove later ? | ||||
| -keep class * extends androidx.fragment.app.Fragment | ||||
|  | ||||
| -dontwarn org.slf4j.impl.StaticLoggerBinder | ||||
|  | ||||
| # Keep `Companion` object fields of serializable classes. | ||||
| # This avoids serializer lookup through `getDeclaredClasses` as done for named companion objects. | ||||
| @@ -90,3 +84,14 @@ | ||||
| # @Serializable and @Polymorphic are used at runtime for polymorphic serialization. | ||||
| -keepattributes RuntimeVisibleAnnotations,AnnotationDefault | ||||
|  | ||||
| -dontwarn io.mockk.** | ||||
| -keep class io.mockk.** { *; } | ||||
|  | ||||
|  | ||||
|  | ||||
| # Kodein | ||||
| -keep, allowobfuscation, allowoptimization class org.kodein.type.TypeReference | ||||
| -keep, allowobfuscation, allowoptimization class org.kodein.type.JVMAbstractTypeToken$Companion$WrappingTest | ||||
|  | ||||
| -keep, allowobfuscation, allowoptimization class * extends org.kodein.type.TypeReference | ||||
| -keep, allowobfuscation, allowoptimization class * extends org.kodein.type.JVMAbstractTypeToken$Companion$WrappingTest | ||||
| @@ -0,0 +1,90 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.annotation.ArrayRes | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.action.ViewActions.replaceText | ||||
| import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView | ||||
| import androidx.test.espresso.assertion.ViewAssertions.doesNotExist | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isNotChecked | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withId | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
|  | ||||
| fun performLogin(someUrl: String? = null) { | ||||
|     onView(withId(R.id.urlView)).perform(click()).perform( | ||||
|         typeTextIntoFocusedView( | ||||
|             if (!someUrl.isNullOrEmpty()) someUrl else "http://10.0.2.2:8888" | ||||
|         ) | ||||
|     ) | ||||
|     onView(withId(R.id.signInButton)).perform(click()) | ||||
|     Thread.sleep(10000) | ||||
| } | ||||
|  | ||||
| fun loginAndInitHome() { | ||||
|  | ||||
|     performLogin() | ||||
|     onView(withText(R.string.gdpr_dialog_title)).check(matches(isDisplayed())) | ||||
|     onView(withText("OK")).perform(click()) | ||||
| } | ||||
|  | ||||
| fun changeAndCancelSetting(oldValue: String, newValue: String, openSettingItem: () -> Unit) { | ||||
|     openSettingItem() | ||||
|     onView( | ||||
|         withId(android.R.id.edit) | ||||
|     ).perform(replaceText(newValue)) | ||||
|     onView( | ||||
|         withId(android.R.id.button2) | ||||
|     ).perform(click()) | ||||
|     openSettingItem() | ||||
|     onView( | ||||
|         withId(android.R.id.edit) | ||||
|     ).check(matches(withText(oldValue))) | ||||
|     onView( | ||||
|         withText(newValue) | ||||
|     ).check(doesNotExist()) | ||||
|     onView( | ||||
|         withId(android.R.id.button2) | ||||
|     ).perform(click()) | ||||
| } | ||||
|  | ||||
| fun changeAndSaveSetting(oldValue: String, newValue: String, openSettingItem: () -> Unit) { | ||||
|     openSettingItem() | ||||
|     onView( | ||||
|         withId(android.R.id.edit) | ||||
|     ).perform(replaceText(newValue)) | ||||
|     onView( | ||||
|         withId(android.R.id.button1) | ||||
|     ).perform(click()) | ||||
|     openSettingItem() | ||||
|     onView( | ||||
|         withId(android.R.id.edit) | ||||
|     ).check(matches(withText(newValue))) | ||||
|     if (oldValue.isNotEmpty()) { | ||||
|         onView( | ||||
|             withText(oldValue) | ||||
|         ).check(doesNotExist()) | ||||
|     } | ||||
|     onView( | ||||
|         withId(android.R.id.button2) | ||||
|     ).perform(click()) | ||||
| } | ||||
|  | ||||
| fun testPreferencesFromArray( | ||||
|     context: Context, | ||||
|     @ArrayRes arrayRes: Int, | ||||
|     openSettingItem: () -> Unit | ||||
| ) { | ||||
|     openSettingItem() | ||||
|     context.resources.getStringArray(arrayRes).forEach { res -> | ||||
|         onView(withText(res)).check(matches(allOf(isDisplayed(), isNotChecked()))) | ||||
|         onView(withText(res)).perform(click()) | ||||
|         onView(withText(res)).check(doesNotExist()) | ||||
|         openSettingItem() | ||||
|         onView(withText(res)).check(matches(allOf(isDisplayed(), isChecked()))) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,137 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| import androidx.test.espresso.action.ViewActions | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isClickable | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isFocused | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isRoot | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isSelected | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withId | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import org.hamcrest.CoreMatchers.not | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class HomeActivityTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         loginAndInitHome() | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testMenu() { | ||||
|         onView(withId(R.id.action_search)).check(matches(isDisplayed())).check( | ||||
|             matches( | ||||
|                 isClickable() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withId(R.id.action_filter)).check(matches(isDisplayed())).check( | ||||
|             matches( | ||||
|                 isClickable() | ||||
|             ) | ||||
|         ) | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|         onView(withText(R.string.readAll)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.menu_home_sources)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.title_activity_settings)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.menu_home_refresh)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.issue_tracker_link)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.action_disconnect)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testMenuActions() { | ||||
|         onView(withId(R.id.action_search)).perform(click()) | ||||
|         onView( | ||||
|             withId(R.id.search_src_text) | ||||
|         ).check(matches(isFocused())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|  | ||||
|         onView(withId(R.id.action_filter)).perform(click()) | ||||
|         onView( | ||||
|             withText(R.string.filter_item_sources) | ||||
|         ).check(matches(isDisplayed())) | ||||
|         onView( | ||||
|             withText(R.string.filter_item_tags) | ||||
|         ).check(matches(isDisplayed())) | ||||
|         onView( | ||||
|             withId(R.id.floatingActionButton2) | ||||
|         ).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|  | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|         onView(withText(R.string.readAll)).perform(click()) | ||||
|         onView(withText(R.string.markall_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|  | ||||
|         onView(withText(R.string.menu_home_sources)).perform(click()) | ||||
|         onView(withId(R.id.fab)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|  | ||||
|         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||
|         onView(withText(R.string.pref_header_general)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|  | ||||
|         onView(withText(R.string.menu_home_refresh)).perform(click()) | ||||
|         onView(withText(R.string.refresh_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|  | ||||
|         /*onView(withText(R.string.issue_tracker_link)).perform(click()) | ||||
|         onView(withText(R.string.markall_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         )*/ | ||||
|  | ||||
|         onView(withText(R.string.action_disconnect)).perform(click()) | ||||
|         onView(withText(R.string.confirm_disconnect_title)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testEmptyView() { | ||||
|         onView(withId(R.id.emptyText)).check(matches(isDisplayed())) | ||||
|         onView( | ||||
|             hasBottombarItemText(R.string.tab_new) | ||||
|         ).check(matches(isDisplayed())).check(matches(isSelected())) | ||||
|         onView( | ||||
|             hasBottombarItemText(R.string.tab_read) | ||||
|         ).check(matches(isDisplayed())).check(matches(not(isSelected()))) | ||||
|         onView( | ||||
|             hasBottombarItemText(R.string.tab_favs) | ||||
|         ).check(matches(isDisplayed())).check(matches(not(isSelected()))) | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,83 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.app.Activity | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.IdlingRegistry | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isClickable | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isNotChecked | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withId | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||
| import org.junit.After | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class LoginActivityTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     private fun getActivity(): Activity? { | ||||
|         var activity: Activity? = null | ||||
|         activityRule.scenario.onActivity { | ||||
|             activity = it | ||||
|         } | ||||
|         return activity | ||||
|     } | ||||
|  | ||||
|     @Before | ||||
|     fun registerIdlingResource() { | ||||
|         IdlingRegistry.getInstance() | ||||
|             .register(CountingIdlingResourceSingleton.countingIdlingResource) | ||||
|     } | ||||
|  | ||||
|     @After | ||||
|     fun unregisterIdlingResource() { | ||||
|         IdlingRegistry.getInstance() | ||||
|             .unregister(CountingIdlingResourceSingleton.countingIdlingResource) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun viewIsInitialized() { | ||||
|         onView(withId(R.id.urlView)).check(matches(isDisplayed())) | ||||
|         onView(withId(R.id.selfSigned)).check(matches(isDisplayed())).check(matches(isNotChecked())) | ||||
|             .check( | ||||
|                 matches(isClickable()) | ||||
|             ) | ||||
|         onView(withId(R.id.withLogin)).check(matches(isDisplayed())) | ||||
|             .check(matches(isNotChecked())).check( | ||||
|                 matches(isClickable()) | ||||
|             ) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun urlError() { | ||||
|         performLogin("10.0.2.2:8888") | ||||
|         onView(withId(R.id.urlView)).perform(click()) | ||||
|         onView(withId(R.id.urlView)).check(matches(withError(R.string.wrong_infos))) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun multiError() { | ||||
|         onView(withId(R.id.signInButton)).perform(click()) | ||||
|         onView(withId(R.id.signInButton)).perform(click()) | ||||
|         onView(withId(R.id.signInButton)).perform(click()) | ||||
|         onView(withText(R.string.warning_wrong_url)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun connect() { | ||||
|         performLogin() | ||||
|         onView(withText(R.string.gdpr_dialog_title)).check(matches(isDisplayed())) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,172 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| import androidx.test.espresso.action.ViewActions | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.action.ViewActions.replaceText | ||||
| import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isEnabled | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isFocused | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isRoot | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withId | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
| import org.hamcrest.CoreMatchers.not | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityGeneralTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         loginAndInitHome() | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext() | ||||
|         ) | ||||
|         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||
|         onView(withText(R.string.pref_header_general)).perform(click()) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testGeneral() { | ||||
|         onView(withText(R.string.pref_api_items_number_title)).check(matches(isDisplayed())) | ||||
|         onView( | ||||
|             withSettingsCheckboxWidget(R.string.pref_general_infinite_loading_title) | ||||
|         ).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withText(R.string.pref_general_category_links)).check(matches(isDisplayed())) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_article_viewer_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), isChecked() | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.reader_static_bar_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.reader_static_bar_title)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withText(R.string.pref_general_category_displaying)).check(matches(isDisplayed())) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_card_view_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.card_height_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.card_height_title)).check( | ||||
|             matches( | ||||
|                 not(isEnabled()) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.switch_unread_count_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), isChecked() | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.display_all_counts_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testGeneralActionsNumberItems() { | ||||
|         onView(withText(R.string.pref_api_items_number_title)).perform(click()) | ||||
|         onView(withId(android.R.id.edit)).check(matches(isFocused())) | ||||
|  | ||||
|         // Value check | ||||
|         onView( | ||||
|             withId(android.R.id.edit) | ||||
|         ).perform(replaceText("AVC")) | ||||
|             .check(matches(withText(""))) | ||||
|         // TODO: should check message error. Not working for api level 30+ | ||||
|         onView( | ||||
|             withId(android.R.id.edit) | ||||
|         ).perform(replaceText("-1")) | ||||
|             .check(matches(withText(""))) | ||||
|         // TODO: should check message error. Not working for api level 30+ | ||||
|         onView( | ||||
|             withId(android.R.id.edit) | ||||
|         ).perform(replaceText("300")) | ||||
|             .check(matches(withText(""))) | ||||
|         onView( | ||||
|             withId(android.R.id.edit) | ||||
|         ).perform(typeTextIntoFocusedView("300")) | ||||
|             .check(matches(withText("30"))) | ||||
|         onView( | ||||
|             withId(android.R.id.edit) | ||||
|         ).perform(replaceText("10")) | ||||
|             .check(matches(withText("10"))) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|  | ||||
|         // Value saving | ||||
|         changeAndCancelSetting("20", "10") { | ||||
|             onView(withText(R.string.pref_api_items_number_title)).perform(click()) | ||||
|         } | ||||
|         changeAndSaveSetting("20", "10") { | ||||
|             onView(withText(R.string.pref_api_items_number_title)).perform(click()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testGeneralActionsCheckboxes() { | ||||
|         // article viewer settings | ||||
|         onView(withSettingsCheckboxFrame(R.string.reader_static_bar_title)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_article_viewer_title)).perform(click()) | ||||
|         onView(withSettingsCheckboxFrame(R.string.reader_static_bar_title)).check( | ||||
|             matches( | ||||
|                 not(isEnabled()) | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|         onView(withSettingsCheckboxFrame(R.string.card_height_title)).check(matches(not(isEnabled()))) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_card_view_title)).perform(click()) | ||||
|         onView(withSettingsCheckboxFrame(R.string.card_height_title)).check(matches(isEnabled())) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,169 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isEnabled | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isNotEnabled | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
| import org.hamcrest.CoreMatchers.not | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityOfflineTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     lateinit var context: Context | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         activityRule.scenario.onActivity { activity -> | ||||
|             context = activity.window.context | ||||
|         } | ||||
|         loginAndInitHome() | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext() | ||||
|         ) | ||||
|         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||
|         onView(withText(R.string.pref_header_offline)).perform(click()) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testOffline() { | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_periodic_refresh)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_items_caching)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_items_caching)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withText(R.string.pref_periodic_refresh_minutes_title)).check( | ||||
|             matches( | ||||
|                 allOf(isNotEnabled(), isDisplayed()) | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_refresh_when_charging)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).check( | ||||
|             matches( | ||||
|                 isNotEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_notify_new_items)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not(isChecked()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).check( | ||||
|             matches( | ||||
|                 isNotEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_update_sources)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), isChecked() | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testOfflineActions() { | ||||
|         onView(withText(R.string.pref_switch_items_caching_off)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.pref_switch_items_caching)).perform(click()) | ||||
|         onView(withText(R.string.pref_switch_items_caching_on)).check(matches(isDisplayed())) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_items_caching)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withText(R.string.pref_periodic_refresh_minutes_title)).check( | ||||
|             matches( | ||||
|                 isNotEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).check( | ||||
|             matches( | ||||
|                 isNotEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).check( | ||||
|             matches( | ||||
|                 isNotEnabled() | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|         onView(withText(R.string.pref_switch_periodic_refresh_off)).check( | ||||
|             matches( | ||||
|                 isDisplayed() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_periodic_refresh)).perform(click()) | ||||
|         onView(withText(R.string.pref_switch_periodic_refresh_on)).check( | ||||
|             matches( | ||||
|                 isDisplayed() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_periodic_refresh_minutes_title)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).check( | ||||
|             matches( | ||||
|                 isEnabled() | ||||
|             ) | ||||
|         ) | ||||
|         changeAndCancelSetting("360", "123") { | ||||
|             onView(withText(R.string.pref_periodic_refresh_minutes_title)).perform(click()) | ||||
|         } | ||||
|         changeAndSaveSetting("360", "123") { | ||||
|             onView(withText(R.string.pref_periodic_refresh_minutes_title)).perform(click()) | ||||
|         } | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_refresh_when_charging)).perform(click()) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_notify_new_items)).perform(click()) | ||||
|         onView(withSettingsCheckboxWidget(R.string.pref_switch_update_sources)).perform(click()) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,86 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isChecked | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
| import org.hamcrest.CoreMatchers.not | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityReaderTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     lateinit var context: Context | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         activityRule.scenario.onActivity { activity -> | ||||
|             context = activity.window.context | ||||
|         } | ||||
|         loginAndInitHome() | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext() | ||||
|         ) | ||||
|         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||
|         onView(withText(R.string.pref_header_viewer)).perform(click()) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testReader() { | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_actions_pager_scroll)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), not( | ||||
|                         isChecked() | ||||
|                     ) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withText(R.string.pref_content_reader_font_size)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.settings_reader_font)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testReaderActions() { | ||||
|         onView(withText(R.string.pref_switch_actions_pager_scroll_off)).check( | ||||
|             matches( | ||||
|                 isDisplayed() | ||||
|             ) | ||||
|         ) | ||||
|         onView(withSettingsCheckboxFrame(R.string.pref_switch_actions_pager_scroll)).perform(click()) | ||||
|         onView(withText(R.string.pref_switch_actions_pager_scroll_on)).check( | ||||
|             matches( | ||||
|                 isDisplayed() | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|         onView(withText(R.string.pref_content_reader_font_size)).perform(click()) | ||||
|         changeAndCancelSetting("16", "10") { | ||||
|             onView(withText(R.string.pref_content_reader_font_size)).perform(click()) | ||||
|         } | ||||
|         changeAndSaveSetting("16", "10") { | ||||
|             onView(withText(R.string.pref_content_reader_font_size)).perform(click()) | ||||
|         } | ||||
|  | ||||
|         testPreferencesFromArray(context, R.array.preloaded_fonts_values) { | ||||
|             onView(withText(R.string.settings_reader_font)).perform(click()) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,104 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isSelected | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
| import org.hamcrest.CoreMatchers.not | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|     lateinit var context: Context | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         activityRule.scenario.onActivity { activity -> | ||||
|             context = activity.window.context | ||||
|         } | ||||
|         loginAndInitHome() | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Test | ||||
|     fun testAllSettings() { | ||||
|  | ||||
|         onView(withText(R.string.pref_header_general)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.pref_header_viewer)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.pref_header_offline)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.pref_header_theme)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.pref_header_links)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.pref_switch_disable_acra)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), | ||||
|                     not(isSelected()) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         onView(withText(R.string.action_about)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Test | ||||
|     fun testThemes() { | ||||
|         testPreferencesFromArray(context, R.array.ModeTitles) { | ||||
|             onView(withText(R.string.pref_header_theme)).perform(click()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Test | ||||
|     fun testExperimentail() { | ||||
|         onView(withText(R.string.pref_header_experimental)).perform(click()) | ||||
|         changeAndCancelSetting("", "10") { | ||||
|             onView(withText(R.string.pref_api_timeout)).perform(click()) | ||||
|         } | ||||
|         changeAndSaveSetting("", "10") { | ||||
|             onView(withText(R.string.pref_api_timeout)).perform(click()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Test | ||||
|     fun testBugReports() { | ||||
|         onView(withText(R.string.pref_switch_disable_acra)).perform(click()) | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Test | ||||
|     fun testLinks() { | ||||
|         onView(withText(R.string.pref_header_links)).perform(click()) | ||||
|         onView(withText(R.string.issue_tracker_link)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.issue_tracker_summary)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.source_code)).check(matches(isDisplayed())) | ||||
|         onView(withText(R.string.translation)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Test | ||||
|     fun testAbout() { | ||||
|         onView(withText(R.string.action_about)).perform(click()) | ||||
|         onView(withText("ACRA")).check(matches(isDisplayed())) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,65 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.action.ViewActions.scrollCompletelyTo | ||||
| import androidx.test.espresso.action.ViewActions.typeTextIntoFocusedView | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withId | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import androidx.test.ext.junit.rules.ActivityScenarioRule | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.LargeTest | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
| import java.util.UUID | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SourcesActivityTest { | ||||
|  | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     lateinit var sourceName: String | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         sourceName = UUID.randomUUID().toString().substring(0, 15) | ||||
|  | ||||
|         loginAndInitHome() | ||||
|         openActionBarOverflowOrOptionsMenu( | ||||
|             ApplicationProvider.getApplicationContext<Context>() | ||||
|         ) | ||||
|         onView(withText(R.string.menu_home_sources)) | ||||
|             .perform(click()) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun addSource() { | ||||
|         onView(withId(R.id.fab)) | ||||
|             .perform(click()) | ||||
|         onView(withId(R.id.nameInput)) | ||||
|             .perform(click()).perform(typeTextIntoFocusedView(sourceName)) | ||||
|         onView(withId(R.id.sourceUri)) | ||||
|             .perform(click()) | ||||
|             .perform(typeTextIntoFocusedView("https://lorem-rss.herokuapp.com/feed?unit=year&interval=1&length=10")) | ||||
|         onView(withId(R.id.tags)) | ||||
|             .perform(click()).perform(typeTextIntoFocusedView("tag1,tag2,tag3")) | ||||
|         onView(withId(R.id.spoutsSpinner)) | ||||
|             .perform(click()) | ||||
|         onView(withText("RSS Feed")) | ||||
|             .perform(scrollCompletelyTo()) | ||||
|             .perform(click()) | ||||
|         onView(withId(R.id.saveBtn)) | ||||
|             .perform(click()) | ||||
|         onView(withText(sourceName)).check(matches(isDisplayed())) | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,101 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.view.View | ||||
| import android.widget.EditText | ||||
| import android.widget.ImageView | ||||
| import android.widget.RelativeLayout | ||||
| import androidx.annotation.DrawableRes | ||||
| import androidx.annotation.StringRes | ||||
| import androidx.core.graphics.drawable.toBitmap | ||||
| import androidx.test.espresso.Root | ||||
| import androidx.test.espresso.matcher.RootMatchers.isPlatformPopup | ||||
| import androidx.test.espresso.matcher.ViewMatchers.hasSibling | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withChild | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withClassName | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withId | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withParent | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withResourceName | ||||
| import androidx.test.espresso.matcher.ViewMatchers.withText | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
| import org.hamcrest.Description | ||||
| import org.hamcrest.Matcher | ||||
| import org.hamcrest.Matchers | ||||
| import org.hamcrest.TypeSafeMatcher | ||||
|  | ||||
|  | ||||
| fun withError(@StringRes id: Int): TypeSafeMatcher<View?> { | ||||
|     return object : TypeSafeMatcher<View?>() { | ||||
|         override fun matchesSafely(view: View?): Boolean { | ||||
|             if (view == null) { | ||||
|                 return false | ||||
|             } | ||||
|             val context = view.context | ||||
|             if (view !is EditText) { | ||||
|                 return false | ||||
|             } | ||||
|             if (view.error == null) { | ||||
|                 return false | ||||
|             } | ||||
|  | ||||
|             return view.error.toString() == context.getString(id) | ||||
|         } | ||||
|  | ||||
|         override fun describeTo(description: Description?) { | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun isPopupWindow(): Matcher<Root> { | ||||
|     return isPlatformPopup() | ||||
| } | ||||
|  | ||||
| fun withDrawable(@DrawableRes id: Int) = object : TypeSafeMatcher<View>() { | ||||
|     override fun describeTo(description: Description) { | ||||
|         description.appendText("ImageView with drawable same as drawable with id $id") | ||||
|     } | ||||
|  | ||||
|     override fun matchesSafely(view: View): Boolean { | ||||
|         val context = view.context | ||||
|         val expectedBitmap = context.getDrawable(id)!!.toBitmap() | ||||
|         try { | ||||
|             return view is ImageView && view.drawable.toBitmap().sameAs(expectedBitmap) | ||||
|         } catch (e: Exception) { | ||||
|             return false | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun hasBottombarItemText(@StringRes id: Int): Matcher<View>? { | ||||
|     return allOf( | ||||
|         withResourceName("fixed_bottom_navigation_icon"), | ||||
|         withParent( | ||||
|             allOf( | ||||
|                 withResourceName("fixed_bottom_navigation_icon_container"), | ||||
|                 hasSibling(withText(id)) | ||||
|             ) | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
|  | ||||
| fun withSettingsCheckboxWidget(@StringRes id: Int): Matcher<View>? { | ||||
|     return allOf( | ||||
|         withId(android.R.id.switch_widget), | ||||
|         withParent( | ||||
|             withSettingsCheckboxFrame(id) | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
|  | ||||
| fun withSettingsCheckboxFrame(@StringRes id: Int): Matcher<View>? { | ||||
|     return allOf( | ||||
|         withId(android.R.id.widget_frame), | ||||
|         hasSibling( | ||||
|             allOf( | ||||
|                 withClassName(Matchers.equalTo(RelativeLayout::class.java.name)), | ||||
|                 withChild( | ||||
|                     withText(id) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     package="bou.amine.apps.readerforselfossv2.android"> | ||||
|     xmlns:tools="http://schemas.android.com/tools"> | ||||
|  | ||||
|     <uses-permission android:name="android.permission.POST_NOTIFICATIONS"/> | ||||
|     <uses-permission android:name="android.permission.INTERNET" /> | ||||
|     <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> | ||||
|  | ||||
| @@ -16,7 +16,8 @@ | ||||
|         android:supportsRtl="true" | ||||
|         android:networkSecurityConfig="@xml/network_security_config" | ||||
|         android:theme="@style/NoBar" | ||||
|         android:dataExtractionRules="@xml/data_extraction_rules"> | ||||
|         android:dataExtractionRules="@xml/data_extraction_rules" | ||||
|         android:configChanges="uiMode"> | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:theme="@style/SplashTheme" | ||||
| @@ -52,7 +53,7 @@ | ||||
|                 android:value=".HomeActivity" /> | ||||
|         </activity> | ||||
|         <activity | ||||
|             android:name=".AddSourceActivity" | ||||
|             android:name=".UpsertSourceActivity" | ||||
|             android:parentActivityName=".SourcesActivity" | ||||
|             android:exported="true"> | ||||
|             <meta-data | ||||
| @@ -69,7 +70,8 @@ | ||||
|             android:name=".ReaderActivity"> | ||||
|         </activity> | ||||
|         <activity | ||||
|             android:name=".ImageActivity"> | ||||
|             android:name=".ImageActivity" | ||||
|             android:theme="@style/Theme.AppCompat.ImageActivity"> | ||||
|         </activity> | ||||
|  | ||||
|         <meta-data android:name="android.webkit.WebView.MetricsOptOut" | ||||
| @@ -79,8 +81,5 @@ | ||||
|             android:value="true" /> | ||||
|  | ||||
|         <meta-data android:name="android.max_aspect" android:value="2.1" /> | ||||
|         <meta-data | ||||
|             android:name="preloaded_fonts" | ||||
|             android:resource="@array/preloaded_fonts" /> | ||||
|     </application> | ||||
| </manifest> | ||||
| @@ -1,201 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Intent | ||||
| import android.os.Bundle | ||||
| import android.view.View | ||||
| import android.widget.* | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.constraintlayout.widget.ConstraintLayout | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityAddSourceBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlValid | ||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.ftinc.scoop.Scoop | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
|  | ||||
| class AddSourceActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private var mSpoutsValue: String? = null | ||||
|  | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var binding: ActivityAddSourceBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(this@AddSourceActivity) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityAddSourceBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolbar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         val drawable = binding.nameInput.background | ||||
|         drawable.setTint(appColors.colorAccent) | ||||
|  | ||||
|  | ||||
|         // TODO: clean | ||||
|         binding.nameInput.background = drawable | ||||
|  | ||||
|         val drawable1 = binding.sourceUri.background | ||||
|         drawable1.setTint(appColors.colorAccent) | ||||
|  | ||||
|         binding.sourceUri.background = drawable1 | ||||
|  | ||||
|         val drawable2 = binding.tags.background | ||||
|         drawable2.setTint(appColors.colorAccent) | ||||
|  | ||||
|         binding.tags.background = drawable2 | ||||
|  | ||||
|         setSupportActionBar(binding.toolbar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         maybeGetDetailsFromIntentSharing(intent, binding.sourceUri, binding.nameInput) | ||||
|  | ||||
|         binding.saveBtn.setTextColor(appColors.colorAccent) | ||||
|  | ||||
|         binding.saveBtn.setOnClickListener { | ||||
|             handleSaveSource( | ||||
|                 binding.tags, | ||||
|                 binding.nameInput.text.toString(), | ||||
|                 binding.sourceUri.text.toString() | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onResume() { | ||||
|         super.onResume() | ||||
|  | ||||
|         val baseUrl = appSettingsService.getBaseUrl() | ||||
|         if (baseUrl.isEmpty() || !baseUrl.isBaseUrlValid(this@AddSourceActivity)) { | ||||
|             mustLoginToAddSource() | ||||
|         } else { | ||||
|             handleSpoutsSpinner(binding.spoutsSpinner, binding.progress, binding.formContainer) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleSpoutsSpinner( | ||||
|         spoutsSpinner: Spinner, | ||||
|         mProgress: ProgressBar, | ||||
|         formContainer: ConstraintLayout | ||||
|     ) { | ||||
|         val spoutsKV = HashMap<String, String>() | ||||
|         spoutsSpinner.onItemSelectedListener = object : AdapterView.OnItemSelectedListener { | ||||
|             override fun onItemSelected(adapterView: AdapterView<*>, view: View?, i: Int, l: Long) { | ||||
|                 if (view != null) { | ||||
|                     val spoutName = (view as TextView).text.toString() | ||||
|                     mSpoutsValue = spoutsKV[spoutName] | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             override fun onNothingSelected(adapterView: AdapterView<*>) { | ||||
|                 mSpoutsValue = null | ||||
|             } | ||||
|         } | ||||
|  | ||||
|  | ||||
|         fun handleSpoutFailure(networkIssue: Boolean = false) { | ||||
|             Toast.makeText( | ||||
|                 this@AddSourceActivity, | ||||
|                 if (networkIssue) R.string.cant_get_spouts_no_network else R.string.cant_get_spouts, | ||||
|                 Toast.LENGTH_SHORT | ||||
|             ).show() | ||||
|             mProgress.visibility = View.GONE | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             try { | ||||
|                 val items = repository.getSpouts() | ||||
|                 if (items != null) { | ||||
|                     val itemsStrings = items.map { it.value.name } | ||||
|                     for ((key, value) in items) { | ||||
|                         spoutsKV[value.name] = key | ||||
|                     } | ||||
|  | ||||
|                     mProgress.visibility = View.GONE | ||||
|                     formContainer.visibility = View.VISIBLE | ||||
|  | ||||
|                     val spinnerArrayAdapter = | ||||
|                         ArrayAdapter( | ||||
|                             this@AddSourceActivity, | ||||
|                             android.R.layout.simple_spinner_item, | ||||
|                             itemsStrings | ||||
|                         ) | ||||
|                     spinnerArrayAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item) | ||||
|                     spoutsSpinner.adapter = spinnerArrayAdapter | ||||
|                 } else { | ||||
|                     handleSpoutFailure() | ||||
|                 } | ||||
|             } catch (e: NetworkUnavailableException) { | ||||
|                 handleSpoutFailure(networkIssue = true) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun maybeGetDetailsFromIntentSharing( | ||||
|         intent: Intent, | ||||
|         sourceUri: EditText, | ||||
|         nameInput: EditText | ||||
|     ) { | ||||
|         if (Intent.ACTION_SEND == intent.action && "text/plain" == intent.type) { | ||||
|             sourceUri.setText(intent.getStringExtra(Intent.EXTRA_TEXT)) | ||||
|             nameInput.setText(intent.getStringExtra(Intent.EXTRA_TITLE)) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun mustLoginToAddSource() { | ||||
|         Toast.makeText(this, getString(R.string.addStringNoUrl), Toast.LENGTH_SHORT).show() | ||||
|         val i = Intent(this, LoginActivity::class.java) | ||||
|         startActivity(i) | ||||
|         finish() | ||||
|     } | ||||
|  | ||||
|     private fun handleSaveSource(tags: EditText, title: String, url: String) { | ||||
|  | ||||
|         val sourceDetailsUnavailable = | ||||
|             title.isEmpty() || url.isEmpty() || mSpoutsValue == null || mSpoutsValue!!.isEmpty() | ||||
|  | ||||
|         when { | ||||
|             sourceDetailsUnavailable -> { | ||||
|                 Toast.makeText(this, R.string.form_not_complete, Toast.LENGTH_SHORT).show() | ||||
|             } | ||||
|             else -> { | ||||
|                 CoroutineScope(Dispatchers.Main).launch { | ||||
|                     val successfullyAddedSource = repository.createSource( | ||||
|                         title, | ||||
|                         url, | ||||
|                         mSpoutsValue!!, | ||||
|                         tags.text.toString(), | ||||
|                         "", | ||||
|                     ) | ||||
|                     if (successfullyAddedSource) { | ||||
|                         finish() | ||||
|                     } else { | ||||
|                         Toast.makeText( | ||||
|                             this@AddSourceActivity, | ||||
|                             R.string.cant_create_source, | ||||
|                             Toast.LENGTH_SHORT | ||||
|                         ).show() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -3,6 +3,7 @@ package bou.amine.apps.readerforselfossv2.android | ||||
| import android.os.Bundle | ||||
| import android.view.MenuItem | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.constraintlayout.motion.widget.MotionLayout | ||||
| import androidx.fragment.app.Fragment | ||||
| import androidx.fragment.app.FragmentActivity | ||||
| import androidx.viewpager2.adapter.FragmentStateAdapter | ||||
| @@ -23,7 +24,6 @@ class ImageActivity : AppCompatActivity() { | ||||
|         setContentView(view) | ||||
|  | ||||
|         setSupportActionBar(binding.toolBar) | ||||
|         supportActionBar?.setDisplayShowTitleEnabled(false) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|  | ||||
|         allImages = intent.getStringArrayListExtra("allImages") as ArrayList<String> | ||||
| @@ -31,12 +31,52 @@ class ImageActivity : AppCompatActivity() { | ||||
|  | ||||
|         binding.pager.adapter = ScreenSlidePagerAdapter(this) | ||||
|         binding.pager.setCurrentItem(position, false) | ||||
|  | ||||
|         val transitionListener = | ||||
|             object : MotionLayout.TransitionListener { | ||||
|                 override fun onTransitionStarted( | ||||
|                     motionLayout: MotionLayout?, | ||||
|                     startId: Int, | ||||
|                     endId: Int, | ||||
|                 ) { | ||||
|                     // Nothing | ||||
|                 } | ||||
|  | ||||
|                 override fun onTransitionChange( | ||||
|                     motionLayout: MotionLayout?, | ||||
|                     startId: Int, | ||||
|                     endId: Int, | ||||
|                     progress: Float, | ||||
|                 ) { | ||||
|                     // Nothing | ||||
|                 } | ||||
|  | ||||
|                 override fun onTransitionCompleted( | ||||
|                     motionLayout: MotionLayout?, | ||||
|                     currentId: Int, | ||||
|                 ) { | ||||
|                     if (motionLayout?.currentState == binding.root.endState) { | ||||
|                         onBackPressedDispatcher.onBackPressed() | ||||
|                         overridePendingTransition(0, 0) | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 override fun onTransitionTrigger( | ||||
|                     motionLayout: MotionLayout?, | ||||
|                     triggerId: Int, | ||||
|                     positive: Boolean, | ||||
|                     progress: Float, | ||||
|                 ) { | ||||
|                     // Nothing | ||||
|                 } | ||||
|             } | ||||
|         binding.root.setTransitionListener(transitionListener) | ||||
|     } | ||||
|  | ||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||
|         when (item.itemId) { | ||||
|             android.R.id.home -> { | ||||
|                 onBackPressed() | ||||
|                 onBackPressedDispatcher.onBackPressed() | ||||
|                 return true | ||||
|             } | ||||
|         } | ||||
| @@ -45,7 +85,6 @@ class ImageActivity : AppCompatActivity() { | ||||
|     } | ||||
|  | ||||
|     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : FragmentStateAdapter(fa) { | ||||
|  | ||||
|         override fun getItemCount(): Int = allImages.size | ||||
|  | ||||
|         override fun createFragment(position: Int): Fragment = ImageFragment.newInstance(allImages[position]) | ||||
|   | ||||
| @@ -2,7 +2,9 @@ package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.animation.Animator | ||||
| import android.animation.AnimatorListenerAdapter | ||||
| import android.annotation.SuppressLint | ||||
| import android.content.Intent | ||||
| import android.net.Uri | ||||
| import android.os.Bundle | ||||
| import android.text.TextUtils | ||||
| import android.view.Menu | ||||
| @@ -10,27 +12,28 @@ import android.view.MenuItem | ||||
| import android.view.View | ||||
| import android.view.inputmethod.EditorInfo | ||||
| import android.widget.TextView | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.appcompat.app.AppCompatDelegate | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityLoginBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlValid | ||||
| import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.mikepenz.aboutlibraries.LibsBuilder | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.acra.ACRA | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class LoginActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private var inValidCount: Int = 0 | ||||
|     private var isWithLogin = false | ||||
|  | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var binding: ActivityLoginBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
| @@ -38,9 +41,10 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(this@LoginActivity) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|  | ||||
|         handleTheme() | ||||
|  | ||||
|         binding = ActivityLoginBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
| @@ -51,14 +55,19 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|         handleBaseUrlFail() | ||||
|  | ||||
|         if (appSettingsService.getBaseUrl().isNotEmpty()) { | ||||
|             showProgress(true) | ||||
|             goToMain() | ||||
|         } | ||||
|  | ||||
|         handleActions() | ||||
|     } | ||||
|  | ||||
|     private fun handleActions() { | ||||
|     @SuppressLint("WrongConstant") // Constant is fetched from the settings | ||||
|     private fun handleTheme() { | ||||
|         AppCompatDelegate.setDefaultNightMode(appSettingsService.getCurrentTheme()) | ||||
|     } | ||||
|  | ||||
|     private fun handleActions() { | ||||
|         binding.passwordView.setOnEditorActionListener( | ||||
|             TextView.OnEditorActionListener { _, id, _ -> | ||||
|                 if (id == R.id.loginView || id == EditorInfo.IME_NULL) { | ||||
| @@ -66,7 +75,7 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|                     return@OnEditorActionListener true | ||||
|                 } | ||||
|                 false | ||||
|             } | ||||
|             }, | ||||
|         ) | ||||
|  | ||||
|         binding.signInButton.setOnClickListener { attemptLogin() } | ||||
| @@ -87,19 +96,28 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|             alertDialog.setMessage(getString(R.string.base_url_error)) | ||||
|             alertDialog.setButton( | ||||
|                 AlertDialog.BUTTON_NEUTRAL, | ||||
|                 "OK" | ||||
|                 "OK", | ||||
|             ) { dialog, _ -> dialog.dismiss() } | ||||
|             alertDialog.show() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun goToMain() { | ||||
|         CountingIdlingResourceSingleton.increment() | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             repository.updateApiInformation() | ||||
|             ACRA.errorReporter.putCustomData( | ||||
|                 "SELFOSS_API_VERSION", | ||||
|                 appSettingsService.getApiVersion().toString() | ||||
|             ) | ||||
|             CountingIdlingResourceSingleton.decrement() | ||||
|         } | ||||
|         val intent = Intent(this, HomeActivity::class.java) | ||||
|         startActivity(intent) | ||||
|         finish() | ||||
|     } | ||||
|  | ||||
|     private fun preferenceError(t: Throwable) { | ||||
|     private fun preferenceError() { | ||||
|         appSettingsService.resetLoginInformation() | ||||
|  | ||||
|         binding.urlView.error = getString(R.string.wrong_infos) | ||||
| @@ -108,24 +126,84 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     } | ||||
|  | ||||
|     private fun attemptLogin() { | ||||
|  | ||||
|         // Reset errors. | ||||
|         binding.urlView.error = null | ||||
|         binding.loginView.error = null | ||||
|         binding.passwordView.error = null | ||||
|  | ||||
|         // Store values at the time of the login attempt. | ||||
|         val url = binding.urlView.text.toString() | ||||
|         val login = binding.loginView.text.toString() | ||||
|         val password = binding.passwordView.text.toString() | ||||
|         val url = binding.urlView.text.toString().trim() | ||||
|         val login = binding.loginView.text.toString().trim() | ||||
|         val password = binding.passwordView.text.toString().trim() | ||||
|  | ||||
|         failInvalidUrl(url) | ||||
|         failLoginDetails(password, login) | ||||
|  | ||||
|         showProgress(true) | ||||
|  | ||||
|         appSettingsService.updateSelfSigned(binding.selfSigned.isChecked) | ||||
|  | ||||
|         repository.refreshLoginInformation(url, login, password) | ||||
|  | ||||
|         CountingIdlingResourceSingleton.increment() | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             try { | ||||
|                 repository.updateApiInformation() | ||||
|             } catch (e: Exception) { | ||||
|                 if (e.message?.startsWith("No transformation found") == true) { | ||||
|                     Toast.makeText( | ||||
|                         applicationContext, | ||||
|                         R.string.application_selfoss_only, | ||||
|                         Toast.LENGTH_LONG, | ||||
|                     ).show() | ||||
|                     preferenceError() | ||||
|                     showProgress(false) | ||||
|                 } | ||||
|             } | ||||
|             val result = repository.login() | ||||
|             if (result) { | ||||
|                 val errorFetching = repository.checkIfFetchFails() | ||||
|                 if (!errorFetching) { | ||||
|                     goToMain() | ||||
|                 } else { | ||||
|                     preferenceError() | ||||
|                 } | ||||
|             } else { | ||||
|                 preferenceError() | ||||
|             } | ||||
|             showProgress(false) | ||||
|             CountingIdlingResourceSingleton.decrement() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun failLoginDetails( | ||||
|         password: String, | ||||
|         login: String, | ||||
|     ) { | ||||
|         var lastFocusedView: View? = null | ||||
|         var cancel = false | ||||
|         var focusView: View? = null | ||||
|  | ||||
|         if (!url.isBaseUrlValid(this@LoginActivity)) { | ||||
|             binding.urlView.error = getString(R.string.login_url_problem) | ||||
|             focusView = binding.urlView | ||||
|         if (isWithLogin) { | ||||
|             if (TextUtils.isEmpty(password)) { | ||||
|                 binding.passwordView.error = getString(R.string.error_invalid_password) | ||||
|                 lastFocusedView = binding.passwordView | ||||
|                 cancel = true | ||||
|             } | ||||
|  | ||||
|             if (TextUtils.isEmpty(login)) { | ||||
|                 binding.loginView.error = getString(R.string.error_field_required) | ||||
|                 lastFocusedView = binding.loginView | ||||
|                 cancel = true | ||||
|             } | ||||
|         } | ||||
|         maybeCancelAndFocusView(cancel, lastFocusedView) | ||||
|     } | ||||
|  | ||||
|     private fun failInvalidUrl(url: String) { | ||||
|         val focusView = binding.urlView | ||||
|         var cancel = false | ||||
|         if (url.isBaseUrlInvalid()) { | ||||
|             cancel = true | ||||
|             binding.urlView.error = getString(R.string.login_url_problem) | ||||
|             inValidCount++ | ||||
|             if (inValidCount == 3) { | ||||
|                 val alertDialog = AlertDialog.Builder(this).create() | ||||
| @@ -133,46 +211,21 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|                 alertDialog.setMessage(getString(R.string.text_wrong_url)) | ||||
|                 alertDialog.setButton( | ||||
|                     AlertDialog.BUTTON_NEUTRAL, | ||||
|                     "OK" | ||||
|                     "OK", | ||||
|                 ) { dialog, _ -> dialog.dismiss() } | ||||
|                 alertDialog.show() | ||||
|                 inValidCount = 0 | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (isWithLogin) { | ||||
|             if (TextUtils.isEmpty(password)) { | ||||
|                 binding.passwordView.error = getString(R.string.error_invalid_password) | ||||
|                 focusView = binding.passwordView | ||||
|                 cancel = true | ||||
|             } | ||||
|  | ||||
|             if (TextUtils.isEmpty(login)) { | ||||
|                 binding.loginView.error = getString(R.string.error_field_required) | ||||
|                 focusView = binding.loginView | ||||
|                 cancel = true | ||||
|             } | ||||
|         maybeCancelAndFocusView(cancel, focusView) | ||||
|     } | ||||
|  | ||||
|     private fun maybeCancelAndFocusView( | ||||
|         cancel: Boolean, | ||||
|         focusView: View?, | ||||
|     ) { | ||||
|         if (cancel) { | ||||
|             focusView?.requestFocus() | ||||
|         } else { | ||||
|             showProgress(true) | ||||
|  | ||||
|             repository.refreshLoginInformation(url, login, password) | ||||
|  | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 val result = repository.login() | ||||
|                 if (result) { | ||||
|                     repository.updateApiVersion() | ||||
|                     goToMain() | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.Main).launch { | ||||
|                         preferenceError(Exception("Not success")) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             showProgress(false) | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -184,12 +237,13 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|             .animate() | ||||
|             .setDuration(shortAnimTime.toLong()) | ||||
|             .alpha( | ||||
|                 if (show) 0F else 1F | ||||
|             ).setListener(object : AnimatorListenerAdapter() { | ||||
|                 if (show) 0F else 1F, | ||||
|             ).setListener( | ||||
|                 object : AnimatorListenerAdapter() { | ||||
|                     override fun onAnimationEnd(animation: Animator) { | ||||
|                         binding.loginForm.visibility = if (show) View.GONE else View.VISIBLE | ||||
|                     } | ||||
|         } | ||||
|                 }, | ||||
|             ) | ||||
|  | ||||
|         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||
| @@ -197,12 +251,13 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|             .animate() | ||||
|             .setDuration(shortAnimTime.toLong()) | ||||
|             .alpha( | ||||
|                 if (show) 1F else 0F | ||||
|             ).setListener(object : AnimatorListenerAdapter() { | ||||
|                 if (show) 1F else 0F, | ||||
|             ).setListener( | ||||
|                 object : AnimatorListenerAdapter() { | ||||
|                     override fun onAnimationEnd(animation: Animator) { | ||||
|                         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||
|                     } | ||||
|         } | ||||
|                 }, | ||||
|             ) | ||||
|     } | ||||
|  | ||||
| @@ -213,13 +268,25 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||
|         return when (item.itemId) { | ||||
|             R.id.issue_tracker -> { | ||||
|                 val browserIntent = | ||||
|                     Intent(Intent.ACTION_VIEW, Uri.parse(AppSettingsService.trackerUrl)) | ||||
|                 startActivity(browserIntent) | ||||
|                 return true | ||||
|             } | ||||
|  | ||||
|             R.id.about -> { | ||||
|                 LibsBuilder() | ||||
|                     .withAboutIconShown(true) | ||||
|                     .withAboutVersionShown(true) | ||||
|                     .withAboutSpecial2("Bug reports") | ||||
|                     .withAboutSpecial2Description(AppSettingsService.trackerUrl) | ||||
|                     .withAboutSpecial1("Project Page") | ||||
|                     .withAboutSpecial1Description(AppSettingsService.sourceUrl) | ||||
|                     .start(this) | ||||
|                 true | ||||
|             } | ||||
|  | ||||
|             else -> super.onOptionsItemSelected(item) | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -8,7 +8,6 @@ import bou.amine.apps.readerforselfossv2.android.databinding.ActivityMainBinding | ||||
| class MainActivity : AppCompatActivity() { | ||||
|     private lateinit var binding: ActivityMainBinding | ||||
|  | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityMainBinding.inflate(layoutInflater) | ||||
|   | ||||
| @@ -3,42 +3,54 @@ package bou.amine.apps.readerforselfossv2.android | ||||
| import android.app.NotificationChannel | ||||
| import android.app.NotificationManager | ||||
| import android.content.Context | ||||
| import android.graphics.drawable.Drawable | ||||
| import android.net.Uri | ||||
| import android.os.Build | ||||
| import android.widget.ImageView | ||||
| import android.widget.Toast | ||||
| import androidx.lifecycle.DefaultLifecycleObserver | ||||
| import androidx.lifecycle.LifecycleOwner | ||||
| import androidx.lifecycle.ProcessLifecycleOwner | ||||
| import androidx.multidex.MultiDexApplication | ||||
| import androidx.preference.PreferenceManager | ||||
| import bou.amine.apps.readerforselfossv2.DI.networkModule | ||||
| import bou.amine.apps.readerforselfossv2.android.testing.TestingHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | ||||
| import bou.amine.apps.readerforselfossv2.dao.DriverFactory | ||||
| import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import com.ftinc.scoop.Scoop | ||||
| import com.github.ln_12.library.ConnectivityStatus | ||||
| import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader | ||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader | ||||
| import io.github.aakira.napier.DebugAntilog | ||||
| import io.github.aakira.napier.Napier | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.MutableStateFlow | ||||
| import kotlinx.coroutines.launch | ||||
| import org.kodein.di.* | ||||
| import org.acra.ACRA | ||||
| import org.acra.ReportField | ||||
| import org.acra.config.httpSender | ||||
| import org.acra.config.toast | ||||
| import org.acra.data.StringFormat | ||||
| import org.acra.ktx.initAcra | ||||
| import org.acra.sender.HttpSender | ||||
| import org.kodein.di.DI | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.bind | ||||
| import org.kodein.di.instance | ||||
| import org.kodein.di.singleton | ||||
|  | ||||
| class MyApp : MultiDexApplication(), DIAware { | ||||
|  | ||||
|     override val di by DI.lazy { | ||||
|         bind<AppSettingsService>() with singleton { AppSettingsService(ACRA.isACRASenderServiceProcess() || TestingHelper().isUnitTest()) } | ||||
|         import(networkModule) | ||||
|         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } | ||||
|         bind<ReaderForSelfossDB>() with singleton { ReaderForSelfossDB(driverFactory.createDriver()) } | ||||
|         bind<Repository>() with singleton { Repository(instance(), instance(), connectivityStatus, instance()) } | ||||
|         bind<Repository>() with | ||||
|                 singleton { | ||||
|                     Repository( | ||||
|                         instance(), | ||||
|                         instance(), | ||||
|                         isConnectionAvailable, | ||||
|                         instance(), | ||||
|                     ) | ||||
|                 } | ||||
|         bind<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } | ||||
|         bind<AppViewModel>() with singleton { AppViewModel(repository = instance()) } | ||||
|     } | ||||
| @@ -48,23 +60,29 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|     private val connectivityStatus: ConnectivityStatus by instance() | ||||
|     private val driverFactory: DriverFactory by instance() | ||||
|  | ||||
|     // TODO: handle with the "previous" way | ||||
|     private val isConnectionAvailable: MutableStateFlow<Boolean> = MutableStateFlow(true) | ||||
|  | ||||
|     override fun onCreate() { | ||||
|         super.onCreate() | ||||
|         Napier.base(DebugAntilog()) | ||||
|  | ||||
|         initDrawerImageLoader() | ||||
|  | ||||
|         initTheme() | ||||
|  | ||||
|         if (!ACRA.isACRASenderServiceProcess()) { | ||||
|             tryToHandleBug() | ||||
|  | ||||
|             handleNotificationChannels() | ||||
|  | ||||
|         ProcessLifecycleOwner.get().lifecycle.addObserver(AppLifeCycleObserver(connectivityStatus, repository)) | ||||
|             ProcessLifecycleOwner.get().lifecycle.addObserver( | ||||
|                 AppLifeCycleObserver( | ||||
|                     connectivityStatus, | ||||
|                     repository, | ||||
|                 ), | ||||
|             ) | ||||
|  | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                 viewModel.networkAvailableProvider.collect { networkAvailable -> | ||||
|                 val toastMessage = if (networkAvailable) { | ||||
|                     val toastMessage = | ||||
|                         if (networkAvailable) { | ||||
|                             repository.handleDBActions() | ||||
|                             R.string.network_connectivity_retrieved | ||||
|                         } else { | ||||
| @@ -74,11 +92,57 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|                     Toast.makeText( | ||||
|                         applicationContext, | ||||
|                         toastMessage, | ||||
|                     Toast.LENGTH_SHORT | ||||
|                         Toast.LENGTH_SHORT, | ||||
|                     ).show() | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         repository.migrate(driverFactory) | ||||
|     } | ||||
|  | ||||
|     override fun attachBaseContext(base: Context?) { | ||||
|         super.attachBaseContext(base) | ||||
|  | ||||
|         initAcra { | ||||
|             reportFormat = StringFormat.JSON | ||||
|             reportContent = | ||||
|                 listOf( | ||||
|                     ReportField.REPORT_ID, | ||||
|                     ReportField.INSTALLATION_ID, | ||||
|                     ReportField.APP_VERSION_CODE, | ||||
|                     ReportField.APP_VERSION_NAME, | ||||
|                     ReportField.BUILD, | ||||
|                     ReportField.ANDROID_VERSION, | ||||
|                     ReportField.BRAND, | ||||
|                     ReportField.PHONE_MODEL, | ||||
|                     ReportField.AVAILABLE_MEM_SIZE, | ||||
|                     ReportField.TOTAL_MEM_SIZE, | ||||
|                     ReportField.STACK_TRACE, | ||||
|                     ReportField.APPLICATION_LOG, | ||||
|                     ReportField.LOGCAT, | ||||
|                     ReportField.INITIAL_CONFIGURATION, | ||||
|                     ReportField.CRASH_CONFIGURATION, | ||||
|                     ReportField.IS_SILENT, | ||||
|                     ReportField.USER_APP_START_DATE, | ||||
|                     ReportField.USER_COMMENT, | ||||
|                     ReportField.USER_CRASH_DATE, | ||||
|                     ReportField.USER_EMAIL, | ||||
|                     ReportField.CUSTOM_DATA, | ||||
|                 ) | ||||
|             toast { | ||||
|                 // required | ||||
|                 text = getString(R.string.crash_toast_text) | ||||
|                 length = Toast.LENGTH_SHORT | ||||
|             } | ||||
|             httpSender { | ||||
|                 uri = | ||||
|                     "https://bugs.amine-bouabdallaoui.fr/report" // best guess, you may need to adjust this | ||||
|                 basicAuthLogin = "qMEscjj89Gwt6cPR" | ||||
|                 basicAuthPassword = "Yo58QFlGzFaWlBzP" | ||||
|                 httpMethod = HttpSender.Method.POST | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleNotificationChannels() { | ||||
| @@ -91,56 +155,38 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|  | ||||
|             val newItemsChannelname = getString(R.string.new_items_channel_sync) | ||||
|             val newItemsChannelimportance = NotificationManager.IMPORTANCE_DEFAULT | ||||
|             val newItemsChannelmChannel = NotificationChannel(AppSettingsService.newItemsChannelId, newItemsChannelname, newItemsChannelimportance) | ||||
|             val newItemsChannelmChannel = | ||||
|                 NotificationChannel( | ||||
|                     AppSettingsService.newItemsChannelId, | ||||
|                     newItemsChannelname, | ||||
|                     newItemsChannelimportance, | ||||
|                 ) | ||||
|  | ||||
|             notificationManager.createNotificationChannel(mChannel) | ||||
|             notificationManager.createNotificationChannel(newItemsChannelmChannel) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun initDrawerImageLoader() { | ||||
|         DrawerImageLoader.init(object : AbstractDrawerImageLoader() { | ||||
|             override fun set(imageView: ImageView, uri: Uri, placeholder: Drawable, tag: String?) { | ||||
|                 Glide.with(imageView.context) | ||||
|                     .load(uri.toString()) | ||||
|                     .apply(RequestOptions.fitCenterTransform().placeholder(placeholder)) | ||||
|                     .into(imageView) | ||||
|             } | ||||
|  | ||||
|             override fun cancel(imageView: ImageView) { | ||||
|                 Glide.with(imageView.context).clear(imageView) | ||||
|             } | ||||
|  | ||||
|             override fun placeholder(ctx: Context, tag: String?): Drawable { | ||||
|                 return baseContext.resources.getDrawable(R.mipmap.ic_launcher) | ||||
|             } | ||||
|         }) | ||||
|     } | ||||
|  | ||||
|     private fun initTheme() { | ||||
|         Scoop.waffleCone() | ||||
|             .addFlavor(getString(R.string.default_theme), R.style.NoBar, true) | ||||
|             .addFlavor(getString(R.string.default_dark_theme), R.style.NoBarDark, false) | ||||
|             .setSharedPreferences(PreferenceManager.getDefaultSharedPreferences(this)) | ||||
|             .initialize() | ||||
|     } | ||||
|  | ||||
|     private fun tryToHandleBug() { | ||||
|         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() | ||||
|  | ||||
|         Thread.setDefaultUncaughtExceptionHandler { thread, e -> | ||||
|             if (e is java.lang.NoClassDefFoundError && e.stackTrace.asList().any { | ||||
|             if (e is NoClassDefFoundError && | ||||
|                 e.stackTrace.asList().any { | ||||
|                     it.toString().contains("android.view.ViewDebug") | ||||
|                 }) { | ||||
|                 Unit | ||||
|                 } | ||||
|             ) { | ||||
|                 // Nothing | ||||
|             } else { | ||||
|                 oldHandler.uncaughtException(thread, e) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class AppLifeCycleObserver(val connectivityStatus: ConnectivityStatus, val repository: Repository) : DefaultLifecycleObserver { | ||||
|  | ||||
|     class AppLifeCycleObserver( | ||||
|         val connectivityStatus: ConnectivityStatus, | ||||
|         val repository: Repository, | ||||
|     ) : DefaultLifecycleObserver { | ||||
|         override fun onResume(owner: LifecycleOwner) { | ||||
|             super.onResume(owner) | ||||
|             repository.connectionMonitored = true | ||||
|   | ||||
| @@ -12,12 +12,9 @@ import androidx.viewpager2.adapter.FragmentStateAdapter | ||||
| import androidx.viewpager2.widget.ViewPager2 | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityReaderBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.fragments.ArticleFragment | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.ftinc.scoop.Scoop | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -26,23 +23,23 @@ import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private var currentItem: Int = 0 | ||||
|     private lateinit var appColors: AppColors | ||||
|  | ||||
|     private lateinit var toolbarMenu: Menu | ||||
|  | ||||
|     private lateinit var binding: ActivityReaderBinding | ||||
|  | ||||
|     private var allItems: ArrayList<SelfossModel.Item> = ArrayList() | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     private fun showMenuItem(willAddToFavorite: Boolean) { | ||||
|         if (willAddToFavorite) { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.WHITE) | ||||
|             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.WHITE) | ||||
|         } else { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.RED) | ||||
|             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.RED) | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -56,27 +53,28 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         appColors = AppColors(this) | ||||
|         binding = ActivityReaderBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolBar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         setSupportActionBar(binding.toolBar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         if (allItems.isEmpty()) { | ||||
|         currentItem = intent.getIntExtra("currentItem", 0) | ||||
|  | ||||
|         allItems = repository.getReaderItems() | ||||
|  | ||||
|         if (allItems.isEmpty() || currentItem > allItems.size) { | ||||
|             finish() | ||||
|         } | ||||
|  | ||||
|         currentItem = intent.getIntExtra("currentItem", 0) | ||||
|  | ||||
|         try { | ||||
|             readItem(allItems[currentItem]) | ||||
|         } catch (e: IndexOutOfBoundsException) { | ||||
|             finish() | ||||
|         } | ||||
|  | ||||
|         binding.pager.adapter = ScreenSlidePagerAdapter(this) | ||||
|         binding.pager.setCurrentItem(currentItem, false) | ||||
| @@ -89,7 +87,7 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|     } | ||||
|  | ||||
|     private fun readItem(item: SelfossModel.Item) { | ||||
|         if (appSettingsService.isMarkOnScrollEnabled()) { | ||||
|         if (appSettingsService.isMarkOnScrollEnabled() && !appSettingsService.getPublicAccess()) { | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 repository.markAsRead(item) | ||||
|             } | ||||
| @@ -103,15 +101,15 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : | ||||
|         FragmentStateAdapter(fa) { | ||||
|  | ||||
|         override fun getItemCount(): Int = allItems.size | ||||
|  | ||||
|         override fun createFragment(position: Int): Fragment = | ||||
|             ArticleFragment.newInstance(allItems[position]) | ||||
|  | ||||
|         override fun createFragment(position: Int): Fragment = ArticleFragment.newInstance(allItems[position]) | ||||
|     } | ||||
|  | ||||
|     override fun onKeyDown(keyCode: Int, event: KeyEvent?): Boolean { | ||||
|     override fun onKeyDown( | ||||
|         keyCode: Int, | ||||
|         event: KeyEvent?, | ||||
|     ): Boolean { | ||||
|         return when (keyCode) { | ||||
|             KeyEvent.KEYCODE_VOLUME_DOWN -> { | ||||
|                 val currentFragment = | ||||
| @@ -142,16 +140,19 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|         inflater.inflate(R.menu.reader_menu, menu) | ||||
|         toolbarMenu = menu | ||||
|  | ||||
|         alignmentMenu() | ||||
|  | ||||
|         if (appSettingsService.getPublicAccess()) { | ||||
|             menu.removeItem(R.id.star) | ||||
|         } else { | ||||
|             if (allItems.isNotEmpty() && allItems[currentItem].starred) { | ||||
|                 canRemoveFromFavorite() | ||||
|             } else { | ||||
|                 canFavorite() | ||||
|             } | ||||
|         alignmentMenu() | ||||
|  | ||||
|             binding.pager.registerOnPageChangeCallback( | ||||
|                 object : ViewPager2.OnPageChangeCallback() { | ||||
|  | ||||
|                     override fun onPageSelected(position: Int) { | ||||
|                         super.onPageSelected(position) | ||||
|  | ||||
| @@ -162,8 +163,9 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|                         } | ||||
|                         readItem(allItems[position]) | ||||
|                     } | ||||
|             } | ||||
|                 }, | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         return true | ||||
|     } | ||||
| @@ -182,20 +184,18 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|         when (item.itemId) { | ||||
|             android.R.id.home -> { | ||||
|                 onBackPressed() | ||||
|                 onBackPressedDispatcher.onBackPressed() | ||||
|                 return true | ||||
|             } | ||||
|             R.id.star -> { | ||||
|                 if (allItems[binding.pager.currentItem].starred) { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.unstarr(allItems[binding.pager.currentItem]) | ||||
|                         // TODO: Handle failure | ||||
|                     } | ||||
|                     afterUnsave() | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.starr(allItems[binding.pager.currentItem]) | ||||
|                         // TODO: Handle failure | ||||
|                     } | ||||
|                     afterSave() | ||||
|                 } | ||||
| @@ -223,8 +223,4 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|         startActivity(intent) | ||||
|         overridePendingTransition(0, 0) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         var allItems: ArrayList<SelfossModel.Item> = ArrayList() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -8,11 +8,9 @@ import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.recyclerview.widget.LinearLayoutManager | ||||
| import bou.amine.apps.readerforselfossv2.android.adapters.SourcesListAdapter | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySourcesBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.android.testing.CountingIdlingResourceSingleton | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import com.ftinc.scoop.Scoop | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -21,22 +19,15 @@ import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var binding: ActivitySourcesBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(this@SourcesActivity) | ||||
|         binding = ActivitySourcesBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolbar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|  | ||||
|         setContentView(view) | ||||
| @@ -45,8 +36,9 @@ class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         binding.fab.rippleColor = appColors.colorAccentDark | ||||
|         binding.fab.backgroundTintList = ColorStateList.valueOf(appColors.colorAccent) | ||||
|         binding.fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||
|         binding.fab.backgroundTintList = | ||||
|             ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||
|     } | ||||
|  | ||||
|     override fun onStop() { | ||||
| @@ -58,38 +50,35 @@ class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|         super.onResume() | ||||
|         val mLayoutManager = LinearLayoutManager(this) | ||||
|  | ||||
|         var items: ArrayList<SelfossModel.Source> | ||||
|         var items: ArrayList<SelfossModel.SourceDetail> | ||||
|  | ||||
|         binding.recyclerView.setHasFixedSize(true) | ||||
|         binding.recyclerView.layoutManager = mLayoutManager | ||||
|  | ||||
|         CountingIdlingResourceSingleton.increment() | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             val response = repository.getSources() | ||||
|             if (response != null) { | ||||
|             val response = repository.getSourcesDetails() | ||||
|             if (response.isNotEmpty()) { | ||||
|                 items = response | ||||
|                 val mAdapter = SourcesListAdapter( | ||||
|                     this@SourcesActivity, items | ||||
|                 val mAdapter = | ||||
|                     SourcesListAdapter( | ||||
|                         this@SourcesActivity, | ||||
|                         items, | ||||
|                     ) | ||||
|                 binding.recyclerView.adapter = mAdapter | ||||
|                 mAdapter.notifyDataSetChanged() | ||||
|                 if (items.isEmpty()) { | ||||
|                     Toast.makeText( | ||||
|                         this@SourcesActivity, | ||||
|                         R.string.nothing_here, | ||||
|                         Toast.LENGTH_SHORT | ||||
|                     ).show() | ||||
|                 } | ||||
|             } else { | ||||
|                 Toast.makeText( | ||||
|                     this@SourcesActivity, | ||||
|                     R.string.cant_get_sources, | ||||
|                     Toast.LENGTH_SHORT | ||||
|                     Toast.LENGTH_SHORT, | ||||
|                 ).show() | ||||
|             } | ||||
|             CountingIdlingResourceSingleton.decrement() | ||||
|         } | ||||
|  | ||||
|         binding.fab.setOnClickListener { | ||||
|             startActivity(Intent(this@SourcesActivity, AddSourceActivity::class.java)) | ||||
|             startActivity(Intent(this@SourcesActivity, UpsertSourceActivity::class.java)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,210 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Intent | ||||
| import android.os.Bundle | ||||
| import android.view.View | ||||
| import android.widget.AdapterView | ||||
| import android.widget.ArrayAdapter | ||||
| import android.widget.TextView | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityUpsertSourceBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | ||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class UpsertSourceActivity : AppCompatActivity(), DIAware { | ||||
|     private var existingSource: SelfossModel.SourceDetail? = null | ||||
|     private var mSpoutsValue: String? = null | ||||
|  | ||||
|     private lateinit var binding: ActivityUpsertSourceBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityUpsertSourceBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         existingSource = repository.getSelectedSource() | ||||
|         if (existingSource != null) { | ||||
|             binding.formContainer.visibility = View.GONE | ||||
|             binding.progress.visibility = View.VISIBLE | ||||
|         } | ||||
|         val title = if (existingSource == null) R.string.add_source else R.string.update_source | ||||
|  | ||||
|         supportFragmentManager.addOnBackStackChangedListener { | ||||
|             if (supportFragmentManager.backStackEntryCount == 0) { | ||||
|                 setTitle(title) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         setSupportActionBar(binding.toolbar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|         supportActionBar?.title = resources.getString(title) | ||||
|  | ||||
|         maybeGetDetailsFromIntentSharing(intent) | ||||
|  | ||||
|         binding.saveBtn.setOnClickListener { | ||||
|             handleSaveSource() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun initFields(items: Map<String, SelfossModel.Spout>) { | ||||
|         binding.nameInput.setText(existingSource!!.title) | ||||
|         binding.tags.setText(existingSource!!.tags?.joinToString(", ")) | ||||
|         binding.sourceUri.setText(existingSource!!.params?.url) | ||||
|         binding.spoutsSpinner.setSelection(items.keys.indexOf(existingSource!!.spout)) | ||||
|         binding.progress.visibility = View.GONE | ||||
|         binding.formContainer.visibility = View.VISIBLE | ||||
|     } | ||||
|  | ||||
|     override fun onResume() { | ||||
|         super.onResume() | ||||
|  | ||||
|         val baseUrl = appSettingsService.getBaseUrl() | ||||
|         if (baseUrl.isEmpty() || baseUrl.isBaseUrlInvalid()) { | ||||
|             mustLoginToAddSource() | ||||
|         } else { | ||||
|             handleSpoutsSpinner() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleSpoutsSpinner() { | ||||
|         val spoutsKV = HashMap<String, String>() | ||||
|         binding.spoutsSpinner.onItemSelectedListener = | ||||
|             object : AdapterView.OnItemSelectedListener { | ||||
|                 override fun onItemSelected( | ||||
|                     adapterView: AdapterView<*>, | ||||
|                     view: View?, | ||||
|                     i: Int, | ||||
|                     l: Long, | ||||
|                 ) { | ||||
|                     if (view != null) { | ||||
|                         val spoutName = (view as TextView).text.toString() | ||||
|                         mSpoutsValue = spoutsKV[spoutName] | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 override fun onNothingSelected(adapterView: AdapterView<*>) { | ||||
|                     mSpoutsValue = null | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|         fun handleSpoutFailure(networkIssue: Boolean = false) { | ||||
|             Toast.makeText( | ||||
|                 this@UpsertSourceActivity, | ||||
|                 if (networkIssue) R.string.cant_get_spouts_no_network else R.string.cant_get_spouts, | ||||
|                 Toast.LENGTH_SHORT, | ||||
|             ).show() | ||||
|             binding.progress.visibility = View.GONE | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             try { | ||||
|                 val items = repository.getSpouts() | ||||
|                 if (items.isNotEmpty()) { | ||||
|                     val itemsStrings = items.map { it.value.name } | ||||
|                     for ((key, value) in items) { | ||||
|                         spoutsKV[value.name] = key | ||||
|                     } | ||||
|  | ||||
|                     binding.progress.visibility = View.GONE | ||||
|                     binding.formContainer.visibility = View.VISIBLE | ||||
|  | ||||
|                     val spinnerArrayAdapter = | ||||
|                         ArrayAdapter( | ||||
|                             this@UpsertSourceActivity, | ||||
|                             android.R.layout.simple_spinner_item, | ||||
|                             itemsStrings, | ||||
|                         ) | ||||
|                     spinnerArrayAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item) | ||||
|                     binding.spoutsSpinner.adapter = spinnerArrayAdapter | ||||
|  | ||||
|                     if (existingSource != null) { | ||||
|                         initFields(items) | ||||
|                     } | ||||
|                 } else { | ||||
|                     handleSpoutFailure() | ||||
|                 } | ||||
|             } catch (e: NetworkUnavailableException) { | ||||
|                 handleSpoutFailure(networkIssue = true) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun maybeGetDetailsFromIntentSharing(intent: Intent) { | ||||
|         if (Intent.ACTION_SEND == intent.action && "text/plain" == intent.type) { | ||||
|             binding.sourceUri.setText(intent.getStringExtra(Intent.EXTRA_TEXT)) | ||||
|             binding.nameInput.setText(intent.getStringExtra(Intent.EXTRA_TITLE)) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun mustLoginToAddSource() { | ||||
|         Toast.makeText(this, getString(R.string.addStringNoUrl), Toast.LENGTH_SHORT).show() | ||||
|         val i = Intent(this, LoginActivity::class.java) | ||||
|         startActivity(i) | ||||
|         finish() | ||||
|     } | ||||
|  | ||||
|     private fun handleSaveSource() { | ||||
|         val url = binding.sourceUri.text.toString() | ||||
|  | ||||
|         val sourceDetailsUnavailable = | ||||
|             title.isEmpty() || url.isEmpty() || mSpoutsValue == null || mSpoutsValue!!.isEmpty() | ||||
|  | ||||
|         when { | ||||
|             sourceDetailsUnavailable -> { | ||||
|                 Toast.makeText(this, R.string.form_not_complete, Toast.LENGTH_SHORT).show() | ||||
|             } | ||||
|             else -> { | ||||
|                 CoroutineScope(Dispatchers.Main).launch { | ||||
|                     val successfullyAddedSource = | ||||
|                         if (existingSource != null) { | ||||
|                             repository.updateSource( | ||||
|                                 existingSource!!.id, | ||||
|                                 binding.nameInput.text.toString(), | ||||
|                                 url, | ||||
|                                 mSpoutsValue!!, | ||||
|                                 binding.tags.text.toString(), | ||||
|                             ) | ||||
|                         } else { | ||||
|                             repository.createSource( | ||||
|                                 binding.nameInput.text.toString(), | ||||
|                                 url, | ||||
|                                 mSpoutsValue!!, | ||||
|                                 binding.tags.text.toString(), | ||||
|                             ) | ||||
|                         } | ||||
|                     if (successfullyAddedSource) { | ||||
|                         finish() | ||||
|                     } else { | ||||
|                         Toast.makeText( | ||||
|                             this@UpsertSourceActivity, | ||||
|                             R.string.cant_create_source, | ||||
|                             Toast.LENGTH_SHORT, | ||||
|                         ).show() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onDestroy() { | ||||
|         super.onDestroy() | ||||
|         repository.unsetSelectedSource() | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,6 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.adapters | ||||
|  | ||||
| import android.app.Activity | ||||
| import android.content.Context | ||||
| import android.view.LayoutInflater | ||||
| import android.view.View | ||||
| import android.view.ViewGroup | ||||
| @@ -9,20 +8,18 @@ import android.widget.ImageView.ScaleType | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.* | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrlInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||
| import bou.amine.apps.readerforselfossv2.utils.getThumbnail | ||||
| import com.amulyakhare.textdrawable.TextDrawable | ||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | ||||
| import com.bumptech.glide.Glide | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| @@ -33,13 +30,10 @@ import org.kodein.di.instance | ||||
|  | ||||
| class ItemCardAdapter( | ||||
|     override val app: Activity, | ||||
|     override var items: ArrayList<SelfossModel.Item>, | ||||
|     private val helper: CustomTabActivityHelper, | ||||
|     override val appColors: AppColors, | ||||
|     override val updateItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
|     override val items: ArrayList<SelfossModel.Item>, | ||||
|     override val updateHomeItems: (ArrayList<SelfossModel.Item>) -> Unit, | ||||
| ) : ItemsAdapter<ItemCardAdapter.ViewHolder>() { | ||||
|     private val c: Context = app.baseContext | ||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||
|     override lateinit var binding: CardItemBinding | ||||
|     private val imageMaxHeight: Int = | ||||
|         c.resources.getDimension(R.dimen.card_image_max_height).toInt() | ||||
|  | ||||
| @@ -47,23 +41,67 @@ class ItemCardAdapter( | ||||
|     override val repository: Repository by instance() | ||||
|     override val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||
|         val binding = CardItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|     override fun onCreateViewHolder( | ||||
|         parent: ViewGroup, | ||||
|         viewType: Int, | ||||
|     ): ViewHolder { | ||||
|         binding = CardItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|         return ViewHolder(binding) | ||||
|     } | ||||
|  | ||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { | ||||
|     private fun handleClickListeners(holderBinding: CardItemBinding, position: Int) { | ||||
|         holderBinding.favButton.setOnClickListener { | ||||
|             val item = items[position] | ||||
|             if (item.starred) { | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     repository.unstarr(item) | ||||
|                 } | ||||
|                 binding.favButton.isSelected = false | ||||
|             } else { | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     repository.starr(item) | ||||
|                 } | ||||
|                 binding.favButton.isSelected = true | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         binding.shareBtn.setOnClickListener { | ||||
|             val item = items[position] | ||||
|             c.shareLink(item.getLinkDecoded(), item.title.getHtmlDecoded()) | ||||
|         } | ||||
|  | ||||
|         binding.browserBtn.setOnClickListener { | ||||
|             c.openItemUrlInBrowserAsNewTask(items[position]) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onBindViewHolder( | ||||
|         holder: ViewHolder, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         with(holder) { | ||||
|             val itm = items[position] | ||||
|  | ||||
|             handleClickListeners(binding, position) | ||||
|             handleLinkOpening(binding, position) | ||||
|  | ||||
|             binding.favButton.isSelected = itm.starred | ||||
|             if (appSettingsService.getPublicAccess()) { | ||||
|                 binding.favButton.visibility = View.GONE | ||||
|             } | ||||
|  | ||||
|             binding.title.text = itm.title.getHtmlDecoded() | ||||
|  | ||||
|             binding.title.setOnTouchListener(LinkOnTouchListener()) | ||||
|  | ||||
|             binding.title.setLinkTextColor(appColors.colorAccent) | ||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) | ||||
|             binding.sourceTitleAndDate.text = try { | ||||
|                 itm.sourceAuthorAndDate() | ||||
|             } catch (e: Exception) { | ||||
|                 e.sendSilentlyWithAcraWithName("ItemCardAdapter parse date") | ||||
|                 itm.sourceAuthorOnly() | ||||
|             } | ||||
|  | ||||
|             if (!appSettingsService.isFullHeightCardsEnabled()) { | ||||
|                 binding.itemImage.maxHeight = imageMaxHeight | ||||
| @@ -80,76 +118,12 @@ class ItemCardAdapter( | ||||
|             } | ||||
|  | ||||
|             if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||
|                 val color = generator.getColor(itm.title.getHtmlDecoded()) | ||||
|  | ||||
|                 val drawable = | ||||
|                         TextDrawable | ||||
|                                 .builder() | ||||
|                                 .round() | ||||
|                                 .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) | ||||
|                 binding.sourceImage.setImageDrawable(drawable) | ||||
|                 binding.sourceImage.setBackgroundAndText(itm.sourcetitle.getHtmlDecoded()) | ||||
|             } else { | ||||
|                 c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.sourceImage) | ||||
|                 c.circularDrawable(itm.getIcon(repository.baseUrl), binding.sourceImage) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun getItemCount(): Int { | ||||
|         return items.size | ||||
|     } | ||||
|  | ||||
|     inner class ViewHolder(val binding: CardItemBinding) : RecyclerView.ViewHolder(binding.root) { | ||||
|         init { | ||||
|             handleClickListeners() | ||||
|             handleCustomTabActions() | ||||
|         } | ||||
|  | ||||
|         private fun handleClickListeners() { | ||||
|  | ||||
|             binding.favButton.setOnClickListener { | ||||
|                 val item = items[bindingAdapterPosition] | ||||
|                 if (item.starred) { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.unstarr(item) | ||||
|                         // TODO: Handle failure | ||||
|                     } | ||||
|                     item.starred = false | ||||
|                     binding.favButton.isSelected = false | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.starr(item) | ||||
|                         // TODO: Handle failure | ||||
|                     } | ||||
|                     item.starred = true | ||||
|                     binding.favButton.isSelected = true | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|                 binding.shareBtn.setOnClickListener { | ||||
|                     val item = items[bindingAdapterPosition] | ||||
|                     c.shareLink(item.getLinkDecoded(), item.title.getHtmlDecoded()) | ||||
|                 } | ||||
|  | ||||
|                 binding.browserBtn.setOnClickListener { | ||||
|                     c.openInBrowserAsNewTask(items[bindingAdapterPosition]) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|         private fun handleCustomTabActions() { | ||||
|             val customTabsIntent = c.buildCustomTabsIntent() | ||||
|             helper.bindCustomTabsService(app) | ||||
|  | ||||
|             binding.root.setOnClickListener { | ||||
|                 c.openItemUrl( | ||||
|                     items, | ||||
|                     bindingAdapterPosition, | ||||
|                     items[bindingAdapterPosition].getLinkDecoded(), | ||||
|                     customTabsIntent, | ||||
|                     appSettingsService.isInternalBrowserEnabled(), | ||||
|                     appSettingsService.isArticleViewerEnabled(), | ||||
|                     app | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     inner class ViewHolder(val binding: CardItemBinding) : RecyclerView.ViewHolder(binding.root) | ||||
| } | ||||
| @@ -1,106 +1,76 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.adapters | ||||
|  | ||||
| import android.app.Activity | ||||
| import android.content.Context | ||||
| import android.view.LayoutInflater | ||||
| import android.view.ViewGroup | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ListItemBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.buildCustomTabsIntent | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularDrawable | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||
| import bou.amine.apps.readerforselfossv2.utils.getThumbnail | ||||
| import com.amulyakhare.textdrawable.TextDrawable | ||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | ||||
| import org.kodein.di.DI | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class ItemListAdapter( | ||||
|     override val app: Activity, | ||||
|     override var items: ArrayList<SelfossModel.Item>, | ||||
|     private val helper: CustomTabActivityHelper, | ||||
|     override val appColors: AppColors, | ||||
|     override val updateItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
|     override val items: ArrayList<SelfossModel.Item>, | ||||
|     override val updateHomeItems: (ArrayList<SelfossModel.Item>) -> Unit, | ||||
| ) : ItemsAdapter<ItemListAdapter.ViewHolder>() { | ||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||
|     private val c: Context = app.baseContext | ||||
|     override lateinit var binding: ListItemBinding | ||||
|  | ||||
|     override val di: DI by closestDI(app) | ||||
|     override val repository: Repository by instance() | ||||
|     override val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||
|         val binding = ListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|     override fun onCreateViewHolder( | ||||
|         parent: ViewGroup, | ||||
|         viewType: Int, | ||||
|     ): ViewHolder { | ||||
|         binding = ListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|         return ViewHolder(binding) | ||||
|     } | ||||
|  | ||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { | ||||
|     override fun onBindViewHolder( | ||||
|         holder: ViewHolder, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         with(holder) { | ||||
|             val itm = items[position] | ||||
|  | ||||
|             handleLinkOpening(binding, position) | ||||
|  | ||||
|             binding.title.text = itm.title.getHtmlDecoded() | ||||
|  | ||||
|             binding.title.setOnTouchListener(LinkOnTouchListener()) | ||||
|  | ||||
|             binding.title.setLinkTextColor(appColors.colorAccent) | ||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) | ||||
|             binding.sourceTitleAndDate.text = try { | ||||
|                 itm.sourceAuthorAndDate() | ||||
|             } catch (e: Exception) { | ||||
|                 e.sendSilentlyWithAcraWithName("ItemListAdapter parse date") | ||||
|                 itm.sourceAuthorOnly() | ||||
|             } | ||||
|  | ||||
|             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { | ||||
|  | ||||
|                 if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||
|                     val color = generator.getColor(itm.title.getHtmlDecoded()) | ||||
|  | ||||
|                     val drawable = | ||||
|                             TextDrawable | ||||
|                                     .builder() | ||||
|                                     .round() | ||||
|                                     .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) | ||||
|  | ||||
|                     binding.itemImage.setImageDrawable(drawable) | ||||
|                     binding.itemImage.setBackgroundAndText(itm.sourcetitle.getHtmlDecoded()) | ||||
|                 } else { | ||||
|                     c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||
|                     c.circularDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||
|                 } | ||||
|             } else { | ||||
|                 c.bitmapCenterCrop(itm.getThumbnail(repository.baseUrl), binding.itemImage) | ||||
|                 c.circularDrawable(itm.getThumbnail(repository.baseUrl), binding.itemImage) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun getItemCount(): Int = items.size | ||||
|  | ||||
|     inner class ViewHolder(val binding: ListItemBinding) : RecyclerView.ViewHolder(binding.root) { | ||||
|  | ||||
|         init { | ||||
|             handleCustomTabActions() | ||||
|         } | ||||
|  | ||||
|         private fun handleCustomTabActions() { | ||||
|             val customTabsIntent = c.buildCustomTabsIntent() | ||||
|             helper.bindCustomTabsService(app) | ||||
|  | ||||
|             binding.root.setOnClickListener { | ||||
|                 c.openItemUrl( | ||||
|                     items, | ||||
|                     bindingAdapterPosition, | ||||
|                     items[bindingAdapterPosition].getLinkDecoded(), | ||||
|                     customTabsIntent, | ||||
|                     appSettingsService.isInternalBrowserEnabled(), | ||||
|                     appSettingsService.isArticleViewerEnabled(), | ||||
|                     app | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     inner class ViewHolder(val binding: ListItemBinding) : RecyclerView.ViewHolder(binding.root) | ||||
| } | ||||
| @@ -1,11 +1,13 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.adapters | ||||
|  | ||||
| import android.app.Activity | ||||
| import android.content.Context | ||||
| import android.graphics.Color | ||||
| import android.widget.TextView | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import androidx.viewbinding.ViewBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -17,30 +19,35 @@ import kotlinx.coroutines.launch | ||||
| import org.kodein.di.DIAware | ||||
|  | ||||
| abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapter<VH>(), DIAware { | ||||
|     abstract var items: ArrayList<SelfossModel.Item> | ||||
|     abstract val items: ArrayList<SelfossModel.Item> | ||||
|     abstract val repository: Repository | ||||
|     abstract val binding: ViewBinding | ||||
|     abstract val appSettingsService: AppSettingsService | ||||
|     abstract val app: Activity | ||||
|     abstract val appColors: AppColors | ||||
|     abstract val updateItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
|     abstract val updateHomeItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
|  | ||||
|     protected val c: Context get() = app.baseContext | ||||
|  | ||||
|     fun updateAllItems(items: ArrayList<SelfossModel.Item>) { | ||||
|         this.items = items | ||||
|         this.items.clear() | ||||
|         this.items.addAll(items) | ||||
|         updateHomeItems(items) | ||||
|         notifyDataSetChanged() | ||||
|         updateItems(this.items) | ||||
|     } | ||||
|  | ||||
|     private fun unmarkSnackbar(i: SelfossModel.Item, position: Int) { | ||||
|         val s = Snackbar | ||||
|     private fun unmarkSnackbar( | ||||
|         item: SelfossModel.Item, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         val s = | ||||
|             Snackbar | ||||
|                 .make( | ||||
|                     app.findViewById(R.id.coordLayout), | ||||
|                     R.string.marked_as_read, | ||||
|                 Snackbar.LENGTH_LONG | ||||
|                     Snackbar.LENGTH_LONG, | ||||
|                 ) | ||||
|                 .setAction(R.string.undo_string) { | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     unreadItemAtIndex(position, false) | ||||
|                 } | ||||
|                     unreadItemAtIndex(item, position, false) | ||||
|                 } | ||||
|  | ||||
|         val view = s.view | ||||
| @@ -49,15 +56,19 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         s.show() | ||||
|     } | ||||
|  | ||||
|     private fun markSnackbar(position: Int) { | ||||
|         val s = Snackbar | ||||
|     private fun markSnackbar( | ||||
|         item: SelfossModel.Item, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         val s = | ||||
|             Snackbar | ||||
|                 .make( | ||||
|                     app.findViewById(R.id.coordLayout), | ||||
|                     R.string.marked_as_unread, | ||||
|                 Snackbar.LENGTH_LONG | ||||
|                     Snackbar.LENGTH_LONG, | ||||
|                 ) | ||||
|                 .setAction(R.string.undo_string) { | ||||
|                 readItemAtIndex(position) | ||||
|                     readItemAtIndex(item, position, false) | ||||
|                 } | ||||
|  | ||||
|         val view = s.view | ||||
| @@ -66,54 +77,76 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         s.show() | ||||
|     } | ||||
|  | ||||
|     protected fun handleLinkOpening(holderBinding: ViewBinding, position: Int) { | ||||
|         holderBinding.root.setOnClickListener { | ||||
|             repository.setReaderItems(items) | ||||
|             c.openItemUrl( | ||||
|                 position, | ||||
|                 items[position].getLinkDecoded(), | ||||
|                 appSettingsService.isArticleViewerEnabled(), | ||||
|                 app, | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun handleItemAtIndex(position: Int) { | ||||
|         if (items[position].unread) { | ||||
|             readItemAtIndex(position) | ||||
|             readItemAtIndex(items[position], position) | ||||
|         } else { | ||||
|             unreadItemAtIndex(position) | ||||
|             unreadItemAtIndex(items[position], position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun readItemAtIndex(position: Int, showSnackbar: Boolean = true) { | ||||
|         val i = items[position] | ||||
|     private fun readItemAtIndex( | ||||
|         item: SelfossModel.Item, | ||||
|         position: Int, | ||||
|         showSnackbar: Boolean = true, | ||||
|     ) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             repository.markAsRead(i) | ||||
|             repository.markAsRead(item) | ||||
|         } | ||||
|         if (repository.displayedItems == ItemType.UNREAD) { | ||||
|             items.remove(i) | ||||
|             items.remove(item) | ||||
|             notifyItemRemoved(position) | ||||
|             updateItems(items) | ||||
|             notifyItemRangeChanged(position, itemCount) | ||||
|             updateHomeItems(items) | ||||
|         } else { | ||||
|             notifyItemChanged(position) | ||||
|         } | ||||
|         if (showSnackbar) { | ||||
|             unmarkSnackbar(i, position) | ||||
|             unmarkSnackbar(item, position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun unreadItemAtIndex(position: Int, showSnackbar: Boolean = true) { | ||||
|     private fun unreadItemAtIndex( | ||||
|         item: SelfossModel.Item, | ||||
|         position: Int, | ||||
|         showSnackbar: Boolean = true, | ||||
|     ) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             repository.unmarkAsRead(items[position]) | ||||
|  | ||||
|             repository.unmarkAsRead(item) | ||||
|         } | ||||
|         notifyItemChanged(position) | ||||
|         if (showSnackbar) { | ||||
|             markSnackbar(position) | ||||
|             markSnackbar(item, position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun addItemAtIndex(item: SelfossModel.Item, position: Int) { | ||||
|     fun addItemAtIndex( | ||||
|         item: SelfossModel.Item, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         items.add(position, item) | ||||
|         notifyItemInserted(position) | ||||
|         updateItems(items) | ||||
|  | ||||
|         updateHomeItems(items) | ||||
|     } | ||||
|  | ||||
|     fun addItemsAtEnd(newItems: List<SelfossModel.Item>) { | ||||
|         val oldSize = items.size | ||||
|         items.addAll(newItems) | ||||
|         notifyItemRangeInserted(oldSize, newItems.size) | ||||
|         updateItems(items) | ||||
|         updateHomeItems(items) | ||||
|     } | ||||
|  | ||||
|     } | ||||
|     override fun getItemCount(): Int = items.size | ||||
| } | ||||
| @@ -2,22 +2,22 @@ package bou.amine.apps.readerforselfossv2.android.adapters | ||||
|  | ||||
| import android.app.Activity | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.view.LayoutInflater | ||||
| import android.view.View | ||||
| import android.view.ViewGroup | ||||
| import android.widget.Button | ||||
| import android.widget.Toast | ||||
| import androidx.constraintlayout.widget.ConstraintLayout | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.UpsertSourceActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.SourceListItemBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularDrawable | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||
| import com.amulyakhare.textdrawable.TextDrawable | ||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -28,68 +28,76 @@ import org.kodein.di.instance | ||||
|  | ||||
| class SourcesListAdapter( | ||||
|     private val app: Activity, | ||||
|     private val items: ArrayList<SelfossModel.Source> | ||||
|     private val items: ArrayList<SelfossModel.SourceDetail>, | ||||
| ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>(), DIAware { | ||||
|     private val c: Context = app.baseContext | ||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||
|     private lateinit var binding: SourceListItemBinding | ||||
|  | ||||
|     override val di: DI by closestDI(app) | ||||
|     private val repository: Repository by instance() | ||||
|  | ||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||
|     override fun onCreateViewHolder( | ||||
|         parent: ViewGroup, | ||||
|         viewType: Int, | ||||
|     ): ViewHolder { | ||||
|         binding = SourceListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|         return ViewHolder(binding.root) | ||||
|     } | ||||
|  | ||||
|     override fun onBindViewHolder(holder: ViewHolder, position: Int) { | ||||
|     override fun onBindViewHolder( | ||||
|         holder: ViewHolder, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         val itm = items[position] | ||||
|  | ||||
|         if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||
|             val color = generator.getColor(itm.title.getHtmlDecoded()) | ||||
|         val deleteBtn: Button = holder.mView.findViewById(R.id.deleteBtn) | ||||
|  | ||||
|             val drawable = | ||||
|                 TextDrawable | ||||
|                     .builder() | ||||
|                     .round() | ||||
|                     .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) | ||||
|             binding.itemImage.setImageDrawable(drawable) | ||||
|         deleteBtn.setOnClickListener { | ||||
|             val (id, title) = items[position] | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 val successfullyDeletedSource = repository.deleteSource(id, title) | ||||
|                 if (successfullyDeletedSource) { | ||||
|                     items.removeAt(position) | ||||
|                     notifyItemRemoved(position) | ||||
|                     notifyItemRangeChanged(position, itemCount) | ||||
|                 } else { | ||||
|             c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||
|                     Toast.makeText( | ||||
|                         app, | ||||
|                         R.string.can_delete_source, | ||||
|                         Toast.LENGTH_SHORT, | ||||
|                     ).show() | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         holder.mView.setOnClickListener { | ||||
|             val source = items[position] | ||||
|  | ||||
|             repository.setSelectedSource(source) | ||||
|             app.startActivity(Intent(app, UpsertSourceActivity::class.java)) | ||||
|         } | ||||
|  | ||||
|         if (itm.getIcon(repository.baseUrl).isEmpty()) { | ||||
|             binding.itemImage.setBackgroundAndText(itm.title.getHtmlDecoded()) | ||||
|         } else { | ||||
|             c.circularDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||
|         } | ||||
|  | ||||
|         if (!itm.error.isNullOrBlank()) { | ||||
|             binding.errorText.visibility = View.VISIBLE | ||||
|             binding.errorText.text = itm.error | ||||
|         } else { | ||||
|             binding.errorText.visibility = View.GONE | ||||
|         } | ||||
|  | ||||
|         binding.sourceTitle.text = itm.title.getHtmlDecoded() | ||||
|     } | ||||
|  | ||||
|     override fun getItemId(position: Int) = position.toLong() | ||||
|  | ||||
|     override fun getItemViewType(position: Int) = position | ||||
|  | ||||
|     override fun getItemCount(): Int = items.size | ||||
|  | ||||
|     inner class ViewHolder(internal val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { | ||||
|  | ||||
|         init { | ||||
|             handleClickListeners() | ||||
|         } | ||||
|  | ||||
|         private fun handleClickListeners() { | ||||
|  | ||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||
|  | ||||
|             deleteBtn.setOnClickListener { | ||||
|                 val (id) = items[adapterPosition] | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     val successfullyDeletedSource = repository.deleteSource(id) | ||||
|                     if (successfullyDeletedSource) { | ||||
|                         items.removeAt(adapterPosition) | ||||
|                         notifyItemRemoved(adapterPosition) | ||||
|                         notifyItemRangeChanged(adapterPosition, itemCount) | ||||
|                     } else { | ||||
|                         Toast.makeText( | ||||
|                             app, | ||||
|                             R.string.can_delete_source, | ||||
|                             Toast.LENGTH_SHORT | ||||
|                         ).show() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     inner class ViewHolder(val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) | ||||
| } | ||||
|   | ||||
| @@ -1,35 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import com.google.gson.GsonBuilder | ||||
| import okhttp3.OkHttpClient | ||||
| import okhttp3.logging.HttpLoggingInterceptor | ||||
| import retrofit2.Call | ||||
| import retrofit2.Retrofit | ||||
| import retrofit2.converter.gson.GsonConverterFactory | ||||
|  | ||||
| class MercuryApi() { | ||||
|     private val service: MercuryService | ||||
|  | ||||
|     init { | ||||
|  | ||||
|         val interceptor = HttpLoggingInterceptor() | ||||
|         interceptor.level = HttpLoggingInterceptor.Level.NONE | ||||
|         val client = OkHttpClient.Builder().addInterceptor(interceptor).build() | ||||
|  | ||||
|         val gson = GsonBuilder() | ||||
|             .setLenient() | ||||
|             .create() | ||||
|         val retrofit = | ||||
|             Retrofit | ||||
|                 .Builder() | ||||
|                 .baseUrl("https://www.amine-louveau.fr") | ||||
|                 .client(client) | ||||
|                 .addConverterFactory(GsonConverterFactory.create(gson)) | ||||
|                 .build() | ||||
|         service = retrofit.create(MercuryService::class.java) | ||||
|     } | ||||
|  | ||||
|     fun parseUrl(url: String): Call<ParsedContent> { | ||||
|         return service.parseUrl(url) | ||||
|     } | ||||
| } | ||||
| @@ -1,59 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import android.os.Parcel | ||||
| import android.os.Parcelable | ||||
| import com.google.gson.annotations.SerializedName | ||||
|  | ||||
| class ParsedContent( | ||||
|     @SerializedName("title") val title: String, | ||||
|     @SerializedName("content") val content: String?, | ||||
|     @SerializedName("date_published") val date_published: String, | ||||
|     @SerializedName("lead_image_url") val lead_image_url: String?, | ||||
|     @SerializedName("dek") val dek: String, | ||||
|     @SerializedName("url") val url: String, | ||||
|     @SerializedName("domain") val domain: String, | ||||
|     @SerializedName("excerpt") val excerpt: String, | ||||
|     @SerializedName("total_pages") val total_pages: Int, | ||||
|     @SerializedName("rendered_pages") val rendered_pages: Int, | ||||
|     @SerializedName("next_page_url") val next_page_url: String | ||||
| ) : Parcelable { | ||||
|  | ||||
|     companion object { | ||||
|         @JvmField | ||||
|         val CREATOR: Parcelable.Creator<ParsedContent> = | ||||
|             object : Parcelable.Creator<ParsedContent> { | ||||
|                 override fun createFromParcel(source: Parcel): ParsedContent = ParsedContent(source) | ||||
|                 override fun newArray(size: Int): Array<ParsedContent?> = arrayOfNulls(size) | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     constructor(source: Parcel) : this( | ||||
|         title = source.readString().orEmpty(), | ||||
|         content = source.readString(), | ||||
|         date_published = source.readString().orEmpty(), | ||||
|         lead_image_url = source.readString(), | ||||
|         dek = source.readString().orEmpty(), | ||||
|         url = source.readString().orEmpty(), | ||||
|         domain = source.readString().orEmpty(), | ||||
|         excerpt = source.readString().orEmpty(), | ||||
|         total_pages = source.readInt(), | ||||
|         rendered_pages = source.readInt(), | ||||
|         next_page_url = source.readString().orEmpty() | ||||
|     ) | ||||
|  | ||||
|     override fun describeContents() = 0 | ||||
|  | ||||
|     override fun writeToParcel(dest: Parcel, flags: Int) { | ||||
|         dest.writeString(title) | ||||
|         dest.writeString(content) | ||||
|         dest.writeString(date_published) | ||||
|         dest.writeString(lead_image_url) | ||||
|         dest.writeString(dek) | ||||
|         dest.writeString(url) | ||||
|         dest.writeString(domain) | ||||
|         dest.writeString(excerpt) | ||||
|         dest.writeInt(total_pages) | ||||
|         dest.writeInt(rendered_pages) | ||||
|         dest.writeString(next_page_url) | ||||
|     } | ||||
| } | ||||
| @@ -1,10 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import retrofit2.Call | ||||
| import retrofit2.http.GET | ||||
| import retrofit2.http.Query | ||||
|  | ||||
| interface MercuryService { | ||||
|     @GET("parser.php") | ||||
|     fun parseUrl(@Query("link") link: String): Call<ParsedContent> | ||||
| } | ||||
| @@ -26,15 +26,15 @@ import org.kodein.di.instance | ||||
| import java.util.* | ||||
| import kotlin.concurrent.schedule | ||||
|  | ||||
| class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(context, params), DIAware { | ||||
|  | ||||
| class LoadingWorker(val context: Context, params: WorkerParameters) : | ||||
|     Worker(context, params), | ||||
|     DIAware { | ||||
|     override val di by lazy { (applicationContext as MyApp).di } | ||||
|     private val repository: Repository by instance() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun doWork(): Result { | ||||
|         if (appSettingsService.isPeriodicRefreshEnabled() && isNetworkAccessible(context)) { | ||||
|  | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 val notificationManager = | ||||
|                     applicationContext.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager | ||||
| @@ -52,11 +52,13 @@ override fun doWork(): Result { | ||||
|  | ||||
|                 repository.handleDBActions() | ||||
|  | ||||
|                 val apiItems = repository.tryToCacheItemsAndGetNewOnes() | ||||
|                 if (appSettingsService.isNotifyNewItemsEnabled()) { | ||||
|                     launch { | ||||
|                     handleNewItemsNotification(repository.tryToCacheItemsAndGetNewOnes(), notificationManager) | ||||
|                         handleNewItemsNotification(apiItems, notificationManager) | ||||
|                     } | ||||
|                 } | ||||
|                 apiItems.map { it.preloadImages(context) } | ||||
|             } | ||||
|         } | ||||
|         return Result.success() | ||||
| @@ -64,33 +66,37 @@ override fun doWork(): Result { | ||||
|  | ||||
|     private fun handleNewItemsNotification( | ||||
|         newItems: List<SelfossModel.Item>?, | ||||
|         notificationManager: NotificationManager | ||||
|         notificationManager: NotificationManager, | ||||
|     ) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             val apiItems = newItems.orEmpty() | ||||
|  | ||||
|  | ||||
|             val newSize = apiItems.filter { it.unread }.size | ||||
|             if (newSize > 0) { | ||||
|  | ||||
|                     val intent = Intent(context, MainActivity::class.java).apply { | ||||
|                 val intent = | ||||
|                     Intent(context, MainActivity::class.java).apply { | ||||
|                         flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK | ||||
|                     } | ||||
|                     val pflags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|                 val pflags = | ||||
|                     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|                         PendingIntent.FLAG_IMMUTABLE | ||||
|                     } else { | ||||
|                         0 | ||||
|                     } | ||||
|                     val pendingIntent: PendingIntent = PendingIntent.getActivity(context, 0, intent, pflags) | ||||
|                 val pendingIntent: PendingIntent = | ||||
|                     PendingIntent.getActivity(context, 0, intent, pflags) | ||||
|  | ||||
|                 val newItemsNotification = | ||||
|                         NotificationCompat.Builder(applicationContext, AppSettingsService.newItemsChannelId) | ||||
|                     NotificationCompat.Builder( | ||||
|                         applicationContext, | ||||
|                         AppSettingsService.newItemsChannelId, | ||||
|                     ) | ||||
|                         .setContentTitle(context.getString(R.string.new_items_notification_title)) | ||||
|                         .setContentText( | ||||
|                             context.getString( | ||||
|                                 R.string.new_items_notification_text, | ||||
|                                     newSize | ||||
|                                 ) | ||||
|                                 newSize, | ||||
|                             ), | ||||
|                         ) | ||||
|                         .setPriority(PRIORITY_DEFAULT) | ||||
|                         .setChannelId(AppSettingsService.newItemsChannelId) | ||||
| @@ -102,7 +108,6 @@ override fun doWork(): Result { | ||||
|                     notificationManager.notify(2, newItemsNotification.build()) | ||||
|                 } | ||||
|             } | ||||
|                 apiItems.map { it.preloadImages(context) } | ||||
|             Timer("", false).schedule(4000) { | ||||
|                 notificationManager.cancel(1) | ||||
|             } | ||||
|   | ||||
| @@ -6,36 +6,39 @@ import android.content.res.TypedArray | ||||
| import android.graphics.Bitmap | ||||
| import android.graphics.Typeface | ||||
| import android.graphics.drawable.ColorDrawable | ||||
| import android.net.Uri | ||||
| import android.os.Bundle | ||||
| import android.view.* | ||||
| import android.util.TypedValue | ||||
| import android.view.GestureDetector | ||||
| import android.view.InflateException | ||||
| import android.view.LayoutInflater | ||||
| import android.view.MenuItem | ||||
| import android.view.MotionEvent | ||||
| import android.view.View | ||||
| import android.view.ViewGroup | ||||
| import android.webkit.WebResourceResponse | ||||
| import android.webkit.WebSettings | ||||
| import android.webkit.WebView | ||||
| import android.webkit.WebViewClient | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.browser.customtabs.CustomTabsIntent | ||||
| import androidx.core.content.res.ResourcesCompat | ||||
| import androidx.core.widget.NestedScrollView | ||||
| import androidx.fragment.app.Fragment | ||||
| import bou.amine.apps.readerforselfossv2.android.ImageActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.api.mercury.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.android.api.mercury.ParsedContent | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.FragmentArticleBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.ParecelableItem | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toModel | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toParcelable | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.buildCustomTabsIntent | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrlInternalBrowser | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isUrlValid | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrlInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openUrlInBrowser | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||
| import bou.amine.apps.readerforselfossv2.model.MercuryModel | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| @@ -53,18 +56,16 @@ import org.kodein.di.DI | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.x.closestDI | ||||
| import org.kodein.di.instance | ||||
| import retrofit2.Call | ||||
| import retrofit2.Callback | ||||
| import retrofit2.Response | ||||
| import java.net.MalformedURLException | ||||
| import java.net.URL | ||||
| import java.util.* | ||||
| import java.util.Locale | ||||
| import java.util.concurrent.ExecutionException | ||||
|  | ||||
| private const val IMAGE_JPG = "image/jpg" | ||||
|  | ||||
| class ArticleFragment : Fragment(), DIAware { | ||||
|     private var fontSize: Int = 16 | ||||
|     private lateinit var item: SelfossModel.Item | ||||
|     private var mCustomTabActivityHelper: CustomTabActivityHelper? = null | ||||
|     private lateinit var url: String | ||||
|     private lateinit var contentText: String | ||||
|     private lateinit var contentSource: String | ||||
| @@ -72,10 +73,8 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     private lateinit var contentTitle: String | ||||
|     private lateinit var allImages: ArrayList<String> | ||||
|     private lateinit var fab: FloatingActionButton | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var textAlignment: String | ||||
|     private var _binding: FragmentArticleBinding? = null | ||||
|     private val binding get() = _binding!! | ||||
|     private lateinit var binding: FragmentArticleBinding | ||||
|  | ||||
|     override val di: DI by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
| @@ -86,16 +85,9 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     private var font = "" | ||||
|     private var staticBar = false | ||||
|  | ||||
|     override fun onStop() { | ||||
|         super.onStop() | ||||
|         if (mCustomTabActivityHelper != null) { | ||||
|             mCustomTabActivityHelper!!.unbindCustomTabsService(activity) | ||||
|         } | ||||
|     } | ||||
|     private val mercuryApi: MercuryApi by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(requireActivity()) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|  | ||||
|         val pi: ParecelableItem = requireArguments().getParcelable(ARG_ITEMS)!! | ||||
| @@ -106,88 +98,36 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     override fun onCreateView( | ||||
|         inflater: LayoutInflater, | ||||
|         container: ViewGroup?, | ||||
|         savedInstanceState: Bundle? | ||||
|         savedInstanceState: Bundle?, | ||||
|     ): View { | ||||
|         try { | ||||
|             _binding = FragmentArticleBinding.inflate(inflater, container, false) | ||||
|             binding = FragmentArticleBinding.inflate(inflater, container, false) | ||||
|  | ||||
|             url = item.getLinkDecoded() | ||||
|             contentText = item.content | ||||
|             contentTitle = item.title.getHtmlDecoded() | ||||
|             contentImage = item.getThumbnail(repository.baseUrl) | ||||
|             contentSource = item.sourceAndDateText(repository.dateUtils) | ||||
|             contentSource = try { | ||||
|                 item.sourceAuthorAndDate() | ||||
|             } catch (e: Exception) { | ||||
|                 e.sendSilentlyWithAcraWithName("Article Fragment parse date") | ||||
|                 item.sourceAuthorOnly() | ||||
|             } | ||||
|             allImages = item.getImages() | ||||
|  | ||||
|             fontSize = appSettingsService.getFontSize() | ||||
|             staticBar = appSettingsService.isStaticBarEnabled() | ||||
|             font = appSettingsService.getFont() | ||||
|  | ||||
|             if (font.isNotEmpty()) { | ||||
|                 resId = requireContext().resources.getIdentifier(font, "font", requireContext().packageName) | ||||
|                 typeface = try { | ||||
|                     ResourcesCompat.getFont(requireContext(), resId)!! | ||||
|                 } catch (e: java.lang.Exception) { | ||||
|                     // Just to be sure | ||||
|                     null | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             refreshAlignment() | ||||
|  | ||||
|             fab = binding.fab | ||||
|  | ||||
|             fab.backgroundTintList = ColorStateList.valueOf(appColors.colorAccent) | ||||
|             fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             fab.rippleColor = appColors.colorAccentDark | ||||
|             fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||
|  | ||||
|             val floatingToolbar: FloatingToolbar = binding.floatingToolbar | ||||
|             floatingToolbar.attachFab(fab) | ||||
|  | ||||
|             floatingToolbar.background = ColorDrawable(appColors.colorAccent) | ||||
|  | ||||
|             val customTabsIntent = requireActivity().buildCustomTabsIntent() | ||||
|             mCustomTabActivityHelper = CustomTabActivityHelper() | ||||
|             mCustomTabActivityHelper!!.bindCustomTabsService(activity) | ||||
|  | ||||
|  | ||||
|             floatingToolbar.setClickListener( | ||||
|                 object : FloatingToolbar.ItemClickListener { | ||||
|                     override fun onItemClick(item: MenuItem) { | ||||
|                         when (item.itemId) { | ||||
|                             R.id.more_action -> getContentFromMercury(customTabsIntent) | ||||
|                             R.id.share_action -> requireActivity().shareLink(url, contentTitle) | ||||
|                             R.id.open_action -> requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|                             R.id.unread_action -> if (context != null) { | ||||
|                                 if (this@ArticleFragment.item.unread) { | ||||
|                                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                                         repository.markAsRead(this@ArticleFragment.item) | ||||
|                                     } | ||||
|                                     this@ArticleFragment.item.unread = false | ||||
|                                     Toast.makeText( | ||||
|                                         context, | ||||
|                                         R.string.marked_as_read, | ||||
|                                         Toast.LENGTH_LONG | ||||
|                                     ).show() | ||||
|                                 } else { | ||||
|                                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                                         repository.unmarkAsRead(this@ArticleFragment.item) | ||||
|                                     } | ||||
|                                     this@ArticleFragment.item.unread = true | ||||
|                                     Toast.makeText( | ||||
|                                         context, | ||||
|                                         R.string.marked_as_unread, | ||||
|                                         Toast.LENGTH_LONG | ||||
|                                     ).show() | ||||
|                                 } | ||||
|                             } | ||||
|                             else -> Unit | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     override fun onItemLongClick(item: MenuItem?) { | ||||
|                     } | ||||
|                 } | ||||
|             ) | ||||
|             val floatingToolbar: FloatingToolbar = handleFloatingToolbar() | ||||
|  | ||||
|             if (staticBar) { | ||||
|                 fab.hide() | ||||
| @@ -199,8 +139,47 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 binding.source.typeface = typeface | ||||
|             } | ||||
|  | ||||
|             handleContent() | ||||
|  | ||||
|             binding.nestedScrollView.setOnScrollChangeListener( | ||||
|                 NestedScrollView.OnScrollChangeListener { _, _, scrollY, _, oldScrollY -> | ||||
|                     if (scrollY > oldScrollY) { | ||||
|                         floatingToolbar.hide() | ||||
|                         fab.hide() | ||||
|                     } else { | ||||
|                         if (staticBar) { | ||||
|                             floatingToolbar.show() | ||||
|                         } else { | ||||
|                             if (floatingToolbar.isShowing) floatingToolbar.hide() else fab.show() | ||||
|                         } | ||||
|                     } | ||||
|                 }, | ||||
|             ) | ||||
|         } catch (e: InflateException) { | ||||
|             e.sendSilentlyWithAcraWithName("webview not available") | ||||
|             if (context != null) { | ||||
|                 AlertDialog.Builder(requireContext()) | ||||
|                     .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) | ||||
|                     .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) | ||||
|                     .setPositiveButton( | ||||
|                         android.R.string.ok, | ||||
|                     ) { _, _ -> | ||||
|                         appSettingsService.disableArticleViewer() | ||||
|                         requireActivity().finish() | ||||
|                     } | ||||
|                     .create() | ||||
|                     .show() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return binding.root | ||||
|     } | ||||
|  | ||||
|     private fun handleContent() { | ||||
|         if (contentText.isEmptyOrNullOrNullString()) { | ||||
|                 getContentFromMercury(customTabsIntent) | ||||
|             if (repository.isNetworkAvailable()) { | ||||
|                 getContentFromMercury() | ||||
|             } | ||||
|         } else { | ||||
|             binding.titleView.text = contentTitle | ||||
|             if (typeface != null) { | ||||
| @@ -221,198 +200,223 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 binding.imageView.visibility = View.GONE | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|             binding.nestedScrollView.setOnScrollChangeListener( | ||||
|                 NestedScrollView.OnScrollChangeListener { _, _, scrollY, _, oldScrollY -> | ||||
|                     if (scrollY > oldScrollY) { | ||||
|                         floatingToolbar.hide() | ||||
|                         fab.hide() | ||||
|     private fun handleFloatingToolbar(): FloatingToolbar { | ||||
|         val floatingToolbar: FloatingToolbar = binding.floatingToolbar | ||||
|         if (appSettingsService.getPublicAccess()) { | ||||
|             floatingToolbar.setMenu(R.menu.reader_toolbar_no_read) | ||||
|         } | ||||
|         floatingToolbar.attachFab(fab) | ||||
|  | ||||
|         floatingToolbar.background = ColorDrawable(resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|         floatingToolbar.setClickListener( | ||||
|             object : FloatingToolbar.ItemClickListener { | ||||
|                 override fun onItemClick(item: MenuItem) { | ||||
|                     when (item.itemId) { | ||||
|                         R.id.share_action -> requireActivity().shareLink(url, contentTitle) | ||||
|                         R.id.open_action -> requireActivity().openItemUrlInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|                         R.id.unread_action -> | ||||
|                             if (context != null) { | ||||
|                                 if (this@ArticleFragment.item.unread) { | ||||
|                                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                                         repository.markAsRead(this@ArticleFragment.item) | ||||
|                                     } | ||||
|                                     this@ArticleFragment.item.unread = false | ||||
|                                     Toast.makeText( | ||||
|                                         context, | ||||
|                                         R.string.marked_as_read, | ||||
|                                         Toast.LENGTH_LONG, | ||||
|                                     ).show() | ||||
|                                 } else { | ||||
|                         if (staticBar) { | ||||
|                             floatingToolbar.show() | ||||
|                         } else { | ||||
|                             if (floatingToolbar.isShowing) floatingToolbar.hide() else fab.show() | ||||
|                                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                                         repository.unmarkAsRead(this@ArticleFragment.item) | ||||
|                                     } | ||||
|                                     this@ArticleFragment.item.unread = true | ||||
|                                     Toast.makeText( | ||||
|                                         context, | ||||
|                                         R.string.marked_as_unread, | ||||
|                                         Toast.LENGTH_LONG, | ||||
|                                     ).show() | ||||
|                                 } | ||||
|                             } | ||||
|  | ||||
|                         else -> Unit | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 override fun onItemLongClick(item: MenuItem?) { | ||||
|                     // We do nothing | ||||
|                 } | ||||
|             }, | ||||
|         ) | ||||
|  | ||||
|         } catch (e: InflateException) { | ||||
|             AlertDialog.Builder(requireContext()) | ||||
|                 .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) | ||||
|                 .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) | ||||
|                 .setPositiveButton(android.R.string.ok | ||||
|                 ) { _, _ -> | ||||
|                     appSettingsService.disableArticleViewer() | ||||
|                     requireActivity().finish() | ||||
|                 } | ||||
|                 .create() | ||||
|                 .show() | ||||
|         } | ||||
|  | ||||
|         return binding.root | ||||
|     } | ||||
|  | ||||
|     override fun onDestroyView() { | ||||
|         super.onDestroyView() | ||||
|         _binding = null | ||||
|         return floatingToolbar | ||||
|     } | ||||
|  | ||||
|     private fun refreshAlignment() { | ||||
|         textAlignment = when (appSettingsService.getActiveAllignment()) { | ||||
|         textAlignment = | ||||
|             when (appSettingsService.getActiveAllignment()) { | ||||
|                 1 -> "justify" | ||||
|                 2 -> "left" | ||||
|                 else -> "justify" | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     private fun getContentFromMercury(customTabsIntent: CustomTabsIntent) { | ||||
|         if (repository.isNetworkAvailable()) { | ||||
|     private fun getContentFromMercury() { | ||||
|         binding.progressBar.visibility = View.VISIBLE | ||||
|         val parser = MercuryApi() | ||||
|  | ||||
|             parser.parseUrl(url).enqueue( | ||||
|                 object : Callback<ParsedContent> { | ||||
|                     override fun onResponse( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         response: Response<ParsedContent> | ||||
|                     ) { | ||||
|                         // TODO: clean all the following after finding the mercury content issue | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             try { | ||||
|                             if (response.body() != null && response.body()!!.content != null && !response.body()!!.content.isNullOrEmpty()) { | ||||
|                                 try { | ||||
|                                     binding.titleView.text = response.body()!!.title | ||||
|                 val response = mercuryApi.query(url) | ||||
|                 if (response.success && response.data != null) { | ||||
|                     handleMercuryData(response.data!!) | ||||
|                 } else { | ||||
|                     openInBrowserAfterFailing() | ||||
|                 } | ||||
|             } catch (e: Exception) { | ||||
|                 openInBrowserAfterFailing() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleMercuryData(data: MercuryModel.ParsedContent) { | ||||
|         if (data.error == true || data.failed == true) { | ||||
|             openInBrowserAfterFailing() | ||||
|         } else { | ||||
|             binding.titleView.text = data.title.orEmpty() | ||||
|             if (typeface != null) { | ||||
|                 binding.titleView.typeface = typeface | ||||
|             } | ||||
|                                     try { | ||||
|                                         // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||
|                                         URL(response.body()!!.url) | ||||
|                                         url = response.body()!!.url | ||||
|                                     } catch (e: MalformedURLException) { | ||||
|                                         // Mercury returned a relative url. We do nothing. | ||||
|                                     } | ||||
|                                 } catch (e: Exception) { | ||||
|                                 } | ||||
|             URL(data.url) | ||||
|             url = data.url!! | ||||
|  | ||||
|                                 try { | ||||
|                                     contentText = response.body()!!.content.orEmpty() | ||||
|             contentText = data.content.orEmpty() | ||||
|             htmlToWebview() | ||||
|                                 } catch (e: Exception) { | ||||
|  | ||||
|             handleLeadImage(data.lead_image_url) | ||||
|  | ||||
|             binding.nestedScrollView.scrollTo(0, 0) | ||||
|             binding.progressBar.visibility = View.GONE | ||||
|         } | ||||
|     } | ||||
|  | ||||
|                                 try { | ||||
|                                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||
|     private fun handleLeadImage(lead_image_url: String?) { | ||||
|         if (!lead_image_url.isNullOrEmpty() && context != null) { | ||||
|             binding.imageView.visibility = View.VISIBLE | ||||
|                                         try { | ||||
|             Glide | ||||
|                 .with(requireContext()) | ||||
|                 .asBitmap() | ||||
|                 .load( | ||||
|                                                     response.body()!!.lead_image_url.orEmpty() | ||||
|                     lead_image_url, | ||||
|                 ) | ||||
|                 .apply(RequestOptions.fitCenterTransform()) | ||||
|                 .into(binding.imageView) | ||||
|                                         } catch (e: Exception) { | ||||
|                                         } | ||||
|         } else { | ||||
|             binding.imageView.visibility = View.GONE | ||||
|         } | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|     } | ||||
|  | ||||
|                                 try { | ||||
|                                     binding.nestedScrollView.scrollTo(0, 0) | ||||
|  | ||||
|                                     binding.progressBar.visibility = View.GONE | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|     private fun handleImageLoading() { | ||||
|         binding.webcontent.webViewClient = | ||||
|             object : WebViewClient() { | ||||
|                 @Deprecated("Deprecated in Java") | ||||
|                 override fun shouldOverrideUrlLoading( | ||||
|                     view: WebView?, | ||||
|                     url: String, | ||||
|                 ): Boolean { | ||||
|                     return if (context != null && url.isUrlValid() && binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||
|                         requireContext().openUrlInBrowser(url) | ||||
|                         true | ||||
|                     } else { | ||||
|                                 try { | ||||
|                                     openInBrowserAfterFailing(customTabsIntent) | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|                             } | ||||
|                         } catch (e: Exception) { | ||||
|                             if (context != null) { | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     override fun onFailure( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         t: Throwable | ||||
|                     ) = openInBrowserAfterFailing(customTabsIntent) | ||||
|                 } | ||||
|             ) | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|     private fun htmlToWebview() { | ||||
|         val stringColor = String.format("#%06X", 0xFFFFFF and appColors.colorAccent) | ||||
|  | ||||
|         val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||
|         val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) | ||||
|  | ||||
|  | ||||
|         binding.webcontent.settings.standardFontFamily = a.getString(0) | ||||
|         binding.webcontent.visibility = View.VISIBLE | ||||
|  | ||||
|         // TODO: Set the color strings programmatically | ||||
|         val (stringTextColor, stringBackgroundColor) = if (appColors.isDarkTheme) { | ||||
|             Pair("#FFFFFF", "#303030") | ||||
|         } else { | ||||
|             Pair("#212121", "#FAFAFA") | ||||
|         } | ||||
|  | ||||
|         binding.webcontent.settings.useWideViewPort = true | ||||
|         binding.webcontent.settings.loadWithOverviewMode = true | ||||
|         binding.webcontent.settings.javaScriptEnabled = false | ||||
|  | ||||
|         binding.webcontent.webViewClient = object : WebViewClient() { | ||||
|             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { | ||||
|                 if (binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||
|                     requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||
|                 } | ||||
|                 return true | ||||
|             } | ||||
|  | ||||
|             override fun shouldInterceptRequest(view: WebView?, url: String): WebResourceResponse? { | ||||
|                 @Deprecated("Deprecated in Java") | ||||
|                 override fun shouldInterceptRequest( | ||||
|                     view: WebView, | ||||
|                     url: String, | ||||
|                 ): WebResourceResponse? { | ||||
|                     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||
|                 if (url.lowercase(Locale.US).contains(".jpg") || url.lowercase(Locale.US).contains(".jpeg")) { | ||||
|                     if (url.lowercase(Locale.US).contains(".jpg") || | ||||
|                         url.lowercase(Locale.US) | ||||
|                             .contains(".jpeg") | ||||
|                     ) { | ||||
|                         try { | ||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.JPEG)) | ||||
|                     }catch ( e : ExecutionException) {} | ||||
|                             val image = | ||||
|                                 Glide.with(view).asBitmap().apply(glideOptions).load(url).submit() | ||||
|                                     .get() | ||||
|                             return WebResourceResponse( | ||||
|                                 IMAGE_JPG, | ||||
|                                 "UTF-8", | ||||
|                                 getBitmapInputStream(image, Bitmap.CompressFormat.JPEG), | ||||
|                             ) | ||||
|                         } catch (e: ExecutionException) { | ||||
|                             // Do nothing | ||||
|                         } | ||||
|                 else if (url.lowercase(Locale.US).contains(".png")) { | ||||
|                     } else if (url.lowercase(Locale.US).contains(".png")) { | ||||
|                         try { | ||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.PNG)) | ||||
|                     }catch ( e : ExecutionException) {} | ||||
|                             val image = | ||||
|                                 Glide.with(view).asBitmap().apply(glideOptions).load(url).submit() | ||||
|                                     .get() | ||||
|                             return WebResourceResponse( | ||||
|                                 IMAGE_JPG, | ||||
|                                 "UTF-8", | ||||
|                                 getBitmapInputStream(image, Bitmap.CompressFormat.PNG), | ||||
|                             ) | ||||
|                         } catch (e: ExecutionException) { | ||||
|                             // Do nothing | ||||
|                         } | ||||
|                 else if (url.lowercase(Locale.US).contains(".webp")) { | ||||
|                     } else if (url.lowercase(Locale.US).contains(".webp")) { | ||||
|                         try { | ||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.WEBP)) | ||||
|                     }catch ( e : ExecutionException) {} | ||||
|                             val image = | ||||
|                                 Glide.with(view).asBitmap().apply(glideOptions).load(url).submit() | ||||
|                                     .get() | ||||
|                             return WebResourceResponse( | ||||
|                                 IMAGE_JPG, | ||||
|                                 "UTF-8", | ||||
|                                 getBitmapInputStream(image, Bitmap.CompressFormat.WEBP), | ||||
|                             ) | ||||
|                         } catch (e: ExecutionException) { | ||||
|                             // Do nothing | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     return super.shouldInterceptRequest(view, url) | ||||
|                 } | ||||
|             } | ||||
|     } | ||||
|  | ||||
|         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||
|             override fun onSingleTapUp(e: MotionEvent?): Boolean { | ||||
|     private fun htmlToWebview() { | ||||
|         if (context != null) { | ||||
|             val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||
|             val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) | ||||
|  | ||||
|             binding.webcontent.settings.standardFontFamily = a.getString(0) | ||||
|             binding.webcontent.visibility = View.VISIBLE | ||||
|  | ||||
|             val colorOnSurface = TypedValue() | ||||
|             requireContext().theme.resolveAttribute(R.attr.colorOnSurface, colorOnSurface, true) | ||||
|  | ||||
|             val colorSurface = TypedValue() | ||||
|             requireContext().theme.resolveAttribute(R.attr.colorSurface, colorSurface, true) | ||||
|  | ||||
|             binding.webcontent.settings.useWideViewPort = true | ||||
|             binding.webcontent.settings.loadWithOverviewMode = true | ||||
|             binding.webcontent.settings.javaScriptEnabled = false | ||||
|  | ||||
|             handleImageLoading() | ||||
|  | ||||
|             val gestureDetector = | ||||
|                 GestureDetector( | ||||
|                     activity, | ||||
|                     object : GestureDetector.SimpleOnGestureListener() { | ||||
|                         override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||
|                             return performClick() | ||||
|                         } | ||||
|         }) | ||||
|                     }, | ||||
|                 ) | ||||
|  | ||||
|             binding.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event) } | ||||
|  | ||||
| @@ -425,16 +429,25 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 val itemUrl = URL(url) | ||||
|                 baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||
|             } catch (e: MalformedURLException) { | ||||
|                 e.sendSilentlyWithAcraWithName("htmlToWebview > $url") | ||||
|             } | ||||
|  | ||||
|         val fontName =  when (font) { | ||||
|             val fontName = | ||||
|                 when (font) { | ||||
|                     getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|                     getString(R.string.roboto_font_id) -> "Roboto" | ||||
|                     getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||
|                     else -> "" | ||||
|                 } | ||||
|  | ||||
|         val fontLinkAndStyle = if (font.isNotEmpty()) { | ||||
|             """<link href="https://fonts.googleapis.com/css?family=${fontName.replace(" ", "+")}" rel="stylesheet"> | ||||
|             val fontLinkAndStyle = | ||||
|                 if (font.isNotEmpty()) { | ||||
|                     """<link href="https://fonts.googleapis.com/css?family=${ | ||||
|                         fontName.replace( | ||||
|                             " ", | ||||
|                             "+", | ||||
|                         ) | ||||
|                     }" rel="stylesheet"> | ||||
|                 |<style> | ||||
|                 |   * { | ||||
|                 |       font-family: '$fontName'; | ||||
| @@ -458,10 +471,15 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |        max-width: 100%; | ||||
|                 |      } | ||||
|                 |      a { | ||||
|                 |        color: $stringColor !important; | ||||
|                 |        color: ${ | ||||
|                     String.format( | ||||
|                         "#%06X", | ||||
|                         0xFFFFFF and resources.getColor(R.color.colorAccent), | ||||
|                     ) | ||||
|                 } !important; | ||||
|                 |      } | ||||
|                 |      *:not(a) { | ||||
|                 |        color: $stringTextColor; | ||||
|                 |        color: ${String.format("#%06X", 0xFFFFFF and colorOnSurface.data)}; | ||||
|                 |      } | ||||
|                 |      * { | ||||
|                 |        font-size: ${fontSize}px; | ||||
| @@ -469,11 +487,26 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |        word-break: break-word; | ||||
|                 |        overflow:hidden; | ||||
|                 |        line-height: 1.5em; | ||||
|                 |        background-color: $stringBackgroundColor; | ||||
|                 |        background-color: ${ | ||||
|                     String.format( | ||||
|                         "#%06X", | ||||
|                         0xFFFFFF and colorSurface.data, | ||||
|                     ) | ||||
|                 }; | ||||
|                 |      } | ||||
|                 |      body, html { | ||||
|                 |        background-color: $stringBackgroundColor !important; | ||||
|                 |        border-color: $stringBackgroundColor  !important; | ||||
|                 |        background-color: ${ | ||||
|                     String.format( | ||||
|                         "#%06X", | ||||
|                         0xFFFFFF and colorSurface.data, | ||||
|                     ) | ||||
|                 } !important; | ||||
|                 |        border-color: ${ | ||||
|                     String.format( | ||||
|                         "#%06X", | ||||
|                         0xFFFFFF and colorSurface.data, | ||||
|                     ) | ||||
|                 }  !important; | ||||
|                 |        padding: 0 !important; | ||||
|                 |        margin: 0 !important; | ||||
|                 |      } | ||||
| @@ -483,19 +516,26 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |      pre, code { | ||||
|                 |        white-space: pre-wrap; | ||||
|                 |        width:100%; | ||||
|                 |        background-color: $stringBackgroundColor; | ||||
|                 |        background-color: ${ | ||||
|                     String.format( | ||||
|                         "#%06X", | ||||
|                         0xFFFFFF and colorSurface.data, | ||||
|                     ) | ||||
|                 }; | ||||
|                 |      } | ||||
|                 |   </style> | ||||
|                 |   $fontLinkAndStyle | ||||
|                 |</head> | ||||
|                 |<body> | ||||
|                 |   $contentText | ||||
|                 |</body>""".trimMargin(), | ||||
|                 |</body> | ||||
|                 """.trimMargin(), | ||||
|                 "text/html", | ||||
|                 "utf-8", | ||||
|             null | ||||
|                 null, | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun scrollDown() { | ||||
|         val height = binding.nestedScrollView.measuredHeight | ||||
| @@ -507,21 +547,19 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         binding.nestedScrollView.smoothScrollBy(0, -height / 2) | ||||
|     } | ||||
|  | ||||
|     private fun openInBrowserAfterFailing(customTabsIntent: CustomTabsIntent) { | ||||
|     private fun openInBrowserAfterFailing() { | ||||
|         binding.progressBar.visibility = View.GONE | ||||
|         requireActivity().openItemUrlInternalBrowser( | ||||
|                 url, | ||||
|                 customTabsIntent, | ||||
|                 requireActivity() | ||||
|         ) | ||||
|         if (context != null) { | ||||
|             requireContext().openItemUrlInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|         } else { | ||||
|             Exception("openInBrowserAfterFailing context is null").sendSilentlyWithAcraWithName("openInBrowserAfterFailing > $context") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         private const val ARG_ITEMS = "items" | ||||
|  | ||||
|         fun newInstance( | ||||
|                 item: SelfossModel.Item | ||||
|         ): ArticleFragment { | ||||
|         fun newInstance(item: SelfossModel.Item): ArticleFragment { | ||||
|             val fragment = ArticleFragment() | ||||
|             val args = Bundle() | ||||
|             args.putParcelable(ARG_ITEMS, item.toParcelable()) | ||||
| @@ -531,9 +569,11 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     } | ||||
|  | ||||
|     fun performClick(): Boolean { | ||||
|         if (binding.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || | ||||
|                 binding.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||
|  | ||||
|         if (allImages != null && ( | ||||
|                     binding.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || | ||||
|                             binding.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE | ||||
|                     ) | ||||
|         ) { | ||||
|             val position: Int = allImages.indexOf(binding.webcontent.hitTestResult.extra) | ||||
|  | ||||
|             val intent = Intent(activity, ImageActivity::class.java) | ||||
| @@ -544,6 +584,4 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         } | ||||
|         return false | ||||
|     } | ||||
|  | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,195 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.fragments | ||||
|  | ||||
| import android.content.Context | ||||
| import android.graphics.Color | ||||
| import android.graphics.drawable.Drawable | ||||
| import android.graphics.drawable.GradientDrawable | ||||
| import android.os.Build | ||||
| import android.os.Bundle | ||||
| import android.text.TextUtils | ||||
| import android.view.LayoutInflater | ||||
| import android.view.View | ||||
| import android.view.View.GONE | ||||
| import android.view.View.VISIBLE | ||||
| import android.view.ViewGroup | ||||
| import bou.amine.apps.readerforselfossv2.android.HomeActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.FilterFragmentBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.utils.getColorHexCode | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.request.target.ViewTarget | ||||
| import com.bumptech.glide.request.transition.Transition | ||||
| import com.google.android.material.bottomsheet.BottomSheetDialogFragment | ||||
| import com.google.android.material.chip.Chip | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.kodein.di.DI | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.x.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | ||||
|     private lateinit var binding: FilterFragmentBinding | ||||
|     override val di: DI by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
|  | ||||
|     private var selectedChip: Chip? = null | ||||
|  | ||||
|     override fun onCreateView( | ||||
|         inflater: LayoutInflater, | ||||
|         container: ViewGroup?, | ||||
|         savedInstanceState: Bundle?, | ||||
|     ): View { | ||||
|         binding = | ||||
|             FilterFragmentBinding.inflate( | ||||
|                 inflater, | ||||
|                 container, | ||||
|                 false, | ||||
|             ) | ||||
|  | ||||
|         val context: Context? = context | ||||
|  | ||||
|         if (context == null) { | ||||
|             dismiss() | ||||
|             Exception("FilterSheetFragment context is null").sendSilentlyWithAcraWithName("FilterSheetFragment > onCreateView") | ||||
|         } else { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                 handleTagChips(context) | ||||
|                 handleSourceChips(context) | ||||
|  | ||||
|                 binding.progressBar2.visibility = GONE | ||||
|                 binding.filterView.visibility = VISIBLE | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         binding.floatingActionButton2.setOnClickListener { | ||||
|             (activity as HomeActivity).getElementsAccordingToTab() | ||||
|             (activity as HomeActivity).fetchOnEmptyList() | ||||
|             dismiss() | ||||
|         } | ||||
|         return binding.root | ||||
|     } | ||||
|  | ||||
|     private suspend fun handleSourceChips(context: Context) { | ||||
|         val sourceGroup = binding.sourcesGroup | ||||
|  | ||||
|         repository.getSourcesDetailsOrStats().forEachIndexed { _, source -> | ||||
|             val c = Chip(context) | ||||
|             c.ellipsize = TextUtils.TruncateAt.END | ||||
|  | ||||
|             Glide.with(context) | ||||
|                 .load(source.getIcon(repository.baseUrl)) | ||||
|                 .into( | ||||
|                     object : ViewTarget<Chip?, Drawable?>(c) { | ||||
|                         override fun onResourceReady( | ||||
|                             resource: Drawable, | ||||
|                             transition: Transition<in Drawable?>?, | ||||
|                         ) { | ||||
|                             try { | ||||
|                                 c.chipIcon = resource | ||||
|                             } catch (e: Exception) { | ||||
|                                 e.sendSilentlyWithAcraWithName("sources > onResourceReady") | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                 ) | ||||
|  | ||||
|             c.text = source.title.getHtmlDecoded() | ||||
|  | ||||
|             c.setOnCloseIconClickListener { | ||||
|                 (it as Chip).isCloseIconVisible = false | ||||
|                 selectedChip = null | ||||
|                 repository.setSourceFilter(null) | ||||
|             } | ||||
|  | ||||
|             c.setOnClickListener { | ||||
|                 if (selectedChip != null) { | ||||
|                     selectedChip!!.isCloseIconVisible = false | ||||
|                 } | ||||
|                 (it as Chip).isCloseIconVisible = true | ||||
|                 selectedChip = it | ||||
|                 repository.setSourceFilter(source) | ||||
|  | ||||
|                 repository.setTagFilter(null) | ||||
|             } | ||||
|  | ||||
|             if (repository.sourceFilter.value?.equals(source) == true) { | ||||
|                 c.isCloseIconVisible = true | ||||
|                 selectedChip = c | ||||
|             } | ||||
|  | ||||
|             c.isEnabled = source.error.isNullOrBlank() | ||||
|  | ||||
|             if (!source.error.isNullOrBlank() && Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { | ||||
|                 c.tooltipText = source.error | ||||
|             } | ||||
|  | ||||
|             sourceGroup.addView(c) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private suspend fun handleTagChips(context: Context) { | ||||
|         val tagGroup = binding.tagsGroup | ||||
|  | ||||
|         val tags = repository.getTags() | ||||
|  | ||||
|         tags.forEachIndexed { _, tag -> | ||||
|             val c = Chip(context) | ||||
|             c.ellipsize = TextUtils.TruncateAt.END | ||||
|             c.text = tag.tag | ||||
|  | ||||
|             if (tag.color.isNotEmpty()) { | ||||
|                 try { | ||||
|                     val gd = GradientDrawable() | ||||
|                     val gdColor = | ||||
|                         try { | ||||
|                             Color.parseColor(tag.getColorHexCode()) | ||||
|                         } catch (e: IllegalArgumentException) { | ||||
|                             e.sendSilentlyWithAcraWithName("color issue " + tag.color + " / " + tag.getColorHexCode()) | ||||
|                             resources.getColor(R.color.colorPrimary) | ||||
|                         } | ||||
|                     gd.setColor(gdColor) | ||||
|                     gd.shape = GradientDrawable.RECTANGLE | ||||
|                     gd.setSize(30, 30) | ||||
|                     gd.cornerRadius = 30F | ||||
|                     c.chipIcon = gd | ||||
|                 } catch (e: Exception) { | ||||
|                     e.sendSilentlyWithAcraWithName("tags > GradientDrawable") | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             c.setOnCloseIconClickListener { | ||||
|                 (it as Chip).isCloseIconVisible = false | ||||
|                 selectedChip = null | ||||
|                 repository.setTagFilter(null) | ||||
|             } | ||||
|  | ||||
|             c.setOnClickListener { | ||||
|                 if (selectedChip != null) { | ||||
|                     selectedChip!!.isCloseIconVisible = false | ||||
|                 } | ||||
|                 (it as Chip).isCloseIconVisible = true | ||||
|                 selectedChip = it | ||||
|                 repository.setTagFilter(tag) | ||||
|  | ||||
|                 repository.setSourceFilter(null) | ||||
|             } | ||||
|  | ||||
|             if (repository.tagFilter.value?.equals(tag) == true) { | ||||
|                 c.isCloseIconVisible = true | ||||
|                 selectedChip = c | ||||
|             } | ||||
|  | ||||
|             tagGroup.addView(c) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         const val TAG = "FilterModalBottomSheet" | ||||
|     } | ||||
| } | ||||
| @@ -11,7 +11,6 @@ import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
|  | ||||
| class ImageFragment : Fragment() { | ||||
|  | ||||
|     private lateinit var imageUrl: String | ||||
|     private val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||
|     private var _binding: FragmentImageBinding? = null | ||||
| @@ -23,12 +22,16 @@ class ImageFragment : Fragment() { | ||||
|         imageUrl = requireArguments().getString("imageUrl")!! | ||||
|     } | ||||
|  | ||||
|     override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { | ||||
|     override fun onCreateView( | ||||
|         inflater: LayoutInflater, | ||||
|         container: ViewGroup?, | ||||
|         savedInstanceState: Bundle?, | ||||
|     ): View? { | ||||
|         _binding = FragmentImageBinding.inflate(inflater, container, false) | ||||
|         val view = binding?.root | ||||
|  | ||||
|         binding!!.photoView.visibility = View.VISIBLE | ||||
|         Glide.with(activity) | ||||
|         Glide.with(requireActivity()) | ||||
|             .asBitmap() | ||||
|             .apply(glideOptions) | ||||
|             .load(imageUrl) | ||||
| @@ -45,9 +48,7 @@ class ImageFragment : Fragment() { | ||||
|     companion object { | ||||
|         private const val ARG_IMAGE = "imageUrl" | ||||
|  | ||||
|         fun newInstance( | ||||
|                 imageUrl : String | ||||
|         ): ImageFragment { | ||||
|         fun newInstance(imageUrl: String): ImageFragment { | ||||
|             val fragment = ImageFragment() | ||||
|             val args = Bundle() | ||||
|             args.putString(ARG_IMAGE, imageUrl) | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package bou.amine.apps.readerforselfossv2.android.model | ||||
|  | ||||
| import android.content.Context | ||||
| import android.webkit.URLUtil | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| import com.bumptech.glide.Glide | ||||
| @@ -13,7 +14,6 @@ fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | ||||
|  | ||||
|     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL).timeout(10000) | ||||
|  | ||||
|  | ||||
|     try { | ||||
|         for (url in imageUrls) { | ||||
|             if (URLUtil.isValidUrl(url)) { | ||||
| @@ -23,6 +23,7 @@ fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | ||||
|             } | ||||
|         } | ||||
|     } catch (e: Error) { | ||||
|         e.sendSilentlyWithAcraWithName("preloadImages") | ||||
|         return false | ||||
|     } | ||||
|  | ||||
| @@ -35,7 +36,7 @@ fun String.toTextDrawableString(): String { | ||||
|         try { | ||||
|             textDrawable.append(s[0]) | ||||
|         } catch (e: StringIndexOutOfBoundsException) { | ||||
|             // We do nothing | ||||
|             e.sendSilentlyWithAcraWithName("toTextDrawableString") | ||||
|         } | ||||
|     } | ||||
|     return textDrawable.toString() | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package bou.amine.apps.readerforselfossv2.android.model | ||||
| import android.os.Parcel | ||||
| import android.os.Parcelable | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import com.google.gson.annotations.SerializedName | ||||
|  | ||||
| fun SelfossModel.Item.toParcelable(): ParecelableItem = | ||||
|     ParecelableItem( | ||||
| @@ -17,8 +16,10 @@ fun SelfossModel.Item.toParcelable() : ParecelableItem = | ||||
|         this.icon, | ||||
|         this.link, | ||||
|         this.sourcetitle, | ||||
|         this.tags.joinToString(",") | ||||
|         this.tags.joinToString(","), | ||||
|         this.author, | ||||
|     ) | ||||
|  | ||||
| fun ParecelableItem.toModel(): SelfossModel.Item = | ||||
|     SelfossModel.Item( | ||||
|         this.id, | ||||
| @@ -31,26 +32,30 @@ fun ParecelableItem.toModel() : SelfossModel.Item = | ||||
|         this.icon, | ||||
|         this.link, | ||||
|         this.sourcetitle, | ||||
|         this.tags.split(",") | ||||
|         this.tags.split(","), | ||||
|         this.author, | ||||
|     ) | ||||
| data class ParecelableItem( | ||||
|     @SerializedName("id") val id: Int, | ||||
|     @SerializedName("datetime") val datetime: String, | ||||
|     @SerializedName("title") val title: String, | ||||
|     @SerializedName("content") val content: String, | ||||
|     @SerializedName("unread") var unread: Boolean, | ||||
|     @SerializedName("starred") var starred: Boolean, | ||||
|     @SerializedName("thumbnail") val thumbnail: String?, | ||||
|     @SerializedName("icon") val icon: String?, | ||||
|     @SerializedName("link") val link: String, | ||||
|     @SerializedName("sourcetitle") val sourcetitle: String, | ||||
|     @SerializedName("tags") val tags: String | ||||
| ) : Parcelable { | ||||
|  | ||||
| data class ParecelableItem( | ||||
|     val id: Int, | ||||
|     val datetime: String, | ||||
|     val title: String, | ||||
|     val content: String, | ||||
|     var unread: Boolean, | ||||
|     var starred: Boolean, | ||||
|     val thumbnail: String?, | ||||
|     val icon: String?, | ||||
|     val link: String, | ||||
|     val sourcetitle: String, | ||||
|     val tags: String, | ||||
|     val author: String?, | ||||
| ) : Parcelable { | ||||
|     companion object { | ||||
|         @JvmField | ||||
|         val CREATOR: Parcelable.Creator<ParecelableItem> = object : Parcelable.Creator<ParecelableItem> { | ||||
|         val CREATOR: Parcelable.Creator<ParecelableItem> = | ||||
|             object : Parcelable.Creator<ParecelableItem> { | ||||
|                 override fun createFromParcel(source: Parcel): ParecelableItem = ParecelableItem(source) | ||||
|  | ||||
|                 override fun newArray(size: Int): Array<ParecelableItem?> = arrayOfNulls(size) | ||||
|             } | ||||
|     } | ||||
| @@ -66,12 +71,16 @@ data class ParecelableItem( | ||||
|         icon = source.readString(), | ||||
|         link = source.readString().orEmpty(), | ||||
|         sourcetitle = source.readString().orEmpty(), | ||||
|         tags = source.readString().orEmpty() | ||||
|         tags = source.readString().orEmpty(), | ||||
|         author = source.readString().orEmpty(), | ||||
|     ) | ||||
|  | ||||
|     override fun describeContents() = 0 | ||||
|  | ||||
|     override fun writeToParcel(dest: Parcel, flags: Int) { | ||||
|     override fun writeToParcel( | ||||
|         dest: Parcel, | ||||
|         flags: Int, | ||||
|     ) { | ||||
|         dest.writeInt(id) | ||||
|         dest.writeString(datetime) | ||||
|         dest.writeString(title) | ||||
| @@ -83,5 +92,6 @@ data class ParecelableItem( | ||||
|         dest.writeString(link) | ||||
|         dest.writeString(sourcetitle) | ||||
|         dest.writeString(tags) | ||||
|         dest.writeString(author) | ||||
|     } | ||||
| } | ||||
| @@ -1,45 +1,38 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.settings | ||||
|  | ||||
| import android.content.Intent | ||||
| import android.net.Uri | ||||
| import android.os.Bundle | ||||
| import android.text.Editable | ||||
| import android.text.InputFilter | ||||
| import android.text.InputType | ||||
| import android.text.TextWatcher | ||||
| import android.view.Menu | ||||
| import android.view.MenuInflater | ||||
| import android.view.MenuItem | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.appcompat.app.AppCompatDelegate | ||||
| import androidx.core.widget.addTextChangedListener | ||||
| import androidx.preference.EditTextPreference | ||||
| import androidx.preference.Preference | ||||
| import androidx.preference.PreferenceFragmentCompat | ||||
| import androidx.preference.PreferenceManager | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySettingsBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openUrlInBrowser | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.ftinc.scoop.Scoop | ||||
| import com.mikepenz.aboutlibraries.LibsBuilder | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
|  | ||||
| private const val TITLE_TAG = "settingsActivityTitle" | ||||
|  | ||||
| class SettingsActivity : AppCompatActivity(), | ||||
|         PreferenceFragmentCompat.OnPreferenceStartFragmentCallback { | ||||
| class SettingsActivity : | ||||
|     AppCompatActivity(), | ||||
|     PreferenceFragmentCompat.OnPreferenceStartFragmentCallback, | ||||
|     DIAware { | ||||
|     override val di by closestDI() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         if (PreferenceManager.getDefaultSharedPreferences(this).getBoolean("dark_theme", false)) { | ||||
|             setTheme(R.style.NoBarDark) | ||||
|         } | ||||
|         val binding = ActivitySettingsBinding.inflate(layoutInflater) | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolbar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         setContentView(binding.root) | ||||
|         if (savedInstanceState == null) { | ||||
|             supportFragmentManager | ||||
| @@ -80,13 +73,14 @@ class SettingsActivity : AppCompatActivity(), | ||||
|  | ||||
|     override fun onPreferenceStartFragment( | ||||
|         caller: PreferenceFragmentCompat, | ||||
|             pref: Preference | ||||
|         pref: Preference, | ||||
|     ): Boolean { | ||||
|         // Instantiate the new Fragment | ||||
|         val args = pref.extras | ||||
|         val fragment = supportFragmentManager.fragmentFactory.instantiate( | ||||
|         val fragment = | ||||
|             supportFragmentManager.fragmentFactory.instantiate( | ||||
|                 classLoader, | ||||
|                 pref.fragment | ||||
|                 pref.fragment.toString(), | ||||
|             ).apply { | ||||
|                 arguments = args | ||||
|                 setTargetFragment(caller, 0) | ||||
| @@ -102,119 +96,181 @@ class SettingsActivity : AppCompatActivity(), | ||||
|     } | ||||
|  | ||||
|     class MainPreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_main, rootKey) | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = | ||||
|                 Preference.OnPreferenceChangeListener { _, newValue -> | ||||
|                     AppCompatDelegate.setDefaultNightMode( | ||||
|                         newValue.toString().toInt() | ||||
|                     ) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||
|                     true | ||||
|                 } | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("action_about")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { _ -> | ||||
|                     context?.let { | ||||
|                         LibsBuilder() | ||||
|                             .withAboutIconShown(true) | ||||
|                             .withAboutVersionShown(true) | ||||
|                             .start(it) | ||||
|                     } | ||||
|                     true | ||||
|                 } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class GeneralPreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_general, rootKey) | ||||
|  | ||||
|             val editTextPreference = preferenceManager.findPreference<EditTextPreference>("prefer_api_items_number") | ||||
|             val editTextPreference = | ||||
|                 preferenceManager.findPreference<EditTextPreference>("prefer_api_items_number") | ||||
|             editTextPreference?.setOnBindEditTextListener { editText -> | ||||
|                 editText.inputType = InputType.TYPE_CLASS_NUMBER | ||||
|                 editText.filters = arrayOf( | ||||
|                 editText.filters = | ||||
|                     arrayOf( | ||||
|                         InputFilter { source, _, _, dest, _, _ -> | ||||
|                             try { | ||||
|                                 val input: Int = (dest.toString() + source.toString()).toInt() | ||||
|                                 if (input in 1..200) return@InputFilter null | ||||
|                             } catch (nfe: NumberFormatException) { | ||||
|                                 Toast.makeText(activity, R.string.items_number_should_be_number, Toast.LENGTH_LONG).show() | ||||
|                                 Toast.makeText( | ||||
|                                     activity, | ||||
|                                     R.string.items_number_should_be_number, | ||||
|                                     Toast.LENGTH_LONG | ||||
|                                 ).show() | ||||
|                             } | ||||
|                             "" | ||||
|                         } | ||||
|                         }, | ||||
|                     ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class ArticleViewerPreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_viewer, rootKey) | ||||
|  | ||||
|             val fontSize = preferenceManager.findPreference<EditTextPreference>("reader_font_size") | ||||
|             fontSize?.setOnBindEditTextListener { editText -> | ||||
|                 editText.inputType = InputType.TYPE_CLASS_NUMBER | ||||
|                 editText.addTextChangedListener { object : TextWatcher { | ||||
|                     override fun beforeTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) {} | ||||
|                     override fun onTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) {} | ||||
|                 editText.addTextChangedListener { | ||||
|                     object : TextWatcher { | ||||
|                         override fun beforeTextChanged( | ||||
|                             charSequence: CharSequence, | ||||
|                             i: Int, | ||||
|                             i1: Int, | ||||
|                             i2: Int, | ||||
|                         ) { | ||||
|                             // We do nothing | ||||
|                         } | ||||
|  | ||||
|                         override fun onTextChanged( | ||||
|                             charSequence: CharSequence, | ||||
|                             i: Int, | ||||
|                             i1: Int, | ||||
|                             i2: Int, | ||||
|                         ) { | ||||
|                             // We do nothing | ||||
|                         } | ||||
|  | ||||
|                         override fun afterTextChanged(editable: Editable) { | ||||
|                             try { | ||||
|                                 editText.textSize = editable.toString().toInt().toFloat() | ||||
|                             } catch (e: NumberFormatException) { | ||||
|                                 e.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > afterTextChanged") | ||||
|                             } | ||||
|                         } | ||||
|                 } } | ||||
|                 editText.filters = arrayOf( | ||||
|                     } | ||||
|                 } | ||||
|                 editText.filters = | ||||
|                     arrayOf( | ||||
|                         InputFilter { source, _, _, dest, _, _ -> | ||||
|                             try { | ||||
|                                 val input = (dest.toString() + source.toString()).toInt() | ||||
|                                 if (input > 0) return@InputFilter null | ||||
|                             } catch (nfe: NumberFormatException) { | ||||
|                                 nfe.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > filters") | ||||
|                             } | ||||
|                             "" | ||||
|                         } | ||||
|                         }, | ||||
|                     ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class OfflinePreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_offline, rootKey) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class ThemePreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_theme, rootKey) | ||||
|             setHasOptionsMenu(true) | ||||
|         } | ||||
|  | ||||
|         override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) { | ||||
|             super.onCreateOptionsMenu(menu, inflater) | ||||
|             inflater.inflate(R.menu.settings_theme, menu) | ||||
|             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = | ||||
|                 Preference.OnPreferenceChangeListener { _, newValue -> | ||||
|                     AppCompatDelegate.setDefaultNightMode( | ||||
|                         newValue.toString().toInt() | ||||
|                     ) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||
|                     true | ||||
|                 } | ||||
|  | ||||
|         override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||
|             val id = item.itemId | ||||
|             if (id == R.id.clear) { | ||||
|                 AppColors.resetColors() | ||||
|                 requireActivity().recreate() | ||||
|             } | ||||
|             return super.onOptionsItemSelected(item) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class LinksPreferenceFragment : PreferenceFragmentCompat() { | ||||
|         private fun openUrl(uri: Uri?) { | ||||
|             val browserIntent = Intent(Intent.ACTION_VIEW, uri) | ||||
|             startActivity(browserIntent) | ||||
|         private fun openUrl(url: String) { | ||||
|             context?.openUrlInBrowser(url) | ||||
|         } | ||||
|  | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_links, rootKey) | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("trackerLink")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { | ||||
|                 openUrl(Uri.parse(AppSettingsService.trackerUrl)) | ||||
|             preferenceManager.findPreference<Preference>("trackerLink")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { | ||||
|                     openUrl(AppSettingsService.trackerUrl) | ||||
|                     true | ||||
|                 } | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("sourceLink")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { | ||||
|                 openUrl(Uri.parse(AppSettingsService.sourceUrl)) | ||||
|             preferenceManager.findPreference<Preference>("sourceLink")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { | ||||
|                     openUrl(AppSettingsService.sourceUrl) | ||||
|                     false | ||||
|                 } | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("translation")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { | ||||
|                 openUrl(Uri.parse(AppSettingsService.translationUrl)) | ||||
|             preferenceManager.findPreference<Preference>("translation")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { | ||||
|                     openUrl(AppSettingsService.translationUrl) | ||||
|                     false | ||||
|                 } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class ExperimentalPreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|         override fun onCreatePreferences( | ||||
|             savedInstanceState: Bundle?, | ||||
|             rootKey: String?, | ||||
|         ) { | ||||
|             setPreferencesFromResource(R.xml.pref_experimental, rootKey) | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -0,0 +1,21 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.testing | ||||
|  | ||||
| import androidx.test.espresso.idling.CountingIdlingResource | ||||
|  | ||||
| object CountingIdlingResourceSingleton { | ||||
|  | ||||
|     private const val RESOURCE = "GLOBAL" | ||||
|  | ||||
|     @JvmField | ||||
|     val countingIdlingResource = CountingIdlingResource(RESOURCE) | ||||
|  | ||||
|     fun increment() { | ||||
|         countingIdlingResource.increment() | ||||
|     } | ||||
|  | ||||
|     fun decrement() { | ||||
|         if (!countingIdlingResource.isIdleNow) { | ||||
|             countingIdlingResource.decrement() | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,19 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.testing | ||||
|  | ||||
| import android.os.Build | ||||
|  | ||||
|  | ||||
| class TestingHelper { | ||||
|     fun isUnitTest(): Boolean { | ||||
|         var device = Build.DEVICE | ||||
|         var product = Build.PRODUCT | ||||
|         if (device == null) { | ||||
|             device = "" | ||||
|         } | ||||
|  | ||||
|         if (product == null) { | ||||
|             product = "" | ||||
|         } | ||||
|         return device == "robolectric" && product == "robolectric" | ||||
|     } | ||||
| } | ||||
| @@ -1,72 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.themes | ||||
|  | ||||
| import android.app.Activity | ||||
| import androidx.annotation.ColorInt | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import com.russhwolf.settings.Settings | ||||
|  | ||||
| class AppColors(a: Activity) { | ||||
|  | ||||
|     @ColorInt val colorPrimary: Int | ||||
|     @ColorInt val colorPrimaryDark: Int | ||||
|     @ColorInt val colorAccent: Int | ||||
|     @ColorInt val colorAccentDark: Int | ||||
|     @ColorInt val colorBackground: Int | ||||
|     @ColorInt val textColor: Int | ||||
|     val isDarkTheme: Boolean | ||||
|  | ||||
|     init { | ||||
|         val settings = Settings() | ||||
|  | ||||
|         colorPrimary = | ||||
|                 settings.getInt( | ||||
|                     "color_primary", | ||||
|                     a.resources.getColor(R.color.colorPrimary) | ||||
|                 ) | ||||
|         colorPrimaryDark = | ||||
|                 settings.getInt( | ||||
|                     "color_primary_dark", | ||||
|                     a.resources.getColor(R.color.colorPrimaryDark) | ||||
|                 ) | ||||
|         colorAccent = | ||||
|                 settings.getInt( | ||||
|                     "color_accent", | ||||
|                     a.resources.getColor(R.color.colorAccent) | ||||
|                 ) | ||||
|         colorAccentDark = | ||||
|                 settings.getInt( | ||||
|                     "color_accent_dark", | ||||
|                     a.resources.getColor(R.color.colorAccentDark) | ||||
|                 ) | ||||
|         isDarkTheme = | ||||
|                 settings.getBoolean( | ||||
|                     "dark_theme", | ||||
|                     false | ||||
|                 ) | ||||
|  | ||||
|         colorBackground = if (isDarkTheme) { | ||||
|             a.setTheme(R.style.NoBarDark) | ||||
|             a.resources.getColor(R.color.darkBackground) | ||||
|         } else { | ||||
|             a.setTheme(R.style.NoBar) | ||||
|             a.resources.getColor(R.color.grey_50) | ||||
|         } | ||||
|  | ||||
|         textColor = if (isDarkTheme) { | ||||
|             a.resources.getColor(R.color.white) | ||||
|         } else { | ||||
|             a.resources.getColor(R.color.grey_900) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         fun resetColors() { | ||||
|             val settings = Settings() | ||||
|             settings.remove("color_primary") | ||||
|             settings.remove("color_primary_dark") | ||||
|             settings.remove("color_accent") | ||||
|             settings.remove("color_accent_dark") | ||||
|             settings.remove("dark_theme") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.themes | ||||
|  | ||||
| enum class Toppings(val value: Int) { | ||||
|     PRIMARY(1), | ||||
|     PRIMARY_DARK(2), | ||||
|     ACCENT(3), | ||||
|     ACCENT_DARK(4) | ||||
| } | ||||
| @@ -5,7 +5,10 @@ import android.content.Intent | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
|  | ||||
| fun Context.shareLink(itemUrl: String, itemTitle: String) { | ||||
| fun Context.shareLink( | ||||
|     itemUrl: String, | ||||
|     itemTitle: String, | ||||
| ) { | ||||
|     val sendIntent = Intent() | ||||
|     sendIntent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|     sendIntent.action = Intent.ACTION_SEND | ||||
| @@ -15,7 +18,7 @@ fun Context.shareLink(itemUrl: String, itemTitle: String) { | ||||
|     startActivity( | ||||
|         Intent.createChooser( | ||||
|             sendIntent, | ||||
|             getString(R.string.share) | ||||
|         ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) | ||||
|             getString(R.string.share), | ||||
|         ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK), | ||||
|     ) | ||||
| } | ||||
| @@ -0,0 +1,65 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils | ||||
|  | ||||
| import android.content.Context | ||||
| import android.graphics.drawable.GradientDrawable | ||||
| import android.util.AttributeSet | ||||
| import android.view.LayoutInflater | ||||
| import android.view.View | ||||
| import android.widget.RelativeLayout | ||||
| import android.widget.TextView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import com.google.android.material.imageview.ShapeableImageView | ||||
| import kotlin.math.abs | ||||
|  | ||||
| class CircleImageView | ||||
|     @JvmOverloads | ||||
|     constructor( | ||||
|         context: Context, | ||||
|         attrs: AttributeSet? = null, | ||||
|         defStyleAttr: Int = 0, | ||||
|     ) : RelativeLayout(context, attrs, defStyleAttr) { | ||||
|         val view: View | ||||
|         val imageView: ShapeableImageView | ||||
|         val textView: TextView | ||||
|  | ||||
|         private val colorScheme = | ||||
|             listOf( | ||||
|                 -0x1a8c8d, | ||||
|                 -0xf9d6e, | ||||
|                 -0x459738, | ||||
|                 -0x6a8a33, | ||||
|                 -0x867935, | ||||
|                 -0x9b4a0a, | ||||
|                 -0xb03c09, | ||||
|                 -0xb22f1f, | ||||
|                 -0xb24954, | ||||
|                 -0x7e387c, | ||||
|                 -0x512a7f, | ||||
|                 -0x759b, | ||||
|                 -0x2b1ea9, | ||||
|                 -0x2ab1, | ||||
|                 -0x48b3, | ||||
|                 -0x5e7781, | ||||
|                 -0x6f5b52, | ||||
|             ) | ||||
|  | ||||
|         init { | ||||
|             view = LayoutInflater.from(context).inflate(R.layout.circle_image_view, this, true) | ||||
|             imageView = view.findViewById(R.id.circleImage) | ||||
|             textView = view.findViewById(R.id.circleText) | ||||
|         } | ||||
|  | ||||
|         fun setBackgroundAndText(text: String) { | ||||
|             val circleDrawable = GradientDrawable() | ||||
|             val color = colorFromIdentifier(text) | ||||
|             circleDrawable.setColor(color) | ||||
|             imageView.setImageDrawable(circleDrawable) | ||||
|  | ||||
|             textView.text = text.toTextDrawableString() | ||||
|         } | ||||
|  | ||||
|         private fun colorFromIdentifier(key: String): Int { | ||||
|             return colorScheme[abs(key.hashCode()) % colorScheme.size] | ||||
|         } | ||||
|     } | ||||
| @@ -1,13 +1,10 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils | ||||
|  | ||||
| import android.app.Activity | ||||
| import android.app.PendingIntent | ||||
| import android.content.ActivityNotFoundException | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.graphics.BitmapFactory | ||||
| import android.net.Uri | ||||
| import android.os.Build | ||||
| import android.text.Spannable | ||||
| import android.text.style.ClickableSpan | ||||
| import android.util.Patterns | ||||
| @@ -15,155 +12,40 @@ import android.view.MotionEvent | ||||
| import android.view.View | ||||
| import android.widget.TextView | ||||
| import android.widget.Toast | ||||
| import androidx.browser.customtabs.CustomTabsIntent | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.ReaderActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
| import okhttp3.HttpUrl.Companion.toHttpUrlOrNull | ||||
|  | ||||
| fun Context.buildCustomTabsIntent(): CustomTabsIntent { | ||||
|  | ||||
|     val actionIntent = Intent(Intent.ACTION_SEND) | ||||
|     actionIntent.type = "text/plain" | ||||
|     val pflags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|         PendingIntent.FLAG_IMMUTABLE | ||||
|     } else { | ||||
|         0 | ||||
|     } | ||||
|     val createPendingShareIntent: PendingIntent = PendingIntent.getActivity( | ||||
|             this, | ||||
|             0, | ||||
|             actionIntent, | ||||
|             pflags | ||||
|         ) | ||||
|  | ||||
|     val intentBuilder = CustomTabsIntent.Builder() | ||||
|  | ||||
|     // TODO: change to primary when it's possible to customize custom tabs title color | ||||
|     //intentBuilder.setToolbarColor(c.getResources().getColor(R.color.colorPrimary)); | ||||
|     intentBuilder.setToolbarColor(resources.getColor(R.color.colorAccentDark)) | ||||
|     intentBuilder.setShowTitle(true) | ||||
|  | ||||
|  | ||||
|     intentBuilder.setStartAnimations( | ||||
|         this, | ||||
|         R.anim.slide_in_right, | ||||
|         R.anim.slide_out_left | ||||
|     ) | ||||
|     intentBuilder.setExitAnimations( | ||||
|         this, | ||||
|         android.R.anim.slide_in_left, | ||||
|         android.R.anim.slide_out_right | ||||
|     ) | ||||
|  | ||||
|     val closeicon = BitmapFactory.decodeResource(resources, R.drawable.ic_close_white_24dp) | ||||
|     intentBuilder.setCloseButtonIcon(closeicon) | ||||
|  | ||||
|     val shareLabel = this.getString(R.string.label_share) | ||||
|     val icon = BitmapFactory.decodeResource( | ||||
|         resources, | ||||
|         R.drawable.ic_share_white_24dp | ||||
|     ) | ||||
|     intentBuilder.setActionButton(icon, shareLabel, createPendingShareIntent) | ||||
|  | ||||
|     return intentBuilder.build() | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrlInternally( | ||||
|     allItems: ArrayList<SelfossModel.Item>, | ||||
|     currentItem: Int, | ||||
|     linkDecoded: String, | ||||
|     customTabsIntent: CustomTabsIntent, | ||||
|     articleViewer: Boolean, | ||||
|     app: Activity | ||||
| ) { | ||||
|     if (articleViewer) { | ||||
|         ReaderActivity.allItems = allItems | ||||
|         val intent = Intent(this, ReaderActivity::class.java) | ||||
|         intent.putExtra("currentItem", currentItem) | ||||
|         app.startActivity(intent) | ||||
|     } else { | ||||
|         this.openItemUrlInternalBrowser( | ||||
|                 linkDecoded, | ||||
|                 customTabsIntent, | ||||
|                 app) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrlInternalBrowser( | ||||
|         linkDecoded: String, | ||||
|         customTabsIntent: CustomTabsIntent, | ||||
|         app: Activity | ||||
| ) { | ||||
|     try { | ||||
|         CustomTabActivityHelper.openCustomTab( | ||||
|                 app, | ||||
|                 customTabsIntent, | ||||
|                 Uri.parse(linkDecoded) | ||||
|         ) { _, uri -> | ||||
|             val intent = Intent(Intent.ACTION_VIEW, uri) | ||||
|             intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|             startActivity(intent) | ||||
|         } | ||||
|     } catch (e: Exception) { | ||||
|         openInBrowser(linkDecoded, app) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrl( | ||||
|     allItems: ArrayList<SelfossModel.Item>, | ||||
|     currentItem: Int, | ||||
|     linkDecoded: String, | ||||
|     customTabsIntent: CustomTabsIntent, | ||||
|     internalBrowser: Boolean, | ||||
|     articleViewer: Boolean, | ||||
|     app: Activity | ||||
|     app: Activity, | ||||
| ) { | ||||
|  | ||||
|     if (!linkDecoded.isUrlValid()) { | ||||
|         Toast.makeText( | ||||
|             this, | ||||
|             this.getString(R.string.cant_open_invalid_url), | ||||
|             Toast.LENGTH_LONG | ||||
|             Toast.LENGTH_LONG, | ||||
|         ).show() | ||||
|     } else { | ||||
|         if (!internalBrowser) { | ||||
|             openInBrowser(linkDecoded, app) | ||||
|         } else if (articleViewer) { | ||||
|             this.openItemUrlInternally( | ||||
|                 allItems, | ||||
|                 currentItem, | ||||
|                 linkDecoded, | ||||
|                 customTabsIntent, | ||||
|                 articleViewer, | ||||
|                 app | ||||
|             ) | ||||
|         } else { | ||||
|             this.openItemUrlInternalBrowser( | ||||
|                     linkDecoded, | ||||
|                     customTabsIntent, | ||||
|                     app | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| private fun openInBrowser(linkDecoded: String, app: Activity) { | ||||
|     val intent = Intent(Intent.ACTION_VIEW) | ||||
|     intent.data = Uri.parse(linkDecoded) | ||||
|     try { | ||||
|         if (articleViewer) { | ||||
|             val intent = Intent(this, ReaderActivity::class.java) | ||||
|             intent.putExtra("currentItem", currentItem) | ||||
|             app.startActivity(intent) | ||||
|     } catch (e: ActivityNotFoundException) { | ||||
|         Toast.makeText(app.baseContext, e.message, Toast.LENGTH_LONG).show() | ||||
|         } else { | ||||
|             this.openUrlInBrowserAsNewTask(linkDecoded) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun String.isUrlValid(): Boolean = | ||||
|     this.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||
|  | ||||
| fun String.isBaseUrlValid(ctx: Context): Boolean { | ||||
| fun String.isBaseUrlInvalid(): Boolean { | ||||
|     val baseUrl = this.toHttpUrlOrNull() | ||||
|     var existsAndEndsWithSlash = false | ||||
|     if (baseUrl != null) { | ||||
| @@ -171,18 +53,41 @@ fun String.isBaseUrlValid(ctx: Context): Boolean { | ||||
|         existsAndEndsWithSlash = "" == pathSegments[pathSegments.size - 1] | ||||
|     } | ||||
|  | ||||
|     return Patterns.WEB_URL.matcher(this).matches() && existsAndEndsWithSlash | ||||
|     return !(Patterns.WEB_URL.matcher(this).matches() && existsAndEndsWithSlash) | ||||
| } | ||||
|  | ||||
| fun Context.openInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
| fun Context.openItemUrlInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
|     this.openUrlInBrowserAsNewTask(i.getLinkDecoded().toStringUriWithHttp()) | ||||
| } | ||||
|  | ||||
| fun Context.openUrlInBrowserAsNewTask(url: String) { | ||||
|  | ||||
|     val intent = Intent(Intent.ACTION_VIEW) | ||||
|     intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|     intent.data = Uri.parse(i.getLinkDecoded().toStringUriWithHttp()) | ||||
|     startActivity(intent) | ||||
|     intent.data = Uri.parse(url) | ||||
|     this.mayBeStartActivity(intent) | ||||
| } | ||||
|  | ||||
| fun Context.openUrlInBrowser(url: String) { | ||||
|     val intent = Intent(Intent.ACTION_VIEW) | ||||
|     intent.data = Uri.parse(url) | ||||
|     this.mayBeStartActivity(intent) | ||||
| } | ||||
|  | ||||
| fun Context.mayBeStartActivity(intent: Intent) { | ||||
|     try { | ||||
|         this.startActivity(intent) | ||||
|     } catch (e: ActivityNotFoundException) { | ||||
|         Toast.makeText(this, getString(R.string.no_browser), Toast.LENGTH_SHORT).show() | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| class LinkOnTouchListener : View.OnTouchListener { | ||||
|     override fun onTouch(v: View?, event: MotionEvent?): Boolean { | ||||
|     override fun onTouch( | ||||
|         v: View?, | ||||
|         event: MotionEvent?, | ||||
|     ): Boolean { | ||||
|         var ret = false | ||||
|         val widget: TextView = v as TextView | ||||
|         val text: CharSequence = widget.text | ||||
| @@ -191,7 +96,8 @@ class LinkOnTouchListener: View.OnTouchListener { | ||||
|         val action = event!!.action | ||||
|  | ||||
|         if (action == MotionEvent.ACTION_UP || | ||||
|             action == MotionEvent.ACTION_DOWN) { | ||||
|             action == MotionEvent.ACTION_DOWN | ||||
|         ) { | ||||
|             var x: Float = event.x | ||||
|             var y: Float = event.y | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,9 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.acra | ||||
|  | ||||
| import org.acra.ACRA | ||||
| import org.acra.ktx.sendSilentlyWithAcra | ||||
|  | ||||
| fun Throwable.sendSilentlyWithAcraWithName(name: String) { | ||||
|     ACRA.errorReporter.putCustomData("error_source", name) | ||||
|     this.sendSilentlyWithAcra() | ||||
| } | ||||
| @@ -0,0 +1,29 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.acra | ||||
|  | ||||
| import android.content.Context | ||||
| import android.os.DeadSystemException | ||||
| import com.google.auto.service.AutoService | ||||
| import org.acra.builder.ReportBuilder | ||||
| import org.acra.config.CoreConfiguration | ||||
| import org.acra.config.ReportingAdministrator | ||||
| import org.acra.data.CrashReportData | ||||
|  | ||||
|  | ||||
| @AutoService(ReportingAdministrator::class) | ||||
| class AcraReportingAdministrator : ReportingAdministrator { | ||||
|     override fun shouldStartCollecting( | ||||
|         context: Context, | ||||
|         config: CoreConfiguration, | ||||
|         reportBuilder: ReportBuilder | ||||
|     ): Boolean { | ||||
|         return reportBuilder.exception !is DeadSystemException | ||||
|     } | ||||
|  | ||||
|     override fun shouldSendReport( | ||||
|         context: Context, | ||||
|         config: CoreConfiguration, | ||||
|         crashReportData: CrashReportData | ||||
|     ): Boolean { | ||||
|         return crashReportData.get("BRAND") != "redroid" | ||||
|     } | ||||
| } | ||||
| @@ -8,5 +8,4 @@ fun TextBadgeItem.removeBadge(): TextBadgeItem { | ||||
|     return this | ||||
| } | ||||
|  | ||||
| fun TextBadgeItem.maybeShow(): TextBadgeItem = | ||||
|     if (this.isHidden) this.show() else this | ||||
| fun TextBadgeItem.maybeShow(): TextBadgeItem = if (this.isHidden) this.show() else this | ||||
|   | ||||
| @@ -1,153 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import android.app.Activity; | ||||
| import android.net.Uri; | ||||
| import android.os.Bundle; | ||||
| import androidx.browser.customtabs.CustomTabsClient; | ||||
| import androidx.browser.customtabs.CustomTabsIntent; | ||||
| import androidx.browser.customtabs.CustomTabsServiceConnection; | ||||
| import androidx.browser.customtabs.CustomTabsSession; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * This is a helper class to manage the connection to the Custom Tabs Service. | ||||
|  */ | ||||
| public class CustomTabActivityHelper implements ServiceConnectionCallback { | ||||
|     private CustomTabsSession mCustomTabsSession; | ||||
|     private CustomTabsClient mClient; | ||||
|     private CustomTabsServiceConnection mConnection; | ||||
|     private ConnectionCallback mConnectionCallback; | ||||
|  | ||||
|     /** | ||||
|      * Opens the URL on a Custom Tab if possible. Otherwise fallsback to opening it on a WebView. | ||||
|      * | ||||
|      * @param activity         The host activity. | ||||
|      * @param customTabsIntent a CustomTabsIntent to be used if Custom Tabs is available. | ||||
|      * @param uri              the Uri to be opened. | ||||
|      * @param fallback         a CustomTabFallback to be used if Custom Tabs is not available. | ||||
|      */ | ||||
|     public static void openCustomTab(Activity activity, | ||||
|                                      CustomTabsIntent customTabsIntent, | ||||
|                                      Uri uri, | ||||
|                                      CustomTabFallback fallback) { | ||||
|         String packageName = CustomTabsHelper.getPackageNameToUse(activity); | ||||
|  | ||||
|         //If we cant find a package name, it means theres no browser that supports | ||||
|         //Chrome Custom Tabs installed. So, we fallback to the webview | ||||
|         if (packageName == null) { | ||||
|             if (fallback != null) { | ||||
|                 fallback.openUri(activity, uri); | ||||
|             } | ||||
|         } else { | ||||
|             customTabsIntent.intent.setPackage(packageName); | ||||
|             customTabsIntent.launchUrl(activity, uri); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Unbinds the Activity from the Custom Tabs Service. | ||||
|      * | ||||
|      * @param activity the activity that is connected to the service. | ||||
|      */ | ||||
|     public void unbindCustomTabsService(Activity activity) { | ||||
|         if (mConnection == null) return; | ||||
|         activity.unbindService(mConnection); | ||||
|         mClient = null; | ||||
|         mCustomTabsSession = null; | ||||
|         mConnection = null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates or retrieves an exiting CustomTabsSession. | ||||
|      * | ||||
|      * @return a CustomTabsSession. | ||||
|      */ | ||||
|     public CustomTabsSession getSession() { | ||||
|         if (mClient == null) { | ||||
|             mCustomTabsSession = null; | ||||
|         } else if (mCustomTabsSession == null) { | ||||
|             mCustomTabsSession = mClient.newSession(null); | ||||
|         } | ||||
|         return mCustomTabsSession; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Register a Callback to be called when connected or disconnected from the Custom Tabs Service. | ||||
|      * | ||||
|      * @param connectionCallback | ||||
|      */ | ||||
|     public void setConnectionCallback(ConnectionCallback connectionCallback) { | ||||
|         this.mConnectionCallback = connectionCallback; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Binds the Activity to the Custom Tabs Service. | ||||
|      * | ||||
|      * @param activity the activity to be binded to the service. | ||||
|      */ | ||||
|     public void bindCustomTabsService(Activity activity) { | ||||
|         if (mClient != null) return; | ||||
|  | ||||
|         String packageName = CustomTabsHelper.getPackageNameToUse(activity); | ||||
|         if (packageName == null) return; | ||||
|  | ||||
|         mConnection = new ServiceConnection(this); | ||||
|         CustomTabsClient.bindCustomTabsService(activity, packageName, mConnection); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return true if call to mayLaunchUrl was accepted. | ||||
|      * @see {@link CustomTabsSession#mayLaunchUrl(Uri, Bundle, List)}. | ||||
|      */ | ||||
|     public boolean mayLaunchUrl(Uri uri, Bundle extras, List<Bundle> otherLikelyBundles) { | ||||
|         if (mClient == null) return false; | ||||
|  | ||||
|         CustomTabsSession session = getSession(); | ||||
|         return session != null && session.mayLaunchUrl(uri, extras, otherLikelyBundles); | ||||
|  | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onServiceConnected(CustomTabsClient client) { | ||||
|         mClient = client; | ||||
|         mClient.warmup(0L); | ||||
|         if (mConnectionCallback != null) mConnectionCallback.onCustomTabsConnected(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onServiceDisconnected() { | ||||
|         mClient = null; | ||||
|         mCustomTabsSession = null; | ||||
|         if (mConnectionCallback != null) mConnectionCallback.onCustomTabsDisconnected(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * A Callback for when the service is connected or disconnected. Use those callbacks to | ||||
|      * handle UI changes when the service is connected or disconnected. | ||||
|      */ | ||||
|     public interface ConnectionCallback { | ||||
|         /** | ||||
|          * Called when the service is connected. | ||||
|          */ | ||||
|         void onCustomTabsConnected(); | ||||
|  | ||||
|         /** | ||||
|          * Called when the service is disconnected. | ||||
|          */ | ||||
|         void onCustomTabsDisconnected(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * To be used as a fallback to open the Uri when Custom Tabs is not available. | ||||
|      */ | ||||
|     public interface CustomTabFallback { | ||||
|         /** | ||||
|          * @param activity The Activity that wants to open the Uri. | ||||
|          * @param uri      The uri to be opened by the fallback. | ||||
|          */ | ||||
|         void openUri(Activity activity, Uri uri); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,129 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.content.Intent; | ||||
| import android.content.IntentFilter; | ||||
| import android.content.pm.PackageManager; | ||||
| import android.content.pm.ResolveInfo; | ||||
| import android.net.Uri; | ||||
| import android.text.TextUtils; | ||||
| import android.util.Log; | ||||
| import androidx.browser.customtabs.CustomTabsService; | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.helpers.KeepAliveService; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
|  | ||||
| @SuppressWarnings("ALL") | ||||
| class CustomTabsHelper { | ||||
|     private static final String TAG = "CustomTabsHelper"; | ||||
|     private static final String STABLE_PACKAGE = "com.android.chrome"; | ||||
|     private static final String BETA_PACKAGE = "com.chrome.beta"; | ||||
|     private static final String DEV_PACKAGE = "com.chrome.dev"; | ||||
|     private static final String LOCAL_PACKAGE = "com.google.android.apps.chrome"; | ||||
|     private static final String EXTRA_CUSTOM_TABS_KEEP_ALIVE = | ||||
|             "android.support.customtabs.extra.KEEP_ALIVE"; | ||||
|  | ||||
|     private static String sPackageNameToUse; | ||||
|  | ||||
|     private CustomTabsHelper() { | ||||
|     } | ||||
|  | ||||
|     public static void addKeepAliveExtra(Context context, Intent intent) { | ||||
|         Intent keepAliveIntent = new Intent().setClassName( | ||||
|                 context.getPackageName(), KeepAliveService.class.getCanonicalName()); | ||||
|         intent.putExtra(EXTRA_CUSTOM_TABS_KEEP_ALIVE, keepAliveIntent); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Goes through all apps that handle VIEW intents and have a warmup service. Picks | ||||
|      * the one chosen by the user if there is one, otherwise makes a best effort to return a | ||||
|      * valid package name. | ||||
|      * <p> | ||||
|      * This is <strong>not</strong> threadsafe. | ||||
|      * | ||||
|      * @param context {@link Context} to use for accessing {@link PackageManager}. | ||||
|      * @return The package name recommended to use for connecting to custom tabs related components. | ||||
|      */ | ||||
|     public static String getPackageNameToUse(Context context) { | ||||
|         if (sPackageNameToUse != null) return sPackageNameToUse; | ||||
|  | ||||
|         PackageManager pm = context.getPackageManager(); | ||||
|         // Get default VIEW intent handler. | ||||
|         Intent activityIntent = new Intent(Intent.ACTION_VIEW, Uri.parse("http://www.example.com")); | ||||
|         ResolveInfo defaultViewHandlerInfo = pm.resolveActivity(activityIntent, 0); | ||||
|         String defaultViewHandlerPackageName = null; | ||||
|         if (defaultViewHandlerInfo != null) { | ||||
|             defaultViewHandlerPackageName = defaultViewHandlerInfo.activityInfo.packageName; | ||||
|         } | ||||
|  | ||||
|         // Get all apps that can handle VIEW intents. | ||||
|         List<ResolveInfo> resolvedActivityList = pm.queryIntentActivities(activityIntent, 0); | ||||
|         List<String> packagesSupportingCustomTabs = new ArrayList<>(); | ||||
|         for (ResolveInfo info : resolvedActivityList) { | ||||
|             Intent serviceIntent = new Intent(); | ||||
|             serviceIntent.setAction(CustomTabsService.ACTION_CUSTOM_TABS_CONNECTION); | ||||
|             serviceIntent.setPackage(info.activityInfo.packageName); | ||||
|             if (pm.resolveService(serviceIntent, 0) != null) { | ||||
|                 packagesSupportingCustomTabs.add(info.activityInfo.packageName); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Now packagesSupportingCustomTabs contains all apps that can handle both VIEW intents | ||||
|         // and service calls. | ||||
|         if (packagesSupportingCustomTabs.isEmpty()) { | ||||
|             sPackageNameToUse = null; | ||||
|         } else if (packagesSupportingCustomTabs.size() == 1) { | ||||
|             sPackageNameToUse = packagesSupportingCustomTabs.get(0); | ||||
|         } else if (!TextUtils.isEmpty(defaultViewHandlerPackageName) | ||||
|                 && !hasSpecializedHandlerIntents(context, activityIntent) | ||||
|                 && packagesSupportingCustomTabs.contains(defaultViewHandlerPackageName)) { | ||||
|             sPackageNameToUse = defaultViewHandlerPackageName; | ||||
|         } else if (packagesSupportingCustomTabs.contains(STABLE_PACKAGE)) { | ||||
|             sPackageNameToUse = STABLE_PACKAGE; | ||||
|         } else if (packagesSupportingCustomTabs.contains(BETA_PACKAGE)) { | ||||
|             sPackageNameToUse = BETA_PACKAGE; | ||||
|         } else if (packagesSupportingCustomTabs.contains(DEV_PACKAGE)) { | ||||
|             sPackageNameToUse = DEV_PACKAGE; | ||||
|         } else if (packagesSupportingCustomTabs.contains(LOCAL_PACKAGE)) { | ||||
|             sPackageNameToUse = LOCAL_PACKAGE; | ||||
|         } | ||||
|         return sPackageNameToUse; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Used to check whether there is a specialized handler for a given intent. | ||||
|      * | ||||
|      * @param intent The intent to check with. | ||||
|      * @return Whether there is a specialized handler for the given intent. | ||||
|      */ | ||||
|     private static boolean hasSpecializedHandlerIntents(Context context, Intent intent) { | ||||
|         try { | ||||
|             PackageManager pm = context.getPackageManager(); | ||||
|             List<ResolveInfo> handlers = pm.queryIntentActivities( | ||||
|                     intent, | ||||
|                     PackageManager.GET_RESOLVED_FILTER); | ||||
|             if (handlers == null || handlers.isEmpty()) { | ||||
|                 return false; | ||||
|             } | ||||
|             for (ResolveInfo resolveInfo : handlers) { | ||||
|                 IntentFilter filter = resolveInfo.filter; | ||||
|                 if (filter == null) continue; | ||||
|                 if (filter.countDataAuthorities() == 0 || filter.countDataPaths() == 0) continue; | ||||
|                 if (resolveInfo.activityInfo == null) continue; | ||||
|                 return true; | ||||
|             } | ||||
|         } catch (RuntimeException e) { | ||||
|             Log.e(TAG, "Runtime exception while getting specialized handlers"); | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return All possible chrome package names that provide custom tabs feature. | ||||
|      */ | ||||
|     public static String[] getPackages() { | ||||
|         return new String[]{"", STABLE_PACKAGE, BETA_PACKAGE, DEV_PACKAGE, LOCAL_PACKAGE}; | ||||
|     } | ||||
| } | ||||
| @@ -1,33 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import android.content.ComponentName; | ||||
| import androidx.browser.customtabs.CustomTabsClient; | ||||
| import androidx.browser.customtabs.CustomTabsServiceConnection; | ||||
|  | ||||
| import java.lang.ref.WeakReference; | ||||
|  | ||||
| /** | ||||
|  * Implementation for the CustomTabsServiceConnection that avoids leaking the | ||||
|  * ServiceConnectionCallback | ||||
|  */ | ||||
| public class ServiceConnection extends CustomTabsServiceConnection { | ||||
|     // A weak reference to the ServiceConnectionCallback to avoid leaking it. | ||||
|     private WeakReference<ServiceConnectionCallback> mConnectionCallback; | ||||
|  | ||||
|     public ServiceConnection(ServiceConnectionCallback connectionCallback) { | ||||
|         mConnectionCallback = new WeakReference<>(connectionCallback); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onCustomTabsServiceConnected(ComponentName name, CustomTabsClient client) { | ||||
|         ServiceConnectionCallback connectionCallback = mConnectionCallback.get(); | ||||
|         if (connectionCallback != null) connectionCallback.onServiceConnected(client); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onServiceDisconnected(ComponentName name) { | ||||
|         ServiceConnectionCallback connectionCallback = mConnectionCallback.get(); | ||||
|         if (connectionCallback != null) connectionCallback.onServiceDisconnected(); | ||||
|     } | ||||
| } | ||||
| @@ -1,19 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import androidx.browser.customtabs.CustomTabsClient; | ||||
|  | ||||
|  | ||||
| public interface ServiceConnectionCallback { | ||||
|     /** | ||||
|      * Called when the service is connected. | ||||
|      * | ||||
|      * @param client a CustomTabsClient | ||||
|      */ | ||||
|     void onServiceConnected(CustomTabsClient client); | ||||
|  | ||||
|     /** | ||||
|      * Called when the service is disconnected. | ||||
|      */ | ||||
|     void onServiceDisconnected(); | ||||
| } | ||||
| @@ -1,15 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs.helpers; | ||||
|  | ||||
| import android.app.Service; | ||||
| import android.content.Intent; | ||||
| import android.os.Binder; | ||||
| import android.os.IBinder; | ||||
|  | ||||
| public class KeepAliveService extends Service { | ||||
|     private static final Binder sBinder = new Binder(); | ||||
|  | ||||
|     @Override | ||||
|     public IBinder onBind(Intent intent) { | ||||
|         return sBinder; | ||||
|     } | ||||
| } | ||||
| @@ -1,14 +0,0 @@ | ||||
| /* From https://github.com/mikepenz/MaterialDrawer/blob/develop/app/src/main/java/com/mikepenz/materialdrawer/app/drawerItems/CustomBaseViewHolder.java */ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.drawer | ||||
|  | ||||
| import android.view.View | ||||
| import android.widget.ImageView | ||||
| import android.widget.TextView | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
|  | ||||
| open class CustomBaseViewHolder(var view: View) : RecyclerView.ViewHolder(view) { | ||||
|     var icon: ImageView = view.findViewById(R.id.material_drawer_icon) | ||||
|     var name: TextView = view.findViewById(R.id.material_drawer_name) | ||||
|     var description: TextView = view.findViewById(R.id.material_drawer_description) | ||||
| } | ||||
| @@ -3,38 +3,37 @@ package bou.amine.apps.readerforselfossv2.android.utils.glide | ||||
| import android.content.Context | ||||
| import android.graphics.Bitmap | ||||
| import android.widget.ImageView | ||||
| import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import com.bumptech.glide.request.target.BitmapImageViewTarget | ||||
| import java.io.ByteArrayInputStream | ||||
| import java.io.ByteArrayOutputStream | ||||
| import java.io.InputStream | ||||
|  | ||||
| fun Context.bitmapCenterCrop(url: String, iv: ImageView) = | ||||
|     Glide.with(this) | ||||
| fun Context.bitmapCenterCrop( | ||||
|     url: String, | ||||
|     iv: ImageView, | ||||
| ) = Glide.with(this) | ||||
|     .asBitmap() | ||||
|     .load(url) | ||||
|     .apply(RequestOptions.centerCropTransform()) | ||||
|     .into(iv) | ||||
|  | ||||
| fun Context.circularBitmapDrawable(url: String, iv: ImageView) = | ||||
|     Glide.with(this) | ||||
|         .asBitmap() | ||||
|         .load(url) | ||||
|         .apply(RequestOptions.centerCropTransform()) | ||||
|         .into(object : BitmapImageViewTarget(iv) { | ||||
|             override fun setResource(resource: Bitmap?) { | ||||
|                 val circularBitmapDrawable = RoundedBitmapDrawableFactory.create( | ||||
|                     resources, | ||||
|                     resource | ||||
|                 ) | ||||
|                 circularBitmapDrawable.isCircular = true | ||||
|                 iv.setImageDrawable(circularBitmapDrawable) | ||||
|             } | ||||
|         }) | ||||
| fun Context.circularDrawable( | ||||
|     url: String, | ||||
|     view: CircleImageView, | ||||
| ) { | ||||
|     view.textView.text = "" | ||||
|  | ||||
| fun getBitmapInputStream(bitmap:Bitmap,compressFormat: Bitmap.CompressFormat): InputStream { | ||||
|     Glide.with(this) | ||||
|         .load(url) | ||||
|         .into(view.imageView) | ||||
| } | ||||
|  | ||||
| fun getBitmapInputStream( | ||||
|     bitmap: Bitmap, | ||||
|     compressFormat: Bitmap.CompressFormat, | ||||
| ): InputStream { | ||||
|     val byteArrayOutputStream = ByteArrayOutputStream() | ||||
|     bitmap.compress(compressFormat, 80, byteArrayOutputStream) | ||||
|     val bitmapData: ByteArray = byteArrayOutputStream.toByteArray() | ||||
|   | ||||
| @@ -19,7 +19,8 @@ class AppViewModel(private val repository: Repository) : ViewModel() { | ||||
|                     if (isConnected && !wasConnected && repository.connectionMonitored) { | ||||
|                         _networkAvailableProvider.emit(true) | ||||
|                         wasConnected = true | ||||
|                     } else if (!isConnected && wasConnected && repository.connectionMonitored){ | ||||
|                     } else if (!isConnected && wasConnected && repository.connectionMonitored) | ||||
|                         { | ||||
|                             _networkAvailableProvider.emit(false) | ||||
|                             wasConnected = false | ||||
|                         } | ||||
|   | ||||
| @@ -1,16 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Copyright 2015 Google Inc. All Rights Reserved. | ||||
|      Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|      you may not use this file except in compliance with the License. | ||||
|      You may obtain a copy of the License at | ||||
|          http://www.apache.org/licenses/LICENSE-2.0 | ||||
|      Unless required by applicable law or agreed to in writing, software | ||||
|      distributed under the License is distributed on an "AS IS" BASIS, | ||||
|      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|      See the License for the specific language governing permissions and | ||||
|      limitations under the License. | ||||
| --> | ||||
| <set xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <translate android:fromXDelta="100%p" android:toXDelta="0" | ||||
|                android:duration="@android:integer/config_mediumAnimTime"/> | ||||
| </set> | ||||
| @@ -1,16 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <!-- Copyright 2015 Google Inc. All Rights Reserved. | ||||
|      Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|      you may not use this file except in compliance with the License. | ||||
|      You may obtain a copy of the License at | ||||
|          http://www.apache.org/licenses/LICENSE-2.0 | ||||
|      Unless required by applicable law or agreed to in writing, software | ||||
|      distributed under the License is distributed on an "AS IS" BASIS, | ||||
|      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|      See the License for the specific language governing permissions and | ||||
|      limitations under the License. | ||||
| --> | ||||
| <set xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <translate android:fromXDelta="0" android:toXDelta="-100%p" | ||||
|                android:duration="@android:integer/config_mediumAnimTime"/> | ||||
| </set> | ||||
| @@ -1,8 +1,11 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <layer-list xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|  | ||||
|     <item | ||||
|         android:drawable="@color/ic_launcher_background"/> | ||||
|     <item> | ||||
|         <shape android:shape="rectangle" > | ||||
|             <solid android:color="?attr/colorSurface" /> | ||||
|         </shape> | ||||
|     </item> | ||||
|  | ||||
|     <item> | ||||
|         <bitmap | ||||
|   | ||||
							
								
								
									
										5
									
								
								androidApp/src/main/res/drawable/checkerboard.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								androidApp/src/main/res/drawable/checkerboard.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| <bitmap | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     android:dither="true" | ||||
|     android:src="@drawable/checktile" | ||||
|     android:tileMode="repeat"/> | ||||
							
								
								
									
										
											BIN
										
									
								
								androidApp/src/main/res/drawable/checktile.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								androidApp/src/main/res/drawable/checktile.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 235 B | 
| @@ -0,0 +1,5 @@ | ||||
| <vector android:height="24dp" android:tint="#000000" | ||||
|     android:viewportHeight="24" android:viewportWidth="24" | ||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <path android:fillColor="@android:color/white" android:pathData="M20,8h-2.81c-0.45,-0.78 -1.07,-1.45 -1.82,-1.96L17,4.41 15.59,3l-2.17,2.17C12.96,5.06 12.49,5 12,5c-0.49,0 -0.96,0.06 -1.41,0.17L8.41,3 7,4.41l1.62,1.63C7.88,6.55 7.26,7.22 6.81,8L4,8v2h2.09c-0.05,0.33 -0.09,0.66 -0.09,1v1L4,12v2h2v1c0,0.34 0.04,0.67 0.09,1L4,16v2h2.81c1.04,1.79 2.97,3 5.19,3s4.15,-1.21 5.19,-3L20,18v-2h-2.09c0.05,-0.33 0.09,-0.66 0.09,-1v-1h2v-2h-2v-1c0,-0.34 -0.04,-0.67 -0.09,-1L20,10L20,8zM14,16h-4v-2h4v2zM14,12h-4v-2h4v2z"/> | ||||
| </vector> | ||||
| @@ -0,0 +1,5 @@ | ||||
| <vector android:height="24dp" android:tint="#FFFFFF" | ||||
|     android:viewportHeight="24" android:viewportWidth="24" | ||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <path android:fillColor="@android:color/white" android:pathData="M4.25,5.61C6.27,8.2 10,13 10,13v6c0,0.55 0.45,1 1,1h2c0.55,0 1,-0.45 1,-1v-6c0,0 3.72,-4.8 5.74,-7.39C20.25,4.95 19.78,4 18.95,4H5.04C4.21,4 3.74,4.95 4.25,5.61z"/> | ||||
| </vector> | ||||
| @@ -1,9 +0,0 @@ | ||||
| <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|         android:width="24dp" | ||||
|         android:height="24dp" | ||||
|         android:viewportWidth="24.0" | ||||
|         android:viewportHeight="24.0"> | ||||
|     <path | ||||
|         android:fillColor="#FF000000" | ||||
|         android:pathData="M20,8h-2.81c-0.45,-0.78 -1.07,-1.45 -1.82,-1.96L17,4.41 15.59,3l-2.17,2.17C12.96,5.06 12.49,5 12,5c-0.49,0 -0.96,0.06 -1.41,0.17L8.41,3 7,4.41l1.62,1.63C7.88,6.55 7.26,7.22 6.81,8L4,8v2h2.09c-0.05,0.33 -0.09,0.66 -0.09,1v1L4,12v2h2v1c0,0.34 0.04,0.67 0.09,1L4,16v2h2.81c1.04,1.79 2.97,3 5.19,3s4.15,-1.21 5.19,-3L20,18v-2h-2.09c0.05,-0.33 0.09,-0.66 0.09,-1v-1h2v-2h-2v-1c0,-0.34 -0.04,-0.67 -0.09,-1L20,10L20,8zM14,16h-4v-2h4v2zM14,12h-4v-2h4v2z"/> | ||||
| </vector> | ||||
| @@ -1,9 +0,0 @@ | ||||
| <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|         android:width="24dp" | ||||
|         android:height="24dp" | ||||
|         android:viewportWidth="24.0" | ||||
|         android:viewportHeight="24.0"> | ||||
|     <path | ||||
|         android:fillColor="#FFFFFFFF" | ||||
|         android:pathData="M13,12h7v1.5h-7zM13,9.5h7L20,11h-7zM13,14.5h7L20,16h-7zM21,4L3,4c-1.1,0 -2,0.9 -2,2v13c0,1.1 0.9,2 2,2h18c1.1,0 2,-0.9 2,-2L23,6c0,-1.1 -0.9,-2 -2,-2zM21,19h-9L12,6h9v13z"/> | ||||
| </vector> | ||||
| @@ -1,5 +0,0 @@ | ||||
| <vector android:height="24dp" android:tint="#FFFFFF" | ||||
|     android:viewportHeight="24.0" android:viewportWidth="24.0" | ||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <path android:fillColor="#FF000000" android:pathData="M19,6.41L17.59,5 12,10.59 6.41,5 5,6.41 10.59,12 5,17.59 6.41,19 12,13.41 17.59,19 19,17.59 13.41,12z"/> | ||||
| </vector> | ||||
| @@ -1,5 +0,0 @@ | ||||
| <vector android:height="24dp" android:tint="#FFFFFF" | ||||
|     android:viewportHeight="24.0" android:viewportWidth="24.0" | ||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <path android:fillColor="#FF000000" android:pathData="M13,3c-4.97,0 -9,4.03 -9,9L1,12l3.89,3.89 0.07,0.14L9,12L6,12c0,-3.87 3.13,-7 7,-7s7,3.13 7,7 -3.13,7 -7,7c-1.93,0 -3.68,-0.79 -4.94,-2.06l-1.42,1.42C8.27,19.99 10.51,21 13,21c4.97,0 9,-4.03 9,-9s-4.03,-9 -9,-9zM12,8v5l4.28,2.54 0.72,-1.21 -3.5,-2.08L13.5,8L12,8z"/> | ||||
| </vector> | ||||
| @@ -1,5 +0,0 @@ | ||||
| <vector android:height="24dp" android:tint="#FFFFFF" | ||||
|     android:viewportHeight="24.0" android:viewportWidth="24.0" | ||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <path android:fillColor="#FF000000" android:pathData="M17.65,6.35C16.2,4.9 14.21,4 12,4c-4.42,0 -7.99,3.58 -7.99,8s3.57,8 7.99,8c3.73,0 6.84,-2.55 7.73,-6h-2.08c-0.82,2.33 -3.04,4 -5.65,4 -3.31,0 -6,-2.69 -6,-6s2.69,-6 6,-6c1.66,0 3.14,0.69 4.22,1.78L13,11h7V4l-2.35,2.35z"/> | ||||
| </vector> | ||||
| @@ -1,7 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <font-family xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|         app:fontProviderAuthority="com.google.android.gms.fonts" | ||||
|         app:fontProviderPackage="com.google.android.gms" | ||||
|         app:fontProviderQuery="Open Sans" | ||||
|         app:fontProviderCerts="@array/com_google_android_gms_fonts_certs"> | ||||
| </font-family> | ||||
| @@ -1,7 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <font-family xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|         app:fontProviderAuthority="com.google.android.gms.fonts" | ||||
|         app:fontProviderPackage="com.google.android.gms" | ||||
|         app:fontProviderQuery="Roboto" | ||||
|         app:fontProviderCerts="@array/com_google_android_gms_fonts_certs"> | ||||
| </font-family> | ||||
| @@ -1,13 +1,12 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.drawerlayout.widget.DrawerLayout | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.drawerlayout.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:id="@+id/drawerContainer" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     tools:context="bou.amine.apps.readerforselfossv2.android.HomeActivity" | ||||
|     android:fitsSystemWindows="true" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||
|     tools:context="bou.amine.apps.readerforselfossv2.android.HomeActivity"> | ||||
|  | ||||
|     <androidx.coordinatorlayout.widget.CoordinatorLayout | ||||
|         android:id="@+id/coordLayout" | ||||
| @@ -32,8 +31,10 @@ | ||||
|                         android:id="@+id/toolBar" | ||||
|                         android:layout_width="match_parent" | ||||
|                         android:layout_height="?attr/actionBarSize" | ||||
|                         app:theme="@style/ToolBarStyle" | ||||
|                         app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|                         android:theme="@style/ToolBarStyle" | ||||
|                         app:popupTheme="?attr/toolbarPopupTheme" | ||||
|  | ||||
|                         /> | ||||
|  | ||||
|                 </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
| @@ -45,19 +46,19 @@ | ||||
|                     <LinearLayout | ||||
|                         android:layout_width="match_parent" | ||||
|                         android:layout_height="match_parent" | ||||
|                         android:orientation="vertical" | ||||
|                         android:background="?android:attr/windowBackground"> | ||||
|                         android:background="?android:attr/windowBackground" | ||||
|                         android:orientation="vertical"> | ||||
|  | ||||
|                         <TextView | ||||
|                             android:id="@+id/emptyText" | ||||
|                             android:layout_width="match_parent" | ||||
|                             android:layout_height="wrap_content" | ||||
|                             android:background="@android:color/transparent" | ||||
|                             android:gravity="center_horizontal" | ||||
|                             android:paddingTop="100dp" | ||||
|                             android:text="@string/nothing_here" | ||||
|                             android:textAlignment="center" | ||||
|                             android:textAppearance="@style/TextAppearance.AppCompat.Headline" | ||||
|                             android:background="@android:color/transparent" | ||||
|                             android:visibility="gone" /> | ||||
|  | ||||
|                         <androidx.recyclerview.widget.RecyclerView | ||||
| @@ -77,17 +78,13 @@ | ||||
|             </LinearLayout> | ||||
|  | ||||
|         </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|  | ||||
|         <com.ashokvarma.bottomnavigation.BottomNavigationBar | ||||
|             android:layout_gravity="bottom" | ||||
|             android:id="@+id/bottomBar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="60dp"/> | ||||
|             android:layout_height="60dp" | ||||
|             android:layout_gravity="bottom" | ||||
|             app:bnbActiveColor="@color/colorAccent" | ||||
|             app:bnbBackgroundColor="?attr/bottomBarBackground" /> | ||||
|     </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|  | ||||
|     <com.mikepenz.materialdrawer.widget.MaterialDrawerSliderView | ||||
|         android:id="@+id/mainDrawer" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="match_parent" | ||||
|         android:layout_gravity="start" | ||||
|         android:fitsSystemWindows="true" /> | ||||
| </androidx.drawerlayout.widget.DrawerLayout> | ||||
| @@ -1,33 +1,40 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.constraintlayout.motion.widget.MotionLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:id="@+id/container" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
|     android:layout_height="match_parent" | ||||
|     app:layoutDescription="@xml/image_close_scene"> | ||||
|  | ||||
|     <com.google.android.material.appbar.AppBarLayout | ||||
|         android:id="@+id/appBarLayout" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent"> | ||||
|         android:layout_height="wrap_content"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|         <androidx.appcompat.widget.Toolbar android:theme="@style/ToolBarStyle" | ||||
|             android:id="@+id/toolBar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" | ||||
|             app:theme="@style/ToolBarStyle" /> | ||||
|  | ||||
|             /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:id="@+id/scrollView" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:fillViewport="true" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/appBarLayout"> | ||||
|  | ||||
|         <androidx.viewpager2.widget.ViewPager2 | ||||
|             android:id="@+id/pager" | ||||
|             android:layout_width="match_parent" | ||||
|         android:layout_height="0dp" | ||||
|             android:layout_height="match_parent" | ||||
|             app:layout_constraintBottom_toBottomOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/appBarLayout" /> | ||||
|             app:layout_constraintStart_toStartOf="parent" /> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
|  | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|  | ||||
| </androidx.constraintlayout.motion.widget.MotionLayout> | ||||
|   | ||||
| @@ -1,11 +1,12 @@ | ||||
| <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:gravity="center_horizontal" | ||||
|     android:orientation="vertical" | ||||
|     tools:context="bou.amine.apps.readerforselfossv2.android.LoginActivity"> | ||||
|  | ||||
|     <com.google.android.material.appbar.AppBarLayout | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
| @@ -14,18 +15,16 @@ | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             app:theme="@style/ToolBarStyle" | ||||
|             android:theme="@style/ToolBarStyle" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|     <LinearLayout | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:orientation="vertical" | ||||
|         android:paddingBottom="@dimen/activity_vertical_margin" | ||||
|         android:paddingLeft="@dimen/activity_horizontal_margin" | ||||
|         android:paddingRight="@dimen/activity_horizontal_margin" | ||||
|         android:paddingTop="@dimen/activity_vertical_margin"> | ||||
|         android:padding="@dimen/activity_horizontal_margin"> | ||||
|         <!-- Login progress --> | ||||
|         <ProgressBar | ||||
|             android:id="@+id/loginProgress" | ||||
| @@ -35,12 +34,8 @@ | ||||
|             android:layout_marginBottom="8dp" | ||||
|             android:visibility="gone" /> | ||||
|  | ||||
|         <ScrollView | ||||
|             android:id="@+id/loginForm" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent"> | ||||
|  | ||||
|         <LinearLayout | ||||
|             android:id="@+id/loginForm" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:orientation="vertical"> | ||||
| @@ -53,14 +48,22 @@ | ||||
|                 android:imeOptions="actionUnspecified" | ||||
|                 android:importantForAutofill="no" | ||||
|                 android:inputType="textUri" | ||||
|                     android:maxLines="1" /> | ||||
|                 android:maxLines="1" | ||||
|                 android:minHeight="48dp" /> | ||||
|  | ||||
|             <com.google.android.material.switchmaterial.SwitchMaterial | ||||
|                     android:text="@string/withLoginSwitch" | ||||
|                 android:id="@+id/selfSigned" | ||||
|                 android:layout_width="match_parent" | ||||
|                     android:layout_height="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:text="@string/disable_ssl" | ||||
|                 android:textAlignment="viewStart" /> | ||||
|  | ||||
|             <com.google.android.material.switchmaterial.SwitchMaterial | ||||
|                 android:id="@+id/withLogin" | ||||
|                     android:layout_weight="1"/> | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:text="@string/withLoginSwitch" | ||||
|                 android:textAlignment="viewStart" /> | ||||
|  | ||||
|             <EditText | ||||
|                 android:id="@+id/loginView" | ||||
| @@ -70,6 +73,7 @@ | ||||
|                 android:hint="@string/prompt_login" | ||||
|                 android:inputType="text" | ||||
|                 android:maxLines="1" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:visibility="gone" /> | ||||
|  | ||||
|             <EditText | ||||
| @@ -80,6 +84,7 @@ | ||||
|                 android:hint="@string/prompt_password" | ||||
|                 android:inputType="textPassword" | ||||
|                 android:maxLines="1" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:visibility="gone" /> | ||||
|  | ||||
|             <Button | ||||
| @@ -93,7 +98,6 @@ | ||||
|                 android:textStyle="bold" /> | ||||
|  | ||||
|         </LinearLayout> | ||||
|         </ScrollView> | ||||
|     </LinearLayout> | ||||
|  | ||||
| </LinearLayout> | ||||
|   | ||||
| @@ -17,8 +17,10 @@ | ||||
|             android:id="@+id/toolBar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             android:theme="@style/ToolBarStyle" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" | ||||
|             app:theme="@style/ToolBarStyle" /> | ||||
|  | ||||
|             /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:id="@+id/layout" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
| @@ -8,11 +9,11 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|         <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             android:theme="@style/ToolBarStyle" /> | ||||
|              /> | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|     <FrameLayout | ||||
|   | ||||
| @@ -10,12 +10,12 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|         <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             app:theme="@style/ToolBarStyle" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|  | ||||
|              /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
| @@ -24,7 +24,8 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:scrollbars="vertical" | ||||
|         app:layout_behavior="@string/appbar_scrolling_view_behavior"> | ||||
|         app:layout_behavior="@string/appbar_scrolling_view_behavior" | ||||
|         tools:listitem="@layout/source_list_item"> | ||||
|     </androidx.recyclerview.widget.RecyclerView> | ||||
|  | ||||
|     <com.google.android.material.floatingactionbutton.FloatingActionButton | ||||
|   | ||||
| @@ -4,7 +4,7 @@ | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     tools:context="bou.amine.apps.readerforselfossv2.android.AddSourceActivity"> | ||||
|     tools:context="bou.amine.apps.readerforselfossv2.android.UpsertSourceActivity"> | ||||
|     <LinearLayout | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content" | ||||
| @@ -14,120 +14,86 @@ | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content"> | ||||
| 
 | ||||
|             <androidx.appcompat.widget.Toolbar | ||||
|             <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|                 android:id="@+id/toolbar" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="?attr/actionBarSize" | ||||
|                 app:theme="@style/ToolBarStyle" | ||||
|                 app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|                 android:layout_height="?attr/actionBarSize" /> | ||||
| 
 | ||||
|         </com.google.android.material.appbar.AppBarLayout> | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:paddingBottom="@dimen/activity_vertical_margin" | ||||
|             android:paddingLeft="@dimen/activity_horizontal_margin" | ||||
|             android:paddingRight="@dimen/activity_horizontal_margin" | ||||
|             android:paddingTop="@dimen/activity_vertical_margin" | ||||
|             android:layout_height="match_parent" | ||||
|             android:layout_width="match_parent" | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
|             app:layout_constraintRight_toRightOf="parent" | ||||
|             app:layout_constraintLeft_toLeftOf="parent" | ||||
|             app:layout_constraintBottom_toBottomOf="parent" | ||||
|             android:id="@+id/formContainer" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent" | ||||
|             android:padding="16dp" | ||||
|             android:visibility="gone" | ||||
|             app:layout_constraintHorizontal_bias="1.0" | ||||
|             app:layout_constraintVertical_bias="0.0"> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:text="@string/add_source" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:id="@+id/textView2" | ||||
|                 android:textAppearance="@style/TextAppearance.AppCompat.Large" | ||||
|                 android:textAlignment="center" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 android:layout_marginEnd="16dp" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:layout_marginStart="16dp" | ||||
|             app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 android:layout_marginLeft="16dp" | ||||
|                 android:gravity="center_horizontal" /> | ||||
|             app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
| 
 | ||||
|             <EditText | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:ems="10" | ||||
|                 android:id="@+id/nameInput" | ||||
|                 android:layout_marginTop="32dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/textView2" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 android:inputType="text" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:autofillHints="false" | ||||
|                 android:hint="@string/add_source_hint_name" | ||||
|                 android:textColorHint="?android:textColorPrimary" | ||||
|                 android:autofillHints="false" /> | ||||
| 
 | ||||
|             <EditText | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:inputType="textUri" | ||||
|                 android:ems="10" | ||||
|                 android:id="@+id/sourceUri" | ||||
|                 android:hint="@string/add_source_hint_url" | ||||
|                 android:textColorHint="?android:textColorPrimary" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/nameInput" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 android:autofillHints="false" /> | ||||
| 
 | ||||
|             <EditText | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:ems="10" | ||||
|                 android:id="@+id/tags" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/sourceUri" | ||||
|                 android:hint="@string/add_source_hint_tags" | ||||
|                 android:textColorHint="?android:textColorPrimary" | ||||
|                 android:inputType="text" | ||||
|                 android:autofillHints="false" /> | ||||
|                 android:textColorHint="?android:textColorPrimary" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|             <EditText | ||||
|                 android:id="@+id/sourceUri" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:autofillHints="false" | ||||
|                 android:hint="@string/add_source_hint_url" | ||||
|                 android:inputType="textUri" | ||||
|                 android:textColorHint="?android:textColorPrimary" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/nameInput" /> | ||||
| 
 | ||||
|             <EditText | ||||
|                 android:id="@+id/tags" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:autofillHints="false" | ||||
|                 android:hint="@string/add_source_hint_tags" | ||||
|                 android:inputType="text" | ||||
|                 android:textColorHint="?android:textColorPrimary" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/sourceUri" /> | ||||
| 
 | ||||
|             <Spinner | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:id="@+id/spoutsSpinner" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/tags" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 android:layout_height="40dp" | ||||
|                 android:theme="@style/App.Spinner"/> | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/tags" /> | ||||
| 
 | ||||
|             <Button | ||||
|                 android:text="@string/add_source_save" | ||||
|                 android:id="@+id/saveBtn" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:id="@+id/saveBtn" | ||||
|                 android:elevation="5dp" | ||||
|                 android:textColor="?attr/colorAccent" | ||||
|                 android:layout_marginEnd="16dp" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:layout_marginStart="16dp" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 android:layout_marginLeft="16dp" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/spoutsSpinner" | ||||
|                 android:elevation="5dp" | ||||
|                 android:text="@string/add_source_save" | ||||
|                 android:textColor="?android:textColorPrimary" | ||||
|                 app:layout_constraintBottom_toBottomOf="parent" | ||||
|                 android:layout_marginBottom="16dp" | ||||
|                 app:layout_constraintVertical_bias="0.0"/> | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/spoutsSpinner" /> | ||||
| 
 | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
| @@ -136,8 +102,6 @@ | ||||
|             style="?android:attr/progressBarStyleLarge" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:visibility="visible" | ||||
|             app:layout_constraintBottom_toBottomOf="parent" | ||||
|             app:layout_constraintLeft_toLeftOf="parent" | ||||
|             app:layout_constraintRight_toRightOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
| @@ -1,23 +1,18 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.cardview.widget.CardView | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:card_view="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:id="@+id/card" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="wrap_content" | ||||
|     android:layout_marginLeft="8dp" | ||||
|     android:layout_marginRight="8dp" | ||||
|     android:layout_marginTop="8dp" | ||||
|     app:layout_constraintHorizontal_bias="0.62" | ||||
|     app:layout_constraintLeft_toLeftOf="parent" | ||||
|     app:layout_constraintRight_toRightOf="parent" | ||||
|     android:layout_margin="8dp" | ||||
|     app:layout_constraintEnd_toEndOf="parent" | ||||
|     app:layout_constraintStart_toStartOf="parent" | ||||
|     app:layout_constraintTop_toTopOf="parent" | ||||
|     card_view:cardElevation="2dp" | ||||
|     card_view:cardUseCompatPadding="true" | ||||
|     card_view:layout_constraintBottom_toBottomOf="parent" | ||||
|     app:cardBackgroundColor="?cardBackgroundColor"> | ||||
|     card_view:layout_constraintBottom_toBottomOf="parent"> | ||||
|  | ||||
|     <androidx.constraintlayout.widget.ConstraintLayout | ||||
|         android:layout_width="match_parent" | ||||
| @@ -29,8 +24,8 @@ | ||||
|             android:layout_height="wrap_content" | ||||
|             android:adjustViewBounds="true" | ||||
|             android:cropToPadding="true" | ||||
|             app:layout_constraintLeft_toLeftOf="parent" | ||||
|             app:layout_constraintRight_toRightOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
|             app:srcCompat="@drawable/background_splash" | ||||
|             card_view:layout_constraintBottom_toTopOf="@+id/constraintLayout" /> | ||||
| @@ -40,18 +35,17 @@ | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             app:layout_constraintBottom_toBottomOf="parent" | ||||
|             app:layout_constraintLeft_toLeftOf="parent" | ||||
|             app:layout_constraintRight_toRightOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/itemImage"> | ||||
|  | ||||
|             <ImageView | ||||
|             <bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||
|                 android:id="@+id/sourceImage" | ||||
|                 android:layout_width="40dp" | ||||
|                 android:layout_height="40dp" | ||||
|                 android:layout_marginLeft="8dp" | ||||
|                 android:layout_marginStart="8dp" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 app:srcCompat="@drawable/background_splash" /> | ||||
|  | ||||
| @@ -59,70 +53,63 @@ | ||||
|                 android:id="@+id/title" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginEnd="8dp" | ||||
|                 android:layout_marginLeft="8dp" | ||||
|                 android:layout_marginRight="8dp" | ||||
|                 android:layout_margin="8dp" | ||||
|                 android:layout_marginStart="8dp" | ||||
|                 android:gravity="start" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:layout_marginEnd="8dp" | ||||
|                 android:textAlignment="viewStart" | ||||
|                 android:textStyle="bold" | ||||
|                 android:textColor="?android:textColorPrimary" | ||||
|                 app:layout_constraintHorizontal_bias="0.0" | ||||
|                 app:layout_constraintLeft_toRightOf="@+id/sourceImage" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="@+id/sourceImage" | ||||
|                 android:textStyle="bold" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toEndOf="@+id/sourceImage" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 tools:text="Titre" /> | ||||
|  | ||||
|             <TextView | ||||
|                 android:id="@+id/sourceTitleAndDate" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginStart="8dp" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:gravity="start" | ||||
|                 android:layout_marginEnd="8dp" | ||||
|                 android:textAlignment="viewStart" | ||||
|                 android:textSize="14sp" | ||||
|                 android:textColor="?android:textColorPrimary" | ||||
|                 app:layout_constraintLeft_toLeftOf="@+id/title" | ||||
|                 android:textSize="14sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/title" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/title" | ||||
|                 tools:text="Google Actualité Il y a 5h" /> | ||||
|  | ||||
|             <RelativeLayout | ||||
|                 android:layout_width="0dp" | ||||
|             <LinearLayout | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:layout_marginEnd="8dp" | ||||
|                 app:layout_constraintBottom_toBottomOf="parent" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/sourceTitleAndDate"> | ||||
|  | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/favButton" | ||||
|                     android:id="@+id/browserBtn" | ||||
|                     android:layout_width="35dp" | ||||
|                     android:layout_height="35dp" | ||||
|                     android:layout_alignParentEnd="true" | ||||
|                     android:layout_alignParentRight="true" | ||||
|                     android:layout_centerVertical="true" | ||||
|                     android:layout_marginEnd="8dp" | ||||
|                     android:layout_marginRight="8dp" | ||||
|                     android:adjustViewBounds="true" | ||||
|                     android:background="@android:color/transparent" | ||||
|                     android:contentDescription="@string/reader_action_open" | ||||
|                     android:elevation="5dp" | ||||
|                     android:padding="4dp" | ||||
|                     android:scaleType="centerCrop" | ||||
|                     app:srcCompat="@drawable/ic_menu_heart_60dp" | ||||
|                     app:tint="@color/ic_menu_heart_color" /> | ||||
|                     app:srcCompat="@drawable/ic_open_in_browser_black_24dp" | ||||
|                     app:tint="?android:attr/textColorPrimary" /> | ||||
|  | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/shareBtn" | ||||
|                     android:layout_width="35dp" | ||||
|                     android:layout_height="35dp" | ||||
|                     android:layout_centerVertical="true" | ||||
|                     android:layout_marginEnd="16dp" | ||||
|                     android:layout_marginRight="16dp" | ||||
|                     android:layout_toLeftOf="@+id/favButton" | ||||
|                     android:layout_toStartOf="@+id/favButton" | ||||
|                     android:layout_marginStart="16dp" | ||||
|                     android:adjustViewBounds="true" | ||||
|                     android:background="@android:color/transparent" | ||||
|                     android:contentDescription="@string/share" | ||||
|                     android:elevation="5dp" | ||||
|                     android:padding="4dp" | ||||
|                     android:scaleType="centerCrop" | ||||
| @@ -130,23 +117,21 @@ | ||||
|                     app:tint="?android:attr/textColorPrimary" /> | ||||
|  | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/browserBtn" | ||||
|                     android:id="@+id/favButton" | ||||
|                     android:layout_width="35dp" | ||||
|                     android:layout_height="35dp" | ||||
|                     android:layout_centerVertical="true" | ||||
|                     android:layout_marginEnd="16dp" | ||||
|                     android:layout_marginRight="16dp" | ||||
|                     android:layout_toLeftOf="@+id/shareBtn" | ||||
|                     android:layout_toStartOf="@+id/shareBtn" | ||||
|                     android:layout_marginStart="16dp" | ||||
|                     android:adjustViewBounds="true" | ||||
|                     android:background="@android:color/transparent" | ||||
|                     android:contentDescription="@string/add_to_favs_reader" | ||||
|                     android:elevation="5dp" | ||||
|                     android:padding="4dp" | ||||
|                     android:scaleType="centerCrop" | ||||
|                     app:srcCompat="@drawable/ic_open_in_browser_black_24dp" | ||||
|                     app:tint="?android:attr/textColorPrimary" /> | ||||
|                     app:srcCompat="@drawable/ic_menu_heart_60dp" | ||||
|                     app:tint="@color/ic_menu_heart_color" /> | ||||
|  | ||||
|             </RelativeLayout> | ||||
|  | ||||
|             </LinearLayout> | ||||
|  | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|   | ||||
							
								
								
									
										26
									
								
								androidApp/src/main/res/layout/circle_image_view.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								androidApp/src/main/res/layout/circle_image_view.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:layout_width="wrap_content" | ||||
|     android:layout_height="wrap_content"> | ||||
|  | ||||
|     <com.google.android.material.imageview.ShapeableImageView | ||||
|         android:id="@+id/circleImage" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:scaleType="centerCrop" | ||||
|         app:shapeAppearanceOverlay="@style/circleImageView" | ||||
|         app:srcCompat="@drawable/background_splash" /> | ||||
|  | ||||
|     <TextView | ||||
|         android:id="@+id/circleText" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:ellipsize="none" | ||||
|         android:gravity="center" | ||||
|         android:singleLine="true" | ||||
|         android:textColor="@color/white" | ||||
|         android:textIsSelectable="false" | ||||
|         android:textSize="20sp" | ||||
|         android:typeface="normal" /> | ||||
| </RelativeLayout> | ||||
							
								
								
									
										96
									
								
								androidApp/src/main/res/layout/filter_fragment.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								androidApp/src/main/res/layout/filter_fragment.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
|  | ||||
|     <ProgressBar | ||||
|         android:id="@+id/progressBar2" | ||||
|         style="?android:attr/progressBarStyle" | ||||
|         android:layout_width="48dp" | ||||
|         android:layout_height="48dp" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         tools:visibility="gone" /> | ||||
|  | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:fillViewport="true"> | ||||
|  | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:id="@+id/filterView" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:visibility="gone" | ||||
|             tools:visibility="visible"> | ||||
|  | ||||
|             <TextView | ||||
|                 android:id="@+id/filterTagsTitle" | ||||
|                 style="@style/MaterialAlertDialog.MaterialComponents.Title.Text" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginStart="24dp" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:text="@string/filter_item_tags" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" /> | ||||
|  | ||||
|             <TextView | ||||
|                 android:id="@+id/filterSourcesTitle" | ||||
|                 style="@style/MaterialAlertDialog.MaterialComponents.Title.Text" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginStart="24dp" | ||||
|                 android:layout_marginTop="24dp" | ||||
|                 android:text="@string/filter_item_sources" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/tagsGroup" /> | ||||
|  | ||||
|  | ||||
|             <com.google.android.material.chip.ChipGroup | ||||
|                 android:id="@+id/tagsGroup" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginStart="16dp" | ||||
|                 android:layout_marginTop="24dp" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/filterTagsTitle" | ||||
|                 app:singleSelection="true"> | ||||
|  | ||||
|             </com.google.android.material.chip.ChipGroup> | ||||
|  | ||||
|             <com.google.android.material.chip.ChipGroup | ||||
|                 android:id="@+id/sourcesGroup" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginStart="16dp" | ||||
|                 android:layout_marginTop="24dp" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/filterSourcesTitle"> | ||||
|  | ||||
|             </com.google.android.material.chip.ChipGroup> | ||||
|  | ||||
|  | ||||
|             <com.google.android.material.floatingactionbutton.FloatingActionButton | ||||
|                 android:id="@+id/floatingActionButton2" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:layout_marginEnd="16dp" | ||||
|                 android:clickable="true" | ||||
|                 android:contentDescription="@string/menu_home_search" | ||||
|                 android:focusable="true" | ||||
|                 app:backgroundTint="@color/colorAccent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 app:rippleColor="@color/colorAccentDark" | ||||
|                 app:srcCompat="@drawable/ic_menu_search_white_24dp" /> | ||||
|  | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
|  | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| @@ -1,5 +1,4 @@ | ||||
| <androidx.coordinatorlayout.widget.CoordinatorLayout | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:layout_width="match_parent" | ||||
| @@ -22,10 +21,22 @@ | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="200dp" | ||||
|                 android:scaleType="centerCrop" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 /> | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" /> | ||||
|  | ||||
|             <TextView | ||||
|                 android:id="@+id/titleView" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginLeft="16dp" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:textAppearance="@style/TextAppearance.AppCompat.Headline" | ||||
|                 android:textStyle="bold" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/imageView" /> | ||||
|  | ||||
|             <TextView | ||||
|                 android:id="@+id/source" | ||||
| @@ -36,40 +47,23 @@ | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:textColor="?android:textColorSecondary" | ||||
|                 android:textSize="12sp" | ||||
|                 app:layout_constraintHorizontal_bias="0.0" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/titleView" /> | ||||
|  | ||||
|             <TextView | ||||
|                 android:id="@+id/titleView" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginLeft="16dp" | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:textAppearance="@style/TextAppearance.AppCompat.Headline" | ||||
|                 android:textStyle="bold" | ||||
|                 app:layout_constraintHorizontal_bias="0.0" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/imageView" /> | ||||
|  | ||||
|  | ||||
|             <WebView | ||||
|                 android:id="@+id/webcontent" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginLeft="16dp" | ||||
|                 android:layout_marginTop="24dp" | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:background="?attr/webviewBackground" | ||||
|                 android:paddingBottom="48dp" | ||||
|                 android:textColorLink="?attr/colorAccent" | ||||
|                 android:visibility="gone" | ||||
|                 android:layout_marginLeft="16dp" | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:layout_marginTop="24dp" | ||||
|                 android:paddingBottom="48dp" | ||||
|                 android:background="?android:colorBackground" | ||||
|                 app:layout_constraintHorizontal_bias="0.0" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/source" | ||||
|                 tools:visibility="visible" /> | ||||
|  | ||||
| @@ -80,10 +74,10 @@ | ||||
|     <FrameLayout | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_gravity="start|bottom|end" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintLeft_toLeftOf="parent" | ||||
|         android:layout_gravity="end|bottom|right"> | ||||
|         app:layout_constraintStart_toStartOf="parent"> | ||||
|  | ||||
|         <com.github.rubensousa.floatingtoolbar.FloatingToolbar | ||||
|             android:id="@+id/floatingToolbar" | ||||
| @@ -96,12 +90,11 @@ | ||||
|             android:id="@+id/fab" | ||||
|             android:layout_width="wrap_content" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_gravity="end|bottom|right" | ||||
|             android:layout_marginBottom="16dp" | ||||
|             android:layout_gravity="end|bottom" | ||||
|             android:layout_marginEnd="16dp" | ||||
|             android:layout_marginRight="16dp" | ||||
|             android:paddingBottom="@dimen/activity_vertical_margin" | ||||
|             android:layout_marginBottom="16dp" | ||||
|             android:paddingTop="@dimen/activity_vertical_margin" | ||||
|             android:paddingBottom="@dimen/activity_vertical_margin" | ||||
|             android:src="@drawable/ic_add_white_24dp" | ||||
|             app:backgroundTint="?attr/colorAccent" | ||||
|             app:fabSize="mini" | ||||
| @@ -112,11 +105,11 @@ | ||||
|         android:id="@+id/progressBar" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:visibility="gone" | ||||
|         android:animateLayoutChanges="true" | ||||
|         android:alpha="0.8" | ||||
|         android:animateLayoutChanges="true" | ||||
|         android:background="@color/black" | ||||
|         android:clickable="false"> | ||||
|         android:clickable="false" | ||||
|         android:visibility="gone"> | ||||
|  | ||||
|         <ProgressBar | ||||
|             style="?android:attr/progressBarStyleLarge" | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
| @@ -9,8 +9,8 @@ | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent" | ||||
|             android:layout_centerVertical="true" | ||||
|         android:layout_centerHorizontal="true" | ||||
|         android:background="@android:color/black" | ||||
|             android:adjustViewBounds="true" | ||||
|             android:background="@drawable/checkerboard" | ||||
|             app:srcCompat="@android:drawable/screen_background_dark" /> | ||||
|  | ||||
| </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| @@ -3,17 +3,18 @@ | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="88dp"> | ||||
|     android:layout_height="wrap_content"> | ||||
|  | ||||
|     <ImageView | ||||
|     <bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||
|         android:id="@+id/itemImage" | ||||
|         android:layout_width="46dp" | ||||
|         android:layout_height="46dp" | ||||
|         android:layout_marginStart="8dp" | ||||
|         android:layout_marginTop="21dp" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:layout_marginBottom="8dp" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         android:layout_marginLeft="8dp" /> | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
|  | ||||
|     <TextView | ||||
|         android:id="@+id/title" | ||||
| @@ -21,42 +22,35 @@ | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginStart="8dp" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:layout_marginEnd="16dp" | ||||
|         android:layout_marginEnd="8dp" | ||||
|         android:ellipsize="end" | ||||
|         android:fontFamily="sans-serif" | ||||
|         android:gravity="start" | ||||
|         android:maxLines="3" | ||||
|         android:textAlignment="viewStart" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="?android:textColorPrimary" | ||||
|         android:textSize="16sp" | ||||
|         android:textStyle="bold" | ||||
|         android:textColor="?android:textColorPrimary" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintHorizontal_bias="0.0" | ||||
|         app:layout_constraintStart_toEndOf="@+id/itemImage" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         tools:text="Titre" | ||||
|         android:layout_marginLeft="8dp" | ||||
|         android:layout_marginRight="16dp" /> | ||||
|         tools:text="Titre" /> | ||||
|  | ||||
|     <TextView | ||||
|         android:id="@+id/sourceTitleAndDate" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginStart="8dp" | ||||
|         android:layout_marginTop="66dp" | ||||
|         android:layout_marginEnd="16dp" | ||||
|         android:layout_marginTop="16dp" | ||||
|         android:layout_marginEnd="8dp" | ||||
|         android:layout_marginBottom="8dp" | ||||
|         android:gravity="start" | ||||
|         android:maxLines="1" | ||||
|         android:textAlignment="viewStart" | ||||
|         android:textSize="14sp" | ||||
|         android:textColor="?android:textColorPrimary" | ||||
|         android:textSize="14sp" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintHorizontal_bias="0.0" | ||||
|         app:layout_constraintStart_toEndOf="@+id/itemImage" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         tools:text="Google Actualité Il y a 5h" | ||||
|         android:layout_marginLeft="8dp" | ||||
|         android:layout_marginRight="16dp" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/title" | ||||
|         tools:text="Google Actualité Il y a 5h" /> | ||||
|  | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| @@ -3,48 +3,74 @@ | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="48dp" | ||||
|     android:layout_height="wrap_content" | ||||
|     android:orientation="vertical"> | ||||
|  | ||||
|  | ||||
|     <ImageView | ||||
|         android:id="@+id/itemImage" | ||||
|         android:layout_width="36dp" | ||||
|         android:layout_height="36dp" | ||||
|         android:importantForAccessibility="no" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintLeft_toLeftOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
|  | ||||
|     <TextView | ||||
|         android:id="@+id/sourceTitle" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="17dp" | ||||
|         android:layout_marginStart="8dp" | ||||
|         android:ellipsize="end" | ||||
|         android:gravity="start" | ||||
|         android:maxLines="1" | ||||
|         android:textAlignment="textStart" | ||||
|         android:textSize="13sp" | ||||
|         android:textColor="?android:textColorPrimary" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toStartOf="@+id/deleteBtn" | ||||
|         app:layout_constraintStart_toEndOf="@+id/itemImage" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         tools:text="source title" /> | ||||
|  | ||||
|     <Button | ||||
|         android:id="@+id/deleteBtn" | ||||
|         style="@style/Widget.AppCompat.Button.Borderless" | ||||
|         android:layout_width="34dp" | ||||
|         android:layout_height="34dp" | ||||
|         android:layout_width="48dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:layout_marginEnd="8dp" | ||||
|         android:layout_marginBottom="8dp" | ||||
|         android:background="@drawable/ic_remove_circle_outline_black_24dp" | ||||
|         android:backgroundTint="?android:textColorSecondary" | ||||
|         android:elevation="4dp" | ||||
|         android:contentDescription="@string/remove_source" | ||||
|         android:elevation="4dp" | ||||
|         app:iconSize="34dp" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         app:layout_constraintVertical_bias="0.0" /> | ||||
|  | ||||
|     <bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||
|         android:id="@+id/itemImage" | ||||
|         android:layout_width="36dp" | ||||
|         android:layout_height="36dp" | ||||
|         android:layout_marginStart="8dp" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:layout_marginBottom="8dp" | ||||
|         android:importantForAccessibility="no" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         app:layout_constraintVertical_bias="0.0" /> | ||||
|  | ||||
|     <TextView | ||||
|         android:id="@+id/sourceTitle" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginStart="8dp" | ||||
|         android:ellipsize="end" | ||||
|         android:maxLines="1" | ||||
|         android:textAlignment="viewStart" | ||||
|         android:textColor="?android:textColorPrimary" | ||||
|         android:textSize="20sp" | ||||
|         android:textStyle="bold" | ||||
|         app:layout_constraintBottom_toTopOf="@+id/errorText" | ||||
|         app:layout_constraintEnd_toStartOf="@+id/deleteBtn" | ||||
|         app:layout_constraintStart_toEndOf="@+id/itemImage" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         tools:text="Source title" /> | ||||
|  | ||||
|     <TextView | ||||
|         android:id="@+id/errorText" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginStart="10sp" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:layout_marginBottom="8dp" | ||||
|         android:textAppearance="@style/TextAppearance.AppCompat.Small" | ||||
|         android:textColor="@color/red" | ||||
|         android:textStyle="italic" | ||||
|         android:visibility="gone" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toStartOf="@+id/deleteBtn" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/itemImage" | ||||
|         tools:text="Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua." | ||||
|         tools:visibility="visible" /> | ||||
|  | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| @@ -8,18 +8,40 @@ | ||||
|         app:showAsAction="ifRoom|collapseActionView" | ||||
|         app:actionViewClass="androidx.appcompat.widget.SearchView" /> | ||||
|  | ||||
|     <item android:id="@+id/readAll" | ||||
|           android:icon="@drawable/ic_menu_done_all_white_24dp" | ||||
|           android:title="@string/readAll" | ||||
|     <item android:id="@+id/action_filter" | ||||
|         android:title="@string/menu_home_filter" | ||||
|         android:icon="@drawable/ic_baseline_filter_alt_24" | ||||
|         android:orderInCategory="1" | ||||
|         app:showAsAction="always" /> | ||||
|  | ||||
|     <item android:id="@+id/readAll" | ||||
|           android:icon="@drawable/ic_menu_done_all_white_24dp" | ||||
|           android:title="@string/readAll" | ||||
|           android:orderInCategory="2" | ||||
|           app:showAsAction="ifRoom"/> | ||||
|  | ||||
|     <item android:id="@+id/action_sources" | ||||
|         android:title="@string/menu_home_sources" | ||||
|         android:orderInCategory="97" | ||||
|         app:showAsAction="never"/> | ||||
|  | ||||
|     <item android:id="@+id/action_settings" | ||||
|         android:title="@string/title_activity_settings" | ||||
|         android:orderInCategory="98" | ||||
|         app:showAsAction="never"/> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/refresh" | ||||
|         android:icon="@drawable/ic_menu_refresh_white_24dp" | ||||
|         android:orderInCategory="99" | ||||
|         app:showAsAction="never" | ||||
|         android:orderInCategory="101" | ||||
|         android:title="@string/menu_home_refresh" /> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/issue_tracker" | ||||
|         app:showAsAction="never" | ||||
|         android:orderInCategory="103" | ||||
|         android:title="@string/issue_tracker_link" /> | ||||
|  | ||||
|     <item android:id="@+id/action_disconnect" | ||||
|           android:title="@string/action_disconnect" | ||||
|           android:orderInCategory="104" | ||||
|   | ||||
| @@ -3,6 +3,13 @@ | ||||
|       xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||
|  | ||||
|  | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/issue_tracker" | ||||
|         app:showAsAction="never" | ||||
|         android:orderInCategory="101" | ||||
|         android:title="@string/issue_tracker_link" /> | ||||
|  | ||||
|     <item android:id="@+id/about" | ||||
|           android:title="@string/action_about" | ||||
|           android:orderInCategory="102" | ||||
|   | ||||
| @@ -8,12 +8,6 @@ | ||||
|         android:title="@string/unmark" | ||||
|         app:showAsAction="ifRoom" /> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/more_action" | ||||
|         android:icon="@drawable/ic_chrome_reader_mode_white_24dp" | ||||
|         android:title="@string/reader_action_more" | ||||
|         app:showAsAction="ifRoom" /> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/open_action" | ||||
|         android:icon="@drawable/ic_open_in_browser_white_24dp" | ||||
|   | ||||
							
								
								
									
										16
									
								
								androidApp/src/main/res/menu/reader_toolbar_no_read.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								androidApp/src/main/res/menu/reader_toolbar_no_read.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| <menu xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/open_action" | ||||
|         android:icon="@drawable/ic_open_in_browser_white_24dp" | ||||
|         android:title="@string/reader_action_open" | ||||
|         app:showAsAction="ifRoom" /> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/share_action" | ||||
|         android:icon="@drawable/ic_share_white_24dp" | ||||
|         android:title="@string/reader_action_share" | ||||
|         app:showAsAction="ifRoom" /> | ||||
|  | ||||
| </menu> | ||||
| @@ -1,10 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <menu xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/clear" | ||||
|         android:icon="@drawable/ic_history_white_24dp" | ||||
|         android:title="@string/drawer_action_clear" | ||||
|         app:showAsAction="ifRoom" /> | ||||
| </menu> | ||||
| @@ -1,5 +1,5 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | ||||
| <resources> | ||||
|     <string name="app_name">"Lector per a Selfoss"</string> | ||||
|     <string name="title_activity_login">"Inicia la sessió"</string> | ||||
|     <string name="prompt_password">"Contrasenya"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"La contrasenya és massa curta"</string> | ||||
|     <string name="error_field_required">"Camp necessari"</string> | ||||
|     <string name="prompt_url">"URL"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"Autenticació (si és necessària)"</string> | ||||
|     <string name="login_url_problem">"Pot ser que falti una \"/\" al final de l'url."</string> | ||||
|     <string name="prompt_login">"Nom d'usuari"</string> | ||||
|     <string name="label_share">"Comparteix"</string> | ||||
|     <string name="readAll">"Llegeix-ho tot"</string> | ||||
|     <string name="action_disconnect">"Desconnecta't"</string> | ||||
|     <string name="title_activity_settings">"Configuració"</string> | ||||
| @@ -23,13 +23,6 @@ | ||||
|     <string name="wrong_infos">"Torneu a comprovar la informació."</string> | ||||
|     <string name="all_posts_not_read">"No s'han llegit totes les publicacions"</string> | ||||
|     <string name="all_posts_read">"S'han llegit totes les publicacions"</string> | ||||
|     <string name="nothing_here">"No hi ha res"</string> | ||||
|     <string name="tab_new">"Nou"</string> | ||||
|     <string name="tab_read">"Tot"</string> | ||||
|     <string name="tab_favs">"Preferits"</string> | ||||
|     <string name="action_about">"Quant a"</string> | ||||
|     <string name="marked_as_read">"Element llegit"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Desfés"</string> | ||||
|     <string name="addStringNoUrl">"Inicieu la sessió per afegir fonts."</string> | ||||
|     <string name="cant_get_sources">"No es pot obtenir la llista de fonts."</string> | ||||
| @@ -47,12 +40,9 @@ | ||||
|     <string name="switch_unread_count_title">"Recompte d'articles no llegits"</string> | ||||
|     <string name="display_all_counts_title">"Recompte d'articles llegits i preferits"</string> | ||||
|     <string name="text_wrong_url">"Sembla que esteu utilitzant un URL no vàlid. Assegureu-vos que és correcte, i si el problema persisteix, poseu-vos en contacte amb mi (a través de l'enllaç de contacte que hi ha a la Botiga). Tingueu en compte que per utilitzar aquesta aplicació cal que també utilitzeu Selfoss. Si no, no podreu accedir a canals RSS."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Obre els enllaços dins de l'aplicació"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Els articles s'obriran dins de l'aplicació"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Els articles s'obriran amb el navegador predeterminat"</string> | ||||
|     <string name="prefer_article_viewer_title">"Obre el visualitzador d'articles"</string> | ||||
|     <string name="prefer_article_viewer_on">"S'obrirà el visualitzador d'articles en lloc del navegador intern"</string> | ||||
|     <string name="prefer_article_viewer_off">"S'obrirà el navegador intern en lloc del visualitzador d'articles"</string> | ||||
|     <string name="pref_article_viewer_title">"Obre els enllaços dins de l'aplicació"</string> | ||||
|     <string name="pref_article_viewer_on">"Els articles s'obriran dins de l'aplicació"</string> | ||||
|     <string name="pref_article_viewer_off">"Els articles s'obriran amb el navegador predeterminat"</string> | ||||
|     <string name="pref_general_category_links">"Gestió d'enllaços"</string> | ||||
|     <string name="pref_general_category_displaying">"Visualització"</string> | ||||
|     <string name="pref_switch_card_view_on">"Els articles es mostraran com a targetes"</string> | ||||
| @@ -65,28 +55,18 @@ | ||||
|     <string name="card_height_on">L\'alçada de les targetes s\'ajustarà al seu contingut</string> | ||||
|     <string name="card_height_off">L\'alçada de les targetes serà fixa</string> | ||||
|     <string name="source_code">Codi font</string> | ||||
|     <string name="drawer_error_loading_tags">S\'ha produït un error en carregar les etiquetes</string> | ||||
|     <string name="drawer_item_filters">Filtres</string> | ||||
|     <string name="drawer_action_clear">Esborra</string> | ||||
|     <string name="drawer_item_tags">Etiquetes</string> | ||||
|     <string name="drawer_item_sources">Fonts</string> | ||||
|     <string name="drawer_action_edit">Edita</string> | ||||
|     <string name="drawer_loading">S\'està carregant…</string> | ||||
|     <string name="filter_item_tags">Etiquetes</string> | ||||
|     <string name="filter_item_sources">Fonts</string> | ||||
|     <string name="menu_home_search">Cerca</string> | ||||
|     <string name="can_delete_source">No es pot suprimir la font</string> | ||||
|     <string name="base_url_error">S\'ha produït un error en comunicar-se amb la instància de Selfoss. Si el problema persisteix, posa\'t en contacte amb mi.</string> | ||||
|     <string name="pref_header_theme">Temes</string> | ||||
|     <string name="default_theme">Predeterminat</string> | ||||
|     <string name="default_dark_theme">Predeterminat/Fosc</string> | ||||
|     <string name="pref_selfoss_category">API de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Nombre d\'elements carregats</string> | ||||
|     <string name="pref_hidden_tags">Etiquetes ocultes</string> | ||||
|     <string name="pref_general_infinite_loading_title">Carrega articles en desplaçar</string> | ||||
|     <string name="translation">Traducció</string> | ||||
|     <string name="cant_open_invalid_url">L\'element URL no és vàlid. Estic intentant solucionar aquest problema perquè l\'aplicació no falli.</string> | ||||
|     <string name="drawer_report_bug">Informa d\'un error</string> | ||||
|     <string name="items_number_should_be_number">El nombre d\'elements ha de ser enter.</string> | ||||
|     <string name="reader_action_more">Més informació</string> | ||||
|     <string name="reader_action_open">Obre al navegador</string> | ||||
|     <string name="reader_action_share">Comparteix</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Es marcaran els articles com a llegits en lliscar el dit d\'un article a l\'altre.</string> | ||||
| @@ -97,7 +77,6 @@ | ||||
|     <string name="markall_dialog_message">Aquesta acció marcarà els elements com a llegits.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Marca com a llegit en lliscar el dit</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">No es marcaran els articles com a llegits en lliscar el dit d\'un article a l\'altre.</string> | ||||
|     <string name="drawer_item_hidden_tags">Etiquetes ocultes</string> | ||||
|     <string name="unmark">Marca com no llegit</string> | ||||
|     <string name="pref_header_offline">Sense connexió i memòria clau</string> | ||||
|     <string name="pref_switch_items_caching_off">Els articles no es guardaran a la memòria del dispositiu i l\'aplicació no es podrà utilitzar sense connexió.</string> | ||||
| @@ -105,14 +84,14 @@ | ||||
|     <string name="pref_switch_items_caching">Guarda els elements per utilitzar-los sense connexió</string> | ||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> | ||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> | ||||
|     <string name="network_connectivity_lost">"Sense connexió!"</string> | ||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Sincronitza els articles</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Els articles no se sincronitzaran en segon pla</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Els articles se sincronitzaran periòdicament</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de sincronització ( >= 15 minuts)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Sincronitza només quan el telèfon s\'està carregant</string> | ||||
|     <string name="loading_notification_title">S\'està carregant...</string> | ||||
|     <string name="loading_notification_title">S\'està carregant…</string> | ||||
|     <string name="loading_notification_text">Selfoss està sincronitzant els articles</string> | ||||
|     <string name="notification_channel_sync">Notificació de sincronització</string> | ||||
|     <string name="new_items_channel_sync">Notificació d\'elements nous</string> | ||||
| @@ -131,4 +110,26 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting."</string> | ||||
|     <string name="menu_home_filter">Filters</string> | ||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||
|     <string name="menu_home_sources">Sources</string> | ||||
|     <string name="update_source">Update source</string> | ||||
|     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||
|     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||
|     <string name="no_browser">Can\'t open a link without a browser installed</string> | ||||
|     <string name="nothing_here">"No hi ha res"</string> | ||||
|     <string name="tab_new">"Nou"</string> | ||||
|     <string name="tab_read">"Tot"</string> | ||||
|     <string name="tab_favs">"Preferits"</string> | ||||
|     <string name="action_about">"Quant a"</string> | ||||
|     <string name="marked_as_read">"Element llegit"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
| </resources> | ||||
| @@ -1,16 +1,16 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | ||||
|     <string name="app_name">"Reader für selfoss"</string> | ||||
| <resources> | ||||
|     <string name="app_name">"Reader für Selfoss"</string> | ||||
|     <string name="title_activity_login">"Anmelden"</string> | ||||
|     <string name="prompt_password">"Passwort"</string> | ||||
|     <string name="action_sign_in">"Fortfahren"</string> | ||||
|     <string name="error_invalid_password">"Passwort ist nicht lang genug"</string> | ||||
|     <string name="error_field_required">"Pflichtfeld"</string> | ||||
|     <string name="prompt_url">"URL"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"Anmeldung erforderlich?"</string> | ||||
|     <string name="login_url_problem">"Ups. Du musst eventuell ein \"/\" am Ende der URL anhängen."</string> | ||||
|     <string name="prompt_login">"Benutzername"</string> | ||||
|     <string name="label_share">"Teilen"</string> | ||||
|     <string name="readAll">"Alle gelesen"</string> | ||||
|     <string name="action_disconnect">"Verbindung trennen"</string> | ||||
|     <string name="title_activity_settings">"Einstellungen"</string> | ||||
| @@ -23,22 +23,15 @@ | ||||
|     <string name="wrong_infos">"Überprüfe deine Angaben noch einmal."</string> | ||||
|     <string name="all_posts_not_read">"Nicht alle Beiträge wurden gelesen"</string> | ||||
|     <string name="all_posts_read">"Alle Beiträge wurden gelesen"</string> | ||||
|     <string name="nothing_here">"Keine Einträge vorhanden"</string> | ||||
|     <string name="tab_new">"Neu"</string> | ||||
|     <string name="tab_read">"Alle"</string> | ||||
|     <string name="tab_favs">"Favoriten"</string> | ||||
|     <string name="action_about">"Über"</string> | ||||
|     <string name="marked_as_read">"Artikel gelesen"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Rückgängig"</string> | ||||
|     <string name="addStringNoUrl">"Melde dich an um Quellen hinzuzufügen."</string> | ||||
|     <string name="cant_get_sources">"Quellen können nicht abgerufen werden."</string> | ||||
|     <string name="cant_create_source">"Quelle kann nicht gespeichert werden."</string> | ||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> | ||||
|     <string name="cant_get_spouts_no_network">"Fehler beim Laden der Spouts-Liste aufgrund von Netzwerkproblemen."</string> | ||||
|     <string name="cant_get_spouts">"Fehler beim Laden der Spouts-Liste, möglicherweise aufgrund eines API-Fehlers."</string> | ||||
|     <string name="form_not_complete">"Das Formular ist nicht vollständig"</string> | ||||
|     <string name="pref_header_links">"Links"</string> | ||||
|     <string name="issue_tracker_link">"Issue Tracker"</string> | ||||
|     <string name="issue_tracker_link">"Ticketsystem"</string> | ||||
|     <string name="issue_tracker_summary">"Melde einen Bug oder rege ein neues Feature an"</string> | ||||
|     <string name="warning_wrong_url">"WARNUNG"</string> | ||||
|     <string name="pref_switch_card_view_title">"Kachelansicht"</string> | ||||
| @@ -47,12 +40,9 @@ | ||||
|     <string name="switch_unread_count_title">"Zeige Anzahl ungelesener Artikel"</string> | ||||
|     <string name="display_all_counts_title">"Zeige Anzahl der Favoriten und gelesenen Artikel"</string> | ||||
|     <string name="text_wrong_url">"Sie scheinen eine ungültige URL verwenden. Stellen Sie sicher, dass die URL richtig ist. Sollte das Problem weiterhin bestehen kontaktieren Sie mich (über den Playstore-Kontakt-Link). Bitte beachten Sie, dass Sie Selfoss benötigen um RSS-Feeds zu lesen."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Öffne Links innerhalb der App"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Artikel werden innerhalb der App geöffnet"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Artikel werden mit deinem Standard-Browser geöffnet"</string> | ||||
|     <string name="prefer_article_viewer_title">"Verwenden Sie den Artikel-viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Artikel-Viewer wird anstelle des internen Browser verwendet"</string> | ||||
|     <string name="prefer_article_viewer_off">"Der internen Browser wird anstelle des Artikel-Viewer verwendet"</string> | ||||
|     <string name="pref_article_viewer_title">"Öffne Links innerhalb der App"</string> | ||||
|     <string name="pref_article_viewer_on">"Artikel werden innerhalb der App geöffnet"</string> | ||||
|     <string name="pref_article_viewer_off">"Artikel werden mit deinem Standard-Browser geöffnet"</string> | ||||
|     <string name="pref_general_category_links">"Umgang mit Links"</string> | ||||
|     <string name="pref_general_category_displaying">"Ansicht"</string> | ||||
|     <string name="pref_switch_card_view_on">"Artikel werden als Kacheln angezeigt"</string> | ||||
| @@ -65,70 +55,81 @@ | ||||
|     <string name="card_height_on">Kartenhöhe passt sich Inhalt an</string> | ||||
|     <string name="card_height_off">Kartenhöhe ist fix</string> | ||||
|     <string name="source_code">Quellcode</string> | ||||
|     <string name="drawer_error_loading_tags">Fehler beim Laden der Tags…</string> | ||||
|     <string name="drawer_item_filters">Filter</string> | ||||
|     <string name="drawer_action_clear">leeren</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Quellen</string> | ||||
|     <string name="drawer_action_edit">bearbeiten</string> | ||||
|     <string name="drawer_loading">Lade…</string> | ||||
|     <string name="filter_item_tags">Tags</string> | ||||
|     <string name="filter_item_sources">Quellen</string> | ||||
|     <string name="menu_home_search">Suche</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
|     <string name="can_delete_source">Quelle konnte nicht gelöscht werden…</string> | ||||
|     <string name="base_url_error">Beim Versuch deine Selfoss-Instanz zu erreichen ist ein Fehler aufgetreten. Solltet dieser Fehler bestehen bleiben, trete bitte mit mir in Kontakt.</string> | ||||
|     <string name="pref_header_theme">Designs</string> | ||||
|     <string name="default_theme">Standard</string> | ||||
|     <string name="default_dark_theme">Standard (Dunkel)</string> | ||||
|     <string name="pref_selfoss_category">selfoss API</string> | ||||
|     <string name="pref_api_items_number_title">Loaded items number</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="pref_api_items_number_title">Anzahl der zu ladenden Artikel</string> | ||||
|     <string name="pref_general_infinite_loading_title">Weitere Artikel beim Navigieren laden</string> | ||||
|     <string name="translation">Übersetzung</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
|     <string name="drawer_report_bug">Melde einen Fehler</string> | ||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||
|     <string name="reader_action_more">Read more</string> | ||||
|     <string name="cant_open_invalid_url">Der Artikel-Link ist ungültig. Ich such nach einer Lösung dieses Problems, damit die App nicht abstürzt.</string> | ||||
|     <string name="items_number_should_be_number">Die Anzahl der Artikel sollte eine Ganzzahl sein.</string> | ||||
|     <string name="reader_action_open">Im Browser öffnen</string> | ||||
|     <string name="reader_action_share">Teilen</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Artikel als gelesen markieren, wenn zwischen den Artikeln gewischt wird.</string> | ||||
|     <string name="add_to_favs_reader">Zu Favoriten hinzufügen</string> | ||||
|     <string name="pref_content_reader_font_size">Article reader content font size</string> | ||||
|     <string name="pref_header_viewer">Article viewer</string> | ||||
|     <string name="refresh_dialog_message">This will refresh your Selfoss instance.</string> | ||||
|     <string name="pref_content_reader_font_size">Schriftgröße im Lesemodus</string> | ||||
|     <string name="pref_header_viewer">Lesemodus</string> | ||||
|     <string name="refresh_dialog_message">Dies wird die Selfoss-Instanz aktualisieren.</string> | ||||
|     <string name="markall_dialog_message">Dies wird alle Elemente als gelesen markieren.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Beim Wischen als gelesen markieren</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">Artikel nicht als gelesen markieren, wenn zwischen den Artikeln gewischt wird.</string> | ||||
|     <string name="unmark">Eintrag als ungelesen markieren</string> | ||||
|     <string name="pref_header_offline">Offline and cache</string> | ||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> | ||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||
|     <string name="pref_header_offline">Offline-Modus und Cache</string> | ||||
|     <string name="pref_switch_items_caching_off">Artikel werden nicht lokal zwischengespeichert wodurch die App nicht offline nutzbar ist.</string> | ||||
|     <string name="pref_switch_items_caching_on">Artikel werden lokal zwischengespeichert wodurch die App offline nutzbar ist.</string> | ||||
|     <string name="pref_switch_items_caching">Artikel lokal zwischenspeichern</string> | ||||
|     <string name="pref_switch_update_sources">Nach neuen Quellen und Tags suchen</string> | ||||
|     <string name="pref_switch_update_sources_summary">Diese Funktion sollte deaktiviert werden, wenn der Server übermäßig viele Datenbankanfragen erhält.</string> | ||||
|     <string name="network_connectivity_lost">"Die Netzwerkverbindung wurde unterbrochen"</string> | ||||
|     <string name="network_connectivity_retrieved">"Netzwerkverbindung ist jetzt verfügbar"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Synchronisiere Artikel</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Artikel werden nicht im Hintergrund synchronisiert</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Die Artikel werden regelmäßig synchronisiert</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Aktualisierungsintervall (>= 15 Minuten)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Nur aktualisieren, wenn das Telefon aufgeladen wird</string> | ||||
|     <string name="loading_notification_title">Lädt...</string> | ||||
|     <string name="loading_notification_title">Lädt…</string> | ||||
|     <string name="loading_notification_text">Selfoss synchronisiert Ihre Artikel</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
|     <string name="new_items_notification_title">New items !</string> | ||||
|     <string name="new_items_notification_text">%1$d new items loaded.</string> | ||||
|     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> | ||||
|     <string name="notification_channel_sync">Synchronisationsbenachrichtigung</string> | ||||
|     <string name="new_items_channel_sync">Benachrichtigung bei neuen Artikeln</string> | ||||
|     <string name="new_items_notification_title">Neue Artikel!</string> | ||||
|     <string name="new_items_notification_text">%1$d neue Artikel geladen.</string> | ||||
|     <string name="pref_switch_notify_new_items">Benachrichtigung bei neuen Artikeln</string> | ||||
|     <string name="shortcut_offline">Offline</string> | ||||
|     <string name="pref_api_timeout">API-Zeitüberschreitung</string> | ||||
|     <string name="pref_header_experimental">Experimentell</string> | ||||
|     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> | ||||
|     <string name="webview_dialog_issue_title">Webview issue</string> | ||||
|     <string name="reader_text_align_left">Align left</string> | ||||
|     <string name="reader_text_align_justify">Justify</string> | ||||
|     <string name="settings_reader_font">Reader font</string> | ||||
|     <string name="reader_static_bar_title">Static bottom bar in the article viewer</string> | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="webview_dialog_issue_message">Webview ist nicht verfügbar. Deaktiviere den Lesemodus, um zukünftige Abstürze zu vermeiden. Lade von nun an die Nachrichten in deinen Browser.</string> | ||||
|     <string name="webview_dialog_issue_title">Webview-Probleme</string> | ||||
|     <string name="reader_text_align_left">Linksbündig</string> | ||||
|     <string name="reader_text_align_justify">Blocksatz</string> | ||||
|     <string name="settings_reader_font">Schriftgröße im Lesemodus</string> | ||||
|     <string name="reader_static_bar_title">Statische untere Leiste im Lesemodus</string> | ||||
|     <string name="reader_static_bar_on">Die untere Leiste wird dauerhaft angezeigt</string> | ||||
|     <string name="reader_static_bar_off">Die untere Leiste kann über einen schwebenden Button angezeigt werden</string> | ||||
|     <string name="remove_source">Quelle entfernen</string> | ||||
|     <string name="pref_theme_title">Heller/Dunkler Modus</string> | ||||
|     <string name="mode_dark">Dunkler Modus</string> | ||||
|     <string name="mode_system">Systemeinstellungen übernehmen</string> | ||||
|     <string name="mode_light">Heller Modus</string> | ||||
|     <string name="gdpr_dialog_title">Diese App teilt keine persönlichen Daten.</string> | ||||
|     <string name="gdpr_dialog_message"><![CDATA[Das Senden von Absturzberichten ist jetzt aktiviert. Die Funktion kann auf der Einstellungsseite deaktiviert werden, beachte aber bitte, dass Absturzberichte für die Anwendungsentwicklung von entscheidender Bedeutung sind.]]></string> | ||||
|     <string name="crash_toast_text">Die App ist abgestürzt. Details werden an den Entwickler gesendet.</string> | ||||
|     <string name="pref_switch_disable_acra">"Automatische Fehlerberichterstattung deaktivieren."</string> | ||||
|     <string name="menu_home_filter">Filter</string> | ||||
|     <string name="application_selfoss_only">Diese App funktioniert nur mit einer Selfoss-Instanz, nicht mit einzelnen RSS-Feeds.</string> | ||||
|     <string name="menu_home_sources">Quellen</string> | ||||
|     <string name="update_source">Quelle aktualisieren</string> | ||||
|     <string name="confirm_disconnect_title">Verbindung trennen?</string> | ||||
|     <string name="confirm_disconnect_description">Die Verbindung zur Selfoss-Instanz wird getrennt.</string> | ||||
|     <string name="no_browser">Can\'t open a link without a browser installed</string> | ||||
|     <string name="nothing_here">"Keine Einträge vorhanden"</string> | ||||
|     <string name="tab_new">"Neu"</string> | ||||
|     <string name="tab_read">"Alle"</string> | ||||
|     <string name="tab_favs">"Favoriten"</string> | ||||
|     <string name="action_about">"Über"</string> | ||||
|     <string name="marked_as_read">"Artikel gelesen"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
| </resources> | ||||
| @@ -1,5 +1,5 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | ||||
| <resources> | ||||
|     <string name="app_name">"Lector para Selfoss"</string> | ||||
|     <string name="title_activity_login">"Iniciar sesión"</string> | ||||
|     <string name="prompt_password">"Contraseña"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"La contraseña no es suficientemente larga"</string> | ||||
|     <string name="error_field_required">"Campo requerido"</string> | ||||
|     <string name="prompt_url">"Url"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"Inicio de sesión requerido ?"</string> | ||||
|     <string name="login_url_problem">"Oops. Puede que necesite añadir un \"/\" al final de la url."</string> | ||||
|     <string name="prompt_login">"Nombre de usuario"</string> | ||||
|     <string name="label_share">"Compartir"</string> | ||||
|     <string name="readAll">"Leer todo"</string> | ||||
|     <string name="action_disconnect">"Desconectar"</string> | ||||
|     <string name="title_activity_settings">"Configuración"</string> | ||||
| @@ -23,19 +23,12 @@ | ||||
|     <string name="wrong_infos">"Revise sus datos de nuevo."</string> | ||||
|     <string name="all_posts_not_read">"No todas las publicaciones fueron leídas"</string> | ||||
|     <string name="all_posts_read">"Todas las publicaciones fueron leídas"</string> | ||||
|     <string name="nothing_here">"Nada aquí"</string> | ||||
|     <string name="tab_new">"Nuevo"</string> | ||||
|     <string name="tab_read">"Todo"</string> | ||||
|     <string name="tab_favs">"Favoritos"</string> | ||||
|     <string name="action_about">"Acerca de"</string> | ||||
|     <string name="marked_as_read">"Artículo leído"</string> | ||||
|     <string name="marked_as_unread">"Artículo no leído"</string> | ||||
|     <string name="undo_string">"Deshacer"</string> | ||||
|     <string name="addStringNoUrl">"Iniciar sesión para añadir fuentes."</string> | ||||
|     <string name="cant_get_sources">"No se puede obtener la lista de fuentes."</string> | ||||
|     <string name="cant_create_source">"No se puede crear la fuente."</string> | ||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||
|     <string name="cant_get_spouts">"Can't get spouts list. There may ben an api issue."</string> | ||||
|     <string name="cant_get_spouts">"No se puede obtener la lista de fuentes."</string> | ||||
|     <string name="form_not_complete">"El formulario no está completo"</string> | ||||
|     <string name="pref_header_links">"Enlaces"</string> | ||||
|     <string name="issue_tracker_link">"Rastreador de Incidencias"</string> | ||||
| @@ -47,12 +40,9 @@ | ||||
|     <string name="switch_unread_count_title">"Mostrar recuento no leído"</string> | ||||
|     <string name="display_all_counts_title">"Mostrar recuento de favoritos y leídos"</string> | ||||
|     <string name="text_wrong_url">"Parece estar tratando de utilizar una dirección URL inválida. Asegúrese de que sea correcta y si el problema persiste, póngase en contacto conmigo (mediante el enlace de contacto de la tienda). Tenga en cuenta que la aplicación necesita utilizar Selfoss. No se puede acceder al contenido RSS sin él."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Abrir enlaces dentro de la aplicación"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Los artículos se abrirán dentro de la aplicación"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Los artículos se abrirán con tu navegador predeterminado"</string> | ||||
|     <string name="prefer_article_viewer_title">"Utilizar el visor de artículo"</string> | ||||
|     <string name="prefer_article_viewer_on">"Se usará el visor de artículos en lugar del navegador interno"</string> | ||||
|     <string name="prefer_article_viewer_off">"Se utilizará el navegador interno en lugar del visor de artículo"</string> | ||||
|     <string name="pref_article_viewer_title">"Abrir enlaces dentro de la aplicación"</string> | ||||
|     <string name="pref_article_viewer_on">"Los artículos se abrirán dentro de la aplicación"</string> | ||||
|     <string name="pref_article_viewer_off">"Los artículos se abrirán con tu navegador predeterminado"</string> | ||||
|     <string name="pref_general_category_links">"Control de enlaces"</string> | ||||
|     <string name="pref_general_category_displaying">"Mostrando"</string> | ||||
|     <string name="pref_switch_card_view_on">"Los artículos se mostrarán como tarjetas"</string> | ||||
| @@ -65,28 +55,18 @@ | ||||
|     <string name="card_height_on">Altura de tarjetas se ajustará a su contenido</string> | ||||
|     <string name="card_height_off">Se fijará la altura de la tarjeta</string> | ||||
|     <string name="source_code">Código fuente</string> | ||||
|     <string name="drawer_error_loading_tags">Error al cargar etiquetas…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpiar</string> | ||||
|     <string name="drawer_item_tags">Etiquetas</string> | ||||
|     <string name="drawer_item_sources">Fuentes</string> | ||||
|     <string name="drawer_action_edit">editar</string> | ||||
|     <string name="drawer_loading">Cargando…</string> | ||||
|     <string name="filter_item_tags">Etiquetas</string> | ||||
|     <string name="filter_item_sources">Fuentes</string> | ||||
|     <string name="menu_home_search">Buscar</string> | ||||
|     <string name="can_delete_source">No se puede eliminar la fuente…</string> | ||||
|     <string name="base_url_error">Hubo un problema al intentar comunicarse con su instancia de Selfoss. Si el problema persiste, póngase en contacto conmigo.</string> | ||||
|     <string name="pref_header_theme">Temas</string> | ||||
|     <string name="default_theme">Predeterminado</string> | ||||
|     <string name="default_dark_theme">Predeterminado/Oscuro</string> | ||||
|     <string name="pref_selfoss_category">Api de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Número de artículos cargados</string> | ||||
|     <string name="pref_hidden_tags">Etiquetas ocultas</string> | ||||
|     <string name="pref_general_infinite_loading_title">Cargar más artículos en desplazamiento</string> | ||||
|     <string name="translation">Traducción</string> | ||||
|     <string name="cant_open_invalid_url">La url del elemento no es válida. Estoy buscando resolver este problema para que la aplicación no colapse.</string> | ||||
|     <string name="drawer_report_bug">Reportar un error</string> | ||||
|     <string name="items_number_should_be_number">El número de artículos debe ser un número entero.</string> | ||||
|     <string name="reader_action_more">Leer más</string> | ||||
|     <string name="reader_action_open">Abrir en el navegador</string> | ||||
|     <string name="reader_action_share">Compartir</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Marcar artículos como leidos al desplazarse entre ellos.</string> | ||||
| @@ -97,7 +77,6 @@ | ||||
|     <string name="markall_dialog_message">Esto marcará todos los artículos como leídos.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Marcar artículos como leídos al deslizar con el dedo hacia los lados</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">No marcar artículos como leídos al deslizar con el dedo hacia los lados.</string> | ||||
|     <string name="drawer_item_hidden_tags">Etiquetas ocultas</string> | ||||
|     <string name="unmark">Marcar artículo como no leído</string> | ||||
|     <string name="pref_header_offline">Sin conexión y caché</string> | ||||
|     <string name="pref_switch_items_caching_off">Los artículos no se guardarán en la memoria del dispositivo y la aplicación no se podrá utilizar sin conexión.</string> | ||||
| @@ -105,14 +84,14 @@ | ||||
|     <string name="pref_switch_items_caching">Guardar elementos para uso sin conexión</string> | ||||
|     <string name="pref_switch_update_sources">Check for new sources and tags</string> | ||||
|     <string name="pref_switch_update_sources_summary">Disable this if your server is receiving excessive amounts of database queries.</string> | ||||
|     <string name="network_connectivity_lost">"Network connection lost"</string> | ||||
|     <string name="network_connectivity_lost">"Sin conexión!"</string> | ||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Sincronizar artículos</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Los artículos no se sincronizarán en segundo plano</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Los artículos se sincronizarán periódicamente</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Sólo refrescar cuando el teléfono está cargando</string> | ||||
|     <string name="loading_notification_title">Cargando...</string> | ||||
|     <string name="loading_notification_title">Cargando…</string> | ||||
|     <string name="loading_notification_text">Selfoss está sincronizando tus artículos</string> | ||||
|     <string name="notification_channel_sync">Notificación de sincronización</string> | ||||
|     <string name="new_items_channel_sync">Notificación de elementos nuevos</string> | ||||
| @@ -131,4 +110,26 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||
|     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||
|     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||
|     <string name="pref_switch_disable_acra">"Disable automatic bug reporting."</string> | ||||
|     <string name="menu_home_filter">Filters</string> | ||||
|     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||
|     <string name="menu_home_sources">Sources</string> | ||||
|     <string name="update_source">Update source</string> | ||||
|     <string name="confirm_disconnect_title">Disconnect ?</string> | ||||
|     <string name="confirm_disconnect_description">You will be disconnected from your selfoss instance.</string> | ||||
|     <string name="no_browser">Can\'t open a link without a browser installed</string> | ||||
|     <string name="nothing_here">"Nada aquí"</string> | ||||
|     <string name="tab_new">"Nuevo"</string> | ||||
|     <string name="tab_read">"Todo"</string> | ||||
|     <string name="tab_favs">"Favoritos"</string> | ||||
|     <string name="action_about">"Acerca de"</string> | ||||
|     <string name="marked_as_read">"Artículo leído"</string> | ||||
|     <string name="marked_as_unread">"Artículo no leído"</string> | ||||
| </resources> | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user