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 44 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 @@ -77,6 +77,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 @@ -87,6 +88,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
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,14 @@ import org.oppia.app.home.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.testing.BindableAdapterTestActivity
import org.oppia.app.testing.ContentCardTestActivity
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 @@ -25,7 +25,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 Down
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.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
8 changes: 8 additions & 0 deletions app/src/main/java/org/oppia/app/home/HomeFragmentPresenter.kt
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
package org.oppia.app.home

import android.content.Context
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
Expand All @@ -18,10 +19,12 @@ import org.oppia.app.home.topiclist.TopicSummaryViewModel
import org.oppia.app.model.TopicList
import org.oppia.app.model.TopicSummary
import org.oppia.app.model.UserAppHistory
import org.oppia.app.story.StoryActivity
import org.oppia.app.viewmodel.ViewModelProvider
import org.oppia.domain.UserAppHistoryController
import org.oppia.domain.exploration.ExplorationDataController
import org.oppia.domain.exploration.TEST_EXPLORATION_ID_5
import org.oppia.domain.topic.TEST_STORY_ID_1
import org.oppia.domain.topic.TopicListController
import org.oppia.util.data.AsyncResult
import org.oppia.util.logging.Logger
Expand Down Expand Up @@ -101,6 +104,11 @@ class HomeFragmentPresenter @Inject constructor(
})
}

// TODO(#134): Remove this method once it is possible to navigate to story activity in normal flow
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
fun openStory(v: View) {
hareshkh marked this conversation as resolved.
Show resolved Hide resolved
val intent = StoryActivity.createStoryActivityIntent(fragment.activity as Context, TEST_STORY_ID_1)
fragment.activity?.startActivity(intent)
}
private fun getUserAppHistoryViewModel(): UserAppHistoryViewModel {
return viewModelProvider.getForFragment(fragment, UserAppHistoryViewModel::class.java)
}
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
@@ -1,6 +1,7 @@
package org.oppia.app.recyclerview

import androidx.databinding.BindingAdapter
import androidx.databinding.ObservableList
import androidx.lifecycle.LiveData
import androidx.recyclerview.widget.RecyclerView

Expand All @@ -10,11 +11,18 @@ import androidx.recyclerview.widget.RecyclerView
* https://android.jlelse.eu/1bd08b4796b4.
*/
@BindingAdapter("data")
fun <T : Any> bindToRecyclerViewAdapter(recyclerView: RecyclerView, liveData: LiveData<List<T>>) {
fun <T : Any> bindToRecyclerViewAdapterWithLiveData(
recyclerView: RecyclerView,
liveData: LiveData<List<T>>
) {
liveData.value?.let { data ->
val adapter = recyclerView.adapter
checkNotNull(adapter) { "Cannot bind data to a RecyclerView missing its adapter." }
check(adapter is BindableAdapter<*>) { "Can only bind data to a BindableAdapter." }
adapter.setDataUnchecked(data)
bindToRecyclerViewAdapter(recyclerView, data)
}
}

private fun <T : Any> bindToRecyclerViewAdapter(recyclerView: RecyclerView, dataList: List<T>) {
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
val adapter = recyclerView.adapter
checkNotNull(adapter) { "Cannot bind data to a RecyclerView missing its adapter." }
check(adapter is BindableAdapter<*>) { "Can only bind data to a BindableAdapter." }
adapter.setDataUnchecked(dataList)
}
5 changes: 5 additions & 0 deletions app/src/main/java/org/oppia/app/story/ExplorationSelector.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
package org.oppia.app.story

interface ExplorationSelector {
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
fun selectExploration(explorationId: String)
}
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
15 changes: 13 additions & 2 deletions app/src/main/java/org/oppia/app/story/StoryActivity.kt
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,27 @@ 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? = intent.getStringExtra(STORY_ACTIVITY_STORY_ID_ARGUMENT_KEY)
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
storyActivityPresenter.handleOnCreate(storyId)
}

override fun onSupportNavigateUp(): Boolean {
return storyActivityPresenter.handleOnSupportNavigationUp()
}

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

companion object {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,16 +8,22 @@ import javax.inject.Inject
/** The presenter for [StoryActivity]. */
@ActivityScope
class StoryActivityPresenter @Inject constructor(private val activity: AppCompatActivity) {
fun handleOnCreate() {
fun handleOnCreate(storyId: String?) {
activity.setContentView(R.layout.story_activity)
activity.supportActionBar?.setDisplayHomeAsUpEnabled(true)
if (getStoryFragment() == null) {
activity.supportFragmentManager.beginTransaction().add(
R.id.story_fragment_placeholder,
StoryFragment()
StoryFragment.newInstance(storyId!!)
).commitNow()
}
}

fun handleOnSupportNavigationUp(): Boolean {
activity.finish()
return true
}

private fun getStoryFragment(): StoryFragment? {
return activity.supportFragmentManager.findFragmentById(R.id.story_fragment_placeholder) as StoryFragment?
}
Expand Down
31 changes: 27 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,20 @@ 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 = "STORY_ID"
BenHenning marked this conversation as resolved.
Show resolved Hide resolved

/** Fragment for displaying a story. */
class StoryFragment : InjectableFragment(), ExplorationSelector {
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, 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 +30,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)) { "Expected storyId to be passed to StoryFragment" }
return storyFragmentPresenter.handleCreateView(inflater, container, storyId)
}

override fun selectExploration(explorationId: String) {
storyFragmentPresenter.handleSelectExploration(explorationId)
}
}
75 changes: 70 additions & 5 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,86 @@ 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
/** The presented for [StoryFragment]. */
BenHenning marked this conversation as resolved.
Show resolved Hide resolved
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.setStoryFragment(fragment as StoryFragment)

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
}
}
Loading