背景介紹
圖靈教育推出的限量款編程日歷2018,因?yàn)楹?jiǎn)約大氣的設(shè)計(jì)和每周一個(gè)編程語(yǔ)言的介紹,在程序員中廣受歡迎。
圖靈教育推出的編程日歷實(shí)體版
不幸的是由于限量1000款,除去贈(zèng)品的300多套,真正在售的只有600多套,很快就被搶購(gòu)一空。值得欣慰的是,前天下午作者將pdf版本的日歷公開下載。
在簡(jiǎn)書中也無(wú)意間看到有人用python片段將壁紙與當(dāng)周的日歷進(jìn)行了融合,這個(gè)想法讓我受到了啟發(fā),從該文章下面的評(píng)論看到很多用戶(特別是mac用戶)反映在 macOS 下,Wand 庫(kù)有點(diǎn)小問(wèn)題,GitHub 有人提到了這個(gè) issue。
我一直在使用的一款軟件Blotter,吸附在桌面上的日歷和待辦事項(xiàng),于是就萌生了一個(gè)將該pdf吸附在桌面上,并根據(jù)當(dāng)前日期展示相應(yīng)日期的應(yīng)用,于是我花半天做了TuringCalendar這款應(yīng)用,開源地址。歡迎有能力的開發(fā)者改進(jìn)這款應(yīng)用。
Blotter截圖
TuringCalendar的現(xiàn)狀
由于時(shí)間倉(cāng)促,這款軟件有一些缺點(diǎn)需要后續(xù)解決。
- 現(xiàn)在的默認(rèn)將日歷頁(yè)放置在右上角,因?yàn)樽笊辖潜籅lotter占了,后面需要做成可配置的。
- 現(xiàn)在是白底的,在淺色背景的桌面上會(huì)比較美觀,在深色背景中就不那么美觀了。關(guān)于這點(diǎn)我在簡(jiǎn)書上問(wèn)過(guò)python代碼的作者,他告訴我用通道混合來(lái)解決,目前尚在研究中。
TuringCalendar截圖
TuringCalendar開發(fā)過(guò)程
將窗口固定在桌面上
macOS管理窗口的類是NSWindow,將窗口固定在桌面上是通過(guò)繼承該類,并override 其中的某些方法做到的。
override init(contentRect: NSRect, styleMask style: NSWindow.StyleMask, backing backingStoreType: NSWindow.BackingStoreType, defer flag: Bool) {
super.init(contentRect: contentRect, styleMask: style, backing: backingStoreType, defer: flag)
self.level = NSWindow.Level(rawValue: NSWindow.Level.RawValue(CGWindowLevelForKey(CGWindowLevelKey.desktopWindow) - 1))
self.collectionBehavior = (NSWindow.CollectionBehavior(rawValue: NSWindow.CollectionBehavior.RawValue(UInt8(NSWindow.CollectionBehavior.canJoinAllSpaces.rawValue) |
UInt8(NSWindow.CollectionBehavior.stationary.rawValue) |
UInt8(NSWindow.CollectionBehavior.ignoresCycle.rawValue)))
)
self.backgroundColor = NSColor.clear
self.isOpaque = false
}
override var canBecomeMain: Bool{
return false;
}
override var canBecomeKey: Bool{
return false;
}
init方法中,指定了窗口的層級(jí)為desktopWindow-1,并且指定了窗口的背景色和一些操作的影響,主要是expose操作的時(shí)候,該窗口不應(yīng)該和其他普通窗口一樣,收縮起來(lái)。同時(shí)override相應(yīng)方法,讓該窗口不可以成為Main窗口和Key窗口。
讀取pdf
讀取pdf是通過(guò)PDFView完成的,需要導(dǎo)入Quartz庫(kù)。在StoryBoard中也有相關(guān)的組件,可以查到日歷每頁(yè)的寬高,在StoryBoard中指定為固定寬高即可。
@IBOutlet var calendarViewer: PDFView!
override func viewDidLoad() {
super.viewDidLoad()
let url = Bundle.main.url(forResource: "calendar", withExtension: "pdf")
let pdf = PDFDocument(url: url!)
let today = GetWeekByDate(date: Date())
calendarViewer.document = pdf
calendarViewer.go(to: (pdf?.page(at: today-1))!)
// Do any additional setup after loading the view.
}
這里發(fā)現(xiàn)一個(gè)坑,PDFView是會(huì)響應(yīng)鼠標(biāo)事件的,上下滑會(huì)在頁(yè)與頁(yè)之間切換,由于PDFView是NSView的子類,因此可以override hitTest方法,讓PDFView不響應(yīng)相關(guān)事件,使用了extension關(guān)鍵字。
extension PDFView{
open override func hitTest(_ point: NSPoint) -> NSView? {
return nil
}
}
得到今天是今年的第幾周
我將原作者提供的pdf文件進(jìn)行了截取,只保留了我們需要的53個(gè)周的數(shù)據(jù)。通過(guò)下面的方法獲取到當(dāng)天是2018年的第幾周,然后讓PDFView跳到相應(yīng)的頁(yè)面。
func GetWeekByDate(date:Date) ->Int{
guard let calendar = NSCalendar(identifier: NSCalendar.Identifier.gregorian) else {
return 0
}
let components = calendar.components([.weekOfYear,.weekOfMonth,.weekday,.weekdayOrdinal], from: date)
return components.weekOfYear!;
}
將窗口固定在右上角
控制窗口這件事是由windowController完成的,獲取到相應(yīng)的window,并調(diào)用setFrameOrigin方法指定窗口的初始x,y坐標(biāo)即可。需要注意的是屏幕的坐標(biāo)左下角是(0,0)。
override func windowDidLoad() {
super.windowDidLoad()
if let window = window, let screen = window.screen {
let screenRect = screen.visibleFrame
let offsetFromLeft = CGFloat(screenRect.maxX - window.frame.width)
let offsetFromTop = CGFloat(0)
let offsetFromBottom = screenRect.maxY - window.frame.height - offsetFromTop
window.setFrameOrigin(NSPoint(x: offsetFromLeft, y: offsetFromBottom))
}
}
與Python版本相比的優(yōu)點(diǎn)
相比于python版,TuringCalendar也有自己的優(yōu)勢(shì),那就是不需要手動(dòng)的去生成壁紙,而且每周要定時(shí)更換;環(huán)境的配置可能有一些坑,很多人都在評(píng)論里說(shuō)配置沒有成功。
最后,歡迎有能力的開發(fā)者改進(jìn)這款應(yīng)用。