-
Notifications
You must be signed in to change notification settings - Fork 487
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add 4Column layout for ultra-wide screens (#1154)
Co-authored-by: Ian Ynda-Hummel <[email protected]>
- Loading branch information
Showing
3 changed files
with
309 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,291 @@ | ||
// | ||
// FourColumnLayout.swift | ||
// Amethyst | ||
// | ||
// Originally created by Ian Ynda-Hummel on 12/15/15. | ||
// Copyright © 2015 Ian Ynda-Hummel. All rights reserved. | ||
// | ||
// Modifications by Craig Disselkoen on 09/03/18. | ||
// Modifications by Reyk Floeter on 10/28/21. | ||
// | ||
|
||
import Silica | ||
|
||
// we'd like to hide these structures and enums behind fileprivate, but | ||
// https://bugs.swift.org/browse/SR-47 | ||
|
||
enum FourColumn { | ||
case left | ||
case middleLeft | ||
case middleRight | ||
case right | ||
} | ||
|
||
enum FourPane { | ||
case main | ||
case secondary | ||
case tertiary | ||
case quaternary | ||
} | ||
|
||
struct FourPaneWidths { | ||
var left: CGFloat = 0 | ||
var middleLeft: CGFloat = 0 | ||
var middleRight: CGFloat = 0 | ||
var right: CGFloat = 0 | ||
} | ||
|
||
struct QuadruplePaneArrangement { | ||
/// number of windows in pane | ||
private let paneCount: [FourPane: UInt] | ||
|
||
/// height of windows in pane | ||
private let paneWindowHeight: [FourPane: CGFloat] | ||
|
||
/// width of windows in pane | ||
private let paneWindowWidth: [FourPane: CGFloat] | ||
|
||
// how panes relate to columns | ||
private let panePosition: [FourPane: FourColumn] | ||
|
||
/// how columns relate to panes | ||
private let columnDesignation: [FourColumn: FourPane] | ||
|
||
/** | ||
- Parameters: | ||
- mainPane: which Column is the main Pane | ||
- numWindows: how many windows total | ||
- numMainPane: how many windows in the main Pane | ||
- screenSize: total size of the screen | ||
- mainPaneRatio: ratio of the screen taken by main pane | ||
*/ | ||
init(mainPane: FourColumn, numWindows: UInt, numMainPane: UInt, screenSize: CGSize, mainPaneRatio: CGFloat) { | ||
// forward and reverse mapping of columns to their designations | ||
self.panePosition = { | ||
switch mainPane { | ||
case .left: return [.main: .left, .secondary: .middleLeft, .tertiary: .middleRight, .quaternary: .right] | ||
case .middleLeft: return [.main: .middleLeft, .secondary: .middleRight, .tertiary: .left, .quaternary: .right] | ||
case .middleRight: return [.main: .middleRight, .secondary: .middleLeft, .tertiary: .right, .quaternary: .left] | ||
case .right: return [.main: .right, .secondary: .middleRight, .tertiary: .middleLeft, .quaternary: .left] | ||
} | ||
}() | ||
// swap keys and values for reverse lookup | ||
self.columnDesignation = Dictionary(uniqueKeysWithValues: panePosition.map({ ($1, $0) })) | ||
|
||
// calculate how many are in each type | ||
let mainPaneCount = min(numWindows, numMainPane) | ||
let nonMainCount: UInt = numWindows - mainPaneCount | ||
// we do tertiary first because a single window produces a zero in integer division by 2 | ||
let nonMainPaneCount: UInt = max(nonMainCount / 3, 1) | ||
let quaternaryPaneCount = nonMainPaneCount | ||
let tertiaryPaneCount = nonMainPaneCount | ||
let secondaryPaneCount = nonMainPaneCount + max(nonMainCount, 3) % 3 | ||
self.paneCount = [.main: mainPaneCount, .secondary: secondaryPaneCount, .tertiary: tertiaryPaneCount, .quaternary: quaternaryPaneCount] | ||
|
||
// calculate heights | ||
let screenHeight = screenSize.height | ||
self.paneWindowHeight = [ | ||
.main: round(screenHeight / CGFloat(mainPaneCount)), | ||
.secondary: secondaryPaneCount == 0 ? 0.0 : round(screenHeight / CGFloat(secondaryPaneCount)), | ||
.tertiary: tertiaryPaneCount == 0 ? 0.0 : round(screenHeight / CGFloat(tertiaryPaneCount)), | ||
.quaternary: quaternaryPaneCount == 0 ? 0.0 : round(screenHeight / CGFloat(quaternaryPaneCount)) | ||
] | ||
|
||
// calculate widths | ||
let screenWidth = screenSize.width | ||
let mainWindowWidth = round(screenWidth / 4) | ||
let nonMainWindowWidth = round(screenWidth / 4) | ||
self.paneWindowWidth = [ | ||
.main: mainWindowWidth, | ||
.secondary: nonMainWindowWidth, | ||
.tertiary: nonMainWindowWidth, | ||
.quaternary: nonMainWindowWidth | ||
] | ||
} | ||
|
||
func count(_ pane: FourPane) -> UInt { | ||
return paneCount[pane]! | ||
} | ||
|
||
func height(_ pane: FourPane) -> CGFloat { | ||
return paneWindowHeight[pane]! | ||
} | ||
|
||
func width(_ pane: FourPane) -> CGFloat { | ||
return paneWindowWidth[pane]! | ||
} | ||
|
||
func firstIndex(_ pane: FourPane) -> UInt { | ||
switch pane { | ||
case .main: return 0 | ||
case .secondary: return count(.main) | ||
case .tertiary: return count(.main) + count(.secondary) | ||
case .quaternary: return count(.main) + count(.secondary) + count(.tertiary) | ||
} | ||
} | ||
|
||
func pane(ofIndex windowIndex: UInt) -> FourPane { | ||
if windowIndex >= firstIndex(.quaternary) { | ||
return .quaternary | ||
} | ||
if windowIndex >= firstIndex(.tertiary) { | ||
return .tertiary | ||
} | ||
if windowIndex >= firstIndex(.secondary) { | ||
return .secondary | ||
} | ||
return .main | ||
} | ||
|
||
/// Given a window index, which Pane does it belong to, and which index within that Pane | ||
func coordinates(at windowIndex: UInt) -> (FourPane, UInt) { | ||
let pane = self.pane(ofIndex: windowIndex) | ||
return (pane, windowIndex - firstIndex(pane)) | ||
} | ||
|
||
/// Get the (height, width) dimensions for a window in the given Pane | ||
func windowDimensions(inPane pane: FourPane) -> (CGFloat, CGFloat) { | ||
return (height(pane), width(pane)) | ||
} | ||
|
||
/// Get the Column assignment for the given Pane | ||
func column(ofPane pane: FourPane) -> FourColumn { | ||
return panePosition[pane]! | ||
} | ||
|
||
func pane(ofColumn column: FourColumn) -> FourPane { | ||
return columnDesignation[column]! | ||
} | ||
|
||
/// Get the column widths in the order (left, middle, right) | ||
func widthsLeftToRight() -> FourPaneWidths { | ||
return FourPaneWidths( | ||
left: width(pane(ofColumn: .left)), | ||
middleLeft: width(pane(ofColumn: .middleLeft)), | ||
middleRight: width(pane(ofColumn: .middleRight)), | ||
right: width(pane(ofColumn: .right)) | ||
) | ||
} | ||
} | ||
|
||
// not an actual Layout, just a base class for the four actual Layouts below | ||
class FourColumnLayout<Window: WindowType>: Layout<Window> { | ||
class var mainColumn: FourColumn { fatalError("Must be implemented by subclass") } | ||
|
||
enum CodingKeys: String, CodingKey { | ||
case mainPaneCount | ||
case mainPaneRatio | ||
} | ||
|
||
private(set) var mainPaneCount: Int = 1 | ||
private(set) var mainPaneRatio: CGFloat = 0.5 | ||
|
||
required init() { | ||
super.init() | ||
} | ||
|
||
required init(from decoder: Decoder) throws { | ||
let values = try decoder.container(keyedBy: CodingKeys.self) | ||
self.mainPaneCount = try values.decode(Int.self, forKey: .mainPaneCount) | ||
self.mainPaneRatio = try values.decode(CGFloat.self, forKey: .mainPaneRatio) | ||
super.init() | ||
} | ||
|
||
override func encode(to encoder: Encoder) throws { | ||
var container = encoder.container(keyedBy: CodingKeys.self) | ||
try container.encode(mainPaneCount, forKey: .mainPaneCount) | ||
try container.encode(mainPaneRatio, forKey: .mainPaneRatio) | ||
} | ||
|
||
override func frameAssignments(_ windowSet: WindowSet<Window>, on screen: Screen) -> [FrameAssignmentOperation<Window>]? { | ||
let windows = windowSet.windows | ||
|
||
guard !windows.isEmpty else { | ||
return [] | ||
} | ||
|
||
let screenFrame = screen.adjustedFrame() | ||
let paneArrangement = QuadruplePaneArrangement( | ||
mainPane: type(of: self).mainColumn, | ||
numWindows: UInt(windows.count), | ||
numMainPane: UInt(mainPaneCount), | ||
screenSize: screenFrame.size, | ||
mainPaneRatio: mainPaneRatio | ||
) | ||
|
||
return windows.reduce([]) { frameAssignments, window -> [FrameAssignmentOperation<Window>] in | ||
var assignments = frameAssignments | ||
var windowFrame = CGRect.zero | ||
let windowIndex: UInt = UInt(frameAssignments.count) | ||
|
||
let (pane, paneIndex) = paneArrangement.coordinates(at: windowIndex) | ||
|
||
let (windowHeight, windowWidth): (CGFloat, CGFloat) = paneArrangement.windowDimensions(inPane: pane) | ||
let column: FourColumn = paneArrangement.column(ofPane: pane) | ||
|
||
let widths = paneArrangement.widthsLeftToRight() | ||
|
||
let xorigin: CGFloat = screenFrame.origin.x + { | ||
switch column { | ||
case .left: return 0.0 | ||
case .middleLeft: return widths.left | ||
case .middleRight: return widths.left + widths.middleLeft | ||
case .right: return widths.left + widths.middleLeft + widths.middleRight | ||
} | ||
}() | ||
|
||
let scaleFactor: CGFloat = screenFrame.width / { | ||
if pane == .main { | ||
return paneArrangement.width(.main) | ||
} | ||
return paneArrangement.width(.secondary) + paneArrangement.width(.tertiary) + paneArrangement.width(.quaternary) | ||
}() | ||
|
||
windowFrame.origin.x = xorigin | ||
windowFrame.origin.y = screenFrame.origin.y + (windowHeight * CGFloat(paneIndex)) | ||
windowFrame.size.width = windowWidth | ||
windowFrame.size.height = windowHeight | ||
|
||
let isMain = windowIndex < paneArrangement.firstIndex(.secondary) | ||
|
||
let resizeRules = ResizeRules(isMain: isMain, unconstrainedDimension: .horizontal, scaleFactor: scaleFactor) | ||
let frameAssignment = FrameAssignment<Window>( | ||
frame: windowFrame, | ||
window: window, | ||
screenFrame: screenFrame, | ||
resizeRules: resizeRules | ||
) | ||
|
||
assignments.append(FrameAssignmentOperation(frameAssignment: frameAssignment, windowSet: windowSet)) | ||
|
||
return assignments | ||
} | ||
} | ||
} | ||
|
||
extension FourColumnLayout { | ||
func recommendMainPaneRawRatio(rawRatio: CGFloat) { | ||
mainPaneRatio = rawRatio | ||
} | ||
|
||
func increaseMainPaneCount() { | ||
mainPaneCount += 1 | ||
} | ||
|
||
func decreaseMainPaneCount() { | ||
mainPaneCount = max(1, mainPaneCount - 1) | ||
} | ||
} | ||
|
||
// implement the two variants | ||
class FourColumnLeftLayout<Window: WindowType>: FourColumnLayout<Window>, PanedLayout { | ||
override static var layoutName: String { return "4Column Left" } | ||
override static var layoutKey: String { return "4column-left" } | ||
override static var mainColumn: FourColumn { return .middleLeft } | ||
} | ||
|
||
class FourColumnRightLayout<Window: WindowType>: FourColumnLayout<Window>, PanedLayout { | ||
override static var layoutName: String { return "4Column Right" } | ||
override static var layoutKey: String { return "4column-right" } | ||
override static var mainColumn: FourColumn { return .middleRight } | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters