Skip to content

Commit

Permalink
Closes mozilla-mobile#1385: Determine and expose reader-ability state
Browse files Browse the repository at this point in the history
  • Loading branch information
csadilek committed Apr 26, 2019
1 parent 25dfcc3 commit 56a2987
Show file tree
Hide file tree
Showing 5 changed files with 254 additions and 25 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ class Session(
fun onMediaAdded(session: Session, media: List<Media>, added: Media) = Unit
fun onCrashStateChanged(session: Session, crashed: Boolean) = Unit
fun onIconChanged(session: Session, icon: Bitmap?) = Unit
fun onReaderableStateUpdated(session: Session, readerable: Boolean) = Unit
}

/**
Expand Down Expand Up @@ -371,6 +372,13 @@ class Session(
notifyObservers(old, new) { onCrashStateChanged(this@Session, new) }
}

/**
* Readerable state, whether or not the current page can be shown in a reader view.
*/
var readerable: Boolean by Delegates.observable(false) { _, _, new ->
notifyObservers { onReaderableStateUpdated(this@Session, new) }
}

/**
* Returns whether or not this session is used for a Custom Tab.
*/
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -609,6 +609,7 @@ class SessionTest {
defaultObserver.onMediaAdded(session, emptyList(), mock())
defaultObserver.onMediaRemoved(session, emptyList(), mock())
defaultObserver.onIconChanged(session, mock())
defaultObserver.onReaderableStateUpdated(session, true)
}

@Test
Expand Down Expand Up @@ -932,4 +933,29 @@ class SessionTest {

assertEquals(bitmapMock, notifiedIcon)
}

