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.
# csinfo.py - An embeddable widget for changeset summary## Copyright 2010 Yuki KODAMA <endflow.net@gmail.com>## This software may be used and distributed according to the terms of the# GNU General Public License version 2, incorporated herein by reference.importreimportosimport binascii
from PyQt4.QtCore import Qt
-from PyQt4.QtGui import QWidget, QFrame, QPalette, QLabel
+from PyQt4.QtGui import QWidget, QLabel
from mercurial import patch, util, error
from mercurial.node import hex
fromtortoisehg.utilimporthglib,pathsfromtortoisehg.hgqt.i18nimport_fromtortoisehg.hgqtimportqtlibPANEL_DEFAULT=('rev','summary','user','dateage','branch','tags','transplant','p4','svn')defcreate(repo,target=None,style=None,custom=None,**kargs):returnFactory(repo,custom,style,target,**kargs)()deffactory(*args,**kargs):returnFactory(*args,**kargs)defpanelstyle(**kargs):kargs['type']='panel'if'contents'notinkargs:kargs['contents']=PANEL_DEFAULTreturnkargsdeflabelstyle(**kargs):kargs['type']='label'returnkargsdefcustom(**kargs):returnkargsclassFactory(object):def__init__(self,repo,custom=None,style=None,target=None,withupdate=False):ifrepoisNone:raise_('must be specified repository')self.repo=repoself.target=targetifcustomisNone:custom={}self.custom=customifstyleisNone:style=panelstyle()self.csstyle=styleself.info=CachedSummaryInfo()self.withupdate=withupdatedef__call__(self,target=None,style=None,custom=None,repo=None):# try to create a context objectiftargetisNone:target=self.targetifrepoisNone:repo=self.repoifstyleisNone:style=self.csstyleelse:# need to override stylesnewstyle=self.csstyle.copy()newstyle.update(style)style=newstyleifcustomisNone:custom=self.customelse:# need to override customsnewcustom=self.custom.copy()newcustom.update(custom)custom=newcustomif'type'notinstyle:raise_("must be specified 'type' in style")type=style['type']asserttypein('panel','label')# create widgetargs=(target,style,custom,repo,self.info)iftype=='panel':widget=SummaryPanel(*args)else:widget=SummaryLabel(*args)ifself.withupdate:widget.update()returnwidgetclassUnknownItem(Exception):passdefcreate_context(repo,target):ifrepoisNoneortargetisNone:returnNonectx=PatchContext(repo,target)ifctxisNone:ctx=ChangesetContext(repo,target)returnctxdefChangesetContext(repo,rev):ifrepoisNoneorrevisNone:returnNonetry:ctx=repo[rev]except(error.LookupError,error.RepoLookupError,error.RepoError):ctx=NonereturnctxdefPatchContext(repo,patchpath,cache={}):ifrepoisNoneorpatchpathisNone:returnNone# check pathifnotos.path.isabs(patchpath)ornotos.path.isfile(patchpath):returnNone# check cachemtime=os.path.getmtime(patchpath)key=repo.root+patchpathholder=cache.get(key,None)ifholderisnotNoneandmtime==holder[0]:returnholder[1]# create a new context objectctx=patchctx(patchpath,repo)cache[key]=(mtime,ctx)returnctxclasspatchctx(object):def__init__(self,patchpath,repo,patchHandle=None):""" Read patch context from file :param patchHandle: If set, then the patch is a temporary. The provided handle is used to read the patch and the patchpath contains the name of the patch. The handle is NOT closed. """self._path=patchpathself._patchname=os.path.basename(patchpath)self._repo=repoifpatchHandle:pf=patchHandlepf_start_pos=pf.tell()else:pf=open(patchpath)try:data=patch.extract(self._repo.ui,pf)tmpfile,msg,user,date,branch,node,p1,p2=dataiftmpfile:os.unlink(tmpfile)finally:ifpatchHandle:pf.seek(pf_start_pos)else:pf.close()ifnotmsgandhasattr(repo,'mq'):# attempt to get commit messagefromhgextimportmqmsg=mq.patchheader(repo.mq.join(self._patchname)).messageifmsg:msg='\n'.join(msg)self._node=nodeself._user=userandhglib.toutf(user)or''self._date=dateandutil.parsedate(date)orNoneself._desc=msgandmsgor''self._branch=branchandhglib.toutf(branch)or''self._parents=[]forpin(p1,p2):ifnotp:continuetry:self._parents.append(repo[p])except(error.LookupError,error.RepoLookupError,error.RepoError):self._parents.append(p)def__str__(self):node=self.node()ifnode:returnnode[:12]return''def__int__(self):returnself.rev()defnode(self):returnself._nodedefrev(self):returnNonedefhex(self):node=self.node()ifnode:returnhex(node)return''defuser(self):returnself._userdefdate(self):returnself._datedefdescription(self):returnself._descdefbranch(self):returnself._branchdeftags(self):return()defparents(self):returnself._parentsdefchildren(self):return()defextra(self):return{}classSummaryInfo(object):LABELS={'rev':_('Revision:'),'revnum':_('Revision:'),'revid':_('Revision:'),'summary':_('Summary:'),'user':_('User:'),'date':_('Date:'),'age':_('Age:'),'dateage':_('Date:'),'branch':_('Branch:'),'tags':_('Tags:'),'rawbranch':_('Branch:'),'rawtags':_('Tags:'),'transplant':_('Transplant:'),'p4':_('Perforce:'),'svn':_('Subversion:'),'shortuser':_('User:')}def__init__(self):passdefget_data(self,item,widget,ctx,custom,**kargs):args=(widget,ctx,custom)defdefault_func(widget,item,ctx):returnNonedefpreset_func(widget,item,ctx):ifitem=='rev':revnum=self.get_data('revnum',*args)revid=self.get_data('revid',*args)ifrevid:return(revnum,revid)returnNoneelifitem=='revnum':returnctx.rev()elifitem=='revid':returnstr(ctx)elifitem=='desc':returnhglib.toutf(ctx.description().replace('\0',''))elifitem=='summary':value=ctx.description().replace('\0','').split('\n')[0]iflen(value)==0:returnNonereturnhglib.toutf(hglib.tounicode(value)[:80])elifitem=='user':returnhglib.toutf(ctx.user())elifitem=='shortuser':returnhglib.toutf(hglib.username(ctx.user()))elifitem=='dateage':date=self.get_data('date',*args)age=self.get_data('age',*args)ifdateandage:return(date,age)returnNoneelifitem=='date':date=ctx.date()ifdate:returnhglib.displaytime(date)returnNoneelifitem=='age':date=ctx.date()ifdate:returnhglib.age(date)returnNoneelifitem=='rawbranch':value=ctx.branch()ifvalue:returnhglib.toutf(value)returnNoneelifitem=='branch':value=self.get_data('rawbranch',*args)ifvalue:repo=ctx._repoifctx.node()notinrepo.branchtags().values():returnNonedblist=hglib.getdeadbranch(repo.ui)ifvalueindblist:returnNonereturnvaluereturnNoneelifitem=='rawtags':value=[hglib.toutf(tag)fortaginctx.tags()]iflen(value)==0:returnNonereturnvalueelifitem=='tags':value=self.get_data('rawtags',*args)ifvalue:htlist=hglib.gethidetags(ctx._repo.ui)tags=[tagfortaginvalueiftagnotinhtlist]iflen(tags)==0:returnNonereturntagsreturnNoneelifitem=='transplant':extra=ctx.extra()try:ts=extra['transplant_source']ifts:returnbinascii.hexlify(ts)exceptKeyError:passreturnNoneelifitem=='p4':extra=ctx.extra()p4cl=extra.get('p4',None)returnp4cland('changelist %s'%p4cl)elifitem=='svn':extra=ctx.extra()cvt=extra.get('convert_revision','')ifcvt.startswith('svn:'):result=cvt.split('/',1)[-1]ifcvt!=result:returnresultreturncvt.split('@')[-1]else:returnNoneelifitem=='ishead':childbranches=[cctx.branch()forcctxinctx.children()]returnctx.branch()notinchildbranchesraiseUnknownItem(item)ifcustom.has_key('data')andnotkargs.get('usepreset',False):try:returncustom['data'](widget,item,ctx)exceptUnknownItem:passtry:returnpreset_func(widget,item,ctx)exceptUnknownItem:passreturndefault_func(widget,item,ctx)defget_label(self,item,widget,ctx,custom,**kargs):defdefault_func(widget,item):return''defpreset_func(widget,item):try:returnself.LABELS[item]exceptKeyError:raiseUnknownItem(item)ifcustom.has_key('label')andnotkargs.get('usepreset',False):try:returncustom['label'](widget,item)exceptUnknownItem:passtry:returnpreset_func(widget,item)exceptUnknownItem:passreturndefault_func(widget,item)defget_markup(self,item,widget,ctx,custom,**kargs):args=(widget,ctx,custom)mono=dict(family='monospace',size='9pt',space='pre')defdefault_func(widget,item,value):return''defpreset_func(widget,item,value):ifitem=='rev':revnum,revid=valuerevid=qtlib.markup(revid,**mono)ifrevnumisnotNoneandrevidisnotNone:return'%s (%s)'%(revnum,revid)return'%s'%revidelifitemin('revid','transplant'):returnqtlib.markup(value,**mono)elifitemin('revnum','p4','svn'):returnstr(value)elifitemin('rawbranch','branch'):opts=dict(fg='black',bg='#aaffaa')returnqtlib.markup(' %s '%value,**opts)elifitemin('rawtags','tags'):opts=dict(fg='black',bg='#ffffaa')tags=[qtlib.markup(' %s '%tag,**opts)fortaginvalue]return' '.join(tags)elifitemin('desc','summary','user','shortuser','date','age'):returnqtlib.markup(value)elifitem=='dateage':returnqtlib.markup('%s (%s)'%value)raiseUnknownItem(item)value=self.get_data(item,*args)ifvalueisNone:returnNoneifcustom.has_key('markup')andnotkargs.get('usepreset',False):try:returncustom['markup'](widget,item,value)exceptUnknownItem:passtry:returnpreset_func(widget,item,value)exceptUnknownItem:passreturndefault_func(widget,item,value)defget_widget(self,item,widget,ctx,custom,**kargs):args=(widget,ctx,custom)defdefault_func(widget,item,markups):ifisinstance(markups,basestring):markups=(markups,)labels=[]fortextinmarkups:label=QLabel()label.setText(text)labels.append(label)returnlabelsmarkups=self.get_markup(item,*args)ifnotmarkups:returnNoneifcustom.has_key('widget')andnotkargs.get('usepreset',False):try:returncustom['widget'](widget,item,markups)exceptUnknownItem:passreturndefault_func(widget,item,markups)classCachedSummaryInfo(SummaryInfo):def__init__(self):SummaryInfo.__init__(self)self.clear_cache()deftry_cache(self,target,func,*args,**kargs):item,widget,ctx,custom=argsiftarget!='widget':# no cache for widgetroot=ctx._repo.rootrepoid=id(ctx._repo)try:cacheinfo=self.cache[root]ifcacheinfo[0]!=repoid:delself.cache[root]# clear cachecacheinfo=NoneexceptKeyError:cacheinfo=NoneifcacheinfoisNone:self.cache[root]=cacheinfo=(repoid,{})revid=ctx.hex()ifnotrevidandhasattr(ctx,'_path'):revid=ctx._pathkey=target+item+revid+str(custom)try:returncacheinfo[1][key]exceptKeyError:passvalue=func(self,*args,**kargs)iftarget!='widget':# do not cache widgetscacheinfo[1][key]=valuereturnvaluedefget_data(self,*args,**kargs):returnself.try_cache('data',SummaryInfo.get_data,*args,**kargs)defget_label(self,*args,**kargs):returnself.try_cache('label',SummaryInfo.get_label,*args,**kargs)defget_markup(self,*args,**kargs):returnself.try_cache('markup',SummaryInfo.get_markup,*args,**kargs)defget_widget(self,*args,**kargs):returnself.try_cache('widget',SummaryInfo.get_widget,*args,**kargs)defclear_cache(self):self.cache={}classSummaryBase(object):def__init__(self,target,custom,repo,info):iftargetisNone:self.target=Noneelse:self.target=str(target)self.custom=customself.repo=repoself.info=infoself.ctx=create_context(repo,self.target)defget_data(self,item,**kargs):returnself.info.get_data(item,self,self.ctx,self.custom,**kargs)defget_label(self,item,**kargs):returnself.info.get_label(item,self,self.ctx,self.custom,**kargs)defget_markup(self,item,**kargs):returnself.info.get_markup(item,self,self.ctx,self.custom,**kargs)defget_widget(self,item,**kargs):returnself.info.get_widget(item,self,self.ctx,self.custom,**kargs)defupdate(self,target=None,custom=None,repo=None):self.ctx=Noneiftype(target)==patchctx:# If a patchctx is specified as target, use it instead# of creating a context from revision or patch fileself.ctx=targettarget=Noneself.target=NoneiftargetisNone:target=self.targetiftargetisnotNone:target=str(target)self.target=targetifcustomisnotNone:self.custom=customifrepoisNone:repo=self.repoifrepoisnotNone:self.repo=repoifself.ctxisNone:self.ctx=create_context(repo,target)ifself.ctxisNone: return False # cannot update
return True
-# FIXME: SummaryPanel porting is not completed-class SummaryPanel(SummaryBase, QWidget):
+class SummaryPanel(SummaryBase, QLabel):
def __init__(self, target, style, custom, repo, info):
SummaryBase.__init__(self, target, custom, repo, info)
-QWidget.__init__(self)
+QLabel.__init__(self)
-# self.set_shadow_type(gtk.SHADOW_NONE) self.csstyle = style
-# self.expander = gtk.Expander()-# self.expander.set_expanded(False)-- # layout table for contents-# self.table = gtklib.LayoutTable(ypad=1, headopts={'weight': 'bold'})- def update(self, target=None, style=None, custom=None, repo=None):
if not SummaryBase.update(self, target, custom, repo):
return False # cannot update
if style is not None:
self.csstyle = style
- if 'label' in self.csstyle:
- label = self.csstyle['label']
- assert isinstance(label, basestring)
- self.set_label(label)
+ if 'selectable' in self.csstyle:+ sel = self.csstyle['selectable']+ val = sel and Qt.TextSelectableByMouse or Qt.TextBrowserInteraction+ self.setTextInteractionFlags(val)++# if 'label' in self.csstyle:
+# label = self.csstyle['label']
+# assert isinstance(label, basestring)
+# self.set_label(label)
# self.set_shadow_type(gtk.SHADOW_ETCHED_IN)
+# if 'padding' in self.csstyle:+# padding = self.csstyle['padding']+ # 'border' range is 0-65535+# assert isinstance(padding, (int, long))+# assert 0 <= padding and padding <= 65535+# self.table.set_border_width(padding)++ # build info+ contents = self.csstyle.get('contents', ())+ if 'margin' in self.csstyle:
margin = self.csstyle['margin']
- # 'border' range is 0-65535 assert isinstance(margin, (int, long))
-assert0 <= margin and margin <=65535-self.set_border_width(margin)+buf = '<table style="margin:%spx">' % margin
+ else:+buf = '<table>'- if 'padding' in self.csstyle:- padding = self.csstyle['padding']- # 'border' range is 0-65535- assert isinstance(padding, (int, long))- assert 0 <= padding and padding <= 65535- self.table.set_border_width(padding)-- contents = self.csstyle.get('contents', None)- assert contents-- use_expander = self.csstyle.get('expander', False)-- # build info- first = True- self.table.clear_rows() for item in contents:
-widgets = self.get_widget(item)
- if not widgets:
+markups = self.get_markup(item)
+ if not markups:
continue
- if isinstance(widgets, QWidget):
-widgets = (widgets,)- if not self.get_child():- if use_expander:- self.add(self.expander)- self.expander.add(self.table)- else:- self.add(self.table)- for widget in widgets:- if use_expander and first:- self.expander.set_label_widget(widget)- first = False- else:- self.table.add_row(self.get_label(item), widget)- self.show_all()+ label = qtlib.markup(self.get_label(item), weight='bold')+ buf += '<tr><td style="padding-right: 6px;">' + label + '</td><td>'+ if isinstance(markups, basestring):
+markups = (markups,)+ buf += ', '.join(markups) + '</td></tr>'+ buf += '</table>'+ self.setText(buf) return True
LABEL_PAT=re.compile(r'(?:(?<=%%)|(?<!%)%\()(\w+)(?:\)s)')classSummaryLabel(SummaryBase,QLabel):def__init__(self,target,style,custom,repo,info):SummaryBase.__init__(self,target,custom,repo,info)QLabel.__init__(self)self.csstyle=styledefupdate(self,target=None,style=None,custom=None,repo=None):ifnotSummaryBase.update(self,target,custom,repo):returnFalse# cannot updateifstyleisnotNone:self.csstyle=styleif'selectable'inself.csstyle:sel=self.csstyle['selectable']val=selandQt.TextSelectableByMouseorQt.TextBrowserInteractionself.setTextInteractionFlags(val)if'width'inself.csstyle:width=self.csstyle.get('width',0)self.setMinimumWidth(width)if'height'inself.csstyle:height=self.csstyle.get('height',0)self.setMinimumHeight(height)contents=self.csstyle.get('contents',None)# build infoinfo=''forsnipincontents:# extract all placeholdersitems=LABEL_PAT.findall(snip)# fetch required datadata={}foriteminitems:markups=self.get_markup(item)ifnotmarkups:continueifisinstance(markups,basestring):markups=(markups,)data[item]=', '.join(markups)iflen(data)==0:continue# insert data & append to labelinfo+=snip%dataself.setText(info)returnTrue
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.