####(c)www.stani.be
import _spe.info as info
INFO=info.copy()

INFO['description']=\
"""Subclassed smdi Parent frame."""

__doc__=INFO['doc']%INFO

####Importing modules-----------------------------------------------------------

#---general modules
import ConfigParser,os,string,sys,thread,time,types,webbrowser, pprint
import _spe,sm.scriptutils,sm.wxp
import dialogs.stcStyleEditor

#---wxPython
import wx
import wx.stc
from wx.lib.evtmgr import eventManager

BLENDER_MESSAGE             = 'Spe must be launched within Blender for this feature.'
STATUS_TEXT_WORKSPACE_POS   = 2 # Todo: can this be done more dynamic?

import Child

####Constants-------------------------------------------------------------------
DEFAULT         = "<default>"
FIREFOX         = '/usr/bin/firefox'
NAUTILUS        = '/usr/bin/nautilus --no-desktop'
DOLPHIN         = '/usr/bin/dolphin'
KONQUEROR       = '/usr/bin/konqueror'
THUNAR          = '/usr/bin/thunar'
PCMANFM         = '/usr/bin/pcmanfm'
HELP_SORRY      = "Sorry, '%s' was not found on your system, getting it from internet instead."
HELP_WWW        = 'http://www.python.org/doc/current/%s/%s.html'
MAIL            = 'mailto:spe.stani.be@gmail.com?subject=About spe...'
PATH            = info.dirname(__file__)
PLATFORM        = sys.platform
PREFIX          = sys.prefix
SIZE            = (600,1)
SKIN            = 'default'
TABS            = ['Shell','Locals','Session','Output','Find','Browser','Recent','Todo','Index','Notes','Donate']
TITLE           = 'SPE %s'
UNNAMED         = 'unnamed'
RECENT          = 'recent.txt'
FOLDERS         = 'folders.txt'
NOTES           = 'notes.txt'
REMEMBER        = 'remember.txt'
if info.LINUX:
    STYLE       = wx.NB_TOP
else:
    STYLE       = wx.NB_BOTTOM


BLENDER_SHORTCUT_SPE    = "spe_blender.py" #a shortcut to SPE from Blender
BLENDER_SHORTCUT_WINPDB = "winpdb_blender.py" #a shortcut to Winpdb from Blender

