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.
# commit.py - Commit dialog for TortoiseHg## Copyright 2007 Brad Schick, brad at gmail . com# Copyright 2007 TK Soh <teekaysoh@gmail.com># Copyright 2007 Steve Borho <steve@borho.org>## This software may be used and distributed according to the terms of the# GNU General Public License version 2, incorporated herein by reference.importosimporterrnoimportgtkimportgobjectimportpangoimporttempfileimportcStringIOimporttimefrommercurialimporthg,util,patch,cmdutilfromtortoisehg.util.i18nimport_fromtortoisehg.utilimportshlib,hglibfromtortoisehg.hgtk.statusimportGStatus,FM_STATUS,FM_CHECKEDfromtortoisehg.hgtk.statusimportFM_PATH,FM_PATH_UTF8fromtortoisehg.hgtkimportcsinfo,gtklib,thgconfig,gdialog,hgcmdclassBranchOperationDialog(gtk.Dialog):def__init__(self,branch,close,mergebranches): gtk.Dialog.__init__(self, parent=None, flags=gtk.DIALOG_MODAL,
buttons=(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
gtk.STOCK_OK, gtk.RESPONSE_OK))
+ gtklib.set_tortoise_icon(self, 'branch.ico') gtklib.set_tortoise_keys(self)
self.set_title(_('Branch Operations'))
self.newbranch = None
self.closebranch=Falseifmergebranches:lbl=gtk.Label(_('Select branch of merge commit'))branchcombo=gtk.combo_box_new_text()fornameinmergebranches:branchcombo.append_text(name)branchcombo.set_active(0)self.vbox.pack_start(lbl,True,True,2)self.vbox.pack_start(branchcombo,True,True,2)self.connect('response',self.merge_response,branchcombo)self.show_all()returnself.connect('response',self.response)lbl=gtk.Label(_('Changes take effect on next commit'))nochanges=gtk.RadioButton(None,_('No branch changes'))self.newbranchradio=gtk.RadioButton(nochanges,_('Open a new named branch'))self.closebranchradio=gtk.RadioButton(nochanges,_('Close current named branch'))self.branchentry=gtk.Entry()self.branchentry.connect('activate',self.activated)hbox=gtk.HBox()hbox.pack_start(self.newbranchradio,False,False,2)hbox.pack_start(self.branchentry,True,True,2)self.vbox.pack_start(hbox,True,True,2)hbox=gtk.HBox()hbox.pack_start(self.closebranchradio,True,True,2)self.vbox.pack_start(hbox,True,True,2)hbox=gtk.HBox()hbox.pack_start(nochanges,True,True,2)self.vbox.pack_start(hbox,True,True,2)self.vbox.pack_start(lbl,True,True,10)self.newbranchradio.connect('toggled',self.nbtoggle)self.newbranchradio.set_active(True)ifbranch:self.newbranch=branchself.branchentry.set_text(branch)self.newbranchradio.set_active(True)elifclose:self.closebranch=closeself.closebranchradio.set_active(True)else:nochanges.set_active(True)self.show_all()defnbtoggle(self,radio):self.branchentry.set_sensitive(radio.get_active())ifradio.get_active():self.branchentry.grab_focus()defactivated(self,entry):self.response(self,response_id=gtk.RESPONSE_OK)defresponse(self,widget,response_id):ifresponse_id==gtk.RESPONSE_OK:ifself.newbranchradio.get_active():self.newbranch=self.branchentry.get_text()elifself.closebranchradio.get_active():self.closebranch=Trueelse:self.newbranch=Noneself.closebranch=Falseself.destroy()defmerge_response(self,widget,response_id,combo):self.closebranch=Falseifresponse_id==gtk.RESPONSE_OK:row=combo.get_active()ifrow==1:self.newbranch=combo.get_model()[row][0]self.destroy()classGCommit(GStatus):"""GTK+ based dialog for displaying repository status and committing changes. Also provides related operations like add, delete, remove, revert, refresh, ignore, diff, and edit. """### Overrides of base class methods ###definit(self):GStatus.init(self)self.mode='commit'self.nextbranch=Noneself.closebranch=Falseself.last_commit_id=Noneself.qnew=Falseself.notify_func=Noneself.patch_text=Nonedefget_help_url(self):return'commit.html'defset_notify_func(self,func,args):self.notify_func=funcself.notify_args=argsdefparse_opts(self):GStatus.parse_opts(self)defget_title(self):root=self.get_reponame()ifself.is_merge():root=_('merging ')+rootuser=self.opts.get('user')or''ifuser:user='as '+userdate=self.opts.get('date')or''pats=' '.join(self.pats)or''ifself.qnew:returnroot+_(' - qnew')elifself.mqmode:patch=self.repo.mq.lookup('qtip')returnroot+_(' - qrefresh ')+patchreturnroot+' '.join([_(' - commit'),pats,user,date])defget_icon(self):return'menucommit.ico'defget_default_setting(self):return'ui.username'defauto_check(self):ifnotself.test_opt('check'):returnforrowinself.filemodel:ifrow[FM_STATUS]in'MAR'androw[FM_PATH]notinself.excludes:row[FM_CHECKED]=Trueself.update_check_count()self.opts['check']=Falsedefget_menu_list(self):defrefresh(menu):self.reload_status()deftoggle(button,type):show=button.get_active()statename='show'+typeifgetattr(self,statename)!=show:frame=getattr(self,type+'_frame')ifshow:frame.show()else:frame.hide()setattr(self,statename,show)return[(_('_View'),[dict(text=_('Advanced'),ascheck=True,func=toggle,args=['advanced'],check=self.showadvanced),dict(text=_('Parents'),ascheck=True,func=toggle,args=['parents'],check=self.showparents),dict(text='----'),dict(text=_('Refresh'),func=refresh,icon=gtk.STOCK_REFRESH)]),(_('_Operations'),[dict(text=_('_Commit'),func=self.commit_clicked,icon=gtk.STOCK_OK),dict(name='undo',text=_('_Undo'),func=self.undo_clicked,icon=gtk.STOCK_UNDO),dict(text='----'),dict(name='diff',text=_('_Diff'),func=self.diff_clicked,icon=gtk.STOCK_JUSTIFY_FILL),dict(name='revert',text=_('Re_vert'),func=self.revert_clicked,icon=gtk.STOCK_MEDIA_REWIND),dict(name='add',text=_('_Add'),func=self.add_clicked,icon=gtk.STOCK_ADD),dict(name='remove',text=_('_Remove'),func=self.remove_clicked,icon=gtk.STOCK_DELETE),dict(name='move',text=_('Move'),func=self.move_clicked,icon=gtk.STOCK_JUMP_TO),dict(name='forget',text=_('_Forget'),func=self.forget_clicked,icon=gtk.STOCK_CLEAR)]),]defsave_settings(self):settings=GStatus.save_settings(self)settings['commit-vpane']=self.vpaned.get_position()settings['showparents']=self.showparentssettings['showadvanced']=self.showadvancedreturnsettingsdefload_settings(self,settings):GStatus.load_settings(self,settings)self.setting_vpos=-1self.showparents=Trueself.showadvanced=Falsetry:self.setting_vpos=settings['commit-vpane']self.showparents=settings['showparents']self.showadvanced=settings['showadvanced']exceptKeyError:passdefget_tbbuttons(self):# insert to head of toolbartbbuttons=GStatus.get_tbbuttons(self)tbbuttons.insert(0,gtk.SeparatorToolItem())undo_btn=self.make_toolbutton(gtk.STOCK_UNDO,_('_Undo'),self.undo_clicked,name='undo',tip=_('undo recent commit'))commit_btn=self.make_toolbutton(gtk.STOCK_OK,_('_Commit'),self.commit_clicked,name='commit',tip=_('commit'))tbbuttons.insert(0,undo_btn)tbbuttons.insert(0,commit_btn)returntbbuttonsdefshould_live(self,widget=None,event=None):# If there are more than a few character typed into the commit# message, ask if the exit should continue.live=Falsebuf=self.text.get_buffer()ifbuf.get_char_count()>10andbuf.get_modified():# response: 0=Yes, 1=No, 2=Cancelresponse=gdialog.CustomPrompt(_('Confirm Exit'),_('Save commit message at exit?'),self,(_('&Yes'),_('&No'),_('&Cancel')),2,2).run()ifresponse==0:begin,end=buf.get_bounds()self.update_recent_messages(buf.get_text(begin,end))buf.set_modified(False)elifresponse==2:live=Trueifnotlive:self._destroying(widget)returnlivedefrefresh_complete(self):self.check_merge()self.check_patch_queue()self.check_undo()self.refresh_branchop()self.update_parent_labels()self.update_commit_button()ifnotself.committer_cbbox.get_active_text():user=self.opts['user']oros.environ.get('HGUSER')or \
self.repo.ui.config('ui','username')ifuser:self.update_recent_committers(hglib.toutf(user))ifnotself.autoinc_entry.get_text():autoinc=self.repo.ui.config('tortoisehg','autoinc','')self.autoinc_entry.set_text(hglib.toutf(autoinc))ifself.qnew_name:self.qnew_name.set_sensitive(notself.is_merge())ifself.qnew:self.qnew_name.grab_focus()# set focus backself.qnew_name.set_position(-1)defget_body(self):self.connect('delete-event',self.delete)status_body=GStatus.get_body(self)vbox=gtk.VBox()# Advanced barself.advanced_frame=gtk.VBox()adv_hbox=gtk.HBox()self.advanced_frame.pack_start(adv_hbox,False,False)adv_hbox.pack_start(gtk.Label(_('Committer:')),False,False,2)liststore=gtk.ListStore(str)self.committer_cbbox=gtk.ComboBoxEntry(liststore)cell=gtk.CellRendererText()self.committer_cbbox.pack_start(cell,True)adv_hbox.pack_start(self.committer_cbbox,True,True,2)self._mru_committers=self.settings.mrul('recent_committers')self.update_recent_committers()committer=os.environ.get('HGUSER')orself.repo.ui.config('ui','username')ifcommitter:self.update_recent_committers(hglib.toutf(committer))self.committer_cbbox.set_active(0)adv_hbox.pack_start(gtk.Label(_('Auto-includes:')),False,False,2)self.autoinc_entry=gtk.Entry()adv_hbox.pack_start(self.autoinc_entry,False,False,2)self.autopush=gtk.CheckButton(_('Push after commit'))pushafterci=self.repo.ui.configbool('tortoisehg','pushafterci')self.autopush.set_active(pushafterci)adv_hbox.pack_start(self.autopush,False,False,2)vbox.pack_start(self.advanced_frame,False,False,2)mbox=gtk.HBox()self.connect('thg-accept',self.thgaccept)self.branchbutton=gtk.Button()self.branchbutton.connect('clicked',self.branch_clicked)mbox.pack_start(self.branchbutton,False,False,2)ifself.is_merge():branches=[p.branch()forpinself.repo.parents()]ifbranches[0]==branches[1]:self.branchbutton.set_sensitive(False)ifhasattr(self.repo,'mq'):label=gtk.Label('QNew: ')mbox.pack_start(label,False,False,2)self.qnew_name=gtk.Entry()self.qnew_name.set_sensitive(notself.is_merge())self.qnew_name.set_width_chars(20)self.qnew_name.connect('changed',self.qnew_changed)mbox.pack_start(self.qnew_name,False,False,2)else:self.qnew_name=Noneliststore=gtk.ListStore(str,# summary line (utf-8)str)# full commit messageself.msg_cbbox=gtk.ComboBox(liststore)cell=gtk.CellRendererText()self.msg_cbbox.pack_start(cell,True)self.msg_cbbox.add_attribute(cell,'text',0)liststore.append([_('Recent commit messages...'),''])self.msg_cbbox.set_active(0)self.popupid=self.msg_cbbox.connect('notify::popup-shown',self.first_msg_popdown)self.msg_cbbox.connect('changed',self.changed_cb)mbox.pack_start(self.msg_cbbox)vbox.pack_start(mbox,False,False)self._mru_messages=self.settings.mrul('recent_messages')# change message fieldframe=gtk.Frame()frame.set_shadow_type(gtk.SHADOW_ETCHED_IN)scroller=gtk.ScrolledWindow()scroller.set_policy(gtk.POLICY_AUTOMATIC,gtk.POLICY_AUTOMATIC)frame.add(scroller)vbox.pack_start(frame)accelgroup=gtk.AccelGroup()self.add_accel_group(accelgroup)mod=gtklib.get_thg_modifier()key,modifier=gtk.accelerator_parse('<alt>q')self.add_accelerator('thg-reflow',accelgroup,key,modifier,gtk.ACCEL_VISIBLE)self.text=gtk.TextView()self.text.connect('populate-popup',self.msg_add_to_popup)self.connect('thg-reflow',self.thgreflow,self.text)self.text.modify_font(pango.FontDescription(self.fontcomment))scroller.add(self.text)gtklib.addspellcheck(self.text,self.repo.ui)vbox2=gtk.VBox()vbox2.pack_start(status_body)# parent changeset infoparents_vbox=gtk.VBox(spacing=1)self.parents_frame=parents_vboxstyle=csinfo.labelstyle(contents=(_('Parent: %(rev)s'),' %(athead)s',' %(branch)s',' %(tags)s',' %(summary)s'),selectable=True)defdata_func(widget,item,ctx):ifitem=='athead':returnwidget.get_data('ishead')orself.mqmoderaisecsinfo.UnknownItem(item)defmarkup_func(widget,item,value):ifitem=='athead'andvalueisFalse:text='[%s]'%_('not at head revision')returngtklib.markup(text,weight='bold')raisecsinfo.UnknownItem(item)custom=csinfo.custom(data=data_func,markup=markup_func)factory=csinfo.factory(self.repo,custom,style)defadd_parent():label=factory()parents_vbox.pack_start(label,False,False)returnlabelself.parent1_label=add_parent()self.parent2_label=add_parent()parents_hbox=gtk.HBox()parents_hbox.pack_start(parents_vbox,False,False,5)vbox2.pack_start(parents_hbox,False,False,2)vbox2.pack_start(gtk.HSeparator(),False,False)self.vpaned=gtk.VPaned()self.vpaned.pack1(vbox,shrink=False)self.vpaned.pack2(vbox2,shrink=False)gtklib.idle_add_single_call(self.realize_settings)returnself.vpaneddefget_preview_tab_name(self):ifself.qneworself.mqmode:res=_('Patch Preview')else:res=_('Commit Preview')returnres### End of overridable methods ###defupdate_recent_committers(self,name=None):""" 'name' argument must be in UTF-8. """ifnameisnotNone:self._mru_committers.add(name)self._mru_committers.compact()self.settings.write()liststore=self.committer_cbbox.get_model()liststore.clear()fornameinself._mru_committers:liststore.append([name])defchanged_cb(self,combobox):model=combobox.get_model()index=combobox.get_active()ifindex>=0:buf=self.text.get_buffer()ifbuf.get_char_count()andbuf.get_modified():response=gdialog.Confirm(_('Confirm Discard Message'),[],self,_('Discard current commit message?')).run()ifresponse!=gtk.RESPONSE_YES:combobox.set_active(-1)returnbuf.set_text(model[index][1])buf.set_modified(False)combobox.set_active(-1)deffirst_msg_popdown(self,combo,shown):combo.disconnect(self.popupid)self.popupid=Noneself.update_recent_messages()defupdate_recent_messages(self,msg=None):ifmsg:self._mru_messages.add(msg)self.settings.write()ifself.popupidisnotNone:returnliststore=self.msg_cbbox.get_model()liststore.clear()formsginself._mru_messages:ifnotmsg:continuesumline=hglib.toutf(hglib.tounicode(msg).splitlines()[0])liststore.append([sumline,msg])defbranch_clicked(self,button):ifself.is_merge():mb=[p.branch()forpinself.repo.parents()]else:mb=Nonedialog=BranchOperationDialog(self.nextbranch,self.closebranch,mb)dialog.run()self.nextbranch=Noneself.closebranch=Falseifdialog.newbranch:self.nextbranch=dialog.newbranchelifdialog.closebranch:self.closebranch=Trueself.refresh_branchop()defupdate_commit_button(self):label=_('Commit')tooltip=_('commit')ifself.qnew:label=_('QNew')tooltip=_('create new MQ patch')elifself.mqmode:label=_('QRefresh')tooltip=_('refresh top MQ patch')else:plus,minus=_('_Commit (+1 head)'),_('_Commit (-1 head)')ctxs,isheads,ismerge=self.get_head_info()ifnotismerge:ifnotisheads[0]:label=plustooltip=_('parent is not a head, ''commit to add a new head')else:ifisheads[0]andisheads[1]:label=minustooltip=_('commit to merge one head')elifnotisheads[0]andnotisheads[1]:label=plustooltip=_('no parent is a head, ''commit to add a new head')btn=self.get_toolbutton('commit')btn.set_label(label)btn.set_tooltip(self.tooltips,tooltip)defget_head_info(self):defishead(ctx):returnlen(ctx.children())==0ifself.mqmode:ctxs=self.repo['.'].parents()else:ctxs=self.repo[None].parents()isheads=[ishead(ctx)forctxinctxs]returnctxs,isheads,len(ctxs)==2defupdate_parent_labels(self):ctxs,isheads,ismerge=self.get_head_info()self.parent1_label.update(ctxs[0])ifnotismerge:self.parent2_label.hide()else:self.parent2_label.update(ctxs[1])self.parent2_label.show()defthgreflow(self,window,textview):buffer=textview.get_buffer()pos=buffer.get_property('cursor-position')start=buffer.get_iter_at_offset(0)end=start.copy()end.forward_to_end()text=hglib.tounicode(buffer.get_text(start,end))ifpos==len(text):pos-=1sentence_begin=text.rfind('\n\n',0,pos)whilesentence_begin>=0andtext.rfind('\n\n',0,pos-1)==sentence_begin-1:sentence_begin-=1ifsentence_begin==-1:sentence_begin=0whilesentence_begin<len(text)andtext[sentence_begin]=='\n':sentence_begin+=1sentence_end=text.find('\n\n',sentence_begin+1)ifsentence_end==-1:sentence_end=len(text)pre_sentence=text[:sentence_begin]post_sentence=text[sentence_end:]sentence=text[sentence_begin:sentence_end]parts=sentence.replace('\r','').replace('\n',' ').replace('\t',' ').split(' ')line_width=int(self.repo.ui.config('tortoisehg','messagewrap',80))new_sentence=['']forpartinparts:iflen(new_sentence[-1])+len(part)+1>line_width:new_sentence.append('')new_sentence[-1]+='%s '%partsentence=u'\n'.join([x.strip()forxinnew_sentence]).encode('utf-8')new_pos=len(pre_sentence+sentence)buffer.set_text(pre_sentence+sentence+post_sentence)new_pos_iter=buffer.get_iter_at_offset(new_pos)buffer.place_cursor(new_pos_iter)m=buffer.create_mark('newcurspos',new_pos_iter,False)self.text.scroll_mark_onscreen(m)buffer.delete_mark(m)defrealize_settings(self):self.vpaned.set_position(self.setting_vpos)ifnotself.showparents:self.parents_frame.hide()ifnotself.showadvanced:self.advanced_frame.hide()defthgaccept(self,window):self.commit_clicked(None)defget_custom_menus(self):defcommit(menuitem,files):ifself.ready_message()andself.isuptodate():self.hg_commit(files)self.reload_status()abs=[self.repo.wjoin(file)forfileinfiles]shlib.shell_notify(abs)ifself.is_merge():return()else:return[(_('_Commit'),commit,'MAR'),]defdelete(self,window,event):ifnotself.should_live():self.destroy()else:returnTruedefrefresh_branchop(self):ifself.nextbranch:text=_('new branch: ')+self.nextbranchelifself.closebranch:text=_('close branch: ')+self.repo[None].branch()else:text=_('branch: ')+self.repo[None].branch()self.branchbutton.set_label(text)defcheck_undo(self):can_undo=os.path.exists(self.repo.sjoin("undo"))and \
self.last_commit_idisnotNoneself.cmd_set_sensitive('undo',can_undo)defcheck_merge(self):ifself.is_merge():# select all changes if repo is mergedforentryinself.filemodel:ifentry[FM_STATUS]in'MARD':entry[FM_CHECKED]=Trueself.update_check_count()# pre-fill commit message, if not modifiedbuf=self.text.get_buffer()ifnotbuf.get_modified():buf.set_text(_('Merge '))buf.set_modified(False)defcheck_patch_queue(self):'See if an MQ patch is applied, switch to qrefresh mode'self.qheader=Noneifself.mqmode:qtipctx=self.repo['qtip']self.qheader=qtipctx.description()self.committer_cbbox.child.set_text(hglib.toutf(qtipctx.user()))buf=self.text.get_buffer()ifbuf.get_char_count()==0ornotbuf.get_modified():ifself.qnew:buf.set_text('')else:buf.set_text(self.qheader)buf.set_modified(False)ifself.qnew:self.reload_status()self.qnew_name.grab_focus()self.qnew_name.set_position(-1)ifself.patch_text:self.diff_notebook.remove_page(self.ppage)self.patch_text=Noneelse:ifnotself.patch_text:self.patch_text=gtk.TextView()self.patch_text.set_wrap_mode(gtk.WRAP_NONE)self.patch_text.set_editable(False)self.patch_text.modify_font(self.difffont)scroller=gtk.ScrolledWindow()scroller.set_policy(gtk.POLICY_AUTOMATIC,gtk.POLICY_AUTOMATIC)scroller.add(self.patch_text)self.ppage=self.diff_notebook.append_page(scroller,gtk.Label(_('Patch Contents')))self.diff_notebook.show_all()revs=cmdutil.revrange(self.repo,['tip'])fp=cStringIO.StringIO()opts=patch.diffopts(self.ui,self.opts)patch.export(self.repo,revs,fp=fp,opts=opts)text=fp.getvalue().splitlines(True)buf=self.diff_highlight_buffer(text)self.patch_text.set_buffer(buf)elifself.qnew:buf=self.text.get_buffer()ifnotbuf.get_modified():buf.set_text('')buf.set_modified(False)ifself.patch_text:self.diff_notebook.remove_page(self.ppage)self.patch_text=Noneelifself.patch_text:buf=self.text.get_buffer()ifnotbuf.get_modified():buf.set_text('')buf.set_modified(False)self.diff_notebook.remove_page(self.ppage)self.patch_text=Noneself.branchbutton.set_sensitive(not(self.mqmodeorself.qnew))defcommit_clicked(self,toolbutton,data=None):ifnotself.isuptodate():returndefget_list(addremove=True):commitable='MAR'ifaddremove:ar_list=self.relevant_checked_files('?!')iflen(ar_list)>0andself.should_addremove(ar_list):commitable+='?!'returnself.relevant_checked_files(commitable)ifself.qnew:commit_list=get_list()self.commit_selected(commit_list)else:ifnotself.ready_message():returnifself.is_merge():commit_list=get_list(addremove=False)# merges must be committed without specifying file list.self.hg_commit([])else:commit_list=get_list()iflen(commit_list)>0:self.commit_selected(commit_list)elifself.qheaderisnotNone:self.commit_selected([])elifself.closebranch:self.commit_selected([])else:gdialog.Prompt(_('Nothing Commited'),_('No committable files selected'),self).run()returnself.reload_status()files=[self.repo.wjoin(x)forxincommit_list]shlib.shell_notify(files)defcommit_selected(self,files):# 1a. get list of chunks not rejectedrepo,ui=self.repo,self.repo.ui# 2. backup changed files, so we can restore them in the endbackups={}backupdir=repo.join('record-backups')try:os.mkdir(backupdir)exceptOSError,err:iferr.errno!=errno.EEXIST:gdialog.Prompt(_('Commit'),_('Unable to create ')+backupdir,self).run()returntry:# backup continuesallchunks=[]forfinfiles:cf=util.pconvert(f)ifcfnotinself.status[0]:continueiffnotinself.filechunks:continuechunks=self.filechunks[f]iflen(chunks)<2:continue# unfiltered files do not go through backup-revert-patch cyclerejected=[cforcinchunks[1:]ifnotc.active]iflen(rejected)==0:continueallchunks.extend(chunks)fd,tmpname=tempfile.mkstemp(prefix=cf.replace('/','_')+'.',dir=backupdir)os.close(fd)util.copyfile(repo.wjoin(cf),tmpname)backups[cf]=tmpnamefp=cStringIO.StringIO()forn,cinenumerate(allchunks):ifc.filename()inbackupsandc.active:c.write(fp)dopatch=fp.tell()fp.seek(0)ifbackups:ifself.qheaderisnotNone:# 3a. apply filtered patch to top patch's parenthg.revert(repo,self._node1,backups.has_key)else:# 3a. apply filtered patch to clean repo (clean)hg.revert(repo,repo.dirstate.parents()[0],backups.has_key)# 3b. (apply)ifdopatch:try:pfiles={}patch.internalpatch(fp,ui,1,repo.root,files=pfiles,eolmode=None)patch.updatedir(ui,repo,pfiles)exceptpatch.PatchError,err:s=str(err)ifs:raiseutil.Abort(s)else:gdialog.Prompt(_('Commit'),_('Unable to apply patch'),self).run()returndelfp# 4. We prepared working directory according to filtered patch.# Now is the time to delegate the job to commit/qrefresh# or the like!# it is important to first chdir to repo root -- we'll call a# highlevel command with list of pathnames relative to repo rootcwd=os.getcwd()os.chdir(repo.root)try:self.hg_commit(files)finally:os.chdir(cwd)returnfinally:# 5. finally restore backed-up filestry:forrealname,tmpnameinbackups.iteritems():util.copyfile(tmpname,repo.wjoin(realname))os.unlink(tmpname)os.rmdir(backupdir)exceptOSError:passdefundo_clicked(self,toolbutton,data=None):response=gdialog.Confirm(_('Confirm Undo Commit'),[],self,_('Undo last commit')).run()ifresponse!=gtk.RESPONSE_YES:returntip=self.get_tip_rev(True)ifnottip==self.last_commit_id:gdialog.Prompt(_('Undo Commit'),_('Unable to undo!\n\n''Tip revision differs from last commit.'),self).run()returntry:self.repo.ui.quiet=Trueself.repo.rollback()self.repo.ui.quiet=Falseself.last_commit_id=Noneself.reload_status()time.sleep(0.5)# give fs some time to pick up changesshlib.shell_notify([os.getcwd()])except:gdialog.Prompt(_('Undo Commit'),_('Errors during rollback!'),self).run()defchangelog_clicked(self,toolbutton,data=None):fromtortoisehg.hgtkimporthistorydlg=history.run(self.ui)dlg.display()returnTruedefshould_addremove(self,files):ifself.test_opt('addremove'):returnTrueresponse=gdialog.Confirm(_('Confirm Add/Remove'),files,self,_('Add/Remove the following files?')).run()ifresponse!=gtk.RESPONSE_YES:returnFalse# This will stay set for further commits (meaning no more# prompts). Problem?self.opts['addremove']=Trueifself.qneworself.qheaderisnotNone:cmdline=['hg','addremove','--verbose']cmdline+=[self.repo.wjoin(x)forxinfiles]dialog=hgcmd.CmdDialog(cmdline,True)dialog.set_transient_for(self)dialog.run()dialog.hide()returnTruedefready_message(self):buf=self.text.get_buffer()ifbuf.get_char_count()==0:gdialog.Prompt(_('Nothing Commited'),_('Please enter commit message'),self).run()self.text.grab_focus()returnFalsetry:sumlen=int(self.repo.ui.config('tortoisehg','summarylen',0))maxlen=int(self.repo.ui.config('tortoisehg','messagewrap',0))except(TypeError,ValueError):gdialog.Prompt(_('Error'),_('Message format configuration error'),self).run()self.msg_config(None)returnFalselines=hglib.tounicode(buf.get_text(buf.get_start_iter(),buf.get_end_iter())).splitlines()ifsumlenandlen(lines[0].rstrip())>sumlen:resp=gdialog.Confirm(_('Confirm Commit'),[],self,_('The summary line length of %i is greater than'' %i.\n\nIgnore format policy and continue'' commit?')%(len(lines[0].rstrip()),sumlen)).run()ifresp!=gtk.RESPONSE_YES:returnFalseifsumlenandlen(lines)>1andlen(lines[1].strip()):resp=gdialog.Confirm(_('Confirm Commit'),[],self,_('The summary line is not followed by a blank'' line.\n\nIgnore format policy and continue'' commit?')).run()ifresp!=gtk.RESPONSE_YES:returnFalseifmaxlen:start=int(sumlen>0)tmp=[len(x.rstrip())>maxlenforxinlines[start:]]errs=[str(x[1]+start+1)forxinzip(tmp,range(len(tmp)))ifx[0]]iferrs:resp=gdialog.Confirm(_('Confirm Commit'),[],self,_('The following lines are over the %i-''character limit: %s.\n\nIgnore format'' policy and continue commit?')%(maxlen,', '.join(errs))).run()ifresp!=gtk.RESPONSE_YES:returnFalsereturnTruedefhg_commit(self,files):# get advanced optionsuser=hglib.fromutf(self.committer_cbbox.get_active_text())ifnotuser:gdialog.Prompt(_('Commit: Invalid username'),_('Your username has not been configured.\n\n''Please configure your username and try again'),self).run()# bring up the config dialog for user to enter their username.# But since we can't be sure they will do it right, we will# have them to retry, to re-trigger the checking mechanism.dlg=thgconfig.ConfigDialog(False)dlg.show_all()dlg.focus_field('ui.username')dlg.run()dlg.hide()self.refreshui()self.refresh_complete()returnself.update_recent_committers(hglib.toutf(user))incs=hglib.fromutf(self.autoinc_entry.get_text())self.opts['include']=[i.strip()foriinincs.split(',')ifi.strip()]autopush=self.autopush.get_active()buf=self.text.get_buffer()begin,end=buf.get_bounds()self.opts['message']=buf.get_text(begin,end)cmdline=['hg','commit','--verbose']ifself.nextbranch:# response: 0=Yes, 1=No, 2=Cancelifself.nextbranchinself.repo.branchtags():ifself.nextbranchin[p.branch()forpinself.repo.parents()]:response=0else:response=gdialog.CustomPrompt(_('Confirm Override Branch'),_('A branch named "%s" already exists,\n''override?')%self.nextbranch,self,(_('&Yes'),_('&No'),_('&Cancel')),2,2).run()else:response=gdialog.CustomPrompt(_('Confirm New Branch'),_('Create new named branch "%s"?')%self.nextbranch,self,(_('&Yes'),_('&No'),_('&Cancel')),2,2).run()ifresponse==0:self.repo.dirstate.setbranch(self.nextbranch)elifresponse==2:returnelifself.closebranch:cmdline.append('--close-branch')# call the threaded CmdDialog to do the commit, so the the large commit# won't get locked up by potential large commit. CmdDialog will also# display the progress of the commit operation.ifself.qnew:cmdline[1]='qnew'cmdline.append('--force')elifself.qheaderisnotNone:cmdline[1]='qrefresh'ifnotfiles:cmdline+=['-X',self.repo.root]elifself.opts['addremove']:cmdline+=['--addremove']ifself.opts['user']oruser:cmdline.extend(['--user',self.opts['user']oruser])ifself.opts['date']:cmdline.extend(['--date',self.opts['date']])files+=self.opts['include']cmdline+=['--message',hglib.fromutf(self.opts['message'])]ifself.qnew:cmdline+=[hglib.fromutf(self.get_qnew_name())]cmdline+=filesifautopush:cmdline=(cmdline,['hg','push'])dialog=hgcmd.CmdDialog(cmdline,True)dialog.set_transient_for(self)dialog.run()dialog.hide()# refresh overlay icons and commit dialogifdialog.return_code()==0:self.closebranch=Falseself.nextbranch=Noneself.filechunks={}# force re-read of chunksbuf=self.text.get_buffer()ifbuf.get_modified():self.update_recent_messages(self.opts['message'])buf.set_modified(False)ifself.qnew:self.qnew_name.set_text('')hglib.invalidaterepo(self.repo)self.mode='commit'self.qnew=Falseelifself.qheaderisNone:self.text.set_buffer(gtk.TextBuffer())self.msg_cbbox.set_active(-1)self.last_commit_id=self.get_tip_rev(True)ifself.notify_func:self.notify_func(self.notify_args)defget_tip_rev(self,refresh=False):ifrefresh:self.repo.invalidate()returnself.repo['tip'].node()defget_qnew_name(self):returnself.qnew_nameandself.qnew_name.get_text().strip()or''defqnew_changed(self,element):qnew=bool(self.get_qnew_name())ifself.qnew!=qnew:self.qnew=qnewself.mode=qnewand'status'or'commit'self.reload_status()defmsg_add_to_popup(self,textview,menu):menu_items=(('----',None),(_('Paste _Filenames'),self.msg_paste_fnames),(_('App_ly Format'),self.msg_word_wrap),(_('C_onfigure Format'),self.msg_config))forlabel,handlerinmenu_items:iflabel=='----':menuitem=gtk.SeparatorMenuItem()else:menuitem=gtk.MenuItem(label)ifhandler:menuitem.connect('activate',handler)menu.append(menuitem)menu.show_all()defmsg_paste_fnames(self,sender):buf=self.text.get_buffer()fnames=[file[FM_PATH_UTF8]forfileinself.filemodeliffile[FM_CHECKED]]buf.delete_selection(True,True)buf.insert_at_cursor('\n'.join(fnames))defmsg_word_wrap(self,sender):try:sumlen=int(self.repo.ui.config('tortoisehg','summarylen',0))maxlen=int(self.repo.ui.config('tortoisehg','messagewrap',0))except(TypeError,ValueError):sumlen=0maxlen=0ifnot(sumlenormaxlen):gdialog.Prompt(_('Info Required'),_('Message format needs to be configured'),self).run()self.msg_config(None)returnbuf=self.text.get_buffer()lines=buf.get_text(buf.get_start_iter(),buf.get_end_iter()).splitlines()ifnotlines:returnifsumlenandlen(lines[0].rstrip())>sumlen:gdialog.Prompt(_('Warning'),_('The summary line length of %i is greater than %i')%(len(lines[0].rstrip()),sumlen),self).run()ifsumlenandlen(lines)>1andlen(lines[1].strip()):gdialog.Prompt(_('Warning'),_('The summary line is not followed by a blank line'),self).run()ifnotmaxlen:returnlnum=int(sumlen>0)whilelnum<len(lines):lines[lnum]=lines[lnum].rstrip()+' 'iflines[lnum].endswith('. '):lines[lnum]+=' 'iflen(lines[lnum].rstrip())>maxlen:ind=lines[lnum].rfind(' ',0,maxlen+1)+1ifind>0:iflnum==len(lines)-1ornotlines[lnum+1].strip():lines.insert(lnum+1,lines[lnum][ind:].lstrip())else:lines[lnum+1]=lines[lnum][ind:].lstrip() \
+lines[lnum+1]lines[lnum]=lines[lnum][0:ind].rstrip()lnum+=1buf.set_text('\n'.join(lines))defmsg_config(self,sender):dlg=thgconfig.ConfigDialog(True)dlg.show_all()dlg.focus_field('tortoisehg.summarylen')dlg.run()dlg.hide()self.refreshui()returndefrun(_ui,*pats,**opts):cmdoptions={'user':opts.get('user',''),'date':opts.get('date',''),'logfile':'','message':'','modified':True,'added':True,'removed':True,'deleted':True,'unknown':True,'ignored':False,'exclude':[],'include':[],'rev':[],'check':True,'git':False,'addremove':False,}returnGCommit(_ui,None,None,pats,cmdoptions)
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.