- Post History
- Subscribe to RSS Feed
- Mark as New
- Mark as Read
- Bookmark
- Subscribe
- Printer Friendly Page
- Report Inappropriate Content
on 09-07-2018 11:44 AM
Greetings,
I would like to share with you the ability to add "Impersonate User" to the Service Portal without having to clone the out-of-box Header Menu widget.
I would like to first thank Service Catalyst for initially creating a widget for Service Portal impersonation. I have taken what they built in a widget for impersonating users and created another widget based on the out-of-box Knowledge Notification Entry widget to inject "Impersonate User" in the the menu drop down and then call the Impersonate User widget when clicked.
We will be creating two widgets to make this work.
Widget #1
Name: Impersonate User
ID: impersonate_user
HTML Template:
<div class="container-fluid" ng-init="getRecentImpersonation()">
<div class="row">
<div class="btn-group btn-group-sm btn-group-justified" role="group">
<label class="btn btn-info btn-sm" ng-model="c.selectedAction" uib-btn-radio="'sp'">Service Portal</label>
<label class="btn btn-primary btn-sm" ng-model="c.selectedAction" uib-btn-radio="'classic'">Classic UI</label>
</div>
</div>
<div class="row top-buffer" ng-if="data.isImpersonating">
<div class="btn-group btn-group-sm btn-group-justified" role="group">
<label ng-click="impersonate(c.data.realUser)" class="btn btn-danger btn-lg btn-block">
<i class="fa fa-stop m-r-sm"></i>
STOP Impersonating</label>
</div>
</div>
<div class="row top-buffer">
<div class="record-picker">
<sn-record-picker field="c.data.toImpersonate" table="'sys_user'" display-field="'name'" on-change="impersonate(c.data.toImpersonate.value)"
display-fields="'user_name'" search-fields="'name'" page-size="100" placeholder="Search for user" options="{cache: true, allowClear: false}"
default-query="'active=true^locked_out=false^web_service_access_only=false^ORweb_service_access_onlyISEMPTY'"
value-field="'sys_id'">
</sn-record-picker>
</div>
</div>
<div class="row top-buffer">
<div class="panel panel-default">
<div class="panel-heading">
<h5 class="panel-title">
${Recent Impersonations}
</h5>
</div>
<a ng-bind="rec.user_display_value" ng-repeat="rec in recentImpersonations" href="javascript:void(0)" class="list-group-item ng-binding ng-scope"
ng-click="impersonate(rec.user_sys_id)"></a>
</div>
</div>
</div>
CSS - SCSS:
.top-buffer {
margin-top:20px;
}
Client Script:
function($scope, $http, $window) {
/* widget controller */
var c = this;
c.data.toImpersonate = {};
c.selectedAction = {};
$scope.impersonate = function(userName){
//If we don't have a user we can't impersonate
if(!userName){
return;
}
//Call to the impersonation api with username/sys_id
$http.post("/api/now/ui/impersonate/" + userName, {}).success(function(){
$scope.showError = false;
if(c.selectedAction === "classic"){
window.location = "/";
}else if(c.selectedAction === "sp" || c.selectedAction === null){
window.location.reload();
}else{
window.location.reload();
}
}).error(function(response){
if(response.error){
$scope.showError = true;
$scope.error = response.error;
console.warn("Impersonate Failed! with error:", response.error);
}
})
};
//call to the impersonation api Get recent impersonations
$scope.getRecentImpersonation = function(){
$http.get("/api/now/ui/impersonate/recent", {}).success(function(response){
$scope.showError = false;
$scope.recentImpersonations = response.result;
}).error(function(response){
if(response.error){
$scope.showError = true;
$scope.error = response.error;
console.warn("Get Recent Impersonations Failed! with error:" + response.error);
}
})
};
}
Server Script:
(function() {
/* populate the 'data' object */
/* e.g., data.table = $sp.getValue('table'); */
data.isImpersonating = new GlideImpersonate().isImpersonating();
data.realUser = gs.getImpersonatingUserName();
})();
Widget #2
Name: Impersonate User Entry
ID: impersonate_user_entry
Server Script:
(function() {
/* populate the 'data' object */
/* e.g., data.table = $sp.getValue('table'); */
data.canImpersonate = new GlideImpersonate().canImpersonate(gs.getUserID());
data.isImpersonating = new GlideImpersonate().isImpersonating();
data.realUser = gs.getImpersonatingUserName();
if (data.canImpersonate || data.isImpersonating) {
data.show_menu_entry = true;
}
else {
data.show_menu_entry = false;
}
})();
Link Function:
function(scope) {
var $uibModal = $injector.get('$uibModal');
// Add entry to navbar
if(scope.data.show_menu_entry && !$('#impersonation')[0]) {
$('#sp-nav-bar ul li.hidden-xs.dropdown ul.dropdown-menu li:first').after('<li style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
$('#sp-nav-bar ul li.visible-xs-block:first').after('<li class="visible-xs-block" style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
}
$("#impersonation").click(function() {
$uibModal.open({
templateUrl: 'impersonate-widget',
scope: scope
})
});
}
Next you will want to open the Impersonate User Entry widget in the platform. In the related lists go to Angular ng-templates and click New.
ID: impersonate-widget
Template:
<div class="panel panel-default">
<h3 class="padder-md">
Impersonate User
</h3>
<div class="modal-header">
<widget id="impersonate_user"></widget>
</div>
</div>
Final Step
Open your service portal home page in the Service Portal Designer and add the Impersonate User Entry widget to the page. Now impersonate User will show in the menu drop down for users with the impersonate role.
Add icons to menu items
If you want to add icons like you see in the screen shot above, follow these steps:
- Open the Stock theme in the platform.
- Add the following to CSS variables:
#sp-nav-bar > ul:nth-child(2) > li.hidden-xs.dropdown.ng-scope.open > ul > li.ng-scope > a:before {
content: "\f08b"; /* Sign Out */
font-family: FontAwesome;
padding-right: 5px;
}
#sp-nav-bar > ul:nth-child(2) > li.hidden-xs.dropdown.ng-scope.open > ul > li:nth-child(1) > a:before {
content: "\f007"; /* Profile */
font-family: FontAwesome;
padding-right: 8px;
}
#impersonation:before {
content: "\f06e"; /* Impersonate User */
font-family: FontAwesome;
padding-right: 6px;
margin-left: -2px;
}
- 11,511 Views
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Exactly followed the steps mentioned, but unable to implement this in kingston instance. But working as expected in Jakarta instance. Please suggest the solution for the above problem. It is urgent. Thanks in advance.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
I have tested this in Kingston and it is working on a developer instance for me. This also works in London. Would you be able to provide me with the issue you are experiencing on a Kingston instance?
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Following the above steps did not make any change in the instance.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Thanks for sharing this - worked as expected in Jakarta and London and is super handy. Do you have any suggestions on how I may be able to embed this in the (cloned) header widget itself? It'd be ideal to not have to add the widget to every portal page. Any help / advice is appreciated!
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
This is fantastic! Great work.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
I am pretty new to ServiceNow portal. Could someone please provide me with instructions on how to perform the last step of adding the Impersonate User Entry widget to the page using the Service Portal Designer?
TIA.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
In the platform view in the left navigation go to Service Portal > Service Portal Configuration. This will open a new window, then click on Designer. You can then select the portal page you want and can then drag and drop the widget to the page.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
But which page should I choose to drag the widget in so that it shows up under the name, profie, logout? Should it be the login page?
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
We put it on our home page, not the login page. Just know that you would have to go to your home page to use it. Otherwise you can add it to all pages should you choose to do so.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Thanks it worked. However, the icons dont show up now.
Added the script to stock as instructed.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
I had followed the exact steps that were stated above. It's working good and my version is Madrid.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
I followed above step and worked well.. But I have noticed that, when we are in home page then only we can get the "Impersonate User" button below the profile. Incase if I opned any other page and try to see under profile and its not visible... Can this applied across entire service portal page?
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
This isn't true for me, atleast on my personal dev. instance. I can see the Impersonate option on every screen in the portal. See some screenshots below.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
It will show but it will disappear if you refresh the screen..
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
You would need to add the widget to each page you want to be able to use it on.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Hi,
For me stop impersonating is not working on London release.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Should work. I have it working in Madrid.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
thanks for the article and i followed the steps to create two widgets and after that from FINAL STEP is is a little bit confusing for me to follow .... can you please elaborate it.
thanks in advance
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
You just need to open your home page of your portal in page designer and add the widget for the impersonate entry to the page.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
This works great! Thanks for sharing!
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
You can accomplish this by adding the widget to the footer of your theme. Here you can see I have added the "Impersonate User Entry", and I have also added a feedback button widget so that both are visible on all pages, no matter what your entry point to the portal is:
You will need to update the Link Function in the Impersonate User Entry widget, though, because the header menu DOM element does not yet exist when the Link Function first runs. The code below will check that the header menu exists before trying to add the menu entry and set up the link action:
function(scope) {
var $uibModal = $injector.get('$uibModal');
var checkExist = setInterval(function() {
if ($('#sp-nav-bar ul li.hidden-xs.dropdown ul.dropdown-menu li:first').length && $('#sp-nav-bar ul li.visible-xs-block:first').length) {
// Add entry to navbar
if(scope.data.show_menu_entry && !$('#impersonation')[0]) {
$('#sp-nav-bar ul li.hidden-xs.dropdown ul.dropdown-menu li:first').after('<li style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
$('#sp-nav-bar ul li.visible-xs-block:first').after('<li class="visible-xs-block" style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
}
$("#impersonation").click(function() {
$uibModal.open({
templateUrl: 'impersonate-widget',
scope: scope
})
});
clearInterval(checkExist);
}
}, 100); // check every 100ms
}
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
This is great. Thanks for sharing.

- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Hi all,
It seems that this customization doesn't work on Orlando.
Can you confirm?
If no, do you know what we have to modify?
Thanks!
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
it is working as expected in Orlando Release.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Thank you for sharing!!
I get an error message.
"GlideImpersonate is not allowed in scoped applications"
Am I doing something wrong?

- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
It is not working for me in Orlando either. All I get is two buttons that say Service Porta and Classic UI but click on them does nothing.
I have an idea to make at least end impersonate available in the portal since you cannot get out of it without logging out if the user you impersonated only has access to the portal.

- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Never mind figured it out. seems you have to put it in the top most section of your home page.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
I have implemented this solution in our portal - running on Paris.
Everything seemed to be working but just discovered a problem.
The menu item visibility does not appear to be respecting roles that permit impersonation. So if a logged in general user selects that item in the user pulldown, then the system hangs and the impersonation popup does not appear (that seems to be respecting the lack of a role) and the portal hangs.
Seems like the menu item visability needs to bel driven by role permission.
How can that be implemented.
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Thank you for sharing!!
I get an error message.
"GlideImpersonate is not allowed in scoped applications"
Am I doing something wrong?
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
Has anyone experienced issue with Automated Testing after implementing this solution from @mhedtke24 ?
I'm having difficulty with both Cloud & Client ATF with the testing hanging around step #26 (confirming the Service Catalog Items has been submitted)
- Mark as Read
- Mark as New
- Bookmark
- Permalink
- Report Inappropriate Content
@kshaw, i change function in server script on Impersonate User Entry widget:
(function() {
/* populate the 'data' object */
/* e.g., data.table = $sp.getValue('table'); */
data.canImpersonate = new GlideImpersonate().canImpersonate(gs.getUserID());
data.isImpersonating = new GlideImpersonate().isImpersonating();
data.realUser = gs.getImpersonatingUserName();
// Check if the user has the impersonator role
var hasImpersonatorRole = gs.hasRole('impersonator');
if ((data.canImpersonate && hasImpersonatorRole) || data.isImpersonating) {
data.show_menu_entry = true;
}
else {
data.show_menu_entry = false;
}
})();