mp3.c
author nathan
Tue, 03 Feb 2009 20:33:04 +0800
branchtrunk
changeset 22 93aaf15c145a
parent 19 306cc35c7faa
child 23 3b14b8aacaa0
permissions -rw-r--r--
remove compatibility for VDR < 1.4.5
nathan@0
     1
/*
nathan@0
     2
 * MP3/MPlayer plugin to VDR (C++)
nathan@0
     3
 *
nathan@22
     4
 * (C) 2001-2009 Stefan Huelswitt <s.huelswitt@gmx.de>
nathan@0
     5
 *
nathan@0
     6
 * This code is free software; you can redistribute it and/or
nathan@0
     7
 * modify it under the terms of the GNU General Public License
nathan@0
     8
 * as published by the Free Software Foundation; either version 2
nathan@0
     9
 * of the License, or (at your option) any later version.
nathan@0
    10
 *
nathan@0
    11
 * This code is distributed in the hope that it will be useful,
nathan@0
    12
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
nathan@0
    13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
nathan@0
    14
 * GNU General Public License for more details.
nathan@0
    15
 *
nathan@0
    16
 * You should have received a copy of the GNU General Public License
nathan@0
    17
 * along with this program; if not, write to the Free Software
nathan@0
    18
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
nathan@0
    19
 * Or, point your browser to http://www.gnu.org/copyleft/gpl.html
nathan@0
    20
 */
nathan@0
    21
nathan@0
    22
#include <stdlib.h>
nathan@0
    23
#include <getopt.h>
nathan@0
    24
#include <strings.h>
nathan@0
    25
#include <typeinfo>
nathan@0
    26
nathan@0
    27
#include "common.h"
nathan@0
    28
nathan@0
    29
#include <vdr/menuitems.h>
nathan@0
    30
#include <vdr/status.h>
nathan@0
    31
#include <vdr/plugin.h>
nathan@0
    32
#include <vdr/interface.h>
nathan@0
    33
#include <vdr/skins.h>
nathan@0
    34
nathan@0
    35
#include "setup.h"
nathan@0
    36
#include "setup-mp3.h"
nathan@0
    37
#include "data-mp3.h"
nathan@0
    38
#include "data-src.h"
nathan@0
    39
#include "player-mp3.h"
nathan@0
    40
#include "menu.h"
nathan@0
    41
#include "menu-async.h"
nathan@0
    42
#include "decoder.h"
nathan@0
    43
#include "i18n.h"
nathan@0
    44
#include "version.h"
nathan@0
    45
#include "service.h"
nathan@0
    46
nathan@0
    47
#ifdef DEBUG
nathan@0
    48
#include <mad.h>
nathan@0
    49
#endif
nathan@0
    50
nathan@0
    51
const char *sourcesSub=0;
nathan@0
    52
cFileSources MP3Sources;
nathan@0
    53
nathan@2
    54
static const char *plugin_name=0;
nathan@12
    55
const char *i18n_name=0;
nathan@2
    56
nathan@0
    57
// --- cMenuSetupMP3 --------------------------------------------------------
nathan@0
    58
nathan@0
    59
class cMenuSetupMP3 : public cMenuSetupPage {
nathan@0
    60
private:
nathan@0
    61
  cMP3Setup data;
nathan@0
    62
  //
nathan@0
    63
  const char *cddb[3], *disp[2], *scan[3], *bgr[3];
nathan@0
    64
  const char *aout[AUDIOOUTMODES];
nathan@0
    65
  int amode, amodes[AUDIOOUTMODES];
nathan@0
    66
protected:
nathan@0
    67
  virtual void Store(void);
nathan@0
    68
public:
nathan@0
    69
  cMenuSetupMP3(void);
nathan@0
    70
  };
nathan@0
    71
nathan@0
    72
cMenuSetupMP3::cMenuSetupMP3(void)
nathan@0
    73
{
nathan@0
    74
  static const char allowed[] = { "abcdefghijklmnopqrstuvwxyz0123456789-_" };
nathan@0
    75
  int numModes=0;
nathan@2
    76
  aout[numModes]=trVDR("DVB"); amodes[numModes]=AUDIOOUTMODE_DVB; numModes++;
nathan@0
    77
#ifdef WITH_OSS
nathan@0
    78
  aout[numModes]=tr("OSS"); amodes[numModes]=AUDIOOUTMODE_OSS; numModes++;
nathan@0
    79
#endif
nathan@0
    80
  data=MP3Setup;
nathan@0
    81
  amode=0;
nathan@0
    82
  for(int i=0; i<numModes; i++)
nathan@0
    83
    if(amodes[i]==data.AudioOutMode) { amode=i; break; }
nathan@0
    84
nathan@0
    85
  SetSection(tr("MP3"));
nathan@0
    86
  Add(new cMenuEditStraItem(tr("Setup.MP3$Audio output mode"),     &amode,numModes,aout));
nathan@0
    87
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Audio mode"),            &data.AudioMode, tr("Round"), tr("Dither")));
nathan@0
    88
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Use 48kHz mode only"),   &data.Only48kHz));
nathan@0
    89
  disp[0]=tr("classic");
nathan@0
    90
  disp[1]=tr("via skin");
nathan@0
    91
  Add(new cMenuEditStraItem(tr("Setup.MP3$Replay display"),        &data.ReplayDisplay, 2, disp));
nathan@0
    92
  Add(new cMenuEditIntItem( tr("Setup.MP3$Display mode"),          &data.DisplayMode, 1, 3));
nathan@0
    93
  bgr[0]=tr("Black");
nathan@0
    94
  bgr[1]=tr("Live");
nathan@0
    95
  bgr[2]=tr("Images");
nathan@0
    96
  Add(new cMenuEditStraItem(tr("Setup.MP3$Background mode"),       &data.BackgrMode, 3, bgr));
nathan@0
    97
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Initial loop mode"),     &data.InitLoopMode));
nathan@0
    98
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Initial shuffle mode"),  &data.InitShuffleMode));
nathan@0
    99
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Abort player at end of list"),&data.AbortAtEOL));
nathan@0
   100
  scan[0]=tr("disabled");
nathan@0
   101
  scan[1]=tr("ID3 only");
nathan@0
   102
  scan[2]=tr("ID3 & Level");
nathan@0
   103
  Add(new cMenuEditStraItem(tr("Setup.MP3$Background scan"),       &data.BgrScan, 3, scan));
nathan@0
   104
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Editor display mode"),   &data.EditorMode, tr("Filenames"), tr("ID3 names")));
nathan@0
   105
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Mainmenu mode"),         &data.MenuMode, tr("Playlists"), tr("Browser")));
nathan@0
   106
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Keep selection menu"),   &data.KeepSelect));
nathan@0
   107
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Title/Artist order"),    &data.TitleArtistOrder, tr("Normal"), tr("Reversed")));
nathan@0
   108
  Add(new cMenuEditBoolItem(tr("Hide mainmenu entry"),             &data.HideMainMenu));
nathan@0
   109
  Add(new cMenuEditIntItem( tr("Setup.MP3$Normalizer level"),      &data.TargetLevel, 0, MAX_TARGET_LEVEL));
nathan@0
   110
  Add(new cMenuEditIntItem( tr("Setup.MP3$Limiter level"),         &data.LimiterLevel, MIN_LIMITER_LEVEL, 100));
nathan@0
   111
  Add(new cMenuEditBoolItem(tr("Setup.MP3$Use HTTP proxy"),        &data.UseProxy));
nathan@0
   112
  Add(new cMenuEditStrItem( tr("Setup.MP3$HTTP proxy host"),       data.ProxyHost,MAX_HOSTNAME,allowed));
nathan@0
   113
  Add(new cMenuEditIntItem( tr("Setup.MP3$HTTP proxy port"),       &data.ProxyPort,1,65535));
nathan@0
   114
  cddb[0]=tr("disabled");
nathan@0
   115
  cddb[1]=tr("local only");
nathan@0
   116
  cddb[2]=tr("local&remote");
nathan@0
   117
  Add(new cMenuEditStraItem(tr("Setup.MP3$CDDB for CD-Audio"),     &data.UseCddb,3,cddb));
nathan@0
   118
  Add(new cMenuEditStrItem( tr("Setup.MP3$CDDB server"),           data.CddbHost,MAX_HOSTNAME,allowed));
nathan@0
   119
  Add(new cMenuEditIntItem( tr("Setup.MP3$CDDB port"),             &data.CddbPort,1,65535));
nathan@0
   120
}
nathan@0
   121
nathan@0
   122
void cMenuSetupMP3::Store(void)
nathan@0
   123
{
nathan@0
   124
  data.AudioOutMode=amodes[amode];
nathan@0
   125
nathan@0
   126
  MP3Setup=data;
nathan@0
   127
  SetupStore("InitLoopMode",     MP3Setup.InitLoopMode   );
nathan@0
   128
  SetupStore("InitShuffleMode",  MP3Setup.InitShuffleMode);
nathan@0
   129
  SetupStore("AudioMode",        MP3Setup.AudioMode      );
nathan@0
   130
  SetupStore("AudioOutMode",     MP3Setup.AudioOutMode   );
nathan@0
   131
  SetupStore("BgrScan",          MP3Setup.BgrScan        );
nathan@0
   132
  SetupStore("EditorMode",       MP3Setup.EditorMode     );
nathan@0
   133
  SetupStore("DisplayMode",      MP3Setup.DisplayMode    );
nathan@0
   134
  SetupStore("BackgrMode",       MP3Setup.BackgrMode     );
nathan@0
   135
  SetupStore("MenuMode",         MP3Setup.MenuMode       );
nathan@0
   136
  SetupStore("TargetLevel",      MP3Setup.TargetLevel    );
nathan@0
   137
  SetupStore("LimiterLevel",     MP3Setup.LimiterLevel   );
nathan@0
   138
  SetupStore("Only48kHz",        MP3Setup.Only48kHz      );
nathan@0
   139
  SetupStore("UseProxy",         MP3Setup.UseProxy       );
nathan@0
   140
  SetupStore("ProxyHost",        MP3Setup.ProxyHost      );
nathan@0
   141
  SetupStore("ProxyPort",        MP3Setup.ProxyPort      );
nathan@0
   142
  SetupStore("UseCddb",          MP3Setup.UseCddb        );
nathan@0
   143
  SetupStore("CddbHost",         MP3Setup.CddbHost       );
nathan@0
   144
  SetupStore("CddbPort",         MP3Setup.CddbPort       );
nathan@0
   145
  SetupStore("AbortAtEOL",       MP3Setup.AbortAtEOL     );
nathan@0
   146
  SetupStore("ReplayDisplay",    MP3Setup.ReplayDisplay  );
nathan@0
   147
  SetupStore("HideMainMenu",     MP3Setup.HideMainMenu   );
nathan@0
   148
  SetupStore("KeepSelect",       MP3Setup.KeepSelect     );
nathan@0
   149
  SetupStore("TitleArtistOrder", MP3Setup.TitleArtistOrder);
nathan@0
   150
}
nathan@0
   151
nathan@0
   152
// --- cAsyncStatus ------------------------------------------------------------
nathan@0
   153
nathan@0
   154
cAsyncStatus asyncStatus;
nathan@0
   155
nathan@0
   156
cAsyncStatus::cAsyncStatus(void)
nathan@0
   157
{
nathan@0
   158
  text=0;
nathan@0
   159
  changed=false;
nathan@0
   160
}
nathan@0
   161
nathan@0
   162
cAsyncStatus::~cAsyncStatus()
nathan@0
   163
{
nathan@0
   164
  free((void *)text);
nathan@0
   165
}
nathan@0
   166
nathan@0
   167
void cAsyncStatus::Set(const char *Text)
nathan@0
   168
{
nathan@0
   169
  Lock();
nathan@0
   170
  free((void *)text);
nathan@0
   171
  text=Text ? strdup(Text) : 0;
nathan@0
   172
  changed=true;
nathan@0
   173
  Unlock();
nathan@0
   174
}
nathan@0
   175
nathan@0
   176
const char *cAsyncStatus::Begin(void)
nathan@0
   177
{
nathan@0
   178
  Lock();
nathan@0
   179
  return text;
nathan@0
   180
}
nathan@0
   181
nathan@0
   182
void cAsyncStatus::Finish(void)
nathan@0
   183
{
nathan@0
   184
  changed=false;
nathan@0
   185
  Unlock();
nathan@0
   186
}
nathan@0
   187