####Subclassed Parent class-----------------------------------------------------
class Panel(wx.Notebook):
    ####Constructors
    def __init__(self, parent, openFiles=[], splash=None, redirect=1, path=PATH,
                 size = SIZE,**settings):
        wx.Notebook.__init__(self,parent=parent,id=wx.ID_ANY,style=STYLE,size=size)
        self.__paths__(path)
        self.__settings__(openFiles,redirect,**settings)
        self.__findReplaceEvents__()
        # Todo: make this a real class instead of this lazy one
        self.workspace=self.emptyWorkspace()

    def __timer__(self):
        self.lock           = thread.allocate_lock()
        #self.Bind(wx.EVT_IDLE,self.onIdle)
        self.Bind(wx.EVT_TIMER,self.onTimer)
        #self.was_idle       = False
        self.timer          = wx.Timer(self)
        wx.CallAfter(self.timer.Start,self.getValue('Redraw'))
        
    def __paths__(self,path,skin='default'):
        self.path           = path
        self.pathDoc        = os.path.join(self.path,       'doc')
        self.pathSkins      = os.path.join(self.path,       'skins')
        self.pathImages     = os.path.join(self.pathSkins,  skin)
        self.pathPlugins    = os.path.join(self.path,       'plugins')
        self.pathTabs       = os.path.join(self.path,       'tabs')
        if self.pathPlugins not in sys.path:
            sys.path.insert(0,self.pathPlugins)
        try:
            os.mkdir(INFO['userPath'])
        except:
            if not os.path.exists(INFO['userPath']):
                print 'Warning: could not find or create user path (%s).'%INFO['userPath']

    def __settings__(self,openFiles,redirect,redraw=None,Blender=None,**kwds):
        #arguments
        self._openFiles         = openFiles
        self._redirect          = redirect
        self._simultaneous      = False
        self.redraw             = redraw
        self.Blender            = Blender
        #panel
        self.argumentsPrevious  = []
        self.beepPrevious       = False
        self.defaultEncoding    = None
        self.findDialog         = None
        self.folders            = []
        self.kiki               = None
        self.remember           = 0
        self.restartMessage     = ''
        self.runner             = None
        if PLATFORM == 'win32':
            self.LIST_STYLE     = wx.LC_SMALL_ICON# todo: verify this better |wx.LC_LIST
        else:
            self.LIST_STYLE     = wx.LC_LIST

    def __findReplaceEvents__(self):
        self.findStr=''
        self.replaceStr=''
        self.findFlags=1
        self.stcFindFlags=0
        #This can't be done with the eventManager unfortunately ;-(
        wx.EVT_COMMAND_FIND(self,-1,self.onFind)
        wx.EVT_COMMAND_FIND_NEXT(self, -1,self.onFind)
        wx.EVT_COMMAND_FIND_REPLACE(self, -1,self.onReplace)
        wx.EVT_COMMAND_FIND_REPLACE_ALL(self, -1,self.onReplaceAll)
        wx.EVT_COMMAND_FIND_CLOSE(self, -1,self.onFindClose)

    #---finish
    def __finish__(self):
        self.__icons__()
        self.__sash__()
        self.__frame__()
        #self.app.childActive.source.SetFocus()

    def __icons__(self):
        self.iconsList=wx.ImageList(16,16)
        self.icons={}
        self.iconsListIndex={}
        iconFiles=sm.osx.listdir(self.pathImages,extensions=['.png'])
        iconFiles.sort()
        for icon in iconFiles:
            self.icons[icon]=self.app.bitmap(icon)
            if self.icons[icon].GetHeight() == 16:
                self.iconsListIndex[icon]=self.iconsList.Add(self.icons[icon])

    def __sash__(self):
        if self.app.DEBUG: print 'Creating tabs...'
        self.config = self.app.config
        self.AssignImageList(self.iconsList)
        tabs = [os.path.splitext(x)[0] for x in sm.osx.listdir(self.pathTabs,extensions=['.py']) if x[:1]!='_']
        tabs.sort()
        tabs = TABS[:-1] + [x for x in tabs if x not in TABS] + [TABS[-1]]
        if not self.app.Blender:
            tabs.remove('Blender')
        self.tabPanel = {}
        for tab in tabs:
            if self.app.DEBUG: print '\t',tab
            __import__('_spe.tabs.'+tab)
            page = self.__dict__[tab.lower()] = eval('_spe.tabs.%s.Panel'%tab)(self)
            if info.DARWIN and tab!='Shell':    text    = ''
            else:                               text    = tab
            self.AddPage(page=page,text=text,imageId=self.iconsListIndex[tab.lower()+'.png'])
            self.tabPanel[tab] = page
        self.tabs = tabs
        if self.get('version')!=INFO['version']:
            self.SetSelection(self.GetPageCount()-1)
            self.set('version',INFO['version'])
        eventManager.Register(self.onTab,wx.EVT_NOTEBOOK_PAGE_CHANGED,self)
        if self.getValue('Redraw') < 10:
            self.set('Redraw',self.getValue('Redraw')*1000)
        self.__timer__()

    def __frame__(self):
        #parent frame
        frame   = self.frame
        frame.SetDropTarget(Child.DropOpen(self.openList))
        icon    = wx.Icon(os.path.join(self.pathImages,'favicon.ico'),wx.BITMAP_TYPE_ICO)
        frame.SetIcon(icon)
        if hasattr(frame,'panelFrame'):
            frame.panelFrame.SetIcon(icon)
        #constructors
        self.preferencesSave()
        self.__remember__(openFiles=self._openFiles)
        self.__menus__()

    def __menus__(self):
        #parentInit.importMenus
        menuBar = self.frame.menuBar
        if self.app.mdi:
            #d#self.frame.menuBar.parentPanel = self
            menuBar.check_sidebar()
            menuBar.check_view()
        if not self.app.children:
            menuBar.enable(0)

    def __remember__(self,openFiles=[]):

        self.__openWorkspace__()  #this prepares the currentworkspace to be used
        try:
            self.get("currentworkspace") #This will fail if there is no current workspace to get
        except:
            pass # Open default workspace?
        self.loadWorkspace()
        if len(openFiles)>0:
            self.rememberSet(1)
            self.openList(openFiles)
        if len(self.app.children)==0:
            self.new(maximize = self.getValue("MaxChildren"))

    #---Workspace
    def _createNewDefaultWorkspace(self):
        #if not os.path.exists(self.workspace['file']):
        file=os.path.join(INFO['userPath'],'defaults.sws')
        if not os.path.isfile(file):
            new_cf=ConfigParser.ConfigParser()

            # Recent
            new_cf.add_section("recent")
            try:
                new_cf.set("recent","1",self.userOpen(RECENT))
            except:
                new_cf.set("recent","1","")

            # Folders
            new_cf.add_section("folders")
            try:
                new_cf.set("folders","1",self.userOpen(FOLDERS))
            except:
                new_cf.set("folders","1","")

            # Notes
            new_cf.add_section("notes")
            try:
                new_cf.set("notes","1",self.userOpen(NOTES))
            except:
                new_cf.set("notes","1","")

            # OpenFiles
            new_cf.add_section("openfiles")
            try:
                new_cf.set("openfiles","1",self.userOpen(REMEMBER))
            except:
                new_cf.set("openfiles","1","")
            f=open(file,"w")
            new_cf.write(f)
            f.close()
            return file
    def __openWorkspace__(self):
        #read the default workspace for the 'global' items
        file=INFO['defaultWorkspace']
        if not os.path.isfile(file):
            self._createNewDefaultWorkspace()
        self.workspace['defaultconfig']=ConfigParser.ConfigParser()
        self.workspace['defaultconfig'].read(file)
        
        #read the specific workspace for the 'local' items
        #this can be blocked in two ways:
        # 1 self.getValue('RememberLastWorkspace') is False
        # 2 incorrect workspace files
        self.workspace['config']=ConfigParser.ConfigParser()
        if self.getValue('RememberLastWorkspace'):
            try:
                file    = self.get("currentworkspace")
                if not os.path.exists(file): file = ''
            except:
                file    = ''
            try:
                self.workspace['config'].read(file)
                self.workspace['file']=file
            except:
                file    = ''
        else:
            file    = ''
        if file is '':
            self.workspace['config']=self.workspace['defaultconfig']
            self.workspace['file']=INFO['defaultWorkspace']
        self.workspace['openfiles']=[]
        if self.getValue('RememberLastWorkspace'):
            #open the workspace files in SPE
            try:
                openFiles=eval(self.getWorkspaceValue("OpenFiles"))
                for i in openFiles:
                    self.workspace['openfiles'].append(i[0])
            except Exception,e:
                if self.app.DEBUG:
                    self.SetStatusText("Error opening workspace file %s: %s"%(file,e))
        else:
            self.setWorkspaceValue("openfiles",str([]))

    def applyWorkspaceTab(self,child):
        """ this function will change the child tab to indicate that it is part of the workspace """
        self.name   = os.path.basename(child.fileName)
        if not (self.frame.dead or child.frame.dead):
            try:
                child.frame.setTitle(self.name,colour=wx.WHITE)
            except Exception, e:
                print e

    def loadWorkspace(self):
        try:
            if self.getValue('CloseChildrenOnNewWorkspace'):
                #Close all open children
                for child in self.app.children: #this doesn't loop through all the children (leaves one left)
                    child.frame.onFrameClose()
        except:
            pass

        #load the Recent Files
        self.recent.files=[]
        default=False
        if self.getValue('globalRecent'): default=True
        try:
            files=eval(self.getWorkspaceValue("Recent",default))
            self.recent.add([file for file in files if file and os.path.exists(file)])
        except:
            pass

        #load the Folders
        default=False
        if self.getValue('globalFolders'): default=True
        try:
            folders=eval(self.getWorkspaceValue("Folders",default))
            self.browser.depth.SetValue(folders[0])
            self.browser.add([file for file in folders[1:] if file])
        except:
            pass

        #load the Notes
        default=False
        if self.getValue('globalNotes'): default=True
        try:
            notes=self.getWorkspaceValue("Notes",default)
            self.notes.SetValue(notes)
        except:
            pass

        #load the openfiles (the old 'remember.txt')
        fileList=[]
        default=False
        if self.getValue('globalFileList'): default=True
        try:
            files=eval(self.getWorkspaceValue("OpenFiles",default))
            fileList=[file for file in files if file]
        except:
            pass
        if fileList:
            self.rememberSet(1)
            self.openList(fileList,
                message     = 0,
                select      = None,
                maximize    = self.getValue("MaxChildren"),
                verbose     = True)
        self.setWorkspaceStatusBarText(self.workspace['file'])

    def setWorkspaceStatusBarText(self,file):
        self.SetActiveStatusText(os.path.basename(file)[:-4],STATUS_TEXT_WORKSPACE_POS)
        
    def emptyWorkspace(self):
        return { 'config' : ConfigParser.ConfigParser() ,
                           'file' : "" ,
                      'openfiles' : [],
                  'defaultconfig' : ConfigParser.ConfigParser()
                       }

    def saveWorkspace(self,filelocation=None):
        if filelocation is '': return
        if self.app.children:
            childActive = self.app.childActive
            fileList=[]
            if self.app.children:
                self.workspace['openfiles']=[]
                for child in self.app.children:
                    if child.fileName != Child.NEWFILE:
                        pos     = child.source.GetCurrentPos()
                        lineno  = child.source.LineFromPosition(pos)
                        col     = child.source.GetColumn(pos)
                        fileList.append((child.fileName,lineno,col))
                        self.workspace['openfiles'].append(child.fileName)
                        self.applyWorkspaceTab(child)
            self.setWorkspaceValue("notes",self.notes.GetValue())
            self.setWorkspaceValue("openfiles",str(fileList))
            self.setWorkspaceValue("recent",str(self.recent.files[:self.getValue('RecentFileAmount')]))
            self.setWorkspaceValue("folders",str([self.browser.depth.GetValue()]+self.browser.getFolders()[1:]))
            if not filelocation: filelocation=self.workspace['file']
            try:
                file=open(filelocation,'w')
                self.workspace['config'].write(file)
                file.close()
                self.workspace['file']=filelocation
                self.setWorkspaceStatusBarText(filelocation)
            except Exception, message:
                print 'Spe warning: could not save workspace options in',filelocation
                print message
            self.applyWorkspaceTab(childActive)

    def getWorkspaceValue(self,type,default=False):
        """        returns the value of the workspace config file key        """
        if default:
            return self.workspace['defaultconfig'].get(type.lower(),"1")
        else:
            return self.workspace['config'].get(type.lower(),"1")

    def setWorkspaceValue(self,type,value):
        """        sets the workspace config file key to a specific value        """
        if not self.workspace['config'].has_section(type): self.workspace['config'].add_section(type)
        self.workspace['config'].set(type.lower(),"1",value)


    ####Menu
    #---File
    def new(self,name=UNNAMED,source='',maximize=None):
        """Create a new empty script window.?"""
        app = self.app
        child = app.ChildFrame(self.frame,
            page        = os.path.basename(name),
            extra       = name,
            fileName    = name,
            source      = source,
            size        = app.size,
            maximize    = maximize)
        self.frame.menuBar.enable(1)
        return child.panel

    def open(self, event=None):
        """Open file(s) dialog."""
        try:
            defaultDir  = info.dirname(self.app.childActive.fileName)
        except:
            defaultDir  = ''
        dlg = wx.FileDialog(self, "Choose a file - www.stani.be",
            defaultDir=defaultDir, defaultFile="",
            wildcard=info.WILDCARD,
            style=wx.OPEN|wx.MULTIPLE)
        if dlg.ShowModal() == wx.ID_OK:
            fileList = dlg.GetPaths()
            if fileList and self.app.children:
                child           = self.app.childActive
                if child and child.fileName==Child.NEWFILE and not child.changed:
                    child.frame.onFrameClose()
            self.openList(fileList)
        dlg.Destroy()

    def open_workspace(self):
        """Open file dialog."""
        try:
            defaultDir=info.dirname(self.workspace['file'])
        except:
            defaultDir=''
        dlg = wx.FileDialog(self, "Choose a file - www.stani.be",
            defaultDir=defaultDir, defaultFile="",
            wildcard=info.WORKSPACE_WILDCARD,
            style=wx.OPEN)
        if dlg.ShowModal() == wx.ID_OK:
            file = dlg.GetPath()
            try:
                self.set("currentworkspace",file)
                self.__openWorkspace__()
                self.loadWorkspace()
            except Exception,e:
                self.message("Could not open workspace:%s\n%s"%(file,e))
        dlg.Destroy()

    def save_workspace(self):
        """Save file dialog."""
        try:
            self.saveWorkspace()
        except Exception,e:
            self.message("Could not save workspace:%s\n%s"%(file,e))

    def save_workspace_as(self):
        """Save file dialog."""
        try:
            defaultFile = self.workspace['file']
        except:
            defaultFile = ''
        if info.WIN:
           defaultFile  = defaultFile.replace("/","\\")
        dlg             = wx.FileDialog(self, "Save Workspace As - www.stani.be",
            defaultFile = defaultFile,
            defaultDir  = info.dirname(defaultFile),
            wildcard    = info.WORKSPACE_WILDCARD,
            style       = wx.SAVE|wx.OVERWRITE_PROMPT|wx.CHANGE_DIR)
        if dlg.ShowModal() == wx.ID_OK:
            file = dlg.GetPath()
            try:
                self.set("currentworkspace",file)
                self.saveWorkspace(file)
            except Exception,e:
                self.message("Could not save workspace:%s\n%s"%(file,e))
        dlg.Destroy()

    def save(self):
        if self.getValue('SaveWorkspaceOnFileSave'):
            self.saveWorkspace()

    #---Edit
    def browse_source(self, event=None):
        """Locate source file of word and open it."""
        if self.app.children:
            fileName=self.app.childActive.source.getWordFileName(whole=1)
            if fileName and fileName[0]!='"':
                self.openList(fileName)
            else:
                if not fileName: fileName=''
                self.SetActiveStatusText('Sorry, can not locate file %s'%fileName)

    def find_replace(self, event=None):
        """Find and Replace dialog and action."""
        if self.app.children:
            #find string
            findStr = self.app.childActive.source.GetSelectedText()
            if findStr and self.findDialog:
                self.findDialog.Destroy()
                self.findDialog = None
            #dialog already open, if yes give focus
            if self.findDialog:
                self.findDialog.Show(1)
                self.findDialog.Raise()
                return
            if not findStr:
                    findStr = self.findStr
            self.numberMessages=0
            #find data
            data    = wx.FindReplaceData(self.findFlags)
            data.SetFindString(findStr)
            data.SetReplaceString(self.replaceStr)
            #dialog
            self.findDialog = wx.FindReplaceDialog(self, data, "Find & Replace",
                    wx.FR_REPLACEDIALOG|wx.FR_NOUPDOWN)
            x, y    = self.frame.GetPosition()
            self.findDialog.SetPosition((x+5,y+200))
            self.findDialog.Show(1)
            self.findDialog.Raise()
            self.findDialog.data = data  # save a reference to it...

    def execute(self):
        """Execute"""
        if self.app.children:
            child   = self.app.childActive
            source  = child.source
            code    = source.GetSelectedText()
            if not code.strip():
                if self.getValue('ExecuteWarning') and not self.messageConfirm('As there is no code selected,\nSPE will run the whole script.\n\nAre sure you want to continue?'):
                    return
                if self.getValue('SaveBeforeRun') and not child.confirmSave():
                    return
                child.setStatus('As there is no code selected, SPE will run the whole script.')
                code = source.GetText()
            self.shell.Execute(code)
            #thread.start_new_thread(self.shell.Execute,(code,))

    def preferences(self):
        """Show preferences dialog box."""
        from dialogs import preferencesDialog
        prefs=preferencesDialog.Create(self,-1,'')
        prefs.ShowModal()

    #---Blender

    def add_spe_to_blender(self):
        """Adds SPE and Winpdb shortcuts to Blender menu"""
        from distutils.file_util import copy_file 
        import Blender
        #important local variables
        #
        srcdir = info.PATH #_spe directory
        dstdir = Blender.Get('uscriptsdir') #preferred Blender script directory (can be '')
        altdir = Blender.Get('scriptsdir') #the other Blender script directory 
        #'uscriptsdir' can be empty - in such case use 'scriptsdir': 
        if not dstdir: dstdir, altdir = altdir, None
        #
        #2. Main operation: try to update the *.py file at dstdir,
        #   optionally remove eventual old location from altdir:
        #
        cpyresult = rmresult = mresult = "" #helpers for message fromatting
        for fname in (BLENDER_SHORTCUT_SPE,BLENDER_SHORTCUT_WINPDB):
            src = os.path.join(srcdir,fname)
            result = copy_file(src, os.path.join(dstdir,fname),update=1)
            if result[1]: #copied! 
                cpyresult += ", " + fname #if suceeded: add fname to the message
                #
                #if we have copied fname with success - there should not be 
                # two fname scripts (one for every Blender scripts directory): 
                # try to remove the unwanted one from the altdir (Blender 'scriptsdir')
                #
                if altdir and os.access(altdir,os.W_OK): 
                    try: #let's try to remove it from unused dir:
                        os.remove(os.path.join(altdir, fname)) 
                        rmresult += ", " + fname #OK, succeed: add fname to the message
                    except:
                        pass #just continue - it is not a big problem
        #
        #3. Update Blender:
        #
        Blender.UpdateMenus()
        #
        #4. Final message to the user:
        #
        #([2:] is used in strings to discard leading ", "):
        msg = "Blender menu updated.\n\n"
        if cpyresult: msg+= "Copied %s to %s.\n\n" % (cpyresult[2:], dstdir)
        if rmresult: msg+= "Removed %s from %s. " % (rmresult[2:], altdir)
        self.message(msg)
        #self.SetStatusText(msg,1)
        
    #---View
    def whitespace(self,event):
        """Toggle visibility white space."""
        for child in self.app.children:
            child.source.SetViewWhiteSpace(event.IsChecked())
        self.set('ViewWhiteSpace',event.IsChecked())
    
    def linenumbers(self,event):
        """Toggle visibility line numbers."""
        for child in self.app.children:
            if event.IsChecked():        
                child.source.SetMarginWidth(1, 50)
            else:
                child.source.SetMarginWidth(1, 0)
            
        self.set('ViewLineNumbers',event.IsChecked())

    def indentation_guides(self,event):
        """Toggle visibility indentation guides."""
        for child in self.app.children:
            child.source.SetIndentationGuides(event.IsChecked())
        self.set('IndentationGuides',event.IsChecked())

    def right_edge_indicator(self,event):
        """Toggle visibility right edge indicator."""
        for child in self.app.children:
            child.source.SetViewEdge(event.IsChecked())
        self.set('ViewEdge',event.IsChecked())

    def end_of_line_marker(self,event):
        """Toggle visibility end of line marker."""
        for child in self.app.children:
            child.source.SetViewEOL(event.IsChecked())
        self.set('ViewEol',event.IsChecked())

    def as_notebook(self,event):
        if hasattr(self.frame,'Tile'):
            if self.app.children:
                self.app.childActive.frame.Maximize()
        else:
            tabs    = getattr(self.frame,'tabs',None)
            if tabs:
                    tabs.Tile(False)

    def as_columns(self,event):
        if hasattr(self.frame,'Tile'):
            self.frame.Tile(wx.VERTICAL)
        else:
            tabs    = getattr(self.frame,'tabs',None)
            if tabs:
                    tabs.Tile(True, wx.VERTICAL)

    def as_rows(self,event):
        if hasattr(self.frame,'Tile'):
            self.frame.Maximize(wx.HORIZONTAL)
        else:
            tabs    = getattr(self.frame,'tabs',None)
            if tabs:
                    tabs.Tile(True, wx.HORIZONTAL)

    def toggle_shell(self):
        """Show/hide shell"""
        frame               = self.frame
        if hasattr(frame,'sash'):
            height          = frame.sash.GetSize()[1]
            if height<10:
                hidden      = 1
            else:
                hidden      = 0
        elif hasattr(frame,'panelFrame'):
            hidden          = not frame.panelFrame.IsShown()
        elif hasattr(frame,'split'):
            hidden          = not frame.split.IsSplit()
        else: return False
        self.showShell(hidden,save=True)
        return hidden

    def showShell(self,show,save=False):
        if save: self.set('ShowShell',show)
        frame = self.frame
        if hasattr(frame,'sash'):
            if show:
                frame.sash.SetDefaultSize(wx.Size(1000,200))
            else:
                frame.sash.SetDefaultSize(wx.Size(1000,1))
            wx.LayoutAlgorithm().LayoutMDIFrame(frame)
        elif hasattr(frame,'panelFrame'):
            frame.panelFrame.Show(show)
            frame.panelFrame.Activate()
        elif hasattr(frame,'split'):
            if show:
                frame.split.SplitHorizontally(frame.tabs, self, -200)
            else:
                frame.split.Unsplit()

