diff options
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 Binary files differnew file mode 100644 index 0000000..5b3fcec --- /dev/null +++ b/vdrmanager/res/drawable/folder.png 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<herrlado@gmail.com> - herrlado<herrlado@gmail.com> - </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<herrlado@gmail.com> +herrlado<herrlado@gmail.com></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 -> %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); |