nathan@0
   188
// --- --------------------------------------------------------------------
nathan@0
   189
nathan@0
   190
static const char *TitleArtist(const char *title, const char *artist)
nathan@0
   191
{
nathan@0
   192
  static char buf[256]; // clearly not multi-thread save!
nathan@6
   193
  const char *fmt;
nathan@0
   194
  if(artist && artist[0]) {
nathan@0
   195
    if(MP3Setup.TitleArtistOrder) fmt="%2$s - %1$s";
nathan@0
   196
    else  fmt="%s - %s";
nathan@0
   197
    }
nathan@0
   198
  else fmt="%s";
nathan@0
   199
  snprintf(buf,sizeof(buf),fmt,title,artist);
nathan@0
   200
  return buf;
nathan@0
   201
}
nathan@0
   202
nathan@0
   203
// --- cMP3Control --------------------------------------------------------
nathan@0
   204
nathan@0
   205
#define MAXROWS 120
nathan@0
   206
nathan@0
   207
class cMP3Control : public cControl {
nathan@0
   208
private:
nathan@0
   209
  cOsd *osd;
nathan@0
   210
  const cFont *font;
nathan@0
   211
  cSkinDisplayReplay *disp;
nathan@0
   212
  int bw, bh, bwc, fw, fh;
nathan@0
   213
  //
nathan@0
   214
  cMP3Player *player;
nathan@0
   215
  bool visible, shown, bigwin, statusActive;
nathan@0
   216
  time_t timeoutShow, greentime, oktime;
nathan@0
   217
  int lastkeytime, number;
nathan@7
   218
  bool selecting;
nathan@0
   219
  //
nathan@0
   220
  cMP3PlayInfo *lastMode;
nathan@0
   221
  time_t fliptime, listtime;
nathan@0
   222
  int hashlist[MAXROWS];
nathan@0
   223
  int flip, flipint, top, rows;
nathan@0
   224
  int lastIndex, lastTotal, lastTop;
nathan@0
   225
  int framesPerSecond;
nathan@0
   226
  //
nathan@0
   227
  bool jumpactive, jumphide, jumpsecs;
nathan@0
   228
  int jumpmm;
nathan@0
   229
  //
nathan@0
   230
  void ShowTimed(int Seconds=0);
nathan@0
   231
  void ShowProgress(bool open=false, bool bigWin=false);
nathan@0
   232
  void ShowStatus(bool force);
nathan@0
   233
  void HideStatus(void);
nathan@0
   234
  void DisplayInfo(const char *s=0);
nathan@0
   235
  void JumpDisplay(void);
nathan@0
   236
  void JumpProcess(eKeys Key);
nathan@0
   237
  void Jump(void);
nathan@0
   238
  void Stop(void);
nathan@22
   239
  void Write(int x, int y, int w, const char *text, int fg=clrWhite, int bg=clrGray50);
nathan@22
   240
  void Fill(int x, int y, int w, int h, int fg);
nathan@0
   241
  inline void Flush(void);
nathan@0
   242
public:
nathan@0
   243
  cMP3Control(void);
nathan@0
   244
  virtual ~cMP3Control();
nathan@0
   245
  virtual eOSState ProcessKey(eKeys Key);
nathan@0
   246
  virtual void Show(void) { ShowTimed(); }
nathan@0
   247
  virtual void Hide(void);
nathan@0
   248
  bool Visible(void) { return visible; }
nathan@0
   249
  static bool SetPlayList(cPlayList *plist);
nathan@0
   250
  };
nathan@0
   251
nathan@0
   252
cMP3Control::cMP3Control(void)
nathan@0
   253
:cControl(player=new cMP3Player)
nathan@0
   254
{
nathan@7
   255
  visible=shown=bigwin=selecting=jumpactive=jumphide=statusActive=false;
nathan@0
   256
  timeoutShow=greentime=oktime=0;
nathan@0
   257
  lastkeytime=number=0;
nathan@0
   258
  lastMode=0;
nathan@0
   259
  framesPerSecond=SecondsToFrames(1);
nathan@0
   260
  osd=0; disp=0;
nathan@0
   261
  font=cFont::GetFont(fontOsd);
nathan@0
   262
  cStatus::MsgReplaying(this,"MP3",0,true);
nathan@0
   263
}
nathan@0
   264
nathan@0
   265
cMP3Control::~cMP3Control()
nathan@0
   266
{
nathan@0
   267
  delete lastMode;
nathan@0
   268
  Hide();
nathan@0
   269
  Stop();
nathan@0
   270
}
nathan@0
   271
nathan@0
   272
void cMP3Control::Stop(void)
nathan@0
   273
{
nathan@0
   274
  cStatus::MsgReplaying(this,0,0,false);
nathan@0
   275
  delete player; player=0;
nathan@0
   276
  mgr->Halt();
nathan@0
   277
  mgr->Flush(); //XXX remove later
nathan@0
   278
}
nathan@0
   279
nathan@0
   280
bool cMP3Control::SetPlayList(cPlayList *plist)
nathan@0
   281
{
nathan@0
   282
  bool res;
nathan@0
   283
  cControl *control=cControl::Control();
nathan@0
   284
  // is there a running MP3 player?
nathan@0
   285
  if(control && typeid(*control)==typeid(cMP3Control)) {
nathan@0
   286
    // add songs to running playlist
nathan@0
   287
    mgr->Add(plist);
nathan@0
   288
    res=true;
nathan@0
   289
    }
nathan@0
   290
  else {
nathan@0
   291
    mgr->Flush();
nathan@0
   292
    mgr->Add(plist);
nathan@0
   293
    cControl::Launch(new cMP3Control);
nathan@0
   294
    res=false;
nathan@0
   295
    }
nathan@0
   296
  delete plist;
nathan@0
   297
  return res;
nathan@0
   298
}
nathan@0
   299
nathan@0
   300
void cMP3Control::ShowTimed(int Seconds)
nathan@0
   301
{
nathan@0
   302
  if(!visible) {
nathan@0
   303
    ShowProgress(true);
nathan@7
   304
    timeoutShow=(Seconds>0) ? time(0)+Seconds : 0;
nathan@0
   305
    }
nathan@0
   306
}
nathan@0
   307
nathan@0
   308
void cMP3Control::Hide(void)
nathan@0
   309
{
nathan@0
   310
  HideStatus();
nathan@7
   311
  timeoutShow=0;
nathan@0
   312
  if(visible) {
nathan@0
   313
    delete osd; osd=0;
nathan@0
   314
    delete disp; disp=0;
nathan@0
   315
    visible=bigwin=false;
nathan@0
   316
#if APIVERSNUM >= 10500
nathan@0
   317
    SetNeedsFastResponse(false);
nathan@0
   318
#else
nathan@0
   319
    needsFastResponse=false;
nathan@0
   320
#endif
nathan@0
   321
    }
nathan@0
   322
}
nathan@0
   323
nathan@0
   324
void cMP3Control::ShowStatus(bool force)
nathan@0
   325
{
nathan@0
   326
  if((asyncStatus.Changed() || (force && !statusActive)) && !jumpactive) {
nathan@0
   327
    const char *text=asyncStatus.Begin();
nathan@0
   328
    if(text) {
nathan@0
   329
      if(MP3Setup.ReplayDisplay || !osd) {
nathan@0
   330
        if(statusActive) Skins.Message(mtStatus,0);
nathan@0
   331
        Skins.Message(mtStatus,text);
nathan@0
   332
        }
nathan@0
   333
      else {
nathan@0
   334
        if(!statusActive) osd->SaveRegion(0,bh-2*fh,bw-1,bh-fh-1);
nathan@0
   335
        osd->DrawText(0,bh-2*fh,text,clrBlack,clrCyan,font,bw,fh,taCenter);
nathan@0
   336
        osd->Flush();
nathan@0
   337
        }
nathan@0
   338
      statusActive=true;
nathan@0
   339
      }
nathan@0
   340
    else
nathan@0
   341
      HideStatus();
nathan@0
   342
    asyncStatus.Finish();
nathan@0
   343
    }
nathan@0
   344
}
nathan@0
   345
nathan@0
   346
void cMP3Control::HideStatus(void)
nathan@0
   347
{
nathan@0
   348
  if(statusActive) {
nathan@0
   349
    if(MP3Setup.ReplayDisplay || !osd)
nathan@0
   350
      Skins.Message(mtStatus,0);
nathan@0
   351
    else {
nathan@0
   352
      osd->RestoreRegion();
nathan@0
   353
      osd->Flush();
nathan@0
   354
      }
nathan@0
   355
    }
nathan@0
   356
  statusActive=false;
nathan@0
   357
}
nathan@0
   358
nathan@0
   359
#define CTAB    11 // some tabbing values for the progress display
nathan@0
   360
#define CTAB2   5
nathan@0
   361
nathan@22
   362
void cMP3Control::Write(int x, int y, int w, const char *text, int fg, int bg)
nathan@22
   363
{
nathan@0
   364
  if(osd) {
nathan@0
   365
    x*=fw; if(x<0) x+=bw;
nathan@0
   366
    y*=fh; if(y<0) y+=bh;
nathan@0
   367
    osd->DrawText(x,y,text,fg,bg,font,w*fw);
nathan@22
   368
    }
nathan@22
   369
}
nathan@22
   370
nathan@22
   371
void cMP3Control::Fill(int x, int y, int w, int h, int fg)
nathan@22
   372
{
nathan@0
   373
  if(osd) {
nathan@0
   374
    x*=fw; if(x<0) x+=bw;
nathan@0
   375
    y*=fh; if(y<0) y+=bh;
nathan@0
   376
    osd->DrawRectangle(x,y,x+w*fw-1,y+h*fh-1,fg);
nathan@22
   377
    }
nathan@0
   378
}
nathan@0
   379
nathan@0
   380
void cMP3Control::Flush(void)
nathan@0
   381
{
nathan@0
   382
  if(MP3Setup.ReplayDisplay) Skins.Flush();
nathan@0
   383
  else if(osd) osd->Flush();
nathan@0
   384
}
nathan@0
   385
nathan@0
   386
