home · contact · privacy
Fixed bug that led to endless loop in nearest_enemy_dir().
[plomrogue] / src / control.c
index c7c7b1b6fb4fff949e9e8a24ec7410c0a826a18a..1e9265e73074da48e2f884e93fa2ee94b00c1ed9 100644 (file)
 /* control.c */
 
 #include "control.h"
-#include <stdint.h> /* for uint8_t */
+#include <stdint.h> /* for uint8_t, uint16_t */
 #include "windows.h" /* for cycle_active_win(), shift_active_win(), struct Win,
                       *  struct WinMeta
                       */
 #include "keybindings.h" /* for get_keycode_to_action(), mod_selected_keyb(),
-                          * move_keyb_mod_selection()
+                          * move_keyb_mod_selection(), get_func_to_keycode()
                           */
-#include "map.h" /* for map_scroll() */
+#include "map.h" /* for map_scroll(), map_center() */
 #include "main.h" /* for world global */
-#include "rexit.h" /* for exit_err() */
-#include "wincontrol.h" /* for scroll_pad(), toggle_window(),
+#include "wincontrol.h" /* for struct WinConf, scroll_pad(), toggle_window(),
                          * growshrink_active_window(), toggle_winconfig(),
-                         * toggle_win_height_type(), toggle_win_width_type()
+                         * toggle_win_size_type()
                          */
-#include "map_object_actions.h" /* for player_wait(), move_player(),
-                                 * player_drop(), player_pick()
-                                 */
+#include "map_object_actions.h" /* for get_moa_id_by_name() */
 #include "command_db.h" /* for is_command_id_shortdsc() */
-#include "misc.h" /* for load_interface_conf(), unload_interface_conf(),
-                   * save_interface_conf(), nav_inventory()
+#include "misc.h" /* for reload_interface_conf(), save_interface_conf(),
+                   * nav_inventory(), turn_over()
                    */
-#include "yx_uint16.h" /* for dir enum */
 #include "map_objects.h" /* for get_player() */
 
 
 
-extern uint16_t get_available_keycode_to_action(char * name)
+/* If "cmd" matches "match" in get_available_keycode_to_action(), execute "f"
+ * with provided char arguments and return 1; else only return 0.
+ */
+static uint8_t try_cmd_0args(int cmd, char * match, void (* f) ());
+static uint8_t try_cmd_1args(int cmd, char * match, void (* f) (char), char c);
+static uint8_t try_cmd_2args(int cmd, char * match,
+                             void (* f) (char, char), char c1, char c2);
+
+/* If "action" is id of command named "match", set player's .arg and .command
+ * and call turn_over().
+ */
+static uint8_t try_player_cmd(int action, char * match, char * command,
+                              uint8_t arg);
+
+/* Return pointer to global keybindings or to keybindings for wingeometry config
+ * (c = "g") or winkeys config (c = "k") or active window's keybindings ("w").
+ */
+static struct KeyBiData * select_keybidata_pointer(char c);
+
+/* Wrappers to make some functions compatible to try_cmd_* single char args. */
+static void wrap_mod_selected_keyb(char c);
+static void wrap_mv_kb_mod(char c1, char c2);
+
+
+
+static uint8_t try_cmd_0args(int cmd, char * match, void (* f) ())
 {
-    uint16_t keycode = get_keycode_to_action(world.kb_global.kbs, name);
-    if (0 != keycode || 0 == world.wmeta->active)
+    if (cmd == get_available_keycode_to_action(match))
     {
-        return keycode;
-    }
-    struct WinConf * wc = get_winconf_by_win(world.wmeta->active);
-    if (0 == wc->view)
-    {
-        keycode = get_keycode_to_action(wc->kb.kbs, name);
-    }
-    else if (1 == wc->view)
-    {
-        keycode = get_keycode_to_action(world.kb_wingeom.kbs, name);
-    }
-    else if (2 == wc->view)
-    {
-        keycode = get_keycode_to_action(world.kb_winkeys.kbs, name);
+        f();
+        return 1;
     }
-    return keycode;
+    return 0;
 }
 
 
 
