diff --git a/components/Button.qml b/components/Button.qml new file mode 100644 index 0000000000000000000000000000000000000000..496d37df20878fc895dc1032475893e726712087 --- /dev/null +++ b/components/Button.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Button { + id:button + + property int buttonHeight: Math.max(22, styleitem.sizeFromContents(100, 6).height) + height: buttonHeight + + QStyleItem { + id:styleitem + elementType:"button" + sunken: pressed + raised: !pressed + hover: containsMouse + enabled:button.enabled + text:button.text + } + + background: + QStyleBackground { + style:styleitem + anchors.fill:parent + } +} + diff --git a/components/ButtonRow.qml b/components/ButtonRow.qml new file mode 100644 index 0000000000000000000000000000000000000000..623c5f44283fa00555d3ac8e07ad15278f732caa --- /dev/null +++ b/components/ButtonRow.qml @@ -0,0 +1,5 @@ +import QtQuick 1.0 +import "../../../components" as Components + +Components.ButtonRow { +} diff --git a/components/CheckBox.qml b/components/CheckBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..632dce2d4000584df89cc036b63823ee5ff857ac --- /dev/null +++ b/components/CheckBox.qml @@ -0,0 +1,26 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jb : Size should not depend on background, we should make it consistent + +Components.CheckBox{ + id:checkbox + property variant text + width:100 + height:18 + + background: QStyleBackground { + id:styleitem + style:QStyleItem { + elementType:"checkbox" + sunken:pressed + on:checked || pressed + hover:containsMouse + text:checkbox.text + enabled:checkbox.enabled + } + } + checkmark: null +} + diff --git a/components/ChoiceList.qml b/components/ChoiceList.qml new file mode 100644 index 0000000000000000000000000000000000000000..5015fbefbdd92866f81b46faf3a70e32840ff29b --- /dev/null +++ b/components/ChoiceList.qml @@ -0,0 +1,52 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.ChoiceList { + id:choicelist + + property int buttonHeight: buttonitem.sizeFromContents(100, 18).height + QStyleItem { id:buttonitem; elementType:"combobox" } + height: buttonHeight + topMargin:4 + bottomMargin:4 + + QStyleItem { + id:styleitem + elementType: "combobox" + sunken: pressed + raised: !pressed + hover: containsMouse + enabled:choicelist.enabled + } + + background: QStyleBackground { + anchors.fill:parent + style: styleitem + } + + listItem: Item { + id:item + + height:22 + anchors.left:parent.left + width:choicelist.width + QStyleBackground { + anchors.fill:parent + style: QStyleItem { + elementType: "menuitem" + text: choicelist.model.get(index).text + selected: highlighted + } + } + } + + popupFrame: QStyleBackground { + property int fw: styleitem.pixelMetric("menupanelwidth"); + anchors.leftMargin: styleitem.pixelMetric("menuhmargin") + fw + anchors.rightMargin: styleitem.pixelMetric("menuhmargin") + fw + anchors.topMargin: styleitem.pixelMetric("menuvmargin") + fw + anchors.bottomMargin: styleitem.pixelMetric("menuvmargin") + fw + style:QStyleItem{elementType:"menu"} + } +} diff --git a/components/GroupBox.qml b/components/GroupBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..0d45810d12d1964cb24c1fa6bfb11b7dc2d468d6 --- /dev/null +++ b/components/GroupBox.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Item { + width:200 + height:46 + + property alias text: styleitem.text + default property alias children: content.children + property bool checkable: false + + QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + id:styleitem + elementType:"groupbox" + } + + Item { + id:content + anchors.topMargin:22 + anchors.leftMargin:6 + anchors.fill:parent + } + } +} diff --git a/components/ProgressBar.qml b/components/ProgressBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..bf29bcac94fb4beb915ec407ac10b20e564d9d3e --- /dev/null +++ b/components/ProgressBar.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.ProgressBar { + id:progressbar + + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 15).height + QStyleItem { id:buttonitem; elementType:"button" } + height: buttonHeight + + background: QStyleBackground{ + anchors.fill:parent + style: QStyleItem { + elementType:"progressbar" + + // XXX: since desktop uses int instead of real, the progressbar + // range [0..1] must be stretched to a good precision + property int factor : 1000000 + + value: progressbar.value * factor + minimum: indeterminate ? 0 : progressbar.minimumValue * factor + maximum: indeterminate ? 0 : progressbar.maximumValue * factor + enabled: progressbar.enabled + } + } + indeterminateProgress:null + progress: null +} + diff --git a/components/RadioButton.qml b/components/RadioButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..3c2c3e5473503f88f95eddc4675d24c747ce7209 --- /dev/null +++ b/components/RadioButton.qml @@ -0,0 +1,26 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jb : Size should not depend on background, we should make it consistent + +Components.RadioButton{ + id:radiobutton + property variant text + width:110 + height:18 + + background: QStyleBackground { + + style: QStyleItem{ + elementType:"radiobutton" + sunken:pressed + on:checked || pressed + hover:containsMouse + text:radiobutton.text + enabled:radiobutton.enabled + } + } + checkmark: null +} + diff --git a/components/ScrollArea.qml b/components/ScrollArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..3a904ddebf7fa86027bfc84457eb17d7ec4ed058 --- /dev/null +++ b/components/ScrollArea.qml @@ -0,0 +1,103 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +FocusScope { + id:scrollarea + width: 100 + height: 100 + + property int contentMargin: 1 + property int __scrollbarExtent : styleitem.pixelMetric("scrollbarExtent"); + property int frameWidth: styleitem.pixelMetric("defaultframewidth"); + property int contentHeight : content.childrenRect.height + property int contentWidth: content.childrenRect.width + property alias color: flickable.color + property bool frame: true + property bool highlightOnFocus: false + + default property alias children: content.children + + property int contentY + property int contentX + + property bool frameAroundContents: styleitem.styleHint("framearoundcontents") + + onContentYChanged: { + vscrollbar.value = contentY + } + + onContentXChanged: { + hscrollbar.value = contentX + } + + QStyleBackground { + style: QStyleItem{ + id:styleitem + elementType: frame ? "frame" : "" + sunken: true + } + anchors.fill: parent + anchors.rightMargin: (frameAroundContents && vscrollbar.visible) ? vscrollbar.width + 4 : -frameWidth + anchors.bottomMargin: (frameAroundContents && hscrollbar.visible) ? hscrollbar.height + 4 : -frameWidth + anchors.topMargin: (frameAroundContents && hscrollbar.visible) ? hscrollbar.height + 4 : -frameWidth + + Rectangle { + id:flickable + color: "transparent" + anchors.fill: parent + anchors.margins: frame ? 2 : 0 + clip: true + + Item { + id: docmargins + anchors.fill:parent + anchors.margins:contentMargin + Item { + id: content + x: -scrollarea.contentX + y: -scrollarea.contentY + } + } + } + } + + ScrollBar { + id: hscrollbar + orientation: Qt.Horizontal + visible: contentWidth > flickable.width + maximumValue: contentWidth > flickable.width ? scrollarea.contentWidth - flickable.width : 0 + minimumValue: 0 + anchors.bottom: parent.bottom + anchors.left: parent.left + anchors.right: parent.right + anchors.rightMargin: { return (frame ? 1 : 0) + ( vscrollbar.visible ? __scrollbarExtent : 0) } + onValueChanged: contentX = value + } + + ScrollBar { + id: vscrollbar + orientation: Qt.Vertical + visible: contentHeight > flickable.height + maximumValue: contentHeight > flickable.height ? scrollarea.contentHeight - flickable.height : 0 + minimumValue: 0 + anchors.right: parent.right + anchors.top: parent.top + anchors.bottom: parent.bottom + anchors.bottomMargin: { return (frame ? 1 : 0) + (hscrollbar.visible ? __scrollbarExtent : 0) } + onValueChanged: contentY = value + } + + QStyleBackground { + z:2 + anchors.fill:parent + anchors.margins:-2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + visible: highlightOnFocus && parent.activeFocus && styleitem.styleHint("focuswidget") + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } +} diff --git a/components/ScrollBar.qml b/components/ScrollBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..ea822b62e82543ae0a68b41699b0f2673b3e2eee --- /dev/null +++ b/components/ScrollBar.qml @@ -0,0 +1,109 @@ +import QtQuick 1.1 +import "../../../components" as Components +import "../plugin" + +MouseArea { + id:scrollbar + property int __scrollbarExtent : styleitem.pixelMetric("scrollbarExtent"); + implicitWidth:orientation == Qt.Horizontal ? 200 : __scrollbarExtent; + implicitHeight:orientation == Qt.Horizontal ? __scrollbarExtent : 200 + + property int orientation : Qt.Horizontal + property alias minimumValue: slider.minimumValue + property alias maximumValue: slider.maximumValue + property alias value: slider.value + + property bool upPressed; + property bool downPressed; + property bool __autoincrement: false + + // Update hover item + onEntered: styleitem.activeControl = bgitem.hitTest(mouseX, mouseY) + onExited: styleitem.activeControl = "none" + onMouseXChanged: styleitem.activeControl = bgitem.hitTest(mouseX, mouseY) + hoverEnabled:true + + Timer { running: upPressed || downPressed; interval: 350 ; onTriggered: __autoincrement = true } + Timer { running: __autoincrement; interval: 60 ; repeat: true ; + onTriggered: upPressed ? decrement() : increment() } + + onPressed: { + var control = bgitem.hitTest(mouseX,mouseY) + if (control == "up") { + upPressed = true + } else if (control == "down") { + downPressed = true + } + } + + onReleased: { + __autoincrement = false; + if (upPressed) { + upPressed = false; + decrement() + } else if (downPressed) { + increment() + downPressed = false; + } + } + + function increment() { + value += 30 + if (value > maximumValue) + value = maximumValue + } + + function decrement() { + value -= 30 + if (value < minimumValue) + value = minimumValue + } + + QStyleBackground { + id:bgitem + anchors.fill:parent + style: QStyleItem { + id:styleitem + elementType: "scrollbar" + hover: activeControl != "none" + activeControl: "none" + sunken: upPressed | downPressed + minimum: slider.minimumValue + maximum: slider.maximumValue + value: slider.value + horizontal: orientation == Qt.Horizontal + enabled: parent.enabled + } + } + + property variant handleRect + function updateHandle() { + handleRect = bgitem.subControlRect("handle") + var grooveRect = bgitem.subControlRect("groove"); + var extra = 0 + if (orientation == Qt.Vertical) { + slider.anchors.topMargin = grooveRect.y + extra + slider.anchors.bottomMargin = height - grooveRect.y - grooveRect.height + extra + } else { + slider.anchors.leftMargin = grooveRect.x + extra + slider.anchors.rightMargin = width - grooveRect.x - grooveRect.width + extra + } + } + + onValueChanged: updateHandle() + onMaximumValueChanged: updateHandle() + onMinimumValueChanged: updateHandle() + Component.onCompleted: updateHandle() + Components.Slider { + id:slider + anchors.fill:parent + orientation:scrollbar.orientation + leftMargin: (orientation === Qt.Horizontal) ? handleRect.width/2 : handleRect.height/2 + rightMargin:leftMargin + handle: Item { width:orientation == Qt.Vertical ? handleRect.height : handleRect.width; + height:orientation == Qt.Vertical ? handleRect.width : handleRect.height} + groove:null + valueIndicator:null + inverted:orientation != Qt.Horizontal + } +} diff --git a/components/Slider.qml b/components/Slider.qml new file mode 100644 index 0000000000000000000000000000000000000000..c6b9c915187794195af2a1d75a6317627fada3bf --- /dev/null +++ b/components/Slider.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jens: ContainsMouse breaks drag functionality + +Components.Slider{ + id:slider + minimumWidth:200 + + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 15).height + QStyleItem { id:buttonitem; elementType:"button" } + height: buttonHeight + + groove: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"slider" + sunken: pressed + maximum:slider.maximumValue + minimum:slider.minimumValue + value:slider.value + horizontal:slider.orientation == Qt.Horizontal + enabled:slider.enabled + } + } + + handle: null + valueIndicator: null +} diff --git a/components/SpinBox.qml b/components/SpinBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..2db428cef894a82b7497096833bb36bb18e2c717 --- /dev/null +++ b/components/SpinBox.qml @@ -0,0 +1,91 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.SpinBox { + id:spinbox + + property variant __upRect; + property variant __downRect; + property int __margin: (height -16)/2 + + // Align height with button + topMargin:__margin + bottomMargin:__margin + + property int buttonHeight: edititem.sizeFromContents(100, 20).height + QStyleItem { id:edititem; elementType:"edit" } + height: buttonHeight + clip:false + + background: + Item { + anchors.fill: parent + property variant __editRect + + Rectangle { + id:editBackground + x: __editRect.x - 1 + y: __editRect.y + width: __editRect.width + height: __editRect.height + } + + Item { + id:focusFrame + anchors.fill: editBackground + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -6 + anchors.leftMargin: -5 + anchors.fill: parent + visible:spinbox.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } + + function updateRect() { + __upRect = spinboxbg.subControlRect("up"); + __downRect = spinboxbg.subControlRect("down"); + __editRect = spinboxbg.subControlRect("edit"); + } + + Component.onCompleted:updateRect() + onWidthChanged:updateRect() + onHeightChanged:updateRect() + + QStyleBackground { + id:spinboxbg + anchors.fill:parent + style: QStyleItem { + id: styleitem + elementType: "spinbox" + sunken: downPressed | upPressed + hover: containsMouse + focus: spinbox.activeFocus + enabled: spinbox.enabled + value: (upPressed? 1 : 0) | + (downPressed== 1 ? 1<<1 : 0) | + (upEnabled? (1<<2) : 0) | + (downEnabled == 1 ? (1<<3) : 0) + } + } + } + + up: Item { + x: __upRect.x + y: __upRect.y + width: __upRect.width + height: __upRect.height + } + + down: Item { + x: __downRect.x + y: __downRect.y + width: __downRect.width + height: __downRect.height + } +} diff --git a/components/Switch.qml b/components/Switch.qml new file mode 100644 index 0000000000000000000000000000000000000000..b82817c9f6071766d31216cedd1106c02b65201e --- /dev/null +++ b/components/Switch.qml @@ -0,0 +1,22 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Switch { + id:widget + minimumWidth:100 + minimumHeight:30 + + groove:QStyleItem { + elementType:"edit" + sunken: true + } + + handle: QStyleItem { + elementType:"button" + width:widget.width/2 + height:widget.height-4 + hover:containsMouse + } +} + diff --git a/components/Tab.qml b/components/Tab.qml new file mode 100644 index 0000000000000000000000000000000000000000..e258185ba5299b603b8240946dfc57f2cdfac167 --- /dev/null +++ b/components/Tab.qml @@ -0,0 +1,7 @@ +import Qt 4.7 + +Item { + id:tab + anchors.fill:parent + property string title +} diff --git a/components/TabBar.qml b/components/TabBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..2d891256f283ec06278b4f691ec939a0d37a3fbd --- /dev/null +++ b/components/TabBar.qml @@ -0,0 +1,90 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + + +Item { + id: tabbar + property int tabHeight: styleitem.sizeFromContents(100, 24).height + height: tabHeight + + property Item tabFrame + onTabFrameChanged:parent = tabFrame + visible: tabFrame ? tabFrame.tabsVisible : true + property int __overlap : styleitem.pixelMetric("tabvshift"); + property string position: tabFrame ? tabFrame.position : "North" + property string tabBarAlignment: styleitem.styleHint("tabbaralignment"); + property int tabOverlap: styleitem.pixelMetric("taboverlap"); + + function tab(index) { + for (var i = 0; i < tabrow.children.length; ++i) { + if (tabrow.children[i].tabindex == index) { + return tabrow.children[i] + } + } + return null; + } + + QStyleItem { + id:styleitem + elementType: "tab" + text: "generic" + } + + Row { + id:tabrow + states: + State { + when: tabBarAlignment == "center" + name: "centered" + AnchorChanges { + target:tabrow + anchors.horizontalCenter: tabbar.horizontalCenter + } + } + Repeater { + id:repeater + model: tabFrame ? tabFrame.tabs.length : null + delegate: Item { + id:tab + property int tabindex: index + property bool selected : tabFrame.current == index + width: textitem.width + 42 + height: tabHeight + z: selected ? 1 : -1 + + QStyleBackground { + style: QStyleItem { + id:style + elementType: "tab" + selected: tab.selected + text: tabbar.position + + activeControl: tabFrame.count == 1 ? + "only" : + index == 0 ? "beginning" : + index == tabFrame.count-1 ? "end" : "middle" + } + anchors.leftMargin: -tabOverlap + (style.text == "North" && (style.activeControl == "middle" || style.activeControl == "end") + && tab.selected ? -__overlap : 0) + + anchors.rightMargin: -tabOverlap + (style.text == "North" && (style.activeControl == "middle" || style.activeControl == "beginning") + && tab.selected ? -__overlap : 0) + anchors.fill:parent + } + + Text { + id:textitem + anchors.centerIn:parent + text: tabFrame.tabs[index].title + elide: Text.ElideRight + } + + MouseArea { + anchors.fill: parent + onPressed: tabFrame.current = index + } + } + } + } +} diff --git a/components/TabFrame.qml b/components/TabFrame.qml new file mode 100644 index 0000000000000000000000000000000000000000..01896e75353cc3ca79f644b3989b994f40d4dfd9 --- /dev/null +++ b/components/TabFrame.qml @@ -0,0 +1,72 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Item{ + id: tabWidget + width:100 + height:100 + property TabBar tabbar + property int current: 0 + property int count: stack.children.length + property bool frame:true + property bool tabsVisible: true + property string position: "North" + default property alias tabs : stack.children + + onCurrentChanged: __setOpacities() + Component.onCompleted: __setOpacities() + onTabbarChanged: { + tabbar.tabFrame = tabWidget + tabbar.anchors.top = tabWidget.top + tabbar.anchors.left = tabWidget.left + tabbar.anchors.right = tabWidget.right + } + + property int __baseOverlap : style.pixelMetric("tabbaseoverlap"); + function __setOpacities() { + for (var i = 0; i < stack.children.length; ++i) { + stack.children[i].opacity = (i == current ? 1 : 0) + } + } + + QStyleBackground { + id: frame + z:-1 + style: QStyleItem { + id:style + elementType: "tabframe" + text: position + value: tabbar && tabsVisible && tabbar.tab(current) ? tabbar.tab(current).x : 0 + minimum: tabbar && tabsVisible && tabbar.tab(current) ? tabbar.tab(current).width : 0 + } + anchors.fill:parent + Item { + id:stack + anchors.fill:parent + anchors.margins: frame ? 2 : 0 + } + anchors.topMargin: tabbar && tabsVisible && position == "North" ? tabbar.height - __baseOverlap : 0 + + states: [ + State { + name: "South" + when: position == "South" && tabbar!= undefined + PropertyChanges { + target: frame + anchors.topMargin: 0 + anchors.bottomMargin: tabbar ? tabbar.height - __baseOverlap: 0 + } + PropertyChanges { + target: tabbar + anchors.topMargin: -__baseOverlap + } + AnchorChanges { + target: tabbar + anchors.top: frame.bottom + anchors.bottom: undefined + } + } + ] + } +} diff --git a/components/TextArea.qml b/components/TextArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..93ad8f3d9c1c5962905c6f7007e93fb308daa93b --- /dev/null +++ b/components/TextArea.qml @@ -0,0 +1,44 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.TextArea { + id:textarea + leftMargin:12 + rightMargin:12 + minimumWidth:200 + desktopBehavior:true + placeholderText:"" + clip:false + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 14).height + QStyleItem { id:buttonitem; elementType:"button" } + minimumHeight: buttonHeight + + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"edit" + sunken:true + focus:textarea.activeFocus + } + } + + Item{ + id:focusFrame + anchors.fill: textarea + parent:textarea.parent + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + anchors.fill: parent + visible:textarea.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } +} diff --git a/components/TextField.qml b/components/TextField.qml new file mode 100644 index 0000000000000000000000000000000000000000..260584e9513147dd6917c11875415b5e54c2d058 --- /dev/null +++ b/components/TextField.qml @@ -0,0 +1,52 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.TextField { + id:textfield + minimumWidth:200 + desktopBehavior:true + placeholderText:"" + topMargin:2 + bottomMargin:2 + leftMargin:6 + rightMargin:6 + width:200 + height: editItem.sizeFromContents(100, 20).height + clip:false + + QStyleItem { + id:editItem + elementType:"edit" + sunken:true + focus:textfield.activeFocus + hover:containsMouse + } + + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"edit" + sunken:true + focus:textfield.activeFocus + } + } + + Item{ + id:focusFrame + anchors.fill: textfield + parent:textfield + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + anchors.fill: parent + visible:textfield.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } +} diff --git a/components/TextScrollArea.qml b/components/TextScrollArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..2a7deab683a5b6a63379c3f590959f1b2bf256a7 --- /dev/null +++ b/components/TextScrollArea.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +ScrollArea { + id:area + color: "white" + width: 280 + height: 120 + contentWidth: 200 + + property alias text: edit.text + property alias wrapMode: edit.wrapMode + highlightOnFocus: true + + TextEdit { + id: edit + text: loremIpsum + loremIpsum; + wrapMode: TextEdit.WordWrap; + width: area.contentWidth + selectByMouse:true + + // keep textcursor within scrollarea + onCursorRectangleChanged: + if (cursorRectangle.y >= area.contentY + area.height - 1.5*cursorRectangle.height) + area.contentY = cursorRectangle.y - area.height + 1.5*cursorRectangle.height + else if (cursorRectangle.y < area.contentY) + area.contentY = cursorRectangle.y + + } +} diff --git a/components/ToolBar.qml b/components/ToolBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..ce61382ee1b9b5a0ebb05db310e2616466aabdcf --- /dev/null +++ b/components/ToolBar.qml @@ -0,0 +1,12 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +QStyleBackground { + id:styleitem + width:200 + height:60 + + style: QStyleItem{elementType:"toolbar"} +} + diff --git a/components/ToolButton.qml b/components/ToolButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..98b2dfebf805f460ad4a19ca8504a71bf77cdc86 --- /dev/null +++ b/components/ToolButton.qml @@ -0,0 +1,18 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Button { + id:button + minimumWidth:30 + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem { + elementType: "toolbutton" + on: pressed | checked + sunken: pressed + raised: containsMouse + hover: containsMouse + } + } +} diff --git a/components/custom/BasicButton.qml b/components/custom/BasicButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..1116f0b0111c43b650ebbf6c1e30199fedbd8dc8 --- /dev/null +++ b/components/custom/BasicButton.qml @@ -0,0 +1,44 @@ +import QtQuick 1.1 +import "./behaviors" // ButtonBehavior +import "./styles/default" as DefaultStyles + +Item { + id: button + + signal clicked + property alias pressed: behavior.pressed + property alias containsMouse: behavior.containsMouse + property alias checkable: behavior.checkable // button toggles between checked and !checked + property alias checked: behavior.checked + + property Component background: defaultStyle.background + + property color backgroundColor: syspal.button + property color textColor: syspal.text; + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + // implementation + + property string __position: "only" + implicitWidth: Math.max(minimumWidth, backgroundLoader.item.width) + implicitHeight: Math.max(minimumHeight, backgroundLoader.item.height) + + Loader { + id: backgroundLoader + anchors.fill: parent + sourceComponent: background + property alias styledItem: button + property alias position: button.__position + } + + ButtonBehavior { + id: behavior + anchors.fill: parent + onClicked: button.clicked() + } + + DefaultStyles.BasicButtonStyle { id: defaultStyle } + SystemPalette { id: syspal } +} diff --git a/components/custom/BusyIndicator.qml b/components/custom/BusyIndicator.qml new file mode 100644 index 0000000000000000000000000000000000000000..00b12ed3fc85c927b8d2e40059835cba7104a04f --- /dev/null +++ b/components/custom/BusyIndicator.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles + +Item { + id: busyIndicator + + // Common API: + property bool running: true + + width: backgroundComponent.item.width; + height: backgroundComponent.item.height; + + property Component background: defaultStyle.background + + Loader { + id: backgroundComponent + property bool running: busyIndicator.opacity > 0 && + busyIndicator.visible && + busyIndicator.running +// onRunningChanged: print("Running: " + running) + anchors.fill: parent + sourceComponent: background + + } + + DefaultStyles.BusyIndicatorStyle { id: defaultStyle } +} diff --git a/components/custom/Button.qml b/components/custom/Button.qml new file mode 100644 index 0000000000000000000000000000000000000000..1bcd8335e15a755ff8d1468de5f2695c4f243a9f --- /dev/null +++ b/components/custom/Button.qml @@ -0,0 +1,39 @@ +import QtQuick 1.1 +import "./styles/default" as DefaultStyles + +BasicButton { + id: button + + property string text + property url iconSource + + property Component label: defaultStyle.label + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + // implementation + + implicitWidth: Math.max(minimumWidth, labelLoader.item.implicitWidth + leftMargin + rightMargin) + implicitHeight: Math.max(minimumHeight, labelLoader.item.implicitHeight + topMargin + bottomMargin) + + minimumWidth: defaultStyle.minimumWidth + minimumHeight: defaultStyle.minimumHeight + + background: defaultStyle.background + + Loader { + id: labelLoader + anchors.fill: parent + anchors.leftMargin: leftMargin + anchors.rightMargin: rightMargin + anchors.topMargin: topMargin + anchors.bottomMargin: bottomMargin + property alias styledItem: button + sourceComponent: label + } + + DefaultStyles.ButtonStyle { id: defaultStyle } +} diff --git a/components/custom/ButtonBlock.qml b/components/custom/ButtonBlock.qml new file mode 100644 index 0000000000000000000000000000000000000000..999d3dd062bf1ca90dd935f3c4a0e661dc05f80e --- /dev/null +++ b/components/custom/ButtonBlock.qml @@ -0,0 +1,216 @@ +import QtQuick 1.0 +import "./behaviors" // ButtonBehavior +import "./visuals" // AdjoiningVisual +import "./styles/default" as DefaultStyles + +// KNOWN ISSUES +// 1) When switching between horizontal and vertical orientation after block has been created, the Grid's move +// transition is called *before* the grid item's have actually moved, resulting in incorrect "adjoins" states +// 2) Can't make items in vertical groups all the same width without access to their implicitWidth, see QTBUG-14957 +// 3) Should be generalized into JoinedGroup and ButtonBlock made a specialization. +// 4) ExclusiveSelection support missing + +// NOTES +// 1) The ButtonBlock implementation has no ultimate dependency on AdjoiningVisual, and can therefor be made to work +// with any Component for the item instances. The use of AdjoiningVisual is only for simplifying the implementation +// and styling of the items in the block. The property defining how an item in the block should be joinged with its +// neighbours is "adjoins" which the ButtonBlock "attaches" to the items in the block (i.e. defines in their +// look-up scope) which means the "button" instances does not need to define this property, only read it to draw +// their border appropriately. This "adjoins" property is a *completely* separate concept from the AdjoiningVisual. +// I.e. the coupling between the ButtonGroup and the "button" elements making up the group is the weakest possible. + + +Item { + id: buttonBlock + + property alias model: repeater.model + property variant bindings: {"text":"text", "iconSource":"iconSource", "enabled":"enabled", "opacity":"opacity"} + + property Component buttonBackground: defaultStyle.background + property Component buttonLabel: defaultStyle.label + + property color backgroundColor: syspal.button + property color textColor: syspal.text; + + property int orientation: Qt.Horizontal + signal clicked(int index) + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + width: grid.width + height: grid.height + + Grid { + id: grid + columns: orientation == Qt.Vertical ? 1 : model.count + anchors.centerIn: parent + property int widestItemWidth: 0 + property int talestItemHeight: 0 + + move: Transition { //mm seems "move" transition is not always called after items has been moved + ScriptAction { script: { + if(orientation == Qt.Horizontal) { + grid.leftmostVisibleIndex = grid.__leftmostVisibleIndex(); + grid.rightmostVisibleIndex = grid.__rightmostVisibleIndex(); + } else { + grid.topmostVisibleIndex = grid.__topmostVisibleIndex(); + grid.bottommostVisibleIndex = grid.__bottommostVisibleIndex(); + } + } + } + } + + property int leftmostVisibleIndex + function __leftmostVisibleIndex() { + var leftmostVisibleIndex = 0; + var leftmostItemX = 10000000; + for(var i = 0; i < children.length; i++) { + var child = children[i]; + if(child.x < leftmostItemX && child.opacity > 0) { + leftmostItemX = child.x; + leftmostVisibleIndex = i; + } + } + return leftmostVisibleIndex; + } + + property int rightmostVisibleIndex + function __rightmostVisibleIndex() { + var rightmostVisibleIndex = 0; + var rightmostItemX = 0; + for(var i = 0; i < children.length; i++) { + var child = children[i]; +//mm print("rightmost child? at: " + child.x + "," + child.y) + if(child.x > rightmostItemX && child.opacity > 0) { + rightmostItemX = child.x; + rightmostVisibleIndex = i; + } + } + return rightmostVisibleIndex; + } + + property int topmostVisibleIndex + function __topmostVisibleIndex() { + var topmostVisibleIndex = 0; + var topmostItemY = 10000000; + for(var i = 0; i < children.length; i++) { + var child = children[i]; + if(child.y < topmostItemY && child.opacity > 0) { + topmostItemY = child.y; + topmostVisibleIndex = i; + } + } + return topmostVisibleIndex; + } + + property int bottommostVisibleIndex + function __bottommostVisibleIndex() { + var bottommostVisibleIndex = 0; + var bottommostItemY = 0; + for(var i = 0; i < children.length; i++) { + var child = children[i]; +//mm print("bottommost child? at: " + child.x + "," + child.y) + if(child.y > bottommostItemY && child.opacity > 0) { + bottommostItemY = child.y; + bottommostVisibleIndex = i; + } + } + return bottommostVisibleIndex; + } + + Repeater { + id: repeater + delegate: AdjoiningVisual { + id: blockButton + styledItem: blockButton + styling: buttonBackground + + property alias pressed: behavior.pressed + property alias containsMouse: behavior.containsMouse + property alias checkable: behavior.checkable // button toggles between checked and !checked + property alias checked: behavior.checked + + property string text + property url iconSource + property color textColor: buttonBlock.textColor + property color backgroundColor: buttonBlock.backgroundColor + + Component.onCompleted: { + // Create the Binding objects defined by the ButtonBlock's "bindings" map property to allow + // the properties of the buttons to be bound to properties in the model with different names + + if (bindings == undefined) // jb : bindings is undefined on Mac + return; + var keys = Object.keys(buttonBlock.bindings); + for(var i = 0; i < keys.length; i++) { + var key = keys[i]; + var bindingComponent = + 'import QtQuick 1.0;' + + 'Binding {' + + ' target: blockButton;' + + ' property: "' + key + '";' + + ' value: buttonBlock.model.get(' + index + ').' + bindings[key] + ';' + + '}'; + Qt.createQmlObject(bindingComponent, blockButton); //mm do we ever need to explicitly delete these? + } + + // Find the widest/talest item to make all buttons the same width/height + if(buttonBlock.orientation == Qt.Vertical) //mm Can't make this work without QTBUG-14957 + grid.widestItemWidth = Math.max(grid.widestItemWidth, width); + else + grid.talestItemHeight = Math.max(grid.talestItemHeight, height); + } + + ButtonBehavior { + id: behavior + anchors.fill: parent + onClicked: buttonBlock.clicked(index) + } + + property int adjoins + adjoins: { //mm see QTBUG-14987 + var adjoins; + if(orientation == Qt.Horizontal) { + adjoins = 0x1|0x2; // left and right + if(index == grid.leftmostVisibleIndex) adjoins &= ~0x1; // not left + if(index == grid.rightmostVisibleIndex) adjoins &= ~0x2; // not right + } else { + adjoins = 0x4|0x8; // top and bottom + if(index == grid.topmostVisibleIndex) adjoins &= ~0x4; // not top + if(index == grid.bottommostVisibleIndex) adjoins &= ~0x8; // not bottom + } + return adjoins; + } + + + // width: orientation == Qt.Vertical ? grid.widestItemWidth : item.width + // height: orientation == Qt.Vertical ? item.height : grid.talestItemHeight + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + width: Math.max(minimumWidth, + labelComponent.item.width + leftMargin + rightMargin) + height: Math.max(minimumHeight, + labelComponent.item.height + topMargin + bottomMargin) + + Loader { + id: labelComponent + property variant styledItem: blockButton + anchors.fill: parent + anchors.leftMargin: leftMargin + anchors.rightMargin: rightMargin + anchors.topMargin: topMargin + anchors.bottomMargin: bottomMargin + sourceComponent: buttonLabel + } + + } + } + } + + SystemPalette { id: syspal; colorGroup: enabled ? SystemPalette.Active : SystemPalette.Disabled } + DefaultStyles.ButtonBlockStyle { id: defaultStyle } +} diff --git a/components/custom/ButtonColumn.qml b/components/custom/ButtonColumn.qml new file mode 100644 index 0000000000000000000000000000000000000000..8d63a855c3b774f50678746cc55507e5594e3320 --- /dev/null +++ b/components/custom/ButtonColumn.qml @@ -0,0 +1,44 @@ +import Qt 4.7 +import "ButtonGroup.js" as Behavior + +/* + Class: ButtonColumn + A ButtonColumn allows you to group Buttons in a column. It provides a selection-behavior as well. + + Note: This component don't support the enabled property. + If you need to disable it you should disable all the buttons inside it. + + <code> + ButtonColumn { + Button { text: "Top" } + Button { text: "Bottom" } + } + </code> +*/ +Column { + id: root + + /* + * Property: exclusive + * [bool=true] Specifies the grouping behavior. If enabled, the checked property on buttons contained + * in the group will be exclusive. + * + * Note that a button in an exclusive group will allways be checkable + */ + property bool exclusive: true + + /* + * Property: checkedButton + * [string] Contains the last checked Button. + */ + property Item checkedButton; + + Component.onCompleted: { + Behavior.create(root, {direction: Qt.Vertical}); + } + + Component.onDestruction: { + Behavior.destroy(); + } + +} diff --git a/components/custom/ButtonGroup.js b/components/custom/ButtonGroup.js new file mode 100644 index 0000000000000000000000000000000000000000..f2d000029f7b1cabcf4803c9f608ca0218421549 --- /dev/null +++ b/components/custom/ButtonGroup.js @@ -0,0 +1,127 @@ +var self; +var clickHandlers = []; +var visibleButtons = []; +var nonVisibleButtons = []; +var direction; +var exclusive; + +function create(that, options) { + self = that; + direction = options.direction || Qt.Horizontal; + exclusive = self.exclusive|| options.exclusive; + self.childrenChanged.connect(rebuild); +// self.widthChanged.connect(resizeChildren); + build(); +} + +function isButton(item) { + if (item && item["__position"] !== undefined) + return true; + return false; +} + +function hasChecked(item) { + if (item && item["checked"] !== undefined) + return true; + return false; +} + +function destroy() { + self.childrenChanged.disconnect(rebuild); +// self.widthChanged.disconnect(resizeChildren); + cleanup(); +} + +function build() { + visibleButtons = []; + nonVisibleButtons = []; + + for (var i = 0, item; (item = self.children[i]); i++) { + if (!hasChecked(item)) + continue; + + item.visibleChanged.connect(rebuild); // Not optimal, but hardly a bottleneck in your app + if (!item.visible) { + nonVisibleButtons.push(item); + continue; + } + visibleButtons.push(item); + + if (exclusive && hasChecked(item)) { + if (item["checkable"]!==undefined) { + item.checkable = true; + } + clickHandlers[i] = checkExclusive(item); + item.clicked.connect(clickHandlers[i]); + } + } + + if (self.checkedButton && !self.checkedButton.visible) + self.checkedButton = undefined; + + var nrButtons = visibleButtons.length; + if (nrButtons == 0) + return; + + if (nrButtons == 1) { + finishButton(visibleButtons[0], "only"); + } else { + finishButton(visibleButtons[0], direction == Qt.Horizontal ? "leftmost" : "top"); + for (var i = 1; i < nrButtons - 1; i++) + finishButton(visibleButtons[i], direction == Qt.Horizontal ? "h_middle": "v_middle"); + finishButton(visibleButtons[nrButtons - 1], direction == Qt.Horizontal ? "rightmost" : "bottom"); + } +} + +function finishButton(button, position) { + if (isButton(button)) { + button.__position = position; + if (direction == Qt.Vertical) { + button.anchors.left = self.left + button.anchors.right = self.right + } + } +} + +function cleanup() { + visibleButtons.forEach(function(item, i) { + if (clickHandlers[i]) + item.clicked.disconnect(clickHandlers[i]); + item.visibleChanged.disconnect(rebuild); + }); + clickHandlers = []; + + nonVisibleButtons.forEach(function(item, i) { + item.visibleChanged.disconnect(rebuild); + }); +} + +function rebuild() { + cleanup(); + build(); +} + +function resizeChildren() { + if (direction != Qt.Horizontal) + return; + + var extraPixels = self.width % visibleButtons; + var buttonSize = (self.width - extraPixels) / visibleButtons; + visibleButtons.forEach(function(item, i) { + if (!item || !item.visible) + return; + item.width = buttonSize + (extraPixels > 0 ? 1 : 0); + if (extraPixels > 0) + extraPixels--; + }); +} + +function checkExclusive(item) { + var button = item; + return function() { + for (var i = 0, ref; (ref = visibleButtons[i]); i++) { + ref.checked = button === ref; + } + self.checkedButton = button; + } +} diff --git a/components/custom/ButtonRow.qml b/components/custom/ButtonRow.qml new file mode 100644 index 0000000000000000000000000000000000000000..7c2d3ea7ad62646096004778e3ed48e2254f122f --- /dev/null +++ b/components/custom/ButtonRow.qml @@ -0,0 +1,43 @@ +import Qt 4.7 +import "ButtonGroup.js" as Behavior + +/* + Class: ButtonRow + A ButtonRow allows you to group Buttons in a row. It provides a selection-behavior as well. + + Note: This component don't support the enabled property. + If you need to disable it you should disable all the buttons inside it. + + <code> + ButtonRow { + Button { text: "Left" } + Button { text: "Right" } + } + </code> +*/ +Row { + id: root + + /* + * Property: exclusive + * [bool=true] Specifies the grouping behavior. If enabled, the checked property on buttons contained + * in the group will be exclusive. + * + * Note that a button in an exclusive group will allways be checkable + */ + property bool exclusive: true + + /* + * Property: checkedButton + * [string] Contains the last checked Button. + */ + property Item checkedButton; + + Component.onCompleted: { + Behavior.create(root, {direction: Qt.Horizontal}); + } + + Component.onDestruction: { + Behavior.destroy(); + } +} diff --git a/components/custom/CheckBox.qml b/components/custom/CheckBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..e8b865564964a523353c8860d4fa3a6ca4ded71f --- /dev/null +++ b/components/custom/CheckBox.qml @@ -0,0 +1,49 @@ +import QtQuick 1.1 +import "./behaviors" +import "./styles/default" as DefaultStyles + +Item { + id: checkBox + + signal clicked + property alias pressed: behavior.pressed + property alias checked: behavior.checked + property alias containsMouse: behavior.containsMouse + + property Component background: defaultStyle.background + property Component checkmark: defaultStyle.checkmark + + property color backgroundColor: syspal.base + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + // implementation + + implicitWidth: minimumWidth + implicitHeight: minimumHeight + + Loader { + id: backgroundLoader + anchors.fill: parent + property alias styledItem: checkBox + sourceComponent: background + } + + Loader { + id: checkmarkLoader + anchors.centerIn: parent + property alias styledItem: checkBox + sourceComponent: checkmark + } + + ButtonBehavior { + id: behavior + anchors.fill: parent + checkable: true + onClicked: checkBox.clicked() + } + + DefaultStyles.CheckBoxStyle { id: defaultStyle } + SystemPalette { id: syspal } +} diff --git a/components/custom/ChoiceList.qml b/components/custom/ChoiceList.qml new file mode 100644 index 0000000000000000000000000000000000000000..8c5ca81bed7ccf926e7d864cd9a6fe6228df0353 --- /dev/null +++ b/components/custom/ChoiceList.qml @@ -0,0 +1,85 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles +import "./private" as Private // for ChoiceListPopup + +// KNOWN ISSUES +// 1) Popout list does not have a scrollbar/scroll indicator or similar +// 2) The ChoiceListPopup should ff dynamically loaded, to support radically different implementations +// 3) Mouse wheel scroll events not handled by the popout ListView (see QTBUG-7369) +// 4) Support for configurable bindings between model's and ChoiceList's properties similar to ButtonBlock's is missing + +Item { + id: choiceList + + property alias model: popup.model + property int currentIndex: popup.currentIndex + + property alias containsMouse: mouseArea.containsMouse //mm needed? + property bool pressed: false //mm needed? + + property color textColor: syspal.text + property color backgroundColor: syspal.button + + property Component background: defaultStyle.background + property Component label: defaultStyle.label + property Component listItem: defaultStyle.listItem + property Component popupFrame: defaultStyle.popupFrame + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + property int labelWidth: labelComponent.item.width + property int labelHeight: labelComponent.item.height + + width: Math.max(minimumWidth, + labelComponent.item.width + leftMargin + rightMargin) + height: Math.max(minimumHeight, + labelComponent.item.height + topMargin + bottomMargin) + + // Implementation + + SystemPalette { id: syspal } + Loader { + property alias styledItem: choiceList + sourceComponent: background + anchors.fill: parent + } + + Loader { + id: labelComponent + property alias model: popup.model + anchors.fill: parent + anchors.leftMargin: leftMargin + anchors.rightMargin: rightMargin + anchors.topMargin: topMargin + anchors.bottomMargin: bottomMargin + sourceComponent: label + } + + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + drag.target: Item {} // disable dragging in case ChoiceList is on a Flickable + onPressed: { + choiceList.pressed = true; + popup.togglePopup(); + + } + onReleased: choiceList.pressed = false + onCanceled: choiceList.pressed = false // mouse stolen e.g. by Flickable + } + + Private.ChoiceListPopup { + id: popup + listItem: choiceList.listItem + popupFrame: choiceList.popupFrame + } + + DefaultStyles.ChoiceListStyle { id: defaultStyle } +} diff --git a/components/custom/ProgressBar.qml b/components/custom/ProgressBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..fadbbc5e083541c13a36ae52bb40af2ea1c03670 --- /dev/null +++ b/components/custom/ProgressBar.qml @@ -0,0 +1,80 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles + +Item { + id: progressBar + SystemPalette{id:syspal} + + property real value: 0 + property real minimumValue: 0 + property real maximumValue: 1 + property bool indeterminate: false + + property color backgroundColor: syspal.base + property color progressColor: syspal.highlight + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + width: minimumWidth + height: minimumHeight + + property Component background: defaultStyle.background + property Component progress: defaultStyle.progress + property Component indeterminateProgress: defaultStyle.indeterminateProgress + + Loader { + id: groove + property alias indeterminate:progressBar.indeterminate + property alias value:progressBar.value + property alias maximumValue:progressBar.maximumValue + property alias minimumValue:progressBar.minimumValue + + sourceComponent: background + anchors.fill: parent + } + + Item { + anchors.fill: parent + anchors.leftMargin: leftMargin + anchors.rightMargin: rightMargin + anchors.topMargin: topMargin + anchors.bottomMargin: bottomMargin + + Loader { + id: progressComponent + property alias styledItem: progressBar + property alias indeterminate:progressBar.indeterminate + property alias value:progressBar.value + property alias maximumValue:progressBar.maximumValue + property alias minimumValue:progressBar.minimumValue + property real complete: (value-minimumValue)/(maximumValue-minimumValue) + + opacity: !indeterminate && value > 0 ? 1 : 0 + width: Math.round((progressBar.width-leftMargin-rightMargin)*(complete)) + height: progressBar.height-topMargin-bottomMargin + anchors.left:parent.left + sourceComponent: progressBar.progress + } + + Loader { + id: indeterminateComponent + property alias styledItem: progressBar + property alias indeterminate:progressBar.indeterminate + property alias value:progressBar.value + property alias maximumValue:progressBar.maximumValue + property alias minimumValue:progressBar.minimumValue + + opacity: indeterminate ? 1 : 0 + anchors.fill: parent + sourceComponent: indeterminateProgress + } + } + + DefaultStyles.ProgressBarStyle { id: defaultStyle } +} diff --git a/components/custom/RadioButton.qml b/components/custom/RadioButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..f89f2773cf4f30aee8e9c2e4eb5699f4c522e240 --- /dev/null +++ b/components/custom/RadioButton.qml @@ -0,0 +1,13 @@ +import QtQuick 1.1 +import "./styles/default" as DefaultStyles + +CheckBox { + id: radioButton + + // implementation + + checkmark: defaultStyle.checkmark + background: defaultStyle.background + + DefaultStyles.RadioButtonStyle { id: defaultStyle} +} diff --git a/components/custom/ScrollDecorator.qml b/components/custom/ScrollDecorator.qml new file mode 100644 index 0000000000000000000000000000000000000000..e61a16d206daa2007c05ae6c87cc6a6273aa58c9 --- /dev/null +++ b/components/custom/ScrollDecorator.qml @@ -0,0 +1,20 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles + +Item { + id: scrollDecorator + + property Flickable flickableItem + + anchors.fill: parent + + ScrollIndicator { + horizontal: true + scrollItem: scrollDecorator.flickableItem + } + + ScrollIndicator { + horizontal: false + scrollItem: scrollDecorator.flickableItem + } +} diff --git a/components/custom/ScrollIndicator.qml b/components/custom/ScrollIndicator.qml new file mode 100644 index 0000000000000000000000000000000000000000..cba1dba8126d608f055d61acf22d2a5c45fb47d6 --- /dev/null +++ b/components/custom/ScrollIndicator.qml @@ -0,0 +1,56 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles + +Item { + id: scrollIndicator + + property Flickable scrollItem + property bool horizontal: false + + property Component content: defaultStyle.content + + //private + anchors.left: horizontal ? parent.left : undefined + anchors.right: parent.right + anchors.top: horizontal ? undefined : parent.top + anchors.bottom: parent.bottom + width: horizontal ? parent.width : defaultStyle.minimumWidth + height: horizontal ? defaultStyle.minimumWidth : parent.height + + opacity: (horizontal ? scrollItem.movingHorizontally : scrollItem.movingVertically) ? 1 : 0 + Behavior on opacity { NumberAnimation { duration: 100 } } + + Item { + id: ratationContainer + + anchors.centerIn: parent + rotation: horizontal ? -90 : 0 + width: horizontal ? scrollIndicator.height : scrollIndicator.width // rotate width and heigh back + height: horizontal ? scrollIndicator.width : scrollIndicator.height + + property variant visibleArea: scrollItem.visibleArea + property real scrollItemSize: horizontal ? scrollItem.width : scrollItem.height + property real scrollItemVisibleAreaPos: horizontal ? visibleArea.xPosition : visibleArea.yPosition + property real scrollItemVisibleAreaScrollRatio: horizontal ? visibleArea.widthRatio : visibleArea.heightRatio + property real scrollItemContentPos: horizontal ? scrollItem.contentX : scrollItem.contentY + property real scrollItemContentSize: horizontal ? scrollItem.contentWidth : scrollItem.contentHeight + + property real offset: scrollItemVisibleAreaPos * scrollItemSize + property real length: scrollItemVisibleAreaScrollRatio * scrollItemSize + property real startOvershoot: Math.max(-scrollItemContentPos, 0) + property real endOvershoot: Math.max(scrollItemContentPos-(scrollItemContentSize-scrollItemSize), 0) + property real start: Math.max(offset + endOvershoot, 0) + property real end: Math.min(offset+length-startOvershoot, scrollItemSize) + + Loader { + x: 0 + y: parent.endOvershoot > 0 ? Math.min(parent.start, parent.scrollItemSize-width) : parent.start + width: 12 + height: Math.max(parent.end-parent.start, width) + + sourceComponent: content + } + } + + DefaultStyles.ScrollIndicatorStyle { id: defaultStyle } +} diff --git a/components/custom/Slider.qml b/components/custom/Slider.qml new file mode 100644 index 0000000000000000000000000000000000000000..789167c66656585bcdaa60a265d2a1847c019e67 --- /dev/null +++ b/components/custom/Slider.qml @@ -0,0 +1,291 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the Qt Components project on Qt Labs. +** +** No Commercial Usage +** This file contains pre-release code and may not be distributed. +** You may use this file in accordance with the terms and conditions contained +** in the Technology Preview License Agreement accompanying this package. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** If you have questions regarding the use of this file, please contact +** Nokia at qt-info@nokia.com. +** +****************************************************************************/ + +import QtQuick 1.0 +import Qt.labs.components 1.0 +import "./styles/default" as DefaultStyles + +Item { + id: slider + + // COMMON API + property int orientation: Qt.Horizontal + property alias minimumValue: range.minimumValue + property alias maximumValue: range.maximumValue + property alias inverted: range.inverted + property bool updateValueWhileDragging: true + property alias pressed: mouseArea.pressed + property alias stepSize: range.stepSize + + // NOTE: this property is in/out, the user can set it, create bindings to it, and + // at the same time the slider wants to update. There's no way in QML to do this kind + // of updates AND allow the user bind it (without a Binding object). That's the + // reason this is an alias to a C++ property in range model. + property alias value: range.value + + // CONVENIENCE TO BE USED BY STYLES + SystemPalette { + id: palette + } + property color progressColor: palette.highlight + property color backgroundColor: palette.window + + property int leftMargin: defaultStyle.leftMargin + property int rightMargin: defaultStyle.rightMargin + + // EXTENSIONS + // Indicate that we want animations in the Slider, people customizing should + // look at it to decide whether or not active animations. + property bool animated: true + + // Value indicator displays the current value near the slider + property bool valueIndicatorVisible: true + property int valueIndicatorMargin: 10 + property string valueIndicatorPosition: _isVertical ? "Left" : "Top" + + // Reimplement this function to control how the value is shown in the + // indicator. + function formatValue(v) { + return Math.round(v); + } + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + // Hooks for customizing the pieces of the slider + property Component groove: defaultStyle.groove + property Component handle: defaultStyle.handle + property Component valueIndicator: defaultStyle.valueIndicator + + // PRIVATE/CONVENIENCE + property bool _isVertical: orientation == Qt.Vertical + + width: _isVertical ? minimumHeight : minimumWidth + height: _isVertical ? minimumWidth : minimumHeight + + DefaultStyles.SliderStyle { id: defaultStyle } + + // This is a template slider, so every piece can be modified by passing a + // different Component. The main elements in the implementation are + // + // - the 'range' does the calculations to map position to/from value, + // it also serves as a data storage for both properties; + // + // - the 'fakeHandle' is what the mouse area drags on the screen, it feeds + // the 'range' position and also reads it when convenient; + // + // - the real 'handle' it is the visual representation of the handle, that + // just follows the 'fakeHandle' position. + // + // When the 'updateValueWhileDragging' is false and we are dragging, we stop + // feeding the range with position information, delaying until the next + // mouse release. + // + // Everything is encapsulated in a contents Item, so for the + // vertical slider, we just swap the height/width, make it + // horizontal, and then use rotation to make it vertical again. + + Item { + id: contents + + width: _isVertical ? slider.height : slider.width + height: _isVertical ? slider.width : slider.height + rotation: _isVertical ? -90 : 0 + + anchors.centerIn: slider + + RangeModel { + id: range + minimumValue: 0 + maximumValue: 100 + value: 0 + stepSize: 1.0 + inverted: false + + positionAtMinimum: leftMargin + positionAtMaximum: contents.width - rightMargin + } + + Loader { + id: grooveLoader + anchors.fill: parent + sourceComponent: groove + + property real handlePosition : handleLoader.x + function positionForValue(value) { + return range.positionForValue(value) - leftMargin; + } + } + + Loader { + id: handleLoader + transform: Translate { x: - handleLoader.width / 2 } + + anchors.verticalCenter: grooveLoader.verticalCenter + + sourceComponent: handle + + x: fakeHandle.x + Behavior on x { + id: behavior + enabled: !mouseArea.drag.active && slider.animated + + PropertyAnimation { + duration: behavior.enabled ? 150 : 0 + easing.type: Easing.OutSine + } + } + } + + Item { + id: fakeHandle + width: handleLoader.width + height: handleLoader.height + transform: Translate { x: - handleLoader.width / 2 } + } + + MouseArea { + id: mouseArea + + anchors.centerIn: parent + anchors.horizontalCenterOffset: (slider.leftMargin - slider.rightMargin) / 2 + + width: parent.width + handleLoader.width - slider.rightMargin - slider.leftMargin + height: parent.height + + drag.target: fakeHandle + drag.axis: Drag.XAxis + drag.minimumX: range.positionAtMinimum + drag.maximumX: range.positionAtMaximum + + onPressed: { + // Clamp the value + var newX = Math.max(mouse.x, drag.minimumX); + newX = Math.min(newX, drag.maximumX); + + // Debounce the press: a press event inside the handler will not + // change its position, the user needs to drag it. + if (Math.abs(newX - fakeHandle.x) > handleLoader.width / 2) + range.position = newX; + } + + onReleased: { + // If we don't update while dragging, this is the only + // moment that the range is updated. + if (!slider.updateValueWhileDragging) + range.position = fakeHandle.x; + } + } + + Loader { + id: valueIndicatorLoader + + transform: Translate { x: - handleLoader.width / 2 } + rotation: _isVertical ? 90 : 0 + visible: valueIndicatorVisible + + // Properties available for the delegate component. Note that the indicatorText + // shows the value for the position the handle is, which is not necessarily the + // available as the current slider.value, since updateValueWhileDragging can + // be set to 'false'. + property string indicatorText: slider.formatValue(range.valueForPosition(handleLoader.x)) + property bool dragging: mouseArea.drag.active + + sourceComponent: valueIndicator + + state: { + if (!_isVertical) + return slider.valueIndicatorPosition; + + if (valueIndicatorPosition == "Right") + return "Bottom"; + if (valueIndicatorPosition == "Top") + return "Right"; + if (valueIndicatorPosition == "Bottom") + return "Left"; + + return "Top"; + } + + anchors.margins: valueIndicatorMargin + + states: [ + State { + name: "Top" + AnchorChanges { + target: valueIndicatorLoader + anchors.bottom: handleLoader.top + anchors.horizontalCenter: handleLoader.horizontalCenter + } + }, + State { + name: "Bottom" + AnchorChanges { + target: valueIndicatorLoader + anchors.top: handleLoader.bottom + anchors.horizontalCenter: handleLoader.horizontalCenter + } + }, + State { + name: "Right" + AnchorChanges { + target: valueIndicatorLoader + anchors.left: handleLoader.right + anchors.verticalCenter: handleLoader.verticalCenter + } + }, + State { + name: "Left" + AnchorChanges { + target: valueIndicatorLoader + anchors.right: handleLoader.left + anchors.verticalCenter: handleLoader.verticalCenter + } + } + ] + } + } + + // Range position normally follow fakeHandle, except when + // 'updateValueWhileDragging' is false. In this case it will only follow + // if the user is not pressing the handle. + Binding { + when: updateValueWhileDragging || !mouseArea.pressed + target: range + property: "position" + value: fakeHandle.x + } + + // During the drag, we simply ignore position set from the range, this + // means that setting a value while dragging will not "interrupt" the + // dragging activity. + Binding { + when: !mouseArea.drag.active + target: fakeHandle + property: "x" + value: range.position + } +} diff --git a/components/custom/SpinBox.qml b/components/custom/SpinBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..1566a8e7213b3fa677ebb00c4ec03d8be1dad456 --- /dev/null +++ b/components/custom/SpinBox.qml @@ -0,0 +1,142 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles + +Item { + id: spinbox + SystemPalette{id:syspal} + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + width: Math.max(minimumWidth, + input.width + leftMargin + rightMargin) + + height: Math.max(minimumHeight, + input.height + topMargin + bottomMargin) + + property real value: 0.0 + property real maximumValue: 99 + property real minimumValue: 0 + property real singlestep: 1 + + property bool upEnabled: value != maximumValue; + property bool downEnabled: value != minimumValue; + + property alias upPressed: mouseUp.pressed + property alias downPressed: mouseDown.pressed + property alias upHovered: mouseUp.containsMouse + property alias downHovered: mouseDown.containsMouse + property alias containsMouse: mouseArea.containsMouse + property alias activeFocus: input.activeFocus // Forward active focus + + property color backgroundColor: syspal.base + property color textColor: syspal.text + + property Component background: defaultStyle.background + property Component up: defaultStyle.up + property Component down: defaultStyle.down + DefaultStyles.SpinBoxStyle { id: defaultStyle } + + function increment() { + value += singlestep + if (value > maximumValue) + value = maximumValue + input.text = value + } + + function decrement() { + value -= singlestep + if (value < minimumValue) + value = minimumValue + input.text = value + } + + function setValue(v) { + var newval = parseFloat(v) + if (newval > maximumValue) + newval = maximumValue + else if (value < minimumValue) + newval = minimumValue + value = newval + input.text = value + } + + // background + Loader { + id: backgroundComponent + anchors.fill: parent + sourceComponent: background + } + + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + } + + TextInput { + id: input + font.pixelSize: 14 + anchors.margins: 5 + anchors.leftMargin: leftMargin + anchors.topMargin: topMargin + anchors.rightMargin: rightMargin + anchors.bottomMargin: bottomMargin + anchors.fill: parent + selectByMouse: true + text: spinbox.value + validator: DoubleValidator { bottom: 11; top: 31 } + onTextChanged: { spinbox.setValue(text); } + color: textColor + opacity: parent.enabled ? 1 : 0.5 + } + + Loader { + id: upButton + property alias pressed : spinbox.upPressed + property alias hover : spinbox.upHovered + property alias enabled : spinbox.upEnabled + sourceComponent: up + MouseArea { + id: mouseUp + anchors.fill: upButton.item + onClicked: increment() + + property bool autoincrement: false; + onReleased: autoincrement = false + Timer { running: mouseUp.pressed; interval: 350 ; onTriggered: mouseUp.autoincrement = true } + Timer { running: mouseUp.autoincrement; interval: 60 ; repeat: true ; onTriggered: increment() } + } + onLoaded: { + item.parent = spinbox + mouseUp.parent = item + } + } + + Loader { + id: downButton + property alias pressed : spinbox.downPressed + property alias hover : spinbox.downHovered + property alias enabled : spinbox.downEnabled + sourceComponent: down + MouseArea { + id: mouseDown + anchors.fill: downButton.item + onClicked: decrement() + + property bool autoincrement: false; + onReleased: autoincrement = false + Timer { running: mouseDown.pressed; interval: 350 ; onTriggered: mouseDown.autoincrement = true } + Timer { running: mouseDown.autoincrement; interval: 60 ; repeat: true ; onTriggered: decrement() } + } + onLoaded: { + item.parent = spinbox + mouseDown.parent = item + } + } +} diff --git a/components/custom/Switch.qml b/components/custom/Switch.qml new file mode 100644 index 0000000000000000000000000000000000000000..8bf18cfe325401a9dd2f28a38d1a6369afd662a1 --- /dev/null +++ b/components/custom/Switch.qml @@ -0,0 +1,85 @@ +import QtQuick 1.1 +import "./styles/default" as DefaultStyles + +Item { + id: toggleSwitch // "switch" is a reserved word + + signal clicked + property bool pressed: mouseArea.pressed + property bool checked: false + property alias containsMouse: mouseArea.containsMouse + + property color switchColor: syspal.button + property color backgroundColor: syspal.alternateBase + property color positiveHighlightColor: syspal.highlight + property color negativeHighlightColor: "transparent" + property color textColor: syspal.text + + property Component groove: defaultStyle.groove + property Component handle: defaultStyle.handle + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + // implementation + + implicitWidth: Math.max(minimumWidth, grooveLoader.item.implicitWidth) + implicitHeight: Math.max(minimumHeight, grooveLoader.item.implicitHeight) + + onCheckedChanged: snapHandleIntoPlace(); + + Loader { + id: grooveLoader + anchors.fill: parent + property alias styledItem: toggleSwitch + property real handleCenterX: handleLoader.item.x + (handleLoader.item.width/2) + sourceComponent: groove + } + + Loader { + id: handleLoader + anchors.top: parent.top + anchors.bottom: parent.bottom + property alias styledItem: toggleSwitch + sourceComponent: handle + + Component.onCompleted: item.x = checked ? mouseArea.drag.maximumX : mouseArea.drag.minimumX + } + + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + + drag.axis: Drag.XAxis + drag.minimumX: 0 + drag.maximumX: toggleSwitch.width - handleLoader.item.width + drag.target: handleLoader.item + + onPressed: toggleSwitch.pressed = true // needed when hover is enabled + onCanceled: { snapHandleIntoPlace(); toggleSwitch.pressed = false; } // mouse stolen e.g. by Flickable + onReleased: { + var wasChecked = checked; + if (drag.active) { + checked = (handleLoader.item.x > (drag.maximumX - drag.minimumX)/2) + } else if (toggleSwitch.pressed && enabled) { // No click if release outside area + checked = !checked; + } + + snapHandleIntoPlace(); + + toggleSwitch.pressed = false + if(checked != wasChecked) + toggleSwitch.clicked(); + } + } + + onWidthChanged: snapHandleIntoPlace() + function snapHandleIntoPlace() { + if(handleLoader.item) + handleLoader.item.x = checked ? mouseArea.drag.maximumX : mouseArea.drag.minimumX; + } + + DefaultStyles.SwitchStyle { id: defaultStyle } + SystemPalette { id: syspal } +} diff --git a/components/custom/TextArea.qml b/components/custom/TextArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..92263fe6ed00ccf2b541195e0aeea514f0b67002 --- /dev/null +++ b/components/custom/TextArea.qml @@ -0,0 +1,156 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles +import "./behaviors" // TextEditMouseBehavior + +FocusScope { + id: textArea + + property alias text: textEdit.text + property alias placeholderText: placeholderTextComponent.text + property alias font: textEdit.font + property bool passwordMode: false + property bool readOnly: textEdit.readOnly // read only + property int inputHint; // values tbd (alias to TextEdit.inputMethodHints? + property alias selectedText: textEdit.selectedText + property alias selectionEnd: textEdit.selectionEnd + property alias selectionStart: textEdit.selectionStart + property alias horizontalAlignment: textEdit.horizontalAlignment + property alias verticalAlignment: textEdit.verticalAlignment + property alias wrapMode: textEdit.wrapMode //mm Missing from spec + property alias textFormat: textEdit.textFormat + property alias cursorPosition: textEdit.cursorPosition + + property color textColor: syspal.text + property color backgroundColor: syspal.base + property alias containsMouse: mouseEditBehavior.containsMouse + + property Component background: defaultStyle.background + property Component hints: defaultStyle.hints + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + function copy() { + textEdit.copy() + } + + function paste() { + textEdit.paste() + } + + function cut() { + textEdit.cut() + } + + function forceActiveFocus() { + textEdit.forceActiveFocus() + } + + function select(start, end) { + textEdit.select(start, end) + } + + function selectAll() { + textEdit.selectAll() + } + + function selectWord() { + textEdit.selectWord() + } + + function positionAt(x, y) { + var p = mapToItem(textEdit, x, y); + return textEdit.positionAt(p.x, p.y); + } + + function positionToRectangle(pos) { + var p = mapToItem(textEdit, pos.x, pos.y); + return textEdit.positionToRectangle(p); + } + + width: Math.max(minimumWidth, + Math.max(textEdit.width, placeholderTextComponent.width) + leftMargin + rightMargin) + height: Math.max(minimumHeight, + Math.max(textEdit.height, placeholderTextComponent.height) + topMargin + bottomMargin) + + + // Implementation + + property alias activeFocus: textEdit.activeFocus // Forward active focus + property alias desktopBehavior: mouseEditBehavior.desktopBehavior + property alias _hints: hintsLoader.item + clip: true + + SystemPalette { id: syspal } + Loader { id: hintsLoader; sourceComponent: hints } + Loader { sourceComponent: background; anchors.fill: parent } + + Flickable { //mm is FocusScope, so TextArea's root doesn't need to be, no? + id: flickable + clip: true + + anchors.fill: parent + anchors.leftMargin: leftMargin + anchors.topMargin: topMargin + anchors.rightMargin: rightMargin + anchors.bottomMargin: bottomMargin + + function ensureVisible(r) { + if (contentX >= r.x) + contentX = r.x; + else if (contentX+width <= r.x+r.width) + contentX = r.x+r.width-width; + if (contentY >= r.y) + contentY = r.y; + else if (contentY+height <= r.y+r.height) + contentY = r.y+r.height-height; + } + + TextEdit { // see QTBUG-14936 + id: textEdit + font.pixelSize: _hints.fontPixelSize + font.bold: _hints.fontBold + + anchors.top: parent.top + anchors.left: parent.left + anchors.right: parent.right + + color: enabled ? textColor: Qt.tint(textColor, "#80ffffff") + wrapMode: desktopBehavior ? TextEdit.NoWrap : TextEdit.WordWrap + onCursorRectangleChanged: flickable.ensureVisible(cursorRectangle) + + onActiveFocusChanged: activeFocus ? openSoftwareInputPanel() : closeSoftwareInputPanel() + } + } + + Text { + id: placeholderTextComponent + x: leftMargin; y: topMargin + font: textEdit.font + opacity: !textEdit.text.length && !textEdit.activeFocus ? 1 : 0 + color: "gray" + clip: true + text: "Enter text" + Behavior on opacity { NumberAnimation { duration: 90 } } + } + + + TextEditMouseBehavior { + id: mouseEditBehavior + anchors.fill: parent + textEdit: textEdit + desktopBehavior: false + copyPasteButtons: ButtonBlock { + opacity: 0 // initially hidden + Behavior on opacity { NumberAnimation { duration: 100 } } + } + } + + DefaultStyles.TextFieldStyle { id: defaultStyle } +} + diff --git a/components/custom/TextField.qml b/components/custom/TextField.qml new file mode 100644 index 0000000000000000000000000000000000000000..2af29427e0a8bce403651fdfcaf42c75fcdee7d3 --- /dev/null +++ b/components/custom/TextField.qml @@ -0,0 +1,196 @@ +import QtQuick 1.0 +import "./styles/default" as DefaultStyles +import "./behaviors" // TextEditMouseBehavior + +// KNOWN ISSUES +// 1) TextField does not loose focus when !enabled if it is a FocusScope (see QTBUG-16161) + +FocusScope { + id: textField + + property alias text: textInput.text + property alias font: textInput.font + + property int inputHint // values tbd + property bool acceptableInput: textInput.acceptableInput // read only + property bool readOnly: textInput.readOnly // read only + property alias placeholderText: placeholderTextComponent.text + property bool passwordMode: false + property alias selectedText: textInput.selectedText + property alias selectionEnd: textInput.selectionEnd + property alias selectionStart: textInput.selectionStart + property alias validator: textInput.validator + property alias inputMask: textInput.inputMask + property alias horizontalalignment: textInput.horizontalAlignment + property alias echoMode: textInput.echoMode + property alias cursorPosition: textInput.cursorPosition + property alias inputMethodHints: textInput.inputMethodHints + + property color textColor: syspal.text + property color backgroundColor: syspal.base + property alias containsMouse: mouseEditBehavior.containsMouse + + property Component background: defaultStyle.background + property Component hints: defaultStyle.hints + + property int minimumWidth: defaultStyle.minimumWidth + property int minimumHeight: defaultStyle.minimumHeight + + property int leftMargin: defaultStyle.leftMargin + property int topMargin: defaultStyle.topMargin + property int rightMargin: defaultStyle.rightMargin + property int bottomMargin: defaultStyle.bottomMargin + + function copy() { + textInput.copy() + } + + function paste() { + textInput.paste() + } + + function cut() { + textInput.cut() + } + + function select(start, end) { + textInput.select(start, end) + } + + function selectAll() { + textInput.selectAll() + } + + function selectWord() { + textInput.selectWord() + } + + function positionAt(x) { + var p = mapToItem(textInput, x, 0); + return textInput.positionAt(p.x); + } + + function positionToRectangle(pos) { + var p = mapToItem(textInput, pos.x, pos.y); + return textInput.positionToRectangle(p); + } + + width: Math.max(minimumWidth, + textInput.width + leftMargin + rightMargin) + + height: Math.max(minimumHeight, + textInput.height + topMargin + bottomMargin) + + // Forward focus property + property alias activeFocus: textInput.activeFocus + + // Implementation + + property alias desktopBehavior: mouseEditBehavior.desktopBehavior + property alias _hints: hintsLoader.item + clip: true + + SystemPalette { id: syspal } + Loader { id: hintsLoader; sourceComponent: hints } + Loader { sourceComponent: background; anchors.fill:parent} + + TextInput { // see QTBUG-14936 + id: textInput + font.pixelSize: _hints.fontPixelSize + font.bold: _hints.fontBold + + anchors.leftMargin: leftMargin + anchors.topMargin: topMargin + anchors.rightMargin: rightMargin + anchors.bottomMargin: bottomMargin + + anchors.left: parent.left + anchors.right: parent.right + anchors.verticalCenter: parent.verticalCenter + + opacity: desktopBehavior || activeFocus ? 1 : 0 + color: enabled ? textColor : Qt.tint(textColor, "#80ffffff") + echoMode: passwordMode ? _hints.passwordEchoMode : TextInput.Normal + + onActiveFocusChanged: { + if (!desktopBehavior) + state = (activeFocus ? "focused" : "") + + if (activeFocus) + openSoftwareInputPanel() + else + closeSoftwareInputPanel() + } + + states: [ + State { + name: "" + PropertyChanges { target: textInput; cursorPosition: 0 } + }, + State { + name: "focused" + PropertyChanges { target: textInput; cursorPosition: textInput.text.length } + } + ] + + transitions: Transition { + to: "focused" + SequentialAnimation { + ScriptAction { script: textInput.cursorVisible = false; } + ScriptAction { script: textInput.cursorPosition = textInput.positionAt(textInput.width); } + NumberAnimation { target: textInput; property: "cursorPosition"; duration: 150 } + ScriptAction { script: textInput.cursorVisible = true; } + } + } + } + + Text { + id: placeholderTextComponent + anchors.fill: textInput + font: textInput.font + opacity: !textInput.text.length && !textInput.activeFocus ? 1 : 0 + color: "gray" + text: "Enter text" + Behavior on opacity { NumberAnimation { duration: 90 } } + } + + Text { + id: unfocusedText + clip: true + anchors.fill: textInput + font: textInput.font + opacity: !desktopBehavior && !passwordMode && textInput.text.length && !textInput.activeFocus ? 1 : 0 + color: textInput.color + elide: Text.ElideRight + text: textInput.text + } + + + TextEditMouseBehavior { + id: mouseEditBehavior + anchors.fill: parent + textInput: textInput + desktopBehavior: false + copyPasteButtons: ButtonBlock { + opacity: 0 // initially hidden + Behavior on opacity { NumberAnimation { duration: 100 } } + } + } + + DefaultStyles.TextFieldStyle { id: defaultStyle } +} + + + + + + + + + + + + + + + diff --git a/components/custom/behaviors/ButtonBehavior.qml b/components/custom/behaviors/ButtonBehavior.qml new file mode 100644 index 0000000000000000000000000000000000000000..391c870a77fc729bb9f777bc831f573bddbb6d72 --- /dev/null +++ b/components/custom/behaviors/ButtonBehavior.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 + +Item { + id: behavior + + signal clicked + property bool pressed: false // Can't be alias of mouseArea.pressed because the latter is read-only + property alias containsMouse: mouseArea.containsMouse + property bool checkable: false + property bool checked: false + property bool triState: false + + onCheckableChanged: { if(!checkable) checked = false } + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + onPressed: behavior.pressed = true // needed when hover is enabled + onEntered: if(pressed && enabled) behavior.pressed = true + onExited: behavior.pressed = false + onCanceled: behavior.pressed = false // mouse stolen e.g. by Flickable + onReleased: { + if(behavior.pressed && behavior.enabled) { // No click if release outside area + behavior.pressed = false + if(behavior.checkable) + behavior.checked = !behavior.checked; + behavior.clicked() + } + } + } +} diff --git a/components/custom/behaviors/ModalPopupBehavior.qml b/components/custom/behaviors/ModalPopupBehavior.qml new file mode 100644 index 0000000000000000000000000000000000000000..108bf02ae219b761964e315726500582dfae1d5a --- /dev/null +++ b/components/custom/behaviors/ModalPopupBehavior.qml @@ -0,0 +1,90 @@ +import QtQuick 1.0 + +// KNOWN ISSUES +// none + +Item { + id: popupBehavior + + property bool showing: false + property bool whenAlso: true // modifier to the "showing" property + property bool consumeCancelClick: true + property int delay: 0 // delay before popout becomes visible + property int deallocationDelay: 3000 // 3 seconds + + property Component popupComponent + + property alias popup: popupLoader.item // read-only + property alias window: popupBehavior.root // read-only + + signal prepareToShow + signal prepareToHide + signal cancelledByClick + + // implementation + + anchors.fill: parent + + onShowingChanged: notifyChange() + onWhenAlsoChanged: notifyChange() + function notifyChange() { + if(showing && whenAlso) { + if(popupLoader.sourceComponent == undefined) { + popupLoader.sourceComponent = popupComponent; + } + } else { + mouseArea.enabled = false; // disable before opacity is changed in case it has fading behavior + if(Qt.isQtObject(popupLoader.item)) { + popupBehavior.prepareToHide(); + popupLoader.item.opacity = 0; + } + } + } + + property Item root: findRoot() + function findRoot() { + var p = parent; + while(p.parent != undefined) + p = p.parent; + + return p; + } + + MouseArea { + id: mouseArea + anchors.fill: parent + enabled: false // enabled only when popout is showing + onPressed: { + popupBehavior.showing = false; + mouse.accepted = consumeCancelClick; + cancelledByClick(); + } + } + + Loader { + id: popupLoader + } + + Timer { // visibility timer + running: Qt.isQtObject(popupLoader.item) && showing && whenAlso + interval: delay + onTriggered: { + popupBehavior.prepareToShow(); + mouseArea.enabled = true; + popup.opacity = 1; + } + } + + Timer { // deallocation timer + running: Qt.isQtObject(popupLoader.item) && popupLoader.item.opacity == 0 + interval: deallocationDelay + onTriggered: popupLoader.sourceComponent = undefined + } + + states: State { + name: "active" + when: Qt.isQtObject(popupLoader.item) && popupLoader.item.opacity > 0 + ParentChange { target: popupBehavior; parent: root } + } + } + diff --git a/components/custom/behaviors/TextEditMouseBehavior.qml b/components/custom/behaviors/TextEditMouseBehavior.qml new file mode 100644 index 0000000000000000000000000000000000000000..02175d5b956a41d9e97713957a26fd7b1248d4b4 --- /dev/null +++ b/components/custom/behaviors/TextEditMouseBehavior.qml @@ -0,0 +1,266 @@ +import QtQuick 1.0 + +// KNOWN ISSUES +// 1) Can't tell if the Paste button should be shown or not, see QTBUG-16190 +// 2) Hard to tell if the Select button should be shown (part of QTBUG-16190?) + +Item { + id: mouseBehavior + + property TextInput textInput + property TextEdit textEdit + property Flickable flickable + property bool desktopBehavior: true + property alias containsMouse: mouseArea.containsMouse + + property Component copyPasteButtons + + // Implementation + + property Item textEditor: Qt.isQtObject(textInput) ? textInput : textEdit + + Connections { + target: textInput + onTextChanged: reset() + onCursorPositionChanged: reset() + onSelectedTextChanged: reset() + onActiveFocusChanged: reset() + } + + function reset() { + copyPastePopup.showing = false; // hide and restart the visibility timer + copyPastePopup.showing = selectedText.length > 0 && !desktopBehavior; + copyPastePopup.wasCancelledByClick = false; + copyPastePopup.wasClosedByCopy = false; + } + + Component.onCompleted: { + textEditor.focus = true; + textEditor.selectByMouse = false; + textEditor.activeFocusOnPress = false; + } + + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + drag.target: Item {} // work-around for Flickable stealing the mouse, which is expected (?), see QTBUG-15231 + + property int pressedPos + property int selectionStartAtPress + property int selectionEndAtPress + property bool hadFocusBeforePress + property bool draggingStartHandle + property bool draggingEndHandle + + function characterPositionAt(mouse) { + var mappedMouse = mapToItem(textEditor, mouse.x, mouse.y); + if(Qt.isQtObject(textInput)) { + return textInput.positionAt(mappedMouse.x); + } else { + return textEdit.positionAt(mappedMouse.x, mappedMouse.y); + } + } + + //mm see QTBUG-15814 + onPressed: { + hadFocusBeforePress = textEditor.activeFocus; + selectionStartAtPress = textEditor.selectionStart; + selectionEndAtPress = textEditor.selectionEnd; + + textEditor.forceActiveFocus(); //mm see QTBUG-16157 + var pos = characterPositionAt(mouse); + if(desktopBehavior) { + textEditor.cursorPosition = pos; + } else { + draggingStartHandle = false; // reset both to false, i.e. no dragging of selection endpoints + draggingEndHandle = false; + if(textEditor.selectionStart != textEditor.selectionEnd) { + draggingEndHandle = (pos > selectionStartAtPress + (selectionEndAtPress-selectionStartAtPress)/2); + draggingStartHandle = !draggingEndHandle; + } + } + + pressedPos = textEditor.cursorPosition; + } + + onPressAndHold: { + if(!desktopBehavior) { + if(!textEditor.selectedText.length) { //mm Somehow just having the onPressAndHold impementation interrupts the text selection + textEditor.cursorPosition = characterPositionAt(mouse); + } + } + } + + onReleased: { + if(!desktopBehavior) { + if(mouse.wasHeld) copyPastePopup.showing = true; + } + } + + onClicked: { + if(!desktopBehavior) { + if(!hadFocusBeforePress) + return; + + var pos = characterPositionAt(mouse); + if(textEditor.selectedText.length) { // clicked on or outside selection + if(copyPastePopup.wasClosedByCopy && pos >= textEditor.selectionStart && pos < textEditor.selectionEnd) + copyPastePopup.showing = true; + else + textEditor.select(textEditor.selectionEnd, textEditor.selectionEnd); //mm use deselect() from QtQuick 1.1 (see QTBUG-16059) + } else { // clicked while there's no selection + if(pos == textEditor.cursorPosition) { // Clicked where the cursor already where + copyPastePopup.showing = !copyPastePopup.wasCancelledByClick; + copyPastePopup.wasCancelledByClick = false; + } else { // Clicked in a new place (where the cursor wasn't) + var endOfWordRegEx = /[^\b]\b/g; + endOfWordRegEx.lastIndex = pos; + var endOfWordPosition = pos; + if(endOfWordRegEx.test(textEditor.text)) // updates lastIndex + endOfWordPosition = endOfWordRegEx.lastIndex; + + if(textEditor.cursorPosition != endOfWordPosition) { + textEditor.cursorPosition = endOfWordPosition; + } else { + copyPastePopup.showing = !copyPastePopup.wasCancelledByClick; + copyPastePopup.wasCancelledByClick = false; + } + } + } + } + + textEditor.openSoftwareInputPanel() + } + + onPositionChanged: { + if(!pressed) + return; + + var pos = characterPositionAt(mouse); + if(desktopBehavior) { + textEditor.select(pressedPos, pos); + } else { + if(draggingStartHandle) { + textEditor.select(selectionEndAtPress, pos); + } else if(draggingEndHandle) { + textEditor.select(selectionStartAtPress, pos); + } else if(mouse.wasHeld && textEditor.cursorPosition != pos) { // there's no selection + textEditor.cursorPosition = pos; + copyPastePopup.showing = true; // show once not pressed any more + } + } + } + + onDoubleClicked: { + if(desktopBehavior) + textEditor.selectAll(); + else { + textEditor.cursorPosition = characterPositionAt(mouse); + textEditor.selectWord(); // select word at cursor position + } + } + } + + + function selectionPopoutPoint() { + var point = {x:0, y:0} + + var selectionStartRect = textEditor.positionToRectangle(textEditor.selectionStart); + var mappedStartPoint = mapFromItem(textEditor, selectionStartRect.x, selectionStartRect.y); + mappedStartPoint.x = Math.max(mappedStartPoint.x, 0); + mappedStartPoint.y = Math.max(mappedStartPoint.y, 0); + + var selectioEndRect = textEditor.positionToRectangle(textEditor.selectionEnd); + var mappedEndPoint = mapFromItem(textEditor, selectioEndRect.x, selectioEndRect.y); + mappedEndPoint.x = Math.min(mappedEndPoint.x, textEditor.width); + mappedEndPoint.y = Math.min(mappedEndPoint.y, textEditor.height); + + var multilineSelection = (selectionStartRect.y != selectioEndRect.y); + if(!multilineSelection) { + point.x = mappedStartPoint.x + (mappedEndPoint.x-mappedStartPoint.x)/2 + } else { + point.x = textEditor.x + textEdit.width/2; + } + + point.y = mappedStartPoint.y + return point; + } + + Item { + id: copyPastePopup + + property alias showing: modalPopup.showing + property bool wasCancelledByClick: false + property bool wasClosedByCopy: false + + property bool showCopyAction: textEditor.selectedText.length > 0 + property bool showCutAction: textEditor.selectedText.length > 0 + property bool showPasteAction: true //textEditor.canPaste //mm see QTBUG-16190 + property bool showSelectAction: textEditor.text.length > 0 && textEditor.selectedText.length < textEditor.text.length // need canSelectWord, see QTBUG-16190 + property bool showSelectAllAction: textEditor.text.length > 0 && textEditor.selectedText.length < textEditor.text.length + + ListModel { + id: popupButtonModel + ListElement { text: "Copy"; opacity: 0 } // index: 0 + ListElement { text: "Cut"; opacity: 0 } // index: 1 + ListElement { text: "Paste"; opacity: 1 } // index: 2 + ListElement { text: "Select"; opacity: 0 } // index: 3 + ListElement { text: "Select all"; opacity: 0 } // index: 4 + } + + onShowCopyActionChanged: popupButtonModel.setProperty(0, "opacity", showCopyAction ? 1 : 0); + onShowCutActionChanged: popupButtonModel.setProperty(1, "opacity", showCutAction ? 1 : 0); + onShowPasteActionChanged: popupButtonModel.setProperty(2, "opacity", showPasteAction ? 1 : 0); + onShowSelectActionChanged: popupButtonModel.setProperty(3, "opacity", showSelectAction ? 1 : 0); + onShowSelectAllActionChanged: popupButtonModel.setProperty(4, "opacity", showSelectAllAction ? 1 : 0); + + Component.onCompleted: { + showCopyActionChanged(); + showCutActionChanged(); + showPasteActionChanged(); + showSelectActionChanged(); + showSelectAllActionChanged(); + } + + function positionPopout(popup, window) { // position poput above the text field's cursor + var popoutPoint = selectionPopoutPoint(); + var mappedPos = mapToItem(window, popoutPoint.x, popoutPoint.y); + popup.x = Math.max(mappedPos.x - popup.width/2, 0); + if(popup.x+popup.width > window.width) + popup.x = window.width-popup.width; + + popup.y = mappedPos.y - popup.height; + if(popup.y < 0) + popup.y += popup.height + textEditor.height; + } + + ModalPopupBehavior { + id: modalPopup + consumeCancelClick: false + whenAlso: !mouseArea.pressed + delay: 300 + onPrepareToShow: copyPastePopup.positionPopout(popup, window) + onCancelledByClick: copyPastePopup.wasCancelledByClick = true + + popupComponent: copyPasteButtons + onPopupChanged: if(popup) popup.model = popupButtonModel + + Connections { + target: modalPopup.popup + onClicked: { + if(index == 0) { + textEditor.copy(); + copyPastePopup.showing = false; + copyPastePopup.wasClosedByCopy = true; + } + if(index == 1) textEditor.cut(); + if(index == 2) textEditor.paste(); + if(index == 3) textEditor.selectWord(); + if(index == 4) textEditor.selectAll(); + } + } + } + } +} + diff --git a/components/custom/components.pro b/components/custom/components.pro new file mode 100644 index 0000000000000000000000000000000000000000..be0663d70948a7a81825e655ec3cf553ebb4de43 --- /dev/null +++ b/components/custom/components.pro @@ -0,0 +1,49 @@ +TEMPLATE = subdirs # XXX: Avoid call the linker +TARGETPATH = Qt/labs/components/custom + +symbian { + INSTALL_IMPORTS = /resource/qt/imports +} else { + INSTALL_IMPORTS = $$[QT_INSTALL_IMPORTS] +} + +QML_FILES = \ + qmldir \ + BasicButton.qml \ + BusyIndicator.qml \ + ButtonBlock.qml \ + ButtonColumn.qml \ + ButtonRow.qml \ + ButtonGroup.js \ + Button.qml \ + CheckBox.qml \ + ChoiceList.qml \ + ProgressBar.qml \ + RadioButton.qml \ + ScrollDecorator.qml \ + ScrollIndicator.qml \ + Slider.qml \ + SpinBox.qml \ + Switch.qml \ + TextArea.qml \ + TextField.qml + +QML_DIRS = \ + behaviors \ + private \ + styles \ + visuals + +qmlfiles.files = $$QML_FILES +qmlfiles.sources = $$QML_FILES +qmlfiles.path = $$INSTALL_IMPORTS/$$TARGETPATH + +qmldirs.files = $$QML_DIRS +qmldirs.sources = $$QML_DIRS +qmldirs.path = $$INSTALL_IMPORTS/$$TARGETPATH + +INSTALLS += qmlfiles qmldirs + +symbian { + DEPLOYMENT += qmlfiles qmldirs +} diff --git a/components/custom/private/ChoiceListPopup.qml b/components/custom/private/ChoiceListPopup.qml new file mode 100644 index 0000000000000000000000000000000000000000..61a0d22203ea32d942ad4e2aa1377b5cdae60b75 --- /dev/null +++ b/components/custom/private/ChoiceListPopup.qml @@ -0,0 +1,269 @@ +import QtQuick 1.0 + +MouseArea { + id: popup + // There is no global toplevel so we have to make one + // We essentially reparent this item to the root item + Component.onCompleted: { + var p = parent; + while (p.parent != undefined) + p = p.parent + parent = p; + } + + anchors.fill: parent // fill the while app area + opacity: popupFrameLoader.item.opacity // let the frame control opacity, so it can set the behavior + + property string behavior: "MacOS" + property bool desktopBehavior: (behavior == "MacOS" || behavior == "Windows" || behavior == "Linux") + property int previousCurrentIndex: -1 // set in state transition last in this file + + property alias model: listView.model + property alias currentIndex: listView.currentIndex + + property Component listItem + property Component listHighlight + property Component popupFrame + + function togglePopup() { state = (state == "" ? "hidden" : "") } + function setCurrentIndex(index) { listView.currentIndex = index; } + function cancelSelection() { listView.currentIndex = previousCurrentIndex; } + function closePopup() { state = "hidden" } + + function positionPopup() { + switch(behavior) { + case "MacOS": + var mappedListPos = mapFromItem(choiceList, 0, 0); + var itemHeight = Math.max(listView.contentHeight/listView.count, 0); + var currentItemY = Math.max(currentIndex*itemHeight, 0); + currentItemY += Math.floor(itemHeight/2 - choiceList.height/2); // correct for choiceLists that are higher than items in the list + + listView.y = mappedListPos.y - currentItemY; + listView.x = mappedListPos.x; + + listView.width = choiceList.width; + listView.height = listView.contentHeight //mm see QTBUG-16037 + + if(listView.y < topMargin) { + var excess = Math.floor(currentItemY - mappedListPos.y); + listView.y = topMargin; + listView.height += excess; + listView.contentY = excess + topMargin; + + if(listView.contentY != excess+topMargin) //mm setting listView.height seems to make it worse + print("!!! ChoiceListPopup.qml: listView.contentY should be " + excess+topMargin + " but is " + listView.contentY) + } + + if(listView.height+listView.contentY > listView.contentHeight) { + listView.height = listView.contentHeight-listView.contentY; + } + + if(listView.y+listView.height+bottomMargin > popup.height) { + listView.height = popup.height-listView.y-bottomMargin; + } + break; + case "Windows": + var point = popup.mapFromItem(choiceList, 0, listView.height); + listView.y = point.y; + listView.x = point.x; + + listView.width = choiceList.width; + listView.height = 200; + + break; + case "MeeGo": + break; + } + } + + Loader { + id: popupFrameLoader + property alias styledItem: popup.parent + anchors.fill: listView + anchors.leftMargin: -item.anchors.leftMargin + anchors.rightMargin: -item.anchors.rightMargin + anchors.topMargin: -item.anchors.topMargin + anchors.bottomMargin: -item.anchors.bottomMargin + sourceComponent: popupFrame + } + + ListView { + id: listView + focus: true + boundsBehavior: desktopBehavior ? ListView.StopAtBounds : ListView.DragOverBounds + keyNavigationWraps: !desktopBehavior + highlightFollowsCurrentItem: false // explicitly handled below + + interactive: !desktopBehavior // disable flicking. also disables key handling + onCurrentItemChanged: { + if(desktopBehavior) { + positionViewAtIndex(currentIndex, ListView.Contain); + } + } + + property int highlightedIndex: -1 + onHighlightedIndexChanged: positionViewAtIndex(highlightedIndex, ListView.Contain) + + property variant highlightedItem: null + onHighlightedItemChanged: { + if(desktopBehavior) { + positionHighlight(); + } + } + + function positionHighlight() { + if(!Qt.isQtObject(highlightItem)) + return; + + if(!Qt.isQtObject(highlightedItem)) { + highlightItem.opacity = 0; // hide when no item is highlighted + } else { + highlightItem.x = highlightedItem.x; + highlightItem.y = highlightedItem.y; + highlightItem.width = highlightedItem.width; + highlightItem.height = highlightedItem.height; + highlightItem.opacity = 1; // show once positioned + } + } + + function hideHighlight() { + highlightedIndex = -1; + highlightedItem = null; // will trigger positionHighlight() what will hide the highlight + } + + delegate: Item { + id: itemDelegate + width: delegateLoader.item.width + height: delegateLoader.item.height + property int theIndex: index // for some reason the loader can't bind directly to the "index" + + Loader { + id: delegateLoader + property variant model: listView.model + property alias index: itemDelegate.theIndex //mm Somehow the "model" gets through automagically, but not index + property Item styledItem: choiceList + property bool highlighted: theIndex == listView.highlightedIndex + sourceComponent: listItem + MouseArea { // handle list selection on mobile platforms + id:itemMouseArea + anchors.fill: parent + onClicked: { setCurrentIndex(index); closePopup(); } + } + } + + states: State { + name: "highlighted" + when: index == listView.highlightedIndex + StateChangeScript { + script: { + if(Qt.isQtObject(listView.highlightedItem)) { + listView.highlightedItem.yChanged.disconnect(listView.positionHighlight); + } + listView.highlightedItem = itemDelegate; + listView.highlightedItem.yChanged.connect(listView.positionHighlight); + } + } + + } + } + + function firstVisibleItem() { return indexAt(contentX+10,contentY+10); } + function lastVisibleItem() { return indexAt(contentX+width-10,contentY+height-10); } + function itemsPerPage() { return lastVisibleItem() - firstVisibleItem(); } + + Keys.onPressed: { + // with the ListView !interactive (non-flicking) we have to handle arrow keys + if (event.key == Qt.Key_Up) { + if(!highlightedItem) highlightedIndex = lastVisibleItem(); + else if(highlightedIndex > 0) highlightedIndex--; + } else if (event.key == Qt.Key_Down) { + if(!highlightedItem) highlightedIndex = firstVisibleItem(); + else if(highlightedIndex+1 < model.count) highlightedIndex++; + } else if (event.key == Qt.Key_PageUp) { + if(!highlightedItem) highlightedIndex = lastVisibleItem(); + else highlightedIndex = Math.max(highlightedIndex-itemsPerPage(), 0); + } else if (event.key == Qt.Key_PageDown) { + if(!highlightedItem) highlightedIndex = firstVisibleItem(); + else highlightedIndex = Math.min(highlightedIndex+itemsPerPage(), model.count-1); + } else if (event.key == Qt.Key_Home) { + highlightedIndex = 0; + } else if (event.key == Qt.Key_End) { + highlightedIndex = model.count-1; + } else if (event.key == Qt.Key_Enter || event.key == Qt.Key_Return) { + if(highlightedIndex != -1) { + popup.setCurrentIndex(highlightedIndex); + } else { + popup.cancelSelection(); + } + + popup.closePopup(); + } else if (event.key == Qt.Key_Escape) { + popup.cancelSelection(); + popup.closePopup(); + } + event.accepted = true; // consume all keys while popout has focus + } + + highlight: popup.listHighlight + } + + enabled: (state != "hidden") // to avoid stray events when poput is about to close + hoverEnabled: true + onClicked: { popup.cancelSelection(); popup.closePopup(); } // clicked outside the list + onPressed: { + var mappedPos = mapToItem(listView.contentItem, mouse.x, mouse.y); + var indexAt = listView.indexAt(mappedPos.x, mappedPos.y); + if(indexAt != -1) { + listView.currentIndex = indexAt; + } + } + onPositionChanged: { + var mappedPos = mapToItem(listView.contentItem, mouse.x, mouse.y); + var indexAt = listView.indexAt(mappedPos.x, mappedPos.y); + if(!pressed) { // hovering + if(indexAt == listView.highlightedIndex) + return; + + if(indexAt >= 0) { + listView.highlightedIndex = indexAt; + } else { + if(mouse.y > listView.y+listView.height && listView.highlightedIndex+1 < listView.count ) { + listView.highlightedIndex++; + } else if(mouse.y < listView.y && listView.highlightedIndex > 0) { + listView.highlightedIndex--; + } else if(mouse.x < popupFrameLoader.x || mouse.x > popupFrameLoader.x+popupFrameLoader.width) { + listView.hideHighlight(); + } + } + } + } + + state: "hidden" // hidden by default + states: [ + State { + name: "" // not hidden, i.e. showing + PropertyChanges { target: popupFrameLoader.item; opacity: 1 } + }, + State { + name: "hidden" + PropertyChanges { target: popupFrameLoader.item; opacity: 0 } + } + ] + + transitions: [ + Transition { to: ""; + ScriptAction { + script: { + previousCurrentIndex = currentIndex; + positionPopup(); + listView.forceActiveFocus(); + } + } + }, + Transition { to: "hidden"; ScriptAction { script: listView.hideHighlight(); } } + ] +} + + + + diff --git a/components/custom/qmldir b/components/custom/qmldir new file mode 100644 index 0000000000000000000000000000000000000000..6e256a81d47c3ed8f88c300def7e2a36836a6d12 --- /dev/null +++ b/components/custom/qmldir @@ -0,0 +1,17 @@ +BasicButton 1.0 BasicButton.qml +BusyIndicator 1.0 BusyIndicator.qml +ButtonBlock 1.0 ButtonBlock.qml +Button 1.0 Button.qml +ButtonColumn 1.0 ButtonColumn.qml +ButtonRow 1.0 ButtonRow.qml +CheckBox 1.0 CheckBox.qml +ChoiceList 1.0 ChoiceList.qml +ProgressBar 1.0 ProgressBar.qml +RadioButton 1.0 RadioButton.qml +ScrollDecorator 1.0 ScrollDecorator.qml +ScrollIndicator 1.0 ScrollIndicator.qml +Slider 1.0 Slider.qml +SpinBox 1.0 SpinBox.qml +Switch 1.0 Switch.qml +TextArea 1.0 TextArea.qml +TextField 1.0 TextField.qml diff --git a/components/custom/styles/default/BasicButtonStyle.qml b/components/custom/styles/default/BasicButtonStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..08b18040e202abaaebcad38d2fefbcabc84f7909 --- /dev/null +++ b/components/custom/styles/default/BasicButtonStyle.qml @@ -0,0 +1,8 @@ +import QtQuick 1.1 + +QtObject { + property int minimumWidth: 40 + property int minimumHeight: 25 + + property Component background: Component { Item { } } +} diff --git a/components/custom/styles/default/BusyIndicatorStyle.qml b/components/custom/styles/default/BusyIndicatorStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..83ef46fd71403eb5e4dc07e47006b95caa9fcca7 --- /dev/null +++ b/components/custom/styles/default/BusyIndicatorStyle.qml @@ -0,0 +1,21 @@ +import QtQuick 1.0 + +QtObject { + property Component background: + Component { + Image { + opacity: running ? 1.0 : 0.7 //mm Should the rotation fade and stop when indicator is !enabled? + source: "images/spinner.png"; + fillMode: Image.PreserveAspectFit + smooth: true + + property int steps: 12 + property int rotationStep: 0 + rotation: rotationStep*(360/steps) + NumberAnimation on rotationStep { + running: busyIndicator.running; from: 0; to: steps; //mm see QTBUG-15652 + loops: Animation.Infinite; duration: 1000 // 1s per revolution + } + } + } +} diff --git a/components/custom/styles/default/ButtonBlockStyle.qml b/components/custom/styles/default/ButtonBlockStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..f0bed8654275b9f32e005c1508eeb8d7921df2f3 --- /dev/null +++ b/components/custom/styles/default/ButtonBlockStyle.qml @@ -0,0 +1,77 @@ +import QtQuick 1.0 + +QtObject { + property int minimumWidth: 90 + property int minimumHeight: 32 + + property int leftMargin : 8 + property int topMargin: 8 + property int rightMargin: 8 + property int bottomMargin: 8 + + property Component background: + Component { + id: defaultBackground + Item { + opacity: enabled ? 1 : 0.7 + Rectangle { // Background center fill + anchors.fill: parent + anchors.leftMargin: adjoining&Qt.Horizontal ? 0 : 2 + anchors.rightMargin: anchors.leftMargin + anchors.topMargin: adjoining&Qt.Vertical ? 0 : 2 + anchors.bottomMargin: anchors.topMargin + + radius: adjoining ? 0 : 5 + color: !styledItem.checked ? backgroundColor : Qt.darker(backgroundColor) + } + BorderImage { + anchors.fill: parent + smooth: true + source: { + if(!adjoining) + return styledItem.pressed || styledItem.checked ? "images/button_pressed.png" : "images/button_normal.png"; + else if(adjoining&Qt.Horizontal) + return styledItem.pressed || styledItem.checked ? "images/buttongroup_h_pressed.png" : "images/buttongroup_h_normal.png"; + else // adjoining&Qt.Vertical + return styledItem.pressed || styledItem.checked ? "images/buttongroup_v_pressed.png" : "images/buttongroup_v_normal.png"; + } + border.left: 6; border.top: 6 + border.right: 6; border.bottom: 6 + } + } + } + + property Component label: + Component { + id: defaultLabel + Item { + width: row.width + height: row.height + anchors.centerIn: parent //mm see QTBUG-15619 + opacity: styledItem.enabled ? 1 : 0.5 + transform: Translate { + x: styledItem.pressed || styledItem.checked ? 1 : 0 + y: styledItem.pressed || styledItem.checked ? 1 : 0 + } + + Row { + id: row + anchors.centerIn: parent + spacing: 4 + Image { + source: styledItem.iconSource + anchors.verticalCenter: parent.verticalCenter + fillMode: Image.Stretch //mm Image should shrink if button is too small, depends on QTBUG-14957 + } + + Text { + color: styledItem.textColor + anchors.verticalCenter: parent.verticalCenter + text: styledItem.text + horizontalAlignment: Text.Center + elide: Text.ElideRight //mm can't make layout work as desired without implicit size support, see QTBUG-14957 + } + } + } + } +} diff --git a/components/custom/styles/default/ButtonStyle.qml b/components/custom/styles/default/ButtonStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..a0e5f33024f9a6aa9ca3f2d3d60f17edf606e14e --- /dev/null +++ b/components/custom/styles/default/ButtonStyle.qml @@ -0,0 +1,102 @@ +import QtQuick 1.1 + +QtObject { + property int minimumWidth: 90 + property int minimumHeight: 32 + + property int leftMargin: 8 + property int topMargin: 8 + property int rightMargin: 8 + property int bottomMargin: 8 + + property Component background: Component { + Item { + opacity: enabled ? 1 : 0.7 + clip: true // clip connected buttons, as they overlap to remove the rounded edjes + property bool isPositioned: position != "only" // only evaluate for rows and columns + + Item { + anchors.fill: parent + // Give connected buttons a negative styling margin, to make + // them overlap and the rounded edge can be clipped away + anchors.leftMargin: isPositioned && (position == "rightmost" || position =="h_middle") ? -leftMargin : 0 + anchors.rightMargin: isPositioned && (position == "leftmost" || position =="h_middle") ? -rightMargin : 0 + anchors.topMargin: isPositioned && (position == "bottom" || position =="v_middle") ? -bottomMargin : 0 + anchors.bottomMargin: isPositioned && (position == "top" || position =="v_middle") ? -topMargin : 0 + + Rectangle { // Background center fill + anchors.fill: parent + anchors.leftMargin: anchors.leftMargin + anchors.rightMargin: anchors.rightMargin + anchors.topMargin: anchors.topMargin + anchors.bottomMargin: anchors.bottomMargin + radius: 5 + color: backgroundColor + } + BorderImage { + anchors.fill: parent + smooth: true + source: pressed || checked ? "images/button_pressed.png" : "images/button_normal.png"; + border.left: 6; border.top: 6 + border.right: 6; border.bottom: 6 + } + } + + // Draw straight border lines between connected buttons + Rectangle { + width: 1 + visible: isPositioned && !checked && !pressed && (position == "rightmost" || position == "h_middle") + anchors.top: parent.top + anchors.topMargin: 2 + anchors.bottomMargin: 2 + anchors.bottom: parent.bottom + anchors.left: parent.left + opacity: 0.4 + color: "white" + } + Rectangle { + width: 1 + opacity: 0.4 + visible: isPositioned && !checked && !pressed && (position == "leftmost" || position == "h_middle") + anchors.top: parent.top + anchors.topMargin: 2 + anchors.bottomMargin: 2 + anchors.bottom: parent.bottom + anchors.right: parent.right + color: "black" + } + } + } + + property Component label: Component { + Item { + implicitWidth: row.implicitWidth + implicitHeight: row.implicitHeight + + opacity: enabled ? 1 : 0.5 + transform: Translate { + x: pressed || checked ? 1 : 0 + y: pressed || checked ? 1 : 0 + } + + Row { + id: row + anchors.centerIn: parent + spacing: 4 + Image { + source: iconSource + anchors.verticalCenter: parent.verticalCenter + fillMode: Image.Stretch //mm Image should shrink if button is too small, depends on QTBUG-14957 + } + + Text { + color: textColor + anchors.verticalCenter: parent.verticalCenter + text: styledItem.text + horizontalAlignment: Text.Center + elide: Text.ElideRight //mm can't make layout work as desired without implicit size support, see QTBUG-14957 + } + } + } + } +} diff --git a/components/custom/styles/default/CheckBoxStyle.qml b/components/custom/styles/default/CheckBoxStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..f5e5fdded9cd84e3245baa4a375b17f9cfe08849 --- /dev/null +++ b/components/custom/styles/default/CheckBoxStyle.qml @@ -0,0 +1,34 @@ +import QtQuick 1.1 + +QtObject { + property int minimumWidth: 32 + property int minimumHeight: 32 + + property Component background: Component { + Item { + width: styledItem.implicitWidth; height: styledItem.implicitHeight + opacity: enabled ? 1 : 0.7 + Rectangle { // Background center fill + anchors.fill: parent + anchors.margins: 1 + radius: 5 + color: backgroundColor + } + BorderImage { + anchors.fill: parent + source: "images/lineedit_normal.png" + smooth: true + border.left: 6; border.top: 3 + border.right: 6; border.bottom: 3 + } + } + } + + property Component checkmark: Component { + Image { + source: "images/checkbox_check.png" + opacity: (!enabled && checked) || pressed == true ? 0.5 : (!checked ? 0 : 1) + Behavior on opacity { NumberAnimation { duration: 150; easing.type: Easing.OutCubic } } + } + } +} diff --git a/components/custom/styles/default/ChoiceListStyle.qml b/components/custom/styles/default/ChoiceListStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..12b10ffd5fb3ae205225c123ce9fbcde38ffb606 --- /dev/null +++ b/components/custom/styles/default/ChoiceListStyle.qml @@ -0,0 +1,100 @@ +import QtQuick 1.0 + +QtObject { + property int minimumWidth: 200 + property int minimumHeight: 32 + + property int leftMargin: 8 + property int topMargin: 8 + property int rightMargin: 34 + property int bottomMargin: 8 + + property Component background: Component { + Item { + opacity: enabled ? 1 : 0.8 + Rectangle { // Background center fill + anchors.fill: parent + anchors.margins: 1 + color: backgroundColor + radius: 5 + } + BorderImage { + anchors.fill: parent + id: backgroundimage + smooth: true + source: styledItem.pressed ? "images/button_pressed.png" : "images/button_normal.png" + width: 80; height: 24 + border.left: 3; border.top: 3 + border.right: 3; border.bottom: 3 + Image { + anchors.verticalCenter: parent.verticalCenter + anchors.right: parent.right + anchors.rightMargin: 10 + opacity: enabled ? 1 : 0.3 + source:"images/spinbox_down.png" + } + } + } + } + + property Component label: Component { + Text { + id:t + color: textColor + text: model && currentIndex >= 0 ? model.get(currentIndex).text : "" + opacity: enabled ? 1 : 0.5 + } + } + + // Popout styling + + property Component popupFrame: Component { + Item { + Behavior on opacity { NumberAnimation { easing.type: Easing.OutQuad; duration: 250 } } + + anchors.leftMargin: 6 + anchors.topMargin: 6 + anchors.rightMargin: 7 + anchors.bottomMargin: 6 + + Rectangle { // Background center fill + anchors.fill: parent + anchors.margins: 1 + color: backgroundColor + radius: 5 + } + BorderImage { + anchors.fill: parent + id: backgroundimage + smooth: true + source: "images/button_normal.png" + width: 80; height: 24 + border.left: 5; border.top: 5 + border.right: 5; border.bottom: 5 + } + } + } + + property Component listItem: Component { + Rectangle { + width: styledItem.width + height: Math.max(itemText.height, 28) + color: highlighted ? "#556699" : "transparent" + radius:2 + border.width:1 + border.color:Qt.darker(color) + clip:true + Text { + id: itemText + anchors.verticalCenter: parent.verticalCenter + anchors.left: parent.left + anchors.leftMargin: 6 + + font.bold: index == currentIndex + color: highlighted ? "white" : styledItem.textColor + anchors.margins: 10 + text: model ? model.get(index).text : "" // list properties can't be automatically be added to the scope, so use get() + } + } + } +} diff --git a/components/custom/styles/default/ProgressBarStyle.qml b/components/custom/styles/default/ProgressBarStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..7cb9795c158cb3a8f4dca09e6b1cb914295b3442 --- /dev/null +++ b/components/custom/styles/default/ProgressBarStyle.qml @@ -0,0 +1,142 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the Qt Components project on Qt Labs. +** +** No Commercial Usage +** This file contains pre-release code and may not be distributed. +** You may use this file in accordance with the terms and conditions contained +** in the Technology Preview License Agreement accompanying this package. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** If you have questions regarding the use of this file, please contact +** Nokia at qt-info@nokia.com. +** +****************************************************************************/ + +import QtQuick 1.0 + +QtObject { + property int minimumWidth: 200 + property int minimumHeight: 25 + + property int leftMargin: 0 + property int rightMargin: 0 + property int topMargin: 0 + property int bottomMargin: 0 + + property Component background: Component { + Item{ + Rectangle{ + anchors.fill:parent + width: parent.width-2 + height: parent.height-2 + color: backgroundColor + radius: 5 + } + BorderImage { + anchors.fill:parent + source:"images/progressbar_groove.png" + border.left:10; border.right:10 + border.top:10; border.bottom:10 + } + } + } + + property Component progress: Component { // progress bar, known duration + BorderImage { // green progress indication + id:progress + opacity: styledItem.enabled ? 1: 0.7 + source: complete > 0.95 ? + "images/progressbar_indeterminate.png" : "images/progressbar_fill.png" + border.left:complete > 0.1 ? 6: 2; + border.right:complete > 0.1 ? 6: 2 + border.top:10; border.bottom:10 + clip:true + + Rectangle{ + anchors.fill:progress + color: styledItem.progressColor + z:-1 + radius:2 + smooth:true + clip:true + anchors.rightMargin:0 + anchors.margins:1 + Image { + id: overlay + NumberAnimation on x { + loops:Animation.Infinite; + from:0; + to:-overlay.sourceSize.width; + duration:2000 + } + width:styledItem.width + sourceSize.width + height:styledItem.height + fillMode:Image.Tile + source: "images/progressbar_overlay.png" + } + } + } + } + + property Component indeterminateProgress: Component { // progress bar, unknown duration + Item { + id: bar + anchors.fill:parent + onWidthChanged:indicator.x = width-indicator.width + BorderImage { + id:indicator + opacity: styledItem.enabled ? 1: 0.7 + Behavior on x { + NumberAnimation{easing.type:Easing.Linear; duration:1000} + } + onXChanged: { + var w = bar.width - indicator.width + if (x == w) x = 0 + else if (x==0) x = w + } + width: 80 + height: parent.height + source:"images/progressbar_indeterminate.png" + border.left:10 ; border.right:10 + border.top:10 ; border.bottom:10 + clip:true + + Rectangle{ + anchors.fill:indicator + color: styledItem.progressColor + z:-1 + radius:2 + smooth:true + clip:true + anchors.rightMargin:0 + anchors.margins:1 + Image { + id: overlay + NumberAnimation on x { + loops:Animation.Infinite; + from:0; + to:-overlay.sourceSize.width; + duration:2000 + } + width:styledItem.width + sourceSize.width + height:styledItem.height + fillMode:Image.Tile + source: "images/progressbar_overlay.png" + } + } + } + } + } +} diff --git a/components/custom/styles/default/RadioButtonStyle.qml b/components/custom/styles/default/RadioButtonStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..0381993e40578578e8cd00bdb51631f7a3f4f557 --- /dev/null +++ b/components/custom/styles/default/RadioButtonStyle.qml @@ -0,0 +1,34 @@ +import QtQuick 1.1 +import "tools" as StyleTools + +QtObject { + property int minimumWidth: 32 + property int minimumHeight: 32 + + property Component background: Component { + Item { + width: styledItem.implicitWidth; height: styledItem.implicitHeight + Rectangle { // Background center fill + anchors.fill: parent + anchors.margins: 1 + radius: width/2 + color: backgroundColor + } + Image { + opacity: enabled ? 1 : 0.7 + source: "images/radiobutton_normal.png" + fillMode: Image.Stretch + anchors.centerIn: parent + } + } + } + + property Component checkmark: Component { + Image { + StyleTools.ColorConverter { id: cc; color: backgroundColor } + source: cc.grayValue() < 70? "images/radiobutton_check_white.png" : "images/radiobutton_check.png" + opacity: (!enabled && checked) || pressed == true ? 0.5 : (!checked ? 0 : 1) + Behavior on opacity { NumberAnimation { duration: 150; easing.type: Easing.OutCubic } } + } + } +} diff --git a/components/custom/styles/default/ScrollIndicatorStyle.qml b/components/custom/styles/default/ScrollIndicatorStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..8e8826cdac22d10556c24a8299e3ebe858fd378c --- /dev/null +++ b/components/custom/styles/default/ScrollIndicatorStyle.qml @@ -0,0 +1,18 @@ +import QtQuick 1.0 + +QtObject { + property int minimumWidth: 12 + property int minimumHeight: 40 + + property Component content: Component { + Item { + Rectangle { + anchors.fill: parent + anchors.margins: 3 + border.color: "black" + color: "gray" + radius: width/2 + } + } + } +} diff --git a/components/custom/styles/default/SliderStyle.qml b/components/custom/styles/default/SliderStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..d32f13eb796d3f1e3f41655b2019f03e7efb0f9d --- /dev/null +++ b/components/custom/styles/default/SliderStyle.qml @@ -0,0 +1,110 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the Qt Components project on Qt Labs. +** +** No Commercial Usage +** This file contains pre-release code and may not be distributed. +** You may use this file in accordance with the terms and conditions contained +** in the Technology Preview License Agreement accompanying this package. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** If you have questions regarding the use of this file, please contact +** Nokia at qt-info@nokia.com. +** +****************************************************************************/ + +import QtQuick 1.0 + +QtObject { + + property int minimumWidth: 200 + property int minimumHeight: 40 + + property int leftMargin : 2 + property int topMargin: 0 + property int rightMargin: 2 + property int bottomMargin: 0 + + property Component groove: Item { + opacity: enabled ? 1.0 : 0.7 + + Rectangle { + color: backgroundColor + anchors.fill: sliderBackground + anchors.margins: 1 + radius: 2 + } + + Rectangle { + property real zeroPos : positionForValue(0) + property real handlePos: handlePosition + color: progressColor + height: 10 + radius: 4 + anchors.verticalCenter: parent.verticalCenter + x: zeroPos + width: handlePos - x + } + + BorderImage { + id: sliderBackground + anchors.verticalCenter: parent.verticalCenter + width: parent.width + border.top: 2 + border.bottom: 2 + border.left: 12 + border.right: 12 + source: "images/slider.png" + } + } + + property Component handle: Item { + width: handleImage.width + height: handleImage.height + anchors.verticalCenter: parent.verticalCenter + + Image { + id: handleImage + Rectangle { + anchors.centerIn: parent + anchors.horizontalCenterOffset: -1 + width: parent.width - 7 + height: parent.height - 7 + smooth: true + color: backgroundColor + radius: Math.floor(parent.width / 2) + z: -1 // behind the image + } + anchors.centerIn: parent; + source: "images/handle.png" + smooth: true + } + } + + property Component valueIndicator: Rectangle { + width: valueText.width + 20 + height: valueText.height + 20 + color: "gray" + opacity: pressed ? 0.9 : 0 + Behavior on opacity { NumberAnimation { duration: 100 } } + radius: 5 + + Text { + id: valueText + anchors.margins: 10 + anchors.centerIn: parent + text: indicatorText + } + } +} diff --git a/components/custom/styles/default/SpinBoxStyle.qml b/components/custom/styles/default/SpinBoxStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..d42267d8dc6729cc466f2ebb7e99ec4a2feb01cd --- /dev/null +++ b/components/custom/styles/default/SpinBoxStyle.qml @@ -0,0 +1,73 @@ +import QtQuick 1.0 + +QtObject { + + property int minimumWidth: 200 + property int minimumHeight: 25 + + property int leftMargin : 8 + property int topMargin: 8 + property int rightMargin: 8 + property int bottomMargin: 8 + + property Component background: + Component { + id: defaultBackground + Item { + opacity: enabled ? 1 : 0.7 + Rectangle { + x: 1 + y: 1 + width: parent.width-2 + height: parent.height-2 + color: backgroundColor + radius: 5 + } + + BorderImage { + anchors.fill: parent + id: backgroundimage + smooth: true + source: "images/lineedit_normal.png" + border.left: 6; border.top: 6 + border.right: 50; border.bottom: 6 + } + } + } + + property Component up: + Component { + id: defaultUp + Item { + anchors.right: parent.right + anchors.top: parent.top + width: 24 + height: parent.height/2 + Image { + anchors.left: parent.left; + anchors.top: parent.top; + anchors.topMargin: 7 + opacity: (upEnabled && enabled) ? (upPressed ? 1 : 0.8) : 0.3 + source: "images/spinbox_up.png" + } + } + } + + property Component down: + Component { + id: defaultDown + Item { + anchors.right: parent.right + anchors.bottom: parent.bottom + width: 24 + height: parent.height/2 + Image { + anchors.left: parent.left; + anchors.bottom: parent.bottom; + anchors.bottomMargin: 7 + opacity: (downEnabled && enabled) ? (downPressed ? 1 : 0.8) : 0.3 + source: "images/spinbox_down.png" + } + } + } +} diff --git a/components/custom/styles/default/SwitchStyle.qml b/components/custom/styles/default/SwitchStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..35e6f235ed057ba768473d72ec60f44984868d3d --- /dev/null +++ b/components/custom/styles/default/SwitchStyle.qml @@ -0,0 +1,71 @@ +import QtQuick 1.1 + +QtObject { + property int minimumWidth: 80 + property int minimumHeight: 32 + + property Component groove: Component { + Item { + opacity: enabled ? 1 : 0.7 + Rectangle { // Background center fill + anchors.fill: parent + anchors.margins: 1 + radius: 5 + color: backgroundColor + } + + Item { // Clipping container of the positive and negative groove highlight + anchors.fill: parent + anchors.margins: 2 + clip: true + + Item { // The highlight item is twice the width of there switch, clipped by its parent, + // and sliding back and forth keeping the center under the handle + height: parent.height + width: 2*parent.width + x: handleCenterX-parent.width-parent.anchors.leftMargin + + Rectangle { // positive background highlight + color: positiveHighlightColor + opacity: 0.8 + anchors.top: parent.top; anchors.bottom: parent.bottom + anchors.left: parent.left; anchors.right: parent.horizontalCenter + } + Rectangle { // negative background highlight + color: negativeHighlightColor + opacity: 0.8 + anchors.top: parent.top; anchors.bottom: parent.bottom + anchors.left: parent.horizontalCenter; anchors.right: parent.right + } + } + } + + BorderImage { // Rounded border + anchors.fill: parent + source: "images/lineedit_normal.png" + border { left: 6; right: 6; top: 3; bottom: 3 } + smooth: true + } + } + } + + property Component handle: Component { + Item { + width: 42 + Rectangle { // center fill + anchors.fill: parent + anchors.margins: 1 + radius: 5 + color: switchColor + } + BorderImage { + anchors.fill: parent + opacity: enabled ? 1 : 0.7 + smooth: true + source: pressed ? "images/button_pressed.png" : "images/button_normal.png" + border { left: 4; top: 4; right: 4; bottom: 4 } + } + Behavior on x { NumberAnimation { easing.type: Easing.OutCubic; duration: 200 } } + } + } +} diff --git a/components/custom/styles/default/TextFieldStyle.qml b/components/custom/styles/default/TextFieldStyle.qml new file mode 100644 index 0000000000000000000000000000000000000000..38da6b0db6d4c8c950ac6eb7c01401912f9d3893 --- /dev/null +++ b/components/custom/styles/default/TextFieldStyle.qml @@ -0,0 +1,41 @@ +import QtQuick 1.0 + +QtObject { + + property int minimumWidth: 200 + property int minimumHeight: 25 + + property int leftMargin : 8 + property int topMargin: 8 + property int rightMargin: 8 + property int bottomMargin: 8 + + property Component background: Component { + Item { // see QTBUG-14873 + Rectangle { // Background center fill + anchors.fill: parent + anchors.margins: 1 + radius: 5 + color: backgroundColor + } + BorderImage { // Background border + opacity: enabled ? 1 : 0.7 + anchors.fill: parent + border.left: 6; border.top: 6 + border.right: 6; border.bottom: 6 + smooth: true + source: "images/lineedit_normal.png" + } + } + } + + property Component hints: Component { + Item { + property color textColor: "#444" + property color backgroundColor: "white" + property int fontPixelSize: 14 + property bool fontBold: false + property int passwordEchoMode: TextInput.PasswordEchoOnEdit + } + } +} diff --git a/components/custom/styles/default/images/button_normal.png b/components/custom/styles/default/images/button_normal.png new file mode 100644 index 0000000000000000000000000000000000000000..e54f1acba8b1ace3ba8cc456d6f457634d75b085 Binary files /dev/null and b/components/custom/styles/default/images/button_normal.png differ diff --git a/components/custom/styles/default/images/button_pressed.png b/components/custom/styles/default/images/button_pressed.png new file mode 100644 index 0000000000000000000000000000000000000000..b7a63f1ad49c5676266eaa1a2e54e41a3aa02e6c Binary files /dev/null and b/components/custom/styles/default/images/button_pressed.png differ diff --git a/components/custom/styles/default/images/buttongroup_h_normal.png b/components/custom/styles/default/images/buttongroup_h_normal.png new file mode 100644 index 0000000000000000000000000000000000000000..bae593818a783187e7b5ff5fceddf99a0c9bd58a Binary files /dev/null and b/components/custom/styles/default/images/buttongroup_h_normal.png differ diff --git a/components/custom/styles/default/images/buttongroup_h_pressed.png b/components/custom/styles/default/images/buttongroup_h_pressed.png new file mode 100644 index 0000000000000000000000000000000000000000..e0c97fd587a87f1718c7199153cdb0cbab74429e Binary files /dev/null and b/components/custom/styles/default/images/buttongroup_h_pressed.png differ diff --git a/components/custom/styles/default/images/checkbox_check.png b/components/custom/styles/default/images/checkbox_check.png new file mode 100644 index 0000000000000000000000000000000000000000..a2f1f02d2d91b719209e71dd0d26adda9182962b Binary files /dev/null and b/components/custom/styles/default/images/checkbox_check.png differ diff --git a/components/custom/styles/default/images/handle.png b/components/custom/styles/default/images/handle.png new file mode 100644 index 0000000000000000000000000000000000000000..bd8fbddfbde6a43773e32fa34ab4f02c7e88ca16 Binary files /dev/null and b/components/custom/styles/default/images/handle.png differ diff --git a/components/custom/styles/default/images/lineedit_normal.png b/components/custom/styles/default/images/lineedit_normal.png new file mode 100644 index 0000000000000000000000000000000000000000..2253d33f92e4a11742df1633473bef7695c5b6e6 Binary files /dev/null and b/components/custom/styles/default/images/lineedit_normal.png differ diff --git a/components/custom/styles/default/images/progress-bar-background.png b/components/custom/styles/default/images/progress-bar-background.png new file mode 100644 index 0000000000000000000000000000000000000000..4f30927ed8c34581a96edf764c98e4a687352bbc Binary files /dev/null and b/components/custom/styles/default/images/progress-bar-background.png differ diff --git a/components/custom/styles/default/images/progress-bar-bar.png b/components/custom/styles/default/images/progress-bar-bar.png new file mode 100644 index 0000000000000000000000000000000000000000..446c13f2c20c5b8ea4e8e3cfbbd689847d9ad5e4 Binary files /dev/null and b/components/custom/styles/default/images/progress-bar-bar.png differ diff --git a/components/custom/styles/default/images/progressbar_fill.png b/components/custom/styles/default/images/progressbar_fill.png new file mode 100644 index 0000000000000000000000000000000000000000..0f8778969a849127acc0d8e83c4ebd6aba53b903 Binary files /dev/null and b/components/custom/styles/default/images/progressbar_fill.png differ diff --git a/components/custom/styles/default/images/progressbar_groove.png b/components/custom/styles/default/images/progressbar_groove.png new file mode 100644 index 0000000000000000000000000000000000000000..263ced8bf3a8430f0b725678a78248fc2a23a72a Binary files /dev/null and b/components/custom/styles/default/images/progressbar_groove.png differ diff --git a/components/custom/styles/default/images/progressbar_indeterminate.png b/components/custom/styles/default/images/progressbar_indeterminate.png new file mode 100644 index 0000000000000000000000000000000000000000..7ea46460b59830404ca61024da78ac8568a09b49 Binary files /dev/null and b/components/custom/styles/default/images/progressbar_indeterminate.png differ diff --git a/components/custom/styles/default/images/progressbar_overlay.png b/components/custom/styles/default/images/progressbar_overlay.png new file mode 100644 index 0000000000000000000000000000000000000000..8e351f965906de0e757a2848440af05727debb98 Binary files /dev/null and b/components/custom/styles/default/images/progressbar_overlay.png differ diff --git a/components/custom/styles/default/images/qt-logo.png b/components/custom/styles/default/images/qt-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..005ee5ca3204e0cbd3f9157477421e8d8e26a35e Binary files /dev/null and b/components/custom/styles/default/images/qt-logo.png differ diff --git a/components/custom/styles/default/images/radiobutton_check.png b/components/custom/styles/default/images/radiobutton_check.png new file mode 100644 index 0000000000000000000000000000000000000000..3d433860694359f394b7a50ba847bf93dbed6606 Binary files /dev/null and b/components/custom/styles/default/images/radiobutton_check.png differ diff --git a/components/custom/styles/default/images/radiobutton_check_white.png b/components/custom/styles/default/images/radiobutton_check_white.png new file mode 100644 index 0000000000000000000000000000000000000000..b68e7a6987c99020644efac3cde7f54caaee6382 Binary files /dev/null and b/components/custom/styles/default/images/radiobutton_check_white.png differ diff --git a/components/custom/styles/default/images/radiobutton_normal.png b/components/custom/styles/default/images/radiobutton_normal.png new file mode 100644 index 0000000000000000000000000000000000000000..6ceab7ee66870c38426577bfbf9eeffe58b9474e Binary files /dev/null and b/components/custom/styles/default/images/radiobutton_normal.png differ diff --git a/components/custom/styles/default/images/slider-background.png b/components/custom/styles/default/images/slider-background.png new file mode 100644 index 0000000000000000000000000000000000000000..3ba9c4731f7bd4f0aedd487592607d1185ba90b2 Binary files /dev/null and b/components/custom/styles/default/images/slider-background.png differ diff --git a/components/custom/styles/default/images/slider-handle-active.png b/components/custom/styles/default/images/slider-handle-active.png new file mode 100644 index 0000000000000000000000000000000000000000..83cec3f4b3d4f8674326dd981a2df2252f4f2e6c Binary files /dev/null and b/components/custom/styles/default/images/slider-handle-active.png differ diff --git a/components/custom/styles/default/images/slider-handle.png b/components/custom/styles/default/images/slider-handle.png new file mode 100644 index 0000000000000000000000000000000000000000..765e610ac3adae3ef202e786aec4d18d5246193b Binary files /dev/null and b/components/custom/styles/default/images/slider-handle.png differ diff --git a/components/custom/styles/default/images/slider.png b/components/custom/styles/default/images/slider.png new file mode 100644 index 0000000000000000000000000000000000000000..3f310e37cecab290854b8552bad74b71ef0f9336 Binary files /dev/null and b/components/custom/styles/default/images/slider.png differ diff --git a/components/custom/styles/default/images/spinbox_down.png b/components/custom/styles/default/images/spinbox_down.png new file mode 100644 index 0000000000000000000000000000000000000000..a6f3f79e969c11f016fd52b2f2db681b44c7b8a0 Binary files /dev/null and b/components/custom/styles/default/images/spinbox_down.png differ diff --git a/components/custom/styles/default/images/spinbox_up.png b/components/custom/styles/default/images/spinbox_up.png new file mode 100644 index 0000000000000000000000000000000000000000..a484194125fa292abfbcc444358c00906990da8b Binary files /dev/null and b/components/custom/styles/default/images/spinbox_up.png differ diff --git a/components/custom/styles/default/images/spinner.png b/components/custom/styles/default/images/spinner.png new file mode 100644 index 0000000000000000000000000000000000000000..664c2b1491498ee0158cb7674602265cc5f5c70c Binary files /dev/null and b/components/custom/styles/default/images/spinner.png differ diff --git a/components/custom/styles/default/images/switch_normal.png b/components/custom/styles/default/images/switch_normal.png new file mode 100644 index 0000000000000000000000000000000000000000..ac46b2112d5cb7a7bf8a476bff4f14c2094f253d Binary files /dev/null and b/components/custom/styles/default/images/switch_normal.png differ diff --git a/components/custom/styles/default/images/switch_pressed.png b/components/custom/styles/default/images/switch_pressed.png new file mode 100644 index 0000000000000000000000000000000000000000..a04a3f4b0f52626f1145359b45cb6c3f850ee9f6 Binary files /dev/null and b/components/custom/styles/default/images/switch_pressed.png differ diff --git a/components/custom/styles/default/tools/ColorConverter.qml b/components/custom/styles/default/tools/ColorConverter.qml new file mode 100644 index 0000000000000000000000000000000000000000..5070ff1eb08195da32b5221adbdcaa1f31c83a2f --- /dev/null +++ b/components/custom/styles/default/tools/ColorConverter.qml @@ -0,0 +1,59 @@ +import QtQuick 1.0 + +// This helper element allows extracting color values +QtObject { + property color color; + + function intValue(dec) + { + var result; + switch (dec) { + case 'a': + result = 10; + break; + case 'b': + result = 11 + break; + case 'c': + result = 12; + break; + case 'd': + result = 13; + break; + case 'e': + result = 14 + case 'f': + result = 15 + break; + default: + result = dec + break; + } + return Number(result); + } + + function convertSubstring(val) { + return 16*intValue(val[0]) + intValue(val[1]) + } + + function grayValue() { + return (red + green + blue)/3 + } + + onColorChanged: { + var string = "" + color + var redString = string.substring(1, 3) + var greenString = string.substring(3,5) + var blueString = string.substring(5,7) + var alphaString = string.substring(7,9) + red = convertSubstring(string.substring(1, 3)) + green = convertSubstring(string.substring(3,5)) + blue = convertSubstring(string.substring(5,7)) + alpha = convertSubstring(string.substring(7,9)) + } + + property int red + property int green + property int blue + property int alpha +} diff --git a/components/custom/visuals/AdjoiningCorner.qml b/components/custom/visuals/AdjoiningCorner.qml new file mode 100644 index 0000000000000000000000000000000000000000..92027cef494c09e1f5fb014a66e52b975bad4f7a --- /dev/null +++ b/components/custom/visuals/AdjoiningCorner.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 + +Item { + property string corner: "" // Can we use Qt::Corner? see http://doc.trolltech.com/4.7/qt.html#Corner-enum + property alias styledItem: loader.styledItem + property alias adjoining: loader.adjoining + + clip: true + + anchors { + left: corner == "TopLeftCorner" || corner == "BottomLeftCorner" ? parent.left : parent.horizontalCenter + right: corner == "TopLeftCorner" || corner == "BottomLeftCorner" ? parent.horizontalCenter : parent.right + top: corner == "TopLeftCorner" || corner == "TopRightCorner" ? parent.top : parent.verticalCenter + bottom: corner == "TopLeftCorner" || corner == "TopRightCorner" ? parent.verticalCenter : parent.bottom + } + + Item { + width: parent.width*2 + height: parent.height*2 + x: corner == "TopLeftCorner" || corner == "BottomLeftCorner" ? 0 : -parent.width + y: corner == "TopLeftCorner" || corner == "TopRightCorner" ? 0 : -parent.height + + Loader { + id: loader + anchors.fill: parent + property Item styledItem:loader.styledItem + property int adjoining: 0x0 + sourceComponent: styling + } + } +} diff --git a/components/custom/visuals/AdjoiningVisual.qml b/components/custom/visuals/AdjoiningVisual.qml new file mode 100644 index 0000000000000000000000000000000000000000..09bc34cc9461db30ff1487198845f0b600495efb --- /dev/null +++ b/components/custom/visuals/AdjoiningVisual.qml @@ -0,0 +1,50 @@ +import QtQuick 1.0 + +Item { + id: adjoiningVisual + property int adjoins: 0 // use enum Qt::DockWidgetArea? see http://doc.trolltech.com/4.7/qt.html#DockWidgetArea-enum + property Item styledItem + property Component styling + +// Qt::LeftDockWidgetArea 0x1 +// Qt::RightDockWidgetArea 0x2 +// Qt::TopDockWidgetArea 0x4 +// Qt::BottomDockWidgetArea 0x8 + + Item { + anchors.fill: parent + + AdjoiningCorner { corner: "TopLeftCorner"; adjoining: topLeftAdjoining(); styledItem: adjoiningVisual.styledItem } + AdjoiningCorner { corner: "TopRightCorner"; adjoining: topRightAdjoining(); styledItem: adjoiningVisual.styledItem } + AdjoiningCorner { corner: "BottomLeftCorner"; adjoining: bottomLeftAdjoining(); styledItem: adjoiningVisual.styledItem } + AdjoiningCorner { corner: "BottomRightCorner"; adjoining: bottomRightAdjoining(); styledItem: adjoiningVisual.styledItem } + } + + function topLeftAdjoining() { + var adjoining = 0; + if(adjoins&0x01) adjoining |= Qt.Horizontal; + if(adjoins&0x04) adjoining |= Qt.Vertical; + return adjoining; + } + + function topRightAdjoining() { + var adjoining = 0; + if(adjoins&0x02) adjoining |= Qt.Horizontal; + if(adjoins&0x04) adjoining |= Qt.Vertical; + return adjoining; + } + + function bottomLeftAdjoining() { + var adjoining = 0; + if(adjoins&0x01) adjoining |= Qt.Horizontal; + if(adjoins&0x08) adjoining |= Qt.Vertical; + return adjoining; + } + + function bottomRightAdjoining() { + var adjoining = 0; + if(adjoins&0x02) adjoining |= Qt.Horizontal; + if(adjoins&0x08) adjoining |= Qt.Vertical; + return adjoining; + } +} diff --git a/components/images/folder_new.png b/components/images/folder_new.png new file mode 100644 index 0000000000000000000000000000000000000000..8d8bb9bd768d8e8ab785d95483ead02ae6800dc5 Binary files /dev/null and b/components/images/folder_new.png differ diff --git a/components/plugin/qmldir b/components/plugin/qmldir new file mode 100644 index 0000000000000000000000000000000000000000..e8452efd6f37818e71e256646eb1d06a60dccf0f --- /dev/null +++ b/components/plugin/qmldir @@ -0,0 +1 @@ +plugin styleplugin diff --git a/components/src.pro b/components/src.pro new file mode 100644 index 0000000000000000000000000000000000000000..83e0a3d4c0b4f86c82ef65d0f88bcf168f12c7c0 --- /dev/null +++ b/components/src.pro @@ -0,0 +1,2 @@ +TEMPLATE = subdirs +SUBDIRS = styleitem diff --git a/components/styleitem/qstyleitem.cpp b/components/styleitem/qstyleitem.cpp new file mode 100644 index 0000000000000000000000000000000000000000..90fa5a36778f46fbc7d80ba18a4517423af168a4 --- /dev/null +++ b/components/styleitem/qstyleitem.cpp @@ -0,0 +1,547 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOTgall +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include "qstyleitem.h" + +#include <QtGui/QPainter> +#include <QtGui/QStyle> +#include <QtGui/QStyleOption> +#include <QtGui/QApplication> +#include <QtGui/QMainWindow> +#include <QtGui/QGroupBox> +#include <QtGui/QToolBar> +#include <QtGui/QMenu> + + +QStyleItem::QStyleItem(QObject*parent) + : QObject(parent), + m_sunken(false), + m_raised(false), + m_active(true), + m_enabled(true), + m_selected(false), + m_focus(false), + m_on(false), + m_horizontal(true), + m_minimum(0), + m_maximum(100), + m_value(0) +{ +} + +void QStyleItem::initStyleOption(QStyleOption *opt) const +{ + if (m_enabled) + opt->state |= QStyle::State_Enabled; + if (m_active) + opt->state |= QStyle::State_Active; + if (m_sunken) + opt->state |= QStyle::State_Sunken; + if (m_raised) + opt->state |= QStyle::State_Raised; + if (m_selected) + opt->state |= QStyle::State_Selected; + if (m_focus) + opt->state |= QStyle::State_HasFocus; + if (m_on) + opt->state |= QStyle::State_On; + if (m_hover) + opt->state |= QStyle::State_MouseOver; + if (m_horizontal) + opt->state |= QStyle::State_Horizontal; +} + +QString QStyleBackground::hitTest(int px, int py) const +{ + QStyle::SubControl subcontrol = QStyle::SC_All; + QStyle::ComplexControl control = QStyle::CC_CustomBase; + QString type = m_style->elementType(); + if (type == QLatin1String("spinbox")) { + control = QStyle::CC_SpinBox; + QStyleOptionSpinBox opt; + opt.rect = QRect(0, 0, width(), height()); + opt.frame = true; + m_style->initStyleOption(&opt); + subcontrol = qApp->style()->hitTestComplexControl(control, &opt, QPoint(px,py), 0); + if (subcontrol == QStyle::SC_SpinBoxUp) + return "up"; + else if (subcontrol == QStyle::SC_SpinBoxDown) + return "down"; + + } else if (type == QLatin1String("slider")) { + control = QStyle::CC_Slider; + QStyleOptionSlider opt; + opt.rect = QRect(0, 0, width(), height()); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.sliderPosition = m_style->value(); + m_style->initStyleOption(&opt); + subcontrol = qApp->style()->hitTestComplexControl(control, &opt, QPoint(px,py), 0); + if (subcontrol == QStyle::SC_SliderHandle) + return "handle"; + } else if (type == QLatin1String("scrollbar")) { + control = QStyle::CC_ScrollBar; + QStyleOptionSlider opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.pageStep = 200; + opt.orientation = m_style->horizontal() ? Qt::Horizontal : Qt::Vertical; + opt.sliderPosition = m_style->value(); + subcontrol = qApp->style()->hitTestComplexControl(control, &opt, QPoint(px,py), 0); + + if (subcontrol == QStyle::SC_ScrollBarSlider) + return "handle"; + if (subcontrol == QStyle::SC_ScrollBarSubLine || subcontrol == QStyle::SC_ScrollBarSubPage) + return "up"; + if (subcontrol == QStyle::SC_ScrollBarAddLine || subcontrol == QStyle::SC_ScrollBarAddPage) + return "down"; + } + return "none"; +} + +QSize QStyleItem::sizeFromContents(int width, int height) const +{ + QString metric = m_type; + if (metric == QLatin1String("checkbox")) { + QStyleOptionButton opt; + initStyleOption(&opt); + opt.text = text(); + return qApp->style()->sizeFromContents(QStyle::CT_CheckBox, &opt, QSize(width,height), &m_dummywidget); + } else if (metric == QLatin1String("button")) { + QStyleOptionButton opt; + initStyleOption(&opt); + opt.text = text(); + return qApp->style()->sizeFromContents(QStyle::CT_PushButton, &opt, QSize(width,height), &m_dummywidget); + } else if (metric == QLatin1String("tab")) { + QStyleOptionTabV3 opt; + initStyleOption(&opt); + opt.text = text(); + return qApp->style()->sizeFromContents(QStyle::CT_TabBarTab, &opt, QSize(width,height), &m_dummywidget); + } else if (metric == QLatin1String("combobox")) { + QStyleOptionComboBox opt; + initStyleOption(&opt); + return qApp->style()->sizeFromContents(QStyle::CT_ComboBox, &opt, QSize(width,height), &m_dummywidget); + } else if (metric == QLatin1String("spinbox")) { + + QStyleOptionSpinBox opt; + initStyleOption(&opt); + return qApp->style()->sizeFromContents(QStyle::CT_SpinBox, &opt, QSize(width,height), &m_dummywidget); + } else if (metric == QLatin1String("edit")) { + QStyleOptionFrameV3 opt; + initStyleOption(&opt); + return qApp->style()->sizeFromContents(QStyle::CT_LineEdit, &opt, QSize(width,height), &m_dummywidget); + } + return QSize(); +} + +int QStyleItem::pixelMetric(const QString &metric) const +{ + if (metric == "scrollbarExtent") + return qApp->style()->pixelMetric(QStyle::PM_ScrollBarExtent); + else if (metric == "defaultframewidth") + return qApp->style()->pixelMetric(QStyle::PM_DefaultFrameWidth); + else if (metric == "taboverlap") + return qApp->style()->pixelMetric(QStyle::PM_TabBarTabOverlap); + else if (metric == "tabbaseoverlap") + return qApp->style()->pixelMetric(QStyle::PM_TabBarBaseOverlap); + else if (metric == "tabbaseheight") + return qApp->style()->pixelMetric(QStyle::PM_TabBarBaseHeight); + else if (metric == "tabvshift") + return qApp->style()->pixelMetric(QStyle::PM_TabBarTabShiftVertical); + else if (metric == "menuhmargin") + return qApp->style()->pixelMetric(QStyle::PM_MenuHMargin); + else if (metric == "menuvmargin") + return qApp->style()->pixelMetric(QStyle::PM_MenuVMargin); + else if (metric == "menupanelwidth") + return qApp->style()->pixelMetric(QStyle::PM_MenuPanelWidth); + return 0; +} + +QVariant QStyleItem::styleHint(const QString &metric) const +{ + if (metric == "focuswidget") + return qApp->style()->styleHint(QStyle::SH_FocusFrame_AboveWidget); + if (metric == "tabbaralignment") { + int result = qApp->style()->styleHint(QStyle::SH_TabBar_Alignment); + if (result == Qt::AlignCenter) + return "center"; + return "left"; + } + if (metric == "framearoundcontents") + return qApp->style()->styleHint(QStyle::SH_ScrollView_FrameOnlyAroundContents); + return 0; +} + + +QRect QStyleBackground::subControlRect(const QString &subcontrolString) const +{ + QStyle::SubControl subcontrol = QStyle::SC_None; + QString m_type = m_style->elementType(); + if (m_type == QLatin1String("spinbox")) { + QStyle::ComplexControl control = QStyle::CC_SpinBox; + QStyleOptionSpinBox opt; + m_style->initStyleOption(&opt); + opt.rect = QRect(0, 0, width(), height()); + opt.frame = true; + if (subcontrolString == QLatin1String("down")) + subcontrol = QStyle::SC_SpinBoxDown; + else if (subcontrolString == QLatin1String("up")) + subcontrol = QStyle::SC_SpinBoxUp; + else if (subcontrolString == QLatin1String("edit")){ + subcontrol = QStyle::SC_SpinBoxEditField; + } + return qApp->style()->subControlRect(control, &opt, subcontrol, 0); + } else if (m_type == QLatin1String("slider")) { + QStyle::ComplexControl control = QStyle::CC_Slider; + QStyleOptionSlider opt; + m_style->initStyleOption(&opt); + opt.rect = QRect(0, 0, width(), height()); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.sliderPosition = m_style->value(); + if (subcontrolString == QLatin1String("handle")) + subcontrol = QStyle::SC_SliderHandle; + else if (subcontrolString == QLatin1String("groove")) + subcontrol = QStyle::SC_SliderGroove; + return qApp->style()->subControlRect(control, &opt, subcontrol, 0); + } else if (m_type == QLatin1String("scrollbar")) { + QStyle::ComplexControl control = QStyle::CC_ScrollBar; + QStyleOptionSlider opt; + m_style->initStyleOption(&opt); + opt.rect = QRect(0, 0, width(), height()); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.pageStep = m_style->horizontal() ? width() : height(); + opt.orientation = m_style->horizontal() ? Qt::Horizontal : Qt::Vertical; + opt.sliderPosition = m_style->value(); + if (subcontrolString == QLatin1String("slider")) + subcontrol = QStyle::SC_ScrollBarSlider; + if (subcontrolString == QLatin1String("groove")) + subcontrol = QStyle::SC_ScrollBarGroove; + else if (subcontrolString == QLatin1String("handle")) + subcontrol = QStyle::SC_ScrollBarSlider; + else if (subcontrolString == QLatin1String("add")) + subcontrol = QStyle::SC_ScrollBarAddPage; + else if (subcontrolString == QLatin1String("sub")) + subcontrol = QStyle::SC_ScrollBarSubPage; + return qApp->style()->subControlRect(control, &opt, subcontrol, 0); + } + return QRect(); +} + +QStyleBackground::QStyleBackground(QDeclarativeItem *parent) + : QDeclarativeItem(parent), + m_menu(0), + m_style(0) +{ + setFlag(QGraphicsItem::ItemHasNoContents, false); + setCacheMode(QGraphicsItem::DeviceCoordinateCache); + setSmooth(true); + + m_menu = new QMenu(); + m_menu->ensurePolished(); +} + +void QStyleBackground::setStyle(QStyleItem *style) +{ + if (m_style != style) { + m_style = style; + //connect(&m_dummywidget, SIGNAL(updateRequest()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(onChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(selectedChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(activeChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(textChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(activeChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(raisedChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(sunkenChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(hoverChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(maximumChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(minimumChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(valueChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(enabledChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(horizontalChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(activeControlChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(focusChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(activeControlChanged()), this, SLOT(updateItem())); + connect(m_style, SIGNAL(elementTypeChanged()), this, SLOT(updateItem())); + emit styleChanged(); + } +} + + +void QStyleBackground::paint(QPainter *painter, const QStyleOptionGraphicsItem *, QWidget *) +{ + QString type = m_style->elementType(); + if (type == QLatin1String("button")) { + QStyle::ControlElement control = QStyle::CE_PushButton; + QStyleOptionButton opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + + // Dirty hack to fix button label positioning on mac + if (qApp->style()->metaObject()->className() == QLatin1String("QMacStyle")) + opt.rect.translate(0,2); + + qApp->style()->drawControl(control, &opt, painter, &m_dummywidget); + } + else if (type == QLatin1String("toolbutton")) { + QStyle::ComplexControl control = QStyle::CC_ToolButton; + QStyleOptionToolButton opt; + m_style->initStyleOption(&opt); + opt.subControls = QStyle::SC_ToolButton; + opt.rect = QRect(0, 0, width(), height()); + QToolBar bar; + QWidget dummy(&bar); + if (opt.state & QStyle::State_Raised || opt.state & QStyle::State_On) + qApp->style()->drawComplexControl(control, &opt, painter, &dummy); + } + else if (type == QLatin1String("tab")) { + QStyle::ControlElement control = QStyle::CE_TabBarTabShape; + QStyleOptionTabV3 opt; + m_style->initStyleOption(&opt); + int overlap = qApp->style()->pixelMetric(QStyle::PM_TabBarTabOverlap); + opt.rect = QRect(overlap, 0, width()-2*overlap, height()); + if (m_style->text() == "South") + opt.shape = QTabBar::RoundedSouth; + if (m_style->activeControl() == QLatin1String("beginning")) + opt.position = QStyleOptionTabV3::Beginning; + else if (m_style->activeControl() == QLatin1String("end")) + opt.position = QStyleOptionTabV3::End; + else if (m_style->activeControl() == QLatin1String("only")) + opt.position = QStyleOptionTabV3::OnlyOneTab; + else + opt.position = QStyleOptionTabV3::Middle; + qApp->style()->drawControl(control, &opt, painter, &m_dummywidget); + } + else if (type == QLatin1String("menu")) { + QStyleOptionMenuItem opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + QStyleHintReturnMask val; + qApp->style()->styleHint(QStyle::SH_Menu_Mask, &opt, &m_dummywidget, &val); + painter->save(); + painter->setClipRegion(val.region); + m_menu->setContextMenuPolicy(Qt::CustomContextMenu); + m_menu->ensurePolished(); + opt.palette = m_menu->palette(); + painter->fillRect(opt.rect, opt.palette.window()); + painter->restore(); + qApp->style()->drawPrimitive(QStyle::PE_PanelMenu, &opt, painter, m_menu); + QStyleOptionFrame frame; + m_style->initStyleOption(&frame); + frame.lineWidth = qApp->style()->pixelMetric(QStyle::PM_MenuPanelWidth); + frame.midLineWidth = 0; + frame.rect = opt.rect; + qApp->style()->drawPrimitive(QStyle::PE_FrameMenu, &frame, painter, m_menu); + // qApp->style()->drawControl(QStyle::CE_MenuVMargin, &opt, painter, m_menu); + } + else if (type == QLatin1String("frame")) { + QStyle::PrimitiveElement control = QStyle::PE_Frame; + QStyleOptionFrameV3 opt; + opt.rect = QRect(0, 0, width(), height()); + opt.frameShape = QFrame::StyledPanel; + opt.lineWidth = 1; + m_style->initStyleOption(&opt); + qApp->style()->drawPrimitive(control, &opt, painter, 0); + } + else if (type == QLatin1String("focusframe")) { + QStyle::ControlElement control = QStyle::CE_FocusFrame; + QStyleOption opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + qApp->style()->drawControl(control, &opt, painter, &m_dummywidget); + } + else if (type == QLatin1String("tabframe")) { + QStyle::PrimitiveElement control = QStyle::PE_FrameTabWidget; + if (m_style->minimum()) { + QStyleOptionTabWidgetFrameV2 opt; + m_style->initStyleOption(&opt); + if (m_style->text() == "South") + opt.shape = QTabBar::RoundedSouth; + opt.selectedTabRect = QRect(m_style->value(), 0, m_style->minimum(), height()); + opt.rect = QRect(0, 0, width(), height()); + qApp->style()->drawPrimitive(control, &opt, painter, 0); + } else { + QStyleOptionTabWidgetFrame opt; + m_style->initStyleOption(&opt); + opt.rect = QRect(0, 0, width(), height()); + qApp->style()->drawPrimitive(control, &opt, painter, 0); + } + } + else if (type == QLatin1String("menuitem")) { + QStyle::ControlElement control = QStyle::CE_MenuItem; + QStyleOptionMenuItem opt; + opt.rect = QRect(0, 0, width(), height()); + opt.text = m_style->text(); + m_style->initStyleOption(&opt); + opt.palette = m_menu->palette(); + qApp->style()->drawControl(control, &opt, painter, m_menu); + } + else if (type == QLatin1String("checkbox")) { + QStyle::ControlElement control = QStyle::CE_CheckBox; + QStyleOptionButton opt; + m_style->initStyleOption(&opt); + if (!(opt.state & QStyle::State_On)) + opt.state |= QStyle::State_Off; + opt.rect = QRect(0, 0, width(), height()); + opt.text = m_style->text(); + qApp->style()->drawControl(control, &opt, painter, 0); + } + else if (type == QLatin1String("radiobutton")) { + QStyle::ControlElement control = QStyle::CE_RadioButton; + QStyleOptionButton opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + opt.text = m_style->text(); + qApp->style()->drawControl(control, &opt, painter, 0); + } + else if (type == QLatin1String("edit")) { + QStyle::PrimitiveElement control = QStyle::PE_PanelLineEdit; + QStyleOptionFrameV3 opt; + opt.rect = QRect(0, 0, width(), height()); + opt.lineWidth = 1; // jens : this must be non-zero + m_style->initStyleOption(&opt); + qApp->style()->drawPrimitive(control, &opt, painter, 0); + } + else if (type == QLatin1String("slidergroove")) { + QStyle::ComplexControl control = QStyle::CC_Slider; + QStyleOptionSlider opt; + opt.rect = QRect(0, 0, width(), height()); + opt.minimum = 0; + opt.maximum = 100; + opt.subControls |= (QStyle::SC_SliderGroove); + opt.activeSubControls = QStyle::SC_SliderHandle; + m_style->initStyleOption(&opt); + qApp->style()->drawComplexControl(control, &opt, painter, 0); + } + else if (type == QLatin1String("combobox")) { + QStyle::ComplexControl control = QStyle::CC_ComboBox; + QStyleOptionComboBox opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + m_dummywidget.activateWindow(); + qApp->style()->drawComplexControl(control, &opt, painter, &m_dummywidget); + } + else if (type == QLatin1String("spinbox")) { + QStyle::ComplexControl control = QStyle::CC_SpinBox; + QStyleOptionSpinBox opt; + opt.rect = QRect(0, 0, width(), height()); + opt.frame = true; + m_style->initStyleOption(&opt); + if (m_style->value() & 0x1) + opt.activeSubControls = QStyle::SC_SpinBoxUp; + else if (m_style->value() & (1<<1)) + opt.activeSubControls = QStyle::SC_SpinBoxDown; + opt.subControls |= QStyle::SC_SpinBoxDown; + opt.subControls |= QStyle::SC_SpinBoxUp; + if (m_style->value() & (1<<2)) + opt.stepEnabled |= QAbstractSpinBox::StepUpEnabled; + if (m_style->value() & (1<<3)) + opt.stepEnabled |= QAbstractSpinBox::StepDownEnabled; + qApp->style()->drawComplexControl(control, &opt, painter, 0); + } + else if (type == QLatin1String("slider")) { + QStyle::ComplexControl control = QStyle::CC_Slider; + QStyleOptionSlider opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.tickPosition = QSlider::TicksBelow; + opt.sliderPosition = m_style->value(); + opt.tickInterval = 1200 / (opt.maximum - opt.minimum); + opt.sliderValue = m_style->value(); + opt.subControls = QStyle::SC_SliderTickmarks | QStyle::SC_SliderGroove | QStyle::SC_SliderHandle; + opt.activeSubControls = QStyle::SC_None; + qApp->style()->drawComplexControl(control, &opt, painter, 0); + } + else if (type == QLatin1String("progressbar")) { + QStyle::ControlElement control = QStyle::CE_ProgressBar; + QStyleOptionProgressBarV2 opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.progress = m_style->value(); + qApp->style()->drawControl(control, &opt, painter, &m_dummywidget); + } + else if (type == QLatin1String("toolbar")) { + QStyle::ControlElement control = QStyle::CE_ToolBar; + QStyleOptionToolBar opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + QMainWindow mw; + QWidget w(&mw); + qApp->style()->drawControl(control, &opt, painter, &w); + } + else if (type == QLatin1String("groupbox")) { + QStyle::ComplexControl control = QStyle::CC_GroupBox; + QStyleOptionGroupBox opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + opt.text = m_style->text(); + opt.lineWidth = 1; + opt.subControls = QStyle::SC_GroupBoxLabel; + // oxygen crashes if we dont pass a widget + qApp->style()->drawComplexControl(control, &opt, painter, &m_dummywidget); + } + else if (type == QLatin1String("scrollbar")) { + QStyle::ComplexControl control = QStyle::CC_ScrollBar; + QStyleOptionSlider opt; + opt.rect = QRect(0, 0, width(), height()); + m_style->initStyleOption(&opt); + opt.minimum = m_style->minimum(); + opt.maximum = m_style->maximum(); + opt.pageStep = m_style->horizontal() ? width() : height(); + opt.orientation = m_style->horizontal() ? Qt::Horizontal : Qt::Vertical; + opt.sliderPosition = m_style->value(); + opt.sliderValue = m_style->value(); + opt.activeSubControls = (m_style->activeControl() == QLatin1String("up")) + ? QStyle::SC_ScrollBarSubLine : + (m_style->activeControl() == QLatin1String("down")) ? + QStyle::SC_ScrollBarAddLine: + QStyle::SC_ScrollBarSlider; + + opt.sliderValue = m_style->value(); + opt.subControls = QStyle::SC_All; + qApp->style()->drawComplexControl(control, &opt, painter, &m_dummywidget); + } +} diff --git a/components/styleitem/qstyleitem.h b/components/styleitem/qstyleitem.h new file mode 100644 index 0000000000000000000000000000000000000000..5d195252a3cc4ad8488ee256b0a7313381aa916b --- /dev/null +++ b/components/styleitem/qstyleitem.h @@ -0,0 +1,197 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef STYLEWRAPPER_H +#define STYLEWRAPPER_H + +#include <QDeclarativeItem> +#include <QtGui/QStyle> +#include <QtGui> +#include <QEvent> + +/** + * This class adds experimental support for + * animated progressbars + */ +class AnimWidget: public QProgressBar +{ +Q_OBJECT +public: + AnimWidget(QWidget *parent = 0): + QProgressBar(parent) { + setMaximum(100); + setMinimum(0); + setValue(50); + // setAttribute(Qt::WA_WState_InPaintEvent); + // setAttribute(Qt::WA_WState_Visible, true); + } +public: + bool event(QEvent *e){ + // emit updateRequest(); + return QProgressBar::event(e); + } +signals: + void updateRequest(); +}; + +class QStyleItem: public QObject +{ + Q_OBJECT + + Q_PROPERTY( bool sunken READ sunken WRITE setSunken NOTIFY sunkenChanged) + Q_PROPERTY( bool raised READ raised WRITE setRaised NOTIFY raisedChanged) + Q_PROPERTY( bool active READ active WRITE setActive NOTIFY activeChanged) + Q_PROPERTY( bool enabled READ enabled WRITE setEnabled NOTIFY enabledChanged) + Q_PROPERTY( bool selected READ selected WRITE setSelected NOTIFY selectedChanged) + Q_PROPERTY( bool focus READ focus WRITE setFocus NOTIFY focusChanged) + Q_PROPERTY( bool on READ on WRITE setOn NOTIFY onChanged) + Q_PROPERTY( bool hover READ hover WRITE setHover NOTIFY hoverChanged) + Q_PROPERTY( bool horizontal READ horizontal WRITE setHorizontal NOTIFY horizontalChanged) + Q_PROPERTY( QString elementType READ elementType WRITE setElementType NOTIFY elementTypeChanged) + Q_PROPERTY( QString text READ text WRITE setText NOTIFY textChanged) + Q_PROPERTY( QString activeControl READ activeControl WRITE setActiveControl NOTIFY activeControlChanged) + + // For range controls + Q_PROPERTY( int minimum READ minimum WRITE setMinimum NOTIFY minimumChanged) + Q_PROPERTY( int maximum READ maximum WRITE setMaximum NOTIFY maximumChanged) + Q_PROPERTY( int value READ value WRITE setValue NOTIFY valueChanged) + +public: + QStyleItem(QObject *parent = 0); + + bool sunken() const { return m_sunken; } + bool raised() const { return m_raised; } + bool active() const { return m_active; } + bool enabled() const { return m_enabled; } + bool selected() const { return m_selected; } + bool focus() const { return m_focus; } + bool on() const { return m_on; } + bool hover() const { return m_hover; } + bool horizontal() const { return m_horizontal; } + int minimum() const { return m_minimum; } + int maximum() const { return m_maximum; } + int value() const { return m_value; } + QString elementType() const { return m_type; } + QString text() const { return m_text; } + QString activeControl() const { return m_activeControl; } + + void setSunken(bool sunken) { if (m_sunken != sunken) {m_sunken = sunken; emit sunkenChanged();}} + void setRaised(bool raised) { if (m_raised!= raised) {m_raised = raised; emit raisedChanged();}} + void setActive(bool active) { if (m_active!= active) {m_active = active; emit activeChanged();}} + void setEnabled(bool enabled) { if (m_enabled!= enabled) {m_enabled = enabled; emit enabledChanged();}} + void setSelected(bool selected) { if (m_selected!= selected) {m_selected = selected; emit selectedChanged();}} + void setFocus(bool focus) { if (m_focus != focus) {m_focus = focus; emit focusChanged();}} + void setOn(bool on) { if (m_on != on) {m_on = on ; emit onChanged();}} + void setHover(bool hover) { if (m_hover != hover) {m_hover = hover ; emit hoverChanged();}} + void setHorizontal(bool horizontal) { if (m_horizontal != horizontal) {m_horizontal = horizontal; emit horizontalChanged();}} + void setMinimum(int minimum) { if (m_minimum!= minimum) {m_minimum = minimum; emit minimumChanged();}} + void setMaximum(int maximum) { if (m_maximum != maximum) {m_maximum = maximum; emit maximumChanged();}} + void setValue(int value) { if (m_value!= value) {m_value = value; emit valueChanged();}} + void setElementType(const QString &str) { if (m_type != str) {m_type = str; emit elementTypeChanged();}} + void setText(const QString &str) { if (m_text != str) {m_text = str; emit textChanged();}} + void setActiveControl(const QString &str) { if (m_activeControl != str) {m_activeControl = str; emit activeControlChanged();}} + + virtual void initStyleOption(QStyleOption *opt) const; +public Q_SLOTS: + int pixelMetric(const QString&) const; + QVariant styleHint(const QString&) const; + QSize sizeFromContents(int width, int height) const; + +Q_SIGNALS: + void elementTypeChanged(); + void textChanged(); + void sunkenChanged(); + void raisedChanged(); + void activeChanged(); + void enabledChanged(); + void selectedChanged(); + void focusChanged(); + void onChanged(); + void hoverChanged(); + void horizontalChanged(); + void minimumChanged(); + void maximumChanged(); + void valueChanged(); + void activeControlChanged(); + +protected: + QString m_type; + QString m_text; + QString m_activeControl; + bool m_sunken; + bool m_raised; + bool m_active; + bool m_enabled; + bool m_selected; + bool m_focus; + bool m_hover; + bool m_on; + bool m_horizontal; + int m_minimum; + int m_maximum; + int m_value; + AnimWidget m_dummywidget; +}; + +class QStyleBackground: public QDeclarativeItem +{ + Q_OBJECT +public: + Q_PROPERTY( QStyleItem* style READ style WRITE setStyle NOTIFY styleChanged) + + QStyleBackground(QDeclarativeItem *parent = 0); + void paint(QPainter *, const QStyleOptionGraphicsItem *, QWidget *); + +public Q_SLOTS: + QStyleItem *style(){return m_style;} + void setStyle(QStyleItem *style); + void updateItem(){update();} + QString hitTest(int x, int y) const; + QRect subControlRect(const QString &subcontrolString) const; + +Q_SIGNALS: + void styleChanged(); + +private: + AnimWidget m_dummywidget; + QWidget *m_menu; + QStyleItem *m_style; +}; + +#endif //STYLEWRAPPER_H diff --git a/components/styleitem/qstyleplugin.cpp b/components/styleitem/qstyleplugin.cpp new file mode 100644 index 0000000000000000000000000000000000000000..05cb45c56eab143ac0b991680a45f1d21dfde0c1 --- /dev/null +++ b/components/styleitem/qstyleplugin.cpp @@ -0,0 +1,83 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include <qdeclarative.h> +#include "qstyleplugin.h" +#include "qstyleitem.h" + +#include <qdeclarativeextensionplugin.h> + +#include <qdeclarativeengine.h> +#include <qdeclarative.h> +#include <qdeclarativeitem.h> +#include <qdeclarativeimageprovider.h> +#include <qdeclarativeview.h> +#include <QApplication> +#include <QImage> + +// Load icons from desktop theme +class DesktopIconProvider : public QDeclarativeImageProvider +{ +public: + DesktopIconProvider() + : QDeclarativeImageProvider(QDeclarativeImageProvider::Pixmap) + { + } + + QPixmap requestPixmap(const QString &id, QSize *size, const QSize &requestedSize) + { + int pos = id.lastIndexOf('/'); + QString iconName = id.right(id.length() - pos); + int width = qApp->style()->pixelMetric(QStyle::PM_ToolBarIconSize); + return QIcon::fromTheme(iconName).pixmap(width); + } +}; + +void StylePlugin::registerTypes(const char *uri) +{ + qmlRegisterType<QStyleItem>(uri, 1, 0, "QStyleItem"); + qmlRegisterType<QStyleBackground>(uri, 1, 0, "QStyleBackground"); +} + +void StylePlugin::initializeEngine(QDeclarativeEngine *engine, const char *uri) +{ + engine->addImageProvider("desktoptheme", new DesktopIconProvider); +} + +Q_EXPORT_PLUGIN2(styleplugin, StylePlugin); diff --git a/components/styleitem/qstyleplugin.h b/components/styleitem/qstyleplugin.h new file mode 100644 index 0000000000000000000000000000000000000000..45aff70f76428f0588b5bdf8e8abf0deeedee9c4 --- /dev/null +++ b/components/styleitem/qstyleplugin.h @@ -0,0 +1,55 @@ +/**************************************************************************** +** +** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef STYLEPLUGIN_H +#define STYLEPLUGIN_H + +#include <QDeclarativeExtensionPlugin> +#include <QtScript/QScriptValue> +#include <QtCore/QTimer> + +class StylePlugin : public QDeclarativeExtensionPlugin +{ + Q_OBJECT +public: + void registerTypes(const char *uri); + void initializeEngine(QDeclarativeEngine *engine, const char *uri); +}; + +#endif // STYLEPLUGIN_H diff --git a/components/styleitem/styleitem.pro b/components/styleitem/styleitem.pro new file mode 100644 index 0000000000000000000000000000000000000000..b254cd1ebb2e9d36352554cfe24d26f0ea870c99 --- /dev/null +++ b/components/styleitem/styleitem.pro @@ -0,0 +1,38 @@ +TEMPLATE = lib +CONFIG += qt plugin +QT += declarative +QT += script + +TARGET = styleplugin + +DESTDIR = ..\\plugin +OBJECTS_DIR = tmp +MOC_DIR = tmp + +HEADERS += qstyleplugin.h \ + qstyleitem.h + +SOURCES += qstyleplugin.cpp \ + qstyleitem.cpp + + +OTHER_FILES += \ + ../widgets/Button.qml \ + ../widgets/CheckBox.qml \ + ../widgets/ChoiceList.qml \ + ../widgets/GroupBox.qml \ + ../widgets/ProgressBar.qml \ + ../widgets/RadioButton.qml \ + ../widgets/ScrollArea.qml \ + ../widgets/ScrollBar.qml \ + ../widgets/Slider.qml \ + ../widgets/SpinBox.qml \ + ../widgets/Switch.qml \ + ../widgets/TextArea.qml \ + ../widgets/TextField.qml \ + ../widgets/ToolBar.qml \ + ../widgets/ToolButton.qml \ + ../gallery.qml \ + ../widgets/Tab.qml \ + ../widgets/TabBar.qml \ + ../widgets/TabFrame.qml diff --git a/src/Button.qml b/src/Button.qml new file mode 100644 index 0000000000000000000000000000000000000000..496d37df20878fc895dc1032475893e726712087 --- /dev/null +++ b/src/Button.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Button { + id:button + + property int buttonHeight: Math.max(22, styleitem.sizeFromContents(100, 6).height) + height: buttonHeight + + QStyleItem { + id:styleitem + elementType:"button" + sunken: pressed + raised: !pressed + hover: containsMouse + enabled:button.enabled + text:button.text + } + + background: + QStyleBackground { + style:styleitem + anchors.fill:parent + } +} + diff --git a/src/ButtonRow.qml b/src/ButtonRow.qml new file mode 100644 index 0000000000000000000000000000000000000000..623c5f44283fa00555d3ac8e07ad15278f732caa --- /dev/null +++ b/src/ButtonRow.qml @@ -0,0 +1,5 @@ +import QtQuick 1.0 +import "../../../components" as Components + +Components.ButtonRow { +} diff --git a/src/CheckBox.qml b/src/CheckBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..632dce2d4000584df89cc036b63823ee5ff857ac --- /dev/null +++ b/src/CheckBox.qml @@ -0,0 +1,26 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jb : Size should not depend on background, we should make it consistent + +Components.CheckBox{ + id:checkbox + property variant text + width:100 + height:18 + + background: QStyleBackground { + id:styleitem + style:QStyleItem { + elementType:"checkbox" + sunken:pressed + on:checked || pressed + hover:containsMouse + text:checkbox.text + enabled:checkbox.enabled + } + } + checkmark: null +} + diff --git a/src/ChoiceList.qml b/src/ChoiceList.qml new file mode 100644 index 0000000000000000000000000000000000000000..5015fbefbdd92866f81b46faf3a70e32840ff29b --- /dev/null +++ b/src/ChoiceList.qml @@ -0,0 +1,52 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.ChoiceList { + id:choicelist + + property int buttonHeight: buttonitem.sizeFromContents(100, 18).height + QStyleItem { id:buttonitem; elementType:"combobox" } + height: buttonHeight + topMargin:4 + bottomMargin:4 + + QStyleItem { + id:styleitem + elementType: "combobox" + sunken: pressed + raised: !pressed + hover: containsMouse + enabled:choicelist.enabled + } + + background: QStyleBackground { + anchors.fill:parent + style: styleitem + } + + listItem: Item { + id:item + + height:22 + anchors.left:parent.left + width:choicelist.width + QStyleBackground { + anchors.fill:parent + style: QStyleItem { + elementType: "menuitem" + text: choicelist.model.get(index).text + selected: highlighted + } + } + } + + popupFrame: QStyleBackground { + property int fw: styleitem.pixelMetric("menupanelwidth"); + anchors.leftMargin: styleitem.pixelMetric("menuhmargin") + fw + anchors.rightMargin: styleitem.pixelMetric("menuhmargin") + fw + anchors.topMargin: styleitem.pixelMetric("menuvmargin") + fw + anchors.bottomMargin: styleitem.pixelMetric("menuvmargin") + fw + style:QStyleItem{elementType:"menu"} + } +} diff --git a/src/GroupBox.qml b/src/GroupBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..0d45810d12d1964cb24c1fa6bfb11b7dc2d468d6 --- /dev/null +++ b/src/GroupBox.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Item { + width:200 + height:46 + + property alias text: styleitem.text + default property alias children: content.children + property bool checkable: false + + QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + id:styleitem + elementType:"groupbox" + } + + Item { + id:content + anchors.topMargin:22 + anchors.leftMargin:6 + anchors.fill:parent + } + } +} diff --git a/src/ProgressBar.qml b/src/ProgressBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..bf29bcac94fb4beb915ec407ac10b20e564d9d3e --- /dev/null +++ b/src/ProgressBar.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.ProgressBar { + id:progressbar + + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 15).height + QStyleItem { id:buttonitem; elementType:"button" } + height: buttonHeight + + background: QStyleBackground{ + anchors.fill:parent + style: QStyleItem { + elementType:"progressbar" + + // XXX: since desktop uses int instead of real, the progressbar + // range [0..1] must be stretched to a good precision + property int factor : 1000000 + + value: progressbar.value * factor + minimum: indeterminate ? 0 : progressbar.minimumValue * factor + maximum: indeterminate ? 0 : progressbar.maximumValue * factor + enabled: progressbar.enabled + } + } + indeterminateProgress:null + progress: null +} + diff --git a/src/RadioButton.qml b/src/RadioButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..3c2c3e5473503f88f95eddc4675d24c747ce7209 --- /dev/null +++ b/src/RadioButton.qml @@ -0,0 +1,26 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jb : Size should not depend on background, we should make it consistent + +Components.RadioButton{ + id:radiobutton + property variant text + width:110 + height:18 + + background: QStyleBackground { + + style: QStyleItem{ + elementType:"radiobutton" + sunken:pressed + on:checked || pressed + hover:containsMouse + text:radiobutton.text + enabled:radiobutton.enabled + } + } + checkmark: null +} + diff --git a/src/ScrollArea.qml b/src/ScrollArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..3a904ddebf7fa86027bfc84457eb17d7ec4ed058 --- /dev/null +++ b/src/ScrollArea.qml @@ -0,0 +1,103 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +FocusScope { + id:scrollarea + width: 100 + height: 100 + + property int contentMargin: 1 + property int __scrollbarExtent : styleitem.pixelMetric("scrollbarExtent"); + property int frameWidth: styleitem.pixelMetric("defaultframewidth"); + property int contentHeight : content.childrenRect.height + property int contentWidth: content.childrenRect.width + property alias color: flickable.color + property bool frame: true + property bool highlightOnFocus: false + + default property alias children: content.children + + property int contentY + property int contentX + + property bool frameAroundContents: styleitem.styleHint("framearoundcontents") + + onContentYChanged: { + vscrollbar.value = contentY + } + + onContentXChanged: { + hscrollbar.value = contentX + } + + QStyleBackground { + style: QStyleItem{ + id:styleitem + elementType: frame ? "frame" : "" + sunken: true + } + anchors.fill: parent + anchors.rightMargin: (frameAroundContents && vscrollbar.visible) ? vscrollbar.width + 4 : -frameWidth + anchors.bottomMargin: (frameAroundContents && hscrollbar.visible) ? hscrollbar.height + 4 : -frameWidth + anchors.topMargin: (frameAroundContents && hscrollbar.visible) ? hscrollbar.height + 4 : -frameWidth + + Rectangle { + id:flickable + color: "transparent" + anchors.fill: parent + anchors.margins: frame ? 2 : 0 + clip: true + + Item { + id: docmargins + anchors.fill:parent + anchors.margins:contentMargin + Item { + id: content + x: -scrollarea.contentX + y: -scrollarea.contentY + } + } + } + } + + ScrollBar { + id: hscrollbar + orientation: Qt.Horizontal + visible: contentWidth > flickable.width + maximumValue: contentWidth > flickable.width ? scrollarea.contentWidth - flickable.width : 0 + minimumValue: 0 + anchors.bottom: parent.bottom + anchors.left: parent.left + anchors.right: parent.right + anchors.rightMargin: { return (frame ? 1 : 0) + ( vscrollbar.visible ? __scrollbarExtent : 0) } + onValueChanged: contentX = value + } + + ScrollBar { + id: vscrollbar + orientation: Qt.Vertical + visible: contentHeight > flickable.height + maximumValue: contentHeight > flickable.height ? scrollarea.contentHeight - flickable.height : 0 + minimumValue: 0 + anchors.right: parent.right + anchors.top: parent.top + anchors.bottom: parent.bottom + anchors.bottomMargin: { return (frame ? 1 : 0) + (hscrollbar.visible ? __scrollbarExtent : 0) } + onValueChanged: contentY = value + } + + QStyleBackground { + z:2 + anchors.fill:parent + anchors.margins:-2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + visible: highlightOnFocus && parent.activeFocus && styleitem.styleHint("focuswidget") + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } +} diff --git a/src/ScrollBar.qml b/src/ScrollBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..ea822b62e82543ae0a68b41699b0f2673b3e2eee --- /dev/null +++ b/src/ScrollBar.qml @@ -0,0 +1,109 @@ +import QtQuick 1.1 +import "../../../components" as Components +import "../plugin" + +MouseArea { + id:scrollbar + property int __scrollbarExtent : styleitem.pixelMetric("scrollbarExtent"); + implicitWidth:orientation == Qt.Horizontal ? 200 : __scrollbarExtent; + implicitHeight:orientation == Qt.Horizontal ? __scrollbarExtent : 200 + + property int orientation : Qt.Horizontal + property alias minimumValue: slider.minimumValue + property alias maximumValue: slider.maximumValue + property alias value: slider.value + + property bool upPressed; + property bool downPressed; + property bool __autoincrement: false + + // Update hover item + onEntered: styleitem.activeControl = bgitem.hitTest(mouseX, mouseY) + onExited: styleitem.activeControl = "none" + onMouseXChanged: styleitem.activeControl = bgitem.hitTest(mouseX, mouseY) + hoverEnabled:true + + Timer { running: upPressed || downPressed; interval: 350 ; onTriggered: __autoincrement = true } + Timer { running: __autoincrement; interval: 60 ; repeat: true ; + onTriggered: upPressed ? decrement() : increment() } + + onPressed: { + var control = bgitem.hitTest(mouseX,mouseY) + if (control == "up") { + upPressed = true + } else if (control == "down") { + downPressed = true + } + } + + onReleased: { + __autoincrement = false; + if (upPressed) { + upPressed = false; + decrement() + } else if (downPressed) { + increment() + downPressed = false; + } + } + + function increment() { + value += 30 + if (value > maximumValue) + value = maximumValue + } + + function decrement() { + value -= 30 + if (value < minimumValue) + value = minimumValue + } + + QStyleBackground { + id:bgitem + anchors.fill:parent + style: QStyleItem { + id:styleitem + elementType: "scrollbar" + hover: activeControl != "none" + activeControl: "none" + sunken: upPressed | downPressed + minimum: slider.minimumValue + maximum: slider.maximumValue + value: slider.value + horizontal: orientation == Qt.Horizontal + enabled: parent.enabled + } + } + + property variant handleRect + function updateHandle() { + handleRect = bgitem.subControlRect("handle") + var grooveRect = bgitem.subControlRect("groove"); + var extra = 0 + if (orientation == Qt.Vertical) { + slider.anchors.topMargin = grooveRect.y + extra + slider.anchors.bottomMargin = height - grooveRect.y - grooveRect.height + extra + } else { + slider.anchors.leftMargin = grooveRect.x + extra + slider.anchors.rightMargin = width - grooveRect.x - grooveRect.width + extra + } + } + + onValueChanged: updateHandle() + onMaximumValueChanged: updateHandle() + onMinimumValueChanged: updateHandle() + Component.onCompleted: updateHandle() + Components.Slider { + id:slider + anchors.fill:parent + orientation:scrollbar.orientation + leftMargin: (orientation === Qt.Horizontal) ? handleRect.width/2 : handleRect.height/2 + rightMargin:leftMargin + handle: Item { width:orientation == Qt.Vertical ? handleRect.height : handleRect.width; + height:orientation == Qt.Vertical ? handleRect.width : handleRect.height} + groove:null + valueIndicator:null + inverted:orientation != Qt.Horizontal + } +} diff --git a/src/Slider.qml b/src/Slider.qml new file mode 100644 index 0000000000000000000000000000000000000000..c6b9c915187794195af2a1d75a6317627fada3bf --- /dev/null +++ b/src/Slider.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jens: ContainsMouse breaks drag functionality + +Components.Slider{ + id:slider + minimumWidth:200 + + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 15).height + QStyleItem { id:buttonitem; elementType:"button" } + height: buttonHeight + + groove: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"slider" + sunken: pressed + maximum:slider.maximumValue + minimum:slider.minimumValue + value:slider.value + horizontal:slider.orientation == Qt.Horizontal + enabled:slider.enabled + } + } + + handle: null + valueIndicator: null +} diff --git a/src/SpinBox.qml b/src/SpinBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..2db428cef894a82b7497096833bb36bb18e2c717 --- /dev/null +++ b/src/SpinBox.qml @@ -0,0 +1,91 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.SpinBox { + id:spinbox + + property variant __upRect; + property variant __downRect; + property int __margin: (height -16)/2 + + // Align height with button + topMargin:__margin + bottomMargin:__margin + + property int buttonHeight: edititem.sizeFromContents(100, 20).height + QStyleItem { id:edititem; elementType:"edit" } + height: buttonHeight + clip:false + + background: + Item { + anchors.fill: parent + property variant __editRect + + Rectangle { + id:editBackground + x: __editRect.x - 1 + y: __editRect.y + width: __editRect.width + height: __editRect.height + } + + Item { + id:focusFrame + anchors.fill: editBackground + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -6 + anchors.leftMargin: -5 + anchors.fill: parent + visible:spinbox.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } + + function updateRect() { + __upRect = spinboxbg.subControlRect("up"); + __downRect = spinboxbg.subControlRect("down"); + __editRect = spinboxbg.subControlRect("edit"); + } + + Component.onCompleted:updateRect() + onWidthChanged:updateRect() + onHeightChanged:updateRect() + + QStyleBackground { + id:spinboxbg + anchors.fill:parent + style: QStyleItem { + id: styleitem + elementType: "spinbox" + sunken: downPressed | upPressed + hover: containsMouse + focus: spinbox.activeFocus + enabled: spinbox.enabled + value: (upPressed? 1 : 0) | + (downPressed== 1 ? 1<<1 : 0) | + (upEnabled? (1<<2) : 0) | + (downEnabled == 1 ? (1<<3) : 0) + } + } + } + + up: Item { + x: __upRect.x + y: __upRect.y + width: __upRect.width + height: __upRect.height + } + + down: Item { + x: __downRect.x + y: __downRect.y + width: __downRect.width + height: __downRect.height + } +} diff --git a/src/Switch.qml b/src/Switch.qml new file mode 100644 index 0000000000000000000000000000000000000000..b82817c9f6071766d31216cedd1106c02b65201e --- /dev/null +++ b/src/Switch.qml @@ -0,0 +1,22 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Switch { + id:widget + minimumWidth:100 + minimumHeight:30 + + groove:QStyleItem { + elementType:"edit" + sunken: true + } + + handle: QStyleItem { + elementType:"button" + width:widget.width/2 + height:widget.height-4 + hover:containsMouse + } +} + diff --git a/src/Tab.qml b/src/Tab.qml new file mode 100644 index 0000000000000000000000000000000000000000..e258185ba5299b603b8240946dfc57f2cdfac167 --- /dev/null +++ b/src/Tab.qml @@ -0,0 +1,7 @@ +import Qt 4.7 + +Item { + id:tab + anchors.fill:parent + property string title +} diff --git a/src/TabBar.qml b/src/TabBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..2d891256f283ec06278b4f691ec939a0d37a3fbd --- /dev/null +++ b/src/TabBar.qml @@ -0,0 +1,90 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + + +Item { + id: tabbar + property int tabHeight: styleitem.sizeFromContents(100, 24).height + height: tabHeight + + property Item tabFrame + onTabFrameChanged:parent = tabFrame + visible: tabFrame ? tabFrame.tabsVisible : true + property int __overlap : styleitem.pixelMetric("tabvshift"); + property string position: tabFrame ? tabFrame.position : "North" + property string tabBarAlignment: styleitem.styleHint("tabbaralignment"); + property int tabOverlap: styleitem.pixelMetric("taboverlap"); + + function tab(index) { + for (var i = 0; i < tabrow.children.length; ++i) { + if (tabrow.children[i].tabindex == index) { + return tabrow.children[i] + } + } + return null; + } + + QStyleItem { + id:styleitem + elementType: "tab" + text: "generic" + } + + Row { + id:tabrow + states: + State { + when: tabBarAlignment == "center" + name: "centered" + AnchorChanges { + target:tabrow + anchors.horizontalCenter: tabbar.horizontalCenter + } + } + Repeater { + id:repeater + model: tabFrame ? tabFrame.tabs.length : null + delegate: Item { + id:tab + property int tabindex: index + property bool selected : tabFrame.current == index + width: textitem.width + 42 + height: tabHeight + z: selected ? 1 : -1 + + QStyleBackground { + style: QStyleItem { + id:style + elementType: "tab" + selected: tab.selected + text: tabbar.position + + activeControl: tabFrame.count == 1 ? + "only" : + index == 0 ? "beginning" : + index == tabFrame.count-1 ? "end" : "middle" + } + anchors.leftMargin: -tabOverlap + (style.text == "North" && (style.activeControl == "middle" || style.activeControl == "end") + && tab.selected ? -__overlap : 0) + + anchors.rightMargin: -tabOverlap + (style.text == "North" && (style.activeControl == "middle" || style.activeControl == "beginning") + && tab.selected ? -__overlap : 0) + anchors.fill:parent + } + + Text { + id:textitem + anchors.centerIn:parent + text: tabFrame.tabs[index].title + elide: Text.ElideRight + } + + MouseArea { + anchors.fill: parent + onPressed: tabFrame.current = index + } + } + } + } +} diff --git a/src/TabFrame.qml b/src/TabFrame.qml new file mode 100644 index 0000000000000000000000000000000000000000..01896e75353cc3ca79f644b3989b994f40d4dfd9 --- /dev/null +++ b/src/TabFrame.qml @@ -0,0 +1,72 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Item{ + id: tabWidget + width:100 + height:100 + property TabBar tabbar + property int current: 0 + property int count: stack.children.length + property bool frame:true + property bool tabsVisible: true + property string position: "North" + default property alias tabs : stack.children + + onCurrentChanged: __setOpacities() + Component.onCompleted: __setOpacities() + onTabbarChanged: { + tabbar.tabFrame = tabWidget + tabbar.anchors.top = tabWidget.top + tabbar.anchors.left = tabWidget.left + tabbar.anchors.right = tabWidget.right + } + + property int __baseOverlap : style.pixelMetric("tabbaseoverlap"); + function __setOpacities() { + for (var i = 0; i < stack.children.length; ++i) { + stack.children[i].opacity = (i == current ? 1 : 0) + } + } + + QStyleBackground { + id: frame + z:-1 + style: QStyleItem { + id:style + elementType: "tabframe" + text: position + value: tabbar && tabsVisible && tabbar.tab(current) ? tabbar.tab(current).x : 0 + minimum: tabbar && tabsVisible && tabbar.tab(current) ? tabbar.tab(current).width : 0 + } + anchors.fill:parent + Item { + id:stack + anchors.fill:parent + anchors.margins: frame ? 2 : 0 + } + anchors.topMargin: tabbar && tabsVisible && position == "North" ? tabbar.height - __baseOverlap : 0 + + states: [ + State { + name: "South" + when: position == "South" && tabbar!= undefined + PropertyChanges { + target: frame + anchors.topMargin: 0 + anchors.bottomMargin: tabbar ? tabbar.height - __baseOverlap: 0 + } + PropertyChanges { + target: tabbar + anchors.topMargin: -__baseOverlap + } + AnchorChanges { + target: tabbar + anchors.top: frame.bottom + anchors.bottom: undefined + } + } + ] + } +} diff --git a/src/TextArea.qml b/src/TextArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..93ad8f3d9c1c5962905c6f7007e93fb308daa93b --- /dev/null +++ b/src/TextArea.qml @@ -0,0 +1,44 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.TextArea { + id:textarea + leftMargin:12 + rightMargin:12 + minimumWidth:200 + desktopBehavior:true + placeholderText:"" + clip:false + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 14).height + QStyleItem { id:buttonitem; elementType:"button" } + minimumHeight: buttonHeight + + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"edit" + sunken:true + focus:textarea.activeFocus + } + } + + Item{ + id:focusFrame + anchors.fill: textarea + parent:textarea.parent + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + anchors.fill: parent + visible:textarea.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } +} diff --git a/src/TextField.qml b/src/TextField.qml new file mode 100644 index 0000000000000000000000000000000000000000..260584e9513147dd6917c11875415b5e54c2d058 --- /dev/null +++ b/src/TextField.qml @@ -0,0 +1,52 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.TextField { + id:textfield + minimumWidth:200 + desktopBehavior:true + placeholderText:"" + topMargin:2 + bottomMargin:2 + leftMargin:6 + rightMargin:6 + width:200 + height: editItem.sizeFromContents(100, 20).height + clip:false + + QStyleItem { + id:editItem + elementType:"edit" + sunken:true + focus:textfield.activeFocus + hover:containsMouse + } + + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"edit" + sunken:true + focus:textfield.activeFocus + } + } + + Item{ + id:focusFrame + anchors.fill: textfield + parent:textfield + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + anchors.fill: parent + visible:textfield.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } +} diff --git a/src/TextScrollArea.qml b/src/TextScrollArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..2a7deab683a5b6a63379c3f590959f1b2bf256a7 --- /dev/null +++ b/src/TextScrollArea.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +ScrollArea { + id:area + color: "white" + width: 280 + height: 120 + contentWidth: 200 + + property alias text: edit.text + property alias wrapMode: edit.wrapMode + highlightOnFocus: true + + TextEdit { + id: edit + text: loremIpsum + loremIpsum; + wrapMode: TextEdit.WordWrap; + width: area.contentWidth + selectByMouse:true + + // keep textcursor within scrollarea + onCursorRectangleChanged: + if (cursorRectangle.y >= area.contentY + area.height - 1.5*cursorRectangle.height) + area.contentY = cursorRectangle.y - area.height + 1.5*cursorRectangle.height + else if (cursorRectangle.y < area.contentY) + area.contentY = cursorRectangle.y + + } +} diff --git a/src/ToolBar.qml b/src/ToolBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..ce61382ee1b9b5a0ebb05db310e2616466aabdcf --- /dev/null +++ b/src/ToolBar.qml @@ -0,0 +1,12 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +QStyleBackground { + id:styleitem + width:200 + height:60 + + style: QStyleItem{elementType:"toolbar"} +} + diff --git a/src/ToolButton.qml b/src/ToolButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..98b2dfebf805f460ad4a19ca8504a71bf77cdc86 --- /dev/null +++ b/src/ToolButton.qml @@ -0,0 +1,18 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Button { + id:button + minimumWidth:30 + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem { + elementType: "toolbutton" + on: pressed | checked + sunken: pressed + raised: containsMouse + hover: containsMouse + } + } +} diff --git a/src/components/ButtonGroup.js b/src/components/ButtonGroup.js new file mode 100644 index 0000000000000000000000000000000000000000..f2d000029f7b1cabcf4803c9f608ca0218421549 --- /dev/null +++ b/src/components/ButtonGroup.js @@ -0,0 +1,127 @@ +var self; +var clickHandlers = []; +var visibleButtons = []; +var nonVisibleButtons = []; +var direction; +var exclusive; + +function create(that, options) { + self = that; + direction = options.direction || Qt.Horizontal; + exclusive = self.exclusive|| options.exclusive; + self.childrenChanged.connect(rebuild); +// self.widthChanged.connect(resizeChildren); + build(); +} + +function isButton(item) { + if (item && item["__position"] !== undefined) + return true; + return false; +} + +function hasChecked(item) { + if (item && item["checked"] !== undefined) + return true; + return false; +} + +function destroy() { + self.childrenChanged.disconnect(rebuild); +// self.widthChanged.disconnect(resizeChildren); + cleanup(); +} + +function build() { + visibleButtons = []; + nonVisibleButtons = []; + + for (var i = 0, item; (item = self.children[i]); i++) { + if (!hasChecked(item)) + continue; + + item.visibleChanged.connect(rebuild); // Not optimal, but hardly a bottleneck in your app + if (!item.visible) { + nonVisibleButtons.push(item); + continue; + } + visibleButtons.push(item); + + if (exclusive && hasChecked(item)) { + if (item["checkable"]!==undefined) { + item.checkable = true; + } + clickHandlers[i] = checkExclusive(item); + item.clicked.connect(clickHandlers[i]); + } + } + + if (self.checkedButton && !self.checkedButton.visible) + self.checkedButton = undefined; + + var nrButtons = visibleButtons.length; + if (nrButtons == 0) + return; + + if (nrButtons == 1) { + finishButton(visibleButtons[0], "only"); + } else { + finishButton(visibleButtons[0], direction == Qt.Horizontal ? "leftmost" : "top"); + for (var i = 1; i < nrButtons - 1; i++) + finishButton(visibleButtons[i], direction == Qt.Horizontal ? "h_middle": "v_middle"); + finishButton(visibleButtons[nrButtons - 1], direction == Qt.Horizontal ? "rightmost" : "bottom"); + } +} + +function finishButton(button, position) { + if (isButton(button)) { + button.__position = position; + if (direction == Qt.Vertical) { + button.anchors.left = self.left + button.anchors.right = self.right + } + } +} + +function cleanup() { + visibleButtons.forEach(function(item, i) { + if (clickHandlers[i]) + item.clicked.disconnect(clickHandlers[i]); + item.visibleChanged.disconnect(rebuild); + }); + clickHandlers = []; + + nonVisibleButtons.forEach(function(item, i) { + item.visibleChanged.disconnect(rebuild); + }); +} + +function rebuild() { + cleanup(); + build(); +} + +function resizeChildren() { + if (direction != Qt.Horizontal) + return; + + var extraPixels = self.width % visibleButtons; + var buttonSize = (self.width - extraPixels) / visibleButtons; + visibleButtons.forEach(function(item, i) { + if (!item || !item.visible) + return; + item.width = buttonSize + (extraPixels > 0 ? 1 : 0); + if (extraPixels > 0) + extraPixels--; + }); +} + +function checkExclusive(item) { + var button = item; + return function() { + for (var i = 0, ref; (ref = visibleButtons[i]); i++) { + ref.checked = button === ref; + } + self.checkedButton = button; + } +} diff --git a/src/components/components.pro b/src/components/components.pro new file mode 100644 index 0000000000000000000000000000000000000000..be0663d70948a7a81825e655ec3cf553ebb4de43 --- /dev/null +++ b/src/components/components.pro @@ -0,0 +1,49 @@ +TEMPLATE = subdirs # XXX: Avoid call the linker +TARGETPATH = Qt/labs/components/custom + +symbian { + INSTALL_IMPORTS = /resource/qt/imports +} else { + INSTALL_IMPORTS = $$[QT_INSTALL_IMPORTS] +} + +QML_FILES = \ + qmldir \ + BasicButton.qml \ + BusyIndicator.qml \ + ButtonBlock.qml \ + ButtonColumn.qml \ + ButtonRow.qml \ + ButtonGroup.js \ + Button.qml \ + CheckBox.qml \ + ChoiceList.qml \ + ProgressBar.qml \ + RadioButton.qml \ + ScrollDecorator.qml \ + ScrollIndicator.qml \ + Slider.qml \ + SpinBox.qml \ + Switch.qml \ + TextArea.qml \ + TextField.qml + +QML_DIRS = \ + behaviors \ + private \ + styles \ + visuals + +qmlfiles.files = $$QML_FILES +qmlfiles.sources = $$QML_FILES +qmlfiles.path = $$INSTALL_IMPORTS/$$TARGETPATH + +qmldirs.files = $$QML_DIRS +qmldirs.sources = $$QML_DIRS +qmldirs.path = $$INSTALL_IMPORTS/$$TARGETPATH + +INSTALLS += qmlfiles qmldirs + +symbian { + DEPLOYMENT += qmlfiles qmldirs +} diff --git a/src/components/qmldir b/src/components/qmldir new file mode 100644 index 0000000000000000000000000000000000000000..6e256a81d47c3ed8f88c300def7e2a36836a6d12 --- /dev/null +++ b/src/components/qmldir @@ -0,0 +1,17 @@ +BasicButton 1.0 BasicButton.qml +BusyIndicator 1.0 BusyIndicator.qml +ButtonBlock 1.0 ButtonBlock.qml +Button 1.0 Button.qml +ButtonColumn 1.0 ButtonColumn.qml +ButtonRow 1.0 ButtonRow.qml +CheckBox 1.0 CheckBox.qml +ChoiceList 1.0 ChoiceList.qml +ProgressBar 1.0 ProgressBar.qml +RadioButton 1.0 RadioButton.qml +ScrollDecorator 1.0 ScrollDecorator.qml +ScrollIndicator 1.0 ScrollIndicator.qml +Slider 1.0 Slider.qml +SpinBox 1.0 SpinBox.qml +Switch 1.0 Switch.qml +TextArea 1.0 TextArea.qml +TextField 1.0 TextField.qml diff --git a/src/images/folder_new.png b/src/images/folder_new.png new file mode 100644 index 0000000000000000000000000000000000000000..8d8bb9bd768d8e8ab785d95483ead02ae6800dc5 Binary files /dev/null and b/src/images/folder_new.png differ diff --git a/src/plugin/qmldir b/src/plugin/qmldir new file mode 100644 index 0000000000000000000000000000000000000000..e8452efd6f37818e71e256646eb1d06a60dccf0f --- /dev/null +++ b/src/plugin/qmldir @@ -0,0 +1 @@ +plugin styleplugin diff --git a/src/styleitem/styleitem.pro b/src/styleitem/styleitem.pro new file mode 100644 index 0000000000000000000000000000000000000000..b254cd1ebb2e9d36352554cfe24d26f0ea870c99 --- /dev/null +++ b/src/styleitem/styleitem.pro @@ -0,0 +1,38 @@ +TEMPLATE = lib +CONFIG += qt plugin +QT += declarative +QT += script + +TARGET = styleplugin + +DESTDIR = ..\\plugin +OBJECTS_DIR = tmp +MOC_DIR = tmp + +HEADERS += qstyleplugin.h \ + qstyleitem.h + +SOURCES += qstyleplugin.cpp \ + qstyleitem.cpp + + +OTHER_FILES += \ + ../widgets/Button.qml \ + ../widgets/CheckBox.qml \ + ../widgets/ChoiceList.qml \ + ../widgets/GroupBox.qml \ + ../widgets/ProgressBar.qml \ + ../widgets/RadioButton.qml \ + ../widgets/ScrollArea.qml \ + ../widgets/ScrollBar.qml \ + ../widgets/Slider.qml \ + ../widgets/SpinBox.qml \ + ../widgets/Switch.qml \ + ../widgets/TextArea.qml \ + ../widgets/TextField.qml \ + ../widgets/ToolBar.qml \ + ../widgets/ToolButton.qml \ + ../gallery.qml \ + ../widgets/Tab.qml \ + ../widgets/TabBar.qml \ + ../widgets/TabFrame.qml diff --git a/src/widgets/Button.qml b/src/widgets/Button.qml new file mode 100644 index 0000000000000000000000000000000000000000..496d37df20878fc895dc1032475893e726712087 --- /dev/null +++ b/src/widgets/Button.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Button { + id:button + + property int buttonHeight: Math.max(22, styleitem.sizeFromContents(100, 6).height) + height: buttonHeight + + QStyleItem { + id:styleitem + elementType:"button" + sunken: pressed + raised: !pressed + hover: containsMouse + enabled:button.enabled + text:button.text + } + + background: + QStyleBackground { + style:styleitem + anchors.fill:parent + } +} + diff --git a/src/widgets/ButtonRow.qml b/src/widgets/ButtonRow.qml new file mode 100644 index 0000000000000000000000000000000000000000..623c5f44283fa00555d3ac8e07ad15278f732caa --- /dev/null +++ b/src/widgets/ButtonRow.qml @@ -0,0 +1,5 @@ +import QtQuick 1.0 +import "../../../components" as Components + +Components.ButtonRow { +} diff --git a/src/widgets/CheckBox.qml b/src/widgets/CheckBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..632dce2d4000584df89cc036b63823ee5ff857ac --- /dev/null +++ b/src/widgets/CheckBox.qml @@ -0,0 +1,26 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jb : Size should not depend on background, we should make it consistent + +Components.CheckBox{ + id:checkbox + property variant text + width:100 + height:18 + + background: QStyleBackground { + id:styleitem + style:QStyleItem { + elementType:"checkbox" + sunken:pressed + on:checked || pressed + hover:containsMouse + text:checkbox.text + enabled:checkbox.enabled + } + } + checkmark: null +} + diff --git a/src/widgets/ChoiceList.qml b/src/widgets/ChoiceList.qml new file mode 100644 index 0000000000000000000000000000000000000000..5015fbefbdd92866f81b46faf3a70e32840ff29b --- /dev/null +++ b/src/widgets/ChoiceList.qml @@ -0,0 +1,52 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.ChoiceList { + id:choicelist + + property int buttonHeight: buttonitem.sizeFromContents(100, 18).height + QStyleItem { id:buttonitem; elementType:"combobox" } + height: buttonHeight + topMargin:4 + bottomMargin:4 + + QStyleItem { + id:styleitem + elementType: "combobox" + sunken: pressed + raised: !pressed + hover: containsMouse + enabled:choicelist.enabled + } + + background: QStyleBackground { + anchors.fill:parent + style: styleitem + } + + listItem: Item { + id:item + + height:22 + anchors.left:parent.left + width:choicelist.width + QStyleBackground { + anchors.fill:parent + style: QStyleItem { + elementType: "menuitem" + text: choicelist.model.get(index).text + selected: highlighted + } + } + } + + popupFrame: QStyleBackground { + property int fw: styleitem.pixelMetric("menupanelwidth"); + anchors.leftMargin: styleitem.pixelMetric("menuhmargin") + fw + anchors.rightMargin: styleitem.pixelMetric("menuhmargin") + fw + anchors.topMargin: styleitem.pixelMetric("menuvmargin") + fw + anchors.bottomMargin: styleitem.pixelMetric("menuvmargin") + fw + style:QStyleItem{elementType:"menu"} + } +} diff --git a/src/widgets/GroupBox.qml b/src/widgets/GroupBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..0d45810d12d1964cb24c1fa6bfb11b7dc2d468d6 --- /dev/null +++ b/src/widgets/GroupBox.qml @@ -0,0 +1,27 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Item { + width:200 + height:46 + + property alias text: styleitem.text + default property alias children: content.children + property bool checkable: false + + QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + id:styleitem + elementType:"groupbox" + } + + Item { + id:content + anchors.topMargin:22 + anchors.leftMargin:6 + anchors.fill:parent + } + } +} diff --git a/src/widgets/ProgressBar.qml b/src/widgets/ProgressBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..bf29bcac94fb4beb915ec407ac10b20e564d9d3e --- /dev/null +++ b/src/widgets/ProgressBar.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.ProgressBar { + id:progressbar + + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 15).height + QStyleItem { id:buttonitem; elementType:"button" } + height: buttonHeight + + background: QStyleBackground{ + anchors.fill:parent + style: QStyleItem { + elementType:"progressbar" + + // XXX: since desktop uses int instead of real, the progressbar + // range [0..1] must be stretched to a good precision + property int factor : 1000000 + + value: progressbar.value * factor + minimum: indeterminate ? 0 : progressbar.minimumValue * factor + maximum: indeterminate ? 0 : progressbar.maximumValue * factor + enabled: progressbar.enabled + } + } + indeterminateProgress:null + progress: null +} + diff --git a/src/widgets/RadioButton.qml b/src/widgets/RadioButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..3c2c3e5473503f88f95eddc4675d24c747ce7209 --- /dev/null +++ b/src/widgets/RadioButton.qml @@ -0,0 +1,26 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jb : Size should not depend on background, we should make it consistent + +Components.RadioButton{ + id:radiobutton + property variant text + width:110 + height:18 + + background: QStyleBackground { + + style: QStyleItem{ + elementType:"radiobutton" + sunken:pressed + on:checked || pressed + hover:containsMouse + text:radiobutton.text + enabled:radiobutton.enabled + } + } + checkmark: null +} + diff --git a/src/widgets/ScrollArea.qml b/src/widgets/ScrollArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..3a904ddebf7fa86027bfc84457eb17d7ec4ed058 --- /dev/null +++ b/src/widgets/ScrollArea.qml @@ -0,0 +1,103 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +FocusScope { + id:scrollarea + width: 100 + height: 100 + + property int contentMargin: 1 + property int __scrollbarExtent : styleitem.pixelMetric("scrollbarExtent"); + property int frameWidth: styleitem.pixelMetric("defaultframewidth"); + property int contentHeight : content.childrenRect.height + property int contentWidth: content.childrenRect.width + property alias color: flickable.color + property bool frame: true + property bool highlightOnFocus: false + + default property alias children: content.children + + property int contentY + property int contentX + + property bool frameAroundContents: styleitem.styleHint("framearoundcontents") + + onContentYChanged: { + vscrollbar.value = contentY + } + + onContentXChanged: { + hscrollbar.value = contentX + } + + QStyleBackground { + style: QStyleItem{ + id:styleitem + elementType: frame ? "frame" : "" + sunken: true + } + anchors.fill: parent + anchors.rightMargin: (frameAroundContents && vscrollbar.visible) ? vscrollbar.width + 4 : -frameWidth + anchors.bottomMargin: (frameAroundContents && hscrollbar.visible) ? hscrollbar.height + 4 : -frameWidth + anchors.topMargin: (frameAroundContents && hscrollbar.visible) ? hscrollbar.height + 4 : -frameWidth + + Rectangle { + id:flickable + color: "transparent" + anchors.fill: parent + anchors.margins: frame ? 2 : 0 + clip: true + + Item { + id: docmargins + anchors.fill:parent + anchors.margins:contentMargin + Item { + id: content + x: -scrollarea.contentX + y: -scrollarea.contentY + } + } + } + } + + ScrollBar { + id: hscrollbar + orientation: Qt.Horizontal + visible: contentWidth > flickable.width + maximumValue: contentWidth > flickable.width ? scrollarea.contentWidth - flickable.width : 0 + minimumValue: 0 + anchors.bottom: parent.bottom + anchors.left: parent.left + anchors.right: parent.right + anchors.rightMargin: { return (frame ? 1 : 0) + ( vscrollbar.visible ? __scrollbarExtent : 0) } + onValueChanged: contentX = value + } + + ScrollBar { + id: vscrollbar + orientation: Qt.Vertical + visible: contentHeight > flickable.height + maximumValue: contentHeight > flickable.height ? scrollarea.contentHeight - flickable.height : 0 + minimumValue: 0 + anchors.right: parent.right + anchors.top: parent.top + anchors.bottom: parent.bottom + anchors.bottomMargin: { return (frame ? 1 : 0) + (hscrollbar.visible ? __scrollbarExtent : 0) } + onValueChanged: contentY = value + } + + QStyleBackground { + z:2 + anchors.fill:parent + anchors.margins:-2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + visible: highlightOnFocus && parent.activeFocus && styleitem.styleHint("focuswidget") + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } +} diff --git a/src/widgets/ScrollBar.qml b/src/widgets/ScrollBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..ea822b62e82543ae0a68b41699b0f2673b3e2eee --- /dev/null +++ b/src/widgets/ScrollBar.qml @@ -0,0 +1,109 @@ +import QtQuick 1.1 +import "../../../components" as Components +import "../plugin" + +MouseArea { + id:scrollbar + property int __scrollbarExtent : styleitem.pixelMetric("scrollbarExtent"); + implicitWidth:orientation == Qt.Horizontal ? 200 : __scrollbarExtent; + implicitHeight:orientation == Qt.Horizontal ? __scrollbarExtent : 200 + + property int orientation : Qt.Horizontal + property alias minimumValue: slider.minimumValue + property alias maximumValue: slider.maximumValue + property alias value: slider.value + + property bool upPressed; + property bool downPressed; + property bool __autoincrement: false + + // Update hover item + onEntered: styleitem.activeControl = bgitem.hitTest(mouseX, mouseY) + onExited: styleitem.activeControl = "none" + onMouseXChanged: styleitem.activeControl = bgitem.hitTest(mouseX, mouseY) + hoverEnabled:true + + Timer { running: upPressed || downPressed; interval: 350 ; onTriggered: __autoincrement = true } + Timer { running: __autoincrement; interval: 60 ; repeat: true ; + onTriggered: upPressed ? decrement() : increment() } + + onPressed: { + var control = bgitem.hitTest(mouseX,mouseY) + if (control == "up") { + upPressed = true + } else if (control == "down") { + downPressed = true + } + } + + onReleased: { + __autoincrement = false; + if (upPressed) { + upPressed = false; + decrement() + } else if (downPressed) { + increment() + downPressed = false; + } + } + + function increment() { + value += 30 + if (value > maximumValue) + value = maximumValue + } + + function decrement() { + value -= 30 + if (value < minimumValue) + value = minimumValue + } + + QStyleBackground { + id:bgitem + anchors.fill:parent + style: QStyleItem { + id:styleitem + elementType: "scrollbar" + hover: activeControl != "none" + activeControl: "none" + sunken: upPressed | downPressed + minimum: slider.minimumValue + maximum: slider.maximumValue + value: slider.value + horizontal: orientation == Qt.Horizontal + enabled: parent.enabled + } + } + + property variant handleRect + function updateHandle() { + handleRect = bgitem.subControlRect("handle") + var grooveRect = bgitem.subControlRect("groove"); + var extra = 0 + if (orientation == Qt.Vertical) { + slider.anchors.topMargin = grooveRect.y + extra + slider.anchors.bottomMargin = height - grooveRect.y - grooveRect.height + extra + } else { + slider.anchors.leftMargin = grooveRect.x + extra + slider.anchors.rightMargin = width - grooveRect.x - grooveRect.width + extra + } + } + + onValueChanged: updateHandle() + onMaximumValueChanged: updateHandle() + onMinimumValueChanged: updateHandle() + Component.onCompleted: updateHandle() + Components.Slider { + id:slider + anchors.fill:parent + orientation:scrollbar.orientation + leftMargin: (orientation === Qt.Horizontal) ? handleRect.width/2 : handleRect.height/2 + rightMargin:leftMargin + handle: Item { width:orientation == Qt.Vertical ? handleRect.height : handleRect.width; + height:orientation == Qt.Vertical ? handleRect.width : handleRect.height} + groove:null + valueIndicator:null + inverted:orientation != Qt.Horizontal + } +} diff --git a/src/widgets/Slider.qml b/src/widgets/Slider.qml new file mode 100644 index 0000000000000000000000000000000000000000..c6b9c915187794195af2a1d75a6317627fada3bf --- /dev/null +++ b/src/widgets/Slider.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +// jens: ContainsMouse breaks drag functionality + +Components.Slider{ + id:slider + minimumWidth:200 + + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 15).height + QStyleItem { id:buttonitem; elementType:"button" } + height: buttonHeight + + groove: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"slider" + sunken: pressed + maximum:slider.maximumValue + minimum:slider.minimumValue + value:slider.value + horizontal:slider.orientation == Qt.Horizontal + enabled:slider.enabled + } + } + + handle: null + valueIndicator: null +} diff --git a/src/widgets/SpinBox.qml b/src/widgets/SpinBox.qml new file mode 100644 index 0000000000000000000000000000000000000000..2db428cef894a82b7497096833bb36bb18e2c717 --- /dev/null +++ b/src/widgets/SpinBox.qml @@ -0,0 +1,91 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.SpinBox { + id:spinbox + + property variant __upRect; + property variant __downRect; + property int __margin: (height -16)/2 + + // Align height with button + topMargin:__margin + bottomMargin:__margin + + property int buttonHeight: edititem.sizeFromContents(100, 20).height + QStyleItem { id:edititem; elementType:"edit" } + height: buttonHeight + clip:false + + background: + Item { + anchors.fill: parent + property variant __editRect + + Rectangle { + id:editBackground + x: __editRect.x - 1 + y: __editRect.y + width: __editRect.width + height: __editRect.height + } + + Item { + id:focusFrame + anchors.fill: editBackground + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -6 + anchors.leftMargin: -5 + anchors.fill: parent + visible:spinbox.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } + + function updateRect() { + __upRect = spinboxbg.subControlRect("up"); + __downRect = spinboxbg.subControlRect("down"); + __editRect = spinboxbg.subControlRect("edit"); + } + + Component.onCompleted:updateRect() + onWidthChanged:updateRect() + onHeightChanged:updateRect() + + QStyleBackground { + id:spinboxbg + anchors.fill:parent + style: QStyleItem { + id: styleitem + elementType: "spinbox" + sunken: downPressed | upPressed + hover: containsMouse + focus: spinbox.activeFocus + enabled: spinbox.enabled + value: (upPressed? 1 : 0) | + (downPressed== 1 ? 1<<1 : 0) | + (upEnabled? (1<<2) : 0) | + (downEnabled == 1 ? (1<<3) : 0) + } + } + } + + up: Item { + x: __upRect.x + y: __upRect.y + width: __upRect.width + height: __upRect.height + } + + down: Item { + x: __downRect.x + y: __downRect.y + width: __downRect.width + height: __downRect.height + } +} diff --git a/src/widgets/Switch.qml b/src/widgets/Switch.qml new file mode 100644 index 0000000000000000000000000000000000000000..b82817c9f6071766d31216cedd1106c02b65201e --- /dev/null +++ b/src/widgets/Switch.qml @@ -0,0 +1,22 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Switch { + id:widget + minimumWidth:100 + minimumHeight:30 + + groove:QStyleItem { + elementType:"edit" + sunken: true + } + + handle: QStyleItem { + elementType:"button" + width:widget.width/2 + height:widget.height-4 + hover:containsMouse + } +} + diff --git a/src/widgets/Tab.qml b/src/widgets/Tab.qml new file mode 100644 index 0000000000000000000000000000000000000000..e258185ba5299b603b8240946dfc57f2cdfac167 --- /dev/null +++ b/src/widgets/Tab.qml @@ -0,0 +1,7 @@ +import Qt 4.7 + +Item { + id:tab + anchors.fill:parent + property string title +} diff --git a/src/widgets/TabBar.qml b/src/widgets/TabBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..2d891256f283ec06278b4f691ec939a0d37a3fbd --- /dev/null +++ b/src/widgets/TabBar.qml @@ -0,0 +1,90 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + + +Item { + id: tabbar + property int tabHeight: styleitem.sizeFromContents(100, 24).height + height: tabHeight + + property Item tabFrame + onTabFrameChanged:parent = tabFrame + visible: tabFrame ? tabFrame.tabsVisible : true + property int __overlap : styleitem.pixelMetric("tabvshift"); + property string position: tabFrame ? tabFrame.position : "North" + property string tabBarAlignment: styleitem.styleHint("tabbaralignment"); + property int tabOverlap: styleitem.pixelMetric("taboverlap"); + + function tab(index) { + for (var i = 0; i < tabrow.children.length; ++i) { + if (tabrow.children[i].tabindex == index) { + return tabrow.children[i] + } + } + return null; + } + + QStyleItem { + id:styleitem + elementType: "tab" + text: "generic" + } + + Row { + id:tabrow + states: + State { + when: tabBarAlignment == "center" + name: "centered" + AnchorChanges { + target:tabrow + anchors.horizontalCenter: tabbar.horizontalCenter + } + } + Repeater { + id:repeater + model: tabFrame ? tabFrame.tabs.length : null + delegate: Item { + id:tab + property int tabindex: index + property bool selected : tabFrame.current == index + width: textitem.width + 42 + height: tabHeight + z: selected ? 1 : -1 + + QStyleBackground { + style: QStyleItem { + id:style + elementType: "tab" + selected: tab.selected + text: tabbar.position + + activeControl: tabFrame.count == 1 ? + "only" : + index == 0 ? "beginning" : + index == tabFrame.count-1 ? "end" : "middle" + } + anchors.leftMargin: -tabOverlap + (style.text == "North" && (style.activeControl == "middle" || style.activeControl == "end") + && tab.selected ? -__overlap : 0) + + anchors.rightMargin: -tabOverlap + (style.text == "North" && (style.activeControl == "middle" || style.activeControl == "beginning") + && tab.selected ? -__overlap : 0) + anchors.fill:parent + } + + Text { + id:textitem + anchors.centerIn:parent + text: tabFrame.tabs[index].title + elide: Text.ElideRight + } + + MouseArea { + anchors.fill: parent + onPressed: tabFrame.current = index + } + } + } + } +} diff --git a/src/widgets/TabFrame.qml b/src/widgets/TabFrame.qml new file mode 100644 index 0000000000000000000000000000000000000000..01896e75353cc3ca79f644b3989b994f40d4dfd9 --- /dev/null +++ b/src/widgets/TabFrame.qml @@ -0,0 +1,72 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Item{ + id: tabWidget + width:100 + height:100 + property TabBar tabbar + property int current: 0 + property int count: stack.children.length + property bool frame:true + property bool tabsVisible: true + property string position: "North" + default property alias tabs : stack.children + + onCurrentChanged: __setOpacities() + Component.onCompleted: __setOpacities() + onTabbarChanged: { + tabbar.tabFrame = tabWidget + tabbar.anchors.top = tabWidget.top + tabbar.anchors.left = tabWidget.left + tabbar.anchors.right = tabWidget.right + } + + property int __baseOverlap : style.pixelMetric("tabbaseoverlap"); + function __setOpacities() { + for (var i = 0; i < stack.children.length; ++i) { + stack.children[i].opacity = (i == current ? 1 : 0) + } + } + + QStyleBackground { + id: frame + z:-1 + style: QStyleItem { + id:style + elementType: "tabframe" + text: position + value: tabbar && tabsVisible && tabbar.tab(current) ? tabbar.tab(current).x : 0 + minimum: tabbar && tabsVisible && tabbar.tab(current) ? tabbar.tab(current).width : 0 + } + anchors.fill:parent + Item { + id:stack + anchors.fill:parent + anchors.margins: frame ? 2 : 0 + } + anchors.topMargin: tabbar && tabsVisible && position == "North" ? tabbar.height - __baseOverlap : 0 + + states: [ + State { + name: "South" + when: position == "South" && tabbar!= undefined + PropertyChanges { + target: frame + anchors.topMargin: 0 + anchors.bottomMargin: tabbar ? tabbar.height - __baseOverlap: 0 + } + PropertyChanges { + target: tabbar + anchors.topMargin: -__baseOverlap + } + AnchorChanges { + target: tabbar + anchors.top: frame.bottom + anchors.bottom: undefined + } + } + ] + } +} diff --git a/src/widgets/TextArea.qml b/src/widgets/TextArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..93ad8f3d9c1c5962905c6f7007e93fb308daa93b --- /dev/null +++ b/src/widgets/TextArea.qml @@ -0,0 +1,44 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.TextArea { + id:textarea + leftMargin:12 + rightMargin:12 + minimumWidth:200 + desktopBehavior:true + placeholderText:"" + clip:false + // Align with button + property int buttonHeight: buttonitem.sizeFromContents(100, 14).height + QStyleItem { id:buttonitem; elementType:"button" } + minimumHeight: buttonHeight + + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"edit" + sunken:true + focus:textarea.activeFocus + } + } + + Item{ + id:focusFrame + anchors.fill: textarea + parent:textarea.parent + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + anchors.fill: parent + visible:textarea.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } +} diff --git a/src/widgets/TextField.qml b/src/widgets/TextField.qml new file mode 100644 index 0000000000000000000000000000000000000000..260584e9513147dd6917c11875415b5e54c2d058 --- /dev/null +++ b/src/widgets/TextField.qml @@ -0,0 +1,52 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.TextField { + id:textfield + minimumWidth:200 + desktopBehavior:true + placeholderText:"" + topMargin:2 + bottomMargin:2 + leftMargin:6 + rightMargin:6 + width:200 + height: editItem.sizeFromContents(100, 20).height + clip:false + + QStyleItem { + id:editItem + elementType:"edit" + sunken:true + focus:textfield.activeFocus + hover:containsMouse + } + + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem{ + elementType:"edit" + sunken:true + focus:textfield.activeFocus + } + } + + Item{ + id:focusFrame + anchors.fill: textfield + parent:textfield + visible:framestyle.styleHint("focuswidget") + QStyleBackground{ + anchors.margins: -2 + anchors.rightMargin:-4 + anchors.bottomMargin:-4 + anchors.fill: parent + visible:textfield.activeFocus + style: QStyleItem { + id:framestyle + elementType:"focusframe" + } + } + } +} diff --git a/src/widgets/TextScrollArea.qml b/src/widgets/TextScrollArea.qml new file mode 100644 index 0000000000000000000000000000000000000000..2a7deab683a5b6a63379c3f590959f1b2bf256a7 --- /dev/null +++ b/src/widgets/TextScrollArea.qml @@ -0,0 +1,31 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +ScrollArea { + id:area + color: "white" + width: 280 + height: 120 + contentWidth: 200 + + property alias text: edit.text + property alias wrapMode: edit.wrapMode + highlightOnFocus: true + + TextEdit { + id: edit + text: loremIpsum + loremIpsum; + wrapMode: TextEdit.WordWrap; + width: area.contentWidth + selectByMouse:true + + // keep textcursor within scrollarea + onCursorRectangleChanged: + if (cursorRectangle.y >= area.contentY + area.height - 1.5*cursorRectangle.height) + area.contentY = cursorRectangle.y - area.height + 1.5*cursorRectangle.height + else if (cursorRectangle.y < area.contentY) + area.contentY = cursorRectangle.y + + } +} diff --git a/src/widgets/ToolBar.qml b/src/widgets/ToolBar.qml new file mode 100644 index 0000000000000000000000000000000000000000..ce61382ee1b9b5a0ebb05db310e2616466aabdcf --- /dev/null +++ b/src/widgets/ToolBar.qml @@ -0,0 +1,12 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +QStyleBackground { + id:styleitem + width:200 + height:60 + + style: QStyleItem{elementType:"toolbar"} +} + diff --git a/src/widgets/ToolButton.qml b/src/widgets/ToolButton.qml new file mode 100644 index 0000000000000000000000000000000000000000..98b2dfebf805f460ad4a19ca8504a71bf77cdc86 --- /dev/null +++ b/src/widgets/ToolButton.qml @@ -0,0 +1,18 @@ +import QtQuick 1.0 +import "../../../components" as Components +import "../plugin" + +Components.Button { + id:button + minimumWidth:30 + background: QStyleBackground { + anchors.fill:parent + style: QStyleItem { + elementType: "toolbutton" + on: pressed | checked + sunken: pressed + raised: containsMouse + hover: containsMouse + } + } +}