void cMP3Control::ShowProgress(bool open, bool bigWin)
nathan@0
   387
{
nathan@0
   388
  int index, total;
nathan@0
   389
nathan@0
   390
  if(player->GetIndex(index,total) && total>=0) {
nathan@0
   391
    if(!visible && open) {
nathan@0
   392
      HideStatus();
nathan@0
   393
      if(MP3Setup.ReplayDisplay) {
nathan@0
   394
        disp=Skins.Current()->DisplayReplay(false);
nathan@0
   395
        if(!disp) return;
nathan@0
   396
        bigWin=false;
nathan@0
   397
        }
nathan@0
   398
      else {
nathan@0
   399
        int bt, bp;
nathan@0
   400
        fw=font->Width(' ')*2;
nathan@0
   401
        fh=font->Height();
nathan@0
   402
        if(bigWin) {
nathan@0
   403
          bw=Setup.OSDWidth;
nathan@0
   404
          bh=Setup.OSDHeight;
nathan@0
   405
          bt=0;
nathan@0
   406
          bp=2*fh;
nathan@0
   407
          rows=(bh-bp-fh/3)/fh;
nathan@0
   408
          }
nathan@0
   409
        else {
nathan@0
   410
          bw=Setup.OSDWidth;
nathan@0
   411
          bh=bp=2*fh;
nathan@0
   412
          bt=Setup.OSDHeight-bh;
nathan@0
   413
          rows=0;
nathan@0
   414
          }
nathan@0
   415
        bwc=bw/fw+1;
nathan@0
   416
        //d(printf("mp3: bw=%d bh=%d bt=%d bp=%d bwc=%d rows=%d fw=%d fh=%d\n",
nathan@0
   417
        //  bw,bh,bt,bp,bwc,rows,fw,fh))
nathan@0
   418
        osd=cOsdProvider::NewOsd(Setup.OSDLeft,Setup.OSDTop+bt);
nathan@0
   419
        if(!osd) return;
nathan@0
   420
        if(bigWin) {
nathan@0
   421
          tArea Areas[] = { { 0,0,bw-1,bh-bp-1,2 }, { 0,bh-bp,bw-1,bh-1,4 } };
nathan@0
   422
          osd->SetAreas(Areas,sizeof(Areas)/sizeof(tArea));
nathan@0
   423
          }
nathan@0
   424
        else {
nathan@0
   425
          tArea Areas[] = { { 0,0,bw-1,bh-1,4 } };
nathan@0
   426
          osd->SetAreas(Areas,sizeof(Areas)/sizeof(tArea));
nathan@0
   427
          }
nathan@0
   428
        osd->DrawRectangle(0,0,bw-1,bh-1,clrGray50);
nathan@0
   429
        osd->Flush();
nathan@0
   430
        }
nathan@0
   431
      ShowStatus(true);
nathan@0
   432
      bigwin=bigWin;
nathan@0
   433
      visible=true;
nathan@0
   434
#if APIVERSNUM >= 10500
nathan@0
   435
      SetNeedsFastResponse(true);
nathan@0
   436
#else
nathan@0
   437
      needsFastResponse=true;
nathan@0
   438
#endif
nathan@0
   439
      fliptime=listtime=0; flipint=0; flip=-1; top=lastTop=-1; lastIndex=lastTotal=-1;
nathan@0
   440
      delete lastMode; lastMode=0;
nathan@0
   441
      }
nathan@0
   442
nathan@0
   443
    cMP3PlayInfo *mode=new cMP3PlayInfo;
nathan@0
   444
    bool valid=mgr->Info(-1,mode);
nathan@0
   445
    bool changed=(!lastMode || mode->Hash!=lastMode->Hash);
nathan@0
   446
    char buf[256];
nathan@0
   447
    if(changed) { d(printf("mp3-ctrl: mode change detected\n")) }
nathan@0
   448
nathan@0
   449
    if(valid) { // send progress to status monitor
nathan@0
   450
      if(changed || mode->Loop!=lastMode->Loop || mode->Shuffle!=lastMode->Shuffle) {
nathan@0
   451
        snprintf(buf,sizeof(buf),"[%c%c] (%d/%d) %s",
nathan@0
   452
                  mode->Loop?'L':'.',mode->Shuffle?'S':'.',mode->Num,mode->MaxNum,TitleArtist(mode->Title,mode->Artist));
nathan@0
   453
        cStatus::MsgReplaying(this,buf,mode->Filename[0]?mode->Filename:0,true);
nathan@0
   454
        }
nathan@0
   455
      }
nathan@0
   456
nathan@0
   457
    if(visible) { // refresh the OSD progress display
nathan@0
   458
      bool flush=false;
nathan@0
   459
nathan@0
   460
      if(MP3Setup.ReplayDisplay) {
nathan@0
   461
        if(!statusActive) {
nathan@0
   462
          if(total>0) disp->SetProgress(index,total);
nathan@0
   463
          disp->SetCurrent(IndexToHMSF(index));
nathan@0
   464
          disp->SetTotal(IndexToHMSF(total));
nathan@0
   465
          bool Play, Forward;
nathan@0
   466
          int Speed;
nathan@0
   467
          if(GetReplayMode(Play,Forward,Speed)) 
nathan@0
   468
            disp->SetMode(Play, Forward, Speed);
nathan@0
   469
          flush=true;
nathan@0
   470
          }
nathan@0
   471
        }
nathan@0
   472
      else {
nathan@0
   473
        if(!selecting && changed && !statusActive) {
nathan@0
   474
          snprintf(buf,sizeof(buf),"(%d/%d)",mode->Num,mode->MaxNum);
nathan@0
   475
          Write(0,-2,CTAB,buf);
nathan@0
   476
          flush=true;
nathan@0
   477
          }
nathan@0
   478
nathan@0
   479
        if(!lastMode || mode->Loop!=lastMode->Loop) {
nathan@0
   480
          if(mode->Loop) Write(-4,-1,0,"L",clrBlack,clrYellow);
nathan@22
   481
          else Fill(-4,-1,2,1,clrGray50);
nathan@0
   482
          flush=true;
nathan@0
   483
          }
nathan@0
   484
        if(!lastMode || mode->Shuffle!=lastMode->Shuffle) {
nathan@0
   485
          if(mode->Shuffle) Write(-2,-1,0,"S",clrWhite,clrRed);
nathan@22
   486
          else Fill(-2,-1,2,1,clrGray50);
nathan@0
   487
          flush=true;
nathan@0
   488
          }
nathan@0
   489
nathan@0
   490
        index/=framesPerSecond; total/=framesPerSecond;
nathan@0
   491
        if(index!=lastIndex || total!=lastTotal) {
nathan@0
   492
          if(total>0) {
nathan@0
   493
            cProgressBar ProgressBar(bw-(CTAB+CTAB2)*fw,fh,index,total);
nathan@0
   494
            osd->DrawBitmap(CTAB*fw,bh-fh,ProgressBar);
nathan@0
   495
            }
nathan@0
   496
          snprintf(buf,sizeof(buf),total?"%02d:%02d/%02d:%02d":"%02d:%02d",index/60,index%60,total/60,total%60);
nathan@0
   497
          Write(0,-1,11,buf);
nathan@0
   498
          flush=true;
nathan@0
   499
          }
nathan@22
   500
        }
nathan@0
   501
nathan@0
   502
      if(!jumpactive) {
nathan@0
   503
        bool doflip=false;
nathan@0
   504
        if(MP3Setup.ReplayDisplay && (!lastMode || mode->Loop!=lastMode->Loop || mode->Shuffle!=lastMode->Shuffle))
nathan@0
   505
          doflip=true;
nathan@0
   506
        if(!valid || changed) {
nathan@0
   507
          fliptime=time(0); flip=0;
nathan@0
   508
	  doflip=true;
nathan@0
   509
	  }
nathan@0
   510
        else if(time(0)>fliptime+flipint) {
nathan@0
   511
	  fliptime=time(0);
nathan@0
   512
	  flip++; if(flip>=MP3Setup.DisplayMode) flip=0;
nathan@0
   513
          doflip=true;
nathan@0
   514
	  }
nathan@0
   515
        if(doflip) {
nathan@0
   516
          buf[0]=0;
nathan@0
   517
          switch(flip) {
nathan@0
   518
	    default:
nathan@0
   519
	      flip=0;
nathan@0
   520
	      // fall through
nathan@0
   521
	    case 0:
nathan@0
   522
	      snprintf(buf,sizeof(buf),"%s",TitleArtist(mode->Title,mode->Artist));
nathan@0
   523
	      flipint=6;
nathan@0
   524
	      break;
nathan@0
   525
	    case 1:
nathan@0
   526
              if(mode->Album[0]) {
nathan@0
   527
      	        snprintf(buf,sizeof(buf),mode->Year>0?"from: %s (%d)":"from: %s",mode->Album,mode->Year);
nathan@0
   528
	        flipint=4;
nathan@0
   529
	        }
nathan@0
   530
              else fliptime=0;
nathan@0
   531
              break;
nathan@0
   532
	    case 2:
nathan@0
   533
              if(mode->MaxBitrate>0)
nathan@0
   534
                snprintf(buf,sizeof(buf),"%.1f kHz, %d-%d kbps, %s",mode->SampleFreq/1000.0,mode->Bitrate/1000,mode->MaxBitrate/1000,mode->SMode);
nathan@0
   535
              else
nathan@0
   536
                snprintf(buf,sizeof(buf),"%.1f kHz, %d kbps, %s",mode->SampleFreq/1000.0,mode->Bitrate/1000,mode->SMode);
nathan@0
   537
	      flipint=3;
nathan@0
   538
	      break;
nathan@0
   539
	    }
nathan@0
   540
          if(buf[0]) {
nathan@0
   541
            if(MP3Setup.ReplayDisplay) {
nathan@0
   542
              char buf2[256];
nathan@0
   543
              snprintf(buf2,sizeof(buf2),"[%c%c] (%d/%d) %s",
nathan@0
   544
                       mode->Loop?'L':'.',mode->Shuffle?'S':'.',mode->Num,mode->MaxNum,buf);
nathan@0
   545
              disp->SetTitle(buf2);
nathan@0
   546
              flush=true;
nathan@0
   547
              }
nathan@0
   548
            else {
nathan@0
   549
              if(!statusActive) {
nathan@0
   550
                DisplayInfo(buf);
nathan@0
   551
                flush=true;
nathan@0
   552
                }
nathan@0
   553
              else { d(printf("mp3-ctrl: display info skip due to status active\n")) }
nathan@0
   554
              }
nathan@0
   555
            }
nathan@0
   556
          }
nathan@0
   557
        }
nathan@0
   558
nathan@0
   559
      if(bigwin) {
nathan@0
   560
        bool all=(top!=lastTop || changed);
nathan@0
   561
        if(all || time(0)>listtime+2) {
nathan@0
   562
          int num=(top>0 && mode->Num==lastMode->Num) ? top : mode->Num - rows/2;
nathan@0
   563
          if(num+rows>mode->MaxNum) num=mode->MaxNum-rows+1;
nathan@0
   564
          if(num<1) num=1;
nathan@0
   565
          top=num;
nathan@0
   566
          for(int i=0 ; i<rows && i<MAXROWS && num<=mode->MaxNum ; i++,num++) {
nathan@0
   567
            cMP3PlayInfo pi;
nathan@0
   568
            mgr->Info(num,&pi); if(!pi.Title[0]) break;
nathan@0
   569
            snprintf(buf,sizeof(buf),"%d.\t%s",num,TitleArtist(pi.Title,pi.Artist));
nathan@22
   570
            int fg=clrWhite, bg=clrGray50;
nathan@0
   571
            int hash=MakeHash(buf);
nathan@0
   572
            if(num==mode->Num) { fg=clrBlack; bg=clrCyan; hash=(hash^77) + 23; }
nathan@0
   573
            if(all || hash!=hashlist[i]) {
nathan@0
   574
              char *s=rindex(buf,'\t');
nathan@0
   575
              if(s) {
nathan@0
   576
                *s++=0;
nathan@0
   577
                Write(0,i,5,buf,fg,bg);
nathan@0
   578
                Write(5,i,bwc-5,s,fg,bg);
nathan@0
   579
                }
nathan@0
   580
              else
nathan@0
   581
                Write(0,i,bwc,buf,fg,bg);
nathan@0
   582
              flush=true;
nathan@0
   583
              hashlist[i]=hash;
nathan@0
   584
              }
nathan@0
   585
            }
nathan@0
   586
          listtime=time(0); lastTop=top;
nathan@0
   587
          }
nathan@0
   588
        }
nathan@0
   589
nathan@0
   590
      if(flush) Flush();
nathan@0
   591
      }
nathan@0
   592
nathan@0
   593
    lastIndex=index; lastTotal=total;
nathan@0
   594
    delete lastMode; lastMode=mode;
nathan@0
   595
    }
nathan@0
   596
}
nathan@0
   597
nathan@0
   598
void cMP3Control::DisplayInfo(const char *s)
nathan@0
   599
{
nathan@0
   600
  if(s) Write(CTAB,-2,bwc-CTAB,s);
nathan@22
   601
  else Fill(CTAB,-2,bwc-CTAB,1,clrGray50);
nathan@0
   602
}
nathan@0
   603
nathan@0
   604
void cMP3Control::JumpDisplay(void)
nathan@0
   605
{
nathan@0
   606
  char buf[64];
nathan@2
   607
  const char *j=trVDR("Jump: "), u=jumpsecs?'s':'m';
nathan@0
   608
  if(!jumpmm) sprintf(buf,"%s- %c",  j,u);
nathan@0
   609
  else        sprintf(buf,"%s%d- %c",j,jumpmm,u);
nathan@0
   610
  if(MP3Setup.ReplayDisplay) {
nathan@0
   611
    disp->SetJump(buf);
nathan@0
   612
    }
nathan@0
   613
  else {
nathan@0
   614
    DisplayInfo(buf);
nathan@22
   615
    }
nathan@0
   616
}
nathan@0
   617
nathan@0
   618
