You can find the Safari App Extension I describe in this blog post on the App Store — if you find this blog post valuable, please consider downloading a copy!
NOTE 11 Jan 2022: This post is out of date, as Apple has now brought back support for Safari Extensions that use similar syntax to extensions for Chrome and Firefox. You might still find this post informative, but I recommend having a look at this and this video from WWDC’21 where Apple developers explain how ‘Safari Web Extensions’ are now developed.
I previously wrote a little browser extension for Chrome and Safari that reduces Facebook to just the functionality I find useful, by removing the newsfeed and most other things that distract me on the front page.
data:image/s3,"s3://crabby-images/0198b/0198bebec502e89106942d70758fcc116ae519a4" alt="Facebook with no distractions"
Facebook with no distractions
With the most recent update of Safari, however, my extension was automatically turned off everytime I closed the browser. And I discovered that Apple has an ongoing plan to integrate extensions to Safari with the usual development of Apple apps, instead of following the more interoperable approach to browser extensions taken by e.g. Chrome and Firefox. At some point in the not to distant future, the only way to get a Safari extension will be by downloading it as an extension attached to an app on the macOS App Store. [11. Oct. 2019 update: This has now happened as per Safari v13.0]
Sigh. I kinda like using Safari, mostly because Apple is putting more and more emphasis on protecting user privacy in times of ubiquitous tracking. Unfortunately, that means that any Safari extensions I’d like to build must be developed using Apple’s IDE Xcode.
I tried to do this but it was painful. Mostly because I couldn’t find a single tutorial online that just walked through a basic example. I did eventually figure things out using Apple’s documentation and talks from the Safari team at WWDC, but it was more way more frustrating and time consuming than necessary.
So, to make this learning quicker and easier for others, here’s a tutorial that walks through a simple example of how to build my "No Distractions" for Facebook
as a Safari App Extension:
Step 1: Get Xcode
If you haven’t already, install Apple’s IDE Xcode.
Step 2: Create a project in Xcode
Open Xcode, create a new project, choose ‘Cocoa App’ under the macOS project templates, and click ‘next’.
data:image/s3,"s3://crabby-images/227dc/227dc1e98b1e56340680bde075ea9c2cadc10311" alt="Create cocoa app"
Create cocoa app
Give the project a name, e.g. ‘Facebook Minimiser’. Untick ‘Use Storyboards’, ‘Include Unit Tests’ and ‘Include UI Tests’, as we’ll just do a minimal example, then click ‘next’.
data:image/s3,"s3://crabby-images/2d5f4/2d5f4bae7c4f265225dc921a8f1da1cef0cb9124" alt="Name project"
Name project
Choose where to save the project - let’s just save on the desktop for this tutorial. Click ‘create’.
data:image/s3,"s3://crabby-images/e1e10/e1e1083bbf8d54d7f9130b5b65e8a130e0ed8224" alt="Save project"
Save project
Now we see the template app open in Xcode.
data:image/s3,"s3://crabby-images/cc050/cc050349dc614271c7c6bebc90d35c0c04e80e84" alt="Template project open in Xcode"
Template project open in Xcode
Step 3: Add a Safari Extension ‘target’ to the app
If you, like me, are new to Xcode, try running the current template app by clicking the ‘play’ triangle icon in the top left. This will build and run the app:
data:image/s3,"s3://crabby-images/6ded7/6ded7d77f9dc76ee4dad445fbfd16980b70a0e76" alt="Build template app"
Build template app
As we haven’t added any user interface to the app, the opened application window will just be blank:
data:image/s3,"s3://crabby-images/5be2d/5be2d32e8b3712c0eaf0ef5b7581a82a7b12cc7a" alt="Blank template app opened"
Blank template app opened
Close the application again:
data:image/s3,"s3://crabby-images/3b99a/3b99adbdc24fe400ffc3667c0cd3613166c74768" alt="Closing template app"
Closing template app
Now we add a Safari Extension ‘target’ to our app. Click ‘File’ -> ‘New’ -> ‘Target…’:
data:image/s3,"s3://crabby-images/0e8bf/0e8bf46c525f2214584a601c0311a487192c0889" alt="Adding extension target"
Adding extension target
Under ‘macOS’, choose ‘Safari Extension’ then click ‘next’.
data:image/s3,"s3://crabby-images/721d0/721d066161c704449911b7ffeeffcf1ee9864e3d" alt="Choosing Safari Extension template"
Choosing Safari Extension template
Give the extension a name, e.g. ‘Facebook Minimiser Extension’ and click ‘Finish’:
data:image/s3,"s3://crabby-images/85a6e/85a6e27adb64e243150e7a39a6cc9e515b4c5426" alt="Name extension target"
Name extension target
When Xcode asks whether it should ‘Activate “Facebook Minimiser Extension” scheme?’, click ‘cancel’.
data:image/s3,"s3://crabby-images/17bf4/17bf44f4107d6ad7a53a873cd61d6fd461737e66" alt="Name extension target"
Name extension target
Now a folder called ‘Facebook Minimiser Extension’ has been added to the project:
data:image/s3,"s3://crabby-images/389c0/389c0ea22b9a27013bf78572094ab21692de7b74" alt="Extension added"
Extension added
Step 4: Try building the empty extension and enable it in Safari
If you click the ‘play’ triangle icon in the top left, the app will again be built and run.
Apple tells us that
When you build your app, Xcode builds your Safari App Extension first, then embeds it inside the finished app bundle. As soon as your app runs, your extension is ready for use in Safari.
Let’s go over to Safari, then go to Safari > Preferences. Click the ‘Extensions’ pane.
data:image/s3,"s3://crabby-images/b2f31/b2f31aa0f9361ef7161370e309698550d211ef6c" alt="Safari preferences extensions pane"
Safari preferences extensions pane
Uh-oh - where’s our extension? Turns out that unless we sign our extension with a developer certificate, it won’t automatically appear. If you haven’t already turned on the Development menu in Safari, go to Safari > Preferences > Advanced, then tick ‘Show Develop menu in menu bar’.
data:image/s3,"s3://crabby-images/aed1a/aed1a0493c07479b8870e884fe498a50e30edfcb" alt="Adding Develop menu"
Adding Develop menu
Now go to ‘Develop’ in the top menu, click ‘Allow Unsigned Extensions’ and type in your password.
data:image/s3,"s3://crabby-images/6fcde/6fcde950fcbcfa34d0de2b0de2fd0c4be23b94ec" alt="Allowing unsigned extensions"
Allowing unsigned extensions
Hooray, now our extension shows up! If we tick to enable it, we see its icon in the toolbar:
data:image/s3,"s3://crabby-images/dc13d/dc13d05bcda62c9cdd83071184571418e43bd5d5" alt="Enabling the extension"
Enabling the extension
Step 5: Injecting a style sheet
Create the style sheet file
The extension doesn’t do anything yet. Go back to Xcode and create a new file by right-clicking the ‘Facebook Minimiser Extension’ folder:
data:image/s3,"s3://crabby-images/78379/78379166e30c4a6020e2bd7d37068eab3563173f" alt="Adding file"
Adding file
Select ‘Empty’ and click ‘next’:
data:image/s3,"s3://crabby-images/7fb99/7fb990e675e3aeac95d6376dba9e55766ea23db1" alt="Choosing file type"
Choosing file type
Let’s name our stylesheet style.css, set the target to ‘Facebook Minimiser Extension’, then click ‘create’:
data:image/s3,"s3://crabby-images/8a4ac/8a4ac32e72938044ebf1ce854f577f60259d2cf4" alt="Naming file and choosing target"
Naming file and choosing target
As an initial test, let’s hide the news feed. In Safari, navigate to Facebook, right click on the page, select ‘Inspect Element’, and find the id or class of the element that contains the feed:
data:image/s3,"s3://crabby-images/8e681/8e6813ee79e2d557c098265b535f9c8bcfec3b0b" alt="Inspecting Facebook HTML"
Inspecting Facebook HTML
Now go back to Xcode and add this corresponding CSS code to style.css:
#topnews_main_stream_408239535924329
{
display: none !important;
}
data:image/s3,"s3://crabby-images/79e62/79e624ce885cd612d44a15556e5a4e1fa676d7a0" alt="Updating style.css to hide news feed"
Updating style.css to hide news feed
Include the style sheet file in the extension’s Info.plist file
In the ‘Facebook Minimiser Extension’ folder, click the **Info.plist* file. Expand the ‘NSExtension’ element and click the plus icon to add a new element 1:
data:image/s3,"s3://crabby-images/3c0c3/3c0c3a0d179e4428dc2d09f883b1b7f074ba3d7a" alt="Add element to NSExtension in **Info.plist**"
Add element to NSExtension in Info.plist
Name this element ‘SFSafariStyleSheet’ and set its type to ‘Array’. Expand the element and click the plus icon to add a new element:
data:image/s3,"s3://crabby-images/eaca5/eaca504e9c0a8a90db92ad1c0fedce4998de3e8a" alt="Specify array 'SFSafariStyleSheet'"
Specify array ‘SFSafariStyleSheet’
Set the type of this element to ‘Dictionary’, expand it and click the plus icon to add a new element:
data:image/s3,"s3://crabby-images/9bcc4/9bcc4825afb2f89e9447ee900af6f7ca03b214f8" alt="Set to dictionary and add element"
Set to dictionary and add element
Name this element ‘Style Sheet’, leave its type as ‘String’, and give it the value ‘style.css’:
data:image/s3,"s3://crabby-images/77190/77190db39ebfbd16c86e48746146ae10d7c0eec8" alt="Point to our style sheet"
Point to our style sheet
Set websites to target
Finally, expand ‘SFSafariWebsiteAccess’ and under ‘Allowed Domains’ set the ‘Item 0’ element’s value to ’*.facebook.com’:
data:image/s3,"s3://crabby-images/84fd3/84fd390f61398029f736596bade9560df41178c8" alt="Inject only on Facebook"
Inject only on Facebook
Now, if we rebuild our app and activate the extension in Safari, we see the news feed has been hidden:
Step 6: Tweak the CSS
Finally, tweak the content of style.css to build my Facebook Minimiser:
/* ELEMENTS TO REMOVE FROM DISPLAY */
#pagelet_ego_pane, /* recommended for you and sponsored content */
#u_0_1t, /* right side bar instant chat people online */
#u_0_1u, /* right side bar instant chat prompt */
#u_0_1v, /* right side bar instant chat prompt */
#pagelet_rhc_footer, /* select language prompt, and Facebook copyright footer */
#createNav, /* create shortcut on the bottom left */
#appsNav, /* explore bar on the left */
#pagelet_trending_tags_and_topics, /* trending topics */
#stories_pagelet_rhc, /* stories */
#topnews_main_stream_408239535924329, /* newsfeed */
.fbChatSidebarBody,
#u_0_1w,
#pagelet_gaming_destination_rhc, /*gaming sidebar*/
#pagelet_marketplace_new_user_top_picks_rhc /*market place sidebar*/
{
display: none !important;
}
/* remove border from where the chat sidebar was */
.fbChatSidebar {
border-left: none !important;
}
/* make the background white instead of the usual depressing grey */
#globalContainer,
#contentCol,
._5vb_,
._5vb_ #contentCol,
#u_0_1s,
._4oes,
._51x_,
body {
background-color: white !important;
}
And now we get a nice, simple, and distraction free Facebook experience:
data:image/s3,"s3://crabby-images/ed870/ed8702bdb8e90ffd323a065ffd3b892832e56132" alt="Distraction-free Facebook!"
Distraction-free Facebook!
Step 7: Celebrate! (almost)
These are the basic steps to embed a basic Safari Extension in an app using Xcode. I did find (and filed a bug report to Apple) that in Mojave, my extensions would at times mysteriously disappear from Safari > Preferences > Extensions - hopefully this won’t occur on your machine and/or Apple will fix this bug ASAP.
The Info.plist file is quite similar to the manifest.json file used in Chrome extensions.↩︎
Comments
I use the open source, zero-tracking utteranc.es widget for comments - it's built on GitHub issues, so you need a GitHub account to comment.