分享
 
 
 

C++ Builder高手进阶 (三)用BCB设计DBTreeView组件(续二)

王朝c/c++·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

//续二

void __fastcall TDBTreeView::FillChildTreeNodes(int ParentID, TTreeNode* Node, bool Nest)

{//用指定值填充子节点

TQuery* AQuery = new TQuery(this);//创建一个数据集控件

AnsiString strSql, strText;

int iID, iParentID;

int iImageIndex, iSelectedIndex;

TTreeNode* TreeNode;

TDBTreeNodeData NodeData;

//设置数据集的各项参数

AQuery->DatabaseName = ((TTable*)(FDataLink->DataSource->DataSet))->DatabaseName;//设置数据库名

AQuery->Close();//关闭数据集

AQuery->SQL->Clear();//清空原SQL语句

strSql = "SELECT * FROM " + ((TTable*)(FDataLink->DataSource->DataSet))->TableName + " WHERE ";

strSql += FParentIDField + "=:PID";

AQuery->SQL->Add(strSql);//指定新的SQL语句

AQuery->ParamByName("PID")->AsInteger = ParentID;

AQuery->Open();//打开数据集

this->Items->BeginUpdate();//开始更新DBTreeView组件显示

while (!AQuery->Eof)//未到数据集尾部

{

strText = AQuery->FieldByName(FDisplayField)->AsString;//取得显示字段内容

iID = AQuery->FieldByName(FPrimaryIDField)->AsInteger;//取得主标识字段内容

iParentID = AQuery->FieldByName(FParentIDField)->AsInteger;//取得父标识字段内容

TreeNode = this->Items->AddChild(Node, strText);//将上面取得的相关数据追加到新节点

iImageIndex = iSelectedIndex = -1;//设置节点位图与选中时位图

if (FOnSetImageIndex) FOnSetImageIndex(this, iID, iParentID, TreeNode->Level, iImageIndex, iSelectedIndex);//如果设置位图发生变化,则触发相关事件

TreeNode->ImageIndex = iImageIndex;//设置节点位图及选中时位图

TreeNode->SelectedIndex = iSelectedIndex;

NodeData.ID = iID;//取得节点主标识,父标识,并按其追加一个新节点

NodeData.ParentID = iParentID;

AddDataToNode(TreeNode, NodeData);

if (Nest) FillChildTreeNodes(iID, TreeNode);//如果指定参数Nest为true,则递归调用以填充所有子节点

AQuery->Next();//移动到下一条记录

}

this->Items->EndUpdate();//结束DBTreeView组件更新

AQuery->Close();//关闭数据集

delete AQuery;//删除临时创建的数据集控件

}

void __fastcall TDBTreeView::AddDataToNode(TTreeNode* Node, TDBTreeNodeData& Data)

{//追加数据到节点

TDBTreeNodeData* pData = new TDBTreeNodeData;

*pData = Data;

Node->Data = pData;

}

//---------------------------------------------------------------------------

TDBTreeNodeData __fastcall TDBTreeView::GetNodeData(TTreeNode* Node)

