Make JavaScript variable external data? - api
In a script tag in my HTML file I have a JavaScript function as such. Notice the part where it I create the variable const event = { . My question how in the same folder as the HTML file could I create a file that is only the event variable and refer to in in the code. Could this externally referenced data be a JSON file? Thanks
async function insertEvents() {
let response;
try {
const event = {
'summary': 'Martha Dentist',
'location': '800 Howard St., San Francisco, CA 94103',
'description': 'Take Martha to dentist.',
'start': {
'dateTime': '2023-02-28T07:00:00-07:00',
'timeZone': 'America/Los_Angeles'
},
'end': {
'dateTime': '2023-02-28T08:00:00-07:00',
'timeZone': 'America/Los_Angeles'
},
'recurrence': [
'RRULE:FREQ=DAILY;COUNT=8'
],
'attendees': [
{'email': 'charles#datatech.com'},
{'email': 'datatech#gmail.com'}
],
'reminders': {
'useDefault': false,
'overrides': [
{'method': 'email', 'minutes': 24 * 60},
{'method': 'popup', 'minutes': 10}
]
}
};
const request = {
'calendarId': 'primary',
'resource': event
};
response = await gapi.client.calendar.events.insert(request);
} catch (err) {
document.getElementById('content').innerText = err.message;
return;
}
const events = response.result.items;
if (!events || events.length == 0) {
document.getElementById('content').innerText = 'No events found.';
return;
}
}
I did not know if you can refer to another JavaScript or JSON file in a function.
Related
Vue.js with iView Table - accessing Elasticsearch search response object
I am using the iView UI kit table in my Vue.js application that consumes an Elasticsearch API with axios. My problem is that I just can't seem to get to access the nested search response object, which is an array list object. I only get to access the 1st level fields, but not the nested ones. I don't know how to set the table row key in the iView table. This is how my axios call and mapper methods look like: listObjects(pageNumber){ const self = this self.$Loading.start() self.axios.get("/api/elasticsearch/") .then(response => { self.ajaxTableData = self.mapObjectToArray(response.data); self.dataCount = self.ajaxTableData.length; if(self.ajaxTableData.length < self.pageSize){ self.tableData = self.ajaxTableData; } else { self.tableData = self.ajaxTableData.slice(0,self.pageSize); } self.$Loading.finish() }) .catch(e => { self.tableData = [] self.$Loading.error() self.errors.push(e) }) }, mapObjectToArray(data){ var mappedData = Object.keys(data).map(key => { return data[key]; }) return mappedData }, The iView table columns look like this: tableColumns: [ { title: 'Study Date', key: 'patientStudy.studyDate', width: 140, sortable: true, sortType: 'desc' }, { title: 'Modality', key: "generalSeries.modality", width: 140, sortable: true }, ... ] The (raw) Elasticsearch documents look like this: [ { "score":1, "id":"3a710fa2c1b3f6125fc168c9308531b59e21d6b3", "type":"dicom", "nestedIdentity":null, "version":-1, "fields":{ "highlightFields":{ }, "sortValues":[ ], "matchedQueries":[ ], "explanation":null, "shard":null, "index":"dicomdata", "clusterAlias":null, "sourceAsMap":{ "generalSeries":[ { "seriesInstanceUid":"999.999.2.19960619.163000.1", "modality":"MR", "studyInstanceUid":"999.999.2.19960619.163000", "seriesNumber":"1" } ], "patientStudy":[ { "studyDate":"19990608" } ] } } ] And this is how the consumed object looks like: As you can see, the fields I need to access are within the "sourceAsMap" object, and then nested in arrays. How can I provide the iView table cell key to access them? UPDATE: I now "remapped" my Elasticsearch object before displaying it in the Vue.js table, and it works now. However, I don't think that the way I did it is very elegant or clean....maybe you can help me to do it in a better way. This is my method to remap the object: getData(data){ let jsonMapped = [] for(let i = 0; i < data.length; i++){ let id = {} id['id'] = data[i].id let generalData = data[i]['sourceAsMap']['generalData'][0] let generalSeries = data[i]['sourceAsMap']['generalSeries'][0] let generalImage = data[i]['sourceAsMap']['generalImage'][0] let generalEquipment = data[i]['sourceAsMap']['generalEquipment'][0] let patient = data[i]['sourceAsMap']['patient'][0] let patientStudy = data[i]['sourceAsMap']['patientStudy'][0] let contrastBolus = data[i]['sourceAsMap']['contrastBolus'][0] let specimen = data[i]['sourceAsMap']['specimen'][0] jsonMapped[i] = Object.assign({}, id, generalData, generalSeries, generalImage, generalEquipment, patient, patientStudy, contrastBolus, specimen) } return jsonMapped }, The result is this: Even though it now works, but how can I optimize this code?
A few functions can help you with your situation let data = [{ key1: ['k1'], key2: ['k2'], key3: [{ subKey1: 'sk1', subKey2: ['sk2'], subObject: [{ name: 'John', surname: 'Doe' }], items: [1, 2, 3, 5, 7] }] }]; function mapIt(data) { if (isSingletonArray(data)) { data = data[0]; } for(const key in data) { if (isSingletonArray(data[key])) { data[key] = mapIt(data[key][0]); } else { data[key] = data[key]; } } return data; } function isSingletonArray(obj) { return typeof obj === 'object' && Array.isArray(obj) && obj.length === 1; } console.log(mapIt(data)); Outputs: { key1: 'k1', key2: 'k2', key3: { subKey1: 'sk1', subKey2: 'sk2', subObject: { name: 'John', surname: 'Doe' }, items: [ 1, 2, 3, 5, 7 ] } } You can check it in your browser. mapIt unwraps the singleton arrays into objects or primitives. But I recommend you to watch out special elastic client libraries for javascript. It could save you from writing extra code.
Vue.js - Element UI - HTML message in MessageBox
I'm using vue-js 2.3 and element-ui. This question is more specific to the MessageBox component for which you can find the documentation here Problem I'd like to be able to enter html message in the MessageBox More specifically I would like to display the data contained in dataForMessage by using a v-for loop. Apparently, we can insert vnode in the message but I have no idea where to find some information about the syntax. https://jsfiddle.net/7ugahcfz/ var Main = { data:function () { return { dataForMessage: [ { name:'Paul', gender:'Male', }, { name:'Anna', gender:'Female', }, ], } }, methods: { open() { const h = this.$createElement; this.$msgbox({ title: 'Message', message: h('p', null, [ h('span', null, 'Message can be '), h('i', { style: 'color: teal' }, 'VNode '), h('span', null, 'but I would like to see the data from '), h('i', { style: 'color: teal' }, 'dataForMessage'), ]) }).then(action => { }); }, } } var Ctor = Vue.extend(Main) new Ctor().$mount('#app')
I think this is what you want. methods: { open() { const h = this.$createElement; let people = this.dataForMessage.map(p => h('li', `${p.name} ${p.gender}`)) const message = h('div', null, [ h('h1', "Model wished"), h('div', "The data contained in dataForMessage are:"), h('ul', people) ]) this.$msgbox({ title: 'Message', message }).then(action => { }); }, } Example.
You can also use html directly and convert to vnodes by using domProps: const html = '<div><h1>Model wished</h1><div>The data contained in dataForMessage are:</div><ul><li>Paul Male</li><li>Anna Female</li></ul></div>' const message = h("div", {domProps:{innerHTML: html}}) (The above is simplified without the loop. Just to get the idea) Fiddle
Updating an array of Data in Vue JS
i'm trying to make a little DataTable in VueJS 2.0, and i've hit a bit of a roadblock, i'm making a call to a backend route to get some JSON from records in the Database, so far so good. I want those records to populate an Array of GridData that i will later iterate through and display on my table. The problem here is that it's not working... Vue.http.post('/getData', data).then((response) => { this.gridData = response.body; }); I have my data setup as follows data: () => { return { searchQuery: "", columns: ['ID', 'Name', 'Campaign', 'Method', 'Limit Per Connection', 'Limit Per Day', 'Active', 'Successes', 'Failures', 'Last Ran'], lastId: 0, rowsPerPage: 10, gridData: [] } } And my function that gets the data is called on the mounted hook mounted() { this.fetchData(this.lastId, this.rowsPerPage); }, Am i doing anything wrong here?
You are using POST HTTP Request, instead of GET.POST is for creating new records in API/DB. I think this should work for you data() { return { searchQuery: "", columns: ['ID', 'Name', 'Campaign', 'Method', 'Limit Per Connection', 'Limit Per Day', 'Active', 'Successes', 'Failures', 'Last Ran'], lastId: 0, rowsPerPage: 10, gridData: [] } }, created() { this.fetchData() }, methods: { fetchData() { this.$http.get('/getData') .then(result => { this.gridData = result.data // or this.gridData = result.json() }) } }
Column Visibility is not restored from a saved state via stateLoadCallback
I have added the Column Visibility button to choose to show or hide certain columns. I'm saving the state in a database, I call the stateSaveCallback function via a click on a button. I cant find documentation about retrieving data this way, so I just link to the page and pass variables to get the data back from the database, and then load that using stateLoadCallback. Now all this works fine, EXCEPT the column visibility is not restored. It is in the JSON data being returned though. Here is my full code: $(document).ready(function() { $.extend( jQuery.fn.dataTableExt.oSort, { "date-uk-pre": function (a){ return parseInt(moment(a, "DD/MM/YYYY").format("X"), 10); }, "date-uk-asc": function (a, b) { return a - b; }, "date-uk-desc": function (a, b) { return b - a; } }); var edit_date_col_num = $('th:contains("Edit Date")').index(); var entry_date_col_num = $('th:contains("Entry Date")').index(); var table = $('.mainTable').DataTable( { pageLength: 50, colReorder: true, stateSave: true, columnDefs: [ { "type": "date-uk", targets: [ edit_date_col_num, entry_date_col_num ] } ], dom: 'Blfrtip', buttons: [ 'copy', 'csv', 'excel', 'print', { extend: 'colvis', collectionLayout: 'fixed four-column', postfixButtons: [ 'colvisRestore' ] } ], <?php $id = $this->input->get('id'); $action = $this->input->get('action'); if(isset($action) && $action == 'load' && isset($id) && $id != '') : ?> "stateLoadCallback": function (settings) { var o; // Send an Ajax request to the server to get the data. Note that // this is a synchronous request since the data is expected back from the // function $.ajax( { "url": EE.BASE + "&C=addons_modules&M=show_module_cp&module=ion&method=state_save&action=load&id=<?php echo $id;?>", "async": false, "dataType": "json", "success": function (response) { response = JSON.parse(response); o = response; } }); return o; }, <?php endif; ?> initComplete: function (settings) { this.api().columns().every( function () { var column = this; var select = $('<select><option value=""></option></select>') .appendTo( $(column.footer()).empty() ) .on( 'change', function () { var val = $.fn.dataTable.util.escapeRegex( $(this).val() ); column .search( val ? '^'+val+'$' : '', true, false ) .draw(); } ); column.data().unique().sort().each( function ( d, j ) { select.append( '<option value="'+d+'">'+d+'</option>' ) } ); } ); // Need to re-apply the selection to the select dropdowns var cols = settings.aoPreSearchCols; for (var i = 0; i < cols.length; i++) { var value = cols[i].sSearch; if (value.length > 0) { value = value.replace("^", "").replace("$",""); console.log(value); $("tfoot select").eq(i).val(value); } } }, } ); // Save a datatables state by clicking the save button $( ".save_state" ).click(function(e) { e.preventDefault(); table.destroy(); $('.mainTable').DataTable( { colReorder: true, stateSave: true, "stateSaveCallback": function (settings, data) { var save_name = $('.save_name').val(); // Send an Ajax request to the server with the state object $.ajax( { "url": EE.BASE + "&C=addons_modules&M=show_module_cp&module=ion&method=state_save&action=save&save_name="+save_name, "data": data, "dataType": "json", "type": "POST", "success": function (response) { //console.log(response); } } ); }, }); //table.state.save(); window.location.replace(EE.BASE + "&C=addons_modules&M=show_module_cp&module=ion&method=applications"); }); $( ".clear_state" ).click(function(e) { e.preventDefault(); table.state.clear(); window.location.replace(EE.BASE + "&C=addons_modules&M=show_module_cp&module=ion&method=applications"); }); } ); Here is the saved JSON with several visible false in the beginning (which are visible once loaded): {"time":"1449338856556","start":"0","length":"50","order":[["0","asc"]],"search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"},"columns":[{"visible":"false","search":{"search":"","smart":"false","regex":"true","caseInsensitive":"true"}},{"visible":"false","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"false","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"false","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"false","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}},{"visible":"true","search":{"search":"","smart":"true","regex":"false","caseInsensitive":"true"}}],"ColReorder":["0","1","2","3","4","5","6","7","8","9","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26","27","28","29","30","31","32","33","34","35","36","37","38","39","40","41","42","43","44","45","46","47","48","49","50","51","52","53","54","55","56","57","58","59","60","61","62","63","64","65","66","67","68","69","70"]} Thanks
In my case datatables rejects old data according to "stateDuration" and "time" properties.. Solution: ignore state duration "stateSave": true, "stateDuration": -1, Above case: "visible":"false" may should be "visible":false
After a while of debugging this myself here's what worked for me.. This issue is that all the values in your JSON are strings and they need to be of correct datatypes for the datatables plugin. Within the "stateSaveCallback" ajax request to save your state I did the following to the json string and then it saved all the values properly which then loaded the state as it should. "stateSaveCallback": function (settings, data) { var save_name = $('.save_name').val(); // Send an Ajax request to the server with the state object $.ajax( { "url": EE.BASE + "&C=addons_modules&M=show_module_cp&module=ion&method=state_save&action=save&save_name="+save_name, //"data": data, "data": JSON.stringify(data), // change to this.. "dataType": "json", "type": "POST", "success": function (response) { //console.log(response); } } ); },
How can I use Ember Data's FixtureAdapter with real API content
I want to populate my Ember Data fixtures with real content taken from an API data dump instead of the standard fixtures. I don't want to use the API directly. I want to do this so I can imitate a local instance of the API data. How can I streamline this and also how might I configure the adapter to allow this? Consider this default FIXTURE: App.Comment = DS.Model.extend({ article: DS.belongsTo('article'), author: DS.belongsTo('user'), dateCreated: DS.attr('date', {readOnly: true}), dateModified: DS.attr('date', {readOnly: true}), description: DS.attr('string') }); App.Comment.FIXTURES = [{ id: 1, temp: 1, author: 1, dateCreated: 'Mon Jul 28 2014 12:00:00 GMT+1000 (EST)', dateModified: null, description: 'lorem ipsum' }]; Consider this API response: {"comments": [ { "articleID": 1, "description": "I am a comment", "authorID": 1, "dateCreated": "2014-09-04T02:39:00", "createdBy": "Elise Chant", "dateModified": "2014-09-04T02:39:00", "id": 1 }, { "articleID": 1, "description": "I am another comment", "authorID": 1, "dateCreated": "2014-09-04T02:48:00", "createdBy": "Elise Chant", "dateModified": "2014-09-04T02:48:00", "id": 2 } ]}
Get your backend to provide a data dump method from the API with the resources, such as: http://myapi/dump The method shall return a zipped folder of files representing the site's resources, such as Comment.js, User.js, Article.js. Each of these files should return the JSON resource wrapped by DS.Model.FIXTURES array, such as App.Comment.FIXTURES = [ { "articleID": 1, "userID": 1, "description": "I am a comment.", "category": 0, "authorID": 1, "dateCreated": "2014-09-04T02:39:00", "id": 1 }, // ... ]; Remember to make sure that each .js file is an included script tag in html and appears after its corresponding model. Check this is available in the Browser's Developer Console. Finally, In order to correctly connect asynchronous relationships such as articleID and authorID, property names need to be normalised like that would be if you were using normal fixture data. So configure the adapter to strip 'ID' from the end of any belongsTo relationships by parsing the FIXTURES payload: if (App.ENV === 'DEV') { App.ApplicationSerializer = DS.JSONSerializer.extend({ // access to the payload extractArray: function(store, type, payload) { var self = this; return payload.map(function(item) { // normalise incoming data return self.normalize(type, item); }); }, normalize: function(type, hash) { if (!hash) { return hash; } var normalizedHash = {}, normalizedProp; for (var prop in hash) { if (!!prop.match(/ID$/)) { // belongs to / has Many attribute // remove 'ID' from the end of the property name normalizedProp = prop.substr(0, prop.length-2); } else { // regular attribute normalizedProp = prop; } normalizedHash[normalizedProp] = hash[prop]; } this.normalizeId(normalizedHash); this.normalizeUsingDeclaredMapping(type, normalizedHash); this.applyTransforms(type, normalizedHash); return normalizedHash; } }); App.ApplicationAdapter = DS.FixtureAdapter.extend({ simulateRemoteResponse: true, latency: 1000, // This "unsets" serializer so that the store will lookup the proper serializer // #see https://github.com/emberjs/data/issues/1333 serializer: function() { return; }.property() }); } else { DS.CustomRESTSerializer = DS.RESTSerializer.extend({ keyForRelationship: function(key, kind) { if (kind === 'belongsTo') { return key + 'ID'; } else { return key; } } }); App.ApplicationAdapter = DS.RESTAdapter.extend({ host: App.HOST, namespace: App.NAMESPACE, ajax: function(url, method, hash) { hash = hash || {}; // hash may be undefined hash.crossDomain = true; hash.xhrFields = {withCredentials: true}; return this._super(url, method, hash); }, defaultSerializer: 'DS/customREST' }); }