Add missing i18n to native macOS receiver selector and refactor

This commit is contained in:
hensm
2019-05-15 13:38:11 +01:00
parent 474dbad1aa
commit e368c4d09c
13 changed files with 155 additions and 108 deletions

View File

@@ -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
} }
} }

View File

@@ -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
} }
} }

View File

@@ -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 {

Binary file not shown.

View 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
}

View File

@@ -0,0 +1,3 @@
enum MediaType : Int, Codable {
case app, tab, screen
}

View File

@@ -0,0 +1,6 @@
struct Receiver : Codable {
let friendlyName: String
let host: String
let id: String
let port: Int
}

View File

@@ -0,0 +1,4 @@
struct ReceiverSelection : Codable {
let receiver: Receiver
let mediaType: MediaType
}

View 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
}

View File

@@ -21,10 +21,10 @@
, "popupMediaSelectToLabel": { , "popupMediaSelectToLabel": {
"message": "to:" "message": "to:"
} }
, "popupCastButtonLabel": { , "popupCastButtonTitle": {
"message": "Cast" "message": "Cast"
} }
, "popupCastingButtonLabel": { , "popupCastingButtonTitle": {
"message": "Casting" "message": "Casting"
} }

View File

@@ -6,10 +6,10 @@
"message": "" "message": ""
} }
, "popupCastButtonLabel": { , "popupCastButtonTitle": {
"message": "Casten" "message": "Casten"
} }
, "popupCastingButtonLabel": { , "popupCastingButtonTitle": {
"message": "Aan het casten" "message": "Aan het casten"
} }

View File

@@ -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")

View File

@@ -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>
); );