1
0
mirror of git://projects.qi-hardware.com/gmenu2x.git synced 2024-07-02 18:54:31 +03:00
gmenu2x/src/linkapp.cpp

602 lines
16 KiB
C++
Raw Normal View History

/***************************************************************************
* Copyright (C) 2006 by Massimiliano Torromeo *
* massimiliano.torromeo@gmail.com *
* *
* 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 2 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, write to the *
* Free Software Foundation, Inc., *
* 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
***************************************************************************/
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <fstream>
#include <sstream>
#include "linkapp.h"
#include "menu.h"
#include "selector.h"
#include "textmanualdialog.h"
#include "debug.h"
using namespace std;
LinkApp::LinkApp(GMenu2X *gmenu2x_, InputManager &inputMgr_,
const char* linkfile)
: Link(gmenu2x_)
, inputMgr(inputMgr_)
{
manual = "";
file = linkfile;
wrapper = false;
dontleave = false;
setClock(336);
setVolume(-1);
//G
//setGamma(0);
setBacklight(-1);
selectordir = "";
selectorfilter = "";
icon = iconPath = "";
selectorbrowser = false;
useRamTimings = false;
string line;
ifstream infile (linkfile, ios_base::in);
while (getline(infile, line, '\n')) {
line = trim(line);
if (line=="") continue;
if (line[0]=='#') continue;
string::size_type position = line.find("=");
string name = trim(line.substr(0,position));
string value = trim(line.substr(position+1));
if (name == "title") {
title = value;
} else if (name == "description") {
description = value;
} else if (name == "icon") {
setIcon(value);
} else if (name == "exec") {
exec = value;
} else if (name == "params") {
params = value;
} else if (name == "workdir") {
workdir = value;
} else if (name == "manual") {
manual = value;
} else if (name == "wrapper") {
if (value=="true") wrapper = true;
} else if (name == "dontleave") {
if (value=="true") dontleave = true;
} else if (name == "clock") {
setClock( atoi(value.c_str()) );
//G
} else if (name == "gamma") {
setGamma( atoi(value.c_str()) );
} else if (name == "backlight") {
setBacklight( atoi(value.c_str()) );
} else if (name == "volume") {
setVolume( atoi(value.c_str()) );
} else if (name == "selectordir") {
setSelectorDir( value );
} else if (name == "selectorbrowser") {
if (value=="true") selectorbrowser = true;
} else if (name == "useramtimings") {
if (value=="true") useRamTimings = true;
} else if (name == "selectorfilter") {
setSelectorFilter( value );
} else if (name == "selectorscreens") {
setSelectorScreens( value );
} else if (name == "selectoraliases") {
setAliasFile( value );
} else {
WARNING("Unrecognized option: '%s'\n", name.c_str());
break;
}
}
infile.close();
if (iconPath.empty()) searchIcon();
edited = false;
}
const string &LinkApp::searchIcon() {
string execicon = exec;
string::size_type pos = exec.rfind(".");
if (pos != string::npos) execicon = exec.substr(0,pos);
execicon += ".png";
string exectitle = execicon;
pos = execicon.rfind("/");
if (pos != string::npos)
string exectitle = execicon.substr(pos+1,execicon.length());
if (!gmenu2x->sc.getSkinFilePath("icons/"+exectitle).empty())
iconPath = gmenu2x->sc.getSkinFilePath("icons/"+exectitle);
else if (fileExists(execicon))
iconPath = execicon;
else
iconPath = gmenu2x->sc.getSkinFilePath("icons/generic.png");
return iconPath;
}
int LinkApp::clock() {
return iclock;
}
const string &LinkApp::clockStr(int maxClock) {
if (iclock>maxClock) setClock(maxClock);
return sclock;
}
void LinkApp::setClock(int mhz) {
iclock = constrain(mhz,200,430);
stringstream ss;
sclock = "";
ss << iclock << "Mhz";
ss >> sclock;
edited = true;
}
int LinkApp::volume() {
return ivolume;
}
const string &LinkApp::volumeStr() {
return svolume;
}
void LinkApp::setVolume(int vol) {
ivolume = constrain(vol,-1,100);
stringstream ss;
svolume = "";
if (ivolume<0)
ss << gmenu2x->confInt["globalVolume"];
else
ss << ivolume;
ss >> svolume;
edited = true;
}
int LinkApp::backlight()
{
return ibacklight;
}
const string &LinkApp::backlightStr()
{
return sbacklight;
}
void LinkApp::setBacklight(int val)
{
ibacklight = constrain(val,-1,100);
stringstream ss;
sbacklight = "";
ss << ibacklight;
ss >> sbacklight;
edited = true;
}
//G
int LinkApp::gamma() {
return igamma;
}
const string &LinkApp::gammaStr() {
return sgamma;
}
void LinkApp::setGamma(int gamma) {
igamma = constrain(gamma,0,100);
stringstream ss;
sgamma = "";
ss << igamma;
ss >> sgamma;
edited = true;
}
// /G
bool LinkApp::targetExists() {
#ifndef TARGET_GP2X
return true; //For displaying elements during testing on pc
#endif
string target = exec;
if (!exec.empty() && exec[0]!='/' && !workdir.empty())
target = workdir + "/" + exec;
return fileExists(target);
}
bool LinkApp::save() {
if (!edited) return false;
ofstream f(file.c_str());
if (f.is_open()) {
if (title!="" ) f << "title=" << title << endl;
if (description!="" ) f << "description=" << description << endl;
if (icon!="" ) f << "icon=" << icon << endl;
if (exec!="" ) f << "exec=" << exec << endl;
if (params!="" ) f << "params=" << params << endl;
if (workdir!="" ) f << "workdir=" << workdir << endl;
if (manual!="" ) f << "manual=" << manual << endl;
if (iclock!=0 ) f << "clock=" << iclock << endl;
if (useRamTimings ) f << "useramtimings=true" << endl;
if (ivolume>0 ) f << "volume=" << ivolume << endl;
//G
if (igamma!=0 ) f << "gamma=" << igamma << endl;
if (ibacklight!=0 ) f << "backlight=" << ibacklight << endl;
if (selectordir!="" ) f << "selectordir=" << selectordir << endl;
if (selectorbrowser ) f << "selectorbrowser=true" << endl;
if (selectorfilter!="" ) f << "selectorfilter=" << selectorfilter << endl;
if (selectorscreens!="") f << "selectorscreens=" << selectorscreens << endl;
if (aliasfile!="" ) f << "selectoraliases=" << aliasfile << endl;
if (wrapper ) f << "wrapper=true" << endl;
if (dontleave ) f << "dontleave=true" << endl;
f.close();
sync();
return true;
} else
ERROR("Error while opening the file '%s' for write.\n", file.c_str());
return false;
}
void LinkApp::drawRun() {
//Darkened background
gmenu2x->s->box(0, 0, gmenu2x->resX, gmenu2x->resY, 0,0,0,150);
string text = gmenu2x->tr.translate("Launching $1",getTitle().c_str(),NULL);
int textW = gmenu2x->font->getTextWidth(text);
int boxW = 62+textW;
int halfBoxW = boxW/2;
//outer box
gmenu2x->s->box(gmenu2x->halfX-2-halfBoxW, gmenu2x->halfY-23, halfBoxW*2+5, 47, gmenu2x->skinConfColors[COLOR_MESSAGE_BOX_BG]);
//inner rectangle
gmenu2x->s->rectangle(gmenu2x->halfX-halfBoxW, gmenu2x->halfY-21, boxW, 42, gmenu2x->skinConfColors[COLOR_MESSAGE_BOX_BORDER]);
int x = gmenu2x->halfX+10-halfBoxW;
/*if (getIcon()!="")
gmenu2x->sc[getIcon()]->blit(gmenu2x->s,x,104);
else
gmenu2x->sc["icons/generic.png"]->blit(gmenu2x->s,x,104);*/
gmenu2x->sc[getIconPath()]->blit(gmenu2x->s,x,gmenu2x->halfY-16);
gmenu2x->s->write( gmenu2x->font, text, x+42, gmenu2x->halfY+1, SFontHAlignLeft, SFontVAlignMiddle );
gmenu2x->s->flip();
}
void LinkApp::run() {
if (selectordir!="")
selector();
else
launch();
}
void LinkApp::showManual() {
if (manual=="" || !fileExists(manual)) return;
// Png manuals
string ext8 = manual.substr(manual.size()-8,8);
if (ext8==".man.png" || ext8==".man.bmp" || ext8==".man.jpg" || manual.substr(manual.size()-9,9)==".man.jpeg") {
//Raise the clock to speed-up the loading of the manual
gmenu2x->setClock(336);
Surface pngman(manual);
Surface bg(gmenu2x->confStr["wallpaper"],false);
stringstream ss;
string pageStatus;
bool close = false, repaint = true;
int page=0, pagecount=pngman.raw->w/320;
ss << pagecount;
string spagecount;
ss >> spagecount;
//Lower the clock
gmenu2x->setClock(gmenu2x->confInt["menuClock"]);
while (!close) {
if (repaint) {
bg.blit(gmenu2x->s, 0, 0);
pngman.blit(gmenu2x->s, -page*320, 0);
gmenu2x->drawBottomBar();
gmenu2x->drawButton(gmenu2x->s, "x", gmenu2x->tr["Exit"],
gmenu2x->drawButton(gmenu2x->s, "right", gmenu2x->tr["Change page"],
gmenu2x->drawButton(gmenu2x->s, "left", "", 5)-10));
ss.clear();
ss << page+1;
ss >> pageStatus;
pageStatus = gmenu2x->tr["Page"]+": "+pageStatus+"/"+spagecount;
gmenu2x->s->write(gmenu2x->font, pageStatus, 310, 230, SFontHAlignRight, SFontVAlignMiddle);
gmenu2x->s->flip();
repaint = false;
}
switch(inputMgr.waitForPressedButton()) {
case MANUAL:
case CLEAR:
case SETTINGS:
close = true;
break;
case LEFT:
if (page > 0) {
page--;
repaint = true;
}
break;
case RIGHT:
if (page < pagecount-1) {
page++;
repaint=true;
}
break;
default:
break;
}
}
return;
}
// Txt manuals
if (manual.substr(manual.size()-8,8)==".man.txt") {
vector<string> txtman;
string line;
ifstream infile(manual.c_str(), ios_base::in);
if (infile.is_open()) {
gmenu2x->setClock(336);
while (getline(infile, line, '\n')) txtman.push_back(line);
infile.close();
TextManualDialog tmd(gmenu2x, getTitle(), getIconPath(), &txtman);
gmenu2x->setClock(gmenu2x->confInt["menuClock"]);
tmd.exec();
}
return;
}
//Readmes
vector<string> readme;
string line;
ifstream infile(manual.c_str(), ios_base::in);
if (infile.is_open()) {
gmenu2x->setClock(336);
while (getline(infile, line, '\n')) readme.push_back(line);
infile.close();
TextDialog td(gmenu2x, getTitle(), "ReadMe", getIconPath(), &readme);
gmenu2x->setClock(gmenu2x->confInt["menuClock"]);
td.exec();
}
}
void LinkApp::selector(int startSelection, const string &selectorDir) {
//Run selector interface
Selector sel(gmenu2x, this, selectorDir);
int selection = sel.exec(startSelection);
if (selection!=-1) {
gmenu2x->writeTmp(selection, sel.getDir());
launch(sel.getFile(), sel.getDir());
}
}
void LinkApp::launch(const string &selectedFile, const string &selectedDir) {
drawRun();
save();
#ifndef TARGET_GP2X
//delay for testing
SDL_Delay(1000);
#endif
//Set correct working directory
string wd = workdir;
if (wd=="") {
string::size_type pos = exec.rfind("/");
if (pos!=string::npos)
wd = exec.substr(0,pos);
}
if (!wd.empty()) {
if (wd[0]!='/') wd = gmenu2x->getExePath() + wd;
chdir(wd.c_str());
}
//selectedFile
if (selectedFile!="") {
string selectedFileExtension;
string selectedFileName;
string dir;
string::size_type i = selectedFile.rfind(".");
if (i != string::npos) {
selectedFileExtension = selectedFile.substr(i,selectedFile.length());
selectedFileName = selectedFile.substr(0,i);
}
if (selectedDir=="")
dir = getSelectorDir();
else
dir = selectedDir;
if (params=="") {
params = cmdclean(dir+selectedFile);
} else {
string origParams = params;
params = strreplace(params,"[selFullPath]",cmdclean(dir+selectedFile));
params = strreplace(params,"[selPath]",cmdclean(dir));
params = strreplace(params,"[selFile]",cmdclean(selectedFileName));
params = strreplace(params,"[selExt]",cmdclean(selectedFileExtension));
if (params == origParams) params += " " + cmdclean(dir+selectedFile);
}
}
if (useRamTimings)
gmenu2x->applyRamTimings();
if (volume()>=0)
gmenu2x->setVolume(volume());
INFO("Executing '%s' (%s %s)\n", title.c_str(), exec.c_str(), params.c_str());
//check if we have to quit
string command = cmdclean(exec);
// Check to see if permissions are desirable
struct stat fstat;
if( stat( command.c_str(), &fstat ) == 0 ) {
struct stat newstat = fstat;
if( S_IRUSR != ( fstat.st_mode & S_IRUSR ) )
newstat.st_mode |= S_IRUSR;
if( S_IXUSR != ( fstat.st_mode & S_IXUSR ) )
newstat.st_mode |= S_IXUSR;
if( fstat.st_mode != newstat.st_mode )
chmod( command.c_str(), newstat.st_mode );
} // else, well.. we are no worse off :)
if (params!="") command += " " + params;
if (gmenu2x->confInt["outputLogs"]) command += " &> " + cmdclean(gmenu2x->getExePath()) + "/log.txt";
if (wrapper) command += "; sync & cd "+cmdclean(gmenu2x->getExePath())+"; exec ./gmenu2x";
if (dontleave) {
system(command.c_str());
} else {
if (gmenu2x->confInt["saveSelection"] && (gmenu2x->confInt["section"]!=gmenu2x->menu->selSectionIndex() || gmenu2x->confInt["link"]!=gmenu2x->menu->selLinkIndex()))
gmenu2x->writeConfig();
if (gmenu2x->fwType == "open2x" && gmenu2x->savedVolumeMode != gmenu2x->volumeMode)
gmenu2x->writeConfigOpen2x();
if (selectedFile=="")
gmenu2x->writeTmp();
gmenu2x->quit();
if (clock()!=gmenu2x->confInt["menuClock"])
gmenu2x->setClock(clock());
//if (gamma()!=0 && gamma()!=gmenu2x->confInt["gamma"])
// gmenu2x->setGamma(gamma());
if((backlight() != 0) && (backlight() != gmenu2x->confInt["backlight"]))
gmenu2x->setBacklight(backlight());
execlp("/bin/sh","/bin/sh","-c",command.c_str(),NULL);
//if execution continues then something went wrong and as we already called SDL_Quit we cannot continue
//try relaunching gmenu2x
chdir(gmenu2x->getExePath().c_str());
execlp("./gmenu2x", "./gmenu2x", NULL);
}
chdir(gmenu2x->getExePath().c_str());
}
const string &LinkApp::getExec() {
return exec;
}
void LinkApp::setExec(const string &exec) {
this->exec = exec;
edited = true;
}
const string &LinkApp::getParams() {
return params;
}
void LinkApp::setParams(const string &params) {
this->params = params;
edited = true;
}
const string &LinkApp::getWorkdir() {
return workdir;
}
void LinkApp::setWorkdir(const string &workdir) {
this->workdir = workdir;
edited = true;
}
const string &LinkApp::getManual() {
return manual;
}
void LinkApp::setManual(const string &manual) {
this->manual = manual;
edited = true;
}
const string &LinkApp::getSelectorDir() {
return selectordir;
}
void LinkApp::setSelectorDir(const string &selectordir) {
this->selectordir = selectordir;
if (this->selectordir!="" && this->selectordir[this->selectordir.length()-1]!='/') this->selectordir += "/";
edited = true;
}
bool LinkApp::getSelectorBrowser() {
return selectorbrowser;
}
void LinkApp::setSelectorBrowser(bool value) {
selectorbrowser = value;
edited = true;
}
bool LinkApp::getUseRamTimings() {
return useRamTimings;
}
void LinkApp::setUseRamTimings(bool value) {
useRamTimings = value;
edited = true;
}
const string &LinkApp::getSelectorFilter() {
return selectorfilter;
}
void LinkApp::setSelectorFilter(const string &selectorfilter) {
this->selectorfilter = selectorfilter;
edited = true;
}
const string &LinkApp::getSelectorScreens() {
return selectorscreens;
}
void LinkApp::setSelectorScreens(const string &selectorscreens) {
this->selectorscreens = selectorscreens;
edited = true;
}
const string &LinkApp::getAliasFile() {
return aliasfile;
}
void LinkApp::setAliasFile(const string &aliasfile) {
if (fileExists(aliasfile)) {
this->aliasfile = aliasfile;
edited = true;
}
}
void LinkApp::renameFile(const string &name) {
file = name;
}