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:
parent
6e56bec206
commit
7cc27cb8dd
203
cwUi.cpp
203
cwUi.cpp
@ -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
7
cwUi.h
@ -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 );
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user