-extern void record_control(int action)
+static uint8_t try_cmd_1args(int cmd, char * match, void (* f) (char), char c)
 {
-    if      (is_command_id_shortdsc(action, "wait"))
+    if (cmd == get_available_keycode_to_action(match))
     {
-        player_wait();
-    }
-    else if (is_command_id_shortdsc(action, "player_u"))
-    {
-        move_player(NORTH);
+        f(c);
+        return 1;
     }
-    else if (is_command_id_shortdsc(action, "player_r"))
+    return 0;
+}
+
+
+
+static uint8_t try_cmd_2args(int cmd, char * match,
+                             void (* f) (char, char), char c1, char c2)
+{
+    if (cmd == get_available_keycode_to_action(match))
     {
-        move_player(EAST);
+        f(c1, c2);
+        return 1;
     }
-    else if (is_command_id_shortdsc(action, "player_d"))
+    return 0;
+}
+
+
+
+static uint8_t try_player_cmd(int action, char * match, char * command,
+                              uint8_t arg)
+{
+    if (is_command_id_shortdsc(action, match))
     {
-        move_player(SOUTH);
+        struct MapObj * player = get_player();
+        player->arg = arg;
+        player->command = get_moa_id_by_name(command);
+        turn_over(get_command_id(match));
+        return 1;
     }
-    else if (is_command_id_shortdsc(action, "player_l"))
+    return 0;
+}
+
+
+
+static struct KeyBiData * select_keybidata_pointer(char c)
+{
+    struct KeyBiData * kbd;
+    kbd = &world.kb_global;
+    if      ('g' == c)
     {
-        move_player(WEST);
+        kbd = &world.kb_wingeom;
     }
-    else if (is_command_id_shortdsc(action, "drop"))
+    else if ('k' == c)
     {
-        player_drop();
+        kbd = &world.kb_winkeys;
     }
-    else if (is_command_id_shortdsc(action, "pick"))
+    else if ('w' == c)
     {
-        player_pick();
+        struct WinConf * wc = get_winconf_by_win(world.wmeta->active);
+        kbd = &wc->kb;
     }
+    return kbd;
 }
 
 
 
