# -*- coding: utf-8 -*- # Copyright (C) 2013 Alex Yatskov # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . import anki import aqt import hashlib import urllib2 # # Audio helpers # def audioBuildFilename(kana, kanji): filename = u'yomichan_{}'.format(kana) if kanji: filename += u'_{}'.format(kanji) filename += u'.mp3' return filename def audioDownload(kana, kanji): url = 'http://assets.languagepod101.com/dictionary/japanese/audiomp3.php?kanji={}'.format(urllib2.quote(kanji.encode('utf-8'))) if kana: url += '&kana={}'.format(urllib2.quote(kana.encode('utf-8'))) try: resp = urllib2.urlopen(url) except urllib2.URLError: return None if resp.code != 200: return None return resp.read() def audioIsPlaceholder(data): m = hashlib.md5() m.update(data) return m.hexdigest() == '7e2c2f954ef6051373ba916f000168dc' def audioInject(note, fields, filename): for field in fields: if field in note: note[field] += u'[sound:{}]'.format(filename) # # Anki # class Anki: def addNote(self, deckName, modelName, fields, tags, audio): collection = self.collection() if collection is None: return note = self.createNote(deckName, modelName, fields, tags) if note is None: return if audio is not None and len(audio['fields']) > 0: data = audioDownload(audio['kana'], audio['kanji']) if data is not None and not audioIsPlaceholder(data): filename = audioBuildFilename(audio['kana'], audio['kanji']) audioInject(note, audio['fields'], filename) self.media().writeData(filename, data) self.startEditing() collection.addNote(note) collection.autosave() return note.id def canAddNote(self, deckName, modelName, fields): return bool(self.createNote(deckName, modelName, fields)) def createNote(self, deckName, modelName, fields, tags=[]): collection = self.collection() if collection is None: return model = collection.models.byName(modelName) if model is None: return deck = collection.decks.byName(deckName) if deck is None: return note = anki.notes.Note(collection, model) note.model()['did'] = deck['id'] note.tags = tags for name, value in fields.items(): if name in note: note[name] = value if not note.dupeOrEmpty(): return note def browseNote(self, noteId): browser = aqt.dialogs.open('Browser', self.window()) browser.form.searchEdit.lineEdit().setText('nid:{0}'.format(noteId)) browser.onSearch() def startEditing(self): self.window().requireReset() def stopEditing(self): if self.collection() is not None: self.window().maybeReset() def window(self): return aqt.mw def addUiAction(self, action): self.window().form.menuTools.addAction(action) def collection(self): return self.window().col def media(self): collection = self.collection() if collection is not None: return collection.media def modelNames(self): collection = self.collection() if collection is not None: return collection.models.allNames() def modelFieldNames(self, modelName): collection = self.collection() if collection is None: return model = collection.models.byName(modelName) if model is not None: return [field['name'] for field in model['flds']] def deckNames(self): collection = self.collection() if collection is not None: return collection.decks.allNames()