diff options
author | Friedemann Kleint <Friedemann.Kleint@qt.io> | 2020-10-12 08:17:07 +0200 |
---|---|---|
committer | Friedemann Kleint <Friedemann.Kleint@qt.io> | 2020-10-12 08:17:07 +0200 |
commit | 30724622333ffc8bce61f7e19217977eebbf9564 (patch) | |
tree | 172487d1478cc478857d46cb445db59ca3601448 | |
parent | 1a66d7be29902fc041fe53ef280c237d3c3f216b (diff) | |
parent | 8847a47aad95d7f85d5e184071bf95c44826c4c7 (diff) |
Merge remote-tracking branch 'origin/5.15' into dev
Change-Id: I7f16e4f178bff9c37af3b7a9d536c5d6f42c66fb
-rw-r--r-- | examples/tutorial/t14.py | 2 | ||||
-rw-r--r-- | examples/widgets/graphicsview/collidingmice/collidingmice.py | 16 | ||||
-rw-r--r-- | examples/widgets/graphicsview/dragdroprobot/dragdroprobot.py | 11 | ||||
-rw-r--r-- | examples/widgets/graphicsview/elasticnodes.py | 7 | ||||
-rw-r--r-- | examples/widgets/itemviews/basicsortfiltermodel.py | 119 | ||||
-rw-r--r-- | sources/pyside2/libpyside/feature_select.cpp | 79 | ||||
-rw-r--r-- | sources/shiboken2/libshiboken/pep384impl.cpp | 3 | ||||
-rw-r--r-- | sources/shiboken2/libshiboken/pep384impl.h | 2 |
8 files changed, 166 insertions, 73 deletions
diff --git a/examples/tutorial/t14.py b/examples/tutorial/t14.py index 9096188b7..d86f59166 100644 --- a/examples/tutorial/t14.py +++ b/examples/tutorial/t14.py @@ -310,7 +310,7 @@ class CannonField(QtWidgets.QWidget): return QtCore.QRect(145, self.height() - 100, 15, 99) def barrelHit(self, pos): - matrix = QtGui.QMatrix() + matrix = QtGui.QTransform() matrix.translate(0, self.height()) matrix.rotate(-self.currentAngle) matrix, invertible = matrix.inverted() diff --git a/examples/widgets/graphicsview/collidingmice/collidingmice.py b/examples/widgets/graphicsview/collidingmice/collidingmice.py index 08a62d0e0..2203cb381 100644 --- a/examples/widgets/graphicsview/collidingmice/collidingmice.py +++ b/examples/widgets/graphicsview/collidingmice/collidingmice.py @@ -47,6 +47,10 @@ from PySide2 import QtCore, QtGui, QtWidgets import mice_rc +def random(boundary): + return QtCore.QRandomGenerator.global_().bounded(boundary) + + class Mouse(QtWidgets.QGraphicsItem): Pi = math.pi TwoPi = 2.0 * Pi @@ -62,10 +66,9 @@ class Mouse(QtWidgets.QGraphicsItem): self.angle = 0.0 self.speed = 0.0 self.mouseEyeDirection = 0.0 - self.color = QtGui.QColor(QtCore.qrand() % 256, QtCore.qrand() % 256, - QtCore.qrand() % 256) + self.color = QtGui.QColor(random(256), random(256), random(256)) - self.setTransform(QtGui.QTransform().rotate(QtCore.qrand() % (360 * 16))) + self.setTransform(QtGui.QTransform().rotate(random(360 * 16))) # In the C++ version of this example, this class is also derived from # QObject in order to receive timer events. PySide2 does not support @@ -171,11 +174,11 @@ class Mouse(QtWidgets.QGraphicsItem): # Add some random movement. if len(dangerMice) > 1 and (QtCore.qrand() % 10) == 0: if QtCore.qrand() % 1: - self.angle += (QtCore.qrand() % 100) / 500.0 + self.angle += random(100) / 500.0 else: - self.angle -= (QtCore.qrand() % 100) / 500.0 + self.angle -= random(100) / 500.0 - self.speed += (-50 + QtCore.qrand() % 100) / 100.0 + self.speed += (-50 + random(100)) / 100.0 dx = math.sin(self.angle) * 10 self.mouseEyeDirection = [dx / 5, 0.0][QtCore.qAbs(dx / 5) < 1] @@ -191,7 +194,6 @@ if __name__ == '__main__': MouseCount = 7 app = QtWidgets.QApplication(sys.argv) - QtCore.qsrand(QtCore.QTime(0,0,0).secsTo(QtCore.QTime.currentTime())) scene = QtWidgets.QGraphicsScene() scene.setSceneRect(-300, -300, 600, 600) diff --git a/examples/widgets/graphicsview/dragdroprobot/dragdroprobot.py b/examples/widgets/graphicsview/dragdroprobot/dragdroprobot.py index 035358e65..8defc6d07 100644 --- a/examples/widgets/graphicsview/dragdroprobot/dragdroprobot.py +++ b/examples/widgets/graphicsview/dragdroprobot/dragdroprobot.py @@ -45,14 +45,17 @@ from PySide2 import QtCore, QtGui, QtWidgets import dragdroprobot_rc +def random(boundary): + return QtCore.QRandomGenerator.global_().bounded(boundary) + + class ColorItem(QtWidgets.QGraphicsItem): n = 0 def __init__(self): super(ColorItem, self).__init__() - self.color = QtGui.QColor(QtCore.qrand() % 256, QtCore.qrand() % 256, - QtCore.qrand() % 256) + self.color = QtGui.QColor(random(256), random(256), random(256)) self.setToolTip( "QColor(%d, %d, %d)\nClick and drag this color onto the robot!" % @@ -87,7 +90,7 @@ class ColorItem(QtWidgets.QGraphicsItem): drag.setMimeData(mime) ColorItem.n += 1 - if ColorItem.n > 2 and QtCore.qrand() % 3 == 0: + if ColorItem.n > 2 and random(3) == 0: image = QtGui.QImage(':/images/head.png') mime.setImageData(image) drag.setPixmap(QtGui.QPixmap.fromImage(image).scaled(30,40)) @@ -262,8 +265,6 @@ if __name__== '__main__': app = QtWidgets.QApplication(sys.argv) - QtCore.qsrand(QtCore.QTime(0, 0, 0).secsTo(QtCore.QTime.currentTime())) - scene = QtWidgets.QGraphicsScene(-200, -200, 400, 400) for i in range(10): diff --git a/examples/widgets/graphicsview/elasticnodes.py b/examples/widgets/graphicsview/elasticnodes.py index 48feffc85..f5d229b13 100644 --- a/examples/widgets/graphicsview/elasticnodes.py +++ b/examples/widgets/graphicsview/elasticnodes.py @@ -46,6 +46,10 @@ import math from PySide2 import QtCore, QtGui, QtWidgets +def random(boundary): + return QtCore.QRandomGenerator.global_().bounded(boundary) + + class Edge(QtWidgets.QGraphicsItem): Pi = math.pi TwoPi = 2.0 * Pi @@ -338,7 +342,7 @@ class GraphWidget(QtWidgets.QGraphicsView): elif key == QtCore.Qt.Key_Space or key == QtCore.Qt.Key_Enter: for item in self.scene().items(): if isinstance(item, Node): - item.setPos(-150 + QtCore.qrand() % 300, -150 + QtCore.qrand() % 300) + item.setPos(-150 + random(300), -150 + random(300)) else: QtWidgets.QGraphicsView.keyPressEvent(self, event) @@ -405,7 +409,6 @@ class GraphWidget(QtWidgets.QGraphicsView): if __name__ == "__main__": app = QtWidgets.QApplication(sys.argv) - QtCore.qsrand(QtCore.QTime(0,0,0).secsTo(QtCore.QTime.currentTime())) widget = GraphWidget() widget.show() diff --git a/examples/widgets/itemviews/basicsortfiltermodel.py b/examples/widgets/itemviews/basicsortfiltermodel.py index 4aea218d2..00441ffe7 100644 --- a/examples/widgets/itemviews/basicsortfiltermodel.py +++ b/examples/widgets/itemviews/basicsortfiltermodel.py @@ -2,7 +2,7 @@ ############################################################################# ## ## Copyright (C) 2013 Riverbank Computing Limited. -## Copyright (C) 2016 The Qt Company Ltd. +## Copyright (C) 2020 The Qt Company Ltd. ## Contact: http://www.qt.io/licensing/ ## ## This file is part of the Qt for Python examples of the Qt Toolkit. @@ -40,51 +40,63 @@ ## ############################################################################# -from PySide2 import QtCore, QtGui, QtWidgets +import sys +from PySide2.QtCore import (QDate, QDateTime, QRegularExpression, + QSortFilterProxyModel, QTime, Qt) +from PySide2.QtGui import QStandardItemModel +from PySide2.QtWidgets import (QApplication, QCheckBox, QComboBox, QGridLayout, + QGroupBox, QHBoxLayout, QLabel, QLineEdit, + QTreeView, QVBoxLayout, QWidget) -class Window(QtWidgets.QWidget): +REGULAR_EXPRESSION = 0 +WILDCARD = 1 +FIXED_STRING = 2 + + +class Window(QWidget): def __init__(self): super(Window, self).__init__() - self.proxyModel = QtCore.QSortFilterProxyModel() + self.proxyModel = QSortFilterProxyModel() self.proxyModel.setDynamicSortFilter(True) - self.sourceGroupBox = QtWidgets.QGroupBox("Original Model") - self.proxyGroupBox = QtWidgets.QGroupBox("Sorted/Filtered Model") + self.sourceGroupBox = QGroupBox("Original Model") + self.proxyGroupBox = QGroupBox("Sorted/Filtered Model") - self.sourceView = QtWidgets.QTreeView() + self.sourceView = QTreeView() self.sourceView.setRootIsDecorated(False) self.sourceView.setAlternatingRowColors(True) - self.proxyView = QtWidgets.QTreeView() + self.proxyView = QTreeView() self.proxyView.setRootIsDecorated(False) self.proxyView.setAlternatingRowColors(True) self.proxyView.setModel(self.proxyModel) self.proxyView.setSortingEnabled(True) - self.sortCaseSensitivityCheckBox = QtWidgets.QCheckBox("Case sensitive sorting") - self.filterCaseSensitivityCheckBox = QtWidgets.QCheckBox("Case sensitive filter") + self.sortCaseSensitivityCheckBox = QCheckBox("Case sensitive sorting") + self.filterCaseSensitivityCheckBox = QCheckBox("Case sensitive filter") - self.filterPatternLineEdit = QtWidgets.QLineEdit() - self.filterPatternLabel = QtWidgets.QLabel("&Filter pattern:") + self.filterPatternLineEdit = QLineEdit() + self.filterPatternLineEdit.setClearButtonEnabled(True) + self.filterPatternLabel = QLabel("&Filter pattern:") self.filterPatternLabel.setBuddy(self.filterPatternLineEdit) - self.filterSyntaxComboBox = QtWidgets.QComboBox() + self.filterSyntaxComboBox = QComboBox() self.filterSyntaxComboBox.addItem("Regular expression", - QtCore.QRegExp.RegExp) + REGULAR_EXPRESSION) self.filterSyntaxComboBox.addItem("Wildcard", - QtCore.QRegExp.Wildcard) + WILDCARD) self.filterSyntaxComboBox.addItem("Fixed string", - QtCore.QRegExp.FixedString) - self.filterSyntaxLabel = QtWidgets.QLabel("Filter &syntax:") + FIXED_STRING) + self.filterSyntaxLabel = QLabel("Filter &syntax:") self.filterSyntaxLabel.setBuddy(self.filterSyntaxComboBox) - self.filterColumnComboBox = QtWidgets.QComboBox() + self.filterColumnComboBox = QComboBox() self.filterColumnComboBox.addItem("Subject") self.filterColumnComboBox.addItem("Sender") self.filterColumnComboBox.addItem("Date") - self.filterColumnLabel = QtWidgets.QLabel("Filter &column:") + self.filterColumnLabel = QLabel("Filter &column:") self.filterColumnLabel.setBuddy(self.filterColumnComboBox) self.filterPatternLineEdit.textChanged.connect(self.filterRegExpChanged) @@ -93,11 +105,11 @@ class Window(QtWidgets.QWidget): self.filterCaseSensitivityCheckBox.toggled.connect(self.filterRegExpChanged) self.sortCaseSensitivityCheckBox.toggled.connect(self.sortChanged) - sourceLayout = QtWidgets.QHBoxLayout() + sourceLayout = QHBoxLayout() sourceLayout.addWidget(self.sourceView) self.sourceGroupBox.setLayout(sourceLayout) - proxyLayout = QtWidgets.QGridLayout() + proxyLayout = QGridLayout() proxyLayout.addWidget(self.proxyView, 0, 0, 1, 3) proxyLayout.addWidget(self.filterPatternLabel, 1, 0) proxyLayout.addWidget(self.filterPatternLineEdit, 1, 1, 1, 2) @@ -109,7 +121,7 @@ class Window(QtWidgets.QWidget): proxyLayout.addWidget(self.sortCaseSensitivityCheckBox, 4, 2) self.proxyGroupBox.setLayout(proxyLayout) - mainLayout = QtWidgets.QVBoxLayout() + mainLayout = QVBoxLayout() mainLayout.addWidget(self.sourceGroupBox) mainLayout.addWidget(self.proxyGroupBox) self.setLayout(mainLayout) @@ -117,7 +129,7 @@ class Window(QtWidgets.QWidget): self.setWindowTitle("Basic Sort/Filter Model") self.resize(500, 450) - self.proxyView.sortByColumn(1, QtCore.Qt.AscendingOrder) + self.proxyView.sortByColumn(1, Qt.AscendingOrder) self.filterColumnComboBox.setCurrentIndex(1) self.filterPatternLineEdit.setText("Andy|Grace") @@ -129,26 +141,28 @@ class Window(QtWidgets.QWidget): self.sourceView.setModel(model) def filterRegExpChanged(self): - syntax_nr = self.filterSyntaxComboBox.itemData(self.filterSyntaxComboBox.currentIndex()) - syntax = QtCore.QRegExp.PatternSyntax(syntax_nr) - - if self.filterCaseSensitivityCheckBox.isChecked(): - caseSensitivity = QtCore.Qt.CaseSensitive - else: - caseSensitivity = QtCore.Qt.CaseInsensitive - - regExp = QtCore.QRegExp(self.filterPatternLineEdit.text(), - caseSensitivity, syntax) - self.proxyModel.setFilterRegExp(regExp) + syntax_nr = self.filterSyntaxComboBox.currentData() + pattern = self.filterPatternLineEdit.text() + if syntax_nr == WILDCARD: + pattern = QRegularExpression.wildcardToRegularExpression(pattern) + elif syntax_nr == FIXED_STRING: + pattern = QRegularExpression.escape(pattern) + + regExp = QRegularExpression(pattern) + if not self.filterCaseSensitivityCheckBox.isChecked(): + options = regExp.patternOptions() + options |= QRegularExpression.CaseInsensitiveOption + regExp.setPatternOptions(options) + self.proxyModel.setFilterRegularExpression(regExp) def filterColumnChanged(self): self.proxyModel.setFilterKeyColumn(self.filterColumnComboBox.currentIndex()) def sortChanged(self): if self.sortCaseSensitivityCheckBox.isChecked(): - caseSensitivity = QtCore.Qt.CaseSensitive + caseSensitivity = Qt.CaseSensitive else: - caseSensitivity = QtCore.Qt.CaseInsensitive + caseSensitivity = Qt.CaseInsensitive self.proxyModel.setSortCaseSensitivity(caseSensitivity) @@ -161,41 +175,38 @@ def addMail(model, subject, sender, date): def createMailModel(parent): - model = QtGui.QStandardItemModel(0, 3, parent) + model = QStandardItemModel(0, 3, parent) - model.setHeaderData(0, QtCore.Qt.Horizontal, "Subject") - model.setHeaderData(1, QtCore.Qt.Horizontal, "Sender") - model.setHeaderData(2, QtCore.Qt.Horizontal, "Date") + model.setHeaderData(0, Qt.Horizontal, "Subject") + model.setHeaderData(1, Qt.Horizontal, "Sender") + model.setHeaderData(2, Qt.Horizontal, "Date") addMail(model, "Happy New Year!", "Grace K. <grace@software-inc.com>", - QtCore.QDateTime(QtCore.QDate(2006, 12, 31), QtCore.QTime(17, 3))) + QDateTime(QDate(2006, 12, 31), QTime(17, 3))) addMail(model, "Radically new concept", "Grace K. <grace@software-inc.com>", - QtCore.QDateTime(QtCore.QDate(2006, 12, 22), QtCore.QTime(9, 44))) + QDateTime(QDate(2006, 12, 22), QTime(9, 44))) addMail(model, "Accounts", "pascale@nospam.com", - QtCore.QDateTime(QtCore.QDate(2006, 12, 31), QtCore.QTime(12, 50))) + QDateTime(QDate(2006, 12, 31), QTime(12, 50))) addMail(model, "Expenses", "Joe Bloggs <joe@bloggs.com>", - QtCore.QDateTime(QtCore.QDate(2006, 12, 25), QtCore.QTime(11, 39))) + QDateTime(QDate(2006, 12, 25), QTime(11, 39))) addMail(model, "Re: Expenses", "Andy <andy@nospam.com>", - QtCore.QDateTime(QtCore.QDate(2007, 1, 2), QtCore.QTime(16, 5))) + QDateTime(QDate(2007, 1, 2), QTime(16, 5))) addMail(model, "Re: Accounts", "Joe Bloggs <joe@bloggs.com>", - QtCore.QDateTime(QtCore.QDate(2007, 1, 3), QtCore.QTime(14, 18))) + QDateTime(QDate(2007, 1, 3), QTime(14, 18))) addMail(model, "Re: Accounts", "Andy <andy@nospam.com>", - QtCore.QDateTime(QtCore.QDate(2007, 1, 3), QtCore.QTime(14, 26))) + QDateTime(QDate(2007, 1, 3), QTime(14, 26))) addMail(model, "Sports", "Linda Smith <linda.smith@nospam.com>", - QtCore.QDateTime(QtCore.QDate(2007, 1, 5), QtCore.QTime(11, 33))) + QDateTime(QDate(2007, 1, 5), QTime(11, 33))) addMail(model, "AW: Sports", "Rolf Newschweinstein <rolfn@nospam.com>", - QtCore.QDateTime(QtCore.QDate(2007, 1, 5), QtCore.QTime(12, 0))) + QDateTime(QDate(2007, 1, 5), QTime(12, 0))) addMail(model, "RE: Sports", "Petra Schmidt <petras@nospam.com>", - QtCore.QDateTime(QtCore.QDate(2007, 1, 5), QtCore.QTime(12, 1))) + QDateTime(QDate(2007, 1, 5), QTime(12, 1))) return model if __name__ == '__main__': - - import sys - - app = QtWidgets.QApplication(sys.argv) + app = QApplication(sys.argv) window = Window() window.setSourceModel(createMailModel(window)) window.show() diff --git a/sources/pyside2/libpyside/feature_select.cpp b/sources/pyside2/libpyside/feature_select.cpp index b26810add..6a21d168d 100644 --- a/sources/pyside2/libpyside/feature_select.cpp +++ b/sources/pyside2/libpyside/feature_select.cpp @@ -445,6 +445,8 @@ void finalize() Py_DECREF(fast_id_array[idx]); } +static bool patch_property_impl(); + void init() { // This function can be called multiple times. @@ -457,6 +459,7 @@ void init() featurePointer = featureProcArray; initSelectableFeature(SelectFeatureSet); registerCleanupFunction(finalize); + patch_property_impl(); is_initialized = true; } // Reset the cache. This is called at any "from __feature__ import". @@ -551,9 +554,8 @@ static PyObject *createProperty(PyObject *getter, PyObject *setter) assert(getter != nullptr); if (setter == nullptr) setter = Py_None; - PyObject *prop = PyObject_CallObject(reinterpret_cast<PyObject *>(&PyProperty_Type), nullptr); - AutoDecRef args(Py_BuildValue("OO", getter, setter)); - PyProperty_Type.tp_init(prop, args, nullptr); + auto obtype = reinterpret_cast<PyObject *>(&PyProperty_Type); + PyObject *prop = PyObject_CallFunctionObjArgs(obtype, getter, setter, nullptr); return prop; } @@ -635,6 +637,77 @@ static bool feature_02_true_property(PyTypeObject *type, PyObject *prev_dict, in ////////////////////////////////////////////////////////////////////////////// // +// These are a number of patches to make Python's property object better +// suitable for us. +// We turn `__doc__` into a lazy attribute saving signature initialization. +// +// Currently, there is no static extension planned, because _PyType_Lookup +// and Limited_API are hard to use at the same time. +// + +typedef struct { + PyObject_HEAD + PyObject *prop_get; + PyObject *prop_set; + PyObject *prop_del; + PyObject *prop_doc; + int getter_doc; +} propertyobject; + +static PyObject *property_doc_get(PyObject *self, void *) +{ + auto po = reinterpret_cast<propertyobject *>(self); + + if (po->prop_doc != nullptr && po->prop_doc != Py_None) { + Py_INCREF(po->prop_doc); + return po->prop_doc; + } + if (po->prop_get) { + // PYSIDE-1019: Fetch the default `__doc__` from fget. We do it late. + auto txt = PyObject_GetAttr(po->prop_get, PyMagicName::doc()); + if (txt != nullptr) { + Py_INCREF(txt); + po->prop_doc = txt; + Py_INCREF(txt); + return txt; + } + PyErr_Clear(); + } + Py_RETURN_NONE; +} + +static int property_doc_set(PyObject *self, PyObject *value, void *) +{ + auto po = reinterpret_cast<propertyobject *>(self); + + Py_INCREF(value); + po->prop_doc = value; + return 0; +} + +static PyGetSetDef property_getset[] = { + // This gets added to the existing getsets + {const_cast<char *>("__doc__"), property_doc_get, property_doc_set, nullptr, nullptr}, + {nullptr, nullptr, nullptr, nullptr, nullptr} +}; + +static bool patch_property_impl() +{ + // Turn `__doc__` into a computed attribute without changing writability. + auto gsp = property_getset; + auto type = &PyProperty_Type; + auto dict = type->tp_dict; + AutoDecRef descr(PyDescr_NewGetSet(type, gsp)); + if (descr.isNull()) + return false; + if (PyDict_SetItemString(dict, gsp->name, descr) < 0) + return false; + // Replace property_descr_get/set by slightly changed versions + return true; +} + +////////////////////////////////////////////////////////////////////////////// +// // PYSIDE-1019: Support switchable extensions // // Feature 0x04..0x40: A fake switchable option for testing diff --git a/sources/shiboken2/libshiboken/pep384impl.cpp b/sources/shiboken2/libshiboken/pep384impl.cpp index c349572f6..fa31ce43f 100644 --- a/sources/shiboken2/libshiboken/pep384impl.cpp +++ b/sources/shiboken2/libshiboken/pep384impl.cpp @@ -91,6 +91,7 @@ static PyMemberDef probe_members[] = { #define probe_tp_members probe_members #define probe_tp_getset probe_getseters #define probe_tp_descr_get make_dummy(10) +#define probe_tp_descr_set make_dummy(18) #define probe_tp_init make_dummy(11) #define probe_tp_alloc make_dummy(12) #define probe_tp_new make_dummy(13) @@ -114,6 +115,7 @@ static PyType_Slot typeprobe_slots[] = { {Py_tp_members, probe_tp_members}, {Py_tp_getset, probe_tp_getset}, {Py_tp_descr_get, probe_tp_descr_get}, + {Py_tp_descr_set, probe_tp_descr_set}, {Py_tp_init, probe_tp_init}, {Py_tp_alloc, probe_tp_alloc}, {Py_tp_new, probe_tp_new}, @@ -163,6 +165,7 @@ check_PyTypeObject_valid() || !PyDict_Check(check->tp_dict) || !PyDict_GetItemString(check->tp_dict, "dummy") || probe_tp_descr_get != check->tp_descr_get + || probe_tp_descr_set != check->tp_descr_set || probe_tp_dictoffset != typetype->tp_dictoffset || probe_tp_init != check->tp_init || probe_tp_alloc != check->tp_alloc diff --git a/sources/shiboken2/libshiboken/pep384impl.h b/sources/shiboken2/libshiboken/pep384impl.h index 4b3e32ea2..7a6f57fcd 100644 --- a/sources/shiboken2/libshiboken/pep384impl.h +++ b/sources/shiboken2/libshiboken/pep384impl.h @@ -115,7 +115,7 @@ typedef struct _typeobject { struct _typeobject *tp_base; PyObject *tp_dict; descrgetfunc tp_descr_get; - void *X33; // descrsetfunc tp_descr_set; + descrsetfunc tp_descr_set; Py_ssize_t tp_dictoffset; initproc tp_init; allocfunc tp_alloc; |