From 3970f1f338fcc06eb33facc3ce6e786691ccd683 Mon Sep 17 00:00:00 2001 From: Allen Webster Date: Fri, 9 Sep 2016 18:56:43 -0400 Subject: [PATCH 1/6] fixed initial file loading name issue; wrapped lexer string.h in include guard --- 4coder_API.html | 51 +++++++++++++++++++++----------------- 4coder_custom_api.h | 2 +- 4coder_default_include.cpp | 6 ++--- 4coder_types.h | 29 +++++++++++++++++++--- 4cpp_lexer.h | 4 ++- 4ed.cpp | 15 ++++++++--- 4ed_api_implementation.cpp | 6 +++-- 4ed_file.cpp | 29 +++++++++------------- 4ed_file_view.cpp | 20 +++++++-------- TODO.txt | 8 +++--- win32_4ed.cpp | 4 +-- 11 files changed, 101 insertions(+), 73 deletions(-) diff --git a/4coder_API.html b/4coder_API.html index b3777aba..cd0488e2 100644 --- a/4coder_API.html +++ b/4coder_API.html @@ -2,8 +2,8 @@

4cpp Lexing Library

Table of Contents

§1 Introduction

This is the documentation for alpha 4.0.11 The documentation is still under construction so some of the links are linking to sections that have not been written yet. What is here should be correct and I suspect useful even without some of the other sections.

If you have questions or discover errors please contact editor@4coder.net or to get help from community members you can post on the 4coder forums hosted on handmade.network at 4coder.handmade.network

§2 4coder Systems

Coming Soon
-

§3 Types and Functions

§3.1 Function List

§3.2 Type List

§3.3 Function Descriptions

§3.3.1: exec_command

bool32 app->exec_command(
Application_Links *app,
Command_ID command_id
)
Parameters
command_id
The command_id parameter specifies which internal command to execute.
Return
This call returns non-zero if command_id named a valid internal command.
Description
A call to exec_command executes an internal command. -If command_id is invalid a warning is posted to *messages*.

See Also

§3.3.2: exec_system_command

bool32 app->exec_system_command(
Application_Links *app,
View_Summary *view,
Buffer_Identifier buffer,
char *path,
int32_t path_len,
char *command,
int32_t command_len,
Command_Line_Input_Flag flags
)
Parameters
view
If the view parameter is non-null it specifies a view to display the command's output buffer.
buffer
The buffer the command will output to is specified by the buffer parameter. +

§3 Types and Functions

§3.1 Function List

§3.2 Type List

§3.3 Function Descriptions

§3.3.1: exec_command

bool32 app->exec_command(
Application_Links *app,
Command_ID command_id
)
Parameters
command_id
The command_id parameter specifies which internal command to execute.
Return
This call returns non-zero if command_id named a valid internal command.
Description
A call to exec_command executes an internal command. +If command_id is invalid a warning is posted to *messages*.

See Also

§3.3.2: exec_system_command

bool32 app->exec_system_command(
Application_Links *app,
View_Summary *view,
Buffer_Identifier buffer,
char *path,
int32_t path_len,
char *command,
int32_t command_len,
Command_Line_Interface_Flag flags
)
Parameters
view
If the view parameter is non-null it specifies a view to display the command's output buffer.
buffer
The buffer the command will output to is specified by the buffer parameter. See Buffer_Identifier for information on how this type specifies a buffer.
path
The path parameter specifies the path in which the command shall be executed. The string need not be null terminated.
path_len
The parameter path_len specifies the length of the path string.
command
The command parameter specifies the command that shall be executed. The string need not be null terminated.
command_len
The parameter command_len specifies the length of the command string.
flags
Flags for the behavior of the call are specified in the flags parameter.
Return
This call returns non-zero on success.
Description
A call to exec_system_command executes a command as if called from the command line, and sends the output to a buffer. The buffer identifier can either name a new buffer that does not exist, name a buffer that does exist, or provide the id of a buffer that does exist.

@@ -128,59 +128,64 @@ that protection flag, the object is still returned from the access call.

the output from an app->exec_system_command call such as *build*. This is to prevent the user from accidentally editing output that they might prefer to keep in tact.

AccessHidden = 0x2
AccessHidden is set on any view that is not currently showing it's file, for instance because it is navigating the file system to open a file.

AccessAll = 0xFF
AccessAll is a catchall access for cases where an access call should always - return an object no matter what it's protection flags are.


§3.4.17: Seek_Boundary_Flag

enum Seek_Boundary_Flag;
Description
A Seek_Boundary_Flag field specifies a set of "boundary" types used in seeks for the -beginning or end of different types of words.

Values
BoundaryWhitespace = 0x1
BoundaryToken = 0x2
BoundaryAlphanumeric = 0x4
BoundaryCamelCase = 0x8

§3.4.18: Command_Line_Input_Flag

enum Command_Line_Input_Flag;
Description
A Command_Line_Input_Flag field specifies the behavior of a call to a command line interface.

Values
CLI_OverlapWithConflict = 0x1
If CLI_OverlapWithConflict is set if output buffer of the new command is already + return an object no matter what it's protection flags are.


§3.4.17: Dirty_State

enum Dirty_State;
Description
A Dirty_State value describes whether changes have been made to a buffer +or to an underlying file since the last sync time between the two. Saving a buffer +to it's file or loading the buffer from the file both act as sync points.

Values
DirtyState_UpToDate
DirtyState_UpToDate indicates that there are no unsaved changes and + the underlying system file still agrees with the buffer's state.

DirtyState_UnsavedChanges
DirtyState_UnsavedChanges indicates that there have been changes in the + buffer since the last sync point.

DirtyState_UnloadedChanges
DirtyState_UnsavedChanges indicates that the underlying file has been + edited since the last sync point with the buffer.


§3.4.18: Seek_Boundary_Flag

enum Seek_Boundary_Flag;
Description
A Seek_Boundary_Flag field specifies a set of "boundary" types used in seeks for the +beginning or end of different types of words.

Values
BoundaryWhitespace = 0x1
BoundaryToken = 0x2
BoundaryAlphanumeric = 0x4
BoundaryCamelCase = 0x8

§3.4.19: Command_Line_Interface_Flag

enum Command_Line_Interface_Flag;
Description
A Command_Line_Interface_Flag field specifies the behavior of a call to a command line interface.

Values
CLI_OverlapWithConflict = 0x1
If CLI_OverlapWithConflict is set if output buffer of the new command is already in use by another command which is still executing, the older command relinquishes control of the buffer and both operate simultaneously with only the newer command outputting to the buffer.

CLI_AlwaysBindToView = 0x2
If CLI_AlwaysBindToView is set the output buffer will always be set in the active view even if it is already set in another open view.

CLI_CursorAtEnd = 0x4
If CLI_CursorAtEnd is set the cursor will be kept at the end of the output buffer, - otherwise the cursor is kept at the beginning.


§3.4.19: Auto_Indent_Flag

enum Auto_Indent_Flag;
Description
An Auto_Indent_Flag field specifies the behavior of an auto indentation operation.

Values
AutoIndent_ClearLine = 0x1
If AutoIndent_ClearLine is set, then any line that is only whitespace will + otherwise the cursor is kept at the beginning.


§3.4.20: Auto_Indent_Flag

enum Auto_Indent_Flag;
Description
An Auto_Indent_Flag field specifies the behavior of an auto indentation operation.

Values
AutoIndent_ClearLine = 0x1
If AutoIndent_ClearLine is set, then any line that is only whitespace will be cleared to contain nothing at all. otherwise the line is filled with whitespace to match the nearby indentation.

AutoIndent_UseTab = 0x2
If AutoIndent_UseTab is set, then when putting in leading whitespace to align code, as many tabs will be used as possible until the fine grained control of spaces - is needed to finish the alignment.


§3.4.20: Set_Buffer_Flag

enum Set_Buffer_Flag;
Description
A Set_Buffer_Flag field specifies the behavior of an operation that sets the buffer of a view.

Values
SetBuffer_KeepOriginalGUI = 0x1
If SetBuffer_KeepOriginalGUI then when the file is set, the view will not switch to it + is needed to finish the alignment.


§3.4.21: Set_Buffer_Flag

enum Set_Buffer_Flag;
Description
A Set_Buffer_Flag field specifies the behavior of an operation that sets the buffer of a view.

Values
SetBuffer_KeepOriginalGUI = 0x1
If SetBuffer_KeepOriginalGUI then when the file is set, the view will not switch to it if some other GUI was currently up, otherwise any GUI that is up is closed and the view - switches to the file.


§3.4.21: Input_Type_Flag

enum Input_Type_Flag;
Description
A Input_Type_Flag field specifies a set of input event types.

Values
EventOnAnyKey = 0x1
If EventOnAnyKey is set, all keyboard events are included in the set.

EventOnEsc = 0x2
If EventOnEsc is set, any press of the escape key is included in the set.

EventOnLeftButton = 0x4
If EventOnLeftButton is set, left clicks are included in the set.

EventOnRightButton = 0x8
If EventOnRightButton is set, right clicks are included in the set.

EventOnWheel = 0x10
If EventOnWheel is set, any wheel movement is included in the set.

EventOnButton = (EventOnLeftButton | EventOnRightButton | EventOnWheel)
If EventOnButton is set, all mouse button events are included in the set.

