]> git.sesse.net Git - vlc/blobdiff - modules/gui/skins2/parser/interpreter.cpp
mtime: Minimize imprecision and prevent overflow on darwin.
[vlc] / modules / gui / skins2 / parser / interpreter.cpp
index db48e34b074d1711ed789bb77dde6ee354abd54d..9f695aafffff9a9a2e9fe4d1c5d82f5cdf3675a9 100644 (file)
@@ -1,11 +1,11 @@
 /*****************************************************************************
  * interpreter.cpp
  *****************************************************************************
- * Copyright (C) 2003 VideoLAN
- * $Id: interpreter.cpp,v 1.2 2004/01/05 22:17:32 asmax Exp $
+ * Copyright (C) 2003 the VideoLAN team
+ * $Id$
  *
  * Authors: Cyril Deguet     <asmax@via.ecp.fr>
- *          Olivier Teulière <ipkiss@via.ecp.fr>
+ *          Olivier Teulière <ipkiss@via.ecp.fr>
  *
  * 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
  *
  * 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, USA.
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  *****************************************************************************/
 
 #include "interpreter.hpp"
+#include "expr_evaluator.hpp"
+#include "../commands/cmd_audio.hpp"
+#include "../commands/cmd_muxer.hpp"
 #include "../commands/cmd_playlist.hpp"
+#include "../commands/cmd_playtree.hpp"
 #include "../commands/cmd_dialogs.hpp"
 #include "../commands/cmd_dummy.hpp"
+#include "../commands/cmd_dvd.hpp"
 #include "../commands/cmd_layout.hpp"
 #include "../commands/cmd_quit.hpp"
+#include "../commands/cmd_minimize.hpp"
 #include "../commands/cmd_input.hpp"
 #include "../commands/cmd_fullscreen.hpp"
+#include "../commands/cmd_on_top.hpp"
+#include "../commands/cmd_show_window.hpp"
+#include "../commands/cmd_snapshot.hpp"
 #include "../src/theme.hpp"
+#include "../src/var_manager.hpp"
 #include "../src/vlcproc.hpp"
