A lot of files were missing copyright field in the header and
the Blender Foundation contributed to them in a sense of bug
fixing and general maintenance.
This change makes it explicit that those files are at least
partially copyrighted by the Blender Foundation.
Note that this does not make it so the Blender Foundation is
the only holder of the copyright in those files, and developers
who do not have a signed contract with the foundation still
hold the copyright as well.
Another aspect of this change is using SPDX format for the
header. We already used it for the license specification,
and now we state it for the copyright as well, following the
FAQ:
https://reuse.software/faq/
503 lines
14 KiB
C
503 lines
14 KiB
C
/* SPDX-FileCopyrightText: 2023 Blender Foundation
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
/** \file
|
|
* \ingroup wm
|
|
*
|
|
* Utilities to help define keymaps.
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include "DNA_object_types.h"
|
|
#include "DNA_space_types.h"
|
|
#include "DNA_userdef_types.h"
|
|
#include "DNA_windowmanager_types.h"
|
|
|
|
#include "BLI_listbase.h"
|
|
#include "BLI_utildefines.h"
|
|
|
|
#include "BKE_context.h"
|
|
|
|
#include "RNA_access.h"
|
|
|
|
#include "WM_api.h"
|
|
#include "WM_types.h"
|
|
|
|
/* menu wrapper for WM_keymap_add_item */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Wrappers for #WM_keymap_add_item
|
|
* \{ */
|
|
|
|
wmKeyMapItem *WM_keymap_add_menu(wmKeyMap *keymap,
|
|
const char *idname,
|
|
const KeyMapItem_Params *params)
|
|
{
|
|
wmKeyMapItem *kmi = WM_keymap_add_item(keymap, "WM_OT_call_menu", params);
|
|
RNA_string_set(kmi->ptr, "name", idname);
|
|
return kmi;
|
|
}
|
|
|
|
wmKeyMapItem *WM_keymap_add_menu_pie(wmKeyMap *keymap,
|
|
const char *idname,
|
|
const KeyMapItem_Params *params)
|
|
{
|
|
wmKeyMapItem *kmi = WM_keymap_add_item(keymap, "WM_OT_call_menu_pie", params);
|
|
RNA_string_set(kmi->ptr, "name", idname);
|
|
return kmi;
|
|
}
|
|
|
|
wmKeyMapItem *WM_keymap_add_panel(wmKeyMap *keymap,
|
|
const char *idname,
|
|
const KeyMapItem_Params *params)
|
|
{
|
|
wmKeyMapItem *kmi = WM_keymap_add_item(keymap, "WM_OT_call_panel", params);
|
|
RNA_string_set(kmi->ptr, "name", idname);
|
|
/* TODO: we might want to disable this. */
|
|
RNA_boolean_set(kmi->ptr, "keep_open", false);
|
|
return kmi;
|
|
}
|
|
|
|
wmKeyMapItem *WM_keymap_add_tool(wmKeyMap *keymap,
|
|
const char *idname,
|
|
const KeyMapItem_Params *params)
|
|
{
|
|
wmKeyMapItem *kmi = WM_keymap_add_item(keymap, "WM_OT_tool_set_by_id", params);
|
|
RNA_string_set(kmi->ptr, "name", idname);
|
|
return kmi;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Introspection
|
|
* \{ */
|
|
|
|
wmKeyMap *WM_keymap_guess_from_context(const bContext *C)
|
|
{
|
|
SpaceLink *sl = CTX_wm_space_data(C);
|
|
const char *km_id = NULL;
|
|
if (sl->spacetype == SPACE_VIEW3D) {
|
|
const enum eContextObjectMode mode = CTX_data_mode_enum(C);
|
|
switch (mode) {
|
|
case CTX_MODE_EDIT_MESH:
|
|
km_id = "Mesh";
|
|
break;
|
|
case CTX_MODE_EDIT_CURVE:
|
|
km_id = "Curve";
|
|
break;
|
|
case CTX_MODE_EDIT_CURVES:
|
|
km_id = "Curves";
|
|
break;
|
|
case CTX_MODE_EDIT_SURFACE:
|
|
km_id = "Curve";
|
|
break;
|
|
case CTX_MODE_EDIT_TEXT:
|
|
km_id = "Font";
|
|
break;
|
|
case CTX_MODE_EDIT_ARMATURE:
|
|
km_id = "Armature";
|
|
break;
|
|
case CTX_MODE_EDIT_METABALL:
|
|
km_id = "Metaball";
|
|
break;
|
|
case CTX_MODE_EDIT_LATTICE:
|
|
km_id = "Lattice";
|
|
break;
|
|
case CTX_MODE_POSE:
|
|
km_id = "Pose";
|
|
break;
|
|
case CTX_MODE_SCULPT:
|
|
km_id = "Sculpt";
|
|
break;
|
|
case CTX_MODE_PAINT_WEIGHT:
|
|
km_id = "Weight Paint";
|
|
break;
|
|
case CTX_MODE_PAINT_VERTEX:
|
|
km_id = "Vertex Paint";
|
|
break;
|
|
case CTX_MODE_PAINT_TEXTURE:
|
|
km_id = "Image Paint";
|
|
break;
|
|
case CTX_MODE_PARTICLE:
|
|
km_id = "Particle";
|
|
break;
|
|
case CTX_MODE_OBJECT:
|
|
km_id = "Object Mode";
|
|
break;
|
|
case CTX_MODE_PAINT_GPENCIL:
|
|
km_id = "Grease Pencil Stroke Paint Mode";
|
|
break;
|
|
case CTX_MODE_EDIT_GPENCIL:
|
|
km_id = "Grease Pencil Stroke Edit Mode";
|
|
break;
|
|
case CTX_MODE_SCULPT_GPENCIL:
|
|
km_id = "Grease Pencil Stroke Sculpt Mode";
|
|
break;
|
|
case CTX_MODE_WEIGHT_GPENCIL:
|
|
km_id = "Grease Pencil Stroke Weight Mode";
|
|
break;
|
|
case CTX_MODE_VERTEX_GPENCIL:
|
|
km_id = "Grease Pencil Stroke Vertex Mode";
|
|
break;
|
|
case CTX_MODE_SCULPT_CURVES:
|
|
km_id = "Sculpt Curves";
|
|
break;
|
|
}
|
|
}
|
|
else if (sl->spacetype == SPACE_IMAGE) {
|
|
const SpaceImage *sima = (SpaceImage *)sl;
|
|
const eSpaceImage_Mode mode = sima->mode;
|
|
switch (mode) {
|
|
case SI_MODE_VIEW:
|
|
km_id = "Image";
|
|
break;
|
|
case SI_MODE_PAINT:
|
|
km_id = "Image Paint";
|
|
break;
|
|
case SI_MODE_MASK:
|
|
km_id = "Mask Editing";
|
|
break;
|
|
case SI_MODE_UV:
|
|
km_id = "UV Editor";
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
return NULL;
|
|
}
|
|
|
|
wmKeyMap *km = WM_keymap_find_all(CTX_wm_manager(C), km_id, 0, 0);
|
|
BLI_assert(km);
|
|
return km;
|
|
}
|
|
|
|
wmKeyMap *WM_keymap_guess_opname(const bContext *C, const char *opname)
|
|
{
|
|
/* Op types purposely skipped for now:
|
|
* BRUSH_OT
|
|
* BOID_OT
|
|
* BUTTONS_OT
|
|
* CONSTRAINT_OT
|
|
* PAINT_OT
|
|
* ED_OT
|
|
* FLUID_OT
|
|
* TEXTURE_OT
|
|
* WORLD_OT
|
|
*/
|
|
|
|
wmKeyMap *km = NULL;
|
|
wmWindowManager *wm = CTX_wm_manager(C);
|
|
SpaceLink *sl = CTX_wm_space_data(C);
|
|
|
|
/* Window */
|
|
if (STRPREFIX(opname, "WM_OT") || STRPREFIX(opname, "ED_OT_undo")) {
|
|
if (STREQ(opname, "WM_OT_tool_set_by_id")) {
|
|
km = WM_keymap_guess_from_context(C);
|
|
}
|
|
|
|
if (km == NULL) {
|
|
km = WM_keymap_find_all(wm, "Window", 0, 0);
|
|
}
|
|
}
|
|
/* Screen & Render */
|
|
else if (STRPREFIX(opname, "SCREEN_OT") || STRPREFIX(opname, "RENDER_OT") ||
|
|
STRPREFIX(opname, "SOUND_OT") || STRPREFIX(opname, "SCENE_OT"))
|
|
{
|
|
km = WM_keymap_find_all(wm, "Screen", 0, 0);
|
|
}
|
|
/* Grease Pencil */
|
|
else if (STRPREFIX(opname, "GPENCIL_OT")) {
|
|
km = WM_keymap_find_all(wm, "Grease Pencil", 0, 0);
|
|
}
|
|
/* Markers */
|
|
else if (STRPREFIX(opname, "MARKER_OT")) {
|
|
km = WM_keymap_find_all(wm, "Markers", 0, 0);
|
|
}
|
|
/* Import/Export */
|
|
else if (STRPREFIX(opname, "IMPORT_") || STRPREFIX(opname, "EXPORT_")) {
|
|
km = WM_keymap_find_all(wm, "Window", 0, 0);
|
|
}
|
|
|
|
/* 3D View */
|
|
else if (STRPREFIX(opname, "VIEW3D_OT")) {
|
|
km = WM_keymap_find_all(wm, "3D View", sl->spacetype, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "OBJECT_OT")) {
|
|
/* exception, this needs to work outside object mode too */
|
|
if (STRPREFIX(opname, "OBJECT_OT_mode_set")) {
|
|
km = WM_keymap_find_all(wm, "Object Non-modal", 0, 0);
|
|
}
|
|
else {
|
|
km = WM_keymap_find_all(wm, "Object Mode", 0, 0);
|
|
}
|
|
}
|
|
/* Object mode related */
|
|
else if (STRPREFIX(opname, "GROUP_OT") || STRPREFIX(opname, "MATERIAL_OT") ||
|
|
STRPREFIX(opname, "PTCACHE_OT") || STRPREFIX(opname, "RIGIDBODY_OT"))
|
|
{
|
|
km = WM_keymap_find_all(wm, "Object Mode", 0, 0);
|
|
}
|
|
|
|
/* Editing Modes */
|
|
else if (STRPREFIX(opname, "MESH_OT")) {
|
|
km = WM_keymap_find_all(wm, "Mesh", 0, 0);
|
|
|
|
/* some mesh operators are active in object mode too, like add-prim */
|
|
if (km && !WM_keymap_poll((bContext *)C, km)) {
|
|
km = WM_keymap_find_all(wm, "Object Mode", 0, 0);
|
|
}
|
|
}
|
|
else if (STRPREFIX(opname, "CURVE_OT") || STRPREFIX(opname, "SURFACE_OT")) {
|
|
km = WM_keymap_find_all(wm, "Curve", 0, 0);
|
|
|
|
/* some curve operators are active in object mode too, like add-prim */
|
|
if (km && !WM_keymap_poll((bContext *)C, km)) {
|
|
km = WM_keymap_find_all(wm, "Object Mode", 0, 0);
|
|
}
|
|
}
|
|
else if (STRPREFIX(opname, "ARMATURE_OT") || STRPREFIX(opname, "SKETCH_OT")) {
|
|
km = WM_keymap_find_all(wm, "Armature", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "POSE_OT") || STRPREFIX(opname, "POSELIB_OT")) {
|
|
km = WM_keymap_find_all(wm, "Pose", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "SCULPT_OT")) {
|
|
switch (CTX_data_mode_enum(C)) {
|
|
case CTX_MODE_SCULPT:
|
|
km = WM_keymap_find_all(wm, "Sculpt", 0, 0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
else if (STRPREFIX(opname, "CURVES_OT")) {
|
|
km = WM_keymap_find_all(wm, "Curves", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "SCULPT_CURVES_OT")) {
|
|
km = WM_keymap_find_all(wm, "Sculpt Curves", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "MBALL_OT")) {
|
|
km = WM_keymap_find_all(wm, "Metaball", 0, 0);
|
|
|
|
/* Some meta-ball operators are active in object mode too, like add-primitive. */
|
|
if (km && !WM_keymap_poll((bContext *)C, km)) {
|
|
km = WM_keymap_find_all(wm, "Object Mode", 0, 0);
|
|
}
|
|
}
|
|
else if (STRPREFIX(opname, "LATTICE_OT")) {
|
|
km = WM_keymap_find_all(wm, "Lattice", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "PARTICLE_OT")) {
|
|
km = WM_keymap_find_all(wm, "Particle", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "FONT_OT")) {
|
|
km = WM_keymap_find_all(wm, "Font", 0, 0);
|
|
}
|
|
/* Paint Face Mask */
|
|
else if (STRPREFIX(opname, "PAINT_OT_face_select")) {
|
|
km = WM_keymap_find_all(wm, "Paint Face Mask (Weight, Vertex, Texture)", 0, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "PAINT_OT")) {
|
|
/* check for relevant mode */
|
|
switch (CTX_data_mode_enum(C)) {
|
|
case CTX_MODE_PAINT_WEIGHT:
|
|
km = WM_keymap_find_all(wm, "Weight Paint", 0, 0);
|
|
break;
|
|
case CTX_MODE_PAINT_VERTEX:
|
|
km = WM_keymap_find_all(wm, "Vertex Paint", 0, 0);
|
|
break;
|
|
case CTX_MODE_PAINT_TEXTURE:
|
|
km = WM_keymap_find_all(wm, "Image Paint", 0, 0);
|
|
break;
|
|
case CTX_MODE_SCULPT:
|
|
km = WM_keymap_find_all(wm, "Sculpt", 0, 0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
/* General 2D View, not bound to a specific spacetype. */
|
|
else if (STRPREFIX(opname, "VIEW2D_OT")) {
|
|
km = WM_keymap_find_all(wm, "View2D", 0, 0);
|
|
}
|
|
/* Image Editor */
|
|
else if (STRPREFIX(opname, "IMAGE_OT")) {
|
|
km = WM_keymap_find_all(wm, "Image", sl->spacetype, 0);
|
|
}
|
|
/* Clip Editor */
|
|
else if (STRPREFIX(opname, "CLIP_OT")) {
|
|
km = WM_keymap_find_all(wm, "Clip", sl->spacetype, 0);
|
|
}
|
|
else if (STRPREFIX(opname, "MASK_OT")) {
|
|
km = WM_keymap_find_all(wm, "Mask Editing", 0, 0);
|
|
}
|
|
/* UV Editor */
|
|
else if (STRPREFIX(opname, "UV_OT")) {
|
|
/* Hack to allow using UV unwrapping ops from 3DView/editmode.
|
|
* Mesh keymap is probably not ideal, but best place I could find to put those. */
|
|
if (sl->spacetype == SPACE_VIEW3D) {
|
|
km = WM_keymap_find_all(wm, "Mesh", 0, 0);
|
|
if (km && !WM_keymap_poll((bContext *)C, km)) {
|
|
km = NULL;
|
|
}
|
|
}
|
|
if (!km) {
|
|
km = WM_keymap_find_all(wm, "UV Editor", 0, 0);
|
|
}
|
|
}
|
|
/* Node Editor */
|
|
else if (STRPREFIX(opname, "NODE_OT")) {
|
|
km = WM_keymap_find_all(wm, "Node Editor", sl->spacetype, 0);
|
|
}
|
|
/* Animation Editor Channels */
|
|
else if (STRPREFIX(opname, "ANIM_OT_channels")) {
|
|
km = WM_keymap_find_all(wm, "Animation Channels", 0, 0);
|
|
}
|
|
/* Animation Generic - after channels */
|
|
else if (STRPREFIX(opname, "ANIM_OT")) {
|
|
if (sl->spacetype == SPACE_VIEW3D) {
|
|
switch (CTX_data_mode_enum(C)) {
|
|
case CTX_MODE_OBJECT:
|
|
km = WM_keymap_find_all(wm, "Object Mode", 0, 0);
|
|
break;
|
|
case CTX_MODE_POSE:
|
|
km = WM_keymap_find_all(wm, "Pose", 0, 0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (km && !WM_keymap_poll((bContext *)C, km)) {
|
|
km = NULL;
|
|
}
|
|
}
|
|
|
|
if (!km) {
|
|
km = WM_keymap_find_all(wm, "Animation", 0, 0);
|
|
}
|
|
}
|
|
/* Graph Editor */
|
|
else if (STRPREFIX(opname, "GRAPH_OT")) {
|
|
km = WM_keymap_find_all(wm, "Graph Editor", sl->spacetype, 0);
|
|
}
|
|
/* Dopesheet Editor */
|
|
else if (STRPREFIX(opname, "ACTION_OT")) {
|
|
km = WM_keymap_find_all(wm, "Dopesheet", sl->spacetype, 0);
|
|
}
|
|
/* NLA Editor */
|
|
else if (STRPREFIX(opname, "NLA_OT")) {
|
|
km = WM_keymap_find_all(wm, "NLA Editor", sl->spacetype, 0);
|
|
}
|
|
/* Script */
|
|
else if (STRPREFIX(opname, "SCRIPT_OT")) {
|
|
km = WM_keymap_find_all(wm, "Script", sl->spacetype, 0);
|
|
}
|
|
/* Text */
|
|
else if (STRPREFIX(opname, "TEXT_OT")) {
|
|
km = WM_keymap_find_all(wm, "Text", sl->spacetype, 0);
|
|
}
|
|
/* Sequencer */
|
|
else if (STRPREFIX(opname, "SEQUENCER_OT")) {
|
|
km = WM_keymap_find_all(wm, "Sequencer", sl->spacetype, 0);
|
|
}
|
|
/* Console */
|
|
else if (STRPREFIX(opname, "CONSOLE_OT")) {
|
|
km = WM_keymap_find_all(wm, "Console", sl->spacetype, 0);
|
|
}
|
|
/* Console */
|
|
else if (STRPREFIX(opname, "INFO_OT")) {
|
|
km = WM_keymap_find_all(wm, "Info", sl->spacetype, 0);
|
|
}
|
|
/* File browser */
|
|
else if (STRPREFIX(opname, "FILE_OT")) {
|
|
km = WM_keymap_find_all(wm, "File Browser", sl->spacetype, 0);
|
|
}
|
|
/* Logic Editor */
|
|
else if (STRPREFIX(opname, "LOGIC_OT")) {
|
|
km = WM_keymap_find_all(wm, "Logic Editor", sl->spacetype, 0);
|
|
}
|
|
/* Outliner */
|
|
else if (STRPREFIX(opname, "OUTLINER_OT")) {
|
|
km = WM_keymap_find_all(wm, "Outliner", sl->spacetype, 0);
|
|
}
|
|
/* Transform */
|
|
else if (STRPREFIX(opname, "TRANSFORM_OT")) {
|
|
/* check for relevant editor */
|
|
switch (sl->spacetype) {
|
|
case SPACE_VIEW3D:
|
|
km = WM_keymap_find_all(wm, "3D View", sl->spacetype, 0);
|
|
break;
|
|
case SPACE_GRAPH:
|
|
km = WM_keymap_find_all(wm, "Graph Editor", sl->spacetype, 0);
|
|
break;
|
|
case SPACE_ACTION:
|
|
km = WM_keymap_find_all(wm, "Dopesheet", sl->spacetype, 0);
|
|
break;
|
|
case SPACE_NLA:
|
|
km = WM_keymap_find_all(wm, "NLA Editor", sl->spacetype, 0);
|
|
break;
|
|
case SPACE_IMAGE:
|
|
km = WM_keymap_find_all(wm, "UV Editor", 0, 0);
|
|
break;
|
|
case SPACE_NODE:
|
|
km = WM_keymap_find_all(wm, "Node Editor", sl->spacetype, 0);
|
|
break;
|
|
case SPACE_SEQ:
|
|
km = WM_keymap_find_all(wm, "Sequencer", sl->spacetype, 0);
|
|
break;
|
|
}
|
|
}
|
|
/* User Interface */
|
|
else if (STRPREFIX(opname, "UI_OT")) {
|
|
km = WM_keymap_find_all(wm, "User Interface", 0, 0);
|
|
}
|
|
|
|
return km;
|
|
}
|
|
|
|
static bool wm_keymap_item_uses_modifier(const wmKeyMapItem *kmi, const int event_modifier)
|
|
{
|
|
if (kmi->ctrl != KM_ANY) {
|
|
if ((kmi->ctrl == KM_NOTHING) != ((event_modifier & KM_CTRL) == 0)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (kmi->alt != KM_ANY) {
|
|
if ((kmi->alt == KM_NOTHING) != ((event_modifier & KM_ALT) == 0)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (kmi->shift != KM_ANY) {
|
|
if ((kmi->shift == KM_NOTHING) != ((event_modifier & KM_SHIFT) == 0)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (kmi->oskey != KM_ANY) {
|
|
if ((kmi->oskey == KM_NOTHING) != ((event_modifier & KM_OSKEY) == 0)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool WM_keymap_uses_event_modifier(const wmKeyMap *keymap, const int event_modifier)
|
|
{
|
|
LISTBASE_FOREACH (const wmKeyMapItem *, kmi, &keymap->items) {
|
|
if ((kmi->flag & KMI_INACTIVE) == 0) {
|
|
if (wm_keymap_item_uses_modifier(kmi, event_modifier)) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void WM_keymap_fix_linking(void) {}
|
|
|
|
/** \} */
|