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 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  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. self.m_hover_item = None
  118. self.m_mouse_down = bool(event.button() == Qt.LeftButton)
  119. self.m_cursor_moving = False
  120. QGraphicsItem.mousePressEvent(self, event)
  121. def mouseMoveEvent(self, event):
  122. if not self.m_mouse_down:
  123. QGraphicsItem.mouseMoveEvent(self, event)
  124. return
  125. event.accept()
  126. if not self.m_cursor_moving:
  127. self.setCursor(QCursor(Qt.CrossCursor))
  128. self.m_cursor_moving = True
  129. for connection in canvas.connection_list:
  130. if (
  131. (connection.group_out_id == self.m_group_id and
  132. connection.port_out_id == self.m_port_id)
  133. or
  134. (connection.group_in_id == self.m_group_id and
  135. connection.port_in_id == self.m_port_id)
  136. ):
  137. connection.widget.setLocked(True)
  138. if not self.m_line_mov:
  139. if options.use_bezier_lines:
  140. self.m_line_mov = CanvasBezierLineMov(self.m_port_mode, self.m_port_type, self)
  141. else:
  142. self.m_line_mov = CanvasLineMov(self.m_port_mode, self.m_port_type, self)
  143. canvas.last_z_value += 1
  144. self.m_line_mov.setZValue(canvas.last_z_value)
  145. canvas.last_z_value += 1
  146. self.parentItem().setZValue(canvas.last_z_value)
  147. item = None
  148. items = canvas.scene.items(event.scenePos(), Qt.ContainsItemShape, Qt.AscendingOrder)
  149. #for i in range(len(items)):
  150. for _, itemx in enumerate(items):
  151. if itemx.type() != CanvasPortType:
  152. continue
  153. if itemx == self:
  154. continue
  155. if item is None or itemx.parentItem().zValue() > item.parentItem().zValue():
  156. item = itemx
  157. if self.m_hover_item and self.m_hover_item != item:
  158. self.m_hover_item.setSelected(False)
  159. if item is not None:
  160. if item.getPortMode() != self.m_port_mode and item.getPortType() == self.m_port_type:
  161. item.setSelected(True)
  162. self.m_hover_item = item
  163. else:
  164. self.m_hover_item = None
  165. else:
  166. self.m_hover_item = None
  167. self.m_line_mov.updateLinePos(event.scenePos())
  168. def mouseReleaseEvent(self, event):
  169. if self.m_mouse_down:
  170. if self.m_line_mov is not None:
  171. item = self.m_line_mov
  172. self.m_line_mov = None
  173. canvas.scene.removeItem(item)
  174. del item
  175. for connection in canvas.connection_list:
  176. if (
  177. (connection.group_out_id == self.m_group_id and
  178. connection.port_out_id == self.m_port_id)
  179. or
  180. (connection.group_in_id == self.m_group_id and
  181. connection.port_in_id == self.m_port_id)
  182. ):
  183. connection.widget.setLocked(False)
  184. if self.m_hover_item:
  185. # TODO: a better way to check already existing connection
  186. for connection in canvas.connection_list:
  187. hover_group_id = self.m_hover_item.getGroupId()
  188. hover_port_id = self.m_hover_item.getPortId()
  189. # FIXME clean ths big if stuff
  190. if (
  191. (connection.group_out_id == self.m_group_id and
  192. connection.port_out_id == self.m_port_id and
  193. connection.group_in_id == hover_group_id and
  194. connection.port_in_id == hover_port_id)
  195. or
  196. (connection.group_out_id == hover_group_id and
  197. connection.port_out_id == hover_port_id and
  198. connection.group_in_id == self.m_group_id and
  199. connection.port_in_id == self.m_port_id)
  200. ):
  201. canvas.callback(ACTION_PORTS_DISCONNECT, connection.connection_id, 0, "")
  202. break
  203. else:
  204. if self.m_port_mode == PORT_MODE_OUTPUT:
  205. conn = "%i:%i:%i:%i" % (self.m_group_id, self.m_port_id,
  206. self.m_hover_item.getGroupId(), self.m_hover_item.getPortId())
  207. canvas.callback(ACTION_PORTS_CONNECT, 0, 0, conn)
  208. else:
  209. conn = "%i:%i:%i:%i" % (self.m_hover_item.getGroupId(),
  210. self.m_hover_item.getPortId(), self.m_group_id, self.m_port_id)
  211. canvas.callback(ACTION_PORTS_CONNECT, 0, 0, conn)
  212. canvas.scene.clearSelection()
  213. if self.m_cursor_moving:
  214. self.unsetCursor()
  215. self.m_hover_item = None
  216. self.m_mouse_down = False
  217. self.m_cursor_moving = False
  218. QGraphicsItem.mouseReleaseEvent(self, event)
  219. def contextMenuEvent(self, event):
  220. event.accept()
  221. canvas.scene.clearSelection()
  222. self.setSelected(True)
  223. menu = QMenu()
  224. discMenu = QMenu("Disconnect", menu)
  225. conn_list = CanvasGetPortConnectionList(self.m_group_id, self.m_port_id)
  226. if len(conn_list) > 0:
  227. for conn_id, group_id, port_id in conn_list:
  228. act_x_disc = discMenu.addAction(CanvasGetFullPortName(group_id, port_id))
  229. act_x_disc.setData(conn_id)
  230. act_x_disc.triggered.connect(canvas.qobject.PortContextMenuDisconnect)
  231. else:
  232. act_x_disc = discMenu.addAction("No connections")
  233. act_x_disc.setEnabled(False)
  234. menu.addMenu(discMenu)
  235. act_x_disc_all = menu.addAction("Disconnect &All")
  236. act_x_sep_1 = menu.addSeparator()
  237. act_x_info = menu.addAction("Get &Info")
  238. act_x_rename = menu.addAction("&Rename")
  239. if not features.port_info:
  240. act_x_info.setVisible(False)
  241. if not features.port_rename:
  242. act_x_rename.setVisible(False)
  243. if not (features.port_info and features.port_rename):
  244. act_x_sep_1.setVisible(False)
  245. act_selected = menu.exec_(event.screenPos())
  246. if act_selected == act_x_disc_all:
  247. self.triggerDisconnect(conn_list)
  248. elif act_selected == act_x_info:
  249. canvas.callback(ACTION_PORT_INFO, self.m_group_id, self.m_port_id, "")
  250. elif act_selected == act_x_rename:
  251. canvas.callback(ACTION_PORT_RENAME, self.m_group_id, self.m_port_id, "")
  252. def setPortSelected(self, yesno):
  253. for connection in canvas.connection_list:
  254. if (
  255. (connection.group_out_id == self.m_group_id and
  256. connection.port_out_id == self.m_port_id)
  257. or
  258. (connection.group_in_id == self.m_group_id and
  259. connection.port_in_id == self.m_port_id)
  260. ):
  261. connection.widget.updateLineSelected()
  262. def itemChange(self, change, value):
  263. if change == QGraphicsItem.ItemSelectedHasChanged:
  264. self.setPortSelected(value)
  265. return QGraphicsItem.itemChange(self, change, value)
  266. def triggerDisconnect(self, conn_list=None):
  267. if not conn_list:
  268. conn_list = CanvasGetPortConnectionList(self.m_group_id, self.m_port_id)
  269. for conn_id, group_id, port_id in conn_list:
  270. canvas.callback(ACTION_PORTS_DISCONNECT, conn_id, 0, "")
  271. def boundingRect(self):
  272. return QRectF(0, 0, self.m_port_width + 12, self.m_port_height)
  273. def paint(self, painter, option, widget):
  274. painter.save()
  275. painter.setRenderHint(QPainter.Antialiasing, bool(options.antialiasing == ANTIALIASING_FULL))
  276. selected = self.isSelected()
  277. theme = canvas.theme
  278. if self.m_port_type == PORT_TYPE_AUDIO_JACK:
  279. poly_color = theme.port_audio_jack_bg_sel if selected else theme.port_audio_jack_bg
  280. poly_pen = theme.port_audio_jack_pen_sel if selected else theme.port_audio_jack_pen
  281. text_pen = theme.port_audio_jack_text_sel if selected else theme.port_audio_jack_text
  282. conn_pen = QPen(theme.port_audio_jack_pen_sel)
  283. elif self.m_port_type == PORT_TYPE_MIDI_JACK:
  284. poly_color = theme.port_midi_jack_bg_sel if selected else theme.port_midi_jack_bg
  285. poly_pen = theme.port_midi_jack_pen_sel if selected else theme.port_midi_jack_pen
  286. text_pen = theme.port_midi_jack_text_sel if selected else theme.port_midi_jack_text
  287. conn_pen = QPen(theme.port_midi_jack_pen_sel)
  288. elif self.m_port_type == PORT_TYPE_MIDI_ALSA:
  289. poly_color = theme.port_midi_alsa_bg_sel if selected else theme.port_midi_alsa_bg
  290. poly_pen = theme.port_midi_alsa_pen_sel if selected else theme.port_midi_alsa_pen
  291. text_pen = theme.port_midi_alsa_text_sel if selected else theme.port_midi_alsa_text
  292. conn_pen = QPen(theme.port_midi_alsa_pen_sel)
  293. elif self.m_port_type == PORT_TYPE_PARAMETER:
  294. poly_color = theme.port_parameter_bg_sel if selected else theme.port_parameter_bg
  295. poly_pen = theme.port_parameter_pen_sel if selected else theme.port_parameter_pen
  296. text_pen = theme.port_parameter_text_sel if selected else theme.port_parameter_text
  297. conn_pen = QPen(theme.port_parameter_pen_sel)
  298. else:
  299. qCritical("PatchCanvas::CanvasPort.paint() - invalid port type '%s'" % port_type2str(self.m_port_type))
  300. return
  301. # To prevent quality worsening
  302. poly_pen = QPen(poly_pen)
  303. poly_pen.setWidthF(poly_pen.widthF() + 0.00001)
  304. if self.m_is_alternate:
  305. poly_color = poly_color.darker(180)
  306. #poly_pen.setColor(poly_pen.color().darker(110))
  307. #text_pen.setColor(text_pen.color()) #.darker(150))
  308. #conn_pen.setColor(conn_pen.color()) #.darker(150))
  309. lineHinting = poly_pen.widthF() / 2
  310. poly_locx = [0, 0, 0, 0, 0]
  311. poly_corner_xhinting = (float(canvas.theme.port_height)/2) % floor(float(canvas.theme.port_height)/2)
  312. if poly_corner_xhinting == 0:
  313. poly_corner_xhinting = 0.5 * (1 - 7 / (float(canvas.theme.port_height)/2))
  314. if self.m_port_mode == PORT_MODE_INPUT:
  315. text_pos = QPointF(3, canvas.theme.port_text_ypos)
  316. if canvas.theme.port_mode == Theme.THEME_PORT_POLYGON:
  317. poly_locx[0] = lineHinting
  318. poly_locx[1] = self.m_port_width + 5 - lineHinting
  319. poly_locx[2] = self.m_port_width + 12 - poly_corner_xhinting
  320. poly_locx[3] = self.m_port_width + 5 - lineHinting
  321. poly_locx[4] = lineHinting
  322. elif canvas.theme.port_mode == Theme.THEME_PORT_SQUARE:
  323. poly_locx[0] = lineHinting
  324. poly_locx[1] = self.m_port_width + 5 - lineHinting
  325. poly_locx[2] = self.m_port_width + 5 - lineHinting
  326. poly_locx[3] = self.m_port_width + 5 - lineHinting
  327. poly_locx[4] = lineHinting
  328. else:
  329. qCritical("PatchCanvas::CanvasPort.paint() - invalid theme port mode '%s'" % canvas.theme.port_mode)
  330. return
  331. elif self.m_port_mode == PORT_MODE_OUTPUT:
  332. text_pos = QPointF(9, canvas.theme.port_text_ypos)
  333. if canvas.theme.port_mode == Theme.THEME_PORT_POLYGON:
  334. poly_locx[0] = self.m_port_width + 12 - lineHinting
  335. poly_locx[1] = 7 + lineHinting
  336. poly_locx[2] = 0 + poly_corner_xhinting
  337. poly_locx[3] = 7 + lineHinting
  338. poly_locx[4] = self.m_port_width + 12 - lineHinting
  339. elif canvas.theme.port_mode == Theme.THEME_PORT_SQUARE:
  340. poly_locx[0] = self.m_port_width + 12 - lineHinting
  341. poly_locx[1] = 5 + lineHinting
  342. poly_locx[2] = 5 + lineHinting
  343. poly_locx[3] = 5 + lineHinting
  344. poly_locx[4] = self.m_port_width + 12 - lineHinting
  345. else:
  346. qCritical("PatchCanvas::CanvasPort.paint() - invalid theme port mode '%s'" % canvas.theme.port_mode)
  347. return
  348. else:
  349. qCritical("PatchCanvas::CanvasPort.paint() - invalid port mode '%s'" % port_mode2str(self.m_port_mode))
  350. return
  351. polygon = QPolygonF()
  352. polygon += QPointF(poly_locx[0], lineHinting)
  353. polygon += QPointF(poly_locx[1], lineHinting)
  354. polygon += QPointF(poly_locx[2], float(canvas.theme.port_height)/2)
  355. polygon += QPointF(poly_locx[3], canvas.theme.port_height - lineHinting)
  356. polygon += QPointF(poly_locx[4], canvas.theme.port_height - lineHinting)
  357. polygon += QPointF(poly_locx[0], lineHinting)
  358. if canvas.theme.port_bg_pixmap:
  359. portRect = polygon.boundingRect().adjusted(-lineHinting+1, -lineHinting+1, lineHinting-1, lineHinting-1)
  360. portPos = portRect.topLeft()
  361. painter.drawTiledPixmap(portRect, canvas.theme.port_bg_pixmap, portPos)
  362. else:
  363. painter.setBrush(poly_color) #.lighter(200))
  364. painter.setPen(poly_pen)
  365. painter.drawPolygon(polygon)
  366. painter.setPen(text_pen)
  367. painter.setFont(self.m_port_font)
  368. painter.drawText(text_pos, self.m_port_name)
  369. if canvas.theme.idx == Theme.THEME_OOSTUDIO and canvas.theme.port_bg_pixmap:
  370. conn_pen.setCosmetic(True)
  371. conn_pen.setWidthF(0.4)
  372. painter.setPen(conn_pen)
  373. if self.m_port_mode == PORT_MODE_INPUT:
  374. connLineX = portRect.left()+1
  375. else:
  376. connLineX = portRect.right()-1
  377. conn_path = QPainterPath()
  378. conn_path.addRect(QRectF(connLineX-1, portRect.top(), 2, portRect.height()))
  379. painter.fillPath(conn_path, conn_pen.brush())
  380. painter.drawLine(QLineF(connLineX, portRect.top(), connLineX, portRect.bottom()))
  381. painter.restore()
  382. # ------------------------------------------------------------------------------------------------------------