##    def showToolbar(self,show,save=False):
##        if save: self.set('ShowToolbar',show)
##        frame = self.frame
##        if hasattr(frame,'toolBar'):
##            if show:
##                frame.sash.SetDefaultSize(wx.Size(1000,200))
##            else:
##                frame.sash.SetDefaultSize(wx.Size(1000,1))
##            wx.LayoutAlgorithm().LayoutMDIFrame(frame)

    #---Tools
    def browse_folder(self):
        """Browse folder"""
        if self.app.children:
            child           = self.app.childActive
            if hasattr(child,'fileName'):
                path        = info.dirname(child.fileName)
                if not os.path.exists(path):
                    path    = os.getcwd()
            else:
                path    = os.getcwd()
            if os.path.exists(THUNAR):
                os.system('%s "%s" &'%(THUNAR,path))
            elif os.path.exists(NAUTILUS):
                os.system('%s "%s" &'%(NAUTILUS,path))
            elif os.path.exists(DOLPHIN):
                os.system('%s "%s" &'%(DOLPHIN,path))
            elif os.path.exists(KONQUEROR):
                os.system('%s "%s" &'%(KONQUEROR,path))
            elif os.path.exists(PCMANFM):
                os.system('%s "%s" &'%(PCMANFM,path))
            else:
                if path[0] == '/': path = 'file://'+path
                webbrowser.open(path)

    def run(self):
        if self.output.IsBusy():
            if self.messageConfirm('SPE will try to kill the running script.\n\nWarning:this might kill SPE as well.\nIt is highly recommended to save all scripts first.\n\nAre you sure you want to continue?'):
                self.output.Kill()
            else:
                self.output._check_run(False)
            return
        if self.app.children:
            child = self.app.childActive
            if not child.confirmSave():
                self.output._check_run(False)
                return
            if child.isNew(): 
                self.output._check_run(False)
                return
            from _spe.dialogs.runDialog import RunDialog
            runDialog           = RunDialog(child.fileName,
                                    self.argumentsPrevious,
                                    self.beepPrevious,
                                    parent=self.frame,
                                    id=-1)
            answer              = runDialog.ShowModal()
            arguments           = runDialog.arguments.GetValue()
            beep                = runDialog.beep.GetValue()
            runDialog.Destroy()
            if answer == wx.ID_OK:
                if not (arguments in self.argumentsPrevious):
                    self.argumentsPrevious.insert(0,arguments)
                self.beepPrevious   = beep
                self.run_with_arguments(arguments,beep=beep,confirm=False)
            else:
                self.output._check_run(False)

    def run_with_arguments(self,arguments='',beep=True,confirm=True):
        if self.output.IsBusy():
            self.output.Kill()
            return
        if self.app.children:
            child = self.app.childActive
            if confirm and not child.confirmSave():
                return
            if child.isNew(): return
            # todo: input stuff from preferences dialog box!
            path, fileName  = os.path.split(child.fileName)
            params          = { 'file':         info.path(child.fileName),
                                'path':         path,
                                'arguments':    arguments,
                                'python':       info.PYTHON_EXEC}
            #label = params['label'] = '"%(file)s" %(arguments)s'%params
            label = params['label'] = '%(file)s %(arguments)s'%params
            os.chdir(path)
            self.output.Execute("""%(python)s -u %(label)s"""%params,label=label,beep=beep)

    def run_debug(self):
        """Run file"""
        if not self.runner:
            try:
                from _spe.plugins.spe_winpdb import Runner
            except ImportError:
                self.messageError('You need to install\n'
                    'winpdb for this feature.')
                return
            self.runner = Runner(self.app)
        self.runner.switch()

    def import_(self):
        """Import"""
        if self.app.children:
            child                   = self.app.childActive
            if not self.getValue('SaveBeforeRun') or child.confirmSave():
                if child.isNew(): return
                self.busyShow()
                name                = child.fileName
                self.shell.write('Importing "%s" ...'%name)
                self.shell.waiting  = 1
                self.shell.Execute('execfile(r"%s")' % name)
