class SettingsWin(QFrame): def __init__(self, parent = None, cenwid = None, status=None): QFrame.__init__(self, parent) self.title = translate('Settings', "Patterns") connect = lambda c, signal, s: self.connect(c, SIGNAL(signal), s) self.setFrameStyle(QFrame.Box) self.listbox = ListBox() self.listbox.setSelectionMode(self.listbox.ExtendedSelection) buttons = ListButtons() self.listbox.addItems(status['patterns']) hbox = QHBoxLayout() hbox.addWidget(self.listbox) self.setLayout(hbox) vbox = QVBoxLayout() sortlistbox = QPushButton(translate("Pattern Settings", '&Sort')) self._sortOrder = Qt.AscendingOrder connect(sortlistbox, 'clicked()', self._sortListBox) vbox.addWidget(sortlistbox) vbox.addLayout(buttons) vbox.addStretch() hbox.addLayout(vbox) connect(buttons, "add", self.addPattern) connect(buttons, "edit", self.editItem) buttons.duplicate.setVisible(False) self.listbox.connectToListButtons(buttons) self.listbox.editButton = buttons.edit connect(self.listbox, 'itemDoubleClicked(QListWidgetItem *)', self._doubleClicked) def _sortListBox(self): if self._sortOrder == Qt.AscendingOrder: self.listbox.sortItems(Qt.DescendingOrder) self._sortOrder = Qt.DescendingOrder else: self.listbox.sortItems(Qt.AscendingOrder) self._sortOrder = Qt.AscendingOrder def saveSettings(self): patterns = [unicode(self.listbox.item(row).text()) for row in xrange(self.listbox.count())] cparser = PuddleConfig() cparser.setSection('editor', 'patterns', patterns) def addPattern(self): l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] row = self.listbox.currentRow() if row < 0: row = 0 (text, ok) = QInputDialog().getItem(self, 'puddletag', translate("Pattern Settings", 'Enter a pattern'), patterns, row) if ok: self.listbox.clearSelection() self.listbox.addItem(text) self.listbox.setCurrentRow(self.listbox.count() - 1) def _doubleClicked(self, item): self.editItem() def editItem(self, row=None): if row is None: row = self.listbox.currentRow() l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] (text, ok) = QInputDialog().getItem (self, 'puddletag', translate("Pattern Settings", 'Enter a pattern'), patterns, row) if ok: item = l(row) item.setText(text) self.listbox.setItemSelected(item, True) def applySettings(self, control): item = self.listbox.item patterns = [item(row).text() for row in xrange(self.listbox.count())] control.setItems(patterns)
class MassTagEdit(QDialog): def __init__(self, tag_sources, profiles=None, parent = None): super(MassTagEdit, self).__init__(parent) self.setWindowTitle(translate('Profile Config', 'Configure Mass Tagging Profiles')) winsettings('masstag_edit', self) self.listbox = ListBox() self.tag_sources = tag_sources okcancel = OKCancel() okcancel.ok.setDefault(True) self.buttonlist = ListButtons() connect = lambda control, signal, slot: self.connect( control, SIGNAL(signal), slot) connect(okcancel, "ok" , self.okClicked) connect(okcancel, "cancel",self.close) connect(self.buttonlist, "add", self.addClicked) connect(self.buttonlist, "edit", self.editClicked) connect(self.buttonlist, "moveup", self.moveUp) connect(self.buttonlist, "movedown", self.moveDown) connect(self.buttonlist, "remove", self.remove) connect(self.buttonlist, "duplicate", self.dupClicked) connect(self.listbox, "itemDoubleClicked (QListWidgetItem *)", self.editClicked) connect(self.listbox, "currentRowChanged(int)", self.enableListButtons) self.enableListButtons(self.listbox.currentRow()) layout = QVBoxLayout() self.setLayout(layout) layout.addWidget(QLabel(translate('Profile Config', 'Masstagging Profiles'))) list_layout = QHBoxLayout() list_layout.addWidget(self.listbox, 1) list_layout.addLayout(self.buttonlist) layout.addLayout(list_layout) layout.addLayout(okcancel) if profiles is not None: self.setProfiles(profiles) def addClicked(self): win = MTProfileEdit(self.tag_sources, parent=self) win.setModal(True) self.connect(win, SIGNAL('profileChanged'), self.addProfile) win.show() def addProfile(self, profile): row = self.listbox.count() self.listbox.addItem(profile.name) self._profiles.append(profile) def dupClicked(self): row = self.listbox.currentRow() if row == -1: return win = MTProfileEdit(self.tag_sources, deepcopy(self._profiles[row]), self) win.setModal(True) self.connect(win, SIGNAL('profileChanged'), self.addProfile) win.show() def editClicked(self, item=None): if item: row = self.listbox.row(item) else: row = self.listbox.currentRow() if row == -1: return win = MTProfileEdit(self.tag_sources, self._profiles[row], self) win.setModal(True) self.connect(win, SIGNAL('profileChanged'), partial(self.replaceProfile, row)) win.show() def enableListButtons(self, val): if val == -1: [button.setEnabled(False) for button in self.buttonlist.widgets[1:]] else: [button.setEnabled(True) for button in self.buttonlist.widgets[1:]] def loadProfiles(self, dirpath=PROFILEDIR): profiles = load_all_mtps(dirpath, self.tag_sources) self.setProfiles(filter(None, profiles)) def moveDown(self): self.listbox.moveDown(self._profiles) def moveUp(self): self.listbox.moveUp(self._profiles) def okClicked(self): self.emit(SIGNAL('profilesChanged'), self._profiles) self.saveProfiles(PROFILEDIR, self._profiles) self.close() def remove(self): row = self.listbox.currentRow() if row == -1: return del(self._profiles[row]) self.listbox.takeItem(row) def replaceProfile(self, row, profile): self._profiles[row] = profile self.listbox.item(row).setText(profile.name) def saveProfiles(self, dirpath, profiles): filenames = {} order = [] for profile in profiles: filename = profile.name + u'.mtp' i = 0 while filename in filenames: filename = u'%s_%d%s' % (profile.name, i, u'.mtp') i += 1 filenames[filename] = profile order.append(profile.name) files = glob.glob(os.path.join(dirpath, '*.mtp')) for f in files: if f not in filenames: try: os.remove(f) except EnvironmentError: pass for filename, profile in filenames.items(): save_mtp(profile, os.path.join(dirpath, filename)) f = open(os.path.join(dirpath, 'order'), 'w') f.write(u'\n'.join(order)) f.close() def setProfiles(self, profiles): self._profiles = profiles for profile in self._profiles: self.listbox.addItem(profile.name)
class MTProfileEdit(QDialog): def __init__(self, tag_sources, profile=None, parent=None): super(MTProfileEdit, self).__init__(parent) self.setWindowTitle(translate('Profile Editor', 'Edit Masstagging Profile')) winsettings('masstag_profile', self) self._configs = [] self.tag_sources = tag_sources self._tsps = [] self._name = QLineEdit(translate('Profile Editor', 'Masstagging Profile')) self._desc = QLineEdit() self.listbox = ListBox() self.okcancel = OKCancel() self.okcancel.ok.setDefault(True) self.buttonlist = ListButtons() self.pattern = QLineEdit('%artist% - %album%/%track% - %title%') self.pattern.setToolTip(translate('Profile Editor', "<p>If no tag information is found in a file, " "the tags retrieved using this pattern will be used instead.</p>")) self.albumBound = QSpinBox() self.albumBound.setRange(0,100) self.albumBound.setValue(70) self.albumBound.setToolTip(translate('Profile Editor', "<p>The artist and album fields will be used in " "determining whether an album matches the retrieved one. " "Each field will be compared using a fuzzy matching algorithm. " "If the resulting average match percentage is greater " "or equal than what you specify here it'll be " "considered to match.</p>")) self.matchFields = QLineEdit('artist, title') self.matchFields.setToolTip(translate('Profile Editor', '<p>The fields listed here will be used in ' 'determining whether a file matches a retrieved track. ' 'Each field will be compared using a fuzzy matching ' 'algorithm. If the resulting average match ' 'percentage is greater than the "Minimum Percentage" ' 'it\'ll be considered to match.</p>')) self.trackBound = QSpinBox() self.trackBound.setRange(0,100) self.trackBound.setValue(80) self.jfdi = QCheckBox(translate('Profile Editor', 'Brute force unmatched files.')) self.jfdi.setToolTip(translate('Profile Editor', "<p>Check to enable brute forcing matches. " " If a proper match isn't found for a file, " 'the files will get sorted by filename, ' 'the retrieved tag sources by filename and ' 'corresponding (unmatched) tracks will matched.</p>')) self.existing = QCheckBox(translate('Profile Editor', 'Update empty fields only.')) self.grid = QGridLayout() self.setLayout(self.grid) self.grid.addLayout(create_buddy( translate('Profile Editor', '&Name:'), self._name), 0, 0, 1, 2) self.grid.addLayout(create_buddy( translate('Profile Editor', '&Description'), self._desc), 1, 0, 1, 2) self.grid.addWidget(self.listbox, 2, 0) self.grid.setRowStretch(2, 1) self.grid.addLayout(self.buttonlist, 2, 1) self.grid.addLayout(create_buddy(translate('Profile Editor', 'Pattern to match filenames against.'), self.pattern, QVBoxLayout()), 3, 0, 1, 2) self.grid.addLayout(create_buddy(translate('Profile Editor', 'Minimum percentage required for album matches.'), self.albumBound), 4, 0, 1, 2) self.grid.addLayout(create_buddy(translate('Profile Editor', 'Match tracks using fields: '), self.matchFields, QVBoxLayout()), 5, 0, 1, 2) self.grid.addLayout(create_buddy(translate('Profile Editor', 'Minimum percentage required for track match.'), self.trackBound), 6, 0, 1, 2) self.grid.addWidget(self.jfdi, 7, 0, 1, 2) self.grid.addWidget(self.existing, 8, 0, 1, 2) self.grid.addLayout(self.okcancel, 9, 0, 1, 2) connect = lambda control, signal, slot: self.connect( control, SIGNAL(signal), slot) connect(self.okcancel, "ok" , self.okClicked) connect(self.okcancel, "cancel",self.close) connect(self.buttonlist, "add", self.addClicked) connect(self.buttonlist, "edit", self.editClicked) connect(self.buttonlist, "moveup", self.moveUp) connect(self.buttonlist, "movedown", self.moveDown) connect(self.buttonlist, "remove", self.remove) connect(self.buttonlist, "duplicate", self.dupClicked) connect(self.listbox, "itemDoubleClicked (QListWidgetItem *)", self.editClicked) connect(self.listbox, "currentRowChanged(int)", self.enableListButtons) if profile is not None: self.setProfile(profile) self.enableListButtons(self.listbox.currentRow()) def addClicked(self): win = TSProfileEdit(self.tag_sources, None, self) win.setModal(True) self.connect(win, SIGNAL('profileChanged'), self.addTSProfile) win.show() def addTSProfile(self, profile): row = self.listbox.count() self.listbox.addItem(profile.tag_source.name) self._tsps.append(profile) def dupClicked(self): row = self.listbox.currentRow() if row == -1: return win = TSProfileEdit(self.tag_sources, self._tsps[row], self) win.setModal(True) self.connect(win, SIGNAL('profileChanged'), self.addTSProfile) win.show() def editClicked(self, item=None): if item: row = self.listbox.row(item) else: row = self.listbox.currentRow() if row == -1: return win = TSProfileEdit(self.tag_sources, self._tsps[row], self) win.setModal(True) self.connect(win, SIGNAL('profileChanged'), partial(self.replaceTSProfile, row)) win.show() def replaceTSProfile(self, row, profile): self._tsps[row] = profile self.listbox.item(row).setText(profile.tag_source.name) def enableListButtons(self, val): if val == -1: [button.setEnabled(False) for button in self.buttonlist.widgets[1:]] else: [button.setEnabled(True) for button in self.buttonlist.widgets[1:]] def moveDown(self): self.listbox.moveDown(self._tsps) def moveUp(self): self.listbox.moveUp(self._tsps) def okClicked(self): fields = [z.strip() for z in unicode(self.matchFields.text()).split(u',')] mtp = MassTagProfile(unicode(self._name.text()), unicode(self._desc.text()), fields, None, unicode(self.pattern.text()), self._tsps, self.albumBound.value() / 100.0, self.trackBound.value() / 100.0, self.jfdi.isChecked(), self.existing.isChecked(), u'') self.emit(SIGNAL('profileChanged'), mtp) self.close() def remove(self): row = self.listbox.currentRow() if row == -1: return del(self._tsps[row]) self.listbox.takeItem(row) def setProfile(self, profile): self._tsps = [tsp for tsp in profile.profiles if tsp.tag_source] [self.listbox.addItem(tsp.tag_source.name) for tsp in self._tsps] self.albumBound.setValue(profile.album_bound * 100) self.pattern.setText(profile.file_pattern) self.matchFields.setText(u', '.join(profile.fields)) self.trackBound.setValue(profile.track_bound * 100) self.jfdi.setChecked(profile.jfdi) self._name.setText(profile.name) self._desc.setText(profile.desc) self.existing.setChecked(profile.leave_existing)
class ButtonsAndList(QFrame): def __init__(self, parent=None, title=u'', add_text=ADD_TEXT, help_text=u''): QFrame.__init__(self, parent) self.title = title connect = lambda c, signal, s: self.connect(c, SIGNAL(signal), s) self.setFrameStyle(QFrame.Box) self.listbox = ListBox() self.listbox.setSelectionMode(self.listbox.ExtendedSelection) buttons = ListButtons() hbox = QHBoxLayout() hbox.addWidget(self.listbox) vbox = QVBoxLayout() sortlistbox = QPushButton(translate("Defaults", '&Sort')) self._sortOrder = Qt.AscendingOrder connect(sortlistbox, 'clicked()', self._sortListBox) vbox.addWidget(sortlistbox) vbox.addLayout(buttons) vbox.addStretch() hbox.addLayout(vbox) if help_text: label = QLabel(help_text) layout = QVBoxLayout() layout.addWidget(label) layout.addLayout(hbox, 1) self.setLayout(layout) else: self.setLayout(hbox) connect(buttons, "add", self.addItem) connect(buttons, "edit", self.editItem) buttons.duplicate.setVisible(False) self.listbox.connectToListButtons(buttons) self.listbox.editButton = buttons.edit connect(self.listbox, 'itemDoubleClicked(QListWidgetItem *)', self._doubleClicked) self.addText = add_text def _doubleClicked(self, item): self.editItem() def _sortListBox(self): if self._sortOrder == Qt.AscendingOrder: self.listbox.sortItems(Qt.DescendingOrder) self._sortOrder = Qt.DescendingOrder else: self.listbox.sortItems(Qt.AscendingOrder) self._sortOrder = Qt.AscendingOrder def addItem(self): l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] row = self.listbox.currentRow() if row < 0: row = 0 (text, ok) = QInputDialog().getItem(self, 'puddletag', self.addText, patterns, row) if ok: self.listbox.clearSelection() self.listbox.addItem(text) self.listbox.setCurrentRow(self.listbox.count() - 1) def addItems(self, items): self.listbox.addItems(items) def editItem(self, row=None): if row is None: row = self.listbox.currentRow() l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] (text, ok) = QInputDialog().getItem(self, 'puddletag', self.addText, patterns, row) if ok: item = l(row) item.setText(text) self.listbox.setItemSelected(item, True) def getItems(self): return [item.text() for item in self.listbox.items()]
class SettingsWin(QFrame): def __init__(self, parent=None, cenwid=None, status=None): QFrame.__init__(self, parent) self.title = translate('Settings', "Patterns") connect = lambda c, signal, s: self.connect(c, SIGNAL(signal), s) self.setFrameStyle(QFrame.Box) self.listbox = ListBox() self.listbox.setSelectionMode(self.listbox.ExtendedSelection) buttons = ListButtons() self.listbox.addItems(status['patterns']) hbox = QHBoxLayout() hbox.addWidget(self.listbox) self.setLayout(hbox) vbox = QVBoxLayout() sortlistbox = QPushButton(translate("Pattern Settings", '&Sort')) self._sortOrder = Qt.AscendingOrder connect(sortlistbox, 'clicked()', self._sortListBox) vbox.addWidget(sortlistbox) vbox.addLayout(buttons) vbox.addStretch() hbox.addLayout(vbox) connect(buttons, "add", self.addPattern) connect(buttons, "edit", self.editItem) buttons.duplicate.setVisible(False) self.listbox.connectToListButtons(buttons) self.listbox.editButton = buttons.edit connect(self.listbox, 'itemDoubleClicked(QListWidgetItem *)', self._doubleClicked) def _sortListBox(self): if self._sortOrder == Qt.AscendingOrder: self.listbox.sortItems(Qt.DescendingOrder) self._sortOrder = Qt.DescendingOrder else: self.listbox.sortItems(Qt.AscendingOrder) self._sortOrder = Qt.AscendingOrder def saveSettings(self): patterns = [ unicode(self.listbox.item(row).text()) for row in xrange(self.listbox.count()) ] cparser = PuddleConfig() cparser.setSection('editor', 'patterns', patterns) def addPattern(self): l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] row = self.listbox.currentRow() if row < 0: row = 0 (text, ok) = QInputDialog().getItem( self, 'puddletag', translate("Pattern Settings", 'Enter a pattern'), patterns, row) if ok: self.listbox.clearSelection() self.listbox.addItem(text) self.listbox.setCurrentRow(self.listbox.count() - 1) def _doubleClicked(self, item): self.editItem() def editItem(self, row=None): if row is None: row = self.listbox.currentRow() l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] (text, ok) = QInputDialog().getItem( self, 'puddletag', translate("Pattern Settings", 'Enter a pattern'), patterns, row) if ok: item = l(row) item.setText(text) self.listbox.setItemSelected(item, True) def applySettings(self, control): item = self.listbox.item patterns = [item(row).text() for row in xrange(self.listbox.count())] control.setItems(patterns)
class ButtonsAndList(QFrame): def __init__(self, parent=None, title=u'', add_text=ADD_TEXT, help_text=u''): QFrame.__init__(self, parent) self.title = title connect = lambda c, signal, s: self.connect(c, SIGNAL(signal), s) self.setFrameStyle(QFrame.Box) self.listbox = ListBox() self.listbox.setSelectionMode(self.listbox.ExtendedSelection) buttons = ListButtons() hbox = QHBoxLayout() hbox.addWidget(self.listbox) vbox = QVBoxLayout() sortlistbox = QPushButton(translate("Defaults", '&Sort')) self._sortOrder = Qt.AscendingOrder connect(sortlistbox, 'clicked()', self._sortListBox) vbox.addWidget(sortlistbox) vbox.addLayout(buttons) vbox.addStretch() hbox.addLayout(vbox) if help_text: label = QLabel(help_text) layout = QVBoxLayout() layout.addWidget(label) layout.addLayout(hbox, 1) self.setLayout(layout) else: self.setLayout(hbox) connect(buttons, "add", self.addItem) connect(buttons, "edit", self.editItem) buttons.duplicate.setVisible(False) self.listbox.connectToListButtons(buttons) self.listbox.editButton = buttons.edit connect(self.listbox, 'itemDoubleClicked(QListWidgetItem *)', self._doubleClicked) self.addText = add_text def _doubleClicked(self, item): self.editItem() def _sortListBox(self): if self._sortOrder == Qt.AscendingOrder: self.listbox.sortItems(Qt.DescendingOrder) self._sortOrder = Qt.DescendingOrder else: self.listbox.sortItems(Qt.AscendingOrder) self._sortOrder = Qt.AscendingOrder def addItem(self): l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] row = self.listbox.currentRow() if row < 0: row = 0 (text, ok) = QInputDialog().getItem(self, 'puddletag', self.addText, patterns, row) if ok: self.listbox.clearSelection() self.listbox.addItem(text) self.listbox.setCurrentRow(self.listbox.count() - 1) def addItems(self, items): self.listbox.addItems(items) def editItem(self, row=None): if row is None: row = self.listbox.currentRow() l = self.listbox.item patterns = [unicode(l(z).text()) for z in range(self.listbox.count())] (text, ok) = QInputDialog().getItem (self, 'puddletag', self.addText, patterns, row) if ok: item = l(row) item.setText(text) self.listbox.setItemSelected(item, True) def getItems(self): return [item.text() for item in self.listbox.items()]