12

Once the button in my flask template is pressed I'd like it to call a python function defined in app.py that I made to be available to be called within the template by typing the following below where I define the function:

Example function in app.py:

@app.route('/foo')
def foo(x,y):
    pass
app.jinja_env.globals.update(foo=foo)

Template:

<button  type="button" onclick="myFunction(this)" name="enable" id="{{counter}}"> Enable </button>

In my button I have the onclick attribute just to test that the correct button out of many is pressed using javascript like such:

{% block scripts %}
    {{ super() }}
    <script>
    function myFunction(elem){
        if(confirm('Are you sure you want to ' + elem.name) == true){
            alert("its done.");            
         }
         else { 
            return false;
         }          
    }
    </script>
{% endblock %}

The issue I'm facing is I need the function that I'm making available within the template to correspond to the correct button. For example, if the button says Enable, then I need to call the enable function defined already or otherwise if the button corresponds to false, I'd like the disable function to be used.

I feel like I'm headed in the right direction but can't get past this part. Please be as detailed as you can.

Jonathan Hall
  • 75,165
  • 16
  • 143
  • 189
Sean
  • 141
  • 1
  • 1
  • 8

1 Answers1

20

If you want to execute your function without generating a request to the server, then your function must be defined in JavaScript. Otherwise, you need to fire an HTTP request.

Now in your case, if all you're trying to do is enable/disable buttons, it would make sense to do all that in javascript (no need to go to the server).

Example:

<button type="button" onclick="disableButton(this)" name="enable">Enable</button>

javascript

function disableButtonState(elem) {
    if(confirm('Are you sure you want to disable this button?') == true) {
        elem.disabled = true;
        alert("its done.");            
    }
    else { 
        return false;
    }          
}

However if what you want is to call a method on your server that, for example, sends an email, then you should use a form POST/GET or AJAX POST/GET

Example:

app.py

@app.route('/foo', methods=['GET', 'POST'])
def foo(x=None, y=None):
    # do something to send email
    pass

template

<form action="/foo" method="post">
    <button type="submit" value="Send Email" />
</form>

When you click the "Send Email" button an HTTP POST request is sent to "/foo" on your application. Your function foo can now extract some data from the request and do whatever it wants to do on the server side and then return a response to the client web browser.

It would suggest going through the Flask Tutorial to get a better understanding of client/server interactions when it comes to web applications built with Flask.

ARZ
  • 2,461
  • 3
  • 34
  • 56
junnytony
  • 3,455
  • 1
  • 22
  • 24
  • Thanks for the reply. So from what you're telling me I just need to use JavaScript for my specific situation. The biggest issue I face is trying to call the actual python function within the JS. can you point me in the right direction or give an example? – Sean Jun 17 '15 at 23:22
  • You can't call a python function directly from JavaScript since in this case your JavaScript is running on a browser on the client and your Python function exists on the server. The only way to make that communication happen is to send an HTTP request to your server. That's what my answer is suggesting. If what you're trying to do is purely visual, like disabling or hiding a button then this can be achieved completely in JavaScript. Is this what you're intending? – junnytony Jun 17 '15 at 23:31
  • See my edit for manipulating button with javascript, if that's what you're trying to do – junnytony Jun 17 '15 at 23:59
  • Ah I see. I am not trying to just change the text on the server page without something actually happening on the backend. The function I have accomplishes this completely what I need disabled on the server side. So I do need a HTTP request to do this then. I'll search about it tomorrow. Thank you for the guidance. – Sean Jun 18 '15 at 04:25