[Qt] QML(2) - State/Transition, ListView, Property
지난 포스팅에선 간단히 오브젝트와 컨테이너를 이용해서 배치하고 이벤트 처리를 하는 예제를 해봤습니다.
이번에는 State와 Property 등에 대해 알아보려고 합니다.
State는 Transition이라는 개념과 같이 나오는데 스위치를 예를 들자면 스위치는 ON/OFF State가 있고 ON/OFF를 할때마다 설정한 값이 바뀌게 되는데 이에 대한 행동이나 작업을 Trasition이라고 합니다.
왼쪽에는 램프가 있고 오른쪽에 스위치를 하나 두었습니다. 스위치를 클릭하면 램프의 상태가 바뀌고 램프는 상태에 따라 Text와 색상이 바뀌께 됩니다.
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
width: 200
height: 100
visible: true
Row {
width:200
height:100
Rectangle {
id:lamp
width:100
height:100
border.color: "black"
Text {
id: lampText
}
state: "off"
states: [
State {
name: "on"
PropertyChanges {
target: lamp
color:"yellow"
}
PropertyChanges {
target: lampText
text:"on"
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
color:"black"
}
},
State {
name: "off"
PropertyChanges {
target: lamp
color:"black"
}
PropertyChanges {
target: lampText
text:"off"
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
color:"white"
}
}
]
}
Rectangle {
id:swit
width:100
height:100
border.color: "black"
MouseArea {
anchors.fill: parent
onClicked: {
lamp.state == "on" ? lamp.state = "off" : lamp.state = "on";
}
}
}
}
}
이 때 on state에서 off state로 이동할 때 처럼 State 간 변화가 있을 때의 행동을 Transitions이라고 합니다.
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
width: 200
height: 100
visible: true
Row {
width:200
height:100
Rectangle {
id:lamp
width:100
height:100
border.color: "black"
Text {
id: lampText
width: parent.width
height: parent.height
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
}
state: "off"
states: [
State {
name: "on"
PropertyChanges {
target: lamp
color:"yellow"
}
PropertyChanges {
target: lampText
text:"on"
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
color:"black"
}
},
State {
name: "off"
PropertyChanges {
target: lamp
color:"black"
}
PropertyChanges {
target: lampText
text:"off"
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
color:"white"
}
}
]
transitions: [
Transition {
from: "off"
to: "on"
ColorAnimation { duration : 1000 }
}
]
}
Rectangle {
id:swit
width:100
height:100
border.color: "black"
Text {
text: "Switch"
width: parent.width
height: parent.height
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
}
MouseArea {
anchors.fill: parent
onClicked: {
lamp.state == "on" ? lamp.state = "off" : lamp.state = "on";
}
}
}
}
}
위 코드에서 off->on으로 상태가 변경될 때는 Color 변경에 대해 Animation을 적용했습니다.
Qt에서 List나 Table 같은 데이터를 표현하기 위해서 Model 부분과 View 부를 제공합니다. 아래 그림을 보면 Model은 Data를 읽어 담아놓는 Container입니다. View는 Model로부터 읽어온 데이터를 표시하거나 핸들링하게 해줍니다.
먼저 List입니다. List도 마찬가지로 Model과 View로 데이터를 관리/표시 하며 Delegate를 통해 표시할 데이터의 Style을 꾸며줄 수 있습니다.
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
width: 300
height: 100
visible: true
ListModel {
id:nameModel
ListElement { name: "park" }
ListElement { name: "kim" }
ListElement { name: "lee" }
} // 데이터를 담는 Model
Component {
id: nameDelegate
Text {
text : name
font.pixelSize: 20
}
} // 데이터를 표현하는 Style
ListView {
anchors.fill: parent
model : nameModel
delegate: nameDelegate
} // 데이터를 보여주는 View
}
데이터를 변경하고 싶을 땐 Model을 Style을 변경하고 싶을 땐 Component(Delegate)를 실제 표시하는 오브젝트를 변경하고 싶을 때는 View를 수정하시면 됩니다.
Delegate는 한 데이터(Element)에 대해 꾸며주는 Style라고 생각하면 좋습니다. 위 예제에서 Element의 데이터가 추가되고 이 추가된 데이터를 표현할 Style을 더 추가해보겠습니다.
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
width: 300
height: nameModel.count * 50
visible: true
id:windows
ListModel {
id:nameModel
ListElement {
name: "park"
age : 20
}
ListElement {
name: "kim"
age : 25
}
ListElement {
name: "lee"
age : 30
}
} // 데이터를 담는 Model
Component {
id: nameDelegate
Rectangle {
width : windows.width
height : windows.height/nameModel.count
border.color: "black"
border.width: 1
Text {
anchors.fill: parent
text: "name : " + name + "\n" + "age : " + age;
verticalAlignment: Text.AlignVCenter
anchors.leftMargin: 10
}
}
} // 데이터를 표현하는 Style
ListView {
anchors.fill: parent
model : nameModel
delegate: nameDelegate
} // 데이터를 보여주는 View
}
여기에 리스트에 데이터를 추가하고 삭제하는 오브젝트도 넣어보겠습니다.
먼저 추가하는 코드입니다. 아래 코드에선 UI 크기 관리를 위해서 변수를 사용하였습니다. 변수는 property 키워드를 붙이고 자료형 + 변수명으로 사용 가능합니다.
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
property int listViewheight : nameModel.count * dataHeight;
property int dataHeight : 50;
property int inputUIHeight : 30;
property int columnSpacing : 10;
property int nCount : 5;
width: 300
height: listViewheight + inputUIHeight + columnSpacing;
visible: true
id:windows
ListModel {
id:nameModel
ListElement {
name: "park"
age : 20
}
ListElement {
name: "kim"
age : 25
}
ListElement {
name: "lee"
age : 30
}
} // 데이터를 담는 Model
Component {
id: nameDelegate
Rectangle {
width : windows.width
height : dataHeight
border.color: "black"
border.width: 1
Text {
anchors.fill: parent
text: "name : " + name + "\n" + "age : " + age;
verticalAlignment: Text.AlignVCenter
anchors.leftMargin: 10
}
}
} // 데이터를 표현하는 Style
Column {
spacing : columnSpacing
width : parent.width
height : parent.height
ListView {
width : parent.width
height : listViewheight
model : nameModel
delegate: nameDelegate
} // 데이터를 보여주는 View
Row {
width : parent.width
height: inputUIHeight
Rectangle {
border.width: 1
width : parent.width/nCount
height : parent.height
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "name"
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
border.color: "black"
border.width: 1
TextInput {
id:nameInput
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
onFocusChanged: {
focus? parent.color = "deepskyblue" : parent.color = "white"
}
}
}
Rectangle {
border.width: 1
width : parent.width/nCount
height : parent.height
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "age"
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
border.color: "black"
border.width: 1
TextInput {
id:ageInput
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
onFocusChanged: {
focus? parent.color = "deepskyblue" : parent.color = "white"
}
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
color: "gray"
border.width: 1
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "input"
}
MouseArea {
anchors.fill: parent
onClicked: {
nameModel.append({ name: nameInput.text,
age : parseInt(ageInput.text)
})
nameInput.text = ""
ageInput.text = ""
nameInput.focus = false
ageInput.focus = false
}
}
}
}
}
}
TextInput이 적용된 Rectangle을 클릭할 시 Focus가 들어가고 그에 따라 Color가 변경됩니다. 데이터를 입력 후 Input 버튼을 누르게 되면 해당 데이터가 리스트에 추가가 됩니다.
다음은 삭제 처리입니다. 삭제를 위해서 클릭한 아이템을 지우는 방식으로 진행했습니다. 어떤 아이템이 클릭되었는지 확인하기 위해서 listView에 highlight component도 추가하였습니다.
import QtQuick 2.15
import QtQuick.Window 2.15
Window {
property int listViewheight : nameModel.count * dataHeight;
property int dataHeight : 50;
property int inputUIHeight : 30;
property int columnSpacing : 10;
property int nCount : 6;
width: 300
height: listViewheight + inputUIHeight + columnSpacing;
visible: true
id:windows
ListModel {
id:nameModel
ListElement {
name: "park"
age : 20
}
ListElement {
name: "kim"
age : 25
}
ListElement {
name: "lee"
age : 30
}
} // 데이터를 담는 Model
Component {
id: nameDelegate
Item {
width : windows.width
height : dataHeight
Text {
anchors.fill: parent
text: "name : " + name + "\n" + "age : " + age;
verticalAlignment: Text.AlignVCenter
anchors.leftMargin: 10
}
MouseArea {
anchors.fill: parent
onClicked: nameList.currentIndex = index
}
}
} // 데이터를 표현하는 Style
Component {
id:listHighLight
Rectangle{
color : "blue"
}
} // Highlight
Column {
spacing : columnSpacing
width : parent.width
height : parent.height
ListView {
id : nameList
width : parent.width
height : listViewheight
model : nameModel
delegate: nameDelegate
highlight: listHighLight
} // 데이터를 보여주는 View
Row {
width : parent.width
height: inputUIHeight
Rectangle {
border.width: 1
width : parent.width/nCount
height : parent.height
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "name"
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
border.color: "black"
border.width: 1
TextInput {
id:nameInput
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
onFocusChanged: {
focus? parent.color = "deepskyblue" : parent.color = "white"
}
}
}
Rectangle {
border.width: 1
width : parent.width/nCount
height : parent.height
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "age"
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
border.color: "black"
border.width: 1
TextInput {
id:ageInput
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
onFocusChanged: {
focus? parent.color = "deepskyblue" : parent.color = "white"
}
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
color: "gray"
border.width: 1
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "input"
}
MouseArea {
anchors.fill: parent
onClicked: {
nameModel.append({ name: nameInput.text,
age : parseInt(ageInput.text)
})
nameInput.text = ""
ageInput.text = ""
nameInput.focus = false
ageInput.focus = false
}
}
}
Rectangle {
width : parent.width/nCount
height : parent.height
color: "red"
border.width: 1
Text {
anchors.fill: parent
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text : "delete"
}
MouseArea {
anchors.fill: parent
onClicked: {
if ( nameModel.count > 0 ) {
nameModel.remove(nameList.currentIndex);
nameList.currentIndex = -1
}
}
}
}
}
}
}
Delegate에 Rectangle 오브젝트를 사용하면 highlight가 제대로 적용되지 않아 Item 오브젝트로 변경하였습니다.