Files
sample-site/devices.html
2025-10-06 21:22:13 +09:00

407 lines
15 KiB
HTML

<!DOCTYPE html>
<html
lang="en-US"
class="light-style"
>
<head>
<meta charset="utf-8" />
<meta
name="viewport"
content="width=device-width, initial-scale=1.0, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0"
/>
<title>Devices - amiPro sample site</title>
<meta name="description" content="" />
<!-- Favicon -->
<link rel="icon" type="image/x-icon" href="files/favicon.ico" />
<!-- Fonts -->
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link
href="https://fonts.googleapis.com/css2?family=Public+Sans:ital,wght@0,300;0,400;0,500;0,600;0,700;1,300;1,400;1,500;1,600;1,700&display=swap"
rel="stylesheet"
/>
<!-- Icons. Uncomment required icon fonts -->
<link rel="stylesheet" href="files/boxicons.css?v=20230405" />
<!-- Core CSS -->
<link rel="stylesheet" href="files/core.css" class="template-customizer-core-css" />
<link rel="stylesheet" href="files/theme-default.css" class="template-customizer-theme-css" />
<link rel="stylesheet" href="files/demo.css" />
<!-- Vendors CSS -->
<link rel="stylesheet" href="files/perfect-scrollbar.css" />
<!-- Page CSS -->
<!-- Helpers -->
<script src="files/helpers.js"></script>
<!--! Template customizer & Theme config files MUST be included after core stylesheets and helpers.js in the <head> section -->
<!--? Config: Mandatory theme config file contain global vars & default theme options, Set your preferred theme option in this file. -->
<script src="files/config.js"></script>
<script src="files/jquery.js"></script>
<script src="files/popper.js"></script>
<script src="files/bootstrap.js"></script>
<script src="files/perfect-scrollbar.js"></script>
<script src="files/menu.js"></script>
<script src="files/main.js"></script>
<script src="files/ua-parser.js"></script>
<script src="files/amipro_utils.js?v=20230401402"></script>
<script src="files/dfido2-lib.js?v=20230918"></script>
<script>
var user_id, reg_session_id;
const i18n_messages = new Map();
var lang_map = new Map();
lang_map.set("en-US", "Add device");
lang_map.set("zh-CN", "添加设备");
lang_map.set("ja", "デバイスを追加");
i18n_messages.set("btn_add", lang_map);
lang_map = new Map();
lang_map.set("en-US", "My devices");
lang_map.set("zh-CN", "我的设备");
lang_map.set("ja", "マイデバイス");
i18n_messages.set("my_devices", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Device registered");
lang_map.set("zh-CN", "添加设备成功");
lang_map.set("ja", "デバイス登録完了。");
i18n_messages.set("msg_register_ok", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Device");
lang_map.set("zh-CN", "设备");
lang_map.set("ja", "デバイス");
i18n_messages.set("title_device", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Registered time");
lang_map.set("zh-CN", "添加时间");
lang_map.set("ja", "登録時間");
i18n_messages.set("title_time", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Actions");
lang_map.set("zh-CN", "操作");
lang_map.set("ja", "操作");
i18n_messages.set("title_act", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Delete");
lang_map.set("zh-CN", "删除");
lang_map.set("ja", "削除");
i18n_messages.set("title_del", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Contact us");
lang_map.set("zh-CN", "联系我们");
lang_map.set("ja", "お問い合わせ");
i18n_messages.set("title_contact", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Log out");
lang_map.set("zh-CN", "登出");
lang_map.set("ja", "ログアウト");
i18n_messages.set("title_logout", lang_map);
lang_map = new Map();
lang_map.set("en-US", "No devices, please add.");
lang_map.set("zh-CN", "无设备,请添加。");
lang_map.set("ja", "デバイスがなし、追加してください。");
i18n_messages.set("title_empty_list", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Deleted device.");
lang_map.set("zh-CN", "设备删除成功。");
lang_map.set("ja", "デバイスを削除しました。");
i18n_messages.set("msg_deldev_ok", lang_map);
lang_map = new Map();
lang_map.set("en-US", "Do you want to delete this device?");
lang_map.set("zh-CN", "确认删除此设备吗?");
lang_map.set("ja", "デバイスを削除しますか?");
i18n_messages.set("msg_confirm_deldev", lang_map);
lang_map = new Map();
lang_map.set("en-US", "The fido2 session is good.");
lang_map.set("zh-CN", "FIDO2会话正常。");
lang_map.set("ja", "FIDO2セッションは正常です。");
i18n_messages.set("msg_session_status_ok", lang_map);
lang_map = new Map();
lang_map.set("en-US", "The fido2 session is invalid. You can refuse accessing to your site's pages.");
lang_map.set("zh-CN", "FIDO2会话无效。您可以拒绝访问您网站的页面。");
lang_map.set("ja", "FIDO2セッションは無効です。あなたのサイトのページにアクセスすることを拒否できます。");
i18n_messages.set("msg_session_status_fail", lang_map);
window.onload = async function() {
setI18NText(i18n_messages)
setSessionStatus();
let url = new URL(window.location.href);
let params = url.searchParams;
reg_session_id = params.get("rid");
user_id = params.get("uid");
if(reg_session_id && 0 < reg_session_id.length){
const reg_unm_json = await getRegistrationUser(reg_session_id);
user_id = ''
if(reg_unm_json && reg_unm_json.status === 'ok'){
user_id = reg_unm_json.username;
await addDevice(reg_unm_json.displayname)
}else{
window.location.href = "login.html";
}
}
$('#user_id').html(user_id);
listDevices();
}
async function setSessionStatus(){
const sessionOk = await validSession();
if(sessionOk){
$('#session_status').html(getI18NText(i18n_messages, 'msg_session_status_ok'))
}else{
$('#session_status').html(getI18NText(i18n_messages, 'msg_session_status_fail'))
}
}
async function addDevice(display_name){
//You can change the display name or ask the user to input it, if the display_name is null.
const result = await registerFido2(user_id, display_name?display_name:'dis_'+user_id);
if(result.status === 'ok'){
//You can try to insert user_id into your user db now if this user_id does not exist in your db.
alert($('#msg_register_ok').html());
listDevices();
}else{
errProcessFido2(result)
}
setSessionStatus();
}
async function delDevice(device_id){
if(window.confirm(getI18NText(i18n_messages, 'msg_confirm_deldev'))){
const result = await delUserDeviceFido2(device_id)
if("ok" === result.status){
alert(getI18NText(i18n_messages, 'msg_deldev_ok'));
listDevices();
}else{
const msg = getI18NErrorMessage(result.errorMessage);
alert(msg?msg:result.errorMessage);
}
}
}
async function listDevices(){
const result = await listUserDevicesFido2()
$("#devices_list").html("");
if("ok" === result.status){
let lst_html = ""
for(let dev of result.devices){
var dev_desc = dev.desc
if(!dev_desc || 0 === dev_desc.length){
let parser = new UAParser(dev.userAgent);
if(parser.getOS().name){
dev_desc = parser.getDevice().model + ',' + parser.getOS().name + ',' + parser.getBrowser().name;
}else{
dev_desc = dev.userAgent;
}
}
var date = new Date(dev.registered_time);
lst_html += '<tr><td><strong>'+ dev_desc +'</strong></td><td>'+
date.toLocaleString()+'</td><td>'+
"<a style=\"padding-left: 0px;\" class=\"dropdown-item\" href=\"javascript:delDevice('"+dev.device_id+"');\">"+
'<i class="bx bx-trash me-1"></i><span id="title_del"> '+ getI18NText(i18n_messages, 'title_del') +'</span></a></div></td></tr>'
}
$("#devices_list").html(lst_html);
}else{
const msg = getI18NErrorMessage(result.errorMessage); //fido2LibErrMsgLanguages.japanese
alert(msg?msg:result.errorMessage);
}
}
</script>
</head>
<body>
<div id="msg_register_ok" style="display:none;">Device registered</div>
<!-- Layout wrapper -->
<div class="layout-wrapper layout-content-navbar">
<div class="layout-container">
<!-- Layout container -->
<div class="layout-page">
<!-- Navbar -->
<nav
class="layout-navbar container-xxl navbar navbar-expand-xl navbar-detached align-items-center bg-navbar-theme"
id="layout-navbar"
>
<div class="layout-menu-toggle navbar-nav align-items-xl-center me-3 me-xl-0 d-xl-none">
<a class="nav-item nav-link px-0 me-xl-4" href="javascript:void(0)">
<i class="bx bx-menu bx-sm"></i>
</a>
</div>
<div class="navbar-nav-right d-flex align-items-center" id="navbar-collapse">
<!-- Search -->
<div class="navbar-nav align-items-center">
<div class="nav-item d-flex align-items-center">
<i class="bx bx-user fs-4 lh-0"></i>
WELCOME <span id="session_status" style="padding-left: 20px;"></span>
</div>
</div>
<!-- /Search -->
<ul class="navbar-nav flex-row align-items-center ms-auto">
<!-- User -->
<li class="nav-item navbar-dropdown dropdown-user dropdown">
<a class="nav-link dropdown-toggle hide-arrow" href="javascript:void(0);" data-bs-toggle="dropdown">
<div class="avatar avatar-online">
<img src="files/avatar.png" alt class="w-px-40 h-auto rounded-circle" />
</div>
</a>
<ul class="dropdown-menu dropdown-menu-end">
<li>
<a class="dropdown-item" href="#">
<div class="d-flex">
<div class="flex-shrink-0 me-3">
<div class="avatar avatar-online">
<img src="files/avatar.png" alt class="w-px-40 h-auto rounded-circle" />
</div>
</div>
<div class="flex-grow-1">
<span class="fw-semibold d-block" id="user_id">User</span>
</div>
</div>
</a>
</li>
<li>
<div class="dropdown-divider"></div>
</li>
<li>
<a class="dropdown-item" href="login.html">
<i class="bx bx-power-off me-2"></i>
<span class="align-middle" id="title_logout">Log Out</span>
</a>
</li>
</ul>
</li>
<!--/ User -->
</ul>
</div>
</nav>
<!-- / Navbar -->
<!-- Content wrapper -->
<div class="content-wrapper">
<!-- Content -->
<div class="container-xxl flex-grow-1 container-p-y">
<!-- Basic Bootstrap Table -->
<div class="card">
<h5 class="card-header" id="my_devices">My devices</h5>
<button style="width:30%;float: right;margin-left: 66%;" class="btn btn-info" onclick="addDevice();"
id="btn_add">Add device</button>
<!-- button style="width:30%;float: right;margin-left: 66%;" class="btn btn-info" onclick="listDevices();"
id="btn_add">List</button -->
<div class="table-responsive text-nowrap">
<table class="table">
<thead>
<tr>
<th id="title_device">Device</th>
<th id="title_time">Registered time</th>
<th id="title_act">Actions</th>
</tr>
</thead>
<tbody class="table-border-bottom-0" id="devices_list">
<tr>
<td colspan="3" style="text-align: center;" id="title_empty_list"></td>
</tr>
</tbody>
</table>
</div>
</div>
<!--/ Basic Bootstrap Table -->
</div>
<!-- / Content -->
<!-- Footer -->
<footer class="content-footer footer bg-footer-theme">
<div class="container-xxl d-flex flex-wrap justify-content-between py-2 flex-md-row flex-column">
<div class="mb-2 mb-md-0">
©
<script>
document.write(new Date().getFullYear());
</script>
<a href="https://www.amipro.me" target="_blank" class="footer-link fw-bolder">amiPro</a>
</div>
<div>
<a
href="mailto:sales@amipro.me?subject=contact"
target="_blank"
class="footer-link me-4"
id="title_contact"
>Contact</a
>
</div>
</div>
</footer>
<!-- / Footer -->
<div class="content-backdrop fade"></div>
</div>
<!-- Content wrapper -->
</div>
<!-- / Layout page -->
</div>
<!-- Overlay -->
<div class="layout-overlay layout-menu-toggle"></div>
</div>
<!-- / Layout wrapper -->
<!-- div class="buy-now">
<a
href="https://themeselection.com/products/sneat-bootstrap-html-admin-template/"
target="_blank"
class="btn btn-danger btn-buy-now"
>Upgrade to Pro</a
>
</div -->
<!-- Page JS -->
<!-- Place this tag in your head or just before your close body tag. -->
<script async defer src="https://buttons.github.io/buttons.js"></script>
</body>
</html>