« get me outta code hell

mtui - Music Text User Interface - user-friendly command line music player
about summary refs log tree commit diff
path: root/ui.js
diff options
context:
space:
mode:
Diffstat (limited to 'ui.js')
-rw-r--r--ui.js131
1 files changed, 123 insertions, 8 deletions
diff --git a/ui.js b/ui.js
index 0a1cbca..378cd77 100644
--- a/ui.js
+++ b/ui.js
@@ -138,6 +138,8 @@ class AppElement extends FocusElement {
     super()
 
     this.backend = backend
+    this.telnetServer = null
+    this.isPartyHost = false
 
     this.bindListeners()
     this.attachListeners()
@@ -212,6 +214,21 @@ class AppElement extends FocusElement {
     this.playbackInfoElement.on('seek ahead', () => this.backend.seekAhead(5))
     this.playbackInfoElement.on('toggle pause', () => this.backend.togglePause())
 
+    this.partyTop = new DisplayElement()
+    this.partyBottom = new DisplayElement()
+    this.addChild(this.partyTop)
+    this.addChild(this.partyBottom)
+    this.partyTop.visible = false
+    this.partyBottom.visible = false
+
+    this.partyTopBanner = new PartyBanner(1)
+    this.partyBottomBanner = new PartyBanner(-1)
+    this.partyTop.addChild(this.partyTopBanner)
+    this.partyBottom.addChild(this.partyBottomBanner)
+
+    this.partyLabel = new Label('')
+    this.partyTop.addChild(this.partyLabel)
+
     // Dialogs
 
     this.openPlaylistDialog = new OpenPlaylistDialog()
@@ -666,12 +683,42 @@ class AppElement extends FocusElement {
 
     this.menubar.fixLayout()
 
-    const mainHeight = this.contentH - 6
+    let topY = this.contentH
+
+    if (this.partyBottom.visible) {
+      this.partyBottom.w = this.contentW
+      this.partyBottom.h = 1
+      this.partyBottom.x = 0
+      this.partyBottom.y = topY - this.partyBottom.h
+      topY = this.partyBottom.top
+      this.partyBottomBanner.w = this.partyBottom.w
+    }
+
+    this.playbackPane.w = this.contentW
+    this.playbackPane.h = 5
+    this.playbackPane.x = 0
+    this.playbackPane.y = topY - this.playbackPane.h
+    topY = this.playbackPane.top
+
+    let bottomY = 1
+
+    if (this.partyTop.visible) {
+      this.partyTop.w = this.contentW
+      this.partyTop.h = 1
+      this.partyTop.x = 0
+      this.partyTop.y = 1
+      bottomY = this.partyTop.bottom
+
+      this.partyTopBanner.w = this.partyTop.w
+      this.partyTopBanner.y = this.partyTop.contentH - 1
+
+      this.alignPartyLabel()
+    }
 
     if (this.paneLeft.visible) {
       this.paneLeft.w = Math.max(Math.floor(0.8 * this.contentW), this.contentW - 80)
-      this.paneLeft.h = mainHeight
-      this.paneLeft.y = 1
+      this.paneLeft.y = bottomY
+      this.paneLeft.h = topY - this.paneLeft.y
       this.paneRight.x = this.paneLeft.right
       this.paneRight.w = this.contentW - this.paneLeft.right
     } else {
@@ -679,11 +726,9 @@ class AppElement extends FocusElement {
       this.paneRight.w = this.contentW
     }
 
-    this.paneRight.y = 1
-    this.paneRight.h = mainHeight
-    this.playbackPane.y = mainHeight + 1
-    this.playbackPane.w = this.contentW
-    this.playbackPane.h = this.contentH - this.playbackPane.y
+    this.paneRight.y = bottomY
+    this.paneRight.h = topY - this.paneRight.y
+    topY = this.paneRight.y
 
     this.tabber.fillParent()
 
@@ -702,6 +747,42 @@ class AppElement extends FocusElement {
     this.playbackInfoElement.fillParent()
   }
 
+  alignPartyLabel() {
+    this.partyLabel.centerInParent()
+    this.partyLabel.y = 0
+  }
+
+  attachAsServerHost(telnetServer) {
+    this.isPartyHost = true
+    this.attachAsServer(telnetServer)
+  }
+
+  attachAsServerClient(telnetServer) {
+    this.isPartyHost = false
+    this.attachAsServer(telnetServer)
+  }
+
+  attachAsServer(telnetServer) {
+    this.telnetServer = telnetServer
+    this.updatePartyLabel()
+
+    this.telnetServer.on('joined', () => this.updatePartyLabel())
+    this.telnetServer.on('left', () => this.updatePartyLabel())
+
+    this.partyTop.visible = true
+    this.partyBottom.visible = true
+    this.fixLayout()
+  }
+
+  updatePartyLabel() {
+    const clients = this.telnetServer.sockets.length
+    const clientsMsg = clients === 1 ? '1-ish connection' : `${clients}-ish connections`
+    let msg = `${process.env.USER} playing for ${clientsMsg}`
+
+    this.partyLabel.text = `  ${msg}  `
+    this.alignPartyLabel()
+  }
+
   keyPressed(keyBuf) {
     if (keyBuf[0] === 0x03) { // Ctrl-C
       this.shutdown()
@@ -2787,4 +2868,38 @@ class Menubar extends ListScrollForm {
   }
 }
 
+class PartyBanner extends DisplayElement {
+  constructor(direction) {
+    super()
+
+    this.direction = direction
+  }
+
+  drawTo(writable) {
+    writable.write(ansi.moveCursor(this.absTop, this.absLeft))
+
+    const timerNum = Date.now() / 2000 * this.direction
+    let lastAttribute = ''
+    const updateAttribute = offsetNum => {
+      const attr = (Math.cos(offsetNum - timerNum) < 0 ? '\x1b[0;1m' : '\x1b[0;2m')
+      if (attr === lastAttribute) {
+        return ''
+      } else {
+        lastAttribute = attr
+        return attr
+      }
+    }
+    let str = new Array(this.w).fill('0').map((_, i) => {
+      const offsetNum = i / this.w * 2 * Math.PI
+      return (
+        updateAttribute(offsetNum) +
+        (Math.sin(offsetNum + timerNum) < 0 ? '-' : '*')
+      )
+    }).join('')
+
+    writable.write(str)
+    writable.write(ansi.resetAttributes())
+  }
+}
+
 module.exports = AppElement