mirror of
https://github.com/hensm/fx_cast.git
synced 2026-06-12 10:39:57 +00:00
Add missing i18n to native macOS receiver selector and refactor
This commit is contained in:
@@ -1,14 +1,13 @@
|
|||||||
import Cocoa
|
import Cocoa
|
||||||
|
|
||||||
|
|
||||||
class AppDelegate: NSObject {
|
class AppDelegate : NSObject, NSApplicationDelegate {
|
||||||
var mainWindow: NSWindow?
|
var mainWindow: NSWindow?
|
||||||
var mainWindowController: NSWindowController?
|
var mainWindowController: NSWindowController?
|
||||||
var mainWindowViewController: ViewController?
|
var mainWindowViewController: ViewController?
|
||||||
}
|
|
||||||
|
|
||||||
extension AppDelegate: NSApplicationDelegate {
|
|
||||||
func applicationDidFinishLaunching(_ aNotification: Notification) {
|
func applicationDidFinishLaunching (_ aNotification: Notification) {
|
||||||
let window = NSPanel(
|
let window = NSPanel(
|
||||||
contentRect: NSZeroRect
|
contentRect: NSZeroRect
|
||||||
, styleMask: [
|
, styleMask: [
|
||||||
@@ -21,7 +20,7 @@ extension AppDelegate: NSApplicationDelegate {
|
|||||||
, backing: .buffered
|
, backing: .buffered
|
||||||
, defer: false)
|
, defer: false)
|
||||||
|
|
||||||
window.title = "fx_cast"
|
window.titleVisibility = .hidden
|
||||||
window.orderFrontRegardless()
|
window.orderFrontRegardless()
|
||||||
window.center()
|
window.center()
|
||||||
|
|
||||||
@@ -39,13 +38,12 @@ extension AppDelegate: NSApplicationDelegate {
|
|||||||
NSApp.activate(ignoringOtherApps: true)
|
NSApp.activate(ignoringOtherApps: true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func applicationDidResignActive(_ aNotification: Notification) {
|
func applicationDidResignActive (_ aNotification: Notification) {
|
||||||
self.mainWindow?.performClose(aNotification)
|
self.mainWindow?.performClose(aNotification)
|
||||||
}
|
}
|
||||||
|
|
||||||
func applicationWillTerminate(_ aNotification: Notification) {}
|
func applicationShouldTerminateAfterLastWindowClosed (
|
||||||
|
_ app: NSApplication) -> Bool {
|
||||||
func applicationShouldTerminateAfterLastWindowClosed(_ app: NSApplication) -> Bool {
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,17 +1,29 @@
|
|||||||
import Cocoa
|
import Cocoa
|
||||||
|
|
||||||
protocol ReceiverViewDelegate: AnyObject {
|
|
||||||
|
protocol ReceiverViewDelegate : AnyObject {
|
||||||
func didCast (_ receiver: Receiver)
|
func didCast (_ receiver: Receiver)
|
||||||
}
|
}
|
||||||
|
|
||||||
class ReceiverView: NSStackView {
|
class ReceiverView : NSStackView {
|
||||||
weak var receiverViewDelegate: ReceiverViewDelegate?
|
weak var receiverViewDelegate: ReceiverViewDelegate?
|
||||||
|
|
||||||
var receiver: Receiver!
|
var receiver: Receiver!
|
||||||
var constraintsSet = false
|
var constraintsSet = false
|
||||||
|
|
||||||
var button: NSButton!
|
var castButton: NSButton!
|
||||||
var spinner: NSProgressIndicator!
|
var castingSpinner: NSProgressIndicator!
|
||||||
|
|
||||||
|
|
||||||
|
var isEnabled: Bool {
|
||||||
|
get {
|
||||||
|
return self.castButton.isEnabled
|
||||||
|
}
|
||||||
|
set {
|
||||||
|
self.castButton.isEnabled = newValue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
override init (frame: CGRect) {
|
override init (frame: CGRect) {
|
||||||
super.init(frame: frame)
|
super.init(frame: frame)
|
||||||
@@ -20,8 +32,8 @@ class ReceiverView: NSStackView {
|
|||||||
super.init(coder: coder)
|
super.init(coder: coder)
|
||||||
}
|
}
|
||||||
|
|
||||||
init (receiver: Receiver) {
|
init (receiver: Receiver, initData: InitData) {
|
||||||
super.init(frame: CGRect(x: 0, y: 0, width: 0, height: 0))
|
super.init(frame: NSZeroRect)
|
||||||
|
|
||||||
self.receiver = receiver
|
self.receiver = receiver
|
||||||
|
|
||||||
@@ -37,21 +49,23 @@ class ReceiverView: NSStackView {
|
|||||||
metaStackView.spacing = 4
|
metaStackView.spacing = 4
|
||||||
|
|
||||||
|
|
||||||
self.button = WideButton(
|
self.castButton = NSButton(
|
||||||
title: "Cast"
|
title: initData.i18n_castButtonTitle
|
||||||
, target: self
|
, target: self
|
||||||
, action: #selector(ReceiverView.onCast))
|
, action: #selector(ReceiverView.onCast))
|
||||||
|
|
||||||
self.button.bezelStyle = .rounded
|
self.castButton.bezelStyle = .rounded
|
||||||
|
self.castButton.widthAnchor.constraint(equalToConstant: 100).isActive = true
|
||||||
|
|
||||||
self.spinner = NSProgressIndicator()
|
|
||||||
self.spinner.style = .spinning
|
self.castingSpinner = NSProgressIndicator()
|
||||||
self.spinner.controlSize = .small
|
self.castingSpinner.style = .spinning
|
||||||
self.spinner.isHidden = true
|
self.castingSpinner.controlSize = .small
|
||||||
|
self.castingSpinner.isHidden = true
|
||||||
|
|
||||||
self.addArrangedSubview(metaStackView)
|
self.addArrangedSubview(metaStackView)
|
||||||
self.addArrangedSubview(self.spinner)
|
self.addArrangedSubview(self.castingSpinner)
|
||||||
self.addArrangedSubview(self.button)
|
self.addArrangedSubview(self.castButton)
|
||||||
|
|
||||||
self.distribution = .fill
|
self.distribution = .fill
|
||||||
}
|
}
|
||||||
@@ -61,34 +75,24 @@ class ReceiverView: NSStackView {
|
|||||||
|
|
||||||
if !constraintsSet {
|
if !constraintsSet {
|
||||||
self.translatesAutoresizingMaskIntoConstraints = false
|
self.translatesAutoresizingMaskIntoConstraints = false
|
||||||
self.leadingAnchor.constraint(equalTo: superview!.leadingAnchor, constant: 8).isActive = true
|
|
||||||
self.trailingAnchor.constraint(equalTo: superview!.trailingAnchor, constant: -8).isActive = true
|
self.leadingAnchor.constraint(
|
||||||
|
equalTo: superview!.leadingAnchor
|
||||||
|
, constant: 8).isActive = true
|
||||||
|
self.trailingAnchor.constraint(
|
||||||
|
equalTo: superview!.trailingAnchor
|
||||||
|
, constant: -8).isActive = true
|
||||||
|
|
||||||
constraintsSet = true
|
constraintsSet = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func disable () {
|
|
||||||
self.button.isEnabled = false
|
|
||||||
}
|
|
||||||
|
|
||||||
@objc
|
@objc
|
||||||
func onCast () {
|
func onCast () {
|
||||||
self.receiverViewDelegate?.didCast(self.receiver)
|
self.receiverViewDelegate?.didCast(self.receiver)
|
||||||
|
|
||||||
self.spinner.isHidden = false
|
self.castingSpinner.isHidden = false
|
||||||
self.spinner.startAnimation(nil)
|
self.castingSpinner.startAnimation(nil)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class WideButton: NSButton {
|
|
||||||
override var intrinsicContentSize: NSSize {
|
|
||||||
var size = super.intrinsicContentSize
|
|
||||||
if size.width < 100 {
|
|
||||||
size.width = 100
|
|
||||||
}
|
|
||||||
|
|
||||||
return size
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,60 +1,13 @@
|
|||||||
import Cocoa
|
import Cocoa
|
||||||
import Foundation
|
|
||||||
|
|
||||||
|
|
||||||
func makeLabel(_ text: String,
|
class ViewController : NSViewController {
|
||||||
size: CGFloat = 0,
|
var initData: InitData!
|
||||||
color: NSColor = NSColor.textColor) -> NSTextField {
|
|
||||||
|
|
||||||
let textField = NSTextField()
|
|
||||||
textField.stringValue = text
|
|
||||||
textField.backgroundColor = .clear
|
|
||||||
textField.isEditable = false
|
|
||||||
textField.isBezeled = false
|
|
||||||
textField.sizeToFit()
|
|
||||||
|
|
||||||
// Text
|
|
||||||
textField.font = NSFont.systemFont(ofSize: size)
|
|
||||||
textField.textColor = color
|
|
||||||
|
|
||||||
return textField
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
struct InitData: Codable {
|
|
||||||
let receivers: [Receiver]
|
|
||||||
let defaultMediaType: MediaType
|
|
||||||
|
|
||||||
let i18n_mediaTypeApp: String
|
|
||||||
let i18n_mediaTypeTab: String
|
|
||||||
let i18n_mediaTypeScreen: String
|
|
||||||
let i18n_mediaSelectCastLabel: String
|
|
||||||
let i18n_mediaSelectToLabel: String
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ReceiverSelection: Codable {
|
|
||||||
let receiver: Receiver
|
|
||||||
let mediaType: MediaType
|
|
||||||
}
|
|
||||||
|
|
||||||
enum MediaType: Int, Codable {
|
|
||||||
case app
|
|
||||||
case tab
|
|
||||||
case screen
|
|
||||||
}
|
|
||||||
|
|
||||||
struct Receiver: Codable {
|
|
||||||
let friendlyName: String
|
|
||||||
let host: String
|
|
||||||
let id: String
|
|
||||||
let port: Int
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class ViewController: NSViewController {
|
|
||||||
var mediaTypePopUpButton: NSPopUpButton!
|
var mediaTypePopUpButton: NSPopUpButton!
|
||||||
var receiverViews = [ReceiverView]()
|
var receiverViews = [ReceiverView]()
|
||||||
|
|
||||||
|
|
||||||
override func loadView () {
|
override func loadView () {
|
||||||
let visualEffectView = NSVisualEffectView()
|
let visualEffectView = NSVisualEffectView()
|
||||||
visualEffectView.blendingMode = .behindWindow
|
visualEffectView.blendingMode = .behindWindow
|
||||||
@@ -76,17 +29,35 @@ class ViewController: NSViewController {
|
|||||||
exit(1)
|
exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
let initData: InitData!
|
|
||||||
|
|
||||||
do {
|
do {
|
||||||
initData = try JSONDecoder().decode(InitData.self, from: data)
|
// Decode and store initialization JSON data
|
||||||
|
self.initData = try JSONDecoder().decode(InitData.self, from: data)
|
||||||
} catch {
|
} catch {
|
||||||
fputs("Error: Failed to parse input data\n", stderr)
|
fputs("Error: Failed to parse input data\n", stderr)
|
||||||
exit(1)
|
exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* View Hierarchy
|
||||||
|
* --------------
|
||||||
|
*
|
||||||
|
* stackView \ (NSStackView)
|
||||||
|
* ├ mediaTypeStackView \ (NSStackView)
|
||||||
|
* │ ├ mediaSelectCastLabel \ (NSTextField)
|
||||||
|
* │ ├ mediaTypePopUpButton \ (NSPopUpButton)
|
||||||
|
* │ └ mediaSelectToLabel \ (NSTextField)
|
||||||
|
* │
|
||||||
|
* ├ receiverSeparator \ (NSBox) ┐
|
||||||
|
* └ receiverView \ (ReceiverView:NSStackView) │
|
||||||
|
* ├ metaStackView \ (NSStackView) │
|
||||||
|
* │ ├ receiver name \ (NSTextField) ├ Repeats
|
||||||
|
* │ └ receiver host \ (NSTextField) │
|
||||||
|
* ├ spinner \ (NSProgressIndicator) │
|
||||||
|
* └ button \ (NSButton) ┘
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
let stackView = NSStackView()
|
let stackView = NSStackView()
|
||||||
stackView.orientation = .vertical
|
stackView.orientation = .vertical
|
||||||
stackView.alignment = .leading
|
stackView.alignment = .leading
|
||||||
@@ -101,7 +72,9 @@ class ViewController: NSViewController {
|
|||||||
, initData.i18n_mediaTypeScreen
|
, initData.i18n_mediaTypeScreen
|
||||||
])
|
])
|
||||||
|
|
||||||
self.mediaTypePopUpButton.selectItem(at: initData.defaultMediaType.rawValue)
|
self.mediaTypePopUpButton.selectItem(
|
||||||
|
at: initData.defaultMediaType.rawValue)
|
||||||
|
|
||||||
|
|
||||||
let mediaTypeStackView = NSStackView(views: [
|
let mediaTypeStackView = NSStackView(views: [
|
||||||
makeLabel(initData.i18n_mediaSelectCastLabel),
|
makeLabel(initData.i18n_mediaSelectCastLabel),
|
||||||
@@ -109,17 +82,29 @@ class ViewController: NSViewController {
|
|||||||
makeLabel(initData.i18n_mediaSelectToLabel)
|
makeLabel(initData.i18n_mediaSelectToLabel)
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
stackView.addArrangedSubview(mediaTypeStackView)
|
stackView.addArrangedSubview(mediaTypeStackView)
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* For each receiver in the initData list, create a new
|
||||||
|
* ReceiverView, set self as a ReceiverViewDelegate and
|
||||||
|
* appends to main stack view.
|
||||||
|
*
|
||||||
|
* Keeps a reference to the receiver view to call disable()
|
||||||
|
* later.
|
||||||
|
*/
|
||||||
for receiver in initData.receivers {
|
for receiver in initData.receivers {
|
||||||
|
// Create separator between last receiver / media type
|
||||||
let receiverSeparator = NSBox()
|
let receiverSeparator = NSBox()
|
||||||
receiverSeparator.boxType = .separator
|
receiverSeparator.boxType = .separator
|
||||||
|
|
||||||
let receiverView = ReceiverView(receiver: receiver)
|
let receiverView = ReceiverView(
|
||||||
|
receiver: receiver
|
||||||
|
, initData: self.initData)
|
||||||
|
|
||||||
receiverView.receiverViewDelegate = self
|
receiverView.receiverViewDelegate = self
|
||||||
|
|
||||||
|
|
||||||
self.receiverViews.append(receiverView)
|
self.receiverViews.append(receiverView)
|
||||||
|
|
||||||
stackView.addArrangedSubview(receiverSeparator)
|
stackView.addArrangedSubview(receiverSeparator)
|
||||||
@@ -127,17 +112,30 @@ class ViewController: NSViewController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Add to main view and set width to resize window
|
||||||
self.view.addSubview(stackView)
|
self.view.addSubview(stackView)
|
||||||
self.view.autoresizesSubviews = true
|
self.view.autoresizesSubviews = true
|
||||||
self.view.frame.size.width = 350
|
self.view.frame.size.width = 350
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override func viewDidAppear () {
|
||||||
|
// Set window title and update visibility
|
||||||
|
let window = self.view.window!
|
||||||
|
window.title = initData.i18n_extensionName
|
||||||
|
window.titleVisibility = .visible
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
extension ViewController: ReceiverViewDelegate {
|
extension ViewController : ReceiverViewDelegate {
|
||||||
func didCast (_ receiver: Receiver) {
|
func didCast (_ receiver: Receiver) {
|
||||||
|
|
||||||
|
// Disable media type UI
|
||||||
|
self.mediaTypePopUpButton.isEnabled = false
|
||||||
|
|
||||||
|
// Disable cast buttons
|
||||||
for receiverView in self.receiverViews {
|
for receiverView in self.receiverViews {
|
||||||
receiverView.disable()
|
receiverView.isEnabled = false
|
||||||
}
|
}
|
||||||
|
|
||||||
do {
|
do {
|
||||||
|
|||||||
BIN
app/NativeMacReceiverSelector/main
Executable file
BIN
app/NativeMacReceiverSelector/main
Executable file
Binary file not shown.
12
app/NativeMacReceiverSelector/models/InitData.swift
Normal file
12
app/NativeMacReceiverSelector/models/InitData.swift
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
struct InitData : Codable {
|
||||||
|
let receivers: [Receiver]
|
||||||
|
let defaultMediaType: MediaType
|
||||||
|
|
||||||
|
let i18n_extensionName: String
|
||||||
|
let i18n_castButtonTitle: String
|
||||||
|
let i18n_mediaTypeApp: String
|
||||||
|
let i18n_mediaTypeTab: String
|
||||||
|
let i18n_mediaTypeScreen: String
|
||||||
|
let i18n_mediaSelectCastLabel: String
|
||||||
|
let i18n_mediaSelectToLabel: String
|
||||||
|
}
|
||||||
3
app/NativeMacReceiverSelector/models/MediaType.swift
Normal file
3
app/NativeMacReceiverSelector/models/MediaType.swift
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
enum MediaType : Int, Codable {
|
||||||
|
case app, tab, screen
|
||||||
|
}
|
||||||
6
app/NativeMacReceiverSelector/models/Receiver.swift
Normal file
6
app/NativeMacReceiverSelector/models/Receiver.swift
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
struct Receiver : Codable {
|
||||||
|
let friendlyName: String
|
||||||
|
let host: String
|
||||||
|
let id: String
|
||||||
|
let port: Int
|
||||||
|
}
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
struct ReceiverSelection : Codable {
|
||||||
|
let receiver: Receiver
|
||||||
|
let mediaType: MediaType
|
||||||
|
}
|
||||||
19
app/NativeMacReceiverSelector/util.swift
Normal file
19
app/NativeMacReceiverSelector/util.swift
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import Cocoa
|
||||||
|
|
||||||
|
func makeLabel(_ text: String,
|
||||||
|
size: CGFloat = 0,
|
||||||
|
color: NSColor = NSColor.textColor) -> NSTextField {
|
||||||
|
|
||||||
|
let textField = NSTextField()
|
||||||
|
textField.stringValue = text
|
||||||
|
textField.backgroundColor = .clear
|
||||||
|
textField.isEditable = false
|
||||||
|
textField.isBezeled = false
|
||||||
|
textField.sizeToFit()
|
||||||
|
|
||||||
|
// Text
|
||||||
|
textField.font = NSFont.systemFont(ofSize: size)
|
||||||
|
textField.textColor = color
|
||||||
|
|
||||||
|
return textField
|
||||||
|
}
|
||||||
@@ -21,10 +21,10 @@
|
|||||||
, "popupMediaSelectToLabel": {
|
, "popupMediaSelectToLabel": {
|
||||||
"message": "to:"
|
"message": "to:"
|
||||||
}
|
}
|
||||||
, "popupCastButtonLabel": {
|
, "popupCastButtonTitle": {
|
||||||
"message": "Cast"
|
"message": "Cast"
|
||||||
}
|
}
|
||||||
, "popupCastingButtonLabel": {
|
, "popupCastingButtonTitle": {
|
||||||
"message": "Casting"
|
"message": "Casting"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,10 +6,10 @@
|
|||||||
"message": ""
|
"message": ""
|
||||||
}
|
}
|
||||||
|
|
||||||
, "popupCastButtonLabel": {
|
, "popupCastButtonTitle": {
|
||||||
"message": "Casten"
|
"message": "Casten"
|
||||||
}
|
}
|
||||||
, "popupCastingButtonLabel": {
|
, "popupCastingButtonTitle": {
|
||||||
"message": "Aan het casten"
|
"message": "Aan het casten"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -60,6 +60,9 @@ export default class NativeMacReceiverSelectorManager
|
|||||||
, data: JSON.stringify({
|
, data: JSON.stringify({
|
||||||
receivers
|
receivers
|
||||||
, defaultMediaType
|
, defaultMediaType
|
||||||
|
|
||||||
|
, i18n_extensionName: _("extensionName")
|
||||||
|
, i18n_castButtonTitle: _("popupCastButtonTitle")
|
||||||
, i18n_mediaTypeApp: _("popupMediaTypeApp")
|
, i18n_mediaTypeApp: _("popupMediaTypeApp")
|
||||||
, i18n_mediaTypeTab: _("popupMediaTypeTab")
|
, i18n_mediaTypeTab: _("popupMediaTypeTab")
|
||||||
, i18n_mediaTypeScreen: _("popupMediaTypeScreen")
|
, i18n_mediaTypeScreen: _("popupMediaTypeScreen")
|
||||||
|
|||||||
@@ -187,11 +187,11 @@ class ReceiverEntry extends Component<ReceiverEntryProps, ReceiverEntryState> {
|
|||||||
onClick={ this.handleCast }
|
onClick={ this.handleCast }
|
||||||
disabled={this.props.isLoading}>
|
disabled={this.props.isLoading}>
|
||||||
{ this.state.isLoading
|
{ this.state.isLoading
|
||||||
? _("popupCastingButtonLabel") +
|
? _("popupCastingButtonTitle") +
|
||||||
(this.state.isLoading
|
(this.state.isLoading
|
||||||
? this.state.ellipsis
|
? this.state.ellipsis
|
||||||
: "")
|
: "")
|
||||||
: _("popupCastButtonLabel") }
|
: _("popupCastButtonTitle") }
|
||||||
</button>
|
</button>
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
|
|||||||
Reference in New Issue
Block a user