mirror of
https://github.com/caronc/apprise-api.git
synced 2025-01-25 07:19:02 +01:00
604 lines
22 KiB
HTML
604 lines
22 KiB
HTML
{% extends 'base.html' %}
|
|
{% load i18n %}
|
|
{% block body %}
|
|
{% if STATEFUL_MODE != 'disabled' %}
|
|
<h4>{% trans "Management for Config ID:" %} <code class="config-id">{{ key }}</code></h4>
|
|
<div class="row">
|
|
<div class="col s12">
|
|
<ul class="tabs config-overview">
|
|
<li class="tab col s3"><a class="active" href="#overview"><i class="material-icons">info</i>
|
|
{% trans "Overview" %}</a></li>
|
|
<li class="tab {% if CONFIG_LOCK %}tab-locked {% endif %}col s3"><a href="#config"><i class="material-icons">{% if not CONFIG_LOCK %}settings{% else %}lock{% endif %}</i> {%trans "Configuration" %}</a>
|
|
</li>
|
|
<li class="tab {% if CONFIG_LOCK %}tab-locked {% endif %}col s3"><a href="#review"><i class="material-icons">{% if not CONFIG_LOCK %}web{% else %}lock{% endif %}</i> {%trans "Review" %}</a>
|
|
</li>
|
|
<li class="tab col s3"><a href="#notify"><i class="material-icons">announcement</i> {%trans "Notifications" %}</a>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
<div id="overview" class="col s12">
|
|
{% if not CONFIG_LOCK %}
|
|
<div class="section">
|
|
<h5>{% trans "Getting Started" %}</h5>
|
|
<ol>
|
|
<li>
|
|
{% blocktrans %}
|
|
Here is where you can store your Apprise configuration associated with the key <code>{{key}}</code>.
|
|
{% endblocktrans %}
|
|
For some examples on how to build a development environment around this, <strong><a href="{% url 'welcome'%}?key={{key}}">click here</a></strong>.
|
|
</li>
|
|
<li>
|
|
{% blocktrans %}
|
|
In the future you can return to this configuration screen at any time by placing the following into your
|
|
browser:
|
|
{% endblocktrans %}
|
|
<code>{{request.scheme}}://{{request.META.HTTP_HOST}}{{BASE_URL}}/cfg/{{key}}</code>
|
|
</li>
|
|
<li>
|
|
{% blocktrans %}
|
|
Use the <strong><i class="material-icons">settings</i> Configuration</strong> section to prepare and save your Apprise configuration.
|
|
{% endblocktrans %}
|
|
<ul>
|
|
<li>
|
|
{% blocktrans %}
|
|
<i class="material-icons">lightbulb_outline</i>You can always refer to the
|
|
<a href="https://github.com/caronc/apprise/wiki#notification-services">Apprise Wiki</a> if you're having troubles assembling your URL(s).
|
|
{% endblocktrans %}
|
|
</li>
|
|
</ul>
|
|
</li>
|
|
<li>
|
|
{% blocktrans %}
|
|
Use the <strong><i class="material-icons">announcement</i> Notifications</strong> section to test out your saved configuration.
|
|
{% endblocktrans %}
|
|
</li>
|
|
</ol>
|
|
</div>
|
|
<div class="section no-config">
|
|
<div class="divider"></div>
|
|
<p>
|
|
<i class="material-icons info">info</i>
|
|
{% blocktrans %}Once you have successfully load <i>at least one Apprise URL</i> on the <strong><i class="material-icons">settings</i> Configuration</strong> section, your loaded entries will display here.
|
|
{% endblocktrans %}
|
|
</p>
|
|
<div class="divider"></div>
|
|
</div>
|
|
{% else %}
|
|
<div class="section">
|
|
<h5>{% trans "Apprise Configuration is Locked" %}</h5>
|
|
<p>
|
|
{% blocktrans %}At this time, the administrator of this server has locked down all configuration. This means
|
|
That pre-created configuration is securely hidden for the purpose of notification transmission only.
|
|
|
|
New configuration can not be set, and existing configuration can not be modified or viewed.
|
|
{% endblocktrans %}</p>
|
|
</div>
|
|
{% endif %}
|
|
<div class="has-config">
|
|
<div class="section">
|
|
<h5>{% trans "Working Remotely With Your Configuration" %}</h5>
|
|
<h6>{% trans "Using The Apprise CLI" %}</h6>
|
|
<p>
|
|
{% blocktrans %}The following command would cause apprise to directly notify all of your services:{% endblocktrans %}
|
|
<pre><code class="bash">apprise --body="Test Message" \<br/>
|
|
apprise{% if request.is_secure %}s{% endif %}://{{request.META.HTTP_HOST}}{{BASE_URL}}/<em>{{key}}</em>/?tags=all</code></pre>
|
|
{% blocktrans %}Send one or more attachments like this:{% endblocktrans %}
|
|
<pre><code class="bash">apprise --body="Test Message" \<br/>
|
|
apprise{% if request.is_secure %}s{% endif %}://{{request.META.HTTP_HOST}}{{BASE_URL}}/<em>{{key}}</em>/?tags=all \<br/>
|
|
--attach=/path/to/an/attachment.jpeg \ <br/>
|
|
--attach=https://raw.githubusercontent.com/caronc/apprise/master/apprise/assets/themes/default/apprise-logo.png<br/>
|
|
</code></pre>
|
|
</p>
|
|
{% if not CONFIG_LOCK %}
|
|
<p>
|
|
{% blocktrans %}The following command would cause apprise to retrieve the configuration loaded and
|
|
send a test notification to all of your added services:{% endblocktrans %}
|
|
<br />
|
|
<pre><code class="bash">apprise --body="Test Message" --tag=all \<br/>
|
|
--config={{request.scheme}}://{{request.META.HTTP_HOST}}{{BASE_URL}}/get/<em>{{key}}</em></code></pre>
|
|
{% blocktrans %}You may also create an <a href="https://github.com/caronc/apprise/wiki/config#cli" target="_blank">Apprise configuration file</a> that contains this line somewhere in it:{% endblocktrans %}
|
|
<pre><code class="bash">include {{request.scheme}}://{{request.META.HTTP_HOST}}{{BASE_URL}}/get/<em>{{key}}</em></code></pre>
|
|
{% blocktrans %}By leveraging the <em>include</em> directive, it will automatically be referenced for future calls to the <code>apprise</code> tool. All future calls using Apprise now simplify to:{% endblocktrans %}
|
|
<pre><code class="bash">apprise --body="Test Message" --tag=all</em></code></pre>
|
|
</p>
|
|
|
|
{% endif %}
|
|
<h6>{% trans "Using CURL" %}</h6>
|
|
<p>
|
|
{% blocktrans %}The following command would cause the apprise api to notify all of your services:{% endblocktrans %}
|
|
<pre><code class="bash">curl -X POST \<br/>
|
|
-F "body=Test Message" \<br/>
|
|
-F "tags=all" \<br/>
|
|
http{% if request.is_secure %}s{% endif %}://{{request.META.HTTP_HOST}}{{BASE_URL}}/notify/<em>{{key}}</em></code></pre>
|
|
{% blocktrans %}Send one or more attachments like this:{% endblocktrans %}
|
|
<pre><code class="bash">curl -X POST \<br/>
|
|
-F "tags=all" \<br/>
|
|
-F "body=Test Message" \<br/>
|
|
-F attach1=@Screenshot-1.png \<br/>
|
|
-F attach2=@/my/path/to/Apprise.doc \<br/>
|
|
http{% if request.is_secure %}s{% endif %}://{{request.META.HTTP_HOST}}{{BASE_URL}}/notify/<em>{{key}}</em></code></pre>
|
|
{% blocktrans %}Sends a notification to our endpoints with an attachment{% endblocktrans %}
|
|
<pre><code class="bash">
|
|
curl -X POST \<br/>
|
|
-F "tag=all" \ <br/>
|
|
-F "attach=https://raw.githubusercontent.com/caronc/apprise/master/apprise/assets/themes/default/apprise-logo.png" \ <br/>
|
|
"{{request.scheme}}://{{request.META.HTTP_HOST}}{{BASE_URL}}/notify/<em>{{key}}</em>"</code></pre>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div id="config" class="col s12">
|
|
{% if not CONFIG_LOCK %}
|
|
<p>
|
|
{% blocktrans %}Define your configuration below:{% endblocktrans %}
|
|
<form id="addconfig" action="{% url "add" key %}" method="post">
|
|
{{ form_cfg }}
|
|
<button class="btn waves-effect waves-light" type="submit" name="action">{% trans "Save Configuration" %}
|
|
<i class="material-icons right">save</i>
|
|
</button>
|
|
</form>
|
|
</p>
|
|
{% else %}
|
|
<h5>{% trans "Your Configuration Is Locked" %}</h5>
|
|
<p>{% blocktrans %}Access to your configuration has been disabled by your administrator.{% endblocktrans %}
|
|
|
|
{% endif %}
|
|
</div>
|
|
|
|
<div id="review" class="col s12">
|
|
{% if not CONFIG_LOCK %}
|
|
<p>
|
|
{% blocktrans %}The following URLs have been detected:{% endblocktrans %}
|
|
</p>
|
|
<div class="section">
|
|
<h5>{% trans "Loaded Configuration" %}</h5>
|
|
<div id="url-list"></div>
|
|
<div id="url-list-progress" class="progress" style="width:70%">
|
|
<div class="indeterminate"></div>
|
|
</div>
|
|
</div>
|
|
{% else %}
|
|
<h5>{% trans "Your Configuration Is Locked" %}</h5>
|
|
<p>{% blocktrans %}Access to your configuration has been disabled by your administrator.{% endblocktrans %}
|
|
{% endif %}
|
|
</div>
|
|
|
|
<div id="notify" class="col s12">
|
|
<p>
|
|
{% blocktrans %}
|
|
You can send a notification using the loaded configuration:
|
|
{% endblocktrans %}
|
|
<form id="donotify" enctype="multipart/form-data" action="{% url "notify" key %}" method="post">
|
|
{{ form_notify }}
|
|
<button class="btn waves-effect waves-light" type="submit" name="action">{% trans "Send Notification" %}
|
|
<i class="material-icons right">send</i>
|
|
</button>
|
|
</form>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
{% else %}
|
|
<div class="section">
|
|
<h4>{% trans "Persistent Store Endpoints" %}</h4>
|
|
<p>{% blocktrans %}The administrator of this system has disabled persistent storage.{% endblocktrans %}</p>
|
|
</div>
|
|
{% endif %}
|
|
{% endblock %}
|
|
{% block jsfooter %}
|
|
|
|
{% if STATEFUL_MODE != 'disabled' %}
|
|
async function main_init(){
|
|
|
|
// disable the notification tab until we know for certain
|
|
// a notification is possible
|
|
document.querySelector('.config-overview li a[href="#notify"]')
|
|
.parentNode.classList.add('disabled');
|
|
|
|
// Disable any has-config entries
|
|
document.querySelector('.has-config')
|
|
.style.display = 'none';
|
|
|
|
// Ensure we show our progress loader and reset our url list
|
|
document.querySelector('#url-list').textContent = ''
|
|
document.querySelector('#url-list-progress').style.display = null;
|
|
|
|
{% if not CONFIG_LOCK %}
|
|
// Ensure no-config sections are visible
|
|
document.querySelector('.no-config')
|
|
.style.display = null;
|
|
{% endif %}
|
|
|
|
// perform a tag retrieval; start with 'all'
|
|
let tags = ['all'];
|
|
|
|
let jsonResponse = await fetch('{% url "json_urls" key %}/?privacy=1', {
|
|
method: 'GET',
|
|
})
|
|
if(jsonResponse.status == 204) {
|
|
// Take an early exit
|
|
document.querySelector('#url-list-progress').style.display = 'none';
|
|
document.querySelector('#url-list').textContent = '{% trans "There are no URLs defined. Click on the ⚙️ Configuration tab and define some." %}'
|
|
return;
|
|
|
|
} else if(jsonResponse.status != 200) {
|
|
// Take an early exit
|
|
document.querySelector('#url-list-progress').style.display = 'none';
|
|
document.querySelector('#url-list').textContent = '{% trans "💣 An error occurred retrieving the list of loaded Apprise URL(s)" %}'
|
|
return;
|
|
}
|
|
|
|
// Initialize our tags making it easy for an end user to
|
|
// choose from. Tags are based off ones found in the saved
|
|
// configuration.
|
|
const data = await jsonResponse.json();
|
|
const external_data = tags.concat(data.tags).reduce(function(result, item) {
|
|
result[item] = null;
|
|
return result;
|
|
}, {})
|
|
|
|
const chipElement = document.querySelector('.chips');
|
|
M.Chips.init(chipElement, {
|
|
placeholder: 'Optional Tag',
|
|
secondaryPlaceholder: 'Another Tag',
|
|
autocompleteOptions: {
|
|
data: external_data,
|
|
minLength: 0
|
|
},
|
|
onChipAdd: function(e, chip) {
|
|
var $this = this;
|
|
$this.chipsData.forEach(function(e, index) {
|
|
if(!(e.tag in external_data))
|
|
$this.deleteChip(index);
|
|
})
|
|
}
|
|
});
|
|
|
|
// our GET parameters to be treated as template values
|
|
var tagRe = new RegExp('[^[A-Za-z0-9_-]+');
|
|
const params = new Proxy(new URLSearchParams(window.location.search), {
|
|
get: (searchParams, prop) => searchParams.get(prop),
|
|
});
|
|
if (params.tag) {
|
|
params.tag.split(tagRe).forEach(function (tag, index) {
|
|
M.Chips.getInstance(chipElement).addChip({tag: tag, image: ''});
|
|
});
|
|
}
|
|
if (params.title) {
|
|
document.querySelector('#id_title').value = params.title;
|
|
}
|
|
if (params.body) {
|
|
document.querySelector('#id_body').value = params.body;
|
|
}
|
|
|
|
// Now build our our loaded list of configuration for our welcome page
|
|
let urlList = document.createElement('ul');
|
|
|
|
// Create a list item for each url retrieved
|
|
data.urls.forEach(function (entry) {
|
|
let code = document.createElement('code');
|
|
let li = document.createElement('li');
|
|
code.textContent = entry.url;
|
|
li.setAttribute('class', 'card-panel');
|
|
li.appendChild(code);
|
|
|
|
// Get our tags associate with the URL
|
|
entry.tags.forEach(function (tag) {
|
|
let chip = document.createElement('div');
|
|
chip.setAttribute('class', 'chip');
|
|
chip.setAttribute('name', tag);
|
|
chip.setAttribute('href', `?tag=${tag}#notify`);
|
|
chip.textContent = `🏷️ ${tag}`;
|
|
li.appendChild(chip);
|
|
|
|
chip.addEventListener('click', function(e) {
|
|
e.preventDefault();
|
|
window.history.pushState(null, null, this.getAttribute('href'));
|
|
document.querySelector('.config-overview li a[href="#notify"]').click()
|
|
|
|
const chipElement = document.querySelector('.chips');
|
|
M.Chips.getInstance(chipElement).addChip({tag: this.getAttribute('name'), image: ''});
|
|
}, false);
|
|
});
|
|
|
|
urlList.appendChild(li);
|
|
});
|
|
|
|
// Store our new list
|
|
document.querySelector('#url-list-progress').style.display = 'none';
|
|
document.querySelector('#url-list').textContent = ''
|
|
if(urlList.childNodes.length > 0) {
|
|
|
|
// Ensure has-config sections are visible
|
|
document.querySelector('.has-config')
|
|
.style.display = null;
|
|
|
|
// Remove our restrictions on sending notifications
|
|
document.querySelector('.config-overview li a[href="#notify"]')
|
|
.parentNode.classList.remove('disabled');
|
|
|
|
{% if not CONFIG_LOCK %}
|
|
// Disable any no-config entries
|
|
document.querySelector('.no-config')
|
|
.style.display = 'none';
|
|
{% endif %}
|
|
|
|
// Save our list to the screen
|
|
document.querySelector('#url-list').appendChild(urlList);
|
|
|
|
{% if not CONFIG_LOCK %}
|
|
//
|
|
// Load our configuration now into the configuration tab
|
|
//
|
|
let response = await fetch('{% url "get" key %}', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json;charset=utf-8'
|
|
},
|
|
});
|
|
|
|
if(response.status == 204)
|
|
{
|
|
// no problem; we simply have no content to retrieve
|
|
return '';
|
|
}
|
|
else if(response.status == 200)
|
|
{
|
|
// configuration found
|
|
|
|
// get our results
|
|
let result = await response.json();
|
|
|
|
// Set our configuration so it's visible
|
|
document.querySelector('#id_config').value = result.config;
|
|
// Set our format
|
|
document.querySelector('#id_format').value = result.format;
|
|
|
|
// dispatch our event to update our select box
|
|
if (typeof(Event) === 'function') {
|
|
var event = new Event('change');
|
|
} else { // for IE11
|
|
var event = document.createEvent('Event');
|
|
event.initEvent('change', true, true);
|
|
}
|
|
document.querySelector('#id_format').dispatchEvent(event);
|
|
}
|
|
{% endif %}
|
|
} else {
|
|
document.querySelector('#url-list').textContent = '{% trans "There are no Apprise URL(s) loaded." %}'
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
function config_init() {
|
|
// over-ride manual submit for a nicer user experience
|
|
document.querySelector('#addconfig').onsubmit = function(event) {
|
|
event.preventDefault();
|
|
const form = this;
|
|
const body = new URLSearchParams(new FormData(form));
|
|
|
|
content = document.querySelector('#id_config')
|
|
.value.replace(/^\s+|\s+$/gm,'');
|
|
if(content.length) {
|
|
// perform our status check
|
|
let response = fetch('{% url "add" key %}', {
|
|
method: 'POST',
|
|
body: body,
|
|
}).then(function(response) {
|
|
if(response.status == 200)
|
|
{
|
|
// update our settings
|
|
main_init();
|
|
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Save" %}',
|
|
'{% trans "Successfully saved the specified URL(s)." %}',
|
|
'success'
|
|
);
|
|
} else if(response.status == 500) {
|
|
// Disk issue
|
|
Swal.fire(
|
|
'{% trans "Save" %}',
|
|
'{% trans "There was an issue writing the configuration to disk. Check your file permissions and try again." %}',
|
|
'error'
|
|
);
|
|
} else {
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Save" %}',
|
|
'{% trans "Failed to save the specified URL(s). Check your syntax and try again." %}',
|
|
'error'
|
|
);
|
|
}
|
|
});
|
|
} else {
|
|
// Perform Delete
|
|
// perform our status check
|
|
let response = fetch('{% url "del" key %}', {
|
|
method: 'POST',
|
|
body: body,
|
|
}).then(function(response) {
|
|
if(response.status == 200 || response.status == 204)
|
|
{
|
|
// update our settings
|
|
main_init();
|
|
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Delete" %}',
|
|
'{% trans "Successfully removed configuration." %}',
|
|
'success'
|
|
);
|
|
} else {
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Delete" %}',
|
|
'{% trans "There was an issue removing the configuration." %}',
|
|
'error'
|
|
);
|
|
}
|
|
});
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function form_file_input_hack() {
|
|
/*
|
|
A small hack to remformat all `<input file>` upload options to a more cleaner
|
|
and presentable look by wrapping it like this:
|
|
|
|
<span class="btn btn-file">
|
|
<i class="material-icons left">cloud_upload</i>
|
|
Browse<input type="file">
|
|
</span>
|
|
*/
|
|
document.querySelectorAll('input[type=file]').forEach(function (entry) {
|
|
const div = document.createElement('div');
|
|
const selected = document.createElement('div');
|
|
selected.style.display = 'none';
|
|
selected.setAttribute('class', 'file-selected');
|
|
const span = document.createElement('span')
|
|
span.setAttribute('class', 'btn btn-file waves-effect waves-light');
|
|
const i = document.createElement('i')
|
|
span.textContent = '{% trans "Browse" %}';
|
|
i.setAttribute('class', 'material-icons right');
|
|
i.textContent = 'folder';
|
|
div.appendChild(span);
|
|
span.appendChild(i);
|
|
entry.before(div)
|
|
span.appendChild(entry);
|
|
span.after(selected);
|
|
|
|
entry.addEventListener('change', function(e){
|
|
const selected = this.parentNode.nextElementSibling;
|
|
const file = e.target.files[0];
|
|
selected.style.display = 'block'
|
|
selected.textContent = file.name;
|
|
});
|
|
});
|
|
}
|
|
function notify_init() {
|
|
// over-ride manual submit for a nicer user experience
|
|
document.querySelector('#donotify').onsubmit = function(event) {
|
|
event.preventDefault();
|
|
|
|
const chipElement = document.querySelector('.chips');
|
|
const chipInput = document.querySelector('.chips > input');
|
|
if(chipInput.value) {
|
|
// This code just handles text typed in the tag section that was
|
|
// not submitted. This forces any lingering un-committed text
|
|
// into a tag just prior to it's submission
|
|
const ev = new KeyboardEvent('keydown', {
|
|
altKey:false,
|
|
bubbles: true,
|
|
cancelBubble: false,
|
|
cancelable: true,
|
|
charCode: 0,
|
|
code: "Enter",
|
|
composed: true,
|
|
ctrlKey: false,
|
|
currentTarget: null,
|
|
defaultPrevented: true,
|
|
detail: 0,
|
|
eventPhase: 0,
|
|
isComposing: false,
|
|
isTrusted: true,
|
|
key: "Enter",
|
|
keyCode: 13,
|
|
location: 0,
|
|
metaKey: false,
|
|
repeat: false,
|
|
returnValue: false,
|
|
shiftKey: false,
|
|
type: "keydown",
|
|
which: 13
|
|
});
|
|
chipInput.dispatchEvent(ev);
|
|
}
|
|
|
|
// store tags (as comma separated string) from materialize chip type into form
|
|
document.querySelector('#id_tag').value = M.Chips.getInstance(chipElement).chipsData.reduce(
|
|
function(s, a){
|
|
s.push(a.tag)
|
|
return s;
|
|
}, []).join(",")
|
|
|
|
// our Form
|
|
const form = new FormData(this);
|
|
|
|
// perform our notification
|
|
Swal.fire(
|
|
'{% trans "Notification" %}',
|
|
'{% trans "Sending notification(s)..." %}',
|
|
);
|
|
Swal.showLoading()
|
|
let response = fetch('{% url "notify" key %}', {
|
|
method: 'POST',
|
|
body: form,
|
|
headers: {
|
|
'Accept': 'text/html',
|
|
'X-Apprise-Log-Level': 'info'
|
|
}
|
|
}).then(function(response) {
|
|
response.text().then(function (html) {
|
|
if(response.status == 200)
|
|
{
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Notification" %}',
|
|
'{% trans "Successfully sent the notification(s)." %}' + html,
|
|
'success'
|
|
);
|
|
} else if(response.status == 424) {
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Notification" %}',
|
|
'{% trans "One or more of the notification(s) were not sent." %}' + html,
|
|
'warning'
|
|
);
|
|
} else {
|
|
// user notification
|
|
Swal.fire(
|
|
'{% trans "Notification" %}',
|
|
'{% trans "Failed to send the notification(s)." %}' + html,
|
|
'error'
|
|
);
|
|
}
|
|
});
|
|
});
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/* Initialize our page */
|
|
main_init();
|
|
{% if not CONFIG_LOCK %}
|
|
/* Initialze our configuration */
|
|
config_init();
|
|
|
|
{% endif %}
|
|
notify_init();
|
|
form_file_input_hack();
|
|
{% endif %}
|
|
{% endblock %}
|
|
|
|
{% block onload %}
|
|
{% if STATEFUL_MODE != 'disabled' %}
|
|
{{ block.super }}
|
|
document.querySelector('label [for="id_tag"]')
|
|
{
|
|
// create a new div with the class 'chips' assigned to it
|
|
const element = document.createElement('div')
|
|
let refNode = document.querySelector('label[for="id_tag"]')
|
|
element.classList.add("chips")
|
|
element.style.margin = '0'
|
|
refNode.parentNode.insertBefore(element, refNode.nextSibling)
|
|
}
|
|
|
|
// Hide tag field since we use the pretty Materialize Chip setup instead
|
|
document.querySelector('#id_tag').style.display = 'none';
|
|
{% endif %}
|
|
{% endblock %}
|