123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622 |
- 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) {
-
- 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)
-
-
- }
-
- func findFileAndLine(forLocation point: NSPoint, inRect rect: NSRect, pageBounds bounds: NSRect, atPageIndex pageIndex: UInt) {
-
- }
-
- func findPageAndLocation(forLine line: Int, inFile file: String, options: Int) {
-
- }
- }
- 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
- }
- }
- }
-
- 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!
- }
- }
- 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 {
- }
-
- 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)
-
-
- let lineSortDescriptor = NSSortDescriptor(key: "line", ascending: true)
- let lineSortDescriptors = [lineSortDescriptor]
- for array in pages {
- (array as! NSMutableArray).sort(using: [NSSortDescriptor(key: "y", ascending: false), NSSortDescriptor(key: "x", ascending: true)])
- }
-
- }
-
- 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]()
-
-
- 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
- }
- }
- 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)
- 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
-
-
-
-
- if !rv {
- NSLog("SyncTeX was unable to find location and page.")
- }
-
- return rv
- }
- }
- 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
- }
-
- }
- 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()
-
- }
- }
|