home *** CD-ROM | disk | FTP | other *** search
/ Windoware / WINDOWARE_1_6.iso / winutil / wshel202 / winshell.txt < prev    next >
Text File  |  1992-06-09  |  43KB  |  763 lines

  1.  
  2.  
  3.  
  4. WINDOWS SHELL
  5. FOR MICROSOFT WINDOWS 3.0
  6. VERSION 2.02
  7.  
  8.  
  9.  
  10.  
  11. by
  12.  
  13.  
  14.  
  15.  
  16. Greg McCain
  17.  
  18. June 5, 1992
  19.  
  20.  
  21.  
  22.  
  23.  
  24.  
  25.  
  26.  
  27.  
  28.  
  29.  
  30. Advisor: Charles Dana
  31. Computer Science Department
  32. School of Engineering
  33. California Polytechnic State University
  34. 199
  35.  
  36.  
  37. TABLE OF CONTENTS
  38.  
  39. ABSTRACT    iii
  40. LIST OF FIGURES    iv
  41. 0. INTRODUCTION    1
  42. I. FEATURES OF WINDOWS SHELL    2
  43. I.A. THE COMMAND LINE    2
  44. I.B. ALIASES    3
  45. I.C. ENVIRONMENT VARIABLES    3
  46. I.D. ACTION BAR    4
  47. I.E. FILE LIST WINDOW    5
  48. I.F. EXTERNAL COMMANDS    6
  49. I.G. SHELL COMMANDS    8
  50. II. DESIGN OF WINDOWS SHELL    9
  51. II.A. GRAPHICAL OBJECTS OF THE WINDOWS SHELL    9
  52. II.A.1 The WSTDIO Window    10
  53. II.A.1.a Pertinent Data    10
  54. II.A.1.b Painting The Display    12
  55. II.A.1.c Obtaining Standard I/O Input    12
  56. II.A.1.d Marking Text With The Mouse    14
  57. II.A.1.e Command History    14
  58. II.A.1.f Wstdio Message Directory    15
  59. II.A.2 The ACTNBAR Window    16
  60. II.A.3 The WINSHELL Window    17
  61. II.B LOGICAL MODULES FOR COMMAND INTERPRETATION    17
  62. II.B.1 The Command Prompt Path    18
  63. II.B.2 The Action Button Command Path    19
  64. II.B.3 The ALIAS.C Module    20
  65. II.B.4 The INTERP.C Module    21
  66. III. DEVELOPMENT PROCESS    22
  67. APPENDIX A. THE EXTERNAL COMMANDS DLL INTERFACE    25
  68. APPENDIX B. FUTURE ENHANCEMENTS    27
  69.  
  70.  
  71. ABSTRACT
  72.  
  73. This document discusses the purpose, features, and design of a command line 
  74. shell for Microsoft Windows 3.0/3.1, the Windows Shell.  The Windows Shell allows 
  75. users to launch both DOS and Windows programs from a command line environment, 
  76. as well as perform disk maintenance operations such as copy, deleting, and moving 
  77. files.
  78. The Windows Shell introduces several new features to the typical command shell 
  79. environment, including interactive editing of Aliases and Environment variables,  an 
  80. "Action Bar" to quickly execute a command, the ability to set "permanent" options in 
  81. shell commands, and customizable fonts and screen colors.
  82. The Windows Shell is designed according to an object oriented paradigm.  It is 
  83. comprised of three major graphical objects, which includes the main WINSHELL 
  84. window, the WSTDIO window, and the ACTNBAR window.  The task of interpreting and 
  85. executing commands is done by the INTERP (or interpreter) module. 
  86. A key design element of WINSHELL is the WSTDIO window, which provides a 
  87. functional interface much like the normal C standard I/O functions for DOS and UNIX.  
  88. Clients using this window need not implement their own task message loop; the 
  89. WSTDIO window will handle messages transparently when it is invoked.  This frees 
  90. the client from having to deal directly with Windows messages, allowing it to focus 
  91. on the task at hand. 
  92.  
  93.  
  94. LIST OF FIGURES
  95.  
  96. FIGURE 
  97.  
  98. 1-1 The Command Line    2
  99. 1-2 The Alias Editor    3
  100. 1-3 The Environment Editor    4
  101. 1-4 Action Button Configuration    5
  102. 1-5 File List Window    5
  103. 1-6 Internal Commands    6
  104. 2-1 Graphical Object Design    9
  105. 2-2 Command Interpretation Paths    18
  106.  
  107.  
  108. 0. INTRODUCTION
  109. The Windows 3.0 environment provides a graphical user interface for executing programs 
  110. and file management.  While this is great for naive users, the graphical interface can be 
  111. cumbersome to experienced users.  The Windows Shell is intended for experienced users, 
  112. providing quick manipulation of files and directories, as well as executing Windows and DOS 
  113. programs.  
  114. The Windows Shell was developed on a IBM PS/2 model 70, running at 16 MHz and using a 
  115. VGA display.  The shell has also been succesfully run on systems running Windows in SVGA 
  116. and EGA mode.
  117. This document is divided into three main sections.  The first presents the Windows Shell as 
  118. seen by the user.  It discusses the feature of Windows Shell, how to access them, and what they 
  119. can do for a user.  The second section presents the Windows Shell as seen by the programmer.  
  120. This includes a discussion of the graphical objects seen in the Window Shell, as well as a 
  121. discussion of how the command interpretation works.  The last section discusses the 
  122. development process of the Windows Shell.  It describes the problems encountered in both 
  123. design and implementation of the Windows Shell, and any other noteworthy considerations that 
  124. were involved in the development process.
  125. The document also provides the specifications of the DLL External Commands interface in 
  126. Appendix A.  The interface is a major consideration when developing external commands for use 
  127. with the Windows Shell.
  128. I. FEATURES OF WINDOWS SHELL
  129.  
  130.  
  131. I.A. THE COMMAND LINE
  132.  
  133. figure 1-1
  134. The Windows Shell provides a command line interface to the windows environment.  The 
  135. command line is similar in appearance to a DOS shell, with the following additions:
  136. - The Windows Shell prompt allows for text marking, cutting, and pasting.
  137. - The Windows Shell offers the "Action Bar" seen on the left side of the shell (see figure 
  138. 1-1.)
  139. - The Windows Shell allows the user to configure the color and font used by each 
  140. instance of the shell.
  141. - The Windows Shell recognizes the '&' character to run a windows program minimized.
  142. - The Windows Shell expands partial file names entered on the command line into the 
  143. entire file name by pressing the TAB key.  If more than one file matches the partial 
  144. name, the user will be given a choice of files.
  145.  
  146. I.B. ALIASES
  147. Windows Shell allows the user to define aliases.  Aliases allow long or complex commands 
  148. to be abbreviated for quick access.   The Alias Editor  (see figure 1-2 below) is provided for quick 
  149. viewing and editing of aliases.
  150.  
  151.  
  152. figure 1-2
  153.  
  154. I.C. ENVIRONMENT VARIABLES
  155.  
  156. Windows Shell provides the Environment Editor (see figure 1-3 below) for quick viewing and 
  157. editing of environment variables.  Windows 3.0 provides environment variables similar to that 
  158. of DOS and UNIX.  Unfortunately, the environment variables in the Windows Shell are not 
  159. inherited by the programs it executes.  Each new task gets a copy of the environment that was 
  160. recorded when windows was started.  However,  modifying the local environment variables in a 
  161. given application can and will effect the application if it uses them the variables.  In the 
  162. Windows Shell, it is sometimes useful to change the path or prompt environment variables.  
  163. Hopefully future versions of Windows will provide inherited environment variables.
  164.  
  165.  
  166. figure 1-3
  167.  
  168. I.D. ACTION BAR
  169. The action bar is a column of eight buttons along the left side of the shell (see figure 1-1)  It 
  170. allows users to execute a predefined command at the press of a button.  The following window 
  171. is used to configure the action bar:
  172.  
  173.  
  174. figure 1-4
  175. The Button Configuration Window allows the users to define the text seen on the buttons and 
  176. the command that is carries out.  The user can access text marked on the Windows Shell using 
  177. the '=' character.  The '=' character is replaced by a string containing the marked text when the 
  178. command is executed.  This makes it easy for a user to mark a block of text and perform actions 
  179. upon it, such as deleting or editing files.  For example, to have a button which will delete any 
  180. marked files, use the string:
  181. del =
  182. in a "button command" box, as seen in figure 1-4.
  183.  
  184. I.E. FILE LIST WINDOW
  185.  
  186.  
  187. figure 1-5
  188. The File List window lists the files in the current directory.  It can be activated using the 
  189. View-File_List menu command, or by pressing Cntrl+F.  The File List allows you to select 
  190. multiple files in the current directory, and to either move, copy, or delete the highlighted files.  It 
  191. allows you to copy the names of the selected files to the command line.  This can be very useful 
  192. when you want to perform a command only on certain files in a directory, but don't know their 
  193. names or you don't want to type them in.  Future version of the Windows Shell might include the 
  194. option to sort the files in the File List Window by size, date, or extension.
  195.  
  196. I.F. EXTERNAL COMMANDS
  197. Most shells provide shell commands which are built into the shell itself.  The Windows Shell 
  198. provides only the most basic shell commands, including: change directory (cd), make directory 
  199. (md), and remove directory (rd).  The rest of the usual shell commands are implemented as 
  200. "external commands".
  201. Each external command is a Windows 3.0 Dynamic Link Library (DLL).  The Windows Shell 
  202. uses a Windows 3.0 DLL loading function to load and run external commands.  Each external 
  203. command must provide a set of functions to execute the command, show an about box, and 
  204. show an options box.
  205. The windows shell provides the "external commands window" to view and configure external 
  206. commands (see figure 1-6).
  207.  
  208.  
  209. figure 1-6
  210.  
  211. The options box allows the user to set "permanent" options on a command that will be invoked 
  212. each time the command is run.  The about box allows the user to view a window which should 
  213. contain useful information on what the command does and how to use it.
  214. "External commands" are implemented as DLL's to allow new commands to be written without 
  215. modification to the shell.  The format of a DLL is such that a normal DOS shell program can 
  216. easily be ported to run under the shell.
  217.  
  218. I.G. SHELL COMMANDS
  219. The user accessible commands contained in Winshell, or Shell Commands, are as follows:
  220. cd <dirname>     : Changes current directory to directory specified in dirname.
  221. md <dirname>     : Creates a directory as specified in dirname.
  222. rd <dirname>     : Deletes a directory as specified in dirname.
  223. <DRIVE> :    : Changes current disk specified to drive specified in DRIVE.
  224. ps    : Lists currently running processes.
  225. min <task_name>: Minimizes program with caption matching task_name.
  226. max <task_name>: Maximizes program with caption matching task_name.
  227. kill <task_name>: Closes program with caption matching task_name.
  228. exit    : Exits current Windows Shell.
  229. exitwin    : Exits Windows.
  230.  
  231.  
  232.  
  233. II. DESIGN OF WINDOWS SHELL
  234. The design of Windows Shell has been broken down into the many separate modules, both 
  235. graphical and logical.  To simplify the explanation of the design, this section is divided into two 
  236. main parts.  The first part will discuss the graphical objects of the Windows Shell.  This includes 
  237. how the individual windows and buttons are designed, how they function, and they the 
  238. communicate with one another.  The second part of this section will discuss the modules used 
  239. in command interpretation.  This section will involve a detailed trace of the flow of control for the 
  240. interpretation of a command.
  241.  
  242. II.A. GRAPHICAL OBJECTS OF THE WINDOWS SHELL
  243. The windows shell has been divided into three main graphical objects, as seen in figure 2-1.  
  244. Each of the objects is implemented in a separate C source file, and communicate with each 
  245. other via Windows messages.  Although the "Main" window as seen in figure 2-1 is at the top of 
  246. the hierarchy, controlling the other two windows, this explanation will be more clear of it starts 
  247. from the bottom.
  248.  
  249.  
  250. figure 2-1
  251.  
  252. II.A.1 The WSTDIO Window
  253. By far the most important window in the shell is the WStdio window.  It is responsible for 
  254. supplying the primitives such as reading and writing characters to a standard I/O type device.  
  255. To do this in a windowing environment, is requires creating a "virtual" standard I/O display.  
  256. This display is represented by a data structure, who's main element is an array of characters 
  257. which represent the characters displayed on the screen.  When characters are written to the 
  258. WStdio window, they a first copied into this character array, and then are actually displayed on 
  259. the screen.
  260. II.A.1.a Pertinent Data
  261. To better understand the implementation of the WStdio window, let's look at the elements of 
  262. the Display data structure as defined by the WStdio window.  The first feature is the array of 
  263. characters representing the display (This is referred to as the "LineBuf".)  Second, are a set of 
  264. elements used to track the state of the display:
  265.  
  266. int  iTopLine;            // Index of the current top line in the LineBuf
  267. int  iBottomLine;            // Index of the current bottom line in the
  268.                     // LineBuf
  269. int  TotalLineCount;        // the current number of line in the LineBuf
  270. int  CurCharOffset;        // offset into LineBuff of current position
  271.  
  272. These first four elements are what control current state of LineBuff itself.  LineBuff is an array of 
  273. characters which is logically divided into fixed length lines.  The actual number of lines 
  274. available in LineBuff is a constant at compile time, but is typically much larger than the number 
  275. of lines that will currently fit in the window on screen.  (This surpluss of lines in memory is used 
  276. as a scrollback buffer, as will be discussed later.)  The indices to the top and bottom lines are 
  277. used because LineBuff is a circular buffer.  These indices always point to the portion of the 
  278. LineBuff that is currently displayed on the screen.  The TotalLineCount variable is the absolute 
  279. number of lines in LineBuff.  CurCharOffset is the absolute position in LineBuff at which the next 
  280. character to be displayed will be written.
  281. The next set of elements is used when actually outputting lines to the screen.  Their uses 
  282. and intentions are explained in the comments.
  283.  
  284. int  nLinesOnScreen;          // the number of lines fitting on the window
  285.                     // in it's current size
  286. int  nCurLineOnScreen;        // the current line the cursor is on in window
  287.                                    // starting from 1 (NOT 0!)
  288. int  yChar,                       // height in pixels of a line
  289. int  xCaretPos;                  // distance (in pixels) from the left hand
  290.                                    // side of the screen that the caret is at
  291. When the user activates the scrollback feature by using the vertical scrollbar or the PAGE 
  292. UP/PAGE DOWN keys, these variables are initialized and used to track the positioning of the 
  293. scrollback:
  294.  
  295. BOOL     isScrolling;        // True is window is in a scrolling back state
  296. int      TopScrollLineNum,    // index of the top line in the LineBuff  
  297.                     // during a scrollback
  298. Another noteworthy element in the Display data structure is used to expedite the actual 
  299. outputting of lines on the screen.
  300.  
  301. int  nUnpurgedLines,              // # of line waiting to be written to
  302.                         // the screen
  303. The WStdio window does not display a line of text immediately when it is received.  If 
  304. possible, the window will wait until a predefined constant number of lines come in before it 
  305. actually displays the lines.  This can dramatically increase the rate at which lines are displayed 
  306. on the screen.  The nUnpurgedLines variable counts the number of lines waiting to be 
  307. displayed.  In the current implementation (vers 1.3), this can lead to lines of text not being 
  308. displayed while the task performs some lengthy operation.  This should be corrected in later 
  309. versions by having a timer periodically purged any unpurged lines.
  310. Finally, there are elements used to track how the user has highlighted any text in the WStdio 
  311. window:
  312.  
  313. WORD wSelectState;
  314. RECT rectInversion;
  315. The wSelectState variable can be in three states: one indicating there is no highlighted 
  316. rectangle, when all bits are turned off.  The second, SS_SELECTING,  indicates the user is 
  317. currently marking a rectangle. The third, SS_RECTSELECTED, indicates that a rectangle is 
  318. currently highlighted.  When in third state, the rectInversion variable will hold the coordinates of 
  319. the highlighted rectangle.
  320. II.A.1.b Painting The Display
  321. Windows dictates that all screen I/O should be done in response to a WM_PAINT message. This 
  322. message informs a window that it needs to repaint a portion of it's client area, and supplies the 
  323. window with the coordinates of a rectangle it needs to repaint.  Thus upon receiving this 
  324. message, the WStdio window calculates what lines need to be painted, and paints them.  When 
  325. a client module requests the WStdio window to display a line (via the DM_PUTS message), the 
  326. WStdio window only has to copy the line into it's internal Linebuff, and then invalidate the 
  327. portion of the window that will be effected by the new line.  By invalidating a portion of the 
  328. window, Windows will generate a WM_PAINT message, and the proper area of the window will 
  329. be repainted, reflecting the new line to be displayed.  This scheme mandates that all screen I/O 
  330. be done in response to the WM_PAINT message, which keeps all screen I/O consistent and in a 
  331. fixed area in the program.
  332. II.A.1.c Obtaining Standard I/O Input
  333. Perhaps the most interesting feature in the implementation of the WStdio window is how it 
  334. obtains input from the user.  To provide a function like getstr(), which does not return a value 
  335. until the user presses the ENTER key, the Windows message loop had to be placed inside the 
  336. getstr() function.  This allows other processes to run and respond to their own inputs while the 
  337. WStdio window is waiting for input.
  338. When a client module sends a DM_GETS message to get a string from the WStdio window, 
  339. the WStdio window calls it's own DisplayGetStr() function.  This function first positions the caret 
  340. at the appropriate position, and then falls into a message loop.  Inside this message loop, the 
  341. function monitors the incoming messages looking for the ENTER key to be pressed, in which 
  342. case it will fall out of the loop and return the text that was entered.  The function also monitors 
  343. the incoming messages for keys like the arrow keys, in which case it will invoke the command 
  344. history, and for WM_CLOSE message.  If a WM_CLOSE message comes in, the DisplayGetStr 
  345. function exits the message loop and returns a value indicating that the function failed.
  346. With this method of implementing the message loop in the input function, the client modules 
  347. are simplified since they need not implement a separate message loop.  The main module of 
  348. the Windows Shell, WINSHELL.C, in fact does not use a message loop.  Instead, it falls into a 
  349. loop that might be expected out of a UNIX type command, in which it displays a command 
  350. prompt, gets an input string, interprets it, and executes the appropriate action.  It does not use a 
  351. message loop at all, like most other Windows WinMain functions have to do.
  352. In order to obtain input from the user, the WStdio window uses a Windows edit control.  
  353. When the user sees a prompt at which he or she can type, that prompt is actually inside a 
  354. separate edit control, and not in the WStdio window itself.  The edit control is always positioned 
  355. at the end of the last character entered, much as a caret would be.  This way, however, leaves 
  356. much of the work of obtaining key-presses and displaying characters to the edit control.  It also 
  357. helps provide the standard controls a user might expect from an input prompt,  such as marking 
  358. text and replacing text.  The user will see an interface consistent with other programs which use 
  359. edit controls. If future versions of Windows change the behavior of these controls, the Windows 
  360. Shell will adapt automatically.
  361. II.A.1.d Marking Text With The Mouse
  362. The job of marking text on the display is quite simple.  The WStdio window responds to a 
  363. WM_LBUTTONDOWN message (indicating the left mouse button is being pressed,) by obtaining 
  364. a mouse capture.  This forces Windows to send all subsequent mouse messages to the window 
  365. obtaining the capture.  While in this state, the WStdio window then responds to all 
  366. WM_MOUSEMOVE messages by inverting a rectangle between the position where the mouse 
  367. was originally pressed and the current position.  When the left mouse button is released, the 
  368. capture is also released, and the rectangle is left highlighted.
  369. The portion of highlighted text can now be accessed by both the WStdio window itself, and 
  370. by client windows via the DM_GETMARKEDTEXT message, which is supplied by the wstdio.c 
  371. module.  The function GetMarkedText(), in the wstdio.c module,  is called in response to this 
  372. message.  It is responsible for determining what characters are actually marked and copying 
  373. them to a buffer.  This is no simple task when proportionally spaced fonts are in use.  The 
  374. function must navigate LineBuff and determine the actual length in pixels of each character in 
  375. the buffer.  It then compares this to coordinates that are marked in the screen, and can 
  376. determine what characters are actually marked.
  377. II.A.1.e Command History
  378. The WStdio is also responsible for providing a command history.  Whenever the user enters 
  379. a command at the command prompt, the WStdio window records the command entered with the 
  380. CommandHistory() function.  This function manages a simple queue of a constant size.  When a 
  381. user enters a command, it is added at the end of the queue, and the first item in the queue is 
  382. discarded if there are more than the constant limit of items in the queue.  When the user 
  383. presses the up and down arrow keys, the WStdio window responds by displaying items from 
  384. this queue on the command line.
  385. II.A.1.f Wstdio Message Directory
  386. The following is a list of messages that the WStdio window provides for client modules.  The 
  387. messages are defined and processed explicity by the WStdio window. The client can invoke 
  388. these by using the Windows SendMessage() function to send the desired message.
  389.  
  390. DM_PUTS    
  391. Writes a string to the display.
  392.  
  393. DM_GETS      
  394. Gets a string from the display.
  395.  
  396. DM_CLEAR     
  397. Clears the Wstdio window.
  398.  
  399.  
  400. DM_SETFONT    
  401. Sets the font the Wstdio window will use.
  402.  
  403. DM_GETNUMCOLUMNS 
  404. Returns the approximate number of columns on the display. A column is space 
  405. enough for about 12 of the widest characters in the current font, and a trailing tab.  
  406. This feature essentially indicates the number of file names that can be displayed 
  407. on one line.
  408.  
  409. DM_GETMARKEDTEXT 
  410. Returns a global handle to memory block containing marked text.  This memory 
  411. must be freed by the user.
  412.  
  413. DM_SETMORE 
  414. Turns the more feature on and off.    When this feature is turned on, the WStdio 
  415. will automatically display a ----more---- at the bottom of the screen after the last 
  416. number of lines displayed has filled up the screen.  The more feature is 
  417. automatically turned off when after DM_GETS message is sent.
  418.  
  419. II.A.2 The ACTNBAR Window
  420. The actnbar window provides the column of user-configurable push buttons along the left 
  421. hand side of the Windows Shell.  Each push button provides a shortcut way for the user to 
  422. execute a command.  The ACTNBAR window is actually a rectangular window surrounding the 
  423. set of push buttons.  Associated with each push button is a caption and a command.  The 
  424. caption is the text that is displayed in the button on the screen;  the command is the command 
  425. string which will be invoked when the button is pressed.
  426. When the ACTNBAR window is created, it creates it's push button children and initializes 
  427. them to defaults saved in the WINSHELL.INI configuration file.  It's job thereafter is to report to 
  428. it's parent window whenever one of it's buttons has been pushed, passing the parent window 
  429. the command string to be executed.  
  430. This is accomplished using the standard Windows WM_COMMAND message.  This 
  431. message is sent to the ACTNBAR window whenever one of it's children push buttons is pressed.  
  432. The ACTNBAR processes this message by first setting it's own caption (the string of text 
  433. appearing in the caption bar of a window, which is not visible in the ACTNBAR window) to the 
  434. command string of the child being pressed.  The ACTNBAR window then relays the same 
  435. WM_COMMAND message to the parent (or client) window.  When the parent receives the 
  436. WM_COMMAND from the ACTNBAR window, it reads caption text of the ACTNBAR window, and 
  437. executes the command string contained therein.
  438. Note that this logical command path is different from that of the other command interpretation 
  439. path.  A command executed in response to the ACTNBAR being pressed is interpreted and 
  440. executed in response to the WM_COMMAND message, and is not obtained via the command 
  441. interpretation loop in the WinMain function.  This is discussed more in section II.B.2.
  442. Also contained inside the ACTNBAR.C module is action button configuration dialog box.  This 
  443. allows users to configure both the caption and command of each action button.
  444.  
  445. II.A.3 The WINSHELL Window
  446. This window is the main window of the application, and controls the other windows as seen 
  447. in figure 2-1.  The graphical duties of this window are quite simple.  The WINSHELL window 
  448. creates and manages two children windows, the WSTDIO window and the ACTNBAR window.  
  449. Both children are sized so that together they fill up the client area of the WINSHELL window.  
  450. Thus the WINSHELL window reacts to changes in it's own size by repositioning and re-sizing 
  451. the children so they always fill it's client area.
  452. WINSHELL also provides the menu bar as seen at the top of the window.  It must respond to 
  453. menu messages and execute the appropriate functions.  These functions include changing the 
  454. font in the WSTDIO window, changing the colors of the Windows Shell, and popping up the 
  455. various configuration dialog boxes.  These duties mostly involve sending a single message to 
  456. the appropriate window to perform the task.  In this way, the WINSHELL window serves more as 
  457. a message router for it's children than anything else.
  458. If this seems too simple, it's because the graphically oriented tasks are only part of the 
  459. tasks the WINSHELL window must perform.  It's main task, that of command interpretation, is 
  460. discussed in the next section.
  461. II.B LOGICAL MODULES FOR COMMAND INTERPRETATION
  462. This section discusses the modules involved in command interpretation.  Now that the 
  463. relationship of the 3 main windows has been defined, the matter of understanding command 
  464. interpretation will be much easier.  First let's look at the design of the command interpreter, as 
  465. seen in figure 2-2.
  466.  
  467.  
  468. figure 2-2
  469.  
  470. The flow of control in figure 2-2 moves left to right.  The diamond shaped modules indicate 
  471. modules which are returning user input.  The circular modules perform some logical operation 
  472. on the input data, and the square boxes will be the end result of the command.
  473. There exist two paths in which a command can be executed by the shell.  The first is by the 
  474. user entering a command at the command prompt.  This is accomplished by looping for user 
  475. input, interpreting it, and executing it.  The second path is invoked via the user pressing an 
  476. Action Button; the command is executed immediately in response to the ACTNBAR's message.
  477. II.B.1 The Command Prompt Path
  478. Command interpretation begins in the WINSHELL.C module.  This module contains the 
  479. WinMain function, which is the entry point of a Windows application.  After performing it's 
  480. initializations and creating it's child windows, the WINSHELL.C module falls into the command 
  481. interpretation loop as follows:
  482. do
  483. {
  484.    DisplayPrompt (hwndDisplay); 
  485.  
  486.    // exit if display says to
  487.    if (dgets (hwndDisplay, szCmdLine, MAX_COMMAND_LENGTH)== -1)
  488.       break;  
  489.  
  490.    ExpandAliasString (szCmdLine, MAX_COMMAND_LENGTH);
  491.  
  492.    iInterp = InterpretCommand (hwndWinShell, hwndDisplay, szCmdLine);
  493. } while (bContinue && iInterp != -1);
  494. The first function in the loop displays the command prompt on the WStdio window.  The next 
  495. job is to obtain a line of text from the user, which is accomplished by the dgets() macro.  Dgets() 
  496. is a macro supplied by the WStdio module which sends the DM_PUTS message to a WStdio 
  497. window.  Note that if dgets() returns -1, it means the user has closed the window, and the loop 
  498. must be exited.  After the user input has been obtained, the input is passed to the 
  499. ExpandAliasString() function defined in the ALIAS.C module, which will expand any aliases found 
  500. in the string.  Finally, the string is passed to the InterpretCommand function in the INTERP.C 
  501. module.  It is in this module that the string is parsed and executed.
  502. Note that commands retrieved from the command line are not given a chance to expand the 
  503. marked text symbol by using the WSTDIO.C module.  The marked text symbol is provided so 
  504. that the symbol in the command string is replaced by the text marked in the WStdio window.  
  505. Although this would be a desirable feature, a minor design flaw stopped me from implementing it 
  506. (see section 3).
  507. II.B.2 The Action Button Command Path
  508.  The ACTNBAR.C module can instigate a command by sending a message to the WINSHELL 
  509. window.  The WINSHELL responds by executing the command using the flow of control as seen 
  510. in figure 2-2.  The following code is executed in response to such a message:
  511. GetWindowText (LOWORD (lParam), szCmdLine, MAX_COMMAND_LENGTH);
  512. ExpandAliasString (szCmdLine, MAX_COMMAND_LENGTH);
  513. ExpandMarkedText  (szCmdLine, MAX_COMMAND_LENGTH);
  514. InterpretCommand  (hwndWinShell, NULL, szCmdLine);
  515. As you can see, the code is very similar to that used in the command interpretation loop in 
  516. the previous section.  The only difference is that commands executed in response to the 
  517. ACTNBAR module get to use the ExapandMarkedText() function, which expands the marked text 
  518. symbol into the text currently marked on the WStdio window.  Also, the command executed in 
  519. response to the ACTNBAR module are passed a WStdio window handle of NULL, prohibiting 
  520. them from outputing any data to the WStdio window.  This is to safeguard the current text in the 
  521. Wstdio window, which might otherwise be overwritten by the command executed by the button. 
  522. II.B.3 The ALIAS.C Module
  523. The ALIAS.C module provides alias expansion for command strings.  It is implemented as a 
  524. Windows DLL, which provides the following advantages.  For all instances of the Windows Shell 
  525. running, they all share a common ALIAS module.  This means that if an alias changed in one 
  526. shell, it is changed for all shells, keeping all aliases consistent.  This methods also expedites 
  527. the loading process of Windows Shell, because the default aliases only have to be read from 
  528. disk once, for the first module.
  529. The implementation of the ALIAS module is quite simple.  It maintains a dynamically 
  530. resizable array of elements.  Each element contains an alias name and an alias value.  The 
  531. module provides a dialog box for adding and deleting aliases, and the ExpandAliasString() 
  532. function, which takes a string and expands any aliases within it.
  533.  
  534. II.B.4 The INTERP.C Module
  535. The third and perhaps most important module used in command interpretation is the 
  536. INTERP.C module.  This module is responsible for parsing a command line, determining what 
  537. type of command it is, and executing the command accordingly.  There are three types of 
  538. commands among which the module must discriminate.
  539. The first type of command the INTERP module looks for are shell commands.  These are 
  540. commands who's code is kept inside the Window Shell.  This is accomplished by a large switch 
  541. statement that checks if the requested command is amongst those known to be external 
  542. commands.  If INTERP determines that command is a shell command, it merely calls the 
  543. function associated with the shell command.  All shell commands are contained in the 
  544. COMMANDS.C module.  These commands are so trivial they will not be discussed.
  545. The second type of command the INTERP module looks for are external commands.  These 
  546. commands are implemented as Windows DLLs.  The INTERP function searches the default 
  547. directory for files who's names match the specified command name and ending with the .WS 
  548. extension.  If a corresponding external command file is found, the DLL is loaded and control is 
  549. passed to it.  The specifications for an external command are outline in appendix A.
  550. The third type of command the INTERP module looks for are executable files.  Windows 
  551. provides the WinExec() function which performs this function.  However, the WinExec function 
  552. only searches for executable files with and extension of .EXE.   DOS on the other hand, allows a 
  553. different type of executable file to end with the .COM extension.  Also, Windows provides a DOS 
  554. shell configuration file which can also be executed, who's extension is .PIF.  In order all these 
  555. types of files to be executed, the INTERP module first gives the WinExec function a crack at 
  556. executing it.  If this fails, INTERP then searches the PATH for a file who's name matches the 
  557. command name and ends in either a .COM or a .PIF.  If either of these are found, an explicit file 
  558. name and path is created and passed to the WinExec function, which will then execute the 
  559. appropriate files.
  560. If none of the above types of commands are found to match the specified command, an 
  561. "unknown command" message is displayed in the WStdio window.
  562.  
  563. III. DEVELOPMENT PROCESS
  564. This section discusses some of the problems and considerations that I ran into while 
  565. designing and implementing the Windows Shell.   On a general note, it seems that many of the 
  566. major design issues I originally set out to implement worked quite well.  From the onset, I 
  567. intended to create a Windows application that did not have a message loop in WinMain function.  
  568. I wanted the main module worry more about the matters of command interpretation than 
  569. windowing.   The message loop was to be hidden in the screen I/O functions, which is how it is 
  570. now implemented.   Also, the idea of having external commands implemented as DLLs worked 
  571. out great.  I really had no idea if either of these ideas were feasible when I started.
  572. Problems Encountered
  573. The remainder of this section will be organized in a problem-solution format.  I will first 
  574. present a problem or consideration, and then discuss how it was solved or overlooked.
  575.  
  576. P -  As mentioned in section II.B.1, commands entered at the command line aren't given a 
  577. chance to expand the marked text symbol into text that is actually marked on the WStdio 
  578. window.  I had originally intended to provide this feature, but the implementation in the WStdio 
  579. window prevented it.  The problem lies in the fact that a command entered by the user is 
  580. executed only after the user finishes typing the command and presses the ENTER key, which 
  581. immediately generates a newline character.  When a newline is output to the WStdio window, 
  582. the marked text is automatically un-marked.  Thus by the time the command is being interpreted, 
  583. any marked text is no longer valid.  The Action Bar does not have this problem because it does 
  584. not generate a newline character when pressed.
  585. S - The solution for this problem was to disallow marked text expansion on the command line.  
  586. While this is more or less avoiding the problem, I have not found an acceptable way to rectify it.
  587.  
  588. P - The Microsoft C functions for manipulating ENVIRONMENT variables do not work in a 
  589. Windows program.  I don't know why.
  590. S - The solution was to get a pointer to the environment area, and do all ENVIRONMENT 
  591. manipulation functions by hand.  This was a messy job, but the only way I found to get it to 
  592. work.
  593.  
  594. P - In Windows, the ENVIRONMENT of the parent is not inherited by the programs is spawns.  
  595. This has made the ENVIRONMENT editor almost useless.
  596. S - There is nothing that can be done for this, except pray for future releases of Windows to 
  597. change this.
  598.  
  599. P - The speed of I/O to the display was too slow, running on a 16MHz VGA system.  For 
  600. example, as lines were output from an LS command, the rate at which lines were displayed was 
  601. annoyingly slow.  This is of course do to the graphical nature of displaying and scrolling text.
  602. S - The solution was to buffer the lines as they came in, and to output them in bursts.  This made 
  603. the code considerably more difficult to understand, but the end result was highly desirable.  I 
  604. found that by just buffering every other line, screen I/O was greatly increased.
  605.  
  606. P - Use of "strtok" C library function is dangerous.  Because I am lexically analyzing a 
  607. command line to determine what to execute, what flags to set, etc, I intended to use strtok.  
  608. However, the C library version is not compatible with Windows, because it allocates memory in 
  609. an incompatible way.  So I decided I would write my own. However, I realized that the way strtok 
  610. is used will not work in a multi tasking environment.  Strtok "remembers" the last parameter you 
  611. gave it, which allows you to call it successively to get the next tokens.  But with different 
  612. programs using strtok simultaneously, it will get garbled.  
  613. S - I considered two possible solutions for this problem.  The first was to redesign my strtok 
  614. function to always require the string to be parsed as a parameter.  The calling routine would 
  615. have to supply two buffers, one holding the source string, the second holding a buffer in which 
  616. strtok could do it's work.  This however, would be quite cumbersome for the client using strtok.  
  617. The second solution is to carefully organize the use of strtok so that it is used by all modules in 
  618. a linear fashion, instead of allowing any one module use it any time it likes.  That means the 
  619. module must know exactly what state the function is in before using strtok.  While the former is 
  620. a more foolproof solution I chose to go with the latter, since it seems to work and required the 
  621. least amount of change.
  622.  
  623.  
  624.  
  625. APPENDIX A. THE EXTERNAL COMMANDS DLL INTERFACE
  626.  
  627.  
  628. OVERVIEW
  629.     Each "External Command" is a DLL that contains the code for that command.  The 
  630. 'GENERIC.C' and associated files in the GENERIC directory provide a template for creating a 
  631. new External Command.  
  632.     An important consideration in implementing a new command is where to store your data.  
  633. In order for your DLL to be completely reentrant, no variable can be stored in the data segment.  
  634. That is, one must not declare variables outside of a function, or declare variables inside a 
  635. function.  The reason for this is that each invocation of a DLL function uses that same data 
  636. segment.  If a DLL function was called reentrantly,  static variables would be overwritten.  If a 
  637. DLL requires more data than will fit on the stack, on must use dynamic allocation.  Future 
  638. version may get rid of this requirement.
  639.  
  640. REQUIREMENTS
  641.  
  642. Each DLL must provide the following 3 functions for use by Windows Shell:
  643.  
  644. int FAR PASCAL ModuleProc (HWND hwndDisplay, int argc, LPSTR argv[]);
  645. @ ORDINAL 3
  646. hwndDisplay    - Window handle of STDIO Display to use for I/O.
  647. argc    - Number of command line arguments.
  648. argv    - Array of pointers to command line arguments.  The first pointer always points to 
  649. the name of the DLL.
  650. This function is called to let the DLL do the function which it is providing.  For example, if this 
  651. were a DLL providing a file deletion function, the DLL would perform the deletion at this time.
  652.  
  653. int FAR PASCAL ShowOptions (HWND hwndParent);
  654. @ ORDINAL 4
  655. hwndDisplay    - Window handle of STDIO Display to use for I/O or as parent.
  656. This function is called to tell the DLL to show it's options box.  The DLL should display a window 
  657. which allows the user to set options in the DLL.
  658.  
  659. int FAR PASCAL ShowAbout (HWND hwndParent);
  660. @ ORDINAL 5
  661. hwndDisplay    - Window handle of STDIO Display to use for I/O or as parent.
  662. This function is called to tell the DLL to show it's about box.  The DLL should display an about 
  663. window at this time.
  664.  
  665. NOTE - It is essential that the DLL export these functions at the specified ordinal value in it's 
  666. .DEF file.  Otherwise, The Windows Shell will not properly access the DLL.
  667.  
  668.  
  669. USING THE DISPLAY
  670. The header file 'wstdio.h' has been provided for outputing lines and other function to the 
  671. display.  The most common of these is dputs(), which you can use to output a line to the display.  
  672. See the header file for the description of the rest of the functions.
  673.  
  674. UTILITY FUNCTIONS
  675. The 'wslib.dll' provides several useful functions for parsing command lines, and yielding to 
  676. other applications.  It is extremely important that you use the YieldToOthers() function your code 
  677. sits in a tight loop for an extended length of time.  See the header file 'wslib.h' for a list of useful 
  678. functions.
  679.  
  680.  
  681. APPENDIX B. FUTURE ENHANCEMENTS
  682.  
  683. Possible Future Enhancements
  684. - User loadable alias files, supporting multiple loads of different files
  685. - Allow more variables to be user configurable, such as the scrollback buffer size.
  686. - Add a timer to the WStdio module to periodically purge all unpurged lines.
  687. - Allow marked text to remain marked as the WStdio window is scrolled.  This would allow the 
  688. marked text symbol to be expanded into the marked text for commands issued from the 
  689. command line.
  690. - Spawn a separate task for each Internal Command as it is run.  This would allow the DLL's to 
  691. use static data, because each time a DLL is called, it would be from a different task.
  692. Page: 2
  693.  
  694. TABLE OF CONTENTS
  695.  
  696.  
  697. ABSTRACT . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .    iii
  698.  
  699. HOW THIS DOCUMENT IS ORGANIZED . . . . . . . . . .. . . . .    iv
  700.  
  701. LIST OF FIGURES . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .     v
  702.  
  703. Section
  704.  
  705.     I.    Features of Windows Shell . . . . . . . . . . . . . . . . . . . . . . . .     1
  706.  
  707.         Command Line . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .    2
  708.  
  709.         Aliasing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .    3
  710.  
  711.         Environment Variables . . . . . . . . . . . . . . . . . . . . . . . . . . .     3
  712.  
  713.         Action Bar . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .    3
  714.  
  715.         External Commands . . . . . . . . . . . . . . . . . . . . . . . . . . . . .     4
  716.  
  717.         Shell Commands . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .     5
  718.  
  719.     II.    Design of Windows Shell . . . . . . . . . . . . . . . . . . . . . . . . .     7
  720.  
  721.         Graphical Objects in the Windows Shell . . . . . . . . . . . . .     7
  722.  
  723.              The WSTDIO Window . . . . .  . . . . . . . . . . . . . . .     8
  724.                 
  725.                 Pertinent Data . . . . . . . . . . . . . . . . . . . . . .    8
  726.  
  727.                 Painting The Display . . . . . . . . . . . . . . . . .    10
  728.  
  729.                 Obtaining Standard I/O input . . . . . . . . . .    11
  730.  
  731.                 Marking Text  . . . . . . . . . . . . . . . . . . . .    12
  732.  
  733.                 Command History . . . . . . . . . . . . . . . . . .    13
  734.  
  735.                 Message Directory . . . . . . . . . . . . . . . . .    13
  736.  
  737.             The ACTNBAR Window . . . . . . . . . . . . . . . . . .    14
  738.  
  739.             The WINSHELL Window . . . . . . . . . . . . . . . . .    15
  740.  
  741.         Logical Modules For Command Interpretation. . . . . . . .    16
  742.  
  743.             The Command Prompt Path . . . . . . . . . . . . . . . .    17
  744.  
  745.             The Action Button Path . . . . . . . . . . . . . . . . . . .     18
  746.  
  747.             The ALIAS.C Module . . . . . . . . . . . . . . . . . . . .    19
  748.  
  749.             The INTERP.C Module . . . . . . . . . . . . . . . . . . .    19
  750.  
  751.     III.     The Development Process . . .  . . . . . . . . . . . . . . . . . .    21
  752.  
  753. Appendix
  754.  
  755.     A. Appendix A: External Commands DLL Interface . . . . . . . . .     23
  756.  
  757.  
  758. iv
  759.  
  760. 20
  761.  
  762.  
  763.