Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixes #166: Story activity UI structure #195

Merged
merged 51 commits into from
Nov 7, 2019
Merged
Show file tree
Hide file tree
Changes from 49 commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
0e9eaea
Introduce a generic data-binding-enabled RecyclerView adapter. No tests
BenHenning Sep 20, 2019
0ae1439
Initial structure commit
hareshkh Sep 29, 2019
14c8633
Minor changes
hareshkh Oct 2, 2019
2df7459
Merges develop
hareshkh Oct 8, 2019
446df70
Adds story fragment layout
hareshkh Oct 10, 2019
fa10f3b
Passes storyId to fragment presenter
hareshkh Oct 10, 2019
8bb838a
Passes storyId to ViewModel
hareshkh Oct 11, 2019
770fdb0
Adds code to retrieve story live data
hareshkh Oct 11, 2019
39fca77
Merges branch 'develop'
hareshkh Oct 12, 2019
01b629d
Adds button in home-activity and back button in action bar
hareshkh Oct 12, 2019
75307df
Sets story title in action bar
hareshkh Oct 12, 2019
8d6c3a7
Adds chapter summary
Oct 15, 2019
9d70736
Adds chapter list to viewModel
hareshkh Oct 15, 2019
62afbf4
Adds chapter list item
hareshkh Oct 15, 2019
3f5e390
Adds recyclerview
hareshkh Oct 17, 2019
cbfadb1
Adds placeholder thumbnail view
hareshkh Oct 18, 2019
32f468c
Merges branch develop
hareshkh Oct 18, 2019
62362f5
Adds chapter progress
hareshkh Oct 18, 2019
3ecbafb
Converts string concat
hareshkh Oct 21, 2019
34c300b
Shows story thumbnail
hareshkh Oct 21, 2019
8f82ff6
Merge branch develop
hareshkh Oct 22, 2019
3a56b6e
Restores .idea
hareshkh Oct 23, 2019
9fb9f33
Merges develop
hareshkh Oct 23, 2019
503817c
Cosmetic changes
hareshkh Oct 23, 2019
77d7ca8
Cosmetic changes to StoryViewModel
hareshkh Oct 23, 2019
0c99161
Newline at end of each file
hareshkh Oct 23, 2019
3bae028
Adds fragment only when it is not there
hareshkh Oct 30, 2019
5504af6
Removes NPE
hareshkh Oct 30, 2019
dbefd4f
Moves chapter count logic to viewModel and string to plurals
hareshkh Oct 30, 2019
fb4b815
Merges branch develop
hareshkh Oct 30, 2019
20bfdb9
Adds StoryItemViewModel and changes to recyclerViewAdapter
hareshkh Oct 31, 2019
43d84a8
Adds two viewTypes to recycler adapter
hareshkh Oct 31, 2019
207f28d
Newlines at end of files
hareshkh Oct 31, 2019
aefd855
Merge branch 'develop'
hareshkh Oct 31, 2019
d23bf1d
Minor changes to BindableAdapterTest
hareshkh Oct 31, 2019
988a37d
Adds test
hareshkh Nov 1, 2019
7b6704e
Adds routing to exploration activity
hareshkh Nov 3, 2019
70d7a16
Adds exploration loading test
hareshkh Nov 3, 2019
c3d075d
Some review comment changes
hareshkh Nov 3, 2019
1d3fa6c
Minor cosmetic changes
hareshkh Nov 3, 2019
f9ae447
Listeners to invoke ExplorationActivity
hareshkh Nov 5, 2019
c51b394
StoryId can be null
hareshkh Nov 5, 2019
d94c543
Merge branch 'develop'
hareshkh Nov 6, 2019
b05e786
Remove unrequired changes
hareshkh Nov 6, 2019
863728d
Merge branch 'develop' into story-activity-ui-structure
BenHenning Nov 6, 2019
da81a13
Merge branch 'story-activity-ui-structure' of github.com:oppia/oppia-…
BenHenning Nov 6, 2019
12886ec
Undo small accidental edits to home fragment/presenter post merge.
BenHenning Nov 6, 2019
8e1253a
Address reviewer comments (including introducing a test activity for
BenHenning Nov 6, 2019
4a3c07b
Add checkNotNull check in test activity to help quick-fail tests if they
BenHenning Nov 6, 2019
8219df2
Removes unrequired gradle change
hareshkh Nov 6, 2019
d296bd4
Add intent extras check in test
hareshkh Nov 6, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ dependencies {
'androidx.test.espresso:espresso-contrib:3.1.0',
'androidx.test.espresso:espresso-core:3.2.0',
'androidx.test.espresso:espresso-intents:3.1.0',
'com.android.support.test.espresso:espresso-contrib:3.1.0',
hareshkh marked this conversation as resolved.
Show resolved Hide resolved
'androidx.test.ext:junit:1.1.1',
'com.google.truth:truth:0.43',
'org.robolectric:robolectric:4.3',
Expand All @@ -88,6 +89,7 @@ dependencies {
'androidx.test.espresso:espresso-contrib:3.1.0',
'androidx.test.espresso:espresso-core:3.2.0',
'androidx.test.espresso:espresso-intents:3.1.0',
'com.android.support.test.espresso:espresso-contrib:3.1.0',
hareshkh marked this conversation as resolved.
Show resolved Hide resolved
'androidx.test.ext:junit:1.1.1',
'androidx.test:runner:1.2.0',
'com.google.truth:truth:0.43',
Expand Down
1 change: 1 addition & 0 deletions app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
</intent-filter>
</activity>
<activity android:name=".story.StoryActivity" />
<activity android:name=".story.testing.StoryFragmentTestActivity" />
<activity android:name=".testing.BindableAdapterTestActivity" />
<activity android:name=".testing.ContentCardTestActivity" />
<activity android:name=".testing.HtmlParserTestActivity" />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,16 @@ import org.oppia.app.home.continueplaying.ContinuePlayingActivity
import org.oppia.app.home.HomeActivity
import org.oppia.app.player.audio.testing.AudioFragmentTestActivity
import org.oppia.app.player.exploration.ExplorationActivity
import org.oppia.app.topic.conceptcard.testing.ConceptCardFragmentTestActivity
import org.oppia.app.player.state.testing.StateFragmentTestActivity
import org.oppia.app.profile.ProfileActivity
import org.oppia.app.story.StoryActivity
import org.oppia.app.story.testing.StoryFragmentTestActivity
import org.oppia.app.testing.BindableAdapterTestActivity
import org.oppia.app.testing.ContentCardTestActivity
import org.oppia.app.testing.ContinuePlayingFragmentTestActivity
import org.oppia.app.testing.HtmlParserTestActivity
import org.oppia.app.topic.TopicActivity
import org.oppia.app.topic.conceptcard.testing.ConceptCardFragmentTestActivity
import org.oppia.app.topic.questionplayer.QuestionPlayerActivity
import javax.inject.Provider

Expand All @@ -26,7 +27,9 @@ import javax.inject.Provider
interface ActivityComponent {
@Subcomponent.Builder
interface Builder {
@BindsInstance fun setActivity(appCompatActivity: AppCompatActivity): Builder
@BindsInstance
fun setActivity(appCompatActivity: AppCompatActivity): Builder

fun build(): ActivityComponent
}

Expand All @@ -46,4 +49,5 @@ interface ActivityComponent {
fun inject(stateFragmentTestActivity: StateFragmentTestActivity)
fun inject(storyActivity: StoryActivity)
fun inject(topicActivity: TopicActivity)
fun inject(storyFragmentTestActivity: StoryFragmentTestActivity)
}
Empty file.
Empty file.
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,9 @@ import dagger.BindsInstance
import dagger.Subcomponent
import org.oppia.app.home.continueplaying.ContinuePlayingFragment
import org.oppia.app.home.HomeFragment
import org.oppia.app.player.audio.AudioFragment
import org.oppia.app.player.exploration.ExplorationFragment
import org.oppia.app.player.state.StateFragment
import org.oppia.app.player.audio.AudioFragment
import org.oppia.app.profile.AddProfileFragment
import org.oppia.app.profile.AdminAuthFragment
import org.oppia.app.profile.ProfileChooserFragment
Expand Down
23 changes: 19 additions & 4 deletions app/src/main/java/org/oppia/app/recyclerview/BindableAdapter.kt
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ class BindableAdapter<T : Any> internal constructor(
// when T1 is not assignable to T2). This likely won't have bad side effects since any time a
// non-empty list is attempted to be bound, this crash will be correctly triggered.
newDataList.firstOrNull()?.let {
check(it.javaClass.isAssignableFrom(dataClassType.java)) {
check(dataClassType.java.isAssignableFrom(it.javaClass)) {
"Trying to bind incompatible data to adapter. Data class type: ${it.javaClass}, " +
"expected adapter class type: $dataClassType."
}
Expand Down Expand Up @@ -148,6 +148,18 @@ class BindableAdapter<T : Any> internal constructor(
return this
}

/** See [registerViewDataBinder]. */
fun <DB : ViewDataBinding> registerViewDataBinderWithSameModelType(
viewType: Int = DEFAULT_VIEW_TYPE,
inflateDataBinding: (LayoutInflater, ViewGroup, Boolean) -> DB,
setViewModel: (DB, T) -> Unit
): Builder<T> {
return registerViewDataBinder(
viewType = viewType, inflateDataBinding = inflateDataBinding, setViewModel = setViewModel,
transformViewModel = { it }
)
}

/**
* Behaves in the same way as [registerViewBinder] except the inflate and bind methods correspond to a [View]
* data-binding typed [DB].
Expand All @@ -158,12 +170,15 @@ class BindableAdapter<T : Any> internal constructor(
* @param setViewModel a function that initializes the view model in the data-bound view (e.g.
* MyDataBinding::setSpecialViewModel). This may also be a function that initializes the view model & other
* view-accessible properties as necessary.
* @param transformViewModel a function that converts the input model to a model of another type (such as for cases
* when subclassing is used to represent more complex lists of data).
* @return this
*/
fun <DB : ViewDataBinding> registerViewDataBinder(
fun <DB : ViewDataBinding, T2 : T> registerViewDataBinder(
viewType: Int = DEFAULT_VIEW_TYPE,
inflateDataBinding: (LayoutInflater, ViewGroup, Boolean) -> DB,
setViewModel: (DB, T) -> Unit
setViewModel: (DB, T2) -> Unit,
transformViewModel: (T) -> T2
): Builder<T> {
checkViewTypeIsUnique(viewType)
val viewHolderFactory: ViewHolderFactory<T> = { viewGroup ->
Expand All @@ -176,7 +191,7 @@ class BindableAdapter<T : Any> internal constructor(
)
object : BindableViewHolder<T>(binding.root) {
override fun bind(data: T) {
setViewModel(binding, data)
setViewModel(binding, transformViewModel(data))
}
}
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
package org.oppia.app.story

/** Listener for cases when the user taps on a specific chapter/exploration to play. */
interface ExplorationSelectionListener {
/** Called when an exploration has been selected by the user. */
fun selectExploration(explorationId: String)
}
17 changes: 13 additions & 4 deletions app/src/main/java/org/oppia/app/story/StoryActivity.kt
Original file line number Diff line number Diff line change
Expand Up @@ -4,25 +4,34 @@ import android.content.Context
import android.content.Intent
import android.os.Bundle
import org.oppia.app.activity.InjectableAppCompatActivity
import org.oppia.app.home.RouteToExplorationListener
import org.oppia.app.player.exploration.ExplorationActivity
import javax.inject.Inject

/** Activity for stories. */
class StoryActivity : InjectableAppCompatActivity() {
class StoryActivity : InjectableAppCompatActivity(), RouteToExplorationListener {
@Inject lateinit var storyActivityPresenter: StoryActivityPresenter

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
activityComponent.inject(this)
storyActivityPresenter.handleOnCreate()
val storyId: String = checkNotNull(intent.getStringExtra(STORY_ACTIVITY_INTENT_EXTRA)) {
"Expected extra story ID to be included for StoryActivity."
}
storyActivityPresenter.handleOnCreate(storyId)
}

override fun routeToExploration(explorationId: String) {
startActivity(ExplorationActivity.createExplorationActivityIntent(this, explorationId))
}

companion object {
const val STORY_ACTIVITY_STORY_ID_ARGUMENT_KEY = "StoryActivity.story_id"
const val STORY_ACTIVITY_INTENT_EXTRA = "StoryActivity.story_id"

/** Returns a new [Intent] to route to [StoryActivity] for a specified story ID. */
fun createStoryActivityIntent(context: Context, storyId: String): Intent {
val intent = Intent(context, StoryActivity::class.java)
intent.putExtra(STORY_ACTIVITY_STORY_ID_ARGUMENT_KEY, storyId)
intent.putExtra(STORY_ACTIVITY_INTENT_EXTRA, storyId)
return intent
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,12 +8,12 @@ import javax.inject.Inject
/** The presenter for [StoryActivity]. */
@ActivityScope
class StoryActivityPresenter @Inject constructor(private val activity: AppCompatActivity) {
fun handleOnCreate() {
fun handleOnCreate(storyId: String) {
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
activity.setContentView(R.layout.story_activity)
if (getStoryFragment() == null) {
activity.supportFragmentManager.beginTransaction().add(
R.id.story_fragment_placeholder,
StoryFragment()
StoryFragment.newInstance(storyId)
).commitNow()
}
}
Expand Down
32 changes: 28 additions & 4 deletions app/src/main/java/org/oppia/app/story/StoryFragment.kt
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,21 @@ import android.view.ViewGroup
import org.oppia.app.fragment.InjectableFragment
import javax.inject.Inject

/** Fragment that displays story with chapter list. */
class StoryFragment : InjectableFragment() {
private const val KEY_STORY_ID_ARGUMENT = "STORY_ID"

/** Fragment for displaying a story. */
class StoryFragment : InjectableFragment(), ExplorationSelectionListener {
companion object {
/** Returns a new [StoryFragment] to display the story corresponding to the specified story ID. */
fun newInstance(storyId: String): StoryFragment {
val storyFragment = StoryFragment()
val args = Bundle()
args.putString(KEY_STORY_ID_ARGUMENT, storyId)
storyFragment.arguments = args
return storyFragment
}
}

@Inject
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
lateinit var storyFragmentPresenter: StoryFragmentPresenter

Expand All @@ -18,7 +31,18 @@ class StoryFragment : InjectableFragment() {
fragmentComponent.inject(this)
}

override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return storyFragmentPresenter.handleCreateView(inflater, container)
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
val args = checkNotNull(arguments) { "Expected arguments to be passed to StoryFragment" }
val storyId =
checkNotNull(args.getString(KEY_STORY_ID_ARGUMENT)) { "Expected storyId to be passed to StoryFragment" }
return storyFragmentPresenter.handleCreateView(inflater, container, storyId)
}

override fun selectExploration(explorationId: String) {
storyFragmentPresenter.handleSelectExploration(explorationId)
}
}
72 changes: 68 additions & 4 deletions app/src/main/java/org/oppia/app/story/StoryFragmentPresenter.kt
Original file line number Diff line number Diff line change
Expand Up @@ -3,21 +3,85 @@ package org.oppia.app.story
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.appcompat.app.AppCompatActivity
import androidx.fragment.app.Fragment
import androidx.lifecycle.Observer
import org.oppia.app.databinding.StoryChapterViewBinding
import org.oppia.app.databinding.StoryFragmentBinding
import org.oppia.app.fragment.FragmentScope
import org.oppia.app.databinding.StoryHeaderViewBinding
import org.oppia.app.home.RouteToExplorationListener
import org.oppia.app.recyclerview.BindableAdapter
import org.oppia.app.story.storyitemviewmodel.StoryChapterSummaryViewModel
import org.oppia.app.story.storyitemviewmodel.StoryHeaderViewModel
import org.oppia.app.story.storyitemviewmodel.StoryItemViewModel
import org.oppia.app.viewmodel.ViewModelProvider
import javax.inject.Inject

/** The presenter for [StoryFragment]. */
@FragmentScope
class StoryFragmentPresenter @Inject constructor(
private val fragment: Fragment
activity: AppCompatActivity,
private val fragment: Fragment,
private val viewModelProvider: ViewModelProvider<StoryViewModel>
) {
fun handleCreateView(inflater: LayoutInflater, container: ViewGroup?): View? {
private val routeToExplorationListener = activity as RouteToExplorationListener

fun handleCreateView(inflater: LayoutInflater, container: ViewGroup?, storyId: String): View? {
val viewModel = getStoryViewModel()
val binding = StoryFragmentBinding.inflate(inflater, container, /* attachToRoot= */ false)
viewModel.setStoryId(storyId)

viewModel.storyNameLiveData.observe(fragment, Observer<String> { storyName ->
(fragment.activity as? AppCompatActivity)?.supportActionBar?.title = storyName
})

binding.storyChapterList.apply {
adapter = createRecyclerViewAdapter()
}

// NB: Both the view model and lifecycle owner must be set in order to correctly bind LiveData elements to
// data-bound view models.
binding.let {
it.lifecycleOwner = fragment
it.viewModel = viewModel
}
return binding.root
}

fun handleSelectExploration(explorationId: String) {
routeToExplorationListener.routeToExploration(explorationId)
}

private fun createRecyclerViewAdapter(): BindableAdapter<StoryItemViewModel> {
return BindableAdapter.Builder
.newBuilder<StoryItemViewModel>()
.registerViewTypeComputer { viewModel ->
when (viewModel) {
is StoryHeaderViewModel -> ViewType.VIEW_TYPE_HEADER.ordinal
is StoryChapterSummaryViewModel -> ViewType.VIEW_TYPE_CHAPTER.ordinal
else -> throw IllegalArgumentException("Encountered unexpected view model: $viewModel")
}
}
.registerViewDataBinder(
viewType = ViewType.VIEW_TYPE_HEADER.ordinal,
inflateDataBinding = StoryHeaderViewBinding::inflate,
setViewModel = StoryHeaderViewBinding::setViewModel,
transformViewModel = { it as StoryHeaderViewModel }
)
.registerViewDataBinder(
viewType = ViewType.VIEW_TYPE_CHAPTER.ordinal,
inflateDataBinding = StoryChapterViewBinding::inflate,
setViewModel = StoryChapterViewBinding::setViewModel,
transformViewModel = { it as StoryChapterSummaryViewModel }
)
.build()
}

private fun getStoryViewModel(): StoryViewModel {
return viewModelProvider.getForFragment(fragment, StoryViewModel::class.java)
}

private enum class ViewType {
VIEW_TYPE_HEADER,
VIEW_TYPE_CHAPTER
}
}
76 changes: 76 additions & 0 deletions app/src/main/java/org/oppia/app/story/StoryViewModel.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
package org.oppia.app.story

import androidx.fragment.app.Fragment
import androidx.lifecycle.LiveData
import androidx.lifecycle.Transformations
import androidx.lifecycle.ViewModel
import org.oppia.app.fragment.FragmentScope
import org.oppia.app.model.ChapterPlayState
import org.oppia.app.model.ChapterSummary
import org.oppia.app.model.StorySummary
import org.oppia.app.story.storyitemviewmodel.StoryChapterSummaryViewModel
import org.oppia.app.story.storyitemviewmodel.StoryHeaderViewModel
import org.oppia.app.story.storyitemviewmodel.StoryItemViewModel
import org.oppia.domain.topic.TopicController
import org.oppia.util.data.AsyncResult
import org.oppia.util.logging.Logger
import javax.inject.Inject

/** The ViewModel for [StoryFragment]. */
@FragmentScope
class StoryViewModel @Inject constructor(
hareshkh marked this conversation as resolved.
Show resolved Hide resolved
fragment: Fragment,
private val topicController: TopicController,
private val logger: Logger
) : ViewModel() {
/** [storyId] needs to be set before any of the live data members can be accessed. */
private lateinit var storyId: String
private lateinit var fragment: StoryFragment
private val explorationSelectionListener = fragment as ExplorationSelectionListener

private val storyResultLiveData: LiveData<AsyncResult<StorySummary>> by lazy {
topicController.getStory(storyId)
}

private val storyLiveData: LiveData<StorySummary> by lazy {
Transformations.map(storyResultLiveData, ::processStoryResult)
}

val storyNameLiveData: LiveData<String> by lazy {
Transformations.map(storyLiveData, StorySummary::getStoryName)
}

val storyChapterLiveData: LiveData<List<StoryItemViewModel>> by lazy {
Transformations.map(storyLiveData, ::processStoryChapterList)
}

fun setStoryId(storyId: String) {
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
this.storyId = storyId
}

private fun processStoryResult(storyResult: AsyncResult<StorySummary>): StorySummary {
if (storyResult.isFailure()) {
logger.e("StoryFragment", "Failed to retrieve Story: ", storyResult.getErrorOrNull()!!)
}

return storyResult.getOrDefault(StorySummary.getDefaultInstance())
}

private fun processStoryChapterList(storySummary: StorySummary): List<StoryItemViewModel> {
val chapterList: List<ChapterSummary> = storySummary.chapterList
val completedCount =
chapterList.filter { chapter -> chapter.chapterPlayState == ChapterPlayState.COMPLETED }.size

// List with only the header
val itemViewModelList: MutableList<StoryItemViewModel> = mutableListOf(
StoryHeaderViewModel(completedCount, chapterList.size) as StoryItemViewModel
)

// Add the rest of the list
itemViewModelList.addAll(chapterList.map { chapter ->
StoryChapterSummaryViewModel(explorationSelectionListener, chapter) as StoryItemViewModel
})

return itemViewModelList
}
}
Loading