summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--vdr-vdrmanager/HISTORY8
-rw-r--r--vdr-vdrmanager/Makefile2
-rw-r--r--vdr-vdrmanager/Makefile.pre.1.7.3682
-rw-r--r--vdr-vdrmanager/README6
-rw-r--r--vdr-vdrmanager/compressor.cpp164
-rw-r--r--vdr-vdrmanager/compressor.h32
-rw-r--r--vdr-vdrmanager/examples/plugin.vdrmanager.conf2
-rw-r--r--vdr-vdrmanager/examples/vdr.vdrmanager3
-rw-r--r--vdr-vdrmanager/handler.cpp16
-rw-r--r--vdr-vdrmanager/helpers.cpp92
-rw-r--r--vdr-vdrmanager/helpers.h2
-rw-r--r--vdr-vdrmanager/select.cpp26
-rw-r--r--vdr-vdrmanager/sock.cpp220
-rw-r--r--vdr-vdrmanager/sock.h23
-rw-r--r--vdr-vdrmanager/vdrmanager.cpp42
-rw-r--r--vdr-vdrmanager/vdrmanagerthread.cpp30
-rw-r--r--vdr-vdrmanager/vdrmanagerthread.h6
-rw-r--r--vdrmanager/AndroidManifest.xml11
-rw-r--r--vdrmanager/build.xml145
-rw-r--r--vdrmanager/proguard-project.txt20
-rw-r--r--vdrmanager/res/drawable/folder.pngbin0 -> 4865 bytes
-rw-r--r--vdrmanager/res/layout-land/vdrmanager.xml33
-rw-r--r--vdrmanager/res/layout/folder_item.xml32
-rw-r--r--vdrmanager/res/layout/header_item.xml4
-rw-r--r--vdrmanager/res/layout/recording_list.xml24
-rw-r--r--vdrmanager/res/layout/vdrmanager.xml35
-rw-r--r--vdrmanager/res/values-de/strings.xml227
-rw-r--r--vdrmanager/res/values/strings.xml13
-rw-r--r--vdrmanager/res/values/styles.xml10
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/data/Channel.java1
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/data/Event.java2
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/data/Recording.java95
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/data/RecordingListItem.java43
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseActivity.java12
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventAdapter.java (renamed from vdrmanager/src/de/bjusystems/vdrmanager/gui/EventAdapter.java)95
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventListActivity.java2
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/ChannelEventAdapter.java7
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/EpgSearchTimesListActivity.java36
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingAdapter.java98
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingListActivity.java181
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/SvdrpProgressDialog.java2
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/TimeEventAdapter.java3
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/gui/VdrManagerActivity.java541
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpAsyncTask.java9
-rw-r--r--vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpClient.java29
45 files changed, 1564 insertions, 902 deletions
diff --git a/vdr-vdrmanager/HISTORY b/vdr-vdrmanager/HISTORY
index 34ed4d1..e62cb01 100644
--- a/vdr-vdrmanager/HISTORY
+++ b/vdr-vdrmanager/HISTORY
@@ -1,5 +1,11 @@
VDR Plugin 'vdrmanager' Revision History
------------------------------------
+
+2013-04-xx: Version 0.10 (http://projects.vdr-developer.org/versions/show/328)
+- Bug #1314: -f has not function, remove it. Deleting timers is always beeing forced
+- Feature #790: use some compression (zlib, gzip) to compress responses server sides
+- Feature #1319: Recording Folders
+
2013-01-xx: Version 0.9 (http://projects.vdr-developer.org/versions/show/312)
- Improved Timer/Recording deletion
- Improved recording information (recordings command)
@@ -39,7 +45,7 @@ VDR Plugin 'vdrmanager' Revision History
2011-11-06: Version 0.3
- Check against svdrphosts.conf file only if a) forced via command
line argument -s or no password was specified via -P
-- arguments parsing by getopts
+- arguments parsing by getopts
2011-10-12: Version 0.2
- Recording Info via recordings command
diff --git a/vdr-vdrmanager/Makefile b/vdr-vdrmanager/Makefile
index 4fae6b6..c43fa6a 100644
--- a/vdr-vdrmanager/Makefile
+++ b/vdr-vdrmanager/Makefile
@@ -47,7 +47,7 @@ SOFILE = libvdr-$(PLUGIN).so
### The object files (add further files here):
-OBJS = $(PLUGIN).o sock.o vdrmanagerthread.o select.o handler.o helpers.o
+OBJS = $(PLUGIN).o sock.o vdrmanagerthread.o select.o handler.o helpers.o compressor.o
### The main target:
diff --git a/vdr-vdrmanager/Makefile.pre.1.7.36 b/vdr-vdrmanager/Makefile.pre.1.7.36
new file mode 100644
index 0000000..90941fb
--- /dev/null
+++ b/vdr-vdrmanager/Makefile.pre.1.7.36
@@ -0,0 +1,82 @@
+#
+# Makefile for a Video Disk Recorder plugin
+#
+# $Id$
+
+# The official name of this plugin.
+# This name will be used in the '-P...' option of VDR to load the plugin.
+# By default the main source file also carries this name.
+#
+PLUGIN = vdrmanager
+
+### The version number of this plugin (taken from the main source file):
+
+VERSION = $(shell grep 'const char \*VERSION *=' $(PLUGIN).cpp | awk '{ print $$5 }' | sed -e 's/[";]//g')
+
+### The C++ compiler and options:
+
+CXX ?= g++
+CXXFLAGS ?= -O2 -Wall -Woverloaded-virtual -fPIC -g
+
+### The directory environment:
+
+DVBDIR = ../../../../DVB
+VDRDIR = ../../..
+LIBDIR = ../../lib
+TMPDIR = /tmp
+
+### Allow user defined options to overwrite defaults:
+
+-include $(VDRDIR)/Make.config
+
+### The version number of VDR (taken from VDR's "config.h"):
+
+APIVERSION = $(shell grep 'define APIVERSION ' $(VDRDIR)/config.h | awk '{ print $$3 }' | sed -e 's/"//g')
+
+### The name of the distribution archive:
+
+ARCHIVE = $(PLUGIN)-$(VERSION)
+PACKAGE = vdr-$(ARCHIVE)
+
+### Includes and Defines (add further entries here):
+
+INCLUDES += -I$(VDRDIR)/include -I$(DVBDIR)/include
+
+DEFINES += -D_GNU_SOURCE -DPLUGIN_NAME_I18N='"$(PLUGIN)"'
+
+### The object files (add further files here):
+
+OBJS = $(PLUGIN).o sock.o vdrmanagerthread.o select.o handler.o helpers.o
+
+### Implicit rules:
+
+%.o: %.cpp
+ $(CXX) -g $(CXXFLAGS) -c $(DEFINES) $(INCLUDES) $<
+
+# Dependencies:
+
+MAKEDEP = $(CXX) -MM -MG
+DEPFILE = .dependencies
+$(DEPFILE): Makefile
+ @$(MAKEDEP) $(DEFINES) $(INCLUDES) $(OBJS:%.o=%.cpp) > $@
+
+-include $(DEPFILE)
+
+### Targets:
+
+all: libvdr-$(PLUGIN).so
+
+libvdr-$(PLUGIN).so: $(OBJS)
+ $(CXX) -g $(CXXFLAGS) -shared $(OBJS) -o $@
+ @cp $@ $(LIBDIR)/$@.$(APIVERSION)
+
+dist: clean
+ @-rm -rf $(TMPDIR)/$(ARCHIVE)
+ @mkdir $(TMPDIR)/$(ARCHIVE)
+ @cp -a * $(TMPDIR)/$(ARCHIVE)
+ @tar czf $(PACKAGE).tgz -C $(TMPDIR) $(ARCHIVE)
+ @-rm -rf $(TMPDIR)/$(ARCHIVE)
+ @echo Distribution package created as $(PACKAGE).tgz
+
+clean:
+ @-rm -f $(OBJS) $(DEPFILE) *.so *.tgz core* *~
diff --git a/vdr-vdrmanager/README b/vdr-vdrmanager/README
index 8e0ed1e..610904f 100644
--- a/vdr-vdrmanager/README
+++ b/vdr-vdrmanager/README
@@ -8,6 +8,6 @@ This helper plugin allows remote programming VDR using
VDR-Manager running on Android devices.
Installation:
-- It's already debianized, so it can be build just by dpkg-buildpackage -b in the source folder
-- gentoo ebuilds can also be downloaded from the download section
-- Otherwise use Makfile
+http://www.vdr-wiki.de/wiki/index.php/Plugin_Installation
+
+If you use a vdr version lower then 1.7.36, use Makefile.pre.1.7.36
diff --git a/vdr-vdrmanager/compressor.cpp b/vdr-vdrmanager/compressor.cpp
new file mode 100644
index 0000000..6a479f3
--- /dev/null
+++ b/vdr-vdrmanager/compressor.cpp
@@ -0,0 +1,164 @@
+/*
+ * compressor.cpp
+ *
+ * Created on: 23.03.2013
+ * Author: bju
+ */
+
+#include "compressor.h"
+
+#include <zlib.h>
+#include <malloc.h>
+#include <string.h>
+#include <stdlib.h>
+
+#define CHUNK 16384
+
+cCompressor::cCompressor() {
+ size = 0;
+ data = NULL;
+}
+
+cCompressor::~cCompressor() {
+
+}
+
+bool cCompressor::CompressGzip(string text) {
+
+ int in_fd[2];
+ if (pipe(in_fd) < 0) {
+ return false;
+ }
+
+ int out_fd[2];
+ if (pipe(out_fd) < 0) {
+ return false;
+ }
+
+ int pid = fork();
+ if (pid < 0) {
+ return false;
+ }
+
+ if (pid == 0) {
+ // child
+ close(in_fd[1]);
+ close(out_fd[0]);
+ dup2(in_fd[0], 0);
+ dup2(out_fd[1], 1);
+
+ execlp("gzip", "gzip", "-c", "-9", NULL);
+
+ exit(-1);
+
+ } else {
+ // parent
+ close(in_fd[0]);
+ close(out_fd[1]);
+
+ write(in_fd[1], text.c_str(), text.length());
+ close(in_fd[1]);
+
+ char buf[32*1024];
+
+ for(;;) {
+ int count = read(out_fd[0], buf, sizeof(buf));
+ if (count < 0) {
+ close(out_fd[0]);
+ return false;
+ }
+ if (count == 0)
+ break;
+
+ char * newdata = (char *)malloc(size + count);
+ if (data != NULL) {
+ memcpy(newdata, data, size);
+ }
+ memcpy(newdata + size, buf, count);
+ if (data != NULL) {
+ free(data);
+ }
+ data = newdata;
+ size += count;
+ }
+
+ close(out_fd[0]);
+ }
+
+ return true;
+}
+
+bool cCompressor::CompressZlib(string text) {
+
+ int ret, flush;
+ unsigned have;
+ z_stream strm;
+ unsigned char in[CHUNK];
+ unsigned char out[CHUNK];
+ int level = 9;
+
+ strm.zalloc = Z_NULL;
+ strm.zfree = Z_NULL;
+ strm.opaque = Z_NULL;
+ ret = deflateInit(&strm, level);
+ if (ret != Z_OK)
+ return false;
+
+ string input = text;
+ uInt len;
+
+ do {
+ len = input.length();
+ if (len > CHUNK) {
+ len = CHUNK;
+ }
+ flush = len > 0 ? Z_NO_FLUSH : Z_FINISH;
+
+ strm.avail_in = len;
+ strm.next_in = (unsigned char *)input.c_str();
+
+ do {
+ strm.avail_out = CHUNK;
+ strm.next_out = out;
+
+ ret = deflate(&strm, flush);
+ if (ret < 0) {
+ if (data != NULL) {
+ free(data);
+ }
+ return false;
+ }
+
+ have = CHUNK - strm.avail_out;
+
+ char * newdata = (char *)malloc(size + have);
+ if (data != NULL) {
+ memcpy(newdata, data, size);
+ }
+ memcpy(newdata + size, out, have);
+ if (data != NULL) {
+ free(data);
+ }
+ data = newdata;
+ size += have;
+
+ } while (strm.avail_out == 0);
+
+ input = input.substr(len);
+
+ } while (flush != Z_FINISH);
+
+ // clean up and return
+ (void)deflateEnd(&strm);
+
+ return true;
+}
+
+char * cCompressor::GetData() {
+ return data;
+}
+
+size_t cCompressor::getDataSize() {
+ return size;
+}
+
diff --git a/vdr-vdrmanager/compressor.h b/vdr-vdrmanager/compressor.h
new file mode 100644
index 0000000..fbcc14f
--- /dev/null
+++ b/vdr-vdrmanager/compressor.h
@@ -0,0 +1,32 @@
+/*
+ * compressor.h
+ *
+ * Created on: 23.03.2013
+ * Author: bju
+ */
+
+#ifndef COMPRESSOR_H_
+#define COMPRESSOR_H_
+
+#include <string>
+
+#define COMPRESSION_NONE 0
+#define COMPRESSION_ZLIB 1
+#define COMPRESSION_GZIP 2
+
+using namespace std;
+
+class cCompressor {
+private:
+ char * data;
+ size_t size;
+public:
+ cCompressor();
+ virtual ~cCompressor();
+ bool CompressGzip(string text);
+ bool CompressZlib(string text);
+ char * GetData();
+ size_t getDataSize();
+};
+#endif /* COMPRESSOR_H_ */
+
diff --git a/vdr-vdrmanager/examples/plugin.vdrmanager.conf b/vdr-vdrmanager/examples/plugin.vdrmanager.conf
index ec4cad3..c0c9a7d 100644
--- a/vdr-vdrmanager/examples/plugin.vdrmanager.conf
+++ b/vdr-vdrmanager/examples/plugin.vdrmanager.conf
@@ -5,5 +5,7 @@
# -p port port number to listen to\n"
# -P password password (none if not given)"
# -s force check against svdrphosts.conf even if -P option was given (password)";
+# -c selects the compression mode to use (zlib or gzip). Default is zlib"
-p 6420
-P change
+-c
diff --git a/vdr-vdrmanager/examples/vdr.vdrmanager b/vdr-vdrmanager/examples/vdr.vdrmanager
index e1d1ddb..ba53505 100644
--- a/vdr-vdrmanager/examples/vdr.vdrmanager
+++ b/vdr-vdrmanager/examples/vdr.vdrmanager
@@ -5,4 +5,5 @@
# -p port port number to listen to\n"
# -P password password (none if not given)"
# -s force check against svdrphosts.conf";
-_EXTRAOPTS="-p 6420 -P change"
+# -c selects the compression mode to use (zlib or gzip). Default is zlib"
+_EXTRAOPTS="-p 6420 -P change -c"
diff --git a/vdr-vdrmanager/handler.cpp b/vdr-vdrmanager/handler.cpp
index 43b500a..ea15296 100644
--- a/vdr-vdrmanager/handler.cpp
+++ b/vdr-vdrmanager/handler.cpp
@@ -17,6 +17,8 @@ bool cHandler::HandleNewClient(cVdrmanagerClientSocket * sock)
bool cHandler::HandleClientRequest(cVdrmanagerClientSocket * sock)
{
+ bool closeSocket = true;
+
while(sock->Read())
{
// get lines
@@ -48,6 +50,12 @@ bool cHandler::HandleClientRequest(cVdrmanagerClientSocket * sock)
sock->SetLoggedIn();
sock->PutLine("!OK\r\n");
}
+ closeSocket = false;
+ }
+ else if (cmd == "COMPRESS")
+ {
+ sock->ActivateCompression();
+ closeSocket = false;
}
else if (cmd == "TIMERS")
{
@@ -97,9 +105,15 @@ bool cHandler::HandleClientRequest(cVdrmanagerClientSocket * sock)
else if (cmd == "QUIT")
{
// close socket
- sock->PutLine(string("Good bye! :-)\n"));
+ sock->PutLine(string("Good bye! :-)\r\n"));
sock->Disconnect();
}
+
+ if (closeSocket) {
+ sock->Disconnect();
+ }
+
+ sock->Flush();
}
}
diff --git a/vdr-vdrmanager/helpers.cpp b/vdr-vdrmanager/helpers.cpp
index 80721f3..b75931c 100644
--- a/vdr-vdrmanager/helpers.cpp
+++ b/vdr-vdrmanager/helpers.cpp
@@ -287,7 +287,9 @@ string cHelpers::DelRecording(cRecording * recording) {
if (!recording || recording->Delete()) {
cReplayControl::ClearLastReplayed(FileName);
Recordings.DelByName(FileName);
+#if VDRVERSNUM > 10727
cVideoDiskUsage::ForceCheck();
+#endif
}
return "START\r\nEND\r\n";
@@ -709,6 +711,10 @@ string cHelpers::ToText(cRecording * recording) {
}
}
+ result += ":";
+ //Feature #1319
+ result += recording->Name();
+
result += "\r\n";
return result;
}
@@ -819,8 +825,10 @@ string cHelpers::ToText(const cEvent * event) {
event->Schedule()->ChannelID());
// search assigned timer
- eTimerMatch TimerMatch = tmNone;
- cTimer * eventTimer = Timers.GetMatch(event, &TimerMatch);
+
+ //eTimerMatch TimerMatch = tmNone;
+ cTimer * eventTimer = Timers.GetMatch(event);
+
// if(eventTimer){
//
// for (cTimer * timer = Timers.First(); timer; timer = Timers.Next(timer)) {
@@ -1066,86 +1074,6 @@ int cHelpers::RecordingLengthInSeconds(cRecording* recording) {
return Duration(recording) * 60;
}
-/** Compress a STL string using zlib with given compression level and return
- * the binary data. */
-string cHelpers::compress_string(const string& str, int compressionlevel) {
- z_stream zs; // z_stream is zlib's control structure
- memset(&zs, 0, sizeof(zs));
-
- if (deflateInit(&zs, compressionlevel) != Z_OK)
- throw(runtime_error("deflateInit failed while compressing."));
-
- zs.next_in = (Bytef*) str.data();
- zs.avail_in = str.size(); // set the z_stream's input
-
- int ret;
- char outbuffer[32768];
- string outstring;
-
-// retrieve the compressed bytes blockwise
- do {
- zs.next_out = reinterpret_cast<Bytef*>(outbuffer);
- zs.avail_out = sizeof(outbuffer);
-
- ret = deflate(&zs, Z_FINISH);
-
- if (outstring.size() < zs.total_out) {
- // append the block to the output string
- outstring.append(outbuffer, zs.total_out - outstring.size());
- }
- } while (ret == Z_OK);
-
- deflateEnd(&zs);
-
- if (ret != Z_STREAM_END) { // an error occurred that was not EOF
- ostringstream oss;
- oss << "Exception during zlib compression: (" << ret << ") " << zs.msg;
- throw(runtime_error(oss.str()));
- }
-
- return outstring;
-}
-
-/** Decompress an STL string using zlib and return the original data. */
-string cHelpers::decompress_string(const string& str) {
- z_stream zs; // z_stream is zlib's control structure
- memset(&zs, 0, sizeof(zs));
-
- if (inflateInit(&zs) != Z_OK)
- throw(runtime_error("inflateInit failed while decompressing."));
-
- zs.next_in = (Bytef*) str.data();
- zs.avail_in = str.size();
-
- int ret;
- char outbuffer[32768];
- string outstring;
-
-// get the decompressed bytes blockwise using repeated calls to inflate
- do {
- zs.next_out = reinterpret_cast<Bytef*>(outbuffer);
- zs.avail_out = sizeof(outbuffer);
-
- ret = inflate(&zs, 0);
-
- if (outstring.size() < zs.total_out) {
- outstring.append(outbuffer, zs.total_out - outstring.size());
- }
-
- } while (ret == Z_OK);
-
- inflateEnd(&zs);
-
- if (ret != Z_STREAM_END) { // an error occurred that was not EOF
- ostringstream oss;
- oss << "Exception during zlib decompression: (" << ret << ") "
- << zs.msg;
- throw(runtime_error(oss.str()));
- }
-
- return outstring;
-}
-
//These three methodes were stolen from vdr-restfulapi project. Thanks!
std::queue<int> cHelpers::ConvertToBinary(int v) {
int b;
diff --git a/vdr-vdrmanager/helpers.h b/vdr-vdrmanager/helpers.h
index b9ebee5..18576be 100644
--- a/vdr-vdrmanager/helpers.h
+++ b/vdr-vdrmanager/helpers.h
@@ -31,8 +31,6 @@ public:
static string ToUpper(string text);
static string ToLower(string text);
static string Trim(string text);
- static string decompress_string(const string& str);
- static string compress_string(const string& str, int compressionlevel = Z_BEST_COMPRESSION);
static long Duration(cRecording* recording);
private:
static string SafeCall(string (*)());
diff --git a/vdr-vdrmanager/select.cpp b/vdr-vdrmanager/select.cpp
index 06d9103..c569f42 100644
--- a/vdr-vdrmanager/select.cpp
+++ b/vdr-vdrmanager/select.cpp
@@ -108,9 +108,8 @@ bool cSelect::Action() {
}
void cSelect::CreatePollfds() {
- // construct pollfd array
- // we need one pollfd for the eventpipe,
- // the serversocket and each clientsocket
+
+ // we poll for the server socket and for each client socket
pollfds = new struct pollfd[clientsocketcount + 1];
pollfds[0].fd = serversocket->GetSocket();
pollfds[0].events = POLLIN;
@@ -119,9 +118,10 @@ void cSelect::CreatePollfds() {
int i = 1;
while (curnode) {
pollfds[i].fd = curnode->socket->GetSocket();
- pollfds[i].events = POLLIN;
- if (curnode->socket->WritePending())
+ pollfds[i].events = POLLIN | POLLHUP;
+ if (curnode->socket->WritePending()) {
pollfds[i].events |= POLLOUT;
+ }
pollfds[i++].revents = 0;
curnode = curnode->next;
}
@@ -149,17 +149,17 @@ bool cSelect::Poll() {
for (int i = 1; i < clientsocketcount + 1; i++) {
cVdrmanagerClientSocket * sock = GetClientSocket(pollfds[i].fd);
if (sock) {
- if (pollfds[i].revents & (POLLIN | POLLHUP)) {
+ if (pollfds[i].revents & POLLOUT) {
+ // possibly outstanding writes
+ sock->Flush();
+ } else if (pollfds[i].revents & (POLLIN | POLLHUP)) {
// client request
handler->HandleClientRequest(sock);
+ }
- // disconnect?
- if (sock->Disconnected()) {
- RemoveClientSocket(sock);
- }
- } else if (pollfds[i].revents & POLLOUT) {
- // possibly outstanding writes
- sock->Flush();
+ // disconnect?
+ if (sock->Disconnected()) {
+ RemoveClientSocket(sock);
}
}
}
diff --git a/vdr-vdrmanager/sock.cpp b/vdr-vdrmanager/sock.cpp
index e2d4d36..a860b53 100644
--- a/vdr-vdrmanager/sock.cpp
+++ b/vdr-vdrmanager/sock.cpp
@@ -3,10 +3,9 @@
*/
#include <unistd.h>
#include <vdr/plugin.h>
-#include <openssl/ssl.h>
-#include <openssl/err.h>
#include "sock.h"
#include "helpers.h"
+#include "compressor.h"
static int clientno = 0;
@@ -15,9 +14,6 @@ static int clientno = 0;
*/
cVdrmanagerSocket::cVdrmanagerSocket() {
sock = -1;
- useSSL = false;
- password = "";
- forceCheckSvdrp = false;
}
cVdrmanagerSocket::~cVdrmanagerSocket() {
@@ -55,36 +51,21 @@ const char * cVdrmanagerSocket::GetPassword() {
return password;
}
-void cVdrmanagerSocket::LogSSLError() {
-
- char * error = ERR_error_string(ERR_get_error(), NULL);
-
- esyslog("SSL error: %s", error);
-
-}
-
/*
* cVdrmonServerSocket
*/
-cVdrmanagerServerSocket::cVdrmanagerServerSocket() : cVdrmanagerSocket() {
-
- sslContext = NULL;
+cVdrmanagerServerSocket::cVdrmanagerServerSocket() :
+ cVdrmanagerSocket() {
}
cVdrmanagerServerSocket::~cVdrmanagerServerSocket() {
}
-bool cVdrmanagerServerSocket::Create(int port, const char * password, bool forceCheckSvrp,
- bool useSSL, const char * pemFile) {
+bool cVdrmanagerServerSocket::Create(int port, const char * password, bool forceCheckSvrp, int compressionMode) {
- this->password = password;
+ this->password = password;
this->forceCheckSvdrp = forceCheckSvrp;
- this->useSSL = useSSL;
-
- // create SSL context
- if (useSSL && !InitSSL(pemFile)) {
- return false;
- }
+ this->compressionMode = compressionMode;
// create socket
sock = socket(PF_INET, SOCK_STREAM, 0);
@@ -133,18 +114,12 @@ cVdrmanagerClientSocket * cVdrmanagerServerSocket::Accept() {
int newsock = accept(sock, (struct sockaddr *) &clientname, &size);
if (newsock > 0) {
// create client socket
- newsocket = new cVdrmanagerClientSocket(password);
+ newsocket = new cVdrmanagerClientSocket(password, compressionMode);
if (!newsocket->Attach(newsock)) {
delete newsocket;
return NULL;
}
- // Attach client SSL
- if (!newsocket->InitSSL(sslContext)) {
- delete newsocket;
- return NULL;
- }
-
if (!IsPasswordSet() || forceCheckSvdrp == true) {
bool accepted = SVDRPhosts.Acceptable(clientname.sin_addr.s_addr);
if (!accepted) {
@@ -155,51 +130,34 @@ cVdrmanagerClientSocket * cVdrmanagerServerSocket::Accept() {
}
dsyslog(
"[vdrmanager] connect from %s, port %hd - %s", inet_ntoa(clientname.sin_addr), ntohs(clientname.sin_port), accepted ? "accepted" : "DENIED");
-
}
- } else if (errno != EINTR && errno != EAGAIN) {
+ } else if (errno != EINTR && errno != EAGAIN
+ )
LOG_ERROR;
- }
return newsocket;
}
-bool cVdrmanagerServerSocket::InitSSL(const char * pemFile) {
-
- sslContext = SSL_CTX_new(SSLv3_server_method());
- if (sslContext == NULL) {
- LogSSLError();
- return false;
- }
-
- if (SSL_CTX_use_certificate_file(sslContext, pemFile, SSL_FILETYPE_PEM) != 1) {
- LogSSLError();
- return false;
- }
-
- if (SSL_CTX_use_PrivateKey_file(sslContext, pemFile, SSL_FILETYPE_PEM) != 1) {
- LogSSLError();
- return false;
- }
-
- return true;
-}
/*
* cVdrmonClientSocket
*/
-cVdrmanagerClientSocket::cVdrmanagerClientSocket(const char * password) {
- readbuf = writebuf = "";
+cVdrmanagerClientSocket::cVdrmanagerClientSocket(const char * password, int compressionMode) {
+ readbuf = "";
+ writebuf = "";
+ sendbuf = NULL;
+ sendsize = 0;
+ sendoffset = 0;
disconnected = false;
+ initDisconnect = false;
client = ++clientno;
this->password = password;
+ this->compressionMode = compressionMode;
login = false;
+ compression = false;
+ initCompression = false;
}
cVdrmanagerClientSocket::~cVdrmanagerClientSocket() {
-
- if (sslContext) {
- SSL_free(sslContext);
- }
}
bool cVdrmanagerClientSocket::IsLineComplete() {
@@ -237,24 +195,13 @@ bool cVdrmanagerClientSocket::GetLine(string& line) {
}
bool cVdrmanagerClientSocket::Read() {
-
if (Disconnected())
return false;
int rc;
bool len = 0;
char buf[2001];
-
- for(;;) {
- if (useSSL) {
- rc = read(sock, buf, sizeof(buf)-1);
- } else {
- rc = SSL_read(sslContext, buf, sizeof(buf)-1);
- }
-
- if (rc <= 0)
- break;
-
+ while ((rc = read(sock, buf, sizeof(buf) - 1)) > 0) {
buf[rc] = 0;
readbuf += buf;
len += rc;
@@ -276,46 +223,74 @@ bool cVdrmanagerClientSocket::Disconnected() {
}
void cVdrmanagerClientSocket::Disconnect() {
- disconnected = true;
+ initDisconnect = true;
}
bool cVdrmanagerClientSocket::PutLine(string line) {
- //TODO http://projects.vdr-developer.org/issues/790
- //string line2 = cHelpers::compress_string(line);
- //unsigned long l = line.size();
- //unsigned long l2 = line2.size();
- //if(l2 == 0){
- //l2 = 1;
- //}
- //dsyslog("[vdrmanager] PutLine, line size is %lu, with zlib it would be %lu (factor %lu)", l, l2, l/l2);
- // add line to write buffer
- writebuf += line;
-
- // data present?
- if (writebuf.length() > 0) {
- int rc;
- if (useSSL) {
- rc = SSL_write(sslContext, writebuf.c_str(), writebuf.length());
- } else {
- rc = write(sock, writebuf.c_str(), writebuf.length());
- }
+
+ // fill writebuf
+ if (line.length() > 0) {
+ writebuf += line;
+ return true;
+ }
+
+ // initialize sendbuf if needed
+ if (sendbuf == NULL) {
+ if (!compression) {
+ sendbuf = (char *)malloc(writebuf.length()+1);
+ strcpy(sendbuf, writebuf.c_str());
+ sendsize = writebuf.length();
+ } else {
+ Compress();
+ }
+ sendoffset = 0;
+ writebuf.clear();
+ }
+
+ // send data
+ if (sendsize > 0) {
+
+ // write so many bytes as possible
+ int rc = write(sock, sendbuf + sendoffset, sendsize);
if (rc < 0 && errno != EAGAIN)
{
LOG_ERROR;
+
+ if (sendbuf != NULL) {
+ free(sendbuf);
+ sendbuf = NULL;
+ }
+
return false;
}
+ sendsize -= rc;
+ sendoffset += rc;
+ }
+
+ if (sendsize == 0) {
+
+ if (sendbuf != NULL) {
+ free(sendbuf);
+ sendbuf = NULL;
+ }
- // move the remainder
- if (rc > 0)
- writebuf = writebuf.substr(rc, writebuf.length() - rc);
+ if (initCompression) {
+ isyslog("Compression is activated now");
+ initCompression = false;
+ compression = true;
+ }
+
+ if (initDisconnect) {
+ initDisconnect = false;
+ disconnected = true;
+ }
}
return true;
}
bool cVdrmanagerClientSocket::Flush() {
- string empty = "";
- return PutLine(empty);
+ return PutLine("");
}
bool cVdrmanagerClientSocket::Attach(int fd) {
@@ -328,7 +303,7 @@ int cVdrmanagerClientSocket::GetClientId() {
}
bool cVdrmanagerClientSocket::WritePending() {
- return writebuf.length() > 0;
+ return sendsize > 0;
}
bool cVdrmanagerClientSocket::IsLoggedIn() {
@@ -339,20 +314,41 @@ void cVdrmanagerClientSocket::SetLoggedIn() {
login = true;
}
-bool::cVdrmanagerClientSocket::InitSSL(SSL_CTX * sslContext) {
-
- // create a new SSL context
- this->sslContext = SSL_new(sslContext);
- if (this->sslContext == NULL) {
- LOG_ERROR_STR("Error creating new SSL context");
- return false;
+void cVdrmanagerClientSocket::ActivateCompression() {
+
+ string mode = "NONE";
+ switch (compressionMode) {
+ case COMPRESSION_GZIP:
+ mode = "GZIP";
+ initCompression = true;
+ break;
+ case COMPRESSION_ZLIB:
+ mode = "ZLIB";
+ initCompression = true;
+ break;
+ default:
+ mode = "NONE";
+ break;
}
- // connect context to the socket
- if (SSL_set_fd(this->sslContext, sock) != 1) {
- SSL_free(this->sslContext);
- this->sslContext = NULL;
- LOG_ERROR_STR("Error connecting SSL and socket");
- return false;
+ PutLine("!OK " + mode + "\r\n");
+}
+
+void cVdrmanagerClientSocket::Compress() {
+ cCompressor compressor = cCompressor();
+
+ switch (compressionMode) {
+ case COMPRESSION_GZIP:
+ compressor.CompressGzip(writebuf);
+ break;
+ case COMPRESSION_ZLIB:
+ compressor.CompressZlib(writebuf);
+ break;
}
+
+ sendbuf = compressor.GetData();
+ sendsize = compressor.getDataSize();
+
+ double ratio = 1.0 * writebuf.length() / sendsize;
+ isyslog("Compression stats: raw %ld, compressed %ld, ratio %f:1", writebuf.length(), sendsize, ratio);
}
diff --git a/vdr-vdrmanager/sock.h b/vdr-vdrmanager/sock.h
index 94da4b2..25a7b83 100644
--- a/vdr-vdrmanager/sock.h
+++ b/vdr-vdrmanager/sock.h
@@ -5,10 +5,10 @@
#ifndef _VDRMON_SOCK
#define _VDRMON_SOCK
+#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string>
-#include <openssl/ssl.h>
using namespace std;
@@ -18,7 +18,7 @@ protected:
int sock;
const char * password;
bool forceCheckSvdrp;
- bool useSSL;
+ int compressionMode;
protected:
cVdrmanagerSocket();
bool IsPasswordSet();
@@ -28,7 +28,6 @@ public:
int GetSocket();
bool MakeDontBlock();
const char * GetPassword();
- void LogSSLError();
};
class cVdrmanagerClientSocket : public cVdrmanagerSocket
@@ -36,12 +35,18 @@ class cVdrmanagerClientSocket : public cVdrmanagerSocket
private:
string readbuf;
string writebuf;
+ char * sendbuf;
+ size_t sendsize;
+ size_t sendoffset;
bool disconnected;
+ bool initDisconnect;
int client;
bool login;
- SSL * sslContext;
+ bool compression;
+ bool initCompression;
+ int compressionMode;
public:
- cVdrmanagerClientSocket(const char * password);
+ cVdrmanagerClientSocket(const char * password, int compressionMode);
virtual ~cVdrmanagerClientSocket();
bool Attach(int fd);
bool IsLineComplete();
@@ -55,19 +60,17 @@ public:
bool WritePending();
bool IsLoggedIn();
void SetLoggedIn();
- bool InitSSL(SSL_CTX * sslContext);
+ void ActivateCompression();
+ void Compress();
};
class cVdrmanagerServerSocket : public cVdrmanagerSocket
{
-private:
- SSL_CTX * sslContext;
public:
cVdrmanagerServerSocket();
virtual ~cVdrmanagerServerSocket();
- bool Create(int port, const char * password, bool forceCheckSvdrp, bool useSSL, const char * pemFile);
+ bool Create(int port, const char * password, bool forceCheckSvdrp, int compressionMode);
cVdrmanagerClientSocket * Accept();
- bool InitSSL(const char * pemFile);
};
#endif
diff --git a/vdr-vdrmanager/vdrmanager.cpp b/vdr-vdrmanager/vdrmanager.cpp
index 96c5f6f..6542c89 100644
--- a/vdr-vdrmanager/vdrmanager.cpp
+++ b/vdr-vdrmanager/vdrmanager.cpp
@@ -12,10 +12,11 @@
#include <vdr/device.h>
#include <vdr/player.h>
#include "vdrmanagerthread.h"
+#include "compressor.h"
#define VDRMANAGER_PORT 6420
-static const char *VERSION = "0.9";
+static const char *VERSION = "0.10";
static const char *DESCRIPTION = "VDR-Manager support plugin";
class cVdrManager: public cPlugin {
@@ -25,9 +26,7 @@ private:
int port;
const char * password;
bool forceCheckSvdrp;
- bool useSSL;
- const char * pemFile;
- bool forceDelete;
+ int compressionMode;
protected:
public:
cVdrManager(void);
@@ -59,9 +58,6 @@ cVdrManager::cVdrManager(void) {
port = VDRMANAGER_PORT;
password = "";
forceCheckSvdrp = false;
- forceDelete = false;
- useSSL = false;
- pemFile = NULL;
}
cVdrManager::~cVdrManager() {
@@ -77,17 +73,16 @@ cMenuSetupPage * cVdrManager::SetupMenu(void) {
}
const char * cVdrManager::CommandLineHelp(void) {
- return " -p port port number to listen to\n -P password password (none if not given). No password forces check against svdrphosts.conf.\n -s force check against svdrphosts.conf, even if a password was given\n -f force delete of a timer or a recording even if they are active\n";
+ return " -p port port number to listen to\n"
+ " -P password password (none if not given). No password forces check against svdrphosts.conf.\n"
+ " -s force check against svdrphosts.conf, even if a password was given\n"
+ " -c compression selects the compression mode to use (zlib or gzip). Default is zlib";
}
bool cVdrManager::ProcessArgs(int argc, char *argv[]) {
int c;
- while ((c = getopt(argc, argv, "c:p:P:s:f")) != -1)
+ while ((c = getopt(argc, argv, "c::p:P:s")) != -1)
switch (c) {
- case 'c':
- pemFile = optarg;
- useSSL = true;
- break;
case 'p':
port = atoi(optarg);
break;
@@ -97,16 +92,18 @@ bool cVdrManager::ProcessArgs(int argc, char *argv[]) {
case 's':
forceCheckSvdrp = true;
break;
- case 'f':
- forceDelete = true;
+ case 'c':
+ if (!optarg) {
+ compressionMode = COMPRESSION_ZLIB;
+ } else if (optarg[0] == 'g') {
+ compressionMode = COMPRESSION_GZIP;
+ } else if (optarg[0] == 'z') {
+ compressionMode = COMPRESSION_ZLIB;
+ } else {
+ return false;
+ }
break;
case '?':
- if (optopt == 'c') {
- fprintf(stderr, "Option -%c requires an argument.\n", optopt);
- } else if (isprint(optopt))
- fprintf(stderr, "Unknown option `-%c'.\n", optopt);
- else
- fprintf(stderr, "Unknown option character `\\x%x'.\n", optopt);
return false;
default:
return false;
@@ -123,7 +120,8 @@ bool cVdrManager::Initialize(void) {
// Initialize any background activities the plugin shall perform.
// Start any background activities the plugin shall perform.
- Thread = new cVdrManagerThread(port, password, forceCheckSvdrp, useSSL, pemFile);
+ Thread = new cVdrManagerThread(port, password, forceCheckSvdrp,
+ compressionMode);
return Thread != NULL;
}
diff --git a/vdr-vdrmanager/vdrmanagerthread.cpp b/vdr-vdrmanager/vdrmanagerthread.cpp
index 977f3d1..51301e5 100644
--- a/vdr-vdrmanager/vdrmanagerthread.cpp
+++ b/vdr-vdrmanager/vdrmanagerthread.cpp
@@ -4,21 +4,17 @@
#include <string.h>
#include <vdr/plugin.h>
#include <vdr/thread.h>
-#include <openssl/ssl.h>
-
#include "vdrmanagerthread.h"
#include "select.h"
#include "helpers.h"
-cVdrManagerThread::cVdrManagerThread(int port, const char * password, bool forceCheckSvdrp,
- bool useSSL, const char * pemFile)
+cVdrManagerThread::cVdrManagerThread(int port, const char * password, bool forceCheckSvdrp, int compressionMode)
{
select = NULL;
- this->port = port;
- this->password = password;
- this->forceCheckSvdrp = forceCheckSvdrp;
- this->useSSL = useSSL;
- this->pemFile = pemFile;
+ this -> port = port;
+ this -> password = password;
+ this -> forceCheckSvdrp = forceCheckSvdrp;
+ this -> compressionMode = compressionMode;
}
cVdrManagerThread::~cVdrManagerThread()
@@ -28,11 +24,6 @@ cVdrManagerThread::~cVdrManagerThread()
void cVdrManagerThread::Action(void)
{
- // initialize SSL
- if (useSSL) {
- InitSSL();
- }
-
// create listener socket
if (!Init())
return;
@@ -53,7 +44,7 @@ bool cVdrManagerThread::Init()
// create server socket
cVdrmanagerServerSocket * sock = new cVdrmanagerServerSocket();
- if (sock == NULL || !sock->Create(port, password, forceCheckSvdrp, useSSL, pemFile))
+ if (sock == NULL || !sock->Create(port, password, forceCheckSvdrp, compressionMode))
return false;
// register server socket
@@ -71,12 +62,3 @@ void cVdrManagerThread::Cleanup()
void cVdrManagerThread::Shutdown()
{
}
-
-bool cVdrManagerThread::InitSSL() {
-
- SSL_library_init();
- SSL_load_error_strings();
- ERR_load_BIO_strings();
- ERR_load_SSL_strings();
-
-}
diff --git a/vdr-vdrmanager/vdrmanagerthread.h b/vdr-vdrmanager/vdrmanagerthread.h
index ac3551b..0dd3646 100644
--- a/vdr-vdrmanager/vdrmanagerthread.h
+++ b/vdr-vdrmanager/vdrmanagerthread.h
@@ -22,17 +22,15 @@ private:
int port;
const char * password;
bool forceCheckSvdrp;
- bool useSSL;
- const char * pemFile;
+ int compressionMode;
public:
- cVdrManagerThread(int port, const char * password, bool forceCheckSvdrp, bool useSSL, const char * pemFile);
+ cVdrManagerThread(int port, const char * password, bool forceCheckSvdrp, int compressionMode);
virtual void Action(void);
void Shutdown();
private:
virtual ~cVdrManagerThread();
void Cleanup();
bool Init();
- bool InitSSL();
};
#endif
diff --git a/vdrmanager/AndroidManifest.xml b/vdrmanager/AndroidManifest.xml
index 10d5357..5551e4d 100644
--- a/vdrmanager/AndroidManifest.xml
+++ b/vdrmanager/AndroidManifest.xml
@@ -1,21 +1,21 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="de.bjusystems.vdrmanager"
- android:versionCode="600"
- android:versionName="0.6.0" >
+ android:versionCode="700"
+ android:versionName="7" >
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
- <uses-permission android:name="android.permission.BROADCAST_SMS" />
<uses-sdk
- android:minSdkVersion="7"
+ android:minSdkVersion="8"
android:targetSdkVersion="14" />
<application
+ android:theme="@style/Theme.Sherlock"
android:name=".app.VdrManagerApp"
android:hardwareAccelerated="true"
android:icon="@drawable/app_logo"
@@ -24,8 +24,7 @@
android:name=".gui.VdrManagerActivity"
android:clearTaskOnLaunch="true"
android:configChanges="locale"
- android:label="@string/app_name"
- android:theme="@style/Theme.Sherlock" >
+ android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
diff --git a/vdrmanager/build.xml b/vdrmanager/build.xml
index 458b5b1..3a29200 100644
--- a/vdrmanager/build.xml
+++ b/vdrmanager/build.xml
@@ -1,10 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
-<project name="VDR-Manager" default="help">
+<project name="VdrManagerActivity" default="help">
<!-- The local.properties file is created and updated by the 'android' tool.
It contains the path to the SDK. It should *NOT* be checked into
Version Control Systems. -->
- <loadproperties srcFile="local.properties" />
+ <property file="local.properties" />
<!-- The ant.properties file can be created by you. It is only edited by the
'android' tool to add properties to it.
@@ -28,6 +28,15 @@
-->
<property file="ant.properties" />
+ <!-- if sdk.dir was not set from one of the property file, then
+ get it from the ANDROID_HOME env var.
+ This must be done before we load project.properties since
+ the proguard config can use sdk.dir -->
+ <property environment="env" />
+ <condition property="sdk.dir" value="${env.ANDROID_HOME}">
+ <isset property="env.ANDROID_HOME" />
+ </condition>
+
<!-- The project.properties file is created and updated by the 'android'
tool, as well as ADT.
@@ -41,25 +50,23 @@
<!-- quick check on sdk.dir -->
<fail
- message="sdk.dir is missing. Make sure to generate local.properties using 'android update project'"
+ message="sdk.dir is missing. Make sure to generate local.properties using 'android update project' or to inject it through the ANDROID_HOME environment variable."
unless="sdk.dir"
/>
-
-<!-- extension targets. Uncomment the ones where you want to do custom work
- in between standard targets -->
-<!--
- <target name="-pre-build">
- </target>
- <target name="-pre-compile">
- </target>
-
- /* This is typically used for code obfuscation.
- Compiled code location: ${out.classes.absolute.dir}
- If this is not done in place, override ${out.dex.input.absolute.dir} */
- <target name="-post-compile">
- </target>
--->
+ <!--
+ Import per project custom build rules if present at the root of the project.
+ This is the place to put custom intermediary targets such as:
+ -pre-build
+ -pre-compile
+ -post-compile (This is typically used for code obfuscation.
+ Compiled code location: ${out.classes.absolute.dir}
+ If this is not done in place, override ${out.dex.input.absolute.dir})
+ -post-package
+ -post-build
+ -pre-clean
+ -->
+ <import file="custom_rules.xml" optional="true" />
<!-- Import the actual build file.
@@ -81,107 +88,5 @@
-->
<!-- version-tag: 1 -->
<import file="${sdk.dir}/tools/ant/build.xml" />
- <tstamp>
- <format property="builddatetime" pattern="dd-MM-yyyy_HH.mm"/>
- </tstamp>
- <xmlproperty file="AndroidManifest.xml" prefix="mymanifest" collapseAttributes="true"/>
-
- <target name="-set-snapshot-mode" depends="-set-mode-check">
- <property name="out.packaged.file" location="${out.absolute.dir}/${ant.project.name}-release-unsigned.apk" />
- <property name="out.final.file" location="${out.absolute.dir}/${ant.project.name}-${mymanifest.manifest.android:versionName}-SNAPSHOT-${builddatetime}.apk" />
-
- <!-- record the current build target -->
- <property name="build.target" value="release" />
-
- <property name="build.is.instrumented" value="false" />
-
- <!-- release mode is only valid if the manifest does not explicitly
- set debuggable to true. default is false. -->
- <xpath input="AndroidManifest.xml" expression="/manifest/application/@android:debuggable"
- output="build.is.packaging.debug" default="false"/>
-
- <!-- signing mode: release -->
- <property name="build.is.signing.debug" value="false" />
-
- <if condition="${build.is.packaging.debug}">
- <then>
- <echo>*************************************************</echo>
- <echo>**** Android Manifest has debuggable=true ****</echo>
- <echo>**** Doing DEBUG packaging with RELEASE keys ****</echo>
- <echo>*************************************************</echo>
- </then>
- <else>
- <!-- property only set in release mode.
- Useful for if/unless attributes in target node
- when using Ant before 1.8 -->
- <property name="build.is.mode.release" value="true"/>
- </else>
- </if>
- </target>
-
-
- <target name="-set-release-mode" depends="-set-mode-check">
- <property name="out.packaged.file" location="${out.absolute.dir}/${ant.project.name}-release-unsigned.apk" />
- <property name="out.final.file" location="${out.absolute.dir}/${ant.project.name}-${mymanifest.manifest.android:versionName}.apk" />
-
- <!-- record the current build target -->
- <property name="build.target" value="release" />
-
- <property name="build.is.instrumented" value="false" />
-
- <!-- release mode is only valid if the manifest does not explicitly
- set debuggable to true. default is false. -->
- <xpath input="AndroidManifest.xml" expression="/manifest/application/@android:debuggable"
- output="build.is.packaging.debug" default="false"/>
-
- <!-- signing mode: release -->
- <property name="build.is.signing.debug" value="false" />
-
- <if condition="${build.is.packaging.debug}">
- <then>
- <echo>*************************************************</echo>
- <echo>**** Android Manifest has debuggable=true ****</echo>
- <echo>**** Doing DEBUG packaging with RELEASE keys ****</echo>
- <echo>*************************************************</echo>
- </then>
- <else>
- <!-- property only set in release mode.
- Useful for if/unless attributes in target node
- when using Ant before 1.8 -->
- <property name="build.is.mode.release" value="true"/>
- </else>
- </if>
- </target>
-
- <target name="snapshot"
- depends="-set-snapshot-mode, -release-obfuscation-check, -package, -release-prompt-for-password, -release-nosign"
- if="has.keystore"
- description="Builds the application. The generated apk file must be signed before
- it is published.">
-
- <!-- only create apk if *not* a library project -->
- <do-only-if-not-library elseText="Library project: do not create apk..." >
- <sequential>
- <property name="out.unaligned.file" location="${out.absolute.dir}/${ant.project.name}-release-unaligned.apk" />
-
- <!-- Signs the APK -->
- <echo>Signing final apk...</echo>
- <signjar
- jar="${out.packaged.file}"
- signedjar="${out.unaligned.file}"
- keystore="${key.store}"
- storepass="${key.store.password}"
- alias="${key.alias}"
- keypass="${key.alias.password}"
- verbose="${verbose}" />
-
- <!-- Zip aligns the APK -->
- <zipalign-helper in.package="${out.unaligned.file}"
- out.package="${out.final.file}" />
- <echo>Release Package: ${out.final.file}</echo>
- </sequential>
- </do-only-if-not-library>
- <record-build-info />
- </target>
</project>
diff --git a/vdrmanager/proguard-project.txt b/vdrmanager/proguard-project.txt
new file mode 100644
index 0000000..f2fe155
--- /dev/null
+++ b/vdrmanager/proguard-project.txt
@@ -0,0 +1,20 @@
+# To enable ProGuard in your project, edit project.properties
+# to define the proguard.config property as described in that file.
+#
+# Add project specific ProGuard rules here.
+# By default, the flags in this file are appended to flags specified
+# in ${sdk.dir}/tools/proguard/proguard-android.txt
+# You can edit the include path and order by changing the ProGuard
+# include property in project.properties.
+#
+# For more details, see
+# http://developer.android.com/guide/developing/tools/proguard.html
+
+# Add any project specific keep options here:
+
+# If your project uses WebView with JS, uncomment the following
+# and specify the fully qualified class name to the JavaScript interface
+# class:
+#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
+# public *;
+#}
diff --git a/vdrmanager/res/drawable/folder.png b/vdrmanager/res/drawable/folder.png
new file mode 100644
index 0000000..5b3fcec
--- /dev/null
+++ b/vdrmanager/res/drawable/folder.png
Binary files differ
diff --git a/vdrmanager/res/layout-land/vdrmanager.xml b/vdrmanager/res/layout-land/vdrmanager.xml
index 52cc9c6..8f5b86e 100644
--- a/vdrmanager/res/layout-land/vdrmanager.xml
+++ b/vdrmanager/res/layout-land/vdrmanager.xml
@@ -8,13 +8,13 @@
<ImageView android:id="@+id/main_logo" android:layout_width="fill_parent"
android:layout_height="80dip" android:src="@drawable/vdr_logo" android:paddingBottom="5dip" />
- <!-- <ListView android:id="@+id/vdrmanager_menu" android:layout_width="fill_parent"
+ <!-- <ListView android:id="@+id/vdrmanager_menu" android:layout_width="fill_parent"
android:layout_height="wrap_content" android:layout_below="@id/main_logo"/> -->
<LinearLayout android:layout_width="fill_parent"
android:layout_height="wrap_content" android:orientation="horizontal" android:padding="3dip">
<Button android:text="@string/action_menu_channels"
- xmlns:android="http://schemas.android.com/apk/res/android"
+
android:drawableTop="@drawable/btn_channels" android:layout_width="wrap_content"
android:id="@+id/action_menu_channels" android:layout_height="wrap_content"
android:textColor="#FFF" android:layout_weight="1"
@@ -23,17 +23,17 @@
<Button android:text="@string/action_menu_recordings"
- xmlns:android="http://schemas.android.com/apk/res/android" android:padding="3dip"
+ android:padding="3dip"
android:drawableTop="@drawable/btn_recordings" android:layout_width="wrap_content"
android:id="@+id/action_menu_recordings" android:layout_height="wrap_content"
android:layout_weight="1"
android:cacheColorHint="?android:attr/colorBackground"
android:background="@android:drawable/list_selector_background"
- android:textColor="#FFF"
+ android:textColor="#FFF"
/>
<Button
- xmlns:android="http://schemas.android.com/apk/res/android"
+
android:id="@+id/action_menu_timers"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
@@ -46,27 +46,8 @@
android:textColor="#FFF" />
</LinearLayout>
- <LinearLayout android:layout_width="fill_parent"
+ <LinearLayout android:layout_width="fill_parent"
android:layout_height="wrap_content" android:orientation="horizontal">
- <Button android:text="@string/action_menu_epg"
- xmlns:android="http://schemas.android.com/apk/res/android"
- android:drawableTop="@drawable/btn_epglist" android:layout_width="wrap_content"
- android:id="@+id/action_menu_epg" android:layout_height="wrap_content"
- android:layout_weight="1"
- android:cacheColorHint="?android:attr/colorBackground"
- android:padding="3dip"
- android:background="@android:drawable/list_selector_background"
- android:textColor="#FFF" />
-
- <Button android:text="@string/action_menu_search"
- xmlns:android="http://schemas.android.com/apk/res/android"
- android:drawableTop="@drawable/btn_search" android:layout_width="wrap_content"
- android:id="@+id/action_menu_search" android:layout_height="wrap_content"
- android:layout_weight="1"
- android:cacheColorHint="?android:attr/colorBackground"
- android:background="@android:drawable/list_selector_background"
- android:padding="3dip"
- android:textColor="#FFF" />
<Button android:text="@string/action_menu_wakeup"
xmlns:android="http://schemas.android.com/apk/res/android" android:layout_weight="1"
@@ -77,6 +58,6 @@
android:background="@android:drawable/list_selector_background"
android:textColor="#FFF" />
</LinearLayout>
-
+
</LinearLayout>
</ScrollView> \ No newline at end of file
diff --git a/vdrmanager/res/layout/folder_item.xml b/vdrmanager/res/layout/folder_item.xml
new file mode 100644
index 0000000..432fa13
--- /dev/null
+++ b/vdrmanager/res/layout/folder_item.xml
@@ -0,0 +1,32 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="fill_parent"
+ android:orientation="horizontal"
+ android:padding="5dip"
+ android:layout_height="wrap_content" >
+
+ <ImageView
+ android:layout_width="24dp"
+ android:layout_height="24dp"
+ android:src="@drawable/folder" />
+
+ <TextView
+ android:id="@+id/header_item"
+ android:textAppearance="?android:textAppearanceMedium"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:paddingLeft="5dip"
+ android:layout_weight="1"
+ android:layout_gravity="center_vertical"
+ android:text="House of Cards" />
+
+
+ <TextView
+ android:id="@+id/count"
+ android:textAppearance="?android:textAppearanceMedium"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:layout_gravity="center_vertical"
+ android:text="3" />
+
+</LinearLayout> \ No newline at end of file
diff --git a/vdrmanager/res/layout/header_item.xml b/vdrmanager/res/layout/header_item.xml
index 0ca07d6..0821980 100644
--- a/vdrmanager/res/layout/header_item.xml
+++ b/vdrmanager/res/layout/header_item.xml
@@ -1,8 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
-android:text="Header"
+android:text="Header"
android:layout_margin="2dip" android:paddingLeft="4dip" android:paddingRight="4dip"
android:background="#444444"
android:textAppearance="?android:textAppearanceMedium"
android:layout_width="fill_parent" android:layout_height="wrap_content" android:id="@+id/header_item"/>
-
+
diff --git a/vdrmanager/res/layout/recording_list.xml b/vdrmanager/res/layout/recording_list.xml
index 66eb5de..d4bd820 100644
--- a/vdrmanager/res/layout/recording_list.xml
+++ b/vdrmanager/res/layout/recording_list.xml
@@ -4,16 +4,34 @@
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
- <LinearLayout android:layout_height="fill_parent"
+ <LinearLayout android:layout_height="fill_parent" android:orientation="vertical"
android:layout_width="fill_parent" android:id="@+id/main_content">
- <ListView xmlns:android="http://schemas.android.com/apk/res/android"
+ <LinearLayout
+ android:layout_width="fill_parent"
+ android:layout_height="wrap_content"
+ android:layout_alignParentBottom="true"
+ android:orientation="horizontal"
+ android:paddingTop="2dp" android:paddingBottom="2dp" android:paddingLeft="5dp" android:paddingRight="5dp" >
+ <TextView
+ android:id="@+id/folder_info"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ android:layout_weight="1"
+ />
+ <TextView android:id="@+id/current_count"
+ android:layout_width="wrap_content"
+ android:layout_height="wrap_content"
+ />
+ </LinearLayout>
+
+ <ListView
android:id="@+id/recording_list" android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_alignWithParentIfMissing="true"
android:layout_alignParentTop="true" android:layout_centerHorizontal="true"
android:cacheColorHint="?android:attr/colorBackground" />
</LinearLayout>
-
+
<LinearLayout android:orientation="vertical"
android:id="@+id/no_connection_layout" android:layout_width="fill_parent"
android:layout_height="fill_parent">
diff --git a/vdrmanager/res/layout/vdrmanager.xml b/vdrmanager/res/layout/vdrmanager.xml
index d963d70..8fdf062 100644
--- a/vdrmanager/res/layout/vdrmanager.xml
+++ b/vdrmanager/res/layout/vdrmanager.xml
@@ -6,7 +6,7 @@
android:layout_height="wrap_content"
>
-
+
<LinearLayout
android:layout_width="fill_parent" android:layout_height="fill_parent"
android:orientation="vertical" android:padding="10dip">
@@ -15,7 +15,7 @@
android:clickable="true"
android:layout_height="80dip" android:src="@drawable/vdr_logo" android:padding="10dip" />
- <!-- <ListView android:id="@+id/vdrmanager_menu" android:layout_width="fill_parent"
+ <!-- <ListView android:id="@+id/vdrmanager_menu" android:layout_width="fill_parent"
android:layout_height="wrap_content" android:layout_below="@id/main_logo"/> -->
<LinearLayout android:layout_width="fill_parent"
@@ -26,9 +26,8 @@
android:textColor="#FFF" android:layout_weight="1"
android:cacheColorHint="?android:attr/colorBackground"
android:background="@android:drawable/list_selector_background"/>
-
+
<Button android:text="@string/action_menu_epg"
- xmlns:android="http://schemas.android.com/apk/res/android"
android:drawableTop="@drawable/btn_epglist" android:layout_width="wrap_content"
android:id="@+id/action_menu_epg" android:layout_height="wrap_content"
android:layout_weight="1"
@@ -37,22 +36,20 @@
android:textColor="#FFF" />
</LinearLayout>
-
+
<LinearLayout android:layout_width="fill_parent" android:padding="10dip"
android:layout_height="wrap_content" android:orientation="horizontal">
-
+
<Button android:text="@string/action_menu_recordings"
- xmlns:android="http://schemas.android.com/apk/res/android"
android:drawableTop="@drawable/btn_recordings" android:layout_width="wrap_content"
android:id="@+id/action_menu_recordings" android:layout_height="wrap_content"
android:layout_weight="1"
android:cacheColorHint="?android:attr/colorBackground"
android:background="@android:drawable/list_selector_background"
- android:textColor="#FFF"
+ android:textColor="#FFF"
/>
-
+
<Button android:text="@string/action_menu_timers"
- xmlns:android="http://schemas.android.com/apk/res/android"
android:drawableTop="@drawable/btn_timer" android:layout_width="wrap_content"
android:id="@+id/action_menu_timers" android:layout_height="wrap_content"
android:layout_weight="1"
@@ -63,29 +60,17 @@
<LinearLayout android:layout_width="fill_parent" android:padding="10dip"
android:layout_height="wrap_content" android:orientation="horizontal">
-
-
-
- <Button android:text="@string/action_menu_search"
- xmlns:android="http://schemas.android.com/apk/res/android"
- android:drawableTop="@drawable/btn_search" android:layout_width="wrap_content"
- android:id="@+id/action_menu_search" android:layout_height="wrap_content"
- android:layout_weight="1"
- android:cacheColorHint="?android:attr/colorBackground"
- android:background="@android:drawable/list_selector_background"
-
- android:textColor="#FFF" />
<Button android:text="@string/action_menu_wakeup"
- xmlns:android="http://schemas.android.com/apk/res/android" android:layout_weight="1"
+ android:layout_weight="1"
android:padding="5dip"
android:drawableTop="@drawable/btn_wakeup" android:layout_width="wrap_content"
android:id="@+id/action_menu_wakeup" android:layout_height="wrap_content"
-
+
android:cacheColorHint="?android:attr/colorBackground"
android:background="@android:drawable/list_selector_background"
android:textColor="#FFF" />
</LinearLayout>
-
+
</LinearLayout>
</ScrollView> \ No newline at end of file
diff --git a/vdrmanager/res/values-de/strings.xml b/vdrmanager/res/values-de/strings.xml
index 28ea91a..7a7ab5f 100644
--- a/vdrmanager/res/values-de/strings.xml
+++ b/vdrmanager/res/values-de/strings.xml
@@ -7,12 +7,9 @@
<string name="action_menu_epg">EPG Liste</string>
<string name="action_menu_search">EPG Suche</string>
<string name="action_menu_wakeup">Aufwecken</string>
-
<string name="app_name">VDR-Manager</string>
-
-
<string name="channel_item_menu_epg">Zeige EPG</string>
- <string name="channel_item_menu_stream">Zeige Livestream</string>
+ <string name="channel_item_menu_stream">Livestream</string>
<string name="channel_item_menu_hide">Verbergen</string>
<string name="channel_item_menu_hide_permanent">Permanent verbergen</string>
<string name="channel_group_menu_hide">Verbergen</string>
@@ -21,9 +18,11 @@
<string name="groupby_group">Gruppe</string>
<string name="groupby_provider">Provider</string>
<string name="groupby_name">Name</string>
+ <string name="groupby_source">Quelle</string>
+ <string name="groupby_channel">nach Kanal</string>
+ <string name="groupby_date">nach Datum</string>
<string name="groupby_name_all_channels_group">Alle Kanäle</string>
<string name="groupby_window_title_templte">Nach %1$s anordnen</string>
-
<string name="common_delete">Löschen</string>
<string name="prefs_current_value">Aktuell:</string>
<string name="prefs_current_value_template">Aktuell: %1$s</string>
@@ -34,38 +33,33 @@
<string name="refresh">Auffrischen</string>
<string name="done">Fertig</string>
<string name="share">Share</string>
+ <string name="addtocal">Zum Kalender</string>
<string name="search_reapt">Wiederholungen</string>
<string name="share_chooser">Share EPG Eintrag</string>
- <string name="about_text">
- Über \"VDR-Manager\" Software für Android\n
- Das ist Version %1$s\n
- Wenn Sie Ihren VDR zu Hause über das Internet und ein Handy mit Android
- erreichen wollen, dann finden sie dieses Programm vielleicht nützlich.
- Für weitere Informationen besuchen Sie http://projects.vdr-developer.org/projects/vdr-manager/wiki.
- Autoren:\n
- bju&lt;herrlado@gmail.com&gt;
- herrlado&lt;herrlado@gmail.com&gt;
- </string>
+ <string name="about_text">Über \"VDR-Manager\" für Android\n
+Dies ist Version %1$s\n
+Falls sie zuhause einen VDR haben, der über Internet erreichbar ist, und sie ein Android-Smartphone besitzen werden sie das Programm eventuell nützlich finden.
+Für weitere Informationen besuchen sie http://projects.vdr-developer.org/projects/vdr-manager/wiki.
+Author:\n
+bju&lt;herrlado@gmail.com&gt;
+herrlado&lt;herrlado@gmail.com&gt;</string>
<string name="about_title">Ãœber</string>
<string name="no_internet_connection">Keine Datenverbindung</string>
<string name="update_will_start_in">Die Liste wird in einer Sekunde aktualisiert</string>
-
<string name="epg_menu_search">Suche EPG</string>
<string name="epg_menu_times">Zeiten einstellen</string>
<string name="epg_item_menu_timer_add">Erzeuge Timer</string>
+ <string name="epg_item_menu_timer_record">Aufnahme</string>
<string name="epg_item_menu_timer_delete">Lösche Timer</string>
<string name="epg_item_menu_timer_modify">Bearbeite Timer</string>
<string name="epg_item_menu_timer_enable">Aktiviere Timer</string>
<string name="epg_item_menu_timer_disable">Deaktiviere Timer</string>
<string name="epg_no_items">Nichts gefunden…</string>
<string name="epg_client_errors">Ein Fehler ist aufgetreten, prüfe Log…</string>
-
<string name="navigae_at_the_end">Ende</string>
<string name="navigae_at_the_start">Anfang</string>
-
<string name="epg_search_times_add">Zeit hinzufügen</string>
<string name="epg_search_times_window">Selbstdefinierte Zeiten</string>
-
<string name="epg_list_time_label">Was kommt</string>
<string name="epg_list_time_now">jetzt</string>
<string name="epg_list_time_next">als nächstes</string>
@@ -75,13 +69,14 @@
<string name="epg_event_share_text">Share</string>
<string name="epg_event_imdb_text">IMDb Suche</string>
<string name="epg_event_omdb_text">OMDB Suche</string>
+ <string name="epg_event_tmdb_text">TMDb Suche</string>
<string name="epg_eent_livetv_text">Stream</string>
<string name="epg_event_create_timer_text">Timer hinzufügen</string>
<string name="epg_event_modify_timer_text">Timer modifizieren</string>
<string name="epg_search_button">Suchen</string>
<string name="epg_duration_template">%1$s min</string>
<string name="epg_duration_template_live">%1$s/%2$s min</string>
- <string name="epg_of_a_channel">EPG -&gt; %1$s, %2$s/%3$s</string>
+ <string name="epg_of_a_channel">%1$s, %2$s/%3$s</string>
<string name="epg_by_channel">EPG nach Kanälen</string>
<string name="epg_by_search">EPG Suche</string>
<string name="epg_by_search_param">EPG Suche: \'%1$s\'</string>
@@ -92,9 +87,7 @@
<string name="epg_window_title_count">%1$s, %2$s Einträge</string>
<string name="channels_window_title_count">%1$s, %2$s/%3$s</string>
<string name="new_timer">Neuer Timer</string>
-
<string name="vdr_error_text">Bei der Kommunikation mit dem VDR ist ein Fehler aufgetreten: %1$s</string>
-
<string name="livetv">Live TV</string>
<string name="settings_livetv_summary">Benötigt Streamdev-server auf dem VDR und ein externes Wiedergabeprogramm wie z.B. VPlayer</string>
<string name="settings_livetv_streamformat">Stream-Format</string>
@@ -105,6 +98,7 @@
<string name="key_remux_command">remux_command</string>
<string name="key_remux_parameter">remux_parameter</string>
<string name="stream_via_as">Stream</string>
+ <string name="stream_choose_audio">Audiospur wählen</string>
<string name="stream_via">über %s</string>
<string name="stream_as">als %s</string>
<string name="remux_title">Remux</string>
@@ -114,14 +108,6 @@
<string name="remux_command_summary">Typischerweise EXT oder Extern z.B. http://vdr:3000/Extern/1</string>
<string name="remux_parameter_title">Parameter</string>
<string name="remux_parameter_summary">Steuert Streaming Qualität. z.B. http://vdr:3000/Extern;DSL1000/1</string>
- <string-array name="livetv_streamformat">
- <item>TS</item>
- <item>PES</item>
- <item>ES</item>
- <item>PS</item>
- </string-array>
-
-
<string name="main_menu_preferences">Einstellungen</string>
<string name="main_menu_info">Infomationen</string>
<string name="main_menu_exit">Beenden</string>
@@ -129,7 +115,6 @@
<string name="main_menu_goto_title">Umstellen auf…</string>
<string name="main_menu_goto_no_vdr">Das gewählte Gerät kann nicht geladen werden :(</string>
<string name="main_menu_switched_to">Umgestellt auf %1$s</string>
-
<string name="vdr_devices_summary">Definiere ein oder weitere VDR Geräte. (aktuell nicht implementiert)</string>
<string name="vdr_devices">VDR Geräte</string>
<string name="vdr_preferences">Netzwerkeinstellungen</string>
@@ -184,11 +169,14 @@
<string name="qui_show_imdb_button_title">Zeige IMDb Button</string>
<string name="qui_show_imdb_button_summary_on">IMDb Button anzeigen</string>
<string name="qui_show_imdb_button_summary_off">IMDb Button nicht anzeigen</string>
+ <string name="qui_imdb_url_title">IMDb Seite</string>
+ <string name="qui_imdb_url_summary">Welche IMDb Seite für die Suche verwenden.</string>
<string name="qui_show_omdb_button_title">Zeige OMDB Button</string>
<string name="qui_show_omdb_button_summary_on">OMDB Button anzeigen</string>
<string name="qui_show_omdb_button_summary_off">OMDB Button nicht anzeigen</string>
- <string name="qui_imdb_url_title">IMDb Seite</string>
- <string name="qui_imdb_url_summary">Welche IMDb Seite für die Suche verwenden.</string>
+ <string name="qui_show_tmdb_button_title">TMDb Button anzeigen</string>
+ <string name="qui_show_tmdb_button_summary_on">TMDb Button wird angezeigt</string>
+ <string name="qui_show_tmdb_button_summary_off">TMDb Button wird nicht angezeigt</string>
<string name="gui_custom_locale_title">Andere Sprache</string>
<string name="gui_custom_locale_sum">Eine andere Sprache aktivieren.</string>
<string name="vdr_advanced_preferences">Erweitert</string>
@@ -212,31 +200,13 @@
<string name="vdr_name_title">Name</string>
<string name="vdr_name_summary">Der Name von diesem Gerät</string>
<string name="vdr_device_delete_qeustion">Soll dieses Gerät gelöscht werden?</string>
- <string-array name="wakeup_methods">
- <item>Rufe ein URL auf</item>
- <item>WOL</item>
- </string-array>
- <string-array name="imdb_urls">
- <item>International</item>
- <item>imdb.com</item>
- <item>uk.imdb.com</item>
- <item>imdb.de</item>
- <item>imdb.es</item>
- <item>imdb.fr</item>
- <item>imdb.it</item>
- <item>imdb.pt</item>
- </string-array>
-
- <string-array name="lang">
- <item>Englisch</item>
- <item>Deutsch</item>
- <item>Italienisch</item>
- </string-array>
- <string-array name="remux_entries">
- <item>EXT</item>
- <item>EXTERN (veraltet)</item>
- </string-array>
-
+ <string name="recstream">Streaming von Aufzeichnungen</string>
+ <string name="recstream_enable_title">Streaming von Aufzeichnungen</string>
+ <string name="recstream_enable_summary">Streaming von Aufzeichnungen aktivieren</string>
+ <string name="recstream_method_title">Streamingmethode</string>
+ <string name="recstream_method_summary">Momentan wird Streaming über VDR live oder das streamdev-plugin unterstützt</string>
+ <string name="settings_live_port">VDR Live-Plugin Port</string>
+ <string name="settings_live_port_summary">VDR Live-Plugin hat standardmäßig den Port 8008.</string>
<string name="progress_connect">Verbinde…</string>
<string name="progress_connect_error">Verbindung fehlgeschlagen!</string>
<string name="progress_connect_timeout">Zeitüberschreitung beim Verbindungsaufbau!</string>
@@ -245,7 +215,9 @@
<string name="progress_whatson_loading">Lade EPG…</string>
<string name="progress_timers_loading">Lade Timer…</string>
<string name="progress_recordings_loading">Lade Aufnahmen…</string>
+ <string name="progress_switching">Umschalten…</string>
<string name="progress_channels_loading">Lade Kanäle…</string>
+ <string name="progress_loading">Lade…</string>
<string name="progress_disconnect">Trenne…</string>
<string name="progress_wakeup_sending">Initalisiere aufwecken…</string>
<string name="progress_wakeup_sent">Aufforderung zum Wecken gesendet</string>
@@ -259,23 +231,130 @@
<string name="progress_connect_finished_abnormal">Verbindung unnormal beeendet…</string>
<string name="progress_connect_finished_abnormal_arg">Verbindung unnormal beeendet: %1$s</string>
<string name="progress_cache_hit">Lade aus Cache.</string>
-
<string name="recording_item_menu_delete">Löschen</string>
<string name="recording_item_menu_stream">Stream</string>
-
- <string name="timer_detail_title_title">Titel:</string>
- <string name="timer_detail_channel_title">Kanal:</string>
- <string name="timer_detail_start_title">Start:</string>
- <string name="timer_detail_end_title">Ende:</string>
- <string name="timer_details_create_title">Erzeuge Timer</string>
- <string name="timer_details_save_title">Speichere Änderungen</string>
- <string name="timer_details_delete_title">Lösche Timer</string>
- <string name="timer_details_add_title">Neuen Timer anlegen</string>
- <string name="timer_details_modify_title">Einen Timer ändern</string>
+ <string name="recent_channels">Letzte Kanäle</string>
+ <string name="recent_channels_no_history">Keine aktuellen Kanäle gefunden…</string>
+ <string name="gui_max_recent_channels_title">Maximale Kanalhistory</string>
+ <string name="gui_max_recent_channels_summary">Wie viele Kanäle sollen in der Kanalhistory verbleiben?</string>
+ <string name="timer_detail_day_title">Tag</string>
+ <string name="every_day">Jeden Tag</string>
+ <string name="never">Nie</string>
+ <string name="day_concat">", "</string>
+ <string name="timer_detail_repeat_title">Wiederholung</string>
+ <string name="timer_detail_use_vps">Verwende VPS:</string>
+ <string name="timer_detail_priority">Priorität:</string>
+ <string name="timer_detail_lifetime">Lebensdauer:</string>
+ <string name="timer_detail_lifetime_hint">z.B. 99</string>
+ <string name="timer_detail_priority_hint">z.B. 50</string>
+ <string name="timer_detail_title_vps">%1$s (VPS)</string>
+ <string name="channel_item_menu_switch">Umschalten</string>
+ <string name="switching_success">umgeschalten zu %s</string>
+ <string name="switching_failed">Umschalten zu %1$s fehlgeschlagen: %2$s</string>
+ <string name="switch_to_channel">Umschalten zu</string>
<string name="sort">Sortieren</string>
- <string name="sortby">Sortieren</string>
- <string name="sortby_time">Zeitlich</string>
- <string name="sortby_alphabet">Alphabetisch</string>
- <string name="sortby_channellist">Nach Kanalliste</string>
-
+ <string name="sortby">Sortiere nach</string>
+ <string name="sortby_time">Zeit</string>
+ <string name="sortby_channel">Kanal</string>
+ <string name="sortby_alphabet">Alphabet</string>
+ <string name="sortby_channellist">Kanalliste</string>
+ <string name="main_menu_clear_search">Lösche Such-History</string>
+ <string name="vdr_host_not_defined">VDR-Server nicht definiert</string>
+ <string name="processing">Ausführen…</string>
+ <string name="gui_recent_channels_order_summary">Sortiere nach zuletzt genutzt oder am häufigsten genutzt</string>
+ <string name="gui_recent_channels_order_title">Sortierung der aktuellen Kanäle</string>
+ <string name="filter">Filter</string>
+ <string name="audio_tracks_template">Audio Spuren: %1$s</string>
+ <string name="audio_track_dolby">D</string>
+ <string name="vdr_timezone_summary">Die Zeitzone des Servers. Das ist wichtig, falls sie verreisen und Timer von einer anderen Zeitzone aus bearbeiten.</string>
+ <string name="vdr_timezone_title">Zeitzone des Servers wählen</string>
+ <string name="zone_list_menu_sort_alphabetically">Sortiere alphabetisch</string>
+ <string name="zone_list_menu_sort_by_timezone">Sortiere nach Zeitzone</string>
+ <string name="probing">Prüfe %1$s …</string>
+ <string name="no_results">Keine Ergebnisse…</string>
+ <string name="recording_started">Aufzeichnung gestartet</string>
+ <string name="set_time">Zeit setzen</string>
+ <string-array name="livetv_streamformat">
+ <item>TS</item>
+ <item>PES</item>
+ <item>ES</item>
+ <item>PS</item>
+ </string-array>
+ <string-array name="wakeup_methods">
+ <item>Rufe ein URL auf</item>
+ <item>WOL</item>
+ </string-array>
+ <string-array name="imdb_urls">
+ <item>International</item>
+ <item>imdb.com</item>
+ <item>uk.imdb.com</item>
+ <item>imdb.de</item>
+ <item>imdb.es</item>
+ <item>imdb.fr</item>
+ <item>imdb.it</item>
+ <item>imdb.pt</item>
+ </string-array>
+ <string-array name="lang">
+ <item>Englisch</item>
+ <item>Deutsch</item>
+ <item>Italienisch</item>
+ <item>ukrainisch</item>
+ </string-array>
+ <string-array name="remux_entries">
+ <item>EXT</item>
+ <item>EXTERN (veraltet)</item>
+ </string-array>
+ <string-array name="recstream_entries">
+ <item>VDR Live plugin</item>
+ <item>VDR streamdev plugin</item>
+ </string-array>
+ <string-array name="recent_channels_order_entries">
+ <item>Am meisten genutzt</item>
+ <item>Zuletzt genutzt</item>
+ </string-array>
+ <string-array name="epg_sort_by_time_alpha">
+ <item>Zeit</item>
+ <item>Alphabet</item>
+ </string-array>
+ <string-array name="epg_sort_by_channels_alpha">
+ <item>Kanäle</item>
+ <item>Alphabet</item>
+ </string-array>
+ <string-array name="recordings_group_by">
+ <item>Nach Zeit</item>
+ <item>Alphabetisch</item>
+ </string-array>
+ <string-array name="channels_group_by">
+ <item>Gruppe</item>
+ <item>Anbieter</item>
+ <item>Quelle</item>
+ <item>Name</item>
+ </string-array>
+ <string-array name="navigation_array">
+ <item>Kanäle</item>
+ <item>EPG nach Zeit</item>
+ <item>EPG nach Kanal</item>
+ <item>Aufzeichnungen</item>
+ <item>Timer</item>
+ </string-array>
+
+ <!--
+ SSL related messages
+ -->
+
+ <string name="certificate_problem_message_title">Zertifikat-Warnung</string>
+ <string name="certificate_problem_message_text" formatted="false">
+Das Zertifikat des Servers wurde nicht akzeptiert:\n\n
+Hostname:\n
+ %s\n
+Gültig ab:\n
+ %s\n
+Gültig bis:\n
+ %s\n\n
+Was wollen Sie tun?
+ </string>
+ <string name="certificate_accept_once">Einmal akzeptieren</string>
+ <string name="certificate_accepted_forever">Immer akzeptieren</string>
+ <string name="certificate_not_accepted">Nicht akzeptieren</string>
+
</resources>
diff --git a/vdrmanager/res/values/strings.xml b/vdrmanager/res/values/strings.xml
index da1fd23..1ccc5ff 100644
--- a/vdrmanager/res/values/strings.xml
+++ b/vdrmanager/res/values/strings.xml
@@ -393,6 +393,8 @@
<string name="probing">Probing %1$s …</string>
<string name="no_results">No results…</string>
<string name="recording_started">Recording started</string>
+ <string name="set_time">Set time</string>
+ <!-- … -->
<!--
SSL related messages
@@ -400,16 +402,17 @@
<string name="certificate_problem_message_title">Certificate warning</string>
<string name="certificate_problem_message_text" formatted="false">
-The server certificate was not accepted:\n
+The server certificate was not accepted:\n\n
Hostname:\n
%s\n
-Created:\n
+Valid from:\n
%s\n
Valid until:\n
- %s
+ %s\n\n
+What do you want to do?
</string>
- <string name="certificate_accept_once">Continue</string>
- <string name="certificate_accepted_forever">Remember certificate</string>
+ <string name="certificate_accept_once">Accept once</string>
+ <string name="certificate_accepted_forever">Accept forever</string>
<string name="certificate_not_accepted">Abort</string>
</resources> \ No newline at end of file
diff --git a/vdrmanager/res/values/styles.xml b/vdrmanager/res/values/styles.xml
index 4c463ba..70ece1a 100644
--- a/vdrmanager/res/values/styles.xml
+++ b/vdrmanager/res/values/styles.xml
@@ -67,5 +67,15 @@
<item name="android:layout_marginBottom">10dp</item>
<item name="android:background">@drawable/line_gradient</item>
</style>
+
+ <style name="Line2">
+ <item name="android:layout_width">fill_parent</item>
+ <item name="android:layout_height">1px</item>
+ <item name="android:layout_marginLeft">10dp</item>
+ <item name="android:layout_marginRight">10dp</item>
+ <item name="android:layout_marginTop">1dp</item>
+ <item name="android:layout_marginBottom">1dp</item>
+ <item name="android:background">@drawable/line_gradient</item>
+ </style>
</resources> \ No newline at end of file
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/data/Channel.java b/vdrmanager/src/de/bjusystems/vdrmanager/data/Channel.java
index d8ca9ec..2b7b0d1 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/data/Channel.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/data/Channel.java
@@ -63,6 +63,7 @@ public class Channel implements Parcelable {
}
public Channel(final String channelData) {
+ System.err.println(channelData);
String[] words = StringUtils.splitPreserveAllTokens(channelData,
C.DATA_SEPARATOR);
this.number = Integer.valueOf(words[0].substring(1));
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/data/Event.java b/vdrmanager/src/de/bjusystems/vdrmanager/data/Event.java
index dcfb010..ef3fe0e 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/data/Event.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/data/Event.java
@@ -111,7 +111,7 @@ public abstract class Event {
}
return TextUtils.substring(description, 0, 30) + "…";
}
- return shortText;
+ return "";
}
public String getDescription() {
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/data/Recording.java b/vdrmanager/src/de/bjusystems/vdrmanager/data/Recording.java
index f2ad445..f510011 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/data/Recording.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/data/Recording.java
@@ -1,15 +1,60 @@
package de.bjusystems.vdrmanager.data;
-import java.util.Date;
+import static de.bjusystems.vdrmanager.gui.Utils.mapSpecialChars;
+import java.util.Date;
import de.bjusystems.vdrmanager.StringUtils;
import de.bjusystems.vdrmanager.app.C;
-import static de.bjusystems.vdrmanager.gui.Utils.mapSpecialChars;
-
public class Recording extends Event{
+ public static String ROOT_FOLDER = "";
+ public static final String FOLDERDELIMCHAR = "~";
+
+ public class Folder {
+
+ public String name;
+
+ public Folder parent;
+
+ private String path;
+
+ public boolean isRoot(){
+ return parent == null;
+ }
+
+ public String getFullPath(){
+ if(this.path != null){
+ return this.path;
+ }
+ if(isRoot()){
+ this.path = "";
+ } else {
+ this.path = parent.getFullPath() + "/" + name;
+ }
+
+ return path;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if(o == this){
+ return true;
+ }
+ return ((Folder)o).name.equals(this.name);
+ }
+ @Override
+ public int hashCode() {
+ return name.hashCode();
+ }
+
+ @Override
+ public String toString() {
+ return name + "("+path+")";
+ }
+ };
+
public Recording(String line) {
final String[] words = StringUtils.splitPreserveAllTokens(line, C.DATA_SEPARATOR);
int idx = 0;
@@ -17,7 +62,7 @@ public class Recording extends Event{
start = new Date(Long.parseLong(words[idx++]) * 1000);
stop = new Date(Long.parseLong(words[idx++]) * 1000);
channelName = mapSpecialChars(words[idx++]);
- title = mapSpecialChars(words[idx++]);
+ eventTitle = mapSpecialChars(words[idx++]);
shortText = mapSpecialChars(words[idx++]);
description = mapSpecialChars(words[idx++]);
fileName = mapSpecialChars(words[idx++]);
@@ -39,8 +84,27 @@ public class Recording extends Event{
timerStopTime = new Date(Long.parseLong(data) * 1000L);
}
}
+
+ if(idx < words.length) { //name
+ String titleRaw = words[idx];
+ int idxdel = titleRaw.lastIndexOf(FOLDERDELIMCHAR);
+ if(idxdel == -1){
+ title = titleRaw;
+ folder = ROOT_FOLDER;
+ } else {
+ title = titleRaw.substring(idxdel+1);
+
+ String foldersRaw = titleRaw.substring(0, idxdel);
+
+ folder = foldersRaw;
+
+ }
+ }
+
}
+ private String folder;
+
private String fileName;
private int fileSize;
@@ -51,11 +115,21 @@ public class Recording extends Event{
private String devInode = null;
+ private String eventTitle = null;
+
/**
* If it is not null, recording is on going or will be on going until this date;
*/
private Date timerStopTime = null;
+ public String getEventTitle() {
+ return eventTitle;
+ }
+
+ public void setEventTitle(String eventTitle) {
+ this.eventTitle = eventTitle;
+ }
+
public Date getTimerStopTime() {
return timerStopTime;
}
@@ -123,4 +197,15 @@ public class Recording extends Event{
public String toString() {
return title;
}
-}
+
+
+ public String getFolder() {
+ return folder;
+ }
+
+ public void setFolder(String folder) {
+ this.folder = folder;
+ }
+
+
+} \ No newline at end of file
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/data/RecordingListItem.java b/vdrmanager/src/de/bjusystems/vdrmanager/data/RecordingListItem.java
new file mode 100644
index 0000000..58d798c
--- /dev/null
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/data/RecordingListItem.java
@@ -0,0 +1,43 @@
+package de.bjusystems.vdrmanager.data;
+
+
+public class RecordingListItem extends EventListItem {
+
+ public String folder;
+
+ public Integer count = 0;
+
+ public RecordingListItem(Recording rec) {
+ super(rec);
+ }
+
+ public RecordingListItem(String dailyHeader) {
+ super(dailyHeader);
+ }
+
+
+
+ @Override
+ public String getTitle() {
+ if(isFolder()){
+ return folder;
+ }
+ return super.getTitle();
+ }
+
+ public boolean isFolder() {
+
+ return folder != null;
+
+ }
+
+ @Override
+ public String getHeader() {
+ if (isFolder()) {
+ return folder;
+ } else {
+ return super.getHeader();
+ }
+ }
+
+} \ No newline at end of file
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseActivity.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseActivity.java
index cdf0842..73d858c 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseActivity.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseActivity.java
@@ -206,7 +206,14 @@ SvdrpExceptionListener, SvdrpFinishedListener<Result> {
super.onCreate(savedInstanceState);
Preferences.setLocale(this);
progress = new ProgressDialog(this);
-
+ progress.setCancelable(false);
+ progress.setCanceledOnTouchOutside(false);
+ // progress.setOnCancelListener(new OnCancelListener() {
+ // @Override
+ // public void onCancel(DialogInterface dialog) {
+ //
+ // }
+ // });
initActionBar();
@@ -379,7 +386,8 @@ SvdrpExceptionListener, SvdrpFinishedListener<Result> {
@Override
public void svdrpEvent(final SvdrpEvent event, final Throwable t) {
- Utils.say(this, t.getLocalizedMessage());
+ progress.dismiss();
+ Utils.say(this, t.getMessage());
}
protected void addListener(final SvdrpAsyncTask<Result, SvdrpClient<Result>> task) {
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/EventAdapter.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventAdapter.java
index e5a8cc4..b022311 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/EventAdapter.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventAdapter.java
@@ -22,16 +22,16 @@ import de.bjusystems.vdrmanager.data.Recording;
import de.bjusystems.vdrmanager.data.TimerMatch;
import de.bjusystems.vdrmanager.data.Timerable;
-abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
+abstract class BaseEventAdapter<T extends EventListItem> extends ArrayAdapter<T> implements
Filterable
// , SectionIndexer
{
- private final int TYPE_ITEM = 0;
- private final int TYPE_HEADER = 1;
+ protected final static int TYPE_ITEM = 0;
+ protected final static int TYPE_HEADER = 1;
protected final int layout;
protected final LayoutInflater inflater;
- protected final List<EventListItem> items = new ArrayList<EventListItem>();
+ protected final List<T> items = new ArrayList<T>();
protected boolean hideDescription = true;
@@ -47,7 +47,7 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
*/
private final Object _Lock = new Object();
- public EventAdapter(final Context context, int layout) {
+ public BaseEventAdapter(final Context context, int layout) {
super(context, layout);
this.layout = layout;
inflater = LayoutInflater.from(context);
@@ -59,7 +59,7 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
}
@Override
- public void add(EventListItem object) {
+ public void add(T object) {
items.add(object);
// if (object.isHeader()) {
// sections.add(object.getHeader());
@@ -79,6 +79,28 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
return TYPE_ITEM;
}
+ public static class EventListItemHeaderHolder {
+ TextView header;
+ }
+
+ private boolean canReuseConvertView(View convertView, int itemViewType){
+ if(convertView == null){
+ return false;
+ }
+ Object o = convertView.getTag();
+ if(itemViewType == TYPE_ITEM){
+ return o instanceof EventListItemHolder;
+ }
+
+ if(itemViewType == TYPE_HEADER){
+ return o instanceof EventListItemHeaderHolder;
+ }
+
+ return false;
+
+ }
+
+
@Override
public View getView(final int position, View convertView,
final ViewGroup parent) {
@@ -88,7 +110,7 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
Object holder = null;
int type = getItemViewType(position);
- if (convertView == null) {
+ if (canReuseConvertView(convertView, type) == false) {
switch (type) {
case TYPE_ITEM:
convertView = inflater.inflate(layout, null);
@@ -104,17 +126,6 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
holder = convertView.getTag();
}
- switch (type) {
- case TYPE_ITEM:
- convertView = inflater.inflate(layout, null);
- holder = getEventViewHolder(item, convertView);
- break;
- case TYPE_HEADER:
- convertView = inflater.inflate(R.layout.header_item, null);
- holder = getHeaderViewHolder(item, convertView);
- break;
- }
-
if (type == TYPE_ITEM) {
fillEventViewHolder((EventListItemHolder) holder, item);
} else {
@@ -124,7 +135,7 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
return convertView;
}
- private EventListItemHolder getEventViewHolder(EventListItem item, View view) {
+ protected EventListItemHolder getEventViewHolder(EventListItem item, View view) {
EventListItemHolder itemHolder = new EventListItemHolder();
@@ -152,27 +163,30 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
itemHolder.state.setVisibility(View.VISIBLE);
if (item.getEvent() instanceof Recording) {
- Recording r = (Recording)item.getEvent();
- if(r.getTimerStopTime() != null){
+ Recording r = (Recording) item.getEvent();
+ if (r.getTimerStopTime() != null) {
itemHolder.state.setImageResource(R.drawable.timer_recording);
+ } else {
+ itemHolder.state.setImageResource(R.drawable.timer_none);
}
} else if (item.getEvent() instanceof Timerable == true) {
TimerMatch match = ((Timerable) item.getEvent()).getTimerMatch();
switch (((Timerable) item.getEvent()).getTimerState()) {
case Active:
- itemHolder.state.setImageResource(Utils.getTimerStateDrawable(match,
- R.drawable.timer_active, R.drawable.timer_active_begin,
+ itemHolder.state.setImageResource(Utils.getTimerStateDrawable(
+ match, R.drawable.timer_active,
+ R.drawable.timer_active_begin,
R.drawable.timer_active_end));
break;
case Inactive:
- itemHolder.state.setImageResource(Utils.getTimerStateDrawable(match,
- R.drawable.timer_inactive,
+ itemHolder.state.setImageResource(Utils.getTimerStateDrawable(
+ match, R.drawable.timer_inactive,
R.drawable.timer_inactive_begin,
R.drawable.timer_inactive_end));
break;
case Recording:
- itemHolder.state.setImageResource(Utils.getTimerStateDrawable(match,
- R.drawable.timer_recording,
+ itemHolder.state.setImageResource(Utils.getTimerStateDrawable(
+ match, R.drawable.timer_recording,
R.drawable.timer_recording_begin,
R.drawable.timer_recording_end));
break;
@@ -219,8 +233,8 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
itemHolder.progress.setProgress(p);
int dura = Utils.getDuration(item.getEvent());
int rest = dura - (dura * p / 100);
- //on live recordings the amount of already recorded time
- if(item.getEvent() instanceof Recording){
+ // on live recordings the amount of already recorded time
+ if (item.getEvent() instanceof Recording) {
rest = dura - rest;
}
itemHolder.duration.setText(getContext().getString(
@@ -228,11 +242,8 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
}
}
- class EventListItemHeaderHolder {
- public TextView header;
- }
- private EventListItemHeaderHolder getHeaderViewHolder(EventListItem item,
+ protected EventListItemHeaderHolder getHeaderViewHolder(EventListItem item,
View view) {
EventListItemHeaderHolder itemHolder = new EventListItemHeaderHolder();
itemHolder.header = (TextView) view.findViewById(R.id.header_item);
@@ -243,11 +254,11 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
return new EventFormatter(event);
}
- private void addSuper(EventListItem item) {
+ protected void addSuper(T item) {
super.add(item);
}
- private void clearSuper() {
+ protected void clearSuper() {
super.clear();
}
@@ -266,14 +277,18 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
public void setHideChannelName(boolean hideChannelName) {
this.hideChannelName = hideChannelName;
}
+
+ protected boolean isHeader(EventListItem item){
+ return item.isHeader();
+ }
// TODO implement locking in performFiltering, check the parent class
// http://stackoverflow.com/questions/5846385/how-to-update-android-listview-with-dynamic-data-in-real-time
public Filter getFilter() {
return new Filter() {
/**
- *
- */
+ *
+ */
EventListItem prevHead = null;
@Override
@@ -281,7 +296,7 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
highlight = arg0.toString().toLowerCase();
ArrayList<EventListItem> result = new ArrayList<EventListItem>();
for (EventListItem event : items) {
- if (event.isHeader()) {
+ if (isHeader(event)) {
prevHead = event;
// result.add(event);
continue;
@@ -308,7 +323,7 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
@Override
protected void publishResults(CharSequence arg0, FilterResults arg1) {
clearSuper();
- for (EventListItem item : (ArrayList<EventListItem>) arg1.values) {
+ for (T item : (ArrayList<T>) arg1.values) {
addSuper(item);
}
notifyDataSetChanged();
@@ -343,4 +358,4 @@ abstract class EventAdapter extends ArrayAdapter<EventListItem> implements
super.clear();
items.clear();
}
-} \ No newline at end of file
+}
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventListActivity.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventListActivity.java
index 2723d41..58b88ea 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventListActivity.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/BaseEventListActivity.java
@@ -51,7 +51,7 @@ SimpleGestureListener {
private SimpleGestureFilter detector;
- protected EventAdapter adapter;
+ protected BaseEventAdapter<EventListItem> adapter;
protected String highlight = null;
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/ChannelEventAdapter.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/ChannelEventAdapter.java
index 7e3d575..475f3e6 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/ChannelEventAdapter.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/ChannelEventAdapter.java
@@ -4,16 +4,17 @@ import android.content.Context;
import de.bjusystems.vdrmanager.R;
import de.bjusystems.vdrmanager.data.Event;
import de.bjusystems.vdrmanager.data.EventFormatter;
+import de.bjusystems.vdrmanager.data.EventListItem;
-class ChannelEventAdapter extends EventAdapter
+class ChannelEventAdapter extends BaseEventAdapter<EventListItem>
{
-
+
public ChannelEventAdapter(final Context context) {
super(context, R.layout.epg_event_item);
hideChannelName = true;
}
-
+
@Override
protected EventFormatter getEventFormatter(Event event) {
return new EventFormatter(event,true);
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/EpgSearchTimesListActivity.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/EpgSearchTimesListActivity.java
index 6cc5a3d..9be94cb 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/EpgSearchTimesListActivity.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/EpgSearchTimesListActivity.java
@@ -1,12 +1,10 @@
package de.bjusystems.vdrmanager.gui;
-import java.util.Calendar;
import java.util.List;
-import java.util.TimeZone;
import android.app.Activity;
-import android.app.TimePickerDialog;
-import android.app.TimePickerDialog.OnTimeSetListener;
+import android.app.AlertDialog;
+import android.content.DialogInterface;
import android.os.Bundle;
import android.view.ContextMenu;
import android.view.ContextMenu.ContextMenuInfo;
@@ -27,13 +25,14 @@ import de.bjusystems.vdrmanager.data.Preferences;
/**
* This class is used for showing what's
* current running on all channels
+
* @author bju
*/
public class EpgSearchTimesListActivity extends Activity
- implements OnClickListener, OnTimeSetListener {
+ implements OnClickListener{
ArrayAdapter<EpgSearchTimeValue> adapter;
-
+
List<EpgSearchTimeValue> values;
@Override
@@ -44,7 +43,7 @@ public class EpgSearchTimesListActivity extends Activity
setContentView(R.layout.epg_search_times_list);
setTitle(R.string.epg_search_times_window);
-
+
// Create adapter for ListView
adapter = new ArrayAdapter<EpgSearchTimeValue>(this, R.layout.epg_search_times_item);
final ListView listView = (ListView) findViewById(R.id.epg_search_times_list);
@@ -60,13 +59,28 @@ public class EpgSearchTimesListActivity extends Activity
}
public void onClick(final View v) {
- final Calendar cal = Calendar.getInstance(TimeZone.getDefault());
// show time selection
- final TimePickerDialog dialog = new TimePickerDialog(this, this, cal.get(Calendar.HOUR_OF_DAY), cal.get(Calendar.MINUTE), Preferences.get().isUse24hFormat());
- dialog.show();
+ final TimePicker timePicker = new TimePicker(this);
+ timePicker.setIs24HourView(Preferences.get().isUse24hFormat());
+ new AlertDialog.Builder(this)
+ .setTitle(R.string.set_time)
+ .setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
+ @Override
+ public void onClick(DialogInterface dialog, int which) {
+ onTimeSet(timePicker.getCurrentHour(), timePicker.getCurrentMinute());
+ }
+ })
+ .setNegativeButton(android.R.string.cancel,
+ new DialogInterface.OnClickListener() {
+ @Override
+ public void onClick(DialogInterface dialog,
+ int which) {
+ dialog.dismiss();
+ }
+ }).setView(timePicker).show();
}
- public void onTimeSet(final TimePicker view, final int hourOfDay, final int minute) {
+ public void onTimeSet(final int hourOfDay, final int minute) {
final EpgSearchTimeValue time = new EpgSearchTimeValue(values.size(), String.format("%02d:%02d", hourOfDay, minute));
values.add(time);
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingAdapter.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingAdapter.java
index 3645596..9d537c1 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingAdapter.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingAdapter.java
@@ -1,13 +1,19 @@
package de.bjusystems.vdrmanager.gui;
import android.content.Context;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.TextView;
import de.bjusystems.vdrmanager.R;
import de.bjusystems.vdrmanager.data.Event;
import de.bjusystems.vdrmanager.data.EventFormatter;
+import de.bjusystems.vdrmanager.data.EventListItem;
+import de.bjusystems.vdrmanager.data.RecordingListItem;
-class RecordingAdapter extends EventAdapter
-{
+class RecordingAdapter extends BaseEventAdapter<EventListItem> {
+
+ protected final static int TYPE_FOLDER = 2;
public RecordingAdapter(final Context context) {
super(context, R.layout.epg_event_item);
@@ -16,8 +22,94 @@ class RecordingAdapter extends EventAdapter
@Override
protected EventFormatter getEventFormatter(Event event) {
- return new EventFormatter(event, true);
+ return new EventFormatter(event, true);
+ }
+
+ @Override
+ public int getViewTypeCount() {
+ return 3;
+ }
+
+ @Override
+ protected boolean isHeader(EventListItem item) {
+ if(item instanceof RecordingListItem == false){
+ return item.isHeader();
+ }
+
+ if(((RecordingListItem)item).isFolder() ){
+ return false;
+ }
+
+ return item.isHeader();
+ }
+
+
+ @Override
+ public int getItemViewType(int position) {
+
+ // get item
+ final RecordingListItem item = (RecordingListItem) getItem(position);
+
+ if (item.isHeader()) {
+ return TYPE_HEADER;
+ } else if (item.isFolder()) {
+ return TYPE_FOLDER;
+ }
+ return TYPE_ITEM;
+ }
+
+
+
+ class EventListItemFolderHolder {
+ public TextView folder;
+ public TextView count;
+ }
+
+ protected EventListItemFolderHolder getFolderViewHolder(EventListItem item,
+ View view) {
+ EventListItemFolderHolder itemHolder = new EventListItemFolderHolder();
+ itemHolder.folder = (TextView) view.findViewById(R.id.header_item);
+ itemHolder.count = (TextView) view.findViewById(R.id.count);
+ return itemHolder;
+ }
+
+
+ @Override
+ public View getView(final int position, View convertView,
+ final ViewGroup parent) {
+
+ // get item
+ final RecordingListItem item = (RecordingListItem) getItem(position);
+
+ if (item.isFolder() == false) {
+ return super.getView(position, convertView, parent);
+ }
+
+ EventListItemFolderHolder holder = null;
+ if (convertView == null || (convertView != null && convertView.getTag() instanceof EventListItemFolderHolder) == false) {
+ convertView = inflater.inflate(R.layout.folder_item, null);
+ holder = getFolderViewHolder(item, convertView);
+ convertView.setTag(holder);
+ } else {
+ holder = (EventListItemFolderHolder) convertView.getTag();
+ }
+
+ holder.folder.setText(Utils.highlight(item.folder, highlight));
+ holder.count.setText(String.valueOf(item.count));
+ return convertView;
}
+ @Override
+ public RecordingListItem getItem(int position) {
+ return (RecordingListItem) super.getItem(position);
+ }
+//
+// protected void addSuper(RecordingListItem item) {
+// super.addSuper(item);
+// }
+//
+// protected void clearSuper() {
+// super.clear();
+// }
} \ No newline at end of file
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingListActivity.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingListActivity.java
index d369ca5..5a74852 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingListActivity.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/RecordingListActivity.java
@@ -3,7 +3,13 @@ package de.bjusystems.vdrmanager.gui;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collections;
+import java.util.Comparator;
import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.Stack;
+import java.util.TreeMap;
+import java.util.TreeSet;
import android.os.Bundle;
import android.view.ContextMenu;
@@ -14,12 +20,14 @@ import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemLongClickListener;
import android.widget.ListView;
+import android.widget.TextView;
import de.bjusystems.vdrmanager.R;
import de.bjusystems.vdrmanager.data.Event;
import de.bjusystems.vdrmanager.data.EventFormatter;
import de.bjusystems.vdrmanager.data.EventListItem;
import de.bjusystems.vdrmanager.data.Preferences;
import de.bjusystems.vdrmanager.data.Recording;
+import de.bjusystems.vdrmanager.data.RecordingListItem;
import de.bjusystems.vdrmanager.tasks.DeleteRecordingTask;
import de.bjusystems.vdrmanager.utils.date.DateFormatter;
import de.bjusystems.vdrmanager.utils.svdrp.RecordingClient;
@@ -35,28 +43,42 @@ import de.bjusystems.vdrmanager.utils.svdrp.SvdrpEvent;
public class RecordingListActivity extends BaseEventListActivity<Recording>
implements OnItemLongClickListener {
- //RecordingClient recordingClient;
+ // RecordingClient recordingClient;
- //public static final int MENU_GROUP_CHANNEL = 2;
+ // public static final int MENU_GROUP_CHANNEL = 2;
public static final int ASC = 0;
public static final int DESC = 1;
- protected static ArrayList<Recording> CACHE = new ArrayList<Recording>();
+ // protected static ArrayList<Recording> CACHE = new ArrayList<Recording>();
+ private static Map<String, List<Recording>> CACHE = new TreeMap<String, List<Recording>>();
+
+ public static final Map<String, Set<String>> FOLDERS = new TreeMap<String, Set<String>>();
+
+ private String currentFolder = Recording.ROOT_FOLDER;
private final int ASC_DESC = ASC;
+ private static final List<Recording> EMPTY = new ArrayList<Recording>(0);
+
+ private final Stack<String> stack = new Stack<String>();
+
+ private TextView folderInfo;
+
+ private TextView currentCount;
+
@Override
protected void onCreate(final Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
- // create an adapter
adapter = new RecordingAdapter(this);
// attach adapter to ListView
listView = (ListView) findViewById(R.id.recording_list);
+ folderInfo = (TextView) findViewById(R.id.folder_info);
+ currentCount = (TextView) findViewById(R.id.current_count);
listView.setAdapter(adapter);
// set click listener
@@ -116,6 +138,32 @@ implements OnItemLongClickListener {
// }
// }
+ @Override
+ public void onItemClick(final AdapterView<?> parent, final View view, final int position,
+ final long id) {
+
+ final RecordingListItem item = (RecordingListItem) adapter
+ .getItem(position);
+ if (item.isFolder()) {
+ if (currentFolder.equals(Recording.ROOT_FOLDER)) {
+ currentFolder = item.folder;
+ } else {
+ currentFolder = currentFolder + Recording.FOLDERDELIMCHAR
+ + item.folder;
+ }
+ stack.push(currentFolder);
+ fillAdapter();
+ } else {
+ super.onItemClick(parent, view, position, id);
+ }
+ }
+
+ private void updateCurrentFolderInfo() {
+ folderInfo.setText("/" + currentFolder.replaceAll("~", "/"));
+ final List<Recording> list = CACHE.get(currentFolder);
+ currentCount.setText(String.valueOf(list.size()));
+ }
+
/*
* (non-Javadoc)
*
@@ -139,7 +187,7 @@ implements OnItemLongClickListener {
@Override
protected void prepareDetailsViewData(final EventListItem event) {
getApp().setCurrentEvent(event.getEvent());
- getApp().setCurrentEpgList(CACHE);
+ getApp().setCurrentEpgList(CACHE.get(currentFolder));
}
@Override
@@ -168,9 +216,9 @@ implements OnItemLongClickListener {
super.onCreateContextMenu(menu, v, menuInfo);
//
// http://projects.vdr-developer.org/issues/863
- //if (Utils.isLive(item)) {
+ // if (Utils.isLive(item)) {
menu.removeItem(R.id.epg_item_menu_live_tv);
- //}
+ // }
}
@Override
@@ -251,16 +299,16 @@ implements OnItemLongClickListener {
/* */
switch (sortBy) {
case MENU_GROUP_DEFAULT: {
- sortItemsByTime(CACHE, true);
+ sortItemsByTime(CACHE.get(currentFolder), true);
break;
}
case MENU_GROUP_ALPHABET: {
- Collections.sort(CACHE, new TitleComparator());
+ Collections.sort(CACHE.get(currentFolder), new TitleComparator());
break;
}
- //case MENU_GROUP_CHANNEL: {
- //sortItemsByChannel(results);
- //}
+ // case MENU_GROUP_CHANNEL: {
+ // sortItemsByChannel(results);
+ // }
}
}
@@ -268,40 +316,118 @@ implements OnItemLongClickListener {
protected void fillAdapter() {
adapter.clear();
-
- if (CACHE.isEmpty()) {
+ final List<Recording> list = CACHE.get(currentFolder);
+ if (list == null || list.isEmpty()) {
return;
}
+ updateCurrentFolderInfo();
+
sort();
final Calendar cal = Calendar.getInstance();
int day = -1;
- for (final Event rec : CACHE) {
+ final Set<String> folders = FOLDERS.get(currentFolder);
+ if (folders != null) {
+ for (final String f : folders) {
+ final RecordingListItem recordingListItem = new RecordingListItem(f);
+ recordingListItem.folder = f;
+ final String sf = currentFolder.length() > 0 ? currentFolder
+ + Recording.FOLDERDELIMCHAR + f : f;
+ final List<Recording> list2 = CACHE.get(sf);
+ if (list2 != null) {
+ recordingListItem.count = list2.size();
+ }
+ adapter.add(recordingListItem);
+ }
+ }
+
+ for (final Event rec : CACHE.get(currentFolder)) {
cal.setTime(rec.getStart());
final int eday = cal.get(Calendar.DAY_OF_YEAR);
if (eday != day) {
day = eday;
- adapter.add(new EventListItem(new DateFormatter(cal)
+ adapter.add(new RecordingListItem(new DateFormatter(cal)
.getDailyHeader()));
}
- adapter.add(new EventListItem(rec));
+ adapter.add(new RecordingListItem((Recording) rec));
adapter.notifyDataSetChanged();
}
}
@Override
+ public void onBackPressed() {
+ if (stack.isEmpty()) {
+ super.onBackPressed();
+ } else {
+ stack.pop();
+ if (stack.isEmpty()) {
+ currentFolder = "";
+ } else {
+ currentFolder = stack.peek();
+ }
+ fillAdapter();
+ }
+
+ }
+
+ @Override
protected boolean finishedSuccessImpl(final List<Recording> results) {
clearCache();
- for(final Recording r :results){
- CACHE.add(r);
+ for (final Recording r : results) {
+ final String folder = r.getFolder();
+ if (folder.length() > 0) {
+ final String[] split = folder.split(Recording.FOLDERDELIMCHAR);
+ String key = null;
+ String value = null;
+ if (split.length == 1) {
+ key = Recording.ROOT_FOLDER;
+ value = split[0];
+ } else {
+ value = split[split.length - 1];
+ // StringBuilder sb = new StringBuilder();
+ // String sep = "";
+ // for(int i = 0; i < split.length - 1; ++i){
+ // sb.append(sep).append(split[i]);
+ // sep = Recording.FOLDERDELIMCHAR;
+ // }
+ key = folder.subSequence(0,
+ folder.length() - (value.length() + 1)).toString();
+
+ }
+
+ Set<String> list = FOLDERS.get(key);
+ if (list == null) {
+ list = new TreeSet<String>(new Comparator<String>() {
+ @Override
+ public int compare(final String lhs, final String rhs) {
+ return lhs.compareToIgnoreCase(rhs);
+ }
+ });
+ FOLDERS.put(key, list);
+ }
+
+ list.add(value);
+
+ // a b
+ // a
+ // c
+ // a~b > k
+
+ }
+ List<Recording> list = CACHE.get(folder);
+ if (list == null) {
+ list = new ArrayList<Recording>();
+ CACHE.put(folder, list);
+ }
+ list.add(r);
}
+
pushResultCountToTitle();
fillAdapter();
return adapter.isEmpty() == false;
-
}
@Override
@@ -317,8 +443,19 @@ implements OnItemLongClickListener {
}
@Override
- protected List<Recording> getCACHE() {
- return CACHE;
+ public void clearCache() {
+ CACHE.clear();
+ FOLDERS.clear();
}
+ @Override
+ protected List<Recording> getCACHE() {
+
+ final List<Recording> list = CACHE.get(currentFolder);
+
+ if (list != null) {
+ return list;
+ }
+ return EMPTY;
+ }
}
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/SvdrpProgressDialog.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/SvdrpProgressDialog.java
index 4c72795..74cd502 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/SvdrpProgressDialog.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/SvdrpProgressDialog.java
@@ -23,6 +23,8 @@ public class SvdrpProgressDialog<T> extends ProgressDialog implements
this.client = client;
progress = new ProgressDialog(context);
progress.setOnCancelListener(this);
+ progress.setCancelable(true);
+ progress.setCanceledOnTouchOutside(false);
}
public void svdrpEvent(final SvdrpEvent sevent) {
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/TimeEventAdapter.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/TimeEventAdapter.java
index dc7545e..f5cca64 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/TimeEventAdapter.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/TimeEventAdapter.java
@@ -4,8 +4,9 @@ import android.content.Context;
import de.bjusystems.vdrmanager.R;
import de.bjusystems.vdrmanager.data.Event;
import de.bjusystems.vdrmanager.data.EventFormatter;
+import de.bjusystems.vdrmanager.data.EventListItem;
-public class TimeEventAdapter extends EventAdapter {
+public class TimeEventAdapter extends BaseEventAdapter<EventListItem> {
public TimeEventAdapter(final Context context) {
super(context, R.layout.epg_event_item);
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/gui/VdrManagerActivity.java b/vdrmanager/src/de/bjusystems/vdrmanager/gui/VdrManagerActivity.java
index 27163e0..8a7330e 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/gui/VdrManagerActivity.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/gui/VdrManagerActivity.java
@@ -29,223 +29,221 @@ import de.bjusystems.vdrmanager.data.db.EPGSearchSuggestionsProvider;
import de.bjusystems.vdrmanager.utils.wakeup.AsyncWakeupTask;
public class VdrManagerActivity extends SherlockActivity implements
-OnClickListener, OnQueryTextListener {
+ OnClickListener, OnQueryTextListener {
- public static final String TAG = "VdrManagerActivity";
+ public static final String TAG = "VdrManagerActivity";
- public static final String VDR_PORTAL = "http://www.vdr-portal.de";
+ public static final String VDR_PORTAL = "http://www.vdr-portal.de";
- private com.actionbarsherlock.widget.SearchView search;
+ private com.actionbarsherlock.widget.SearchView search;
- private View actionMenuWakup;
+ private View actionMenuWakup;
- @Override
- protected void onCreate(final Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
+ @Override
+ protected void onCreate(final Bundle savedInstanceState) {
+ super.onCreate(savedInstanceState);
- // Preferences.initVDR(this);
+ // Preferences.initVDR(this);
- // if(Preferences.get().getCurrentVdr() == null){
- // finish();
- // return;
- // }
+ // if(Preferences.get().getCurrentVdr() == null){
+ // finish();
+ // return;
+ // }
- if (Preferences.initVDR(this) == false) {
+ if (Preferences.initVDR(this) == false) {
final Intent intent = new Intent();
- intent.setClass(this, VdrListActivity.class);
- intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- intent.putExtra(Intents.EMPTY_CONFIG, Boolean.TRUE);
- startActivity(intent);
- Toast.makeText(this, R.string.no_vdr, Toast.LENGTH_SHORT).show();
- finish();
- return;
- }
-
- Preferences.setLocale(this);
-
- // this.getActionBar().setDisplayShowCustomEnabled(true);
- // this.getActionBar().setDisplayShowTitleEnabled(false);
- // setTitle(getString(R.string.app_name));
- // attach view
- setContentView(R.layout.vdrmanager);
-
- // Preferences.loadPreferences(this);
-
- findViewById(R.id.action_menu_channels).setOnClickListener(this);
- findViewById(R.id.action_menu_recordings).setOnClickListener(this);
- findViewById(R.id.action_menu_timers).setOnClickListener(this);
- findViewById(R.id.action_menu_epg).setOnClickListener(this);
- final View v = findViewById(R.id.action_menu_search);
- if (v != null) {
- v.setOnClickListener(this);
- }
- findViewById(R.id.main_logo).setOnClickListener(this);
- actionMenuWakup = findViewById(R.id.action_menu_wakeup);
-
- // add and register buttons
- // createButtons();
- }
-
- @Override
- public boolean onCreateOptionsMenu(final com.actionbarsherlock.view.Menu menu) {
- final com.actionbarsherlock.view.MenuInflater inflater = getSupportMenuInflater();
- inflater.inflate(R.menu.main_menu, menu);
-
- // search = new SearchView(getSupportActionBar().getThemedContext());
-
- search = (SearchView) menu.findItem(R.id.menu_search).getActionView();
-
- // search = (SearchView)
- // .getActionView();
- //
- // Object o = menu.findItem(R.id.menu_search);
-
- final SearchManager searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
- search.setSearchableInfo(searchManager
- .getSearchableInfo(getComponentName()));
-
- // search.setOnQueryTextListener(this);
- return true;
- }
-
- @Override
- protected void onResume() {
- Preferences.setLocale(this);
- if (Preferences.get().isWakeupEnabled() == false) {
- actionMenuWakup.setVisibility(View.GONE);
- actionMenuWakup.setOnClickListener(null);
- } else {
- actionMenuWakup.setVisibility(View.VISIBLE);
- actionMenuWakup.setOnClickListener(this);
- }
- super.onResume();
- }
-
- @Override
- public boolean onOptionsItemSelected(
- final com.actionbarsherlock.view.MenuItem item) {
-
- switch (item.getItemId()) {
- case R.id.main_menu_preferences: {
- final Intent intent = new Intent(this, PreferencesActivity.class);
- final int flags = Intent.FLAG_ACTIVITY_NEW_TASK
- | Intent.FLAG_ACTIVITY_SINGLE_TOP
- | Intent.FLAG_ACTIVITY_CLEAR_TOP;
- intent.setFlags(flags);
- startActivity(intent);
- finish();
- break;
- }
- case R.id.main_menu_info: {
- if(!isFinishing()){
- return true;
- }
- About.show(this);
- break;
- }
- case R.id.main_menu_exit: {
- finish();
- break;
- }
-
- case R.id.main_menu_clear_search: {
- final SearchRecentSuggestions suggestions = new SearchRecentSuggestions(
- this, EPGSearchSuggestionsProvider.AUTHORITY,
- EPGSearchSuggestionsProvider.MODE);
- suggestions.clearHistory();
- break;
- }
-
- // case R.id.menu_search: {
- // if(Build.VERSION.SDK_INT <11){
- // onSearchRequested();
- // }
- // break;
- // }
- case R.id.main_menu_goto: {
- try {
- final Cursor cursor = ((AndroidDatabaseResults) DBAccess
- .get(this).getVdrDAO().iterator().getRawResults())
- .getRawCursor();
- startManagingCursor(cursor);
- final AlertDialog ad = new AlertDialog.Builder(this)
- .setSingleChoiceItems(cursor, findVdrCursor(cursor),
- "name", new DialogInterface.OnClickListener() {
-
- @Override
- public void onClick(final DialogInterface dialog,
- final int which) {
- cursor.moveToPosition(which);
- final int id = cursor.getInt(cursor
- .getColumnIndex("_id"));
- final Vdr vdr = DBAccess
- .get(VdrManagerActivity.this)
- .getVdrDAO().queryForId(id);
- if (vdr == null) {
- Toast.makeText(
- VdrManagerActivity.this,
- R.string.main_menu_goto_no_vdr,
- Toast.LENGTH_SHORT).show();
- } else {
- Preferences.setCurrentVdr(
- VdrManagerActivity.this,
- vdr);
- Toast.makeText(
- VdrManagerActivity.this,
- getString(
- R.string.main_menu_switched_to,
- vdr.getName()),
- Toast.LENGTH_SHORT).show();
- final Intent intent = getIntent();
- overridePendingTransition(0, 0);
- intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
- finish();
-
- overridePendingTransition(0, 0);
- startActivity(intent);
- }
- dialog.dismiss();
- }
- })//
- .setTitle(R.string.main_menu_goto_title)//
- .create();
- ad.show();
-
- } catch (final Exception ex) {
- Log.w(TAG, ex);
- }
-
- break;
- }
- }
- return true;
- }
-
- private int findVdrCursor(final Cursor c) {
- if (Preferences.get().getCurrentVdr() == null) {
- return -1;
- }
-
- final int cid = Preferences.get().getCurrentVdr().getId();
-
- int position = 0;
- c.moveToPosition(-1);
- while (c.moveToNext()) {
- if (c.getInt(c.getColumnIndex("_id")) == cid) {
- break;
- }
- position++;
- }
- return position;
- }
-
- @Override
- public void onBackPressed() {
- if (Preferences.get().isQuiteOnBackButton()) {
+ intent.setClass(this, VdrListActivity.class);
+ intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
+ intent.putExtra(Intents.EMPTY_CONFIG, Boolean.TRUE);
+ startActivity(intent);
+ Toast.makeText(this, R.string.no_vdr, Toast.LENGTH_SHORT).show();
+ finish();
+ return;
+ }
+
+ Preferences.setLocale(this);
+
+ // this.getActionBar().setDisplayShowCustomEnabled(true);
+ // this.getActionBar().setDisplayShowTitleEnabled(false);
+ // setTitle(getString(R.string.app_name));
+ // attach view
+ setContentView(R.layout.vdrmanager);
+
+ // Preferences.loadPreferences(this);
+
+ findViewById(R.id.action_menu_channels).setOnClickListener(this);
+ findViewById(R.id.action_menu_recordings).setOnClickListener(this);
+ findViewById(R.id.action_menu_timers).setOnClickListener(this);
+ findViewById(R.id.action_menu_epg).setOnClickListener(this);
+// View v = findViewById(R.id.action_menu_search);
+// if (v != null) {
+// v.setOnClickListener(this);
+// }
+ findViewById(R.id.main_logo).setOnClickListener(this);
+ actionMenuWakup = findViewById(R.id.action_menu_wakeup);
+
+ // add and register buttons
+ // createButtons();
+ }
+
+ public boolean onCreateOptionsMenu(com.actionbarsherlock.view.Menu menu) {
+ com.actionbarsherlock.view.MenuInflater inflater = getSupportMenuInflater();
+ inflater.inflate(R.menu.main_menu, menu);
+
+ // search = new SearchView(getSupportActionBar().getThemedContext());
+
+ search = (SearchView) menu.findItem(R.id.menu_search).getActionView();
+
+ // search = (SearchView)
+ // .getActionView();
+ //
+ // Object o = menu.findItem(R.id.menu_search);
+
+ SearchManager searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
+ search.setSearchableInfo(searchManager
+ .getSearchableInfo(getComponentName()));
+
+ // search.setOnQueryTextListener(this);
+ return true;
+ }
+
+ @Override
+ protected void onResume() {
+ Preferences.setLocale(this);
+ if (Preferences.get().isWakeupEnabled() == false) {
+ actionMenuWakup.setVisibility(View.GONE);
+ actionMenuWakup.setOnClickListener(null);
+ } else {
+ actionMenuWakup.setVisibility(View.VISIBLE);
+ actionMenuWakup.setOnClickListener(this);
+ }
+ super.onResume();
+ }
+
+ @Override
+ public boolean onOptionsItemSelected(
+ final com.actionbarsherlock.view.MenuItem item) {
+
+ switch (item.getItemId()) {
+ case R.id.main_menu_preferences: {
+ Intent intent = new Intent(this, PreferencesActivity.class);
+ int flags = Intent.FLAG_ACTIVITY_NEW_TASK
+ | Intent.FLAG_ACTIVITY_SINGLE_TOP
+ | Intent.FLAG_ACTIVITY_CLEAR_TOP;
+ intent.setFlags(flags);
+ startActivity(intent);
+ finish();
+ break;
+ }
+ case R.id.main_menu_info: {
+ if(isFinishing()){
+ break;
+ }
+ About.show(this);
+ break;
+ }
+ case R.id.main_menu_exit: {
+ finish();
+ break;
+ }
+
+ case R.id.main_menu_clear_search: {
+ SearchRecentSuggestions suggestions = new SearchRecentSuggestions(
+ this, EPGSearchSuggestionsProvider.AUTHORITY,
+ EPGSearchSuggestionsProvider.MODE);
+ suggestions.clearHistory();
+ break;
+ }
+
+ // case R.id.menu_search: {
+ // if(Build.VERSION.SDK_INT <11){
+ // onSearchRequested();
+ // }
+ // break;
+ // }
+ case R.id.main_menu_goto: {
+ try {
+ final Cursor cursor = ((AndroidDatabaseResults) DBAccess
+ .get(this).getVdrDAO().iterator().getRawResults())
+ .getRawCursor();
+ startManagingCursor(cursor);
+ final AlertDialog ad = new AlertDialog.Builder(this)
+ .setSingleChoiceItems(cursor, findVdrCursor(cursor),
+ "name", new DialogInterface.OnClickListener() {
+
+ public void onClick(DialogInterface dialog,
+ int which) {
+ cursor.moveToPosition(which);
+ int id = cursor.getInt(cursor
+ .getColumnIndex("_id"));
+ Vdr vdr = DBAccess
+ .get(VdrManagerActivity.this)
+ .getVdrDAO().queryForId(id);
+ if (vdr == null) {
+ Toast.makeText(
+ VdrManagerActivity.this,
+ R.string.main_menu_goto_no_vdr,
+ Toast.LENGTH_SHORT).show();
+ } else {
+ Preferences.setCurrentVdr(
+ VdrManagerActivity.this,
+ vdr);
+ Toast.makeText(
+ VdrManagerActivity.this,
+ getString(
+ R.string.main_menu_switched_to,
+ vdr.getName()),
+ Toast.LENGTH_SHORT).show();
+ Intent intent = getIntent();
+ overridePendingTransition(0, 0);
+ intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
+ finish();
+
+ overridePendingTransition(0, 0);
+ startActivity(intent);
+ }
+ dialog.dismiss();
+ }
+ })//
+ .setTitle(R.string.main_menu_goto_title)//
+ .create();
+ ad.show();
+
+ } catch (Exception ex) {
+ Log.w(TAG, ex);
+ }
+
+ break;
+ }
+ }
+ return true;
+ }
+
+ private int findVdrCursor(Cursor c) {
+ if (Preferences.get().getCurrentVdr() == null) {
+ return -1;
+ }
+
+ int cid = Preferences.get().getCurrentVdr().getId();
+
+ int position = 0;
+ c.moveToPosition(-1);
+ while (c.moveToNext()) {
+ if (c.getInt(c.getColumnIndex("_id")) == cid) {
+ break;
+ }
+ position++;
+ }
+ return position;
+ }
+
+ @Override
+ public void onBackPressed() {
+ if (Preferences.get().isQuiteOnBackButton()) {
finish();
} else {
- super.onBackPressed();
- }
+ super.onBackPressed();
+ }
try {
// reassign a new and empty key store
@@ -254,70 +252,69 @@ OnClickListener, OnQueryTextListener {
Log.e(getClass().getName(), "Can't clear session key store");
}
- }
-
- public void startActivity(final Class<?> clazz) {
- final Intent intent = new Intent(Intent.ACTION_VIEW);
- intent.setClass(this, clazz);
- startActivity(intent);
- }
-
- @Override
- public void onClick(final View v) {
- final int id = v.getId();
-
- switch (id) {
- case R.id.action_menu_channels:
- startActivity(ChannelListActivity.class);
- break;
- case R.id.action_menu_recordings:
- startActivity(RecordingListActivity.class);
- break;
- case R.id.action_menu_timers:
- startActivity(TimerListActivity.class);
- break;
- case R.id.action_menu_epg:
- startActivity(TimeEpgListActivity.class);
- break;
- case R.id.action_menu_search:
- onSearchRequested();
- break;
- case R.id.action_menu_wakeup:
- final AsyncWakeupTask wakeupTask = new AsyncWakeupTask(this);
- wakeupTask.execute();
- break;
- case R.id.main_logo:
- final Intent i = new Intent(Intent.ACTION_VIEW);
- i.setData(Uri.parse(VDR_PORTAL));
- startActivity(i);
- break;
- }
-
- }
-
- protected void startSearchManager() {
- final Bundle appData = new Bundle();
- startSearch(null, false, appData, false);
- }
-
- @Override
- public boolean onSearchRequested() {
- search.setVisibility(View.VISIBLE);
- // Bundle appData = new Bundle();
- // appData.putBoolean(SearchableActivity.JARGON, true);
- // startSearch(null, false, appData, false);
- return true;
- }
-
- @Override
- public boolean onQueryTextSubmit(final String query) {
- // TODO Auto-generated method stub
- return false;
- }
-
- @Override
- public boolean onQueryTextChange(final String newText) {
- // TODO Auto-generated method stub
- return false;
- }
+ }
+
+ public void startActivity(Class<?> clazz) {
+ Intent intent = new Intent(Intent.ACTION_VIEW);
+ intent.setClass(this, clazz);
+ startActivity(intent);
+ }
+
+ public void onClick(View v) {
+ int id = v.getId();
+
+ switch (id) {
+ case R.id.action_menu_channels:
+ startActivity(ChannelListActivity.class);
+ break;
+ case R.id.action_menu_recordings:
+ startActivity(RecordingListActivity.class);
+ break;
+ case R.id.action_menu_timers:
+ startActivity(TimerListActivity.class);
+ break;
+ case R.id.action_menu_epg:
+ startActivity(TimeEpgListActivity.class);
+ break;
+// case R.id.action_menu_search:
+// onSearchRequested();
+// break;
+ case R.id.action_menu_wakeup:
+ final AsyncWakeupTask wakeupTask = new AsyncWakeupTask(this);
+ wakeupTask.execute();
+ break;
+ case R.id.main_logo:
+ Intent i = new Intent(Intent.ACTION_VIEW);
+ i.setData(Uri.parse(VDR_PORTAL));
+ startActivity(i);
+ break;
+ }
+
+ }
+
+ protected void startSearchManager() {
+ Bundle appData = new Bundle();
+ startSearch(null, false, appData, false);
+ }
+
+ @Override
+ public boolean onSearchRequested() {
+ search.setVisibility(View.VISIBLE);
+ // Bundle appData = new Bundle();
+ // appData.putBoolean(SearchableActivity.JARGON, true);
+ // startSearch(null, false, appData, false);
+ return true;
+ }
+
+ @Override
+ public boolean onQueryTextSubmit(String query) {
+ // TODO Auto-generated method stub
+ return false;
+ }
+
+ @Override
+ public boolean onQueryTextChange(String newText) {
+ // TODO Auto-generated method stub
+ return false;
+ }
}
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpAsyncTask.java b/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpAsyncTask.java
index 224ca48..505b084 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpAsyncTask.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpAsyncTask.java
@@ -4,7 +4,6 @@ import java.util.ArrayList;
import java.util.List;
import android.os.AsyncTask;
-import de.bjusystems.vdrmanager.utils.svdrp.CertificateProblemListener.CertificateProblemAction;
public class SvdrpAsyncTask<Result, Client extends SvdrpClient<Result>> extends
AsyncTask<Void, Object, Void> implements SvdrpListener,
@@ -22,9 +21,9 @@ SvdrpExceptionListener, SvdrpResultListener<Result> {
List<SvdrpFinishedListener<Result>> finishedListeners = new ArrayList<SvdrpFinishedListener<Result>>();
- CertificateProblemListener certificateProblemListener = null;
+ //CertificateProblemListener certificateProblemListener = null;
- CertificateProblemAction certificateProblemAction;
+ //CertificateProblemAction certificateProblemAction;
public SvdrpAsyncTask(final Client client) {
this.client = client;
@@ -159,4 +158,8 @@ SvdrpExceptionListener, SvdrpResultListener<Result> {
public void svdrpEvent(final Result result) {
results.add(result);
}
+
+ public void abort(){
+ client.abort();
+ }
}
diff --git a/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpClient.java b/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpClient.java
index ba87100..e269a5b 100644
--- a/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpClient.java
+++ b/vdrmanager/src/de/bjusystems/vdrmanager/utils/svdrp/SvdrpClient.java
@@ -12,6 +12,8 @@ import java.util.ArrayList;
import java.util.List;
import java.util.Timer;
import java.util.TimerTask;
+import java.util.zip.GZIPInputStream;
+import java.util.zip.InflaterInputStream;
import android.util.Log;
import de.bjusystems.vdrmanager.app.C;
@@ -374,14 +376,39 @@ public abstract class SvdrpClient<Result> {
return;
}
+ // activate compression
+ Log.i(TAG, "Activate compression");
+ writeLine("compress");
+
// send command
informListener(SvdrpEvent.COMMAND_SENDING);
writeLine(command);
informListener(SvdrpEvent.COMMAND_SENT);
Log.i(TAG, SvdrpEvent.COMMAND_SENT + ":" + command);
- // read first line
+ // get the answer for the compress command or
+ // the first line of the answer for the command
String line = readLine();
+ if (line.startsWith("!OK")) {
+ final String[] words = line.split(" ");
+ if (words.length > 1) {
+ final String mode = words[1].toUpperCase();
+ if (mode.equals("ZLIB")) {
+ Log.i(TAG, "ZLIB compression activated");
+ inputStream = new InflaterInputStream(inputStream);
+ } else if (mode.equals("GZIP")) {
+ Log.i(TAG, "GZIP compression activated");
+ inputStream = new GZIPInputStream(inputStream);
+ } else {
+ Log.i(TAG, "NO compression activated");
+ }
+ }
+ line = readLine();
+ } else {
+ Log.i(TAG, "NO compression activated");
+ }
+
+ // correct answer?
if (!line.startsWith("START")) {
Log.w(TAG, line);
throw new IOException("Answer not wellformed: " + line);