// Copyright (c) 2014-2018, The Monero Project // // All rights reserved. // // Redistribution and use in source and binary forms, with or without modification, are // permitted provided that the following conditions are met: // // 1. Redistributions of source code must retain the above copyright notice, this list of // conditions and the following disclaimer. // // 2. 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. // // 3. Neither the name of the copyright holder 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 HOLDER 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. import QtQuick 2.9 import QtQuick.Controls 1.2 import QtQuick.Layouts 1.2 import QtGraphicalEffects 1.0 import QtQuick.Controls.Styles 1.2 import "." as MoneroComponents import "effects/" as MoneroEffects Item { id: datePicker z: parent.z + 1 property bool expanded: false property date currentDate property bool showCurrentDate: true property color backgroundColor : MoneroComponents.Style.appWindowBorderColor property color errorColor : "red" property bool error: false property alias inputLabel: inputLabel signal dateChanged(); height: 50 onExpandedChanged: if(expanded) appWindow.currentItem = datePicker function hide() { datePicker.expanded = false } function containsPoint(px, py) { if(px < 0) return false if(px > width) return false if(py < 0) return false if(py > height + calendarRect.height) return false return true } Rectangle { id: inputLabelRect color: "transparent" height: 22 width: parent.width MoneroComponents.TextPlain { id: inputLabel anchors.top: parent.top anchors.topMargin: 2 anchors.left: parent.left font.family: MoneroComponents.Style.fontLight.name font.pixelSize: 14 font.bold: false textFormat: Text.RichText color: MoneroComponents.Style.defaultFontColor themeTransition: false MouseArea { anchors.fill: parent acceptedButtons: Qt.NoButton cursorShape: parent.hoveredLink ? Qt.PointingHandCursor : Qt.ArrowCursor } } } Item { id: head anchors.top: inputLabelRect.bottom anchors.topMargin: 6 * scaleRatio anchors.left: parent.left anchors.right: parent.right height: 28 Rectangle { anchors.top: parent.top anchors.left: parent.left anchors.right: parent.right height: parent.height - 1 anchors.leftMargin: datePicker.expanded ? 1 : 0 anchors.rightMargin: datePicker.expanded ? 1 : 0 radius: 4 y: 1 color: datePicker.backgroundColor } RowLayout { id: dateInput anchors.verticalCenter: parent.verticalCenter anchors.left: parent.left anchors.leftMargin: 2 anchors.right: parent.right property string headerFontColor: MoneroComponents.Style.blackTheme ? "#e6e6e6" : "#333333" spacing: 0 function setDate(date) { var day = date.getDate() var month = date.getMonth() + 1 dayInput.text = day < 10 ? "0" + day : day monthInput.text = month < 10 ? "0" + month : month yearInput.text = date.getFullYear() } Connections { target: datePicker onCurrentDateChanged: { dateInput.setDate(datePicker.currentDate) } } TextInput { id: dayInput readOnly: true Layout.preferredWidth: childrenRect.width + 40 font.family: MoneroComponents.Style.fontRegular.name font.pixelSize: 14 color: datePicker.error ? errorColor : parent.headerFontColor selectionColor: MoneroComponents.Style.dimmedFontColor selectByMouse: true horizontalAlignment: TextInput.AlignHCenter maximumLength: 2 validator: IntValidator{bottom: 01; top: 31;} KeyNavigation.tab: monthInput text: { if(datePicker.showCurrentDate) { var day = datePicker.currentDate.getDate() return day < 10 ? "0" + day : day } } onFocusChanged: { if(focus === false) { if(text.length === 0 || text === "0" || text === "00") text = "01" else if(text.length === 1) text = "0" + text } } } MoneroComponents.TextPlain { font.family: MoneroComponents.Style.fontRegular.name font.pixelSize: 14 color: datePicker.error ? errorColor : MoneroComponents.Style.defaultFontColor text: "-" themeTransition: false } TextInput { id: monthInput readOnly: true Layout.preferredWidth: childrenRect.width + 40 font.family: MoneroComponents.Style.fontRegular.name font.pixelSize: 14 color: datePicker.error ? errorColor : parent.headerFontColor selectionColor: MoneroComponents.Style.dimmedFontColor selectByMouse: true horizontalAlignment: TextInput.AlignHCenter maximumLength: 2 validator: IntValidator{bottom: 01; top: 12;} KeyNavigation.tab: yearInput text: { if(datePicker.showCurrentDate) { var month = datePicker.currentDate.getMonth() + 1 return month < 10 ? "0" + month : month } } onFocusChanged: { if(focus === false) { if(text.length === 0 || text === "0" || text === "00") text = "01" else if(text.length === 1) text = "0" + text } } } MoneroComponents.TextPlain { font.family: MoneroComponents.Style.fontRegular.name font.pixelSize: 14 color: datePicker.error ? errorColor : MoneroComponents.Style.defaultFontColor text: "-" themeTransition: false } TextInput { id: yearInput Layout.preferredWidth: childrenRect.width + 60 font.family: MoneroComponents.Style.fontRegular.name font.pixelSize: 14 color: datePicker.error ? errorColor : parent.headerFontColor selectionColor: MoneroComponents.Style.dimmedFontColor selectByMouse: true horizontalAlignment: TextInput.AlignHCenter maximumLength: 4 validator: IntValidator{bottom: 1000; top: 9999;} text: if(datePicker.showCurrentDate) datePicker.currentDate.getFullYear() onFocusChanged: { if(focus === false) { var d = new Date() var y = d.getFullYear() if(text.length != 4 || text[0] === "0") text = y } } } Rectangle { Layout.preferredHeight: parent.height Layout.fillWidth: true color: "transparent" Image { id: button anchors.right: parent.right anchors.rightMargin: 10 * scaleRatio anchors.verticalCenter: parent.verticalCenter source: "qrc:///images/whiteDropIndicator.png" visible: false } ColorOverlay { source: button anchors.fill: button color: MoneroComponents.Style.defaultFontColor rotation: datePicker.expanded ? 180 : 0 opacity: 1 } MouseArea { anchors.fill: parent onClicked: datePicker.expanded = !datePicker.expanded hoverEnabled: true cursorShape: Qt.PointingHandCursor } } } } Rectangle { id: calendarRect anchors.left: parent.left anchors.right: parent.right anchors.top: head.bottom anchors.topMargin: 10 * scaleRatio color: MoneroComponents.Style.middlePanelBackgroundColor border.width: 1 border.color: MoneroComponents.Style.appWindowBorderColor height: datePicker.expanded ? calendar.height + 2 : 0 clip: true Behavior on height { NumberAnimation { duration: 100; easing.type: Easing.InQuad } } Rectangle { anchors.left: parent.left anchors.right: parent.right anchors.leftMargin: 1 anchors.rightMargin: 1 anchors.top: parent.top color: MoneroComponents.Style.appWindowBorderColor height: 1 } Calendar { id: calendar anchors.left: parent.left anchors.right: parent.right anchors.top: parent.top anchors.margins: 1 anchors.bottomMargin: 10 * scaleRatio height: 220 frameVisible: false style: CalendarStyle { gridVisible: false background: Rectangle { color: MoneroComponents.Style.middlePanelBackgroundColor } dayDelegate: Item { z: parent.z + 1 implicitHeight: implicitWidth implicitWidth: calendar.width / 7 Rectangle { id: dayRect anchors.fill: parent radius: parent.implicitHeight / 2 color: { if(dayArea.pressed && styleData.visibleMonth) return MoneroComponents.Style.blackTheme ? "#20FFFFFF" : "#10000000" return "transparent"; } } MoneroComponents.TextPlain { id: dayText anchors.centerIn: parent font.family: MoneroComponents.Style.fontMonoRegular.name font.pixelSize: { if(!styleData.visibleMonth) return 12 return 14 } font.bold: { if(dayArea.pressed || styleData.visibleMonth) return true; return false; } text: styleData.date.getDate() themeTransition: false color: { if(!styleData.visibleMonth) return MoneroComponents.Style.lightGreyFontColor if(dayArea.pressed) return MoneroComponents.Style.defaultFontColor if(styleData.today) return MoneroComponents.Style.orange return MoneroComponents.Style.defaultFontColor } } MouseArea { id: dayArea anchors.fill: parent hoverEnabled: true onEntered: dayRect.color = MoneroComponents.Style.blackTheme ? "#20FFFFFF" : "#10000000" onExited: dayRect.color = "transparent" cursorShape: Qt.PointingHandCursor onClicked: { if(styleData.visibleMonth) { currentDate = styleData.date datePicker.expanded = false } else { var date = styleData.date if(date.getMonth() > calendar.visibleMonth) calendar.showNextMonth() else calendar.showPreviousMonth() } datePicker.dateChanged(); } } } dayOfWeekDelegate: Item { implicitHeight: 20 implicitWidth: calendar.width / 7 MoneroComponents.TextPlain { anchors.centerIn: parent elide: Text.ElideRight font.family: MoneroComponents.Style.fontMonoRegular.name font.pixelSize: 12 color: MoneroComponents.Style.lightGreyFontColor themeTransition: false text: { var locale = Qt.locale() return locale.dayName(styleData.dayOfWeek, Locale.ShortFormat) } } } navigationBar: Rectangle { color: MoneroComponents.Style.middlePanelBackgroundColor implicitWidth: calendar.width implicitHeight: 30 MoneroComponents.TextPlain { anchors.centerIn: parent font.family: MoneroComponents.Style.fontMonoRegular.name font.pixelSize: 14 color: MoneroComponents.Style.dimmedFontColor themeTransition: false text: styleData.title } Item { anchors.left: parent.left anchors.leftMargin: 4 * scaleRatio anchors.top: parent.top anchors.bottom: parent.bottom width: height Image { id: prevMonthIcon anchors.centerIn: parent source: "qrc:///images/prevMonth.png" visible: false } ColorOverlay { source: prevMonthIcon anchors.fill: prevMonthIcon color: MoneroComponents.Style.defaultFontColor opacity: 0.5 } MouseArea { hoverEnabled: true cursorShape: Qt.PointingHandCursor anchors.fill: parent onClicked: calendar.showPreviousMonth() } } Item { anchors.right: parent.right anchors.rightMargin: 4 * scaleRatio anchors.top: parent.top anchors.bottom: parent.bottom width: height Image { id: nextMonthIcon anchors.centerIn: parent source: "qrc:///images/prevMonth.png" visible: false } ColorOverlay { source: nextMonthIcon anchors.fill: nextMonthIcon color: MoneroComponents.Style.defaultFontColor opacity: 0.5 rotation: 180 } MouseArea { hoverEnabled: true cursorShape: Qt.PointingHandCursor anchors.fill: parent onClicked: calendar.showNextMonth() } } } } } } }