void cMP3Control::JumpProcess(eKeys Key)
nathan@0
   619
{
nathan@0
   620
 int n=Key-k0, d=jumpsecs?1:60;
nathan@0
   621
  switch (Key) {
nathan@0
   622
    case k0 ... k9:
nathan@0
   623
      if(jumpmm*10+n <= lastTotal/d) jumpmm=jumpmm*10+n;
nathan@0
   624
      JumpDisplay();
nathan@0
   625
      break;
nathan@0
   626
    case kBlue:
nathan@0
   627
      jumpsecs=!jumpsecs;
nathan@0
   628
      JumpDisplay();
nathan@0
   629
      break;
nathan@0
   630
    case kPlay:
nathan@0
   631
    case kUp:
nathan@0
   632
      jumpmm-=lastIndex/d;
nathan@0
   633
      // fall through
nathan@0
   634
    case kFastRew:
nathan@0
   635
    case kFastFwd:
nathan@0
   636
    case kLeft:
nathan@0
   637
    case kRight:
nathan@0
   638
      player->SkipSeconds(jumpmm*d * ((Key==kLeft || Key==kFastRew) ? -1:1));
nathan@0
   639
      // fall through
nathan@0
   640
    default:
nathan@0
   641
      jumpactive=false;
nathan@0
   642
      break;
nathan@0
   643
    }
nathan@0
   644
nathan@0
   645
  if(!jumpactive) {
nathan@0
   646
    if(jumphide) Hide();
nathan@0
   647
    else if(MP3Setup.ReplayDisplay) disp->SetJump(0);
nathan@0
   648
    }
nathan@0
   649
}
nathan@0
   650
nathan@0
   651
void cMP3Control::Jump(void)
nathan@0
   652
{
nathan@0
   653
  jumpmm=0; jumphide=jumpsecs=false;
nathan@0
   654
  if(!visible) {
nathan@0
   655
    ShowTimed(); if(!visible) return;
nathan@0
   656
    jumphide=true;
nathan@0
   657
    }
nathan@0
   658
  JumpDisplay();
nathan@0
   659
  jumpactive=true; fliptime=0; flip=-1;
nathan@0
   660
}
nathan@0
   661
nathan@0
   662
eOSState cMP3Control::ProcessKey(eKeys Key)
nathan@0
   663
{
nathan@0
   664
  if(!player->Active()) return osEnd;
nathan@0
   665
nathan@7
   666
  if(timeoutShow && time(0)>timeoutShow) Hide();
nathan@0
   667
  ShowProgress();
nathan@0
   668
  ShowStatus(Key==kNone && !Skins.IsOpen());
nathan@0
   669
nathan@0
   670
  if(jumpactive && Key!=kNone) { JumpProcess(Key); return osContinue; }
nathan@0
   671
nathan@0
   672
  switch(Key) {
nathan@0
   673
    case kUp:
nathan@0
   674
    case kUp|k_Repeat:
nathan@0
   675
    case kNext:
nathan@0
   676
    case kNext|k_Repeat:    
nathan@0
   677
      mgr->Next(); player->Play();
nathan@0
   678
      break;
nathan@0
   679
    case kDown:
nathan@0
   680
    case kDown|k_Repeat:
nathan@0
   681
    case kPrev:
nathan@0
   682
    case kPrev|k_Repeat:
nathan@0
   683
      if(!player->PrevCheck()) mgr->Prev();
nathan@0
   684
      player->Play();
nathan@0
   685
      break;
nathan@0
   686
    case kLeft:
nathan@0
   687
    case kLeft|k_Repeat:
nathan@0
   688
      if(bigwin) {
nathan@0
   689
        if(top>0) { top-=rows; if(top<1) top=1; }
nathan@0
   690
        break;
nathan@0
   691
        }
nathan@0
   692
      // fall through
nathan@0
   693
    case kFastRew:
nathan@0
   694
    case kFastRew|k_Repeat:
nathan@0
   695
      if(!player->IsStream()) player->SkipSeconds(-JUMPSIZE);
nathan@0
   696
      break;
nathan@0
   697
    case kRight:
nathan@0
   698
    case kRight|k_Repeat:
nathan@0
   699
      if(bigwin) {
nathan@0
   700
        if(top>0) top+=rows;
nathan@0
   701
        break;
nathan@0
   702
        }
nathan@0
   703
      // fall through
nathan@0
   704
    case kFastFwd:
nathan@0
   705
    case kFastFwd|k_Repeat:
nathan@0
   706
      if(!player->IsStream()) player->SkipSeconds(JUMPSIZE);
nathan@0
   707
      break;
nathan@0
   708
    case kRed:
nathan@0
   709
      if(!player->IsStream()) Jump();
nathan@0
   710
      break;
nathan@0
   711
    case kGreen:
nathan@0
   712
      if(lastMode) {
nathan@0
   713
        if(time(0)>greentime) {
nathan@0
   714
          if(lastMode->Loop || (!lastMode->Loop && !lastMode->Shuffle)) mgr->ToggleLoop();
nathan@0
   715
          if(lastMode->Shuffle) mgr->ToggleShuffle();
nathan@0
   716
          }
nathan@0
   717
        else {
nathan@0
   718
          if(!lastMode->Loop) mgr->ToggleLoop();
nathan@0
   719
          else if(!lastMode->Shuffle) mgr->ToggleShuffle();
nathan@0
   720
          else mgr->ToggleLoop();
nathan@0
   721
          }
nathan@0
   722
        greentime=time(0)+MULTI_TIMEOUT;
nathan@0
   723
        }
nathan@0
   724
      break;
nathan@0
   725
    case kPlay:
nathan@0
   726
      player->Play();
nathan@0
   727
      break;
nathan@0
   728
    case kPause:
nathan@0
   729
    case kYellow:
nathan@0
   730
      if(!player->IsStream()) player->Pause();
nathan@0
   731
      break;
nathan@0
   732
    case kStop:
nathan@0
   733
    case kBlue:
nathan@0
   734
      Hide();
nathan@0
   735
      Stop();
nathan@0
   736
      return osEnd;
nathan@0
   737
    case kBack:
nathan@0
   738
      Hide();
nathan@2
   739
      cRemote::CallPlugin(plugin_name);
nathan@0
   740
      return osBack;
nathan@0
   741
nathan@0
   742
    case k0 ... k9:
nathan@0
   743
      number=number*10+Key-k0;
nathan@0
   744
      if(lastMode && number>0 && number<=lastMode->MaxNum) {
nathan@7
   745
        if(!visible) ShowTimed(SELECTHIDE_TIMEOUT);
nathan@7
   746
        else if(timeoutShow>0) timeoutShow=time(0)+SELECTHIDE_TIMEOUT;
nathan@0
   747
        selecting=true; lastkeytime=time_ms();
nathan@0
   748
        char buf[32];
nathan@0
   749
        if(MP3Setup.ReplayDisplay) {
nathan@2
   750
          snprintf(buf,sizeof(buf),"%s%d-/%d",trVDR("Jump: "),number,lastMode->MaxNum);
nathan@0
   751
          disp->SetJump(buf);
nathan@0
   752
          }
nathan@0
   753
        else {
nathan@0
   754
          snprintf(buf,sizeof(buf),"(%d-/%d)",number,lastMode->MaxNum);
nathan@0
   755
          Write(0,-2,CTAB,buf);
nathan@0
   756
          }
nathan@0
   757
        Flush();
nathan@0
   758
        break;
nathan@0
   759
        }
nathan@0
   760
      number=0; lastkeytime=0;
nathan@0
   761
      // fall through
nathan@0
   762
    case kNone:
nathan@0
   763
      if(selecting && time_ms()-lastkeytime>SELECT_TIMEOUT) {
nathan@0
   764
        if(number>0) { mgr->Goto(number); player->Play();  }
nathan@0
   765
        if(lastMode) lastMode->Hash=-1;
nathan@7
   766
        number=0; selecting=false;
nathan@7
   767
        if(MP3Setup.ReplayDisplay && disp) disp->SetJump(0);
nathan@0
   768
        }
nathan@0
   769
      break;
nathan@0
   770
    case kOk:
nathan@0
   771
      if(time(0)>oktime || MP3Setup.ReplayDisplay) {
nathan@0
   772
        visible ? Hide() : ShowTimed();
nathan@0
   773
        }
nathan@0
   774
      else {
nathan@0
   775
        if(visible && !bigwin) { Hide(); ShowProgress(true,true); }
nathan@0
   776
        else { Hide(); ShowTimed(); }
nathan@0
   777
        }
nathan@0
   778
      oktime=time(0)+MULTI_TIMEOUT;
nathan@0
   779
      ShowStatus(true);
nathan@0
   780
      break;
nathan@0
   781
    default:
nathan@0
   782
      return osUnknown;
nathan@0
   783
    }
nathan@0
   784
  return osContinue;
nathan@0
   785
}
nathan@0
   786
nathan@0
   787
// --- cMenuID3Info ------------------------------------------------------------
nathan@0
   788
nathan@0
   789
class cMenuID3Info : public cOsdMenu {
nathan@0
   790
private:
nathan@0
   791
  cOsdItem *Item(const char *name, const char *text);
nathan@0
   792
  cOsdItem *Item(const char *name, const char *format, const float num);
nathan@0
   793
  void Build(cSongInfo *info, const char *name);
nathan@0
   794
public:
nathan@0
   795
  cMenuID3Info(cSong *song);
nathan@0
   796
  cMenuID3Info(cSongInfo *si, const char *name);
nathan@0
   797
  virtual eOSState ProcessKey(eKeys Key);
nathan@0
   798
  };
nathan@0
   799
nathan@0
   800
cMenuID3Info::cMenuID3Info(cSong *song)
nathan@0
   801
:cOsdMenu(tr("ID3 information"),12)
nathan@0
   802
{
nathan@0
   803
  Build(song->Info(),song->Name());
nathan@0
   804
}
nathan@0
   805
nathan@0
   806
cMenuID3Info::cMenuID3Info(cSongInfo *si, const char *name)
nathan@0
   807
:cOsdMenu(tr("ID3 information"),12)
nathan@0
   808
{
nathan@0
   809
  Build(si,name);
nathan@0
   810
}
nathan@0
   811
nathan@0
   812
void cMenuID3Info::Build(cSongInfo *si, const char *name)
nathan@0
   813
{
nathan@0
   814
  if(si) {
nathan@0
   815
    Item(tr("Filename"),name);
nathan@0
   816
    if(si->HasInfo() && si->Total>0) {
nathan@0
   817
      char *buf=0;
nathan@0
   818
      asprintf(&buf,"%02d:%02d",si->Total/60,si->Total%60);
nathan@0
   819
      Item(tr("Length"),buf);
nathan@0
   820
      free(buf);
nathan@0
   821
      Item(tr("Title"),si->Title);
nathan@0
   822
      Item(tr("Artist"),si->Artist);
nathan@0
   823
      Item(tr("Album"),si->Album);
nathan@0
   824
      Item(tr("Year"),0,(float)si->Year);
nathan@0
   825
      Item(tr("Samplerate"),"%.1f kHz",si->SampleFreq/1000.0);
nathan@0
   826
      Item(tr("Bitrate"),"%.f kbit/s",si->Bitrate/1000.0);
nathan@2
   827
      Item(trVDR("Channels"),0,(float)si->Channels);
nathan@0
   828
      }
nathan@0
   829
    Display();
nathan@0
   830
    }
nathan@0
   831
}
nathan@0
   832
nathan@0
   833
cOsdItem *cMenuID3Info::Item(const char *name, const char *format, const float num)
nathan@0
   834
{
nathan@0
   835
  cOsdItem *item;
nathan@0
   836
  if(num>=0.0) {
nathan@0
   837
    char *buf=0;
nathan@0
   838
    asprintf(&buf,format?format:"%.f",num);
nathan@0
   839
    item=Item(name,buf);
nathan@0
   840
    free(buf);
nathan@0
   841
    }
nathan@0
   842
  else item=Item(name,"");
nathan@0
   843
  return item;
nathan@0
   844
}
nathan@0
   845
nathan@0
   846
cOsdItem *cMenuID3Info::Item(const char *name, const char *text)
nathan@0
   847
{
nathan@0
   848
  char *buf=0;
nathan@0
   849
  asprintf(&buf,"%s:\t%s",name,text?text:"");
nathan@0
   850
  cOsdItem *item = new cOsdItem(buf,osBack);
nathan@0
   851
  item->SetSelectable(false);
nathan@0
   852
  free(buf);
nathan@0
   853
  Add(item); return item;
nathan@0
   854
}
nathan@0
   855
nathan@0
   856
eOSState cMenuID3Info::ProcessKey(eKeys Key)
nathan@0
   857
{
nathan@0
   858
  eOSState state = cOsdMenu::ProcessKey(Key);
nathan@0
   859
nathan@0
   860
  if(state==osUnknown) {
nathan@0
   861
     switch(Key) {
nathan@0
   862
       case kRed:
nathan@0
   863
       case kGreen:
nathan@0
   864
       case kYellow:
nathan@0
   865
       case kBlue:   return osContinue;
nathan@0
   866
       case kMenu:   return osEnd;
nathan@0
   867
       default: break;
nathan@0
   868
       }
nathan@0
   869
     }
nathan@0
   870
  return state;
nathan@0
   871
}
nathan@0
   872
nathan@0
   873
// --- cMenuInstantBrowse -------------------------------------------------------
nathan@0
   874
nathan@0
   875
class cMenuInstantBrowse : public cMenuBrowse {
nathan@0
   876
private:
nathan@0
   877
  const char *selecttext, *alltext;
nathan@0
   878
  virtual void SetButtons(void);
nathan@0
   879
  virtual eOSState ID3Info(void);
nathan@0
   880
public:
nathan@0
   881
  cMenuInstantBrowse(cFileSource *Source, const char *Selecttext, const char *Alltext);
nathan@0
   882
  virtual eOSState ProcessKey(eKeys Key);
nathan@0
   883
  };
nathan@0
   884
nathan@0
   885
cMenuInstantBrowse::cMenuInstantBrowse(cFileSource *Source, const char *Selecttext, const char *Alltext)
nathan@0
   886
:cMenuBrowse(Source,true,true,tr("Directory browser"),excl_br)
nathan@0
   887
{
nathan@0
   888
  selecttext=Selecttext; alltext=Alltext;
nathan@0
   889
  SetButtons();
nathan@0
   890
}
nathan@0
   891
nathan@0
   892
void cMenuInstantBrowse::SetButtons(void)
nathan@0
   893
{
nathan@0
   894
  SetHelp(selecttext, currentdir?tr("Parent"):0, currentdir?0:alltext, tr("ID3 info"));
nathan@0
   895
  Display();
nathan@0
   896
}
nathan@0
   897
nathan@0
   898
eOSState cMenuInstantBrowse::ID3Info(void)
nathan@0
   899
{
nathan@0
   900
  cFileObj *item=CurrentItem();
nathan@0
   901
  if(item && item->Type()==otFile) {
nathan@0
   902
    cSong *song=new cSong(item);
nathan@0
   903
    cSongInfo *si;
nathan@0
   904
    if(song && (si=song->Info())) {
nathan@0
   905
      AddSubMenu(new cMenuID3Info(si,item->Path()));
nathan@0
   906
      }
nathan@0
   907
    delete song;
nathan@0
   908
    }
nathan@0
   909
  return osContinue;
nathan@0
   910
}
nathan@0
   911
nathan@0
   912
eOSState cMenuInstantBrowse::ProcessKey(eKeys Key)
nathan@0
   913
{
nathan@0
   914
  eOSState state=cOsdMenu::ProcessKey(Key);
nathan@0
   915
  if(state==osUnknown) {
nathan@0
   916
     switch (Key) {
nathan@0
   917
       case kYellow: lastselect=new cFileObj(source,0,0,otBase);
nathan@0
   918
                     return osBack;
nathan@0
   919
       default: break;
nathan@0
   920
       }
nathan@0
   921
     }
nathan@0
   922
  if(state==osUnknown) state=cMenuBrowse::ProcessStdKey(Key,state);
nathan@0
   923
  return state;
nathan@0
   924
}
nathan@0
   925
nathan@0
   926
// --- cMenuPlayListItem -------------------------------------------------------
nathan@0
   927
nathan@0
   928
class cMenuPlayListItem : public cOsdItem {
nathan@0
   929
  private:
nathan@0
   930
  bool showID3;
nathan@0
   931
  cSong *song;
nathan@0
   932
public:
nathan@0
   933
  cMenuPlayListItem(cSong *Song, bool showid3);
nathan@0
   934
  cSong *Song(void) { return song; }
nathan@0
   935
  virtual void Set(void);
nathan@0
   936
  void Set(bool showid3);
nathan@0
   937
  };
nathan@0
   938
nathan@0
   939
cMenuPlayListItem::cMenuPlayListItem(cSong *Song, bool showid3)
nathan@0
   940
{
nathan@0
   941
  song=Song;
nathan@0
   942
  Set(showid3);
nathan@0
   943
}
nathan@0
   944
nathan@0
   945
void cMenuPlayListItem::Set(bool showid3)
nathan@0
   946
{
nathan@0
   947
  showID3=showid3;
nathan@0
   948
  Set();
nathan@0
   949
}
nathan@0
   950
nathan@0
   951
void cMenuPlayListItem::Set(void)
nathan@0
   952
{
nathan@0
   953
  char *buffer=0;
nathan@0
   954
  cSongInfo *si=song->Info(false);
nathan@0
   955
  if(showID3 && !si) si=song->Info();
nathan@0
   956
  if(showID3 && si && si->Title)
nathan@0
   957
    asprintf(&buffer, "%d.\t%s",song->Index()+1,TitleArtist(si->Title,si->Artist));
nathan@0
   958
  else
nathan@0
   959
    asprintf(&buffer, "%d.\t<%s>",song->Index()+1,song->Name());
nathan@0
   960
  SetText(buffer,false);
nathan@0
   961
}
nathan@0
   962
nathan@0
   963
// --- cMenuPlayList ------------------------------------------------------
nathan@0
   964
nathan@0
   965
class cMenuPlayList : public cOsdMenu {
nathan@0
   966
private:
nathan@0
   967
  cPlayList *playlist;
nathan@0
   968
  bool browsing, showid3;
nathan@0
   969
  void Buttons(void);
nathan@0
   970
  void Refresh(bool all = false);
nathan@0
   971
  void Add(void);
nathan@0
   972
  virtual void Move(int From, int To);
nathan@0
   973
  eOSState Remove(void);
nathan@0
   974
  eOSState ShowID3(void);
nathan@0
   975
  eOSState ID3Info(void);
nathan@0
   976
public:
nathan@0
   977
  cMenuPlayList(cPlayList *Playlist);
nathan@0
   978
  virtual eOSState ProcessKey(eKeys Key);
nathan@0
   979
  };
nathan@0
   980
nathan@0
   981
cMenuPlayList::cMenuPlayList(cPlayList *Playlist)
nathan@0
   982
:cOsdMenu(tr("Playlist editor"),4)
nathan@0
   983
{
nathan@0
   984
  browsing=showid3=false;
nathan@0
   985
  playlist=Playlist;
nathan@0
   986
  if(MP3Setup.EditorMode) showid3=true;
nathan@0
   987
nathan@0
   988
  cSong *mp3 = playlist->First();
nathan@0
   989
  while(mp3) {
nathan@0
   990
    cOsdMenu::Add(new cMenuPlayListItem(mp3,showid3));
nathan@0
   991
    mp3 = playlist->cList<cSong>::Next(mp3);
nathan@0
   992
    }
nathan@0
   993
  Buttons(); Display();
nathan@0
   994
}
nathan@0
   995
nathan@0
   996
void cMenuPlayList::Buttons(void)
nathan@0
   997
{
nathan@2
   998
  SetHelp(tr("Add"), showid3?tr("Filenames"):tr("ID3 names"), tr("Remove"), trVDR(BUTTON"Mark"));
nathan@0
   999
}
nathan@0
  1000
nathan@0
  1001
void cMenuPlayList::Refresh(bool all)
nathan@0
  1002
{
nathan@0
  1003
  cMenuPlayListItem *cur=(cMenuPlayListItem *)((all || Count()<2) ? First() : Get(Current()));
nathan@0
  1004
  while(cur) {
nathan@0
  1005
    cur->Set(showid3);
nathan@0
  1006
    cur=(cMenuPlayListItem *)Next(cur);
nathan@0
  1007
    }
nathan@0
  1008
}
nathan@0
  1009
nathan@0
  1010
void cMenuPlayList::Add(void)
nathan@0
  1011
{
nathan@0
  1012
  cFileObj *item=cMenuInstantBrowse::GetSelected();
nathan@0
  1013
  if(item) {
nathan@0
  1014
    Status(tr("Scanning directory..."));
nathan@0
  1015
    cInstantPlayList *newpl=new cInstantPlayList(item);
nathan@0
  1016
    if(newpl->Load()) {
nathan@0
  1017
      if(newpl->Count()) {
nathan@0
  1018
        if(newpl->Count()==1 || Interface->Confirm(tr("Add recursivly?"))) {
nathan@0
  1019
          cSong *mp3=newpl->First();
nathan@0
  1020
          while(mp3) {
nathan@0
  1021
            cSong *n=new cSong(mp3);
nathan@0
  1022
            if(Count()>0) {
nathan@0
  1023
              cMenuPlayListItem *current=(cMenuPlayListItem *)Get(Current());
nathan@0
  1024
              playlist->Add(n,current->Song());
nathan@0
  1025
              cOsdMenu::Add(new cMenuPlayListItem(n,showid3),true,current);
nathan@0
  1026
              }
nathan@0
  1027
            else {
nathan@0
  1028
              playlist->Add(n);
nathan@0
  1029
              cOsdMenu::Add(new cMenuPlayListItem(n,showid3),true);
nathan@0
  1030
              }
nathan@0
  1031
            mp3=newpl->cList<cSong>::Next(mp3);
nathan@0
  1032
            }
nathan@0
  1033
          playlist->Save();
nathan@0
  1034
          Refresh(); Display();
nathan@0
  1035
          }
nathan@0
  1036
        }
nathan@0
  1037
      else Error(tr("Empty directory!"));
nathan@0
  1038
      }
nathan@0
  1039
    else Error(tr("Error scanning directory!"));
nathan@0
  1040
    delete newpl;
nathan@0
  1041
    Status(0);
nathan@0
  1042
    }
nathan@0
  1043
}
nathan@0
  1044
nathan@0
  1045
void cMenuPlayList::Move(int From, int To)
nathan@0
  1046
{
nathan@0
  1047
  playlist->Move(From,To); playlist->Save();
nathan@0
  1048
  cOsdMenu::Move(From,To);
nathan@0
  1049
  Refresh(true); Display();
nathan@0
  1050
}
nathan@0
  1051
nathan@0
  1052
eOSState cMenuPlayList::ShowID3(void)
nathan@0
  1053
{
nathan@0
  1054
  showid3=!showid3;
nathan@0
  1055
  Buttons(); Refresh(true); Display();
nathan@0
  1056
  return osContinue;
nathan@0
  1057
}
nathan@0
  1058
nathan@0
  1059
eOSState cMenuPlayList::ID3Info(void)
nathan@0
  1060
{
nathan@0
  1061
  if(Count()>0) {
nathan@0
  1062
    cMenuPlayListItem *current = (cMenuPlayListItem *)Get(Current());
nathan@0
  1063
    AddSubMenu(new cMenuID3Info(current->Song()));
nathan@0
  1064
    }
nathan@0
  1065
  return osContinue;
nathan@0
  1066
}
nathan@0
  1067
nathan@0
  1068
eOSState cMenuPlayList::Remove(void)
nathan@0
  1069
{
nathan@0
  1070
  if(Count()>0) {
nathan@0
  1071
    cMenuPlayListItem *current = (cMenuPlayListItem *)Get(Current());
nathan@0
  1072
    if(Interface->Confirm(tr("Remove entry?"))) {
nathan@0
  1073
      playlist->Del(current->Song()); playlist->Save();
nathan@0
  1074
      cOsdMenu::Del(Current());
nathan@0
  1075
      Refresh(); Display();
nathan@0
  1076
      }
nathan@0
  1077
    }
nathan@0
  1078
  return osContinue;
nathan@0
  1079
}
nathan@0
  1080
nathan@0
  1081
eOSState cMenuPlayList::ProcessKey(eKeys Key)
nathan@0
  1082
{
nathan@0
  1083
  eOSState state = cOsdMenu::ProcessKey(Key);
nathan@0
  1084
nathan@0
  1085
  if(browsing && !HasSubMenu() && state==osContinue) { Add(); browsing=false; }
nathan@0
  1086
nathan@0
  1087
  if(state==osUnknown) {
nathan@0
  1088
     switch(Key) {
nathan@0
  1089
       case kOk:     return ID3Info();
nathan@0
  1090
       case kRed:    browsing=true;
nathan@0
  1091
                     return AddSubMenu(new cMenuInstantBrowse(MP3Sources.GetSource(),tr("Add"),tr("Add all")));
nathan@0
  1092
       case kGreen:  return ShowID3();
nathan@0
  1093
       case kYellow: return Remove();
nathan@0
  1094
       case kBlue:   Mark(); return osContinue;
nathan@0
  1095
       case kMenu:   return osEnd;
nathan@0
  1096
       default: break;
nathan@0
  1097
       }
nathan@0
  1098
     }
nathan@0
  1099
  return state;
nathan@0
  1100
}
nathan@0
  1101