-#include "../vars/playlist.hpp"
-#include "../vars/vlcvars.hpp"
-#include "../vars/time.hpp"
-#include "../vars/volume.hpp"
 
 
 Interpreter::Interpreter( intf_thread_t *pIntf ): SkinObject( pIntf )
 {
+    /// Create the generic commands
+#define REGISTER_CMD( name, cmd ) \
+    m_commandMap[name] = CmdGenericPtr( new cmd( getIntf() ) );
+#define REGISTER_CMD1( name, cmd, a1 ) \
+    m_commandMap[name] = CmdGenericPtr( new cmd( getIntf(), a1 ) );
+
+    REGISTER_CMD( "none", CmdDummy )
+    REGISTER_CMD( "dialogs.changeSkin()", CmdDlgChangeSkin )
+    REGISTER_CMD( "dialogs.fileSimple()", CmdDlgFileSimple )
+    REGISTER_CMD( "dialogs.file()", CmdDlgFile )
+    REGISTER_CMD( "dialogs.directory()", CmdDlgDirectory )
+    REGISTER_CMD( "dialogs.disc()", CmdDlgDisc )
+    REGISTER_CMD( "dialogs.net()", CmdDlgNet )
+    REGISTER_CMD( "dialogs.playlist()", CmdDlgPlaylist )
+    REGISTER_CMD( "dialogs.messages()", CmdDlgMessages )
+    REGISTER_CMD( "dialogs.prefs()", CmdDlgPrefs )
+    REGISTER_CMD( "dialogs.fileInfo()", CmdDlgFileInfo )
+    REGISTER_CMD( "dialogs.streamingWizard()", CmdDlgStreamingWizard )
+
+    REGISTER_CMD( "dialogs.popup()", CmdDlgShowPopupMenu )
+    REGISTER_CMD( "dialogs.audioPopup()", CmdDlgShowAudioPopupMenu )
+    REGISTER_CMD( "dialogs.videoPopup()", CmdDlgShowVideoPopupMenu )
+    REGISTER_CMD( "dialogs.miscPopup()", CmdDlgShowMiscPopupMenu )
+
+    REGISTER_CMD( "dvd.nextTitle()", CmdDvdNextTitle )
+    REGISTER_CMD( "dvd.previousTitle()", CmdDvdPreviousTitle )
+    REGISTER_CMD( "dvd.nextChapter()", CmdDvdNextChapter )
+    REGISTER_CMD( "dvd.previousChapter()", CmdDvdPreviousChapter )
+    REGISTER_CMD( "dvd.rootMenu()", CmdDvdRootMenu )
+    REGISTER_CMD( "playlist.load()", CmdDlgPlaylistLoad )
+    REGISTER_CMD( "playlist.save()", CmdDlgPlaylistSave )
+    REGISTER_CMD( "playlist.add()", CmdDlgAdd )
+    REGISTER_CMD( "playlist.next()", CmdPlaylistNext )
+    REGISTER_CMD( "playlist.previous()", CmdPlaylistPrevious )
+    REGISTER_CMD1( "playlist.setRandom(true)", CmdPlaylistRandom, true )
+    REGISTER_CMD1( "playlist.setRandom(false)", CmdPlaylistRandom, false )
+    REGISTER_CMD1( "playlist.setLoop(true)", CmdPlaylistLoop, true )
+    REGISTER_CMD1( "playlist.setLoop(false)", CmdPlaylistLoop, false )
+    REGISTER_CMD1( "playlist.setRepeat(true)", CmdPlaylistRepeat, true )
+    REGISTER_CMD1( "playlist.setRepeat(false)", CmdPlaylistRepeat, false )
+    VarTree &rVarTree = VlcProc::instance( getIntf() )->getPlaytreeVar();
+    REGISTER_CMD1( "playlist.del()", CmdPlaytreeDel, rVarTree )
+    REGISTER_CMD1( "playtree.del()", CmdPlaytreeDel, rVarTree )
+    REGISTER_CMD( "playlist.sort()", CmdPlaytreeSort )
+    REGISTER_CMD( "playtree.sort()", CmdPlaytreeSort )
+    REGISTER_CMD( "vlc.fullscreen()", CmdFullscreen )
+    REGISTER_CMD( "vlc.play()", CmdPlay )
+    REGISTER_CMD( "vlc.pause()", CmdPause )
+    REGISTER_CMD( "vlc.stop()", CmdStop )
+    REGISTER_CMD( "vlc.faster()", CmdFaster )
+    REGISTER_CMD( "vlc.slower()", CmdSlower )
+    REGISTER_CMD( "vlc.mute()", CmdMute )
+    REGISTER_CMD( "vlc.volumeUp()", CmdVolumeUp )
+    REGISTER_CMD( "vlc.volumeDown()", CmdVolumeDown )
+    REGISTER_CMD( "vlc.minimize()", CmdMinimize )
+    REGISTER_CMD( "vlc.onTop()", CmdOnTop )
+    REGISTER_CMD( "vlc.snapshot()", CmdSnapshot )
+    REGISTER_CMD( "vlc.toggleRecord()", CmdToggleRecord )
+    REGISTER_CMD( "vlc.nextFrame()", CmdNextFrame )
+    REGISTER_CMD( "vlc.quit()", CmdQuit )
+    REGISTER_CMD1( "equalizer.enable()", CmdSetEqualizer, true )
+    REGISTER_CMD1( "equalizer.disable()", CmdSetEqualizer, false )
+
+    // Register the constant bool variables in the var manager
+    VarManager *pVarManager = VarManager::instance( getIntf() );
+    VarBool *pVarTrue = new VarBoolTrue( getIntf() );
+    pVarManager->registerVar( VariablePtr( pVarTrue ), "true" );
+    VarBool *pVarFalse = new VarBoolFalse( getIntf() );
+    pVarManager->registerVar( VariablePtr( pVarFalse ), "false" );
+
+#undef  REGISTER_CMD
+#undef  REGISTER_CMD1
 }
 
 
