Compare commits
7 Commits
v125030711
...
8d4f3c0068
Author | SHA1 | Date | |
---|---|---|---|
8d4f3c0068 | |||
02d503e03a | |||
24b9320d6d | |||
ceba58e98f | |||
c3ee07dd85 | |||
93d99192b3 | |||
359dec2ca0 |
43
.gitea/workflows/common_coverage.yml
Normal file
43
.gitea/workflows/common_coverage.yml
Normal file
@ -0,0 +1,43 @@
|
||||
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: coverage
|
||||
run: |
|
||||
./gradlew androidApp:connectedAndroidTest
|
||||
- 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
|
@ -3,9 +3,11 @@ on:
|
||||
pull_request:
|
||||
branches:
|
||||
- master
|
||||
- chore-crowdin-ci
|
||||
|
||||
jobs:
|
||||
EspressoReports:
|
||||
runs-on: ubuntu-latest
|
||||
uses: ./.gitea/workflows/common_coverage.yml
|
||||
Lint:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
|
11
CHANGELOG.md
11
CHANGELOG.md
@ -1,3 +1,14 @@
|
||||
**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.
|
||||
|
@ -96,6 +96,7 @@ android {
|
||||
// tests
|
||||
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
|
||||
testInstrumentationRunnerArguments["clearPackageData"] = "true"
|
||||
testInstrumentationRunnerArguments["useTestStorageService"] = "true"
|
||||
}
|
||||
packaging {
|
||||
resources {
|
||||
@ -109,6 +110,8 @@ android {
|
||||
proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro")
|
||||
}
|
||||
getByName("debug") {
|
||||
isTestCoverageEnabled = true
|
||||
enableAndroidTestCoverage = true
|
||||
}
|
||||
}
|
||||
flavorDimensions.add("build")
|
||||
@ -197,14 +200,15 @@ dependencies {
|
||||
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.6.2")
|
||||
androidTestImplementation("androidx.test:rules:1.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.5.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.6.1")
|
||||
testImplementation("androidx.test:core-ktx:1.7.0-alpha01")
|
||||
|
||||
implementation("ch.acra:acra-http:$acraVersion")
|
||||
implementation("ch.acra:acra-toast:$acraVersion")
|
||||
|
@ -71,6 +71,8 @@ class SourcesActivityTest {
|
||||
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())
|
||||
}
|
||||
|
||||
|
@ -37,22 +37,6 @@ class ReaderActivity :
|
||||
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)
|
||||
@ -73,14 +57,21 @@ class ReaderActivity :
|
||||
finish()
|
||||
}
|
||||
|
||||
try {
|
||||
readItem(allItems[currentItem])
|
||||
} catch (e: IndexOutOfBoundsException) {
|
||||
finish()
|
||||
}
|
||||
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() {
|
||||
@ -89,14 +80,20 @@ class ReaderActivity :
|
||||
binding.indicator.setViewPager(binding.pager)
|
||||
}
|
||||
|
||||
private fun readItem(item: SelfossModel.Item) {
|
||||
if (appSettingsService.isMarkOnScrollEnabled() && !appSettingsService.getPublicAccess()) {
|
||||
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()
|
||||
@ -141,8 +138,7 @@ class ReaderActivity :
|
||||
}
|
||||
|
||||
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
|
||||
|
||||
alignmentMenu()
|
||||
@ -150,87 +146,50 @@ class ReaderActivity :
|
||||
if (appSettingsService.getPublicAccess()) {
|
||||
menu.removeItem(R.id.star)
|
||||
} else {
|
||||
if (allItems.isNotEmpty() && allItems[currentItem].starred) {
|
||||
canRemoveFromFavorite()
|
||||
} else {
|
||||
canFavorite()
|
||||
}
|
||||
|
||||
binding.pager.registerOnPageChangeCallback(
|
||||
object : ViewPager2.OnPageChangeCallback() {
|
||||
override fun onPageSelected(position: Int) {
|
||||
super.onPageSelected(position)
|
||||
|
||||
if (!allItems.isNullOrEmpty() && allItems.size >= position) {
|
||||
if (allItems[position].starred) {
|
||||
canRemoveFromFavorite()
|
||||
} else {
|
||||
canFavorite()
|
||||
}
|
||||
readItem(allItems[position])
|
||||
}
|
||||
}
|
||||
},
|
||||
)
|
||||
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 -> {
|
||||
onBackPressedDispatcher.onBackPressed()
|
||||
return true
|
||||
}
|
||||
|
||||
R.id.star -> {
|
||||
if (allItems[binding.pager.currentItem].starred) {
|
||||
CoroutineScope(Dispatchers.IO).launch {
|
||||
repository.unstarr(allItems[binding.pager.currentItem])
|
||||
}
|
||||
afterUnsave()
|
||||
} else {
|
||||
CoroutineScope(Dispatchers.IO).launch {
|
||||
repository.starr(allItems[binding.pager.currentItem])
|
||||
}
|
||||
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)
|
||||
alignmentMenu()
|
||||
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 refreshFragment() {
|
||||
finish()
|
||||
overridePendingTransition(0, 0)
|
||||
startActivity(intent)
|
||||
overridePendingTransition(0, 0)
|
||||
private fun switchAlignmentSetting(alignment: Int) {
|
||||
appSettingsService.changeAllignment(alignment)
|
||||
alignmentMenu()
|
||||
|
||||
val fragmentManager = supportFragmentManager
|
||||
val fragments = fragmentManager.fragments
|
||||
|
||||
for (fragment in fragments) {
|
||||
if (fragment is ArticleFragment) {
|
||||
fragment.refreshAlignment()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -263,13 +263,15 @@ class ArticleFragment :
|
||||
)
|
||||
}
|
||||
|
||||
private fun refreshAlignment() {
|
||||
fun refreshAlignment() {
|
||||
textAlignment =
|
||||
when (appSettingsService.getActiveAllignment()) {
|
||||
1 -> "justify"
|
||||
2 -> "left"
|
||||
else -> "justify"
|
||||
}
|
||||
|
||||
htmlToWebview()
|
||||
}
|
||||
|
||||
@Suppress("detekt:SwallowedException")
|
||||
|
@ -1,3 +1,5 @@
|
||||
@file:Suppress("ktlint")
|
||||
/*
|
||||
package bou.amine.apps.readerforselfossv2.android.tests.robolectric
|
||||
|
||||
import android.view.Menu
|
||||
@ -25,3 +27,4 @@ fun Menu.assertVisible(
|
||||
val item = this.findItem(id)
|
||||
assertTrue(item.isVisible)
|
||||
}
|
||||
*/
|
||||
|
@ -1,3 +1,5 @@
|
||||
@file:Suppress("ktlint")
|
||||
/*
|
||||
package bou.amine.apps.readerforselfossv2.android.tests.robolectric
|
||||
|
||||
import android.widget.Button
|
||||
@ -57,7 +59,8 @@ class LoginActivityTest {
|
||||
}
|
||||
}
|
||||
|
||||
/* @Test
|
||||
*/
|
||||
/* @Test
|
||||
fun connect() {
|
||||
Robolectric.buildActivity(LoginActivity::class.java).use { controller ->
|
||||
controller.setup() // Moves the Activity to the RESUMED state
|
||||
@ -72,4 +75,7 @@ class LoginActivityTest {
|
||||
assertEquals(expectedIntent.component, actual.component)
|
||||
}
|
||||
}*/
|
||||
/*
|
||||
|
||||
}
|
||||
*/
|
||||
|
@ -1,3 +1,5 @@
|
||||
@file:Suppress("ktlint")
|
||||
/*
|
||||
package bou.amine.apps.readerforselfossv2.android.tests.robolectric
|
||||
|
||||
import org.robolectric.RobolectricTestRunner
|
||||
@ -8,3 +10,4 @@ class RobotElectriqueRunner(
|
||||
) : RobolectricTestRunner(testClass) {
|
||||
override fun buildGlobalConfig(): Config = Config.Builder().setSdk(25, 30, 33).build()
|
||||
}
|
||||
*/
|
||||
|
@ -0,0 +1,8 @@
|
||||
**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
|
Reference in New Issue
Block a user