#                sm.scriptutils.importMod(name,mainDict=self.shell.locals)
                self.shell.waiting  = 0
                self.shell.prompt()
                if self.redraw: self.redraw()
                self.activateShell()
                self.busyHide()

    def debug(self):
        child                   = self.app.childActive
        if not child.confirmSave():
            return
        if child.isNew(): return
        if not self.runner:
            try:
                from _spe.plugins.spe_winpdb import Runner
            except ImportError:
                self.messageError('You need to install\n'
                    'winpdb for this feature.')
                return
            self.runner = Runner(self.app)
        self.runner.debug()

    def browse_object_with_pyfilling(self):
        """Browse object with pyfilling"""
        from wx.py.filling import FillingFrame
        name=self.messageEntry('Enter object:')
        try:
            object=self.shell.interp.locals[name]
        except:
            self.SetActiveStatusText('%s is not defined'%name,1)
            return
        filling=FillingFrame(parent=self, id=-1, title='PyFilling',
                 pos=wx.DefaultPosition, size=wx.Size(600,300),
                 style=wx.DEFAULT_FRAME_STYLE, rootObject=object,
                 rootLabel=str(object), rootIsNamespace=0, static=0)
        filling.Show(1)
        wx.FutureCall(1000,filling.Raise)

    def test_regular_expression_with_kiki(self):
        """Test regular expression with Kiki..."""
        try:
            self.kiki.Raise()
        except:
            try:
                from kiki import kiki
            except ImportError:
                from plugins.kiki import kiki
            INFO['kikiPath']=info.dirname(kiki.__file__)
            self.kiki=kiki.speCreate(self,info=INFO)
        self.SetStatusText('Kiki is succesfully started.',1)

    def design_a_gui_with_wxglade(self):
        try:
            import subprocess
            glade   = subprocess.Popen(['which','wxglade'],
                stdout=subprocess.PIPE).stdout.read().strip()
            os.spawnl(os.P_NOWAIT,glade,glade)
            return
        except:
            pass
        try:
            from wxglade import __file__ as fileName
        except ImportError:
            from plugins.wxGlade import __file__ as fileName
        except:
            self.messageError('You need to install\n'
                'wxglade (python-wxglade)\n'
                'for this feature.')
            return
        path    = info.dirname(fileName)
        glade   = '%s'%os.path.join(path,'wxglade.py')
        if info.WIN and ' ' in glade:
            glade = '"%s"'%glade
        os.spawnl(os.P_NOWAIT,info.PYTHON_EXEC,info.PYTHON_EXEC,glade)
        self.SetStatusText('wxGlade is succesfully started.',1)

    def design_a_gui_with_xrc(self):
        try:
            from wx.tools.XRCed import xrced
        except ImportError:
            from plugins.XRCed import xrced
        except:
            self.messageError('You need to install\n'
                'the wxpython tools (python-wxtools)\n'
                'for this feature.')
            return
        import plugins
        path   = os.path.dirname(plugins.__file__)
        xrced   = '%s'%os.path.join(path,'spe_xrced.py')
        if info.WIN and ' ' in xrced:
            xrced   = '"%s"'%xrced
        os.spawnl(os.P_NOWAIT,info.PYTHON_EXEC,info.PYTHON_EXEC,xrced)
        self.SetStatusText('XRC editor is succesfully started.',1)

    #---Links
    def contact_author(self):
        WebBrowser=self.get('WebBrowser')
        if WebBrowser==DEFAULT:
            webbrowser.open(MAIL)
        else:
            os.system("%s '%s'"%(WebBrowser,MAIL))

    #---Help
    def keyboard_shortcuts(self):
        from dialogs import helpShortcutsDialog
        helpShortcutsDialog.create(self,path=self.path).Show(1)

    def python_help(self,what):
        pythonDocs  = self.get('PythonDocs')
        wwwHelp     = HELP_WWW%(what,what)
        if pythonDocs==DEFAULT:
            if sys.platform=='win32':
                helpActivePython=os.path.join(PREFIX,"Doc","ActivePython.chm")
                helpEntoughtPython=os.path.join(PREFIX,"Enthought","Doc","enthought_python.chm")
                if os.path.exists(helpActivePython):
                    self.messageHtml(helpActivePython)
                elif os.path.exists(helpEntoughtPython):
                    self.messageHtml(helpEntoughtPython)
                else:
                    library=os.path.join(PREFIX,"Doc",what,"index.html")
                    if os.path.exists(library):
                        self.messageHtml(library)
                    else:
                        self.messageHtml(wwwHelp)
                        self.SetStatusText(HELP_SORRY%library,1)
            else:
                linux1=''.join([PREFIX, "/share/doc/python", str(sys.version_info[0]),
                                ".", str(sys.version_info[1]), "/html/%s/index.html"%what])
                linux2=''.join([PREFIX, "/share/doc/python-docs-",
                                str(sys.version_info[0]), ".", str(sys.version_info[1]), ".",
                                str(sys.version_info[2]), "/html/%s/index.html"%what])
                if os.path.exists(linux1):self.messageHtml(linux1)
                elif os.path.exists(linux2):self.messageHtml(linux2)
                else:
                    self.messageHtml(wwwHelp)
                    self.SetStatusText(HELP_SORRY%linux1,1)
        else:
            # TODO: pythondocs laten loopen over alle mogelijkheden, dit iets eleganter.
            helpActivePython=os.path.join(pythonDocs,"Doc","ActivePython.chm")
            if sys.platform=='win32' and os.path.exists(helpActivePython):
                library = helpActivePython
            else:
                library = os.path.join(pythonDocs,what,"index.html")
            if os.path.exists(library):
                self.messageHtml(library)
            else:
                self.messageHtml(wwwHelp)
                self.SetStatusText(HELP_SORRY%library,1)

    def python_documentation_server(self):
        from pydoc import __file__ as fileName
        os.spawnl(os.P_NOWAIT,info.PYTHON_EXEC,info.PYTHON_EXEC,fileName,'-g')
        self.messageHtml('http://localhost:7464/')

    def wxwindows_documentation(self):
        WxPythonDocs=self.get('WxPythonDocs')
        if WxPythonDocs==DEFAULT:
            WxPythonDocs    = os.path.join(info.dirname(wx.__file__),'docs')
        WxPythonDocs2       = os.path.join(info.dirname(WxPythonDocs),'docs')
        WxPythonDocs3        = 'C:\\Program Files\\wxPython2.8 Docs and Demos\\docs'
        path = None
        for dir in [WxPythonDocs,WxPythonDocs2,WxPythonDocs3]:
            for docs in ['wx.chm','wxPythonDocs.html','wxPythonManual.html']:
                if not path:
                    p = os.path.join(dir,docs)
                    if os.path.exists(p):
                        path = p
        if path:
            self.messageHtml(path)
        elif os.path.exists(os.path.join(WxPythonDocs2,"wxDocsViewer.app")):
            os.spawnl(os.P_NOWAIT,info.PYTHON_EXEC,info.PYTHON_EXEC,
                os.path.join(WxPythonDocs,"wxDocsViewer.app/Contents/Resources/viewdocs.py"))
        else:
            self.SetActiveStatusText('wxPython documentation could not be found. Check the path in the preferences dialog.')

    def about(self):
        from dialogs import helpDialog
        helpDialog.create(self,self.path,'about.htm', replacements=INFO)

    #---Backstage
    def openList(self,fileList,lineno=None,col=1,message=1,select='line',\
                    maximize=None, verbose=False):
        """Open a list of files."""
        if type(fileList)!=types.ListType:
            fileList=[fileList]
        child = None
        for fileName in fileList:
            if type(fileName)==types.TupleType:
                fileName, lineno, col = fileName
            if not os.path.exists(fileName): continue
            if self.app.DEBUG:
                print 'Opening %s'%fileName
            child=self.getChildByFileName(fileName)
            if child:
                #opened already
                child.frame.Raise()
                if lineno:child.scrollTo(lineno,col,select=select)
                else:self.SetStatusText("'%s' is already open"%fileName,1)
            else:
                #not opened yet
                try:
                    source=open(fileName).read()
                except:
                    source='' # todo: make this an option
                child=self.new(name=fileName,source=source,maximize=maximize)
                #if this file is in the workspace, then change the name
                self.recent.add([fileName])
                if lineno:child.scrollTo(lineno,col,select=select)
