Skip to content

Commit

Permalink
Fix #4224 and #4306, part of #1051: Expand written translations suppo…
Browse files Browse the repository at this point in the history
…rt to titles & descriptions (#4555)

## Explanation
Fixes #4224
Fixes #4306
Fixes part of #1051

This PR introduces expanded support for translating content strings. In particular, previous support for written content translations only extended as far as exploration state content and interactions. Now, all topics, stories, chapters, exploration, and revision cards titles and descriptions now also support translations. When considered in conjunction with app string translations, this now means that nearly the entire app is translatable.

### Technical approach
At a high-level, this PR expands the existing pattern for content translations for titles and descriptions for structures: explorations, topics, revision cards (subtopics), stories, and chapters (story nodes). This required changing existing structures (and corresponding test assets) to use ``SubtitledHtml`` instead of raw strings for their titles and descriptions, and to now include written translation tables. New ephemeral structures have been introduced to compute the translations that can be consumed by the UI to ensure the correct string is displayed.

Note that the test JSON assets have not been updated since it was a bit easier to just translate the title/description strings into ``SubtitledHtml``s during parsing time (which seems like a reasonable stopgap since the JSON loading pipeline is temporary and only developer-facing).

Due to new ephemeral structures needing to be computed in a bunch of places, profile ID now needs to be passed in more places. This has led to some minor refactoring to push more UI components over to passing around the ``ProfileId`` structure rather than the internal integer ID.

This PR also fixes part of #1051 by migrating more fragment presenters over to injecting their view models rather than retrieving them from Jetpack's view model provider (which means each of these changed presenters no longer expose potential 
memory leaks in the corresponding view model).

