Advanced Dark Mode
Advanced Dark Mode
WWDC 2018
NSAppearance
- Cocoa’s UI theme system
- One view hierarchy for both light and dar
- Underlies Dark Mode, High Contrast mode, and Touch Bar
data:image/s3,"s3://crabby-images/53cae/53cae1a221ba5a550123f37e1c7980a86faa21fe" alt=""
data:image/s3,"s3://crabby-images/63307/63307ddc825f23e8d4b9734c6698893ee332196c" alt=""
NSAppearanceCustomization
- Adopted by NSView and NSWindow
- In 10.14, NSApplication conforms too
data:image/s3,"s3://crabby-images/8505d/8505df673f44d4a0522c2e66b408441dceba6566" alt=""
Appearance Source ( new )
- Makes a window inherit appearance from another object
data:image/s3,"s3://crabby-images/b65b0/b65b0a13a060812380b4529677cc0fa623aaaed8" alt=""
- Assign
nil
to reset default inheritance
data:image/s3,"s3://crabby-images/af95c/af95c9ae0ac466027052bc6b7326d5a9bc13c148" alt=""
data:image/s3,"s3://crabby-images/2c154/2c154571ef073a49fe7e9be4005bceb81475f32f" alt=""
Custom Views
- Do appearance-sensitive work in
data:image/s3,"s3://crabby-images/0fef9/0fef9fbbb4823815e295107f7c18174f2647f597" alt=""
- Appearance changes will cause
- Set
needsUpdateConstraints needsLayout, or needsDisplay
to trigger manually
data:image/s3,"s3://crabby-images/dc478/dc47870825f8dfe941f6d2f158605d77663b7a8e" alt=""
data:image/s3,"s3://crabby-images/0cd63/0cd632fe0c11cbaa117de56582d4c2cf55ce5265" alt=""
data:image/s3,"s3://crabby-images/3cf68/3cf6824641891626b3ea47195794035875caacdb" alt=""
data:image/s3,"s3://crabby-images/7da43/7da4369bc59af598d38afa799a7e624dd0c3cb37" alt=""
Glow behind image
data:image/s3,"s3://crabby-images/291ab/291ab71b26df91b4709a852b121e553fd3fbbe43" alt=""
data:image/s3,"s3://crabby-images/a833c/a833c52b88290e09df7747e1b35d283848d9cf10" alt=""
Customizing for High Contrast
- High-contrast appearance names
data:image/s3,"s3://crabby-images/b2338/b233859cf377fe8c21f9d604c490bdc7f93fe152" alt=""
data:image/s3,"s3://crabby-images/8860c/8860c9c099e55e0d9ae90354d4abef26daac9691" alt=""
data:image/s3,"s3://crabby-images/1d2ed/1d2edd2c509c25eb663269d27935b091a1e33c1b" alt=""
data:image/s3,"s3://crabby-images/04579/045792ad472c5bae879a3714702489305fab5422" alt=""
- Not available directly
- Only via System Preferences
data:image/s3,"s3://crabby-images/fed3e/fed3e3aca47f6454c741413ea296ea7bae8337f7" alt=""
- Pass to
bestMatch(from:)
to provide custom behavior
data:image/s3,"s3://crabby-images/fe284/fe2849ae1cf64e44922c10db711790440f435cb5" alt=""
Managing Custom Sublayers
- Custom sublayers don’t inherit appearance automatically
- If possible, switch to using views
- Otherwise, manage layers manually with
-viewDidChangeEffectiveAppearance()
-NSAppearance.current
Detecting Appearance Change ( new )
data:image/s3,"s3://crabby-images/49d69/49d69625e1afd0eeb81ae8dfcf88d7ae46f981d5" alt=""
- Good time to
- Perform custom invalidation
- Drop caches - No need to invalidate view itself !
Current Appearance
data:image/s3,"s3://crabby-images/19ca4/19ca40f2a18492349d976f18c7d7486dc38e069e" alt=""
- Thread-local state
- LikeNSGraphicsContext.current
- Appearance used to resolve colors and images
- Current appearance is set up automatically for
data:image/s3,"s3://crabby-images/86442/864429d88cc327ac3489fce9fcc284911978f8a0" alt=""
- You can also set it yourself
data:image/s3,"s3://crabby-images/d3f5e/d3f5e41f3449c9aa18c5e057f160e11c5a1c4799" alt=""
NSImage as Layer Contents
- Don’t
data:image/s3,"s3://crabby-images/30494/3049428c72b6d8b11ae8ed31238fe0866225a426" alt=""
- Do
- If possible, switch to NSImageView
- Automatically updates for appearance
- Create a CGImage inupdateLayer()
data:image/s3,"s3://crabby-images/c4b83/c4b83cccaae649405acd9e0a45574bcdc62859e1" alt=""
Materials
- Dynamic backgrounds
- Use effects like blur, tinting, and gradients
- Provide depth, context, and beauty to the interface
- Automatically used in
- Window titlebars and backgrounds
- Table views
- Sidebar split view items
- Popovers
- Menus
data:image/s3,"s3://crabby-images/99172/991729df7087b62ad8542562f85c72e6f0c602f6" alt=""
NSVisualEffectView
- A view that shows a material
data:image/s3,"s3://crabby-images/9292c/9292cf6f49c4d859f91d2095c159c0a57a06cb78" alt=""
- Controls whether material uses active window look
- By default, matches enclosing window
data:image/s3,"s3://crabby-images/fbef5/fbef5e6741dc467f7abd8f3684c8147ead746ab7" alt=""
- Controls whether material “punches through” window
data:image/s3,"s3://crabby-images/7c47b/7c47b01c227217e892271b42e44ac25af8bf48db" alt=""
- Behind-window by default
data:image/s3,"s3://crabby-images/fa92e/fa92ecc913282343b90d47819b2b05df85248c56" alt=""
data:image/s3,"s3://crabby-images/b5f7c/b5f7c5ef193df0daa8bd24943fc11bd1c40d9985" alt=""
- Encapsulates effect definition
data:image/s3,"s3://crabby-images/a8048/a8048585b1180825eff7768d025cc5d4661a6388" alt=""
Semantic Materials ( new )
data:image/s3,"s3://crabby-images/6f2f2/6f2f2e0aa8d2e633f4427c56741c22ae7a12e21d" alt=""
Materials
Nonsemantic materials are deprecated ⚠️
- Light
- Dark
- Medium light
- Ultra dard
data:image/s3,"s3://crabby-images/d50ac/d50aca51cd64fb552a33f854ba2481d8b8513482" alt=""
data:image/s3,"s3://crabby-images/de037/de03748666a1103d36a2bd1336416c24beae055b" alt=""
data:image/s3,"s3://crabby-images/998be/998be43a12acc7d0d0238001a49bd9d39ef1e34a" alt=""
Desktop-Tinted Materials
- Automatic support
- NSWindow, NSScrollView, NSTableView, NSCollectionView - Custom NSBox support
data:image/s3,"s3://crabby-images/d8e03/d8e030b29e5c34847a2f240827bff172132623ac" alt=""
- NSBox and NSVisualEffectView support
data:image/s3,"s3://crabby-images/55064/550646f87fe9ae849419ef2cae78cac189876301" alt=""
- Untinted color in light
- Desktop-tinted in dark
- Tint effect might be disabled
data:image/s3,"s3://crabby-images/6176b/6176b35225b0bca9240f4001d634ff1a4e9d6910" alt=""
data:image/s3,"s3://crabby-images/9acb2/9acb260b485a0f1a0b2f44c2d142d18286ef3a83" alt=""
data:image/s3,"s3://crabby-images/bc059/bc059255700607b9c0fdd17de3d5186bef61631a" alt=""
data:image/s3,"s3://crabby-images/2e6f4/2e6f4df9517df332a75343d39fb4192890cb787f" alt=""
data:image/s3,"s3://crabby-images/e3e45/e3e45e5a8938772ac56d9e6100cc1f9ae20b6d65" alt=""
data:image/s3,"s3://crabby-images/323de/323de725760cbb55e9dbc2649adc156eadd3964d" alt=""
- Effect is provided by Quartz window server
- Updates asynchronously from your app
- But can’t draw with the color ( or get RGB values )
- Instead, use
maskImage
Mask Image
- Masks the material to a shape
- Drawing-handler images supported
- Works with any material
- Alpha channel used for mask
- Only material (not subviews) masked
- Provide a resizable image
capInsets
andresizingMode
Vibrancy
- Lightens or darkens the content behind it
- Similar to dodge and burn in a photo editor
data:image/s3,"s3://crabby-images/be9a4/be9a4d9d21e086842fb9782bf74ab122504eb536" alt=""
data:image/s3,"s3://crabby-images/a62ee/a62ee4371fbde08698b13300bef1522e00d0fd6c" alt=""
data:image/s3,"s3://crabby-images/b3a24/b3a24b02f320c8b799c1124c0d278e61383e464e" alt=""
- Provided by “vibrant” NSAppearance objects
data:image/s3,"s3://crabby-images/3dae5/3dae5f1d056911a47acce09553a6451599fff2ef" alt=""
- Override
allowsVibrancy
to returntrue
- Blending effect automatically applies to your view and its descendants
- Typically, draw with a label color depending on prominence
- Avoid non grayscale colors
data:image/s3,"s3://crabby-images/e487e/e487ebf721932d6a69cd4010e0b7c6bb2c7d8c5b" alt=""
Vibrant Blending Tips
- Vibrant blend mode applies to subtree
- Typically only leaf views should be vibrant
- Break non vibrant drawing out into a sibling view
- Avoid overlapping vibrant and non-vibrant views
- Don’t subclass Cocoa controls to override
allowsVibrancy
- Only override
allowsVibrancy
if you also override drawing
data:image/s3,"s3://crabby-images/17d11/17d11a575276a42e25d118698639189b1431f753" alt=""
data:image/s3,"s3://crabby-images/51056/5105637b4d9a33323149f23f2fcd8507e4407fb3" alt=""
Background Styles ( new )
- NSTableView now sets background style recursively
- Label colors automatically adapt to emphasized style
data:image/s3,"s3://crabby-images/e929f/e929f1be106321594ae879b6aa27407a60dcf9ea" alt=""
data:image/s3,"s3://crabby-images/c39a5/c39a5f527d25de073bbea2225f7646b3f40c2ee2" alt=""
data:image/s3,"s3://crabby-images/04010/040109ed3d44c06c9a6edc3b36d24ff0be2e7ef0" alt=""
Custom Selections
data:image/s3,"s3://crabby-images/384d3/384d3430ae704dd548a5c5daf1ee32c9bf8dde24" alt=""
Selection Material
- Selection material follow the accent color
- Avoid drawing custom blue selections
- Use NSVisualEffectView instead
data:image/s3,"s3://crabby-images/f050f/f050f8037889da57924f700be10479241a1f8714" alt=""
data:image/s3,"s3://crabby-images/d4a70/d4a70ce8388b37cc2e07c99147e0de4a4d94ee96" alt=""
Backward Deployment
- System colors ( 10.10 + )
data:image/s3,"s3://crabby-images/4e8e1/4e8e1940281da4e301b85acce3dff0d610c9b3d5" alt=""
- Asset catalog colors
data:image/s3,"s3://crabby-images/93a17/93a176788c52f431d50408e3cb072ab8f7109317" alt=""
data:image/s3,"s3://crabby-images/fed1d/fed1de5ba7eb2214704392c701905876e6faa97e" alt=""
- Desktop-tinted materials
- Desktop-tinted NSVisualEffectView materials are new in 10.14
- Classes providing automatic materials
data:image/s3,"s3://crabby-images/59956/59956a7b73406145599efee714eaed337b7afc94" alt=""
- Manually enable or disable Dar Mode
- Automatically opted in with 10.14 SDK
- Info.plist key:NSRequiresAquaSystemAppearance
Set to NO to enable Dark Mode with earlier SDKs
Set to YES to disable it temporarily
Find Hardcoded Colors
- Static named color
data:image/s3,"s3://crabby-images/1e592/1e5925427d820413e7a83a7eaab273047fef1d8f" alt=""
- Hardcoded components
data:image/s3,"s3://crabby-images/0d58a/0d58a66b73c14f2502f8ae76f0530a204d199461" alt=""
Offscreen Drawing
- Prefer block-based NSImage over lockFocus
data:image/s3,"s3://crabby-images/451d2/451d26bd4477e6348a17c0cf6e8995309a429908" alt=""
- Other offscreen drawing techniques, including
data:image/s3,"s3://crabby-images/75d15/75d155e7d1724100152d57576e7345990d4d0452" alt=""
data:image/s3,"s3://crabby-images/14007/1400701ad3fce2138e930a0e9640369852eed8c9" alt=""
data:image/s3,"s3://crabby-images/7ccd1/7ccd12d918c98c68e5ca4e90f203e17d57265657" alt=""
Appearances in Interface Builder
- Views set to Aqua won’t change dynamically
- Choose “Inherited” to match the superview
data:image/s3,"s3://crabby-images/1d432/1d43252d2a31a19295794061176d84d564d702d9" alt=""
- NSVisualEffectView now automatically uses the correct vibrant appearance
- Avoid overriding the appearance
- Interface Builder: select “Inherited”
- In code: setappearance
tonil
data:image/s3,"s3://crabby-images/255bb/255bbac22c4e732c0bdf19e16216e606815421f5" alt=""
data:image/s3,"s3://crabby-images/e0167/e0167885f64f0fb9505823df4504956ba59e300c" alt=""
Testing Appearances ( new )
- Switch appearances from the Debug Bar and Touch Bar
data:image/s3,"s3://crabby-images/cc666/cc6668e3349f979898fc7a2ff730205b696a468f" alt=""
Using the View Debugger ( new )
Enhancements in Xcode 10
- System color names
- Asset Catalog color names
- NSAppearance configuration
data:image/s3,"s3://crabby-images/e7f20/e7f206c59ba89c1600f2e7a1845c1770bfdf274e" alt=""
data:image/s3,"s3://crabby-images/c9c7b/c9c7b09d6ff4e7bbe0350ab9c4b3f8dde7d91130" alt=""