2015-01-14 12:12:47 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2015 Synced Synapse. All rights reserved.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2015-03-09 22:35:18 +01:00
|
|
|
package org.xbmc.kore.ui;
|
2015-01-14 12:12:47 +01:00
|
|
|
|
|
|
|
import android.content.Intent;
|
|
|
|
import android.graphics.Point;
|
2015-04-08 23:28:19 +02:00
|
|
|
import android.net.Uri;
|
2015-01-14 12:12:47 +01:00
|
|
|
import android.os.Bundle;
|
2015-01-23 23:23:59 +01:00
|
|
|
import android.os.Handler;
|
2015-01-14 12:12:47 +01:00
|
|
|
import android.preference.PreferenceManager;
|
|
|
|
import android.support.v4.view.ViewPager;
|
|
|
|
import android.support.v4.widget.DrawerLayout;
|
|
|
|
import android.support.v7.widget.Toolbar;
|
|
|
|
import android.text.TextUtils;
|
2015-02-14 06:26:19 +01:00
|
|
|
import android.view.KeyEvent;
|
2015-01-14 12:12:47 +01:00
|
|
|
import android.view.Menu;
|
|
|
|
import android.view.MenuItem;
|
|
|
|
import android.view.ViewTreeObserver;
|
|
|
|
import android.widget.ImageView;
|
|
|
|
import android.widget.Toast;
|
|
|
|
|
2015-03-09 22:35:18 +01:00
|
|
|
import org.xbmc.kore.R;
|
|
|
|
import org.xbmc.kore.Settings;
|
|
|
|
import org.xbmc.kore.host.HostConnectionObserver;
|
|
|
|
import org.xbmc.kore.host.HostManager;
|
|
|
|
import org.xbmc.kore.jsonrpc.ApiCallback;
|
2015-04-10 20:51:44 +02:00
|
|
|
import org.xbmc.kore.jsonrpc.HostConnection;
|
2015-03-09 22:35:18 +01:00
|
|
|
import org.xbmc.kore.jsonrpc.method.Application;
|
|
|
|
import org.xbmc.kore.jsonrpc.method.AudioLibrary;
|
2015-03-20 00:00:31 +01:00
|
|
|
import org.xbmc.kore.jsonrpc.method.GUI;
|
2015-03-09 22:35:18 +01:00
|
|
|
import org.xbmc.kore.jsonrpc.method.Input;
|
2015-04-08 23:28:19 +02:00
|
|
|
import org.xbmc.kore.jsonrpc.method.Player;
|
|
|
|
import org.xbmc.kore.jsonrpc.method.Playlist;
|
2015-03-09 22:35:18 +01:00
|
|
|
import org.xbmc.kore.jsonrpc.method.System;
|
|
|
|
import org.xbmc.kore.jsonrpc.method.VideoLibrary;
|
|
|
|
import org.xbmc.kore.jsonrpc.type.GlobalType;
|
|
|
|
import org.xbmc.kore.jsonrpc.type.ListType;
|
|
|
|
import org.xbmc.kore.jsonrpc.type.PlayerType;
|
2015-04-08 23:28:19 +02:00
|
|
|
import org.xbmc.kore.jsonrpc.type.PlaylistType;
|
2015-03-09 22:35:18 +01:00
|
|
|
import org.xbmc.kore.service.NotificationService;
|
|
|
|
import org.xbmc.kore.ui.hosts.AddHostActivity;
|
|
|
|
import org.xbmc.kore.ui.views.CirclePageIndicator;
|
|
|
|
import org.xbmc.kore.utils.LogUtils;
|
|
|
|
import org.xbmc.kore.utils.TabsAdapter;
|
|
|
|
import org.xbmc.kore.utils.UIUtils;
|
2015-04-08 23:28:19 +02:00
|
|
|
|
|
|
|
import java.net.MalformedURLException;
|
|
|
|
import java.net.URL;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
2015-01-14 12:12:47 +01:00
|
|
|
|
|
|
|
import butterknife.ButterKnife;
|
|
|
|
import butterknife.InjectView;
|
|
|
|
|
|
|
|
|
2015-02-11 00:17:16 +01:00
|
|
|
public class RemoteActivity extends BaseActivity
|
2015-01-14 12:12:47 +01:00
|
|
|
implements HostConnectionObserver.PlayerEventsObserver,
|
|
|
|
NowPlayingFragment.NowPlayingListener,
|
|
|
|
SendTextDialogFragment.SendTextDialogListener {
|
|
|
|
private static final String TAG = LogUtils.makeLogTag(RemoteActivity.class);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Host manager singleton
|
|
|
|
*/
|
|
|
|
private HostManager hostManager = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To register for observing host events
|
|
|
|
*/
|
|
|
|
private HostConnectionObserver hostConnectionObserver;
|
|
|
|
|
|
|
|
private NavigationDrawerFragment navigationDrawerFragment;
|
|
|
|
|
|
|
|
@InjectView(R.id.background_image) ImageView backgroundImage;
|
|
|
|
@InjectView(R.id.pager_indicator) CirclePageIndicator pageIndicator;
|
|
|
|
@InjectView(R.id.pager) ViewPager viewPager;
|
|
|
|
@InjectView(R.id.default_toolbar) Toolbar toolbar;
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onCreate(Bundle savedInstanceState) {
|
|
|
|
super.onCreate(savedInstanceState);
|
|
|
|
|
|
|
|
// Set default values for the preferences
|
|
|
|
PreferenceManager.setDefaultValues(this, R.xml.preferences, false);
|
|
|
|
|
|
|
|
setContentView(R.layout.activity_remote);
|
|
|
|
ButterKnife.inject(this);
|
|
|
|
|
|
|
|
hostManager = HostManager.getInstance(this);
|
|
|
|
|
|
|
|
// Check if we have any hosts setup
|
|
|
|
if (hostManager.getHostInfo() == null) {
|
|
|
|
final Intent intent = new Intent(this, AddHostActivity.class);
|
|
|
|
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
|
|
|
|
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
|
|
|
startActivity(intent);
|
|
|
|
finish();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set up the drawer.
|
|
|
|
navigationDrawerFragment = (NavigationDrawerFragment) getSupportFragmentManager()
|
|
|
|
.findFragmentById(R.id.navigation_drawer);
|
|
|
|
navigationDrawerFragment.setUp(R.id.navigation_drawer, (DrawerLayout) findViewById(R.id.drawer_layout));
|
|
|
|
|
|
|
|
// Set up pager and fragments
|
|
|
|
TabsAdapter tabsAdapter = new TabsAdapter(this, getSupportFragmentManager())
|
|
|
|
.addTab(NowPlayingFragment.class, null, R.string.now_playing, 1)
|
|
|
|
.addTab(RemoteFragment.class, null, R.string.remote, 2)
|
|
|
|
.addTab(PlaylistFragment.class, null, R.string.playlist, 3);
|
|
|
|
|
|
|
|
viewPager.setAdapter(tabsAdapter);
|
|
|
|
pageIndicator.setViewPager(viewPager);
|
|
|
|
pageIndicator.setOnPageChangeListener(defaultOnPageChangeListener);
|
|
|
|
|
|
|
|
viewPager.setCurrentItem(1);
|
|
|
|
viewPager.setOffscreenPageLimit(2);
|
|
|
|
|
|
|
|
setupActionBar();
|
|
|
|
|
2015-04-08 23:28:19 +02:00
|
|
|
// If we should start playing something
|
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
// // Setup system bars and content padding
|
|
|
|
// setupSystemBarsColors();
|
|
|
|
// // Set the padding of views.
|
|
|
|
// // Only set top and right, to allow bottom to overlap in each fragment
|
|
|
|
// UIUtils.setPaddingForSystemBars(this, viewPager, true, true, false);
|
|
|
|
// UIUtils.setPaddingForSystemBars(this, pageIndicator, true, true, false);
|
|
|
|
}
|
|
|
|
|
2015-05-21 21:08:05 +02:00
|
|
|
@Override
|
|
|
|
public void onStart() {
|
|
|
|
super.onStart();
|
|
|
|
handleStartIntent(getIntent());
|
|
|
|
}
|
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
@Override
|
|
|
|
public void onResume() {
|
|
|
|
super.onResume();
|
|
|
|
hostConnectionObserver = hostManager.getHostConnectionObserver();
|
2015-02-15 20:11:32 +01:00
|
|
|
hostConnectionObserver.registerPlayerObserver(this, true);
|
|
|
|
// Force a refresh, mainly to update the time elapsed on the fragments
|
|
|
|
hostConnectionObserver.forceRefreshResults();
|
2015-01-14 12:12:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPause() {
|
|
|
|
super.onPause();
|
|
|
|
hostConnectionObserver.unregisterPlayerObserver(this);
|
|
|
|
hostConnectionObserver = null;
|
|
|
|
}
|
|
|
|
|
2015-02-14 06:26:19 +01:00
|
|
|
/**
|
|
|
|
* Override hardware volume keys and send to Kodi
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public boolean dispatchKeyEvent(KeyEvent event) {
|
2015-02-15 20:34:56 +01:00
|
|
|
// Check whether we should intercept this
|
|
|
|
boolean useVolumeKeys = PreferenceManager
|
|
|
|
.getDefaultSharedPreferences(this)
|
|
|
|
.getBoolean(Settings.KEY_PREF_USE_HARDWARE_VOLUME_KEYS,
|
|
|
|
Settings.DEFAULT_PREF_USE_HARDWARE_VOLUME_KEYS);
|
|
|
|
if (useVolumeKeys) {
|
|
|
|
int action = event.getAction();
|
|
|
|
int keyCode = event.getKeyCode();
|
|
|
|
switch (keyCode) {
|
|
|
|
case KeyEvent.KEYCODE_VOLUME_UP:
|
|
|
|
if (action == KeyEvent.ACTION_DOWN) {
|
2015-04-08 23:28:19 +02:00
|
|
|
new Application
|
|
|
|
.SetVolume(GlobalType.IncrementDecrement.INCREMENT)
|
|
|
|
.execute(hostManager.getConnection(), null, null);
|
2015-02-15 20:34:56 +01:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
case KeyEvent.KEYCODE_VOLUME_DOWN:
|
|
|
|
if (action == KeyEvent.ACTION_DOWN) {
|
2015-04-08 23:28:19 +02:00
|
|
|
new Application
|
|
|
|
.SetVolume(GlobalType.IncrementDecrement.DECREMENT)
|
|
|
|
.execute(hostManager.getConnection(), null, null);
|
2015-02-15 20:34:56 +01:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2015-02-14 06:26:19 +01:00
|
|
|
}
|
2015-02-15 20:34:56 +01:00
|
|
|
|
|
|
|
return super.dispatchKeyEvent(event);
|
2015-02-14 06:26:19 +01:00
|
|
|
}
|
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
@Override
|
|
|
|
public boolean onCreateOptionsMenu(Menu menu) {
|
|
|
|
if (!navigationDrawerFragment.isDrawerOpen()) {
|
|
|
|
// Only show items in the action bar relevant to this screen if the drawer is not showing.
|
|
|
|
// Otherwise, let the drawer decide what to show in the action bar.
|
|
|
|
getMenuInflater().inflate(R.menu.remote, menu);
|
|
|
|
}
|
|
|
|
return super.onCreateOptionsMenu(menu);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean onOptionsItemSelected(MenuItem item) {
|
|
|
|
// Handle action bar item clicks here.
|
|
|
|
switch (item.getItemId()) {
|
|
|
|
case R.id.action_wake_up:
|
|
|
|
UIUtils.sendWolAsync(this, hostManager.getHostInfo());
|
|
|
|
return true;
|
|
|
|
case R.id.action_quit:
|
|
|
|
Application.Quit actionQuit = new Application.Quit();
|
|
|
|
// Fire and forget
|
|
|
|
actionQuit.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
|
|
|
case R.id.action_suspend:
|
|
|
|
System.Suspend actionSuspend = new System.Suspend();
|
|
|
|
// Fire and forget
|
|
|
|
actionSuspend.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
2015-03-29 13:48:51 +02:00
|
|
|
case R.id.action_reboot:
|
|
|
|
System.Reboot actionReboot = new System.Reboot();
|
|
|
|
// Fire and forget
|
|
|
|
actionReboot.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
2015-01-14 12:12:47 +01:00
|
|
|
case R.id.action_shutdown:
|
|
|
|
System.Shutdown actionShutdown = new System.Shutdown();
|
|
|
|
// Fire and forget
|
|
|
|
actionShutdown.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
|
|
|
case R.id.send_text:
|
|
|
|
SendTextDialogFragment dialog =
|
|
|
|
SendTextDialogFragment.newInstance(getString(R.string.send_text));
|
|
|
|
dialog.show(getSupportFragmentManager(), null);
|
2015-01-23 23:23:59 +01:00
|
|
|
return true;
|
2015-01-25 15:17:22 +01:00
|
|
|
case R.id.toggle_fullscreen:
|
2015-03-20 00:00:31 +01:00
|
|
|
GUI.SetFullscreen actionSetFullscreen = new GUI.SetFullscreen();
|
|
|
|
// Input.ExecuteAction actionSetFullscreen = new Input.ExecuteAction(Input.ExecuteAction.TOGGLEFULLSCREEN);
|
2015-01-25 15:17:22 +01:00
|
|
|
actionSetFullscreen.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
2015-01-23 23:23:59 +01:00
|
|
|
case R.id.clean_video_library:
|
|
|
|
VideoLibrary.Clean actionCleanVideo = new VideoLibrary.Clean();
|
|
|
|
actionCleanVideo.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
|
|
|
case R.id.clean_audio_library:
|
|
|
|
AudioLibrary.Clean actionCleanAudio = new AudioLibrary.Clean();
|
|
|
|
actionCleanAudio.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
|
|
|
case R.id.update_video_library:
|
|
|
|
VideoLibrary.Scan actionScanVideo = new VideoLibrary.Scan();
|
|
|
|
actionScanVideo.execute(hostManager.getConnection(), null, null);
|
|
|
|
return true;
|
|
|
|
case R.id.update_audio_library:
|
|
|
|
AudioLibrary.Scan actionScanAudio = new AudioLibrary.Scan();
|
|
|
|
actionScanAudio.execute(hostManager.getConnection(), null, null);
|
2015-01-14 12:12:47 +01:00
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return super.onOptionsItemSelected(item);
|
|
|
|
}
|
|
|
|
|
2015-01-23 23:23:59 +01:00
|
|
|
/**
|
|
|
|
* Issue commands to update the Audio and Video libraries, sequentially
|
|
|
|
*/
|
|
|
|
private void updateLibraries() {
|
|
|
|
final Handler callbackHandler = new Handler();
|
|
|
|
VideoLibrary.Scan actionScanVideo = new VideoLibrary.Scan();
|
|
|
|
actionScanVideo.execute(hostManager.getConnection(), new ApiCallback<String>() {
|
|
|
|
@Override
|
2015-03-16 19:33:40 +01:00
|
|
|
public void onSuccess(String result) {
|
2015-01-23 23:23:59 +01:00
|
|
|
// Great, now update the Audio library
|
|
|
|
AudioLibrary.Scan actionScanAudio = new AudioLibrary.Scan();
|
|
|
|
actionScanAudio.execute(hostManager.getConnection(), null, callbackHandler);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onError(int errorCode, String description) { }
|
|
|
|
}, callbackHandler);
|
|
|
|
}
|
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
/**
|
|
|
|
* Callbacks from Send text dialog
|
|
|
|
*/
|
|
|
|
public void onSendTextFinished(String text, boolean done) {
|
|
|
|
Input.SendText action = new Input.SendText(text, done);
|
|
|
|
action.execute(hostManager.getConnection(), null, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void onSendTextCancel() {
|
|
|
|
// Nothing to do
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void setupActionBar() {
|
|
|
|
setToolbarTitle(toolbar, 1);
|
|
|
|
setSupportActionBar(toolbar);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setToolbarTitle(Toolbar toolbar, int position) {
|
|
|
|
if (toolbar != null) {
|
|
|
|
switch (position) {
|
|
|
|
case 0:
|
|
|
|
toolbar.setTitle(R.string.now_playing);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
toolbar.setTitle(R.string.remote);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
toolbar.setTitle(R.string.playlist);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-08 23:28:19 +02:00
|
|
|
/**
|
|
|
|
* Handles the intent that started this activity, namely to start playing something on Kodi
|
|
|
|
* @param intent Start intent for the activity
|
|
|
|
*/
|
|
|
|
private void handleStartIntent(Intent intent) {
|
|
|
|
final String action = intent.getAction();
|
|
|
|
// Check action
|
2015-04-18 19:23:31 +02:00
|
|
|
if ((action == null) ||
|
|
|
|
!(action.equals(Intent.ACTION_SEND) || action.equals(Intent.ACTION_VIEW)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
Uri youTubeUri = null;
|
|
|
|
if (action.equals(Intent.ACTION_SEND)) {
|
|
|
|
// Get the URI, which is stored in Extras
|
|
|
|
youTubeUri = getYouTubeUri(intent.getStringExtra(Intent.EXTRA_TEXT));
|
|
|
|
if (youTubeUri == null) return;
|
|
|
|
} else if (action.equals(Intent.ACTION_VIEW)) {
|
|
|
|
if (intent.getData() == null) return;
|
|
|
|
youTubeUri = Uri.parse(intent.getData().toString());
|
|
|
|
}
|
2015-04-08 23:28:19 +02:00
|
|
|
|
|
|
|
final String videoId = getYouTubeVideoId(youTubeUri);
|
|
|
|
if (videoId == null) return;
|
|
|
|
|
2015-05-21 21:08:05 +02:00
|
|
|
// final String kodiAddonUrl = "plugin://plugin.video.youtube/?path=/root/search&action=play_video&videoid="
|
|
|
|
// + videoId;
|
|
|
|
final String kodiAddonUrl = "plugin://plugin.video.youtube/play/?video_id=" + videoId;
|
2015-04-08 23:28:19 +02:00
|
|
|
|
2015-04-10 20:51:44 +02:00
|
|
|
// Check if any video player is active and play or queue the file depending on that
|
|
|
|
final HostConnection connection = hostManager.getConnection();
|
|
|
|
final Handler callbackHandler = new Handler();
|
|
|
|
Player.GetActivePlayers getActivePlayers = new Player.GetActivePlayers();
|
|
|
|
getActivePlayers.execute(connection, new ApiCallback<ArrayList<PlayerType.GetActivePlayersReturnType>>() {
|
|
|
|
@Override
|
|
|
|
public void onSuccess(ArrayList<PlayerType.GetActivePlayersReturnType> result) {
|
|
|
|
boolean videoIsPlaying = false;
|
|
|
|
|
|
|
|
for (PlayerType.GetActivePlayersReturnType player : result) {
|
|
|
|
if (player.type.equals(PlayerType.GetActivePlayersReturnType.VIDEO))
|
|
|
|
videoIsPlaying = true;
|
|
|
|
}
|
|
|
|
|
2015-05-21 21:08:05 +02:00
|
|
|
queueMediaFile(kodiAddonUrl, !videoIsPlaying, connection, callbackHandler);
|
2015-04-10 20:51:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onError(int errorCode, String description) {
|
|
|
|
LogUtils.LOGD(TAG, "Couldn't get active player when handling start intent.");
|
|
|
|
Toast.makeText(RemoteActivity.this,
|
|
|
|
String.format(getString(R.string.error_get_active_player), description),
|
|
|
|
Toast.LENGTH_SHORT).show();
|
|
|
|
}
|
|
|
|
}, callbackHandler);
|
2015-04-08 23:28:19 +02:00
|
|
|
}
|
|
|
|
|
2015-04-10 20:51:44 +02:00
|
|
|
/**
|
2015-05-21 21:08:05 +02:00
|
|
|
* Queues the given media file and optionally starts the playlist
|
2015-04-10 20:51:44 +02:00
|
|
|
* @param file File to play
|
2015-05-21 21:08:05 +02:00
|
|
|
* @param startPlaylist Whether to start playing the playlist after add
|
2015-04-10 20:51:44 +02:00
|
|
|
* @param connection Host connection
|
|
|
|
* @param callbackHandler Handler to use for posting callbacks
|
|
|
|
*/
|
2015-05-21 21:08:05 +02:00
|
|
|
private void queueMediaFile(final String file, final boolean startPlaylist,
|
|
|
|
final HostConnection connection, final Handler callbackHandler) {
|
2015-04-08 23:28:19 +02:00
|
|
|
PlaylistType.Item item = new PlaylistType.Item();
|
|
|
|
item.file = file;
|
2015-04-10 20:51:44 +02:00
|
|
|
Playlist.Add action = new Playlist.Add(PlaylistType.VIDEO_PLAYLISTID, item);
|
|
|
|
action.execute(connection, new ApiCallback<String>() {
|
2015-04-08 23:28:19 +02:00
|
|
|
@Override
|
|
|
|
public void onSuccess(String result ) {
|
2015-05-21 21:08:05 +02:00
|
|
|
if (startPlaylist) {
|
|
|
|
Player.Open action = new Player.Open(PlaylistType.VIDEO_PLAYLISTID);
|
|
|
|
action.execute(connection, new ApiCallback<String>() {
|
|
|
|
@Override
|
|
|
|
public void onSuccess(String result) {
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onError(int errorCode, String description) {
|
|
|
|
Toast.makeText(RemoteActivity.this,
|
|
|
|
String.format(getString(R.string.error_play_media_file), description),
|
|
|
|
Toast.LENGTH_SHORT).show();
|
|
|
|
}
|
|
|
|
}, callbackHandler);
|
|
|
|
}
|
|
|
|
|
2015-04-10 20:51:44 +02:00
|
|
|
// Force a refresh of the playlist fragment
|
|
|
|
String tag = "android:switcher:" + viewPager.getId() + ":" + 3;
|
|
|
|
PlaylistFragment playlistFragment = (PlaylistFragment)getSupportFragmentManager()
|
|
|
|
.findFragmentByTag(tag);
|
|
|
|
if (playlistFragment != null) {
|
|
|
|
playlistFragment.forceRefreshPlaylist();
|
|
|
|
}
|
2015-04-08 23:28:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onError(int errorCode, String description) {
|
|
|
|
Toast.makeText(RemoteActivity.this,
|
|
|
|
String.format(getString(R.string.error_queue_media_file), description),
|
|
|
|
Toast.LENGTH_SHORT).show();
|
|
|
|
}
|
|
|
|
}, callbackHandler);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the YouTube Uri that the YouTube app passes in EXTRA_TEXT
|
|
|
|
* YouTube sends something like: [Video title]: [YouTube URL] so we need
|
|
|
|
* to get the second part
|
|
|
|
*
|
|
|
|
* @param extraText EXTRA_TEXT passed in the intent
|
|
|
|
* @return Uri present in extraText if present
|
|
|
|
*/
|
|
|
|
private Uri getYouTubeUri(String extraText) {
|
|
|
|
if (extraText == null) return null;
|
|
|
|
|
|
|
|
for (String word : extraText.split(" ")) {
|
|
|
|
if (word.startsWith("http://") || word.startsWith("https://")) {
|
|
|
|
try {
|
|
|
|
URL validUri = new URL(word);
|
|
|
|
return Uri.parse(word);
|
|
|
|
} catch (MalformedURLException exc) {
|
|
|
|
LogUtils.LOGD(TAG, "Got a malformed URL in an intent: " + word);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the youtube video ID from its URL
|
|
|
|
*
|
|
|
|
* @param playuri Youtube URL
|
|
|
|
* @return Youtube Video ID
|
|
|
|
*/
|
|
|
|
private String getYouTubeVideoId(Uri playuri) {
|
|
|
|
if (playuri.getHost().endsWith("youtube.com") || playuri.getHost().endsWith("youtu.be")) {
|
|
|
|
// We'll need to get the v= parameter from the URL
|
|
|
|
final Pattern pattern =
|
2015-05-21 21:08:05 +02:00
|
|
|
Pattern.compile("(?:https?:\\/\\/)?(?:www\\.|m\\.)?youtu(?:.be\\/|be\\.com\\/watch\\?v=)([\\w-]+)",
|
2015-04-08 23:28:19 +02:00
|
|
|
Pattern.CASE_INSENSITIVE);
|
|
|
|
// final Pattern pattern = Pattern.compile("^http(:?s)?:\\/\\/(?:www\\.)?(?:youtube\\.com|youtu\\.be)\\/watch\\?(?=.*v=([\\w-]+))(?:\\S+)?$", Pattern.CASE_INSENSITIVE);
|
|
|
|
// final Pattern pattern = Pattern.compile(".*v=([a-z0-9_\\-]+)(?:&.)*", Pattern.CASE_INSENSITIVE);
|
|
|
|
final Matcher matcher = pattern.matcher(playuri.toString());
|
|
|
|
if (matcher.matches()) {
|
|
|
|
return matcher.group(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
// Default page change listener, that doesn't scroll images
|
|
|
|
ViewPager.OnPageChangeListener defaultOnPageChangeListener = new ViewPager.OnPageChangeListener() {
|
|
|
|
@Override
|
|
|
|
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) { }
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageSelected(int position) {
|
|
|
|
setToolbarTitle(toolbar, position);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageScrollStateChanged(int state) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets or clear the image background
|
2015-02-15 20:11:32 +01:00
|
|
|
* @param url Image url
|
2015-01-14 12:12:47 +01:00
|
|
|
*/
|
|
|
|
private void setImageViewBackground(String url) {
|
|
|
|
if (url != null) {
|
|
|
|
Point displaySize = new Point();
|
|
|
|
getWindowManager().getDefaultDisplay().getSize(displaySize);
|
|
|
|
|
|
|
|
UIUtils.loadImageIntoImageview(hostManager, url, backgroundImage,
|
|
|
|
displaySize.x, displaySize.y / 2);
|
|
|
|
|
|
|
|
final int pixelsPerPage = displaySize.x / 4;
|
|
|
|
|
|
|
|
backgroundImage.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
|
|
|
|
@Override
|
|
|
|
public boolean onPreDraw() {
|
|
|
|
backgroundImage.getViewTreeObserver().removeOnPreDrawListener(this);
|
|
|
|
// Position the image
|
|
|
|
int offsetX = (viewPager.getCurrentItem() - 1) * pixelsPerPage;
|
|
|
|
backgroundImage.scrollTo(offsetX, 0);
|
|
|
|
|
|
|
|
pageIndicator.setOnPageChangeListener(new ViewPager.OnPageChangeListener() {
|
|
|
|
@Override
|
|
|
|
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
|
|
|
|
int offsetX = (int) ((position - 1 + positionOffset) * pixelsPerPage);
|
|
|
|
backgroundImage.scrollTo(offsetX, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageSelected(int position) {
|
|
|
|
setToolbarTitle(toolbar, position);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageScrollStateChanged(int state) { }
|
|
|
|
});
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
backgroundImage.setImageDrawable(null);
|
|
|
|
pageIndicator.setOnPageChangeListener(defaultOnPageChangeListener);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HostConnectionObserver.PlayerEventsObserver interface callbacks
|
|
|
|
*/
|
|
|
|
private String lastImageUrl = null;
|
|
|
|
public void playerOnPlay(PlayerType.GetActivePlayersReturnType getActivePlayerResult,
|
|
|
|
PlayerType.PropertyValue getPropertiesResult,
|
|
|
|
ListType.ItemsAll getItemResult) {
|
|
|
|
String imageUrl = (TextUtils.isEmpty(getItemResult.fanart)) ?
|
|
|
|
getItemResult.thumbnail : getItemResult.fanart;
|
|
|
|
if ((imageUrl != null) && !imageUrl.equals(lastImageUrl)) {
|
|
|
|
setImageViewBackground(imageUrl);
|
|
|
|
}
|
|
|
|
lastImageUrl = imageUrl;
|
2015-02-15 20:11:32 +01:00
|
|
|
|
|
|
|
// Check whether we should show a notification
|
|
|
|
boolean showNotification = PreferenceManager
|
|
|
|
.getDefaultSharedPreferences(this)
|
|
|
|
.getBoolean(Settings.KEY_PREF_SHOW_NOTIFICATION, Settings.DEFAULT_PREF_SHOW_NOTIFICATION);
|
|
|
|
if (showNotification) {
|
|
|
|
// Let's start the notification service
|
|
|
|
LogUtils.LOGD(TAG, "Starting notification service");
|
|
|
|
startService(new Intent(this, NotificationService.class));
|
|
|
|
}
|
2015-01-14 12:12:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void playerOnPause(PlayerType.GetActivePlayersReturnType getActivePlayerResult,
|
|
|
|
PlayerType.PropertyValue getPropertiesResult,
|
|
|
|
ListType.ItemsAll getItemResult) {
|
|
|
|
playerOnPlay(getActivePlayerResult, getPropertiesResult, getItemResult);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void playerOnStop() {
|
|
|
|
if (lastImageUrl != null) {
|
|
|
|
setImageViewBackground(null);
|
|
|
|
}
|
|
|
|
lastImageUrl = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void playerNoResultsYet() {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
public void playerOnConnectionError(int errorCode, String description) {
|
|
|
|
playerOnStop();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void systemOnQuit() {
|
|
|
|
Toast.makeText(this, R.string.xbmc_quit, Toast.LENGTH_SHORT).show();
|
|
|
|
playerOnStop();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void inputOnInputRequested(String title, String type, String value) {
|
|
|
|
SendTextDialogFragment dialog =
|
|
|
|
SendTextDialogFragment.newInstance(title);
|
|
|
|
dialog.show(getSupportFragmentManager(), null);
|
|
|
|
}
|
|
|
|
|
2015-02-15 20:11:32 +01:00
|
|
|
public void observerOnStopObserving() {}
|
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
/**
|
|
|
|
* Now playing fragment listener
|
|
|
|
*/
|
|
|
|
public void SwitchToRemotePanel() {
|
|
|
|
viewPager.setCurrentItem(1);
|
|
|
|
}
|
2015-01-23 23:23:59 +01:00
|
|
|
|
2015-01-14 12:12:47 +01:00
|
|
|
}
|