全国高分辨率土地利用数据服务 土地利用数据服务 土地覆盖数据服务 坡度数据服务 土壤侵蚀数据服务 全国各省市DEM数据服务 耕地资源空间分布数据服务 草地资源空间分布数据服务 林地资源空间分布数据服务 水域资源空间分布数据服务 建设用地空间分布数据服务 地形、地貌、土壤数据服务 分坡度耕地数据服务 全国大宗农作物种植范围空间分布数据服务
多种卫星遥感数据反演植被覆盖度数据服务 地表反照率数据服务 比辐射率数据服务 地表温度数据服务 地表蒸腾与蒸散数据服务 归一化植被指数数据服务 叶面积指数数据服务 净初级生产力数据服务 净生态系统生产力数据服务 生态系统总初级生产力数据服务 生态系统类型分布数据服务 土壤类型质地养分数据服务 生态系统空间分布数据服务 增强型植被指数数据服务
多年平均气温空间分布数据服务 多年平均降水量空间分布数据服务 湿润指数数据服务 大于0℃积温空间分布数据服务 光合有效辐射分量数据服务 显热/潜热信息数据服务 波文比信息数据服务 地表净辐射通量数据服务 光合有效辐射数据服务 温度带分区数据服务 山区小气候因子精细数据服务
全国夜间灯光指数数据服务 全国GDP公里格网数据服务 全国建筑物总面积公里格网数据服务 全国人口密度数据服务 全国县级医院分布数据服务 人口调查空间分布数据服务 收入统计空间分布数据服务 矿山面积统计及分布数据服务 载畜量及空间分布数据服务 农作物种植面积统计数据服务 农田分类面积统计数据服务 农作物长势遥感监测数据服务 医疗资源统计数据服务 教育资源统计数据服务 行政辖区信息数据服务
Landsat 8 高分二号 高分一号 SPOT-6卫星影像 法国Pleiades高分卫星 资源三号卫星 风云3号 中巴资源卫星 NOAA/AVHRR MODIS Landsat TM 环境小卫星 Landsat MSS 天绘一号卫星影像
上面我们实现了从shp文件加载图层、图层管理的功能。里面只是简单的实现了这些功能而已,为的是抛砖引玉。
// 删除图层
void CLayerManageDlg::OnBtnDel()
{
int iCurSel = m_listLayer.GetSelectionMark();
if(iCurSel<0) return;
CString sztemp;
sztemp.Format("要删除图层:“%s”吗?",m_listLayer.GetItemText(iCurSel,0));
if(IDNO == MessageBox(sztemp,"确认",MB_ICONQUESTION+MB_YESNO)) return;
IMapPtr ipMap;
HRESULT hr = m_ipMapControl->get_Map(&ipMap);
if (FAILED(hr)) return ;
// 获取要删除的图层
ILayerPtr ipLayer;
hr = ipMap->get_Layer(iCurSel,&ipLayer);
if(FAILED(hr)) return ;
ipMap->DeleteLayer(ipLayer);
if(FAILED(hr)) return ;
m_listLayer.DeleteItem(iCurSel);
}
保存、编译、运行,先点击工具栏添加图层数据,然后点击图层管理按钮,出现图层管理的对话框,在ListCtrl中选择一行,点击按钮:上移、下移、删除,可以看到相应的效果。
在图层列表中每行记录前面的复选框是用来控制图层是否可见的,因为在ListCtrl的事件函数里面没有专门处理复选框的点击消息的函数,所以我们从ClistCtrl派生一个新的类专门处理它。
在Classview页右击项目名称在出现的菜单里面选择New Class…,添加一个新类ClistCtrlEx,类型选择MFC Class 基类选择ClistCtrl。
在生成的新类中添加一个自定义消息,当用户点击复选框时向父窗口发送此消息:
#define WM_CLICK_CHECKBOX WM_USER+100
然后添加一个WM_LBUTTONDOWN事件处理函数,并在其中加入如下代码:
void CListCtrlEx::OnLButtonDown(UINT nFlags, CPoint point)
{
CListCtrl::OnLButtonDown(nFlags, point);
LVHITTESTINFO lvhti;
lvhti.pt = point;
SubItemHitTest(&lvhti);
// 点击复选框时向父窗口发WM_CLICK_CHECKBOX消息
if(!(lvhti.flags&LVHT_ONITEMLABEL)&&(lvhti.flags&LVHT_ONITEMSTATEICON))
{
BOOL bChecked = GetCheck(lvhti.iItem);
GetParent()->PostMessage(WM_CLICK_CHECKBOX,(WPARAM)lvhti.iItem,(LPARAM)bChecked);
}
}
接下来修改图层设置对话框类ClayerManageDlg,在头文件里把成员变量m_listLayer由CListCtrl改为ClistCtrlEx,并添加CListCtrlEx的头文件包含语句,然后添加处理CListCtrlEx自定义消息的处理函数,在头文件里添加如下函数:
afx_msg LRESULT onClickCheckbox(WPARAM wParam,LPARAM lParam);
在cpp文件里添加消息映射和实现函数:
ON_MESSAGE(WM_CLICK_CHECKBOX,onClickCheckbox)
// 响应点击ListCtrl的Checkbox事件
LRESULT CLayerManageDlg::onClickCheckbox(WPARAM wParam,LPARAM lParam)
{
int iItem = (int)wParam;// 点击的Item
BOOL bChecked = (BOOL)lParam;// 当前Checkbox状态
IMapPtr ipMap;
HRESULT hr = m_ipMapControl->get_Map(&ipMap);
if (FAILED(hr)) return 0L;
// 根据索引获取图层
ILayerPtr ipLayer;
hr = ipMap->get_Layer(iItem,&ipLayer);
if (FAILED(hr)) return 0L;
// 设置图层可见状态
ipLayer->put_Visible(bChecked);
RefreshMap();
return 0L;
}
保存修改、编译程序,运行,当打开图层管理点击某个图层前面的复选框时会看到图层隐藏或显示。
三、本章小结
上面我们实现了从shp文件加载图层、图层管理的功能。里面只是简单的实现了这些功能而已,为的是抛砖引玉。你可以根据自己的实际要求,也可以把图层管理做出象ArcMap里面一样,用一棵树来控制图层的显示和顺序的调整。其实只是表现形式不同,用到的AO里面的接口和函数都是一样的。我们在下一节中将实现对地图要素一些操作,如:要素的选择、要素属性的查看、修改、要素删除等功能。