Browse Source

add query

root 2 years ago
parent
commit
a73a69bd87
4 changed files with 408 additions and 20 deletions
  1. 43 9
      chem/index.html
  2. 34 0
      chem/query/index.html
  3. 321 10
      js/chem.js
  4. 10 1
      log/index.html

+ 43 - 9
chem/index.html

@@ -24,6 +24,12 @@
             overflow-y: auto;
         }
 
+        pre {
+            overflow-x: auto;
+            overflow-y: auto;
+            white-space: pre-wrap;
+        }
+
         .btn-default:focus,
         .btn-default:focus-visible,
         .btn-default:focus-within,
@@ -73,7 +79,7 @@
                     <button type="button" class="btn btn-default" onclick="balUp()" id="balShare" title="导入配平结果"> <span
                             class="glyphicon glyphicon-chevron-right" aria-hidden="true"></span></button>
                 </span>
-                <input id="balInput" type="text" class="form-control" oninput="input()">
+                <input id="balInput" type="text" class="form-control" oninput="input()" autocomplete="off">
                 <span class="input-group-btn">
                     <button type="button" class="btn btn-default" onclick="balance()" id="balBtn">配平</button>
                     <button type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">
@@ -89,38 +95,66 @@
                         <li class="divider"></li>
                         <li><a href="/chemist" data-toggle="tooltip" data-placement="left" title="将打开新页面">配平接口</a>
                         </li>
-                        <li class="divider"></li>
-                        <li><a href="#" data-toggle="tooltip" data-placement="left" title="将会对所有人可见">发送到数据库</a></li>
+                        <!-- <li class="divider"></li>
+                        <li><a href="#" data-toggle="tooltip" data-placement="left" title="将会对所有人可见">发送到数据库</a></li> -->
                     </ul>
                 </span>
             </div>
             <hr>
-            <div class="well" id="frame">
+            <div class="well frame">
             </div>
             <hr>
             <div class="input-group">
                 <span class="input-group-btn">
                     <button type="button" class="btn btn-default" onclick="qryUp()" id="qryShare" title="导入配平结果"> <span
                             class="glyphicon glyphicon-chevron-right" aria-hidden="true"></span></button>
+                    <button type="button" class="btn btn-default" data-toggle="button" onclick="qryToggleMatch()"
+                        id="qryMatch">匹配分子</button>
                 </span>
-                <input id="qryInput" type="text" class="form-control" oninput="input()">
+                <input id="qryInput" type="text" class="form-control qryInput" autocomplete="off" oninput="input2()">
+                <span class="input-group-addon qryInputHidable" style="display:none">=</span>
+                <input id="qryInput2" type="text" class="form-control qryInput qryInputHidable" style="display:none"
+                    autocomplete="off" oninput="input2()">
                 <span class="input-group-btn">
-                    <button type="button" class="btn btn-default" onclick="balance()" id="qryBtn">查询</button>
+                    <button type="button" class="btn btn-default" onclick="query()" id="qryBtn">查询方程式</button>
                     <button type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">
                         <span class="caret"></span>
                         <span class="sr-only">切换下拉菜单</span>
                     </button>
                     <ul class="dropdown-menu" role="menu">
                         <li class="dropdown-header">功能</li>
-                        <li><a href="#" onclick="setBal()">查询</a></li>
+                        <li><a href="#1" onclick="setQryEq()">查询方程式</a></li>
+                        <li><a href="#1" onclick="setQryEq2()" data-toggle="tooltip" data-placement="left"
+                                title="限制等号左右">查询方程式*</a></li>
+                        <li><a href="#4" onclick="setQryMo()">查询分子</a></li>
                         <li class="divider"></li>
-                        <li><a href="/chemist" data-toggle="tooltip" data-placement="left" title="将打开新页面">上传到数据库</a>
+                        <li><a href="#5" onclick="setAddEq()">上传方程式</a></li>
+                        <li><a href="#5" onclick="setAddMo()">上传分子</a></li>
+                        <li class="divider"></li>
+                        <li><a href="#1" onclick="setUpdEq()">修改方程式</a></li>
+                        <li><a href="#4" onclick="setUpdMo()">修改分子</a></li>
                         </li>
                     </ul>
                 </span>
             </div>
