The next addition to the program class is to handle the New program action. Previously, the clear function of the edit box was called. This function is part of the QPlainTextEdit base class that the edit box class is derived from. When this function was called, it generated a document changed signal, which the edit box processed and ended up clearing the program unit that was attached to the edit box. However, the program class should have this responsibility since there may not be an attached edit box.
The new action routine in the main window class was modified to call the new clear function in the program class. This new clear function calls a new clear function of the program unit member (a program model instance). Eventually, all of the program units of the program (for subroutines and functions) will be deleted with only the program unit for the main routine being cleared.
The new program model class clear function clears the line information list, program code vector, error list and all of the dictionaries. This required new clear functions be added to the dictionary and information dictionary classes. A new clear function was also added to the abstract information class, which was implemented for the constant number and string information classes. Finally, the clear function sends the new program cleared signal, which is connected to the edit box clear funtion.
The clear function was reimplemented in the edit box class. This was necessary so that a flag could be set before calling the clear function of the QPlainTextEdit base class. This flag indicates to the document changed slot of the edit box routine that the signal should be ignored as it didn't originate by editing the program. The recreating line flag, that is used to ignore document changes when lines are replaced with their recreated text, was used for this purpose. This flag was renamed to the more appropriate ignore change.
A minor problem was also corrected in the main window class. Previously, before a program was saved, the main window instance called the capture modified line routine of the edit box to make sure that any changes made to the current line are saved. This normally takes place when the cursor is moved from the line being changed. This check also needs to take place before the check to see if the program should be saved before it is cleared by the new action. This functionality was put into the new program capture edit changes routine since it is called from two different places.
[commit 8d6e5703c0]
Sunday, December 29, 2013
Tuesday, December 24, 2013
Program – Program Unit
Continuing with the transition to the new program class, the pointer to the program model was moved from the main window class to the program class and renamed to unit. When support for subroutines and functions is added, the single unit will become a list of units with the first one holding the main routine and the list growing for each subroutine and function added to the program. An access function for the program unit was also added to the program class.
[commit 5bc9d15f43]
[commit 5bc9d15f43]
Saturday, December 21, 2013
Program – New Program Class
Currently a program is loaded by the main window class and given to the edit box using the set plain text access function. This will set the text document of the edit box, which causes a document changed signal. When this signal is processed, the edit box updates the program unit that is attached. Because the text cursor is not valid during this operation, the recreated text from the program model cannot be inserted into the text document (the first of two issues mentioned in the last post).
One way to resolve this issue is to load the program directly into the program unit bypassing the edit box. The program generates signals that lines have been changed by line number. The edit box receives these signals, retrieves the recreated text for the lines from the program and puts them into the document. Special handling is needed for lines that contain errors since there would be no code for these lines to recreate text from.
Eventually, when support for subroutines and functions is added, a loaded program will consist of a number of program units, one for the main routine and one for each of the subroutines and functions. The program will consist of a list of program units. When a program is loaded, the load routine will create a new program unit for each routine or function. Likewise for the save routine, which accesses the list of program units.
The list of program units will be contained in a new program class. This class will contain the routines for loading and saving programs. The current program file path name will also be contained in this class. New source and header files were added for the new program class. To start simple, the new class only contains the program file path. Since the file path is included in the applications settings, save and restore settings routines were also implemented.
[commit 17f3e958ed]
One way to resolve this issue is to load the program directly into the program unit bypassing the edit box. The program generates signals that lines have been changed by line number. The edit box receives these signals, retrieves the recreated text for the lines from the program and puts them into the document. Special handling is needed for lines that contain errors since there would be no code for these lines to recreate text from.
Eventually, when support for subroutines and functions is added, a loaded program will consist of a number of program units, one for the main routine and one for each of the subroutines and functions. The program will consist of a list of program units. When a program is loaded, the load routine will create a new program unit for each routine or function. Likewise for the save routine, which accesses the list of program units.
The list of program units will be contained in a new program class. This class will contain the routines for loading and saving programs. The current program file path name will also be contained in this class. New source and header files were added for the new program class. To start simple, the new class only contains the program file path. Since the file path is included in the applications settings, save and restore settings routines were also implemented.
[commit 17f3e958ed]
Sunday, December 15, 2013
Edit Box – Recreated Line Replacement
When a program line is changed or inserted, the line should be recreated and the recreated text should replace the line entered in the edit box. To accomplish this, the program model class will send a signal with the line number of a line that is changed or inserted. The edit box will receive this signal, retrieve the recreated text for the line, and replace the text of the line with this recreated text.
The new program changed signal was added to the program model class. The update line routine was modified to send this signal when a line is changed or inserted. When a line is changed, the actual code of the line may not have changed, for example, if spaces were added or removed, or the case of keywords was changed. In this case, the program code will not be modified, but this signal still needs to be sent so that the edit box reflects the correct recreated program code.
The new program changed slot routine was added to the edit box class, which starts be retrieving the recreated text for the changed line. A text cursor is obtained for the edit box and its position is set to the beginning position of the block containing the line. The cursor is moved to the end of the block keeping the anchor at the beginning, which selects the entire line. The recreated text is inserted at the cursor and since text is selected, the selected text is replaced.
The program model line text routine was modified to return a null string if the line has an error. The program changed slot routine does not replace the text if the text is null indicating that the line has an error. This required the recreator class recreate routine to be modified where the output string is initialized to an empty string (a pair of double quotes) instead of being cleared. Clearing a string creates a null string, and a null string is not quite the same as an empty string (a null string is empty but an empty string is not null).
When text is replaced in the document using a text cursor, document changed and cursor moved signals are generated from the document. These signals need to be ignored when the line is being recreated, otherwise an infinite loop occurs because the document change signal updates the program, which generates another program changed signal, an so on. A flag was added to the edit box and is set before replacing text and cleared afterward. The document changed and cursor moved slot routines were modified to do nothing if this flag is set.
There are two unresolved issues resulting from these changes. The first issue occurs during the initial loading of a program. As a new program is being loaded, each line added to the program should be recreated to the edit box. However, this cannot occur because until the program has been loaded into the document of the edit box, the text cursor is not valid, so can't be used to replace text. The second issue occurs when a line is replaced with recreated text; extra undo commands are added to the undo stack.
[commit 4b34bd2dde]
The new program changed signal was added to the program model class. The update line routine was modified to send this signal when a line is changed or inserted. When a line is changed, the actual code of the line may not have changed, for example, if spaces were added or removed, or the case of keywords was changed. In this case, the program code will not be modified, but this signal still needs to be sent so that the edit box reflects the correct recreated program code.
The new program changed slot routine was added to the edit box class, which starts be retrieving the recreated text for the changed line. A text cursor is obtained for the edit box and its position is set to the beginning position of the block containing the line. The cursor is moved to the end of the block keeping the anchor at the beginning, which selects the entire line. The recreated text is inserted at the cursor and since text is selected, the selected text is replaced.
The program model line text routine was modified to return a null string if the line has an error. The program changed slot routine does not replace the text if the text is null indicating that the line has an error. This required the recreator class recreate routine to be modified where the output string is initialized to an empty string (a pair of double quotes) instead of being cleared. Clearing a string creates a null string, and a null string is not quite the same as an empty string (a null string is empty but an empty string is not null).
When text is replaced in the document using a text cursor, document changed and cursor moved signals are generated from the document. These signals need to be ignored when the line is being recreated, otherwise an infinite loop occurs because the document change signal updates the program, which generates another program changed signal, an so on. A flag was added to the edit box and is set before replacing text and cleared afterward. The document changed and cursor moved slot routines were modified to do nothing if this flag is set.
There are two unresolved issues resulting from these changes. The first issue occurs during the initial loading of a program. As a new program is being loaded, each line added to the program should be recreated to the edit box. However, this cannot occur because until the program has been loaded into the document of the edit box, the text cursor is not valid, so can't be used to replace text. The second issue occurs when a line is replaced with recreated text; extra undo commands are added to the undo stack.
[commit 4b34bd2dde]
Saturday, December 14, 2013
Edit Box – Lines Changed Signal
The edit box was using the lines changed signal to notify the program model instance when lines changed in the document that included the starting line number, the number of lines deleted and the number of lines inserted and the text of the lines. The program model update slot routine was connected to and processed this signal. Since the edit box can access the program model directly, this signal is not needed. The tester class is already accessing this routine directly.
The lines changed signal was removed and the two emits of this signal were changed to calling the update routine directly. The update routine was changed from a public slot to a normal public member function in the program model class. A few comments were added and word 'slot' was added to the comments of all the slot functions so that these would be easier to identify.
[commit c864bdde07]
The lines changed signal was removed and the two emits of this signal were changed to calling the update routine directly. The update routine was changed from a public slot to a normal public member function in the program model class. A few comments were added and word 'slot' was added to the comments of all the slot functions so that these would be easier to identify.
[commit c864bdde07]
Program – Error List Handling
The program model class was keeping a list of any errors detected when lines are translated. After a group of line changes were processed, if the list of errors changed, the entire error list was sent to the edit box via a signal. The edit box stored this list of errors and used it to generate its extra selection list, which is used to highlight the errors. Since the edit box now has access to the program model, it was no longer necessary for the edit box to keep a copy of this list.
So that the edit box can keep its extra selection list up to date, the program model was modified to send signals for when an error item has been inserted, changed, or removed. The connected edit box slots will update the extra selection list accordingly. When the program model is done updating the error list, it send a signal that the error list has changed. The connected edit box slot sets the extra selections to the base QPlainTextEdit class from updated extra selection list. The edit box no longer scans through the error list to generate its extra selection list.
The functionality for finding the next or previous error from the current cursor location was moved from the edit box class to the program model class. The current line number and column are passed to these routines, which are set to the next error line number and column upon returning along with a flag of whether the end or beginning of the program was passed so that a message can be issued.
A new routine was added to the program model class to handle when the current line is edited and contains an error. The error is shifted if the edit takes place before the error, or deleted if the edit takes place within the error. If the error list changes, the appropriate signals are sent to the edit box. This routine was not made a slot since the edit box can call it directly.
During the initial loading of the program, the text cursor in the edit box is not valid. Any errors in the program are sent as inserted error signals. Since the text cursor is not valid, the extra selections cannot be created (each extra selection contains a format and a cursor, which needs to be set to a valid cursor). The errors are temporarily saved in a list. After the text cursor becomes valid, extra selections are created from this list (which is then cleared).
The error list class used by the program model to hold the list of errors previously kept track of the first and last index affected by changes to the list. These indexes were used by the edit box to maintain its extra selection list. With the new direct change signals, these indexes are no longer needed, so the error list was changed to having a simple changed flag. The error list changed signal is only sent when this flag is set. The has changed access function was modified to clear this flag after it is read so a separate reset function is not needed.
Finally, the main window class status bar update slot was modified to receive the error message by an argument in the signal instead of it retrieving the message for the current line from the edit box, which retrieved the message from its copy of the error list. The routine sending this signal in the edit box class was modified to send the error message, which is retrieved from the program model via its list of errors.
[commit 19c5f06d0c]
So that the edit box can keep its extra selection list up to date, the program model was modified to send signals for when an error item has been inserted, changed, or removed. The connected edit box slots will update the extra selection list accordingly. When the program model is done updating the error list, it send a signal that the error list has changed. The connected edit box slot sets the extra selections to the base QPlainTextEdit class from updated extra selection list. The edit box no longer scans through the error list to generate its extra selection list.
The functionality for finding the next or previous error from the current cursor location was moved from the edit box class to the program model class. The current line number and column are passed to these routines, which are set to the next error line number and column upon returning along with a flag of whether the end or beginning of the program was passed so that a message can be issued.
A new routine was added to the program model class to handle when the current line is edited and contains an error. The error is shifted if the edit takes place before the error, or deleted if the edit takes place within the error. If the error list changes, the appropriate signals are sent to the edit box. This routine was not made a slot since the edit box can call it directly.
During the initial loading of the program, the text cursor in the edit box is not valid. Any errors in the program are sent as inserted error signals. Since the text cursor is not valid, the extra selections cannot be created (each extra selection contains a format and a cursor, which needs to be set to a valid cursor). The errors are temporarily saved in a list. After the text cursor becomes valid, extra selections are created from this list (which is then cleared).
The error list class used by the program model to hold the list of errors previously kept track of the first and last index affected by changes to the list. These indexes were used by the edit box to maintain its extra selection list. With the new direct change signals, these indexes are no longer needed, so the error list was changed to having a simple changed flag. The error list changed signal is only sent when this flag is set. The has changed access function was modified to clear this flag after it is read so a separate reset function is not needed.
Finally, the main window class status bar update slot was modified to receive the error message by an argument in the signal instead of it retrieving the message for the current line from the edit box, which retrieved the message from its copy of the error list. The routine sending this signal in the edit box class was modified to send the error message, which is retrieved from the program model via its list of errors.
[commit 19c5f06d0c]
Tuesday, December 10, 2013
Edit Box – Program Unit Access
The edit class box needs access to the program model, specifically to the program unit currently opened in the instance of an edit box. Eventually when subroutines and functions are implemented, any one of them could will be opened with their own edit box. There could be several edit box instances opened at any given time. Right now there is just a single program unit, the main routine, which will be opened in a single edit box instance.
To allow an edit box instance access to its program unit, a new pointer to a program unit instance (program model class) was added to the constructor of the edit box class. This pointer is stored in a new program unit pointer member variable. The connection of the line changes signal (from the edit box class), and the error list changed signal (from the program unit) are now made in the constructor of the edit box instead of the constructor of the main window class.
The constructor of the main window class was modified to create the program model first (the program unit for the main routine) before creating the edit box instance, which now requires the pointer to program unit instance that it will be editing.
[commit aa2125609a]
To allow an edit box instance access to its program unit, a new pointer to a program unit instance (program model class) was added to the constructor of the edit box class. This pointer is stored in a new program unit pointer member variable. The connection of the line changes signal (from the edit box class), and the error list changed signal (from the program unit) are now made in the constructor of the edit box instead of the constructor of the main window class.
The constructor of the main window class was modified to create the program model first (the program unit for the main routine) before creating the edit box instance, which now requires the pointer to program unit instance that it will be editing.
[commit aa2125609a]
Saturday, December 7, 2013
Program (Recreator) and GUI Integration
The recreator is fully integrated with the program model such that program lines can be converted back into text. When lines are entered into the program, the lines need to be recreated back to text and put into the edit box (the GUI), specifically into the text document of the edit box. Like the temporary program view (being used for debugging) is the viewer of the data held by the program model, the edit box is the viewer of the data contained in the document. The edit box also allows editing, so it is more than just a viewer.
The document of the edit box is really just the text representation of the program. The program model holds the actual data of the program. Ideally, the program model would be the document of the edit box and it would convert text to program code and back while editing. However, Qt does not have an abstract text document class from which a document sub-class could be built that would hold its data in another form like program code. The QTextDocument class is meant for text.
Alternatively, a new viewer could be designed that would allow all the text editing features (cut, copy, paste, undo, redo, etc.) like the QPlainTextEdit class that the edit box class is based on. Designing one would be quite an effort. Therefore, the edit box will the viewer for two data models at the same time, the text document (to allow text editing) and the program model (for holding the program code). The program model will be the master of the data, with the text document being updated as the program changes.
This implies that the edit box either own the program model with the program code or at least have easy access to it like via a pointer. The later approach will be used since the main window class will ultimately be the owner of the program. Eventually there will be a list of program models, one for the main routine and several for the subroutines and functions of the program. There will only be associated edit box instances when the main routine, subroutines or functions are open for editing.
Since the edit box will now have access to the program model, signals from the program model (for program changes) do not need to contain actual data. For instance, when a program line has changed, its recreated text is needed to update the text document. The signal could contain both the line number and text (already recreated). Looking at the edit box to document interface, when the document changes, only the position, number of characters removed and inserted are contained in the signal. The edit box must obtain the actual text changes by querying the document. So, when the program changes, only the line number will be sent and the edit box will request the recreated text from the program model.
The document of the edit box is really just the text representation of the program. The program model holds the actual data of the program. Ideally, the program model would be the document of the edit box and it would convert text to program code and back while editing. However, Qt does not have an abstract text document class from which a document sub-class could be built that would hold its data in another form like program code. The QTextDocument class is meant for text.
Alternatively, a new viewer could be designed that would allow all the text editing features (cut, copy, paste, undo, redo, etc.) like the QPlainTextEdit class that the edit box class is based on. Designing one would be quite an effort. Therefore, the edit box will the viewer for two data models at the same time, the text document (to allow text editing) and the program model (for holding the program code). The program model will be the master of the data, with the text document being updated as the program changes.
This implies that the edit box either own the program model with the program code or at least have easy access to it like via a pointer. The later approach will be used since the main window class will ultimately be the owner of the program. Eventually there will be a list of program models, one for the main routine and several for the subroutines and functions of the program. There will only be associated edit box instances when the main routine, subroutines or functions are open for editing.
Since the edit box will now have access to the program model, signals from the program model (for program changes) do not need to contain actual data. For instance, when a program line has changed, its recreated text is needed to update the text document. The signal could contain both the line number and text (already recreated). Looking at the edit box to document interface, when the document changes, only the position, number of characters removed and inserted are contained in the signal. The edit box must obtain the actual text changes by querying the document. So, when the program changes, only the line number will be sent and the edit box will request the recreated text from the program model.
Wednesday, December 4, 2013
Information Dictionaries – Improved Design
The design of the info dictionaries required the program model to create the instances for the additional info for the dictionary (in this case, the constant number and string dictionaries) and pass this instance for the creation of the dictionary. The program model owned and was responsible for these instances. This is possibly problematic because it did not guarantee that an additional info instance of the correct type was passed to the info dictionary.
The design was changed where new constant number and string classes derived from the base info dictionary class were added. In their constructors, the additional info instance is created of the correct type and they own the instance in the abstract info member pointer in the base class. A destructor was added to the base class to delete this instance, which required a virtual destructor in the abstract info class so that the derived info class destructor gets called.
While not needed until the run-time module is implemented, access functions for the arrays in the additional info of the constant string and number dictionaries were added. These functions simply call the access functions in the derived info classes. However, a type cast to the derived class is needed since the base class defines the additional info instance pointer as an abstract info class pointer.
[commit 451edd6346]
The design was changed where new constant number and string classes derived from the base info dictionary class were added. In their constructors, the additional info instance is created of the correct type and they own the instance in the abstract info member pointer in the base class. A destructor was added to the base class to delete this instance, which required a virtual destructor in the abstract info class so that the derived info class destructor gets called.
While not needed until the run-time module is implemented, access functions for the arrays in the additional info of the constant string and number dictionaries were added. These functions simply call the access functions in the derived info classes. However, a type cast to the derived class is needed since the base class defines the additional info instance pointer as an abstract info class pointer.
[commit 451edd6346]
Subscribe to:
Posts (Atom)