-CmdGeneric *Interpreter::parseAction( const string &rAction, Theme *pTheme )
+Interpreter *Interpreter::instance( intf_thread_t *pIntf )
 {
-    // XXX should not be so hardcoded!
-    CmdGeneric *pCommand = NULL;
-    if( rAction == "none" )
-    {
-        pCommand = new CmdDummy( getIntf() );
-    }
-    else if( rAction == "dialogs.changeSkin()" )
-    {
-        pCommand = new CmdDlgChangeSkin( getIntf() );
-    }
-    else if( rAction == "dialogs.fileSimple()" )
-    {
-        pCommand = new CmdDlgFileSimple( getIntf() );
-    }
-    else if( rAction == "dialogs.file()" )
-    {
-        pCommand = new CmdDlgFile( getIntf() );
-    }
-    else if( rAction == "dialogs.disc()" )
-    {
-        pCommand = new CmdDlgDisc( getIntf() );
-    }
-    else if( rAction == "dialogs.net()" )
-    {
-        pCommand = new CmdDlgNet( getIntf() );
-    }
-    else if( rAction == "dialogs.messages()" )
-    {
-        pCommand = new CmdDlgMessages( getIntf() );
-    }
-    else if( rAction == "dialogs.prefs()" )
-    {
-        pCommand = new CmdDlgPrefs( getIntf() );
-    }
-    else if( rAction == "dialogs.fileInfo()" )
-    {
-        pCommand = new CmdDlgFileInfo( getIntf() );
-    }
-    else if( rAction == "dialogs.popup()" )
-    {
-        pCommand = new CmdDlgPopupMenu( getIntf() );
-    }
-    else if( rAction == "playlist.add()" )
-    {
-        pCommand = new CmdDlgAdd( getIntf() );
-    }
-    else if( rAction == "playlist.del()" )
-    {
-        VarList &rVar = VlcProc::instance( getIntf() )->getPlaylistVar();
-        pCommand = new CmdPlaylistDel( getIntf(), rVar );
-    }
-    else if( rAction == "playlist.next()" )
-    {
-        pCommand = new CmdPlaylistNext( getIntf() );
-    }
-    else if( rAction == "playlist.previous()" )
+    if( ! pIntf->p_sys->p_interpreter )
     {
-        pCommand = new CmdPlaylistPrevious( getIntf() );
+        Interpreter *pInterpreter = new (std::nothrow) Interpreter( pIntf );
+        if( pInterpreter )
+            pIntf->p_sys->p_interpreter = pInterpreter;
     }
-    else if( rAction == "playlist.sort()" )
+    return pIntf->p_sys->p_interpreter;
+}
+
+
+void Interpreter::destroy( intf_thread_t *pIntf )
+{
+    delete pIntf->p_sys->p_interpreter;
+    pIntf->p_sys->p_interpreter = NULL;
+}
+
+
+CmdGeneric *Interpreter::parseAction( const string &rAction, Theme *pTheme )
+{
+    // Try to find the command in the global command map
+    if( m_commandMap.find( rAction ) != m_commandMap.end() )
     {
-        pCommand = new CmdPlaylistSort( getIntf() );
+        return m_commandMap[rAction].get();
     }
-    else if( rAction == "vlc.fullscreen()" )
+
+    CmdGeneric *pCommand = NULL;
+
+    if( rAction.find( ";" ) != string::npos )
     {
-        pCommand = new CmdFullscreen( getIntf() );
+        // Several actions are defined...
+        list<CmdGeneric *> actionList;
+        string rightPart = rAction;
+        string::size_type pos = rightPart.find( ";" );
+        while( pos != string::npos )
+        {
+            string leftPart = rightPart.substr( 0, pos );
+            // Remove any whitespace at the end of the left part, and parse it
+            leftPart =
+                leftPart.substr( 0, leftPart.find_last_not_of( " \t" ) + 1 );
+            actionList.push_back( parseAction( leftPart, pTheme ) );
+            // Now remove any whitespace at the beginning of the right part,
+            // and go on checking for further actions in it...
+            rightPart = rightPart.substr( pos, rightPart.size() );
+            if ( rightPart.find_first_not_of( " \t;" ) == string::npos )
+            {
+                // The right part is completely buggy, it's time to leave the
+                // loop...
+                rightPart = "";
+                break;
+            }
+
+            rightPart =
+                rightPart.substr( rightPart.find_first_not_of( " \t;" ),
+                                  rightPart.size() );
+            pos = rightPart.find( ";" );
+        }
+        actionList.push_back( parseAction( rightPart, pTheme ) );
+
+        // The list is filled, we remove NULL pointers from it, just in case...
+        actionList.remove( NULL );
+
+        pCommand = new CmdMuxer( getIntf(), actionList );
     }
-    else if( rAction == "vlc.quit()" )
+    else if( rAction.find( ".setLayout(" ) != string::npos )
     {
-        pCommand = new CmdQuit( getIntf() );
+        int leftPos = rAction.find( ".setLayout(" );
+        string windowId = rAction.substr( 0, leftPos );
+        // 11 is the size of ".setLayout("
+        int rightPos = rAction.find( ")", windowId.size() + 11 );
+        string layoutId = rAction.substr( windowId.size() + 11,
+                                          rightPos - (windowId.size() + 11) );
+
+        TopWindow *pWin = pTheme->getWindowById( windowId );
+        GenericLayout *pLayout = pTheme->getLayoutById( layoutId );
+        if( !pWin )
+        {
+            msg_Err( getIntf(), "unknown window (%s)", windowId.c_str() );
+        }
+        else if( !pLayout )
+        {
+            msg_Err( getIntf(), "unknown layout (%s)", layoutId.c_str() );
+        }
+        // Check that the layout doesn't correspond to another window
+        else if( pWin != pLayout->getWindow() )
+        {
+            msg_Err( getIntf(), "layout %s is not associated to window %s",
+                     layoutId.c_str(), windowId.c_str() );
+        }
+        else
+        {
+            pCommand = new CmdLayout( getIntf(), *pWin, *pLayout );
+        }
     }
-    else if( rAction == "vlc.stop()" )
+    else if( rAction.find( ".maximize()" ) != string::npos )
     {
-        pCommand = new CmdStop( getIntf() );
+        int leftPos = rAction.find( ".maximize()" );
+        string windowId = rAction.substr( 0, leftPos );
+
+        TopWindow *pWin = pTheme->getWindowById( windowId );
+        if( !pWin )
+        {
+            msg_Err( getIntf(), "unknown window (%s)", windowId.c_str() );
+        }
+        else
+        {
+            pCommand = new CmdMaximize( getIntf(),
+                                        pTheme->getWindowManager(),
+                                        *pWin );
+        }
     }
-    else if( rAction == "vlc.slower()" )
+    else if( rAction.find( ".unmaximize()" ) != string::npos )
     {
-        pCommand = new CmdSlower( getIntf() );
+        int leftPos = rAction.find( ".unmaximize()" );
+        string windowId = rAction.substr( 0, leftPos );
+
+        TopWindow *pWin = pTheme->getWindowById( windowId );
+        if( !pWin )
+        {
+            msg_Err( getIntf(), "unknown window (%s)", windowId.c_str() );
+        }
+        else
+        {
+            pCommand = new CmdUnmaximize( getIntf(),
+                                          pTheme->getWindowManager(),
+                                          *pWin );
+        }
     }
-    else if( rAction == "vlc.faster()" )
+    else if( rAction.find( ".show()" ) != string::npos )
     {
-        pCommand = new CmdFaster( getIntf() );
+        int leftPos = rAction.find( ".show()" );
+        string windowId = rAction.substr( 0, leftPos );
+
+        if( windowId == "playlist_window" &&
+            !var_InheritBool( getIntf(), "skinned-playlist") )
+        {
+            list<CmdGeneric *> list;
+            list.push_back( new CmdDlgPlaylist( getIntf() ) );
+            TopWindow *pWin = pTheme->getWindowById( windowId );
+            if( pWin )
+                list.push_back( new CmdHideWindow( getIntf(),
+                                                   pTheme->getWindowManager(),
+                                                   *pWin ) );
+            pCommand = new CmdMuxer( getIntf(), list );
+        }
+        else
+        {
+            TopWindow *pWin = pTheme->getWindowById( windowId );
+            if( pWin )
+            {
+                pCommand = new CmdShowWindow( getIntf(),
+                                              pTheme->getWindowManager(),
+                                              *pWin );
+            }
+            else
+            {
+                // It was maybe the id of a popup
+                Popup *pPopup = pTheme->getPopupById( windowId );
+                if( pPopup )
+                {
+                    pCommand = new CmdShowPopup( getIntf(), *pPopup );
+                }
+                else
+                {
+                    msg_Err( getIntf(), "unknown window or popup (%s)",
+                             windowId.c_str() );
+                }
+            }
+        }
     }
-    else if( rAction.find( ".setLayout(" ) != string::npos )
+    else if( rAction.find( ".hide()" ) != string::npos )
     {
-        int leftPos = rAction.find( ".setLayout(" );
+        int leftPos = rAction.find( ".hide()" );
         string windowId = rAction.substr( 0, leftPos );
-        // 11 is the size of ".setLayout("
-        int rightPos = rAction.find( ")", windowId.size() + 11 );
-        string layoutId = rAction.substr( windowId.size() + 11,
-                                          rightPos - (windowId.size() + 11) );
-        // XXX check the IDs (isalpha())
-        pCommand = new CmdLayout( getIntf(), windowId, layoutId );
+        if( windowId == "playlist_window" &&
+           !var_InheritBool( getIntf(), "skinned-playlist") )
+        {
+            list<CmdGeneric *> list;
+            list.push_back( new CmdDlgPlaylist( getIntf() ) );
+            TopWindow *pWin = pTheme->getWindowById( windowId );
+            if( pWin )
+                list.push_back( new CmdHideWindow( getIntf(),
+                                                   pTheme->getWindowManager(),
+                                                   *pWin ) );
+            pCommand = new CmdMuxer( getIntf(), list );
+        }
+        else
+        {
+            TopWindow *pWin = pTheme->getWindowById( windowId );
+            if( pWin )
+            {
+                pCommand = new CmdHideWindow( getIntf(),
+                                              pTheme->getWindowManager(),
+                                              *pWin );
+            }
+            else
+            {
+                msg_Err( getIntf(), "unknown window (%s)", windowId.c_str() );
+            }
+        }
     }
 
     if( pCommand )
@@ -145,6 +329,10 @@ CmdGeneric *Interpreter::parseAction( const string &rAction, Theme *pTheme )
         // Add the command in the pool
         pTheme->m_commands.push_back( CmdGenericPtr( pCommand ) );
     }
