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-05
python中zip()函數(shù)遍歷多個(gè)列表方法
在本篇文章里小編給大家整理的是一篇關(guān)于python中zip()函數(shù)遍歷多個(gè)列表方法,對(duì)此有興趣的朋友們可以學(xué)習(xí)下。2021-02-02
macbook如何徹底刪除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-07
Python機(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