##                if child.fileName in self.workspace['openfiles']:
##                    #this is a workspace file and should
##                    self.applyWorkspaceTab(child)
            lineno=None
            col=1
        if len(self.app.children)>0:self.frame.menuBar.enable(1)
        return child

    def runFile(self,fileName,source=None,separate=0,profiling=0):
        """
        separate: run in seperate namespace to avoid conflicts with the spe namespace
        """
        self.shell.write("Running '%s' ..."%fileName)
        #self.busyShow()
        self.shell.waiting  = 1
        self.shell.interp.push('import sm.scriptutils')
        if separate:
            nameSpace       = 'namespace["%s"]'%fileName
            self.shell.interp.push('%s={"__fileName__":"%s"}'%(nameSpace,fileName))
        else:
            nameSpace       = 'locals()'
        try:
            runCommand      = 'sm.scriptutils.run(fileName=r"%s",source=%s,mainDict=%s,profiling=%s)'\
                %(fileName,source,nameSpace,profiling)
            self.shell.interp.push(runCommand)
        except Exception,message:
            print '\n(Spe internal warning: %s!)'%message
        self.shell.prompt()
        if self.redraw:self.redraw()
        self.activateShell()
        #self.busyHide()

    def activateShell(self):
        if self.app.mdi:
            self.showShell(True,save=True)
        else:
            self.frame.Raise()
        self.SetSelection(0)

    ####Events
    #---smdi events
    def onActivate(self,event):
        """Check and update, if files are changed when parent frame is activated."""
        if self.app.DEBUG:
            print 'Event:  Parent: %s.onActivate'%self.__class__
        if not self.timer.IsRunning():
            #self.Bind(wx.EVT_IDLE,self.onIdle)
            self.Bind(wx.EVT_TIMER,self.onTimer)
            self.timer.Start(self.getValue('Redraw'))
        reloaded=[]
        try:
            for child in self.app.children:
                if child.checkTime():
                    reloaded.append(os.path.basename(child.fileName))
            if reloaded: self.SetStatusText('Reloaded %s'%','.join(reloaded),1)
            if event: event.Skip()
        except Exception,m:
            if self.app.DEBUG:
                print 'Warning: Parent: %s.onActivate failed\n%s\n%s\n'%(self.__class__,Exception,m)
                
    def onDeactivate(self,event=None):
        if self.app.children:
            childActive = self.app.childActive
            if (not self.frame.dead) and childActive and hasattr(childActive,'source'):
                childActive.source.AutoCompCancel()
                childActive.source.CallTipCancel()
        self.timer.Stop()
        self.Unbind(wx.EVT_TIMER)