+    else
+    {
+        msg_Warn( getIntf(), "unknown action: %s", rAction.c_str() );
+    }
 
     return pCommand;
 }
@@ -152,71 +340,196 @@ CmdGeneric *Interpreter::parseAction( const string &rAction, Theme *pTheme )
 
 VarBool *Interpreter::getVarBool( const string &rName, Theme *pTheme )
 {
-    VarBool *pVar = NULL;
+    VarManager *pVarManager = VarManager::instance( getIntf() );
 
-    if( rName == "vlc.isPlaying" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->getIsPlayingVar();
-    }
-    else if( rName == "vlc.isSeekablePlaying" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->getIsSeekablePlayingVar();
-    }
-    else if( rName == "vlc.isMute" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->getIsMuteVar();
-    }
-    else if( rName.find( ".isVisible" ) != string::npos )
+    // Convert the expression into Reverse Polish Notation
+    ExprEvaluator evaluator( getIntf() );
+    evaluator.parse( rName );
+
+    list<VarBool*> varStack;
+
+    // Get the first token from the RPN stack
+    string token = evaluator.getToken();
+    while( !token.empty() )
     {
-        int leftPos = rName.find( ".isVisible" );
-        string windowId = rName.substr( 0, leftPos );
-        // XXX Need to check the IDs (isalpha())?
-        GenericWindow *pWin = pTheme->getWindowById( windowId );
-        if( pWin )
+        if( token == "and" )
+        {
+            // Get the 2 last variables on the stack
+            if( varStack.empty() )
+            {
+                msg_Err( getIntf(), "invalid boolean expression: %s",
+                         rName.c_str());
+                return NULL;
+            }
+            VarBool *pVar1 = varStack.back();
+            varStack.pop_back();
+            if( varStack.empty() )
+            {
+                msg_Err( getIntf(), "invalid boolean expression: %s",
+                         rName.c_str());
+                return NULL;
+            }
+            VarBool *pVar2 = varStack.back();
+            varStack.pop_back();
+
+            // Create a composite boolean variable
+            VarBool *pNewVar = new VarBoolAndBool( getIntf(), *pVar1, *pVar2 );
+            varStack.push_back( pNewVar );
+            // Register this variable in the manager
+            pVarManager->registerVar( VariablePtr( pNewVar ) );
+        }
+        else if( token == "or" )
         {
-            pVar = &pWin->getVisibleVar();
+            // Get the 2 last variables on the stack
+            if( varStack.empty() )
+            {
+                msg_Err( getIntf(), "invalid boolean expression: %s",
+                         rName.c_str());
+                return NULL;
+            }
+            VarBool *pVar1 = varStack.back();
+            varStack.pop_back();
+            if( varStack.empty() )
+            {
+                msg_Err( getIntf(), "invalid boolean expression: %s",
+                         rName.c_str());
+                return NULL;
+            }
+            VarBool *pVar2 = varStack.back();
+            varStack.pop_back();
+
+            // Create a composite boolean variable
+            VarBool *pNewVar = new VarBoolOrBool( getIntf(), *pVar1, *pVar2 );
+            varStack.push_back( pNewVar );
+            // Register this variable in the manager
+            pVarManager->registerVar( VariablePtr( pNewVar ) );
+        }
+        else if( token == "not" )
+        {
+            // Get the last variable on the stack
+            if( varStack.empty() )
+            {
+                msg_Err( getIntf(), "invalid boolean expression: %s",
+                         rName.c_str());
+                return NULL;
+            }
+            VarBool *pVar = varStack.back();
+            varStack.pop_back();
+
+            // Create a composite boolean variable
+            VarBool *pNewVar = new VarNotBool( getIntf(), *pVar );
+            varStack.push_back( pNewVar );
+            // Register this variable in the manager
+            pVarManager->registerVar( VariablePtr( pNewVar ) );
         }
         else
         {
-            msg_Warn( getIntf(), "Unknown window (%s)", windowId.c_str() );
+            // Try first to get the variable from the variable manager
+            // Indeed, if the skin designer is stupid enough to call a layout
+            // "dvd", we want "dvd.isActive" to resolve as the built-in action
+            // and not as the "layoutId.isActive" one.
+            VarBool *pVar = (VarBool*)pVarManager->getVar( token, "bool" );
+            if( pVar )
+            {
+                varStack.push_back( pVar );
+            }
+            else if( token.find( ".isVisible" ) != string::npos )
+            {
+                int leftPos = token.find( ".isVisible" );
+                string windowId = token.substr( 0, leftPos );
+                TopWindow *pWin = pTheme->getWindowById( windowId );
+                if( pWin )
+                {
+                    // Push the visibility variable onto the stack
+                    varStack.push_back( &pWin->getVisibleVar() );
+                }
+                else
+                {
+                    msg_Err( getIntf(), "unknown window (%s)",
+                             windowId.c_str() );
+                    return NULL;
+                }
+            }
+            else if( token.find( ".isMaximized" ) != string::npos )
+            {
+                int leftPos = token.find( ".isMaximized" );
+                string windowId = token.substr( 0, leftPos );
+                TopWindow *pWin = pTheme->getWindowById( windowId );
+                if( pWin )
+                {
+                    // Push the "maximized" variable onto the stack
+                    varStack.push_back( &pWin->getMaximizedVar() );
+                }
+                else
+                {
+                    msg_Err( getIntf(), "unknown window (%s)",
+                             windowId.c_str() );
+                    return NULL;
+                }
+            }
+            else if( token.find( ".isActive" ) != string::npos )
+            {
+                int leftPos = token.find( ".isActive" );
+                string layoutId = token.substr( 0, leftPos );
+                GenericLayout *pLayout = pTheme->getLayoutById( layoutId );
+                if( pLayout )
+                {
+                    // Push the isActive variable onto the stack
+                    varStack.push_back( &pLayout->getActiveVar() );
+                }
+                else
+                {
+                    msg_Err( getIntf(), "unknown layout (%s)",
+                             layoutId.c_str() );
+                    return NULL;
+                }
+            }
+            else
+            {
+                msg_Err( getIntf(), "cannot resolve boolean variable: %s",
+                         token.c_str());
+                return NULL;
+            }
         }
+        // Get the first token from the RPN stack
+        token = evaluator.getToken();
     }
 
-    return pVar;
+    // The stack should contain a single variable
+    if( varStack.size() != 1 )
+    {
+        msg_Err( getIntf(), "invalid boolean expression: %s", rName.c_str() );
+        return NULL;
+    }
+    return varStack.back();
 }
 
 
 VarPercent *Interpreter::getVarPercent( const string &rName, Theme *pTheme )
 {
-    VarPercent *pVar = NULL;
+    VarManager *pVarManager = VarManager::instance( getIntf() );
+    return static_cast<VarPercent*>(pVarManager->getVar( rName, "percent" ));
+}
 