nathan@0
  1102
// --- cPlaylistRename --------------------------------------------------------
nathan@0
  1103
nathan@0
  1104
class cPlaylistRename : public cOsdMenu {
nathan@0
  1105
private:
nathan@0
  1106
  static char *newname;
nathan@0
  1107
  const char *oldname;
nathan@0
  1108
  char data[64];
nathan@0
  1109
public:
nathan@0
  1110
  cPlaylistRename(const char *Oldname);
nathan@0
  1111
  virtual eOSState ProcessKey(eKeys Key);
nathan@0
  1112
  static const char *GetNewname(void) { return newname; }
nathan@0
  1113
  };
nathan@0
  1114
nathan@0
  1115
char *cPlaylistRename::newname = NULL;
nathan@0
  1116
nathan@0
  1117
cPlaylistRename::cPlaylistRename(const char *Oldname)
nathan@0
  1118
:cOsdMenu(tr("Rename playlist"), 15)
nathan@0
  1119
{
nathan@0
  1120
  free(newname); newname=0;
nathan@0
  1121
nathan@0
  1122
  oldname=Oldname;
nathan@0
  1123
  char *buf=NULL;
nathan@0
  1124
  asprintf(&buf,"%s\t%s",tr("Old name:"),oldname);
nathan@0
  1125
  cOsdItem *old = new cOsdItem(buf,osContinue);
nathan@0
  1126
  old->SetSelectable(false);
nathan@0
  1127
  Add(old);
nathan@0
  1128
  free(buf);
nathan@0
  1129
nathan@0
  1130
  data[0]=0;
nathan@0
  1131
  Add(new cMenuEditStrItem( tr("New name"), data, sizeof(data)-1, tr(FileNameChars)),true);
nathan@0
  1132
}
nathan@0
  1133
nathan@0
  1134
eOSState cPlaylistRename::ProcessKey(eKeys Key)
nathan@0
  1135
{
nathan@0
  1136
  eOSState state = cOsdMenu::ProcessKey(Key);
nathan@0
  1137
nathan@0
  1138
  if (state == osUnknown) {
nathan@0
  1139
     switch (Key) {
nathan@0
  1140
       case kOk:     if(data[0] && strcmp(data,oldname)) newname=strdup(data);
nathan@0
  1141
                     return osBack;
nathan@0
  1142
       case kRed:
nathan@0
  1143
       case kGreen:
nathan@0
  1144
       case kYellow:
nathan@0
  1145
       case kBlue:   return osContinue;
nathan@0
  1146
       default: break;
nathan@0
  1147
       }
nathan@0
  1148
     }
nathan@0
  1149
  return state;
nathan@0
  1150
}
nathan@0
  1151
nathan@0
  1152
// --- cMenuMP3Item -----------------------------------------------------
nathan@0
  1153
nathan@0
  1154
class cMenuMP3Item : public cOsdItem {
nathan@0
  1155
  private:
nathan@0
  1156
  cPlayList *playlist;
nathan@0
  1157
  virtual void Set(void);
nathan@0
  1158
public:
nathan@0
  1159
  cMenuMP3Item(cPlayList *PlayList);
nathan@0
  1160
  cPlayList *List(void) { return playlist; }
nathan@0
  1161
  };
nathan@0
  1162
nathan@0
  1163
cMenuMP3Item::cMenuMP3Item(cPlayList *PlayList)
nathan@0
  1164
{
nathan@0
  1165
  playlist=PlayList;
nathan@0
  1166
  Set();
nathan@0
  1167
}
nathan@0
  1168
nathan@0
  1169
void cMenuMP3Item::Set(void)
nathan@0
  1170
{
nathan@0
  1171
  char *buffer=0;
nathan@0
  1172
  asprintf(&buffer," %s",playlist->BaseName());
nathan@0
  1173
  SetText(buffer,false);
nathan@0
  1174
}
nathan@0
  1175
nathan@0
  1176
// --- cMenuMP3 --------------------------------------------------------
nathan@0
  1177
nathan@0
  1178
class cMenuMP3 : public cOsdMenu {
nathan@0
  1179
private:
nathan@0
  1180
  cPlayLists *lists;
nathan@0
  1181
  bool renaming, sourcing, instanting;
nathan@0
  1182
  int buttonnum;
nathan@0
  1183
  eOSState Play(void);
nathan@0
  1184
  eOSState Edit(void);
nathan@0
  1185
  eOSState New(void);
nathan@0
  1186
  eOSState Delete(void);
nathan@0
  1187
  eOSState Rename(bool second);
nathan@0
  1188
  eOSState Source(bool second);
nathan@0
  1189
  eOSState Instant(bool second);
nathan@0
  1190
  void ScanLists(void);
nathan@0
  1191
  eOSState SetButtons(int num);
nathan@0
  1192
public:
nathan@0
  1193
  cMenuMP3(void);
nathan@0
  1194
  ~cMenuMP3(void);
nathan@0
  1195
  virtual eOSState ProcessKey(eKeys Key);
nathan@0
  1196
  };
nathan@0
  1197
nathan@0
  1198
cMenuMP3::cMenuMP3(void)
nathan@0
  1199
:cOsdMenu(tr("MP3"))
nathan@0
  1200
{
nathan@0
  1201
  renaming=sourcing=instanting=false;
nathan@0
  1202
  lists=new cPlayLists;
nathan@0
  1203
  ScanLists(); SetButtons(1);
nathan@0
  1204
  if(MP3Setup.MenuMode) Instant(false);
nathan@0
  1205
}
nathan@0
  1206
nathan@0
  1207
cMenuMP3::~cMenuMP3(void)
nathan@0
  1208
{
nathan@0
  1209
  delete lists;
nathan@0
  1210
}
nathan@0
  1211
nathan@0
  1212
eOSState cMenuMP3::SetButtons(int num)
nathan@0
  1213
{
nathan@0
  1214
  switch(num) {
nathan@0
  1215
    case 1:
nathan@2
  1216
      SetHelp(trVDR(BUTTON"Edit"), tr("Source"), tr("Browse"), ">>");
nathan@0
  1217
      break;
nathan@0
  1218
    case 2:
nathan@2
  1219
      SetHelp("<<", trVDR(BUTTON"New"), trVDR(BUTTON"Delete"), tr("Rename"));
nathan@0
  1220
      break;
nathan@0
  1221
    }
nathan@0
  1222
  buttonnum=num; Display();
nathan@0
  1223
  return osContinue;
nathan@0
  1224
}
nathan@0
  1225
nathan@0
  1226
void cMenuMP3::ScanLists(void)
nathan@0
  1227
{
nathan@0
  1228
  Clear();
nathan@0
  1229
  Status(tr("Scanning playlists..."));
nathan@0
  1230
  bool res=lists->Load(MP3Sources.GetSource());
nathan@0
  1231
  Status(0);
nathan@0
  1232
  if(res) {
nathan@0
  1233
    cPlayList *plist=lists->First();
nathan@0
  1234
    while(plist) {
nathan@0
  1235
      Add(new cMenuMP3Item(plist));
nathan@0
  1236
      plist=lists->Next(plist);
nathan@0
  1237
      }
nathan@0
  1238
    }
nathan@0
  1239
  else Error(tr("Error scanning playlists!"));
nathan@0
  1240
}
nathan@0
  1241
nathan@0
  1242
eOSState cMenuMP3::Delete(void)
nathan@0
  1243
{
nathan@0
  1244
  if(Count()>0) {
nathan@0
  1245
    if(Interface->Confirm(tr("Delete playlist?")) &&
nathan@0
  1246
       Interface->Confirm(tr("Are you sure?")) ) {
nathan@0
  1247
      cPlayList *plist = ((cMenuMP3Item *)Get(Current()))->List();
nathan@0
  1248
      if(plist->Delete()) {
nathan@0
  1249
        lists->Del(plist);
nathan@0
  1250
        cOsdMenu::Del(Current());
nathan@0
  1251
        Display();
nathan@0
  1252
        }
nathan@0
  1253
      else Error(tr("Error deleting playlist!"));
nathan@0
  1254
      }
nathan@0
  1255
    }
nathan@0
  1256
  return osContinue;
nathan@0
  1257
}
nathan@0
  1258
nathan@0
  1259
eOSState cMenuMP3::New(void)
nathan@0
  1260
{
nathan@0
  1261
  cPlayList *plist=new cPlayList(MP3Sources.GetSource(),0,0);
nathan@0
  1262
  char name[32];
nathan@0
  1263
  int i=0;
nathan@0
  1264
  do {
nathan@0
  1265
    if(i) sprintf(name,"%s%d",tr("unnamed"),i++);
nathan@0
  1266
    else { strcpy(name,tr("unnamed")); i++; }
nathan@0
  1267
    } while(plist->TestName(name));
nathan@0
  1268
nathan@0
  1269
  if(plist->Create(name)) {
nathan@0
  1270
    lists->Add(plist);
nathan@0
  1271
    Add(new cMenuMP3Item(plist), true);
nathan@0
  1272
nathan@0
  1273
    isyslog("MP3: playlist %s added", plist->Name());
nathan@0
  1274
    return AddSubMenu(new cMenuPlayList(plist));
nathan@0
  1275
    }
nathan@0
  1276
  Error(tr("Error creating playlist!"));
nathan@0
  1277
  delete plist;
nathan@0
  1278
  return osContinue;
nathan@0
  1279
}
nathan@0
  1280
nathan@0
  1281
eOSState cMenuMP3::Rename(bool second)
nathan@0
  1282
{
nathan@0
  1283
  if(HasSubMenu() || Count() == 0) return osContinue;
nathan@0
  1284
nathan@0
  1285
  cPlayList *plist = ((cMenuMP3Item *)Get(Current()))->List();
nathan@0
  1286
  if(!second) {
nathan@0
  1287
    renaming=true;
nathan@0
  1288
    return AddSubMenu(new cPlaylistRename(plist->BaseName()));
nathan@0
  1289
    }
nathan@0
  1290
  renaming=false;
nathan@0
  1291
  const char *newname=cPlaylistRename::GetNewname();
nathan@0
  1292
  if(newname) {
nathan@0
  1293
    if(plist->Rename(newname)) {
nathan@0
  1294
      RefreshCurrent();
nathan@0
  1295
      DisplayCurrent(true);
nathan@0
  1296
      }
nathan@0
  1297
    else Error(tr("Error renaming playlist!"));
nathan@0
  1298
    }
nathan@0
  1299
  return osContinue;
nathan@0
  1300
}
nathan@0
  1301
nathan@0
  1302
eOSState cMenuMP3::Edit(void)
nathan@0
  1303
{
nathan@0
  1304
  if(HasSubMenu() || Count() == 0) return osContinue;
nathan@0
  1305
nathan@0
  1306
  cPlayList *plist = ((cMenuMP3Item *)Get(Current()))->List();
nathan@0
  1307
  if(!plist->Load()) Error(tr("Error loading playlist!"));
nathan@0
  1308
  else if(!plist->IsWinAmp()) {
nathan@0
  1309
    isyslog("MP3: editing playlist %s", plist->Name());
nathan@0
  1310
    return AddSubMenu(new cMenuPlayList(plist));
nathan@0
  1311
    }
nathan@0
  1312
  else Error(tr("Can't edit a WinAmp playlist!"));
nathan@0
  1313
  return osContinue;
nathan@0
  1314
}
nathan@0
  1315
nathan@0
  1316
eOSState cMenuMP3::Play(void)
nathan@0
  1317
{
nathan@0
  1318
  if(HasSubMenu() || Count() == 0) return osContinue;
nathan@0
  1319
nathan@0
  1320
  Status(tr("Loading playlist..."));
nathan@0
  1321
  cPlayList *newpl=new cPlayList(((cMenuMP3Item *)Get(Current()))->List());
nathan@0
  1322
  if(newpl->Load() && newpl->Count()) {
nathan@0
  1323
    isyslog("mp3: playback started with playlist %s", newpl->Name());
nathan@0
  1324
    cMP3Control::SetPlayList(newpl);
nathan@0
  1325
    if(MP3Setup.KeepSelect) { Status(0); return osContinue; }
nathan@0
  1326
    return osEnd;
nathan@0
  1327
    }
nathan@0
  1328
  Status(0);
nathan@0
  1329
  delete newpl;
nathan@0
  1330
  Error(tr("Error loading playlist!"));
nathan@0
  1331
  return osContinue;
nathan@0
  1332
}
nathan@0
  1333
