ReactOS 0.4.15-dev-7788-g1ad9096
history.c
Go to the documentation of this file.
1/*
2 * HISTORY.C - command line history.
3 *
4 *
5 * History:
6 *
7 * 14/01/95 (Tim Norman)
8 * started.
9 *
10 * 08/08/95 (Matt Rains)
11 * i have cleaned up the source code. changes now bring this source
12 * into guidelines for recommended programming practice.
13 *
14 * 27-Jul-1998 (John P Price <linux-guru@gcfl.net>)
15 * added config.h include
16 *
17 * 25-Jan-1999 (Eric Kohl)
18 * Cleanup!
19 * Unicode and redirection safe!
20 *
21 * 25-Jan-1999 (Paolo Pantaleo <paolopan@freemail.it>)
22 * Added lots of comments (beginning studying the source)
23 * Added command.com's F3 support (see cmdinput.c)
24 *
25 */
26
27
28
29/*
30 * HISTORY.C - command line history. Second version
31 *
32 *
33 * History:
34 *
35 * 06/12/99 (Paolo Pantaleo <paolopan@freemail.it>)
36 * started.
37 *
38 */
39
40#include "precomp.h"
41
42#ifdef FEATURE_HISTORY
43
44typedef struct tagHISTORY
45{
50
51static INT size, max_size = 100;
52
55
57
60VOID History(INT dir, LPTSTR commandline);
63
64/*service functions*/
66static VOID add_at_bottom(LPTSTR string);
67/*VOID add_before_last(LPTSTR string);*/
68VOID set_size(INT new_size);
69
70
72{
73 LPTSTR tmp;
74 INT tmp_int;
75 LPHIST_ENTRY h_tmp;
76 TCHAR szBuffer[2048];
77
78 tmp=_tcschr(param,_T('/'));
79
80 if (tmp)
81 {
82 param=tmp;
83 switch (_totupper(param[1]))
84 {
85 case _T('F'):/*delete history*/
87 break;
88
89 case _T('R'):/*read history from standard in*/
90 for(;;)
91 {
92 ConInString(szBuffer,sizeof(szBuffer)/sizeof(TCHAR));
93 if (*szBuffer!=_T('\0'))
94 History(0,szBuffer);
95 else
96 break;
97 }
98 break;
99
100 case _T('A'):/*add an antry*/
101 History(0,param+2);
102 break;
103
104 case _T('S'):/*set history size*/
105 if ((tmp_int=_ttoi(param+2)))
106 set_size(tmp_int);
107 break;
108
109 default:
110 return 1;
111 }
112 }
113 else
114 {
115 for (h_tmp = Top->prev; h_tmp != Bottom; h_tmp = h_tmp->prev)
116 ConOutPrintf(_T("%s\n"), h_tmp->string);
117 }
118 return 0;
119}
120
122{
123 ASSERT(Top && Bottom);
124
125 while (new_size<size)
126 del(Top->prev);
127
128 max_size=new_size;
129}
130
131
133{
134 size = 0;
135
136 Top = cmd_alloc(sizeof(HIST_ENTRY));
137 if (!Top)
138 {
139 WARN("Cannot allocate memory for Top!\n");
140 return;
141 }
142 Bottom = cmd_alloc(sizeof(HIST_ENTRY));
143 if (!Bottom)
144 {
145 WARN("Cannot allocate memory for Bottom!\n");
146 cmd_free(Top);
147 Top = NULL;
148 return;
149 }
150
151 Top->prev = Bottom;
152 Top->next = NULL;
153 Top->string = NULL;
154
155 Bottom->prev = NULL;
156 Bottom->next = Top;
157 Bottom->string = NULL;
158
160}
161
162
164{
165 ASSERT(Top && Bottom);
166
167 while (Bottom->next != Top)
168 del(Bottom->next);
169
170 cmd_free(Top);
172}
173
174
176{
177 LPHIST_ENTRY tmp;
178
179 ASSERT(Top && Bottom);
180
181 if (size == 0)
182 return;
183
184 if (curr_ptr == Bottom)
186
187 if (curr_ptr == Top)
188 curr_ptr = Top->prev;
189
190
191 tmp = curr_ptr;
193 del(tmp);
194 History(-1, str);
195}
196
197
198static
200{
201 ASSERT(Top && Bottom);
202
203 if (item==NULL || item==Top || item==Bottom)
204 {
205 TRACE ("del in " __FILE__ ": returning\n"
206 "item is 0x%08x (Bottom is0x%08x)\n",
207 item, Bottom);
208 return;
209 }
210
211 /*free string's mem*/
212 if (item->string)
213 cmd_free(item->string);
214
215 /*set links in prev and next item*/
216 item->next->prev=item->prev;
217 item->prev->next=item->next;
218
219 cmd_free(item);
220
221 size--;
222}
223
224static
226{
227 LPHIST_ENTRY tmp;
228
229 ASSERT(Top && Bottom);
230
231 /*delete first entry if maximum number of entries is reached*/
232 while (size>=max_size)
233 del(Top->prev);
234
235 while (_istspace(*string))
236 string++;
237
238 if (*string==_T('\0'))
239 return;
240
241 /*if new entry is the same than the last do not add it*/
242 if (size)
243 {
244 if (_tcscmp(string,Bottom->next->string)==0)
245 return;
246 }
247
248 /*create new empty Bottom*/
249 tmp = cmd_alloc(sizeof(HIST_ENTRY));
250 if (!tmp)
251 {
252 WARN("Cannot allocate memory for new Bottom!\n");
253 return;
254 }
255
256 /*fill old bottom with string, it will become new Bottom->next*/
257 Bottom->string = cmd_alloc((_tcslen(string)+1)*sizeof(TCHAR));
258 if (!Bottom->string)
259 {
260 WARN("Cannot allocate memory for Bottom->string!\n");
261 cmd_free(tmp);
262 return;
263 }
264 _tcscpy(Bottom->string,string);
265
266 tmp->next = Bottom;
267 tmp->prev = NULL;
268 tmp->string = NULL;
269
270 Bottom->prev = tmp;
271
272 /*save the new Bottom value*/
273 Bottom = tmp;
274
275 /*set new size*/
276 size++;
277}
278
279
281{
282 ASSERT(Top && Bottom);
283
285}
286
288{
290
291 ASSERT(Top && Bottom);
292
293 if (dir == 0)
294 return NULL;
295
296 if (dir < 0)
297 {
298 /* key up */
299 if (entry->next == Top || entry == Top)
300 {
301#ifdef WRAP_HISTORY
302 entry = Bottom;
303#else
304 return NULL;
305#endif
306 }
307 entry = entry->next;
308 }
309 else
310 {
311 /* key down */
312 if (entry->prev == Bottom || entry == Bottom)
313 {
314#ifdef WRAP_HISTORY
315 entry = Top;
316#else
317 return NULL;
318#endif
319 }
320 entry = entry->prev;
321 }
322
323 return entry->string;
324}
325
326VOID History(INT dir, LPTSTR commandline)
327{
328 ASSERT(Top && Bottom);
329
330 if (dir==0)
331 {
332 add_at_bottom(commandline);
334 return;
335 }
336
337 if (size==0)
338 {
339 commandline[0]=_T('\0');
340 return;
341 }
342
343 if (dir<0)/*key up*/
344 {
345 if (curr_ptr->next==Top || curr_ptr==Top)
346 {
347#ifdef WRAP_HISTORY
349#else
350 curr_ptr = Top;
351 commandline[0]=_T('\0');
352 return;
353#endif
354 }
355
357 if (curr_ptr->string)
358 _tcscpy(commandline,curr_ptr->string);
359 }
360
361 if (dir>0)
362 {
364 {
365#ifdef WRAP_HISTORY
366 curr_ptr = Top;
367#else
369 commandline[0]=_T('\0');
370 return;
371#endif
372 }
373
375 if (curr_ptr->string)
376 _tcscpy(commandline,curr_ptr->string);
377 }
378}
379
380#endif //#if FEATURE_HISTORY
unsigned int dir
Definition: maze.c:112
#define ConOutPrintf(szStr,...)
Definition: console.h:41
VOID History_move_to_bottom(VOID)
Definition: history.c:280
static LPHIST_ENTRY Bottom
Definition: history.c:54
VOID InitHistory(VOID)
Definition: history.c:132
static INT max_size
Definition: history.c:51
VOID History(INT dir, LPTSTR commandline)
Definition: history.c:326
VOID set_size(INT new_size)
Definition: history.c:121
static INT size
Definition: history.c:51
static LPHIST_ENTRY Top
Definition: history.c:53
static VOID add_at_bottom(LPTSTR string)
Definition: history.c:225
struct tagHISTORY HIST_ENTRY
VOID History_del_current_entry(LPTSTR str)
Definition: history.c:175
struct tagHISTORY * LPHIST_ENTRY
LPCTSTR PeekHistory(INT dir)
Definition: history.c:287
static LPHIST_ENTRY curr_ptr
Definition: history.c:56
static VOID del(LPHIST_ENTRY item)
Definition: history.c:199
VOID CleanHistory(VOID)
Definition: history.c:163
INT CommandHistory(LPTSTR param)
Definition: history.c:71
#define WARN(fmt,...)
Definition: debug.h:112
#define cmd_free(ptr)
Definition: cmddbg.h:31
#define cmd_alloc(size)
Definition: cmddbg.h:29
#define NULL
Definition: types.h:112
GLsizeiptr size
Definition: glext.h:5919
GLfloat param
Definition: glext.h:5796
#define _istspace
Definition: tchar.h:1504
#define _tcscmp
Definition: tchar.h:1424
#define _tcscpy
Definition: tchar.h:623
#define _totupper
Definition: tchar.h:1509
#define _tcschr
Definition: tchar.h:1406
uint32_t entry
Definition: isohybrid.c:63
static VOID ConInString(LPWSTR lpInput, DWORD dwLength)
Definition: label.c:56
#define ASSERT(a)
Definition: mode.c:44
static ATOM item
Definition: dde.c:856
const WCHAR * str
#define TRACE(s)
Definition: solgame.cpp:4
struct tagHISTORY * prev
Definition: history.c:46
struct tagHISTORY * next
Definition: history.c:47
LPTSTR string
Definition: history.c:48
int32_t INT
Definition: typedefs.h:58
#define _T(x)
Definition: vfdio.h:22
char TCHAR
Definition: xmlstorage.h:189
#define _ttoi
Definition: xmlstorage.h:195
const CHAR * LPCTSTR
Definition: xmlstorage.h:193
CHAR * LPTSTR
Definition: xmlstorage.h:192
#define _tcslen
Definition: xmlstorage.h:198