Mercurial and Git clients can push and pull from this alias URL to interact with this repository. You can change to which repository an alias points by going to the Aliases link on the project page.
# manifestdialog.py - Dialog and widget for TortoiseHg manifest view## Copyright (C) 2003-2010 LOGILAB S.A. <http://www.logilab.fr/># Copyright (C) 2010 Yuya Nishihara <yuya@tcha.org>## This program is free software; you can redistribute it and/or modify it under# the terms of the GNU General Public License as published by the Free Software# Foundation; either version 2 of the License, or (at your option) any later# version.importosfrommercurialimportutilfromPyQt4.QtCoreimport*fromPyQt4.QtGuiimport*fromtortoisehg.utilimportpaths,hglibfromtortoisehg.hgqt.i18nimport_fromtortoisehg.hgqtimportqtlib,qscilib,fileview,status,thgrepofromtortoisehg.hgqtimportvisdiff,revert,revpanel,workbenchfromtortoisehg.hgqt.filedialogsimportFileLogDialog,FileDiffDialogfromtortoisehg.hgqt.manifestmodelimportManifestModelclassManifestDialog(QMainWindow):""" Qt4 dialog to display all files of a repo at a given revision """finished=pyqtSignal(int)linkActivated=pyqtSignal(QString)def__init__(self,repo,rev=None,parent=None):QMainWindow.__init__(self,parent)self._repo=repoself.setWindowIcon(qtlib.geticon('hg-annotate'))self.resize(400,300)self._manifest_widget=ManifestWidget(repo,rev)self._manifest_widget.revChanged.connect(self._updatewindowtitle)self._manifest_widget.pathChanged.connect(self._updatewindowtitle)self._manifest_widget.grepRequested.connect(self._openSearchWidget)self._manifest_widget.setContentsMargins(10,10,10,10)self.setCentralWidget(self._manifest_widget)self.addToolBar(self._manifest_widget.toolbar)self.setStatusBar(QStatusBar())self._manifest_widget.showMessage.connect(self.statusBar().showMessage)self._manifest_widget.linkActivated.connect(self.linkActivated)self._readsettings()self._updatewindowtitle()@pyqtSlot()def_updatewindowtitle(self):self.setWindowTitle(_('Manifest %s@%s')%(self._manifest_widget.path,self._manifest_widget.rev))defcloseEvent(self,event):self._writesettings()super(ManifestDialog,self).closeEvent(event)self.finished.emit(0)# mimic QDialog exitdef_readsettings(self):s=QSettings()self.restoreGeometry(s.value('manifest/geom').toByteArray())self._manifest_widget.loadSettings(s,'manifest')def_writesettings(self):s=QSettings()s.setValue('manifest/geom',self.saveGeometry())self._manifest_widget.saveSettings(s,'manifest')defsetSource(self,path,rev,line=None):self._manifest_widget.setSource(path,rev,line)defsetSearchPattern(self,text):"""Set search pattern [unicode]"""self._manifest_widget._fileview.searchbar.setPattern(text)defsetSearchCaseInsensitive(self,ignorecase):"""Set if search is case insensitive"""self._manifest_widget._fileview.searchbar.setCaseInsensitive(ignorecase)@pyqtSlot(unicode,dict)def_openSearchWidget(self,pattern,opts):opts=dict((str(k),str(v))fork,vinopts.iteritems())fromtortoisehg.hgqtimportrunrun.grep(self._repo.ui,hglib.fromunicode(pattern),**opts)@pyqtSlot(unicode,object,int)def_openInEditor(self,path,rev,line):"""Open editor to show the specified file"""_openineditor(self._repo,path,rev,line,pattern=self._fileview.searchbar.pattern(),parent=self)classManifestWidget(QWidget,qtlib.TaskWidget):"""Display file tree and contents at the specified revision"""revChanged=pyqtSignal(object)"""Emitted (rev) when the current revision changed"""pathChanged=pyqtSignal(unicode)"""Emitted (path) when the current file path changed"""showMessage=pyqtSignal(unicode)"""Emitted when to show revision summary as a hint"""grepRequested=pyqtSignal(unicode,dict)"""Emitted (pattern, opts) when user request to search changelog"""linkActivated=pyqtSignal(QString)"""Emitted (path) when user clicks on link"""filecontextmenu=Nonesubrepocontextmenu=Nonedefcanswitch(self):returnFalsedef__init__(self,repo,rev=None,parent=None):super(ManifestWidget,self).__init__(parent)self._repo=repoself._rev=revself._selectedrev=revself._diff_dialogs={}self._nav_dialogs={}self._initwidget()self._initactions()self._setupmodel()self._treeview.setCurrentIndex(self._treemodel.index(0,0))self.setRev(self._rev)def_initwidget(self):self.setLayout(QVBoxLayout())self._splitter=QSplitter()self.layout().addWidget(self._splitter)self.layout().setContentsMargins(2,2,2,2)navlayout=QVBoxLayout(spacing=0)navlayout.setContentsMargins(0,0,0,0)self._toolbar=QToolBar()self._toolbar.setIconSize(QSize(16,16))self._toolbar.setStyleSheet(qtlib.tbstylesheet)self._treeview=QTreeView(self,headerHidden=True,dragEnabled=True)self._treeview.setContextMenuPolicy(Qt.CustomContextMenu)self._treeview.customContextMenuRequested.connect(self.menuRequest)self._treeview.doubleClicked.connect(self.onDoubleClick)navlayout.addWidget(self._toolbar)navlayout.addWidget(self._treeview)navlayoutw=QWidget()navlayoutw.setLayout(navlayout)self._splitter.addWidget(navlayoutw)self._splitter.setStretchFactor(0,1)vbox=QVBoxLayout(spacing=0)vbox.setMargin(0)self.revpanel=revpanel.RevPanelWidget(self._repo)self.revpanel.linkActivated.connect(self.linkActivated)vbox.addWidget(self.revpanel,0)self._fileview=fileview.HgFileView(self._repo,self)vbox.addWidget(self._fileview,0)w=QWidget()w.setLayout(vbox)self._splitter.addWidget(w)self._splitter.setStretchFactor(1,3)self._fileview.revisionSelected.connect(self.setRev)self._fileview.linkActivated.connect(self.linkActivated)fornamein('showMessage','grepRequested'):getattr(self._fileview,name).connect(getattr(self,name))defloadSettings(self,qs,prefix):prefix+='/manifest'self._fileview.loadSettings(qs,prefix+'/fileview')self._splitter.restoreState(qs.value(prefix+'/splitter').toByteArray())expanded=qs.value(prefix+'/revpanel.expanded',False).toBool()self.revpanel.set_expanded(expanded)defsaveSettings(self,qs,prefix):prefix+='/manifest'self._fileview.saveSettings(qs,prefix+'/fileview')qs.setValue(prefix+'/splitter',self._splitter.saveState())qs.setValue(prefix+'/revpanel.expanded',self.revpanel.is_expanded())def_initactions(self):self._statusfilter=status.StatusFilterButton(statustext='MASC',text=_('Status'))self._toolbar.addWidget(self._statusfilter)self._actions={}forname,desc,icon,key,tip,cbin[('navigate',_('File history'),'hg-log','Shift+Return',_('Show the history of the selected file'),self.navigate),('diffnavigate',_('Compare file revisions'),'compare-files',None,_('Compare revisions of the selected file'),self.diffNavigate),('diff',_('Visual Diff'),'visualdiff','Ctrl+D',_('View file changes in external diff tool'),self.vdiff),('ldiff',_('Visual Diff to Local'),'ldiff','Shift+Ctrl+D',_('View changes to current in external diff tool'),self.vdifflocal),('edit',_('View at Revision'),'view-at-revision','Alt+Ctrl+E',_('View file as it appeared at this revision'),self.editfile),('ledit',_('Edit Local'),'edit-file','Shift+Ctrl+E',_('Edit current file in working copy'),self.editlocal),('revert',_('Revert to Revision'),'hg-revert','Alt+Ctrl+T',_('Revert file(s) to contents at this revision'),self.revertfile),('opensubrepo',_('Open subrepository'),'thg-repository-open','Alt+Ctrl+O',_('Open the selected subrepository'),self.opensubrepo),('explore',_('Explore subrepository'),'system-file-manager','Alt+Ctrl+E',_('Open the selected subrepository in a file browser'),self.explore),('terminal',_('Open terminal in subrepository'),'utilities-terminal','Alt+Ctrl+T',_('Open a shell terminal in the selected subrepository root'),self.terminal),]:act=QAction(desc,self)ificon:act.setIcon(qtlib.getmenuicon(icon))ifkey:act.setShortcut(key)iftip:act.setStatusTip(tip)ifcb:act.triggered.connect(cb)self._actions[name]=actself.addAction(act)defnavigate(self,filename=None):self._navigate(filename,FileLogDialog,self._nav_dialogs)defdiffNavigate(self,filename=None):self._navigate(filename,FileDiffDialog,self._diff_dialogs)defvdiff(self):ifself.pathisNone:returnpats=[self.path]opts={'change':self.rev}dlg=visdiff.visualdiff(self._repo.ui,self._repo,pats,opts)ifdlg:dlg.exec_()defvdifflocal(self):ifself.pathisNone:returnpats=[self.path]asserttype(self.rev)isintopts={'rev':['rev(%d)'%self.rev]}dlg=visdiff.visualdiff(self._repo.ui,self._repo,pats,opts)ifdlg:dlg.exec_()defeditfile(self):ifself.pathisNone:returnifself.revisNone:qtlib.editfiles(self._repo,[self.path],parent=self)else:base,_=visdiff.snapshot(self._repo,[self.path],self._repo[self.rev])files=[os.path.join(base,self.path)]qtlib.editfiles(self._repo,files,parent=self)defeditlocal(self):ifself.pathisNone:returnqtlib.editfiles(self._repo,[self.path],parent=self)defrevertfile(self):ifself.pathisNone:returnrev=self.revifrevisNone:rev=self._repo['.'].rev()dlg=revert.RevertDialog(self._repo,[self.path],rev,self)dlg.exec_()def_navigate(self,filename,dlgclass,dlgdict):ifnotfilename:filename=self.pathiffilenamenotindlgdict:repoviewer=self.window()ifnotisinstance(repoviewer,workbench.Workbench):repoviewer=Nonedlg=dlgclass(self._repo,filename,repoviewer)dlgdict[filename]=dlgufname=hglib.tounicode(filename)dlg.setWindowTitle(_('Hg file log viewer - %s')%ufname)dlg=dlgdict[filename]dlg.goto(self.rev)dlg.show()dlg.raise_()dlg.activateWindow()defopensubrepo(self):path=self._repo.wjoin(self.path)ifos.path.isdir(path):self.linkActivated.emit(u'subrepo:'+hglib.tounicode(path))else:QMessageBox.warning(self,_("Cannot open subrepository"),_("The selected subrepository does not exist on the working directory"))defexplore(self):root=self._repo.wjoin(self.path)ifos.path.isdir(root):QDesktopServices.openUrl(QUrl.fromLocalFile(root))defterminal(self):root=self._repo.wjoin(self.path)ifos.path.isdir(root):qtlib.openshell(root,self.path)defshowEvent(self,event):QWidget.showEvent(self,event)ifself._selectedrev!=self._rev:# If the selected revision is not the same as the current revision# we must "reload" the manifest contents with the selected revisionself.setRev(self._selectedrev)#@pyqtSlot(QModelIndex)defonDoubleClick(self,index):itemissubrepo=(self._treemodel.fileStatus(index)=='S')ifitemissubrepo:self.opensubrepo()else:self.vdiff()defmenuRequest(self,point):selmodel=self._treeview.selectionModel()ifnotselmodel.selectedRows():returnpoint=self._treeview.viewport().mapToGlobal(point)currentindex=self._treeview.currentIndex()itemissubrepo=(self._treemodel.fileStatus(currentindex)=='S')# Subrepos and regular items have different context menusifitemissubrepo:contextmenu=self.subrepocontextmenuactionlist=['opensubrepo','explore','terminal']else:contextmenu=self.filecontextmenuactionlist=['diff','ldiff','edit','ledit','revert','navigate','diffnavigate']ifnotcontextmenu:contextmenu=QMenu(self)foractinactionlist:ifact:contextmenu.addAction(self._actions[act])else:contextmenu.addSeparator()ifitemissubrepo:self.subrepocontextmenu=contextmenuelse:self.filecontextmenu=contextmenuifactionlist:contextmenu.exec_(point)@propertydeftoolbar(self):"""Return toolbar for manifest widget"""returnself._toolbar@pyqtSlot(unicode,bool,bool,bool)deffind(self,pattern,icase=False,wrap=False,forward=True):returnself._fileview.find(pattern,icase,wrap,forward)@pyqtSlot(unicode,bool)defhighlightText(self,pattern,icase=False):self._fileview.highlightText(pattern,icase)def_setupmodel(self):self._treemodel=ManifestModel(self._repo,self._rev,statusfilter=self._statusfilter.status(),parent=self)oldmodel=self._treeview.model()oldselmodel=self._treeview.selectionModel()self._treeview.setModel(self._treemodel)ifoldmodel:oldmodel.deleteLater()ifoldselmodel:oldselmodel.deleteLater()selmodel=self._treeview.selectionModel()selmodel.currentChanged.connect(self._updatecontent)selmodel.currentChanged.connect(self._emitPathChanged)self._statusfilter.statusChanged.connect(self._treemodel.setStatusFilter)self._statusfilter.statusChanged.connect(self._autoexpandtree)self._autoexpandtree()@pyqtSlot()def_autoexpandtree(self):"""expand file tree if the number of the items isn't large"""if'C'notinself._statusfilter.status():self._treeview.expandAll()defreload(self):# TODOpassdefsetRepo(self,repo):self._repo=repo#self._fileview.setRepo(repo)self._fileview.repo=repoiflen(repo)<=self._rev:self._rev=len(repo)-1self._setupmodel()@propertydefrev(self):"""Return current revision"""returnself._revdefselectRev(self,rev):""" Select the revision that must be set when the dialog is shown again """self._selectedrev=rev@pyqtSlot(int)@pyqtSlot(object)defsetRev(self,rev):"""Change revision to show"""self._selectedrev=revself.revpanel.set_revision(rev)self.revpanel.update(repo=self._repo)ifrev==self._rev:returnself._rev=revpath=self.pathself.revChanged.emit(rev)self._setupmodel()ctx=self._repo[rev]ifpathandpathinctx:# recover file selection after reloading the modelself.setPath(path)self._fileview.setContext(ctx)self._fileview.displayFile(self.path,self.status)# update sensitivity of actionsreal=type(rev)isintself._actions['ldiff'].setEnabled(real)foractin['diff','edit']:self._actions[act].setEnabled(realorrevisNone)self._actions['revert'].setEnabled(real)@pyqtSlot(unicode,object)@pyqtSlot(unicode,object,int)defsetSource(self,path,rev,line=None):"""Change path and revision to show at once"""ifself._rev!=rev:self._rev=revself._setupmodel()self.revChanged.emit(rev)ifpath!=self.path:self.setPath(path)ctx=self._repo[rev]ifself.pathinctx:self._fileview.displayFile(path,self.status)ifline:self._fileview.showLine(int(line)-1)else:self._fileview.clearDisplay()@propertydefpath(self):"""Return currently selected path"""returnself._treemodel.filePath(self._treeview.currentIndex())@propertydefstatus(self):"""Return currently selected path"""returnself._treemodel.fileStatus(self._treeview.currentIndex())@pyqtSlot(unicode)defsetPath(self,path):"""Change path to show"""self._treeview.setCurrentIndex(self._treemodel.indexFromPath(path))defdisplayFile(self):ctx,path=self._treemodel.fileSubrepoCtxFromPath(self.path)ifctxisNone:ctx=self._repo[self._rev]else:ctx._repo.tabwidth=self._repo.tabwidthctx._repo.maxdiff=self._repo.maxdiffself._fileview.setContext(ctx)self._fileview.displayFile(path,self.status)@pyqtSlot()def_updatecontent(self):self.displayFile()@pyqtSlot()def_emitPathChanged(self):self.pathChanged.emit(self.path)defconnectsearchbar(manifestwidget,searchbar):"""Connect searchbar to manifest widget"""searchbar.conditionChanged.connect(manifestwidget.highlightText)searchbar.searchRequested.connect(manifestwidget.find)def_openineditor(repo,path,rev,line=None,pattern=None,parent=None):"""Open editor to show the specified file [unicode]"""path=hglib.fromunicode(path)pattern=hglib.fromunicode(pattern)base=visdiff.snapshot(repo,[path],repo[rev])[0]files=[os.path.join(base,path)]qtlib.editfiles(repo,files,line,pattern,parent=self)defrun(ui,*pats,**opts):repo=opts.get('repo')orthgrepo.repository(ui,paths.find_root())dlg=ManifestDialog(repo,opts.get('rev'))# set initial state after dialog visibledefinit():try:ifpats:path=hglib.canonpaths(pats)[0]elif'canonpath'inopts:path=opts['canonpath']else:returnline=opts.get('line')andint(opts['line'])orNonedlg.setSource(path,opts.get('rev'),line)ifopts.get('pattern'): dlg.setSearchPattern(opts['pattern'])
if dlg._manifest_widget._fileview.actionAnnMode.isEnabled():
dlg._manifest_widget._fileview.actionAnnMode.trigger()
- dlg.setSearchCaseInsensitive(opts['ignorecase'])
+ if 'ignorecase' in opts:+ dlg.setSearchCaseInsensitive(opts['ignorecase'])
except IndexError:
pass
dlg.setSearchPattern(hglib.tounicode(opts.get('pattern')) or '')
QTimer.singleShot(0,init)returndlg
Attach a Trello Card
Add a tag
Your session has expired
You are no longer logged in. Please log in and try your request again.
Filter RSS Feed
This RSS feed URL allows you to see the contents of your current filter using any feed reader.
This link includes a special authentication token. If you share the URL with anyone else, they can see this RSS feed's activity. You can disable these tokens when needed.
Your current filter is unsaved; changing it won't affect this RSS feed.