EventOnMouseMove = 0x20
This is not totally implemented yet.

EventOnMouse = (EventOnButton | EventOnMouseMove)
This is not totally implemented yet.

EventAll = 0xFF
EventAll is a catch all name for including all possible events in the set.


§3.4.22: Mouse_Cursor_Show_Type

enum Mouse_Cursor_Show_Type;
Description
A Mouse_Cursor_Show_Type value specifes a mode for 4coder to handle the mouse cursor.

Values
MouseCursorShow_Never
The MouseCursorShow_Never mode never shows the cursor.

MouseCursorShow_Always
The MouseCursorShow_Never mode always shows the cursor.


§3.4.23: Buffer_Seek_Type

enum Buffer_Seek_Type;
Description
The Buffer_Seek_Type is is used in a Buffer_Seek to identify which -coordinates are suppose to be used for the seek.

Values
buffer_seek_pos
This value indicates absolute positioning where positions are measured as the number of bytes from the start of the file.

buffer_seek_wrapped_xy
This value indicates xy positioning with wrapped lines where the x and y values are in pixels.

buffer_seek_unwrapped_xy
This value indicates xy positioning with unwrapped lines where the x and y values are in pixels.

buffer_seek_line_char
This value indicates line-character, or line-column positioning. These coordinates are 1 based to match standard line numbering.

See Also

§3.4.24: View_Split_Position

enum View_Split_Position;
Description
A View_Split_Position specifies where a new view should be placed as a result of -a view split operation.

Values
ViewSplit_Top
This value indicates that the new view should be above the existing view.

ViewSplit_Bottom
This value indicates that the new view should be below the existing view.

ViewSplit_Left
This value indicates that the new view should be left of the existing view.

ViewSplit_Right
This value indicates that the new view should be right of the existing view.


§3.4.25: Generic_Command

union Generic_Command {
Command_ID cmdid;
Custom_Command_Function * command;
};
Description
Generic_Command acts as a name for a command, and can name an + switches to the file.


§3.4.22: Input_Type_Flag

enum Input_Type_Flag;
Description
A Input_Type_Flag field specifies a set of input event types.

Values
EventOnAnyKey = 0x1
If EventOnAnyKey is set, all keyboard events are included in the set.

EventOnEsc = 0x2
If EventOnEsc is set, any press of the escape key is included in the set.

EventOnLeftButton = 0x4
If EventOnLeftButton is set, left clicks are included in the set.

EventOnRightButton = 0x8
If EventOnRightButton is set, right clicks are included in the set.

EventOnWheel = 0x10
If EventOnWheel is set, any wheel movement is included in the set.

EventOnButton = (EventOnLeftButton | EventOnRightButton | EventOnWheel)
If EventOnButton is set, all mouse button events are included in the set.

EventOnMouseMove = 0x20
This is not totally implemented yet.

EventOnMouse = (EventOnButton | EventOnMouseMove)
This is not totally implemented yet.

EventAll = 0xFF
EventAll is a catch all name for including all possible events in the set.


§3.4.23: Mouse_Cursor_Show_Type

enum Mouse_Cursor_Show_Type;
Description
A Mouse_Cursor_Show_Type value specifes a mode for 4coder to handle the mouse cursor.

Values
MouseCursorShow_Never
The MouseCursorShow_Never mode never shows the cursor.

MouseCursorShow_Always
The MouseCursorShow_Never mode always shows the cursor.


§3.4.24: Buffer_Seek_Type

enum Buffer_Seek_Type;
Description
The Buffer_Seek_Type is is used in a Buffer_Seek to identify which +coordinates are suppose to be used for the seek.

Values
buffer_seek_pos
This value indicates absolute positioning where positions are measured as the number of bytes from the start of the file.

buffer_seek_wrapped_xy
This value indicates xy positioning with wrapped lines where the x and y values are in pixels.

buffer_seek_unwrapped_xy
This value indicates xy positioning with unwrapped lines where the x and y values are in pixels.

buffer_seek_line_char
This value indicates line-character, or line-column positioning. These coordinates are 1 based to match standard line numbering.

See Also

§3.4.25: View_Split_Position

enum View_Split_Position;
Description
A View_Split_Position specifies where a new view should be placed as a result of +a view split operation.

Values
ViewSplit_Top
This value indicates that the new view should be above the existing view.

ViewSplit_Bottom
This value indicates that the new view should be below the existing view.

ViewSplit_Left
This value indicates that the new view should be left of the existing view.

ViewSplit_Right
This value indicates that the new view should be right of the existing view.


§3.4.26: Generic_Command

union Generic_Command {
Command_ID cmdid;
Custom_Command_Function * command;
};
Description
Generic_Command acts as a name for a command, and can name an internal command or a custom command.

Fields
cmdid
If this Generic_Command represents an internal command the cmdid field will have a value less than cmdid_count, and this field is the command id for the command.

command
If this Generic_Command does not represent an internal command the command - field is the pointer to the custom command..


§3.4.26: Key_Event_Data

struct Key_Event_Data {
Key_Code keycode;
Key_Code character;
Key_Code character_no_caps_lock;
char modifiers[MDFR_INDEX_COUNT];
};
Description
Key_Event_Data describes a key event, including the + field is the pointer to the custom command..


§3.4.27: Key_Event_Data

struct Key_Event_Data {
Key_Code keycode;
Key_Code character;
Key_Code character_no_caps_lock;
char modifiers[MDFR_INDEX_COUNT];
};
Description
Key_Event_Data describes a key event, including the translation to a character, the translation to a character ignoring the state of caps lock, and an array of all the modifiers that were pressed at the time of the event.

Fields
keycode
This field is the raw keycode which is always non-zero in valid key events.

character
This field is the keycode after translation to a character, this is 0 if there is no translation.

character_no_caps_lock
This field is like the field character, except that the state of caps lock is ignored in the translation.

modifiers
This field is an array indicating the state of modifiers at the time of the key press. The array is indexed using the values of Key_Modifier. A 1 indicates that the corresponding - modifier was held, and a 0 indicates that it was not held.


§3.4.27: Mouse_State

struct Mouse_State {
char l;
char r;
char press_l;
char press_r;
char release_l;
char release_r;
char wheel;
char out_of_window;
int32_t x;
int32_t y;
};
Description
Mouse_State describes an entire mouse state complete with the + modifier was held, and a 0 indicates that it was not held.


§3.4.28: Mouse_State

struct Mouse_State {
char l;
char r;
char press_l;
char press_r;
char release_l;
char release_r;
char wheel;
char out_of_window;
int32_t x;
int32_t y;
};
Description
Mouse_State describes an entire mouse state complete with the position, left and right button states, the wheel state, and whether -or not the mouse if in the window.

Fields
l
This field indicates that the left button is held.

r
This field indicates that the right button is held.

press_l
This field indicates that the left button was pressed this frame.

press_r
This field indicates that the right button was pressed this frame.

release_l
This field indicates that the left button was released this frame.

release_r
This field indicates that the right button was released this frame.

wheel
This field is 0 when the wheel has not moved, it is 1 for a downward motion and -1 for an upward motion.

out_of_window
This field indicates that the mouse is outside of the window.

x
This field contains the x position of the mouse relative to the window where the left side is 0.

y
This field contains the y position of the mouse relative to the window where the top side is 0.


§3.4.28: Range

union Range {
struct {
int32_t min;
int32_t max;
};
struct {
int32_t start;
int32_t end;
};
};
Description
Range describes an integer range typically used for ranges within a buffer. +or not the mouse if in the window.

Fields
l
This field indicates that the left button is held.

r
This field indicates that the right button is held.

press_l
This field indicates that the left button was pressed this frame.

press_r
This field indicates that the right button was pressed this frame.

release_l
This field indicates that the left button was released this frame.

release_r
This field indicates that the right button was released this frame.

wheel
This field is 0 when the wheel has not moved, it is 1 for a downward motion and -1 for an upward motion.

out_of_window
This field indicates that the mouse is outside of the window.

x
This field contains the x position of the mouse relative to the window where the left side is 0.

y
This field contains the y position of the mouse relative to the window where the top side is 0.


§3.4.29: Range

union Range {
struct {
int32_t min;
int32_t max;
};
struct {
int32_t start;
int32_t end;
};
};
Description
Range describes an integer range typically used for ranges within a buffer. Ranges tend are usually not passed as a Range struct into the API, but this struct is used to return ranges.

