FileBrowser.qml Example File

multimedia/video/qmlvideo/qml/qmlvideo/FileBrowser.qml

  /****************************************************************************
  **
  ** Copyright (C) 2016 The Qt Company Ltd.
  ** Contact: https://www.qt.io/licensing/
  **
  ** This file is part of the Qt Mobility Components.
  **
  ** $QT_BEGIN_LICENSE:BSD$
  ** Commercial License Usage
  ** Licensees holding valid commercial Qt licenses may use this file in
  ** accordance with the commercial license agreement provided with the
  ** Software or, alternatively, in accordance with the terms contained in
  ** a written agreement between you and The Qt Company. For licensing terms
  ** and conditions see https://www.qt.io/terms-conditions. For further
  ** information use the contact form at https://www.qt.io/contact-us.
  **
  ** BSD License Usage
  ** Alternatively, 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 The Qt Company Ltd 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$
  **
  ****************************************************************************/

  import QtQuick 2.0
  import Qt.labs.folderlistmodel 2.0

  Rectangle {
      id: fileBrowser
      color: "transparent"

      property string folder

      property int itemHeight: Math.min(parent.width, parent.height) / 15
      property int buttonHeight: Math.min(parent.width, parent.height) / 12

      signal fileSelected(string file)

      function selectFile(file) {
          if (file !== "") {
              folder = loader.item.folders.folder
              fileBrowser.fileSelected(file)
          }
          loader.sourceComponent = undefined
      }

      Loader {
          id: loader
      }

      function show() {
          loader.sourceComponent = fileBrowserComponent
          loader.item.parent = fileBrowser
          loader.item.anchors.fill = fileBrowser
          loader.item.folder = fileBrowser.folder
      }

      Component {
          id: fileBrowserComponent

          Rectangle {
              id: root
              color: "black"
              property bool showFocusHighlight: false
              property variant folders: folders1
              property variant view: view1
              property alias folder: folders1.folder
              property color textColor: "white"

              FolderListModel {
                  id: folders1
                  folder: folder
              }

              FolderListModel {
                  id: folders2
                  folder: folder
              }

              SystemPalette {
                  id: palette
              }

              Component {
                  id: folderDelegate

                  Rectangle {
                      id: wrapper
                      function launch() {
                          var path = "file://";
                          if (filePath.length > 2 && filePath[1] === ':') // Windows drive logic, see QUrl::fromLocalFile()
                              path += '/';
                          path += filePath;
                          if (folders.isFolder(index))
                              down(path);
                          else
                              fileBrowser.selectFile(path)
                      }
                      width: root.width
                      height: folderImage.height
                      color: "transparent"

                      Rectangle {
                          id: highlight
                          visible: false
                          anchors.fill: parent
                          anchors.leftMargin: 5
                          anchors.rightMargin: 5
                          color: "#212121"
                      }

                      Item {
                          id: folderImage
                          width: itemHeight
                          height: itemHeight
                          Image {
                              id: folderPicture
                              source: "qrc:/folder.png"
                              width: itemHeight * 0.9
                              height: itemHeight * 0.9
                              anchors.left: parent.left
                              anchors.margins: 5
                              visible: folders.isFolder(index)
                          }
                      }

                      Text {
                          id: nameText
                          anchors.fill: parent;
                          verticalAlignment: Text.AlignVCenter
                          text: fileName
                          anchors.leftMargin: itemHeight + 10
                          color: (wrapper.ListView.isCurrentItem && root.showFocusHighlight) ? palette.highlightedText : textColor
                          elide: Text.ElideRight
                      }

                      MouseArea {
                          id: mouseRegion
                          anchors.fill: parent
                          onPressed: {
                              root.showFocusHighlight = false;
                              wrapper.ListView.view.currentIndex = index;
                          }
                          onClicked: { if (folders === wrapper.ListView.view.model) launch() }
                      }

                      states: [
                          State {
                              name: "pressed"
                              when: mouseRegion.pressed
                              PropertyChanges { target: highlight; visible: true }
                              PropertyChanges { target: nameText; color: palette.highlightedText }
                          }
                      ]
                  }
              }

              ListView {
                  id: view1
                  anchors.top: titleBar.bottom
                  anchors.bottom: cancelButton.top
                  width: parent.width
                  model: folders1
                  delegate: folderDelegate
                  highlight: Rectangle {
                      color: "#212121"
                      visible: root.showFocusHighlight && view1.count != 0
                      width: view1.currentItem == null ? 0 : view1.currentItem.width
                  }
                  highlightMoveVelocity: 1000
                  pressDelay: 100
                  focus: true
                  state: "current"
                  states: [
                      State {
                          name: "current"
                          PropertyChanges { target: view1; x: 0 }
                      },
                      State {
                          name: "exitLeft"
                          PropertyChanges { target: view1; x: -root.width }
                      },
                      State {
                          name: "exitRight"
                          PropertyChanges { target: view1; x: root.width }
                      }
                  ]
                  transitions: [
                      Transition {
                          to: "current"
                          SequentialAnimation {
                              NumberAnimation { properties: "x"; duration: 250 }
                          }
                      },
                      Transition {
                          NumberAnimation { properties: "x"; duration: 250 }
                          NumberAnimation { properties: "x"; duration: 250 }
                      }
                  ]
                  Keys.onPressed: root.keyPressed(event.key)
              }

              ListView {
                  id: view2
                  anchors.top: titleBar.bottom
                  anchors.bottom: parent.bottom
                  x: parent.width
                  width: parent.width
                  model: folders2
                  delegate: folderDelegate
                  highlight: Rectangle {
                      color: "#212121"
                      visible: root.showFocusHighlight && view2.count != 0
                      width: view1.currentItem == null ? 0 : view1.currentItem.width
                  }
                  highlightMoveVelocity: 1000
                  pressDelay: 100
                  states: [
                      State {
                          name: "current"
                          PropertyChanges { target: view2; x: 0 }
                      },
                      State {
                          name: "exitLeft"
                          PropertyChanges { target: view2; x: -root.width }
                      },
                      State {
                          name: "exitRight"
                          PropertyChanges { target: view2; x: root.width }
                      }
                  ]
                  transitions: [
                      Transition {
                          to: "current"
                          SequentialAnimation {
                              NumberAnimation { properties: "x"; duration: 250 }
                          }
                      },
                      Transition {
                          NumberAnimation { properties: "x"; duration: 250 }
                      }
                  ]
                  Keys.onPressed: root.keyPressed(event.key)
              }

              Rectangle {
                  width: parent.width
                  height: buttonHeight + 10
                  anchors.bottom: parent.bottom
                  color: "black"
              }

              Rectangle {
                  id: cancelButton
                  width: parent.width
                  height: buttonHeight
                  color: "#212121"
                  anchors.bottom: parent.bottom
                  anchors.left: parent.left
                  anchors.right: parent.right
                  anchors.margins: 5
                  radius: buttonHeight / 15

                  Text {
                      anchors.fill: parent
                      text: "Cancel"
                      color: "white"
                      horizontalAlignment: Text.AlignHCenter
                      verticalAlignment: Text.AlignVCenter
                  }

                  MouseArea {
                      anchors.fill: parent
                      onClicked: fileBrowser.selectFile("")
                  }
              }

              Keys.onPressed: {
                  root.keyPressed(event.key);
                  if (event.key === Qt.Key_Return || event.key === Qt.Key_Select || event.key === Qt.Key_Right) {
                      view.currentItem.launch();
                      event.accepted = true;
                  } else if (event.key === Qt.Key_Left) {
                      up();
                  }
              }

              Rectangle {
                  id: titleBar
                  width: parent.width
                  height: buttonHeight + 10
                  anchors.top: parent.top
                  color: "black"

                  Rectangle {
                      width: parent.width;
                      height: buttonHeight
                      color: "#212121"
                      anchors.margins: 5
                      anchors.top: parent.top
                      anchors.left: parent.left
                      anchors.right: parent.right
                      radius: buttonHeight / 15

                      Rectangle {
                          id: upButton
                          width: buttonHeight
                          height: buttonHeight
                          color: "transparent"
                          Image {
                              width: itemHeight
                              height: itemHeight
                              anchors.centerIn: parent
                              source: "qrc:/up.png"
                          }
                          MouseArea { id: upRegion; anchors.centerIn: parent
                              width: buttonHeight
                              height: buttonHeight
                              onClicked: up()
                          }
                          states: [
                              State {
                                  name: "pressed"
                                  when: upRegion.pressed
                                  PropertyChanges { target: upButton; color: palette.highlight }
                              }
                          ]
                      }

                      Text {
                          anchors.left: upButton.right; anchors.right: parent.right; height: parent.height
                          anchors.leftMargin: 5; anchors.rightMargin: 5
                          text: folders.folder
                          color: "white"
                          elide: Text.ElideLeft;
                          horizontalAlignment: Text.AlignLeft;
                          verticalAlignment: Text.AlignVCenter
                      }
                  }
              }

              function down(path) {
                  if (folders == folders1) {
                      view = view2
                      folders = folders2;
                      view1.state = "exitLeft";
                  } else {
                      view = view1
                      folders = folders1;
                      view2.state = "exitLeft";
                  }
                  view.x = root.width;
                  view.state = "current";
                  view.focus = true;
                  folders.folder = path;
              }

              function up() {
                  var path = folders.parentFolder;
                  if (path.toString().length === 0 || path.toString() === 'file:')
                      return;
                  if (folders == folders1) {
                      view = view2
                      folders = folders2;
                      view1.state = "exitRight";
                  } else {
                      view = view1
                      folders = folders1;
                      view2.state = "exitRight";
                  }
                  view.x = -root.width;
                  view.state = "current";
                  view.focus = true;
                  folders.folder = path;
              }

              function keyPressed(key) {
                  switch (key) {
                  case Qt.Key_Up:
                  case Qt.Key_Down:
                  case Qt.Key_Left:
                  case Qt.Key_Right:
                      root.showFocusHighlight = true;
                      break;
                  default:
                      // do nothing
                      break;
                  }
              }
          }
      }
  }