The goal here is to make app templates usable for default templates that we can ship with Blender. These only have a custom startup.blend currently and so are quite limited compared to app templates that fully customize Blender. But still it seems like the same kind of concept where we should be sharing the code and UI. It is useful to be able to save a startup.blend per template, and I can imagine some scripting being useful in the future as well. Changes made: * File > New and Ctrl+N now list the templates, replacing a separate Application Templates menu that was not as easy to discover. * File menu now shows name of active template above Save Startup File and Load Factory Settings to indicate these are saved/loaded per template. * The "Default" template was renamed to "General". * Workspaces can now be added from any of the template startup.blend files when clicking the (+) button in the topbar. * User preferences are now fully shared between app templates, unless the template includes a custom userpref.blend. I think this will be useful in general, not all app templates need their own keymaps for example. * Previously Save User Preferences would save the current app template and then Blender would start using that template by default. I've disabled this, to me it seems it was unintentional, or at least not clear at all that saving user preferences also makes the current Differential Revision: https://developer.blender.org/D3690
602 lines
18 KiB
C
602 lines
18 KiB
C
/*
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
*/
|
|
|
|
/** \file blender/editors/screen/workspace_edit.c
|
|
* \ingroup edscr
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "BLI_utildefines.h"
|
|
#include "BLI_fileops.h"
|
|
#include "BLI_listbase.h"
|
|
#include "BLI_path_util.h"
|
|
#include "BLI_string.h"
|
|
|
|
#include "BKE_appdir.h"
|
|
#include "BKE_blendfile.h"
|
|
#include "BKE_context.h"
|
|
#include "BKE_idcode.h"
|
|
#include "BKE_main.h"
|
|
#include "BKE_layer.h"
|
|
#include "BKE_library.h"
|
|
#include "BKE_report.h"
|
|
#include "BKE_scene.h"
|
|
#include "BKE_screen.h"
|
|
#include "BKE_workspace.h"
|
|
|
|
#include "BLO_readfile.h"
|
|
|
|
#include "DNA_object_types.h"
|
|
#include "DNA_screen_types.h"
|
|
#include "DNA_windowmanager_types.h"
|
|
#include "DNA_workspace_types.h"
|
|
|
|
#include "ED_datafiles.h"
|
|
#include "ED_object.h"
|
|
#include "ED_screen.h"
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "RNA_access.h"
|
|
#include "RNA_define.h"
|
|
|
|
#include "DEG_depsgraph.h"
|
|
|
|
#include "UI_interface.h"
|
|
#include "UI_resources.h"
|
|
|
|
#include "BLT_translation.h"
|
|
|
|
#include "WM_api.h"
|
|
#include "WM_types.h"
|
|
#include "WM_toolsystem.h"
|
|
|
|
#include "screen_intern.h"
|
|
|
|
|
|
/** \name Workspace API
|
|
*
|
|
* \brief API for managing workspaces and their data.
|
|
* \{ */
|
|
|
|
WorkSpace *ED_workspace_add(Main *bmain, const char *name)
|
|
{
|
|
return BKE_workspace_add(bmain, name);
|
|
}
|
|
|
|
/**
|
|
* Changes the object mode (if needed) to the one set in \a workspace_new.
|
|
* Object mode is still stored on object level. In future it should all be workspace level instead.
|
|
*/
|
|
static void workspace_change_update(
|
|
WorkSpace *workspace_new, const WorkSpace *workspace_old,
|
|
bContext *C, wmWindowManager *wm)
|
|
{
|
|
/* needs to be done before changing mode! (to ensure right context) */
|
|
UNUSED_VARS(workspace_old, workspace_new, C, wm);
|
|
#if 0
|
|
Object *ob_act = CTX_data_active_object(C)
|
|
eObjectMode mode_old = workspace_old->object_mode;
|
|
eObjectMode mode_new = workspace_new->object_mode;
|
|
|
|
if (mode_old != mode_new) {
|
|
ED_object_mode_compat_set(C, ob_act, mode_new, &wm->reports);
|
|
ED_object_mode_toggle(C, mode_new);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static bool workspace_change_find_new_layout_cb(const WorkSpaceLayout *layout, void *UNUSED(arg))
|
|
{
|
|
/* return false to stop the iterator if we've found a layout that can be activated */
|
|
return workspace_layout_set_poll(layout) ? false : true;
|
|
}
|
|
|
|
static WorkSpaceLayout *workspace_change_get_new_layout(
|
|
Main *bmain, WorkSpace *workspace_new, wmWindow *win)
|
|
{
|
|
/* ED_workspace_duplicate may have stored a layout to activate once the workspace gets activated. */
|
|
WorkSpaceLayout *layout_old = WM_window_get_active_layout(win);
|
|
WorkSpaceLayout *layout_new;
|
|
bScreen *screen_new;
|
|
|
|
if (win->workspace_hook->temp_workspace_store) {
|
|
layout_new = win->workspace_hook->temp_layout_store;
|
|
}
|
|
else {
|
|
layout_new = BKE_workspace_hook_layout_for_workspace_get(win->workspace_hook, workspace_new);
|
|
if (!layout_new) {
|
|
layout_new = BKE_workspace_layouts_get(workspace_new)->first;
|
|
}
|
|
}
|
|
screen_new = BKE_workspace_layout_screen_get(layout_new);
|
|
|
|
if (screen_new->winid) {
|
|
/* screen is already used, try to find a free one */
|
|
WorkSpaceLayout *layout_temp = BKE_workspace_layout_iter_circular(
|
|
workspace_new, layout_new, workspace_change_find_new_layout_cb,
|
|
NULL, false);
|
|
if (!layout_temp) {
|
|
/* fallback solution: duplicate layout from old workspace */
|
|
layout_temp = ED_workspace_layout_duplicate(bmain, workspace_new, layout_old, win);
|
|
}
|
|
layout_new = layout_temp;
|
|
}
|
|
|
|
return layout_new;
|
|
}
|
|
|
|
/**
|
|
* \brief Change the active workspace.
|
|
*
|
|
* Operator call, WM + Window + screen already existed before
|
|
* Pretty similar to #ED_screen_change since changing workspace also changes screen.
|
|
*
|
|
* \warning Do NOT call in area/region queues!
|
|
* \returns if workspace changing was successful.
|
|
*/
|
|
bool ED_workspace_change(
|
|
WorkSpace *workspace_new, bContext *C, wmWindowManager *wm, wmWindow *win)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
WorkSpace *workspace_old = WM_window_get_active_workspace(win);
|
|
WorkSpaceLayout *layout_new = workspace_change_get_new_layout(bmain, workspace_new, win);
|
|
bScreen *screen_new = BKE_workspace_layout_screen_get(layout_new);
|
|
bScreen *screen_old = BKE_workspace_active_screen_get(win->workspace_hook);
|
|
|
|
win->workspace_hook->temp_layout_store = NULL;
|
|
if (workspace_old == workspace_new) {
|
|
/* Could also return true, everything that needs to be done was done (nothing :P), but nothing changed */
|
|
return false;
|
|
}
|
|
|
|
screen_new = screen_change_prepare(screen_old, screen_new, bmain, C, win);
|
|
BLI_assert(BKE_workspace_layout_screen_get(layout_new) == screen_new);
|
|
|
|
if (screen_new == NULL) {
|
|
return false;
|
|
}
|
|
|
|
BKE_workspace_hook_layout_for_workspace_set(win->workspace_hook, workspace_new, layout_new);
|
|
BKE_workspace_active_set(win->workspace_hook, workspace_new);
|
|
|
|
/* update screen *after* changing workspace - which also causes the
|
|
* actual screen change and updates context (including CTX_wm_workspace) */
|
|
screen_change_update(C, win, screen_new);
|
|
workspace_change_update(workspace_new, workspace_old, C, wm);
|
|
|
|
BLI_assert(CTX_wm_workspace(C) == workspace_new);
|
|
|
|
WM_toolsystem_unlink_all(C, workspace_old);
|
|
WM_toolsystem_reinit_all(C, win);
|
|
|
|
/* Automatic mode switching. */
|
|
if (workspace_new->object_mode != workspace_old->object_mode) {
|
|
ED_object_mode_generic_enter(C, workspace_new->object_mode);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Duplicate a workspace including its layouts. Does not activate the workspace, but
|
|
* it stores the screen-layout to be activated (BKE_workspace_temp_layout_store)
|
|
*/
|
|
WorkSpace *ED_workspace_duplicate(
|
|
WorkSpace *workspace_old, Main *bmain, wmWindow *win)
|
|
{
|
|
WorkSpaceLayout *layout_active_old = BKE_workspace_active_layout_get(win->workspace_hook);
|
|
ListBase *layouts_old = BKE_workspace_layouts_get(workspace_old);
|
|
WorkSpace *workspace_new = ED_workspace_add(bmain, workspace_old->id.name + 2);
|
|
|
|
/* TODO(campbell): tools */
|
|
|
|
for (WorkSpaceLayout *layout_old = layouts_old->first; layout_old; layout_old = layout_old->next) {
|
|
WorkSpaceLayout *layout_new = ED_workspace_layout_duplicate(bmain, workspace_new, layout_old, win);
|
|
|
|
if (layout_active_old == layout_old) {
|
|
win->workspace_hook->temp_layout_store = layout_new;
|
|
}
|
|
}
|
|
return workspace_new;
|
|
}
|
|
|
|
/**
|
|
* \return if succeeded.
|
|
*/
|
|
bool ED_workspace_delete(
|
|
WorkSpace *workspace, Main *bmain, bContext *C, wmWindowManager *wm)
|
|
{
|
|
ID *workspace_id = (ID *)workspace;
|
|
|
|
if (BLI_listbase_is_single(&bmain->workspaces)) {
|
|
return false;
|
|
}
|
|
|
|
for (wmWindow *win = wm->windows.first; win; win = win->next) {
|
|
WorkSpace *prev = workspace_id->prev;
|
|
WorkSpace *next = workspace_id->next;
|
|
|
|
ED_workspace_change((prev != NULL) ? prev : next, C, wm, win);
|
|
}
|
|
BKE_libblock_free(bmain, workspace_id);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Some editor data may need to be synced with scene data (3D View camera and layers).
|
|
* This function ensures data is synced for editors in active layout of \a workspace.
|
|
*/
|
|
void ED_workspace_scene_data_sync(
|
|
WorkSpaceInstanceHook *hook, Scene *scene)
|
|
{
|
|
bScreen *screen = BKE_workspace_active_screen_get(hook);
|
|
BKE_screen_view3d_scene_sync(screen, scene);
|
|
}
|
|
|
|
/** \} Workspace API */
|
|
|
|
|
|
/** \name Workspace Operators
|
|
*
|
|
* \{ */
|
|
|
|
static WorkSpace *workspace_context_get(bContext *C)
|
|
{
|
|
ID *id = UI_context_active_but_get_tab_ID(C);
|
|
if (id && GS(id->name) == ID_WS) {
|
|
return (WorkSpace *)id;
|
|
}
|
|
|
|
wmWindow *win = CTX_wm_window(C);
|
|
return WM_window_get_active_workspace(win);
|
|
}
|
|
|
|
static bool workspace_context_poll(bContext *C)
|
|
{
|
|
return workspace_context_get(C) != NULL;
|
|
}
|
|
|
|
static int workspace_new_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
wmWindow *win = CTX_wm_window(C);
|
|
WorkSpace *workspace = workspace_context_get(C);
|
|
|
|
workspace = ED_workspace_duplicate(workspace, bmain, win);
|
|
|
|
WM_event_add_notifier(C, NC_SCREEN | ND_WORKSPACE_SET, workspace);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void WORKSPACE_OT_duplicate(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "New Workspace";
|
|
ot->description = "Add a new workspace";
|
|
ot->idname = "WORKSPACE_OT_duplicate";
|
|
|
|
/* api callbacks */
|
|
ot->poll = workspace_context_poll;
|
|
ot->exec = workspace_new_exec;
|
|
}
|
|
|
|
static int workspace_delete_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
WorkSpace *workspace = workspace_context_get(C);
|
|
WM_event_add_notifier(C, NC_SCREEN | ND_WORKSPACE_DELETE, workspace);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void WORKSPACE_OT_delete(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Delete Workspace";
|
|
ot->description = "Delete the active workspace";
|
|
ot->idname = "WORKSPACE_OT_delete";
|
|
|
|
/* api callbacks */
|
|
ot->poll = workspace_context_poll;
|
|
ot->exec = workspace_delete_exec;
|
|
}
|
|
|
|
static bool workspace_append_activate_poll(bContext *C)
|
|
{
|
|
wmOperatorType *ot = WM_operatortype_find("WM_OT_append", false);
|
|
return WM_operator_poll(C, ot);
|
|
}
|
|
|
|
static int workspace_append(bContext *C, const char *directory, const char *idname)
|
|
{
|
|
wmOperatorType *ot = WM_operatortype_find("WM_OT_append", false);
|
|
PointerRNA opptr;
|
|
int retval;
|
|
|
|
WM_operator_properties_create_ptr(&opptr, ot);
|
|
RNA_string_set(&opptr, "directory", directory);
|
|
RNA_string_set(&opptr, "filename", idname);
|
|
RNA_boolean_set(&opptr, "autoselect", false);
|
|
|
|
retval = WM_operator_name_call_ptr(C, ot, WM_OP_EXEC_DEFAULT, &opptr);
|
|
|
|
WM_operator_properties_free(&opptr);
|
|
|
|
return retval;
|
|
}
|
|
|
|
static int workspace_append_activate_exec(bContext *C, wmOperator *op)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
char idname[MAX_ID_NAME - 2], filepath[FILE_MAX];
|
|
|
|
if (!RNA_struct_property_is_set(op->ptr, "idname") ||
|
|
!RNA_struct_property_is_set(op->ptr, "filepath"))
|
|
{
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
RNA_string_get(op->ptr, "idname", idname);
|
|
RNA_string_get(op->ptr, "filepath", filepath);
|
|
|
|
if (workspace_append(C, filepath, idname) != OPERATOR_CANCELLED) {
|
|
WorkSpace *appended_workspace = BLI_findstring(&bmain->workspaces, idname, offsetof(ID, name) + 2);
|
|
BLI_assert(appended_workspace != NULL);
|
|
|
|
/* Reorder to last position. */
|
|
BKE_id_reorder(&bmain->workspaces, &appended_workspace->id, NULL, true);
|
|
|
|
/* Changing workspace changes context. Do delayed! */
|
|
WM_event_add_notifier(C, NC_SCREEN | ND_WORKSPACE_SET, appended_workspace);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
static void WORKSPACE_OT_append_activate(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Append and Activate Workspace";
|
|
ot->description = "Append a workspace and make it the active one in the current window";
|
|
ot->idname = "WORKSPACE_OT_append_activate";
|
|
|
|
/* api callbacks */
|
|
ot->exec = workspace_append_activate_exec;
|
|
ot->poll = workspace_append_activate_poll;
|
|
|
|
RNA_def_string(ot->srna, "idname", NULL, MAX_ID_NAME - 2, "Identifier",
|
|
"Name of the workspace to append and activate");
|
|
RNA_def_string(ot->srna, "filepath", NULL, FILE_MAX, "Filepath",
|
|
"Path to the library");
|
|
}
|
|
|
|
static WorkspaceConfigFileData *workspace_config_file_read(const char *app_template)
|
|
{
|
|
const char *cfgdir = BKE_appdir_folder_id(BLENDER_USER_CONFIG, app_template);
|
|
char startup_file_path[FILE_MAX] = {0};
|
|
|
|
if (cfgdir) {
|
|
BLI_join_dirfile(startup_file_path, sizeof(startup_file_path), cfgdir, BLENDER_STARTUP_FILE);
|
|
}
|
|
|
|
bool has_path = BLI_exists(startup_file_path);
|
|
return (has_path) ? BKE_blendfile_workspace_config_read(startup_file_path, NULL, 0, NULL) : NULL;
|
|
}
|
|
|
|
static WorkspaceConfigFileData *workspace_system_file_read(const char *app_template)
|
|
{
|
|
if (app_template == NULL) {
|
|
return BKE_blendfile_workspace_config_read(NULL, datatoc_startup_blend, datatoc_startup_blend_size, NULL);
|
|
}
|
|
|
|
char template_dir[FILE_MAX];
|
|
if (!BKE_appdir_app_template_id_search(app_template, template_dir, sizeof(template_dir))) {
|
|
return NULL;
|
|
}
|
|
|
|
char startup_file_path[FILE_MAX];
|
|
BLI_join_dirfile(startup_file_path, sizeof(startup_file_path), template_dir, BLENDER_STARTUP_FILE);
|
|
|
|
bool has_path = BLI_exists(startup_file_path);
|
|
return (has_path) ? BKE_blendfile_workspace_config_read(startup_file_path, NULL, 0, NULL) : NULL;
|
|
}
|
|
|
|
static void workspace_append_button(
|
|
uiLayout *layout, wmOperatorType *ot_append, const WorkSpace *workspace, const Main *from_main)
|
|
{
|
|
const ID *id = (ID *)workspace;
|
|
PointerRNA opptr;
|
|
char lib_path[FILE_MAX_LIBEXTRA];
|
|
const char *filepath = from_main->name;
|
|
|
|
if (strlen(filepath) == 0) {
|
|
filepath = BLO_EMBEDDED_STARTUP_BLEND;
|
|
}
|
|
|
|
BLI_path_join(
|
|
lib_path, sizeof(lib_path), filepath, BKE_idcode_to_name(GS(id->name)), NULL);
|
|
|
|
BLI_assert(STREQ(ot_append->idname, "WORKSPACE_OT_append_activate"));
|
|
uiItemFullO_ptr(
|
|
layout, ot_append, workspace->id.name + 2, ICON_NONE, NULL,
|
|
WM_OP_EXEC_DEFAULT, 0, &opptr);
|
|
RNA_string_set(&opptr, "idname", id->name + 2);
|
|
RNA_string_set(&opptr, "filepath", lib_path);
|
|
}
|
|
|
|
static void workspace_add_menu(bContext *C, uiLayout *layout, void *template_v)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
const char *app_template = template_v;
|
|
bool has_startup_items = false;
|
|
|
|
wmOperatorType *ot_append = WM_operatortype_find("WORKSPACE_OT_append_activate", true);
|
|
WorkspaceConfigFileData *startup_config = workspace_config_file_read(app_template);
|
|
WorkspaceConfigFileData *builtin_config = workspace_system_file_read(app_template);
|
|
|
|
if (startup_config) {
|
|
for (WorkSpace *workspace = startup_config->workspaces.first; workspace; workspace = workspace->id.next) {
|
|
uiLayout *row = uiLayoutRow(layout, false);
|
|
if (BLI_findstring(&bmain->workspaces, workspace->id.name, offsetof(ID, name))) {
|
|
uiLayoutSetActive(row, false);
|
|
}
|
|
|
|
workspace_append_button(row, ot_append, workspace, startup_config->main);
|
|
has_startup_items = true;
|
|
}
|
|
}
|
|
|
|
if (builtin_config) {
|
|
bool has_title = false;
|
|
|
|
for (WorkSpace *workspace = builtin_config->workspaces.first; workspace; workspace = workspace->id.next) {
|
|
if (startup_config && BLI_findstring(&startup_config->workspaces, workspace->id.name, offsetof(ID, name))) {
|
|
continue;
|
|
}
|
|
|
|
if (!has_title) {
|
|
if (has_startup_items) {
|
|
uiItemS(layout);
|
|
}
|
|
has_title = true;
|
|
}
|
|
|
|
uiLayout *row = uiLayoutRow(layout, false);
|
|
if (BLI_findstring(&bmain->workspaces, workspace->id.name, offsetof(ID, name))) {
|
|
uiLayoutSetActive(row, false);
|
|
}
|
|
|
|
workspace_append_button(row, ot_append, workspace, builtin_config->main);
|
|
}
|
|
}
|
|
|
|
if (startup_config) {
|
|
BKE_blendfile_workspace_config_data_free(startup_config);
|
|
}
|
|
if (builtin_config) {
|
|
BKE_blendfile_workspace_config_data_free(builtin_config);
|
|
}
|
|
}
|
|
|
|
static int workspace_add_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event))
|
|
{
|
|
uiPopupMenu *pup = UI_popup_menu_begin(C, op->type->name, ICON_NONE);
|
|
uiLayout *layout = UI_popup_menu_layout(pup);
|
|
|
|
uiItemMenuF(layout, IFACE_("General"), ICON_NONE, workspace_add_menu, NULL);
|
|
|
|
ListBase templates;
|
|
BKE_appdir_app_templates(&templates);
|
|
|
|
for (LinkData *link = templates.first; link; link = link->next) {
|
|
char *template = link->data;
|
|
char display_name[FILE_MAX];
|
|
|
|
BLI_path_to_display_name(display_name, sizeof(display_name), template);
|
|
|
|
/* Steals ownership of link data string. */
|
|
uiItemMenuF(layout, display_name, ICON_NONE, workspace_add_menu, template);
|
|
}
|
|
|
|
BLI_freelistN(&templates);
|
|
|
|
uiItemS(layout);
|
|
uiItemO(layout, "Duplicate Current", ICON_NONE, "WORKSPACE_OT_duplicate");
|
|
|
|
UI_popup_menu_end(C, pup);
|
|
|
|
return OPERATOR_INTERFACE;
|
|
}
|
|
|
|
static void WORKSPACE_OT_add(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Add Workspace";
|
|
ot->description = "Add a new workspace by duplicating the current one or appending one "
|
|
"from the user configuration";
|
|
ot->idname = "WORKSPACE_OT_add";
|
|
|
|
/* api callbacks */
|
|
ot->invoke = workspace_add_invoke;
|
|
}
|
|
|
|
static int workspace_reorder_to_back_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
WorkSpace *workspace = workspace_context_get(C);
|
|
|
|
BKE_id_reorder(&bmain->workspaces, &workspace->id, NULL, true);
|
|
WM_event_add_notifier(C, NC_WINDOW, NULL);
|
|
|
|
return OPERATOR_INTERFACE;
|
|
}
|
|
|
|
static void WORKSPACE_OT_reorder_to_back(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Workspace Reorder to Back";
|
|
ot->description = "Reorder workspace to be first in the list";
|
|
ot->idname = "WORKSPACE_OT_reorder_to_back";
|
|
|
|
/* api callbacks */
|
|
ot->poll = workspace_context_poll;
|
|
ot->exec = workspace_reorder_to_back_exec;
|
|
}
|
|
|
|
static int workspace_reorder_to_front_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
WorkSpace *workspace = workspace_context_get(C);
|
|
|
|
BKE_id_reorder(&bmain->workspaces, &workspace->id, NULL, false);
|
|
WM_event_add_notifier(C, NC_WINDOW, NULL);
|
|
|
|
return OPERATOR_INTERFACE;
|
|
}
|
|
|
|
static void WORKSPACE_OT_reorder_to_front(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Workspace Reorder to Front";
|
|
ot->description = "Reorder workspace to be first in the list";
|
|
ot->idname = "WORKSPACE_OT_reorder_to_front";
|
|
|
|
/* api callbacks */
|
|
ot->poll = workspace_context_poll;
|
|
ot->exec = workspace_reorder_to_front_exec;
|
|
}
|
|
|
|
void ED_operatortypes_workspace(void)
|
|
{
|
|
WM_operatortype_append(WORKSPACE_OT_duplicate);
|
|
WM_operatortype_append(WORKSPACE_OT_delete);
|
|
WM_operatortype_append(WORKSPACE_OT_add);
|
|
WM_operatortype_append(WORKSPACE_OT_append_activate);
|
|
WM_operatortype_append(WORKSPACE_OT_reorder_to_back);
|
|
WM_operatortype_append(WORKSPACE_OT_reorder_to_front);
|
|
}
|
|
|
|
/** \} Workspace Operators */
|