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 - embeddable changeset summary## Copyright 2009 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.'''embeddable changeset summary'''importreimportosimportgtkimportbinasciifrommercurialimportpatch,util,errorfrommercurial.nodeimporthexfromtortoisehg.util.i18nimport_fromtortoisehg.utilimporthglib,pathsfromtortoisehg.hgtkimportgtklibPANEL_DEFAULT=('rev','summary','user','dateage','branch','tags','transplant')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):self._path=patchpathself._patchname=os.path.basename(patchpath)self._repo=repopf=open(patchpath)try:data=patch.extract(self._repo.ui,pf)tmpfile,msg,user,date,branch,node,p1,p2=dataiftmpfile:os.unlink(tmpfile)finally: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:')}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): if item == 'rev':
revnum = self.get_data('revnum', *args)
revid = self.get_data('revid', *args)
- return (revnum, revid)
+ if revid:+ return (revnum, revid)
+ return None elif item == 'revnum':
return ctx.rev()
elif item == '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=='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':returnhglib.toutf(ctx.branch())elifitem=='branch':value=self.get_data('rawbranch',*args)ifvalue:repo=ctx._repoifctx.node()notinrepo.branchtags().values():returnNonedblist=self.get_config(repo,'tortoisehg','deadbranch')ifdblistandvaluein[hglib.toutf(b.strip()) \
forbindblist.split(',')]:returnNonereturnNoneelifitem=='rawtags':value=[hglib.toutf(tag)fortaginctx.tags()]iflen(value)==0:returnNonereturnvalueelifitem=='tags':value=self.get_data('rawtags',*args)ifvalue:repo=ctx._repohtags=self.get_config(repo,'tortoisehg','hidetags','')htags=[hglib.toutf(b.strip())forbinhtags.split()]value=[tagfortaginvalueiftagnotinhtags]iflen(value)==0:returnNonereturnNoneelifitem=='transplant':extra=ctx.extra()try:ts=extra['transplant_source']ifts:returnbinascii.hexlify(ts)exceptKeyError:passreturnNoneelifitem=='ishead':returnlen(ctx.children())==0raiseUnknownItem(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(face='monospace',size='9000')defdefault_func(widget,item,value):return''defpreset_func(widget,item,value): if item == 'rev':
revnum, revid = value
revid = gtklib.markup(revid, **mono)
- if revnum:
+ if revnum is not None and revid is not None:
return '%s (%s)' % (revnum, revid)
- else:- return '%s' % revid
+ return '%s' % revid
elif item in ('revid', 'transplant'):
return gtklib.markup(value, **mono)
elif item == 'revnum':
returnstr(value)elifitemin('rawbranch','branch'):returngtklib.markup(' %s '%value,color='black',background='#aaffaa')elifitemin('rawtags','tags'):opts=dict(color='black',background='#ffffaa')tags=[gtklib.markup(' %s '%tag,**opts)fortaginvalue]return' '.join(tags)elifitemin('desc','summary','user','date','age'):returngtklib.markup(value)elifitem=='dateage':returngtklib.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=gtk.Label()label.set_markup(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)defget_config(self,repo,section,key,default=None,untrusted=False):cachekey=repo.root+section+keytry:returnself.confcache[cachekey]exceptKeyError:passvalue=repo.ui.config(section,key,default,untrusted)self.confcache[cachekey]=valuereturnvaluedefclear_cache(self):self.cache={}self.confcache={}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):iftargetisNone:target=self.targetiftargetisnotNone:target=str(target)self.target=targetifcustomisnotNone:self.custom=customifrepoisNone:repo=self.repoifrepoisnotNone:self.repo=repoself.ctx=create_context(repo,target)ifself.ctxisNone:returnFalse# cannot updatereturnTrueclassSummaryPanel(SummaryBase,gtk.Frame):def__init__(self,target,style,custom,repo,info):SummaryBase.__init__(self,target,custom,repo,info)gtk.Frame.__init__(self)self.set_shadow_type(gtk.SHADOW_NONE)self.csstyle=styleself.expander=gtk.Expander()self.expander.set_expanded(False)# layout table for contentsself.table=gtklib.LayoutTable(ypad=1,headopts={'weight':'bold'})defupdate(self,target=None,style=None,custom=None,repo=None):ifnotSummaryBase.update(self,target,custom,repo):returnFalse# cannot updateifstyleisnotNone:self.csstyle=styleif'label'inself.csstyle:label=self.csstyle['label']assertisinstance(label,basestring)self.set_label(label)self.set_shadow_type(gtk.SHADOW_ETCHED_IN)if'margin'inself.csstyle:margin=self.csstyle['margin']# 'border' range is 0-65535assertisinstance(margin,(int,long))assert0<=marginandmargin<=65535self.set_border_width(margin)if'padding'inself.csstyle:padding=self.csstyle['padding']# 'border' range is 0-65535assertisinstance(padding,(int,long))assert0<=paddingandpadding<=65535self.table.set_border_width(padding)contents=self.csstyle.get('contents',None)assertcontentssel=self.csstyle.get('selectable',False)assertisinstance(sel,bool)use_expander=self.csstyle.get('expander',False)# build infofirst=Trueself.table.clear_rows()foritemincontents:widgets=self.get_widget(item)ifnotwidgets:continueifisinstance(widgets,gtk.Widget):widgets=(widgets,)ifnotself.get_child():ifuse_expander:self.add(self.expander)self.expander.add(self.table)else:self.add(self.table)forwidgetinwidgets:ifhasattr(widget,'set_selectable'):widget.set_selectable(sel)ifuse_expanderandfirst:self.expander.set_label_widget(widget)first=Falseelse:self.table.add_row(self.get_label(item),widget)self.show_all()returnTrueclassSummaryLabel(SummaryBase,gtk.Label):def__init__(self,target,style,custom,repo,info):SummaryBase.__init__(self,target,custom,repo,info)gtk.Label.__init__(self)self.set_alignment(0,0.5)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']assertisinstance(sel,bool)self.set_selectable(sel)if'width'inself.csstyleor'height'inself.csstyle:width=self.csstyle.get('width',-1)assertisinstance(width,(int,long))height=self.csstyle.get('height',-1)assertisinstance(height,(int,long))self.set_size_request(width,height)self.set_line_wrap(True)contents=self.csstyle.get('contents',None)assertcontents# build infoinfo=''pat=re.compile(r'(?:(?<=%%)|(?<!%)%\()(\w+)(?:\)s)')forsnipincontents:# extract all placeholdersitems=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.set_markup(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.