mirror of https://github.com/chriskiehl/Gooey.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
160 lines
5.0 KiB
160 lines
5.0 KiB
'''
|
|
Created on Jan 24, 2014
|
|
|
|
@author: Chris
|
|
|
|
##How things work these days (though, likely to change)
|
|
|
|
The decorator is used solely as a nice way to get the location
|
|
of the executing script. It no longer returns a decorated version
|
|
of the client code, but in fact completely hijacks the execution.
|
|
So, rather than returning a reference to the client's main, it now
|
|
returns itself, thus short-circuiting the execution of the client
|
|
program.
|
|
|
|
What it DOES do now is grab where the client module is stored and
|
|
read it in as a file so that it can hack away at it.
|
|
|
|
The first step, as before, is getting the ArgumentParser reference
|
|
so that the needed values can be extracted. This is done by reading
|
|
the source file up to the point where the `parse_args()` method is
|
|
called. This puts us smack in the middle of the client's `main` method.
|
|
|
|
This first half guarantees that all imports, modules, variable assignments,
|
|
etc.. are caught (unlike before).
|
|
|
|
Next step: getting the rest of the source code that's relevant
|
|
|
|
The file is again read up to the `parse_args` call, but this time everything
|
|
leading up to that point is dropped and we keep only the remainder of the file.
|
|
So, now the top and the bottom is located, but the bottom needs to be trimmed a
|
|
little more -- we want to drop everything remaining in the main method.
|
|
|
|
So, we `dropwhile` lines are currently indented (and thus still part of the `main`
|
|
method)
|
|
|
|
Finally, we arrive at the end, which gives us an exact copy of the original source
|
|
file, minus all of it's main logic. The two pieces are then sandwiched together,
|
|
saved to a file, and imported as a new module. Now all that has to be done is call
|
|
it (moddified) main function, and bam! It returns to fully populated parser object
|
|
to us. No more complicated ast stuff. Just a little bit of string parsing and we're
|
|
done.
|
|
|
|
'''
|
|
|
|
from argparse import ArgumentParser as RealArgParser, ArgumentParser
|
|
|
|
from functools import partial
|
|
import os
|
|
import sys
|
|
import types
|
|
|
|
import wx
|
|
|
|
from gooey.gui.lang import i18n
|
|
from gooey.python_bindings import argparse_to_json
|
|
import source_parser
|
|
|
|
|
|
ROOT_DIR = os.path.dirname(__import__(__name__.split('.')[0]).__file__)
|
|
TMP_DIR = os.path.join(ROOT_DIR, '_tmp')
|
|
|
|
def Gooey(f=None, advanced=True,
|
|
language='english', show_config=True,
|
|
program_name=None, program_description=None):
|
|
'''
|
|
Decorator for client code's main function.
|
|
Entry point for the GUI generator.
|
|
|
|
Scans the client code for argparse data.
|
|
If found, extracts it and build the proper
|
|
configuration gui windows (basic or advanced).
|
|
'''
|
|
|
|
params = locals()
|
|
|
|
def build(payload):
|
|
def inner():
|
|
main_module_path = get_caller_path()
|
|
_, filename = os.path.split(main_module_path)
|
|
cleaned_source = clean_source(main_module_path)
|
|
|
|
filepath = os.path.join(TMP_DIR, filename)
|
|
with open(filepath, 'w') as f:
|
|
f.write(cleaned_source)
|
|
|
|
run_cmd = 'python {}'.format(filepath)
|
|
|
|
# Must be called before anything else
|
|
app = wx.App(False)
|
|
|
|
i18n.load(language)
|
|
|
|
# load gui components after loading the language pack
|
|
from gooey.gui.client_app import ClientApp
|
|
from gooey.gui.client_app import EmptyClientApp
|
|
from gooey.gui.windows.base_window import BaseWindow
|
|
from gooey.gui.windows.advanced_config import AdvancedConfigPanel
|
|
from gooey.gui.windows.basic_config_panel import BasicConfigPanel
|
|
|
|
if show_config:
|
|
parser = get_parser(main_module_path)
|
|
|
|
meta = {
|
|
'target': run_cmd,
|
|
'program_name': program_name,
|
|
'program_description': program_description or parser.description,
|
|
'show_config': show_config,
|
|
'show_advanced': advanced,
|
|
'default_size': (610, 530),
|
|
'requireds_cols': 1,
|
|
'optionals_cols': 2
|
|
}
|
|
client_app = ClientApp(parser, payload)
|
|
|
|
build_spec = dict(meta.items() + argparse_to_json.convert(parser).items())
|
|
|
|
if advanced:
|
|
BodyPanel = partial(AdvancedConfigPanel, build_spec=build_spec)
|
|
else:
|
|
BodyPanel = BasicConfigPanel
|
|
# User doesn't want to display configuration screen
|
|
# Just jump straight to the run panel
|
|
else:
|
|
BodyPanel = BasicConfigPanel
|
|
client_app = EmptyClientApp(payload)
|
|
|
|
frame = BaseWindow(BodyPanel, build_spec, params)
|
|
|
|
if not show_config:
|
|
frame.ManualStart()
|
|
frame.Show(True)
|
|
app.MainLoop()
|
|
|
|
inner.__name__ = payload.__name__
|
|
return inner
|
|
|
|
if callable(f):
|
|
return build(f)
|
|
return build
|
|
|
|
|
|
def clean_source(module_path):
|
|
with open(module_path, 'r') as f:
|
|
return ''.join(
|
|
line for line in f.readlines()
|
|
if '@gooey' not in line.lower())
|
|
|
|
|
|
def get_parser(module_path):
|
|
return source_parser.extract_parser(module_path)
|
|
|
|
def get_caller_path():
|
|
tmp_sys = __import__('sys')
|
|
return tmp_sys.argv[0]
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
pass
|