set_title("Login"); $page->set_heading("Login"); $page->add_block(new NavBlock()); $page->add_block(new Block( "Login There", "There should be a login box to the left" )); } public function display_user_list(Page $page, $table, $paginator): void { $page->set_title("User List"); $page->set_heading("User List"); $page->add_block(new NavBlock()); $page->add_block(new Block("Users", $table . $paginator)); } public function display_user_links(Page $page, User $user, $parts): void { # $page->add_block(new Block("User Links", join(", ", $parts), "main", 10)); } public function display_user_block(Page $page, User $user, $parts): void { $html = emptyHTML('Logged in as ', $user->name); foreach ($parts as $part) { $html->appendChild(BR()); $html->appendChild(A(["href" => $part["link"]], $part["name"])); } $b = new Block("User Links", $html, "left", 90); $b->is_content = false; $page->add_block($b); } public function display_signup_page(Page $page): void { global $config, $user; $tac = $config->get_string("login_tac", ""); if ($config->get_bool("login_tac_bbcode")) { $tac = send_event(new TextFormattingEvent($tac))->formatted; } $email_required = ( $config->get_bool("user_email_required") && !$user->can(Permissions::CREATE_OTHER_USER) ); $form = SHM_SIMPLE_FORM( "user_admin/create", TABLE( ["class" => "form"], TBODY( TR( TH("Name"), TD(INPUT(["type" => 'text', "name" => 'name', "required" => true])) ), TR( TH("Password"), TD(INPUT(["type" => 'password', "name" => 'pass1', "required" => true])) ), TR( TH(rawHTML("Repeat Password")), TD(INPUT(["type" => 'password', "name" => 'pass2', "required" => true])) ), TR( TH($email_required ? "Email" : rawHTML("Email (Optional)")), TD(INPUT(["type" => 'email', "name" => 'email', "required" => $email_required])) ), TR( TD(["colspan" => "2"], rawHTML(captcha_get_html())) ), ), TFOOT( TR(TD(["colspan" => "2"], INPUT(["type" => "submit", "value" => "Create Account"]))) ) ) ); $html = emptyHTML( $tac ? P(rawHTML($tac)) : null, $form ); $page->set_title("Create Account"); $page->set_heading("Create Account"); $page->add_block(new NavBlock()); $page->add_block(new Block("Signup", $html)); } public function display_user_creator(): void { global $page; $form = SHM_SIMPLE_FORM( "user_admin/create_other", TABLE( ["class" => "form"], TBODY( TR( TH("Name"), TD(INPUT(["type" => 'text', "name" => 'name', "required" => true])) ), TR( TH("Password"), TD(INPUT(["type" => 'password', "name" => 'pass1', "required" => true])) ), TR( TH(rawHTML("Repeat Password")), TD(INPUT(["type" => 'password', "name" => 'pass2', "required" => true])) ), TR( TH(rawHTML("Email")), TD(INPUT(["type" => 'email', "name" => 'email'])) ), TR( TD(["colspan" => 2], rawHTML("(Email is optional for admin-created accounts)")), ), ), TFOOT( TR(TD(["colspan" => "2"], INPUT(["type" => "submit", "value" => "Create Account"]))) ) ) ); $page->add_block(new Block("Create User", (string)$form, "main", 75)); } public function display_signups_disabled(Page $page): void { $page->set_title("Signups Disabled"); $page->set_heading("Signups Disabled"); $page->add_block(new NavBlock()); $page->add_block(new Block( "Signups Disabled", "The board admin has disabled the ability to create new accounts~" )); } public function display_login_block(Page $page): void { $page->add_block(new Block("Login", $this->create_login_block(), "left", 90)); } public function create_login_block(): HTMLElement { global $config, $user; $form = SHM_SIMPLE_FORM( "user_admin/login", TABLE( ["style" => "width: 100%", "class" => "form"], TBODY( TR( TH(LABEL(["for" => "user"], "Name")), TD(INPUT(["id" => "user", "type" => "text", "name" => "user", "autocomplete" => "username"])) ), TR( TH(LABEL(["for" => "pass"], "Password")), TD(INPUT(["id" => "pass", "type" => "password", "name" => "pass", "autocomplete" => "current-password"])) ) ), TFOOT( TR(TD(["colspan" => "2"], INPUT(["type" => "submit", "value" => "Log In"]))) ) ) ); $html = emptyHTML(); $html->appendChild($form); if ($config->get_bool("login_signup_enabled") && $user->can(Permissions::CREATE_USER)) { $html->appendChild(SMALL(A(["href" => make_link("user_admin/create")], "Create Account"))); } return $html; } private function _ip_list(string $name, array $ips): HTMLElement { $td = TD("$name: "); $n = 0; foreach ($ips as $ip => $count) { $td->appendChild(BR()); $td->appendChild("$ip ($count)"); if (++$n >= 20) { $td->appendChild(BR()); $td->appendChild("..."); break; } } return $td; } public function display_ip_list(Page $page, array $uploads, array $comments, array $events): void { $html = TABLE( ["id" => "ip-history"], TR( $this->_ip_list("Uploaded from", $uploads), $this->_ip_list("Commented from", $comments), $this->_ip_list("Logged Events", $events) ), TR( TD(["colspan" => "3"], "(Most recent at top)") ) ); $page->add_block(new Block("IPs", $html, "main", 70)); } public function display_user_page(User $duser, $stats): void { global $page; assert(is_array($stats)); $stats[] = 'User ID: '.$duser->id; $page->set_title(html_escape($duser->name)."'s Page"); $page->set_heading(html_escape($duser->name)."'s Page"); $page->add_block(new NavBlock()); $page->add_block(new Block("Stats", join("
", $stats), "main", 10)); } public function build_operations(User $duser, UserOperationsBuildingEvent $event): string { global $config, $user; $html = emptyHTML(); // just a fool-admin protection so they dont mess around with anon users. if ($duser->id != $config->get_int('anon_id')) { if ($user->can(Permissions::EDIT_USER_NAME)) { $html->appendChild(SHM_USER_FORM( $duser, "user_admin/change_name", "Change Name", TBODY(TR( TH("New name"), TD(INPUT(["type" => 'text', "name" => 'name', "value" => $duser->name])) )), "Set" )); } $html->appendChild(SHM_USER_FORM( $duser, "user_admin/change_pass", "Change Password", TBODY( TR( TH("Password"), TD(INPUT(["type" => 'password', "name" => 'pass1', "autocomplete" => 'new-password'])) ), TR( TH("Repeat Password"), TD(INPUT(["type" => 'password', "name" => 'pass2', "autocomplete" => 'new-password'])) ), ), "Set" )); $html->appendChild(SHM_USER_FORM( $duser, "user_admin/change_email", "Change Email", TBODY(TR( TH("Address"), TD(INPUT(["type" => 'text', "name" => 'address', "value" => $duser->email, "autocomplete" => 'email', "inputmode" => 'email'])) )), "Set" )); if ($user->can(Permissions::EDIT_USER_CLASS)) { global $_shm_user_classes; $select = SELECT(["name" => "class"]); foreach ($_shm_user_classes as $name => $values) { $select->appendChild( OPTION(["value" => $name, "selected" => $name == $duser->class->name], ucwords($name)) ); } $html->appendChild(SHM_USER_FORM( $duser, "user_admin/change_class", "Change Class", TBODY(TR(TD($select))), "Set" )); } if ($user->can(Permissions::DELETE_USER)) { $html->appendChild(SHM_USER_FORM( $duser, "user_admin/delete_user", "Delete User", TBODY( TR(TD(LABEL(INPUT(["type" => 'checkbox', "name" => 'with_images']), "Delete images"))), TR(TD(LABEL(INPUT(["type" => 'checkbox', "name" => 'with_comments']), "Delete comments"))), ), TFOOT( TR(TD(INPUT(["type" => 'button', "class" => 'shm-unlocker', "data-unlock-sel" => '.deluser', "value" => 'Unlock']))), TR(TD(INPUT(["type" => 'submit', "class" => 'deluser', "value" => 'Delete User', "disabled" => 'true']))), ) )); } foreach ($event->parts as $part) { $html .= $part; } } return (string)$html; } public function get_help_html(): HTMLElement { global $user; $output = emptyHTML(P("Search for posts posted by particular individuals.")); $output->appendChild(SHM_COMMAND_EXAMPLE( "poster=username", 'Returns posts posted by "username".' )); $output->appendChild(SHM_COMMAND_EXAMPLE( "poster_id=123", 'Returns posts posted by user 123.' )); if ($user->can(Permissions::VIEW_IP)) { $output->appendChild(SHM_COMMAND_EXAMPLE( "poster_ip=127.0.0.1", "Returns posts posted from IP 127.0.0.1." )); } return $output; } /** * @param Page $page * @param UserClass[] $classes * @param \ReflectionClassConstant[] $permissions */ public function display_user_classes(Page $page, array $classes, array $permissions): void { $table = TABLE(["class" => "zebra"]); $row = TR(); $row->appendChild(TH("Permission")); foreach ($classes as $class) { $n = $class->name; if ($class->parent) { $n .= " ({$class->parent->name})"; } $row->appendChild(TH($n)); } $row->appendChild(TH("Description")); $table->appendChild($row); foreach ($permissions as $perm) { $row = TR(); $row->appendChild(TH($perm->getName())); foreach ($classes as $class) { $opacity = array_key_exists($perm->getValue(), $class->abilities) ? 1 : 0.2; if ($class->can($perm->getValue())) { $cell = TD(["style" => "color: green; opacity: $opacity;"], "✔"); } else { $cell = TD(["style" => "color: red; opacity: $opacity;"], "✘"); } $row->appendChild($cell); } $doc = $perm->getDocComment(); if ($doc) { $doc = preg_replace('/\/\*\*|\n\s*\*\s*|\*\//', '', $doc); $row->appendChild(TD(["style" => "text-align: left;"], $doc)); } else { $row->appendChild(TD("")); } $table->appendChild($row); } $page->set_title("User Classes"); $page->set_heading("User Classes"); $page->add_block(new NavBlock()); $page->add_block(new Block("Classes", $table, "main", 10)); } }