Display search result items

Still plenty of TODO items, but making great progress

Make sure all the tests pass too
parser-fixes
Bradlee Speice 2015-02-26 23:53:20 -05:00
parent 71fb362ffe
commit e552d4d5a6
8 changed files with 193 additions and 20 deletions

View File

@ -3,7 +3,7 @@ package org.bspeice.minimalbible.activity.search;
import android.app.SearchManager;
import android.content.Intent;
import android.os.Bundle;
import android.widget.Toast;
import android.support.v7.widget.Toolbar;
import org.bspeice.minimalbible.Injector;
import org.bspeice.minimalbible.MinimalBible;
@ -16,6 +16,8 @@ import java.util.List;
import javax.inject.Inject;
import butterknife.ButterKnife;
import butterknife.InjectView;
import dagger.ObjectGraph;
@ -25,6 +27,12 @@ public class BasicSearch extends BaseActivity
@Inject
SearchProvider searchProvider;
@InjectView(R.id.content)
SearchResultsListView resultsView;
@InjectView(R.id.toolbar)
Toolbar toolbar;
private ObjectGraph searchObjGraph;
private void buildObjGraph() {
@ -50,11 +58,18 @@ public class BasicSearch extends BaseActivity
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_search_results);
inject(this);
ButterKnife.inject(this);
setSupportActionBar(toolbar);
// We don't set toolbar insets assuming that fitsSystemWindows="true"
// Also don't set "Up" enabled, back is enough.
handleSearch(getIntent());
}
// Used for launchMode="singleTop"
@Override
protected void onNewIntent(Intent intent) {
@ -67,8 +82,25 @@ public class BasicSearch extends BaseActivity
return;
String query = intent.getStringExtra(SearchManager.QUERY);
Toast.makeText(this, "Searching for: " + query, Toast.LENGTH_SHORT).show();
List<Verse> results = searchProvider.basicTextSearch(query);
Toast.makeText(this, "Found results: " + results.size(), Toast.LENGTH_SHORT).show();
displayTitle(query, results.size());
displaySearch(results);
}
public void displayTitle(String query, Integer resultsSize) {
// We can't go through the actual `toolbar` object, we have to
// getSupportActionBar() first.
// http://stackoverflow.com/a/26506858/1454178
getSupportActionBar().setTitle(buildTitle(query, resultsSize));
}
public String buildTitle(String query, Integer resultsSize) {
return "\"" + query + "\" - " + resultsSize + " results";
}
// TODO: Inject the book into BasicSearch instead of pulling it out of searchProvider?
public void displaySearch(List<Verse> searchResults) {
resultsView.initialize(searchProvider.getBook(), searchResults);
}
}

View File