nathan@0
  1334
eOSState cMenuMP3::Source(bool second)
nathan@0
  1335
{
nathan@0
  1336
  if(HasSubMenu()) return osContinue;
nathan@0
  1337
nathan@0
  1338
  if(!second) {
nathan@0
  1339
    sourcing=true;
nathan@0
  1340
    return AddSubMenu(new cMenuSource(&MP3Sources,tr("MP3 source")));
nathan@0
  1341
    }
nathan@0
  1342
  sourcing=false;
nathan@0
  1343
  cFileSource *src=cMenuSource::GetSelected();
nathan@0
  1344
  if(src) {
nathan@0
  1345
    MP3Sources.SetSource(src);
nathan@0
  1346
    ScanLists();
nathan@0
  1347
    Display();
nathan@0
  1348
    }
nathan@0
  1349
  return osContinue;
nathan@0
  1350
}
nathan@0
  1351
nathan@0
  1352
eOSState cMenuMP3::Instant(bool second)
nathan@0
  1353
{
nathan@0
  1354
  if(HasSubMenu()) return osContinue;
nathan@0
  1355
nathan@0
  1356
  if(!second) {
nathan@0
  1357
    instanting=true;
nathan@2
  1358
    return AddSubMenu(new cMenuInstantBrowse(MP3Sources.GetSource(),trVDR(BUTTON"Play"),tr("Play all")));
nathan@0
  1359
    }
nathan@0
  1360
  instanting=false;
nathan@0
  1361
  cFileObj *item=cMenuInstantBrowse::GetSelected();
nathan@0
  1362
  if(item) {
nathan@0
  1363
    Status(tr("Building playlist..."));
nathan@0
  1364
    cInstantPlayList *newpl = new cInstantPlayList(item);
nathan@0
  1365
    if(newpl->Load() && newpl->Count()) {
nathan@0
  1366
      isyslog("mp3: playback started with instant playlist %s", newpl->Name());
nathan@0
  1367
      cMP3Control::SetPlayList(newpl);
nathan@0
  1368
      if(MP3Setup.KeepSelect) { Status(0); return Instant(false); }
nathan@0
  1369
      return osEnd;
nathan@0
  1370
      }
nathan@0
  1371
    Status(0);
nathan@0
  1372
    delete newpl;
nathan@0
  1373
    Error(tr("Error building playlist!"));
nathan@0
  1374
    }
nathan@0
  1375
  return osContinue;
nathan@0
  1376
}
nathan@0
  1377
nathan@0
  1378
eOSState cMenuMP3::ProcessKey(eKeys Key)
nathan@0
  1379
{
nathan@0
  1380
  eOSState state = cOsdMenu::ProcessKey(Key);
nathan@0
  1381
nathan@0
  1382
  if(!HasSubMenu() && state==osContinue) { // eval the return value from submenus
nathan@0
  1383
    if(renaming) return Rename(true);
nathan@0
  1384
    if(sourcing) return Source(true);
nathan@0
  1385
    if(instanting) return Instant(true);
nathan@0
  1386
    }
nathan@0
  1387
nathan@0
  1388
  if(state == osUnknown) {
nathan@0
  1389
    switch(Key) {
nathan@0
  1390
      case kOk:     return Play();
nathan@0
  1391
      case kRed:    return (buttonnum==1 ? Edit() : SetButtons(1)); 
nathan@0
  1392
      case kGreen:  return (buttonnum==1 ? Source(false) : New());
nathan@0
  1393
      case kYellow: return (buttonnum==1 ? Instant(false) : Delete());
nathan@0
  1394
      case kBlue:   return (buttonnum==1 ? SetButtons(2) : Rename(false));
nathan@0
  1395
      case kMenu:   return osEnd;
nathan@0
  1396
      default:      break;
nathan@0
  1397
      }
nathan@0
  1398
    }
nathan@0
  1399
  return state;
nathan@0
  1400
}
nathan@0
  1401
nathan@0
  1402
// --- PropagateImage ----------------------------------------------------------
nathan@0
  1403
nathan@0
  1404
void PropagateImage(const char *image)
nathan@0
  1405
{
nathan@0
  1406
  cPlugin *graphtft=cPluginManager::GetPlugin("graphtft");
nathan@0
  1407
  if(graphtft) graphtft->SetupParse("CoverImage",image ? image:"");
nathan@0
  1408
}
nathan@0
  1409
nathan@0
  1410
// --- cPluginMP3 --------------------------------------------------------------
nathan@0
  1411
nathan@2
  1412
static const char *DESCRIPTION    = trNOOP("A versatile audio player");
nathan@0
  1413
static const char *MAINMENUENTRY  = "MP3";
nathan@0
  1414
nathan@0
  1415
class cPluginMp3 : public cPlugin {
nathan@0
  1416
private:
nathan@0
  1417
  bool ExternalPlay(const char *path, bool test);
nathan@0
  1418
public:
nathan@0
  1419
  cPluginMp3(void);
nathan@0
  1420
  virtual ~cPluginMp3();
nathan@9
  1421
  virtual const char *Version(void) { return PluginVersion; }
nathan@0
  1422
  virtual const char *Description(void) { return tr(DESCRIPTION); }
nathan@0
  1423
  virtual const char *CommandLineHelp(void);
nathan@0
  1424
  virtual bool ProcessArgs(int argc, char *argv[]);
nathan@0
  1425
  virtual bool Initialize(void);
nathan@0
  1426
  virtual void Housekeeping(void);
nathan@0
  1427
  virtual const char *MainMenuEntry(void);
nathan@0
  1428
  virtual cOsdObject *MainMenuAction(void);
nathan@0
  1429
  virtual cMenuSetupPage *SetupMenu(void);
nathan@0
  1430
  virtual bool SetupParse(const char *Name, const char *Value);
nathan@0
  1431
  virtual bool Service(const char *Id, void *Data);
nathan@0
  1432
  virtual const char **SVDRPHelpPages(void);
nathan@0
  1433
  virtual cString SVDRPCommand(const char *Command, const char *Option, int &ReplyCode);
nathan@0
  1434
  };
nathan@0
  1435
nathan@0
  1436
cPluginMp3::cPluginMp3(void)
nathan@0
  1437
{
nathan@0
  1438
  // Initialize any member varaiables here.
nathan@0
  1439
  // DON'T DO ANYTHING ELSE THAT MAY HAVE SIDE EFFECTS, REQUIRE GLOBAL
nathan@0
  1440
  // VDR OBJECTS TO EXIST OR PRODUCE ANY OUTPUT!
nathan@0
  1441
}
nathan@0
  1442
nathan@0
  1443
cPluginMp3::~cPluginMp3()
nathan@0
  1444
{
nathan@2
  1445
  InfoCache.Shutdown();
nathan@0
  1446
  delete mgr;
nathan@0
  1447
}
nathan@0
  1448
nathan@0
  1449
const char *cPluginMp3::CommandLineHelp(void)
nathan@0
  1450
{
nathan@0
  1451
  static char *help_str=0;
nathan@0
  1452
  
nathan@0
  1453
  free(help_str);    //                                     for easier orientation, this is column 80|
nathan@0
  1454
  asprintf(&help_str,"  -m CMD,   --mount=CMD    use CMD to mount/unmount/eject mp3 sources\n"
nathan@0
  1455
                     "                           (default: %s)\n"
nathan@0
  1456
                     "  -n CMD,   --network=CMD  execute CMD before & after network access\n"
nathan@0
  1457
                     "                           (default: %s)\n"
nathan@0
  1458
                     "  -C DIR,   --cache=DIR    store ID3 cache file in DIR\n"
nathan@0
  1459
                     "                           (default: %s)\n"
nathan@0
  1460
                     "  -B DIR,   --cddb=DIR     search CDDB files in DIR\n"
nathan@0
  1461
                     "                           (default: %s)\n"
nathan@0
  1462
                     "  -D DEV,   --dsp=DEV      device for OSS output\n"
nathan@0
  1463
                     "                           (default: %s)\n"
nathan@0
  1464
                     "  -i CMD,   --iconv=CMD    use CMD to convert background images\n"
nathan@0
  1465
                     "                           (default: %s)\n"
nathan@19
  1466
                     "  -I IMG,   --defimage=IMG use IMG as default background image\n"
nathan@19
  1467
                     "                           (default: none)\n"
nathan@0
  1468
                     "  -c DIR,   --icache=DIR   cache converted images in DIR\n"
nathan@0
  1469
                     "                           (default: %s)\n"
nathan@0
  1470
                     "  -S SUB,   --sources=SUB  search sources config in SUB subdirectory\n"
nathan@0
  1471
                     "                           (default: %s)\n",
nathan@0
  1472
                     
nathan@0
  1473
                     mountscript,
nathan@0
  1474
                     netscript ? netscript:"none",
nathan@0
  1475
                     cachedir ? cachedir:"video dir",
nathan@0
  1476
#ifdef HAVE_SNDFILE
nathan@0
  1477
                     cddbpath,
nathan@0
  1478
#else
nathan@0
  1479
                     "none",
nathan@0
  1480
#endif
nathan@0
  1481
#ifdef WITH_OSS
nathan@0
  1482
                     dspdevice,
nathan@0
  1483
#else
nathan@0
  1484
                     "none",
nathan@0
  1485
#endif
nathan@0
  1486
                     imageconv,
nathan@0
  1487
                     imagecache,
nathan@0
  1488
                     sourcesSub ? sourcesSub:"empty"
nathan@0
  1489
                     );
nathan@0
  1490
  return help_str;
nathan@0
  1491
}
nathan@0
  1492
nathan@0
  1493
bool cPluginMp3::ProcessArgs(int argc, char *argv[])
nathan@0
  1494
{
nathan@0
  1495
  static struct option long_options[] = {
nathan@0
  1496
      { "mount",    required_argument, NULL, 'm' },
nathan@0
  1497
      { "network",  required_argument, NULL, 'n' },
nathan@0
  1498
      { "cddb",     required_argument, NULL, 'B' },
nathan@0
  1499
      { "dsp",      required_argument, NULL, 'D' },
nathan@0
  1500
      { "cache",    required_argument, NULL, 'C' },
nathan@0
  1501
      { "icache",   required_argument, NULL, 'c' },
nathan@0
  1502
      { "iconv",    required_argument, NULL, 'i' },
nathan@19
  1503
      { "defimage", required_argument, NULL, 'I' },
nathan@0
  1504
      { "sources",  required_argument, NULL, 'S' },
nathan@0
  1505
      { NULL }
nathan@0
  1506
    };
nathan@0
  1507
nathan@0
  1508
  int c, option_index = 0;
nathan@0
  1509
  while((c=getopt_long(argc,argv,"c:i:m:n:B:C:D:S:",long_options,&option_index))!=-1) {
nathan@0
  1510
    switch (c) {
nathan@0
  1511
      case 'i': imageconv=optarg; break;
nathan@0
  1512
      case 'c': imagecache=optarg; break;
nathan@0
  1513
      case 'm': mountscript=optarg; break;
nathan@0
  1514
      case 'n': netscript=optarg; break;
nathan@0
  1515
      case 'C': cachedir=optarg; break;
nathan@19
  1516
      case 'I': def_usr_img=optarg; break;
nathan@0
  1517
      case 'S': sourcesSub=optarg; break;
nathan@0
  1518
      case 'B':
nathan@0
  1519
#ifdef HAVE_SNDFILE
nathan@0
  1520
                cddbpath=optarg; break;
nathan@0
  1521
#else
nathan@0
  1522
                fprintf(stderr, "mp3: libsndfile support has not been compiled in!\n"); return false;
nathan@0
  1523
#endif
nathan@0
  1524
      case 'D':
nathan@0
  1525
#ifdef WITH_OSS
nathan@0
  1526
                dspdevice=optarg; break;
nathan@0
  1527
#else
nathan@0
  1528
                fprintf(stderr, "mp3: OSS output has not been compiled in!\n"); return false;
nathan@0
  1529
#endif
nathan@0
  1530
      default:  return false;
nathan@0
  1531
      }
nathan@0
  1532
    }
nathan@0
  1533
  return true;
nathan@0
  1534
}
nathan@0
  1535
nathan@0
  1536
