Merge remote-tracking branch 'origin/feature/namespace' into main
This commit is contained in:
		| @@ -3,193 +3,193 @@ | |||||||
|   "packages": [ |   "packages": [ | ||||||
|     { |     { | ||||||
|       "name": "alert_banner", |       "name": "alert_banner", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/alert_banner-1.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/alert_banner-1.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "animated_toggle_switch", |       "name": "animated_toggle_switch", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/animated_toggle_switch-0.8.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/animated_toggle_switch-0.8.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "args", |       "name": "args", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/args-2.6.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/args-2.6.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "async", |       "name": "async", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/async-2.11.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/async-2.11.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "boolean_selector", |       "name": "boolean_selector", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "box_transform", |       "name": "box_transform", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/box_transform-0.4.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/box_transform-0.4.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "characters", |       "name": "characters", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/characters-1.3.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/characters-1.3.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "clock", |       "name": "clock", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/clock-1.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/clock-1.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "collection", |       "name": "collection", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/collection-1.19.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/collection-1.18.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.4" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "convert", |       "name": "convert", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/convert-3.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/convert-3.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "cron", |       "name": "cron", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/cron-0.6.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/cron-0.6.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "crypto", |       "name": "crypto", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/crypto-3.0.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/crypto-3.0.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "cupertino_icons", |       "name": "cupertino_icons", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/cupertino_icons-1.0.8", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/cupertino_icons-1.0.8", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.1" |       "languageVersion": "3.1" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dart_earcut", |       "name": "dart_earcut", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dart_earcut-1.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dart_earcut-1.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dashed_path", |       "name": "dashed_path", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dashed_path-1.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dashed_path-1.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "datetime_picker_formfield", |       "name": "datetime_picker_formfield", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/datetime_picker_formfield-2.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/datetime_picker_formfield-2.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "desktop_window", |       "name": "desktop_window", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/desktop_window-0.4.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/desktop_window-0.4.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "device_info_plus", |       "name": "device_info_plus", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "device_info_plus_platform_interface", |       "name": "device_info_plus_platform_interface", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus_platform_interface-7.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/device_info_plus_platform_interface-7.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dio", |       "name": "dio", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dio-5.7.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dio-5.7.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dio_web_adapter", |       "name": "dio_web_adapter", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dio_web_adapter-2.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dio_web_adapter-2.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dotted_line", |       "name": "dotted_line", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dotted_line-3.2.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dotted_line-3.2.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "el_tooltip", |       "name": "el_tooltip", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/el_tooltip-2.2.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/el_tooltip-2.2.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "fake_async", |       "name": "fake_async", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/fake_async-1.3.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/fake_async-1.3.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "ffi", |       "name": "ffi", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/ffi-2.1.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/ffi-2.1.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "file", |       "name": "file", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/file-7.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/file-7.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "fixnum", |       "name": "fixnum", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/fixnum-1.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/fixnum-1.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.1" |       "languageVersion": "3.1" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter", |       "name": "flutter", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/packages/flutter", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/packages/flutter", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_advanced_switch", |       "name": "flutter_advanced_switch", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_advanced_switch-3.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_advanced_switch-3.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_box_transform", |       "name": "flutter_box_transform", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.6", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.6", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_colorpicker", |       "name": "flutter_colorpicker", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.14" |       "languageVersion": "2.14" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_event_calendar", |       "name": "flutter_event_calendar", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_event_calendar-1.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_event_calendar-1.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
| @@ -201,523 +201,523 @@ | |||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_lints", |       "name": "flutter_lints", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_lints-4.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_lints-4.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.1" |       "languageVersion": "3.1" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_map", |       "name": "flutter_map", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_map-7.0.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_map-7.0.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_spinkit", |       "name": "flutter_spinkit", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_spinkit-5.2.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_spinkit-5.2.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_svg", |       "name": "flutter_svg", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_svg-2.0.10+1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_svg-2.0.10+1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_test", |       "name": "flutter_test", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/packages/flutter_test", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/packages/flutter_test", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_web_plugins", |       "name": "flutter_web_plugins", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/packages/flutter_web_plugins", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/packages/flutter_web_plugins", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "font_awesome_flutter", |       "name": "font_awesome_flutter", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/font_awesome_flutter-10.8.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/font_awesome_flutter-10.8.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "get_it", |       "name": "get_it", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/get_it-7.7.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/get_it-7.7.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "go_router", |       "name": "go_router", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/go_router-14.3.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/go_router-14.3.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "hover_menu", |       "name": "hover_menu", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/hover_menu-1.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/hover_menu-1.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.16" |       "languageVersion": "2.16" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "http", |       "name": "http", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/http-1.2.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/http-1.2.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "http_parser", |       "name": "http_parser", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/http_parser-4.0.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/http_parser-4.0.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "injectable", |       "name": "injectable", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/injectable-2.4.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/injectable-2.4.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "intl", |       "name": "intl", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/intl-0.19.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/intl-0.19.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "irondash_engine_context", |       "name": "irondash_engine_context", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "irondash_message_channel", |       "name": "irondash_message_channel", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/irondash_message_channel-0.7.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/irondash_message_channel-0.7.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "json_string", |       "name": "json_string", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/json_string-3.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/json_string-3.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "json_util", |       "name": "json_util", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/json_util-2.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/json_util-2.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "latlong2", |       "name": "latlong2", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/latlong2-0.9.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/latlong2-0.9.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "leak_tracker", |       "name": "leak_tracker", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker-10.0.7", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "leak_tracker_flutter_testing", |       "name": "leak_tracker_flutter_testing", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.8", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "leak_tracker_testing", |       "name": "leak_tracker_testing", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "lints", |       "name": "lints", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/lints-4.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/lints-4.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.1" |       "languageVersion": "3.1" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "lists", |       "name": "lists", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/lists-1.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/lists-1.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "localstorage", |       "name": "localstorage", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/localstorage-5.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/localstorage-5.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "logger", |       "name": "logger", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/logger-2.4.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/logger-2.4.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "logging", |       "name": "logging", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/logging-1.2.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/logging-1.2.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "matcher", |       "name": "matcher", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/matcher-0.12.16+1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/matcher-0.12.16+1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "material_color_utilities", |       "name": "material_color_utilities", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "meta", |       "name": "meta", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/meta-1.15.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/meta-1.15.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "mgrs_dart", |       "name": "mgrs_dart", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/mgrs_dart-2.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/mgrs_dart-2.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "number_text_input_formatter", |       "name": "number_text_input_formatter", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path", |       "name": "path", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path-1.9.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path-1.9.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_parsing", |       "name": "path_parsing", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_parsing-1.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_parsing-1.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_provider", |       "name": "path_provider", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider-2.1.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_provider-2.1.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_provider_android", |       "name": "path_provider_android", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_provider_foundation", |       "name": "path_provider_foundation", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_provider_linux", |       "name": "path_provider_linux", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_provider_platform_interface", |       "name": "path_provider_platform_interface", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_platform_interface-2.1.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_platform_interface-2.1.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path_provider_windows", |       "name": "path_provider_windows", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "petitparser", |       "name": "petitparser", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/petitparser-6.0.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/petitparser-6.0.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "pixel_snap", |       "name": "pixel_snap", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/pixel_snap-0.1.5", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/pixel_snap-0.1.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "platform", |       "name": "platform", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/platform-3.1.6", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/platform-3.1.6", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "plugin_platform_interface", |       "name": "plugin_platform_interface", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/plugin_platform_interface-2.1.8", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/plugin_platform_interface-2.1.8", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "polylabel", |       "name": "polylabel", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/polylabel-1.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/polylabel-1.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.13" |       "languageVersion": "2.13" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "proj4dart", |       "name": "proj4dart", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/proj4dart-2.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/proj4dart-2.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "scoped_model", |       "name": "scoped_model", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/scoped_model-2.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/scoped_model-2.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shamsi_date", |       "name": "shamsi_date", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shamsi_date-0.15.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shamsi_date-0.15.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences", |       "name": "shared_preferences", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.1" |       "languageVersion": "3.1" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences_android", |       "name": "shared_preferences_android", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences_foundation", |       "name": "shared_preferences_foundation", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences_linux", |       "name": "shared_preferences_linux", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences_platform_interface", |       "name": "shared_preferences_platform_interface", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_platform_interface-2.4.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_platform_interface-2.4.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences_web", |       "name": "shared_preferences_web", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "shared_preferences_windows", |       "name": "shared_preferences_windows", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "simple_gesture_detector", |       "name": "simple_gesture_detector", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/simple_gesture_detector-0.2.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/simple_gesture_detector-0.2.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "sky_engine", |       "name": "sky_engine", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/bin/cache/pkg/sky_engine", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/bin/cache/pkg/sky_engine", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "source_span", |       "name": "source_span", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/source_span-1.10.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/source_span-1.10.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "sprintf", |       "name": "sprintf", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/sprintf-7.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/sprintf-7.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "stack_trace", |       "name": "stack_trace", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/stack_trace-1.12.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/stack_trace-1.11.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.4" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "star_menu", |       "name": "star_menu", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/star_menu-4.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/star_menu-4.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "stream_channel", |       "name": "stream_channel", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/stream_channel-2.1.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/stream_channel-2.1.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "string_scanner", |       "name": "string_scanner", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/string_scanner-1.3.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/string_scanner-1.2.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.1" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "super_clipboard", |       "name": "super_clipboard", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/super_clipboard-0.8.19", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/super_clipboard-0.8.19", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "super_drag_and_drop", |       "name": "super_drag_and_drop", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/super_drag_and_drop-0.8.19", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/super_drag_and_drop-0.8.19", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "super_native_extensions", |       "name": "super_native_extensions", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "syncfusion_flutter_calendar", |       "name": "syncfusion_flutter_calendar", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_calendar-27.1.55", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_calendar-27.1.55", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "syncfusion_flutter_core", |       "name": "syncfusion_flutter_core", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_core-27.1.55", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_core-27.1.55", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "syncfusion_flutter_datepicker", |       "name": "syncfusion_flutter_datepicker", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_datepicker-27.1.55", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_datepicker-27.1.55", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "table_calendar", |       "name": "table_calendar", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/table_calendar-3.1.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/table_calendar-3.1.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "term_glyph", |       "name": "term_glyph", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/term_glyph-1.2.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/term_glyph-1.2.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "test_api", |       "name": "test_api", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/test_api-0.7.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/test_api-0.7.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.4" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "timezone", |       "name": "timezone", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/timezone-0.9.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/timezone-0.9.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "typed_data", |       "name": "typed_data", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/typed_data-1.3.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/typed_data-1.3.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "unicode", |       "name": "unicode", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/unicode-0.3.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/unicode-0.3.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "uuid", |       "name": "uuid", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/uuid-4.5.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/uuid-4.5.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vector_graphics", |       "name": "vector_graphics", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics-1.1.11+1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics-1.1.11+1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vector_graphics_codec", |       "name": "vector_graphics_codec", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics_codec-1.1.11+1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics_codec-1.1.11+1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vector_graphics_compiler", |       "name": "vector_graphics_compiler", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics_compiler-1.1.11+1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics_compiler-1.1.11+1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vector_math", |       "name": "vector_math", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_math-2.1.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vector_math-2.1.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.14" |       "languageVersion": "2.14" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vm_service", |       "name": "vm_service", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vm_service-14.3.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vm_service-14.2.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "web", |       "name": "web", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/web-0.5.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/web-0.5.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "win32", |       "name": "win32", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/win32-5.7.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/win32-5.7.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.5" |       "languageVersion": "3.5" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "win32_registry", |       "name": "win32_registry", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/win32_registry-1.1.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/win32_registry-1.1.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "wkt_parser", |       "name": "wkt_parser", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/wkt_parser-2.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/wkt_parser-2.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "xdg_directories", |       "name": "xdg_directories", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/xdg_directories-1.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/xdg_directories-1.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "xml", |       "name": "xml", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/xml-6.5.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/xml-6.5.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
| @@ -728,10 +728,10 @@ | |||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     } |     } | ||||||
|   ], |   ], | ||||||
|   "generated": "2025-01-09T14:13:17.895902Z", |   "generated": "2025-02-17T08:56:44.408862Z", | ||||||
|   "generator": "pub", |   "generator": "pub", | ||||||
|   "generatorVersion": "3.6.0", |   "generatorVersion": "3.5.3", | ||||||
|   "flutterRoot": "file:///home/opencloud/Tools/flutter", |   "flutterRoot": "file:///home/mr/snap/flutter/common/flutter", | ||||||
|   "flutterVersion": "3.27.1", |   "flutterVersion": "3.24.3", | ||||||
|   "pubCache": "file:///home/opencloud/.pub-cache" |   "pubCache": "file:///home/mr/.pub-cache" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,485 +1,485 @@ | |||||||
| alert_banner | alert_banner | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/alert_banner-1.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/alert_banner-1.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/alert_banner-1.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/alert_banner-1.0.1/lib/ | ||||||
| animated_toggle_switch | animated_toggle_switch | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/animated_toggle_switch-0.8.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/animated_toggle_switch-0.8.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/animated_toggle_switch-0.8.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/animated_toggle_switch-0.8.3/lib/ | ||||||
| args | args | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/args-2.6.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/args-2.6.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/args-2.6.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/args-2.6.0/lib/ | ||||||
| async | async | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/async-2.11.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/async-2.11.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/async-2.11.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/async-2.11.0/lib/ | ||||||
| boolean_selector | boolean_selector | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/lib/ | ||||||
| box_transform | box_transform | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/box_transform-0.4.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/box_transform-0.4.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/box_transform-0.4.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/box_transform-0.4.4/lib/ | ||||||
| characters | characters | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/characters-1.3.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/characters-1.3.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/characters-1.3.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/characters-1.3.0/lib/ | ||||||
| clock | clock | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/clock-1.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/clock-1.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/clock-1.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/clock-1.1.1/lib/ | ||||||
| collection | collection | ||||||
| 3.4 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/collection-1.19.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/collection-1.18.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/collection-1.19.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/collection-1.18.0/lib/ | ||||||
| convert | convert | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/convert-3.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/convert-3.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/convert-3.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/convert-3.1.1/lib/ | ||||||
| cron | cron | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/cron-0.6.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/cron-0.6.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/cron-0.6.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/cron-0.6.1/lib/ | ||||||
| crypto | crypto | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/crypto-3.0.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/crypto-3.0.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/crypto-3.0.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/crypto-3.0.3/lib/ | ||||||
| cupertino_icons | cupertino_icons | ||||||
| 3.1 | 3.1 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/cupertino_icons-1.0.8/ | file:///home/mr/.pub-cache/hosted/pub.dev/cupertino_icons-1.0.8/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/cupertino_icons-1.0.8/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/cupertino_icons-1.0.8/lib/ | ||||||
| dart_earcut | dart_earcut | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dart_earcut-1.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/dart_earcut-1.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dart_earcut-1.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dart_earcut-1.1.0/lib/ | ||||||
| dashed_path | dashed_path | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/lib/ | ||||||
| datetime_picker_formfield | datetime_picker_formfield | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/datetime_picker_formfield-2.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/datetime_picker_formfield-2.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/datetime_picker_formfield-2.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/datetime_picker_formfield-2.0.1/lib/ | ||||||
| desktop_window | desktop_window | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/desktop_window-0.4.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/desktop_window-0.4.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/desktop_window-0.4.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/desktop_window-0.4.1/lib/ | ||||||
| device_info_plus | device_info_plus | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2/lib/ | ||||||
| device_info_plus_platform_interface | device_info_plus_platform_interface | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus_platform_interface-7.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/device_info_plus_platform_interface-7.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus_platform_interface-7.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/device_info_plus_platform_interface-7.0.1/lib/ | ||||||
| dio | dio | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dio-5.7.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/dio-5.7.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dio-5.7.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dio-5.7.0/lib/ | ||||||
| dio_web_adapter | dio_web_adapter | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dio_web_adapter-2.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/dio_web_adapter-2.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dio_web_adapter-2.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dio_web_adapter-2.0.0/lib/ | ||||||
| dotted_line | dotted_line | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dotted_line-3.2.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/dotted_line-3.2.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dotted_line-3.2.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dotted_line-3.2.3/lib/ | ||||||
| el_tooltip | el_tooltip | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/el_tooltip-2.2.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/el_tooltip-2.2.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/el_tooltip-2.2.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/el_tooltip-2.2.1/lib/ | ||||||
| fake_async | fake_async | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fake_async-1.3.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/fake_async-1.3.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fake_async-1.3.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/fake_async-1.3.1/lib/ | ||||||
| ffi | ffi | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/ffi-2.1.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/ffi-2.1.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/ffi-2.1.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/ffi-2.1.3/lib/ | ||||||
| file | file | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/file-7.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/file-7.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/file-7.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/file-7.0.1/lib/ | ||||||
| fixnum | fixnum | ||||||
| 3.1 | 3.1 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fixnum-1.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/fixnum-1.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fixnum-1.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/fixnum-1.1.1/lib/ | ||||||
| flutter_advanced_switch | flutter_advanced_switch | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_advanced_switch-3.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_advanced_switch-3.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_advanced_switch-3.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_advanced_switch-3.1.0/lib/ | ||||||
| flutter_box_transform | flutter_box_transform | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.6/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.6/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.6/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.6/lib/ | ||||||
| flutter_colorpicker | flutter_colorpicker | ||||||
| 2.14 | 2.14 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/lib/ | ||||||
| flutter_event_calendar | flutter_event_calendar | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_event_calendar-1.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_event_calendar-1.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_event_calendar-1.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_event_calendar-1.0.0/lib/ | ||||||
| flutter_lints | flutter_lints | ||||||
| 3.1 | 3.1 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_lints-4.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_lints-4.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_lints-4.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_lints-4.0.0/lib/ | ||||||
| flutter_map | flutter_map | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_map-7.0.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_map-7.0.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_map-7.0.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_map-7.0.2/lib/ | ||||||
| flutter_spinkit | flutter_spinkit | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_spinkit-5.2.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_spinkit-5.2.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_spinkit-5.2.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_spinkit-5.2.1/lib/ | ||||||
| flutter_svg | flutter_svg | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_svg-2.0.10+1/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_svg-2.0.10+1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_svg-2.0.10+1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_svg-2.0.10+1/lib/ | ||||||
| font_awesome_flutter | font_awesome_flutter | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/font_awesome_flutter-10.8.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/font_awesome_flutter-10.8.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/font_awesome_flutter-10.8.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/font_awesome_flutter-10.8.0/lib/ | ||||||
| get_it | get_it | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/get_it-7.7.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/get_it-7.7.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/get_it-7.7.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/get_it-7.7.0/lib/ | ||||||
| go_router | go_router | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/go_router-14.3.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/go_router-14.3.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/go_router-14.3.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/go_router-14.3.0/lib/ | ||||||
| hover_menu | hover_menu | ||||||
| 2.16 | 2.16 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/lib/ | ||||||
| http | http | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/http-1.2.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/http-1.2.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/http-1.2.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/http-1.2.2/lib/ | ||||||
| http_parser | http_parser | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/http_parser-4.0.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/http_parser-4.0.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/http_parser-4.0.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/http_parser-4.0.2/lib/ | ||||||
| injectable | injectable | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/injectable-2.4.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/injectable-2.4.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/injectable-2.4.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/injectable-2.4.2/lib/ | ||||||
| intl | intl | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/intl-0.19.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/intl-0.19.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/intl-0.19.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/intl-0.19.0/lib/ | ||||||
| irondash_engine_context | irondash_engine_context | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4/lib/ | ||||||
| irondash_message_channel | irondash_message_channel | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/irondash_message_channel-0.7.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/irondash_message_channel-0.7.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/irondash_message_channel-0.7.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/irondash_message_channel-0.7.0/lib/ | ||||||
| json_string | json_string | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/json_string-3.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/json_string-3.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/json_string-3.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/json_string-3.0.1/lib/ | ||||||
| json_util | json_util | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/json_util-2.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/json_util-2.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/json_util-2.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/json_util-2.0.0/lib/ | ||||||
| latlong2 | latlong2 | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/latlong2-0.9.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/latlong2-0.9.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/latlong2-0.9.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/latlong2-0.9.1/lib/ | ||||||
| leak_tracker | leak_tracker | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker-10.0.7/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker-10.0.7/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5/lib/ | ||||||
| leak_tracker_flutter_testing | leak_tracker_flutter_testing | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.8/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.8/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5/lib/ | ||||||
| leak_tracker_testing | leak_tracker_testing | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/lib/ | ||||||
| lints | lints | ||||||
| 3.1 | 3.1 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/lints-4.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/lints-4.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/lints-4.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/lints-4.0.0/lib/ | ||||||
| lists | lists | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/lists-1.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/lists-1.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/lists-1.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/lists-1.0.1/lib/ | ||||||
| localstorage | localstorage | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/localstorage-5.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/localstorage-5.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/localstorage-5.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/localstorage-5.0.0/lib/ | ||||||
| logger | logger | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/logger-2.4.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/logger-2.4.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/logger-2.4.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/logger-2.4.0/lib/ | ||||||
| logging | logging | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/logging-1.2.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/logging-1.2.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/logging-1.2.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/logging-1.2.0/lib/ | ||||||
| matcher | matcher | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/ | file:///home/mr/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/lib/ | ||||||
| material_color_utilities | material_color_utilities | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/lib/ | ||||||
| meta | meta | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/meta-1.15.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/meta-1.15.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/meta-1.15.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/meta-1.15.0/lib/ | ||||||
| mgrs_dart | mgrs_dart | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/mgrs_dart-2.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/mgrs_dart-2.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/mgrs_dart-2.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/mgrs_dart-2.0.0/lib/ | ||||||
| number_text_input_formatter | number_text_input_formatter | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/ | file:///home/mr/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/lib/ | ||||||
| path | path | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path-1.9.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/path-1.9.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path-1.9.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path-1.9.0/lib/ | ||||||
| path_parsing | path_parsing | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_parsing-1.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_parsing-1.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_parsing-1.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_parsing-1.0.1/lib/ | ||||||
| path_provider | path_provider | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider-2.1.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider-2.1.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider-2.1.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider-2.1.4/lib/ | ||||||
| path_provider_android | path_provider_android | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4/lib/ | ||||||
| path_provider_foundation | path_provider_foundation | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0/lib/ | ||||||
| path_provider_linux | path_provider_linux | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/lib/ | ||||||
| path_provider_platform_interface | path_provider_platform_interface | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_platform_interface-2.1.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_platform_interface-2.1.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_platform_interface-2.1.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_platform_interface-2.1.2/lib/ | ||||||
| path_provider_windows | path_provider_windows | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/lib/ | ||||||
| petitparser | petitparser | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/petitparser-6.0.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/petitparser-6.0.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/petitparser-6.0.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/petitparser-6.0.2/lib/ | ||||||
| pixel_snap | pixel_snap | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/pixel_snap-0.1.5/ | file:///home/mr/.pub-cache/hosted/pub.dev/pixel_snap-0.1.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/pixel_snap-0.1.5/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/pixel_snap-0.1.5/lib/ | ||||||
| platform | platform | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/platform-3.1.6/ | file:///home/mr/.pub-cache/hosted/pub.dev/platform-3.1.6/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/platform-3.1.6/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/platform-3.1.6/lib/ | ||||||
| plugin_platform_interface | plugin_platform_interface | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/plugin_platform_interface-2.1.8/ | file:///home/mr/.pub-cache/hosted/pub.dev/plugin_platform_interface-2.1.8/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/plugin_platform_interface-2.1.8/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/plugin_platform_interface-2.1.8/lib/ | ||||||
| polylabel | polylabel | ||||||
| 2.13 | 2.13 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/polylabel-1.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/polylabel-1.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/polylabel-1.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/polylabel-1.0.1/lib/ | ||||||
| proj4dart | proj4dart | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/proj4dart-2.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/proj4dart-2.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/proj4dart-2.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/proj4dart-2.1.0/lib/ | ||||||
| scoped_model | scoped_model | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/scoped_model-2.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/scoped_model-2.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/scoped_model-2.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/scoped_model-2.0.0/lib/ | ||||||
| shamsi_date | shamsi_date | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shamsi_date-0.15.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/shamsi_date-0.15.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shamsi_date-0.15.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shamsi_date-0.15.0/lib/ | ||||||
| shared_preferences | shared_preferences | ||||||
| 3.1 | 3.1 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3/lib/ | ||||||
| shared_preferences_android | shared_preferences_android | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2/lib/ | ||||||
| shared_preferences_foundation | shared_preferences_foundation | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3/lib/ | ||||||
| shared_preferences_linux | shared_preferences_linux | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/lib/ | ||||||
| shared_preferences_platform_interface | shared_preferences_platform_interface | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_platform_interface-2.4.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_platform_interface-2.4.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_platform_interface-2.4.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_platform_interface-2.4.1/lib/ | ||||||
| shared_preferences_web | shared_preferences_web | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1/lib/ | ||||||
| shared_preferences_windows | shared_preferences_windows | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/lib/ | ||||||
| simple_gesture_detector | simple_gesture_detector | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/simple_gesture_detector-0.2.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/simple_gesture_detector-0.2.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/simple_gesture_detector-0.2.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/simple_gesture_detector-0.2.1/lib/ | ||||||
| source_span | source_span | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/source_span-1.10.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/source_span-1.10.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/source_span-1.10.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/source_span-1.10.0/lib/ | ||||||
| sprintf | sprintf | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/sprintf-7.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/sprintf-7.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/sprintf-7.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/sprintf-7.0.0/lib/ | ||||||
| stack_trace | stack_trace | ||||||
| 3.4 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stack_trace-1.12.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/stack_trace-1.11.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stack_trace-1.12.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/stack_trace-1.11.1/lib/ | ||||||
| star_menu | star_menu | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/star_menu-4.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/star_menu-4.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/star_menu-4.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/star_menu-4.0.1/lib/ | ||||||
| stream_channel | stream_channel | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/lib/ | ||||||
| string_scanner | string_scanner | ||||||
| 3.1 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/string_scanner-1.3.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/string_scanner-1.2.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/string_scanner-1.3.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/string_scanner-1.2.0/lib/ | ||||||
| super_clipboard | super_clipboard | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/super_clipboard-0.8.19/ | file:///home/mr/.pub-cache/hosted/pub.dev/super_clipboard-0.8.19/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/super_clipboard-0.8.19/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/super_clipboard-0.8.19/lib/ | ||||||
| super_drag_and_drop | super_drag_and_drop | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/super_drag_and_drop-0.8.19/ | file:///home/mr/.pub-cache/hosted/pub.dev/super_drag_and_drop-0.8.19/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/super_drag_and_drop-0.8.19/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/super_drag_and_drop-0.8.19/lib/ | ||||||
| super_native_extensions | super_native_extensions | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19/ | file:///home/mr/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19/lib/ | ||||||
| syncfusion_flutter_calendar | syncfusion_flutter_calendar | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_calendar-27.1.55/ | file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_calendar-27.1.55/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_calendar-27.1.55/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_calendar-27.1.55/lib/ | ||||||
| syncfusion_flutter_core | syncfusion_flutter_core | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_core-27.1.55/ | file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_core-27.1.55/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_core-27.1.55/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_core-27.1.55/lib/ | ||||||
| syncfusion_flutter_datepicker | syncfusion_flutter_datepicker | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_datepicker-27.1.55/ | file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_datepicker-27.1.55/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/syncfusion_flutter_datepicker-27.1.55/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/syncfusion_flutter_datepicker-27.1.55/lib/ | ||||||
| table_calendar | table_calendar | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/table_calendar-3.1.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/table_calendar-3.1.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/table_calendar-3.1.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/table_calendar-3.1.2/lib/ | ||||||
| term_glyph | term_glyph | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/lib/ | ||||||
| test_api | test_api | ||||||
| 3.4 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/test_api-0.7.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/test_api-0.7.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/test_api-0.7.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/test_api-0.7.2/lib/ | ||||||
| timezone | timezone | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/timezone-0.9.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/timezone-0.9.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/timezone-0.9.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/timezone-0.9.4/lib/ | ||||||
| typed_data | typed_data | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/typed_data-1.3.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/typed_data-1.3.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/typed_data-1.3.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/typed_data-1.3.2/lib/ | ||||||
| unicode | unicode | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/unicode-0.3.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/unicode-0.3.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/unicode-0.3.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/unicode-0.3.1/lib/ | ||||||
| uuid | uuid | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/uuid-4.5.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/uuid-4.5.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/uuid-4.5.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/uuid-4.5.1/lib/ | ||||||
| vector_graphics | vector_graphics | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics-1.1.11+1/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics-1.1.11+1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics-1.1.11+1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics-1.1.11+1/lib/ | ||||||
| vector_graphics_codec | vector_graphics_codec | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics_codec-1.1.11+1/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics_codec-1.1.11+1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics_codec-1.1.11+1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics_codec-1.1.11+1/lib/ | ||||||
| vector_graphics_compiler | vector_graphics_compiler | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics_compiler-1.1.11+1/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics_compiler-1.1.11+1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_graphics_compiler-1.1.11+1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_graphics_compiler-1.1.11+1/lib/ | ||||||
| vector_math | vector_math | ||||||
| 2.14 | 2.14 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_math-2.1.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_math-2.1.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_math-2.1.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_math-2.1.4/lib/ | ||||||
| vm_service | vm_service | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vm_service-14.3.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/vm_service-14.2.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vm_service-14.3.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vm_service-14.2.5/lib/ | ||||||
| web | web | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/web-0.5.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/web-0.5.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/web-0.5.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/web-0.5.1/lib/ | ||||||
| win32 | win32 | ||||||
| 3.5 | 3.5 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/win32-5.7.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/win32-5.7.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/win32-5.7.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/win32-5.7.1/lib/ | ||||||
| win32_registry | win32_registry | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/win32_registry-1.1.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/win32_registry-1.1.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/win32_registry-1.1.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/win32_registry-1.1.3/lib/ | ||||||
| wkt_parser | wkt_parser | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/wkt_parser-2.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/wkt_parser-2.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/wkt_parser-2.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/wkt_parser-2.0.0/lib/ | ||||||
| xdg_directories | xdg_directories | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/xdg_directories-1.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/xdg_directories-1.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/xdg_directories-1.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/xdg_directories-1.1.0/lib/ | ||||||
| xml | xml | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/xml-6.5.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/xml-6.5.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/xml-6.5.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/xml-6.5.0/lib/ | ||||||
| oc_front | oc_front | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Source/opencloud/oc-local/src/oc-front/ | file:///home/mr/Documents/OC/oc-front/ | ||||||
| file:///home/opencloud/Source/opencloud/oc-local/src/oc-front/lib/ | file:///home/mr/Documents/OC/oc-front/lib/ | ||||||
| flutter_flow_chart | flutter_flow_chart | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Source/opencloud/oc-local/src/oc-front/library/flutter_flow_chart/ | file:///home/mr/Documents/OC/oc-front/library/flutter_flow_chart/ | ||||||
| file:///home/opencloud/Source/opencloud/oc-local/src/oc-front/library/flutter_flow_chart/lib/ | file:///home/mr/Documents/OC/oc-front/library/flutter_flow_chart/lib/ | ||||||
| sky_engine | sky_engine | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/Tools/flutter/bin/cache/pkg/sky_engine/ | file:///home/mr/snap/flutter/common/flutter/bin/cache/pkg/sky_engine/ | ||||||
| file:///home/opencloud/Tools/flutter/bin/cache/pkg/sky_engine/lib/ | file:///home/mr/snap/flutter/common/flutter/bin/cache/pkg/sky_engine/lib/ | ||||||
| flutter | flutter | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter/ | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter/lib/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter/lib/ | ||||||
| flutter_test | flutter_test | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter_test/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter_test/ | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter_test/lib/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter_test/lib/ | ||||||
| flutter_web_plugins | flutter_web_plugins | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter_web_plugins/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter_web_plugins/ | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter_web_plugins/lib/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter_web_plugins/lib/ | ||||||
| 2 | 2 | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| 3.27.1 | 3.24.3 | ||||||
| @@ -1,16 +1,16 @@ | |||||||
| # This is a generated file; do not edit or check into version control. | # This is a generated file; do not edit or check into version control. | ||||||
| desktop_window=/home/opencloud/.pub-cache/hosted/pub.dev/desktop_window-0.4.1/ | desktop_window=/home/mr/.pub-cache/hosted/pub.dev/desktop_window-0.4.1/ | ||||||
| device_info_plus=/home/opencloud/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2/ | device_info_plus=/home/mr/.pub-cache/hosted/pub.dev/device_info_plus-10.1.2/ | ||||||
| irondash_engine_context=/home/opencloud/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4/ | irondash_engine_context=/home/mr/.pub-cache/hosted/pub.dev/irondash_engine_context-0.5.4/ | ||||||
| path_provider=/home/opencloud/.pub-cache/hosted/pub.dev/path_provider-2.1.4/ | path_provider=/home/mr/.pub-cache/hosted/pub.dev/path_provider-2.1.4/ | ||||||
| path_provider_android=/home/opencloud/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4/ | path_provider_android=/home/mr/.pub-cache/hosted/pub.dev/path_provider_android-2.2.4/ | ||||||
| path_provider_foundation=/home/opencloud/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0/ | path_provider_foundation=/home/mr/.pub-cache/hosted/pub.dev/path_provider_foundation-2.4.0/ | ||||||
| path_provider_linux=/home/opencloud/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/ | path_provider_linux=/home/mr/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/ | ||||||
| path_provider_windows=/home/opencloud/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/ | path_provider_windows=/home/mr/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/ | ||||||
| shared_preferences=/home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3/ | shared_preferences=/home/mr/.pub-cache/hosted/pub.dev/shared_preferences-2.2.3/ | ||||||
| shared_preferences_android=/home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2/ | shared_preferences_android=/home/mr/.pub-cache/hosted/pub.dev/shared_preferences_android-2.2.2/ | ||||||
| shared_preferences_foundation=/home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3/ | shared_preferences_foundation=/home/mr/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.3/ | ||||||
| shared_preferences_linux=/home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/ | shared_preferences_linux=/home/mr/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/ | ||||||
| shared_preferences_web=/home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1/ | shared_preferences_web=/home/mr/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.1/ | ||||||
| shared_preferences_windows=/home/opencloud/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/ | shared_preferences_windows=/home/mr/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/ | ||||||
| super_native_extensions=/home/opencloud/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19/ | super_native_extensions=/home/mr/.pub-cache/hosted/pub.dev/super_native_extensions-0.8.19/ | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -41,7 +41,7 @@ class CollaborativeAreaLocal { | |||||||
|     await _service.delete(null, id, {}); |     await _service.delete(null, id, {}); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future<void> createCollaborativeArea(String name, BuildContext? context) async { |   static Future<void> createCollaborativeArea(BuildContext context, String name) async { | ||||||
|     Workspace n = Workspace(name: name); |     Workspace n = Workspace(name: name); | ||||||
|     await _service.post(context, n.serialize(), {}).then((value) { |     await _service.post(context, n.serialize(), {}).then((value) { | ||||||
|       if (value.data != null) { |       if (value.data != null) { | ||||||
|   | |||||||
| @@ -1,9 +1,13 @@ | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; |  | ||||||
| import 'package:oc_front/core/sections/end_drawer.dart'; | import 'package:oc_front/core/sections/end_drawer.dart'; | ||||||
| import 'package:oc_front/core/sections/header/header.dart'; | import 'package:oc_front/core/sections/header/header.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/data.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
|  | import 'package:oc_front/models/resources/workflow.dart'; | ||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/models/workspace.dart'; | import 'package:oc_front/models/workspace.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/workspace_service.dart'; | import 'package:oc_front/core/services/specialized_services/workspace_service.dart'; | ||||||
| import 'package:oc_front/pages/catalog.dart'; | import 'package:oc_front/pages/catalog.dart'; | ||||||
| @@ -22,7 +26,7 @@ class WorkspaceLocal { | |||||||
|   static final WorkspaceService _service = WorkspaceService(); |   static final WorkspaceService _service = WorkspaceService(); | ||||||
|   static List<AbstractItem> items = []; |   static List<AbstractItem> items = []; | ||||||
|  |  | ||||||
|   static Future<void> init(BuildContext context, bool changeCurrent) async { |   static Future<void> init(BuildContext? context, bool changeCurrent) async { | ||||||
|     var value = await _service.all(context); |     var value = await _service.all(context); | ||||||
|     if (value.data != null && value.data!.values.isNotEmpty ) { |     if (value.data != null && value.data!.values.isNotEmpty ) { | ||||||
|       var vals = value.data!.values; |       var vals = value.data!.values; | ||||||
| @@ -144,7 +148,7 @@ class WorkspaceLocal { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   static List<AbstractItem> byWorkspace(String id) { |   static List<AbstractItem> byWorkspace(String id) { | ||||||
|     List<AbstractItem<FlowData>> d = []; |     List<AbstractItem> d = []; | ||||||
|     var w = workspaces[id]!; |     var w = workspaces[id]!; | ||||||
|     d = [ ...d, ...w.datas.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; |     d = [ ...d, ...w.datas.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; | ||||||
|     d = [ ...d, ...w.computes.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; |     d = [ ...d, ...w.computes.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; | ||||||
| @@ -156,7 +160,7 @@ class WorkspaceLocal { | |||||||
|    |    | ||||||
|   static List<AbstractItem> byTopic(String topic, bool all) { |   static List<AbstractItem> byTopic(String topic, bool all) { | ||||||
|     if (all) { |     if (all) { | ||||||
|       List<AbstractItem<FlowData>> d = []; |       List<AbstractItem> d = []; | ||||||
|       for (var w in workspaces.values) { |       for (var w in workspaces.values) { | ||||||
|         d = [ ...d, ...w.datas.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; |         d = [ ...d, ...w.datas.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; | ||||||
|         d = [ ...d, ...w.computes.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; |         d = [ ...d, ...w.computes.where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; | ||||||
| @@ -171,7 +175,7 @@ class WorkspaceLocal { | |||||||
|  |  | ||||||
|   static AbstractItem? getItem(String id, bool all) { |   static AbstractItem? getItem(String id, bool all) { | ||||||
|     if (all) { |     if (all) { | ||||||
|       List<AbstractItem<FlowData>> d = []; |       List<AbstractItem> d = []; | ||||||
|       for (var w in workspaces.values) { |       for (var w in workspaces.values) { | ||||||
|         d = [ ...d, ...w.datas.where((element) => element.id.toString() == id).where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; |         d = [ ...d, ...w.datas.where((element) => element.id.toString() == id).where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; | ||||||
|         d = [ ...d, ...w.computes.where((element) => element.id.toString() == id).where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; |         d = [ ...d, ...w.computes.where((element) => element.id.toString() == id).where((element) => !d.map( (d2) => d2.id).contains(element.id)) ]; | ||||||
| @@ -189,13 +193,15 @@ class WorkspaceLocal { | |||||||
|     if (!WorkspaceLocal.hasItem(item) && workspaces[current] != null && workspaces[current]!.id != null) {  |     if (!WorkspaceLocal.hasItem(item) && workspaces[current] != null && workspaces[current]!.id != null) {  | ||||||
|       items.add(item);  |       items.add(item);  | ||||||
|       if (item.topic == "data") { workspaces[current]!.datas.add(item as DataItem); } |       if (item.topic == "data") { workspaces[current]!.datas.add(item as DataItem); } | ||||||
|       if (item.topic == "processing") { workspaces[current]!.processings.add(item as ProcessingItem); } |       if (item.topic == "processing") {  workspaces[current]!.processings.add(item as ProcessingItem); } | ||||||
|       if (item.topic == "compute") { workspaces[current]!.computes.add(item as ComputeItem); } |       if (item.topic == "compute") { workspaces[current]!.computes.add(item as ComputeItem); } | ||||||
|       if (item.topic == "storage") { workspaces[current]!.storages.add(item as StorageItem); } |       if (item.topic == "storage") { workspaces[current]!.storages.add(item as StorageItem); } | ||||||
|       if (item.topic == "workflow") { workspaces[current]!.workflows.add(item as WorkflowItem); } |       if (item.topic == "workflow") { workspaces[current]!.workflows.add(item as WorkflowItem); } | ||||||
|       try { _service.put(null, workspaces[current]!.id!, workspaces[current]!.serialize(), {}); |       try {  | ||||||
|       } catch (e) { /* */ } |         _service.put(null, workspaces[current]!.id!, workspaces[current]!.serialize(), {}); | ||||||
|       _service.put(null, item.id ?? "", workspaces[current]!.serialize(), {}); |       } catch (e) {  | ||||||
|  |         _service.put(null, item.id ?? "", workspaces[current]!.serialize(), {}); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   static void removeItem(AbstractItem item) { |   static void removeItem(AbstractItem item) { | ||||||
| @@ -205,9 +211,11 @@ class WorkspaceLocal { | |||||||
|     if (item.topic == "compute") { workspaces[current]!.computes.removeWhere( (e) => e.id == item.id); } |     if (item.topic == "compute") { workspaces[current]!.computes.removeWhere( (e) => e.id == item.id); } | ||||||
|     if (item.topic == "storage") { workspaces[current]!.storages.removeWhere( (e) => e.id == item.id); } |     if (item.topic == "storage") { workspaces[current]!.storages.removeWhere( (e) => e.id == item.id); } | ||||||
|     if (item.topic == "workflow") { workspaces[current]!.workflows.removeWhere( (e) => e.id == item.id); } |     if (item.topic == "workflow") { workspaces[current]!.workflows.removeWhere( (e) => e.id == item.id); } | ||||||
|     try { _service.put(null, workspaces[current]!.id!, workspaces[current]!.serialize(), {}); |     try {  | ||||||
|     } catch (e) { /* */ } |       _service.put(null, workspaces[current]!.id!, workspaces[current]!.serialize(), {}); | ||||||
|      _service.put(null, item.id ?? "", workspaces[current]!.serialize(), {}); |     } catch (e) {  | ||||||
|  |       _service.put(null, item.id ?? "", workspaces[current]!.serialize(), {}); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   static bool hasItemByID(String id) {  |   static bool hasItemByID(String id) {  | ||||||
|     return items.where((element) => element.id == id).isNotEmpty; |     return items.where((element) => element.id == id).isNotEmpty; | ||||||
|   | |||||||
| @@ -1,11 +1,12 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:oc_front/pages/catalog.dart'; |  | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; |  | ||||||
| import 'package:oc_front/pages/shared.dart'; | import 'package:oc_front/pages/shared.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | import 'package:oc_front/pages/catalog.dart'; | ||||||
| import 'package:oc_front/widgets/items/item_row.dart'; | import 'package:oc_front/widgets/items/item_row.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/core/models/workspace_local.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | ||||||
|  | import 'package:oc_front/core/models/shared_workspace_local.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/workspace_service.dart'; | import 'package:oc_front/core/services/specialized_services/workspace_service.dart'; | ||||||
|  |  | ||||||
| @@ -62,7 +63,7 @@ class EndDrawerWidgetState extends State<EndDrawerWidget> { | |||||||
|               child: Row( mainAxisAlignment: MainAxisAlignment.center, children: [  |               child: Row( mainAxisAlignment: MainAxisAlignment.center, children: [  | ||||||
|                 Padding(padding: EdgeInsets.only(right: 5), child: Icon(Icons.share, size: 12, color: Colors.grey)), |                 Padding(padding: EdgeInsets.only(right: 5), child: Icon(Icons.share, size: 12, color: Colors.grey)), | ||||||
|                 Text(  (WorkspaceLocal.workspaces[WorkspaceLocal.current]?.shared) == null ?  |                 Text(  (WorkspaceLocal.workspaces[WorkspaceLocal.current]?.shared) == null ?  | ||||||
|                   "actually not shared" : "share with ${WorkspaceLocal.workspaces[WorkspaceLocal.current]!.shared!}", |                   "actually not shared" : "share with ${CollaborativeAreaLocal.workspaces[WorkspaceLocal.workspaces[WorkspaceLocal.current]!.shared!]}", | ||||||
|                 style: TextStyle( fontSize: 14, color: Colors.grey ),), |                 style: TextStyle( fontSize: 14, color: Colors.grey ),), | ||||||
|               ]), |               ]), | ||||||
|             ), |             ), | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:localstorage/localstorage.dart'; | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; | import 'package:oc_front/core/models/workspace_local.dart'; | ||||||
| import 'package:oc_front/core/sections/header/default.dart'; | import 'package:oc_front/core/sections/header/default.dart'; | ||||||
| import 'package:oc_front/core/sections/header/menu.dart'; | import 'package:oc_front/core/sections/header/menu.dart'; | ||||||
| @@ -14,7 +15,10 @@ class SearchConstants { | |||||||
|   static String? get() { return _searchHost[AppRouter.currentRoute.route]; } |   static String? get() { return _searchHost[AppRouter.currentRoute.route]; } | ||||||
|   static void set(String? search) { _searchHost[AppRouter.currentRoute.route] = search; } |   static void set(String? search) { _searchHost[AppRouter.currentRoute.route] = search; } | ||||||
|   static void remove() { _searchHost.remove(AppRouter.currentRoute.route); } |   static void remove() { _searchHost.remove(AppRouter.currentRoute.route); } | ||||||
|   static void clear() { _searchHost.clear(); } |   static void clear() {  | ||||||
|  |     localStorage.setItem("search", ""); | ||||||
|  |     _searchHost.clear();  | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| class HeaderConstants { | class HeaderConstants { | ||||||
| @@ -56,7 +60,7 @@ class HeaderConstants { | |||||||
| } | } | ||||||
|  |  | ||||||
| class HeaderWidget extends StatefulWidget { | class HeaderWidget extends StatefulWidget { | ||||||
|   HeaderWidget () : super(key: null); |   const HeaderWidget () : super(key: null); | ||||||
|   @override HeaderWidgetState createState() => HeaderWidgetState(); |   @override HeaderWidgetState createState() => HeaderWidgetState(); | ||||||
| } | } | ||||||
| class HeaderWidgetState extends State<HeaderWidget> { | class HeaderWidgetState extends State<HeaderWidget> { | ||||||
| @@ -79,24 +83,34 @@ class HeaderWidgetState extends State<HeaderWidget> { | |||||||
|         child: Row(crossAxisAlignment: CrossAxisAlignment.stretch,  |         child: Row(crossAxisAlignment: CrossAxisAlignment.stretch,  | ||||||
|             mainAxisAlignment: MainAxisAlignment.end, |             mainAxisAlignment: MainAxisAlignment.end, | ||||||
|             children: [ |             children: [ | ||||||
|  |               InkWell( onTap: () { | ||||||
|  |                   AppRouter.currentRoute.factory.back(context); | ||||||
|  |                 }, | ||||||
|  |                 child: | ||||||
|  |                   Container(width: 50, height: 50,  | ||||||
|  |                     color: Colors.black, | ||||||
|  |                     child: Center(child: Icon(Icons.keyboard_backspace, color: Colors.white)) | ||||||
|  |                   ), | ||||||
|  |               ), | ||||||
|                 ShallowTextInputWidget( |                 ShallowTextInputWidget( | ||||||
|                     filled: midColor, |                   filled: midColor, | ||||||
|                     width: getMainWidth(context) - 451, |                   current: AppRouter.currentRoute.factory.getSearch(), | ||||||
|                     type: CollaborativeAreaType.workspace, |                   width: getMainWidth(context) - 501, | ||||||
|                     hint: "search in resources...", |                   type: CollaborativeAreaType.workspace, | ||||||
|                     iconLoad: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? Icons.search : null, |                   hint: "search in resources...", | ||||||
|                     iconRemove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? Icons.screen_search_desktop_outlined : null, |                   iconLoad: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? Icons.search : null, | ||||||
|                     tooltipLoad: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? "search" : null, |                   iconRemove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? Icons.screen_search_desktop_outlined : null, | ||||||
|                     tooltipRemove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? "distributed search" : null, |                   tooltipLoad: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? "search" : null, | ||||||
|                     canLoad: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? (String? str) => str != null && str.isNotEmpty : null, |                   tooltipRemove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? "distributed search" : null, | ||||||
|                     canRemove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? (String? str) => str != null && str.isNotEmpty : null, |                   canLoad: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? (String? str) => str != null && str.isNotEmpty : null, | ||||||
|                     change: (value) => SearchConstants.set(value), |                   canRemove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? (String? str) => str != null && str.isNotEmpty : null, | ||||||
|                     loadStr: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? (String val) async { |                   change: (value) => SearchConstants.set(value), | ||||||
|                       AppRouter.currentRoute.factory.search(context, false);  |                   loadStr: PermsService.getPerm(Perms.SEARCH_INTERNAL) ? (String val) async { | ||||||
|                     } : null, |                     AppRouter.currentRoute.factory.search(context, false);  | ||||||
|                     remove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? (String val) async { |                   } : null, | ||||||
|                       AppRouter.currentRoute.factory.search(context, true);  |                   remove: PermsService.getPerm(Perms.SEARCH_EXTERNAL) ? (String val) async { | ||||||
|                     } : null, |                     AppRouter.currentRoute.factory.search(context, true);  | ||||||
|  |                   } : null, | ||||||
|               ), |               ), | ||||||
|               Container( padding: EdgeInsets.only(left: 50), |               Container( padding: EdgeInsets.only(left: 50), | ||||||
|                 decoration: BoxDecoration( border: Border( left: BorderSide( color: Colors.white ))), |                 decoration: BoxDecoration( border: Border( left: BorderSide( color: Colors.white ))), | ||||||
|   | |||||||
| @@ -5,13 +5,13 @@ import 'package:oc_front/core/services/auth.service.dart'; | |||||||
| import 'package:oc_front/core/services/router.dart'; | import 'package:oc_front/core/services/router.dart'; | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/pages/catalog.dart'; | ||||||
|  |  | ||||||
| class HeaderMenuWidget extends StatefulWidget{ | class HeaderMenuWidget extends StatefulWidget{ | ||||||
|   HeaderMenuWidget (): super(key: HeaderConstants.getKey()); |   HeaderMenuWidget (): super(key: HeaderConstants.getKey()); | ||||||
|   @override HeaderMenuWidgetState createState() => HeaderMenuWidgetState(); |   @override HeaderMenuWidgetState createState() => HeaderMenuWidgetState(); | ||||||
| } | } | ||||||
| class HeaderMenuWidgetState extends State<HeaderMenuWidget> { | class HeaderMenuWidgetState extends State<HeaderMenuWidget> { | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     return Container(  |     return Container(  | ||||||
|         width: getWidth(context), |         width: getWidth(context), | ||||||
| @@ -23,12 +23,16 @@ class HeaderMenuWidgetState extends State<HeaderMenuWidget> { | |||||||
|         child: Stack(children: [   |         child: Stack(children: [   | ||||||
|           AppRouter.currentRoute.factory.searchFill() ? Container() : Positioned( top: 0, left: 30,  |           AppRouter.currentRoute.factory.searchFill() ? Container() : Positioned( top: 0, left: 30,  | ||||||
|             child: InkWell( onTap: () { |             child: InkWell( onTap: () { | ||||||
|  |               CatalogFactory.key.currentState?.widget.isSearch = true; | ||||||
|  |               CatalogFactory.key.currentState?.widget.items = []; | ||||||
|               AppRouter.zones.first.go(context, {}); |               AppRouter.zones.first.go(context, {}); | ||||||
|  |               AppRouter.zones.first.factory.getKey().currentState?.setState(() {}); | ||||||
|  |               HeaderConstants.headerKey.currentState?.setState(() {}); | ||||||
|             },  |             },  | ||||||
|             child: Wrap( alignment: WrapAlignment.center, children: [  |             child: Wrap( alignment: WrapAlignment.center, children: [  | ||||||
|             SvgPicture.asset("assets/images/icon.svg", width: 70, height: 70, semanticsLabel: 'OpenCloud Logo'), |               SvgPicture.asset("assets/images/icon.svg", width: 70, height: 70, semanticsLabel: 'OpenCloud Logo'), | ||||||
|             Container( height: 50, alignment: Alignment.centerLeft, margin: const EdgeInsets.only(left: 20), |               Container( height: 50, alignment: Alignment.centerLeft, margin: const EdgeInsets.only(left: 20), | ||||||
|               child: Text(AppRouter.currentRoute.label ?? "Where am I ?", style: TextStyle( color: Colors.grey))) |                 child: Text(AppRouter.currentRoute.label ?? "Where am I ?", style: TextStyle( color: Colors.grey))) | ||||||
|           ]))), |           ]))), | ||||||
|           Padding( padding: const EdgeInsets.only(left: 50),  |           Padding( padding: const EdgeInsets.only(left: 50),  | ||||||
|           child: Row(crossAxisAlignment: CrossAxisAlignment.stretch,  |           child: Row(crossAxisAlignment: CrossAxisAlignment.stretch,  | ||||||
|   | |||||||
| @@ -18,9 +18,8 @@ class APIService<T extends SerializerDeserializer> { | |||||||
|  |  | ||||||
|   Dio _dio = Dio( |   Dio _dio = Dio( | ||||||
|     BaseOptions( |     BaseOptions( | ||||||
|       baseUrl: config.get('HOST', |       baseUrl: const String.fromEnvironment('HOST', defaultValue: 'http://localhost:8080'), // you can keep this blank | ||||||
|           defaultValue: 'http://localhost:8080'), // you can keep this blank |       headers: { 'Content-Type': 'application/json; charset=UTF-8', 'Access-Control-Allow-Origin': '*' }, | ||||||
|       headers: {'Content-Type': 'application/json; charset=UTF-8'}, |  | ||||||
|     ), |     ), | ||||||
|   )..interceptors.add(LogInterceptor( |   )..interceptors.add(LogInterceptor( | ||||||
|       requestHeader: true, |       requestHeader: true, | ||||||
| @@ -30,7 +29,7 @@ class APIService<T extends SerializerDeserializer> { | |||||||
|     _dio = Dio( |     _dio = Dio( | ||||||
|       BaseOptions( |       BaseOptions( | ||||||
|         baseUrl: baseURL, // you can keep this blank |         baseUrl: baseURL, // you can keep this blank | ||||||
|         headers: {'Content-Type': 'application/json; charset=UTF-8'}, |         headers: { 'Content-Type': 'application/json; charset=UTF-8', 'Access-Control-Allow-Origin': '*' }, | ||||||
|       ), |       ), | ||||||
|     )..interceptors.add( |     )..interceptors.add( | ||||||
|         LogInterceptor( |         LogInterceptor( | ||||||
| @@ -102,45 +101,29 @@ class APIService<T extends SerializerDeserializer> { | |||||||
|     } catch (e) {/* */} |     } catch (e) {/* */} | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<APIResponse<T>> _main( |   Future<APIResponse<T>> _main(String url, dynamic body, String method, String succeed, bool force,  | ||||||
|       String url, |                                BuildContext? context, Options? options) async { | ||||||
|       dynamic body, |  | ||||||
|       String method, |  | ||||||
|       String succeed, |  | ||||||
|       bool force, |  | ||||||
|       BuildContext? context, |  | ||||||
|       Options? options) async { |  | ||||||
|     var err = "";  |     var err = "";  | ||||||
|     try { |       try { | ||||||
|       var type = localStorage.getItem('tokenType') ?? "bearer"; |         var type = localStorage.getItem('tokenType') ?? "bearer"; | ||||||
|       _dio.options.headers["Authorization"] = |         _dio.options.headers["Authorization"] = "${type[0].toUpperCase() + type.substring(1)} ${localStorage.getItem('accessToken') ?? ""}"; | ||||||
|           "${type[0].toUpperCase() + type.substring(1)} ${localStorage.getItem('accessToken') ?? ""}"; |         _dio.interceptors.clear();  | ||||||
|       _dio.interceptors.clear(); |         var response = await _request(url, method, body, options); | ||||||
|       var response = await _request(url, method, body, options); |         if (response.statusCode != null && response.statusCode! < 400) { | ||||||
|       if (response.statusCode != null && response.statusCode! < 400) { |           if (method == "delete") { cache.remove(url); return APIResponse<T>(); } | ||||||
|         if (method == "delete") { |           APIResponse<T> resp = APIResponse<T>().deserialize(response.data);  | ||||||
|           cache.remove(url); |           if (resp.error == "") {  | ||||||
|           return APIResponse<T>(); |             if (context != null && succeed != "") { | ||||||
|         } |               // ignore: use_build_context_synchronously | ||||||
|         APIResponse<T> resp = APIResponse<T>().deserialize(response.data); |               try { | ||||||
|         if (resp.error == "") { |                 showAlertBanner(context, () {}, InfoAlertBannerChild(text: succeed), // <-- Put any widget here you want! | ||||||
|           if (context != null && succeed != "") { |                 alertBannerLocation:  AlertBannerLocation.bottom,); | ||||||
|             // ignore: use_build_context_synchronously |               } catch (e) { /* */ } | ||||||
|             showAlertBanner( |             } | ||||||
|               context, |             return resp; | ||||||
|               () {}, |  | ||||||
|               InfoAlertBannerChild( |  | ||||||
|                   text: succeed), // <-- Put any widget here you want! |  | ||||||
|               alertBannerLocation: AlertBannerLocation.bottom, |  | ||||||
|             ); |  | ||||||
|           } |           } | ||||||
|           return resp; |           err = resp.error ?? "internal error"; | ||||||
|         } |         } else if (response.statusCode == 401) { err = "not authorized"; } | ||||||
|         err = resp.error ?? "internal error"; |  | ||||||
|       } |  | ||||||
|       if (response.statusCode == 401) { |  | ||||||
|         err = "not authorized"; |  | ||||||
|       } |  | ||||||
|     } catch (e, s) { |     } catch (e, s) { | ||||||
|       print(e); |       print(e); | ||||||
|       print(s); |       print(s); | ||||||
| @@ -149,11 +132,10 @@ class APIService<T extends SerializerDeserializer> { | |||||||
|     //if (err.contains("token") && err.contains("expired")) {  AuthService().unAuthenticate();  } |     //if (err.contains("token") && err.contains("expired")) {  AuthService().unAuthenticate();  } | ||||||
|     if (context != null) { |     if (context != null) { | ||||||
|       // ignore: use_build_context_synchronously |       // ignore: use_build_context_synchronously | ||||||
|       showAlertBanner( |       try { | ||||||
|         context, () {}, |         showAlertBanner( context, () {}, AlertAlertBannerChild(text: err),// <-- Put any widget here you want! | ||||||
|         AlertAlertBannerChild(text: err), // <-- Put any widget here you want! |                        alertBannerLocation:  AlertBannerLocation.bottom,); | ||||||
|         alertBannerLocation: AlertBannerLocation.bottom, |       } catch (e) { /* */ } | ||||||
|       ); |  | ||||||
|     }  |     }  | ||||||
|     throw Exception(err); |     throw Exception(err); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -1,21 +1,15 @@ | |||||||
| import 'package:localstorage/localstorage.dart'; | import 'package:localstorage/localstorage.dart'; | ||||||
| import 'package:oc_front/core/conf/conf_reader.dart'; | import 'package:oc_front/core/conf/conf_reader.dart'; | ||||||
| import 'package:oc_front/core/sections/header/header.dart'; |  | ||||||
| import 'package:oc_front/core/services/api_service.dart'; | import 'package:oc_front/core/services/api_service.dart'; | ||||||
| import 'package:oc_front/core/services/perms_service.dart'; | import 'package:oc_front/core/services/perms_service.dart'; | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
|  |  | ||||||
| class AuthService { | class AuthService { | ||||||
|   static var config = AppConfig(); |   static var isAuth = const bool.fromEnvironment('AUTH_MODE', defaultValue: false); | ||||||
|   static var isAuth = false; |   static const _clientID = String.fromEnvironment('CLIENT_ID', defaultValue: 'test-client'); | ||||||
|  |  | ||||||
|   AuthService() { |  | ||||||
|     isAuth = config.get('AUTH_MODE') == 'true'; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static APIService<SimpleData> service = APIService( |   static APIService<SimpleData> service = APIService( | ||||||
|     baseURL: |     baseURL: AppConfig().get('AUTH_HOST', defaultValue: 'http://localhost:8080/auth'), | ||||||
|         config.get('AUTH_HOST', defaultValue: 'http://localhost:8080/auth'), |  | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   static Future<void> init() async { |   static Future<void> init() async { | ||||||
| @@ -42,7 +36,8 @@ class AuthService { | |||||||
|     if (!isAuth) { |     if (!isAuth) { | ||||||
|       return true; |       return true; | ||||||
|     } |     } | ||||||
|     return (localStorage.getItem('accessToken') ?? "") != ""; |     return (localStorage.getItem('accessToken') ?? "") != ""  | ||||||
|  |     && localStorage.getItem('username') != ""; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static String? getUsername() { |   static String? getUsername() { | ||||||
| @@ -53,18 +48,17 @@ class AuthService { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future<void> login(String username, String password) async { |   static Future<void> login(String username, String password) async { | ||||||
|     var token = await service.post("/ldap/login", |     var token = await service.post("/login?client_id=$_clientID", <String, dynamic> { | ||||||
|         <String, dynamic>{"username": username, "password": password}, null); |       "username": username, | ||||||
|  |       "password": password | ||||||
|  |     }, null); | ||||||
|     if (token.code == 200 && token.data != null) { |     if (token.code == 200 && token.data != null) { | ||||||
|       localStorage.setItem('accessToken', token.data?.value['access_token']); |       localStorage.setItem('accessToken', token.data?.value['access_token']); | ||||||
|       localStorage.setItem('tokenType', token.data?.value['token_type']); |       localStorage.setItem('tokenType', token.data?.value['token_type']); | ||||||
|       localStorage.setItem('username', username); |       localStorage.setItem('username', username); | ||||||
|       localStorage.setItem( |       localStorage.setItem('expiresIn', DateTime.now().add( | ||||||
|           'expiresIn', |         Duration(seconds: token.data?.value['expires_in'])).toIso8601String()); | ||||||
|           DateTime.now() |       mainKey?.currentState?.setState(() {}); | ||||||
|               .add(Duration(seconds: token.data?.value['expires_in'])) |  | ||||||
|               .toIso8601String()); |  | ||||||
|       HeaderConstants.headerKey.currentState?.setState(() {}); |  | ||||||
|       PermsService.init(token.data?.value['access_token']); |       PermsService.init(token.data?.value['access_token']); | ||||||
|       refresh(token.data?.value['access_token'] ?? "", username, |       refresh(token.data?.value['access_token'] ?? "", username, | ||||||
|           Duration(seconds: token.data?.value['expires_in'])); |           Duration(seconds: token.data?.value['expires_in'])); | ||||||
| @@ -72,7 +66,7 @@ class AuthService { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future<void> logout() async { |   static Future<void> logout() async { | ||||||
|     var token = await service.delete("/ldap/logout", null); |     var token = await service.delete("/logout?client_id=$_clientID", null); | ||||||
|     if (token.code == 200) { |     if (token.code == 200) { | ||||||
|       localStorage.setItem('accessToken', ''); |       localStorage.setItem('accessToken', ''); | ||||||
|       localStorage.setItem('username', ''); |       localStorage.setItem('username', ''); | ||||||
| @@ -85,22 +79,18 @@ class AuthService { | |||||||
|     if (!isConnected()) { |     if (!isConnected()) { | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|     var isIntrospected = await service.get("/introspect", true, null); |     // ignore: invalid_return_type_for_catch_error | ||||||
|  |     var isIntrospected = await service.get("/introspect", true, null).catchError((e) => mainKey?.currentState?.setState(() {})); | ||||||
|     return isIntrospected.code == 200; |     return isIntrospected.code == 200; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future<void> refresh( |   static Future<void> refresh( | ||||||
|       String accessToken, String username, Duration duration) async { |       String accessToken, String username, Duration duration) async { | ||||||
|     Future.delayed(duration, () { |     Future.delayed(duration, () { | ||||||
|       service |       service.post("/refresh?client_id=$_clientID", <String, dynamic> { | ||||||
|           .post( |         "access_token": accessToken, | ||||||
|               "/refresh", |         "username": username | ||||||
|               <String, dynamic>{ |       }, null).then((token) { | ||||||
|                 "access_token": accessToken, |  | ||||||
|                 "username": username |  | ||||||
|               }, |  | ||||||
|               null) |  | ||||||
|           .then((token) { |  | ||||||
|         if (token.code == 200 && token.data != null) { |         if (token.code == 200 && token.data != null) { | ||||||
|           PermsService.init(token.data?.value['access_token']); |           PermsService.init(token.data?.value['access_token']); | ||||||
|           localStorage.setItem( |           localStorage.setItem( | ||||||
|   | |||||||
							
								
								
									
										49
									
								
								lib/core/services/enum_service.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								lib/core/services/enum_service.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | |||||||
|  | import 'package:oc_front/models/response.dart'; | ||||||
|  | import 'package:oc_front/core/services/api_service.dart'; | ||||||
|  |  | ||||||
|  | class EnumService { | ||||||
|  |   static final APIService<EnumData> _service = APIService<EnumData>( | ||||||
|  |     baseURL: const String.fromEnvironment('ITEM_HOST',  | ||||||
|  |       defaultValue: 'http://localhost:8080/catalog') | ||||||
|  |   ); | ||||||
|  |   static String subPath = "/enum/"; | ||||||
|  |   static Map<String, Map<String,dynamic>> enums = {}; | ||||||
|  |  | ||||||
|  |   static int? get(String path, dynamic name) { | ||||||
|  |     var n = enums[path]; | ||||||
|  |     if (n == null) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |     var names = "$name"; | ||||||
|  |     for (var nn in n.keys) { | ||||||
|  |       if (n[nn] == names || nn == names) { | ||||||
|  |         return int.parse(nn); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static void init() { | ||||||
|  |     _load("infrastructure"); | ||||||
|  |     _load("storage/type"); | ||||||
|  |     _load("storage/size"); | ||||||
|  |     _load("resource/type"); | ||||||
|  |     _load("booking/status"); | ||||||
|  |     _load("status"); | ||||||
|  |     _load("pricing/strategy/buy"); | ||||||
|  |     _load("pricing/strategy/data"); | ||||||
|  |     _load("pricing/strategy/time"); | ||||||
|  |     _load("pricing/strategy/storage"); | ||||||
|  |     _load("pricing/strategy/privilege/storage"); | ||||||
|  |     _load("pricing/strategy/privilege"); | ||||||
|  |     _load("pricing/refund/type"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static void _load(String name) { | ||||||
|  |     _service.get("$subPath$name", false, null).then((response) { | ||||||
|  |       if (response.code == 200) { | ||||||
|  |         enums[name] = response.data!.value; | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,5 +1,7 @@ | |||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
|  |  | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
|  |  | ||||||
| enum Perms {  | enum Perms {  | ||||||
|   SEARCH_INTERNAL,// ignore: constant_identifier_names |   SEARCH_INTERNAL,// ignore: constant_identifier_names | ||||||
|   SEARCH_EXTERNAL, // ignore: constant_identifier_names |   SEARCH_EXTERNAL, // ignore: constant_identifier_names | ||||||
| @@ -23,21 +25,21 @@ enum Perms { | |||||||
| } | } | ||||||
|  |  | ||||||
| Map<Perms, String> perms = { | Map<Perms, String> perms = { | ||||||
|   Perms.SEARCH_INTERNAL: 'GET__catalog_compute_search_search'.toUpperCase(), |   Perms.SEARCH_INTERNAL: 'GET__CATALOG_COMPUTE_SEARCH_SEARCH'.toUpperCase(), | ||||||
|   Perms.SEARCH_EXTERNAL: 'Search External'.toUpperCase(), |   Perms.SEARCH_EXTERNAL: 'Search External'.toUpperCase(), | ||||||
|   Perms.WORKSPACE_SHARE: 'POST__shared_collaborative_area_id_workspace_id2'.toUpperCase(), |   Perms.WORKSPACE_SHARE: 'POST__SHARED_COLLABORATIVE_AREA_ID_WORKSPACE_ID2'.toUpperCase(), | ||||||
|   Perms.WORKFLOW_CREATE: 'POST__workflow_'.toUpperCase(), |   Perms.WORKFLOW_CREATE: 'POST__WORKFLOW_'.toUpperCase(), | ||||||
|   Perms.WORKFLOW_UNSHARE: 'DELETE__shared_collaborative_area_id_workflow_id2'.toUpperCase(), |   Perms.WORKFLOW_UNSHARE: 'DELETE__SHARED_COLLABORATIVE_AREA_ID_WORKFLOW_ID2'.toUpperCase(), | ||||||
|   Perms.PEER_SHARE: 'POST__shared_collaborative_area_id_peer_id2'.toUpperCase(), |   Perms.PEER_SHARE: 'POST__SHARED_COLLABORATIVE_AREA_ID_PEER_ID2'.toUpperCase(), | ||||||
|   Perms.PEER_UNSHARE: 'DELETE__shared_collaborative_area_id_peer_id2'.toUpperCase(), |   Perms.PEER_UNSHARE: 'DELETE__SHARED_COLLABORATIVE_AREA_ID_PEER_ID2'.toUpperCase(), | ||||||
|   Perms.COLLABORATIVE_AREA_CREATE: 'POST__shared_collaborative_area_'.toUpperCase(), |   Perms.COLLABORATIVE_AREA_CREATE: 'POST__SHARED_COLLABORATIVE_AREA_'.toUpperCase(), | ||||||
|   Perms.COLLABORATIVE_AREA_EDIT: 'PUT__shared_collaborative_area_id'.toUpperCase(), |   Perms.COLLABORATIVE_AREA_EDIT: 'PUT__SHARED_COLLABORATIVE_AREA_ID'.toUpperCase(), | ||||||
|   Perms.COLLABORATIVE_AREA_DELETE: 'DELETE__shared_collaborative_area_id'.toUpperCase(), |   Perms.COLLABORATIVE_AREA_DELETE: 'DELETE__SHARED_COLLABORATIVE_AREA_ID'.toUpperCase(), | ||||||
|   Perms.WORKSPACE_UNSHARE: 'DELETE__shared_collaborative_area_id_workspace_id2'.toUpperCase(), |   Perms.WORKSPACE_UNSHARE: 'DELETE__SHARED_COLLABORATIVE_AREA_ID_WORKSPACE_ID2'.toUpperCase(), | ||||||
|   Perms.WORKFLOW_EDIT: 'PUT__workflow_id'.toUpperCase(), |   Perms.WORKFLOW_EDIT: 'PUT__WORKFLOW_ID'.toUpperCase(), | ||||||
|   Perms.WORKFLOW_DELETE: 'DELETE__workflow_id'.toUpperCase(), |   Perms.WORKFLOW_DELETE: 'DELETE__WORKFLOW_ID'.toUpperCase(), | ||||||
|   Perms.WORKFLOW_BOOKING: 'POST__datacenter_booking_'.toUpperCase(), |   Perms.WORKFLOW_BOOKING: 'POST__DATACENTER_BOOKING_'.toUpperCase(), | ||||||
|   Perms.WORKFLOW_SHARE: 'POST__shared_collaborative_area_id_workflow_id2'.toUpperCase(), |   Perms.WORKFLOW_SHARE: 'POST__SHARED_COLLABORATIVE_AREA_ID_WORKFLOW_ID2'.toUpperCase(), | ||||||
| }; | }; | ||||||
|  |  | ||||||
| class PermsService { | class PermsService { | ||||||
| @@ -77,6 +79,7 @@ class PermsService { | |||||||
|           _perms[w] = false; |           _perms[w] = false; | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  |       mainKey?.currentState?.setState(() {}); | ||||||
|     } catch (e) {/**/} |     } catch (e) {/**/} | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -57,7 +57,7 @@ class AppRouter { | |||||||
|   static final RouterItem workflowItem = RouterItem(icon: Icons.rebase_edit, label: "workflow manager", route: "workflow",  |   static final RouterItem workflowItem = RouterItem(icon: Icons.rebase_edit, label: "workflow manager", route: "workflow",  | ||||||
|       factory: WorkflowFactory()); |       factory: WorkflowFactory()); | ||||||
|   static final RouterItem workflowIDItem = RouterItem(description: "", help: "", route: "workflow/:id", factory: WorkflowFactory(), args: ["id"]); |   static final RouterItem workflowIDItem = RouterItem(description: "", help: "", route: "workflow/:id", factory: WorkflowFactory(), args: ["id"]); | ||||||
|   static final RouterItem catalogItem = RouterItem(description: "", help: "", route: "catalog/:id", factory: CatalogItemFactory(), args: ["id"]); |   static final RouterItem catalogItem = RouterItem(label: "resource", description: "", help: "", route: "catalog/:id", factory: CatalogItemFactory(), args: ["id"]); | ||||||
|   static final RouterItem catalog= RouterItem(icon: Icons.book_outlined, label: "catalog searcher", route: "catalog", factory: CatalogFactory()); |   static final RouterItem catalog= RouterItem(icon: Icons.book_outlined, label: "catalog searcher", route: "catalog", factory: CatalogFactory()); | ||||||
|   static final RouterItem scheduler = RouterItem(icon: Icons.schedule, label: "scheduled tasks", route: "scheduler", factory: SchedulerFactory()); |   static final RouterItem scheduler = RouterItem(icon: Icons.schedule, label: "scheduled tasks", route: "scheduler", factory: SchedulerFactory()); | ||||||
|   static final RouterItem compute = RouterItem(icon: Icons.dns_outlined, label: "my compute", route: "compute", factory: DatacenterFactory()); |   static final RouterItem compute = RouterItem(icon: Icons.dns_outlined, label: "my compute", route: "compute", factory: DatacenterFactory()); | ||||||
|   | |||||||
| @@ -4,34 +4,20 @@ import 'package:oc_front/core/services/specialized_services/abstract_service.dar | |||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
|  |  | ||||||
| class BookingExecutionService extends AbstractService<WorkflowExecutions> { | class BookingExecutionService extends AbstractService<WorkflowExecution> { | ||||||
|   @override |   @override APIService<WorkflowExecution> service = APIService<WorkflowExecution>( | ||||||
|   late final APIService<WorkflowExecutions> service; |     baseURL: const String.fromEnvironment('DATACENTER_HOST', defaultValue: 'http://localhost:8080/datacenter') | ||||||
|   @override |   ); | ||||||
|   String subPath = "/booking/"; |   @override String subPath = "/booking/"; | ||||||
|  |  | ||||||
|   BookingExecutionService() { |   @override Future<APIResponse<WorkflowExecution>> search(BuildContext? context, List<String> words, Map<String, dynamic> params) {   | ||||||
|     service = APIService<WorkflowExecutions>( |  | ||||||
|         baseURL: super.conf.get('DATACENTER_HOST', |  | ||||||
|             defaultValue: 'http://localhost:8080/datacenter')); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   Future<APIResponse<WorkflowExecutions>> search( |  | ||||||
|       BuildContext? context, List<String> words, Map<String, dynamic> params) { |  | ||||||
|     print("${subPath}search/${words.join("/")}"); |  | ||||||
|     return service.get("${subPath}search/${words.join("/")}", false, context); |     return service.get("${subPath}search/${words.join("/")}", false, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override Future<APIResponse<WorkflowExecution>> post(BuildContext? context, Map<String, dynamic> body, Map<String, String> params) {  | ||||||
|   Future<APIResponse<WorkflowExecutions>> post(BuildContext? context, |  | ||||||
|       Map<String, dynamic> body, Map<String, String> params) { |  | ||||||
|     return throw UnimplementedError(); |     return throw UnimplementedError(); | ||||||
|   } |   } | ||||||
|  |   @override Future<APIResponse<WorkflowExecution>> put(BuildContext? context, String id, Map<String, dynamic> body, Map<String, String> params) {  | ||||||
|   @override |  | ||||||
|   Future<APIResponse<WorkflowExecutions>> put(BuildContext? context, String id, |  | ||||||
|       Map<String, dynamic> body, Map<String, String> params) { |  | ||||||
|     return throw UnimplementedError(); |     return throw UnimplementedError(); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/core/services/api_service.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/abstract_service.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/response.dart'; | ||||||
|  |  | ||||||
|  | class DatacenterService extends AbstractService<Resource> { | ||||||
|  |   @override APIService<Resource> service = APIService<Resource>( | ||||||
|  |     baseURL: const String.fromEnvironment('DATACENTER_HOST', defaultValue: 'http://localhost:8080/datacenter') | ||||||
|  |   ); | ||||||
|  |   @override String subPath = "/"; | ||||||
|  |  | ||||||
|  |   @override Future<APIResponse<Resource>> search(BuildContext? context, List<String> words, Map<String, dynamic> params) {   | ||||||
|  |     return throw UnimplementedError(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Future<APIResponse<Resource>> post(BuildContext? context, Map<String, dynamic> body, Map<String, String> params) {  | ||||||
|  |     return throw UnimplementedError(); | ||||||
|  |   } | ||||||
|  |   @override Future<APIResponse<Resource>> put(BuildContext? context, String id, Map<String, dynamic> body, Map<String, String> params) {  | ||||||
|  |     return throw UnimplementedError(); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -27,12 +27,7 @@ class LogsService extends AbstractService<LogsResult> { | |||||||
|       } |       } | ||||||
|       v.add("$p=\"${params[p]}\""); |       v.add("$p=\"${params[p]}\""); | ||||||
|     }  |     }  | ||||||
|     print( |     return service.get("${subPath}query_range?query={${v.join(", ")}}&start=${params["start"].toString().substring(0, 10)}&end=${params["end"].toString().substring(0, 10)}", false, context); | ||||||
|         "${subPath}query_range?query={${v.join(", ")}}&start=${params["start"].toString().substring(0, 10)}&end=${params["end"].toString().substring(0, 10)}"); |  | ||||||
|     return service.get( |  | ||||||
|         "${subPath}query_range?query={${v.join(", ")}}&start=${params["start"].toString().substring(0, 10)}&end=${params["end"].toString().substring(0, 10)}", |  | ||||||
|         false, |  | ||||||
|         context); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:oc_front/core/services/api_service.dart'; |  | ||||||
| import 'package:oc_front/core/services/specialized_services/abstract_service.dart'; |  | ||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:oc_front/core/services/api_service.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/abstract_service.dart'; | ||||||
|  |  | ||||||
| class ResourceService extends AbstractService<Resource> { | class ResourceService extends AbstractService<Resource> { | ||||||
|   @override |   @override | ||||||
|   | |||||||
| @@ -5,11 +5,10 @@ import 'package:oc_front/models/response.dart'; | |||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
|  |  | ||||||
| class WorkflowExecutionService extends AbstractService<WorkflowExecutions> { | class WorkflowExecutionService extends AbstractService<WorkflowExecutions> { | ||||||
|   @override |   @override APIService<WorkflowExecutions> service = APIService<WorkflowExecutions>( | ||||||
|   late final APIService<WorkflowExecutions> service; |     baseURL: const String.fromEnvironment('SCHEDULER_HOST', defaultValue: 'http://localhost:8080/scheduler') | ||||||
|  |   ); | ||||||
|   @override |   @override String subPath = "/execution/"; | ||||||
|   String subPath = "/"; |  | ||||||
|  |  | ||||||
|   WorkflowExecutionService() { |   WorkflowExecutionService() { | ||||||
|     service = APIService<WorkflowExecutions>( |     service = APIService<WorkflowExecutions>( | ||||||
|   | |||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/core/services/api_service.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/abstract_service.dart'; | ||||||
|  | import 'package:oc_front/models/response.dart'; | ||||||
|  | import 'package:oc_front/models/workflow.dart'; | ||||||
|  |  | ||||||
|  | class SchedulerService extends AbstractService<WorkflowExecution> { | ||||||
|  |   @override APIService<WorkflowExecution> service = APIService<WorkflowExecution>( | ||||||
|  |     baseURL: const String.fromEnvironment('SCHEDULER_HOST', defaultValue: 'http://localhost:8080/scheduler') | ||||||
|  |   ); | ||||||
|  |   @override String subPath = "/"; | ||||||
|  |  | ||||||
|  |   Future<APIResponse<WorkflowExecution>> schedule(BuildContext? context, String id, Map<String, dynamic> body, Map<String, dynamic> params) {   | ||||||
|  |     print("$subPath$id"); | ||||||
|  |     return service.post("$subPath$id", body, context); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Future<APIResponse<WorkflowExecution>> search(BuildContext? context, List<String> words, Map<String, dynamic> params) {   | ||||||
|  |     return throw UnimplementedError(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Future<APIResponse<WorkflowExecution>> post(BuildContext? context, Map<String, dynamic> body, Map<String, String> params) {  | ||||||
|  |     return throw UnimplementedError(); | ||||||
|  |   } | ||||||
|  |   @override Future<APIResponse<WorkflowExecution>> put(BuildContext? context, String id, Map<String, dynamic> body, Map<String, String> params) {  | ||||||
|  |     return throw UnimplementedError(); | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										121
									
								
								lib/main.dart
									
									
									
									
									
								
							
							
						
						
									
										121
									
								
								lib/main.dart
									
									
									
									
									
								
							| @@ -9,6 +9,7 @@ import 'package:oc_front/core/sections/header/header.dart'; | |||||||
| import 'package:oc_front/core/sections/header/menu.dart'; | import 'package:oc_front/core/sections/header/menu.dart'; | ||||||
| import 'package:oc_front/core/sections/left_menu.dart'; | import 'package:oc_front/core/sections/left_menu.dart'; | ||||||
| import 'package:oc_front/core/services/auth.service.dart'; | import 'package:oc_front/core/services/auth.service.dart'; | ||||||
|  | import 'package:oc_front/core/services/enum_service.dart'; | ||||||
| import 'package:oc_front/core/services/router.dart'; | import 'package:oc_front/core/services/router.dart'; | ||||||
| import 'package:oc_front/core/sections/end_drawer.dart'; | import 'package:oc_front/core/sections/end_drawer.dart'; | ||||||
| import 'package:oc_front/widgets/dialog/login.dart'; | import 'package:oc_front/widgets/dialog/login.dart'; | ||||||
| @@ -35,7 +36,9 @@ class MyApp extends StatelessWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     AuthService.init(); |     AuthService.init(); | ||||||
|     return MaterialApp.router(routerConfig: GoRouter(routes: AppRouter.routes)); |     EnumService.init(); | ||||||
|  |     SearchConstants.clear(); | ||||||
|  |     return MaterialApp.router(  routerConfig: GoRouter( routes: AppRouter.routes ) ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -83,7 +86,7 @@ double getMainHeight(BuildContext context) { | |||||||
| double getMainWidth(BuildContext context) { | double getMainWidth(BuildContext context) { | ||||||
|   return getWidth(context) - 50; |   return getWidth(context) - 50; | ||||||
| } | } | ||||||
|  | bool loginIsSet = false; | ||||||
| class MainPageState extends State<MainPage> { | class MainPageState extends State<MainPage> { | ||||||
|   bool isCtrl = false; |   bool isCtrl = false; | ||||||
|   final FocusNode node = FocusNode(); |   final FocusNode node = FocusNode(); | ||||||
| @@ -104,14 +107,13 @@ class MainPageState extends State<MainPage> { | |||||||
|     // than having to individually change instances of widgets.i |     // than having to individually change instances of widgets.i | ||||||
|     scaffoldKey = GlobalKey<ScaffoldState>(); |     scaffoldKey = GlobalKey<ScaffoldState>(); | ||||||
|     isCtrl = false;  |     isCtrl = false;  | ||||||
|     if (!AuthService.isConnected()) { |     if (!AuthService.isConnected() && !loginIsSet) { | ||||||
|       print("isConnected: ${AuthService.isConnected()}"); |           Future.delayed(const Duration(milliseconds: 500), () { | ||||||
|       Future.delayed(const Duration(milliseconds: 500), () { |             loginIsSet = true; | ||||||
|         showDialog( |             showDialog( | ||||||
|             barrierDismissible: false, |                 barrierDismissible: false, | ||||||
|             context: context, |                 context: context ?? context, builder: (context) { | ||||||
|             builder: (context) { |                 return AlertDialog( | ||||||
|               return AlertDialog( |  | ||||||
|                   insetPadding: EdgeInsets.zero, |                   insetPadding: EdgeInsets.zero, | ||||||
|                   backgroundColor: Colors.white, |                   backgroundColor: Colors.white, | ||||||
|                   shape: RoundedRectangleBorder( |                   shape: RoundedRectangleBorder( | ||||||
| @@ -120,63 +122,46 @@ class MainPageState extends State<MainPage> { | |||||||
|             }); |             }); | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|     return FutureBuilder( |     return FutureBuilder(future: AuthService.init(),  | ||||||
|         future: AuthService.init(), |       builder: (e, s) { | ||||||
|         builder: (e, s) { |         WorkspaceLocal.init(context, false);         | ||||||
|           WorkspaceLocal.init(context, false); |         HeaderConstants.height = HeaderConstants.isNoHeader(AppRouter.currentRoute.route) || AppRouter.currentRoute.factory.searchFill() ? 50 : 100; | ||||||
|           CollaborativeAreaLocal.init(context, false); |         return Scaffold( key: scaffoldKey, endDrawer: EndDrawerWidget(), body:  | ||||||
|  |         SingleChildScrollView( | ||||||
|           HeaderConstants.height = |           controller: ScrollController(), | ||||||
|               HeaderConstants.isNoHeader(AppRouter.currentRoute.route) || |           scrollDirection: Axis.horizontal, | ||||||
|                       AppRouter.currentRoute.factory.searchFill() |           child: SingleChildScrollView( | ||||||
|                   ? 50 |             child: Column( children: [ | ||||||
|                   : 100; |               HeaderMenuWidget(), | ||||||
|           return Scaffold( |               Row( children : [ | ||||||
|               key: scaffoldKey, |                 Container( padding: const EdgeInsets.symmetric(vertical: 30), | ||||||
|               endDrawer: EndDrawerWidget(), |                   decoration: BoxDecoration( color: darkColor), | ||||||
|               body: SingleChildScrollView( |                   width: 50, height: getHeight(context) - 50,  | ||||||
|                   controller: ScrollController(), |                   child: SingleChildScrollView( child: LeftMenuWidget() )), | ||||||
|                   scrollDirection: Axis.horizontal, |                 SizedBox( width: getMainWidth(context), height: getHeight(context) - 50, | ||||||
|                   child: SingleChildScrollView( |                   child: KeyboardListener( | ||||||
|                       child: Column(children: [ |                     focusNode: node, | ||||||
|                     HeaderMenuWidget(), |                     onKeyEvent: (event) async { | ||||||
|                     Row(children: [ |                       if ( event.logicalKey == LogicalKeyboardKey.controlLeft ) { | ||||||
|                       Container( |                         isCtrl = (event is KeyDownEvent);  | ||||||
|                           padding: const EdgeInsets.symmetric(vertical: 30), |                         node.requestFocus(); | ||||||
|                           decoration: BoxDecoration(color: darkColor), |                       } else if( (event is KeyDownEvent) && event.logicalKey == LogicalKeyboardKey.enter) { | ||||||
|                           width: 50, |                         AppRouter.currentRoute.factory.search(context, isCtrl); | ||||||
|                           height: getHeight(context) - 50, |                         node.requestFocus(); | ||||||
|                           child: |                       } | ||||||
|                               SingleChildScrollView(child: LeftMenuWidget())), |                     }, | ||||||
|                       SizedBox( |                     child: Column( | ||||||
|                           width: getMainWidth(context), |                       mainAxisAlignment: MainAxisAlignment.start, | ||||||
|                           height: getHeight(context) - 50, |                       children: <Widget>[ | ||||||
|                           child: KeyboardListener( |                         HeaderWidget(), | ||||||
|                             focusNode: node, |                         widget.page ?? Container()  // CatalogPageWidget(), | ||||||
|                             onKeyEvent: (event) async { |                   ], | ||||||
|                               if (event.logicalKey == |                 ), | ||||||
|                                   LogicalKeyboardKey.controlLeft) { |               )), | ||||||
|                                 isCtrl = (event is KeyDownEvent); |             ]) | ||||||
|                                 node.requestFocus(); |           ]) | ||||||
|                               } else if ((event is KeyDownEvent) && |         ) | ||||||
|                                   event.logicalKey == |       )); | ||||||
|                                       LogicalKeyboardKey.enter) { |     }); | ||||||
|                                 AppRouter.currentRoute.factory |  | ||||||
|                                     .search(context, isCtrl); |  | ||||||
|                                 node.requestFocus(); |  | ||||||
|                               } |  | ||||||
|                             }, |  | ||||||
|                             child: Column( |  | ||||||
|                               mainAxisAlignment: MainAxisAlignment.start, |  | ||||||
|                               children: <Widget>[ |  | ||||||
|                                 HeaderWidget(), |  | ||||||
|                                 widget.page ?? |  | ||||||
|                                     Container() // CatalogPageWidget(), |  | ||||||
|                               ], |  | ||||||
|                             ), |  | ||||||
|                           )), |  | ||||||
|                     ]) |  | ||||||
|                   ])))); |  | ||||||
|         }); |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										222
									
								
								lib/models/resources/compute.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										222
									
								
								lib/models/resources/compute.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,222 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/core/services/enum_service.dart'; | ||||||
|  | import 'package:oc_front/models/abstract.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  |  | ||||||
|  | class ComputeItem extends AbstractItem<ComputePricing, ComputePartnership, ComputeInstance, ComputeItem> { | ||||||
|  |     // special attributes | ||||||
|  |   int? infrastructureEnum; | ||||||
|  |   String? architecture; | ||||||
|  |  | ||||||
|  |   ComputeItem({ | ||||||
|  |     this.infrastructureEnum, | ||||||
|  |     this.architecture | ||||||
|  |   }): super();   | ||||||
|  |  | ||||||
|  |   @override String get topic => "compute"; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic data) { | ||||||
|  |     try { data = data as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ComputeItem(); }    | ||||||
|  |     var w = ComputeItem( | ||||||
|  |       infrastructureEnum: data.containsKey("infrastructure") ? EnumService.get("infrastructure", data["infrastructure"]) : null, | ||||||
|  |       architecture: data.containsKey("architecture") && data["architecture"] != null ? data["architecture"] : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(data, ComputeInstance()); | ||||||
|  |     if (w.logo != null) { // get image dimensions | ||||||
|  |       var image = Image.network(w.logo!); | ||||||
|  |       image.image | ||||||
|  |         .resolve(const ImageConfiguration()) | ||||||
|  |         .addListener( | ||||||
|  |             ImageStreamListener( | ||||||
|  |                 (ImageInfo info, bool _) { | ||||||
|  |                   w.width = info.image.width.toDouble(); | ||||||
|  |                   w.height = info.image.height.toDouble(); | ||||||
|  |                 })); | ||||||
|  |     } | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "infrastructure": EnumService.enums["infrastructure"] != null  | ||||||
|  |       && EnumService.enums["infrastructure"]!["$infrastructureEnum"] != null ? EnumService.enums["infrastructure"]!["$infrastructureEnum"] : infrastructureEnum, | ||||||
|  |       "architecture": architecture, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     Map<String, dynamic> obj = infos(); | ||||||
|  |     obj["infrastructure"] = infrastructureEnum; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ComputeInstance extends AbstractInstance<ComputePricing, ComputePartnership> { | ||||||
|  |   String? securityLevel; | ||||||
|  |   List<String>? powerSources = []; | ||||||
|  |   double? annualEnergyConsumption; | ||||||
|  |   Map<String,CPU> cpus = {}; | ||||||
|  |   Map<String,GPU> gpus = {}; | ||||||
|  |   List<ComputeNode> nodes = []; | ||||||
|  |  | ||||||
|  |   ComputeInstance({ | ||||||
|  |     this.securityLevel, | ||||||
|  |     this.powerSources = const [], | ||||||
|  |     this.annualEnergyConsumption, | ||||||
|  |     this.cpus = const {}, | ||||||
|  |     this.gpus = const {}, | ||||||
|  |     this.nodes = const [], | ||||||
|  |   }): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "security_level": securityLevel, | ||||||
|  |       "power_sources": powerSources, | ||||||
|  |       "annual_co2_emissions": annualEnergyConsumption, | ||||||
|  |       "cpus": toMapJson(cpus), | ||||||
|  |       "gpus": toMapJson(gpus), | ||||||
|  |       "nodes": toListJson(nodes), | ||||||
|  |       "inputs": toListJson(inputs), | ||||||
|  |       "outputs": toListJson(outputs), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   ComputeInstance deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ComputeInstance(); } | ||||||
|  |     var w = ComputeInstance( | ||||||
|  |       securityLevel: json.containsKey("security_level") && json["security_level"] != null ? json["security_level"] : null, | ||||||
|  |       powerSources: json.containsKey("power_sources") && json["power_sources"] != null ? List<String>.from(json["power_sources"]) : [], | ||||||
|  |       annualEnergyConsumption: json.containsKey("annual_co2_emissions") && json["annual_co2_emissions"] != null ? json["annual_co2_emissions"] : null, | ||||||
|  |       //cpus: json.containsKey("cpus") && json["cpus"] != null ? fromMapJson(json["cpus"], CPU()) : {}, | ||||||
|  |       // gpus: json.containsKey("gpus") && json["gpus"] != null ? fromMapJson(json["gpus"], GPU()) : {}, | ||||||
|  |       //nodes: json.containsKey("nodes") && json["nodes"] != null ? fromListJson(json["nodes"], ComputeNode()) : [], | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json, ComputePartnership()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     var obj = infos(); | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ComputePartnership extends AbstractPartnerShip<ComputePricing> { | ||||||
|  |   Map<String, dynamic> maxAllowedCPUsCores = {}; | ||||||
|  |   Map<String, dynamic> maxAllowedGPUsMemoryGB = {}; | ||||||
|  |   double? maxAllowedRAM; | ||||||
|  |  | ||||||
|  |   ComputePartnership({ | ||||||
|  |     this.maxAllowedCPUsCores = const {}, | ||||||
|  |     this.maxAllowedGPUsMemoryGB = const {}, | ||||||
|  |     this.maxAllowedRAM, | ||||||
|  |   }): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   ComputePartnership deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ComputePartnership(); } | ||||||
|  |     var w = ComputePartnership( | ||||||
|  |       maxAllowedCPUsCores: json.containsKey("allowed_cpus") && json["allowed_cpus"] != null ? json["allowed_cpus"] : {}, | ||||||
|  |       maxAllowedGPUsMemoryGB: json.containsKey("allowed_gpus") && json["allowed_gpus"] != null ? json["allowed_gpus"] : {}, | ||||||
|  |       maxAllowedRAM: json.containsKey("allowed_ram") && json["allowed_ram"] != null ? double.parse("${json["allowed_ram"]}") : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json, ComputePricing()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     Map<String, dynamic> obj = { | ||||||
|  |       "allowed_cpus": maxAllowedCPUsCores, | ||||||
|  |       "allowed_gpus": maxAllowedGPUsMemoryGB, | ||||||
|  |       "allowed_ram": maxAllowedRAM, | ||||||
|  |     }; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ComputePricing extends AbstractPricing { | ||||||
|  |   Map<String, dynamic> cpusPrice = {}; | ||||||
|  |   Map<String, dynamic> gpusPrice = {}; | ||||||
|  |   double? ramPrice; | ||||||
|  |  | ||||||
|  |   ComputePricing({ | ||||||
|  |     this.cpusPrice = const {}, | ||||||
|  |     this.gpusPrice = const {}, | ||||||
|  |     this.ramPrice, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override ComputePricing deserialize(json) { | ||||||
|  |     var w = ComputePricing( | ||||||
|  |       cpusPrice: json.containsKey("cpus") && json["cpus"] != null ? json["cpus"] : {}, | ||||||
|  |       gpusPrice: json.containsKey("gpus") && json["gpus"] != null ? json["gpus"] : {}, | ||||||
|  |       ramPrice: json.containsKey("ram") && json["ram"] != null ? json["ram"] : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     var obj = { | ||||||
|  |       "cpus": cpusPrice, | ||||||
|  |       "gpus": gpusPrice, | ||||||
|  |       "ram": ramPrice, | ||||||
|  |     }; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ComputeNode extends SerializerDeserializer<ComputeNode> { | ||||||
|  |   String? name; | ||||||
|  |   int? quantity; | ||||||
|  |   Map<String,dynamic> cpus = {}; | ||||||
|  |   Map<String,dynamic> gpus = {}; | ||||||
|  |   RAM? ram; | ||||||
|  |  | ||||||
|  |   ComputeNode({ | ||||||
|  |     this.cpus = const {}, | ||||||
|  |     this.gpus = const {}, | ||||||
|  |     this.ram, | ||||||
|  |     this.name, | ||||||
|  |     this.quantity, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   ComputeNode deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ComputeNode(); } | ||||||
|  |     return ComputeNode( | ||||||
|  |       name: json.containsKey("name") && json["name"] != null ? json["name"] : null, | ||||||
|  |       quantity: json.containsKey("quantity") && json["quantity"] != null ? json["quantity"] : null, | ||||||
|  |       cpus: json.containsKey("cpus") && json["cpus"] != null ? fromMapJson(json["cpus"], CPU()) : {}, | ||||||
|  |       gpus: json.containsKey("gpus") && json["gpus"] != null ? fromMapJson(json["gpus"], GPU()) : {}, | ||||||
|  |       ram: json.containsKey("ram") && json["ram"] != null ? RAM().deserialize(json["ram"]) : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "name": name, | ||||||
|  |       "quantity": quantity, | ||||||
|  |       "cpus": cpus, | ||||||
|  |       "gpus": gpus, | ||||||
|  |       "ram": ram!.serialize(), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										166
									
								
								lib/models/resources/data.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								lib/models/resources/data.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,166 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/models/abstract.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  |  | ||||||
|  | class DataItem extends AbstractItem<DataPricing, DataPartnership, DataInstance, DataItem> { | ||||||
|  |   // special attributes | ||||||
|  |   String? type; | ||||||
|  |   String? source; | ||||||
|  |   String? quality; | ||||||
|  |   bool openData = false; | ||||||
|  |   bool static = false; | ||||||
|  |   bool personalData = false; | ||||||
|  |   bool anonymizedPersonalData = false; | ||||||
|  |   double? size; | ||||||
|  |   String? example; | ||||||
|  |   DateTime? updatePeriod; | ||||||
|  |    | ||||||
|  |   DataItem({ | ||||||
|  |     this.type, | ||||||
|  |     this.source, | ||||||
|  |     this.quality, | ||||||
|  |     this.openData = false, | ||||||
|  |     this.static = false, | ||||||
|  |     this.personalData = false, | ||||||
|  |     this.anonymizedPersonalData = false, | ||||||
|  |     this.size, | ||||||
|  |     this.example, | ||||||
|  |     this.updatePeriod, | ||||||
|  |   }): super();   | ||||||
|  |  | ||||||
|  |   @override String get topic => "data"; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic data) { | ||||||
|  |     try { data = data as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return DataItem(); }    | ||||||
|  |     var w = DataItem( | ||||||
|  |       type: data.containsKey("type") && data["type"] != null ? data["type"] : null, | ||||||
|  |       source: data.containsKey("source") && data["source"] != null ? data["source"] : null, | ||||||
|  |       quality: data.containsKey("quality") && data["quality"] != null ? data["quality"] : null, | ||||||
|  |       openData: data.containsKey("open_data") && data["open_data"] != null ? data["open_data"] : false, | ||||||
|  |       static: data.containsKey("static") && data["static"] != null ? data["static"] : false, | ||||||
|  |       personalData: data.containsKey("personal_data") && data["l"] != null ? data["personal_data"] : false, | ||||||
|  |       anonymizedPersonalData: data.containsKey("anonymized_personal_data") && data["anonymized_personal_data"] != null ? data["anonymized_personal_data"] : false, | ||||||
|  |       size: data.containsKey("size") && data["size"] != null ? data["size"] : null, | ||||||
|  |       example: data.containsKey("example") && data["example"] != null ? data["example"] : null, | ||||||
|  |       updatePeriod: data.containsKey("update_period") && data["update_period"] != null ? DateTime.parse(data["update_period"]) : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(data, DataInstance()); | ||||||
|  |     if (w.logo != null) { // get image dimensions | ||||||
|  |       var image = Image.network(w.logo!); | ||||||
|  |       image.image | ||||||
|  |         .resolve(const ImageConfiguration()) | ||||||
|  |         .addListener( | ||||||
|  |             ImageStreamListener( | ||||||
|  |                 (ImageInfo info, bool _) { | ||||||
|  |                   w.width = info.image.width.toDouble(); | ||||||
|  |                   w.height = info.image.height.toDouble(); | ||||||
|  |                 })); | ||||||
|  |     } | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "type": type, | ||||||
|  |       "quality": quality, | ||||||
|  |       "open_data": openData, | ||||||
|  |       "static": static, | ||||||
|  |       "personal_data": personalData, | ||||||
|  |       "anonymized_personal_data": anonymizedPersonalData, | ||||||
|  |       "size": size, | ||||||
|  |       "example": example, | ||||||
|  |       "update_period": updatePeriod?.toIso8601String(), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     var obj = infos(); | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class DataInstance extends AbstractInstance<DataPricing, DataPartnership> { | ||||||
|  |   String? source; | ||||||
|  |   DataInstance( | ||||||
|  |     {this.source} | ||||||
|  |   ): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "inputs": toListJson(inputs), | ||||||
|  |       "outputs": toListJson(outputs), | ||||||
|  |     }; | ||||||
|  |   }  | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   DataInstance deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return DataInstance(); } | ||||||
|  |     var w = DataInstance( | ||||||
|  |       source: json.containsKey("source") && json["source"] != null ? json["source"] : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json, DataPartnership()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     var obj = toJSON(); | ||||||
|  |     obj["source"] = source; | ||||||
|  |     obj.addAll(infos()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class DataPartnership extends AbstractPartnerShip<DataPricing> { | ||||||
|  |   double? maxDownloadableGBAllowed; | ||||||
|  |   bool personalDataAllowed = false; | ||||||
|  |   bool anonymizedPersonalDataAllowed = false; | ||||||
|  |  | ||||||
|  |   DataPartnership({ | ||||||
|  |     this.maxDownloadableGBAllowed, | ||||||
|  |     this.personalDataAllowed = false, | ||||||
|  |     this.anonymizedPersonalDataAllowed = false, | ||||||
|  |   }): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   DataPartnership deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return DataPartnership(); } | ||||||
|  |     var w = DataPartnership( | ||||||
|  |       maxDownloadableGBAllowed: json.containsKey("max_downloadable_gb_allowed") && json["max_downloadable_gb_allowed"] != null ? json["max_downloadable_gb_allowed"] : null, | ||||||
|  |       personalDataAllowed: json.containsKey("personal_data_allowed") && json["personal_data_allowed"] != null ? json["personal_data_allowed"] : false, | ||||||
|  |       anonymizedPersonalDataAllowed: json.containsKey("anonymized_personal_data_allowed") && json["anonymized_personal_data_allowed"] != null ? json["anonymized_personal_data_allowed"] : false, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json, DataPricing()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     Map<String, dynamic> obj = { | ||||||
|  |       "max_downloadable_gb_allowed": maxDownloadableGBAllowed, | ||||||
|  |       "personal_data_allowed": personalDataAllowed, | ||||||
|  |       "anonymized_personal_data_allowed": anonymizedPersonalDataAllowed, | ||||||
|  |     }; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class DataPricing extends AbstractPricing { | ||||||
|  |   @override DataPricing deserialize(json) { | ||||||
|  |     var w = DataPricing(); | ||||||
|  |     w.mapFromJSON(json); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										357
									
								
								lib/models/resources/processing.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										357
									
								
								lib/models/resources/processing.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,357 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/core/services/enum_service.dart'; | ||||||
|  | import 'package:oc_front/models/abstract.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  |  | ||||||
|  | class ProcessingItem extends AbstractItem<ProcessingPricing, ProcessingPartnership, ProcessingInstance, ProcessingItem> { | ||||||
|  |     // special attributes | ||||||
|  |   int? infrastructureEnum; | ||||||
|  |   bool isService = false; | ||||||
|  |   bool openSource = false; | ||||||
|  |   String? license; | ||||||
|  |   String? maturity; | ||||||
|  |   ProcessingUsage? usage;   | ||||||
|  |    | ||||||
|  |   ProcessingItem({ | ||||||
|  |     this.infrastructureEnum, | ||||||
|  |     this.isService = false, | ||||||
|  |     this.openSource = false, | ||||||
|  |     this.license, | ||||||
|  |     this.maturity, | ||||||
|  |     this.usage, | ||||||
|  |   }): super();   | ||||||
|  |  | ||||||
|  |   @override String get topic => "processing"; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic data) { | ||||||
|  |     try { data = data as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ProcessingItem(); }    | ||||||
|  |     var w = ProcessingItem( | ||||||
|  |       infrastructureEnum: data.containsKey("infrastructure") ? EnumService.get("infrastructure", data["infrastructure"]) : null, | ||||||
|  |       isService: data.containsKey("is_service") && data["is_service"] != null ? data["is_service"] : false, | ||||||
|  |       openSource: data.containsKey("open_source") && data["open_source"] != null ? data["open_source"] : false, | ||||||
|  |       license: data.containsKey("license") && data["license"] != null ? data["license"] : null, | ||||||
|  |       maturity: data.containsKey("maturity") && data["maturity"] != null ? data["maturity"] : null, | ||||||
|  |       usage: data.containsKey("usage") && data["usage"] != null ? ProcessingUsage().deserialize(data["usage"]) : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(data, ProcessingInstance()); | ||||||
|  |     if (w.logo != null) { // get image dimensions | ||||||
|  |       var image = Image.network(w.logo!); | ||||||
|  |       image.image | ||||||
|  |         .resolve(const ImageConfiguration()) | ||||||
|  |         .addListener( | ||||||
|  |             ImageStreamListener( | ||||||
|  |                 (ImageInfo info, bool _) { | ||||||
|  |                   w.width = info.image.width.toDouble(); | ||||||
|  |                   w.height = info.image.height.toDouble(); | ||||||
|  |                 })); | ||||||
|  |     } | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "infrastructure": EnumService.enums["infrastructure"] != null  | ||||||
|  |       && EnumService.enums["infrastructure"]!["$infrastructureEnum"] != null ?  | ||||||
|  |       EnumService.enums["infrastructure"]!["$infrastructureEnum"] : infrastructureEnum, | ||||||
|  |       "is_service": isService, | ||||||
|  |       "open_source": openSource, | ||||||
|  |       "license": license, | ||||||
|  |       "maturity": maturity, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     var obj = infos(); | ||||||
|  |     obj["infrastructure"] = infrastructureEnum; | ||||||
|  |     obj["usage"] = usage?.serialize(); | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ProcessingAccess extends SerializerDeserializer<ProcessingAccess> { | ||||||
|  |   Containered? container; | ||||||
|  |  | ||||||
|  |   ProcessingAccess({ | ||||||
|  |     this.container, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override ProcessingAccess deserialize(dynamic json) { | ||||||
|  |     try { | ||||||
|  |       json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { | ||||||
|  |       return ProcessingAccess(); | ||||||
|  |     } | ||||||
|  |     return ProcessingAccess( | ||||||
|  |       container: json.containsKey("container") && json["container"] != null ? Containered().deserialize(json["container"]) : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "container": container?.serialize(), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ProcessingInstance extends AbstractInstance<ProcessingPricing, ProcessingPartnership> { | ||||||
|  |   ProcessingAccess? access; | ||||||
|  |   ProcessingInstance( | ||||||
|  |     {this.access} | ||||||
|  |   ): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   ProcessingInstance deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ProcessingInstance(); } | ||||||
|  |     var w = ProcessingInstance(); | ||||||
|  |     w.access = json.containsKey("access") && json["access"] != null ? ProcessingAccess().deserialize(json['access']) : null; | ||||||
|  |     w.mapFromJSON(json, ProcessingPartnership()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     var obj = toJSON(); | ||||||
|  |     obj["access"] = access?.serialize(); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "inputs": toListJson(inputs), | ||||||
|  |       "outputs": toListJson(outputs), | ||||||
|  |     }; | ||||||
|  |   }  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ProcessingPartnership extends AbstractPartnerShip<ProcessingPricing> { | ||||||
|  |   ProcessingPartnership(): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   ProcessingPartnership deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return ProcessingPartnership(); } | ||||||
|  |     var w = ProcessingPartnership(); | ||||||
|  |     w.mapFromJSON(json, ProcessingPricing()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ProcessingPricing extends AbstractPricing { | ||||||
|  |   @override ProcessingPricing deserialize(json) { | ||||||
|  |     var w = ProcessingPricing(); | ||||||
|  |     w.mapFromJSON(json); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ProcessingUsage extends SerializerDeserializer<ProcessingUsage> { | ||||||
|  |     Map<String,CPU> cpus = {}; | ||||||
|  |     Map<String,GPU> gpus = {}; | ||||||
|  |     RAM? ram; | ||||||
|  |     double? storageSize; | ||||||
|  |     String? hypothesis; | ||||||
|  |     String? scalingModel; | ||||||
|  |  | ||||||
|  |     ProcessingUsage({ | ||||||
|  |       this.cpus = const {}, | ||||||
|  |       this.gpus = const {}, | ||||||
|  |       this.ram, | ||||||
|  |       this.storageSize, | ||||||
|  |       this.hypothesis, | ||||||
|  |       this.scalingModel, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     @override | ||||||
|  |   ProcessingUsage deserialize(json) { | ||||||
|  |     return ProcessingUsage( | ||||||
|  |       cpus: json.containsKey("cpus") && json["cpus"] != null ? fromMapJson(json["cpus"], CPU()) : {}, | ||||||
|  |       gpus: json.containsKey("gpus") && json["gpus"] != null ? fromMapJson(json["gpus"], GPU()) : {}, | ||||||
|  |       ram: json.containsKey("ram") && json["ram"] != null ? RAM().deserialize(json["ram"]) : null, | ||||||
|  |       storageSize: json.containsKey("storage_size") && json["storage_size"] != null ? json["storage_size"]?.toDouble() : null, | ||||||
|  |       hypothesis: json.containsKey("hypothesis") && json["hypothesis"] != null ? json["hypothesis"] : null, | ||||||
|  |       scalingModel: json.containsKey("scaling_model") && json["scaling_model"] != null ? json["scaling_model"] : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "cpus": toMapJson(cpus), | ||||||
|  |       "gpus": toMapJson(gpus), | ||||||
|  |       "ram": ram?.serialize(), | ||||||
|  |       "storage_size": storageSize, | ||||||
|  |       "hypothesis": hypothesis, | ||||||
|  |       "scaling_model": scalingModel, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class CPU extends SerializerDeserializer<CPU> { | ||||||
|  |   CPU({ | ||||||
|  |     this.cores, | ||||||
|  |     this.platform, | ||||||
|  |     this.architecture, | ||||||
|  |     this.minimumMemory, | ||||||
|  |     this.shared = false, | ||||||
|  |   }); | ||||||
|  |   double? cores; | ||||||
|  |   String? platform; | ||||||
|  |   bool shared = false; | ||||||
|  |   String? architecture; | ||||||
|  |   double? minimumMemory; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return CPU(); }   | ||||||
|  |     return CPU( | ||||||
|  |       cores: json.containsKey("cores") && json["cores"] != null ? json["cores"]?.toDouble() : null, | ||||||
|  |       platform: json.containsKey("platform") && json["platform"] != null ? json["platform"] : null, | ||||||
|  |       architecture: json.containsKey("architecture") && json["architecture"] != null ? json["architecture"] : null, | ||||||
|  |       minimumMemory: json.containsKey("minimumMemory") && json["minimumMemory"] != null ? json["minimumMemory"]?.toDouble() : null, | ||||||
|  |       shared: json.containsKey("shared") && json["shared"] != null ? json["shared"] : false, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() => { | ||||||
|  |     "cores": cores, | ||||||
|  |     "platform": platform, | ||||||
|  |     "architecture": architecture, | ||||||
|  |     "minimumMemory": minimumMemory, | ||||||
|  |     "shared": shared, | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  | class GPU extends SerializerDeserializer<GPU> { | ||||||
|  |   GPU({ | ||||||
|  |     this.cudaCores, | ||||||
|  |     this.memory, | ||||||
|  |     this.model, | ||||||
|  |     this.tensorCores, | ||||||
|  |   }); | ||||||
|  |   double? cudaCores; | ||||||
|  |   double? memory; | ||||||
|  |   String? model; | ||||||
|  |   double? tensorCores; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return GPU(); }   | ||||||
|  |     return GPU( | ||||||
|  |       cudaCores: json.containsKey("cuda_cores") && json["cuda_cores"] != null ? json["cuda_cores"]?.toDouble() : null, | ||||||
|  |       memory: json.containsKey("memory") && json["memory"] != null ? json["memory"]?.toDouble() : null, | ||||||
|  |       model: json.containsKey("model") && json["model"] != null ? json["model"] : null, | ||||||
|  |       tensorCores: json.containsKey("tensor_cores") && json["tensor_cores"] != null ? json["tensor_cores"]?.toDouble() : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() => { | ||||||
|  |     "cuda_cores": cudaCores, | ||||||
|  |     "memory": memory, | ||||||
|  |     "model": model, | ||||||
|  |     "tensor_cores": tensorCores, | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  | class RAM extends SerializerDeserializer<RAM> { | ||||||
|  |   RAM({ | ||||||
|  |     this.ecc = false, | ||||||
|  |     this.size, | ||||||
|  |   }); | ||||||
|  |   bool ecc = false; | ||||||
|  |   double? size; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return RAM(); }   | ||||||
|  |     return RAM( | ||||||
|  |       ecc: json.containsKey("ecc") && json["ecc"] != null ? json["ecc"] : false, | ||||||
|  |       size: json.containsKey("size") && json["size"] != null ? json["size"]?.toDouble() : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() => { | ||||||
|  |     "ecc": ecc, | ||||||
|  |     "size": size, | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Expose extends SerializerDeserializer<Expose> { | ||||||
|  |   Expose({ | ||||||
|  |     this.PAT, | ||||||
|  |     this.port, | ||||||
|  |     this.path, | ||||||
|  |   }); | ||||||
|  |    | ||||||
|  |   int? port; | ||||||
|  |   int? PAT; | ||||||
|  |   String? path; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Expose(); } | ||||||
|  |     return Expose( | ||||||
|  |       port: json.containsKey("port") && json["port"] != null ? json["port"] : null, | ||||||
|  |       PAT: json.containsKey("PAT") && json["PAT"] != null ? json["PAT"] : null, | ||||||
|  |       path: json.containsKey("reverse") && json["reverse"] != null ? json["reverse"] : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() => { | ||||||
|  |     "port": port, | ||||||
|  |     "PAT": PAT, | ||||||
|  |     "reverse": path, | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Containered extends SerializerDeserializer<Containered> { | ||||||
|  |   Containered({ | ||||||
|  |     this.image, | ||||||
|  |     this.args, | ||||||
|  |     this.command, | ||||||
|  |     this.env, | ||||||
|  |     this.volumes, | ||||||
|  |     this.exposes = const [], | ||||||
|  |   });  | ||||||
|  |  | ||||||
|  |   String? args; | ||||||
|  |   String? image; | ||||||
|  |   String? command; | ||||||
|  |   Map<String, dynamic>? env; | ||||||
|  |   Map<String, dynamic>? volumes; | ||||||
|  |   List<Expose> exposes = []; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Containered(); } | ||||||
|  |     return Containered( | ||||||
|  |       args: json.containsKey("args") && json["args"] != null ? json["args"] : null, | ||||||
|  |       image: json.containsKey("image") && json["image"] != null ? json["image"] : null, | ||||||
|  |       command: json.containsKey("command") && json["command"] != null ? json["command"] : null, | ||||||
|  |       env: json.containsKey("env") && json["env"] != null ? json["env"] : null, | ||||||
|  |       volumes: json.containsKey("volumes") && json["volumes"] != null ? json["volumes"] : null, | ||||||
|  |       exposes: json.containsKey("exposes") && json["exposes"] != null ? fromListJson(json["exposes"], Expose()) : [], | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     var w = { | ||||||
|  |       "args": args, | ||||||
|  |       "image": image, | ||||||
|  |       "command": command, | ||||||
|  |       "env": env, | ||||||
|  |       "volumes": volumes, | ||||||
|  |       "exposes": toListJson(exposes), | ||||||
|  |     }; | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										529
									
								
								lib/models/resources/resources.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										529
									
								
								lib/models/resources/resources.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,529 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:oc_front/models/abstract.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/data.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
|  | import 'package:oc_front/models/resources/workflow.dart'; | ||||||
|  |  | ||||||
|  | class Resource implements SerializerDeserializer<Resource> { | ||||||
|  |   List<DataItem> datas = []; | ||||||
|  |   List<ProcessingItem> processings = []; | ||||||
|  |   List<StorageItem> storages = []; | ||||||
|  |   List<ComputeItem> computes = []; | ||||||
|  |   List<WorkflowItem> workflows = []; | ||||||
|  |  | ||||||
|  |   Resource({ | ||||||
|  |     this.datas = const [], | ||||||
|  |     this.processings = const [], | ||||||
|  |     this.storages = const [], | ||||||
|  |     this.computes = const [], | ||||||
|  |     this.workflows = const [], | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override Resource deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Resource(); } | ||||||
|  |     return Resource( | ||||||
|  |       computes: json.containsKey("compute_resource") ? fromListJson(json["compute_resource"], ComputeItem()) : [], | ||||||
|  |       datas: json.containsKey("data_resource") ? fromListJson(json["data_resource"], DataItem()) : [], | ||||||
|  |       processings: json.containsKey("processing_resource") ? fromListJson(json["processing_resource"], ProcessingItem()) : [], | ||||||
|  |       storages: json.containsKey("storage_resource") ? fromListJson(json["storage_resource"], StorageItem()) : [], | ||||||
|  |       workflows: json.containsKey("workflow_resource") ? fromListJson(json["workflow_resource"], WorkflowItem()) : [], | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "compute_resource": toListJson<ComputeItem>(computes), | ||||||
|  |       "data_resource": toListJson<DataItem>(datas), | ||||||
|  |       "processing_resource": toListJson<ProcessingItem>(processings), | ||||||
|  |       "storage_resource": toListJson<StorageItem>(storages), | ||||||
|  |       "workflow_resource": toListJson<WorkflowItem>(workflows), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Owner extends SerializerDeserializer<Owner> { | ||||||
|  |   String? name; | ||||||
|  |   String? logo; | ||||||
|  |  | ||||||
|  |   Owner({ | ||||||
|  |     this.name, | ||||||
|  |     this.logo, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override Owner deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Owner(); } | ||||||
|  |     return Owner( | ||||||
|  |       name: json.containsKey("name") ? json["name"] : null, | ||||||
|  |       logo: json.containsKey("logo") ? json["logo"] : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "name": name, | ||||||
|  |       "logo": logo, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | abstract class Infos { | ||||||
|  |   Map<String, dynamic> infos(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Artifact extends SerializerDeserializer<Artifact> { | ||||||
|  |   String? attrPath; | ||||||
|  |   String? attrFrom; | ||||||
|  |   bool readOnly = true; | ||||||
|  |  | ||||||
|  |   Artifact({ | ||||||
|  |     this.attrPath, | ||||||
|  |     this.attrFrom, | ||||||
|  |     this.readOnly = true, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override Artifact deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Artifact(); } | ||||||
|  |     return Artifact( | ||||||
|  |       attrPath: json.containsKey("attr_path") ? json["attr_path"] : null, | ||||||
|  |       attrFrom: json.containsKey("attr_from") ? json["attr_from"] : null, | ||||||
|  |       readOnly: json.containsKey("readonly") ? json["readonly"] : true, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "attr_path": attrPath, | ||||||
|  |       "attr_from": attrFrom, | ||||||
|  |       "readonly": readOnly, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | class Param extends SerializerDeserializer<Param> { | ||||||
|  |   String? name; | ||||||
|  |   String? attr; | ||||||
|  |   dynamic value; | ||||||
|  |   String? origin; | ||||||
|  |   bool optionnal = false; | ||||||
|  |   bool readOnly = true; | ||||||
|  |  | ||||||
|  |   Param({ | ||||||
|  |     this.name, | ||||||
|  |     this.attr, | ||||||
|  |     this.value, | ||||||
|  |     this.origin, | ||||||
|  |     this.optionnal = false, | ||||||
|  |     this.readOnly = true, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override Param deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Param(); } | ||||||
|  |     return Param( | ||||||
|  |       name: json.containsKey("name") ? json["name"] : null, | ||||||
|  |       attr: json.containsKey("attr") ? json["attr"] : null, | ||||||
|  |       value: json.containsKey("value") ? json["value"] : null, | ||||||
|  |       origin: json.containsKey("origin") ? json["origin"] : null, | ||||||
|  |       optionnal: json.containsKey("optionnal") ? json["optionnal"] : false, | ||||||
|  |       readOnly: json.containsKey("readonly") ? json["readonly"] : false, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "name": name, | ||||||
|  |       "attr": attr, | ||||||
|  |       "value": value, | ||||||
|  |       "origin": origin, | ||||||
|  |       "optionnal": optionnal, | ||||||
|  |       "readonly": readOnly, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Credential extends SerializerDeserializer<Credential> { | ||||||
|  |   String? login; | ||||||
|  |   String? password; | ||||||
|  |  | ||||||
|  |   Credential({ | ||||||
|  |     this.login, | ||||||
|  |     this.password, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Credential deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>;  | ||||||
|  |     } catch (e) { return Credential(); } | ||||||
|  |     return Credential( | ||||||
|  |       login: json.containsKey("login") ? json["login"] : null, | ||||||
|  |       password: json.containsKey("password") ? json["password"] : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "login": login, | ||||||
|  |       "password": password, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | abstract class AbstractItem<X extends AbstractPricing, Y extends AbstractPartnerShip<X>, S extends AbstractInstance<X,Y>, T extends FlowData> extends FlowData implements SerializerDeserializer<T>, Infos { | ||||||
|  |   String? id; | ||||||
|  |   String? name; | ||||||
|  |   String? logo; | ||||||
|  |   String? type; | ||||||
|  |   String? creatorID; | ||||||
|  |   String? updaterID; | ||||||
|  |   DateTime? createdAt; | ||||||
|  |   DateTime? updatedAt; | ||||||
|  |   List<Owner> owners; | ||||||
|  |   String? description; | ||||||
|  |   String? restrictions; | ||||||
|  |   String? shortDescription; | ||||||
|  |   int selectedInstance = 0; | ||||||
|  |  | ||||||
|  |   List<AbstractInstance<X,Y>> instances = []; | ||||||
|  |  | ||||||
|  |   String get topic => ""; | ||||||
|  |  | ||||||
|  |   AbstractItem({ | ||||||
|  |     this.id, | ||||||
|  |     this.type, | ||||||
|  |     this.name, | ||||||
|  |     this.logo, | ||||||
|  |     this.creatorID, | ||||||
|  |     this.updaterID, | ||||||
|  |     this.createdAt, | ||||||
|  |     this.updatedAt, | ||||||
|  |     this.description, | ||||||
|  |     this.shortDescription, | ||||||
|  |     this.owners = const [], | ||||||
|  |     this.selectedInstance = 0, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   void addEnv(List<dynamic> infos) { | ||||||
|  |     var inst = getSelectedInstance(); | ||||||
|  |     if (inst == null) { return; } | ||||||
|  |     inst.env = []; | ||||||
|  |     for (var info in infos) { | ||||||
|  |       inst.env.add(Param(name: info["name"], attr: info["attr"], value: info["value"],  | ||||||
|  |       origin: info["origin"], optionnal: info["optionnal"], readOnly: info["readonly"])); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   AbstractInstance<X,Y>? getSelectedInstance() { | ||||||
|  |     if (selectedInstance == -1) { return instances.isEmpty ? null : instances[0]; } | ||||||
|  |     return  instances.isNotEmpty ? instances[selectedInstance] : null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override String getID() { | ||||||
|  |     return id ?? ""; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override String getType() { | ||||||
|  |     return type ?? ""; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   @override String getName() { | ||||||
|  |     return name ?? ""; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos(); | ||||||
|  |  | ||||||
|  |   double? width; | ||||||
|  |   double? height; | ||||||
|  |   @override | ||||||
|  |   double? getWidth() { | ||||||
|  |     return width; | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   double? getHeight() { | ||||||
|  |     return height; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, dynamic> setVariable(String key, dynamic value, Map<String, dynamic> map) { | ||||||
|  |     map[key] = value; | ||||||
|  |     return map; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   dynamic getVariable(String key, Map<String, dynamic> map) { | ||||||
|  |     return map[key]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, dynamic> toJSON() { | ||||||
|  |     return { | ||||||
|  |       "id": id, | ||||||
|  |       "type": type ?? topic, | ||||||
|  |       "name": name, | ||||||
|  |       "logo": logo, | ||||||
|  |       "owners": toListJson(owners), | ||||||
|  |       "creator_id": creatorID, | ||||||
|  |       "updater_id": updaterID, | ||||||
|  |       "creation_date": createdAt?.toIso8601String(), | ||||||
|  |       "update_date": updatedAt?.toIso8601String(), | ||||||
|  |       "description": description, | ||||||
|  |       "short_description": shortDescription, | ||||||
|  |       "selected_instance": selectedInstance, | ||||||
|  |       "instances": instances.map((e) => e.serialize()).toList(), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void mapFromJSON(dynamic json, S ex) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return; }    | ||||||
|  |     this.id = json.containsKey("id") ? json["id"] : null; | ||||||
|  |     this.type = json.containsKey("type") ? json["type"] : topic; | ||||||
|  |     this.name = json.containsKey("name") ? json["name"] : null; | ||||||
|  |     this.logo = json.containsKey("logo") ? json["logo"] : null; | ||||||
|  |     this.creatorID = json.containsKey("creator_id") ? json["creator_id"] : null; | ||||||
|  |     this.updaterID = json.containsKey("updater_id") ? json["updater_id"] : null; | ||||||
|  |     this.description = json.containsKey("description") ? json["description"] : null; | ||||||
|  |     this.owners = json.containsKey("owners") ? fromListJson(json["owners"], Owner()) : []; | ||||||
|  |     this.instances = json.containsKey("instances") ? fromListJson(json["instances"], ex) : []; | ||||||
|  |     this.updatedAt = json.containsKey("update_date") ? DateTime.parse(json["update_date"]) : null; | ||||||
|  |     this.selectedInstance = json.containsKey("selected_instance") ? json["selected_instance"] : 0; | ||||||
|  |     this.shortDescription = json.containsKey("short_description") ? json["short_description"] : null; | ||||||
|  |     this.createdAt = json.containsKey("creation_date") ? DateTime.parse(json["creation_date"]) : null; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Location extends SerializerDeserializer<Location> { | ||||||
|  |   double? latitude; | ||||||
|  |   double? longitude; | ||||||
|  |  | ||||||
|  |   Location({ | ||||||
|  |     this.latitude, | ||||||
|  |     this.longitude, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Location deserialize(json) { | ||||||
|  |     return Location( | ||||||
|  |       latitude: json.containsKey("latitude") ? json["latitude"] : null, | ||||||
|  |       longitude: json.containsKey("longitude") ? json["longitude"] : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "latitude": latitude, | ||||||
|  |       "longitude": longitude, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |    | ||||||
|  | abstract class AbstractInstance<X extends AbstractPricing, S extends AbstractPartnerShip<X>> extends SerializerDeserializer<AbstractInstance<X,S>> implements Infos { | ||||||
|  |   String? id; | ||||||
|  |   String? name;  | ||||||
|  |   int? countryCode; | ||||||
|  |   Location? location; | ||||||
|  |   List<S> partnerships = []; | ||||||
|  |   List<Param> env = []; | ||||||
|  |   List<Param> inputs = []; | ||||||
|  |   List<Param> outputs = []; | ||||||
|  |   Credential? credential; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   bool isEnv(String key) { | ||||||
|  |     for (var e in env) { | ||||||
|  |       if (e.name?.contains(key) ?? false || key.contains(e.name ?? "none")) { return true; } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   bool isEnvAttr(String attr, String origin, bool isOrigin) { | ||||||
|  |     for (var e in env) { | ||||||
|  |       if (e.attr == attr && ((isOrigin && e.origin != null) || (!isOrigin && e.origin == origin))) { return true; } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos(); | ||||||
|  |  | ||||||
|  |   void mapFromJSON(dynamic json, S ex) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return; }  | ||||||
|  |     this.countryCode = json.containsKey("country_code") ? json["country_code"] : null; | ||||||
|  |     this.id = json.containsKey("id") ? json["id"] : null; | ||||||
|  |     this.name = json.containsKey("name") ? json["name"] : null; | ||||||
|  |     this.env = json.containsKey("env") ? fromListJson(json["env"], Param()) : []; | ||||||
|  |     this.inputs = json.containsKey("inputs") ? fromListJson(json["inputs"], Param()) : []; | ||||||
|  |     this.outputs = json.containsKey("outputs") ? fromListJson(json["outputs"], Param()) : []; | ||||||
|  |     this.location = json.containsKey("location") ? Location().deserialize(json["location"]) : null; | ||||||
|  |     this.credential = json.containsKey("credential") ? Credential().deserialize(json["credential"]) : null; | ||||||
|  |     this.partnerships = json.containsKey("partnerships") ? fromListJson(json["partnerships"], ex) : []; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, dynamic> toJSON() { | ||||||
|  |     return { | ||||||
|  |       "country_code": countryCode, | ||||||
|  |       "id": id, | ||||||
|  |       "name": name, | ||||||
|  |       "location": location?.serialize(), | ||||||
|  |       "env": toListJson(env), | ||||||
|  |       "inputs": toListJson(inputs), | ||||||
|  |       "outputs": toListJson(outputs), | ||||||
|  |       "credential": credential?.serialize(), // TODO CREDENTIAL FORM | ||||||
|  |       "partnerships": partnerships.map((e) => e.serialize()).toList(), | ||||||
|  |     };  | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | abstract class AbstractPartnerShip<S extends AbstractPricing> extends SerializerDeserializer<AbstractPartnerShip<S>> { | ||||||
|  |   String? namespace; | ||||||
|  |   List<AbstractPricing> pricings = []; | ||||||
|  |  | ||||||
|  |   void mapFromJSON(dynamic json, S ex) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return; }  | ||||||
|  |     this.namespace = json.containsKey("namespace") ? json["namespace"] : null; | ||||||
|  |     this.pricings = json.containsKey("pricings") ? fromListJson(json["pricings"], ex) : []; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, dynamic> toJSON() { | ||||||
|  |     return { | ||||||
|  |       "namespace": namespace, | ||||||
|  |       "pricings": pricings, | ||||||
|  |     };  | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | abstract class AbstractPricing extends SerializerDeserializer<AbstractPricing> { | ||||||
|  |   PricingStrategy? pricing; | ||||||
|  |   int? refundTypeEnum; | ||||||
|  |   int? refundRatio; | ||||||
|  |   List<dynamic> additionnalRefundTypeEnum = []; | ||||||
|  |   int? privilegeStrategyEnum; | ||||||
|  |   int? garantedDelaySecond; | ||||||
|  |   bool exceeding = false; | ||||||
|  |   int? exceedingRatio; | ||||||
|  |  | ||||||
|  |   void mapFromJSON(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return; }  | ||||||
|  |     pricing = json.containsKey("pricing") ? PricingStrategy().deserialize(json["pricing"]) : null; | ||||||
|  |     refundTypeEnum = json.containsKey("refund_type") ? json["refund_type"] : null; | ||||||
|  |     refundRatio = json.containsKey("refund_ratio") ? json["refund_ratio"] : null; | ||||||
|  |     additionnalRefundTypeEnum = json.containsKey("additionnal_refund_type") ? json["additionnal_refund_type"] : []; | ||||||
|  |     privilegeStrategyEnum = json.containsKey("privilege_strategy") ? json["privilege_strategy"] : null; | ||||||
|  |     garantedDelaySecond = json.containsKey("garanted_delay") ? json["garanted_delay"] : null; | ||||||
|  |     exceeding = json.containsKey("exceeding") ? json["exceeding"] : false; | ||||||
|  |     exceedingRatio = json.containsKey("exceeding_ratio") ? json["exceeding_ratio"] : null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, dynamic> toJSON() { | ||||||
|  |     return { | ||||||
|  |       "pricing": pricing?.serialize(), | ||||||
|  |       "refund_type": refundTypeEnum, | ||||||
|  |       "refund_ratio": refundRatio, | ||||||
|  |       "additionnal_refund_type": additionnalRefundTypeEnum, | ||||||
|  |       "privilege_strategy": privilegeStrategyEnum, | ||||||
|  |       "garanted_delay": garantedDelaySecond, | ||||||
|  |       "exceeding": exceeding, | ||||||
|  |       "exceeding_ratio": exceedingRatio, | ||||||
|  |     };  | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class PricingStrategy extends SerializerDeserializer<PricingStrategy> { | ||||||
|  |   double? price; | ||||||
|  |   String? currency; | ||||||
|  |   int? buyingStrategyEnum; | ||||||
|  |   int? timeStrategyEnum; | ||||||
|  |   int? overrideStrategyEnum; | ||||||
|  |  | ||||||
|  |   PricingStrategy({ | ||||||
|  |     this.price, | ||||||
|  |     this.currency, | ||||||
|  |     this.buyingStrategyEnum, | ||||||
|  |     this.timeStrategyEnum, | ||||||
|  |     this.overrideStrategyEnum, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   PricingStrategy deserialize(dynamic json) { | ||||||
|  |     return PricingStrategy( | ||||||
|  |       price: json.containsKey("price") && json["price"] != null ? json["price"] : null, | ||||||
|  |       currency: json.containsKey("currency") && json["currency"] != null  ? json["currency"] : null, | ||||||
|  |       buyingStrategyEnum: json.containsKey("buying_strategy") ? json["buying_strategy"] : null, | ||||||
|  |       timeStrategyEnum: json.containsKey("time_strategy") ? json["time_strategy"] : null, | ||||||
|  |       overrideStrategyEnum: json.containsKey("override_strategy") ? json["override_strategy"] : null, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "price": price, | ||||||
|  |       "currency": currency, | ||||||
|  |       "buying_strategy": buyingStrategyEnum, | ||||||
|  |       "time_strategy": timeStrategyEnum, | ||||||
|  |       "override_strategy": overrideStrategyEnum, | ||||||
|  |     };  | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class Model extends SerializerDeserializer<Model> { | ||||||
|  |   dynamic value; | ||||||
|  |   String? type; | ||||||
|  |   bool readonly = false; | ||||||
|  |  | ||||||
|  |   Model({ | ||||||
|  |     this.value, | ||||||
|  |     this.type, | ||||||
|  |     this.readonly = false, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Model(); } | ||||||
|  |     return Model( | ||||||
|  |       value: json.containsKey("value") ? json["value"] : null, | ||||||
|  |       type: json.containsKey("type") ? json["type"] : null, | ||||||
|  |       readonly: json.containsKey("readonly") ? json["readonly"] : false, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() => { | ||||||
|  |     "value": value, | ||||||
|  |     "type": type, | ||||||
|  |     "readonly": readonly, | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Type? getTopicType(String topic) { | ||||||
|  |   if (topic == "processing") { return ProcessingItem; } | ||||||
|  |   else if (topic == "data") { return DataItem; } | ||||||
|  |   else if (topic == "compute") { return ComputeItem; } | ||||||
|  |   else if (topic == "storage") { return StorageItem; } | ||||||
|  |   else if (topic == "workflow") { return WorkflowItem; } | ||||||
|  |   else { return null; } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | String getTopic(Type type) { | ||||||
|  |   if (type == AbstractItem) { return "resource"; } | ||||||
|  |   if (type == ProcessingItem) { return "processing"; } | ||||||
|  |   if (type == DataItem) { return "data"; } | ||||||
|  |   if (type == ComputeItem) { return "compute"; } | ||||||
|  |   if (type == StorageItem) { return "storage"; } | ||||||
|  |   if (type == WorkflowItem) { return "workflow"; } | ||||||
|  |   return ""; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool isComputing(String topic) => topic == "processing"; | ||||||
|  | bool isData(String topic) => topic == "data"; | ||||||
|  | bool isCompute(String topic) => topic == "compute"; | ||||||
|  | bool isStorage(String topic) => topic == "storage"; | ||||||
|  | bool isWorkflow(String topic) => topic == "workflow"; | ||||||
|  |  | ||||||
|  | Color getColor(String topic) => isData(topic) ? Colors.blue : isComputing(topic) ? Colors.green :  | ||||||
|  |                                 isCompute(topic) ? Colors.orange : isStorage(topic) ? redColor : Colors.grey; | ||||||
							
								
								
									
										166
									
								
								lib/models/resources/storage.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								lib/models/resources/storage.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,166 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/core/services/enum_service.dart'; | ||||||
|  | import 'package:oc_front/models/abstract.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  |  | ||||||
|  | class StorageItem extends AbstractItem<StoragePricing, StoragePartnership, StorageInstance, StorageItem> { | ||||||
|  |   StorageItem({ | ||||||
|  |     this.acronym, | ||||||
|  |     this.typeEnum, | ||||||
|  |   }): super();   | ||||||
|  |  | ||||||
|  |  @override String get topic => "storage"; | ||||||
|  |   // special attributes | ||||||
|  |   String? acronym; | ||||||
|  |   int? typeEnum; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic data) { | ||||||
|  |     try { data = data as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return StorageItem(); }    | ||||||
|  |     var w = StorageItem( | ||||||
|  |       acronym: data.containsKey("acronym") && data["acronym"] != null ? data["acronym"] : null, | ||||||
|  |       typeEnum: data.containsKey("storage_type") && data["storage_type"] != null ? EnumService.get("storage/type", data["storage_type"]) : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(data, StorageInstance()); | ||||||
|  |     if (w.logo != null) { // get image dimensions | ||||||
|  |       var image = Image.network(w.logo!); | ||||||
|  |       image.image | ||||||
|  |         .resolve(const ImageConfiguration()) | ||||||
|  |         .addListener( | ||||||
|  |             ImageStreamListener( | ||||||
|  |                 (ImageInfo info, bool _) { | ||||||
|  |                   w.width = info.image.width.toDouble(); | ||||||
|  |                   w.height = info.image.height.toDouble(); | ||||||
|  |                 })); | ||||||
|  |     } | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "acronym": acronym, | ||||||
|  |       "storage_type": EnumService.enums["storage/type"] != null  | ||||||
|  |         && EnumService.enums["storage/type"]!["$typeEnum"] != null ?  | ||||||
|  |         EnumService.enums["storage/type"]!["$typeEnum"] : typeEnum, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     var obj = infos(); | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class StorageInstance extends AbstractInstance<StoragePricing, StoragePartnership> { | ||||||
|  |   String? source; | ||||||
|  |   bool local = false; | ||||||
|  |   String? securityLevel; | ||||||
|  |   int? storageSizeEnum; | ||||||
|  |   int? size; | ||||||
|  |   bool encryption = false; | ||||||
|  |   String? redundancy; | ||||||
|  |   String? throughput; | ||||||
|  |  | ||||||
|  |   StorageInstance({ | ||||||
|  |     this.source, | ||||||
|  |     this.local = false, | ||||||
|  |     this.securityLevel, | ||||||
|  |     this.storageSizeEnum, | ||||||
|  |     this.size, | ||||||
|  |     this.encryption = false, | ||||||
|  |     this.redundancy, | ||||||
|  |     this.throughput, | ||||||
|  |   }): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   StorageInstance deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return StorageInstance(); } | ||||||
|  |     var w = StorageInstance( | ||||||
|  |       source: json.containsKey("source") && json["source"] != null ? json["source"] : null, | ||||||
|  |       local: json.containsKey("local") && json["local"] != null ? json["local"] : false, | ||||||
|  |       securityLevel: json.containsKey("security_level") && json["security_level"] != null ? json["security_level"] : null, | ||||||
|  |       storageSizeEnum: json.containsKey("size_type") ? EnumService.get("storage/size", json["size_type"]) : null, | ||||||
|  |       size: json.containsKey("size") && json["size"] != null ? json["size"] : null, | ||||||
|  |       encryption: json.containsKey("encryption") && json["encryption"] != null ? json["encryption"] : false, | ||||||
|  |       redundancy: json.containsKey("redundancy") && json["redundancy"] != null ? json["redundancy"] : null, | ||||||
|  |       throughput: json.containsKey("throughput") && json["throughput"] != null ? json["throughput"] : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json, StoragePartnership()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return { | ||||||
|  |       "local": local, | ||||||
|  |       "security_level": securityLevel, | ||||||
|  |       "size_type": EnumService.enums["storage/size"] != null  | ||||||
|  |       && EnumService.enums["storage/size"]!["$storageSizeEnum"] != null ?  | ||||||
|  |       EnumService.enums["storage/size"]!["$storageSizeEnum"] : storageSizeEnum, | ||||||
|  |       "size": size, | ||||||
|  |       "encryption": encryption, | ||||||
|  |       "redundancy": redundancy, | ||||||
|  |       "throughput": throughput, | ||||||
|  |       "inputs": toListJson(inputs), | ||||||
|  |       "outputs": toListJson(outputs), | ||||||
|  |     }; | ||||||
|  |   }  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     var obj = infos(); | ||||||
|  |     obj["source"] = source; | ||||||
|  |     obj["size_type"] = storageSizeEnum; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class StoragePartnership extends AbstractPartnerShip<StoragePricing> { | ||||||
|  |   double? maxSizeGBAllowed; | ||||||
|  |   bool onlyEncryptedAllowed = false; | ||||||
|  |  | ||||||
|  |   StoragePartnership({ | ||||||
|  |     this.maxSizeGBAllowed, | ||||||
|  |     this.onlyEncryptedAllowed = false, | ||||||
|  |   }): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   StoragePartnership deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return StoragePartnership(); } | ||||||
|  |     var w = StoragePartnership( | ||||||
|  |       maxSizeGBAllowed: json.containsKey("allowed_gb") && json["allowed_gb"] != null ? json["allowed_gb"] : null, | ||||||
|  |       onlyEncryptedAllowed: json.containsKey("personal_data_allowed") && json["personal_data_allowed"] != null ? json["personal_data_allowed"] : false, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(json, StoragePricing()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     Map<String, dynamic> obj = { | ||||||
|  |       "allowed_gb": maxSizeGBAllowed, | ||||||
|  |       "personal_data_allowed": onlyEncryptedAllowed, | ||||||
|  |     }; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class StoragePricing extends AbstractPricing { | ||||||
|  |   @override StoragePricing deserialize(json) { | ||||||
|  |     var w = StoragePricing(); | ||||||
|  |     w.mapFromJSON(json); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										101
									
								
								lib/models/resources/workflow.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								lib/models/resources/workflow.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,101 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  |  | ||||||
|  | class WorkflowItem extends AbstractItem<WorkflowPricing, WorkflowPartnership, WorkflowInstance, WorkflowItem> { | ||||||
|  |   // special attributes | ||||||
|  |   String? workflowID; | ||||||
|  |    | ||||||
|  |   WorkflowItem({ | ||||||
|  |     this.workflowID, | ||||||
|  |   }): super();   | ||||||
|  |  | ||||||
|  |   @override String get topic => "workflow"; | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic data) { | ||||||
|  |     try { data = data as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return WorkflowItem(); }    | ||||||
|  |     var w = WorkflowItem( | ||||||
|  |       workflowID: data.containsKey("workflow_id") && data["workflow_id"] != null ? data["workflow_id"] : null, | ||||||
|  |     ); | ||||||
|  |     w.mapFromJSON(data, WorkflowInstance()); | ||||||
|  |     if (w.logo != null) { // get image dimensions | ||||||
|  |       var image = Image.network(w.logo!); | ||||||
|  |       image.image | ||||||
|  |         .resolve(const ImageConfiguration()) | ||||||
|  |         .addListener( | ||||||
|  |             ImageStreamListener( | ||||||
|  |                 (ImageInfo info, bool _) { | ||||||
|  |                   w.width = info.image.width.toDouble(); | ||||||
|  |                   w.height = info.image.height.toDouble(); | ||||||
|  |                 })); | ||||||
|  |     } | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return {}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     Map<String, dynamic> obj ={ | ||||||
|  |       "workflow_id": workflowID, | ||||||
|  |     }; | ||||||
|  |     obj.addAll(toJSON()); | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class WorkflowInstance extends AbstractInstance<WorkflowPricing, WorkflowPartnership> { | ||||||
|  |   WorkflowInstance(): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   WorkflowInstance deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return WorkflowInstance(); } | ||||||
|  |     var w = WorkflowInstance(); | ||||||
|  |     w.mapFromJSON(json, WorkflowPartnership()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> infos() { | ||||||
|  |     return {}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class WorkflowPartnership extends AbstractPartnerShip<WorkflowPricing> { | ||||||
|  |   WorkflowPartnership(): super(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   WorkflowPartnership deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return WorkflowPartnership(); } | ||||||
|  |     var w = WorkflowPartnership(); | ||||||
|  |     w.mapFromJSON(json, WorkflowPricing()); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class WorkflowPricing extends AbstractPricing { | ||||||
|  |   @override WorkflowPricing deserialize(json) { | ||||||
|  |     var w = WorkflowPricing(); | ||||||
|  |     w.mapFromJSON(json); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |   @override | ||||||
|  |   Map<String, dynamic> serialize() { | ||||||
|  |     return toJSON(); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,5 +1,10 @@ | |||||||
| import 'package:oc_front/models/abstract.dart'; | import 'package:oc_front/models/abstract.dart'; | ||||||
| import 'package:oc_front/models/logs.dart'; | import 'package:oc_front/models/logs.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/data.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:oc_front/models/search.dart'; | ||||||
| import 'package:oc_front/models/shared.dart'; | import 'package:oc_front/models/shared.dart'; | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
| @@ -20,6 +25,7 @@ Map<Type, SerializerDeserializer> refs = <Type, SerializerDeserializer> { | |||||||
|   Check: Check(), |   Check: Check(), | ||||||
|   CollaborativeArea: CollaborativeArea(), |   CollaborativeArea: CollaborativeArea(), | ||||||
|   SimpleData: SimpleData(), |   SimpleData: SimpleData(), | ||||||
|  |   EnumData: EnumData(), | ||||||
| }; | }; | ||||||
|  |  | ||||||
| class APIResponse<T extends SerializerDeserializer> { | class APIResponse<T extends SerializerDeserializer> { | ||||||
| @@ -53,7 +59,9 @@ class APIResponse<T extends SerializerDeserializer> { | |||||||
|         code: j.containsKey("code") && j["code"] != null ? j["code"] : 200, |         code: j.containsKey("code") && j["code"] != null ? j["code"] : 200, | ||||||
|         error: j.containsKey("error") && j["error"] != null ? j["error"] : "", |         error: j.containsKey("error") && j["error"] != null ? j["error"] : "", | ||||||
|       ); |       ); | ||||||
|     } catch (e) { return APIResponse<T>( data: refs[T]!.deserialize(data), ); } |     } catch (e) {  | ||||||
|  |       return APIResponse<T>( data: refs[T]?.deserialize(data), );  | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -66,17 +74,34 @@ class SimpleData extends SerializerDeserializer<SimpleData> { | |||||||
|   @override Map<String, dynamic> serialize() => { }; |   @override Map<String, dynamic> serialize() => { }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class EnumData extends SerializerDeserializer<EnumData> { | ||||||
|  |   EnumData({ this.value = const {} }); | ||||||
|  |   Map<String, dynamic> value = {}; | ||||||
|  |   @override deserialize(dynamic json) {  | ||||||
|  |     return EnumData(value: json as Map<String, dynamic>);  | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() => { }; | ||||||
|  | } | ||||||
|  |  | ||||||
| class RawData extends SerializerDeserializer<RawData> { | class RawData extends SerializerDeserializer<RawData> { | ||||||
|   RawData({ this.values = const []}); |   RawData({ this.values = const []}); | ||||||
|   List<dynamic> values = []; |   List<dynamic> values = []; | ||||||
|   @override deserialize(dynamic json) {  |   @override deserialize(dynamic json) {  | ||||||
|     return RawData(values: json ?? []); } |     try { | ||||||
|  |       return RawData(values: json ?? []);  | ||||||
|  |     } catch (e) {  | ||||||
|  |       print(e); | ||||||
|  |       print(json); | ||||||
|  |       return RawData();  | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   @override Map<String, dynamic> serialize() => { }; |   @override Map<String, dynamic> serialize() => { }; | ||||||
| } | } | ||||||
|  |  | ||||||
| abstract class ShallowData { | abstract class ShallowData { | ||||||
|   String getID(); |   String getID(); | ||||||
|   String getName(); |   String getName(); | ||||||
|  |   Map<String, dynamic> serialize(); | ||||||
| } | } | ||||||
|  |  | ||||||
| class Shallow { | class Shallow { | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								lib/models/schedule.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								lib/models/schedule.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | import 'package:oc_front/models/abstract.dart'; | ||||||
|  |  | ||||||
|  | class Scheduler extends SerializerDeserializer<Scheduler> { | ||||||
|  |   DateTime? start; | ||||||
|  |   DateTime? end; | ||||||
|  |   double? duration; | ||||||
|  |   String? cron; | ||||||
|  |  | ||||||
|  |   Scheduler({ | ||||||
|  |     this.start, | ||||||
|  |     this.end, | ||||||
|  |     this.duration, | ||||||
|  |     this.cron, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Scheduler deserialize(json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return Scheduler(); } | ||||||
|  |     var w = Scheduler( | ||||||
|  |       start: json.containsKey("start") && json["start"] != null ? DateTime.parse(json["start"]) : null, | ||||||
|  |       end: json.containsKey("end") && json["end"] != null ? DateTime.parse(json["end"]) : null, | ||||||
|  |       duration: json.containsKey("duration_s") && json["duration_s"] != null ? json["duration_s"] : -1, | ||||||
|  |       cron: json.containsKey("cron") && json["cron"] != null ? json["cron"] : null, | ||||||
|  |     ); | ||||||
|  |     return w; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "start": start?.toIso8601String(), | ||||||
|  |       "end": end?.toIso8601String(), | ||||||
|  |       "duration_s": duration, | ||||||
|  |       "cron": cron, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -79,13 +79,14 @@ class CollaborativeArea extends SerializerDeserializer<CollaborativeArea> { | |||||||
|       rule : json.containsKey("collaborative_area") ? CollaborativeAreaRule().deserialize(json["collaborative_area"]) : CollaborativeAreaRule(), |       rule : json.containsKey("collaborative_area") ? CollaborativeAreaRule().deserialize(json["collaborative_area"]) : CollaborativeAreaRule(), | ||||||
|       id: json.containsKey("id") ? json["id"] : null, |       id: json.containsKey("id") ? json["id"] : null, | ||||||
|       name: json.containsKey("name") ? json["name"] : null, |       name: json.containsKey("name") ? json["name"] : null, | ||||||
|  |       creatorID: json.containsKey("creator_id") ? json["creator_id"] : null, | ||||||
|       description: json.containsKey("description") ? json["description"] : null, |       description: json.containsKey("description") ? json["description"] : null, | ||||||
|       version: json.containsKey("version") ? json["version"] : null, |       version: json.containsKey("version") ? json["version"] : null, | ||||||
|       attributes: json.containsKey("attributes") ? json["attributes"] : {}, |       attributes: json.containsKey("attributes") ? json["attributes"] : {}, | ||||||
|       workspaces: json.containsKey("shared_workspaces") ? fromListJson(json["shared_workspaces"], Workspace()) : [], |       workspaces: json.containsKey("shared_workspaces") && json["shared_workspaces"] != null ? fromListJson(json["shared_workspaces"], Workspace()) : [], | ||||||
|       workflows: json.containsKey("shared_workflows") ? fromListJson(json["shared_workflows"], Workflow()) : [], |       workflows: json.containsKey("shared_workflows") && json["shared_workflows"] != null ? fromListJson(json["shared_workflows"], Workflow()) : [], | ||||||
|       peers: json.containsKey("shared_peers") ? fromListJson(json["shared_peers"], Peer()) : [], |       peers: json.containsKey("shared_peers") && json["shared_peers"] != null ? fromListJson(json["shared_peers"], Peer()) : [], | ||||||
|       rules: json.containsKey("rules") ? json["rules"] : [], |       rules: json.containsKey("shared_rules") && json["shared_rules"] != null ? json["shared_rules"] : [], | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override |   @override | ||||||
|   | |||||||
| @@ -1,39 +1,40 @@ | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_colorpicker/flutter_colorpicker.dart'; | import 'package:flutter_colorpicker/flutter_colorpicker.dart'; | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; |  | ||||||
| import 'package:oc_front/models/abstract.dart'; | import 'package:oc_front/models/abstract.dart'; | ||||||
| import 'package:oc_front/models/logs.dart'; | import 'package:oc_front/models/logs.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/data.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
|  | import 'package:oc_front/models/resources/workflow.dart'; | ||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:oc_front/widgets/forms/sub_keys_forms.dart'; | ||||||
|  |  | ||||||
| class Check extends SerializerDeserializer<Check> { | class Check extends SerializerDeserializer<Check> { | ||||||
|   bool is_available = false; |   bool isAvailable = false; | ||||||
|  |  | ||||||
|   Check({ |   Check({ | ||||||
|     this.is_available = false, |     this.isAvailable = false, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override deserialize(dynamic json) { |   @override deserialize(dynamic json) { | ||||||
|     try { json = json as Map<String, dynamic>; |     try { json = json as Map<String, dynamic>; | ||||||
|     } catch (e) { return Check(); }    |     } catch (e) { return Check(); }    | ||||||
|     return Check( |     return Check( | ||||||
|       is_available: json.containsKey("is_available") ? json["is_available"] : false, |       isAvailable: json.containsKey("is_available") ? json["is_available"] : false, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override Map<String, dynamic> serialize() { |   @override Map<String, dynamic> serialize() { | ||||||
|     return { |     return { | ||||||
|       "is_available": is_available, |       "is_available": isAvailable, | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| class WorkflowExecutions extends SerializerDeserializer<WorkflowExecutions> { | class WorkflowExecutions extends SerializerDeserializer<WorkflowExecutions> { | ||||||
|   List<WorkflowExecution> executions = []; |   List<WorkflowExecution> executions = []; | ||||||
|   String? executionData; |  | ||||||
|   int? status; |  | ||||||
|   String? workflowId; |  | ||||||
|    |    | ||||||
|   WorkflowExecutions({ |   WorkflowExecutions({ | ||||||
|     this.executions = const [], |     this.executions = const [], | ||||||
| @@ -56,7 +57,7 @@ class WorkflowExecutions extends SerializerDeserializer<WorkflowExecutions> { | |||||||
| class WorkflowExecution extends SerializerDeserializer<WorkflowExecution> { | class WorkflowExecution extends SerializerDeserializer<WorkflowExecution> { | ||||||
|   String? id; |   String? id; | ||||||
|   String? name; |   String? name; | ||||||
|   String? executionData; |   String? startDate; | ||||||
|   String? endDate; |   String? endDate; | ||||||
|   int? status; |   int? status; | ||||||
|   String? workflowId; |   String? workflowId; | ||||||
| @@ -66,7 +67,7 @@ class WorkflowExecution extends SerializerDeserializer<WorkflowExecution> { | |||||||
|  |  | ||||||
|   WorkflowExecution({ |   WorkflowExecution({ | ||||||
|     this.id, |     this.id, | ||||||
|     this.executionData, |     this.startDate, | ||||||
|     this.status, |     this.status, | ||||||
|     this.workflowId, |     this.workflowId, | ||||||
|     this.name, |     this.name, | ||||||
| @@ -79,7 +80,7 @@ class WorkflowExecution extends SerializerDeserializer<WorkflowExecution> { | |||||||
|     return WorkflowExecution( |     return WorkflowExecution( | ||||||
|       id: json.containsKey("id") ? json["id"] : "", |       id: json.containsKey("id") ? json["id"] : "", | ||||||
|       endDate: json.containsKey("end_date") ? json["end_date"] : "", |       endDate: json.containsKey("end_date") ? json["end_date"] : "", | ||||||
|       executionData: json.containsKey("execution_date") ? json["execution_date"] : "", |       startDate: json.containsKey("execution_date") ? json["execution_date"] : "", | ||||||
|       status: json.containsKey("state") ? json["state"] : 1, |       status: json.containsKey("state") ? json["state"] : 1, | ||||||
|       workflowId: json.containsKey("workflow_id") ? json["workflow_id"] : "", |       workflowId: json.containsKey("workflow_id") ? json["workflow_id"] : "", | ||||||
|       name: json.containsKey("name") ? json["name"] : "", |       name: json.containsKey("name") ? json["name"] : "", | ||||||
| @@ -91,7 +92,7 @@ class WorkflowExecution extends SerializerDeserializer<WorkflowExecution> { | |||||||
|       "id": id, |       "id": id, | ||||||
|       "name": name, |       "name": name, | ||||||
|       "end_date": endDate, |       "end_date": endDate, | ||||||
|       "execution_data": executionData, |       "execution_data": startDate, | ||||||
|       "status": status, |       "status": status, | ||||||
|       "workflow_id": workflowId, |       "workflow_id": workflowId, | ||||||
|     }; |     }; | ||||||
| @@ -108,8 +109,6 @@ class Workflow extends SerializerDeserializer<Workflow>  implements ShallowData | |||||||
|   List<dynamic> processing; |   List<dynamic> processing; | ||||||
|   List<dynamic> workflows; |   List<dynamic> workflows; | ||||||
|   Graph? graph; |   Graph? graph; | ||||||
|   Scheduler? schedule; |  | ||||||
|   bool scheduleActive = false; |  | ||||||
|   List<dynamic> shared; |   List<dynamic> shared; | ||||||
|  |  | ||||||
|   Workflow({ |   Workflow({ | ||||||
| @@ -121,14 +120,12 @@ class Workflow extends SerializerDeserializer<Workflow>  implements ShallowData | |||||||
|     this.processing = const [], |     this.processing = const [], | ||||||
|     this.workflows = const [], |     this.workflows = const [], | ||||||
|     this.graph, |     this.graph, | ||||||
|     this.schedule, |  | ||||||
|     this.scheduleActive = false, |  | ||||||
|     this.shared = const [], |     this.shared = const [], | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override String getID() => id ?? ""; |   @override String getID() => id ?? ""; | ||||||
|   @override String getName() => name ?? ""; |   @override String getName() => name ?? ""; | ||||||
|   @override String getDescription() => ""; |   String getDescription() => ""; | ||||||
|  |  | ||||||
|   @override deserialize(dynamic json) { |   @override deserialize(dynamic json) { | ||||||
|     try { json = json as Map<String, dynamic>; |     try { json = json as Map<String, dynamic>; | ||||||
| @@ -140,11 +137,9 @@ class Workflow extends SerializerDeserializer<Workflow>  implements ShallowData | |||||||
|       processing: json.containsKey("processings") ? json["processings"] : [], |       processing: json.containsKey("processings") ? json["processings"] : [], | ||||||
|       compute: json.containsKey("computes") ? json["computes"] : [], |       compute: json.containsKey("computes") ? json["computes"] : [], | ||||||
|       data: json.containsKey("datas") ? json["datas"] : [], |       data: json.containsKey("datas") ? json["datas"] : [], | ||||||
|       scheduleActive: json.containsKey("schedule_active") ? json["schedule_active"] : false, |  | ||||||
|       storage: json.containsKey("storages") ? json["storages"] : [], |       storage: json.containsKey("storages") ? json["storages"] : [], | ||||||
|       shared: json.containsKey("shared") ? json["shared"] : [], |       shared: json.containsKey("shared") ? json["shared"] : [], | ||||||
|       graph: json.containsKey("graph") ? Graph().deserialize(json["graph"]) : null, |       graph: json.containsKey("graph") ? Graph().deserialize(json["graph"]) : null, | ||||||
|       schedule: json.containsKey("schedule") ?  Scheduler().deserialize(json["schedule"]) : null, |  | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override Map<String, dynamic> serialize() { |   @override Map<String, dynamic> serialize() { | ||||||
| @@ -152,12 +147,10 @@ class Workflow extends SerializerDeserializer<Workflow>  implements ShallowData | |||||||
|       "id": id, |       "id": id, | ||||||
|       "name": name, |       "name": name, | ||||||
|       "datas": data, |       "datas": data, | ||||||
|       "computes" : compute, |  | ||||||
|       "storages": storage, |       "storages": storage, | ||||||
|       "processings": processing, |       "computes" : compute, | ||||||
|       "workflows": workflows, |       "workflows": workflows, | ||||||
|       "schedule_active": scheduleActive, |       "processings": processing, | ||||||
|       "schedule": schedule?.serialize(), |  | ||||||
|     }; |     }; | ||||||
|     if (graph != null) { |     if (graph != null) { | ||||||
|       obj["graph"] = graph!.serialize(); |       obj["graph"] = graph!.serialize(); | ||||||
| @@ -168,24 +161,17 @@ class Workflow extends SerializerDeserializer<Workflow>  implements ShallowData | |||||||
|   void fromDashboard(Map<String, dynamic> j) { |   void fromDashboard(Map<String, dynamic> j) { | ||||||
|     id = j["id"]; |     id = j["id"]; | ||||||
|     name = j["name"]; |     name = j["name"]; | ||||||
|     scheduleActive = j["schedule_active"]; |  | ||||||
|     if (j.containsKey("graph")) {  |     if (j.containsKey("graph")) {  | ||||||
|       graph = Graph(); |       graph = Graph(); | ||||||
|       graph!.fromDashboard(j["graph"]); |       graph!.fromDashboard(j["graph"]); | ||||||
|     } |     } | ||||||
|     if (j.containsKey("schedule")) { |  | ||||||
|       schedule = Scheduler(); |  | ||||||
|       schedule!.fromDashboard(j["schedule"]); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|   Map<String, dynamic> toDashboard() { |   Map<String, dynamic> toDashboard() { | ||||||
|     return { |     return { | ||||||
|       "id": id, |       "id": id, | ||||||
|       "name": name, |       "name": name, | ||||||
|       "graph": graph?.toDashboard(), |  | ||||||
|       "schedule_active": scheduleActive, |  | ||||||
|       "schedule": schedule?.toDashboard(), |  | ||||||
|       "shared": shared, |       "shared": shared, | ||||||
|  |       "graph": graph?.toDashboard(), | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -221,7 +207,7 @@ class Scheduler extends SerializerDeserializer<Scheduler> { | |||||||
|         end = DateTime.parse(j["end"]); |         end = DateTime.parse(j["end"]); | ||||||
|       } |       } | ||||||
|        |        | ||||||
|     } catch (e) {} |     } catch (e) { /**/ } | ||||||
|   } |   } | ||||||
|   Map<String, dynamic> toDashboard() { |   Map<String, dynamic> toDashboard() { | ||||||
|     return { |     return { | ||||||
| @@ -279,6 +265,159 @@ class Graph extends SerializerDeserializer<Graph> { | |||||||
|     this.links = const [], |     this.links = const [], | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|  |   Map<String, List<dynamic>> getInfosToUpdate(SubMapFormsType type) { | ||||||
|  |       Map<String, List<dynamic>> infos = {}; | ||||||
|  |       for (var item in items.values) { | ||||||
|  |         var inst = item.getElement()?.getSelectedInstance(); | ||||||
|  |         if (inst == null) { return infos; } | ||||||
|  |         infos[item.id ?? ""] = (type == SubMapFormsType.INPUT ? inst.inputs : (  | ||||||
|  |           type == SubMapFormsType.OUTPUT ? inst.outputs : inst.env)).map( (e) => e.serialize()).toList(); | ||||||
|  |       } | ||||||
|  |       return infos; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, List<dynamic>> getEnvToUpdate() { | ||||||
|  |       return getInfosToUpdate(SubMapFormsType.ENV); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, List<dynamic>> getInputToUpdate() { | ||||||
|  |       return getInfosToUpdate(SubMapFormsType.INPUT); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, List<dynamic>> getOutputToUpdate() { | ||||||
|  |       return getInfosToUpdate(SubMapFormsType.OUTPUT); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     GraphItem getItemByElementID(String id) { | ||||||
|  |       return items[id] ?? GraphItem(); | ||||||
|  |     } | ||||||
|  |   Map<String,GraphItem?> getArrowByElementID(String id) { | ||||||
|  |     Map<String,GraphItem?> arr = {}; | ||||||
|  |     for (var link in links) { | ||||||
|  |       var nested = false; | ||||||
|  |       var reverse = false; | ||||||
|  |       String? from = link.source?.id; | ||||||
|  |       String? to = link.destination?.id; | ||||||
|  |       bool isInLink = (from?.contains(id) ?? false) || (to?.contains(id) ?? false); | ||||||
|  |       if (isInLink && (["storage", "compute"].contains(getItemByElementID(from ?? "").getElement()?.getType())  | ||||||
|  |       || ["storage", "compute"].contains(getItemByElementID(to ?? "").getElement()?.getType()))) { | ||||||
|  |         nested = true; | ||||||
|  |         reverse = link.source?.id?.contains(id) ?? false; | ||||||
|  |       } | ||||||
|  |       if (nested || isInLink) { | ||||||
|  |           arr[reverse ? (to ?? "") : (from ?? "") ] = getItemByElementID(reverse ? (to ?? "") : (from ?? "") ); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return arr; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void fillEnv(AbstractItem? mainItem, GraphItem? item, SubMapFormsType type, Map<String,int> alreadySeen) { | ||||||
|  |     if (mainItem == null || item == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     AbstractItem? el = item.getElement(); | ||||||
|  |     if (el?.getSelectedInstance() == null) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     var inst = el!.getSelectedInstance()!; | ||||||
|  |      | ||||||
|  |     var inst2 = mainItem.getSelectedInstance()!; | ||||||
|  |     var what = type == SubMapFormsType.INPUT ? inst.inputs : (  | ||||||
|  |       type == SubMapFormsType.OUTPUT ? inst.outputs : inst.env); | ||||||
|  |     var what2 = type == SubMapFormsType.INPUT ? inst2.inputs : (  | ||||||
|  |       type == SubMapFormsType.OUTPUT ? inst2.outputs : inst2.env); | ||||||
|  |     for (var e in what2) { | ||||||
|  |       if (e.attr != null && e.value == null) { | ||||||
|  |         e.value = inst2.serialize()[e.attr!]; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     // should find arrow env info and add it to the env | ||||||
|  |     List<Param> extParams = []; | ||||||
|  |     var arrows = links.where( (e) => (e.source?.id?.contains(item.id ?? "") ?? false) || (e.destination?.id?.contains(item.id ?? "") ?? false)); | ||||||
|  |     for (var arrow in arrows) { | ||||||
|  |       for (var info in arrow.infos) { | ||||||
|  |         var i = info as Map<String, dynamic>; | ||||||
|  |         for (var entry in i.entries) { | ||||||
|  |           if (entry.value == null) { continue; } | ||||||
|  |  | ||||||
|  |           var varName =  "LINK_${el.getName().toUpperCase().replaceAll(" ", "_")}_${entry.key.toUpperCase()}"; | ||||||
|  |           /*alreadySeen[varName] = (alreadySeen[varName] ?? -1) + 1; | ||||||
|  |           if ((alreadySeen[varName] ?? 0) > 1) { | ||||||
|  |             varName = "${varName}_${alreadySeen[varName]}"; | ||||||
|  |           }*/ | ||||||
|  |           if ((entry.value is String) && !isEnvAttr(entry.value, what2)) { | ||||||
|  |             extParams.add(Param( name: varName, | ||||||
|  |               attr: entry.key, value: entry.value, origin: item.id, readOnly: true)); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     for ( var param in what) { | ||||||
|  |       if (param.attr == null) { continue; } | ||||||
|  |       var varName = param.name != null && (param.name!.contains("LINK_")  | ||||||
|  |             || param.name!.contains("DATA_") | ||||||
|  |             || param.name!.contains("PROCESSING_") | ||||||
|  |             || param.name!.contains("STORAGE_") | ||||||
|  |             || param.name!.contains("WORKFLOW_") | ||||||
|  |             || param.name!.contains("COMPUTE") ) ? param.name : "${el.topic.toUpperCase()}_${el.getName().toUpperCase().replaceAll(" ", "_")}_${param.attr!.toUpperCase()}"; | ||||||
|  |       /*alreadySeen[varName] = (alreadySeen[varName] ?? -1) + 1; | ||||||
|  |       if ((alreadySeen[varName] ?? 0) > 0) { | ||||||
|  |         varName = "${varName}_${alreadySeen[varName]}"; | ||||||
|  |       }*/ | ||||||
|  |       dynamic env; | ||||||
|  |       if (param.value == null) {  | ||||||
|  |         var s =  el.serialize(); | ||||||
|  |         s.addAll(el.getSelectedInstance()?.serialize() ?? {}); | ||||||
|  |         env = s[param.attr!]; | ||||||
|  |       } else { | ||||||
|  |         env = param.value; | ||||||
|  |       } | ||||||
|  |       // bool isSrc = param.origin == null; | ||||||
|  |       //if (isSrc) { continue; } | ||||||
|  |       if (!isEnvAttr(env, what2)) { | ||||||
|  |         extParams.add(Param( name: varName, | ||||||
|  |           attr: param.attr, value: env, origin: item.id, readOnly: true)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     for (var param in extParams) { | ||||||
|  |       what2.add(param); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   bool isEnvAttr(String value, List<Param> params) { | ||||||
|  |     for (var e in params) { | ||||||
|  |       if (value == e.value) { return true; } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void fill() { | ||||||
|  |     Map<String, int> alreadySeen = {}; | ||||||
|  |     var its = items.values.toList(); | ||||||
|  |     for (var type in [SubMapFormsType.INPUT, SubMapFormsType.OUTPUT, SubMapFormsType.ENV]) { | ||||||
|  |       for (var item in its) { | ||||||
|  |         var envs = type == SubMapFormsType.INPUT ? item.getElement()?.getSelectedInstance()?.inputs : ( | ||||||
|  |           type == SubMapFormsType.OUTPUT ? item.getElement()?.getSelectedInstance()?.outputs : item.getElement()?.getSelectedInstance()?.env); | ||||||
|  |         if (envs != null) { | ||||||
|  |             envs.removeWhere( (e) => e.readOnly && e.name != null  | ||||||
|  |             && (e.name!.contains("LINK_")  | ||||||
|  |             || e.name!.contains("DATA_") | ||||||
|  |             || e.name!.contains("PROCESSING_") | ||||||
|  |             || e.name!.contains("STORAGE_") | ||||||
|  |             || e.name!.contains("WORKFLOW_") | ||||||
|  |             || e.name!.contains("COMPUTE") )); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       for (var item in [...its, ...its.reversed]) { | ||||||
|  |         var itss = getArrowByElementID(item.id ?? ""); | ||||||
|  |         for (var i in itss.values) { | ||||||
|  |           fillEnv(item.getElement(), i, type, alreadySeen); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void fromDashboard(Map<String, dynamic> j) { |   void fromDashboard(Map<String, dynamic> j) { | ||||||
|     items = {}; |     items = {}; | ||||||
|     for (var el in (j["elements"] as Map<dynamic, dynamic>).values) { |     for (var el in (j["elements"] as Map<dynamic, dynamic>).values) { | ||||||
| @@ -292,9 +431,11 @@ class Graph extends SerializerDeserializer<Graph> { | |||||||
|       return d; |       return d; | ||||||
|     }).toList(); |     }).toList(); | ||||||
|     j["zoom"] = zoom; |     j["zoom"] = zoom; | ||||||
|  |     fill(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> toDashboard() { |   Map<String, dynamic> toDashboard() { | ||||||
|  |     fill(); | ||||||
|     List<Map<String, dynamic>> elements = []; |     List<Map<String, dynamic>> elements = []; | ||||||
|     List<Map<String, dynamic>> arrows = []; |     List<Map<String, dynamic>> arrows = []; | ||||||
|     for (var el in items.values) { |     for (var el in items.values) { | ||||||
| @@ -313,28 +454,70 @@ class Graph extends SerializerDeserializer<Graph> { | |||||||
|   @override deserialize(dynamic json) { |   @override deserialize(dynamic json) { | ||||||
|     try { json = json as Map<String, dynamic>; |     try { json = json as Map<String, dynamic>; | ||||||
|     } catch (e) { return Graph(); }    |     } catch (e) { return Graph(); }    | ||||||
|     return Graph( |     var g =  Graph( | ||||||
|       zoom: json.containsKey("zoom") ? double.parse(json["zoom"].toString()) : 0, |       zoom: json.containsKey("zoom") ? double.parse(json["zoom"].toString()) : 0, | ||||||
|       links: json.containsKey("links") ? fromListJson(json["links"], GraphLink()) : [], |       links: json.containsKey("links") ? fromListJson(json["links"], GraphLink()) : [], | ||||||
|       items: json.containsKey("items") ? fromMapJson<GraphItem>(json["items"], GraphItem()) : {}, |       items: json.containsKey("items") ? fromMapJson<GraphItem>(json["items"], GraphItem()) : {}, | ||||||
|     ); |     ); | ||||||
|  |     g.fill(); | ||||||
|  |     return g; | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     fill(); | ||||||
|  |     return { | ||||||
|  |       "zoom": zoom, | ||||||
|  |       "items": toMapJson(items), | ||||||
|  |       "links": toListJson(links), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class StorageProcessingGraphLink extends SerializerDeserializer<StorageProcessingGraphLink> { | ||||||
|  |   bool write = false; | ||||||
|  |   String? source; | ||||||
|  |   String? destination; | ||||||
|  |   String? filename; | ||||||
|  |  | ||||||
|  |   StorageProcessingGraphLink({ | ||||||
|  |     this.write = false, | ||||||
|  |     this.source, | ||||||
|  |     this.destination, | ||||||
|  |     this.filename, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override deserialize(dynamic json) { | ||||||
|  |     try { json = json as Map<String, dynamic>; | ||||||
|  |     } catch (e) { return StorageProcessingGraphLink(); }    | ||||||
|  |     return StorageProcessingGraphLink( | ||||||
|  |       write: json.containsKey("write") ? json["write"] : false, | ||||||
|  |       source: json.containsKey("source") ? json["source"] : "", | ||||||
|  |       destination: json.containsKey("destination") ? json["destination"] : "", | ||||||
|  |       filename: json.containsKey("filename") && json["filename"] != null ? json["filename"] : "", | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |   @override Map<String, dynamic> serialize() { | ||||||
|  |     return { | ||||||
|  |       "write": write, | ||||||
|  |       "source": source, | ||||||
|  |       "destination": destination, | ||||||
|  |       "filename": filename, | ||||||
|  |     }; | ||||||
|   } |   } | ||||||
|   @override Map<String, dynamic> serialize() => { |  | ||||||
|     "zoom": zoom, |  | ||||||
|     "items": toMapJson(items), |  | ||||||
|     "links": toListJson(links), |  | ||||||
|   }; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| class GraphLink extends SerializerDeserializer<GraphLink> { | class GraphLink extends SerializerDeserializer<GraphLink> { | ||||||
|   Position? source; |   Position? source; | ||||||
|   Position? destination; |   Position? destination; | ||||||
|   GraphLinkStyle? style; |   GraphLinkStyle? style; | ||||||
|  |   List<StorageProcessingGraphLink> infos = []; | ||||||
|  |   List<Param> env = []; | ||||||
|  |  | ||||||
|   GraphLink({ |   GraphLink({ | ||||||
|     this.source, |     this.source, | ||||||
|     this.destination, |     this.destination, | ||||||
|     this.style, |     this.style, | ||||||
|  |     this.infos = const [], | ||||||
|  |     this.env = const [], | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   void fromDashboard(Map<String, dynamic> j) { |   void fromDashboard(Map<String, dynamic> j) { | ||||||
| @@ -342,6 +525,8 @@ class GraphLink extends SerializerDeserializer<GraphLink> { | |||||||
|     destination = Position(id: j["to"]["id"], x: j["to"]["x"], y: j["to"]["y"]); |     destination = Position(id: j["to"]["id"], x: j["to"]["x"], y: j["to"]["y"]); | ||||||
|     style = GraphLinkStyle(); |     style = GraphLinkStyle(); | ||||||
|     style!.fromDashboard(j["params"]); |     style!.fromDashboard(j["params"]); | ||||||
|  |     infos = fromListJson(j["infos"], StorageProcessingGraphLink()); | ||||||
|  |     env = fromListJson(j["env"], Param()); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> toDashboard() { |   Map<String, dynamic> toDashboard() { | ||||||
| @@ -357,6 +542,8 @@ class GraphLink extends SerializerDeserializer<GraphLink> { | |||||||
|         "y": destination?.y ?? 0, |         "y": destination?.y ?? 0, | ||||||
|       }, |       }, | ||||||
|       "params": style?.toDashboard(), |       "params": style?.toDashboard(), | ||||||
|  |       "infos":  toListJson(infos), | ||||||
|  |       "env": env.map( (el) => el.serialize()).toList(), | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -364,9 +551,11 @@ class GraphLink extends SerializerDeserializer<GraphLink> { | |||||||
|     try { json = json as Map<String, dynamic>; |     try { json = json as Map<String, dynamic>; | ||||||
|     } catch (e) { return GraphLink(); }    |     } catch (e) { return GraphLink(); }    | ||||||
|     return GraphLink( |     return GraphLink( | ||||||
|       source: json.containsKey("source") ? Position().deserialize(json["source"]) : null, |       source: json.containsKey("source") && json["source"] != null  ? Position().deserialize(json["source"]) : null, | ||||||
|       destination: json.containsKey("destination") ? Position().deserialize(json["destination"]) : null, |       destination: json.containsKey("destination") && json["destination"] != null  ? Position().deserialize(json["destination"]) : null, | ||||||
|       style: json.containsKey("style") ? GraphLinkStyle().deserialize(json["style"]) : null, |       style: json.containsKey("style") && json["style"] != null  ? GraphLinkStyle().deserialize(json["style"]) : null, | ||||||
|  |       infos: json.containsKey("storage_link_infos") && json["storage_link_infos"] != null ? fromListJson(json["storage_link_infos"], StorageProcessingGraphLink()) : [], | ||||||
|  |       env: json.containsKey("env") && json["env"] != null ? fromListJson(json["env"], Param()) : [], | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override Map<String, dynamic> serialize() { |   @override Map<String, dynamic> serialize() { | ||||||
| @@ -380,6 +569,8 @@ class GraphLink extends SerializerDeserializer<GraphLink> { | |||||||
|     if (style != null) { |     if (style != null) { | ||||||
|       obj["style"] = style!.serialize(); |       obj["style"] = style!.serialize(); | ||||||
|     } |     } | ||||||
|  |     obj["storage_link_infos"] = toListJson(infos); | ||||||
|  |     obj["env"] = toListJson(env); | ||||||
|     return obj; |     return obj; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -521,33 +712,19 @@ class GraphItem extends SerializerDeserializer<GraphItem> { | |||||||
|     return null; |     return null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|   void fromDashboard(Map<String, dynamic> j) { |   void fromDashboard(Map<String, dynamic> j) { | ||||||
|     id = j["id"]; |     id = j["id"]; | ||||||
|     position = Position(x: j["x"], y: j["y"]); |     position = Position(x: j["x"], y: j["y"]); | ||||||
|     width = j["width"]; |     width = j["width"]; | ||||||
|     height = j["height"]; |     height = j["height"]; | ||||||
|      |      | ||||||
|     var abs = WorkspaceLocal.getItem(j["element"]?["id"] ?? "", true) as AbstractItem<FlowData>?; |     if (j["element"] != null) { | ||||||
|     if (abs != null) { |       if (j["element"]["type"] == "data") {  data = DataItem().deserialize(j["element"]);  | ||||||
|       if (abs.topic == "data") {  |       } else if (j["element"]["type"] == "processing") {  processing = ProcessingItem().deserialize(j["element"]); | ||||||
|         data = DataItem().deserialize(abs.serialize());  |       } else if (j["element"]["type"] == "compute") { compute = ComputeItem().deserialize(j["element"]);  | ||||||
|         data!.model = ResourceModel().deserialize(j["element"]["resource_model"]); |       } else if (j["element"]["type"] == "storage") { storage = StorageItem().deserialize(j["element"]);  | ||||||
|       } else if (abs.topic == "processing") {  |       } else if (j["element"]["type"] == "workflow") { workflow = WorkflowItem().deserialize(j["element"]);  | ||||||
|         processing = ProcessingItem().deserialize(abs.serialize()); |  | ||||||
|         processing!.model = ResourceModel().deserialize(j["element"]["resource_model"]); |  | ||||||
|         if ((j["element"] as Map<String, dynamic>).containsKey("container")) { |  | ||||||
|           processing!.container = Containered().deserialize(j["element"]["container"]); |  | ||||||
|           processing!.container?.env?.removeWhere((key, value) => key == "" || value == ""); |  | ||||||
|           processing!.container?.volumes?.removeWhere((key, value) => key == "" || value == ""); |  | ||||||
|           processing!.expose.removeWhere((element) => element.port == null || element.port == 0 || (element.PAT == 0 && element.path == "")); |  | ||||||
|  |  | ||||||
|         } |  | ||||||
|       } else if (abs.topic == "compute") {  |  | ||||||
|         compute = ComputeItem().deserialize(abs.serialize());  |  | ||||||
|       } else if (abs.topic == "storage") {  |  | ||||||
|         storage = StorageItem().deserialize(abs.serialize());  |  | ||||||
|       } else if (abs.topic == "workflow") {  |  | ||||||
|         workflow = WorkflowItem().deserialize(abs.serialize());  |  | ||||||
|       } else { |       } else { | ||||||
|         compute = null; |         compute = null; | ||||||
|         data = null; |         data = null; | ||||||
| @@ -566,7 +743,8 @@ class GraphItem extends SerializerDeserializer<GraphItem> { | |||||||
|  |  | ||||||
|   Map<String, dynamic> toDashboard() { |   Map<String, dynamic> toDashboard() { | ||||||
|     Map<String, dynamic> element = {}; |     Map<String, dynamic> element = {}; | ||||||
|     for(var el in [data, processing, storage, compute, workflow]) { |     List<AbstractItem?> items = [data, processing, storage, compute, workflow]; | ||||||
|  |     for(var el in items) { | ||||||
|       if (el != null && el.getID() != "") { |       if (el != null && el.getID() != "") { | ||||||
|         element = el.serialize(); |         element = el.serialize(); | ||||||
|         break; |         break; | ||||||
|   | |||||||
| @@ -1,7 +1,10 @@ | |||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; |  | ||||||
| import 'package:oc_front/models/abstract.dart'; | import 'package:oc_front/models/abstract.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/data.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
|  | import 'package:oc_front/models/resources/workflow.dart'; | ||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
|  |  | ||||||
| class Workspace extends SerializerDeserializer<Workspace> implements ShallowData { | class Workspace extends SerializerDeserializer<Workspace> implements ShallowData { | ||||||
|   String? id; |   String? id; | ||||||
| @@ -44,13 +47,15 @@ class Workspace extends SerializerDeserializer<Workspace> implements ShallowData | |||||||
|       workflows: json.containsKey("workflow_resources") ? fromListJson(json["workflow_resources"], WorkflowItem()) : [] |       workflows: json.containsKey("workflow_resources") ? fromListJson(json["workflow_resources"], WorkflowItem()) : [] | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override Map<String, dynamic> serialize() => { |   @override Map<String, dynamic> serialize() {  | ||||||
|     "id": id, |     return { | ||||||
|     "name": name, |       "id": id, | ||||||
|     "processings": processings.map((e) => e.id).toList(), |       "name": name, | ||||||
|     "storages": storages.map((e) => e.id).toList(), |       "processings": processings.map((e) => e.id).toList(), | ||||||
|     "computes": computes.map((e) => e.id).toList(), |       "storages": storages.map((e) => e.id).toList(), | ||||||
|     "datas": datas.map((e) => e.id).toList(), |       "computes": computes.map((e) => e.id).toList(), | ||||||
|     "workflows": workflows.map((e) => e.id).toList(), |       "datas": datas.map((e) => e.id).toList(), | ||||||
|   }; |       "workflows": workflows.map((e) => e.id).toList(), | ||||||
|  |     }; | ||||||
|  |   } | ||||||
| } | } | ||||||
| @@ -6,4 +6,6 @@ abstract class AbstractFactory { | |||||||
|   Widget factory(GoRouterState state, List<String> args); |   Widget factory(GoRouterState state, List<String> args); | ||||||
|   bool searchFill(); |   bool searchFill(); | ||||||
|   void search(BuildContext context, bool special); |   void search(BuildContext context, bool special); | ||||||
|  |   String? getSearch(); | ||||||
|  |   void back(BuildContext context); | ||||||
| } | } | ||||||
| @@ -1,23 +1,68 @@ | |||||||
|  | import 'package:oc_front/main.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
|  | import 'package:oc_front/widgets/catalog.dart'; | ||||||
|  | import 'package:localstorage/localstorage.dart'; | ||||||
|  | import 'package:oc_front/pages/abstract_page.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
| import 'package:oc_front/core/sections/header/header.dart'; | import 'package:oc_front/core/sections/header/header.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/resource_service.dart'; | import 'package:oc_front/core/services/specialized_services/resource_service.dart'; | ||||||
| import 'package:oc_front/main.dart'; |  | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/widgets/catalog.dart'; |  | ||||||
| import 'package:oc_front/pages/abstract_page.dart'; |  | ||||||
|  |  | ||||||
| class CatalogFactory implements AbstractFactory { | class CatalogFactory implements AbstractFactory { | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|   static GlobalKey<CatalogPageWidgetState> key = GlobalKey<CatalogPageWidgetState>(); |   static GlobalKey<CatalogPageWidgetState> key = GlobalKey<CatalogPageWidgetState>(); | ||||||
|   @override bool searchFill() {  return key.currentState?.widget.items.isEmpty ?? true; } |   @override void back(BuildContext context) { | ||||||
|  |     var s = (localStorage.getItem("search") ?? ""); | ||||||
|  |     if (s != "") { | ||||||
|  |       localStorage.setItem("search", s.split(",").sublist(1).join(",")); | ||||||
|  |       SearchConstants.set(s.split(",").sublist(1).join(",")); | ||||||
|  |       if ((localStorage.getItem("search") ?? "") == "") { | ||||||
|  |         SearchConstants.remove(); | ||||||
|  |         key.currentState?.widget.isSearch = true; | ||||||
|  |         key.currentState?.widget.items = []; | ||||||
|  |         HeaderConstants.headerKey.currentState?.setState(() {}); | ||||||
|  |         HeaderConstants.headerWidget?.setState(() {}); | ||||||
|  |         CatalogFactory.key.currentState?.setState(() {});  | ||||||
|  |       } else { | ||||||
|  |         CatalogFactory().search(context, false); | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       SearchConstants.remove(); | ||||||
|  |       key.currentState?.widget.isSearch = true; | ||||||
|  |       key.currentState?.widget.items = []; | ||||||
|  |       HeaderConstants.headerKey.currentState?.setState(() {}); | ||||||
|  |       HeaderConstants.headerWidget?.setState(() {}); | ||||||
|  |       CatalogFactory.key.currentState?.setState(() {}); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   @override bool searchFill() {  return (key.currentState?.widget.items.isEmpty ?? true) && (key.currentState?.widget.isSearch ?? true); } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) { return CatalogPageWidget(); } |   @override Widget factory(GoRouterState state, List<String> args) { return CatalogPageWidget(); } | ||||||
|  |   @override String? getSearch() {  | ||||||
|  |     if ((localStorage.getItem("search") ?? "") == "") { return null; } | ||||||
|  |     return localStorage.getItem("search")!.split(",")[0];  | ||||||
|  |   } | ||||||
|   @override void search(BuildContext context, bool special) { |   @override void search(BuildContext context, bool special) { | ||||||
|     if (special) { return; } |     if (special) { return; } // T | ||||||
|     CatalogFactory.key.currentState?.widget.search.search(context, [ SearchConstants.get()! ], {}).then((value) { |     key.currentState?.widget.isSearch = true; | ||||||
|       if (value.data == null) { return; } |     var s = (localStorage.getItem("search") ?? ""); | ||||||
|       key.currentState?.widget.items = [ ...value.data!.workflows, |     if (s != "") { | ||||||
|         ...value.data!.processings, ...value.data!.datas, ...value.data!.storages, ...value.data!.computes,]; |       if (SearchConstants.get() == null) {  | ||||||
|  |         localStorage.setItem("search", s);  | ||||||
|  |       } else if (s.split(",")[0] != SearchConstants.get()) { | ||||||
|  |         localStorage.setItem("search", "${SearchConstants.get()!},$s"); | ||||||
|  |       } | ||||||
|  |     } else if ((SearchConstants.get() ?? "") == "") {  return; | ||||||
|  |     } else {  localStorage.setItem("search", SearchConstants.get()!); } | ||||||
|  |     CatalogFactory.key.currentState?.widget.search.search(context, [  | ||||||
|  |       localStorage.getItem("search")!.split(",")[0] ], {}).then((value) { | ||||||
|  |         if (value.data == null) { | ||||||
|  |           key.currentState?.widget.items = []; | ||||||
|  |         } else { | ||||||
|  |           key.currentState?.widget.isSearch = false; | ||||||
|  |           key.currentState?.widget.items = [ ...value.data!.workflows, | ||||||
|  |             ...value.data!.processings, ...value.data!.datas, ...value.data!.storages, ...value.data!.computes,]; | ||||||
|  |         } | ||||||
|       HeaderConstants.headerKey.currentState?.setState(() {}); |       HeaderConstants.headerKey.currentState?.setState(() {}); | ||||||
|       HeaderConstants.headerWidget?.setState(() {}); |       HeaderConstants.headerWidget?.setState(() {}); | ||||||
|       CatalogFactory.key.currentState?.setState(() {});       // ignore: invalid_use_of_protected_member  |       CatalogFactory.key.currentState?.setState(() {});       // ignore: invalid_use_of_protected_member  | ||||||
| @@ -28,6 +73,7 @@ class CatalogFactory implements AbstractFactory { | |||||||
| // ignore: must_be_immutable | // ignore: must_be_immutable | ||||||
| class CatalogPageWidget extends StatefulWidget { | class CatalogPageWidget extends StatefulWidget { | ||||||
|   double? itemWidth; |   double? itemWidth; | ||||||
|  |   bool isSearch = true; | ||||||
|   List<AbstractItem> items = []; |   List<AbstractItem> items = []; | ||||||
|   final ResourceService search = ResourceService(); |   final ResourceService search = ResourceService(); | ||||||
|   CatalogPageWidget ({ |   CatalogPageWidget ({ | ||||||
| @@ -37,7 +83,16 @@ class CatalogPageWidget extends StatefulWidget { | |||||||
| } | } | ||||||
| class CatalogPageWidgetState extends State<CatalogPageWidget> {   | class CatalogPageWidgetState extends State<CatalogPageWidget> {   | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     if (widget.items.isEmpty) { return Container(); } |     if (widget.items.isEmpty) {  | ||||||
|  |       if (widget.isSearch) { return Container(); } | ||||||
|  |       return Container( | ||||||
|  |         width: getMainWidth(context),  | ||||||
|  |         height: getMainHeight(context) - 50, | ||||||
|  |         color: Colors.grey.shade300,  | ||||||
|  |         child: const Center(child: Text("NO RESOURCES FOUND",  | ||||||
|  |           style: TextStyle(fontSize: 30, color: Colors.grey)) | ||||||
|  |       ), | ||||||
|  |     ); } | ||||||
|     return Column( children : [ |     return Column( children : [ | ||||||
|         SizedBox( width: getMainWidth(context), |         SizedBox( width: getMainWidth(context), | ||||||
|           height: getMainHeight(context) - 50, |           height: getMainHeight(context) - 50, | ||||||
| @@ -46,35 +101,3 @@ class CatalogPageWidgetState extends State<CatalogPageWidget> { | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| /* |  | ||||||
|  |  | ||||||
|             PermsService.getPerm(Perms.WORKSPACE_SHARE) ? ShallowDropdownInputWidget( |  | ||||||
|             iconLoad: Icons.share_rounded, |  | ||||||
|             tooltipLoad:  'share', |  | ||||||
|             tooltipRemove: 'unshare', |  | ||||||
|             color: Colors.white, |  | ||||||
|             filled: const Color.fromRGBO(38, 166, 154, 1), |  | ||||||
|             hintColor: midColor, |  | ||||||
|             type: CollaborativeAreaType.workspace, |  | ||||||
|             all: () async => CollaborativeAreaLocal.workspaces.values.map(  |  | ||||||
|               (e) => Shallow(id: e.id ?? "", name: e.name ?? "") ).toList(), |  | ||||||
|             current: WorkspaceLocal.workspaces[WorkspaceLocal.current]?.shared, |  | ||||||
|             width: (getMainWidth(context) / 3), |  | ||||||
|             canLoad: (String? change) => CollaborativeAreaLocal.workspaces[change] == null  |  | ||||||
|               || !CollaborativeAreaLocal.workspaces[change]!.workspaces.map(  |  | ||||||
|                 (e) => e.id ).contains(WorkspaceLocal.current), |  | ||||||
|             canRemove: (String? change) => CollaborativeAreaLocal.workspaces[change] == null  |  | ||||||
|               || CollaborativeAreaLocal.workspaces[change]!.workspaces.map(  |  | ||||||
|                 (e) => e.id ).contains(WorkspaceLocal.current),  |  | ||||||
|             load: (String val) async { |  | ||||||
|               await SharedService().addWorkspace(context, val, WorkspaceLocal.current ?? ""); |  | ||||||
|               // ignore: use_build_context_synchronously |  | ||||||
|               CollaborativeAreaLocal.init(context, false); |  | ||||||
|             }, |  | ||||||
|             remove: (String val) async { |  | ||||||
|               await SharedService().removeWorkspace(context, val, WorkspaceLocal.current ?? ""); |  | ||||||
|                // ignore: use_build_context_synchronously |  | ||||||
|               CollaborativeAreaLocal.init(context, false); |  | ||||||
|             }) : Container(width: (getMainWidth(context) / 3), height: 50,  |  | ||||||
|               color: const Color.fromRGBO(38, 166, 154, 1)), |  | ||||||
| */ |  | ||||||
| @@ -1,8 +1,11 @@ | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
|  | import 'package:localstorage/localstorage.dart'; | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; | import 'package:oc_front/core/models/workspace_local.dart'; | ||||||
|  | import 'package:oc_front/core/sections/header/header.dart'; | ||||||
|  | import 'package:oc_front/core/services/router.dart'; | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:oc_front/models/resources/resources.dart'; | ||||||
| import 'package:oc_front/pages/abstract_page.dart'; | import 'package:oc_front/pages/abstract_page.dart'; | ||||||
| import 'package:oc_front/pages/catalog.dart'; | import 'package:oc_front/pages/catalog.dart'; | ||||||
| import 'package:oc_front/widgets/items/item.dart'; | import 'package:oc_front/widgets/items/item.dart'; | ||||||
| @@ -11,6 +14,13 @@ import 'package:oc_front/widgets/items/item_row.dart'; | |||||||
| class CatalogItemFactory implements AbstractFactory { | class CatalogItemFactory implements AbstractFactory { | ||||||
|   static GlobalKey<CatalogItemPageWidgetState> key = GlobalKey<CatalogItemPageWidgetState>(); |   static GlobalKey<CatalogItemPageWidgetState> key = GlobalKey<CatalogItemPageWidgetState>(); | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|  |   @override void back(BuildContext context) {  | ||||||
|  |     search(context, false); | ||||||
|  |   } | ||||||
|  |   @override String? getSearch() {  | ||||||
|  |     if ((localStorage.getItem("search") ?? "") == "") { return null; } | ||||||
|  |     return localStorage.getItem("search")!.split(",")[0];  | ||||||
|  |   } | ||||||
|   @override bool searchFill() { return false; } |   @override bool searchFill() { return false; } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) {  |   @override Widget factory(GoRouterState state, List<String> args) {  | ||||||
|     var id = state.pathParameters[args.first]; |     var id = state.pathParameters[args.first]; | ||||||
| @@ -19,13 +29,22 @@ class CatalogItemFactory implements AbstractFactory { | |||||||
|       return CatalogItemPageWidget(item : item!);  |       return CatalogItemPageWidget(item : item!);  | ||||||
|     } catch (e) {  |     } catch (e) {  | ||||||
|       var item =  WorkspaceLocal.getItem(id ?? "", false); |       var item =  WorkspaceLocal.getItem(id ?? "", false); | ||||||
|       if (item != null) { return CatalogItemPageWidget(item : item as AbstractItem); } |       if (item != null) { return CatalogItemPageWidget(item : item); } | ||||||
|       return Container();  |       return Container();  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   @override void search(BuildContext context, bool special) { } |   @override void search(BuildContext context, bool special) { | ||||||
|  |     if (special) { return; } // T | ||||||
|  |       var s = SearchConstants.get(); | ||||||
|  |       AppRouter.catalog.go(context, {}); | ||||||
|  |       Future.delayed(Duration(milliseconds: 10), () { | ||||||
|  |         SearchConstants.set(s); | ||||||
|  |         CatalogFactory().search(context, false); | ||||||
|  |       }); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class CatalogItemPageWidget extends StatefulWidget { | class CatalogItemPageWidget extends StatefulWidget { | ||||||
|   AbstractItem item; |   AbstractItem item; | ||||||
|   CatalogItemPageWidget ({ required this.item }) : super(key: CatalogItemFactory.key); |   CatalogItemPageWidget ({ required this.item }) : super(key: CatalogItemFactory.key); | ||||||
| @@ -35,7 +54,7 @@ class CatalogItemPageWidgetState extends State<CatalogItemPageWidget> { | |||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     return Column( children: [ |     return Column( children: [ | ||||||
|       ItemRowWidget(contextWidth: getMainWidth(context), item: widget.item, readOnly: true,), |       ItemRowWidget(contextWidth: getMainWidth(context), item: widget.item, readOnly: true,), | ||||||
|       ItemWidget(item: widget.item,), |       ItemWidget(item: widget.item), | ||||||
|     ]); |     ]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -4,18 +4,20 @@ import 'package:go_router/go_router.dart'; | |||||||
| import 'package:oc_front/core/services/specialized_services/booking_service.dart'; | import 'package:oc_front/core/services/specialized_services/booking_service.dart'; | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:intl/intl.dart' as intl; | import 'package:intl/intl.dart' as intl; | ||||||
| import 'package:oc_front/models/workflow.dart'; |  | ||||||
| import 'package:oc_front/pages/abstract_page.dart'; | import 'package:oc_front/pages/abstract_page.dart'; | ||||||
| import 'package:oc_front/widgets/sheduler_items/schedule.dart'; | import 'package:oc_front/widgets/sheduler_items/schedule.dart'; | ||||||
|  |  | ||||||
| class DatacenterFactory implements AbstractFactory { | class DatacenterFactory implements AbstractFactory { | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|  |   @override String? getSearch() { return ""; } | ||||||
|  |   @override void back(BuildContext context) { } | ||||||
|   static GlobalKey<ComputePageWidgetState> key = GlobalKey<ComputePageWidgetState>(); |   static GlobalKey<ComputePageWidgetState> key = GlobalKey<ComputePageWidgetState>(); | ||||||
|   @override bool searchFill() { return false; } |   @override bool searchFill() { return false; } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) { return ComputePageWidget(); } |   @override Widget factory(GoRouterState state, List<String> args) { return ComputePageWidget(); } | ||||||
|   @override void search(BuildContext context, bool special) { } |   @override void search(BuildContext context, bool special) { } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class ComputePageWidget extends StatefulWidget { | class ComputePageWidget extends StatefulWidget { | ||||||
|   bool isList = true; |   bool isList = true; | ||||||
|   DateTime start = DateTime.now(); |   DateTime start = DateTime.now(); | ||||||
| @@ -169,7 +171,7 @@ class ComputePageWidgetState extends State<ComputePageWidget> { | |||||||
|                 ) |                 ) | ||||||
|               ])) |               ])) | ||||||
|           ), |           ), | ||||||
|           ScheduleWidget( service: widget._service, key: k, start: widget.start, end : widget.end, isList: widget.isList, isBox: false) |           ScheduleWidget( key: k, start: widget.start, end : widget.end, isList: widget.isList, isBox: false) | ||||||
|         ]); |         ]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -1,18 +1,32 @@ | |||||||
|  | import 'dart:math'; | ||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:font_awesome_flutter/font_awesome_flutter.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:latlong2/latlong.dart'; | import 'package:latlong2/latlong.dart'; | ||||||
| import 'package:flutter_map/flutter_map.dart'; | import 'package:flutter_map/flutter_map.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/datacenter_service.dart'; | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
| import 'package:oc_front/pages/abstract_page.dart'; | import 'package:oc_front/pages/abstract_page.dart'; | ||||||
|  | import 'package:oc_front/widgets/items/item_row.dart'; | ||||||
|  |  | ||||||
|  |  | ||||||
| class  MapFactory implements AbstractFactory { | class  MapFactory implements AbstractFactory { | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|  |   @override String? getSearch() { return ""; } | ||||||
|  |   @override void back(BuildContext context) { } | ||||||
|   static GlobalKey<MapPageWidgetState> key = GlobalKey<MapPageWidgetState>(); |   static GlobalKey<MapPageWidgetState> key = GlobalKey<MapPageWidgetState>(); | ||||||
|   @override bool searchFill() { return false; } |   @override bool searchFill() { return false; } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) { return MapPageWidget(); } |   @override Widget factory(GoRouterState state, List<String> args) { return MapPageWidget(); } | ||||||
|   @override void search(BuildContext context, bool special) { } |   @override void search(BuildContext context, bool special) { } | ||||||
| } | } | ||||||
|  | double menuSize = 0; | ||||||
| class MapPageWidget extends StatefulWidget { | class MapPageWidget extends StatefulWidget { | ||||||
|  |   bool isShowed = false; | ||||||
|  |   final DatacenterService _service = DatacenterService(); | ||||||
|   MapPageWidget(): super(key: MapFactory.key); |   MapPageWidget(): super(key: MapFactory.key); | ||||||
|   @override MapPageWidgetState createState() => MapPageWidgetState(); |   @override MapPageWidgetState createState() => MapPageWidgetState(); | ||||||
|   static void search(BuildContext context) { } |   static void search(BuildContext context) { } | ||||||
| @@ -22,26 +36,193 @@ class MapPageWidgetState extends State<MapPageWidget> { | |||||||
|   double currentZoom = 2.0; |   double currentZoom = 2.0; | ||||||
|   LatLng currentCenter = const LatLng(51.5, -0.09); |   LatLng currentCenter = const LatLng(51.5, -0.09); | ||||||
|   static final MapController _mapController = MapController(); |   static final MapController _mapController = MapController(); | ||||||
|   void _zoom() { |   bool selected = true; | ||||||
|     currentZoom = currentZoom - 1; |  | ||||||
|     _mapController.move(currentCenter, currentZoom); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     return Expanded(  |     return FutureBuilder(future: widget._service.all(context), builder: (BuildContext context, AsyncSnapshot snapshot) { | ||||||
|       child : FlutterMap( |         Map<String, Map<AbstractItem, LatLng>> coordinates = {}; | ||||||
|         mapController: _mapController, |         List<Marker> markerCoordinates = []; | ||||||
|         options: MapOptions( |         if (snapshot.data != null&& snapshot.data!.data != null && snapshot.data!.data!.values.isNotEmpty) { | ||||||
|                   initialCenter: currentCenter, |           for (var element in snapshot.data!.data!.values) { | ||||||
|                   initialZoom: currentZoom, |             if (element["type"] == "storage") {  | ||||||
|                 ), |               StorageItem resource = StorageItem().deserialize(element); | ||||||
|         children: [ |               var instance = resource.getSelectedInstance(); | ||||||
|             TileLayer( |               if (instance == null || instance.location == null) { continue; } | ||||||
|               urlTemplate: 'https://tile.openstreetmap.org/{z}/{x}/{y}.png', |               if (coordinates[resource.topic] == null) { coordinates[resource.topic] = {}; } | ||||||
|               userAgentPackageName: 'dev.fleaflet.flutter_map.example', |               coordinates[resource.topic]![resource] = LatLng(instance.location!.latitude ?? 0, instance.location!.longitude ?? 0); | ||||||
|             ) |             } else if (element["type"] == "compute") { | ||||||
|         ], |               ComputeItem resource = ComputeItem().deserialize(element); | ||||||
|       ) |               var instance = resource.getSelectedInstance(); | ||||||
|  |               if (instance == null || instance.location == null) { continue; } | ||||||
|  |               if (coordinates[resource.topic] == null) { coordinates[resource.topic] = {}; } | ||||||
|  |               var lr = Random().nextInt(max(1, 100)) / 10000; | ||||||
|  |               var lfr = Random().nextInt(max(1, 100)) / 10000; | ||||||
|  |               coordinates[resource.topic]![resource] = LatLng((instance.location!.latitude ?? 0) + lr, (instance.location!.longitude ?? 0) + lfr); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |           for (var topic in coordinates.keys) { | ||||||
|  |             for (var coord in coordinates[topic]!.keys) { | ||||||
|  |               markerCoordinates.add(Marker( | ||||||
|  |                 width: 25, | ||||||
|  |                 height: 30, | ||||||
|  |                 point: coordinates[topic]![coord]!, | ||||||
|  |                 child: HoverMenu( width: 110, title: Container(  alignment: Alignment.center, | ||||||
|  |                     constraints: BoxConstraints( maxHeight: 100, maxWidth: 100 ),  | ||||||
|  |                     child: Icon(FontAwesomeIcons.locationDot,  | ||||||
|  |                       shadows: <Shadow>[Shadow(color: Color.fromRGBO(0, 0, 0, 1), blurRadius: 10.0)], | ||||||
|  |                       color: getColor(topic)) ), | ||||||
|  |                     items: [ Container(color: Colors.white,  | ||||||
|  |                       child: ItemRowWidget(low: true, contextWidth: 290, item: coord))  ] | ||||||
|  |                   ), | ||||||
|  |               )); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |           Rect rect = Rect.fromCenter( center: MediaQuery.of(context).size.center(Offset.zero), | ||||||
|  |             width: selected ? menuSize : 0, height: (getMainHeight(context) - 50) > 0 ? (getMainHeight(context) - 50) : 0); | ||||||
|  |           return Expanded(  | ||||||
|  |             child : FlutterMap( | ||||||
|  |                 mapController: _mapController, | ||||||
|  |                 options: MapOptions( | ||||||
|  |                           initialCenter: currentCenter, | ||||||
|  |                           initialZoom: currentZoom, | ||||||
|  |                         ), | ||||||
|  |                 children: [ | ||||||
|  |                     TileLayer( | ||||||
|  |                       urlTemplate: 'https://tile.openstreetmap.org/{z}/{x}/{y}.png', | ||||||
|  |                       userAgentPackageName: 'dev.fleaflet.flutter_map.example', | ||||||
|  |                     ), | ||||||
|  |                     MarkerLayer( | ||||||
|  |                       markers: markerCoordinates, | ||||||
|  |                     ), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |           ); | ||||||
|  |         } else { | ||||||
|  |           return const CircularProgressIndicator(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class HoverMenuController { | ||||||
|  |   HoverMenuState? currentState; | ||||||
|  |  | ||||||
|  |   void hideSubMenu() { | ||||||
|  |     currentState?.hideSubMenu(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class HoverMenu extends StatefulWidget { | ||||||
|  |   final Widget title; | ||||||
|  |   final double? width; | ||||||
|  |   final List<Widget> items; | ||||||
|  |   final HoverMenuController? controller; | ||||||
|  |   bool isHovered = false; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   HoverMenu({ | ||||||
|  |     Key? key, | ||||||
|  |     required this.title, | ||||||
|  |     this.items = const [], | ||||||
|  |     this.width, | ||||||
|  |     this.controller, | ||||||
|  |   }) : super(key: key); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   HoverMenuState createState() => HoverMenuState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class HoverMenuState extends State<HoverMenu> { | ||||||
|  |   OverlayEntry? _overlayEntry; | ||||||
|  |   final _focusNode = FocusNode(); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void initState() { | ||||||
|  |     super.initState(); | ||||||
|  |     _focusNode.addListener(_onFocusChanged); | ||||||
|  |  | ||||||
|  |     if (widget.controller != null) { | ||||||
|  |       widget.controller?.currentState = this; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void dispose() { | ||||||
|  |     _focusNode.dispose(); | ||||||
|  |     super.dispose(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void _onFocusChanged() { | ||||||
|  |     if (_focusNode.hasFocus) { | ||||||
|  |       _overlayEntry = _createOverlayEntry(); | ||||||
|  |       Overlay.of(context).insert(_overlayEntry!); | ||||||
|  |     } else { | ||||||
|  |       _overlayEntry?.remove(); | ||||||
|  |       _removeOverlay(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void _removeOverlay() { | ||||||
|  |     widget.isHovered = false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void hideSubMenu() { | ||||||
|  |     _focusNode.unfocus(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return TextButton( | ||||||
|  |       style: TextButton.styleFrom( | ||||||
|  |                   elevation: 0.0, | ||||||
|  |                   shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(0)), | ||||||
|  |                   splashFactory: NoSplash.splashFactory, | ||||||
|  |                   padding: EdgeInsets.zero, | ||||||
|  |                   foregroundColor: Colors.transparent, | ||||||
|  |                   surfaceTintColor: Colors.transparent, | ||||||
|  |                   backgroundColor: Colors.transparent, | ||||||
|  |                   shadowColor: Colors.transparent, | ||||||
|  |                 ), | ||||||
|  |       focusNode: _focusNode, | ||||||
|  |       onHover: (isHovered) { | ||||||
|  |         if (isHovered && !widget.isHovered) { | ||||||
|  |           _focusNode.requestFocus(); | ||||||
|  |           isHovered = true; | ||||||
|  |         } else { | ||||||
|  |           _focusNode.unfocus(); | ||||||
|  |           widget.isHovered = false; | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       onPressed: () {}, | ||||||
|  |       child: widget.title, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   OverlayEntry _createOverlayEntry() { | ||||||
|  |     final renderBox = context.findRenderObject() as RenderBox; | ||||||
|  |     final size = renderBox.size; | ||||||
|  |     final offset = renderBox.localToGlobal(Offset.zero); | ||||||
|  |  | ||||||
|  |     return OverlayEntry( | ||||||
|  |       maintainState: true, | ||||||
|  |       builder: (context) => Positioned( | ||||||
|  |         left: offset.dx - 300, | ||||||
|  |         top: offset.dy + size.height, | ||||||
|  |         width: 300, | ||||||
|  |         child: TextButton( | ||||||
|  |             style: TextButton.styleFrom(  | ||||||
|  |               padding: EdgeInsets.zero, | ||||||
|  |               shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(0)), | ||||||
|  |               splashFactory: NoSplash.splashFactory, | ||||||
|  |               backgroundColor: Colors.transparent, | ||||||
|  |             ), | ||||||
|  |             onPressed: () {}, | ||||||
|  |             child: ListView( | ||||||
|  |                 padding: EdgeInsets.zero, | ||||||
|  |                 shrinkWrap: true, | ||||||
|  |                 children: widget.items)), | ||||||
|  |       ), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -9,12 +9,15 @@ import 'package:oc_front/widgets/sheduler_items/schedule.dart'; | |||||||
|  |  | ||||||
| class  SchedulerFactory implements AbstractFactory { | class  SchedulerFactory implements AbstractFactory { | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|  |   @override String? getSearch() { return ""; } | ||||||
|  |   @override void back(BuildContext context) { } | ||||||
|   static GlobalKey<SchedulerPageWidgetState> key = GlobalKey<SchedulerPageWidgetState>(); |   static GlobalKey<SchedulerPageWidgetState> key = GlobalKey<SchedulerPageWidgetState>(); | ||||||
|   @override bool searchFill() { return false; } |   @override bool searchFill() { return false; } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) { return SchedulerPageWidget(); } |   @override Widget factory(GoRouterState state, List<String> args) { return SchedulerPageWidget(); } | ||||||
|   @override void search(BuildContext context, bool special) { } |   @override void search(BuildContext context, bool special) { } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class SchedulerPageWidget extends StatefulWidget { | class SchedulerPageWidget extends StatefulWidget { | ||||||
|   bool isList = true; |   bool isList = true; | ||||||
|   DateTime start = DateTime.now(); |   DateTime start = DateTime.now(); | ||||||
| @@ -164,7 +167,7 @@ class SchedulerPageWidgetState extends State<SchedulerPageWidget> { | |||||||
|                 ) |                 ) | ||||||
|               ])) |               ])) | ||||||
|           ), |           ), | ||||||
|           ScheduleWidget( service: widget._service, key: k, start: widget.start, end : widget.end, isList: widget.isList, ) |           ScheduleWidget( key: k, start: widget.start, end : widget.end, isList: widget.isList, ) | ||||||
|         ]); |         ]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,33 +1,32 @@ | |||||||
| import 'dart:async'; | import 'dart:async'; | ||||||
| import 'dart:math'; | import 'package:intl/intl.dart'; | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:intl/intl.dart'; | import 'package:oc_front/models/shared.dart'; | ||||||
| import 'package:oc_front/core/models/shared_workspace_local.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
|  | import 'package:oc_front/widgets/catalog.dart'; | ||||||
|  | import 'package:oc_front/models/response.dart'; | ||||||
|  | import 'package:oc_front/pages/abstract_page.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; | import 'package:oc_front/core/models/workspace_local.dart'; | ||||||
| import 'package:oc_front/core/sections/header/header.dart'; | import 'package:oc_front/core/sections/header/header.dart'; | ||||||
| import 'package:oc_front/core/services/perms_service.dart'; | import 'package:oc_front/core/services/perms_service.dart'; | ||||||
| import 'package:oc_front/core/services/router.dart'; | import 'package:oc_front/widgets/items/shallow_item_row.dart'; | ||||||
|  | import 'package:oc_front/widgets/dialog/shallow_creation.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | ||||||
|  | import 'package:oc_front/core/models/shared_workspace_local.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/peer_service.dart'; | import 'package:oc_front/core/services/specialized_services/peer_service.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/shared_service.dart'; | import 'package:oc_front/core/services/specialized_services/shared_service.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/workflow_service.dart'; | import 'package:oc_front/core/services/specialized_services/workflow_service.dart'; | ||||||
| import 'package:oc_front/main.dart'; |  | ||||||
| import 'package:oc_front/models/response.dart'; |  | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/models/shared.dart'; |  | ||||||
| import 'package:oc_front/models/workflow.dart'; |  | ||||||
| import 'package:oc_front/pages/abstract_page.dart'; |  | ||||||
| import 'package:oc_front/widgets/catalog.dart'; |  | ||||||
| import 'package:oc_front/widgets/dialog/shallow_creation.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; |  | ||||||
| import 'package:oc_front/widgets/items/shallow_item_row.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; |  | ||||||
|  |  | ||||||
| enum CollaborativeAreaType { global, collaborative_area, workspace, workflow, peer, resource } | enum CollaborativeAreaType { global, collaborative_area, workspace, workflow, peer, resource } | ||||||
|  |  | ||||||
| class  SharedFactory implements AbstractFactory { | class  SharedFactory implements AbstractFactory { | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|  |   @override String? getSearch() { return ""; } | ||||||
|  |   @override void back(BuildContext context) { } | ||||||
|   static GlobalKey<SharedPageWidgetState> key = GlobalKey<SharedPageWidgetState>(); |   static GlobalKey<SharedPageWidgetState> key = GlobalKey<SharedPageWidgetState>(); | ||||||
|   @override bool searchFill() { return false; } |   @override bool searchFill() { return false; } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) { return SharedPageWidget(); } |   @override Widget factory(GoRouterState state, List<String> args) { return SharedPageWidget(); } | ||||||
| @@ -36,6 +35,7 @@ class  SharedFactory implements AbstractFactory { | |||||||
|  |  | ||||||
| class SharedPageWidget extends StatefulWidget { | class SharedPageWidget extends StatefulWidget { | ||||||
|   CollaborativeAreaType type = CollaborativeAreaType.global; |   CollaborativeAreaType type = CollaborativeAreaType.global; | ||||||
|  |   bool showDialog = false; | ||||||
|   SharedPageWidget(): super(key: SharedFactory.key); |   SharedPageWidget(): super(key: SharedFactory.key); | ||||||
|   @override SharedPageWidgetState createState() => SharedPageWidgetState(); |   @override SharedPageWidgetState createState() => SharedPageWidgetState(); | ||||||
|   static void search(BuildContext context) { } |   static void search(BuildContext context) { } | ||||||
| @@ -56,7 +56,6 @@ class SharedPageWidgetState extends State<SharedPageWidget> { | |||||||
|         child : Icon(icon,  |         child : Icon(icon,  | ||||||
|           color: widget.type == workspaceType ? Colors.white : Colors.white, size: 20)))); |           color: widget.type == workspaceType ? Colors.white : Colors.white, size: 20)))); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     GlobalKey<ShallowTextInputWidgetState> key = GlobalKey<ShallowTextInputWidgetState>(); |     GlobalKey<ShallowTextInputWidgetState> key = GlobalKey<ShallowTextInputWidgetState>(); | ||||||
|     if (CollaborativeAreaLocal.current == null) {  |     if (CollaborativeAreaLocal.current == null) {  | ||||||
| @@ -64,52 +63,59 @@ class SharedPageWidgetState extends State<SharedPageWidget> { | |||||||
|         HeaderConstants.setTitle("Choose a Collaborative Area"); |         HeaderConstants.setTitle("Choose a Collaborative Area"); | ||||||
|         HeaderConstants.setDescription("select a shared workspace to continue"); |         HeaderConstants.setDescription("select a shared workspace to continue"); | ||||||
|       }); |       }); | ||||||
|       Future.delayed( const Duration(milliseconds: 100), () { |       if (!widget.showDialog) { | ||||||
|         showDialog( |         Future.delayed( const Duration(milliseconds: 100), () { | ||||||
|           barrierDismissible: false, |           widget.showDialog = true; | ||||||
|           context: context, builder: (BuildContext ctx) => AlertDialog( |           showDialog( | ||||||
|             titlePadding: EdgeInsets.zero, |             barrierDismissible: false, | ||||||
|             insetPadding: EdgeInsets.zero, |             context: context,  | ||||||
|             backgroundColor: Colors.white, |             builder: (BuildContext ctx) => AlertDialog( | ||||||
|             shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(0)), |               titlePadding: EdgeInsets.zero, | ||||||
|             title: ShallowCreationDialogWidget( |               insetPadding: EdgeInsets.zero, | ||||||
|               formKey: key, |               backgroundColor: Colors.white, | ||||||
|               canClose: () => CollaborativeAreaLocal.current != null, |               shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(0)), | ||||||
|               context: context, |               title: ShallowCreationDialogWidget( | ||||||
|               load: (p0) async { |                 formKey: key, | ||||||
|                 CollaborativeAreaLocal.current = p0; |                 canClose: () => CollaborativeAreaLocal.current != null, | ||||||
|                 HeaderConstants.setTitle("Collaborative Area <${CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.name ?? ""}>"); |                 context: context, | ||||||
|                 HeaderConstants.setDescription(CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.description ?? ""); |                 load: (p0) async { | ||||||
|                 Future.delayed(const Duration(seconds: 1), () => SharedFactory.key.currentState?.setState(() {})); |                   CollaborativeAreaLocal.current = p0; | ||||||
|               }, |                   HeaderConstants.setTitle("Collaborative Area <${CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.name ?? ""}>"); | ||||||
|               form: [ |                   HeaderConstants.setDescription(CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.description ?? ""); | ||||||
|                 ShallowTextInputWidget( |                   Future.delayed(const Duration(seconds: 1), () => SharedFactory.key.currentState?.setState(() { widget.showDialog = false; })); | ||||||
|                   change :(p0) => key.currentState?.setState(() {}), |                 }, | ||||||
|                   canLoad: (po) => po != null && po.isNotEmpty, |                 form: [ | ||||||
|                   type: CollaborativeAreaType.collaborative_area, |                   ShallowTextInputWidget( | ||||||
|                   width: getMainWidth(context) <= 540 ? getMainWidth(context) - 140 : 400, |                     change :(p0) => key.currentState?.setState(() {}), | ||||||
|                   attr: "description", |                     canLoad: (po) => po != null && po.isNotEmpty, | ||||||
|                   color: Colors.black, |                     type: CollaborativeAreaType.collaborative_area, | ||||||
|                   hintColor: Colors.grey, |                     width: getMainWidth(context) <= 540 ? getMainWidth(context) - 140 : 400, | ||||||
|                   hint: "enter collaborative area description...", |                     attr: "description", | ||||||
|                   filled: midColor, |                     color: Colors.black, | ||||||
|                 ) |                     hintColor: Colors.grey, | ||||||
|               ], |                     hint: "enter collaborative area description...", | ||||||
|               create: PermsService.getPerm(Perms.COLLABORATIVE_AREA_CREATE) ? (p0) async => await SharedService().post(context, p0, {}).then((value) { |                     filled: midColor, | ||||||
|                 if (value.data != null) { |                   ) | ||||||
|                     CollaborativeAreaLocal.current = value.data!.id; |                 ], | ||||||
|                 } |                 create: PermsService.getPerm(Perms.COLLABORATIVE_AREA_CREATE) ? (p0) async => await SharedService().post(context, p0, {}).then((value) { | ||||||
|                 CollaborativeAreaLocal.init(context, true); |                   if (value.data != null) { | ||||||
|  |                       CollaborativeAreaLocal.current = value.data!.id; | ||||||
|  |                   } | ||||||
|  |                   CollaborativeAreaLocal.init(context, true); | ||||||
|  |  | ||||||
|                 HeaderConstants.setTitle("Collaborative Area <${CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.name ?? ""}>"); |                   HeaderConstants.setTitle("Collaborative Area <${CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.name ?? ""}>"); | ||||||
|                 HeaderConstants.setDescription(CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.description ?? ""); |                   HeaderConstants.setDescription(CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.description ?? ""); | ||||||
|                 Future.delayed(const Duration(seconds: 1), () => SharedFactory.key.currentState?.setState(() {})); |                   Future.delayed(const Duration(seconds: 1), () => SharedFactory.key.currentState?.setState(() { widget.showDialog = false;  })); | ||||||
|               }) : null, |                 }) : null, | ||||||
|               type: CollaborativeAreaType.collaborative_area, |                 type: CollaborativeAreaType.collaborative_area, | ||||||
|               all: () async => CollaborativeAreaLocal.workspaces.values.map(  |                 all: () async {  | ||||||
|                 (e) => Shallow(id: e.id ?? "", name: e.name ?? "") ).toList(), |                   await CollaborativeAreaLocal.init(context, true); | ||||||
|             ))); |                   return CollaborativeAreaLocal.workspaces.values.map(  | ||||||
|       }); |                   (e) => Shallow(id: e.id ?? "", name: e.name ?? "") ).toList(); | ||||||
|  |                 } | ||||||
|  |               ))); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|     } else { |     } else { | ||||||
|       Future.delayed( const Duration(milliseconds: 100), () { |       Future.delayed( const Duration(milliseconds: 100), () { | ||||||
|         HeaderConstants.setTitle("Collaborative Area <${CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.name ?? ""}>"); |         HeaderConstants.setTitle("Collaborative Area <${CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current]?.name ?? ""}>"); | ||||||
| @@ -119,107 +125,7 @@ class SharedPageWidgetState extends State<SharedPageWidget> { | |||||||
|     Widget w = WorkspaceSharedPageWidget(type: widget.type); |     Widget w = WorkspaceSharedPageWidget(type: widget.type); | ||||||
|     List<Widget> addMenu = []; |     List<Widget> addMenu = []; | ||||||
|     CollaborativeArea? current = CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]; |     CollaborativeArea? current = CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]; | ||||||
|     if (widget.type == CollaborativeAreaType.workspace) { |     addMenu.add(getDropdown(widget.type, current, this, context, false)); | ||||||
|       addMenu.add( Row( mainAxisAlignment: MainAxisAlignment.end, |  | ||||||
|         children : [ Container( padding: EdgeInsets.only(left: 20), decoration: BoxDecoration( |  | ||||||
|           border: Border(left: BorderSide(color: Colors.white)) |  | ||||||
|         ), child: ShallowDropdownInputWidget(  |  | ||||||
|           tooltipLoad: "share", |  | ||||||
|           tooltipRemove: "unshare", |  | ||||||
|           iconLoad: Icons.share, |  | ||||||
|           type: widget.type,  |  | ||||||
|           filled: lightColor, |  | ||||||
|           hintColor: midColor, |  | ||||||
|           color: Colors.white, |  | ||||||
|           prefixIcon: Icon(Icons.shopping_cart, color: Colors.white), |  | ||||||
|           current: WorkspaceLocal.current, |  | ||||||
|           all: () async => WorkspaceLocal.getWorkspacesShallow(), |  | ||||||
|           width: getMainWidth(context) / 3, |  | ||||||
|           canLoad: (String? change) => current == null || !current.workspaces.map( (e) => e.id ).contains(change), |  | ||||||
|           canRemove: (String? change) => current == null || current.workspaces.map( (e) => e.id ).contains(change),  |  | ||||||
|           load: PermsService.getPerm(Perms.WORKSPACE_SHARE) ? (String val) async { |  | ||||||
|             await service.addWorkspace(context, CollaborativeAreaLocal.current ?? "", val); |  | ||||||
|             await CollaborativeAreaLocal.init(context, false); |  | ||||||
|             setState(() {}); |  | ||||||
|           } : null, |  | ||||||
|           remove: PermsService.getPerm(Perms.WORKSPACE_UNSHARE) ? (String val) async { |  | ||||||
|             await service.removeWorkspace(context, CollaborativeAreaLocal.current ?? "", val); |  | ||||||
|             await CollaborativeAreaLocal.init(context, false); |  | ||||||
|             setState(() {}); |  | ||||||
|           } : null)) |  | ||||||
|       ])); |  | ||||||
|     } |  | ||||||
|     if (widget.type == CollaborativeAreaType.workflow) { |  | ||||||
|       addMenu.add( Row( mainAxisAlignment: MainAxisAlignment.end, |  | ||||||
|         children : [ Container( padding: EdgeInsets.only(left: 20), decoration: BoxDecoration( |  | ||||||
|           border: Border(left: BorderSide(color: Colors.white)) |  | ||||||
|         ), child: ShallowDropdownInputWidget(  |  | ||||||
|           tooltipLoad: "share", |  | ||||||
|           tooltipRemove: "unshare", |  | ||||||
|           iconLoad: Icons.share, |  | ||||||
|           filled: lightColor, |  | ||||||
|           hintColor: midColor, |  | ||||||
|           color: Colors.white, |  | ||||||
|           type: widget.type, all: () async { |  | ||||||
|             List<Shallow> shals = []; |  | ||||||
|             await WorflowService().all(context).then((value) { |  | ||||||
|                 if (value.data != null) { |  | ||||||
|                   shals = value.data!.values.map((e) => Shallow(id: e["id"], name: e["name"])).toList(); |  | ||||||
|                 } |  | ||||||
|               }); |  | ||||||
|             return shals; |  | ||||||
|           }, |  | ||||||
|           width: getMainWidth(context) / 3, |  | ||||||
|           canLoad: (String? change) => current == null || !current.workflows.map( (e) => e.id ).contains(change), |  | ||||||
|           canRemove: (String? change) => current == null || current.workflows.map( (e) => e.id ).contains(change),  |  | ||||||
|           load: PermsService.getPerm(Perms.WORKFLOW_SHARE) ? (String change) async { |  | ||||||
|             await service.addWorkflow(context, CollaborativeAreaLocal.current ?? "", change); |  | ||||||
|             await CollaborativeAreaLocal.init(context, false); |  | ||||||
|             setState(() {}); |  | ||||||
|           } : null, |  | ||||||
|           remove: PermsService.getPerm(Perms.WORKFLOW_UNSHARE) ? (String change) async { |  | ||||||
|             await service.removeWorkflow(context, CollaborativeAreaLocal.current ?? "", change); |  | ||||||
|             await CollaborativeAreaLocal.init(context, false); |  | ||||||
|             setState(() {}); |  | ||||||
|           } : null)) |  | ||||||
|       ])); |  | ||||||
|     } |  | ||||||
|     if (widget.type == CollaborativeAreaType.peer) { |  | ||||||
|       addMenu.add( Row( mainAxisAlignment: MainAxisAlignment.end, |  | ||||||
|         children : [ |  | ||||||
|         Container( padding: EdgeInsets.only(left: 20), decoration: BoxDecoration( |  | ||||||
|           border: Border(left: BorderSide(color: Colors.white)) |  | ||||||
|         ), child: ShallowDropdownInputWidget(  |  | ||||||
|           tooltipLoad: "add", |  | ||||||
|           iconLoad: Icons.add, |  | ||||||
|           filled: lightColor, |  | ||||||
|           hintColor: midColor, |  | ||||||
|           color: Colors.white, |  | ||||||
|           type: widget.type, all: () async { |  | ||||||
|             List<Shallow> shals = []; |  | ||||||
|             await PeerService().all(context).then((value) { |  | ||||||
|                 if (value.data != null) { |  | ||||||
|                   shals = value.data!.values.map((e) => Shallow(id: e["id"], name: e["name"])).toList(); |  | ||||||
|                 } |  | ||||||
|               }); |  | ||||||
|             return shals; |  | ||||||
|           }, |  | ||||||
|           width: getMainWidth(context) / 3, |  | ||||||
|           canLoad: (String? change) => current == null || !current.peers.map( (e) => e.id ).contains(change), |  | ||||||
|           canRemove: (String? change) => current == null || current.peers.map( (e) => e.id ).contains(change),  |  | ||||||
|           load: PermsService.getPerm(Perms.PEER_SHARE) ? (String change) async { |  | ||||||
|             await service.addPeer(context, CollaborativeAreaLocal.current ?? "", change); |  | ||||||
|             await CollaborativeAreaLocal.init(context, false); |  | ||||||
|             setState(() {}); |  | ||||||
|           } : null, |  | ||||||
|           remove: PermsService.getPerm(Perms.PEER_UNSHARE) ?  (String change) async { |  | ||||||
|             await service.removePeer(context, CollaborativeAreaLocal.current ?? "", change); |  | ||||||
|             await CollaborativeAreaLocal.init(context, false); |  | ||||||
|             setState(() {}); |  | ||||||
|           } : null)) |  | ||||||
|       ])); |  | ||||||
|     } |  | ||||||
|     print(w); |  | ||||||
|     return Column( children: [  |     return Column( children: [  | ||||||
|         Container( |         Container( | ||||||
|           height: 50, |           height: 50, | ||||||
| @@ -318,9 +224,10 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|       List<Widget> badges = []; |       List<Widget> badges = []; | ||||||
|       List<Widget> bbadges = []; |       List<Widget> bbadges = []; | ||||||
|       badges.add(Container( margin: const EdgeInsets.only(left: 5), padding: EdgeInsets.symmetric(horizontal: 10, vertical: 5), |       badges.add(Container( margin: const EdgeInsets.only(left: 5), padding: EdgeInsets.symmetric(horizontal: 10, vertical: 5), | ||||||
|         decoration: BoxDecoration( borderRadius: BorderRadius.circular(4), |         decoration: BoxDecoration(  | ||||||
|         color: isData(w.topic) ? Colors.blue :  isComputing(w.topic) ? Colors.green :  |           borderRadius: BorderRadius.circular(4), | ||||||
|                isCompute(w.topic) ? Colors.orange : isStorage(w.topic) ? redColor : Colors.grey ), |           color: getColor(w.topic)  | ||||||
|  |         ), | ||||||
|         child:Text(w.topic, style: TextStyle( color: Colors.white, fontSize: 11) ))); |         child:Text(w.topic, style: TextStyle( color: Colors.white, fontSize: 11) ))); | ||||||
|       bbadges.add(Container(  margin: const EdgeInsets.only(left: 5),  |       bbadges.add(Container(  margin: const EdgeInsets.only(left: 5),  | ||||||
|         decoration: BoxDecoration( color: Colors.grey, borderRadius: BorderRadius.circular(4) ), |         decoration: BoxDecoration( color: Colors.grey, borderRadius: BorderRadius.circular(4) ), | ||||||
| @@ -337,10 +244,16 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|  |  | ||||||
|   List<Widget> getCardItems(List<ShallowData> data) { |   List<Widget> getCardItems(List<ShallowData> data) { | ||||||
|     List<Widget> items = []; |     List<Widget> items = []; | ||||||
|     for (var w in data) { |     List<ShallowData> neoData = []; | ||||||
|  |     for (var d in data) { | ||||||
|  |       try { neoData.firstWhere( (e) => e.getID() == d.getID()); | ||||||
|  |       } catch (e) { neoData.add(d); } | ||||||
|  |     } | ||||||
|  |     for (var w in neoData) { | ||||||
|       List<Widget> badges = []; |       List<Widget> badges = []; | ||||||
|       if (w is Peer && w.getID( |       if (w is Peer && ( | ||||||
|       ) == CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]?.rule?.creator) { |           w.getID() != CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]?.rule?.creator | ||||||
|  |           || w.getID() != CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]?.creatorID)) { | ||||||
|         badges.add(Padding( padding: const EdgeInsets.only(left: 5), child: Icon(Icons.star, color: Colors.orange.shade300 ))); |         badges.add(Padding( padding: const EdgeInsets.only(left: 5), child: Icon(Icons.star, color: Colors.orange.shade300 ))); | ||||||
|       } else if (widget.type == CollaborativeAreaType.workspace) { |       } else if (widget.type == CollaborativeAreaType.workspace) { | ||||||
|         badges.add(Container( margin: const EdgeInsets.only(left: 5), padding: EdgeInsets.all(5), color: Colors.grey.shade200, |         badges.add(Container( margin: const EdgeInsets.only(left: 5), padding: EdgeInsets.all(5), color: Colors.grey.shade200, | ||||||
| @@ -349,11 +262,13 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|       items.add(ShallowItemRowWidget( |       items.add(ShallowItemRowWidget( | ||||||
|         color: Colors.grey.shade200, |         color: Colors.grey.shade200, | ||||||
|         item: w, badges: badges, |         item: w, badges: badges, | ||||||
|         delete: w is Peer && PermsService.getPerm(Perms.PEER_UNSHARE) && w.getID() != CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]?.rule?.creator ? (String? change) async { |         delete: w is Peer ? (PermsService.getPerm(Perms.PEER_UNSHARE) && ( | ||||||
|  |           w.getID() != CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]?.rule?.creator | ||||||
|  |           && w.getID() != CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current ?? ""]?.creatorID) ? (String? change) async { | ||||||
|           await SharedService().removePeer(context, CollaborativeAreaLocal.current ?? "", change ?? ""); |           await SharedService().removePeer(context, CollaborativeAreaLocal.current ?? "", change ?? ""); | ||||||
|           await CollaborativeAreaLocal.init(context, false); |           await CollaborativeAreaLocal.init(context, false); | ||||||
|           setState(() {}); |           setState(() {}); | ||||||
|         } : (w is Workflow && PermsService.getPerm(Perms.WORKSPACE_UNSHARE)) ? (String? change) async { |         } : null) : (w is Workflow && PermsService.getPerm(Perms.WORKSPACE_UNSHARE)) ? (String? change) async { | ||||||
|           await SharedService().removeWorkflow(context, CollaborativeAreaLocal.current ?? "", change ?? ""); |           await SharedService().removeWorkflow(context, CollaborativeAreaLocal.current ?? "", change ?? ""); | ||||||
|           await CollaborativeAreaLocal.init(context, false); |           await CollaborativeAreaLocal.init(context, false); | ||||||
|           setState(() {}); |           setState(() {}); | ||||||
| @@ -381,7 +296,7 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|       Peer? creator; |       Peer? creator; | ||||||
|       SharedService service = SharedService(); |       SharedService service = SharedService(); | ||||||
|       try { creator = space.peers.firstWhere( (e) => (space.rule?.creator ?? "") == e.id); |       try { creator = space.peers.firstWhere( (e) => (space.rule?.creator ?? "") == e.id); | ||||||
|       } catch (e) { } |       } catch (e) { /**/ } | ||||||
|       Map<String, List<AbstractItem>> datas = {}; |       Map<String, List<AbstractItem>> datas = {}; | ||||||
|       for (var w in space.workspaces) { |       for (var w in space.workspaces) { | ||||||
|         datas[w.getName()] =<AbstractItem> [ |         datas[w.getName()] =<AbstractItem> [ | ||||||
| @@ -406,9 +321,9 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|               padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 50), |               padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 50), | ||||||
|               child: Text(space.description ?? "",  |               child: Text(space.description ?? "",  | ||||||
|                   style: const TextStyle(fontSize: 15, fontWeight: FontWeight.w400, color: Colors.grey)) |                   style: const TextStyle(fontSize: 15, fontWeight: FontWeight.w400, color: Colors.grey)) | ||||||
|             ), // TODO |             ), | ||||||
|             Text("COLLABORATIVE AREA ACCESS RULES", textAlign: TextAlign.start, |             Padding( padding: EdgeInsets.only(left: 30), child: Text("COLLABORATIVE AREA ACCESS RULES", textAlign: TextAlign.start, | ||||||
|             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip), |             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip)), | ||||||
|             Container(  |             Container(  | ||||||
|               width: getMainWidth(context) - 50, |               width: getMainWidth(context) - 50, | ||||||
|               margin: EdgeInsets.symmetric(vertical: 20), |               margin: EdgeInsets.symmetric(vertical: 20), | ||||||
| @@ -427,39 +342,14 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|             ), |             ), | ||||||
|              |              | ||||||
|             Row( children: [ |             Row( children: [ | ||||||
|               Text("COLLABORATIVE AREA PEERS", textAlign: TextAlign.start, |               Padding( padding: EdgeInsets.only(left: 30), child: Text("COLLABORATIVE AREA PEERS", textAlign: TextAlign.start, | ||||||
|             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip),  |             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip)),  | ||||||
|             PermsService.getPerm(Perms.PEER_SHARE) ? Container(  |             PermsService.getPerm(Perms.PEER_SHARE) ? Container(  | ||||||
|               margin: EdgeInsets.only(left: 20), padding: EdgeInsets.only(left: 15), |               margin: EdgeInsets.only(left: 20), padding: EdgeInsets.only(left: 15), | ||||||
|               decoration: BoxDecoration( |               decoration: BoxDecoration( | ||||||
|                 border: Border(left: BorderSide(color: Colors.grey)) |                 border: Border(left: BorderSide(color: Colors.grey)) | ||||||
|               ), child:ShallowDropdownInputWidget(  |               ), child: getDropdown(CollaborativeAreaType.peer, CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current], this, context, true) | ||||||
|               tooltipLoad: "add peer", |               ) : Container()]), | ||||||
|               hint: "add peer", |  | ||||||
|               iconLoad: Icons.add, |  | ||||||
|               type: widget.type,  |  | ||||||
|               filled: midColor, |  | ||||||
|               hintColor: Colors.grey, |  | ||||||
|               color: Colors.black, |  | ||||||
|               prefixIcon: Icon(Icons.person, color: Colors.grey), |  | ||||||
|               all: () async { |  | ||||||
|                 List<Shallow> shals = []; |  | ||||||
|                 await PeerService().all(context).then((value) { |  | ||||||
|                     if (value.data != null) { |  | ||||||
|                       shals = value.data!.values.map((e) => Shallow(id: e["id"], name: e["name"])).where( (e) => !space.peers.map( (e) => e.id ).contains(e.id)).toList(); |  | ||||||
|                     } |  | ||||||
|                     setState(() {}); |  | ||||||
|                   }); |  | ||||||
|                 return shals; |  | ||||||
|               }, |  | ||||||
|               width: getMainWidth(context) / 3, |  | ||||||
|               canLoad: (String? change) => true, |  | ||||||
|               load: (String val) async { |  | ||||||
|                 await service.addPeer(context, CollaborativeAreaLocal.current ?? "", val); |  | ||||||
|                 await CollaborativeAreaLocal.init(context, false); |  | ||||||
|                 setState(() {}); |  | ||||||
|               }, |  | ||||||
|             ) ) : Container()]), |  | ||||||
|             Container(  |             Container(  | ||||||
|               width: getMainWidth(context) - 50, |               width: getMainWidth(context) - 50, | ||||||
|               margin: EdgeInsets.symmetric(vertical: 20), |               margin: EdgeInsets.symmetric(vertical: 20), | ||||||
| @@ -467,8 +357,8 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|               padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 50), |               padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 50), | ||||||
|               child: Wrap( alignment: WrapAlignment.center, children: getCardItems(space.peers)), |               child: Wrap( alignment: WrapAlignment.center, children: getCardItems(space.peers)), | ||||||
|             ), |             ), | ||||||
|             space.rules.isNotEmpty ? Text("RULES", textAlign: TextAlign.start, |             space.rules.isNotEmpty ? Padding( padding: EdgeInsets.only(left: 30), child: Text("RULES", textAlign: TextAlign.start, | ||||||
|             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip) : Container(), |             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip)) : Container(), | ||||||
|             space.rules.isNotEmpty ? Container(  |             space.rules.isNotEmpty ? Container(  | ||||||
|               width: getMainWidth(context) - 50, |               width: getMainWidth(context) - 50, | ||||||
|               margin: EdgeInsets.symmetric(vertical: 20), |               margin: EdgeInsets.symmetric(vertical: 20), | ||||||
| @@ -483,30 +373,13 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|               }).toList() ) |               }).toList() ) | ||||||
|             ) : Container(), |             ) : Container(), | ||||||
|             Row( children: [ |             Row( children: [ | ||||||
|               Text("WORKSPACES / RESOURCES", textAlign: TextAlign.start, |               Padding( padding: EdgeInsets.only(left: 30), child: Text("WORKSPACES / RESOURCES", textAlign: TextAlign.start, | ||||||
|             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip),  |             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip)),  | ||||||
|             PermsService.getPerm(Perms.WORKSPACE_SHARE) ? Container(  |             PermsService.getPerm(Perms.WORKSPACE_SHARE) ? Container(  | ||||||
|               margin: EdgeInsets.only(left: 20), padding: EdgeInsets.only(left: 15), |               margin: EdgeInsets.only(left: 20), padding: EdgeInsets.only(left: 15), | ||||||
|               decoration: BoxDecoration( |               decoration: BoxDecoration( | ||||||
|                 border: Border(left: BorderSide(color: Colors.grey)) |                 border: Border(left: BorderSide(color: Colors.grey)) | ||||||
|               ), child: ShallowDropdownInputWidget(  |               ), child: getDropdown(CollaborativeAreaType.workspace, CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current], this, context, true)) : Container() ]), | ||||||
|               tooltipLoad: "add workspace", |  | ||||||
|               hint: "add workspace", |  | ||||||
|               iconLoad: Icons.add, |  | ||||||
|               type: widget.type,  |  | ||||||
|               filled: midColor, |  | ||||||
|               hintColor: Colors.grey, |  | ||||||
|               color: Colors.black, |  | ||||||
|               prefixIcon: Icon(Icons.shopping_cart, color: Colors.grey), |  | ||||||
|               all: () async => WorkspaceLocal.getWorkspacesShallow().where( (e) => !space.workspaces.map( (e) => e.id ).contains(e.id)).toList(), |  | ||||||
|               width: getMainWidth(context) / 3, |  | ||||||
|               canLoad: (String? change) => true, |  | ||||||
|               load: (String val) async { |  | ||||||
|                 await service.addWorkspace(context, CollaborativeAreaLocal.current ?? "", val); |  | ||||||
|                 await CollaborativeAreaLocal.init(context, false); |  | ||||||
|                 setState(() {}); |  | ||||||
|               }, |  | ||||||
|             )) : Container() ]), |  | ||||||
|             Container(  |             Container(  | ||||||
|               width: getMainWidth(context) - 50, |               width: getMainWidth(context) - 50, | ||||||
|               margin: EdgeInsets.symmetric(vertical: 20), |               margin: EdgeInsets.symmetric(vertical: 20), | ||||||
| @@ -515,38 +388,13 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|               child: Wrap( alignment: WrapAlignment.center, children: getCardWorkspaceItems(datas)) |               child: Wrap( alignment: WrapAlignment.center, children: getCardWorkspaceItems(datas)) | ||||||
|               ), |               ), | ||||||
|             Row( children: [ |             Row( children: [ | ||||||
|               Text("WORKFLOWS", textAlign: TextAlign.start, |               Padding( padding: EdgeInsets.only(left: 30), child: Text("WORKFLOWS", textAlign: TextAlign.start, | ||||||
|             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip),  |             style: TextStyle(color: darkColor, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.clip)),  | ||||||
|             PermsService.getPerm(Perms.WORKFLOW_SHARE) ? Container(  |             PermsService.getPerm(Perms.WORKFLOW_SHARE) ? Container(  | ||||||
|               margin: EdgeInsets.only(left: 20), padding: EdgeInsets.only(left: 15), |               margin: EdgeInsets.only(left: 20), padding: EdgeInsets.only(left: 15), | ||||||
|               decoration: BoxDecoration( |               decoration: BoxDecoration( | ||||||
|                 border: Border(left: BorderSide(color: Colors.grey)) |                 border: Border(left: BorderSide(color: Colors.grey)) | ||||||
|               ), child:ShallowDropdownInputWidget(  |               ), child: getDropdown(CollaborativeAreaType.workflow, CollaborativeAreaLocal.workspaces[CollaborativeAreaLocal.current], this, context, true)) : Container()]), | ||||||
|               tooltipLoad: "add workflow", |  | ||||||
|               hint: "add workflow", |  | ||||||
|               iconLoad: Icons.add, |  | ||||||
|               type: widget.type,  |  | ||||||
|               filled: midColor, |  | ||||||
|               hintColor: Colors.grey, |  | ||||||
|               color: Colors.black, |  | ||||||
|               prefixIcon: Icon(Icons.rebase_edit, color: Colors.grey), |  | ||||||
|               all: () async { |  | ||||||
|                 List<Shallow> shals = []; |  | ||||||
|                 await WorflowService().all(context).then((value) { |  | ||||||
|                     if (value.data != null) { |  | ||||||
|                       shals = value.data!.values.map((e) => Shallow(id: e["id"], name: e["name"])).where( (e) => !space.workflows.map( (e) => e.id ).contains(e.id)).toList(); |  | ||||||
|                     } |  | ||||||
|                   }); |  | ||||||
|                 return shals; |  | ||||||
|               }, |  | ||||||
|               width: getMainWidth(context) / 3, |  | ||||||
|               canLoad: (String? change) => true, |  | ||||||
|               load: (String val) async { |  | ||||||
|                 await service.addWorkflow(context, CollaborativeAreaLocal.current ?? "", val); |  | ||||||
|                 await CollaborativeAreaLocal.init(context, false); |  | ||||||
|                 setState(() {}); |  | ||||||
|               }, |  | ||||||
|             )) : Container()]), |  | ||||||
|             Container(  |             Container(  | ||||||
|               width: getMainWidth(context) - 50, |               width: getMainWidth(context) - 50, | ||||||
|               margin: EdgeInsets.symmetric(vertical: 20), |               margin: EdgeInsets.symmetric(vertical: 20), | ||||||
| @@ -557,6 +405,7 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|           ],) |           ],) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
|  |      | ||||||
|     if (CollaborativeAreaLocal.current == null) {  |     if (CollaborativeAreaLocal.current == null) {  | ||||||
|       return Container( |       return Container( | ||||||
|         color: midColor, |         color: midColor, | ||||||
| @@ -576,11 +425,14 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|       data = space?.peers ?? []; |       data = space?.peers ?? []; | ||||||
|     } |     } | ||||||
|     if (widget.type == CollaborativeAreaType.workspace) { |     if (widget.type == CollaborativeAreaType.workspace) { | ||||||
|       for (var w in data) { |       List<ShallowData> neoData = []; | ||||||
|         if (widget.type == CollaborativeAreaType.workspace) { |       for (var d in data) { | ||||||
|           if (WorkspaceLocal.workspaces[w.getID()] == null) { continue; } |         try { neoData.firstWhere( (e) => e.getID() == d.getID()); | ||||||
|           items.add(WorkspaceSharedItemPageWidget(name: "Workspace <${WorkspaceLocal.workspaces[w.getID()]!.name}>", id: w.getID())); |         } catch (e) { neoData.add(d); } | ||||||
|         }  |       } | ||||||
|  |       for (var w in neoData) { | ||||||
|  |         if (WorkspaceLocal.workspaces[w.getID()] == null) { continue; } | ||||||
|  |         items.add(WorkspaceSharedItemPageWidget(name: "Workspace <${WorkspaceLocal.workspaces[w.getID()]!.name}>", id: w.getID()));  | ||||||
|       } |       } | ||||||
|     } else { |     } else { | ||||||
|       items = getCardItems(data); |       items = getCardItems(data); | ||||||
| @@ -603,6 +455,7 @@ class WorkspaceSharedPageWidgetState extends State<WorkspaceSharedPageWidget> { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class WorkspaceSharedItemPageWidget extends StatefulWidget { | class WorkspaceSharedItemPageWidget extends StatefulWidget { | ||||||
|   bool open = true; |   bool open = true; | ||||||
|   String id = ""; |   String id = ""; | ||||||
| @@ -639,3 +492,128 @@ class WorkspaceSharedItemPageWidgetState extends State<WorkspaceSharedItemPageWi | |||||||
|         ]); |         ]); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |   Widget getDropdown(CollaborativeAreaType type, CollaborativeArea? current, State<dynamic> state, BuildContext context, bool mainPage ) { | ||||||
|  |     if (type == CollaborativeAreaType.workspace) { | ||||||
|  |       return Row( mainAxisAlignment: MainAxisAlignment.end, | ||||||
|  |         children : [ Container( padding: EdgeInsets.only(left:  mainPage ? 0 : 20), decoration: BoxDecoration( | ||||||
|  |           border:  mainPage ? null : Border(left: BorderSide(color: Colors.white)) | ||||||
|  |         ), child: ShallowDropdownInputWidget(  | ||||||
|  |           tooltipLoad: "share", | ||||||
|  |           tooltipRemove: "unshare", | ||||||
|  |           iconLoad: Icons.share, | ||||||
|  |           type: type,  | ||||||
|  |           filled: mainPage ? Colors.white : lightColor, | ||||||
|  |           hintColor:  mainPage ? Colors.grey : midColor, | ||||||
|  |           color: mainPage ? Colors.black : Colors.white, | ||||||
|  |           prefixIcon: Icon(Icons.shopping_cart, color: Colors.white), | ||||||
|  |           current: WorkspaceLocal.current, | ||||||
|  |           all: () async => WorkspaceLocal.getWorkspacesShallow(), | ||||||
|  |           width: getMainWidth(context) / 3, | ||||||
|  |           canLoad: (String? change) => current != null && !current.workspaces.map( (e) => e.id ).contains(change), | ||||||
|  |           canRemove: (String? change) => current != null && current.workspaces.map( (e) => e.id ).contains(change),  | ||||||
|  |           load: PermsService.getPerm(Perms.WORKSPACE_SHARE) ? (String val) async { | ||||||
|  |             await SharedService().addWorkspace(context, CollaborativeAreaLocal.current ?? "", val); | ||||||
|  |             await CollaborativeAreaLocal.init(context, false); | ||||||
|  |             state.setState(() {}); | ||||||
|  |             if (mainPage) { | ||||||
|  |               SharedFactory.key.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|  |           } : null, | ||||||
|  |           remove: PermsService.getPerm(Perms.WORKSPACE_UNSHARE) ? (String val) async { | ||||||
|  |             await SharedService().removeWorkspace(context, CollaborativeAreaLocal.current ?? "", val); | ||||||
|  |             await CollaborativeAreaLocal.init(context, false); | ||||||
|  |             state.setState(() {}); | ||||||
|  |             if (mainPage) { | ||||||
|  |               SharedFactory.key.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|  |           } : null)) | ||||||
|  |       ]); | ||||||
|  |     } | ||||||
|  |     if (type == CollaborativeAreaType.workflow) { | ||||||
|  |       return Row( mainAxisAlignment: MainAxisAlignment.end, | ||||||
|  |         children : [ Container( padding: EdgeInsets.only(left:  mainPage ? 0 : 20), decoration: BoxDecoration( | ||||||
|  |           border:  mainPage ? null : Border(left: BorderSide(color: Colors.white)) | ||||||
|  |         ), child: ShallowDropdownInputWidget(  | ||||||
|  |           tooltipLoad: "share", | ||||||
|  |           tooltipRemove: "unshare", | ||||||
|  |           iconLoad: Icons.share, | ||||||
|  |           filled: mainPage ? Colors.white : lightColor, | ||||||
|  |           hintColor:  mainPage ? Colors.grey : midColor, | ||||||
|  |           color: mainPage ? Colors.black : Colors.white, | ||||||
|  |           type: type, all: () async { | ||||||
|  |             List<Shallow> shals = []; | ||||||
|  |             await WorflowService().all(context).then((value) { | ||||||
|  |                 if (value.data != null) { | ||||||
|  |                   shals = value.data!.values.map((e) => Shallow(id: e["id"], name: e["name"])).toList(); | ||||||
|  |                 } | ||||||
|  |               }); | ||||||
|  |             return shals; | ||||||
|  |           }, | ||||||
|  |           width: getMainWidth(context) / 3, | ||||||
|  |           canLoad: (String? change) => current != null && !current.workflows.map( (e) => e.id ).contains(change), | ||||||
|  |           canRemove: (String? change) => current != null && current.workflows.map( (e) => e.id ).contains(change), | ||||||
|  |           load: PermsService.getPerm(Perms.WORKFLOW_SHARE) ? (String change) async { | ||||||
|  |             await SharedService().addWorkflow(context, CollaborativeAreaLocal.current ?? "", change); | ||||||
|  |             await CollaborativeAreaLocal.init(context, false); | ||||||
|  |             state.setState(() {}); | ||||||
|  |             if (mainPage) { | ||||||
|  |               SharedFactory.key.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|  |           } : null, | ||||||
|  |           remove: PermsService.getPerm(Perms.WORKFLOW_UNSHARE) ? (String change) async { | ||||||
|  |             await SharedService().removeWorkflow(context, CollaborativeAreaLocal.current ?? "", change); | ||||||
|  |             await CollaborativeAreaLocal.init(context, false); | ||||||
|  |             state.setState(() {}); | ||||||
|  |             if (mainPage) { | ||||||
|  |               SharedFactory.key.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|  |           } : null)) | ||||||
|  |       ]); | ||||||
|  |     } | ||||||
|  |     if (type == CollaborativeAreaType.peer) { | ||||||
|  |       return Row( mainAxisAlignment: MainAxisAlignment.end, | ||||||
|  |         children : [ | ||||||
|  |         Container( padding: EdgeInsets.only(left: mainPage ? 0 : 20), decoration: BoxDecoration( | ||||||
|  |           border:  mainPage ? null : Border(left: BorderSide(color : Colors.white)) | ||||||
|  |         ), child: ShallowDropdownInputWidget(  | ||||||
|  |           tooltipLoad: "add", | ||||||
|  |           iconLoad: Icons.add, | ||||||
|  |           filled: mainPage ? Colors.white : lightColor, | ||||||
|  |           hintColor:  mainPage ? Colors.grey : midColor, | ||||||
|  |           color: mainPage ? Colors.black : Colors.white, | ||||||
|  |           type: type, all: () async { | ||||||
|  |             List<Shallow> shals = []; | ||||||
|  |             await PeerService().all(context).then((value) { | ||||||
|  |                 if (value.data != null) { | ||||||
|  |                   shals = value.data!.values.where( (e) { | ||||||
|  |                       return e["id"] != current?.creatorID; | ||||||
|  |                     } | ||||||
|  |                   ).map((e) => Shallow(id: e["id"], name: e["name"])).toList(); | ||||||
|  |                 } | ||||||
|  |               }); | ||||||
|  |             return shals; | ||||||
|  |           }, | ||||||
|  |           width: getMainWidth(context) / 3, | ||||||
|  |           canLoad: (String? change) => current != null && !current.peers.map( (e) => e.id ).contains(change), | ||||||
|  |           canRemove: (String? change) => current != null && current.peers.map( (e) => e.id ).contains(change),  | ||||||
|  |           load: PermsService.getPerm(Perms.PEER_SHARE) ? (String change) async { | ||||||
|  |             await SharedService().addPeer(context, CollaborativeAreaLocal.current ?? "", change); | ||||||
|  |             await CollaborativeAreaLocal.init(context, false); | ||||||
|  |             state.setState(() {}); | ||||||
|  |             if (mainPage) { | ||||||
|  |               SharedFactory.key.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|  |           } : null, | ||||||
|  |           remove: PermsService.getPerm(Perms.PEER_UNSHARE) ?  (String change) async { | ||||||
|  |             await SharedService().removePeer(context, CollaborativeAreaLocal.current ?? "", change); | ||||||
|  |             await CollaborativeAreaLocal.init(context, false); | ||||||
|  |             state.setState(() {}); | ||||||
|  |             if (mainPage) { | ||||||
|  |               SharedFactory.key.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|  |           } : null)) | ||||||
|  |       ]); | ||||||
|  |     } | ||||||
|  |     return Container(); | ||||||
|  |   } | ||||||
| @@ -6,35 +6,41 @@ import 'package:oc_front/core/models/workspace_local.dart'; | |||||||
| import 'package:oc_front/core/services/perms_service.dart'; | import 'package:oc_front/core/services/perms_service.dart'; | ||||||
| import 'package:oc_front/core/services/specialized_services/workflow_service.dart'; | import 'package:oc_front/core/services/specialized_services/workflow_service.dart'; | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/data.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/resources/storage.dart'; | ||||||
|  | import 'package:oc_front/models/resources/workflow.dart'; | ||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
| import 'package:oc_front/pages/abstract_page.dart'; | import 'package:oc_front/pages/abstract_page.dart'; | ||||||
| import 'package:oc_front/pages/shared.dart'; | import 'package:oc_front/pages/shared.dart'; | ||||||
| import 'package:oc_front/widgets/dialog/shallow_creation.dart'; | import 'package:oc_front/widgets/dialog/shallow_creation.dart'; | ||||||
| import 'package:oc_front/widgets/forms/compute_forms.dart'; | import 'package:oc_front/widgets/forms/resource_forms.dart'; | ||||||
| import 'package:oc_front/widgets/forms/data_forms.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/processing_forms.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/scheduler_forms.dart'; | import 'package:oc_front/widgets/forms/scheduler_forms.dart'; | ||||||
| import 'package:oc_front/widgets/forms/storage_forms.dart'; | import 'package:oc_front/widgets/forms/storage_processing_link_forms.dart'; | ||||||
| import 'package:oc_front/widgets/items/item_row.dart'; | import 'package:oc_front/widgets/items/item_row.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | ||||||
|  |  | ||||||
| Dashboard dash = Dashboard(name: ""); | Dashboard dash = Dashboard(name: ""); | ||||||
| class WorkflowFactory implements AbstractFactory { | class WorkflowFactory implements AbstractFactory { | ||||||
|   @override GlobalKey getKey() { return key; } |   @override GlobalKey getKey() { return key; } | ||||||
|  |   @override String? getSearch() { return ""; } | ||||||
|  |   @override void back(BuildContext context) { } | ||||||
|   static GlobalKey<WorkflowPageWidgetState> key = GlobalKey<WorkflowPageWidgetState>(); |   static GlobalKey<WorkflowPageWidgetState> key = GlobalKey<WorkflowPageWidgetState>(); | ||||||
|   @override bool searchFill() { return false; } |   @override bool searchFill() { return false; } | ||||||
|   @override Widget factory(GoRouterState state, List<String> args) {  |   @override Widget factory(GoRouterState state, List<String> args) {  | ||||||
|     String? id; |     String? id; | ||||||
|     try { id = state.pathParameters[args.first]; |     try { id = state.pathParameters[args.first]; | ||||||
|     } catch (e) { } |     } catch (e) { /**/ } | ||||||
|     return WorkflowPageWidget(id: id);  |     return WorkflowPageWidget(id: id);  | ||||||
|   } |   } | ||||||
|   @override void search(BuildContext context, bool special) { } |   @override void search(BuildContext context, bool special) { } | ||||||
| } | } | ||||||
| bool getAll = true; | bool getAll = true; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class WorkflowPageWidget extends StatefulWidget { | class WorkflowPageWidget extends StatefulWidget { | ||||||
|   String? id; |   String? id; | ||||||
|   WorkflowPageWidget ({ this.id }) : super(key: WorkflowFactory.key); |   WorkflowPageWidget ({ this.id }) : super(key: WorkflowFactory.key); | ||||||
| @@ -55,19 +61,20 @@ final WorflowService _service = WorflowService(); | |||||||
|     var e = item as AbstractItem; |     var e = item as AbstractItem; | ||||||
|     return Container(color: Colors.white, child: ItemRowWidget(low: true, contextWidth: 290, item: e)); |     return Container(color: Colors.white, child: ItemRowWidget(low: true, contextWidth: 290, item: e)); | ||||||
|   } |   } | ||||||
|  |   Widget getArrowForms(ArrowPainter? arrow) { | ||||||
|  |     if (arrow == null) { return Container(); } | ||||||
|  |     var from = dash.getElement(arrow.fromID); | ||||||
|  |     var to = dash.getElement(arrow.toID); | ||||||
|  |     if ((from?.element?.getType() == "storage" && to?.element?.getType() == "processing")  | ||||||
|  |     || (from?.element?.getType() == "processing" && to?.element?.getType() == "storage")) { | ||||||
|  |       return StorageProcessingLinkFormsWidget( dash: dash, item: arrow); | ||||||
|  |     } | ||||||
|  |     return Container(); | ||||||
|  |   } | ||||||
|   List<Widget> getForms(FlowData? obj, String id) { |   List<Widget> getForms(FlowData? obj, String id) { | ||||||
|     var objAbs = obj as AbstractItem?; |     var objAbs = obj as AbstractItem?; | ||||||
|     if (objAbs == null) { return []; } |     if (objAbs == null) { return []; } | ||||||
|     List<Widget> res = []; |     List<Widget> res = [ ResourceFormsWidget(item: objAbs, dash: dash, elementID: id) ]; | ||||||
|     if ( objAbs.topic == "processing") { |  | ||||||
|       res = [ProcessingFormsWidget(item: objAbs as ProcessingItem, dash: dash, elementID: id)]; |  | ||||||
|     } else if ( objAbs.topic == "data" ) { |  | ||||||
|       res = [DataFormsWidget(item: objAbs as DataItem)]; |  | ||||||
|     } else if ( objAbs.topic == "storage" ) { |  | ||||||
|       res = [StorageFormsWidget(item: objAbs as StorageItem)]; |  | ||||||
|     } else if ( objAbs.topic == "compute" ) { |  | ||||||
|       res = [ComputeFormsWidget(item: objAbs as ComputeItem)]; |  | ||||||
|     } |  | ||||||
|     return [ Wrap(  |     return [ Wrap(  | ||||||
|       alignment: WrapAlignment.center, |       alignment: WrapAlignment.center, | ||||||
|       children: [ |       children: [ | ||||||
| @@ -84,18 +91,20 @@ final WorflowService _service = WorflowService(); | |||||||
|   Widget? getTopRight(FlowData? obj) { |   Widget? getTopRight(FlowData? obj) { | ||||||
|     var objAbs = obj as AbstractItem?; |     var objAbs = obj as AbstractItem?; | ||||||
|     if (objAbs == null) { return null; } |     if (objAbs == null) { return null; } | ||||||
|     if (objAbs.topic == "compute" ) { |     if (objAbs.topic == "compute") { | ||||||
|       var compute = objAbs as ComputeItem; |       var instance = objAbs as ComputeItem; | ||||||
|       if (compute.technology == 0) { |       if (instance.infrastructureEnum != null) { | ||||||
|         return Icon(FontAwesomeIcons.docker, size: 16); |         if (instance.infrastructureEnum == 0) { | ||||||
|       } else if (compute.technology == 1) { |           return Icon(FontAwesomeIcons.docker, size: 16); | ||||||
|         return Icon(FontAwesomeIcons.lifeRing, size: 16); |         } else if (instance.infrastructureEnum == 1) { | ||||||
|       } else if (compute.technology == 2) { |           return Icon(FontAwesomeIcons.lifeRing, size: 16); | ||||||
|         return Icon(FontAwesomeIcons.cubes, size: 16); |         } else if (instance.infrastructureEnum == 2) { | ||||||
|       } else if (compute.technology == 3) { |           return Icon(FontAwesomeIcons.cubes, size: 16); | ||||||
|         return Icon(FontAwesomeIcons.hardDrive, size: 16); |         } else if (instance.infrastructureEnum == 3) { | ||||||
|       } else if (compute.technology == 4) { |           return Icon(FontAwesomeIcons.hardDrive, size: 16); | ||||||
|         return Icon(FontAwesomeIcons.v, size: 16); |         } else if (instance.infrastructureEnum == 4) { | ||||||
|  |           return Icon(FontAwesomeIcons.v, size: 16); | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return null; |     return null; | ||||||
| @@ -136,10 +145,11 @@ final WorflowService _service = WorflowService(); | |||||||
|     } |     } | ||||||
|     await _service.get(context, dash.id ?? "").then((value) async { |     await _service.get(context, dash.id ?? "").then((value) async { | ||||||
|       if (value.data != null) { |       if (value.data != null) { | ||||||
|  |         // ignore: use_build_context_synchronously | ||||||
|         await WorkspaceLocal.init(context, false); |         await WorkspaceLocal.init(context, false); | ||||||
|         WorkspaceLocal.changeWorkspaceByName("${value.data?.name ?? ""}_workspace"); |         WorkspaceLocal.changeWorkspaceByName("${value.data?.name ?? ""}_workspace"); | ||||||
|         dash.clear(); |         dash.clear(); | ||||||
|         dash.deserialize(value.data!.toDashboard()); |         dash.deserialize(value.data!.toDashboard(), false); | ||||||
|         Future.delayed(const Duration(seconds: 1), () { |         Future.delayed(const Duration(seconds: 1), () { | ||||||
|           dash.name = name; |           dash.name = name; | ||||||
|           dash.shouldSave = true; |           dash.shouldSave = true; | ||||||
| @@ -149,26 +159,20 @@ final WorflowService _service = WorflowService(); | |||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> saveDash(String? id) async { |   Future<void> saveDash(String? id, BuildContext? context) async { | ||||||
|     if (id == null || !dash.isOpened || !dash.shouldSave  |     if (id == null || !dash.isOpened || !dash.shouldSave || !PermsService.getPerm(Perms.WORKFLOW_EDIT)) { return; } | ||||||
|     || !PermsService.getPerm(Perms.WORKFLOW_EDIT)) { return; } |     var datas =  dash.elements.where( (e) => e.element?.serialize()["type"] == "data"); | ||||||
|     var datas = WorkspaceLocal.byTopic("data", true).where( |     var compute = dash.elements.where( (e) => e.element?.serialize()["type"] == "compute"); | ||||||
|         (element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) ); |     var storage = dash.elements.where( (e) => e.element?.serialize()["type"] == "storage"); | ||||||
|     var compute = WorkspaceLocal.byTopic("compute", true).where( |     var processing = dash.elements.where( (e) => e.element?.serialize()["type"] == "processing"); | ||||||
|         (element) => dash.elements.map( (e) => e.element?.getID()).contains(element.id) ); |     var workflows = dash.elements.where( (e) => e.element?.serialize()["type"] == "workflow"); | ||||||
|     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( |     var updateW = Workflow( | ||||||
|       name: dash.name, |       name: dash.name, | ||||||
|       graph: Graph(), |       graph: Graph(), | ||||||
|       data: datas.map((e) => e.id).toSet().toList(), |       data: datas.map((e) => e.id).toSet().toList(), | ||||||
|       compute: compute.map((e) => e.id).toSet().toList(), |       compute: compute.map((e) => e.id).toSet().toList(), | ||||||
|       storage: storage.map((e) => e.id).toSet().toList(), |       storage: storage.map((e) => e.id).toSet().toList(), | ||||||
|       processing: computing.map((e) => e.id).toSet().toList(), |       processing: processing.map((e) => e.id).toSet().toList(), | ||||||
|       workflows: workflows.map((e) => e.id).toSet().toList(), |       workflows: workflows.map((e) => e.id).toSet().toList(), | ||||||
|     ); |     ); | ||||||
|     updateW.fromDashboard(dash.serialize()); |     updateW.fromDashboard(dash.serialize()); | ||||||
| @@ -177,8 +181,8 @@ final WorflowService _service = WorflowService(); | |||||||
|       item.position?.x = (item.position?.x ?? 0) + (item.width! / 2) + 7.5; |       item.position?.x = (item.position?.x ?? 0) + (item.width! / 2) + 7.5; | ||||||
|       item.position?.y = (item.position?.y ?? 0) + (item.height! / 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)) { |       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?.x = (i.source?.x ?? 0) + (item.width! / 2) + 7.5; | ||||||
|         i.source?.y = (i.source?.y ?? 0) + (item.width! / 2) + 7; |         i.source?.y = (i.source?.y ?? 0) + (item.width! / 2) + 7.5; | ||||||
|       } |       } | ||||||
|       for (var i in (updateW.graph?.links ?? [] as List<GraphLink>).where((element) => id == element.destination?.id)) { |       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?.x = (i.destination?.x ?? 0) + (item.width! / 2) + 7.5; | ||||||
| @@ -187,31 +191,24 @@ final WorflowService _service = WorflowService(); | |||||||
|     } |     } | ||||||
|     updateW.graph?.zoom = dash.getZoomFactor(); |     updateW.graph?.zoom = dash.getZoomFactor(); | ||||||
|     dash.addToHistory(); |     dash.addToHistory(); | ||||||
|     await _service.put(context, id, updateW.serialize(), {}).then( (e) async { |       _service.put(null, id, updateW.serialize(), {}).then( (e) async { | ||||||
|       if (dash.addChange) { |         dash.applyInfos(updateW.graph?.getEnvToUpdate() ?? {}, updateW.toDashboard()); | ||||||
|         dash.addChange = false; |         if (dash.addChange) { | ||||||
|         await WorkspaceLocal.init(context, false); |           dash.addChange = false; | ||||||
|         WorkspaceLocal.changeWorkspaceByName("${dash.name}_workspace"); |           // ignore: use_build_context_synchronously | ||||||
|         dash.selectedLeftMenuKey.currentState?.setState(() { }); |           await WorkspaceLocal.init(context, false); | ||||||
|       } |           WorkspaceLocal.changeWorkspaceByName("${dash.name}_workspace"); | ||||||
|     });   |           dash.selectedLeftMenuKey.currentState?.setState(() { }); | ||||||
|  |         } | ||||||
|  |       });   | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   FlowData? transformToData(Map<String, dynamic> data) { |   FlowData? transformToData(Map<String, dynamic> data) { | ||||||
|     var d = WorkspaceLocal.getItem(data["id"] ?? "", true); |     if (data["type"] == "data") { return DataItem().deserialize(data); } | ||||||
|     if (d == null) { return null; } |     if (data["type"] == "compute") { return ComputeItem().deserialize(data); } | ||||||
|     d.model = ResourceModel().deserialize(data["resource_model"]); |     if (data["type"] == "storage") { return StorageItem().deserialize(data); } | ||||||
|     if (d.topic == "data") { return d as DataItem; } |     if (data["type"] == "processing") { return ProcessingItem().deserialize(data); } | ||||||
|     if (d.topic == "compute") { return d as ComputeItem; } |     if (data["type"] == "workflows") { return WorkflowItem().deserialize(data); } | ||||||
|     if (d.topic == "storage") { return d as StorageItem; } |  | ||||||
|     if (d.topic == "processing") {  |  | ||||||
|       d = d as ProcessingItem; |  | ||||||
|       if (data.containsKey("container")) { |  | ||||||
|         d.container = Containered().deserialize(data["container"]); |  | ||||||
|       } |  | ||||||
|       return d;  |  | ||||||
|     } |  | ||||||
|     if (d.topic == "workflows") { return d as WorkflowItem; } |  | ||||||
|     return null; |     return null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -249,12 +246,14 @@ final WorflowService _service = WorflowService(); | |||||||
|             WorkspaceLocal.changeWorkspaceByName(p0.split("~")[1]);   |             WorkspaceLocal.changeWorkspaceByName(p0.split("~")[1]);   | ||||||
|             await dash.load!(p0);          |             await dash.load!(p0);          | ||||||
|           }  |           }  | ||||||
|  |           dash.inDialog = false; | ||||||
|           dash.notifyListeners(); |           dash.notifyListeners(); | ||||||
|       }, |       }, | ||||||
|       create: PermsService.getPerm(Perms.WORKFLOW_CREATE) ? (p0) async => await _service.post(context, p0, {}).then( (value) async { |       create: PermsService.getPerm(Perms.WORKFLOW_CREATE) ? (p0) async => await _service.post(context, p0, {}).then( (value) async { | ||||||
|           dash.clear(); |           dash.clear(); | ||||||
|           dash.id = value.data?.getID() ?? ""; |           dash.id = value.data?.getID() ?? ""; | ||||||
|           dash.name = value.data?.getName() ?? ""; |           dash.name = value.data?.getName() ?? ""; | ||||||
|  |           dash.inDialog = false; | ||||||
|           dash.notifyListeners(); |           dash.notifyListeners(); | ||||||
|           await WorkspaceLocal.init(context, true); |           await WorkspaceLocal.init(context, true); | ||||||
|           dash.isOpened = true; |           dash.isOpened = true; | ||||||
| @@ -288,26 +287,33 @@ final WorflowService _service = WorflowService(); | |||||||
|     dash.midDashColor = midColor; |     dash.midDashColor = midColor; | ||||||
|     dash.transformToData = transformToData; |     dash.transformToData = transformToData; | ||||||
|     dash.infoItemWidget = getForms; |     dash.infoItemWidget = getForms; | ||||||
|  |     dash.infoLinkWidget = getArrowForms; | ||||||
|     dash.infoWidget = getDashInfoForms; |     dash.infoWidget = getDashInfoForms; | ||||||
|     dash.widthOffset = 50; |     dash.widthOffset = 50; | ||||||
|     dash.arrowStyleRules = [ |     dash.arrowStyleRules = [ | ||||||
|       (dash) { |       (dash) { | ||||||
|         for (var arrow in dash.arrows) { |         for (var arrow in dash.arrows) { | ||||||
|           var from = dash.elements.firstWhere((element) => arrow.fromID.contains(element.id)).element; |           try { | ||||||
|           var to = dash.elements.firstWhere((element) => arrow.toID.contains(element.id)).element; |             var from = dash.elements.firstWhere((element) => arrow.fromID.split("_")[0] == element.id).element; | ||||||
|           if ((from is ProcessingItem && to is ComputeItem) || (to is ProcessingItem && from is ComputeItem)) { |             var to = dash.elements.firstWhere((element) => arrow.toID.split("_")[0] == element.id).element; | ||||||
|             arrow.params.color = Colors.orange; |             if ((from is ProcessingItem && to is ComputeItem) || (to is ProcessingItem && from is ComputeItem)) { | ||||||
|             arrow.params.dashSpace = 2; |               arrow.params.color = Colors.orange; | ||||||
|             arrow.params.dashWidth = 2; |               arrow.params.dashSpace = 2; | ||||||
|           } else if ((from is ProcessingItem && to is StorageItem) || (to is ProcessingItem && from is StorageItem)) { |               arrow.params.dashWidth = 2; | ||||||
|             arrow.params.color = redColor; |             } else if ((from is ProcessingItem && to is StorageItem) || (to is ProcessingItem && from is StorageItem)) { | ||||||
|             arrow.params.dashSpace = 2; |               arrow.params.color = redColor; | ||||||
|             arrow.params.dashWidth = 2; |               arrow.params.dashSpace = 2; | ||||||
|           } else if ((from is ProcessingItem && to is DataItem) || (to is ProcessingItem && from is DataItem)) { |               arrow.params.dashWidth = 2; | ||||||
|             arrow.params.color = Colors.blue; |             } else if ((from is ProcessingItem && to is DataItem) || (to is ProcessingItem && from is DataItem)) { | ||||||
|             arrow.params.dashSpace = 2; |               arrow.params.color = Colors.blue; | ||||||
|             arrow.params.dashWidth = 2; |               arrow.params.dashSpace = 2; | ||||||
|           } else { |               arrow.params.dashWidth = 2; | ||||||
|  |             } else { | ||||||
|  |               arrow.params.color = Colors.black; | ||||||
|  |               arrow.params.dashSpace = 0; | ||||||
|  |               arrow.params.dashWidth = 0; | ||||||
|  |             } | ||||||
|  |           } catch (e) { | ||||||
|             arrow.params.color = Colors.black; |             arrow.params.color = Colors.black; | ||||||
|             arrow.params.dashSpace = 0; |             arrow.params.dashSpace = 0; | ||||||
|             arrow.params.dashWidth = 0; |             arrow.params.dashWidth = 0; | ||||||
| @@ -319,32 +325,7 @@ final WorflowService _service = WorflowService(); | |||||||
|     dash.saveRules = [ |     dash.saveRules = [ | ||||||
|       (dash) { |       (dash) { | ||||||
|         dash.error = null; |         dash.error = null; | ||||||
|         if (dash.scheduleActive) { |          | ||||||
|           if (dash.elements.isEmpty || dash.elements.where((element) => element.element is ProcessingItem).isEmpty) { |  | ||||||
|             dash.error = "You need at least one processing element"; |  | ||||||
|             dash.scheduleActive = false; |  | ||||||
|           }  |  | ||||||
|           var processings = dash.elements.where((element) => element.element is ProcessingItem).map((e) => e.element as ProcessingItem); |  | ||||||
|           var computes = dash.elements.where((element) => element.element is ComputeItem).map((e) => e.element as ComputeItem); |  | ||||||
|           if (processings.length != computes.length) { |  | ||||||
|              dash.error = "You need the same number of processing and compute elements"; |  | ||||||
|              dash.scheduleActive = false; |  | ||||||
|           } |  | ||||||
|           for (var p in processings) { |  | ||||||
|             var links =  dash.arrows.where((element) => element.fromID.contains(p.getID()) || element.toID.contains(p.getID())); |  | ||||||
|             try { |  | ||||||
|               computes.firstWhere( (e) => links.first.toID.contains(e.getID()) || links.first.fromID.contains(e.getID()) ); |  | ||||||
|             } catch (e) { |  | ||||||
|                dash.error = "You need to link each processing element to a compute element"; |  | ||||||
|                dash.scheduleActive = false; |  | ||||||
|             } |  | ||||||
|           } |  | ||||||
|           if (dash.error != null) { |  | ||||||
|             print(dash.error); |  | ||||||
|             setState(() {}); |  | ||||||
|           } |  | ||||||
|           return dash.scheduleActive; |  | ||||||
|         } |  | ||||||
|         return true; |         return true; | ||||||
|       } |       } | ||||||
|     ]; |     ]; | ||||||
|   | |||||||
| @@ -1,14 +1,14 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:oc_front/widgets/items/item_row.dart'; | import 'package:oc_front/widgets/items/item_row.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/core/models/workspace_local.dart'; | ||||||
|  |  | ||||||
| class CatalogWidget extends StatefulWidget { | class CatalogWidget extends StatefulWidget { | ||||||
|   double? itemWidth; |   double? itemWidth; | ||||||
|   bool readOnly = false; |   bool readOnly = false; | ||||||
|   final List<AbstractItem>? items; |   final List<AbstractItem>? items; | ||||||
|   CatalogWidget ({ Key? key, this.items, this.itemWidth, this.readOnly = false }): super(key: key); |   CatalogWidget ({ super.key, this.items, this.itemWidth, this.readOnly = false }); | ||||||
|   @override CatalogWidgetState createState() => CatalogWidgetState(); |   @override CatalogWidgetState createState() => CatalogWidgetState(); | ||||||
| } | } | ||||||
| class CatalogWidgetState extends State<CatalogWidget> { | class CatalogWidgetState extends State<CatalogWidget> { | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import 'package:flutter_spinkit/flutter_spinkit.dart'; | |||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:oc_front/core/services/auth.service.dart'; | import 'package:oc_front/core/services/auth.service.dart'; | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:oc_front/pages/workflow.dart'; | ||||||
|  |  | ||||||
| class LoginWidget extends StatefulWidget { | class LoginWidget extends StatefulWidget { | ||||||
|   LoginWidget ({ Key? key }): super(key: key); |   LoginWidget ({ Key? key }): super(key: key); | ||||||
| @@ -30,26 +31,32 @@ class LoginWidgetState extends State<LoginWidget> { | |||||||
|             loading = false; |             loading = false; | ||||||
|             error = "Invalid username or password"; |             error = "Invalid username or password"; | ||||||
|           }); |           }); | ||||||
|  |         }).then( (e) { | ||||||
|  |           if (error == null) { | ||||||
|  |               // ignore: use_build_context_synchronously | ||||||
|  |               setState(() { | ||||||
|  |                 loading = false; | ||||||
|  |               }); | ||||||
|  |               loginIsSet = false; | ||||||
|  |               dash.inDialog = false; | ||||||
|  |               context.pop(); | ||||||
|  |               mainKey?.currentState?.setState(() {}); | ||||||
|  |             } | ||||||
|         }); |         }); | ||||||
|         if (error == null) { |  | ||||||
|           // ignore: use_build_context_synchronously |  | ||||||
|           setState(() { |  | ||||||
|             loading = false; |  | ||||||
|           }); |  | ||||||
|           context.pop(); |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     child: Padding( |     child: Container( padding: const EdgeInsets.all(50), child: Column(children: [ | ||||||
|       padding: const EdgeInsets.all(50), child: Column(mainAxisSize: MainAxisSize.min, children: [ |       getMainHeight(context) < 600 ? Container() : SizedBox( width: getMainWidth(context) / 4, height: getMainHeight(context) / 4, | ||||||
|       const Center(child: Icon(Icons.person_search, size: 150, color: Colors.grey,)), |       child: FittedBox( | ||||||
|  |           child:const Center(child: Icon(Icons.person_search, size: 150, color: Colors.grey,)))), | ||||||
|       Center(child: Padding( padding: const EdgeInsets.only(top: 5, bottom: 20),  |       Center(child: Padding( padding: const EdgeInsets.only(top: 5, bottom: 20),  | ||||||
|         child: Text("WELCOME ON OPENCLOUD", style: TextStyle(fontSize: 25, fontWeight: FontWeight.w600, |         child: Text("WELCOME ON OPENCLOUD", style: TextStyle(fontSize: 25, fontWeight: FontWeight.w600, | ||||||
|         color: lightColor ) ))), |         color: lightColor ), overflow: TextOverflow.ellipsis, ))), | ||||||
|       Container( margin: const EdgeInsets.only(bottom: 10), child: Center(child: Row( mainAxisAlignment: MainAxisAlignment.center, |       Container( margin: const EdgeInsets.only(bottom: 10),  | ||||||
|  |       child: Center(child: Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|         children: [  |         children: [  | ||||||
|         Container(  |         Container(  | ||||||
|           width: getMainWidth(context) / 3, |           width: MediaQuery.of(context).size.width / 3, | ||||||
|           alignment : Alignment.center,   |           alignment : Alignment.center,   | ||||||
|           child: TextField( |           child: TextField( | ||||||
|           controller: usernameCtrl, |           controller: usernameCtrl, | ||||||
| @@ -69,7 +76,7 @@ class LoginWidgetState extends State<LoginWidget> { | |||||||
|       Center(child: Row( mainAxisAlignment: MainAxisAlignment.center, |       Center(child: Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|         children: [  |         children: [  | ||||||
|         Container(  |         Container(  | ||||||
|           width: getMainWidth(context) / 3, |           width: MediaQuery.of(context).size.width / 3, | ||||||
|           alignment : Alignment.center,   |           alignment : Alignment.center,   | ||||||
|           child: TextField( |           child: TextField( | ||||||
|           controller: passwordCtrl, |           controller: passwordCtrl, | ||||||
| @@ -105,20 +112,22 @@ class LoginWidgetState extends State<LoginWidget> { | |||||||
|                   loading = false; |                   loading = false; | ||||||
|                   error = "Invalid username or password"; |                   error = "Invalid username or password"; | ||||||
|                 }); |                 }); | ||||||
|  |               }).then( (e) { | ||||||
|  |                 if (error == null) { | ||||||
|  |                     // ignore: use_build_context_synchronously | ||||||
|  |                   setState(() { | ||||||
|  |                     loading = false; | ||||||
|  |                   }); | ||||||
|  |                   loginIsSet = false; | ||||||
|  |                   dash.inDialog = false; | ||||||
|  |                   context.pop(); | ||||||
|  |                   mainKey?.currentState?.setState(() {}); | ||||||
|  |                 } | ||||||
|               }); |               }); | ||||||
|                |  | ||||||
|               if (error == null) { |  | ||||||
|                 // ignore: use_build_context_synchronously |  | ||||||
|                 setState(() { |  | ||||||
|                   loading = true; |  | ||||||
|                 }); |  | ||||||
|                 context.pop(); |  | ||||||
|                 //mainKey?.currentState!.setState(() {}); |  | ||||||
|               } |  | ||||||
|             }, |             }, | ||||||
|             mouseCursor: SystemMouseCursors.click, |             mouseCursor: SystemMouseCursors.click, | ||||||
|             child: Container( |             child: Container( | ||||||
|               width: getMainWidth(context) / 3, |               width: MediaQuery.of(context).size.width / 3, | ||||||
|               padding: const EdgeInsets.symmetric(vertical: 20), |               padding: const EdgeInsets.symmetric(vertical: 20), | ||||||
|               color: usernameCtrl.text == "" || passwordCtrl.text == "" ? Colors.grey : lightColor,  |               color: usernameCtrl.text == "" || passwordCtrl.text == "" ? Colors.grey : lightColor,  | ||||||
|           child: Center( child: loading ? SpinKitWave(color: Colors.white, size: 20) : Text("LOGIN", style: TextStyle( |           child: Center( child: loading ? SpinKitWave(color: Colors.white, size: 20) : Text("LOGIN", style: TextStyle( | ||||||
|   | |||||||
| @@ -3,9 +3,11 @@ import 'package:oc_front/main.dart'; | |||||||
| import 'package:oc_front/models/response.dart'; | import 'package:oc_front/models/response.dart'; | ||||||
| import 'package:oc_front/core/services/router.dart'; | import 'package:oc_front/core/services/router.dart'; | ||||||
| import 'package:oc_front/pages/shared.dart'; | import 'package:oc_front/pages/shared.dart'; | ||||||
|  | import 'package:oc_front/pages/workflow.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class ShallowCreationDialogWidget extends StatefulWidget { | class ShallowCreationDialogWidget extends StatefulWidget { | ||||||
|   GlobalKey<ShallowTextInputWidgetState>? formKey; |   GlobalKey<ShallowTextInputWidgetState>? formKey; | ||||||
|   BuildContext context; |   BuildContext context; | ||||||
| @@ -43,6 +45,7 @@ class ShallowCreationDialogState extends State<ShallowCreationDialogWidget> { | |||||||
|                   Tooltip( message: "back", child: InkWell(  |                   Tooltip( message: "back", child: InkWell(  | ||||||
|                   mouseCursor: SystemMouseCursors.click, |                   mouseCursor: SystemMouseCursors.click, | ||||||
|                   onTap: () {  |                   onTap: () {  | ||||||
|  |                     dash.inDialog = false; | ||||||
|                     AppRouter.catalog.go(context, {});  |                     AppRouter.catalog.go(context, {});  | ||||||
|                   }, |                   }, | ||||||
|                   child: const Icon(Icons.arrow_back, color: Colors.black))), |                   child: const Icon(Icons.arrow_back, color: Colors.black))), | ||||||
| @@ -50,7 +53,10 @@ class ShallowCreationDialogState extends State<ShallowCreationDialogWidget> { | |||||||
|                 widget.canClose != null && !widget.canClose!() ? Container() :  Row ( mainAxisAlignment: MainAxisAlignment.end, children: [ |                 widget.canClose != null && !widget.canClose!() ? Container() :  Row ( mainAxisAlignment: MainAxisAlignment.end, children: [ | ||||||
|                   Tooltip( message: "close", child: InkWell(  |                   Tooltip( message: "close", child: InkWell(  | ||||||
|                   mouseCursor: SystemMouseCursors.click, |                   mouseCursor: SystemMouseCursors.click, | ||||||
|                   onTap: () { Navigator.pop(context); }, |                   onTap: () {  | ||||||
|  |                     dash.inDialog = false; | ||||||
|  |                     Navigator.pop(context);  | ||||||
|  |                   }, | ||||||
|                   child: const Icon(Icons.close, color: Colors.black))), |                   child: const Icon(Icons.close, color: Colors.black))), | ||||||
|                 ]),  |                 ]),  | ||||||
|               ],), |               ],), | ||||||
| @@ -62,6 +68,7 @@ class ShallowCreationDialogState extends State<ShallowCreationDialogWidget> { | |||||||
|               width: getMainWidth(context) <= 540 ? getMainWidth(context) - 140 : 400, |               width: getMainWidth(context) <= 540 ? getMainWidth(context) - 140 : 400, | ||||||
|               load: (e) async { |               load: (e) async { | ||||||
|                 await widget.load!(e); |                 await widget.load!(e); | ||||||
|  |                 dash.inDialog = false; | ||||||
|                 Navigator.pop(widget.context); |                 Navigator.pop(widget.context); | ||||||
|               }, |               }, | ||||||
|               iconLoad: Icons.open_in_browser_outlined, |               iconLoad: Icons.open_in_browser_outlined, | ||||||
| @@ -83,6 +90,7 @@ class ShallowCreationDialogState extends State<ShallowCreationDialogWidget> { | |||||||
|               width: getMainWidth(context) <= 540 ? getMainWidth(context) - 140 : 400, |               width: getMainWidth(context) <= 540 ? getMainWidth(context) - 140 : 400, | ||||||
|               load: (e) async { |               load: (e) async { | ||||||
|                 await widget.create!(e); |                 await widget.create!(e); | ||||||
|  |                 dash.inDialog = false; | ||||||
|                 Navigator.pop(widget.context); |                 Navigator.pop(widget.context); | ||||||
|               }, |               }, | ||||||
|               forms: widget.form, |               forms: widget.form, | ||||||
|   | |||||||
| @@ -1,22 +0,0 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/sub_text_input.dart'; |  | ||||||
|  |  | ||||||
| class ComputeFormsWidget extends StatefulWidget { |  | ||||||
|   dynamic item; |  | ||||||
|   ComputeFormsWidget ({ super.key, required this.item }); |  | ||||||
|   @override ComputeFormsWidgetState createState() => ComputeFormsWidgetState(); |  | ||||||
| } |  | ||||||
| class ComputeFormsWidgetState extends State<ComputeFormsWidget> { |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     return Column( children: [ |  | ||||||
|       SubTextInputWidget(subkey: "technology", width: 180, empty: false, change: (value) {  |  | ||||||
|       }, initialValue: widget.item.getTechnology(), readOnly: true,), |  | ||||||
|       SubTextInputWidget(subkey: "architecture", width: 180, empty: false, change: (value) {  |  | ||||||
|       }, initialValue: widget.item.architecture, readOnly: true,), |  | ||||||
|       SubTextInputWidget(subkey: "access", width: 180, empty: false, change: (value) {  |  | ||||||
|       }, initialValue: widget.item.getAccess(), readOnly: true,), |  | ||||||
|       SubTextInputWidget(subkey: "localisation", width: 180, empty: false, change: (value) {  |  | ||||||
|       }, initialValue: widget.item.localisation, readOnly: true,), |  | ||||||
|     ]); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
							
								
								
									
										105
									
								
								lib/widgets/forms/container_forms.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								lib/widgets/forms/container_forms.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/widgets/forms/sub_expose_forms.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
|  | class ContainerFormsWidget extends StatefulWidget { | ||||||
|  |   int instanceID = 0; | ||||||
|  |   Dashboard dash; | ||||||
|  |   AbstractItem item; | ||||||
|  |   String elementID; | ||||||
|  |   ContainerFormsWidget({ super.key, required this.item, required this.dash, required this.elementID }); | ||||||
|  |   @override ContainerFormsWidgetState createState() => ContainerFormsWidgetState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ContainerFormsWidgetState extends State<ContainerFormsWidget> { | ||||||
|  |   @override Widget build(BuildContext context) { | ||||||
|  |     List<Widget> widgets = []; | ||||||
|  |     var instance = widget.item.getSelectedInstance(); | ||||||
|  |     if (instance != null && instance is ProcessingInstance && instance.access?.container != null) { | ||||||
|  |       var container = instance.access!.container!; | ||||||
|  |       widgets.add(Container( | ||||||
|  |         padding: const EdgeInsets.only(bottom: 10), | ||||||
|  |         width: 180, | ||||||
|  |         child: Column(  | ||||||
|  |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |           children: [ | ||||||
|  |             Text("<CONTAINER>", style: const TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center), | ||||||
|  |             SubTextInputWidget(subkey: "image", width: 180, empty: false, change: (value) { }, | ||||||
|  |               initialValue: container.image, readOnly: true), | ||||||
|  |             SubTextInputWidget(subkey: "command", width: 180, empty: false, change: (value) { | ||||||
|  |                 container.command = value; | ||||||
|  |                 for (var el in widget.dash.elements) { | ||||||
|  |                     if (el.id == widget.elementID) { | ||||||
|  |                       el.element = widget.item; | ||||||
|  |                       break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |               }, initialValue: container.command, readOnly: false,), | ||||||
|  |             SubTextInputWidget(subkey: "args", width: 180, empty: false, change: (value) { | ||||||
|  |                 container.args = value; | ||||||
|  |                 for (var el in widget.dash.elements) { | ||||||
|  |                     if (el.id == widget.elementID) { | ||||||
|  |                       el.element = widget.item; | ||||||
|  |                       break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |               },  | ||||||
|  |               initialValue: container.args, readOnly: false,) | ||||||
|  |           ],) | ||||||
|  |         )); | ||||||
|  |         widgets.add(Container( | ||||||
|  |         width: 200, decoration: BoxDecoration( border: Border(bottom: BorderSide(color: Colors.grey))), | ||||||
|  |       )); | ||||||
|  |       widgets.add(Container( | ||||||
|  |             padding: const EdgeInsets.symmetric(vertical: 10), | ||||||
|  |             width: 180, | ||||||
|  |             child: Column(  | ||||||
|  |               mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |               children: [ | ||||||
|  |                 const Padding(padding: EdgeInsets.only(bottom: 5), child: Text("<EXPOSE>",  | ||||||
|  |                   style: TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center)), | ||||||
|  |                 Row( children: [ | ||||||
|  |                   InkWell( onTap: () { | ||||||
|  |                       container.exposes.add(Expose()); | ||||||
|  |                       var el = widget.dash.getElement(widget.elementID); | ||||||
|  |                       el!.element = widget.item as dynamic; | ||||||
|  |                       setState(() {}); | ||||||
|  |                     }, child: | ||||||
|  |                     Container( margin: const EdgeInsets.only(top: 5), | ||||||
|  |                       decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.grey, width: 1)), | ||||||
|  |                       width: 125, height: 30, | ||||||
|  |                       child: const Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                         children: [ Padding( padding: EdgeInsets.only(right: 5), child: Icon(Icons.add)), Text("add expose")]), | ||||||
|  |                     ) | ||||||
|  |                   ), | ||||||
|  |                   InkWell( onTap: () { | ||||||
|  |                     if (container.exposes.isEmpty) { return; } | ||||||
|  |                       container.exposes = container.exposes.sublist(0, container.exposes.length - 1); | ||||||
|  |                       var el = widget.dash.getElement(widget.elementID); | ||||||
|  |                       el!.element = widget.item as dynamic; | ||||||
|  |                       setState(() {}); | ||||||
|  |                     }, child: | ||||||
|  |                     Container( margin: const EdgeInsets.only(left: 5, top: 5), | ||||||
|  |                       decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.grey, width: 1)), | ||||||
|  |                       width: 45, height: 30, | ||||||
|  |                       child: const Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                         children: [ Icon(Icons.delete, color: Colors.black) ]), | ||||||
|  |                     ) | ||||||
|  |                   ) | ||||||
|  |                 ]), | ||||||
|  |               ],) | ||||||
|  |           )); | ||||||
|  |       for (var expose in container.exposes) { | ||||||
|  |         widgets.add(SubExposeFormsWidget( readOnly: true, width: 180, dash: widget.dash, empty: widgets.isEmpty,  | ||||||
|  |         item: expose, elementID: widget.elementID)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return Column(children: widgets); | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										58
									
								
								lib/widgets/forms/credentials_forms.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								lib/widgets/forms/credentials_forms.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
|  | class CredentialsFormsWidget extends StatefulWidget { | ||||||
|  |   int instanceID = 0; | ||||||
|  |   Dashboard dash; | ||||||
|  |   AbstractItem item; | ||||||
|  |   String elementID; | ||||||
|  |   CredentialsFormsWidget({ super.key, required this.item, required this.dash, required this.elementID }); | ||||||
|  |   @override CredentialsFormsWidgetState createState() => CredentialsFormsWidgetState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class CredentialsFormsWidgetState extends State<CredentialsFormsWidget> { | ||||||
|  |   @override Widget build(BuildContext context) { | ||||||
|  |     List<Widget> widgets = []; | ||||||
|  |     var instance = widget.item.getSelectedInstance(); | ||||||
|  |     if (instance != null  && instance.credential != null) { | ||||||
|  |       var creds = instance.credential!; | ||||||
|  |       widgets.add(Container( margin: EdgeInsets.only(bottom: 15), | ||||||
|  |         width: 200, decoration: BoxDecoration( border: Border(bottom: BorderSide(color: Colors.grey))), | ||||||
|  |       )); | ||||||
|  |       widgets.add(Container( | ||||||
|  |         padding: const EdgeInsets.only(bottom: 10), | ||||||
|  |         width: 180, | ||||||
|  |         child: Column(  | ||||||
|  |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |           children: [ | ||||||
|  |             Text("<CREDENTIALS>", style: const TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center), | ||||||
|  |             SubTextInputWidget(subkey: "login", width: 180, empty: false, change: (value) { | ||||||
|  |               creds.password = value; | ||||||
|  |                 for (var el in widget.dash.elements) { | ||||||
|  |                     if (el.id == widget.elementID) { | ||||||
|  |                       el.element = widget.item; | ||||||
|  |                       break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |             },  initialValue: creds.login), | ||||||
|  |             SubTextInputWidget(subkey: "password", width: 180, empty: false, change: (value) { | ||||||
|  |                 creds.password = value; | ||||||
|  |                 for (var el in widget.dash.elements) { | ||||||
|  |                     if (el.id == widget.elementID) { | ||||||
|  |                       el.element = widget.item; | ||||||
|  |                       break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |               }, initialValue: creds.password, readOnly: false,), | ||||||
|  |           ],) | ||||||
|  |         )); | ||||||
|  |         widgets.add(Container( padding: EdgeInsets.only(bottom: 15), width: 200 )); | ||||||
|  |     } | ||||||
|  |     return Column(children: widgets); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/web_reference_forms.dart'; |  | ||||||
|  |  | ||||||
| class DataFormsWidget extends StatefulWidget { |  | ||||||
|   DataItem item; |  | ||||||
|   String purpose = "";  |  | ||||||
|   Function validate = () {}; |  | ||||||
|   DataFormsWidget ({ super.key, required this.item }); |  | ||||||
|   @override DataFormsWidgetState createState() => DataFormsWidgetState(); |  | ||||||
| } |  | ||||||
| class DataFormsWidgetState extends State<DataFormsWidget> { |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     return Column( children: [ |  | ||||||
|       WebReferenceFormsWidget(item: widget.item), |  | ||||||
|     ]); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,147 +0,0 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; |  | ||||||
| import 'package:oc_front/core/services/perms_service.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; |  | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/models/workflow.dart'; |  | ||||||
| import 'package:oc_front/pages/workflow.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/sub_keys_forms.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/sub_expose_forms.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/sub_map_forms.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/sub_text_input.dart'; |  | ||||||
|  |  | ||||||
| Map<String, Map<String, AbstractItem>> proxyWfItem = {}; |  | ||||||
| // ADD EXPOSE |  | ||||||
| class ProcessingFormsWidget extends StatefulWidget { |  | ||||||
|   String elementID; |  | ||||||
|   ProcessingItem item; |  | ||||||
|   Dashboard dash; |  | ||||||
|   ProcessingFormsWidget ({ super.key, required this.item, required this.dash, required this.elementID }); |  | ||||||
|   @override ProcessingFormsWidgetState createState() => ProcessingFormsWidgetState(); |  | ||||||
| } |  | ||||||
| class ProcessingFormsWidgetState extends State<ProcessingFormsWidget> { |  | ||||||
|   Widget getInputAndOutputVariableForms(bool readOnly) { |  | ||||||
|     var inList = widget.dash.GetArrowByElementID(widget.elementID, true); |  | ||||||
|     var outList = widget.dash.GetArrowByElementID(widget.elementID, false); |  | ||||||
|     List<Widget> res = []; |  | ||||||
|     List<GraphItem> inItems = []; |  | ||||||
|     List<GraphItem> outItems = []; |  | ||||||
|     for (var inItem in inList) { |  | ||||||
|       var element = widget.dash.getElement(inItem.fromID.substring(0,36)); |  | ||||||
|       if (element == null) { continue; } |  | ||||||
|       var g = GraphItem(); |  | ||||||
|       g.fromDashboard(element.serialize()); |  | ||||||
|       inItems.add(g); |  | ||||||
|     } |  | ||||||
|     for (var outItem in outList) { |  | ||||||
|       var element = widget.dash.getElement(outItem.toID.substring(0,36)); |  | ||||||
|       if (element == null) { continue; } |  | ||||||
|       var g = GraphItem(); |  | ||||||
|       g.fromDashboard(element.serialize()); |  | ||||||
|       outItems.add(g); |  | ||||||
|     } |  | ||||||
|     if (inItems.isNotEmpty) { |  | ||||||
|       res.add(SubKeysMapFormsWidget(dash: widget.dash, isInput: true, item: widget.item, elementID: widget.elementID,  |  | ||||||
|         categoryKey: "container", varKey: "env", graphItems: inItems, readOnly: readOnly)); |  | ||||||
|     } |  | ||||||
|     if (outItems.isNotEmpty) { |  | ||||||
|       res.add(SubKeysMapFormsWidget(dash: widget.dash, isInput: false, item: widget.item, elementID: widget.elementID,  |  | ||||||
|         categoryKey: "container", varKey: "env", graphItems: outItems, readOnly: readOnly)); |  | ||||||
|     } |  | ||||||
|     return Column( children: res ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     bool readOnly = !PermsService.getPerm(Perms.WORKFLOW_EDIT); |  | ||||||
|     List<Widget> categories = []; |  | ||||||
|     var l = widget.item.model?.model?.keys ?? []; |  | ||||||
|     for (var child in l) { |  | ||||||
|       var sub = widget.item.model!.model![child]!; |  | ||||||
|       List<Widget> children = []; |  | ||||||
|       for (var st in sub.keys) {   |  | ||||||
|         if (sub[st]!.type?.contains("map") ?? false) { |  | ||||||
|           children.add( |  | ||||||
|             SubMapFormsWidget(dash: dash, empty: children.isEmpty, item: widget.item,  |  | ||||||
|                           readOnly: readOnly, |  | ||||||
|                           elementID: widget.elementID, categoryKey: child, varKey: st) |  | ||||||
|           ); |  | ||||||
|         } else if (sub[st]!.type == "string") { |  | ||||||
|           children.add(SubTextInputWidget(subkey: st,  |  | ||||||
|             readOnly: readOnly, |  | ||||||
|             initialValue: widget.item.getVariable([child, st], widget.item.serialize()),  |  | ||||||
|             width: 180, empty: children.isEmpty, change: (value) {  |  | ||||||
|               widget.item.model ?? Model(); |  | ||||||
|               Future.delayed(const Duration(seconds: 2), () { |  | ||||||
|                 if (widget.item.getVariable([child, st], widget.item.serialize()) == value) { |  | ||||||
|                   dash.saveDash(dash.id); |  | ||||||
|                 } |  | ||||||
|               }); |  | ||||||
|               var el = dash.getElement(widget.elementID); |  | ||||||
|               widget.item = widget.item.deserialize(widget.item.setVariable([child, st], value, widget.item.serialize())) as dynamic; |  | ||||||
|               el!.element = widget.item as dynamic; |  | ||||||
|             }) |  | ||||||
|           ); |  | ||||||
|         }   |  | ||||||
|       } |  | ||||||
|       categories.add(Container( |  | ||||||
|           padding: const EdgeInsets.symmetric(vertical: 10), |  | ||||||
|           width: 180, |  | ||||||
|           child: Column(  |  | ||||||
|             mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|             children: [ |  | ||||||
|               Text("<${child.toUpperCase()}>", style: const TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center), |  | ||||||
|               Padding(padding: const EdgeInsets.only(bottom: 10), child: SubTextInputWidget(subkey: "image", width: 180, empty: false, change: (value) {},  |  | ||||||
|                 initialValue: widget.item.container?.image, readOnly: true,)), |  | ||||||
|               ...children, |  | ||||||
|               getInputAndOutputVariableForms(readOnly), |  | ||||||
|             ],) |  | ||||||
|         )); |  | ||||||
|     } |  | ||||||
|     // EXPOSE  |  | ||||||
|     categories.add(Container( |  | ||||||
|           padding: const EdgeInsets.symmetric(vertical: 10), |  | ||||||
|           width: 180, |  | ||||||
|           child: Column(  |  | ||||||
|             mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|             children: [ |  | ||||||
|               const Padding(padding: EdgeInsets.only(bottom: 5), child: Text("<EXPOSE>",  |  | ||||||
|                 style: TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center)), |  | ||||||
|               readOnly ? Container() : Row( children: [ |  | ||||||
|                 InkWell( onTap: () { |  | ||||||
|                     widget.item.expose.add(Expose()); |  | ||||||
|                     var el = dash.getElement(widget.elementID); |  | ||||||
|                     el!.element = widget.item as dynamic; |  | ||||||
|                     setState(() {}); |  | ||||||
|                   }, child: |  | ||||||
|                   Container( margin: const EdgeInsets.only(top: 5), |  | ||||||
|                     decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.grey, width: 1)), |  | ||||||
|                     width: 125, height: 30, |  | ||||||
|                     child: const Row( mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|                       children: [ Padding( padding: EdgeInsets.only(right: 5), child: Icon(Icons.add)), Text("add expose")]), |  | ||||||
|                   ) |  | ||||||
|                 ), |  | ||||||
|                 InkWell( onTap: () { |  | ||||||
|                   if (widget.item.expose.isEmpty) { return; } |  | ||||||
|                     widget.item.expose = widget.item.expose.sublist(0, widget.item.expose.length - 1); |  | ||||||
|                     var el = dash.getElement(widget.elementID); |  | ||||||
|                     el!.element = widget.item as dynamic; |  | ||||||
|                     setState(() {}); |  | ||||||
|                   }, child: |  | ||||||
|                   Container( margin: const EdgeInsets.only(left: 5, top: 5), |  | ||||||
|                     decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.grey, width: 1)), |  | ||||||
|                     width: 45, height: 30, |  | ||||||
|                     child: const Row( mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|                       children: [ Icon(Icons.delete, color: Colors.black) ]), |  | ||||||
|                   ) |  | ||||||
|                 ) |  | ||||||
|               ]), |  | ||||||
|             ],) |  | ||||||
|         )); |  | ||||||
|     for (var expose in widget.item.expose) { |  | ||||||
|       categories.add(SubExposeFormsWidget( readOnly: readOnly, width: 180, dash: dash, empty: categories.isEmpty,  |  | ||||||
|       item: expose, elementID: widget.elementID)); |  | ||||||
|     } |  | ||||||
|     return SizedBox( height: getHeight(context) - 330, child: SingleChildScrollView( child: Column(  |  | ||||||
|       children: categories )) ); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
							
								
								
									
										92
									
								
								lib/widgets/forms/resource_forms.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								lib/widgets/forms/resource_forms.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | |||||||
|  | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/models/workflow.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/core/services/perms_service.dart'; | ||||||
|  | import 'package:oc_front/widgets/forms/credentials_forms.dart'; | ||||||
|  | import 'package:oc_front/widgets/forms/sub_keys_forms.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  | import 'package:oc_front/widgets/forms/container_forms.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_dropdown_input%20.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
|  | class ResourceFormsWidget extends StatefulWidget { | ||||||
|  |   AbstractItem item; | ||||||
|  |   Dashboard dash; | ||||||
|  |   String elementID; | ||||||
|  |   ResourceFormsWidget ({ super.key, required this.item, required this.dash, required this.elementID }); | ||||||
|  |   @override ResourceFormsWidgetState createState() => ResourceFormsWidgetState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ResourceFormsWidgetState extends State<ResourceFormsWidget> { | ||||||
|  |   List<Widget> getWidgets(Map<String, dynamic> infos) { | ||||||
|  |     List<Widget> widgets = []; | ||||||
|  |     for (var key in infos.keys) { | ||||||
|  |       if (infos[key] == null && infos[key] != 0) { continue; } | ||||||
|  |       if (infos[key] is Map) { widgets.addAll(getWidgets(infos[key])); | ||||||
|  |       } if (infos[key] is List) { | ||||||
|  |         if (infos[key].isEmpty) { continue; } | ||||||
|  |         widgets.add(SubTextInputWidget(subkey: key.replaceAll("_", " "), width: 180, empty: false, readOnly: true, | ||||||
|  |               change: (value) {}, initialValue: (infos[key] as List<dynamic>).join(",") )); | ||||||
|  |       } else if (infos[key] is Map) { | ||||||
|  |           widgets.addAll(getWidgets(infos[key] as Map<String, dynamic>)); | ||||||
|  |       } else { | ||||||
|  |         widgets.add(SubTextInputWidget(subkey: key.replaceAll("_", " "), width: 180, empty: false, readOnly: true, | ||||||
|  |           change: (value) {}, initialValue: infos[key] is List ? infos[key].join(",") : "${infos[key]}" )); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (widgets.isNotEmpty) { | ||||||
|  |       widgets.add(SizedBox( width: 200, height: 15) ); | ||||||
|  |     } | ||||||
|  |     return widgets; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Widget getInputAndOutputVariableForms(bool readOnly) { | ||||||
|  |     List<Widget> res = []; | ||||||
|  |     res.add(SubKeysMapFormsWidget(dash: widget.dash, type: SubMapFormsType.INPUT, item: widget.item, elementID: widget.elementID,  | ||||||
|  |       readOnly: readOnly)); | ||||||
|  |     res.add(SubKeysMapFormsWidget(dash: widget.dash, type: SubMapFormsType.OUTPUT, item: widget.item, elementID: widget.elementID,  | ||||||
|  |       readOnly: readOnly)); | ||||||
|  |     res.add(SubKeysMapFormsWidget(dash: widget.dash, type: SubMapFormsType.ENV, item: widget.item, elementID: widget.elementID,  | ||||||
|  |         readOnly: readOnly)); | ||||||
|  |     return Column( children: res ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   @override Widget build(BuildContext context) { | ||||||
|  |     List<Widget> instancesCat = []; | ||||||
|  |     List<Widget> childrenReadOnly = getWidgets(widget.item.infos()); | ||||||
|  |     List<DropdownMenuItem<String>> dpItems = []; | ||||||
|  |     for (var (i, instance) in widget.item.instances.indexed) { | ||||||
|  |       dpItems.add(DropdownMenuItem(value: '$i', child: Text('${instance.name}', overflow: TextOverflow.ellipsis,))); | ||||||
|  |     } | ||||||
|  |     if (dpItems.isNotEmpty) { | ||||||
|  |       childrenReadOnly.add(Padding( padding: EdgeInsets.only(top: 20), | ||||||
|  |         child : Container( | ||||||
|  |           width: 200, decoration: BoxDecoration( border: Border(bottom: BorderSide(color: Colors.grey))), | ||||||
|  |         ))); | ||||||
|  |       childrenReadOnly.add(Padding(padding: EdgeInsets.only(bottom: 15), child:  | ||||||
|  |         SubDropdownInputWidget( dropdownMenuEntries: dpItems, subkey: "", width: 180, empty: false,  | ||||||
|  |           initialValue: "${widget.item.selectedInstance}", change: (value) { | ||||||
|  |             if (value != null) { setState(() { widget.item.selectedInstance = int.parse(value); }); } | ||||||
|  |           }, | ||||||
|  |         )) | ||||||
|  |       ); | ||||||
|  |       if (widget.item.instances.length > (widget.item.selectedInstance)) { | ||||||
|  |         childrenReadOnly.addAll(getWidgets(widget.item.instances[(widget.item.selectedInstance)].infos())); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     instancesCat.add(ContainerFormsWidget(dash: widget.dash, item: widget.item, elementID: widget.elementID)); | ||||||
|  |     instancesCat.add(CredentialsFormsWidget(dash: widget.dash, item: widget.item, elementID: widget.elementID)); | ||||||
|  |     if (instancesCat.isNotEmpty) { | ||||||
|  |       instancesCat.add(Container( | ||||||
|  |           width: 200, decoration: BoxDecoration( border: Border(bottom: BorderSide(color: Colors.grey))), | ||||||
|  |         )); | ||||||
|  |     } | ||||||
|  |     bool readOnly = !PermsService.getPerm(Perms.WORKFLOW_EDIT); | ||||||
|  |     // missing env input and output variables | ||||||
|  |     return SizedBox( height: getHeight(context) - 330, child: SingleChildScrollView( child:  Column(  | ||||||
|  |       children: [ getInputAndOutputVariableForms(readOnly), ...childrenReadOnly,  ...instancesCat,  ]))); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,72 +1,120 @@ | |||||||
| import 'package:alert_banner/exports.dart'; |  | ||||||
| import 'package:cron/cron.dart'; | import 'package:cron/cron.dart'; | ||||||
| import 'package:intl/intl.dart' as intl; | import 'package:oc_front/core/services/specialized_services/workflow_execution_service.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:oc_front/core/services/specialized_services/workflow_scheduler_service.dart'; | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; |  | ||||||
| import 'package:datetime_picker_formfield/datetime_picker_formfield.dart'; |  | ||||||
| import 'package:oc_front/core/models/shared_workspace_local.dart'; |  | ||||||
| import 'package:oc_front/core/services/perms_service.dart'; |  | ||||||
| import 'package:oc_front/core/services/specialized_services/workflow_service.dart'; |  | ||||||
| import 'package:oc_front/core/services/specialized_services/check_service.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:intl/intl.dart' as intl; | ||||||
|  | import 'package:alert_banner/exports.dart'; | ||||||
|  | import 'package:oc_front/models/resources/compute.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
|  | import 'package:oc_front/models/workflow.dart'; | ||||||
| import 'package:oc_front/pages/shared.dart'; | import 'package:oc_front/pages/shared.dart'; | ||||||
| import 'package:oc_front/pages/workflow.dart'; | import 'package:oc_front/pages/workflow.dart'; | ||||||
| import 'package:oc_front/widgets/dialog/alert.dart'; | import 'package:oc_front/widgets/dialog/alert.dart'; | ||||||
| import 'package:oc_front/widgets/dialog/confirm_box.dart'; | import 'package:oc_front/widgets/dialog/confirm_box.dart'; | ||||||
|  | import 'package:oc_front/core/services/perms_service.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | import 'package:oc_front/widgets/inputs/shallow_text_input.dart'; | ||||||
|  | import 'package:oc_front/core/models/shared_workspace_local.dart'; | ||||||
|  | import 'package:datetime_picker_formfield/datetime_picker_formfield.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/check_service.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/workflow_service.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class SchedulerFormsWidget extends StatefulWidget { | class SchedulerFormsWidget extends StatefulWidget { | ||||||
|  |   Scheduler schedule = Scheduler(); | ||||||
|   Dashboard item; |   Dashboard item; | ||||||
|   String purpose = "";  |  | ||||||
|   bool? booking; |   bool? booking; | ||||||
|  |   bool valid = false; | ||||||
|  |   bool shouldSearch = true; | ||||||
|  |   int scheduleFutureCount = 0; | ||||||
|  |   int scheduleBeforeCount = 0; | ||||||
|   String? error; |   String? error; | ||||||
|   String? errorEndDate; |   String? errorEndDate; | ||||||
|   String? errorCron; |   String? errorCron; | ||||||
|   Function validate = () {}; |   Function validate = () {}; | ||||||
|  |   final SchedulerService _schedulerService = SchedulerService(); | ||||||
|  |   final WorkflowExecutionService _executionService = WorkflowExecutionService(); | ||||||
|   SchedulerFormsWidget ({ super.key, required this.item, }); |   SchedulerFormsWidget ({ super.key, required this.item, }); | ||||||
|   @override SchedulerFormsWidgetState createState() => SchedulerFormsWidgetState(); |   @override SchedulerFormsWidgetState createState() => SchedulerFormsWidgetState(); | ||||||
| } | } | ||||||
| class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | ||||||
|   CheckService check = CheckService(); |   CheckService check = CheckService(); | ||||||
|   void save(List<GlobalKey<FormFieldState>> formKeys) { |   void save(List<GlobalKey<FormFieldState>> formKeys) { | ||||||
|     dash.scheduleActive = !dash.scheduleActive; |     print("save"); | ||||||
|     widget.error = null; |     widget.error = null; | ||||||
|     widget.errorEndDate = null; |     widget.errorEndDate = null; | ||||||
|     widget.errorCron = null; |     widget.errorCron = null; | ||||||
|  |     if (dash.elements.isEmpty || dash.elements.where((element) => element.element is ProcessingItem).isEmpty) { | ||||||
|  |       dash.error = "You need at least one processing element"; | ||||||
|  |     }  | ||||||
|  |     var processings = dash.elements.where((element) => element.element is ProcessingItem); | ||||||
|  |     var computes = dash.elements.where((element) => element.element is ComputeItem); | ||||||
|  |     for (var p in processings) { | ||||||
|  |       var links =  dash.arrows.where((element) => element.fromID.contains(p.id) || element.toID.contains(p.id)); | ||||||
|  |       try { | ||||||
|  |         List<ComputeItem> c = []; | ||||||
|  |         for (var link in links) { | ||||||
|  |           c.addAll(computes.where( (e) => link.toID.contains(e.id) || link.fromID.contains(e.id)).map( (e) => e.element as ComputeItem)); | ||||||
|  |         } | ||||||
|  |         if (c.isEmpty) { throw Exception("no compute element linked"); } | ||||||
|  |       } catch (e) { | ||||||
|  |         dash.error = "You need to link each processing element to a compute element"; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (dash.error != null) { | ||||||
|  |       showAlertBanner( context, () {}, AlertAlertBannerChild(text: dash.error.toString()),// <-- Put any widget here you want! | ||||||
|  |                        alertBannerLocation:  AlertBannerLocation.bottom,); | ||||||
|  |       setState(() {}); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|     for (var k in formKeys) { |     for (var k in formKeys) { | ||||||
|       if (k.currentState != null) { |       if (k.currentState != null) { | ||||||
|         if (!k.currentState!.validate() && dash.scheduleActive) { |         if (!k.currentState!.validate()) { | ||||||
|           dash.scheduleActive = !dash.scheduleActive; |  | ||||||
|           return; |           return; | ||||||
|         } else { k.currentState!.save();} |         } else { k.currentState!.save();} | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     DateTime now = DateTime.now().add(const Duration(minutes: 1)); |     DateTime now = DateTime.now().add(const Duration(minutes: 1)); | ||||||
|     if (dash.scheduler["start"] == null || DateTime.parse(dash.scheduler["start"]!).isBefore(now)) { |     if (widget.schedule.start == null || widget.schedule.start!.isBefore(now)) { | ||||||
|       dash.scheduler["start"] = now.toUtc().toIso8601String(); |       widget.schedule.start = now.toUtc(); | ||||||
|       if (dash.scheduler["end"] != null) { |       if (widget.schedule.end != null) { | ||||||
|         dash.scheduler["end"] = now.add(const Duration(minutes: 1)).toUtc().toIso8601String(); |         widget.schedule.end = now.add(const Duration(minutes: 1)).toUtc(); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     widget.item.saveDash(widget.item.id); |     Duration durationBefore = widget.schedule.start!.difference(DateTime.now().toUtc()) + Duration(seconds: 5); | ||||||
|  |     widget._schedulerService.schedule(context, widget.item.id ?? "", widget.schedule.serialize(), {}).then((value) { | ||||||
|  |       setState(() { widget.valid = true; }); | ||||||
|  |       Future.delayed(durationBefore, () { | ||||||
|  |         try { | ||||||
|  |           setState(() {}); | ||||||
|  |         } catch (e) { /* */ } | ||||||
|  |       }); | ||||||
|  |       Future.delayed(const Duration(seconds: 10), () { | ||||||
|  |         try { | ||||||
|  |           setState(() { widget.valid = false; }); | ||||||
|  |         } catch (e) { /* */ } | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|   } |   } | ||||||
|   void checkBooking(List<GlobalKey<FormFieldState>> formKeys, void Function(List<GlobalKey<FormFieldState>> )? f){ |   void checkBooking(List<GlobalKey<FormFieldState>> formKeys, void Function(List<GlobalKey<FormFieldState>> )? f){ | ||||||
|     if (widget.item.scheduler["start"] == null) { |     if (widget.schedule.start == null) { | ||||||
|       DateTime now = DateTime.now().add(const Duration(minutes: 5)); |       DateTime now = DateTime.now().add(const Duration(minutes: 5)); | ||||||
|       widget.item.scheduler["start"] = now.toUtc().toIso8601String(); |       widget.schedule.start = now.toUtc(); | ||||||
|     } |     } | ||||||
|     var s = DateTime.parse(widget.item.scheduler["start"]).toUtc().toIso8601String(); |     var s = widget.schedule.start!.toUtc().toIso8601String(); | ||||||
|     var e = ""; |     var e = ""; | ||||||
|     if (widget.item.scheduler["end"] == null) { |     if (widget.schedule.end == null) { | ||||||
|       e = DateTime.parse(widget.item.scheduler["start"]).add(const Duration(seconds: 5)).toUtc().toIso8601String(); |       e = widget.schedule.start!.add(const Duration(seconds: 5)).toUtc().toIso8601String(); | ||||||
|     } else { |     } else { | ||||||
|       e = DateTime.parse(widget.item.scheduler["end"]).toUtc().toIso8601String(); |       e = widget.schedule.end!.toUtc().toIso8601String(); | ||||||
|     } |     } | ||||||
|     check.search(context, [widget.item.id ?? "", s.substring(0, 19), e.substring(0, 19)], {}).then( |     check.search(context, [widget.item.id ?? "", s.substring(0, 19), e.substring(0, 19)], {}).then( | ||||||
|       (v) { |       (v) { | ||||||
|         if (v.data == null) { return; } |         if (v.data == null) { return; } | ||||||
|         widget.booking = v.data!.is_available; |         widget.booking = v.data!.isAvailable; | ||||||
|         if (v.data!.is_available) { |         if (v.data!.isAvailable) { | ||||||
|           if (f != null) { f(formKeys);  |           if (f != null) { f(formKeys);  | ||||||
|           } else { |           } else { | ||||||
|             showAlertBanner( context, () {},  |             showAlertBanner( context, () {},  | ||||||
| @@ -84,6 +132,31 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|  |     if (widget.shouldSearch && widget.item.name != "") { | ||||||
|  |       widget.shouldSearch = false; | ||||||
|  |       widget._executionService.search(null, [widget.item.name], {}).then((value) { | ||||||
|  |         if (value.data != null) { | ||||||
|  |           try { | ||||||
|  |             setState(() {   | ||||||
|  |             widget.scheduleFutureCount = 0; | ||||||
|  |             widget.scheduleBeforeCount = 0; | ||||||
|  |             for (var exec in value.data!.executions) { | ||||||
|  |               if (exec.startDate != null && DateTime.parse(exec.startDate!).isAfter(DateTime.now().toUtc())) { | ||||||
|  |                 widget.scheduleFutureCount++; | ||||||
|  |               } else { | ||||||
|  |                 widget.scheduleBeforeCount++; | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }); | ||||||
|  |           } catch (e) { /* */ } | ||||||
|  |            | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } else { | ||||||
|  |       Future.delayed(const Duration(milliseconds: 100), () { | ||||||
|  |         widget.shouldSearch = true; | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|     bool isService = true; |     bool isService = true; | ||||||
|     try { |     try { | ||||||
|       widget.item.elements.firstWhere((element) => (element.element?.serialize()["is_service"] ?? false) == true); |       widget.item.elements.firstWhere((element) => (element.element?.serialize()["is_service"] ?? false) == true); | ||||||
| @@ -95,22 +168,22 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|     DateTime? end; |     DateTime? end; | ||||||
|  |  | ||||||
|     Duration delayed = const Duration(minutes: 5); |     Duration delayed = const Duration(minutes: 5); | ||||||
|     if (widget.item.scheduler["start"] != null) { |     if (widget.schedule.start != null) { | ||||||
|       start = DateTime.parse(widget.item.scheduler["start"]!); |       start = widget.schedule.start!; | ||||||
|       if (start.isBefore(DateTime.now()) && !dash.scheduleActive) { |       if (start.isBefore(DateTime.now())) { | ||||||
|         start = DateTime.now().add(const Duration(minutes: 5)); |         start = DateTime.now().add(const Duration(minutes: 5)); | ||||||
|         widget.item.scheduler["start"] = start.toUtc().toIso8601String(); |         widget.schedule.start = start.toUtc(); | ||||||
|       } |       } | ||||||
|       if (start.isBefore(DateTime.now())) { |       if (start.isBefore(DateTime.now())) { | ||||||
|         // get difference between now and start |         // get difference between now and start | ||||||
|         delayed = start.difference(DateTime.now()); |         delayed = start.difference(DateTime.now()); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (widget.item.scheduler["end"] != null) { |     if (widget.schedule.end != null) { | ||||||
|       end = DateTime.parse(widget.item.scheduler["end"]!); |       end = widget.schedule.end!; | ||||||
|       if (end.isBefore(DateTime.now()) && !dash.scheduleActive) { |       if (end.isBefore(DateTime.now())) { | ||||||
|         end = DateTime.now().add(const Duration(minutes: 5)); |         end = DateTime.now().add(const Duration(minutes: 5)); | ||||||
|         widget.item.scheduler["end"] = end.toUtc().toIso8601String(); |         widget.schedule.end = end.toUtc(); | ||||||
|       } |       } | ||||||
|       if (end.isBefore(DateTime.now())) { |       if (end.isBefore(DateTime.now())) { | ||||||
|         // get difference between now and start |         // get difference between now and start | ||||||
| @@ -164,51 +237,11 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|         child: shallow ), |         child: shallow ), | ||||||
|       const SizedBox(height: 20, width: 200 ), |       const SizedBox(height: 20, width: 200 ), | ||||||
|       isService ? Text("Warning a processing is a service, if no end execution it will run forever.") : Container(), |       isService ? Text("Warning a processing is a service, if no end execution it will run forever.") : Container(), | ||||||
|       Tooltip( message: "executions name", |  | ||||||
|         child: Container( height: 40, margin: const EdgeInsets.only(top: 5), |  | ||||||
|                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), |  | ||||||
|                               child: TextFormField( key: formKeys[0], readOnly: readOnly, |  | ||||||
|                               initialValue: "${widget.item.scheduler["name"] ?? "${widget.item.name}_executions"}", |  | ||||||
|                               enabled: !dash.scheduleActive && !readOnly, |  | ||||||
|                               onChanged: (value) {  |  | ||||||
|                                 Future.delayed(const Duration(seconds: 100), () { |  | ||||||
|                                   if (widget.item.scheduler["name"] == value) {  |  | ||||||
|                                     widget.item.saveDash(widget.item.id);   |  | ||||||
|                                   } |  | ||||||
|                                 }); |  | ||||||
|                                 widget.item.scheduler["name"] = value; |  | ||||||
|                               }, |  | ||||||
|                               onSaved: (value) {  |  | ||||||
|                                 widget.item.scheduler["name"] = value ?? "${widget.item.scheduler["name"] ?? "${widget.item.name}_executions"}"; |  | ||||||
|                               }, |  | ||||||
|                               validator: (value) { |  | ||||||
|                                 if (value == null || value.isEmpty) { |  | ||||||
|                                   setState(() { widget.error = 'missing name';  });    |  | ||||||
|                                 } |  | ||||||
|                                 return value == null || value.isEmpty ? "not empty" :  null; |  | ||||||
|                               }, |  | ||||||
|                               style: const TextStyle(fontSize: 12), |  | ||||||
|                               decoration: InputDecoration( |  | ||||||
|                                 floatingLabelBehavior: FloatingLabelBehavior.always, |  | ||||||
|                                 fillColor: Colors.white, |  | ||||||
|                                 filled: true, |  | ||||||
|                                 hintText: "enter executions name...", |  | ||||||
|                                 labelText: "executions name*", |  | ||||||
|                                 hintStyle: TextStyle(fontSize: 10), |  | ||||||
|                                 labelStyle: TextStyle(fontSize: 10), |  | ||||||
|                                 focusedErrorBorder: OutlineInputBorder(borderSide: BorderSide(color: widget.item.error != null || widget.error != null ? Colors.red : Colors.black)), |  | ||||||
|                                 errorBorder: OutlineInputBorder(borderSide: BorderSide(color: widget.item.error != null || widget.error != null ? Colors.red : Colors.black)), |  | ||||||
|                                 focusedBorder: OutlineInputBorder(borderSide: BorderSide(color: widget.item.error != null || widget.error != null ? Colors.red : Colors.black)), |  | ||||||
|                                 enabledBorder: OutlineInputBorder(borderSide: BorderSide(color: widget.item.error != null || widget.error != null ? Colors.red : Colors.grey)), |  | ||||||
|                                 border: OutlineInputBorder(borderSide: BorderSide(color: widget.item.error != null || widget.error != null ? Colors.red : Colors.grey)), |  | ||||||
|                                 contentPadding: EdgeInsets.symmetric(horizontal: 10, vertical: 5), |  | ||||||
|                               ), |  | ||||||
|                           ))), |  | ||||||
|       Tooltip( message: "start executions", |       Tooltip( message: "start executions", | ||||||
|         child: Container( height: 40, margin: const EdgeInsets.only(top: 5), |         child: Container( height: 40, margin: const EdgeInsets.only(top: 5), | ||||||
|                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), |                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), | ||||||
|                               child: DateTimeField( key: formKeys[1], |                               child: DateTimeField( key: formKeys[1], | ||||||
|                                 enabled: !dash.scheduleActive && !readOnly, |                                 enabled: !readOnly, | ||||||
|                                 resetIcon: null, |                                 resetIcon: null, | ||||||
|                                 onShowPicker: (context, currentValue) async {  |                                 onShowPicker: (context, currentValue) async {  | ||||||
|                                   var date = await showDatePicker(                                     |                                   var date = await showDatePicker(                                     | ||||||
| @@ -228,7 +261,7 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|                                       return w; |                                       return w; | ||||||
|                                     }, |                                     }, | ||||||
|                                     context: context, |                                     context: context, | ||||||
|                                     firstDate: dash.scheduleActive ? DateTime(1900) : DateTime.now().add(const Duration(minutes: 5)), |                                     firstDate: DateTime.now().add(const Duration(minutes: 5)), | ||||||
|                                     initialDate: DateTime.parse( start?.toLocal().toIso8601String()  |                                     initialDate: DateTime.parse( start?.toLocal().toIso8601String()  | ||||||
|                                       ?? currentValue?.toIso8601String()  |                                       ?? currentValue?.toIso8601String()  | ||||||
|                                       ?? DateTime.now().add(const Duration(minutes: 5)).toUtc().toIso8601String()).toLocal(), |                                       ?? DateTime.now().add(const Duration(minutes: 5)).toUtc().toIso8601String()).toLocal(), | ||||||
| @@ -265,9 +298,9 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|                                       if (time == null) { return DateTime.now().add( const Duration(minutes: 1)); } |                                       if (time == null) { return DateTime.now().add( const Duration(minutes: 1)); } | ||||||
|                                       count++; |                                       count++; | ||||||
|                                       date = DateTime(date.year, date.month, date.day, time.hour, time.minute); |                                       date = DateTime(date.year, date.month, date.day, time.hour, time.minute); | ||||||
|                                       widget.item.scheduler["start"] = date.toUtc().toIso8601String();   |                                       widget.schedule.start = date.toUtc();   | ||||||
|                                     } |                                     } | ||||||
|                                     widget.item.saveDash(widget.item.id);    |                                     widget.item.saveDash(widget.item.id, context);    | ||||||
|                                   } |                                   } | ||||||
|                                   return date; |                                   return date; | ||||||
|                                 }, |                                 }, | ||||||
| @@ -296,18 +329,18 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|         child: Container( height: 40, margin: const EdgeInsets.only(top: 5),  |         child: Container( height: 40, margin: const EdgeInsets.only(top: 5),  | ||||||
|                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), |                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), | ||||||
|                               child: DateTimeField( key: formKeys[2], |                               child: DateTimeField( key: formKeys[2], | ||||||
|                                 enabled: !dash.scheduleActive && !readOnly, |                                 enabled: !readOnly, | ||||||
|                                 validator: (value) { |                                 validator: (value) { | ||||||
|                                   if (value == null && widget.item.scheduler["cron"] != null && widget.item.scheduler["cron"].isNotEmpty) { |                                   if (value == null && widget.schedule.cron != null && widget.schedule.cron!.isNotEmpty) { | ||||||
|                                     setState(() { |                                     setState(() { | ||||||
|                                       widget.errorEndDate = 'missing start date'; |                                       widget.errorEndDate = 'missing start date'; | ||||||
|                                     });    |                                     });    | ||||||
|                                   } |                                   } | ||||||
|                                   return value == null && widget.item.scheduler["cron"] != null && widget.item.scheduler["cron"].isNotEmpty ? "not empty" :  null; |                                   return value == null && widget.schedule.cron != null && widget.schedule.cron!.isNotEmpty ? "not empty" :  null; | ||||||
|                                 }, |                                 }, | ||||||
|                                 onChanged: (value) {  |                                 onChanged: (value) {  | ||||||
|                                   if (value == null) { |                                   if (value == null) { | ||||||
|                                     widget.item.scheduler.remove("end"); |                                     widget.schedule.end = null; | ||||||
|                                   } |                                   } | ||||||
|                                 }, |                                 }, | ||||||
|                                 resetIcon: const Icon(Icons.close, size: 15), |                                 resetIcon: const Icon(Icons.close, size: 15), | ||||||
| @@ -329,7 +362,7 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|                                       return w; |                                       return w; | ||||||
|                                     }, |                                     }, | ||||||
|                                     context: context, |                                     context: context, | ||||||
|                                     firstDate: dash.scheduleActive ? DateTime(1900) : DateTime.now().add(const Duration(minutes: 5)), |                                     firstDate: DateTime.now().add(const Duration(minutes: 5)), | ||||||
|                                     initialDate: DateTime.parse( end?.toLocal().toIso8601String()  |                                     initialDate: DateTime.parse( end?.toLocal().toIso8601String()  | ||||||
|                                       ?? currentValue?.toIso8601String()  |                                       ?? currentValue?.toIso8601String()  | ||||||
|                                       ?? DateTime.now().add(const Duration(minutes: 5)).toUtc().toIso8601String()).toLocal(), |                                       ?? DateTime.now().add(const Duration(minutes: 5)).toUtc().toIso8601String()).toLocal(), | ||||||
| @@ -342,7 +375,7 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|                                     var count = 0; |                                     var count = 0; | ||||||
|                                     while(date!.microsecondsSinceEpoch <= (DateTime.now().microsecondsSinceEpoch)  |                                     while(date!.microsecondsSinceEpoch <= (DateTime.now().microsecondsSinceEpoch)  | ||||||
|                                       || time == null |                                       || time == null | ||||||
|                                       || (date.microsecondsSinceEpoch) <= (DateTime.parse(widget.item.scheduler["start"] ?? DateTime.now().toIso8601String()).microsecondsSinceEpoch)) { |                                       || (date.microsecondsSinceEpoch) <= (widget.schedule.start  ?? DateTime.now()).microsecondsSinceEpoch) { | ||||||
|                                       if (count > 0) {  |                                       if (count > 0) {  | ||||||
|                                         showAlertBanner( context, () {},  // ignore: use_build_context_synchronously |                                         showAlertBanner( context, () {},  // ignore: use_build_context_synchronously | ||||||
|                                         const AlertAlertBannerChild(text: "must be at least 1 minute from now to let system check info && upper starting date"),// <-- Put any widget here you want! |                                         const AlertAlertBannerChild(text: "must be at least 1 minute from now to let system check info && upper starting date"),// <-- Put any widget here you want! | ||||||
| @@ -370,9 +403,9 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|                                       if (time == null) { return null; } |                                       if (time == null) { return null; } | ||||||
|                                       count++; |                                       count++; | ||||||
|                                       date = DateTime(date.year, date.month, date.day, time.hour, time.minute); |                                       date = DateTime(date.year, date.month, date.day, time.hour, time.minute); | ||||||
|                                       widget.item.scheduler["end"] = date.toUtc().toIso8601String();    |                                       widget.schedule.end = date.toUtc();    | ||||||
|                                     } |                                     } | ||||||
|                                     widget.item.saveDash(widget.item.id);                           |                                     widget.item.saveDash(widget.item.id, context);                           | ||||||
|                                   } |                                   } | ||||||
|                                   return date; |                                   return date; | ||||||
|                                 }, |                                 }, | ||||||
| @@ -400,19 +433,19 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|         child: Container( height: 40, margin: const EdgeInsets.only(top: 5), |         child: Container( height: 40, margin: const EdgeInsets.only(top: 5), | ||||||
|                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), |                               padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), | ||||||
|                               child: TextFormField( key: formKeys[3], |                               child: TextFormField( key: formKeys[3], | ||||||
|                               enabled: !dash.scheduleActive && !readOnly, |                               enabled: !readOnly, | ||||||
|                               initialValue: widget.item.scheduler["cron"], |                               initialValue: widget.schedule.cron, | ||||||
|                               onChanged: (value) {  |                               onChanged: (value) {  | ||||||
|                                 Future.delayed(const Duration(seconds: 100), () { |                                 Future.delayed(const Duration(seconds: 100), () { | ||||||
|                                   if (widget.item.scheduler["cron"] == value) {  |                                   if (widget.schedule.cron  == value) {  | ||||||
|                                     widget.item.saveDash(widget.item.id);   |                                     widget.item.saveDash(widget.item.id, context);   | ||||||
|                                   } |                                   } | ||||||
|                                 }); |                                 }); | ||||||
|                                 widget.item.scheduler["cron"] = value; |                                 widget.schedule.cron = value; | ||||||
|                               }, |                               }, | ||||||
|                               onSaved: (value) {  |                               onSaved: (value) {  | ||||||
|                                 if (value != null) { |                                 if (value != null) { | ||||||
|                                   widget.item.scheduler["cron"] = value; |                                   widget.schedule.cron = value; | ||||||
|                                 } |                                 } | ||||||
|                               }, |                               }, | ||||||
|                               validator: (value) {  |                               validator: (value) {  | ||||||
| @@ -450,55 +483,65 @@ class SchedulerFormsWidgetState extends State<SchedulerFormsWidget> { | |||||||
|       Container( |       Container( | ||||||
|         width: 200, |         width: 200, | ||||||
|         height: 20, |         height: 20, | ||||||
|         decoration: const BoxDecoration(border: Border(bottom: BorderSide(color: Colors.grey, width: 1))), |         decoration: BoxDecoration(border: Border(bottom: BorderSide(color:  | ||||||
|  |         PermsService.getPerm(Perms.WORKFLOW_BOOKING)? Colors.grey : Colors.transparent, width: 1))), | ||||||
|       ), |       ), | ||||||
|       const SizedBox( |       const SizedBox( | ||||||
|         width: 200, |         width: 200, | ||||||
|         height: 10, |         height: 10, | ||||||
|       ), |       ), | ||||||
|       Tooltip( message: "check booking", |       PermsService.getPerm(Perms.WORKFLOW_BOOKING)? Tooltip( message: "check booking", child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|                               child: InkWell( mouseCursor: SystemMouseCursors.click,  |         onTap: () { PermsService.getPerm(Perms.WORKFLOW_BOOKING) ? checkBooking(formKeys, null) : null; | ||||||
|                               onTap: () { |         }, child: Container( margin: const EdgeInsets.only(bottom: 5, left: 10, right: 10), | ||||||
|                                 PermsService.getPerm(Perms.WORKFLOW_BOOKING) ? checkBooking(formKeys, null) : null; |                               decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  | ||||||
|                               }, child: Container( margin: const EdgeInsets.only(bottom: 5, left: 10, right: 10), |                                   border: Border.all(color: widget.booking == null ? ( | ||||||
|                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  |  | ||||||
|                                   border: Border.all(color: widget.booking == null && !dash.scheduleActive ? ( |  | ||||||
|                                     PermsService.getPerm(Perms.WORKFLOW_BOOKING) && PermsService.getPerm(Perms.WORKFLOW_EDIT) ? Colors.black : Colors.grey) : ( |                                     PermsService.getPerm(Perms.WORKFLOW_BOOKING) && PermsService.getPerm(Perms.WORKFLOW_EDIT) ? Colors.black : Colors.grey) : ( | ||||||
|                                       widget.booking == true || dash.scheduleActive  ? Colors.green : redColor), width: 1)), |                                       widget.booking == true  ? Colors.green : redColor), width: 1)), | ||||||
|                                 width: 200, height: 30, |                                 width: 200, height: 30, | ||||||
|                                 child: Icon( |                                 child: Icon( Icons.verified_outlined,  | ||||||
|                                   Icons.verified_outlined,  |                                   color: widget.booking == null ? Colors.black :  (widget.booking == true ? Colors.green : redColor)), | ||||||
|                                   color: widget.booking == null && !dash.scheduleActive ? Colors.black :  (widget.booking == true || dash.scheduleActive ? Colors.green : redColor)), |  | ||||||
|                               )) |                               )) | ||||||
|                             ), |                             ): Container(), | ||||||
|       Tooltip( message: dash.scheduleActive ? "unbook" : "book", |       PermsService.getPerm(Perms.WORKFLOW_BOOKING) ? Tooltip( message: "book", child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|                               child: InkWell( mouseCursor: SystemMouseCursors.click,  |  | ||||||
|                               onTap: () { |                               onTap: () { | ||||||
|                                 PermsService.getPerm(Perms.WORKFLOW_BOOKING) && PermsService.getPerm(Perms.WORKFLOW_EDIT) ? (dash.scheduleActive ? setState(() {  save(formKeys); }) : checkBooking(formKeys,  save)) : null; |                                 setState(() {  save(formKeys); }); | ||||||
|                               }, child: Container( margin: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), |                               }, child: Container( margin: const EdgeInsets.only(top: 5, bottom: 10, left: 10, right: 10), | ||||||
|                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  |                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  | ||||||
|                                   border: Border.all(color: dash.scheduleActive ? Colors.green : ( PermsService.getPerm(Perms.WORKFLOW_BOOKING) ?Colors.black : Colors.grey ))), |                                   border: Border.all(color: dash.error != null ? Colors.red : ( PermsService.getPerm(Perms.WORKFLOW_BOOKING) ?(widget.valid ? Colors.green : Colors.black) : Colors.grey ))), | ||||||
|                                 width: 200, height: 30, |                                 width: 200, height: 30, | ||||||
|                                 child: Icon( |                                 child: Icon(Icons.schedule_send, color: dash.error != null ? Colors.red : (widget.valid ? Colors.green : Colors.black)), | ||||||
|                                   dash.scheduleActive ? Icons.cancel_schedule_send : Icons.schedule_send, color: dash.scheduleActive ? Colors.green : Colors.black), |  | ||||||
|                               )) |                               )) | ||||||
|                             ), |                             ) : Container(), | ||||||
|  |       Column( children: [ | ||||||
|  |         Container( | ||||||
|  |           height: 15, width: 200, | ||||||
|  |           decoration: const BoxDecoration(border: Border(top: BorderSide(color: Colors.grey))), | ||||||
|  |         ), | ||||||
|  |         Container( alignment: Alignment.center, padding: const EdgeInsets.symmetric(horizontal: 10), | ||||||
|  |           child:Text( textAlign: TextAlign.center, overflow: TextOverflow.ellipsis, | ||||||
|  |             style: const TextStyle(fontSize: 13, color: Colors.grey ), | ||||||
|  |             "Was booked ${widget.scheduleBeforeCount} times.")), | ||||||
|  |         Container( alignment: Alignment.center, padding: const EdgeInsets.only(left: 10, right: 10, bottom: 15), | ||||||
|  |           child:Text( textAlign: TextAlign.center, overflow: TextOverflow.ellipsis, | ||||||
|  |             style: const TextStyle(fontSize: 13, color: Colors.grey ), | ||||||
|  |             "Is booked ${widget.scheduleFutureCount} times.")), | ||||||
|  |       ]), | ||||||
|       widget.item.info["shared"] != null && (widget.item.info["shared"] as List<dynamic>).isNotEmpty ? Column( children: [ |       widget.item.info["shared"] != null && (widget.item.info["shared"] as List<dynamic>).isNotEmpty ? Column( children: [ | ||||||
|         Container( |         Container( | ||||||
|           height: 30, |           height: 30, | ||||||
|           width: 200, |           width: 200, | ||||||
|           decoration: const BoxDecoration(border: Border(top: BorderSide(color: Colors.grey))), |           decoration: const BoxDecoration(border: Border(top: BorderSide(color: Colors.grey))), | ||||||
|         ), |         ), | ||||||
|         Container( alignment: Alignment.center, padding: const EdgeInsets.symmetric(horizontal: 10), |         Container( alignment: Alignment.center, | ||||||
|           child:Text( textAlign: TextAlign.center, overflow: TextOverflow.ellipsis, |           child:Text( overflow: TextOverflow.ellipsis, | ||||||
|             style: const TextStyle(fontSize: 14, color: Colors.black, fontWeight: FontWeight.bold), |             style: const TextStyle(fontSize: 14, color: Colors.black, fontWeight: FontWeight.bold), | ||||||
|             "Workflow is shared in ${(widget.item.info["shared"] as List<dynamic>).length} workspace(s)")), |             "Is shared ${(widget.item.info["shared"] as List<dynamic>).length} time(s).")), | ||||||
|         ...(widget.item.info["shared"] as List<dynamic>).where( (e) => CollaborativeAreaLocal.getCollaborativeArea(e) != null  |         ...(widget.item.info["shared"] as List<dynamic>).where( (e) => CollaborativeAreaLocal.getCollaborativeArea(e) != null  | ||||||
|         ).map((e) { |         ).map((e) { | ||||||
|           var sw = CollaborativeAreaLocal.getCollaborativeArea(e); |           var sw = CollaborativeAreaLocal.getCollaborativeArea(e); | ||||||
|           return Container( alignment: Alignment.center, padding: const EdgeInsets.symmetric(horizontal: 10, vertical: 5), |           return Container( alignment: Alignment.center, padding: const EdgeInsets.symmetric(horizontal: 10, vertical: 5), | ||||||
|             child: Row(  children: [  |             child: Row(  children: [  | ||||||
|               const Padding(padding: EdgeInsets.only(right: 10), child: Icon(Icons.work, color: Colors.grey, size: 15)), |               const Padding(padding: EdgeInsets.only( left: 10, right: 10), child: Icon(Icons.work, color: Colors.grey, size: 15)), | ||||||
|               Text(style: const TextStyle(fontSize: 12, color: Colors.grey), |               Text(style: const TextStyle(fontSize: 12, color: Colors.grey), | ||||||
|               "Workspace: ${sw != null && sw.name != null ?  |               "Workspace: ${sw != null && sw.name != null ?  | ||||||
|               "${sw.name!.substring(0, sw.name!.length > 15 ? 12 : sw.name!.length)}${sw.name!.length > 15 ? "..." : ""}" : ""}") ])); |               "${sw.name!.substring(0, sw.name!.length > 15 ? 12 : sw.name!.length)}${sw.name!.length > 15 ? "..." : ""}" : ""}") ])); | ||||||
|   | |||||||
| @@ -1,18 +0,0 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/widgets/forms/web_reference_forms.dart'; |  | ||||||
|  |  | ||||||
| class StorageFormsWidget extends StatefulWidget { |  | ||||||
|   StorageItem item; |  | ||||||
|   String purpose = "";  |  | ||||||
|   Function validate = () {}; |  | ||||||
|   StorageFormsWidget ({ super.key, required this.item }); |  | ||||||
|   @override StorageFormsWidgetState createState() => StorageFormsWidgetState(); |  | ||||||
| } |  | ||||||
| class StorageFormsWidgetState extends State<StorageFormsWidget> { |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     return Column( children: [ |  | ||||||
|       WebReferenceFormsWidget(item: widget.item), |  | ||||||
|     ]); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
							
								
								
									
										154
									
								
								lib/widgets/forms/storage_processing_link_forms.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								lib/widgets/forms/storage_processing_link_forms.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,154 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_advanced_switch/flutter_advanced_switch.dart'; | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/models/workflow.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
|  | class StorageProcessingLinkFormsWidget extends StatefulWidget { | ||||||
|  |   Dashboard dash; | ||||||
|  |   ArrowPainter item; | ||||||
|  |   bool readOnly = false; | ||||||
|  |   StorageProcessingLinkFormsWidget ({ super.key, required this.dash, required this.item, this.readOnly = false }); | ||||||
|  |   @override StorageProcessingLinkFormsWidgetState createState() => StorageProcessingLinkFormsWidgetState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class StorageProcessingLinkFormsWidgetState extends State<StorageProcessingLinkFormsWidget> { | ||||||
|  |  | ||||||
|  |   List<Param> getParams(String fromID, String toID) { | ||||||
|  |     List<Param> arr = []; | ||||||
|  |     var els = widget.dash.elements.where( (e) => fromID.contains(e.id) || toID.contains(e.id)); | ||||||
|  |     for (var element in els) { | ||||||
|  |       var g = GraphItem(); | ||||||
|  |       g.fromDashboard(element.serialize()); | ||||||
|  |       var e = g.getElement(); | ||||||
|  |       if (e == null) { continue; } | ||||||
|  |       if (e.getSelectedInstance() != null) { | ||||||
|  |         for (var env in e.getSelectedInstance()!.env) { | ||||||
|  |           if (env.name?.contains("LINK") ?? true) { continue; } | ||||||
|  |           arr.add(env); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return arr.where( (e) => e.name != null && e.value != null).toList(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   @override Widget build(BuildContext context) { | ||||||
|  |     List<Widget> children = [ | ||||||
|  |       Padding( padding: const EdgeInsets.only(top: 10), | ||||||
|  |           child: Text("<ENV VARIABLES>",  | ||||||
|  |             style: const TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center)), | ||||||
|  |     ]; | ||||||
|  |     var params = getParams(widget.item.fromID, widget.item.toID); | ||||||
|  |     widget.item.env = params.map( (e) => e.serialize()).toList(); | ||||||
|  |     for (var param in params) { | ||||||
|  |       children.add(SubTextInputWidget(subkey: param.name ?? "", width: 180, empty: false, readOnly: true, | ||||||
|  |           change: (value) {}, initialValue: param.value ?? "", copyLabel: true)); | ||||||
|  |     } | ||||||
|  |     children.add(Container(width: 200, padding: const EdgeInsets.only(top: 10, bottom: 10), | ||||||
|  |           decoration: const BoxDecoration(border: Border(bottom: BorderSide(color: Colors.grey, width: 1))))); | ||||||
|  |     List<Widget> inf = []; | ||||||
|  |     int count = 0; | ||||||
|  |     for(var info in widget.item.infos) { | ||||||
|  |       count++; | ||||||
|  |       inf.add(Padding( padding: EdgeInsets.only(top: 10, bottom: 5),  | ||||||
|  |         child : Row( crossAxisAlignment: CrossAxisAlignment.center, children: [ | ||||||
|  |             Padding(padding: EdgeInsets.only(left: 10, right: 10), child: Text("N°$count")), | ||||||
|  |             Container(width: 140, decoration: const BoxDecoration(border: Border(bottom: BorderSide(color: Colors.grey, width: 1)))) | ||||||
|  |           ]))); | ||||||
|  |       for (var key in (info as Map<String, dynamic>).keys) { | ||||||
|  |           if (info[key] is bool) { | ||||||
|  |             inf.add(AdvancedSwitch( | ||||||
|  |               width: 150.0, | ||||||
|  |               height: 25.0, | ||||||
|  |               initialValue: info[key] as bool? ?? false, | ||||||
|  |               enabled: !widget.readOnly, | ||||||
|  |               activeChild: Text(key, style: const TextStyle(color: Colors.white)), | ||||||
|  |               inactiveChild: Text(key == "write" ? "read" : "no $key", style: const TextStyle(color: Colors.white)), | ||||||
|  |               activeColor: Colors.green, | ||||||
|  |               inactiveColor: redColor, | ||||||
|  |               onChanged: (v) { | ||||||
|  |                 try {  | ||||||
|  |                   info[key] = v; | ||||||
|  |                   Future.delayed(const Duration(seconds: 1), () { | ||||||
|  |                     for (var a in widget.dash.arrows) { | ||||||
|  |                       if (a.fromID == widget.item.fromID && a.toID == widget.item.toID) { | ||||||
|  |                         a.infos = widget.item.infos; | ||||||
|  |                       } | ||||||
|  |                     } | ||||||
|  |                     widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |                   }); | ||||||
|  |                 } catch (e) { /* */ } | ||||||
|  |                  | ||||||
|  |               }, | ||||||
|  |             )); | ||||||
|  |           } else if (info[key] is String?) { | ||||||
|  |             inf.add(SubTextInputWidget(subkey: key, width: 180, empty: false, change: (v) { | ||||||
|  |               try { | ||||||
|  |                 info[key] = v; | ||||||
|  |                 Future.delayed(const Duration(seconds: 1), () { | ||||||
|  |                   if (info[key] != v) { return; } | ||||||
|  |                   for (var a in widget.dash.arrows) { | ||||||
|  |                     if (a.fromID == widget.item.fromID && a.toID == widget.item.toID) { | ||||||
|  |                       a.infos = widget.item.infos; | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                   widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |                 }); | ||||||
|  |               } catch (e) { | ||||||
|  |                 print(e); | ||||||
|  |               } | ||||||
|  |                  | ||||||
|  |             }, initialValue: "${info[key] ?? ""}", readOnly: false, noLabel: false)); | ||||||
|  |           } | ||||||
|  |            | ||||||
|  |            | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return SizedBox( height: getHeight(context) - 230, child: SingleChildScrollView( child: Column(children: [ | ||||||
|  |       ...children, | ||||||
|  |       Row( mainAxisAlignment: MainAxisAlignment.center, children: [ | ||||||
|  |               InkWell( onTap: () { | ||||||
|  |                   widget.item.infos.add({ | ||||||
|  |                             "write": false, | ||||||
|  |                             "source": null, | ||||||
|  |                             "destination": null, | ||||||
|  |                             "filename": null, | ||||||
|  |                   }); | ||||||
|  |                   setState(() { | ||||||
|  |                     widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |                   }); | ||||||
|  |                 }, child: Container( margin: const EdgeInsets.only(top: 10), | ||||||
|  |                     decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  | ||||||
|  |                     border: Border.all(color: Colors.grey, width: 1)), | ||||||
|  |                     width: 125, height: 30, | ||||||
|  |                     child: Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                       children: [ Padding( padding: EdgeInsets.only(right: 5), child: Icon(Icons.add, color: Colors.black,)),  | ||||||
|  |                       Text("link infos", | ||||||
|  |                         style: TextStyle( color: Colors.black))]), | ||||||
|  |                   ),   | ||||||
|  |               ), | ||||||
|  |               InkWell( onTap: () { | ||||||
|  |                       if (widget.item.infos.isEmpty) { return; } | ||||||
|  |                       setState(() {  | ||||||
|  |                         widget.item.infos.removeLast(); | ||||||
|  |                         widget.dash.saveDash(widget.dash.id, context);  | ||||||
|  |                       }); | ||||||
|  |                     }, child: | ||||||
|  |                     Container( margin: const EdgeInsets.only(left: 5, top: 10), | ||||||
|  |                       decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  | ||||||
|  |                         border: Border.all(color: widget.item.infos.isEmpty ? Colors.grey : Colors.black , width: 1)), | ||||||
|  |                       width: 50, height: 30, | ||||||
|  |                       child: Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                         children: [ Icon( Icons.delete, size: 18, | ||||||
|  |                           color: widget.item.infos.isEmpty ? Colors.grey : Colors.black ) ]), | ||||||
|  |                     ) | ||||||
|  |                   ),  | ||||||
|  |             ]), | ||||||
|  |           ...inf | ||||||
|  |         ] ) )); | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										174
									
								
								lib/widgets/forms/sub_add_forms.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								lib/widgets/forms/sub_add_forms.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/widgets/forms/sub_keys_forms.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
|  | class SubAddFormsWidget extends StatefulWidget { | ||||||
|  |   bool readOnly; | ||||||
|  |   SubMapFormsType type; | ||||||
|  |   String elementID = ""; | ||||||
|  |   AbstractItem item; | ||||||
|  |   Dashboard dash;  | ||||||
|  |   bool empty = false; | ||||||
|  |   List<Param> forms = []; | ||||||
|  |   SubAddFormsWidget ({ super.key, required this.dash, this.readOnly = false, | ||||||
|  |     this.empty = false, required this.item, required this.elementID, | ||||||
|  |     required this.type }); | ||||||
|  |   @override SubAddFormsWidgetState createState() => SubAddFormsWidgetState(); | ||||||
|  | } | ||||||
|  | class SubAddFormsWidgetState extends State<SubAddFormsWidget> { | ||||||
|  |   Param? addedparam; | ||||||
|  |   @override Widget build(BuildContext context) { | ||||||
|  |     AbstractInstance instance = widget.item.getSelectedInstance()!; | ||||||
|  |     var f = (widget.type == SubMapFormsType.INPUT ? instance.inputs : (  | ||||||
|  |       widget.type == SubMapFormsType.OUTPUT ? instance.outputs : instance.env)).where( (e) => !e.readOnly).toList(); | ||||||
|  |     widget.forms = f; | ||||||
|  |     if (addedparam != null) { | ||||||
|  |       widget.forms.add(addedparam!); | ||||||
|  |     } | ||||||
|  |     List<Widget> children = []; | ||||||
|  |     for (var param in widget.forms) { | ||||||
|  |       children.add(Row( mainAxisAlignment: MainAxisAlignment.center, children: [ | ||||||
|  |         SubTextInputWidget(subkey: "key", readOnly: widget.readOnly, | ||||||
|  |           initialValue: param.name, width: 81, empty: widget.empty,  | ||||||
|  |           change: (value) {  | ||||||
|  |             setState(() { | ||||||
|  |               param.name = value; | ||||||
|  |               param.attr = value; | ||||||
|  |               if (addedparam?.value != null && addedparam?.name != null) { | ||||||
|  |                 addedparam = null; | ||||||
|  |               } | ||||||
|  |               var sel = widget.item.getSelectedInstance(); | ||||||
|  |               if (sel != null) { | ||||||
|  |                 if (widget.type == SubMapFormsType.INPUT) { sel.inputs = [  | ||||||
|  |                     ...sel.inputs.where( (e) => e.readOnly), | ||||||
|  |                     ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                   ]; | ||||||
|  |                 } else if (widget.type == SubMapFormsType.OUTPUT) {   | ||||||
|  |                   sel.outputs = [ | ||||||
|  |                     ...sel.outputs.where( (e) => e.readOnly), | ||||||
|  |                     ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                   ]; | ||||||
|  |                 } else {  | ||||||
|  |                   sel.env = [ | ||||||
|  |                     ...sel.env.where( (e) => e.readOnly), | ||||||
|  |                     ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                   ];  | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             Future.delayed(const Duration(milliseconds: 100), () { | ||||||
|  |               if (param.name == value) { | ||||||
|  |                 widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |             var el = widget.dash.getElement(widget.elementID); | ||||||
|  |             el!.element = widget.item as dynamic; | ||||||
|  |             setState(() {}); | ||||||
|  |           }); | ||||||
|  |         }), | ||||||
|  |         const Padding(padding: EdgeInsets.only(left: 5, right: 5, top: 15), child: Text("=", textAlign: TextAlign.center,)), | ||||||
|  |         SubTextInputWidget(subkey: "value", initialValue: param.value, width: 81, empty: widget.empty,  | ||||||
|  |           readOnly: widget.readOnly, change: (value) {  | ||||||
|  |             param.value = value; | ||||||
|  |             if (addedparam?.value != null && addedparam?.name != null) { | ||||||
|  |               addedparam = null; | ||||||
|  |             } | ||||||
|  |             var sel = widget.item.getSelectedInstance(); | ||||||
|  |             if (sel != null) { | ||||||
|  |                 if (widget.type == SubMapFormsType.INPUT) { sel.inputs = [  | ||||||
|  |                     ...sel.inputs.where( (e) => e.readOnly), | ||||||
|  |                     ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                   ]; | ||||||
|  |                 } else if (widget.type == SubMapFormsType.OUTPUT) {   | ||||||
|  |                   sel.outputs = [ | ||||||
|  |                     ...sel.outputs.where( (e) => e.readOnly), | ||||||
|  |                     ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                   ]; | ||||||
|  |                 } else {  | ||||||
|  |                   sel.env = [ | ||||||
|  |                     ...sel.env.where( (e) => e.readOnly ), | ||||||
|  |                     ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                   ];  | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             var el = widget.dash.getElement(widget.elementID); | ||||||
|  |             el!.element = widget.item as dynamic; | ||||||
|  |             setState(() {}); | ||||||
|  |             Future.delayed(const Duration(milliseconds: 100), () { | ||||||
|  |               if (param.value == value) { | ||||||
|  |                 widget.dash.saveDash(widget.dash.id, context); | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |         }),      | ||||||
|  |       ])); | ||||||
|  |     } | ||||||
|  |     return Column( crossAxisAlignment: CrossAxisAlignment.center, children : [ | ||||||
|  |       widget.readOnly ? Container() : Row( mainAxisAlignment: MainAxisAlignment.center, children: [ | ||||||
|  |               InkWell( onTap: () { | ||||||
|  |                   if (widget.forms.isNotEmpty && (widget.forms[widget.forms.length - 1].name == null | ||||||
|  |                   || widget.forms[widget.forms.length - 1].value == null)) { return;} | ||||||
|  |                   widget.forms.add(Param(readOnly: false)); | ||||||
|  |                   addedparam = widget.forms.last; | ||||||
|  |                   var el = widget.dash.getElement(widget.elementID); | ||||||
|  |                   el!.element = widget.item as dynamic; | ||||||
|  |                   setState(() {}); | ||||||
|  |                 }, child: Container( margin: const EdgeInsets.only(top: 10), | ||||||
|  |                     decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  | ||||||
|  |                     border: Border.all(color: widget.forms.isNotEmpty && (widget.forms[widget.forms.length - 1].name == null | ||||||
|  |                   || widget.forms[widget.forms.length - 1].value == null) ? Colors.grey : Colors.grey, width: 1)), | ||||||
|  |                     width: 125, height: 30, | ||||||
|  |                     child: Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                       children: [ Padding( padding: EdgeInsets.only(right: 5), child: Icon(Icons.add, color: widget.forms.isNotEmpty && (widget.forms[widget.forms.length - 1].name == null | ||||||
|  |                   || widget.forms[widget.forms.length - 1].value == null) ? Colors.grey : Colors.black,)),  | ||||||
|  |                       Text(widget.type == SubMapFormsType.ENV ? "env" : (widget.type == SubMapFormsType.INPUT ? "input" : "output"), | ||||||
|  |                         style: TextStyle( color: widget.forms.isNotEmpty && (widget.forms[widget.forms.length - 1].name == null | ||||||
|  |                   || widget.forms[widget.forms.length - 1].value == null) ? Colors.grey : Colors.black))]), | ||||||
|  |                   ),   | ||||||
|  |               ), | ||||||
|  |               InkWell( onTap: () { | ||||||
|  |                       if (addedparam != null) { | ||||||
|  |                         addedparam = null; | ||||||
|  |                         return setState(() {}); | ||||||
|  |                       } | ||||||
|  |                       if (widget.forms.isEmpty) { return; } | ||||||
|  |                       widget.forms.sublist(0, widget.forms.length - 1); | ||||||
|  |                       var el = widget.dash.getElement(widget.elementID); | ||||||
|  |                       var sel = widget.item.getSelectedInstance(); | ||||||
|  |                       if (sel != null) { | ||||||
|  |                           widget.forms.removeLast(); | ||||||
|  |                           if (widget.type == SubMapFormsType.INPUT) { sel.inputs = [  | ||||||
|  |                               ...sel.inputs.where( (e) => e.readOnly), | ||||||
|  |                               ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                             ]; | ||||||
|  |                           } else if (widget.type == SubMapFormsType.OUTPUT) {   | ||||||
|  |                             sel.outputs = [ | ||||||
|  |                               ...sel.outputs.where( (e) => e.readOnly), | ||||||
|  |                               ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                             ]; | ||||||
|  |                           } else {  | ||||||
|  |                             sel.env = [ | ||||||
|  |                               ...sel.env.where( (e) => e.readOnly), | ||||||
|  |                               ...widget.forms.where((e) => e.name != null && e.value != null) | ||||||
|  |                             ];  | ||||||
|  |                           } | ||||||
|  |                       } | ||||||
|  |                       el!.element = widget.item as dynamic; | ||||||
|  |                       setState(() { widget.dash.saveDash(widget.dash.id, context); }); | ||||||
|  |                     }, child: | ||||||
|  |                     Container( margin: const EdgeInsets.only(left: 5, top: 10), | ||||||
|  |                       decoration: BoxDecoration(borderRadius: BorderRadius.circular(5),  | ||||||
|  |                         border: Border.all(color: widget.forms.isEmpty ? Colors.grey : Colors.black , width: 1)), | ||||||
|  |                       width: 50, height: 30, | ||||||
|  |                       child: Row( mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                         children: [ Icon( Icons.delete, size: 18, | ||||||
|  |                           color: widget.forms.isEmpty ? Colors.grey : Colors.black ) ]), | ||||||
|  |                     ) | ||||||
|  |                   ),  | ||||||
|  |             ]), | ||||||
|  |       ...children | ||||||
|  |     ]);  | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,9 +1,10 @@ | |||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/models/resources/processing.dart'; | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class SubExposeFormsWidget extends StatefulWidget { | class SubExposeFormsWidget extends StatefulWidget { | ||||||
|   bool readOnly; |   bool readOnly; | ||||||
|   Expose item; |   Expose item; | ||||||
| @@ -29,7 +30,7 @@ class SubExposeFormsWidgetState extends State<SubExposeFormsWidget> { | |||||||
|                     widget.item.port = int.parse(value); |                     widget.item.port = int.parse(value); | ||||||
|                     Future.delayed(const Duration(seconds: 2), () { |                     Future.delayed(const Duration(seconds: 2), () { | ||||||
|                       if (widget.item.port == int.parse(value) && int.parse(value) != 0) { |                       if (widget.item.port == int.parse(value) && int.parse(value) != 0) { | ||||||
|                         widget.dash.saveDash(widget.dash.id); |                         widget.dash.saveDash(widget.dash.id, context); | ||||||
|                       } |                       } | ||||||
|                     }); |                     }); | ||||||
|                   } catch (e) {  widget.item.port = null; } |                   } catch (e) {  widget.item.port = null; } | ||||||
| @@ -44,7 +45,7 @@ class SubExposeFormsWidgetState extends State<SubExposeFormsWidget> { | |||||||
|                     widget.item.PAT = int.parse(value); |                     widget.item.PAT = int.parse(value); | ||||||
|                     Future.delayed(const Duration(seconds: 2), () { |                     Future.delayed(const Duration(seconds: 2), () { | ||||||
|                       if (widget.item.PAT == int.parse(value) && int.parse(value) != 0) { |                       if (widget.item.PAT == int.parse(value) && int.parse(value) != 0) { | ||||||
|                         widget.dash.saveDash(widget.dash.id); |                         widget.dash.saveDash(widget.dash.id, context); | ||||||
|                       } |                       } | ||||||
|                     }); |                     }); | ||||||
|                   } catch (e) { widget.item.PAT = null;  } |                   } catch (e) { widget.item.PAT = null;  } | ||||||
| @@ -57,7 +58,7 @@ class SubExposeFormsWidgetState extends State<SubExposeFormsWidget> { | |||||||
|                   try { |                   try { | ||||||
|                     widget.item.path = value; |                     widget.item.path = value; | ||||||
|                     Future.delayed(const Duration(seconds: 2), () { |                     Future.delayed(const Duration(seconds: 2), () { | ||||||
|                       if (widget.item.path == value) {  widget.dash.saveDash(widget.dash.id); } |                       if (widget.item.path == value) {  widget.dash.saveDash(widget.dash.id, context); } | ||||||
|                     }); |                     }); | ||||||
|                   } catch (e) { widget.item.path = null;  } |                   } catch (e) { widget.item.path = null;  } | ||||||
|                   var el = widget.dash.getElement(widget.elementID); |                   var el = widget.dash.getElement(widget.elementID); | ||||||
|   | |||||||
| @@ -1,62 +1,97 @@ | |||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_advanced_switch/flutter_advanced_switch.dart'; | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:oc_front/main.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
|  | import 'package:oc_front/widgets/forms/sub_add_forms.dart'; | ||||||
| import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | import 'package:oc_front/widgets/inputs/sub_text_input.dart'; | ||||||
|  | enum SubMapFormsType {  | ||||||
|  |   ENV,  | ||||||
|  |   INPUT, | ||||||
|  |   OUTPUT  | ||||||
|  | } | ||||||
|  | // ignore: must_be_immutable | ||||||
| class SubKeysMapFormsWidget extends StatefulWidget { | class SubKeysMapFormsWidget extends StatefulWidget { | ||||||
|   bool readOnly = false; |   bool readOnly = false; | ||||||
|   FlowData item; |   AbstractItem item; | ||||||
|   Dashboard dash;  |   Dashboard dash;  | ||||||
|   String varKey = ""; |  | ||||||
|   bool empty = false; |   bool empty = false; | ||||||
|   bool isInput = true; |   SubMapFormsType type; | ||||||
|   String elementID = ""; |   String elementID = ""; | ||||||
|   String categoryKey = ""; |   SubKeysMapFormsWidget({ super.key, required this.dash, required this.type, this.readOnly = false, | ||||||
|   List<GraphItem> graphItems = []; |     this.empty = false, required this.item, required this.elementID }); | ||||||
|   SubKeysMapFormsWidget({ super.key, required this.dash, required this.isInput, this.readOnly = false, |  | ||||||
|     this.empty = false, required this.item, required this.elementID, required this.graphItems, |  | ||||||
|     required this.categoryKey, required this.varKey }); |  | ||||||
|   @override SubKeysMapFormsWidgetState createState() => SubKeysMapFormsWidgetState(); |   @override SubKeysMapFormsWidgetState createState() => SubKeysMapFormsWidgetState(); | ||||||
| } | } | ||||||
| class SubKeysMapFormsWidgetState extends State<SubKeysMapFormsWidget> { | class SubKeysMapFormsWidgetState extends State<SubKeysMapFormsWidget> { | ||||||
|  |   bool _save = false; | ||||||
|  |  | ||||||
|  |   List<Param> getParams(String fromID, String toID) { | ||||||
|  |     List<Param> arr = []; | ||||||
|  |     var els = widget.dash.elements.where( (e) => fromID.contains(e.id) || toID.contains(e.id)); | ||||||
|  |     for (var element in els) { | ||||||
|  |       var g = GraphItem(); | ||||||
|  |       g.fromDashboard(element.serialize()); | ||||||
|  |       var e = g.getElement(); | ||||||
|  |       if (e != null && e.getSelectedInstance() != null) { | ||||||
|  |         arr = [...arr, ...e.getSelectedInstance()!.env]; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return arr; | ||||||
|  |   } | ||||||
|  |   List<Widget> getInstanceInOutput(AbstractItem? item, SubMapFormsType type) { | ||||||
|  |     List<Widget> children = []; | ||||||
|  |     if (item == null || item.getSelectedInstance() == null) { | ||||||
|  |       return []; | ||||||
|  |     }  | ||||||
|  |     AbstractInstance instance = item.getSelectedInstance()!; | ||||||
|  |     List<Param> params = widget.type == SubMapFormsType.INPUT ? instance.inputs : ( widget.type == SubMapFormsType.OUTPUT ? instance.outputs : instance.env); | ||||||
|  |     for ( var param in params) { | ||||||
|  |       if (!param.readOnly) { continue; } | ||||||
|  |       dynamic env; | ||||||
|  |       if (param.value == null && param.attr != null) {  | ||||||
|  |         var s = item.serialize(); | ||||||
|  |         s.addAll(item.getSelectedInstance()?.serialize() ?? {}); | ||||||
|  |         env = s[param.attr!]; | ||||||
|  |       } else {  env = param.value; } | ||||||
|  |       _save = true; | ||||||
|  |       if (env is bool) { | ||||||
|  |         var ctrl = ValueNotifier<bool>(env); | ||||||
|  |         children.add(AdvancedSwitch( | ||||||
|  |               width: 150.0, | ||||||
|  |               height: 25.0, | ||||||
|  |               controller: ctrl, | ||||||
|  |               enabled: false, | ||||||
|  |               activeChild: Text(param.name!, style: const TextStyle(color: Colors.white)), | ||||||
|  |               inactiveChild: Text("no ${param.name!}", style: const TextStyle(color: Colors.white)), | ||||||
|  |               activeColor: Colors.green, | ||||||
|  |               inactiveColor: redColor, | ||||||
|  |               onChanged: (value) {}, | ||||||
|  |         )); | ||||||
|  |       } else { | ||||||
|  |         children.add(SubTextInputWidget( subkey: param.name!, width: 180, empty: false, change: (value) { },  | ||||||
|  |                      initialValue: env, readOnly: true, noLabel: false, copyLabel: true )); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return children; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     List<Widget> children = []; |     List<Widget> children = []; | ||||||
|  |     var newwidget = getInstanceInOutput(widget.item, widget.type); | ||||||
|     bool save = false; |     children.add(Column( children : [ | ||||||
|     for (var graphItem in widget.graphItems) { |         Padding( padding: const EdgeInsets.only(top: 10), | ||||||
|       int count = 0; |           child: Text("<${widget.type == SubMapFormsType.INPUT ? "INPUT" : (widget.type == SubMapFormsType.OUTPUT ? "OUTPUT" : "")} ENV VARIABLES>",  | ||||||
|       var el = graphItem.getElement(); |             style: const TextStyle(fontSize: 13, fontWeight: FontWeight.bold), textAlign: TextAlign.center)), | ||||||
|       if (el == null || el.model == null) { continue; } |         ...newwidget, | ||||||
|       for ( var r in el.model!.refs.keys) { |         SubAddFormsWidget(dash: widget.dash, item: widget.item, elementID: widget.elementID, type: widget.type), | ||||||
|         var env = widget.item.getVariable(["container", "env"], widget.item.serialize()); |         Container(width: 200, padding: const EdgeInsets.only(top: 10, bottom: 10), | ||||||
|         if (env == null || env is Map<String, dynamic>) { continue; } |           decoration: const BoxDecoration(border: Border(bottom: BorderSide(color: Colors.grey, width: 1)))), | ||||||
|         var n = "${el.topic.toUpperCase()}_${el.getName().toUpperCase().replaceAll(" ", "_")}_${r.toUpperCase()}_$count"; |     ])); | ||||||
|         if (env[n] == null) { |     if (_save) { | ||||||
|           save = true; |       widget.dash.saveDash(widget.dash.id, context); | ||||||
|         } |  | ||||||
|         env[n]= "{{ ${ widget.isInput ? "in" : "out" }_${graphItem.id}_$r }}"; |  | ||||||
|         widget.item.setVariable(["container", "env"], env, el.serialize()); |  | ||||||
|         children.add( Padding(padding: const EdgeInsets.only(bottom: 10),  |  | ||||||
|           child: SubTextInputWidget(subkey: n, width: 180, empty: false, change: (value) {},  |  | ||||||
|           initialValue: n, readOnly: true, noLabel: true))); |  | ||||||
|         count++; |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|     if (save) { |     return Column( children : children );  | ||||||
|       widget.dash.saveDash(widget.dash.id); |  | ||||||
|     } |  | ||||||
|     if (children.isEmpty) { |  | ||||||
|       return Container(); |  | ||||||
|     } |  | ||||||
|     return Column( children : [ |  | ||||||
|       Container(width: 200, padding: const EdgeInsets.only(top: 10, bottom: 10), |  | ||||||
|         decoration: const BoxDecoration(border: Border(bottom: BorderSide(color: Colors.grey, width: 1)))), |  | ||||||
|       Padding( padding: const EdgeInsets.only(top: 10), |  | ||||||
|         child: Text("<${widget.isInput ? "INPUT ENV VARIABLE" : "OUTPUT ENV VARIABLE"}>", style: const TextStyle(fontSize: 13, fontWeight: FontWeight.bold),  |  | ||||||
|                     textAlign: TextAlign.center)), |  | ||||||
|       Column( children: children), |  | ||||||
|     ]);  |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -1,115 +0,0 @@ | |||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/sub_text_input.dart'; |  | ||||||
|  |  | ||||||
| class MapForm { |  | ||||||
|   String key = ""; |  | ||||||
|   String value = ""; |  | ||||||
|   MapForm({ required this.key, required this.value }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| class SubMapFormsWidget extends StatefulWidget { |  | ||||||
|   bool readOnly; |  | ||||||
|   String categoryKey = ""; |  | ||||||
|   String varKey = ""; |  | ||||||
|   String elementID = ""; |  | ||||||
|   FlowData item; |  | ||||||
|   Dashboard dash;  |  | ||||||
|   bool empty = false; |  | ||||||
|   List<MapForm> forms = []; |  | ||||||
|   SubMapFormsWidget ({ super.key, required this.dash, this.readOnly = false, |  | ||||||
|     this.empty = false, required this.item, required this.elementID, |  | ||||||
|     required this.categoryKey, required this.varKey }); |  | ||||||
|   @override SubMapFormsWidgetState createState() => SubMapFormsWidgetState(); |  | ||||||
| } |  | ||||||
| class SubMapFormsWidgetState extends State<SubMapFormsWidget> { |  | ||||||
|  |  | ||||||
|   Map<String, dynamic> toMap() { |  | ||||||
|     Map<String, dynamic> m = {}; |  | ||||||
|     for (var form in widget.forms) { |  | ||||||
|       m[form.key] = form.value; |  | ||||||
|     } |  | ||||||
|     return m; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     Map<String, dynamic>? m = widget.item.getVariable([widget.categoryKey, widget.varKey], widget.item.serialize()); |  | ||||||
|     var l = [widget.categoryKey, widget.varKey]; |  | ||||||
|     List<Widget> children = []; |  | ||||||
|     List<String> empty = []; |  | ||||||
|     widget.forms = []; |  | ||||||
|     var i = 0; |  | ||||||
|     for (var key in (m?.keys.toList() ?? empty)) { |  | ||||||
|       if (((m![key] as String?)?.contains('{{') ?? false)) { |  | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
|       widget.forms.add(MapForm(key: key, value: m[key])); |  | ||||||
|       children.add(Row( children: [ |  | ||||||
|         SubTextInputWidget(subkey: "key", readOnly: widget.readOnly, |  | ||||||
|           initialValue: key, width: 77.5, empty: widget.empty,  |  | ||||||
|           change: (value) {  |  | ||||||
|             setState(() { |  | ||||||
|               widget.forms[i].key = value; |  | ||||||
|               Future.delayed(const Duration(seconds: 2), () { |  | ||||||
|                 widget.dash.saveDash(widget.dash.id); |  | ||||||
|             }); |  | ||||||
|             var el = widget.dash.getElement(widget.elementID); |  | ||||||
|             widget.item = widget.item.deserialize(widget.item.setVariable(l, toMap(), widget.item.serialize())) as dynamic; |  | ||||||
|             el!.element = widget.item as dynamic; |  | ||||||
|           }); |  | ||||||
|         }), |  | ||||||
|         const Padding(padding: EdgeInsets.only(left: 5, right: 5, top: 15), child: Text("=", textAlign: TextAlign.center,)), |  | ||||||
|         SubTextInputWidget(subkey: "value", initialValue: widget.forms[i].value, width: 77.5, empty: widget.empty,  |  | ||||||
|           readOnly: widget.readOnly, change: (value) {  |  | ||||||
|             Future.delayed(const Duration(seconds: 2), () { |  | ||||||
|                 widget.dash.saveDash(widget.dash.id); |  | ||||||
|             }); |  | ||||||
|             widget.forms[i].value = value; |  | ||||||
|             var el = widget.dash.getElement(widget.elementID); |  | ||||||
|             widget.item = widget.item.deserialize( |  | ||||||
|               widget.item.setVariable(l, toMap(), widget.item.serialize())) as dynamic; |  | ||||||
|             el!.element = widget.item as dynamic; |  | ||||||
|         }),      |  | ||||||
|       ])); |  | ||||||
|       i++; |  | ||||||
|     } |  | ||||||
|     return Column( children : [ |  | ||||||
|       Container(width: 200, padding: const EdgeInsets.only(top: 10), |  | ||||||
|         decoration: const BoxDecoration(border: Border(bottom: BorderSide(color: Colors.grey, width: 1))),), |  | ||||||
|       widget.readOnly ? Container() : Row( children: [ |  | ||||||
|               InkWell( onTap: () { |  | ||||||
|                   widget.forms.add(MapForm(key: "", value: "")); |  | ||||||
|                   var el = widget.dash.getElement(widget.elementID); |  | ||||||
|                   widget.item = widget.item.deserialize( |  | ||||||
|                     widget.item.setVariable(l, toMap(), widget.item.serialize())) as dynamic; |  | ||||||
|                   el!.element = widget.item as dynamic; |  | ||||||
|                   setState(() {}); |  | ||||||
|                 }, child: Container( margin: const EdgeInsets.only(top: 10), |  | ||||||
|                     decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.grey, width: 1)), |  | ||||||
|                     width: 125, height: 30, |  | ||||||
|                     child: Row( mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|                       children: [ const Padding( padding: EdgeInsets.only(right: 5), child: Icon(Icons.add)), Text("add ${widget.varKey} vars")]), |  | ||||||
|                   ),   |  | ||||||
|               ), |  | ||||||
|               InkWell( onTap: () { |  | ||||||
|                       if (widget.forms.isEmpty) { return;} |  | ||||||
|                       widget.forms.sublist(0, widget.forms.length - 1); |  | ||||||
|                       widget.item = widget.item.deserialize( |  | ||||||
|                         widget.item.setVariable(l, toMap(), widget.item.serialize())) as dynamic; |  | ||||||
|                       var el = widget.dash.getElement(widget.elementID); |  | ||||||
|                       el!.element = widget.item as dynamic; |  | ||||||
|                       setState(() { widget.dash.saveDash(widget.dash.id); }); |  | ||||||
|                     }, child: |  | ||||||
|                     Container( margin: const EdgeInsets.only(left: 5, top: 10), |  | ||||||
|                       decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.grey, width: 1)), |  | ||||||
|                       width: 45, height: 30, |  | ||||||
|                       child: const Row( mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|                         children: [ Icon( Icons.delete, color: Colors.black ) ]), |  | ||||||
|                     ) |  | ||||||
|                   ),  |  | ||||||
|             ]), |  | ||||||
|       ...children |  | ||||||
|     ]);  |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/widgets/inputs/sub_text_input.dart'; |  | ||||||
|  |  | ||||||
| class WebReferenceFormsWidget extends StatefulWidget { |  | ||||||
|   dynamic item; |  | ||||||
|   WebReferenceFormsWidget ({ super.key, required this.item }); |  | ||||||
|   @override WebReferenceFormsWidgetState createState() => WebReferenceFormsWidgetState(); |  | ||||||
| } |  | ||||||
| class WebReferenceFormsWidgetState extends State<WebReferenceFormsWidget> { |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     return Column( children: [ |  | ||||||
|       SubTextInputWidget(subkey: "path", width: 180, empty: false, change: (value) {  |  | ||||||
|         widget.item.path = value.split(","); |  | ||||||
|       }, initialValue: widget.item.path, readOnly: true,), |  | ||||||
|       SubTextInputWidget(subkey: "protocol", width: 180, empty: false, change: (value) {  |  | ||||||
|         widget.item.protocol = value.split(","); |  | ||||||
|       }, initialValue: widget.item.protocol, readOnly: true,), |  | ||||||
|       SubTextInputWidget(subkey: "type", width: 180, empty: false, change: (value) {  |  | ||||||
|         widget.item.type = value.split(","); |  | ||||||
|       }, initialValue: widget.item.type, readOnly: true,), |  | ||||||
|     ]); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -5,6 +5,7 @@ import 'package:oc_front/pages/shared.dart'; | |||||||
|  |  | ||||||
| class ShallowDropdownInputWidget extends StatefulWidget { | class ShallowDropdownInputWidget extends StatefulWidget { | ||||||
|   double? width; |   double? width; | ||||||
|  |   double? height; | ||||||
|   CollaborativeAreaType type = CollaborativeAreaType.workspace; |   CollaborativeAreaType type = CollaborativeAreaType.workspace; | ||||||
|   Future<List<Shallow>> Function()? all; |   Future<List<Shallow>> Function()? all; | ||||||
|   Future<void> Function(String)? load; |   Future<void> Function(String)? load; | ||||||
| @@ -17,7 +18,7 @@ class ShallowDropdownInputWidget extends StatefulWidget { | |||||||
|   Widget? prefixIcon; |   Widget? prefixIcon; | ||||||
|   IconData? iconLoad; |   IconData? iconLoad; | ||||||
|   IconData? iconRemove; |   IconData? iconRemove; | ||||||
|  |   String? label; | ||||||
|   String? hint; |   String? hint; | ||||||
|  |  | ||||||
|   String? tooltipLoad; |   String? tooltipLoad; | ||||||
| @@ -30,8 +31,8 @@ class ShallowDropdownInputWidget extends StatefulWidget { | |||||||
|   bool deletion = false; |   bool deletion = false; | ||||||
|  |  | ||||||
|   ShallowDropdownInputWidget ({ Key? key, this.width, this.current, required this.all, this.prefixIcon, |   ShallowDropdownInputWidget ({ Key? key, this.width, this.current, required this.all, this.prefixIcon, | ||||||
|   this.iconLoad, this.iconRemove, this.hint, this.filled, this.hintColor, this.color, |   this.iconLoad, this.iconRemove, this.hint, this.filled, this.hintColor, this.color, this.height, | ||||||
|   this.tooltipLoad, this.tooltipRemove, this.deletion = false, this.maptoDropdown, |   this.tooltipLoad, this.tooltipRemove, this.deletion = false, this.maptoDropdown, this.label, | ||||||
|   required this.type, this.canLoad, this.canRemove, this.load, this.remove, this.change }): super(key: key); |   required this.type, this.canLoad, this.canRemove, this.load, this.remove, this.change }): super(key: key); | ||||||
|   @override ShallowDropdownInputWidgetState createState() => ShallowDropdownInputWidgetState(); |   @override ShallowDropdownInputWidgetState createState() => ShallowDropdownInputWidgetState(); | ||||||
| } | } | ||||||
| @@ -54,11 +55,12 @@ class ShallowDropdownInputWidgetState extends State<ShallowDropdownInputWidget> | |||||||
|             data: Theme.of(context).copyWith( |             data: Theme.of(context).copyWith( | ||||||
|               canvasColor: widget.filled ??Colors.white, |               canvasColor: widget.filled ??Colors.white, | ||||||
|             ), |             ), | ||||||
|             child: Container( height: 50, width: (widget.width ?? getMainWidth(context)) - (widget.load == null ? 0 : 50) - (widget.remove == null ? 0 : 50),  |             child: Container( height: widget.height ?? 50,  | ||||||
|  |             width: (widget.width ?? getMainWidth(context)) - (widget.load == null ? 0 : 50) - (widget.remove == null ? 0 : 50),  | ||||||
|             decoration: const BoxDecoration( |             decoration: const BoxDecoration( | ||||||
|               color: Colors.white, |               color: Colors.white, | ||||||
|             ),   |             ),   | ||||||
|             child:  DropdownButtonFormField( |             child: Center(child: DropdownButtonFormField( | ||||||
|                       onChanged: (value) { |                       onChanged: (value) { | ||||||
|                         setState(() { |                         setState(() { | ||||||
|                           widget.current = value; |                           widget.current = value; | ||||||
| @@ -78,6 +80,8 @@ class ShallowDropdownInputWidgetState extends State<ShallowDropdownInputWidget> | |||||||
|                        ), |                        ), | ||||||
|                       decoration: InputDecoration(  |                       decoration: InputDecoration(  | ||||||
|                         filled: true, |                         filled: true, | ||||||
|  |                         label: widget.label == null ? null : Text(widget.label!), | ||||||
|  |                         alignLabelWithHint: false, | ||||||
|                         prefixIconColor: Colors.grey, |                         prefixIconColor: Colors.grey, | ||||||
|                         prefixIcon: widget.prefixIcon, |                         prefixIcon: widget.prefixIcon, | ||||||
|                         suffixIconColor: widget.hintColor ?? Colors.grey , |                         suffixIconColor: widget.hintColor ?? Colors.grey , | ||||||
| @@ -88,13 +92,14 @@ class ShallowDropdownInputWidgetState extends State<ShallowDropdownInputWidget> | |||||||
|                         contentPadding: EdgeInsets.only(left: (widget.width ?? 400) < 200 ? 0 : 30, right: (widget.width ?? 400) < 200 ? 0 : 30, top: 18, bottom: 18), |                         contentPadding: EdgeInsets.only(left: (widget.width ?? 400) < 200 ? 0 : 30, right: (widget.width ?? 400) < 200 ? 0 : 30, top: 18, bottom: 18), | ||||||
|                       ), |                       ), | ||||||
|                       items: items,  |                       items: items,  | ||||||
|                       )))), |                     ))))), | ||||||
|           widget.load == null ? Container() : Tooltip( |           widget.load == null ? Container() : Tooltip( | ||||||
|             message: widget.tooltipLoad ?? "load $t", |             message: widget.tooltipLoad ?? "load $t", | ||||||
|             child:InkWell(  |             child:InkWell(  | ||||||
|             mouseCursor: SystemMouseCursors.click, |             mouseCursor: SystemMouseCursors.click, | ||||||
|             onTap: () async { |             onTap: () async { | ||||||
|               if (widget.canLoad == null || !widget.canLoad!(widget.current) || widget.load == null || widget.current == null) { |               if (widget.canLoad == null || !widget.canLoad!(widget.current)  | ||||||
|  |               || widget.load == null || widget.current == null) { | ||||||
|                 return; |                 return; | ||||||
|               } |               } | ||||||
|               await widget.load!(widget.current!); |               await widget.load!(widget.current!); | ||||||
|   | |||||||
| @@ -54,6 +54,8 @@ class ShallowTextInputWidget extends StatefulWidget { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| class ShallowTextInputWidgetState extends State<ShallowTextInputWidget> { | class ShallowTextInputWidgetState extends State<ShallowTextInputWidget> { | ||||||
|  |   TextEditingController ctrl = TextEditingController(); | ||||||
|  |  | ||||||
|   bool validForms() { |   bool validForms() { | ||||||
|     for (var form in widget.forms) { |     for (var form in widget.forms) { | ||||||
|       if (!form.validate()) { |       if (!form.validate()) { | ||||||
| @@ -65,6 +67,7 @@ class ShallowTextInputWidgetState extends State<ShallowTextInputWidget> { | |||||||
|  |  | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     var t = widget.type == CollaborativeAreaType.workspace ? "workspace" : (widget.type == CollaborativeAreaType.workflow ? "workflow" : "peer"); |     var t = widget.type == CollaborativeAreaType.workspace ? "workspace" : (widget.type == CollaborativeAreaType.workflow ? "workflow" : "peer"); | ||||||
|  |     ctrl.text = widget.current ?? ""; | ||||||
|     return Row( mainAxisAlignment: widget.alignment, children: [ |     return Row( mainAxisAlignment: widget.alignment, children: [ | ||||||
|           Tooltip( message: widget.hint ?? "current $t", child: |           Tooltip( message: widget.hint ?? "current $t", child: | ||||||
|           Theme( |           Theme( | ||||||
| @@ -83,7 +86,7 @@ class ShallowTextInputWidgetState extends State<ShallowTextInputWidget> { | |||||||
|                           } |                           } | ||||||
|                         }); |                         }); | ||||||
|                       }, |                       }, | ||||||
|                       initialValue: widget.current, |                       controller: ctrl, | ||||||
|                       style: TextStyle(color: widget.color ?? Colors.black, fontSize: 15), |                       style: TextStyle(color: widget.color ?? Colors.black, fontSize: 15), | ||||||
|                       decoration: InputDecoration(  |                       decoration: InputDecoration(  | ||||||
|                         filled: true, |                         filled: true, | ||||||
|   | |||||||
| @@ -4,10 +4,11 @@ class SubDropdownInputWidget extends StatefulWidget { | |||||||
|   String subkey; |   String subkey; | ||||||
|   double width; |   double width; | ||||||
|   bool empty; |   bool empty; | ||||||
|  |   String? initialValue; | ||||||
|   List<DropdownMenuItem<String>> dropdownMenuEntries = []; |   List<DropdownMenuItem<String>> dropdownMenuEntries = []; | ||||||
|   void Function(String?)? change = (value) {}; |   void Function(String?)? change = (value) {}; | ||||||
|   SubDropdownInputWidget ({ Key? key, required this.dropdownMenuEntries, |   SubDropdownInputWidget ({ super.key, required this.dropdownMenuEntries, this.initialValue, | ||||||
|   required this.subkey, required this.width, required this.empty, required this.change }): super(key: key); |   required this.subkey, required this.width, required this.empty, required this.change }); | ||||||
|   @override SubDropdownInputWidgetState createState() => SubDropdownInputWidgetState(); |   @override SubDropdownInputWidgetState createState() => SubDropdownInputWidgetState(); | ||||||
| } | } | ||||||
| class SubDropdownInputWidgetState extends State<SubDropdownInputWidget> { | class SubDropdownInputWidgetState extends State<SubDropdownInputWidget> { | ||||||
| @@ -17,9 +18,11 @@ class SubDropdownInputWidgetState extends State<SubDropdownInputWidget> { | |||||||
|               child: Container( margin: EdgeInsets.only(top: widget.empty ? 0 : 15), |               child: Container( margin: EdgeInsets.only(top: widget.empty ? 0 : 15), | ||||||
|                 width: widget.width, height: 30, |                 width: widget.width, height: 30, | ||||||
|                 child: DropdownButtonFormField(  |                 child: DropdownButtonFormField(  | ||||||
|  |                   isExpanded: true, | ||||||
|                   items: widget.dropdownMenuEntries, |                   items: widget.dropdownMenuEntries, | ||||||
|  |                   value: widget.initialValue, | ||||||
|                   onChanged: widget.change, |                   onChanged: widget.change, | ||||||
|                   style: const TextStyle(fontSize: 12), |                   style: const TextStyle(fontSize: 12,color: Colors.black, overflow: TextOverflow.ellipsis), | ||||||
|                   decoration: InputDecoration( |                   decoration: InputDecoration( | ||||||
|                     hintText: "select ${widget.subkey}...", |                     hintText: "select ${widget.subkey}...", | ||||||
|                     fillColor: Colors.white, |                     fillColor: Colors.white, | ||||||
|   | |||||||
| @@ -10,9 +10,10 @@ class SubTextInputWidget extends StatefulWidget { | |||||||
|   bool empty; |   bool empty; | ||||||
|   bool noLabel; |   bool noLabel; | ||||||
|   bool readOnly = false; |   bool readOnly = false; | ||||||
|  |   bool copyLabel = false; | ||||||
|   void Function(String) change = (value) {}; |   void Function(String) change = (value) {}; | ||||||
|   SubTextInputWidget ({ Key? key,  |   SubTextInputWidget ({ Key? key,  | ||||||
|   required this.subkey, this.readOnly = false, this.noLabel = false, |   required this.subkey, this.readOnly = false, this.noLabel = false, this.copyLabel = false, | ||||||
|    this.initialValue, required this.width, required this.empty, required this.change }):  |    this.initialValue, required this.width, required this.empty, required this.change }):  | ||||||
|     super(key: key); |     super(key: key); | ||||||
|   @override SubTextInputWidgetState createState() => SubTextInputWidgetState(); |   @override SubTextInputWidgetState createState() => SubTextInputWidgetState(); | ||||||
| @@ -47,7 +48,7 @@ class SubTextInputWidgetState extends State<SubTextInputWidget> { | |||||||
|                   ), |                   ), | ||||||
|                 ))), |                 ))), | ||||||
|       widget.readOnly ? InkWell( onTap: () { |       widget.readOnly ? InkWell( onTap: () { | ||||||
|                     Clipboard.setData(ClipboardData(text: widget.initialValue!)); |                     Clipboard.setData(ClipboardData(text: widget.copyLabel ? "\$${widget.subkey}" : widget.initialValue!)); | ||||||
|                     showAlertBanner(context, () {}, const InfoAlertBannerChild(text: "successfully add to clipboard"), // <-- Put any widget here you want! |                     showAlertBanner(context, () {}, const InfoAlertBannerChild(text: "successfully add to clipboard"), // <-- Put any widget here you want! | ||||||
|                       alertBannerLocation:  AlertBannerLocation.bottom,); |                       alertBannerLocation:  AlertBannerLocation.bottom,); | ||||||
|                   }, child: Container( margin: EdgeInsets.only(left: 5, top: widget.empty ? 0 : 15), |                   }, child: Container( margin: EdgeInsets.only(left: 5, top: widget.empty ? 0 : 15), | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								lib/widgets/items/infos.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								lib/widgets/items/infos.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  |  | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
|  | class InfosWidget extends StatefulWidget { | ||||||
|  |   AbstractItem item; | ||||||
|  |   InfosWidget ({ super.key, required this.item }); | ||||||
|  |   @override InfosWidgetState createState() => InfosWidgetState(); | ||||||
|  | } | ||||||
|  | class InfosWidgetState extends State<InfosWidget> { | ||||||
|  |   @override Widget build(BuildContext context) { | ||||||
|  |     List<Widget> children = []; | ||||||
|  |     var obj = widget.item.infos(); | ||||||
|  |     for (var key in obj.keys) { | ||||||
|  |       children.add( | ||||||
|  |         Padding(padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 100), | ||||||
|  |           child: Text("$key : ${obj[key] ?? "no $key"}",  | ||||||
|  |           style: const TextStyle(fontSize: 15, fontWeight: FontWeight.w600))), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |     return Wrap( children: children); | ||||||
|  |   }  | ||||||
|  | } | ||||||
| @@ -1,9 +1,12 @@ | |||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:oc_front/widgets/items/items_details/data_item.dart'; | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/models/response.dart'; | ||||||
|  | import 'package:oc_front/pages/shared.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/shallow_dropdown_input.dart'; | ||||||
|  | import 'package:oc_front/widgets/inputs/sub_dropdown_input%20.dart'; | ||||||
|  |  | ||||||
|  | // ignore: must_be_immutable | ||||||
| class ItemWidget extends StatefulWidget { | class ItemWidget extends StatefulWidget { | ||||||
|   AbstractItem item; |   AbstractItem item; | ||||||
|   ItemWidget ({ super.key, required this.item }); |   ItemWidget ({ super.key, required this.item }); | ||||||
| @@ -11,28 +14,88 @@ class ItemWidget extends StatefulWidget { | |||||||
| } | } | ||||||
| class ItemWidgetState extends State<ItemWidget> { | class ItemWidgetState extends State<ItemWidget> { | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|     Widget w = Container(); |     List<Widget> widgets = [ | ||||||
|     /*  if (isData(widget.item.topic)) { w = DataItemWidget(item: widget.item as DataItem); } |       Container( margin: EdgeInsets.only(bottom: 20), | ||||||
|     else if (isComputing(widget.item.topic)) { w = DataItemWidget(item: widget.item as DataItem); } |         decoration: BoxDecoration(border: Border(bottom: BorderSide(color: midColor))), | ||||||
|     else if (isCompute(widget.item.topic)) { w = DataItemWidget(item: widget.item as DataItem); } |         width: getMainWidth(context) / 2, | ||||||
|     else if (isStorage(widget.item.topic)) { w = DataItemWidget(item: widget.item as DataItem); } */ |         child: Center(child: Padding( padding: EdgeInsets.only(bottom: 20),  | ||||||
|  |           child: Text("RESOURCE INFORMATIONS", style: TextStyle(fontSize: 18, color: Colors.grey, fontWeight: FontWeight.w500))))) | ||||||
|     return Container( |     ]; | ||||||
|             height: getHeight(context) - 300, |     var infos = widget.item.infos(); | ||||||
|  |     var count = 0; | ||||||
|  |     for (var info in infos.keys) { | ||||||
|  |       count++; | ||||||
|  |       widgets.add(Padding( padding: EdgeInsets.symmetric(vertical: 5), child : Row(children: [ | ||||||
|  |         Padding(  padding: EdgeInsets.only(left: 50, right: 10), child : Text("${info.toUpperCase().replaceAll("_", " ")} :", style: TextStyle(fontSize: 15, color: Colors.grey))), | ||||||
|  |         Text("${infos[info] is bool ? (infos[info] == true ? "yes" : "no") : infos[info] ?? "unknown"}",  | ||||||
|  |           overflow: TextOverflow.ellipsis, | ||||||
|  |           style: TextStyle(fontSize: 15, color: Colors.black, fontWeight: FontWeight.w500)) | ||||||
|  |       ]))); | ||||||
|  |     } | ||||||
|  |     if (count == 0 ) { | ||||||
|  |         widgets.add(Center(child: Padding( padding: EdgeInsets.symmetric(vertical: 5), child : Row(children: [ | ||||||
|  |           Padding(  padding: EdgeInsets.only(left: 50, right: 10), child : Text("NO INFORMATION", style: TextStyle(fontSize: 15, color: Colors.grey))), | ||||||
|  |         ])))); | ||||||
|  |       } | ||||||
|  |     List<Widget> widgetsInstance = []; | ||||||
|  |     List<Shallow> dpItems = []; | ||||||
|  |     for (var (i, instance) in widget.item.instances.indexed) { | ||||||
|  |       dpItems.add(Shallow(id: "$i", name: instance.name ?? ""));    | ||||||
|  |     } | ||||||
|  |     if (dpItems.isNotEmpty) { | ||||||
|  |       widgetsInstance.add(Center( child: Padding(padding: EdgeInsets.only(bottom: 15), child:  | ||||||
|  |         ShallowDropdownInputWidget( all: () async => dpItems, width: (getWidth(context) / 2) - 25,  | ||||||
|  |         label: "instances", type: CollaborativeAreaType.resource, height: 65, | ||||||
|  |           current: "${widget.item.selectedInstance}", change: (value) { | ||||||
|  |             if (value != null) { setState(() { widget.item.selectedInstance = int.parse(value); }); } | ||||||
|  |           }, | ||||||
|  |         )) | ||||||
|  |       )); | ||||||
|  |     } | ||||||
|  |     if (widget.item.instances.length > widget.item.selectedInstance) { | ||||||
|  |       var instance = widget.item.instances[widget.item.selectedInstance]; | ||||||
|  |       widgetsInstance.add(Container(height: 20, width: getWidth(context) / 2,)); | ||||||
|  |       var count = 0; | ||||||
|  |       for (var info in instance.infos().keys) { | ||||||
|  |         if (instance.infos()[info] == null || instance.infos()[info] is List || instance.infos()[info] is Map) { continue; } | ||||||
|  |         count++; | ||||||
|  |         widgetsInstance.add(Center(child: Padding( padding: EdgeInsets.symmetric(vertical: 5), child : Row(children: [ | ||||||
|  |           Padding(  padding: EdgeInsets.only(left: 50, right: 10), child : Text("${info.toUpperCase().replaceAll("_", " ")} :", style: TextStyle(fontSize: 15, color: Colors.grey))), | ||||||
|  |           Text("${instance.infos()[info] is bool ? (instance.infos()[info] == true ? "yes" : "no") : instance.infos()[info] ?? "unknown"}",  | ||||||
|  |             overflow: TextOverflow.ellipsis, | ||||||
|  |             style: TextStyle(fontSize: 15, color: Colors.black, fontWeight: FontWeight.w500)) | ||||||
|  |         ])))); | ||||||
|  |       } | ||||||
|  |       if (count == 0 ) { | ||||||
|  |         widgetsInstance.add(Padding( padding: EdgeInsets.symmetric(vertical: 5), child : Row(children: [ | ||||||
|  |           Padding(  padding: EdgeInsets.only(left: 50, right: 10), child : Text("NO INSTANCE INFORMATION", style: TextStyle(fontSize: 15, color: Colors.grey))), | ||||||
|  |         ]))); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     Widget w = Column( mainAxisSize: MainAxisSize.max, children: widgets ); | ||||||
|  |     Widget w2 = Column( mainAxisSize: MainAxisSize.max, children: widgetsInstance ); | ||||||
|  |     return SizedBox( | ||||||
|  |             height: getHeight(context) - 200, | ||||||
|             child: SingleChildScrollView( |             child: SingleChildScrollView( | ||||||
|               child: Column( children: [ |               child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ | ||||||
|               widget.item.description == null ? Container() : Container( |               widget.item.description == null ? Container() : Container( | ||||||
|                 width: getMainWidth(context), |                 width: getMainWidth(context), | ||||||
|                 alignment: Alignment.center, |                 alignment: Alignment.center, | ||||||
|                 decoration: BoxDecoration(border: Border(bottom: BorderSide(color: midColor))), |                 height: 100, | ||||||
|                 padding: const EdgeInsets.all(30),  |                 decoration: BoxDecoration( | ||||||
|                 child: Text(widget.item.description!,  |                   color: Colors.white, | ||||||
|  |                   border: Border(bottom: BorderSide(color: midColor))), | ||||||
|  |                 padding: const EdgeInsets.symmetric(vertical: 30, horizontal: 50),  | ||||||
|  |                 child: Text(widget.item.description!.length > 350 ? "${widget.item.description!.substring(0, 347)}..." : widget.item.description!, | ||||||
|                 style: TextStyle(fontSize: 15, color: Colors.grey, fontWeight: FontWeight.w500))),  |                 style: TextStyle(fontSize: 15, color: Colors.grey, fontWeight: FontWeight.w500))),  | ||||||
|               Container(padding: const EdgeInsets.all(30), |               Row( children: [  | ||||||
|                   color: midColor, |                 Container(padding: const EdgeInsets.symmetric(vertical: 20), | ||||||
|                   width: getMainWidth(context) / 2, |                   height: getHeight(context) - 300, | ||||||
|                   child: w |                   decoration: BoxDecoration(border: Border(right: BorderSide(color: midColor))), | ||||||
|             ) |                   alignment: Alignment.topLeft, width: getMainWidth(context) / 2,  child: w), | ||||||
|  |                 Container(height: getHeight(context) - 300, | ||||||
|  |                   alignment: Alignment.topRight, width: getMainWidth(context) / 2, child: w2) | ||||||
|  |               ]) | ||||||
|           ] |           ] | ||||||
|         ) |         ) | ||||||
|       ) |       ) | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| import 'dart:convert'; |  | ||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/search.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:oc_front/core/models/workspace_local.dart'; |  | ||||||
| import 'package:oc_front/core/services/router.dart'; | import 'package:oc_front/core/services/router.dart'; | ||||||
|  | import 'package:oc_front/models/resources/resources.dart'; | ||||||
|  | import 'package:oc_front/core/models/workspace_local.dart'; | ||||||
|  |  | ||||||
| const List<GlobalKey<State>> _empty = []; | const List<GlobalKey<State>> _empty = []; | ||||||
| // ignore: must_be_immutable | // ignore: must_be_immutable | ||||||
| @@ -38,7 +37,7 @@ class ItemRowWidgetState extends State<ItemRowWidget> { | |||||||
|             constraints: BoxConstraints(maxWidth: imageSize, minWidth: imageSize), |             constraints: BoxConstraints(maxWidth: imageSize, minWidth: imageSize), | ||||||
|             child: image ?? Image.network('https://get-picto.com/wp-content/uploads/2024/01/logo-instagram-png.webp',  |             child: image ?? Image.network('https://get-picto.com/wp-content/uploads/2024/01/logo-instagram-png.webp',  | ||||||
|               height: imageSize, width: imageSize)), |               height: imageSize, width: imageSize)), | ||||||
|           Container( |           SizedBox( | ||||||
|             width: widget.low ? widget.contextWidth - 20 :  widget.contextWidth - (imageSize + 20) - endWidth, |             width: widget.low ? widget.contextWidth - 20 :  widget.contextWidth - (imageSize + 20) - endWidth, | ||||||
|             child: Padding(padding: widget.contextWidth != getMainWidth(context) ?  |             child: Padding(padding: widget.contextWidth != getMainWidth(context) ?  | ||||||
|             const EdgeInsets.symmetric(horizontal: 10) : const EdgeInsets.symmetric(horizontal: 20), |             const EdgeInsets.symmetric(horizontal: 10) : const EdgeInsets.symmetric(horizontal: 20), | ||||||
| @@ -48,10 +47,7 @@ class ItemRowWidgetState extends State<ItemRowWidget> { | |||||||
|                   widget.low ? Container() : Container(padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), |                   widget.low ? Container() : Container(padding: const EdgeInsets.symmetric(vertical: 5, horizontal: 10), | ||||||
|                     margin: const EdgeInsets.only(right: 20), |                     margin: const EdgeInsets.only(right: 20), | ||||||
|                     decoration: BoxDecoration( |                     decoration: BoxDecoration( | ||||||
|                       color: isData(widget.item.topic) ? Colors.blue :  |                       color: getColor(widget.item.topic), | ||||||
|                       isComputing(widget.item.topic) ? Colors.green :  |  | ||||||
|                       isCompute(widget.item.topic) ? Colors.orange :  |  | ||||||
|                       isStorage(widget.item.topic) ? redColor : Colors.grey, |  | ||||||
|                       borderRadius: BorderRadius.circular(4), |                       borderRadius: BorderRadius.circular(4), | ||||||
|                     ), |                     ), | ||||||
|                     child: Text( getMainWidth(context) < 600 ? "" : widget.item.topic.toString(),  |                     child: Text( getMainWidth(context) < 600 ? "" : widget.item.topic.toString(),  | ||||||
| @@ -61,13 +57,13 @@ class ItemRowWidgetState extends State<ItemRowWidget> { | |||||||
|                     style: TextStyle(fontSize: widget.low ? 14 : 20, overflow: TextOverflow.ellipsis, fontWeight: FontWeight.w600, color: Color(0xFFF67C0B9))), |                     style: TextStyle(fontSize: widget.low ? 14 : 20, overflow: TextOverflow.ellipsis, fontWeight: FontWeight.w600, color: Color(0xFFF67C0B9))), | ||||||
|                   ) |                   ) | ||||||
|                 ]), |                 ]), | ||||||
|                 Text( "From ${widget.item.owner ?? "unknown owner"}",  |                 widget.item.owners.isEmpty ? Container() : Text( "From ${widget.item.owners.map( (e) => e.name).join(", ")}",  | ||||||
|                   style: const TextStyle(fontSize: 14, color: Colors.grey, overflow: TextOverflow.ellipsis)), |                   style: const TextStyle(fontSize: 14, color: Colors.grey, overflow: TextOverflow.ellipsis)), | ||||||
|                 Text(widget.item.shortDescription ?? "", style: const TextStyle(fontSize: 12, overflow: TextOverflow.ellipsis)), |                 Text(widget.item.shortDescription ?? "", style: const TextStyle(fontSize: 12, overflow: TextOverflow.ellipsis)), | ||||||
|               ],) |               ],) | ||||||
|             ) |             ) | ||||||
|           ), |           ), | ||||||
|           widget.low ? Container() : Container( |           widget.low ? Container() : SizedBox( | ||||||
|             width: endWidth, |             width: endWidth, | ||||||
|             child: Row(  |             child: Row(  | ||||||
|               mainAxisAlignment: MainAxisAlignment.center, |               mainAxisAlignment: MainAxisAlignment.center, | ||||||
| @@ -91,7 +87,8 @@ class ItemRowWidgetState extends State<ItemRowWidget> { | |||||||
|                     borderRadius: BorderRadius.circular(4), |                     borderRadius: BorderRadius.circular(4), | ||||||
|                   ), |                   ), | ||||||
|                   child: Icon(WorkspaceLocal.hasItem(widget.item) ? Icons.remove_shopping_cart : Icons.add_shopping_cart,  |                   child: Icon(WorkspaceLocal.hasItem(widget.item) ? Icons.remove_shopping_cart : Icons.add_shopping_cart,  | ||||||
|                     color: Colors.white, size: 20 ))  |                     color: Colors.white, size: 20 ) | ||||||
|  |                 )  | ||||||
|             ), |             ), | ||||||
|             ...(ratio > 1 ? [Padding( padding: const EdgeInsets.only(left: 20),  |             ...(ratio > 1 ? [Padding( padding: const EdgeInsets.only(left: 20),  | ||||||
|               child: InkWell(  |               child: InkWell(  | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:oc_front/models/search.dart'; |  | ||||||
|  |  | ||||||
| class DataItemWidget extends StatefulWidget { |  | ||||||
|   DataItem item; |  | ||||||
|   DataItemWidget ({ super.key, required this.item }); |  | ||||||
|   @override DataItemWidgetState createState() => DataItemWidgetState(); |  | ||||||
| } |  | ||||||
| class DataItemWidgetState extends State<DataItemWidget> { |  | ||||||
|   @override Widget build(BuildContext context) { |  | ||||||
|     return Wrap( children: [ |  | ||||||
|       Padding(padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 100), |  | ||||||
|         child: Text("type : ${widget.item.type ?? "unknown type"}",  |  | ||||||
|           style: const TextStyle(fontSize: 15, fontWeight: FontWeight.w600))), |  | ||||||
|       Padding(padding: const EdgeInsets.symmetric(horizontal: 100, vertical: 20),  |  | ||||||
|         child: Text("protocol : ${widget.item.protocols.isEmpty ? "no protocol founded" : widget.item.protocols.join(",")}",  |  | ||||||
|           style: const TextStyle(fontSize: 15, fontWeight: FontWeight.w600))), |  | ||||||
|       Padding(padding: const EdgeInsets.symmetric(horizontal: 100, vertical: 20),  |  | ||||||
|         child: Text("ex : ${widget.item.exemple ?? "no example"}",  |  | ||||||
|           style: const TextStyle(fontSize: 15, fontWeight: FontWeight.w600))), |  | ||||||
|     ]); |  | ||||||
|   }  |  | ||||||
| } |  | ||||||
| @@ -1,24 +1,35 @@ | |||||||
| import 'package:alert_banner/exports.dart'; | import 'package:alert_banner/exports.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
| import 'package:json_string/json_string.dart'; |  | ||||||
| import 'package:oc_front/core/services/specialized_services/logs_service.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/logs.dart'; | import 'package:oc_front/models/logs.dart'; | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
| import 'package:oc_front/widgets/dialog/alert.dart'; | import 'package:oc_front/widgets/dialog/alert.dart'; | ||||||
|  | import 'package:json_string/json_string.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/logs_service.dart'; | ||||||
|  |  | ||||||
|  | bool isLoading = true; | ||||||
|  | Map<String, bool> valid = {}; | ||||||
| class LogsWidget extends StatefulWidget { | class LogsWidget extends StatefulWidget { | ||||||
|   String? level; |   String? level; | ||||||
|   String search = ""; |   String search = ""; | ||||||
|  |  | ||||||
|   WorkflowExecution? exec; |   WorkflowExecution? exec; | ||||||
|  |   List<Log> logs = []; | ||||||
|   LogsWidget ({ Key? key, this.search = "", this.level, this.exec }): super(key: key); |   LogsWidget ({ Key? key, this.search = "", this.level, this.exec }): super(key: key); | ||||||
|   @override LogsWidgetState createState() => LogsWidgetState(); |   @override LogsWidgetState createState() => LogsWidgetState(); | ||||||
| } | } | ||||||
|  |  | ||||||
| class LogsWidgetState extends State<LogsWidget> { | class LogsWidgetState extends State<LogsWidget> { | ||||||
|   @override Widget build(BuildContext context) { |   @override Widget build(BuildContext context) { | ||||||
|  |     if (!isLoading) { | ||||||
|  |       isLoading = true;  | ||||||
|  |       Future.delayed(const Duration(milliseconds: 100), () { | ||||||
|  |         try { setState(() { }); | ||||||
|  |         } catch (e) { /**/ } | ||||||
|  |       }); | ||||||
|  |       return Container( height: getMainHeight(context) - 100, | ||||||
|  |         child: Center( child: CircularProgressIndicator()) ); | ||||||
|  |     } | ||||||
|     if (widget.exec == null) { |     if (widget.exec == null) { | ||||||
|       return Container(); |       return Container(); | ||||||
|     } else { |     } else { | ||||||
| @@ -26,7 +37,7 @@ class LogsWidgetState extends State<LogsWidget> { | |||||||
|       String end = ""; |       String end = ""; | ||||||
|       try { |       try { | ||||||
|         if (widget.exec!.endDate != null && widget.exec!.endDate != "") { |         if (widget.exec!.endDate != null && widget.exec!.endDate != "") { | ||||||
|           var startD = DateTime.parse(widget.exec!.executionData!); |           var startD = DateTime.parse(widget.exec!.startDate!); | ||||||
|           var endD = DateTime.parse(widget.exec!.endDate!); |           var endD = DateTime.parse(widget.exec!.endDate!); | ||||||
|           var diff = endD.difference(startD); |           var diff = endD.difference(startD); | ||||||
|           if (diff.inDays < 30) { |           if (diff.inDays < 30) { | ||||||
| @@ -38,20 +49,22 @@ class LogsWidgetState extends State<LogsWidget> { | |||||||
|             end = (startD.add( const Duration(days: 29)).microsecondsSinceEpoch).toString(); |             end = (startD.add( const Duration(days: 29)).microsecondsSinceEpoch).toString(); | ||||||
|           } |           } | ||||||
|         } else { |         } else { | ||||||
|           start = (DateTime.parse(widget.exec!.executionData!).subtract( const Duration(days: 14)).microsecondsSinceEpoch).toString(); |           start = (DateTime.parse(widget.exec!.startDate!).subtract( const Duration(days: 14)).microsecondsSinceEpoch).toString(); | ||||||
|           end = (DateTime.parse(widget.exec!.executionData!).add( const Duration(days: 14)).microsecondsSinceEpoch).toString(); |           end = (DateTime.parse(widget.exec!.startDate!).add( const Duration(days: 14)).microsecondsSinceEpoch).toString(); | ||||||
|         } |         } | ||||||
|       } catch(e) { /* */ } |       } catch(e) { /* */ } | ||||||
|  |       Future.delayed(const Duration(minutes: 1), () { | ||||||
|  |         try { setState(() {}); | ||||||
|  |         } catch (e) { /**/ } | ||||||
|  |       }); | ||||||
|       return FutureBuilder(future: LogsService().search(context, [], { |       return FutureBuilder(future: LogsService().search(context, [], { | ||||||
|               "workflow_execution_id":  widget.exec!.id, |               "workflow_execution_id":  widget.exec!.id, | ||||||
|               "start": start, |               "start": start, | ||||||
|               "end":  end |               "end":  end | ||||||
|             }), builder: (a, b) { |             }), builder: (a, b) { | ||||||
|               Future.delayed(const Duration(minutes: 1), () { |  | ||||||
|                 setState(() {}); |  | ||||||
|               }); |  | ||||||
|               List<Log> logs = []; |               List<Log> logs = []; | ||||||
|               if (b.data != null && b.data!.data != null) { |               if (b.data != null && b.data!.data != null) { | ||||||
|  |                 isLoading = false; | ||||||
|                 var d = b.data!.data!; |                 var d = b.data!.data!; | ||||||
|                 for( var r in d.result) { |                 for( var r in d.result) { | ||||||
|                   for (var element in r.logs) { |                   for (var element in r.logs) { | ||||||
| @@ -63,6 +76,10 @@ class LogsWidgetState extends State<LogsWidget> { | |||||||
|               }   |               }   | ||||||
|               List<LogWidget> itemRows = logs.where((element) => (element.message?.toLowerCase().contains(widget.search.toLowerCase()) ?? true) |               List<LogWidget> itemRows = logs.where((element) => (element.message?.toLowerCase().contains(widget.search.toLowerCase()) ?? true) | ||||||
|                 && (widget.level?.contains(element.level ?? "") ?? true) ).map((e) => LogWidget(item: e)).toList(); |                 && (widget.level?.contains(element.level ?? "") ?? true) ).map((e) => LogWidget(item: e)).toList(); | ||||||
|  |               if (isLoading) { | ||||||
|  |                 return Container( height: getMainHeight(context) - 100, | ||||||
|  |                     child: Center( child: CircularProgressIndicator()) ); | ||||||
|  |               } | ||||||
|               return Stack( children: [  |               return Stack( children: [  | ||||||
|                   SingleChildScrollView( child: itemRows.isEmpty ?  |                   SingleChildScrollView( child: itemRows.isEmpty ?  | ||||||
|                   Container( height: getMainHeight(context) - 100, |                   Container( height: getMainHeight(context) - 100, | ||||||
|   | |||||||
| @@ -1,14 +1,13 @@ | |||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:flutter_box_transform/flutter_box_transform.dart'; |  | ||||||
| import 'package:oc_front/core/services/specialized_services/abstract_service.dart'; |  | ||||||
| import 'package:oc_front/core/services/specialized_services/booking_service.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:oc_front/widgets/logs.dart'; | import 'package:oc_front/widgets/logs.dart'; | ||||||
| import 'package:flutter_spinkit/flutter_spinkit.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
|  | import 'package:flutter_box_transform/flutter_box_transform.dart'; | ||||||
| import 'package:oc_front/widgets/lib/tranformablebox.dart' as fork; | import 'package:oc_front/widgets/lib/tranformablebox.dart' as fork; | ||||||
| import 'package:oc_front/widgets/sheduler_items/scheduler_calendar.dart'; |  | ||||||
| import 'package:oc_front/widgets/sheduler_items/scheduler_item.dart'; | import 'package:oc_front/widgets/sheduler_items/scheduler_item.dart'; | ||||||
|  | import 'package:oc_front/widgets/sheduler_items/scheduler_calendar.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/abstract_service.dart'; | ||||||
|  | import 'package:oc_front/core/services/specialized_services/workflow_execution_service.dart'; | ||||||
|  |  | ||||||
| double menuSize = 300; | double menuSize = 300; | ||||||
| // ignore: must_be_immutable | // ignore: must_be_immutable | ||||||
| @@ -19,8 +18,9 @@ class ScheduleWidget extends StatefulWidget { | |||||||
|   bool loading = true; |   bool loading = true; | ||||||
|   bool isList = true; |   bool isList = true; | ||||||
|   bool isBox = true; |   bool isBox = true; | ||||||
|   AbstractService<WorkflowExecutions> service; |   String? selectedID; | ||||||
|   ScheduleWidget ({ super.key, required this.start, required this.end, required this.service, |   AbstractService<WorkflowExecutions> service = WorkflowExecutionService(); | ||||||
|  |   ScheduleWidget ({ super.key, required this.start, required this.end, | ||||||
|     this.isBox =true, this.isList = true, this.loading = false}); |     this.isBox =true, this.isList = true, this.loading = false}); | ||||||
|   @override ScheduleWidgetState createState() => ScheduleWidgetState(); |   @override ScheduleWidgetState createState() => ScheduleWidgetState(); | ||||||
| } | } | ||||||
| @@ -43,19 +43,16 @@ class ScheduleWidgetState extends State<ScheduleWidget> { | |||||||
|         "${widget.start.year}-${widget.start.month > 9 ? widget.start.month : "0${widget.start.month}"}-${widget.start.day > 9 ? widget.start.day : "0${widget.start.day}"}",  |         "${widget.start.year}-${widget.start.month > 9 ? widget.start.month : "0${widget.start.month}"}-${widget.start.day > 9 ? widget.start.day : "0${widget.start.day}"}",  | ||||||
|         "${widget.end.year}-${widget.end.month > 9 ? widget.end.month : "0${widget.end.month}"}-${widget.end.day > 9 ? widget.end.day : "0${widget.end.day}"}"], {}), |         "${widget.end.year}-${widget.end.month > 9 ? widget.end.month : "0${widget.end.month}"}-${widget.end.day > 9 ? widget.end.day : "0${widget.end.day}"}"], {}), | ||||||
|       builder: (ctx, as) {  |       builder: (ctx, as) {  | ||||||
|         Future.delayed(const Duration(minutes: 1), () { |  | ||||||
|           setState(() {}); |  | ||||||
|         }); |  | ||||||
|         Map<String, List<WorkflowExecution>> data = {}; |         Map<String, List<WorkflowExecution>> data = {}; | ||||||
|         if (as.hasData && as.data!.data != null) { |         if (as.hasData && as.data!.data != null) { | ||||||
|           for (var element in as.data!.data!.executions) { |           for (var element in as.data!.data!.executions) { | ||||||
|             if (element.executionData == null) { continue; } |             if (element.startDate == null) { continue; } | ||||||
|             DateTime dateTime = DateTime.parse(element.executionData!); |             DateTime dateTime = DateTime.parse(element.startDate!); | ||||||
|             DateTime date = DateTime(dateTime.year, dateTime.month, dateTime.day); |             DateTime date = DateTime(dateTime.year, dateTime.month, dateTime.day); | ||||||
|             var str = "${date.toIso8601String()}Z"; |             var str = "${date.toIso8601String()}Z"; | ||||||
|             if (data[str] == null) { data[str] = []; } |             if (data[str] == null) { data[str] = []; } | ||||||
|             data[str]!.add(element); |             data[str]!.add(element); | ||||||
|             data[str]!.sort((a, b) => DateTime.parse(a.executionData!).compareTo(DateTime.parse(b.executionData!))); |             data[str]!.sort((a, b) => DateTime.parse(a.startDate!).compareTo(DateTime.parse(b.startDate!))); | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|         bool isInfo = getMainWidth(context) <= 600 && selected != null && widget.isBox; |         bool isInfo = getMainWidth(context) <= 600 && selected != null && widget.isBox; | ||||||
| @@ -63,12 +60,14 @@ class ScheduleWidgetState extends State<ScheduleWidget> { | |||||||
|         List<Widget> children = []; |         List<Widget> children = []; | ||||||
|         if (selected != null) { |         if (selected != null) { | ||||||
|           for (var wf in data[selected!] ?? (<WorkflowExecution>[])) { |           for (var wf in data[selected!] ?? (<WorkflowExecution>[])) { | ||||||
|             DateTime d2 = DateTime.parse(wf.executionData!).toLocal(); |             DateTime d2 = DateTime.parse(wf.startDate!).toLocal(); | ||||||
|             children.add( InkWell( |             children.add( InkWell( | ||||||
|               onTap: () => setState(() { selectedReal = wf.executionData; }), |               onTap: () => setState(() {  | ||||||
|  |                 selectedReal = wf.startDate;  | ||||||
|  |               }), | ||||||
|               child: Container( margin: const EdgeInsets.all(10), |               child: Container( margin: const EdgeInsets.all(10), | ||||||
|               decoration: BoxDecoration(  |               decoration: BoxDecoration(  | ||||||
|                 border: Border.all(color: selectedReal != null && selectedReal == wf.executionData ? lightColor : Colors.transparent, width: 2), |                 border: Border.all(color: selectedReal != null && selectedReal == wf.startDate ? lightColor : Colors.transparent, width: 2), | ||||||
|                 borderRadius: BorderRadius.circular(4), color: Colors.white |                 borderRadius: BorderRadius.circular(4), color: Colors.white | ||||||
|               ), |               ), | ||||||
|               child: Container( |               child: Container( | ||||||
| @@ -96,14 +95,13 @@ class ScheduleWidgetState extends State<ScheduleWidget> { | |||||||
|                   )) |                   )) | ||||||
|             )); |             )); | ||||||
|           } |           } | ||||||
|            |  | ||||||
|         } |         } | ||||||
|         String? selectedID; |          | ||||||
|         WorkflowExecution? sel; |         WorkflowExecution? sel; | ||||||
|         if (selectedReal != null) { |         if (selectedReal != null) { | ||||||
|           try { |           try { | ||||||
|             sel = data[selected!]!.firstWhere((element) => element.executionData == selectedReal); |             sel = data[selected!]!.firstWhere((element) => element.startDate == selectedReal); | ||||||
|             selectedID = sel.id; |             widget.selectedID = sel.id; | ||||||
|           } catch(e) { /* */ } |           } catch(e) { /* */ } | ||||||
|         } |         } | ||||||
|         menuSize = isInfo ? getMainWidth(context) : (menuSize > getMainWidth(context) / 2 ? getMainWidth(context) / 2 : menuSize); |         menuSize = isInfo ? getMainWidth(context) : (menuSize > getMainWidth(context) / 2 ? getMainWidth(context) / 2 : menuSize); | ||||||
| @@ -133,7 +131,9 @@ class ScheduleWidgetState extends State<ScheduleWidget> { | |||||||
|             child: Column( |             child: Column( | ||||||
|               children: [ |               children: [ | ||||||
|                 Row( children: [ |                 Row( children: [ | ||||||
|                   InkWell( onTap: () => setState(() { widget.isDayPlanner = true; }), |                   InkWell( onTap: () => setState(() {  | ||||||
|  |                     widget.isDayPlanner = true;  | ||||||
|  |                   }), | ||||||
|                   child: Tooltip( message: "day planning", child: |                   child: Tooltip( message: "day planning", child: | ||||||
|                     Container( height: 50, width: (isInfo ? getMainWidth(context) : (selected != null ? menuSize : 0)) / (selectedReal != null ? 2 : 1 ), |                     Container( height: 50, width: (isInfo ? getMainWidth(context) : (selected != null ? menuSize : 0)) / (selectedReal != null ? 2 : 1 ), | ||||||
|                       alignment: Alignment.center, |                       alignment: Alignment.center, | ||||||
| @@ -143,7 +143,10 @@ class ScheduleWidgetState extends State<ScheduleWidget> { | |||||||
|                       child: Icon(Icons.calendar_today_outlined, color: widget.isDayPlanner ? Colors.white : Colors.grey), |                       child: Icon(Icons.calendar_today_outlined, color: widget.isDayPlanner ? Colors.white : Colors.grey), | ||||||
|                     ) |                     ) | ||||||
|                   )), |                   )), | ||||||
|                   InkWell( onTap: () => setState(() { widget.isDayPlanner = false; }), |                   InkWell( onTap: () => setState(() {  | ||||||
|  |                     widget.isDayPlanner = false;   | ||||||
|  |  | ||||||
|  |                   }), | ||||||
|                   child: Tooltip( message: "monitor task", child: |                   child: Tooltip( message: "monitor task", child: | ||||||
|                     Container( height: 50, width: selectedReal == null ? 0 : ( |                     Container( height: 50, width: selectedReal == null ? 0 : ( | ||||||
|                       (isInfo ? getMainWidth(context) : (selected != null ? menuSize : 0)) / 2), |                       (isInfo ? getMainWidth(context) : (selected != null ? menuSize : 0)) / 2), | ||||||
| @@ -162,8 +165,8 @@ class ScheduleWidgetState extends State<ScheduleWidget> { | |||||||
|                     SingleChildScrollView( child: Column(  |                     SingleChildScrollView( child: Column(  | ||||||
|                       mainAxisAlignment: children.isEmpty || widget.loading ? MainAxisAlignment.center : MainAxisAlignment.start, |                       mainAxisAlignment: children.isEmpty || widget.loading ? MainAxisAlignment.center : MainAxisAlignment.start, | ||||||
|                       children: [  |                       children: [  | ||||||
|                       ...( widget.isDayPlanner ? children : ( selectedID != null ? [ |                       ...( widget.isDayPlanner ? children : ( widget.selectedID != null ? [ | ||||||
|                         widget.loading ?  const SpinKitCircle(color: Colors.white,) : LogsWidget(exec: sel, search: search, level: level) |                         LogsWidget(exec: sel, search: search, level: level) | ||||||
|                       ] : [])), |                       ] : [])), | ||||||
|                       children.isEmpty ? Container( height: 100, alignment: Alignment.center, child: const Text("No event found", style: TextStyle(color: Colors.grey, fontSize: 20))) : Container() |                       children.isEmpty ? Container( height: 100, alignment: Alignment.center, child: const Text("No event found", style: TextStyle(color: Colors.grey, fontSize: 20))) : Container() | ||||||
|                     ]),  |                     ]),  | ||||||
|   | |||||||
| @@ -70,7 +70,7 @@ class SchedulerCalendarWidgetState extends State<SchedulerCalendarWidget> { | |||||||
|                 } |                 } | ||||||
|                 children.add(InkWell( onTap: () => widget.parent!.setState(() { |                 children.add(InkWell( onTap: () => widget.parent!.setState(() { | ||||||
|                   selected = day.toIso8601String(); |                   selected = day.toIso8601String(); | ||||||
|                   selectedReal = ev.executionData; |                   selectedReal = ev.startDate; | ||||||
|                   if (selectedReal  == null) { |                   if (selectedReal  == null) { | ||||||
|                     widget.parent!.widget.isDayPlanner = true; |                     widget.parent!.widget.isDayPlanner = true; | ||||||
|                   } |                   } | ||||||
| @@ -158,9 +158,9 @@ class SchedulerCalendarWidgetState extends State<SchedulerCalendarWidget> { | |||||||
|           shouldFillViewport: true, |           shouldFillViewport: true, | ||||||
|           eventLoader: (day) { |           eventLoader: (day) { | ||||||
|             return widget.data[day.toIso8601String()] != null ? widget.data[day.toIso8601String()]!.map((e) { |             return widget.data[day.toIso8601String()] != null ? widget.data[day.toIso8601String()]!.map((e) { | ||||||
|               DateTime dateTime = DateTime.parse(e.executionData!); |               DateTime dateTime = DateTime.parse(e.startDate!); | ||||||
|               return Event("[${dateTime.hour > 9 ? dateTime.hour : "0${dateTime.hour}"}:${dateTime.minute > 9 ? dateTime.minute : "0${dateTime.minute}"}:${dateTime.second > 9 ? dateTime.second : "0${dateTime.second}"}] ${e.name}",  |               return Event("[${dateTime.hour > 9 ? dateTime.hour : "0${dateTime.hour}"}:${dateTime.minute > 9 ? dateTime.minute : "0${dateTime.minute}"}:${dateTime.second > 9 ? dateTime.second : "0${dateTime.second}"}] ${e.name}",  | ||||||
|                  colors[(e.status ?? 1) - 1], e.executionData ); |                  colors[(e.status ?? 1) - 1], e.startDate ); | ||||||
|             }).toList() : []; |             }).toList() : []; | ||||||
|           }, |           }, | ||||||
|         )); |         )); | ||||||
| @@ -169,10 +169,10 @@ class SchedulerCalendarWidgetState extends State<SchedulerCalendarWidget> { | |||||||
|  |  | ||||||
| class Event { | class Event { | ||||||
|   final String title; |   final String title; | ||||||
|   String? executionData; |   String? startDate; | ||||||
|   Color color; |   Color color; | ||||||
|  |  | ||||||
|   Event(this.title, this.color, this.executionData); |   Event(this.title, this.color, this.startDate); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String toString() => title; |   String toString() => title; | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_box_transform/flutter_box_transform.dart'; |  | ||||||
| import 'package:oc_front/core/sections/header/header.dart'; |  | ||||||
| import 'package:oc_front/main.dart'; | import 'package:oc_front/main.dart'; | ||||||
| import 'package:oc_front/models/workflow.dart'; | import 'package:oc_front/models/workflow.dart'; | ||||||
| import 'package:oc_front/widgets/sheduler_items/schedule.dart'; | import 'package:oc_front/widgets/sheduler_items/schedule.dart'; | ||||||
| @@ -26,23 +24,25 @@ class SchedulerItemWidgetState extends State<SchedulerItemWidget> { | |||||||
|           for (var element in widget.data.keys.toList()..sort((a, b) => DateTime.parse(a).compareTo(DateTime.parse(b)))) { |           for (var element in widget.data.keys.toList()..sort((a, b) => DateTime.parse(a).compareTo(DateTime.parse(b)))) { | ||||||
|             List<Widget> widgets = []; |             List<Widget> widgets = []; | ||||||
|             for (var ev in widget.data[element] ?? ([] as List<WorkflowExecution>)) { |             for (var ev in widget.data[element] ?? ([] as List<WorkflowExecution>)) { | ||||||
|               widget.keys[ev.executionData!] = GlobalKey(); |               widget.keys[ev.startDate!] = GlobalKey(); | ||||||
|               var d2 = DateTime.parse(ev.executionData!).toLocal(); |               var d2 = DateTime.parse(ev.startDate!).toLocal(); | ||||||
|               DateTime? d3; |               DateTime? d3; | ||||||
|               try { d3 = DateTime.parse(ev.endDate!).toLocal(); |               try { d3 = DateTime.parse(ev.endDate!).toLocal(); | ||||||
|               } catch (e) { /* */ } |               } catch (e) { /* */ } | ||||||
|               widgets.add(InkWell( |               widgets.add(InkWell( | ||||||
|                 onTap: () => widget.parent?.setState(() {  |                 onTap: () => widget.parent?.setState(() {  | ||||||
|                   selected = selected != element || ev.executionData != selectedReal  ? element : null;   |                   selected = selected != element || ev.startDate != selectedReal  ? element : null;   | ||||||
|                   selectedReal = selected == null ? null : ev.executionData;  |                   widget.parent!.widget.selectedID = selected; | ||||||
|  |                   selectedReal = selected == null ? null : ev.startDate;  | ||||||
|  |                   print("there"); | ||||||
|                   if (selectedReal  == null) { |                   if (selectedReal  == null) { | ||||||
|                     widget.parent!.widget.isDayPlanner = true; |                     widget.parent!.widget.isDayPlanner = true; | ||||||
|                   } |                   } | ||||||
|                 }),   |                 }),   | ||||||
|                 child: Container( key: widget.keys[ev.executionData!], |                 child: Container( key: widget.keys[ev.startDate!], | ||||||
|                 padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 50), |                 padding: const EdgeInsets.symmetric(vertical: 20, horizontal: 50), | ||||||
|                 decoration: BoxDecoration(  |                 decoration: BoxDecoration(  | ||||||
|                   border: selectedReal == ev.executionData ?  |                   border: selectedReal == ev.startDate ?  | ||||||
|                     Border.all(color: lightColor, width: 2) : Border(top: BorderSide(color: midColor)), |                     Border.all(color: lightColor, width: 2) : Border(top: BorderSide(color: midColor)), | ||||||
|                 ), |                 ), | ||||||
|                 child: Row(children: [ |                 child: Row(children: [ | ||||||
| @@ -59,7 +59,7 @@ class SchedulerItemWidgetState extends State<SchedulerItemWidget> { | |||||||
|                   SizedBox( width: (widget.width - 330) / 2, |                   SizedBox( width: (widget.width - 330) / 2, | ||||||
|                     child: Padding(  |                     child: Padding(  | ||||||
|                       padding: const EdgeInsets.only(left: 20),  |                       padding: const EdgeInsets.only(left: 20),  | ||||||
|                       child: Text(ev.name?.toUpperCase() ?? "", overflow: TextOverflow.ellipsis, |                       child: Text("${ev.name?.toUpperCase().split("EXECUTION")[0].replaceAll("_", "") ?? ""} EXECUTION", overflow: TextOverflow.ellipsis, | ||||||
|                       style: const TextStyle(color: Colors.black, fontWeight: FontWeight.w500)), |                       style: const TextStyle(color: Colors.black, fontWeight: FontWeight.w500)), | ||||||
|                     )), |                     )), | ||||||
|                   SizedBox( width: (widget.width - 340) / 2, |                   SizedBox( width: (widget.width - 340) / 2, | ||||||
|   | |||||||
| @@ -3,223 +3,223 @@ | |||||||
|   "packages": [ |   "packages": [ | ||||||
|     { |     { | ||||||
|       "name": "async", |       "name": "async", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/async-2.11.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/async-2.11.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "boolean_selector", |       "name": "boolean_selector", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "box_transform", |       "name": "box_transform", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/box_transform-0.4.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/box_transform-0.4.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "characters", |       "name": "characters", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/characters-1.3.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/characters-1.3.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "clock", |       "name": "clock", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/clock-1.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/clock-1.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "collection", |       "name": "collection", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/collection-1.18.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/collection-1.18.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "crypto", |       "name": "crypto", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/crypto-3.0.3", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/crypto-3.0.3", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dashed_path", |       "name": "dashed_path", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dashed_path-1.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dashed_path-1.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "dotted_line", |       "name": "dotted_line", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/dotted_line-3.2.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/dotted_line-3.2.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "fake_async", |       "name": "fake_async", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/fake_async-1.3.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/fake_async-1.3.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "fixnum", |       "name": "fixnum", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/fixnum-1.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/fixnum-1.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter", |       "name": "flutter", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/packages/flutter", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/packages/flutter", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_box_transform", |       "name": "flutter_box_transform", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_colorpicker", |       "name": "flutter_colorpicker", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.14" |       "languageVersion": "2.14" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "flutter_test", |       "name": "flutter_test", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/packages/flutter_test", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/packages/flutter_test", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "hover_menu", |       "name": "hover_menu", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/hover_menu-1.1.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/hover_menu-1.1.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.16" |       "languageVersion": "2.16" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "leak_tracker", |       "name": "leak_tracker", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "leak_tracker_flutter_testing", |       "name": "leak_tracker_flutter_testing", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "leak_tracker_testing", |       "name": "leak_tracker_testing", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "matcher", |       "name": "matcher", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/matcher-0.12.16+1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/matcher-0.12.16+1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "material_color_utilities", |       "name": "material_color_utilities", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "meta", |       "name": "meta", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/meta-1.15.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/meta-1.15.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "number_text_input_formatter", |       "name": "number_text_input_formatter", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "path", |       "name": "path", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/path-1.9.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/path-1.9.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "sky_engine", |       "name": "sky_engine", | ||||||
|       "rootUri": "file:///home/opencloud/Tools/flutter/bin/cache/pkg/sky_engine", |       "rootUri": "file:///home/mr/snap/flutter/common/flutter/bin/cache/pkg/sky_engine", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "source_span", |       "name": "source_span", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/source_span-1.10.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/source_span-1.10.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "sprintf", |       "name": "sprintf", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/sprintf-7.0.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/sprintf-7.0.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "stack_trace", |       "name": "stack_trace", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/stack_trace-1.11.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/stack_trace-1.11.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "stream_channel", |       "name": "stream_channel", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/stream_channel-2.1.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/stream_channel-2.1.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.19" |       "languageVersion": "2.19" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "string_scanner", |       "name": "string_scanner", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/string_scanner-1.2.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/string_scanner-1.2.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.18" |       "languageVersion": "2.18" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "term_glyph", |       "name": "term_glyph", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/term_glyph-1.2.1", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/term_glyph-1.2.1", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.12" |       "languageVersion": "2.12" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "test_api", |       "name": "test_api", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/test_api-0.7.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/test_api-0.7.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.2" |       "languageVersion": "3.2" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "typed_data", |       "name": "typed_data", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/typed_data-1.3.2", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/typed_data-1.3.2", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.17" |       "languageVersion": "2.17" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "uuid", |       "name": "uuid", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/uuid-4.4.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/uuid-4.4.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vector_math", |       "name": "vector_math", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_math-2.1.4", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vector_math-2.1.4", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "2.14" |       "languageVersion": "2.14" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "very_good_analysis", |       "name": "very_good_analysis", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/very_good_analysis-5.1.0", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/very_good_analysis-5.1.0", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.0" |       "languageVersion": "3.0" | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "vm_service", |       "name": "vm_service", | ||||||
|       "rootUri": "file:///home/opencloud/.pub-cache/hosted/pub.dev/vm_service-14.2.5", |       "rootUri": "file:///home/mr/.pub-cache/hosted/pub.dev/vm_service-14.2.5", | ||||||
|       "packageUri": "lib/", |       "packageUri": "lib/", | ||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     }, |     }, | ||||||
| @@ -230,10 +230,10 @@ | |||||||
|       "languageVersion": "3.3" |       "languageVersion": "3.3" | ||||||
|     } |     } | ||||||
|   ], |   ], | ||||||
|   "generated": "2024-12-17T16:53:00.428443Z", |   "generated": "2025-02-17T08:57:16.568019Z", | ||||||
|   "generator": "pub", |   "generator": "pub", | ||||||
|   "generatorVersion": "3.5.4", |   "generatorVersion": "3.5.3", | ||||||
|   "flutterRoot": "file:///home/opencloud/Tools/flutter", |   "flutterRoot": "file:///home/mr/snap/flutter/common/flutter", | ||||||
|   "flutterVersion": "3.24.5", |   "flutterVersion": "3.24.3", | ||||||
|   "pubCache": "file:///home/opencloud/.pub-cache" |   "pubCache": "file:///home/mr/.pub-cache" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,153 +1,153 @@ | |||||||
| async | async | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/async-2.11.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/async-2.11.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/async-2.11.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/async-2.11.0/lib/ | ||||||
| boolean_selector | boolean_selector | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/boolean_selector-2.1.1/lib/ | ||||||
| box_transform | box_transform | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/box_transform-0.4.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/box_transform-0.4.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/box_transform-0.4.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/box_transform-0.4.4/lib/ | ||||||
| characters | characters | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/characters-1.3.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/characters-1.3.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/characters-1.3.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/characters-1.3.0/lib/ | ||||||
| clock | clock | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/clock-1.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/clock-1.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/clock-1.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/clock-1.1.1/lib/ | ||||||
| collection | collection | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/collection-1.18.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/collection-1.18.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/collection-1.18.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/collection-1.18.0/lib/ | ||||||
| crypto | crypto | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/crypto-3.0.3/ | file:///home/mr/.pub-cache/hosted/pub.dev/crypto-3.0.3/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/crypto-3.0.3/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/crypto-3.0.3/lib/ | ||||||
| dashed_path | dashed_path | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dashed_path-1.0.1/lib/ | ||||||
| dotted_line | dotted_line | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dotted_line-3.2.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/dotted_line-3.2.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/dotted_line-3.2.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/dotted_line-3.2.2/lib/ | ||||||
| fake_async | fake_async | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fake_async-1.3.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/fake_async-1.3.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fake_async-1.3.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/fake_async-1.3.1/lib/ | ||||||
| fixnum | fixnum | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fixnum-1.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/fixnum-1.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/fixnum-1.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/fixnum-1.1.0/lib/ | ||||||
| flutter_box_transform | flutter_box_transform | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_box_transform-0.4.4/lib/ | ||||||
| flutter_colorpicker | flutter_colorpicker | ||||||
| 2.14 | 2.14 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/flutter_colorpicker-1.1.0/lib/ | ||||||
| hover_menu | hover_menu | ||||||
| 2.16 | 2.16 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/hover_menu-1.1.1/lib/ | ||||||
| leak_tracker | leak_tracker | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker-10.0.5/lib/ | ||||||
| leak_tracker_flutter_testing | leak_tracker_flutter_testing | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_flutter_testing-3.0.5/lib/ | ||||||
| leak_tracker_testing | leak_tracker_testing | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/leak_tracker_testing-3.0.1/lib/ | ||||||
| matcher | matcher | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/ | file:///home/mr/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/matcher-0.12.16+1/lib/ | ||||||
| material_color_utilities | material_color_utilities | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/material_color_utilities-0.11.1/lib/ | ||||||
| meta | meta | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/meta-1.15.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/meta-1.15.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/meta-1.15.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/meta-1.15.0/lib/ | ||||||
| number_text_input_formatter | number_text_input_formatter | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/ | file:///home/mr/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/number_text_input_formatter-1.0.0+8/lib/ | ||||||
| path | path | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path-1.9.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/path-1.9.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/path-1.9.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/path-1.9.0/lib/ | ||||||
| source_span | source_span | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/source_span-1.10.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/source_span-1.10.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/source_span-1.10.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/source_span-1.10.0/lib/ | ||||||
| sprintf | sprintf | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/sprintf-7.0.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/sprintf-7.0.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/sprintf-7.0.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/sprintf-7.0.0/lib/ | ||||||
| stack_trace | stack_trace | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stack_trace-1.11.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/stack_trace-1.11.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stack_trace-1.11.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/stack_trace-1.11.1/lib/ | ||||||
| stream_channel | stream_channel | ||||||
| 2.19 | 2.19 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/stream_channel-2.1.2/lib/ | ||||||
| string_scanner | string_scanner | ||||||
| 2.18 | 2.18 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/string_scanner-1.2.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/string_scanner-1.2.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/string_scanner-1.2.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/string_scanner-1.2.0/lib/ | ||||||
| term_glyph | term_glyph | ||||||
| 2.12 | 2.12 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/ | file:///home/mr/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/term_glyph-1.2.1/lib/ | ||||||
| test_api | test_api | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/test_api-0.7.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/test_api-0.7.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/test_api-0.7.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/test_api-0.7.2/lib/ | ||||||
| typed_data | typed_data | ||||||
| 2.17 | 2.17 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/typed_data-1.3.2/ | file:///home/mr/.pub-cache/hosted/pub.dev/typed_data-1.3.2/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/typed_data-1.3.2/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/typed_data-1.3.2/lib/ | ||||||
| uuid | uuid | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/uuid-4.4.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/uuid-4.4.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/uuid-4.4.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/uuid-4.4.0/lib/ | ||||||
| vector_math | vector_math | ||||||
| 2.14 | 2.14 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_math-2.1.4/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_math-2.1.4/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vector_math-2.1.4/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vector_math-2.1.4/lib/ | ||||||
| very_good_analysis | very_good_analysis | ||||||
| 3.0 | 3.0 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/very_good_analysis-5.1.0/ | file:///home/mr/.pub-cache/hosted/pub.dev/very_good_analysis-5.1.0/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/very_good_analysis-5.1.0/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/very_good_analysis-5.1.0/lib/ | ||||||
| vm_service | vm_service | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vm_service-14.2.5/ | file:///home/mr/.pub-cache/hosted/pub.dev/vm_service-14.2.5/ | ||||||
| file:///home/opencloud/.pub-cache/hosted/pub.dev/vm_service-14.2.5/lib/ | file:///home/mr/.pub-cache/hosted/pub.dev/vm_service-14.2.5/lib/ | ||||||
| flutter_flow_chart | flutter_flow_chart | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Source/opencloud/oc-local/src/oc-front/library/flutter_flow_chart/ | file:///home/mr/Documents/OC/oc-front/library/flutter_flow_chart/ | ||||||
| file:///home/opencloud/Source/opencloud/oc-local/src/oc-front/library/flutter_flow_chart/lib/ | file:///home/mr/Documents/OC/oc-front/library/flutter_flow_chart/lib/ | ||||||
| sky_engine | sky_engine | ||||||
| 3.2 | 3.2 | ||||||
| file:///home/opencloud/Tools/flutter/bin/cache/pkg/sky_engine/ | file:///home/mr/snap/flutter/common/flutter/bin/cache/pkg/sky_engine/ | ||||||
| file:///home/opencloud/Tools/flutter/bin/cache/pkg/sky_engine/lib/ | file:///home/mr/snap/flutter/common/flutter/bin/cache/pkg/sky_engine/lib/ | ||||||
| flutter | flutter | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter/ | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter/lib/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter/lib/ | ||||||
| flutter_test | flutter_test | ||||||
| 3.3 | 3.3 | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter_test/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter_test/ | ||||||
| file:///home/opencloud/Tools/flutter/packages/flutter_test/lib/ | file:///home/mr/snap/flutter/common/flutter/packages/flutter_test/lib/ | ||||||
| 2 | 2 | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| 3.24.5 | 3.24.3 | ||||||
| @@ -5,5 +5,5 @@ export 'src/dashboard.dart'; | |||||||
| export 'src/elements/connection_params.dart'; | export 'src/elements/connection_params.dart'; | ||||||
| export 'src/elements/flow_element.dart'; | export 'src/elements/flow_element.dart'; | ||||||
| export 'src/flow_chart.dart'; | export 'src/flow_chart.dart'; | ||||||
| export 'src/ui/draw_arrow.dart' show ArrowParams, ArrowStyle, ArrowDirection; | export 'src/ui/draw_arrow.dart' show ArrowParams, ArrowStyle, ArrowDirection, ArrowPainter; | ||||||
| export 'src/ui/grid_background.dart' show GridBackgroundParams; | export 'src/ui/grid_background.dart' show GridBackgroundParams; | ||||||
|   | |||||||
| @@ -28,13 +28,11 @@ class Dashboard extends ChangeNotifier { | |||||||
|   GlobalKey<FlowChartMenuState> chartMenuKey = GlobalKey<FlowChartMenuState>(); |   GlobalKey<FlowChartMenuState> chartMenuKey = GlobalKey<FlowChartMenuState>(); | ||||||
|   GlobalKey<ChartWidgetState> chartKey = GlobalKey<ChartWidgetState>(); |   GlobalKey<ChartWidgetState> chartKey = GlobalKey<ChartWidgetState>(); | ||||||
|   GlobalKey<FlowChartState> flutterChartKey = GlobalKey<FlowChartState>(); |   GlobalKey<FlowChartState> flutterChartKey = GlobalKey<FlowChartState>(); | ||||||
|  |   bool inDialog = false; | ||||||
|   List<Map<String, dynamic>> tempHistory = []; |   List<Map<String, dynamic>> tempHistory = []; | ||||||
|   List<Map<String, dynamic>> history = []; |   List<Map<String, dynamic>> history = []; | ||||||
|  |  | ||||||
|   Map<String, dynamic> scheduler = {}; |  | ||||||
|   Map<String, dynamic> info = {}; |   Map<String, dynamic> info = {}; | ||||||
|   bool scheduleActive = false; |  | ||||||
|   String? id; |   String? id; | ||||||
|   String name = ""; |   String name = ""; | ||||||
|   bool isMenu = true; |   bool isMenu = true; | ||||||
| @@ -47,8 +45,9 @@ class Dashboard extends ChangeNotifier { | |||||||
|   double defaultDashWidth = 0; |   double defaultDashWidth = 0; | ||||||
|   double defaultBackWidth = 10; |   double defaultBackWidth = 10; | ||||||
|   double defaultForwardWidth = 10; |   double defaultForwardWidth = 10; | ||||||
|   Future<void> Function(String? id)? save; |   Future<void> Function(String? id, BuildContext? context)? save; | ||||||
|   List<Widget> Function(FlowData? obj, String id)? infoItemWidget; |   List<Widget> Function(FlowData? obj, String id)? infoItemWidget; | ||||||
|  |   Widget Function(ArrowPainter item)? infoLinkWidget; | ||||||
|   List<Widget> Function()? infoWidget; |   List<Widget> Function()? infoWidget; | ||||||
|   FlowData? Function(Map<String, dynamic> json)? transformToData; |   FlowData? Function(Map<String, dynamic> json)? transformToData; | ||||||
|   bool addChange = false; |   bool addChange = false; | ||||||
| @@ -74,7 +73,6 @@ class Dashboard extends ChangeNotifier { | |||||||
|     this.save, |     this.save, | ||||||
|     this.dashColor = Colors.grey, |     this.dashColor = Colors.grey, | ||||||
|     this.midDashColor = Colors.blueGrey, |     this.midDashColor = Colors.blueGrey, | ||||||
|     this.scheduler = const {}, |  | ||||||
|     Offset? handlerFeedbackOffset, |     Offset? handlerFeedbackOffset, | ||||||
|     this.isMenu = true, |     this.isMenu = true, | ||||||
|     this.defaultDashSpace = 0, |     this.defaultDashSpace = 0, | ||||||
| @@ -93,7 +91,7 @@ class Dashboard extends ChangeNotifier { | |||||||
|     // This is a workaround to set the handlerFeedbackOffset |     // This is a workaround to set the handlerFeedbackOffset | ||||||
|     // to improve the user experience on devices with touch screens |     // to improve the user experience on devices with touch screens | ||||||
|     // This will prevent the handler being covered by user's finger |     // This will prevent the handler being covered by user's finger | ||||||
|     if (loadedGraph != null) { deserialize(loadedGraph!); } |     if (loadedGraph != null) { deserialize(loadedGraph!, false); } | ||||||
|     if (handlerFeedbackOffset != null) { |     if (handlerFeedbackOffset != null) { | ||||||
|       this.handlerFeedbackOffset = handlerFeedbackOffset; |       this.handlerFeedbackOffset = handlerFeedbackOffset; | ||||||
|     } else { |     } else { | ||||||
| @@ -110,18 +108,15 @@ class Dashboard extends ChangeNotifier { | |||||||
|     tempHistory = []; |     tempHistory = []; | ||||||
|     history = []; |     history = []; | ||||||
|   } |   } | ||||||
|   Future<void> saveDash(String? id) async { |   Future<void> saveDash(String? id, BuildContext? context) async { | ||||||
|     shouldSave = true; |  | ||||||
|     for (var element in saveRules) { |     for (var element in saveRules) { | ||||||
|       if (element(this)) { |       if (!element(this)) { | ||||||
|         shouldSave = true; |  | ||||||
|       } else { |  | ||||||
|         shouldSave = false; |         shouldSave = false; | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (save != null && shouldSave) { |     if (save != null && shouldSave) { | ||||||
|       save!(id); |       save!(id, context); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   Future<void> Function(String cat)? load; |   Future<void> Function(String cat)? load; | ||||||
| @@ -131,14 +126,12 @@ class Dashboard extends ChangeNotifier { | |||||||
|     final d = Dashboard( |     final d = Dashboard( | ||||||
|       name: map['name'] as String, |       name: map['name'] as String, | ||||||
|       isMenu: map['isMenu'] as bool, |       isMenu: map['isMenu'] as bool, | ||||||
|       scheduler: map['schedule'] as Map<String, String>? ?? {}, |  | ||||||
|       defaultDashSpace: map['defaultDashSpace'] as double? ?? 0, |       defaultDashSpace: map['defaultDashSpace'] as double? ?? 0, | ||||||
|       defaultDashWidth: map['defaultDashWidth'] as double? ?? 0, |       defaultDashWidth: map['defaultDashWidth'] as double? ?? 0, | ||||||
|       defaultArrowDirection: ArrowDirection.values[ |       defaultArrowDirection: ArrowDirection.values[ | ||||||
|           map['defaultArrowDirection'] as int? ?? 0], |           map['defaultArrowDirection'] as int? ?? 0], | ||||||
|       defaultArrowStyle: ArrowStyle.values[map['arrowStyle'] as int? ?? 0], |       defaultArrowStyle: ArrowStyle.values[map['arrowStyle'] as int? ?? 0], | ||||||
|     ); |     ); | ||||||
|     d..scheduleActive = map['schedule_active'] as bool? ?? false; |  | ||||||
|     d..arrows = List<ArrowPainter>.from( |     d..arrows = List<ArrowPainter>.from( | ||||||
|         (map['arrows'] as List<dynamic>).map<ArrowPainter>( |         (map['arrows'] as List<dynamic>).map<ArrowPainter>( | ||||||
|           (x) => ArrowPainter.fromMap(x as Map<String, dynamic>), |           (x) => ArrowPainter.fromMap(x as Map<String, dynamic>), | ||||||
| @@ -170,15 +163,14 @@ class Dashboard extends ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   void copyFromMap(Map<String, dynamic> map) { |   void copyFromMap(Map<String, dynamic> map) { | ||||||
|     scheduleActive = map['schedule_active'] as bool? ?? false; |     debugPrintStack(stackTrace: StackTrace.current, label: 'my_label', maxFrames: 5); | ||||||
|     scheduler = map['schedule'] as Map<String, String>? ?? {}; |  | ||||||
|     defaultArrowStyle = ArrowStyle.values[map['arrowStyle'] as int? ?? 0]; |     defaultArrowStyle = ArrowStyle.values[map['arrowStyle'] as int? ?? 0]; | ||||||
|     defaultDashSpace = map['defaultDashSpace'] as double? ?? 0; |     defaultDashSpace = map['defaultDashSpace'] as double? ?? 0; | ||||||
|     defaultDashWidth = map['defaultDashWidth'] as double? ?? 0; |     defaultDashWidth = map['defaultDashWidth'] as double? ?? 0; | ||||||
|     defaultArrowDirection = ArrowDirection.values[ |     defaultArrowDirection = ArrowDirection.values[ | ||||||
|           map['defaultArrowDirection'] as int? ?? 0]; |           map['defaultArrowDirection'] as int? ?? 0]; | ||||||
|     arrows = List<ArrowPainter>.from( |     arrows = List<ArrowPainter>.from( | ||||||
|         (map['arrows'] as List<dynamic>).map<ArrowPainter>( |           (map['arrows'] as List<dynamic>).map<ArrowPainter>( | ||||||
|           (x) => ArrowPainter.fromMap(x as Map<String, dynamic>), |           (x) => ArrowPainter.fromMap(x as Map<String, dynamic>), | ||||||
|         ), |         ), | ||||||
|       ); |       ); | ||||||
| @@ -257,26 +249,25 @@ class Dashboard extends ChangeNotifier { | |||||||
|     d["id"]=id; |     d["id"]=id; | ||||||
|     d["name"]=name; |     d["name"]=name; | ||||||
|     d["graph"]=graph; |     d["graph"]=graph; | ||||||
|     d["schedule"]=scheduler; |  | ||||||
|     d["schedule_active"]=scheduleActive; |  | ||||||
|     return d; |     return d; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void deserialize(Map<String, dynamic> graph) { |   void deserialize(Map<String, dynamic> graph, bool noHistory) { | ||||||
|     elements = []; |     elements = []; | ||||||
|     arrows = []; |     arrows = []; | ||||||
|     info["shared"] = graph["shared"] ?? []; |     info["shared"] = graph["shared"] ?? []; | ||||||
|     scheduler = graph['schedule'] ?? {}; |  | ||||||
|     scheduleActive = graph['schedule_active'] ?? false; |  | ||||||
|     setZoomFactor(graph["graph"]?["zoom"] ?? 1.0); |     setZoomFactor(graph["graph"]?["zoom"] ?? 1.0); | ||||||
|     for(var el in graph['graph']?['elements'] ?? []) { |     for(var el in graph['graph']?['elements'] ?? []) { | ||||||
|       List<ConnectionParams> nexts = []; |       List<ConnectionParams> nexts = []; | ||||||
|       var flow = FlowElement.deserialize(this, el); |       var flow = FlowElement.deserialize(this, el); | ||||||
|       for(var ar in graph['graph']['arrows']) { |       for(var ar in graph['graph']['arrows']) { | ||||||
|  |          | ||||||
|         if (ar['from']['id'] != flow.id) { continue; } |         if (ar['from']['id'] != flow.id) { continue; } | ||||||
|         nexts.add(ConnectionParams( |         nexts.add(ConnectionParams( | ||||||
|           srcElementId: ar['from']['id'], |           srcElementId: ar['from']['id'], | ||||||
|           destElementId: ar['to']['id'], |           destElementId: ar['to']['id'], | ||||||
|  |           infos: ar['infos'], | ||||||
|  |           env: ar['env'], | ||||||
|           arrowParams: ArrowParams.fromMap(ar["params"]), |           arrowParams: ArrowParams.fromMap(ar["params"]), | ||||||
|           pivots: [ |           pivots: [ | ||||||
|             Pivot(Offset(ar['from']['x'], (ar['from']['y']))),  |             Pivot(Offset(ar['from']['x'], (ar['from']['y']))),  | ||||||
| @@ -291,16 +282,35 @@ class Dashboard extends ChangeNotifier { | |||||||
|           arr.add(build); |           arr.add(build); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       try { |       if (flow.element != null) { | ||||||
|         FlowData data = arr.firstWhere((element) => element.getID() == flow.element?.getID()); |  | ||||||
|         flow.kind = ElementKind.widget; |         flow.kind = ElementKind.widget; | ||||||
|         flow.widget = chartKey.currentState?.widget.flowChart.widget.itemWidget(data); |         flow.widget = chartKey.currentState?.widget.flowChart.widget.itemWidget(flow.element!); | ||||||
|       } catch (e) { print(e); } |       } | ||||||
|       elements.add(flow); |       elements.add(flow); | ||||||
|     } |     } | ||||||
|     selectedMenuKey.currentState?.setState(() { }); |     selectedMenuKey.currentState?.setState(() { }); | ||||||
|     chartMenuKey.currentState?.setState(() { }); |     chartMenuKey.currentState?.setState(() { }); | ||||||
|     addToHistory(); |     if (!noHistory) { | ||||||
|  |       addToHistory(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   void applyInfos(Map<String, List<dynamic>> elInfos, Map<String, dynamic> graph) { | ||||||
|  |     for(var id in elInfos.keys) { | ||||||
|  |       try { | ||||||
|  |         var element = elements.firstWhere((element) => element.id == id); | ||||||
|  |         element.element?.addEnv(elInfos[id] ?? []); | ||||||
|  |       } catch (e) { /* */ } | ||||||
|  |     } | ||||||
|  |     for(var ar in graph['graph']['arrows']) { | ||||||
|  |         var arr = arrows.where((a) => a.fromID == ar['from']['id'] && a.toID == ar['to']['id']).toList(); | ||||||
|  |         for (var a in arr) { | ||||||
|  |           a.infos = ar['infos']; | ||||||
|  |           a.env = ar['env']; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     selectedLeftMenuKey.currentState?.setState(() { }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// add listener called when a new connection is created |   /// add listener called when a new connection is created | ||||||
| @@ -325,30 +335,31 @@ class Dashboard extends ChangeNotifier { | |||||||
|     handlerFeedbackOffset = offset; |     handlerFeedbackOffset = offset; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   List<ArrowPainter> GetArrowByElementID(String id, bool isinput) { |   List<ArrowPainter> getArrowByElementID(String id, bool isinput) { | ||||||
|     return arrows.where((element) => (isinput && element.toID.contains(id)) || (!isinput && element.fromID.contains(id))).toList(); |     return arrows.where((element) =>  | ||||||
|  |       (isinput && element.toID.contains(id)) || (!isinput && element.fromID.contains(id))).toList(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void addArrows(ArrowPainter f) { |   void addArrows(ArrowPainter f, BuildContext? context) { | ||||||
|     arrows.add(f); |     arrows.add(f); | ||||||
|     for (var f in arrowStyleRules) { |     for (var f in arrowStyleRules) { | ||||||
|       arrows = f(this); |       arrows = f(this); | ||||||
|     } |     } | ||||||
|     addChange = true; |     addChange = true; | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void removeArrows(bool Function(ArrowPainter) f) { |   void removeArrows(bool Function(ArrowPainter) f, BuildContext? context) { | ||||||
|     arrows.removeWhere((element) => f(element)); |     arrows.removeWhere((element) => f(element)); | ||||||
|     for (var f in arrowStyleRules) { |     for (var f in arrowStyleRules) { | ||||||
|       arrows = f(this); |       arrows = f(this); | ||||||
|     } |     } | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void removeElements(bool Function(FlowElement<FlowData>) f) { |   void removeElements(bool Function(FlowElement<FlowData>) f, BuildContext? context) { | ||||||
|     elements.removeWhere((element) => f(element)); |     elements.removeWhere((element) => f(element)); | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void clear() { |   void clear() { | ||||||
| @@ -356,9 +367,7 @@ class Dashboard extends ChangeNotifier { | |||||||
|     arrows.clear(); |     arrows.clear(); | ||||||
|     tempHistory = []; |     tempHistory = []; | ||||||
|     history = []; |     history = []; | ||||||
|     scheduler = {}; |  | ||||||
|     info = {}; |     info = {}; | ||||||
|     scheduleActive = false; |  | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|   bool noHistory = false; |   bool noHistory = false; | ||||||
| @@ -375,14 +384,13 @@ class Dashboard extends ChangeNotifier { | |||||||
|     Future.delayed(Duration(seconds: 1), () { |     Future.delayed(Duration(seconds: 1), () { | ||||||
|       chartMenuKey.currentState?.setState(() { }); |       chartMenuKey.currentState?.setState(() { }); | ||||||
|     }); |     }); | ||||||
|     ; |  | ||||||
|   } |   } | ||||||
|   bool isBack = false; |   bool isBack = false; | ||||||
|   void back() {  |   void back() {  | ||||||
|     if (canBack()) { |     if (canBack()) { | ||||||
|       noHistory = true; |       noHistory = true; | ||||||
|       tempHistory.removeLast();  |       tempHistory.removeLast();  | ||||||
|       if (tempHistory.length > 0) { |       if (tempHistory.isNotEmpty) { | ||||||
|         copyFromMap(tempHistory.last); |         copyFromMap(tempHistory.last); | ||||||
|       } |       } | ||||||
|       chartKey.currentState?.setState(() { }); |       chartKey.currentState?.setState(() { }); | ||||||
| @@ -413,28 +421,29 @@ class Dashboard extends ChangeNotifier { | |||||||
|     FlowElement element, |     FlowElement element, | ||||||
|     bool resizable, { |     bool resizable, { | ||||||
|     bool notify = true, |     bool notify = true, | ||||||
|  |     BuildContext? context | ||||||
|   }) { |   }) { | ||||||
|     element.isResizing = resizable; |     element.isResizing = resizable; | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|     if (notify) notifyListeners(); |     if (notify) notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   FlowElement? getElement(String id, {bool notify = true}) { |   FlowElement? getElement(String id, {bool notify = true}) { | ||||||
|     try { return elements.firstWhere((element) { |     try { return elements.firstWhere((element) { | ||||||
|       return element.id == id; |       return id.contains(element.id); | ||||||
|     }); } |     }); } | ||||||
|     catch (e) { return null; } |     catch (e) { return null; } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// add a [FlowElement] to the dashboard |   /// add a [FlowElement] to the dashboard | ||||||
|   void addElement(FlowElement element, {bool notify = true}) { |   void addElement(FlowElement element, BuildContext? context, {bool notify = true}) { | ||||||
|     if (element.id.isEmpty) { |     if (element.id.isEmpty) { | ||||||
|       element.id = const Uuid().v4(); |       element.id = const Uuid().v4(); | ||||||
|     } |     } | ||||||
|     element.setScale(1, gridBackgroundParams.scale); |     element.setScale(1, gridBackgroundParams.scale); | ||||||
|     elements.add(element); |     elements.add(element); | ||||||
|     addChange = true; |     addChange = true; | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|     if (notify) { |     if (notify) { | ||||||
|       notifyListeners(); |       notifyListeners(); | ||||||
|     } |     } | ||||||
| @@ -554,16 +563,16 @@ class Dashboard extends ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// remove all elements |   /// remove all elements | ||||||
|   void removeAllElements({bool notify = true}) { |   void removeAllElements(BuildContext? context, {bool notify = true}) { | ||||||
|     elements.clear(); |     elements.clear(); | ||||||
|     if (notify) notifyListeners(); |     if (notify) notifyListeners(); | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// remove the [handler] connection of [element] |   /// remove the [handler] connection of [element] | ||||||
|   void removeElementConnection( |   void removeElementConnection( | ||||||
|     FlowElement element, |     FlowElement element, | ||||||
|     Handler handler, { |     Handler handler, BuildContext? context, { | ||||||
|     bool notify = true, |     bool notify = true, | ||||||
|   }) { |   }) { | ||||||
|     Alignment alignment; |     Alignment alignment; | ||||||
| @@ -600,7 +609,7 @@ class Dashboard extends ChangeNotifier { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (notify) notifyListeners(); |     if (notify) notifyListeners(); | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// dissect an element connection |   /// dissect an element connection | ||||||
| @@ -690,19 +699,20 @@ class Dashboard extends ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// remove all the connection from the [element] |   /// remove all the connection from the [element] | ||||||
|   void removeElementConnections(FlowElement element, {bool notify = true}) { |   void removeElementConnections(FlowElement element, BuildContext? context, {bool notify = true}) { | ||||||
|     element.next.clear(); |     element.next.clear(); | ||||||
|     if (notify) notifyListeners(); |     if (notify) notifyListeners(); | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// remove all the elements with [id] from the dashboard |   /// remove all the elements with [id] from the dashboard | ||||||
|   void removeElementById(String id, {bool notify = true}) { |   void removeElementById(String id, BuildContext? context, {bool notify = true}) { | ||||||
|     // remove the element |     // remove the element | ||||||
|     var elementId = ''; |     var elementId = ''; | ||||||
|     elements.removeWhere((element) { |     elements.removeWhere((element) { | ||||||
|       if (element.id == id) { |       if (element.id == id) { | ||||||
|         elementId = element.id; |         elementId = element.id; | ||||||
|  |         arrows.removeWhere((a) => a.fromID.contains(elementId) || a.toID.contains(elementId)); | ||||||
|       } |       } | ||||||
|       return element.id == id; |       return element.id == id; | ||||||
|     }); |     }); | ||||||
| @@ -714,20 +724,23 @@ class Dashboard extends ChangeNotifier { | |||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|     if (notify) notifyListeners(); |     if (notify) notifyListeners(); | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /// remove element |   /// remove element | ||||||
|   /// return true if it has been removed |   /// return true if it has been removed | ||||||
|   bool removeElement(FlowElement element, {bool notify = true}) { |   bool removeElement(FlowElement element, BuildContext? context, {bool notify = true}) { | ||||||
|     // remove the element |     // remove the element | ||||||
|     var found = false; |     var found = false; | ||||||
|     final elementId = element.id; |     final elementId = element.id; | ||||||
|     elements.removeWhere((e) { |     elements.removeWhere((e) { | ||||||
|       if (e.id == element.id) found = true; |       if (e.id == element.id) { | ||||||
|  |         found = true; | ||||||
|  |  | ||||||
|  |       }  | ||||||
|       return e.id == element.id; |       return e.id == element.id; | ||||||
|     }); |     }); | ||||||
|  |     arrows.removeWhere( (e) => e.fromID.contains(elementId) || e.toID.contains(elementId)); | ||||||
|     // remove all connections to the element |     // remove all connections to the element | ||||||
|     for (final e in elements) { |     for (final e in elements) { | ||||||
|       e.next.removeWhere( |       e.next.removeWhere( | ||||||
| @@ -735,7 +748,7 @@ class Dashboard extends ChangeNotifier { | |||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|     if (notify) notifyListeners(); |     if (notify) notifyListeners(); | ||||||
|     saveDash(id); |     saveDash(id, context); | ||||||
|     return found; |     return found; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,14 +11,19 @@ class ConnectionParams { | |||||||
|     required this.srcElementId, |     required this.srcElementId, | ||||||
|     required this.destElementId, |     required this.destElementId, | ||||||
|     required this.arrowParams, |     required this.arrowParams, | ||||||
|  |     this.env = const [], | ||||||
|  |     this.infos = const [], | ||||||
|     List<Pivot>? pivots, |     List<Pivot>? pivots, | ||||||
|   }) : pivots = pivots ?? []; |   }) : pivots = pivots ?? []; | ||||||
|  |   List<dynamic> env = []; | ||||||
|  |   List<dynamic> infos = []; | ||||||
|   /// |   /// | ||||||
|   factory ConnectionParams.fromMap(Map<String, dynamic> map) { |   factory ConnectionParams.fromMap(Map<String, dynamic> map) { | ||||||
|     return ConnectionParams( |     return ConnectionParams( | ||||||
|       srcElementId: map['srcElementId'] as String, |       srcElementId: map['srcElementId'] as String, | ||||||
|       destElementId: map['destElementId'] as String, |       destElementId: map['destElementId'] as String, | ||||||
|  |       infos: map['infos'] as List<dynamic>? ?? [], | ||||||
|  |       env: map['env'] as List<dynamic>? ?? [], | ||||||
|       arrowParams: ArrowParams.fromMap(map['arrowParams'] as Map<String, dynamic>), |       arrowParams: ArrowParams.fromMap(map['arrowParams'] as Map<String, dynamic>), | ||||||
|       pivots: (map['pivots'] as List?) |       pivots: (map['pivots'] as List?) | ||||||
|               ?.map<Pivot>( |               ?.map<Pivot>( | ||||||
| @@ -37,7 +42,6 @@ class ConnectionParams { | |||||||
|   final String destElementId; |   final String destElementId; | ||||||
|   final String srcElementId; |   final String srcElementId; | ||||||
|  |  | ||||||
|  |  | ||||||
|   /// Arrow parameters. |   /// Arrow parameters. | ||||||
|   final ArrowParams arrowParams; |   final ArrowParams arrowParams; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -374,6 +374,7 @@ class FlowElement<T extends FlowData> extends ChangeNotifier { | |||||||
|     graphElement['width'] = size.width; |     graphElement['width'] = size.width; | ||||||
|     graphElement['height'] = size.height; |     graphElement['height'] = size.height; | ||||||
|     graphElement['element']=element?.serialize(); |     graphElement['element']=element?.serialize(); | ||||||
|  |     graphElement['next'] = next.map((x) => x.toMap()).toList(); | ||||||
|     return graphElement; |     return graphElement; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -385,6 +386,13 @@ class FlowElement<T extends FlowData> extends ChangeNotifier { | |||||||
|         position: Offset(double.parse("${map['x']}"), double.parse("${map['y']}")), |         position: Offset(double.parse("${map['x']}"), double.parse("${map['y']}")), | ||||||
|         size: Size(double.parse("${map['width']}"), double.parse("${map['height']}")), |         size: Size(double.parse("${map['width']}"), double.parse("${map['height']}")), | ||||||
|         element: (dashboard.transformToData != null ? dashboard.transformToData!(map['element'] ?? {}) : null) as T?, |         element: (dashboard.transformToData != null ? dashboard.transformToData!(map['element'] ?? {}) : null) as T?, | ||||||
|  |         /*next: map['next'] != null && (map['next'] as List).isNotEmpty | ||||||
|  |           ? List<ConnectionParams>.from( | ||||||
|  |               (map['next'] as List<dynamic>).map<dynamic>( | ||||||
|  |                 (x) => ConnectionParams.fromMap(x as Map<String, dynamic>), | ||||||
|  |               ), | ||||||
|  |             ) | ||||||
|  |           : []*/ | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,29 +1,27 @@ | |||||||
| // ignore: directives_ordering | // ignore: directives_ordering | ||||||
| import 'dart:developer'; | import 'package:uuid/uuid.dart'; | ||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
| import 'package:flutter/foundation.dart'; | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; |  | ||||||
| import 'package:flutter_flow_chart/src/flow_chart_left_menu.dart'; |  | ||||||
| import 'package:flutter_flow_chart/src/flow_chart_menu.dart'; |  | ||||||
| import 'package:flutter_flow_chart/src/flow_chart_selected_menu.dart'; |  | ||||||
| import 'package:flutter_flow_chart/src/ui/draw_arrow.dart'; | import 'package:flutter_flow_chart/src/ui/draw_arrow.dart'; | ||||||
|  | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
|  | import 'package:flutter_flow_chart/src/flow_chart_menu.dart'; | ||||||
| import 'package:flutter_flow_chart/src/ui/element_widget.dart'; | import 'package:flutter_flow_chart/src/ui/element_widget.dart'; | ||||||
| import 'package:flutter_flow_chart/src/ui/grid_background.dart'; | import 'package:flutter_flow_chart/src/ui/grid_background.dart'; | ||||||
| import 'package:flutter_flow_chart/src/ui/segment_handler.dart'; | import 'package:flutter_flow_chart/src/ui/segment_handler.dart'; | ||||||
| import 'package:uuid/uuid.dart'; | import 'package:flutter_flow_chart/src/flow_chart_left_menu.dart'; | ||||||
|  | import 'package:flutter_flow_chart/src/flow_chart_selected_menu.dart'; | ||||||
|  |  | ||||||
|  |  | ||||||
| /// Main flow chart Widget. | /// Main flow chart Widget. | ||||||
| /// It displays the background grid, all the elements and connection lines | /// It displays the background grid, all the elements and connection lines | ||||||
| abstract class FlowData { | abstract class FlowData { | ||||||
|   String getID(); |   String getID(); | ||||||
|   String getName(); |   String getName(); | ||||||
|  |   String getType(); | ||||||
|   double? getWidth(); |   double? getWidth(); | ||||||
|   double? getHeight(); |   double? getHeight(); | ||||||
|  |   void addEnv(List<dynamic> infos); | ||||||
|   Map<String, dynamic> setVariable(List<String> keys, dynamic value, Map<String, dynamic> map); |  | ||||||
|   dynamic getVariable(List<String> keys, Map<String, dynamic> map); |  | ||||||
|   Map<String, dynamic> serialize(); |   Map<String, dynamic> serialize(); | ||||||
|   FlowData deserialize(Map<String, dynamic> data); |   FlowData deserialize(Map<String, dynamic> data); | ||||||
| } | } | ||||||
| @@ -321,17 +319,21 @@ class FlowChartState<T extends FlowData> extends State<FlowChart> { | |||||||
|         Future.delayed(Duration(milliseconds: 100), () => widget.dashboard.load!(widget.dashboard.id!) );     |         Future.delayed(Duration(milliseconds: 100), () => widget.dashboard.load!(widget.dashboard.id!) );     | ||||||
|       } else { |       } else { | ||||||
|         Future.delayed(Duration(milliseconds: 100), () { |         Future.delayed(Duration(milliseconds: 100), () { | ||||||
|           showDialog( |           if (!widget.dashboard.inDialog) { | ||||||
|           barrierDismissible: false, |             widget.dashboard.inDialog = true; | ||||||
|           context: context, builder: (context) { |             showDialog( | ||||||
|           return AlertDialog( |               barrierDismissible: false, | ||||||
|             titlePadding: EdgeInsets.zero, |               context: context, builder: (context) { | ||||||
|             insetPadding: EdgeInsets.zero, |               return AlertDialog( | ||||||
|             backgroundColor: Colors.white, |                 titlePadding: EdgeInsets.zero, | ||||||
|             shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(0)), |                 insetPadding: EdgeInsets.zero, | ||||||
|             title: widget.onDashboardAlertOpened!(context, widget.dashboard)); |                 backgroundColor: Colors.white, | ||||||
|  |                 shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(0)), | ||||||
|  |                 title: widget.onDashboardAlertOpened!(context, widget.dashboard)); | ||||||
|                |                | ||||||
|         }); }); |             }); | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|       } |       } | ||||||
|     } else { |     } else { | ||||||
|       widget.dashboard.isOpened  = true; |       widget.dashboard.isOpened  = true; | ||||||
| @@ -357,14 +359,14 @@ class FlowChartState<T extends FlowData> extends State<FlowChart> { | |||||||
|       onKeyEvent: (event) { |       onKeyEvent: (event) { | ||||||
|         bool change = false; |         bool change = false; | ||||||
|         if (event.logicalKey == LogicalKeyboardKey.controlLeft) { |         if (event.logicalKey == LogicalKeyboardKey.controlLeft) { | ||||||
|           isCtrl = event is KeyDownEvent ||   event is KeyRepeatEvent; |           isCtrl = event is KeyDownEvent || event is KeyRepeatEvent; | ||||||
|         } |         } | ||||||
|         if (event is KeyDownEvent || event.logicalKey == LogicalKeyboardKey.keyZ && isCtrl) { |         /*if ((event is KeyDownEvent || event.logicalKey == LogicalKeyboardKey.keyZ) && isCtrl) { | ||||||
|           widget.dashboard.back(); |           widget.dashboard.back(); | ||||||
|         } |         } | ||||||
|         if (event is KeyDownEvent || event.logicalKey == LogicalKeyboardKey.keyY && isCtrl) { |         if ((event is KeyDownEvent || event.logicalKey == LogicalKeyboardKey.keyY) && isCtrl) { | ||||||
|           widget.dashboard.forward(); |           widget.dashboard.forward(); | ||||||
|         } |         }*/ | ||||||
|         if (event is KeyDownEvent && event.logicalKey == LogicalKeyboardKey.add) { |         if (event is KeyDownEvent && event.logicalKey == LogicalKeyboardKey.add) { | ||||||
|           change = true; |           change = true; | ||||||
|           for (var el in widget.dashboard.elementSelected) { |           for (var el in widget.dashboard.elementSelected) { | ||||||
| @@ -379,18 +381,23 @@ class FlowChartState<T extends FlowData> extends State<FlowChart> { | |||||||
|                 kind: el.kind, |                 kind: el.kind, | ||||||
|                 handlers: el.handlers, |                 handlers: el.handlers, | ||||||
|                 handlerSize: el.handlerSize, |                 handlerSize: el.handlerSize, | ||||||
|               )); |               ), context); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         if (event is KeyDownEvent && event.logicalKey == LogicalKeyboardKey.delete) { |         if (event is KeyDownEvent && event.logicalKey == LogicalKeyboardKey.delete) { | ||||||
|           change = true; |           change = true; | ||||||
|           widget.dashboard.removeElements( (el) => el.isSelected ); |           widget.dashboard.removeElements((element) { | ||||||
|  |               if (element.isSelected) { | ||||||
|  |                 widget.dashboard.arrows.removeWhere( (e) => e.toID.contains(element.id) || e.fromID.contains(element.id)); | ||||||
|  |               } | ||||||
|  |             return element.isSelected; | ||||||
|  |           }, context); | ||||||
|           for (var arrow in widget.dashboard.arrowsSelected) { |           for (var arrow in widget.dashboard.arrowsSelected) { | ||||||
|               for (var el in widget.dashboard.elements.where((element) => element.id == arrow.fromID.split("_")[0])) { |               for (var el in widget.dashboard.elements.where((element) => element.id == arrow.fromID.split("_")[0])) { | ||||||
|                 el.next.removeAt(int.parse(arrow.fromID.split("_")[1])); |                 el.next.removeAt(int.parse(arrow.fromID.split("_")[1])); | ||||||
|               } |               } | ||||||
|           } |           } | ||||||
|           widget.dashboard.removeArrows( (el) => el.isSelected ); |           widget.dashboard.removeArrows( (el) => el.isSelected, context ); | ||||||
|         } |         } | ||||||
|         if (change) { |         if (change) { | ||||||
|           DrawingArrow.instance.notifyListeners(); |           DrawingArrow.instance.notifyListeners(); | ||||||
| @@ -459,7 +466,7 @@ class FlowChartState<T extends FlowData> extends State<FlowChart> { | |||||||
|                       Handler.rightCenter, |                       Handler.rightCenter, | ||||||
|                     ], |                     ], | ||||||
|                   ); |                   ); | ||||||
|               widget.dashboard.addElement(el); |               widget.dashboard.addElement(el, context); | ||||||
|             }, |             }, | ||||||
|           ))] |           ))] | ||||||
|         ), |         ), | ||||||
| @@ -495,8 +502,8 @@ class FlowChartState<T extends FlowData> extends State<FlowChart> { | |||||||
|         onDragEnd: (d) => node.requestFocus(), |         onDragEnd: (d) => node.requestFocus(), | ||||||
|         childWhenDragging: Opacity(opacity: .5,  |         childWhenDragging: Opacity(opacity: .5,  | ||||||
|         child: Padding( padding: const EdgeInsets.all(10),  |         child: Padding( padding: const EdgeInsets.all(10),  | ||||||
|           child: Container( child: widget.itemWidget(e), alignment: Alignment.center, |           child: Container( alignment: Alignment.center, | ||||||
|             constraints: BoxConstraints(maxHeight: realSize - 10, maxWidth: realSize - 10), ))), |             constraints: BoxConstraints(maxHeight: realSize - 10, maxWidth: realSize - 10), child: widget.itemWidget(e), ))), | ||||||
|         feedback: Container( alignment: Alignment.center, constraints: BoxConstraints(maxHeight: realSize, maxWidth: realSize), |         feedback: Container( alignment: Alignment.center, constraints: BoxConstraints(maxHeight: realSize, maxWidth: realSize), | ||||||
|           child: widget.itemWidget(e) ), |           child: widget.itemWidget(e) ), | ||||||
|         child: InkWell( mouseCursor: SystemMouseCursors.grab, child: Padding( padding: const EdgeInsets.all(10),  |         child: InkWell( mouseCursor: SystemMouseCursors.grab, child: Padding( padding: const EdgeInsets.all(10),  | ||||||
| @@ -548,6 +555,8 @@ class _DrawingArrowWidgetState extends State<DrawingArrowWidget> { | |||||||
|       painter: ArrowPainter( |       painter: ArrowPainter( | ||||||
|         fromID: DrawingArrow.instance.fromID, |         fromID: DrawingArrow.instance.fromID, | ||||||
|         toID: "", |         toID: "", | ||||||
|  |         env: DrawingArrow.instance.env, | ||||||
|  |         infos: DrawingArrow.instance.infos, | ||||||
|         params: DrawingArrow.instance.params, |         params: DrawingArrow.instance.params, | ||||||
|         from: DrawingArrow.instance.from, |         from: DrawingArrow.instance.from, | ||||||
|         to: DrawingArrow.instance.to, |         to: DrawingArrow.instance.to, | ||||||
| @@ -878,9 +887,7 @@ class ChartWidgetState<T extends FlowData> extends State<ChartWidget> { | |||||||
|                     ), |                     ), | ||||||
|                   // Draw arrows         |                   // Draw arrows         | ||||||
|                   for (int i = 0; i < widget.dashboard.elements.length; i++) |                   for (int i = 0; i < widget.dashboard.elements.length; i++) | ||||||
|                      |  | ||||||
|                     for (int n = 0; n < widget.dashboard.elements[i].next.length; n++)  |                     for (int n = 0; n < widget.dashboard.elements[i].next.length; n++)  | ||||||
|                        |  | ||||||
|                       DrawArrow( |                       DrawArrow( | ||||||
|                         flow: this, |                         flow: this, | ||||||
|                         key: UniqueKey(), |                         key: UniqueKey(), | ||||||
| @@ -889,6 +896,8 @@ class ChartWidgetState<T extends FlowData> extends State<ChartWidget> { | |||||||
|                         destElement: widget.dashboard.elements[widget.dashboard.findElementIndexById( |                         destElement: widget.dashboard.elements[widget.dashboard.findElementIndexById( | ||||||
|                           widget.dashboard.elements[i].next[n].destElementId, |                           widget.dashboard.elements[i].next[n].destElementId, | ||||||
|                         )], |                         )], | ||||||
|  |                         env: widget.dashboard.elements[i].next[n].env, | ||||||
|  |                         infos: widget.dashboard.elements[i].next[n].infos, | ||||||
|                         arrowParams: widget.dashboard.elements[i].next[n].arrowParams, |                         arrowParams: widget.dashboard.elements[i].next[n].arrowParams, | ||||||
|                         pivots: widget.dashboard.elements[i].next[n].pivots, |                         pivots: widget.dashboard.elements[i].next[n].pivots, | ||||||
|                       ), |                       ), | ||||||
|   | |||||||
| @@ -37,17 +37,20 @@ class FlowChartMenuState extends State<FlowChartMenu> { | |||||||
|                               onTap: () { |                               onTap: () { | ||||||
|                                 widget.dashboard.name = "graph_${DateTime.now().toString().replaceAll(" ", "_").substring(0, DateTime.now().toString().length - 7)}"; |                                 widget.dashboard.name = "graph_${DateTime.now().toString().replaceAll(" ", "_").substring(0, DateTime.now().toString().length - 7)}"; | ||||||
|                                 widget.dashboard.isOpened = true; |                                 widget.dashboard.isOpened = true; | ||||||
|                                 showDialog( |                                 if (!widget.dashboard.inDialog) { | ||||||
|                                     barrierDismissible: false, |                                   widget.dashboard.inDialog = true; | ||||||
|                                     context: context, builder: (context) { |                                   showDialog( | ||||||
|                                     return AlertDialog( |                                       barrierDismissible: false, | ||||||
|                                       titlePadding: EdgeInsets.zero, |                                       context: context, builder: (context) { | ||||||
|                                       insetPadding: EdgeInsets.zero, |                                       return AlertDialog( | ||||||
|                                       backgroundColor: Colors.white, |                                         titlePadding: EdgeInsets.zero, | ||||||
|                                       shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(0)), |                                         insetPadding: EdgeInsets.zero, | ||||||
|                                       title: widget.chart.widget.flowChart.widget.onDashboardAlertOpened!( |                                         backgroundColor: Colors.white, | ||||||
|                                         context, widget.dashboard)); |                                         shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(0)), | ||||||
|                                   }); |                                         title: widget.chart.widget.flowChart.widget.onDashboardAlertOpened!( | ||||||
|  |                                           context, widget.dashboard)); | ||||||
|  |                                     }); | ||||||
|  |                                 } | ||||||
|                               },  |                               },  | ||||||
|                               child: Icon(Icons.folder, color: Colors.white))))), |                               child: Icon(Icons.folder, color: Colors.white))))), | ||||||
|                         ])), |                         ])), | ||||||
| @@ -57,8 +60,6 @@ class FlowChartMenuState extends State<FlowChartMenu> { | |||||||
|                         ), |                         ), | ||||||
|                         child: Padding( padding: EdgeInsets.symmetric(horizontal: 10),  |                         child: Padding( padding: EdgeInsets.symmetric(horizontal: 10),  | ||||||
|                           child: PopupMenuButton<DisplayEnum>( |                           child: PopupMenuButton<DisplayEnum>( | ||||||
|                             child:  |  | ||||||
|                             Row( children: [ Icon(Icons.fullscreen, color: Colors.white), Icon(Icons.arrow_drop_down, size: 10, color: Colors.white) ]), |  | ||||||
|                             initialValue: null, |                             initialValue: null, | ||||||
|                             onSelected: (DisplayEnum value) { |                             onSelected: (DisplayEnum value) { | ||||||
|                               if (value == DisplayEnum.MENU) {  widget.dashboard.isMenu = !widget.dashboard.isMenu; } |                               if (value == DisplayEnum.MENU) {  widget.dashboard.isMenu = !widget.dashboard.isMenu; } | ||||||
| @@ -83,7 +84,9 @@ class FlowChartMenuState extends State<FlowChartMenu> { | |||||||
|                                       child: Text(widget.dashboard.isInfo ? 'hide info' : 'show info', textAlign: TextAlign.center,)) |                                       child: Text(widget.dashboard.isInfo ? 'hide info' : 'show info', textAlign: TextAlign.center,)) | ||||||
|                                 ]), |                                 ]), | ||||||
|                               ), |                               ), | ||||||
|                             ] |                             ], | ||||||
|  |                             child:  | ||||||
|  |                             Row( children: [ Icon(Icons.fullscreen, color: Colors.white), Icon(Icons.arrow_drop_down, size: 10, color: Colors.white) ]) | ||||||
|                             ),) |                             ),) | ||||||
|                         ) |                         ) | ||||||
|                       ), |                       ), | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import 'package:flutter_flow_chart/src/ui/draw_arrow.dart'; | |||||||
| import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | import 'package:flutter_flow_chart/flutter_flow_chart.dart'; | ||||||
| import 'package:flutter_colorpicker/flutter_colorpicker.dart'; | import 'package:flutter_colorpicker/flutter_colorpicker.dart'; | ||||||
| import 'package:number_text_input_formatter/number_text_input_formatter.dart'; | import 'package:number_text_input_formatter/number_text_input_formatter.dart'; | ||||||
|  | import 'package:uuid/uuid.dart'; | ||||||
|  |  | ||||||
| class FlowChartSelectedMenu extends StatefulWidget { | class FlowChartSelectedMenu extends StatefulWidget { | ||||||
|   Dashboard dashboard; |   Dashboard dashboard; | ||||||
| @@ -31,15 +32,14 @@ class FlowChartSelectedMenuState extends State<FlowChartSelectedMenu> { | |||||||
|                             Tooltip( message: "remove", |                             Tooltip( message: "remove", | ||||||
|                               child: InkWell( mouseCursor: SystemMouseCursors.click,  |                               child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|                               onTap: () {  |                               onTap: () {  | ||||||
|                                 widget.dashboard.removeArrows((element) {  |                                 widget.dashboard.removeElements((element) { | ||||||
|                                   if (element.isSelected && element.elementIndex != null && element.connIndex != null) { |                                   if (element.isSelected) { | ||||||
|                                     widget.dashboard.elements[element.elementIndex!].next.removeAt(element.connIndex!); |                                     widget.dashboard.arrows.removeWhere( (e) => e.toID.contains(element.id) || e.fromID.contains(element.id)); | ||||||
|                                   } |                                   } | ||||||
|                                   return element.isSelected; |                                   return element.isSelected; | ||||||
|                                 });  |                                 }, context);  | ||||||
|                                 widget.dashboard.removeElements((element) => element.isSelected);  |  | ||||||
|                                 Future.delayed(Duration(milliseconds: 100), () { |                                 Future.delayed(Duration(milliseconds: 100), () { | ||||||
|                                   widget.dashboard.chartKey.currentState?.setState(() { }); |                                   widget.dashboard.flutterChartKey.currentState?.setState(() { }); | ||||||
|                                 }); |                                 }); | ||||||
|                               }, child: Container( margin: EdgeInsets.all(10), |                               }, child: Container( margin: EdgeInsets.all(10), | ||||||
|                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.black, width: 1)), |                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.black, width: 1)), | ||||||
| @@ -51,7 +51,9 @@ class FlowChartSelectedMenuState extends State<FlowChartSelectedMenu> { | |||||||
|                               child: InkWell( mouseCursor: SystemMouseCursors.click,  |                               child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|                               onTap: () { |                               onTap: () { | ||||||
|                                 for (var sel in widget.dashboard.elementSelected) { |                                 for (var sel in widget.dashboard.elementSelected) { | ||||||
|                                   widget.dashboard.addElement(FlowElement.fromMap(widget.dashboard, sel.toMap())); |                                   var el =FlowElement.fromMap(widget.dashboard, sel.toMap()); | ||||||
|  |                                   el.id = Uuid().v8(); | ||||||
|  |                                   widget.dashboard.addElement(FlowElement.fromMap(widget.dashboard, sel.toMap()), context); | ||||||
|                                   widget.dashboard.elements.last.position += Offset(50, 50); |                                   widget.dashboard.elements.last.position += Offset(50, 50); | ||||||
|                                 } |                                 } | ||||||
|                                 Future.delayed(Duration(milliseconds: 100), () { |                                 Future.delayed(Duration(milliseconds: 100), () { | ||||||
| @@ -67,6 +69,42 @@ class FlowChartSelectedMenuState extends State<FlowChartSelectedMenu> { | |||||||
|                         ) : Container() |                         ) : Container() | ||||||
|         ]) |         ]) | ||||||
|       )); |       )); | ||||||
|  |     } else if (widget.isDashboardInfo && widget.dashboard.arrowsSelected.length == 1) { | ||||||
|  |       w = Container( | ||||||
|  |         width: 200, | ||||||
|  |         height: widget.height, | ||||||
|  |         color: widget.dashboard.midDashColor, | ||||||
|  |         child: SingleChildScrollView( child: Column( children: [  | ||||||
|  |           widget.dashboard.infoItemWidget != null ? | ||||||
|  |           widget.dashboard.infoLinkWidget!(widget.dashboard.arrowsSelected.first) : Container(), | ||||||
|  |           widget.dashboard.arrowsSelected.isNotEmpty || widget.dashboard.elementSelected.isNotEmpty ? Container(  | ||||||
|  |                           width: 200, | ||||||
|  |                           margin: EdgeInsets.only(top: 15), | ||||||
|  |                           decoration: BoxDecoration(border: Border( | ||||||
|  |                             top: BorderSide(color: Colors.grey, width: 1))), | ||||||
|  |                           child: Column(crossAxisAlignment: CrossAxisAlignment.center, children: [ | ||||||
|  |                             Tooltip( message: "remove", | ||||||
|  |                               child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|  |                               onTap: () { | ||||||
|  |                                 widget.dashboard.removeArrows((element) {  | ||||||
|  |                                   if (element.isSelected && element.elementIndex != null && element.connIndex != null) { | ||||||
|  |                                     widget.dashboard.elements[element.elementIndex!].next.removeAt(element.connIndex!); | ||||||
|  |                                   } | ||||||
|  |                                   return element.isSelected; | ||||||
|  |                                 }, context);  | ||||||
|  |                                 Future.delayed(Duration(milliseconds: 100), () { | ||||||
|  |                                   widget.dashboard.flutterChartKey.currentState?.setState(() { }); | ||||||
|  |                                 }); | ||||||
|  |                               }, child: Container( margin: EdgeInsets.all(10), | ||||||
|  |                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.black, width: 1)), | ||||||
|  |                                 width: 200, height: 30, | ||||||
|  |                                 child: Icon(Icons.delete_outline, color: Colors.black), | ||||||
|  |                               )) | ||||||
|  |                             ), | ||||||
|  |                           ]) | ||||||
|  |                         ) : Container() | ||||||
|  |         ]) | ||||||
|  |       )); | ||||||
|     } else if (widget.isDashboardInfo && widget.dashboard.infoWidget != null) { |     } else if (widget.isDashboardInfo && widget.dashboard.infoWidget != null) { | ||||||
|        w = Container( |        w = Container( | ||||||
|         width: 200, |         width: 200, | ||||||
| @@ -515,15 +553,14 @@ class FlowChartSelectedMenuState extends State<FlowChartSelectedMenu> { | |||||||
|                             Tooltip( message: "remove", |                             Tooltip( message: "remove", | ||||||
|                               child: InkWell( mouseCursor: SystemMouseCursors.click,  |                               child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|                               onTap: () { |                               onTap: () { | ||||||
|                                 widget.dashboard.removeArrows((element) {  |                                 widget.dashboard.removeElements((element) { | ||||||
|                                   if (element.isSelected && element.elementIndex != null && element.connIndex != null) { |                                   if (element.isSelected) { | ||||||
|                                     widget.dashboard.elements[element.elementIndex!].next.removeAt(element.connIndex!); |                                     widget.dashboard.arrows.removeWhere( (e) => e.toID.contains(element.id) || e.fromID.contains(element.id)); | ||||||
|                                   } |                                   } | ||||||
|                                   return element.isSelected; |                                   return element.isSelected; | ||||||
|                                 });  |                                 }, context); | ||||||
|                                 widget.dashboard.removeElements((element) => element.isSelected);  |  | ||||||
|                                 Future.delayed(Duration(milliseconds: 100), () { |                                 Future.delayed(Duration(milliseconds: 100), () { | ||||||
|                                   widget.dashboard.chartKey.currentState?.setState(() { }); |                                   widget.dashboard.flutterChartKey.currentState?.setState(() { }); | ||||||
|                                 }); |                                 }); | ||||||
|                               }, child: Container( margin: EdgeInsets.all(10), |                               }, child: Container( margin: EdgeInsets.all(10), | ||||||
|                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.black, width: 1)), |                                 decoration: BoxDecoration(borderRadius: BorderRadius.circular(5), border: Border.all(color: Colors.black, width: 1)), | ||||||
| @@ -535,7 +572,7 @@ class FlowChartSelectedMenuState extends State<FlowChartSelectedMenu> { | |||||||
|                               child: InkWell( mouseCursor: SystemMouseCursors.click,  |                               child: InkWell( mouseCursor: SystemMouseCursors.click,  | ||||||
|                               onTap: () { |                               onTap: () { | ||||||
|                                 for (var sel in widget.dashboard.elementSelected) { |                                 for (var sel in widget.dashboard.elementSelected) { | ||||||
|                                   widget.dashboard.addElement(FlowElement.fromMap(widget.dashboard, sel.toMap())); |                                   widget.dashboard.addElement(FlowElement.fromMap(widget.dashboard, sel.toMap()), context); | ||||||
|                                   widget.dashboard.elements.last.position += Offset(50, 50); |                                   widget.dashboard.elements.last.position += Offset(50, 50); | ||||||
|                                 } |                                 } | ||||||
|                                 Future.delayed(Duration(milliseconds: 100), () { |                                 Future.delayed(Duration(milliseconds: 100), () { | ||||||
|   | |||||||
| @@ -211,6 +211,8 @@ class DrawingArrow extends ChangeNotifier { | |||||||
|  |  | ||||||
|   /// Arrow parameters. |   /// Arrow parameters. | ||||||
|   ArrowParams params = ArrowParams(); |   ArrowParams params = ArrowParams(); | ||||||
|  |   List<dynamic> env = []; | ||||||
|  |   List<dynamic> infos = []; | ||||||
|   String fromID = ""; |   String fromID = ""; | ||||||
|   String toID = ""; |   String toID = ""; | ||||||
|  |  | ||||||
| @@ -264,13 +266,16 @@ class DrawArrow extends StatefulWidget { | |||||||
|     required this.srcElement, |     required this.srcElement, | ||||||
|     required this.destElement, |     required this.destElement, | ||||||
|     required List<Pivot> pivots, |     required List<Pivot> pivots, | ||||||
|  |     this.infos = const [], | ||||||
|  |     this.env = const [], | ||||||
|     super.key, |     super.key, | ||||||
|     ArrowParams? arrowParams, |     ArrowParams? arrowParams, | ||||||
|   })  : arrowParams = arrowParams ?? ArrowParams(), |   })  : arrowParams = arrowParams ?? ArrowParams(), | ||||||
|         pivots = PivotsNotifier(pivots); |         pivots = PivotsNotifier(pivots); | ||||||
|  |  | ||||||
|   final int index; |   final int index; | ||||||
|  |    List<dynamic> env = []; | ||||||
|  |   List<dynamic> infos = []; | ||||||
|   /// |   /// | ||||||
|   final ArrowParams arrowParams; |   final ArrowParams arrowParams; | ||||||
|  |  | ||||||
| @@ -340,6 +345,8 @@ class DrawArrowState extends State<DrawArrow> { | |||||||
|           builder: (context) { |           builder: (context) { | ||||||
|             var painter = ArrowPainter( |             var painter = ArrowPainter( | ||||||
|                 connIndex: widget.index, |                 connIndex: widget.index, | ||||||
|  |                 infos: widget.infos, | ||||||
|  |                 env: widget.env, | ||||||
|                 elementIndex: widget.flow.widget.dashboard.elements.indexOf(widget.srcElement), |                 elementIndex: widget.flow.widget.dashboard.elements.indexOf(widget.srcElement), | ||||||
|                 fromID: "${widget.srcElement.id}_${widget.index}", |                 fromID: "${widget.srcElement.id}_${widget.index}", | ||||||
|                 toID: "${widget.destElement.id}_${widget.index}", |                 toID: "${widget.destElement.id}_${widget.index}", | ||||||
| @@ -351,8 +358,8 @@ class DrawArrowState extends State<DrawArrow> { | |||||||
|               ); |               ); | ||||||
|             if ( widget.flow.widget.dashboard.arrows.where( |             if ( widget.flow.widget.dashboard.arrows.where( | ||||||
|                   (element) => element.fromID == "${widget.srcElement.id}_${widget.index}").isEmpty) { |                   (element) => element.fromID == "${widget.srcElement.id}_${widget.index}").isEmpty) { | ||||||
|                 widget.flow.widget.dashboard.addArrows(painter);  |                 widget.flow.widget.dashboard.addArrows(painter, context);  | ||||||
|                 widget.flow.widget.dashboard.saveDash(widget.flow.widget.dashboard.id); |                 widget.flow.widget.dashboard.saveDash(widget.flow.widget.dashboard.id, context); | ||||||
|             } else { |             } else { | ||||||
|               var i = widget.flow.widget.dashboard.arrows.indexWhere( |               var i = widget.flow.widget.dashboard.arrows.indexWhere( | ||||||
|                   (element) => element.fromID == "${widget.srcElement.id}_${widget.index}"); |                   (element) => element.fromID == "${widget.srcElement.id}_${widget.index}"); | ||||||
| @@ -390,7 +397,7 @@ class GraphParamsWidgetState extends State<GraphParamsWidget> { | |||||||
|       child: Row(children: [ |       child: Row(children: [ | ||||||
|         IconButton(onPressed: () { |         IconButton(onPressed: () { | ||||||
|           widget.comp.setState(() { |           widget.comp.setState(() { | ||||||
|             widget.comp.widget.dashboard.removeArrows((el) => el.fromID == "${widget.element.id}${widget.index}"); |             widget.comp.widget.dashboard.removeArrows((el) => el.fromID == "${widget.element.id}${widget.index}", context); | ||||||
|             widget.element.next.removeAt(widget.index); |             widget.element.next.removeAt(widget.index); | ||||||
|           }); |           }); | ||||||
|         }, icon: Icon(Icons.delete)) |         }, icon: Icon(Icons.delete)) | ||||||
| @@ -421,13 +428,13 @@ class ArrowInfoWidgetState extends State<ArrowInfoWidget> { | |||||||
| /// [ArrowParams.startArrowPosition] and | /// [ArrowParams.startArrowPosition] and | ||||||
| /// [ArrowParams.endArrowPosition] alignment. | /// [ArrowParams.endArrowPosition] alignment. | ||||||
| class ArrowPainter extends CustomPainter { | class ArrowPainter extends CustomPainter { | ||||||
|    |  | ||||||
|   /// |  | ||||||
|   ArrowPainter({ |   ArrowPainter({ | ||||||
|     this.elementIndex, |     this.elementIndex, | ||||||
|     this.connIndex, |     this.connIndex, | ||||||
|     this.toID = "", |     this.toID = "", | ||||||
|     this.fromID = "", |     this.fromID = "", | ||||||
|  |     this.infos = const [], | ||||||
|  |     this.env  = const [], | ||||||
|     this.isSelected = false, |     this.isSelected = false, | ||||||
|     required this.params, |     required this.params, | ||||||
|     this.from = Offset.zero, |     this.from = Offset.zero, | ||||||
| @@ -435,9 +442,9 @@ class ArrowPainter extends CustomPainter { | |||||||
|     List<Pivot>? pivots,  |     List<Pivot>? pivots,  | ||||||
|   }) : pivots = pivots ?? []; |   }) : pivots = pivots ?? []; | ||||||
|  |  | ||||||
|   /// |   List<dynamic> env; | ||||||
|  |   List<dynamic> infos; | ||||||
|   ArrowParams params; |   ArrowParams params; | ||||||
|   /// |  | ||||||
|   String toID; |   String toID; | ||||||
|   String fromID; |   String fromID; | ||||||
|   Offset to; |   Offset to; | ||||||
| @@ -462,6 +469,8 @@ class ArrowPainter extends CustomPainter { | |||||||
|   final arrowAngle=  25 * math.pi / 180; |   final arrowAngle=  25 * math.pi / 180; | ||||||
|  |  | ||||||
|   ArrowPainter deserialize(Map<String, dynamic> map) { |   ArrowPainter deserialize(Map<String, dynamic> map) { | ||||||
|  |     infos = map['infos']; | ||||||
|  |     env = map['env']; | ||||||
|     params = ArrowParams.fromMap(map['params']); |     params = ArrowParams.fromMap(map['params']); | ||||||
|     fromID = map['from']['id']; |     fromID = map['from']['id']; | ||||||
|     toID = map['to']['id']; |     toID = map['to']['id']; | ||||||
| @@ -469,9 +478,10 @@ class ArrowPainter extends CustomPainter { | |||||||
|     to = Offset(map['to']['x'], map['to']['y']); |     to = Offset(map['to']['x'], map['to']['y']); | ||||||
|     return this; |     return this; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Map<String, dynamic> serialize() { |   Map<String, dynamic> serialize() { | ||||||
|     Map<String, dynamic> graphElement = {}; |     Map<String, dynamic> graphElement = {}; | ||||||
|  |     graphElement['infos'] = infos; | ||||||
|  |     graphElement['env'] = env; | ||||||
|     graphElement['from'] = { "id" : fromID.split("_")[0], "x" : from.dx, "y" : from.dy }; |     graphElement['from'] = { "id" : fromID.split("_")[0], "x" : from.dx, "y" : from.dy }; | ||||||
|     graphElement['to'] = { "id" : toID.split("_")[0], "x" : to.dx, "y" : to.dy }; |     graphElement['to'] = { "id" : toID.split("_")[0], "x" : to.dx, "y" : to.dy }; | ||||||
|     graphElement['params'] = params.toMap(); |     graphElement['params'] = params.toMap(); | ||||||
| @@ -484,6 +494,8 @@ class ArrowPainter extends CustomPainter { | |||||||
|       elementIndex: map['elementIndex'] as int, |       elementIndex: map['elementIndex'] as int, | ||||||
|       toID: map['toID'] != null ? map['toID'] as String : "", |       toID: map['toID'] != null ? map['toID'] as String : "", | ||||||
|       fromID: map['fromID'] as String, |       fromID: map['fromID'] as String, | ||||||
|  |       infos: map['infos'] as List, | ||||||
|  |       env: map['env'] as List, | ||||||
|       isSelected: map['isSelected'] != null ? map['isSelected'] as bool : false, |       isSelected: map['isSelected'] != null ? map['isSelected'] as bool : false, | ||||||
|       params: ArrowParams.fromMap(map['params']), |       params: ArrowParams.fromMap(map['params']), | ||||||
|       from: Offset( |       from: Offset( | ||||||
| @@ -511,6 +523,8 @@ class ArrowPainter extends CustomPainter { | |||||||
|         "fromDy" : from.dy, |         "fromDy" : from.dy, | ||||||
|         "toDx" : to.dx, |         "toDx" : to.dx, | ||||||
|         "toDy" : to.dy, |         "toDy" : to.dy, | ||||||
|  |         "infos" : infos, | ||||||
|  |         "env" : env, | ||||||
|         'pivots': pivots.map((e) => e.toMap()).toList(), |         'pivots': pivots.map((e) => e.toMap()).toList(), | ||||||
|       }; |       }; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -151,12 +151,15 @@ class ElementWidgetState<T extends FlowData>  extends State<ElementWidget> { | |||||||
|               for (var element in widget.dashboard.elements) {  |               for (var element in widget.dashboard.elements) {  | ||||||
|                 element.isSelected = false;  |                 element.isSelected = false;  | ||||||
|                 element.dashboard.chartKey.currentState?. setState(() { }); |                 element.dashboard.chartKey.currentState?. setState(() { }); | ||||||
|  |                 widget.dashboard.selectedMenuKey.currentState?. setState(() { }); | ||||||
|               } |               } | ||||||
|             } |             } | ||||||
|             widget.element.isSelected = !widget.element.isSelected;  |             Future.delayed(Duration(milliseconds: 100), () {  | ||||||
|             for (var sel in widget.dashboard.arrows) { sel.isSelected = false; } |               DrawingArrow.instance.notifyListeners();  | ||||||
|             widget.dashboard.selectedMenuKey.currentState?. setState(() { }); |               widget.element.isSelected = !widget.element.isSelected;  | ||||||
|             Future.delayed(Duration(seconds: 1), () { DrawingArrow.instance.notifyListeners(); }); |               for (var sel in widget.dashboard.arrows) { sel.isSelected = false; } | ||||||
|  |               widget.dashboard.selectedMenuKey.currentState?. setState(() { }); | ||||||
|  |             }); | ||||||
|           }); |           }); | ||||||
|           widget.onElementPressed?.call(context, tapLocation); |           widget.onElementPressed?.call(context, tapLocation); | ||||||
|         }, |         }, | ||||||
| @@ -234,7 +237,7 @@ class ElementWidgetState<T extends FlowData>  extends State<ElementWidget> { | |||||||
|                   sel.changePosition(sel.position + diff); |                   sel.changePosition(sel.position + diff); | ||||||
|                 } |                 } | ||||||
|               } |               } | ||||||
|               widget.dashboard.save!(widget.dashboard.id); |               widget.dashboard.saveDash(widget.dashboard.id, context); | ||||||
|             }, |             }, | ||||||
|           ), |           ), | ||||||
|         ), |         ), | ||||||
| @@ -263,7 +266,7 @@ class ElementWidgetState<T extends FlowData>  extends State<ElementWidget> { | |||||||
|             child: Row( mainAxisAlignment: MainAxisAlignment.center,  |             child: Row( mainAxisAlignment: MainAxisAlignment.center,  | ||||||
|               children: (!widget.isHovered ? [] : [ |               children: (!widget.isHovered ? [] : [ | ||||||
|               IconButton(tooltip: "remove element", onPressed: () { |               IconButton(tooltip: "remove element", onPressed: () { | ||||||
|                 widget.dashboard.removeElement(widget.element); |                 widget.dashboard.removeElement(widget.element, context); | ||||||
|               }, icon: Icon(Icons.delete_outline)), |               }, icon: Icon(Icons.delete_outline)), | ||||||
|               IconButton(tooltip: "copy element", onPressed: () { |               IconButton(tooltip: "copy element", onPressed: () { | ||||||
|                 FlowElement<T> newElement = FlowElement<T>( |                 FlowElement<T> newElement = FlowElement<T>( | ||||||
| @@ -274,7 +277,7 @@ class ElementWidgetState<T extends FlowData>  extends State<ElementWidget> { | |||||||
|                   size: widget.element.size, |                   size: widget.element.size, | ||||||
|                   widget: widget.element.widget, |                   widget: widget.element.widget, | ||||||
|                 ); |                 ); | ||||||
|                 widget.dashboard.addElement(newElement); |                 widget.dashboard.addElement(newElement, context); | ||||||
|               }, icon: Icon(Icons.copy, size: 20)), |               }, icon: Icon(Icons.copy, size: 20)), | ||||||
|             ])) |             ])) | ||||||
|           ), |           ), | ||||||
|   | |||||||
							
								
								
									
										30
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -69,10 +69,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: collection |       name: collection | ||||||
|       sha256: a1ace0a119f20aabc852d165077c036cd864315bd99b7eaa10a60100341941bf |       sha256: ee67cb0715911d28db6bf4af1026078bd6f0128b07a5f66fb2ed94ec6783c09a | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.19.0" |     version: "1.18.0" | ||||||
|   convert: |   convert: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -411,18 +411,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: leak_tracker |       name: leak_tracker | ||||||
|       sha256: "7bb2830ebd849694d1ec25bf1f44582d6ac531a57a365a803a6034ff751d2d06" |       sha256: "3f87a60e8c63aecc975dda1ceedbc8f24de75f09e4856ea27daf8958f2f0ce05" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "10.0.7" |     version: "10.0.5" | ||||||
|   leak_tracker_flutter_testing: |   leak_tracker_flutter_testing: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: leak_tracker_flutter_testing |       name: leak_tracker_flutter_testing | ||||||
|       sha256: "9491a714cca3667b60b5c420da8217e6de0d1ba7a5ec322fab01758f6998f379" |       sha256: "932549fb305594d82d7183ecd9fa93463e9914e1b67cacc34bc40906594a1806" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.8" |     version: "3.0.5" | ||||||
|   leak_tracker_testing: |   leak_tracker_testing: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -707,7 +707,7 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: flutter |     description: flutter | ||||||
|     source: sdk |     source: sdk | ||||||
|     version: "0.0.0" |     version: "0.0.99" | ||||||
|   source_span: |   source_span: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -728,10 +728,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: stack_trace |       name: stack_trace | ||||||
|       sha256: "9f47fd3630d76be3ab26f0ee06d213679aa425996925ff3feffdec504931c377" |       sha256: "73713990125a6d93122541237550ee3352a2d84baad52d375a4cad2eb9b7ce0b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.12.0" |     version: "1.11.1" | ||||||
|   star_menu: |   star_menu: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -752,10 +752,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: string_scanner |       name: string_scanner | ||||||
|       sha256: "688af5ed3402a4bde5b3a6c15fd768dbf2621a614950b17f04626c431ab3c4c3" |       sha256: "556692adab6cfa87322a115640c11f13cb77b3f076ddcc5d6ae3c20242bedcde" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.3.0" |     version: "1.2.0" | ||||||
|   super_clipboard: |   super_clipboard: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -824,10 +824,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: test_api |       name: test_api | ||||||
|       sha256: "664d3a9a64782fcdeb83ce9c6b39e78fd2971d4e37827b9b06c3aa1edc5e760c" |       sha256: "5b8a98dafc4d5c4c9c72d8b31ab2b23fc13422348d2997120294d3bac86b4ddb" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.7.3" |     version: "0.7.2" | ||||||
|   timezone: |   timezone: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -896,10 +896,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: vm_service |       name: vm_service | ||||||
|       sha256: f6be3ed8bd01289b34d679c2b62226f63c0e69f9fd2e50a6b3c1c729a961041b |       sha256: "5c5f338a667b4c644744b661f309fb8080bb94b18a7e91ef1dbd343bed00ed6d" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "14.3.0" |     version: "14.2.5" | ||||||
|   web: |   web: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user