+            <div class="input-group addInput">
+                <span id="addId"><span class="input-group-addon">id</span>
+                    <input type="text" class="form-control" id="addIdText" oninput="input2()" autocomplete="off"
+                        placeholder="1">
+                </span>
+                <span><span class="input-group-addon">反应条件</span>
+                    <input id="addCondition" type="text" class="form-control" autocomplete="off"
+                        placeholder="加热;高温;点燃;通电;光照;MnO2">
+                </span>
+                <span><span class="input-group-addon">描述</span>
+                    <textarea id="addDescription" type="text" class="form-control" autocomplete="off"
+                        style="resize:vertical"></textarea>
+                </span>
+                <span class="input-group-addon">:)</span>
+            </div>
+            <pre class="ok" style="font-family: 'Consolas'"></pre>
             <hr>
-            <div class="well" id="frame">
+            <div class="well frame">
             </div>
         </div>
     </div>

+ 34 - 0
chem/query/index.html

@@ -0,0 +1,34 @@
+<!DOCTYPE html>
+
+<head>
+    <!-- <link async rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
+        integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous"> -->
+    <script type="text/javascript" src="../../js/jquery.min.js" crossorigin="anonymous"></script>
+    <link rel="stylesheet" href="../../css/bootstrap.min.css" crossorigin="anonymous">
+    <script type="text/javascript" src="../../js/bootstrap.min.js" crossorigin="anonymous"></script>
+    <link rel="stylesheet" href="../../css/style.css">
+    <link rel="preconnect" href="https://fonts.gstatic.com">
+    <link href="https://fonts.loli.net/css2?family=Anonymous+Pro:ital,wght@0,400;0,700;1,400;1,700
+&family=Noto+Serif+SC:wght@300;400;500;600;700&display=swap" rel="stylesheet">
+    <title>化学</title>
+
+</head>
+
+<body>
+    <h4 class="well">!! !。?<div class="well text-warning"><div class="alert alert-danger">WARNING!!!!</div><h3>YOUUUUUUUU&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;re being watched.</h3></div><div style="font-size:10px">Do not loapp.post('/chess/setboard', (req, res) => {
+        board = req.body;
+        const from = req.body.from;
+        const result = { succeed: true };
+        wsClients['chess'].forEach((client) => {
+            // console.log(client)
+            client.send(JSON.stringify({
+                from,
+                content: req.body.content,
+                message: req.body.message
+            }));
+        });
+        res.json(result);
+    });.</div></h4>
+</body>
+
+</html>

+ 321 - 10
js/chem.js

