2023-08-01 13:01:01 +00:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
|
|
|
import QtQuick
|
|
|
|
import QtQuick.Controls
|
|
|
|
import QtQuick.Layouts
|
2023-08-09 14:25:15 +00:00
|
|
|
import QtQuick.Dialogs
|
2023-08-01 13:01:01 +00:00
|
|
|
import Fk
|
|
|
|
|
|
|
|
Item {
|
|
|
|
id: root
|
|
|
|
|
|
|
|
ToolBar {
|
|
|
|
id: bar
|
|
|
|
width: parent.width
|
|
|
|
RowLayout {
|
|
|
|
anchors.fill: parent
|
|
|
|
ToolButton {
|
|
|
|
icon.source: AppPath + "/image/modmaker/back"
|
|
|
|
onClicked: mainStack.pop();
|
|
|
|
}
|
|
|
|
Label {
|
|
|
|
text: Backend.translate("Replay Manager")
|
|
|
|
horizontalAlignment: Qt.AlignHCenter
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
ToolButton {
|
|
|
|
icon.source: AppPath + "/image/modmaker/menu"
|
|
|
|
onClicked: menu.open()
|
|
|
|
|
|
|
|
Menu {
|
|
|
|
id: menu
|
|
|
|
y: bar.height
|
2023-08-09 14:25:15 +00:00
|
|
|
MenuItem {
|
|
|
|
text: qsTr("Replay from file")
|
|
|
|
onTriggered: {
|
|
|
|
fdialog.open();
|
|
|
|
}
|
|
|
|
}
|
2023-08-01 13:01:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Rectangle {
|
|
|
|
width: parent.width
|
|
|
|
height: parent.height - bar.height
|
|
|
|
anchors.top: bar.bottom
|
2023-08-02 13:40:00 +00:00
|
|
|
color: "#A0EFEFEF"
|
2023-08-01 13:01:01 +00:00
|
|
|
clip: true
|
|
|
|
|
|
|
|
ListView {
|
|
|
|
id: list
|
|
|
|
clip: true
|
|
|
|
anchors.fill: parent
|
|
|
|
model: ListModel {
|
|
|
|
id: model
|
|
|
|
}
|
|
|
|
delegate: Item {
|
|
|
|
width: root.width
|
|
|
|
height: 64
|
|
|
|
|
|
|
|
Image {
|
|
|
|
id: generalPic
|
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.left: parent.left
|
|
|
|
anchors.margins: 8
|
|
|
|
width: 48
|
|
|
|
height: 48
|
|
|
|
source: SkinBank.getGeneralExtraPic(general, "avatar/") ?? SkinBank.getGeneralPicture(general)
|
|
|
|
sourceClipRect: sourceSize.width > 200 ? Qt.rect(61, 0, 128, 128) : undefined
|
|
|
|
|
|
|
|
Rectangle {
|
|
|
|
anchors.fill: parent
|
|
|
|
color: "transparent"
|
|
|
|
border.width: 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ColumnLayout {
|
|
|
|
anchors.left: generalPic.right
|
|
|
|
anchors.margins: 8
|
|
|
|
Text {
|
|
|
|
text: {
|
|
|
|
const win = winner.split("+").indexOf(role) !== -1;
|
|
|
|
const winStr = win ? Backend.translate("Game Win") : Backend.translate("Game Lose");
|
|
|
|
return "<b>" + Backend.translate(general) + "</b> " + Backend.translate(role) + " " + winStr;
|
|
|
|
}
|
|
|
|
font.pixelSize: 20
|
|
|
|
textFormat: Text.RichText
|
|
|
|
}
|
|
|
|
Text {
|
|
|
|
text: {
|
|
|
|
const y = repDate.slice(0,4);
|
|
|
|
const month = repDate.slice(4,6);
|
|
|
|
const d = repDate.slice(6,8);
|
|
|
|
const h = repDate.slice(8,10);
|
|
|
|
const m = repDate.slice(10,12);
|
|
|
|
const s = repDate.slice(12,14);
|
|
|
|
const dateStr = y + "-" + month + "-" + d + " " + h + ":" + m + ":" + s;
|
|
|
|
|
|
|
|
return playerName + " " + Backend.translate(gameMode) + " " + dateStr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Button {
|
|
|
|
id: replayBtn
|
|
|
|
text: Backend.translate("Play the Replay")
|
|
|
|
anchors.right: delBtn.left
|
|
|
|
anchors.rightMargin: 8
|
|
|
|
onClicked: {
|
|
|
|
config.observing = true;
|
|
|
|
config.replaying = true;
|
2023-08-09 14:25:15 +00:00
|
|
|
Backend.playRecord("recording/" + fileName);
|
2023-08-01 13:01:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Button {
|
|
|
|
id: delBtn
|
|
|
|
text: Backend.translate("Delete Replay")
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.rightMargin: 8
|
|
|
|
onClicked: {
|
|
|
|
Backend.removeRecord(fileName);
|
|
|
|
removeModel(index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-09 14:25:15 +00:00
|
|
|
FileDialog {
|
|
|
|
id: fdialog
|
|
|
|
nameFilters: ["FK Rep Files (*.fk.rep)"];
|
|
|
|
onAccepted: {
|
|
|
|
config.observing = true;
|
|
|
|
config.replaying = true;
|
|
|
|
let str = selectedFile.toString(); // QUrl -> string
|
|
|
|
Backend.playRecord(str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-01 13:01:01 +00:00
|
|
|
function updateList() {
|
|
|
|
model.clear();
|
|
|
|
const data = Backend.ls("recording");
|
|
|
|
data.reverse();
|
|
|
|
data.forEach(s => {
|
|
|
|
const d = s.split(".");
|
|
|
|
if (d.length !== 8) return;
|
|
|
|
// s: <time>.<screenName>.<mode>.<general>.<role>.<winner>.fk.rep
|
|
|
|
const [t, name, mode, general, role, winner] = d;
|
|
|
|
|
|
|
|
model.append({
|
|
|
|
fileName: s,
|
|
|
|
repDate: t,
|
|
|
|
playerName: name,
|
|
|
|
gameMode: mode,
|
|
|
|
general: general,
|
|
|
|
role: role,
|
|
|
|
winner: winner,
|
|
|
|
})
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function removeModel(index) {
|
|
|
|
model.remove(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
Component.onCompleted: {
|
|
|
|
updateList();
|
|
|
|
}
|
|
|
|
}
|