##        wait    = self.getValue('Redraw')/1000.0
##        while self.onTimer(None,end=True):
##            time.sleep(wait)
        
    def onClose(self,event=None):
        """Called when the parent frame is closed."""
        if self.app.DEBUG:
            print 'Event:  Parent: %s.onClose'%self.__class__
        for child in self.app.children:
            if not child.confirmSave():
                child.Raise()
                child.SetStatusText('Please save this file before quitting SPE.')
                return False
        eventManager.DeregisterWindow(self)
        self.timer.Stop()
        self.frame.dead = 1
        #if (not self.app.DEBUG) and self.getValue('RedirectShell'):
        self.redirect(0)
        if self.remember and self.app.children:
            active=self.app.childActive
            if self.app.mdi and active:
                self.set('MaxChildren',active.frame.IsMaximized())
        #save the window size
        mainWindow=self.app.GetTopWindow()
        selfSize=mainWindow.GetRect()
        if not mainWindow.IsMaximized():
            self.set("sizex",selfSize.GetSize().x)
            if info.DARWIN and wx.VERSION_STRING>'2.8':
                self.set("sizey",selfSize.GetSize().y-47) #dirty hack
            else:
                self.set("sizey",selfSize.GetSize().y)
            self.set("posx",selfSize.GetPosition().x)
            self.set("posy",selfSize.GetPosition().y)
            self.set("maximize","False")
        else:
            self.set("maximize","True")
        try:
            if self.getValue('SaveOnExit'):
                self.saveWorkspace()
            if self.remember:
                self.saveWorkspace(INFO['defaultWorkspace'])
        except Exception, message:
            self.messageEmail("""\
Spe Warning: can't save user settings (%s).
Please report these details and operating system to %s."""%(message,INFO['author_email']))
        if not self.getValue('RememberLastWorkspace'): 
            self.set('currentworkspace',"")
        return 1

    def onClosePanelFrame(self,event=None):
        self.showShell(show=False,save=True)
        self.frame.menuBar.check_view()

    def onTimer(self,event=None,end=False):
        if self.app.IsActive():
            #child
            if self.app.children:
                child   = self.app.childActive
                if not self.lock.locked():
                    child.idle(None,end)
            #redraw
            if self.redraw:
                self.redraw()
        else:
            self.onDeactivate()

    def onMove(self,event=None):
        """Called when the parent frame is moved."""
        if self.app.DEBUG:
            print 'Event:  Parent: %s.onMove'%self.__class__
        if self.redraw:self.redraw()

    def onSize(self,event=None):
        """Called when the parent frame is resized."""
        if self.app.DEBUG:
            print 'Event:  Parent: %s.onSize'%self.__class__
        if self.redraw:self.redraw()

    #---extra
    def onArgs(self,*args,**keyw):
        self.openList(*args,**keyw)
        self.shell.prompt()

    def excepthook(self, type, value, traceback) :
        webbrowser.open('''mailto:%s?subject=SPE %s error report&body="%s %s\n%s"'''%(INFO['author_email'],type,type,value,traceback))

    def onFind(self,event,message=1):
        if self.app.children:
            source=self.app.childActive.source
            try:
                self.findStr=event.GetFindString()
                self.findFlags=event.GetFlags()
                flags=0
                if wx.FR_WHOLEWORD & self.findFlags:
                    flags|=wx.stc.STC_FIND_WHOLEWORD
                if wx.FR_MATCHCASE & self.findFlags:
                    flags|=wx.stc.STC_FIND_MATCHCASE
                self.stcFindFlags=flags
            except:
                pass
            current=source.GetCurrentPos()
            position=source.FindText(current,len(source.GetText()),self.findStr,
                    self.stcFindFlags)
            if position==-1:#wrap around
                self.wrapped=1
                position=source.FindText(0,current+len(self.findStr),self.findStr,self.stcFindFlags)
                self.SetActiveStatusText("Wrapped around to find '%s'"%self.findStr,1)
            if position==-1 and message and self.numberMessages<1:
                #not found
                self.numberMessages=1
                self.message("'%s' not found!"%self.findStr)
                self.numberMessages=0
            else:
                #found
                line    = source.LineFromPosition(position)
                source.EnsureVisible(line)
                source.GotoPos(position)
                source.SetSelection(position,position+len(self.findStr))
            return position

    def onFindClose(self,event):
        event.GetDialog().Destroy()
        self.numberMessages=0

    def onReplace(self,event,message=1):
        if self.app.children:
            ## Next line avoid infinite loop
            findStr=event.GetFindString()
            self.replaceStr=event.GetReplaceString()
            if findStr==self.replaceStr:
                return -1
            source=self.app.childActive.source
            selection=source.GetSelectedText()
            if not(event.GetFlags() & wx.FR_WHOLEWORD):
                findStr=findStr.lower()
                selection=selection.lower()
                if findStr==self.replaceStr.lower():
                    return -1
            if selection==findStr:
                position=source.GetSelectionStart()
                source.ReplaceSelection(self.replaceStr)
                source.SetSelection(position,position+len(self.replaceStr))
            position=self.onFind(event,message=message)
            return position

    def onReplaceAll(self,event):
        if self.app.children:
            source=self.app.childActive.source
            count=0
            self.wrapped=0
            position=start=source.GetCurrentPos()
            while position>-1 and ((not self.wrapped) or position<start):
                position=self.onReplace(event,message=0)
                if position != -1: count+=1
            if count:
                self.SetActiveStatusText("'%s' is %s times replaced with '%s'"%\
                    (event.GetFindString(),count,event.GetReplaceString()),1)
            elif not count and self.numberMessages<1:
                self.numberMessages=1
                self.message("'%s' not found!"%event.GetFindString())
                self.numberMessages=0

    ####Tabs
    def onTab(self,event):
        tab     = event.GetSelection()
        tabName = self.tabs[tab]
        method  = 'update'+tabName
        if hasattr(self,method):
            getattr(self,method)()
        if info.DARWIN:
            self.SetPageText(event.GetOldSelection(),'')
            self.SetPageText(tab,tabName)
        event.Skip()


    ####Sidebar
    def onTodoJump(self,event):
        file,line=self.todoList[event.GetData()]
        self.openList(file,line-1)

    def todoMax(self,child):
        try:
            child.updateTodo()
            return child.todoMax
        except:
            return 1

    def updateTodo(self):
        self.todo.DeleteAllItems()
        todoIndex=0
        self.todo.list=[]
        todoMax=[self.todoMax(child) for child in self.app.children]+[1]
        todoMax=max(todoMax)
        for child in self.app.children:
            todoIndexStart=todoIndex
            for task in child.todoList:
                line,urgency,entry=task
                item=self.todo.InsertStringItem(todoIndex, os.path.basename(child.fileName))
                self.todo.SetStringItem(todoIndex, 1, str(line+1))
                self.todo.SetStringItem(todoIndex, 2, str(urgency))
                self.todo.SetStringItem(todoIndex, 3, entry)
                self.todo.list.append((child.fileName,line+1))
                self.todo.SetItemData(item,todoIndex)
                todoIndex+=1
            if child.todoMax==todoMax:
                for i in child.todoHighlights:
                    item=self.todo.GetItem(todoIndexStart+i.wx)
                    item.SetBackgroundColour(wx.Colour(255,255,0))
                    self.todo.SetItem(item)

    #---index
    def onIndexJump(self,event):
        stripped,entry,line,colour,icon,file=self.indexList[event.GetData()]
        self.openList(file,line-1)

    def updateIndex(self):
        self.index.DeleteAllItems()
        indexList= self.index.list = []
        for child in self.app.children:
            child.updateIndex()
            indexList.extend(child.indexData)
        indexList.sort()
        firstLetter=''
        x=0
        index=0
        for element in indexList:
            stripped,entry,line,colour,icon,fileName=element
            if stripped[0]!=firstLetter:
                firstLetter=stripped[0]
                self.index.InsertStringItem(x, firstLetter)
                x+=1
            item=self.index.InsertImageStringItem(x, '%s@%s'%(entry.split('(')[0],os.path.basename(fileName)),icon)
            self.index.SetItemData(item,index)
            item=self.index.GetItem(item)
            item.SetTextColour(colour)
            self.index.SetItem(item)
            x+=1
            index+=1
        if self.index.GetItemCount():
            try:
                self.index.SetColumnWidth(-1,100)
            except:
                pass
    ####Methods
    def busyShow(self):
        wx.BeginBusyCursor()

    def busyHide(self):
        wx.EndBusyCursor()

    def checkBlender(self):
        """Check if source is running in Blender, otherwise give error message."""
        if self.Blender:return 1
        else:
            self.messageError(BLENDER_MESSAGE)
            return 0

    def _check_run_debug(self,bool):
        #assing method for check run tool button
        if self.app.children:
            child                   = self.app.childActive
            if child.frame.menuBar:
                child.frame.menuBar.check_run(bool)
            else:
                self.frame.menuBar.check_run(bool)
    
    def redirect(self,value):
        self.shell.redirectStdin(value)
        self.shell.redirectStdout(value)
        self.shell.redirectStderr(value)

    def rememberSet(self,value):
        self.remember   = value
        self.set('RememberLastWorkspace',int(value))
        self.frame.menuBar.check_remember(self.remember)

    def SetActiveStatusText(self,x,pos=1):
        try:
            if self.app.children:
                self.app.childActive.SetStatusText(x,pos)
            else:
                self.SetStatusText(x,pos)
        except:
            self.SetStatusText(x,pos)

    def userOpen(self,fileName):
        f=open(os.path.join(INFO['userPath'],fileName),'r')
        content=f.read()
        f.close()
        return content

    def userSave(self,fileName,content):
        f=open(os.path.join(INFO['userPath'],fileName),'w')
        f.write(content)
        f.close()

    #---get
    def getChildByFileName(self,fileName):
        if fileName=='<source>':
            if self.app.children:
                return self.app.childActive
            else:
                return None
        fileName = os.path.normcase(fileName)
        for child in self.app.children:
            if os.path.normcase(child.fileName)==fileName:
                return child
        return None

    def getFileNames(self):
        return [child.fileName for child in self.app.children]

    def getText(self):
        """Get raw text of current script window."""
        if self.app.children:
            return self.app.childActive.source.GetText()
        else:
            return ''

    #---messages
    def message(self,message,style=wx.OK | wx.ICON_INFORMATION):
        """Show a message with Ok button. (style offers other options)"""
        dlg = wx.MessageDialog(self, message, self.app.title, style)
        answer = dlg.ShowModal()
        dlg.Destroy()
        return answer

    def messageConfirm(self,message):
        """Show a confirm message (yes,no)"""
        answer=self.message(message,style=wx.YES_NO|wx.ICON_QUESTION)
        return self.messageIsOk(answer)

    def messageIsOk(self,answer):
        return answer==wx.ID_OK or answer==wx.ID_YES

    def messageCancel(self,message):
        """Show a yes,no or cancel message"""
        if self.app.DEBUG:
            print 'Dialog: Parent: %s.messageCancel'%self.__class__
        return self.message(message,style=wx.YES_NO|wx.ICON_QUESTION | wx.CANCEL)

    def messageError(self,message):
        """Display error message."""
        self.message(message,style=wx.OK | wx.ICON_ERROR)

    def messageEmail(self,message):
        try:
            webbrowser.open('mailto:%s?subject=SPE error (automatic report)&body=%s'%(INFO['author_email'],message))
        except:
            pass

    def messageEntry(self,message,default=''):
        """Show entry dialog box for user input."""
        dlg = wx.TextEntryDialog(self, message,self.app.title, default)
        if dlg.ShowModal() == wx.ID_OK:value=dlg.GetValue()
        else:value=None
        dlg.Destroy()
        return value

    def messageScrolled(self,message):
        """Show multiline scrollable message box."""
        from dialogs import speDialog
        if sys.platform!='win32':message='<font size=-2>%s</font>'%message
        speDialog.create(self, message, self.path)

    def messageHtml(self,fileName,doc=None):
        """Launch html with webbrowser."""
        if doc:
            fileName=os.path.join(doc,'doc',fileName)
        if fileName[0]=='/': fileName = 'file://'+fileName
        WebBrowser=self.get('WebBrowser')
        if WebBrowser==DEFAULT:
            if os.path.isfile(FIREFOX):
                os.system("%s %s &"%(FIREFOX,fileName))
            else:
                webbrowser.open(fileName, 1)
        else:
            os.system("%s %s &"%(WebBrowser,fileName))

    #---preferences
    def preferencesSave(self):
        self.preferencesUpdate()
        try:
            defaults=open(INFO['defaultsUser'],'w')
            self.config.write(defaults)
            defaults.close()
        except Exception, message:
            print 'Spe warning: could not save user options in',INFO['defaultsUser']
            print message

    def preferencesUpdate(self):
        if self.frame.dead: return
        self.redirect((not self.app.DEBUG) and self.getValue('RedirectShell'))
        self.showShell(self.getValue('ShowShell'))
        self.frame.menuBar.check_view()
        if hasattr(self.frame,'tabs'):
            self.frame.tabs.EnableToolTip(self.getValue('ToolTipsForFileTabs'))
        for child in self.app.children:
            child.source.update()
        #dialogs.stcStyleEditor.SetStyles(self.tabPanel['Shell'], self.config)
        #restart?!
        restart = ''
        if self.app.mdiName     != self.get('Mdi'):
            restart += '- switch between multiple and single document interface\n  (%s>%s)\n'%(self.app.mdiName,self.get('Mdi'))
        if self.app.shortcuts   != self.get('Shortcuts'):
            restart += '- to use other keyboad shortcuts\n  (%s>%s)\n'%(self.app.shortcuts, self.get('Shortcuts'))
        #encoding
        encoding                = self.get('Encoding').split(',')[0].split(' ')[0]
        if encoding!=self.defaultEncoding:
            self.defaultEncoding = encoding
            if encoding=='<default>':
                wx.SetDefaultPyEncoding(INFO['encoding'])
            else:
                wx.SetDefaultPyEncoding(encoding)
        #restart
        if restart!=self.restartMessage:
            self.restartMessage = restart
            if restart:
                self.message('Please restart SPE to apply following changes:\n%s'%restart)

    def set(self,name,value,save=1):
        self.config.set('Default',name,str(value).replace('%(','%{').replace(')s','}s'))
        if save: self.preferencesSave()

    def get(self,name):
        return self.config.get('Default',name).replace('%{','%(').replace('}s',')s')

    def getValue(self,name):
        return eval(self.config.get('Default',name))


