Kore/app/src/main/java/org/xbmc/kore/ui/sections/file/MediaFileListFragment.java

802 lines
33 KiB
Java
Raw Normal View History

/*
* Copyright 2015 DanhDroid. 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.
*/
package org.xbmc.kore.ui.sections.file;
import android.content.Context;
2015-03-28 19:33:48 +01:00
import android.content.res.Resources;
import android.os.Bundle;
import android.os.Handler;
import android.os.Parcel;
import android.os.Parcelable;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.PopupMenu;
import android.widget.TextView;
import android.widget.Toast;
import org.xbmc.kore.R;
import org.xbmc.kore.host.HostManager;
import org.xbmc.kore.jsonrpc.ApiCallback;
import org.xbmc.kore.jsonrpc.HostConnection;
import org.xbmc.kore.jsonrpc.method.Files;
import org.xbmc.kore.jsonrpc.method.Player;
import org.xbmc.kore.jsonrpc.method.Playlist;
import org.xbmc.kore.jsonrpc.type.ItemType;
import org.xbmc.kore.jsonrpc.type.ListType;
import org.xbmc.kore.jsonrpc.type.PlayerType;
import org.xbmc.kore.jsonrpc.type.PlaylistType;
import org.xbmc.kore.ui.AbstractListFragment;
import org.xbmc.kore.ui.viewgroups.RecyclerViewEmptyViewSupport;
2015-03-28 19:33:48 +01:00
import org.xbmc.kore.utils.LogUtils;
import org.xbmc.kore.utils.UIUtils;
2015-03-28 19:33:48 +01:00
import org.xbmc.kore.utils.Utils;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
2016-05-03 21:42:30 +02:00
import java.util.regex.Pattern;
/**
2015-03-22 16:55:18 +01:00
* Presents a list of files of different types (Video/Music)
*/
public class MediaFileListFragment extends AbstractListFragment {
2015-03-28 19:33:48 +01:00
private static final String TAG = LogUtils.makeLogTag(MediaFileListFragment.class);
public static final String MEDIA_TYPE = "mediaType";
public static final String SORT_METHOD = "sortMethod";
public static final String PATH_CONTENTS = "pathContents";
public static final String ROOT_PATH_CONTENTS = "rootPathContents";
public static final String ROOT_VISITED = "rootVisited";
2016-05-03 21:42:30 +02:00
public static final String ROOT_PATH = "rootPath";
public static final String DELAY_LOAD = "delayLoad";
private static final String ADDON_SOURCE = "addons:";
private HostManager hostManager;
/**
* Handler on which to post RPC callbacks
*/
private Handler callbackHandler = new Handler();
2015-03-22 16:55:18 +01:00
String mediaType = Files.Media.FILES;
ListType.Sort sortMethod = null;
String parentDirectory = null;
int playlistId = PlaylistType.MUSIC_PLAYLISTID; // this is the ID of the music player
// private MediaFileListAdapter adapter = null;
boolean browseRootAlready = false;
FileLocation loadOnVisible = null;
2015-03-22 16:55:18 +01:00
ArrayList<FileLocation> rootFileLocation = new ArrayList<>();
Queue<FileLocation> mediaQueueFileLocation = new LinkedList<>();
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putString(MEDIA_TYPE, mediaType);
outState.putParcelable(SORT_METHOD, sortMethod);
try {
outState.putParcelableArrayList(PATH_CONTENTS, (ArrayList<FileLocation>) ((MediaFileListAdapter) getAdapter()).getFileItemList());
} catch (NullPointerException npe) {
// adapter is null probably nothing was save in bundle because the directory is empty
// ignore this so that the empty message would display later on
}
outState.putParcelableArrayList(ROOT_PATH_CONTENTS, rootFileLocation);
outState.putBoolean(ROOT_VISITED, browseRootAlready);
}
@Override
protected RecyclerViewEmptyViewSupport.OnItemClickListener createOnItemClickListener() {
return new RecyclerViewEmptyViewSupport.OnItemClickListener() {
@Override
public void onItemClick(View view, int position) {
handleFileSelect(((MediaFileListAdapter) getAdapter()).getItem(position));
}
};
}
@Override
protected RecyclerView.Adapter createAdapter() {
return new MediaFileListAdapter(getActivity(), R.layout.grid_item_file);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View root = super.onCreateView(inflater, container, savedInstanceState);
Bundle args = getArguments();
2016-05-03 21:42:30 +02:00
FileLocation rootPath = null;
if (args != null) {
2016-05-03 21:42:30 +02:00
rootPath = args.getParcelable(ROOT_PATH);
mediaType = args.getString(MEDIA_TYPE, Files.Media.FILES);
if (mediaType.equalsIgnoreCase(Files.Media.VIDEO)) {
playlistId = PlaylistType.VIDEO_PLAYLISTID;
} else if (mediaType.equalsIgnoreCase(Files.Media.PICTURES)) {
playlistId = PlaylistType.PICTURE_PLAYLISTID;
}
sortMethod = args.getParcelable(SORT_METHOD);
}
hostManager = HostManager.getInstance(getActivity());
2015-03-28 19:33:48 +01:00
getEmptyView().setOnClickListener(new View.OnClickListener() {
2015-03-28 19:33:48 +01:00
@Override
public void onClick(View v) {
if (!atRootDirectory())
browseSources();
2015-03-28 19:33:48 +01:00
}
});
if (savedInstanceState != null) {
mediaType = savedInstanceState.getString(MEDIA_TYPE, Files.Media.FILES);
//currentPath = savedInstanceState.getString(CURRENT_PATH);
if (mediaType.equalsIgnoreCase(Files.Media.VIDEO)) {
playlistId = PlaylistType.VIDEO_PLAYLISTID;
} else if (mediaType.equalsIgnoreCase(Files.Media.PICTURES)) {
playlistId = PlaylistType.PICTURE_PLAYLISTID;
}
sortMethod = savedInstanceState.getParcelable(SORT_METHOD);
ArrayList<FileLocation> list = savedInstanceState.getParcelableArrayList(PATH_CONTENTS);
rootFileLocation = savedInstanceState.getParcelableArrayList(ROOT_PATH_CONTENTS);
browseRootAlready = savedInstanceState.getBoolean(ROOT_VISITED);
((MediaFileListAdapter) getAdapter()).setFilelistItems(list);
}
2016-05-03 21:42:30 +02:00
else if (rootPath != null) {
loadOnVisible = rootPath;
// setUserVisibleHint may have already fired
setUserVisibleHint(getUserVisibleHint() || !args.getBoolean(DELAY_LOAD, false));
2016-05-03 21:42:30 +02:00
}
else {
browseSources();
}
return root;
}
@Override
public void setUserVisibleHint (boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isVisibleToUser && loadOnVisible != null) {
FileLocation rootPath = loadOnVisible;
loadOnVisible = null;
browseRootAlready = true;
browseDirectory(rootPath);
}
}
void handleFileSelect(FileLocation f) {
// if selection is a directory, browse the the level below
if (f.isDirectory) {
// a directory - store the path of this directory so that we can reverse travel if
// we want to
if (f.isRootDir()) {
if (browseRootAlready)
browseDirectory(f);
else {
browseSources();
}
} else {
browseDirectory(f);
}
} else {
playMediaFile(f.file);
}
}
public void onBackPressed() {
// Emulate a click on ..
handleFileSelect(((MediaFileListAdapter) getAdapter()).getItem(0));
}
public boolean atRootDirectory() {
if (getAdapter().getItemCount() == 0)
return true;
FileLocation fl = ((MediaFileListAdapter) getAdapter()).getItem(0);
if (fl == null)
return true;
else
// if we still see "..", it is not the real root directory
return fl.isRootDir() &&
(fl.title != null) && !fl.title.contentEquals("..");
2015-03-28 19:33:48 +01:00
}
/**
* Gets and presents the list of media sources
*/
private void browseSources() {
Files.GetSources action = new Files.GetSources(mediaType);
action.execute(hostManager.getConnection(), new ApiCallback<List<ItemType.Source>>() {
@Override
public void onSuccess(List<ItemType.Source> result) {
2015-03-28 19:33:48 +01:00
if (!isAdded()) return;
// save this to compare when the user select a node
rootFileLocation.clear();
FileLocation fl;
for (ItemType.Source item : result) {
if ((item.file != null) &&
(!item.file.contains(ADDON_SOURCE))) {
fl = new FileLocation(item.label, item.file, true);
fl.setRootDir(true);
rootFileLocation.add(fl);
}
}
2015-03-22 16:55:18 +01:00
browseRootAlready = true;
getEmptyView().setText(getString(R.string.source_empty));
((MediaFileListAdapter) getAdapter()).setFilelistItems(rootFileLocation);
}
@Override
public void onError(int errorCode, String description) {
2015-03-28 19:33:48 +01:00
if (!isAdded()) return;
Toast.makeText(getActivity(),
String.format(getString(R.string.error_getting_source_info), description),
Toast.LENGTH_SHORT).show();
}
}, callbackHandler);
}
/**
* Gets and presents the files of the specified directory
* @param dir Directory to browse
*/
private void browseDirectory(final FileLocation dir) {
if (dir.isRootDir()) {
// this is a root directory
parentDirectory = dir.file;
} else {
// check to make sure that this is not our root path
String rootPath = null;
String path;
for (FileLocation fl : rootFileLocation) {
path = fl.file;
if ((path != null) && (dir.file != null) &&
(dir.file.contentEquals(path))) {
rootPath = fl.file;
break;
}
}
if (rootPath != null) {
parentDirectory = rootPath;
dir.setRootDir(true);
} else if (dir.file != null) {
parentDirectory = getParentDirectory(dir.file);
2015-03-22 16:55:18 +01:00
}
}
2015-03-28 19:33:48 +01:00
String[] properties = new String[] {
ListType.FieldsFiles.TITLE, ListType.FieldsFiles.ARTIST,
//ListType.FieldsFiles.ALBUMARTIST, ListType.FieldsFiles.GENRE,
//ListType.FieldsFiles.YEAR, ListType.FieldsFiles.RATING,
ListType.FieldsFiles.ALBUM, ListType.FieldsFiles.TRACK, ListType.FieldsFiles.DURATION,
//ListType.FieldsFiles.COMMENT,
//ListType.FieldsFiles.LYRICS, ListType.FieldsFiles.MUSICBRAINZTRACKID,
//ListType.FieldsFiles.MUSICBRAINZARTISTID, ListType.FieldsFiles.MUSICBRAINZALBUMID,
//ListType.FieldsFiles.MUSICBRAINZALBUMARTISTID, ListType.FieldsFiles.PLAYCOUNT,
//ListType.FieldsFiles.FANART,
//ListType.FieldsFiles.DIRECTOR, ListType.FieldsFiles.TRAILER,
ListType.FieldsFiles.TAGLINE,
//ListType.FieldsFiles.PLOT, ListType.FieldsFiles.PLOTOUTLINE, ListType.FieldsFiles.ORIGINALTITLE,
//ListType.FieldsFiles.LASTPLAYED, ListType.FieldsFiles.WRITER, ListType.FieldsFiles.STUDIO,
//ListType.FieldsFiles.MPAA, ListType.FieldsFiles.CAST, ListType.FieldsFiles.COUNTRY,
//ListType.FieldsFiles.IMDBNUMBER, ListType.FieldsFiles.PREMIERED,
//ListType.FieldsFiles.PRODUCTIONCODE,
ListType.FieldsFiles.RUNTIME,
//ListType.FieldsFiles.SET,
//ListType.FieldsFiles.SHOWLINK, ListType.FieldsFiles.STREAMDETAILS,
//ListType.FieldsFiles.TOP250, ListType.FieldsFiles.VOTES,
//ListType.FieldsFiles.FIRSTAIRED,
ListType.FieldsFiles.SEASON, ListType.FieldsFiles.EPISODE,
ListType.FieldsFiles.SHOWTITLE, ListType.FieldsFiles.THUMBNAIL, ListType.FieldsFiles.FILE,
//ListType.FieldsFiles.RESUME, ListType.FieldsFiles.ARTISTID, ListType.FieldsFiles.ALBUMID,
//ListType.FieldsFiles.TVSHOWID, ListType.FieldsFiles.SETID, ListType.FieldsFiles.WATCHEDEPISODES,
//ListType.FieldsFiles.DISC, ListType.FieldsFiles.TAG, ListType.FieldsFiles.ART,
//ListType.FieldsFiles.GENREID,
ListType.FieldsFiles.DISPLAYARTIST,
//ListType.FieldsFiles.ALBUMARTISTID, ListType.FieldsFiles.DESCRIPTION, ListType.FieldsFiles.THEME,
//ListType.FieldsFiles.MOOD, ListType.FieldsFiles.STYLE, ListType.FieldsFiles.ALBUMLABEL,
//ListType.FieldsFiles.SORTTITLE, ListType.FieldsFiles.EPISODEGUIDE,
//ListType.FieldsFiles.UNIQUEID, ListType.FieldsFiles.DATEADDED,
ListType.FieldsFiles.SIZE, ListType.FieldsFiles.LASTMODIFIED, ListType.FieldsFiles.MIMETYPE
};
Files.GetDirectory action = new Files.GetDirectory(dir.file,
2015-03-28 19:33:48 +01:00
mediaType,
sortMethod,
2015-03-28 19:33:48 +01:00
properties);
action.execute(hostManager.getConnection(), new ApiCallback<List<ListType.ItemFile>>() {
@Override
public void onSuccess(List<ListType.ItemFile> result) {
2015-03-22 16:55:18 +01:00
if (!isAdded()) return;
2015-03-28 19:33:48 +01:00
ArrayList<FileLocation> flList = new ArrayList<>();
2015-03-28 19:33:48 +01:00
2016-05-03 21:42:30 +02:00
if (dir.hasParent) {
// insert the parent directory as the first item in the list
FileLocation fl = new FileLocation("..", parentDirectory, true);
fl.setRootDir(dir.isRootDir());
flList.add(0, fl);
}
for (ListType.ItemFile i : result) {
2015-03-28 19:33:48 +01:00
flList.add(FileLocation.newInstanceFromItemFile(getActivity(), i));
}
((MediaFileListAdapter) getAdapter()).setFilelistItems(flList);
browseRootAlready = false;
}
@Override
public void onError(int errorCode, String description) {
2015-03-22 16:55:18 +01:00
if (!isAdded()) return;
2015-03-28 19:33:48 +01:00
Toast.makeText(getActivity(),
String.format(getString(R.string.error_getting_source_info), description),
Toast.LENGTH_SHORT).show();
}
}, callbackHandler);
}
/**
* Starts playing the given media file
* @param filename Filename to start playing
*/
private void playMediaFile(final String filename) {
PlaylistType.Item item = new PlaylistType.Item();
item.file = filename;
Player.Open action = new Player.Open(item);
action.execute(hostManager.getConnection(), new ApiCallback<String>() {
@Override
public void onSuccess(String result) {
while (!mediaQueueFileLocation.isEmpty()) {
queueMediaFile(mediaQueueFileLocation.poll().file);
}
}
@Override
public void onError(int errorCode, String description) {
2015-03-22 16:55:18 +01:00
if (!isAdded()) return;
Toast.makeText(getActivity(),
String.format(getString(R.string.error_play_media_file), description),
Toast.LENGTH_SHORT).show();
}
}, callbackHandler);
}
/**
* Queues the given media file on the active playlist, and starts it if nothing is playing
* @param filename File to queue
*/
private void queueMediaFile(final String filename) {
final HostConnection connection = hostManager.getConnection();
PlaylistType.Item item = new PlaylistType.Item();
item.file = filename;
Playlist.Add action = new Playlist.Add(playlistId, item);
action.execute(connection, new ApiCallback<String>() {
@Override
public void onSuccess(String result ) {
startPlaylistIfNoActivePlayers(connection, playlistId, callbackHandler);
}
@Override
public void onError(int errorCode, String description) {
2015-03-22 16:55:18 +01:00
if (!isAdded()) return;
Toast.makeText(getActivity(),
String.format(getString(R.string.error_queue_media_file), description),
Toast.LENGTH_SHORT).show();
}
}, callbackHandler);
}
/**
* Starts a playlist if no active players are playing
* @param connection Host connection
* @param playlistId PlaylistId to start
* @param callbackHandler Handler on which to post method callbacks
*/
private void startPlaylistIfNoActivePlayers(final HostConnection connection,
final int playlistId,
final Handler callbackHandler) {
Player.GetActivePlayers action = new Player.GetActivePlayers();
action.execute(connection, new ApiCallback<ArrayList<PlayerType.GetActivePlayersReturnType>>() {
@Override
public void onSuccess(ArrayList<PlayerType.GetActivePlayersReturnType> result ) {
// find out if any player is running. If it is not, start one
if (result.isEmpty()) {
2015-11-10 00:40:48 +01:00
Player.Open action = new Player.Open(Player.Open.TYPE_PLAYLIST, playlistId);
action.execute(connection, new ApiCallback<String>() {
@Override
public void onSuccess(String result) { }
@Override
public void onError(int errorCode, String description) {
if (!isAdded()) return;
Toast.makeText(getActivity(),
String.format(getString(R.string.error_play_media_file), description),
Toast.LENGTH_SHORT).show();
}
}, callbackHandler);
}
}
@Override
public void onError(int errorCode, String description) {
if (!isAdded()) return;
Toast.makeText(getActivity(),
String.format(getString(R.string.error_get_active_player), description),
Toast.LENGTH_SHORT).show();
}
}, callbackHandler);
}
/**
* return the path of the parent based on path
* @param path of the current media file
* @return path of the parent
*/
public static String getParentDirectory(final String path) {
String p = path;
String pathSymbol = "/"; // unix style
if (path.contains("\\")) {
pathSymbol = "\\"; // windows style
}
2015-03-22 16:55:18 +01:00
// if path ends with /, remove it before removing the directory name
if (path.endsWith(pathSymbol)) {
p = path.substring(0, path.length() - 1);
}
2015-03-22 16:55:18 +01:00
2015-04-14 00:29:20 +02:00
if (p.lastIndexOf(pathSymbol) != -1) {
p = p.substring(0, p.lastIndexOf(pathSymbol));
}
p = p + pathSymbol; // add it back to make it look like path
return p;
}
/**
* return the filename of a given path, if path is a directory, return directory name
* @param path of the current file
* @return filename or directory name
*/
public static String getFilenameFromPath(final String path) {
String p = path;
String pathSymbol = "/"; // unix style
if (path.contains("\\")) {
pathSymbol = "\\"; // windows style
}
// if path ends with /, remove it
if (path.endsWith(pathSymbol)) {
p = path.substring(0, path.length() - 1);
}
if (p.lastIndexOf(pathSymbol) != -1) {
p = p.substring(p.lastIndexOf(pathSymbol)+1);
}
return p;
}
Refactored AbstractDetailsFragment This introduces the MVC model to details fragments. It moves as much view and control code out of the concrete fragments into the abstract classes. * Added UML class and sequence diagrams under doc/diagrams to clarify the new setup * Introduces new abstract classes * AbstractFragment class to hold the DataHolder * AbstractInfoFragment class to display media information * AbstractAddtionalInfoFragment class to allow *InfoFragments to add additional UI elements and propagate refresh requests. See for an example TVShowInfoFragment which adds TVShowProgressFragment to display next episodes and season progression. * Introduces new RefreshItem class to encapsulate all refresh functionality from AbstractDetailsFragment * Introduces new SharedElementTransition utility class to encapsulate all shared element transition code * Introduces new CastFragment class to encapsulate all code for displaying casts reducing code duplication * Introduces DataHolder class to replace passing the ViewHolder from the *ListFragment to the *DetailsFragment or *InfoFragment * Refactored AbstractDetailsFragment into two classes: o AbstractDetailsFragment: for fragments requiring a tab bar o AbstractInfoFragment: for fragments showing media information We used to use <NAME>DetailsFragments for both fragments that show generic info about some media item and fragments that hold all details for some media item. For example, artist details showed artist info and used tabs to show artist albums and songs as well. Now Details fragments are used to show all details, Info fragments only show media item information like description, title, rating, etc. * Moved swiperefreshlayout code from AbstractCursorListFragment to AbstractListFragment
2016-12-30 09:27:24 +01:00
@Override
public void onRefresh() {
}
private class MediaFileListAdapter extends RecyclerView.Adapter {
Context ctx;
int resource;
List<FileLocation> fileLocationItems;
2015-03-28 19:33:48 +01:00
int artWidth;
int artHeight;
private View.OnClickListener itemMenuClickListener = new View.OnClickListener() {
@Override
public void onClick(View v) {
final int position = (Integer)v.getTag();
if (fileLocationItems != null) {
final FileLocation loc = fileLocationItems.get(position);
if (!loc.isDirectory) {
final PopupMenu popupMenu = new PopupMenu(getActivity(), v);
popupMenu.getMenuInflater().inflate(R.menu.filelist_item, popupMenu.getMenu());
popupMenu.setOnMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() {
@Override
public boolean onMenuItemClick(MenuItem item) {
switch (item.getItemId()) {
case R.id.action_queue_item:
queueMediaFile(loc.file);
return true;
case R.id.action_play_item:
playMediaFile(loc.file);
return true;
case R.id.action_play_from_this_item:
mediaQueueFileLocation.clear();
FileLocation fl;
// start playing the selected one, then queue the rest
mediaQueueFileLocation.add(loc);
for (int i = position + 1; i < fileLocationItems.size(); i++) {
fl = fileLocationItems.get(i);
if (!fl.isDirectory) {
mediaQueueFileLocation.add(fl);
}
}
playMediaFile(loc.file);
2015-03-22 16:55:18 +01:00
return true;
}
return false;
}
});
popupMenu.show();
}
}
}
};
MediaFileListAdapter(Context context, int resource) {
super();
this.ctx = context;
this.resource = resource;
this.fileLocationItems = null;
2015-03-28 19:33:48 +01:00
// Get the art dimensions
Resources resources = context.getResources();
artWidth = (int)(resources.getDimension(R.dimen.filelist_art_width) /
UIUtils.IMAGE_RESIZE_FACTOR);
artHeight = (int)(resources.getDimension(R.dimen.filelist_art_heigth) /
UIUtils.IMAGE_RESIZE_FACTOR);
}
/**
* Manually set the items on the adapter
* Calls notifyDataSetChanged()
*
* @param items list of files/directories
*/
public void setFilelistItems(List<FileLocation> items) {
this.fileLocationItems = items;
notifyDataSetChanged();
}
2015-03-22 16:55:18 +01:00
public List<FileLocation> getFileItemList() {
if (fileLocationItems == null)
return new ArrayList<>();
return new ArrayList<>(fileLocationItems);
}
public FileLocation getItem(int position) {
if (fileLocationItems == null) {
return null;
2015-03-22 16:55:18 +01:00
} else {
return fileLocationItems.get(position);
}
}
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(ctx)
.inflate(resource, parent, false);
return new ViewHolder(view, getContext(), hostManager, artWidth, artHeight, itemMenuClickListener);
}
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
FileLocation fileLocation = this.getItem(position);
((ViewHolder) holder).bindView(fileLocation, position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public int getItemCount() {
if (fileLocationItems == null) {
return 0;
2015-03-22 16:55:18 +01:00
} else {
return fileLocationItems.size();
}
}
}
/**
* View holder pattern
*/
private static class ViewHolder extends RecyclerView.ViewHolder {
ImageView art;
TextView title;
2015-03-28 19:33:48 +01:00
TextView details;
TextView sizeDuration;
ImageView contextMenu;
HostManager hostManager;
int artWidth;
int artHeight;
Context context;
ViewHolder(View itemView, Context context, HostManager hostManager, int artWidth, int artHeight,
View.OnClickListener itemMenuClickListener) {
super(itemView);
this.context = context;
this.hostManager = hostManager;
this.artWidth = artWidth;
this.artHeight = artHeight;
art = itemView.findViewById(R.id.art);
title = itemView.findViewById(R.id.title);
details = itemView.findViewById(R.id.details);
contextMenu = itemView.findViewById(R.id.list_context_menu);
sizeDuration = itemView.findViewById(R.id.size_duration);
contextMenu.setOnClickListener(itemMenuClickListener);
}
public void bindView(FileLocation fileLocation, int position) {
title.setText(UIUtils.applyMarkup(context, fileLocation.title));
details.setText(UIUtils.applyMarkup(context, fileLocation.details));
sizeDuration.setText(fileLocation.sizeDuration);
UIUtils.loadImageWithCharacterAvatar(context, hostManager,
fileLocation.artUrl, fileLocation.title,
art, artWidth, artHeight);
// For the popup menu
if (fileLocation.isDirectory) {
contextMenu.setVisibility(View.GONE);
} else {
contextMenu.setVisibility(View.VISIBLE);
contextMenu.setTag(position);
}
}
}
2015-03-28 19:33:48 +01:00
public static class FileLocation implements Parcelable {
public final String title;
public final String details;
public final String sizeDuration;
public final String artUrl;
public final String file;
public final boolean isDirectory;
2016-05-03 21:42:30 +02:00
public final boolean hasParent;
private boolean isRoot;
public boolean isRootDir() { return this.isRoot; }
public void setRootDir(boolean root) { this.isRoot = root; }
2015-03-28 19:33:48 +01:00
public FileLocation(String title, String path, boolean isDir) {
this(title, path, isDir, null, null, null);
}
2016-05-03 21:42:30 +02:00
static final Pattern noParent = Pattern.compile("plugin://[^/]*/?");
2015-03-28 19:33:48 +01:00
public FileLocation(String title, String path, boolean isDir, String details, String sizeDuration, String artUrl) {
this.title = title;
this.file = path;
this.isDirectory = isDir;
2016-05-03 21:42:30 +02:00
this.hasParent = !noParent.matcher(path).matches();
this.isRoot = false;
2015-03-28 19:33:48 +01:00
this.details = details;
this.sizeDuration = sizeDuration;
this.artUrl = artUrl;
}
public static FileLocation newInstanceFromItemFile(Context context, ListType.ItemFile itemFile) {
String title, details, sizeDuration, artUrl;
switch (itemFile.type) {
case ListType.ItemBase.TYPE_MOVIE:
title = itemFile.title;
details = itemFile.tagline;
2016-06-16 20:40:36 +02:00
sizeDuration = (itemFile.size > 0) && (itemFile.runtime > 0) ?
UIUtils.formatFileSize(itemFile.size) + " | " + UIUtils.formatTime(itemFile.runtime) :
(itemFile.size > 0) ? UIUtils.formatFileSize(itemFile.size) :
(itemFile.runtime > 0)? UIUtils.formatTime(itemFile.runtime) : null;
2015-03-28 19:33:48 +01:00
artUrl = itemFile.thumbnail;
break;
case ListType.ItemBase.TYPE_EPISODE:
title = itemFile.title;
details = String.format(context.getString(R.string.season_episode), itemFile.season, itemFile.episode);
2016-06-16 20:40:36 +02:00
sizeDuration = (itemFile.size > 0) && (itemFile.runtime > 0) ?
UIUtils.formatFileSize(itemFile.size) + " | " + UIUtils.formatTime(itemFile.runtime) :
(itemFile.size > 0) ? UIUtils.formatFileSize(itemFile.size) :
(itemFile.runtime > 0)? UIUtils.formatTime(itemFile.runtime) : null;
2015-03-28 19:33:48 +01:00
artUrl = itemFile.thumbnail;
break;
case ListType.ItemBase.TYPE_MUSIC_VIDEO:
title = itemFile.title;
details = Utils.listStringConcat(itemFile.artist, ", ") + " | " + itemFile.album;
2016-06-16 20:40:36 +02:00
sizeDuration = (itemFile.size > 0) && (itemFile.runtime > 0) ?
UIUtils.formatFileSize(itemFile.size) + " | " + UIUtils.formatTime(itemFile.runtime) :
(itemFile.size > 0) ? UIUtils.formatFileSize(itemFile.size) :
(itemFile.runtime > 0)? UIUtils.formatTime(itemFile.runtime) : null;
2015-03-28 19:33:48 +01:00
artUrl = itemFile.thumbnail;
break;
case ListType.ItemBase.TYPE_ALBUM:
title = itemFile.displayartist + " | " + itemFile.album;
details = getFilenameFromPath(itemFile.file);
artUrl = itemFile.thumbnail;
sizeDuration = (itemFile.size > 0) && (itemFile.duration > 0) ?
UIUtils.formatFileSize(itemFile.size) + " | " + UIUtils.formatTime(itemFile.duration) :
(itemFile.size > 0) ? UIUtils.formatFileSize(itemFile.size) :
(itemFile.duration > 0)? UIUtils.formatTime(itemFile.duration) : null;
break;
2015-03-28 19:33:48 +01:00
case ListType.ItemBase.TYPE_SONG:
title = itemFile.label;
details = getFilenameFromPath(itemFile.file);
2015-03-28 19:33:48 +01:00
artUrl = itemFile.thumbnail;
2016-06-16 20:40:36 +02:00
sizeDuration = (itemFile.size > 0) && (itemFile.duration > 0) ?
UIUtils.formatFileSize(itemFile.size) + " | " + UIUtils.formatTime(itemFile.duration) :
(itemFile.size > 0) ? UIUtils.formatFileSize(itemFile.size) :
(itemFile.duration > 0)? UIUtils.formatTime(itemFile.duration) : null;
2015-03-28 19:33:48 +01:00
break;
case ListType.ItemBase.TYPE_PICTURE:
default:
title = itemFile.label;
details = null;
artUrl = itemFile.thumbnail;
sizeDuration = UIUtils.formatFileSize(itemFile.size);
break;
}
return new FileLocation(title, itemFile.file,
itemFile.filetype.equalsIgnoreCase(ListType.ItemFile.FILETYPE_DIRECTORY),
details, sizeDuration, artUrl);
}
private FileLocation(Parcel in) {
2015-03-28 19:33:48 +01:00
this.title = in.readString();
this.file = in.readString();
this.isDirectory = (in.readInt() != 0);
2016-05-03 21:42:30 +02:00
this.hasParent = (in.readInt() != 0);
this.isRoot = (in.readInt() != 0);
2015-03-28 19:33:48 +01:00
this.details = in.readString();
this.sizeDuration = in.readString();
this.artUrl = in.readString();
}
public int describeContents() {
return 0;
}
public void writeToParcel(Parcel out, int flags) {
2015-03-28 19:33:48 +01:00
out.writeString(title);
out.writeString(file);
out.writeInt(isDirectory ? 1 : 0);
2016-05-03 21:42:30 +02:00
out.writeInt(hasParent ? 1 : 0);
out.writeInt(isRoot ? 1 : 0);
2015-03-28 19:33:48 +01:00
out.writeString(details);
out.writeString(sizeDuration);
out.writeString(artUrl);
}
public static final Parcelable.Creator<FileLocation> CREATOR = new Parcelable.Creator<FileLocation>() {
public FileLocation createFromParcel(Parcel in) {
return new FileLocation(in);
}
public FileLocation[] newArray(int size) {
return new FileLocation[size];
}
};
}
}