### Caveats
This PR doesn't technically directly address #4224 and #4306, but it does indirectly. The new structures are linked to the assets download script which has been subsequently updated to cross-reference the translations for topics, stories, chapters, and subtopics from Oppia web's repository (which has temporary hardcoded translations for many of the titles and descriptions). This means that:
- Not everything will be translated despite the app now supporting it (example: revision card content isn't available to cross-reference).
- Many translation languages are missing (the team is currently focusing on Brazilian Portuguese ahead of the upcoming Beta MR1 app launch).
- Longer term, Oppia web will provide the translations directly with the structures like it does for explorations (this is an ongoing backend project).

Another caveat is that this PR further reveals the specific need to be able to compute a ``DataProvider`` based on the result of another. This is a limitation that the current ``DataProviders`` utility methods cannot handle, and the workaround is to call ``retrieveData()`` on the computed ``DataProvider`` directly which has updating inconsistency issues. Some of the providers modified as part of this PR should be updated to leverage a new dynamic transform method (see #4564 for the tracking issue). This was unfortunately dropped from the project due to limited time, but #4484 is the WIP PR.

### Test changes (or lack thereof)
Due to this PR being especially high priority, tests have been omitted from this PR in favor of merging it faster. Instead, tests will be added in a follow-up PR (#4567 is tracking to make sure that this happens). The changes have been manually tested, and will be tested more prior to many users interacting with the new functionality.

## Essential Checklist
- [x] The PR title and explanation each start with "Fix #bugnum: " (If this PR fixes part of an issue, prefix the title with "Fix part of #bugnum: ...".)
- [x] Any changes to [scripts/assets](https://github.com/oppia/oppia-android/tree/develop/scripts/assets) files have their rationale included in the PR explanation.
- [x] The PR follows the [style guide](https://github.com/oppia/oppia-android/wiki/Coding-style-guide).
- [x] The PR does not contain any unnecessary code changes from Android Studio ([reference](https://github.com/oppia/oppia-android/wiki/Guidance-on-submitting-a-PR#undo-unnecessary-changes)).
- [x] The PR is made from a branch that's **not** called "develop" and is up-to-date with "develop".
- [x] The PR is **assigned** to the appropriate reviewers ([reference](https://github.com/oppia/oppia-android/wiki/Guidance-on-submitting-a-PR#clarification-regarding-assignees-and-reviewers-section)).

## For UI-specific PRs only
While this PR is highly end user-facing, for the sake of time screenshots and video demonstrations have been omitted from the PR. They will be added at a later date (after the PR is merged), and #4567 is tracking the work item to make sure that this happens.

Commits:

* Create dedicated alpha application component.

This simplifies application component management significantly and
allows individual build flavors to have their own unique module lists.

* Add beta & GA update notices.

This also introduces dedicated beta & GA build flavors which is a
necessary prerequisite.

It also introduces an extra beta, alpha, and dev mode labels for the splash
screen (the latter 2 were extra) with 2 second minimum wait timers for
beta and alpha to ensure they are seen. A 5-second safety timer was
added to ensure the splash screen can always be passed even if something
goes wrong at the domain level (since there are now quite a few moving
pieces to determine the user's current onboarding state).

* Add build tests for the new beta & GA flavors.

* Initial support for dynamic data provider xforms.

* Stabilization improvements.

To ensure that the new dynamic DataProvider can't result in a soft lock
if in-memory cache-backed providers (such as PersistentCacheStore)
circularly reference each other (since they depend on a single thread),
InMemoryBlockingCache was updated to use the standard background in
conjunction with an actor channel. This ensures the lockless
synchronization mechanism can continue without cross-cache contention
occurring due to using a shared test.

This commit also subsequently updates all blocking caches to use the
same actor model, instead, and outright removes the blocking dispatcher
to ensure these situations can never occur in the future.

* Update codebase to Kotlin 1.6.

* Add support for build-specific asset handling.

* Fix build & Kotlin warnings (that are now errors).

One test needed to be ignored due to the Kotlin coroutines update, so
follow-up work will be needed to address this.

Note that this commit also restricts access to the Kotlin reflection
library (since it shouldn't be used broadly).

This commit also updates Gradle to use Kotlin 1.6 (though only the app
build has been verified).

Proguard building support hasn't yet been verified.

* Fix broken test per earlier changes.

* Fix general broken tests & builds.

Tests broken due to changes to the app startup experience haven't yet
been fixed.

* Lint fixes.

* Introduce coordinated executors for tests.

This commit replaces coroutine dispatchers with lower-level executors
that perform the actual state management and coordination for all
multi-threaded task handling in the app. This is actually a major
simplification as it allows for:
- Proper threading resource sharing between the app and its libraries
(such as OkHttp)
- Proper thread synchronization between executors and coroutines in
tests without leveraging the extremely complicated
CoroutineExecutorService
- Less custom coroutine infrastructure which can be difficult to
maintain

* Fix broken tests & lint issues.

* Post-merge & static check fixes.

Does not include new tests that need to be added, or lint fixes.

* Lots of test fixes.

This also includes fixing some core issues in
AsyncDataSubscriptionManager and InMemoryBlockingCache that went
previously unnoticed.

More tests still need to be added yet.

* First part of new tests.

(This is an in-progress commit that's needed so that I can transfer
changes to a different machine).

* First part of adding tests for GA notices.

There's a bunch left to do here, this is mainly needed so that I can
transfer changes to a different machine.

* Update TransformAndroidManifestTest.kt

Correct typos.

* Fix tests & static checks.

This also removes temporary debug code and TODOs, and finishes the tests
for SplashActivity.

* Post-merge fixes.

* Test fixes.

* Fix Gradle test.

* Initial commit for expanded xlations support.

This commit introduces the base changes needed to introduce translations
support for all other content strings in the app outside core lessons.

* Undo executors, Kotlin upgrade, and dynamic xform.

This undoes the work of PRs #4484 and others not yet created in order to
isolate the expanded translations support directly off of the develop
branch to expedite getting it finished for the upcoming beta release.

* Part 1 of post-"rebase" fixes.

* Some post-merge fixes.

* Undo changes from #4436.

Isolating domain assets shouldn't be necessary for the expanded
translations support, and it will be easier to manage this PR with fewer
extra things being added to it.

* Fix various tests & update test assets.

Also, remove temporary TODOs.

* Static fixes.

* Add some string resource checks/tools.

Also, fixes major performance issue with all file-based CI checks.

* Ensure newline consistency in translated strings.

Also, fix reporting in new validation check script.

* Add tests & fix static checks.

* Fix broken tests.

* Update version codes & pt-BR strings.

The strings were manually pulled Translatewiki.

* Follow-up adjustments after self-review.

* Undo changes from #4565.

This commit allows the PR to be "rebased" back onto develop so that it
can be prepped for merging.
  • Loading branch information
BenHenning authored Sep 8, 2022
1 parent 1fcd774 commit 5be4b79
Show file tree
Hide file tree
Showing 125 changed files with 2,280 additions and 988 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -5,15 +5,27 @@ import org.oppia.android.app.home.RouteToTopicPlayStoryListener
import org.oppia.android.app.model.CompletedStory
import org.oppia.android.app.shim.IntentFactoryShim
import org.oppia.android.app.viewmodel.ObservableViewModel
import org.oppia.android.domain.translation.TranslationController

/** Completed story view model for the recycler view in [CompletedStoryListFragment]. */
class CompletedStoryItemViewModel(
private val activity: AppCompatActivity,
private val internalProfileId: Int,
val completedStory: CompletedStory,
val entityType: String,
private val intentFactoryShim: IntentFactoryShim
private val intentFactoryShim: IntentFactoryShim,
translationController: TranslationController
) : ObservableViewModel(), RouteToTopicPlayStoryListener {
val completedStoryName by lazy {
translationController.extractString(
completedStory.storyTitle, completedStory.storyWrittenTranslationContext
)
}
val topicName by lazy {
translationController.extractString(
completedStory.topicTitle, completedStory.topicWrittenTranslationContext
)
}

fun onCompletedStoryItemClicked() {
routeToTopicPlayStory(internalProfileId, completedStory.topicId, completedStory.storyId)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import org.oppia.android.app.shim.IntentFactoryShim
import org.oppia.android.app.viewmodel.ObservableViewModel
import org.oppia.android.domain.oppialogger.OppiaLogger
import org.oppia.android.domain.topic.TopicController
import org.oppia.android.domain.translation.TranslationController
import org.oppia.android.util.data.AsyncResult
import org.oppia.android.util.data.DataProviders.Companion.toLiveData
import org.oppia.android.util.parser.html.StoryHtmlParserEntityType
Expand All @@ -22,6 +23,7 @@ class CompletedStoryListViewModel @Inject constructor(
private val intentFactoryShim: IntentFactoryShim,
private val topicController: TopicController,
private val oppiaLogger: OppiaLogger,
private val translationController: TranslationController,
@StoryHtmlParserEntityType private val entityType: String
) : ObservableViewModel() {
/** [internalProfileId] needs to be set before any of the live data members can be accessed. */
Expand Down Expand Up @@ -73,7 +75,8 @@ class CompletedStoryListViewModel @Inject constructor(
internalProfileId,
completedStory,
entityType,
intentFactoryShim
intentFactoryShim,
translationController
)
}
)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,16 +1,28 @@
package org.oppia.android.app.devoptions.markchapterscompleted

import org.oppia.android.app.model.ChapterPlayState
import org.oppia.android.app.model.ChapterSummary
import org.oppia.android.app.model.EphemeralChapterSummary
import org.oppia.android.domain.translation.TranslationController

/** [MarkChaptersCompletedItemViewModel] for displaying a chapter summary. */
class ChapterSummaryViewModel(
val chapterIndex: Int,
val chapterSummary: ChapterSummary,
ephemeralChapterSummary: EphemeralChapterSummary,
val nextStoryIndex: Int,
val storyId: String,
val topicId: String
val topicId: String,
translationController: TranslationController
) : MarkChaptersCompletedItemViewModel() {
/** The summary of the chapter being displayed. */
val chapterSummary = ephemeralChapterSummary.chapterSummary

/** The localized title of the chapter being displayed. */
val chapterTitle by lazy {
translationController.extractString(
chapterSummary.title, ephemeralChapterSummary.writtenTranslationContext
)
}

/** Returns whether the chapter represented by the current view model is completed. */
fun checkIfChapterIsCompleted(): Boolean =
chapterSummary.chapterPlayState == ChapterPlayState.COMPLETED
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ import androidx.recyclerview.widget.RecyclerView
import org.oppia.android.app.fragment.FragmentScope
import org.oppia.android.app.model.ProfileId
import org.oppia.android.app.recyclerview.BindableAdapter
import org.oppia.android.app.viewmodel.ViewModelProvider
import org.oppia.android.databinding.MarkChaptersCompletedChapterSummaryViewBinding
import org.oppia.android.databinding.MarkChaptersCompletedFragmentBinding
import org.oppia.android.databinding.MarkChaptersCompletedStorySummaryViewBinding
Expand All @@ -22,7 +21,7 @@ import javax.inject.Inject
class MarkChaptersCompletedFragmentPresenter @Inject constructor(
private val activity: AppCompatActivity,
private val fragment: Fragment,
private val viewModelProvider: ViewModelProvider<MarkChaptersCompletedViewModel>,
private val viewModel: MarkChaptersCompletedViewModel,
private val modifyLessonProgressController: ModifyLessonProgressController
) : ChapterSelector {
private lateinit var binding: MarkChaptersCompletedFragmentBinding
Expand All @@ -49,13 +48,13 @@ class MarkChaptersCompletedFragmentPresenter @Inject constructor(

binding.apply {
this.lifecycleOwner = fragment
this.viewModel = getMarkChaptersCompletedViewModel()
this.viewModel = this@MarkChaptersCompletedFragmentPresenter.viewModel
}

this.selectedExplorationIdList = selectedExplorationIdList

profileId = ProfileId.newBuilder().setInternalId(internalProfileId).build()
getMarkChaptersCompletedViewModel().setProfileId(profileId)
viewModel.setProfileId(profileId)

linearLayoutManager = LinearLayoutManager(activity.applicationContext)

Expand All @@ -68,7 +67,7 @@ class MarkChaptersCompletedFragmentPresenter @Inject constructor(
binding.markChaptersCompletedAllCheckBoxContainer.setOnClickListener {
if (binding.isAllChecked == null || binding.isAllChecked == false) {
binding.isAllChecked = true
getMarkChaptersCompletedViewModel().getItemList().forEach { viewModel ->
viewModel.getItemList().forEach { viewModel ->
if (viewModel is ChapterSummaryViewModel) {
if (!viewModel.checkIfChapterIsCompleted())
chapterSelected(
Expand All @@ -84,8 +83,7 @@ class MarkChaptersCompletedFragmentPresenter @Inject constructor(
binding.markChaptersCompletedMarkCompletedTextView.setOnClickListener {
modifyLessonProgressController.markMultipleChaptersCompleted(
profileId = profileId,
chapterMap = getMarkChaptersCompletedViewModel().getChapterMap()
.filterKeys { selectedExplorationIdList.contains(it) }
chapterMap = viewModel.getChapterMap().filterKeys { selectedExplorationIdList.contains(it) }
)
activity.finish()
}
Expand Down Expand Up @@ -122,7 +120,7 @@ class MarkChaptersCompletedFragmentPresenter @Inject constructor(
model: ChapterSummaryViewModel
) {
binding.viewModel = model
val notCompletedChapterCount = getMarkChaptersCompletedViewModel().getItemList().count {
val notCompletedChapterCount = viewModel.getItemList().count {
it is ChapterSummaryViewModel && !it.checkIfChapterIsCompleted()
}
if (notCompletedChapterCount == 0) {
Expand Down Expand Up @@ -155,16 +153,12 @@ class MarkChaptersCompletedFragmentPresenter @Inject constructor(
}
}

private fun getMarkChaptersCompletedViewModel(): MarkChaptersCompletedViewModel {
return viewModelProvider.getForFragment(fragment, MarkChaptersCompletedViewModel::class.java)
}

override fun chapterSelected(chapterIndex: Int, nextStoryIndex: Int, explorationId: String) {
if (!selectedExplorationIdList.contains(explorationId)) {
selectedExplorationIdList.add(explorationId)
}
if (selectedExplorationIdList.size ==
getMarkChaptersCompletedViewModel().getItemList().count {
viewModel.getItemList().count {
it is ChapterSummaryViewModel && !it.checkIfChapterIsCompleted()
}
) {
Expand All @@ -182,14 +176,13 @@ class MarkChaptersCompletedFragmentPresenter @Inject constructor(
override fun chapterUnselected(chapterIndex: Int, nextStoryIndex: Int) {
for (index in chapterIndex until nextStoryIndex) {
val explorationId =
(getMarkChaptersCompletedViewModel().getItemList()[index] as ChapterSummaryViewModel)
.chapterSummary.explorationId
(viewModel.getItemList()[index] as ChapterSummaryViewModel).chapterSummary.explorationId
if (selectedExplorationIdList.contains(explorationId)) {
selectedExplorationIdList.remove(explorationId)
}
}
if (selectedExplorationIdList.size !=
getMarkChaptersCompletedViewModel().getItemList().count {
viewModel.getItemList().count {
it is ChapterSummaryViewModel && !it.checkIfChapterIsCompleted()
}
) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,11 +3,12 @@ package org.oppia.android.app.devoptions.markchapterscompleted
import androidx.lifecycle.LiveData
import androidx.lifecycle.Transformations
import org.oppia.android.app.fragment.FragmentScope
import org.oppia.android.app.model.EphemeralStorySummary
import org.oppia.android.app.model.ProfileId
import org.oppia.android.app.model.StorySummary
import org.oppia.android.app.viewmodel.ObservableViewModel
import org.oppia.android.domain.devoptions.ModifyLessonProgressController
import org.oppia.android.domain.oppialogger.OppiaLogger
import org.oppia.android.domain.translation.TranslationController
import org.oppia.android.util.data.AsyncResult
import org.oppia.android.util.data.DataProviders.Companion.toLiveData
import javax.inject.Inject
Expand All @@ -19,7 +20,8 @@ import javax.inject.Inject
@FragmentScope
class MarkChaptersCompletedViewModel @Inject constructor(
private val oppiaLogger: OppiaLogger,
private val modifyLessonProgressController: ModifyLessonProgressController
private val modifyLessonProgressController: ModifyLessonProgressController,
private val translationController: TranslationController
) : ObservableViewModel() {

private lateinit var profileId: ProfileId
Expand All @@ -34,20 +36,22 @@ class MarkChaptersCompletedViewModel @Inject constructor(
Transformations.map(storyMapLiveData, ::processStoryMap)
}

private val storyMapLiveData: LiveData<Map<String, List<StorySummary>>> by lazy { getStoryMap() }
private val storyMapLiveData: LiveData<Map<String, List<EphemeralStorySummary>>> by lazy {
getStoryMap()
}

private fun getStoryMap(): LiveData<Map<String, List<StorySummary>>> {
private fun getStoryMap(): LiveData<Map<String, List<EphemeralStorySummary>>> {
return Transformations.map(storyMapResultLiveData, ::processStoryMapResult)
}

private val storyMapResultLiveData:
LiveData<AsyncResult<Map<String, List<StorySummary>>>> by lazy {
LiveData<AsyncResult<Map<String, List<EphemeralStorySummary>>>> by lazy {
modifyLessonProgressController.getStoryMapWithProgress(profileId).toLiveData()
}

private fun processStoryMapResult(
storyMap: AsyncResult<Map<String, List<StorySummary>>>
): Map<String, List<StorySummary>> {
storyMap: AsyncResult<Map<String, List<EphemeralStorySummary>>>
): Map<String, List<EphemeralStorySummary>> {
return when (storyMap) {
is AsyncResult.Failure -> {
oppiaLogger.e(
Expand All @@ -61,24 +65,30 @@ class MarkChaptersCompletedViewModel @Inject constructor(
}

private fun processStoryMap(
storyMap: Map<String, List<StorySummary>>
storyMap: Map<String, List<EphemeralStorySummary>>
): List<MarkChaptersCompletedItemViewModel> {
itemList.clear()
var nextStoryIndex: Int
var chapterIndex = 0
storyMap.forEach { storyMapItem ->
storyMapItem.value.forEach { storySummary ->
itemList.add(StorySummaryViewModel(storyName = storySummary.storyName))
storyMapItem.value.forEach { ephemeralStorySummary ->
val storySummary = ephemeralStorySummary.storySummary
val storyTitle =
translationController.extractString(
storySummary.storyTitle, ephemeralStorySummary.writtenTranslationContext
)
itemList.add(StorySummaryViewModel(storyTitle))
chapterIndex++
nextStoryIndex = chapterIndex + storySummary.chapterCount
storySummary.chapterList.forEach { chapterSummary ->
ephemeralStorySummary.chaptersList.forEach { ephemeralChapterSummary ->
itemList.add(
ChapterSummaryViewModel(
chapterIndex = chapterIndex,
chapterSummary = chapterSummary,
ephemeralChapterSummary = ephemeralChapterSummary,
nextStoryIndex = nextStoryIndex,
storyId = storySummary.storyId,
topicId = storyMapItem.key
topicId = storyMapItem.key,
translationController
)
)
chapterIndex++
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package org.oppia.android.app.devoptions.markchapterscompleted

/** [MarkChaptersCompletedItemViewModel] for displaying a story. */
class StorySummaryViewModel(val storyName: String) : MarkChaptersCompletedItemViewModel()
class StorySummaryViewModel(val storyTitle: String) : MarkChaptersCompletedItemViewModel()
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import androidx.recyclerview.widget.LinearLayoutManager
import org.oppia.android.app.fragment.FragmentScope
import org.oppia.android.app.model.ProfileId
import org.oppia.android.app.recyclerview.BindableAdapter
import org.oppia.android.app.viewmodel.ViewModelProvider
import org.oppia.android.databinding.MarkStoriesCompletedFragmentBinding
import org.oppia.android.databinding.MarkStoriesCompletedStorySummaryViewBinding
import org.oppia.android.domain.devoptions.ModifyLessonProgressController
Expand All @@ -20,7 +19,7 @@ import javax.inject.Inject
class MarkStoriesCompletedFragmentPresenter @Inject constructor(
private val activity: AppCompatActivity,
private val fragment: Fragment,
private val viewModelProvider: ViewModelProvider<MarkStoriesCompletedViewModel>,
private val viewModel: MarkStoriesCompletedViewModel,
private val modifyLessonProgressController: ModifyLessonProgressController
) : StorySelector {
private lateinit var binding: MarkStoriesCompletedFragmentBinding
Expand All @@ -47,13 +46,13 @@ class MarkStoriesCompletedFragmentPresenter @Inject constructor(

binding.apply {
this.lifecycleOwner = fragment
this.viewModel = getMarkStoriesCompletedViewModel()
this.viewModel = this@MarkStoriesCompletedFragmentPresenter.viewModel
}

this.selectedStoryIdList = selectedStoryIdList

profileId = ProfileId.newBuilder().setInternalId(internalProfileId).build()
getMarkStoriesCompletedViewModel().setProfileId(profileId)
viewModel.setProfileId(profileId)

linearLayoutManager = LinearLayoutManager(activity.applicationContext)

Expand All @@ -70,7 +69,7 @@ class MarkStoriesCompletedFragmentPresenter @Inject constructor(

binding.markStoriesCompletedAllCheckBox.setOnCheckedChangeListener { _, isChecked ->
if (isChecked) {
getMarkStoriesCompletedViewModel().getStorySummaryMap().values.forEach { viewModel ->
viewModel.getStorySummaryMap().values.forEach { viewModel ->
if (!viewModel.isCompleted)
storySelected(viewModel.storySummary.storyId)
}
Expand All @@ -81,8 +80,9 @@ class MarkStoriesCompletedFragmentPresenter @Inject constructor(
binding.markStoriesCompletedMarkCompletedTextView.setOnClickListener {
modifyLessonProgressController.markMultipleStoriesCompleted(
profileId,
getMarkStoriesCompletedViewModel().getStorySummaryMap()
.filterKeys { selectedStoryIdList.contains(it) }.mapValues { it.value.topicId }
viewModel.getStorySummaryMap().filterKeys {
selectedStoryIdList.contains(it)
}.mapValues { it.value.topicId }
)
activity.finish()
}
Expand All @@ -105,9 +105,7 @@ class MarkStoriesCompletedFragmentPresenter @Inject constructor(
model: StorySummaryViewModel
) {
binding.viewModel = model
if (getMarkStoriesCompletedViewModel().getStorySummaryMap().values
.count { !it.isCompleted } == 0
) {
if (viewModel.getStorySummaryMap().values.count { !it.isCompleted } == 0) {
this.binding.isAllChecked = true
}
if (model.isCompleted) {
Expand All @@ -125,17 +123,13 @@ class MarkStoriesCompletedFragmentPresenter @Inject constructor(
}
}

private fun getMarkStoriesCompletedViewModel(): MarkStoriesCompletedViewModel {
return viewModelProvider.getForFragment(fragment, MarkStoriesCompletedViewModel::class.java)
}

override fun storySelected(storyId: String) {
if (!selectedStoryIdList.contains(storyId)) {
selectedStoryIdList.add(storyId)
}

if (selectedStoryIdList.size ==
getMarkStoriesCompletedViewModel().getStorySummaryMap().values.count { !it.isCompleted }
viewModel.getStorySummaryMap().values.count { !it.isCompleted }
) {
binding.isAllChecked = true
}
Expand All @@ -147,7 +141,7 @@ class MarkStoriesCompletedFragmentPresenter @Inject constructor(
}

if (selectedStoryIdList.size !=
getMarkStoriesCompletedViewModel().getStorySummaryMap().values.count { !it.isCompleted }
viewModel.getStorySummaryMap().values.count { !it.isCompleted }
) {
binding.isAllChecked = false
}
Expand Down
Loading

0 comments on commit 5be4b79

Please sign in to comment.