Skip to content
This repository has been archived by the owner on Feb 22, 2022. It is now read-only.

Commit

Permalink
Add support for og:image icons.
Browse files Browse the repository at this point in the history
  • Loading branch information
leonbreedt committed Nov 3, 2018
1 parent 576b460 commit 833fb33
Show file tree
Hide file tree
Showing 8 changed files with 137 additions and 43 deletions.
10 changes: 5 additions & 5 deletions Example/Example/Base.lproj/Main.storyboard
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
<?xml version="1.0" encoding="UTF-8"?>
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="13771" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="BYZ-38-t0r">
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="14460.31" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="BYZ-38-t0r">
<device id="retina4_7" orientation="portrait">
<adaptation id="fullscreen"/>
</device>
<dependencies>
<deployment identifier="iOS"/>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="13772"/>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="14460.20"/>
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
</dependencies>
Expand All @@ -25,10 +25,10 @@
<nil key="highlightedColor"/>
</label>
<imageView userInteractionEnabled="NO" contentMode="scaleToFill" horizontalHuggingPriority="251" verticalHuggingPriority="251" translatesAutoresizingMaskIntoConstraints="NO" id="cSf-Nm-dlD">
<rect key="frame" x="171" y="362" width="32" height="32"/>
<rect key="frame" x="155.5" y="362" width="64" height="64"/>
<constraints>
<constraint firstAttribute="width" constant="32" id="JlN-aE-C65"/>
<constraint firstAttribute="height" constant="32" id="Ojd-ki-eD9"/>
<constraint firstAttribute="width" constant="64" id="JlN-aE-C65"/>
<constraint firstAttribute="height" constant="64" id="Ojd-ki-eD9"/>
</constraints>
</imageView>
</subviews>
Expand Down
11 changes: 8 additions & 3 deletions Example/Example/ViewController.swift
Original file line number Diff line number Diff line change
Expand Up @@ -21,21 +21,26 @@ import FavIcon
class ViewController: UIViewController {
@IBOutlet private weak var imageView: UIImageView!
@IBOutlet private weak var statusLabel: UILabel!

let url = "https://youtube.com"

override func viewDidLoad() {
super.viewDidLoad()

statusLabel.text = "Loading..."
do {
try FavIcon.downloadPreferred("https://apple.com") { result in
try FavIcon.downloadPreferred(url) { result in
if case let .success(image) = result {
self.statusLabel.text = "Loaded."
self.statusLabel.text = "Loaded (\(image.size.width)x\(image.size.height))"
self.imageView.image = image
} else if case let .failure(error) = result {
self.statusLabel.text = "Failed: \(error.localizedDescription)."
print("failed to download preferred favicon for \(self.url): \(error)")
}
}
} catch let error {
statusLabel.text = "Failed."
print("failed to download preferred favicon for apple.com: \(error)")
print("failed to download preferred favicon for \(self.url): \(error)")
}
}
}
Expand Down
17 changes: 11 additions & 6 deletions Sources/FavIcon/Detection.swift
Original file line number Diff line number Diff line change
Expand Up @@ -72,12 +72,17 @@ func detectHTMLHeadIcons(_ document: HTMLDocument, baseURL: URL) -> [Icon] {
}

for meta in document.query(xpath: "/html/head/meta") {
guard let name = meta.attributes["name"]?.lowercased() else { continue }
guard let content = meta.attributes["content"] else { continue }
guard let url = URL(string: content, relativeTo: baseURL) else { continue }
guard let size = microsoftSizeHints[name] else { continue }

icons.append(Icon(url: url, type: .microsoftPinnedSite, width: size.width, height: size.height))
if let name = meta.attributes["name"]?.lowercased() {
guard let content = meta.attributes["content"] else { continue }
guard let url = URL(string: content, relativeTo: baseURL) else { continue }
guard let size = microsoftSizeHints[name] else { continue }
icons.append(Icon(url: url, type: .microsoftPinnedSite, width: size.width, height: size.height))
} else if let property = meta.attributes["property"]?.lowercased() {
guard let content = meta.attributes["content"] else { continue }
guard property == "og:image" else { continue }
guard let imageURL = URL(string: content, relativeTo: baseURL) else { continue }
icons.append(Icon(url: imageURL, type: .openGraphImage))
}
}

return icons
Expand Down
64 changes: 54 additions & 10 deletions Sources/FavIcon/FavIcon.swift
Original file line number Diff line number Diff line change
Expand Up @@ -173,7 +173,7 @@ public final class FavIcon {
}
}

