Commit 4dd43c97 authored by sumpfralle's avatar sumpfralle

added automatic preferences load/save during startup/exit of the program

 * the preferences file is written to ~/.pycam/preferences.conf (*nix) resp. below APP_DATA (Windows)


git-svn-id: https://pycam.svn.sourceforge.net/svnroot/pycam/trunk@285 bbaffbd6-741e-11dd-a85d-61de82d9cad9
parent 94e0fa35
Version 0.2.3 - UNRELEASED Version 0.2.3 - UNRELEASED
* GUI change: tool and process settings can be combined into tasks * GUI change: tool and process settings can be combined into tasks
* added export of EMC tool definitions * added export of EMC tool definitions
* store configured settings to a file in the user's home directory
* fixed "overlap" calculation * fixed "overlap" calculation
* added a warning dialog (GUI) for missing dependencies during startup (especially useful for Windows) * added a warning dialog (GUI) for missing dependencies during startup (especially useful for Windows)
* improved GUI for model scaling * improved GUI for model scaling
......
...@@ -19,6 +19,7 @@ import OpenGL.GLUT as GLUT ...@@ -19,6 +19,7 @@ import OpenGL.GLUT as GLUT
#import gtk.gtkgl #import gtk.gtkgl
import gtk import gtk
import pango import pango
import ConfigParser
import time import time
import re import re
import os import os
...@@ -50,6 +51,29 @@ COLORS = { ...@@ -50,6 +51,29 @@ COLORS = {
"color_toolpath_return": (0.5, 1.0, 0.5), "color_toolpath_return": (0.5, 1.0, 0.5),
} }
PREFERENCES_DEFAULTS = {
"enable_ode": True,
"boundary_mode": -1,
"unit": "mm",
"show_model": True,
"show_axes": True,
"show_dimensions": True,
"show_bounding_box": True,
"show_toolpath": True,
"show_drill_progress": False,
"color_background": COLORS["color_background"],
"color_model": COLORS["color_model"],
"color_bounding_box": COLORS["color_bounding_box"],
"color_cutter": COLORS["color_cutter"],
"color_toolpath_return": COLORS["color_toolpath_return"],
"view_light": True,
"view_shadow": True,
"view_polygon": True,
"drill_progress_max_fps": 2,
}
""" the listed items will be loaded/saved via the preferences file in the
user's home directory on startup/shutdown"""
# floating point color values are only available since gtk 2.16 # floating point color values are only available since gtk 2.16
GTK_COLOR_MAX = 65535.0 GTK_COLOR_MAX = 65535.0
...@@ -387,21 +411,21 @@ class ProjectGui: ...@@ -387,21 +411,21 @@ class ProjectGui:
self.gui.add_from_file(gtk_build_file) self.gui.add_from_file(gtk_build_file)
self.window = self.gui.get_object("ProjectWindow") self.window = self.gui.get_object("ProjectWindow")
# file loading # file loading
self.last_settings_file = None self.last_task_settings_file = None
self.last_model_file = None self.last_model_file = None
self.last_toolpath_file = None self.last_toolpath_file = None
# define callbacks and accelerator keys for the menu actions # define callbacks and accelerator keys for the menu actions
for objname, callback, data, accel_key in ( for objname, callback, data, accel_key in (
("LoadSettings", self.load_settings_file, None, "<Control>t"), ("LoadTaskSettings", self.load_task_settings_file, None, "<Control>t"),
("SaveSettings", self.save_settings_file, lambda: self.last_settings_file, None), ("SaveTaskSettings", self.save_task_settings_file, lambda: self.last_task_settings_file, None),
("SaveAsSettings", self.save_settings_file, None, None), ("SaveAsTaskSettings", self.save_task_settings_file, None, None),
("LoadModel", self.load_model_file, None, "<Control>l"), ("LoadModel", self.load_model_file, None, "<Control>l"),
("SaveModel", self.save_model, lambda: self.last_model_file, "<Control>s"), ("SaveModel", self.save_model, lambda: self.last_model_file, "<Control>s"),
("SaveAsModel", self.save_model, None, "<Control><Shift>s"), ("SaveAsModel", self.save_model, None, "<Control><Shift>s"),
("ExportGCode", self.save_toolpath, None, "<Control><Shift>e"), ("ExportGCode", self.save_toolpath, None, "<Control><Shift>e"),
("ExportEMCToolDefinition", self.export_emc_tools, None, None), ("ExportEMCToolDefinition", self.export_emc_tools, None, None),
("Quit", self.destroy, None, "<Control>q"), ("Quit", self.destroy, None, "<Control>q"),
("GeneralSettings", self.toggle_settings_window, None, "<Control>p"), ("GeneralSettings", self.toggle_preferences_window, None, "<Control>p"),
("Toggle3DView", self.toggle_3d_view, None, "<Control>v")): ("Toggle3DView", self.toggle_3d_view, None, "<Control>v")):
item = self.gui.get_object(objname) item = self.gui.get_object(objname)
if objname == "Toggle3DView": if objname == "Toggle3DView":
...@@ -417,11 +441,12 @@ class ProjectGui: ...@@ -417,11 +441,12 @@ class ProjectGui:
# other events # other events
self.window.connect("destroy", self.destroy) self.window.connect("destroy", self.destroy)
# the settings window # the settings window
self.gui.get_object("CloseSettingsWindow").connect("clicked", self.toggle_settings_window, False) self.gui.get_object("CloseSettingsWindow").connect("clicked", self.toggle_preferences_window, False)
self.settings_window = self.gui.get_object("GeneralSettingsWindow") self.gui.get_object("ResetPreferencesButton").connect("clicked", self.reset_preferences)
self.settings_window.connect("delete-event", self.toggle_settings_window, False) self.preferences_window = self.gui.get_object("GeneralSettingsWindow")
self._settings_window_position = None self.preferences_window.connect("delete-event", self.toggle_preferences_window, False)
self._settings_window_visible = False self._preferences_window_position = None
self._preferences_window_visible = False
# "about" window # "about" window
self.about_window = self.gui.get_object("AboutWindow") self.about_window = self.gui.get_object("AboutWindow")
self.gui.get_object("About").connect("activate", self.toggle_about_window, True) self.gui.get_object("About").connect("activate", self.toggle_about_window, True)
...@@ -446,9 +471,8 @@ class ProjectGui: ...@@ -446,9 +471,8 @@ class ProjectGui:
unit_field.connect("changed", self.update_view) unit_field.connect("changed", self.update_view)
unit_field.connect("changed", self.update_unit_labels) unit_field.connect("changed", self.update_unit_labels)
def set_unit(text): def set_unit(text):
unit_field.set_active((text == "mm") and 0 or 1) unit_field.set_active(0 if text == "mm" else 1)
self.settings.add_item("unit", unit_field.get_active_text, set_unit) self.settings.add_item("unit", unit_field.get_active_text, set_unit)
self.settings.set("unit", "mm")
# boundary mode (move inside/along/around the boundaries) # boundary mode (move inside/along/around the boundaries)
boundary_mode_control = self.gui.get_object("BoundaryModeControl") boundary_mode_control = self.gui.get_object("BoundaryModeControl")
def set_boundary_mode(value): def set_boundary_mode(value):
...@@ -477,7 +501,6 @@ class ProjectGui: ...@@ -477,7 +501,6 @@ class ProjectGui:
# scale model to an axis dimension # scale model to an axis dimension
self.gui.get_object("ScaleDimensionAxis").connect("changed", self.switch_scale_axis) self.gui.get_object("ScaleDimensionAxis").connect("changed", self.switch_scale_axis)
self.gui.get_object("ScaleDimensionButton").connect("clicked", self.scale_model_axis_fit) self.gui.get_object("ScaleDimensionButton").connect("clicked", self.scale_model_axis_fit)
self.gui.get_object("ScaleDimensionsProportionally").set_active(True)
# visual and general settings # visual and general settings
for name, objname in (("show_model", "ShowModelCheckBox"), for name, objname in (("show_model", "ShowModelCheckBox"),
("show_axes", "ShowAxesCheckBox"), ("show_axes", "ShowAxesCheckBox"),
...@@ -519,9 +542,6 @@ class ProjectGui: ...@@ -519,9 +542,6 @@ class ProjectGui:
self.settings.add_item(name, get_color_wrapper(obj), set_color_wrapper(obj)) self.settings.add_item(name, get_color_wrapper(obj), set_color_wrapper(obj))
# repaint the 3d view after a color change # repaint the 3d view after a color change
obj.connect("color-set", self.update_view) obj.connect("color-set", self.update_view)
# pre-define the colors
for name in COLORS.keys():
self.settings.set(name, COLORS[name])
# set the availability of ODE # set the availability of ODE
if ode_objects.is_ode_available(): if ode_objects.is_ode_available():
self.settings.set("enable_ode", True) self.settings.set("enable_ode", True)
...@@ -532,18 +552,8 @@ class ProjectGui: ...@@ -532,18 +552,8 @@ class ProjectGui:
self.gui.get_object("SettingEnableODE").set_sensitive(False) self.gui.get_object("SettingEnableODE").set_sensitive(False)
# TODO: remove this as soon as non-ODE toolpath generation respects material allowance # TODO: remove this as soon as non-ODE toolpath generation respects material allowance
self.gui.get_object("MaterialAllowanceControl").set_sensitive(False) self.gui.get_object("MaterialAllowanceControl").set_sensitive(False)
# preconfigure some values
self.settings.set("show_model", True)
self.settings.set("show_toolpath", True)
self.settings.set("show_bounding_box", True)
self.settings.set("show_axes", True)
self.settings.set("show_dimensions", True)
self.settings.set("view_light", True)
self.settings.set("view_shadow", True)
self.settings.set("view_polygon", True)
skip_obj = self.gui.get_object("DrillProgressFrameSkipControl") skip_obj = self.gui.get_object("DrillProgressFrameSkipControl")
self.settings.add_item("drill_progress_max_fps", skip_obj.get_value, skip_obj.set_value) self.settings.add_item("drill_progress_max_fps", skip_obj.get_value, skip_obj.set_value)
self.settings.set("drill_progress_max_fps", 2)
# drill settings # drill settings
for objname, key in ( for objname, key in (
("ToolRadiusControl", "tool_radius"), ("ToolRadiusControl", "tool_radius"),
...@@ -625,7 +635,7 @@ class ProjectGui: ...@@ -625,7 +635,7 @@ class ProjectGui:
self._accel_group = uimanager.get_accel_group() self._accel_group = uimanager.get_accel_group()
self.window.add_accel_group(self._accel_group) self.window.add_accel_group(self._accel_group)
self.about_window.add_accel_group(self._accel_group) self.about_window.add_accel_group(self._accel_group)
self.settings_window.add_accel_group(self._accel_group) self.preferences_window.add_accel_group(self._accel_group)
# load menu data # load menu data
gtk_menu_file = get_data_file_location(GTKMENU_FILE) gtk_menu_file = get_data_file_location(GTKMENU_FILE)
if gtk_menu_file is None: if gtk_menu_file is None:
...@@ -643,7 +653,9 @@ class ProjectGui: ...@@ -643,7 +653,9 @@ class ProjectGui:
window_box.pack_start(self.menubar, False) window_box.pack_start(self.menubar, False)
window_box.reorder_child(self.menubar, 0) window_box.reorder_child(self.menubar, 0)
# some more initialization # some more initialization
self.load_settings() self.reset_preferences()
self.load_preferences()
self.load_task_settings()
self.update_all_controls() self.update_all_controls()
if not self.no_dialog: if not self.no_dialog:
self.window.show() self.window.show()
...@@ -701,7 +713,7 @@ class ProjectGui: ...@@ -701,7 +713,7 @@ class ProjectGui:
return self._physics_cache return self._physics_cache
def update_save_actions(self): def update_save_actions(self):
self.gui.get_object("SaveSettings").set_sensitive(not self.last_settings_file is None) self.gui.get_object("SaveTaskSettings").set_sensitive(not self.last_task_settings_file is None)
self.gui.get_object("SaveModel").set_sensitive(not self.last_model_file is None) self.gui.get_object("SaveModel").set_sensitive(not self.last_model_file is None)
def update_tasklist_controls(self, widget=None, data=None): def update_tasklist_controls(self, widget=None, data=None):
...@@ -877,20 +889,20 @@ class ProjectGui: ...@@ -877,20 +889,20 @@ class ProjectGui:
return True return True
@gui_activity_guard @gui_activity_guard
def toggle_settings_window(self, widget=None, event=None, state=None): def toggle_preferences_window(self, widget=None, event=None, state=None):
if state is None: if state is None:
# the "delete-event" issues the additional "event" argument # the "delete-event" issues the additional "event" argument
state = event state = event
if state is None: if state is None:
state = not self._settings_window_visible state = not self._preferences_window_visible
if state: if state:
if self._settings_window_position: if self._preferences_window_position:
self.settings_window.move(*self._settings_window_position) self.preferences_window.move(*self._preferences_window_position)
self.settings_window.show() self.preferences_window.show()
else: else:
self._settings_window_position = self.settings_window.get_position() self._preferences_window_position = self.preferences_window.get_position()
self.settings_window.hide() self.preferences_window.hide()
self._settings_window_visible = state self._preferences_window_visible = state
# don't close the window - just hide it (for "delete-event") # don't close the window - just hide it (for "delete-event")
return True return True
...@@ -1174,6 +1186,59 @@ class ProjectGui: ...@@ -1174,6 +1186,59 @@ class ProjectGui:
if not no_dialog and not self.no_dialog: if not no_dialog and not self.no_dialog:
show_error_dialog(self.window, "Failed to save model file") show_error_dialog(self.window, "Failed to save model file")
@gui_activity_guard
def reset_preferences(self, widget=None):
""" reset all preferences to their default values """
for key, value in PREFERENCES_DEFAULTS.items():
self.settings.set(key, value)
def load_preferences(self):
""" load all settings that are available in the Preferences window from
a file in the user's home directory """
config_filename = pycam.Gui.Settings.get_config_filename()
if config_filename is None:
# failed to create the personal preferences directory
return
config = ConfigParser.ConfigParser()
if not config.read(config_filename):
# no config file was read
return
# report any ignored (obsolete) preference keys present in the file
for item, value in config.items("DEFAULT"):
if not item in PREFERENCES_DEFAULTS.keys():
print "Warning: skipping obsolete preference item: %s" % str(item)
for item in PREFERENCES_DEFAULTS.keys():
if not config.has_option("DEFAULT", item):
# a new preference setting is missing in the (old) file
continue
value_raw = config.get("DEFAULT", item)
old_value = self.settings.get(item)
if isinstance(old_value, basestring):
# keep strings as they are
value = value_raw
else:
# parse tuples, integers, bools, ...
value = eval(value_raw)
self.settings.set(item, value)
def save_preferences(self):
""" save all settings that are available in the Preferences window to
a file in the user's home directory """
config_filename = pycam.Gui.Settings.get_config_filename()
if config_filename is None:
# failed to create the personal preferences directory
print >>sys.stderr, "Warning: Failed to create a preferences directory your user home directory."
return
config = ConfigParser.ConfigParser()
for item in PREFERENCES_DEFAULTS.keys():
config.set("DEFAULT", item, self.settings.get(item))
try:
config_file = file(config_filename, "w")
config.write(config_file)
config_file.close()
except IOError, err_msg:
print >>sys.stderr, "Warning: Failed to write preferences file (%s): %s" % (config_filename, err_msg)
@gui_activity_guard @gui_activity_guard
def shift_model(self, widget, use_form_values=True): def shift_model(self, widget, use_form_values=True):
if use_form_values: if use_form_values:
...@@ -1286,6 +1351,7 @@ class ProjectGui: ...@@ -1286,6 +1351,7 @@ class ProjectGui:
# wait until if is finished # wait until if is finished
while self._progress_running: while self._progress_running:
time.sleep(0.5) time.sleep(0.5)
self.save_preferences()
gtk.main_quit() gtk.main_quit()
def open(self, filename): def open(self, filename):
...@@ -1334,24 +1400,24 @@ class ProjectGui: ...@@ -1334,24 +1400,24 @@ class ProjectGui:
if not no_dialog and not self.no_dialog: if not no_dialog and not self.no_dialog:
show_error_dialog(self.window, "Failed to save EMC tool file") show_error_dialog(self.window, "Failed to save EMC tool file")
def open_settings_file(self, filename): def open_task_settings_file(self, filename):
""" This function is used by the commandline handler """ """ This function is used by the commandline handler """
self.last_settings_file = filename self.last_task_settings_file = filename
self.load_settings_file(filename=filename) self.load_task_settings_file(filename=filename)
self.update_save_actions() self.update_save_actions()
@gui_activity_guard @gui_activity_guard
def load_settings_file(self, widget=None, filename=None): def load_task_settings_file(self, widget=None, filename=None):
if callable(filename): if callable(filename):
filename = filename() filename = filename()
if not filename: if not filename:
filename = self.get_filename_via_dialog("Loading settings ...", filename = self.get_filename_via_dialog("Loading settings ...",
mode_load=True, type_filter=FILTER_CONFIG) mode_load=True, type_filter=FILTER_CONFIG)
if filename: if filename:
self.last_settings_file = filename self.last_task_settings_file = filename
self.update_save_actions() self.update_save_actions()
if filename: if filename:
self.load_settings(filename) self.load_task_settings(filename)
def load_model(self, model): def load_model(self, model):
# load the new model only if the import worked # load the new model only if the import worked
...@@ -1365,8 +1431,8 @@ class ProjectGui: ...@@ -1365,8 +1431,8 @@ class ProjectGui:
self.append_to_queue(self.toggle_3d_view, value=True) self.append_to_queue(self.toggle_3d_view, value=True)
self.append_to_queue(self.update_view) self.append_to_queue(self.update_view)
def load_settings(self, filename=None): def load_task_settings(self, filename=None):
settings = pycam.Gui.Settings.SettingsFileManager() settings = pycam.Gui.Settings.ProcessSettings()
if not filename is None: if not filename is None:
settings.load_file(filename) settings.load_file(filename)
self.tool_list = settings.get_tools() self.tool_list = settings.get_tools()
...@@ -1544,7 +1610,7 @@ class ProjectGui: ...@@ -1544,7 +1610,7 @@ class ProjectGui:
self.gui.get_object("toolpath_down").set_sensitive((not new_index is None) and (new_index + 1 < len(self.toolpath))) self.gui.get_object("toolpath_down").set_sensitive((not new_index is None) and (new_index + 1 < len(self.toolpath)))
@gui_activity_guard @gui_activity_guard
def save_settings_file(self, widget=None, filename=None): def save_task_settings_file(self, widget=None, filename=None):
no_dialog = False no_dialog = False
if callable(filename): if callable(filename):
filename = filename() filename = filename()
...@@ -1555,12 +1621,12 @@ class ProjectGui: ...@@ -1555,12 +1621,12 @@ class ProjectGui:
filename = self.get_filename_via_dialog("Save settings to ...", filename = self.get_filename_via_dialog("Save settings to ...",
mode_load=False, type_filter=FILTER_CONFIG) mode_load=False, type_filter=FILTER_CONFIG)
if filename: if filename:
self.last_settings_file = filename self.last_task_settings_file = filename
self.update_save_actions() self.update_save_actions()
# no filename given -> exit # no filename given -> exit
if not filename: if not filename:
return return
settings = pycam.Gui.Settings.SettingsFileManager() settings = pycam.Gui.Settings.ProcessSettings()
if not settings.write_to_file(filename, self.tool_list, self.process_list, self.task_list) and not no_dialog and not self.no_dialog: if not settings.write_to_file(filename, self.tool_list, self.process_list, self.task_list) and not no_dialog and not self.no_dialog:
show_error_dialog(self.window, "Failed to save settings file") show_error_dialog(self.window, "Failed to save settings file")
......
...@@ -3,12 +3,40 @@ import StringIO ...@@ -3,12 +3,40 @@ import StringIO
import sys import sys
import os import os
GET_INDEX = 0 CONFIG_DIR = "pycam"
SET_INDEX = 1
VALUE_INDEX = 2 def get_config_dirname():
try:
from win32com.shell import shellcon, shell
homedir = shell.SHGetFolderPath(0, shellcon.CSIDL_APPDATA, 0, 0)
config_dir = os.path.join(homedir, CONFIG_DIR)
except ImportError: # quick semi-nasty fallback for non-windows/win32com case
homedir = os.path.expanduser("~")
# hide the config directory for unixes
config_dir = os.path.join(homedir, "." + CONFIG_DIR)
if not os.path.isdir(config_dir):
try:
os.makedirs(config_dir)
except OSError:
config_dir = None
return config_dir
def get_config_filename(filename=None):
if filename is None:
filename = "preferences.conf"
config_dir = get_config_dirname()
if config_dir is None:
return None
else:
return os.path.join(config_dir, filename)
class Settings: class Settings:
GET_INDEX = 0
SET_INDEX = 1
VALUE_INDEX = 2
def __init__(self): def __init__(self):
self.items = {} self.items = {}
self.values = {} self.values = {}
...@@ -17,35 +45,39 @@ class Settings: ...@@ -17,35 +45,39 @@ class Settings:
self.items[key] = [None, None, None] self.items[key] = [None, None, None]
self.define_get_func(key, get_func) self.define_get_func(key, get_func)
self.define_set_func(key, set_func) self.define_set_func(key, set_func)
self.items[key][VALUE_INDEX] = None self.items[key][self.VALUE_INDEX] = None
def define_get_func(self, key, get_func=None): def define_get_func(self, key, get_func=None):
if not self.items.has_key(key): if not self.items.has_key(key):
return return
if get_func is None: if get_func is None:
get_func = lambda: self.items[key][VALUE_INDEX] get_func = lambda: self.items[key][self.VALUE_INDEX]
self.items[key][GET_INDEX] = get_func self.items[key][self.GET_INDEX] = get_func
def define_set_func(self, key, set_func=None): def define_set_func(self, key, set_func=None):
if not self.items.has_key(key): if not self.items.has_key(key):
return return
def default_set_func(value): def default_set_func(value):
self.items[key][VALUE_INDEX] = value self.items[key][self.VALUE_INDEX] = value
if set_func is None: if set_func is None:
set_func = default_set_func set_func = default_set_func
self.items[key][SET_INDEX] = set_func self.items[key][self.SET_INDEX] = set_func
def get(self, key, default=None): def get(self, key, default=None):
if self.items.has_key(key): if self.items.has_key(key):
return self.items[key][GET_INDEX]() return self.items[key][self.GET_INDEX]()
else: else:
return default return default
def set(self, key, value): def set(self, key, value):
if not self.items.has_key(key): if not self.items.has_key(key):
self.add_item(key) self.add_item(key)
self.items[key][SET_INDEX](value) self.items[key][self.SET_INDEX](value)
self.items[key][VALUE_INDEX] = value self.items[key][self.VALUE_INDEX] = value
def has_key(self, key):
""" expose the "has_key" function of the items list """
return self.items.has_key(key)
def __str__(self): def __str__(self):
result = {} result = {}
...@@ -54,7 +86,7 @@ class Settings: ...@@ -54,7 +86,7 @@ class Settings:
return str(result) return str(result)
class SettingsFileManager: class ProcessSettings:
DEFAULT_CONFIG = """ DEFAULT_CONFIG = """
[ToolDefault] [ToolDefault]
......
...@@ -10,9 +10,9 @@ ...@@ -10,9 +10,9 @@
<menuitem action="Quit"/> <menuitem action="Quit"/>
</menu> </menu>
<menu action="SettingsMenu"> <menu action="SettingsMenu">
<menuitem action="LoadSettings"/> <menuitem action="LoadTaskSettings"/>
<menuitem action="SaveSettings"/> <menuitem action="SaveTaskSettings"/>
<menuitem action="SaveAsSettings"/> <menuitem action="SaveAsTaskSettings"/>
<separator /> <separator />
<menuitem action="ExportEMCToolDefinition"/> <menuitem action="ExportEMCToolDefinition"/>
<separator /> <separator />
......
...@@ -614,6 +614,7 @@ ...@@ -614,6 +614,7 @@
<property name="can_focus">True</property> <property name="can_focus">True</property>
<property name="receives_default">False</property> <property name="receives_default">False</property>
<property name="tooltip_text" translatable="yes">Scale the model along all axes proportionally.</property> <property name="tooltip_text" translatable="yes">Scale the model along all axes proportionally.</property>
<property name="active">True</property>
<property name="draw_indicator">True</property> <property name="draw_indicator">True</property>
</object> </object>
<packing> <packing>
...@@ -2178,19 +2179,19 @@ This combination is added to the above task list.</property> ...@@ -2178,19 +2179,19 @@ This combination is added to the above task list.</property>
<property name="tooltip">Write all current toolpaths to a file.</property> <property name="tooltip">Write all current toolpaths to a file.</property>
<property name="stock_id">gtk-execute</property> <property name="stock_id">gtk-execute</property>
</object> </object>
<object class="GtkAction" id="LoadSettings"> <object class="GtkAction" id="LoadTaskSettings">
<property name="label">_Load Settings</property> <property name="label">_Load Task Settings</property>
<property name="tooltip">Loads a set of processing templates from a file.</property> <property name="tooltip">Load tool, process and task definitions from a file.</property>
<property name="stock_id">gtk-open</property> <property name="stock_id">gtk-open</property>
</object> </object>
<object class="GtkAction" id="SaveSettings"> <object class="GtkAction" id="SaveTaskSettings">
<property name="label">_Save Settings</property> <property name="label">_Save Task Settings</property>
<property name="tooltip">Save the currently visible set of processing templates to a file.</property> <property name="tooltip">Save the tool, process and task definitions.</property>
<property name="stock_id">gtk-save</property> <property name="stock_id">gtk-save</property>
</object> </object>
<object class="GtkAction" id="SaveAsSettings"> <object class="GtkAction" id="SaveAsTaskSettings">
<property name="label">Save Settings _as ...</property> <property name="label">Save Task Settings _as ...</property>
<property name="tooltip">Save the currently visible set of processing templates to a new file.</property> <property name="tooltip">Save the tool, process and task definitions to a new file.</property>
<property name="stock_id">gtk-save-as</property> <property name="stock_id">gtk-save-as</property>
</object> </object>
<object class="GtkAction" id="GeneralSettings"> <object class="GtkAction" id="GeneralSettings">
...@@ -2223,7 +2224,7 @@ This combination is added to the above task list.</property> ...@@ -2223,7 +2224,7 @@ This combination is added to the above task list.</property>
</object> </object>
<object class="GtkDialog" id="GeneralSettingsWindow"> <object class="GtkDialog" id="GeneralSettingsWindow">
<property name="border_width">5</property> <property name="border_width">5</property>
<property name="title" translatable="yes">PyCAM Settings</property> <property name="title" translatable="yes">PyCAM Preferences</property>
<property name="destroy_with_parent">True</property> <property name="destroy_with_parent">True</property>
<property name="type_hint">normal</property> <property name="type_hint">normal</property>
<child internal-child="vbox"> <child internal-child="vbox">
...@@ -2796,7 +2797,18 @@ This combination is added to the above task list.</property> ...@@ -2796,7 +2797,18 @@ This combination is added to the above task list.</property>
<property name="visible">True</property> <property name="visible">True</property>
<property name="layout_style">end</property> <property name="layout_style">end</property>
<child> <child>
<placeholder/> <object class="GtkButton" id="ResetPreferencesButton">
<property name="label">gtk-revert-to-saved</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">True</property>
<property name="use_stock">True</property>
</object>
<packing>
<property name="expand">False</property>
<property name="fill">False</property>
<property name="position">0</property>
</packing>
</child> </child>
<child> <child>
<object class="GtkButton" id="CloseSettingsWindow"> <object class="GtkButton" id="CloseSettingsWindow">
...@@ -2822,6 +2834,7 @@ This combination is added to the above task list.</property> ...@@ -2822,6 +2834,7 @@ This combination is added to the above task list.</property>
</object> </object>
</child> </child>
<action-widgets> <action-widgets>
<action-widget response="0">ResetPreferencesButton</action-widget>
<action-widget response="0">CloseSettingsWindow</action-widget> <action-widget response="0">CloseSettingsWindow</action-widget>
</action-widgets> </action-widgets>
</object> </object>
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment