2 * Copyright (C) 2013,2015 Canonical, Ltd.
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; version 3.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 import Ubuntu.Components 1.3
19 import Ubuntu.Components.Themes.Ambiance 1.3
20 import Ubuntu.Components.Popups 1.3
21 import Ubuntu.Components.ListItems 1.3
22 import "../Components"
23 import "../Components/SearchHistoryModel"
27 objectName: "pageHeader"
28 implicitHeight: headerContainer.height + bottomContainer.height + (showSignatureLine ? units.gu(2) : 0)
30 property bool showBackButton: false
31 property bool backIsClose: false
34 property bool storeEntryEnabled: false
35 property bool searchEntryEnabled: false
36 property bool settingsEnabled: false
37 property bool favoriteEnabled: false
38 property bool favorite: false
39 property ListModel searchHistory: SearchHistoryModel
40 property alias searchQuery: searchTextField.text
41 property alias searchHint: searchTextField.placeholderText
42 property bool showSignatureLine: true
44 property alias bottomItem: bottomContainer.children
45 property int paginationCount: 0
46 property int paginationIndex: -1
48 property var scopeStyle: null
52 signal settingsClicked()
53 signal favoriteClicked()
54 signal searchTextFieldFocused()
56 onScopeStyleChanged: refreshLogo()
57 onSearchQueryChanged: {
58 // Make sure we are at the search page if the search query changes behind our feet
60 headerContainer.showSearch = true;
64 function triggerSearch() {
65 if (searchEntryEnabled) {
66 headerContainer.showSearch = true;
67 searchTextField.forceActiveFocus();
71 function closePopup(keepFocus) {
72 if (headerContainer.popover != null) {
73 headerContainer.popover.unfocusOnDestruction = !keepFocus;
74 PopupUtils.close(headerContainer.popover);
75 } else if (!keepFocus) {
80 function resetSearch(keepFocus) {
82 searchHistory.addQuery(searchTextField.text);
84 searchTextField.text = "";
85 closePopup(keepFocus);
89 searchTextField.focus = false;
90 if (!searchTextField.text) {
91 headerContainer.showSearch = false;
95 function openSearchHistory() {
96 if (openSearchAnimation.running) {
97 openSearchAnimation.openSearchHistory = true;
98 } else if (root.searchHistory.count > 0 && headerContainer.popover == null) {
99 headerContainer.popover = PopupUtils.open(popoverComponent, searchTextField,
101 "contentWidth": searchTextField.width,
102 "edgeMargins": units.gu(1)
108 function refreshLogo() {
109 if (root.scopeStyle ? root.scopeStyle.headerLogo != "" : false) {
110 header.contents = imageComponent.createObject();
111 } else if (header.contents) {
112 header.contents.destroy();
113 header.contents = null;
118 target: root.scopeStyle
119 onHeaderLogoChanged: root.refreshLogo()
123 anchors { fill: parent; margins: units.gu(1); bottomMargin: units.gu(3) + bottomContainer.height }
124 visible: headerContainer.showSearch
126 closePopup(/* keepFocus */false);
127 if (!searchTextField.text) {
128 headerContainer.showSearch = false;
130 mouse.accepted = false;
136 objectName: "headerContainer"
137 clip: contentY < height
138 anchors { left: parent.left; top: parent.top; right: parent.right }
139 height: header.contentHeight
140 contentHeight: headersColumn.height
142 contentY: showSearch ? 0 : height
144 property bool showSearch: false
145 property var popover: null
149 objectName: "headerBackground"
150 style: scopeStyle.headerBackground
153 Behavior on contentY {
154 UbuntuNumberAnimation {
155 id: openSearchAnimation
156 property bool openSearchHistory: false
159 if (!running && openSearchAnimation.openSearchHistory) {
160 openSearchAnimation.openSearchHistory = false;
161 root.openSearchHistory();
169 anchors { left: parent.left; right: parent.right }
173 anchors { left: parent.left; right: parent.right }
174 opacity: headerContainer.clip || headerContainer.showSearch ? 1 : 0 // setting visible false cause column to relayout
175 __separator_visible: false
176 // Required to keep PageHeadStyle noise down as it expects the Page's properties around.
177 property var styledItem: searchHeader
178 property color dividerColor: "transparent" // Doesn't matter as we don't have PageHeadSections
179 property color panelColor: background.topColor
180 panelForegroundColor: config.foregroundColor
181 config: PageHeadConfiguration {
182 foregroundColor: root.scopeStyle ? root.scopeStyle.headerForeground : theme.palette.normal.baseText
187 headerContainer.showSearch = false;
191 property var contents: TextField {
193 objectName: "searchTextField"
194 inputMethodHints: Qt.ImhNoPredictiveText
195 hasClearButton: false
198 leftMargin: units.gu(1)
199 topMargin: units.gu(1)
200 bottomMargin: units.gu(1)
201 rightMargin: root.width > units.gu(60) ? root.width - units.gu(40) : units.gu(1)
204 secondaryItem: AbstractButton {
205 height: searchTextField.height
207 enabled: searchTextField.text.length > 0
210 objectName: "clearIcon"
212 anchors.margins: units.gu(.75)
213 source: "image://theme/clear"
214 opacity: searchTextField.text.length > 0
216 Behavior on opacity {
217 UbuntuNumberAnimation { duration: UbuntuAnimation.FastDuration }
222 root.resetSearch(true);
223 root.openSearchHistory();
227 onActiveFocusChanged: {
229 root.searchTextFieldFocused();
230 root.openSearchHistory();
236 closePopup(/* keepFocus */true);
244 objectName: "innerPageHeader"
245 anchors { left: parent.left; right: parent.right }
246 height: headerContainer.height
247 opacity: headerContainer.clip || !headerContainer.showSearch ? 1 : 0 // setting visible false cause column to relayout
248 __separator_visible: false
249 property var styledItem: header
250 property color dividerColor: "transparent" // Doesn't matter as we don't have PageHeadSections
251 property color panelColor: background.topColor
252 panelForegroundColor: config.foregroundColor
253 config: PageHeadConfiguration {
255 foregroundColor: root.scopeStyle ? root.scopeStyle.headerForeground : theme.palette.normal.baseText
257 iconName: backIsClose ? "close" : "back"
258 visible: root.showBackButton
259 onTriggered: root.backClicked()
265 text: i18n.ctr("Button: Open the Ubuntu Store", "Store")
266 iconName: "ubuntu-store-symbolic"
267 visible: root.storeEntryEnabled
268 onTriggered: root.storeClicked();
272 text: i18n.ctr("Button: Start a search in the current dash scope", "Search")
274 visible: root.searchEntryEnabled
276 headerContainer.showSearch = true;
277 searchTextField.forceActiveFocus();
281 objectName: "settings"
282 text: i18n.ctr("Button: Show the current dash scope settings", "Settings")
284 visible: root.settingsEnabled
285 onTriggered: root.settingsClicked()
288 objectName: "favorite"
289 text: root.favorite ? i18n.tr("Remove from Favorites") : i18n.tr("Add to Favorites")
290 iconName: root.favorite ? "starred" : "non-starred"
291 visible: root.favoriteEnabled
292 onTriggered: root.favoriteClicked()
297 property var contents: null
298 Component.onCompleted: root.refreshLogo()
304 anchors { fill: parent; topMargin: units.gu(1.5); bottomMargin: units.gu(1.5) }
307 objectName: "titleImage"
309 source: root.scopeStyle ? root.scopeStyle.headerLogo : ""
310 fillMode: Image.PreserveAspectFit
311 horizontalAlignment: Image.AlignLeft
312 sourceSize.height: height
326 property bool unfocusOnDestruction: false
328 Component.onDestruction: {
329 headerContainer.popover = null;
330 if (unfocusOnDestruction) {
344 objectName: "recentSearches"
348 showDivider: index < recentSearches.count - 1
351 searchHistory.addQuery(text);
352 searchTextField.text = text;
353 closePopup(/* keepFocus */false);
363 visible: showSignatureLine
365 top: headerContainer.bottom
368 bottom: bottomContainer.top
371 color: root.scopeStyle ? root.scopeStyle.headerDividerColor : "#e0e0e0"
380 color: Qt.darker(parent.color, 1.1)
385 visible: bottomBorder.visible
386 spacing: units.gu(.5)
388 objectName: "paginationRepeater"
389 model: root.paginationCount
391 objectName: "paginationDots_" + index
394 source: (index == root.paginationIndex) ? "graphics/pagination_dot_on.png" : "graphics/pagination_dot_off.png"
398 top: headerContainer.bottom
399 horizontalCenter: headerContainer.horizontalCenter
400 topMargin: units.gu(.5)
404 // FIXME this doesn't work with solid scope backgrounds due to z-ordering
407 visible: bottomBorder.visible
409 top: bottomContainer.top
417 // FIXME this should be a shader when bottomItem exists
418 // to support image backgrounds
421 color: if (bottomItem && bottomItem.background) {
422 Qt.lighter(Qt.rgba(bottomItem.background.topColor.r,
423 bottomItem.background.topColor.g,
424 bottomItem.background.topColor.b, 1.0), 1.2);
425 } else if (!bottomItem && root.scopeStyle) {
426 Qt.lighter(Qt.rgba(root.scopeStyle.background.r,
427 root.scopeStyle.background.g,
428 root.scopeStyle.background.b, 1.0), 1.2);
439 bottom: parent.bottom
441 height: childrenRect.height