class StringSearchFilter(SearchFilter): """ Contains: - a label - an entry :ivar entry: the entry :ivar label: the label """ def __init__(self, label, chars=0, container=None): """ Create a new StringSearchFilter object. :param label: label of the search filter :param chars: maximum number of chars used by the search entry """ self._container = container SearchFilter.__init__(self, label=label) self.title_label = gtk.Label(label) self.pack_start(self.title_label, False, False) self.title_label.show() self._options = {} self.mode = ProxyComboBox() self.mode.connect('content-changed', self._on_mode__content_changed) self.pack_start(self.mode, False, False, 6) self.entry = HintedEntry() self.entry.set_hint(_("Search")) self.entry.show_hint() self.entry.props.secondary_icon_sensitive = False self.entry.set_icon_from_stock(gtk.ENTRY_ICON_PRIMARY, gtk.STOCK_FIND) self.entry.set_icon_from_stock(gtk.ENTRY_ICON_SECONDARY, gtk.STOCK_CLEAR) self.entry.set_icon_tooltip_text(gtk.ENTRY_ICON_SECONDARY, _("Clear the search")) self.entry.connect("icon-release", self._on_entry__icon_release) self.entry.connect('activate', self._on_entry__activate) self.entry.connect('changed', self._on_entry__changed) if chars: self.entry.set_width_chars(chars) self.pack_start(self.entry, False, False, 6) self.entry.show() for option in (ContainsAll, ContainsExactly, DoesNotContain): self._add_option(option) self.mode.select_item_by_position(0) def _add_option(self, option_type, position=-2): option = option_type() num = abs(position) self.mode.insert_item(num, option.name, option_type) self._options[option_type] = option # # Callbacks # def _on_mode__content_changed(self, combo): self.emit('changed') def _on_entry__activate(self, entry): self.emit('changed') def _on_entry__changed(self, entry): entry.props.secondary_icon_sensitive = bool(entry.get_text()) def _position_filter_menu(self, data): window = self.entry.get_icon_window(gtk.ENTRY_ICON_PRIMARY) x, y = window.get_origin() y += window.get_size()[1] border = self.entry.style_get_property('progress-border') if border is not None: y += border.bottom return (x, y, True) def _on_entry__icon_release(self, entry, icon_pos, event): if icon_pos == gtk.ENTRY_ICON_SECONDARY: entry.set_text("") entry.grab_focus() self.emit('changed') elif icon_pos == gtk.ENTRY_ICON_PRIMARY: # We don't need create popup filters if haven't search columns. if (not self._container or not hasattr(self._container, 'menu') or not self._container.menu): return self._container.menu.popup(None, None, self._position_filter_menu, 0, event.time) # # SearchFilter # def get_state(self): option = self.mode.get_selected_data() return StringQueryState(filter=self, text=self.entry.get_text(), mode=option and option.mode) def set_state(self, text, mode=None): self.entry.set_text(text) if mode is not None: self.mode.select_item_by_position(mode) def get_title_label(self): return self.title_label def get_mode_combo(self): return self.mode def get_description(self): desc = self.entry.get_text() if desc: mode = self.mode.get_selected_label() return '%s %s "%s"' % ( self.title_label.get_text(), mode, desc, ) # # Public API # def enable_advanced(self): self.mode.show() def set_label(self, label): self.title_label.set_text(label)
class CrashReportDialog(object): def __init__(self, parent): self._parent = parent self._report_submitter = ReportSubmitter() self._report_submitter.connect('submitted', self._on_report__submitted) self._report_submitter.connect('failed', self._on_report__failed) self._create_dialog() self.submitted = False self.deferred = defer.Deferred() def _create_dialog(self): app_info = get_utility(IAppInfo, None) self._dialog = HIGAlertDialog(parent=self._parent, flags=gtk.DIALOG_MODAL, type=gtk.MESSAGE_WARNING) self._dialog.set_details_label(_("Details ...")) primary_fmt = _( 'We\'r sorry to inform you that an error occurred while ' 'running %s. Please help us improving Stoq by sending a ' 'automatically generated report about the incident.\n' 'Click on details to see the report text.') self._dialog.set_primary(primary_fmt % (app_info.get('name'), ), bold=False) self._create_details() self._create_comments() self._create_email() self._no_button = self._dialog.add_button(_('No thanks'), gtk.RESPONSE_NO) self._yes_button = self._dialog.add_button(_('Send report'), gtk.RESPONSE_YES) self._insert_tracebacks() def _create_details(self): sw = gtk.ScrolledWindow() self._dialog.set_details_widget(sw) sw.set_shadow_type(gtk.SHADOW_ETCHED_OUT) sw.show() view = gtk.TextView() sw.add(view) view.set_size_request(500, 350) view.show() self._details_buffer = view.get_buffer() def _create_comments(self): sw = gtk.ScrolledWindow() sw.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC) self._dialog.main_vbox.pack_start(sw, False, False, 6) sw.set_shadow_type(gtk.SHADOW_ETCHED_OUT) sw.show() view = gtk.TextView() view.set_wrap_mode(gtk.WRAP_WORD) view.set_accepts_tab(False) view_style = view.get_style() def focus_in(view, event): if self._comments_buffer.props.text != _DEFAULT_COMMENT: return self._comments_buffer.set_text("") view.modify_text(gtk.STATE_NORMAL, view_style.text[gtk.STATE_NORMAL]) view.connect('focus-in-event', focus_in) def focus_out(view, event): if self._comments_buffer.props.text: return self._comments_buffer.set_text(_DEFAULT_COMMENT) view.modify_text(gtk.STATE_NORMAL, view_style.text[gtk.STATE_INSENSITIVE]) view.connect('focus-out-event', focus_out) view.set_size_request(-1, 100) sw.add(view) self._comments_buffer = view.get_buffer() self._comments_buffer.create_tag("highlight") self._comments_buffer.insert_with_tags_by_name( self._comments_buffer.get_iter_at_offset(0), _DEFAULT_COMMENT, 'highlight') view.modify_text(gtk.STATE_NORMAL, view_style.text[gtk.STATE_INSENSITIVE]) view.show() self._comments_view = view def _create_email(self): self._email_entry = HintedEntry() self._email_entry.set_hint(_DEFAULT_EMAIL) self._email_entry.show_hint() self._dialog.main_vbox.pack_start(self._email_entry, False, False, 6) self._email_entry.show() def _insert_tracebacks(self): report = self._report_submitter.report lines = [report.get('log', '')] for key in sorted(report): # Tracebacks already apear in the log if key in ('tracebacks', 'log'): continue lines.append('%s: %s' % (key, report[key])) self._details_buffer.set_text("\n".join(lines)) def _finish(self): self._dialog.set_primary(_( "Thanks for submitting the crash report!\n" "We will use it to make Stoq a better software."), bold=False) self._yes_button.set_label(_("Close")) self._comments_view.set_sensitive(False) self._email_entry.set_sensitive(False) self._yes_button.set_sensitive(True) def _show_report(self, data): message = data.get('message') if message is not None: if data.get('report-url'): label = gtk.LinkButton(data['report-url'], message) else: label = gtk.Label(message) self._dialog.vbox.pack_start(label) label.show() self._finish() def _show_error(self): label = gtk.Label(_("Failed to submit bugreport")) self._dialog.vbox.pack_start(label) label.show() self._finish() def _submit_report(self): self._no_button.hide() self._yes_button.set_sensitive(False) self._yes_button.set_label(_('Sending...')) if self._parent: self._parent.destroy() comments = self._comments_buffer.props.text if comments == _DEFAULT_COMMENT: comments = "" self._report_submitter.report['comments'] = comments email = self._email_entry.get_text() if email == _DEFAULT_EMAIL: email = "" self._report_submitter.report['email'] = email self._report_submitter.submit() def _on_dialog__response(self, dialog, response): if response == gtk.RESPONSE_YES: if self.submitted: self._destroy() self._submit_report() return self._destroy() def _destroy(self): self._dialog.destroy() self.deferred.callback(None) def run(self): self._dialog.connect('response', self._on_dialog__response) self._dialog.show_all() return self.deferred def _on_report__failed(self, response, failure): self._show_error() self.submitted = True def _on_report__submitted(self, response, data): # If the requested successed but the script failed, data is None. if not data: self._show_error() else: self._show_report(data) self.submitted = True
class CrashReportDialog(object): def __init__(self, parent, callback): self._parent = parent self._report_submitter = ReportSubmitter() self._report_submitter.connect('submitted', self._on_report__submitted) self._report_submitter.connect('failed', self._on_report__failed) self._create_dialog() self.submitted = False self._callback = callback def _create_dialog(self): app_info = get_utility(IAppInfo, None) self._dialog = HIGAlertDialog(parent=self._parent, flags=gtk.DIALOG_MODAL, type=gtk.MESSAGE_WARNING) self._dialog.set_details_label(_("Details ...")) primary_fmt = _('We\'r sorry to inform you that an error occurred while ' 'running %s. Please help us improving Stoq by sending a ' 'automatically generated report about the incident.\n' 'Click on details to see the report text.') self._dialog.set_primary(primary_fmt % (app_info.get('name'), ), bold=False) self._create_details() self._create_comments() self._create_email() self._no_button = self._dialog.add_button(_('No thanks'), gtk.RESPONSE_NO) self._yes_button = self._dialog.add_button(_('Send report'), gtk.RESPONSE_YES) self._insert_tracebacks() def _create_details(self): sw = gtk.ScrolledWindow() self._dialog.set_details_widget(sw) sw.set_shadow_type(gtk.SHADOW_ETCHED_OUT) sw.show() view = gtk.TextView() sw.add(view) view.set_size_request(500, 350) view.show() self._details_buffer = view.get_buffer() def _create_comments(self): sw = gtk.ScrolledWindow() sw.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC) self._dialog.main_vbox.pack_start(sw, False, False, 6) sw.set_shadow_type(gtk.SHADOW_ETCHED_OUT) sw.show() view = gtk.TextView() view.set_wrap_mode(gtk.WRAP_WORD) view.set_accepts_tab(False) view_style = view.get_style() def focus_in(view, event): if self._comments_buffer.props.text != _DEFAULT_COMMENT: return self._comments_buffer.set_text("") view.modify_text( gtk.STATE_NORMAL, view_style.text[gtk.STATE_NORMAL]) view.connect('focus-in-event', focus_in) def focus_out(view, event): if self._comments_buffer.props.text: return self._comments_buffer.set_text(_DEFAULT_COMMENT) view.modify_text( gtk.STATE_NORMAL, view_style.text[gtk.STATE_INSENSITIVE]) view.connect('focus-out-event', focus_out) view.set_size_request(-1, 100) sw.add(view) self._comments_buffer = view.get_buffer() self._comments_buffer.create_tag("highlight") self._comments_buffer.insert_with_tags_by_name( self._comments_buffer.get_iter_at_offset(0), _DEFAULT_COMMENT, 'highlight') view.modify_text( gtk.STATE_NORMAL, view_style.text[gtk.STATE_INSENSITIVE]) view.show() self._comments_view = view def _create_email(self): self._email_entry = HintedEntry() self._email_entry.set_hint(_DEFAULT_EMAIL) self._email_entry.show_hint() self._dialog.main_vbox.pack_start(self._email_entry, False, False, 6) self._email_entry.show() def _insert_tracebacks(self): report = self._report_submitter.report lines = [report.get('log', '')] for key in sorted(report): # Tracebacks already apear in the log if key in ('tracebacks', 'log'): continue lines.append('%s: %s' % (key, report[key])) self._details_buffer.set_text("\n".join(lines)) def _finish(self): self._dialog.set_primary( _("Thanks for submitting the crash report!\n" "We will use it to make Stoq a better software."), bold=False) self._yes_button.set_label(_("Close")) self._comments_view.set_sensitive(False) self._email_entry.set_sensitive(False) self._yes_button.set_sensitive(True) def _show_report(self, data): message = data.get('message') if message is not None: if data.get('report-url'): label = gtk.LinkButton(data['report-url'], message) else: label = gtk.Label(message) self._dialog.vbox.pack_start(label) label.show() self._finish() def _show_error(self): label = gtk.Label(_("Failed to submit bugreport")) self._dialog.vbox.pack_start(label) label.show() self._finish() def _submit_report(self): self._no_button.hide() self._yes_button.set_sensitive(False) self._yes_button.set_label(_('Sending...')) if self._parent: self._parent.destroy() comments = self._comments_buffer.props.text if comments == _DEFAULT_COMMENT: comments = "" self._report_submitter.report['comments'] = comments email = self._email_entry.get_text() if email == _DEFAULT_EMAIL: email = "" self._report_submitter.report['email'] = email self._report_submitter.submit() def _on_dialog__response(self, dialog, response): if response == gtk.RESPONSE_YES: if self.submitted: self._destroy() self._submit_report() return self._destroy() def _destroy(self): self._dialog.destroy() if self._callback is not None: self._callback() def run(self): self._dialog.connect('response', self._on_dialog__response) self._dialog.show_all() def _on_report__failed(self, response, failure): self._show_error() self.submitted = True def _on_report__submitted(self, response, data): # If the requested successed but the script failed, data is None. if not data: self._show_error() else: self._show_report(data) self.submitted = True
class StringSearchFilter(SearchFilter): """ Contains: - a label - an entry :ivar entry: the entry :ivar label: the label """ def __init__(self, label, chars=0, container=None): """ Create a new StringSearchFilter object. :param label: label of the search filter :param chars: maximum number of chars used by the search entry """ self._container = container SearchFilter.__init__(self, label=label) self.title_label = gtk.Label(label) self.pack_start(self.title_label, False, False) self.title_label.show() self._options = {} self.mode = ProxyComboBox() self.mode.connect('content-changed', self._on_mode__content_changed) self.pack_start(self.mode, False, False, 6) self.entry = HintedEntry() self.entry.set_hint(_("Search")) self.entry.show_hint() self.entry.props.secondary_icon_sensitive = False data = environ.get_resource_string('stoq', 'pixmaps', 'stoq-funnel-16x16.png') image = pixbuf_from_string(data) self.entry.set_icon_from_pixbuf(gtk.ENTRY_ICON_PRIMARY, image) self.entry.set_icon_tooltip_text(gtk.ENTRY_ICON_PRIMARY, _("Add a filter")) self.entry.set_icon_from_stock(gtk.ENTRY_ICON_SECONDARY, gtk.STOCK_CLEAR) self.entry.set_icon_tooltip_text(gtk.ENTRY_ICON_SECONDARY, _("Clear the search")) self.entry.connect("icon-release", self._on_entry__icon_release) self.entry.connect('activate', self._on_entry__activate) self.entry.connect('changed', self._on_entry__changed) if chars: self.entry.set_width_chars(chars) self.pack_start(self.entry, False, False, 6) self.entry.show() for option in (ContainsAll, ContainsExactly, DoesNotContain): self._add_option(option) self.mode.select_item_by_position(0) def _add_option(self, option_type, position=-2): option = option_type() num = abs(position) self.mode.insert_item(num, option.name, option_type) self._options[option_type] = option # # Callbacks # def _on_mode__content_changed(self, combo): self.emit('changed') def _on_entry__activate(self, entry): self.emit('changed') def _on_entry__changed(self, entry): entry.props.secondary_icon_sensitive = bool(entry.get_text()) def _position_filter_menu(self, data): window = self.entry.get_icon_window(gtk.ENTRY_ICON_PRIMARY) x, y = window.get_origin() y += window.get_size()[1] border = self.entry.style_get_property('progress-border') if border is not None: y += border.bottom return (x, y, True) def _on_entry__icon_release(self, entry, icon_pos, event): if icon_pos == gtk.ENTRY_ICON_SECONDARY: entry.set_text("") entry.grab_focus() self.emit('changed') elif icon_pos == gtk.ENTRY_ICON_PRIMARY: # We don't need create popup filters if haven't search columns. if (not self._container or not hasattr(self._container, 'menu') or not self._container.menu): return self._container.menu.popup(None, None, self._position_filter_menu, 0, event.time) # # SearchFilter # def get_state(self): option = self.mode.get_selected_data() return StringQueryState(filter=self, text=unicode(self.entry.get_text()), mode=option and option.mode) def set_state(self, text, mode=None): self.entry.set_text(text) if mode is not None: self.mode.select_item_by_position(mode) def get_title_label(self): return self.title_label def get_mode_combo(self): return self.mode def get_description(self): desc = self.entry.get_text() if desc: mode = self.mode.get_selected_label() return '%s %s "%s"' % (self.title_label.get_text(), mode, desc,) # # Public API # def enable_advanced(self): # Do not show the funnel icon if its an advanced filter self.entry.set_icon_from_pixbuf(gtk.ENTRY_ICON_PRIMARY, None) self.mode.show() def set_label(self, label): self.title_label.set_text(label)