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.
# gdialog.py - base dialog for gtools## Copyright 2007 Brad Schick, brad at gmail . com# Copyright 2008 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.importosimportthreadingimportcStringIOimportsysimportshutilimporttempfileimportgtkimportatexitfrommercurialimportcmdutil,util,ui,hg,commandsfromtortoisehg.util.i18nimport_fromtortoisehg.utilimportsettings,hglib,paths,shlibfromtortoisehg.hgtkimportgtklibclassSimpleMessage(gtklib.MessageDialog):defrun(self):response=gtklib.MessageDialog.run(self)self.destroy()returnresponseclassPrompt(SimpleMessage):def__init__(self,title,message,parent,type=gtk.MESSAGE_INFO):SimpleMessage.__init__(self,parent,gtk.DIALOG_MODAL,type,gtk.BUTTONS_CLOSE)self.set_title('TortoiseHg')self.set_markup(gtklib.markup(hglib.toutf(title),weight='bold'))self.format_secondary_text(hglib.toutf(message))mod=gtklib.get_thg_modifier()key,modifier=gtk.accelerator_parse(mod+'Return')accel_group=gtk.AccelGroup()self.add_accel_group(accel_group)try:buttons=self.get_children()[0].get_children()[1].get_children()buttons[0].add_accelerator('clicked',accel_group,key,modifier,gtk.ACCEL_VISIBLE)exceptIndexError:passclassCustomPrompt(gtk.MessageDialog):''' Custom prompt dialog. Provide a list of choices with ampersands to delineate response given for each choice (and keyboard accelerator). Default must be the index of one of the choice responses. '''# ret = CustomPrompt('Title', 'Message', self, ('&Yes', 'N&o'), 1).run()# ret will be (gtk.RESPONSE_DELETE_EVENT, 0 (for yes), or 1 (for no)def__init__(self,title,message,parent,choices,default=None,esc=None):gtk.MessageDialog.__init__(self,parent,gtk.DIALOG_MODAL,gtk.MESSAGE_QUESTION)self.set_title(hglib.toutf(title))self.format_secondary_markup(gtklib.markup(hglib.toutf(message),weight='bold'))accel_group=gtk.AccelGroup()self.add_accel_group(accel_group)fori,sinenumerate(choices):button=self.add_button(s.replace('&','_'),i)try:char=s[s.index('&')+1].lower()button.add_accelerator('clicked',accel_group,ord(char),0,gtk.ACCEL_VISIBLE)exceptValueError:passifdefault:self.set_default_response(default)self.esc=escdefrun(self):response=gtklib.MessageDialog.run(self)ifresponse==gtk.RESPONSE_DELETE_EVENTandself.esc!=None:response=self.escself.destroy()returnresponseclassConfirm(SimpleMessage):"""Dialog returns gtk.RESPONSE_YES or gtk.RESPONSE_NO """def__init__(self,title,files,parent,primary):SimpleMessage.__init__(self,parent,gtk.DIALOG_MODAL,gtk.MESSAGE_QUESTION,gtk.BUTTONS_YES_NO)self.set_title(hglib.toutf(title))self.set_markup(gtklib.markup(hglib.toutf(primary),weight='bold'))message=''fori,finenumerate(files):message+=' '+f+'\n'ifi==9:message+=' ...\n'breakself.format_secondary_text(hglib.toutf(message))accel_group=gtk.AccelGroup()self.add_accel_group(accel_group)buttons=self.get_children()[0].get_children()[1].get_children()buttons[1].add_accelerator("clicked",accel_group,ord("y"),0,gtk.ACCEL_VISIBLE)buttons[0].add_accelerator("clicked",accel_group,ord("n"),0,gtk.ACCEL_VISIBLE)classGDialog(gtk.Window):"""GTK+ based dialog for displaying mercurial information The following methods are meant to be overridden by subclasses. At this point GCommit is really the only intended subclass. parse_opts(self) get_title(self) get_minsize(self) get_defsize(self) get_tbbuttons(self) get_menu_list(self) get_help_url(self) get_default_setting(self) get_body(self) get_extras(self) prepare_display(self) should_live(self, widget, event) save_settings(self) load_settings(self, settings) """# "Constants"def__init__(self,ui,repo,cwd,pats,opts):gtk.Window.__init__(self,gtk.WINDOW_TOPLEVEL)self.cwd=cwdoros.getcwd()self.ui=uiself.ui.setconfig('ui','interactive','off')ifrepo:self.repo=repoelse:root=paths.find_root()ifroot:self.repo=hg.repository(ui,path=root)else:self.repo=Noneself.pats=patsself.opts=optsself.tmproot=Noneself.toolbuttons={}self.menuitems={}self.settings=settings.Settings(self.__class__.__name__)self.init()defrefreshui(self):self.ui=ui.ui()self.ui.setconfig('ui','interactive','off')ifself.repo:self.repo=hg.repository(self.ui,path=self.repo.root)### Following methods are meant to be overridden by subclasses ###definit(self):passdefparse_opts(self):passdefget_title(self):return''defget_icon(self):return''defget_minsize(self):return(395,200)defget_defsize(self):returnself._setting_defsizedefget_tbbuttons(self):return[]defget_menu_list(self):returnNonedefget_help_url(self):returnNonedefget_default_setting(self):returnNonedefget_body(self):returnNonedefget_extras(self):returnNonedefprepare_display(self):passdefshould_live(self,widget=None,event=None):self._destroying(widget)returnFalsedefsave_settings(self):settings={}rect=self.get_allocation()ifself.ismaximizedorself.isiconified:settings['gdialog-rect']=self._setting_defsizesettings['gdialog-pos']=self._setting_winposelse:settings['gdialog-rect']=(rect.width,rect.height)settings['gdialog-pos']=self.lastpossettings['gdialog-ismax']=self.ismaximizedreturnsettingsdefload_settings(self,settings):self._setting_defsize=(678,585)self._setting_winpos=(0,0)self._setting_wasmax=Falsetry:self._setting_defsize=settings['gdialog-rect']self._setting_winpos=settings['gdialog-pos']self._setting_wasmax=settings['gdialog-ismax']exceptKeyError:passdefshow_toolbar_on_start(self):returnTrue### End of overridable methods ###defdisplay(self,opengui=True):self._parse_config()self._load_settings()ifopengui:self._setup_gtk()self._parse_opts()self.prepare_display()self.show_all()else:self._parse_opts()self.tooltips=gtk.Tooltips()deftest_opt(self,opt):returnself.opts.get(opt,False)def_parse_config(self):# defaultsself.fontcomment='monospace 10'self.fontdiff='monospace 10'self.fontlist='Sans 9'self.diffbottom=''forattr,settinginself.ui.configitems('gtools'):ifsetting:setattr(self,attr,setting)ifnotself.diffbottom:self.diffbottom=Falseelifself.diffbottom.lower()=='false'orself.diffbottom=='0':self.diffbottom=Falseelse:self.diffbottom=Truedef_parse_opts(self):# Remove dry_run since Hg only honors it for certain commandsself.opts['dry_run']=Falseself.opts['force_editor']=Falseself.parse_opts()defmerge_opts(self,defaults,mergelist=()):"""Merge default options with the specified local options and globals. Results is defaults + merglist + globals """newopts={}forhgoptindefaults:newopts[hgopt[1].replace('-','_')]=hgopt[2]formergeoptinmergelist:newopts[mergeopt]=self.opts[mergeopt]newopts.update(self.global_opts())returnnewoptsdefglobal_opts(self):globalopts={}hgglobals=[opt[1]foroptincommands.globaloptsifopt[1]!='help']hgglobals=[f.replace('-','_')forfinhgglobals]forkeyinself.opts:ifkeyinhgglobals:globalopts[key]=self.opts[key]returnglobaloptsdefcount_revs(self):cnt=0ifself.test_opt('rev'):forrevinself.opts['rev']:cnt+=len(rev.split(cmdutil.revrangesep,1))returncntdefmake_toolbutton(self,stock,label,handler,userdata=None,menu=None,tip=None,toggle=False,icon=None,name=None):ifmenu:tbutton=gtk.MenuToolButton(stock)tbutton.set_menu(menu)eliftoggle:tbutton=gtk.ToggleToolButton(stock)else:tbutton=gtk.ToolButton(stock) if tip:
tbutton.set_tooltip(self.tooltips, tip)
if icon:
- path = paths.get_tortoise_icon(icon)- if path:- image = gtk.Image()
- image.set_from_file(path)- tbutton.set_icon_widget(image)
+ image = self.icon_from_name(icon)
+ tbutton.set_icon_widget(image)
tbutton.set_use_underline(True)
tbutton.set_label(label)
tbutton.connect('clicked', handler, userdata)
ifname:self.toolbuttons[name]=tbuttonreturntbuttondefget_toolbutton(self,name):returnself.toolbuttons[name]defget_menuitem(self,name):returnself.menuitems[name]defget_widgets(self,name):widgets=[]widgets.append(self.toolbuttons.get(name))widgets.append(self.menuitems.get(name))returnwidgetsdefcmd_set_sensitive(self,name,sensitive):forwinself.get_widgets(name):ifw:w.set_sensitive(sensitive)defcmd_set_active(self,name,active):forwinself.get_widgets(name):ifwandhasattr(w,'set_active'):w.set_active(active)defcmd_get_active(self,name,fallback=None):prev=Noneforwinself.get_widgets(name):ifwandhasattr(w,'set_active'):active=w.get_active()ifprevisnotNoneandprev!=active:returnfallbackprev=activereturnprevdefcmd_handler_block_by_func(self,name,func):forwinself.get_widgets(name):ifw:w.handler_block_by_func(func)defcmd_handler_unblock_by_func(self,name,func):forwinself.get_widgets(name):ifw:w.handler_unblock_by_func(func)defget_reponame(self):returnhglib.get_reponame(self.repo)defhelpcontents(self,item):'User selected Help->Contents from menu bar'url=self.get_help_url()ifnoturl:returnifnoturl.startswith('http'):url='http://tortoisehg.org/manual/0.9/'+urlshlib.browse_url(url)deflaunch(self,item,app):importsys# Spawn background process and exitifhasattr(sys,"frozen"):args=[sys.argv[0],app]else:args=[sys.executable]+[sys.argv[0],app]ifapp.endswith('config')andself.get_default_setting():args+=['--focus',self.get_default_setting()]ifos.name=='nt':args=['"%s"'%argforarginargs]oldcwd=os.getcwd()root=paths.find_root(oldcwd)try:os.chdir(root)os.spawnv(os.P_NOWAIT,sys.executable,args)finally:os.chdir(oldcwd)defwindowstate(self,window,event):ifevent.changed_mask>k.gdk.WINDOW_STATE_MAXIMIZED:ifevent.new_window_state>k.gdk.WINDOW_STATE_MAXIMIZED:self.ismaximized=Trueelse:self.ismaximized=Falseifevent.changed_mask>k.gdk.WINDOW_STATE_ICONIFIED:ifevent.new_window_state>k.gdk.WINDOW_STATE_ICONIFIED:self.isiconified=Trueelse:self.isiconified=False def setfocus(self, window, event):
self.lastpos = self.get_position()
+ def icon_from_name(self, icon):+ if icon.startswith('gtk'):+ img = gtk.image_new_from_stock(icon, gtk.ICON_SIZE_MENU)+ else:+ img = gtk.Image()+ ico = paths.get_tortoise_icon(icon)+ if not ico:+ return img+ try:+ width, height = gtk.icon_size_lookup(gtk.ICON_SIZE_MENU)+ buf = gtk.gdk.pixbuf_new_from_file_at_size(ico, width, height)+ img.set_from_pixbuf(buf)+ except: # don't let broken gtk+ to break dialogs+ pass+ return img+ def _setup_gtk(self):
self.set_title(self.get_title())
gtklib.set_tortoise_icon(self, self.get_icon())
gtklib.set_tortoise_keys(self)self.ismaximized=Falseself.isiconified=Falseself.lastpos=self._setting_winposself.connect('window-state-event',self.windowstate)self.connect('set-focus',self.setfocus)# Minimum sizeminx,miny=self.get_minsize()self.set_size_request(minx,miny)# Initial sizedefx,defy=self.get_defsize()self.set_default_size(defx,defy)ifself._setting_wasmax:self.maximize()# Restore position if it is still on screenscreen=self.get_screen()w,h=screen.get_width(),screen.get_height()x,y=self._setting_winposifx>=0andx<wandy>=0andy<h:self.move(x,y)self.tooltips=gtk.Tooltips()toolbar=gtk.Toolbar()tbuttons=self.get_tbbuttons()fortbuttonintbuttons:toolbar.insert(tbutton,-1)self.toolbar=toolbar# Subclass returns the main bodybody=self.get_body()# Subclass provides extra stuff in bottom hboxextras=self.get_extras()menus=self.get_menu_list()ifmenus:allmenus=[(_('_Tools'),[dict(text=_('Repository Explorer'),func=self.launch,args=['log'],icon='menulog.ico'),dict(text=_('Commit'),func=self.launch,args=['commit'],icon='menucommit.ico'),dict(text=_('Datamine'),func=self.launch,args=['datamine'],icon='menurepobrowse.ico'),dict(text=_('Recovery'),func=self.launch,args=['recover'],icon='general.ico'),dict(text=_('Serve'),func=self.launch,args=['serve'],icon='proxy.ico'),dict(text=_('Shelve'),func=self.launch,args=['shelve'],icon='shelve.ico'),dict(text=_('Synchronize'),func=self.launch,args=['synch'],icon='menusynch.ico'),dict(text=_('Settings'),func=self.launch,args=['repoconfig'],icon='settings_repo.ico')])]+menus+[(_('_Help'),[dict(text=_('Contents'),func=self.helpcontents,icon=gtk.STOCK_INFO),dict(text=_('About'),func=self.launch,args=['about'],icon=gtk.STOCK_ABOUT)])]menubar=gtk.MenuBar()fortitle,itemsinallmenus:m_items=gtklib.MenuItems()fordinitems:text=d['text']name=d.get('name')func=d.get('func')ascheck=d.get('ascheck',False)args=d.get('args',[])icon=d.get('icon')check=d.get('check',False)sensitive=d.get('sensitive',None)iftext=='----':item=gtk.SeparatorMenuItem()else:ifascheck:item=gtk.CheckMenuItem(text) item.set_active(check)
elif icon:
item = gtk.ImageMenuItem(text)
- if icon.startswith('gtk'):- img = gtk.image_new_from_stock(- icon, gtk.ICON_SIZE_MENU)- else:- img = gtk.Image()- ico = paths.get_tortoise_icon(icon)
- if ico:- try:- width, height = gtk.icon_size_lookup(- gtk.ICON_SIZE_MENU)- pixbuf = gtk.gdk.pixbuf_new_from_file_at_size(- ico, width, height)- img.set_from_pixbuf(pixbuf)- except:- # don't let broken gtk+ to break dialogs- pass+ img = self.icon_from_name(icon)
item.set_image(img)
else:
item = gtk.MenuItem(text)
ifsensitiveisnotNone:item.set_sensitive(sensitive)item.connect('activate',func,*args)ifname:self.menuitems[name]=itemm_items.append(item)item=gtk.MenuItem(title)item.set_submenu(m_items.create_menu())menubar.append(item)vbox=gtk.VBox(False,0)self.add(vbox)ifmenus:vbox.pack_start(menubar,False,False,0)self.toolbar_box=gtk.VBox()vbox.pack_start(self.toolbar_box,False,False,0)ifself.show_toolbar_on_start():self._show_toolbar(True)vbox.pack_start(body,True,True,0)ifextras:vbox.pack_end(extras,False,False,0)self.connect('destroy',self._destroying)def_show_toolbar(self,show):ifself.toolbarinself.toolbar_box.get_children():self.toolbar.set_property('visible',show)elifshow:self.toolbar_box.pack_start(self.toolbar,False,False,0)self.toolbar.show_all()def_destroying(self,gtkobj):settings=self.save_settings()self.settings.set_value('dialogs',settings)self.settings.write()def_load_settings(self):settings=self.settings.get_value('dialogs',{})self.load_settings(settings)def_hg_call_wrapper(self,title,command,showoutput=True):"""Run the specified command and display any resulting aborts, messages, and errors """textout=''saved=sys.stderrerrors=cStringIO.StringIO()try:sys.stderr=errorsself.ui.pushbuffer()try:command()except(util.Abort,IOError,OSError),inst:Prompt(title+_(' Aborted'),str(inst),self).run()returnFalse,''finally:sys.stderr=savedtextout=self.ui.popbuffer()prompttext=''ifshowoutput:prompttext=textout+'\n'prompttext+=errors.getvalue()errors.close()iflen(prompttext)>1:Prompt(title+_(' Messages and Errors'),prompttext,self).run()returnTrue,textoutdef_do_diff(self,canonpats,options):fromtortoisehg.hgtkimportvisdiffoptions['canonpats']=canonpatsdialog=visdiff.run(self.ui,**options)ifnotdialog:returndialog.show_all()dialog.run()dialog.hide()def_diff_file(self,stat,file):self._do_diff(fileand[file]or[],self.opts)def_view_files(self,files,otherparent):fromtortoisehg.hgtkimportthgconfigdefcleanup():shutil.rmtree(self.tmproot)ifnotself.tmproot:self.tmproot=tempfile.mkdtemp(prefix='gtools.')atexit.register(cleanup)defsnapshot_node(ui,repo,files,node,tmproot):''' snapshot files as of some revision (adapted from Extdiff extension) '''ctx=repo[node]mf=ctx.manifest()dirname=os.path.basename(repo.root)ifdirname=="":dirname="root"dirname='%s.%s'%(dirname,str(ctx))base=os.path.join(tmproot,dirname)try:os.mkdir(base)except:passui.note(_('making snapshot of %d files from rev %s\n')%(len(files),str(ctx)))forfninfiles:ifnotfninmf:# skipping new file after a merge ?continuewfn=util.pconvert(fn)ui.note(' %s\n'%wfn)dest=os.path.join(base,wfn)destdir=os.path.dirname(dest)ifnotos.path.isdir(destdir):os.makedirs(destdir)data=repo.wwritedata(wfn,repo.file(wfn).read(mf[wfn]))open(dest,'wb').write(data)returndirnamedefdoedit():pathroot=self.repo.rootcopynode=None# if we aren't looking at the wc, copy the node...ifotherparent:copynode=self._node1elifself._node2:copynode=self._node2ifcopynode:pf=[util.pconvert(f)forfinfiles]copydir=snapshot_node(self.ui,self.repo,pf,copynode,self.tmproot)pathroot=os.path.join(self.tmproot,copydir)paths=['"'+os.path.join(pathroot,f)+'"'forfinfiles]command=editor+' '+' '.join(paths)util.system(command,environ={'HGUSER':self.ui.username()},onerr=self.ui,errprefix=_('edit failed'))editor=(self.ui.config('tortoisehg','editor')orself.ui.config('gtools','editor')oros.environ.get('HGEDITOR')orself.ui.config('ui','editor')oros.environ.get('EDITOR','vi'))ifos.path.basename(editor)in('vi','vim','hgeditor'):Prompt(_('No visual editor configured'),_('Please configure a visual editor.'),self).run()dlg=thgconfig.ConfigDialog(False)dlg.show_all()dlg.focus_field('tortoisehg.editor')dlg.run()dlg.hide()self.ui=ui.ui()self._parse_config()returnlfile=util.localpath(files[0])thread=threading.Thread(target=doedit,name='edit:'+lfile)thread.setDaemon(True)thread.start()
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.