-extern uint8_t player_control(int key)
+static void wrap_mod_selected_keyb(char c)
 {
-    if      (key == get_available_keycode_to_action("wait"))
-    {
-        player_wait();
-    }
-    else if (key == get_available_keycode_to_action("player_u"))
+        mod_selected_keyb(select_keybidata_pointer(c));
+}
+
+
+
+static void wrap_mv_kb_mod(char c1, char c2)
+{
+        move_keyb_mod_selection(select_keybidata_pointer(c1), c2);
+}
+
+
+
+extern uint16_t get_available_keycode_to_action(char * name)
+{
+    uint16_t keycode = get_keycode_to_action(world.kb_global.kbs, name);
+    if (0 != keycode || 0 == world.wmeta->active)
     {
-        move_player(NORTH);
+        return keycode;
     }
-    else if (key == get_available_keycode_to_action("player_r"))
+    struct WinConf * wc = get_winconf_by_win(world.wmeta->active);
+    if (0 == wc->view)
     {
-        move_player(EAST);
+        keycode = get_keycode_to_action(wc->kb.kbs, name);
     }
-    else if (key == get_available_keycode_to_action("player_d"))
+    else if (1 == wc->view)
     {
-        move_player(SOUTH);
+        keycode = get_keycode_to_action(world.kb_wingeom.kbs, name);
     }
-    else if (key == get_available_keycode_to_action("player_l"))
+    else if (2 == wc->view)
     {
-        move_player(WEST);
+        keycode = get_keycode_to_action(world.kb_winkeys.kbs, name);
     }
-    else if (key == get_available_keycode_to_action("drop"))
+    return keycode;
+}
+
+
+
+extern uint8_t player_control_by_key(int key)
+{
+    char * action_name = get_func_to_keycode(world.kb_global.kbs, key);
+    if (NULL == action_name && 0 != world.wmeta->active)
     {
-        player_drop();
+        struct WinConf * wc = get_winconf_by_win(world.wmeta->active);
+        action_name = get_func_to_keycode(wc->kb.kbs, key);
     }
-    else if (key == get_available_keycode_to_action("pick"))
+    if (NULL != action_name)
     {
-        player_pick();
+        uint8_t action_id = get_command_id(action_name);
+        return player_control_by_id(action_id);
     }
-    else
+    return 0;
+}
+
+
+
+
+extern uint8_t player_control_by_id(int action)
+{
+    if (   try_player_cmd(action, "wait", "wait", 0)
+        || try_player_cmd(action, "drop", "drop", world.inventory_sel)
+        || try_player_cmd(action, "pick", "pick_up", 0)
+        || try_player_cmd(action, "use", "use", world.inventory_sel)
+        || try_player_cmd(action, "player_u", "move", 'N')
+        || try_player_cmd(action, "player_d", "move", 'S')
+        || try_player_cmd(action, "player_r", "move", 'E')
+        || try_player_cmd(action, "player_l", "move", 'W'))
     {
-        return 0;
+        return 1;
     }
-    return 1;
+    return 0;
 }
 
 
 
 extern uint8_t wingeom_control(int key)
 {
-    char * err_shift  = "Trouble with shift_active_win() in wingeom_control().";
-    char * err_resize = "Trouble with growshrink_active_window() in "
-                        "wingeom_control().";
-    if      (key == get_available_keycode_to_action("to_height_t"))
-    {
-        toggle_win_height_type(world.wmeta->active);
-    }
-    else if (key == get_available_keycode_to_action("to_width_t"))
-    {
-        toggle_win_width_type(world.wmeta->active);
-    }
-    else if (key == get_available_keycode_to_action("grow_h"))
+    if (   try_cmd_1args(key, "to_height_t", toggle_win_size_type, 'y')
+        || try_cmd_1args(key, "to_width_t", toggle_win_size_type, 'x')
+        || try_cmd_1args(key, "grow_h", growshrink_active_window, '*')
+        || try_cmd_1args(key, "shri_h", growshrink_active_window, '_')
+        || try_cmd_1args(key, "grow_v", growshrink_active_window, '+')
+        || try_cmd_1args(key, "shri_v", growshrink_active_window, '-')
+        || try_cmd_1args(key, "shift_f", shift_active_win, 'f')
+        || try_cmd_1args(key, "shift_b", shift_active_win, 'b'))
     {
-        exit_err(growshrink_active_window('*'), err_resize);
-    }
-    else if (key == get_available_keycode_to_action("shri_h"))
-    {
-        exit_err(growshrink_active_window('_'), err_resize);
-    }
-    else if (key == get_available_keycode_to_action("grow_v"))
-    {
-        exit_err(growshrink_active_window('+'), err_resize);
-    }
-    else if (key == get_available_keycode_to_action("shri_v"))
-    {
-        exit_err(growshrink_active_window('-'), err_resize);
-    }
-    else if (key == get_available_keycode_to_action("shift_f"))
-    {
-        exit_err(shift_active_win(world.wmeta, 'f'), err_shift);
-    }
-    else if (key == get_available_keycode_to_action("shift_b"))
-    {
-        exit_err(shift_active_win(world.wmeta, 'b'), err_shift);
-    }
-    else
-    {
-        return 0;
+        return 1;
     }
-    return 1;
+    return 0;
 }
 
 
 
 extern uint8_t winkeyb_control(int key)
 {
-    struct WinConf * wc = get_winconf_by_win(world.wmeta->active);
-    if      (key == get_available_keycode_to_action("w_keys_u"))
+    if (   try_cmd_1args(key, "w_keys_m", wrap_mod_selected_keyb, 'w')
+        || try_cmd_2args(key, "w_keys_u", wrap_mv_kb_mod, 'w', 'u')
+        || try_cmd_2args(key, "w_keys_d", wrap_mv_kb_mod, 'w', 'd'))
     {
-        move_keyb_mod_selection(&wc->kb, 'u');
-    }
-    else if (key == get_available_keycode_to_action("w_keys_d"))
-    {
-        move_keyb_mod_selection(&wc->kb, 'd');
-    }
-    else if (key == get_available_keycode_to_action("w_keys_m"))
-    {
-        mod_selected_keyb(&wc->kb);
-    }
-    else
-    {
-        return 0;
+        return 1;
     }
-    return 1;
+    return 0;
 }
 
 
 
 extern uint8_t meta_control(int key)
 {
-    struct WinMeta * win_meta = world.wmeta;
-    struct Win * win_map      = get_win_by_id('m');
-    char * err_toggle = "Trouble with toggle_window() in meta_control().";
-    if      (key == get_available_keycode_to_action("quit"))
-    {
-        return 1;
-    }
-    else if (key == get_available_keycode_to_action("winconf"))
-    {
-        toggle_winconfig(world.wmeta->active);
-    }
-    else if (key == get_available_keycode_to_action("cyc_win_f"))
-    {
-        cycle_active_win(world.wmeta, 'f');
-    }
-    else if (key == get_available_keycode_to_action("cyc_win_b"))
-    {
-        cycle_active_win(world.wmeta, 'b');
-    }
-    else if (key == get_available_keycode_to_action("scrl_r"))
-    {
-        scroll_pad(win_meta, '+');
-    }
-    else if (key == get_available_keycode_to_action("scrl_l"))
-    {
-        scroll_pad(win_meta, '-');
-    }
-    else if (key == get_available_keycode_to_action("to_a_keywin"))
-    {
-       exit_err(toggle_window(win_meta, get_win_by_id('k')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_g_keywin"))
-    {
-        exit_err(toggle_window(win_meta, get_win_by_id('0')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_wg_keywin"))
-    {
-        exit_err(toggle_window(win_meta, get_win_by_id('1')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_wk_keywin"))
-    {
-        exit_err(toggle_window(win_meta, get_win_by_id('2')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_mapwin"))
-    {
-        exit_err(toggle_window(win_meta, win_map), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_infowin"))
-    {
-        exit_err(toggle_window(win_meta, get_win_by_id('i')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_inv"))
-    {
-        exit_err(toggle_window(win_meta, get_win_by_id('c')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("to_logwin"))
-    {
-        exit_err(toggle_window(win_meta, get_win_by_id('l')), err_toggle);
-    }
-    else if (key == get_available_keycode_to_action("save_conf"))
-    {
-        save_interface_conf();
-    }
-    else if (key == get_available_keycode_to_action("g_keys_u"))
-    {
-        move_keyb_mod_selection(&world.kb_global, 'u');
-    }
-    else if (key == get_available_keycode_to_action("g_keys_d"))
-    {
-        move_keyb_mod_selection(&world.kb_global, 'd');
-    }
-    else if (key == get_available_keycode_to_action("g_keys_m"))
-    {
-        mod_selected_keyb(&world.kb_global);
-    }
-    else if (key == get_available_keycode_to_action("wg_keys_u"))
-    {
-        move_keyb_mod_selection(&world.kb_wingeom, 'u');
-    }
-    else if (key == get_available_keycode_to_action("wg_keys_d"))
-    {
-        move_keyb_mod_selection(&world.kb_wingeom, 'd');
-    }
-    else if (key == get_available_keycode_to_action("wg_keys_m"))
-    {
-        mod_selected_keyb(&world.kb_wingeom);
-    }
-    else if (key == get_available_keycode_to_action("wk_keys_u"))
-    {
-        move_keyb_mod_selection(&world.kb_winkeys, 'u');
-    }
-    else if (key == get_available_keycode_to_action("wk_keys_d"))
-    {
-        move_keyb_mod_selection(&world.kb_winkeys, 'd');
-    }
-    else if (key == get_available_keycode_to_action("wk_keys_m"))
-    {
-        mod_selected_keyb(&world.kb_winkeys);
-    }
-    else if (key == get_available_keycode_to_action("map_u"))
-    {
-        map_scroll(win_map, world.map->size, NORTH);
-    }
-    else if (key == get_available_keycode_to_action("map_d"))
-    {
-        map_scroll(win_map, world.map->size, SOUTH);
-    }
-    else if (key == get_available_keycode_to_action("map_r"))
-    {
-        map_scroll(win_map, world.map->size, EAST);
-    }
-    else if (key == get_available_keycode_to_action("map_l"))
-    {
-        map_scroll(win_map, world.map->size, WEST);
-    }
-    else if (key == get_available_keycode_to_action("map_c"))
-    {
-        struct MapObj * player = get_player();
-        win_map->center = player->pos;
-    }
-    else if (key == get_available_keycode_to_action("inv_u"))
-    {
-        nav_inventory('u');
-    }
-    else if (key == get_available_keycode_to_action("inv_d"))
-    {
-        nav_inventory('d');
-    }
-    else if (key == get_available_keycode_to_action("reload_conf"))
-    {
-        unload_interface_conf();
-        load_interface_conf();
-    }
-    else if (key == get_available_keycode_to_action("winconf"))
-    {
-        toggle_winconfig(world.wmeta->active);
-    }
-    return 0;
+    uint8_t ret = (key == get_available_keycode_to_action("quit"));
+    if (   (0 == ret)
+        && (   try_cmd_0args(key, "winconf", toggle_winconfig)
+            || try_cmd_0args(key, "reload_conf", reload_interface_conf)
+            || try_cmd_0args(key, "save_conf", save_interface_conf)
+            || try_cmd_0args(key, "map_c", map_center)
+            || try_cmd_1args(key, "scrl_r", scroll_pad, '+')
+            || try_cmd_1args(key, "scrl_l", scroll_pad, '-')
+            || try_cmd_1args(key, "to_a_keywin", toggle_window, 'k')
+            || try_cmd_1args(key, "to_g_keywin", toggle_window, '0')
+            || try_cmd_1args(key, "to_wg_keywin", toggle_window, '1')
+            || try_cmd_1args(key, "to_wk_keywin", toggle_window, '2')
+            || try_cmd_1args(key, "to_mapwin", toggle_window, 'm')
+            || try_cmd_1args(key, "to_infowin", toggle_window, 'i')
+            || try_cmd_1args(key, "to_inv", toggle_window, 'c')
+            || try_cmd_1args(key, "to_logwin", toggle_window, 'l')
+            || try_cmd_1args(key, "cyc_win_f", cycle_active_win, 'f')
+            || try_cmd_1args(key, "cyc_win_b", cycle_active_win, 'b')
+            || try_cmd_1args(key, "g_keys_m", wrap_mod_selected_keyb, 'G')
+            || try_cmd_1args(key, "wg_keys_m", wrap_mod_selected_keyb, 'g')
+            || try_cmd_1args(key, "wk_keys_m", wrap_mod_selected_keyb, 'k')
+            || try_cmd_1args(key, "inv_u", nav_inventory, 'u')
+            || try_cmd_1args(key, "inv_d", nav_inventory, 'd')
+            || try_cmd_1args(key, "map_u", map_scroll, 'N')
+            || try_cmd_1args(key, "map_d", map_scroll, 'S')
+            || try_cmd_1args(key, "map_r", map_scroll, 'E')
+            || try_cmd_1args(key, "map_l", map_scroll, 'W')
+            || try_cmd_2args(key, "g_keys_u", wrap_mv_kb_mod, 'G', 'u')
+            || try_cmd_2args(key, "g_keys_d", wrap_mv_kb_mod, 'G', 'd')
+            || try_cmd_2args(key, "wg_keys_u", wrap_mv_kb_mod, 'g', 'u')
+            || try_cmd_2args(key, "wg_keys_d", wrap_mv_kb_mod, 'g', 'd')
+            || try_cmd_2args(key, "wk_keys_u", wrap_mv_kb_mod, 'k', 'u')
+            || try_cmd_2args(key, "wk_keys_d", wrap_mv_kb_mod, 'k', 'd')))
+    {
+        ;
+    }
+    return ret;
 }