{//取得指定节点的数据

return *(TDBTreeNodeData*)(Node->Data);

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::Loaded(void)

{//调用原Load方法

TCustomTreeView::Loaded();

/* TODO : Loaded */

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::Notification(Classes::TComponent* AComponent, Classes::TOperation Operation)

{

//事件响应

TCustomTreeView::Notification(AComponent, Operation);//调用原方法

if ((Operation == opRemove) && (FDataLink != NULL) && (AComponent == DataSource))//如果操作为opRemove以及数据连接不为空且指定组件为数据源,则

DataSource = NULL;//使数据源为空

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::Edit(const tagTVITEMA &Item)

{//编辑节点

TCustomTreeView::Edit(Item);//调用原方法

if (FAllowModifyDB)//如果允许修改,则

{

TTreeNode* Node;

AnsiString DatabaseName = ((TTable*)(FDataLink->DataSource->DataSet))->DatabaseName;

AnsiString TableName = ((TTable*)(FDataLink->DataSource->DataSet))->TableName;

TDBTreeNodeData NodeData;

TQuery* AQuery = new TQuery(this);//新建一个数据集

AnsiString strSql;

if ((Item.state & TVIF_PARAM) != 0) Node = (TTreeNode*)(Item.lParam);

else Node = Items->GetNode(Item.hItem);

NodeData = GetNodeData(Node);

strSql = "UPDATE " + TableName + " SET " + FDisplayField + " =:NewDispText WHERE " + FPrimaryIDField + "=:ID";//用指定条件更新数据集

AQuery->Close();//关闭数据集

AQuery->DatabaseName = DatabaseName;//设置数据库连接

AQuery->SQL->Clear();//清空原SQL语句

AQuery->SQL->Add(strSql);//追加SQL语句

AQuery->ParamByName("NewDispText")->AsString = Node->Text;//对SQL中的参数进行赋值

AQuery->ParamByName("ID")->AsInteger = NodeData.ID;

AQuery->ExecSQL();//打开数据集

delete AQuery;//删除临时创建的数据集控件

}

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::Expand(TTreeNode* Node)

{//展开指定节点

TCustomTreeView::Expand(Node);//调用原原方法

TDBTreeNodeData NodeData;

TTreeNode* ANode;

for (ANode = Node->getFirstChild(); ANode; ANode = Node->GetNextChild(ANode))//遍历指定节点的子节点

{

NodeData = GetNodeData(ANode);//取得节点数据

if (ANode->Count == 0)//如果节点数据为0,则填充其子节点

FillChildTreeNodes(NodeData.ID, ANode);

}

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::KeyDown(Word &Key, Classes::TShiftState Shift)

{

//键盘处理

TWinControl::KeyDown(Key, Shift);//调用原键盘处理方法

if (Key == VK_F2 && Shift == TShiftState())//如果是Shift+F2则

{

/* Handle 'F2' key */

if (this->Selected != NULL)//编辑当前选中的节点

this->Selected->EditText();

}

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::Change(TTreeNode* Node)

{//处理节点发生变化事件

TCustomTreeView::Change(Node);//调用原方法

static TTreeNode* OldNode = NULL;

TTreeNode* SelectedNode = this->Selected;//取得已选节点指针

TDBTreeNodeData NodeData;

TTable *ATable = (TTable*)(FDataLink->DataSource->DataSet);//指定数据集

if (OldNode == SelectedNode) return;//如果旧节点等于选中的节点则直接返回

if (ATable == NULL) return;//如果数据集为空则直接返回

if (SelectedNode == NULL) return;//如果没有已选节点则直接返回

NodeData = GetNodeData(SelectedNode);//取得指定节点数据

ATable->SetKey();//按指定条件查找数据记录

ATable->FieldByName(FPrimaryIDField)->AsInteger = NodeData.ID;

ATable->GotoKey();

OldNode = SelectedNode;//旧节点等于现在已选节点

}

//---------------------------------------------------------------------------

bool __fastcall TDBTreeView::CustomDrawItem(TTreeNode* Node, TCustomDrawState State, TCustomDrawStage Stage, bool &PaintImages)

{

//自绘节点

bool Result;//调用原自绘方法

Result = TCustomTreeView::CustomDrawItem(Node, State, Stage, PaintImages);

/* TODO : ... */

//TDBTreeNodeData NodeData = GetNodeData(Node);

return Result;

}

//---------------------------------------------------------------------------

void __fastcall TDBTreeView::FullExpand(void)

{//展开所有节点

if (!Active)//如果DBTreeView不是活动状态,则抛出一个异常

throw Exception("DBTreeView is Inactive.");

this->Items->BeginUpdate();//开始更新DBTreeView

ClearAllNodes();//清空所有节点

FillChildTreeNodes(0, NULL, true);//填充子节点

this->Items->EndUpdate();//结束更新

TCustomTreeView::FullExpand();//调用原方法实现展开所有节点

}

//---------------------------------------------------------------------------

void __fastcall TFieldNameProperty::GetValues(Classes::TGetStrProc Proc)

{//取得字段值

int i;

TDBTreeView* ADBTreeView;

ADBTreeView = (TDBTreeView*)GetComponent(0);

if (ADBTreeView->DataSource != NULL)//如果DBTreeView的数据源不为空,则

{//遍历所有字段,并将其填充到相关字段属性中

for (i = 0; i < ADBTreeView->DataSource->DataSet->FieldCount; i ++)

Proc(ADBTreeView->DataSource->DataSet->Fields->Fields[i]->FieldName);

}

else

{//否则抛出异常

throw Exception("Missing DataSource property.");

}

}

TTypeInfo* AnsiStringTypeInfo(void)

{

//定义类型信息

TTypeInfo* TypeInfo = new TTypeInfo;

TypeInfo->Name = "AnsiString";

TypeInfo->Kind = tkLString;

return TypeInfo;

}

《未完,见一篇DBTreeView开发总结》

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有