-    if( rName == "time" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->getTimeVar();
-    }
-    else if( rName == "volume" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->getVolumeVar();
-    }
-    else if( rName == "playlist.slider" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->
-                getPlaylistVar().getPositionVar();
-    }
 
-    return pVar;
+VarList *Interpreter::getVarList( const string &rName, Theme *pTheme )
+{
+    VarManager *pVarManager = VarManager::instance( getIntf() );
+    return static_cast<VarList*>(pVarManager->getVar( rName, "list" ));
 }
 
 
-VarList *Interpreter::getVarList( const string &rName, Theme *pTheme )
+VarTree *Interpreter::getVarTree( const string &rName, Theme *pTheme )
 {
-    VarList *pVar = NULL;
+    VarManager *pVarManager = VarManager::instance( getIntf() );
+    return static_cast<VarTree*>(pVarManager->getVar( rName, "tree" ));
+}
 
-    if( rName == "playlist" )
-    {
-        pVar = &VlcProc::instance( getIntf() )->getPlaylistVar();
-    }
 
-    return pVar;
+string Interpreter::getConstant( const string &rValue )
+{
+    // Check if the value is a registered constant; if not, keep as is.
+    string val = VarManager::instance( getIntf() )->getConst( rValue );
+    return val.empty() ? rValue : val;
 }