Compare commits
	
		
			398 Commits
		
	
	
		
			v122092701
			...
			cbe57b1b54
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| cbe57b1b54 | |||
| 02d503e03a | |||
| 24b9320d6d | |||
| ceba58e98f | |||
| c3ee07dd85 | |||
| 93d99192b3 | |||
| 359dec2ca0 | |||
| 62354ec70a | |||
| 18a17251ac | |||
| 5e91724ee2 | |||
| 212d259a33 | |||
| 3bf60f1146 | |||
| ef13e300f0 | |||
| f170d1157d | |||
| af4752f0f0 | |||
| f0fa1a17b6 | |||
| bb84d1541c | |||
| c9227b2c1c | |||
| 6eaad0c7c5 | |||
| a1c98aa7d0 | |||
| d5ec118679 | |||
| a1c0241a58 | |||
| f38936f9b4 | |||
| a90ccec707 | |||
| 2564b19726 | |||
| 61c7bb20cc | |||
| 6a0f5baf0a | |||
| 39f9505c00 | |||
| 6a6d447456 | |||
| 0bb4fe6aed | |||
| 7df4c3368c | |||
| c69635b5ae | |||
| 3a829df70e | |||
| 7a0202689f | |||
| b20f6888f5 | |||
| 6b96eb358d | |||
| dfc1bf9fa3 | |||
| b173664ff0 | |||
| bc20a421ae | |||
| 794500355a | |||
| 44f9dd53d3 | |||
| 717d6b664c | |||
| e23289a3dc | |||
| 2f5ebe2420 | |||
| 1893904135 | |||
| a4cb28ba81 | |||
| ae3cada1c7 | |||
| 309500276f | |||
| ce255b23cd | |||
| 3b3a575dae | |||
| 7bcf4574b4 | |||
| c79ab5e92b | |||
| 54dbda76ab | |||
| 11c39ae87c | |||
| 6645902ec8 | |||
| 0a07a5dfad | |||
| d88d38fd3b | |||
| 28fe38aa17 | |||
| d524c30732 | |||
| 8c00aa65da | |||
| ae81261cb1 | |||
| 03c567ee33 | |||
| d23dd82fc2 | |||
| 2e7a168424 | |||
| 5bc2f614af | |||
| 934c112db5 | |||
| ad7549a89f | |||
| fb9ceecabd | |||
| 61b9fd30e0 | |||
| 806e56e20b | |||
| cd8b7aaf9d | |||
| 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 | |||
| e2afff0b8e | |||
| a382fc89ea | |||
| 3f0a3903ae | |||
| f46f98cef0 | |||
| bf6f1a917e | |||
| 71c0a4d340 | |||
| 63c550ead3 | |||
| 366b2e10f1 | |||
| d2436bb976 | |||
| ef994460c1 | |||
| 758708e18d | |||
| c0381144d1 | |||
| cda3ba6cb4 | |||
| a4636cc0c8 | |||
| 60c24fc75a | |||
| 5853a19937 | |||
| 99f2c04bf6 | 
							
								
								
									
										126
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										126
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -1,126 +0,0 @@ | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: test | ||||
|  | ||||
| steps: | ||||
|   - name: Anylyse | ||||
|     image: mingc/android-build-box:latest | ||||
|     failure: ignore | ||||
|     detach: true | ||||
|     commands: | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Analysing..." | ||||
|       - ./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\"" | ||||
|       - echo "---------------------------------------------------------" | ||||
|     environment: | ||||
|       SONAR_HOST_URL: | ||||
|         from_secret: sonarScannerHostUrl | ||||
|       SONAR_LOGIN: | ||||
|         from_secret: sonarScannerLogin | ||||
|   - name: BuildAndTest | ||||
|     image: mingc/android-build-box:latest | ||||
|     commands: | ||||
|       - echo "Building..." | ||||
|       - ./gradlew :androidApp:build -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Testing..." | ||||
|       - echo "---------------------------------------------------------" | ||||
|     environment: | ||||
|       SONAR_HOST_URL: | ||||
|         from_secret: sonarScannerHostUrl | ||||
|       SONAR_LOGIN: | ||||
|         from_secret: sonarScannerLogin | ||||
| trigger: | ||||
|   event: | ||||
|     - push | ||||
|     - pull_request | ||||
|  | ||||
| --- | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: Publish | ||||
|  | ||||
| steps: | ||||
|   - name: createTag | ||||
|     image: ubuntu:latest | ||||
|     commands: | ||||
|       - apt-get update && apt-get install -y git | ||||
|       - ./build.sh --publish --from-ci | ||||
|       - git remote add pushing https://$GITEA_USR:$GITEA_PASS@gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform.git | ||||
|       - git push pushing --tags | ||||
|     environment: | ||||
|       GITEA_USR: | ||||
|         from_secret: giteaUsr | ||||
|       GITEA_PASS: | ||||
|         from_secret: giteaPass | ||||
|  | ||||
|   - name: scpFiles | ||||
|     image: appleboy/drone-scp | ||||
|     settings: | ||||
|       host: amine-louveau.fr | ||||
|       username: ubuntu | ||||
|       key: | ||||
|         from_secret: privateKey | ||||
|       port: 22 | ||||
|       target: /home/ubuntu/ | ||||
|       source: version.txt | ||||
|  | ||||
|   - name: deploy | ||||
|     image: appleboy/drone-ssh | ||||
|     settings: | ||||
|       host: amine-louveau.fr | ||||
|       user: ubuntu | ||||
|       key: | ||||
|         from_secret: privateKey | ||||
|       command_timeout: 2m | ||||
|       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/ | ||||
|  | ||||
| trigger: | ||||
|   event: | ||||
|     - promote | ||||
|   target: | ||||
|     - production | ||||
|  | ||||
| --- | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: Release | ||||
|  | ||||
| steps: | ||||
|   - name: build | ||||
|     image: mingc/android-build-box:latest | ||||
|     commands: | ||||
|       - echo "Generate APK" | ||||
|       - ./gradlew :androidApp:assembleGithubConfigRelease -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Get Key" | ||||
|       - wget https://amine-louveau.fr/key | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Zipalign" | ||||
|       - $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 | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Sign" | ||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner sign -v --out signed.apk --ks ./key --ks-key-alias $YOUR_KEY_ALIAS --ks-pass pass:$YOUR_KEYSTORE_PASSWORD --v1-signing-enabled true --v2-signing-enabled true androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Verify" | ||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk | ||||
|     environment: | ||||
|       YOUR_KEYSTORE_PASSWORD: | ||||
|         from_secret: keyPass | ||||
|       YOUR_KEY_ALIAS: | ||||
|         from_secret: keyAlias | ||||
|  | ||||
|   - name: gitea_release | ||||
|     image: plugins/gitea-release | ||||
|     settings: | ||||
|       api_key: | ||||
|         from_secret: giteaAPI | ||||
|       base_url: https://gitea.amine-louveau.fr | ||||
|       files: signed.apk | ||||
| trigger: | ||||
|   event: | ||||
|     - tag | ||||
							
								
								
									
										36
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| root = true | ||||
|  | ||||
| [*] | ||||
| insert_final_newline = true | ||||
|  | ||||
| [.editorconfig] | ||||
| insert_final_newline = false | ||||
| ij_kotlin_line_break_after_multiline_when_entry = false | ||||
|  | ||||
| [*.{kt,kts}] | ||||
| #  Disable wildcard imports entirely | ||||
| ij_kotlin_name_count_to_use_star_import = 2147483647 | ||||
| ij_kotlin_name_count_to_use_star_import_for_members = 2147483647 | ||||
| end_of_line = lf | ||||
| ij_kotlin_allow_trailing_comma = true | ||||
| ij_kotlin_allow_trailing_comma_on_call_site = true | ||||
| ij_kotlin_imports_layout = *, java.**, javax.**, kotlin.**, ^ | ||||
| ij_kotlin_indent_before_arrow_on_new_line = false | ||||
| ij_kotlin_line_break_after_multiline_when_entry = true | ||||
| ij_kotlin_packages_to_use_import_on_demand = unset | ||||
| indent_size = 4 | ||||
| indent_style = space | ||||
| ktlint_argument_list_wrapping_ignore_when_parameter_count_greater_or_equal_than = unset | ||||
| ktlint_chain_method_rule_force_multiline_when_chain_operator_count_greater_or_equal_than = 4 | ||||
| ktlint_class_signature_rule_force_multiline_when_parameter_count_greater_or_equal_than = 1 | ||||
| ktlint_code_style = ktlint_official | ||||
| ktlint_enum_entry_name_casing = upper_or_camel_cases | ||||
| ktlint_function_naming_ignore_when_annotated_with = unset | ||||
| ktlint_function_signature_body_expression_wrapping = multiline | ||||
| ktlint_function_signature_rule_force_multiline_when_parameter_count_greater_or_equal_than = 2 | ||||
| ktlint_ignore_back_ticked_identifier = false | ||||
| ktlint_property_naming_constant_naming = screaming_snake_case | ||||
| max_line_length = 140 | ||||
|  | ||||
| [**/build] | ||||
| ktlint = disabled | ||||
							
								
								
									
										10
									
								
								.gitea/workflows/assets/crowdin.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.gitea/workflows/assets/crowdin.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| project_id_env: CROWDIN_PROJECT_ID | ||||
| api_token_env: CROWDIN_PERSONAL_TOKEN | ||||
| base_path: "../../../" | ||||
|  | ||||
| files: | ||||
|   - source: /androidApp/src/main/res/values/strings.xml | ||||
|     translation: /androidApp/src/main/res/values-%android_code%/%original_file_name% | ||||
|     translate_attributes: '0' | ||||
|     content_segmentation: '0' | ||||
| preserve_hierarchy: true | ||||
							
								
								
									
										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: | ||||
