cwUI.h/cpp : Added elementChildCount/UuId() and elementPhysChildCount/UuId().

emptyParent() now deletes child elements.
Added  _remove_ele_from_hash_table() to fix bug where destroyed elements were not removed from hash table.
This commit is contained in:
kevin 2024-10-19 12:36:12 -04:00
parent 6e56bec206
commit 7cc27cb8dd
2 changed files with 199 additions and 21 deletions

203
cwUi.cpp
View File

@ -141,18 +141,32 @@ namespace cw
void _store_ele_in_hash_table( ui_t* p, ele_t* e ) void _store_ele_in_hash_table( ui_t* p, ele_t* e )
{ {
unsigned parentUuId = e->logical_parent == nullptr ? kInvalidIdx : e->logical_parent->uuId; unsigned parentUuId = e->logical_parent == nullptr ? kInvalidId : e->logical_parent->uuId;
if( parentUuId == kInvalidId || e->appId == kInvalidId ) if( parentUuId == kInvalidId || e->appId == kInvalidId )
return; return;
unsigned short hash_idx = _gen_hash_index( parentUuId, e->appId ); unsigned short hash_idx = _gen_hash_index( parentUuId, e->appId );
// if the first bucket is empty ...
if( p->hashA[ hash_idx ].ele == nullptr ) if( p->hashA[ hash_idx ].ele == nullptr )
p->hashA[ hash_idx ].ele = e; p->hashA[ hash_idx ].ele = e; // ... then fill it
else else
{ {
bucket_t* b = mem::allocZ<bucket_t>(); // otherwise look for an empty bucket
bucket_t* b = p->hashA[ hash_idx ].link;
for(; b!=nullptr; b=b->link)
if( b->ele == nullptr )
{
b->ele = e; // an empty bucket was found - fill it
return;
}
// create a new bucket
b = mem::allocZ<bucket_t>();
// and insert it as the second bucket
b->link = p->hashA[ hash_idx ].link; b->link = p->hashA[ hash_idx ].link;
b->ele = e; b->ele = e;
p->hashA[hash_idx].link = b; p->hashA[hash_idx].link = b;
@ -173,9 +187,54 @@ namespace cw
} }
return kInvalidId; return kInvalidId;
} }
void _remove_ele_from_hash_table_0( ui_t* p, const ele_t* e )
{
// Note: hashA[] has hashN+1 elements
for(unsigned i=0; i<=hashN; ++i)
{
bucket_t* b = p->hashA + i;
for(; b!= nullptr; b=b->link)
if( b->ele == e )
{
b->ele = nullptr;
break;
}
}
}
void _remove_ele_from_hash_table( ui_t* p, const ele_t* e )
{
if( e == nullptr )
return;
unsigned parentUuId = e->logical_parent == nullptr ? kInvalidId : e->logical_parent->uuId;
unsigned appId = e->appId;
if( parentUuId == kInvalidId || appId == kInvalidId )
{
//_remove_ele_from_hash_table_0(p,e);
return;
}
unsigned hash_idx = _gen_hash_index(parentUuId,appId);
bucket_t* b = p->hashA + hash_idx;
// locate the bucket to delete
for(; b!=nullptr; b=b->link)
{
if( b->ele == e )
{
b->ele = nullptr;
return;
}
}
_remove_ele_from_hash_table_0(p,e);
}
void _print_eles( ui_t* p ) void _print_eles( ui_t* p )
{ {
for(unsigned i=0; i<p->eleN; ++i) for(unsigned i=0; i<p->eleN; ++i)
@ -199,7 +258,7 @@ namespace cw
return _is_child_of( parent, ele->phys_parent ) || _is_child_of( parent, ele->logical_parent ); return _is_child_of( parent, ele->phys_parent ) || _is_child_of( parent, ele->logical_parent );
} }
void _destroy_element( ele_t* e ) void _destroy_element( ui_t* p, ele_t* e )
{ {
if( e == nullptr ) if( e == nullptr )
return; return;
@ -219,8 +278,10 @@ namespace cw
// free each element // free each element
if( p->eleA != nullptr ) if( p->eleA != nullptr )
for(unsigned i=0; i<p->eleN; ++i) for(unsigned i=0; i<p->eleN; ++i)
_destroy_element( p->eleA[i] ); {
_destroy_element( p, p->eleA[i] );
p->eleA[i] = nullptr;
}
appIdMapRecd_t* m = p->appIdMap; appIdMapRecd_t* m = p->appIdMap;
while( m!=nullptr ) while( m!=nullptr )
@ -1975,10 +2036,28 @@ cw::rc_t cw::ui::setLogLine( handle_t h, unsigned uuId, const char* text )
cw::rc_t cw::ui::emptyParent( handle_t h, unsigned uuId ) cw::rc_t cw::ui::emptyParent( handle_t h, unsigned uuId )
{ {
rc_t rc = kOkRC;
unsigned childN = elementPhysChildCount(h,uuId);
unsigned* childUuIdA = nullptr;
ui_t* p = _handleToPtr(h); if( childN > 0 )
rc_t rc = kOkRC; {
childUuIdA = mem::alloc<unsigned>(childN);
unsigned actualChildN = 0;
if((rc = elementPhysChildUuId(h,uuId,childUuIdA,childN,actualChildN)) != kOkRC )
goto errLabel;
assert( actualChildN == childN );
for(unsigned i=0; i<childN; ++i)
if((rc = destroyElement(h,childUuIdA[i])) != kOkRC )
{
rc = cwLogError(rc,"Child element destroy failed.");
goto errLabel;
}
}
/*
const char* mFmt = "{ \"op\":\"empty\", \"uuId\":%i }"; const char* mFmt = "{ \"op\":\"empty\", \"uuId\":%i }";
const int mbufN = 256; const int mbufN = 256;
char mbuf[mbufN]; char mbuf[mbufN];
@ -1994,8 +2073,10 @@ cw::rc_t cw::ui::emptyParent( handle_t h, unsigned uuId )
cwLogError(rc,"'empty' msg transmit failed."); cwLogError(rc,"'empty' msg transmit failed.");
goto errLabel; goto errLabel;
} }
*/
errLabel: errLabel:
mem::release(childUuIdA);
return rc; return rc;
} }
@ -2134,6 +2215,87 @@ cw::rc_t cw::ui::clearBlob( handle_t h, unsigned uuId )
return kOkRC; return kOkRC;
} }
unsigned cw::ui::elementChildCount( handle_t h, unsigned uuId )
{
ui_t* p = _handleToPtr(h);
unsigned n = 0;
ele_t* ele;
if((ele = _uuIdToEle( p, uuId)) != nullptr )
for(unsigned i=0; i<p->eleN; ++i)
if( p->eleA[i] != nullptr && _is_child_of(ele, p->eleA[i] ))
++n;
return n;
}
cw::rc_t cw::ui::elementChildUuId( handle_t h, unsigned uuId, unsigned* bufA, unsigned bufN, unsigned& actualN )
{
rc_t rc = kOkRC;
ui_t* p = _handleToPtr(h);
unsigned n = 0;
ele_t* ele;
if((ele = _uuIdToEle( p, uuId)) != nullptr )
for(unsigned i=0; i<p->eleN; ++i)
if( p->eleA[i] != nullptr && _is_child_of(ele, p->eleA[i] ))
{
if( n >= bufN )
{
rc = cwLogError(kBufTooSmallRC,"The child ele. id buffer is too small.");
goto errLabel;
}
bufA[n++] = p->eleA[i]->uuId;
}
actualN = n;
errLabel:
return rc;
}
unsigned cw::ui::elementPhysChildCount( handle_t h, unsigned uuId )
{
ui_t* p = _handleToPtr(h);
unsigned n = 0;
ele_t* ele;
if((ele = _uuIdToEle( p, uuId)) != nullptr )
for(unsigned i=0; i<p->eleN; ++i)
if( p->eleA[i] != nullptr && p->eleA[i]->phys_parent == ele )
++n;
return n;
}
cw::rc_t cw::ui::elementPhysChildUuId( handle_t h, unsigned uuId, unsigned* bufA, unsigned bufN, unsigned& actualN )
{
rc_t rc = kOkRC;
ui_t* p = _handleToPtr(h);
unsigned n = 0;
ele_t* ele;
actualN = 0;
if((ele = _uuIdToEle( p, uuId)) != nullptr )
for(unsigned i=0; i<p->eleN; ++i)
if( p->eleA[i] != nullptr && p->eleA[i]->phys_parent == ele )
{
if( n >= bufN )
{
rc = cwLogError(kBufTooSmallRC,"The child ele. id buffer is too small.");
goto errLabel;
}
bufA[n++] = p->eleA[i]->uuId;
}
actualN = n;
errLabel:
return rc;
}
cw::rc_t cw::ui::destroyElement( handle_t h, unsigned uuId ) cw::rc_t cw::ui::destroyElement( handle_t h, unsigned uuId )
{ {
@ -2152,27 +2314,36 @@ cw::rc_t cw::ui::destroyElement( handle_t h, unsigned uuId )
// mark the element for deletion // mark the element for deletion
del_ele->destroyFl = true; del_ele->destroyFl = true;
_remove_ele_from_hash_table(p, del_ele);
// mark all child elements of 'del_ele' for deletion
// mark all child elements of 'del_ele' for deletion and remove them from the hash table
for(unsigned i=0; i<p->eleN; ++i) for(unsigned i=0; i<p->eleN; ++i)
if( p->eleA[i] != nullptr ) if( p->eleA[i] != nullptr && p->eleA[i] != del_ele )
p->eleA[i]->destroyFl = _is_child_of(del_ele, p->eleA[i] ); if((p->eleA[i]->destroyFl = _is_child_of(del_ele, p->eleA[i] )) == true )
_remove_ele_from_hash_table(p, p->eleA[i]);
// Note that all ele's that are going to be deleted
// must be first removed from the hash table.
// The ele's can't be removed from the hash table
// as they are deleted because the removal fromo the hash table
// requires accessing the logical parent - which may have already been deleted.
// release all elements that are marked for deletion // release all elements that are marked for deletion
for(unsigned i=0; i<p->eleN; ++i) for(unsigned i=0; i<p->eleN; ++i)
if( p->eleA[i] != nullptr && p->eleA[i]->destroyFl ) if( p->eleA[i] != nullptr && p->eleA[i]->destroyFl )
{ {
_destroy_element( p->eleA[i] ); _destroy_element( p, p->eleA[i] );
p->eleA[i] = nullptr; p->eleA[i] = nullptr;
} }
snprintf(mbuf,mbufN, "{ \"op\":\"destroy\", \"uuId\":%i }", del_ele->uuId );
snprintf(mbuf,mbufN, "{ \"op\":\"destroy\", \"uuId\":%i }", uuId );
_websockSend(p,kInvalidId,mbuf); _websockSend(p,kInvalidId,mbuf);
errLabel: errLabel:
if(rc != kOkRC ) if(rc != kOkRC )
rc = cwLogError(rc,"Element delete failed."); rc = cwLogError(rc,"Element delete failed: uuid:%i.",uuId);
return rc; return rc;
} }

7
cwUi.h
View File

@ -166,6 +166,13 @@ namespace cw
// Register parent/child/name app id's // Register parent/child/name app id's
rc_t registerAppIdMap( handle_t h, const appIdMap_t* map, unsigned mapN ); rc_t registerAppIdMap( handle_t h, const appIdMap_t* map, unsigned mapN );
unsigned elementChildCount( handle_t h, unsigned uuId );
rc_t elementChildUuId( handle_t h, unsigned uuId, unsigned* bufA, unsigned bufN, unsigned& actualN );
unsigned elementPhysChildCount( handle_t h, unsigned uuId );
rc_t elementPhysChildUuId( handle_t h, unsigned uuId, unsigned* bufA, unsigned bufN, unsigned& actualN );
// Release an element an all of it's children. // Release an element an all of it's children.
rc_t destroyElement( handle_t h, unsigned uuId ); rc_t destroyElement( handle_t h, unsigned uuId );