@Test
fun `observer is notified when readerable state updated`() {
val observer = mock(Session.Observer::class.java)

val session = Session("https://www.mozilla.org")
session.register(observer)
assertFalse(session.readerable)

session.readerable = true

verify(observer).onReaderableStateUpdated(
eq(session),
eq(true))

// We want to notify observers every time readerability is determined,
// not only when the state changed.
session.readerable = true

verify(observer, times(2)).onReaderableStateUpdated(
eq(session),
eq(true))

assertTrue(session.readerable)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,27 @@
/* Avoid adding ID selector rules in this style sheet, since they could
* inadvertently match elements in the article content. */

const supportedProtocols = ["http:", "https:"];

// Prevent false positives for these sites. This list is taken from Fennec:
// https://dxr.mozilla.org/mozilla-central/rev/7d47e7fa2489550ffa83aae67715c5497048923f/toolkit/components/reader/Readerable.js#45
const blockedHosts = ["amazon.com", "github.com", "mail.google.com", "pinterest.com", "reddit.com", "twitter.com", "youtube.com"];

class ReaderView {

static isReaderable() {
if (!supportedProtocols.includes(location.protocol)) {
return false;
}

if (blockedHosts.some(blockedHost => location.hostname.endsWith(blockedHost))) {
return false;
}

if (location.pathname == "/") {
return false;
}

return isProbablyReaderable(document);
}

Expand Down Expand Up @@ -261,15 +279,24 @@ class ReaderView {
}
}

// TODO remove (for testing purposes only)
let port = browser.runtime.connectNative("mozacReaderview");
port.postMessage(`Hello from ReaderView on page ${location.hostname}`);

port.onMessage.addListener((response) => {
console.log(`Received: ${JSON.stringify(response)} on page ${location.hostname}`);
port.onMessage.addListener((message) => {
switch (message.action) {
case 'show':
readerView.show({fontSize: 3, fontType: "serif", colorScheme: "light"});
break;
case 'hide':
readerView.hide();
break;
case 'checkReaderable':
port.postMessage({readerable: ReaderView.isReaderable()});
break;
default:
console.error(`Received invalid action ${message.action}`);
}
});


// TODO remove hostname check (for testing purposes only)
// e.g. https://blog.mozilla.org/firefox/reader-view
if (ReaderView.isReaderable() && location.hostname.endsWith("blog.mozilla.org")) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,8 +38,11 @@ typealias OnReaderViewAvailableChange = (available: Boolean) -> Unit
* @property sessionManager a reference to the application's [SessionManager].
* @property onReaderViewAvailableChange a callback invoked to indicate whether
* or not reader view is available for the page loaded by the currently selected
* (active) session.
* session. The callback will be invoked when a page is loaded or refreshed,
* on any navigation (back or forward), and when the selected session
* changes.
*/
@Suppress("TooManyFunctions")
class ReaderViewFeature(
private val context: Context,
private val engine: Engine,
Expand Down Expand Up @@ -93,6 +96,8 @@ class ReaderViewFeature(
ReaderViewFeature.install(engine)
}

checkReaderable()

controlsInteractor.start()
}

Expand All @@ -101,26 +106,24 @@ class ReaderViewFeature(
return
}

// TODO https://github.com/mozilla-mobile/android-components/issues/2624
val messageHandler = object : MessageHandler {
override fun onPortConnected(port: Port) {
ReaderViewFeature.ports[port.engineSession] = port
ports[port.engineSession] = port
checkReaderable()
}

override fun onPortDisconnected(port: Port) {
ReaderViewFeature.ports.remove(port.engineSession)
ports.remove(port.engineSession)
}

override fun onPortMessage(message: Any, port: Port) {
Logger.info("Received message: $message")
val response = JSONObject().apply {
put("response", "Message received! Hello from Android!")
if (message is JSONObject) {
activeSession?.readerable = message.optBoolean(READERABLE_RESPONSE_MESSAGE_KEY, false)
}
ReaderViewFeature.sendMessage(response, port.engineSession!!)
}
}

ReaderViewFeature.registerMessageHandler(sessionManager.getOrCreateEngineSession(session), messageHandler)
registerMessageHandler(sessionManager.getOrCreateEngineSession(session), messageHandler)
}

override fun stop() {
Expand All @@ -136,19 +139,32 @@ class ReaderViewFeature(
override fun onSessionSelected(session: Session) {
// TODO restore selected state of whether the controls are open or not
registerContentMessageHandler(activeSession)
checkReaderable()
super.onSessionSelected(session)
}

override fun onSessionRemoved(session: Session) {
ReaderViewFeature.ports.remove(sessionManager.getEngineSession(session))
ports.remove(sessionManager.getEngineSession(session))
}

override fun onUrlChanged(session: Session, url: String) {
checkReaderable()
}

override fun onReaderableStateUpdated(session: Session, readerable: Boolean) {
onReaderViewAvailableChange(readerable)
}

fun showReaderView() {
// TODO send message to show reader view (-> see ReaderView.show())
activeSession?.let {
sendMessage(JSONObject().put(ACTION_MESSAGE_KEY, ACTION_SHOW), it)
}
}

fun hideReaderView() {
// TODO send message to hide reader view (-> see ReaderView.hide())
activeSession?.let {
sendMessage(JSONObject().put(ACTION_MESSAGE_KEY, ACTION_HIDE), it)
}
}

/**
Expand All @@ -165,12 +181,43 @@ class ReaderViewFeature(
controlsPresenter.hide()
}

internal fun checkReaderable() {
activeSession?.let {
if (ports.containsKey(sessionManager.getEngineSession(it))) {
sendMessage(JSONObject().put(ACTION_MESSAGE_KEY, ACTION_CHECK_READERABLE), it)
}
}
}

private fun sendMessage(msg: Any, session: Session) {
val port = ports[sessionManager.getEngineSession(session)]
port?.postMessage(msg) ?: throw IllegalStateException("No port connected for the provided session")
}

companion object {
@VisibleForTesting(otherwise = VisibleForTesting.PRIVATE)
@VisibleForTesting
internal const val READER_VIEW_EXTENSION_ID = "mozacReaderview"

@VisibleForTesting
internal const val READER_VIEW_EXTENSION_URL = "resource://android/assets/extensions/readerview/"

@VisibleForTesting
internal const val ACTION_MESSAGE_KEY = "action"

@VisibleForTesting
internal const val ACTION_SHOW = "show"

@VisibleForTesting
internal const val ACTION_HIDE = "hide"

@VisibleForTesting
internal const val ACTION_CHECK_READERABLE = "checkReaderable"

@VisibleForTesting
internal const val READERABLE_RESPONSE_MESSAGE_KEY = "readerable"

@Volatile
@VisibleForTesting
internal var installedWebExt: WebExtension? = null

@Volatile
Expand Down Expand Up @@ -205,10 +252,5 @@ class ReaderViewFeature(

installedWebExt?.let { registerContentMessageHandler(it) }
}

fun sendMessage(msg: Any, engineSession: EngineSession) {
val port = ports[engineSession]
port?.postMessage(msg) ?: throw IllegalStateException("No port connected for the provided session")
}
}
}
Loading

0 comments on commit 56a2987

Please sign in to comment.