|   BuildAndTestAndCoverage: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out repository code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: "Check android app changes" | ||||
|         id: check-android-changes | ||||
|         uses: tj-actions/changed-files@v45 | ||||
|         with: | ||||
|           files: | | ||||
|             androidApp/src/** | ||||
|       - name: Fetch tags | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|         run: git fetch --tags -p | ||||
|       - uses: actions/setup-java@v4 | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|           cache: gradle | ||||
|       - uses: gradle/actions/setup-gradle@v3 | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|       - uses: android-actions/setup-android@v3 | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|       - name: Configure gradle... | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|         run: mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true" >> ~/.gradle/gradle.properties | ||||
|       - name: Build and test | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|         run: ./gradlew build -x testReleaseUnitTest -x testDebugUnitTest -x testGithubConfigReleaseUnitTest -x testGithubConfigDebugUnitTest # These tests will be done | ||||
|       # TESTS ARE RUN LOCALLY | ||||
|       #      - 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 | ||||
|       - name: coverage | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|         run: | | ||||
|           ./gradlew :koverHtmlReport | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         if: steps.check-android-changes.outputs.any_modified == 'true' | ||||
|         with: | ||||
|           name: coverage | ||||
|           path: build/reports/kover/html | ||||
|           retention-days: 1 | ||||
|           overwrite: true | ||||
|           include-hidden-files: true | ||||
| #      TESTS ARE RUN LOCALLY | ||||
| #      - name: Clean | ||||
| #        if: always() | ||||
| #        run: | | ||||
| #          docker compose -f .gitea/workflows/assets/docker-compose.yml stop | ||||
							
								
								
									
										56
									
								
								.gitea/workflows/common_coverage.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								.gitea/workflows/common_coverage.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| name: Coverage | ||||
| on: | ||||
|   workflow_call: | ||||
|  | ||||
| jobs: | ||||
|   BuildAndTestAndCoverage: | ||||
|     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' | ||||
|           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 | ||||
|       - 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 | ||||
|       - name: Tests | ||||
|         uses: reactivecircus/android-emulator-runner@v2 | ||||
|         with: | ||||
|           api-level: 36 | ||||
|           target: google_apis | ||||
|           arch: x86_64 | ||||
|           script: ./gradlew androidApp:connectedAndroidTest | ||||
|           pre-emulator-launch-script: sdkmanager --list | grep system-images | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         if: failure() | ||||
|         with: | ||||
|           name: failure-espresso | ||||
|           path: build/reports/androidTests/connected/screenshots | ||||
|           retention-days: 2 | ||||
|           overwrite: true | ||||
|           include-hidden-files: true | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         with: | ||||
|           name: coverage-espresso | ||||
|           path: build/reports/coverage/androidTest/githubConfig/debug/connected | ||||
|           retention-days: 1 | ||||
|           overwrite: true | ||||
|           include-hidden-files: true | ||||
|       - name: Clean | ||||
|         if: always() | ||||
|         run: | | ||||
|           docker compose -f .gitea/workflows/assets/docker-compose.yml stop | ||||
							
								
								
									
										128
									
								
								.gitea/workflows/on_merge_on_release.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								.gitea/workflows/on_merge_on_release.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| 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 | ||||
|           ref: master | ||||
|       - 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: master | ||||
|       - 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' | ||||
|           cache: gradle | ||||
|       - 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 | ||||
							
								
								
									
										93
									
								
								.gitea/workflows/on_pr.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								.gitea/workflows/on_pr.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | ||||
| name: Check PR code | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - master | ||||
|  | ||||
| jobs: | ||||
|   EspressoReports: | ||||
|     runs-on: ubuntu-latest | ||||
|     uses: ./.gitea/workflows/common_coverage.yml | ||||
| #  Lint: | ||||
| #    runs-on: ubuntu-latest | ||||
| #    steps: | ||||
| #      - name: Check out repository code | ||||
| #        uses: actions/checkout@v4 | ||||
| #      - uses: actions/setup-java@v4 | ||||
| #        with: | ||||
| #          distribution: 'temurin' | ||||
| #          java-version: '17' | ||||
| #          cache: gradle | ||||
| #      - name: Install klint | ||||
| #        run: curl -sSLO https://github.com/pinterest/ktlint/releases/download/1.5.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.7/detekt-cli-1.23.7.zip && unzip detekt-cli-1.23.7.zip | ||||
| #      - name: Linting... | ||||
| #        run: ktlint 'shared/**/*.kt' 'androidApp/**/*.kt' '!shared/build' | ||||
| #      - name: Detecting... | ||||
| #        run: ./detekt-cli-1.23.7/bin/detekt-cli -c detekt.yml --excludes '**/shared/build/**/*.kt' | ||||
| #  translations: | ||||
| #    runs-on: ubuntu-latest | ||||
| #    steps: | ||||
| #      - name: Check out repository code | ||||
| #        uses: actions/checkout@v4 | ||||
| #        with: | ||||
| #          fetch-depth: 0 | ||||
| #      - name: "Check translations changes" | ||||
| #        id: check-translations-changes | ||||
| #        uses: tj-actions/changed-files@v45 | ||||
| #        with: | ||||
| #          files: | | ||||
| #            androidApp/src/main/res/values/strings.xml | ||||
| #      - name: upload translation sources | ||||
| #        if: steps.check-api-changes.outputs.any_modified == 'true' | ||||
| #        uses: crowdin/github-action@v2 | ||||
| #        with: | ||||
| #          config: './.gitea/workflows/assets/crowdin.yml' | ||||
| #          upload_sources: true | ||||
| #          upload_translations: false | ||||
| #          download_translations: false | ||||
| #          create_pull_request: false | ||||
| #          push_translations: false | ||||
| #        env: | ||||
| #          CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }} | ||||
| #          CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }} | ||||
| #      - name: wait | ||||
| #        if: steps.check-api-changes.outputs.any_modified == 'true' | ||||
| #        run: sleep 10s | ||||
| #      - name: download translations | ||||
| #        if: steps.check-api-changes.outputs.any_modified == 'true' | ||||
| #        uses: crowdin/github-action@v2 | ||||
| #        with: | ||||
| #          config: './.gitea/workflows/assets/crowdin.yml' | ||||
| #          upload_sources: false | ||||
| #          upload_translations: false | ||||
| #          download_translations: true | ||||
| #          create_pull_request: false | ||||
| #          push_translations: false | ||||
| #        env: | ||||
| #          CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }} | ||||
| #          CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }} | ||||
| #      - name: Check for uncommitted changes | ||||
| #        if: steps.check-api-changes.outputs.any_modified == 'true' | ||||
| #        id: check-changes | ||||
| #        uses: mskri/check-uncommitted-changes-action@v1.0.1 | ||||
| #      - name: Commit Changes | ||||
| #        if: steps.check-api-changes.outputs.any_modified == 'true' && steps.check-changes.outputs.changes != '' | ||||
| #        run: | | ||||
| #          git config --global user.email aminecmi+giteadrone@pm.me | ||||
| #          git config --global user.name giteadrone | ||||
| #          git add ./androidApp/src/main/res/* | ||||
| #          git commit -m "translation: translation files" | ||||
| #      - name: Push changes | ||||
| #        if: steps.check-api-changes.outputs.any_modified == 'true' && steps.check-changes.outputs.changes != '' | ||||
| #        uses: appleboy/git-push-action@v1.0.0 | ||||
| #        with: | ||||
| #          author_name: giteadrone | ||||
| #          author_email: aminecmi+giteadrone@pm.me | ||||
| #          remote: ${{ secrets.REMOTE_URL }} | ||||
| #          ssh_key: ${{ secrets.PRIVATE_KEY }} | ||||
| #          branch: ${{ github.head_ref || github.ref_name }} | ||||
| #  build: | ||||
| #    needs: Lint | ||||
| #    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" | ||||
| ``` | ||||
|   | ||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -320,4 +320,9 @@ fabric.properties | ||||
| # End of https://www.toptal.com/developers/gitignore/api/gradle,kotlin,androidstudio,android,xcode,swift | ||||
|  | ||||
|  | ||||
| crowdin.properties | ||||
| crowdin.properties | ||||
|  | ||||
| .kotlin/ | ||||
| build-cache/ | ||||
|  | ||||
| act | ||||
|   | ||||
							
								
								
									
										441
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										441
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,444 @@ | ||||
| **v125030711 | ||||
|  | ||||
| - Merge pull request 'fix: initial status loading issues.' (#192) from connectivity into master | ||||
| - chore: check changes for translations and android. | ||||
| - fix: initial status loading issues. | ||||
| - Merge pull request 'chore: new connectivity dep. Closes #84.' (#189) from connectivity into master | ||||
| - chore: new connectivity dep. Closes #84. | ||||
| - Changelog for v125030681 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125030681 | ||||
|  | ||||
| - chore: do not send reports on simulators. | ||||
| - Merge pull request 'chore: do not send reports on simulators.' (#188) from chore-acra-simulator into master | ||||
| - chore: do not send reports on simulators. | ||||
| - Merge pull request 'fix: Url validation was not failing login. Added tests.' (#186) from fix-invalid-url into master | ||||
| - Merge pull request 'chore: crowding ci integration.' (#187) from chore-crowdin-ci into master | ||||
| - chore: we don't need to check if the url is valid in upsert screen. | ||||
| - fix: Url validation was not failing login. Added tests. | ||||
| - chore: crowding ci integration. | ||||
| - Show a confirmation dialog before deleting sources (#185) | ||||
| - Changelog for v125020581 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125020581 | ||||
|  | ||||
| - fix: url can be empty ? | ||||
| - Changelog for v125020471 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125020471 | ||||
|  | ||||
| - chore: no more docker-compose. | ||||
| - bump: gradle plugin. | ||||
| - Merge pull request 'fix: check index exists.' (#183) from fix-index into master | ||||
| - fix: check index exists. | ||||
| - Changelog for v125020411 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125020411 | ||||
|  | ||||
| - Merge pull request 'bump' (#182) from bump into master | ||||
| - chore: non transiant R classes. | ||||
| - Merge pull request 'fix: One more missing context.' (#181) from fix-one-more-context into master | ||||
| - bump | ||||
| - fix: One more missing context. | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010241 | ||||
|  | ||||
| - Merge pull request 'fix: Link not opening.' (#178) from fix-open-link into master | ||||
| - refactor: context fragments issues. | ||||
| - logs: Context issues. | ||||
| - fix: Handle empty url issue, again. | ||||
| - fix: Link not opening. | ||||
| - Changelog for v125010201 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010201 | ||||
|  | ||||
| - fix: Handle empty url issue. | ||||
| - Merge pull request 'Removed the floating bar.' (#177) from floating-bar into master | ||||
| - chore: changing actions in reader fragment. | ||||
| - Changelog for v125010131 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010131 | ||||
|  | ||||
| - fix: reload the adapter when it's needed. Fixes #128. (#176) | ||||
| - feat: basic auth and images loading. Fixes #172. (#175) | ||||
| - Changelog for v125010111 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010111 | ||||
|  | ||||
| - Debug trying to fix context issues. (#174) | ||||
| - Changelog for v125010031 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v125010031 | ||||
|  | ||||
| - Merge pull request 'Bump dependencies' (#173) from upgarde into master | ||||
| - chore: "faster" action. | ||||
| - fastlane: icon change. | ||||
| - chore: ignoring a pixel issue. | ||||
| - test: fixed an ui test issue. | ||||
| - fix: center the loading thing. | ||||
| - test: items displaying. | ||||
| - bump: sqldelight. | ||||
| - bump: material, desugar jdk, jsoup, kodein, settings, napier, mock. | ||||
| - bump: androix and coroutines. | ||||
| - bump: ktor. Closes #67. | ||||
| - Changelog for v124123651 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **v124123651 | ||||
|  | ||||
| - Merge pull request 'Bugfixes' (#171) from bugfixes into master | ||||
| - config: crowdin | ||||
| - chore: can links be empty ? | ||||
| - fix: Context issues in article fragment. | ||||
| - fix: Context issues in fragment sheet. | ||||
| - fix: build. | ||||
| - chore: compile issue fix. | ||||
| - chore: filter some bugs. | ||||
| - bugfix: catch users using something other than selfoss. | ||||
| - bugfix: No browser, no link. | ||||
| - translations | ||||
| - chore: remove log. | ||||
| - translation | ||||
| - Changelog for v124123641 | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|  | ||||
| **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 [](https://build.amine-louveau.fr/Louvorg/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,36 +1,49 @@ | ||||
| import java.io.ByteArrayOutputStream | ||||
|  | ||||
| val ignoreGitVersion: String by project | ||||
| val acraVersion = "5.12.0" | ||||
|  | ||||
| plugins { | ||||
|     id("com.android.application") | ||||
|     kotlin("android") | ||||
|     kotlin("kapt") | ||||
|     id("com.mikepenz.aboutlibraries.plugin") | ||||
|     id("org.jetbrains.kotlinx.kover") | ||||
|     id("app.cash.sqldelight") version "2.0.2" | ||||
| } | ||||
|  | ||||
| fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||
|     var result: String = ByteArrayOutputStream().use { outputStream -> | ||||
|         project.exec { | ||||
|             commandLine = cmd.split(" ") | ||||
|             standardOutput = outputStream | ||||
|             isIgnoreExitValue = ignore ?: false | ||||
| fun Project.execWithOutput( | ||||
|     cmd: String, | ||||
|     ignore: Boolean = false, | ||||
| ): String { | ||||
|     val result: String = | ||||
|         ByteArrayOutputStream().use { outputStream -> | ||||
|             project.exec { | ||||
|                 commandLine = cmd.split(" ") | ||||
|                 standardOutput = outputStream | ||||
|                 isIgnoreExitValue = ignore | ||||
|             } | ||||
|             outputStream.toString() | ||||
|         } | ||||
|         outputStream.toString() | ||||
|     } | ||||
|     return result | ||||
| } | ||||
|  | ||||
| fun gitVersion(): String { | ||||
|     var process = "" | ||||
|     val maybeTagOfCurrentCommit = execWithOutput("git -C ../ describe --contains HEAD", true) | ||||
|     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") | ||||
|     } else { | ||||
|         println("Tag found on current commit") | ||||
|         execWithOutput("git -C ../ describe --contains HEAD") | ||||
|     } | ||||
|     return process.replace("^0", "").replace("'", "").substring(1).replace("\\.", "").trim() | ||||
|     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=-refname --format='%(refname:short)' --count=1") | ||||
|         } else { | ||||
|             println("Tag found on current commit") | ||||
|             execWithOutput("git -C ../ describe --contains HEAD") | ||||
|         } | ||||
|     return process | ||||
|         .replace("^0", "") | ||||
|         .replace("'", "") | ||||
|         .substring(1) | ||||
|         .replace("\\.", "") | ||||
|         .trim() | ||||
| } | ||||
|  | ||||
| fun versionCodeFromGit(): Int { | ||||
| @@ -53,20 +66,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 = 35 | ||||
|     buildFeatures { | ||||
|         viewBinding = true | ||||
|     } | ||||
|     defaultConfig { | ||||
|         applicationId = "bou.amine.apps.readerforselfossv2.android" | ||||
|         minSdk = 21 | ||||
|         targetSdk = 31 | ||||
|         minSdk = 25 | ||||
|         targetSdk = 34 // 35 when edge-to-edge is handled | ||||
|         versionCode = versionCodeFromGit() | ||||
|         versionName = versionNameFromGit() | ||||
|  | ||||
| @@ -78,6 +95,13 @@ android { | ||||
|  | ||||
|         // tests | ||||
|         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||
|         testInstrumentationRunnerArguments["clearPackageData"] = "true" | ||||
|         testInstrumentationRunnerArguments["useTestStorageService"] = "true" | ||||
|     } | ||||
|     packaging { | ||||
|         resources { | ||||
|             excludes += "/META-INF/{AL2.0,LGPL2.1}" | ||||
|         } | ||||
|     } | ||||
|     buildTypes { | ||||
|         getByName("release") { | ||||
| @@ -86,9 +110,11 @@ 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) | ||||
|             isTestCoverageEnabled = true | ||||
|             enableAndroidTestCoverage = true | ||||
|             installation { | ||||
|                 installOptions("-g", "-r") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     flavorDimensions.add("build") | ||||
| @@ -98,104 +124,160 @@ 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.1.4") | ||||
|  | ||||
|     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("androidx.appcompat:appcompat:1.7.0") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.10.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("com.google.android.material:material:1.12.0") | ||||
|     implementation("androidx.recyclerview:recyclerview:1.4.0-rc01") | ||||
|     implementation("androidx.legacy:legacy-support-v4:1.0.0") | ||||
|     implementation("androidx.vectordrawable:vectordrawable:1.2.0-alpha02") | ||||
|     implementation("androidx.vectordrawable:vectordrawable:1.2.0") | ||||
|     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") | ||||
|     implementation("androidx.annotation:annotation:1.9.1") | ||||
|     implementation("androidx.work:work-runtime-ktx:2.10.0") | ||||
|     implementation("androidx.constraintlayout:constraintlayout:2.2.0") | ||||
|     implementation("org.jsoup:jsoup:1.18.3") | ||||
|  | ||||
|     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:1.1.5") | ||||
|  | ||||
|     //multidex | ||||
|     // 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.16.0") | ||||
|     implementation("com.github.bumptech.glide:okhttp3-integration:4.16.0") | ||||
|  | ||||
|     // Themes | ||||
|     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") | ||||
|     implementation("com.leinardi.android:speed-dial:3.3.0") | ||||
|  | ||||
|     // Pager | ||||
|     implementation("me.relex:circleindicator:2.1.6") | ||||
|     implementation("androidx.viewpager2:viewpager2:1.1.0-beta01") | ||||
|     implementation("androidx.viewpager2:viewpager2:1.1.0") | ||||
|  | ||||
|     //Dependency Injection | ||||
|     implementation("org.kodein.di:kodein-di:7.14.0") | ||||
|     implementation("org.kodein.di:kodein-di-framework-android-x:7.14.0") | ||||
|     implementation("org.kodein.di:kodein-di-framework-android-x-viewmodel:7.14.0") | ||||
|     // Dependency Injection | ||||
|     implementation("org.kodein.di:kodein-di:7.23.1") | ||||
|     implementation("org.kodein.di:kodein-di-framework-android-x:7.23.1") | ||||
|     implementation("org.kodein.di:kodein-di-framework-android-x-viewmodel:7.23.1") | ||||
|  | ||||
|     //Settings | ||||
|     implementation("com.russhwolf:multiplatform-settings-no-arg:0.9") | ||||
|     // Settings | ||||
|     implementation("com.russhwolf:multiplatform-settings-no-arg:1.3.0") | ||||
|  | ||||
|     //Logging | ||||
|     implementation("io.github.aakira:napier:2.6.1") | ||||
|     // Logging | ||||
|     implementation("io.github.aakira:napier:2.7.1") | ||||
|  | ||||
|     //PhotoView | ||||
|     // PhotoView | ||||
|     implementation("com.github.chrisbanes:PhotoView:2.3.0") | ||||
|  | ||||
|     implementation("androidx.core:core-ktx:1.8.0") | ||||
|     implementation("androidx.core:core-ktx:1.15.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("app.cash.sqldelight:android-driver:2.0.2") | ||||
|  | ||||
|     // test | ||||
|     testImplementation("junit:junit:4.13.2") | ||||
|     testImplementation("io.mockk:mockk:1.13.14") | ||||
|     testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.10.1") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.6.1") | ||||
|     androidTestImplementation("androidx.test:runner:1.7.0-alpha01") | ||||
|     androidTestImplementation("androidx.test:rules:1.7.0-alpha01") | ||||
|     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.6.0-alpha02") | ||||
|     androidTestUtil("androidx.test.services:test-services:1.6.0-alpha02") | ||||
|     testImplementation("org.robolectric:robolectric:4.14.1") | ||||
|     testImplementation("androidx.test:core-ktx:1.7.0-alpha01") | ||||
|     androidTestImplementation("androidx.test.uiautomator:uiautomator:2.3.0") | ||||
|  | ||||
|     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 { | ||||
|     excludeFields = arrayOf("generated") | ||||
|     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 | ||||
| } | ||||
|  | ||||
| // Screenshot failure handling | ||||
| val reportsDirectory = file("$buildDir/reports/androidTests/connected") | ||||
|  | ||||
| val clearScreenshotsTask = | ||||
|     tasks.register<Exec>("clearScreenshots") { | ||||
|         println("AMINE : clear") | ||||
|         commandLine = listOf("adb", "shell", "rm", "-r", "/sdcard/Pictures/selfoss_tests") | ||||
|     } | ||||
|  | ||||
| val createScreenshotDirectoryTask = | ||||
|     tasks.register<Exec>("createScreenshotDirectory") { | ||||
|         println("AMINE : create directory") | ||||
|         group = "reporting" | ||||
|         commandLine = listOf("adb", "shell", "mkdir", "-p", "/sdcard/Pictures/selfoss_tests") | ||||
|     } | ||||
|  | ||||
| val fetchScreenshotsTask = | ||||
|     tasks.register<Exec>("fetchScreenshots") { | ||||
|         println("AMINE : fetch") | ||||
|         group = "reporting" | ||||
|         executable(android.adbExecutable.toString()) | ||||
|         commandLine = listOf("adb", "pull", "/sdcard/Pictures/selfoss_tests/.", reportsDirectory.toString()) | ||||
|  | ||||
|         finalizedBy(clearScreenshotsTask) | ||||
|         dependsOn(createScreenshotDirectoryTask) | ||||
|  | ||||
|         doFirst { | ||||
|             reportsDirectory.mkdirs() | ||||
|         } | ||||
|     } | ||||
|  | ||||
| tasks.whenTaskAdded { | ||||
|     if (this.name == "connectedAndroidTest") { | ||||
|         this.finalizedBy(fetchScreenshotsTask) | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										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,220 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import android.os.Environment.DIRECTORY_PICTURES | ||||
| import android.os.Environment.getExternalStoragePublicDirectory | ||||
| import android.util.Log | ||||
| import androidx.annotation.ArrayRes | ||||
| import androidx.test.espresso.Espresso | ||||
| import androidx.test.espresso.Espresso.onData | ||||
| 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.base.DefaultFailureHandler | ||||
| 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 androidx.test.platform.app.InstrumentationRegistry.getInstrumentation | ||||
| import androidx.test.runner.screenshot.BasicScreenCaptureProcessor | ||||
| import androidx.test.runner.screenshot.Screenshot | ||||
| import androidx.test.uiautomator.UiDevice | ||||
| import androidx.test.uiautomator.UiSelector | ||||
| import org.hamcrest.CoreMatchers.allOf | ||||
| import org.hamcrest.Matchers.hasToString | ||||
| import org.junit.BeforeClass | ||||
| import org.junit.rules.TestWatcher | ||||
| import org.junit.runner.Description | ||||
| import java.io.File | ||||
| import java.io.IOException | ||||
| import java.util.Locale | ||||
|  | ||||
| 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()) | ||||
| } | ||||
|  | ||||
| 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()))) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun testAddSourceWithUrl( | ||||
|     url: String, | ||||
|     sourceName: String, | ||||
| ) { | ||||
|     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(url)) | ||||
|     onView(withId(R.id.tags)) | ||||
|         .perform(click()) | ||||
|         .perform(typeTextIntoFocusedView("tag1,tag2,tag3")) | ||||
|     onView(withId(R.id.spoutsSpinner)) | ||||
|         .perform(click()) | ||||
|     onData(hasToString("RSS Feed")).perform(click()) | ||||
|     onView(withId(R.id.saveBtn)) | ||||
|         .perform(click()) | ||||
|     onView(withText(sourceName)).check(matches(isDisplayed())) | ||||
| } | ||||
|  | ||||
| open class WithANRException { | ||||
|     companion object { | ||||
|         // Running count of the number of Android Not Responding dialogues to prevent endless dismissal. | ||||
|         private var anrCount = 0 | ||||
|  | ||||
|         // `RootViewWithoutFocusException` class is private, need to match the message (instead of using type matching). | ||||
|         private val rootViewWithoutFocusExceptionMsg = | ||||
|             java.lang.String.format( | ||||
|                 Locale.ROOT, | ||||
|                 "Waited for the root of the view hierarchy to have " + | ||||
|                     "window focus and not request layout for 10 seconds. If you specified a non " + | ||||
|                     "default root matcher, it may be picking a root that never takes focus. " + | ||||
|                     "Root:", | ||||
|             ) | ||||
|  | ||||
|         private fun handleAnrDialogue() { | ||||
|             val device = UiDevice.getInstance(getInstrumentation()) | ||||
|             // If running the device in English Locale | ||||
|             val waitButton = device.findObject(UiSelector().textContains("wait")) | ||||
|             if (waitButton.exists()) waitButton.click() | ||||
|         } | ||||
|  | ||||
|         @JvmStatic | ||||
|         @BeforeClass | ||||
|         fun setUpHandler() { | ||||
|             Espresso.setFailureHandler { error, viewMatcher -> | ||||
|  | ||||
|                 if (error.message!!.contains(rootViewWithoutFocusExceptionMsg) && anrCount < 3) { | ||||
|                     anrCount++ | ||||
|                     handleAnrDialogue() | ||||
|                 } else { // chain all failures down to the default espresso handler | ||||
|                     DefaultFailureHandler(getInstrumentation().targetContext).handle(error, viewMatcher) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| class MyScreenCaptureProcessor( | ||||
|     parentFolderPath: String, | ||||
| ) : BasicScreenCaptureProcessor() { | ||||
|     init { | ||||
|         this.mDefaultScreenshotPath = | ||||
|             File( | ||||
|                 File( | ||||
|                     getExternalStoragePublicDirectory(DIRECTORY_PICTURES), | ||||
|                     "selfoss_tests", | ||||
|                 ).absolutePath, | ||||
|                 "screenshots/$parentFolderPath", | ||||
|             ) | ||||
|     } | ||||
|  | ||||
|     override fun getFilename(prefix: String): String = prefix | ||||
| } | ||||
|  | ||||
| fun takeScreenshot( | ||||
|     parentFolderPath: String = "", | ||||
|     screenShotName: String, | ||||
| ) { | ||||
|     Log.d("Screenshots", "Taking screenshot of '$screenShotName'") | ||||
|     val screenCapture = Screenshot.capture() | ||||
|     val processors = setOf(MyScreenCaptureProcessor(parentFolderPath)) | ||||
|     try { | ||||
|         screenCapture.apply { | ||||
|             name = screenShotName | ||||
|             process(processors) | ||||
|         } | ||||
|         Log.d("Screenshots", "Screenshot taken") | ||||
|     } catch (ex: IOException) { | ||||
|         Log.e("Screenshots", "Could not take the screenshot", ex) | ||||
|     } | ||||
| } | ||||
|  | ||||
| class ScreenshotTakingRule : TestWatcher() { | ||||
|     override fun failed( | ||||
|         e: Throwable?, | ||||
|         description: Description, | ||||
|     ) { | ||||
|         val parentFolderPath = "failures/${description.className}" | ||||
|         takeScreenshot(parentFolderPath = parentFolderPath, screenShotName = description.methodName) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,110 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| 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.core.app.ApplicationProvider | ||||
| import androidx.test.espresso.Espresso.openActionBarOverflowOrOptionsMenu | ||||
| 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 && (view !is EditText || view.error == null)) { | ||||
|                 return false | ||||
|             } | ||||
|             val context = view!!.context | ||||
|  | ||||
|             return (view as EditText).error.toString() == context.getString(id) | ||||
|         } | ||||
|  | ||||
|         override fun describeTo(description: Description?) { | ||||
|             // Nothing | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun isPopupWindow(): Matcher<Root> = 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") | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:SwallowedException") | ||||
|     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>? = | ||||
|     allOf( | ||||
|         withResourceName("fixed_bottom_navigation_icon"), | ||||
|         withParent( | ||||
|             allOf( | ||||
|                 withResourceName("fixed_bottom_navigation_icon_container"), | ||||
|                 hasSibling(withText(id)), | ||||
|             ), | ||||
|         ), | ||||
|     ) | ||||
|  | ||||
| fun withSettingsCheckboxWidget( | ||||
|     @StringRes id: Int, | ||||
| ): Matcher<View>? = | ||||
|     allOf( | ||||
|         withId(android.R.id.switch_widget), | ||||
|         withParent( | ||||
|             withSettingsCheckboxFrame(id), | ||||
|         ), | ||||
|     ) | ||||
|  | ||||
| fun withSettingsCheckboxFrame( | ||||
|     @StringRes id: Int, | ||||
| ): Matcher<View>? = | ||||
|     allOf( | ||||
|         withId(android.R.id.widget_frame), | ||||
|         hasSibling( | ||||
|             allOf( | ||||
|                 withClassName(Matchers.equalTo(RelativeLayout::class.java.name)), | ||||
|                 withChild( | ||||
|                     withText(id), | ||||
|                 ), | ||||
|             ), | ||||
|         ), | ||||
|     ) | ||||
|  | ||||
| fun openMenu() { | ||||
|     openActionBarOverflowOrOptionsMenu( | ||||
|         ApplicationProvider.getApplicationContext<Context>(), | ||||
|     ) | ||||
| } | ||||
| @@ -0,0 +1,126 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| 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.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class HomeActivityTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         loginAndInitHome() | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testMenu() { | ||||
|         onView(withId(R.id.action_search)).check(matches(not(isDisplayed()))).check( | ||||
|             matches( | ||||
|                 isClickable(), | ||||
|             ), | ||||
|         ) | ||||
|         onView(withId(R.id.action_filter)).check(matches(isDisplayed())).check( | ||||
|             matches( | ||||
|                 isClickable(), | ||||
|             ), | ||||
|         ) | ||||
|         openMenu() | ||||
|         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(com.google.android.material.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()) | ||||
|  | ||||
|         openMenu() | ||||
|         onView(withText(R.string.readAll)).perform(click()) | ||||
|         onView(withText(R.string.markall_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openMenu() | ||||
|  | ||||
|         onView(withText(R.string.menu_home_sources)).perform(click()) | ||||
|         onView(withId(R.id.fab)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openMenu() | ||||
|  | ||||
|         onView(withText(R.string.title_activity_settings)).perform(click()) | ||||
|         onView(withText(R.string.pref_header_general)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openMenu() | ||||
|  | ||||
|         onView(withText(R.string.menu_home_refresh)).perform(click()) | ||||
|         onView(withText(R.string.refresh_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openMenu() | ||||
|  | ||||
|         /*onView(withText(R.string.issue_tracker_link)).perform(click()) | ||||
|         onView(withText(R.string.markall_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openMenu()*/ | ||||
|  | ||||
|         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,101 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| 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.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class LoginActivityTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     @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.login_url_problem))) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun connectError() { | ||||
|         performLogin("http://10.0.2.2:8889") | ||||
|         onView(withId(R.id.urlView)).perform(click()) | ||||
|         onView(withId(R.id.urlView)).check(matches(withError(R.string.wrong_infos))) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun urlSlashError() { | ||||
|         performLogin("https://google.fr/toto") | ||||
|         onView(withId(R.id.urlView)).perform(click()) | ||||
|         onView(withId(R.id.urlView)).check(matches(withError(R.string.login_url_problem))) | ||||
|     } | ||||
|  | ||||
|     @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,163 @@ | ||||
| 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.swipeUp | ||||
| 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.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityGeneralTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     @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()) | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:LongMethod") | ||||
|     @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(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(withId(R.id.settings)).perform(swipeUp()) | ||||
|         onView(withSettingsCheckboxWidget(R.string.display_all_counts_title)).check( | ||||
|             matches( | ||||
|                 allOf( | ||||
|                     isDisplayed(), | ||||
|                     not(isChecked()), | ||||
|                 ), | ||||
|             ), | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:ForbiddenComment") | ||||
|     @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() { | ||||
|         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,182 @@ | ||||
| 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.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityOfflineTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     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()) | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:LongMethod") | ||||
|     @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(), | ||||
|                 ), | ||||
|             ), | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:LongMethod") | ||||
|     @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,93 @@ | ||||
| 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.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityReaderTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     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,101 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| 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.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SettingsActivityTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     lateinit var context: Context | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         activityRule.scenario.onActivity { activity -> | ||||
|             context = activity.window.context | ||||
|         } | ||||
|         loginAndInitHome() | ||||
|         openMenu() | ||||
|         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,92 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import androidx.test.espresso.AmbiguousViewMatcherException | ||||
| import androidx.test.espresso.Espresso.onView | ||||
| import androidx.test.espresso.action.ViewActions | ||||
| import androidx.test.espresso.action.ViewActions.click | ||||
| import androidx.test.espresso.action.ViewActions.swipeDown | ||||
| import androidx.test.espresso.assertion.ViewAssertions.doesNotExist | ||||
| import androidx.test.espresso.assertion.ViewAssertions.matches | ||||
| import androidx.test.espresso.matcher.ViewMatchers.isDisplayed | ||||
| 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.junit.After | ||||
| import org.junit.Before | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.rules.RuleChain | ||||
| import org.junit.runner.RunWith | ||||
| import java.util.UUID | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @LargeTest | ||||
| class SourcesActivityTest : WithANRException() { | ||||
|     @get:Rule | ||||
|     val activityRule = ActivityScenarioRule(LoginActivity::class.java) | ||||
|  | ||||
|     @JvmField | ||||
|     @Rule | ||||
|     val ruleChain: RuleChain = | ||||
|         RuleChain | ||||
|             .outerRule(activityRule) | ||||
|             .around(ScreenshotTakingRule()) | ||||
|  | ||||
|     lateinit var sourceName: String | ||||
|  | ||||
|     @Before | ||||
|     fun init() { | ||||
|         sourceName = UUID.randomUUID().toString().substring(0, 15) | ||||
|  | ||||
|         loginAndInitHome() | ||||
|         goToSources() | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun addSource() { | ||||
|         testAddSourceWithUrl( | ||||
|             "https://lorem-rss.herokuapp.com/feed?unit=year&interval=1&length=10", | ||||
|             sourceName, | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:SwallowedException") | ||||
|     @Test | ||||
|     fun addSourceCheckContent() { | ||||
|         testAddSourceWithUrl("https://news.google.com/rss?hl=en-US&gl=US&ceid=US:en", sourceName) | ||||
|         onView(isRoot()).perform(ViewActions.pressBack()) | ||||
|         openMenu() | ||||
|         onView(withText(R.string.menu_home_refresh)).perform(click()) | ||||
|         onView(withText(R.string.refresh_dialog_message)).check(matches(isDisplayed())) | ||||
|         onView( | ||||
|             withId(android.R.id.button1), | ||||
|         ).perform(click()) | ||||
|         Thread.sleep(10000) | ||||
|         onView(withId(R.id.swipeRefreshLayout)).perform(swipeDown()) | ||||
|         Thread.sleep(2000) | ||||
|         try { | ||||
|             onView(withId(R.id.sourceTitleAndDate)).check(matches(isDisplayed())) | ||||
|         } catch (e: AmbiguousViewMatcherException) { | ||||
|             assert(true) | ||||
|         } | ||||
|         goToSources() | ||||
|     } | ||||
|  | ||||
|     @After | ||||
|     fun deleteTheCreatedSource() { | ||||
|         onView(withText(sourceName)).check(matches(isDisplayed())) | ||||
|         onView(withId(R.id.deleteBtn)).perform(click()) | ||||
|         onView(withText(R.string.confirm_delete_title)).check(matches(isDisplayed())) | ||||
|         onView(withId(android.R.id.button1)).perform(click()) | ||||
|         onView(withText(sourceName)).check(doesNotExist()) | ||||
|     } | ||||
|  | ||||
|     private fun goToSources() { | ||||
|         openMenu() | ||||
|         onView(withText(R.string.menu_home_sources)) | ||||
|             .perform(click()) | ||||
|     } | ||||
| } | ||||
| @@ -2,6 +2,7 @@ | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/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" /> | ||||
|  | ||||
| @@ -15,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" | ||||
| @@ -51,7 +53,7 @@ | ||||
|                 android:value=".HomeActivity" /> | ||||
|         </activity> | ||||
|         <activity | ||||
|             android:name=".AddSourceActivity" | ||||
|             android:name=".UpsertSourceActivity" | ||||
|             android:parentActivityName=".SourcesActivity" | ||||
|             android:exported="true"> | ||||
|             <meta-data | ||||
| @@ -68,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" | ||||
| @@ -78,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,172 +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.utils.isBaseUrlInvalid | ||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||
| 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 AddSourceActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private var mSpoutsValue: String? = null | ||||
|  | ||||
|     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?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityAddSourceBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         setSupportActionBar(binding.toolbar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         maybeGetDetailsFromIntentSharing(intent, binding.sourceUri, binding.nameInput) | ||||
|  | ||||
|         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.isBaseUrlInvalid()) { | ||||
|             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 | ||||
| @@ -10,8 +11,8 @@ import bou.amine.apps.readerforselfossv2.android.databinding.ActivityImageBindin | ||||
| import bou.amine.apps.readerforselfossv2.android.fragments.ImageFragment | ||||
|  | ||||
| class ImageActivity : AppCompatActivity() { | ||||
|     private lateinit var allImages : ArrayList<String> | ||||
|     private var position : Int = 0 | ||||
|     private lateinit var allImages: ArrayList<String> | ||||
|     private var position: Int = 0 | ||||
|  | ||||
|     private lateinit var binding: ActivityImageBinding | ||||
|  | ||||
| @@ -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 | ||||
|             } | ||||
|         } | ||||
| @@ -44,10 +84,11 @@ class ImageActivity : AppCompatActivity() { | ||||
|         return super.onOptionsItemSelected(item) | ||||
|     } | ||||
|  | ||||
|     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : FragmentStateAdapter(fa) { | ||||
|  | ||||
|     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,10 +12,12 @@ 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.testing.CountingIdlingResourceSingleton | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -21,24 +25,29 @@ 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 const val MAX_INVALID_LOGIN_BEFORE_ALERT_DISPLAYED = 3 | ||||
|  | ||||
| class LoginActivity : | ||||
|     AppCompatActivity(), | ||||
|     DIAware { | ||||
|     private var inValidCount: Int = 0 | ||||
|     private var isWithLogin = false | ||||
|  | ||||
|     private lateinit var binding: ActivityLoginBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|     private val repository: Repository by instance() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM) | ||||
|  | ||||
|         handleTheme() | ||||
|  | ||||
|         binding = ActivityLoginBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
| @@ -50,14 +59,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) { | ||||
| @@ -65,7 +79,7 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|                     return@OnEditorActionListener true | ||||
|                 } | ||||
|                 false | ||||
|             } | ||||
|             }, | ||||
|         ) | ||||
|  | ||||
|         binding.signInButton.setOnClickListener { attemptLogin() } | ||||
| @@ -86,19 +100,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) | ||||
| @@ -107,71 +130,119 @@ 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().trim() | ||||
|         val login = binding.loginView.text.toString().trim() | ||||
|         val password = binding.passwordView.text.toString().trim() | ||||
|         val url = | ||||
|             binding.urlView.text | ||||
|                 .toString() | ||||
|                 .trim() | ||||
|         val login = | ||||
|             binding.loginView.text | ||||
|                 .toString() | ||||
|                 .trim() | ||||
|         val password = | ||||
|             binding.passwordView.text | ||||
|                 .toString() | ||||
|                 .trim() | ||||
|  | ||||
|         var cancel = false | ||||
|         var focusView: View? = null | ||||
|         val cancelUrl = failInvalidUrl(url) | ||||
|         if (cancelUrl) return | ||||
|         val cancelDetails = failLoginDetails(password, login) | ||||
|         if (cancelDetails) return | ||||
|         showProgress(true) | ||||
|  | ||||
|         if (url.isBaseUrlInvalid()) { | ||||
|             binding.urlView.error = getString(R.string.login_url_problem) | ||||
|             focusView = binding.urlView | ||||
|             cancel = true | ||||
|             inValidCount++ | ||||
|             if (inValidCount == 3) { | ||||
|                 val alertDialog = AlertDialog.Builder(this).create() | ||||
|                 alertDialog.setTitle(getString(R.string.warning_wrong_url)) | ||||
|                 alertDialog.setMessage(getString(R.string.text_wrong_url)) | ||||
|                 alertDialog.setButton( | ||||
|                     AlertDialog.BUTTON_NEUTRAL, | ||||
|                     "OK" | ||||
|                 ) { dialog, _ -> dialog.dismiss() } | ||||
|                 alertDialog.show() | ||||
|                 inValidCount = 0 | ||||
|         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, | ||||
|     ): Boolean { | ||||
|         var lastFocusedView: View? = null | ||||
|         var cancel = false | ||||
|         if (isWithLogin) { | ||||
|             if (TextUtils.isEmpty(password)) { | ||||
|                 binding.passwordView.error = getString(R.string.error_invalid_password) | ||||
|                 focusView = binding.passwordView | ||||
|                 lastFocusedView = binding.passwordView | ||||
|                 cancel = true | ||||
|             } | ||||
|  | ||||
|             if (TextUtils.isEmpty(login)) { | ||||
|                 binding.loginView.error = getString(R.string.error_field_required) | ||||
|                 focusView = binding.loginView | ||||
|                 lastFocusedView = binding.loginView | ||||
|                 cancel = true | ||||
|             } | ||||
|         } | ||||
|         maybeCancelAndFocusView(cancel, lastFocusedView) | ||||
|         return cancel | ||||
|     } | ||||
|  | ||||
|     private fun failInvalidUrl(url: String): Boolean { | ||||
|         val focusView = binding.urlView | ||||
|         var cancel = false | ||||
|         if (url.isBaseUrlInvalid()) { | ||||
|             cancel = true | ||||
|             binding.urlView.error = getString(R.string.login_url_problem) | ||||
|             inValidCount++ | ||||
|             if (inValidCount == MAX_INVALID_LOGIN_BEFORE_ALERT_DISPLAYED) { | ||||
|                 val alertDialog = AlertDialog.Builder(this).create() | ||||
|                 alertDialog.setTitle(getString(R.string.warning_wrong_url)) | ||||
|                 alertDialog.setMessage(getString(R.string.text_wrong_url)) | ||||
|                 alertDialog.setButton( | ||||
|                     AlertDialog.BUTTON_NEUTRAL, | ||||
|                     "OK", | ||||
|                 ) { dialog, _ -> dialog.dismiss() } | ||||
|                 alertDialog.show() | ||||
|                 inValidCount = 0 | ||||
|             } | ||||
|         } | ||||
|         maybeCancelAndFocusView(cancel, focusView) | ||||
|         return cancel | ||||
|     } | ||||
|  | ||||
|     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) | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -183,26 +254,28 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|             .animate() | ||||
|             .setDuration(shortAnimTime.toLong()) | ||||
|             .alpha( | ||||
|                 if (show) 0F else 1F | ||||
|             ).setListener(object : AnimatorListenerAdapter() { | ||||
|             override fun onAnimationEnd(animation: Animator) { | ||||
|                 binding.loginForm.visibility = if (show) View.GONE else View.VISIBLE | ||||
|             } | ||||
|         } | ||||
|         ) | ||||
|                 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 | ||||
|         binding.loginProgress | ||||
|             .animate() | ||||
|             .setDuration(shortAnimTime.toLong()) | ||||
|             .alpha( | ||||
|                 if (show) 1F else 0F | ||||
|             ).setListener(object : AnimatorListenerAdapter() { | ||||
|             override fun onAnimationEnd(animation: Animator) { | ||||
|                 binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||
|             } | ||||
|         } | ||||
|         ) | ||||
|                 if (show) 1F else 0F, | ||||
|             ).setListener( | ||||
|                 object : AnimatorListenerAdapter() { | ||||
|                     override fun onAnimationEnd(animation: Animator) { | ||||
|                         binding.loginProgress.visibility = if (show) View.VISIBLE else View.GONE | ||||
|                     } | ||||
|                 }, | ||||
|             ) | ||||
|     } | ||||
|  | ||||
|     override fun onCreateOptionsMenu(menu: Menu): Boolean { | ||||
| @@ -212,13 +285,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.BUG_URL)) | ||||
|                 startActivity(browserIntent) | ||||
|                 return true | ||||
|             } | ||||
|  | ||||
|             R.id.about -> { | ||||
|                 LibsBuilder() | ||||
|                     .withAboutIconShown(true) | ||||
|                     .withAboutVersionShown(true) | ||||
|                     .withAboutSpecial2("Bug reports") | ||||
|                     .withAboutSpecial2Description(AppSettingsService.BUG_URL) | ||||
|                     .withAboutSpecial1("Project Page") | ||||
|                     .withAboutSpecial1Description(AppSettingsService.SOURCE_URL) | ||||
|                     .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,79 +3,142 @@ 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.viewmodel.AppViewModel | ||||
| import bou.amine.apps.readerforselfossv2.android.testing.TestingHelper | ||||
| import bou.amine.apps.readerforselfossv2.dao.DriverFactory | ||||
| import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | ||||
| import bou.amine.apps.readerforselfossv2.di.networkModule | ||||
| 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.github.ln_12.library.ConnectivityStatus | ||||
| import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader | ||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader | ||||
| import bou.amine.apps.readerforselfossv2.service.ConnectivityService | ||||
| import io.github.aakira.napier.DebugAntilog | ||||
| import io.github.aakira.napier.Napier | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.kodein.di.* | ||||
|  | ||||
| class MyApp : MultiDexApplication(), DIAware { | ||||
| 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<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } | ||||
|         bind<AppViewModel>() with singleton { AppViewModel(repository = instance()) } | ||||
|         bind<ConnectivityService>() with singleton { ConnectivityService() } | ||||
|         bind<Repository>() with | ||||
|             singleton { | ||||
|                 Repository( | ||||
|                     instance(), | ||||
|                     instance(), | ||||
|                     instance(), | ||||
|                     instance(), | ||||
|                 ) | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     private val repository: Repository by instance() | ||||
|     private val viewModel: AppViewModel by instance() | ||||
|     private val connectivityStatus: ConnectivityStatus by instance() | ||||
|     private val driverFactory: DriverFactory by instance() | ||||
|     private val connectivityService: ConnectivityService by instance() | ||||
|  | ||||
|     override fun onCreate() { | ||||
|         super.onCreate() | ||||
|         Napier.base(DebugAntilog()) | ||||
|  | ||||
|         initDrawerImageLoader() | ||||
|         if (!ACRA.isACRASenderServiceProcess()) { | ||||
|             tryToHandleBug() | ||||
|  | ||||
|         tryToHandleBug() | ||||
|             handleNotificationChannels() | ||||
|  | ||||
|         handleNotificationChannels() | ||||
|             ProcessLifecycleOwner.get().lifecycle.addObserver( | ||||
|                 AppLifeCycleObserver( | ||||
|                     connectivityService, | ||||
|                 ), | ||||
|             ) | ||||
|  | ||||
|         ProcessLifecycleOwner.get().lifecycle.addObserver(AppLifeCycleObserver(connectivityStatus, repository)) | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                 connectivityService.networkAvailableProvider.collect { networkAvailable -> | ||||
|                     val toastMessage = | ||||
|                         if (networkAvailable) { | ||||
|                             repository.handleDBActions() | ||||
|                             R.string.network_connectivity_retrieved | ||||
|                         } else { | ||||
|                             R.string.network_connectivity_lost | ||||
|                         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             viewModel.networkAvailableProvider.collect { networkAvailable -> | ||||
|                 val toastMessage = if (networkAvailable) { | ||||
|                     repository.handleDBActions() | ||||
|                     R.string.network_connectivity_retrieved | ||||
|                 } else { | ||||
|                     R.string.network_connectivity_lost | ||||
|                     Toast | ||||
|                         .makeText( | ||||
|                             applicationContext, | ||||
|                             toastMessage, | ||||
|                             Toast.LENGTH_SHORT, | ||||
|                         ).show() | ||||
|                 } | ||||
|  | ||||
|                 Toast.makeText( | ||||
|                     applicationContext, | ||||
|                     toastMessage, | ||||
|                     Toast.LENGTH_SHORT | ||||
|                 ).show() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         repository.migrate(driverFactory) | ||||
|     } | ||||
|  | ||||
|     override fun attachBaseContext(base: Context?) { | ||||
|         super.attachBaseContext(base) | ||||
|  | ||||
|         initAcra { | ||||
|             sendReportsInDevMode = false | ||||
|             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() { | ||||
| @@ -84,62 +147,49 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|  | ||||
|             val name = getString(R.string.notification_channel_sync) | ||||
|             val importance = NotificationManager.IMPORTANCE_LOW | ||||
|             val mChannel = NotificationChannel(AppSettingsService.syncChannelId, name, importance) | ||||
|             val mChannel = NotificationChannel(AppSettingsService.SYNC_CHANNEL_ID, name, importance) | ||||
|  | ||||
|             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.NEW_ITEMS_CHANNEL, | ||||
|                     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 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 connectivityService: ConnectivityService, | ||||
|     ) : DefaultLifecycleObserver { | ||||
|         override fun onResume(owner: LifecycleOwner) { | ||||
|             super.onResume(owner) | ||||
|             repository.connectionMonitored = true | ||||
|             connectivityStatus.start() | ||||
|             connectivityService.start() | ||||
|         } | ||||
|  | ||||
|         override fun onPause(owner: LifecycleOwner) { | ||||
|             repository.connectionMonitored = false | ||||
|             connectivityStatus.stop() | ||||
|             connectivityService.stop() | ||||
|             super.onPause(owner) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -22,34 +22,22 @@ import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
| class ReaderActivity : | ||||
|     AppCompatActivity(), | ||||
|     DIAware { | ||||
|     private var currentItem: Int = 0 | ||||
|  | ||||
|     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) | ||||
|         } else { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.RED) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun canFavorite() { | ||||
|         showMenuItem(true) | ||||
|     } | ||||
|  | ||||
|     private fun canRemoveFromFavorite() { | ||||
|         showMenuItem(false) | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:SwallowedException") | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityReaderBinding.inflate(layoutInflater) | ||||
| @@ -61,16 +49,29 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|         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) | ||||
|  | ||||
|         readItem(allItems[currentItem]) | ||||
|         readItem() | ||||
|  | ||||
|         binding.pager.adapter = ScreenSlidePagerAdapter(this) | ||||
|         binding.pager.setCurrentItem(currentItem, false) | ||||
|  | ||||
|         binding.pager.registerOnPageChangeCallback( | ||||
|             object : ViewPager2.OnPageChangeCallback() { | ||||
|                 override fun onPageSelected(position: Int) { | ||||
|                     super.onPageSelected(position) | ||||
|                     currentItem = position | ||||
|                     updateStarIcon() | ||||
|                     readItem() | ||||
|                 } | ||||
|             }, | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     override fun onResume() { | ||||
| @@ -79,48 +80,56 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|         binding.indicator.setViewPager(binding.pager) | ||||
|     } | ||||
|  | ||||
|     private fun readItem(item: SelfossModel.Item) { | ||||
|         if (appSettingsService.isMarkOnScrollEnabled()) { | ||||
|     private fun readItem() { | ||||
|         val item = allItems.getOrNull(currentItem) | ||||
|         if (appSettingsService.isMarkOnScrollEnabled() && !appSettingsService.getPublicAccess() && item != null) { | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 repository.markAsRead(item) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun updateStarIcon() { | ||||
|         val isStarred = allItems.getOrNull(currentItem)?.starred ?: false | ||||
|         toolbarMenu.findItem(R.id.star)?.icon?.setTint(if (isStarred) Color.RED else Color.WHITE) | ||||
|     } | ||||
|  | ||||
|     override fun onSaveInstanceState(oldInstanceState: Bundle) { | ||||
|         super.onSaveInstanceState(oldInstanceState) | ||||
|         oldInstanceState.clear() | ||||
|     } | ||||
|  | ||||
|     private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : | ||||
|         FragmentStateAdapter(fa) { | ||||
|  | ||||
|     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 { | ||||
|         return when (keyCode) { | ||||
|     override fun onKeyDown( | ||||
|         keyCode: Int, | ||||
|         event: KeyEvent?, | ||||
|     ): Boolean = | ||||
|         when (keyCode) { | ||||
|             KeyEvent.KEYCODE_VOLUME_DOWN -> { | ||||
|                 val currentFragment = | ||||
|                     supportFragmentManager.findFragmentByTag("f" + binding.pager.currentItem) as ArticleFragment | ||||
|                 currentFragment.scrollDown() | ||||
|                 currentFragment.volumeButtonScrollDown() | ||||
|                 true | ||||
|             } | ||||
|  | ||||
|             KeyEvent.KEYCODE_VOLUME_UP -> { | ||||
|                 val currentFragment = | ||||
|                     supportFragmentManager.findFragmentByTag("f" + binding.pager.currentItem) as ArticleFragment | ||||
|                 currentFragment.scrollUp() | ||||
|                 currentFragment.volumeButtonScrollUp() | ||||
|                 true | ||||
|             } | ||||
|  | ||||
|             else -> { | ||||
|                 super.onKeyDown(keyCode, event) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun alignmentMenu() { | ||||
|         val showJustify = appSettingsService.getActiveAllignment() == AppSettingsService.ALIGN_LEFT | ||||
| @@ -129,93 +138,58 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|     } | ||||
|  | ||||
|     override fun onCreateOptionsMenu(menu: Menu): Boolean { | ||||
|         val inflater = menuInflater | ||||
|         inflater.inflate(R.menu.reader_menu, menu) | ||||
|         menuInflater.inflate(R.menu.reader_menu, menu) | ||||
|         toolbarMenu = menu | ||||
|  | ||||
|         if (allItems.isNotEmpty() && allItems[currentItem].starred) { | ||||
|             canRemoveFromFavorite() | ||||
|         } else { | ||||
|             canFavorite() | ||||
|         } | ||||
|         alignmentMenu() | ||||
|  | ||||
|         binding.pager.registerOnPageChangeCallback( | ||||
|             object : ViewPager2.OnPageChangeCallback() { | ||||
|  | ||||
|                 override fun onPageSelected(position: Int) { | ||||
|                     super.onPageSelected(position) | ||||
|  | ||||
|                     if (allItems[position].starred) { | ||||
|                         canRemoveFromFavorite() | ||||
|                     } else { | ||||
|                         canFavorite() | ||||
|                     } | ||||
|                     readItem(allItems[position]) | ||||
|                 } | ||||
|             } | ||||
|         ) | ||||
|         if (appSettingsService.getPublicAccess()) { | ||||
|             menu.removeItem(R.id.star) | ||||
|         } else { | ||||
|             updateStarIcon() | ||||
|         } | ||||
|  | ||||
|         return true | ||||
|     } | ||||
|  | ||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||
|         fun afterSave() { | ||||
|             allItems[binding.pager.currentItem] = | ||||
|                 allItems[binding.pager.currentItem].toggleStar() | ||||
|             canRemoveFromFavorite() | ||||
|         } | ||||
|  | ||||
|         fun afterUnsave() { | ||||
|             allItems[binding.pager.currentItem] = allItems[binding.pager.currentItem].toggleStar() | ||||
|             canFavorite() | ||||
|         } | ||||
|  | ||||
|         when (item.itemId) { | ||||
|             android.R.id.home -> { | ||||
|                 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() | ||||
|                 } | ||||
|             } | ||||
|             R.id.align_left -> { | ||||
|                 switchAlignmentSetting(AppSettingsService.ALIGN_LEFT) | ||||
|                 refreshFragment() | ||||
|             } | ||||
|             R.id.align_justify -> { | ||||
|                 switchAlignmentSetting(AppSettingsService.JUSTIFY) | ||||
|                 refreshFragment() | ||||
|             } | ||||
|             android.R.id.home -> onBackPressedDispatcher.onBackPressed() | ||||
|             R.id.star -> toggleFavorite() | ||||
|             R.id.align_left -> switchAlignmentSetting(AppSettingsService.ALIGN_LEFT) | ||||
|             R.id.align_justify -> switchAlignmentSetting(AppSettingsService.JUSTIFY) | ||||
|         } | ||||
|         return super.onOptionsItemSelected(item) | ||||
|     } | ||||
|  | ||||
|     private fun switchAlignmentSetting(allignment: Int) { | ||||
|         appSettingsService.changeAllignment(allignment) | ||||
|     private fun toggleFavorite() { | ||||
|         val item = allItems.getOrNull(currentItem) ?: return | ||||
|  | ||||
|         val starred = item.starred | ||||
|  | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             if (starred) { | ||||
|                 repository.unstarr(item) | ||||
|             } else { | ||||
|                 repository.starr(item) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         item.toggleStar() | ||||
|         updateStarIcon() | ||||
|     } | ||||
|  | ||||
|     private fun switchAlignmentSetting(alignment: Int) { | ||||
|         appSettingsService.changeAllignment(alignment) | ||||
|         alignmentMenu() | ||||
|     } | ||||
|  | ||||
|     private fun refreshFragment() { | ||||
|         finish() | ||||
|         overridePendingTransition(0, 0) | ||||
|         startActivity(intent) | ||||
|         overridePendingTransition(0, 0) | ||||
|     } | ||||
|         val fragmentManager = supportFragmentManager | ||||
|         val fragments = fragmentManager.fragments | ||||
|  | ||||
|     companion object { | ||||
|         var allItems: ArrayList<SelfossModel.Item> = ArrayList() | ||||
|         for (fragment in fragments) { | ||||
|             if (fragment is ArticleFragment) { | ||||
|                 fragment.refreshAlignment() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -8,6 +8,7 @@ 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.testing.CountingIdlingResourceSingleton | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| @@ -17,12 +18,13 @@ import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
| class SourcesActivity : | ||||
|     AppCompatActivity(), | ||||
|     DIAware { | ||||
|     private lateinit var binding: ActivitySourcesBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|     private val repository: Repository by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         binding = ActivitySourcesBinding.inflate(layoutInflater) | ||||
| @@ -37,7 +39,8 @@ class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         binding.fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||
|         binding.fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||
|         binding.fab.backgroundTintList = | ||||
|             ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||
|     } | ||||
|  | ||||
|     override fun onStop() { | ||||
| @@ -49,38 +52,36 @@ 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 | ||||
|                 ).show() | ||||
|                 Toast | ||||
|                     .makeText( | ||||
|                         this@SourcesActivity, | ||||
|                         R.string.cant_get_sources, | ||||
|                         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,200 @@ | ||||
| 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.model.NetworkUnavailableException | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| 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() | ||||
|  | ||||
|     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() | ||||
|         handleSpoutsSpinner() | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:SwallowedException") | ||||
|     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 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,18 +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.utils.* | ||||
| 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 | ||||
| @@ -31,11 +30,10 @@ import org.kodein.di.instance | ||||
|  | ||||
| class ItemCardAdapter( | ||||
|     override val app: Activity, | ||||
|     override var items: ArrayList<SelfossModel.Item>, | ||||
|     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() | ||||
|  | ||||
| @@ -43,23 +41,71 @@ 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(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 | ||||
| @@ -72,73 +118,18 @@ class ItemCardAdapter( | ||||
|                 binding.itemImage.setImageDrawable(null) | ||||
|             } else { | ||||
|                 binding.itemImage.visibility = View.VISIBLE | ||||
|                 c.bitmapCenterCrop(itm.getThumbnail(repository.baseUrl), binding.itemImage) | ||||
|                 c.bitmapCenterCrop(itm.getThumbnail(repository.baseUrl), binding.itemImage, appSettingsService) | ||||
|             } | ||||
|  | ||||
|             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, appSettingsService) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun getItemCount(): Int { | ||||
|         return items.size | ||||
|     } | ||||
|  | ||||
|     inner class ViewHolder(val binding: CardItemBinding) : RecyclerView.ViewHolder(binding.root) { | ||||
|         init { | ||||
|             handleClickListeners() | ||||
|             handleLinkOpening() | ||||
|         } | ||||
|  | ||||
|         private fun handleClickListeners() { | ||||
|  | ||||
|             binding.favButton.setOnClickListener { | ||||
|                 val item = items[bindingAdapterPosition] | ||||
|                 if (item.starred) { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.unstarr(item) | ||||
|                     } | ||||
|                     item.starred = false | ||||
|                     binding.favButton.isSelected = false | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.starr(item) | ||||
|                     } | ||||
|                     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 handleLinkOpening() { | ||||
|             binding.root.setOnClickListener { | ||||
|                 c.openItemUrl( | ||||
|                     items, | ||||
|                     bindingAdapterPosition, | ||||
|                     items[bindingAdapterPosition].getLinkDecoded(), | ||||
|                     appSettingsService.isArticleViewerEnabled(), | ||||
|                     app | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     inner class ViewHolder( | ||||
|         val binding: CardItemBinding, | ||||
|     ) : RecyclerView.ViewHolder(binding.root) | ||||
| } | ||||
|   | ||||
| @@ -1,97 +1,79 @@ | ||||
| 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.utils.LinkOnTouchListener | ||||
| 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>, | ||||
|     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 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(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, appSettingsService) | ||||
|                 } | ||||
|             } else { | ||||
|                 c.bitmapCenterCrop(itm.getThumbnail(repository.baseUrl), binding.itemImage) | ||||
|                 c.circularDrawable(itm.getThumbnail(repository.baseUrl), binding.itemImage, appSettingsService) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun getItemCount(): Int = items.size | ||||
|  | ||||
|     inner class ViewHolder(val binding: ListItemBinding) : RecyclerView.ViewHolder(binding.root) { | ||||
|  | ||||
|         init { | ||||
|             handleLinkOpening() | ||||
|         } | ||||
|  | ||||
|         private fun handleLinkOpening() { | ||||
|             binding.root.setOnClickListener { | ||||
|                 c.openItemUrl( | ||||
|                     items, | ||||
|                     bindingAdapterPosition, | ||||
|                     items[bindingAdapterPosition].getLinkDecoded(), | ||||
|                     appSettingsService.isArticleViewerEnabled(), | ||||
|                     app | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     inner class ViewHolder( | ||||
|         val binding: ListItemBinding, | ||||
|     ) : RecyclerView.ViewHolder(binding.root) | ||||
| } | ||||
|   | ||||
| @@ -1,10 +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.utils.openItemUrl | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -15,31 +18,38 @@ import kotlinx.coroutines.Dispatchers | ||||
| 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 class ItemsAdapter<VH : RecyclerView.ViewHolder?> : | ||||
|     RecyclerView.Adapter<VH>(), | ||||
|     DIAware { | ||||
|     abstract val items: ArrayList<SelfossModel.Item> | ||||
|     abstract val repository: Repository | ||||
|     abstract val binding: ViewBinding | ||||
|     abstract val appSettingsService: AppSettingsService | ||||
|     abstract val app: Activity | ||||
|     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(position: Int) { | ||||
|         val s = Snackbar | ||||
|             .make( | ||||
|                 app.findViewById(R.id.coordLayout), | ||||
|                 R.string.marked_as_read, | ||||
|                 Snackbar.LENGTH_LONG | ||||
|             ) | ||||
|             .setAction(R.string.undo_string) { | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     unreadItemAtIndex(position, false) | ||||
|     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, | ||||
|                 ).setAction(R.string.undo_string) { | ||||
|                     unreadItemAtIndex(item, position, false) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|         val view = s.view | ||||
|         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||
| @@ -47,16 +57,19 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         s.show() | ||||
|     } | ||||
|  | ||||
|     private fun markSnackbar(position: Int) { | ||||
|         val s = Snackbar | ||||
|             .make( | ||||
|                 app.findViewById(R.id.coordLayout), | ||||
|                 R.string.marked_as_unread, | ||||
|                 Snackbar.LENGTH_LONG | ||||
|             ) | ||||
|             .setAction(R.string.undo_string) { | ||||
|                 readItemAtIndex(position) | ||||
|             } | ||||
|     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, | ||||
|                 ).setAction(R.string.undo_string) { | ||||
|                     readItemAtIndex(item, position, false) | ||||
|                 } | ||||
|  | ||||
|         val view = s.view | ||||
|         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||
| @@ -64,54 +77,79 @@ 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(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.appcompat.app.AlertDialog | ||||
| 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.service.AppSettingsService | ||||
| 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,65 +28,96 @@ import org.kodein.di.instance | ||||
|  | ||||
| class SourcesListAdapter( | ||||
|     private val app: Activity, | ||||
|     private val items: ArrayList<SelfossModel.Source> | ||||
| ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>(), DIAware { | ||||
|     private val c: Context = app.baseContext | ||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||
|     private lateinit var binding: SourceListItemBinding | ||||
|  | ||||
|     private val items: ArrayList<SelfossModel.SourceDetail>, | ||||
| ) : RecyclerView.Adapter<SourcesListAdapter.ViewHolder>(), | ||||
|     DIAware { | ||||
|     override val di: DI by closestDI(app) | ||||
|     private val repository : Repository by instance() | ||||
|  | ||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||
|         binding = SourceListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|         return ViewHolder(binding.root) | ||||
|     override fun onCreateViewHolder( | ||||
|         parent: ViewGroup, | ||||
|         viewType: Int, | ||||
|     ): ViewHolder { | ||||
|         val binding = SourceListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
|         return ViewHolder(binding) | ||||
|     } | ||||
|  | ||||
|     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 drawable = | ||||
|                 TextDrawable | ||||
|                     .builder() | ||||
|                     .round() | ||||
|                     .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) | ||||
|             binding.itemImage.setImageDrawable(drawable) | ||||
|         } else { | ||||
|             c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.itemImage) | ||||
|         } | ||||
|  | ||||
|         binding.sourceTitle.text = itm.title.getHtmlDecoded() | ||||
|     override fun onBindViewHolder( | ||||
|         holder: ViewHolder, | ||||
|         position: Int, | ||||
|     ) { | ||||
|         holder.bind(items[position], position) | ||||
|     } | ||||
|  | ||||
|     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) { | ||||
|     inner class ViewHolder( | ||||
|         val binding: SourceListItemBinding, | ||||
|     ) : RecyclerView.ViewHolder(binding.root) { | ||||
|         private val context: Context = app.applicationContext | ||||
|         private val repository: Repository by instance() | ||||
|         private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|         init { | ||||
|             handleClickListeners() | ||||
|         fun bind( | ||||
|             source: SelfossModel.SourceDetail, | ||||
|             position: Int, | ||||
|         ) { | ||||
|             binding.apply { | ||||
|                 sourceTitle.text = source.title.getHtmlDecoded() | ||||
|                 if (source.getIcon(repository.baseUrl).isEmpty()) { | ||||
|                     itemImage.setBackgroundAndText(source.title.getHtmlDecoded()) | ||||
|                 } else { | ||||
|                     context.circularDrawable(source.getIcon(repository.baseUrl), itemImage, appSettingsService) | ||||
|                 } | ||||
|  | ||||
|                 errorText.apply { | ||||
|                     visibility = if (!source.error.isNullOrBlank()) View.VISIBLE else View.GONE | ||||
|                     text = source.error | ||||
|                 } | ||||
|  | ||||
|                 deleteBtn.setOnClickListener { showDeleteConfirmationDialog(source, position) } | ||||
|  | ||||
|                 root.setOnClickListener { | ||||
|                     repository.setSelectedSource(source) | ||||
|                     app.startActivity(Intent(app, UpsertSourceActivity::class.java)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         private fun handleClickListeners() { | ||||
|         private fun showDeleteConfirmationDialog( | ||||
|             source: SelfossModel.SourceDetail, | ||||
|             position: Int, | ||||
|         ) { | ||||
|             AlertDialog | ||||
|                 .Builder(app) | ||||
|                 .setTitle(app.getString(R.string.confirm_delete_title)) | ||||
|                 .setMessage(app.getString(R.string.confirm_delete_message, source.title)) | ||||
|                 .setPositiveButton(android.R.string.ok) { _, _ -> deleteSource(source, position) } | ||||
|                 .setNegativeButton(android.R.string.cancel, null) | ||||
|                 .show() | ||||
|         } | ||||
|  | ||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||
|  | ||||
|             deleteBtn.setOnClickListener { | ||||
|                 val (id) = items[adapterPosition] | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     val successfullyDeletedSource = repository.deleteSource(id) | ||||
|         private fun deleteSource( | ||||
|             source: SelfossModel.SourceDetail, | ||||
|             position: Int, | ||||
|         ) { | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 val successfullyDeletedSource = repository.deleteSource(source.id, source.title) | ||||
|                 launch(Dispatchers.Main) { | ||||
|                     if (successfullyDeletedSource) { | ||||
|                         items.removeAt(adapterPosition) | ||||
|                         notifyItemRemoved(adapterPosition) | ||||
|                         notifyItemRangeChanged(adapterPosition, itemCount) | ||||
|                         items.removeAt(position) | ||||
|                         notifyItemRemoved(position) | ||||
|                         notifyItemRangeChanged(position, itemCount) | ||||
|                     } else { | ||||
|                         Toast.makeText( | ||||
|                             app, | ||||
|                             R.string.can_delete_source, | ||||
|                             Toast.LENGTH_SHORT | ||||
|                         ).show() | ||||
|                         Toast | ||||
|                             .makeText( | ||||
|                                 app, | ||||
|                                 R.string.can_delete_source, | ||||
|                                 Toast.LENGTH_SHORT, | ||||
|                             ).show() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -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> | ||||
| } | ||||
| @@ -0,0 +1,120 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.background | ||||
|  | ||||
| import android.app.NotificationManager | ||||
| import android.app.PendingIntent | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.os.Build | ||||
| import androidx.core.app.NotificationCompat | ||||
| import androidx.core.app.NotificationCompat.PRIORITY_DEFAULT | ||||
| import androidx.core.app.NotificationCompat.PRIORITY_LOW | ||||
| import androidx.work.Worker | ||||
| import androidx.work.WorkerParameters | ||||
| import bou.amine.apps.readerforselfossv2.android.MainActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.MyApp | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.model.preloadImages | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.network.isNetworkAccessible | ||||
| 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.instance | ||||
| import java.util.Timer | ||||
| import kotlin.concurrent.schedule | ||||
|  | ||||
| private const val NOTIFICATION_DELAY = 4000L | ||||
|  | ||||
| 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 | ||||
|  | ||||
|                 val notification = | ||||
|                     NotificationCompat | ||||
|                         .Builder(applicationContext, AppSettingsService.SYNC_CHANNEL_ID) | ||||
|                         .setContentTitle(context.getString(R.string.loading_notification_title)) | ||||
|                         .setContentText(context.getString(R.string.loading_notification_text)) | ||||
|                         .setOngoing(true) | ||||
|                         .setPriority(PRIORITY_LOW) | ||||
|                         .setChannelId(AppSettingsService.SYNC_CHANNEL_ID) | ||||
|                         .setSmallIcon(R.drawable.ic_stat_cloud_download_black_24dp) | ||||
|  | ||||
|                 notificationManager.notify(1, notification.build()) | ||||
|  | ||||
|                 repository.handleDBActions() | ||||
|  | ||||
|                 val apiItems = repository.tryToCacheItemsAndGetNewOnes() | ||||
|                 if (appSettingsService.isNotifyNewItemsEnabled()) { | ||||
|                     launch { | ||||
|                         handleNewItemsNotification(apiItems, notificationManager) | ||||
|                     } | ||||
|                 } | ||||
|                 apiItems.map { it.preloadImages(context, appSettingsService) } | ||||
|             } | ||||
|         } | ||||
|         return Result.success() | ||||
|     } | ||||
|  | ||||
|     private fun handleNewItemsNotification( | ||||
|         newItems: List<SelfossModel.Item>?, | ||||
|         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 { | ||||
|                         flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK | ||||
|                     } | ||||
|                 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 newItemsNotification = | ||||
|                     NotificationCompat | ||||
|                         .Builder( | ||||
|                             applicationContext, | ||||
|                             AppSettingsService.NEW_ITEMS_CHANNEL, | ||||
|                         ).setContentTitle(context.getString(R.string.new_items_notification_title)) | ||||
|                         .setContentText( | ||||
|                             context.getString( | ||||
|                                 R.string.new_items_notification_text, | ||||
|                                 newSize, | ||||
|                             ), | ||||
|                         ).setPriority(PRIORITY_DEFAULT) | ||||
|                         .setChannelId(AppSettingsService.NEW_ITEMS_CHANNEL) | ||||
|                         .setContentIntent(pendingIntent) | ||||
|                         .setAutoCancel(true) | ||||
|                         .setSmallIcon(R.drawable.ic_tab_fiber_new_black_24dp) | ||||
|  | ||||
|                 Timer("", false).schedule(NOTIFICATION_DELAY) { | ||||
|                     notificationManager.notify(2, newItemsNotification.build()) | ||||
|                 } | ||||
|             } | ||||
|             Timer("", false).schedule(NOTIFICATION_DELAY) { | ||||
|                 notificationManager.cancel(1) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,111 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.background | ||||
|  | ||||
| import android.app.NotificationManager | ||||
| import android.app.PendingIntent | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.os.Build | ||||
| import androidx.core.app.NotificationCompat | ||||
| import androidx.core.app.NotificationCompat.PRIORITY_DEFAULT | ||||
| import androidx.core.app.NotificationCompat.PRIORITY_LOW | ||||
| import androidx.work.Worker | ||||
| import androidx.work.WorkerParameters | ||||
| import bou.amine.apps.readerforselfossv2.android.MainActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.MyApp | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.model.preloadImages | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.network.isNetworkAccessible | ||||
| 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.instance | ||||
| import java.util.* | ||||
| import kotlin.concurrent.schedule | ||||
|  | ||||
| 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 | ||||
|  | ||||
|             val notification = | ||||
|                 NotificationCompat.Builder(applicationContext, AppSettingsService.syncChannelId) | ||||
|                     .setContentTitle(context.getString(R.string.loading_notification_title)) | ||||
|                     .setContentText(context.getString(R.string.loading_notification_text)) | ||||
|                     .setOngoing(true) | ||||
|                     .setPriority(PRIORITY_LOW) | ||||
|                     .setChannelId(AppSettingsService.syncChannelId) | ||||
|                     .setSmallIcon(R.drawable.ic_stat_cloud_download_black_24dp) | ||||
|  | ||||
|             notificationManager.notify(1, notification.build()) | ||||
|  | ||||
|             repository.handleDBActions() | ||||
|  | ||||
|             if (appSettingsService.isNotifyNewItemsEnabled()) { | ||||
|                 launch { | ||||
|                     handleNewItemsNotification(repository.tryToCacheItemsAndGetNewOnes(), notificationManager) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     return Result.success() | ||||
| } | ||||
|  | ||||
|     private fun handleNewItemsNotification( | ||||
|         newItems: List<SelfossModel.Item>?, | ||||
|         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 { | ||||
|                         flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK | ||||
|                     } | ||||
|                     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 newItemsNotification = | ||||
|                         NotificationCompat.Builder(applicationContext, AppSettingsService.newItemsChannelId) | ||||
|                             .setContentTitle(context.getString(R.string.new_items_notification_title)) | ||||
|                             .setContentText( | ||||
|                                 context.getString( | ||||
|                                     R.string.new_items_notification_text, | ||||
|                                     newSize | ||||
|                                 ) | ||||
|                             ) | ||||
|                             .setPriority(PRIORITY_DEFAULT) | ||||
|                             .setChannelId(AppSettingsService.newItemsChannelId) | ||||
|                             .setContentIntent(pendingIntent) | ||||
|                             .setAutoCancel(true) | ||||
|                             .setSmallIcon(R.drawable.ic_tab_fiber_new_black_24dp) | ||||
|  | ||||
|                     Timer("", false).schedule(4000) { | ||||
|                         notificationManager.notify(2, newItemsNotification.build()) | ||||
|                     } | ||||
|                 } | ||||
|                 apiItems.map { it.preloadImages(context) } | ||||
|             Timer("", false).schedule(4000) { | ||||
|                 notificationManager.cancel(1) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,85 +1,101 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.fragments | ||||
|  | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.content.res.ColorStateList | ||||
| 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.util.TypedValue | ||||
| import android.view.* | ||||
| import android.util.TypedValue.DATA_NULL_UNDEFINED | ||||
| import android.view.GestureDetector | ||||
| import android.view.InflateException | ||||
| import android.view.LayoutInflater | ||||
| 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.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.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.bottombar.addHomeMadeActionItem | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.getColorFromAttr | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapFitCenter | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getGlideImageForResource | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isUrlValid | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.maybeIfContext | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrlInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openUrlInBrowserAsNewTask | ||||
| 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.service.ConnectivityService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| import bou.amine.apps.readerforselfossv2.utils.getThumbnail | ||||
| import bou.amine.apps.readerforselfossv2.utils.isEmptyOrNullOrNullString | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import com.github.rubensousa.floatingtoolbar.FloatingToolbar | ||||
| import com.google.android.material.floatingactionbutton.FloatingActionButton | ||||
| import com.leinardi.android.speeddial.SpeedDialView | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.acra.ktx.sendSilentlyWithAcra | ||||
| 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" | ||||
| private const val IMAGE_PNG = "image/png" | ||||
| private const val IMAGE_WEBP = "image/webp" | ||||
|  | ||||
| class ArticleFragment : Fragment(), DIAware { | ||||
|     private var fontSize: Int = 16 | ||||
| private const val WHITE_COLOR_HEX = 0xFFFFFF | ||||
|  | ||||
| private const val DEFAULT_FONT_SIZE = 16 | ||||
|  | ||||
| class ArticleFragment : | ||||
|     Fragment(), | ||||
|     DIAware { | ||||
|     private var colorOnSurface: Int = 0 | ||||
|     private var colorSurface: Int = 0 | ||||
|     private var fontSize: Int = DEFAULT_FONT_SIZE | ||||
|     private lateinit var item: SelfossModel.Item | ||||
|     private lateinit var url: String | ||||
|     private var url: String? = null | ||||
|     private lateinit var contentText: String | ||||
|     private lateinit var contentSource: String | ||||
|     private lateinit var contentImage: String | ||||
|     private lateinit var contentTitle: String | ||||
|     private lateinit var allImages : ArrayList<String> | ||||
|     private lateinit var fab: FloatingActionButton | ||||
|     private lateinit var allImages: ArrayList<String> | ||||
|     private lateinit var fab: SpeedDialView | ||||
|     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() | ||||
|     override val di: DI by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|     private val connectivityService: ConnectivityService by instance() | ||||
|  | ||||
|     private var typeface: Typeface? = null | ||||
|     private var resId: Int = 0 | ||||
|     private var font = "" | ||||
|     private var staticBar = false | ||||
|  | ||||
|     private val mercuryApi: MercuryApi by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
| @@ -89,343 +105,366 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         item = pi.toModel() | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:LongMethod") | ||||
|     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) | ||||
|  | ||||
|             try { | ||||
|                 url = item.getLinkDecoded() | ||||
|             } catch (e: Exception) { | ||||
|                 e.sendSilentlyWithAcra() | ||||
|             } | ||||
|  | ||||
|             colorOnSurface = getColorFromAttr(com.google.android.material.R.attr.colorOnSurface) | ||||
|             colorSurface = getColorFromAttr(com.google.android.material.R.attr.colorSurface) | ||||
|  | ||||
|             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(resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||
|  | ||||
|             val floatingToolbar: FloatingToolbar = binding.floatingToolbar | ||||
|             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().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?) { | ||||
|                     } | ||||
|                 } | ||||
|             ) | ||||
|  | ||||
|             if (staticBar) { | ||||
|                 fab.hide() | ||||
|                 floatingToolbar.show() | ||||
|             } | ||||
|             handleFloatingToolbar() | ||||
|  | ||||
|             binding.source.text = contentSource | ||||
|             if (typeface != null) { | ||||
|                 binding.source.typeface = typeface | ||||
|             } | ||||
|  | ||||
|             if (contentText.isEmptyOrNullOrNullString()) { | ||||
|                 getContentFromMercury() | ||||
|             } else { | ||||
|                 binding.titleView.text = contentTitle | ||||
|                 if (typeface != null) { | ||||
|                     binding.titleView.typeface = typeface | ||||
|                 } | ||||
|  | ||||
|                 htmlToWebview() | ||||
|  | ||||
|                 if (!contentImage.isEmptyOrNullOrNullString() && context != null) { | ||||
|                     binding.imageView.visibility = View.VISIBLE | ||||
|                     Glide | ||||
|                         .with(requireContext()) | ||||
|                         .asBitmap() | ||||
|                         .load(contentImage) | ||||
|                         .apply(RequestOptions.fitCenterTransform()) | ||||
|                         .into(binding.imageView) | ||||
|                 } else { | ||||
|                     binding.imageView.visibility = View.GONE | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             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() | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             ) | ||||
|  | ||||
|             handleContent() | ||||
|         } 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() | ||||
|             e.sendSilentlyWithAcraWithName("webview not available") | ||||
|             maybeIfContext { | ||||
|                 AlertDialog | ||||
|                     .Builder(it) | ||||
|                     .setMessage(it.getString(R.string.webview_dialog_issue_message)) | ||||
|                     .setTitle(it.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 | ||||
|     } | ||||
|     private fun handleContent() { | ||||
|         if (contentText.isEmptyOrNullOrNullString()) { | ||||
|             if (connectivityService.isNetworkAvailable() && url.isUrlValid()) { | ||||
|                 getContentFromMercury(url!!) | ||||
|             } | ||||
|         } else { | ||||
|             binding.titleView.text = contentTitle | ||||
|             if (typeface != null) { | ||||
|                 binding.titleView.typeface = typeface | ||||
|             } | ||||
|  | ||||
|     private fun refreshAlignment() { | ||||
|         textAlignment = when (appSettingsService.getActiveAllignment()) { | ||||
|             1 -> "justify" | ||||
|             2 -> "left" | ||||
|             else -> "justify" | ||||
|             htmlToWebview() | ||||
|  | ||||
|             if (!contentImage.isEmptyOrNullOrNullString() && context != null) { | ||||
|                 binding.imageView.visibility = View.VISIBLE | ||||
|                 maybeIfContext { it.bitmapFitCenter(contentImage, binding.imageView, appSettingsService) } | ||||
|             } else { | ||||
|                 binding.imageView.visibility = View.GONE | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun getContentFromMercury() { | ||||
|         if (repository.isNetworkAvailable()) { | ||||
|             binding.progressBar.visibility = View.VISIBLE | ||||
|             val parser = MercuryApi() | ||||
|     private fun handleFloatingToolbar() { | ||||
|         fab = binding.speedDial | ||||
|         fab.mainFabClosedIconColor = colorOnSurface | ||||
|         fab.mainFabOpenedIconColor = colorOnSurface | ||||
|  | ||||
|             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 | ||||
|                         try { | ||||
|                             if (response.body() != null && response.body()!!.content != null && !response.body()!!.content.isNullOrEmpty()) { | ||||
|                                 try { | ||||
|                                     binding.titleView.text = response.body()!!.title | ||||
|                                     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) { | ||||
|                                 } | ||||
|         maybeIfContext { handleFloatingToolbarActionItems(it) } | ||||
|  | ||||
|                                 try { | ||||
|                                     contentText = response.body()!!.content.orEmpty() | ||||
|                                     htmlToWebview() | ||||
|                                 } catch (e: Exception) { | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||
|                                         binding.imageView.visibility = View.VISIBLE | ||||
|                                         try { | ||||
|                                             Glide | ||||
|                                                 .with(requireContext()) | ||||
|                                                 .asBitmap() | ||||
|                                                 .load( | ||||
|                                                     response.body()!!.lead_image_url.orEmpty() | ||||
|                                                 ) | ||||
|                                                 .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) { | ||||
|                                     } | ||||
|                                 } | ||||
|                             } else { | ||||
|                                 try { | ||||
|                                     openInBrowserAfterFailing() | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|                             } | ||||
|                         } catch (e: Exception) { | ||||
|                             if (context != null) { | ||||
|                             } | ||||
|         fab.setOnActionSelectedListener { actionItem -> | ||||
|             when (actionItem.id) { | ||||
|                 R.id.share_action -> requireActivity().shareLink(url, contentTitle) | ||||
|                 R.id.open_action -> requireActivity().openItemUrlInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|                 R.id.unread_action -> | ||||
|                     if (this@ArticleFragment.item.unread) { | ||||
|                         CoroutineScope(Dispatchers.IO).launch { | ||||
|                             repository.markAsRead(this@ArticleFragment.item) | ||||
|                         } | ||||
|                         this@ArticleFragment.item.unread = false | ||||
|                         maybeIfContext { | ||||
|                             Toast | ||||
|                                 .makeText( | ||||
|                                     it, | ||||
|                                     R.string.marked_as_read, | ||||
|                                     Toast.LENGTH_LONG, | ||||
|                                 ).show() | ||||
|                         } | ||||
|                     } else { | ||||
|                         CoroutineScope(Dispatchers.IO).launch { | ||||
|                             repository.unmarkAsRead(this@ArticleFragment.item) | ||||
|                         } | ||||
|                         this@ArticleFragment.item.unread = true | ||||
|                         maybeIfContext { | ||||
|                             Toast | ||||
|                                 .makeText( | ||||
|                                     it, | ||||
|                                     R.string.marked_as_unread, | ||||
|                                     Toast.LENGTH_LONG, | ||||
|                                 ).show() | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     override fun onFailure( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         t: Throwable | ||||
|                     ) = openInBrowserAfterFailing() | ||||
|                 } | ||||
|             ) | ||||
|                 else -> Unit | ||||
|             } | ||||
|             false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleFloatingToolbarActionItems(c: Context) { | ||||
|         fab.addHomeMadeActionItem( | ||||
|             R.id.share_action, | ||||
|             resources.getDrawable(R.drawable.ic_share_white_24dp), | ||||
|             R.string.reader_action_share, | ||||
|             colorOnSurface, | ||||
|             colorSurface, | ||||
|             c, | ||||
|         ) | ||||
|         fab.addHomeMadeActionItem( | ||||
|             R.id.open_action, | ||||
|             resources.getDrawable(R.drawable.ic_open_in_browser_white_24dp), | ||||
|             R.string.reader_action_open, | ||||
|             colorOnSurface, | ||||
|             colorSurface, | ||||
|             c, | ||||
|         ) | ||||
|         fab.addHomeMadeActionItem( | ||||
|             R.id.unread_action, | ||||
|             resources.getDrawable(R.drawable.ic_baseline_white_eye_24dp), | ||||
|             R.string.unmark, | ||||
|             colorOnSurface, | ||||
|             colorSurface, | ||||
|             c, | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     fun refreshAlignment() { | ||||
|         textAlignment = | ||||
|             when (appSettingsService.getActiveAllignment()) { | ||||
|                 1 -> "justify" | ||||
|                 2 -> "left" | ||||
|                 else -> "justify" | ||||
|             } | ||||
|  | ||||
|         htmlToWebview() | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:SwallowedException") | ||||
|     private fun getContentFromMercury(url: String) { | ||||
|         binding.progressBar.visibility = View.VISIBLE | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             try { | ||||
|                 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 | ||||
|             } | ||||
|             URL(data.url) | ||||
|             url = data.url!! | ||||
|  | ||||
|             contentText = data.content.orEmpty() | ||||
|             htmlToWebview() | ||||
|  | ||||
|             handleLeadImage(data.lead_image_url) | ||||
|  | ||||
|             binding.nestedScrollView.scrollTo(0, 0) | ||||
|             binding.progressBar.visibility = View.GONE | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleLeadImage(leadImageUrl: String?) { | ||||
|         if (!leadImageUrl.isNullOrEmpty()) { | ||||
|             maybeIfContext { | ||||
|                 binding.imageView.visibility = View.VISIBLE | ||||
|                 it.bitmapFitCenter(leadImageUrl, binding.imageView, appSettingsService) | ||||
|             } | ||||
|         } else { | ||||
|             binding.imageView.visibility = View.GONE | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleImageLoading() { | ||||
|         binding.webcontent.webViewClient = | ||||
|             object : WebViewClient() { | ||||
|                 @Deprecated("Deprecated in Java") | ||||
|                 override fun shouldOverrideUrlLoading( | ||||
|                     view: WebView?, | ||||
|                     url: String, | ||||
|                 ): Boolean = | ||||
|                     if (url.isUrlValid() && | ||||
|                         binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE | ||||
|                     ) { | ||||
|                         maybeIfContext { it.openUrlInBrowserAsNewTask(url) } | ||||
|                         true | ||||
|                     } else { | ||||
|                         false | ||||
|                     } | ||||
|  | ||||
|                 @Suppress("detekt:SwallowedException", "detekt:ReturnCount") | ||||
|                 @Deprecated("Deprecated in Java") | ||||
|                 override fun shouldInterceptRequest( | ||||
|                     view: WebView, | ||||
|                     url: String, | ||||
|                 ): WebResourceResponse? { | ||||
|                     val (mime: String?, compression: Bitmap.CompressFormat) = | ||||
|                         if (url | ||||
|                                 .lowercase(Locale.US) | ||||
|                                 .contains(".jpg") || | ||||
|                             url.lowercase(Locale.US).contains(".jpeg") | ||||
|                         ) { | ||||
|                             Pair(IMAGE_JPG, Bitmap.CompressFormat.JPEG) | ||||
|                         } else if (url.lowercase(Locale.US).contains(".png")) { | ||||
|                             Pair(IMAGE_PNG, Bitmap.CompressFormat.PNG) | ||||
|                         } else if (url.lowercase(Locale.US).contains(".webp")) { | ||||
|                             Pair(IMAGE_WEBP, Bitmap.CompressFormat.WEBP) | ||||
|                         } else { | ||||
|                             return super.shouldInterceptRequest(view, url) | ||||
|                         } | ||||
|  | ||||
|                     try { | ||||
|                         val image = view.getGlideImageForResource(url, appSettingsService) | ||||
|                         return WebResourceResponse( | ||||
|                             mime, | ||||
|                             "UTF-8", | ||||
|                             getBitmapInputStream(image, compression), | ||||
|                         ) | ||||
|                     } catch (e: ExecutionException) { | ||||
|                         return super.shouldInterceptRequest(view, url) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     @Suppress("detekt:LongMethod", "detekt:ImplicitDefaultLocale") | ||||
|     private fun htmlToWebview() { | ||||
|         maybeIfContext { | ||||
|             val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||
|             val a: TypedArray = it.obtainStyledAttributes(resId, attrs) | ||||
|  | ||||
|         val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||
|         val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) | ||||
|  | ||||
|  | ||||
|         binding.webcontent.settings.standardFontFamily = a.getString(0) | ||||
|             binding.webcontent.settings.standardFontFamily = a.getString(0) | ||||
|             "" | ||||
|         } | ||||
|         binding.webcontent.visibility = View.VISIBLE | ||||
|  | ||||
|         val colorOnSurface = TypedValue() | ||||
|         requireContext().theme.resolveAttribute(R.attr.colorOnSurface, colorOnSurface, true) | ||||
|         val colorSurfaceString = | ||||
|             String.format( | ||||
|                 "#%06X", | ||||
|                 WHITE_COLOR_HEX and (if (colorSurface != DATA_NULL_UNDEFINED) colorSurface else WHITE_COLOR_HEX), | ||||
|             ) | ||||
|  | ||||
|         val colorSurface = TypedValue() | ||||
|         requireContext().theme.resolveAttribute(R.attr.colorSurface, colorSurface, true) | ||||
|         val colorOnSurfaceString = | ||||
|             String.format( | ||||
|                 "#%06X", | ||||
|                 WHITE_COLOR_HEX and (if (colorOnSurface != DATA_NULL_UNDEFINED) colorOnSurface else 0), | ||||
|             ) | ||||
|  | ||||
|         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 | ||||
|             } | ||||
|         handleImageLoading() | ||||
|         try { | ||||
|             val gestureDetector = | ||||
|                 GestureDetector( | ||||
|                     activity, | ||||
|                     object : GestureDetector.SimpleOnGestureListener() { | ||||
|                         override fun onSingleTapUp(e: MotionEvent): Boolean = performClick() | ||||
|                     }, | ||||
|                 ) | ||||
|  | ||||
|             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")) { | ||||
|                     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) {} | ||||
|                 } | ||||
|                 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) {} | ||||
|                 } | ||||
|                 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) {} | ||||
|                 } | ||||
|  | ||||
|                 return super.shouldInterceptRequest(view, url) | ||||
|             binding.webcontent.setOnTouchListener { _, event -> | ||||
|                 gestureDetector.onTouchEvent( | ||||
|                     event, | ||||
|                 ) | ||||
|             } | ||||
|         } catch (e: IllegalStateException) { | ||||
|             e.sendSilentlyWithAcraWithName("Gesture detector issue ?") | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||
|             override fun onSingleTapUp(e: MotionEvent?): Boolean { | ||||
|                 return performClick() | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|         binding.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event)} | ||||
|  | ||||
|         binding.webcontent.settings.layoutAlgorithm = | ||||
|                 WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||
|             WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||
|  | ||||
|         var baseUrl: String? = null | ||||
|  | ||||
|         try { | ||||
|             val itemUrl = URL(url) | ||||
|             val itemUrl = URL(url.orEmpty()) | ||||
|             baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||
|         } catch (e: MalformedURLException) { | ||||
|             e.sendSilentlyWithAcraWithName("htmlToWebview > ${url.orEmpty()}") | ||||
|         } | ||||
|  | ||||
|         val fontName =  when (font) { | ||||
|             getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|             getString(R.string.roboto_font_id) -> "Roboto" | ||||
|             else -> "" | ||||
|         } | ||||
|         val fontName: String = | ||||
|             maybeIfContext { | ||||
|                 when (font) { | ||||
|                     it.getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|                     it.getString(R.string.roboto_font_id) -> "Roboto" | ||||
|                     it.getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||
|                     else -> "" | ||||
|                 } | ||||
|             }?.toString().orEmpty() | ||||
|  | ||||
|         val fontLinkAndStyle = if (font.isNotEmpty()) { | ||||
|             """<link href="https://fonts.googleapis.com/css?family=${fontName.replace(" ", "+")}" rel="stylesheet"> | ||||
|         val fontLinkAndStyle = | ||||
|             if (fontName.isNotEmpty()) { | ||||
|                 """<link href="https://fonts.googleapis.com/css?family=${ | ||||
|                     fontName.replace( | ||||
|                         " ", | ||||
|                         "+", | ||||
|                     ) | ||||
|                 }" rel="stylesheet"> | ||||
|                 |<style> | ||||
|                 |   * { | ||||
|                 |       font-family: '$fontName'; | ||||
|                 |   } | ||||
|                 |</style> | ||||
|             """.trimMargin() | ||||
|         } else { | ||||
|             "" | ||||
|         } | ||||
|  | ||||
|         binding.webcontent.loadDataWithBaseURL( | ||||
|             baseUrl, | ||||
|             """<html> | ||||
|                 """.trimMargin() | ||||
|             } else { | ||||
|                 "" | ||||
|             } | ||||
|         try { | ||||
|             binding.webcontent.loadDataWithBaseURL( | ||||
|                 baseUrl, | ||||
|                 """<html> | ||||
|                 |<head> | ||||
|                 |   <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||
|                 |   <style> | ||||
| @@ -436,10 +475,15 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |        max-width: 100%; | ||||
|                 |      } | ||||
|                 |      a { | ||||
|                 |        color: ${String.format("#%06X", 0xFFFFFF and resources.getColor(R.color.colorAccent))} !important; | ||||
|                 |        color: ${ | ||||
|                     String.format( | ||||
|                         "#%06X", | ||||
|                         WHITE_COLOR_HEX and (maybeIfContext { it.resources.getColor(R.color.colorAccent) } as Int), | ||||
|                     ) | ||||
|                 } !important; | ||||
|                 |      } | ||||
|                 |      *:not(a) { | ||||
|                 |        color: ${String.format("#%06X", 0xFFFFFF and colorOnSurface.data)}; | ||||
|                 |        color: $colorOnSurfaceString; | ||||
|                 |      } | ||||
|                 |      * { | ||||
|                 |        font-size: ${fontSize}px; | ||||
| @@ -447,11 +491,11 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |        word-break: break-word; | ||||
|                 |        overflow:hidden; | ||||
|                 |        line-height: 1.5em; | ||||
|                 |        background-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)}; | ||||
|                 |        background-color: $colorSurfaceString; | ||||
|                 |      } | ||||
|                 |      body, html { | ||||
|                 |        background-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)} !important; | ||||
|                 |        border-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)}  !important; | ||||
|                 |        background-color: $colorSurfaceString !important; | ||||
|                 |        border-color: $colorSurfaceString  !important; | ||||
|                 |        padding: 0 !important; | ||||
|                 |        margin: 0 !important; | ||||
|                 |      } | ||||
| @@ -461,41 +505,45 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |      pre, code { | ||||
|                 |        white-space: pre-wrap; | ||||
|                 |        width:100%; | ||||
|                 |        background-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)}; | ||||
|                 |        background-color: $colorSurfaceString; | ||||
|                 |      } | ||||
|                 |   </style> | ||||
|                 |   $fontLinkAndStyle | ||||
|                 |</head> | ||||
|                 |<body> | ||||
|                 |   $contentText | ||||
|                 |</body>""".trimMargin(), | ||||
|             "text/html", | ||||
|             "utf-8", | ||||
|             null | ||||
|         ) | ||||
|                 |</body> | ||||
|                 """.trimMargin(), | ||||
|                 "text/html", | ||||
|                 "utf-8", | ||||
|                 null, | ||||
|             ) | ||||
|         } catch (e: IllegalStateException) { | ||||
|             e.sendSilentlyWithAcraWithName("Context required is still null ?") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun scrollDown() { | ||||
|     fun volumeButtonScrollDown() { | ||||
|         val height = binding.nestedScrollView.measuredHeight | ||||
|         binding.nestedScrollView.smoothScrollBy(0, height/2) | ||||
|         binding.nestedScrollView.smoothScrollBy(0, height / 2) | ||||
|     } | ||||
|  | ||||
|     fun scrollUp() { | ||||
|     fun volumeButtonScrollUp() { | ||||
|         val height = binding.nestedScrollView.measuredHeight | ||||
|         binding.nestedScrollView.smoothScrollBy(0, -height/2) | ||||
|         binding.nestedScrollView.smoothScrollBy(0, -height / 2) | ||||
|     } | ||||
|  | ||||
|     private fun openInBrowserAfterFailing() { | ||||
|         binding.progressBar.visibility = View.GONE | ||||
|         requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|         maybeIfContext { | ||||
|             it.openItemUrlInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     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()) | ||||
| @@ -505,10 +553,13 @@ 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) { | ||||
|  | ||||
|             val position : Int = allImages.indexOf(binding.webcontent.hitTestResult.extra) | ||||
|         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) | ||||
|             intent.putExtra("allImages", allImages) | ||||
| @@ -518,6 +569,4 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         } | ||||
|         return false | ||||
|     } | ||||
|  | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,213 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.fragments | ||||
|  | ||||
| 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.android.utils.glide.imageIntoViewTarget | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.maybeIfContext | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| 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.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 | ||||
|  | ||||
| private const val DRAWABLE_SIZE = 30 | ||||
|  | ||||
| class FilterSheetFragment : | ||||
|     BottomSheetDialogFragment(), | ||||
|     DIAware { | ||||
|     private lateinit var binding: FilterFragmentBinding | ||||
|     override val di: DI by closestDI() | ||||
|     private val repository: Repository by instance() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     private var selectedChip: Chip? = null | ||||
|  | ||||
|     override fun onCreateView( | ||||
|         inflater: LayoutInflater, | ||||
|         container: ViewGroup?, | ||||
|         savedInstanceState: Bundle?, | ||||
|     ): View { | ||||
|         binding = | ||||
|             FilterFragmentBinding.inflate( | ||||
|                 inflater, | ||||
|                 container, | ||||
|                 false, | ||||
|             ) | ||||
|  | ||||
|         try { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                 handleTagChips() | ||||
|                 handleSourceChips() | ||||
|  | ||||
|                 binding.progressBar2.visibility = GONE | ||||
|                 binding.filterView.visibility = VISIBLE | ||||
|             } | ||||
|         } catch (e: IllegalStateException) { | ||||
|             dismiss() | ||||
|             e.sendSilentlyWithAcraWithName("FilterSheetFragment > onCreateView") | ||||
|         } | ||||
|  | ||||
|         binding.floatingActionButton2.setOnClickListener { | ||||
|             (activity as HomeActivity).getElementsAccordingToTab() | ||||
|             (activity as HomeActivity).fetchOnEmptyList() | ||||
|             dismiss() | ||||
|         } | ||||
|         return binding.root | ||||
|     } | ||||
|  | ||||
|     private suspend fun handleSourceChips() { | ||||
|         val sourceGroup = binding.sourcesGroup | ||||
|  | ||||
|         repository.getSourcesDetailsOrStats().forEachIndexed { _, source -> | ||||
|             val c: Chip? = | ||||
|                 maybeIfContext { | ||||
|                     Chip(it) | ||||
|                 } as Chip? | ||||
|  | ||||
|             if (c == null) { | ||||
|                 return | ||||
|             } | ||||
|  | ||||
|             c.ellipsize = TextUtils.TruncateAt.END | ||||
|  | ||||
|             maybeIfContext { | ||||
|                 it.imageIntoViewTarget( | ||||
|                     source.getIcon(repository.baseUrl), | ||||
|                     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") | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     appSettingsService, | ||||
|                 ) | ||||
|             } | ||||
|  | ||||
|             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() { | ||||
|         val tagGroup = binding.tagsGroup | ||||
|  | ||||
|         val tags = repository.getTags() | ||||
|  | ||||
|         tags.forEachIndexed { _, tag -> | ||||
|             val c: Chip? = maybeIfContext { Chip(it) } as Chip? | ||||
|             if (c == null) { | ||||
|                 return | ||||
|             } | ||||
|  | ||||
|             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(DRAWABLE_SIZE, DRAWABLE_SIZE) | ||||
|                     gd.cornerRadius = DRAWABLE_SIZE.toFloat() | ||||
|                     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" | ||||
|     } | ||||
| } | ||||
| @@ -6,16 +6,21 @@ import android.view.View | ||||
| import android.view.ViewGroup | ||||
| import androidx.fragment.app.Fragment | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.FragmentImageBinding | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapWithCache | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import org.kodein.di.DI | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.x.closestDI | ||||
| import org.kodein.di.instance | ||||
|  | ||||
| class ImageFragment : Fragment() { | ||||
|  | ||||
|     private lateinit var imageUrl : String | ||||
|     private val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||
| class ImageFragment : | ||||
|     Fragment(), | ||||
|     DIAware { | ||||
|     override val di: DI by closestDI() | ||||
|     private val appSettingsService: AppSettingsService by instance() | ||||
|     private lateinit var imageUrl: String | ||||
|     private var _binding: FragmentImageBinding? = null | ||||
|     private val binding get() = _binding | ||||
|     val binding get() = _binding | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
| @@ -23,16 +28,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) | ||||
|                 .asBitmap() | ||||
|                 .apply(glideOptions) | ||||
|                 .load(imageUrl) | ||||
|                 .into(binding!!.photoView) | ||||
|         requireActivity().bitmapWithCache(imageUrl, binding!!.photoView, appSettingsService) | ||||
|  | ||||
|         return view | ||||
|     } | ||||
| @@ -45,9 +50,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) | ||||
| @@ -55,4 +58,4 @@ class ImageFragment : Fragment() { | ||||
|             return fragment | ||||
|         } | ||||
|     } | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -2,27 +2,26 @@ 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.android.utils.glide.preloadImage | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
|  | ||||
| fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | ||||
| fun SelfossModel.Item.preloadImages( | ||||
|     context: Context, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ): Boolean { | ||||
|     val imageUrls = this.getImages() | ||||
|  | ||||
|     val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL).timeout(10000) | ||||
|  | ||||
|  | ||||
|     try { | ||||
|         for (url in imageUrls) { | ||||
|             if ( URLUtil.isValidUrl(url)) { | ||||
|                 Glide.with(context).asBitmap() | ||||
|                     .apply(glideOptions) | ||||
|                     .load(url).submit() | ||||
|             if (URLUtil.isValidUrl(url)) { | ||||
|                 context.preloadImage(url, appSettingsService) | ||||
|             } | ||||
|         } | ||||
|     } catch (e : Error) { | ||||
|     } catch (e: Error) { | ||||
|         e.sendSilentlyWithAcraWithName("preloadImages") | ||||
|         return false | ||||
|     } | ||||
|  | ||||
| @@ -35,8 +34,8 @@ fun String.toTextDrawableString(): String { | ||||
|         try { | ||||
|             textDrawable.append(s[0]) | ||||
|         } catch (e: StringIndexOutOfBoundsException) { | ||||
|             // We do nothing | ||||
|             e.sendSilentlyWithAcraWithName("toTextDrawableString") | ||||
|         } | ||||
|     } | ||||
|     return textDrawable.toString() | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -3,9 +3,8 @@ 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 = | ||||
| fun SelfossModel.Item.toParcelable(): ParecelableItem = | ||||
|     ParecelableItem( | ||||
|         this.id, | ||||
|         this.datetime, | ||||
| @@ -17,9 +16,11 @@ fun SelfossModel.Item.toParcelable() : ParecelableItem = | ||||
|         this.icon, | ||||
|         this.link, | ||||
|         this.sourcetitle, | ||||
|         this.tags.joinToString(",") | ||||
|         this.tags.joinToString(","), | ||||
|         this.author, | ||||
|     ) | ||||
| fun ParecelableItem.toModel() : SelfossModel.Item = | ||||
|  | ||||
| fun ParecelableItem.toModel(): SelfossModel.Item = | ||||
|     SelfossModel.Item( | ||||
|         this.id, | ||||
|         this.datetime, | ||||
| @@ -31,28 +32,32 @@ 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> { | ||||
|             override fun createFromParcel(source: Parcel): ParecelableItem = ParecelableItem(source) | ||||
|             override fun newArray(size: Int): Array<ParecelableItem?> = arrayOfNulls(size) | ||||
|         } | ||||
|         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) | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     constructor(source: Parcel) : this( | ||||
| @@ -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,7 +1,5 @@ | ||||
| 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 | ||||
| @@ -16,12 +14,27 @@ import androidx.preference.Preference | ||||
| import androidx.preference.PreferenceFragmentCompat | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySettingsBinding | ||||
| 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 bou.amine.apps.readerforselfossv2.service.AppSettingsService.Companion.API_ITEMS_NUMBER | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService.Companion.CURRENT_THEME | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService.Companion.READER_FONT_SIZE | ||||
| 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 { | ||||
| const val MAX_ITEMS_NUMBER = 200 | ||||
|  | ||||
| private const val MIN_ITEMS_NUMBER = 1 | ||||
|  | ||||
| class SettingsActivity : | ||||
|     AppCompatActivity(), | ||||
|     PreferenceFragmentCompat.OnPreferenceStartFragmentCallback, | ||||
|     DIAware { | ||||
|     override val di by closestDI() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
| @@ -30,9 +43,9 @@ class SettingsActivity : AppCompatActivity(), | ||||
|         setContentView(binding.root) | ||||
|         if (savedInstanceState == null) { | ||||
|             supportFragmentManager | ||||
|                     .beginTransaction() | ||||
|                     .replace(R.id.settings, MainPreferenceFragment()) | ||||
|                     .commit() | ||||
|                 .beginTransaction() | ||||
|                 .replace(R.id.settings, MainPreferenceFragment()) | ||||
|                 .commit() | ||||
|         } else { | ||||
|             title = savedInstanceState.getCharSequence(TITLE_TAG) | ||||
|         } | ||||
| @@ -55,144 +68,202 @@ class SettingsActivity : AppCompatActivity(), | ||||
|         outState.putCharSequence(TITLE_TAG, title) | ||||
|     } | ||||
|  | ||||
|     override fun onSupportNavigateUp(): Boolean { | ||||
|         return if (supportFragmentManager.popBackStackImmediate()) { | ||||
|     override fun onSupportNavigateUp(): Boolean = | ||||
|         if (supportFragmentManager.popBackStackImmediate()) { | ||||
|             supportActionBar?.title = getText(R.string.title_activity_settings) | ||||
|             false | ||||
|         } else { | ||||
|             super.onBackPressed() | ||||
|             true | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onPreferenceStartFragment( | ||||
|             caller: PreferenceFragmentCompat, | ||||
|             pref: Preference | ||||
|         caller: PreferenceFragmentCompat, | ||||
|         pref: Preference, | ||||
|     ): Boolean { | ||||
|         // Instantiate the new Fragment | ||||
|         val args = pref.extras | ||||
|         val fragment = supportFragmentManager.fragmentFactory.instantiate( | ||||
|                 classLoader, | ||||
|                 pref.fragment | ||||
|         ).apply { | ||||
|             arguments = args | ||||
|             setTargetFragment(caller, 0) | ||||
|         } | ||||
|         val fragment = | ||||
|             supportFragmentManager.fragmentFactory | ||||
|                 .instantiate( | ||||
|                     classLoader, | ||||
|                     pref.fragment.toString(), | ||||
|                 ).apply { | ||||
|                     arguments = args | ||||
|                     setTargetFragment(caller, 0) | ||||
|                 } | ||||
|         // Replace the existing Fragment with the new Fragment | ||||
|         supportFragmentManager.beginTransaction() | ||||
|                 .replace(R.id.settings, fragment) | ||||
|                 .addToBackStack(null) | ||||
|                 .commit() | ||||
|         supportFragmentManager | ||||
|             .beginTransaction() | ||||
|             .replace(R.id.settings, fragment) | ||||
|             .addToBackStack(null) | ||||
|             .commit() | ||||
|         title = pref.title | ||||
|         supportActionBar?.title = title | ||||
|         return true | ||||
|     } | ||||
|  | ||||
|     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>(CURRENT_THEME)?.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>(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 | ||||
|                                 if (input in MIN_ITEMS_NUMBER..MAX_ITEMS_NUMBER) 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") | ||||
|             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) {} | ||||
|                     override fun afterTextChanged(editable: Editable) { | ||||
|                         try { | ||||
|                             editText.textSize = editable.toString().toInt().toFloat() | ||||
|                         } catch (e: NumberFormatException) { | ||||
|                 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?) { | ||||
|             setPreferencesFromResource(R.xml.pref_theme, rootKey) | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = Preference.OnPreferenceChangeListener { _, newValue -> | ||||
|                 AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||
|                 true | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     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)) | ||||
|                 true | ||||
|             } | ||||
|             preferenceManager.findPreference<Preference>("trackerLink")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { | ||||
|                     openUrl(AppSettingsService.BUG_URL) | ||||
|                     true | ||||
|                 } | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("sourceLink")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { | ||||
|                 openUrl(Uri.parse(AppSettingsService.sourceUrl)) | ||||
|                 false | ||||
|             } | ||||
|             preferenceManager.findPreference<Preference>("sourceLink")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { | ||||
|                     openUrl(AppSettingsService.SOURCE_URL) | ||||
|                     false | ||||
|                 } | ||||
|  | ||||
|             preferenceManager.findPreference<Preference>("translation")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { | ||||
|                 openUrl(Uri.parse(AppSettingsService.translationUrl)) | ||||
|                 false | ||||
|             } | ||||
|             preferenceManager.findPreference<Preference>("translation")?.onPreferenceClickListener = | ||||
|                 Preference.OnPreferenceClickListener { | ||||
|                     openUrl(AppSettingsService.TRANSLATION_URL) | ||||
|                     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,20 @@ | ||||
| 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,18 @@ | ||||
| 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" | ||||
|     } | ||||
| } | ||||
| @@ -2,20 +2,60 @@ package bou.amine.apps.readerforselfossv2.android.utils | ||||
|  | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.util.TypedValue | ||||
| import androidx.annotation.AttrRes | ||||
| import androidx.annotation.ColorInt | ||||
| import androidx.fragment.app.Fragment | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.acra.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
|  | ||||
| fun Context.shareLink(itemUrl: String, itemTitle: String) { | ||||
|     val sendIntent = Intent() | ||||
|     sendIntent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|     sendIntent.action = Intent.ACTION_SEND | ||||
|     sendIntent.putExtra(Intent.EXTRA_TEXT, itemUrl.toStringUriWithHttp()) | ||||
|     sendIntent.putExtra(Intent.EXTRA_SUBJECT, itemTitle) | ||||
|     sendIntent.type = "text/plain" | ||||
|     startActivity( | ||||
|         Intent.createChooser( | ||||
|             sendIntent, | ||||
|             getString(R.string.share) | ||||
|         ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) | ||||
|     ) | ||||
| } | ||||
| fun Context.shareLink( | ||||
|     itemUrl: String?, | ||||
|     itemTitle: String, | ||||
| ) { | ||||
|     if (itemUrl.isUrlValid()) { | ||||
|         val sendIntent = Intent() | ||||
|         sendIntent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|         sendIntent.action = Intent.ACTION_SEND | ||||
|         sendIntent.putExtra(Intent.EXTRA_TEXT, itemUrl!!.toStringUriWithHttp()) | ||||
|         sendIntent.putExtra(Intent.EXTRA_SUBJECT, itemTitle) | ||||
|         sendIntent.type = "text/plain" | ||||
|         startActivity( | ||||
|             Intent | ||||
|                 .createChooser( | ||||
|                     sendIntent, | ||||
|                     getString(R.string.share), | ||||
|                 ).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK), | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ColorInt | ||||
| fun Fragment.getColorFromAttr( | ||||
|     @AttrRes attrColor: Int, | ||||
|     resolveRefs: Boolean = true, | ||||
| ): Int { | ||||
|     val typedValue = TypedValue() | ||||
|     maybeIfContextWithLog { this.requireContext().theme.resolveAttribute(attrColor, typedValue, resolveRefs) } | ||||
|     return typedValue.data | ||||
| } | ||||
|  | ||||
| @Suppress("detekt:SwallowedException") | ||||
| fun Fragment.maybeIfContext(fn: (Context) -> Any): Any? { | ||||
|     try { | ||||
|         return fn(this.requireContext()) | ||||
|     } catch (e: Exception) { | ||||
|         // Do nothing | ||||
|         return null | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Fragment.maybeIfContextWithLog(fn: (Context) -> Any): Any? { | ||||
|     try { | ||||
|         return fn(this.requireContext()) | ||||
|     } catch (e: Exception) { | ||||
|         e.sendSilentlyWithAcraWithName("Fragment context issue...") | ||||
|         return null | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,63 @@ | ||||
| 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 = 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 | ||||
| @@ -18,40 +15,35 @@ import android.widget.Toast | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.ReaderActivity | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
| import bou.amine.apps.readerforselfossv2.utils.isEmptyOrNullOrNullString | ||||
| import okhttp3.HttpUrl.Companion.toHttpUrlOrNull | ||||
|  | ||||
| fun Context.openItemUrl( | ||||
|     allItems: ArrayList<SelfossModel.Item>, | ||||
|     currentItem: Int, | ||||
|     linkDecoded: String, | ||||
|     linkDecoded: String?, | ||||
|     articleViewer: Boolean, | ||||
|     app: Activity | ||||
|     app: Activity, | ||||
| ) { | ||||
|  | ||||
|     if (!linkDecoded.isUrlValid()) { | ||||
|         Toast.makeText( | ||||
|             this, | ||||
|             this.getString(R.string.cant_open_invalid_url), | ||||
|             Toast.LENGTH_LONG | ||||
|         ).show() | ||||
|         Toast | ||||
|             .makeText( | ||||
|                 this, | ||||
|                 this.getString(R.string.cant_open_invalid_url), | ||||
|                 Toast.LENGTH_LONG, | ||||
|             ).show() | ||||
|     } else { | ||||
|         if (articleViewer) { | ||||
|             ReaderActivity.allItems = allItems | ||||
|             val intent = Intent(this, ReaderActivity::class.java) | ||||
|             intent.putExtra("currentItem", currentItem) | ||||
|             app.startActivity(intent) | ||||
|         } else { | ||||
|             val intent = Intent(Intent.ACTION_VIEW) | ||||
|             intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|             intent.data = Uri.parse(linkDecoded.toStringUriWithHttp()) | ||||
|             startActivity(intent) | ||||
|             this.openUrlInBrowserAsNewTask(linkDecoded!!) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun String.isUrlValid(): Boolean = | ||||
|     this.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||
| fun String?.isUrlValid(): Boolean = | ||||
|     !this.isEmptyOrNullOrNullString() && this!!.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||
|  | ||||
| fun String.isBaseUrlInvalid(): Boolean { | ||||
|     val baseUrl = this.toHttpUrlOrNull() | ||||
| @@ -64,15 +56,39 @@ fun String.isBaseUrlInvalid(): Boolean { | ||||
|     return !(Patterns.WEB_URL.matcher(this).matches() && existsAndEndsWithSlash) | ||||
| } | ||||
|  | ||||
| fun Context.openInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
| fun Context.openItemUrlInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
|     this.openUrlInBrowserAsNewTask(i.getLinkDecoded()) | ||||
| } | ||||
|  | ||||
| fun Context.openUrlInBrowserAsNewTask(url: String?) { | ||||
|     if (url.isUrlValid()) { | ||||
|         val intent = Intent(Intent.ACTION_VIEW) | ||||
|         intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|         intent.data = Uri.parse(url) | ||||
|         this.mayBeStartActivity(intent) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Context.openUrlInBrowser(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) | ||||
| } | ||||
|  | ||||
| @Suppress("detekt:SwallowedException") | ||||
| 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 | ||||
|   | ||||
| @@ -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,26 @@ | ||||
| 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 = | ||||
|         reportBuilder.exception !is DeadSystemException && | ||||
|             (reportBuilder.exception != null && reportBuilder.exception!!::class.simpleName != "CannotDeliverBroadcastException") | ||||
|  | ||||
|     override fun shouldSendReport( | ||||
|         context: Context, | ||||
|         config: CoreConfiguration, | ||||
|         crashReportData: CrashReportData, | ||||
|     ): Boolean = crashReportData.get("BRAND") != "redroid" && !crashReportData.get("PHONE_MODEL").toString().startsWith("sdk_gphone") | ||||
| } | ||||
| @@ -1,6 +1,13 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.bottombar | ||||
|  | ||||
| import android.content.Context | ||||
| import android.graphics.drawable.Drawable | ||||
| import androidx.annotation.IdRes | ||||
| import androidx.annotation.StringRes | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import com.ashokvarma.bottomnavigation.TextBadgeItem | ||||
| import com.leinardi.android.speeddial.SpeedDialActionItem | ||||
| import com.leinardi.android.speeddial.SpeedDialView | ||||
|  | ||||
| fun TextBadgeItem.removeBadge(): TextBadgeItem { | ||||
|     this.setText("") | ||||
| @@ -8,5 +15,26 @@ 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 | ||||
|  | ||||
| @Suppress("detekt:LongParameterList") | ||||
| fun SpeedDialView.addHomeMadeActionItem( | ||||
|     @IdRes actionId: Int, | ||||
|     actionIcon: Drawable, | ||||
|     @StringRes labelId: Int, | ||||
|     colorOnSurface: Int, | ||||
|     colorSurface: Int, | ||||
|     context: Context, | ||||
| ) { | ||||
|     this.addActionItem( | ||||
|         SpeedDialActionItem | ||||
|             .Builder(actionId, actionIcon) | ||||
|             .setFabBackgroundColor(context.resources.getColor(R.color.colorAccent)) | ||||
|             .setFabImageTintColor(colorOnSurface) | ||||
|             .setLabel(context.getString(labelId)) | ||||
|             .setLabelClickable(false) | ||||
|             .setLabelBackgroundColor(colorOnSurface) | ||||
|             .setLabelColor(colorSurface) | ||||
|             .create(), | ||||
|     ) | ||||
| } | ||||
|   | ||||
| @@ -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) | ||||
| } | ||||
| @@ -2,41 +2,135 @@ package bou.amine.apps.readerforselfossv2.android.utils.glide | ||||
|  | ||||
| import android.content.Context | ||||
| import android.graphics.Bitmap | ||||
| import android.graphics.drawable.Drawable | ||||
| import android.webkit.WebView | ||||
| import android.widget.ImageView | ||||
| import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.CircleImageView | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.isEmptyOrNullOrNullString | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||
| import com.bumptech.glide.load.model.GlideUrl | ||||
| import com.bumptech.glide.load.model.LazyHeaders | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import com.bumptech.glide.request.target.BitmapImageViewTarget | ||||
| import com.bumptech.glide.request.target.ViewTarget | ||||
| import com.google.android.material.chip.Chip | ||||
| import java.io.ByteArrayInputStream | ||||
| import java.io.ByteArrayOutputStream | ||||
| import java.io.InputStream | ||||
| import kotlin.io.encoding.Base64 | ||||
| import kotlin.io.encoding.ExperimentalEncodingApi | ||||
|  | ||||
| fun Context.bitmapCenterCrop(url: String, iv: ImageView) = | ||||
|     Glide.with(this) | ||||
|         .asBitmap() | ||||
|         .load(url) | ||||
|         .apply(RequestOptions.centerCropTransform()) | ||||
|         .into(iv) | ||||
| private const val PRELOAD_IMAGE_TIMEOUT = 10000 | ||||
|  | ||||
| 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) | ||||
|             } | ||||
|         }) | ||||
| @Suppress("detekt:ReturnCount") | ||||
| @OptIn(ExperimentalEncodingApi::class) | ||||
| fun String.toGlideUrl(appSettingsService: AppSettingsService): Any { // GlideUrl Or String | ||||
|     if (this.isEmptyOrNullOrNullString()) { | ||||
|         return "" | ||||
|     } | ||||
|     if (appSettingsService.getBasicUserName().isNotEmpty()) { | ||||
|         val authString = "${appSettingsService.getBasicUserName()}:${appSettingsService.getBasicPassword()}" | ||||
|         val authBuf = Base64.encode(authString.toByteArray(Charsets.UTF_8)) | ||||
|  | ||||
| fun getBitmapInputStream(bitmap:Bitmap,compressFormat: Bitmap.CompressFormat): InputStream { | ||||
|         return GlideUrl( | ||||
|             this, | ||||
|             LazyHeaders | ||||
|                 .Builder() | ||||
|                 .addHeader("Authorization", "Basic $authBuf") | ||||
|                 .build(), | ||||
|         ) | ||||
|     } else { | ||||
|         return GlideUrl( | ||||
|             this, | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun WebView.getGlideImageForResource( | ||||
|     url: String, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) = Glide | ||||
|     .with(this) | ||||
|     .asBitmap() | ||||
|     .apply(RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL)) | ||||
|     .load(url.toGlideUrl(appSettingsService)) | ||||
|     .submit() | ||||
|     .get() | ||||
|  | ||||
| fun Context.preloadImage( | ||||
|     url: String, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) = Glide | ||||
|     .with(this) | ||||
|     .asBitmap() | ||||
|     .apply(RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL).timeout(PRELOAD_IMAGE_TIMEOUT)) | ||||
|     .load(url.toGlideUrl(appSettingsService)) | ||||
|     .submit() | ||||
|  | ||||
| fun Context.imageIntoViewTarget( | ||||
|     url: String, | ||||
|     target: ViewTarget<Chip?, Drawable?>, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) = Glide | ||||
|     .with(this) | ||||
|     .load(url.toGlideUrl(appSettingsService)) | ||||
|     .into(target) | ||||
|  | ||||
| fun Context.bitmapWithCache( | ||||
|     url: String, | ||||
|     iv: ImageView, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) = Glide | ||||
|     .with(this) | ||||
|     .asBitmap() | ||||
|     .apply(RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL)) | ||||
|     .load(url.toGlideUrl(appSettingsService)) | ||||
|     .into(iv) | ||||
|  | ||||
| fun Context.bitmapCenterCrop( | ||||
|     url: String, | ||||
|     iv: ImageView, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) = Glide | ||||
|     .with(this) | ||||
|     .asBitmap() | ||||
|     .load(url.toGlideUrl(appSettingsService)) | ||||
|     .apply(RequestOptions.centerCropTransform()) | ||||
|     .into(iv) | ||||
|  | ||||
| fun Context.bitmapFitCenter( | ||||
|     url: String, | ||||
|     iv: ImageView, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) = Glide | ||||
|     .with(this) | ||||
|     .asBitmap() | ||||
|     .load(url.toGlideUrl(appSettingsService)) | ||||
|     .apply(RequestOptions.fitCenterTransform()) | ||||
|     .into(iv) | ||||
|  | ||||
| fun Context.circularDrawable( | ||||
|     url: String, | ||||
|     view: CircleImageView, | ||||
|     appSettingsService: AppSettingsService, | ||||
| ) { | ||||
|     view.textView.text = "" | ||||
|  | ||||
|     Glide | ||||
|         .with(this) | ||||
|         .load(url.toGlideUrl(appSettingsService)) | ||||
|         .into(view.imageView) | ||||
| } | ||||
|  | ||||
| private const val BITMAP_INPUT_STREAM_COMPRESSION_QUALITY = 80 | ||||
|  | ||||
| fun getBitmapInputStream( | ||||
|     bitmap: Bitmap, | ||||
|     compressFormat: Bitmap.CompressFormat, | ||||
| ): InputStream { | ||||
|     val byteArrayOutputStream = ByteArrayOutputStream() | ||||
|     bitmap.compress(compressFormat, 80, byteArrayOutputStream) | ||||
|     bitmap.compress(compressFormat, BITMAP_INPUT_STREAM_COMPRESSION_QUALITY, byteArrayOutputStream) | ||||
|     val bitmapData: ByteArray = byteArrayOutputStream.toByteArray() | ||||
|     return ByteArrayInputStream(bitmapData) | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package bou.amine.apps.readerforselfossv2.android.utils.network | ||||
| import android.content.Context | ||||
| import android.net.ConnectivityManager | ||||
| import android.net.NetworkCapabilities | ||||
| import android.os.Build | ||||
| import com.google.android.material.snackbar.Snackbar | ||||
|  | ||||
| lateinit var s: Snackbar | ||||
| @@ -11,19 +10,13 @@ lateinit var s: Snackbar | ||||
| fun isNetworkAccessible(context: Context): Boolean { | ||||
|     val connectivityManager = context.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager | ||||
|  | ||||
|     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|         val network = connectivityManager.activeNetwork ?: return false | ||||
|         val networkCapabilities = connectivityManager.getNetworkCapabilities(network) ?: return false | ||||
|     val networkCapabilities = connectivityManager.getNetworkCapabilities(connectivityManager.activeNetwork) ?: return false | ||||
|  | ||||
|         return when { | ||||
|             networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_BLUETOOTH) -> true | ||||
|             networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) -> true | ||||
|             networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET) -> true | ||||
|             networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) -> true | ||||
|             else -> false | ||||
|         } | ||||
|     } else { | ||||
|         val network = connectivityManager.activeNetworkInfo ?: return false | ||||
|         return network.isConnectedOrConnecting | ||||
|     return when { | ||||
|         networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_BLUETOOTH) -> true | ||||
|         networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) -> true | ||||
|         networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET) -> true | ||||
|         networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) -> true | ||||
|         else -> false | ||||
|     } | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -1,30 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.viewmodel | ||||
|  | ||||
| import androidx.lifecycle.ViewModel | ||||
| import androidx.lifecycle.viewModelScope | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import kotlinx.coroutines.flow.MutableSharedFlow | ||||
| import kotlinx.coroutines.flow.asSharedFlow | ||||
| import kotlinx.coroutines.launch | ||||
|  | ||||
| class AppViewModel(private val repository: Repository) : ViewModel() { | ||||
|     private val _networkAvailableProvider = MutableSharedFlow<Boolean>() | ||||
|     val networkAvailableProvider = _networkAvailableProvider.asSharedFlow() | ||||
|     private var wasConnected = true | ||||
|  | ||||
|     init { | ||||
|         viewModelScope.launch { | ||||
|             repository.isConnectionAvailable.collect { isConnected -> | ||||
|                 if (repository.connectionMonitored) { | ||||
|                     if (isConnected && !wasConnected && repository.connectionMonitored) { | ||||
|                         _networkAvailableProvider.emit(true) | ||||
|                         wasConnected = true | ||||
|                     } 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> | ||||
							
								
								
									
										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,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" | ||||
| @@ -28,12 +27,14 @@ | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content"> | ||||
|  | ||||
|                     <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|                     <androidx.appcompat.widget.Toolbar | ||||
|                         android:id="@+id/toolBar" | ||||
|                         android:layout_width="match_parent" | ||||
|                         android:layout_height="?attr/actionBarSize" | ||||
|                         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 | ||||
| @@ -69,7 +70,7 @@ | ||||
|                             android:paddingBottom="60dp" | ||||
|                             android:scrollbars="vertical" | ||||
|                             app:layout_behavior="@string/appbar_scrolling_view_behavior" | ||||
|                             tools:listitem="@layout/list_item"/> | ||||
|                             tools:listitem="@layout/list_item" /> | ||||
|                     </LinearLayout> | ||||
|  | ||||
|                 </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
| @@ -77,6 +78,7 @@ | ||||
|             </LinearLayout> | ||||
|  | ||||
|         </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|  | ||||
|         <com.ashokvarma.bottomnavigation.BottomNavigationBar | ||||
|             android:id="@+id/bottomBar" | ||||
|             android:layout_width="match_parent" | ||||
| @@ -85,11 +87,4 @@ | ||||
|             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" | ||||
|  | ||||
|              /> | ||||
|             /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|     <androidx.viewpager2.widget.ViewPager2 | ||||
|         android:id="@+id/pager" | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:id="@+id/scrollView" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="0dp" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/appBarLayout" /> | ||||
|         android:layout_height="match_parent" | ||||
|         android:fillViewport="true" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/appBarLayout"> | ||||
|  | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|         <androidx.viewpager2.widget.ViewPager2 | ||||
|             android:id="@+id/pager" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent" | ||||
|             app:layout_constraintBottom_toBottomOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" /> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
|  | ||||
|  | ||||
| </androidx.constraintlayout.motion.widget.MotionLayout> | ||||
|   | ||||
| @@ -1,31 +1,31 @@ | ||||
| <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"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|  | ||||
|              /> | ||||
|             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:gravity="center" | ||||
|         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" | ||||
| @@ -33,67 +33,72 @@ | ||||
|             android:layout_width="wrap_content" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginBottom="8dp" | ||||
|             android:visibility="gone"/> | ||||
|             android:visibility="gone" /> | ||||
|  | ||||
|         <ScrollView | ||||
|         <LinearLayout | ||||
|             android:id="@+id/loginForm" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent"> | ||||
|             android:layout_height="match_parent" | ||||
|             android:orientation="vertical"> | ||||
|  | ||||
|             <LinearLayout | ||||
|             <EditText | ||||
|                 android:id="@+id/urlView" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:orientation="vertical"> | ||||
|                 android:hint="@string/prompt_url" | ||||
|                 android:imeOptions="actionUnspecified" | ||||
|                 android:importantForAutofill="no" | ||||
|                 android:inputType="textUri" | ||||
|                 android:maxLines="1" | ||||
|                 android:minHeight="48dp" /> | ||||
|  | ||||
|                 <EditText | ||||
|                     android:id="@+id/urlView" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:hint="@string/prompt_url" | ||||
|                     android:imeOptions="actionUnspecified" | ||||
|                     android:importantForAutofill="no" | ||||
|                     android:inputType="textUri" | ||||
|                     android:maxLines="1" /> | ||||
|             <com.google.android.material.switchmaterial.SwitchMaterial | ||||
|                 android:id="@+id/selfSigned" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:text="@string/disable_ssl" | ||||
|                 android:textAlignment="viewStart" /> | ||||
|  | ||||
|                 <com.google.android.material.switchmaterial.SwitchMaterial | ||||
|                     android:text="@string/withLoginSwitch" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="0dp" | ||||
|                     android:id="@+id/withLogin" | ||||
|                     android:layout_weight="1"/> | ||||
|             <com.google.android.material.switchmaterial.SwitchMaterial | ||||
|                 android:id="@+id/withLogin" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:text="@string/withLoginSwitch" | ||||
|                 android:textAlignment="viewStart" /> | ||||
|  | ||||
|                 <EditText | ||||
|                     android:id="@+id/loginView" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:autofillHints="username" | ||||
|                     android:hint="@string/prompt_login" | ||||
|                     android:inputType="text" | ||||
|                     android:maxLines="1" | ||||
|                     android:visibility="gone" /> | ||||
|             <EditText | ||||
|                 android:id="@+id/loginView" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:autofillHints="username" | ||||
|                 android:hint="@string/prompt_login" | ||||
|                 android:inputType="text" | ||||
|                 android:maxLines="1" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:visibility="gone" /> | ||||
|  | ||||
|                 <EditText | ||||
|                     android:id="@+id/passwordView" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:autofillHints="password" | ||||
|                     android:hint="@string/prompt_password" | ||||
|                     android:inputType="textPassword" | ||||
|                     android:maxLines="1" | ||||
|                     android:visibility="gone" /> | ||||
|             <EditText | ||||
|                 android:id="@+id/passwordView" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:autofillHints="password" | ||||
|                 android:hint="@string/prompt_password" | ||||
|                 android:inputType="textPassword" | ||||
|                 android:maxLines="1" | ||||
|                 android:minHeight="48dp" | ||||
|                 android:visibility="gone" /> | ||||
|  | ||||
|                 <Button | ||||
|                     android:id="@+id/signInButton" | ||||
|                     style="?android:textAppearanceSmall" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginTop="16dp" | ||||
|                     android:layout_marginBottom="16dp" | ||||
|                     android:text="@string/action_sign_in" | ||||
|                     android:textStyle="bold" /> | ||||
|             <Button | ||||
|                 android:id="@+id/signInButton" | ||||
|                 style="?android:textAppearanceSmall" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:layout_marginBottom="16dp" | ||||
|                 android:text="@string/action_sign_in" | ||||
|                 android:textStyle="bold" /> | ||||
|  | ||||
|             </LinearLayout> | ||||
|         </ScrollView> | ||||
|         </LinearLayout> | ||||
|     </LinearLayout> | ||||
|  | ||||
| </LinearLayout> | ||||
| </LinearLayout> | ||||
| @@ -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" | ||||
| @@ -17,116 +17,83 @@ | ||||
|             <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: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"> | ||||
|             app:layout_constraintLeft_toLeftOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
|             <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" /> | ||||
| 
 | ||||
|             <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"/> | ||||
|                 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="?android:textColorPrimary" | ||||
|                 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> | ||||
| 
 | ||||
| @@ -135,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,18 +1,14 @@ | ||||
| <?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" | ||||
| @@ -28,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" /> | ||||
| @@ -39,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" /> | ||||
|  | ||||
| @@ -58,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" | ||||
| @@ -129,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="?attr/webviewBackground" | ||||
|                 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" /> | ||||
|  | ||||
| @@ -77,46 +71,23 @@ | ||||
|  | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
|  | ||||
|     <FrameLayout | ||||
|         android:layout_width="match_parent" | ||||
|     <com.leinardi.android.speeddial.SpeedDialView | ||||
|         android:id="@+id/speedDial" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="wrap_content" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintLeft_toLeftOf="parent" | ||||
|         android:layout_gravity="end|bottom|right"> | ||||
|  | ||||
|         <com.github.rubensousa.floatingtoolbar.FloatingToolbar | ||||
|             android:id="@+id/floatingToolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             android:layout_gravity="bottom" | ||||
|             app:floatingMenu="@menu/reader_toolbar" /> | ||||
|  | ||||
|         <com.google.android.material.floatingactionbutton.FloatingActionButton | ||||
|             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_marginEnd="16dp" | ||||
|             android:layout_marginRight="16dp" | ||||
|             android:paddingBottom="@dimen/activity_vertical_margin" | ||||
|             android:paddingTop="@dimen/activity_vertical_margin" | ||||
|             android:src="@drawable/ic_add_white_24dp" | ||||
|             app:backgroundTint="?attr/colorAccent" | ||||
|             app:fabSize="mini" | ||||
|             app:rippleColor="?attr/colorAccentDark" /> | ||||
|     </FrameLayout> | ||||
|         android:layout_gravity="bottom|end" | ||||
|         app:layout_behavior="@string/speeddial_scrolling_view_snackbar_behavior" | ||||
|         app:sdMainFabClosedSrc="@drawable/ic_add_white_24dp" /> | ||||
|  | ||||
|     <FrameLayout | ||||
|         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" | ||||
| @@ -126,4 +97,5 @@ | ||||
|             android:progressTint="?attr/colorAccent" /> | ||||
|     </FrameLayout> | ||||
|  | ||||
| </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|  | ||||
| </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|   | ||||
| @@ -1,16 +1,16 @@ | ||||
| <?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"> | ||||
|  | ||||
|     <com.github.chrisbanes.photoview.PhotoView | ||||
|         android:id="@+id/photoView" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:layout_centerVertical="true" | ||||
|         android:layout_centerHorizontal="true" | ||||
|         android:background="@android:color/black" | ||||
|         app:srcCompat="@android:drawable/screen_background_dark" /> | ||||
|         <com.github.chrisbanes.photoview.PhotoView | ||||
|             android:id="@+id/photoView" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent" | ||||
|             android:layout_centerVertical="true" | ||||
|             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/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="1" | ||||
|           app:showAsAction="always"/> | ||||
|           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" | ||||
|   | ||||
| @@ -1,13 +1,6 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <menu xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||
| 
 | ||||
|     <item | ||||
|         android:id="@+id/unread_action" | ||||
|         android:icon="@drawable/ic_baseline_white_eye_24dp" | ||||
|         android:title="@string/unmark" | ||||
|         app:showAsAction="ifRoom" /> | ||||
| 
 | ||||
|     <item | ||||
|         android:id="@+id/open_action" | ||||
|         android:icon="@drawable/ic_open_in_browser_white_24dp" | ||||
| @@ -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,15 +23,7 @@ | ||||
|     <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> | ||||
|     <string name="cant_create_source">"No es pot crear la font."</string> | ||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||
| @@ -62,28 +54,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> | ||||
| @@ -94,7 +76,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> | ||||
| @@ -102,14 +83,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> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <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="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> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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,14 @@ | ||||
|     <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> | ||||
| @@ -62,74 +54,79 @@ | ||||
|     <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="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="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="remove_source">Quelle entfernen</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> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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,11 @@ | ||||
|     <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> | ||||
| @@ -62,28 +54,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> | ||||
| @@ -94,7 +76,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> | ||||
| @@ -102,14 +83,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> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <string name="reader_text_align_left">Alinear a la izquierda</string> | ||||
|     <string name="reader_text_align_justify">Justificado</string> | ||||
|     <string name="settings_reader_font">Modo lectura</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="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> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -1,135 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | ||||
|     <string name="app_name">"Reader for Selfoss"</string> | ||||
|     <string name="title_activity_login">"Log in"</string> | ||||
|     <string name="prompt_password">"Password"</string> | ||||
|     <string name="action_sign_in">"Go"</string> | ||||
|     <string name="error_invalid_password">"Password not long enough"</string> | ||||
|     <string name="error_field_required">"Field required"</string> | ||||
|     <string name="prompt_url">"Url"</string> | ||||
|     <string name="withLoginSwitch">"Login required ?"</string> | ||||
|     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||
|     <string name="prompt_login">"Username"</string> | ||||
|     <string name="label_share">"Share"</string> | ||||
|     <string name="readAll">"Read all"</string> | ||||
|     <string name="action_disconnect">"Disconnect"</string> | ||||
|     <string name="title_activity_settings">"Settings"</string> | ||||
|     <string name="pref_header_general">"General"</string> | ||||
|     <string name="add_source_hint_tags">"Tag1, Tag2, Tag3"</string> | ||||
|     <string name="add_source_hint_url">"Link"</string> | ||||
|     <string name="add_source_hint_name">"Name"</string> | ||||
|     <string name="add_source">"Add a source"</string> | ||||
|     <string name="add_source_save">"Save"</string> | ||||
|     <string name="wrong_infos">"Check your details again."</string> | ||||
|     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||
|     <string name="all_posts_read">"All posts were read"</string> | ||||
|     <string name="nothing_here">"Nothing here"</string> | ||||
|     <string name="tab_new">"New"</string> | ||||
|     <string name="tab_read">"All"</string> | ||||
|     <string name="tab_favs">"Favorites"</string> | ||||
|     <string name="action_about">"About"</string> | ||||
|     <string name="marked_as_read">"Item read"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Undo"</string> | ||||
|     <string name="addStringNoUrl">"Log in to add sources."</string> | ||||
|     <string name="cant_get_sources">"Can't get sources list."</string> | ||||
|     <string name="cant_create_source">"Can't create source."</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="form_not_complete">"The form is not complete"</string> | ||||
|     <string name="pref_header_links">"Links"</string> | ||||
|     <string name="issue_tracker_link">"Issue Tracker"</string> | ||||
|     <string name="issue_tracker_summary">"Report a bug or ask for a new feature"</string> | ||||
|     <string name="warning_wrong_url">"WARNING"</string> | ||||
|     <string name="pref_switch_card_view_title">"Card View"</string> | ||||
|     <string name="share">"Share"</string> | ||||
|     <string name="switch_unread_count">"Display the unread count as a badge for the bottom bar."</string> | ||||
|     <string name="switch_unread_count_title">"Display unread count"</string> | ||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> | ||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> | ||||
|     <string name="pref_general_category_links">"Link handling"</string> | ||||
|     <string name="pref_general_category_displaying">"Displaying"</string> | ||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||
|     <string name="pref_switch_card_view_off">"The articles will be displayed as a list"</string> | ||||
|     <string name="menu_home_refresh">"Update remote"</string> | ||||
|     <string name="refresh_success_response">"The remote is updated, you can now reload the articles list"</string> | ||||
|     <string name="refresh_failer_message">"The update didn't work, try again later, or check your selfoss logs."</string> | ||||
|     <string name="refresh_in_progress">"Refresh in progress"</string> | ||||
|     <string name="card_height_title">Full height cards</string> | ||||
|     <string name="card_height_on">Cards height will adjust to its content</string> | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Source code</string> | ||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">clear</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">edit</string> | ||||
|     <string name="drawer_loading">Loading …</string> | ||||
|     <string name="menu_home_search">Search</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||
|     <string name="pref_header_theme">Themes</string> | ||||
|     <string name="default_theme">Default</string> | ||||
|     <string name="default_dark_theme">Default/Dark</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="translation">Translation</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">Report a bug</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="reader_action_open">Open in browser</string> | ||||
|     <string name="reader_action_share">Share</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||
|     <string name="add_to_favs_reader">Add to favorites</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="markall_dialog_message">This will mark all the items as read.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</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="unmark">Mark item as unread</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="network_connectivity_lost">"Network connection lost"</string> | ||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</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="shortcut_offline">Offline</string> | ||||
|     <string name="pref_api_timeout">Api Timeout</string> | ||||
|     <string name="pref_header_experimental">Experimental</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="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> | ||||
| </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">"Reader for Selfoss"</string> | ||||
|     <string name="title_activity_login">"Login"</string> | ||||
|     <string name="prompt_password">"Mot de passe"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"Mot de passe trop court"</string> | ||||
|     <string name="error_field_required">"Champ requis"</string> | ||||
|     <string name="prompt_url">"Url Selfoss"</string> | ||||
|     <string name="disable_ssl">"Désactiver la vérification SSL"</string> | ||||
|     <string name="withLoginSwitch">"Avec login ?"</string> | ||||
|     <string name="login_url_problem">"Petit souci. Il manque peut-être un / à la fin ?"</string> | ||||
|     <string name="prompt_login">"Utilisateur"</string> | ||||
|     <string name="label_share">"Partager"</string> | ||||
|     <string name="readAll">"Tout lire"</string> | ||||
|     <string name="action_disconnect">"Déconnecter"</string> | ||||
|     <string name="title_activity_settings">"Paramètres"</string> | ||||
| @@ -23,16 +23,8 @@ | ||||
|     <string name="wrong_infos">"Vérifiez vos informations."</string> | ||||
|     <string name="all_posts_not_read">"Tous les posts n'ont pas été lus"</string> | ||||
|     <string name="all_posts_read">"Tous les posts sont lus"</string> | ||||
|     <string name="nothing_here">"Il n'y a rien ici !"</string> | ||||
|     <string name="tab_new">"Non lus"</string> | ||||
|     <string name="tab_read">"Tous"</string> | ||||
|     <string name="tab_favs">"Favoris"</string> | ||||
|     <string name="action_about">"À propos"</string> | ||||
|     <string name="marked_as_read">"Marqué comme lu"</string> | ||||
|     <string name="marked_as_unread">"Marqué comme non lu"</string> | ||||
|     <string name="undo_string">"Annuler"</string> | ||||
|     <string name="addStringNoUrl">"Identifiez-vous pour ajouter une source."</string> | ||||
|     <string name="cant_get_sources">"Impossible de récupérer la liste des sources"</string> | ||||
|     <string name="cant_get_sources">"Impossible de récupérer la liste des sources."</string> | ||||
|     <string name="cant_create_source">"Impossible de créer la source."</string> | ||||
|     <string name="cant_get_spouts_no_network">"Impossible d'obtenir la liste des spouts en raison d'un problème de réseau."</string> | ||||
|     <string name="cant_get_spouts">"Impossible d'obtenir la liste des spouts. Cela pourrait venir de l'api."</string> | ||||
| @@ -43,13 +35,13 @@ | ||||
|     <string name="warning_wrong_url">"ATTENTION"</string> | ||||
|     <string name="pref_switch_card_view_title">"Vue en carte"</string> | ||||
|     <string name="share">"Partager"</string> | ||||
|     <string name="switch_unread_count">"Afficher le nombre d'articles non lus sur la barre en bas de l'écran"</string> | ||||
|     <string name="switch_unread_count">"Afficher le nombre d'articles non lus sur la barre en bas de l'écran."</string> | ||||
|     <string name="switch_unread_count_title">"Afficher le nombre de non lus"</string> | ||||
|     <string name="display_all_counts_title">"Afficher le nombre de favoris et d'articles lus"</string> | ||||
|     <string name="text_wrong_url">"Vous semblez essayer de vous connecter avec une URL invalide. Assurez-vous que c'est la bonne, et si le problème persiste, contactez-moi via le lien du play store. Notez aussi que l'application ne peut fonctionner sans l'application web Selfoss. Vous ne pouvez pas utiliser l'application pour accéder directement aux flux RSS."</string> | ||||
|     <string name="pref_article_viewer_title">"Ouvrir les liens dans l'application"</string> | ||||
|     <string name="pref_article_viewer_on">"Les articles s'ouvriront dans l'application"</string> | ||||
|     <string name="pref_article_viewer_off">"Les articles s'ouvriront dans votre naviguateur par défaut"</string> | ||||
|     <string name="pref_article_viewer_off">"Les articles s'ouvriront dans votre navigateur par défaut"</string> | ||||
|     <string name="pref_general_category_links">"Gestion des liens"</string> | ||||
|     <string name="pref_general_category_displaying">"Affichage"</string> | ||||
|     <string name="pref_switch_card_view_on">"Les articles seront affichés en forme de carte"</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">La taille de la carte s\'adaptera au contenu</string> | ||||
|     <string name="card_height_off">La taille de la carte sera fixe</string> | ||||
|     <string name="source_code">Code source</string> | ||||
|     <string name="drawer_error_loading_tags">Erreur lors du chargement des tags…</string> | ||||
|     <string name="drawer_item_filters">Filtres</string> | ||||
|     <string name="drawer_action_clear">raz</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">éditer</string> | ||||
|     <string name="drawer_loading">Chargement …</string> | ||||
|     <string name="filter_item_tags">Tags</string> | ||||
|     <string name="filter_item_sources">Sources</string> | ||||
|     <string name="menu_home_search">Rechercher</string> | ||||
|     <string name="can_delete_source">Impossible de supprimer la source…</string> | ||||
|     <string name="base_url_error">Il y a eu un souci lors de la communication avec votre instance Selfoss. Si le problèmes persiste, contactez-moi pour trouver une solution.</string> | ||||
|     <string name="base_url_error">Il y a eu un souci lors de la communication avec votre instance Selfoss. Si le problème persiste, contactez-moi pour trouver une solution.</string> | ||||
|     <string name="pref_header_theme">Thèmes</string> | ||||
|     <string name="default_theme">Par défaut</string> | ||||
|     <string name="default_dark_theme">Par défaut/Foncé</string> | ||||
|     <string name="pref_selfoss_category">Api Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Nombre d\'articles chargés</string> | ||||
|     <string name="pref_hidden_tags">Tags Cachés</string> | ||||
|     <string name="pref_general_infinite_loading_title">Charger plus d\'articles au scroll</string> | ||||
|     <string name="translation">Traduction</string> | ||||
|     <string name="cant_open_invalid_url">L’url de l’élément n’est pas valide. En attendant la résolution du problème, le lien ne s\'ouvrira pas.</string> | ||||
|     <string name="drawer_report_bug">Signaler un bug</string> | ||||
|     <string name="items_number_should_be_number">Le nombre d\'articles doit être un entier.</string> | ||||
|     <string name="reader_action_more">Lire plus</string> | ||||
|     <string name="reader_action_open">Ouvrir</string> | ||||
|     <string name="reader_action_share">Partager</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Marquer les articles comme lus à la navigation dans le lecteur d\'article.</string> | ||||
| @@ -92,9 +74,8 @@ | ||||
|     <string name="pref_header_viewer">Lecteur d\'articles</string> | ||||
|     <string name="refresh_dialog_message">En validant, votre instance Selfoss sera mise à jour.</string> | ||||
|     <string name="markall_dialog_message">Marquer tous les éléments comme lus ?</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Marquer comme lu à la navigation.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Marquer comme lu à la navigation</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">Ne pas marquer les articles comme lus à la navigation.</string> | ||||
|     <string name="drawer_item_hidden_tags">Tags Cachés</string> | ||||
|     <string name="unmark">Marquer l\'article comme non lu</string> | ||||
|     <string name="pref_header_offline">Hors ligne et cache</string> | ||||
|     <string name="pref_switch_items_caching_off">Les articles ne seront pas enregistrés et l\'application ne sera pas utilisable hors ligne.</string> | ||||
| @@ -105,11 +86,11 @@ | ||||
|     <string name="network_connectivity_lost">"Connexion au réseau perdue"</string> | ||||
|     <string name="network_connectivity_retrieved">"Connexion réseau de nouveau disponible"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Synchroniser les articles</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Les articles ne seront pas synchronisés en arrière plan</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles seront périodiquement synchronisées</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de synchronisation ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Les articles ne seront pas synchronisés en arrière-plan</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles seront périodiquement synchronisés</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalle de synchronisation ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Synchroniser uniquement lorsque le téléphone est en charge</string> | ||||
|     <string name="loading_notification_title">Chargement ...</string> | ||||
|     <string name="loading_notification_title">Chargement …</string> | ||||
|     <string name="loading_notification_text">Selfoss synchronise vos articles</string> | ||||
|     <string name="notification_channel_sync">Notification de synchronisation</string> | ||||
|     <string name="new_items_channel_sync">Notification de nouveaux articles</string> | ||||
| @@ -119,17 +100,33 @@ | ||||
|     <string name="shortcut_offline">Hors ligne</string> | ||||
|     <string name="pref_api_timeout">Timeout de l\'api</string> | ||||
|     <string name="pref_header_experimental">Expérimental</string> | ||||
|     <string name="webview_dialog_issue_message">Pas de Webview disponible. Désactivation du lecteur d\'article pour éviter les futures erreurs. Les articles seront lu via votre navigateur à l\'avenir.</string> | ||||
|     <string name="webview_dialog_issue_message">Pas de Webview disponible. Désactivation du lecteur d\'article pour éviter les futures erreurs. Les articles seront lus via votre navigateur à l\'avenir.</string> | ||||
|     <string name="webview_dialog_issue_title">Problème de Webview</string> | ||||
|     <string name="reader_text_align_left">Aligner à gauche</string> | ||||
|     <string name="reader_text_align_justify">Justifier le texte</string> | ||||
|     <string name="settings_reader_font">Police du lecteur d\'articles</string> | ||||
|     <string name="reader_static_bar_title">Barre statique pour le visionneur d\'articles</string> | ||||
|     <string name="reader_static_bar_on">La barre sera affichée</string> | ||||
|     <string name="reader_static_bar_off">La barre sera affichée grâce au bouton</string> | ||||
|     <string name="remove_source">Supprimer la source</string> | ||||
|     <string name="pref_theme_title">Thème Clair/Sombre</string> | ||||
|     <string name="mode_dark">Thème sombre</string> | ||||
|     <string name="mode_system">Utiliser les paramètres système</string> | ||||
|     <string name="mode_light">Thème clair</string> | ||||
|     <string name="gdpr_dialog_title">L\'application ne partage aucune information personnelle.</string> | ||||
|     <string name="gdpr_dialog_message"><![CDATA[Le rapport de plantage est activés par défaut. Il peut être désactivé depuis les paramètres de l\'application. Notez que les rapports de plantage sont essentiels pour le développement de l\'application.]]></string> | ||||
|     <string name="crash_toast_text">Un bug s\'est produit. Le développeur en sera informé.</string> | ||||
|     <string name="pref_switch_disable_acra">"Désactiver les rapports de plantage."</string> | ||||
|     <string name="menu_home_filter">Filtres</string> | ||||
|     <string name="application_selfoss_only">Cette application ne fonctionne qu\'avec l\'api de Selfoss, et aucun autre.</string> | ||||
|     <string name="menu_home_sources">Sources</string> | ||||
|     <string name="update_source">Mise à jour des sources</string> | ||||
|     <string name="confirm_disconnect_title">Se déconnecter ?</string> | ||||
|     <string name="confirm_disconnect_description">Vous allez être déconnecté de votre instance Selfoss.</string> | ||||
|     <string name="no_browser">Can\'t open a link without a browser installed</string> | ||||
|     <string name="nothing_here">"Il n'y a rien ici !"</string> | ||||
|     <string name="tab_new">"Non lus"</string> | ||||
|     <string name="tab_read">"Tous"</string> | ||||
|     <string name="tab_favs">"Favoris"</string> | ||||
|     <string name="action_about">"À propos"</string> | ||||
|     <string name="marked_as_read">"Marqué comme lu"</string> | ||||
|     <string name="marked_as_unread">"Marqué comme non lu"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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">"Conectar"</string> | ||||
|     <string name="prompt_password">"Contrasinal"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"O contrasinal non é suficientemente longo"</string> | ||||
|     <string name="error_field_required">"Campo requirido"</string> | ||||
|     <string name="prompt_url">"URL"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"É preciso iniciar sesión?"</string> | ||||
|     <string name="login_url_problem">"Ups! Pode que precises engadir un \"/\" o final da URL."</string> | ||||
|     <string name="prompt_login">"Nome de usuario"</string> | ||||
|     <string name="label_share">"Compartir"</string> | ||||
|     <string name="readAll">"Ler todos"</string> | ||||
|     <string name="action_disconnect">"Desconectar"</string> | ||||
|     <string name="title_activity_settings">"Axustes"</string> | ||||
| @@ -23,19 +23,11 @@ | ||||
|     <string name="wrong_infos">"Comprobar os teus detalles de novo."</string> | ||||
|     <string name="all_posts_not_read">"Non se leron todas as publicacións"</string> | ||||
|     <string name="all_posts_read">"Leronse todas as publicacións"</string> | ||||
|     <string name="nothing_here">"Non hai nada aquí"</string> | ||||
|     <string name="tab_new">"Novo"</string> | ||||
|     <string name="tab_read">"Todos"</string> | ||||
|     <string name="tab_favs">"Favoritos"</string> | ||||
|     <string name="action_about">"Acerca de"</string> | ||||
|     <string name="marked_as_read">"Elemento lido"</string> | ||||
|     <string name="marked_as_unread">"Elemento non lido"</string> | ||||
|     <string name="undo_string">"Desfacer"</string> | ||||
|     <string name="addStringNoUrl">"Accede pra engadir fontes."</string> | ||||
|     <string name="cant_get_sources">"Non se pode obter a lista de fontes."</string> | ||||
|     <string name="cant_create_source">"Non se pode crear unha fonte."</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">"Non se pode obter a lista de spouts por mor dun erro de rede."</string> | ||||
|     <string name="cant_get_spouts">"Non se pode obter a lista de spoits. Pode que haxa algún problema coa api."</string> | ||||
|     <string name="form_not_complete">"O formulario non está completo"</string> | ||||
|     <string name="pref_header_links">"Ligazóns"</string> | ||||
|     <string name="issue_tracker_link">"Rastrexador de Incidencias"</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">A altura das tarxetas axustarase ao seu contido</string> | ||||
|     <string name="card_height_off">A altura das tarxetas será fixa</string> | ||||
|     <string name="source_code">Código fonte</string> | ||||
|     <string name="drawer_error_loading_tags">Produciuse un erro ao cargar as etiquetas…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpar</string> | ||||
|     <string name="drawer_item_tags">Etiquetas</string> | ||||
|     <string name="drawer_item_sources">Fontes</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">Fontes</string> | ||||
|     <string name="menu_home_search">Procurar</string> | ||||
|     <string name="can_delete_source">Non se puido eliminar a fonte…</string> | ||||
|     <string name="base_url_error">Houno unha incidencia ao tratar de comunicarse coa túa instancia de Selfoss. Se o problema persiste, prégolle que se poña en contacto conmigo.</string> | ||||
|     <string name="pref_header_theme">Temas</string> | ||||
|     <string name="default_theme">Predeterminado</string> | ||||
|     <string name="default_dark_theme">Predeterminado/Escuro</string> | ||||
|     <string name="pref_selfoss_category">API de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Número de elementos cargados</string> | ||||
|     <string name="pref_hidden_tags">Etiquetas ocultas</string> | ||||
|     <string name="pref_general_infinite_loading_title">Cargar máis artigos ao desprazarse</string> | ||||
|     <string name="translation">Traducción</string> | ||||
|     <string name="cant_open_invalid_url">A URL do elemento non é válida. Estou tratando de solucionar isto pra que a aplicación non falle.</string> | ||||
|     <string name="drawer_report_bug">Informar dun erro</string> | ||||
|     <string name="items_number_should_be_number">O número de elementos debería ser un enteiro.</string> | ||||
|     <string name="reader_action_more">Ler máis</string> | ||||
|     <string name="reader_action_open">Abrir no navegador</string> | ||||
|     <string name="reader_action_share">Compartir</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Marcar artigos como lidos cando se desliza o dedo dun a outro.</string> | ||||
| @@ -94,7 +76,6 @@ | ||||
|     <string name="markall_dialog_message">Isto marcara todos os elementos como lidos.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Marcar artigos como lidos ao deslizar co dedo cara os lados</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">Non marcar artigos como lidos ao deslizar co dedo cara os lados.</string> | ||||
|     <string name="drawer_item_hidden_tags">Etiquetas ocultas</string> | ||||
|     <string name="unmark">Marcar artículo como non lido</string> | ||||
|     <string name="pref_header_offline">Sen conexión e caché</string> | ||||
|     <string name="pref_switch_items_caching_off">Os artigos non se gardaran na memoria do dispositivo e non se poderá utilizar a aplicación sen conexión.</string> | ||||
| @@ -109,7 +90,7 @@ | ||||
|     <string name="pref_switch_periodic_refresh_on">Os artigos sincronizaranse 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ó refrescar cando o teléfono se está a cargar</string> | ||||
|     <string name="loading_notification_title">Cargando...</string> | ||||
|     <string name="loading_notification_title">Cargando…</string> | ||||
|     <string name="loading_notification_text">Selfoss está sincronizando os teus ar tigos</string> | ||||
|     <string name="notification_channel_sync">Notificación de sincronización</string> | ||||
|     <string name="new_items_channel_sync">Notificación de actualizacións</string> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <string name="reader_text_align_left">Aliñar á esquerda</string> | ||||
|     <string name="reader_text_align_justify">Xustificado</string> | ||||
|     <string name="settings_reader_font">Modo lector</string> | ||||
|     <string name="reader_static_bar_title">Barra inferior estática na vista de artigos</string> | ||||
|     <string name="reader_static_bar_on">A barra inferior mostrarase sempre</string> | ||||
|     <string name="reader_static_bar_off">A barra inferior pode mostrarse a través do botón flotante</string> | ||||
|     <string name="remove_source">Eliminar fonte</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="mode_dark">Modo escuro</string> | ||||
|     <string name="mode_system">Seguir axustes do sistema</string> | ||||
|     <string name="mode_light">Modo claro</string> | ||||
|     <string name="gdpr_dialog_title">A aplicación non comparte ningún dato persoal seu.</string> | ||||
|     <string name="gdpr_dialog_message"><![CDATA[O envío de informes de erros está habilitado. Pode deshabilitarse dende a páxina de axustes. Ten en conta que os informes de erros son esenciais para o desenvolvemento da aplicación.]]></string> | ||||
|     <string name="crash_toast_text">Ocurriu un erro. Enviando os detalles o desenvolvedor.</string> | ||||
|     <string name="pref_switch_disable_acra">"Deshabilitar o reporte automático de erros."</string> | ||||
|     <string name="menu_home_filter">Filtros</string> | ||||
|     <string name="application_selfoss_only">Esta aplicación só funciona cunha instancia de Selfoss, e con ningún outro filtro RSS.</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">"Non hai nada aquí"</string> | ||||
|     <string name="tab_new">"Novo"</string> | ||||
|     <string name="tab_read">"Todos"</string> | ||||
|     <string name="tab_favs">"Favoritos"</string> | ||||
|     <string name="action_about">"Acerca de"</string> | ||||
|     <string name="marked_as_read">"Elemento lido"</string> | ||||
|     <string name="marked_as_unread">"Elemento non lido"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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">"Reader for Selfoss"</string> | ||||
|     <string name="title_activity_login">"Masuk"</string> | ||||
|     <string name="prompt_password">"Kata sandi"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"Kata sandinya tidak cukup panjang"</string> | ||||
|     <string name="error_field_required">"Kolom wajib diisi"</string> | ||||
|     <string name="prompt_url">"URL"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"Harus masuk?"</string> | ||||
|     <string name="login_url_problem">"Ups. Anda mungkin harus menambahkan \"/\" di akhir url."</string> | ||||
|     <string name="prompt_login">"Nama pengguna"</string> | ||||
|     <string name="label_share">"Bagikan"</string> | ||||
|     <string name="readAll">"Baca semua"</string> | ||||
|     <string name="action_disconnect">"Putuskan sambungan"</string> | ||||
|     <string name="title_activity_settings">"Pengaturan"</string> | ||||
| @@ -23,19 +23,11 @@ | ||||
|     <string name="wrong_infos">"Periksa kembali detail Anda."</string> | ||||
|     <string name="all_posts_not_read">"Semua pos belum dibaca"</string> | ||||
|     <string name="all_posts_read">"Semua pos sudah dibaca"</string> | ||||
|     <string name="nothing_here">"Tidak ada di sini"</string> | ||||
|     <string name="tab_new">"Baru"</string> | ||||
|     <string name="tab_read">"Semua"</string> | ||||
|     <string name="tab_favs">"Favorit"</string> | ||||
|     <string name="action_about">"Tentang"</string> | ||||
|     <string name="marked_as_read">"Membaca item"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Urung"</string> | ||||
|     <string name="addStringNoUrl">"Masuk untuk menambah sumber."</string> | ||||
|     <string name="cant_get_sources">"Tidak bisa mendapatkan daftar sumber."</string> | ||||
|     <string name="cant_create_source">"Tidak dapat membuat sumber."</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">"Tidak bisa masuk ke daftar Spouts."</string> | ||||
|     <string name="form_not_complete">"Formulirnya belum selesai"</string> | ||||
|     <string name="pref_header_links">"Tautan"</string> | ||||
|     <string name="issue_tracker_link">"Pelacak Masalah"</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">Tinggi kartu akan disesuaikan dengan konten</string> | ||||
|     <string name="card_height_off">Ukuran kartu akan tetap</string> | ||||
|     <string name="source_code">Kode sumber</string> | ||||
|     <string name="drawer_error_loading_tags">Kesalahan saat memuat tag…</string> | ||||
|     <string name="drawer_item_filters">Filter</string> | ||||
|     <string name="drawer_action_clear">kosongkan</string> | ||||
|     <string name="drawer_item_tags">Tag</string> | ||||
|     <string name="drawer_item_sources">Sumber</string> | ||||
|     <string name="drawer_action_edit">suntung</string> | ||||
|     <string name="drawer_loading">Memuat …</string> | ||||
|     <string name="filter_item_tags">Tag</string> | ||||
|     <string name="filter_item_sources">Sumber</string> | ||||
|     <string name="menu_home_search">Cari</string> | ||||
|     <string name="can_delete_source">Tidak dapat menghapus sumber…</string> | ||||
|     <string name="base_url_error">Ada masalah saat berkomunikasi dengan Selfoss Anda. Jika masalah berlanjut, tolong hubungi saya.</string> | ||||
|     <string name="pref_header_theme">Tema</string> | ||||
|     <string name="default_theme">Bawaan</string> | ||||
|     <string name="default_dark_theme">Bawaan/Gelap</string> | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Item nomor dimuat</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="pref_general_infinite_loading_title">Muat lebih banyak artikel saat membalik halaman</string> | ||||
|     <string name="translation">Terjemahan</string> | ||||
|     <string name="cant_open_invalid_url">Alamat tautan proyek tidak valid. Saya mencoba memecahkan masalah ini untuk menghindari aplikasi berhenti.</string> | ||||
|     <string name="drawer_report_bug">Laporkan bug</string> | ||||
|     <string name="items_number_should_be_number">Jumlah item harus berupa bilangan bulat.</string> | ||||
|     <string name="reader_action_more">Baca lebih lanjut</string> | ||||
|     <string name="reader_action_open">Buka di peramban</string> | ||||
|     <string name="reader_action_share">Bagikan</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||
| @@ -94,7 +76,6 @@ | ||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</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="unmark">Mark item as unread</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> | ||||
| @@ -102,14 +83,14 @@ | ||||
|     <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="network_connectivity_lost">"Network connection lost"</string> | ||||
|     <string name="network_connectivity_lost">"Koneksi jaringan hilang"</string> | ||||
|     <string name="network_connectivity_retrieved">"Network connection is now available"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <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="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">"Tidak ada di sini"</string> | ||||
|     <string name="tab_new">"Baru"</string> | ||||
|     <string name="tab_read">"Semua"</string> | ||||
|     <string name="tab_favs">"Favorit"</string> | ||||
|     <string name="action_about">"Tentang"</string> | ||||
|     <string name="marked_as_read">"Membaca item"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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">"Lettore RSS per Selfoss"</string> | ||||
|     <string name="title_activity_login">"Accedi"</string> | ||||
|     <string name="prompt_password">"Password"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"La password non è sufficientemente lunga"</string> | ||||
|     <string name="error_field_required">"Campo obbligatorio"</string> | ||||
|     <string name="prompt_url">"URL"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"È richiesto l'accesso?"</string> | ||||
|     <string name="login_url_problem">"Oops. Potrebbe essere necessario aggiungere un \"/\" alla fine dell'url."</string> | ||||
|     <string name="prompt_login">"Nome utente"</string> | ||||
|     <string name="label_share">"Condividi"</string> | ||||
|     <string name="readAll">"Segna tutte come lette"</string> | ||||
|     <string name="action_disconnect">"Scollegati"</string> | ||||
|     <string name="title_activity_settings">"Impostazioni"</string> | ||||
| @@ -23,15 +23,7 @@ | ||||
|     <string name="wrong_infos">"Controlla nuovamente i dati."</string> | ||||
|     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||
|     <string name="all_posts_read">"Tutti i messaggi sono stati letti"</string> | ||||
|     <string name="nothing_here">"Non c'è niente qui"</string> | ||||
|     <string name="tab_new">"Nuovi"</string> | ||||
|     <string name="tab_read">"Tutti"</string> | ||||
|     <string name="tab_favs">"Preferiti"</string> | ||||
|     <string name="action_about">"Informazioni"</string> | ||||
|     <string name="marked_as_read">"Articolo letto"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Annulla"</string> | ||||
|     <string name="addStringNoUrl">"Autenticati per aggiungere fonti."</string> | ||||
|     <string name="cant_get_sources">"Can't get sources list."</string> | ||||
|     <string name="cant_create_source">"Can't create source."</string> | ||||
|     <string name="cant_get_spouts_no_network">"Can't get spouts list because of a network issue."</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">Cards height will adjust to its content</string> | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Codice sorgente</string> | ||||
|     <string name="drawer_error_loading_tags">Errore nel caricamento dei tag…</string> | ||||
|     <string name="drawer_item_filters">Filtri</string> | ||||
|     <string name="drawer_action_clear">cancella</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Fonti</string> | ||||
|     <string name="drawer_action_edit">modifica</string> | ||||
|     <string name="drawer_loading">Caricamento…</string> | ||||
|     <string name="filter_item_tags">Tags</string> | ||||
|     <string name="filter_item_sources">Fonti</string> | ||||
|     <string name="menu_home_search">Cerca</string> | ||||
|     <string name="can_delete_source">Non è possibile eliminare la fonte…</string> | ||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||
|     <string name="pref_header_theme">Temi</string> | ||||
|     <string name="default_theme">Predefinito</string> | ||||
|     <string name="default_dark_theme">Predefinito (Scuro)</string> | ||||
|     <string name="pref_selfoss_category">Api di Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Numero di elementi caricati</string> | ||||
|     <string name="pref_hidden_tags">Tag nascosti</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Traduzioni</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">Segnala un bug</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="reader_action_open">Open in browser</string> | ||||
|     <string name="reader_action_share">Share</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||
| @@ -94,7 +76,6 @@ | ||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</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="unmark">Segna come non letto</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> | ||||
| @@ -109,7 +90,7 @@ | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <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="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">"Non c'è niente qui"</string> | ||||
|     <string name="tab_new">"Nuovi"</string> | ||||
|     <string name="tab_read">"Tutti"</string> | ||||
|     <string name="tab_favs">"Preferiti"</string> | ||||
|     <string name="action_about">"Informazioni"</string> | ||||
|     <string name="marked_as_read">"Articolo letto"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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">"Reader for Selfoss"</string> | ||||
|     <string name="title_activity_login">"로그인"</string> | ||||
|     <string name="prompt_password">"비밀번호"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"패스워드가 짧습니다."</string> | ||||
|     <string name="error_field_required">"필수 항목"</string> | ||||
|     <string name="prompt_url">"Url"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"로그인이 필요합니까?"</string> | ||||
|     <string name="login_url_problem">"죄송합니다. Url의 끝에 \"/\"를 추가할 필요가 있습니다."</string> | ||||
|     <string name="prompt_login">"사용자 이름"</string> | ||||
|     <string name="label_share">"공유"</string> | ||||
|     <string name="readAll">"모두 읽기"</string> | ||||
|     <string name="action_disconnect">"연결 해제"</string> | ||||
|     <string name="title_activity_settings">"설정"</string> | ||||
| @@ -23,19 +23,11 @@ | ||||
|     <string name="wrong_infos">"세부 정보를 다시 확인하세요."</string> | ||||
|     <string name="all_posts_not_read">"모든 게시물을 읽지 않았습니다."</string> | ||||
|     <string name="all_posts_read">"모든 게시물을 읽었습니다."</string> | ||||
|     <string name="nothing_here">"비어있음"</string> | ||||
|     <string name="tab_new">"새로운"</string> | ||||
|     <string name="tab_read">"전체"</string> | ||||
|     <string name="tab_favs">"즐겨찾기"</string> | ||||
|     <string name="action_about">"정보"</string> | ||||
|     <string name="marked_as_read">"항목 읽기"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"실행 취소"</string> | ||||
|     <string name="addStringNoUrl">"로그인 소스를 추가 해야 합니다."</string> | ||||
|     <string name="cant_get_sources">"소스 리스트를 얻을 수 없습니다."</string> | ||||
|     <string name="cant_create_source">"소스를 만들 수 없습니다."</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">"Spouts 목록을 가져올 수 없습니다."</string> | ||||
|     <string name="form_not_complete">"양식이 완료되지 않았습니다."</string> | ||||
|     <string name="pref_header_links">"링크"</string> | ||||
|     <string name="issue_tracker_link">"이슈 트래커"</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">Cards height will adjust to its content</string> | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Source code</string> | ||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">clear</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">edit</string> | ||||
|     <string name="drawer_loading">Loading …</string> | ||||
|     <string name="filter_item_tags">Tags</string> | ||||
|     <string name="filter_item_sources">Sources</string> | ||||
|     <string name="menu_home_search">Search</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||
|     <string name="pref_header_theme">Themes</string> | ||||
|     <string name="default_theme">Default</string> | ||||
|     <string name="default_dark_theme">Default/Dark</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="translation">Translation</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">Report a bug</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="reader_action_open">Open in browser</string> | ||||
|     <string name="reader_action_share">Share</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||
| @@ -94,7 +76,6 @@ | ||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</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="unmark">Mark item as unread</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> | ||||
| @@ -109,7 +90,7 @@ | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <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="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">"비어있음"</string> | ||||
|     <string name="tab_new">"새로운"</string> | ||||
|     <string name="tab_read">"전체"</string> | ||||
|     <string name="tab_favs">"즐겨찾기"</string> | ||||
|     <string name="action_about">"정보"</string> | ||||
|     <string name="marked_as_read">"항목 읽기"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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">"Selfoss Reader"</string> | ||||
|     <string name="title_activity_login">"Inloggen"</string> | ||||
|     <string name="prompt_password">"Wachtwoord"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"Wachtwoord niet lang genoeg"</string> | ||||
|     <string name="error_field_required">"Dit veld is verplicht"</string> | ||||
|     <string name="prompt_url">"Selfoss server"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"Authenticatie vereist?"</string> | ||||
|     <string name="login_url_problem">"Oeps, ben je soms de \"/\" vergeten aan het eind?"</string> | ||||
|     <string name="prompt_login">"Gebruikersnaam"</string> | ||||
|     <string name="label_share">"Delen"</string> | ||||
|     <string name="readAll">"Alles lezen"</string> | ||||
|     <string name="action_disconnect">"Verbinding verbreken"</string> | ||||
|     <string name="title_activity_settings">"Instellingen"</string> | ||||
| @@ -23,19 +23,11 @@ | ||||
|     <string name="wrong_infos">"Controleer de gegevens nogmaals."</string> | ||||
|     <string name="all_posts_not_read">"Fout bij markeren als gelezen"</string> | ||||
|     <string name="all_posts_read">"Alle artikelen gemarkeerd als gelezen"</string> | ||||
|     <string name="nothing_here">"Niets gevonden"</string> | ||||
|     <string name="tab_new">"Nieuw"</string> | ||||
|     <string name="tab_read">"Alle"</string> | ||||
|     <string name="tab_favs">"Favorieten"</string> | ||||
|     <string name="action_about">"Over"</string> | ||||
|     <string name="marked_as_read">"Artikel gelezen"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Ongedaan maken"</string> | ||||
|     <string name="addStringNoUrl">"Login om bronnen toe te voegen"</string> | ||||
|     <string name="cant_get_sources">"Kan de lijst met bronnen niet ophalen"</string> | ||||
|     <string name="cant_create_source">"Kan bron niet creëeren"</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">"Ophalen spouts mislukt"</string> | ||||
|     <string name="form_not_complete">"Formulier is niet volledig ingevuld"</string> | ||||
|     <string name="pref_header_links">"Links"</string> | ||||
|     <string name="issue_tracker_link">"Bug tracker"</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">Hoogte aanpassen aan de hand van kaartinhoud</string> | ||||
|     <string name="card_height_off">Vaste hoogte</string> | ||||
|     <string name="source_code">Broncode</string> | ||||
|     <string name="drawer_error_loading_tags">Fout bij het laden van tags…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">wissen</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Bronnen</string> | ||||
|     <string name="drawer_action_edit">bewerken</string> | ||||
|     <string name="drawer_loading">Bezig met laden …</string> | ||||
|     <string name="filter_item_tags">Tags</string> | ||||
|     <string name="filter_item_sources">Bronnen</string> | ||||
|     <string name="menu_home_search">Zoeken</string> | ||||
|     <string name="can_delete_source">Kan de bron niet verwijderen…</string> | ||||
|     <string name="base_url_error">Er was een probleem bij het communiceren met uw Selfoss Instance. Als het probleem blijft, neem dan contact met mij op.</string> | ||||
|     <string name="pref_header_theme">Thema \'s</string> | ||||
|     <string name="default_theme">Standaard</string> | ||||
|     <string name="default_dark_theme">Standaard/Donker</string> | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Geladen items nummer</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="pref_general_infinite_loading_title">Laad meer artikelen door te bladeren</string> | ||||
|     <string name="translation">Vertaling</string> | ||||
|     <string name="cant_open_invalid_url">De URL is ongeldig. Ik probeer dit probleem op te lossen, zodat de toepassing niet wordt afgesloten.</string> | ||||
|     <string name="drawer_report_bug">Een fout melden</string> | ||||
|     <string name="items_number_should_be_number">Het aantal items moet een geheel getal zijn.</string> | ||||
|     <string name="reader_action_more">Lees meer</string> | ||||
|     <string name="reader_action_open">Openen in browser</string> | ||||
|     <string name="reader_action_share">Delen</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||
| @@ -94,7 +76,6 @@ | ||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</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="unmark">Mark item as unread</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> | ||||
| @@ -104,12 +85,12 @@ | ||||
|     <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_retrieved">"Network connection is now available"</string> | ||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||
|     <string name="pref_switch_periodic_refresh">Artikel synchronisieren</string> | ||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <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="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">"Niets gevonden"</string> | ||||
|     <string name="tab_new">"Nieuw"</string> | ||||
|     <string name="tab_read">"Alle"</string> | ||||
|     <string name="tab_favs">"Favorieten"</string> | ||||
|     <string name="action_about">"Over"</string> | ||||
|     <string name="marked_as_read">"Artikel gelezen"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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">"Reader for Selfoss"</string> | ||||
|     <string name="title_activity_login">"Entrar"</string> | ||||
|     <string name="prompt_password">"Senha"</string> | ||||
| @@ -7,10 +7,10 @@ | ||||
|     <string name="error_invalid_password">"Senha muito pequena"</string> | ||||
|     <string name="error_field_required">"Campo obrigatório"</string> | ||||
|     <string name="prompt_url">"Url"</string> | ||||
|     <string name="disable_ssl">"Disable SSL"</string> | ||||
|     <string name="withLoginSwitch">"É necessário o login ?"</string> | ||||
|     <string name="login_url_problem">"Oops. Talvez você precise adicionar uma \"/\" no final da url."</string> | ||||
|     <string name="prompt_login">"Usuário"</string> | ||||
|     <string name="label_share">"Compartilhar"</string> | ||||
|     <string name="readAll">"Ler todos"</string> | ||||
|     <string name="action_disconnect">"Desconectar"</string> | ||||
|     <string name="title_activity_settings">"Configurações"</string> | ||||
| @@ -23,19 +23,11 @@ | ||||
|     <string name="wrong_infos">"Verifique os detalhes novamente."</string> | ||||
|     <string name="all_posts_not_read">"Nenhum post foi lido"</string> | ||||
|     <string name="all_posts_read">"Todos os posts foram lidos"</string> | ||||
|     <string name="nothing_here">"Nada aqui"</string> | ||||
|     <string name="tab_new">"Novo"</string> | ||||
|     <string name="tab_read">"Todos"</string> | ||||
|     <string name="tab_favs">"Favoritos"</string> | ||||
|     <string name="action_about">"Sobre"</string> | ||||
|     <string name="marked_as_read">"Item lido"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="undo_string">"Desfazer"</string> | ||||
|     <string name="addStringNoUrl">"Faça login para adicionar fontes."</string> | ||||
|     <string name="cant_get_sources">"Não é possível obter a lista de fontes."</string> | ||||
|     <string name="cant_create_source">"Não é possível criar fonte."</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">"Não é possível obter a lista de spouts."</string> | ||||
|     <string name="form_not_complete">"O formulário não está completo"</string> | ||||
|     <string name="pref_header_links">"Links"</string> | ||||
|     <string name="issue_tracker_link">"Rastreador de problemas"</string> | ||||
| @@ -62,28 +54,18 @@ | ||||
|     <string name="card_height_on">Cards com altura ajustáveis de acordo com o conteúdo</string> | ||||
|     <string name="card_height_off">Cards com altura de tamanho fixo</string> | ||||
|     <string name="source_code">Código fonte</string> | ||||
|     <string name="drawer_error_loading_tags">Erro ao carregar as tags…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpar</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Fontes</string> | ||||
|     <string name="drawer_action_edit">editar</string> | ||||
|     <string name="drawer_loading">Carregando …</string> | ||||
|     <string name="filter_item_tags">Tags</string> | ||||
|     <string name="filter_item_sources">Fontes</string> | ||||
|     <string name="menu_home_search">Procurar</string> | ||||
|     <string name="can_delete_source">Não foi possível apagar a fonte…</string> | ||||
|     <string name="base_url_error">Houve um problema ao tentar se comunicar com o seu Selfoss. Se o problema persistir, entre em contato comigo.</string> | ||||
|     <string name="pref_header_theme">Temas</string> | ||||
|     <string name="default_theme">Padrão</string> | ||||
|     <string name="default_dark_theme">Padrão/Escuro</string> | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Quantidade de itens carregados</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="pref_general_infinite_loading_title">Carregar mais artigos ao realizar o scroll</string> | ||||
|     <string name="translation">Traduções</string> | ||||
|     <string name="cant_open_invalid_url">A url está inválida. Estou tentando resolver esse problema para que o aplicativo não encerre.</string> | ||||
|     <string name="drawer_report_bug">Reportar erro</string> | ||||
|     <string name="items_number_should_be_number">O número dos itens deve ser um número inteiro.</string> | ||||
|     <string name="reader_action_more">Leia mais</string> | ||||
|     <string name="reader_action_open">Abrir no navegador</string> | ||||
|     <string name="reader_action_share">Compartilhar</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_on">Se esta configuração estiver ativada, os artigos serão marcados como lidos ao deslizar para a esquerda e para a direita no leitor do artigo.</string> | ||||
| @@ -94,7 +76,6 @@ | ||||
|     <string name="markall_dialog_message">Isso marcará todos os itens como lidos.</string> | ||||
|     <string name="pref_switch_actions_pager_scroll">Marcar Como Lida ao Abrir</string> | ||||
|     <string name="pref_switch_actions_pager_scroll_off">Não marca artigos como lido quando abrir.</string> | ||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> | ||||
|     <string name="unmark">Mark item as unread</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> | ||||
| @@ -109,7 +90,7 @@ | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -124,12 +105,28 @@ | ||||
|     <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="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 aqui"</string> | ||||
|     <string name="tab_new">"Novo"</string> | ||||
|     <string name="tab_read">"Todos"</string> | ||||
|     <string name="tab_favs">"Favoritos"</string> | ||||
|     <string name="action_about">"Sobre"</string> | ||||
|     <string name="marked_as_read">"Item lido"</string> | ||||
|     <string name="marked_as_unread">"Item unread"</string> | ||||
|     <string name="confirm_delete_title">Confirm Deletion</string> | ||||
|     <string name="confirm_delete_message">Are you sure you want to delete the following source?\n%s</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