123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591 |
- //
- // KMPDFSynchronizer.swift
- // PDF Reader Pro
- //
- // Created by lizhe on 2024/2/28.
- //
- import Cocoa
- enum KMPDFSynchronizerType: Int {
- case defaultOptions = 0
- case showReadingBarMask = 1
- case flippedMask = 2
- }
- let PDFSYNC_TO_PDF: (CGFloat) -> CGFloat = { coord in
- return CGFloat(coord) / 65536.0
- }
- let SKPDFSynchronizerPdfsyncExtension = "pdfsync"
- var SKPDFSynchronizerTexExtensions: [String] = ["tex", "ltx", "latex", "ctx", "lyx", "rnw"]
- let STACK_BUFFER_SIZE = 256
- func caseInsensitiveStringEqual(_ item1: UnsafeRawPointer, _ item2: UnsafeRawPointer, _ size: ((UnsafeRawPointer) -> Int)?) -> Bool {
- return CFStringCompare(item1 as! CFString, item2 as! CFString, CFStringCompareFlags.compareCaseInsensitive) == CFComparisonResult.compareEqualTo
- }
-
- protocol KMPDFSynchronizerDelegate: AnyObject {
- func synchronizer(_ synchronizer: KMPDFSynchronizer, foundLine line: Int, inFile file: String)
- func synchronizer(_ synchronizer: KMPDFSynchronizer, foundLocation point: NSPoint, atPageIndex pageIndex: UInt, options: Int)
- }
- class KMPDFSynchronizer: NSObject {
- weak var delegate: KMPDFSynchronizerDelegate?
- var fileName: String = ""
- var shouldKeepRunning: Bool {
- OSMemoryBarrier()
- return shouldKeepRunningFlag == 1
- }
-
- private var queue: DispatchQueue = DispatchQueue(label: "net.sourceforge.queue.KMPDFSynchronizer")
- private var lockQueue: DispatchQueue = DispatchQueue(label: "net.sourceforge.lockQueue.KMPDFSynchronizer")
- private var syncFileName: String = "" {
- didSet {
- if syncFileName.count == 0 {
- lastModDate = nil
- } else {
- do {
- let attributes = try FileManager.default.attributesOfItem(atPath: syncFileName)
- lastModDate = attributes[.modificationDate] as? Date
- } catch {
-
- }
- }
- }
- }
- private var lastModDate: Date?
- private var isPdfsync: Bool = true
- private var fileManager: FileManager = FileManager()
- private var pages: [Any] = []
- private var lines: NSMapTable<AnyObject, AnyObject>?
- private var filenames: NSMapTable<AnyObject, AnyObject>?
- private var scanner: synctex_scanner_p?
- private var shouldKeepRunningFlag: Int32 = 1
-
- func setFileName(_ newFileName: String) {
- // We compare filenames in canonical form throughout, so we need to make sure fileName also is in canonical form
- if let canonicalFileName = NSURL(fileURLWithPath: newFileName).resolvingSymlinksInPath?.standardizedFileURL.path {
- DispatchQueue.global().async { [weak self] in
- guard let self = self else { return }
- self.lockQueue.sync {
- if self.fileName != canonicalFileName {
- if self.fileName != newFileName {
- self.syncFileName = ""
- self.lastModDate = nil
- }
- self.fileName = canonicalFileName
- }
- }
- }
- }
- }
-
- func terminate() {
- delegate = nil
- let originalValue = OSAtomicCompareAndSwap32Barrier(1, 0, &shouldKeepRunningFlag)
- // originalValue 为原来的值,如果原来的值是 1,表示已经交换成功,如果是 0,表示没有成功交换
- // 如果需要对返回值进行处理,可以在这里添加逻辑
- }
-
- func findFileAndLine(forLocation point: NSPoint, inRect rect: NSRect, pageBounds bounds: NSRect, atPageIndex pageIndex: UInt) {
- // Implement this method
- }
-
- func findPageAndLocation(forLine line: Int, inFile file: String, options: Int) {
- // Implement this method
- }
- }
- //MARK: Support
- extension KMPDFSynchronizer {
- func sourceFile(forFileName file: String, isTeX: Bool, removeQuotes: Bool) -> String {
- var fileName = file
- if removeQuotes && fileName.count > 2 && fileName.first == "\"" && fileName.last == "\"" {
- fileName = String(fileName.dropFirst().dropLast())
- }
- if !(fileName as NSString).isAbsolutePath {
- fileName = (self.fileName as NSString).deletingLastPathComponent + "/" + fileName
- }
- if isTeX && !FileManager.default.fileExists(atPath: fileName) && !SKPDFSynchronizerTexExtensions.contains((fileName as NSString).pathExtension.lowercased()) {
- for ext in SKPDFSynchronizerTexExtensions {
- let tryFile = fileName + "." + ext
- if FileManager.default.fileExists(atPath: tryFile) {
- fileName = tryFile
- break
- }
- }
- }
- // Swift's `standardizedFileURL` property does both `resolvingSymlinksInPath` and `standardizingPath`
- return URL(fileURLWithPath: fileName).standardizedFileURL.path
- }
-
- func defaultSourceFile() -> String {
- let file = (self.fileName as NSString).deletingPathExtension
- for `extension` in SKPDFSynchronizerTexExtensions {
- let tryFile = file + "." + `extension`
- if FileManager.default.fileExists(atPath: tryFile) {
- return tryFile
- }
- }
- return file + "." + SKPDFSynchronizerTexExtensions.first!
- }
- }
- //MARK: PDFSync
- extension KMPDFSynchronizer{
- func recordForIndex(_ records: NSMapTable<AnyObject, AnyObject>, _ recordIndex: Int) -> KMPDFSyncRecord {
- if let record = records.object(forKey: recordIndex as AnyObject) as? KMPDFSyncRecord {
- return record
- } else {
- let record = KMPDFSyncRecord(recordIndex: recordIndex)
- records.setObject(record, forKey: recordIndex as AnyObject)
- return record
- }
- }
-
- func loadPdfsyncFile(_ theFileName: String) -> Bool {
- pages.removeAll()
-
- if lines != nil {
- lines?.removeAllObjects()
- } else {
- // let keyPointerFunctions = NSPointerFunctions(options: [.strongMemory, .objectPersonality])
- // keyPointerFunctions.isEqualFunction = { (a, b) in
- // caseInsensitiveStringEqual(a, b) { <#UnsafeRawPointer#> in
- // <#code#>
- // }
- // guard let strA = a as? String, let strB = b as? String else { return false }
- // return strA.caseInsensitiveCompare(strB) == .orderedSame
- // }
- // keyPointerFunctions.hashFunction = { (ptr) in
- // guard let str = ptr as? String else { return 0 }
- // return str.hash
- // }
- // let valuePointerFunctions = NSPointerFunctions(options: [.strongMemory, .objectPersonality])
- // lines = NSMapTable(keyPointerFunctions: keyPointerFunctions, valuePointerFunctions: valuePointerFunctions,capacity: 0)
- }
-
- syncFileName = theFileName
- isPdfsync = true
-
- guard let pdfsyncString = try? String(contentsOfFile: theFileName, encoding: .utf8) else { return false }
-
- var rv = false
-
- let records = NSMapTable<NSNumber, AnyObject>.strongToStrongObjects()
- let files = NSMutableArray()
- var recordIndex = 0, line = 0, pageIndex = 0
- var x = 0.0, y = 0.0
- var record: KMPDFSyncRecord?
- var array: NSMutableArray?
- var ch: unichar = 0
- let sc = Scanner(string: pdfsyncString)
- let newlines = CharacterSet.newlines
-
- sc.charactersToBeSkipped = CharacterSet.whitespaces
-
- var file: NSString?
- var tryFile: String
-
- let scanString = sc.scanUpToCharacters(from: newlines)
- let scanCharactersString = sc.scanCharacters(from: newlines)
- if scanString?.count != 0 && scanCharactersString?.count != 0 {
- file = sourceFile(forFileName: scanString! as String, isTeX: true, removeQuotes: true) as NSString
- files.add(file!)
-
- array = NSMutableArray()
- lines!.setObject(array!, forKey: file!)
-
- sc.scanString("version", into: nil)
- sc.scanInt(nil)
- sc.scanCharacters(from: newlines, into: nil)
-
- // while sc.shouldKeepRunning() && sc.scanCharacter(&ch) {
- // switch ch {
- // case "l":
- // if sc.scanInt(&recordIndex) && sc.scanInt(&line) {
- // // we ignore the column
- // sc.scanInt(nil)
- // record = recordForIndex(records, recordIndex)
- // record!.file = file! as String
- // record!.line = line
- // lines!.object(forKey: file! as NSString)!.add(record!)
- // }
- // case "p":
- // // we ignore * and + modifiers
- // if !sc.scanString("*", into: nil) {
- // sc.scanString("+", into: nil)
- // }
- // if sc.scanInt(&recordIndex) && sc.scanDouble(&x) && sc.scanDouble(&y) {
- // record = recordForIndex(records, recordIndex)
- // record!.pageIndex = pages.count - 1
- // record!.point = NSMakePoint(CGFloat(PDFSYNC_TO_PDF(x) + pdfOffset.x), CGFloat(PDFSYNC_TO_PDF(y) + pdfOffset.y))
- // (pages.lastObject as! NSMutableArray).add(record!)
- // }
- // case "s":
- // // start of a new page, the scanned integer should always equal [pages count]+1
- // var tempPageIndex = 0
- // if !sc.scanInt(&tempPageIndex) {
- // pageIndex = pages.count + 1
- // } else {
- // pageIndex = tempPageIndex
- // }
- // while pageIndex > pages.count {
- // array = NSMutableArray()
- // pages.add(array!)
- // }
- // case "(":
- // // start of a new source file
- // var tempFile: NSString?
- // if sc.scanUpToCharacters(from: newlines, into: &tempFile as? String) {
- // file = sourceFile(forFileName: tempFile! as String, isTeX: true, removeQuotes: true) as NSString
- // files.add(file!)
- // if lines!.object(forKey: file!) == nil {
- // array = NSMutableArray()
- // lines!.setObject(array!, forKey: file!)
- // }
- // }
- // case ")":
- // // closing of a source file
- // if files.count > 0 {
- // files.removeLastObject()
- // file = files.lastObject as? NSString
- // }
- // default:
- // // shouldn't reach
- // break
- // }
- //
- // sc.scanUpToCharacters(from: newlines, into: nil)
- // sc.scanCharacters(from: newlines, into: nil)
- // }
-
- let lineSortDescriptor = NSSortDescriptor(key: "line", ascending: true)
- let lineSortDescriptors = [lineSortDescriptor]
- // for array in lines!.objectEnumerator() {
- // (array as! NSMutableArray).sort(using: lineSortDescriptors)
- // }
- for array in pages {
- (array as! NSMutableArray).sort(using: [NSSortDescriptor(key: "y", ascending: false), NSSortDescriptor(key: "x", ascending: true)])
- }
-
- // rv = sc.shouldKeepRunning
- }
-
- return rv
- }
-
- func pdfsyncFindFileLine(_ linePtr: inout Int, file filePtr: inout String?, forLocation point: NSPoint, inRect rect: NSRect, pageBounds bounds: NSRect, atPageIndex pageIndex: Int) -> Bool {
- var rv = false
- if pageIndex < pages.count {
-
- var record: KMPDFSyncRecord?
- var beforeRecord: KMPDFSyncRecord?
- var afterRecord: KMPDFSyncRecord?
- var atRecords = [Double: KMPDFSyncRecord]()
-
- // for case let tempRecord as KMPDFSyncRecord in pages[pageIndex] {
- // if tempRecord.line == 0 {
- // continue
- // }
- // let p = tempRecord.point
- // if p.y > NSMaxY(rect) {
- // beforeRecord = tempRecord
- // } else if p.y < NSMinY(rect) {
- // afterRecord = tempRecord
- // break
- // } else if p.x < NSMinX(rect) {
- // beforeRecord = tempRecord
- // } else if p.x > NSMaxX(rect) {
- // afterRecord = tempRecord
- // break
- // } else {
- // atRecords[abs(p.x - point.x)] = tempRecord
- // }
- // }
-
- var nearestRecord: KMPDFSyncRecord?
- if !atRecords.isEmpty {
- let nearest = atRecords.keys.sorted()[0]
- nearestRecord = atRecords[nearest]
- } else if let beforeRecord = beforeRecord, let afterRecord = afterRecord {
- let beforePoint = beforeRecord.point
- let afterPoint = afterRecord.point
- if beforePoint.y - point.y < point.y - afterPoint.y {
- nearestRecord = beforeRecord
- } else if beforePoint.y - point.y > point.y - afterPoint.y {
- nearestRecord = afterRecord
- } else if beforePoint.x - point.x < point.x - afterPoint.x {
- nearestRecord = beforeRecord
- } else if beforePoint.x - point.x > point.x - afterPoint.x {
- nearestRecord = afterRecord
- } else {
- nearestRecord = beforeRecord
- }
- } else if let beforeRecord = beforeRecord {
- nearestRecord = beforeRecord
- } else if let afterRecord = afterRecord {
- nearestRecord = afterRecord
- }
-
- if let record = nearestRecord {
- linePtr = record.line
- filePtr = record.file
- rv = true
- }
- }
- if !rv {
- print("PDFSync was unable to find file and line.")
- }
- return rv
- }
-
- func pdfsyncFindPage(_ pageIndexPtr: inout Int, location pointPtr: inout NSPoint, forLine line: Int, inFile file: String) -> Bool {
- var rv = false
- if let theLines = lines!.object(forKey: file as NSString) as? [KMPDFSyncRecord] {
-
- var record: KMPDFSyncRecord?
- var beforeRecord: KMPDFSyncRecord?
- var afterRecord: KMPDFSyncRecord?
- var atRecord: KMPDFSyncRecord?
-
- for tempRecord in theLines {
- if tempRecord.pageIndex == NSNotFound {
- continue
- }
- let l = tempRecord.line
- if l < line {
- beforeRecord = tempRecord
- } else if l > line {
- afterRecord = tempRecord
- break
- } else {
- atRecord = tempRecord
- break
- }
- }
-
- if let atRecord = atRecord {
- record = atRecord
- } else if let beforeRecord = beforeRecord, let afterRecord = afterRecord {
- let beforeLine = beforeRecord.line
- let afterLine = afterRecord.line
- if beforeLine - line > line - afterLine {
- record = afterRecord
- } else {
- record = beforeRecord
- }
- } else if let beforeRecord = beforeRecord {
- record = beforeRecord
- } else if let afterRecord = afterRecord {
- record = afterRecord
- }
-
- if let record = record {
- pageIndexPtr = record.pageIndex
- pointPtr = record.point
- rv = true
- }
- }
- if !rv {
- print("PDFSync was unable to find location and page.")
- }
- return rv
- }
- }
- //MARK: SyncTeX
- extension KMPDFSynchronizer {
- func loadSynctexFile(forFile theFileName: String) -> Bool {
- var rv = false
- if let scanner = scanner {
- synctex_scanner_free(scanner)
- }
- scanner = synctex_scanner_new_with_output_file(theFileName.cString(using: .utf8), nil, 1)
- // if let scanner = scanner {
- // let fileRep = synctex_scanner_get_synctex(scanner)
- // syncFileName = sourceFile(forFileName: String(cString: fileRep!), isTeX: false, removeQuotes: false)
- // if let filenames = filenames {
- // NSResetMapTable(filenames)
- // } else {
- // let keyPointerFunctions = NSPointerFunctions(options: [.strongMemory, .objectPersonality])
- // keyPointerFunctions.isEqualFunction = { (a, b) in
- // guard let strA = a as? String, let strB = b as? String else { return false }
- // return strA.caseInsensitiveCompare(strB) == .orderedSame
- // }
- // keyPointerFunctions.hashFunction = { (ptr) in
- // guard let str = ptr as? String else { return 0 }
- // return str.hash
- // }
- // let valuePointerFunctions = NSPointerFunctions(options: [.mallocMemory, .cstringPersonality, .copyIn])
- // filenames = NSMapTable(keyOptions: keyPointerFunctions, valueOptions: valuePointerFunctions)
- // }
- // var node = synctex_scanner_input(scanner)
- // repeat {
- // if let fileRep = synctex_scanner_get_name(scanner, synctex_node_tag(node)) {
- // filenames!.setObject(String(cString: fileRep), forKey: sourceFile(forFileName: String(cString: fileRep), isTeX: true, removeQuotes: false))
- // }
- // } while ((node = synctex_node_next(node)) != nil)
- // isPdfsync = false
- // rv = shouldKeepRunning
- // }
- return rv
- }
-
- func synctexFindFileLine(_ linePtr: inout Int, file filePtr: inout String?, forLocation point: NSPoint, inRect rect: NSRect, pageBounds bounds: NSRect, atPageIndex pageIndex: Int) -> Bool {
- var rv = false
- if synctex_edit_query(scanner, Int32(pageIndex + 1), Float(Double(point.x)), Float(Double(NSMaxY(bounds) - point.y))) > 0 {
- var node: synctex_node_p?
- var file: UnsafePointer<Int8>?
- while rv == false && (node = synctex_scanner_next_result(scanner)) != nil {
- if let tempFile = synctex_scanner_get_name(scanner, synctex_node_tag(node)) {
- linePtr = Int(max(synctex_node_line(node), 1) - 1)
- filePtr = sourceFile(forFileName: String(cString: tempFile), isTeX: true, removeQuotes: false)
- rv = true
- }
- }
- }
- if rv == false {
- NSLog("SyncTeX was unable to find file and line.")
- }
- return rv
- }
-
- func synctexFindPage(pageIndexPtr: UnsafeMutablePointer<UInt>, pointPtr: UnsafeMutablePointer<CGPoint>, forLine line: Int, inFile file: String) -> Bool {
-
- guard let filenames = filenames else { return false }
-
- var rv = false
- // var filename: UnsafeMutableRawPointer = NSMapGet(filenames, file) ?? UnsafeMutableRawPointer(<#Builtin.RawPointer#>)
- //?? NSMapGet(filenames, (file as NSString).resolvingSymlinksInPath.standardizingPath)
-
- // if filename == nil {
- // for fn in filenames.allKeys {
- // if let fnString = fn as? String, fnString.lastPathComponent.caseInsensitiveCompare(file.lastPathComponent) == .orderedSame {
- // filename = NSMapGet(filenames, fn) as? UnsafePointer<Int8>
- // break
- // }
- // }
- //
- // if filename == nil {
- // filename = (file as NSString).lastPathComponent.utf8String
- // }
- // }
-
- // if synctex_display_query(scanner, filename, Int32(line) + 1, 0, -1) > 0 {
- // if let node = synctex_scanner_next_result(scanner) {
- // let page = UInt(synctex_node_page(node))
- // pageIndexPtr.pointee = max(page, 1) - 1
- // pointPtr.pointee = CGPoint(x: CGFloat(synctex_node_visible_h(node)), y: CGFloat(synctex_node_visible_v(node)))
- // rv = true
- // }
- // }
-
- if !rv {
- NSLog("SyncTeX was unable to find location and page.")
- }
-
- return rv
- }
- }
- //MARK: Generic
- extension KMPDFSynchronizer {
- func loadSyncFileIfNeeded() -> Bool {
- let theFileName = fileName
- var rv = false
-
- if theFileName.count != 0 {
- var theSyncFileName = self.syncFileName
-
- var modDate: NSDate?
- if theSyncFileName.count != 0 && fileManager.fileExists(atPath: theSyncFileName) {
- do {
- let attributes = try FileManager.default.attributesOfItem(atPath: theFileName)
- modDate = attributes[.modificationDate] as? NSDate
- } catch {
-
- }
- let currentModDate = self.lastModDate
-
- if (currentModDate != nil) && modDate?.compare(currentModDate!) != ComparisonResult.orderedDescending {
- rv = true
- } else if (isPdfsync) {
- rv = self.loadPdfsyncFile(theSyncFileName)
- } else {
- rv = self.loadSynctexFile(forFile: theFileName)
- }
- } else {
- rv = self.loadSynctexFile(forFile: theFileName)
- if rv == false {
- theSyncFileName = (theFileName as NSString).deletingPathExtension.stringByAppendingPathExtension("pdfsync")
- if fileManager.fileExists(atPath: theSyncFileName) {
- rv = self.loadPdfsyncFile(theSyncFileName)
- }
- }
- }
- }
-
- if rv == false {
- print("Unable to find or load synctex or pdfsync file.")
- }
- return rv
- }
-
- }
- //MARK: Finding API
- extension KMPDFSynchronizer {
- func findFileAndLine(forLocation point: NSPoint, inRect rect: NSRect, pageBounds bounds: NSRect, atPageIndex pageIndex: Int) {
- queue.async {
- guard self.shouldKeepRunning, self.loadSyncFileIfNeeded() else { return }
-
- var foundLine = 0
- var foundFile: String?
- var success = false
-
- if self.isPdfsync {
- success = self.pdfsyncFindFileLine(&foundLine, file: &foundFile, forLocation: point, inRect: rect, pageBounds: bounds, atPageIndex: pageIndex)
- } else {
- success = self.synctexFindFileLine(&foundLine, file: &foundFile, forLocation: point, inRect: rect, pageBounds: bounds, atPageIndex: pageIndex)
- }
-
- if success, self.shouldKeepRunning {
- DispatchQueue.main.async {
- self.delegate?.synchronizer(self, foundLine: foundLine, inFile: foundFile ?? "")
- }
- }
- }
- }
-
- func findPageAndLocation(forLine line: Int, inFile file: String?, options: Int) {
- let fixedFile = file ?? defaultSourceFile()
-
- // queue.async {
- // guard let fixedFile = fixedFile, self.shouldKeepRunning, self.loadSyncFileIfNeeded() else { return }
- //
- // var foundPageIndex = NSNotFound
- // var foundPoint = NSZeroPoint
- // var foundOptions = options
- //
- // if self.isPdfsync {
- // self.pdfsyncFindPage(&foundPageIndex, location: &foundPoint, forLine: line, inFile: fixedFile)
- // } else {
- // self.synctexFindPage(&foundPageIndex, location: &foundPoint, forLine: line, inFile: fixedFile)
- // }
- //
- // if self.shouldKeepRunning() {
- // if self.isPdfsync {
- // foundOptions &= ~KMPDFSynchronizerType.flippedMask
- // } else {
- // foundOptions |= KMPDFSynchronizerType.flippedMask
- // }
- // DispatchQueue.main.async {
- // self.delegate?.synchronizer(self, foundLocation: foundPoint, atPageIndex: foundPageIndex, options: foundOptions)
- // }
- // }
- // }
- }
- }
|