reCAPTCHA not loading properly in XHTML document - xhtml
I am writing an SHTML CSS & JS document into an XML document which uses an XHTML parser.
When I try to load the captcha into it, it briefly shows up as a white box then disappears, never to be seen again inside the renderer.
I understand that this might be caused by the scripts being loaded in the wrong order but I don't have the luxury of the async defer keywords that are recommended by google simply because the parser won't accept them.
<script src="https://www.google.com/recaptcha/api.js" type="text/javascript"></script>
<script type="text/javascript"><![CDATA[
class Token {
constructor(tokenInstance) {
this.props = tokenInstance;
this.getStatus();
this.setOnConfirm();
this.setAdditionalProps();
}
setAdditionalProps() {
this.props.claimed = false;
this.props.claimTokensMessage = "Claim free DAI, DeFi & ETH tokens";
this.props.claimedMessage = "You have already claimed DAI, DeFi & ETH tokens, stay posted for new freebies!";
this.props.serverError = "Server Error, please try again later";
this.props.success = "ETH, DeFi & DAI tokens will be sent to your address shortly!";
this.props.fillCaptchaMessage = "Please click the captcha below to receive the tokens";
}
getBaseURL() {
return "http://192.168.26.189:8080/api/"
}
getStatus() {
let suffix = "claimed?userAddress=" + this.props.ownerAddress;
fetch(this.getBaseURL() + suffix).then((res) => {
if(parseInt(res.status) === 200) {
document.getElementById("message").innerHTML = this.props.claimTokensMessage;
} else if(parseInt(res.status) === 412) {
this.props.claimed = true;
document.getElementById("message").innerHTML = this.props.claimedMessage;
} else {
document.getElementById("message").innerHTML = this.props.serverError;
}
}).catch((err) => {
document.getElementById("message").innerHTML = err;
window.onConfirm = function() { window.close() };
});
}
setOnConfirm() {
window.onConfirm = () => {
if(this.props.claimed) {
window.close();
} else {
web3.personal.sign({ data: "To receive free tokens, you must reveal your public address to a smart contract. Is that ok?" }, (err, val) => {
if(err) { throw err; }
//user completes the request by filling the captcha
document.getElementById("message").innerHTML = this.props.success;
document.getElementById("captcha").submit();
window.onConfirm = function () { window.close(); };
});
}
};
}
render() {
let suffix = "discover?userAddress=" + this.props.ownerAddress;
return`
<div class="ui container">
<div class="ui segment">
<img src="">
<h3 id="message"></h3>
<form action=${this.getBaseURL() + suffix} id="captcha" method="post">
<button class="g-recaptcha" data-sitekey="6LeK070UAAAAAHuzSEGgoqbuLGuJq-GRDd-LA4kH" data-callback="onSubmit" hidden>fill captcha</button>
</form>
</div>
</div>
`;
}
}
web3.tokens.dataChanged = (oldTokens, updatedTokens) => {
const currentTokenInstance = web3.tokens.data.currentInstance;
document.getElementById('root').innerHTML = new Token(currentTokenInstance).render();
};
]]></script>
<div id="root"></div>
Related
Simple pug html form, make it send immediately on change of value rather than wait for submit button
I have a very simple pug file: for item in itemList form(method='post', action='/change') table tr td(width=100) td(width=200) | #{item.name} input(type='hidden', name='field' value=item.name) input(type='hidden', name='style' value='doublevalue') td(width=100) input(type='number', name='value' min=-20.0 max=80.00 step=0.01 value=+item.value) td(width=100) input(type='submit', value='Update') p end As you can see it produces a few trivial forms like this: (Each form is one 'line' which is a simple table.) (On the script side, it just reads each 'line' from a MySQL table, there are 10 or so of them.) So on the www page, the user either types in new number (say "8") or clicks the small arrows (say Up, changing it to 7.2 in the example) then the user must click submit and it sends the post. Quite simply, I would like it to be that when the user clicks a small arrows (say Up, changing it to 7.2 in the example) it immediately sends a submit-post. How do I achieve this? (It would be fine if the send happens, any time the user types something in the field, and/or, when the user clicks the Small Up And Down Buttons. Either/both is fine.) May be relevant: My pug file (and all my pug files) have this sophisticated line of code as line 1: include TOP.pug And I have a marvellous file called TOP.pug: html head style. html { font-family: sans-serif } td { font-family: monospace } body
I have a solution with javascript. // check if there are input[type="number"] to prevent errors if (document.querySelector('input[type="number"]')) { // add event for each of them document.querySelectorAll('input[type="number"]').forEach(function(el) { el.addEventListener('change', function (e) { // on change submit the parent (closest) form e.currentTarget.closest('form').submit() }); }); } Actually it is short but if you want to support Internet Explorer you have to add the polyfill script too. Internet Explorer does not support closest() with this snippet below we teach it. // polyfills for matches() and closest() if (!Element.prototype.matches) Element.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector; if (!Element.prototype.closest) { Element.prototype.closest = function(s) { var el = this; do { if (el.matches(s)) return el; el = el.parentElement || el.parentNode; } while (el !== null && el.nodeType === 1); return null; }; } Ajax form submit to node.js If you are interested in an ajax solution I put some code below just to blow your mind ;-) It should work instantly, I use it on one of my sites. You could use jQuery and save lines of code but I like it pure. (The ajax function and polyfills are utils so paste it anywhere) HTML (example) <form> <input type="hidden" name="field" value="field1"> <input type="hidden" name="style" value="style1"> <input type="number" name="value"> <input type="submit" value="update"> </form> <form> <input type="hidden" name="field" value="field2"> <input type="hidden" name="style" value="style2"> <input type="number" name="value"> <input type="submit" value="update"> </form> Javascript: event listener and prepare ajax call (note the callbacks). // check if there are forms to prevent errors if (document.querySelector('form')) { // add submit event for each form document.querySelectorAll('form').forEach(function (el) { el.addEventListener('submit', function (e) { e.currentTarget.preventDefault(); submitData(e.currentTarget); }); }); } // check if there are input[type="number"] to prevent errors if (document.querySelector('input[type="number"]')) { // add change event for each of them document.querySelectorAll('input[type="number"]').forEach(function (el) { el.addEventListener('change', function (e) { submitData(e.currentTarget.closest('form')); }); }); } // collect form data and send it function submitData(form) { // send data through (global) ajax function ajax({ url: '/change', method: 'POST', data: { field: form.querySelector('input[name="field"]').value, style: form.querySelector('input[name="style"]').value, value: form.querySelector('input[name="value"]').value, }, // callback on success success: function (response) { // HERE COMES THE RESPONSE console.log(response); // error is defined in (node.js res.json({error: ...})) if (response.error) { // make something red form.style.border = '1px solid red'; } if (!response.error) { // everything ok, make it green form.style.border = '1px solid green'; } // remove above styling setTimeout(function () { form.style.border = 'none'; }, 1000); }, // callback on error error: function (error) { console.log('server error occurred: ' + error) } }); } As told javascript utils (paste it anywhere like a library) // reusable ajax function function ajax(obj) { let a = {}; a.url = ''; a.method = 'GET'; a.data = null; a.dataString = ''; a.async = true; a.postHeaders = [ ['Content-type', 'application/x-www-form-urlencoded'], ['X-Requested-With', 'XMLHttpRequest'] ]; a.getHeaders = [ ['X-Requested-With', 'XMLHttpRequest'] ]; a = Object.assign(a, obj); a.method = a.method.toUpperCase(); if (typeof a.data === 'string') a.dataString = encodeURIComponent(a.data); else for (let item in a.data) a.dataString += item + '=' + encodeURIComponent(a.data[item]) + '&'; let xhReq = new XMLHttpRequest(); if (window.ActiveXObject) xhReq = new ActiveXObject('Microsoft.XMLHTTP'); if (a.method == 'GET') { if (typeof a.data !== 'undefined' && a.data !== null) a.url = a.url + '?' + a.dataString; xhReq.open(a.method, a.url, a.async); for (let x = 0; x < a.getHeaders.length; x++) xhReq.setRequestHeader(a.getHeaders[x][0], a.getHeaders[x][1]); xhReq.send(null); } else { xhReq.open(a.method, a.url, a.async); for (let x = 0; x < a.postHeaders.length; x++) xhReq.setRequestHeader(a.postHeaders[x][0], a.postHeaders[x][1]); xhReq.send(a.dataString); } xhReq.onreadystatechange = function () { if (xhReq.readyState == 4) { let response; try { response = JSON.parse(xhReq.responseText) } catch (e) { response = xhReq.responseText; } //console.log(response); if (xhReq.status == 200) { obj.success(response); } else { obj.error(response); } } } } // (one more) polyfill for Object.assign if (typeof Object.assign !== 'function') { // Must be writable: true, enumerable: false, configurable: true Object.defineProperty(Object, 'assign', { value: function assign(target, varArgs) { // .length of function is 2 if (target === null || target === undefined) { throw new TypeError('Cannot convert undefined or null to object'); } var to = Object(target); for (var index = 1; index < arguments.length; index++) { var nextSource = arguments[index]; if (nextSource !== null && nextSource !== undefined) { for (var nextKey in nextSource) { // Avoid bugs when hasOwnProperty is shadowed if (Object.prototype.hasOwnProperty.call(nextSource, nextKey)) { to[nextKey] = nextSource[nextKey]; } } } } return to; }, writable: true, configurable: true }); } // polyfills for matches() and closest() if (!Element.prototype.matches) Element.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector; if (!Element.prototype.closest) { Element.prototype.closest = function (s) { var el = this; do { if (el.matches(s)) return el; el = el.parentElement || el.parentNode; } while (el !== null && el.nodeType === 1); return null; }; } In node.js (e.g. express route) // the route in node.js app.post('/change', (req, res) => { // your logic here let field = req.body.field; let style = req.body.style; let value = req.body.value; // ... // response result res.json({ databaseError: false, // or true additionalStuff: 'message, markup and other things ...', }); });
Integrate Firebase authentication in an AngularDart app
I can not find a clear and complete explanation on how to integrate Firebase authentication (Google, e-mail, phone, ...) into an AngularDart (version 5) app. Can you give it to me? Or, can you tell me where to find it? Thank you!
firebase_dart_ui works for me. The repository contains an example, too.
Likewise. I spent a long time trying to find something...anything. While new to Angular Dart these examples that actually worked where a good find. The repo has examples for simple auth + phone and storage etc. The md talks a little about what you need to setup in the Firebase console, you'll need to allow your host access for the Firebase generated keys in cloud console as well. As Gazihan mentioned, the firebase_dart_ui package will do signing in with Google etc if you want to use Firebase UI, agreed, it's hard to follow in terms of finding steps. I had to change dev dependencies to get it to working but it did: dev_dependencies: build_runner: ^1.7.0 build_test: ^0.10.8 build_web_compilers: ^2.4.0 The following user/pwd signin example from the FirebaseDart repo is more straight forward and could help. First, get your App details from your projects Firebase settings page. index.dart import 'dart:convert'; import 'dart:html'; import 'package:firebase/firebase.dart' as fb; main() async { try { fb.initializeApp( apiKey: "your api key", authDomain: "your auth domain", databaseURL: "your database url", storageBucket: "your storage bucket"); AuthApp(); } on fb.FirebaseJsNotLoadedException catch (e) { print(e); } } class AuthApp { final fb.Auth auth; final FormElement registerForm; final InputElement email, password; final AnchorElement logout; final TableElement authInfo; final ParagraphElement error; final SelectElement persistenceState, verifyEmailLanguage; final ButtonElement verifyEmail; final DivElement registeredUser, verifyEmailContainer; AuthApp() : this.auth = fb.auth(), this.email = querySelector("#email"), this.password = querySelector("#password"), this.authInfo = querySelector("#auth_info"), this.error = querySelector("#register_form p"), this.logout = querySelector("#logout_btn"), this.registerForm = querySelector("#register_form"), this.persistenceState = querySelector("#persistent_state"), this.verifyEmail = querySelector('#verify_email'), this.verifyEmailLanguage = querySelector('#verify_email_language'), this.registeredUser = querySelector("#registered_user"), this.verifyEmailContainer = querySelector("#verify_email_container") { logout.onClick.listen((e) { e.preventDefault(); auth.signOut(); }); this.registerForm.onSubmit.listen((e) { e.preventDefault(); var emailValue = email.value.trim(); var passwordvalue = password.value.trim(); _registerUser(emailValue, passwordvalue); }); // After opening if (auth.currentUser != null) { _setLayout(auth.currentUser); } // When auth state changes auth.onAuthStateChanged.listen(_setLayout); verifyEmail.onClick.listen((e) async { verifyEmail.disabled = true; verifyEmail.text = 'Sending verification email...'; try { // you will get the verification email in selected language auth.languageCode = _getSelectedValue(verifyEmailLanguage); // url should be any authorized domain in your console - we use here, // for this example, authDomain because it is whitelisted by default // More info: https://firebase.google.com/docs/auth/web/passing-state-in-email-actions await auth.currentUser.sendEmailVerification( fb.ActionCodeSettings(url: "followmyvoyage.firebaseapp.com")); verifyEmail.text = 'Verification email sent!'; } catch (e) { verifyEmail ..text = e.toString() ..style.color = 'red'; } }); } _registerUser(String email, String password) async { if (email.isNotEmpty && password.isNotEmpty) { var trySignin = false; try { // Modifies persistence state. More info: https://firebase.google.com/docs/auth/web/auth-state-persistence var selectedPersistence = _getSelectedValue(persistenceState); await auth.setPersistence(selectedPersistence); await auth.createUserWithEmailAndPassword(email, password); } on fb.FirebaseError catch (e) { if (e.code == "auth/email-already-in-use") { trySignin = true; } } catch (e) { error.text = e.toString(); } if (trySignin) { try { await auth.signInWithEmailAndPassword(email, password); } catch (e) { error.text = e.toString(); } } } else { error.text = "Please fill correct e-mail and password."; } } String _getSelectedValue(SelectElement element) => element.options[element.selectedIndex].value; void _setLayout(fb.User user) { if (user != null) { registerForm.style.display = "none"; registeredUser.style.display = "block"; email.value = ""; password.value = ""; error.text = ""; var data = <String, dynamic>{ 'email': user.email, 'emailVerified': user.emailVerified, 'isAnonymous': user.isAnonymous, 'metadata.creationTime': user.metadata.creationTime, 'metadata.lastSignInTime': user.metadata.lastSignInTime }; data.forEach((k, v) { if (v != null) { var row = authInfo.addRow(); row.addCell() ..text = k ..classes.add('header'); row.addCell().text = "$v"; } }); print('User.toJson:'); print(const JsonEncoder.withIndent(' ').convert(user)); verifyEmailContainer.style.display = user.emailVerified ? "none" : "block"; } else { registerForm.style.display = "block"; registeredUser.style.display = "none"; authInfo.children.clear(); } } } index.html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Auth demo</title> <style type="text/css"> #register_form p { color: red; } #auth_info td { padding: 1px; } #auth_info td.header { font-weight: bold; text-align: right; } </style> <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> </head> <body> <div class="container"> <h1>Auth demo</h1> <form id="register_form" style="display: none"> <h2>Register</h2> <p></p> <label for="persistent_state">State persistence type</label> <select id="persistent_state"> <option value="local">LOCAL</option> <option value="session">SESSION</option> <option value="none">NONE</option> </select> <br> <input type="email" placeholder="Your E-mail" id="email"> <input type="password" placeholder="Your Password" id="password"> <input type="submit" value="Register me"> </form> <div id="registered_user" style="display: none"> <table id="auth_info"> </table> <div id="verify_email_container"> <h2>Send verify email</h2> <label for="verify_email_language">Email language</label> <select id="verify_email_language"> <option value="en">English</option> <option value="cs">Czech</option> <option value="de">German</option> <option value="fr">French</option> </select> <button id="verify_email">Verify email</button> </div> Log out </div> </div> <script src="https://www.gstatic.com/firebasejs/6.6.1/firebase-app.js"></script> <script src="https://www.gstatic.com/firebasejs/6.6.1/firebase-auth.js"></script> <script defer src="index.dart.js" type="application/javascript"></script> </body> </html> pubspec.yaml dependencies: firebase: ^5.0.0 http: '>=0.11.3 <0.13.0' js: ^0.6.0 This answer is a bit all over the place, between the two examples talked about hopefully it's useful.
Youtube-Js-Api / How can I include the request’s response in my actual webpage?
I'm working on an app with both Firebase (web app) and the youtube API. The idea is to let users share their playlists and interact. 1) First, the user must identify himself. This part works very well (firebase part) 2) Then, the user must accept some conditions (the scopes) from the youtube API. 3) The Youtube API return the result of the request. The issue is that youtube API recreate for every item of the array the HTML structure : HTML > HEAD > BODY. Plus, the response skips my header and display the all thing in a blank page instead of reacting like an include PHP (i know this is javascript but still.. ). I know the solution rest in the function executerequest but i can't figurate how to do it. The code of the request is in the middle of the body but for the purpose of my post, i did separate it. var GoogleAuth; var SCOPE = 'https://www.googleapis.com/auth/youtube.force-ssl'; function handleClientLoad() { // Load the API's client and auth2 modules. // Call the initClient function after the modules load. gapi.load('client:auth2', initClient); } function initClient() { // Retrieve the discovery document for version 3 of YouTube Data API. // In practice, your app can retrieve one or more discovery documents. var discoveryUrl = 'https://www.googleapis.com/discovery/v1/apis/youtube/v3/rest'; // Initialize the gapi.client object, which app uses to make API requests. // Get API key and client ID from API Console. // 'scope' field specifies space-delimited list of access scopes. gapi.client.init({ 'apiKey': '...', 'discoveryDocs': [discoveryUrl], 'clientId': '....', 'scope': SCOPE }).then(function () { GoogleAuth = gapi.auth2.getAuthInstance(); // Listen for sign-in state changes. GoogleAuth.isSignedIn.listen(updateSigninStatus); // Handle initial sign-in state. (Determine if user is already signed in.) var user = GoogleAuth.currentUser.get(); setSigninStatus(); // Call handleAuthClick function when user clicks on // "Sign In/Authorize" button. $('#sign-in-or-out-button').click(function() { handleAuthClick(); }); $('#revoke-access-button').click(function() { revokeAccess(); }); }); } function handleAuthClick() { if (GoogleAuth.isSignedIn.get()) { // User is authorized and has clicked 'Sign out' button. $('#sign-in-or-out-button').html('Sign out'); $('#revoke-access-button').css('display', 'inline-block'); GoogleAuth.signOut(); } else { // User is not signed in. Start Google auth flow. GoogleAuth.signIn(); } } function revokeAccess() { GoogleAuth.disconnect(); } function setSigninStatus(isSignedIn) { var user = GoogleAuth.currentUser.get(); var isAuthorized = user.hasGrantedScopes(SCOPE); if (isAuthorized) { $('#sign-in-or-out-button').html('Sign out'); $('#revoke-access-button').css('display', 'inline-block'); $('#auth-status').html('Connecté ' + ' Granted'); defineRequest(); console.log('connecté'); } else { $('#roomRed').html('display', 'block'); $('#sign-in-or-out-button').html('Sign In/Authorize'); $('#revoke-access-button').css('display', 'none'); $('#auth-status').html('Déconnecté' + ' Denied'); console.log('déconnecté'); } // This helper method displays a message on the page. } function updateSigninStatus(isSignedIn) { setSigninStatus(); } function createResource(properties) { var resource = {}; var normalizedProps = properties; for (var p in properties) { var value = properties[p]; if (p && p.substr(-2, 2) == '[]') { var adjustedName = p.replace('[]', ''); if (value) { normalizedProps[adjustedName] = value.split(','); } delete normalizedProps[p]; } } for (var p in normalizedProps) { // Leave properties that don't have values out of inserted resource. if (normalizedProps.hasOwnProperty(p) && normalizedProps[p]) { var propArray = p.split('.'); var ref = resource; for (var pa = 0; pa < propArray.length; pa++) { var key = propArray[pa]; if (pa == propArray.length - 1) { ref[key] = normalizedProps[p]; } else { ref = ref[key] = ref[key] || {}; } } }; } return resource; } function removeEmptyParams(params) { for (var p in params) { if (!params[p] || params[p] == 'undefined') { delete params[p]; } } return params; } function executeRequest(request) { request.execute(function(response) { console.log(response); for(var i = 0; i< response.items.length; i++){ console.log(response.items[i].player.embedHtml); document.write(response.items[i].player.embedHtml); } }); } function buildApiRequest(requestMethod, path, params, properties) { params = removeEmptyParams(params); var request; if (properties) { var resource = createResource(properties); request = gapi.client.request({ 'body': resource, 'method': requestMethod, 'path': path, 'params': params }); } else { request = gapi.client.request({ 'method': requestMethod, 'path': path, 'params': params }); } executeRequest(request); } /***** END BOILERPLATE CODE *****/ function defineRequest() { // See full sample for buildApiRequest() code, which is not // specific to a particular youtube or youtube method. buildApiRequest('GET', '/youtube/v3/playlists', { 'mine': 'true', 'maxResults': '25', 'part': 'snippet,contentDetails,player', 'onBehalfOfContentOwner': '', 'onBehalfOfContentOwnerChannel': '' }); /* buildApiRequest('GET', '/youtube/v3/playlistItems', { 'playlistId': "PLsvlo6Soc2pc2ZlereiehdPRhm0eKjSxI", 'maxResults': '25', 'part': 'snippet,contentDetails' }); */ } <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Mes vidéo </title> <style> </style> </head> <body> <button id="sign-in-or-out-button" style="margin-left: 25px">Sign In/Authorize</button> <button id="revoke-access-button" style="display: none; margin-left: 25px">Revoke access</button> <div id="auth-status" style="display: inline; padding-left: 25px"></div><hr> <div id="video-container"></div> <script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.3/jquery.min.js"></script> <script async defer src="https://apis.google.com/js/api.js" onload="this.onload=function(){};handleClientLoad()" onreadystatechange="if (this.readyState === 'complete') this.onload()"> </script> </body> </html> Thank you by advance
<a onclick"sth" .. cause IE11 warn and prompt for adding about:blank as Trusted Site
I have jqgrid on asp .NET web page. When page is loaded and grid loads its data, after it receives json, before data is shown on grid, I get error message from IE: Internet Explorer 11.0.9600.17728 It wants me to add about:blank to Trusted Sites. When i click "close" rows appear. Row`s html is as follows: <a onclick=" OpenWindow('/13_1/Workflow.TasksWebPresUnit/TaskDetails/Index/? taskDn=WFL///70000/.321360&isModal=True& returnUrl=/13_1/Workflow.TasksWebPresUnit& userId=5& componentsToSet=test*testc', 'Details', '90%', '0', false, '¡',true);" href="#">P01.07 Verification </a> No matter what I put in onclick, can be "blahblah", same with href, can be "foobar", <a onclick=" abcd" href="xyz">P01.07 Verification </a> i get that error window. But, when I construct row's html that it not contain "onclick=sth", then i do not get this error. Is there any way to stop IE from warning user or maybe I`m doing somethiong wrong? Any help / clarification appreciated. p.s. If i add about:blank to trusted sites, the problem is gone, but i don`t understand fully if this is secure solution. Update 1: script function OpenWindow(url, title, width, height, checkValue, multiSepar, showCloseButton) { var progressIndImg = GLOBAL_CSS_PATH + '/Technology/modal/images/waiting.gif'; if (checkValue) { var queryVals = parseQueryString(url); var relatedComp = queryVals['componentToFill']; var relatedVal = document.getElementById(relatedComp); if (relatedVal.value != null && relatedVal.value != '') { var win = dhtmlmodal.open('ModalBox', 'iframe', url, title, 'width=' + width + ', height=' + height + ',center=1,resize=1,scrolling=1', '', progressIndImg, showCloseButton); win.onclose = function () { return true; } } } else { var win = dhtmlmodal.open('ModalBox', 'iframe', url, title, 'width=' + width + ', height=' + height + ',center=1,resize=1,scrolling=1', '', progressIndImg, showCloseButton); win.onclose = function () { return true; } } // } else { // setInterval(checkForMessages, 200); // } // var ie7 = (navigator.appVersion.indexOf('MSIE 7.') == -1) ? false : true; // if (ie7 != true) { var onmessage = function (e) { try { var objects = JSON.parse(e.data); if (window.addEventListener) { window.removeEventListener('message', onmessage, false); } else if (window.attachEvent) { window.detachEvent('onmessage', onmessage); } if (objects['methodName'] != null) window[objects['methodName']](objects); else FillData(objects, true, multiSepar); var close = objects['close']; if (close == 'true') { win.hide(); win.close(); } } catch (err) { if (console) console.error(err); } }; if (window.addEventListener) { window.addEventListener('message', onmessage, false); } else if (window.attachEvent) { window.attachEvent('onmessage', onmessage); }
Azure Media Player Silverlight fallback not working
I have used the azure media player with my project, it will play multiple adaptive bit rate streamed videos in an asp.net page, the best part is, it is working superb in html5 and flash but it will get stuck at spinner image in silverlight fallback. Below is the code I have used. I have also tried to get errors but it is not hitting the event listener code added for errors, but the play and pause events are working fine where flash and html5 is used, but silverlight fallback is not working at all. <link href="https://amp.azure.net/libs/amp/1.3.0/skins/amp-default/azuremediaplayer.min.css" rel="stylesheet"> <script src="https://amp.azure.net/libs/amp/1.3.0/azuremediaplayer.min.js"></script> <div class="marginBlock"> <h3> <asp:Label ID="lblTitle" runat="server"><%=Title.ToString()%></asp:Label> </h3> <video id="<%=mediaPlayerID %>" class="azuremediaplayer amp-default-skin amp-big-play-centered"> <p class="amp-no-js"> To view this video please enable JavaScript, and consider upgrading to a web browser that supports HTML 5 video. </p> </video> </div> <p> <asp:Label ID="lblDescription" runat="server"><%=Description.ToString()%> </asp:Label> </p> <script> $(document).ready(function () { var playOptions = { "nativeControlsForTouch": false, techOrder: ['azureHtml5JS', 'flashSS', 'silverlightSS', 'html5'], autoplay: false, controls: true, width: '100%', height: '400', logo: { enabled: false }, poster: "<%=ImageSelector%>" } var azurePlayer = amp('<%=mediaPlayerID%>', playOptions); azurePlayer.src([{ src: "<%=VideoURL%>", type: 'application/vnd.ms-sstr+xml' }]); azurePlayer.addEventListener("error", function () { var errorDetails = azurePlayer.error(); var code = errorDetails.code; var message = errorDetails.message; alert(errorDetails + ' ' + code + " " + message); if (azurePlayer.error().code & amp.errorCode.abortedErrStart) { console.log("abortedErrStart"); } else if (azurePlayer.error().code & amp.errorCode.networkErrStart) { // MEDIA_ERR_NETWORK errors console.log("networkErrStart"); } else if (azurePlayer.error().code & amp.errorCode.decodeErrStart) { // MEDIA_ERR_DECODE errors console.log("decodeErrStart"); } else if (azurePlayer.error().code & amp.errorCode.srcErrStart) { // MEDIA_ERR_SRC_NOT_SUPPORTED errors console.log("srcErrStart"); } else if (azurePlayer.error().code & amp.errorCode.encryptErrStart) { // MEDIA_ERR_ENCRYPTED errors console.log("encryptErrStart"); } else if (azurePlayer.error().code & amp.errorCode.srcPlayerMismatchStart) { // SRC_PLAYER_MISMATCH errors console.log("srcPlayerMismatchStart"); } else { // unknown errors console.log("unknown"); } }); azurePlayer.addEventListener('play', function () { console.log('play'); }); azurePlayer.addEventListener('pause', function () { console.log('pause'); }); }); Updated Noticed I am getting following error for IE < 11. Also I disabled the flash in firefox and removed the silverlight from techOrder, then it should hit the error event listener, it is not hitting. This is also important for me to handle the analytics for error. Play and Pause event listener are working fine. Update 8/28/2015: Fixed the JS error, it is because of multiple call to cdn of azure mentioned in link above, moved the code in master page and load it only once, browser like chrome handles duplicity of code easily but not IE. After all the research I am lost why it is not working. So added the following JS that will check for Silverlight and Flash and gracefully handle the error and update our analytics as well. function getBrowserInformation() { var ua = navigator.userAgent, tem, M = ua.match(/(opera|chrome|safari|firefox|msie|trident(?=\/))\/?\s*(\d+)/i) || []; if (/trident/i.test(M[1])) { tem = /\brv[ :]+(\d+)/g.exec(ua) || []; return { name: 'IE ', version: (tem[1] || '') }; } if (M[1] === 'Chrome') { tem = ua.match(/\bOPR\/(\d+)/) if (tem != null) { return { name: 'Opera', version: tem[1] }; } } M = M[2] ? [M[1], M[2]] : [navigator.appName, navigator.appVersion, '-?']; if ((tem = ua.match(/version\/(\d+)/i)) != null) { M.splice(1, 1, tem[1]); } return { name: M[0], version: M[1] };}; function checkForAzureErrors() { function isSilverlightInstalled() { var isSilverlightInstalled = false; try { //check on IE try { var slControl = new ActiveXObject('AgControl.AgControl'); isSilverlightInstalled = true; } catch (e) { //either not installed or not IE. Check Firefox/Safari if (navigator.plugins["Silverlight Plug-In"]) { isSilverlightInstalled = true; } } } catch (e) { console.log(e); } return isSilverlightInstalled; } function isFlashInstalled() { try { return Boolean(new ActiveXObject('ShockwaveFlash.ShockwaveFlash')); } catch (exception) { return ('undefined' != typeof navigator.mimeTypes['application/x-shockwave-flash']); } } function addErrorMessage() { $($("#mediaPlayer.marginBlock").find('h3')).text('Media is not supported on this browser or device.'); $($("#mediaPlayer.marginBlock").find('video')).css('display', 'none'); $($("#mediaPlayer.marginBlock").find('p')).css('display', 'none'); $('.azuremediaplayer').css('display', 'none'); ga("send", "event", "Videos", "error", getBrowserInformation().name + getBrowserInformation().version + ": is silverlight Installed " + isSilverlightInstalled() + " and is Flash Installed " + isFlashInstalled()); } function checkBrowser() { if ((getBrowserInformation().name === 'MSIE' || getBrowserInformation().name === 'IE')) { if (getBrowserInformation().version < 11) { addErrorMessage(); } } else if (getBrowserInformation().name === 'Firefox') { addErrorMessage(); } } if ((getBrowserInformation().name === 'MSIE' || getBrowserInformation().name === 'IE')) { if (getBrowserInformation().version < 9) { addErrorMessage() } } for (var key in amp.players) { if (amp.players.hasOwnProperty(key)) { if (isSilverlightInstalled()) { if (!amp.players[key].isReady_) { checkBrowser(); } } else if (!isFlashInstalled()) { checkBrowser(); } } }} This function is called after 5 second of page load in document.ready function, giving it enough time to load and make the isReady_boolean true. SetTimeout(function () { checkForAzureErrors(); }, 5000); Still I am waiting for some angel to resolve this issue. Updates: Partially Fixed Need to add the xap refrences like older version, it will play silverlight but there is a catch, it will only work if you have one video per page. <script> amp.options.flashSS.swf = "http://amp.azure.net/libs/amp/1.3.0/techs/StrobeMediaPlayback.2.0.swf" amp.options.flashSS.plugin = "http://amp.azure.net/libs/amp/1.3.0/techs/MSAdaptiveStreamingPlugin-osmf2.0.swf" amp.options.silverlightSS.xap = "http://amp.azure.net/libs/amp/1.3.0/techs/SmoothStreamingPlayer.xap" </script> Fixed As per comments from Amit Rajput #Parshii Currently, as per the documentation, Azure Media Player doesn't support multi-instance playback. While it may work on some techs, it is not a tested scenario at this time. Please feel free to add it to the UserVoice forum (http://aka.ms/ampuservoice). As per my testing it is working in html5 and flash but not Silverlight, for silverlight support we can try using iframes as per comments from rnrneverdies Single Instance Media Player is working in all techs.
#Parshii Currently, as per the documentation, Azure Media Player doesn't support multi-instance playback. While it may work on some techs, it is not a tested scenario at this time. Please feel free to add it to the UserVoice forum (http://aka.ms/ampuservoice).
This may be not a complete answer, but could help you. I made the following plugin for the Azure Media Player 1.3.0 which logs all the activity performed by the user and also the errors. Set up it as: var mylogFunction = function(data) { console.log(data); }; var options = { techOrder: ["azureHtml5JS", "flashSS", "silverlightSS", "html5"], nativeControlsForTouch: false, loop: false, logo: { enabled: false }, heuristicProfile: "Quick Start", //"High Quality", // could be "Quick Start" customPlayerSettings: { customHeuristicSettings: { preRollInSec: 4, windowSizeHeuristics: true } }, plugins: { DebugLog: { logFunction: mylogFunction } } }; var amPlayer = amp("yourvideotagid", options); Plugin Code: var amp; (function (amp) { amp.plugin('DebugLog', DebugLog); function DebugLog(options) { var player = this; var log = function (data) { console.log("Azure Media Player Log", data); } if (options) { if (options['logFunction']) { log = options['logFunction']; } } init(); function init() { player.ready(handleReady); player.addEventListener(amp.eventName.error, handleError); } function handleReady() { player.addEventListener(amp.eventName.loadedmetadata, handleLoadedMetaData); var data = { ampVersion: "1.3.0", appName: options['appName'], userAgent: navigator.userAgent, options: { autoplay: player.options().autoplay, heuristicProfile: player.options().heuristicProfile, techOrder: JSON.stringify(player.options().techOrder) } }; logData("InstanceCreated", 1, data); } function handleError() { var err = player.error(); var data = { sessionId: player.currentSrc(), currentTime: player.currentTime(), code: "0x" + err.code.toString(16), message: err.message }; logData("Error", 0, data); } function handleLoadedMetaData() { player.addEventListener(amp.eventName.playbackbitratechanged, handlePlaybackBitrateChanged); player.addEventListener(amp.eventName.playing, handlePlaying); player.addEventListener(amp.eventName.seeking, handleSeeking); player.addEventListener(amp.eventName.pause, handlePaused); player.addEventListener(amp.eventName.waiting, handleWaiting); player.addEventListener(amp.eventName.ended, handleEnded); if (player.audioBufferData()) { player.audioBufferData().addEventListener(amp.bufferDataEventName.downloadfailed, function () { var data = { sessionId: player.currentSrc(), currentTime: player.currentTime(), bufferLevel: player.audioBufferData().bufferLevel, url: player.audioBufferData().downloadFailed.mediaDownload.url, code: "0x" + player.audioBufferData().downloadFailed.code.toString(16), message: player.audioBufferData().downloadFailed }; logData("DownloadFailed", 0, data); }); } if (player.videoBufferData()) { player.videoBufferData().addEventListener(amp.bufferDataEventName.downloadfailed, function () { var data = { sessionId: player.currentSrc(), currentTime: player.currentTime(), bufferLevel: player.videoBufferData().bufferLevel, url: player.videoBufferData().downloadFailed.mediaDownload.url, code: "0x" + player.videoBufferData().downloadFailed.code.toString(16), message: player.videoBufferData().downloadFailed }; logData("DownloadFailed", 0, data); }); } var data = { sessionId: player.currentSrc(), isLive: player.isLive(), duration: player.duration(), tech: player.currentTechName(), protection: ((player.currentProtectionInfo() && player.currentProtectionInfo()[0]) ? player.currentProtectionInfo()[0].type : "clear") }; logData("PresentationInfo", 1, data); } function handlePlaybackBitrateChanged(event) { logData("BitrateChanged", 1, eventData(event)); } function handleWaiting(event) { logData("Waiting", 0, eventData(event)); } function handlePlaying(event) { logData("Playing", 1, eventData(event)); } function handleSeeking(event) { logData("Seeking", 1, eventData(event)); } function handlePaused(event) { logData("Paused", 1, eventData(event)); } function handleEnded(event) { logData("Ended", 1, eventData(event)); } function logData(eventId, level, data) { var eventLog = { eventId: eventId, level: level, data: data }; log(eventLog); } function eventData(event) { return { sessionId: player.currentSrc(), currentTime: player.currentTime(), isLive: player.isLive(), event: event.type, presentationTimeInSec: event.presentationTimeInSec, message: event.message ? event.message : "" }; } } })(amp || (amp = {}));