bool cPluginMp3::Initialize(void)
nathan@22
  1537
{
nathan@22
  1538
  if(!CheckVDRVersion(1,4,5,"mp3")) return false;
nathan@2
  1539
  plugin_name="mp3";
nathan@2
  1540
#if APIVERSNUM < 10507
nathan@2
  1541
  i18n_name="mp3";
nathan@2
  1542
#else
nathan@2
  1543
  i18n_name="vdr-mp3";
nathan@2
  1544
#endif
nathan@0
  1545
  MP3Sources.Load(AddDirectory(ConfigDirectory(sourcesSub),"mp3sources.conf"));
nathan@0
  1546
  if(MP3Sources.Count()<1) {
nathan@0
  1547
     esyslog("ERROR: you should have defined at least one source in mp3sources.conf");
nathan@0
  1548
     fprintf(stderr,"No source(s) defined in mp3sources.conf\n");
nathan@0
  1549
     return false;
nathan@0
  1550
     }
nathan@0
  1551
  InfoCache.Load();
nathan@2
  1552
#if APIVERSNUM < 10507
nathan@0
  1553
  RegisterI18n(Phrases);
nathan@2
  1554
#endif
nathan@0
  1555
  mgr=new cPlayManager;
nathan@0
  1556
  if(!mgr) {
nathan@0
  1557
    esyslog("ERROR: creating playmanager failed");
nathan@0
  1558
    fprintf(stderr,"Creating playmanager failed\n");
nathan@0
  1559
    return false;
nathan@0
  1560
    }
nathan@0
  1561
  d(printf("mp3: using %s\n",mad_version))
nathan@0
  1562
  d(printf("mp3: compiled with %s\n",MAD_VERSION))
nathan@0
  1563
  return true;
nathan@0
  1564
}
nathan@0
  1565
nathan@0
  1566
void cPluginMp3::Housekeeping(void)
nathan@0
  1567
{
nathan@0
  1568
  InfoCache.Save();
nathan@0
  1569
}
nathan@0
  1570
nathan@0
  1571
const char *cPluginMp3::MainMenuEntry(void)
nathan@0
  1572
{
nathan@0
  1573
  return MP3Setup.HideMainMenu ? 0 : tr(MAINMENUENTRY);
nathan@0
  1574
}
nathan@0
  1575
nathan@0
  1576
cOsdObject *cPluginMp3::MainMenuAction(void)
nathan@0
  1577
{
nathan@0
  1578
  return new cMenuMP3;
nathan@0
  1579
}
nathan@0
  1580
nathan@0
  1581
cMenuSetupPage *cPluginMp3::SetupMenu(void)
nathan@0
  1582
{
nathan@0
  1583
  return new cMenuSetupMP3;
nathan@0
  1584
}
nathan@0
  1585
nathan@0
  1586
bool cPluginMp3::SetupParse(const char *Name, const char *Value)
nathan@0
  1587
{
nathan@0
  1588
  if      (!strcasecmp(Name, "InitLoopMode"))     MP3Setup.InitLoopMode    = atoi(Value);
nathan@0
  1589
  else if (!strcasecmp(Name, "InitShuffleMode"))  MP3Setup.InitShuffleMode = atoi(Value);
nathan@0
  1590
  else if (!strcasecmp(Name, "AudioMode"))        MP3Setup.AudioMode       = atoi(Value);
nathan@0
  1591
  else if (!strcasecmp(Name, "BgrScan"))          MP3Setup.BgrScan         = atoi(Value);
nathan@0
  1592
  else if (!strcasecmp(Name, "EditorMode"))       MP3Setup.EditorMode      = atoi(Value);
nathan@0
  1593
  else if (!strcasecmp(Name, "DisplayMode"))      MP3Setup.DisplayMode     = atoi(Value);
nathan@0
  1594
  else if (!strcasecmp(Name, "BackgrMode"))       MP3Setup.BackgrMode      = atoi(Value);
nathan@0
  1595
  else if (!strcasecmp(Name, "MenuMode"))         MP3Setup.MenuMode        = atoi(Value);
nathan@0
  1596
  else if (!strcasecmp(Name, "TargetLevel"))      MP3Setup.TargetLevel     = atoi(Value);
nathan@0
  1597
  else if (!strcasecmp(Name, "LimiterLevel"))     MP3Setup.LimiterLevel    = atoi(Value);
nathan@0
  1598
  else if (!strcasecmp(Name, "Only48kHz"))        MP3Setup.Only48kHz       = atoi(Value);
nathan@0
  1599
  else if (!strcasecmp(Name, "UseProxy"))         MP3Setup.UseProxy        = atoi(Value);
nathan@0
  1600
  else if (!strcasecmp(Name, "ProxyHost"))        strn0cpy(MP3Setup.ProxyHost,Value,MAX_HOSTNAME);
nathan@0
  1601
  else if (!strcasecmp(Name, "ProxyPort"))        MP3Setup.ProxyPort       = atoi(Value);
nathan@0
  1602
  else if (!strcasecmp(Name, "UseCddb"))          MP3Setup.UseCddb         = atoi(Value);
nathan@0
  1603
  else if (!strcasecmp(Name, "CddbHost"))         strn0cpy(MP3Setup.CddbHost,Value,MAX_HOSTNAME);
nathan@0
  1604
  else if (!strcasecmp(Name, "CddbPort"))         MP3Setup.CddbPort        = atoi(Value);
nathan@0
  1605
  else if (!strcasecmp(Name, "AbortAtEOL"))       MP3Setup.AbortAtEOL      = atoi(Value);
nathan@0
  1606
  else if (!strcasecmp(Name, "AudioOutMode")) {
nathan@0
  1607
    MP3Setup.AudioOutMode = atoi(Value);
nathan@0
  1608
#ifndef WITH_OSS
nathan@0
  1609
    if(MP3Setup.AudioOutMode==AUDIOOUTMODE_OSS) {
nathan@0
  1610
      esyslog("WARNING: AudioOutMode OSS not supported, falling back to DVB");
nathan@0
  1611
      MP3Setup.AudioOutMode=AUDIOOUTMODE_DVB;
nathan@0
  1612
      }
nathan@0
  1613
#endif
nathan@0
  1614
    }
nathan@0
  1615
  else if (!strcasecmp(Name, "ReplayDisplay"))      MP3Setup.ReplayDisplay = atoi(Value);
nathan@0
  1616
  else if (!strcasecmp(Name, "HideMainMenu"))       MP3Setup.HideMainMenu  = atoi(Value);
nathan@0
  1617
  else if (!strcasecmp(Name, "KeepSelect"))         MP3Setup.KeepSelect    = atoi(Value);
nathan@0
  1618
  else if (!strcasecmp(Name, "TitleArtistOrder"))   MP3Setup.TitleArtistOrder = atoi(Value);
nathan@0
  1619
  else return false;
nathan@0
  1620
  return true;
nathan@0
  1621
}
nathan@0
  1622
nathan@0
  1623
bool cPluginMp3::ExternalPlay(const char *path, bool test)
nathan@0
  1624
{
nathan@0
  1625
  char real[PATH_MAX+1];
nathan@0
  1626
  if(realpath(path,real)) {
nathan@0
  1627
    cFileSource *src=MP3Sources.FindSource(real);
nathan@0
  1628
    if(src) {
nathan@0
  1629
      cFileObj *item=new cFileObj(src,0,0,otFile);
nathan@0
  1630
      if(item) {
nathan@0
  1631
        item->SplitAndSet(real);
nathan@0
  1632
        if(item->GuessType()) {
nathan@0
  1633
          if(item->Exists()) {
nathan@0
  1634
            cInstantPlayList *pl=new cInstantPlayList(item);
nathan@0
  1635
            if(pl && pl->Load() && pl->Count()) {
nathan@0
  1636
              if(!test) cMP3Control::SetPlayList(pl);
nathan@0
  1637
              else delete pl;
nathan@0
  1638
              delete item;
nathan@0
  1639
              return true;
nathan@0
  1640
              }
nathan@0
  1641
            else dsyslog("MP3 service: error building playlist");
nathan@0
  1642
            delete pl;
nathan@0
  1643
            }
nathan@0
  1644
          else dsyslog("MP3 service: cannot play '%s'",path);
nathan@0
  1645
          }
nathan@0
  1646
        else dsyslog("MP3 service: GuessType() failed for '%s'",path);
nathan@0
  1647
        delete item;
nathan@0
  1648
        }
nathan@0
  1649
      }
nathan@0
  1650
    else dsyslog("MP3 service: cannot find source for '%s', real '%s'",path,real);
nathan@0
  1651
    }
nathan@0
  1652
  else if(errno!=ENOENT && errno!=ENOTDIR)
nathan@0
  1653
    esyslog("ERROR: realpath: %s: %s",path,strerror(errno));
nathan@0
  1654
  return false;
nathan@0
  1655
}
nathan@0
  1656
nathan@0
  1657
bool cPluginMp3::Service(const char *Id, void *Data)
nathan@0
  1658
{
nathan@0
  1659
  if(!strcasecmp(Id,"MP3-Play-v1")) {
nathan@0
  1660
    if(Data) {
nathan@0
  1661
      struct MPlayerServiceData *msd=(struct MPlayerServiceData *)Data;
nathan@0
  1662
      msd->result=ExternalPlay(msd->data.filename,false);
nathan@0
  1663
      }
nathan@0
  1664
    return true;
nathan@0
  1665
    }
nathan@0
  1666
  else if(!strcasecmp(Id,"MP3-Test-v1")) {
nathan@0
  1667
    if(Data) {
nathan@0
  1668
      struct MPlayerServiceData *msd=(struct MPlayerServiceData *)Data;
nathan@0
  1669
      msd->result=ExternalPlay(msd->data.filename,true);
nathan@0
  1670
      }
nathan@0
  1671
    return true;
nathan@0
  1672
    }
nathan@0
  1673
  return false;
nathan@0
  1674
}
nathan@0
  1675
nathan@0
  1676
const char **cPluginMp3::SVDRPHelpPages(void)
nathan@0
  1677
{
nathan@0
  1678
  static const char *HelpPages[] = {
nathan@0
  1679
    "PLAY <filename>\n"
nathan@0
  1680
    "    Triggers playback of file 'filename'.",
nathan@0
  1681
    "TEST <filename>\n"
nathan@0
  1682
    "    Tests is playback of file 'filename' is possible.",
nathan@0
  1683
    "CURR\n"
nathan@0
  1684
    "    Returns filename of song currently being replayed.",
nathan@0
  1685
    NULL
nathan@0
  1686
    };
nathan@0
  1687
  return HelpPages;
nathan@0
  1688
}
nathan@0
  1689
nathan@0
  1690
cString cPluginMp3::SVDRPCommand(const char *Command, const char *Option, int &ReplyCode)
nathan@0
  1691
{
nathan@0
  1692
  if(!strcasecmp(Command,"PLAY")) {
nathan@0
  1693
    if(*Option) {
nathan@0
  1694
      if(ExternalPlay(Option,false)) return "Playback triggered";
nathan@0
  1695
      else { ReplyCode=550; return "Playback failed"; }
nathan@0
  1696
      }
nathan@0
  1697
    else { ReplyCode=501; return "Missing filename"; }
nathan@0
  1698
    }
nathan@0
  1699
  else if(!strcasecmp(Command,"TEST")) {
nathan@0
  1700
    if(*Option) {
nathan@0
  1701
      if(ExternalPlay(Option,true)) return "Playback possible";
nathan@0
  1702
      else { ReplyCode=550; return "Playback not possible"; }
nathan@0
  1703
      }
nathan@0
  1704
    else { ReplyCode=501; return "Missing filename"; }
nathan@0
  1705
    }
nathan@0
  1706
  else if(!strcasecmp(Command,"CURR")) {
nathan@0
  1707
    cControl *control=cControl::Control();
nathan@0
  1708
    if(control && typeid(*control)==typeid(cMP3Control)) {
nathan@0
  1709
      cMP3PlayInfo mode;
nathan@0
  1710
      if(mgr->Info(-1,&mode)) return mode.Filename;
nathan@0
  1711
      else return "<unknown>";
nathan@0
  1712
      }
nathan@0
  1713
    else { ReplyCode=550; return "No running playback"; }
nathan@0
  1714
    }
nathan@0
  1715
  return NULL;
nathan@0
  1716
}
nathan@0
  1717
nathan@0
  1718
VDRPLUGINCREATOR(cPluginMp3); // Don't touch this!