# The LearningOnline Network with CAPA
# Routines to control the menu
#
# $Id: lonmenu.pm,v 1.560 2025/01/07 22:21:56 raeburn Exp $
#
# Copyright Michigan State University Board of Trustees
#
# This file is part of the LearningOnline Network with CAPA (LON-CAPA).
#
# LON-CAPA is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# LON-CAPA is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with LON-CAPA; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
# /home/httpd/html/adm/gpl.txt
#
# http://www.lon-capa.org/
#
#
=head1 NAME
Apache::lonmenu
=head1 SYNOPSIS
Loads contents of /home/httpd/lonTabs/mydesk.tab,
used to generate inline menu, and Main Menu page.
This is part of the LearningOnline Network with CAPA project
described at http://www.lon-capa.org.
=head1 GLOBAL VARIABLES
=over
=item @desklines
Each element of this array contains a line of mydesk.tab that doesn't start with
cat, prim or scnd.
It gets filled in the BEGIN block of this module.
=item %category_names
The keys of this hash are the abbreviations used in mydesk.tab in those lines that
start with cat, the values are strings representing titles.
It gets filled in the BEGIN block of this module.
=item %category_members
TODO
=item %category_positions
The keys of this hash are the abbreviations used in mydesk.tab in those lines that
start with cat, its values are position vectors (column, row).
It gets filled in the BEGIN block of this module.
=item $readdesk
Indicates that mydesk.tab has been read.
It is set to 'done' in the BEGIN block of this module.
=item @primary_menu
The elements of this array reference arrays that are made up of the components
of those lines of mydesk.tab that start with prim:.
It is used by primary_menu() to generate the corresponding menu.
It gets filled in the BEGIN block of this module.
=item %primary_sub_menu
The keys of this hash reference are the names of items in the primary_menu array
which have sub-menus. For each key, the corresponding value is a reference to
an array containing components extracted from lines in mydesk.tab which begin
with primsub:.
This hash, which is used by primary_menu to generate sub-menus, is populated in
the BEGIN block.
=item @secondary_menu
The elements of this array reference arrays that are made up of the components
of those lines of mydesk.tab that start with scnd.
It is used by secondary_menu() to generate the corresponding menu.
It gets filled in the BEGIN block of this module.
=back
=head1 SUBROUTINES
=over
=item prep_menuitems(\@menuitem,$target,$listclass,$linkattr)
This routine wraps a menuitem in proper HTML. It is used by primary_menu() and
secondary_menu().
=item primary_menu()
This routine evaluates @primary_menu and returns a two item array,
with the array elements containing XHTML for the left and right sides of
the menu that contains the following links: About, Message, Roles, Help, Logout
@primary_menu is filled within the BEGIN block of this module with
entries from mydesk.tab
=item secondary_menu()
Same as primary_menu() but operates on @secondary_menu.
=item create_submenu()
Creates XHTML for unordered list of sub-menu items which belong to a
particular top-level menu item. Uses hover pseudo class in css to display
dropdown list when mouse hovers over top-level item. Support for IE6
(no hover psuedo class) via LC_hoverable class for <li> tag for top-
level item, which employs jQuery to handle behavior on mouseover.
Inputs: 6 - (a) link and (b) target for anchor href in top level item,
(c) title for text wrapped by anchor tag in top level item,
(d) reference to array of arrays of sub-menu items,
(e) boolean to indicate whether to call &mt() to translate
name of menu item,
(f) optional class for <li> element in primary menu, for which
sub menu is being generated.
The underlying datastructure used in (d) contains data from mydesk.tab.
It consists of an array which has an array for each item appearing in
the menu (e.g. [["link", "title", "condition"]] for a single-item menu).
create_submenu() supports also the creation of XHTML for nested dropdown
menus represented by unordered lists. This is done by replacing the
scalar used for the link with an arrayreference containing the menuitems
for the nested menu. This can be done recursively so that the next menu
may also contain nested submenus.
Example:
[ # begin of datastructure
["/home/", "Home", "condition1"], # 1st item of the 1st layer menu
[ # 2nd item of the 1st layer menu
[ # anon. array for nested menu
["/path1", "Path1", undef], # 1st item of the 2nd layer menu
["/path2", "Path2", undef], # 2nd item of the 2nd layer menu
[ # 3rd item of the 2nd layer menu
[[...], [...], ..., [...]], # containing another menu layer
"Sub-Sub-Menu", # title for this container
undef
]
], # end of array/nested menu
"Sub-Menu", # title for the container item
undef
] # end of 2nd item of the 1st layer menu
]
=item innerregister()
This gets called in order to register a URL in the body of the document
=item clear()
=item switch()
Switch a button or create a link
Switch acts on the javascript that is executed when a button is clicked.
The javascript is usually similar to "go('/adm/roles')" or "cstrgo(..)".
=item secondlevel()
=item openmenu()
=item inlinemenu()
=item rawconfig()
=item utilityfunctions()
Output from this routine is a number of javascript functions called by
items in the inline menu, and in some cases items in the Main Menu page.
=item serverform()
=item constspaceform()
=item get_nav_status()
=item hidden_button_check()
=item roles_selector()
=item jump_to_role()
=back
=cut
package Apache::lonmenu;
use strict;
use Apache::lonnet;
use Apache::lonhtmlcommon();
use Apache::loncommon();
use Apache::lonenc();
use Apache::lonlocal;
use Apache::lonmsg();
use LONCAPA qw(:DEFAULT :match);
use HTML::Entities();
use Apache::lonwishlist();
use vars qw(@desklines %category_names %category_members %category_positions
$readdesk @primary_menu %primary_submenu @secondary_menu %secondary_submenu);
my @inlineremote;
sub prep_menuitem {
my ($menuitem,$target,$listclass,$linkattr) = @_;
return '' unless(ref($menuitem) eq 'ARRAY');
my ($link,$targetattr);
if ($$menuitem[1]) { # graphical Link
$link = "<img class=\"LC_noBorder\""
. " src=\"" . &Apache::loncommon::lonhttpdurl($$menuitem[1]) . "\""
. " alt=\"" . &mt($$menuitem[2]) . "\" />";
} else { # textual Link
$link = &mt($$menuitem[3]);
}
if ($target ne '') {
$targetattr = ' target="'.$target.'"';
}
return ($listclass?'<li class="'.$listclass.'">':'<li>').'<a'
# highlighting for new messages
. ( $$menuitem[4] eq 'newmsg' ? ' class="LC_new_message"' : '')
. qq| href="$$menuitem[0]"$targetattr $linkattr>$link</a></li>|;
}
# primary_menu() evaluates @primary_menu and returns a two item array,
# with the array elements containing XHTML for the left and right sides of
# the menu that contains the following links:
# Personal, About, Message, Roles, Help, Logout
# @primary_menu is filled within the BEGIN block of this module with
# entries from mydesk.tab
sub primary_menu {
my ($crstype,$ltimenu,$menucoll,$menuref,$links_disabled,$links_target,$collapsible) = @_;
my (%menu,%ltiexc,%menuopts);
# each element of @primary contains following array:
# (link url, icon path, alt text, link text, condition, position)
my $public;
if ((($env{'user.name'} eq 'public') && ($env{'user.domain'} eq 'public'))
|| (($env{'user.name'} eq '') && ($env{'user.domain'} eq ''))) {
$public = 1;
}
my $rolecount;
if (($crstype eq 'Placement') && (!$env{'request.role.adv'})) {
my $update=$env{'user.update.time'};
if (!$update) {
$update = $env{'user.login.time'};
}
my %roles_in_env;
$rolecount = &Apache::lonroles::roles_from_env(\%roles_in_env,$update);
}
my $lti;
if ($env{'request.lti.login'}) {
$lti = 1;
if (ref($ltimenu) eq 'HASH') {
foreach my $item ('fullname','logout') {
unless ($ltimenu->{$item}) {
$ltiexc{$item} = 1;
}
}
}
}
my ($listclass,$linkattr,$target);
if ($links_disabled) {
$listclass = 'LCisDisabled';
$linkattr = 'aria-disabled="true"';
}
if ($links_target ne '') {
$target = $links_target;
} else {
my ($ltitarget,$deeplinktarget);
if ($env{'request.lti.login'}) {
$ltitarget = $env{'request.lti.target'};
}
if ($env{'request.deeplink.login'}) {
$deeplinktarget = $env{'request.deeplink.target'};
}
if (($ltitarget eq 'iframe') || ($deeplinktarget eq '_self')) {
$target = '_self';
} else {
$target = '_top';
}
}
if (($menucoll) && (ref($menuref) eq 'HASH')) {
%menuopts = %{$menuref};
}
foreach my $menuitem (@primary_menu) {
# evaluate conditions
next if ref($menuitem) ne 'ARRAY'; #
next if $$menuitem[4] eq 'nonewmsg' # show links depending on
&& &Apache::lonmsg::mynewmail(); # whether a new msg
next if $$menuitem[4] eq 'newmsg' # arrived or not
&& !&Apache::lonmsg::mynewmail(); #
next if $$menuitem[4] !~ /public/ ##we've a public user,
&& $public; ##who should not see all
##links
next if $$menuitem[4] eq 'onlypublic'# hide links which are
&& !$public; # only visible to public
# users
next if $$menuitem[4] eq 'roles' ##show links depending on
&& (&Apache::loncommon::show_course() ##term 'Courses' or
|| $lti); ##'Roles' wanted
next if $$menuitem[4] eq 'courses' ##and not LTI access
&& (!&Apache::loncommon::show_course()
|| $lti);
next if $$menuitem[4] eq 'notlti'
&& $lti;
next if $$menuitem[4] eq 'ltiexc'
&& exists($ltiexc{lc($menuitem->[3])});
my $title = $menuitem->[3];
if (($crstype eq 'Placement') && (!$env{'request.role.adv'})) {
if ($menuitem->[4] eq 'courses') {
next unless ($rolecount>1);
} else {
next unless (($title eq 'Personal') || ($title eq 'Logout'));
}
}
my $position = $menuitem->[5];
if ($position eq '') {
$position = 'right';
}
if ($env{'request.course.id'} && $menucoll) {
if (($menuitem->[6]) && (!$menuopts{$menuitem->[6]})) {
if ($menuitem->[6] eq 'pers') {
if ($menuopts{'name'} && !$ltiexc{'fullname'} &&
$env{'user.name'} && $env{'user.domain'}) {
$menu{$position} .= '<li><a href="#">'.
&Apache::loncommon::plainname($env{'user.name'},
$env{'user.domain'}).'</a></li>';
next;
} else {
next;
}
} else {
next;
}
}
}
if (defined($primary_submenu{$title})) {
my $link;
if ($menuitem->[0] ne '') {
$link = $menuitem->[0];
} else {
$link = '#';
}
my @primsub;
if (ref($primary_submenu{$title}) eq 'ARRAY') {
foreach my $item (@{$primary_submenu{$title}}) {
next if (($crstype eq 'Placement') && (!$env{'request.role.adv'}));
next if (($item->[2] eq 'wishlist') && (!$env{'user.adv'}));
next if ((($item->[2] eq 'portfolio') ||
($item->[2] eq 'blog')) &&
(!&Apache::lonnet::usertools_access('','',$item->[2],
undef,'tools')));
if (($item->[2] eq 'browsepub') && ($item->[0] eq '/res/')) {
if ($env{'request.role'} =~ /^au\./) {
$item->[0] .= $env{'request.role.domain'}.'/?launch=1';
} elsif ($env{'request.role'} =~ m{^ca\./($match_domain)/($match_username)$}) {
$item->[0] .= $1.'/'.$2.'/?launch=1';
} elsif (&Apache::lonnet::allowed('bre',$env{'user.domain'})) {
$item->[0] .= $env{'user.domain'}.'/?launch=1';
} elsif (&Apache::lonnet::allowed('bro','/res/')) {
$item->[0] .= '?launch=1';
} else {
next;
}
}
if ($env{'request.course.id'} && $menucoll) {
next if ($item->[3]) && (!$menuopts{$item->[3]});
}
push(@primsub,$item);
}
if ($title eq 'Personal') {
if ($env{'user.name'} && $env{'user.domain'} && !$ltiexc{'fullname'}) {
unless (($env{'request.course.id'}) && ($menucoll) && (!$menuopts{'name'})) {
$title = &Apache::loncommon::plainname($env{'user.name'},$env{'user.domain'});
}
}
next if (($env{'request.course.id'}) && ($menucoll) && ($title eq 'Personal') &&
(!@primsub));
if ($title eq 'Personal') {
$title = &mt($title);
}
} else {
$title = &mt($title);
}
if (@primsub > 0) {
$menu{$position} .= &create_submenu($link,$target,$title,\@primsub,1,undef,$listclass,$linkattr);
} elsif ($link) {
$menu{$position} .= ($listclass?'<li class="'.$listclass.'">':'<li>').
'<a href="'.$link.'" target="'.$target.'" '.$linkattr.'>'.$title.'</a></li>';
}
}
} elsif ($$menuitem[3] eq 'Help') { # special treatment for helplink
next if ($crstype eq 'Placement');
if ($public) {
my $origmail = $Apache::lonnet::perlvar{'lonSupportEMail'};
my $defdom = &Apache::lonnet::default_login_domain();
my $to = &Apache::loncommon::build_recipient_list(undef,
'helpdeskmail',
$defdom,$origmail);
if ($to ne '') {
$menu{$position} .= &prep_menuitem($menuitem,$target,$listclass,$linkattr);
}
} else {
$menu{$position} .= ($listclass?'<li class="'.$listclass.'">':'<li>').
&Apache::loncommon::top_nav_help('Help',$linkattr).
'</li>';
}
} elsif ($$menuitem[3] eq 'Log In') {
if ($public) {
if (&Apache::lonnet::get_saml_landing()) {
$$menuitem[0] = '/adm/login';
}
}
$menu{$position} .= prep_menuitem($menuitem,$target,$listclass,$linkattr);
} else {
$menu{$position} .= prep_menuitem($menuitem,$target,$listclass,$linkattr);
}
}
my @output = ('','');
if ($menu{'left'} ne '') {
if ($collapsible) {
$menu{'left'} = ($listclass?'<li class="'.$listclass.'">':'<li>').
' </li>'.$menu{'left'};
}
$output[0] = "<ol class=\"LC_primary_menu LC_floatleft\">$menu{'left'}</ol>";
}
if ($menu{'right'} ne '') {
$output[1] = "<ol class=\"LC_primary_menu LC_floatright LC_right\">$menu{'right'}</ol>";
}
return @output;
}
#returns hashref {user=>'',dom=>''} containing:
# own name, domain if user is au
# name, domain of parent author if user is ca or aa
#empty return if user is not an author or not on homeserver
#
#TODO this should probably be moved somewhere more central
#since it can be used by different parts of the system
sub getauthor {
return unless $env{'request.role'}=~/^(ca|aa|au)/; #nothing to do if user isn't some kind of author
#co- or assistant author?
my ($dom, $user) = ($env{'request.role'} =~ /^(?:ca|aa)\.\/($match_domain)\/($match_username)$/)
? ($1, $2) #domain, username of the parent author
: @env{ ('request.role.domain', 'user.name') }; #own domain, username
# current server == home server?
my $home = &Apache::lonnet::homeserver($user,$dom);
foreach (&Apache::lonnet::current_machine_ids()){
return {user => $user, dom => $dom} if $_ eq $home;
}
# if wrong server
return;
}
sub secondary_menu {
my ($httphost,$ltiscope,$ltimenu,$noprimary,$menucoll,$menuref,
$links_disabled,$links_target) = @_;
my $menu;
my $crstype = &Apache::loncommon::course_type();
my $crs_sec = $env{'request.course.id'} . ($env{'request.course.sec'}
? "/$env{'request.course.sec'}"
: '');
my $canedit = &Apache::lonnet::allowed('mdc', $env{'request.course.id'});
my $canvieweditor = &Apache::lonnet::allowed('cev', $env{'request.course.id'});
my $canviewroster = $env{'course.'.$env{'request.course.id'}.'.student_classlist_view'};
if ($canviewroster eq 'disabled') {
undef($canviewroster);
}
my $canviewgrps = &Apache::lonnet::allowed('vcg', $crs_sec);
my $canmodifyuser = &Apache::lonnet::allowed('cst', $crs_sec);
my $canviewusers = &Apache::lonnet::allowed('vcl', $crs_sec);
my $canviewwnew = &Apache::lonnet::allowed('whn', $crs_sec);
my $canviewpara = &Apache::lonnet::allowed('vpa', $crs_sec);
my $canmodpara = &Apache::lonnet::allowed('opa', $crs_sec);
my $canvgr = &Apache::lonnet::allowed('vgr', $crs_sec);
my $canmgr = &Apache::lonnet::allowed('mgr', $crs_sec);
my $canplc = &Apache::lonnet::allowed('plc', $crs_sec);
my $author = &getauthor();
my ($is_author,$is_coauthor);
if ($author) {
if ($env{'request.role'} =~ /^au\./) {
$is_author = 1;
} elsif ($env{'request.role'} =~ /^ca\./) {
$is_coauthor = 1;
}
}
my ($cdom,$cnum,$showsyllabus,$showfeeds,$showresv,$grouptools,
$lti,$ltimapres,%ltiexc,%menuopts);
$grouptools = 0;
if ($env{'request.course.id'}) {
$cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
$cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
unless ($canedit || $canvieweditor) {
unless (&Apache::lonnet::is_on_map("public/$cdom/$cnum/syllabus")) {
if (($env{'course.'.$env{'request.course.id'}.'.externalsyllabus'}) ||
($env{'course.'.$env{'request.course.id'}.'.uploadedsyllabus'}) ||
($env{'course.'.$env{'request.course.id'}.'.updatedsyllabus'}) ||
($env{'request.course.syllabustime'})) {
$showsyllabus = 1;
}
}
if ($env{'request.course.feeds'}) {
$showfeeds = 1;
}
}
unless ($canmgr || $canvgr) {
my %slots = &Apache::lonnet::get_course_slots($cnum,$cdom);
if (keys(%slots) > 0) {
$showresv = 1;
}
}
if ($env{'request.course.groups'} ne '') {
foreach my $group (split(/:/,$env{'request.course.groups'})) {
next unless ($group =~ /^\w+$/);
my @privs = split(/:/,$env{"user.priv.$env{'request.role'}./$cdom/$cnum/$group"});
shift(@privs);
if (@privs) {
$grouptools ++;
}
}
}
if ($env{'request.lti.login'}) {
$lti = 1;
if (ref($ltimenu) eq 'HASH') {
foreach my $item ('fullname','coursetitle','role','logout','grades') {
unless ($ltimenu->{$item}) {
$ltiexc{$item} = 1;
}
}
}
if (($ltiscope eq 'map') || ($ltiscope eq 'resource')) {
$ltimapres = 1;
}
}
}
if (($menucoll) && (ref($menuref) eq 'HASH')) {
%menuopts = %{$menuref};
}
my ($listclass,$linkattr,$target);
if ($links_disabled) {
$listclass = 'LCisDisabled';
$linkattr = 'aria-disabled="true"';
}
my ($canlistcoauthors,$canmodifycoauthor);
if ($env{'request.role'} eq "au./$env{'user.domain'}/") {
my $extent = "$env{'user.domain'}/$env{'user.name'}";
if ((&Apache::lonnet::allowed('cca',$extent)) ||
(&Apache::lonnet::allowed('caa',$extent))) {
$canmodifycoauthor = 1;
}
} elsif ($env{'request.role'} =~ m{^(aa|ca)\./($match_domain/$match_username)$}) {
my ($role,$extent) = ($1,$2);
if (&Apache::lonnet::allowed('vca',$extent)) {
if ($env{"environment.internal.manager./$extent"}) {
$canmodifycoauthor = 1;
} else {
$canlistcoauthors = 1;
}
} elsif (&Apache::lonnet::allowed('vaa',$extent)) {
$canlistcoauthors = 1;
}
}
my ($roleswitcher_js,$roleswitcher_form);
if ($links_target ne '') {
$target = $links_target;
} else {
my ($ltitarget,$deeplinktarget);
if ($env{'request.lti.login'}) {
$ltitarget = $env{'request.lti.target'};
}
if ($env{'request.deeplink.login'}) {
$deeplinktarget = $env{'request.deeplink.target'};
}
if (($ltitarget eq 'iframe') || ($deeplinktarget eq '_self')) {
$target = '_self';
} else {
$target = '_top';
}
}
foreach my $menuitem (@secondary_menu) {
# evaluate conditions
next if ref($menuitem) ne 'ARRAY';
next if (($crstype eq 'Placement') && ($$menuitem[3] ne 'Roles') && (!$env{'request.role.adv'}));
next if $$menuitem[4] ne 'always'
&& $$menuitem[4] ne 'coauthor'
&& $$menuitem[4] ne 'author'
&& $$menuitem[4] ne 'authorspace'
&& $$menuitem[4] ne 'vca'
&& $$menuitem[4] ne 'mca'
&& !$env{'request.course.id'};
next if $$menuitem[4] =~ /^crsedit/
&& (!$canedit && !$canvieweditor);
next if $$menuitem[4] eq 'crseditCourse'
&& ($crstype eq 'Community');
next if $$menuitem[4] eq 'crseditCommunity'
&& ($crstype ne 'Community');
next if $$menuitem[4] eq 'nvgr'
&& ($canvgr || $ltiexc{'grades'});
next if $$menuitem[4] eq 'vgr'
&& !$canvgr;
next if $$menuitem[4] eq 'viewusers'
&& !$canmodifyuser && !$canviewusers;
next if $$menuitem[4] eq 'noviewusers'
&& ($canmodifyuser || $canviewusers || !$canviewroster);
next if $$menuitem[4] eq 'mgr'
&& !$canmgr;
next if $$menuitem[4] eq 'showresv'
&& !$showresv;
next if $$menuitem[4] eq 'whn'
&& !$canviewwnew;
next if $$menuitem[4] eq 'params'
&& (!$canmodpara && !$canviewpara);
next if $$menuitem[4] eq 'showgroups'
&& ($canviewgrps || !$grouptools);
next if $$menuitem[4] eq 'showsyllabus'
&& !$showsyllabus;
next if $$menuitem[4] eq 'showfeeds'
&& !$showfeeds;
next if $$menuitem[4] eq 'plc'
&& !$canplc;
next if $$menuitem[4] eq 'authorspace'
&& !$author;
next if $$menuitem[4] eq 'author'
&& !$is_author;
next if $$menuitem[4] eq 'coauthor'
&& !$is_coauthor;
next if $$menuitem[4] eq 'vca'
&& (!$canlistcoauthors || $canmodifycoauthor);
next if $$menuitem[4] eq 'vaa'
&& (!$canlistcoauthors || $canmodifycoauthor);
next if $$menuitem[4] eq 'mca'
&& !$canmodifycoauthor;
next if $$menuitem[4] eq 'notltimapres'
&& $ltimapres;
next if $$menuitem[4] eq 'notlti'
&& $lti;
next if $$menuitem[4] eq 'lti'
&& (!$lti || !$noprimary);
next if $$menuitem[3] eq 'Logout'
&& $ltiexc{'logout'};
my $title = $menuitem->[3];
if ($env{'request.course.id'} && $menucoll) {
unless ($$menuitem[5] eq 'roles') {
next if (($$menuitem[5]) && (!$menuopts{$$menuitem[5]}));
}
}
if (defined($secondary_submenu{$title})) {
my $link;
if ($menuitem->[0] ne '') {
$link = $menuitem->[0];
} else {
$link = '#';
}
my @scndsub;
if (ref($secondary_submenu{$title}) eq 'ARRAY') {
foreach my $item (@{$secondary_submenu{$title}}) {
if (ref($item) eq 'ARRAY') {
next if ($item->[2] eq 'vgr' && !$canvgr);
next if ($item->[2] eq 'opa' && !$canmodpara);
next if ($item->[2] eq 'vpa' && !$canviewpara);
next if ($item->[2] eq 'viewusers' && !($canmodifyuser || $canviewusers));
next if ($item->[2] eq 'mgr' && !$canmgr);
next if ($item->[2] eq 'vcg' && !$canviewgrps);
next if ($item->[2] eq 'crsedit' && !$canedit && !$canvieweditor);
next if ($item->[2] eq 'params' && !$canmodpara && !$canviewpara);
next if ($item->[2] eq 'author' && !$is_author);
next if ($item->[2] eq 'vca' && !$canlistcoauthors);
next if ($item->[2] eq 'lti' && !$lti);
if ($item->[2] =~ /^lti(portfolio|wishlist|blog)$/) {
my $tool = $1;
next if !$lti;
next if (!&Apache::lonnet::usertools_access('','',$tool,
undef,'tools'));
}
push(@scndsub,$item);
}
}
if ($title eq 'Personal' && $env{'user.name'} && $env{'user.domain'}) {
unless ($ltiexc{'fullname'}) {
$title = &Apache::loncommon::plainname($env{'user.name'},$env{'user.domain'});
}
}
if (@scndsub > 0) {
$menu .= &create_submenu($link,$target,&mt($title),\@scndsub,1,undef,
$listclass,$linkattr);
} elsif ($link ne '#') {
$menu .= ($listclass?'<li class="'.$listclass.'">':'<li>').
'<a href="'.$link.'" target="'.$target.'" '.$linkattr.'>'.
&mt($title).'</a></li>';
}
}
} elsif ($$menuitem[3] eq 'Roles' && $env{'request.course.id'}) {
# special treatment for role selector
my ($switcher,$has_opa_priv);
($roleswitcher_js,$roleswitcher_form,$switcher,$has_opa_priv) =
&roles_selector(
$env{'course.' . $env{'request.course.id'} . '.domain'},
$env{'course.' . $env{'request.course.id'} . '.num'},
$httphost,$target,$menucoll,$menuref
);
if (($$menuitem[5]) && (!$menuopts{$$menuitem[5]})) {
next unless ($has_opa_priv);
}
$menu .= $switcher;
} elsif ($$menuitem[3] eq 'Help') { # special treatment for helplink
next if ($crstype eq 'Placement');
$menu .= '<li>'.&Apache::loncommon::top_nav_help('Help').'</li>';
} else {
if ($$menuitem[3] eq 'Syllabus' && $env{'request.course.id'}) {
my $url = $$menuitem[0];
$url =~ s{\[cdom\]/\[cnum\]}{$cdom/$cnum};
if (&Apache::lonnet::is_on_map($url)) {
unless ($$menuitem[0] =~ /(\?|\&)register=1/) {
$$menuitem[0] .= (($$menuitem[0]=~/\?/)? '&' : '?').'register=1';
}
} else {
$$menuitem[0] =~ s{\&?register=1}{};
}
if ($env{'course.'.$env{'request.course.id'}.'.externalsyllabus'} =~ m{^http://}) {
if (($ENV{'SERVER_PORT'} == 443) || ($env{'request.use_absolute'} =~ m{^https://})) {
unless ((&Apache::lonnet::uses_sts()) || (&Apache::lonnet::waf_allssl())) {
unless ($$menuitem[0] =~ m{^https?://}) {
$$menuitem[0] = 'http://'.$ENV{'SERVER_NAME'}.$$menuitem[0];
}
unless ($$menuitem[0] =~ /(\&|\?)usehttp=1/) {
$$menuitem[0] .= (($$menuitem[0]=~/\?/) ? '&' : '?').'usehttp=1';
}
}
}
}
$$menuitem[0] = &HTML::Entities::encode($$menuitem[0],'&<>"');
}
$menu .= &prep_menuitem(\@$menuitem,$target,$listclass,$linkattr);
}
}
if ($menu =~ /\[url\].*\[symb\]/) {
my $escurl = &escape( &Apache::lonenc::check_encrypt(
$env{'request.noversionuri'}));
my $escsymb = &escape( &Apache::lonenc::check_encrypt(
$env{'request.symb'}));
if ( $env{'request.state'} eq 'construct'
and ( $env{'request.noversionuri'} eq ''
|| !defined($env{'request.noversionuri'})))
{
my $londocroot = $Apache::lonnet::perlvar{'lonDocRoot'};
($escurl = $env{'request.filename'}) =~ s{^\Q$londocroot\E}{};
$escurl = &escape($escurl);
}
$menu =~ s/\[url\]/$escurl/g;
$menu =~ s/\[symb\]/$escsymb/g;
} elsif (($menu =~ m{/adm/preferences\?}) && ($menu =~ /\[returnurl\]/)) {
my $returnurl = $ENV{'REQUEST_URI'};
if ($ENV{'REQUEST_URI'} =~ m{/adm/preferences\?action=authorsettings\&returnurl=([^\&]+)$}) {
$returnurl = $1;
}
if (($returnurl =~ m{^/adm/createuser($|\?action=)}) ||
($returnurl =~ m{^/priv/$match_domain/$match_username}) ||
($returnurl =~ m{^/res(/?$|/$match_domain/$match_username)})) {
$returnurl =~ s{\?.*$}{};
$returnurl = '&returnurl='.&HTML::Entities::encode($returnurl,'"<>&\'');
} else {
undef($returnurl);
}
$menu =~ s/\[returnurl\]/$returnurl/;
}
$menu =~ s/\[uname\]/$$author{user}/g;
$menu =~ s/\[udom\]/$$author{dom}/g;
$menu =~ s/\[javascript\]/javascript:/g;
if ($env{'request.course.id'}) {
$menu =~ s/\[cnum\]/$cnum/g;
$menu =~ s/\[cdom\]/$cdom/g;
}
if ($menu) {
$menu = "<ul id=\"LC_secondary_menu\">$menu</ul>";
}
if ($roleswitcher_form) {
$menu .= "\n$roleswitcher_js\n$roleswitcher_form";
}
return $menu;
}
sub create_submenu {
my ($link,$target,$title,$submenu,$translate,$addclass,$listclass,$linkattr) = @_;
return unless (ref($submenu) eq 'ARRAY');
my $targetattr;
if (($target ne '') && ($link ne '#')) {
$targetattr = ' target="'.$target.'"';
}
my $menu = '<li class="LC_hoverable '.$addclass.'">'.
'<a href="'.$link.'"'.$targetattr.'>'.
'<span class="LC_nobreak">'.$title.
'<span class="LC_fontsize_small" style="font-weight:normal;">'.
' ▼</span></span></a>'.
'<ul>';
# $link and $title are only used in the initial string written in $menu
# as seen above, not needed for nested submenus
$menu .= &build_submenu($target, $submenu, $translate, '1', $listclass, $linkattr);
$menu .= '</ul></li>';
return $menu;
}
# helper routine for create_submenu
# build the dropdown (and nested submenus) recursively
# see perldoc create_submenu documentation for further information
sub build_submenu {
my ($target, $submenu, $translate, $first_level, $listclass, $linkattr) = @_;
unless (@{$submenu}) {
return '';
}
my $menu = '';
my $count = 0;
my $numsub = scalar(@{$submenu});
foreach my $item (@{$submenu}) {
$count ++;
if (ref($item) eq 'ARRAY') {
my $href = $item->[0];
my $bordertop;
my $borderbot;
my $title;
if ($translate) {
$title = &mt($item->[1]);
} else {
$title = $item->[1];
}
if ($count == 1 && !$first_level) {
$bordertop = 'border-top: 1px solid black;';
}
if ($count == $numsub) {
$borderbot = 'border-bottom: 1px solid black;';
}
# href is a reference to another submenu
if (ref($href) eq 'ARRAY') {
$menu .= '<li style="margin:0;padding:0;'.$bordertop . $borderbot . '">';
$menu .= '<p><span class="LC_primary_menu_innertitle">'
. $title . '</span><span class="LC_primary_menu_innerarrow">▶</span></p>';
$menu .= '<ul>';
$menu .= &build_submenu($target, $href, $translate);
$menu .= '</ul>';
$menu .= '</li>';
} else { # href is the actual hyperlink and does not represent another submenu
# for the current menu title
if ($href =~ /(aboutme|rss\.html)$/) {
next unless (($env{'user.name'} ne '') && ($env{'user.domain'} ne ''));
$href =~ s/\[domain\]/$env{'user.domain'}/g;
$href =~ s/\[user\]/$env{'user.name'}/g;
} elsif (($href =~ m{^/adm/preferences\?}) && ($href =~ /\[returnurl\]/)) {
my $returnurl = $ENV{'REQUEST_URI'};
if ($ENV{'REQUEST_URI'} =~ m{/adm/preferences\?action=(?:changedomcoord|authorsettings)\&returnurl=([^\&]+)$}) {
$returnurl = $1;
}
if (($returnurl =~ m{^/adm/createuser($|\?action=)}) ||
($returnurl =~ m{^/priv/$match_domain/$match_username}) ||
($returnurl =~ m{^/res(/?$|/$match_domain/$match_username)})) {
$returnurl =~ s{\?.*$}{};
$returnurl = '&returnurl='.&HTML::Entities::encode($returnurl,'"<>&\'');
} else {
undef($returnurl);
}
$href =~ s/\[returnurl\]/$returnurl/;
}
my $targetattr;
unless (($href eq '') || ($href =~ /^\#/)) {
if ($target ne '') {
$targetattr = ' target="'.$target.'"';
}
}
$menu .= '<li ';
$menu .= ($listclass?'class="'.$listclass.'" ':'');
$menu .= 'style="margin:0;padding:0;'. $bordertop . $borderbot .'">';
$menu .= '<a href="'.$href.'"'.$targetattr.' '.$linkattr.'>' . $title . '</a>';
$menu .= '</li>';
}
}
}
return $menu;
}
sub innerregister {
my ($forcereg,$bread_crumbs,$group,$pagebuttonshide,$hostname,
$ltiscope,$ltiuri,$showncrumbsref) = @_;
my $const_space = ($env{'request.state'} eq 'construct');
my $in_daxe = 0;
if ($env{'request.noversionuri'} =~ m{^/res/adm/pages/}) { return ''; }
$env{'request.registered'} = 1;
undef(@inlineremote);
my ($mapurl,$resurl,$crstype,$navmap);
if ($env{'request.course.id'}) {
#
#course_type: Course, Community, or Placement
#
$crstype = &Apache::loncommon::course_type();
if ($env{'request.symb'}) {
my $ignorenull;
unless ($env{'request.noversionuri'} eq '/adm/navmaps') {
$ignorenull = 1;
}
my $symb = &Apache::lonnet::symbread('','',$ignorenull);
($mapurl, my $rid, $resurl) = &Apache::lonnet::decode_symb($symb);
my $coursetitle = $env{'course.'.$env{'request.course.id'}.'.description'};
my $maptitle = &Apache::lonnet::gettitle($mapurl);
my $restitle = &Apache::lonnet::gettitle($symb);
my (@crumbs,@mapcrumbs);
if (($env{'request.noversionuri'} ne '/adm/navmaps') && ($mapurl ne '') &&
(!(($crstype eq 'Placement') && !$env{'request.role.adv'}))) {
unless ($ltiscope eq 'resource') {
if (($mapurl ne $env{'course.'.$env{'request.course.id'}.'.url'}) &&
!(($ltiscope eq 'map') && (&Apache::lonnet::clutter($resurl) eq $ltiuri))) {
$navmap = Apache::lonnavmaps::navmap->new();
if (ref($navmap)) {
@mapcrumbs = $navmap->recursed_crumbs($mapurl,$restitle);
}
}
}
}
unless ((($crstype eq 'Placement') && (!$env{'request.role.adv'})) ||
($ltiscope eq 'map') || ($ltiscope eq 'resource')) {
@crumbs = ({text => $crstype.' Contents',
href => "Javascript:gopost('/adm/navmaps','')"});
}
if ($mapurl ne $env{'course.'.$env{'request.course.id'}.'.url'}) {
if (@mapcrumbs) {
push(@crumbs,@mapcrumbs);
} elsif (!(($crstype eq 'Placement') && (!$env{'request.role.adv'})) &&
($ltiscope ne 'map') && ($ltiscope ne 'resource')) {
push(@crumbs, {text => '...',
no_mt => 1});
}
}
unless ((($crstype eq 'Placement') && (!$env{'request.role.adv'})) || (@mapcrumbs) ||
(!$maptitle) || ($maptitle eq 'default.sequence') ||
($mapurl eq $env{'course.'.$env{'request.course.id'}.'.url'}) ||
($ltiscope eq 'resource')) {
push @crumbs, {text => $maptitle, no_mt => 1,
href => &Apache::lonnet::clutter($mapurl).'?navmap=1'};
}
if ($restitle && !@mapcrumbs) {
push(@crumbs,{text => $restitle, no_mt => 1});
}
my @tools;
if ($env{'request.filename'} =~ /\.page$/) {
my %breadcrumb_tools = &Apache::lonhtmlcommon::current_breadcrumb_tools();
if (ref($breadcrumb_tools{'tools'}) eq 'ARRAY') {
@tools = @{$breadcrumb_tools{'tools'}};
}
}
&Apache::lonhtmlcommon::clear_breadcrumbs();
&Apache::lonhtmlcommon::add_breadcrumb(@crumbs);
if (@tools) {
&Apache::lonhtmlcommon::add_breadcrumb_tool('tools',@tools);
}
} else {
$resurl = $env{'request.noversionuri'};
my $courseurl = &Apache::lonnet::courseid_to_courseurl($env{'request.course.id'});
my $title = &mt('View Resource');
if ($resurl =~ m{^\Q/uploaded$courseurl/supplemental/\E(default|\d+)/}) {
&Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},['folderpath','title']);
&Apache::lonhtmlcommon::clear_breadcrumbs();
if ($env{'form.title'}) {
$title = $env{'form.title'};
}
my ($trail,$cnum,$cdom);
if ($env{'form.folderpath'}) {
$cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
$cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
&Apache::loncommon::validate_folderpath(1,'',$cnum,$cdom);
}
if ($env{'form.folderpath'}) {
&prepare_functions($resurl,$forcereg,$group,undef,undef,1,$hostname);
$title = &HTML::Entities::encode($title,'\'"<>&');
($trail) =
&Apache::lonhtmlcommon::docs_breadcrumbs(undef,$crstype,undef,$title,1,1);
} else {
&Apache::lonhtmlcommon::add_breadcrumb(
{text => "Supplemental $crstype Content",
href => "javascript:gopost('/adm/supplemental','')"});
$title = &HTML::Entities::encode(&mt('View Resource'),'\'"<>&');
($trail) =
&Apache::lonhtmlcommon::docs_breadcrumbs(undef,$crstype,undef,$title,1,1);
}
if (ref($showncrumbsref)) {
$$showncrumbsref = 1;
}
return $trail;
} elsif ($resurl =~ m{^\Q/uploaded$courseurl/portfolio/syllabus/}) {
&Apache::lonhtmlcommon::clear_breadcrumbs();
&prepare_functions('/public'.$courseurl."/syllabus",
$forcereg,$group,undef,undef,1,$hostname);
$title = &HTML::Entities::encode(&mt('Syllabus File'),'\'"<>&');
my ($trail) =
&Apache::lonhtmlcommon::docs_breadcrumbs(undef,$crstype,undef,$title,1,1);
if (ref($showncrumbsref)) {
$$showncrumbsref = 1;
}
return $trail;
} elsif (($resurl eq '/public'.$courseurl.'/syllabus') &&
($env{'form.folderpath'})) {
if ($env{'form.title'}) {
$title = $env{'form.title'};
} else {
$title = 'Syllabus';
}
&prepare_functions($resurl,$forcereg,$group,undef,undef,1,$hostname);
$title = &HTML::Entities::encode($title,'\'"<>&');
my ($trail) =
&Apache::lonhtmlcommon::docs_breadcrumbs(undef,$crstype,undef,$title,1,1);
return $trail;
}
unless ($env{'request.state'} eq 'construct') {
&Apache::lonhtmlcommon::clear_breadcrumbs();
&Apache::lonhtmlcommon::add_breadcrumb({text => 'View Resource'});
}
}
} elsif (! $const_space){
#a situation when we're looking at a resource outside of context of a
#course or construction space (e.g. with cumulative rights)
&Apache::lonhtmlcommon::clear_breadcrumbs();
unless ($env{'request.noversionuri'} =~ m{^/adm/$match_domain/$match_username/aboutme$}) {
&Apache::lonhtmlcommon::add_breadcrumb({text => 'View Resource'});
}
}
# =============================================================================
# ============================ This is for URLs that actually can be registered
return '' unless ( ($env{'request.noversionuri'}!~m{^/(res/)*adm/})
|| $forcereg );
my ($cdom,$cnum,%perms,$cfile,$switchserver,$home,$forceedit,
$forceview,$editbutton);
if (($resurl =~ m{^/?adm/($match_domain)/($match_username)/aboutme$}) ||
($env{'request.role'} !~/^(aa|ca|au)/)) {
$editbutton = &prepare_functions($resurl,$forcereg,$group,'','','',$hostname);
}
if ($editbutton eq '') {
$editbutton = &clear(6,1);
}
#
# This applies in course context
#
if ($env{'request.course.id'}) {
$cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
$cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
$perms{'mdc'} = &Apache::lonnet::allowed('mdc',$env{'request.course.id'});
$perms{'cev'} = &Apache::lonnet::allowed('cev',$env{'request.course.id'});
my @privs;
my $gradable_exttool;
if ($env{'request.symb'} ne '') {
if ($env{'request.noversionuri'} =~ m{^/adm/$cdom/$cnum/(\d+)/ext\.tool$}) {
if (&Apache::lonnet::EXT('resource.0.gradable') =~ /^yes$/i) {
$gradable_exttool = 1;
push(@privs,('mgr','vgr'));
}
} elsif ($env{'request.filename'}=~/$LONCAPA::assess_re/) {
push(@privs,('mgr','vgr'));
}
push(@privs,('opa','vpa'));
}
foreach my $priv (@privs) {
$perms{$priv} = &Apache::lonnet::allowed($priv,$env{'request.course.id'});
if (!$perms{$priv} && $env{'request.course.sec'} ne '') {
$perms{$priv} =
&Apache::lonnet::allowed($priv,"$env{'request.course.id'}/$env{'request.course.sec'}");
}
}
#
# Determine whether or not to show Grades and Submissions buttons
#
if (($env{'request.symb'} ne '') &&
(($env{'request.filename'}=~/$LONCAPA::assess_re/) || ($gradable_exttool))) {
if ($perms{'mgr'}) {
&switch('','',7,2,'pgrd.png','Content Grades','grades[_4]',
"gocmd('/adm/grades','gradingmenu')",
'Content Grades');
} elsif ($perms{'vgr'}) {
&switch('','',7,2,'subm.png','Content Submissions','missions[_1]',
"gocmd('/adm/grades','submission')",
'Content Submissions');
}
}
if (($env{'request.symb'} ne '') && (($perms{'opa'}) || ($perms{'vpa'}))) {
&switch('','',7,3,'pparm.png','Content Settings','parms[_2]',
"gocmd('/adm/parmset','set')",
'Content Settings');
}
# End grades/submissions check
#
# This applies to items inside a folder/page modifiable in the course.
#
if (($env{'request.symb'}=~/^uploaded/) && (($perms{'mdc'}) || ($perms{'cev'}))) {
my $text = 'Edit Folder';
if (($mapurl =~ /\.page$/) ||
($env{'request.symb'}=~
m{uploaded/$cdom/$cnum/default_\d+\.page$})) {
$text = 'Edit Page';
}
&switch('','',7,4,'docs-22x22.png',$text,'parms[_2]',
"gocmd('/adm/coursedocs','direct')",
'Folder/Page Content');
}
# End modifiable folder/page container check
#
# Determine whether to show View As button for shortcut to display problem, answer, and submissions
#
if (($env{'request.symb'} ne '') &&
($env{'request.filename'}=~/$LONCAPA::assess_re/) &&
(($perms{'mgr'}) || ($perms{'vgr'}))) {
my ($viewas,$text,$change,$visibility,$vuname,$vudom,$vid,$leftvis,$defdom,
$domselector,$righticon);
my %lt = &Apache::lonlocal::texthash(
view => 'View',
upda => 'Update',
);
my $possdomstr = $env{'course.'.$env{'request.course.id'}.'.internal.userdomains'};
if ($possdomstr =~ /,/) {
my @possdoms = split(/,/,$possdomstr);
if ($env{'request.user_in_effect'} =~ /^$match_username:($match_domain)$/) {
$defdom = $1;
} elsif (grep(/^\Q$cdom\E$/,@possdoms)) {
$defdom = $cdom;
} elsif (&Apache::lonnet::domain($possdoms[0]) ne '') {
$defdom = $possdoms[0];
}
$domselector = &Apache::loncommon::select_dom_form($defdom,'vudom','','','',\@possdoms);
} elsif (($possdomstr ne '') && (&Apache::lonnet::domain($possdomstr) ne '')) {
if ($env{'request.user_in_effect'} =~ /^$match_username:($match_domain)$/) {
$defdom = $1;
} else {
$defdom = $possdomstr;
}
}
if ($env{'request.user_in_effect'} =~ /^($match_username):($match_domain)$/) {
($vuname,$vudom) = ($1,$2);
unless (&Apache::lonnet::is_advanced_user($vudom,$vuname)) {
$vid = (&Apache::lonnet::idrget($vudom,$vuname))[1];
}
$viewas = $env{'request.user_in_effect'};
$text = $lt{'upda'};
$change = 'off';
$visibility = 'inline';
$leftvis = 'none';
$defdom = $vudom;
$righticon = '✖';
} else {
$text = $lt{'view'};
$change = 'on';
$visibility = 'none';
$leftvis = 'inline';
if ($defdom eq '') {
$defdom = $cdom;
}
}
my $sellink = &Apache::loncommon::selectstudent_link('userview','vuname','vudom','','','vuidentifier');
my $selscript=&Apache::loncommon::studentbrowser_javascript();
my $shownsymb = &HTML::Entities::encode(&Apache::lonenc::check_encrypt($env{'request.symb'}),'<>&"');
my $input;
my @items = (
'<label><input type="radio" name="vuidentifier" value="uname" checked="checked" onclick="javascript:toggleIdentifier(this.form);" />',
'</label><input name="vuname" type="text" size="6" value="'.$vuname.'" id="LC_vuname" />',
'<label><input type="radio" name="vuidentifier" value="uid" onclick="javascript:toggleIdentifier(this.form);" />',
'</label><input name="vid" type="hidden" size="6" value="'.$vid.'" id="LC_vid" />'
);
if ($domselector) {
push(@items,$domselector);
$input = &mt('[_1]User:[_2] or [_3]ID:[_4] at [_5] | ',@items);
} else {
$input = &mt('[_1]Username:[_2] or [_3]ID:[_4] | ',@items).
'<input name="vudom" type="hidden" value="'.$defdom.'" />';
}
$input .= '<input name="LC_viewas" type="hidden" value="'.$viewas.'" />',
'<input name="symb" type="hidden" value="'.$shownsymb.'" />';
my $chooser = <<END;
$selscript
<a href="javascript:toggleViewAsUser('$change');" class="LC_menubuttons_link">
<span id="usexpand" class="LC_menubuttons_inline_text" style="display:$leftvis">► </span>
</a>
<fieldset id="LC_selectuser" style="display:$visibility">
<form name="userview" action="" method="post" onsubmit="event.preventDefault(); return validCourseUser(this,'$change');">
<span class="LC_menubuttons_inline_text LC_nobreak">
$input
$sellink
</span>
<input type="submit" value="$text" />
</form>
</fieldset>
<a href="javascript:toggleViewAsUser('$change');" class="LC_menubuttons_link">
<span id="uscollapse" class="LC_menubuttons_inline_text">$righticon</span>
</a>
END
&switch('','',7,5,'viewuser.png','View As','user[_1]',
'toggleViewAsUser('."'$change'".')',
'View As','','',$chooser);
}
# End view as user check
}
# End course context
# Prepare the rest of the buttons
my ($menuitems,$got_prt,$got_wishlist,$crsauthor,$toplevel_cstr,$crsauthor_cstr);
if ($const_space) {
#
# We are in construction space
#
my $londocroot = $Apache::lonnet::perlvar{'lonDocRoot'};
my ($udom,$uname,$thisdisfn) =
($env{'request.filename'}=~m{^\Q$londocroot/priv/\E([^/]+)/([^/]+)/(.*)$});
if (($env{'request.course.id'}) &&
($env{'course.'.$env{'request.course.id'}.'.num'} eq $uname) &&
($env{'course.'.$env{'request.course.id'}.'.domain'} eq $udom)) {
$crsauthor_cstr = 1;
}
my $currdir = '/priv/'.$udom.'/'.$uname.'/'.$thisdisfn;
if ($currdir =~ m-/$-) {
if ($thisdisfn eq '') {
$toplevel_cstr = 1;
}
my $esc_currdir = &Apache::loncommon::escape_single($currdir);
$menuitems=(<<ENDMENUITEMS);
s&6&3&pub.png&Publish&dir[_2]&gocstr('/adm/publish','$esc_currdir')&Publish this Directory
s&7&4&docs-22x22.png&Edit Metadata&defaults[_1]&gopost('${esc_currdir}default.meta','')&Edit metadata for this Directory
s&7&2&prt.png&Print&printout[_1]&gocstr('/adm/printout','$esc_currdir')&Print contents of directory
s&7&1&del.png&Delete&dir[_3]&gocstr('/adm/cfile?action=delete','$esc_currdir')&Delete this Directory
ENDMENUITEMS
unless ($crsauthor_cstr) {
if (($env{'environment.canarchive'}) &&
($uname eq $env{'user.name'}) && ($udom eq $env{'user.domain'})) {
$menuitems .= (<<ENDMENUITEMS);
s&7&7&archive.png&Export&dir[_1]&gocstr('/adm/cfile?action=archive','$esc_currdir')&Export Authoring Space Archive
ENDMENUITEMS
}
}
} else {
$currdir =~ s|[^/]+$||;
my $cleandisfn = &Apache::loncommon::escape_single($thisdisfn);
my $esc_currdir = &Apache::loncommon::escape_single($currdir);
my $pubfile = "/res/$udom/$uname/$thisdisfn";
my $candelete = 1;
if (-e $londocroot.$pubfile) {
unless (&Apache::lonnet::metadata($pubfile,'obsolete')) {
undef($candelete);
}
}
#
# Probably should be in mydesk.tab
#
if (($crsauthor_cstr) && ($pubfile eq "/res/$udom/$uname/default.rights")) {
$menuitems=(<<ENDMENUITEMS);
s&6&1&list.png&Directory&dir[_1]&golist('$esc_currdir')&List current directory
s&6&3&pub.png&Publish&resource[_3]&gocstr('/adm/publish','/priv/$udom/$uname/$cleandisfn')&Publish this resource
ENDMENUITEMS
} else {
$menuitems=(<<ENDMENUITEMS);
s&6&1&list.png&Directory&dir[_1]&golist('$esc_currdir')&List current directory
s&6&2&rtrv.png&Retrieve&version[_1]&gocstr('/adm/retrieve','/priv/$udom/$uname/$cleandisfn')&Retrieve old version
s&6&3&pub.png&Publish&resource[_3]&gocstr('/adm/publish','/priv/$udom/$uname/$cleandisfn')&Publish this resource
s&7&3©.png&Copy&resource[_4]&gocstr('/adm/cfile?action=copy','/priv/$udom/$uname/$cleandisfn')&Copy this resource
ENDMENUITEMS
}
#
# Rename and Delete only available if obsolete or unpublished
#
if ($candelete) {
$menuitems .= (<<ENDMENUITEMS);
s&7&4&rename.png&Rename&resource[_5]&gocstr('/adm/cfile?action=rename','/priv/$udom/$uname/$cleandisfn')&Rename this resource
s&7&1&del.png&Delete&resource[_2]&gocstr('/adm/cfile?action=delete','/priv/$udom/$uname/$cleandisfn')&Delete this resource
ENDMENUITEMS
}
#
# Print only makes sense for certain mime types
#
if ($thisdisfn=~/\.(xml|html|htm|xhtml|xhtm|tex)$/ || $thisdisfn=~/$LONCAPA::assess_re/) {
$menuitems .= (<<ENDMENUITEMS);
s&7&2&prt.png&Print&printout[_1]&gocstr('/adm/printout','/priv/$udom/$uname/$cleandisfn')&Prepare a printable document
ENDMENUITEMS
}
#
# "Exit Daxe" in Functions menu when using Daxe
#
if ((($env{'form.editmode'} eq 'daxe') &&
($thisdisfn=~/\.(xml|html|htm|xhtml|xhtm)$/)) ||
(($env{'form.problemmode'} eq 'daxe') &&
($thisdisfn=~/$LONCAPA::assess_re/))) {
my %editors = &Apache::loncommon::permitted_editors();
if ($editors{'daxe'}) {
my $privfile = &Apache::loncommon::escape_single("/priv/$udom/$uname/$thisdisfn");
$in_daxe = 1;
$menuitems .= (<<ENDMENUITEMS);
my $privfile = &Apache::loncommon::escape_single("/priv/$udom/$uname/$thisdisfn");
s&7&6&tolastloc.png&Exit Daxe&resource[_1]&go('$privfile')&Exit editing this resource
ENDMENUITEMS
}
}
}
#
# Editing options usually accessed via "Settings" in inline menu need to be
# accessed in a different way, when Authoring Space is accessed in course
# context
#
if ($env{'request.role'} !~/^(aa|ca|au)/) {
my $privfile = &Apache::loncommon::escape_single("/priv/$udom/$uname/$thisdisfn");
$menuitems .= (<<ENDMENUITEMS);
s&7&5&editops.png&Options&edit[_1]&gocstr('/adm/preferences?action=authorsettings','$privfile')&Authoring Space Options
ENDMENUITEMS
}
if (ref($bread_crumbs) eq 'ARRAY') {
&Apache::lonhtmlcommon::clear_breadcrumbs();
foreach my $crumb (@{$bread_crumbs}){
&Apache::lonhtmlcommon::add_breadcrumb($crumb);
}
}
} elsif ( defined($env{'request.course.id'}) &&
$env{'request.symb'} ne '' ) {
#
# We are in a course and looking at a registered URL
# Should probably be in mydesk.tab
#
$menuitems = "c&3&1";
if ($ltiscope eq 'resource') {
# Suppress display of backward arrow for LTI Provider if scope is resource.
# Suppress display of forward arrow for LTI Provider if scope is resource.
} elsif ($ltiscope eq 'map') {
# Suppress display of backward arrow for LTI Provider if scope is map and this is first resource.
# Suppress display of forward arrow for LTI Provider if scope is map and this is the last resource.
my $showforw = 1;
my $showback = 1;
my $navmap = Apache::lonnavmaps::navmap->new();
if (ref($navmap)) {
my $mapres = $navmap->getResourceByUrl($ltiuri);
if (ref($mapres)) {
if ($navmap->isLastResource($mapres,$env{'request.symb'})) {
$showforw = 0;
}
if ($navmap->isFirstResource($mapres,$env{'request.symb'})) {
$showback = 0;
}
}
}
if ($showback) {
$menuitems.="
s&2&1&back.png&&&gopost('/adm/flip','back:'+currentURL)&Previous content resource&&1";
}
if ($showforw) {
$menuitems.="
s&2&3&forw.png&&&gopost('/adm/flip','forward:'+currentURL)&Next content resource&&3";
}
} elsif (($crstype ne 'Placement') || ($env{'request.role.adv'})) {
$menuitems.="
s&2&1&back.png&&&gopost('/adm/flip','back:'+currentURL)&Previous content resource&&1
s&2&3&forw.png&&&gopost('/adm/flip','forward:'+currentURL)&Next content resource&&3";
} else {
# Suppress display of backward arrow for Placement Tests
# Suppress display of forward arrow for Placement Tests if this is the last resource.
my $showforw = 1;
if ($env{'request.symb'}) {
my $navmap = Apache::lonnavmaps::navmap->new();
if (ref($navmap)) {
if (&Apache::lonplacementtest::is_lastres($env{'request.symb'},$navmap)) {
$showforw = 0;
}
}
}
if ($showforw) {
$menuitems.="
s&2&3&forw.png&&&gopost('/adm/flip','forward:'+currentURL)&Next content resource&&3";
}
}
$menuitems .= (<<ENDMENUITEMS);
c&6&3
c&8&1
c&8&2
s&8&3&prt.png&Print&printout[_1]&gopost('/adm/printout',currentURL)&Prepare a printable document
ENDMENUITEMS
$got_prt = 1;
if (($env{'user.adv'}) && ($env{'request.uri'} =~ /^\/res/)
&& (!$env{'request.enc'})) {
my $privurl = $env{'request.uri'};
$privurl =~ s{^/res/}{/priv/};
my ($cnum,$cdom) = &Apache::loncommon::crsauthor_url($privurl);
if ($cnum) {
$crsauthor = 1;
} else {
# wishlist is only available for users with access to resource-pool
# and links can only be set for resources within the resource-pool
$menuitems .= (<<ENDMENUITEMS);
s&9&1&wishlist-link.png&Stored Links&wishlistlink[_2]&set_wishlistlink()&Save a link for this resource in my personal Stored Links repository&&1
ENDMENUITEMS
$got_wishlist = 1;
}
}
my $currentURL = &Apache::loncommon::get_symb();
my ($symb_old,$symb_old_enc) = &Apache::loncommon::clean_symb($currentURL);
my $annotation = &Apache::loncommon::get_annotation($symb_old,$symb_old_enc);
$menuitems.="s&9&3&";
if(length($annotation) > 0){
$menuitems.="anot2.png";
}else{
$menuitems.="anot.png";
}
$menuitems.="&Notes&&annotate()&";
$menuitems.="Make notes and annotations about this resource&&1\n";
my $is_mobile;
if ($env{'browser.mobile'}) {
$is_mobile = 1;
}
unless ($env{'request.noversionuri'}=~/\/(bulletinboard|smppg|navmaps|syllabus|aboutme|viewclasslist|portfolio)(\?|$)/) {
if ((!$env{'request.enc'}) && ($env{'request.noversionuri'} !~ m{^/adm/wrapper/ext/}) &&
($env{'request.noversionuri'} !~ m{^/uploaded/$match_domain/$match_courseid/(docs/|default_\d+\.page$)}) &&
($env{'request.noversionuri'} !~ m{^/adm/.+/ext\.tool$})) {
$menuitems.=(<<ENDREALRES);
s&6&3&catalog.png&Info&info[_1]&catalog_info(currentURL,'$is_mobile')&Show Metadata
ENDREALRES
}
unless (($env{'request.noversionuri'} =~ m{^/uploaded/$match_domain/$match_courseid/(docs/|default_\d+\.page$)}) ||
($env{'request.noversionuri'} =~ m{^\Q/adm/wrapper/\E(ext|uploaded)/}) ||
($env{'request.noversionuri'} =~ m{^/adm/.+/ext\.tool$})) {
$menuitems.=(<<ENDREALRES);
s&8&1&eval.png&Evaluate&this[_1]&gopost('/adm/evaluate',currentURL,1)&Provide my evaluation of this resource
ENDREALRES
}
unless ($env{'request.noversionuri'} =~ m{^\Q/adm/wrapper/\E(ext|uploaded)/}) {
$menuitems.=(<<ENDREALRES);
s&8&2&fdbk.png&Communicate&discuss[_1]&gopost('/adm/feedback',currentURL,1)&Provide feedback messages or contribute to the course discussion about this resource
ENDREALRES
}
}
}
if ($env{'request.uri'} =~ /^\/res/) {
unless ($got_prt) {
$menuitems .= (<<ENDMENUITEMS);
s&8&3&prt.png&Print&printout[_1]&gopost('/adm/printout',currentURL)&Prepare a printable document
ENDMENUITEMS
$got_prt = 1;
}
unless (($got_wishlist) || ($crsauthor)) {
if (($env{'user.adv'}) && (!$env{'request.enc'})) {
# wishlist is only available for users with access to resource-pool
$menuitems .= (<<ENDMENUITEMS);
s&9&1&wishlist-link.png&Stored Links&wishlistlink[_2]&set_wishlistlink()&Save a link for this resource in your personal Stored Links repository&&1
ENDMENUITEMS
$got_wishlist = 1;
}
}
}
my $buttons='';
foreach (split(/\n/,$menuitems)) {
my ($command,@rest)=split(/\&/,$_);
my $idx=10*$rest[0]+$rest[1];
if (&hidden_button_check() eq 'yes') {
if ($idx == 21 ||$idx == 23) {
$buttons.=&switch('','',@rest);
} else {
$buttons.=&clear(@rest);
}
} else {
if ($command eq 's') {
$buttons.=&switch('','',@rest);
} else {
$buttons.=&clear(@rest);
}
}
}
my ($showprogress,$linkprotout);
if (($crstype eq 'Placement') && (!$env{'request.role.adv'})) {
$showprogress = &placement_progress();
}
if ($env{'request.deeplink.login'}) {
$linkprotout = &linkprot_exit();
}
my $addremote=0;
foreach (@inlineremote) { if ($_ ne '') { $addremote=1; last;} }
if ($addremote) {
my ($countdown,$buttonshide);
if ($env{'request.filename'} =~ /\.page$/) {
my %breadcrumb_tools = &Apache::lonhtmlcommon::current_breadcrumb_tools();
if (ref($breadcrumb_tools{'tools'}) eq 'ARRAY') {
$countdown = $breadcrumb_tools{'tools'}->[0];
}
$buttonshide = $pagebuttonshide;
} else {
$countdown = &countdown_timer();
$buttonshide = &hidden_button_check();
}
&Apache::lonhtmlcommon::clear_breadcrumb_tools();
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'navigation', @inlineremote[21,23]);
if ($buttonshide eq 'yes') {
if ($countdown) {
&Apache::lonhtmlcommon::add_breadcrumb_tool('tools',$countdown);
}
if ($linkprotout) {
&Apache::lonhtmlcommon::add_breadcrumb_tool('tools',$linkprotout);
}
if ($showprogress) {
&Apache::lonhtmlcommon::add_breadcrumb_tool('tools',$showprogress);
}
} else {
my @tools = @inlineremote[93,91,81,82,83];
if ($countdown) {
unshift(@tools,$countdown);
}
if ($linkprotout) {
unshift(@tools,$linkprotout);
}
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'tools',@tools);
#exit editing link/icon when using daxe in construction space
#publish button in construction space
if ($env{'request.state'} eq 'construct'){
if ($in_daxe) {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', $inlineremote[76]);
}
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', $inlineremote[63]);
} else {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'tools', $inlineremote[63]);
}
&advtools_crumbs(@inlineremote);
#options link/icon in constructions space viewed with course role
if (($env{'request.state'} eq 'construct') &&
($env{'request.role'} !~/^(aa|ca|au)/)) {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', $inlineremote[75]);
}
}
} else {
if ($showprogress) {
&Apache::lonhtmlcommon::add_breadcrumb_tool('tools',$showprogress);
}
if ($linkprotout) {
&Apache::lonhtmlcommon::add_breadcrumb_tool('tools',$linkprotout);
}
}
my ($topic_help,$topic_help_text);
if ($toplevel_cstr && !$crsauthor_cstr) {
if ((($ENV{'SERVER_PORT'} == 443) ||
($Apache::lonnet::protocol{$Apache::lonnet::perlvar{'lonHostID'}} eq 'https')) &&
(&Apache::lonnet::usertools_access($env{'user.name'},$env{'user.domain'},'webdav'))) {
$topic_help = 'Authoring_WebDAV,Authoring_WebDAV_Mac_10v6,Authoring_WebDAV_Mac_10v10,'.
'Authoring_WebDAV_Windows_v7,Authoring_WebDAV_Linux_Centos';
$topic_help_text = 'About WebDAV access';
}
}
if (ref($showncrumbsref)) {
$$showncrumbsref = 1;
}
return &Apache::lonhtmlcommon::scripttag('', 'start')
. &Apache::lonhtmlcommon::breadcrumbs(undef,undef,0,'','','','',$topic_help,$topic_help_text)
. &Apache::lonhtmlcommon::scripttag('', 'end');
}
sub get_editbutton {
my ($cfile,$home,$switchserver,$forceedit,$forceview,$forcereg,$hostname) = @_;
my $jscall;
if (($forceview) && ($env{'form.todocs'})) {
my ($folderpath,$command,$navmap);
if ($env{'request.symb'}) {
$folderpath = &Apache::loncommon::symb_to_docspath($env{'request.symb'},\$navmap);
} elsif ($env{'form.folderpath'} =~ /^supplemental/) {
$folderpath = $env{'form.folderpath'};
$command = '&forcesupplement=1';
}
$folderpath = &escape(&HTML::Entities::encode(&escape($folderpath),'<>&"'));
$jscall = "go('/adm/coursedocs?folderpath=$folderpath$command')";
} else {
my $suppanchor;
if ($env{'form.folderpath'}) {
$suppanchor = $env{'form.anchor'};
}
my $shownsymb;
if ($env{'request.symb'}) {
$shownsymb = &Apache::lonenc::check_encrypt($env{'request.symb'});
}
$jscall = &Apache::lonhtmlcommon::jump_to_editres($cfile,$home,$switchserver,
$forceedit,$forcereg,$env{'request.symb'},$shownsymb,
&escape($env{'form.folderpath'}),
&escape($env{'form.title'}),$hostname,
$env{'form.idx'},&escape($env{'form.suppurl'}),
$env{'form.todocs'},$suppanchor);
}
if ($jscall) {
my $icon = 'pcstr.png';
my $label = 'Edit';
if ($forceview) {
$icon = 'tolastloc.png';
$label = 'Exit Editing';
}
&switch('','',6,1,$icon,$label,'resource[_2]',
$jscall,"Edit this resource");
return 1;
}
return;
}
sub prepare_functions {
my ($resurl,$forcereg,$group,$bread_crumbs,$advtools,$docscrumbs,$hostname) = @_;
unless ($env{'request.registered'}) {
undef(@inlineremote);
}
my ($cdom,$cnum,%perms,$cfile,$switchserver,$home,$forceedit,
$forceview);
if ($env{'request.course.id'}) {
$cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
$cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
$perms{'mdc'} = &Apache::lonnet::allowed('mdc',$env{'request.course.id'});
}
my $editbutton = '';
my $viewsrcbutton = '';
my $clientip = &Apache::lonnet::get_requestor_ip();
#
# Determine whether or not to display 'Edit' or 'View Source' icon/button
#
if ($resurl =~ m{^/?adm/($match_domain)/($match_username)/aboutme$}) {
my $blocked = &Apache::loncommon::blocking_status('about',$clientip,$2,$1);
my $file=&Apache::lonnet::declutter($env{'request.filename'});
($cfile,$home,$switchserver,$forceedit,$forceview) =
&Apache::lonnet::can_edit_resource($file,$cnum,$cdom,
&Apache::lonnet::clutter($resurl),$env{'request.symb'},$group);
if (($cfile) && ($home ne '') && ($home ne 'no_host') && (!$blocked)) {
$editbutton = &get_editbutton($cfile,$home,$switchserver,
$forceedit,$forceview,$forcereg);
}
} elsif (!$env{'request.course.id'}) {
if (($env{'user.author'}) && ($resurl eq '/adm/viewcoauthors')) {
if ($env{'request.role'} =~/^(ca|au)/) {
my ($audom,$auname);
if ($env{'request.role'} eq "au./$env{'user.domain'}/") {
($audom,$auname) = ($env{'user.domain'},$env{'user.name'});
} elsif ($env{'request.role'} =~ m{^ca\./($match_domain)/($match_username)}) {
($audom,$auname) = ($1,$2);
}
if (($audom ne '') && ($auname ne '')) {
my $file=&Apache::lonnet::declutter($env{'request.filename'});
($cfile,$home,$switchserver,$forceedit,$forceview) =
&Apache::lonnet::can_edit_resource($file,$auname,$audom,
$resurl);
if ($cfile) {
$editbutton = &get_editbutton($resurl,'','',$forceedit,
$forceview);
}
}
}
} elsif (($env{'user.author'}) && ($env{'request.filename'}) &&
($env{'request.role'} !~/^(aa|ca|au)/)) {
#
# Currently do not have the role of author or co-author.
# Do we have authoring privileges for the resource?
#
my $file=&Apache::lonnet::declutter($env{'request.filename'});
($cfile,$home,$switchserver,$forceedit,$forceview) =
&Apache::lonnet::can_edit_resource($file,$cnum,$cdom,
&Apache::lonnet::clutter($resurl),$env{'request.symb'},$group);
if (($cfile) && ($home ne '') && ($home ne 'no_host')) {
$editbutton = &get_editbutton($cfile,$home,$switchserver,
$forceedit,$forceview,$forcereg);
}
}
} elsif ($env{'request.course.id'}) {
#
# This applies in course context
#
if (($perms{'mdc'}) &&
(($resurl =~ m{^/?public/$cdom/$cnum/syllabus}) ||
($resurl =~ m{^/?uploaded/$cdom/$cnum/portfolio/syllabus/}) ||
(($resurl =~ m{^/?uploaded/$cdom/$cnum/default_\d+\.sequence$}) && ($env{'form.navmap'})))) {
if ($resurl =~ m{^/}) {
$cfile = $resurl;
} else {
$cfile = "/$resurl";
}
$home = &Apache::lonnet::homeserver($cnum,$cdom);
if ($env{'form.forceedit'}) {
$forceview = 1;
} else {
$forceedit = 1;
}
if ($cfile =~ m{^/uploaded/$cdom/$cnum/default_\d+\.sequence$}) {
my $text = 'Edit Folder';
&switch('','',7,4,'docs-22x22.png','Edit Folder','parms[_2]',
"gocmd('/adm/coursedocs','direct')",
'Folder/Page Content');
$editbutton = 1;
} else {
$editbutton = &get_editbutton($cfile,$home,$switchserver,
$forceedit,$forceview,$forcereg,
$hostname);
}
} elsif (($resurl eq '/adm/extresedit') &&
(($env{'form.symb'}) || ($env{'form.folderpath'}))) {
($cfile,$home,$switchserver,$forceedit,$forceview) =
&Apache::lonnet::can_edit_resource($resurl,$cnum,$cdom,$resurl,
$env{'form.symb'});
if ($cfile ne '') {
$editbutton = &get_editbutton($cfile,$home,$switchserver,
$forceedit,$forceview,$forcereg);
}
} elsif (($resurl =~ m{^/?adm/viewclasslist$}) &&
(&Apache::lonnet::allowed('opa',$env{'request.course.id'}))) {
($cfile,$home,$switchserver,$forceedit,$forceview) =
&Apache::lonnet::can_edit_resource($resurl,$cnum,$cdom,$resurl,
$env{'form.symb'});
$editbutton = &get_editbutton($cfile,$home,$switchserver,
$forceedit,$forceview,$forcereg);
} elsif (($resurl !~ m{^/?adm/($match_domain)/($match_username)/aboutme$}) &&
($resurl ne '/cgi-bin/printout.pl')) {
if ($env{'request.filename'}) {
my $file;
my $londocroot = $Apache::lonnet::perlvar{'lonDocRoot'};
if ($env{'request.filename'} =~ m{^\Q$londocroot\E/priv/}) {
$file = $env{'request.filename'};
$file =~ s{^\Q$londocroot\E/}{};
} else {
$file=&Apache::lonnet::declutter($env{'request.filename'});
}
($cfile,$home,$switchserver,$forceedit,$forceview) =
&Apache::lonnet::can_edit_resource($file,$cnum,$cdom,
&Apache::lonnet::clutter($resurl),$env{'request.symb'},$group);
if ($cfile ne '') {
$editbutton = &get_editbutton($cfile,$home,$switchserver,
$forceedit,$forceview,$forcereg,
$hostname);
}
if ((($cfile eq '') || (!$editbutton)) &&
($resurl =~ /$LONCAPA::assess_re/)) {
my $showurl = &Apache::lonnet::clutter($resurl);
my $crs_sec = $env{'request.course.id'} . (($env{'request.course.sec'} ne '')
? "/$env{'request.course.sec'}"
: '');
if ((&Apache::lonnet::allowed('cre','/')) &&
(&Apache::lonnet::metadata($resurl,'sourceavail') eq 'open')) {
$viewsrcbutton = 1;
} elsif (&Apache::lonnet::allowed('vxc',$crs_sec)) {
if ($showurl =~ m{^\Q/res/$cdom/\E($match_username)/}) {
my $auname = $1;
if (($env{'request.course.adhocsrcaccess'} ne '') &&
(grep(/^\Q$auname\E$/,split(/,/,$env{'request.course.adhocsrcaccess'})))) {
$viewsrcbutton = 1;
} elsif ((&Apache::lonnet::metadata($resurl,'sourceavail') eq 'open') &&
(&Apache::lonnet::allowed('bre',$crs_sec))) {
$viewsrcbutton = 1;
}
}
}
if ($viewsrcbutton) {
&switch('','',6,1,'pcstr.png','View Source','resource[_2]','open_source()',
'View source code');
}
}
}
}
}
# End determination of 'Edit' icon/button display
if ($env{'request.course.id'}) {
# This applies to about me page for users in a course
if ($resurl =~ m{^/?adm/($match_domain)/($match_username)/aboutme$}) {
my ($sdom,$sname) = ($1,$2);
unless (&Apache::lonnet::is_course($sdom,$sname)) {
my $blocked = &Apache::loncommon::blocking_status('about',$clientip,$sname,$sdom);
unless ($blocked) {
&switch('','',6,4,'mail-message-new-22x22.png','Message to user',
'',
"go('/adm/email?compose=individual&recname=$sname&recdom=$sdom')",
'Send message to specific user');
}
}
my $hideprivileged = 1;
if (&Apache::lonnet::in_course($sdom,$sname,$cdom,$cnum,undef,
$hideprivileged)) {
foreach my $priv ('vsa','vgr','srm') {
$perms{$priv} = &Apache::lonnet::allowed($priv,$env{'request.course.id'});
if (!$perms{$priv} && $env{'request.course.sec'} ne '') {
$perms{$priv} =
&Apache::lonnet::allowed($priv,"$env{'request.course.id'}/$env{'request.course.sec'}");
}
}
if ($perms{'vsa'}) {
&switch('','',6,5,'trck-22x22.png','Activity',
'',
"go('/adm/trackstudent?selected_student=$sname:$sdom')",
'View recent activity by this person');
}
if ($perms{'vgr'}) {
&switch('','',6,6,'rsrv-22x22.png','Reservations',
'',
"go('/adm/slotrequest?command=showresv&origin=aboutme&uname=$sname&udom=$sdom')",
'Slot reservation history');
}
if ($perms{'srm'}) {
&switch('','',6,7,'contact-new-22x22.png','Records',
'',
"go('/adm/email?recordftf=retrieve&recname=$sname&recdom=$sdom')",
'Add records');
}
}
}
if (($env{'form.folderpath'} =~ /^supplemental/) &&
(&Apache::lonnet::allowed('mdc',$env{'request.course.id'})) &&
(($resurl =~ m{^/adm/wrapper/ext/}) ||
($resurl =~ m{^/adm/$cdom/$cnum/\d+/ext\.tool$}) ||
($resurl =~ m{^/uploaded/$cdom/$cnum/supplemental/}) ||
($resurl eq '/adm/supplemental') ||
($resurl =~ m{^/public/$cdom/$cnum/syllabus$}) ||
($resurl =~ m{^/adm/$match_domain/$match_username/aboutme$}))) {
my @folders=split('&',$env{'form.folderpath'});
if ((@folders > 2) || ($resurl ne '/adm/supplemental')) {
my $suppanchor;
if ($resurl =~ m{^/adm/wrapper/ext/}) {
$suppanchor = $env{'form.anchor'};
}
my $esc_path=&escape(&HTML::Entities::encode(&escape($env{'form.folderpath'}),'<>&"'));
my $link = '/adm/coursedocs?command=direct&forcesupplement=1&supppath='.
"$esc_path&anchor=$suppanchor";
if ($env{'request.use_absolute'} ne '') {
$link = $env{'request.use_absolute'}.$link;
}
&switch('','',7,4,'docs-22x22.png','Edit Folder','parms[_2]',
"location.href='$link'",'Folder/Page Content');
}
}
}
# End checking for items for about me page for users in a course
if ($docscrumbs) {
&Apache::lonhtmlcommon::clear_breadcrumb_tools();
&advtools_crumbs(@inlineremote);
return $editbutton;
} elsif ($env{'request.registered'}) {
return $editbutton || $viewsrcbutton;
} else {
if (ref($bread_crumbs) eq 'ARRAY') {
if (@inlineremote > 0) {
if (ref($advtools) eq 'ARRAY') {
@{$advtools} = @inlineremote;
}
}
return;
} elsif (@inlineremote > 0) {
&Apache::lonhtmlcommon::clear_breadcrumb_tools();
&advtools_crumbs(@inlineremote);
return &Apache::lonhtmlcommon::scripttag('', 'start')
. &Apache::lonhtmlcommon::breadcrumbs(undef,undef,0)
. &Apache::lonhtmlcommon::scripttag('', 'end');
}
}
}
sub advtools_crumbs {
my @funcs = @_;
if ($env{'request.noversionuri'} =~ m{^/adm/$match_domain/$match_username/aboutme$}) {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', @funcs[61,64,65,66,67,74]);
} elsif ($env{'request.noversionuri'} !~ m{^/adm/(navmaps|viewclasslist)(\?|$)}) {
if ($env{'request.state'} eq 'construct') {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', @funcs[61,73,74,71,72,77]);
} else {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', @funcs[61,71,72,73,74,75,92]);
}
} elsif ($env{'request.noversionuri'} eq '/adm/viewclasslist') {
&Apache::lonhtmlcommon::add_breadcrumb_tool(
'advtools', $funcs[61]);
}
return;
}
# ================================================================== Raw Config
sub clear {
my ($row,$col)=@_;
$inlineremote[10*$row+$col]='';
return '';
}
# ============================================ Switch a button or create a link
# Switch acts on the javascript that is executed when a button is clicked.
# The javascript is usually similar to "go('/adm/roles')" or "cstrgo(..)".
sub switch {
my ($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat,$nobreak,$form)=@_;
$act=~s/\$uname/$uname/g;
$act=~s/\$udom/$udom/g;
$top=&mt($top);
$bot=&mt($bot);
$desc=&mt($desc);
my $idx=10*$row+$col;
$category_members{$cat}.=':'.$idx;
# Inline Menu
if ($nobreak==2) { return ''; }
my $text=$top.' '.$bot;
$text=~s/\s*\-\s*//gs;
my $pic=
'<img alt="'.$text.'" src="'.
&Apache::loncommon::lonhttpdurl('/res/adm/pages/'.$img).
'" align="'.($nobreak==3?'right':'left').'" class="LC_icon" />';
if ($env{'browser.interface'} eq 'faketextual') {
# Main Menu
if ($nobreak==3) {
$inlineremote[$idx]="\n".
'<td class="LC_menubuttons_text" align="right">'.$text.
'</td><td align="left">'.
'<a href="javascript:'.$act.';">'.$pic.'</a></td></tr>';
} elsif ($nobreak) {
$inlineremote[$idx]="\n<tr>".
'<td align="left">'.
'<a href="javascript:'.$act.';">'.$pic.'</a></td>
<td class="LC_menubuttons_text" align="left"><a class="LC_menubuttons_link" href="javascript:'.$act.';"><span class="LC_menubuttons_inline_text">'.$text.'</span></a></td>';
} else {
$inlineremote[$idx]="\n<tr>".
'<td align="left">'.
'<a href="javascript:'.$act.';">'.$pic.
'</a></td><td class="LC_menubuttons_text" colspan="3">'.
'<a class="LC_menubuttons_link" href="javascript:'.$act.';"><span class="LC_menubuttons_inline_text">'.$desc.'</span></a></td></tr>';
}
} else {
# Inline Menu
my @tools = (93,91,81,82,83);
unless ($env{'request.state'} eq 'construct') {
push(@tools,63);
}
if ((($env{'environment.icons'} eq 'iconsonly') ||
($env{'environment.icons'} eq '') && ($env{'request.lti.login'})) &&
(grep(/^$idx$/,@tools))) {
$inlineremote[$idx] =
'<a title="'.$desc.'" class="LC_menubuttons_link" href="javascript:'.$act.';">'.$pic.'</a>';
} else {
$inlineremote[$idx] =
'<a title="'.$desc.'" class="LC_menubuttons_link" href="javascript:'.$act.';">'.$pic.
'<span class="LC_menubuttons_inline_text">'.$top.' </span></a>'.$form;
}
}
return '';
}
sub secondlevel {
my $output='';
my
($uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat)=@_;
if ($prt eq 'any') {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
} elsif ($prt=~/^r(\w+)/) {
if ($rol eq $1) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
}
return $output;
}
sub inlinemenu {
undef(@inlineremote);
undef(%category_members);
# calling rawconfig with "1" will evaluate mydesk.tab, even if there is no active remote control
&rawconfig(1);
my $output='<table><tr>';
for (my $col=1; $col<=2; $col++) {
$output.='<td class="LC_mainmenu_col_fieldset">';
for (my $row=1; $row<=8; $row++) {
foreach my $cat (keys(%category_members)) {
if ($category_positions{$cat} ne "$col,$row") { next; }
#$output.='<table><tr><td colspan="4" class="LC_menubuttons_category">'.&mt($category_names{$cat}).'</td></tr>';
$output.='<div class="LC_Box LC_400Box">';
$output.='<h3 class="LC_hcell">'.&mt($category_names{$cat}).'</h3>';
$output.='<table>';
my %active=();
foreach my $menu_item (split(/\:/,$category_members{$cat})) {
if ($inlineremote[$menu_item]) {
$active{$menu_item}=1;
}
}
foreach my $item (sort(keys(%active))) {
$output.=$inlineremote[$item];
}
$output.='</table>';
$output.='</div>';
}
}
$output.="</td>";
}
$output.="</tr></table>";
return $output;
}
sub rawconfig {
#
# This evaluates mydesk.tab
# Need to add more positions and more privileges to deal with all
# menu items.
#
my $textualoverride=shift;
my $output='';
return '' unless $textualoverride;
my $uname=$env{'user.name'};
my $udom=$env{'user.domain'};
my $adv=$env{'user.adv'};
my $show_course=&Apache::loncommon::show_course();
my $author=$env{'user.author'};
my $crs='';
my $crstype='';
if ($env{'request.course.id'}) {
$crs='/'.$env{'request.course.id'};
if ($env{'request.course.sec'}) {
$crs.='_'.$env{'request.course.sec'};
}
$crs=~s/\_/\//g;
$crstype = &Apache::loncommon::course_type();
}
my $pub=($env{'request.state'} eq 'published');
my $con=($env{'request.state'} eq 'construct');
my $rol=$env{'request.role'};
my $requested_domain;
if ($rol) {
$requested_domain = $env{'request.role.domain'};
}
foreach my $line (@desklines) {
my ($row,$col,$pro,$prt,$img,$top,$bot,$act,$desc,$cat)=split(/\:/,$line);
$prt=~s/\$uname/$uname/g;
$prt=~s/\$udom/$udom/g;
if ($prt =~ /\$crs/) {
next unless ($env{'request.course.id'});
next if ($crstype eq 'Community');
$prt=~s/\$crs/$crs/g;
} elsif ($prt =~ /\$cmty/) {
next unless ($env{'request.course.id'});
next if ($crstype ne 'Community');
$prt=~s/\$cmty/$crs/g;
}
if ($prt =~ m/\$requested_domain/) {
if ((!$requested_domain) && ($pro eq 'pbre') && ($env{'user.adv'})) {
$prt=~s/\$requested_domain/$env{'user.domain'}/g;
} else {
$prt=~s/\$requested_domain/$requested_domain/g;
}
}
if ($category_names{$cat}!~/\w/) { $cat='oth'; }
if ($pro eq 'clear') {
$output.=&clear($row,$col);
} elsif ($pro eq 'any') {
$output.=&secondlevel(
$uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat);
} elsif ($pro eq 'smp') {
unless ($adv) {
$output.=&secondlevel(
$uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro eq 'adv') {
if ($adv) {
$output.=&secondlevel(
$uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro eq 'shc') {
if ($show_course) {
$output.=&secondlevel(
$uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro eq 'nsc') {
if (!$show_course) {
$output.=&secondlevel(
$uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat);
}
} elsif (($pro=~/^p(\w+)/) && ($prt)) {
my $priv = $1;
if ($priv =~ /^mdc(Course|Community)/) {
if ($crstype eq $1) {
$priv = 'mdc';
} else {
next;
}
} elsif ($priv eq 'cca') {
next if ($rol eq 'cm');
}
if ((($priv eq 'bre') && (&Apache::lonnet::allowed($priv,$prt) eq 'F')) ||
(($priv ne 'bre') && (&Apache::lonnet::allowed($priv,$prt)))) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro eq 'course') {
if (($env{'request.course.fn'}) && ($crstype ne 'Community')) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro eq 'community') {
if (($env{'request.course.fn'}) && ($crstype eq 'Community')) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro =~ /^courseenv_(.*)$/) {
my $key = $1;
if ($crstype ne 'Community') {
my $coursepref = $env{'course.'.$env{'request.course.id'}.'.'.$key};
if ($key eq 'canuse_pdfforms') {
if ($env{'request.course.id'} && $coursepref eq '') {
my %domdefs = &Apache::lonnet::get_domain_defaults($env{'course.'.$env{'request.course.id'}.'.domain'});
$coursepref = $domdefs{'canuse_pdfforms'};
}
}
if ($coursepref) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
}
} elsif ($pro =~ /^communityenv_(.*)$/) {
my $key = $1;
if ($crstype eq 'Community') {
my $coursepref = $env{'course.'.$env{'request.course.id'}.'.'.$key};
if ($key eq 'canuse_pdfforms') {
if ($env{'request.course.id'} && $coursepref eq '') {
my %domdefs = &Apache::lonnet::get_domain_defaults($env{'course.'.$env{'request.course.id'}.'.domain'});
$coursepref = $domdefs{'canuse_pdfforms'};
}
}
if ($coursepref) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
}
} elsif ($pro =~ /^course_(.*)$/) {
# Check for permissions inside of a course
if (($env{'request.course.id'}) && ($crstype ne 'Community') &&
(&Apache::lonnet::allowed($1,$env{'request.course.id'}.
($env{'request.course.sec'}?'/'.$env{'request.course.sec'}:''))
)) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro =~ /^community_(.*)$/) {
# Check for permissions inside of a community
if (($env{'request.course.id'}) && ($crstype eq 'Community') &&
(&Apache::lonnet::allowed($1,$env{'request.course.id'}.
($env{'request.course.sec'}?'/'.$env{'request.course.sec'}:''))
)) {
$output.=&switch($uname,$udom,$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
} elsif ($pro eq 'author') {
if ($author) {
if ((($prt eq 'rca') && ($env{'request.role'}=~/^ca/)) ||
(($prt eq 'raa') && ($env{'request.role'}=~/^aa/)) ||
(($prt eq 'rau') && ($env{'request.role'}=~/^au/))) {
# Check that we are on the correct machine
my $cadom=$requested_domain;
my $caname=$env{'user.name'};
if (($prt eq 'rca') || ($prt eq 'raa')) {
($cadom,$caname)=
($env{'request.role'}=~/($match_domain)\/($match_username)$/);
}
$act =~ s/\$caname/$caname/g;
$act =~ s/\$cadom/$cadom/g;
my $home = &Apache::lonnet::homeserver($caname,$cadom);
my $allowed=0;
my @ids=&Apache::lonnet::current_machine_ids();
foreach my $id (@ids) { if ($id eq $home) { $allowed=1; } }
if ($allowed) {
$output.=&switch($caname,$cadom,
$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
}
}
} elsif ($pro eq 'coauthor') {
if ($env{'request.role'}=~ m{^(ca|aa)\./($match_domain)/($match_username)$}) {
my ($role,$audom,$auname) = ($1,$2,$3);
if ((($prt eq 'raa') && ($role eq 'aa')) ||
(($prt eq 'rca') && ($role eq 'ca') &&
(!$env{"environment.internal.manager./$audom/$auname"}))) {
$output.=&switch($auname,$audom,
$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
}
} elsif ($pro eq 'coauthorenv_manager') {
if ($env{'request.role'}=~ m{^ca\./($match_domain)/($match_username)$}) {
my ($audom,$auname) = ($1,$2);
if ($env{"environment.internal.manager./$audom/$auname"}) {
$output.=&switch($auname,$audom,
$row,$col,$img,$top,$bot,$act,$desc,$cat);
}
}
} elsif ($pro eq 'tools') {
my @tools = ('aboutme','blog','portfolio');
if (grep(/^\Q$prt\E$/,@tools)) {
if (!&Apache::lonnet::usertools_access($env{'user.name'},
$env{'user.domain'},
$prt,undef,'tools')) {
$output.=&clear($row,$col);
next;
}
} elsif (($prt eq 'reqcrsnsc') || ($prt eq 'reqcrsshc')) {
if (($prt eq 'reqcrsnsc') && ($show_course)) {
next;
}
if (($prt eq 'reqcrsshc') && (!$show_course)) {
next;
}
my $showreqcrs = &check_for_rcrs();
if (!$showreqcrs) {
$output.=&clear($row,$col);
next;
}
}
$prt='any';
$output.=&secondlevel(
$uname,$udom,$rol,$crs,$pub,$con,$row,$col,$prt,$img,$top,$bot,$act,$desc,$cat);
}
}
return $output;
}
sub check_for_rcrs {
my $showreqcrs = 0;
my @reqtypes = ('official','unofficial','community','textbook','placement');
foreach my $type (@reqtypes) {
if (&Apache::lonnet::usertools_access($env{'user.name'},
$env{'user.domain'},
$type,undef,'requestcourses')) {
$showreqcrs = 1;
last;
}
}
if (!$showreqcrs) {
foreach my $type (@reqtypes) {
if ($env{'environment.reqcrsotherdom.'.$type} ne '') {
my @domains = split(',',$env{'environment.reqcrsotherdom.'.$type});
foreach my $entry (@domains) {
my ($extdom,$extopt) = split(':',$entry);
if (&Apache::lonnet::will_trust('reqcrs',$env{'user.domain'},$extdom)) {
$showreqcrs = 1;
last;
}
}
if ($showreqcrs) {
last;
}
}
}
}
return $showreqcrs;
}
sub dc_popup_js {
my %lt = &Apache::lonlocal::texthash(
more => '(More ...)',
less => '(Less ...)',
);
return <<"END";
function showCourseID() {
document.getElementById('dccid').style.display='block';
document.getElementById('dccid').style.textAlign='left';
document.getElementById('dccid').style.textFace='normal';
document.getElementById('dccidtext').innerHTML ='<a href="javascript:hideCourseID();" class="LC_menubuttons_link">$lt{'less'}</a>';
return;
}
function hideCourseID() {
document.getElementById('dccid').style.display='none';
document.getElementById('dccidtext').innerHTML ='<a href="javascript:showCourseID()" class="LC_menubuttons_link">$lt{'more'}</a>';
return;
}
END
}
sub countdown_toggle_js {
return <<"END";
function toggleCountdown() {
var countdownid = document.getElementById('duedatecountdown');
var currstyle = countdownid.style.display;
if (currstyle == 'inline') {
countdownid.style.display = 'none';
document.getElementById('ddcountcollapse').innerHTML='';
document.getElementById('ddcountexpand').innerHTML='◄ ';
} else {
countdownid.style.display = 'inline';
document.getElementById('ddcountcollapse').innerHTML='► ';
document.getElementById('ddcountexpand').innerHTML='';
}
return;
}
END
}
# This creates a "done button" for timed events. The confirmation box is a jQuery
# dialog widget. If the interval parameter requires a proctor key for the timed
# event to be marked done, there will also be a textbox where that can be entered.
# Clicking OK will set the value of LC_interval_done to 'true', and, if needed will
# set the value of LC_interval_done_proctorpass to the text entered in that box,
# and submit the corresponding form.
#
# The &zero_time() routine in lonhomework.pm is called when a page is rendered if
# LC_interval_done is true.
#
sub done_button_js {
my ($type,$width,$height,$proctor,$donebuttontext) = @_;
return unless (($type eq 'map') || ($type eq 'resource'));
my %lt = &Apache::lonlocal::texthash(
title => 'WARNING!',
preamble => 'You are trying to end this timed event early.',
map => 'Confirming that you are done will cause the time to expire and prevent you from changing any answers in the current folder.',
resource => 'Confirming that you are done will cause the time to expire for this question, and prevent you from changing your answer(s).',
okdone => 'Click "OK" if you are completely finished.',
cancel => 'Click "Cancel" to continue working.',
proctor => 'Ask a proctor to enter the key, then click "OK" if you are completely finished.',
ok => 'OK',
exit => 'Cancel',
key => 'Key:',
nokey => 'A proctor key is required',
);
my $shownsymb = &HTML::Entities::encode(&Apache::lonenc::check_encrypt($env{'request.symb'}));
my $navmap = Apache::lonnavmaps::navmap->new();
my ($missing,$tried) = (0,0);
if (ref($navmap)) {
my @resources=();
if ($type eq 'map') {
my ($mapurl,$rid,$resurl)=&Apache::lonnet::decode_symb($env{'request.symb'});
if ($env{'request.symb'} =~ /\.page$/) {
@resources=$navmap->retrieveResources($resurl,sub { $_[0]->is_problem() });
} else {
@resources=$navmap->retrieveResources($mapurl,sub { $_[0]->is_problem() });
}
} else {
my $res = $navmap->getBySymb($env{'request.symb'});
if (ref($res)) {
if ($res->is_problem()) {
push(@resources,$res);
}
}
}
foreach my $res (@resources) {
if (ref($res->parts()) eq 'ARRAY') {
foreach my $part (@{$res->parts()}) {
if (!$res->tries($part)) {
$missing++;
} else {
$tried++;
}
}
}
}
}
if ($missing) {
$lt{'miss'} .= '<p class="LC_error">';
if ($type eq 'map') {
$lt{'miss'} .= &mt('Submissions are missing for [quant,_1,question part,question parts] in this folder.',$missing);
} else {
$lt{'miss'} .= &mt('Submissions are missing for [quant,_1,part] in this question.',$missing);
}
if ($missing > 1) {
$lt{'miss'} .= ' '.&mt('If you confirm you are done you will be unable to submit answers for them.').'</span>';
} else {
$lt{'miss'} .= ' '.&mt('If you confirm you are done you will be unable to submit an answer for it.').'</p>';
}
}
$donebuttontext = &HTML::Entities::encode($donebuttontext,'<>&"');
if ($proctor) {
if ($height !~ /^\d+$/) {
$height = 400;
if ($missing) {
$height += 60;
}
}
if ($width !~ /^\d+$/) {
$width = 400;
if ($missing) {
$width += 60;
}
}
return <<END;
<form method="post" name="LCdoneButton" action="">
<input type="hidden" name="LC_interval_done" value="" />
<input type="hidden" name="LC_interval_done_proctorpass" value="" />
<input type="hidden" name="symb" value="$shownsymb" />
<button id="LC_done-confirm-opener" type="button">$donebuttontext</button>
</form>
<div id="LC_done-confirm" title="$lt{'title'}">
<p>$lt{'preamble'} $lt{$type}</p>
$lt{'miss'}
<p>$lt{'proctor'}</p>
<form name="LCdoneButtonProctor" action="">
<label>$lt{'key'}<input type="password" name="LC_interval_done_proctorkey" value="" /></label>
<input type="submit" tabindex="-1" style="position:absolute; top:-1000px" />
</form>
<p>$lt{'cancel'}</p>
</div>
<script type="text/javascript">
// <![CDATA[
\$( "#LC_done-confirm" ).dialog({ autoOpen: false });
\$( "#LC_done-confirm-opener" ).on("click", function() {
\$( "#LC_done-confirm" ).dialog("open");
\$( "#LC_done-confirm" ).dialog({
height: $height,
width: $width,
modal: true,
resizable: false,
buttons: [
{
text: "$lt{'ok'}",
click: function() {
var proctorkey = \$( '[name="LC_interval_done_proctorkey"]' )[0].value;
if ((proctorkey == '') || (proctorkey == null)) {
alert("$lt{'nokey'}");
} else {
\$( '[name="LC_interval_done"]' )[0].value = 'true';
\$( '[name="LC_interval_done_proctorpass"]' )[0].value = proctorkey;
\$( '[name="LCdoneButton"]' )[0].submit();
}
},
},
{
text: "$lt{'exit'}",
click: function() {
\$("#LC_done-confirm").dialog( "close" );
}
}
],
close: function() {
\$( '[name="LC_interval_done_proctorkey"]' )[0].value = '';
}
});
\$( "#LC_done-confirm" ).find( "form" ).on( "submit", function( event ) {
event.preventDefault();
\$( '[name="LC_interval_done"]' )[0].value = 'true';
\$( '[name="LC_interval_done_proctorpass"]' )[0].value = \$( '[name="LC_interval_done_proctorkey"]' )[0].value;
\$( '[name="LCdoneButton"]' )[0].submit();
});
});
// ]]>
</script>
END
} else {
if ($height !~ /^\d+$/) {
$height = 320;
if ($missing) {
$height += 60;
}
}
if ($width !~ /^\d+$/) {
$width = 320;
if ($missing) {
$width += 60;
}
}
if ($missing) {
$lt{'miss'} = '</p>'.$lt{'miss'}.'<p>';
}
return <<END;
<form method="post" name="LCdoneButton" action="">
<input type="hidden" name="LC_interval_done" value="" />
<input type="hidden" name="symb" value="$shownsymb" />
<button id="LC_done-confirm-opener" type="button">$donebuttontext</button>
</form>
<div id="LC_done-confirm" title="$lt{'title'}">
<p>$lt{'preamble'} $lt{$type} $lt{'miss'} $lt{'okdone'} $lt{'cancel'}</p>
</div>
<script type="text/javascript">
// <![CDATA[
\$( "#LC_done-confirm" ).dialog({ autoOpen: false });
\$( "#LC_done-confirm-opener" ).click(function() {
\$( "#LC_done-confirm" ).dialog( "open" );
\$( "#LC_done-confirm" ).dialog({
resizable: false,
height: $height,
width: $width,
modal: true,
buttons: [
{
text: "$lt{'ok'}",
click: function() {
\$( this ).dialog( "close" );
\$( '[name="LC_interval_done"]' )[0].value = 'true';
\$( '[name="LCdoneButton"]' )[0].submit();
},
},
{
text: "$lt{'exit'}",
click: function() {
\$( this ).dialog( "close" );
},
},
],
});
});
// ]]>
</script>
END
}
}
sub view_as_js {
my ($url,$symb) = @_;
my %lt = &Apache::lonlocal::texthash(
ente => 'Enter a username or a student/employee ID',
info => 'Information you entered does not match a valid course user',
);
&js_escape(\%lt);
return <<"END";
function toggleViewAsUser(change) {
if (document.getElementById('LC_selectuser')) {
var seluserid = document.getElementById('LC_selectuser');
var currstyle = seluserid.style.display;
if (change == 'off') {
document.userview.elements['LC_viewas'].value = '';
document.userview.elements['vuname'].value = '';
document.userview.elements['vid'].value = '';
document.userview.submit();
return;
}
if ((document.getElementById('usexpand')) && (document.getElementById('uscollapse'))) {
if (currstyle == 'inline') {
seluserid.style.display = 'none';
document.getElementById('usexpand').innerHTML='► ';
document.getElementById('uscollapse').innerHTML='';
} else {
seluserid.style.display = 'inline';
document.getElementById('usexpand').innerHTML='';
document.getElementById('uscollapse').innerHTML='◄ ';
toggleIdentifier(document.userview);
}
}
}
return;
}
function validCourseUser(form,change) {
var possuname = form.elements['vuname'].value;
var possuid = form.elements['vid'].value;
var domelem = form.elements['vudom'];
var possudom = '';
if ((domelem.tagName === 'INPUT') && ((domelem.type === 'text') || (domelem.type === 'hidden'))) {
possudom = domelem.value;
} else if (domelem.tagName === 'SELECT') {
possudom = domelem.options[domelem.selectedIndex].value;
}
if ((possuname == '') && (possuid == '')) {
if (change == 'off') {
form.elements['LC_viewas'].value = '';
form.submit();
} else {
alert("$lt{'ente'}");
}
return;
}
var http = new XMLHttpRequest();
var url = "/adm/courseuser";
var params = "uname="+possuname+"&uid="+possuid+"&udom="+possudom;
http.open("POST", url, true);
http.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
http.onreadystatechange = function() {
if (http.readyState == 4 && http.status == 200) {
var data = JSON.parse(http.responseText);
if (Array.isArray(data.match)) {
var len = data.match.length;
if (len == 2) {
if (data.match[0] != '' && data.match[1] != '') {
form.elements['LC_viewas'].value = data.match[0]+':'+data.match[1];
form.submit();
}
} else {
alert("$lt{'info'}");
}
}
}
return;
}
http.send(params);
return false;
}
function toggleIdentifier(form) {
if ((document.getElementById('LC_vuname')) && (document.getElementById('LC_vid'))) {
var radioelem = form.elements['vuidentifier'];
if (radioelem.length > 0) {
var i;
for (i=0; i<radioelem.length; i++) {
if (radioelem[i].checked == true) {
if (radioelem[i].value == 'uname') {
document.getElementById('LC_vuname').type = 'text';
document.getElementById('LC_vid').type = 'hidden';
document.getElementById('LC_vid').value = '';
} else {
document.getElementById('LC_vuname').type = 'hidden';
document.getElementById('LC_vuname').value = '';
document.getElementById('LC_vid').type = 'text';
}
break;
}
}
}
}
return;
}
END
}
sub utilityfunctions {
my ($httphost) = @_;
my $currenturl=&Apache::lonnet::clutter(&Apache::lonnet::fixversion((split(/\?/,$env{'request.noversionuri'}))[0]));
my $currentsymb=&Apache::lonenc::check_encrypt($env{'request.symb'});
if ($currenturl =~ m{^/adm/wrapper/ext/}) {
if ($env{'request.external.querystring'}) {
$currenturl .= ($currenturl=~/\?/)?'&':'?'.$env{'request.external.querystring'};
}
my ($anchor) = ($env{'request.symb'} =~ /(\#[^\#]+)$/);
if (($anchor) && ($currenturl !~ /\Q$anchor\E$/)) {
$currenturl .= $1;
}
}
$currenturl=&Apache::lonenc::check_encrypt(&unescape($currenturl));
my $dc_popup_cid;
if ($env{'user.adv'} && exists($env{'user.role.dc./'.
$env{'course.'.$env{'request.course.id'}.
'.domain'}.'/'})) {
$dc_popup_cid = &dc_popup_js();
}
my $start_page_annotate =
&Apache::loncommon::start_page('Annotator',undef,
{'only_body' => 1,
'js_ready' => 1,
'bgcolor' => '#BBBBBB',
'add_entries' => {
'onload' => 'javascript:document.goannotate.submit();'}});
my $end_page_annotate =
&Apache::loncommon::end_page({'js_ready' => 1});
my $jumptores = &Apache::lonhtmlcommon::javascript_jumpto_resource();
my $esc_url=&escape($currenturl);
my $esc_symb=&escape($currentsymb);
my $newname = &mt('New Name');
&js_escape(\$newname);
my $countdown = &countdown_toggle_js();
my $viewuser;
if (($env{'request.course.id'}) &&
($env{'request.symb'} ne '') &&
($env{'request.filename'}=~/$LONCAPA::assess_re/)) {
my $canview;
foreach my $priv ('mgr','vgr') {
$canview = &Apache::lonnet::allowed($priv,$env{'request.course.id'});
if (!$canview && $env{'request.course.sec'} ne '') {
$canview =
&Apache::lonnet::allowed($priv,"$env{'request.course.id'}/$env{'request.course.sec'}");
}
last if ($canview);
}
if ($canview) {
$viewuser = &view_as_js($esc_url,$esc_symb);
}
}
my ($ltitarget,$deeplinktarget);
if ($env{'request.lti.login'}) {
$ltitarget = $env{'request.lti.target'};
}
if ($env{'request.deeplink.login'}) {
$deeplinktarget = $env{'request.deeplink.target'};
}
my $annotateurl = '/adm/annotation';
if ($httphost) {
$annotateurl = '/adm/annotations';
}
my $hostvar = '
function setLCHost() {
var lcHostname="";
';
if ($httphost =~ m{^https?\://}) {
$hostvar .= ' var lcServer="'.$httphost.'";'."\n".
' var hostReg = /^https?:\/\/([^\/]+)$/i;'."\n".
' var match = hostReg.exec(lcServer);'."\n".
' if (match.length) {'."\n".
' if (match[1] == location.hostname) {'."\n".
' lcHostname=lcServer;'."\n".
' }'."\n".
' }'."\n";
}
$hostvar .= ' return lcHostname;'."\n".
'}'."\n";
return (<<ENDUTILITY)
$hostvar
var currentURL=unescape("$esc_url");
var reloadURL=unescape("$esc_url");
var currentSymb=unescape("$esc_symb");
$dc_popup_cid
$jumptores
function gopost(url,postdata) {
if (url!='') {
var lcHostname = setLCHost();
this.document.server.action=lcHostname+url;
this.document.server.postdata.value=postdata;
this.document.server.command.value='';
this.document.server.url.value='';
this.document.server.symb.value='';
this.document.server.submit();
}
}
function gocmd(url,cmd) {
if (url!='') {
var lcHostname = setLCHost();
this.document.server.action=lcHostname+url;
this.document.server.postdata.value='';
this.document.server.command.value=cmd;
this.document.server.url.value=currentURL;
this.document.server.symb.value=currentSymb;
this.document.server.submit();
}
}
function gocstr(url,filename) {
if (url == '/adm/cfile?action=delete') {
this.document.cstrdelete.filename.value = filename
this.document.cstrdelete.submit();
return;
}
if ((url == '/adm/cfile?action=copy') || (url == '/adm/cfile?action=rename')) {
this.document.cstrcopy.filename.value = filename;
var oldname = filename.substring(filename.lastIndexOf("/") + 1);
var newname=prompt('$newname',oldname);
if (newname == "" || !newname || newname == oldname) {
return;
}
if (url == '/adm/cfile?action=rename') {
this.document.cstrcopy.action.value = 'rename';
} else {
this.document.cstrcopy.action.value = 'copy';
}
this.document.cstrcopy.newfilename.value = newname;
this.document.cstrcopy.submit();
return;
}
if (url == '/adm/printout') {
this.document.cstrprint.postdata.value = filename
this.document.cstrprint.curseed.value = 0;
this.document.cstrprint.problemtype.value = 0;
if (this.document.lonhomework) {
if ((this.document.lonhomework.rndseed) && (this.document.lonhomework.rndseed.value != null) && (this.document.lonhomework.rndseed.value != '')) {
this.document.cstrprint.curseed.value = this.document.lonhomework.rndseed.value
}
if (this.document.lonhomework.problemtype) {
if (this.document.lonhomework.problemtype.value) {
this.document.cstrprint.problemtype.value =
this.document.lonhomework.problemtype.value;
} else if (this.document.lonhomework.problemtype.options) {
for (var i=0; i<this.document.lonhomework.problemtype.options.length; i++) {
if (this.document.lonhomework.problemtype.options[i].selected) {
if (this.document.lonhomework.problemtype.options[i].value != null && this.document.lonhomework.problemtype.options[i].value != '') {
this.document.cstrprint.problemtype.value = this.document.lonhomework.problemtype.options[i].value
}
}
}
}
}
}
this.document.cstrprint.submit();
return;
}
if (url == '/adm/preferences?action=authorsettings') {
document.location.href=url+'&returnurl='+filename;
return;
}
if (url !='') {
this.document.constspace.filename.value = filename;
this.document.constspace.action = url;
this.document.constspace.submit();
}
}
function golist(url) {
if (url!='' && url!= null) {
currentURL = null;
currentSymb= null;
var lcHostname = setLCHost();
var ltitarget = '$ltitarget';
var deeplinktarget = '$deeplinktarget';
if ((ltitarget == 'iframe') || (deeplinktarget == '_self')) {
document.location.href=lcHostname+url;
} else {
top.location.href=lcHostname+url;
}
}
}
function catalog_info(url,isMobile) {
if (isMobile == 1) {
openMyModal(url+'.meta?modal=1',500,400,'yes');
} else {
loncatinfo=window.open(url+'.meta',"LONcatInfo",'height=500,width=400,resizable=yes,scrollbars=yes,location=no,menubar=no,toolbar=no');
}
}
function chat_win() {
var lcHostname = setLCHost();
lonchat=window.open(lcHostname+'/res/adm/pages/chatroom.html',"LONchat",'height=320,width=480,resizable=yes,location=no,menubar=no,toolbar=no');
}
function group_chat(group) {
var lcHostname = setLCHost();
var url = lcHostname+'/adm/groupchat?group='+group;
var winName = 'LONchat_'+group;
grpchat=window.open(url,winName,'height=320,width=280,resizable=yes,location=no,menubar=no,toolbar=no');
}
function annotate() {
w_Annotator_flag=1;
annotator=window.open('','Annotator','width=365,height=265,scrollbars=0');
annotator.document.write(
'$start_page_annotate'
+"<form name='goannotate' target='Annotator' method='post' "
+"action='$annotateurl'>"
+"<input type='hidden' name='symbnew' value='"+currentSymb+"' />"
+"<\\/form>"
+'$end_page_annotate');
annotator.document.close();
}
function open_StoredLinks_Import(rat) {
var newWin;
var lcHostname = setLCHost();
if (rat) {
newWin = window.open(lcHostname+'/adm/wishlist?inhibitmenu=yes&mode=import&rat='+rat,
'wishlistImport','scrollbars=1,resizable=1,menubar=0');
}
else {
newWin = window.open(lcHostname+'/adm/wishlist?inhibitmenu=yes&mode=import',
'wishlistImport','scrollbars=1,resizable=1,menubar=0');
}
newWin.focus();
}
function open_source() {
sourcewin=window.open('/adm/source?inhibitmenu=yes&viewonly=1&filename='+currentURL,'LONsource',
'height=500,width=600,resizable=yes,location=no,menubar=no,toolbar=no,scrollbars=yes');
}
function open_aboutLC() {
var isMobile = "$env{'browser.mobile'}";
var url = '/adm/about.html';
if (isMobile == 1) {
openMyModal(url,600,400,'yes');
} else {
window.open(url,"aboutLONCAPA","height=400,width=600,scrollbars=1,resizable=1,menubar=0,location=1");
}
return;
}
(function (\$) {
\$(document).ready(function () {
\$.single=function(a){return function(b){a[0]=b;return a}}(\$([1]));
/*\@cc_on
if (!window.XMLHttpRequest) {
\$('.LC_hoverable').each(function () {
this.attachEvent('onmouseenter', function (evt) { \$.single(evt.srcElement).addClass('hover'); });
this.attachEvent('onmouseleave', function (evt) { \$.single(evt.srcElement).removeClass('hover'); });
});
}
\@*/
});
}(jQuery));
$countdown
$viewuser
ENDUTILITY
}
sub serverform {
my $target;
unless (($env{'request.lti.login'}) && ($env{'request.lti.target'} eq 'iframe')) {
$target = ' target="_top"';
}
if (($env{'request.deeplink.login'}) && ($env{'request.deeplink.target'} eq '_self')) {
$target = ' target="_self"';
}
my $submit = &mt('Submit');
return(<<ENDSERVERFORM);
<form name="server" action="/adm/logout" method="post"$target aria-hidden="true">
<input type="hidden" name="postdata" value="none" />
<input type="hidden" name="command" value="none" />
<input type="hidden" name="url" value="none" />
<input type="hidden" name="symb" value="none" />
<input type="submit" value="$submit" class="LC_visually_hidden" tabindex="-1" disabled="disabled" />
</form>
ENDSERVERFORM
}
sub constspaceform {
my ($frameset) = @_;
my ($target,$printtarget);
if ($frameset) {
$target = ' target="_parent"';
$printtarget = ' target="_parent"';
} else {
unless ((($env{'request.lti.login'}) && ($env{'request.lti.target'} eq 'iframe')) ||
(($env{'request.deeplink.login'}) && ($env{'request.deeplink.target'} eq '_self'))) {
$target = ' target="_top"';
$printtarget = ' target="_top"';
}
}
return(<<ENDCONSTSPACEFORM);
<form name="constspace" action="/adm/logout" method="post"$target>
<input type="hidden" name="filename" value="" />
</form>
<form name="cstrdelete" action="/adm/cfile" method="post"$target>
<input type="hidden" name="action" value="delete" />
<input type="hidden" name="filename" value="" />
</form>
<form name="cstrprint" action="/adm/printout" method="post"$printtarget>
<input type="hidden" name="postdata" value="" />
<input type="hidden" name="curseed" value="" />
<input type="hidden" name="problemtype" value="" />
</form>
<form name="cstrcopy" action="/adm/cfile" method="post"$target>
<input type="hidden" name="action" value="copy" />
<input type="hidden" name="filename" value="" />
<input type="hidden" name="newfilename" value="" />
</form>
ENDCONSTSPACEFORM
}
sub hidden_button_check {
if ( $env{'request.course.id'} eq ''
|| $env{'request.role.adv'} ) {
return;
}
my $buttonshide = &Apache::lonnet::EXT('resource.0.buttonshide');
return $buttonshide;
}
sub roles_selector {
my ($cdom,$cnum,$httphost,$target,$menucoll,$menuref) = @_;
my $crstype = &Apache::loncommon::course_type();
my $now = time;
my (%courseroles,%seccount,%courseprivs,%roledesc);
my $is_cc;
my ($js,$form,$switcher,$has_opa_priv);
my $ccrole;
if ($crstype eq 'Community') {
$ccrole = 'co';
} else {
$ccrole = 'cc';
}
my ($privref,$gotsymb,$destsymb);
my $destinationurl = $ENV{'REQUEST_URI'};
if ($destinationurl =~ /(\?|\&)symb=/) {
$gotsymb = 1;
} elsif ($destinationurl =~ m{^/enc/}) {
my $plainurl = &Apache::lonenc::unencrypted($destinationurl);
if ($plainurl =~ /(\?|\&)symb=/) {
$gotsymb = 1;
}
}
unless ($gotsymb) {
$destsymb = &Apache::lonnet::symbread();
if ($destsymb ne '') {
$destsymb = &Apache::lonenc::check_encrypt($destsymb);
}
}
my $reqprivs = &required_privs();
if (ref($reqprivs) eq 'HASH') {
my $destination = $destinationurl;
$destination =~ s/(\?.*)$//;
if (exists($reqprivs->{$destination})) {
if ($reqprivs->{$destination} =~ /,/) {
@{$privref} = split(/,/,$reqprivs->{$destination});
} else {
$privref = [$reqprivs->{$destination}];
}
}
}
if ($env{'user.role.'.$ccrole.'./'.$cdom.'/'.$cnum}) {
my ($start,$end) = split(/\./,$env{'user.role.'.$ccrole.'./'.$cdom.'/'.$cnum});
if ((($start) && ($start<0)) ||
(($end) && ($end<$now)) ||
(($start) && ($now<$start))) {
$is_cc = 0;
} else {
$is_cc = 1;
}
}
if ($is_cc) {
&get_all_courseroles($cdom,$cnum,\%courseroles,\%seccount,\%courseprivs);
} elsif ($env{'request.role'} =~ m{^\Qcr/$cdom/$cdom-domainconfig/\E(\w+)\.\Q/$cdom/$cnum\E}) {
&get_customadhoc_roles($cdom,$cnum,\%courseroles,\%seccount,\%courseprivs,\%roledesc,$privref);
} else {
my %gotnosection;
foreach my $item (keys(%env)) {
if ($item =~ m-^user\.role\.([^.]+)\./\Q$cdom\E/\Q$cnum\E/?(\w*)$-) {
my $role = $1;
my $sec = $2;
next if ($role eq 'gr');
my ($start,$end) = split(/\./,$env{$item});
next if (($start && $start > $now) || ($end && $end < $now));
if ($sec eq '') {
if (!$gotnosection{$role}) {
$seccount{$role} ++;
$gotnosection{$role} = 1;
}
}
if ((ref($privref) eq 'ARRAY') && (@{$privref} > 0)) {
my $cnumsec = $cnum;
if ($sec ne '') {
$cnumsec .= "/$sec";
}
$courseprivs{"$role./$cdom/$cnumsec./"} =
$env{"user.priv.$role./$cdom/$cnumsec./"};
$courseprivs{"$role./$cdom/$cnumsec./$cdom/"} =
$env{"user.priv.$role./$cdom/$cnumsec./$cdom/"};
$courseprivs{"$role./$cdom/$cnumsec./$cdom/$cnumsec"} =
$env{"user.priv.$role./$cdom/$cnumsec./$cdom/$cnumsec"};
}
if (ref($courseroles{$role}) eq 'ARRAY') {
if ($sec ne '') {
if (!grep(/^\Q$sec\E$/,@{$courseroles{$role}})) {
push(@{$courseroles{$role}},$sec);
$seccount{$role} ++;
}
}
} else {
@{$courseroles{$role}} = ();
if ($sec ne '') {
$seccount{$role} ++;
push(@{$courseroles{$role}},$sec);
}
}
}
}
}
my @roles_order = ($ccrole,'in','ta','ep','ad','st');
my $numdiffsec;
if (keys(%seccount) == 1) {
foreach my $key (keys(%seccount)) {
$numdiffsec = $seccount{$key};
}
}
if ((keys(%seccount) > 1) || ($numdiffsec > 1)) {
my $targetattr;
if ($target ne '') {
$targetattr = ' target="'.$target.'"';
}
my @submenu;
$js = &jump_to_role($cdom,$cnum,\%seccount,\%courseroles,\%courseprivs,
\%roledesc,$privref,$menucoll,$menuref);
$form =
'<form name="rolechooser" method="post" action="'.$httphost.'/adm/roles"'.$targetattr.' aria-hidden="true">'."\n".
' <input type="hidden" name="destinationurl" value="'.
&HTML::Entities::encode($destinationurl).'" />'."\n".
' <input type="hidden" name="gotorole" value="1" />'."\n".
' <input type="hidden" name="selectrole" value="" />'."\n".
' <input type="hidden" name="switchrole" value="" />'."\n";
if ($destsymb ne '') {
$form .= ' <input type="hidden" name="destsymb" value="'.
&HTML::Entities::encode($destsymb).'" />'."\n";
}
$form .= '<input type="submit" value="'.&mt('Submit').'" class="LC_visually_hidden" tabindex="-1" disabled="disabled" />'."\n".
'</form>'."\n";
foreach my $role (@roles_order) {
my $include;
if (defined($courseroles{$role})) {
if ($env{'request.role'} =~ m{^\Q$role\E}) {
if ($seccount{$role} > 1) {
$include = 1;
} else {
if ($env{'user.priv.'.$env{'request.role'}."./$cdom/$cnum"} =~/opa\&([^\:]*)/) {
$has_opa_priv = 1;
}
}
} else {
$include = 1;
}
}
if ($include) {
if ($env{"user.priv.$role./$cdom/$cnum./$cdom/$cnum"} =~/opa\&([^\:]*)/) {
$has_opa_priv = 1;
}
push(@submenu,['javascript:adhocRole('."'$role'".')',
&Apache::lonnet::plaintext($role,$crstype)]);
}
}
foreach my $role (sort(keys(%courseroles))) {
if ($role =~ /^cr/) {
my $include;
if ($env{'request.role'} =~ m{^\Q$role\E}) {
if ($seccount{$role} > 1) {
$include = 1;
}
} else {
$include = 1;
}
if ($include) {
my $rolename;
if ($role =~ m{^cr/$cdom/$cdom\-domainconfig/(\w+)(?:/\w+|$)}) {
$rolename = $roledesc{$role};
if ($rolename eq '') {
$rolename = &mt('Helpdesk [_1]',$1);
}
} else {
$rolename = &Apache::lonnet::plaintext($role);
}
if ($env{"user.priv.$role./$cdom/$cnum./$cdom/$cnum"} =~/opa\&([^\:]*)/) {
$has_opa_priv = 1;
}
push(@submenu,['javascript:adhocRole('."'$role'".')',
$rolename]);
}
}
}
if (@submenu > 0) {
$switcher = &create_submenu('#',$target,&mt('Switch role'),\@submenu);
}
}
return ($js,$form,$switcher,$has_opa_priv);
}
sub get_all_courseroles {
my ($cdom,$cnum,$courseroles,$seccount,$courseprivs) = @_;
unless ((ref($courseroles) eq 'HASH') && (ref($seccount) eq 'HASH') &&
(ref($courseprivs) eq 'HASH')) {
return;
}
my ($result,$cached) =
&Apache::lonnet::is_cached_new('getcourseroles',$cdom.'_'.$cnum);
if (defined($cached)) {
if (ref($result) eq 'HASH') {
if ((ref($result->{'roles'}) eq 'HASH') &&
(ref($result->{'seccount'}) eq 'HASH') &&
(ref($result->{'privs'}) eq 'HASH')) {
%{$courseroles} = %{$result->{'roles'}};
%{$seccount} = %{$result->{'seccount'}};
%{$courseprivs} = %{$result->{'privs'}};
return;
}
}
}
my %gotnosection;
my %adv_roles =
&Apache::lonnet::get_course_adv_roles($env{'request.course.id'},1);
foreach my $role (keys(%adv_roles)) {
my ($urole,$usec) = split(/:/,$role);
if (!$gotnosection{$urole}) {
$seccount->{$urole} ++;
$gotnosection{$urole} = 1;
}
if (ref($courseroles->{$urole}) eq 'ARRAY') {
if ($usec ne '') {
if (!grep(/^Q$usec\E$/,@{$courseroles->{$urole}})) {
push(@{$courseroles->{$urole}},$usec);
$seccount->{$urole} ++;
}
}
} else {
@{$courseroles->{$urole}} = ();
if ($usec ne '') {
$seccount->{$urole} ++;
push(@{$courseroles->{$urole}},$usec);
}
}
my $area = '/'.$cdom.'/'.$cnum;
if ($usec ne '') {
$area .= '/'.$usec;
}
if ($role =~ /^cr\//) {
&Apache::lonnet::custom_roleprivs($courseprivs,$urole,$cdom,$cnum,$urole.'.'.$area,$area);
} else {
&Apache::lonnet::standard_roleprivs($courseprivs,$urole,$cdom,$urole.'.'.$area,$cnum,$area);
}
}
my %sections_count = &Apache::loncommon::get_sections($cdom,$cnum,['st']);
@{$courseroles->{'st'}} = ();
&Apache::lonnet::standard_roleprivs($courseprivs,'st',$cdom,"st./$cdom/$cnum",$cnum,"/$cdom/$cnum");
if (keys(%sections_count) > 0) {
push(@{$courseroles->{'st'}},keys(%sections_count));
$seccount->{'st'} = scalar(keys(%sections_count));
}
$seccount->{'st'} ++; # Increment for a section-less student role.
my $rolehash = {
'roles' => $courseroles,
'seccount' => $seccount,
'privs' => $courseprivs,
};
&Apache::lonnet::do_cache_new('getcourseroles',$cdom.'_'.$cnum,$rolehash);
return;
}
sub get_customadhoc_roles {
my ($cdom,$cnum,$courseroles,$seccount,$courseprivs,$roledesc,$privref) = @_;
unless ((ref($courseroles) eq 'HASH') && (ref($seccount) eq 'HASH') &&
(ref($courseprivs) eq 'HASH') && (ref($roledesc) eq 'HASH')) {
return;
}
my $is_helpdesk = 0;
my $now = time;
foreach my $role ('dh','da') {
if ($env{"user.role.$role./$cdom/"}) {
my ($start,$end)=split(/\./,$env{"user.role.$role./$cdom/"});
if (!($start && ($now<$start)) && !($end && ($now>$end))) {
$is_helpdesk = 1;
last;
}
}
}
if ($is_helpdesk) {
my ($possroles,$description) = &Apache::lonnet::get_my_adhocroles($cdom.'_'.$cnum);
my %available;
if (ref($possroles) eq 'ARRAY') {
map { $available{$_} = 1; } @{$possroles};
}
my %domdefaults = &Apache::lonnet::get_domain_defaults($cdom);
if (ref($domdefaults{'adhocroles'}) eq 'HASH') {
if (keys(%{$domdefaults{'adhocroles'}})) {
my $numsec = 1;
my @sections;
my ($allseclist,$cached) =
&Apache::lonnet::is_cached_new('courseseclist',$cdom.'_'.$cnum);
if (defined($cached)) {
if ($allseclist ne '') {
@sections = split(/,/,$allseclist);
$numsec += scalar(@sections);
}
} else {
my %sections_count = &Apache::loncommon::get_sections($cdom,$cnum);
@sections = sort(keys(%sections_count));
$numsec += scalar(@sections);
$allseclist = join(',',@sections);
&Apache::lonnet::do_cache_new('courseseclist',$cdom.'_'.$cnum,$allseclist);
}
my (%adhoc,$gotprivs);
my $prefix = "cr/$cdom/$cdom".'-domainconfig';
foreach my $role (keys(%{$domdefaults{'adhocroles'}})) {
next if (($role eq '') || ($role =~ /\W/));
$seccount->{"$prefix/$role"} = $numsec;
$roledesc->{"$prefix/$role"} = $description->{$role};
if ((ref($privref) eq 'ARRAY') && (@{$privref} > 0)) {
if (exists($env{"user.priv.$prefix/$role./$cdom/$cnum./"})) {
$courseprivs->{"$prefix/$role./$cdom/$cnum./"} =
$env{"user.priv.$prefix/$role./$cdom/$cnum./"};
$courseprivs->{"$prefix/$role./$cdom/$cnum./$cdom/"} =
$env{"user.priv.$prefix/$role./$cdom/$cnum./$cdom/"};
$courseprivs->{"$prefix/$role./$cdom/$cnum./$cdom/$cnum"} =
$env{"user.priv.$prefix/$role./$cdom/$cnum./$cdom/$cnum"};
} else {
unless ($gotprivs) {
my ($adhocroles,$privscached) =
&Apache::lonnet::is_cached_new('adhocroles',$cdom);
if ((defined($privscached)) && (ref($adhocroles) eq 'HASH')) {
%adhoc = %{$adhocroles};
} else {
my $confname = &Apache::lonnet::get_domainconfiguser($cdom);
my %roledefs = &Apache::lonnet::dump('roles',$cdom,$confname,'rolesdef_');
foreach my $key (keys(%roledefs)) {
(undef,my $rolename) = split(/_/,$key);
if ($rolename ne '') {
my ($systempriv,$domainpriv,$coursepriv) = split(/\_/,$roledefs{$key});
$coursepriv = &Apache::lonnet::course_adhocrole_privs($rolename,$cdom,$cnum,$coursepriv);
$adhoc{$rolename} = join('_',($systempriv,$domainpriv,$coursepriv));
}
}
&Apache::lonnet::do_cache_new('adhocroles',$cdom,\%adhoc);
}
$gotprivs = 1;
}
($courseprivs->{"$prefix/$role./$cdom/$cnum./"},
$courseprivs->{"$prefix/$role./$cdom/$cnum./$cdom/"},
$courseprivs->{"$prefix/$role./$cdom/$cnum./$cdom/$cnum"}) =
split(/\_/,$adhoc{$role});
}
}
if ($available{$role}) {
$courseroles->{"$prefix/$role"} = \@sections;
}
}
}
}
}
return;
}
sub jump_to_role {
my ($cdom,$cnum,$seccount,$courseroles,$courseprivs,$roledesc,$privref,
$menucoll,$menuref) = @_;
my %lt = &Apache::lonlocal::texthash(
this => 'This role has section(s) associated with it.',
ente => 'Enter a specific section.',
orlb => 'Enter a specific section, or leave blank for no section.',
avai => 'Available sections are:',
youe => 'You entered an invalid section choice:',
plst => 'Please try again.',
role => 'The role you selected is not permitted to view the current page.',
swit => 'Switch role, but display Main Menu page instead?',
);
&js_escape(\%lt);
my $js;
if (ref($courseroles) eq 'HASH') {
$js = ' var secpick = new Array("'.$lt{'ente'}.'","'.$lt{'orlb'}.'");'."\n".
' var numsec = new Array();'."\n".
' var rolesections = new Array();'."\n".
' var rolenames = new Array();'."\n".
' var roleseclist = new Array();'."\n";
my @items = keys(%{$courseroles});
for (my $i=0; $i<@items; $i++) {
$js .= ' rolenames['.$i.'] = "'.$items[$i].'";'."\n";
my ($secs,$secstr);
if (ref($courseroles->{$items[$i]}) eq 'ARRAY') {
my @sections = sort { $a <=> $b } @{$courseroles->{$items[$i]}};
$secs = join('","',@sections);
$secstr = join(', ',@sections);
}
$js .= ' rolesections['.$i.'] = new Array("'.$secs.'");'."\n".
' roleseclist['.$i.'] = "'.$secstr.'";'."\n".
' numsec['.$i.'] = "'.$seccount->{$items[$i]}.'";'."\n";
}
}
my $checkroles = 0;
my $fallback = '/adm/menu';
my $displaymsg = $lt{'swit'};
if ((ref($privref) eq 'ARRAY') && (@{$privref} > 0) && (ref($courseprivs) eq 'HASH')) {
my %disallowed;
foreach my $role (sort(keys(%{$courseprivs}))) {
my $trole;
if ($role =~ m{^(.+?)\Q./$cdom/$cnum\E}) {
$trole = $1;
}
if (($trole ne '') && ($trole ne 'cm')) {
$disallowed{$trole} = 1;
foreach my $priv (@{$privref}) {
if ($courseprivs->{$role} =~ /\Q:$priv\E($|:|\&\w+)/) {
delete($disallowed{$trole});
last;
}
}
}
}
if (keys(%disallowed) > 0) {
$checkroles = 1;
$js .= " var disallow = new Array('".join("','",keys(%disallowed))."');\n".
" var rolecheck = 1;\n";
if ($menucoll) {
if (ref($menuref) eq 'HASH') {
if ($menuref->{'main'} eq 'n') {
$fallback = '/adm/navmaps';
if (&Apache::loncommon::course_type() eq 'Community') {
$displaymsg = &mt('Switch role, but display Community Contents page instead?');
} else {
$displaymsg = &mt('Switch role, but display Course Contents page instead?');
}
&js_escape(\$displaymsg);
}
}
}
}
}
&js_escape(\$fallback);
if (!$checkroles) {
$js .= " var disallow = new Array();\n".
" rolecheck = 0;\n";
}
return <<"END";
<script type="text/javascript">
//<![CDATA[
function adhocRole(newrole) {
$js
if (newrole == '') {
return;
}
var fullrole = newrole+'./$cdom/$cnum';
var selidx = '';
for (var i=0; i<rolenames.length; i++) {
if (rolenames[i] == newrole) {
selidx = i;
}
}
if (rolecheck > 0) {
for (var i=0; i<disallow.length; i++) {
if (disallow[i] == newrole) {
if (confirm("$lt{'role'}\\n$displaymsg")) {
document.rolechooser.destinationurl.value = '$fallback';
} else {
return;
}
}
}
}
var secok = 1;
var secchoice = '';
if (selidx >= 0) {
if (numsec[selidx] > 1) {
secok = 0;
var numrolesec = rolesections[selidx].length;
var msgidx = numsec[selidx] - numrolesec;
secchoice = prompt("$lt{'this'} "+secpick[msgidx]+"\\n$lt{'avai'} "+roleseclist[selidx],"");
if (secchoice == '') {
if (msgidx > 0) {
secok = 1;
}
} else {
for (var j=0; j<rolesections[selidx].length; j++) {
if (rolesections[selidx][j] == secchoice) {
secok = 1;
}
}
}
} else {
if (rolesections[selidx].length == 1) {
secchoice = rolesections[selidx][0];
}
}
}
if (secok == 1) {
if (secchoice != '') {
fullrole += '/'+secchoice;
}
} else {
if (secchoice != null) {
alert("$lt{'youe'} \\""+secchoice+"\\".\\n $lt{'plst'}");
}
return;
}
if (fullrole == "$env{'request.role'}") {
return;
}
itemid = retrieveIndex('gotorole');
if (itemid != -1) {
document.rolechooser.elements[itemid].name = fullrole;
}
document.rolechooser.switchrole.value = fullrole;
document.rolechooser.selectrole.value = '1';
document.rolechooser.submit();
return;
}
function retrieveIndex(item) {
for (var i=0;i<document.rolechooser.elements.length;i++) {
if (document.rolechooser.elements[i].name == item) {
return i;
}
}
return -1;
}
// ]]>
</script>
END
}
sub required_privs {
my $privs = {
'/adm/parmset' => 'opa,vpa',
'/adm/courseprefs' => 'opa,vpa',
'/adm/whatsnew' => 'whn',
'/adm/populate' => 'cst,vpa,vcl',
'/adm/trackstudent' => 'vsa',
'/adm/statistics' => 'mgr,vgr',
'/adm/setblock' => 'dcm,vcb',
'/adm/coursedocs' => 'mdc',
};
unless ($env{'course.'.$env{'request.course.id'}.'.grading'} eq 'spreadsheet') {
$privs->{'/adm/classcalc'} = 'vgr',
$privs->{'/adm/assesscalc'} = 'vgr',
$privs->{'/adm/studentcalc'} = 'vgr';
}
return $privs;
}
sub countdown_timer {
if (($env{'request.course.id'}) && ($env{'request.symb'} ne '') &&
(($env{'request.filename'}=~/$LONCAPA::assess_re/) ||
(($env{'request.symb'} =~ /ext\.tool$/) &&
(&Apache::lonnet::EXT('resource.0.gradable',$env{'request.symb'}) =~ /^yes$/i)))) {
my ($type,$hastimeleft,$slothastime);
my $now = time;
if ($env{'request.filename'} =~ /\.task$/) {
$type = 'Task';
} elsif ($env{'request.symb'} =~ /ext\.tool$/) {
$type = 'tool';
} else {
$type = 'problem';
}
my ($status,$accessmsg,$slot_name,$slot);
if ($type eq 'tool') {
($status,$accessmsg,$slot_name,$slot) =
&Apache::lonhomework::check_slot_access('0',$type,$env{'request.symb'},['0']);
} else {
($status,$accessmsg,$slot_name,$slot) =
&Apache::lonhomework::check_slot_access('0',$type);
}
if ($slot_name ne '') {
if (ref($slot) eq 'HASH') {
if (($slot->{'starttime'} < $now) &&
($slot->{'endtime'} > $now)) {
$slothastime = 1;
}
}
}
if ($status ne 'CAN_ANSWER') {
return;
}
my $duedate = &Apache::lonnet::EXT("resource.0.duedate");
my @interval=&Apache::lonnet::EXT("resource.0.interval");
my ($timelimit,$usesdone,$donebuttontext,$proctor,$secret);
if (@interval > 1) {
($timelimit,my $donesuffix) = split(/_/,$interval[0],2);
if ($donesuffix =~ /^done\:([^\:]+)\:(.*)$/) {
$usesdone = 'done';
$donebuttontext = $1;
(undef,$proctor,$secret) = split(/_/,$2);
} elsif ($donesuffix =~ /^done(|_.+)$/) {
$donebuttontext = &mt('Done');
($usesdone,$proctor,$secret) = split(/_/,$donesuffix);
}
my $first_access=&Apache::lonnet::get_first_access($interval[1]);
if ($first_access > 0) {
if ($first_access+$timelimit > time) {
$hastimeleft = 1;
}
}
}
if (($duedate && $duedate > time) ||
(!$duedate && $hastimeleft) ||
($slot_name ne '' && $slothastime)) {
my ($collapse,$expand,$alttxt,$title,$currdisp,$donebutton);
if ((@interval > 1 && $hastimeleft) ||
($type eq 'Task' && $slothastime)) {
$currdisp = 'inline';
$collapse = '► ';
if ((@interval > 1) && ($hastimeleft)) {
if ($usesdone eq 'done') {
$donebutton = &done_button_js($interval[1],'','',$proctor,$donebuttontext);
}
}
} else {
$currdisp = 'none';
$expand = '◄ ';
}
unless ($env{'environment.icons'} eq 'iconsonly') {
$alttxt = &mt('Timer');
$title = $alttxt.' ';
}
my $desc = &mt('Countdown to due date/time');
return <<END;
$donebutton
<a href="javascript:toggleCountdown();" class="LC_menubuttons_link">
<span id="ddcountcollapse" class="LC_menubuttons_inline_text">
$collapse
</span></a>
<span id="duedatecountdown" class="LC_menubuttons_inline_text" style="display: $currdisp;" role="timer"></span>
<a href="javascript:toggleCountdown();" class="LC_menubuttons_link">
<span id="ddcountexpand" class="LC_menubuttons_inline_text" >$expand</span>
<img src="/res/adm/pages/timer.png" title="$desc" class="LC_icon" alt="$alttxt" /><span class="LC_menubuttons_inline_text">$title</span></a>
END
}
}
return;
}
sub placement_progress {
my ($totalpoints,$incomplete) = &Apache::lonplacementtest::check_completion(undef,undef,1);
my $complete = 100 - $incomplete;
return '<span class="LC_placement_prog">'.
&mt('Test is [_1]% complete',$complete).'</span>';
}
sub linkprot_exit {
if (($env{'request.course.id'}) && ($env{'request.deeplink.login'})) {
my ($deeplink_symb,$deeplink);
my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
if (($cnum ne '') && ($cdom ne '')) {
$deeplink_symb = &Apache::loncommon::deeplink_login_symb($cnum,$cdom);
if ($deeplink_symb) {
if ($deeplink_symb =~ /\.(page|sequence)$/) {
my $mapname = &Apache::lonnet::deversion((&Apache::lonnet::decode_symb($deeplink_symb))[2]);
my $navmap = Apache::lonnavmaps::navmap->new();
if (ref($navmap)) {
$deeplink = $navmap->get_mapparam(undef,$mapname,'0.deeplink');
}
} else {
$deeplink = &Apache::lonnet::EXT('resource.0.deeplink',$deeplink_symb);
}
if ($deeplink ne '') {
my ($state,$others,$listed,$scope,$protect,$display,$target,$exit) = split(/,/,$deeplink);
my %lt = &Apache::lonlocal::texthash(
title => 'Exit Tool',
okdone => 'Click "OK" to exit embedded tool',
cancel => 'Click "Cancel" to continue working.',
ok => 'OK',
exit => 'Cancel',
);
if ($exit) {
my ($show,$text) = split(/:/,$exit);
unless ($show eq 'no') {
my $height = 250;
my $width = 300;
my $exitbuttontext;
if ($text eq '') {
$exitbuttontext = &mt('Exit Tool');
} else {
$exitbuttontext = $text;
}
return <<END;
<form method="post" name="LCexitButton" action="/adm/linkexit">
<input type="hidden" name="LC_deeplink_exit" value="" />
<button id="LC_exit-confirm-opener" type="button">$exitbuttontext</button>
</form>
<div id="LC_exit-confirm" title="$lt{'title'}">
<p>$lt{'okdone'} $lt{'cancel'}</p>
</div>
<script type="text/javascript">
// <![CDATA[
\$( "#LC_exit-confirm" ).dialog({ autoOpen: false });
\$( "#LC_exit-confirm-opener" ).click(function() {
\$( "#LC_exit-confirm" ).dialog( "open" );
\$( "#LC_exit-confirm" ).dialog({
resizable: false,
height: $height,
width: $width,
modal: true,
buttons: [
{
text: "$lt{'ok'}",
click: function() {
\$( this ).dialog( "close" );
\$( '[name="LC_deeplink_exit"]' )[0].value = 'true';
\$( '[name="LCexitButton"]' )[0].submit();
},
},
{
text: "$lt{'exit'}",
click: function() {
\$( this ).dialog( "close" );
},
},
],
});
});
// ]]>
</script>
END
}
}
}
}
}
}
return;
}
# ================================================================ Main Program
BEGIN {
if (! defined($readdesk)) {
{
my $tabfile = $Apache::lonnet::perlvar{'lonTabDir'}.'/mydesk.tab';
if ( CORE::open( my $config,"<$tabfile") ) {
while (my $configline=<$config>) {
$configline=(split(/\#/,$configline))[0];
$configline=~s/^\s+//;
chomp($configline);
if ($configline=~/^cat\:/) {
my @entries=split(/\:/,$configline);
$category_positions{$entries[2]}=$entries[1];
$category_names{$entries[2]}=$entries[3];
} elsif ($configline=~/^prim\:/) {
my @entries = (split(/\:/, $configline))[1..7];
push(@primary_menu,\@entries);
} elsif ($configline=~/^primsub\:/) {
my ($parent,@entries) = (split(/\:/, $configline))[1..5];
push(@{$primary_submenu{$parent}},\@entries);
} elsif ($configline=~/^scnd\:/) {
my @entries = (split(/\:/, $configline))[1..6];
push(@secondary_menu,\@entries);
} elsif ($configline=~/^scndsub\:/) {
my ($parent,@entries) = (split(/\:/, $configline))[1..4];
push(@{$secondary_submenu{$parent}},\@entries);
} elsif ($configline) {
push(@desklines,$configline);
}
}
CORE::close($config);
}
}
$readdesk='done';
}
}
1;
__END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>