@ -10,22 +10,22 @@ import org.crosswire.jsword.index.IndexManager
* This is the entry point for handling the actual bible search. Likely will support
* an "advanced" search in the future, but for now, basicTextSearch is what you get.
*/
class SearchProvider(val indexManager: IndexManager, val b: Book?) {
class SearchProvider(val indexManager: IndexManager, val book: Book?) {
val defaultSearchType = SearchType.ANY_WORDS
[suppress("UNUSED_PARAMETER")]
public fun basicTextSearch(text: String): List<Verse> {
if (!isSearchAvailable()) {
Log.w("SearchProvider", "Search unavailable, index status of ${b?.getInitials()}: ${b?.getIndexStatus()}")
Log.w("SearchProvider", "Search unavailable, index status of ${book?.getInitials()}: ${book?.getIndexStatus()}")
return listOf()
}
val searchText = defaultSearchType decorate text
// We already checked for null in isSearchAvailable(), but Kotlin
// doesn't keep track of that (yet)
val results = b!!.find(searchText)
return results map { it as Verse }
return book!!.find(searchText)
.map { it as Verse }
}
/**
@ -34,7 +34,7 @@ class SearchProvider(val indexManager: IndexManager, val b: Book?) {
* This check MUST guarantee that the book is not null.
*/
public fun isSearchAvailable(): Boolean =
b != null &&
indexManager isIndexed b
book != null &&
indexManager isIndexed book
}

View File

@ -0,0 +1,77 @@
package org.bspeice.minimalbible.activity.search
import android.content.Context
import android.util.AttributeSet
import android.widget.LinearLayout
import android.support.v7.widget.RecyclerView
import android.support.v7.widget.LinearLayoutManager
import android.view.LayoutInflater
import org.bspeice.minimalbible.R
import org.crosswire.jsword.passage.Verse
import android.widget.TextView
import org.bspeice.minimalbible.service.format.osisparser.OsisParser
import org.crosswire.jsword.book.Book
import android.view.ViewGroup
/**
* Created by bspeice on 2/26/15.
*/
class SearchResultsListView(val ctx: Context, val attrs: AttributeSet) : LinearLayout(ctx, attrs) {
val layoutManager = LinearLayoutManager(ctx)
val inflater = ctx.getSystemService(Context.LAYOUT_INFLATER_SERVICE) as LayoutInflater
val contentView = inflater.inflate(R.layout.view_search_results_list, this, true)
val searchResults = contentView.findViewById(R.id.search_results) as RecyclerView;
{
searchResults setLayoutManager layoutManager
}
fun initialize(b: Book, resultsList: List<Verse>) {
searchResults.setAdapter(SearchResultsAdapter(b, resultsList))
}
}
// TODO: Handle clicking an item and navigating on the main screen
class SearchResultsAdapter(val b: Book, val results: List<Verse>)
: RecyclerView.Adapter<ResultViewHolder>() {
override fun onCreateViewHolder(parent: ViewGroup?, viewType: Int): ResultViewHolder? {
val resultView = SearchResultView(parent)
return ResultViewHolder(resultView)
}
override fun onBindViewHolder(holder: ResultViewHolder?, position: Int) {
holder?.bind(b, results[position])
}
override fun getItemCount(): Int = results.size()
}
/**
* The ViewHolder object for an individual search result
* TODO: Bold the text found in the query
*/
class ResultViewHolder(val view: SearchResultView) : RecyclerView.ViewHolder(view.contentView) {
// TODO: Need a nicer way of displaying the book name - currently is ALL CAPS
fun buildVerseName(v: Verse) = "${v.getBook().name()} ${v.getChapter()}:${v.getVerse()}"
fun buildVerseContent(b: Book, v: Verse, o: OsisParser) = o.parseVerse(b, v)
fun bind(b: Book, verse: Verse) {
view.verseName setText buildVerseName(verse)
view.verseContent setText buildVerseContent(b, verse, OsisParser())
}
}
/**
* A custom view to wrap showing a search result
*/
class SearchResultView(val group: ViewGroup?) {
val inflater = LayoutInflater.from(group?.getContext())
val contentView = inflater.inflate(R.layout.view_search_result, group, false)
val verseName = contentView.findViewById(R.id.verseName) as TextView
val verseContent = contentView.findViewById(R.id.verseContent) as TextView
}

View File

@ -13,13 +13,13 @@ import org.bspeice.minimalbible.activity.viewer.BookAdapter.ChapterInfo
import rx.subjects.PublishSubject
import android.text.SpannableStringBuilder
import android.util.TypedValue
import org.bspeice.minimalbible.service.format.osisparser.OsisParser
import android.util.Log
import android.content.Context
import android.util.AttributeSet
import android.support.v7.widget.LinearLayoutManager
import android.widget.LinearLayout
import android.view.View
import org.bspeice.minimalbible.service.format.osisparser.OsisParser
class BibleView(val ctx: Context, val attrs: AttributeSet) : LinearLayout(ctx, attrs) {
@ -120,7 +120,7 @@ class BookAdapter(val b: Book, val prefs: BibleViewerPreferences)
override fun getItemCount(): Int = chapterList.size()
public fun bindScrollHandler(provider: PublishSubject<BookScrollEvent>,
lM: RecyclerView.LayoutManager) {
lM: RecyclerView.LayoutManager) {
provider subscribe {
val event = it
lM scrollToPosition
@ -145,7 +145,8 @@ class PassageView(val v: TextView, val b: Book)
fun getAllVerses(verses: Progression<Int>, info: ChapterInfo): SpannableStringBuilder {
val builder = SpannableStringBuilder()
verses.forEach { OsisParser(builder).appendVerse(b, buildOrdinal(it, info)) }
val parser = OsisParser()
verses.forEach { parser.appendVerse(b, buildOrdinal(it, info), builder) }
return builder
}

View File

@ -16,23 +16,41 @@ import org.bspeice.minimalbible.service.format.osisparser.handler.DivineHandler
/**
* Parse out the OSIS XML into whatever we want!
* This takes in a SpannableStringBuilder to modify. Normally I'm not a fan
* of mutability, but due to the need for absolute efficiency in this class,
* that's what we're going with.
* TODO: Speed up parsing. This is the single most expensive repeated operation
*/
class OsisParser(val builder: SpannableStringBuilder) : DefaultHandler() {
class OsisParser() : DefaultHandler() {
// Don't pass a verse as part of the constructor, but still guarantee
// that it will exist
var verseContent: VerseContent by Delegates.notNull()
var builder: SpannableStringBuilder by Delegates.notNull()
// TODO: Implement a stack to keep min API 8
val handlerStack = ArrayDeque<TagHandler>()
fun appendVerse(b: Book, v: Verse): VerseContent {
fun appendVerse(b: Book, v: Verse,
builder: SpannableStringBuilder): VerseContent {
verseContent = VerseContent(v)
this.builder = builder
BookData(b, v).getSAXEventProvider() provideSAXEvents this
return verseContent
}
/**
* Parse a verse and return its content
* Only good for parsing a single verse at a time,
* but gives a cleaner API to work with (and means that
* we can just use the default constructor)
*/
fun parseVerse(b: Book, v: Verse): SpannableStringBuilder {
val mBuilder = SpannableStringBuilder()
appendVerse(b, v, mBuilder)
return mBuilder
}
override fun startElement(uri: String, localName: String,
qName: String, attributes: Attributes) {
when (localName) {

View File

@ -1,11 +1,26 @@
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
<!-- Set fitsSystemWindows to true since we don't want
the search results being overlaid by the nav bar -->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
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:fitsSystemWindows="true"
android:orientation="vertical"
tools:context="org.bspeice.minimalbible.activity.search.BasicSearch">
</RelativeLayout>
<android.support.v7.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/colorPrimary"
android:minHeight="?attr/actionBarSize" />
<!-- TODO: Add a divider between results -->
<!-- TODO: Add some sort of padding/margin -->
<org.bspeice.minimalbible.activity.search.SearchResultsListView
android:id="@+id/content"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</LinearLayout>

View File

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="?android:attr/listPreferredItemHeight"
android:orientation="vertical">
<TextView
android:id="@+id/verseName"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textAppearance="?android:attr/textAppearanceMedium" />
<TextView
android:id="@+id/verseContent"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:ellipsize="end"
android:textAppearance="?android:attr/textAppearanceSmall" />
</LinearLayout>

View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.RecyclerView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/search_results"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</LinearLayout>