]> git.saurik.com Git - wxWidgets.git/blame - src/common/hash.cpp
Reverted wxChoice size fix due to problems in W2K and below
[wxWidgets.git] / src / common / hash.cpp
CommitLineData
c801d85f 1/////////////////////////////////////////////////////////////////////////////
7cb32b4b 2// Name: src/common/hash.cpp
c801d85f
KB
3// Purpose: wxHashTable implementation
4// Author: Julian Smart
bcaa23de 5// Modified by: VZ at 25.02.00: type safe hashes with WX_DECLARE_HASH()
c801d85f
KB
6// Created: 01/02/97
7// RCS-ID: $Id$
55d99c7a 8// Copyright: (c) Julian Smart
65571936 9// Licence: wxWindows licence
c801d85f
KB
10/////////////////////////////////////////////////////////////////////////////
11
bcaa23de
VZ
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
c801d85f
KB
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24#pragma hdrstop
25#endif
26
27#ifndef WX_PRECOMP
28#include "wx/list.h"
29#endif
30
31#include "wx/hash.h"
32
6e86701b 33#if wxUSE_OLD_HASH_TABLE
df5168c4 34
c801d85f
KB
35#include <string.h>
36#include <stdarg.h>
37
bcaa23de
VZ
38// ----------------------------------------------------------------------------
39// wxWin macros
40// ----------------------------------------------------------------------------
41
c801d85f 42IMPLEMENT_DYNAMIC_CLASS(wxHashTable, wxObject)
c801d85f 43
bcaa23de
VZ
44// ============================================================================
45// implementation
46// ============================================================================
47
48// ----------------------------------------------------------------------------
49// wxHashTablleBase for working with "void *" data
50// ----------------------------------------------------------------------------
51
52wxHashTableBase::wxHashTableBase()
53{
f803b25b 54 m_deleteContents = false;
bcaa23de
VZ
55 m_hashTable = (wxListBase **)NULL;
56 m_hashSize = 0;
57 m_count = 0;
58 m_keyType = wxKEY_NONE;
59}
60
61void wxHashTableBase::Create(wxKeyType keyType, size_t size)
62{
63 Destroy();
64
65 m_hashSize = size;
66 m_keyType = keyType;
67 m_hashTable = new wxListBase *[size];
68 for ( size_t n = 0; n < m_hashSize; n++ )
69 {
70 m_hashTable[n] = (wxListBase *) NULL;
71 }
72}
73
74void wxHashTableBase::Destroy()
75{
76 if ( m_hashTable )
77 {
78 for ( size_t n = 0; n < m_hashSize; n++ )
79 {
80 delete m_hashTable[n];
81 }
82
83 delete [] m_hashTable;
84
85 m_hashTable = (wxListBase **)NULL;
86
87 m_count = 0;
88 }
89}
90
91void wxHashTableBase::DeleteContents(bool flag)
92{
93 m_deleteContents = flag;
94 for ( size_t n = 0; n < m_hashSize; n++ )
95 {
96 if ( m_hashTable[n] )
97 {
98 m_hashTable[n]->DeleteContents(flag);
99 }
100 }
101}
102
103wxNodeBase *wxHashTableBase::GetNode(long key, long value) const
104{
3ca6a5f0 105 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
bcaa23de
VZ
106
107 wxNodeBase *node;
108 if ( m_hashTable[slot] )
109 {
110 node = m_hashTable[slot]->Find(wxListKey(value));
111 }
112 else
113 {
114 node = (wxNodeBase *)NULL;
115 }
116
117 return node;
118}
119
ba8c1601
MB
120#if WXWIN_COMPATIBILITY_2_4
121
c2bb85e9
VZ
122// ----------------------------------------------------------------------------
123// wxHashTableLong
124// ----------------------------------------------------------------------------
125
a95e38c0
VZ
126wxHashTableLong::~wxHashTableLong()
127{
128 Destroy();
129}
130
c2bb85e9
VZ
131void wxHashTableLong::Init(size_t size)
132{
133 m_hashSize = size;
134 m_values = new wxArrayLong *[size];
135 m_keys = new wxArrayLong *[size];
136
137 for ( size_t n = 0; n < m_hashSize; n++ )
138 {
139 m_values[n] =
140 m_keys[n] = (wxArrayLong *)NULL;
141 }
142
143 m_count = 0;
144}
145
b7aaabf8
VS
146void wxHashTableLong::Create(size_t size)
147{
148 Init(size);
149}
150
c2bb85e9
VZ
151void wxHashTableLong::Destroy()
152{
153 for ( size_t n = 0; n < m_hashSize; n++ )
154 {
155 delete m_values[n];
156 delete m_keys[n];
157 }
158
159 delete [] m_values;
160 delete [] m_keys;
161 m_hashSize = 0;
162 m_count = 0;
163}
164
165void wxHashTableLong::Put(long key, long value)
166{
167 wxCHECK_RET( m_hashSize, _T("must call Create() first") );
168
3ca6a5f0 169 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
c2bb85e9
VZ
170
171 if ( !m_keys[slot] )
172 {
173 m_keys[slot] = new wxArrayLong;
174 m_values[slot] = new wxArrayLong;
175 }
176
177 m_keys[slot]->Add(key);
178 m_values[slot]->Add(value);
179
180 m_count++;
181}
182
183long wxHashTableLong::Get(long key) const
184{
185 wxCHECK_MSG( m_hashSize, wxNOT_FOUND, _T("must call Create() first") );
186
3ca6a5f0 187 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
c2bb85e9
VZ
188
189 wxArrayLong *keys = m_keys[slot];
190 if ( keys )
191 {
192 size_t count = keys->GetCount();
193 for ( size_t n = 0; n < count; n++ )
194 {
195 if ( keys->Item(n) == key )
196 {
197 return m_values[slot]->Item(n);
198 }
199 }
200 }
201
202 return wxNOT_FOUND;
203}
204
205long wxHashTableLong::Delete(long key)
206{
207 wxCHECK_MSG( m_hashSize, wxNOT_FOUND, _T("must call Create() first") );
208
3ca6a5f0 209 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
c2bb85e9
VZ
210
211 wxArrayLong *keys = m_keys[slot];
212 if ( keys )
213 {
214 size_t count = keys->GetCount();
215 for ( size_t n = 0; n < count; n++ )
216 {
217 if ( keys->Item(n) == key )
218 {
219 long val = m_values[slot]->Item(n);
220
221 keys->RemoveAt(n);
222 m_values[slot]->RemoveAt(n);
223
224 m_count--;
225
226 return val;
227 }
228 }
229 }
230
231 return wxNOT_FOUND;
232}
233
bd83cb56
VZ
234// ----------------------------------------------------------------------------
235// wxStringHashTable: more efficient than storing strings in a list
236// ----------------------------------------------------------------------------
237
238wxStringHashTable::wxStringHashTable(size_t sizeTable)
239{
240 m_keys = new wxArrayLong *[sizeTable];
241 m_values = new wxArrayString *[sizeTable];
242
243 m_hashSize = sizeTable;
244 for ( size_t n = 0; n < m_hashSize; n++ )
245 {
246 m_values[n] = (wxArrayString *)NULL;
247 m_keys[n] = (wxArrayLong *)NULL;
248 }
249}
250
251wxStringHashTable::~wxStringHashTable()
252{
253 Destroy();
254}
255
256void wxStringHashTable::Destroy()
257{
258 for ( size_t n = 0; n < m_hashSize; n++ )
259 {
260 delete m_values[n];
261 delete m_keys[n];
262 }
263
264 delete [] m_values;
265 delete [] m_keys;
266 m_hashSize = 0;
267}
268
269void wxStringHashTable::Put(long key, const wxString& value)
270{
271 wxCHECK_RET( m_hashSize, _T("must call Create() first") );
272
273 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
274
275 if ( !m_keys[slot] )
276 {
277 m_keys[slot] = new wxArrayLong;
278 m_values[slot] = new wxArrayString;
279 }
280
281 m_keys[slot]->Add(key);
282 m_values[slot]->Add(value);
283}
284
285wxString wxStringHashTable::Get(long key, bool *wasFound) const
286{
525d8583 287 wxCHECK_MSG( m_hashSize, wxEmptyString, _T("must call Create() first") );
bd83cb56
VZ
288
289 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
290
291 wxArrayLong *keys = m_keys[slot];
292 if ( keys )
293 {
294 size_t count = keys->GetCount();
295 for ( size_t n = 0; n < count; n++ )
296 {
297 if ( keys->Item(n) == key )
298 {
299 if ( wasFound )
f803b25b 300 *wasFound = true;
bd83cb56
VZ
301
302 return m_values[slot]->Item(n);
303 }
304 }
305 }
306
307 if ( wasFound )
f803b25b 308 *wasFound = false;
bd83cb56 309
525d8583 310 return wxEmptyString;
bd83cb56
VZ
311}
312
53e112a0
JS
313bool wxStringHashTable::Delete(long key) const
314{
f803b25b 315 wxCHECK_MSG( m_hashSize, false, _T("must call Create() first") );
53e112a0
JS
316
317 size_t slot = (size_t)abs((int)(key % (long)m_hashSize));
318
319 wxArrayLong *keys = m_keys[slot];
320 if ( keys )
321 {
322 size_t count = keys->GetCount();
323 for ( size_t n = 0; n < count; n++ )
324 {
325 if ( keys->Item(n) == key )
326 {
f8305caf 327 keys->RemoveAt(n);
53e112a0 328 m_values[slot]->RemoveAt(n);
f803b25b 329 return true;
53e112a0
JS
330 }
331 }
332 }
333
f803b25b 334 return false;
53e112a0
JS
335}
336
ba8c1601
MB
337#endif // WXWIN_COMPATIBILITY_2_4
338
bcaa23de
VZ
339// ----------------------------------------------------------------------------
340// old not type safe wxHashTable
341// ----------------------------------------------------------------------------
342
debe6624 343wxHashTable::wxHashTable (int the_key_type, int size)
c801d85f 344{
17d8ee1c
JS
345 n = 0;
346 hash_table = (wxList**) NULL;
347 Create(the_key_type, size);
5692876f 348 m_count = 0;
f803b25b 349 m_deleteContents = false;
17d8ee1c 350/*
c801d85f
KB
351 n = size;
352 current_position = -1;
c67daf87 353 current_node = (wxNode *) NULL;
c801d85f
KB
354
355 key_type = the_key_type;
356 hash_table = new wxList *[size];
357 int i;
358 for (i = 0; i < size; i++)
c67daf87 359 hash_table[i] = (wxList *) NULL;
17d8ee1c 360*/
c801d85f
KB
361}
362
bcaa23de 363wxHashTable::~wxHashTable ()
c801d85f 364{
e55ad60e
RR
365 Destroy();
366}
367
bcaa23de 368void wxHashTable::Destroy()
e55ad60e
RR
369{
370 if (!hash_table) return;
c801d85f
KB
371 int i;
372 for (i = 0; i < n; i++)
373 if (hash_table[i])
374 delete hash_table[i];
375 delete[] hash_table;
e55ad60e 376 hash_table = NULL;
c801d85f
KB
377}
378
debe6624 379bool wxHashTable::Create(int the_key_type, int size)
c801d85f 380{
17d8ee1c
JS
381 Destroy();
382
c801d85f
KB
383 n = size;
384 current_position = -1;
c67daf87 385 current_node = (wxNode *) NULL;
c801d85f
KB
386
387 key_type = the_key_type;
c801d85f
KB
388 hash_table = new wxList *[size];
389 int i;
390 for (i = 0; i < size; i++)
c67daf87 391 hash_table[i] = (wxList *) NULL;
f803b25b 392 return true;
c801d85f
KB
393}
394
4e67bfc7
VS
395
396void wxHashTable::DoCopy(const wxHashTable& table)
397{
398 n = table.n;
ea450825 399 m_count = table.m_count;
4e67bfc7
VS
400 current_position = table.current_position;
401 current_node = NULL; // doesn't matter - Next() will reconstruct it
402 key_type = table.key_type;
403
404 hash_table = new wxList *[n];
405 for (int i = 0; i < n; i++) {
406 if (table.hash_table[i] == NULL)
407 hash_table[i] = NULL;
408 else {
409 hash_table[i] = new wxList(key_type);
410 *(hash_table[i]) = *(table.hash_table[i]);
411 }
412 }
413}
414
debe6624 415void wxHashTable::Put (long key, long value, wxObject * object)
c801d85f
KB
416{
417 // Should NEVER be
418 long k = (long) key;
c801d85f
KB
419
420 int position = (int) (k % n);
ffae916f
BJ
421 if (position < 0) position = -position;
422
c801d85f 423 if (!hash_table[position])
c7fb814a 424 {
cbb26e3f 425 hash_table[position] = new wxList (wxKEY_INTEGER);
f803b25b 426 if (m_deleteContents) hash_table[position]->DeleteContents(true);
c7fb814a 427 }
c801d85f
KB
428
429 hash_table[position]->Append (value, object);
5692876f 430 m_count++;
c801d85f
KB
431}
432
50920146 433void wxHashTable::Put (long key, const wxChar *value, wxObject * object)
c801d85f
KB
434{
435 // Should NEVER be
bcaa23de 436 long k = (long) key;
c801d85f
KB
437
438 int position = (int) (k % n);
ffae916f 439 if (position < 0) position = -position;
bcaa23de 440
c801d85f 441 if (!hash_table[position])
c7fb814a 442 {
cbb26e3f 443 hash_table[position] = new wxList (wxKEY_STRING);
f803b25b 444 if (m_deleteContents) hash_table[position]->DeleteContents(true);
c7fb814a 445 }
c801d85f
KB
446
447 hash_table[position]->Append (value, object);
5692876f 448 m_count++;
c801d85f
KB
449}
450
debe6624 451void wxHashTable::Put (long key, wxObject * object)
c801d85f
KB
452{
453 // Should NEVER be
454 long k = (long) key;
bcaa23de 455
c801d85f 456 int position = (int) (k % n);
ffae916f
BJ
457 if (position < 0) position = -position;
458
c801d85f 459 if (!hash_table[position])
c7fb814a 460 {
c801d85f 461 hash_table[position] = new wxList (wxKEY_INTEGER);
f803b25b 462 if (m_deleteContents) hash_table[position]->DeleteContents(true);
c7fb814a 463 }
bcaa23de 464
c801d85f 465 hash_table[position]->Append (k, object);
5692876f 466 m_count++;
c801d85f
KB
467}
468
50920146 469void wxHashTable::Put (const wxChar *key, wxObject * object)
c801d85f
KB
470{
471 int position = (int) (MakeKey (key) % n);
ffae916f 472 if (position < 0) position = -position;
c801d85f
KB
473
474 if (!hash_table[position])
c7fb814a 475 {
c801d85f 476 hash_table[position] = new wxList (wxKEY_STRING);
f803b25b 477 if (m_deleteContents) hash_table[position]->DeleteContents(true);
c7fb814a 478 }
c801d85f
KB
479
480 hash_table[position]->Append (key, object);
5692876f 481 m_count++;
c801d85f
KB
482}
483
debe6624 484wxObject *wxHashTable::Get (long key, long value) const
c801d85f
KB
485{
486 // Should NEVER be
487 long k = (long) key;
c801d85f
KB
488
489 int position = (int) (k % n);
ffae916f
BJ
490 if (position < 0) position = -position;
491
c801d85f 492 if (!hash_table[position])
c67daf87 493 return (wxObject *) NULL;
c801d85f
KB
494 else
495 {
496 wxNode *node = hash_table[position]->Find (value);
497 if (node)
b1d4dd7a 498 return node->GetData ();
c801d85f 499 else
bcaa23de 500 return (wxObject *) NULL;
c801d85f
KB
501 }
502}
503
50920146 504wxObject *wxHashTable::Get (long key, const wxChar *value) const
c801d85f
KB
505{
506 // Should NEVER be
507 long k = (long) key;
bcaa23de 508
c801d85f 509 int position = (int) (k % n);
ffae916f
BJ
510 if (position < 0) position = -position;
511
c801d85f 512 if (!hash_table[position])
c67daf87 513 return (wxObject *) NULL;
c801d85f
KB
514 else
515 {
516 wxNode *node = hash_table[position]->Find (value);
517 if (node)
b1d4dd7a 518 return node->GetData ();
c801d85f 519 else
bcaa23de 520 return (wxObject *) NULL;
c801d85f
KB
521 }
522}
523
debe6624 524wxObject *wxHashTable::Get (long key) const
c801d85f
KB
525{
526 // Should NEVER be
527 long k = (long) key;
c801d85f
KB
528
529 int position = (int) (k % n);
ffae916f
BJ
530 if (position < 0) position = -position;
531
c801d85f 532 if (!hash_table[position])
c67daf87 533 return (wxObject *) NULL;
c801d85f
KB
534 else
535 {
536 wxNode *node = hash_table[position]->Find (k);
b1d4dd7a 537 return node ? node->GetData () : (wxObject*)NULL;
c801d85f
KB
538 }
539}
540
50920146 541wxObject *wxHashTable::Get (const wxChar *key) const
c801d85f
KB
542{
543 int position = (int) (MakeKey (key) % n);
ffae916f 544 if (position < 0) position = -position;
c801d85f
KB
545
546 if (!hash_table[position])
c67daf87 547 return (wxObject *) NULL;
c801d85f
KB
548 else
549 {
550 wxNode *node = hash_table[position]->Find (key);
b1d4dd7a 551 return node ? node->GetData () : (wxObject*)NULL;
c801d85f
KB
552 }
553}
554
debe6624 555wxObject *wxHashTable::Delete (long key)
c801d85f
KB
556{
557 // Should NEVER be
558 long k = (long) key;
c801d85f
KB
559
560 int position = (int) (k % n);
ffae916f
BJ
561 if (position < 0) position = -position;
562
c801d85f 563 if (!hash_table[position])
c67daf87 564 return (wxObject *) NULL;
c801d85f
KB
565 else
566 {
567 wxNode *node = hash_table[position]->Find (k);
568 if (node)
bcaa23de 569 {
b1d4dd7a 570 wxObject *data = node->GetData ();
bcaa23de
VZ
571 delete node;
572 m_count--;
573 return data;
574 }
c801d85f 575 else
bcaa23de 576 return (wxObject *) NULL;
c801d85f
KB
577 }
578}
579
50920146 580wxObject *wxHashTable::Delete (const wxChar *key)
c801d85f
KB
581{
582 int position = (int) (MakeKey (key) % n);
ffae916f
BJ
583 if (position < 0) position = -position;
584
c801d85f 585 if (!hash_table[position])
c67daf87 586 return (wxObject *) NULL;
c801d85f
KB
587 else
588 {
589 wxNode *node = hash_table[position]->Find (key);
590 if (node)
bcaa23de 591 {
b1d4dd7a 592 wxObject *data = node->GetData ();
bcaa23de
VZ
593 delete node;
594 m_count--;
595 return data;
596 }
c801d85f 597 else
bcaa23de 598 return (wxObject *) NULL;
c801d85f
KB
599 }
600}
601
debe6624 602wxObject *wxHashTable::Delete (long key, int value)
c801d85f
KB
603{
604 // Should NEVER be
bcaa23de 605 long k = (long) key;
c801d85f
KB
606
607 int position = (int) (k % n);
ffae916f
BJ
608 if (position < 0) position = -position;
609
c801d85f 610 if (!hash_table[position])
c67daf87 611 return (wxObject *) NULL;
c801d85f
KB
612 else
613 {
614 wxNode *node = hash_table[position]->Find (value);
615 if (node)
bcaa23de 616 {
b1d4dd7a 617 wxObject *data = node->GetData ();
bcaa23de
VZ
618 delete node;
619 m_count--;
620 return data;
621 }
c801d85f 622 else
bcaa23de 623 return (wxObject *) NULL;
c801d85f
KB
624 }
625}
626
50920146 627wxObject *wxHashTable::Delete (long key, const wxChar *value)
c801d85f
KB
628{
629 int position = (int) (key % n);
ffae916f
BJ
630 if (position < 0) position = -position;
631
c801d85f 632 if (!hash_table[position])
c67daf87 633 return (wxObject *) NULL;
c801d85f
KB
634 else
635 {
636 wxNode *node = hash_table[position]->Find (value);
637 if (node)
bcaa23de 638 {
b1d4dd7a 639 wxObject *data = node->GetData ();
bcaa23de
VZ
640 delete node;
641 m_count--;
642 return data;
643 }
c801d85f 644 else
bcaa23de 645 return (wxObject *) NULL;
c801d85f
KB
646 }
647}
648
50920146 649long wxHashTable::MakeKey (const wxChar *string) const
c801d85f
KB
650{
651 long int_key = 0;
652
653 while (*string)
50920146 654 int_key += (wxUChar) *string++;
c801d85f
KB
655
656 return int_key;
657}
658
bcaa23de 659void wxHashTable::BeginFind ()
c801d85f
KB
660{
661 current_position = -1;
c67daf87 662 current_node = (wxNode *) NULL;
c801d85f
KB
663}
664
1a6d9c76 665wxHashTable::Node* wxHashTable::Next ()
c801d85f 666{
c67daf87 667 wxNode *found = (wxNode *) NULL;
f803b25b 668 bool end = false;
c801d85f
KB
669 while (!end && !found)
670 {
671 if (!current_node)
bcaa23de
VZ
672 {
673 current_position++;
674 if (current_position >= n)
675 {
676 current_position = -1;
677 current_node = (wxNode *) NULL;
f803b25b 678 end = true;
bcaa23de
VZ
679 }
680 else
681 {
682 if (hash_table[current_position])
683 {
b1d4dd7a 684 current_node = hash_table[current_position]->GetFirst ();
bcaa23de
VZ
685 found = current_node;
686 }
687 }
688 }
c801d85f 689 else
bcaa23de 690 {
b1d4dd7a 691 current_node = current_node->GetNext ();
bcaa23de
VZ
692 found = current_node;
693 }
c801d85f
KB
694 }
695 return found;
696}
697
debe6624 698void wxHashTable::DeleteContents (bool flag)
c801d85f
KB
699{
700 int i;
c7fb814a 701 m_deleteContents = flag;
c801d85f
KB
702 for (i = 0; i < n; i++)
703 {
704 if (hash_table[i])
bcaa23de 705 hash_table[i]->DeleteContents (flag);
c801d85f
KB
706 }
707}
708
bcaa23de 709void wxHashTable::Clear ()
c801d85f 710{
19230604
RD
711 int i;
712 if (hash_table)
c801d85f 713 {
19230604
RD
714 for (i = 0; i < n; i++)
715 {
716 if (hash_table[i])
717 hash_table[i]->Clear ();
718 }
c801d85f 719 }
5692876f 720 m_count = 0;
c801d85f
KB
721}
722
6e86701b 723#else // if !wxUSE_OLD_HASH_TABLE
1a6d9c76
MB
724
725wxHashTableBase_Node::wxHashTableBase_Node( long key, void* value,
726 wxHashTableBase* table )
727 : m_value( value ), m_hashPtr( table )
728{
729 m_key.integer = key;
730}
731
732wxHashTableBase_Node::wxHashTableBase_Node( const wxChar* key, void* value,
733 wxHashTableBase* table )
734 : m_value( value ), m_hashPtr( table )
735{
736 m_key.string = wxStrcpy( new wxChar[wxStrlen( key ) + 1], key );
737}
738
739wxHashTableBase_Node::~wxHashTableBase_Node()
740{
741 if( m_hashPtr ) m_hashPtr->DoRemoveNode( this );
742}
743
744//
745
746wxHashTableBase::wxHashTableBase()
747 : m_size( 0 ), m_count( 0 ), m_table( NULL ), m_keyType( wxKEY_NONE ),
748 m_deleteContents( false )
749{
750}
751
1a6d9c76
MB
752void wxHashTableBase::Create( wxKeyType keyType, size_t size )
753{
754 m_keyType = keyType;
755 m_size = size;
756 m_table = new wxHashTableBase_Node*[ m_size ];
757
758 for( size_t i = 0; i < m_size; ++i )
759 m_table[i] = NULL;
760}
761
762void wxHashTableBase::Clear()
763{
764 for( size_t i = 0; i < m_size; ++i )
765 {
766 Node* end = m_table[i];
767
768 if( end == NULL )
769 continue;
770
771 Node *curr, *next = end->GetNext();
772
773 do
774 {
775 curr = next;
776 next = curr->GetNext();
777
778 DoDestroyNode( curr );
779
780 delete curr;
781 }
782 while( curr != end );
783
784 m_table[i] = NULL;
785 }
786
787 m_count = 0;
788}
789
790void wxHashTableBase::DoRemoveNode( wxHashTableBase_Node* node )
791{
792 size_t bucket = ( m_keyType == wxKEY_INTEGER ?
793 node->m_key.integer :
794 MakeKey( node->m_key.string ) ) % m_size;
795
796 if( node->GetNext() == node )
797 {
798 // single-node chain (common case)
799 m_table[bucket] = NULL;
800 }
801 else
802 {
803 Node *start = m_table[bucket], *curr;
804 Node* prev = start;
805
806 for( curr = prev->GetNext(); curr != node;
f803b25b 807 prev = curr, curr = curr->GetNext() ) ;
1a6d9c76
MB
808
809 DoUnlinkNode( bucket, node, prev );
810 }
811
812 DoDestroyNode( node );
813}
814
815void wxHashTableBase::DoDestroyNode( wxHashTableBase_Node* node )
816{
817 // if it is called from DoRemoveNode, node has already been
818 // removed, from other places it does not matter
819 node->m_hashPtr = NULL;
820
821 if( m_keyType == wxKEY_STRING )
822 delete[] node->m_key.string;
823 if( m_deleteContents )
824 DoDeleteContents( node );
825}
826
827void wxHashTableBase::Destroy()
828{
829 Clear();
830
831 delete[] m_table;
832
833 m_table = NULL;
834 m_size = 0;
835}
836
837void wxHashTableBase::DoInsertNode( size_t bucket, wxHashTableBase_Node* node )
838{
839 if( m_table[bucket] == NULL )
840 {
841 m_table[bucket] = node->m_next = node;
842 }
843 else
844 {
845 Node *prev = m_table[bucket];
846 Node *next = prev->m_next;
847
848 prev->m_next = node;
849 node->m_next = next;
850 m_table[bucket] = node;
851 }
852
853 ++m_count;
854}
855
856void wxHashTableBase::DoPut( long key, long hash, void* data )
857{
858 wxASSERT( m_keyType == wxKEY_INTEGER );
859
860 size_t bucket = size_t(hash) % m_size;
861 Node* node = new wxHashTableBase_Node( key, data, this );
862
863 DoInsertNode( bucket, node );
864}
865
866void wxHashTableBase::DoPut( const wxChar* key, long hash, void* data )
867{
868 wxASSERT( m_keyType == wxKEY_STRING );
869
870 size_t bucket = size_t(hash) % m_size;
871 Node* node = new wxHashTableBase_Node( key, data, this );
872
873 DoInsertNode( bucket, node );
874}
875
876void* wxHashTableBase::DoGet( long key, long hash ) const
877{
878 wxASSERT( m_keyType == wxKEY_INTEGER );
879
880 size_t bucket = size_t(hash) % m_size;
881
882 if( m_table[bucket] == NULL )
883 return NULL;
884
885 Node *first = m_table[bucket]->GetNext(),
886 *curr = first;
887
888 do
889 {
890 if( curr->m_key.integer == key )
891 return curr->m_value;
892
893 curr = curr->GetNext();
894 }
895 while( curr != first );
896
897 return NULL;
898}
899
900void* wxHashTableBase::DoGet( const wxChar* key, long hash ) const
901{
902 wxASSERT( m_keyType == wxKEY_STRING );
903
904 size_t bucket = size_t(hash) % m_size;
905
906 if( m_table[bucket] == NULL )
907 return NULL;
908
909 Node *first = m_table[bucket]->GetNext(),
910 *curr = first;
911
912 do
913 {
914 if( wxStrcmp( curr->m_key.string, key ) == 0 )
915 return curr->m_value;
916
917 curr = curr->GetNext();
918 }
919 while( curr != first );
920
921 return NULL;
922}
923
924void wxHashTableBase::DoUnlinkNode( size_t bucket, wxHashTableBase_Node* node,
925 wxHashTableBase_Node* prev )
926{
927 if( node == m_table[bucket] )
928 m_table[bucket] = prev;
929
930 if( prev == node && prev == node->GetNext() )
931 m_table[bucket] = NULL;
932 else
933 prev->m_next = node->m_next;
934
935 DoDestroyNode( node );
936 --m_count;
937}
938
939void* wxHashTableBase::DoDelete( long key, long hash )
940{
941 wxASSERT( m_keyType == wxKEY_INTEGER );
942
943 size_t bucket = size_t(hash) % m_size;
944
945 if( m_table[bucket] == NULL )
946 return NULL;
947
948 Node *first = m_table[bucket]->GetNext(),
949 *curr = first,
950 *prev = m_table[bucket];
951
952 do
953 {
954 if( curr->m_key.integer == key )
955 {
956 void* retval = curr->m_value;
957 curr->m_value = NULL;
958
959 DoUnlinkNode( bucket, curr, prev );
960 delete curr;
961
962 return retval;
963 }
964
965 prev = curr;
966 curr = curr->GetNext();
967 }
968 while( curr != first );
969
970 return NULL;
971}
972
973void* wxHashTableBase::DoDelete( const wxChar* key, long hash )
974{
975 wxASSERT( m_keyType == wxKEY_STRING );
976
977 size_t bucket = size_t(hash) % m_size;
978
979 if( m_table[bucket] == NULL )
980 return NULL;
981
982 Node *first = m_table[bucket]->GetNext(),
983 *curr = first,
984 *prev = m_table[bucket];
985
986 do
987 {
988 if( wxStrcmp( curr->m_key.string, key ) == 0 )
989 {
990 void* retval = curr->m_value;
991 curr->m_value = NULL;
992
993 DoUnlinkNode( bucket, curr, prev );
994 delete curr;
995
996 return retval;
997 }
998
999 prev = curr;
1000 curr = curr->GetNext();
1001 }
1002 while( curr != first );
1003
1004 return NULL;
1005}
1006
1007long wxHashTableBase::MakeKey( const wxChar *str )
1008{
1009 long int_key = 0;
1010
1011 while( *str )
1012 int_key += (wxUChar)*str++;
1013
1014 return int_key;
1015}
1016
2441fb44
VZ
1017// ----------------------------------------------------------------------------
1018// wxHashTable
1019// ----------------------------------------------------------------------------
1a6d9c76
MB
1020
1021wxHashTable::wxHashTable( const wxHashTable& table )
2441fb44 1022 : wxHashTableBase()
1a6d9c76
MB
1023{
1024 DoCopy( table );
1025}
1026
1027const wxHashTable& wxHashTable::operator=( const wxHashTable& table )
1028{
1029 Destroy();
1030 DoCopy( table );
1031
1032 return *this;
1033}
1034
6aa33060 1035void wxHashTable::DoCopy( const wxHashTable& WXUNUSED(table) )
1a6d9c76
MB
1036{
1037 Create( m_keyType, m_size );
1038
7cb32b4b 1039 wxFAIL;
1a6d9c76
MB
1040}
1041
1042void wxHashTable::DoDeleteContents( wxHashTableBase_Node* node )
1043{
1044 delete ((wxHashTable_Node*)node)->GetData();
1045}
1046
1047void wxHashTable::GetNextNode( size_t bucketStart )
1048{
1049 for( size_t i = bucketStart; i < m_size; ++i )
1050 {
1051 if( m_table[i] != NULL )
1052 {
1053 m_curr = ((Node*)m_table[i])->GetNext();
1054 m_currBucket = i;
1055 return;
1056 }
1057 }
1058
1059 m_curr = NULL;
1060 m_currBucket = 0;
1061}
1062
1063wxHashTable::Node* wxHashTable::Next()
1064{
1065 if( m_curr == NULL )
1066 GetNextNode( 0 );
1067 else
1068 {
1069 m_curr = m_curr->GetNext();
1070
1071 if( m_curr == ( (Node*)m_table[m_currBucket] )->GetNext() )
1072 GetNextNode( m_currBucket + 1 );
1073 }
1074
1075 return m_curr;
1076}
1077
6e86701b 1078#endif // !wxUSE_OLD_HASH_TABLE