2022-06-27 18:52:48 +02:00
class CCPMethodEditorController extends HTMLElement {
# boot ;
# rootdoc ;
2022-07-18 18:08:53 +02:00
# serviceurl ;
2023-03-29 12:32:20 +02:00
# infrastructures ;
2022-06-27 18:52:48 +02:00
# locked = false ;
2022-07-18 18:08:53 +02:00
# isupdate = false ;
2022-09-08 16:03:28 +02:00
# cloneornew _dialog = null ;
# dragging _method = null ;
2022-07-18 18:08:53 +02:00
2022-06-27 18:52:48 +02:00
# tmp _inputs = [ ]
# tmp _outputs = [ ]
# current = null ;
# method _template = {
id : "" ,
title : "New Method" ,
description : "New empty method" ,
version : "1.0.0" ,
jobControlOptions : "async-execute" ,
2022-07-21 15:42:55 +02:00
metadata : [ ] ,
2022-06-27 18:52:48 +02:00
inputs : { } ,
outputs : { } ,
additionalParameters : {
parameters : [
{
name : "deploy-script" ,
2022-09-08 16:03:28 +02:00
value : [ ]
2022-06-27 18:52:48 +02:00
} ,
{
name : "execute-script" ,
2022-09-08 16:03:28 +02:00
value : [ ]
2022-06-27 18:52:48 +02:00
} ,
{
name : "undeploy-script" ,
2022-09-08 16:03:28 +02:00
value : [ ]
2022-06-27 18:52:48 +02:00
}
]
} ,
links : [ ]
}
2022-09-08 16:03:28 +02:00
# scripts = `
< link rel = "canonical" href = "https://getbootstrap.com/docs/5.0/components/modal/" >
`
2022-06-27 18:52:48 +02:00
# style = `
< link rel = "stylesheet" href = "https://cdn.dev.d4science.org/ccp/css/common.css" > < / l i n k >
2022-09-08 16:03:28 +02:00
< link href = "https://cdn.jsdelivr.net/npm/bootstrap@5.0.2/dist/css/bootstrap.min.css" rel = "stylesheet" integrity = "sha384-EVSTQN3/azprG1Anm3QDgpJLIm9Nao0Yz1ztcQTwFspd3yD65VohhpuuCOmLASjC" crossorigin = "anonymous" >
2022-06-27 18:52:48 +02:00
< style >
. ccp - method - editor {
position : relative ;
}
. ccp - option {
background - color : # eeffff ;
color : # 0099 CC ;
border : solid 1 px # 0099 CC ;
2023-03-29 12:32:20 +02:00
display : inline - flex ;
flex - direction : row ;
justify - content : space - between ;
align - items : center ;
gap : 1 rem ;
font - size : 1 rem ;
2022-06-27 18:52:48 +02:00
}
2023-03-29 12:32:20 +02:00
. ccp - option > . btn {
padding : 0 ;
2022-06-27 18:52:48 +02:00
}
2022-07-21 15:42:55 +02:00
2022-09-13 15:40:17 +02:00
ul . author _list , ul . context _list {
2022-07-21 15:42:55 +02:00
list - style : none ;
display : flex ;
flex - direction : row ;
gap : 2 px ;
font - size : small ;
font - weight : 300 ;
}
li . author _list _item {
display : inline - block ;
padding : 0.25 em 0.4 em ;
line - height : 1 ;
text - align : center ;
white - space : nowrap ;
vertical - align : baseline ;
border - radius : 0.25 rem ;
background - color : # eeffff ;
color : # 9900 CC ;
border : solid 1 px # 9900 CC ;
}
2022-09-13 15:40:17 +02:00
li . context _list _item {
display : inline - block ;
padding : 0.25 em 0.4 em ;
line - height : 1 ;
text - align : center ;
white - space : nowrap ;
vertical - align : baseline ;
border - radius : 0.25 rem ;
background - color : # eeffff ;
color : # 0099 CC ;
border : solid 1 px # 0099 CC ;
}
2023-04-06 16:41:10 +02:00
li . category _list _item {
display : inline - block ;
padding : 0.25 em 0.4 em ;
line - height : 1 ;
text - align : center ;
white - space : nowrap ;
vertical - align : baseline ;
border - radius : 0.25 rem ;
background - color : # eeffff ;
color : # 0099 CC ;
border : solid 1 px # 0099 CC ;
}
2022-06-27 18:52:48 +02:00
< / s t y l e >
`
# erase _icon = `
< svg viewBox = "0 0 24 24" >
< path d = "M16.24,3.56L21.19,8.5C21.97,9.29 21.97,10.55 21.19,11.34L12,20.53C10.44,22.09 7.91,22.09 6.34,20.53L2.81,17C2.03,16.21 2.03,14.95 2.81,14.16L13.41,3.56C14.2,2.78 15.46,2.78 16.24,3.56M4.22,15.58L7.76,19.11C8.54,19.9 9.8,19.9 10.59,19.11L14.12,15.58L9.17,10.63L4.22,15.58Z" / >
`
# plus _icon = `
< svg viewBox = "0 0 24 24" >
< path d = "M19,13H13V19H11V13H5V11H11V5H13V11H19V13Z" / >
< / s v g >
`
# disc _icon = `
< svg viewBox = "0 0 24 24" >
< path d = "M15,9H5V5H15M12,19A3,3 0 0,1 9,16A3,3 0 0,1 12,13A3,3 0 0,1 15,16A3,3 0 0,1 12,19M17,3H5C3.89,3 3,3.9 3,5V19A2,2 0 0,0 5,21H19A2,2 0 0,0 21,19V7L17,3Z" / >
< / s v g >
`
# output _icon = `
< svg viewBox = "0 0 24 24" >
< path d = "M23,12L19,8V11H10V13H19V16M1,18V6C1,4.89 1.9,4 3,4H15A2,2 0 0,1 17,6V9H15V6H3V18H15V15H17V18A2,2 0 0,1 15,20H3A2,2 0 0,1 1,18Z" / >
< / s v g >
`
2022-07-18 18:08:53 +02:00
# delete _icon = `
2023-02-09 12:21:39 +01:00
< svg viewBox = "0 0 24 24" >
2022-07-18 18:08:53 +02:00
< path d = "M19,4H15.5L14.5,3H9.5L8.5,4H5V6H19M6,19A2,2 0 0,0 8,21H16A2,2 0 0,0 18,19V7H6V19Z" / >
< / s v g >
`
2023-05-24 14:49:40 +02:00
# annotation _input _icon = `
< svg viewBox = "0 -960 960 960" >
< path d = "M450-234h60v-129h130v-60H510v-130h-60v130H320v60h130v129ZM220-80q-24 0-42-18t-18-42v-680q0-24 18-42t42-18h361l219 219v521q0 24-18 42t-42 18H220Zm331-554v-186H220v680h520v-494H551ZM220-820v186-186 680-680Z" / >
< / s v g >
`
2022-06-27 18:52:48 +02:00
constructor ( ) {
super ( ) ;
this . # boot = document . querySelector ( "d4s-boot-2" )
this . # rootdoc = this . attachShadow ( { "mode" : "open" } )
2022-07-18 18:08:53 +02:00
this . # serviceurl = this . getAttribute ( "serviceurl" )
this . initMethod ( )
2023-03-29 12:32:20 +02:00
this . fetchInfrastructures ( )
this . connectNewEditRequest ( )
2022-06-27 18:52:48 +02:00
}
connectedCallback ( ) {
}
2023-03-29 12:32:20 +02:00
connectNewEditRequest ( ) {
document . addEventListener ( "neweditrequest" , ev => {
this . cloneOrEditMethod ( ev . detail )
} )
}
2022-06-27 18:52:48 +02:00
2023-03-29 12:32:20 +02:00
isInfrastructureRegistered ( infra ) {
return ( this . # infrastructures . filter ( i => i . id === infra . id ) ) . length > 0
2022-09-08 16:03:28 +02:00
}
2023-03-29 12:32:20 +02:00
fetchInfrastructures ( ) {
this . # boot . secureFetch ( this . # serviceurl + "/infrastructures" ) .
2022-06-27 18:52:48 +02:00
then ( resp => {
2023-03-29 12:32:20 +02:00
if ( resp . status !== 200 ) throw "Unable to fetch infrastructures" ;
2022-06-27 18:52:48 +02:00
return resp . json ( )
} ) . then ( data => {
2023-03-29 12:32:20 +02:00
this . # infrastructures = data
2022-06-27 18:52:48 +02:00
this . render ( )
} ) . catch ( err => {
alert ( err )
2022-07-21 12:35:07 +02:00
} )
2022-06-27 18:52:48 +02:00
}
2022-07-18 18:08:53 +02:00
saveMethod ( ) {
if ( this . # locked ) return ;
2022-06-27 18:52:48 +02:00
if ( this . # current != null ) {
this . adoptTemporaries ( )
2023-05-23 19:55:04 +02:00
const text = ` Please confirm ${ this . # isupdate ? "updating" : "creation" } of ${ this . # current . title } version ${ this . # current . version } `
2022-07-18 18:08:53 +02:00
if ( window . confirm ( text ) ) {
this . lockRender ( )
const url = this . # serviceurl + "/methods"
const args = {
body : JSON . stringify ( this . # current ) ,
method : this . # isupdate ? "PUT" : "POST" ,
headers : { "Content-type" : "application/json" }
}
this . # boot . secureFetch ( url , args ) . then (
( resp ) => {
if ( resp . status === 201 || resp . status === 204 ) {
2023-06-21 15:42:11 +02:00
return resp . text ( )
2022-07-21 15:42:55 +02:00
} else throw "Error saving process: " + resp . status
2022-07-18 18:08:53 +02:00
} ) . then ( data => {
2023-02-06 14:05:08 +01:00
if ( ! this . # isupdate ) {
2023-06-21 15:46:34 +02:00
this . # current = JSON . parse ( data )
2023-06-21 15:27:42 +02:00
this . # isupdate = true
this . # tmp _inputs = Object . keys ( this . # current . inputs ) . map ( k => this . # current . inputs [ k ] )
this . # tmp _outputs = Object . keys ( this . # current . outputs ) . map ( k => this . # current . outputs [ k ] )
2023-02-06 14:05:08 +01:00
}
2023-06-21 15:27:42 +02:00
this . unlockRender ( )
2022-07-18 18:08:53 +02:00
} ) . catch ( err => {
alert ( err )
this . unlockRender ( )
} )
}
}
}
deleteMethod ( ) {
if ( this . # locked ) return ;
if ( this . # current != null ) {
2023-05-23 19:55:04 +02:00
const text = ` Please confirm deletion of ${ this . # current . title } version ${ this . # current . version } `
2022-07-18 18:08:53 +02:00
if ( window . confirm ( text ) ) {
this . lockRender ( )
const url = this . # serviceurl + "/methods/" + this . # current . id
const args = {
method : "DELETE"
}
this . # boot . secureFetch ( url , args ) . then (
( resp ) => {
if ( resp . status === 404 || resp . status === 204 ) {
return null
2022-07-21 12:35:07 +02:00
} else throw "Error deleting method: " + resp . status
2022-07-18 18:08:53 +02:00
} ) . then ( data => {
2023-06-21 15:44:31 +02:00
this . # isupdate = false
this . initMethod ( )
2022-07-18 18:08:53 +02:00
this . unlockRender ( )
} ) . catch ( err => {
alert ( err )
this . unlockRender ( )
} )
}
2022-06-27 18:52:48 +02:00
}
}
2022-07-18 18:08:53 +02:00
initMethod ( ) {
this . # current = JSON . parse ( JSON . stringify ( this . # method _template ) )
this . # current . id = Math . abs ( ( Math . random ( ) * 10e11 ) | 0 )
2022-07-21 12:35:07 +02:00
this . # current . metadata = [ ]
2023-03-29 12:32:20 +02:00
this . # tmp _inputs = [
{
id : "ccpimage" ,
title : "Runtime" ,
description : "The image of the runtime to use for method execution. This depends on the infrastructure specific protocol for interacting with registries." ,
minOccurs : 1 ,
maxOccurs : 1 ,
schema : {
type : "string" ,
format : "url" ,
contentMediaType : "text/plain" ,
default : "" ,
readonly : true ,
}
}
]
2022-07-18 18:08:53 +02:00
this . # tmp _outputs = [ ]
}
resetMethod ( ) {
this . initMethod ( )
2022-07-21 12:35:07 +02:00
this . render ( )
2022-07-18 18:08:53 +02:00
}
2022-07-21 18:58:14 +02:00
cloneOrEditMethod ( method ) {
const subject = this . # boot . subject
const matchingauthors = method . metadata ? method . metadata . filter ( md => md . role === "author" && md . href . endsWith ( "/" + subject ) ) . length : 0
if ( matchingauthors === 0 ) this . cloneMethod ( method . id ) ;
2022-09-08 16:03:28 +02:00
else {
this . # dragging _method = method . id
this . # cloneornew _dialog . style . display = "block"
this . # cloneornew _dialog . classList . add ( "show" )
}
2022-07-21 18:58:14 +02:00
}
2022-06-27 18:52:48 +02:00
cloneMethod ( method ) {
if ( this . # locked ) return ;
this . lockRender ( )
2022-07-21 15:42:55 +02:00
this . # boot . secureFetch ( this . # serviceurl + "/methods/" + method + "/clone" ) . then (
2022-06-27 18:52:48 +02:00
( resp ) => {
if ( resp . status === 200 ) {
return resp . json ( )
2022-07-21 15:42:55 +02:00
} else throw "Error retrieving process: " + resp . status
2022-06-27 18:52:48 +02:00
}
) . then ( data => {
this . # current = data
2022-07-18 18:08:53 +02:00
this . # isupdate = false
this . # tmp _inputs = Object . keys ( this . # current . inputs ) . map ( k => this . # current . inputs [ k ] )
this . # tmp _outputs = Object . keys ( this . # current . outputs ) . map ( k => this . # current . outputs [ k ] )
this . unlockRender ( )
} ) . catch ( err => {
this . unlockRender ( )
} )
}
editMethod ( method ) {
if ( this . # locked ) return ;
this . lockRender ( )
2022-07-21 15:42:55 +02:00
this . # boot . secureFetch ( this . # serviceurl + "/methods/" + method + "/updatable" ) . then (
2022-07-18 18:08:53 +02:00
( resp ) => {
if ( resp . status === 200 ) {
return resp . json ( )
2022-07-21 15:42:55 +02:00
} else throw "Error retrieving process: " + resp . status
2022-07-18 18:08:53 +02:00
}
) . then ( data => {
this . # current = data
this . # isupdate = true
2022-06-27 18:52:48 +02:00
this . # tmp _inputs = Object . keys ( this . # current . inputs ) . map ( k => this . # current . inputs [ k ] )
this . # tmp _outputs = Object . keys ( this . # current . outputs ) . map ( k => this . # current . outputs [ k ] )
this . unlockRender ( )
} ) . catch ( err => {
this . unlockRender ( )
} )
}
adoptTemporaries ( ) {
this . # current . inputs = { }
this . # tmp _inputs . forEach ( t => {
this . # current . inputs [ t . id ] = t
} )
this . # current . outputs = { }
this . # tmp _outputs . forEach ( t => {
this . # current . outputs [ t . id ] = t
} )
}
getInputAt ( i ) {
return this . # tmp _inputs [ i ]
}
deleteTmpInputAt ( i ) {
this . # tmp _inputs . splice ( i , 1 )
}
deleteTmpOutputAt ( i ) {
this . # tmp _outputs . splice ( i , 1 )
}
lockRender ( ) {
this . # locked = true
this . # rootdoc . querySelector ( ".plexiglass" ) . classList . toggle ( "d-none" )
}
unlockRender ( ) {
this . # rootdoc . querySelector ( ".plexiglass" ) . classList . toggle ( "d-none" )
this . render ( )
this . # locked = false
2023-06-21 15:27:42 +02:00
if ( this . parentElement ) this . parentElement . scrollIntoViewIfNeeded ( ) ;
2022-06-27 18:52:48 +02:00
}
render ( ) {
this . # rootdoc . innerHTML = `
2022-09-08 16:03:28 +02:00
<!-- Modal -- >
< div class = "modal fade" style = "background-color:rgba(0,0,0,0.3)" id = "cloneornew" tabindex = "-1" role = "dialog" >
< div class = "modal-dialog" role = "document" >
< div class = "modal-content shadow-lg border-primary" >
< div class = "modal-body" >
Choose whether you want to clone this method or edit it .
< / d i v >
< div class = "modal-footer" >
< button name = "clone" type = "button" class = "btn btn-info" > Clone < / b u t t o n >
< button name = "edit" type = "button" class = "btn btn-primary" > Edit < / b u t t o n >
< / d i v >
< / d i v >
< / d i v >
< / d i v >
2022-06-27 18:52:48 +02:00
< div class = "ccp-method-editor" >
< div class = "d-none plexiglass" >
< svg class = "spinner" viewBox = "0 0 66 66" >
< circle class = "path" fill = "none" stroke - width = "6" stroke - linecap = "round" cx = "33" cy = "33" r = "30" > < / c i r c l e >
< / s v g >
< / d i v >
$ { this . # style }
< div class = "ccp-method-editor-form" >
< div class = "card" >
< div class = "card-header" >
2022-07-18 18:08:53 +02:00
< div class = "ccp-toolbar-header" >
< div >
< span name = "header" class = "mr-2" > $ { this . # current . title } < / s p a n >
< / d i v >
< div class = "ccp-toolbar-right" >
$ { this . renderSaveButton ( ) }
$ { this . renderResetButton ( ) }
$ { this . # isupdate ? this . renderDeleteButton ( ) : "" }
< / d i v >
2022-06-27 18:52:48 +02:00
< / d i v >
< / d i v >
< div class = "card-body" >
2022-07-21 15:42:55 +02:00
< div class = "mb-3 row" >
$ { this . renderAuthors ( ) }
< / d i v >
2023-04-06 15:31:42 +02:00
<!-- div class = "mb-3 row" >
2022-09-13 15:40:17 +02:00
$ { this . renderContexts ( ) }
2023-04-06 15:31:42 +02:00
< / d i v - - >
2022-06-27 18:52:48 +02:00
< div class = "mb-3 row" >
< div class = "col" >
< label class = "form-label" > Title < / l a b e l >
< input name = "title" class = "form-control" type = "text" required = "required" value = "${this.#current.title}" / >
< / d i v >
< div class = "col" >
< label class = "form-label" > Version < / l a b e l >
< input name = "version" class = "form-control" type = "text" required = "required" value = "${this.#current.version}" / >
< / d i v >
< / d i v >
< div class = "mb-3" >
< label class = "form-label" > Description < / l a b e l >
< input name = "description" class = "form-control" type = "text" required = "required" value = "${this.#current.description}" / >
< / d i v >
2023-04-06 16:27:14 +02:00
< div class = "row" >
< div class = "col mb-3" >
< label class = "form-label" > Keywords < / l a b e l >
< input name = "keyword-input" class = "form-control" type = "text" / >
< div name = "keyword-list" class = "form-text" >
$ { this . renderKeywords ( ) }
< / d i v >
< / d i v >
< div class = "col mb-3" >
2023-04-06 16:30:44 +02:00
< label class = "form-label" > Categories < / l a b e l >
2023-04-06 16:43:46 +02:00
< input list = "categoryhints" name = "category-input" class = "form-control" type = "text" / >
2023-04-06 16:30:44 +02:00
$ { this . renderCategoryHints ( ) }
2023-04-06 16:27:14 +02:00
< div name = "category-list" class = "form-text" >
$ { this . renderCategories ( ) }
< / d i v >
2022-06-27 18:52:48 +02:00
< / d i v >
< / d i v >
2023-03-29 12:32:20 +02:00
< div name = "infrastructures" class = "mb-3" >
< label class = "form-label" > Compatible Infrastrucures < / l a b e l >
< select name = "infrastructure-input" class = "form-control" >
$ { this . renderInfrastructureOptions ( ) }
2022-06-27 18:52:48 +02:00
< / s e l e c t >
2023-03-29 12:32:20 +02:00
< div name = "infrastructure-list" class = "form-text" >
$ { this . renderInfrastructures ( ) }
2022-06-27 18:52:48 +02:00
< / d i v >
< / d i v >
< / d i v >
< / d i v >
< details class = "card" >
< summary class = "card-header" >
2022-07-18 18:08:53 +02:00
< div class = "ccp-toolbar-header" >
< div >
< span class = "mr-2" > Inputs < / s p a n >
< / d i v >
< div class = "ccp-toolbar-right" >
2023-05-24 14:49:40 +02:00
$ { this . renderStandardInputButtons ( ) }
2022-07-18 18:08:53 +02:00
$ { this . renderPlusButton ( "add-input" ) }
< / d i v >
< / d i v >
2022-06-27 18:52:48 +02:00
< / s u m m a r y >
< div class = "card-body" name = "input-list" >
< / d i v >
< / d e t a i l s >
< details class = "card" >
< summary class = "card-header" name = "output-buttons" >
2022-07-18 18:08:53 +02:00
< div class = "ccp-toolbar-header" >
< div >
< span class = "mr-2" > Outputs < / s p a n >
< / d i v >
< div class = "ccp-toolbar-right" >
$ { this . renderStandardOutputButtons ( ) }
$ { this . renderPlusButton ( "add-output" ) }
< / d i v >
< / d i v >
2022-06-27 18:52:48 +02:00
< / s u m m a r y >
< div class = "card-body" name = "output-list" >
< d4s - ccp - output - editor > < / d 4 s - c c p - o u t p u t - e d i t o r >
< / d i v >
< / d i v >
< details class = "card" name = "script-list" >
< summary class = "card-header" >
2022-07-18 18:08:53 +02:00
< div class = "ccp-toolbar-header" >
< div >
< span > Scripts < / s p a n >
< select name = "script-selector" style = "border:none; padding:.3rem" >
< option value = "deploy-script" > Deploy < / o p t i o n >
< option value = "execute-script" > Execute < / o p t i o n >
< option value = "undeploy-script" > Undeploy < / o p t i o n >
< / s e l e c t >
< / d i v >
< / d i v >
2022-06-27 18:52:48 +02:00
< / s u m m a r y >
< div class = "card-body" name = "input_container" >
$ { this . renderScripts ( ) }
< / d i v >
< / d i v >
< / d i v >
< / d i v >
`
this . renderInputs ( )
this . renderOutputs ( )
2022-09-08 16:03:28 +02:00
this . # cloneornew _dialog = this . # rootdoc . querySelector ( "#cloneornew" )
this . # rootdoc . querySelector ( "#cloneornew button[name=clone]" ) . addEventListener ( "click" , ev => {
this . # cloneornew _dialog . classList . remove ( "show" )
this . # cloneornew _dialog . style . display = "none"
this . cloneMethod ( this . # dragging _method )
this . # dragging _method = null
} )
this . # rootdoc . querySelector ( "#cloneornew button[name=edit]" ) . addEventListener ( "click" , ev => {
this . # cloneornew _dialog . classList . remove ( "show" )
this . # cloneornew _dialog . style . display = "none"
this . editMethod ( this . # dragging _method )
this . # dragging _method = null
} )
2022-07-18 18:08:53 +02:00
this . # rootdoc . querySelector ( "input[name=title]" ) . addEventListener ( "input" , ev => {
this . # current . title = ev . currentTarget . value
this . # rootdoc . querySelector ( "span[name=header]" ) . innerText = this . # current . title
} )
this . # rootdoc . querySelector ( "input[name=version]" ) . addEventListener ( "input" , ev => {
this . # current . version = ev . currentTarget . value
} )
this . # rootdoc . querySelector ( "input[name=description]" ) . addEventListener ( "input" , ev => {
this . # current . description = ev . currentTarget . value
} )
2022-06-27 18:52:48 +02:00
this . # rootdoc . addEventListener ( "drop" , ev => {
2023-02-09 13:22:38 +01:00
if ( ev . dataTransfer && ev . dataTransfer . getData ( 'text/plain+ccpmethod' ) ) {
const method = JSON . parse ( ev . dataTransfer . getData ( 'application/json+ccpmethod' ) )
ev . stopImmediatePropagation ( )
ev . preventDefault ( )
ev . stopPropagation ( )
this . cloneOrEditMethod ( method )
}
2022-06-27 18:52:48 +02:00
} )
this . # rootdoc . addEventListener ( "dragover" , ev => {
ev . preventDefault ( )
} )
this . # rootdoc . querySelector ( "button[name=reset]" ) . addEventListener ( "click" , ev => {
2022-07-18 18:08:53 +02:00
if ( window . confirm ( "All unsaved data will be lost. Proceed?" ) ) {
this . resetMethod ( )
}
2022-06-27 18:52:48 +02:00
ev . preventDefault ( )
ev . stopPropagation ( )
} )
this . # rootdoc . querySelector ( "button[name=save]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
2022-07-18 18:08:53 +02:00
this . saveMethod ( )
2022-06-27 18:52:48 +02:00
} )
2022-07-18 18:08:53 +02:00
if ( this . # isupdate ) {
this . # rootdoc . querySelector ( "button[name=delete]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
this . deleteMethod ( )
} )
}
2022-06-27 18:52:48 +02:00
this . # rootdoc . querySelector ( "input[name=keyword-input]" ) . addEventListener ( "keypress" , ev => {
if ( ev . key === "Enter" || ev . which === 13 ) {
ev . preventDefault ( )
ev . stopPropagation ( )
const val = ev . target . value
ev . target . value = null
if ( ! this . # current . keywords ) {
this . # current . keywords = [ val ]
} else {
this . # current . keywords . push ( val )
}
this . reRenderKeywords ( )
}
} )
this . # rootdoc . querySelector ( "div[name=keyword-list]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
if ( ev . target . getAttribute ( 'name' ) === "delete-keyword" ) {
const index = ev . target . getAttribute ( "data-index" )
this . # current . keywords . splice ( index , 1 )
this . reRenderKeywords ( )
this . # rootdoc . querySelector ( "input[name=keyword-input]" ) . focus ( )
}
} )
2023-04-06 16:41:10 +02:00
this . # rootdoc . querySelector ( "input[name=category-input]" ) . addEventListener ( "keypress" , ev => {
if ( ev . key === "Enter" || ev . which === 13 ) {
ev . preventDefault ( )
ev . stopPropagation ( )
const val = ev . target . value
ev . target . value = null
2023-04-06 17:12:52 +02:00
if ( this . # current . metadata . filter ( md => md . role === "category" && md . title === val ) . length === 0 ) {
2023-04-06 16:41:10 +02:00
this . # current . metadata . push ( { role : "category" , title : val } )
2023-04-06 16:47:32 +02:00
this . reRenderCategories ( )
2023-04-06 16:41:10 +02:00
}
}
} )
2023-04-06 16:43:46 +02:00
this . # rootdoc . querySelector ( "div[name=category-list]" ) . addEventListener ( "click" , ev => {
2023-04-06 16:41:10 +02:00
ev . preventDefault ( )
ev . stopPropagation ( )
if ( ev . target . getAttribute ( 'name' ) === "delete-category" ) {
2023-04-06 17:25:33 +02:00
const val = ev . target . parentElement . title
this . # current . metadata = this . # current . metadata . filter ( md => md . role !== "category" || md . title !== val )
2023-04-06 16:41:10 +02:00
this . reRenderCategories ( )
this . # rootdoc . querySelector ( "input[name=category-input]" ) . focus ( )
}
} )
2023-03-29 12:32:20 +02:00
this . # rootdoc . querySelector ( "div[name=infrastructures]" ) . addEventListener ( "change" , ev => {
if ( ev . target . getAttribute ( "name" ) === "infrastructure-input" && ev . target . value ) {
2022-06-27 18:52:48 +02:00
ev . preventDefault ( )
ev . stopPropagation ( )
const id = ev . target . value
const display = ev . target . options [ ev . target . selectedIndex ] . text
let link = {
2023-03-29 12:32:20 +02:00
rel : "compatibleWith" , title : display , href : "infrastructures/" + id
2022-06-27 18:52:48 +02:00
}
if ( ! this . # current . links ) {
this . # current . links = [ link ]
} else {
this . # current . links . push ( link )
}
2023-03-29 12:32:20 +02:00
this . reRenderInfrastructures ( ev . currentTarget )
2022-06-27 18:52:48 +02:00
}
} )
2023-03-29 12:32:20 +02:00
this . # rootdoc . querySelector ( "div[name=infrastructure-list]" ) . addEventListener ( "click" , ev => {
2022-06-27 18:52:48 +02:00
ev . preventDefault ( )
ev . stopPropagation ( )
2023-03-29 12:32:20 +02:00
if ( ev . target . getAttribute ( 'name' ) === "delete-infrastructure" ) {
2022-06-27 18:52:48 +02:00
const index = ev . target . getAttribute ( "data-index" )
this . # current . links . splice ( index , 1 )
2023-03-29 12:32:20 +02:00
this . reRenderInfrastructures ( )
2022-06-27 18:52:48 +02:00
}
} )
this . # rootdoc . querySelector ( "button[name=add-input]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
this . # tmp _inputs . push (
{
id : "new_input" ,
title : "New input" ,
description : "A new input field" ,
minOccurs : 1 ,
maxOccurs : 1 ,
schema : {
type : "string" ,
format : null ,
2022-07-18 18:08:53 +02:00
contentMediaType : "text/plain" ,
2022-06-27 18:52:48 +02:00
default : ""
}
}
)
this . renderInputs ( )
} )
2023-05-24 14:49:40 +02:00
this . # rootdoc . querySelector ( "button[name=add-ccpannotation]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
this . # tmp _inputs . push (
{
2023-05-25 09:10:12 +02:00
id : "ccpnote" ,
2023-05-24 14:56:56 +02:00
title : "Annotations for execution" ,
description : "The value of this parameter will be associated as annotation to the execution" ,
2023-05-24 14:49:40 +02:00
minOccurs : 1 ,
maxOccurs : 1 ,
schema : {
type : "string" ,
format : null ,
contentMediaType : "text/plain" ,
default : ""
}
}
)
this . renderInputs ( )
} )
2022-06-27 18:52:48 +02:00
this . # rootdoc . querySelector ( "div[name=input-list]" ) . addEventListener ( "click" , ev => {
const evname = ev . target . getAttribute ( 'name' )
if ( evname === "delete-input" ) {
ev . preventDefault ( )
ev . stopPropagation ( )
const index = Number ( ev . target . getAttribute ( "data-index" ) )
console . log ( "deleting input at index" , index )
this . deleteTmpInputAt ( index )
this . renderInputs ( )
}
} )
this . # rootdoc . querySelector ( "button[name=add-output]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
this . # tmp _outputs . push (
{
id : "new_output" ,
title : "New ouput" ,
description : "A new output field" ,
minOccurs : 1 ,
maxOccurs : 1 ,
2022-09-13 15:12:26 +02:00
metadata : [
{
"role" : "file" ,
"title" : "newoutput.txt" ,
"href" : "newoutput.txt"
}
] ,
2022-06-27 18:52:48 +02:00
schema : {
type : "string" ,
2022-07-18 18:08:53 +02:00
contentMediaType : "text/plain"
2022-06-27 18:52:48 +02:00
}
}
)
this . renderOutputs ( )
} )
this . # rootdoc . querySelector ( "button[name=add-stdout]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
this . # tmp _outputs . push (
{
id : "stdout" ,
title : "Standard output" ,
description : "Standard output channel" ,
minOccurs : 1 ,
maxOccurs : 1 ,
2022-09-13 15:12:26 +02:00
metadata : [
{
"role" : "file" ,
"title" : "stdout" ,
"href" : "stdout"
}
] ,
2022-06-27 18:52:48 +02:00
schema : {
type : "string" ,
contentMediaType : "text/plain"
}
}
)
this . renderOutputs ( )
} )
this . # rootdoc . querySelector ( "button[name=add-stderr]" ) . addEventListener ( "click" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
this . # tmp _outputs . push (
{
id : "stderr" ,
title : "Standard error" ,
description : "Standard error channel" ,
minOccurs : 1 ,
maxOccurs : 1 ,
2022-09-13 15:12:26 +02:00
metadata : [
{
"role" : "file" ,
"title" : "stderr" ,
"href" : "stderr"
}
] ,
2022-06-27 18:52:48 +02:00
schema : {
type : "string" ,
contentMediaType : "text/plain"
}
}
)
this . renderOutputs ( )
} )
this . # rootdoc . querySelector ( "div[name=output-list]" ) . addEventListener ( "click" , ev => {
const evname = ev . target . getAttribute ( 'name' )
if ( evname === "delete-output" ) {
ev . preventDefault ( )
ev . stopPropagation ( )
const index = Number ( ev . target . getAttribute ( "data-index" ) )
console . log ( "deleting output at index" , index )
this . deleteTmpOutputAt ( index )
this . renderOutputs ( )
}
} )
this . # rootdoc . querySelector ( "select[name=script-selector]" ) . addEventListener ( "change" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
const scriptname = ev . target . value
const areas = Array . prototype . slice . call ( this . # rootdoc . querySelectorAll ( "textarea.script-area" ) )
areas . forEach ( a => {
if ( a . getAttribute ( "name" ) === scriptname ) a . classList . remove ( "d-none" ) ;
else a . classList . add ( "d-none" )
} )
} )
this . # rootdoc . querySelector ( "details[name=script-list]" ) . addEventListener ( "input" , ev => {
ev . preventDefault ( )
ev . stopPropagation ( )
const scriptname = ev . target . getAttribute ( "name" )
const script = this . # current . additionalParameters . parameters . filter ( p => p . name === scriptname ) [ 0 ]
2022-09-08 16:03:28 +02:00
if ( script ) script . value = ev . target . value . split ( /\r?\n/ ) ;
2022-06-27 18:52:48 +02:00
} )
}
2022-07-21 15:42:55 +02:00
renderAuthors ( ) {
return `
< ul class = "author_list" >
$ { this . # current . metadata .
filter ( md => md . role === "author" ) .
2023-04-03 15:31:14 +02:00
map ( a => ` <li class="badge bg-warning text-dark" title="Author: ${ a . title } "> ${ a . title } </li> ` ) . join ( ) }
2022-09-13 15:40:17 +02:00
< / u l >
`
}
2023-04-06 15:20:31 +02:00
renderCategoryHints ( ) {
2023-04-06 16:06:08 +02:00
const ml = document . querySelector ( "d4s-ccp-methodlist" )
2023-04-06 16:15:17 +02:00
const cats = ml . getCategoryHints ( )
2023-04-12 15:49:47 +02:00
if ( cats . indexOf ( "Uncategorised" ) === - 1 ) cats . push ( "Uncategorised" )
2023-04-06 15:20:31 +02:00
return `
< datalist id = "categoryhints" >
2023-04-06 15:22:54 +02:00
$ { cats . map ( c => ` <option value=" ${ c } "> ${ c } </option> ` ) }
2023-04-06 15:20:31 +02:00
< / d a t a l i s t >
`
}
2022-09-13 15:40:17 +02:00
renderContexts ( ) {
return `
< ul class = "context_list" >
$ { this . # current . metadata .
filter ( md => md . role === "context" ) .
map ( c => ` <li class="context_list_item" title=" ${ c . title } "> ${ c . title } </li> ` ) . join ( "" ) }
2022-07-21 15:42:55 +02:00
< / u l >
`
}
2022-06-27 18:52:48 +02:00
renderSaveButton ( ) {
return `
< button title = "Save" name = "save" class = "btn btn-primary ccp-toolbar-button" >
$ { this . # disc _icon }
< / b u t t o n >
`
}
2022-07-18 18:08:53 +02:00
renderDeleteButton ( ) {
return `
< button title = "Delete" name = "delete" class = "btn btn-danger ccp-toolbar-button" >
$ { this . # delete _icon }
< / b u t t o n >
`
}
2022-06-27 18:52:48 +02:00
renderResetButton ( ) {
return `
< button name = "reset" title = "Reset" class = "btn btn-primary ccp-toolbar-button" >
< svg viewBox = "0 0 24 24" >
$ { this . # erase _icon }
< / s v g >
< / b u t t o n >
`
}
renderPlusButton ( name ) {
return `
< button name = "${name}" title = "Add" name = "reset" class = "btn btn-primary ccp-toolbar-button" >
$ { this . # plus _icon }
< / b u t t o n >
`
}
2023-05-24 14:49:40 +02:00
renderStandardInputButtons ( ) {
return `
< button name = "add-ccpannotation" title = "Add annotation input" name = "reset" class = "btn btn-info ccp-toolbar-button" >
$ { this . # annotation _input _icon }
< / b u t t o n >
`
}
2022-06-27 18:52:48 +02:00
renderStandardOutputButtons ( ) {
return `
2023-05-24 14:49:40 +02:00
< button name = "add-stdout" title = "Add stdout" class = "btn btn-success ccp-toolbar-button" >
2022-06-27 18:52:48 +02:00
$ { this . # output _icon }
< / b u t t o n >
2023-05-24 14:49:40 +02:00
< button name = "add-stderr" title = "Add stderr" class = "btn btn-danger ccp-toolbar-button" >
2022-06-27 18:52:48 +02:00
$ { this . # output _icon }
< / b u t t o n >
`
}
reRenderKeywords ( ) {
this . # rootdoc . querySelector ( "div[name=keyword-list]" ) . innerHTML = this . renderKeywords ( )
}
renderKeywords ( ) {
if ( this . # current . keywords ) {
return this . # current . keywords . map ( ( k , i ) => {
return `
2023-03-29 12:32:20 +02:00
< div class = "ccp-option badge ccp-keyword" title = "${k}" alt = "${k}" >
2022-06-27 18:52:48 +02:00
< span > $ { k } < / s p a n >
< span class = "btn text-danger ccp-toolbar-button" name = "delete-keyword" data - index = "${i}" > x < / s p a n >
< / d i v >
`
} ) . join ( "\n" )
} else {
return ""
}
}
2023-04-06 16:41:10 +02:00
2023-04-06 16:49:04 +02:00
reRenderCategories ( ) {
2023-04-06 16:41:10 +02:00
this . # rootdoc . querySelector ( "div[name=category-list]" ) . innerHTML = this . renderCategories ( )
}
renderCategories ( ) {
2023-04-06 17:14:01 +02:00
if ( this . # current . metadata ) {
2023-04-06 16:41:10 +02:00
return this . # current . metadata . filter ( md => md . role === "category" ) . map ( ( k , i ) => {
return `
2023-04-06 16:52:27 +02:00
< div class = "ccp-option badge ccp-category" title = "${k.title}" alt = "${k.title}" >
< span > $ { k . title } < / s p a n >
2023-04-06 16:41:10 +02:00
< span class = "btn text-danger ccp-toolbar-button" name = "delete-category" data - index = "${i}" > x < / s p a n >
< / d i v >
`
} ) . join ( "\n" )
} else {
return ""
}
}
2022-06-27 18:52:48 +02:00
2023-03-29 12:32:20 +02:00
reRenderInfrastructures ( ) {
this . # rootdoc . querySelector ( "select[name=infrastructure-input]" ) . innerHTML = this . renderInfrastructureOptions ( )
this . # rootdoc . querySelector ( "div[name=infrastructure-list]" ) . innerHTML = this . renderInfrastructures ( )
2022-06-27 18:52:48 +02:00
}
2023-03-29 12:32:20 +02:00
renderInfrastructureOptions ( ) {
const selectedinfras = this . # current . links . filter ( l => l . rel === "compatibleWith" ) . map ( i => i . href . split ( "/" ) [ 1 ] )
const available = this . # infrastructures . filter ( i => { return selectedinfras . indexOf ( i . id ) === - 1 } )
2022-06-27 18:52:48 +02:00
return `
< option > < / o p t i o n >
$ {
2023-03-29 12:32:20 +02:00
available . map ( infra => {
2022-06-27 18:52:48 +02:00
return `
2023-03-29 12:32:20 +02:00
< option value = "${infra.id}" title = "${infra.description}" > $ { infra . name } < / o p t i o n >
2022-06-27 18:52:48 +02:00
`
} ) . join ( "\n" )
}
`
}
2023-03-29 12:32:20 +02:00
renderInfrastructures ( ) {
2022-06-27 18:52:48 +02:00
return this . # current . links . filter ( l => l . rel === "compatibleWith" ) . map ( ( l , i ) => {
return `
2023-03-29 12:32:20 +02:00
< div class = "ccp-option badge ccp-keyword" title = "${l.title}" alt = "${l.title}" >
2022-06-27 18:52:48 +02:00
< span > $ { l . title } < / s p a n >
2023-03-29 12:32:20 +02:00
< span class = "btn text-danger ccp-toolbar-button" name = "delete-infrastructure" data - index = "${i}" > x < / s p a n >
2022-06-27 18:52:48 +02:00
< / d i v >
`
} ) . join ( "\n" )
}
renderInputs ( ) {
const parent = this . # rootdoc . querySelector ( "div[name=input-list]" )
parent . innerHTML = ""
return this . # tmp _inputs . map ( ( inp , i ) => {
const c = document . createElement ( "d4s-ccp-input-editor" ) ;
parent . appendChild ( c )
c . render ( inp , i )
} )
}
renderOutputs ( ) {
const parent = this . # rootdoc . querySelector ( "div[name=output-list]" )
parent . innerHTML = ""
return this . # tmp _outputs . map ( ( outp , i ) => {
const c = document . createElement ( "d4s-ccp-output-editor" ) ;
parent . appendChild ( c )
c . render ( outp , i )
} )
}
renderScripts ( ) {
2023-03-29 12:32:20 +02:00
const val = 'deploy-script'
2022-06-27 18:52:48 +02:00
return this . # current . additionalParameters . parameters . map (
2022-09-08 16:03:28 +02:00
( script , i ) => {
let code = script . value . length ? script . value . join ( "\r\n" ) : ""
2023-03-29 12:32:20 +02:00
return `
< textarea rows = "5" class = "script-area form-control ${script.name === val ? '' : 'd-none'}" name = "${script.name}" > $ { code } < / t e x t a r e a >
2022-09-08 16:03:28 +02:00
`
}
2022-06-27 18:52:48 +02:00
) . join ( "\n" )
}
}
window . customElements . define ( 'd4s-ccp-methodeditor' , CCPMethodEditorController ) ;