completion(downloadResults)
completion(sortResults(downloadResults))
}
}
}
Expand Down Expand Up @@ -217,17 +217,19 @@ public final class FavIcon {
height: Int? = nil,
completion: @escaping (IconDownloadResult) -> Void) throws {
scan(url) { icons in
guard let icon = chooseIcon(icons, width: width, height: height) else {
let sortedIcons = sortIcons(icons, preferredWidth: width, preferredHeight: height)
if sortedIcons.count == 0 {
DispatchQueue.main.async {
completion(.failure(error: IconError.noIconsDetected))
}
return
}

download([icon]) { results in
download(sortedIcons) { results in
let downloadResult: IconDownloadResult
if results.count > 0 {
downloadResult = results[0]
let sortedResults = sortResults(results, preferredWidth: width, preferredHeight: height)
if sortedResults.count > 0 {
downloadResult = sortedResults[0]
} else {
downloadResult = .failure(error: IconError.noIconsDetected)
}
Expand Down Expand Up @@ -257,12 +259,13 @@ public final class FavIcon {
guard let url = URL(string: url) else { throw IconError.invalidBaseURL }
try downloadPreferred(url, width: width, height: height, completion: completion)
}

static func chooseIcon(_ icons: [Icon], width: Int? = nil, height: Int? = nil) -> Icon? {
guard icons.count > 0 else { return nil }
static func sortIcons(_ icons: [Icon], preferredWidth: Int? = nil, preferredHeight: Int? = nil) -> [Icon] {
guard icons.count > 0 else { return [] }

let iconsInPreferredOrder = icons.sorted { left, right in
if let preferredWidth = width, let preferredHeight = height,
// Fall back to comparing dimensions.
if let preferredWidth = preferredWidth, let preferredHeight = preferredHeight,
let widthLeft = left.width, let heightLeft = left.height,
let widthRight = right.width, let heightRight = right.height {
// Which is closest to preferred size?
Expand All @@ -289,7 +292,43 @@ public final class FavIcon {
return left.type.rawValue < right.type.rawValue
}

return iconsInPreferredOrder.first!
return iconsInPreferredOrder
}

static func sortResults(_ results: [IconDownloadResult], preferredWidth: Int? = nil, preferredHeight: Int? = nil) -> [IconDownloadResult] {
guard results.count > 0 else { return [] }

let resultsInPreferredOrder = results.sorted { left, right in
switch (left, right) {
case (.success(let leftImage), .success(let rightImage)):
if let preferredWidth = preferredWidth, let preferredHeight = preferredHeight {
let widthLeft = leftImage.size.width
let heightLeft = leftImage.size.height
let widthRight = rightImage.size.width
let heightRight = rightImage.size.height

// Which is closest to preferred size?
let deltaA = abs(widthLeft - CGFloat(preferredWidth)) * abs(heightLeft - CGFloat(preferredHeight))
let deltaB = abs(widthRight - CGFloat(preferredWidth)) * abs(heightRight - CGFloat(preferredHeight))

return deltaA < deltaB
} else {
// Fall back to largest image.
return leftImage.area > rightImage.area
}
case (.success, .failure):
// An image is better than none.
return true
case (.failure, .success):
// An image is better than none.
return false
default:
return true
}
}

return resultsInPreferredOrder

}
}

Expand All @@ -302,3 +341,8 @@ extension Icon {
}
}

extension ImageType {
var area: CGFloat {
return size.width * size.height
}
}
2 changes: 2 additions & 0 deletions Sources/FavIcon/IconType.swift
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,8 @@ public enum IconType: UInt {
case microsoftPinnedSite
/// An icon defined in a Web Application Manifest JSON file, mainly Android/Chrome.
case webAppManifest
/// An icon defined by the og:image meta property.
case openGraphImage
}


9 changes: 7 additions & 2 deletions Tests/FavIconTests/DetectionTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,7 @@ class DetectionTests: XCTestCase {
let document = HTMLDocument(string: html)
let icons = detectHTMLHeadIcons(document, baseURL: URL(string: "https://localhost")!)

XCTAssertEqual(20, icons.count)
XCTAssertEqual(21, icons.count)

XCTAssertEqual("https://localhost/shortcut.ico", icons[0].url.absoluteString)
XCTAssertEqual(IconType.shortcut.rawValue, icons[0].type.rawValue)
Expand Down Expand Up @@ -197,6 +197,11 @@ class DetectionTests: XCTestCase {
XCTAssertEqual(IconType.microsoftPinnedSite.rawValue, icons[19].type.rawValue)
XCTAssertEqual(310, icons[19].width!)
XCTAssertEqual(310, icons[19].height!)

XCTAssertEqual("https://www.facebook.com/images/fb_icon_325x325.png", icons[20].url.absoluteString)
XCTAssertEqual(IconType.openGraphImage.rawValue, icons[20].type.rawValue)
XCTAssertNil(icons[20].width)
XCTAssertNil(icons[20].height)
}

func testIssue6_ContentTypeWithEmptyComponent() {
Expand All @@ -221,7 +226,7 @@ class DetectionTests: XCTestCase {
XCTAssertEqual(0, document.query(xpath: "/BrowserConfig").count)
XCTAssertEqual(0, document.query(xpath: "").count)
}

private func pathForTestBundleResource(fileName: String) -> String {
let testBundle = Bundle(for: FavIconTests.self)
return testBundle.path(forResource: fileName, ofType: "")!
Expand Down
66 changes: 49 additions & 17 deletions Tests/FavIconTests/FavIconTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -37,9 +37,37 @@ class FavIconTests: XCTestCase {
wait(for: [completed], timeout: 15)

XCTAssertNotNil(actualIcons)
XCTAssertEqual(1, actualIcons.count)
XCTAssertEqual(2, actualIcons.count)
XCTAssertEqual(URL(string: "https://apple.com/favicon.ico")!, actualIcons[0].url)
}

func testIssue24_LowResIcons() {
let url = "https://www.facebook.com"
var actualResult: IconDownloadResult!

let completed = expectation(description: "download: \(url)")
do {
try FavIcon.downloadPreferred(url) { result in
actualResult = result
completed.fulfill()
}
} catch let error {
XCTFail("failed to download icons: \(error)")
}
wait(for: [completed], timeout: 15)

XCTAssertNotNil(actualResult)

switch actualResult! {
case .success(let image):
XCTAssertEqual(325.0, image.size.width)
XCTAssertEqual(325.0, image.size.height)
break
case .failure(let error):
XCTFail("unexpected error returned for download: \(error)")
break
}
}

func testDownloading() {
let url = "https://apple.com"
Expand All @@ -56,12 +84,12 @@ class FavIconTests: XCTestCase {
}
wait(for: [completed], timeout: 15)

XCTAssertEqual(1, actualResults.count)
XCTAssertEqual(2, actualResults.count)

switch actualResults[0] {
case .success(let image):
XCTAssertEqual(64, image.size.width)
XCTAssertEqual(64, image.size.height)
XCTAssertEqual(1200, image.size.width)
XCTAssertEqual(630, image.size.height)
break
case .failure(let error):
XCTFail("unexpected error returned for download: \(error)")
Expand All @@ -82,32 +110,36 @@ class FavIconTests: XCTestCase {
Icon(url: URL(string: "https://google.com/favicon.ico")!, type: .shortcut)
]

var icon = FavIcon.chooseIcon(mixedIcons, width: 50, height: 50)
var sortedIcons = FavIcon.sortIcons(mixedIcons, preferredWidth: 50, preferredHeight: 50)
var icon = sortedIcons[0]

XCTAssertNotNil(icon)
XCTAssertEqual(64, icon!.width)
XCTAssertEqual(64, icon!.height)
XCTAssertEqual(64, icon.width)
XCTAssertEqual(64, icon.height)

icon = FavIcon.chooseIcon(mixedIcons, width: 28, height: 28)
sortedIcons = FavIcon.sortIcons(mixedIcons, preferredWidth: 28, preferredHeight: 28)
icon = sortedIcons[0]

XCTAssertNotNil(icon)
XCTAssertEqual(32, icon!.width)
XCTAssertEqual(32, icon!.height)
XCTAssertEqual(32, icon.width)
XCTAssertEqual(32, icon.height)

icon = FavIcon.chooseIcon(mixedIcons)
sortedIcons = FavIcon.sortIcons(mixedIcons)
icon = sortedIcons[0]

XCTAssertNotNil(icon)
XCTAssertEqual(144, icon!.width)
XCTAssertEqual(144, icon!.height)
XCTAssertEqual(144, icon.width)
XCTAssertEqual(144, icon.height)

icon = FavIcon.chooseIcon(noSizeIcons)
sortedIcons = FavIcon.sortIcons(noSizeIcons)
icon = sortedIcons[0]

XCTAssertNotNil(icon)
XCTAssertEqual(IconType.shortcut.rawValue, icon!.type.rawValue)
XCTAssertEqual(IconType.shortcut.rawValue, icon.type.rawValue)

icon = FavIcon.chooseIcon([])
sortedIcons = FavIcon.sortIcons([])

XCTAssertNil(icon)
XCTAssertEqual(0, sortedIcons.count)
}
}

1 change: 1 addition & 0 deletions Tests/FavIconTests/HTML.html
Original file line number Diff line number Diff line change
Expand Up @@ -25,5 +25,6 @@
<link rel="shortcut icon"
type="image/png"
href="">
<meta property="og:image" content="https://www.facebook.com/images/fb_icon_325x325.png">
</head>
</html>

0 comments on commit 833fb33

Please sign in to comment.