-Throughout the API ranges are thought of in the form [min,max

Fields
min
This is the smaller value in the range, it is also the 'start'.

max
This is the larger value in the range, it is also the 'end'.

start
This is the start of the range, it is also the 'min'.

end
This is the end of the range, it is also the 'max'.


§3.4.29: File_Info

struct File_Info {
char * filename;
int32_t filename_len;
int32_t folder;
};
Description
File_Info describes the name and type of a file.

Fields
filename
This field is a null terminated string specifying the name of the file.

filename_len
This field specifies the length of the filename string not counting the null terminator.

folder
This field indicates that the description is for a folder not a file.

See Also
File_List

§3.4.30: File_List

struct File_List {
void * block;
File_Info * infos;
int32_t count;
int32_t block_size;
};
Description
File_List is a list of File_Info structs.

Fields
block
This field is for inernal use.

infos
This field is an array of File_Info structs.

count
This field specifies the number of struts in the info array.

block_size
This field is for internal use.


§3.4.31: Buffer_Identifier

struct Buffer_Identifier {
char * name;
int32_t name_len;
int32_t id;
};
Description
Buffer_Identifier acts as a loosely typed description of a buffer that +Throughout the API ranges are thought of in the form [min,max

Fields
min
This is the smaller value in the range, it is also the 'start'.

max
This is the larger value in the range, it is also the 'end'.

start
This is the start of the range, it is also the 'min'.

end
This is the end of the range, it is also the 'max'.


§3.4.30: File_Info

struct File_Info {
char * filename;
int32_t filename_len;
int32_t folder;
};
Description
File_Info describes the name and type of a file.

Fields
filename
This field is a null terminated string specifying the name of the file.

filename_len
This field specifies the length of the filename string not counting the null terminator.

folder
This field indicates that the description is for a folder not a file.

See Also
File_List

§3.4.31: File_List

struct File_List {
void * block;
File_Info * infos;
int32_t count;
int32_t block_size;
};
Description
File_List is a list of File_Info structs.

Fields
block
This field is for inernal use.

infos
This field is an array of File_Info structs.

count
This field specifies the number of struts in the info array.

block_size
This field is for internal use.


§3.4.32: Buffer_Identifier

struct Buffer_Identifier {
char * name;
int32_t name_len;
int32_t id;
};
Description
Buffer_Identifier acts as a loosely typed description of a buffer that can either be a name or an id. If the

Fields
name
This field is the name of the buffer; it need not be null terminated. - If id is specified this pointer should be NULL.

name_len
This field specifies the length of the name string.

id
This field is the id of the buffer. If name is specified this should be 0.


§3.4.32: GUI_Scroll_Vars

struct GUI_Scroll_Vars {
float scroll_y;
int32_t target_y;
int32_t prev_target_y;
float scroll_x;
int32_t target_x;
int32_t prev_target_x;
};
Description
This struct is a part of an incomplete feature.

Fields
scroll_y
TODO

target_y
TODO

prev_target_y
TODO

scroll_x
TODO

target_x
TODO

prev_target_x
TODO


§3.4.33: Full_Cursor

struct Full_Cursor {
int32_t pos;
int32_t line;
int32_t character;
float unwrapped_x;
float unwrapped_y;
float wrapped_x;
float wrapped_y;
};
Description
Full_Cursor describes the position of a cursor in every buffer + If id is specified this pointer should be NULL.

name_len
This field specifies the length of the name string.

id
This field is the id of the buffer. If name is specified this should be 0.


§3.4.33: GUI_Scroll_Vars

struct GUI_Scroll_Vars {
float scroll_y;
int32_t target_y;
int32_t prev_target_y;
float scroll_x;
int32_t target_x;
int32_t prev_target_x;
};
Description
This struct is a part of an incomplete feature.

Fields
scroll_y
TODO

target_y
TODO

prev_target_y
TODO

scroll_x
TODO

target_x
TODO

prev_target_x
TODO


§3.4.34: Full_Cursor

struct Full_Cursor {
int32_t pos;
int32_t line;
int32_t character;
float unwrapped_x;
float unwrapped_y;
float wrapped_x;
float wrapped_y;
};
Description
Full_Cursor describes the position of a cursor in every buffer coordinate system supported by 4coder. This cursor type requires that -the buffer is associated with a view to give the x/y values meaning.

Fields
pos
This field contains the cursor's position in absolute positioning.

line
This field contains the number of the line where the cursor is located. This field is one based.

character
This field contains the number of the column where the cursor is located. This field is one based.

unwrapped_x
This field contains the x position measured with unwrapped lines.

unwrapped_y
This field contains the y position measured with unwrapped lines.

wrapped_x
This field contains the x position measured with wrapped lines.

wrapped_y
This field contains the y position measured with wrapped lines.

See Also
4coder_Buffer_Positioning_System

§3.4.34: Partial_Cursor

struct Partial_Cursor {
int32_t pos;
int32_t line;
int32_t character;
};
Description
Partial_Cursor describes the position of a cursor in all of +the buffer is associated with a view to give the x/y values meaning.

Fields
pos
This field contains the cursor's position in absolute positioning.

line
This field contains the number of the line where the cursor is located. This field is one based.

character
This field contains the number of the column where the cursor is located. This field is one based.

unwrapped_x
This field contains the x position measured with unwrapped lines.

unwrapped_y
This field contains the y position measured with unwrapped lines.

wrapped_x
This field contains the x position measured with wrapped lines.

wrapped_y
This field contains the y position measured with wrapped lines.

See Also
4coder_Buffer_Positioning_System

§3.4.35: Partial_Cursor

struct Partial_Cursor {
int32_t pos;
int32_t line;
int32_t character;
};
Description
Partial_Cursor describes the position of a cursor in all of the coordinate systems that a invariant to the View. In other words the coordinate systems available here can be used on a buffer that is -not currently associated with a View.

Fields
pos
This field contains the cursor's position in absolute positioning.

line
This field contains the number of the line where the cursor is located. This field is one based.

character
This field contains the number of the column where the cursor is located. This field is one based.

See Also
4coder_Buffer_Positioning_System

§3.4.35: Buffer_Seek

struct Buffer_Seek {
Buffer_Seek_Type type;
union {
struct {
int32_t pos;
};
struct {
bool32 round_down;
float x;
float y;
};
struct {
int32_t line;
int32_t character;
};
};
};
Description
Buffer_Seek describes the destination of a seek operation. There are helpers +not currently associated with a View.

Fields
pos
This field contains the cursor's position in absolute positioning.

line
This field contains the number of the line where the cursor is located. This field is one based.

character
This field contains the number of the column where the cursor is located. This field is one based.

See Also
4coder_Buffer_Positioning_System

§3.4.36: Buffer_Seek

struct Buffer_Seek {
Buffer_Seek_Type type;
union {
struct {
int32_t pos;
};
struct {
bool32 round_down;
float x;
float y;
};
struct {
int32_t line;
int32_t character;
};
};
};
Description
Buffer_Seek describes the destination of a seek operation. There are helpers for concisely creating Buffer_Seek structs. They can be found in 4coder_buffer_types.h.

Fields
type
The type field determines the coordinate system of the seek operation.

pos
The pos field specified the pos when the seek is in absolute position.

round_down
For xy coordinate seeks, rounding down means that any x in the box of the character lands on that character. For instance when clicking rounding down is the user's expected behavior. Not rounding down means that the right hand portion of the character's box, which is closer to the next character, will land on that next character. The unrounded behavior is the expected behavior when moving vertically - and keeping the preferred x.

x
The x coordinate for xy type seeks.

y
The y coordinate for xy type seeks.

line
The line number of a line-character type seek.

character
The character number of a line-character type seek.

See Also
Buffer_Seek_Type
4coder_Buffer_Positioning_System

§3.4.36: Buffer_Edit

struct Buffer_Edit {
int32_t str_start;
int32_t len;
int32_t start;
int32_t end;
};
Description
Buffer_Edit describes a range of a buffer and string to replace that range. + and keeping the preferred x.

x
The x coordinate for xy type seeks.

y
The y coordinate for xy type seeks.

line
The line number of a line-character type seek.

character
The character number of a line-character type seek.

See Also
Buffer_Seek_Type
4coder_Buffer_Positioning_System

§3.4.37: Buffer_Edit

struct Buffer_Edit {
int32_t str_start;
int32_t len;
int32_t start;
int32_t end;
};
Description
Buffer_Edit describes a range of a buffer and string to replace that range. A Buffer_Edit has to be paired with a string that contains the actual text that -will be replaced into the buffer.

Fields
str_start
The str_start field specifies the first character in the accompanying string that corresponds with this edit.

len
The len field specifies the length of the string being written into the buffer.

start
The start field specifies the start of the range in the buffer to replace in absolute position.

end
The end field specifies one past the end of the range in the buffer to replace in absolute position.


§3.4.37: Buffer_Summary

struct Buffer_Summary {
bool32 exists;
bool32 ready;
int32_t buffer_id;
Access_Flag lock_flags;
int32_t size;
int32_t line_count;
char * file_name;
int32_t file_name_len;
char * buffer_name;
int32_t buffer_name_len;
bool32 is_lexed;
bool32 tokens_are_ready;
int32_t map_id;
bool32 unwrapped_lines;
};
Description
Buffer_Summary acts as a handle to a buffer and describes the state of the buffer.

Fields
exists
This field indicates whether the Buffer_Summary describes a buffer that is open in 4coder. +will be replaced into the buffer.

Fields
str_start
The str_start field specifies the first character in the accompanying string that corresponds with this edit.

len
The len field specifies the length of the string being written into the buffer.

start
The start field specifies the start of the range in the buffer to replace in absolute position.

end
The end field specifies one past the end of the range in the buffer to replace in absolute position.


§3.4.38: Buffer_Summary

struct Buffer_Summary {
bool32 exists;
bool32 ready;
int32_t buffer_id;
Access_Flag lock_flags;
int32_t size;
int32_t line_count;
char * file_name;
int32_t file_name_len;
char * buffer_name;
int32_t buffer_name_len;
Dirty_State dirty;
bool32 is_lexed;
bool32 tokens_are_ready;
int32_t map_id;
bool32 unwrapped_lines;
};
Description
Buffer_Summary acts as a handle to a buffer and describes the state of the buffer.

Fields
exists
This field indicates whether the Buffer_Summary describes a buffer that is open in 4coder. When this field is false the summary is referred to as a "null summary".

ready
If this is not a null summary, this field indicates whether the buffer has finished loading.

buffer_id
If this is not a null summary this field is the id of the associated buffer. - If this is a null summary then buffer_id is 0.

lock_flags
If this is not a null summary, this field contains flags describing the protection status of the buffer.

size
If this is not a null summary, this field specifies the size of the text in the buffer.

line_count
If this is not a null summary, this field specifies the number of lines in the buffer.

file_name
If this is not a null summary, this field specifies the file name associated to this buffer.

file_name_len
This field specifies the length of the file_name string.

buffer_name
If this is not a null summary, this field specifies the name of the buffer.

buffer_name_len
This field specifies the length of the buffer_name string.

is_lexed
If this is not a null summary, this field indicates whether the buffer is set to lex tokens.

tokens_are_ready
If this is not a null summary, this field indicates whether the buffer has up to date tokens available. + If this is a null summary then buffer_id is 0.

lock_flags
If this is not a null summary, this field contains flags describing the protection status of the buffer.

size
If this is not a null summary, this field specifies the size of the text in the buffer.

line_count
If this is not a null summary, this field specifies the number of lines in the buffer.

file_name
If this is not a null summary, this field specifies the file name associated to this buffer.

file_name_len
This field specifies the length of the file_name string.

buffer_name
If this is not a null summary, this field specifies the name of the buffer.

buffer_name_len
This field specifies the length of the buffer_name string.

dirty
This field indicates the dirty state of the buffer.

is_lexed
If this is not a null summary, this field indicates whether the buffer is set to lex tokens.

tokens_are_ready
If this is not a null summary, this field indicates whether the buffer has up to date tokens available. If this field is false, it may simply mean the tokens are still being generated in a background task and will be available later. If that is the case, is_lexed will be true to indicate that the buffer is trying to get - it's tokens up to date.

map_id
If this is not a null summary, this field specifies the id of the command map for this buffer.

unwrapped_lines
If this is not a null summary, this field indicates whether the buffer 'prefers' wrapped lines.

See Also
Access_Flag

§3.4.38: View_Summary

struct View_Summary {
bool32 exists;
int32_t view_id;
int32_t buffer_id;
Access_Flag lock_flags;
Full_Cursor cursor;
Full_Cursor mark;
float preferred_x;
float line_height;
bool32 unwrapped_lines;
bool32 show_whitespace;
i32_Rect file_region;
GUI_Scroll_Vars scroll_vars;
};
Description
View_Summary acts as a handle to a view and describes the state of the view.

Fields
exists
This field indicates whether the View_Summary describes a view that is open in 4coder. + it's tokens up to date.

map_id
If this is not a null summary, this field specifies the id of the command map for this buffer.

unwrapped_lines
If this is not a null summary, this field indicates whether the buffer 'prefers' wrapped lines.

See Also
Access_Flag
Dirty_State

§3.4.39: View_Summary

struct View_Summary {
bool32 exists;
int32_t view_id;
int32_t buffer_id;
Access_Flag lock_flags;
Full_Cursor cursor;
Full_Cursor mark;
float preferred_x;
float line_height;
bool32 unwrapped_lines;
bool32 show_whitespace;
i32_Rect file_region;
GUI_Scroll_Vars scroll_vars;
};
Description
View_Summary acts as a handle to a view and describes the state of the view.

Fields
exists
This field indicates whether the View_Summary describes a view that is open in 4coder. When this field is false the summary is referred to as a "null summary".

view_id
If this is not a null summary, this field is the id of the associated view. - If this is a null summary then view_id is 0.

buffer_id
If this is not a null summary, then this is the id of the buffer this view currently sees.

lock_flags
If this is not a null summary, this field contains flags describing the protection status of the view.

cursor
If this is not a null summary, this describes the position of the cursor.

mark
If this is not a null summary, this describes the position of the mark.

preferred_x
If this is not a null summary, this is the x position that is maintained in vertical navigation.

line_height
If this is not a null summary, this specifies the height of a line rendered in the view.

unwrapped_lines
If this is not a null summary, this indicates that the view is set to render with unwrapped lines.

show_whitespace
If this is not a null summary, this indicates that the view is set to highlight white space.

file_region
If this is not a null summary, this describes the screen position in which this view's buffer is displayed.

scroll_vars
If this is not a null summary, this describes the scrolling position inside the view.

See Also
Access_Flag
Full_Cursor

§3.4.39: User_Input

struct User_Input {
User_Input_Type_ID type;
bool32 abort;
union {
Key_Event_Data key;
Mouse_State mouse;
};
Generic_Command command;
};
Description
User_Input describes a user input event which can be either a key press or mouse event.

Fields
type
This field specifies whether the event was a key press or mouse event.

abort
This field indicates that an abort event has occurred and the command needs to shut down.

key
This field describes a key press event.

mouse
This field describes a mouse input event.

command
If this event would trigger a command, this field specifies what the command would be.

See Also
User_Input_Type_ID
Generic_Command

§3.4.40: Query_Bar

struct Query_Bar {
String prompt;
String string;
};
Description
Query_Bar is a struct used to store information in the user's control -that will be displayed as a drop down bar durring an interactive command.

Fields
prompt
This specifies the prompt portion of the drop down bar.

string
This specifies the main string portion of the drop down bar.


§3.4.41: Event_Message

struct Event_Message {
int32_t type;
};
Description
This feature is not implemented.

Fields
type
This feature is not implemented.


§3.4.42: Theme_Color

struct Theme_Color {
Style_Tag tag;
int_color color;
};
Description
Theme_Color stores a style tag/color pair, for the purpose of setting and getting colors in the theme.

Fields
tag
The style slot in the style palette.

color
The color in the slot.

See Also
Style_Tag
int_color

§3.4.43: Buffer_Batch_Edit_Type

enum Buffer_Batch_Edit_Type;
Description
A Buffer_Batch_Edit_Type is a type of batch operation.

Values
BatchEdit_Normal
The BatchEdit_Normal operation is always correct but does the most work if there are tokens to correct.

BatchEdit_PreserveTokens
The BatchEdit_PreserveTokens operation is one in which none of the edits add, delete, or change any tokens. - This usually applies when whitespace is being replaced with whitespace.


§3.4.44: Buffer_Batch_Edit

struct Buffer_Batch_Edit {
char * str;
int32_t str_len;
Buffer_Edit * edits;
int32_t edit_count;
};
Description
This struct is used to bundle the parameters of the buffer_batch_edit function. It is convenient + If this is a null summary then view_id is 0.

buffer_id
If this is not a null summary, then this is the id of the buffer this view currently sees.

lock_flags
If this is not a null summary, this field contains flags describing the protection status of the view.

cursor
If this is not a null summary, this describes the position of the cursor.

mark
If this is not a null summary, this describes the position of the mark.

preferred_x
If this is not a null summary, this is the x position that is maintained in vertical navigation.

line_height
If this is not a null summary, this specifies the height of a line rendered in the view.

unwrapped_lines
If this is not a null summary, this indicates that the view is set to render with unwrapped lines.

show_whitespace
If this is not a null summary, this indicates that the view is set to highlight white space.

file_region
If this is not a null summary, this describes the screen position in which this view's buffer is displayed.

scroll_vars
If this is not a null summary, this describes the scrolling position inside the view.

See Also
Access_Flag
Full_Cursor

§3.4.40: User_Input

struct User_Input {
User_Input_Type_ID type;
bool32 abort;
union {
Key_Event_Data key;
Mouse_State mouse;
};
Generic_Command command;
};
Description
User_Input describes a user input event which can be either a key press or mouse event.

Fields
type
This field specifies whether the event was a key press or mouse event.

abort
This field indicates that an abort event has occurred and the command needs to shut down.

key
This field describes a key press event.

mouse
This field describes a mouse input event.

command
If this event would trigger a command, this field specifies what the command would be.

See Also
User_Input_Type_ID
Generic_Command

§3.4.41: Query_Bar

struct Query_Bar {
String prompt;
String string;
};
Description
Query_Bar is a struct used to store information in the user's control +that will be displayed as a drop down bar durring an interactive command.

Fields
prompt
This specifies the prompt portion of the drop down bar.

string
This specifies the main string portion of the drop down bar.


§3.4.42: Event_Message

struct Event_Message {
int32_t type;
};
Description
This feature is not implemented.

Fields
type
This feature is not implemented.


§3.4.43: Theme_Color

struct Theme_Color {
Style_Tag tag;
int_color color;
};
Description
Theme_Color stores a style tag/color pair, for the purpose of setting and getting colors in the theme.

Fields
tag
The style slot in the style palette.

color
The color in the slot.

See Also
Style_Tag
int_color

§3.4.44: Buffer_Batch_Edit_Type

enum Buffer_Batch_Edit_Type;
Description
A Buffer_Batch_Edit_Type is a type of batch operation.

Values
BatchEdit_Normal
The BatchEdit_Normal operation is always correct but does the most work if there are tokens to correct.

BatchEdit_PreserveTokens
The BatchEdit_PreserveTokens operation is one in which none of the edits add, delete, or change any tokens. + This usually applies when whitespace is being replaced with whitespace.


§3.4.45: Buffer_Batch_Edit

struct Buffer_Batch_Edit {
char * str;
int32_t str_len;
Buffer_Edit * edits;
int32_t edit_count;
};
Description
This struct is used to bundle the parameters of the buffer_batch_edit function. It is convenient for a few functions that return a batch edit to the user.

Fields
str
The pointer to the edit string buffer.

str_len
The length of the edit string buffer.

edits
The array of edits to be applied.

edit_count
The number of edits in the array.

See Also
buffer_batch_edit

§4 String Library

§4.1 String Intro

Coming Soon

§4.2 String Function List

§4.3 String Function Descriptions

§4.3.1: char_is_slash

fstr_bool char_is_slash(
char c
)
Description
This call returns non-zero if c is \ or /.


§4.3.2: char_is_upper

fstr_bool char_is_upper(
char c
)
Description
If c is an uppercase letter this call returns true.


§4.3.3: char_is_lower

fstr_bool char_is_lower(
char c
)
Description
If c is a lower letter this call returns true.


§4.3.4: char_to_upper

char char_to_upper(
char c
)
Description
If c is a lowercase letter this call returns the uppercase equivalent, otherwise it returns c.


§4.3.5: char_to_lower

char char_to_lower(
char c
)
Description
If c is an uppercase letter this call returns the lowercase equivalent, otherwise it returns c.


§4.3.6: char_is_whitespace

fstr_bool char_is_whitespace(
char c
)
Description
This call returns non-zero if c is whitespace.


§4.3.7: char_is_alpha_numeric

fstr_bool char_is_alpha_numeric(
char c
)
Description
This call returns non-zero if c is any alphanumeric character including underscore.


§4.3.8: char_is_alpha_numeric_true

fstr_bool char_is_alpha_numeric_true(
char c
)
Description
This call returns non-zero if c is any alphanumeric character no including underscore.


§4.3.9: char_is_alpha

fstr_bool char_is_alpha(
char c
)
Description
This call returns non-zero if c is any alphabetic character including underscore.


§4.3.10: char_is_alpha_true

fstr_bool char_is_alpha_true(
char c
)
Description
This call returns non-zero if c is any alphabetic character.


§4.3.11: char_is_hex

fstr_bool char_is_hex(
char c
)
Description
This call returns non-zero if c is any valid hexadecimal digit.


§4.3.12: char_is_numeric

fstr_bool char_is_numeric(
char c
)
Description
This call returns non-zero if c is any valid decimal digit.


§4.3.13: make_string_cap

String make_string_cap(
void *str,
int32_t size,
int32_t mem_size
)
Parameters
str
The str parameter provides the of memory with which the string shall operate.
size
The size parameter expresses the initial size of the string. If the memory does not already contain a useful string this should be zero.
mem_size
The mem_size parameter expresses the full size of the memory provided by str.
Description
This call returns the String created from the parameters.


§4.3.14: make_string

String make_string(
void *str,
int32_t size
)
Parameters
str
The str parameter provides the of memory with which the string shall operate.
size
The size parameter expresses the initial size of the string. diff --git a/4coder_custom_api.h b/4coder_custom_api.h index 2ff4be34..32b31963 100644 --- a/4coder_custom_api.h +++ b/4coder_custom_api.h @@ -1,5 +1,5 @@ #define EXEC_COMMAND_SIG(n) bool32 n(Application_Links *app, Command_ID command_id) -#define EXEC_SYSTEM_COMMAND_SIG(n) bool32 n(Application_Links *app, View_Summary *view, Buffer_Identifier buffer, char *path, int32_t path_len, char *command, int32_t command_len, Command_Line_Input_Flag flags) +#define EXEC_SYSTEM_COMMAND_SIG(n) bool32 n(Application_Links *app, View_Summary *view, Buffer_Identifier buffer, char *path, int32_t path_len, char *command, int32_t command_len, Command_Line_Interface_Flag flags) #define CLIPBOARD_POST_SIG(n) void n(Application_Links *app, int32_t clipboard_id, char *str, int32_t len) #define CLIPBOARD_COUNT_SIG(n) int32_t n(Application_Links *app, int32_t clipboard_id) #define CLIPBOARD_INDEX_SIG(n) int32_t n(Application_Links *app, int32_t clipboard_id, int32_t item_index, char *out, int32_t len) diff --git a/4coder_default_include.cpp b/4coder_default_include.cpp index 1e23d233..f41463e1 100644 --- a/4coder_default_include.cpp +++ b/4coder_default_include.cpp @@ -364,11 +364,10 @@ buffer_seek_string_forward(Application_Links *app, Buffer_Summary *buffer, read_str.size = size; char chunk[1024]; - int32_t chunk_size = sizeof(chunk); Stream_Chunk stream = {0}; stream.max_end = end; - if (init_stream_chunk(&stream, app, buffer, pos, chunk, chunk_size)){ + if (init_stream_chunk(&stream, app, buffer, pos, chunk, sizeof(chunk))){ int32_t still_looping = 1; do{ for(; pos < stream.end; ++pos){ @@ -419,11 +418,10 @@ buffer_seek_string_backward(Application_Links *app, Buffer_Summary *buffer, read_str.size = size; char chunk[1024]; - int32_t chunk_size = sizeof(chunk); Stream_Chunk stream = {0}; stream.min_start = min; - if (init_stream_chunk(&stream, app, buffer, pos, chunk, chunk_size)){ + if (init_stream_chunk(&stream, app, buffer, pos, chunk, sizeof(chunk))){ int32_t still_looping = 1; do{ for(; pos >= stream.start; --pos){ diff --git a/4coder_types.h b/4coder_types.h index 4eb4426b..af566a86 100644 --- a/4coder_types.h +++ b/4coder_types.h @@ -201,7 +201,7 @@ access call. An access call is usually one the returns a summary struct. If a 4coder object has a particular protection flag set and the corresponding bit is not set in the access field, that 4coder object is hidden. On the other hand if a protection flag is set in the access parameter and the object does not have -that protection flag, the object is still returned from the access call.) TODO */ +that protection flag, the object is still returned from the access call.) */ ENUM(uint32_t, Access_Flag){ /* DOC(AccessOpen does not include any bits, it indicates that the access should only return objects that have no protection flags set.) */ @@ -218,6 +218,23 @@ ENUM(uint32_t, Access_Flag){ AccessAll = 0xFF }; +/* DOC(A Dirty_State value describes whether changes have been made to a buffer +or to an underlying file since the last sync time between the two. Saving a buffer +to it's file or loading the buffer from the file both act as sync points.) */ +ENUM(uint32_t, Dirty_State){ + /* DOC(DirtyState_UpToDate indicates that there are no unsaved changes and + the underlying system file still agrees with the buffer's state.) */ + DirtyState_UpToDate, + + /* DOC(DirtyState_UnsavedChanges indicates that there have been changes in the + buffer since the last sync point.) */ + DirtyState_UnsavedChanges, + + /* DOC(DirtyState_UnsavedChanges indicates that the underlying file has been + edited since the last sync point with the buffer.) */ + DirtyState_UnloadedChanges +}; + /* DOC(A Seek_Boundary_Flag field specifies a set of "boundary" types used in seeks for the beginning or end of different types of words.) */ ENUM(uint32_t, Seek_Boundary_Flag){ @@ -227,8 +244,8 @@ ENUM(uint32_t, Seek_Boundary_Flag){ BoundaryCamelCase = 0x8 }; -/* DOC(A Command_Line_Input_Flag field specifies the behavior of a call to a command line interface.) */ -ENUM(uint32_t, Command_Line_Input_Flag){ +/* DOC(A Command_Line_Interface_Flag field specifies the behavior of a call to a command line interface.) */ +ENUM(uint32_t, Command_Line_Interface_Flag){ /* DOC(If CLI_OverlapWithConflict is set if output buffer of the new command is already in use by another command which is still executing, the older command relinquishes control of the buffer and both operate simultaneously with only the newer command outputting to @@ -561,7 +578,8 @@ struct Buffer_Edit{ }; /* DOC(Buffer_Summary acts as a handle to a buffer and describes the state of the buffer.) -DOC_SEE(Access_Flag) */ +DOC_SEE(Access_Flag) +DOC_SEE(Dirty_State) */ struct Buffer_Summary{ /* DOC( This field indicates whether the Buffer_Summary describes a buffer that is open in 4coder. @@ -595,6 +613,9 @@ struct Buffer_Summary{ /* DOC(This field specifies the length of the buffer_name string.) */ int32_t buffer_name_len; + /* DOC(This field indicates the dirty state of the buffer.) */ + Dirty_State dirty; + /* DOC(If this is not a null summary, this field indicates whether the buffer is set to lex tokens.) */ bool32 is_lexed; /* DOC(If this is not a null summary, this field indicates whether the buffer has up to date tokens available. diff --git a/4cpp_lexer.h b/4cpp_lexer.h index c19cc9ad..6e8f48ae 100644 --- a/4cpp_lexer.h +++ b/4cpp_lexer.h @@ -15,7 +15,9 @@ #define FCPP_INTERNAL FCPP_LINK #include -#define FSTRING_IMPLEMENTATION +#if !defined(FSTRING_GUARD) +# define FSTRING_IMPLEMENTATION +#endif #include "4coder_string.h" #include "4cpp_lexer_types.h" #include "4cpp_lexer_tables.c" diff --git a/4ed.cpp b/4ed.cpp index 3e349a9f..9c52cb70 100644 --- a/4ed.cpp +++ b/4ed.cpp @@ -1989,13 +1989,20 @@ App_Step_Sig(app_step){ models->hooks[hook_start](&models->app_links); } + char space[512]; + String cl_filename = make_fixed_width_string(space); + copy_ss(&cl_filename, models->hot_directory.string); + + i32 cl_filename_len = cl_filename.size; + i32 i = 0; Panel *panel = models->layout.used_sentinel.next; for (; i < models->settings.init_files_count; ++i, panel = panel->next){ - String filename = make_string_slowly(models->settings.init_files[i]); + cl_filename.size = cl_filename_len; + append_sc(&cl_filename, models->settings.init_files[i]); if (i < models->layout.panel_count){ - view_open_file(system, models, panel->view, filename); + view_open_file(system, models, panel->view, cl_filename); view_show_file(panel->view); Assert("Earlier" && panel->view->file_data.file != 0); #if 0 @@ -2009,14 +2016,14 @@ App_Step_Sig(app_step){ #endif } else{ - view_open_file(system, models, 0, filename); + view_open_file(system, models, 0, cl_filename); } + } if (i < models->layout.panel_count){ view_set_file(panel->view, models->message_buffer, models); view_show_file(panel->view); - ++i; panel = panel->next; } diff --git a/4ed_api_implementation.cpp b/4ed_api_implementation.cpp index 9230e1cb..797e9f59 100644 --- a/4ed_api_implementation.cpp +++ b/4ed_api_implementation.cpp @@ -31,6 +31,8 @@ fill_buffer_summary(Buffer_Summary *buffer, Editing_File *file, Working_Set *wor buffer->file_name = file->name.source_path.str; buffer->buffer_name = file->name.live_name.str; + buffer->dirty = file->state.dirty; + buffer->is_lexed = file->settings.tokens_exist; if (file->state.token_array.tokens && @@ -180,9 +182,9 @@ DOC_SEE(Command_ID) return(result); } -// TODO(allen): This is a bit of a mess and needs to be fixed soon +// TODO(allen): This is a bit of a mess and needs to be fixed soon. API_EXPORT bool32 -Exec_System_Command(Application_Links *app, View_Summary *view, Buffer_Identifier buffer, char *path, int32_t path_len, char *command, int32_t command_len, Command_Line_Input_Flag flags)/* +Exec_System_Command(Application_Links *app, View_Summary *view, Buffer_Identifier buffer, char *path, int32_t path_len, char *command, int32_t command_len, Command_Line_Interface_Flag flags)/* DOC_PARAM(view, If the view parameter is non-null it specifies a view to display the command's output buffer.) DOC_PARAM(buffer, The buffer the command will output to is specified by the buffer parameter. See Buffer_Identifier for information on how this type specifies a buffer.) diff --git a/4ed_file.cpp b/4ed_file.cpp index a479513b..6ac5fd8c 100644 --- a/4ed_file.cpp +++ b/4ed_file.cpp @@ -100,12 +100,6 @@ struct Undo_Data{ b32 current_block_normal; }; -enum File_Sync_State{ - SYNC_GOOD, - SYNC_BEHIND_OS, - SYNC_UNSAVED -}; - struct Text_Effect{ i32 start, end; u32 color; @@ -143,7 +137,7 @@ struct Editing_File_State{ Text_Effect paste_effect; - File_Sync_State sync; + Dirty_State dirty; u32 ignore_behind_os; File_Edit_Positions edit_pos_space[16]; @@ -671,6 +665,7 @@ touch_file(Working_Set *working_set, Editing_File *file){ struct Hot_Directory{ String string; File_List file_list; + // TODO(allen): eliminate slash char slash; }; @@ -765,7 +760,7 @@ inline b32 buffer_needs_save(Editing_File *file){ b32 result = 0; if (!file->settings.unimportant){ - if (file->state.sync == SYNC_UNSAVED){ + if (file->state.dirty == DirtyState_UnsavedChanges){ result = 1; } } @@ -776,8 +771,8 @@ inline b32 buffer_can_save(Editing_File *file){ b32 result = 0; if (!file->settings.unimportant){ - if (file->state.sync == SYNC_UNSAVED || - file->state.sync == SYNC_BEHIND_OS){ + if (file->state.dirty == DirtyState_UnsavedChanges || + file->state.dirty == DirtyState_UnloadedChanges){ result = 1; } } @@ -814,26 +809,26 @@ file_set_to_loading(Editing_File *file){ inline void file_mark_clean(Editing_File *file){ - if (file->state.sync != SYNC_BEHIND_OS){ - file->state.sync = SYNC_GOOD; + if (file->state.dirty != DirtyState_UnloadedChanges){ + file->state.dirty = DirtyState_UpToDate; } } inline void file_mark_dirty(Editing_File *file){ - if (file->state.sync != SYNC_BEHIND_OS){ - file->state.sync = SYNC_UNSAVED; + if (file->state.dirty != DirtyState_UnloadedChanges){ + file->state.dirty = DirtyState_UnsavedChanges; } } inline void file_mark_behind_os(Editing_File *file){ - file->state.sync = SYNC_BEHIND_OS; + file->state.dirty = DirtyState_UnloadedChanges; } -inline File_Sync_State +inline Dirty_State file_get_sync(Editing_File *file){ - return (file->state.sync); + return (file->state.dirty); } internal void diff --git a/4ed_file_view.cpp b/4ed_file_view.cpp index 274781df..58d8661d 100644 --- a/4ed_file_view.cpp +++ b/4ed_file_view.cpp @@ -782,7 +782,7 @@ starts_new_line(u8 character){ inline void file_synchronize_times(System_Functions *system, Editing_File *file){ - file->state.sync = SYNC_GOOD; + file->state.dirty = DirtyState_UpToDate; } internal b32 @@ -3217,9 +3217,9 @@ get_exhaustive_info(System_Functions *system, Working_Set *working_set, Exhausti result.message = null_string; if (result.is_loaded){ switch (file_get_sync(file)){ - case SYNC_GOOD: result.message = message_loaded; break; - case SYNC_BEHIND_OS: result.message = message_unsynced; break; - case SYNC_UNSAVED: result.message = message_unsaved; break; + case DirtyState_UpToDate: result.message = message_loaded; break; + case DirtyState_UnsavedChanges: result.message = message_unsaved; break; + case DirtyState_UnloadedChanges: result.message = message_unsynced; break; } } @@ -4037,8 +4037,8 @@ step_file_view(System_Functions *system, View *view, View *active_view, Input_Su message = null_string; if (!file->settings.unimportant){ switch (file_get_sync(file)){ - case SYNC_BEHIND_OS: message = message_unsynced; break; - case SYNC_UNSAVED: message = message_unsaved; break; + case DirtyState_UnloadedChanges: message = message_unsynced; break; + case DirtyState_UnsavedChanges: message = message_unsaved; break; } } @@ -4058,8 +4058,8 @@ step_file_view(System_Functions *system, View *view, View *active_view, Input_Su message = null_string; if (!file->settings.unimportant){ switch (file_get_sync(file)){ - case SYNC_BEHIND_OS: message = message_unsynced; break; - case SYNC_UNSAVED: message = message_unsaved; break; + case DirtyState_UnloadedChanges: message = message_unsynced; break; + case DirtyState_UnsavedChanges: message = message_unsaved; break; } } @@ -5132,13 +5132,13 @@ draw_file_bar(Render_Target *target, View *view, Editing_File *file, i32_Rect re if (!file->settings.unimportant){ switch (file_get_sync(file)){ - case SYNC_BEHIND_OS: + case DirtyState_UnloadedChanges: { persist String out_of_sync = make_lit_string(" !"); intbar_draw_string(target, &bar, out_of_sync, pop2_color); }break; - case SYNC_UNSAVED: + case DirtyState_UnsavedChanges: { persist String out_of_sync = make_lit_string(" *"); intbar_draw_string(target, &bar, out_of_sync, pop2_color); diff --git a/TODO.txt b/TODO.txt index ca7c95df..7708cf84 100644 --- a/TODO.txt +++ b/TODO.txt @@ -41,7 +41,7 @@ ; [X] switch to file "4ed.cpp" with "win32_4ed.cpp" open ; [X] inserting new line at top of file ~ scrolling jump when window is unaligned ; [X] saving/killing *compilation* doesn't work -; [X] line wrapping also doesn't work +; [X] line wrapping also doesn't work ; [X] save as corruptitates the filename ; [X] crash when leaving maximized mode and view get's weird ; [X] decrease scroll speed on short windows @@ -87,7 +87,6 @@ ; BEFORE I SHIP ; -; [X] why are command line files not loading any more? ; [X] tokens in the custom API ; [X] token seeking on custom side ; [X] auto indent on the custom side @@ -95,10 +94,7 @@ ; [] inserting lines at end of block comment ; [] clean up and comment the auto indent code to allow for customizations ; [] more built in options for auto indenting -; [] expose dirty flags ; [] occasionally missing the (!) mark on files on windows -; [] scroll down on compilation buffer durring compilation -; ; ; TODOS @@ -143,6 +139,8 @@ ; [X] file out of sync ; [X] mouse down/up distinction ; [X] case insensitive interactive switch buffer +; [X] expose dirty flags +; [X] why are command line files not loading any more? ; ; [] binary buffers diff --git a/win32_4ed.cpp b/win32_4ed.cpp index 312bbf2e..e4a84323 100644 --- a/win32_4ed.cpp +++ b/win32_4ed.cpp @@ -904,7 +904,7 @@ win32_init_drive_strings(Drive_Strings *dstrings){ // for the ability to handle them very quickly when nothing strange is // going on. internal int32_t -win32_canonical_ansi_name(Drive_Strings *dstrings, char *src, i32 len, char *dst, i32 max){ +win32_canonical_ascii_name(Drive_Strings *dstrings, char *src, i32 len, char *dst, i32 max){ char *wrt = dst; char *wrt_stop = dst + max; char *src_stop = src + len; @@ -978,7 +978,7 @@ win32_canonical_ansi_name(Drive_Strings *dstrings, char *src, i32 len, char *dst internal Sys_Get_Canonical_Sig(system_get_canonical){ - i32 result = win32_canonical_ansi_name(&win32vars.dstrings, filename, len, buffer, max); + i32 result = win32_canonical_ascii_name(&win32vars.dstrings, filename, len, buffer, max); return(result); } From 077e6963b170719b2a712e211b00496a54d57456 Mon Sep 17 00:00:00 2001 From: Allen Webster Date: Fri, 9 Sep 2016 18:59:00 -0400 Subject: [PATCH 2/6] ooops --- 4cpp_lexer.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/4cpp_lexer.h b/4cpp_lexer.h index 6e8f48ae..7409dfec 100644 --- a/4cpp_lexer.h +++ b/4cpp_lexer.h @@ -17,8 +17,8 @@ #include #if !defined(FSTRING_GUARD) # define FSTRING_IMPLEMENTATION +# include "4coder_string.h" #endif -#include "4coder_string.h" #include "4cpp_lexer_types.h" #include "4cpp_lexer_tables.c" From 170237329f9869f4433bfa3a2fa2ebcda1d9580f Mon Sep 17 00:00:00 2001 From: insofaras Date: Sat, 10 Sep 2016 01:10:43 +0100 Subject: [PATCH 3/6] linux cursor fix, ty ChronalDragon --- linux_4ed.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/linux_4ed.cpp b/linux_4ed.cpp index 53a03240..6ef37cf0 100644 --- a/linux_4ed.cpp +++ b/linux_4ed.cpp @@ -3279,7 +3279,7 @@ main(int argc, char **argv) Cursor xcursors[APP_MOUSE_CURSOR_COUNT] = { None, - XCreateFontCursor(linuxvars.XDisplay, XC_arrow), + None, XCreateFontCursor(linuxvars.XDisplay, XC_xterm), XCreateFontCursor(linuxvars.XDisplay, XC_sb_h_double_arrow), XCreateFontCursor(linuxvars.XDisplay, XC_sb_v_double_arrow) From 404e08260f12a3cccd0d7f554f099184fb7e6122 Mon Sep 17 00:00:00 2001 From: insofaras Date: Sat, 10 Sep 2016 03:57:45 +0100 Subject: [PATCH 4/6] linux send_exit_signal --- linux_4ed.cpp | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/linux_4ed.cpp b/linux_4ed.cpp index 6ef37cf0..e0ce31f9 100644 --- a/linux_4ed.cpp +++ b/linux_4ed.cpp @@ -806,6 +806,11 @@ IS_FULLSCREEN_SIG(system_is_fullscreen){ return result; } +internal +SEND_EXIT_SIGNAL_SIG(system_send_exit_signal){ + linuxvars.keep_running = 0; +} + // // Clipboard // @@ -1502,6 +1507,7 @@ LinuxLoadSystemCode(){ linuxvars.system.show_mouse_cursor = system_show_mouse_cursor; linuxvars.system.toggle_fullscreen = system_toggle_fullscreen; linuxvars.system.is_fullscreen = system_is_fullscreen; + linuxvars.system.send_exit_signal = system_send_exit_signal; // clipboard linuxvars.system.post_clipboard = system_post_clipboard; @@ -3433,6 +3439,8 @@ main(int argc, char **argv) linuxvars.input.clipboard = null_string; } + b32 keep_running = linuxvars.keep_running; + linuxvars.app.step( &linuxvars.system, &linuxvars.target, @@ -3443,7 +3451,7 @@ main(int argc, char **argv) if(result.perform_kill){ break; - } else { + } else if(!keep_running && !linuxvars.keep_running){ linuxvars.keep_running = 1; } From f0ee2fb6f1d46924ccaa7e3687a4c6e634f737f8 Mon Sep 17 00:00:00 2001 From: Alex Baines Date: Sat, 10 Sep 2016 08:27:21 +0100 Subject: [PATCH 5/6] improve the linux error popup a bit --- linux_4ed.cpp | 74 ++++++++++++++++++++++++++++++++++----------------- 1 file changed, 49 insertions(+), 25 deletions(-) diff --git a/linux_4ed.cpp b/linux_4ed.cpp index e0ce31f9..7d4cb425 100644 --- a/linux_4ed.cpp +++ b/linux_4ed.cpp @@ -2220,10 +2220,23 @@ LinuxFatalErrorMsg(const char* msg) exit(1); } - int win_w = 450; - int win_h = 150 + (strlen(msg) / 40) * 24; + const int num_cols = 50; + int win_w = (num_cols + 10) * 9; + int win_h = 140; - Window w = XCreateSimpleWindow(dpy, DefaultRootWindow(dpy), 0, 0, win_w, win_h, 0, 0, 0x2EA44F); + { + const char *start_p = msg, *space_p = NULL; + for(const char* p = msg; *p; ++p){ + if(*p == ' ') space_p = p; + if(*p == '\n' || p - start_p > num_cols){ + win_h += 18; + start_p = space_p ? space_p + 1 : p; + space_p = NULL; + } + } + } + + Window w = XCreateSimpleWindow(dpy, DefaultRootWindow(dpy), 0, 0, win_w, win_h, 0, 0, 0x227A3B); XStoreName(dpy, w, "4coder Error"); XSizeHints* sh = XAllocSizeHints(); @@ -2245,26 +2258,26 @@ LinuxFatalErrorMsg(const char* msg) Atom WM_DELETE_WINDOW = XInternAtom(dpy, "WM_DELETE_WINDOW", False); XSetWMProtocols(dpy, w, &WM_DELETE_WINDOW, 1); + LinuxSetIcon(dpy, w); + XMapRaised(dpy, w); XSync(dpy, False); - XSelectInput(dpy, w, ExposureMask | StructureNotifyMask | PointerMotionMask | ButtonPressMask | ButtonReleaseMask); + XSelectInput(dpy, w, ExposureMask | StructureNotifyMask | PointerMotionMask | ButtonPressMask | ButtonReleaseMask | KeyPressMask); - XFontStruct* font = XLoadQueryFont(dpy, "-*-fixed-*-*-*-*-*-140-*-*-*-*-iso8859-1"); + XFontStruct* font = XLoadQueryFont(dpy, "-*-fixed-bold-*-*-*-*-140-*-*-*-*-iso8859-1"); if(!font){ exit(1); } XGCValues gcv; gcv.foreground = WhitePixel(dpy, 0); - gcv.background = 0x2EA44F; gcv.line_width = 2; gcv.font = font->fid; - GC gc = XCreateGC(dpy, w, GCForeground | GCBackground | GCFont | GCLineWidth, &gcv); - + GC gc1 = XCreateGC(dpy, w, GCForeground | GCFont | GCLineWidth, &gcv); gcv.foreground = BlackPixel(dpy, 0); - GC gc2 = XCreateGC(dpy, w, GCForeground | GCBackground | GCFont | GCLineWidth, &gcv); + GC gc2 = XCreateGC(dpy, w, GCForeground | GCFont | GCLineWidth, &gcv); int button_trigger = 0; int button_hi = 0; @@ -2279,7 +2292,6 @@ LinuxFatalErrorMsg(const char* msg) if(ev.type == ConfigureNotify){ redraw = 1; - win_w = ev.xconfigure.width; win_h = ev.xconfigure.height; } @@ -2298,6 +2310,13 @@ LinuxFatalErrorMsg(const char* msg) } } + if(ev.type == KeyPress){ + KeySym sym = XLookupKeysym(&ev.xkey, 0); + if(sym == XK_Escape || sym == XK_Return){ + exit(1); + } + } + if(ev.type == ButtonPress && ev.xbutton.button == Button1){ if(button_hi) button_trigger = 1; redraw = 1; @@ -2318,6 +2337,10 @@ LinuxFatalErrorMsg(const char* msg) exit(1); } +#define DRAW_STR(x, y, str, len) \ + XDrawString(dpy, w, gc2, (x)+1, (y)+1, (str), (len)); \ + XDrawString(dpy, w, gc1, (x) , (y) , (str), (len)) + if(redraw){ XClearWindow(dpy, w); @@ -2329,38 +2352,39 @@ LinuxFatalErrorMsg(const char* msg) const char title[] = "4coder - Fatal Error"; int width = XTextWidth(font, title, sizeof(title)-1); int x = (win_w/2) - (width/2); - XDrawString(dpy, w, gc2, x+2, y+2, title, sizeof(title)-1); - XDrawString(dpy, w, gc, x, y, title, sizeof(title)-1); + DRAW_STR(x, y, title, sizeof(title)-1); } y += 36; - - int width = XTextWidth(font, "x", 1) * 40; + int width = XTextWidth(font, "x", 1) * num_cols; int x = (win_w/2) - (width/2); + for(const char* p = line_start; *p; ++p){ if(*p == ' ') last_space = p; - if(p - line_start > 40){ - if(!last_space) last_space = p; + if(p - line_start > num_cols || *p == '\n' || !p[1]){ - XDrawString(dpy, w, gc2, x+2, y+2, line_start, last_space - line_start); - XDrawString(dpy, w, gc, x, y, line_start, last_space - line_start); - line_start = *last_space == ' ' ? last_space + 1 : p; + const char* new_line_start = last_space + 1; + if(!last_space || *p == '\n' || !p[1]){ + new_line_start = last_space = (p + !p[1]); + } + + DRAW_STR(x, y, line_start, last_space - line_start); + + line_start = new_line_start; last_space = NULL; y += 18; } } - XDrawString(dpy, w, gc2, x+2, y+2, line_start, strlen(line_start)); - XDrawString(dpy, w, gc, x, y, line_start, strlen(line_start)); - - XDrawRectangles(dpy, w, gc, &button_rect, 1); + XDrawRectangles(dpy, w, gc1, &button_rect, 1); if(button_hi || button_trigger){ XDrawRectangle(dpy, w, gc2, button_rect.x+1, button_rect.y+1, button_rect.width-2, button_rect.height-2); } - XDrawString(dpy, w, gc2, button_rect.x + 22, button_rect.y + 17, "Drat!", 5); - XDrawString(dpy, w, gc, button_rect.x + 20, button_rect.y + 15, "Drat!", 5); + + DRAW_STR(button_rect.x + 20, button_rect.y + 15, "Drat!", 5); } } +#undef DRAW_STR } internal int From 8bd7c4611abdee8379bdafc7b9290fd499292cd9 Mon Sep 17 00:00:00 2001 From: Allen Webster Date: Sat, 10 Sep 2016 11:22:25 -0400 Subject: [PATCH 6/6] indent whole tokens working --- 4coder_auto_indent.cpp | 75 ++++++++++++++++++++++++++++++++----- 4coder_default_bindings.cpp | 2 +- 4coder_default_include.cpp | 12 +++++- 4ed.cpp | 2 +- TODO.txt | 2 +- 5 files changed, 79 insertions(+), 14 deletions(-) diff --git a/4coder_auto_indent.cpp b/4coder_auto_indent.cpp index ba791130..a98397e9 100644 --- a/4coder_auto_indent.cpp +++ b/4coder_auto_indent.cpp @@ -458,6 +458,56 @@ get_indentation_marks(Application_Links *app, Partition *part, Buffer_Summary *b return(indent_marks); } +static void +get_indent_lines_minimum(Application_Links *app, Buffer_Summary *buffer, + int32_t start_pos, int32_t end_pos, + int32_t *line_start_out, int32_t *line_end_out){ + int32_t line_start = buffer_get_line_index(app, buffer, start_pos); + int32_t line_end = buffer_get_line_index(app, buffer, end_pos) + 1; + + *line_start_out = line_start; + *line_end_out = line_end; +} + +static void +get_indent_lines_whole_tokens(Application_Links *app, Buffer_Summary *buffer, Cpp_Token_Array tokens, + int32_t start_pos, int32_t end_pos, + int32_t *line_start_out, int32_t *line_end_out){ + int32_t line_start = buffer_get_line_index(app, buffer, start_pos); + int32_t line_end = buffer_get_line_index(app, buffer, end_pos); + + for (;line_start > 0;){ + int32_t line_start_pos = 0; + Cpp_Token *token = get_first_token_at_line(app, buffer, tokens, line_start, &line_start_pos); + if (token->start < line_start_pos){ + line_start = buffer_get_line_index(app, buffer, token->start); + } + else{ + break; + } + } + + for (;line_end+1 < buffer->line_count;){ + int32_t next_line_start_pos = 0; + Cpp_Token *token = get_first_token_at_line(app, buffer, tokens, line_end+1, &next_line_start_pos); + if (token && token->start < next_line_start_pos){ + line_end = buffer_get_line_index(app, buffer, token->start+token->size); + } + else{ + break; + } + } + if (line_end >= buffer->line_count){ + line_end = buffer->line_count; + } + else{ + line_end += 1; + } + + *line_start_out = line_start; + *line_end_out = line_end; +} + static bool32 buffer_auto_indent(Application_Links *app, Partition *part, Buffer_Summary *buffer, int32_t start, int32_t end, int32_t tab_width, Auto_Indent_Flag flags){ @@ -468,25 +518,32 @@ buffer_auto_indent(Application_Links *app, Partition *part, Buffer_Summary *buff Temp_Memory temp = begin_temp_memory(part); - // Stage 1: Setup - // Read the tokens to be used for indentation. - // Get the first and last lines to indent. + // Stage 1: Read the tokens to be used for indentation. Cpp_Token_Array tokens; tokens.count = app->buffer_token_count(app, buffer); tokens.max_count = tokens.count; tokens.tokens = push_array(part, Cpp_Token, tokens.count); app->buffer_read_tokens(app, buffer, 0, tokens.count, tokens.tokens); - int32_t line_start = buffer_get_line_index(app, buffer, start); - int32_t line_end = buffer_get_line_index(app, buffer, end) + 1; + // Stage 2: Decide where the first and last lines are. + // The lines in the range [line_start,line_end) will be indented. + int32_t do_whole_tokens = 1; - // Stage 2: Decide Indent Amounts - // Get an array representing how much each line in [line_start,line_end] - // should be indented. + int32_t line_start = 0, line_end = 0; + if (do_whole_tokens){ + get_indent_lines_whole_tokens(app, buffer, tokens, start, end, &line_start, &line_end); + } + else{ + get_indent_lines_minimum(app, buffer, start, end, &line_start, &line_end); + } + + // Stage 3: Decide Indent Amounts + // Get an array representing how much each line in + // the range [line_start,line_end) should be indented. int32_t *indent_marks = get_indentation_marks(app, part, buffer, tokens, line_start, line_end, tab_width); - // Stage 3: Set the Line Indents + // Stage 4: Set the Line Indents Indent_Options opts = {0}; opts.empty_blank_lines = (flags & AutoIndent_ClearLine); opts.use_tabs = (flags & AutoIndent_UseTab); diff --git a/4coder_default_bindings.cpp b/4coder_default_bindings.cpp index c92be080..65480ad2 100644 --- a/4coder_default_bindings.cpp +++ b/4coder_default_bindings.cpp @@ -282,7 +282,7 @@ default_keys(Bind_Helper *context){ bind(context, key_f2, MDFR_NONE, toggle_mouse); bind(context, key_page_up, MDFR_CTRL, toggle_fullscreen); - bind(context, key_f4, MDFR_ALT, exit_4coder); + bind(context, 'E', MDFR_ALT, exit_4coder); end_map(context); diff --git a/4coder_default_include.cpp b/4coder_default_include.cpp index f41463e1..7869cad3 100644 --- a/4coder_default_include.cpp +++ b/4coder_default_include.cpp @@ -571,7 +571,8 @@ buffer_get_line_index(Application_Links *app, Buffer_Summary *buffer, int32_t po } static Cpp_Token* -get_first_token_at_line(Application_Links *app, Buffer_Summary *buffer, Cpp_Token_Array tokens, int32_t line){ +get_first_token_at_line(Application_Links *app, Buffer_Summary *buffer, Cpp_Token_Array tokens, int32_t line, + int32_t *line_start_out = 0){ int32_t line_start = buffer_get_line_start(app, buffer, line); Cpp_Get_Token_Result get_token = cpp_get_token(&tokens, line_start); @@ -579,7 +580,14 @@ get_first_token_at_line(Application_Links *app, Buffer_Summary *buffer, Cpp_Toke get_token.token_index += 1; } - Cpp_Token *result = tokens.tokens + get_token.token_index; + if (line_start_out){ + *line_start_out = line_start; + } + + Cpp_Token *result = 0; + if (get_token.token_index < tokens.count){ + result = tokens.tokens + get_token.token_index; + } return(result); } diff --git a/4ed.cpp b/4ed.cpp index 9c52cb70..a4a06428 100644 --- a/4ed.cpp +++ b/4ed.cpp @@ -2505,7 +2505,7 @@ App_Step_Sig(app_step){ "-4coder now supports proper, borderless, fullscreen with the flag -F\n" " and fullscreen can be toggled with .\n" " (This sometimes causes artifacts on the Windows task bar)\n" - "- to exit\n" + "- to exit\n" "-hook on exit for the customization system\n" "-tokens now exposed in customization system\n" "-mouse release events in customization system\n" diff --git a/TODO.txt b/TODO.txt index 7708cf84..1b792d3a 100644 --- a/TODO.txt +++ b/TODO.txt @@ -90,7 +90,7 @@ ; [X] tokens in the custom API ; [X] token seeking on custom side ; [X] auto indent on the custom side -; [] indent whole comments +; [X] indent whole comments ; [] inserting lines at end of block comment ; [] clean up and comment the auto indent code to allow for customizations ; [] more built in options for auto indenting