From 7321f5bf1323568400ea2de56bb0b346681bbe3e Mon Sep 17 00:00:00 2001
From: Sebastiano Giacomini <92300303+Sebastiano-G@users.noreply.github.com>
Date: Tue, 2 Jul 2024 00:05:21 +0200
Subject: [PATCH] #14 Inverse relationships in View Page & Term Page
UI update still in progress. View pages have been enriched with a novel section to display incoming properties related to the entity. The Term Page "Appears in" section is now organized according to Classes. Term Pages becomes available also for imported entities.
---
app.py | 65 +-
forms.py | 14 +
mapping.py | 152 +-
queries.py | 22 +-
static/css/main.css | 93 +-
static/js/backup js.js | 4460 +++++++++++++++++++++++++++++++++++++++
static/js/main.js | 878 ++++----
templates/modify.html | 33 +-
templates/record.html | 2 +-
templates/template.html | 31 +-
templates/term.html | 19 +-
templates/view.html | 58 +-
utils.py | 25 +-
13 files changed, 5375 insertions(+), 477 deletions(-)
create mode 100644 static/js/backup js.js
diff --git a/app.py b/app.py
index 704f293..a174f20 100644
--- a/app.py
+++ b/app.py
@@ -520,7 +520,7 @@ def POST(self, page):
# create a new template
elif actions.action.startswith('createTemplate'):
- print('create template')
+ print('create template:', actions)
is_git_auth = github_sync.is_git_auth()
res_type = sorted([ urllib.parse.unquote(actions[class_input].strip()) for class_input in actions if class_input.startswith("uri_class")])
res_type = conf.main_entity if res_type == [] else res_type
@@ -995,7 +995,7 @@ def GET(self, name):
base = conf.base
record = base+name
res_class = queries.getClass(conf.base+name)
- data, stage, title, properties, data_labels, extractions_data = None, None, None, None, {}, {}
+ data, stage, title, properties, data_labels, extractions_data, new_dict_classes, properties_sorted = None, None, None, None, {}, {}, {}, {}
try:
res_template = u.get_template_from_class(res_class)
@@ -1019,11 +1019,46 @@ def GET(self, name):
except Exception as e:
pass
+ try:
+ incoming_links = queries.get_records_from_object(base+name)
+ class_sorted = {}
+ for result in incoming_links['results']['bindings']:
+ result_class = result['class']['value']
+ result_property_uri = result['property']['value']
+ result_property = result_property_uri + "," + u.get_LOV_namespace(result_property_uri)
+ result_subject = result['subject']['value'] + ',' + result['label']['value']
+ if result_class not in class_sorted:
+ class_sorted[result_class] = { result_property : [result_subject] }
+ else:
+ if result_property in class_sorted[result_class]:
+ class_sorted[result_class][result_property].append(result_subject)
+ else:
+ class_sorted[result_class][result_property] = [result_subject]
+
+ with open(TEMPLATE_LIST) as tpl_list:
+ templates = json.load(tpl_list)
+ for k in list(class_sorted.keys()):
+ template = next((t["name"], t["template"]) for t in templates if t["type"] == sorted(k.split("; ")))
+ template_name, template_file = template
+ with open(template_file) as tpl_file:
+ template_fields = json.load(tpl_file)
+ property_label = list(class_sorted[k].keys())[0]
+ property_name = next(f["label"] for f in template_fields if f["property"] == property_label.split(',',1)[0])
+ if property_label in properties_sorted:
+ properties_sorted[property_label].extend(class_sorted[k][property_label])
+ else:
+ properties_sorted[property_label] = class_sorted[k][property_label]
+ new_dict_classes[template_name] = {'class':k, 'results': { property_label+','+property_name : class_sorted[k][property_label]} }
+ except Exception as e:
+ pass
+
+
return render.view(user=session['username'], graphdata=data_labels,
graphID=name, title=title, stage=stage, base=base,properties=properties,
- is_git_auth=is_git_auth,project=conf.myProject,knowledge_extractor=extractions_data)
+ is_git_auth=is_git_auth,project=conf.myProject,knowledge_extractor=extractions_data,
+ inverses_by_class=new_dict_classes, inverses_by_properties = properties_sorted)
def POST(self,name):
""" Record web page
@@ -1050,15 +1085,31 @@ def GET(self, name):
the ID of the term, generally the last part of the URL
"""
web.header("Cache-Control", "no-cache, max-age=0, must-revalidate, no-store")
- data = queries.describeTerm(name)
+ uri = mapping.getRightURIbase(name)
+ data = queries.describeTerm(uri)
is_git_auth = github_sync.is_git_auth()
- count = len([ result["subject"]["value"] \
+ results_by_class = {}
+ appears_in = [ result["subject"]["value"] \
for result in data["results"]["bindings"] \
- if (name in result["object"]["value"] and result["object"]["type"] == 'uri') ])
+ if (name in result["object"]["value"] and result["object"]["type"] == 'uri') ]
+
+
+ with open(TEMPLATE_LIST) as tpl_list:
+ res_templates = json.load(tpl_list)
+ for res_uri in appears_in:
+ res_class = sorted(queries.getClass(res_uri))
+ res_type = next(t["name"] for t in res_templates if t["type"] == res_class)
+ if res_type in results_by_class:
+ results_by_class[res_type]['results'].append(res_uri)
+ else:
+ results_by_class[res_type] = {'class':res_class, 'results':[res_uri]}
+
+ count = len(appears_in)
return render.term(user=session['username'], data=data, count=count,
- is_git_auth=is_git_auth,project=conf.myProject,base=conf.base,name=name)
+ is_git_auth=is_git_auth,project=conf.myProject,base=conf.base,
+ uri=uri,name=name,results=results_by_class)
def POST(self,name):
""" controlled vocabulary term web page
diff --git a/forms.py b/forms.py
index 0b9a1ae..12512b3 100644
--- a/forms.py
+++ b/forms.py
@@ -78,7 +78,9 @@ def get_form(json_form, from_dict=False, subtemplate=False):
classes = classes+' websitePreview' if field['type'] == 'WebsitePreview' else classes
classes = classes+' disabled' if 'disabled' in field and field['disabled'] == "True" else classes
classes = classes+ ' ' + field['cardinality'] if 'cardinality' in field else classes
+ classes = classes+ ' ' + field['dataReuse'] if 'dataReuse' in field else classes
autocomplete = field['cache_autocomplete'] if 'cache_autocomplete' in field and len(field['cache_autocomplete']) > 0 else ''
+ rdf_property = field['property'] if 'property' in field else ''
mandatory = field['mandatory'] if 'mandatory' in field and field['mandatory'] == 'True' else 'False'
# text box
@@ -100,6 +102,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
class_= classes,
value=default,
lang=conf.mainLang,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class) , )
else:
@@ -112,6 +115,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
class_= classes,
value=default,
lang=conf.mainLang,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -124,6 +128,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
pre = prepend,
class_= classes,
value=default,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -136,6 +141,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
pre = prepend,
class_= classes,
value=default,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class) , )
@@ -149,6 +155,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
class_= classes,
value=default,
lang=conf.mainLang,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -159,6 +166,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
id=myid,
pre = prepend,
class_= classes,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
elif field['calendar'] == 'Day':
@@ -167,6 +175,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
id=myid,
pre = prepend,
class_= classes,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
elif field['calendar'] == 'Year':
@@ -176,6 +185,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
pre = prepend,
class_= classes,
value=default,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -187,6 +197,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
id=myid,
pre = prepend,
class_= classes,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -200,6 +211,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
pre = prepend_title,
class_= classes+' checkbox_group',
checked=False,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -212,6 +224,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
pre = '',
class_= classes+' checkbox_group following_checkbox',
checked=False,
+ data_property = rdf_property,
data_mandatory = mandatory,
data_class=res_class), )
@@ -227,6 +240,7 @@ def get_form(json_form, from_dict=False, subtemplate=False):
value=default,
data_mandatory = mandatory,
data_class=res_class,
+ data_property = rdf_property,
data_subtemplate = resource_class,
data_subtemplateID = field['import_subtemplate']), ) + get_form(field['import_subtemplate'], subtemplate=True)
diff --git a/mapping.py b/mapping.py
index c82784b..73466e5 100644
--- a/mapping.py
+++ b/mapping.py
@@ -34,6 +34,7 @@
server = sparql.SPARQLServer(conf.myEndpoint)
dir_path = os.path.dirname(os.path.realpath(__file__))
RESOURCE_TEMPLATES = conf.resource_templates
+ASK_CLASS = conf.ask_form
TEMPLATE_LIST = conf.template_list
def getValuesFromFields(fieldPrefix, recordData, fields=None, field_type=None):
@@ -48,12 +49,9 @@ def getValuesFromFields(fieldPrefix, recordData, fields=None, field_type=None):
values = value.split(',', 1)
results.add(( values[0].strip(), urllib.parse.unquote(values[1]) )) # (id, label)
else:
- print("key:",key, fieldPrefix)
if key.startswith(fieldPrefix+'_') and ',' in value: # multiple values from text box (entities) and URL
values = value.split(',', 1)
- print("val:",values)
results.add(( values[0].strip(), urllib.parse.unquote(values[1]) )) # (id, label)
- print(results)
elif key == fieldPrefix: # uri from dropdown (single value from controlled vocabulary) and URL
if fields:
field = next(field for field in fields if field["id"] == fieldPrefix)
@@ -131,10 +129,12 @@ def inputToRDF(recordData, userID, stage, graphToClear=None,tpl_form=None):
for predicate_obj, inner_dict in binding.items():
if predicate_obj.endswith('_property'):
predicate = URIRef(inner_dict['value'])
- obj_value = binding[predicate_obj.replace("_property", "")]['value']
- obj_type = binding[predicate_obj.replace("_property", "")]['type']
- obj = URIRef(obj_value) if obj_type == "uri" else Literal(obj_value, datatype="http://www.w3.org/2001/XMLSchema#string")
- label = binding[predicate_obj.replace("_property", "") + "_label"]['value'] if predicate_obj.replace("_property", "") + "_label" in binding else None
+ short_predicate_obj = predicate_obj.replace("_property", "")
+ obj_value = binding[short_predicate_obj]['value']
+ obj_type = binding[short_predicate_obj]['type']
+ obj_datatype = binding[short_predicate_obj]['datatype'] if 'datatype' in binding[short_predicate_obj] else ''
+ obj = URIRef(obj_value) if obj_type == "uri" else Literal(obj_value, datatype=obj_datatype)
+ label = binding[short_predicate_obj + "_label"]['value'] if predicate_obj.replace("_property", "") + "_label" in binding else None
wd.add((subject, URIRef(predicate), obj))
print(subject, predicate, obj, label)
if label:
@@ -161,6 +161,7 @@ def inputToRDF(recordData, userID, stage, graphToClear=None,tpl_form=None):
else getValuesFromFields(field['id'], recordData, field_type=field['value']) if 'value' in field and field['value'] == 'URL' \
else getLiteralValuesFromFields(field['id'], recordData) if 'value' in field and field['value'] == 'Literal' else recordData[field['id']]
# TODO disambiguate as URI, value
+ print("VALUE:", value)
if field["disambiguate"] == 'True': # use the key 'disambiguate' as title of the graph
main_lang = value['mainLang']
main_value = [label for label in value['results'] if label[1] == main_lang]
@@ -287,7 +288,8 @@ def inputToRDF(recordData, userID, stage, graphToClear=None,tpl_form=None):
# process a new subrecord, send its data to the triplestore, and link it to the main record
subrecord_id = subrecord
subrecord_template = field['import_subtemplate']
- processed_subrecord = process_new_subrecord(recordData,userID,stage,subrecord_template,subrecord)
+ allow_data_reuse = fields if 'dataReuse' in field and field['dataReuse']=='allowDataReuse' else False
+ processed_subrecord = process_new_subrecord(recordData,userID,stage,subrecord_template,subrecord,supertemplate=None,allow_data_reuse=allow_data_reuse)
subrecord_id, retrieved_label = processed_subrecord
wd.add(( URIRef(base+graph_name), URIRef(field['property']), URIRef(base+subrecord_id) ))
wd.add(( URIRef(base+subrecord_id), RDFS.label, Literal(retrieved_label, datatype="http://www.w3.org/2001/XMLSchema#string")))
@@ -315,57 +317,97 @@ def inputToRDF(recordData, userID, stage, graphToClear=None,tpl_form=None):
return 'records/'+recordID+'.ttl'
-def process_new_subrecord(data, userID, stage, sub_tpl, subrecord_id):
+def process_new_subrecord(data, userID, stage, sub_tpl, subrecord_id, supertemplate=None, allow_data_reuse=False):
# prepare a new dict to store data of subrecord-x
new_record_data = {'recordID': subrecord_id,}
-
+ label = 'No Label!'
with open(sub_tpl) as fields:
subtemplate = json.load(fields)
+ subtemplate = sorted(subtemplate, key=lambda x: x['type'] == 'subtemplate')
# process the input data related to subrecord-x
for subtemplate_field in subtemplate:
- subfield_id = subtemplate_field['id']
-
- # Subtemplate
- if subtemplate_field['type'] == 'Subtemplate':
- key = subfield_id+"_"+subrecord_id
- # Process inner-subrecords and retrieve their ids,labels in order to provide a link to them in the upper-level subrecord
- if key+"-subrecords" in data:
- new_record_data[subfield_id] = [[]]
- inner_subtemplate = subtemplate_field['import_subtemplate']
- for inner_subrecord in data[key+"-subrecords"].split(","):
- if ";" in inner_subrecord:
- processed_subrecord = inner_subrecord.split(";",1)
- else:
- processed_subrecord = process_new_subrecord(data,userID,stage,inner_subtemplate,inner_subrecord)
- new_record_data[subfield_id][0].append(processed_subrecord) # store the id,label pair inside the subrecord dict
-
- # Date
- elif subtemplate_field['type'] == 'Date':
- key = subtemplate_field['id']+"_"+subrecord_id
- new_record_data[subtemplate_field['id']] = data[key]
-
- # Knowledge Extraction
- elif subtemplate_field['type'] == 'KnowledgeExtractor' and 'extractions-dict' in data:
- new_record_data['extractions-dict'] = data['extractions-dict']
- for keyword_key,keyword_value in data.items():
- print("FFF", subrecord_id)
- if keyword_key.startswith('keyword_'+subrecord_id):
- print("QQQQ",keyword_key,keyword_value)
- new_record_data[keyword_key]=keyword_value
-
- # Multiple values fields: Literals or URI
- elif 'value' in subtemplate_field and (subtemplate_field['value'] == 'Literal' or subtemplate_field['value'] in ['URI','URL','Place']):
- keys = [input_id for input_id in data.keys() if input_id.startswith(subtemplate_field['id']+"_") and input_id.endswith("_"+subrecord_id)]
- for key in keys:
- shortened_key = key.rsplit("_",1)[0]
- new_record_data[shortened_key] = data[key]
-
- # Label: disambiguate field
- if subtemplate_field['disambiguate'] == "True":
- main_lang_input_field = subfield_id+'_mainLang_'+subrecord_id
- main_lang = data[main_lang_input_field] if main_lang_input_field in data else "No main lang"
- label_input_field = subfield_id+"_"+main_lang+"_"+subrecord_id
- label = data[label_input_field] if label_input_field in data else "No label"
+ if subtemplate_field['hidden'] == 'False':
+ subfield_id = subtemplate_field['id']
+ rdf_property = subtemplate_field['property']
+
+ # Subtemplate
+ if subtemplate_field['type'] == 'Subtemplate':
+ key = subfield_id+"_"+subrecord_id
+ # Process inner-subrecords and retrieve their ids,labels in order to provide a link to them in the upper-level subrecord
+ if key+"-subrecords" in data:
+ new_record_data[subfield_id] = [[]]
+ inner_subtemplate = subtemplate_field['import_subtemplate']
+ data_reuse = subtemplate if 'dataReuse' in subtemplate_field and subtemplate_field['dataReuse']=='allowDataReuse' else False
+ for inner_subrecord in data[key+"-subrecords"].split(","):
+ if ";" in inner_subrecord:
+ processed_subrecord = inner_subrecord.split(";",1)
+ else:
+ processed_subrecord = process_new_subrecord(data,userID,stage,inner_subtemplate,inner_subrecord,subrecord_id,data_reuse)
+ new_record_data[subfield_id][0].append(processed_subrecord) # store the id,label pair inside the subrecord dict
+
+ # Date
+ elif subtemplate_field['type'] == 'Date':
+ key = subtemplate_field['id']+"_"+subrecord_id
+ if key in data:
+ new_record_data[subtemplate_field['id']] = data[key]
+ elif allow_data_reuse:
+ upper_level_field_ids = [field['id'] for field in allow_data_reuse if field['property'] == rdf_property and field['type'] == subtemplate_field['type']]
+ if len(upper_level_field_id) > 0:
+ upper_level_field_id = upper_level_field_ids[0]
+ look_for_id = upper_level_field_id+'_'+supertemplate if supertemplate else upper_level_field_id
+ new_record_data[subtemplate_field['id']] = data[look_for_id]
+ data[key] = data[look_for_id]
+
+ # Knowledge Extraction
+ elif subtemplate_field['type'] == 'KnowledgeExtractor' and 'extractions-dict' in data:
+ new_record_data['extractions-dict'] = data['extractions-dict']
+ for keyword_key,keyword_value in data.items():
+ if keyword_key.startswith('keyword_'+subrecord_id):
+ new_record_data[keyword_key]=keyword_value
+
+ # Multiple values fields: Literals or URI
+ elif 'value' in subtemplate_field and (subtemplate_field['value'] == 'Literal' or subtemplate_field['value'] in ['URI','URL','Place']):
+ keys = [input_id for input_id in data.keys() if input_id.startswith(subtemplate_field['id']+"_") and input_id.endswith("_"+subrecord_id)]
+ if len(keys) > 0:
+ for key in keys:
+ shortened_key = key.rsplit("_",1)[0]
+ new_record_data[shortened_key] = data[key]
+
+ # Label: disambiguate field
+ if subtemplate_field['disambiguate'] == "True":
+ main_lang_input_field = subfield_id+'_mainLang_'+subrecord_id
+ main_lang = data[main_lang_input_field] if main_lang_input_field in data else "No main lang"
+ label_input_field = subfield_id+"_"+main_lang+"_"+subrecord_id
+ label = data[label_input_field] if label_input_field in data else "No label"
+ elif allow_data_reuse:
+ upper_level_field_ids = [field['id'] for field in allow_data_reuse if field['property'] == rdf_property and field['type'] == subtemplate_field['type']]
+ if len(upper_level_field_ids) > 0:
+ upper_level_field_id = upper_level_field_ids[0]
+ if supertemplate:
+ keys = [input_id for input_id in data.keys() if input_id.startswith(upper_level_field_id+"_") and input_id.endswith("_"+supertemplate)]
+ else:
+ keys = [input_id for input_id in data.keys() if input_id.startswith(upper_level_field_id+"_")]
+ for key in keys:
+ cut_number = 1 if supertemplate else 0
+ shortened_key = key.rsplit("_",cut_number)[0]
+ shortened_key = key.split("_")[1]
+ shortened_key = subfield_id+'_'+shortened_key
+ new_record_data[shortened_key] = data[key]
+ data[shortened_key+'_'+subrecord_id] = data[key]
+
+
+ # Label: disambiguate field
+ if subtemplate_field['disambiguate'] == "True":
+ main_lang_input_field = subfield_id+'_mainLang_'+subrecord_id
+ main_lang = data[main_lang_input_field] if main_lang_input_field in data else "No main lang"
+ label_input_field = upper_level_field_id+"_"+main_lang+"_"+ supertemplate if supertemplate else upper_level_field_id+'_'+main_lang
+ with open(TEMPLATE_LIST,'r') as tpl_file:
+ tpl_list = json.load(tpl_file)
+ disambiguation_label = [t['name'] for t in tpl_list if t['template']==sub_tpl][0]
+ label = data[label_input_field] + " - " + disambiguation_label if label_input_field in data else "No label"
+ new_label_input_id = subfield_id + '_' + main_lang
+ new_record_data[new_label_input_id] = label
+
@@ -377,7 +419,7 @@ def process_new_subrecord(data, userID, stage, sub_tpl, subrecord_id):
result = [subrecord_id,label]
return result
-def find_label(tpl):
+""" def find_label(tpl):
# Retrieve the field associated with the Primary Key (i.e., the label) of the Record
with open(tpl) as tpl_file:
tpl_fields = json.load(tpl_file)
@@ -385,4 +427,4 @@ def find_label(tpl):
label_field_id = fields_id[0] if fields_id != [] else False
# TODO: add a mechanism to handle potential Templates without a Primary Key in case it's needed
- return label_field_id
\ No newline at end of file
+ return label_field_id """
\ No newline at end of file
diff --git a/queries.py b/queries.py
index 859f23b..e0e3734 100644
--- a/queries.py
+++ b/queries.py
@@ -326,10 +326,11 @@ def disambiguate_pattern(properties,fields,k):
def describeTerm(name):
""" ask if the resource exists, then describe it."""
- ask = """ASK { ?s ?p <""" +conf.base+name+ """> .}"""
+ ask = """ASK { ?s ?p <""" +name+ """> .}"""
results = hello_blazegraph(ask)
if results["boolean"] == True: # new entity
- describe = """DESCRIBE <"""+conf.base+name+ """>"""
+ describe = """DESCRIBE <"""+name+ """>"""
+ print("DESCRIBE", describe, hello_blazegraph(describe))
return hello_blazegraph(describe)
else: # vocab term
ask = """ASK { ?s ?p ?o .
@@ -537,4 +538,21 @@ def saveHiddenTriples(graph, tpl):
sparql.setQuery(queryNGraph)
sparql.setReturnFormat(JSON)
results = sparql.query().convert()
+ print(results)
return results
+
+def get_records_from_object(graph_uri):
+ query = """
+ SELECT DISTINCT ?subject ?property ?class ?label
+ WHERE {
+ ?subject ?property <"""+graph_uri+""">.
+ ?subject rdfs:label ?label .
+ ?subject a ?class .
+ }
+ """
+ sparql = SPARQLWrapper(conf.myEndpoint)
+ sparql.setQuery(query)
+ sparql.setReturnFormat(JSON)
+ results = sparql.query().convert()
+ print(results)
+ return results
\ No newline at end of file
diff --git a/static/css/main.css b/static/css/main.css
index fa3238b..d91a70c 100644
--- a/static/css/main.css
+++ b/static/css/main.css
@@ -184,6 +184,12 @@ button.active {
border-color: #F5F6FA #F5F6FA rgba(100, 23, 180, 1);
}
+.nav-tabs a.dark-nav.nav-link.active {
+ text-decoration: none !important;
+ background-color: #f5f6fa3a !important;
+ border-color: rgba(100, 23, 180, 1);
+}
+
.logo_header {
display: inline-block;
padding: 5px;
@@ -518,11 +524,11 @@ article p {
padding: 4em !important;
}
-#recordForm .homeheading:nth-child(2) {
+.corners.row .homeheading:nth-child(2) {
padding-right: 0.5em !important;
}
-#recordForm .homeheading:nth-child(3) {
+.corners.row .homeheading:nth-child(3) {
padding-top: 0.35em !important;
padding-left: 0.5em !important;
}
@@ -735,7 +741,7 @@ th {
.label .title {
flex: 1;
- font-size: 1.2em;
+ font-size: 1em;
}
.label img, .label i {
@@ -1109,6 +1115,9 @@ table.extraction-table {
margin-top: 2em;
margin-bottom: -2.5em;
}
+table.extraction-table + span + .block_field {
+ margin-top: 3.5em;
+}
table.url-table th, table.url-table td, table.extraction-table th, table.extraction-table td {
border: 1px solid black;
@@ -1152,6 +1161,11 @@ table.url-table i, table.extraction-table i {
text-transform: uppercase;
color: grey;
font-size: 0.9em !important;
+ display: inline-block;
+}
+
+.subtemplateField:last-child {
+ margin-bottom: 2em;
}
/*display: inline !important;*/
@@ -1254,10 +1268,16 @@ table.url-table i, table.extraction-table i {
transition-duration: 0.4s;
}
-.relatedResources {
- margin: 20px 0px;
+.related-resources-container .articleSubtitle {
+ margin: 2em 0 0 0;
+ display: inline-block;
+}
+.related-resources-container button {
+ display: inline-block;
+ margin-bottom: 0 !important;
}
+
/* historian/collection template */
/*.info-item {font-style: italic;}*/
.sideBoxes {
@@ -1265,6 +1285,52 @@ table.url-table i, table.extraction-table i {
margin-bottom: 3em;
}
+.inverse-subtitle {
+ padding-left: 1em;
+ padding-top: 1em;
+ text-transform: uppercase;
+ font-weight: 600;
+ color: rgb(100, 23, 180);
+ display: inline-block;
+}
+
+.inverse-results-container {
+ display: flex;
+ flex-wrap: wrap;
+ align-items: flex-start;
+ gap: 20px;
+ border-bottom: 2px solid rgba(100, 23, 180, 0.12);
+ padding-left: 1em;
+}
+
+.inverse-property {
+ display: flex;
+ flex-direction: column;
+ justify-content: center;
+ padding: 10px 30px 10px 0;
+}
+
+.inverse-property:first-of-type {
+ width: 20%
+}
+
+.inverse-property span {
+ font-weight: 400;
+ color: #333;
+
+}
+
+.inverses {
+ margin: 3em 0 -1em 0;
+ max-height: 1000px;
+ background-color: silver;
+}
+
+.inverses .section-subtitle {
+ font-weight: 400;
+ color: grey;
+}
+
.info-bio {
margin-top: 2em;
}
@@ -1290,6 +1356,13 @@ table.url-table i, table.extraction-table i {
font-size: 32px;
}
+.biblio-navigate {
+ font-weight: 500;
+ text-decoration: underline;
+ color:#333;
+ cursor:pointer;
+}
+
.bibliospace {
padding-top: 25px !important;
padding-bottom: 10px !important;
@@ -2528,7 +2601,11 @@ button#showTemplates {
overflow: hidden;
transition: max-height 0.3s ease, padding 0.3s ease, margin-top 0.3s ease, margin-bottom 0.3s ease; /* Corretta sintassi per la transizione */
padding: 1em;
- max-height: 1000px;
+ max-height: 10000px;
+}
+
+.subform:first-child {
+ margin-top: 1.5em;
}
.subform.closed {
@@ -2548,6 +2625,10 @@ button#showTemplates {
align-items: center;
}
+.subform .title {
+ font-size: 0.8em;
+}
+
.subform-buttons {
display: flex;
}
diff --git a/static/js/backup js.js b/static/js/backup js.js
new file mode 100644
index 0000000..cab52d5
--- /dev/null
+++ b/static/js/backup js.js
@@ -0,0 +1,4460 @@
+
+if (graph.length) {var inGraph = "FROM <"+graph+">"} else {var inGraph = ""}
+const wdImg = ' '
+const wdImgIcon = ' '
+const geoImg = ' ';
+const viafImg = ' ';
+const viafImgIcon = ' '
+const wikidataEndpoint = "https://query.wikidata.org/sparql"
+$(document).ready(function() {
+
+ // loader
+ $(".se-pre-con").fadeOut("slow");
+
+ // disable submit form when pressing return
+ $("input[type='text'], input[type='textarea']").on('keyup keypress', function(e) {
+ var keyCode = e.keyCode || e.which;
+ if (keyCode === 13) {
+ e.preventDefault();
+ return false;
+ }
+ });
+
+ // create a new TEMPLATE
+ $("#selectTemplateClassButton").on('click', function() {
+ // show modal
+ $("#selectTemplateClassModal").toggleClass('open-modal');
+ $('body').append($("
"));
+ });
+
+ /* check class_name */
+ $("#selectTemplateClass [name='class_name']").on('click', function() {
+ // show error message in case a name is already in use
+ var templatesNames = templatesObject.map(obj => obj.name.toLowerCase());
+ $(this).off('keyup').on('keyup', function() {
+ $(this).removeClass('error-input');
+ $(this).next('.error-message').remove();
+ var val = $(this).val();
+ if (templatesNames.includes(val.toLowerCase())) {
+ $(this).addClass('error-input');
+ $(this).after($('
This name is already in use. Try a new one'))
+ }
+ })
+
+ });
+
+ /* check class_uri */
+ $("#selectTemplateClass [name='class_uri']").on('click', function() {
+ $(this).off('keyup').on('keyup', function(e) {
+ if (e.which == 13 && $(this).val().length > 0) {
+ var id = new Date().valueOf().toString();
+ $(this).next('section').append("
"+$(this).val()+" ");
+ $(this).val('')
+ }
+ })
+ });
+
+ /* remove template creation modal */
+ $("#selectTemplateClass [value='cancelTemplate'], #selectTemplateClass .fa-times").on('click', function(e) {
+ e.preventDefault();
+ $("#selectTemplateClassModal").toggleClass('open-modal');
+ $("body div.modal-bg").remove();
+ return false;
+ });
+
+ /* save new template */
+ $("#selectTemplateClass [value='createTemplate']").on('click', function(e) {
+ e.preventDefault();
+ validateTemplateClass();
+ });
+
+ // message after saving
+ $("#save_record").on('click', function(e) {
+ e.preventDefault();
+ var sel = document.getElementById('res_name');
+
+ if (sel !== undefined) {
+ // when selecting the template
+ if (sel && sel.value == 'None') {
+ Swal.fire({ title: 'choose a template please'});
+ setTimeout(function() { document.getElementById('recordForm').submit();}, 500);
+ } else {
+ // when creating and saving the record
+ var check_mandatory = checkMandatoryFields()
+ if (check_mandatory) {
+ Swal.fire({ title: 'Saved!'});
+ if ($('#recordForm').length) {
+ var element_id = 'recordForm';
+ } else {
+ var element_id = 'modifyForm';
+ }
+ setTimeout(function() { document.getElementById(element_id).submit();}, 500);
+ }
+ }
+ }
+ else {
+ // when saving the record
+ Swal.fire({ title: 'Saved!'});
+ setTimeout(function() { document.getElementById('recordForm').submit();}, 500);
+ };
+ });
+
+ // check templates' constraints
+ $("#updateTemplate").on('click', function(e) {
+ e.preventDefault();
+
+ // make sure the primary key is mandatory
+ var primary_key = $('.disambiguate[checked="checked"');
+ primary_key.parent().parent().find('input[id*="mandatory"]').attr("checked", "checked");
+
+ // prevent mandatory fields to be hidden
+ var mandatory_fields = $('input[type="checkbox"][id*="mandatory"][checked="checked"]');
+ mandatory_fields.each(function() {
+ var hidden_field_checkbox = $(this).parent().parent().find('input[type="checkbox"][id*="hidden"]');
+ if (hidden_field_checkbox.attr('checked') == 'checked') {
+ Swal.fire({ title:"Hidden fields cannot be mandatory"});
+ return false;
+ };
+ });
+
+ // save the template in case everything is ok
+ Swal.fire({ title: 'Saved!'});
+ setTimeout(function() { document.getElementById('templateForm').submit();}, 500);
+ });
+
+ // table of contents
+ $('section.label.col-12').each(function() {
+ var section = $(this);
+ var itemTitle = $(this).text();
+ var listItem = $("
"+itemTitle+" ");
+ listItem.on('click', function() {
+ $('html, body').animate({
+ scrollTop: section.parent().offset().top - 100
+ }, 800);
+ })
+ $('.fields-list').append(listItem);
+ })
+
+ // disable forms
+ $(".disabled").attr("disabled","disabled");
+
+ // URL detection
+ $('.info-url').each(function(element) {
+ var str_text = $(this).html();
+ var regex = /(\b(https?|ftp):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/gim;
+ // Replace plain text links by hyperlinks
+ var replaced_text = str_text.replace(regex, "
$1 ");
+ // Echo link
+ $(this).html(replaced_text);
+ });
+
+ // tooltips
+ $('.tip').tooltip();
+
+ // fields without tooltip
+ $('.input_or_select').not(':has(.tip)').css("padding-left","3.2em");
+
+ // check prior records and alert if duplicate
+ checkPriorRecords('disambiguate');
+
+ // Named Entity Recognition in long texts
+ const areas = document.querySelectorAll('#recordForm textarea, #modifyForm textarea');
+ var tags = document.createElement('div');
+ tags.setAttribute('class','tags-nlp');
+ areas.forEach(element => { element.after(tags); });
+
+ // Textbox > URL: suggestion
+ const textboxURL = document.querySelectorAll('#recordForm input.urlField, #modifyForm input.urlField, #recordForm input.multimediaField, #modifyForm input.multimediaField, #recordForm input.websitePreview, #modifyForm input.websitePreview, #recordForm input.searchWikidata, #modifyForm input.searchWikidata');
+ textboxURL.forEach(element => {
+ var suggestionTags = document.createElement('div');
+ suggestionTags.setAttribute('class','tags-url');
+ const parent = element.parentNode;
+ if (element.className.includes('multimediaField')) {
+ const previousURLS = Array.from(parent.querySelectorAll('.multimediaTag'));
+ previousURLS.forEach(tag => {
+ tag.remove();
+ suggestionTags.appendChild(tag);
+ });
+ } else {
+ console.log(element)
+ const previousURLS = Array.from(parent.querySelectorAll('span.tag, input.hiddenInput'));
+ previousURLS.forEach(tag => {
+ tag.remove();
+ suggestionTags.appendChild(tag);
+ });
+ };
+ element.after(suggestionTags.cloneNode(true));
+ });
+
+ // Suggest vocabularies links
+ $("input[type='text'].searchSkos").each(function() {
+ let vocabs_link = [];
+ var id = $(this).attr("id");
+
+ // check which SKOS vocabularies have been associated with the input field
+ if (id in query_templates) {
+ var selected_vocabs = query_templates[id];
+ selected_vocabs.forEach(function(obj, idx) {
+ var vocab_name = Object.keys(obj)[0]
+ var vocab_name_clean = vocab_name.replace("-", " ");
+ var vocab_link = vocab_name_clean + "," + obj[vocab_name].url;
+ vocabs_link.push(vocab_link);
+ });
+ }
+
+ // create a div to store the shortcuts to Thesauri
+ const div = $("
");
+ if ($(this).prev().length > 0) {
+ div.insertBefore($(this).prev());
+ } else {
+ div.insertBefore($(this));
+ }
+
+ // create a shortcut for each vocabulary
+ vocabs_link.forEach(function(link) {
+ const name = link.split(",")[0].toUpperCase();
+ const url = link.split(",")[1];
+ div.append("
" + name + " ");
+ });
+ });
+
+ // search WD, VIAF, my data, vocabs, years + add URLs
+ $(".main_content").on("click", "input[type='text']", function () { // make the onclick function valid for later generated inputs
+ searchID = $(this).attr('id');
+
+ if ( $(this).hasClass('searchWikidata') && $(this).hasClass('wikidataConstraint') && $(this).hasClass('catalogueConstraint')) {
+ searchWDCatalogueAdvanced(searchID);
+ } else if ( $(this).hasClass('searchWikidata') && !($(this).hasClass('wikidataConstraint')) && !($(this).hasClass('catalogueConstraint')) ) {
+ searchWD(searchID);
+ } else if ( $(this).hasClass('searchWikidata') && $(this).hasClass('wikidataConstraint')) {
+ searchWDAdvanced(searchID);
+ } else if ( $(this).hasClass('searchWikidata') && $(this).hasClass('catalogueConstraint')) {
+ searchCatalogueAdvanced(searchID);
+ };
+
+ if ( $(this).hasClass('searchGeonames') ) {
+ searchGeonames(searchID);
+ };
+
+ if ( $(this).hasClass('searchSkos') ) {
+ searchSkos(searchID);
+ };
+
+ if ( $(this).hasClass('searchGeneral') ) {
+ searchCatalogue('search');
+ };
+
+ if ( $(this).hasClass('searchLOV') ) {
+ searchLOV(searchID);
+ };
+
+ if ( $(this).hasClass('urlField') ) {
+ addURL(searchID);
+ };
+
+ if ( $(this).hasClass('yearField') ) {
+ searchYear(searchID);
+ };
+
+ if ( $(this).hasClass('multimediaField')) {
+ addMultimedia(searchID);
+ }
+
+ if ( $(this).hasClass('websitePreview')) {
+ addURL(searchID, iframe=true);
+ }
+
+ if ( $(this).hasClass('manual-entity')) {
+ addManualEntity(searchID);
+ }
+
+ if ( $(this).attr('subtemplate') != undefined) {
+ searchCatalogueByClass(searchID);
+ }
+
+ });
+
+ // remove modal preview
+ $(document).on('click', '.closePreview', function () {
+ $(".modal-previewMM").remove();
+ $('#showRight').show();
+ });
+
+ // remove tag onclick
+ $(document).on('click', '.tag', function () {
+ $(this).next().remove();
+ if ($(this).prev().hasClass("MMtag") || $(this).prev().hasClass("iframePreview")) {
+ $(this).prev().remove();
+ }
+ $(this).remove();
+
+ //colorForm();
+ });
+
+ // autoresize textarea
+ $('textarea').each(function () {
+ var styleAttr= 'height:' + (this.scrollHeight)/2 + 'px;overflow-y:hidden;';
+ if (this.classList.contains('hiddenInput')) { styleAttr += 'display:none;'}
+ this.setAttribute('style', styleAttr);
+ }).on('input', function () {
+ this.style.height = 'auto';
+ this.style.height = (this.scrollHeight) + 'px';
+ });
+
+ // remove exceding whitespaces in text area
+ $('textarea[id*="values__"]').each(function () {
+ $(this).val($.trim($(this).val()).replace(/\s*[\r\n]+\s*/g, '\n'));
+ });
+
+ // Show documentation in the right sidebar
+ if ($('header').hasClass('needDoc')) {
+ var menuRight = document.getElementById( 'cbp-spmenu-s2' ),
+ showRight = document.getElementById( 'showRight' ),
+ body = document.body;
+ showRight.onclick = function() {
+ classie.toggle( this, 'active' );
+ classie.toggle( menuRight, 'cbp-spmenu-open' );
+ };
+ };
+
+ // hide lookup when creating a record
+ $("#lookup").hide();
+ // append WD icon to input fields
+ $('.searchWikidata').parent().prev().append(wdImg);
+ $('.searchWikidata').parent().prev().append(viafImg);
+ $('.searchGeonames').parent().prev().append(geoImg);
+ $('.wikiEntity').append(wdImgIcon);
+ $('.geoEntity').append(geoImg);
+ $('.viafEntity').append(viafImg);
+ // append Entity Autocompletion toggle switch to input fields
+ $('.searchWikidata').parent().append($('
\
+ \
+ \
+ \
+ \
+
\
+ '))
+ // generate input fields for Entities manual definition
+ $('input~.autocompletion-container .switch').on('click', function() {
+ console.log($(this).parent().prev())
+ $(this).parent().prev('div').toggleClass('active');
+ if (! $(this).find('input').prop('checked')) {
+ var inputField = $(this).parent().prev().prev('.searchWikidata');
+ var id = inputField.attr('id');
+ inputField.hide();
+ inputField.after($('
URI '));
+ inputField.next().next().after($('
Label '));
+
+ } else {
+ var inputField = $(this).parent().parent().find('.searchWikidata');
+ var id = inputField.attr('id');
+ console.log(inputField,id)
+ $('span.manual-entity[data-target="'+id+'"], span.manual-entity[data-target="'+id+'"]+input.manual-entity').remove();
+ inputField.show();
+
+ }
+ })
+ // hide placeholder if filled
+ //colorForm();
+
+ // style mandatory fields
+ $("[data-mandatory='True']").parent().prev(".label").find('.title').append("
* ")
+
+ // prevent POST when deleting records
+ $('.delete').click(function(e) {
+ var result = confirm("Are you sure you want to delete this record?");
+ if (result) { } else { e.preventDefault(); return false; };
+ });
+ // prevent POST when deleting templates
+ $('.delete_template').click(function(e) {
+ var result = confirm("Are you sure you want to delete the template? You will not be able to create new records with this template. Existing records using this template will not be deleted, but you will not be able to modify them.");
+ if (result) { } else { e.preventDefault(); return false; };
+ });
+
+ // change select aspect everywhere
+ $('section > select').addClass('custom-select');
+
+ // sort alphabetically in EXPLORE
+ $('.wrapAllparent').each(function () {
+ $(this).append("
");
+ });
+
+ // hide intro message in explore section
+ $(".opentab").on("click", function(el) {
+ $(".intro_explore").hide();
+ $("main").css("background-color","#F5F6FA");
+ });
+
+ // tooltips
+ $('.tip').tooltip();
+
+
+
+ $('.tab-content .list').each(function () {
+ var letter = $('a', this).text().toUpperCase().charAt(0);
+ var encoded_letter = letter;
+ var data_target = this.id; // e.g. web_resource_T
+ var res_id = data_target.substring(0, data_target.length - 2); // e.g. web_resource
+ if (!/[A-Za-z0-9]/.test(letter)) {
+ // bootstrap.min.js cannot handle non-alphanumerical characters, including %
+ //→ replace them with an encoded string.
+ encoded_letter = encodeURIComponent(letter).replace("%", "_");
+ data_target = res_id + "_" + encoded_letter;
+ }
+ if (!$(this).parent().find('[data-letter="'+ encoded_letter +'"][id="'+ data_target +'"]').length) {
+ $(this).parent().append('
');
+ $(this).parent().parent().find($('.alphabet')).append('
'+ letter +' ');
+ };
+ $(this).parent().find('[data-letter="'+ encoded_letter +'"]').append(this);
+ $('.toBeWrapped.'+res_id).each(function() {
+ console.log(res_id)
+ if (!$(this).parent().hasClass('accordion-group')) {
+ $(this).wrapAll("
");
+
+ }
+ });
+
+ });
+ //$(".wrapAllparent").children(".toBeWrapped").wrapAll("
");
+ // $('.toBeWrapped').each(function () {
+ // $(this).wrapAll("
");
+ // });
+
+
+ // sort alphabet list
+ const alphabets = document.querySelectorAll(`[id^="alphabet"]`);
+ alphabets.forEach(element => { sortList(element.id); });
+
+ // focus on click
+ $('.resource_collapse').on('click', function (e) {
+ $(e.currentTarget).parent('span').addClass('active');
+ });
+
+ // close other dropdowns when opening one
+ var $myGroup = $('.accordion-group');
+ $('.collapse').on('show.bs.collapse', function () {
+ $('.resource_collapse').parent('span').removeClass('active');
+ $('.info_collapse').removeClass('alphaActive');
+ $myGroup.find('.collapse').collapse('hide');
+ var id = $(this).attr('id');
+ var dropLabel = $('.resource_collapse[data-target="#'+id+'"]');
+ dropLabel.parent('span').addClass('active');
+ // in browse by name the label of the tab is different
+ var alphaLabel = $('.info_collapse[data-target="#'+id+'"]');
+ alphaLabel.addClass('alphaActive');
+ });
+
+ // show more in EXPLORE
+ $(".showMore").hide();
+
+ // trigger tabs in EXPLORE
+ var triggerTabList = [].slice.call(document.querySelectorAll('#resource_classes_tab a'))
+ triggerTabList.forEach(function (triggerEl) {
+ var tabTrigger = new bootstrap.Tab(triggerEl)
+
+ triggerEl.addEventListener('click', function (event) {
+ event.preventDefault()
+ tabTrigger.show()
+ })
+ });
+ // show related resources in "term" page
+ $(".showRes").on("click", {count: $(".showRes").data("count"), uri: $(".showRes").data("uri"), limit_query: $(".showRes").data("limit"), offset_query: $(".showRes").data("offset")}, searchResources);
+
+ // sortable blocks in TEMPLATE setup
+ moveUpAndDown() ;
+
+ // remove fields from form TEMPLATE
+ $(".trash").click(function(e){
+ e.preventDefault();
+ $(this).parent().remove();
+ });
+
+ // detect URLs in inputs - popup send to wayback machine
+ detectInputWebPage("detect_web_page");
+
+ // language tags handling
+ $('[lang]').each(function() {
+ modify_lang_inputs($(this));
+ });
+
+ // multiple languages final visualization
+ $('.info-item [xml\\:lang]').each(function(){
+ visualize_subrecord_literals($(this));
+ })
+
+ // visualize subrecords ('Subtemplate' fields)
+ $('.subtemplateField').each(function() {
+ visualize_subrecord($(this));
+ });
+
+ // display subtemplates ('Subtemplate' fields)
+ $("[data-subtemplate]").each(function() {
+ // get the class of the subtemplate and its fields
+ var subtemplateClass = $(this).attr("data-subtemplate");
+ var subtemplateFields = $('[data-class="'+subtemplateClass+'"]');
+ var subtemplateId = $(this).attr('id');
+ var fieldName = $(this).parent().prev().text();
+
+ /* if ($(this).hasClass("oneValue")) {
+ // merged subtemplates: hide the 'subtemplate' field then link it to its sub-fields
+ $(this).parent().parent().hide();
+ $(this).attr('type', 'hidden');
+ var formId = $('.corners form').attr('id'); // either 'recordForm' or 'modifyForm'
+ var allowDataReuse = $(this).hasClass('allowDataReuse');
+
+ // define a new id (timespan) for the subrecord or retrieve the existing one
+ let subformId = '';
+ var now = new Date().valueOf();
+ var timespanId = (now / 1000).toString().replace('.', '-');
+ if($(this).next('span').next('.hiddenInput').length) {
+ var existingSubform = $(this).next().next().val();
+ subformId = existingSubform.split(',')[0];
+ } else {
+ subformId = timespanId;
+ }
+ console.log(subtemplateFields);
+ // adapt the subrecord's input fields to the defined schema
+ subtemplateFields.each(function() {
+ if (allowDataReuse) {
+ console.log(this);
+ var rdfProperty = $(this).attr('data-property');
+ console.log(rdfProperty)
+ var upperLevelClass = $('[data-subtemplate="'+$(this).attr('data-class')+'"]').attr('data-class');
+ console.log(upperLevelClass)
+ if ($('[data-class="'+upperLevelClass+'"][data-property="'+rdfProperty+'"]').length > 0) {
+ $(this).parent().parent().hide();
+ }
+ }
+ var baseId = $(this).attr('id').split('_')[0];
+
+ // subrecord's input fields and the main language hidden input must be like: 'inputFieldID_subrecordID'
+ if (formId === 'recordForm') {
+ // modify the main language id for primary keys
+ if ($(this).hasClass('disambiguate')) {
+ var mainLangInputId = baseId+'_mainLang_'+subformId;
+ $('#'+baseId+'_mainLang').attr('id', mainLangInputId);
+ $('#'+mainLangInputId).attr('name',mainLangInputId);
+ }
+ // modify sub-fields' ids
+ $(this).attr('id', $(this).attr('id')+"_"+subformId);
+ $(this).attr('name', $(this).attr('id'));
+ $(this).attr('data-subform',subformId);
+ }
+
+ // modify show_language onclick (literal input fields)
+ $('#'+'languages-'+baseId).find('a').each(function() {
+ var onclickAttr = $(this).attr('onclick');
+ var regex = /'([^"]*)'/g;
+ var originalIdExtended = onclickAttr.match(regex)[0];
+ var originalId = originalIdExtended.substring(1, originalIdExtended.length-1)
+ $(this).attr('onclick', onclickAttr.replace(originalId, originalId+'_'+subformId));
+ });
+ });
+
+ // associate the 'Subtemplate' input field to its Subrecord
+ var hiddenSubrecordLink = $('
');
+ $('#'+formId).append(hiddenSubrecordLink);
+
+ // check if a subrecord is already associated with this 'subtemplate' field and fill the subtemplate with its data
+ if(subformId !== timespanId) {
+ $(this).next().hide();
+ $(this).next().next().remove();
+ var currentUrl = window.location.href.split("/");
+ var currentAction = currentUrl[currentUrl.length-1].split("-")[0]; // modify or review
+ var requestUrl = '/'+currentAction+"-"+subformId;
+ $.ajax({
+ type:'GET',
+ url:requestUrl,
+ dataType:"html",
+ success:function(data) {
+ subtemplateFields.each(function() {
+ var subtemplateFieldId = $(this).attr('id');
+ var idRoot = subtemplateFieldId.split('_')[0];
+ var cloneElements = $(data).find('[id^="'+idRoot+'"]').parent();
+ let mainLang = '';
+ if (cloneElements.find('#'+idRoot+'_mainLang').length) { mainLang = cloneElements.find('#'+idRoot+'_mainLang').val().toUpperCase()}
+ cloneElements.find('input').each(function() {
+ cloneId = $(this).attr('name')+'_'+subformId;
+ $(this).attr('name', cloneId);
+ $(this).attr('id', cloneId);
+ $(this).addClass('subrecord-field');
+ $(this).attr('data-subform', subformId)
+ });
+ $(this).parent().replaceWith(cloneElements);
+ cloneElements.find('[lang]').each(function() {
+ modify_lang_inputs($(this));
+ });
+ cloneElements.parent().find('.main-lang').removeClass('main-lang');
+ cloneElements.parent().find('[title="text language: '+mainLang+'"]').addClass('main-lang');
+ });
+ }
+ });
+ }
+ } */
+ /* else { */
+ // multiple subtemplates
+ $(this).hide();
+
+ // get the class of the subtemplate and hide it, then clone each field before creating a subrecord
+ subtemplateFields.each(function() {
+ // mark the original input fields to be easily recognised and retrieved when necessary
+ // hide them to not be displayed in the back-end input dictionary
+ $(this).addClass('original-subtemplate');
+ $(this).parent().parent().hide();
+ var name = $(this).parent().prev().find('span').text().trim().replace('**','');
+ $('.fields-list li').each(function() {
+ if ($(this).text().trim() === name) {
+ $(this).remove();
+ }
+ });
+ console.log(name)
+ });
+ // get the display name assigned to the 'Subtemplate' field and create a button for adding new subrecords
+ var fieldName = $(this).parent().prev().text();
+ const createBtn = $('
playlist_add Define a new'+fieldName+' ');
+ createBtn.on('click', function() {
+ createSubrecord(subtemplateClass,fieldName,createBtn)
+ });
+
+ if ($(this).hasClass("oneValue")) {
+ let subformId;
+ var now = new Date().valueOf();
+ var timespanId = (now / 1000).toString().replace('.', '-');
+ if($(this).next('span').next('.hiddenInput').length) {
+ var existingSubform = $(this).next().next().val();
+ subformId = existingSubform.split(',')[0];
+ } else {
+ subformId = timespanId;
+ }
+
+ var hiddenSubrecordLink = $('
');
+ $('#modifyForm, #recordForm').append(hiddenSubrecordLink);
+
+ $(this).after(createBtn);
+ createSubrecord(subtemplateClass,fieldName,createBtn,subformId,"oneValue");
+ createBtn.remove();
+
+
+ } else {
+ $(this).after(createBtn);
+ }
+
+
+ // create hidden fields to store subrecords information when loading a previously created Record (only in modify/review page)
+ if ($('.corners form').attr('id') === "modifyForm") {
+ var subtemplateFieldId = $(this).attr('id');
+ var subrecords = "";
+ $('[data-input="'+subtemplateFieldId+'"').each(function() {
+ subrecords+=$(this).attr('id')+";"+$(this).text()+",";
+ })
+ $('#modifyForm').append('
');
+
+ }
+ /* } */
+ });
+
+ $('textarea').each(function() {
+ $(this).on('click', nlpText($(this).attr('id')))
+ })
+
+});
+
+$(window).on('resize', function() {
+
+ $('.tips-div').each(function() {
+ var target = $(this).attr('data-target');
+ var yasqeObj = $('#'+target+'>.yasqe');
+ var offset = yasqeObj.offset();
+ var left = offset.left;
+ var top = offset.top;
+ var width = yasqeObj.width()+1;
+ var height = yasqeObj.height();
+
+ $(this).css({
+ left: left+"px",
+ top: top+"px",
+ width: width+"px",
+ height: height+"px",
+ position: "absolute",
+ "z-index": "5",
+ "background-color": "white",
+ border: "1px solid #D1D1D1",
+ })
+ })
+});
+
+
+/////////////////////////
+// MULTIPLE LANGUAGES ///
+/////////////////////////
+function modify_lang_inputs(el) {
+ var base_id = $(el).attr('id').split('_')[0];
+
+
+ // check if it's first of type
+ if ($(el).is('[lang]:first-of-type')) {
+ $(el).parent().prev().append($('
translate '));
+
+ if (!($(el).hasClass('subrecord-field'))) {
+ var new_id = base_id+"_"+$(el).attr('lang');
+ $(el).attr('name',new_id);
+ $(el).attr('id',new_id);
+ var lang = $(el).attr('lang').toUpperCase();
+ const languages_list = $('
');
+ const first_lang = $('
'+lang+' ');
+
+ // primary keys: specify main language
+ if ($(el).hasClass('disambiguate')) {
+ if ($('#'+base_id+'_mainLang').length == 0) {
+ const hidden_main_lang = $('
')
+ $(el).after(hidden_main_lang);
+ first_lang.addClass('main-lang');
+ } else if ($('#'+base_id+'_mainLang').val() === $(el).attr('lang')) {
+ first_lang.addClass('main-lang');
+ }
+ }
+ languages_list.append(first_lang);
+ $(el).before(languages_list);
+ }
+ }
+ else {
+
+ // language tags handling: other lang (only in modify and review)
+ const main_lang = $('#'+base_id+'_mainLang').val();
+ var lang = $(el).attr('lang');
+ const languages_list = $('#languages-'+base_id);
+ const other_lang = $('
'+lang.toUpperCase()+' ');
+ if ($(el).hasClass('disambiguate') && lang===main_lang) {
+ var first_lang = languages_list.find('i').next('a');
+ other_lang.addClass('main-lang');
+ other_lang.addClass('selected-lang');
+ first_lang.removeClass('selected-lang');
+ first_lang.before(other_lang);
+ $('#'+base_id+'_'+first_lang.text().toLowerCase()).hide();
+ $(el).show();
+ } else {
+ languages_list.append(other_lang);
+ $(el).hide();
+ }
+ label_section.append(languages_list);
+ }
+}
+
+function language_form(el) {
+ if ($('#lang-form').length > 0) {
+ $('#lang-form').remove()
+ } else {
+ var height = $(el).offset().top + 32 + "px";
+ var left = $(el).offset().left - 13 +"px";
+ var current_lang = $(el).parent().next().find('.selected-lang').text().toLowerCase();
+ var input = $(el).parent().next().find('textarea, input').filter(':visible');
+ var field_id = input.attr('id');
+
+ // set a variable to modify the subrecord's list of fields in case the textbox is part of a subtemplate
+ var subform = $('#'+field_id).attr('data-subform');
+ var modify_subform = subform ? subform : null
+
+
+ const lang_form = $('
');
+ const change_language = $('
');
+ const add_language = $('
')
+ const main_lang = $('
');
+ main_lang.find('select').on('change', function() {change_main_lang(this,modify_subform)});
+ const remove_lang = $('
');
+ remove_lang.on('click', function() {remove_lang(this,modify_subform)});
+ $.ajax({
+ type: 'GET',
+ url: "https://raw.githubusercontent.com/mattcg/language-subtag-registry/master/data/json/registry.json",
+ dataType: 'json',
+ success: function(data) {
+ let languageObjects = [];
+ // Loop through the array of objects
+ for (let obj of data) {
+ // Check if the object has "type": "language"
+ if (obj.Type === "language" && !("Deprecated" in obj)) {
+ var lang = obj.Description[0];
+ var tag = obj.Subtag;
+
+ // prepare the 'Change Language' and 'Add Language' options
+ var change_select = $("
"+lang+" ("+tag+") ");
+ var add_select = $("
"+lang+" ("+tag+") ");
+ change_select.on("click", function() {
+ changeCurrentLanguage(this,modify_subform);
+ });
+ add_select.on("click", function() {
+ console.log(modify_subform)
+ addNewLanguage(this,modify_subform)
+ });
+ if (tag === current_lang) {
+ change_language.find('input').attr('placeholder',lang+' ('+tag+')');
+ change_select.addClass('current-lang');
+ change_language.find('div').prepend(change_select);
+ add_language.find('div').append(add_select);
+ } else {
+ change_language.find('div').append(change_select);
+ add_language.find('div').append(add_select);
+ }
+ }
+ }
+ // hide the input options and append their parent divs to #lang-form
+ change_language.find('div').hide();
+ add_language.find('div').hide();
+ lang_form.append(change_language, add_language);
+
+ // prepare the dropdown for selecting the main language of a primary key input field
+ if (input.hasClass('disambiguate')) {
+ var current_languages = $(el).parent().next().find('.lang-item');
+ current_languages.each(function() {
+ var subtag = $(this).text().toLowerCase();
+ var extended_language = change_language.find('[href="#'+subtag+'"]').attr('lang');
+ var lang_option = $('
'+extended_language+' ('+subtag+') ');
+ if (subtag == $(el).parent().find('.main-lang').text().toLowerCase()) {
+ lang_option.attr('selected', 'selected');
+ }
+ main_lang.find('select').append(lang_option);
+ })
+ lang_form.append(main_lang);
+ };
+
+ lang_form.append(remove_lang)
+ lang_form.css({'top':height, 'left':left});
+ $('main').append(lang_form);
+ },
+ error: function(data) {
+ console.log("Resource not available");
+ }
+ })
+ }
+}
+
+function activateFilter(el){
+ if ($(el).next('div').attr('style') == 'display: none;') {
+ $(el).next('div').show();
+ $(el).keyup(function(){
+ let input_val = $(el).val();
+ if (input_val !== '') {
+ $(el).next('div').find('a:not([lang*="' + input_val + '"])').hide();
+ $(el).next('div').find('a[lang*="' + input_val + '"]').show();
+ } else {
+ $(el).next('div').find('a').show();
+ }
+ })
+ } else {
+ $(el).next('div').hide();
+ }
+}
+
+function addNewLanguage(el,record) {
+ console.log(record)
+ var new_lang = $(el).attr('href').replace("#","");
+ var last_lang_id = $(el).parent().parent().parent().attr('data-input');
+ const new_lang_input = $('#'+last_lang_id).clone();
+ var new_lang_input_id = last_lang_id.split('_')[0] + "_" + new_lang;
+ if (record) {new_lang_input_id += '_' + record}
+ $('#languages-'+last_lang_id.split('_')[0]).find('.selected-lang').removeClass('selected-lang');
+ $('#languages-'+last_lang_id.split('_')[0]).append("
"+new_lang.toUpperCase()+" ");
+
+ new_lang_input.attr('id', new_lang_input_id);
+ new_lang_input.attr('name', new_lang_input_id);
+ new_lang_input.attr('lang', new_lang);
+ new_lang_input.val('');
+ if (new_lang_input.is('textarea')) {
+ new_lang_input.on('click', function() {
+ nlpText(new_lang_input_id);
+ });
+ }
+
+
+ $('[id^="'+last_lang_id.split('_')[0]+'"]').hide();
+ $('#'+last_lang_id).after(new_lang_input);
+ $(el).parent().parent().parent().remove();
+}
+
+function changeCurrentLanguage(el,record) {
+ var new_lang = $(el).attr('href').replace("#","");
+ var id = $(el).parent().parent().parent().attr('data-input');
+ var current_lang = $('#languages-'+id.split('_')[0]).find('.selected-lang').eq(0);
+ var current_lang_field_id = id.split('_')[0] +"_"+current_lang.text().toLowerCase();
+ if (current_lang.text().toLowerCase() !== new_lang) {
+ var new_id = id.split('_')[0] + '_' + new_lang;
+ if (record) {
+ new_id += '_' + record;
+ current_lang_field_id += '_' + record;
+ console.log(current_lang_field_id)
+ }
+ var title = 'text language: '+new_lang.toUpperCase();
+ current_lang.attr('title',title);
+ current_lang.attr('onclick','show_lang("'+new_id+'")');
+ current_lang.text(new_lang.toUpperCase());
+ $('#'+current_lang_field_id).attr('name',new_id);
+ $('#'+current_lang_field_id).attr('id',new_id);
+ $('#'+new_id).attr('lang',new_lang);
+ $(el).parent().parent().parent().remove();
+
+ // check whether this value is the primary key of a subrecord
+ $('[value*="'+id+'"]').each(function() {
+ var new_value = $(this).val().replace(id,new_id);
+ $(this).val(new_value);
+ });
+ }
+ if (current_lang.hasClass('main-lang')) {
+ var main_lang_id = '#'+id.split('_')[0]+'_mainLang';
+ if (record) {
+ main_lang_id += '_' + record;
+ }
+ $(main_lang_id).val(new_lang);
+ }
+
+
+}
+
+function change_main_lang(el,record) {
+ var id = $(el).parent().parent().attr('data-input');
+ let field_base = id.split('_')[0];
+ $('#languages-'+field_base).find('.main-lang').removeClass('main-lang');
+ var new_main_lang = $(el).val();
+ $('#languages-'+field_base).find('[title="text language: '+new_main_lang.toUpperCase()+'"]').addClass('main-lang');
+ $(el).parent().parent().remove();
+ var main_lang_input_id = '#'+field_base+'_mainLang';
+ if (record) {main_lang_input_id+='_'+record}
+ $(main_lang_input_id).val(new_main_lang);
+}
+
+function removeCurrentLanguage(el,record) {
+ var current_field = $(el).parent().parent().attr('data-input');
+ let field_base = current_field.split('_')[0];
+ var current_lang_tag = $('#languages-'+field_base).find('.selected-lang');
+ if (current_lang_tag.next('a').length > 0) {
+ current_lang_tag.next('a').addClass('selected-lang');
+ var next_lang = current_lang_tag.next('a').text().toLowerCase();
+ current_lang_tag.remove();
+ $('#'+current_field).remove();
+ console.log('#'+field_base+'_'+next_lang)
+ $('#'+field_base+'_'+next_lang).show();
+ } else if (current_lang_tag.prev('a').length > 0) {
+ current_lang_tag.prev('a').addClass('selected-lang');
+ var prev_lang = current_lang_tag.prev('a').text().toLowerCase();
+ current_lang_tag.remove();
+ $('#'+current_field).remove();
+ var show_lang_id = '#'+field_base+'_'+prev_lang;
+ if (record) {show_lang_id+='_'+record}
+ $(show_lang_id).show();
+ } else {
+ alert('Not allowed. Change current language, instead');
+ }
+ $('[data-input="'+current_field+'"]').remove();
+}
+
+function show_lang(field_id) {
+ let field_base = field_id.split('_')[0];
+ $('[id^="'+field_base+'_"]').hide();
+ $('#'+field_id).show();
+ var target_lang = field_id.split('_')[1];
+ $('#languages-'+field_base).find('.selected-lang').removeClass('selected-lang');
+ console.log('[title="text language: '+target_lang.toUpperCase()+'"]')
+ $('#languages-'+field_base).find('[title="text language: '+target_lang.toUpperCase()+'"]').addClass('selected-lang')
+}
+
+/////////////////////////
+//// SUBRECORDS VIEW ////
+/////////////////////////
+function visualize_subrecord(el) {
+ var hide = el.hasClass('oneValue');
+ var subtemplate_values = el.find('p');
+ subtemplate_values.each(function() {
+ var externalLink = $(this).find('a').eq(0);
+
+ if (!hide) {
+ // organize subrecords as an accordion
+ $(this).addClass('subtemplateValue')
+ $(this).append('
')
+ externalLink.prepend("
")
+
+ // show and hide inner values
+ $(this).bind('click', function(e){
+ if (!($(e.target).parent().context.localName == "i")) {
+ e.preventDefault();
+ if ($(e.target).parent().is($(this))) {
+ $($(e.target).parent()).find('div').eq(0).toggleClass('hidden-subrecord');
+ $(e.target).parent().toggleClass('subtemplateValueOpen');
+ }
+ }
+ })
+ } else {
+ $(this).parent().hide();
+ }
+
+ var calledValue = $(this);
+ var calledRecord = externalLink.attr('href').replace("term", "view");
+ $.ajax({
+ type:'GET',
+ url:calledRecord,
+ dataType:"html",
+ success:function(data) {
+ var calledRecordData = $(data).find('.articleBox').find('.col-md-8.row').find('section');
+
+ if (!hide) {
+ calledRecordData.each(function() {
+ var cls = $(this).attr('class');
+ var new_cls = cls.replace("col-md-5", "col-md-12")
+ $(this).attr('class', new_cls);
+ $(this).find('.wikiEntity').append(wdImgIcon);
+ $(this).find('.geoEntity').append(geoImg);
+ $(this).find('.viafEntity').append(viafImg);
+ $(this).find('[xml\\:lang]').each(function() {
+ visualize_subrecord_literals($(this));
+ })
+ })
+ calledValue.append($('
').append(calledRecordData));
+ } else {
+ calledRecordData.find('.wikiEntity').append(wdImgIcon);
+ calledRecordData.find('.geoEntity').append(geoImg);
+ calledRecordData.find('.viafEntity').append(viafImg);
+ calledRecordData.find('[xml\\:lang]').each(function() {
+ visualize_subrecord_literals($(this));
+ })
+ calledValue.parent().after(calledRecordData)
+ }
+
+
+
+ var new_values = calledValue.find('.subtemplateField');
+ new_values.each(function() {
+ visualize_subrecord($(this));
+ });
+
+ },
+ error:function() {
+ console.log("Error: requested resource is not available");
+ }
+ })
+ });
+}
+
+function visualize_subrecord_literals(el) {
+ var lang = $(el).attr('xml:lang');
+ const language_item = $('
'+lang.toUpperCase()+' ');
+ if ($(el).prev('p').length != 1) {
+ const languages_list = $('
');
+ language_item.addClass('selected-lang');
+ $(el).before(languages_list.append(language_item));
+ } else {
+ $(el).parent().find('.info-language').append(language_item);
+ $(el).hide();
+ }
+ language_item.on('click', function() {
+ var new_lang = $(this).text().toLowerCase();
+ $(this).parent().parent().find('p').hide();
+ $(this).parent().parent().find('p[xml\\:lang="'+new_lang+'"]').show();
+ $(this).parent().find('.selected-lang').removeClass('selected-lang');
+ $(this).addClass('selected-lang');
+ });
+}
+
+//////////////
+// BACKEND //
+//////////////
+
+function validateTemplateClass() {
+ // validate
+ var class_name = $("input[name='class_name']").val();
+ var class_uris = $('#uri-container').find('input');
+ if (class_name == "" || class_uris.length == 0 ) {
+ alert("Name and URI must be filled out");
+
+ return false;
+ } else if ($("input[name='class_name']").hasClass('error-input')) {
+ console.log("B")
+ alert("Check your template Name");
+
+ return false;
+ } else {
+ document.getElementById('selectTemplateClass').submit();
+ }
+ // lookup for previous classes
+ // redirect to page - modify app py to accept object in Template class
+
+};
+
+////////////////
+// ADD RECORD //
+////////////////
+
+function colorForm() {
+ $('.searchWikidata').each( function() {
+ if ($(this).next('span').length > 0) {
+ $(this).removeAttr('placeholder');
+ $(this).parent().prev('.label').css('color','lightgrey');
+ $(this).parent().prev('.label').children('img').css('opacity','0.5');
+ $(this).nextAll('span').css('color','lightgrey').css('border-color','lightgrey');
+
+ $($(this).parent().parent()).hover(function(){
+ $(this).children().addClass('color_hover');
+ $(this).children().children('span').addClass('color_hover').addClass('bkg_hover');
+ }, function() {
+ $(this).children().removeClass('color_hover');
+ $(this).children().children('span').removeClass('color_hover').removeClass('bkg_hover');
+ });
+
+ } else {
+ $(this).parent().prev('.label').css('color','black');
+ $(this).parent().prev('.label').children('img').css('opacity','1');
+ $(this).nextAll('span').css('color','black').css('border-color','black');
+ };
+ });
+
+ $('.freeText').each( function() {
+ if ($(this).val().length > 0) {
+ $(this).parent().prev('.label').css('color','lightgrey');
+ $(this).parent().prev('.label').children('img').css('opacity','0.5');
+ $(this).css('color','lightgrey');
+ $($(this).parent().parent()).hover(function(){
+ $(this).children().addClass('color_hover');
+ $(this).children().children().addClass('color_hover');
+ }, function() {
+ $(this).children().removeClass('color_hover');
+ $(this).children().children().removeClass('color_hover');
+ });
+ } else {
+ $(this).parent().prev('.label').css('color','black');
+ $(this).parent().prev('.label').children('img').css('opacity','1');
+ $(this.value).css('color','black');
+ };
+ });
+};
+
+// delay a function
+function throttle(f, delay){
+ var timer = null;
+ return function(){
+ var context = this, args = arguments;
+ clearTimeout(timer);
+ timer = window.setTimeout(function(){
+ f.apply(context, args);
+ },
+ delay || 300);
+ };
+};
+
+
+//////////////////////////////
+// Query services functions //
+//////////////////////////////
+
+// Ancillary function: call VIAF API
+function callViafAPI(querySubstring, doneCallback){
+ var requestUrl = "https://viaf.org/viaf/AutoSuggest?query=" + querySubstring + "&callback=?";
+ $.getJSON(requestUrl, doneCallback);
+}
+
+// Ancillary function: make a SPARQL query (Wikidata/catalogue, for advanced search only)
+function makeSPARQLQuery( endpointUrl, sparqlQuery, doneCallback ) {
+ var settings = {
+ headers: {
+ Accept: 'application/sparql-results+json'},
+ data: { query: sparqlQuery }
+ };
+ return $.ajax( endpointUrl, settings ).then( doneCallback );
+}
+
+// Ancillary function: set #searchresult div position (css)
+function setSearchResult(searchterm){
+ var position = $('#'+searchterm).position();
+ var leftpos = position.left+80;
+ var offset = $('#'+searchterm).offset();
+ var height = $('#'+searchterm).height();
+ var width = $('#'+searchterm).width();
+ var top = offset.top + height + "px";
+ var right = offset.left + width + "px";
+
+ $('#searchresult').css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':'600px',
+ 'max-height': '300px',
+ 'overflow-y': 'auto'
+ });
+ $("#searchresult").empty();
+}
+
+// SEARCH GEONAMES
+// search in geonames and my catalogue
+function searchGeonames(searchterm) {
+ // wikidata autocomplete on keyup
+ $('#'+searchterm).keyup(function(e) {
+ $("#searchresult").show();
+ var q = $('#'+searchterm).val();
+
+ $.getJSON("http://api.geonames.org/searchJSON", {
+ q: q,
+ username: "palread",
+ maxRows: 10,
+ lang: "en",
+ uselang: "en",
+ format: "json",
+ },
+ function(data) {
+ // autocomplete positioning;
+ var offset = $('#'+searchterm).offset();
+ var leftpos = offset.left+15;
+ var height = $('#'+searchterm).height();
+ var top = offset.top + height + 15 + "px";
+ var max_width = '600px';
+
+
+ $('#searchresult').css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':max_width,
+ });
+ $("#searchresult").empty();
+
+ // catalogue lookup in case nothing is found
+ if(!data.geonames.length){
+ $("#searchresult").append("
No matches in Geonames... looking in the catalogue
");
+ // remove messages after 3 seconds
+ setTimeout(function(){
+ if ($('.noresults').length > 0) {
+ $('.noresults').remove();
+ }
+ }, 3000);
+
+ var query = "prefix bds:
select distinct ?s ?o ?desc "+inGraph+" where { ?s rdfs:label ?o . OPTIONAL { ?s rdfs:comment ?desc} . ?o bds:search '"+q+"*' .}"
+ var encoded = encodeURIComponent(query)
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint+'?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json'},
+ success: function(returnedJson) {
+ // $("#searchresult").empty();
+ console.log(returnedJson);
+ // if (!returnedJson.length) {
+ // // $("#searchresult").empty();
+ // $("#searchresult").append("No results in Wikidata and catalogue
");
+ // // remove messages after 3 seconds
+ // setTimeout(function(){ if ($('.noresults').length > 0) { $('.noresults').remove(); } }, 3000);
+ // };
+
+ for (i = 0; i < returnedJson.results.bindings.length; i++) {
+ var myUrl = returnedJson.results.bindings[i].s.value;
+ // exclude named graphs from results
+ if ( myUrl.substring(myUrl.length-1) != "/") {
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1)
+ if (returnedJson.results.bindings[i].desc !== undefined) {var desc = '- '+returnedJson.results.bindings[i].desc.value} else {var desc = ''}
+ $("#searchresult").append("");
+ };
+ };
+
+ // add tag if the user chooses an item from the catalogue
+ $('a[data-id^="'+base+'"]').each( function() {
+ $(this).bind('click', function(e) {
+ e.preventDefault();
+ var oldID = this.getAttribute('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#'+searchterm).after(""+oldLabel+" ");
+ $("#searchresult").hide();
+ $('#'+searchterm).val('');
+ });
+
+ });
+
+ }
+ });
+ // end my catalogue
+ };
+
+ // fill the dropdown
+ $.each(data.geonames, function(i, item) {
+ $("#searchresult").append("");
+
+ // add tag if the user chooses an item from wd
+ $('a[data-id="'+ item.geonameId+'"]').each( function() {
+ $(this).bind('click', function(e) {
+ e.preventDefault();
+ $('#'+searchterm).after(""+item.name+" ");
+ $("#searchresult").hide();
+ $('#'+searchterm).val('');
+ //colorForm();
+ });
+
+ });
+ });
+ }
+ );
+ });
+
+ // if the user presses enter - create a new entity
+ $('#'+searchterm).keypress(function(e) {
+ if(e.which == 13) {
+ e.preventDefault();
+ var now = new Date().valueOf();
+ var newID = 'MD'+now;
+ if (!$('#'+searchterm).val() == '') {
+ $('#'+searchterm).after(""+$('#'+searchterm).val()+" ");
+ };
+ $("#searchresult").hide();
+ $('#'+searchterm).val('');
+ //colorForm();
+ };
+ });
+};
+
+// SEARCH CATALOGUE
+// search bar menu
+function searchCatalogue(searchterm) {
+ $('#'+searchterm).keyup(function(e) {
+ $("#searchresultmenu").show();
+ var q = $('#'+searchterm).val();
+ var query = "prefix bds: select distinct ?s ?o "+inGraph+" where { ?o bds:search '"+q+"*'. ?o bds:minRelevance '0.3'^^xsd:double . ?s rdfs:label ?o ; a ?class .}"
+ var encoded = encodeURIComponent(query)
+ if (q == '') { $("#searchresultmenu").hide();}
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint+'?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json; charset=utf-8'},
+ success: function(returnedJson) {
+ $("#searchresultmenu").empty();
+ // autocomplete positioning
+ setSearchResult(searchterm);
+
+ if (!returnedJson.length) {
+ $("#searchresultmenu").empty();
+ var nores = "Searching...
";
+ $("#searchresultmenu").append(nores);
+ // remove messages after 1 second
+ setTimeout(function(){
+ if ($('.noresults').length > 0) {
+ $('.noresults').remove();
+ }
+ }, 1000);
+ };
+
+ for (i = 0; i < returnedJson.results.bindings.length; i++) {
+ var myUrl = returnedJson.results.bindings[i].s.value;
+ // exclude named graphs from results
+ if ( myUrl.substring(myUrl.length-1) != "/") {
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1)
+ $("#searchresultmenu").append("");
+ };
+ };
+
+ }
+ });
+ });
+};
+
+// search catalogue through advanced triple patterns
+function searchCatalogueAdvanced(searchterm) {
+ let newSparqlQuery = "";
+ var rawQuery = query_templates[searchterm];
+ var sparqlQuery = rawQuery.replaceAll('<','<').replaceAll('>','>').replaceAll(/"/g, '"');
+ var endpointUrl = myPublicEndpoint;
+
+ $('#'+searchterm).off('keyup').on('keyup', function() {
+ $("#searchresult").show();
+ setSearchResult(searchterm);
+ $("#searchresult").empty();
+
+ var value = $('#'+searchterm).val().toLowerCase();
+ newSparqlQuery = sparqlQuery.replace("insertQueryTerm",value);
+
+ if (value.length>0) {
+ makeSPARQLQuery( endpointUrl, newSparqlQuery, function( data ) {
+ $("#searchresult").empty();
+ data.results.bindings.forEach(function(obj,idx) {
+ let idSplit = obj.item.value.split('/');
+ var catalogueId = idSplit[idSplit.length-1];
+ if ($('#searchresult [data-id="'+catalogueId+'"]').length == 0) {
+ var newItemDiv = $('');
+ newItemDiv.find('[data-id]').bind('click', function(e) {
+ e.preventDefault();
+ var oldID = $(this).attr('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).after("" + oldLabel + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+ $("#searchresult").append(newItemDiv);
+ };
+ });
+ if ($("#searchresult div").length == 0 && value.length <= 3){
+ $("#searchresult").append('No matches found: try to type more characters
')
+ } else if ($("#searchresult div").length == 0){
+ $("#searchresult").append('No matches found
')
+ };
+
+ });
+ }
+ });
+
+}
+
+// search catalogue's records belonging to a desired class
+function searchCatalogueByClass(searchterm) {
+ // get the required class
+ var resource_class = $('#'+searchterm).attr('subtemplate');
+
+ // get an array of subrecords created within the same webpage and not saved yet:
+ // they must belong to the same required class
+ var yet_to_save_keys = [];
+ var yet_to_save_resources = [];
+ $('.disambiguate[class*="' + resource_class + '"]').each(function() {
+ yet_to_save_keys.push($(this).val());
+ var key_id = $(this).attr('id');
+ var subrecord = $('input[type="hidden"][value*="'+key_id+'"]');
+ yet_to_save_resources.push(subrecord.attr('id'));
+ });
+
+ // on key up look for suggestions based on the new input string
+ $('#'+searchterm).keyup(function(e) {
+ var useful_yet_to_save_keys = yet_to_save_keys.filter(function(value) {
+ return value.toLowerCase().includes($('#'+searchterm).val().toLowerCase()) && value.trim() !== '';
+ });
+
+ // autocomplete positioning;
+ var offset = $('#'+searchterm).offset();
+ var leftpos = offset.left+15;
+ var height = $('#'+searchterm).height();
+ var top = offset.top + height + "px";
+ var max_width = '600px';
+ $('#searchresult').css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':max_width,
+ });
+ $("#searchresult").show();
+
+ // prepare the query
+ var query_term = $('#'+searchterm).val();
+ var query = "prefix bds: select distinct ?s ?o where { ?o bds:search '"+query_term+"*'. ?o bds:minRelevance '0.3'^^xsd:double . ?s rdfs:label ?o ; a <"+resource_class+"> .}"
+ var encoded = encodeURIComponent(query);
+
+ // send the query request to the catalogue
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint + '?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json' },
+ success: function (returnedJson) {
+ $("#searchresult").empty();
+ var url = myPublicEndpoint + '?query=' + encoded
+ // show results
+ if (!returnedJson.results.bindings.length) {
+ $("#searchresult").append("No results in catalogue
");
+ // remove messages after 3 seconds
+ setTimeout(function(){ if ($('.noresults').length > 0) { $('.noresults').remove(); } }, 3000);
+ } else {
+ for (let i = 0; i < returnedJson.results.bindings.length; i++) {
+ // get the URL and the label for each retrieved element
+ var myUrl = returnedJson.results.bindings[i].s.value;
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1);
+ $("#searchresult").append("");
+ }
+ }
+
+ // add tag if the user chooses an item from the catalogue
+ $('a[data-id^="' + base + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var oldID = this.getAttribute('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).next('i').after(""+oldLabel+" ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ if ($('[name="'+searchterm+'-subrecords"]').length) {
+ $('[name="'+searchterm+'-subrecords"]').val($('[name="'+searchterm+'-subrecords"]').val()+","+oldID+";"+oldLabel);
+ } else {
+ const new_sub = $(" ")
+ $('#recordForm').append(new_sub)
+ }
+ });
+
+ });;
+
+ // once external resources have been added, include newly created resources (yet to be saved)
+ for (let j = 0; j < useful_yet_to_save_keys.length; j++) {
+ var resource_id = yet_to_save_resources[j];
+ var resource_name = useful_yet_to_save_keys[j];
+ $('#searchresult').append("")
+ }
+
+ // add tag if the user chooses an item from yet to save resources
+ $('a[target]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var target = $(this).attr('target');
+ var label = $(this).text();
+ $('#' + searchterm).next('i').after("" + label + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ if ($('[name="'+searchterm+'-subrecords"]').length) {
+ $('[name="'+searchterm+'-subrecords"]').val($('[name="'+searchterm+'-subrecords"]').val()+","+target+";"+label);
+ } else {
+ const new_sub = $(" ")
+ $('#recordForm').append(new_sub)
+ }
+ });
+
+ });
+ },
+ error: function (error) {
+ reject(error);
+ }
+ });
+ })
+};
+
+// a function to look through the catalogue while querying Wikidata and VIAF
+function searchCatalogueIntermediate(q) {
+ return new Promise(function(resolve, reject) {
+ var query = "prefix bds: select distinct ?s ?o "+inGraph+" where { ?o bds:search '"+q+"*'. ?o bds:minRelevance '0.3'^^xsd:double . ?s rdfs:label ?o ; a ?class .}"
+ var encoded = encodeURIComponent(query);
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint + '?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json' },
+ success: function (returnedJson) {
+ // $("#searchresult").empty();
+ var url = myPublicEndpoint + '?query=' + encoded
+ console.log(url);
+ console.log(returnedJson);
+ // if (!returnedJson.length) {
+ // // $("#searchresult").empty();
+ // $("#searchresult").append("No results in Wikidata and catalogue
");
+ // // remove messages after 3 seconds
+ // setTimeout(function(){ if ($('.noresults').length > 0) { $('.noresults').remove(); } }, 3000);
+ // };
+ resolve(returnedJson.results.bindings);
+ },
+ error: function (error) {
+ reject(error);
+ }
+ });
+ });
+}
+
+// SEARCH WIKIDATA (includes SEARCH VIAF)
+// search in wikidata, viaf and my catalogue
+function searchWD(searchterm) {
+ // wikidata autocomplete on keyup
+ $('#'+searchterm).keyup(function(e) {
+ $("#searchresult").show();
+ var q = $('#'+searchterm).val();
+
+ $.getJSON("https://www.wikidata.org/w/api.php?callback=?", {
+ search: q,
+ action: "wbsearchentities",
+ language: "en",
+ uselang: "en",
+ format: "json",
+ strictlanguage: true,
+ limit: 5,
+ },
+ function(data) {
+ // autocomplete positioning;
+ var height = $('#'+searchterm).height();
+ var offset = $('#'+searchterm).offset();
+ var leftpos = offset.left+15;
+ var top = offset.top + height + 15 + "px";
+ var max_width = '600px';
+ console.log(max_width);
+
+ $('#searchresult').css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':max_width,
+ });
+ $("#searchresult").empty();
+
+ // VIAF lookup in case nothing is found in Wikidata
+ if (!data.search || !data.search.length) {
+ callViafAPI(q, function (viafData) {
+ if (viafData.result) {
+ // to avoid repetitions of the same element: $("#searchresult").find("[data-id="+item.viafid+"]").length === 0
+ $.each(viafData.result, function (i, item) {
+ if ($("#searchresult").find("[data-id="+item.viafid+"]").length === 0 && $("#searchresult > .viafitem").length <5) {
+ $("#searchresult").append(""); // no item.DESCRIPTION!
+
+ // add tag if the user chooses an item from viaf
+ $('a[data-id="' + item.viafid + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var input_name = (searchterm.split('_').length == 2) ? searchterm.split('')[0] + item.viafid + searchterm.split('_')[1] : searchterm + '-' + item.viafid;
+ $('#' + searchterm).next('div').append("" + item.term + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ //colorForm();
+ });
+ });
+
+ }
+ });
+ }
+ else if ($('#' + searchterm).val().length > 0) {
+ $("#searchresult").append("No matches in Wikidata and VIAF
");
+ // remove messages after 3 seconds
+ setTimeout(function () {
+ if ($('.noresults').length > 0) {
+ $('.noresults').remove();
+ }
+ }, 3000);
+ }
+ });
+ } else {
+ // fill the dropdown with WD results
+ $.each(data.search, function (i, item) {
+ $("#searchresult").append("");
+
+ // add tag if the user chooses an item from wd
+ $('a[data-id="' + item.title + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var input_name = (searchterm.split('_').length == 2) ? searchterm.split('_')[0] + "_" + item.title + "_" + searchterm.split('_')[1] : searchterm + '_' + item.title;
+ $('#' + searchterm).next('div').append("" + item.label + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ //colorForm();
+ });
+ });
+ });
+ }
+
+ // look for the query term in the catalogue (regardless of VIAF and WD results)
+ var cataloguePromise = searchCatalogueIntermediate(q)
+ cataloguePromise.then(function(catalogueEntries) {
+ $(".fromCatalogue").remove(); // to remove previously retrieved elements
+ for (i = 0; i < catalogueEntries.length; i++) {
+ var myUrl = catalogueEntries[i].s.value;
+ console.log(catalogueEntries)
+ // exclude named graphs from results
+ if (myUrl.substring(myUrl.length - 1) != "/") {
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1);
+ if (catalogueEntries[i].desc !== undefined) {
+ var desc = '- ' + catalogueEntries[i].desc.value;
+ } else {
+ var desc = '';
+ }
+ if ($("#searchresult div.wditem a[data-id='" + catalogueEntries[i].s.value + "']").length == 0) {
+ $("#searchresult").append("");
+ }
+ }
+ }
+
+ // add tag if the user chooses an item from the catalogue
+ $('a[data-id^="' + base + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var oldID = this.getAttribute('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).next('div').append("" + oldLabel + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+
+ });
+ })
+
+ // end catalogue
+
+ })
+ });
+
+ // if the user presses enter - create a new entity
+ $('#'+searchterm).keypress(function(e) {
+ if(e.which == 13) {
+ e.preventDefault();
+ var now = new Date().valueOf();
+ var newID = 'MD'+now;
+ if (!$('#'+searchterm).val() == '') {
+ $('#'+searchterm).after(""+$('#'+searchterm).val()+" ");
+ };
+ $("#searchresult").hide();
+ $('#'+searchterm).val('');
+ //colorForm();
+ };
+ });
+};
+
+// search Wikidata with SPARQL patterns
+function searchWDAdvanced(searchterm) {
+ let newSparqlQuery = "";
+ var rawQuery = query_templates[searchterm];
+ var sparqlQuery = rawQuery.replaceAll('<','<').replaceAll('>','>').replaceAll(/"/g, '"');
+ var endpointUrl = wikidataEndpoint;
+
+ $('#'+searchterm).off('keyup').on('keyup', function() {
+ $("#searchresult").show();
+ setSearchResult(searchterm);
+ var value = $('#'+searchterm).val().toLowerCase();
+ newSparqlQuery = sparqlQuery.replace("insertQueryTerm",value);
+
+
+ if (value.length>0) {
+ makeSPARQLQuery( endpointUrl, newSparqlQuery, function( data ) {
+ $("#searchresult").empty();
+ data.results.bindings.forEach(function(obj,idx) {
+ let idSplit = obj.item.value.split('/');
+ var wdId = idSplit[idSplit.length-1];
+ var newItemDiv = $('');
+ newItemDiv.find('[data-id]').bind('click', function(e) {
+ e.preventDefault();
+ var oldID = $(this).attr('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).after("" + oldLabel + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+ $("#searchresult").append(newItemDiv);
+ });
+ if (!data.results.bindings.length){
+ $("#searchresult").append('No matches in Wikidata: try to type more characters
');
+
+ // search for results in VIAF and CATALOGUE in case no result is returned by WD:
+ // VIAF:
+ callViafAPI(value, function(viafData) {
+ if (viafData.result) {
+ // to avoid repetitions of the same element: $("#searchresult").find("[data-id="+item.viafid+"]").length === 0
+ $.each(viafData.result, function (i, item) {
+ if ($("#searchresult").find("[data-id="+item.viafid+"]").length === 0 && $("#searchresult > .viafitem").length <5) {
+ $("#searchresult").append(""); // no item.DESCRIPTION!
+
+ // add tag if the user chooses an item from viaf
+ $('a[data-id="' + item.viafid + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var input_name = (searchterm.split('_').length == 2) ? searchterm.split('')[0] + item.viafid + searchterm.split('_')[1] : searchterm + '-' + item.viafid;
+ $('#' + searchterm).after("" + item.term + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+ });
+
+ }
+ });
+ }
+ });
+ // CATALOGUE:
+ var catalogueSearchPromise = searchCatalogueIntermediate(value)
+ catalogeRequest = catalogueSearchPromise.then(function(catalogueData) {
+ $(".fromCatalogue").remove(); // to remove previously retrieved elements
+ $.each(catalogueData, function (i, item) {
+ var myUrl = item.s.value;
+ // exclude named graphs from results
+ if (myUrl.substring(myUrl.length - 1) != "/") {
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1);
+ if (item.desc !== undefined) {
+ var desc = '- ' + catalogueEntries[i].desc.value;
+ } else {
+ var desc = '';
+ }
+ if ($("#searchresult div.wditem a[data-id='" + item.s.value + "']").length == 0) {
+ $("#searchresult").append("");
+ }
+ }
+ });
+
+ // add tag if the user chooses an item from the catalogue
+ $('a[data-id^="' + base + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var oldID = this.getAttribute('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).after("" + oldLabel + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+
+ });
+ })
+ }
+ });
+ }
+
+ });
+}
+
+// search for entities through SPARQL triple patterns in Wikidata and Catalogue
+function searchWDCatalogueAdvanced(searchterm){
+ let newSparqlQueryWD = "";
+ let newSparqlQueryCatalogue = "";
+ var rawQueryWD = query_templates[searchterm]['wikidata'];
+ var rawQueryCatalogue = query_templates[searchterm]['catalogue'];
+ var sparqlQueryWD = rawQueryWD.replaceAll('<','<').replaceAll('>','>').replaceAll(/"/g, '"');
+ var sparqlQueryCatalogue = rawQueryCatalogue.replaceAll('<','<').replaceAll('>','>').replaceAll(/"/g, '"');
+
+ $('#'+searchterm).off('keyup').on('keyup', function() {
+ $("#searchresult").show();
+ setSearchResult(searchterm);
+ var value = $('#'+searchterm).val().toLowerCase();
+ newSparqlQueryWD = sparqlQueryWD.replace("insertQueryTerm",value);
+ newSparqlQueryCatalogue = sparqlQueryCatalogue.replace("insertQueryTerm",value);
+
+ if (value.length>0) {
+ $("#searchresult").empty();
+ requestWD = makeSPARQLQuery( myPublicEndpoint.replace('/sparql','/wd'), newSparqlQueryWD, function( data ) {
+ data.results.bindings.forEach(function(obj,idx) {
+ let idSplit = obj.item.value.split('/');
+ var wdId = idSplit[idSplit.length-1];
+ var newItemDiv = $('');
+ newItemDiv.find('[data-id]').bind('click', function(e) {
+ e.preventDefault();
+ var oldID = $(this).attr('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).after("" + oldLabel + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+ $("#searchresult").append(newItemDiv);
+ });
+ // call VIAF API in case no result is available in WD
+ if (!data.search || !data.search.length) {
+ callViafAPI(value, function (viafData) {
+ if (viafData.result && $('#'+searchterm).val().toLowerCase() === value && $("#searchresult div").length === 0) {
+ // to avoid repetitions of the same element: $("#searchresult").find("[data-id="+item.viafid+"]").length === 0
+ $.each(viafData.result, function (i, item) {
+ if ($("#searchresult").find("[data-id="+item.viafid+"]").length === 0 && $("#searchresult > .viafitem").length <5) {
+ $("#searchresult").append(""); // no item.DESCRIPTION!
+
+ // add tag if the user chooses an item from viaf
+ $('a[data-id="' + item.viafid + '"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ var input_name = (searchterm.split('_').length == 2) ? searchterm.split('')[0] + item.viafid + searchterm.split('_')[1] : searchterm + '-' + item.viafid;
+ $('#' + searchterm).after("" + item.term + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ //colorForm();
+ });
+ });
+ };
+ });
+ }
+ });
+ };
+ });
+ requestCatalogue = makeSPARQLQuery( myPublicEndpoint, newSparqlQueryCatalogue, function( data ) {
+ data.results.bindings.forEach(function(obj,idx) {
+ let idSplit = obj.item.value.split('/');
+ var catalogueId = idSplit[idSplit.length-1];
+ if ($('#searchresult [data-id="'+catalogueId+'"]').length == 0) {
+ var newItemDiv = $('');
+ newItemDiv.find('[data-id]').bind('click', function(e) {
+ e.preventDefault();
+ var oldID = $(this).attr('data-id').substr(this.getAttribute('data-id').lastIndexOf('/') + 1);
+ var oldLabel = $(this).text();
+ $('#' + searchterm).after("" + oldLabel + " ");
+ $("#searchresult").hide();
+ $('#' + searchterm).val('');
+ });
+ $("#searchresult").append(newItemDiv);
+ };
+ });
+ });
+
+ };
+ });
+}
+
+// add a manually defined entity
+function addManualEntity(searchterm) {
+ var baseId = searchterm.replace('_uri','').replace('_label','');
+ console.log(baseId)
+
+
+ $('#'+searchterm).keypress(function(e) {
+ console.log(e.which)
+ console.log(uri, label)
+ var uri = $("#"+baseId+'_uri').val()
+ var label = $("#"+baseId+'_label').val()
+ if (e.which == 13 && uri != '' && label != '') {
+ console.log(uri, label)
+
+ e.preventDefault();
+ console.log("ciao")
+ console.log($("#"+baseId+'_label').next())
+ $("#"+baseId+'_label').next().append($("" + label + " "))
+ } else if (e.which == 13 && (uri == '' || label == '')) {
+ alert('Please, provide a label and a URI')
+ }
+ });
+}
+
+// search a rdf property in LOV
+function searchLOV(searchterm) {
+ $('#'+searchterm).off('keyup').on('keyup',function(e) {
+ var q = $('#'+searchterm).val();
+ var searchres_div = $('#'+searchterm).next().attr('id');
+ if (q === '') {
+ $("#"+searchres_div).empty();
+ } else {
+ $("#"+searchres_div).show();
+ $.ajax({
+ type: 'GET',
+ url: "https://lov.linkeddata.es/dataset/lov/api/v2/term/autocomplete?q="+q+"&type=property",
+ success: function(data) {
+ // autocomplete positioning
+ var position = $('#'+searchterm).position();
+ var leftpos = position.left + 15 - $('#'+searchterm).parent().parent().position().left;
+ var offset = $('#'+searchterm).offset();
+ var height = $('#'+searchterm).height();
+ var width = $('#'+searchterm).width();
+ var top = offset.top + height - 294 + "px";
+ var right = offset.left + width + "px";
+
+ $('#'+searchres_div).css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':'600px',
+ });
+ $("#"+searchres_div).empty();
+
+ // fill the dropdown
+ $.each(data.results, function(i, item) {
+ $("#"+searchres_div).append("");
+
+ // add tag if the user chooses an item from wd
+ $('a[data-id="'+ item.prefixedName+'"]').each( function() {
+ $(this).bind('click', function(e) {
+ e.preventDefault();
+ $("#"+searchres_div).hide();
+ $('#'+searchterm).val(item.uri);
+ });
+
+ });
+ });
+ }
+
+
+ });
+ }
+ });
+};
+
+// addURL:URL value in textbox field + create iframe previews
+function addURL(searchterm, iframe=false) {
+ console.log(iframe)
+ var itemTable
+ if ($("table.url-table[data-input='"+searchterm+"']").length === 0) {
+ itemTable = $("\
+ \
+ \
+ URL \
+ Actions \
+ \
+ \
+ \
+
");
+ } else {
+ itemTable = $("table.url-table[data-input='"+searchterm+"']");
+ }
+
+ $('#'+searchterm).off('keyup').on('keyup', function(e) {
+
+ let regexURL = /(http|https)?(:\/\/)?(www\.)?[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,4}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)/;
+
+ if(e.which == 13 || e.which == 44) { // 44 code for commas
+ e.preventDefault();
+ var now = new Date().valueOf();
+ var newID = 'MD'+now;
+ // check the input value against the regex
+ if ($('#'+searchterm).val().length > 0 && regexURL.test($('#'+searchterm).val()) ) {
+ // generate iframe if requested
+ if (iframe) {
+ var url;
+ if (!$('#'+searchterm).val().startsWith("https://") && !$('#'+searchterm).val().startsWith("http://")) {
+ url = "https://" + $('#'+searchterm).val();
+ } else {
+ url = $('#'+searchterm).val();
+ }
+ itemTable.find('tbody').append($("\
+ "+$('#'+searchterm).val()+" \
+ \
+ \
+ \
+ \
+ "));
+ $('#'+searchterm).next('.tags-url').prepend(itemTable);
+ }
+ else {
+ itemTable.find('tbody').append($("\
+ "+$('#'+searchterm).val()+" \
+ \
+ \
+ \
+ \
+ "));
+ $('#'+searchterm).next('.tags-url').append(itemTable);
+ }
+
+ // popover Wayback Machine
+ var tooltip_save = '\
+ ';
+
+ var tooltip_saved = '\
+ ';
+
+ $('#'+searchterm).parent().append(tooltip_save);
+ $('#'+searchterm).parent().append(tooltip_saved);
+ $(".savetheweb").popover({
+ html: true,
+ title : "Need to save a source for the future? ",
+ content: "If you have a web page that is important to you, \
+ we can save it using the \
+ Wayback Machine
\
+ Shall we?
\
+ yes \
+ Maybe later \
+
",
+ placement: "bottom",
+ }).popover('show');
+
+ }
+ else if ($('#'+searchterm).val().length > 0 && !regexURL.test($('#'+searchterm).val()) ) {
+ alert('Insert a valid URL');
+ }
+ $('#'+searchterm).val('');
+ //colorForm();
+ };
+ });
+}
+
+// searchYear: gYear Value in Date Field
+function searchYear(searchYear) {
+ $('#'+searchYear).keyup(function(e) {
+ // input values must be digits (47 → 58), backspace (8), enter (13)
+ if ((e.which > 47 && e.which < 58) || e.which == 8 || e.which == 13) {
+ $("#searchresult").show();
+
+ var position = $('#'+searchYear).position();
+ var leftpos = position.left+80;
+ var offset = $('#'+searchYear).offset();
+ var height = $('#'+searchYear).height();
+ var width = $('#'+searchYear).width();
+ var top = offset.top + height + "px";
+ var right = offset.left + width + "px";
+
+ $('#searchresult').css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':'600px',
+ 'max-height': '300px',
+ 'overflow-y': 'auto'
+ });
+ $("#searchresult").empty();
+
+ let options = []; // create an empty array to be populated with the 'year' options
+ let inputYearString = $('#'+searchYear).val();
+ // remove 'A.C.' or 'B.C.', if present, when the user presses backspace
+ if (e.which == 8 && (inputYearString.includes("A.C") || inputYearString.includes("B.C"))) {
+ var expression = / [AB].C/i;
+ var regex = new RegExp(expression);
+ var newValue = inputYearString.replace(regex, "");
+ $('#'+searchYear).val(newValue);
+ inputYearString = newValue;
+ }
+ // generate options for numbers containing 2→4 digits
+ if (inputYearString.length > 1 && inputYearString.length < 5) {
+ let inputYear = parseInt(inputYearString);
+ options.push(inputYearString); // suggests the input value
+ if (inputYear >= 10 && inputYear < 21) {
+ // suggests numbers from 100 to 999 (two input digits requested)
+ for (let i=0; i<100; i++) {
+ options.push(inputYearString+parseInt(i));
+ }
+ } else if (inputYear >= 100 && inputYear < 206) {
+ // suggest numbers from 1000 to 2050 (three input digits requested)
+ for (let i=0; i<10; i++) {
+ options.push(inputYearString+parseInt(i));
+ }
+ }
+ // generate options after one digit 'd' has been inserted (available options: 'd A.C.', 'd B.C.')
+ } else if (inputYearString.length == 1) {
+ options.push(inputYearString);
+ } else if (inputYearString.length > 4) {
+ $("#searchresult").hide();
+ $('#'+searchYear).val(inputYearString.substring(0, 4));
+ alert("Year out of range");
+ } else {
+ $("#searchresult").hide();
+ }
+ // add the options to a scrollable list and add a tag in case a year has been clicked
+ if (inputYearString.length > 0) {
+ $.each(options, function(i, item) {
+ $("#searchresult").append("")
+ $("#searchresult").append("")
+
+ $('a[data-id="'+ item +'B.C."], a[data-id="'+ item +'A.C."]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ $("#searchresult").hide();
+ $('#' + searchYear).val($(this).text());
+ });
+ });
+ });
+ }
+ } else {
+ var newlength = $('#'+searchYear).val().length - 1;
+ $('#'+searchYear).val($('#'+searchYear).val().substring(0, newlength));
+ alert("Only digit accepted")
+ }
+ })
+}
+
+// search through SKOS vocabularies
+function searchSkos(searchterm) {
+
+ $('#' + searchterm).keyup(function(e) {
+ if ($('#' + searchterm).val().length > 1) {
+ $("#searchresult").hide();
+ var requests = []; // prepare an array to collect all the requests to the selected thesauri's endpoints
+ var results_array = []; // prepare an array to access the results returned by the query
+ var vocabs_array = []; // prepare an array to store the selected thesauri's names
+
+ var skos_vocabs = query_templates[searchterm]; // look at the back-end app for query_templates
+ skos_vocabs.forEach(function(obj, idx) {
+ // isolate each SKOS resource associated with the input field and prepare an ajax request
+ var vocabulary_name = Object.keys(obj)[0]
+ if (obj[vocabulary_name].type == "API") {
+
+ // retrieve the parameters of the request to properly call the API
+ var query_parameters = Object.assign({}, obj[vocabulary_name].parameters);
+ // get the keys of the parameters object: the first key (keys[0]) must be associated with the query-term (i.e., input value)
+ var keys = Object.keys(query_parameters);
+ query_parameters[keys[0]] = $('#' + searchterm).val() + "*";
+ const request = $.getJSON(obj[vocabulary_name].endpoint, query_parameters);
+
+ requests.push(request);
+ results_array.push(obj[vocabulary_name].results);
+ vocabs_array.push(vocabulary_name);
+
+ } else if (obj[vocabulary_name].type == "SPARQL") {
+
+ // the string 'QUERY-TERM' inside the query must be replaced with the input value; special charachters must be checked
+ var query = (obj[vocabulary_name].query).replace("QUERY-TERM", ("^" + $('#' + searchterm).val())).replace(">", ">").replace("<", "<").replace(/"/g, '"');
+ var request_parameters = {
+ type: 'GET',
+ url: '/sparqlanything?q=' + encodeURIComponent(query)
+ }
+ const request = $.ajax(request_parameters);
+
+ requests.push(request);
+ results_array.push(obj[vocabulary_name].results);
+ vocabs_array.push(vocabulary_name);
+
+ }
+ });
+
+ Promise.all(requests)
+ .then(function(results) {
+ const options = []; // array to include ALL the resulting terms of ALL the queries
+ console.log(results); // results = ALL the resulting objects of ALL the queries
+ results.forEach(function(data, index) {
+ console.log(data) // the resulting object of a query
+ var path = results_array[index];
+ var main_path = path.array.split(".");
+ let result = data;
+ main_path.forEach(key => {
+ result = result[key];
+ });
+
+ result.forEach(function(res) {
+ // extract a label for each term
+ let label_path = path.label.split(".");
+ let label = res;
+ label_path.forEach(key => {
+ label = label[key];
+ });
+ // extract the URI value for each term
+ let uri_path = path.uri.split(".");
+ let uri = res;
+ uri_path.forEach(key => {
+ uri = uri[key];
+ });
+ let add = uri + "," + label + "," + vocabs_array[index];
+ options.push(add);
+ });
+ });
+
+ $("#searchresult").show();
+
+ // autocomplete positioning;
+ var offset = $('#'+searchterm).offset();
+ var leftpos = offset.left+15;
+ var height = $('#'+searchterm).height();
+ var top = offset.top + height + 15 + "px";
+ var max_width = '600px';
+
+ $('#searchresult').css( {
+ 'position': 'absolute',
+ 'margin-left': leftpos+'px',
+ 'top': top,
+ 'z-index':1000,
+ 'background-color': 'white',
+ 'border':'solid 1px grey',
+ 'max-width':max_width,
+ });
+ $("#searchresult").empty();
+ options.forEach(function(option) {
+ // each option (i.e., retrieved term) has this structure: URI,LABEL,VOCABULARY
+ const resource_uri = option.split(",");
+ var uri = resource_uri.shift();
+ var vocabulary = resource_uri.pop();
+ var label = resource_uri.join(","); // Join is needed for labels containing ","
+
+ // how to display the vocabulary name: e.g., 'vocab-one' → 'VOCAB ONE'
+ if (vocabulary.includes("-")) {
+ var vocabulary_noun = vocabulary.replace("-", " ").toUpperCase();
+ } else {
+ var vocabulary_noun = vocabulary.toUpperCase();
+ }
+ // create a list of options
+ $("#searchresult").append("")
+
+ $('a[data-id="'+ uri +'"]').each(function () {
+ $(this).bind('click', function (e) {
+ e.preventDefault();
+ if (!skos_vocabs.includes("oneVocableAccepted") || $('#' + searchterm).nextAll("span").length == 0) {
+ $('#' + searchterm).after("" + label+" - "+vocabulary_noun + " ");
+ }
+ else if (skos_vocabs.includes("oneVocableAccepted") && $('#' + searchterm).nextAll("span").length > 0) {
+ alert("Only one term is accepted!");
+ }
+ $("#searchresult").hide();
+ $('#' + searchterm).val("");
+ });
+ });
+ });
+
+ })
+ .catch(function(error) {
+ console.error('Ajax Error:', error);
+ });
+ } else { $("#searchresult").hide();}
+ });
+}
+
+// multiple multimedia Links
+function addMultimedia(searchterm) {
+ var itemTable
+ if ($("table.url-table[data-input='"+searchterm+"']").length === 0) {
+ itemTable = $("\
+ \
+ \
+ URL \
+ Actions \
+ \
+ \
+ \
+
");
+ } else {
+ itemTable = $("table.url-table[data-input='"+searchterm+"']");
+ }
+
+ $('#'+searchterm).keypress(function(e) {
+ let regexURL = /(http|https)?(:\/\/)?(www\.)?[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z0-9]{2,4}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)/;
+ let imageFormats = ["apng", "avif", "gif", "ico", "jpg", "jpeg", "jfif", "pjpeg", "pjp", "png", "svg", "webp"];
+ let audioFormats = ["mp3", "wav", "ogg"];
+ let videoFormats = ["mp4", "ogg", "webm"];
+ if(e.which == 13 || e.which == 44) { // 44 code for ,
+ e.preventDefault();
+ var now = new Date().valueOf();
+ var newID = 'MM'+now; // id for multimedia tags
+
+
+ if ($('#'+searchterm).val().length > 0 && regexURL.test($('#'+searchterm).val()) ) {
+ // IMAGE
+ if ($('#'+searchterm).hasClass("Image") && stringEndsWith($('#'+searchterm).val(), imageFormats)) {
+ itemTable.find('tbody').append($("\
+ "+$('#'+searchterm).val()+" \
+ \
+ \
+ \
+ \
+ "));
+ $('#'+searchterm).next('.tags-url').prepend(itemTable);
+ } // AUDIO
+ else if ($('#'+searchterm).hasClass("Audio") && stringEndsWith($('#'+searchterm).val(), audioFormats)) {
+ itemTable.find('tbody').append($("\
+ "+$('#'+searchterm).val()+" \
+ \
+ \
+ \
+ \
+ "));
+ $('#'+searchterm).next('.tags-url').prepend(itemTable);
+ } // VIDEO
+ else if ($('#'+searchterm).hasClass("Video") && stringEndsWith($('#'+searchterm).val(), videoFormats)) {
+ itemTable.find('tbody').append($("\
+ "+$('#'+searchterm).val()+" \
+ \
+ \
+ \
+ \
+ "));
+ $('#'+searchterm).next('.tags-url').prepend(itemTable);
+ }
+ else {
+ alert('Invalid File Format');
+ }
+ }
+ else if ($('#'+searchterm).val().length > 0 && !regexURL.test($('#'+searchterm).val()) ) {
+ alert('Insert a valid URL');
+ }
+ $('#'+searchterm).val('');
+ //colorForm();
+ };
+ });
+}
+
+// check whether the input url $('#'+searchterm).val() ends with a valid format
+function stringEndsWith(string, formatList) {
+ for (var i = 0; i < formatList.length; i++) {
+ var character = formatList[i];
+ if (string.endsWith(character)) {
+ return true;
+ }
+ }
+ return false;
+}
+
+// delete URLs input values
+function deleteUrlInput(el) {
+ if ($(el).parent().parent('tr').parent('tbody').find('tr').length === 1) {
+ $(el).parent().parent('tr').parent('tbody').parent('table').remove();
+ } else {
+ $(el).parent().parent('tr').remove();
+ }
+ return false
+}
+
+// expand URLs input value
+function expandUrlInput(event, el) {
+ event.preventDefault();
+ // retrieve the resource url
+ var url = $(el).parent().parent('tr').find('span').text();
+ if (!url.startsWith("https://") && !url.startsWith("http://")) {
+ url = "https://" + url;
+ }
+
+ var mediaType = $(el).attr('title').replace("expand-", "");
+ console.log(mediaType)
+ // create a modal preview
+ if (mediaType == "image") {
+ var modal = $("This is a preview of your multimedia file:"+url+" ");
+ } else if (mediaType =="video") {
+ var modal = $("This is a preview of your multimedia file:"+url+" ");
+ } else if (mediaType == 'audio') {
+ var modal = $("This is a preview of your multimedia file:"+url+" ");
+ } else if (mediaType == 'iframe') {
+ var modal = $("This is a preview of your website:"+url+" ");
+ }
+ $(el).after(modal);
+ $('#showRight').hide();
+ return false;
+};
+
+// NLP
+function nlpText(searchterm) {
+ console.log('f')
+ var lang = searchterm.split('_')[1];
+ var baseID = searchterm.split('_')[0];
+ $('textarea#'+searchterm).keypress( throttle(function(e) {
+ if(e.which == 13) {
+ //$('textarea#'+searchterm).parent().parent().append('
');
+ $(this).next('.tags-nlp').empty();
+ var textNLP = $('#'+searchterm).val();
+ var encoded = encodeURIComponent(textNLP)
+
+ // call CLEF api (spacy)
+ $.ajax({
+ type: 'GET',
+ url: 'nlp?q=' + encoded + '&lang=' + lang,
+ success: function(listTopics) {
+ console.log(listTopics);
+ for (var i = 0; i < listTopics.length; i++) {
+ console.log("listTopics[i]",listTopics[i]);
+ // query WD for reconciliation
+ $.getJSON("https://www.wikidata.org/w/api.php?callback=?", {
+ search: listTopics[i].result,
+ action: "wbsearchentities",
+ language: "en",
+ limit: 1,
+ uselang: "en",
+ format: "json",
+ strictlanguage: true,
+ },
+ function(data) {
+ console.log(data);
+ $.each(data.search, function(i, item) {
+ $('textarea#'+searchterm).parent().find('.tags-nlp').append(''+item.label+' ');
+ });
+ });
+ };
+
+ // return something or message
+ }
+ });
+ };
+
+ }) );
+};
+
+// lookup when creating new records
+function checkPriorRecords(elem) {
+ $('.'+elem).keyup(function(e) {
+ $("#lookup").show();
+ var q = $('.'+elem).val();
+ var classes = $(this).attr('class');
+ var expression = /\(([^)]+)\)/i;
+ var regex = new RegExp(expression);
+ if (classes.match(regex)) {
+ var res_class = ' a <'+classes.match(regex)[1]+'> ; ';
+ } else {var res_class = ''};
+ var query = "prefix bds: select distinct ?s ?o "+inGraph+" where { ?s "+res_class+" rdfs:label ?o . ?o bds:search '"+q+"' .} LIMIT 5"
+ var encoded = encodeURIComponent(query);
+
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint+'?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json; charset=utf-8'},
+ success: function(returnedJson) {
+ $("#lookup").empty();
+ if (!returnedJson.results.bindings.length) {
+ //$("#lookup").append("We found the following resources that are similar to the one you mention. ")
+ } else {
+ $("#lookup").append("We already have some resources that match with yours. If this is the case, consider suggesting a different resource!
")
+ for (i = 0; i < returnedJson.results.bindings.length; i++) {
+
+ // exclude named graphs from results
+ var myUrl = returnedJson.results.bindings[i].s.value;
+ if ( myUrl.substring(myUrl.length-1) != "/") {
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1)
+ $("#lookup").append("");
+ };
+ };
+ $("#lookup").append("Ok got it! ")
+ // close lookup suggestions
+ $('#close_section').on('click', function() {
+ var target = $(this).parent();
+ target.hide();
+ });
+ };
+ }
+ });
+
+ });
+};
+
+// set the webpage to display a new subform
+function replace_existing_subforms() {
+ if ($('.subform_section').length) {
+ $('.subform_section').each(function () {
+ var right_css = parseInt($(this).css('right'));
+ $(this).css('right', (right_css + 30) + "%");
+ });
+ } else {
+ $('body').after("
");
+ }
+}
+
+// create subrecords
+function createSubrecord(resourceClass, fieldName, el, subformId=null,cardinality=null ) {
+ console.log(cardinality)
+ // prepare a new subrecord id
+ if (!subformId) {
+ var now = new Date().valueOf();
+ subformId = (now / 1000).toString().replace('.', '-');
+ }
+ var formId = $('.corners').eq(0).find('form').eq(0).attr('id'); // either 'recordForm' or 'modifyForm'
+ /* replace_existing_subforms(); */
+
+ // prepare the new subrecord form
+ const subrecordSection = $("");
+ const subrecordForm = $("");
+
+ // create a clone for each input belonging to the requested (sub-)template
+ $("[data-class='"+resourceClass+"'][class~='original-subtemplate']").each(function() {
+ // CREATE A CLONE ELEMENT
+ const cloneElement = $(this).parent().parent().clone();
+ cloneElement.attr("style", "display: block"); // make it visible
+ cloneElement.find('input').attr('data-subform',subformId); // associate the input field with the subrecord id
+ cloneElement.find('input').removeClass('original-subtemplate');
+ // associate proper identifiers to input fields belonging to the subrecord form
+ var inputId = cloneElement.find('input:not([type="hidden"])').attr('id');
+ cloneElement.find('input:not([type="hidden"])').attr('id', inputId+"_"+subformId.toString());
+ cloneElement.find('input:not([type="hidden"])').attr('name', inputId+"_"+subformId.toString());
+
+ // SET LITERAL INPUT FIELDS
+ if (cloneElement.find('[lang]').length>0) {
+ var literalInput = cloneElement.find('[lang]');
+ var languageListSection = literalInput.parent().prev();
+ languageListSection.find('a').each(function() {
+ var onclickAttr = $(this).attr('onclick');
+ var regex = /'([^"]*)'/g;
+ var originalIdExtended = onclickAttr.match(regex)[0];
+ var originalId = originalIdExtended.substring(1, originalIdExtended.length-1)
+ $(this).attr('onclick', onclickAttr.replace(originalId, originalId+'_'+subformId));
+ });
+ }
+ // add a main-lang hidden input in case of primary key
+ if (cloneElement.find('input.disambiguate').next('[type="hidden"]').length > 0) {
+ var primaryKeyLangId = cloneElement.find('input.disambiguate').next('[type="hidden"]').attr('id');
+ cloneElement.find('input[type="hidden"]').attr('id', primaryKeyLangId+"_"+subformId.toString());
+ cloneElement.find('input[type="hidden"]').attr('name', primaryKeyLangId+"_"+subformId.toString());
+ }
+
+ // SET SUBTEMPLATE FIELDS '+' BUTTON
+ cloneElement.find('[data-subtemplate]').each(function(){
+ var subtemplateClass = $(this).attr('data-subtemplate');
+ var fieldName = $(this).parent().prev().text();
+ var addSubrecordBtn = $(this).next('i');
+ addSubrecordBtn.on('click', function(){
+ createSubrecord(subtemplateClass,fieldName,addSubrecordBtn);
+ })
+ })
+
+ // retrieve previously provided values in case they are available (i.e., modify subrecords):
+ let clonedElementValues = [];
+ // a) single value fields
+ if ($('#'+formId+' #'+inputId+"_"+subformId).length >0) {
+ const toBeModified = $('#'+formId+' #'+inputId+'_'+subformId);
+ cloneElement.find('input').val(toBeModified.val());
+ }
+ // b) multiple values fields
+ if ($('#'+formId+' [name^="'+inputId+'_"][name$="_'+subformId+'"]:not([name="'+inputId.split('_')[0]+'_'+subformId+'"])').length >0) {
+
+ var importedValues = $('#'+formId+' [name^="'+inputId.split('_')[0]+'_"][name$="_'+subformId+'"]:not([name="'+inputId.split('_')[0]+'_'+subformId+'"])');
+ cloneElement.find('.label div a').remove();
+ if ($('#'+inputId).hasClass('searchWikidata') || $('#'+inputId).hasClass('searchVocab') || $('#'+inputId).hasClass('searchGeonamaes')) {
+ importedValues.each(function(){
+ // imported values and URIs
+ var value = $(this).val();
+ var code = value.split(",")[0];
+ var label = decodeURIComponent(value.split(",")[1]);
+ var importedValueSpan = $(""+label+" ");
+ clonedElementValues.push(importedValueSpan);
+ clonedElementValues.push($(this));
+ $(this).remove();
+ });
+ } else {
+ importedValues.each(function(){
+ // multiple-lang literal values
+ clonedElementValues.push($(this));
+ cloneElement.find('input').remove();
+ if($(this).attr('lang') != undefined) {
+ let lang = $(this).attr('lang');
+ const newLangItem = $(''+lang+' ');
+ cloneElement.find('div').append(newLangItem);
+ } else {
+ let mainLang = $(this).val();
+ cloneElement.find('div a[title="text language: '+mainLang.toUpperCase()+'"]').addClass('main-lang');
+ }
+ });
+ cloneElement.find('div a').eq(0).addClass('selected-lang');
+ clonedElementValues[0].show();
+ }
+ }
+ // c) subrecords fields (inner subrecords)
+ if ($('[name="'+inputId+'_'+subformId+'-subrecords"]').length>0) {
+
+ // retrieve subrecords
+ var subrecords = $('[name="'+inputId+'_'+subformId+'-subrecords"]').val().split(',');
+ var subtemplateFieldId = $(this).attr('name').replace('-subrecords', '');
+ var subrecord_cls = $('#'+subtemplateFieldId).attr('subtemplate')
+ for (let i=0; i 0) {
+ var mainLang = $('#'+subrecordLabelField.attr('id').split('_')[0] + '_mainLang_' + code).val();
+ label = $('#'+subrecordLabelField.attr('id').split('_')[0]+'_'+mainLang+'_'+code).val();
+ }
+ }
+ var subrecordValueSpan = $(""+label+" ")
+ var modifyButton = $(' ')
+ var deleteButton = $(' ')
+ clonedElementValues.push(subrecordValueSpan, modifyButton, deleteButton);
+ }
+
+ }
+
+
+ cloneElement.find('.input_or_select').eq(0).append(clonedElementValues);
+ subrecordForm.append(cloneElement);
+
+ })
+
+ // add knowledge extractor if required
+ var resourceTemplate = $('[data-subtemplate="'+resourceClass+'"').attr('data-subtemplateid');
+ console.log($('[data-subtemplate="'+resourceClass+'"'), resourceTemplate)
+ if (extractorsArray.includes(resourceTemplate)) {
+ generateExtractionField(subformId,subrecordForm);
+ }
+ if (cardinality==null) {
+ // subform accordion
+ const subrecordTitle = $("New instance of "+fieldName+" ");
+ subrecordSection.append(subrecordTitle);
+ // save or cancel subrecord (buttons)
+ const subrecordButtons = $("");
+ const saveSubrecordButton = $("task_alt ");
+ $('#'+formId).append(newSubrecord);
+ }
+ // hide_subform
+ }
+ return false;
+ });
+
+ // CANCEL SUBRECORD
+ cancelSubrecordButton.on('click', function(e) {
+ // hide_subform
+ cancel_subrecord(this);
+ });
+
+ subrecordButtons.append(cancelSubrecordButton, saveSubrecordButton);
+ subrecordForm.append(subrecordButtons);
+ }
+ subrecordSection.append(subrecordForm);
+ /* $('.main_content').eq(0).prepend(subrecordSection); */
+ $(el).before(subrecordSection)
+}
+
+// hide and show subrecord input fields
+function toggleSubform(element,label=null) {
+ $(element).find('.chevron').toggleClass('rotate');
+ $(element).toggleClass('closed-title');
+ $('.subform').toggleClass('closed');
+
+ // change Subrecord title
+ if (label) {
+ const subformId = $(element).next('.subform').attr('id').replace('-form', '')
+ $(element).html(label+"");
+ $(element).removeClass('italic');
+ $('html, body').animate({
+ scrollTop: $(element).parent().offset().top - 200
+ }, 800);
+ }
+}
+
+// CANCEL SUBRECORD (before adding it to #recordForm)
+function cancel_subrecord(subrecord_section) {
+ if ($('.subform_section').length > 1) {
+ $('.subform_section').each(function () {
+ var right_css = parseInt($(this).css('right'));
+ $(this).css('right', 'calc('+right_css+'px - 30%)');
+ });
+ } else {
+ $('.modal-previewMM').remove();
+ }
+ $('main form').append($('#searchresult'));
+ $(subrecord_section).closest('.subform_section').remove();
+};
+
+// DELETE or MODIFY SUBRECORD (after it has been added to #recordForm)
+function modifySubrecord(subId, keep) {
+ // get the 'Subtemplate' input field and the 'Subtemplate' class
+ var originalSubtemplateId = $('[name*="-subrecords"][value*="'+subId+'"').attr('name').replace("-subrecords", "");
+ var originalSubtemplateClass = $('#'+originalSubtemplateId).attr('data-subtemplate');
+
+ if (!keep) {
+ // remove all inputs
+ var subrecordsListString = $('[name$="-subrecords"][value*='+subId+'"]');
+ var subrecordsListArray = subrecordsListString.split(',');
+ var idx = subrecordsListArray.indexOf(sub_id);
+ var newList = subrecordsListArray.splice(idx, 1);
+ subrecordsList.val(newList.join(','));
+ }
+ else {
+ // recreate subrecord_section
+ var fieldName = $('#'+subId+'-tag').parent().prev().text();
+ var el = $('#'+subId+'-tag').prevAll('.fa-plus-circle').first();
+
+ // import data from triplestore in case the subrecord has not been loaded yet
+ if (! $('[data-subform="'+subId+'"').length) {
+ var currentUrl = window.location.href;
+ var subrecordUrl = currentUrl.replace(/(modify|review)-\d+-\d+/, `$1-${subId}`);
+ $.ajax({
+ type:'GET',
+ url:subrecordUrl,
+ dataType:"html",
+ success:function(data) {
+ var relevantField = $(data).find('[class*="'+originalSubtemplateClass+'"]');
+ $('#modifyForm').append(relevantField);
+ createSubrecord(originalSubtemplateClass,fieldName,el,subformId=subId);
+ }
+ });
+ } else {
+ createSubrecord(originalSubtemplateClass,fieldName,el,subformId=subId);
+ }
+
+ }
+
+ // remove delete/modify icons
+ $('#'+subId+'-tag').next('i').remove();
+ $('#'+subId+'-tag').next('i').remove();
+ $('#'+subId+'-tag').remove();
+ $('#'+subId).remove();
+}
+
+function delete_inner_subrecord(inner_inputs) {
+ for (let i = 0; i < inner_inputs.length; i++) {
+ if ($("#"+inner_inputs[i]).attr('subtemplate')) {
+ console.log(inner_inputs[i])
+ var recursion_inputs = $("[id*="+inner_inputs[i]+"__]");
+ console.log(recursion_inputs)
+ if (recursion_inputs.length) {
+ for (let y = 0; y < recursion_inputs.length; y++) {
+ delete_inner_subrecord($(recursion_inputs[y]).val().split(","));
+ }
+ }
+ }
+ $("[id*="+inner_inputs[i]+"]").remove();
+ }
+}
+
+////////////////////
+// PUBLISH RECORD //
+///////////////////
+
+// spot a uri in the field and pop up the request to send to wayback machine
+function detectInputWebPage(input_elem) {
+ // if the element includes an input text
+ var input_field = $('.'+input_elem).children("input");
+
+ var tooltip_save = '\
+ ';
+
+ var tooltip_saved = '\
+ ';
+
+ if (input_field.length) {
+ input_field.each(function() {
+ if ( !$(this).hasClass("disable_popover") ) {
+ $(this).on("blur", function() {
+ var input_val = $(this).val();
+ var expression = /^(https?:\/\/(?:www\.|(?!www))[a-zA-Z0-9][a-zA-Z0-9-]+[a-zA-Z0-9]\.[^\s]{2,}|www\.[a-zA-Z0-9][a-zA-Z0-9-]+[a-zA-Z0-9]\.[^\s]{2,}|^https?:\/\/(?:www\.|(?!www))[a-zA-Z0-9]+\.[^\s]{2,}|www\.[a-zA-Z0-9]+\.[^\s]{2,})$/gi;
+ var regex = new RegExp(expression);
+ if (input_val.match(regex)) {
+ $(this).parent().append(tooltip_save);
+ $(this).parent().append(tooltip_saved);
+ $(".savetheweb").popover({
+ html: true,
+ title : "Need to save a source for the future? ",
+ content: "If you have a web page that is important to you, \
+ we can save it using the \
+ Wayback Machine
\
+ Shall we?
\
+ yes \
+ Maybe later \
+
",
+ placement: "bottom",
+ }).popover('show');
+
+
+
+ }
+ });
+ };
+
+ });
+ };
+};
+
+// destroy popovers for wayback machine
+function destroyPopover(el='savetheweb') {
+ $("."+el).popover('hide');
+
+}
+
+// call an internal api to send a post request to Wayback machine
+function saveTheWeb(input_url) {
+ console.log(input_url);
+ $(".savetheweb").popover('hide');
+ $(".savedtheweb").popover({
+ html: true,
+ title : "x Thank you! ",
+ content: "We sent a request to the Wayback machine.
",
+ placement: "bottom",
+ }).popover('show');
+
+ $.ajax({
+ type: 'GET',
+ url: "/savetheweb-"+encodeURI(input_url),
+ success: function(returnedJson) {
+ console.log(returnedJson);
+ }
+ });
+
+
+}
+
+
+///////////////
+// TERM PAGE //
+///////////////
+
+// search catalogue resources on click and offset
+function searchResources(event) {
+ var uri = event.data.uri;
+ var count = event.data.count;
+ var offset_query = event.data.offset_query;
+ var limit_query = event.data.limit_query;
+
+ if (offset_query == "0") {
+ var query = "select distinct ?o ?label "+inGraph+" where { ?o ?p <"+uri+"> ; rdfs:label ?label . } ORDER BY ?o LIMIT "+limit_query+" "
+ } else {
+ var query = "select distinct ?o ?label "+inGraph+" where { ?o ?p <"+uri+"> ; rdfs:label ?label . } ORDER BY ?o OFFSET "+offset_query+" LIMIT "+limit_query+" "
+ }
+ var encoded = encodeURIComponent(query)
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint+'?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json; charset=utf-8'},
+ success: function(returnedJson) {
+ if (!returnedJson.results.bindings.length) {
+ $(".relatedResources").append("No more resources
");
+ } else {
+ for (i = 0; i < returnedJson.results.bindings.length; i++) {
+ var myUrl = returnedJson.results.bindings[i].o.value;
+ // exclude named graphs from results
+ if ( myUrl.substring(myUrl.length-1) != "/") {
+ var resID = myUrl.substr(myUrl.lastIndexOf('/') + 1)
+ var newItem = $(" " + decodeURIComponent( unescape(returnedJson.results.bindings[i].label.value)) + " ").hide();
+ $(".relatedResources").prepend(newItem);
+ newItem.show('slow');
+ };
+ };
+ };
+ }
+ });
+ // update offset query
+ var offset_query = offset_query+limit_query ;
+ $(".showRes").html("show more");
+ event.data.offset_query = offset_query;
+ if (event.data.offset_query > count) {
+ $(".showRes").hide();
+ //$(".hideRes").show();
+ }
+};
+
+
+//////////////
+// EXPLORE //
+//////////////
+
+// sort alphabetically
+function sortList(ul) {
+ var ul = document.getElementById(ul);
+
+ Array.from(ul.getElementsByTagName("span"))
+ .sort((a, b) => a.textContent.localeCompare(b.textContent))
+ .forEach(span => ul.appendChild(span));
+};
+
+
+// get values by property in EXPLORE page, e.g. creators
+function getPropertyValue(elemID, prop, typeProp, typeField, elemClass='') {
+ if (elemClass.length) {var class_restriction = "?s a <"+elemClass+"> . "} else {var class_restriction = ''};
+ // TODO extend for vocabulary terms
+ if ((typeProp == 'URI' || typeProp == 'Place' || typeProp == 'URL') && (typeField == 'Textbox' || typeField == 'Dropdown'|| typeField == 'Checkbox' || typeField == 'Vocab') ) {
+ var query = "select distinct ?o ?oLabel (COUNT(?s) AS ?count) "+inGraph+" where { GRAPH ?g { ?s <"+prop+"> ?o. "+class_restriction+" ?o rdfs:label ?oLabel . } ?g ?stage . FILTER( str(?stage) != 'not modified' ) } GROUP BY ?o ?oLabel ORDER BY DESC(?count) lcase(?oLabel)";
+ } else if ((typeProp=='Date' || typeProp=='gYear' || typeProp=='gYearMonth') && typeField == 'Date') {
+ var query = "select distinct ?o (COUNT(?s) AS ?count) "+inGraph+" where { GRAPH ?g { ?s <"+prop+"> ?o. "+class_restriction+" } ?g ?stage . FILTER( str(?stage) != 'not modified' ) } GROUP BY ?o ORDER BY DESC(?count) lcase(?o)";
+ } else {var query = "none"};
+
+ const len = 10;
+ var encoded = encodeURIComponent(query);
+ console.log(query);
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint+'?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json'},
+ success: function(returnedJson) {
+ console.log(returnedJson);
+ var allresults = [];
+ var results = [];
+ for (i = 0; i < returnedJson.results.bindings.length; i++) {
+ var res = returnedJson.results.bindings[i].o.value;
+ if (typeProp != 'Date' && typeProp != 'gYear' && typeProp != 'gYearMonth') {
+ var resLabel = returnedJson.results.bindings[i].oLabel.value;
+ var xsdProp = false;
+ } else if (typeProp == 'gYear') {
+ var resLabel = parseInt(returnedJson.results.bindings[i].o.value);
+ var xsdProp = typeProp;
+ } else {
+ var resLabel = returnedJson.results.bindings[i].o.value;
+ var xsdProp = typeProp;
+ }
+ var count = returnedJson.results.bindings[i].count.value;
+ var result = ""+resLabel+" ("+count+") ";
+ if (allresults.indexOf(result) === -1) {
+ allresults.push(result);
+ results.push($(result).hide());
+ $("#"+elemID).append($(result).hide());
+ };
+
+
+ };
+
+ // show more in EXPLORE
+ if (results.length > len) {
+ // show first batch
+ $("#"+elemID).find("button:lt("+len+")").show('smooth');
+ $("#"+elemID).next(".showMore").show();
+
+ // show more based on var len
+ let counter = 1;
+ $("#"+elemID).next(".showMore").on("click", function() {
+ ++counter;
+ var offset = counter*len;
+ var limit = offset+len;
+ console.log(counter, offset, limit);
+ $("#"+elemID).find("button:lt("+limit+")").show('smooth');
+ });
+
+ } else if (results.length > 0 && results.length <= len) {
+ $("#"+elemID).find("button:not(.showMore)").show('smooth');
+ };
+
+ } // end function
+
+ });
+
+};
+
+// get records by value and property in EXPLORE
+function getRecordsByPropValue(el, resElem, elemClass='', typeProp=false) {
+ if (elemClass.length) {var class_restriction = "?s a <"+elemClass+"> . "} else {var class_restriction = ''};
+ $(el).toggleClass("alphaActive");
+ if ($(resElem).length) {$(resElem).empty();}
+ var prop = $(el).data("property");
+ console.log(prop);
+ if (typeProp == 'false') {
+ var val = $(el).data("value");
+ var query = "select distinct ?s ?sLabel "+inGraph+" where { GRAPH ?g { ?s <"+prop+"> <"+val+">; rdfs:label ?sLabel . "+class_restriction+" } ?g ?stage . FILTER( str(?stage) != 'not modified' ) } ORDER BY ?sLabel"
+ } else {
+ var val = '"'+$(el).data("value")+ '"^^xsd:'+typeProp.charAt(0).toLowerCase() + typeProp.slice(1);
+ var query = "select distinct ?s ?sLabel "+inGraph+" where { GRAPH ?g { ?s <"+prop+"> "+val+"; rdfs:label ?sLabel . "+class_restriction+" } ?g ?stage . FILTER( str(?stage) != 'not modified' ) } ORDER BY ?sLabel"
+ }
+ console.log(query);
+ var encoded = encodeURIComponent(query);
+ $.ajax({
+ type: 'GET',
+ url: myPublicEndpoint+'?query=' + encoded,
+ headers: { Accept: 'application/sparql-results+json'},
+ success: function(returnedJson) {
+ for (i = 0; i < returnedJson.results.bindings.length; i++) {
+ var res = returnedJson.results.bindings[i].s.value;
+ var resID = res.substr(res.lastIndexOf('/') + 1)
+ var resLabel = returnedJson.results.bindings[i].sLabel.value;
+ $(resElem).append("");
+ };
+ }
+ });
+};
+
+
+//////////////
+// TEMPLATE //
+//////////////
+
+// update index of fields in template page (to store the final order)
+function updateindex() {
+ $('.sortable .block_field').each(function(){
+ var idx = $(".block_field").index(this);
+ $(this).attr( "data-index", idx );
+ var everyChild = this.getElementsByTagName("*");
+ for (var i = 0; i< everyChild.length; i++) {
+ var childid = everyChild[i].id;
+ var childname = everyChild[i].name;
+ if (childid != undefined) {
+ if (!isNaN(+childid.charAt(0))) { everyChild[i].id = idx+'__'+childid.split(/__(.+)/)[1]}
+ else {everyChild[i].id = idx+'__'+childid;}
+ };
+ if (childname != undefined) {
+ if (!isNaN(+childname.charAt(0))) { everyChild[i].name = idx+'__'+childname.split(/__(.+)/)[1]}
+ else {everyChild[i].name = idx+'__'+childname;}
+ };
+
+ if (everyChild[i].id.includes("property")) {
+ searchLOV(everyChild[i].id);
+ };
+ };
+ });
+};
+
+// move blocks up/down when clicking on arrow
+function moveUpAndDown() {
+ var selected=0;
+ var itemlist = $('.sortable');
+ var nodes = $(itemlist).children();
+ var len=$(itemlist).children().length;
+ // initialize index
+ updateindex();
+
+ $(".sortable .block_field").click(function(){
+ selected= $(this).index();
+ });
+
+ $(".up").click(function(e){
+ e.preventDefault();
+ if(selected>0) {
+ jQuery($(itemlist).children().eq(selected-1)).before(jQuery($(itemlist).children().eq(selected)));
+ selected=selected-1;
+ updateindex();
+ };
+
+ });
+
+ $(".down").click(function(e){
+ e.preventDefault();
+ if(selected < len) {
+ jQuery($(itemlist).children().eq(selected+1)).after(jQuery($(itemlist).children().eq(selected)));
+ selected=selected+1;
+ updateindex();
+ };
+ });
+
+
+};
+
+// if field type is selected
+function is_selected(st, field) {
+ if (st == field) {return "selected='selected'"} else {return ""};
+};
+
+// add new field in template
+// backend_file argument is currently used to load the information about the available SKOS vocabularies
+function add_field(field, res_type, backend_file=null) {
+ console.log(field);
+ var contents = "";
+ var temp_id = Date.now().toString(); // to be replaced with label id before submitting
+ console.log(temp_id)
+ var field_type = "\
+ FIELD TYPE \
+ \
+ Select \
+ Textbox (text values or name of entities) \
+ Textarea (long textual descriptions) \
+ Dropdown (select one value from a list) \
+ Checkbox (multiple choice) \
+ Date (select a day/month/year) \
+ Multimedia (audio, image, video) \
+ Vocabulary (SKOS) \
+ Website Preview (iframe) \
+ Knowledge Extraction \
+ Subtemplate \
+ \
+ ";
+
+ var field_name = "";
+
+ var field_prepend = "";
+
+ var field_property = " ";
+
+ var field_value = "\
+ VALUE TYPE \
+ \
+ Select \
+ Free text (Literal) \
+ Entity (URI from Wikidata, VIAF, or catalogue) \
+ Location (from geonames) \
+ URL \
+ \
+ ";
+
+ var field_calendar = "\
+ CALENDAR TYPE \
+ \
+ Select \
+ Full Date \
+ Month and Year \
+ Year \
+ \
+ ";
+
+ if (backend_file != null) {
+ var skos_vocabs = backend_file.split("//");
+ var skos_labels = "";
+ for (let i = 0; i < skos_vocabs.length; i++) {
+ skos_labels = skos_labels + ""+skos_vocabs[i].toUpperCase()+" ";
+ }
+ } else {
+ var skos_labels = "";
+ }
+ var field_available_vocabularies = "\
+ VOCABULARIES LIST \
+ "+skos_labels+"\
+ ADD A NEW VOCABULARY \
+ \
+ ";
+
+ var accepted_values_vocabularies = "\
+ NUMBER OF VOCABLES \
+ \
+ ";
+
+ var field_multimedia = "\
+ MULTIMEDIA TYPE \
+ \
+ Select \
+ Audio \
+ Image \
+ Video \
+ All \
+ \
+ ";
+
+ var field_placeholder = "";
+
+ var field_values = "";
+
+ var field_browse = "";
+
+ var field_mandatory = "";
+
+ var field_hide = "";
+
+ var field_subtemplate_name = "";
+
+ var field_subtemplate_class = "";
+
+ var field_subtemplate_import = "\
+ IMPORT A TEMPLATE \
+ "+importable_templates+" \
+ ";
+
+ var field_check_subtemplate = ""
+
+ var field_cardinality = "";
+
+ var open_addons = "";
+ var close_addons = " ";
+ var up_down = ' ';
+
+ contents += field_type + field_name + field_prepend + field_property + open_addons;
+ if (field =='Textbox') { contents += field_value + field_placeholder + field_mandatory + field_hide; }
+ else if (field =='Textarea') { contents += field_placeholder + field_mandatory + field_hide; }
+ else if (field =='Date') { contents += field_calendar + field_mandatory + field_hide + field_browse ; }
+ else if (field =='Skos') { contents += field_available_vocabularies + accepted_values_vocabularies + field_placeholder + field_mandatory + field_hide + field_browse ; }
+ else if (field =='Multimedia') { contents += field_multimedia + field_placeholder + field_mandatory + field_hide; }
+ else if (field =='WebsitePreview') { contents += field_placeholder + field_mandatory + field_hide; }
+ else if (field =='Subtemplate') { contents = field_type + field_name + field_prepend + field_property + field_subtemplate_import + field_subtemplate_name + field_subtemplate_class + field_check_subtemplate + field_cardinality + field_mandatory + field_hide + field_browse + open_addons; }
+ else if (field =='KnowledgeExtractor') {
+ if ($("select option:selected[value='KnowledgeExtractor']").length > 0) {
+ alert("Max. 1 Knowledge Extraction field allowed");
+ return;
+ }
+ contents = field_type + field_name + field_prepend + field_property + open_addons;
+ }
+ else {contents += field_values + field_mandatory + field_hide + field_browse; };
+ contents += close_addons + up_down;
+ $(".sortable").append("");
+ updateindex();
+ moveUpAndDown() ;
+
+ $(".trash").click(function(e){
+ e.preventDefault();
+ $(this).parent().remove();
+ });
+};
+
+function import_subtemplate(el) {
+ var requested_template = el.value;
+ var requested_name = el.options[el.selectedIndex].text;
+ var name_field = $(el).parent().next().find('input[type="text"]').eq(0);
+ var class_field = $(el).parent().next().next().find('input[type="text"]').eq(0);
+ var edit_field = $(el).parent().next().next().next('input[type="button"]');
+ $(name_field).parent().show();
+ $(class_field).parent().show();
+ // make fields not modifiable unless creating a new subtemplate
+ name_field.attr("disabled", true);
+ class_field.attr("disabled", true);
+
+ if (requested_template !== "CreateNewSubtemplate") {
+ // hide new template button
+ edit_field.hide();
+ // import an existing template
+ var encoded_template = encodeURIComponent(requested_template.replace("resource_templates/", ""));
+ var url = window.location.href.split("/");
+ var url_tpl = url[url.length-1];
+ var url_request = '/'+url_tpl+'?template=' + encoded_template;
+ $.ajax({
+ type: 'GET',
+ url: url_request,
+ datatype: 'text',
+ success: function(result_json) {
+ var results = result_json.substring(1, result_json.length - 1).split(", ", 2);
+ var resource_class = results[0].replaceAll("'", ""), resource_template = results[1];
+ name_field.val(requested_name);
+ class_field.val(resource_class);
+ },
+ error: function(xhr, status, error) {
+ console.error("AJAX error:", error);
+ }
+ });
+ } else {
+ // create a new subtemplate
+ name_field.attr("disabled", false); // re-activate disabled input fields
+ class_field.attr("disabled", false);
+ name_field.focus(); // autofocus the name input field
+ name_field.val(""); class_field.val(""); // make the fields empty
+ edit_field.show();
+ }
+
+}
+
+function edit_subtemplate(el) {
+ var class_field = $(el).parent().prev().find('input').val();
+ var name_field = $(el).parent().prev().prev().find('input').val();
+ if (class_field==='' || name_field==='') {
+ alert("Please, specify a label and a class");
+ } else {
+ var modified_cls = name_field.toLowerCase().replaceAll(" ", "_");
+ var url = "/template-" + modified_cls;
+ console.log(url);
+ $.ajax({
+ type: 'POST',
+ url: '/welcome-1?action=createTemplate&class_uri='+encodeURIComponent(class_field)+'&class_name='+encodeURIComponent(name_field),
+ success: function(data) {
+ setTimeout(function() { window.open(url, "_blank") }, 500);
+ }
+ })
+ }
+}
+
+// if value == literal add disambiguate checkbox
+function add_disambiguate(temp_id, el) {
+ var field_disambiguate = "";
+
+ var field_browse = "";
+
+ if (el.value == 'Literal') {
+ $("input[id*='browse__"+temp_id+"']").parent().remove();
+ $(el).parent().parent().append(field_disambiguate);
+ updateindex();
+ moveUpAndDown() ;
+ } else if (el.value == 'URI' || el.value == 'URL') {
+ if ($("input[id*='disambiguate__"+temp_id+"']") != undefined) {
+ $("input[id*='browse__"+temp_id+"']").parent().remove();
+ $("section[id*='addons__"+temp_id+"']").after(field_browse);
+ $("input[id*='disambiguate__"+temp_id+"']").parent().remove();
+ } else { $("section[id*='addons__"+temp_id+"']").after(field_browse); }
+
+ // YASQE editor for SPARQL query patterns
+ if (el.value == 'URI') {
+ var field_SPARQL_constraint = $("\
+ SPARQL CONSTRAINTS \
+ \
+ Select a service \
+ Wikidata \
+ This catalogue SPARQL endpoint \
+ \
+ ");
+ field_SPARQL_constraint.find('select').on('change', function() {
+ // generate a SPARQL editor instance
+ SPARQL_constraint_editor(el,this,temp_id);
+ });
+ $(el).parent().after(field_SPARQL_constraint);
+ }
+
+ updateindex();
+ moveUpAndDown() ;
+ } else if (el.value == 'Place') {
+ $("input[id*='disambiguate__"+temp_id+"']").parent().remove();
+ $("input[id*='browse__"+temp_id+"']").parent().remove();
+ $("section[id*='addons__"+temp_id+"']").after(field_browse);
+ updateindex();
+ moveUpAndDown() ;
+ }
+
+};
+
+// if one disambiguate is checked, disable others
+function disable_other_cb(ckType) {
+ var ckName = document.getElementsByClassName('disambiguate');
+ var checked = document.getElementById(ckType.id);
+
+ if (checked.checked) {
+ for(var i=0; i < ckName.length; i++){
+ ckName[i].checked = false;
+ // if(!ckName[i].checked){ ckName[i].disabled = true; }
+ // else{ ckName[i].disabled = false;}
+ }
+ checked.checked = true;
+ }
+ else {
+ for(var i=0; i < ckName.length; i++){
+ ckName[i].disabled = false;
+ }
+ }
+
+ // make the field mandatory
+ var mandatory_checkbox_id = ckType.id.replace("disambiguate", "mandatory");
+ console.log(mandatory_checkbox_id)
+ $('#'+mandatory_checkbox_id).prop('checked', true);
+};
+
+
+// generate an instance of the YASQE editor to introduce a new SPARQL query constraint
+function SPARQL_constraint_editor(field,el,temp_id,reset=false) {
+ var select_input = $(el);
+ let endpoint = "";
+ let value_to_set = "";
+ var selected_option = select_input.val();
+
+ // hide existing YASQE editors
+ $(el).parent().find('div[id*="'+temp_id+'"]').hide();
+
+
+ // set the editor parameters based on the selected service:
+ // no service selected
+ if (selected_option === 'None') {
+ return null
+ }
+ // wikidata
+ if (selected_option === 'WD') {
+ endpoint = wikidataEndpoint;
+ var id = $(field).attr('id').split('__')[0] + '__wikidataConstraint__' + temp_id;
+ if ($('[name="'+id+'"]').length>0 && reset==false) {
+ value_to_set = $('[name="'+id+'"]').val().replaceAll('<','<').replaceAll('>','>').replaceAll(/"/g, '"');;
+ } else {
+ value_to_set = `SELECT ?item ?itemLabel ?itemDescription WHERE {
+ ?item wdt:P31 wd:Q5 .
+ ?item wdt:P106 wd:Q2526255 .
+ SERVICE wikibase:mwapi {
+ bd:serviceParam wikibase:endpoint "www.wikidata.org";
+ wikibase:api "EntitySearch";
+ mwapi:search "insertQueryTerm";
+ mwapi:language "en".
+ ?item wikibase:apiOutputItem mwapi:item.
+ }
+ SERVICE wikibase:label {bd:serviceParam wikibase:language "en".}
+}`
+ }
+ }
+ // catalogue
+ if (selected_option === 'catalogue') {
+ endpoint = myPublicEndpoint
+ var id = $(field).attr('id').split('__')[0] + '__catalogueConstraint__' + temp_id;
+ if ($('[name="'+id+'"]').length>0 && reset==false) {
+ value_to_set = $('[name="'+id+'"]').val().replaceAll('<','<').replaceAll('>','>').replaceAll(/"/g, '"');;
+ } else {
+ value_to_set = `PREFIX bds:
+SELECT DISTINCT * WHERE {
+ ?item rdfs:label ?itemLabel .
+ ?item rdf:type ?itemClass .
+ ?itemLabel bds:search "insertQueryTerm*" .
+ OPTIONAL { ?item rdfs:comment ?desc } .
+}`
+ }
+ };
+
+ var cls = selected_option + "_" + id;
+ // retrieve the previously created yasqe editor if available else create a new one
+ if ($('.'+cls).length > 0) {
+ $('.'+cls).show();
+ } else {
+ // set the HTML environment for the YASQE editor and add buttons
+ var field_constraints = $("
");
+ var button_save = $("");
+ var button_delete = $("");
+ var button_reset = $("");
+ var button_help = $("");
+
+ // save the SPARQL constraint on click
+ button_save.find('i').on('click', function() {
+ // generate an hidden input to store the query constraints:
+ // retrieve the query from the YASQE editor and set it as the value of the hidden input
+ var value = yasqe_to_hidden_field(this, keep=true);
+
+ // modify the hidden input in case it already exists else create a new one
+ if ($('[name="'+id+'"]').length >0) {
+ $('[name="'+id+'"]').val(value);
+ } else {
+ const new_hidden_field = $("");
+ new_hidden_field.val(value);
+ select_input.after(new_hidden_field);
+ }
+ // reset the dropdown to its first option ('value="None"')
+ select_input.find('option:first').prop('selected', true);
+ });
+
+ // delete the SPARQL constraint on click
+ button_delete.find('i').on('click', function() {
+ $(this).parent().parent().parent().remove();
+ $('[name="'+id+'"]').remove();
+ });
+
+ // refresh the SPARQL constraint and get back to the default value
+ button_reset.find('i').on('click', function() {
+ $(this).parent().parent().parent().remove();
+ $('[name="'+id+'"]').remove();
+ SPARQL_constraint_editor(field,el,temp_id,reset=true);
+ });
+
+ // get some tips
+ button_help.find('i').on('click', function() {
+ var buttonsDiv = $(this).parent().parent();
+ var goBackButton = $("");
+ goBackButton.find('i').on('click', function() {
+ $('.tips-div').remove();
+ buttonsDiv.find('span').show();
+ $(this).parent().remove();
+ })
+ buttonsDiv.find('span').hide();
+ buttonsDiv.append(goBackButton);
+
+
+ // create a new div to show the tips and anchor it to the .yasqe div
+ const yasqeObj = $(this).parent().parent().prev();
+ var offset = yasqeObj.offset();
+ var left = offset.left;
+ var top = offset.top;
+ var width = yasqeObj.width()+1;
+ var height = yasqeObj.height();
+
+ const newDiv = $('
');
+ newDiv.css({
+ left: left+"px",
+ top: top+"px",
+ width: width+"px",
+ height: height+"px",
+ position: "absolute",
+ "z-index": "5",
+ "background-color": "white",
+ border: "1px solid #D1D1D1",
+ })
+
+ // retrieve the tips
+ $.getJSON("./static/json/help.json", function(data) {
+ const wikidataTipsObj = data.wikidataConstraint;
+ $.each(wikidataTipsObj, function(pageNumber, pageContent) {
+
+ // create a clone of the model div and populate it with HTML content
+ let cloneDiv = newDiv.clone();
+ cloneDiv.html(pageContent);
+
+ // set the clone div attributes and css properties
+ $(cloneDiv).attr('data-number', pageNumber.replace('page',''));
+ if (pageNumber !== 'page1') {
+ $(cloneDiv).hide();
+ } else {
+ $(cloneDiv).addClass('active-page');
+ }
+ $('body').append(cloneDiv);
+ });
+
+ $('.tips-div[data-target="yasqe_'+cls+'"]').each(function() {
+ var prevButton = $(' ');
+ var nextButton = $(' ');
+
+ // previous page
+ prevButton.on('click', function() {
+ var currentPage = $('.tips-div.active-page[data-target="yasqe_'+cls+'"]').attr('data-number');
+ var newPageNumber = parseInt(currentPage)-1;
+ var targetPage = $('.tips-div[data-target="yasqe_'+cls+'"][data-number="'+newPageNumber+'"]');
+ $('.tips-div.active-page[data-target="yasqe_'+cls+'"]').hide();
+ $('.tips-div.active-page[data-target="yasqe_'+cls+'"]').removeClass('active-page');
+ targetPage.show();
+ targetPage.addClass('active-page');
+ });
+
+ // next page
+ nextButton.on('click', function() {
+ console.log('c')
+ var currentPage = $('.tips-div.active-page[data-target="yasqe_'+cls+'"]').attr('data-number');
+ console.log(currentPage)
+
+ var newPageNumber = parseInt(currentPage)+1;
+ var targetPage = $('.tips-div[data-target="yasqe_'+cls+'"][data-number="'+newPageNumber+'"]');
+ $('.tips-div.active-page[data-target="yasqe_'+cls+'"]').hide();
+ $('.tips-div.active-page[data-target="yasqe_'+cls+'"]').removeClass('active-page');
+ targetPage.show();
+ targetPage.addClass('active-page');
+ });
+
+ if ($(this).prev('div').length>0) {
+ $(this).append(prevButton)
+ };
+ if ($(this).next('div').length>0) {
+ $(this).append(nextButton)
+ };
+ })
+
+ });
+
+ });
+ // end of tips section
+
+ select_input.after(field_constraints);
+
+ // complete the creation of the YASQE editor
+ var yasqe = YASQE(document.getElementById("yasqe_"+cls), {
+ sparql: {
+ showQueryButton: false,
+ endpoint: endpoint,
+ }
+ });
+ yasqe.setValue(value_to_set);
+ $('.'+cls).append('
');
+ $('.'+cls+' .yasqe-buttons').append(button_save, button_delete, button_reset, button_help);
+ }
+}
+
+// make hidden fields recognisable
+function hide_field(el) {
+ var checked = document.getElementById(el.id);
+ if (checked.checked == true) {
+ $("#"+el.id).closest('.block_field').css('opacity', 0.6);
+ } else {
+ $("#"+el.id).closest('.block_field').css('opacity', 1);
+ }
+}
+
+// when changing field type, change the form
+function change_fields(sel) {
+ var new_field_type = sel.value;
+ var block_field = $(sel).parent().parent();
+
+ var idx = sel.id;
+ var preserve_data_index = idx.split("__")[0]; // get the data-index value
+ var temp_id = idx.substr(idx.lastIndexOf("__")).replace('__', '');
+
+ var regExp = /\(([^)]+)\)/;
+ var matches = regExp.exec(sel.classList.value);
+ var res_type = matches[1];
+
+ // create a new field, associate it with a variable (new_field_block), assign it the correct data-index attr
+ add_field(new_field_type, res_type, available_skos_vocabularies);
+ const new_field_block = $('.sortable .block_field:last-child');
+ var new_field_block_idx = new_field_block.attr('data-index');
+ new_field_block.attr('data-index', preserve_data_index);
+
+ // get the sections of the substited field to preserve previously inserted values
+ // (e.g.: Display name, RDF property, etc.) based on common labels, i.e. common inputs.
+ // each input field is associated with a section and a label.
+ const previous_field_sections = block_field.find("> section");
+ const previous_field_labels = previous_field_sections.map(function() {
+ return $(this).find("label:first-child").text();
+ }).get();
+
+ // check the elements which make up the input fields of the new field:
+ // in case the element (the section's label) already exists in the substituted field, reuse it
+ // otherwise, keep the new input fields and associate them with the right index and identifier
+ new_field_block.find("section, a, label").each(function() {
+ const label = $(this).find("label:first-child").text();
+ const index = previous_field_labels.indexOf(label);
+ console.log(index, label);
+ if (index !== -1 && index !== 0) {
+ // N.B.: index = -1 means that an input field included in the original field form is not included in the new one.
+ // N.B.: index = 0 refers to the element to change the field. If included it would show the previous field value.
+ const replacementSection = previous_field_sections.eq(index).clone();
+ $(this).replaceWith(replacementSection);
+ } else {
+ // fix the id/name attributes of the main sections (i.e. $('section.row'))
+ var new_section_id = $(this).attr('id');
+ var previous_id = new_section_id.replace(new_field_block_idx, preserve_data_index);
+ $(this).attr('id', previous_id);
+ $(this).attr('name', previous_id);
+
+ // fix the id/name attributes of nested elements
+ $(this).find('[id^="' + new_field_block_idx + '__"]').each(function() {
+ const newId = $(this).attr('id').replace(new_field_block_idx + '__', preserve_data_index + '__');
+ if (newId.split("__").length === 3) {
+ const updatedId = newId.replace(newId.split("__")[2], temp_id);
+ $(this).attr('id', updatedId);
+ } else {
+ $(this).attr('id', newId);
+ }
+ });
+
+ $(this).find('[name^="' + new_field_block_idx + '__"]').each(function() {
+ const newName = $(this).attr('name').replace(new_field_block_idx + '__', preserve_data_index + '__');
+ if (newName.split("__").length === 3) {
+ const updatedName = newName.replace(newName.split("__")[2], temp_id);
+ $(this).attr('name', updatedName);
+ } else {
+ $(this).attr('name', newName);
+ }
+ });
+
+ }
+ });
+
+ block_field.replaceWith(new_field_block);
+};
+
+// add_SKOS_vocab: allow users to add a new SKOS vocabulabury with sparql endpoint
+
+function add_skos_vocab(element) {
+ $(element).closest('label').hide(); // remove the button "Add a new vocabulary"
+
+ // generate the form
+ var form = "\
+ \
+ \
+ \
+ "
+ $(element).closest('.row').after(form);
+ var yasqe = YASQE($('section.skos_vocab_generator #yasqe'), {
+ sparql: {
+ showQueryButton: false,
+ endpoint: myPublicEndpoint,
+ }
+ });
+ yasqe.setValue("SELECT DISTINCT ?label ?uri \nWHERE {?uri ?p ?label} \nLIMIT 10");
+ yasqe.setSize(width='100%',height='200px');
+}
+
+function save_vocab(element) {
+ // access the section containing the list of available vocabularies
+ var vocabs_section = $(element).parent().parent().children().eq(0).find(".col-md-8").eq(0);
+
+ // extract a label, a url, a query, and an endpoint to store a new vocab
+ var label = $('#vocabLabel').val();
+ var url = $('#vocabUrl').val();
+ var query = yasqe_to_hidden_field($(element).parent());
+ console.log(query)
+ var endpoint = $('#vocabEndpoint').val();
+ // combine the pieces of information together and check whether some info is missing
+ var infoArray = [label, url, query, endpoint];
+ var infoTogether = infoArray.join("__");
+ if (check_input_form(infoArray)) {
+ return null
+ };
+
+ // get the number of available vocabs (defined in template.html as a string)
+ var i = available_skos_vocabularies.split("//").length+1;
+ available_skos_vocabularies += "//" + label;
+ console.log(i);
+ // get the 'for' attribute of the label for the first vocab of the list (e.g.: "vocab6__1690443665556")
+ // to retrieve the id number of the field (e.g.: "1690443665556")
+ var temp_id_list = vocabs_section.find("label").eq((1)).attr('for').split("__");
+ var temp_id = temp_id_list[temp_id_list.length - 1];
+ // get the index of the field in the template
+ var idx = parseInt(vocabs_section.find("label").eq((1)).attr('id'));
+ console.log(i, temp_id, idx);
+
+ // generate a new checkbox to select the vocabulary and add it at the end of the list
+ var new_voc = ""+label.toUpperCase()+" ";
+ var lastChild = vocabs_section.children().last();
+ lastChild.prev().after(new_voc);
+
+ // delete the form
+ $('.skos_vocab_generator').remove();
+ $('.add_vocabulary_button').show();
+};
+
+function check_input_form(input_array) {
+ // check the name of the newly imported vocabulary (available_skos_vocabularies is defined in template.html)
+ if (available_skos_vocabularies.toUpperCase().split("//").includes(input_array[0].toUpperCase())) {
+ alert('A vocabulary named '+input_array[0]+' already exists: choose a new label');
+ return true;
+ }
+ // check missing information
+ for (let i=0; idiv');
+ yasqe_lines.each(function() {
+ var tokens = $(this).find('pre span span');
+ tokens.each(function() {
+ if (!$(this).hasClass('cm-ws')) {
+ value+= $(this).text();
+ } else {
+ value+=' ';
+ }
+ });
+ value += '\n';
+ });
+ if (keep===false){ yasqe_div.remove(); } else { yasqe_div.hide(); }
+ return value
+}
+
+//////////////////////////////
+//// KNOWLEDGE EXTRACTION ////
+//////////////////////////////
+
+/* Knowledge Extraction input field handling */
+
+// generate extraction input field (during form loading)
+function generateExtractionField(resId, subtemplate=null) {
+ // retrieve field's description and name
+ var resIndex = extractorsArray.indexOf(resId);
+ resIndex = resIndex === -1 ? 0 : resIndex;
+ var fieldName = extractorsNames[resIndex];
+ var fieldDescription = extractorsPrepend[resIndex];
+
+ // predefined HTML node (extraction input field)
+ var extractionRow = $('');
+
+ // add the extraction node to the form
+ if (subtemplate===null) {
+ subtemplate = $('#recordForm .homeheading, #modifyForm .homeheading');
+ }
+ console.log(subtemplate)
+ subtemplate.append(extractionRow);
+
+ // retrieve previously generated extraction graphs
+ if (resId in extractionsObj) {
+ for (let i=0; i 0) {
+ generateExtractionTagList(subtemplate,resId, extractionResults, resId+'-'+extractionId);
+ }
+ }
+ }
+}
+
+// generate tags from previously extracted URI-label pairs
+function generateExtractionTagList(subtemplate,recordId,results,id) {
+ // delete previously retrieved entities if any
+ $("#graph-"+id).remove();
+ // if new results exist, create a new list item to collect each retrieved URI,label pair in the form of a tag (containing an hidden input)
+ const table = $(subtemplate).find('#imported-graphs-'+recordId).prev('table');
+ table.find('tbody').append($("\
+ \
+ \
+ \
+ \
+ \
+ "));
+
+
+ for (let idx = 0; idx < results.length; idx++) {
+ for (const key in results[idx]) {
+
+ if (results[idx][key].type === "literal" && !results[idx][key].value.startsWith("https://") && !results[idx][key].value.startsWith("http://")) {
+ var label = results[idx][key].value;
+ } else if (results[idx][key].type === "uri" || results[idx][key].value.startsWith("https://") || results[idx][key].value.startsWith("http://")) {
+ var uri = results[idx][key].value;
+ }
+ }
+ table.find("#graph-" + id).append("" + label + " ");
+ }
+ table.removeClass('hidden');
+}
+
+/* Generate, Delete, Modify extraction module */
+
+// generate the extraction form within the extraction input field
+function generateExtractor(ul,modifyId=null) {
+ // update the extraction count within the extractions Object
+ var extractorId = ul.replace('imported-graphs-','');
+ let extractionInternalId;
+
+ if (modifyId === null) {
+ // generate new internal Id (sequential numbering) for the extraction
+ if (extractorId in extractionsObj) {
+
+ // get the number of extractions
+ var extractionArray = extractionsObj[extractorId];
+ var extractionLength = extractionArray.length;
+ var lastExtractionInternalId = extractionArray[extractionLength-1].internalId;
+ extractionInternalId = parseInt(lastExtractionInternalId) + 1;
+
+ // initialiaze a new sub-Object to store information about the novel extraction
+ const newExtractionObj = { "internalId": extractionInternalId };
+ newExtractionObj["metadata"] = {};
+
+ // append the sub-Object to the extractions Array
+ extractionsObj[extractorId].push(newExtractionObj);
+
+ } else {
+ // initialize an Array of extractions Objects within the main extractionsObject
+ extractionInternalId = 1;
+ extractionsObj[extractorId] = [{"internalId": extractionInternalId, "metadata": {}}];
+ }
+ } else {
+ // reuse the existing internal id when modifying an extraction
+ extractionInternalId = modifyId
+ }
+
+ // create a select element containing the options to perform the extraction
+ var extractor = $("\
+ \
+ EXTRACTOR TYPE \
+ \
+ \
+ \
+ ");
+ $('.import-form .col-md-12').append(extractor);
+ console.log(extractor)
+ $('#'+ul).hide();
+};
+
+// remove all the information related to the deleted extraction graph
+function deleteExtractor(id) {
+ // remove key entities (hidden input fields) from DOM
+ var hiddenInput = $("#recordForm, #modifyForm").find('[type="hidden"][name*="'+id+'-"]');
+ hiddenInput.remove();
+ $('#graph-'+id).remove();
+
+ // remove the extraction information from the extractions' Object
+ var splitId = id.split("-");
+ var recordId = splitId[0]+"-"+splitId[1]
+ var extractionNum = splitId[2]
+ extractionsObj[recordId] = extractionsObj[recordId].filter(obj => obj.internalId != parseInt(extractionNum));
+}
+
+function modifyExtractor(record,id) {
+ // look for the extraction information within the extractions Object
+ var extractionNumber = parseInt(id.replace(record,'').replace('-',''));
+ var extractions = extractionsObj[record];
+ const extraction = extractions.find(obj => obj.internalId == extractionNumber);
+ const extractionParameters = extraction ? extraction["metadata"] : undefined;
+ const extractorType = extractionParameters.type
+
+ // generate the form for the extraction
+ generateExtractor("imported-graphs-"+record,extractionNumber);
+ var extractor = $('#extractor');
+ extractor.val(extractorType);
+ addExtractionForm(extractor,record,extractionNumber);
+
+ // fill the extraction form with previously provided values
+ if (extractorType == 'api') {
+ const url = extractionParameters.url;
+ const query = extractionParameters.query;
+ const results = extractionParameters.results;
+
+ // set the URL
+ $('#ApiUrl').val(url);
+ // set the query
+ for (const [key, value] of Object.entries(query)) {
+ const addButton = $('.add-parameter');
+ generateExtractionParameter(addButton);
+ const newParameterDiv = addButton.prev('.api-query-parameter').find('input');
+ newParameterDiv.eq(0).val(key);
+ newParameterDiv.eq(1).val(value);
+ }
+ // set the results' keys
+ $('.api-results-parameter input[value]').each(function() {
+ console.log($(this))
+ $(this).next().val(results[$(this).val().toLowerCase()]);
+ })
+
+ } else if (extractorType == 'sparql' || extractorType == 'file') {
+ const url = extractionParameters.url;
+ const query = extractionParameters.query;
+
+ // set the URL
+ $('#SparqlUrl').val(url);
+
+ // set the SPARQL query
+ $('#yasqe > .yasqe').remove();
+ var yasqe = YASQE(document.getElementById("yasqe"), {
+ sparql: {
+ showQueryButton: false,
+ }
+ });
+ yasqe.setValue(query);
+
+ }
+}
+
+/* Extraction Form */
+
+// create a new form based on the selected option (API, SPARQL, static file)
+function addExtractionForm(element,extractorId,extractionInternalId) {
+ if ($('.block_field.col-md-12').length > 0) {
+ $('.block_field.col-md-12 section').not(":first").remove();
+ }
+ var extractionId = extractorId+'-'+extractionInternalId.toString(); // it will be used to create hidden inputs later
+ var extractionType = $(element).find(":selected").val(); // selected option (API, SPARQL, or static file)
+
+ $('.block_field .extractor-1, .block_field hr').remove() // remove previously created forms (in case the user changes the selected option)
+ if (extractionType == 'api') {
+ var form = $(" \
+ \
+ \
+ ")
+ } else if (extractionType == 'sparql') {
+ var form = $(" \
+