2017-01-29 00:03:23 +00:00
|
|
|
/*
|
|
|
|
4coder_default_hooks.cpp - Sets up the hooks for the default framework.
|
|
|
|
|
|
|
|
TYPE: 'internal-for-default-system'
|
|
|
|
*/
|
|
|
|
|
|
|
|
// TOP
|
|
|
|
|
|
|
|
#if !defined(FCODER_DEFAULT_HOOKS_CPP)
|
|
|
|
#define FCODER_DEFAULT_HOOKS_CPP
|
|
|
|
|
|
|
|
#include "4coder_default_framework.h"
|
|
|
|
#include "4coder_helper/4coder_bind_helper.h"
|
2017-03-27 22:36:42 +00:00
|
|
|
#include "4coder_project_commands.cpp"
|
2017-01-29 00:03:23 +00:00
|
|
|
|
2017-06-12 17:40:54 +00:00
|
|
|
#include "languages/4coder_language_cpp.h"
|
2017-05-21 01:22:20 +00:00
|
|
|
#include "languages/4coder_language_rust.h"
|
|
|
|
#include "languages/4coder_language_cs.h"
|
|
|
|
#include "languages/4coder_language_java.h"
|
|
|
|
|
2017-06-23 23:07:18 +00:00
|
|
|
START_HOOK_SIG(default_start){
|
2017-01-29 00:03:23 +00:00
|
|
|
default_4coder_initialize(app);
|
2017-06-23 23:07:18 +00:00
|
|
|
default_4coder_side_by_side_panels(app, files, file_count);
|
2017-01-29 00:03:23 +00:00
|
|
|
|
2017-03-27 22:36:42 +00:00
|
|
|
if (automatically_load_project){
|
|
|
|
load_project(app);
|
|
|
|
}
|
|
|
|
|
2017-01-29 00:03:23 +00:00
|
|
|
// no meaning for return
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE(allen|a4.0.9): All command calls can now go through this hook
|
|
|
|
// If this hook is not implemented a default behavior of calling the
|
|
|
|
// command is used. It is important to note that paste_next does not
|
|
|
|
// work without this hook.
|
|
|
|
// NOTE(allen|a4.0.10): As of this version the word_complete command
|
|
|
|
// also relies on this particular command caller hook.
|
|
|
|
COMMAND_CALLER_HOOK(default_command_caller){
|
|
|
|
View_Summary view = get_active_view(app, AccessAll);
|
|
|
|
|
|
|
|
view_paste_index[view.view_id].next_rewrite = 0;
|
|
|
|
exec_command(app, cmd);
|
|
|
|
view_paste_index[view.view_id].rewrite = view_paste_index[view.view_id].next_rewrite;
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
HOOK_SIG(default_exit){
|
|
|
|
// if this returns zero it cancels the exit.
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
HOOK_SIG(default_view_adjust){
|
|
|
|
int32_t count = 0;
|
|
|
|
int32_t new_wrap_width = 0;
|
|
|
|
for (View_Summary view = get_view_first(app, AccessAll);
|
|
|
|
view.exists;
|
|
|
|
get_view_next(app, &view, AccessAll)){
|
|
|
|
new_wrap_width += view.view_region.x1 - view.view_region.x0;
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_wrap_width /= count;
|
|
|
|
new_wrap_width = (int32_t)(new_wrap_width * .9f);
|
|
|
|
|
|
|
|
int32_t new_min_base_width = (int32_t)(new_wrap_width * .77f);
|
|
|
|
if (automatically_adjust_wrapping){
|
|
|
|
adjust_all_buffer_wrap_widths(app, new_wrap_width, new_min_base_width);
|
|
|
|
default_wrap_width = new_wrap_width;
|
|
|
|
default_min_base_width = new_min_base_width;
|
|
|
|
}
|
|
|
|
|
|
|
|
// no meaning for return
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(allen): Eliminate this hook if you can.
|
|
|
|
OPEN_FILE_HOOK_SIG(default_file_settings){
|
|
|
|
// NOTE(allen|a4.0.8): The get_parameter_buffer was eliminated
|
|
|
|
// and instead the buffer is passed as an explicit parameter through
|
|
|
|
// the function call. That is where buffer_id comes from here.
|
2017-03-29 16:32:06 +00:00
|
|
|
Buffer_Summary buffer = get_buffer(app, buffer_id, AccessAll);
|
2017-01-29 00:03:23 +00:00
|
|
|
Assert(buffer.exists);
|
|
|
|
|
2017-03-29 16:32:06 +00:00
|
|
|
bool32 treat_as_code = false;
|
2017-04-15 21:47:23 +00:00
|
|
|
bool32 treat_as_todo = false;
|
2017-03-29 16:32:06 +00:00
|
|
|
bool32 wrap_lines = true;
|
2017-01-29 00:03:23 +00:00
|
|
|
|
2017-02-26 21:44:40 +00:00
|
|
|
int32_t extension_count = 0;
|
|
|
|
char **extension_list = get_current_code_extensions(&extension_count);
|
|
|
|
|
2017-05-21 01:22:20 +00:00
|
|
|
Parse_Context_ID parse_context_id = 0;
|
|
|
|
|
2017-02-12 06:01:01 +00:00
|
|
|
if (buffer.file_name != 0 && buffer.size < (16 << 20)){
|
2017-04-15 21:47:23 +00:00
|
|
|
String name = make_string(buffer.file_name, buffer.file_name_len);
|
|
|
|
String ext = file_extension(name);
|
2017-02-26 21:44:40 +00:00
|
|
|
for (int32_t i = 0; i < extension_count; ++i){
|
|
|
|
if (match(ext, extension_list[i])){
|
|
|
|
treat_as_code = true;
|
2017-05-21 01:22:20 +00:00
|
|
|
|
|
|
|
if (match(ext, "cs")){
|
|
|
|
if (parse_context_language_cs == 0){
|
|
|
|
init_language_cs(app);
|
|
|
|
}
|
|
|
|
parse_context_id = parse_context_language_cs;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (match(ext, "java")){
|
|
|
|
if (parse_context_language_java == 0){
|
|
|
|
init_language_java(app);
|
|
|
|
}
|
|
|
|
parse_context_id = parse_context_language_java;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (match(ext, "rs")){
|
|
|
|
if (parse_context_language_rust == 0){
|
|
|
|
init_language_rust(app);
|
|
|
|
}
|
|
|
|
parse_context_id = parse_context_language_rust;
|
|
|
|
}
|
|
|
|
|
2017-06-12 17:40:54 +00:00
|
|
|
if (match(ext, "cpp") || match(ext, "h") || match(ext, "c") || match(ext, "hpp") || match(ext, "cc")){
|
|
|
|
if (parse_context_language_cpp == 0){
|
|
|
|
init_language_cpp(app);
|
|
|
|
}
|
|
|
|
parse_context_id = parse_context_language_cpp;
|
|
|
|
}
|
|
|
|
|
2017-02-26 21:44:40 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-01-29 00:03:23 +00:00
|
|
|
}
|
2017-04-15 21:47:23 +00:00
|
|
|
|
|
|
|
if (!treat_as_code){
|
|
|
|
String lead_name = front_of_directory(name);
|
|
|
|
if (match_insensitive(lead_name, "todo.txt")){
|
|
|
|
treat_as_todo = true;
|
|
|
|
}
|
|
|
|
}
|
2017-01-29 00:03:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (treat_as_code){
|
2017-02-12 06:01:01 +00:00
|
|
|
wrap_lines = false;
|
2017-01-29 00:03:23 +00:00
|
|
|
}
|
2017-02-12 06:01:01 +00:00
|
|
|
if (buffer.file_name == 0){
|
|
|
|
wrap_lines = false;
|
2017-01-29 00:03:23 +00:00
|
|
|
}
|
|
|
|
|
2017-03-29 16:32:06 +00:00
|
|
|
int32_t map_id = (treat_as_code)?((int32_t)default_code_map):((int32_t)mapid_file);
|
|
|
|
|
2017-01-29 00:03:23 +00:00
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_WrapPosition, default_wrap_width);
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_MinimumBaseWrapPosition, default_min_base_width);
|
2017-03-29 16:32:06 +00:00
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_MapID, map_id);
|
2017-05-21 01:22:20 +00:00
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_ParserContext, parse_context_id);
|
2017-01-29 00:03:23 +00:00
|
|
|
|
2017-04-15 21:47:23 +00:00
|
|
|
if (treat_as_todo){
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_WrapLine, true);
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_LexWithoutStrings, true);
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_VirtualWhitespace, true);
|
|
|
|
}
|
|
|
|
else if (treat_as_code && enable_code_wrapping && buffer.size < (1 << 18)){
|
2017-01-29 00:03:23 +00:00
|
|
|
// NOTE(allen|a4.0.12): There is a little bit of grossness going on here.
|
|
|
|
// If we set BufferSetting_Lex to true, it will launch a lexing job.
|
|
|
|
// If a lexing job is active when we set BufferSetting_VirtualWhitespace, the call can fail.
|
|
|
|
// Unfortunantely without tokens virtual whitespace doesn't really make sense.
|
|
|
|
// So for now I have it automatically turning on lexing when virtual whitespace is turned on.
|
|
|
|
// Cleaning some of that up is a goal for future versions.
|
2017-02-12 06:01:01 +00:00
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_WrapLine, true);
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_VirtualWhitespace, true);
|
2017-01-29 00:03:23 +00:00
|
|
|
}
|
|
|
|
else{
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_WrapLine, wrap_lines);
|
|
|
|
buffer_set_setting(app, &buffer, BufferSetting_Lex, treat_as_code);
|
|
|
|
}
|
|
|
|
|
|
|
|
// no meaning for return
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2017-02-26 20:13:06 +00:00
|
|
|
OPEN_FILE_HOOK_SIG(default_new_file){
|
|
|
|
Buffer_Summary buffer = get_buffer(app, buffer_id, AccessOpen);
|
|
|
|
char str[] = "/*\nNew File\n*/\n\n\n";
|
|
|
|
buffer_replace_range(app, &buffer, 0, 0, str, sizeof(str)-1);
|
2017-02-26 21:44:40 +00:00
|
|
|
|
|
|
|
// no meaning for return
|
|
|
|
return(0);
|
2017-02-26 20:13:06 +00:00
|
|
|
}
|
|
|
|
|
2017-01-29 00:03:23 +00:00
|
|
|
OPEN_FILE_HOOK_SIG(default_file_save){
|
2017-04-18 15:41:49 +00:00
|
|
|
Buffer_Summary buffer = get_buffer(app, buffer_id, AccessAll);
|
2017-01-29 00:03:23 +00:00
|
|
|
Assert(buffer.exists);
|
|
|
|
|
|
|
|
#if defined(FCODER_AUTO_INDENT_CPP)
|
|
|
|
int32_t is_virtual = 0;
|
|
|
|
if (automatically_indent_text_on_save && buffer_get_setting(app, &buffer, BufferSetting_VirtualWhitespace, &is_virtual)){
|
|
|
|
if (is_virtual){
|
|
|
|
auto_tab_whole_file_by_summary(app, &buffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// no meaning for return
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2017-04-18 15:41:49 +00:00
|
|
|
OPEN_FILE_HOOK_SIG(default_end_file){
|
|
|
|
Buffer_Summary buffer = get_buffer(app, buffer_id, AccessAll);
|
|
|
|
Assert(buffer.exists);
|
|
|
|
|
|
|
|
char space[1024];
|
|
|
|
String str = make_fixed_width_string(space);
|
|
|
|
append(&str, "Ending file: ");
|
|
|
|
append(&str, make_string(buffer.buffer_name, buffer.buffer_name_len));
|
|
|
|
append(&str, "\n");
|
|
|
|
|
|
|
|
print_message(app, str.str, str.size);
|
|
|
|
|
|
|
|
// no meaning for return
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2017-01-29 00:03:23 +00:00
|
|
|
// NOTE(allen|a4.0.9): The input filter allows you to modify the input
|
|
|
|
// to a frame before 4coder starts processing it at all.
|
|
|
|
//
|
|
|
|
// Right now it only has access to the mouse state, but it will be
|
|
|
|
// extended to have access to the key presses soon.
|
|
|
|
INPUT_FILTER_SIG(default_suppress_mouse_filter){
|
|
|
|
if (suppressing_mouse){
|
|
|
|
*mouse = null_mouse_state;
|
|
|
|
mouse->x = -100;
|
|
|
|
mouse->y = -100;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE(allen|a4): scroll rule information
|
|
|
|
//
|
|
|
|
// The parameters:
|
|
|
|
// target_x, target_y
|
|
|
|
// This is where the view would like to be for the purpose of
|
|
|
|
// following the cursor, doing mouse wheel work, etc.
|
|
|
|
//
|
|
|
|
// scroll_x, scroll_y
|
|
|
|
// These are pointers to where the scrolling actually is. If you bind
|
|
|
|
// the scroll rule it is you have to update these in some way to move
|
|
|
|
// the actual location of the scrolling.
|
|
|
|
//
|
|
|
|
// view_id
|
|
|
|
// This corresponds to which view is computing it's new scrolling position.
|
|
|
|
// This id DOES correspond to the views that View_Summary contains.
|
|
|
|
// This will always be between 1 and 16 (0 is a null id).
|
|
|
|
// See below for an example of having state that carries across scroll udpates.
|
|
|
|
//
|
|
|
|
// is_new_target
|
|
|
|
// If the target of the view is different from the last target in either x or y
|
|
|
|
// this is true, otherwise it is false.
|
|
|
|
//
|
|
|
|
// The return:
|
|
|
|
// Should be true if and only if scroll_x or scroll_y are changed.
|
|
|
|
//
|
|
|
|
// Don't try to use the app pointer in a scroll rule, you're asking for trouble.
|
|
|
|
//
|
|
|
|
// If you don't bind scroll_rule, nothing bad will happen, yo will get default
|
|
|
|
// 4coder scrolling behavior.
|
|
|
|
//
|
|
|
|
|
|
|
|
struct Scroll_Velocity{
|
|
|
|
float x, y;
|
|
|
|
};
|
|
|
|
|
|
|
|
Scroll_Velocity scroll_velocity_[16] = {0};
|
|
|
|
Scroll_Velocity *scroll_velocity = scroll_velocity_ - 1;
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
smooth_camera_step(float target, float *current, float *vel, float S, float T){
|
|
|
|
int32_t result = 0;
|
|
|
|
float curr = *current;
|
|
|
|
float v = *vel;
|
|
|
|
if (curr != target){
|
|
|
|
if (curr > target - .1f && curr < target + .1f){
|
|
|
|
curr = target;
|
|
|
|
v = 1.f;
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
float L = curr + T*(target - curr);
|
|
|
|
|
|
|
|
int32_t sign = (target > curr) - (target < curr);
|
|
|
|
float V = curr + sign*v;
|
|
|
|
|
|
|
|
if (sign > 0) curr = (L<V)?(L):(V);
|
|
|
|
else curr = (L>V)?(L):(V);
|
|
|
|
|
|
|
|
if (curr == V){
|
|
|
|
v *= S;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*current = curr;
|
|
|
|
*vel = v;
|
|
|
|
result = 1;
|
|
|
|
}
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
SCROLL_RULE_SIG(smooth_scroll_rule){
|
|
|
|
Scroll_Velocity *velocity = scroll_velocity + view_id;
|
|
|
|
int32_t result = 0;
|
|
|
|
if (velocity->x == 0.f){
|
|
|
|
velocity->x = 1.f;
|
|
|
|
velocity->y = 1.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (smooth_camera_step(target_y, scroll_y, &velocity->y, 80.f, 1.f/2.f)){
|
|
|
|
result = 1;
|
|
|
|
}
|
|
|
|
if (smooth_camera_step(target_x, scroll_x, &velocity->x, 80.f, 1.f/2.f)){
|
|
|
|
result = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_all_default_hooks(Bind_Helper *context){
|
|
|
|
set_hook(context, hook_exit, default_exit);
|
|
|
|
set_hook(context, hook_view_size_change, default_view_adjust);
|
|
|
|
|
2017-06-23 23:07:18 +00:00
|
|
|
set_start_hook(context, default_start);
|
2017-01-29 00:03:23 +00:00
|
|
|
set_open_file_hook(context, default_file_settings);
|
2017-02-26 20:13:06 +00:00
|
|
|
set_new_file_hook(context, default_new_file);
|
2017-01-29 00:03:23 +00:00
|
|
|
set_save_file_hook(context, default_file_save);
|
2017-06-16 23:10:50 +00:00
|
|
|
|
|
|
|
#if defined(FCODER_STICKY_JUMP)
|
|
|
|
set_end_file_hook(context, end_file_close_jump_list);
|
|
|
|
#else
|
2017-04-18 15:41:49 +00:00
|
|
|
set_end_file_hook(context, default_end_file);
|
2017-06-16 23:10:50 +00:00
|
|
|
#endif
|
2017-04-18 15:41:49 +00:00
|
|
|
|
2017-01-29 00:03:23 +00:00
|
|
|
set_command_caller(context, default_command_caller);
|
|
|
|
set_input_filter(context, default_suppress_mouse_filter);
|
|
|
|
set_scroll_rule(context, smooth_scroll_rule);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// BOTTOM
|
|
|
|
|