@@ -153,7 +153,7 @@ function parseEquation(str) {
 
 function renderEquation(str) {
     str = str.replace(/[\[{]/g, "(").replace(/[\]}]/g, ")");
-    str=str.replace(/([\+\=\.;])\1+/g,'$1')
+    str = str.replace(/([\+\=\.;])\1+/g, '$1')
     str = str.replace(/[^\dA-Za-z<>\(\)\+\-=\.;]/g, "");
     // console.log('Rendering equation', str)
     str = str.replace(/([A-Za-z]+)/g, "\\text{$1}");
@@ -171,6 +171,7 @@ var mode = 'bal', balInput, balText = '', inputText
 $().ready(function () {
     balInput = $("#balInput")[0]
     setBal();
+    setQryEq();
     $("#balInput").keydown(function (e) {
         if (e.keyCode == 13 && mode == 'bal') {
             $("#balBtn")[0].click();
@@ -179,7 +180,7 @@ $().ready(function () {
     $(function () { $("[data-toggle='tooltip']").tooltip(); });
 })
 function setBal() {
-    $('#frame')[0].innerHTML = renderEquation(balText);
+    $('.frame')[0].innerHTML = renderEquation(balText);
     MathJax.typeset()
     $('#balBtn').text('配平')
     $('#balShare').removeClass('disabled')
@@ -210,34 +211,35 @@ function setWeigh2() {
 function input() {
     inputText = balInput.value
     if (mode == 'bal') {
-        $('#frame')[0].innerHTML = renderEquation((inputText == '') ? 'CrI3+Cl2+KOH=K2CrO4+KIO4+KCl+H2O' : inputText) + '<br>'
+        $('.frame')[0].innerHTML = renderEquation((inputText == '') ? 'CrI3+Cl2+KOH=K2CrO4+KIO4+KCl+H2O' : inputText) + '<br>'
             + '<span class="glyphicon glyphicon-chevron-down" aria-hidden="true"></span><br>' + ((balText != '') ? (renderEquation(balText)) : ('...'));
         MathJax.typeset()
     } else if (mode == 'weigh') {
-        $('#frame')[0].innerHTML = renderEquation((inputText == '') ? 'CH3CHO+2Ag(NH3)2OH=CH3COONH4+2Ag+3NH3+H2O' : inputText) + '<br>'
+        $('.frame')[0].innerHTML = renderEquation((inputText == '') ? 'CH3CHO+2Ag(NH3)2OH=CH3COONH4+2Ag+3NH3+H2O' : inputText) + '<br>'
             + weighEquation((inputText == '') ? 'CH3CHO+2Ag(NH3)2OH=CH3COONH4+2Ag+3NH3+H2O' : inputText);
         MathJax.typeset()
     } else if (mode == 'weigh2') {
-        $('#frame')[0].innerHTML = renderEquation((inputText == '') ? 'CH3CHO+2Ag(NH3)2OH=CH3COONH4+2Ag+3NH3+H2O' : inputText) + '<br>'
+        $('.frame')[0].innerHTML = renderEquation((inputText == '') ? 'CH3CHO+2Ag(NH3)2OH=CH3COONH4+2Ag+3NH3+H2O' : inputText) + '<br>'
             + weighEquation((inputText == '') ? 'CH3CHO+2Ag(NH3)2OH=CH3COONH4+2Ag+3NH3+H2O' : inputText, 1);
         MathJax.typeset()
     }
 }
 var running;
 function balance() {
+    input()
     if (running || mode != 'bal') return;
     $('#balBtn').text('配平...')
     $('#balBtn').addClass('disabled')
-    $('#frame').addClass('text-muted')
+    $('.frame').addClass('text-muted')
     running = 1;
-    $.get('/chem?' + ((inputText == '') ? 'CrI3+Cl2+KOH=K2CrO4+KIO4+KCl+H2O' : inputText), function (e) {
-        $('#frame')[0].innerHTML = (e.charAt(0) == '!') ? ('<pre class="text-danger bg-danger">' + e.slice(1, e.length) + '</pre>') : (renderEquation((inputText == '') ? 'CrI3+Cl2+KOH=K2CrO4+KIO4+KCl+H2O' : inputText)
+    return $.get('/chem?' + ((inputText == '') ? 'CrI3+Cl2+KOH=K2CrO4+KIO4+KCl+H2O' : inputText), function (e) {
+        $('.frame')[0].innerHTML = (e.charAt(0) == '!') ? ('<pre class="text-danger bg-danger">' + e.slice(1, e.length) + '</pre>') : (renderEquation((inputText == '') ? 'CrI3+Cl2+KOH=K2CrO4+KIO4+KCl+H2O' : inputText)
             + '<br><span class="glyphicon glyphicon-chevron-down" aria-hidden="true"></span><br>' + renderEquation(e));
         if (e.charAt(0) != '!') balText = e
         MathJax.typeset()
         $('#balBtn').text('配平')
         $('#balBtn').removeClass('disabled')
-        $('#frame').removeClass('text-muted')
+        $('.frame').removeClass('text-muted')
         running = 0;
     })
 }
@@ -249,6 +251,315 @@ function balUp() {
 
 //QUERY-------------------------------
 
+var modeq = 'query', nameq = 'eq', strict = false, matchMode = 'mole'
+function setQryEq() {
+    $('#qryMatch').show()
+    $('.addInput').hide()
+    $('.ok').show()
+    modeq = 'query', nameq = 'eq', strict = false
+    $('.qryInputHidable').hide()
+    $('#qryBtn').text('查询方程式')
+    $('#qryInput').attr('placeholder', 'O2=H2O')
+    input2();
+}
+function setQryEq2() {
+    $('#qryMatch').show()
+    $('.addInput').hide()
+    $('.ok').show()
+    modeq = 'query', nameq = 'eq', strict = true
+    $('.qryInputHidable').show()
+    $('#qryBtn').text('查询方程式*')
+    $('#qryInput').attr('placeholder', 'H2O')
+    input2();
+}
+function setQryMo() {
+    $('#qryMatch').show()
+    $('.addInput').hide()
+    $('.ok').show()
+    modeq = 'query', nameq = 'mo'
+    $('.qryInputHidable').hide()
+    $('#qryBtn').text('查询分子')
+    input2();
+}
+function setAddEq() {
+    $('#qryMatch').hide()
+    $('.addInput').show()
+    $('#addId').hide()
+    $('.ok').hide()
+    modeq = 'add', nameq = 'eq'
+    $('.qryInputHidable').hide()
+    $('#qryBtn').text('上传方程式')
+    $('#qryInput').attr('placeholder', 'H2+O2=H2O')
+}
+function setAddMo() {
+    $('#qryMatch').hide()
+    $('.addInput').show()
+    $('#addId').hide()
+    $('.ok').hide()
+    modeq = 'add', nameq = 'mo'
+    $('.qryInputHidable').hide()
+    $('#qryBtn').text('上传分子')
+}
+function setUpdEq() {
+    $('#qryMatch').hide()
+    $('.addInput').show()
+    $('#addId').show()
+    $('.ok').hide()
+    modeq = 'upd', nameq = 'eq'
+    $('.qryInputHidable').hide()
+    $('#qryBtn').text('修改方程式')
+    $('#qryInput').attr('placeholder', '')
+}
+function setUpdMo() {
+    $('#qryMatch').hide()
+    $('.addInput').show()
+    $('#addId').show()
+    $('.ok').hide()
+    modeq = 'upd', nameq = 'mo'
+    $('.qryInputHidable').hide()
+    $('#qryBtn').text('修改分子')
+}
+function getRegex() {
+    var ret = ''
+    if (strict) {
+        var acont = $('#qryInput').val(), bcont = $('#qryInput2').val();
+        if (!acont && !bcont) acont = $('#qryInput').attr('placeholder')
+        acont = acont.replace(/([\+\=\.;])+/g, '$1')
+        bcont = bcont.replace(/([\+\=\.;])+/g, '$1')
+        ret = '^'
+        if (acont) {
+            var as = acont.split('+')
+            for (let i = 0; i < as.length; i++) {
+                if (matchMode == 'mole') ret += '(?=([^=]*\\+|)[0-9]?' + as[i] + '([+=]|$))'
+                else ret += '(?=[^=]*' + as[i] + '([^a-z]|$))'
+            }
+        }
+        if (bcont) {
+            var bs = bcont.split('+');
+            for (let i = 0; i < bs.length; i++) {
+                if (matchMode == 'mole') ret += '(?=.*=(.*\\+|)[0-9]?' + bs[i] + '([+=]|$))'
+                else ret += '(?=.*=.*' + bs[i] + '([^a-z]|$))'
+            }
+        }
+        ret += '.*'
+    } else {
+        ret = $('#qryInput').val()
+        if (!ret) ret = $('#qryInput').attr('placeholder')
+        ret = ret.replace(/([\+\=\.;])+/g, '$1')
+        var scont = ret.split('=')
+        if (scont.length > 2) return;
+        var as = scont[0].split('+');
+        ret = matchMode == 'mole' ? '^' : ''
+        for (let i = 0; i < as.length; i++) {
+            if (matchMode == 'mole') ret += "(?=(.*[+=]|)[0-9]?" + as[i] + "([+=]|$))"
+            else ret += '(?=.*' + as[i] + '([^a-z]|$))'
+        }
+        if (scont.length > 1) {
+            var bs = scont[1].split('+')
+            for (let i = 0; i < bs.length; i++) {
+                if (matchMode == 'mole') ret += "(?=(.*[+=]|)[0-9]?" + bs[i] + "([+=]|$))"
+                else ret += '(?=.*' + as[i] + '([^a-z]|$))'
+            }
+        }
+        ret += '.*'
+    }
+    return ret
+}
+function query() {
+    if (modeq == 'query' || modeq == 'add') {
+        var cont = getRegex();
+        $('.ok')[0].innerHTML = cont;
+        var bd = JSON.stringify({
+            content: cont,
+        })
+        console.log(bd)
+        fetch('/chem/query/' + nameq, {
+            method: 'POST',
+            headers: {
+                'Content-type': 'application/json',
+            },
+            body: bd
+        }).then(res => {
+            res.text().then(resj => {
+                if (resj[0] == '!') {
+                    $('.frame')[1].innerHTML = '<pre class="text-danger bg-danger">' + resj + '</pre>';
+                } else {
+                    resj = JSON.parse(resj)
+                    $('.frame')[1].innerHTML = '';
+                    for (let i = 0; i < resj.length; i++) {
+                        console.log(resj[i])
+                        $('.frame')[1].innerHTML += renderEquation(resj[i].content) + '<br>';
+                        $('.frame')[1].innerHTML += '<span class="label label-default">' + resj[i].id + '</span>';
+                        if (resj[i].conditions) $('.frame')[1].innerHTML += '(' + resj[i].conditions + ')';
+                        $('.frame')[1].innerHTML += resj[i].descriptions + '<br>';
+                        if (resj[i].rel) {
+                            $('.frame')[1].innerHTML += 'rel: <span class="label label-default">' + resj[i].rel + '</span><br>';
+                        }
+                    }
+                    MathJax.typeset()
+                }
+            });
+        })
+    }
+    if (modeq == 'add') {
+        $('#balInput').val($('#qryInput').val() ? $('#qryInput').val() : $('#qryInput').attr('placeholder'))
+        setBal();
+        balance().then(e => {
+            console.log(e)
+            if (e[0] == '!') {
+                return;
+            }
+            input();
+            if (!$('#addDescription').val()) {
+                alert('无描述')
+                return;
+            }
+            var resp = confirm(e + '请确认')
+            if (!resp) return;
+            $('#qryInput').val(e)
+            var bd = JSON.stringify({
+                content: $('#qryInput').val(),
+                conditions: $('#addCondition').val(),
+                descriptions: $('#addDescription').val(),
+            })
+            console.log(bd)
+            fetch('/chem/add/' + nameq, {
+                method: 'POST',
+                headers: {
+                    'Content-type': 'application/json',
+                },
+                body: bd
+            }).then(re => {
+                fetch('/chem/query/' + nameq, {
+                    method: 'POST',
+                    headers: {
+                        'Content-type': 'application/json',
+                    },
+                    body: JSON.stringify({
+                        content: $('#qryInput').val(),
+                    })
+                }).then(res => {
+                    res.text().then(resj => {
+                        console.log(resj)
+                        if (resj[0] == '!') {
+                            $('.frame')[1].innerHTML = '<pre class="text-danger bg-danger">' + resj + '</pre>';
+                        } else {
+                            resj = JSON.parse(resj)
+                            $('.frame')[1].innerHTML = '';
+                            for (let i = 0; i < resj.length; i++) {
+                                console.log(resj[i])
+                                $('.frame')[1].innerHTML += renderEquation(resj[i].content) + '<br>';
+                                $('.frame')[1].innerHTML += '<span class="label label-default">' + resj[i].id + '</span>';
+                                if (resj[i].conditions) $('.frame')[1].innerHTML += '(' + resj[i].conditions + ')';
+                                $('.frame')[1].innerHTML += resj[i].descriptions + '<br>';
+                                if (resj[i].rel) {
+                                    $('.frame')[1].innerHTML += 'rel: <span class="label label-default">' + resj[i].rel + '</span><br>';
+                                }
+                            }
+                            MathJax.typeset()
+                        }
+                    });
+                })
+            })
+        })
+    }
+    if (modeq == 'upd') {
+        $('#balInput').val($('#qryInput').val() ? $('#qryInput').val() : $('#qryInput').attr('placeholder'))
+        setBal();
+        balance().then(e => {
+            console.log(e)
+            if (e[0] == '!') {
+                return;
+            }
+            input();
+            if (!$('#addDescription').val()) {
+                alert('无描述')
+                return;
+            }
+            var resp = confirm(e + '请确认')
+            if (!resp) return;
+            $('#qryInput').val(e)
+            var bd = JSON.stringify({
+                content: $('#qryInput').val(),
+                id: $('#addIdText').val(),
+                conditions: $('#addCondition').val(),
+                descriptions: $('#addDescription').val(),
+            })
+            console.log(bd)
+            fetch('/chem/upd/' + nameq, {
+                method: 'POST',
+                headers: {
+                    'Content-type': 'application/json',
+                },
+                body: bd
+            })
+            input2();
+        })
+    }
+}
+
+function input2() {
+    if (modeq == 'upd') {
+        var bd = JSON.stringify({
+            content: $('#addIdText').val(),
+        })
+        if (!$('#addIdText').val()) {
+            $('.frame')[1].innerHTML = '';
+            return;
+        }
+        console.log(bd)
+        fetch('/chem/query/' + nameq + 'id', {
+            method: 'POST',
+            headers: {
+                'Content-type': 'application/json',
+            },
+            body: bd
+        }).then(res => {
+            res.text().then(resj => {
+                if (resj[0] == '!') {
+                    $('.frame')[1].innerHTML = '<pre class="text-danger bg-danger">' + resj + '</pre>';
+                } else {
+                    resj = JSON.parse(resj)[0]
+                    if (resj) {
+                        $('.frame')[1].innerHTML = '';
+                        console.log(resj)
+                        $('.frame')[1].innerHTML += renderEquation(resj.content) + '<br>';
+                        $('.frame')[1].innerHTML += '<span class="label label-default">' + resj.id + '</span>';
+                        if (resj.conditions) $('.frame')[1].innerHTML += '(' + resj.conditions + ')';
+                        $('.frame')[1].innerHTML += resj.descriptions + '<br>';
+                        if (resj.rel) {
+                            $('.frame')[1].innerHTML += 'rel: <span class="label label-default">' + resj.rel + '</span><br>';
+                        }
+                        $('#qryInput').val(resj.content)
+                        $('#addCondition').val(resj.conditions)
+                        $('#addDescription').val(resj.descriptions)
+                        MathJax.typeset()
+                    } else $('.frame')[1].innerHTML = '';
+                }
+            });
+        })
+    } else if (modeq == 'query') {
+        $('.ok')[0].innerHTML = getRegex()
+    }
+}
+
+function qryToggleMatch() {
+    if (matchMode == 'mole') {
+        $('#qryMatch')[0].innerHTML = '匹配元素'
+        matchMode = 'elem'
+    } else {
+        $('#qryMatch')[0].innerHTML = '匹配分子'
+        matchMode = 'mole'
+    }
+}
+
 function qryUp() {
-    $('#qryInput').val($('#balInput').val());
+    var sp = $('#balInput').val().split('=')
+    if (sp.length > 1) {
+        if (modeq == 'query') {
+            $('#qryInput').val(sp[0])
+            $('#qryInput2').val(sp[1])
+            setQryEq2();
+        } else $('#qryInput').val($('#balInput').val());
+    } else $('#qryInput').val($('#balInput').val());
 }

+ 10 - 1
log/index.html

@@ -90,6 +90,15 @@
         <h1>网站日志</h1>
     </div>
 
+    <h3>
+        2022.12.12
+        <small><a href="/chem">chem</a>/query</small>
+    </h3>
+    <p>实现了对数据库中化学方程式的查询,虽然目前数据库里只放了两个式子,但等我实现添加功能后会往里加的。</p>
+    <p>大量运用了正则表达式(没有你我活不了啊!)。</p>
+    <p>认识到了 SQL 注入的危害,并用简陋的手段做了防护。</p>
+    <hr>
+
     <h3>
         2022.12.11
         <small>release</small>
@@ -103,7 +112,7 @@
         2022.12.4
         <small>dark</small>
     </h3>
-    <p>在首页加上了黑暗模式,以及非常愚蠢的秘密消息。</p>
+    <p>在首页加上了黑暗模式,以及非常愚蠢的秘密消息(其实更多是对 Bootstrap 响应式布局的测试)。</p>
     <p>进一步完善按钮样式。</p>
     <hr>