2024-07-05 09:24:40 +02:00
|
|
|
import 'package:flutter/material.dart';
|
2024-07-17 13:28:02 +02:00
|
|
|
import 'package:flutter_flow_chart/flutter_flow_chart.dart';
|
2024-07-05 09:24:40 +02:00
|
|
|
import 'package:go_router/go_router.dart';
|
2024-08-30 12:52:32 +02:00
|
|
|
import 'package:oc_front/core/models/shared_workspace_local.dart';
|
2024-07-17 13:28:02 +02:00
|
|
|
import 'package:oc_front/core/models/workspace_local.dart';
|
|
|
|
import 'package:oc_front/core/sections/header/header.dart';
|
2024-08-30 12:52:32 +02:00
|
|
|
import 'package:oc_front/core/services/specialized_services/shared_service.dart';
|
2024-07-05 09:24:40 +02:00
|
|
|
import 'package:oc_front/core/services/specialized_services/workflow_service.dart';
|
|
|
|
import 'package:oc_front/models/response.dart';
|
2024-07-17 13:28:02 +02:00
|
|
|
import 'package:oc_front/models/search.dart';
|
2024-08-08 08:42:32 +02:00
|
|
|
import 'package:oc_front/models/workflow.dart';
|
2024-07-05 09:24:40 +02:00
|
|
|
import 'package:oc_front/pages/abstract_page.dart';
|
2024-08-30 12:52:32 +02:00
|
|
|
import 'package:oc_front/pages/shared.dart';
|
|
|
|
import 'package:oc_front/widgets/dialog/shallow_creation.dart';
|
|
|
|
import 'package:oc_front/widgets/forms/processing_forms.dart';
|
2024-08-08 08:42:32 +02:00
|
|
|
import 'package:oc_front/widgets/forms/scheduler_forms.dart';
|
2024-07-17 13:28:02 +02:00
|
|
|
import 'package:oc_front/widgets/items/item_row.dart';
|
2024-08-30 12:52:32 +02:00
|
|
|
import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart';
|
2024-08-08 08:42:32 +02:00
|
|
|
|
|
|
|
Dashboard dash = Dashboard(
|
|
|
|
name: "workflow_${DateTime.now().toString().replaceAll(" ", "_").substring(0, DateTime.now().toString().length - 7)}");
|
2024-07-05 09:24:40 +02:00
|
|
|
class WorkflowFactory implements AbstractFactory {
|
|
|
|
static GlobalKey<WorkflowPageWidgetState> key = GlobalKey<WorkflowPageWidgetState>();
|
|
|
|
@override bool searchFill() { return false; }
|
2024-08-30 12:52:32 +02:00
|
|
|
@override Widget factory(GoRouterState state, List<String> args) {
|
|
|
|
String? id;
|
|
|
|
try { id = state.pathParameters[args.first];
|
|
|
|
} catch (e) { }
|
|
|
|
return WorkflowPageWidget(id: id);
|
|
|
|
}
|
2024-07-05 09:24:40 +02:00
|
|
|
@override void search(BuildContext context) { }
|
|
|
|
}
|
2024-08-22 15:46:16 +02:00
|
|
|
bool getAll = true;
|
2024-07-05 09:24:40 +02:00
|
|
|
|
|
|
|
class WorkflowPageWidget extends StatefulWidget {
|
2024-08-30 12:52:32 +02:00
|
|
|
String? id;
|
|
|
|
WorkflowPageWidget ({ this.id }) : super(key: WorkflowFactory.key);
|
2024-07-05 09:24:40 +02:00
|
|
|
@override WorkflowPageWidgetState createState() => WorkflowPageWidgetState();
|
|
|
|
static void search(BuildContext context) { }
|
|
|
|
static Widget factory() { return WorkflowPageWidget(); }
|
|
|
|
}
|
|
|
|
class WorkflowPageWidgetState extends State<WorkflowPageWidget> {
|
2024-07-17 13:28:02 +02:00
|
|
|
final WorflowService _service = WorflowService();
|
|
|
|
Widget itemBuild(Object item) {
|
|
|
|
var e = item as AbstractItem;
|
2024-08-08 08:42:32 +02:00
|
|
|
return Tooltip( message: item.name ?? "",
|
|
|
|
child: e.logo != null ? Image.network(e.logo ?? "", fit: BoxFit.fill)
|
|
|
|
: Image.network('https://get-picto.com/wp-content/uploads/2024/01/logo-instagram-png.webp',
|
|
|
|
fit: BoxFit.fill));
|
2024-07-17 13:28:02 +02:00
|
|
|
}
|
|
|
|
Widget itemTooltipBuild(Object item) {
|
|
|
|
var e = item as AbstractItem;
|
2024-08-30 12:52:32 +02:00
|
|
|
return Container(color: Colors.white, child: ItemRowWidget(low: true, contextWidth: 290, item: e));
|
2024-07-17 13:28:02 +02:00
|
|
|
}
|
2024-08-08 08:42:32 +02:00
|
|
|
List<Widget> getForms(FlowData? obj) {
|
|
|
|
return obj == null ? [] : [
|
2024-08-30 12:52:32 +02:00
|
|
|
ProcessingFormsWidget(item: obj as AbstractItem, dash: dash),
|
2024-08-08 08:42:32 +02:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
List<Widget> getDashInfoForms() {
|
|
|
|
return [
|
|
|
|
SchedulerFormsWidget(item: dash),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> loadDash(String selected) async {
|
2024-08-30 12:52:32 +02:00
|
|
|
dash.shouldSave = false;
|
2024-08-08 08:42:32 +02:00
|
|
|
if (selected.split("~").length > 1) {
|
|
|
|
dash.name = selected.split("~")[1];
|
|
|
|
dash.id = selected.split("~")[0];
|
|
|
|
} else {
|
|
|
|
dash.name = selected;
|
|
|
|
}
|
|
|
|
await _service.get(context, dash.id ?? "").then((value) {
|
|
|
|
if (value.data != null) {
|
2024-08-30 12:52:32 +02:00
|
|
|
dash.clear();
|
2024-08-08 08:42:32 +02:00
|
|
|
dash.deserialize(value.data!.toDashboard());
|
2024-08-30 12:52:32 +02:00
|
|
|
Future.delayed(const Duration(seconds: 1), () => dash.shouldSave = true);
|
2024-08-08 08:42:32 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> saveDash(String? id) async {
|
2024-08-30 12:52:32 +02:00
|
|
|
if (id == null || !dash.isOpened || !dash.shouldSave) { return; }
|
2024-08-08 08:42:32 +02:00
|
|
|
var datas = WorkspaceLocal.byTopic("data", true).where(
|
|
|
|
(element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) );
|
|
|
|
var dataCenter = WorkspaceLocal.byTopic("datacenter", true).where(
|
|
|
|
(element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) );
|
|
|
|
var storage = WorkspaceLocal.byTopic("storage", true).where(
|
|
|
|
(element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) );
|
|
|
|
var computing = WorkspaceLocal.byTopic("processing", true).where(
|
|
|
|
(element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) );
|
|
|
|
var workflows = WorkspaceLocal.byTopic("workflows", true).where(
|
|
|
|
(element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) );
|
|
|
|
var updateW = Workflow(
|
|
|
|
name: dash.name,
|
|
|
|
graph: Graph(),
|
|
|
|
data: datas.map((e) => e.id).toSet().toList(),
|
|
|
|
datacenter: dataCenter.map((e) => e.id).toSet().toList(),
|
|
|
|
storage: storage.map((e) => e.id).toSet().toList(),
|
|
|
|
processing: computing.map((e) => e.id).toSet().toList(),
|
|
|
|
workflows: workflows.map((e) => e.id).toSet().toList(),
|
|
|
|
);
|
|
|
|
updateW.fromDashboard(dash.serialize());
|
|
|
|
for (var item in (updateW.graph?.items.values ?? [] as List<GraphItem>)) {
|
|
|
|
if (item.position == null) { continue; }
|
2024-08-30 12:52:32 +02:00
|
|
|
item.position?.x = (item.position?.x ?? 0) + (item.width! / 2) + 7.5;
|
|
|
|
item.position?.y = (item.position?.y ?? 0) + (item.height! / 2) + 7.5;
|
|
|
|
for (var i in (updateW.graph?.links ?? [] as List<GraphLink>).where((element) => id == element.source?.id)) {
|
|
|
|
i.source?.x = (i.source?.x ?? 0) + (item.width! / 2) + 7;
|
|
|
|
i.source?.y = (i.source?.y ?? 0) + (item.width! / 2) + 7;
|
|
|
|
}
|
|
|
|
for (var i in (updateW.graph?.links ?? [] as List<GraphLink>).where((element) => id == element.destination?.id)) {
|
|
|
|
i.destination?.x = (i.destination?.x ?? 0) + (item.width! / 2) + 7.5;
|
|
|
|
i.destination?.y = (i.destination?.y ?? 0) + (item.width! / 2) + 7.5;
|
|
|
|
}
|
2024-08-08 08:42:32 +02:00
|
|
|
}
|
|
|
|
updateW.graph?.zoom = dash.getZoomFactor();
|
2024-08-30 12:52:32 +02:00
|
|
|
print("SAVE DASH");
|
|
|
|
dash.addToHistory();
|
|
|
|
await _service.put(context, id, updateW.serialize(), {}).then( (e) {
|
|
|
|
if (dash.addChange) {
|
|
|
|
dash.addChange = false;
|
|
|
|
WorkspaceLocal.init(context, false);
|
|
|
|
dash.selectedLeftMenuKey.currentState?.setState(() { });
|
|
|
|
}
|
|
|
|
});
|
2024-08-08 08:42:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
FlowData? transformToData(Map<String, dynamic> data) {
|
|
|
|
var d = WorkspaceLocal.getItem(data["id"] ?? "", true);
|
|
|
|
if (d == null) { return null; }
|
|
|
|
d.model = ResourceModel().deserialize(data["resource_model"]);
|
|
|
|
if (d.topic == "data") { return d as DataItem; }
|
|
|
|
if (d.topic == "datacenter") { return d as DataCenterItem; }
|
|
|
|
if (d.topic == "storage") { return d as StorageItem; }
|
|
|
|
if (d.topic == "processing") { return d as ProcessingItem; }
|
|
|
|
if (d.topic == "workflows") { return d as WorkflowItem; }
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-08-30 12:52:32 +02:00
|
|
|
Widget onDashboardMenu(Dashboard dash) {
|
|
|
|
return ShallowDropdownInputWidget(
|
|
|
|
iconLoad: Icons.share,
|
|
|
|
tooltipLoad: 'share',
|
|
|
|
tooltipRemove: 'unshare',
|
|
|
|
filled: const Color.fromRGBO(38,166, 154, 1),
|
|
|
|
color: Colors.white,
|
|
|
|
hintColor: Colors.grey.shade300,
|
|
|
|
type: SharedWorkspaceType.workflow,
|
|
|
|
all: () async => SharedWorkspaceLocal.workspaces.values.map(
|
|
|
|
(e) => Shallow(id: e.id ?? "", name: e.name ?? "") ).toList(),
|
|
|
|
current: WorkspaceLocal.workspaces[WorkspaceLocal.current]?.shared,
|
|
|
|
width: (MediaQuery.of(context).size.width / 3),
|
|
|
|
canLoad: (String? change) {
|
|
|
|
return SharedWorkspaceLocal.workspaces[change] == null
|
|
|
|
|| !SharedWorkspaceLocal.workspaces[change]!.workflows.map( (e) => e.id
|
|
|
|
).contains(dash.id);
|
|
|
|
},
|
|
|
|
canRemove: (String? change) => SharedWorkspaceLocal.workspaces[change] == null
|
|
|
|
|| SharedWorkspaceLocal.workspaces[change]!.workflows.map( (e) => e.id
|
|
|
|
).contains(dash.id),
|
|
|
|
load: (String val) async {
|
|
|
|
await SharedService().addWorkflow(context, val, dash.id ?? "");
|
|
|
|
SharedWorkspaceLocal.init(context, false);
|
|
|
|
dash.selectedMenuKey.currentState?.setState(() { });
|
|
|
|
},
|
|
|
|
remove: (String val) async {
|
|
|
|
await SharedService().removeWorkflow(context, val, dash.id ?? "");
|
|
|
|
SharedWorkspaceLocal.init(context, false);
|
|
|
|
dash.selectedMenuKey.currentState?.setState(() { });
|
|
|
|
});
|
|
|
|
}
|
2024-08-08 08:42:32 +02:00
|
|
|
Widget menuExtension() {
|
|
|
|
var quart = MediaQuery.of(context).size.width / 6;
|
2024-08-30 12:52:32 +02:00
|
|
|
return ShallowDropdownInputWidget(
|
|
|
|
current: WorkspaceLocal.current,
|
|
|
|
width: quart > 80 ? quart : 80,
|
|
|
|
all: () async => WorkspaceLocal.getWorkspacesShallow(),
|
|
|
|
type: SharedWorkspaceType.workspace,
|
|
|
|
change: (String? change) {
|
|
|
|
WorkspaceLocal.changeWorkspace(change.toString());
|
|
|
|
}
|
|
|
|
);
|
2024-08-27 15:38:21 +02:00
|
|
|
}
|
2024-07-17 13:28:02 +02:00
|
|
|
Widget onDashboardAlertOpened(BuildContext context, Dashboard dash) {
|
2024-08-30 12:52:32 +02:00
|
|
|
return ShallowCreationDialogWidget(
|
|
|
|
canClose: () => dash.isOpened,
|
|
|
|
context: context,
|
|
|
|
load: (p0) async {
|
|
|
|
dash.isOpened = true;
|
|
|
|
if (dash.load != null) {
|
|
|
|
WorkspaceLocal.changeWorkspaceByName(p0.split("~")[1]);
|
|
|
|
await dash.load!(p0);
|
|
|
|
}
|
|
|
|
dash.notifyListeners();
|
|
|
|
},
|
|
|
|
create: (p0) async => await _service.post(context, p0, {}).then( (value) {
|
|
|
|
dash.clear();
|
|
|
|
dash.id = value.data?.getID() ?? "";
|
|
|
|
dash.name = value.data?.getName() ?? "";
|
|
|
|
dash.notifyListeners();
|
|
|
|
WorkspaceLocal.init(context, true);
|
|
|
|
dash.isOpened = true;
|
|
|
|
}
|
|
|
|
),
|
|
|
|
maptoDropdown: (e) => DropdownMenuItem<String>(
|
|
|
|
value: "${e.id}~${e.name}",
|
|
|
|
child: Text(e.name),
|
|
|
|
),
|
|
|
|
type: SharedWorkspaceType.workflow,
|
|
|
|
all: () async {
|
|
|
|
List<Shallow> res = [];
|
|
|
|
await _service.all(context).then(
|
|
|
|
(e) {
|
|
|
|
if (e.data != null) {
|
|
|
|
res = e.data!.values.map((e) => Shallow(id: e["id"] ?? "", name: e["name"] ?? "")).toList();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
);
|
2024-07-17 13:28:02 +02:00
|
|
|
}
|
|
|
|
@override Widget build(BuildContext context) {
|
2024-08-08 08:42:32 +02:00
|
|
|
dash.load = loadDash;
|
|
|
|
dash.save = saveDash;
|
|
|
|
dash.transformToData = transformToData;
|
|
|
|
dash.infoItemWidget = getForms;
|
|
|
|
dash.infoWidget = getDashInfoForms;
|
2024-07-17 13:28:02 +02:00
|
|
|
var quart = MediaQuery.of(context).size.width / 6;
|
2024-08-30 12:52:32 +02:00
|
|
|
dash.defaultName = "workflow_${DateTime.now().toString().replaceAll(" ", "_"
|
|
|
|
).substring(0, DateTime.now().toString().length - 7)}";
|
2024-07-17 13:28:02 +02:00
|
|
|
return FlowChart<AbstractItem>(
|
|
|
|
onDashboardAlertOpened: onDashboardAlertOpened,
|
2024-07-17 17:45:39 +02:00
|
|
|
dashboard: dash,
|
2024-08-30 12:52:32 +02:00
|
|
|
current: widget.id,
|
2024-07-17 13:28:02 +02:00
|
|
|
itemWidget: itemBuild,
|
2024-08-27 15:38:21 +02:00
|
|
|
menuWidget: onDashboardMenu,
|
2024-08-08 08:42:32 +02:00
|
|
|
categories: const ["processing", "data", "datacenter", "storage", "workflows"],
|
2024-08-22 15:46:16 +02:00
|
|
|
draggableItemBuilder: (cat) => WorkspaceLocal.byTopic(cat, false),
|
2024-07-17 13:28:02 +02:00
|
|
|
itemWidgetTooltip: itemTooltipBuild,
|
|
|
|
innerMenuWidth: quart > 80 ? quart : 80,
|
|
|
|
width: MediaQuery.of(context).size.width,
|
|
|
|
height: MediaQuery.of(context).size.height - HeaderConstants.height,
|
|
|
|
onNewConnection: (p1, p2) { },
|
2024-08-30 12:52:32 +02:00
|
|
|
menuExtension: menuExtension,
|
2024-07-17 13:28:02 +02:00
|
|
|
onDashboardTapped: (context, position) { },
|
|
|
|
onScaleUpdate: (newScale) { },
|
|
|
|
onDashboardSecondaryTapped: (context, position) { },
|
|
|
|
onDashboardLongTapped: (context, position) { },
|
|
|
|
onDashboardSecondaryLongTapped: (context, position) { },
|
|
|
|
onElementLongPressed: (context, position, element) { },
|
|
|
|
onElementSecondaryLongTapped: (context, position, element) { },
|
|
|
|
onElementPressed: (context, position, element) { },
|
|
|
|
onElementSecondaryTapped: (context, position, element) { },
|
|
|
|
onHandlerPressed: (context, position, handler, element) { },
|
|
|
|
onHandlerLongPressed: (context, position, handler, element) { },
|
|
|
|
onPivotSecondaryPressed: (context, pivot) { },
|
2024-07-17 17:45:39 +02:00
|
|
|
);
|
2024-07-05 09:24:40 +02:00
|
|
|
}
|
|
|
|
}
|