/**************************************************************************** * Copyright (C) 2012 FIX94 * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . ****************************************************************************/ #include #include #include #include "MusicPlayer.hpp" #include "SoundHandler.hpp" #include "devicemounter/DeviceHandler.hpp" #include "list/ListGenerator.hpp" #include "gui/text.hpp" #include "gecko/gecko.hpp" #define MUSIC_DEPTH 10 Musicplayer MusicPlayer; static vector FileNames; static vector::const_iterator CurrentFileName; void Musicplayer::Cleanup() { Stop(); DisplayTime = 0; CurrentPosition = 0; MusicChanged = false; MusicStopped = true; FileNames.clear(); } static inline void FileNameAdder(char *Path) { /* No need for more checks */ FileNames.push_back(Path); } void Musicplayer::Init(Config &cfg, const string& musicDir, const string& themeMusicDir) { if(usingPlaylist) { InitPlaylist(cfg, curPlaylist, pl_device); return; } Cleanup(); FadeRate = cfg.getInt("GENERAL", "music_fade_rate", 8); Volume = cfg.getInt("GENERAL", "sound_volume_music", 255); SetVolume(0); MusicFile.SetVoice(0); vector Types = stringToVector(".mp3|.ogg", '|'); GetFiles(musicDir.c_str(), Types, FileNameAdder, false, MUSIC_DEPTH); GetFiles(themeMusicDir.c_str(), Types, FileNameAdder, false, MUSIC_DEPTH); if(cfg.getBool("GENERAL", "randomize_music", true) && FileNames.size() > 0) { srand(unsigned(time(NULL))); random_shuffle(FileNames.begin(), FileNames.end()); } usingPlaylist = false; OneSong = (FileNames.size() == 1); CurrentFileName = FileNames.begin(); } int Musicplayer::InitPlaylist(Config &cfg, const char *playlist, u8 device) { ifstream filestr; filestr.open(playlist); if(filestr.fail()) return 0; filestr.seekg(0,ios_base::end); int size = filestr.tellg(); if(size <= 0) return -1; filestr.seekg(0,ios_base::beg); string song; FileNames.clear(); while(!filestr.eof()) { getline(filestr, song, '\r'); if(song.find(".mp3") == string::npos && song.find(".ogg") == string::npos)// if not song path continue to next line continue; while(song.find("\\") != string::npos)// convert all '\' to '/' song.replace(song.find("\\"), 1, "/"); string::size_type p = song.find("/");// remove drive letter and anything else before first / song.erase(0, p); const char *songPath = fmt("%s:%s", DeviceName[device], song.c_str()); FileNames.push_back(songPath); } filestr.close(); curPlaylist = playlist; usingPlaylist = true; pl_device = device; if(cfg.getBool("GENERAL", "randomize_music", true) && FileNames.size() > 0) { srand(unsigned(time(NULL))); random_shuffle(FileNames.begin(), FileNames.end()); } OneSong = (FileNames.size() == 1); CurrentFileName = FileNames.begin(); LoadCurrentFile(); return 1; } void Musicplayer::SetFadeRate(u8 faderate) { FadeRate = faderate; } void Musicplayer::SetMaxVolume(u8 volume) { Volume = volume; SetVolume(volume); } void Musicplayer::SetVolume(u8 volume) { CurrentVolume = volume; MusicFile.SetVolume(CurrentVolume); } void Musicplayer::Previous() { if(FileNames.empty() || PosFromPrevFile()) return; if(CurrentFileName == FileNames.begin()) CurrentFileName = FileNames.end(); --CurrentFileName; LoadCurrentFile(); } void Musicplayer::Next() { if(FileNames.empty() || PosFromPrevFile()) return; ++CurrentFileName; if(CurrentFileName == FileNames.end()) CurrentFileName = FileNames.begin(); LoadCurrentFile(); } bool Musicplayer::PosFromPrevFile() { if(!CurrentPosition) return false; MusicFile.Load((*CurrentFileName).c_str()); SoundHandle.Decoder(MusicFile.GetVoice())->Seek(CurrentPosition); SetVolume(CurrentVolume); MusicFile.Play(); CurrentPosition = 0; MusicStopped = false; MusicChanged = false; return true; } void Musicplayer::Stop() { if(!MusicFile.IsPlaying()) return; MusicFile.Pause();// why not Stop() CurrentPosition = SoundHandle.Decoder(MusicFile.GetVoice())->Tell(); MusicFile.FreeMemory(); MusicStopped = true; } void Musicplayer::Tick(bool attenuate)// attenuate means fade to zero volume { if(FileNames.empty()) return; if(!attenuate && CurrentVolume < Volume) SetVolume(CurrentVolume + FadeRate > Volume ? Volume : CurrentVolume + FadeRate); else if(attenuate && CurrentVolume > 0) SetVolume(CurrentVolume - FadeRate < 0 ? 0 : CurrentVolume - FadeRate); if(!attenuate && !MusicFile.IsPlaying()) Next(); } void Musicplayer::LoadCurrentFile() { LoadFile(CurrentFileName->c_str()); } void Musicplayer::LoadFile(const char *name, bool display_change) { if(FileNames.empty()) { FileNames.push_back(PLUGIN_DOMAIN); CurrentFileName = FileNames.begin(); } else if(FileNames.size() == 1 && strcmp(name, PLUGIN_DOMAIN) == 0) { MusicFile.FreeMemory();// GuiSound is MusicFile in gui_sound.cpp MusicStopped = true; return; } MusicFile.Load(name); SetVolume(CurrentVolume); MusicFile.Play(); CurrentPosition = 0; MusicStopped = false; MusicChanged = display_change; } /* For our GUI */ wstringEx Musicplayer::GetFileName() { wstringEx CurrentFile; string CurrentFileStr((*CurrentFileName).begin()+(*CurrentFileName).find_last_of('/')+1, (*CurrentFileName).begin()+(*CurrentFileName).find_last_of('.')); CurrentFile.fromUTF8(CurrentFileStr.c_str()); return CurrentFile; } bool Musicplayer::SongChanged() { if(!MusicChanged) return false; MusicChanged = false; return true; }