Python實(shí)現(xiàn)截圖生成符合markdown的鏈接
背景
之前是用的是typora來(lái)寫(xiě)的文章,最近typora最近開(kāi)始收費(fèi)了,所以就不想用了,于是找到了一個(gè)替代品MarkText,感覺(jué)跟typora差不多
整體樣子就像上面,簡(jiǎn)約風(fēng),個(gè)人挺喜歡的。唯一的一個(gè)問(wèn)題就是粘貼圖片的時(shí)候,圖片只能放在本地,雖然marktext有圖片上傳的功能,但是只支持GitHub的圖床,設(shè)置了過(guò)后經(jīng)常會(huì)上傳失敗,導(dǎo)致還是存在本地,存在本地的弊端就是文章在轉(zhuǎn)移的需要把圖片都帶上或者復(fù)制到掘金等平臺(tái)的時(shí)候,圖片會(huì)失效。于是打算用python寫(xiě)個(gè)自動(dòng)生成markdownUrl的程序。
思路
程序的整體思路是,首先需要給界面來(lái)設(shè)置存儲(chǔ)的類(lèi)型,即是選擇存在阿里OSS還是七牛的Kodo還是其他的云存儲(chǔ),還有設(shè)置key和secret以及不同存儲(chǔ)類(lèi)型所需要的屬性,接著界面可以顯示復(fù)制的圖片,以及上傳成功后的markdownUrl和httpUrl,界面大體如下
子界面:
主界面:
實(shí)現(xiàn)
程序整體選擇用python來(lái)實(shí)現(xiàn),因?yàn)橹坝眠^(guò)QT,所以GUI的框架用的是pyqt5,數(shù)據(jù)庫(kù)用的是sqlite3,還有阿里云和七牛的sdk等。
整個(gè)界面有一個(gè)主窗口和一個(gè)子窗口構(gòu)成,主界面在初始化的同時(shí)初始化數(shù)據(jù)庫(kù)
class ImgFrame(QMainWindow): ? ? def __init__(self): ? ? ? ? super().__init__() ? ? ? ? self.http_url = None ? ? ? ? self.markdown_url = None ? ? ? ? self.clipboard = None ? ? ? ? self.img = None ? ? ? ? # 初始化數(shù)據(jù)庫(kù) ? ? ? ? self.db = init_db() ? ? ? ? self.init_ui() ? ? def init_ui(self): ? ? ? ? self.setGeometry(300, 300, 500, 500) ? ? ? ? self.setWindowTitle('MarkDown-Img') ? ? ? ? widget = QWidget() ? ? ? ? setupAction = QAction(QIcon('setup.png'), '設(shè)置', self) ? ? ? ? setupAction.setStatusTip('Exit application') ? ? ? ? setupAction.triggered.connect(self.a) ? ? ? ? menubar = QMenuBar(self) ? ? ? ? menubar.setGeometry(QtCore.QRect(0, 0, 251, 23)) ? ? ? ? menubar.setObjectName("menubar") ? ? ? ? setup = menubar.addMenu('系統(tǒng)') ? ? ? ? setup.addAction(setupAction) ? ? ? ? menubar.setVisible(True) ? ? ? ? menubar.setNativeMenuBar(False) ? ? ? ? self.setMenuBar(menubar) ? ? ? ? self.img = QLabel() ? ? ? ? layout = QVBoxLayout() ? ? ? ? layout.addWidget(markdown_widget(self)) ? ? ? ? layout.addWidget(url_widget(self)) ? ? ? ? layout.addWidget(self.img) ? ? ? ? layout.setAlignment(Qt.AlignCenter) ? ? ? ? self.clipboard = QApplication.clipboard() ? ? ? ? self.clipboard.dataChanged.connect(self.paste) ? ? ? ? widget.setLayout(layout) ? ? ? ? self.setCentralWidget(widget) ? ? ? ? self.show()
def init_db(): ? ? connect = sqlite3.connect('markdown-img.db') ? ? global conn ? ? #全局變量conn ? ? conn = connect ? ? cursor = connect.cursor() ? ? cursor.execute(sql) ? ? #返回游標(biāo) ? ? return cursor
子窗口的主要作用就是設(shè)置云存儲(chǔ)所需要的各種字段,然后存儲(chǔ)到數(shù)據(jù)庫(kù)中
class secondFrame(QWidget): ? ? def __init__(self, db): ? ? ? ? super().__init__() ? ? ? ? # self.init_ui() ? ? ? ? self.db = db ? ? ? ? self.resize(400, 100) ? ? ? ? self.setWindowTitle('存儲(chǔ)設(shè)置') ? ? ? ? formlayout = QFormLayout() ? ? ? ? storageLabel = QLabel("存儲(chǔ)") ? ? ? ? self.storageBox = QComboBox() ? ? ? ? self.storageBox.addItems(['阿里OSS', '七牛Kodo']) ? ? ? ? self.endpointLabel = QLabel("endpoint") ? ? ? ? self.endpointLineEdit = QLineEdit("") ? ? ? ? self.endpointLineEdit.setStyleSheet("width:200px") ? ? ? ? self.qntLabel = QLabel("七牛域名") ? ? ? ? self.qnLineEdit = QLineEdit("") ? ? ? ? self.qnLineEdit.setStyleSheet("width:200px") ? ? ? ? keyLabel = QLabel("access_key") ? ? ? ? self.keyLineEdit = QLineEdit("") ? ? ? ? self.keyLineEdit.setStyleSheet("width:350px") ? ? ? ? secretLabel = QLabel("secret_key") ? ? ? ? self.secretLineEdit = QLineEdit() ? ? ? ? self.secretLineEdit.setStyleSheet("width:350px") ? ? ? ? self.secretLineEdit.setText('') ? ? ? ? bucketLabel = QLabel("bucket_name") ? ? ? ? self.bucketLineEdit = QLineEdit("") ? ? ? ? confirmButton = QPushButton("確定") ? ? ? ? formlayout.addRow(storageLabel, self.storageBox) ? ? ? ? formlayout.addRow(bucketLabel, self.bucketLineEdit) ? ? ? ? formlayout.addRow(self.endpointLabel, self.endpointLineEdit) ? ? ? ? formlayout.addRow(self.qntLabel, self.qnLineEdit) ? ? ? ? self.qntLabel.setVisible(False) ? ? ? ? self.qnLineEdit.setVisible(False) ? ? ? ? formlayout.addRow(keyLabel, self.keyLineEdit) ? ? ? ? formlayout.addRow(secretLabel, self.secretLineEdit) ? ? ? ? formlayout.addRow(confirmButton) ? ? ? ? self.storageBox.currentIndexChanged[int].connect(self.changeLabel) ? ? ? ? confirmButton.clicked.connect(self.confirm) ? ? ? ? self.setLayout(formlayout) ? ? ? ? self.setVisible(True)
子窗口是通過(guò)主窗口的菜單欄的設(shè)置菜單觸發(fā)
setupAction = QAction(QIcon('setup.png'), '設(shè)置', self) setupAction.setStatusTip('Exit application') setupAction.triggered.connect(self.openSecondFrame) def openSecondFrame(self): self.frame = secondFrame(self.db) self.frame.show()
監(jiān)聽(tīng)剪貼板的功能通過(guò)pyqt中的clipboard來(lái)監(jiān)聽(tīng)剪貼板的實(shí)現(xiàn)
self.clipboard.dataChanged.connect(self.paste) def paste(self): data = self.clipboard.mimeData() if data.hasImage(): print(data.formats()) pixmap = self.clipboard.pixmap() height = pixmap.height() width = pixmap.width() if height > 300 and width > 300: self.img.setPixmap(shrink_img(pixmap)) else: self.img.setPixmap(pixmap) fileName = 'ink_' + ''.join(str(uuid.uuid1()).split('-')) + '.png' self.clipboard.pixmap().save(fileName, 'PNG') urls = generate_url(self.upload(fileName)) print(urls) self.img.setScaledContents(True) self.markdown_url.setText(urls['markdown_url']) self.http_url.setText(urls['http_url']) pyperclip.copy(urls['markdown_url']) def shrink_img(pixmap): scale = 0.3 height = pixmap.height() width = pixmap.width() shrink_height = int(height * scale) shrink_width = int(width * scale) size = QSize(shrink_width, shrink_height) image = pixmap.toImage() return QPixmap.fromImage(image.scaled(size, Qt.IgnoreAspectRatio))
? def upload(self, filename): ? ? ? ? type, storage = self.get_storage_data() ? ? ? ? if type == '阿里OSS': ? ? ? ? ? ? url = upload2ali(storage, filename) ? ? ? ? ? ? return url ? ? def get_storage_data(self): ? ? ? ? self.db.execute(select_sql) ? ? ? ? data = self.db.fetchone() ? ? ? ? bucket_name = data[1] ? ? ? ? extra = data[2] ? ? ? ? key = data[3] ? ? ? ? secret = data[4] ? ? ? ? if data[0] == '阿里OSS': ? ? ? ? ? ? bucket = init_ali(bucket_name, extra, key, secret) ? ? ? ? ? ? return '阿里OSS', {'bucket': bucket, 'bucket_name': bucket_name, 'extra': extra}
當(dāng)圖片上傳成功后,默認(rèn)會(huì)生成markdown的圖片url,然后將這個(gè)url設(shè)置到剪貼板中,然后在marktext中只需要粘貼就能貼上圖床的圖片,因?yàn)閜yqt為了防止死循環(huán),不允許通過(guò)剪貼板設(shè)置內(nèi)容,所以我們可以通過(guò)pyperclip來(lái)設(shè)置剪貼板
pyperclip.copy(urls['markdown_url'])
目前只支持阿里oss的,七牛的也是一樣,只是上傳的sdk會(huì)有些不同,看官方文檔即可。程序我也在慢慢完善。本文更多的是記錄提供個(gè)思路
本文的所有圖片都是通過(guò)這個(gè)程序生成鏈接然后直接到文章中
以上就是Python實(shí)現(xiàn)截圖生成符合markdown的鏈接的詳細(xì)內(nèi)容,更多關(guān)于Python截圖生成鏈接的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- python使用html2text庫(kù)實(shí)現(xiàn)從HTML轉(zhuǎn)markdown的方法詳解
- Python3自動(dòng)生成MySQL數(shù)據(jù)字典的markdown文本的實(shí)現(xiàn)
- 獲取CSDN文章內(nèi)容并轉(zhuǎn)換為markdown文本的python
- Python實(shí)現(xiàn)Word文檔轉(zhuǎn)換Markdown的示例
- Python爬蟲(chóng)爬取微博熱搜保存為 Markdown 文件的源碼
- Python實(shí)戰(zhàn)之markdown轉(zhuǎn)pdf(包含公式轉(zhuǎn)換)
- python工具之清理 Markdown 中沒(méi)有引用的圖片
相關(guān)文章
使用selenium模擬登錄解決滑塊驗(yàn)證問(wèn)題的實(shí)現(xiàn)
這篇文章主要介紹了使用selenium模擬登錄解決滑塊驗(yàn)證問(wèn)題的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05python中zip()函數(shù)遍歷多個(gè)列表方法
在本篇文章里小編給大家整理的是一篇關(guān)于python中zip()函數(shù)遍歷多個(gè)列表方法,對(duì)此有興趣的朋友們可以學(xué)習(xí)下。2021-02-02macbook如何徹底刪除python的實(shí)現(xiàn)方法
本文主要介紹了macbook如何徹底刪除python的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07Python機(jī)器學(xué)習(xí)之決策樹(shù)
這篇文章主要介紹了Python機(jī)器學(xué)習(xí)之決策樹(shù),文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)python的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-04-04關(guān)于numpy數(shù)組中元素單個(gè)選取或部分選取問(wèn)題
這篇文章主要介紹了關(guān)于numpy數(shù)組中元素單個(gè)選取或部分選取問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07