1 // BrowseRefsDlg.cpp : implementation file
\r
5 #include "TortoiseProc.h"
\r
6 #include "BrowseRefsDlg.h"
\r
10 // CBrowseRefsDlg dialog
\r
12 IMPLEMENT_DYNAMIC(CBrowseRefsDlg, CResizableStandAloneDialog)
\r
14 CBrowseRefsDlg::CBrowseRefsDlg(CWnd* pParent /*=NULL*/)
\r
15 : CResizableStandAloneDialog(CBrowseRefsDlg::IDD, pParent)
\r
20 CBrowseRefsDlg::~CBrowseRefsDlg()
\r
24 void CBrowseRefsDlg::DoDataExchange(CDataExchange* pDX)
\r
26 CDialog::DoDataExchange(pDX);
\r
27 DDX_Control(pDX, IDC_TREE_REF, m_RefTreeCtrl);
\r
28 DDX_Control(pDX, IDC_LIST_REF_LEAFS, m_ListRefLeafs);
\r
32 BEGIN_MESSAGE_MAP(CBrowseRefsDlg, CResizableStandAloneDialog)
\r
33 ON_BN_CLICKED(IDOK, &CBrowseRefsDlg::OnBnClickedOk)
\r
34 ON_NOTIFY(TVN_SELCHANGED, IDC_TREE_REF, &CBrowseRefsDlg::OnTvnSelchangedTreeRef)
\r
35 ON_NOTIFY(NM_RCLICK, IDC_LIST_REF_LEAFS, &CBrowseRefsDlg::OnNMRClickListRefLeafs)
\r
39 // CBrowseRefsDlg message handlers
\r
41 void CBrowseRefsDlg::OnBnClickedOk()
\r
46 BOOL CBrowseRefsDlg::OnInitDialog()
\r
48 CResizableStandAloneDialog::OnInitDialog();
\r
50 AddAnchor(IDC_TREE_REF, TOP_LEFT, BOTTOM_LEFT);
\r
51 AddAnchor(IDC_LIST_REF_LEAFS, TOP_LEFT, BOTTOM_RIGHT);
\r
53 m_ListRefLeafs.SetExtendedStyle(m_ListRefLeafs.GetExtendedStyle()|LVS_EX_FULLROWSELECT);
\r
54 m_ListRefLeafs.InsertColumn(0,L"Name",0,150);
\r
55 m_ListRefLeafs.InsertColumn(1,L"Date Last Commit",0,100);
\r
56 m_ListRefLeafs.InsertColumn(2,L"Last Commit",0,300);
\r
57 m_ListRefLeafs.InsertColumn(3,L"Hash",0,80);
\r
59 AddAnchor(IDOK,BOTTOM_RIGHT);
\r
60 AddAnchor(IDCANCEL,BOTTOM_RIGHT);
\r
68 CShadowTree* CShadowTree::GetNextSub(CString& nameLeft)
\r
70 int posSlash=nameLeft.Find('/');
\r
75 nameLeft.Empty();//Nothing left
\r
79 nameSub=nameLeft.Left(posSlash);
\r
80 nameLeft=nameLeft.Mid(posSlash+1);
\r
82 if(nameSub.IsEmpty())
\r
85 CShadowTree& nextNode=m_ShadowTree[nameSub];
\r
86 nextNode.m_csName=nameSub;
\r
87 nextNode.m_pParent=this;
\r
91 typedef std::map<CString,CString> MAP_STRING_STRING;
\r
93 void CBrowseRefsDlg::Refresh()
\r
95 // m_RefMap.clear();
\r
96 // g_Git.GetMapHashToFriendName(m_RefMap);
\r
99 m_RefTreeCtrl.DeleteAllItems();
\r
100 m_TreeRoot.m_ShadowTree.clear();
\r
101 m_TreeRoot.m_csName="Refs";
\r
102 m_TreeRoot.m_hTree=m_RefTreeCtrl.InsertItem(L"Refs",NULL,NULL);
\r
103 m_RefTreeCtrl.SetItemData(m_TreeRoot.m_hTree,(DWORD_PTR)&m_TreeRoot);
\r
106 g_Git.Run(L"git for-each-ref --format="
\r
108 L"%(objectname)%04"
\r
109 L"%(authordate:relative)%04"
\r
117 MAP_STRING_STRING refMap;
\r
119 //First sort on ref name
\r
120 while(!(singleRef=allRefs.Tokenize(L"\r\n",linePos)).IsEmpty())
\r
123 CString refName=singleRef.Tokenize(L"\04",valuePos);
\r
124 CString refRest=singleRef.Mid(valuePos);
\r
125 refMap[refName]=refRest;
\r
130 // for(MAP_HASH_NAME::iterator iterRef=m_RefMap.begin();iterRef!=m_RefMap.end();++iterRef)
\r
131 // for(STRING_VECTOR::iterator iterRefName=iterRef->second.begin();iterRefName!=iterRef->second.end();++iterRefName)
\r
132 // refName[*iterRefName]=iterRef->first;
\r
134 //Populate ref tree
\r
135 for(MAP_STRING_STRING::iterator iterRefMap=refMap.begin();iterRefMap!=refMap.end();++iterRefMap)
\r
137 CShadowTree& treeLeaf=GetTreeNode(iterRefMap->first);
\r
138 CString values=iterRefMap->second;
\r
141 treeLeaf.m_csRef= values.Tokenize(L"\04",valuePos);
\r
142 treeLeaf.m_csDate= values.Tokenize(L"\04",valuePos);
\r
143 treeLeaf.m_csSubject= values.Tokenize(L"\04",valuePos);
\r
144 treeLeaf.m_csAuthor= values.Tokenize(L"\04",valuePos);
\r
148 g_Git.Run(L"git symbolic-ref HEAD",&currHead,CP_UTF8);
\r
150 currHead.Trim(L"\r\n\t ");
\r
152 if(!SelectRef(currHead))
\r
153 //Probably not on a branch. Select root node.
\r
154 m_RefTreeCtrl.Expand(m_TreeRoot.m_hTree,TVE_EXPAND);
\r
158 bool CBrowseRefsDlg::SelectRef(CString refName)
\r
160 if(wcsnicmp(refName,L"refs/",5)!=0)
\r
161 return false; // Not a ref name
\r
163 CShadowTree& treeLeafHead=GetTreeNode(refName);
\r
164 if(treeLeafHead.m_pParent==NULL)
\r
165 return false; //Weird... should not occur.
\r
167 //This is the current head.
\r
168 m_RefTreeCtrl.Select(treeLeafHead.m_pParent->m_hTree,TVGN_CARET);
\r
170 for(int indexPos = 0; indexPos < m_ListRefLeafs.GetItemCount(); ++indexPos)
\r
172 CShadowTree* pCurrShadowTree = (CShadowTree*)m_ListRefLeafs.GetItemData(indexPos);
\r
173 if(pCurrShadowTree == &treeLeafHead)
\r
175 m_ListRefLeafs.SetItemState(indexPos,LVIS_SELECTED,LVIS_SELECTED);
\r
182 CShadowTree& CBrowseRefsDlg::GetTreeNode(CString refName, CShadowTree* pTreePos)
\r
186 if(wcsnicmp(refName,L"refs/",5)==0)
\r
187 refName=refName.Mid(5);
\r
188 pTreePos=&m_TreeRoot;
\r
190 if(refName.IsEmpty())
\r
191 return *pTreePos;//Found leaf
\r
193 CShadowTree* pNextTree=pTreePos->GetNextSub(refName);
\r
194 if(pNextTree==NULL)
\r
196 //Should not occur when all ref-names are valid.
\r
201 if(!refName.IsEmpty())
\r
203 //When the refName is not empty, this node is not a leaf, so lets add it to the tree control.
\r
204 //Leafs are for the list control.
\r
205 if(pNextTree->m_hTree==NULL)
\r
207 //New tree. Create node in control.
\r
208 pNextTree->m_hTree=m_RefTreeCtrl.InsertItem(pNextTree->m_csName,pTreePos->m_hTree,NULL);
\r
209 m_RefTreeCtrl.SetItemData(pNextTree->m_hTree,(DWORD_PTR)pNextTree);
\r
213 return GetTreeNode(refName,pNextTree);
\r
217 void CBrowseRefsDlg::OnTvnSelchangedTreeRef(NMHDR *pNMHDR, LRESULT *pResult)
\r
219 LPNMTREEVIEW pNMTreeView = reinterpret_cast<LPNMTREEVIEW>(pNMHDR);
\r
222 FillListCtrlForTreeNode(pNMTreeView->itemNew.hItem);
\r
225 void CBrowseRefsDlg::FillListCtrlForTreeNode(HTREEITEM treeNode)
\r
227 m_ListRefLeafs.DeleteAllItems();
\r
229 CShadowTree* pTree=(CShadowTree*)(m_RefTreeCtrl.GetItemData(treeNode));
\r
235 FillListCtrlForShadowTree(pTree,L"",true);
\r
238 void CBrowseRefsDlg::FillListCtrlForShadowTree(CShadowTree* pTree, CString refNamePrefix, bool isFirstLevel)
\r
240 if(pTree->IsLeaf())
\r
242 int indexItem=m_ListRefLeafs.InsertItem(m_ListRefLeafs.GetItemCount(),L"");
\r
244 m_ListRefLeafs.SetItemData(indexItem,(DWORD_PTR)pTree);
\r
245 m_ListRefLeafs.SetItemText(indexItem,0,refNamePrefix+pTree->m_csName);
\r
246 m_ListRefLeafs.SetItemText(indexItem,1,pTree->m_csDate);
\r
247 m_ListRefLeafs.SetItemText(indexItem,2,pTree->m_csSubject);
\r
248 m_ListRefLeafs.SetItemText(indexItem,3,pTree->m_csRef);
\r
253 CString csThisName;
\r
255 csThisName=refNamePrefix+pTree->m_csName+L"/";
\r
256 for(CShadowTree::TShadowTreeMap::iterator itSubTree=pTree->m_ShadowTree.begin(); itSubTree!=pTree->m_ShadowTree.end(); ++itSubTree)
\r
258 FillListCtrlForShadowTree(&itSubTree->second,csThisName,false);
\r
263 void CBrowseRefsDlg::OnNMRClickListRefLeafs(NMHDR *pNMHDR, LRESULT *pResult)
\r
265 // LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);
\r
268 int selectedItemCount=m_ListRefLeafs.GetSelectedCount();
\r
271 std::vector<CShadowTree*> selectedTrees;
\r
272 selectedTrees.reserve(selectedItemCount);
\r
273 POSITION pos=m_ListRefLeafs.GetFirstSelectedItemPosition();
\r
276 selectedTrees.push_back(
\r
277 (CShadowTree*)m_ListRefLeafs.GetItemData(
\r
278 m_ListRefLeafs.GetNextSelectedItem(pos)));
\r
282 popupMenu.CreatePopupMenu();
\r
284 if(selectedItemCount==1)
\r
286 popupMenu.AppendMenu(MF_STRING,eCmd_ViewLog,L"View log");
\r
288 // CShadowTree* pTree = (CShadowTree*)m_ListRefLeafs.GetItemData(pNMHDR->idFrom);
\r
294 const MSG* pCurrMsg=GetCurrentMessage();
\r
295 eCmd cmd=(eCmd)popupMenu.TrackPopupMenuEx(TPM_LEFTALIGN|TPM_RETURNCMD, pCurrMsg->pt.x, pCurrMsg->pt.y, this, 0);
\r
301 dlg.SetStartRef(selectedTrees[0]->m_csRef);
\r