electrum

Electrum Bitcoin wallet
git clone https://git.parazyd.org/electrum
Log | Files | Refs | Submodules

contact_list.py (5457B)


      1 #!/usr/bin/env python
      2 #
      3 # Electrum - lightweight Bitcoin client
      4 # Copyright (C) 2015 Thomas Voegtlin
      5 #
      6 # Permission is hereby granted, free of charge, to any person
      7 # obtaining a copy of this software and associated documentation files
      8 # (the "Software"), to deal in the Software without restriction,
      9 # including without limitation the rights to use, copy, modify, merge,
     10 # publish, distribute, sublicense, and/or sell copies of the Software,
     11 # and to permit persons to whom the Software is furnished to do so,
     12 # subject to the following conditions:
     13 #
     14 # The above copyright notice and this permission notice shall be
     15 # included in all copies or substantial portions of the Software.
     16 #
     17 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
     18 # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
     19 # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
     20 # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
     21 # BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
     22 # ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
     23 # CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
     24 # SOFTWARE.
     25 
     26 from enum import IntEnum
     27 
     28 from PyQt5.QtGui import QStandardItemModel, QStandardItem
     29 from PyQt5.QtCore import Qt, QPersistentModelIndex, QModelIndex
     30 from PyQt5.QtWidgets import (QAbstractItemView, QMenu)
     31 
     32 from electrum.i18n import _
     33 from electrum.bitcoin import is_address
     34 from electrum.util import block_explorer_URL
     35 from electrum.plugin import run_hook
     36 
     37 from .util import MyTreeView, webopen
     38 
     39 
     40 class ContactList(MyTreeView):
     41 
     42     class Columns(IntEnum):
     43         NAME = 0
     44         ADDRESS = 1
     45 
     46     headers = {
     47         Columns.NAME: _('Name'),
     48         Columns.ADDRESS: _('Address'),
     49     }
     50     filter_columns = [Columns.NAME, Columns.ADDRESS]
     51 
     52     def __init__(self, parent):
     53         super().__init__(parent, self.create_menu,
     54                          stretch_column=self.Columns.NAME,
     55                          editable_columns=[self.Columns.NAME])
     56         self.setModel(QStandardItemModel(self))
     57         self.setSelectionMode(QAbstractItemView.ExtendedSelection)
     58         self.setSortingEnabled(True)
     59         self.update()
     60 
     61     def on_edited(self, idx, user_role, text):
     62         _type, prior_name = self.parent.contacts.pop(user_role)
     63         self.parent.set_contact(text, user_role)
     64         self.update()
     65 
     66     def create_menu(self, position):
     67         menu = QMenu()
     68         idx = self.indexAt(position)
     69         column = idx.column() or self.Columns.NAME
     70         selected_keys = []
     71         for s_idx in self.selected_in_column(self.Columns.NAME):
     72             sel_key = self.model().itemFromIndex(s_idx).data(Qt.UserRole)
     73             selected_keys.append(sel_key)
     74         if not selected_keys or not idx.isValid():
     75             menu.addAction(_("New contact"), lambda: self.parent.new_contact_dialog())
     76             menu.addAction(_("Import file"), lambda: self.parent.import_contacts())
     77             menu.addAction(_("Export file"), lambda: self.parent.export_contacts())
     78         else:
     79             column_title = self.model().horizontalHeaderItem(column).text()
     80             column_data = '\n'.join(self.model().itemFromIndex(s_idx).text()
     81                                     for s_idx in self.selected_in_column(column))
     82             menu.addAction(_("Copy {}").format(column_title), lambda: self.place_text_on_clipboard(column_data, title=column_title))
     83             if column in self.editable_columns:
     84                 item = self.model().itemFromIndex(idx)
     85                 if item.isEditable():
     86                     # would not be editable if openalias
     87                     persistent = QPersistentModelIndex(idx)
     88                     menu.addAction(_("Edit {}").format(column_title), lambda p=persistent: self.edit(QModelIndex(p)))
     89             menu.addAction(_("Pay to"), lambda: self.parent.payto_contacts(selected_keys))
     90             menu.addAction(_("Delete"), lambda: self.parent.delete_contacts(selected_keys))
     91             URLs = [block_explorer_URL(self.config, 'addr', key) for key in filter(is_address, selected_keys)]
     92             if URLs:
     93                 menu.addAction(_("View on block explorer"), lambda: [webopen(u) for u in URLs])
     94 
     95         run_hook('create_contact_menu', menu, selected_keys)
     96         menu.exec_(self.viewport().mapToGlobal(position))
     97 
     98     def update(self):
     99         if self.maybe_defer_update():
    100             return
    101         current_key = self.current_item_user_role(col=self.Columns.NAME)
    102         self.model().clear()
    103         self.update_headers(self.__class__.headers)
    104         set_current = None
    105         for key in sorted(self.parent.contacts.keys()):
    106             contact_type, name = self.parent.contacts[key]
    107             items = [QStandardItem(x) for x in (name, key)]
    108             items[self.Columns.NAME].setEditable(contact_type != 'openalias')
    109             items[self.Columns.ADDRESS].setEditable(False)
    110             items[self.Columns.NAME].setData(key, Qt.UserRole)
    111             row_count = self.model().rowCount()
    112             self.model().insertRow(row_count, items)
    113             if key == current_key:
    114                 idx = self.model().index(row_count, self.Columns.NAME)
    115                 set_current = QPersistentModelIndex(idx)
    116         self.set_current_idx(set_current)
    117         # FIXME refresh loses sort order; so set "default" here:
    118         self.sortByColumn(self.Columns.NAME, Qt.AscendingOrder)
    119         self.filter()
    120         run_hook('update_contacts_tab', self)