Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

canvasport.py 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # PatchBay Canvas engine using QGraphicsView/Scene
  4. # Copyright (C) 2010-2019 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Global)
  19. from math import floor
  20. from PyQt5.QtCore import qCritical, Qt, QLineF, QPointF, QRectF, QTimer
  21. from PyQt5.QtGui import QCursor, QFont, QFontMetrics, QPainter, QPainterPath, QPen, QPolygonF
  22. from PyQt5.QtWidgets import QGraphicsItem, QMenu
  23. # ------------------------------------------------------------------------------------------------------------
  24. # Imports (Custom)
  25. from . import (
  26. canvas,
  27. features,
  28. options,
  29. port_mode2str,
  30. port_type2str,
  31. CanvasPortType,
  32. ANTIALIASING_FULL,
  33. ACTION_PORT_INFO,
  34. ACTION_PORT_RENAME,
  35. ACTION_PORTS_CONNECT,
  36. ACTION_PORTS_DISCONNECT,
  37. PORT_MODE_INPUT,
  38. PORT_MODE_OUTPUT,
  39. PORT_TYPE_AUDIO_JACK,
  40. PORT_TYPE_MIDI_ALSA,
  41. PORT_TYPE_MIDI_JACK,
  42. PORT_TYPE_PARAMETER,
  43. )
  44. from .canvasbezierlinemov import CanvasBezierLineMov
  45. from .canvaslinemov import CanvasLineMov
  46. from .theme import Theme
  47. from .utils import CanvasGetFullPortName, CanvasGetPortConnectionList
  48. # ------------------------------------------------------------------------------------------------------------
  49. class CanvasPort(QGraphicsItem):
  50. def __init__(self, group_id, port_id, port_name, port_mode, port_type, is_alternate, parent):
  51. QGraphicsItem.__init__(self)
  52. self.setParentItem(parent)
  53. # Save Variables, useful for later
  54. self.m_group_id = group_id
  55. self.m_port_id = port_id
  56. self.m_port_mode = port_mode
  57. self.m_port_type = port_type
  58. self.m_port_name = port_name
  59. self.m_is_alternate = is_alternate
  60. # Base Variables
  61. self.m_port_width = 15
  62. self.m_port_height = canvas.theme.port_height
  63. self.m_port_font = QFont()
  64. self.m_port_font.setFamily(canvas.theme.port_font_name)
  65. self.m_port_font.setPixelSize(canvas.theme.port_font_size)
  66. self.m_port_font.setWeight(canvas.theme.port_font_state)
  67. self.m_line_mov = None
  68. self.m_hover_item = None
  69. self.m_mouse_down = False
  70. self.m_cursor_moving = False
  71. self.setFlags(QGraphicsItem.ItemIsSelectable)
  72. if options.auto_select_items:
  73. self.setAcceptHoverEvents(True)
  74. def getGroupId(self):
  75. return self.m_group_id
  76. def getPortId(self):
  77. return self.m_port_id
  78. def getPortMode(self):
  79. return self.m_port_mode
  80. def getPortType(self):
  81. return self.m_port_type
  82. def getPortName(self):
  83. return self.m_port_name
  84. def getFullPortName(self):
  85. return self.parentItem().getGroupName() + ":" + self.m_port_name
  86. def getPortWidth(self):
  87. return self.m_port_width
  88. def getPortHeight(self):
  89. return self.m_port_height
  90. def setPortMode(self, port_mode):
  91. self.m_port_mode = port_mode
  92. self.update()
  93. def setPortType(self, port_type):
  94. self.m_port_type = port_type
  95. self.update()
  96. def setPortName(self, port_name):
  97. if QFontMetrics(self.m_port_font).width(port_name) < QFontMetrics(self.m_port_font).width(self.m_port_name):
  98. QTimer.singleShot(0, canvas.scene.update)
  99. self.m_port_name = port_name
  100. self.update()
  101. def setPortWidth(self, port_width):
  102. if port_width < self.m_port_width:
  103. QTimer.singleShot(0, canvas.scene.update)
  104. self.m_port_width = port_width
  105. self.update()
  106. def type(self):
  107. return CanvasPortType
  108. def hoverEnterEvent(self, event):
  109. if options.auto_select_items:
  110. self.setSelected(True)
  111. QGraphicsItem.hoverEnterEvent(self, event)
  112. def hoverLeaveEvent(self, event):
  113. if options.auto_select_items:
  114. self.setSelected(False)
  115. QGraphicsItem.hoverLeaveEvent(self, event)
  116. def mousePressEvent(self, event):
  117. if event.button() == Qt.MiddleButton or event.source() == Qt.MouseEventSynthesizedByApplication:
  118. event.ignore()
  119. return
  120. if self.m_mouse_down:
  121. self.handleMouseRelease()
  122. self.m_hover_item = None
  123. self.m_mouse_down = bool(event.button() == Qt.LeftButton)
  124. self.m_cursor_moving = False
  125. QGraphicsItem.mousePressEvent(self, event)
  126. def mouseMoveEvent(self, event):
  127. if not self.m_mouse_down:
  128. QGraphicsItem.mouseMoveEvent(self, event)
  129. return
  130. event.accept()
  131. if not self.m_cursor_moving:
  132. self.setCursor(QCursor(Qt.CrossCursor))
  133. self.m_cursor_moving = True
  134. for connection in canvas.connection_list:
  135. if (
  136. (connection.group_out_id == self.m_group_id and
  137. connection.port_out_id == self.m_port_id)
  138. or
  139. (connection.group_in_id == self.m_group_id and
  140. connection.port_in_id == self.m_port_id)
  141. ):
  142. connection.widget.setLocked(True)
  143. if not self.m_line_mov:
  144. if options.use_bezier_lines:
  145. self.m_line_mov = CanvasBezierLineMov(self.m_port_mode, self.m_port_type, self)
  146. else:
  147. self.m_line_mov = CanvasLineMov(self.m_port_mode, self.m_port_type, self)
  148. canvas.last_z_value += 1
  149. self.m_line_mov.setZValue(canvas.last_z_value)
  150. canvas.last_z_value += 1
  151. self.parentItem().setZValue(canvas.last_z_value)
  152. item = None
  153. items = canvas.scene.items(event.scenePos(), Qt.ContainsItemShape, Qt.AscendingOrder)
  154. #for i in range(len(items)):
  155. for _, itemx in enumerate(items):
  156. if itemx.type() != CanvasPortType:
  157. continue
  158. if itemx == self:
  159. continue
  160. if item is None or itemx.parentItem().zValue() > item.parentItem().zValue():
  161. item = itemx
  162. if self.m_hover_item and self.m_hover_item != item:
  163. self.m_hover_item.setSelected(False)
  164. if item is not None:
  165. if item.getPortMode() != self.m_port_mode and item.getPortType() == self.m_port_type:
  166. item.setSelected(True)
  167. self.m_hover_item = item
  168. else:
  169. self.m_hover_item = None
  170. else:
  171. self.m_hover_item = None
  172. self.m_line_mov.updateLinePos(event.scenePos())
  173. def handleMouseRelease(self):
  174. if self.m_mouse_down:
  175. if self.m_line_mov is not None:
  176. item = self.m_line_mov
  177. self.m_line_mov = None
  178. canvas.scene.removeItem(item)
  179. del item
  180. for connection in canvas.connection_list:
  181. if (
  182. (connection.group_out_id == self.m_group_id and
  183. connection.port_out_id == self.m_port_id)
  184. or
  185. (connection.group_in_id == self.m_group_id and
  186. connection.port_in_id == self.m_port_id)
  187. ):
  188. connection.widget.setLocked(False)
  189. if self.m_hover_item:
  190. # TODO: a better way to check already existing connection
  191. for connection in canvas.connection_list:
  192. hover_group_id = self.m_hover_item.getGroupId()
  193. hover_port_id = self.m_hover_item.getPortId()
  194. # FIXME clean this big if stuff
  195. if (
  196. (connection.group_out_id == self.m_group_id and
  197. connection.port_out_id == self.m_port_id and
  198. connection.group_in_id == hover_group_id and
  199. connection.port_in_id == hover_port_id)
  200. or
  201. (connection.group_out_id == hover_group_id and
  202. connection.port_out_id == hover_port_id and
  203. connection.group_in_id == self.m_group_id and
  204. connection.port_in_id == self.m_port_id)
  205. ):
  206. canvas.callback(ACTION_PORTS_DISCONNECT, connection.connection_id, 0, "")
  207. break
  208. else:
  209. if self.m_port_mode == PORT_MODE_OUTPUT:
  210. conn = "%i:%i:%i:%i" % (self.m_group_id, self.m_port_id,
  211. self.m_hover_item.getGroupId(), self.m_hover_item.getPortId())
  212. canvas.callback(ACTION_PORTS_CONNECT, 0, 0, conn)
  213. else:
  214. conn = "%i:%i:%i:%i" % (self.m_hover_item.getGroupId(),
  215. self.m_hover_item.getPortId(), self.m_group_id, self.m_port_id)
  216. canvas.callback(ACTION_PORTS_CONNECT, 0, 0, conn)
  217. canvas.scene.clearSelection()
  218. if self.m_cursor_moving:
  219. self.unsetCursor()
  220. self.m_hover_item = None
  221. self.m_mouse_down = False
  222. self.m_cursor_moving = False
  223. def mouseReleaseEvent(self, event):
  224. if event.button() == Qt.LeftButton:
  225. self.handleMouseRelease()
  226. QGraphicsItem.mouseReleaseEvent(self, event)
  227. def contextMenuEvent(self, event):
  228. event.accept()
  229. canvas.scene.clearSelection()
  230. self.setSelected(True)
  231. menu = QMenu()
  232. discMenu = QMenu("Disconnect", menu)
  233. conn_list = CanvasGetPortConnectionList(self.m_group_id, self.m_port_id)
  234. if len(conn_list) > 0:
  235. for conn_id, group_id, port_id in conn_list:
  236. act_x_disc = discMenu.addAction(CanvasGetFullPortName(group_id, port_id))
  237. act_x_disc.setData(conn_id)
  238. act_x_disc.triggered.connect(canvas.qobject.PortContextMenuDisconnect)
  239. else:
  240. act_x_disc = discMenu.addAction("No connections")
  241. act_x_disc.setEnabled(False)
  242. menu.addMenu(discMenu)
  243. act_x_disc_all = menu.addAction("Disconnect &All")
  244. act_x_sep_1 = menu.addSeparator()
  245. act_x_info = menu.addAction("Get &Info")
  246. act_x_rename = menu.addAction("&Rename")
  247. if not features.port_info:
  248. act_x_info.setVisible(False)
  249. if not features.port_rename:
  250. act_x_rename.setVisible(False)
  251. if not (features.port_info and features.port_rename):
  252. act_x_sep_1.setVisible(False)
  253. act_selected = menu.exec_(event.screenPos())
  254. if act_selected == act_x_disc_all:
  255. self.triggerDisconnect(conn_list)
  256. elif act_selected == act_x_info:
  257. canvas.callback(ACTION_PORT_INFO, self.m_group_id, self.m_port_id, "")
  258. elif act_selected == act_x_rename:
  259. canvas.callback(ACTION_PORT_RENAME, self.m_group_id, self.m_port_id, "")
  260. def setPortSelected(self, yesno):
  261. for connection in canvas.connection_list:
  262. if (
  263. (connection.group_out_id == self.m_group_id and
  264. connection.port_out_id == self.m_port_id)
  265. or
  266. (connection.group_in_id == self.m_group_id and
  267. connection.port_in_id == self.m_port_id)
  268. ):
  269. connection.widget.updateLineSelected()
  270. def itemChange(self, change, value):
  271. if change == QGraphicsItem.ItemSelectedHasChanged:
  272. self.setPortSelected(value)
  273. return QGraphicsItem.itemChange(self, change, value)
  274. def triggerDisconnect(self, conn_list=None):
  275. if not conn_list:
  276. conn_list = CanvasGetPortConnectionList(self.m_group_id, self.m_port_id)
  277. for conn_id, group_id, port_id in conn_list:
  278. canvas.callback(ACTION_PORTS_DISCONNECT, conn_id, 0, "")
  279. def boundingRect(self):
  280. return QRectF(0, 0, self.m_port_width + 12, self.m_port_height)
  281. def paint(self, painter, option, widget):
  282. painter.save()
  283. painter.setRenderHint(QPainter.Antialiasing, bool(options.antialiasing == ANTIALIASING_FULL))
  284. selected = self.isSelected()
  285. theme = canvas.theme
  286. if self.m_port_type == PORT_TYPE_AUDIO_JACK:
  287. poly_color = theme.port_audio_jack_bg_sel if selected else theme.port_audio_jack_bg
  288. poly_pen = theme.port_audio_jack_pen_sel if selected else theme.port_audio_jack_pen
  289. text_pen = theme.port_audio_jack_text_sel if selected else theme.port_audio_jack_text
  290. conn_pen = QPen(theme.port_audio_jack_pen_sel)
  291. elif self.m_port_type == PORT_TYPE_MIDI_JACK:
  292. poly_color = theme.port_midi_jack_bg_sel if selected else theme.port_midi_jack_bg
  293. poly_pen = theme.port_midi_jack_pen_sel if selected else theme.port_midi_jack_pen
  294. text_pen = theme.port_midi_jack_text_sel if selected else theme.port_midi_jack_text
  295. conn_pen = QPen(theme.port_midi_jack_pen_sel)
  296. elif self.m_port_type == PORT_TYPE_MIDI_ALSA:
  297. poly_color = theme.port_midi_alsa_bg_sel if selected else theme.port_midi_alsa_bg
  298. poly_pen = theme.port_midi_alsa_pen_sel if selected else theme.port_midi_alsa_pen
  299. text_pen = theme.port_midi_alsa_text_sel if selected else theme.port_midi_alsa_text
  300. conn_pen = QPen(theme.port_midi_alsa_pen_sel)
  301. elif self.m_port_type == PORT_TYPE_PARAMETER:
  302. poly_color = theme.port_parameter_bg_sel if selected else theme.port_parameter_bg
  303. poly_pen = theme.port_parameter_pen_sel if selected else theme.port_parameter_pen
  304. text_pen = theme.port_parameter_text_sel if selected else theme.port_parameter_text
  305. conn_pen = QPen(theme.port_parameter_pen_sel)
  306. else:
  307. qCritical("PatchCanvas::CanvasPort.paint() - invalid port type '%s'" % port_type2str(self.m_port_type))
  308. return
  309. # To prevent quality worsening
  310. poly_pen = QPen(poly_pen)
  311. poly_pen.setWidthF(poly_pen.widthF() + 0.00001)
  312. if self.m_is_alternate:
  313. poly_color = poly_color.darker(180)
  314. #poly_pen.setColor(poly_pen.color().darker(110))
  315. #text_pen.setColor(text_pen.color()) #.darker(150))
  316. #conn_pen.setColor(conn_pen.color()) #.darker(150))
  317. lineHinting = poly_pen.widthF() / 2
  318. poly_locx = [0, 0, 0, 0, 0]
  319. poly_corner_xhinting = (float(canvas.theme.port_height)/2) % floor(float(canvas.theme.port_height)/2)
  320. if poly_corner_xhinting == 0:
  321. poly_corner_xhinting = 0.5 * (1 - 7 / (float(canvas.theme.port_height)/2))
  322. if self.m_port_mode == PORT_MODE_INPUT:
  323. text_pos = QPointF(3, canvas.theme.port_text_ypos)
  324. if canvas.theme.port_mode == Theme.THEME_PORT_POLYGON:
  325. poly_locx[0] = lineHinting
  326. poly_locx[1] = self.m_port_width + 5 - lineHinting
  327. poly_locx[2] = self.m_port_width + 12 - poly_corner_xhinting
  328. poly_locx[3] = self.m_port_width + 5 - lineHinting
  329. poly_locx[4] = lineHinting
  330. elif canvas.theme.port_mode == Theme.THEME_PORT_SQUARE:
  331. poly_locx[0] = lineHinting
  332. poly_locx[1] = self.m_port_width + 5 - lineHinting
  333. poly_locx[2] = self.m_port_width + 5 - lineHinting
  334. poly_locx[3] = self.m_port_width + 5 - lineHinting
  335. poly_locx[4] = lineHinting
  336. else:
  337. qCritical("PatchCanvas::CanvasPort.paint() - invalid theme port mode '%s'" % canvas.theme.port_mode)
  338. return
  339. elif self.m_port_mode == PORT_MODE_OUTPUT:
  340. text_pos = QPointF(9, canvas.theme.port_text_ypos)
  341. if canvas.theme.port_mode == Theme.THEME_PORT_POLYGON:
  342. poly_locx[0] = self.m_port_width + 12 - lineHinting
  343. poly_locx[1] = 7 + lineHinting
  344. poly_locx[2] = 0 + poly_corner_xhinting
  345. poly_locx[3] = 7 + lineHinting
  346. poly_locx[4] = self.m_port_width + 12 - lineHinting
  347. elif canvas.theme.port_mode == Theme.THEME_PORT_SQUARE:
  348. poly_locx[0] = self.m_port_width + 12 - lineHinting
  349. poly_locx[1] = 5 + lineHinting
  350. poly_locx[2] = 5 + lineHinting
  351. poly_locx[3] = 5 + lineHinting
  352. poly_locx[4] = self.m_port_width + 12 - lineHinting
  353. else:
  354. qCritical("PatchCanvas::CanvasPort.paint() - invalid theme port mode '%s'" % canvas.theme.port_mode)
  355. return
  356. else:
  357. qCritical("PatchCanvas::CanvasPort.paint() - invalid port mode '%s'" % port_mode2str(self.m_port_mode))
  358. return
  359. polygon = QPolygonF()
  360. polygon += QPointF(poly_locx[0], lineHinting)
  361. polygon += QPointF(poly_locx[1], lineHinting)
  362. polygon += QPointF(poly_locx[2], float(canvas.theme.port_height)/2)
  363. polygon += QPointF(poly_locx[3], canvas.theme.port_height - lineHinting)
  364. polygon += QPointF(poly_locx[4], canvas.theme.port_height - lineHinting)
  365. polygon += QPointF(poly_locx[0], lineHinting)
  366. if canvas.theme.port_bg_pixmap:
  367. portRect = polygon.boundingRect().adjusted(-lineHinting+1, -lineHinting+1, lineHinting-1, lineHinting-1)
  368. portPos = portRect.topLeft()
  369. painter.drawTiledPixmap(portRect, canvas.theme.port_bg_pixmap, portPos)
  370. else:
  371. painter.setBrush(poly_color) #.lighter(200))
  372. painter.setPen(poly_pen)
  373. painter.drawPolygon(polygon)
  374. painter.setPen(text_pen)
  375. painter.setFont(self.m_port_font)
  376. painter.drawText(text_pos, self.m_port_name)
  377. if canvas.theme.idx == Theme.THEME_OOSTUDIO and canvas.theme.port_bg_pixmap:
  378. conn_pen.setCosmetic(True)
  379. conn_pen.setWidthF(0.4)
  380. painter.setPen(conn_pen)
  381. if self.m_port_mode == PORT_MODE_INPUT:
  382. connLineX = portRect.left()+1
  383. else:
  384. connLineX = portRect.right()-1
  385. conn_path = QPainterPath()
  386. conn_path.addRect(QRectF(connLineX-1, portRect.top(), 2, portRect.height()))
  387. painter.fillPath(conn_path, conn_pen.brush())
  388. painter.drawLine(QLineF(connLineX, portRect.top(), connLineX, portRect.bottom()))
  389. painter.restore()
  390. # ------------------------------------------------------------------------------------------------------------